使用后如何正确删除信号量?
Posted
技术标签:
【中文标题】使用后如何正确删除信号量?【英文标题】:How to properly remove semaphores after usage? 【发布时间】:2021-01-17 03:20:48 【问题描述】:我正在使用大量的信号量来同步多进程对共享内存的访问。
我通过
初始化信号量int scount = 80000;
int semid = semget(IPC_PRIVATE, scount, 0666 | IPC_CREAT);
效果很好。
在程序结束时,我删除了通过
设置的信号量semctl(semid, 0, IPC_RMID);
但是,当我通过
检查系统中信号量的数量时ipcs -S
在控制台中,我看到它们还在那里。
seminfo:
semmap: 30 (# of entries in semaphore map)
semmni: 8 (# of semaphore identifiers)
semmns: 80064 (# of semaphores in system) <------ ????
semmnu: 0 (# of undo structures in system)
semmsl: 87381 (max # of semaphores per id)
semopm: 5 (max # of operations per semop call)
semume: 10 (max # of undo entries per process)
semusz: 32 (size in bytes of undo structure)
semvmx: 32767 (semaphore maximum value)
semaem: 16384 (adjust on exit max value)
我也尝试在控制台中手动删除它们
ipcrm -s [semid]
这似乎可以很好地删除设置(即semid
在命令之前的ipcs -s
中列出并在之后消失)。然而ipcs -S
仍然显示
semmns: 80064 (# of semaphores in system)
如何在使用后正确清理信号量,使它们不再显示在"semmns: .... (# of semaphores in system)"
中?
我问的原因是,如果信号量没有正确清理,我下一次运行程序将无法运行,因为打开的信号量数量将超过限制。我需要重新启动才能将其重置为 0。
PS:我在 Mac OS 上,但也希望它在 Linux 上工作。
semget
不返回错误。当我通过以下方式初始化信号量集时,下一步会发生错误:
semun_t semun = .val = 1; // initial semaphore value => 1 = released
for (int i=0; i<scount; i++)
if(semctl(semid, i, SETVAL, semun) == -1) perror("semctl init");exit(1);
我得到的错误是:
semctl init: Invalid argument
【问题讨论】:
你调用的函数是正确的。似乎无效的semctl
调用的返回值是多少?删除是由创建信号量的同一进程发出的吗?
semctl
在初始化期间抛出错误Invalid argument
。我编辑了我的问题以反映这一点。谢谢。
所以你搜索sem_unlink
?
@KamilCuk No. AFAIK sem_unlink
用于删除命名信号量。在semget
中创建信号量时,我使用IPC_PRIVATE
而不是key
。
【参考方案1】:
对于遇到同样问题的人:
ipcs -S
调用时在semmns
下显示的数字是到目前为止已使用的信号量 的数量。 not 是否意味着它们仍在使用中。因此,即使您正确删除了信号量,计数器semmns
也不会减少。
semctl init: Invalid argument
错误来自于对集合中的信号量数量使用了太大的值。
【讨论】:
以上是关于使用后如何正确删除信号量?的主要内容,如果未能解决你的问题,请参考以下文章
RTL基本知识:如何正确在敏感信号列表中包含function中的信号