/*
 * 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];
}