zoukankan      html  css  js  c++  java
  • 10.信号

    1、概念

    信号提供了一种处理异步事件的方法。

    不存在编号为0的信号,kill函数对信号编号0有特殊的应用。POSIX.1 将此种信号编号值称为空信号。

     

    2、信号的相关动作

    a、忽略此信号。大多数信号都可使用这种方式进行处理,但有两种信号却决不能被忽略,它们是SIGKILL和SIGSTOP。这两种信号不能不忽略的原因是:它们向超级用户提供了使进程终止和停止的可靠方法;

    b、捕捉信号。为了做到这一点,要通知内核在某种信号发生时调用一个用户函数。(SIGTERM是终止信号,kill命令传送的系统默认信号是终止信号)。注意,不能捕捉SIGKILL和SIGSTOP信号;

    c、执行系统默认动作。针对大多数信号的系统默认动作是终止进程。

     

    3、程序启动(exec处理信号的方式)

    当执行一个程序时,所有信号的状态都是系统默认或忽略,除非调用exec的进程忽略该信号。确切地说,exec函数将原先设置为要捕捉的信号都更改为它们的默认动作,其他信号的状态不变。如在一个交互式shell如何处理针对后台进程的中断和退出信号,如:

    cc main.c &

    shell自动将后台进程对中断和退出信号的处理方式设置为忽略,于是,当按中断键时就不会影响到后台进程。

     

    进程启动(fork处理信号的方式)

    当一个进程调用fork时,其子进程继承父进程的信号处理方式,因为子进程在开始时复制了父进程的存储映像,所以信号捕捉函数的地址在子进程中是有意义的。

     

    4、不可靠信号的两个例子

    前提,早期版本中的问题是在进程每次接到信号对其进行处理时,随即将该信号动作复位为默认值。如:

        int sig_init();                    // my signal handling function
        ...
        signal(SIGINT,sig_init);        // establish handler 
        ...
        
        sig_init()
        {
            signal(SIGINT,sig_init);    // reestablish handler for next time
            ...                            // process the signal
        }

    问题:从信号发生之后到在信号处理程序中调用signal函数之前这段时间中有一个时间窗口。在此段时间中,可能发生另一次中断信号。第二个信号会导致执行默认动作,而针对终端信号的默认动作是终止该进程。

     

    另一个问题:在进程不希望某种信号发生时,它不能关闭该信号。进程能做的一切就是忽略该信号。有时希望通知系统“阻止下列信号发生,如果它们确实发生了,请记住了它们。”能够显现这种缺陷的经典实例有下列程序段,它捕捉了一个信号,然后设置一个表示该信号已发生的标志:

        int sig_init_flag;                // set nonzero when signal occurs
        main()
        {
            int sig_init();                // my signal handling function
            ...
            signal(SIGINT,sig_init);    // establish handler
            ...
            while(sig_init_flag == 0)
                pause();                // go to sleep,waiting for signal
            ...
        }
        sig_init()
        {
            signal(SIGINT,sig_init);    // reestablish handler for next time
            sig_init_flag = 1;            // set flag for main loop to examine
        }

    其中,进程调用pause函数使自己休眠,直至捕捉到一个信号。当捕捉到信号时,信号处理程序将标志sig_init_flag设置为非0值。从信号处理程序返回后,内核自动将该进程唤醒,它检测到该标志为非0值,然后执行它所需做的工作。但是这里也有一个时间窗口,在此窗口中操作可能失误。如果在测试sig_init_flag(while语句)之后和调用pause之前发生SIGINT信号,则此进程在调用pause时入睡,并且长眠不醒(假定此信号不会再次发生)。于是,这次发生的信号也就丢失了。pause函数使调用进程在接到一个信号前挂起。

     

    5、中断的系统调用

    早期UNIX系统的一个特性是:如果进程在执行一个低速系统调用而阻塞期间捕捉到一个信号,则该系统调用就被中断。该系统调用返回出错,其errno被设置为EINTR,这样处理的理由是:因为一个信号发生了,进程捕捉到了它,这意味着已经发生了某种事情,所以是个理应当唤醒其他被阻塞的系统调用的好机会。

     

    与被中断的系统调用相关的问题是必须显式地处理出错返回。典型的代码序列(假定进行一个读操作,它被中断,我们希望重新启动它)可能如下所示:

        again:
            if((n = read(fd,buf,BUFFSIZE)) < 0) {
                if(errno == EINTR)
                    goto again;        // just an interrupted system call
                    
                // handler other errors
            }

    为了帮助应用程序使其不必处理被中断的系统调用,4.2 BSD 引入了某些被中断的系统调用的自动重启动。自动重启动的系统调用包括ioctl、read、readv、write、writev、wait和waitpid。正如前述,其中前5个函数只有对低速设备进行操作时才会被信号中断。而wait和waitpid在捕捉到信号时总是被中断。因为这种自动重启动的处理方式会带来问题,所以某些应用程序并不希望这些函数被中断后重启动。为此4.3 BSD允许进程基于每个信号禁用此功能(见程序清单10-12)。

    引入自动重启动的理由是:有时用户并不知道所使用的输入、输出设备是否是低速设备,如果我们编写的程序可以用交互方式运行,则它可能读、写低速终端设备。如果在程序中捕捉信号,而且系统不提供重启动功能,则对每次读、写系统调用都要进行是否出错返回的测试,如果是被中断的,则再调用读、写系统调用。

     

    6、可重入函数

    进程捕捉到信号并对其进行处理时,进程正在执行的指令序列就被信号处理程序临时中断,它首先执行该信号处理程序中的指令。如果从信号处理程序返回(例如没有调用exit或longjmp),则继续执行在捕捉到信号时进程正在执行的正常指令序列。但在信号处理程序中,不能判断捕捉到信号时进程在何处执行。

    a、如果进程正在执行malloc,在其堆中分配另外的存储空间,而此时由于捕捉到信号而插入执行该信号处理程序,其中又调用malloc,这时会发生什么?

    b、若进程正在执行getpwnam,这种将其结果存放在静态存储单元(可能是静态变量,当发生两次getpwnam时,该变量值只保存最后一次getpwnam的值,并且恢复不了第一次getpwnam时的值)中的函数,期间插入执行信号处理程序,它又调用这样的函数,这时又会发生什么?

     

    在malloc中,可能会对进程造成破坏,因为malloc通常为它所分配的存储区维护一个链接表,而插入执行信号处理程序时,进程可能正在更改此链接表。

    在getpwnam中,返回给正常调用者的信息可能被返回给信号处理程序的信息覆盖。

     

    函数时不可重入的原因为:

    a、已知它们使用静态数据结构;

    b、它们调用malloc或free;

    c、它们是标准IO函数;

    标准IO库的很多实现都以不可重入方式使用全局数据结构。

     

    可重入函数另一个说法,当函数正在执行而被中断,而且中断返回,函数继续执行后的结果,与没有发生中断执行后的结果一样,那么这个函数就是可重入函数。

     

    应当了解,即使在信号处理程序中使用了可重入函数(不会破坏被中断的进程的函数),但是由于每个线程只有一个errno变量,所以信号处理程序可能会修改其原先值。如,一个信号处理程序,它恰好在main刚刚设置errno之后被调用,而该信号处理程序调用read这类函数,则它可能更改errno的值,从而取代了刚刚由main设置的值。因此,作为一个通用的规则,当在信号处理程序中调用了可重入函数时,应当在其前保存,在其后恢复errno。

     

    例子:

        static void my_alarm(void)
        {
            struct passwd *rootptr;
     
            printf("in signal handler
    ");
            if((rootptr = getpwnam("root")) == NULL)
                printf("getpwnam(root) error!
    ");
        
            alarm(1);
        }
        
        int main()
        {
            struct passwd *ptr;                        // 只是指针,内存由getpwnam分配,并放在静态存储区中
        
            signal(SIGALRM,my_alarm);
            alarm(1);
    
            for(;;) {
                if((ptr = getpwnam("zzc")) == NULL)
                    printf("getpwnam error!
    ");
        
                if(strcmp(ptr->pw_name,"zzc") != 0)
                    printf("return value corrupted!,pw_name = %s
    ",ptr->pw_name);
            }
        }

    结果:

        :~/program/c$ ./test
        in signal handler
    
        ^C(强制中断进程运行)

    如果把信号处理函数中的

        if((rootptr = getpwnam("root")) == NULL)
                printf("getpwnam(root) error!
    ");

    注释掉的话(没有覆盖main中getpwnam函数的内容),结果为:

        :~/program/c$ ./test
        in signal handler
        in signal handler
        in signal handler
        in signal handler
        in signal handler
        ...
        ^C

     

    7、可靠信号术语和语义

    在产生了信号时,内核在进程表中设置一个某种形式的标志。当对信号采取了这种行动时,我们说向进程递送了一个信号。在信号产生和递送之间的时间间隔内,称信号是未决的(pending)。

    如果为进程产生了一个设置为阻塞(该信号在信号屏蔽字中的对应位被设置)的信号,而且对该信号的动作是系统默认动作或捕捉该信号,则为该进程将此信号保持为未决状态,直到该进程

    a、对此信号解除阻塞;

    b、将此信号的动作更改为忽略。注意,内核在递送一个原来被阻塞的信号给进程时(而不是在产生该信号时),才决定对它的处理方式。于是进程在信号递送给它之前仍可改变对该信号的动作(即忽略该函数或者默认动作或者进行信号处理)。

    进程调用sigpending函数来判定哪些信号是设置为阻塞并处于未决状态的。

     

    每个进程都有一个信号屏蔽字,它规定了当前要阻塞递送到该进程的信号集,对于每种可能产生的信号,该屏蔽字中都有一位与之对应。对于某种信号,若其对应位已设置,则它当前是被阻塞的。进程可以调用sigprocmask来检测和更改其当前信号屏蔽字。

     

    8、kill、raise、alarm、pause函数

    kill函数将信号发送给进程或进程组。

     

    raise函数则允许进程向自身发送信号。

     

    alarm函数

    如果我们向捕捉SIGALRM信号,则必须在调用alarm之前设置该信号的处理程序。如果我们先调用alarm,然后在我们能够设置SIGALRM处理程序之前已接收到该信号,那么进程将终止。

     

    pause函数

    pause函数使调用进程挂起, 直到捕捉到一个信号,执行了信号处理程序并从其返回时,pause才返回。

     

    9、sleep简单而不完整的实现

    使用alarm和pause,进程可使自己休眠一段指定的时间。如下面的sleep1函数:

        static void sig_alarm()
        {
            ...
        }
        unsigned int sleep1(unsigned int nsecs)
        {
            if(signal(SIGALRM,sig_alarm) == SIG_ERR)
                return (nsecs);
                
            alarm(nsecs);            // start the timer
            pause();                // next caught signal wakes us up
            return (alarm(0));        // turn off timer,return unslept time
        }

    在第一次调用alarm和调用pause之间有一个竞争条件:在一个繁忙的系统中,可能alarm在调用pause之前超时,并调用了信号处理程序。如果发生这种情况,则在调用pause后,如果没有捕捉到其他信号,则调用者将永远被挂起。

    解决方法:

    a、使用setjmp;

    b、使用sigprocmask和sigsuspend;

     

    方法a:

        static jmp_buf env_alrm;
        static void sig_alrm(int signo)
        {
            longjmp(env_alrm,1);
        }
        unsigned int sleep2(unsigned int nsecs)
        {
            if(signal(SIGALRM,sig_alrm) == SIG_ERR)
                return (nsecs);
                
            if(setjmp(env_alrm) == 0) {
                alarm(nsecs);                // start the timer
                pause();                    // next caught signal wakes us up
            }
            
            return (alarm(0));                // turn off timer,return unslept time
        }

    在此,竞争条件已被避免。即使pause从未执行,在发生SIGALRM时,sleep2也会返回。

    出现的另一个问题:它涉及与其他信号的交互,如果SIGALRM中断了某个其他信号处理程序,则调用longjmp会提早终止该信号处理程序。如:

        // 超过5s
        static void sig_int(int signo)
        {
            int i,j;
            volatile int k;
     
            printf("
    sig_int starting!
    ");
            for(i = 0;i < 300000;i++)
                for(j = 0;j < 40000;j++)
                    k += i * j;
     
            printf("sig_int finished!
    ");
        }
     
        int main()
        {
            unsigned int unslept;
     
            if(signal(SIGINT,sig_int) == SIG_ERR) {
                printf("signal(SIGINT) error!
    ");
                exit(1);
            }
            unslept = sleep2(5);
            printf("sleep2 returnd :%u
    ",unslept);
     
            exit(0);
        }

    输出结果为:

        :~/program/c$ ./test
        ^?                      // we type the interrupt character
        sig_int starting
        sleep2 returnd :0

    除了用来实现sleep函数外,alarm还常用于对可能阻塞的操作设置时间上限值。例如,程序中有一个读低速设备的可能阻塞的操作,我们希望超过一定时间量后就停止执行该操作:

        static void sig_alrm(int);
        
        int main()
        {
            int n;
            char line[MAXLINES];
            
            if(signal(SIGALRM,sig_alrm) == SIG_ERR) {
                printf("signal(SIGALRM) error!
    ");
                exit(1);
            }
            
            alarm(10);
            if((n = read(STDIN_FILENO,line,MAXLINE)) < 0) {
                printf("read error!
    ");
                exit(1);
            }
            alarm(0);
            
            write(STDOUT_FILENO,line,n);
            exit(0);
        }
        
        static void sig_alrm(int signo)
        {
            // nothing to do,just return to interrupt the read
        }

    遇到的问题:①跟上次程序类似,在第一次alarm调用和read调用之间有一个竞争条件。如果内核在这两个函数之间使进程阻塞,而其时间长度又超过闹钟时间,则read可能永远阻塞。②在read时候被alarm中断,如果系统调用是自动重启动的,则当从SIGALRM信号处理程序返回时,read并不被中断,在这种情形下,设置时间限制不起作用。

    解决方法:

        static void sig_alrm(int);
        static jmp_buf env_alrm;
        
        int main()
        {
            int n;
            char line[MAXLINES];
            
            if(signal(SIGALRM,sig_alrm) == SIG_ALRM) {
                printf("signal(SIGALRM) error!
    ");
                exit(1);
            }
            if(setjmp(env_alrm) != 0) {
                printf("read timeout!
    ");
                exit(1);
            }
            
            alarm(10);
            
            if((n = read(STDIN_FILENO,line,MAXLINE)) < 0) {
                printf("read error!
    ");
                exit(1);
            }
            
            alarm(0);
            
            write(STDOUT_FILENO,line,n);
            exit(0);
        }
        
        static void sig_alrm(int signo)
        {
            longjmp(env_alrm,1);
        }

    不管系统是否重新启动中断的系统调用,该程序都会如期的那样工作,但是要知道,该程序跟上次的程序一样存在于其他信号处理程序交互的问题。

     

    10、信号集

    所有应用程序在使用信号集前,都要对信号集调用sigemptyset或sigfillset一次,因为C编译器把未赋初值的外部和静态变量都初始化为0,而这是否与给定系统上信号集的实现相对应却并不清楚。

     

    11、sigprocmask函数

    sigprocmask函数用来检测或获取或更改当前信号屏蔽字(即阻塞还是非阻塞)。

    在调用sigprocmask后如果有任何未决的、不再阻塞的信号,则在sigprocmask返回前,至少会将其中一个信号递送给该进程。

    sigprocmask是仅为单线程的进程定义的。为处理多线程的进程中信号的屏蔽,提供了另一个单独的函数。

     

    12、sigpending函数

    sigpending函数返回未决的信号集,其中的各个信号对于调用进程是阻塞的而不能递送,因而也一定是当前未决的。如:

        static void sig_quit(int);
     
        int main()
        {
            sigset_t newmask,oldmask,pendmask;
    
            if(signal(SIGQUIT,sig_quit) == SIG_ERR)
                printf("can't catch SIGQUIT!
    ");
        
            // block SIGQUIT and save current signal mask
            sigemptyset(&newmask);
            sigaddset(&newmask,SIGQUIT);
        
            if(sigprocmask(SIG_BLOCK,&newmask,&oldmask) < 0)    //
                printf("SIG_BLOCK error!
    ");                    
                                                                
            sleep(5);
        
            if(sigpending(&pendmask) < 0)
                printf("sigpending error!
    ");
        
            if(sigismember(&pendmask,SIGQUIT))
                printf("SIGQUIT pending!
    ");                    //// reset signal mask which unblocks SIGQUIT
            if(sigprocmask(SIG_SETMASK,&oldmask,NULL) < 0)        //
                printf("SIG_SETMASK error!
    ");
            printf("SIGQUIT unblocked!
    ");
        
            sleep(5);
            exit(0);
        }
        
        static void sig_quit(int signo)
        {
            printf("caught SIGQUIT!
    ");
            if(signal(SIGQUIT,SIG_DFL) == SIG_ERR)
                printf("can't reset SIGQUIT!
    ");
        }

    运行结果:

        :~/program/c$ ./test
        ^                                        产生信号一次(在5秒中之内)
        SIGQUIT pending!                        从sleep返回后
        caught SIGQUIT!                            在信号处理程序中
        SIGQUIT unblocked!                        从sigprocmask返回后
        :~/program/c$

    ①与②之间,SIGQUIT信号是被阻塞的,所以也是未决的。

    ③调用后,SIGQUIT信号的阻塞被释放,在sigprocmask返回前,任何未决的、不再阻塞的信号会被递送给该进程。所以,SIGQUIT的信号处理函数会被调用,最后才处理sigprocmask后面的程序。

        :~/program/c$ ./test
        SIGQUIT unblocked!                        没有产生信号
        :~/program/c$

    SIGQUIT处理程序(sig_quit)中的printf语句先执行,然后再执行sigprocmask之后的printf语句,因为sigprocmask(SIG_SETMASK,&oldmask,NULL)执行完后,会立即递交SIGQUIT信号(之前被阻塞,现在解除阻塞),相应的信号处理函数会被立即调用。

    然后该进程再休眠5秒钟,如果在此期间再产生退出信号,那么因为在上次捕捉到该信号时,已将其处理方式设置为默认动作,所以这一次它就会使该进程终止。

     

    13、sigaction函数

    此函数的功能是检查或修改与指定信号相关联的处理动作(或同时执行这两种操作)。此函数取代了UNIX早期版本使用的signal函数。

    int sigaction(int signo,const struct sigaction *restrict act,struct sigaction *restrict oact);

    其中,参数signo是检测或修改其具体动作的信号编号。

    若act指针非空,则要修改其动作;

    若oact指针非空,则系统经由oact指针返回该信号的上一个动作。此函数使用下列结构:

        struct sigaction {
            void (*sa_handler)(int);            // addr of signal handler,or SIG_IGN,or SIG_DEF
                                                
            sigset_t sa_mask;                    // additional signals to block
            int sa_flags;                        // signal options
            
            // alternative handler
            void (*sa_sigaction)(int,siginfo_t *,void *);
        };

    当更改信号动作时,如果sa_handler字段包含一个信号捕捉函数的地址(与常量SIG_IGN或SIG_DFL相对),则sa_mask字段说明了一个信号集,在调用该信号捕捉函数之前,这一信号集要加到进程的信号屏蔽字中。仅当从信号捕捉函数返回时再将进程的信号屏蔽字复位为原先值。

     

    在信号处理程序被调用时,操作系统建立的新信号屏蔽字包括正被递送的信号。因此保证了在处理一个给定的信号时,如果这种信号再次发生,那么它会被阻塞到对前一个信号的处理结束为止。若同一种信号多次发生,通常并不将它们排队,所以如果在某种信号被阻塞时它发生了五次,那么对这种信号解除阻塞后,其信号处理函数通常只会被调用一次。

    一旦对给定的信号设置了一个动作,那么在调用sigaction显式地改变它之前,该设置就一直有效。不像旧的不可靠信号语义signal函数,对给定的信号设置了一个动作后,需要在信号处理函数中再次设置一次。

    sa_sigaction字段是一个替代的信号处理程序,当在sigaction结构中使用了SA_SIGNOF标志时,使用该信号处理程序。对于sa_sigaction字段和sa_handler字段这两者,其实现可能使用同一存储器,所以应用程序只能一次使用这两个字段中的一个。

     

    下列程序用sigaction实现的signal函数

        Sigfunc *signal(int signo,Sigfunc *func)
        {
            struct sigaction act,oact;
            
            act.sa_handler = func;
            sigemptyset(&act.sa_mask);
            act.sa_flags = 0;
            
            if(signo == SIGALRM) {
        #ifdef SA_INTERRUPT
                act.sa_flags |= SA_INTERRUPT;
        #endif
            } else {
        #ifdef SA_RESTART
                act.sa_flags |= SA_RESTART;
        #endif
            }
            
            if(sigaction(signo,&act,&oact) < 0)
                return (SIG_ERR);
            return (oact.sa_handler);
        }

    某些早期系统定义了SA_INTERRUPT标志。这些系统的默认方式是重新启动被中断的系统调用,而指定此标志则使系统调用被中断后不再重启动。Linux定义了SA_UNTERRUPT标志,以便与使用该标志的应用程序兼容。但是,如若信号处理程序是用sigaction设置的,那么其默认方式是不重新启动系统调用。除非说明了SA_RESTART标志,否则sigaction函数不再重启动被中断的系统调用。

     

    14、sigsetjmp和siglongjmp函数

    调用longjmp有一个问题。当捕捉到一个信号时,进入信号捕捉函数,此时当前信号被自动地加到进程的信号屏蔽字中。这阻止了后来产生的这种信号中断该信号处理程序。如果用longjmp跳出信号处理程序,那么,对此进程的信号屏蔽字会发生什么呢?

    为了允许这两种形式存在,定义了两个新函数sigsetjmp和siglongjmp。

        int sigsetjmp(sigjmp_buf env,int savemask);
        void siglongjmp(sigjmp_buf env,int val);

    如果savemask非0,则sigsetjmp在env中保存进程的当前信号屏蔽字。调用siglongjmp时,如果带非0savemask的sigsetjmp调用已经保存了env,则siglongjmp从其中恢复保存的信号屏蔽字。如:

    下面程序演示了在信号处理程序被调用时,系统所设置的信号屏蔽字如何自动地包括刚被捕捉到的信号:

        static sigjmp_buf jmpbuf;
        static volatile sig_atomic_t canjump;
        static void sig_usr1(int),sig_alarm(int);
    
        static void pr_mask(const char *str)
        {
            sigset_t sigset;
            int errno_save;
    
            errno_save = errno;
            if(sigprocmask(0,NULL,&sigset) < 0) {
                printf("sigprocmask error!
    ");
                exit(1);
            }
    
            printf("%s
    ",str);
    
            if(sigismember(&sigset,SIGINT))        printf("SIGINT
    ");
            if(sigismember(&sigset,SIGQUIT))    printf("SIGQUIT
    ");
            if(sigismember(&sigset,SIGUSR1))    printf("SIGUSR1
    ");
            if(sigismember(&sigset,SIGALRM))    printf("SIGALRM!
    ");
    
            printf("
    ");
            errno = errno_save;
        }
    
        int main()
        {
            if(signal(SIGUSR1,sig_usr1) == SIG_ERR) {
                printf("signal(SIGUSR1) error!
    ");
                exit(1);
            }
    
            if(signal(SIGALRM,sig_alarm) == SIG_ERR) {
                printf("signal(SIGALRM) error!
    ");
                exit(1);
            }
    
            if(sigsetjmp(jmpbuf,1)) {
                pr_mask("ending main:");
                exit(0);
            }
    
            canjump = 1;
    
            for(;;)
                pause();
        }
    
        static void sig_usr1(int signo)
        {
            time_t starttime;
        
            if(canjump == 0)
                return;
    
            pr_mask("starting sig_usr1:");
            alarm(3);
    
            starttime = time(NULL);
            for(;;) {
                if(time(NULL) > starttime + 5)
                    break;
            }
    
            pr_mask("finishing sig_usr1:");
    
            canjump = 0;
            siglongjmp(jmpbuf,1);
        }
    
        static void sig_alarm(int signo)
        {
            pr_mask("in sig_alarm:");
        }

    运行结果为:

        :~/program/c$ ./test &
        [1] 1280
        :~/program/c$ kill -USR1 1280
        :~/program/c$ starting sig_usr1:
        SIGUSR1
    
        in sig_alarm:
        SIGUSR1
        SIGALRM!
    
        finishing sig_usr1:
        SIGUSR1
    
        ending main:
    
    
        [1]+  Done                    ./test

     

    15、sigsuspend函数

    更改进程的信号屏蔽字可以阻塞所选择的信号,或解除对它们的阻塞,使用这种技术可以保护不希望由信号中断的代码临界区。如果希望对一个信号解除阻塞,然后pause以等待以前被阻塞的信号发生,则又将如何呢?假定信号是SIGINT,实现这一点的一种不正确的方法是:

        sigset_t newmask,oldmask;
        
        sigemptyset(&newmask);
        sigaddset(&newmask,SIGINT);
        
        // block SIGINT and save current signal mask
        if(sigprocmask(SIG_BLOCK,&newmask,&oldmask) < 0)
            printf("SIG_BLOCK error!
    ");
            
        // critical region of code
        if(sigprocmask(SIG_SETMASK,&oldmask,NULL) < 0)
            printf("SIG_SETMASK error!
    ");                    // ---------------- [1] -------------
            
        // window is open
        pause();        // wait for signal to occur            // ---------------- [2]--------------
        
        // continue to process

    如果在信号阻塞时将其发送给进程,那么该信号的传递就被推迟直到对它解除了阻塞。对应用程序而言,该信号好像发生在解除对SIGINT的阻塞和pause之间。如果发生了这种情况,或者如果在解除阻塞时刻和pause之间确实发生了信号,那么就产生了问题。因为我们可能不会再见到该信号(在[1]和[2]之间发生了SIGINT信号,则跳到信号处理函数中进行处理,处理后返回执行pause函数,pause因为等待SIGINT信号而导致阻塞,所以需要原子操作),所以从这种意义上说,在此时间窗口中发生的信号丢失了,这样就使得pause永远阻塞。

    为了纠正此问题,需要一个原子操作中先恢复信号屏蔽字,然后使进程休眠。这种功能是由sigsuspend函数提供的:

    int sigsuspend(const sigset_t *sigmask);

    将进程的信号屏蔽字设置为由sigmask指向的值。在捕捉到一个信号或发生了一个会终止该进程的信号之前,该进程被挂起。如果捕捉到一个信号而且从该信号处理程序返回,则sigsuspend返回,并且将该进程的信号屏蔽字设置为调用sigsuspend之前的值。

    下面程序显示了保护临界区,使其不被特定信号中断的正确方法:

        static void sig_init(int signo);
    
        static void pr_mask(const char *str)
        {    
            sigset_t sigset;
            int errno_save;
    
            errno_save = errno;
            if(sigprocmask(0,NULL,&sigset) < 0) {
                printf("sigprocmask error!
    ");
                exit(1);
            }
    
            printf("%s
    ",str);
    
            if(sigismember(&sigset,SIGINT))        printf("SIGINT
    ");
            if(sigismember(&sigset,SIGQUIT))    printf("SIGQUIT
    ");
            if(sigismember(&sigset,SIGUSR1))    printf("SIGUSR1
    ");
            if(sigismember(&sigset,SIGALRM))    printf("SIGALRM!
    ");
    
            printf("
    ");
            errno = errno_save;
        }
    
        int main()
        {
            sigset_t newmask,oldmask,waitmask;
    
            pr_mask("program start :");
    
            if(signal(SIGINT,sig_init) == SIG_ERR) {
                printf("signal(SIGINT) error!
    ");
                exit(1);
            }
            sigemptyset(&waitmask);
            sigaddset(&waitmask,SIGUSR1);
    
            sigemptyset(&newmask);
            sigaddset(&newmask,SIGINT);
    
            // block SIGINT and save current signal mask
            if(sigprocmask(SIG_BLOCK,&newmask,&oldmask) < 0) {
                printf("SIG_BLOCK error!
    ");
                exit(1);
            }
    
            // critical region of code
            pr_mask("in critical region :");
    
            // pause,allowing all signals except SIGUSR1
            if(sigsuspend(&waitmask) != -1) {
                printf("sigsuspend error!
    ");
                exit(1);
            }
    
            pr_mask("after return from sigsuspend:");
    
            // reset signal mask which unblocks SIGINT
            if(sigprocmask(SIG_SETMASK,&oldmask,NULL) < 0) {
                printf("SIG_SETMASK error!
    ");
                exit(1);
            }
    
            pr_mask("program exit:");
    
            exit(0);
        }
    
        static void sig_init(int signo)
        {
            pr_mask("in sig_init:");
        }

    输出结果为:

        :~/program/c$ ./test
        program start :
    
        in critical region :
        SIGINT
    
        ^C                                    // 键入中断字符
        in sig_init:
        SIGINT
        SIGUSR1
    
        after return from sigsuspend:
        SIGINT
    
        program exit:
    
        :~/program/c$

     

    可以用信号实现父、子进程之间的同步:

        static void charactatime(char *str)
        {
            char *ptr;
            int c;
    
            setbuf(stdout,NULL);
            for(ptr = str;(c = *ptr++) != 0;)
                putc(c,stdout);
        }
    
        static volatile sig_atomic_t sigflag;
        static sigset_t newmask,oldmask,zeromask;
    
        static void sig_usr(int signo)
        {
            sigflag = 1;
        }
    
        void TELL_WAIT()
        {
            if(signal(SIGUSR1,sig_usr) == SIG_ERR) {
                printf("signal(SIGUSR1) error!
    ");
                exit(1);
            }
            if(signal(SIGUSR2,sig_usr) == SIG_ERR) {
                printf("signal(SIGUSR2) error!
    ");
                exit(1);
            }
            sigemptyset(&zeromask);
            sigemptyset(&newmask);
            sigaddset(&newmask,SIGUSR1);
            sigaddset(&newmask,SIGUSR2);
    
            // block SIGUSR1 and SIGUSR2,and save current signal mask
            if(sigprocmask(SIG_BLOCK,&newmask,&oldmask) < 0) {
                printf("SIG_BLOCK error!
    ");
                exit(1);
            }
        }
    
        void TELL_PARENT(pid_t pid)
        {
            kill(pid,SIGUSR2);
        }
    
        void WAIT_PARENT()
        {
            while(sigflag == 0)
                sigsuspend(&zeromask);
    
            sigflag = 0;
    
            // reset signal mask to original value
            if(sigprocmask(SIG_SETMASK,&oldmask,NULL) < 0) {
                printf("SIG_SETMASK error!
    ");
                exit(1);
            }
        }
    
        void TELL_CHILD(pid_t pid)
        {
            kill(pid,SIGUSR1);
        }
    
        void WAIT_CHILD()
        {
            while(sigflag == 0)
                sigsuspend(&zeromask);
    
            sigflag = 0;
    
            // reset signal mask to original value.
            if(sigprocmask(SIG_SETMASK,&oldmask,NULL) < 0) {
                printf("SIG_SETMASK error!
    ");
                exit(1);
            }
        }
    
        int main()
        {
            pid_t pid;
    
            TELL_WAIT();
    
            if((pid = fork()) < 0) {
                printf("fork error!
    ");
                exit(1);
            } else if(pid == 0) {
                WAIT_PARENT();
                charactatime("output from child!
    ");
            } else {
                charactatime("output from parent!
    ");
                TELL_CHILD(pid);
            }
    
            exit(0);
        }

    如果在等待信号发生时希望去休眠,则适用sigsuspend函数是非常适当的,但是如果在等待信号期间希望调用其他系统函数,那么将会怎样呢?不幸的是,在单线程环境下对此问题没有妥善的解决方法。如果可以使用多线程,则可专门安排一个线程处理信号。

    如果不使用线程,那么我们能尽力做到最好的是,当信号发生时,在信号捕捉程序中对一个全局变量置1.

     

    16、sleep函数

    unsigned int sleep(unsigned int seconds);

    此函数使调用进程被挂起,直到满足以下条件之一:

    a、已经过了seconds所指定的墙上的时钟时间;

    b、调用进程捕捉到一个信号并从信号处理程序返回;

     

    在第一种情形中,返回值是0.当由于捕捉到某个信号sleep提早返回时(第二种情形),返回值是未睡够的秒数。

  • 相关阅读:
    阅读笔记--- 04
    站立会议--06个人进度
    站立会议--05 个人
    站立会议---04个人
    场景分析
    站立会议---03个人
    站立会议---02 个人进度
    计算某一天距离今天多少天,多少小时,多少分钟
    改变图片颜色
    手动调整导航控制器中的viewcontroller
  • 原文地址:https://www.cnblogs.com/sheshiji/p/3704179.html
Copyright © 2011-2022 走看看