123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174 |
- // 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");
- }
|