/* -*- mode: C; c-basic-offset: 3; -*- */
#include <assert.h>
#include <elf.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/ptrace.h>
#include <sys/types.h>
#include <sys/uio.h>
#include <sys/user.h>
#include <sys/wait.h>
static int
err_out(const char *msg)
{
perror(msg);
return 1;
}
static int
non_empty(const char *buf, size_t len)
{
size_t i;
int c = 0;
volatile const char *p = buf;
for (i = 0; i != len; i++)
c |= p[i];
return c;
}
static int
do_child(void)
{
if (ptrace(PTRACE_TRACEME, 0, NULL, NULL) == -1)
return err_out("ptrace traceme");
raise(SIGUSR1);
return 0;
}
int
main(void)
{
char buf[1024];
struct iovec iov;
pid_t cpid, pid;
int status;
cpid = fork();
if (cpid == -1)
return err_out("fork");
if (cpid == 0)
return do_child();
pid = wait(&status);
if (pid == -1)
return err_out("wait");
/* Intentionally provide an uninitialized buffer to ptrace. */
iov.iov_len = sizeof(buf);
iov.iov_base = buf;
if (ptrace(0x4204, cpid, NT_PRSTATUS, &iov) == -1)
return err_out("ptrace getregset");
assert(iov.iov_base == buf);
assert(iov.iov_len > 0 && iov.iov_len < sizeof(buf));
/* We're assuming here that NT_PRSTATUS never contains
all-zeros. */
assert(non_empty(buf, iov.iov_len));
puts("OK");
return 0;
}