semop异常分析

semop 常见的异常分析

System V 信号量的P\V 操作通过函数semop实现,在使用的过程中经常会遇到一些异常,常见的如:

    errno 4,Interrupted system call
    errno 34,Numerical result out of range
    errno 11,Resource temporarily unavailable

下面分别介绍这三种错误出现的原因和处理方法。

int  semop(int  semid,struct sembuf *sops,size_t nsops)

    参数semid 为信号量集的标识符;
    参数 sops 指向进行操作的结构体数组的首地址;
    参数 nsops指出将要进行操作的信号的个数。
    semop 函数调用成功返回 0,失败返回 -1。

semop的第二个参数 sops 指向的结构体数组中,每个 sembuf结构体对应一个特定信号的操作,该结构定义在 linux/sem.h,如下所示:
     struct sembuf
     {
        unsigned short  sem_num;     //信号在信号集中的索引,0代表第一个信号,1代表第二个信号  
        short          sem_op;     //操作类型
        short          sem_flg;   //操作标志
    };

    sem_op 参数:

         sem_op > 0        信号加上 sem_op的值,表示进程释放控制的资源;
         sem_op = 0        如果没有设置 IPC_NOWAIT,则调用进程进入睡眠状态,直到信号量的值为0;否则进程不回睡眠,直接返回EAGAIN
        sem_op < 0        信号加上 sem_op的值。若没有设置 IPC_NOWAIT ,则调用进程阻 塞,直到资源可用;否则进程直接返回EAGAIN

    sem_flg 参数:

         该参数可取的值包括IPC_NOWAIT、SEM_UNDO或者0。
         IPC_NOWAIT:由于P\V操作是原子操作,如果设置了IPC_NOWAIT,当有其他进程在修改信号量的值时,就会直接返回,不会阻塞等待, errno为 EAGAIN:Resource temporarily unavailable。可以根据需要忽略这个异常,或者不要设置IPC_NOWAIT
         SEM_UNDO:如果设置了这个参数系统就会为调用进程维护一个UNDO的计数,当进程正常或者异常退出时,系统会根据这个计数恢复之前多余的P\V操作, 防止造成资源的死锁,例如,做了2次P操作,1次V操作, 退出时系统会做1次的V操作。
      
    对于SEM_UNDO来说,内核记录的信息是跟进程相关的。一个进程在P操作的时候对应该进程的UNDO计数就多一个,V操作的时候那么计数就减一。在设置SEM_UNDO的时候一定要注意使用的场景,否则就会导致Numerical result out of range错误。
1)如果P\V操作都是在一个进程中完成,就可以设置该标志, 但要注意P\V操作时要同时设置,否则也会导致计数值一直增加而溢出;
2)如果一个进程做P操作,另外一个进程做V操作,就不能设置该标志,因为对单一的进程来说UNDO计数会一直增加而溢出,计数的上限是32767(和信号量的最大值相同)

Interrupted system call,关于这个错误网上讲解的比较多了,当进程阻塞在某个系统调用的时候,如果进程捕获到一个中断信号,阻塞就会被打断,产生Interrupted system call错误,针对这个错误需要进程重做P操作,系统不会帮助重启,如:进程中有timer 触发的时候就会出现这个错误

do
{
    nRet =  semop(semid,sops,nsops); // P操作
}while( (nRet == -1) && (errno == 4));

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
semop函数是Linux系统中用于对信号量进行操作的系统调用函数。 函数原型: ```c #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> int semop(int semid, struct sembuf *sops, size_t nsops); ``` 参数说明: - `semid`:信号量集标识符; - `sops`:指向一个由sembuf结构构成的数组,每个sembuf结构用于描述对一个信号量的操作; - `nsops`:信号量操作的个数。 sembuf结构定义如下: ```c struct sembuf { ushort sem_num; /* 信号量的编号 */ short sem_op; /* 信号量的操作 */ short sem_flg; /* 信号量的标志 */ }; ``` - `sem_num`:指定操作哪个信号量; - `sem_op`:指定信号量的操作,可以是正数、负数或零; - `sem_flg`:指定信号量的操作标志,可以是IPC_NOWAIT(非阻塞)、SEM_UNDO(自动撤销)或0(默认)。 函数返回值为0表示操作成功,否则表示操作失败。 semop函数可以对一个或多个信号量进行操作,如增加、减少或设置信号量的值。在进行信号量操作时,需要先定义一个sembuf结构数组,并将这个数组作为参数传递给semop函数。 常用的信号量操作如下: - `sem_op > 0`:将信号量的值加上sem_op; - `sem_op < 0`:将信号量的值减去sem_op; - `sem_op = 0`:如果信号量的值为0,则等待信号量变为非0,否则什么也不做。 需要注意的是,如果信号量的值为0,且设置了IPC_NOWAIT标志,则semop函数将立即返回,不会等待信号量变为非0。 如果进程在操作信号量时突然终止,那么系统会自动撤销该进程对信号量的所有操作。如果设置了SEM_UNDO标志,则在进程终止时,系统会自动将信号量恢复到进程开始操作时的值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

sunxiaopengsun

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值