唤醒等待队列中的等待进程的函数wake_up()函数的核心实现函数是__wake_up_common()函数。
__wake_up_common(wait_queue_head_t *q, int mode,
int nr_exclusive, int wake_flags, void *key)
参数介绍:
q : 是等待队列头;
mode: 是进程的状态模式
其取值为: TASK_INTERRUPTIBLE, TASK_UNITERRUPTIBLE
nr_exclusive:是 number exclusive;
wake_flags : 是同步唤醒sync,还是异步唤醒 async;
key : 一般为NULL;
static void __wake_up_common(wait_queue_head_t *q, int mode,
int nr_exclusive, int wake_flags, void *key)
{
wait_queue_t *curr, *next;
list_for_each_entry_safe(curr, next, &q->task_list, task_list)
{
unsigned int flags = curr->flags;
if( curr->func(curr, mode, wake_flags, key) &&
(flags & WQ_FLAG_EXCLUSIVE) && !--nr_exclusive )
break;
}
}
解读一下 __wake_up_common()函数:
list_for_each_entry_safe()函数用于遍历等待队列中的数据项,并且在curr中存放当前的wait_queue_t项的地址;
curr->func()是用来唤醒等待队列中的进程的实现函数;
curr->func = autoremove_wake_function() 或 default_wake_function
其实autoremove_wake_function()与default_wake_function()函数的本质上是相同的,他们最终是在调用try_to_wake_up()函数的。
try_to_wake_up()函数是从等待队列中移走一个进程,然后将其放入到就绪队列中等待调度器schedule()的调度。
所以curr->func()函数的作用就是从等待队列中移走进程,将其放到就绪队列中,达到实现进程唤醒的目的。
__wake_up_common()函数是每次唤醒所有的等待进程,还是只唤醒一个,还是唤醒N个。这取决于 nr_exclusive 参数
当 nr_exclusive = 0时,__wake_up_common()每次唤醒所有的等待进程。
当 nr_exclusive = 1时,并且等待队列中的等待进程时互斥等待唤醒进程,则__wake_up_common()每次只唤醒一个等待进程。
当 nr_exclusive = N时,__wake_up_common()每次唤醒N个等待进程或者N-1个非互斥等待进程与1个互斥等待进程;
__wake_up()函数的实现:
void __wake_up(wait_queue_head_t *q, unsigned int mode,
int nr_exclusive, void *key)
{
unsigned int flags;
spin_lock_irqsave(&p->lock,flags);
__wake_up_common(q,mode,nr_exclusive,0,key);
spin_unlock_irqstore(&p->lock, flags);
}
wake_up()函数的实现:
#define wake_up(wq) __wake_up(wq, TASK_NORMAL, 1,NULL)
#define wake_up_nr(wq,nr) __wake_up(wq, TASK_NORMAL,nr, NULL)
#define wake_up_all(wq) __wake_up(wq, TASK_NORMAL, 0, NULL)
#define wake_up_interruptible(wq) \
__wake_up(wq,TASK_INTERRUPTIBLE,1,NULL)
#define wake_up_interruptible_nr(wq,nr) \
__wake_up(wq, TASK_INTERRUPTIBLE,nr,NULL)
#define wake_up_interruptible_all(wq) \
__wake_up(wq, TASK_INTERRUPTIBLE,0,NULL)