;------------------------------------------------------------------------------ ; ; Copyright (c) 2021, AMD Inc. All rights reserved.
; SPDX-License-Identifier: BSD-2-Clause-Patent ; ; Module Name: ; ; AmdSev.nasm ; ; Abstract: ; ; This provides helper used by the MpFunc.nasm. If AMD SEV-ES is active ; then helpers perform the additional setups (such as GHCB). ; ;------------------------------------------------------------------------------- %define SIZE_4KB 0x1000 RegisterGhcbGpa: ; ; Register GHCB GPA when SEV-SNP is enabled ; lea edi, [esi + MP_CPU_EXCHANGE_INFO_FIELD (SevSnpIsEnabled)] cmp byte [edi], 1 ; SevSnpIsEnabled jne RegisterGhcbGpaDone ; Save the rdi and rsi to used for later comparison push rdi push rsi mov edi, eax mov esi, edx or eax, 18 ; Ghcb registration request wrmsr rep vmmcall rdmsr mov r12, rax and r12, 0fffh cmp r12, 19 ; Ghcb registration response jne GhcbGpaRegisterFailure ; Verify that GPA is not changed and eax, 0fffff000h cmp edi, eax jne GhcbGpaRegisterFailure cmp esi, edx jne GhcbGpaRegisterFailure pop rsi pop rdi jmp RegisterGhcbGpaDone ; ; Request the guest termination ; GhcbGpaRegisterFailure: xor edx, edx mov eax, 256 ; GHCB terminate wrmsr rep vmmcall ; We should not return from the above terminate request, but if we do ; then enter into the hlt loop. DoHltLoop: cli hlt jmp DoHltLoop RegisterGhcbGpaDone: OneTimeCallRet RegisterGhcbGpa ; ; The function checks whether SEV-ES is enabled, if enabled ; then setup the GHCB page. ; SevEsSetupGhcb: lea edi, [esi + MP_CPU_EXCHANGE_INFO_FIELD (SevEsIsEnabled)] cmp byte [edi], 1 ; SevEsIsEnabled jne SevEsSetupGhcbExit ; ; program GHCB ; Each page after the GHCB is a per-CPU page, so the calculation programs ; a GHCB to be every 8KB. ; mov eax, SIZE_4KB shl eax, 1 ; EAX = SIZE_4K * 2 mov ecx, ebx mul ecx ; EAX = SIZE_4K * 2 * CpuNumber mov edi, esi add edi, MP_CPU_EXCHANGE_INFO_FIELD (GhcbBase) add rax, qword [edi] mov rdx, rax shr rdx, 32 mov rcx, 0xc0010130 OneTimeCall RegisterGhcbGpa wrmsr SevEsSetupGhcbExit: OneTimeCallRet SevEsSetupGhcb ; ; The function checks whether SEV-ES is enabled, if enabled, use ; the GHCB ; SevEsGetApicId: lea edi, [esi + MP_CPU_EXCHANGE_INFO_FIELD (SevEsIsEnabled)] cmp byte [edi], 1 ; SevEsIsEnabled jne SevEsGetApicIdExit ; ; Since we don't have a stack yet, we can't take a #VC ; exception. Use the GHCB protocol to perform the CPUID ; calls. ; mov rcx, 0xc0010130 rdmsr shl rdx, 32 or rax, rdx mov rdi, rax ; RDI now holds the original GHCB GPA ; ; For SEV-SNP, the recommended handling for getting the x2APIC ID ; would be to use the SNP CPUID table to fetch CPUID.00H:EAX and ; CPUID:0BH:EBX[15:0] instead of the GHCB MSR protocol vmgexits ; below. ; ; To avoid the unecessary ugliness to accomplish that here, the BSP ; has performed these checks in advance (where #VC handler handles ; the CPUID table lookups automatically) and cached them in a flag ; so those checks can be skipped here. ; mov eax, [esi + MP_CPU_EXCHANGE_INFO_FIELD (SevSnpIsEnabled)] cmp al, 1 jne CheckExtTopoAvail ; ; Even with SEV-SNP, the actual x2APIC ID in CPUID.0BH:EDX ; fetched from the hypervisor the same way SEV-ES does it. ; mov eax, [esi + MP_CPU_EXCHANGE_INFO_FIELD (ExtTopoAvail)] cmp al, 1 je GetApicIdSevEs ; The 8-bit APIC ID fallback is also the same as with SEV-ES jmp NoX2ApicSevEs CheckExtTopoAvail: mov rdx, 0 ; CPUID function 0 mov rax, 0 ; RAX register requested or rax, 4 wrmsr rep vmmcall rdmsr cmp edx, 0bh jb NoX2ApicSevEs ; CPUID level below CPUID_EXTENDED_TOPOLOGY mov rdx, 0bh ; CPUID function 0x0b mov rax, 040000000h ; RBX register requested or rax, 4 wrmsr rep vmmcall rdmsr test edx, 0ffffh jz NoX2ApicSevEs ; CPUID.0BH:EBX[15:0] is zero GetApicIdSevEs: mov rdx, 0bh ; CPUID function 0x0b mov rax, 0c0000000h ; RDX register requested or rax, 4 wrmsr rep vmmcall rdmsr ; Processor is x2APIC capable; 32-bit x2APIC ID is now in EDX jmp RestoreGhcb NoX2ApicSevEs: ; Processor is not x2APIC capable, so get 8-bit APIC ID mov rdx, 1 ; CPUID function 1 mov rax, 040000000h ; RBX register requested or rax, 4 wrmsr rep vmmcall rdmsr shr edx, 24 RestoreGhcb: mov rbx, rdx ; Save x2APIC/APIC ID mov rdx, rdi ; RDI holds the saved GHCB GPA shr rdx, 32 mov eax, edi wrmsr mov rdx, rbx ; x2APIC ID or APIC ID is in EDX jmp GetProcessorNumber SevEsGetApicIdExit: OneTimeCallRet SevEsGetApicId