- 论坛徽章:
- 0
|
本帖最后由 SharkBones 于 2011-10-14 17:00 编辑
回复 6# embeddedlwp
是我描述的不清楚,应该这样描述:已经加入runqueue的进程不会再被放回等待队列,而原来就在等待队列中没有被唤醒的进程则依然在队列中。
可以从代码中看出来,在include/linux/wait.h中:- #define __WAITQUEUE_INITIALIZER(name, tsk) { \
- .private = tsk, \
- .func = default_wake_function, \
- .task_list = { NULL, NULL } }
- #define DECLARE_WAITQUEUE(name, tsk) \
- wait_queue_t name = __WAITQUEUE_INITIALIZER(name, tsk)
复制代码 default_wake_function定义在kernel/sched.c中,如下:- int default_wake_function(wait_queue_t *curr, unsigned mode, int wake_flags,
- void *key)
- {
- return try_to_wake_up(curr->private, mode, wake_flags);
- }
复制代码 调用了tyr_to_wake_up,- static int try_to_wake_up(struct task_struct *p, unsigned int state,
- int wake_flags)
- {
- int cpu, orig_cpu, this_cpu, success = 0;
- unsigned long flags;
- struct rq *rq, *orig_rq;
- ...
- if (cpu == this_cpu)
- schedstat_inc(p, se.nr_wakeups_local);
- else
- schedstat_inc(p, se.nr_wakeups_remote);
- activate_task(rq, p, 1);
- success = 1
- ...
复制代码 active_task将一个task放置于runqueue中,然后调用schedule,最终调用_ _remove_wait_queue()把该task从等待队列中移除。
另一方面,wake_up调用了__wake_up_common函数,- static void __wake_up_common(wait_queue_head_t *q, unsigned 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 flags = curr->flags;
- if (curr->func(curr, mode, wake_flags, key) &&
- (flags & WQ_FLAG_EXCLUSIVE) && !--nr_exclusive)
- break;
- }
- }
复制代码 可以看到curr->func调用的就是default_wake_function,nr_exclusive减为0的时候就不在唤醒等待队列中的task了。 |
|