1*87e9bf23SRichard Henderson /* 2*87e9bf23SRichard Henderson * AArch64 specific prctl functions for linux-user 3*87e9bf23SRichard Henderson * 4*87e9bf23SRichard Henderson * SPDX-License-Identifier: GPL-2.0-or-later 5*87e9bf23SRichard Henderson */ 6*87e9bf23SRichard Henderson #ifndef AARCH64_TARGET_PRCTL_H 7*87e9bf23SRichard Henderson #define AARCH64_TARGET_PRCTL_H 8*87e9bf23SRichard Henderson 9*87e9bf23SRichard Henderson static abi_long do_prctl_get_vl(CPUArchState *env) 10*87e9bf23SRichard Henderson { 11*87e9bf23SRichard Henderson ARMCPU *cpu = env_archcpu(env); 12*87e9bf23SRichard Henderson if (cpu_isar_feature(aa64_sve, cpu)) { 13*87e9bf23SRichard Henderson return ((cpu->env.vfp.zcr_el[1] & 0xf) + 1) * 16; 14*87e9bf23SRichard Henderson } 15*87e9bf23SRichard Henderson return -TARGET_EINVAL; 16*87e9bf23SRichard Henderson } 17*87e9bf23SRichard Henderson #define do_prctl_get_vl do_prctl_get_vl 18*87e9bf23SRichard Henderson 19*87e9bf23SRichard Henderson static abi_long do_prctl_set_vl(CPUArchState *env, abi_long arg2) 20*87e9bf23SRichard Henderson { 21*87e9bf23SRichard Henderson /* 22*87e9bf23SRichard Henderson * We cannot support either PR_SVE_SET_VL_ONEXEC or PR_SVE_VL_INHERIT. 23*87e9bf23SRichard Henderson * Note the kernel definition of sve_vl_valid allows for VQ=512, 24*87e9bf23SRichard Henderson * i.e. VL=8192, even though the current architectural maximum is VQ=16. 25*87e9bf23SRichard Henderson */ 26*87e9bf23SRichard Henderson if (cpu_isar_feature(aa64_sve, env_archcpu(env)) 27*87e9bf23SRichard Henderson && arg2 >= 0 && arg2 <= 512 * 16 && !(arg2 & 15)) { 28*87e9bf23SRichard Henderson ARMCPU *cpu = env_archcpu(env); 29*87e9bf23SRichard Henderson uint32_t vq, old_vq; 30*87e9bf23SRichard Henderson 31*87e9bf23SRichard Henderson old_vq = (env->vfp.zcr_el[1] & 0xf) + 1; 32*87e9bf23SRichard Henderson vq = MAX(arg2 / 16, 1); 33*87e9bf23SRichard Henderson vq = MIN(vq, cpu->sve_max_vq); 34*87e9bf23SRichard Henderson 35*87e9bf23SRichard Henderson if (vq < old_vq) { 36*87e9bf23SRichard Henderson aarch64_sve_narrow_vq(env, vq); 37*87e9bf23SRichard Henderson } 38*87e9bf23SRichard Henderson env->vfp.zcr_el[1] = vq - 1; 39*87e9bf23SRichard Henderson arm_rebuild_hflags(env); 40*87e9bf23SRichard Henderson return vq * 16; 41*87e9bf23SRichard Henderson } 42*87e9bf23SRichard Henderson return -TARGET_EINVAL; 43*87e9bf23SRichard Henderson } 44*87e9bf23SRichard Henderson #define do_prctl_set_vl do_prctl_set_vl 45*87e9bf23SRichard Henderson 46*87e9bf23SRichard Henderson static abi_long do_prctl_reset_keys(CPUArchState *env, abi_long arg2) 47*87e9bf23SRichard Henderson { 48*87e9bf23SRichard Henderson ARMCPU *cpu = env_archcpu(env); 49*87e9bf23SRichard Henderson 50*87e9bf23SRichard Henderson if (cpu_isar_feature(aa64_pauth, cpu)) { 51*87e9bf23SRichard Henderson int all = (PR_PAC_APIAKEY | PR_PAC_APIBKEY | 52*87e9bf23SRichard Henderson PR_PAC_APDAKEY | PR_PAC_APDBKEY | PR_PAC_APGAKEY); 53*87e9bf23SRichard Henderson int ret = 0; 54*87e9bf23SRichard Henderson Error *err = NULL; 55*87e9bf23SRichard Henderson 56*87e9bf23SRichard Henderson if (arg2 == 0) { 57*87e9bf23SRichard Henderson arg2 = all; 58*87e9bf23SRichard Henderson } else if (arg2 & ~all) { 59*87e9bf23SRichard Henderson return -TARGET_EINVAL; 60*87e9bf23SRichard Henderson } 61*87e9bf23SRichard Henderson if (arg2 & PR_PAC_APIAKEY) { 62*87e9bf23SRichard Henderson ret |= qemu_guest_getrandom(&env->keys.apia, 63*87e9bf23SRichard Henderson sizeof(ARMPACKey), &err); 64*87e9bf23SRichard Henderson } 65*87e9bf23SRichard Henderson if (arg2 & PR_PAC_APIBKEY) { 66*87e9bf23SRichard Henderson ret |= qemu_guest_getrandom(&env->keys.apib, 67*87e9bf23SRichard Henderson sizeof(ARMPACKey), &err); 68*87e9bf23SRichard Henderson } 69*87e9bf23SRichard Henderson if (arg2 & PR_PAC_APDAKEY) { 70*87e9bf23SRichard Henderson ret |= qemu_guest_getrandom(&env->keys.apda, 71*87e9bf23SRichard Henderson sizeof(ARMPACKey), &err); 72*87e9bf23SRichard Henderson } 73*87e9bf23SRichard Henderson if (arg2 & PR_PAC_APDBKEY) { 74*87e9bf23SRichard Henderson ret |= qemu_guest_getrandom(&env->keys.apdb, 75*87e9bf23SRichard Henderson sizeof(ARMPACKey), &err); 76*87e9bf23SRichard Henderson } 77*87e9bf23SRichard Henderson if (arg2 & PR_PAC_APGAKEY) { 78*87e9bf23SRichard Henderson ret |= qemu_guest_getrandom(&env->keys.apga, 79*87e9bf23SRichard Henderson sizeof(ARMPACKey), &err); 80*87e9bf23SRichard Henderson } 81*87e9bf23SRichard Henderson if (ret != 0) { 82*87e9bf23SRichard Henderson /* 83*87e9bf23SRichard Henderson * Some unknown failure in the crypto. The best 84*87e9bf23SRichard Henderson * we can do is log it and fail the syscall. 85*87e9bf23SRichard Henderson * The real syscall cannot fail this way. 86*87e9bf23SRichard Henderson */ 87*87e9bf23SRichard Henderson qemu_log_mask(LOG_UNIMP, "PR_PAC_RESET_KEYS: Crypto failure: %s", 88*87e9bf23SRichard Henderson error_get_pretty(err)); 89*87e9bf23SRichard Henderson error_free(err); 90*87e9bf23SRichard Henderson return -TARGET_EIO; 91*87e9bf23SRichard Henderson } 92*87e9bf23SRichard Henderson return 0; 93*87e9bf23SRichard Henderson } 94*87e9bf23SRichard Henderson return -TARGET_EINVAL; 95*87e9bf23SRichard Henderson } 96*87e9bf23SRichard Henderson #define do_prctl_reset_keys do_prctl_reset_keys 97*87e9bf23SRichard Henderson 98*87e9bf23SRichard Henderson static abi_long do_prctl_set_tagged_addr_ctrl(CPUArchState *env, abi_long arg2) 99*87e9bf23SRichard Henderson { 100*87e9bf23SRichard Henderson abi_ulong valid_mask = PR_TAGGED_ADDR_ENABLE; 101*87e9bf23SRichard Henderson ARMCPU *cpu = env_archcpu(env); 102*87e9bf23SRichard Henderson 103*87e9bf23SRichard Henderson if (cpu_isar_feature(aa64_mte, cpu)) { 104*87e9bf23SRichard Henderson valid_mask |= PR_MTE_TCF_MASK; 105*87e9bf23SRichard Henderson valid_mask |= PR_MTE_TAG_MASK; 106*87e9bf23SRichard Henderson } 107*87e9bf23SRichard Henderson 108*87e9bf23SRichard Henderson if (arg2 & ~valid_mask) { 109*87e9bf23SRichard Henderson return -TARGET_EINVAL; 110*87e9bf23SRichard Henderson } 111*87e9bf23SRichard Henderson env->tagged_addr_enable = arg2 & PR_TAGGED_ADDR_ENABLE; 112*87e9bf23SRichard Henderson 113*87e9bf23SRichard Henderson if (cpu_isar_feature(aa64_mte, cpu)) { 114*87e9bf23SRichard Henderson switch (arg2 & PR_MTE_TCF_MASK) { 115*87e9bf23SRichard Henderson case PR_MTE_TCF_NONE: 116*87e9bf23SRichard Henderson case PR_MTE_TCF_SYNC: 117*87e9bf23SRichard Henderson case PR_MTE_TCF_ASYNC: 118*87e9bf23SRichard Henderson break; 119*87e9bf23SRichard Henderson default: 120*87e9bf23SRichard Henderson return -EINVAL; 121*87e9bf23SRichard Henderson } 122*87e9bf23SRichard Henderson 123*87e9bf23SRichard Henderson /* 124*87e9bf23SRichard Henderson * Write PR_MTE_TCF to SCTLR_EL1[TCF0]. 125*87e9bf23SRichard Henderson * Note that the syscall values are consistent with hw. 126*87e9bf23SRichard Henderson */ 127*87e9bf23SRichard Henderson env->cp15.sctlr_el[1] = 128*87e9bf23SRichard Henderson deposit64(env->cp15.sctlr_el[1], 38, 2, arg2 >> PR_MTE_TCF_SHIFT); 129*87e9bf23SRichard Henderson 130*87e9bf23SRichard Henderson /* 131*87e9bf23SRichard Henderson * Write PR_MTE_TAG to GCR_EL1[Exclude]. 132*87e9bf23SRichard Henderson * Note that the syscall uses an include mask, 133*87e9bf23SRichard Henderson * and hardware uses an exclude mask -- invert. 134*87e9bf23SRichard Henderson */ 135*87e9bf23SRichard Henderson env->cp15.gcr_el1 = 136*87e9bf23SRichard Henderson deposit64(env->cp15.gcr_el1, 0, 16, ~arg2 >> PR_MTE_TAG_SHIFT); 137*87e9bf23SRichard Henderson arm_rebuild_hflags(env); 138*87e9bf23SRichard Henderson } 139*87e9bf23SRichard Henderson return 0; 140*87e9bf23SRichard Henderson } 141*87e9bf23SRichard Henderson #define do_prctl_set_tagged_addr_ctrl do_prctl_set_tagged_addr_ctrl 142*87e9bf23SRichard Henderson 143*87e9bf23SRichard Henderson static abi_long do_prctl_get_tagged_addr_ctrl(CPUArchState *env) 144*87e9bf23SRichard Henderson { 145*87e9bf23SRichard Henderson ARMCPU *cpu = env_archcpu(env); 146*87e9bf23SRichard Henderson abi_long ret = 0; 147*87e9bf23SRichard Henderson 148*87e9bf23SRichard Henderson if (env->tagged_addr_enable) { 149*87e9bf23SRichard Henderson ret |= PR_TAGGED_ADDR_ENABLE; 150*87e9bf23SRichard Henderson } 151*87e9bf23SRichard Henderson if (cpu_isar_feature(aa64_mte, cpu)) { 152*87e9bf23SRichard Henderson /* See do_prctl_set_tagged_addr_ctrl. */ 153*87e9bf23SRichard Henderson ret |= extract64(env->cp15.sctlr_el[1], 38, 2) << PR_MTE_TCF_SHIFT; 154*87e9bf23SRichard Henderson ret = deposit64(ret, PR_MTE_TAG_SHIFT, 16, ~env->cp15.gcr_el1); 155*87e9bf23SRichard Henderson } 156*87e9bf23SRichard Henderson return ret; 157*87e9bf23SRichard Henderson } 158*87e9bf23SRichard Henderson #define do_prctl_get_tagged_addr_ctrl do_prctl_get_tagged_addr_ctrl 159*87e9bf23SRichard Henderson 160*87e9bf23SRichard Henderson #endif /* AARCH64_TARGET_PRCTL_H */ 161