diff --git a/MdeModulePkg/Core/Dxe/DxeMain.inf b/MdeModulePkg/Core/Dxe/DxeMain.inf index 8f64b8af0c..b4ad1f5629 100644 --- a/MdeModulePkg/Core/Dxe/DxeMain.inf +++ b/MdeModulePkg/Core/Dxe/DxeMain.inf @@ -92,7 +92,8 @@ UefiCpuPkg/UefiCpuPkg.dec OvmfPkg/OvmfPkg.dec ArmPkg/ArmPkg.dec - + ArmVirtPkg/ArmVirtPkg.dec + [LibraryClasses] BaseMemoryLib CacheMaintenanceLib diff --git a/MdeModulePkg/Core/Dxe/SysCall/AARCH64/CoreBootServices.S b/MdeModulePkg/Core/Dxe/SysCall/AARCH64/CoreBootServices.S index 068db95b1d..2b0f498d87 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/AARCH64/CoreBootServices.S +++ b/MdeModulePkg/Core/Dxe/SysCall/AARCH64/CoreBootServices.S @@ -69,8 +69,22 @@ call: // (x4) &CoreSp //------------------------------------------------------------------------------ ASM_FUNC(ArmCallRing3) - // Save FP and LR on Core Stack. - stp x29, x30, [sp, #-0x10]! + // Save registers. + sub sp, sp, #0x100 + + stp q8, q9, [sp, #0xe0] + stp q10, q11, [sp, #0xc0] + stp q12, q13, [sp, #0xa0] + stp q14, q15, [sp, #0x80] + + stp xzr, x16, [sp, #0x70] + stp x17, x18, [sp, #0x60] + stp x19, x20, [sp, #0x50] + stp x21, x22, [sp, #0x40] + stp x23, x24, [sp, #0x30] + stp x25, x26, [sp, #0x20] + stp x27, x28, [sp, #0x10] + stp x29, x30, [sp] // Disable interrupts. msr daifset, #0xf isb @@ -112,9 +126,21 @@ ASM_FUNC(ReturnToCore) msr far_el1, xzr // Switch to Core Stack. mov sp, x1 - // Restore Stack. - ldp x29, x30, [sp] - add sp, sp, #0x10 + // Restore registers and Stack. + ldp q8, q9, [sp, #0xe0] + ldp q10, q11, [sp, #0xc0] + ldp q12, q13, [sp, #0xa0] + ldp q14, q15, [sp, #0x80] + + ldr x16, [sp, #0x78] + ldp x17, x18, [sp, #0x60] + ldp x19, x20, [sp, #0x50] + ldp x21, x22, [sp, #0x40] + ldp x23, x24, [sp, #0x30] + ldp x25, x26, [sp, #0x20] + ldp x27, x28, [sp, #0x10] + ldp x29, x30, [sp] + add sp, sp, #0x100 // Enable interrupts. msr daifclr, #0xf isb diff --git a/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeMsr.c b/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeMsr.c index 88b06ac53a..ab2f1b536b 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeMsr.c +++ b/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeMsr.c @@ -6,12 +6,14 @@ **/ #include +#include #include #include #include "DxeMain.h" STATIC UINTN mCoreSp; +UINTN gUartBaseAddress; EFI_STATUS EFIAPI @@ -70,6 +72,12 @@ SysCallBootService ( DisableSMAP (); CopyMem ((VOID *)((UINTN)Physical + sizeof (UINTN)), (VOID *)UserRsp, 8 * sizeof (UINTN)); + + SetUefiImageMemoryAttributes ( + gUartBaseAddress, + EFI_PAGE_SIZE, + EFI_MEMORY_XP + ); EnableSMAP (); Status = CallBootService ( @@ -78,6 +86,12 @@ SysCallBootService ( (RING3_STACK *)(UINTN)Physical ); + SetUefiImageMemoryAttributes ( + gUartBaseAddress, + EFI_PAGE_SIZE, + EFI_MEMORY_XP | EFI_MEMORY_USER + ); + CoreFreePages (Physical, EFI_SIZE_TO_PAGES (9 * sizeof (UINTN))); return Status; @@ -86,10 +100,17 @@ SysCallBootService ( VOID EFIAPI InitializeMsr ( - VOID + IN OUT EFI_CONFIGURATION_TABLE *Table, + IN UINTN NumberOfEntries ) { - UINTN Tcr; + UINTN Tcr; + UINTN Index; + EARLY_PL011_BASE_ADDRESS *UartBase; + EFI_PHYSICAL_ADDRESS Physical; + EFI_HOB_GENERIC_HEADER *Ring3Hob; + UINT16 HobLength; + EFI_STATUS Status; // // If HCR_EL2.NV is 1 and the current Exception level is EL1, // then EL1 read accesses to the CurrentEL register return a value of 0x2 in bits[3:2]. @@ -103,6 +124,49 @@ InitializeMsr ( Tcr = ArmGetTCR (); Tcr |= TCR_EL1_HPD0_MASK | TCR_EL1_HPD1_MASK; ArmSetTCR (Tcr); + // + // Problem 1: Uart is memory maped. + // + for (Index = 0; Index < NumberOfEntries; ++Index) { + if (CompareGuid (&gEfiHobListGuid, &(Table[Index].VendorGuid))) { + UartBase = GET_GUID_HOB_DATA (Table[Index].VendorTable); + gUartBaseAddress = UartBase->DebugAddress; + // + // Copy Hob into Ring3. + // + Status = CoreAllocatePages ( + AllocateAnyPages, + EfiRing3MemoryType, + 1, + &Physical + ); + if (EFI_ERROR (Status)) { + DEBUG ((DEBUG_ERROR, "Core: Failed to allocate memory for Ring3Hob.\n")); + ASSERT (FALSE); + } + DEBUG ((DEBUG_ERROR, "UartBaseAddress = %p.\n", gUartBaseAddress)); + + Ring3Hob = (EFI_HOB_GENERIC_HEADER *)(UINTN)Physical; + + HobLength = (UINT16)((sizeof (EFI_HOB_GUID_TYPE) + sizeof (EARLY_PL011_BASE_ADDRESS) + 0x7) & (~0x7)); + + Ring3Hob->HobType = EFI_HOB_TYPE_GUID_EXTENSION; + Ring3Hob->HobLength = HobLength; + Ring3Hob->Reserved = 0; + + CopyGuid (&((EFI_HOB_GUID_TYPE *)Ring3Hob)->Name, &gEarlyPL011BaseAddressGuid); + + Ring3Hob = (EFI_HOB_GENERIC_HEADER *)((UINTN)Ring3Hob + HobLength); + + Ring3Hob->HobType = EFI_HOB_TYPE_END_OF_HOB_LIST; + Ring3Hob->HobLength = sizeof (EFI_HOB_GENERIC_HEADER); + Ring3Hob->Reserved = 0; + + Table[Index].VendorTable = (VOID *)(UINTN)Physical; + UartBase = GET_GUID_HOB_DATA (Table[Index].VendorTable); + UartBase->DebugAddress = gUartBaseAddress; + } + } if (ArmHasPan ()) { // diff --git a/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeMsr.c b/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeMsr.c index bc844910fd..fb7c2c35b6 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeMsr.c +++ b/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeMsr.c @@ -10,7 +10,8 @@ VOID EFIAPI InitializeMsr ( - VOID + IN OUT EFI_CONFIGURATION_TABLE *Table, + IN UINTN NumberOfEntries ) { diff --git a/MdeModulePkg/Core/Dxe/SysCall/IA32/InitializeMsr.c b/MdeModulePkg/Core/Dxe/SysCall/IA32/InitializeMsr.c index 8c1859163e..e08d97a336 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/IA32/InitializeMsr.c +++ b/MdeModulePkg/Core/Dxe/SysCall/IA32/InitializeMsr.c @@ -12,7 +12,8 @@ VOID EFIAPI InitializeMsr ( - VOID + IN OUT EFI_CONFIGURATION_TABLE *Table, + IN UINTN NumberOfEntries ) { UINT64 Msr; diff --git a/MdeModulePkg/Core/Dxe/SysCall/Initialization.c b/MdeModulePkg/Core/Dxe/SysCall/Initialization.c index 58651a4963..facc9d543d 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/Initialization.c +++ b/MdeModulePkg/Core/Dxe/SysCall/Initialization.c @@ -18,7 +18,8 @@ VOID *gRing3Interfaces; VOID EFIAPI InitializeMsr ( - VOID + IN OUT EFI_CONFIGURATION_TABLE *Table, + IN UINTN NumberOfEntries ); EFI_STATUS @@ -141,7 +142,10 @@ InitializeRing3 ( SetUefiImageMemoryAttributes ((UINTN)gRing3CallStackBase, SizeOfStack, EFI_MEMORY_XP | EFI_MEMORY_USER); DEBUG ((DEBUG_ERROR, "Core: gRing3CallStackTop = %p\n", gRing3CallStackTop)); - InitializeMsr (); + InitializeMsr ( + gRing3Data->SystemTable.ConfigurationTable, + gRing3Data->SystemTable.NumberOfTableEntries + ); return Status; } diff --git a/MdeModulePkg/Core/Dxe/SysCall/SupportedProtocols.c b/MdeModulePkg/Core/Dxe/SysCall/SupportedProtocols.c index 90fd9761a1..852ec7c663 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/SupportedProtocols.c +++ b/MdeModulePkg/Core/Dxe/SysCall/SupportedProtocols.c @@ -13,6 +13,10 @@ EFI_FILE_PROTOCOL mRing3FileProtocol; EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *mRing3SimpleFileSystemPointer; +#if defined (MDE_CPU_AARCH64) +extern UINTN gUartBaseAddress; +#endif + typedef struct { EFI_FILE_PROTOCOL Protocol; EFI_FILE_PROTOCOL *Ring3File; @@ -69,8 +73,16 @@ GoToRing3 ( EFI_MEMORY_XP | EFI_MEMORY_USER ); } +#elif defined (MDE_CPU_AARCH64) + // + // Necessary fix for DEBUG printings. + // + SetUefiImageMemoryAttributes ( + gUartBaseAddress, + EFI_PAGE_SIZE, + EFI_MEMORY_XP | EFI_MEMORY_USER + ); #endif - Status = CallRing3 (Input); #if defined (MDE_CPU_X64) || defined (MDE_CPU_IA32) @@ -81,6 +93,21 @@ GoToRing3 ( EFI_MEMORY_XP ); } +#elif defined (MDE_CPU_AARCH64) + // + // Problem 2: Uart memory maped page is not allocated at the very beginnig + // and can be used for translation table later. + // + DisableSMAP (); + // + // Problem 3: QEMU ramdomly breaks GP registers' context. + // + SetUefiImageMemoryAttributes ( + gUartBaseAddress, + EFI_PAGE_SIZE, + EFI_MEMORY_XP + ); + EnableSMAP (); #endif CoreFreePages (Ring3Pages, PagesNumber); diff --git a/MdeModulePkg/Core/Dxe/SysCall/X64/InitializeMsr.c b/MdeModulePkg/Core/Dxe/SysCall/X64/InitializeMsr.c index 171fdc5e7d..fded58d6fa 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/X64/InitializeMsr.c +++ b/MdeModulePkg/Core/Dxe/SysCall/X64/InitializeMsr.c @@ -12,7 +12,8 @@ VOID EFIAPI InitializeMsr ( - VOID + IN OUT EFI_CONFIGURATION_TABLE *Table, + IN UINTN NumberOfEntries ) { UINT64 Msr;