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));
semop 常见的异常分析System V 信号量的P\V 操作通过函数semop实现,在使用的过程中经常会遇到一些异常,常见的如:errno 4,Interrupted system callerrno 34,Numerical result out of rangeerrno 11,Resource temporarily unavailable下面分别...
1、插件shared_preferences找不到gradle,详细日志如下:
* What went wrong:
A problem occurred configuring project ':shared_preferences'.
> Could not resolve
al
l artifacts for configuration ':shared_preferences:cla...
错误现象:(
semop
函数调用,strerror(errno)输出结果)Interrupted system c
al
l平台:RedHat LinuxLINUX文档关于EINTR的描述是这样子的: While blocked in this system c
al
l, the process caught a sign
al
.UNIX文档[IEEE Std 1003.1-2008]关于EINTR的描述是这样子的: The
semop
() function was interrupted by a sign
al
.这样的两句话如果关从字面上理解的话,就是在
semop
等待的过程中出现INTR信号。可是
1>函数功能:得到一个信号量集标识符或创建一个信号量集对象;
2>函数原型:int semget(key_t key, int nsems, int semflg);
3>函数返回值:成功时:返回值为信号量集的标识符 失败时:返回值为-1
4>参数:(1)key:0(IPC_PRIVATE)建立新信号量集对象;
最近在移植和测试syslogd代码时,碰到两个问题:Resource temporarily unavailable(EAGAIN)和
Numeric
al
result
out of
range
。
移植的时候,在初始化全局变量的时候,有sembuf的初始化。如下图
struct sembuf SMwup[1] = = { {1, -1, IPC_NOWAIT} }; //此部分对应V操作
信号量与其他IPC方式不同,它不是进行消息的交互,而是从当一个计数器的功能,它用于实现进程间的互斥和同步。(给我的感觉是有点像JAVA中线程的????)
举一个例子:两个人想要进入同一个密室去处理一些事情,钥匙也只有一把,每次只能有一个人拿一把钥匙进入密室。第一个人拿着钥匙进入密室,第二个人只能在外边等待,直到第一个人拿着钥匙出来之后,放回钥匙,才能进入密室处理自己的事情。信号量在这个过程中就充当着钥匙的作用,而密室则是充当临界资源的角色
信号量用于进程间的同步,若要在进程间传递数据需要结合共享内
在Linux中不管是启动桌面程序还是在控制台执行命令都会有一个返回值,可以叫做错误号(perror),在linux中执行命令后可通过命令:echo $?来查看上一条命令的返回值。要查看返回值的具体含义,可通过perror number查看。
Java中可通过 Runtime.getRuntime().exec(cmd) 执行命令,exec(prams)返回的是一个Process,可通过P