xref: /freebsd/sys/compat/linux/linux_ptrace.c (revision f3065e767def62d9b593dd7528c0eb121a7e1439)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2017 Edward Tomasz Napierala <trasz@FreeBSD.org>
5  *
6  * This software was developed by SRI International and the University of
7  * Cambridge Computer Laboratory under DARPA/AFRL contract (FA8750-10-C-0237)
8  * ("CTSRD"), as part of the DARPA CRASH research programme.
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
23  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29  * SUCH DAMAGE.
30  */
31 
32 #include <sys/cdefs.h>
33 __FBSDID("$FreeBSD$");
34 
35 #include <sys/param.h>
36 #include <sys/lock.h>
37 #include <sys/proc.h>
38 #include <sys/ptrace.h>
39 #include <sys/sx.h>
40 #include <sys/syscallsubr.h>
41 
42 #include <machine/../linux/linux.h>
43 #include <machine/../linux/linux_proto.h>
44 #include <compat/linux/linux_emul.h>
45 #include <compat/linux/linux_errno.h>
46 #include <compat/linux/linux_misc.h>
47 #include <compat/linux/linux_signal.h>
48 #include <compat/linux/linux_util.h>
49 
50 #define	LINUX_PTRACE_TRACEME		0
51 #define	LINUX_PTRACE_PEEKTEXT		1
52 #define	LINUX_PTRACE_PEEKDATA		2
53 #define	LINUX_PTRACE_PEEKUSER		3
54 #define	LINUX_PTRACE_POKETEXT		4
55 #define	LINUX_PTRACE_POKEDATA		5
56 #define	LINUX_PTRACE_POKEUSER		6
57 #define	LINUX_PTRACE_CONT		7
58 #define	LINUX_PTRACE_KILL		8
59 #define	LINUX_PTRACE_SINGLESTEP		9
60 #define	LINUX_PTRACE_GETREGS		12
61 #define	LINUX_PTRACE_SETREGS		13
62 #define	LINUX_PTRACE_GETFPREGS		14
63 #define	LINUX_PTRACE_SETFPREGS		15
64 #define	LINUX_PTRACE_ATTACH		16
65 #define	LINUX_PTRACE_DETACH		17
66 #define	LINUX_PTRACE_SYSCALL		24
67 #define	LINUX_PTRACE_SETOPTIONS		0x4200
68 #define	LINUX_PTRACE_GETEVENTMSG	0x4201
69 #define	LINUX_PTRACE_GETSIGINFO		0x4202
70 #define	LINUX_PTRACE_GETREGSET		0x4204
71 #define	LINUX_PTRACE_SEIZE		0x4206
72 #define	LINUX_PTRACE_GET_SYSCALL_INFO	0x420e
73 
74 #define	LINUX_PTRACE_EVENT_EXEC		4
75 #define	LINUX_PTRACE_EVENT_EXIT		6
76 
77 #define	LINUX_PTRACE_O_TRACESYSGOOD	1
78 #define	LINUX_PTRACE_O_TRACEFORK	2
79 #define	LINUX_PTRACE_O_TRACEVFORK	4
80 #define	LINUX_PTRACE_O_TRACECLONE	8
81 #define	LINUX_PTRACE_O_TRACEEXEC	16
82 #define	LINUX_PTRACE_O_TRACEVFORKDONE	32
83 #define	LINUX_PTRACE_O_TRACEEXIT	64
84 #define	LINUX_PTRACE_O_TRACESECCOMP	128
85 #define	LINUX_PTRACE_O_EXITKILL		1048576
86 #define	LINUX_PTRACE_O_SUSPEND_SECCOMP	2097152
87 
88 #define	LINUX_NT_PRSTATUS		0x1
89 #define	LINUX_NT_PRFPREG		0x2
90 #define	LINUX_NT_X86_XSTATE		0x202
91 
92 #define	LINUX_PTRACE_O_MASK	(LINUX_PTRACE_O_TRACESYSGOOD |	\
93     LINUX_PTRACE_O_TRACEFORK | LINUX_PTRACE_O_TRACEVFORK |	\
94     LINUX_PTRACE_O_TRACECLONE | LINUX_PTRACE_O_TRACEEXEC |	\
95     LINUX_PTRACE_O_TRACEVFORKDONE | LINUX_PTRACE_O_TRACEEXIT |	\
96     LINUX_PTRACE_O_TRACESECCOMP | LINUX_PTRACE_O_EXITKILL |	\
97     LINUX_PTRACE_O_SUSPEND_SECCOMP)
98 
99 #define	LINUX_PTRACE_SYSCALL_INFO_NONE	0
100 #define	LINUX_PTRACE_SYSCALL_INFO_ENTRY	1
101 #define	LINUX_PTRACE_SYSCALL_INFO_EXIT	2
102 
103 #define LINUX_PTRACE_PEEKUSER_ORIG_RAX	120
104 #define LINUX_PTRACE_PEEKUSER_RIP	128
105 #define LINUX_PTRACE_PEEKUSER_CS	136
106 #define LINUX_PTRACE_PEEKUSER_DS	184
107 
108 static int
109 map_signum(int lsig, int *bsigp)
110 {
111 	int bsig;
112 
113 	if (lsig == 0) {
114 		*bsigp = 0;
115 		return (0);
116 	}
117 
118 	if (lsig < 0 || lsig > LINUX_SIGRTMAX)
119 		return (EINVAL);
120 
121 	bsig = linux_to_bsd_signal(lsig);
122 	if (bsig == SIGSTOP)
123 		bsig = 0;
124 
125 	*bsigp = bsig;
126 	return (0);
127 }
128 
129 int
130 linux_ptrace_status(struct thread *td, pid_t pid, int status)
131 {
132 	struct ptrace_lwpinfo lwpinfo;
133 	struct linux_pemuldata *pem;
134 	register_t saved_retval;
135 	int error;
136 
137 	saved_retval = td->td_retval[0];
138 	error = kern_ptrace(td, PT_LWPINFO, pid, &lwpinfo, sizeof(lwpinfo));
139 	td->td_retval[0] = saved_retval;
140 	if (error != 0) {
141 		linux_msg(td, "PT_LWPINFO failed with error %d", error);
142 		return (status);
143 	}
144 
145 	pem = pem_find(td->td_proc);
146 	KASSERT(pem != NULL, ("%s: proc emuldata not found.\n", __func__));
147 
148 	LINUX_PEM_SLOCK(pem);
149 	if ((pem->ptrace_flags & LINUX_PTRACE_O_TRACESYSGOOD) &&
150 	    lwpinfo.pl_flags & PL_FLAG_SCE)
151 		status |= (LINUX_SIGTRAP | 0x80) << 8;
152 	if ((pem->ptrace_flags & LINUX_PTRACE_O_TRACESYSGOOD) &&
153 	    lwpinfo.pl_flags & PL_FLAG_SCX) {
154 		if (lwpinfo.pl_flags & PL_FLAG_EXEC)
155 			status |= (LINUX_SIGTRAP | LINUX_PTRACE_EVENT_EXEC << 8) << 8;
156 		else
157 			status |= (LINUX_SIGTRAP | 0x80) << 8;
158 	}
159 	if ((pem->ptrace_flags & LINUX_PTRACE_O_TRACEEXIT) &&
160 	    lwpinfo.pl_flags & PL_FLAG_EXITED)
161 		status |= (LINUX_SIGTRAP | LINUX_PTRACE_EVENT_EXIT << 8) << 8;
162 	LINUX_PEM_SUNLOCK(pem);
163 
164 	return (status);
165 }
166 
167 static int
168 linux_ptrace_peek(struct thread *td, pid_t pid, void *addr, void *data)
169 {
170 	int error;
171 
172 	error = kern_ptrace(td, PT_READ_I, pid, addr, 0);
173 	if (error == 0)
174 		error = copyout(td->td_retval, data, sizeof(l_int));
175 	else if (error == ENOMEM)
176 		error = EIO;
177 	td->td_retval[0] = error;
178 
179 	return (error);
180 }
181 
182 static int
183 linux_ptrace_peekuser(struct thread *td, pid_t pid, void *addr, void *data)
184 {
185 	struct reg b_reg;
186 	uint64_t val;
187 	int error;
188 
189 	error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
190 	if (error != 0)
191 		return (error);
192 
193 	switch ((uintptr_t)addr) {
194 #ifdef __amd64__
195 	case LINUX_PTRACE_PEEKUSER_ORIG_RAX:
196 		val = b_reg.r_rax;
197 		break;
198 	case LINUX_PTRACE_PEEKUSER_RIP:
199 		val = b_reg.r_rip;
200 		break;
201 	case LINUX_PTRACE_PEEKUSER_CS:
202 		val = b_reg.r_cs;
203 		break;
204 	case LINUX_PTRACE_PEEKUSER_DS:
205 		val = b_reg.r_ds;
206 		break;
207 #endif /* __amd64__ */
208 	default:
209 		linux_msg(td, "PTRACE_PEEKUSER offset %ld not implemented; "
210 		    "returning EINVAL", (uintptr_t)addr);
211 		return (EINVAL);
212 	}
213 
214 	error = copyout(&val, data, sizeof(val));
215 	td->td_retval[0] = error;
216 
217 	return (error);
218 }
219 
220 static int
221 linux_ptrace_pokeuser(struct thread *td, pid_t pid, void *addr, void *data)
222 {
223 
224 	linux_msg(td, "PTRACE_POKEUSER not implemented; returning EINVAL");
225 	return (EINVAL);
226 }
227 
228 static int
229 linux_ptrace_setoptions(struct thread *td, pid_t pid, l_ulong data)
230 {
231 	struct linux_pemuldata *pem;
232 	int mask;
233 
234 	mask = 0;
235 
236 	if (data & ~LINUX_PTRACE_O_MASK) {
237 		linux_msg(td, "unknown ptrace option %lx set; "
238 		    "returning EINVAL",
239 		    data & ~LINUX_PTRACE_O_MASK);
240 		return (EINVAL);
241 	}
242 
243 	pem = pem_find(td->td_proc);
244 	KASSERT(pem != NULL, ("%s: proc emuldata not found.\n", __func__));
245 
246 	/*
247 	 * PTRACE_O_EXITKILL is ignored, we do that by default.
248 	 */
249 
250 	LINUX_PEM_XLOCK(pem);
251 	if (data & LINUX_PTRACE_O_TRACESYSGOOD) {
252 		pem->ptrace_flags |= LINUX_PTRACE_O_TRACESYSGOOD;
253 	} else {
254 		pem->ptrace_flags &= ~LINUX_PTRACE_O_TRACESYSGOOD;
255 	}
256 	LINUX_PEM_XUNLOCK(pem);
257 
258 	if (data & LINUX_PTRACE_O_TRACEFORK)
259 		mask |= PTRACE_FORK;
260 
261 	if (data & LINUX_PTRACE_O_TRACEVFORK)
262 		mask |= PTRACE_VFORK;
263 
264 	if (data & LINUX_PTRACE_O_TRACECLONE)
265 		mask |= PTRACE_VFORK;
266 
267 	if (data & LINUX_PTRACE_O_TRACEEXEC)
268 		mask |= PTRACE_EXEC;
269 
270 	if (data & LINUX_PTRACE_O_TRACEVFORKDONE)
271 		mask |= PTRACE_VFORK; /* XXX: Close enough? */
272 
273 	if (data & LINUX_PTRACE_O_TRACEEXIT) {
274 		pem->ptrace_flags |= LINUX_PTRACE_O_TRACEEXIT;
275 	} else {
276 		pem->ptrace_flags &= ~LINUX_PTRACE_O_TRACEEXIT;
277 	}
278 
279 	return (kern_ptrace(td, PT_SET_EVENT_MASK, pid, &mask, sizeof(mask)));
280 }
281 
282 static int
283 linux_ptrace_geteventmsg(struct thread *td, pid_t pid, l_ulong data)
284 {
285 
286 	linux_msg(td, "PTRACE_GETEVENTMSG not implemented; returning EINVAL");
287 	return (EINVAL);
288 }
289 
290 static int
291 linux_ptrace_getsiginfo(struct thread *td, pid_t pid, l_ulong data)
292 {
293 	struct ptrace_lwpinfo lwpinfo;
294 	l_siginfo_t l_siginfo;
295 	int error, sig;
296 
297 	error = kern_ptrace(td, PT_LWPINFO, pid, &lwpinfo, sizeof(lwpinfo));
298 	if (error != 0) {
299 		linux_msg(td, "PT_LWPINFO failed with error %d", error);
300 		return (error);
301 	}
302 
303 	if ((lwpinfo.pl_flags & PL_FLAG_SI) == 0) {
304 		error = EINVAL;
305 		linux_msg(td, "no PL_FLAG_SI, returning %d", error);
306 		return (error);
307 	}
308 
309 	sig = bsd_to_linux_signal(lwpinfo.pl_siginfo.si_signo);
310 	memset(&l_siginfo, 0, sizeof(l_siginfo));
311 	siginfo_to_lsiginfo(&lwpinfo.pl_siginfo, &l_siginfo, sig);
312 	error = copyout(&l_siginfo, (void *)data, sizeof(l_siginfo));
313 	return (error);
314 }
315 
316 static int
317 linux_ptrace_getregs(struct thread *td, pid_t pid, void *data)
318 {
319 	struct reg b_reg;
320 	struct linux_pt_regset l_regset;
321 	int error;
322 
323 	error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
324 	if (error != 0)
325 		return (error);
326 
327 	bsd_to_linux_regset(&b_reg, &l_regset);
328 	error = linux_ptrace_getregs_machdep(td, pid, &l_regset);
329 	if (error != 0)
330 		return (error);
331 
332 	error = copyout(&l_regset, (void *)data, sizeof(l_regset));
333 	return (error);
334 }
335 
336 static int
337 linux_ptrace_setregs(struct thread *td, pid_t pid, void *data)
338 {
339 	struct reg b_reg;
340 	struct linux_pt_regset l_regset;
341 	int error;
342 
343 	error = copyin(data, &l_regset, sizeof(l_regset));
344 	if (error != 0)
345 		return (error);
346 	linux_to_bsd_regset(&b_reg, &l_regset);
347 	error = kern_ptrace(td, PT_SETREGS, pid, &b_reg, 0);
348 	return (error);
349 }
350 
351 static int
352 linux_ptrace_getregset_prstatus(struct thread *td, pid_t pid, l_ulong data)
353 {
354 	struct reg b_reg;
355 	struct linux_pt_regset l_regset;
356 	struct iovec iov;
357 	size_t len;
358 	int error;
359 
360 	error = copyin((const void *)data, &iov, sizeof(iov));
361 	if (error != 0) {
362 		linux_msg(td, "copyin error %d", error);
363 		return (error);
364 	}
365 
366 	error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
367 	if (error != 0)
368 		return (error);
369 
370 	bsd_to_linux_regset(&b_reg, &l_regset);
371 	error = linux_ptrace_getregs_machdep(td, pid, &l_regset);
372 	if (error != 0)
373 		return (error);
374 
375 	len = MIN(iov.iov_len, sizeof(l_regset));
376 	error = copyout(&l_regset, (void *)iov.iov_base, len);
377 	if (error != 0) {
378 		linux_msg(td, "copyout error %d", error);
379 		return (error);
380 	}
381 
382 	iov.iov_len = len;
383 	error = copyout(&iov, (void *)data, sizeof(iov));
384 	if (error != 0) {
385 		linux_msg(td, "iov copyout error %d", error);
386 		return (error);
387 	}
388 
389 	return (error);
390 }
391 
392 static int
393 linux_ptrace_getregset(struct thread *td, pid_t pid, l_ulong addr, l_ulong data)
394 {
395 
396 	switch (addr) {
397 	case LINUX_NT_PRSTATUS:
398 		return (linux_ptrace_getregset_prstatus(td, pid, data));
399 	case LINUX_NT_PRFPREG:
400 		linux_msg(td, "PTRAGE_GETREGSET NT_PRFPREG not implemented; "
401 		    "returning EINVAL");
402 		return (EINVAL);
403 	case LINUX_NT_X86_XSTATE:
404 		linux_msg(td, "PTRAGE_GETREGSET NT_X86_XSTATE not implemented; "
405 		    "returning EINVAL");
406 		return (EINVAL);
407 	default:
408 		linux_msg(td, "PTRACE_GETREGSET request %#lx not implemented; "
409 		    "returning EINVAL", addr);
410 		return (EINVAL);
411 	}
412 }
413 
414 static int
415 linux_ptrace_seize(struct thread *td, pid_t pid, l_ulong addr, l_ulong data)
416 {
417 
418 	linux_msg(td, "PTRACE_SEIZE not implemented; returning EINVAL");
419 	return (EINVAL);
420 }
421 
422 static int
423 linux_ptrace_get_syscall_info(struct thread *td, pid_t pid,
424     l_ulong len, l_ulong data)
425 {
426 	struct ptrace_lwpinfo lwpinfo;
427 	struct ptrace_sc_ret sr;
428 	struct reg b_reg;
429 	struct syscall_info si;
430 	int error;
431 
432 	error = kern_ptrace(td, PT_LWPINFO, pid, &lwpinfo, sizeof(lwpinfo));
433 	if (error != 0) {
434 		linux_msg(td, "PT_LWPINFO failed with error %d", error);
435 		return (error);
436 	}
437 
438 	memset(&si, 0, sizeof(si));
439 
440 	if (lwpinfo.pl_flags & PL_FLAG_SCE) {
441 		si.op = LINUX_PTRACE_SYSCALL_INFO_ENTRY;
442 		si.entry.nr = lwpinfo.pl_syscall_code;
443 		/*
444 		 * The use of PT_GET_SC_ARGS there is special,
445 		 * implementation of PT_GET_SC_ARGS for Linux-ABI
446 		 * callers emulates Linux bug which strace(1) depends
447 		 * on: at initialization it tests whether ptrace works
448 		 * by calling close(2), or some other single-argument
449 		 * syscall, _with six arguments_, and then verifies
450 		 * whether it can fetch them all using this API;
451 		 * otherwise it bails out.
452 		 */
453 		error = kern_ptrace(td, PT_GET_SC_ARGS, pid,
454 		    &si.entry.args, sizeof(si.entry.args));
455 		if (error != 0) {
456 			linux_msg(td, "PT_GET_SC_ARGS failed with error %d",
457 			    error);
458 			return (error);
459 		}
460 	} else if (lwpinfo.pl_flags & PL_FLAG_SCX) {
461 		si.op = LINUX_PTRACE_SYSCALL_INFO_EXIT;
462 		error = kern_ptrace(td, PT_GET_SC_RET, pid, &sr, sizeof(sr));
463 
464 		if (error != 0) {
465 			linux_msg(td, "PT_GET_SC_RET failed with error %d",
466 			    error);
467 			return (error);
468 		}
469 
470 		if (sr.sr_error == 0) {
471 			si.exit.rval = sr.sr_retval[0];
472 			si.exit.is_error = 0;
473 		} else if (sr.sr_error == EJUSTRETURN) {
474 			/*
475 			 * EJUSTRETURN means the actual value to return
476 			 * has already been put into td_frame; instead
477 			 * of extracting it and trying to determine whether
478 			 * it's an error or not just bail out and let
479 			 * the ptracing process fall back to another method.
480 			 */
481 			si.op = LINUX_PTRACE_SYSCALL_INFO_NONE;
482 		} else if (sr.sr_error == ERESTART) {
483 			si.exit.rval = -LINUX_ERESTARTSYS;
484 			si.exit.is_error = 1;
485 		} else {
486 			si.exit.rval = bsd_to_linux_errno(sr.sr_error);
487 			si.exit.is_error = 1;
488 		}
489 	} else {
490 		si.op = LINUX_PTRACE_SYSCALL_INFO_NONE;
491 	}
492 
493 	error = kern_ptrace(td, PT_GETREGS, pid, &b_reg, 0);
494 	if (error != 0)
495 		return (error);
496 
497 	linux_ptrace_get_syscall_info_machdep(&b_reg, &si);
498 
499 	len = MIN(len, sizeof(si));
500 	error = copyout(&si, (void *)data, len);
501 	if (error == 0)
502 		td->td_retval[0] = sizeof(si);
503 
504 	return (error);
505 }
506 
507 int
508 linux_ptrace(struct thread *td, struct linux_ptrace_args *uap)
509 {
510 	void *addr;
511 	pid_t pid;
512 	int error, sig;
513 
514 	if (!allow_ptrace)
515 		return (ENOSYS);
516 
517 	pid  = (pid_t)uap->pid;
518 	addr = (void *)uap->addr;
519 
520 	switch (uap->req) {
521 	case LINUX_PTRACE_TRACEME:
522 		error = kern_ptrace(td, PT_TRACE_ME, 0, 0, 0);
523 		break;
524 	case LINUX_PTRACE_PEEKTEXT:
525 	case LINUX_PTRACE_PEEKDATA:
526 		error = linux_ptrace_peek(td, pid, addr, (void *)uap->data);
527 		if (error != 0)
528 			goto out;
529 		/*
530 		 * Linux expects this syscall to read 64 bits, not 32.
531 		 */
532 		error = linux_ptrace_peek(td, pid,
533 		    (void *)(uap->addr + 4), (void *)(uap->data + 4));
534 		break;
535 	case LINUX_PTRACE_PEEKUSER:
536 		error = linux_ptrace_peekuser(td, pid, addr, (void *)uap->data);
537 		break;
538 	case LINUX_PTRACE_POKETEXT:
539 	case LINUX_PTRACE_POKEDATA:
540 		error = kern_ptrace(td, PT_WRITE_D, pid, addr, uap->data);
541 		if (error != 0)
542 			goto out;
543 		/*
544 		 * Linux expects this syscall to write 64 bits, not 32.
545 		 */
546 		error = kern_ptrace(td, PT_WRITE_D, pid,
547 		    (void *)(uap->addr + 4), uap->data >> 32);
548 		break;
549 	case LINUX_PTRACE_POKEUSER:
550 		error = linux_ptrace_pokeuser(td, pid, addr, (void *)uap->data);
551 		break;
552 	case LINUX_PTRACE_CONT:
553 		error = map_signum(uap->data, &sig);
554 		if (error != 0)
555 			break;
556 		error = kern_ptrace(td, PT_CONTINUE, pid, (void *)1, sig);
557 		break;
558 	case LINUX_PTRACE_KILL:
559 		error = kern_ptrace(td, PT_KILL, pid, addr, uap->data);
560 		break;
561 	case LINUX_PTRACE_SINGLESTEP:
562 		error = map_signum(uap->data, &sig);
563 		if (error != 0)
564 			break;
565 		error = kern_ptrace(td, PT_STEP, pid, (void *)1, sig);
566 		break;
567 	case LINUX_PTRACE_GETREGS:
568 		error = linux_ptrace_getregs(td, pid, (void *)uap->data);
569 		break;
570 	case LINUX_PTRACE_SETREGS:
571 		error = linux_ptrace_setregs(td, pid, (void *)uap->data);
572 		break;
573 	case LINUX_PTRACE_ATTACH:
574 		error = kern_ptrace(td, PT_ATTACH, pid, addr, uap->data);
575 		break;
576 	case LINUX_PTRACE_DETACH:
577 		error = map_signum(uap->data, &sig);
578 		if (error != 0)
579 			break;
580 		error = kern_ptrace(td, PT_DETACH, pid, (void *)1, sig);
581 		break;
582 	case LINUX_PTRACE_SYSCALL:
583 		error = map_signum(uap->data, &sig);
584 		if (error != 0)
585 			break;
586 		error = kern_ptrace(td, PT_SYSCALL, pid, (void *)1, sig);
587 		break;
588 	case LINUX_PTRACE_SETOPTIONS:
589 		error = linux_ptrace_setoptions(td, pid, uap->data);
590 		break;
591 	case LINUX_PTRACE_GETEVENTMSG:
592 		error = linux_ptrace_geteventmsg(td, pid, uap->data);
593 		break;
594 	case LINUX_PTRACE_GETSIGINFO:
595 		error = linux_ptrace_getsiginfo(td, pid, uap->data);
596 		break;
597 	case LINUX_PTRACE_GETREGSET:
598 		error = linux_ptrace_getregset(td, pid, uap->addr, uap->data);
599 		break;
600 	case LINUX_PTRACE_SEIZE:
601 		error = linux_ptrace_seize(td, pid, uap->addr, uap->data);
602 		break;
603 	case LINUX_PTRACE_GET_SYSCALL_INFO:
604 		error = linux_ptrace_get_syscall_info(td, pid, uap->addr, uap->data);
605 		break;
606 	default:
607 		linux_msg(td, "ptrace(%ld, ...) not implemented; "
608 		    "returning EINVAL", uap->req);
609 		error = EINVAL;
610 		break;
611 	}
612 
613 out:
614 	if (error == EBUSY)
615 		error = ESRCH;
616 
617 	return (error);
618 }
619