1/* SPDX-License-Identifier: GPL-2.0-only */ 2/* 3 * Copyright (C) 2020 - Google Inc 4 * Author: Andrew Scull <ascull@google.com> 5 */ 6 7#include <linux/linkage.h> 8 9#include <asm/assembler.h> 10#include <asm/kvm_asm.h> 11#include <asm/kvm_mmu.h> 12 13 .text 14 15SYM_FUNC_START(__host_exit) 16 stp x0, x1, [sp, #-16]! 17 18 get_host_ctxt x0, x1 19 20 /* Store the host regs x2 and x3 */ 21 stp x2, x3, [x0, #CPU_XREG_OFFSET(2)] 22 23 /* Retrieve the host regs x0-x1 from the stack */ 24 ldp x2, x3, [sp], #16 // x0, x1 25 26 /* Store the host regs x0-x1 and x4-x17 */ 27 stp x2, x3, [x0, #CPU_XREG_OFFSET(0)] 28 stp x4, x5, [x0, #CPU_XREG_OFFSET(4)] 29 stp x6, x7, [x0, #CPU_XREG_OFFSET(6)] 30 stp x8, x9, [x0, #CPU_XREG_OFFSET(8)] 31 stp x10, x11, [x0, #CPU_XREG_OFFSET(10)] 32 stp x12, x13, [x0, #CPU_XREG_OFFSET(12)] 33 stp x14, x15, [x0, #CPU_XREG_OFFSET(14)] 34 stp x16, x17, [x0, #CPU_XREG_OFFSET(16)] 35 36 /* Store the host regs x18-x29, lr */ 37 save_callee_saved_regs x0 38 39 /* Save the host context pointer in x29 across the function call */ 40 mov x29, x0 41 bl handle_trap 42 43 /* Restore host regs x0-x17 */ 44 ldp x0, x1, [x29, #CPU_XREG_OFFSET(0)] 45 ldp x2, x3, [x29, #CPU_XREG_OFFSET(2)] 46 ldp x4, x5, [x29, #CPU_XREG_OFFSET(4)] 47 ldp x6, x7, [x29, #CPU_XREG_OFFSET(6)] 48 49 /* x0-7 are use for panic arguments */ 50__host_enter_for_panic: 51 ldp x8, x9, [x29, #CPU_XREG_OFFSET(8)] 52 ldp x10, x11, [x29, #CPU_XREG_OFFSET(10)] 53 ldp x12, x13, [x29, #CPU_XREG_OFFSET(12)] 54 ldp x14, x15, [x29, #CPU_XREG_OFFSET(14)] 55 ldp x16, x17, [x29, #CPU_XREG_OFFSET(16)] 56 57 /* Restore host regs x18-x29, lr */ 58 restore_callee_saved_regs x29 59 60 /* Do not touch any register after this! */ 61__host_enter_without_restoring: 62 eret 63 sb 64SYM_FUNC_END(__host_exit) 65 66/* 67 * void __noreturn __hyp_do_panic(bool restore_host, u64 spsr, u64 elr, u64 par); 68 */ 69SYM_FUNC_START(__hyp_do_panic) 70 /* Load the format arguments into x1-7 */ 71 mov x6, x3 72 get_vcpu_ptr x7, x3 73 74 mrs x3, esr_el2 75 mrs x4, far_el2 76 mrs x5, hpfar_el2 77 78 /* Prepare and exit to the host's panic funciton. */ 79 mov lr, #(PSR_F_BIT | PSR_I_BIT | PSR_A_BIT | PSR_D_BIT |\ 80 PSR_MODE_EL1h) 81 msr spsr_el2, lr 82 ldr lr, =panic 83 msr elr_el2, lr 84 85 /* 86 * Set the panic format string and enter the host, conditionally 87 * restoring the host context. 88 */ 89 cmp x0, xzr 90 ldr x0, =__hyp_panic_string 91 b.eq __host_enter_without_restoring 92 b __host_enter_for_panic 93SYM_FUNC_END(__hyp_do_panic) 94 95.macro host_el1_sync_vect 96 .align 7 97.L__vect_start\@: 98 stp x0, x1, [sp, #-16]! 99 mrs x0, esr_el2 100 lsr x0, x0, #ESR_ELx_EC_SHIFT 101 cmp x0, #ESR_ELx_EC_HVC64 102 ldp x0, x1, [sp], #16 103 b.ne __host_exit 104 105 /* Check for a stub HVC call */ 106 cmp x0, #HVC_STUB_HCALL_NR 107 b.hs __host_exit 108 109 /* 110 * Compute the idmap address of __kvm_handle_stub_hvc and 111 * jump there. Since we use kimage_voffset, do not use the 112 * HYP VA for __kvm_handle_stub_hvc, but the kernel VA instead 113 * (by loading it from the constant pool). 114 * 115 * Preserve x0-x4, which may contain stub parameters. 116 */ 117 ldr x5, =__kvm_handle_stub_hvc 118 ldr_l x6, kimage_voffset 119 120 /* x5 = __pa(x5) */ 121 sub x5, x5, x6 122 br x5 123.L__vect_end\@: 124.if ((.L__vect_end\@ - .L__vect_start\@) > 0x80) 125 .error "host_el1_sync_vect larger than vector entry" 126.endif 127.endm 128 129.macro invalid_host_el2_vect 130 .align 7 131 /* If a guest is loaded, panic out of it. */ 132 stp x0, x1, [sp, #-16]! 133 get_loaded_vcpu x0, x1 134 cbnz x0, __guest_exit_panic 135 add sp, sp, #16 136 137 /* 138 * The panic may not be clean if the exception is taken before the host 139 * context has been saved by __host_exit or after the hyp context has 140 * been partially clobbered by __host_enter. 141 */ 142 b hyp_panic 143.endm 144 145.macro invalid_host_el1_vect 146 .align 7 147 mov x0, xzr /* restore_host = false */ 148 mrs x1, spsr_el2 149 mrs x2, elr_el2 150 mrs x3, par_el1 151 b __hyp_do_panic 152.endm 153 154/* 155 * The host vector does not use an ESB instruction in order to avoid consuming 156 * SErrors that should only be consumed by the host. Guest entry is deferred by 157 * __guest_enter if there are any pending asynchronous exceptions so hyp will 158 * always return to the host without having consumerd host SErrors. 159 * 160 * CONFIG_KVM_INDIRECT_VECTORS is not applied to the host vectors because the 161 * host knows about the EL2 vectors already, and there is no point in hiding 162 * them. 163 */ 164 .align 11 165SYM_CODE_START(__kvm_hyp_host_vector) 166 invalid_host_el2_vect // Synchronous EL2t 167 invalid_host_el2_vect // IRQ EL2t 168 invalid_host_el2_vect // FIQ EL2t 169 invalid_host_el2_vect // Error EL2t 170 171 invalid_host_el2_vect // Synchronous EL2h 172 invalid_host_el2_vect // IRQ EL2h 173 invalid_host_el2_vect // FIQ EL2h 174 invalid_host_el2_vect // Error EL2h 175 176 host_el1_sync_vect // Synchronous 64-bit EL1 177 invalid_host_el1_vect // IRQ 64-bit EL1 178 invalid_host_el1_vect // FIQ 64-bit EL1 179 invalid_host_el1_vect // Error 64-bit EL1 180 181 invalid_host_el1_vect // Synchronous 32-bit EL1 182 invalid_host_el1_vect // IRQ 32-bit EL1 183 invalid_host_el1_vect // FIQ 32-bit EL1 184 invalid_host_el1_vect // Error 32-bit EL1 185SYM_CODE_END(__kvm_hyp_host_vector) 186