BUG: scheduling while atomic 分析【转】

本文转载自:https://blog.csdn.net/cfy_phonex/article/details/12090943

遇到一个典型的schedule问题。

  1. < 3>[26578.636839] C1 [ swapper/1] BUG: scheduling while atomic: swapper/1/0/0x00000002
  2. < 6>[26578.636869] C0 [ kworker/u:1] CPU1 is up
  3. < 4>[26578.636900] C1 [ swapper/1] Modules linked in: bcm15500_i2c_ts
  4. < 4>[26578.636961] C1 [ swapper/1] [<c00146d0>] (unwind_backtrace+0x0/0x11c) from [<c0602684>] (__schedule+0x70/0x6e0)
  5. < 4>[26578.636991] C1 [ swapper/1] [<c0602684>] (__schedule+0x70/0x6e0) from [<c06030ec>] (schedule_preempt_disabled+0x14/0x20)
  6. < 4>[26578.637052] C1 [ swapper/1] [<c06030ec>] (schedule_preempt_disabled+0x14/0x20) from [<c000f05c>] (cpu_idle+0xf0/0x104)
  7. < 4>[26578.637083] C1 [ swapper/1] [<c000f05c>] (cpu_idle+0xf0/0x104) from [<c05e98e0>] (cpu_die+0x2c/0x5c)
  8. < 3>[26578.637510] C1 [ swapper/1] BUG: scheduling while atomic: swapper/1/0/0x00000002
  9. < 4>[26578.637510] C1 [ swapper/1] Modules linked in: bcm15500_i2c_ts
  10. < 4>[26578.637602] C1 [ swapper/1] [<c00146d0>] (unwind_backtrace+0x0/0x11c) from [<c0602684>] (__schedule+0x70/0x6e0)
  11. < 4>[26578.637663] C1 [ swapper/1] [<c0602684>] (__schedule+0x70/0x6e0) from [<c06030ec>] (schedule_preempt_disabled+0x14/0x20)
  12. < 4>[26578.637724] C1 [ swapper/1] [<c06030ec>] (schedule_preempt_disabled+0x14/0x20) from [<c000f05c>] (cpu_idle+0xf0/0x104)
  13. < 4>[26578.637754] C1 [ swapper/1] [<c000f05c>] (cpu_idle+0xf0/0x104) from [<c05e98e0>] (cpu_die+0x2c/0x5c)
  14. < 3>[26578.648069] C1 [ swapper/1] BUG: scheduling while atomic: swapper/1/0/0x00000002


查看源代码

  1. /*
  2. * __schedule() is the main scheduler function.
  3. */
  4. static void __sched __schedule(void)
  5. {
  6. struct task_struct *prev, *next;
  7. unsigned long *switch_count;
  8. struct rq *rq;
  9. int cpu;
  10. need_resched:
  11. preempt_disable();
  12. cpu = smp_processor_id();
  13. rq = cpu_rq(cpu);
  14. rcu_note_context_switch(cpu);
  15. prev = rq->curr;
  16. schedule_debug(prev);
  17.     ....
  1. /*
  2. * Print scheduling while atomic bug:
  3. */
  4. static noinline void __schedule_bug(struct task_struct *prev)
  5. {
  6. if (oops_in_progress)
  7. return;
  8. printk(KERN_ERR "BUG: scheduling while atomic: %s/%d/0x%08x\n",
  9. prev->comm, prev->pid, preempt_count());
  10. debug_show_held_locks(prev);
  11. print_modules();
  12. if (irqs_disabled())
  13. print_irqtrace_events(prev);
  14. dump_stack();
  15. }
  16. /*
  17. * Various schedule()-time debugging checks and statistics:
  18. */
  19. static inline void schedule_debug(struct task_struct *prev)
  20. {
  21. /*
  22. * Test if we are atomic. Since do_exit() needs to call into
  23. * schedule() atomically, we ignore that path for now.
  24. * Otherwise, whine if we are scheduling when we should not be.
  25. */
  26. if (unlikely(in_atomic_preempt_off() && !prev->exit_state))
  27. __schedule_bug(prev);
  28. rcu_sleep_check();
  29. profile_hit(SCHED_PROFILING, __builtin_return_address( 0));
  30. schedstat_inc(this_rq(), sched_count);
  31. }


可以看出, 满足如下条件将会打印该出错信息

unlikely(in_atomic_preempt_off() && !prev->exit_state

为0表示TASK_RUNNING状态,当前进程在运行; 并且处于原子状态,,那么就不能切换给其它的进程

  1. Linux/include/linux/sched.h
  2. /*
  3. * Task state bitmask. NOTE! These bits are also
  4. * encoded in fs/proc/array.c: get_task_state().
  5. *
  6. * We have two separate sets of flags: task->state
  7. * is about runnability, while task->exit_state are
  8. * about the task exiting. Confusing, but this way
  9. * modifying one set can't modify the other one by
  10. * mistake.
  11. */
  12. #define TASK_RUNNING 0
  13. #define TASK_INTERRUPTIBLE 1
  14. #define TASK_UNINTERRUPTIBLE 2
  15. #define __TASK_STOPPED 4
  16. #define __TASK_TRACED 8
  17. /* in tsk->exit_state */
  18. #define EXIT_ZOMBIE 16
  19. #define EXIT_DEAD 32
  20. /* in tsk->state again */
  21. #define TASK_DEAD 64
  22. #define TASK_WAKEKILL 128
  23. #define TASK_WAKING 256
  24. #define TASK_STATE_MAX 512
  1. kernel/include/linux/hardirq.h
  2. #if defined(CONFIG_PREEMPT_COUNT)
  3. # define PREEMPT_CHECK_OFFSET 1
  4. #else
  5. # define PREEMPT_CHECK_OFFSET 0
  6. #endif
  7. /*
  8. * Are we running in atomic context? WARNING: this macro cannot
  9. * always detect atomic context; in particular, it cannot know about
  10. * held spinlocks in non-preemptible kernels. Thus it should not be
  11. * used in the general case to determine whether sleeping is possible.
  12. * Do not use in_atomic() in driver code.
  13. */
  14. #define in_atomic() ((preempt_count() & ~PREEMPT_ACTIVE) != 0)
  15. /*
  16. * Check whether we were atomic before we did preempt_disable():
  17. * (used by the scheduler, *after* releasing the kernel lock)
  18. */
  19. #define in_atomic_preempt_off() \
  20. ((preempt_count() & ~PREEMPT_ACTIVE) != PREEMPT_CHECK_OFFSET)

结论整理

linux内核打印"BUG: scheduling while atomic"和"bad: scheduling from the idle thread"错误的时候,

通常是在中断处理函数中调用了可以休眠的函数,如semaphore,mutex,sleep之类的可休眠的函数,

而linux内核要求在中断处理的时候,不允许系统调度,不允许抢占,要等到中断处理完成才能做其他事情。

因此,要充分考虑中断处理的时间,一定不能太久。
另外一个能产生此问题的是在idle进程里面,做了不该做的事情。现在Linux用于很多手持式设备,为了降低功耗,

通常的作法是在idle进程里面降低CPU或RAM的频率、关闭一些设备等等。要保证这些动作的原子性才能确保

不发生"bad: scheduling from the idle thread"这样的错误!

禁止内核抢占是指内核不会主动的抢占你的process,但是现在是你在自己的程序中主动call schedule(),

kernel并不能阻止你这么作。

Scheduling while atomic" means that a thread has called schedule() during an operation which is supposed to be atomic (ie uninterrupted).

  1. 190 NOTE: ***** WARNING *****
  2. 191 NEVER SLEEP IN A COMPLETION HANDLER. These are normally called
  3. 192 during hardware interrupt processing. If you can, defer substantial
  4. 193 work to a tasklet (bottom half) to keep system latencies low. You'll
  5. 194 probably need to use spinlocks to protect data structures you manipulate
  6. 195 in completion handlers.
    1. GFP_ATOMIC is used when
    2. (a) you are inside a completion handler, an interrupt, bottom half, tasklet or timer, or
    3. (b) you are holding a spinlock or rwlock (does not apply to semaphores), or
    4. (c) current->state != TASK_RUNNING, this is the case only after you've changed it.
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入式小庄老师

要是觉得不错,就给我点支持吧

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值