ptrace32.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:8k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /* 
  2.  * 32bit ptrace for x86-64.
  3.  *
  4.  * Copyright 2001,2002 Andi Kleen, SuSE Labs.
  5.  * Some parts copied from arch/i386/kernel/ptrace.c. See that file for earlier 
  6.  * copyright.
  7.  * 
  8.  * This allows to access 64bit processes too; but there is no way to see the extended 
  9.  * register contents.
  10.  *
  11.  * $Id: ptrace32.c,v 1.13 2002/07/18 13:44:12 ak Exp $
  12.  */ 
  13. #include <linux/kernel.h>
  14. #include <linux/stddef.h>
  15. #include <linux/sched.h>
  16. #include <linux/mm.h>
  17. #include <asm/ptrace.h>
  18. #include <asm/uaccess.h>
  19. #include <asm/user32.h>
  20. #include <asm/user.h>
  21. #include <asm/errno.h>
  22. #include <asm/debugreg.h>
  23. #include <asm/i387.h>
  24. #include <asm/fpu32.h>
  25. #include <linux/mm.h>
  26. #define R32(l,q) 
  27. case offsetof(struct user32, regs.l): stack[offsetof(struct pt_regs, q)/8] = val; break
  28. static int putreg32(struct task_struct *child, unsigned regno, u32 val)
  29. {
  30. int i;
  31. __u64 *stack = (__u64 *)(child->thread.rsp0 - sizeof(struct pt_regs)); 
  32. switch (regno) {
  33. case offsetof(struct user32, regs.fs):
  34.         if (val && (val & 3) != 3) return -EIO;
  35.         child->thread.fs = val & 0xffff; 
  36. break;
  37. case offsetof(struct user32, regs.gs):
  38.     if (val && (val & 3) != 3) return -EIO;
  39. child->thread.gs = val & 0xffff;
  40. break;
  41. case offsetof(struct user32, regs.ds):
  42. if (val && (val & 3) != 3) return -EIO; 
  43. child->thread.ds = val & 0xffff;
  44. break;
  45. case offsetof(struct user32, regs.es):
  46. child->thread.es = val & 0xffff;
  47. break;
  48.     case offsetof(struct user32, regs.ss):
  49. if ((val & 3) != 3) return -EIO; 
  50.         stack[offsetof(struct pt_regs, ss)/8] = val & 0xffff;
  51.         break;
  52. case offsetof(struct user32, regs.cs):
  53. if ((val & 3) != 3) return -EIO;
  54. stack[offsetof(struct pt_regs, cs)/8] = val & 0xffff;
  55. break;
  56. R32(ebx, rbx); 
  57. R32(ecx, rcx);
  58. R32(edx, rdx);
  59. R32(edi, rdi);
  60. R32(esi, rsi);
  61. R32(ebp, rbp);
  62. R32(eax, rax);
  63. R32(orig_eax, orig_rax);
  64. R32(eip, rip);
  65. R32(esp, rsp);
  66. case offsetof(struct user32, regs.eflags): 
  67. stack[offsetof(struct pt_regs, eflags)/8] = val & 0x44dd5; 
  68. break;
  69. case offsetof(struct user32, u_debugreg[0]) ... offsetof(struct user32, u_debugreg[6]):
  70. child->thread.debugreg[(regno-offsetof(struct user32, u_debugreg[0]))/4] = val; 
  71. break; 
  72. case offsetof(struct user32, u_debugreg[7]):
  73. val &= ~DR_CONTROL_RESERVED;
  74. /* You are not expected to understand this ... I don't neither. */
  75. for(i=0; i<4; i++)
  76. if ((0x5454 >> ((val >> (16 + 4*i)) & 0xf)) & 1)
  77.        return -EIO;
  78. child->thread.debugreg[7] = val; 
  79. break; 
  80.     
  81. default:
  82. if (regno > sizeof(struct user32) || (regno & 3))
  83. return -EIO;
  84.        
  85. /* Other dummy fields in the virtual user structure are ignored */ 
  86. break; 
  87. }
  88. return 0;
  89. }
  90. #undef R32
  91. #define R32(l,q) 
  92. case offsetof(struct user32, regs.l): *val = stack[offsetof(struct pt_regs, q)/8]; break
  93. static int getreg32(struct task_struct *child, unsigned regno, u32 *val)
  94. {
  95. __u64 *stack = (__u64 *)(child->thread.rsp0 - sizeof(struct pt_regs)); 
  96. switch (regno) {
  97. case offsetof(struct user32, regs.fs):
  98.         *val = child->thread.fs; 
  99. break;
  100. case offsetof(struct user32, regs.gs):
  101. *val = child->thread.gs;
  102. break;
  103. case offsetof(struct user32, regs.ds):
  104. *val = child->thread.ds;
  105. break;
  106. case offsetof(struct user32, regs.es):
  107. *val = child->thread.es;
  108. break;
  109. R32(cs, cs);
  110. R32(ss, ss);
  111. R32(ebx, rbx); 
  112. R32(ecx, rcx);
  113. R32(edx, rdx);
  114. R32(edi, rdi);
  115. R32(esi, rsi);
  116. R32(ebp, rbp);
  117. R32(eax, rax);
  118. R32(orig_eax, orig_rax);
  119. R32(eip, rip);
  120. R32(eflags, eflags);
  121. R32(esp, rsp);
  122. case offsetof(struct user32, u_debugreg[0]) ... offsetof(struct user32, u_debugreg[7]):
  123. *val = child->thread.debugreg[(regno-offsetof(struct user32, u_debugreg[0]))/4]; 
  124. break; 
  125.     
  126. default:
  127. if (regno > sizeof(struct user32) || (regno & 3))
  128. return -EIO;
  129. /* Other dummy fields in the virtual user structure are ignored */ 
  130. *val = 0;
  131. break; 
  132. }
  133. return 0;
  134. }
  135. #undef R32
  136. static struct task_struct *find_target(int request, int pid, int *err)
  137. struct task_struct *child;
  138. *err = -EPERM; 
  139. if (pid == 1)
  140. return NULL; 
  141. *err = -ESRCH;
  142. read_lock(&tasklist_lock);
  143. child = find_task_by_pid(pid);
  144. if (child)
  145. get_task_struct(child);
  146. read_unlock(&tasklist_lock);
  147. if (child) { 
  148. *err = -ESRCH;
  149. if (!(child->ptrace & PT_PTRACED))
  150. goto out;
  151. if (child->state != TASK_STOPPED) {
  152. if (request != PTRACE_KILL)
  153. goto out;
  154. }
  155. if (child->p_pptr != current)
  156. goto out;
  157. return child; 
  158.  out:
  159. free_task_struct(child);
  160. return NULL; 
  161. extern asmlinkage long sys_ptrace(long request, long pid, unsigned long addr, unsigned long data);
  162. asmlinkage long sys32_ptrace(long request, u32 pid, u32 addr, u32 data)
  163. {
  164. struct task_struct *child;
  165. struct pt_regs *childregs; 
  166. int ret;
  167. __u32 val;
  168. switch (request) { 
  169. case PTRACE_TRACEME:
  170. case PTRACE_ATTACH:
  171. case PTRACE_SYSCALL:
  172. case PTRACE_CONT:
  173. case PTRACE_KILL:
  174. case PTRACE_SINGLESTEP:
  175. case PTRACE_DETACH:
  176. case PTRACE_SETOPTIONS:
  177. ret = sys_ptrace(request, pid, addr, data); 
  178. return ret;
  179. case PTRACE_PEEKTEXT:
  180. case PTRACE_PEEKDATA:
  181. case PTRACE_POKEDATA:
  182. case PTRACE_POKETEXT:
  183. case PTRACE_POKEUSR:       
  184. case PTRACE_PEEKUSR:
  185. case PTRACE_GETREGS:
  186. case PTRACE_SETREGS:
  187. case PTRACE_SETFPREGS:
  188. case PTRACE_GETFPREGS:
  189. case PTRACE_SETFPXREGS:
  190. case PTRACE_GETFPXREGS:
  191. break;
  192. default:
  193. return -EIO;
  194. child = find_target(request, pid, &ret);
  195. if (!child)
  196. return ret;
  197. childregs = (struct pt_regs *)(child->thread.rsp0 - sizeof(struct pt_regs)); 
  198. switch (request) {
  199. case PTRACE_PEEKDATA:
  200. case PTRACE_PEEKTEXT:
  201. ret = 0;
  202. if (access_process_vm(child, addr, &val, sizeof(u32), 0)!=sizeof(u32))
  203. ret = -EIO;
  204. else
  205. ret = put_user(val, (unsigned int *)(u64)data); 
  206. break; 
  207. case PTRACE_POKEDATA:
  208. case PTRACE_POKETEXT:
  209. ret = 0;
  210. if (access_process_vm(child, addr, &data, sizeof(u32), 1)!=sizeof(u32))
  211. ret = -EIO; 
  212. break;
  213. case PTRACE_PEEKUSR:
  214. ret = getreg32(child, addr, &val);
  215. if (ret == 0)
  216. ret = put_user(val, (__u32 *)(unsigned long) data);
  217. break;
  218. case PTRACE_POKEUSR:
  219. ret = putreg32(child, addr, data);
  220. break;
  221. case PTRACE_GETREGS: { /* Get all gp regs from the child. */
  222. int i;
  223.    if (!access_ok(VERIFY_WRITE, (unsigned *)(unsigned long)data, 16*4)) {
  224. ret = -EIO;
  225. break;
  226. }
  227. ret = 0;
  228. for ( i = 0; i <= 16*4 ; i += sizeof(__u32) ) {
  229. getreg32(child, i, &val);
  230. ret |= __put_user(val,(u32 *) (unsigned long) data);
  231. data += sizeof(u32);
  232. }
  233. break;
  234. }
  235. case PTRACE_SETREGS: { /* Set all gp regs in the child. */
  236. unsigned long tmp;
  237. int i;
  238.    if (!access_ok(VERIFY_READ, (unsigned *)(unsigned long)data, 16*4)) {
  239. ret = -EIO;
  240. break;
  241. }
  242. empty_fpu(child); 
  243. ret = 0; 
  244. for ( i = 0; i <= 16*4; i += sizeof(u32) ) {
  245. ret |= __get_user(tmp, (u32 *) (unsigned long) data);
  246. putreg32(child, i, tmp);
  247. data += sizeof(u32);
  248. }
  249. break;
  250. }
  251. case PTRACE_SETFPREGS:
  252. empty_fpu(child); 
  253. save_i387_ia32(child, (void *)(u64)data, childregs, 1);
  254. ret = 0; 
  255. break;
  256. case PTRACE_GETFPREGS:
  257. empty_fpu(child); 
  258. restore_i387_ia32(child, (void *)(u64)data, 1);
  259. ret = 0;
  260. break;
  261. case PTRACE_GETFPXREGS: { 
  262. struct user32_fxsr_struct *u = (void *)(u64)data; 
  263. empty_fpu(child); 
  264. ret = copy_to_user(u, &child->thread.i387.fxsave, sizeof(*u));
  265. ret |= __put_user(childregs->cs, &u->fcs);
  266. ret |= __put_user(child->thread.ds, &u->fos); 
  267. if (ret) 
  268. ret = -EFAULT;
  269. break; 
  270. case PTRACE_SETFPXREGS: { 
  271. struct user32_fxsr_struct *u = (void *)(u64)data; 
  272. empty_fpu(child); 
  273. /* no error checking to be bug to bug compatible with i386 */ 
  274. copy_from_user(&child->thread.i387.fxsave, u, sizeof(*u));
  275.         child->thread.i387.fxsave.mxcsr &= 0xffbf;
  276. ret = 0; 
  277. break; 
  278. default:
  279. ret = -EINVAL;
  280. break;
  281. }
  282. free_task_struct(child);
  283. return ret;
  284. }