提交 a2250238 编写于 作者: P Peter Zijlstra 提交者: Ingo Molnar

sched/core: Explain sleep/wakeup in a better way

There were a few questions wrt. how sleep-wakeup works. Try and explain
it more.
Requested-by: NWill Deacon <will.deacon@arm.com>
Signed-off-by: NPeter Zijlstra (Intel) <peterz@infradead.org>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: NIngo Molnar <mingo@kernel.org>
上级 3c3fcb45
...@@ -265,17 +265,6 @@ extern char ___assert_task_state[1 - 2*!!( ...@@ -265,17 +265,6 @@ extern char ___assert_task_state[1 - 2*!!(
smp_store_mb((tsk)->state, (state_value)); \ smp_store_mb((tsk)->state, (state_value)); \
} while (0) } while (0)
/*
* set_current_state() includes a barrier so that the write of current->state
* is correctly serialised wrt the caller's subsequent test of whether to
* actually sleep:
*
* set_current_state(TASK_UNINTERRUPTIBLE);
* if (do_i_need_to_sleep())
* schedule();
*
* If the caller does not need such serialisation then use __set_current_state()
*/
#define __set_current_state(state_value) \ #define __set_current_state(state_value) \
do { \ do { \
current->task_state_change = _THIS_IP_; \ current->task_state_change = _THIS_IP_; \
...@@ -289,6 +278,14 @@ extern char ___assert_task_state[1 - 2*!!( ...@@ -289,6 +278,14 @@ extern char ___assert_task_state[1 - 2*!!(
#else #else
/*
* @tsk had better be current, or you get to keep the pieces.
*
* The only reason is that computing current can be more expensive than
* using a pointer that's already available.
*
* Therefore, see set_current_state().
*/
#define __set_task_state(tsk, state_value) \ #define __set_task_state(tsk, state_value) \
do { (tsk)->state = (state_value); } while (0) do { (tsk)->state = (state_value); } while (0)
#define set_task_state(tsk, state_value) \ #define set_task_state(tsk, state_value) \
...@@ -299,11 +296,34 @@ extern char ___assert_task_state[1 - 2*!!( ...@@ -299,11 +296,34 @@ extern char ___assert_task_state[1 - 2*!!(
* is correctly serialised wrt the caller's subsequent test of whether to * is correctly serialised wrt the caller's subsequent test of whether to
* actually sleep: * actually sleep:
* *
* for (;;) {
* set_current_state(TASK_UNINTERRUPTIBLE); * set_current_state(TASK_UNINTERRUPTIBLE);
* if (do_i_need_to_sleep()) * if (!need_sleep)
* break;
*
* schedule(); * schedule();
* }
* __set_current_state(TASK_RUNNING);
*
* If the caller does not need such serialisation (because, for instance, the
* condition test and condition change and wakeup are under the same lock) then
* use __set_current_state().
*
* The above is typically ordered against the wakeup, which does:
*
* need_sleep = false;
* wake_up_state(p, TASK_UNINTERRUPTIBLE);
*
* Where wake_up_state() (and all other wakeup primitives) imply enough
* barriers to order the store of the variable against wakeup.
*
* Wakeup will do: if (@state & p->state) p->state = TASK_RUNNING, that is,
* once it observes the TASK_UNINTERRUPTIBLE store the waking CPU can issue a
* TASK_RUNNING store which can collide with __set_current_state(TASK_RUNNING).
*
* This is obviously fine, since they both store the exact same value.
* *
* If the caller does not need such serialisation then use __set_current_state() * Also see the comments of try_to_wake_up().
*/ */
#define __set_current_state(state_value) \ #define __set_current_state(state_value) \
do { current->state = (state_value); } while (0) do { current->state = (state_value); } while (0)
......
...@@ -1995,14 +1995,15 @@ static void ttwu_queue(struct task_struct *p, int cpu, int wake_flags) ...@@ -1995,14 +1995,15 @@ static void ttwu_queue(struct task_struct *p, int cpu, int wake_flags)
* @state: the mask of task states that can be woken * @state: the mask of task states that can be woken
* @wake_flags: wake modifier flags (WF_*) * @wake_flags: wake modifier flags (WF_*)
* *
* Put it on the run-queue if it's not already there. The "current" * If (@state & @p->state) @p->state = TASK_RUNNING.
* thread is always on the run-queue (except when the actual *
* re-schedule is in progress), and as such you're allowed to do * If the task was not queued/runnable, also place it back on a runqueue.
* the simpler "current->state = TASK_RUNNING" to mark yourself *
* runnable without the overhead of this. * Atomic against schedule() which would dequeue a task, also see
* * set_current_state().
* Return: %true if @p was woken up, %false if it was already running. *
* or @state didn't match @p's state. * Return: %true if @p->state changes (an actual wakeup was done),
* %false otherwise.
*/ */
static int static int
try_to_wake_up(struct task_struct *p, unsigned int state, int wake_flags) try_to_wake_up(struct task_struct *p, unsigned int state, int wake_flags)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册