使用后如何正确删除信号量?

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 错误来自于对集合中的信号量数量使用了太大的值。

【讨论】:

以上是关于使用后如何正确删除信号量?的主要内容,如果未能解决你的问题,请参考以下文章

如何正确销毁 C 中多个进程使用的共享未命名信号量?

正确销毁命名的 System V 信号量

RTL基本知识:如何正确在敏感信号列表中包含function中的信号

捕获信号时如何正确等待bash子进程完成

如何正确使用信号灯同步 Alamofire 请求 Swift4 的主线程?

删除类 UNIX 系统上的所有 SYSTEM V 共享内存和信号量