- 浏览: 156526 次
- 性别:
- 来自: 深圳
文章分类
最新评论
-
lyaqys:
lz实现的OptimisticExclusiveLock有点问 ...
java park/unpark 【java并发】基于JUC CAS原理,自己实现简单独占锁
原文:http://blog.chinaunix.net/uid-23629988-id-199153.html
今天帮助同事调试一个问题,最终确定是在多线程下使用sigwait引发的。
该线程大致流程如下:
void thread(void *data)
{
int wait_sig = *(int*)data;
sigset_t sigset;
sigemptyset(&sigset);
sigaddset(&sigset, wait_ig);
while (1) {
int signal;
if (0 != sigwait(&sigset, &signal)) {
break;
}
}
}
出现的问题是这个断言失败,也就是说sigwait失败了。后来打印出sigwait的返回值,发现sigwait的失败的原因时EINTR,也就说 sigwait被一个信号中断了,但是不知道信号来自何处。说到此处,先说明一下,sigwait这个函数很奇怪,跟一般的linux API不同。sigwait出错的时候,并不设置errno,而直接把errno错误值返回。
通过增加一个新的判断
int ret = sigwait(&sigset, &signal));
if (EINTR == ret) {
continue;
}
else if (ret) {
break;
}
毕竟sigwait作为一个阻塞操作,因为收到信号而失败,是可以接受的行为,所以要对EINTR进行特殊处理。
加上这一句后,发现原来在另外一个线程有一个非法内存错误,所以产生了一个SIGSEGV信号。到此,虽然这个问题是由于这个内存问题引起的,但是实际上从这个问题的现象上看,在多线程下使用sigwait容易引起一些问题。
1.在POSIX标准中,当进程收到信号时,如果是多线程的情况,我们是无法确定是哪一个线程处理这个信号。而sigwait是从进程中pending的信号中,取走指定的信号。这样的话,如果要确保sigwait这个线程收到该信号,那么所有线程含主线程以及这个sigwait线程则必须block住这个信号。否则如果在两次sigwait之间,收到了指定信号,该信号很有可能被任意一个线程处理掉。
2.sigwait的名字以及在man中的介绍容易引起人的误解。
1)sigwait,从名字上看只等待指定信号集。这样很容易让人感觉除指定信号外,其他信号不会促使sigwait的返回;
2)来看看man中的说明The sigwait() function suspends execution of the calling thread until the delivery of one of the signals specified in the signal set set.这里也说了,sigwait会一直suspend直到指定信号发生。另外,man中的errors说明也容易引起误解。man中只有一个错误,就是EINVAL,给人感觉sigwait不会被信号中断。
ERRORS
EINVAL set contains an invalid signal number.
(我的系统是FC12)
但是按照我一直以来的经验,基本上所有的阻塞操作都会被信号中断,除非设置了RESTART标志。所有我建议同事加了以上的代码。
后来看到man中还有这么一句,
NOTES
sigwait() is implemented using sigtimedwait(2).
于是又查看了sigtimedwait。发现sigtimedwait的手册还是比较完整的。
ERRORS
EAGAIN No signal in set was delivered within the timeout period specified to sigtimedwait().
EINTR The wait was interrupted by a signal handler; see signal(7). (This handler was for a signal other than one of those in set.)
EINVAL timeout was invalid.
其实这次的代码是一个开源工具的代码,它使用信号机制作为多线程的通讯工具。个人感觉这种实现并不合适。因为在它的代码中,并没有在所有的线程中屏蔽掉需要sigwait的信号,另外该工具会产生多个线程,每个线程需要sigwait的信号是不同的。按照上面的代码,实际上它只是通过sigwait等到相应的信号来唤醒该线程,那么还不如使用pthread_cond_wait来代替sigwait呢。
今天帮助同事调试一个问题,最终确定是在多线程下使用sigwait引发的。
该线程大致流程如下:
void thread(void *data)
{
int wait_sig = *(int*)data;
sigset_t sigset;
sigemptyset(&sigset);
sigaddset(&sigset, wait_ig);
while (1) {
int signal;
if (0 != sigwait(&sigset, &signal)) {
break;
}
}
}
出现的问题是这个断言失败,也就是说sigwait失败了。后来打印出sigwait的返回值,发现sigwait的失败的原因时EINTR,也就说 sigwait被一个信号中断了,但是不知道信号来自何处。说到此处,先说明一下,sigwait这个函数很奇怪,跟一般的linux API不同。sigwait出错的时候,并不设置errno,而直接把errno错误值返回。
通过增加一个新的判断
int ret = sigwait(&sigset, &signal));
if (EINTR == ret) {
continue;
}
else if (ret) {
break;
}
毕竟sigwait作为一个阻塞操作,因为收到信号而失败,是可以接受的行为,所以要对EINTR进行特殊处理。
加上这一句后,发现原来在另外一个线程有一个非法内存错误,所以产生了一个SIGSEGV信号。到此,虽然这个问题是由于这个内存问题引起的,但是实际上从这个问题的现象上看,在多线程下使用sigwait容易引起一些问题。
1.在POSIX标准中,当进程收到信号时,如果是多线程的情况,我们是无法确定是哪一个线程处理这个信号。而sigwait是从进程中pending的信号中,取走指定的信号。这样的话,如果要确保sigwait这个线程收到该信号,那么所有线程含主线程以及这个sigwait线程则必须block住这个信号。否则如果在两次sigwait之间,收到了指定信号,该信号很有可能被任意一个线程处理掉。
2.sigwait的名字以及在man中的介绍容易引起人的误解。
1)sigwait,从名字上看只等待指定信号集。这样很容易让人感觉除指定信号外,其他信号不会促使sigwait的返回;
2)来看看man中的说明The sigwait() function suspends execution of the calling thread until the delivery of one of the signals specified in the signal set set.这里也说了,sigwait会一直suspend直到指定信号发生。另外,man中的errors说明也容易引起误解。man中只有一个错误,就是EINVAL,给人感觉sigwait不会被信号中断。
ERRORS
EINVAL set contains an invalid signal number.
(我的系统是FC12)
但是按照我一直以来的经验,基本上所有的阻塞操作都会被信号中断,除非设置了RESTART标志。所有我建议同事加了以上的代码。
后来看到man中还有这么一句,
NOTES
sigwait() is implemented using sigtimedwait(2).
于是又查看了sigtimedwait。发现sigtimedwait的手册还是比较完整的。
ERRORS
EAGAIN No signal in set was delivered within the timeout period specified to sigtimedwait().
EINTR The wait was interrupted by a signal handler; see signal(7). (This handler was for a signal other than one of those in set.)
EINVAL timeout was invalid.
其实这次的代码是一个开源工具的代码,它使用信号机制作为多线程的通讯工具。个人感觉这种实现并不合适。因为在它的代码中,并没有在所有的线程中屏蔽掉需要sigwait的信号,另外该工具会产生多个线程,每个线程需要sigwait的信号是不同的。按照上面的代码,实际上它只是通过sigwait等到相应的信号来唤醒该线程,那么还不如使用pthread_cond_wait来代替sigwait呢。
发表评论
-
探索 Pexpect,第 2 部分:Pexpect 的实例分析
2013-08-19 11:08 1729原文: http://www.ibm.com/develope ... -
Sudo提权出现:xx用户不在 sudoers 文件中
2013-08-03 20:22 920Sudo提权出现:xx用户不在 sudoers 文件中 症状 ... -
atoi源码
2013-05-14 19:32 1290原文: http://blog.csdn.net/eroswa ... -
为重负网络优化 Nginx 和 Node.js
2013-05-13 01:12 1033原文:http://linux.cn/forum.php?mo ... -
《APUE》:线程和fork(父子进程锁)
2013-04-29 21:07 1200《Unix环境高级编程》这本书附带了许多短小精美的小程序,我在 ... -
ubuntu eclipse 问题
2013-04-05 03:30 818Eclipse 3.6 在 Ubuntu 10.04 下会出现 ... -
linux多线程之pthread_cancel结束线程(防止死锁)
2013-03-28 18:28 1275linux多线程之pthread_cancel结束线程 摘要: ... -
c++ 多线程编程的时候遇到了一个编译问
2013-03-27 15:22 831今天在进行多线程编程的时候遇到了一个编译问题:error: a ... -
socket参数详解:KeepAlive
2013-03-19 13:24 1245TCP协议中有长连接和短连接之分。短连接在数据包发送完成后就会 ... -
ehcache分布式例子
2013-03-19 11:52 645配置文件: recluster_ehcache_0.xml ... -
Voldemort — 分布式 key-value 存储系统
2013-03-19 00:40 1035Voldemort — 分布式 key-value 存储系统 ... -
java 慎用 new Random()
2013-03-11 18:14 938package tt; import java.util.* ... -
crontab中运行python程序出错,提示ImportError: No module named解决全过程
2013-03-11 16:18 1258原文地址:http://blog.csdn.net/langl ... -
Linux crontab
2013-03-08 18:27 1088crontab是Linux下最常用的计划任务服务。本文跟大家分 ... -
tinyxml 实例
2013-02-28 15:00 679tinyxml 实例 例子见附件 -
C++ 中char*,const char*,string 轉換
2013-02-26 11:34 11361. string转const char* string ... -
WARNING: source src1: skipped 14 document(s) with zero/NULL ids
2012-12-18 21:17 1940select 中缺少主键id -
libmysqlclient.so.15 is needed by .....
2012-12-18 16:22 1193libmysqlclient.so.15 is needed ... -
Redis 内存
2012-12-12 16:28 904NoSQLFan 关注NoSQL相关的新闻与技术 ... -
apache.commons工具
2012-12-03 10:22 1070本文主要介绍自己在开发过程中总结的一些Java工具类,主要包括 ...
相关推荐
本文将深入探讨如何在多线程环境中编写安全的信号处理函数,尤其是在程序逻辑需要而非错误导致的情况下。文章分为两个部分:首先介绍编写安全的异步信号处理函数所需考虑的因素,随后提供一种在多线程环境中以同步...
- **多线程环境下的资源共享**:由于子进程继承了父进程的信号处理函数,且未对信号进行适当的屏蔽,导致多个线程同时尝试执行同一信号处理函数,从而造成所有线程都陷入阻塞状态。 #### 四、解决方案 为了解决...
Linux多线程环境中的信号处理不同于进程的信号处理。一方面线程间信号处理函数的共享性使得信号处理更为复杂,另一方面普通异步信号又可转换为同步方式来简化处理。 本文首先介绍信号处理在进程中和线程间的...
这些函数库提供了丰富的功能,包括线程创建、线程同步、线程通信、实时任务调度等,帮助开发者创建高效、可靠的多线程应用程序。 1. 线程创建函数: - pthread_create():创建一个新的线程,执行指定的函数。 - ...
Posix Pthread API 是 Linux 和 Unix 操作系统中用于多线程编程的应用程序接口(API)。它提供了一组函数,允许开发者创建、管理和控制线程,以实现高性能、可靠的并发编程。本文档总结了 Posix Pthread API 的主要...
在Linux系统中,多线程编程常常涉及到线程同步,以确保数据的一致性和避免竞态条件。信号(Signal)是Unix和类Unix系统中进程间通信的一种方式,也可以用于线程间的同步。本篇文章将重点讲解如何在C语言中使用信号...
这在多线程环境下特别有用,可以避免信号被错误地传递到不适宜的线程。 可靠信号是指能够保证送达的信号,即使在信号处理函数执行期间,相同的信号再次到达,系统也会保留这些信号,直到处理函数返回后再进行处理。...
一种可能的解决方案是使用`pthread_sigmask()`函数来控制线程对信号的可见性,或者使用`sigwait()`函数在你的主线程中等待特定信号的到来。 此外,你的代码示例中包含了JMS消息处理的片段,但没有提供完整的信息。...
在编写多线程程序时,需要注意以下几点: - 如果线程陷入无限循环或长时间阻塞状态,需要确保循环体或阻塞函数中包含取消点,例如调用`pthread_testcancel()`。 - 在可能的取消点前后添加适当的同步机制,以确保...
MIPS版本的线程调试支持更复杂,新增的`td_ta_map_lwp2thr`接口和多个条件编译段,表明在MIPS上调试线程时有特定的需求。 10. `linker/debugger.c` 和 `dlfcn.c`: 这两个文件中,MIPS版本包含了对信号处理和初始...
卡塔托尼 如此简单的容器初始化实际上很容易让人脑筋急... 理想情况下,我们只是为其他项目编写一个使用signalfd(2)的补丁程序,而不是创建一个新项目,但是花了一些时间查看tini和dumb-init我们认为此类补丁程序将更
**蜂鸣器模块**:根据Clkbeep1,2,3和clkcount产生高低中音,以及根据sigwait控制暂停/开始。 **点阵模块**:根据节拍信号lie和hang输出不同的显示符号。 **仿真与源程序** 报告中提到了各模块的仿真波形分析,包括...
3. **信号的接收**:进程可以通过系统调用(如sigwait或sigtimedwait)等待特定信号的到来,或者选择不阻塞地接收信号。接收信号后,进程可以选择忽略、执行默认动作或者执行自定义的信号处理函数。 4. **信号的...
按下button时,jps计数器加一,更改节拍,并将jps信号发送给蜂鸣器、点阵和数码管模块。button3控制暂停信号sigwait,翻转该信号可以暂停或开始节拍。 4. **数码管显示模块ra**:利用Clk扫描频率,jps和speed作为...
- **2.5 信号集**:sigwait()和sigwaitinfo()等函数可以等待并获取信号集合中的信号。 - **2.6 信号的排队与丢失**:如果同一类型的信号多次发送到一个进程,可能会发生排队或丢失的情况。 - **2.7 信号处理函数的...
- **同步原语**:如`sigwait`、`sigwaitinfo`等函数用于等待信号的发生。 - **信号集操作**:如`sigemptyset`、`sigaddset`等函数用于操作信号集。 #### 二十五、基本程序/系统接口 第25章提供了基本的程序和系统...