]> git.proxmox.com Git - mirror_edk2.git/blobdiff - OvmfPkg/QemuVideoDxe/Gop.c
UefiCpuPkg: Move AsmRelocateApLoopStart from Mpfuncs.nasm to AmdSev.nasm
[mirror_edk2.git] / OvmfPkg / QemuVideoDxe / Gop.c
index f6e9b6c14f4790aedfc1f02140eeb360f1cc8025..7a9fe208c99c8753b7c0855b924149029f2b3a78 100644 (file)
@@ -1,81 +1,88 @@
 /** @file\r
   Graphics Output Protocol functions for the QEMU video controller.\r
 \r
-  Copyright (c) 2007 - 2010, Intel Corporation. All rights reserved.<BR>\r
+  Copyright (c) 2007 - 2018, Intel Corporation. All rights reserved.<BR>\r
 \r
-  This program and the accompanying materials\r
-  are licensed and made available under the terms and conditions of the BSD License\r
-  which accompanies this distribution. The full text of the license may be found at\r
-  http://opensource.org/licenses/bsd-license.php\r
-\r
-  THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,\r
-  WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.\r
+  SPDX-License-Identifier: BSD-2-Clause-Patent\r
 \r
 **/\r
 \r
 #include "Qemu.h"\r
-#include <IndustryStandard/Acpi.h>\r
-#include <Library/BltLib.h>\r
 \r
 STATIC\r
 VOID\r
 QemuVideoCompleteModeInfo (\r
-  IN  QEMU_VIDEO_MODE_DATA           *ModeData,\r
+  IN  QEMU_VIDEO_MODE_DATA                  *ModeData,\r
   OUT EFI_GRAPHICS_OUTPUT_MODE_INFORMATION  *Info\r
   )\r
 {\r
   Info->Version = 0;\r
   if (ModeData->ColorDepth == 8) {\r
-    Info->PixelFormat = PixelBitMask;\r
-    Info->PixelInformation.RedMask = PIXEL_RED_MASK;\r
-    Info->PixelInformation.GreenMask = PIXEL_GREEN_MASK;\r
-    Info->PixelInformation.BlueMask = PIXEL_BLUE_MASK;\r
+    Info->PixelFormat                   = PixelBitMask;\r
+    Info->PixelInformation.RedMask      = PIXEL_RED_MASK;\r
+    Info->PixelInformation.GreenMask    = PIXEL_GREEN_MASK;\r
+    Info->PixelInformation.BlueMask     = PIXEL_BLUE_MASK;\r
     Info->PixelInformation.ReservedMask = 0;\r
   } else if (ModeData->ColorDepth == 24) {\r
-    Info->PixelFormat = PixelBitMask;\r
-    Info->PixelInformation.RedMask = PIXEL24_RED_MASK;\r
-    Info->PixelInformation.GreenMask = PIXEL24_GREEN_MASK;\r
-    Info->PixelInformation.BlueMask = PIXEL24_BLUE_MASK;\r
+    Info->PixelFormat                   = PixelBitMask;\r
+    Info->PixelInformation.RedMask      = PIXEL24_RED_MASK;\r
+    Info->PixelInformation.GreenMask    = PIXEL24_GREEN_MASK;\r
+    Info->PixelInformation.BlueMask     = PIXEL24_BLUE_MASK;\r
     Info->PixelInformation.ReservedMask = 0;\r
   } else if (ModeData->ColorDepth == 32) {\r
-    DEBUG ((EFI_D_INFO, "PixelBlueGreenRedReserved8BitPerColor\n"));\r
-    Info->PixelFormat = PixelBlueGreenRedReserved8BitPerColor;\r
+    DEBUG ((DEBUG_INFO, "PixelBlueGreenRedReserved8BitPerColor\n"));\r
+    Info->PixelFormat                   = PixelBlueGreenRedReserved8BitPerColor;\r
+    Info->PixelInformation.RedMask      = 0;\r
+    Info->PixelInformation.GreenMask    = 0;\r
+    Info->PixelInformation.BlueMask     = 0;\r
+    Info->PixelInformation.ReservedMask = 0;\r
+  } else {\r
+    DEBUG ((DEBUG_ERROR, "%a: Invalid ColorDepth %u", __FUNCTION__, ModeData->ColorDepth));\r
+    ASSERT (FALSE);\r
   }\r
+\r
   Info->PixelsPerScanLine = Info->HorizontalResolution;\r
 }\r
 \r
-\r
 STATIC\r
 EFI_STATUS\r
 QemuVideoCompleteModeData (\r
-  IN  QEMU_VIDEO_PRIVATE_DATA           *Private,\r
-  OUT EFI_GRAPHICS_OUTPUT_PROTOCOL_MODE *Mode\r
+  IN  QEMU_VIDEO_PRIVATE_DATA            *Private,\r
+  OUT EFI_GRAPHICS_OUTPUT_PROTOCOL_MODE  *Mode\r
   )\r
 {\r
   EFI_GRAPHICS_OUTPUT_MODE_INFORMATION  *Info;\r
   EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR     *FrameBufDesc;\r
-  QEMU_VIDEO_MODE_DATA           *ModeData;\r
+  QEMU_VIDEO_MODE_DATA                  *ModeData;\r
 \r
   ModeData = &Private->ModeData[Mode->Mode];\r
-  Info = Mode->Info;\r
+  Info     = Mode->Info;\r
   QemuVideoCompleteModeInfo (ModeData, Info);\r
 \r
   Private->PciIo->GetBarAttributes (\r
-                        Private->PciIo,\r
-                        0,\r
-                        NULL,\r
-                        (VOID**) &FrameBufDesc\r
-                        );\r
+                    Private->PciIo,\r
+                    Private->FrameBufferVramBarIndex,\r
+                    NULL,\r
+                    (VOID **)&FrameBufDesc\r
+                    );\r
 \r
   Mode->FrameBufferBase = FrameBufDesc->AddrRangeMin;\r
   Mode->FrameBufferSize = Info->HorizontalResolution * Info->VerticalResolution;\r
   Mode->FrameBufferSize = Mode->FrameBufferSize * ((ModeData->ColorDepth + 7) / 8);\r
-  DEBUG ((EFI_D_INFO, "FrameBufferBase: 0x%x, FrameBufferSize: 0x%x\n", Mode->FrameBufferBase, Mode->FrameBufferSize));\r
-\r
+  Mode->FrameBufferSize = EFI_PAGES_TO_SIZE (\r
+                            EFI_SIZE_TO_PAGES (Mode->FrameBufferSize)\r
+                            );\r
+  DEBUG ((\r
+    DEBUG_INFO,\r
+    "FrameBufferBase: 0x%Lx, FrameBufferSize: 0x%Lx\n",\r
+    Mode->FrameBufferBase,\r
+    (UINT64)Mode->FrameBufferSize\r
+    ));\r
+\r
+  FreePool (FrameBufDesc);\r
   return EFI_SUCCESS;\r
 }\r
 \r
-\r
 //\r
 // Graphics Output Protocol Member Functions\r
 //\r
@@ -87,6 +94,7 @@ QemuVideoGraphicsOutputQueryMode (
   OUT UINTN                                 *SizeOfInfo,\r
   OUT EFI_GRAPHICS_OUTPUT_MODE_INFORMATION  **Info\r
   )\r
+\r
 /*++\r
 \r
 Routine Description:\r
@@ -113,11 +121,7 @@ Routine Description:
 \r
   Private = QEMU_VIDEO_PRIVATE_DATA_FROM_GRAPHICS_OUTPUT_THIS (This);\r
 \r
-  if (Private->HardwareNeedsStarting) {\r
-    return EFI_NOT_STARTED;\r
-  }\r
-\r
-  if (Info == NULL || SizeOfInfo == NULL || ModeNumber >= This->Mode->MaxMode) {\r
+  if ((Info == NULL) || (SizeOfInfo == NULL) || (ModeNumber >= This->Mode->MaxMode)) {\r
     return EFI_INVALID_PARAMETER;\r
   }\r
 \r
@@ -128,7 +132,7 @@ Routine Description:
 \r
   *SizeOfInfo = sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION);\r
 \r
-  ModeData = &Private->ModeData[ModeNumber];\r
+  ModeData                      = &Private->ModeData[ModeNumber];\r
   (*Info)->HorizontalResolution = ModeData->HorizontalResolution;\r
   (*Info)->VerticalResolution   = ModeData->VerticalResolution;\r
   QemuVideoCompleteModeInfo (ModeData, *Info);\r
@@ -139,9 +143,10 @@ Routine Description:
 EFI_STATUS\r
 EFIAPI\r
 QemuVideoGraphicsOutputSetMode (\r
-  IN  EFI_GRAPHICS_OUTPUT_PROTOCOL *This,\r
-  IN  UINT32                       ModeNumber\r
+  IN  EFI_GRAPHICS_OUTPUT_PROTOCOL  *This,\r
+  IN  UINT32                        ModeNumber\r
   )\r
+\r
 /*++\r
 \r
 Routine Description:\r
@@ -159,9 +164,10 @@ Routine Description:
 \r
 --*/\r
 {\r
-  QEMU_VIDEO_PRIVATE_DATA    *Private;\r
-  QEMU_VIDEO_MODE_DATA       *ModeData;\r
-//  UINTN                             Count;\r
+  QEMU_VIDEO_PRIVATE_DATA        *Private;\r
+  QEMU_VIDEO_MODE_DATA           *ModeData;\r
+  RETURN_STATUS                  Status;\r
+  EFI_GRAPHICS_OUTPUT_BLT_PIXEL  Black;\r
 \r
   Private = QEMU_VIDEO_PRIVATE_DATA_FROM_GRAPHICS_OUTPUT_THIS (This);\r
 \r
@@ -171,44 +177,78 @@ Routine Description:
 \r
   ModeData = &Private->ModeData[ModeNumber];\r
 \r
-  if (Private->LineBuffer) {\r
-    gBS->FreePool (Private->LineBuffer);\r
-  }\r
-\r
-  Private->LineBuffer = NULL;\r
-  Private->LineBuffer = AllocatePool (4 * ModeData->HorizontalResolution);\r
-  if (Private->LineBuffer == NULL) {\r
-    return EFI_OUT_OF_RESOURCES;\r
-  }\r
-\r
   switch (Private->Variant) {\r
-  case QEMU_VIDEO_CIRRUS_5430:\r
-  case QEMU_VIDEO_CIRRUS_5446:\r
-    InitializeCirrusGraphicsMode (Private, &QemuVideoCirrusModes[ModeData->ModeNumber]);\r
-    break;\r
-  case QEMU_VIDEO_BOCHS:\r
-    InitializeBochsGraphicsMode (Private, &QemuVideoBochsModes[ModeData->ModeNumber]);\r
-    break;\r
-  default:\r
-    ASSERT (FALSE);\r
-    gBS->FreePool (Private->LineBuffer);\r
-    Private->LineBuffer = NULL;\r
-    return EFI_DEVICE_ERROR;\r
+    case QEMU_VIDEO_CIRRUS_5430:\r
+    case QEMU_VIDEO_CIRRUS_5446:\r
+      InitializeCirrusGraphicsMode (Private, &QemuVideoCirrusModes[ModeData->InternalModeIndex]);\r
+      break;\r
+    case QEMU_VIDEO_BOCHS_MMIO:\r
+    case QEMU_VIDEO_BOCHS:\r
+      InitializeBochsGraphicsMode (Private, ModeData);\r
+      break;\r
+    default:\r
+      ASSERT (FALSE);\r
+      return EFI_DEVICE_ERROR;\r
   }\r
 \r
-  This->Mode->Mode = ModeNumber;\r
+  This->Mode->Mode                       = ModeNumber;\r
   This->Mode->Info->HorizontalResolution = ModeData->HorizontalResolution;\r
-  This->Mode->Info->VerticalResolution = ModeData->VerticalResolution;\r
-  This->Mode->SizeOfInfo = sizeof(EFI_GRAPHICS_OUTPUT_MODE_INFORMATION);\r
+  This->Mode->Info->VerticalResolution   = ModeData->VerticalResolution;\r
+  This->Mode->SizeOfInfo                 = sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION);\r
 \r
   QemuVideoCompleteModeData (Private, This->Mode);\r
 \r
-  BltLibConfigure (\r
-    (VOID*)(UINTN) This->Mode->FrameBufferBase,\r
-    This->Mode->Info\r
-    );\r
+  //\r
+  // Re-initialize the frame buffer configure when mode changes.\r
+  //\r
+  Status = FrameBufferBltConfigure (\r
+             (VOID *)(UINTN)This->Mode->FrameBufferBase,\r
+             This->Mode->Info,\r
+             Private->FrameBufferBltConfigure,\r
+             &Private->FrameBufferBltConfigureSize\r
+             );\r
+  if (Status == RETURN_BUFFER_TOO_SMALL) {\r
+    //\r
+    // Frame buffer configure may be larger in new mode.\r
+    //\r
+    if (Private->FrameBufferBltConfigure != NULL) {\r
+      FreePool (Private->FrameBufferBltConfigure);\r
+    }\r
+\r
+    Private->FrameBufferBltConfigure =\r
+      AllocatePool (Private->FrameBufferBltConfigureSize);\r
+    ASSERT (Private->FrameBufferBltConfigure != NULL);\r
+\r
+    //\r
+    // Create the configuration for FrameBufferBltLib\r
+    //\r
+    Status = FrameBufferBltConfigure (\r
+               (VOID *)(UINTN)This->Mode->FrameBufferBase,\r
+               This->Mode->Info,\r
+               Private->FrameBufferBltConfigure,\r
+               &Private->FrameBufferBltConfigureSize\r
+               );\r
+  }\r
 \r
-  Private->HardwareNeedsStarting  = FALSE;\r
+  ASSERT (Status == RETURN_SUCCESS);\r
+\r
+  //\r
+  // Per UEFI Spec, need to clear the visible portions of the output display to black.\r
+  //\r
+  ZeroMem (&Black, sizeof (Black));\r
+  Status = FrameBufferBlt (\r
+             Private->FrameBufferBltConfigure,\r
+             &Black,\r
+             EfiBltVideoFill,\r
+             0,\r
+             0,\r
+             0,\r
+             0,\r
+             This->Mode->Info->HorizontalResolution,\r
+             This->Mode->Info->VerticalResolution,\r
+             0\r
+             );\r
+  ASSERT_RETURN_ERROR (Status);\r
 \r
   return EFI_SUCCESS;\r
 }\r
@@ -216,17 +256,18 @@ Routine Description:
 EFI_STATUS\r
 EFIAPI\r
 QemuVideoGraphicsOutputBlt (\r
-  IN  EFI_GRAPHICS_OUTPUT_PROTOCOL          *This,\r
-  IN  EFI_GRAPHICS_OUTPUT_BLT_PIXEL         *BltBuffer, OPTIONAL\r
-  IN  EFI_GRAPHICS_OUTPUT_BLT_OPERATION     BltOperation,\r
-  IN  UINTN                                 SourceX,\r
-  IN  UINTN                                 SourceY,\r
-  IN  UINTN                                 DestinationX,\r
-  IN  UINTN                                 DestinationY,\r
-  IN  UINTN                                 Width,\r
-  IN  UINTN                                 Height,\r
-  IN  UINTN                                 Delta\r
+  IN  EFI_GRAPHICS_OUTPUT_PROTOCOL       *This,\r
+  IN  EFI_GRAPHICS_OUTPUT_BLT_PIXEL      *BltBuffer  OPTIONAL,\r
+  IN  EFI_GRAPHICS_OUTPUT_BLT_OPERATION  BltOperation,\r
+  IN  UINTN                              SourceX,\r
+  IN  UINTN                              SourceY,\r
+  IN  UINTN                              DestinationX,\r
+  IN  UINTN                              DestinationY,\r
+  IN  UINTN                              Width,\r
+  IN  UINTN                              Height,\r
+  IN  UINTN                              Delta\r
   )\r
+\r
 /*++\r
 \r
 Routine Description:\r
@@ -256,9 +297,11 @@ Returns:
 \r
 --*/\r
 {\r
-  EFI_STATUS                      Status;\r
-  EFI_TPL                         OriginalTPL;\r
+  EFI_STATUS               Status;\r
+  EFI_TPL                  OriginalTPL;\r
+  QEMU_VIDEO_PRIVATE_DATA  *Private;\r
 \r
+  Private = QEMU_VIDEO_PRIVATE_DATA_FROM_GRAPHICS_OUTPUT_THIS (This);\r
   //\r
   // We have to raise to TPL Notify, so we make an atomic write the frame buffer.\r
   // We would not want a timer based event (Cursor, ...) to come in while we are\r
@@ -267,26 +310,27 @@ Returns:
   OriginalTPL = gBS->RaiseTPL (TPL_NOTIFY);\r
 \r
   switch (BltOperation) {\r
-  case EfiBltVideoToBltBuffer:\r
-  case EfiBltBufferToVideo:\r
-  case EfiBltVideoFill:\r
-  case EfiBltVideoToVideo:\r
-    Status = BltLibGopBlt (\r
-      BltBuffer,\r
-      BltOperation,\r
-      SourceX,\r
-      SourceY,\r
-      DestinationX,\r
-      DestinationY,\r
-      Width,\r
-      Height,\r
-      Delta\r
-      );\r
-    break;\r
-\r
-  default:\r
-    Status = EFI_INVALID_PARAMETER;\r
-    ASSERT (FALSE);\r
+    case EfiBltVideoToBltBuffer:\r
+    case EfiBltBufferToVideo:\r
+    case EfiBltVideoFill:\r
+    case EfiBltVideoToVideo:\r
+      Status = FrameBufferBlt (\r
+                 Private->FrameBufferBltConfigure,\r
+                 BltBuffer,\r
+                 BltOperation,\r
+                 SourceX,\r
+                 SourceY,\r
+                 DestinationX,\r
+                 DestinationY,\r
+                 Width,\r
+                 Height,\r
+                 Delta\r
+                 );\r
+      break;\r
+\r
+    default:\r
+      Status = EFI_INVALID_PARAMETER;\r
+      break;\r
   }\r
 \r
   gBS->RestoreTPL (OriginalTPL);\r
@@ -299,9 +343,8 @@ QemuVideoGraphicsOutputConstructor (
   QEMU_VIDEO_PRIVATE_DATA  *Private\r
   )\r
 {\r
-  EFI_STATUS                   Status;\r
-  EFI_GRAPHICS_OUTPUT_PROTOCOL *GraphicsOutput;\r
-\r
+  EFI_STATUS                    Status;\r
+  EFI_GRAPHICS_OUTPUT_PROTOCOL  *GraphicsOutput;\r
 \r
   GraphicsOutput            = &Private->GraphicsOutput;\r
   GraphicsOutput->QueryMode = QemuVideoGraphicsOutputQueryMode;\r
@@ -314,28 +357,34 @@ QemuVideoGraphicsOutputConstructor (
   Status = gBS->AllocatePool (\r
                   EfiBootServicesData,\r
                   sizeof (EFI_GRAPHICS_OUTPUT_PROTOCOL_MODE),\r
-                  (VOID **) &Private->GraphicsOutput.Mode\r
+                  (VOID **)&Private->GraphicsOutput.Mode\r
                   );\r
   if (EFI_ERROR (Status)) {\r
     return Status;\r
   }\r
+\r
   Status = gBS->AllocatePool (\r
                   EfiBootServicesData,\r
                   sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION),\r
-                  (VOID **) &Private->GraphicsOutput.Mode->Info\r
+                  (VOID **)&Private->GraphicsOutput.Mode->Info\r
                   );\r
   if (EFI_ERROR (Status)) {\r
-    return Status;\r
+    goto FreeMode;\r
   }\r
-  Private->GraphicsOutput.Mode->MaxMode = (UINT32) Private->MaxMode;\r
+\r
+  Private->GraphicsOutput.Mode->MaxMode = (UINT32)Private->MaxMode;\r
   Private->GraphicsOutput.Mode->Mode    = GRAPHICS_OUTPUT_INVALIDE_MODE_NUMBER;\r
-  Private->HardwareNeedsStarting        = TRUE;\r
-  Private->LineBuffer                   = NULL;\r
+  Private->FrameBufferBltConfigure      = NULL;\r
+  Private->FrameBufferBltConfigureSize  = 0;\r
 \r
   //\r
   // Initialize the hardware\r
   //\r
-  GraphicsOutput->SetMode (GraphicsOutput, 0);\r
+  Status = GraphicsOutput->SetMode (GraphicsOutput, 0);\r
+  if (EFI_ERROR (Status)) {\r
+    goto FreeInfo;\r
+  }\r
+\r
   DrawLogo (\r
     Private,\r
     Private->ModeData[Private->GraphicsOutput.Mode->Mode].HorizontalResolution,\r
@@ -343,12 +392,22 @@ QemuVideoGraphicsOutputConstructor (
     );\r
 \r
   return EFI_SUCCESS;\r
+\r
+FreeInfo:\r
+  FreePool (Private->GraphicsOutput.Mode->Info);\r
+\r
+FreeMode:\r
+  FreePool (Private->GraphicsOutput.Mode);\r
+  Private->GraphicsOutput.Mode = NULL;\r
+\r
+  return Status;\r
 }\r
 \r
 EFI_STATUS\r
 QemuVideoGraphicsOutputDestructor (\r
   QEMU_VIDEO_PRIVATE_DATA  *Private\r
   )\r
+\r
 /*++\r
 \r
 Routine Description:\r
@@ -361,14 +420,17 @@ Returns:
 \r
 --*/\r
 {\r
+  if (Private->FrameBufferBltConfigure != NULL) {\r
+    FreePool (Private->FrameBufferBltConfigure);\r
+  }\r
+\r
   if (Private->GraphicsOutput.Mode != NULL) {\r
     if (Private->GraphicsOutput.Mode->Info != NULL) {\r
       gBS->FreePool (Private->GraphicsOutput.Mode->Info);\r
     }\r
+\r
     gBS->FreePool (Private->GraphicsOutput.Mode);\r
   }\r
 \r
   return EFI_SUCCESS;\r
 }\r
-\r
-\r