linux-user/riscv: Add syscall riscv_hwprobe
This patch adds the new syscall for the "RISC-V Hardware Probing Interface" (https://docs.kernel.org/riscv/hwprobe.html). Reviewed-by: Palmer Dabbelt <palmer@rivosinc.com> Signed-off-by: Robbin Ehn <rehn@rivosinc.com> Message-Id: <06a4543df2aa6101ca9a48f21a3198064b4f1f87.camel@rivosinc.com> Signed-off-by: Alistair Francis <alistair.francis@wdc.com>
This commit is contained in:
parent
c0716c81b2
commit
9e1c7d982d
@ -228,6 +228,7 @@
|
|||||||
#define TARGET_NR_accept4 242
|
#define TARGET_NR_accept4 242
|
||||||
#define TARGET_NR_arch_specific_syscall 244
|
#define TARGET_NR_arch_specific_syscall 244
|
||||||
#define TARGET_NR_riscv_flush_icache (TARGET_NR_arch_specific_syscall + 15)
|
#define TARGET_NR_riscv_flush_icache (TARGET_NR_arch_specific_syscall + 15)
|
||||||
|
#define TARGET_NR_riscv_hwprobe (TARGET_NR_arch_specific_syscall + 14)
|
||||||
#define TARGET_NR_prlimit64 261
|
#define TARGET_NR_prlimit64 261
|
||||||
#define TARGET_NR_fanotify_init 262
|
#define TARGET_NR_fanotify_init 262
|
||||||
#define TARGET_NR_fanotify_mark 263
|
#define TARGET_NR_fanotify_mark 263
|
||||||
|
@ -251,6 +251,7 @@
|
|||||||
#define TARGET_NR_recvmmsg 243
|
#define TARGET_NR_recvmmsg 243
|
||||||
#define TARGET_NR_arch_specific_syscall 244
|
#define TARGET_NR_arch_specific_syscall 244
|
||||||
#define TARGET_NR_riscv_flush_icache (TARGET_NR_arch_specific_syscall + 15)
|
#define TARGET_NR_riscv_flush_icache (TARGET_NR_arch_specific_syscall + 15)
|
||||||
|
#define TARGET_NR_riscv_hwprobe (TARGET_NR_arch_specific_syscall + 14)
|
||||||
#define TARGET_NR_wait4 260
|
#define TARGET_NR_wait4 260
|
||||||
#define TARGET_NR_prlimit64 261
|
#define TARGET_NR_prlimit64 261
|
||||||
#define TARGET_NR_fanotify_init 262
|
#define TARGET_NR_fanotify_init 262
|
||||||
|
@ -8983,6 +8983,147 @@ static int do_getdents64(abi_long dirfd, abi_long arg2, abi_long count)
|
|||||||
}
|
}
|
||||||
#endif /* TARGET_NR_getdents64 */
|
#endif /* TARGET_NR_getdents64 */
|
||||||
|
|
||||||
|
#if defined(TARGET_NR_riscv_hwprobe)
|
||||||
|
|
||||||
|
#define RISCV_HWPROBE_KEY_MVENDORID 0
|
||||||
|
#define RISCV_HWPROBE_KEY_MARCHID 1
|
||||||
|
#define RISCV_HWPROBE_KEY_MIMPID 2
|
||||||
|
|
||||||
|
#define RISCV_HWPROBE_KEY_BASE_BEHAVIOR 3
|
||||||
|
#define RISCV_HWPROBE_BASE_BEHAVIOR_IMA (1 << 0)
|
||||||
|
|
||||||
|
#define RISCV_HWPROBE_KEY_IMA_EXT_0 4
|
||||||
|
#define RISCV_HWPROBE_IMA_FD (1 << 0)
|
||||||
|
#define RISCV_HWPROBE_IMA_C (1 << 1)
|
||||||
|
|
||||||
|
#define RISCV_HWPROBE_KEY_CPUPERF_0 5
|
||||||
|
#define RISCV_HWPROBE_MISALIGNED_UNKNOWN (0 << 0)
|
||||||
|
#define RISCV_HWPROBE_MISALIGNED_EMULATED (1 << 0)
|
||||||
|
#define RISCV_HWPROBE_MISALIGNED_SLOW (2 << 0)
|
||||||
|
#define RISCV_HWPROBE_MISALIGNED_FAST (3 << 0)
|
||||||
|
#define RISCV_HWPROBE_MISALIGNED_UNSUPPORTED (4 << 0)
|
||||||
|
#define RISCV_HWPROBE_MISALIGNED_MASK (7 << 0)
|
||||||
|
|
||||||
|
struct riscv_hwprobe {
|
||||||
|
abi_llong key;
|
||||||
|
abi_ullong value;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void risc_hwprobe_fill_pairs(CPURISCVState *env,
|
||||||
|
struct riscv_hwprobe *pair,
|
||||||
|
size_t pair_count)
|
||||||
|
{
|
||||||
|
const RISCVCPUConfig *cfg = riscv_cpu_cfg(env);
|
||||||
|
|
||||||
|
for (; pair_count > 0; pair_count--, pair++) {
|
||||||
|
abi_llong key;
|
||||||
|
abi_ullong value;
|
||||||
|
__put_user(0, &pair->value);
|
||||||
|
__get_user(key, &pair->key);
|
||||||
|
switch (key) {
|
||||||
|
case RISCV_HWPROBE_KEY_MVENDORID:
|
||||||
|
__put_user(cfg->mvendorid, &pair->value);
|
||||||
|
break;
|
||||||
|
case RISCV_HWPROBE_KEY_MARCHID:
|
||||||
|
__put_user(cfg->marchid, &pair->value);
|
||||||
|
break;
|
||||||
|
case RISCV_HWPROBE_KEY_MIMPID:
|
||||||
|
__put_user(cfg->mimpid, &pair->value);
|
||||||
|
break;
|
||||||
|
case RISCV_HWPROBE_KEY_BASE_BEHAVIOR:
|
||||||
|
value = riscv_has_ext(env, RVI) &&
|
||||||
|
riscv_has_ext(env, RVM) &&
|
||||||
|
riscv_has_ext(env, RVA) ?
|
||||||
|
RISCV_HWPROBE_BASE_BEHAVIOR_IMA : 0;
|
||||||
|
__put_user(value, &pair->value);
|
||||||
|
break;
|
||||||
|
case RISCV_HWPROBE_KEY_IMA_EXT_0:
|
||||||
|
value = riscv_has_ext(env, RVF) &&
|
||||||
|
riscv_has_ext(env, RVD) ?
|
||||||
|
RISCV_HWPROBE_IMA_FD : 0;
|
||||||
|
value |= riscv_has_ext(env, RVC) ?
|
||||||
|
RISCV_HWPROBE_IMA_C : pair->value;
|
||||||
|
__put_user(value, &pair->value);
|
||||||
|
break;
|
||||||
|
case RISCV_HWPROBE_KEY_CPUPERF_0:
|
||||||
|
__put_user(RISCV_HWPROBE_MISALIGNED_FAST, &pair->value);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
__put_user(-1, &pair->key);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cpu_set_valid(abi_long arg3, abi_long arg4)
|
||||||
|
{
|
||||||
|
int ret, i, tmp;
|
||||||
|
size_t host_mask_size, target_mask_size;
|
||||||
|
unsigned long *host_mask;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* cpu_set_t represent CPU masks as bit masks of type unsigned long *.
|
||||||
|
* arg3 contains the cpu count.
|
||||||
|
*/
|
||||||
|
tmp = (8 * sizeof(abi_ulong));
|
||||||
|
target_mask_size = ((arg3 + tmp - 1) / tmp) * sizeof(abi_ulong);
|
||||||
|
host_mask_size = (target_mask_size + (sizeof(*host_mask) - 1)) &
|
||||||
|
~(sizeof(*host_mask) - 1);
|
||||||
|
|
||||||
|
host_mask = alloca(host_mask_size);
|
||||||
|
|
||||||
|
ret = target_to_host_cpu_mask(host_mask, host_mask_size,
|
||||||
|
arg4, target_mask_size);
|
||||||
|
if (ret != 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0 ; i < host_mask_size / sizeof(*host_mask); i++) {
|
||||||
|
if (host_mask[i] != 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static abi_long do_riscv_hwprobe(CPUArchState *cpu_env, abi_long arg1,
|
||||||
|
abi_long arg2, abi_long arg3,
|
||||||
|
abi_long arg4, abi_long arg5)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
struct riscv_hwprobe *host_pairs;
|
||||||
|
|
||||||
|
/* flags must be 0 */
|
||||||
|
if (arg5 != 0) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check cpu_set */
|
||||||
|
if (arg3 != 0) {
|
||||||
|
ret = cpu_set_valid(arg3, arg4);
|
||||||
|
if (ret != 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
} else if (arg4 != 0) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* no pairs */
|
||||||
|
if (arg2 == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
host_pairs = lock_user(VERIFY_WRITE, arg1,
|
||||||
|
sizeof(*host_pairs) * (size_t)arg2, 0);
|
||||||
|
if (host_pairs == NULL) {
|
||||||
|
return -TARGET_EFAULT;
|
||||||
|
}
|
||||||
|
risc_hwprobe_fill_pairs(cpu_env, host_pairs, arg2);
|
||||||
|
unlock_user(host_pairs, arg1, sizeof(*host_pairs) * (size_t)arg2);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif /* TARGET_NR_riscv_hwprobe */
|
||||||
|
|
||||||
#if defined(TARGET_NR_pivot_root) && defined(__NR_pivot_root)
|
#if defined(TARGET_NR_pivot_root) && defined(__NR_pivot_root)
|
||||||
_syscall2(int, pivot_root, const char *, new_root, const char *, put_old)
|
_syscall2(int, pivot_root, const char *, new_root, const char *, put_old)
|
||||||
#endif
|
#endif
|
||||||
@ -13665,6 +13806,11 @@ static abi_long do_syscall1(CPUArchState *cpu_env, int num, abi_long arg1,
|
|||||||
return ret;
|
return ret;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if defined(TARGET_NR_riscv_hwprobe)
|
||||||
|
case TARGET_NR_riscv_hwprobe:
|
||||||
|
return do_riscv_hwprobe(cpu_env, arg1, arg2, arg3, arg4, arg5);
|
||||||
|
#endif
|
||||||
|
|
||||||
default:
|
default:
|
||||||
qemu_log_mask(LOG_UNIMP, "Unsupported syscall: %d\n", num);
|
qemu_log_mask(LOG_UNIMP, "Unsupported syscall: %d\n", num);
|
||||||
return -TARGET_ENOSYS;
|
return -TARGET_ENOSYS;
|
||||||
|
Loading…
Reference in New Issue
Block a user