Posted on 2012-12-14 10:58
鑫龍 閱讀(383)
評論(0) 編輯 收藏 引用 所屬分類:
linux編程
這是一個網(wǎng)友的提問:
在 UNIX的system()函數(shù)實現(xiàn)過程中,要求在父進程中忽略掉SIGINT和SIGQUIT信號,但是要將SIGCHLD信號阻塞(在子進程中將 SIGINT和SIGQUIT信號設為默認,SIGCHLD信號解鎖)。子進程執(zhí)行完畢后,在父進程中調(diào)用waitpid(pid_t, &state, 0)。問題:
1、若父進程已被waitpid阻塞,在子進程返回時,此時在父進程中SIGCHLD被阻塞(BLOCK),父進程收不到SIGCHLD信號,waitpid()函數(shù)能否正確返回,收集到子進程的信息?
2、 waitpid若能正確完成,在以后父進程中,將SIGCHLD信號UNBLOCK,用sigprocmask()函數(shù)解鎖,書上說,在 sigprocmask()函數(shù)返回以前,會將以前阻塞的信號發(fā)送給進程,父進程是否還能收到SIGCHLD信號?若能收到何必在開始時將SIGCHLD 進程阻塞。
簡單的對這個問題的解釋是wait及其變體并不是通過sigchld信號來知道子進程狀態(tài)的。
sigprocmask 阻塞的是有signal或sigaction設置的信號處理程序,即帶有SIGCHLD_Handle()等處理函數(shù)。wait不是靠接收sigchld 信號獲得子進程的退出狀態(tài)的,如果進程中同時設置了signal和wait,則子進程退出后發(fā)出sigchld信號,交到signal的信號處理程序處 理,wait接收到子進程退出狀態(tài)。
只是接收sigchld,而不調(diào)用wait還是會使子進程僵死的。一般的只有調(diào)用wait才能使子進程不成為僵死進程(除了2次fork 等或其他一些手段)。
概括下:waitpid不是依靠SIGCHLD是否到達來判斷子進程是否退出,但是如果設置了SIGCHLD的處理函數(shù),那么就需要等待SIGCHLD信號 的發(fā)生并完成信號處理函數(shù),waitpid才能接收到子進程的退出狀態(tài)。在APUE中的system()實現(xiàn)中阻塞了SIGCHLD信號,但是并沒有設置 信號處理函數(shù),所以waitpid在阻塞了SIGCHLD的情況下依然能正常返回,因為SIGCHLD在未設置信號處理函數(shù)的情況下不會影響到 waitpid的工作。至于為什么要阻塞SIGCHLD信號呢?那就是為了防止其他程序(main除了會調(diào)用system還會使用其他程序)設置了 SIGCHLD的信號處理函數(shù),如果其他程序設置了SIGCHLD信號處理函數(shù),在waitpid等待子程序的返回前,要去處理SIGCHLD信號處理程 序,如果阻塞了該信號,就不會去處理該信號處理程序,防止多余信息在system()中的出現(xiàn)。