首先linux内核中获取信号量的三个函数:
void down(struct semaphoore *sem);
int down_interruptible(struct semaphore *sem);
int down_trylock(struct semaphore *sem);
前两个在获取信号量的时候都可能休眠,不同的是down_interruptible可以被用户进程中断,如果被中断,返回非零值,而调用者不会拥有该信号量,而down会在获取失败的时候一直等待。最后一个函数永远不会休眠,如果信号量在调用时不可用,会立即返回一个非零值。
内核中的中断,tasklet和定时器都不能在其执行过程中休眠,其实tasklet和定时器也是终端,tasklet是在软终端的基础上实现的。例如在中断处理程序中调用了down或down_interruptible,那么如果此时信号量不可用,终端例程将会休眠,而这是不允许的,系统将会崩溃。tasklet和工作队列的一个关键区别是:tasklet会在很短的时间内很快执行,并且以原子模式执行,而工作队列函数可具有更成的延迟并且不必原子化,也就是说工作队列是可以休眠的。