mirror of
https://github.com/torvalds/linux.git
synced 2026-04-19 23:34:00 -04:00
Add a test case to check ptrace behavior in the case when vector extension is supported by the system, but vector context is not yet enabled for the traced process. Signed-off-by: Sergey Matyukevich <geomatsi@gmail.com> Reviewed-by: Andy Chiu <andybnac@gmail.com> Tested-by: Andy Chiu <andybnac@gmail.com> Link: https://patch.msgid.link/20251214163537.1054292-6-geomatsi@gmail.com [pjw@kernel.org: dropped duplicate sys/wait.h include] Signed-off-by: Paul Walmsley <pjw@kernel.org>
92 lines
2.0 KiB
C
92 lines
2.0 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
#include "../hwprobe/hwprobe.h"
|
|
#include <asm/vendor/thead.h>
|
|
#include <stdbool.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <sys/wait.h>
|
|
|
|
bool is_xtheadvector_supported(void)
|
|
{
|
|
struct riscv_hwprobe pair;
|
|
|
|
pair.key = RISCV_HWPROBE_KEY_VENDOR_EXT_THEAD_0;
|
|
riscv_hwprobe(&pair, 1, 0, NULL, 0);
|
|
return pair.value & RISCV_HWPROBE_VENDOR_EXT_XTHEADVECTOR;
|
|
}
|
|
|
|
bool is_vector_supported(void)
|
|
{
|
|
struct riscv_hwprobe pair;
|
|
|
|
pair.key = RISCV_HWPROBE_KEY_IMA_EXT_0;
|
|
riscv_hwprobe(&pair, 1, 0, NULL, 0);
|
|
return pair.value & RISCV_HWPROBE_EXT_ZVE32X;
|
|
}
|
|
|
|
unsigned long get_vr_len(void)
|
|
{
|
|
unsigned long vlenb;
|
|
|
|
if (is_vector_supported()) {
|
|
asm volatile("csrr %[vlenb], vlenb" : [vlenb] "=r"(vlenb));
|
|
return vlenb;
|
|
}
|
|
|
|
if (is_xtheadvector_supported()) {
|
|
asm volatile (
|
|
// 0 | zimm[10:0] | rs1 | 1 1 1 | rd | 1010111 | vsetvli
|
|
// vsetvli t4, x0, e8, m1, d1
|
|
".4byte 0b00000000000000000111111011010111\n\t"
|
|
"mv %[vlenb], t4\n\t"
|
|
: [vlenb] "=r"(vlenb) : : "memory", "t4");
|
|
return vlenb;
|
|
}
|
|
|
|
printf("WARNING: vector not supported\n");
|
|
return 0;
|
|
}
|
|
|
|
int launch_test(char *next_program, int test_inherit, int xtheadvector)
|
|
{
|
|
char *exec_argv[4], *exec_envp[1];
|
|
int rc, pid, status;
|
|
|
|
pid = fork();
|
|
if (pid < 0) {
|
|
printf("fork failed %d", pid);
|
|
return -1;
|
|
}
|
|
|
|
if (!pid) {
|
|
exec_argv[0] = next_program;
|
|
exec_argv[1] = test_inherit != 0 ? "x" : NULL;
|
|
exec_argv[2] = xtheadvector != 0 ? "x" : NULL;
|
|
exec_argv[3] = NULL;
|
|
exec_envp[0] = NULL;
|
|
/* launch the program again to check inherit */
|
|
rc = execve(next_program, exec_argv, exec_envp);
|
|
if (rc) {
|
|
perror("execve");
|
|
printf("child execve failed %d\n", rc);
|
|
exit(-1);
|
|
}
|
|
}
|
|
|
|
rc = waitpid(-1, &status, 0);
|
|
if (rc < 0) {
|
|
printf("waitpid failed\n");
|
|
return -3;
|
|
}
|
|
|
|
if ((WIFEXITED(status) && WEXITSTATUS(status) == -1) ||
|
|
WIFSIGNALED(status)) {
|
|
printf("child exited abnormally\n");
|
|
return -4;
|
|
}
|
|
|
|
return WEXITSTATUS(status);
|
|
}
|