root/tools/testing/selftests/powerpc/ptrace/ptrace-gpr.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");
}