/*
* Spawn a child and set it up for ptrace()-ing
*
* Copyright (c) 2008 Analog Devices Inc.
*
* Licensed under the GPL-2 or later
*/
/*
* To use:
* - add this line after your normal includes:
* #include "spawn_ptrace_child.c"
* - add this line to the top of your main():
* make_a_baby(argc, argv);
* - access the child pid via the "pid" variable
*/
#include <errno.h> /* errno */
#include <signal.h> /* signal() */
#include <stdbool.h> /* true */
#include <string.h> /* strcmp() */
#include <unistd.h> /* execlp() sleep() vfork() */
#include <sys/ptrace.h> /* ptrace() */
#include <sys/wait.h>
#include "test.h"
static pid_t pid;
#ifdef __sparc__
/* sparce swaps addr/data for get/set regs */
# define maybe_swap(request, addr, data) \
do { \
if (request == PTRACE_GETREGS || request == PTRACE_SETREGS) { \
void *__s = addr; \
addr = data; \
data = __s; \
} \
} while (0)
#else
# define maybe_swap(...)
#endif
#define vptrace(request, pid, addr, data) \
({ \
errno = 0; \
long __ret; \
void *__addr = (void *)(addr); \
void *__data = (void *)(data); \
maybe_swap(request, __addr, __data); \
__ret = ptrace(request, pid, __addr, __data); \
if (__ret && errno) \
perror("ptrace(" #request ", " #pid ", " #addr ", " #data ")"); \
__ret; \
})
static void make_a_baby(int argc, char *argv[])
{
if (argc > 1 && !strcmp(argv[1], "child")) {
/* if we're the child, just sit around doing nothing */
int i = 60;
while (i--) {
close(-100);
sleep(1);
}
exit(1);
}
signal(SIGCHLD, SIG_IGN);
pid = vfork();
if (pid == -1) {
tst_resm(TFAIL, "vfork() failed");
tst_exit();
} else if (pid) {
int status;
if (wait(&status) != pid) {
tst_brkm(TBROK | TERRNO, NULL, "wait(%i) failed: %#x", pid, status);
kill(pid, SIGKILL);
exit(1);
}
if (!WIFSTOPPED(status)) {
tst_brkm(TBROK, NULL, "child status not stopped: %#x", status);
kill(pid, SIGKILL);
exit(1);
}
return;
}
errno = 0;
long ret = ptrace(PTRACE_TRACEME, 0, NULL, NULL);
if (ret && errno) {
tst_resm(TFAIL, "PTRACE_TRACEME failed");
tst_exit();
}
execlp(argv[0], argv[0], "child", NULL);
tst_resm(TFAIL, "execlp() failed");
tst_exit();
}
#define SPT(x) [PTRACE_##x] = #x,
static char *strings[] = {
SPT(TRACEME)
SPT(PEEKTEXT)
SPT(PEEKDATA)
SPT(PEEKUSER)
SPT(POKETEXT)
SPT(POKEDATA)
SPT(POKEUSER)
#ifdef PTRACE_GETREGS
SPT(GETREGS)
#endif
#ifdef PTRACE_SETREGS
SPT(SETREGS)
#endif
#ifdef PTRACE_GETSIGINFO
SPT(GETSIGINFO)
#endif
#ifdef PTRACE_SETSIGINFO
SPT(SETSIGINFO)
#endif
#ifdef PTRACE_GETFGREGS
SPT(GETFGREGS)
#endif
#ifdef PTRACE_SETFGREGS
SPT(SETFGREGS)
#endif
SPT(KILL)
SPT(SINGLESTEP)
};
static inline char *strptrace(enum __ptrace_request request)
{
return strings[request];
}