The patch attached is to fix the issue that LMFA feature is failed on OVMF.  The root cause is that OVMF platform reserved 128k top memory range for EMU variable range before PEI memory range re-locate to specified memory range, which is quite different from real platform. The original implementation has an assumption that AllocatePage will NOT be invoked until PEI memory range is re-located to preassigned memory range, , which is TRUE in most real platform, but FALSE on OVMF platform. So This patch is to enhanced the memory resource HOB re-organization algorithm to eliminate the assumption.



git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@10225 6f19259b-4bc3-4df7-8a09-765794883524
diff --git a/MdeModulePkg/Core/Dxe/Image/Image.c b/MdeModulePkg/Core/Dxe/Image/Image.c
index 04f8368..25f1720 100644
--- a/MdeModulePkg/Core/Dxe/Image/Image.c
+++ b/MdeModulePkg/Core/Dxe/Image/Image.c
@@ -356,7 +356,7 @@
          // relative to top address

          //

          if ((INT64)PcdGet64(PcdLoadModuleAtFixAddressEnable) < 0) {

-         	 ImageContext->ImageAddress = gLoadModuleAtFixAddressConfigurationTable.DxeCodeTopAddress + (INT64)ImageContext->ImageAddress;

+         	 ImageContext->ImageAddress = gLoadModuleAtFixAddressConfigurationTable.DxeCodeTopAddress + (INT64)(INTN)ImageContext->ImageAddress;

          }

          //

          // Check if the memory range is avaliable.

diff --git a/MdeModulePkg/Core/Pei/Dispatcher/Dispatcher.c b/MdeModulePkg/Core/Pei/Dispatcher/Dispatcher.c
index 455c79b..85a1ad7 100644
--- a/MdeModulePkg/Core/Pei/Dispatcher/Dispatcher.c
+++ b/MdeModulePkg/Core/Pei/Dispatcher/Dispatcher.c
@@ -241,6 +241,50 @@
 //

 #define MINIMUM_INITIAL_MEMORY_SIZE 0x10000

 /**

+  This function is to test if the memory range described in resource HOB is available or not. 

+  

+  This function should only be invoked when Loading Module at Fixed Address(LMFA) feature is enabled. Some platform may allocate the 

+  memory before PeiLoadFixAddressHook in invoked. so this function is to test if the memory range described by the input resource HOB is

+  available or not.

+

+  @param PrivateData         Pointer to the private data passed in from caller

+  @param ResourceHob         Pointer to a resource HOB which described the memory range described by the input resource HOB

+**/

+BOOLEAN

+PeiLoadFixAddressIsMemoryRangeAvailable (

+  IN PEI_CORE_INSTANCE                  *PrivateData,

+  IN EFI_HOB_RESOURCE_DESCRIPTOR        *ResourceHob

+  )

+{

+	EFI_HOB_MEMORY_ALLOCATION          *MemoryHob;

+	BOOLEAN                             IsAvailable;

+	EFI_PEI_HOB_POINTERS                Hob;

+	

+  IsAvailable = TRUE;

+	if (PrivateData == NULL || ResourceHob == NULL) {

+	  return FALSE;

+	}

+	//

+  // test if the memory range describe in the HOB is already allocated.

+  //

+  for (Hob.Raw = PrivateData->HobList.Raw; !END_OF_HOB_LIST(Hob); Hob.Raw = GET_NEXT_HOB(Hob)) {

+    //                                                              

+    // See if this is a memory allocation HOB                     

+    //

+    if (GET_HOB_TYPE (Hob) == EFI_HOB_TYPE_MEMORY_ALLOCATION) { 

+      MemoryHob = Hob.MemoryAllocation;

+      if(MemoryHob->AllocDescriptor.MemoryBaseAddress == ResourceHob->PhysicalStart && 

+         MemoryHob->AllocDescriptor.MemoryBaseAddress + MemoryHob->AllocDescriptor.MemoryLength == ResourceHob->PhysicalStart + ResourceHob->ResourceLength) {

+         IsAvailable = FALSE;

+         break;  

+       }

+     }

+  }

+  

+  return IsAvailable;

+       

+}

+/**

   Hook function for Loading Module at Fixed Address feature

   

   This function should only be invoked when Loading Module at Fixed Address(LMFA) feature is enabled. When feature is

@@ -269,6 +313,7 @@
   EFI_PEI_HOB_POINTERS               NextHob;

   EFI_PHYSICAL_ADDRESS               MaxMemoryBaseAddress;

   UINT64                             MaxMemoryLength;

+  EFI_HOB_MEMORY_ALLOCATION          *MemoryHob;

   //

   // Initialize Local Variables

   //

@@ -362,6 +407,61 @@
     } 

   }

   //

+  // Some platform is already allocated pages before the HOB re-org. Here to build dedicated resource HOB to describe

+  //  the allocated memory range

+  //

+  for (Hob.Raw = PrivateData->HobList.Raw; !END_OF_HOB_LIST(Hob); Hob.Raw = GET_NEXT_HOB(Hob)) {

+    //                                                              

+    // See if this is a memory allocation HOB                     

+    //

+    if (GET_HOB_TYPE (Hob) == EFI_HOB_TYPE_MEMORY_ALLOCATION) {

+      MemoryHob = Hob.MemoryAllocation;

+      for (NextHob.Raw = PrivateData->HobList.Raw; !END_OF_HOB_LIST(NextHob); NextHob.Raw = GET_NEXT_HOB(NextHob)) {

+        //

+        // See if this is a resource descriptor HOB

+        //

+        if (GET_HOB_TYPE (NextHob) == EFI_HOB_TYPE_RESOURCE_DESCRIPTOR) {

+        	NextResourceHob = NextHob.ResourceDescriptor;

+          //

+          // If range described in this hob is not system memory or heigher than MAX_ADDRESS, ignored.

+          //

+          if (NextResourceHob->ResourceType == EFI_RESOURCE_SYSTEM_MEMORY && NextResourceHob->PhysicalStart + NextResourceHob->ResourceLength > MAX_ADDRESS) {

+            continue;

+          }

+          //

+          // If the range describe in memory allocation HOB  belongs to the memroy range described by the resource hob

+          //          

+          if (MemoryHob->AllocDescriptor.MemoryBaseAddress >= NextResourceHob->PhysicalStart && 

+              MemoryHob->AllocDescriptor.MemoryBaseAddress + MemoryHob->AllocDescriptor.MemoryLength <= NextResourceHob->PhysicalStart + NextResourceHob->ResourceLength) {

+             //

+             // Build seperate resource hob for this allocated range

+             //                     

+             if (MemoryHob->AllocDescriptor.MemoryBaseAddress > NextResourceHob->PhysicalStart) {

+               BuildResourceDescriptorHob (

+                 EFI_RESOURCE_SYSTEM_MEMORY,                       

+                 NextResourceHob->ResourceAttribute,

+                 NextResourceHob->PhysicalStart,                             

+                 (MemoryHob->AllocDescriptor.MemoryBaseAddress - NextResourceHob->PhysicalStart)      

+               );

+             }

+             if (MemoryHob->AllocDescriptor.MemoryBaseAddress + MemoryHob->AllocDescriptor.MemoryLength < NextResourceHob->PhysicalStart + NextResourceHob->ResourceLength) {

+               BuildResourceDescriptorHob (

+                 EFI_RESOURCE_SYSTEM_MEMORY,                       

+                 NextResourceHob->ResourceAttribute,

+                 MemoryHob->AllocDescriptor.MemoryBaseAddress + MemoryHob->AllocDescriptor.MemoryLength,                            

+                 (NextResourceHob->PhysicalStart + NextResourceHob->ResourceLength -(MemoryHob->AllocDescriptor.MemoryBaseAddress + MemoryHob->AllocDescriptor.MemoryLength))    

+               );

+             }

+             NextResourceHob->PhysicalStart = MemoryHob->AllocDescriptor.MemoryBaseAddress;

+             NextResourceHob->ResourceLength = MemoryHob->AllocDescriptor.MemoryLength;

+             break;

+          }

+        }

+      }

+    }

+  }

+

+  //

   // Try to find and validate the TOP address.

   //  

   if ((INT64)PcdGet64(PcdLoadModuleAtFixAddressEnable) > 0 ) {

@@ -396,11 +496,12 @@
             // See if Top address specified by user is valid.

             //

             if (ResourceHob->PhysicalStart + TotalReservedMemorySize < TopLoadingAddress && 

-                (ResourceHob->PhysicalStart + ResourceHob->ResourceLength - MINIMUM_INITIAL_MEMORY_SIZE) >= TopLoadingAddress) {

+                (ResourceHob->PhysicalStart + ResourceHob->ResourceLength - MINIMUM_INITIAL_MEMORY_SIZE) >= TopLoadingAddress && 

+                PeiLoadFixAddressIsMemoryRangeAvailable(PrivateData, ResourceHob)) {

               CurrentResourceHob = ResourceHob; 

               CurrentHob = Hob;

               break;

-            }

+           }

         }

       }  

     }  

@@ -428,7 +529,7 @@
               //

               // See if Top address specified by user is valid.

               //

-              if (ResourceHob->ResourceLength > TotalReservedMemorySize) {

+              if (ResourceHob->ResourceLength > TotalReservedMemorySize && PeiLoadFixAddressIsMemoryRangeAvailable(PrivateData, ResourceHob)) {

                  DEBUG ((EFI_D_INFO, "(0x%lx, 0x%lx)\n",  

                           (ResourceHob->PhysicalStart + TotalReservedMemorySize -MINIMUM_INITIAL_MEMORY_SIZE), 

                           (ResourceHob->PhysicalStart + ResourceHob->ResourceLength -MINIMUM_INITIAL_MEMORY_SIZE) 

@@ -440,7 +541,8 @@
       //

       // Assert here 

       //

-      ASSERT (FALSE);      

+      ASSERT (FALSE);   

+      return;   

     }     

   } else {

     //

@@ -462,7 +564,7 @@
         //

         if (ResourceHob->ResourceType == EFI_RESOURCE_SYSTEM_MEMORY && 

             ResourceHob->PhysicalStart + ResourceHob->ResourceLength <= MAX_ADDRESS &&

-            ResourceHob->ResourceLength > TotalReservedMemorySize) {

+            ResourceHob->ResourceLength > TotalReservedMemorySize && PeiLoadFixAddressIsMemoryRangeAvailable(PrivateData, ResourceHob)) {

           //

           // See if this is the highest largest system memory region below MaxAddress

           //

@@ -479,7 +581,8 @@
       //

       // Assert here 

       //

-      ASSERT (FALSE);      

+      ASSERT (FALSE);

+      return;  

     } else {

       TopLoadingAddress = CurrentResourceHob->PhysicalStart + CurrentResourceHob->ResourceLength ; 

     }         

@@ -487,10 +590,10 @@
   

   if (CurrentResourceHob != NULL) {

     //

-    // rebuild hob for PEI memmory and reserved memory

+    // rebuild resource HOB for PEI memmory and reserved memory

     //

     BuildResourceDescriptorHob (

-      EFI_RESOURCE_SYSTEM_MEMORY,                       // MemoryType,

+      EFI_RESOURCE_SYSTEM_MEMORY,                       

       (

       EFI_RESOURCE_ATTRIBUTE_PRESENT |

       EFI_RESOURCE_ATTRIBUTE_INITIALIZED |

@@ -500,15 +603,15 @@
       EFI_RESOURCE_ATTRIBUTE_WRITE_THROUGH_CACHEABLE |

       EFI_RESOURCE_ATTRIBUTE_WRITE_BACK_CACHEABLE

       ),

-      (TopLoadingAddress - TotalReservedMemorySize),                             // MemoryBegin

-      TotalReservedMemorySize      // MemoryLength

+      (TopLoadingAddress - TotalReservedMemorySize),                             

+      TotalReservedMemorySize     

     );

     //

-    // rebuild hob for the remain memory if necessary

+    // rebuild resource for the remain memory if necessary

     //

     if (CurrentResourceHob->PhysicalStart < TopLoadingAddress - TotalReservedMemorySize) {

       BuildResourceDescriptorHob (

-        EFI_RESOURCE_SYSTEM_MEMORY,                       // MemoryType,

+        EFI_RESOURCE_SYSTEM_MEMORY,                       

         (

          EFI_RESOURCE_ATTRIBUTE_PRESENT |

          EFI_RESOURCE_ATTRIBUTE_INITIALIZED |

@@ -517,8 +620,8 @@
          EFI_RESOURCE_ATTRIBUTE_WRITE_THROUGH_CACHEABLE |

          EFI_RESOURCE_ATTRIBUTE_WRITE_BACK_CACHEABLE

          ),

-         CurrentResourceHob->PhysicalStart,                             // MemoryBegin

-         (TopLoadingAddress - TotalReservedMemorySize - CurrentResourceHob->PhysicalStart)      // MemoryLength

+         CurrentResourceHob->PhysicalStart,                             

+         (TopLoadingAddress - TotalReservedMemorySize - CurrentResourceHob->PhysicalStart)      

        );

     }

     if (CurrentResourceHob->PhysicalStart + CurrentResourceHob->ResourceLength  > TopLoadingAddress ) {

@@ -802,8 +905,7 @@
                 //

                 PeiLoadFixAddressHook(Private);

                 //

-                // if Loading Module at Fixed Address is enabled, This is the first invoke to page

-                // allocation for Pei Code range. This memory range should be reserved for loading PEIMs

+                // if Loading Module at Fixed Address is enabled, Allocating memory range for Pei code range.

                 //

                 LoadFixPeiCodeBegin = AllocatePages((UINTN)PcdGet32(PcdLoadFixAddressPeiCodePageNumber));

                 DEBUG ((EFI_D_INFO, "LOADING MODULE FIXED INFO: PeiCodeBegin = 0x%lX, PeiCodeTop= 0x%lX\n", (UINT64)(UINTN)LoadFixPeiCodeBegin, (UINT64)((UINTN)LoadFixPeiCodeBegin + PcdGet32(PcdLoadFixAddressPeiCodePageNumber) * EFI_PAGE_SIZE)));