mysleep函数linux,[Linux]继续探究mysleep函数(竞态条件)

本文探讨了如何改进mysleep函数,通过使用sigsuspend函数避免信号处理中的竞态条件。作者解释了原有方法的问题,并介绍了如何通过信号屏蔽和sigsuspend确保在闹钟响起后的正确挂起。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

之前我们探究过mysleep的简单用法,我们实现的代码是这样的:

#include

#include

void myhandler(int sig)

{

}

unsigned int mysleep(unsigned int timeout)

{

struct sigaction act,oact;

act.sa_handler = myhandler;

sigemptyset(&act.sa_mask);

act.sa_flags = 0;

sigaction(SIGALRM,&act,&oact); //信号注册函数

alarm(timeout); //闹钟timeout秒后响

pause(); //挂起等待

unsigned int ret = alarm(0); //清空闹钟

sigaction(SIGALRM,&oact,NULL);

return ret;

}

int main()

{

printf("ready sleeping!\n");

mysleep(3);

printf("i am waking!\n");

return 0;

}

我们首先注册了捕捉信号的函数,信号为SIGALRM,然后调用了alarm函数来设置闹钟,此时pause来挂起等待,然后内核切换到别的进程运行,在timeout秒后闹钟产生SIGALRM信号,从内核态到用户态的过程中,收到用户自定义的处理函数,就在用户态处理函数,进入处理函数myhandler时,SIGALRM信号会被自动屏蔽,当处理完函数后,自动解除屏蔽,进入到内核态执行系统调用,最后切换到用户态执行主函数控制逻辑。

这里存在一个问题,比如刚刚说的alarm函数调用完成后,pause挂起等待,当把所有的逻辑都处理完成alarm返回时,进入用户态继续pause,是不是没有任何意义呢。还有一点,我们不知道pause挂起等待是在alarm函数之内还是执行后调用的,这就出现了异步情况,我们称这种现象为竞态条件。

我们如何解决这类问题呢。。我们试着将SIGALRM信号屏蔽起来,当执行完alarm函数后再自动解除屏蔽。这样就保证是在alarm函数执行到时间后挂起的状态。但是还有一种可能是当解除屏蔽之后还有可能使SIGALRM递达,因此这种方法还是有缺陷的。我们只能用原子性的操作来避免这种问题的出现。

这时又引出了一个函数:sigsuspend包含了pause的挂起等待功能,同时解决了竞态条件的问题(与exec和pause一样,没有成功的返回值)

#include

int sigsuspend(const sigset_t *sigmask);

调用sigsuspend时,进程的信号屏蔽字由sigmask参数指定,可以通过指定sigmask来临时解除对某个信号的屏蔽,然后挂起等待,当sigsuspend返回时,进程的信号屏蔽字恢复为原来的值,如果原来对该信号是屏蔽的,从sigsuspend返回后仍然是屏蔽的。

实现代码如下:

#include

#include

void myhandler(int sig)

{

}

unsigned int mysleep(unsigned int timeout)

{

struct sigaction act,oact;

sigset_t newmask,oldmask,suspmask;

act.sa_handler = myhandler;

sigemptyset(&act.sa_mask);

act.sa_flags = 0;

sigemptyset(&newmask);

sigaddset(&newmask,SIGALRM);

sigprocmask(SIG_BLOCK,&newmask,&oldmask);

sigaction(SIGALRM,&act,&oact); //信号注册函数

alarm(timeout); //闹钟timeout秒后响

suspmask = oldmask;

sigdelset(&suspmask,SIGALRM);

sigsuspend(&suspmask);

//pause(); //挂起等待

unsigned int ret = alarm(0); //清空闹钟

sigaction(SIGALRM,&oact,NULL);

sigprocmask(SIG_SETMASK,&oldmask,NULL);

return ret;

}

int main()

{

printf("ready sleeping!\n");

mysleep(3);

printf("i am waking!\n");

return 0;

}

运行结果:

4dd2d38c4c2d7535d7b8e53f0aae482b.png

3秒之后:

de6222cb018ce131fc5d26f537f3871a.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值