题记:
这里主要是用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);
}