175 lines
3.6 KiB
C
175 lines
3.6 KiB
C
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||
|
/*
|
||
|
* Ptrace test for GPR/FPR registers
|
||
|
*
|
||
|
* Copyright (C) 2015 Anshuman Khandual, IBM Corporation.
|
||
|
*/
|
||
|
#include "ptrace.h"
|
||
|
#include "ptrace-gpr.h"
|
||
|
#include "reg.h"
|
||
|
#include <time.h>
|
||
|
|
||
|
/* Tracer and Tracee Shared Data */
|
||
|
int shm_id;
|
||
|
int *cptr, *pptr;
|
||
|
|
||
|
extern void gpr_child_loop(int *read_flag, int *write_flag,
|
||
|
unsigned long *gpr_buf, double *fpr_buf);
|
||
|
|
||
|
unsigned long child_gpr_val, parent_gpr_val;
|
||
|
double child_fpr_val, parent_fpr_val;
|
||
|
|
||
|
static int child(void)
|
||
|
{
|
||
|
unsigned long gpr_buf[32];
|
||
|
double fpr_buf[32];
|
||
|
int i;
|
||
|
|
||
|
cptr = (int *)shmat(shm_id, NULL, 0);
|
||
|
memset(gpr_buf, 0, sizeof(gpr_buf));
|
||
|
memset(fpr_buf, 0, sizeof(fpr_buf));
|
||
|
|
||
|
for (i = 0; i < 32; i++) {
|
||
|
gpr_buf[i] = child_gpr_val;
|
||
|
fpr_buf[i] = child_fpr_val;
|
||
|
}
|
||
|
|
||
|
gpr_child_loop(&cptr[0], &cptr[1], gpr_buf, fpr_buf);
|
||
|
|
||
|
shmdt((void *)cptr);
|
||
|
|
||
|
FAIL_IF(validate_gpr(gpr_buf, parent_gpr_val));
|
||
|
FAIL_IF(validate_fpr_double(fpr_buf, parent_fpr_val));
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int trace_gpr(pid_t child)
|
||
|
{
|
||
|
__u64 tmp, fpr[32], *peeked_fprs;
|
||
|
unsigned long gpr[18];
|
||
|
|
||
|
FAIL_IF(start_trace(child));
|
||
|
|
||
|
// Check child GPRs match what we expect using GETREGS
|
||
|
FAIL_IF(show_gpr(child, gpr));
|
||
|
FAIL_IF(validate_gpr(gpr, child_gpr_val));
|
||
|
|
||
|
// Check child FPRs match what we expect using GETFPREGS
|
||
|
FAIL_IF(show_fpr(child, fpr));
|
||
|
memcpy(&tmp, &child_fpr_val, sizeof(tmp));
|
||
|
FAIL_IF(validate_fpr(fpr, tmp));
|
||
|
|
||
|
// Check child FPRs match what we expect using PEEKUSR
|
||
|
peeked_fprs = peek_fprs(child);
|
||
|
FAIL_IF(!peeked_fprs);
|
||
|
FAIL_IF(validate_fpr(peeked_fprs, tmp));
|
||
|
free(peeked_fprs);
|
||
|
|
||
|
// Write child GPRs using SETREGS
|
||
|
FAIL_IF(write_gpr(child, parent_gpr_val));
|
||
|
|
||
|
// Write child FPRs using SETFPREGS
|
||
|
memcpy(&tmp, &parent_fpr_val, sizeof(tmp));
|
||
|
FAIL_IF(write_fpr(child, tmp));
|
||
|
|
||
|
// Check child FPRs match what we just set, using PEEKUSR
|
||
|
peeked_fprs = peek_fprs(child);
|
||
|
FAIL_IF(!peeked_fprs);
|
||
|
FAIL_IF(validate_fpr(peeked_fprs, tmp));
|
||
|
|
||
|
// Write child FPRs using POKEUSR
|
||
|
FAIL_IF(poke_fprs(child, (unsigned long *)peeked_fprs));
|
||
|
|
||
|
// Child will check its FPRs match before exiting
|
||
|
FAIL_IF(stop_trace(child));
|
||
|
|
||
|
return TEST_PASS;
|
||
|
}
|
||
|
|
||
|
#ifndef __LONG_WIDTH__
|
||
|
#define __LONG_WIDTH__ (sizeof(long) * 8)
|
||
|
#endif
|
||
|
|
||
|
static uint64_t rand_reg(void)
|
||
|
{
|
||
|
uint64_t result;
|
||
|
long r;
|
||
|
|
||
|
r = random();
|
||
|
|
||
|
// Small values are typical
|
||
|
result = r & 0xffff;
|
||
|
if (r & 0x10000)
|
||
|
return result;
|
||
|
|
||
|
// Pointers tend to have high bits set
|
||
|
result |= random() << (__LONG_WIDTH__ - 31);
|
||
|
if (r & 0x100000)
|
||
|
return result;
|
||
|
|
||
|
// And sometimes we want a full 64-bit value
|
||
|
result ^= random() << 16;
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
int ptrace_gpr(void)
|
||
|
{
|
||
|
unsigned long seed;
|
||
|
int ret, status;
|
||
|
pid_t pid;
|
||
|
|
||
|
seed = getpid() ^ time(NULL);
|
||
|
printf("srand(%lu)\n", seed);
|
||
|
srand(seed);
|
||
|
|
||
|
child_gpr_val = rand_reg();
|
||
|
child_fpr_val = rand_reg();
|
||
|
parent_gpr_val = rand_reg();
|
||
|
parent_fpr_val = rand_reg();
|
||
|
|
||
|
shm_id = shmget(IPC_PRIVATE, sizeof(int) * 2, 0777|IPC_CREAT);
|
||
|
pid = fork();
|
||
|
if (pid < 0) {
|
||
|
perror("fork() failed");
|
||
|
return TEST_FAIL;
|
||
|
}
|
||
|
if (pid == 0)
|
||
|
exit(child());
|
||
|
|
||
|
if (pid) {
|
||
|
pptr = (int *)shmat(shm_id, NULL, 0);
|
||
|
while (!pptr[1])
|
||
|
asm volatile("" : : : "memory");
|
||
|
|
||
|
ret = trace_gpr(pid);
|
||
|
if (ret) {
|
||
|
kill(pid, SIGTERM);
|
||
|
shmdt((void *)pptr);
|
||
|
shmctl(shm_id, IPC_RMID, NULL);
|
||
|
return TEST_FAIL;
|
||
|
}
|
||
|
|
||
|
pptr[0] = 1;
|
||
|
shmdt((void *)pptr);
|
||
|
|
||
|
ret = wait(&status);
|
||
|
shmctl(shm_id, IPC_RMID, NULL);
|
||
|
if (ret != pid) {
|
||
|
printf("Child's exit status not captured\n");
|
||
|
return TEST_FAIL;
|
||
|
}
|
||
|
|
||
|
return (WIFEXITED(status) && WEXITSTATUS(status)) ? TEST_FAIL :
|
||
|
TEST_PASS;
|
||
|
}
|
||
|
|
||
|
return TEST_PASS;
|
||
|
}
|
||
|
|
||
|
int main(int argc, char *argv[])
|
||
|
{
|
||
|
return test_harness(ptrace_gpr, "ptrace_gpr");
|
||
|
}
|