| /* SPDX-License-Identifier: GPL-2.0 */ |
| /* arch/arm64/include/asm/kvm_ptrauth.h: Guest/host ptrauth save/restore |
| * Copyright 2019 Arm Limited |
| * Authors: Mark Rutland <mark.rutland@arm.com> |
| * Amit Daniel Kachhap <amit.kachhap@arm.com> |
| */ |
| |
| #ifndef __ASM_KVM_PTRAUTH_H |
| #define __ASM_KVM_PTRAUTH_H |
| |
| #ifdef __ASSEMBLY__ |
| |
| #include <asm/sysreg.h> |
| |
| #ifdef CONFIG_ARM64_PTR_AUTH |
| |
| #define PTRAUTH_REG_OFFSET(x) (x - CPU_APIAKEYLO_EL1) |
| |
| /* |
| * CPU_AP*_EL1 values exceed immediate offset range (512) for stp |
| * instruction so below macros takes CPU_APIAKEYLO_EL1 as base and |
| * calculates the offset of the keys from this base to avoid an extra add |
| * instruction. These macros assumes the keys offsets follow the order of |
| * the sysreg enum in kvm_host.h. |
| */ |
| .macro ptrauth_save_state base, reg1, reg2 |
| mrs_s \reg1, SYS_APIAKEYLO_EL1 |
| mrs_s \reg2, SYS_APIAKEYHI_EL1 |
| stp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APIAKEYLO_EL1)] |
| mrs_s \reg1, SYS_APIBKEYLO_EL1 |
| mrs_s \reg2, SYS_APIBKEYHI_EL1 |
| stp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APIBKEYLO_EL1)] |
| mrs_s \reg1, SYS_APDAKEYLO_EL1 |
| mrs_s \reg2, SYS_APDAKEYHI_EL1 |
| stp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APDAKEYLO_EL1)] |
| mrs_s \reg1, SYS_APDBKEYLO_EL1 |
| mrs_s \reg2, SYS_APDBKEYHI_EL1 |
| stp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APDBKEYLO_EL1)] |
| mrs_s \reg1, SYS_APGAKEYLO_EL1 |
| mrs_s \reg2, SYS_APGAKEYHI_EL1 |
| stp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APGAKEYLO_EL1)] |
| .endm |
| |
| .macro ptrauth_restore_state base, reg1, reg2 |
| ldp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APIAKEYLO_EL1)] |
| msr_s SYS_APIAKEYLO_EL1, \reg1 |
| msr_s SYS_APIAKEYHI_EL1, \reg2 |
| ldp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APIBKEYLO_EL1)] |
| msr_s SYS_APIBKEYLO_EL1, \reg1 |
| msr_s SYS_APIBKEYHI_EL1, \reg2 |
| ldp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APDAKEYLO_EL1)] |
| msr_s SYS_APDAKEYLO_EL1, \reg1 |
| msr_s SYS_APDAKEYHI_EL1, \reg2 |
| ldp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APDBKEYLO_EL1)] |
| msr_s SYS_APDBKEYLO_EL1, \reg1 |
| msr_s SYS_APDBKEYHI_EL1, \reg2 |
| ldp \reg1, \reg2, [\base, #PTRAUTH_REG_OFFSET(CPU_APGAKEYLO_EL1)] |
| msr_s SYS_APGAKEYLO_EL1, \reg1 |
| msr_s SYS_APGAKEYHI_EL1, \reg2 |
| .endm |
| |
| /* |
| * Both ptrauth_switch_to_guest and ptrauth_switch_to_hyp macros will |
| * check for the presence ARM64_HAS_ADDRESS_AUTH, which is defined as |
| * (ARM64_HAS_ADDRESS_AUTH_ARCH || ARM64_HAS_ADDRESS_AUTH_IMP_DEF) and |
| * then proceed ahead with the save/restore of Pointer Authentication |
| * key registers if enabled for the guest. |
| */ |
| .macro ptrauth_switch_to_guest g_ctxt, reg1, reg2, reg3 |
| alternative_if_not ARM64_HAS_ADDRESS_AUTH |
| b .L__skip_switch\@ |
| alternative_else_nop_endif |
| mrs \reg1, hcr_el2 |
| and \reg1, \reg1, #(HCR_API | HCR_APK) |
| cbz \reg1, .L__skip_switch\@ |
| add \reg1, \g_ctxt, #CPU_APIAKEYLO_EL1 |
| ptrauth_restore_state \reg1, \reg2, \reg3 |
| .L__skip_switch\@: |
| .endm |
| |
| .macro ptrauth_switch_to_hyp g_ctxt, h_ctxt, reg1, reg2, reg3 |
| alternative_if_not ARM64_HAS_ADDRESS_AUTH |
| b .L__skip_switch\@ |
| alternative_else_nop_endif |
| mrs \reg1, hcr_el2 |
| and \reg1, \reg1, #(HCR_API | HCR_APK) |
| cbz \reg1, .L__skip_switch\@ |
| add \reg1, \g_ctxt, #CPU_APIAKEYLO_EL1 |
| ptrauth_save_state \reg1, \reg2, \reg3 |
| add \reg1, \h_ctxt, #CPU_APIAKEYLO_EL1 |
| ptrauth_restore_state \reg1, \reg2, \reg3 |
| isb |
| .L__skip_switch\@: |
| .endm |
| |
| #else /* !CONFIG_ARM64_PTR_AUTH */ |
| .macro ptrauth_switch_to_guest g_ctxt, reg1, reg2, reg3 |
| .endm |
| .macro ptrauth_switch_to_hyp g_ctxt, h_ctxt, reg1, reg2, reg3 |
| .endm |
| #endif /* CONFIG_ARM64_PTR_AUTH */ |
| |
| #else /* !__ASSEMBLY */ |
| |
| #define __ptrauth_save_key(ctxt, key) \ |
| do { \ |
| u64 __val; \ |
| __val = read_sysreg_s(SYS_ ## key ## KEYLO_EL1); \ |
| ctxt_sys_reg(ctxt, key ## KEYLO_EL1) = __val; \ |
| __val = read_sysreg_s(SYS_ ## key ## KEYHI_EL1); \ |
| ctxt_sys_reg(ctxt, key ## KEYHI_EL1) = __val; \ |
| } while(0) |
| |
| #define ptrauth_save_keys(ctxt) \ |
| do { \ |
| __ptrauth_save_key(ctxt, APIA); \ |
| __ptrauth_save_key(ctxt, APIB); \ |
| __ptrauth_save_key(ctxt, APDA); \ |
| __ptrauth_save_key(ctxt, APDB); \ |
| __ptrauth_save_key(ctxt, APGA); \ |
| } while(0) |
| |
| #endif /* __ASSEMBLY__ */ |
| #endif /* __ASM_KVM_PTRAUTH_H */ |