debugreg.h 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. #ifndef _ASM_X86_DEBUGREG_H
  2. #define _ASM_X86_DEBUGREG_H
  3. #include <linux/bug.h>
  4. #include <uapi/asm/debugreg.h>
  5. DECLARE_PER_CPU(unsigned long, cpu_dr7);
  6. #ifndef CONFIG_PARAVIRT
  7. /*
  8. * These special macros can be used to get or set a debugging register
  9. */
  10. #define get_debugreg(var, register) \
  11. (var) = native_get_debugreg(register)
  12. #define set_debugreg(value, register) \
  13. native_set_debugreg(register, value)
  14. #endif
  15. static inline unsigned long native_get_debugreg(int regno)
  16. {
  17. unsigned long val = 0; /* Damn you, gcc! */
  18. switch (regno) {
  19. case 0:
  20. asm("mov %%db0, %0" :"=r" (val));
  21. break;
  22. case 1:
  23. asm("mov %%db1, %0" :"=r" (val));
  24. break;
  25. case 2:
  26. asm("mov %%db2, %0" :"=r" (val));
  27. break;
  28. case 3:
  29. asm("mov %%db3, %0" :"=r" (val));
  30. break;
  31. case 6:
  32. asm("mov %%db6, %0" :"=r" (val));
  33. break;
  34. case 7:
  35. asm("mov %%db7, %0" :"=r" (val));
  36. break;
  37. default:
  38. BUG();
  39. }
  40. return val;
  41. }
  42. static inline void native_set_debugreg(int regno, unsigned long value)
  43. {
  44. switch (regno) {
  45. case 0:
  46. asm("mov %0, %%db0" ::"r" (value));
  47. break;
  48. case 1:
  49. asm("mov %0, %%db1" ::"r" (value));
  50. break;
  51. case 2:
  52. asm("mov %0, %%db2" ::"r" (value));
  53. break;
  54. case 3:
  55. asm("mov %0, %%db3" ::"r" (value));
  56. break;
  57. case 6:
  58. asm("mov %0, %%db6" ::"r" (value));
  59. break;
  60. case 7:
  61. asm("mov %0, %%db7" ::"r" (value));
  62. break;
  63. default:
  64. BUG();
  65. }
  66. }
  67. static inline void hw_breakpoint_disable(void)
  68. {
  69. /* Zero the control register for HW Breakpoint */
  70. set_debugreg(0UL, 7);
  71. /* Zero-out the individual HW breakpoint address registers */
  72. set_debugreg(0UL, 0);
  73. set_debugreg(0UL, 1);
  74. set_debugreg(0UL, 2);
  75. set_debugreg(0UL, 3);
  76. }
  77. static inline int hw_breakpoint_active(void)
  78. {
  79. return __this_cpu_read(cpu_dr7) & DR_GLOBAL_ENABLE_MASK;
  80. }
  81. extern void aout_dump_debugregs(struct user *dump);
  82. extern void hw_breakpoint_restore(void);
  83. #ifdef CONFIG_X86_64
  84. DECLARE_PER_CPU(int, debug_stack_usage);
  85. static inline void debug_stack_usage_inc(void)
  86. {
  87. __this_cpu_inc(debug_stack_usage);
  88. }
  89. static inline void debug_stack_usage_dec(void)
  90. {
  91. __this_cpu_dec(debug_stack_usage);
  92. }
  93. int is_debug_stack(unsigned long addr);
  94. void debug_stack_set_zero(void);
  95. void debug_stack_reset(void);
  96. #else /* !X86_64 */
  97. static inline int is_debug_stack(unsigned long addr) { return 0; }
  98. static inline void debug_stack_set_zero(void) { }
  99. static inline void debug_stack_reset(void) { }
  100. static inline void debug_stack_usage_inc(void) { }
  101. static inline void debug_stack_usage_dec(void) { }
  102. #endif /* X86_64 */
  103. #ifdef CONFIG_CPU_SUP_AMD
  104. extern void set_dr_addr_mask(unsigned long mask, int dr);
  105. #else
  106. static inline void set_dr_addr_mask(unsigned long mask, int dr) { }
  107. #endif
  108. #endif /* _ASM_X86_DEBUGREG_H */