OvmfPkg/BaseMemEncryptTdxLib: Handle retry result of MapGPA

REF: https://bugzilla.tianocore.org/show_bug.cgi?id=4572

According to section 3.2 of the [GHCI] document, if the return status
of MapGPA is "TDG.VP.VMCALL_RETRY", TD must retry this operation for the
pages in the region starting at the GPA specified in R11.

In this patch, when a retry state is detected, TDVF needs to retry the
mapping with the specified address from the output results of TdVmCall.

Reference:
[GHCI]: TDX Guest-Host-Communication Interface v1.0
https://cdrdv2.intel.com/v1/dl/getContent/726790

Cc: Erdem Aktas <erdemaktas@google.com>
Cc: James Bottomley <jejb@linux.ibm.com>
Cc: Jiewen Yao <jiewen.yao@intel.com>
Cc: Min Xu <min.m.xu@intel.com>
Cc: Tom Lendacky <thomas.lendacky@amd.com>
Cc: Michael Roth <michael.roth@amd.com>
Cc: Gerd Hoffmann <kraxel@redhat.com>
Acked-by: Gerd Hoffmann <kraxel@redhat.com>
Reviewed-by: Jiewen Yao <jiewen.yao@intel.com>
Signed-off-by: Ceping Sun <cepingx.sun@intel.com>
This commit is contained in:
Ceping Sun 2023-11-08 19:38:27 +08:00 committed by mergify[bot]
parent 212cf07aaa
commit 8a41004643
1 changed files with 40 additions and 1 deletions

View File

@ -38,6 +38,8 @@ typedef enum {
STATIC PAGE_TABLE_POOL *mPageTablePool = NULL;
#define MAX_RETRIES_PER_PAGE 3
/**
Returns boolean to indicate whether to indicate which, if any, memory encryption is enabled
@ -527,6 +529,13 @@ SetOrClearSharedBit (
EFI_STATUS Status;
EDKII_MEMORY_ACCEPT_PROTOCOL *MemoryAcceptProtocol;
UINT64 MapGpaRetryAddr;
UINT32 RetryCount;
UINT64 EndAddress;
MapGpaRetryAddr = 0;
RetryCount = 0;
AddressEncMask = GetMemEncryptionAddressMask ();
//
@ -540,7 +549,37 @@ SetOrClearSharedBit (
PhysicalAddress &= ~AddressEncMask;
}
TdStatus = TdVmCall (TDVMCALL_MAPGPA, PhysicalAddress, Length, 0, 0, NULL);
EndAddress = PhysicalAddress + Length;
while (RetryCount < MAX_RETRIES_PER_PAGE) {
TdStatus = TdVmCall (TDVMCALL_MAPGPA, PhysicalAddress, Length, 0, 0, &MapGpaRetryAddr);
if (TdStatus != TDVMCALL_STATUS_RETRY) {
break;
}
DEBUG ((DEBUG_VERBOSE, "%a: TdVmcall(MAPGPA) Retry PhysicalAddress is %llx, MapGpaRetryAddr is %llx\n", __func__, PhysicalAddress, MapGpaRetryAddr));
if ((MapGpaRetryAddr < PhysicalAddress) || (MapGpaRetryAddr >= EndAddress)) {
DEBUG ((
DEBUG_ERROR,
"%a: TdVmcall(MAPGPA) failed with MapGpaRetryAddr(%llx) less than PhysicalAddress(%llx) or more than or equal to EndAddress(%llx) \n",
__func__,
MapGpaRetryAddr,
PhysicalAddress,
EndAddress
));
break;
}
if (MapGpaRetryAddr == PhysicalAddress) {
RetryCount++;
continue;
}
PhysicalAddress = MapGpaRetryAddr;
Length = EndAddress - PhysicalAddress;
RetryCount = 0;
}
if (TdStatus != 0) {
DEBUG ((DEBUG_ERROR, "%a: TdVmcall(MAPGPA) failed with %llx\n", __func__, TdStatus));
ASSERT (FALSE);