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