apue-alarm和pause函数,关于信号

12 篇文章 0 订阅
4 篇文章 0 订阅

题记:

这里主要是用alarm和pause来实现sleep函数。会分析其中存在的bug,会很精辟。当然,大牛的恩赐啦~~不然我介些小生怎看到如此的经典呢?!


1实现sleep

1.1一个含有3个BUG的sleep的实现

#include <signal.h>
#include <unistd.h>

static void sig_alrm(int signo)
{

}

unsigned int sleep1(unsigned int nsecs)
{
	if (signal(SIGALRM, sig_alrm) == SIG_ERR)
		return (nsecs);

	alarm(nsecs);
	pause();
	return (alarm(0));
}


出现了3个问题:

1,alarm会擦出之前alarm的设置

2,这次不理解诶了

3,alarm和pause之前会存在竞争。具体来说就是,在一个繁忙的系统中,可能执行了alarm之后就切换到其他进程了。可能要在alarm超时之后才调用到pause,那pause不就死定了吗?


1.2针对第三个问题的解决方案:

#include <setjmp.h>
#include <signal.h>
#include <unistd.h>

static jmp_buf env_alrm;
static void sig_alrm(int signo)
{
	longjmp(evn_alrm, 1);
}

unsigned int sleep2(unsigned int nsecs)
{
	if (signal(SIGALRM, sig_alrm) == SIG_ERR)
		return (nsecs);
	if (setjmp(env_alrm) == 0) {
		alarm(nsecs);
		pause();
	}
	return (alarm(0));
}


1.3但是这个解决方案引来一个问题,longjmp会中断其他的信号处理。(longjmp的威力太强了。呵呵~~)

1.3.1实验:

#include <stdio.h>
#include <signal.h>
unsigned int sleep2(unsigned int);
static void sig_int(int);

int main(void)
{
	unsigned int unslept;
	
	if (signal(SIGINT, sig_int) == SIG_ERR)
		printf("%m\n"), exit(-1);

	unslept = sleep2(5);
	printf("sleep2 returned: %u\n", unslept);

	exit(0);
}

static void sig_int(int signo)
{
	int		i,j;
	volatile int k;
	
	printf("\nsig_int starting\n");
	for (i = 0; i < 300000; i++)
		for (j = 0; j < 4000; j++)
			k += i * j;

	printf("sig_int finished\n");
}

1.3.2输出:

[root@localhost test]#./main

^C

sig_int starting

sig_int finished

sleep2 returned: 1



2实现限制低速设备的阻塞(比如从终端read字符)

2.1一个失败的实现

#include <stdio.h>
#include <signal.h>
#include <unistd.h>

static void sig_alrm(int);

int main(void)
{
	int		n;
	char	line[4096];

	if (signal(SIGALRM, sig_alrm) == SIG_ERR)
		printf("%m\n"), exit(-1);
	alarm(10);
	if ((n = read(STDIN_FILENO, line, 4096)) < 0)
		printf("%m\n"), exit(-1);
	alarm(0);

	write(STDOUT_FILENO, line, n);
	exit(0);
}

static void sig_alrm(int signo)
{
	//just to interrupt the read call
}

因为我的read系统调用是自动重启的,所以alarm的效果没有的。

而且即使不是自动重启,但是alarm和read之间有竞争,等alarm超时后回到介个进程的read时,alarm的效果就一点都没有啦~~~


2.2修改后的版本

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <setjmp.h>

static void sig_alrm(int);
static jmp_buf env_alrm;

int main(void)
{
	int		n;
	char	line[4096];
	
	if (signal(SIGALRM, sig_alrm) == SIG_ERR)
		printf("%m\n"), exit(-1);
	if (setjmp(env_alrm) != 0)
		printf("read timeout\n"), exit(0);

	alarm(10);
	if ((n = read(STDIN_FILENO, line, 4096)) < 0)
		printf("%m\n"), exit(-1);
	alarm(0);

	write(STDOUT_FILENO, line, n);
	exit(0);
}

static void sig_alrm(int signo)
{
	longjmp(env_alrm, 1);
}




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值