linux内核层是什么,从用户层到内核层 - Linux内核中的信号机制_Linux编程_Linux公社-Linux系统门户网站...
1.簡介
如果進程要處理某一信號,那么要在進程中注冊該信號。注冊信號主要用來確定信號值及進程針對該信號值的動作之間的映射關系,即進程將要處理哪個進程和該信號被傳遞給進程時,將執行何種操作。主要有兩個函數實現信號的注冊:signal()和sigaction()。
2.signal()
signal()的函數原型如下:
void (*signal(int signum, void (*handler)(int)))(int);
在使用該調用的進程中加入以下頭文件:
#include
上述聲明格式比較復雜,如果不清楚如何使用,也可以通過下面這種類型定義的格式來使用(POSIX的定義):
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
但這種格式在不同的系統中有不同的類型定義,所以要使用這種格式,最好還是參考一下手冊。在調用中,參數signum指出要設置處理方法的信號。第二個參數handler是一個處理函數,或者是
SIG_IGN:忽略參數signum所指的信號。
SIG_DFL:恢復參數signum所指信號的處理方法為默認值。
傳遞給信號處理例程的整數參數是信號值,這樣可以使得一個信號處理例程處理多個信號。系統調用signal()返回值是指定信號signum前一次的處理例程或者錯誤時返回錯誤代碼SIG_ERR。
signal()通過系統調用sys_signal()為一個指定的信號設置用戶態處理函數。sys_signal()定義如下:
/*
* For backwards compatibility.? Functionality superseded by sigaction.
*/
asmlinkage unsigned long
sys_signal(int sig, __sighandler_t handler)
{
struct k_sigaction new_sa, old_sa;
int ret;
new_sa.sa.sa_handler = handler;
new_sa.sa.sa_flags = SA_ONESHOT | SA_NOMASK;
ret = do_sigaction(sig, &new_sa, &old_sa);
return ret ? ret : (unsigned long)old_sa.sa.sa_handler;
}
__sighandler_t的定義如下:
typedef void __signalfn_t(int);
typedef __signalfn_t __user *__sighandler_t;
信號由sys_signal()的第一個參數指定,信號處理函數的地址由第二個參數指定。sys_signal()根據這兩個參數設置一個k_sigaction結構,然后調用do_sigaction(),該函數的定義我們會在后面具體講解。
2.sigaction()
sigaction()的函數原型如下:
sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
sigaction()對應的系統調用為do_sigaction(),下面我們具體講解do_sigaction()函數,其定義如下:
2.1do_sigaction()
int
do_sigaction(int sig, const struct k_sigaction *act, struct k_sigaction *oact)
{
struct k_sigaction *k;
if (!valid_signal(sig) || sig < 1 || (act && sig_kernel_only(sig)))
return -EINVAL;
k = ¤tt->sighand->action[sig-1];
spin_lock_irq(¤tt->sighand->siglock);
if (signal_pending(current)) {
/*
* If there might be a fatal signal pending on multiple
* threads, make sure we take it before changing the action.
*/
spin_unlock_irq(¤tt->sighand->siglock);
return -ERESTARTNOINTR;
}
if (oact)//把原來的k_sigaction保存到oact結構中,這里是對整個數據結構進行復制
*oact = *k;
if (act) {
/*
* POSIX 3.3.1.3:
*? "Setting a signal action to SIG_IGN for a signal that is
*? pending shall cause the pending signal to be discarded,
*? whether or not it is blocked."
*
*? "Setting a signal action to SIG_DFL for a signal that is
*? pending and whose default action is to ignore the signal
*? (for example, SIGCHLD), shall cause the pending signal to
*? be discarded, whether or not it is blocked"
*/
if (act->sa.sa_handler == SIG_IGN ||
(act->sa.sa_handler == SIG_DFL &&
sig_kernel_ignore(sig))) {
/*
* This is a fairly rare case, so we only take the
* tasklist_lock once we're sure we'll need it.
* Now we must do this little unlock and relock
* dance to maintain the lock hierarchy.
*/
struct task_struct *t = current;
spin_unlock_irq(&t->sighand->siglock);
read_lock(&tasklist_lock);
spin_lock_irq(&t->sighand->siglock);
*k = *act; //把新的k_sigaction結構復制到進程的sighand->action中
sigdelsetmask(&k->sa.sa_mask,
sigmask(SIGKILL) | sigmask(SIGSTOP));
rm_from_queue(sigmask(sig), &t->signal->shared_pending);
do {
rm_from_queue(sigmask(sig), &t->pending);
recalc_sigpending_tsk(t);
t = next_thread(t);
} while (t != current);
spin_unlock_irq(¤t->sighand->siglock);
read_unlock(&tasklist_lock);
return 0;
}
*k = *act; //把新的k_sigaction結構復制到進程的sighand->action中
sigdelsetmask(&k->sa.sa_mask,
sigmask(SIGKILL) | sigmask(SIGSTOP));
}
spin_unlock_irq(¤tt->sighand->siglock);
return 0;
}
總結
以上是生活随笔為你收集整理的linux内核层是什么,从用户层到内核层 - Linux内核中的信号机制_Linux编程_Linux公社-Linux系统门户网站...的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 海通证券开户用什么交易软件?
- 下一篇: linux系统如何查看是否是线程死锁,多