词条 | semctl |
释义 | semctl()系统调用:semctl(); 原型:int semctl(int semid,int semnum,int cmd,union semunarg); 返回值:如果成功,则为一个正数。 如果失败,则为-1:errno=EACCESS(权限不够) EFAULT(arg指向的地址无效) EIDRM(信号量集已经删除) EINVAL(信号量集不存在,或者semid无效) EPERM(EUID没有cmd的权利) ERANGE(信号量值超出范围) 系统调用semctl用来执行在信号量集上的控制操作。这和在消息队列中的系统调用msgctl是十分相似的。但这两个系统调用的参数略有不同。因为信号量一般是作为一个信号量集使用的,而不是一个单独的信号量。所以在信号量集的操作中,不但要知道IPC关键字值,也要知道信号量集中的具体的信号量。这两个系统调用都使用了参数cmd,它用来指出要操作的具体命令。两个系统调用中的最后一个参数也不一样。在系统调用msgctl中,最后一个参数是指向内核中使用的数据结构的指针。我们使用此数据结构来取得有关消息队列的一些信息,以及设置或者改变队列的存取权限和使用者。但在信号量中支持额外的可选的命令,这样就要求有一个更为复杂的数据结构。 系统调用semctl()的第一个参数是关键字值。第二个参数是操作信号在信号集中的编号,第一个信号的编号是0。 参数cmd中可以使用的命令如下: ·IPC_STAT读取一个信号量集的数据结构semid_ds,并将其存储在semun中的buf参数中。 ·IPC_SET设置信号量集的数据结构semid_ds中的元素ipc_perm,其值取自semun中的buf参数。 ·IPC_RMID将信号量集从内存中删除。 ·GETALL用于读取信号量集中的所有信号量的值。 ·GETNCNT返回正在等待资源的进程数目。 ·GETPID返回最后一个执行semop操作的进程的PID。 ·GETVAL返回信号量集中的一个单个的信号量的值。 ·GETZCNT返回这在等待完全空闲的资源的进程数目。 ·SETALL设置信号量集中的所有的信号量的值。 ·SETVAL设置信号量集中的一个单独的信号量的值。 参数arg代表一个semun的实例。semun是在linux/sem.h中定义的: /*arg for semctl systemcalls.*/ union semun{ intval;/*value for SETVAL*/ structsemid_ds*buf;/*buffer for IPC_STAT&IPC_SET*/ ushort*array;/*array for GETALL&SETALL*/ structseminfo*__buf;/*buffer for IPC_INFO*/ void*__pad; val当执行SETVAL命令时使用。buf在IPC_STAT/IPC_SET命令中使用。代表了内核中使用的信号量的数据结构。array在使用GETALL/SETALL命令时使用的指针。 下面的程序返回信号量的值。当使用GETVAL命令时,调用中的最后一个参数被忽略: intget_sem_val(intsid,intsemnum) { return(semctl(sid,semnum,GETVAL,0)); } 下面是一个实际应用的例子: #defineMAX_PRINTERS5 printer_usage() { int x; for(x=0;x<MAX_PRINTERS;x++) printf("Printer%d:%d\\\r",x,get_sem_val(sid,x)); } 下面的程序可以用来初始化一个新的信号量值: void init_semaphore(int sid,int semnum,int initval) { union semun semopts; semopts.val=initval; semctl(sid,semnum,SETVAL,semopts); } 注意系统调用semctl中的最后一个参数是一个联合类型的副本,而不是一个指向联合类型的指针。 需要注意的是,对于semun联合体,最好自己定义,否则GCC编译器可能会报“semun大小未知”。 |
随便看 |
百科全书收录4421916条中文百科知识,基本涵盖了大多数领域的百科知识,是一部内容开放、自由的电子版百科全书。