免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2297 | 回复: 0
打印 上一主题 下一主题

请教arch/x86/kernel/traps.c中的do_debug函数是做什么的 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-02-08 10:41 |只看该作者 |倒序浏览
本帖最后由 cuic139 于 2012-02-09 13:15 编辑

请教谁知道arch/x86/kernel/traps.c中的do_debug函数是做什么的,调用流程和调用原因是怎样的?,谢谢

  1. /*
  2. * Our handling of the processor debug registers is non-trivial.
  3. * We do not clear them on entry and exit from the kernel. Therefore
  4. * it is possible to get a watchpoint trap here from inside the kernel.
  5. * However, the code in ./ptrace.c has ensured that the user can
  6. * only set watchpoints on userspace addresses. Therefore the in-kernel
  7. * watchpoint trap can only occur in code which is reading/writing
  8. * from user space. Such code must not hold kernel locks (since it
  9. * can equally take a page fault), therefore it is safe to call
  10. * force_sig_info even though that claims and releases locks.
  11. *
  12. * Code in ./signal.c ensures that the debug control register
  13. * is restored before we deliver any signal, and therefore that
  14. * user code runs with the correct debug control register even though
  15. * we clear it here.
  16. *
  17. * Being careful here means that we don't have to be as careful in a
  18. * lot of more complicated places (task switching can be a bit lazy
  19. * about restoring all the debug state, and ptrace doesn't have to
  20. * find every occurrence of the TF bit that could be saved away even
  21. * by user code)
  22. *
  23. * May run on IST stack.
  24. */
  25. dotraplinkage void __kprobes do_debug(struct pt_regs *regs, long error_code)
  26. {
  27.         struct task_struct *tsk = current;
  28.         int user_icebp = 0;
  29.         unsigned long dr6;
  30.         int si_code;

  31.         get_debugreg(dr6, 6);

  32.         /* Filter out all the reserved bits which are preset to 1 */
  33.         dr6 &= ~DR6_RESERVED;

  34.         /*
  35.          * If dr6 has no reason to give us about the origin of this trap,
  36.          * then it's very likely the result of an icebp/int01 trap.
  37.          * User wants a sigtrap for that.
  38.          */
  39.         if (!dr6 && user_mode(regs))
  40.                 user_icebp = 1;

  41.         /* Catch kmemcheck conditions first of all! */
  42.         if ((dr6 & DR_STEP) && kmemcheck_trap(regs))
  43.                 return;

  44.         /* DR6 may or may not be cleared by the CPU */
  45.         set_debugreg(0, 6);

  46.         /*
  47.          * The processor cleared BTF, so don't mark that we need it set.
  48.          */
  49.         clear_tsk_thread_flag(tsk, TIF_BLOCKSTEP);

  50.         /* Store the virtualized DR6 value */
  51.         tsk->thread.debugreg6 = dr6;

  52. #ifdef  CONFIG_KDB
  53.         if (kdb(LKDB_REASON_DEBUG, error_code, regs))
  54.         {
  55.                 return;
  56.         }
  57. #endif  /* CONFIG_KDB */

  58.         if (notify_die(DIE_DEBUG, "debug", regs, PTR_ERR(&dr6), error_code,
  59.                                                         SIGTRAP) == NOTIFY_STOP)
  60.                 return;

  61.         /* It's safe to allow irq's after DR6 has been saved */
  62.         preempt_conditional_sti(regs);

  63.         if (regs->flags & X86_VM_MASK) {
  64.                 handle_vm86_trap((struct kernel_vm86_regs *) regs,
  65.                                 error_code, 1);
  66.                 preempt_conditional_cli(regs);
  67.                 return;
  68.         }

  69.         /*
  70.          * Single-stepping through system calls: ignore any exceptions in
  71.          * kernel space, but re-enable TF when returning to user mode.
  72.          *
  73.          * We already checked v86 mode above, so we can check for kernel mode
  74.          * by just checking the CPL of CS.
  75.          */
  76.         if ((dr6 & DR_STEP) && !user_mode(regs)) {
  77.                 tsk->thread.debugreg6 &= ~DR_STEP;
  78.                 set_tsk_thread_flag(tsk, TIF_SINGLESTEP);
  79.                 regs->flags &= ~X86_EFLAGS_TF;
  80.         }
  81.         si_code = get_si_code(tsk->thread.debugreg6);
  82.         if (tsk->thread.debugreg6 & (DR_STEP | DR_TRAP_BITS) || user_icebp)
  83.                 send_sigtrap(tsk, regs, error_code, si_code);
  84.         preempt_conditional_cli(regs);

  85.         return;
  86. }
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP