mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-09 14:43:16 +00:00
8c29f01654
The hypervisor can enable various new features (SEV_FEATURES[1:63]) and start a SNP guest. Some of these features need guest side implementation. If any of these features are enabled without it, the behavior of the SNP guest will be undefined. It may fail booting in a non-obvious way making it difficult to debug. Instead of allowing the guest to continue and have it fail randomly later, detect this early and fail gracefully. The SEV_STATUS MSR indicates features which the hypervisor has enabled. While booting, SNP guests should ascertain that all the enabled features have guest side implementation. In case a feature is not implemented in the guest, the guest terminates booting with GHCB protocol Non-Automatic Exit(NAE) termination request event, see "SEV-ES Guest-Hypervisor Communication Block Standardization" document (currently at https://developer.amd.com/wp-content/resources/56421.pdf), section "Termination Request". Populate SW_EXITINFO2 with mask of unsupported features that the hypervisor can easily report to the user. More details in the AMD64 APM Vol 2, Section "SEV_STATUS MSR". [ bp: - Massage. - Move snp_check_features() call to C code. Note: the CC:stable@ aspect here is to be able to protect older, stable kernels when running on newer hypervisors. Or not "running" but fail reliably and in a well-defined manner instead of randomly. ] Fixes: cbd3d4f7c4e5 ("x86/sev: Check SEV-SNP features support") Signed-off-by: Nikunj A Dadhania <nikunj@amd.com> Signed-off-by: Borislav Petkov (AMD) <bp@alien8.de> Reviewed-by: Tom Lendacky <thomas.lendacky@amd.com> Cc: <stable@kernel.org> Link: https://lore.kernel.org/r/20230118061943.534309-1-nikunj@amd.com
134 lines
6.5 KiB
ReStructuredText
134 lines
6.5 KiB
ReStructuredText
.. SPDX-License-Identifier: GPL-2.0
|
|
|
|
=====================
|
|
AMD Memory Encryption
|
|
=====================
|
|
|
|
Secure Memory Encryption (SME) and Secure Encrypted Virtualization (SEV) are
|
|
features found on AMD processors.
|
|
|
|
SME provides the ability to mark individual pages of memory as encrypted using
|
|
the standard x86 page tables. A page that is marked encrypted will be
|
|
automatically decrypted when read from DRAM and encrypted when written to
|
|
DRAM. SME can therefore be used to protect the contents of DRAM from physical
|
|
attacks on the system.
|
|
|
|
SEV enables running encrypted virtual machines (VMs) in which the code and data
|
|
of the guest VM are secured so that a decrypted version is available only
|
|
within the VM itself. SEV guest VMs have the concept of private and shared
|
|
memory. Private memory is encrypted with the guest-specific key, while shared
|
|
memory may be encrypted with hypervisor key. When SME is enabled, the hypervisor
|
|
key is the same key which is used in SME.
|
|
|
|
A page is encrypted when a page table entry has the encryption bit set (see
|
|
below on how to determine its position). The encryption bit can also be
|
|
specified in the cr3 register, allowing the PGD table to be encrypted. Each
|
|
successive level of page tables can also be encrypted by setting the encryption
|
|
bit in the page table entry that points to the next table. This allows the full
|
|
page table hierarchy to be encrypted. Note, this means that just because the
|
|
encryption bit is set in cr3, doesn't imply the full hierarchy is encrypted.
|
|
Each page table entry in the hierarchy needs to have the encryption bit set to
|
|
achieve that. So, theoretically, you could have the encryption bit set in cr3
|
|
so that the PGD is encrypted, but not set the encryption bit in the PGD entry
|
|
for a PUD which results in the PUD pointed to by that entry to not be
|
|
encrypted.
|
|
|
|
When SEV is enabled, instruction pages and guest page tables are always treated
|
|
as private. All the DMA operations inside the guest must be performed on shared
|
|
memory. Since the memory encryption bit is controlled by the guest OS when it
|
|
is operating in 64-bit or 32-bit PAE mode, in all other modes the SEV hardware
|
|
forces the memory encryption bit to 1.
|
|
|
|
Support for SME and SEV can be determined through the CPUID instruction. The
|
|
CPUID function 0x8000001f reports information related to SME::
|
|
|
|
0x8000001f[eax]:
|
|
Bit[0] indicates support for SME
|
|
Bit[1] indicates support for SEV
|
|
0x8000001f[ebx]:
|
|
Bits[5:0] pagetable bit number used to activate memory
|
|
encryption
|
|
Bits[11:6] reduction in physical address space, in bits, when
|
|
memory encryption is enabled (this only affects
|
|
system physical addresses, not guest physical
|
|
addresses)
|
|
|
|
If support for SME is present, MSR 0xc00100010 (MSR_AMD64_SYSCFG) can be used to
|
|
determine if SME is enabled and/or to enable memory encryption::
|
|
|
|
0xc0010010:
|
|
Bit[23] 0 = memory encryption features are disabled
|
|
1 = memory encryption features are enabled
|
|
|
|
If SEV is supported, MSR 0xc0010131 (MSR_AMD64_SEV) can be used to determine if
|
|
SEV is active::
|
|
|
|
0xc0010131:
|
|
Bit[0] 0 = memory encryption is not active
|
|
1 = memory encryption is active
|
|
|
|
Linux relies on BIOS to set this bit if BIOS has determined that the reduction
|
|
in the physical address space as a result of enabling memory encryption (see
|
|
CPUID information above) will not conflict with the address space resource
|
|
requirements for the system. If this bit is not set upon Linux startup then
|
|
Linux itself will not set it and memory encryption will not be possible.
|
|
|
|
The state of SME in the Linux kernel can be documented as follows:
|
|
|
|
- Supported:
|
|
The CPU supports SME (determined through CPUID instruction).
|
|
|
|
- Enabled:
|
|
Supported and bit 23 of MSR_AMD64_SYSCFG is set.
|
|
|
|
- Active:
|
|
Supported, Enabled and the Linux kernel is actively applying
|
|
the encryption bit to page table entries (the SME mask in the
|
|
kernel is non-zero).
|
|
|
|
SME can also be enabled and activated in the BIOS. If SME is enabled and
|
|
activated in the BIOS, then all memory accesses will be encrypted and it will
|
|
not be necessary to activate the Linux memory encryption support. If the BIOS
|
|
merely enables SME (sets bit 23 of the MSR_AMD64_SYSCFG), then Linux can activate
|
|
memory encryption by default (CONFIG_AMD_MEM_ENCRYPT_ACTIVE_BY_DEFAULT=y) or
|
|
by supplying mem_encrypt=on on the kernel command line. However, if BIOS does
|
|
not enable SME, then Linux will not be able to activate memory encryption, even
|
|
if configured to do so by default or the mem_encrypt=on command line parameter
|
|
is specified.
|
|
|
|
Secure Nested Paging (SNP)
|
|
==========================
|
|
|
|
SEV-SNP introduces new features (SEV_FEATURES[1:63]) which can be enabled
|
|
by the hypervisor for security enhancements. Some of these features need
|
|
guest side implementation to function correctly. The below table lists the
|
|
expected guest behavior with various possible scenarios of guest/hypervisor
|
|
SNP feature support.
|
|
|
|
+-----------------+---------------+---------------+------------------+
|
|
| Feature Enabled | Guest needs | Guest has | Guest boot |
|
|
| by the HV | implementation| implementation| behaviour |
|
|
+=================+===============+===============+==================+
|
|
| No | No | No | Boot |
|
|
| | | | |
|
|
+-----------------+---------------+---------------+------------------+
|
|
| No | Yes | No | Boot |
|
|
| | | | |
|
|
+-----------------+---------------+---------------+------------------+
|
|
| No | Yes | Yes | Boot |
|
|
| | | | |
|
|
+-----------------+---------------+---------------+------------------+
|
|
| Yes | No | No | Boot with |
|
|
| | | | feature enabled |
|
|
+-----------------+---------------+---------------+------------------+
|
|
| Yes | Yes | No | Graceful boot |
|
|
| | | | failure |
|
|
+-----------------+---------------+---------------+------------------+
|
|
| Yes | Yes | Yes | Boot with |
|
|
| | | | feature enabled |
|
|
+-----------------+---------------+---------------+------------------+
|
|
|
|
More details in AMD64 APM[1] Vol 2: 15.34.10 SEV_STATUS MSR
|
|
|
|
[1] https://www.amd.com/system/files/TechDocs/40332.pdf
|