linux-user: Split out do_prctl and subroutines
Since the prctl constants are supposed to be generic, supply any that are not provided by the host. Split out subroutines for PR_GET_FP_MODE, PR_SET_FP_MODE, PR_GET_VL, PR_SET_VL, PR_RESET_KEYS, PR_SET_TAGGED_ADDR_CTRL, PR_GET_TAGGED_ADDR_CTRL. Return EINVAL for guests that do not support these options rather than pass them on to the host. Reviewed-by: Laurent Vivier <laurent@vivier.eu> Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Message-Id: <20211227150127.2659293-2-richard.henderson@linaro.org> Signed-off-by: Laurent Vivier <laurent@vivier.eu>
This commit is contained in:
parent
c1e8e3a746
commit
87e9bf2323
160
linux-user/aarch64/target_prctl.h
Normal file
160
linux-user/aarch64/target_prctl.h
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
/*
|
||||||
|
* AArch64 specific prctl functions for linux-user
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
*/
|
||||||
|
#ifndef AARCH64_TARGET_PRCTL_H
|
||||||
|
#define AARCH64_TARGET_PRCTL_H
|
||||||
|
|
||||||
|
static abi_long do_prctl_get_vl(CPUArchState *env)
|
||||||
|
{
|
||||||
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
if (cpu_isar_feature(aa64_sve, cpu)) {
|
||||||
|
return ((cpu->env.vfp.zcr_el[1] & 0xf) + 1) * 16;
|
||||||
|
}
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
#define do_prctl_get_vl do_prctl_get_vl
|
||||||
|
|
||||||
|
static abi_long do_prctl_set_vl(CPUArchState *env, abi_long arg2)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* We cannot support either PR_SVE_SET_VL_ONEXEC or PR_SVE_VL_INHERIT.
|
||||||
|
* Note the kernel definition of sve_vl_valid allows for VQ=512,
|
||||||
|
* i.e. VL=8192, even though the current architectural maximum is VQ=16.
|
||||||
|
*/
|
||||||
|
if (cpu_isar_feature(aa64_sve, env_archcpu(env))
|
||||||
|
&& arg2 >= 0 && arg2 <= 512 * 16 && !(arg2 & 15)) {
|
||||||
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
uint32_t vq, old_vq;
|
||||||
|
|
||||||
|
old_vq = (env->vfp.zcr_el[1] & 0xf) + 1;
|
||||||
|
vq = MAX(arg2 / 16, 1);
|
||||||
|
vq = MIN(vq, cpu->sve_max_vq);
|
||||||
|
|
||||||
|
if (vq < old_vq) {
|
||||||
|
aarch64_sve_narrow_vq(env, vq);
|
||||||
|
}
|
||||||
|
env->vfp.zcr_el[1] = vq - 1;
|
||||||
|
arm_rebuild_hflags(env);
|
||||||
|
return vq * 16;
|
||||||
|
}
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
#define do_prctl_set_vl do_prctl_set_vl
|
||||||
|
|
||||||
|
static abi_long do_prctl_reset_keys(CPUArchState *env, abi_long arg2)
|
||||||
|
{
|
||||||
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
|
||||||
|
if (cpu_isar_feature(aa64_pauth, cpu)) {
|
||||||
|
int all = (PR_PAC_APIAKEY | PR_PAC_APIBKEY |
|
||||||
|
PR_PAC_APDAKEY | PR_PAC_APDBKEY | PR_PAC_APGAKEY);
|
||||||
|
int ret = 0;
|
||||||
|
Error *err = NULL;
|
||||||
|
|
||||||
|
if (arg2 == 0) {
|
||||||
|
arg2 = all;
|
||||||
|
} else if (arg2 & ~all) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
if (arg2 & PR_PAC_APIAKEY) {
|
||||||
|
ret |= qemu_guest_getrandom(&env->keys.apia,
|
||||||
|
sizeof(ARMPACKey), &err);
|
||||||
|
}
|
||||||
|
if (arg2 & PR_PAC_APIBKEY) {
|
||||||
|
ret |= qemu_guest_getrandom(&env->keys.apib,
|
||||||
|
sizeof(ARMPACKey), &err);
|
||||||
|
}
|
||||||
|
if (arg2 & PR_PAC_APDAKEY) {
|
||||||
|
ret |= qemu_guest_getrandom(&env->keys.apda,
|
||||||
|
sizeof(ARMPACKey), &err);
|
||||||
|
}
|
||||||
|
if (arg2 & PR_PAC_APDBKEY) {
|
||||||
|
ret |= qemu_guest_getrandom(&env->keys.apdb,
|
||||||
|
sizeof(ARMPACKey), &err);
|
||||||
|
}
|
||||||
|
if (arg2 & PR_PAC_APGAKEY) {
|
||||||
|
ret |= qemu_guest_getrandom(&env->keys.apga,
|
||||||
|
sizeof(ARMPACKey), &err);
|
||||||
|
}
|
||||||
|
if (ret != 0) {
|
||||||
|
/*
|
||||||
|
* Some unknown failure in the crypto. The best
|
||||||
|
* we can do is log it and fail the syscall.
|
||||||
|
* The real syscall cannot fail this way.
|
||||||
|
*/
|
||||||
|
qemu_log_mask(LOG_UNIMP, "PR_PAC_RESET_KEYS: Crypto failure: %s",
|
||||||
|
error_get_pretty(err));
|
||||||
|
error_free(err);
|
||||||
|
return -TARGET_EIO;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
#define do_prctl_reset_keys do_prctl_reset_keys
|
||||||
|
|
||||||
|
static abi_long do_prctl_set_tagged_addr_ctrl(CPUArchState *env, abi_long arg2)
|
||||||
|
{
|
||||||
|
abi_ulong valid_mask = PR_TAGGED_ADDR_ENABLE;
|
||||||
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
|
||||||
|
if (cpu_isar_feature(aa64_mte, cpu)) {
|
||||||
|
valid_mask |= PR_MTE_TCF_MASK;
|
||||||
|
valid_mask |= PR_MTE_TAG_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (arg2 & ~valid_mask) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
env->tagged_addr_enable = arg2 & PR_TAGGED_ADDR_ENABLE;
|
||||||
|
|
||||||
|
if (cpu_isar_feature(aa64_mte, cpu)) {
|
||||||
|
switch (arg2 & PR_MTE_TCF_MASK) {
|
||||||
|
case PR_MTE_TCF_NONE:
|
||||||
|
case PR_MTE_TCF_SYNC:
|
||||||
|
case PR_MTE_TCF_ASYNC:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Write PR_MTE_TCF to SCTLR_EL1[TCF0].
|
||||||
|
* Note that the syscall values are consistent with hw.
|
||||||
|
*/
|
||||||
|
env->cp15.sctlr_el[1] =
|
||||||
|
deposit64(env->cp15.sctlr_el[1], 38, 2, arg2 >> PR_MTE_TCF_SHIFT);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Write PR_MTE_TAG to GCR_EL1[Exclude].
|
||||||
|
* Note that the syscall uses an include mask,
|
||||||
|
* and hardware uses an exclude mask -- invert.
|
||||||
|
*/
|
||||||
|
env->cp15.gcr_el1 =
|
||||||
|
deposit64(env->cp15.gcr_el1, 0, 16, ~arg2 >> PR_MTE_TAG_SHIFT);
|
||||||
|
arm_rebuild_hflags(env);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#define do_prctl_set_tagged_addr_ctrl do_prctl_set_tagged_addr_ctrl
|
||||||
|
|
||||||
|
static abi_long do_prctl_get_tagged_addr_ctrl(CPUArchState *env)
|
||||||
|
{
|
||||||
|
ARMCPU *cpu = env_archcpu(env);
|
||||||
|
abi_long ret = 0;
|
||||||
|
|
||||||
|
if (env->tagged_addr_enable) {
|
||||||
|
ret |= PR_TAGGED_ADDR_ENABLE;
|
||||||
|
}
|
||||||
|
if (cpu_isar_feature(aa64_mte, cpu)) {
|
||||||
|
/* See do_prctl_set_tagged_addr_ctrl. */
|
||||||
|
ret |= extract64(env->cp15.sctlr_el[1], 38, 2) << PR_MTE_TCF_SHIFT;
|
||||||
|
ret = deposit64(ret, PR_MTE_TAG_SHIFT, 16, ~env->cp15.gcr_el1);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#define do_prctl_get_tagged_addr_ctrl do_prctl_get_tagged_addr_ctrl
|
||||||
|
|
||||||
|
#endif /* AARCH64_TARGET_PRCTL_H */
|
@ -19,27 +19,4 @@ struct target_pt_regs {
|
|||||||
#define TARGET_MCL_FUTURE 2
|
#define TARGET_MCL_FUTURE 2
|
||||||
#define TARGET_MCL_ONFAULT 4
|
#define TARGET_MCL_ONFAULT 4
|
||||||
|
|
||||||
#define TARGET_PR_SVE_SET_VL 50
|
|
||||||
#define TARGET_PR_SVE_GET_VL 51
|
|
||||||
|
|
||||||
#define TARGET_PR_PAC_RESET_KEYS 54
|
|
||||||
# define TARGET_PR_PAC_APIAKEY (1 << 0)
|
|
||||||
# define TARGET_PR_PAC_APIBKEY (1 << 1)
|
|
||||||
# define TARGET_PR_PAC_APDAKEY (1 << 2)
|
|
||||||
# define TARGET_PR_PAC_APDBKEY (1 << 3)
|
|
||||||
# define TARGET_PR_PAC_APGAKEY (1 << 4)
|
|
||||||
|
|
||||||
#define TARGET_PR_SET_TAGGED_ADDR_CTRL 55
|
|
||||||
#define TARGET_PR_GET_TAGGED_ADDR_CTRL 56
|
|
||||||
# define TARGET_PR_TAGGED_ADDR_ENABLE (1UL << 0)
|
|
||||||
/* MTE tag check fault modes */
|
|
||||||
# define TARGET_PR_MTE_TCF_SHIFT 1
|
|
||||||
# define TARGET_PR_MTE_TCF_NONE (0UL << TARGET_PR_MTE_TCF_SHIFT)
|
|
||||||
# define TARGET_PR_MTE_TCF_SYNC (1UL << TARGET_PR_MTE_TCF_SHIFT)
|
|
||||||
# define TARGET_PR_MTE_TCF_ASYNC (2UL << TARGET_PR_MTE_TCF_SHIFT)
|
|
||||||
# define TARGET_PR_MTE_TCF_MASK (3UL << TARGET_PR_MTE_TCF_SHIFT)
|
|
||||||
/* MTE tag inclusion mask */
|
|
||||||
# define TARGET_PR_MTE_TAG_SHIFT 3
|
|
||||||
# define TARGET_PR_MTE_TAG_MASK (0xffffUL << TARGET_PR_MTE_TAG_SHIFT)
|
|
||||||
|
|
||||||
#endif /* AARCH64_TARGET_SYSCALL_H */
|
#endif /* AARCH64_TARGET_SYSCALL_H */
|
||||||
|
1
linux-user/alpha/target_prctl.h
Normal file
1
linux-user/alpha/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/arm/target_prctl.h
Normal file
1
linux-user/arm/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/cris/target_prctl.h
Normal file
1
linux-user/cris/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/hexagon/target_prctl.h
Normal file
1
linux-user/hexagon/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/hppa/target_prctl.h
Normal file
1
linux-user/hppa/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/i386/target_prctl.h
Normal file
1
linux-user/i386/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/m68k/target_prctl.h
Normal file
1
linux-user/m68k/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/microblaze/target_prctl.h
Normal file
1
linux-user/microblaze/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
88
linux-user/mips/target_prctl.h
Normal file
88
linux-user/mips/target_prctl.h
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
* MIPS specific prctl functions for linux-user
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
*/
|
||||||
|
#ifndef MIPS_TARGET_PRCTL_H
|
||||||
|
#define MIPS_TARGET_PRCTL_H
|
||||||
|
|
||||||
|
static abi_long do_prctl_get_fp_mode(CPUArchState *env)
|
||||||
|
{
|
||||||
|
abi_long ret = 0;
|
||||||
|
|
||||||
|
if (env->CP0_Status & (1 << CP0St_FR)) {
|
||||||
|
ret |= PR_FP_MODE_FR;
|
||||||
|
}
|
||||||
|
if (env->CP0_Config5 & (1 << CP0C5_FRE)) {
|
||||||
|
ret |= PR_FP_MODE_FRE;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#define do_prctl_get_fp_mode do_prctl_get_fp_mode
|
||||||
|
|
||||||
|
static abi_long do_prctl_set_fp_mode(CPUArchState *env, abi_long arg2)
|
||||||
|
{
|
||||||
|
bool old_fr = env->CP0_Status & (1 << CP0St_FR);
|
||||||
|
bool old_fre = env->CP0_Config5 & (1 << CP0C5_FRE);
|
||||||
|
bool new_fr = arg2 & PR_FP_MODE_FR;
|
||||||
|
bool new_fre = arg2 & PR_FP_MODE_FRE;
|
||||||
|
const unsigned int known_bits = PR_FP_MODE_FR | PR_FP_MODE_FRE;
|
||||||
|
|
||||||
|
/* If nothing to change, return right away, successfully. */
|
||||||
|
if (old_fr == new_fr && old_fre == new_fre) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
/* Check the value is valid */
|
||||||
|
if (arg2 & ~known_bits) {
|
||||||
|
return -TARGET_EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
/* Setting FRE without FR is not supported. */
|
||||||
|
if (new_fre && !new_fr) {
|
||||||
|
return -TARGET_EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
if (new_fr && !(env->active_fpu.fcr0 & (1 << FCR0_F64))) {
|
||||||
|
/* FR1 is not supported */
|
||||||
|
return -TARGET_EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
if (!new_fr && (env->active_fpu.fcr0 & (1 << FCR0_F64))
|
||||||
|
&& !(env->CP0_Status_rw_bitmask & (1 << CP0St_FR))) {
|
||||||
|
/* cannot set FR=0 */
|
||||||
|
return -TARGET_EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
if (new_fre && !(env->active_fpu.fcr0 & (1 << FCR0_FREP))) {
|
||||||
|
/* Cannot set FRE=1 */
|
||||||
|
return -TARGET_EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
|
||||||
|
int i;
|
||||||
|
fpr_t *fpr = env->active_fpu.fpr;
|
||||||
|
for (i = 0; i < 32 ; i += 2) {
|
||||||
|
if (!old_fr && new_fr) {
|
||||||
|
fpr[i].w[!FP_ENDIAN_IDX] = fpr[i + 1].w[FP_ENDIAN_IDX];
|
||||||
|
} else if (old_fr && !new_fr) {
|
||||||
|
fpr[i + 1].w[FP_ENDIAN_IDX] = fpr[i].w[!FP_ENDIAN_IDX];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (new_fr) {
|
||||||
|
env->CP0_Status |= (1 << CP0St_FR);
|
||||||
|
env->hflags |= MIPS_HFLAG_F64;
|
||||||
|
} else {
|
||||||
|
env->CP0_Status &= ~(1 << CP0St_FR);
|
||||||
|
env->hflags &= ~MIPS_HFLAG_F64;
|
||||||
|
}
|
||||||
|
if (new_fre) {
|
||||||
|
env->CP0_Config5 |= (1 << CP0C5_FRE);
|
||||||
|
if (env->active_fpu.fcr0 & (1 << FCR0_FREP)) {
|
||||||
|
env->hflags |= MIPS_HFLAG_FRE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
env->CP0_Config5 &= ~(1 << CP0C5_FRE);
|
||||||
|
env->hflags &= ~MIPS_HFLAG_FRE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#define do_prctl_set_fp_mode do_prctl_set_fp_mode
|
||||||
|
|
||||||
|
#endif /* MIPS_TARGET_PRCTL_H */
|
@ -35,10 +35,4 @@ static inline abi_ulong target_shmlba(CPUMIPSState *env)
|
|||||||
return 0x40000;
|
return 0x40000;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* MIPS-specific prctl() options */
|
|
||||||
#define TARGET_PR_SET_FP_MODE 45
|
|
||||||
#define TARGET_PR_GET_FP_MODE 46
|
|
||||||
#define TARGET_PR_FP_MODE_FR (1 << 0)
|
|
||||||
#define TARGET_PR_FP_MODE_FRE (1 << 1)
|
|
||||||
|
|
||||||
#endif /* MIPS_TARGET_SYSCALL_H */
|
#endif /* MIPS_TARGET_SYSCALL_H */
|
||||||
|
1
linux-user/mips64/target_prctl.h
Normal file
1
linux-user/mips64/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#include "../mips/target_prctl.h"
|
@ -32,10 +32,4 @@ static inline abi_ulong target_shmlba(CPUMIPSState *env)
|
|||||||
return 0x40000;
|
return 0x40000;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* MIPS-specific prctl() options */
|
|
||||||
#define TARGET_PR_SET_FP_MODE 45
|
|
||||||
#define TARGET_PR_GET_FP_MODE 46
|
|
||||||
#define TARGET_PR_FP_MODE_FR (1 << 0)
|
|
||||||
#define TARGET_PR_FP_MODE_FRE (1 << 1)
|
|
||||||
|
|
||||||
#endif /* MIPS64_TARGET_SYSCALL_H */
|
#endif /* MIPS64_TARGET_SYSCALL_H */
|
||||||
|
1
linux-user/nios2/target_prctl.h
Normal file
1
linux-user/nios2/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/openrisc/target_prctl.h
Normal file
1
linux-user/openrisc/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/ppc/target_prctl.h
Normal file
1
linux-user/ppc/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/riscv/target_prctl.h
Normal file
1
linux-user/riscv/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/s390x/target_prctl.h
Normal file
1
linux-user/s390x/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/sh4/target_prctl.h
Normal file
1
linux-user/sh4/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/sparc/target_prctl.h
Normal file
1
linux-user/sparc/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
@ -6294,9 +6294,155 @@ abi_long do_arch_prctl(CPUX86State *env, int code, abi_ulong addr)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
#endif /* defined(TARGET_ABI32 */
|
#endif /* defined(TARGET_ABI32 */
|
||||||
|
|
||||||
#endif /* defined(TARGET_I386) */
|
#endif /* defined(TARGET_I386) */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* These constants are generic. Supply any that are missing from the host.
|
||||||
|
*/
|
||||||
|
#ifndef PR_SET_NAME
|
||||||
|
# define PR_SET_NAME 15
|
||||||
|
# define PR_GET_NAME 16
|
||||||
|
#endif
|
||||||
|
#ifndef PR_SET_FP_MODE
|
||||||
|
# define PR_SET_FP_MODE 45
|
||||||
|
# define PR_GET_FP_MODE 46
|
||||||
|
# define PR_FP_MODE_FR (1 << 0)
|
||||||
|
# define PR_FP_MODE_FRE (1 << 1)
|
||||||
|
#endif
|
||||||
|
#ifndef PR_SVE_SET_VL
|
||||||
|
# define PR_SVE_SET_VL 50
|
||||||
|
# define PR_SVE_GET_VL 51
|
||||||
|
# define PR_SVE_VL_LEN_MASK 0xffff
|
||||||
|
# define PR_SVE_VL_INHERIT (1 << 17)
|
||||||
|
#endif
|
||||||
|
#ifndef PR_PAC_RESET_KEYS
|
||||||
|
# define PR_PAC_RESET_KEYS 54
|
||||||
|
# define PR_PAC_APIAKEY (1 << 0)
|
||||||
|
# define PR_PAC_APIBKEY (1 << 1)
|
||||||
|
# define PR_PAC_APDAKEY (1 << 2)
|
||||||
|
# define PR_PAC_APDBKEY (1 << 3)
|
||||||
|
# define PR_PAC_APGAKEY (1 << 4)
|
||||||
|
#endif
|
||||||
|
#ifndef PR_SET_TAGGED_ADDR_CTRL
|
||||||
|
# define PR_SET_TAGGED_ADDR_CTRL 55
|
||||||
|
# define PR_GET_TAGGED_ADDR_CTRL 56
|
||||||
|
# define PR_TAGGED_ADDR_ENABLE (1UL << 0)
|
||||||
|
#endif
|
||||||
|
#ifndef PR_MTE_TCF_SHIFT
|
||||||
|
# define PR_MTE_TCF_SHIFT 1
|
||||||
|
# define PR_MTE_TCF_NONE (0UL << PR_MTE_TCF_SHIFT)
|
||||||
|
# define PR_MTE_TCF_SYNC (1UL << PR_MTE_TCF_SHIFT)
|
||||||
|
# define PR_MTE_TCF_ASYNC (2UL << PR_MTE_TCF_SHIFT)
|
||||||
|
# define PR_MTE_TCF_MASK (3UL << PR_MTE_TCF_SHIFT)
|
||||||
|
# define PR_MTE_TAG_SHIFT 3
|
||||||
|
# define PR_MTE_TAG_MASK (0xffffUL << PR_MTE_TAG_SHIFT)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "target_prctl.h"
|
||||||
|
|
||||||
|
static abi_long do_prctl_inval0(CPUArchState *env)
|
||||||
|
{
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static abi_long do_prctl_inval1(CPUArchState *env, abi_long arg2)
|
||||||
|
{
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef do_prctl_get_fp_mode
|
||||||
|
#define do_prctl_get_fp_mode do_prctl_inval0
|
||||||
|
#endif
|
||||||
|
#ifndef do_prctl_set_fp_mode
|
||||||
|
#define do_prctl_set_fp_mode do_prctl_inval1
|
||||||
|
#endif
|
||||||
|
#ifndef do_prctl_get_vl
|
||||||
|
#define do_prctl_get_vl do_prctl_inval0
|
||||||
|
#endif
|
||||||
|
#ifndef do_prctl_set_vl
|
||||||
|
#define do_prctl_set_vl do_prctl_inval1
|
||||||
|
#endif
|
||||||
|
#ifndef do_prctl_reset_keys
|
||||||
|
#define do_prctl_reset_keys do_prctl_inval1
|
||||||
|
#endif
|
||||||
|
#ifndef do_prctl_set_tagged_addr_ctrl
|
||||||
|
#define do_prctl_set_tagged_addr_ctrl do_prctl_inval1
|
||||||
|
#endif
|
||||||
|
#ifndef do_prctl_get_tagged_addr_ctrl
|
||||||
|
#define do_prctl_get_tagged_addr_ctrl do_prctl_inval0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static abi_long do_prctl(CPUArchState *env, abi_long option, abi_long arg2,
|
||||||
|
abi_long arg3, abi_long arg4, abi_long arg5)
|
||||||
|
{
|
||||||
|
abi_long ret;
|
||||||
|
|
||||||
|
switch (option) {
|
||||||
|
case PR_GET_PDEATHSIG:
|
||||||
|
{
|
||||||
|
int deathsig;
|
||||||
|
ret = get_errno(prctl(PR_GET_PDEATHSIG, &deathsig,
|
||||||
|
arg3, arg4, arg5));
|
||||||
|
if (!is_error(ret) && arg2 && put_user_s32(deathsig, arg2)) {
|
||||||
|
return -TARGET_EFAULT;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
case PR_GET_NAME:
|
||||||
|
{
|
||||||
|
void *name = lock_user(VERIFY_WRITE, arg2, 16, 1);
|
||||||
|
if (!name) {
|
||||||
|
return -TARGET_EFAULT;
|
||||||
|
}
|
||||||
|
ret = get_errno(prctl(PR_GET_NAME, (uintptr_t)name,
|
||||||
|
arg3, arg4, arg5));
|
||||||
|
unlock_user(name, arg2, 16);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
case PR_SET_NAME:
|
||||||
|
{
|
||||||
|
void *name = lock_user(VERIFY_READ, arg2, 16, 1);
|
||||||
|
if (!name) {
|
||||||
|
return -TARGET_EFAULT;
|
||||||
|
}
|
||||||
|
ret = get_errno(prctl(PR_SET_NAME, (uintptr_t)name,
|
||||||
|
arg3, arg4, arg5));
|
||||||
|
unlock_user(name, arg2, 0);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
case PR_GET_FP_MODE:
|
||||||
|
return do_prctl_get_fp_mode(env);
|
||||||
|
case PR_SET_FP_MODE:
|
||||||
|
return do_prctl_set_fp_mode(env, arg2);
|
||||||
|
case PR_SVE_GET_VL:
|
||||||
|
return do_prctl_get_vl(env);
|
||||||
|
case PR_SVE_SET_VL:
|
||||||
|
return do_prctl_set_vl(env, arg2);
|
||||||
|
case PR_PAC_RESET_KEYS:
|
||||||
|
if (arg3 || arg4 || arg5) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
return do_prctl_reset_keys(env, arg2);
|
||||||
|
case PR_SET_TAGGED_ADDR_CTRL:
|
||||||
|
if (arg3 || arg4 || arg5) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
return do_prctl_set_tagged_addr_ctrl(env, arg2);
|
||||||
|
case PR_GET_TAGGED_ADDR_CTRL:
|
||||||
|
if (arg2 || arg3 || arg4 || arg5) {
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
}
|
||||||
|
return do_prctl_get_tagged_addr_ctrl(env);
|
||||||
|
case PR_GET_SECCOMP:
|
||||||
|
case PR_SET_SECCOMP:
|
||||||
|
/* Disable seccomp to prevent the target disabling syscalls we need. */
|
||||||
|
return -TARGET_EINVAL;
|
||||||
|
default:
|
||||||
|
/* Most prctl options have no pointer arguments */
|
||||||
|
return get_errno(prctl(option, arg2, arg3, arg4, arg5));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#define NEW_STACK_SIZE 0x40000
|
#define NEW_STACK_SIZE 0x40000
|
||||||
|
|
||||||
|
|
||||||
@ -10635,290 +10781,7 @@ static abi_long do_syscall1(void *cpu_env, int num, abi_long arg1,
|
|||||||
return ret;
|
return ret;
|
||||||
#endif
|
#endif
|
||||||
case TARGET_NR_prctl:
|
case TARGET_NR_prctl:
|
||||||
switch (arg1) {
|
return do_prctl(cpu_env, arg1, arg2, arg3, arg4, arg5);
|
||||||
case PR_GET_PDEATHSIG:
|
|
||||||
{
|
|
||||||
int deathsig;
|
|
||||||
ret = get_errno(prctl(arg1, &deathsig, arg3, arg4, arg5));
|
|
||||||
if (!is_error(ret) && arg2
|
|
||||||
&& put_user_s32(deathsig, arg2)) {
|
|
||||||
return -TARGET_EFAULT;
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#ifdef PR_GET_NAME
|
|
||||||
case PR_GET_NAME:
|
|
||||||
{
|
|
||||||
void *name = lock_user(VERIFY_WRITE, arg2, 16, 1);
|
|
||||||
if (!name) {
|
|
||||||
return -TARGET_EFAULT;
|
|
||||||
}
|
|
||||||
ret = get_errno(prctl(arg1, (unsigned long)name,
|
|
||||||
arg3, arg4, arg5));
|
|
||||||
unlock_user(name, arg2, 16);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
case PR_SET_NAME:
|
|
||||||
{
|
|
||||||
void *name = lock_user(VERIFY_READ, arg2, 16, 1);
|
|
||||||
if (!name) {
|
|
||||||
return -TARGET_EFAULT;
|
|
||||||
}
|
|
||||||
ret = get_errno(prctl(arg1, (unsigned long)name,
|
|
||||||
arg3, arg4, arg5));
|
|
||||||
unlock_user(name, arg2, 0);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef TARGET_MIPS
|
|
||||||
case TARGET_PR_GET_FP_MODE:
|
|
||||||
{
|
|
||||||
CPUMIPSState *env = ((CPUMIPSState *)cpu_env);
|
|
||||||
ret = 0;
|
|
||||||
if (env->CP0_Status & (1 << CP0St_FR)) {
|
|
||||||
ret |= TARGET_PR_FP_MODE_FR;
|
|
||||||
}
|
|
||||||
if (env->CP0_Config5 & (1 << CP0C5_FRE)) {
|
|
||||||
ret |= TARGET_PR_FP_MODE_FRE;
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
case TARGET_PR_SET_FP_MODE:
|
|
||||||
{
|
|
||||||
CPUMIPSState *env = ((CPUMIPSState *)cpu_env);
|
|
||||||
bool old_fr = env->CP0_Status & (1 << CP0St_FR);
|
|
||||||
bool old_fre = env->CP0_Config5 & (1 << CP0C5_FRE);
|
|
||||||
bool new_fr = arg2 & TARGET_PR_FP_MODE_FR;
|
|
||||||
bool new_fre = arg2 & TARGET_PR_FP_MODE_FRE;
|
|
||||||
|
|
||||||
const unsigned int known_bits = TARGET_PR_FP_MODE_FR |
|
|
||||||
TARGET_PR_FP_MODE_FRE;
|
|
||||||
|
|
||||||
/* If nothing to change, return right away, successfully. */
|
|
||||||
if (old_fr == new_fr && old_fre == new_fre) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
/* Check the value is valid */
|
|
||||||
if (arg2 & ~known_bits) {
|
|
||||||
return -TARGET_EOPNOTSUPP;
|
|
||||||
}
|
|
||||||
/* Setting FRE without FR is not supported. */
|
|
||||||
if (new_fre && !new_fr) {
|
|
||||||
return -TARGET_EOPNOTSUPP;
|
|
||||||
}
|
|
||||||
if (new_fr && !(env->active_fpu.fcr0 & (1 << FCR0_F64))) {
|
|
||||||
/* FR1 is not supported */
|
|
||||||
return -TARGET_EOPNOTSUPP;
|
|
||||||
}
|
|
||||||
if (!new_fr && (env->active_fpu.fcr0 & (1 << FCR0_F64))
|
|
||||||
&& !(env->CP0_Status_rw_bitmask & (1 << CP0St_FR))) {
|
|
||||||
/* cannot set FR=0 */
|
|
||||||
return -TARGET_EOPNOTSUPP;
|
|
||||||
}
|
|
||||||
if (new_fre && !(env->active_fpu.fcr0 & (1 << FCR0_FREP))) {
|
|
||||||
/* Cannot set FRE=1 */
|
|
||||||
return -TARGET_EOPNOTSUPP;
|
|
||||||
}
|
|
||||||
|
|
||||||
int i;
|
|
||||||
fpr_t *fpr = env->active_fpu.fpr;
|
|
||||||
for (i = 0; i < 32 ; i += 2) {
|
|
||||||
if (!old_fr && new_fr) {
|
|
||||||
fpr[i].w[!FP_ENDIAN_IDX] = fpr[i + 1].w[FP_ENDIAN_IDX];
|
|
||||||
} else if (old_fr && !new_fr) {
|
|
||||||
fpr[i + 1].w[FP_ENDIAN_IDX] = fpr[i].w[!FP_ENDIAN_IDX];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (new_fr) {
|
|
||||||
env->CP0_Status |= (1 << CP0St_FR);
|
|
||||||
env->hflags |= MIPS_HFLAG_F64;
|
|
||||||
} else {
|
|
||||||
env->CP0_Status &= ~(1 << CP0St_FR);
|
|
||||||
env->hflags &= ~MIPS_HFLAG_F64;
|
|
||||||
}
|
|
||||||
if (new_fre) {
|
|
||||||
env->CP0_Config5 |= (1 << CP0C5_FRE);
|
|
||||||
if (env->active_fpu.fcr0 & (1 << FCR0_FREP)) {
|
|
||||||
env->hflags |= MIPS_HFLAG_FRE;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
env->CP0_Config5 &= ~(1 << CP0C5_FRE);
|
|
||||||
env->hflags &= ~MIPS_HFLAG_FRE;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif /* MIPS */
|
|
||||||
#ifdef TARGET_AARCH64
|
|
||||||
case TARGET_PR_SVE_SET_VL:
|
|
||||||
/*
|
|
||||||
* We cannot support either PR_SVE_SET_VL_ONEXEC or
|
|
||||||
* PR_SVE_VL_INHERIT. Note the kernel definition
|
|
||||||
* of sve_vl_valid allows for VQ=512, i.e. VL=8192,
|
|
||||||
* even though the current architectural maximum is VQ=16.
|
|
||||||
*/
|
|
||||||
ret = -TARGET_EINVAL;
|
|
||||||
if (cpu_isar_feature(aa64_sve, env_archcpu(cpu_env))
|
|
||||||
&& arg2 >= 0 && arg2 <= 512 * 16 && !(arg2 & 15)) {
|
|
||||||
CPUARMState *env = cpu_env;
|
|
||||||
ARMCPU *cpu = env_archcpu(env);
|
|
||||||
uint32_t vq, old_vq;
|
|
||||||
|
|
||||||
old_vq = (env->vfp.zcr_el[1] & 0xf) + 1;
|
|
||||||
vq = MAX(arg2 / 16, 1);
|
|
||||||
vq = MIN(vq, cpu->sve_max_vq);
|
|
||||||
|
|
||||||
if (vq < old_vq) {
|
|
||||||
aarch64_sve_narrow_vq(env, vq);
|
|
||||||
}
|
|
||||||
env->vfp.zcr_el[1] = vq - 1;
|
|
||||||
arm_rebuild_hflags(env);
|
|
||||||
ret = vq * 16;
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
case TARGET_PR_SVE_GET_VL:
|
|
||||||
ret = -TARGET_EINVAL;
|
|
||||||
{
|
|
||||||
ARMCPU *cpu = env_archcpu(cpu_env);
|
|
||||||
if (cpu_isar_feature(aa64_sve, cpu)) {
|
|
||||||
ret = ((cpu->env.vfp.zcr_el[1] & 0xf) + 1) * 16;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
case TARGET_PR_PAC_RESET_KEYS:
|
|
||||||
{
|
|
||||||
CPUARMState *env = cpu_env;
|
|
||||||
ARMCPU *cpu = env_archcpu(env);
|
|
||||||
|
|
||||||
if (arg3 || arg4 || arg5) {
|
|
||||||
return -TARGET_EINVAL;
|
|
||||||
}
|
|
||||||
if (cpu_isar_feature(aa64_pauth, cpu)) {
|
|
||||||
int all = (TARGET_PR_PAC_APIAKEY | TARGET_PR_PAC_APIBKEY |
|
|
||||||
TARGET_PR_PAC_APDAKEY | TARGET_PR_PAC_APDBKEY |
|
|
||||||
TARGET_PR_PAC_APGAKEY);
|
|
||||||
int ret = 0;
|
|
||||||
Error *err = NULL;
|
|
||||||
|
|
||||||
if (arg2 == 0) {
|
|
||||||
arg2 = all;
|
|
||||||
} else if (arg2 & ~all) {
|
|
||||||
return -TARGET_EINVAL;
|
|
||||||
}
|
|
||||||
if (arg2 & TARGET_PR_PAC_APIAKEY) {
|
|
||||||
ret |= qemu_guest_getrandom(&env->keys.apia,
|
|
||||||
sizeof(ARMPACKey), &err);
|
|
||||||
}
|
|
||||||
if (arg2 & TARGET_PR_PAC_APIBKEY) {
|
|
||||||
ret |= qemu_guest_getrandom(&env->keys.apib,
|
|
||||||
sizeof(ARMPACKey), &err);
|
|
||||||
}
|
|
||||||
if (arg2 & TARGET_PR_PAC_APDAKEY) {
|
|
||||||
ret |= qemu_guest_getrandom(&env->keys.apda,
|
|
||||||
sizeof(ARMPACKey), &err);
|
|
||||||
}
|
|
||||||
if (arg2 & TARGET_PR_PAC_APDBKEY) {
|
|
||||||
ret |= qemu_guest_getrandom(&env->keys.apdb,
|
|
||||||
sizeof(ARMPACKey), &err);
|
|
||||||
}
|
|
||||||
if (arg2 & TARGET_PR_PAC_APGAKEY) {
|
|
||||||
ret |= qemu_guest_getrandom(&env->keys.apga,
|
|
||||||
sizeof(ARMPACKey), &err);
|
|
||||||
}
|
|
||||||
if (ret != 0) {
|
|
||||||
/*
|
|
||||||
* Some unknown failure in the crypto. The best
|
|
||||||
* we can do is log it and fail the syscall.
|
|
||||||
* The real syscall cannot fail this way.
|
|
||||||
*/
|
|
||||||
qemu_log_mask(LOG_UNIMP,
|
|
||||||
"PR_PAC_RESET_KEYS: Crypto failure: %s",
|
|
||||||
error_get_pretty(err));
|
|
||||||
error_free(err);
|
|
||||||
return -TARGET_EIO;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return -TARGET_EINVAL;
|
|
||||||
case TARGET_PR_SET_TAGGED_ADDR_CTRL:
|
|
||||||
{
|
|
||||||
abi_ulong valid_mask = TARGET_PR_TAGGED_ADDR_ENABLE;
|
|
||||||
CPUARMState *env = cpu_env;
|
|
||||||
ARMCPU *cpu = env_archcpu(env);
|
|
||||||
|
|
||||||
if (cpu_isar_feature(aa64_mte, cpu)) {
|
|
||||||
valid_mask |= TARGET_PR_MTE_TCF_MASK;
|
|
||||||
valid_mask |= TARGET_PR_MTE_TAG_MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((arg2 & ~valid_mask) || arg3 || arg4 || arg5) {
|
|
||||||
return -TARGET_EINVAL;
|
|
||||||
}
|
|
||||||
env->tagged_addr_enable = arg2 & TARGET_PR_TAGGED_ADDR_ENABLE;
|
|
||||||
|
|
||||||
if (cpu_isar_feature(aa64_mte, cpu)) {
|
|
||||||
switch (arg2 & TARGET_PR_MTE_TCF_MASK) {
|
|
||||||
case TARGET_PR_MTE_TCF_NONE:
|
|
||||||
case TARGET_PR_MTE_TCF_SYNC:
|
|
||||||
case TARGET_PR_MTE_TCF_ASYNC:
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Write PR_MTE_TCF to SCTLR_EL1[TCF0].
|
|
||||||
* Note that the syscall values are consistent with hw.
|
|
||||||
*/
|
|
||||||
env->cp15.sctlr_el[1] =
|
|
||||||
deposit64(env->cp15.sctlr_el[1], 38, 2,
|
|
||||||
arg2 >> TARGET_PR_MTE_TCF_SHIFT);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Write PR_MTE_TAG to GCR_EL1[Exclude].
|
|
||||||
* Note that the syscall uses an include mask,
|
|
||||||
* and hardware uses an exclude mask -- invert.
|
|
||||||
*/
|
|
||||||
env->cp15.gcr_el1 =
|
|
||||||
deposit64(env->cp15.gcr_el1, 0, 16,
|
|
||||||
~arg2 >> TARGET_PR_MTE_TAG_SHIFT);
|
|
||||||
arm_rebuild_hflags(env);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
case TARGET_PR_GET_TAGGED_ADDR_CTRL:
|
|
||||||
{
|
|
||||||
abi_long ret = 0;
|
|
||||||
CPUARMState *env = cpu_env;
|
|
||||||
ARMCPU *cpu = env_archcpu(env);
|
|
||||||
|
|
||||||
if (arg2 || arg3 || arg4 || arg5) {
|
|
||||||
return -TARGET_EINVAL;
|
|
||||||
}
|
|
||||||
if (env->tagged_addr_enable) {
|
|
||||||
ret |= TARGET_PR_TAGGED_ADDR_ENABLE;
|
|
||||||
}
|
|
||||||
if (cpu_isar_feature(aa64_mte, cpu)) {
|
|
||||||
/* See above. */
|
|
||||||
ret |= (extract64(env->cp15.sctlr_el[1], 38, 2)
|
|
||||||
<< TARGET_PR_MTE_TCF_SHIFT);
|
|
||||||
ret = deposit64(ret, TARGET_PR_MTE_TAG_SHIFT, 16,
|
|
||||||
~env->cp15.gcr_el1);
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#endif /* AARCH64 */
|
|
||||||
case PR_GET_SECCOMP:
|
|
||||||
case PR_SET_SECCOMP:
|
|
||||||
/* Disable seccomp to prevent the target disabling syscalls we
|
|
||||||
* need. */
|
|
||||||
return -TARGET_EINVAL;
|
|
||||||
default:
|
|
||||||
/* Most prctl options have no pointer arguments */
|
|
||||||
return get_errno(prctl(arg1, arg2, arg3, arg4, arg5));
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
#ifdef TARGET_NR_arch_prctl
|
#ifdef TARGET_NR_arch_prctl
|
||||||
case TARGET_NR_arch_prctl:
|
case TARGET_NR_arch_prctl:
|
||||||
|
1
linux-user/x86_64/target_prctl.h
Normal file
1
linux-user/x86_64/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
1
linux-user/xtensa/target_prctl.h
Normal file
1
linux-user/xtensa/target_prctl.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
/* No special prctl support required. */
|
Loading…
Reference in New Issue
Block a user