xref: /openbmc/qemu/linux-user/aarch64/target_prctl.h (revision 87e9bf23236d3c9da84f2b6164e06be3ecfd45e0)
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