ptrace_user.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. /*
  2. * Copyright (C) 2002 Jeff Dike (jdike@karaya.com)
  3. * Licensed under the GPL
  4. */
  5. #include <stdio.h>
  6. #include <errno.h>
  7. #include <unistd.h>
  8. #include <linux/stddef.h>
  9. #include "ptrace_user.h"
  10. /* Grr, asm/user.h includes asm/ptrace.h, so has to follow ptrace_user.h */
  11. #include <asm/user.h>
  12. #include "kern_util.h"
  13. #include "sysdep/thread.h"
  14. #include "user.h"
  15. #include "os.h"
  16. int ptrace_getregs(long pid, unsigned long *regs_out)
  17. {
  18. if (ptrace(PTRACE_GETREGS, pid, 0, regs_out) < 0)
  19. return -errno;
  20. return 0;
  21. }
  22. int ptrace_setregs(long pid, unsigned long *regs)
  23. {
  24. if (ptrace(PTRACE_SETREGS, pid, 0, regs) < 0)
  25. return -errno;
  26. return 0;
  27. }
  28. int ptrace_getfpregs(long pid, unsigned long *regs)
  29. {
  30. if (ptrace(PTRACE_GETFPREGS, pid, 0, regs) < 0)
  31. return -errno;
  32. return 0;
  33. }
  34. int ptrace_setfpregs(long pid, unsigned long *regs)
  35. {
  36. if (ptrace(PTRACE_SETFPREGS, pid, 0, regs) < 0)
  37. return -errno;
  38. return 0;
  39. }
  40. static void write_debugregs(int pid, unsigned long *regs)
  41. {
  42. struct user *dummy;
  43. int nregs, i;
  44. dummy = NULL;
  45. nregs = sizeof(dummy->u_debugreg)/sizeof(dummy->u_debugreg[0]);
  46. for(i = 0; i < nregs; i++){
  47. if((i == 4) || (i == 5)) continue;
  48. if(ptrace(PTRACE_POKEUSR, pid, &dummy->u_debugreg[i],
  49. regs[i]) < 0)
  50. printk("write_debugregs - ptrace failed on "
  51. "register %d, value = 0x%x, errno = %d\n", i,
  52. regs[i], errno);
  53. }
  54. }
  55. static void read_debugregs(int pid, unsigned long *regs)
  56. {
  57. struct user *dummy;
  58. int nregs, i;
  59. dummy = NULL;
  60. nregs = sizeof(dummy->u_debugreg)/sizeof(dummy->u_debugreg[0]);
  61. for(i = 0; i < nregs; i++){
  62. regs[i] = ptrace(PTRACE_PEEKUSR, pid,
  63. &dummy->u_debugreg[i], 0);
  64. }
  65. }
  66. /* Accessed only by the tracing thread */
  67. static unsigned long kernel_debugregs[8] = { [ 0 ... 7 ] = 0 };
  68. static int debugregs_seq = 0;
  69. void arch_enter_kernel(void *task, int pid)
  70. {
  71. read_debugregs(pid, TASK_DEBUGREGS(task));
  72. write_debugregs(pid, kernel_debugregs);
  73. }
  74. void arch_leave_kernel(void *task, int pid)
  75. {
  76. read_debugregs(pid, kernel_debugregs);
  77. write_debugregs(pid, TASK_DEBUGREGS(task));
  78. }
  79. void ptrace_pokeuser(unsigned long addr, unsigned long data)
  80. {
  81. if((addr < offsetof(struct user, u_debugreg[0])) ||
  82. (addr > offsetof(struct user, u_debugreg[7])))
  83. return;
  84. addr -= offsetof(struct user, u_debugreg[0]);
  85. addr = addr >> 2;
  86. if(kernel_debugregs[addr] == data) return;
  87. kernel_debugregs[addr] = data;
  88. debugregs_seq++;
  89. }
  90. static void update_debugregs_cb(void *arg)
  91. {
  92. int pid = *((int *) arg);
  93. write_debugregs(pid, kernel_debugregs);
  94. }
  95. void update_debugregs(int seq)
  96. {
  97. int me;
  98. if(seq == debugregs_seq) return;
  99. me = os_getpid();
  100. initial_thread_cb(update_debugregs_cb, &me);
  101. }
  102. /*
  103. * Overrides for Emacs so that we follow Linus's tabbing style.
  104. * Emacs will notice this stuff at the end of the file and automatically
  105. * adjust the settings for this buffer only. This must remain at the end
  106. * of the file.
  107. * ---------------------------------------------------------------------------
  108. * Local variables:
  109. * c-file-style: "linux"
  110. * End:
  111. */