aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/sched/core.c
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/sched/core.c')
-rw-r--r--kernel/sched/core.c20
1 files changed, 15 insertions, 5 deletions
diff --git a/kernel/sched/core.c b/kernel/sched/core.c
index fe89afac4d09..ee61f5affd20 100644
--- a/kernel/sched/core.c
+++ b/kernel/sched/core.c
@@ -525,8 +525,10 @@ void resched_task(struct task_struct *p)
525 set_tsk_need_resched(p); 525 set_tsk_need_resched(p);
526 526
527 cpu = task_cpu(p); 527 cpu = task_cpu(p);
528 if (cpu == smp_processor_id()) 528 if (cpu == smp_processor_id()) {
529 set_preempt_need_resched();
529 return; 530 return;
531 }
530 532
531 /* NEED_RESCHED must be visible before we test polling */ 533 /* NEED_RESCHED must be visible before we test polling */
532 smp_mb(); 534 smp_mb();
@@ -1391,6 +1393,14 @@ static void sched_ttwu_pending(void)
1391 1393
1392void scheduler_ipi(void) 1394void scheduler_ipi(void)
1393{ 1395{
1396 /*
1397 * Fold TIF_NEED_RESCHED into the preempt_count; anybody setting
1398 * TIF_NEED_RESCHED remotely (for the first time) will also send
1399 * this IPI.
1400 */
1401 if (tif_need_resched())
1402 set_preempt_need_resched();
1403
1394 if (llist_empty(&this_rq()->wake_list) 1404 if (llist_empty(&this_rq()->wake_list)
1395 && !tick_nohz_full_cpu(smp_processor_id()) 1405 && !tick_nohz_full_cpu(smp_processor_id())
1396 && !got_nohz_idle_kick()) 1406 && !got_nohz_idle_kick())
@@ -1714,7 +1724,7 @@ void sched_fork(struct task_struct *p)
1714#endif 1724#endif
1715#ifdef CONFIG_PREEMPT_COUNT 1725#ifdef CONFIG_PREEMPT_COUNT
1716 /* Want to start with kernel preemption disabled. */ 1726 /* Want to start with kernel preemption disabled. */
1717 task_thread_info(p)->preempt_count = 1; 1727 task_thread_info(p)->preempt_count = PREEMPT_DISABLED;
1718#endif 1728#endif
1719#ifdef CONFIG_SMP 1729#ifdef CONFIG_SMP
1720 plist_node_init(&p->pushable_tasks, MAX_PRIO); 1730 plist_node_init(&p->pushable_tasks, MAX_PRIO);
@@ -2425,6 +2435,7 @@ need_resched:
2425 put_prev_task(rq, prev); 2435 put_prev_task(rq, prev);
2426 next = pick_next_task(rq); 2436 next = pick_next_task(rq);
2427 clear_tsk_need_resched(prev); 2437 clear_tsk_need_resched(prev);
2438 clear_preempt_need_resched();
2428 rq->skip_clock_update = 0; 2439 rq->skip_clock_update = 0;
2429 2440
2430 if (likely(prev != next)) { 2441 if (likely(prev != next)) {
@@ -2536,11 +2547,10 @@ EXPORT_SYMBOL(preempt_schedule);
2536 */ 2547 */
2537asmlinkage void __sched preempt_schedule_irq(void) 2548asmlinkage void __sched preempt_schedule_irq(void)
2538{ 2549{
2539 struct thread_info *ti = current_thread_info();
2540 enum ctx_state prev_state; 2550 enum ctx_state prev_state;
2541 2551
2542 /* Catch callers which need to be fixed */ 2552 /* Catch callers which need to be fixed */
2543 BUG_ON(ti->preempt_count || !irqs_disabled()); 2553 BUG_ON(preempt_count() || !irqs_disabled());
2544 2554
2545 prev_state = exception_enter(); 2555 prev_state = exception_enter();
2546 2556
@@ -4207,7 +4217,7 @@ void init_idle(struct task_struct *idle, int cpu)
4207 raw_spin_unlock_irqrestore(&rq->lock, flags); 4217 raw_spin_unlock_irqrestore(&rq->lock, flags);
4208 4218
4209 /* Set the preempt count _outside_ the spinlocks! */ 4219 /* Set the preempt count _outside_ the spinlocks! */
4210 task_thread_info(idle)->preempt_count = 0; 4220 task_thread_info(idle)->preempt_count = PREEMPT_ENABLED;
4211 4221
4212 /* 4222 /*
4213 * The idle tasks have their own, simple scheduling class: 4223 * The idle tasks have their own, simple scheduling class: