2019-06-04 16:11:33 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-only */
|
2017-10-20 22:30:54 +08:00
|
|
|
/*
|
|
|
|
* AMD Memory Encryption Support
|
|
|
|
*
|
|
|
|
* Copyright (C) 2017 Advanced Micro Devices, Inc.
|
|
|
|
*
|
|
|
|
* Author: Tom Lendacky <thomas.lendacky@amd.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/linkage.h>
|
|
|
|
|
|
|
|
#include <asm/processor-flags.h>
|
|
|
|
#include <asm/msr.h>
|
|
|
|
#include <asm/asm-offsets.h>
|
|
|
|
|
|
|
|
.text
|
|
|
|
.code32
|
2019-10-11 19:51:04 +08:00
|
|
|
SYM_FUNC_START(get_sev_encryption_bit)
|
2017-10-20 22:30:54 +08:00
|
|
|
xor %eax, %eax
|
|
|
|
|
|
|
|
#ifdef CONFIG_AMD_MEM_ENCRYPT
|
|
|
|
push %ebx
|
|
|
|
push %ecx
|
|
|
|
push %edx
|
|
|
|
|
|
|
|
/* Check if running under a hypervisor */
|
|
|
|
movl $1, %eax
|
|
|
|
cpuid
|
|
|
|
bt $31, %ecx /* Check the hypervisor bit */
|
|
|
|
jnc .Lno_sev
|
|
|
|
|
|
|
|
movl $0x80000000, %eax /* CPUID to check the highest leaf */
|
|
|
|
cpuid
|
|
|
|
cmpl $0x8000001f, %eax /* See if 0x8000001f is available */
|
|
|
|
jb .Lno_sev
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check for the SEV feature:
|
|
|
|
* CPUID Fn8000_001F[EAX] - Bit 1
|
|
|
|
* CPUID Fn8000_001F[EBX] - Bits 5:0
|
|
|
|
* Pagetable bit position used to indicate encryption
|
|
|
|
*/
|
|
|
|
movl $0x8000001f, %eax
|
|
|
|
cpuid
|
|
|
|
bt $1, %eax /* Check if SEV is available */
|
|
|
|
jnc .Lno_sev
|
|
|
|
|
|
|
|
movl $MSR_AMD64_SEV, %ecx /* Read the SEV MSR */
|
|
|
|
rdmsr
|
|
|
|
bt $MSR_AMD64_SEV_ENABLED_BIT, %eax /* Check if SEV is active */
|
|
|
|
jnc .Lno_sev
|
|
|
|
|
|
|
|
movl %ebx, %eax
|
|
|
|
andl $0x3f, %eax /* Return the encryption bit location */
|
|
|
|
jmp .Lsev_exit
|
|
|
|
|
|
|
|
.Lno_sev:
|
|
|
|
xor %eax, %eax
|
|
|
|
|
|
|
|
.Lsev_exit:
|
|
|
|
pop %edx
|
|
|
|
pop %ecx
|
|
|
|
pop %ebx
|
|
|
|
|
|
|
|
#endif /* CONFIG_AMD_MEM_ENCRYPT */
|
|
|
|
|
|
|
|
ret
|
2019-10-11 19:51:04 +08:00
|
|
|
SYM_FUNC_END(get_sev_encryption_bit)
|
2017-10-20 22:30:54 +08:00
|
|
|
|
|
|
|
.code64
|
2019-10-11 19:51:04 +08:00
|
|
|
SYM_FUNC_START(set_sev_encryption_mask)
|
2017-10-20 22:30:54 +08:00
|
|
|
#ifdef CONFIG_AMD_MEM_ENCRYPT
|
|
|
|
push %rbp
|
|
|
|
push %rdx
|
|
|
|
|
|
|
|
movq %rsp, %rbp /* Save current stack pointer */
|
|
|
|
|
|
|
|
call get_sev_encryption_bit /* Get the encryption bit position */
|
|
|
|
testl %eax, %eax
|
|
|
|
jz .Lno_sev_mask
|
|
|
|
|
2018-03-28 06:07:11 +08:00
|
|
|
bts %rax, sme_me_mask(%rip) /* Create the encryption mask */
|
2017-10-20 22:30:54 +08:00
|
|
|
|
|
|
|
.Lno_sev_mask:
|
|
|
|
movq %rbp, %rsp /* Restore original stack pointer */
|
|
|
|
|
|
|
|
pop %rdx
|
|
|
|
pop %rbp
|
|
|
|
#endif
|
|
|
|
|
2018-03-28 06:07:11 +08:00
|
|
|
xor %rax, %rax
|
2017-10-20 22:30:54 +08:00
|
|
|
ret
|
2019-10-11 19:51:04 +08:00
|
|
|
SYM_FUNC_END(set_sev_encryption_mask)
|
2017-10-20 22:30:54 +08:00
|
|
|
|
|
|
|
.data
|
2018-03-28 06:07:11 +08:00
|
|
|
|
|
|
|
#ifdef CONFIG_AMD_MEM_ENCRYPT
|
|
|
|
.balign 8
|
2019-10-11 19:50:52 +08:00
|
|
|
SYM_DATA(sme_me_mask, .quad 0)
|
2018-03-28 06:07:11 +08:00
|
|
|
#endif
|