1 /* SPDX-License-Identifier: GPL-2.0-only */
3 * sev_verify_cbit.S - Code for verification of the C-bit position reported
4 * by the Hypervisor when running with SEV enabled.
6 * Copyright (c) 2020 Joerg Roedel (jroedel@suse.de)
8 * sev_verify_cbit() is called before switching to a new long-mode page-table
11 * Verify that the C-bit position is correct by writing a random value to
12 * an encrypted memory location while on the current page-table. Then it
13 * switches to the new page-table to verify the memory content is still the
14 * same. After that it switches back to the current page-table and when the
15 * check succeeded it returns. If the check failed the code invalidates the
16 * stack pointer and goes into a hlt loop. The stack-pointer is invalidated to
17 * make sure no interrupt or exception can get the CPU out of the hlt loop.
19 * New page-table pointer is expected in %rdi (first parameter)
22 SYM_FUNC_START(sev_verify_cbit)
23 #ifdef CONFIG_AMD_MEM_ENCRYPT
24 /* First check if a C-bit was detected */
25 movq sme_me_mask(%rip), %rsi
29 /* sme_me_mask != 0 could mean SME or SEV - Check also for SEV */
30 movq sev_status(%rip), %rsi
34 /* Save CR4 in %rsi */
37 /* Disable Global Pages */
39 andq $(~X86_CR4_PGE), %rdx
43 * Verified that running under SEV - now get a random value using
44 * RDRAND. This instruction is mandatory when running as an SEV guest.
46 * Don't bail out of the loop if RDRAND returns errors. It is better to
47 * prevent forward progress than to work with a non-random value here.
52 /* Store value to memory and keep it in %rdx */
53 movq %rdx, sev_check_data(%rip)
55 /* Backup current %cr3 value to restore it later */
58 /* Switch to new %cr3 - This might unmap the stack */
62 * Compare value in %rdx with memory location. If C-bit is incorrect
63 * this would read the encrypted data and make the check fail.
65 cmpq %rdx, sev_check_data(%rip)
67 /* Restore old %cr3 */
70 /* Restore previous CR4 */
73 /* Check CMPQ result */
77 * The check failed, prevent any forward progress to prevent ROP
78 * attacks, invalidate the stack and go into a hlt loop.
86 /* Return page-table pointer */
89 SYM_FUNC_END(sev_verify_cbit)