From 90ecffbc25fac692b7c6f2089e2c77c559893d45 Mon Sep 17 00:00:00 2001 From: Mikhail Krichanov Date: Fri, 10 Jan 2025 17:51:23 +0300 Subject: [PATCH] Ring3: Refactored User and SysCall stacks allocation. --- MdeModulePkg/Core/Dxe/DxeMain.h | 16 +++-- MdeModulePkg/Core/Dxe/DxeMain/DxeMain.c | 10 --- MdeModulePkg/Core/Dxe/Image/Image.c | 44 ++++++++++++- .../Dxe/SysCall/AARCH64/InitializeAARCH64.c | 4 +- .../Core/Dxe/SysCall/ARM/InitializeARM.c | 4 +- .../Core/Dxe/SysCall/Initialization.c | 62 ++++--------------- 6 files changed, 69 insertions(+), 71 deletions(-) diff --git a/MdeModulePkg/Core/Dxe/DxeMain.h b/MdeModulePkg/Core/Dxe/DxeMain.h index 057904e427..4df7d6596b 100644 --- a/MdeModulePkg/Core/Dxe/DxeMain.h +++ b/MdeModulePkg/Core/Dxe/DxeMain.h @@ -114,7 +114,7 @@ SPDX-License-Identifier: BSD-2-Clause-Patent /// #define DEPEX_STACK_SIZE_INCREMENT 0x1000 -#define USER_STACK_SIZE 0x20000 +#define STACK_SIZE 0x20000 #define RING3_INTERFACES_PAGES 20 typedef struct { @@ -230,6 +230,8 @@ typedef struct { VOID *HiiData; BOOLEAN IsUserImage; UINTN UserPageTable; + UINTN SysCallStackTop; + UINTN UserStackTop; } LOADED_IMAGE_PRIVATE_DATA; typedef struct { @@ -284,10 +286,8 @@ extern LOADED_IMAGE_PRIVATE_DATA * mCurrentImage; extern RING3_DATA *gRing3Data; extern VOID *gRing3Interfaces; -extern VOID *gCoreSysCallStackBase; -extern VOID *gCoreSysCallStackTop; -extern VOID *gRing3CallStackBase; -extern VOID *gRing3CallStackTop; +extern UINTN gCoreSysCallStackTop; +extern UINTN gRing3CallStackTop; extern VOID *gRing3EntryPoint; extern UINTN gUserPageTable; extern UINTN gCorePageTable; @@ -2786,7 +2786,11 @@ FreeProtocolsList ( UINTN EFIAPI InitializeUserPageTable ( - IN LOADED_IMAGE_PRIVATE_DATA *Image + IN LOADED_IMAGE_PRIVATE_DATA *Image, + IN UINTN SysCallStackBase, + IN UINTN SysCallStackSize, + IN UINTN UserStackBase, + IN UINTN UserStackSize ); #endif diff --git a/MdeModulePkg/Core/Dxe/DxeMain/DxeMain.c b/MdeModulePkg/Core/Dxe/DxeMain/DxeMain.c index e489a2a89e..bc55cb3183 100644 --- a/MdeModulePkg/Core/Dxe/DxeMain/DxeMain.c +++ b/MdeModulePkg/Core/Dxe/DxeMain/DxeMain.c @@ -791,16 +791,6 @@ CoreExitBootServices ( RING3_INTERFACES_PAGES ); - CoreFreePages ( - (EFI_PHYSICAL_ADDRESS)(UINTN)gCoreSysCallStackBase, - EFI_SIZE_TO_PAGES (USER_STACK_SIZE) - ); - - CoreFreePages ( - (EFI_PHYSICAL_ADDRESS)(UINTN)gRing3CallStackBase, - EFI_SIZE_TO_PAGES (USER_STACK_SIZE) - ); - FreeProtocolsList (); } diff --git a/MdeModulePkg/Core/Dxe/Image/Image.c b/MdeModulePkg/Core/Dxe/Image/Image.c index 0cba178621..4b076ec213 100644 --- a/MdeModulePkg/Core/Dxe/Image/Image.c +++ b/MdeModulePkg/Core/Dxe/Image/Image.c @@ -1033,6 +1033,30 @@ CoreUnloadAndCloseImage ( CoreFreePool (Image); } +STATIC +UINTN +EFIAPI +AllocateStack ( + IN UINTN Size, + OUT UINTN *Base + ) +{ + UINTN TopOfStack; + + ASSERT (Base != NULL); + ASSERT (IS_ALIGNED (Size, EFI_PAGE_SIZE)); + + *Base = (UINTN)AllocatePages (EFI_SIZE_TO_PAGES (Size)); + ASSERT (*Base != 0); + // + // Compute the top of the allocated stack. Pre-allocate a UINTN for safety. + // + TopOfStack = *Base + Size - CPU_STACK_ALIGNMENT; + TopOfStack = ALIGN_VALUE (TopOfStack, CPU_STACK_ALIGNMENT); + + return TopOfStack; +} + /** Loads an EFI image into memory and returns a handle to the image. @@ -1108,6 +1132,8 @@ CoreLoadImageCommon ( UEFI_IMAGE_LOADER_IMAGE_CONTEXT ImageContext; UINT8 ImageOrigin; EFI_FV_FILE_ATTRIBUTES FileAttributes; + UINTN SysCallStackBase; + UINTN UserStackBase; SecurityStatus = EFI_SUCCESS; @@ -1445,7 +1471,19 @@ CoreLoadImageCommon ( ProtectUefiImage (&Image->Info, ImageOrigin, &ImageContext, Image->IsUserImage); if ((gRing3Data != NULL) && Image->IsUserImage) { - Image->UserPageTable = InitializeUserPageTable (Image); + Image->SysCallStackTop = AllocateStack (STACK_SIZE, &SysCallStackBase); + SetUefiImageMemoryAttributes (SysCallStackBase, STACK_SIZE, EFI_MEMORY_XP); + + Image->UserStackTop = AllocateStack (STACK_SIZE, &UserStackBase); + SetUefiImageMemoryAttributes (UserStackBase, STACK_SIZE, EFI_MEMORY_XP | EFI_MEMORY_USER); + + Image->UserPageTable = InitializeUserPageTable ( + Image, + SysCallStackBase, + STACK_SIZE, + UserStackBase, + STACK_SIZE + ); } RegisterMemoryProfileImage ( @@ -1703,7 +1741,9 @@ CoreStartImage ( gCpu->GetMemoryAttributes (gCpu, (EFI_PHYSICAL_ADDRESS)(UINTN)Image->EntryPoint, &Attributes); ASSERT ((Attributes & EFI_MEMORY_USER) != 0); - gUserPageTable = Image->UserPageTable; + gUserPageTable = Image->UserPageTable; + gRing3CallStackTop = Image->UserStackTop; + gCoreSysCallStackTop = Image->SysCallStackTop; Image->Status = GoToRing3 ( 2, diff --git a/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeAARCH64.c b/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeAARCH64.c index 42ea8ac88f..73400bfa43 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeAARCH64.c +++ b/MdeModulePkg/Core/Dxe/SysCall/AARCH64/InitializeAARCH64.c @@ -19,9 +19,9 @@ EFI_STATUS EFIAPI ArmCallRing3 ( IN RING3_CALL_DATA *Data, - IN VOID *StackPointer, + IN UINTN StackPointer, IN VOID *EntryPoint, - IN VOID *SysCallStack, + IN UINTN SysCallStack, IN VOID *CoreStack, IN UINTN UserPageTable ); diff --git a/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeARM.c b/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeARM.c index 26984d4e58..4d81587422 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeARM.c +++ b/MdeModulePkg/Core/Dxe/SysCall/ARM/InitializeARM.c @@ -18,9 +18,9 @@ EFI_STATUS EFIAPI ArmCallRing3 ( IN RING3_CALL_DATA *Data, - IN VOID *StackPointer, + IN UINTN StackPointer, IN VOID *EntryPoint, - IN VOID *SysCallStack, + IN UINTN SysCallStack, IN VOID *CoreStack, IN UINTN UserPageTable ); diff --git a/MdeModulePkg/Core/Dxe/SysCall/Initialization.c b/MdeModulePkg/Core/Dxe/SysCall/Initialization.c index 171f70f962..c1e6e927fc 100644 --- a/MdeModulePkg/Core/Dxe/SysCall/Initialization.c +++ b/MdeModulePkg/Core/Dxe/SysCall/Initialization.c @@ -9,10 +9,8 @@ #include "DxeMain.h" -VOID *gCoreSysCallStackTop; -VOID *gCoreSysCallStackBase; -VOID *gRing3CallStackTop; -VOID *gRing3CallStackBase; +UINTN gCoreSysCallStackTop; +UINTN gRing3CallStackTop; VOID *gRing3EntryPoint; RING3_DATA *gRing3Data; VOID *gRing3Interfaces; @@ -48,8 +46,6 @@ InitializeRing3 ( ) { EFI_STATUS Status; - VOID *TopOfStack; - UINTN SizeOfStack; EFI_PHYSICAL_ADDRESS Physical; UINTN Index; EFI_CONFIGURATION_TABLE *Conf; @@ -157,42 +153,6 @@ InitializeRing3 ( EFI_MEMORY_XP | EFI_MEMORY_USER ); - SizeOfStack = EFI_SIZE_TO_PAGES (USER_STACK_SIZE) * EFI_PAGE_SIZE; - - // - // Allocate 128KB for the Core SysCall Stack. - // - gCoreSysCallStackBase = AllocatePages (EFI_SIZE_TO_PAGES (USER_STACK_SIZE)); - ASSERT (gCoreSysCallStackBase != NULL); - - // - // Compute the top of the allocated stack. Pre-allocate a UINTN for safety. - // - TopOfStack = (VOID *)((UINTN)gCoreSysCallStackBase + SizeOfStack - CPU_STACK_ALIGNMENT); - TopOfStack = ALIGN_POINTER (TopOfStack, CPU_STACK_ALIGNMENT); - - gCoreSysCallStackTop = TopOfStack; - - SetUefiImageMemoryAttributes ((UINTN)gCoreSysCallStackBase, SizeOfStack, EFI_MEMORY_XP); - DEBUG ((DEBUG_ERROR, "Core: gCoreSysCallStackTop = %p\n", gCoreSysCallStackTop)); - - // - // Allocate 128KB for the User Stack. - // - gRing3CallStackBase = AllocatePages (EFI_SIZE_TO_PAGES (USER_STACK_SIZE)); - ASSERT (gRing3CallStackBase != NULL); - - // - // Compute the top of the allocated stack. Pre-allocate a UINTN for safety. - // - TopOfStack = (VOID *)((UINTN)gRing3CallStackBase + SizeOfStack - CPU_STACK_ALIGNMENT); - TopOfStack = ALIGN_POINTER (TopOfStack, CPU_STACK_ALIGNMENT); - - gRing3CallStackTop = TopOfStack; - - SetUefiImageMemoryAttributes ((UINTN)gRing3CallStackBase, SizeOfStack, EFI_MEMORY_XP | EFI_MEMORY_USER); - DEBUG ((DEBUG_ERROR, "Core: gRing3CallStackTop = %p\n", gRing3CallStackTop)); - InitializeMsr ( gRing3Data->SystemTable.ConfigurationTable, gRing3Data->SystemTable.NumberOfTableEntries @@ -206,7 +166,11 @@ InitializeRing3 ( UINTN EFIAPI InitializeUserPageTable ( - IN LOADED_IMAGE_PRIVATE_DATA *Image + IN LOADED_IMAGE_PRIVATE_DATA *Image, + IN UINTN SysCallStackBase, + IN UINTN SysCallStackSize, + IN UINTN UserStackBase, + IN UINTN UserStackSize ) { UINTN UserPageTable; @@ -222,7 +186,7 @@ InitializeUserPageTable ( MakeUserPageTableTemplate (&UserPageTable, &UserPageTableSize); // - // Map gRing3Data, gRing3Interfaces, gRing3CallStackBase, DxeRing3 + // Map gRing3Data, gRing3Interfaces, UserStackBase, DxeRing3 // gCpu->SetUserMemoryAttributes ( gCpu, @@ -243,8 +207,8 @@ InitializeUserPageTable ( gCpu->SetUserMemoryAttributes ( gCpu, UserPageTable, - (UINTN)gRing3CallStackBase, - EFI_SIZE_TO_PAGES (USER_STACK_SIZE) * EFI_PAGE_SIZE, + UserStackBase, + UserStackSize, EFI_MEMORY_XP | EFI_MEMORY_USER ); @@ -264,7 +228,7 @@ InitializeUserPageTable ( } // - // Map CoreBootServices, gCoreSysCallStackBase + // Map CoreBootServices, SysCallStackBase // gCpu->SetUserMemoryAttributes ( gCpu, @@ -277,8 +241,8 @@ InitializeUserPageTable ( gCpu->SetUserMemoryAttributes ( gCpu, UserPageTable, - (UINTN)gCoreSysCallStackBase, - EFI_SIZE_TO_PAGES (USER_STACK_SIZE) * EFI_PAGE_SIZE, + SysCallStackBase, + SysCallStackSize, EFI_MEMORY_XP );