| /** @file | |
| Graphics Output Protocol functions for the QEMU video controller. | |
| Copyright (c) 2007 - 2018, Intel Corporation. All rights reserved.<BR> | |
| SPDX-License-Identifier: BSD-2-Clause-Patent | |
| **/ | |
| #include "Qemu.h" | |
| #include <Library/DxeServicesTableLib.h> | |
| STATIC | |
| VOID | |
| QemuVideoCompleteModeInfo ( | |
| IN QEMU_VIDEO_MODE_DATA *ModeData, | |
| OUT EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *Info | |
| ) | |
| { | |
| Info->Version = 0; | |
| if (ModeData->ColorDepth == 8) { | |
| Info->PixelFormat = PixelBitMask; | |
| Info->PixelInformation.RedMask = PIXEL_RED_MASK; | |
| Info->PixelInformation.GreenMask = PIXEL_GREEN_MASK; | |
| Info->PixelInformation.BlueMask = PIXEL_BLUE_MASK; | |
| Info->PixelInformation.ReservedMask = 0; | |
| } else if (ModeData->ColorDepth == 24) { | |
| Info->PixelFormat = PixelBitMask; | |
| Info->PixelInformation.RedMask = PIXEL24_RED_MASK; | |
| Info->PixelInformation.GreenMask = PIXEL24_GREEN_MASK; | |
| Info->PixelInformation.BlueMask = PIXEL24_BLUE_MASK; | |
| Info->PixelInformation.ReservedMask = 0; | |
| } else if (ModeData->ColorDepth == 32) { | |
| DEBUG ((DEBUG_INFO, "PixelBlueGreenRedReserved8BitPerColor\n")); | |
| Info->PixelFormat = PixelBlueGreenRedReserved8BitPerColor; | |
| Info->PixelInformation.RedMask = 0; | |
| Info->PixelInformation.GreenMask = 0; | |
| Info->PixelInformation.BlueMask = 0; | |
| Info->PixelInformation.ReservedMask = 0; | |
| } else { | |
| DEBUG ((DEBUG_ERROR, "%a: Invalid ColorDepth %u", __func__, ModeData->ColorDepth)); | |
| ASSERT (FALSE); | |
| } | |
| Info->PixelsPerScanLine = Info->HorizontalResolution; | |
| } | |
| STATIC | |
| EFI_STATUS | |
| QemuVideoCompleteModeData ( | |
| IN QEMU_VIDEO_PRIVATE_DATA *Private, | |
| OUT EFI_GRAPHICS_OUTPUT_PROTOCOL_MODE *Mode | |
| ) | |
| { | |
| EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *Info; | |
| EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *FrameBufDesc; | |
| QEMU_VIDEO_MODE_DATA *ModeData; | |
| EFI_STATUS Status; | |
| ModeData = &Private->ModeData[Mode->Mode]; | |
| Info = Mode->Info; | |
| QemuVideoCompleteModeInfo (ModeData, Info); | |
| Private->PciIo->GetBarAttributes ( | |
| Private->PciIo, | |
| Private->FrameBufferVramBarIndex, | |
| NULL, | |
| (VOID **)&FrameBufDesc | |
| ); | |
| Mode->FrameBufferBase = FrameBufDesc->AddrRangeMin; | |
| Mode->FrameBufferSize = Info->HorizontalResolution * Info->VerticalResolution; | |
| Mode->FrameBufferSize = Mode->FrameBufferSize * ((ModeData->ColorDepth + 7) / 8); | |
| Mode->FrameBufferSize = EFI_PAGES_TO_SIZE ( | |
| EFI_SIZE_TO_PAGES (Mode->FrameBufferSize) | |
| ); | |
| DEBUG (( | |
| DEBUG_INFO, | |
| "FrameBufferBase: 0x%Lx, FrameBufferSize: 0x%Lx\n", | |
| Mode->FrameBufferBase, | |
| (UINT64)Mode->FrameBufferSize | |
| )); | |
| if (FeaturePcdGet (PcdRemapFrameBufferWriteCombine)) { | |
| Status = gDS->SetMemorySpaceCapabilities ( | |
| FrameBufDesc->AddrRangeMin, | |
| FrameBufDesc->AddrLen, | |
| EFI_MEMORY_UC | EFI_MEMORY_WC | EFI_MEMORY_XP | |
| ); | |
| ASSERT_EFI_ERROR (Status); | |
| Status = gDS->SetMemorySpaceAttributes ( | |
| FrameBufDesc->AddrRangeMin, | |
| FrameBufDesc->AddrLen, | |
| EFI_MEMORY_WC | EFI_MEMORY_XP | |
| ); | |
| ASSERT_EFI_ERROR (Status); | |
| } | |
| FreePool (FrameBufDesc); | |
| return EFI_SUCCESS; | |
| } | |
| // | |
| // Graphics Output Protocol Member Functions | |
| // | |
| EFI_STATUS | |
| EFIAPI | |
| QemuVideoGraphicsOutputQueryMode ( | |
| IN EFI_GRAPHICS_OUTPUT_PROTOCOL *This, | |
| IN UINT32 ModeNumber, | |
| OUT UINTN *SizeOfInfo, | |
| OUT EFI_GRAPHICS_OUTPUT_MODE_INFORMATION **Info | |
| ) | |
| /*++ | |
| Routine Description: | |
| Graphics Output protocol interface to query video mode | |
| Arguments: | |
| This - Protocol instance pointer. | |
| ModeNumber - The mode number to return information on. | |
| Info - Caller allocated buffer that returns information about ModeNumber. | |
| SizeOfInfo - A pointer to the size, in bytes, of the Info buffer. | |
| Returns: | |
| EFI_SUCCESS - Mode information returned. | |
| EFI_BUFFER_TOO_SMALL - The Info buffer was too small. | |
| EFI_DEVICE_ERROR - A hardware error occurred trying to retrieve the video mode. | |
| EFI_NOT_STARTED - Video display is not initialized. Call SetMode () | |
| EFI_INVALID_PARAMETER - One of the input args was NULL. | |
| --*/ | |
| { | |
| QEMU_VIDEO_PRIVATE_DATA *Private; | |
| QEMU_VIDEO_MODE_DATA *ModeData; | |
| Private = QEMU_VIDEO_PRIVATE_DATA_FROM_GRAPHICS_OUTPUT_THIS (This); | |
| if ((Info == NULL) || (SizeOfInfo == NULL) || (ModeNumber >= This->Mode->MaxMode)) { | |
| return EFI_INVALID_PARAMETER; | |
| } | |
| *Info = AllocatePool (sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION)); | |
| if (*Info == NULL) { | |
| return EFI_OUT_OF_RESOURCES; | |
| } | |
| *SizeOfInfo = sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION); | |
| ModeData = &Private->ModeData[ModeNumber]; | |
| (*Info)->HorizontalResolution = ModeData->HorizontalResolution; | |
| (*Info)->VerticalResolution = ModeData->VerticalResolution; | |
| QemuVideoCompleteModeInfo (ModeData, *Info); | |
| return EFI_SUCCESS; | |
| } | |
| EFI_STATUS | |
| EFIAPI | |
| QemuVideoGraphicsOutputSetMode ( | |
| IN EFI_GRAPHICS_OUTPUT_PROTOCOL *This, | |
| IN UINT32 ModeNumber | |
| ) | |
| /*++ | |
| Routine Description: | |
| Graphics Output protocol interface to set video mode | |
| Arguments: | |
| This - Protocol instance pointer. | |
| ModeNumber - The mode number to be set. | |
| Returns: | |
| EFI_SUCCESS - Graphics mode was changed. | |
| EFI_DEVICE_ERROR - The device had an error and could not complete the request. | |
| EFI_UNSUPPORTED - ModeNumber is not supported by this device. | |
| --*/ | |
| { | |
| QEMU_VIDEO_PRIVATE_DATA *Private; | |
| QEMU_VIDEO_MODE_DATA *ModeData; | |
| RETURN_STATUS Status; | |
| EFI_GRAPHICS_OUTPUT_BLT_PIXEL Black; | |
| Private = QEMU_VIDEO_PRIVATE_DATA_FROM_GRAPHICS_OUTPUT_THIS (This); | |
| if (ModeNumber >= This->Mode->MaxMode) { | |
| return EFI_UNSUPPORTED; | |
| } | |
| ModeData = &Private->ModeData[ModeNumber]; | |
| switch (Private->Variant) { | |
| case QEMU_VIDEO_CIRRUS_5430: | |
| case QEMU_VIDEO_CIRRUS_5446: | |
| InitializeCirrusGraphicsMode (Private, &QemuVideoCirrusModes[ModeData->InternalModeIndex]); | |
| break; | |
| case QEMU_VIDEO_BOCHS_MMIO: | |
| case QEMU_VIDEO_BOCHS: | |
| InitializeBochsGraphicsMode (Private, ModeData); | |
| break; | |
| default: | |
| ASSERT (FALSE); | |
| return EFI_DEVICE_ERROR; | |
| } | |
| This->Mode->Mode = ModeNumber; | |
| This->Mode->Info->HorizontalResolution = ModeData->HorizontalResolution; | |
| This->Mode->Info->VerticalResolution = ModeData->VerticalResolution; | |
| This->Mode->SizeOfInfo = sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION); | |
| QemuVideoCompleteModeData (Private, This->Mode); | |
| // | |
| // Re-initialize the frame buffer configure when mode changes. | |
| // | |
| Status = FrameBufferBltConfigure ( | |
| (VOID *)(UINTN)This->Mode->FrameBufferBase, | |
| This->Mode->Info, | |
| Private->FrameBufferBltConfigure, | |
| &Private->FrameBufferBltConfigureSize | |
| ); | |
| if (Status == RETURN_BUFFER_TOO_SMALL) { | |
| // | |
| // Frame buffer configure may be larger in new mode. | |
| // | |
| if (Private->FrameBufferBltConfigure != NULL) { | |
| FreePool (Private->FrameBufferBltConfigure); | |
| } | |
| Private->FrameBufferBltConfigure = | |
| AllocatePool (Private->FrameBufferBltConfigureSize); | |
| ASSERT (Private->FrameBufferBltConfigure != NULL); | |
| // | |
| // Create the configuration for FrameBufferBltLib | |
| // | |
| Status = FrameBufferBltConfigure ( | |
| (VOID *)(UINTN)This->Mode->FrameBufferBase, | |
| This->Mode->Info, | |
| Private->FrameBufferBltConfigure, | |
| &Private->FrameBufferBltConfigureSize | |
| ); | |
| } | |
| ASSERT (Status == RETURN_SUCCESS); | |
| // | |
| // Per UEFI Spec, need to clear the visible portions of the output display to black. | |
| // | |
| ZeroMem (&Black, sizeof (Black)); | |
| Status = FrameBufferBlt ( | |
| Private->FrameBufferBltConfigure, | |
| &Black, | |
| EfiBltVideoFill, | |
| 0, | |
| 0, | |
| 0, | |
| 0, | |
| This->Mode->Info->HorizontalResolution, | |
| This->Mode->Info->VerticalResolution, | |
| 0 | |
| ); | |
| ASSERT_RETURN_ERROR (Status); | |
| return EFI_SUCCESS; | |
| } | |
| EFI_STATUS | |
| EFIAPI | |
| QemuVideoGraphicsOutputBlt ( | |
| IN EFI_GRAPHICS_OUTPUT_PROTOCOL *This, | |
| IN EFI_GRAPHICS_OUTPUT_BLT_PIXEL *BltBuffer OPTIONAL, | |
| IN EFI_GRAPHICS_OUTPUT_BLT_OPERATION BltOperation, | |
| IN UINTN SourceX, | |
| IN UINTN SourceY, | |
| IN UINTN DestinationX, | |
| IN UINTN DestinationY, | |
| IN UINTN Width, | |
| IN UINTN Height, | |
| IN UINTN Delta | |
| ) | |
| /*++ | |
| Routine Description: | |
| Graphics Output protocol instance to block transfer for CirrusLogic device | |
| Arguments: | |
| This - Pointer to Graphics Output protocol instance | |
| BltBuffer - The data to transfer to screen | |
| BltOperation - The operation to perform | |
| SourceX - The X coordinate of the source for BltOperation | |
| SourceY - The Y coordinate of the source for BltOperation | |
| DestinationX - The X coordinate of the destination for BltOperation | |
| DestinationY - The Y coordinate of the destination for BltOperation | |
| Width - The width of a rectangle in the blt rectangle in pixels | |
| Height - The height of a rectangle in the blt rectangle in pixels | |
| Delta - Not used for EfiBltVideoFill and EfiBltVideoToVideo operation. | |
| If a Delta of 0 is used, the entire BltBuffer will be operated on. | |
| If a subrectangle of the BltBuffer is used, then Delta represents | |
| the number of bytes in a row of the BltBuffer. | |
| Returns: | |
| EFI_INVALID_PARAMETER - Invalid parameter passed in | |
| EFI_SUCCESS - Blt operation success | |
| --*/ | |
| { | |
| EFI_STATUS Status; | |
| EFI_TPL OriginalTPL; | |
| QEMU_VIDEO_PRIVATE_DATA *Private; | |
| Private = QEMU_VIDEO_PRIVATE_DATA_FROM_GRAPHICS_OUTPUT_THIS (This); | |
| // | |
| // We have to raise to TPL Notify, so we make an atomic write the frame buffer. | |
| // We would not want a timer based event (Cursor, ...) to come in while we are | |
| // doing this operation. | |
| // | |
| OriginalTPL = gBS->RaiseTPL (TPL_NOTIFY); | |
| switch (BltOperation) { | |
| case EfiBltVideoToBltBuffer: | |
| case EfiBltBufferToVideo: | |
| case EfiBltVideoFill: | |
| case EfiBltVideoToVideo: | |
| Status = FrameBufferBlt ( | |
| Private->FrameBufferBltConfigure, | |
| BltBuffer, | |
| BltOperation, | |
| SourceX, | |
| SourceY, | |
| DestinationX, | |
| DestinationY, | |
| Width, | |
| Height, | |
| Delta | |
| ); | |
| break; | |
| default: | |
| Status = EFI_INVALID_PARAMETER; | |
| break; | |
| } | |
| gBS->RestoreTPL (OriginalTPL); | |
| return Status; | |
| } | |
| EFI_STATUS | |
| QemuVideoGraphicsOutputConstructor ( | |
| QEMU_VIDEO_PRIVATE_DATA *Private | |
| ) | |
| { | |
| EFI_STATUS Status; | |
| EFI_GRAPHICS_OUTPUT_PROTOCOL *GraphicsOutput; | |
| GraphicsOutput = &Private->GraphicsOutput; | |
| GraphicsOutput->QueryMode = QemuVideoGraphicsOutputQueryMode; | |
| GraphicsOutput->SetMode = QemuVideoGraphicsOutputSetMode; | |
| GraphicsOutput->Blt = QemuVideoGraphicsOutputBlt; | |
| // | |
| // Initialize the private data | |
| // | |
| Status = gBS->AllocatePool ( | |
| EfiBootServicesData, | |
| sizeof (EFI_GRAPHICS_OUTPUT_PROTOCOL_MODE), | |
| (VOID **)&Private->GraphicsOutput.Mode | |
| ); | |
| if (EFI_ERROR (Status)) { | |
| return Status; | |
| } | |
| Status = gBS->AllocatePool ( | |
| EfiBootServicesData, | |
| sizeof (EFI_GRAPHICS_OUTPUT_MODE_INFORMATION), | |
| (VOID **)&Private->GraphicsOutput.Mode->Info | |
| ); | |
| if (EFI_ERROR (Status)) { | |
| goto FreeMode; | |
| } | |
| Private->GraphicsOutput.Mode->MaxMode = (UINT32)Private->MaxMode; | |
| Private->GraphicsOutput.Mode->Mode = GRAPHICS_OUTPUT_INVALIDE_MODE_NUMBER; | |
| Private->FrameBufferBltConfigure = NULL; | |
| Private->FrameBufferBltConfigureSize = 0; | |
| // | |
| // Initialize the hardware | |
| // | |
| Status = GraphicsOutput->SetMode (GraphicsOutput, 0); | |
| if (EFI_ERROR (Status)) { | |
| goto FreeInfo; | |
| } | |
| DrawLogo ( | |
| Private, | |
| Private->ModeData[Private->GraphicsOutput.Mode->Mode].HorizontalResolution, | |
| Private->ModeData[Private->GraphicsOutput.Mode->Mode].VerticalResolution | |
| ); | |
| return EFI_SUCCESS; | |
| FreeInfo: | |
| FreePool (Private->GraphicsOutput.Mode->Info); | |
| FreeMode: | |
| FreePool (Private->GraphicsOutput.Mode); | |
| Private->GraphicsOutput.Mode = NULL; | |
| return Status; | |
| } | |
| EFI_STATUS | |
| QemuVideoGraphicsOutputDestructor ( | |
| QEMU_VIDEO_PRIVATE_DATA *Private | |
| ) | |
| /*++ | |
| Routine Description: | |
| Arguments: | |
| Returns: | |
| None | |
| --*/ | |
| { | |
| if (Private->FrameBufferBltConfigure != NULL) { | |
| FreePool (Private->FrameBufferBltConfigure); | |
| } | |
| if (Private->GraphicsOutput.Mode != NULL) { | |
| if (Private->GraphicsOutput.Mode->Info != NULL) { | |
| gBS->FreePool (Private->GraphicsOutput.Mode->Info); | |
| } | |
| gBS->FreePool (Private->GraphicsOutput.Mode); | |
| } | |
| return EFI_SUCCESS; | |
| } |