使用系统调用来实现PV操作.doc_第1页
使用系统调用来实现PV操作.doc_第2页
使用系统调用来实现PV操作.doc_第3页
使用系统调用来实现PV操作.doc_第4页
使用系统调用来实现PV操作.doc_第5页
全文预览已结束

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

如何使用系统调用来实现PV操作Big_boy 发表于 2006-1-10 21:20:00 建议 书籍:unix环境高级编程Unix下PV操作的实现;Unix下并没有直接给出PV操作的函数,必须通过调用提供的相关操作,完成我们所需要的PV操作;1、 Unix下信号量的使用:int semget(key_t key,int nsems,int permflags);Unix使用semget调用创建或者获得信号量集合,三个参数中,其中key是唯一标示该信号量集合的标号,以后要获得已经创建的信号量都是通过它;nsems是要创建的信号量的个数,如果个数大于1,那么这将是一个信号量集合,其索引号是用0至到nsems-1;permflags是一组标志,其作用主要是说明该信号量的访问权限;算法中#define SEM_KEY 1002定义了要用到的信号量集的key;还有两个必须用到的关于信号量的系统调用:int (int semid,int sem_num,int command,union semun ctl_arg);Int semop(int semid,struct sembuf *op_array,size_t num_ops);semctl是用于对信号量的控制,如信号量值的设置获取等;semop是对信号量的操作,在下面将看到我定义的PV操作将使用到semop函数;semctlsemctl()系统调用: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()的第一个参数是关键字值。第二个参数是信号量数目。 参数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.*/ unionsemun 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;xMAX_PRINTERS;x+) printf(Printer%d:%dnr,x,get_sem_val(sid,x); 下面的程序可以用来初始化一个新的信号量值: void init_semaphore(int sid,int semnum,int initval) union semunsemopts; semopts.val=initval; semctl(sid,semnum,SETVAL,semopts); 注意系统调用semctl中的最后一个参数是一个联合类型的副本,而不是一个指向联合类型的指针。semop系统调用功能描述:操作一个或一组信号。 用法:#include #include #include int semop(int semid, struct sembuf *sops, unsigned nsops); int semtimedop(int semid, struct sembuf *sops, unsigned nsops, struct timespec *timeout); 参数:semid:信号集的识别码,可通过semget获取。 sops:指向存储信号操作结构的数组指针,信号操作结构的原型如下 struct sembuf unsigned short sem_num; /* semaphore number */ short sem_op; /* semaphore operation */ short sem_flg; /* operation flags */ ; 这三个字段的意义分别为: sem_num:操作信号在信号集中的编号,第一个信号的编号是0。 sem_op:如果其值为正数,该值会加到现有的信号内含值中。通常用于释放所控资源的使用权;如果sem_op的值为负数,而其绝对值又大于信号的现值,操作将会阻塞,直到信号值大于或等于sem_op的绝对值。通常用于获取资源的使用权;如果sem_op的值为0,则操作将暂时阻塞,直到信号的值变为0。 sem_flg:信号操作标志,可能的选择有两种 IPC_NOWAIT /对信号的操作不能满足时,semop()不会阻塞,并立即返回,同时设定错误信息。 IPC_UNDO /程序结束时(不论正常或不正常),保证信号值会被重设为semop()调用前的值。这样做的目的在于避免程序在异常情况下结束时未将锁定的资源解锁,造成该资源永远锁定。 nsops:信号操作结构的数量,恒大于或等于1。 timeout:当semtimedop()调用致使进程进入睡眠时,睡眠时间不能超过本参数指定的值。如果睡眠超时,semtimedop()将失败返回,并设定错误值为EAGAIN。如果本参数的值为NULL,semtimedop()将永远睡眠等待。 返回说明:成功执行时,两个系统调用都返回0。失败返回-1,errno被设为以下的某个值 E2BIG:一次对信号的操作数超出系统的限制 EACCES:调用进程没有权能执行请求的操作,并且不具有CAP_IPC_OWNER权能 EAGAIN:信号操作暂时不能满足,需要重试 EFAULT:sops或timeout指针指向的空间不可访问 EFBIG:sem_num指定的值无效 EIDRM:信号集已被移除 EINTR:系统调用阻塞时,被信号中断 EINVAL:参数无效 ENOMEM:内存不足 ERANGE:信号所允许的值越界 词条图册更多图册2、 Unix下PV操作的实现:int p(int semid,int sem_index)struct sembuf sb;sb.sem_num = sem_index;sb.sem_op = -1;sb.sem_flg = SEM_UNDO;if(semop(semid,&sb,1)=-1)printf(PID(%d)-P():failed!n,getpid();return(FALSE);return(TRUE);int v(int semid,int sem_index)struct sembuf sb;sb.sem_num = sem_index;sb.sem_op = 1;sb.sem_flg = SEM_UNDO;if(semop(semid,&sb,1)=-1)printf(PID(%d)-V():failed!n,getpid();return(FALSE);return(TRUE);以上是PV操作的函数定义,通过传递给p()或v()一个semid和sem_index,对需要的信号量集合中的指定信号量进行操作;3、 Unix下共享存储的实现:Unix提供shmet、shmat和shmdt等一系列的相关系统调用用作共享存储的实现;通过shmget函数来创建共享内存:int shmget(ke

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论