2023-04-13 13:16:14 +06:00

645 lines
18 KiB
C

/** @file
Copyright (C) 2019, vit9696. All rights reserved.
All rights reserved.
This program and the accompanying materials
are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution. The full text of the license may be found at
http://opensource.org/licenses/bsd-license.php
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
**/
#include "OcConsoleLibInternal.h"
#include "ConsoleGopInternal.h"
#include <Protocol/AppleEg2Info.h>
#include <Protocol/ConsoleControl.h>
#include <Protocol/GraphicsOutput.h>
#include <Protocol/SimpleTextOut.h>
#include <Library/BaseMemoryLib.h>
#include <Library/BaseOverflowLib.h>
#include <Library/BaseLib.h>
#include <Library/DebugLib.h>
#include <Library/OcBlitLib.h>
#include <Library/MemoryAllocationLib.h>
#include <Library/MtrrLib.h>
#include <Library/OcConsoleLib.h>
#include <Library/OcMiscLib.h>
#include <Library/UefiBootServicesTableLib.h>
#include <Library/UefiRuntimeServicesTableLib.h>
STATIC CONSOLE_GOP_CONTEXT mGop;
STATIC
EFI_STATUS
EFIAPI
ConsoleHandleProtocol (
IN EFI_HANDLE Handle,
IN EFI_GUID *Protocol,
OUT VOID **Interface
)
{
EFI_STATUS Status;
Status = mGop.OriginalHandleProtocol (Handle, Protocol, Interface);
if ((Status != EFI_UNSUPPORTED) && EFI_ERROR (Status)) {
return Status;
}
if ((mGop.ConsoleGop != NULL) && CompareGuid (&gEfiGraphicsOutputProtocolGuid, Protocol)) {
*Interface = mGop.ConsoleGop;
return EFI_SUCCESS;
}
if (!EFI_ERROR (Status)) {
return Status;
}
if (CompareGuid (&gEfiUgaDrawProtocolGuid, Protocol)) {
//
// EfiBoot from 10.4 can only use UgaDraw protocol.
//
Status = gBS->LocateProtocol (
&gEfiUgaDrawProtocolGuid,
NULL,
Interface
);
if (!EFI_ERROR (Status)) {
return EFI_SUCCESS;
}
}
return EFI_UNSUPPORTED;
}
EFI_STATUS
OcProvideConsoleGop (
IN BOOLEAN Route
)
{
EFI_STATUS Status;
EFI_GRAPHICS_OUTPUT_PROTOCOL *OriginalGop;
EFI_GRAPHICS_OUTPUT_PROTOCOL *Gop;
UINTN HandleCount;
EFI_HANDLE *HandleBuffer;
UINTN Index;
//
// Shell may replace gST->ConsoleOutHandle, so we have to ensure
// that HandleProtocol always reports valid chosen GOP.
//
if (Route) {
mGop.OriginalHandleProtocol = gBS->HandleProtocol;
gBS->HandleProtocol = ConsoleHandleProtocol;
gBS->Hdr.CRC32 = 0;
gBS->CalculateCrc32 (gBS, gBS->Hdr.HeaderSize, &gBS->Hdr.CRC32);
}
OriginalGop = NULL;
Status = gBS->HandleProtocol (
gST->ConsoleOutHandle,
&gEfiGraphicsOutputProtocolGuid,
(VOID **)&OriginalGop
);
if (!EFI_ERROR (Status)) {
DEBUG ((
DEBUG_INFO,
"OCC: GOP exists on ConsoleOutHandle and has %u modes\n",
(UINT32)OriginalGop->Mode->MaxMode
));
if (OriginalGop->Mode->Info->PixelFormat == PixelBltOnly) {
//
// ASUS Z690F has GOP aggregator on console handle with blit only
// formats. This GOP obviously does not work on macOS, so we need
// to uninstall it, but only when we have an alternative.
//
DEBUG ((
DEBUG_INFO,
"OCC: Looking for GOP replacement due to blit-only GOP\n"
));
} else if (OriginalGop->Mode->MaxMode == 0) {
//
// No modes on MacPro5,1 with Mac EFI incompatible GPU.
// Here we need to uninstall ConOut GOP in favour of GPU GOP.
//
DEBUG ((
DEBUG_INFO,
"OCC: Looking for GOP replacement due to invalid mode count\n"
));
} else {
mGop.ConsoleGop = OriginalGop;
return EFI_ALREADY_STARTED;
}
Status = gBS->LocateHandleBuffer (
ByProtocol,
&gEfiGraphicsOutputProtocolGuid,
NULL,
&HandleCount,
&HandleBuffer
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OCC: No handles with GOP protocol - %r\n", Status));
return EFI_UNSUPPORTED;
}
Status = EFI_NOT_FOUND;
for (Index = 0; Index < HandleCount; ++Index) {
if (HandleBuffer[Index] != gST->ConsoleOutHandle) {
Status = gBS->HandleProtocol (
HandleBuffer[Index],
&gEfiGraphicsOutputProtocolGuid,
(VOID **)&Gop
);
break;
}
}
DEBUG ((DEBUG_INFO, "OCC: Alternative GOP status is - %r\n", Status));
FreePool (HandleBuffer);
if ( !EFI_ERROR (Status)
&& (OriginalGop->Mode->Info->PixelFormat == PixelBltOnly))
{
if ((UINT32)Gop->Mode->Info->PixelFormat >= PixelBltOnly) {
Status = EFI_NOT_FOUND;
}
DEBUG ((
DEBUG_INFO,
"OCC: Checking alternative GOP mode %u - %r\n",
Gop->Mode->Info->PixelFormat,
Status
));
//
// We cannot uninstall this GOP as ASUS will reinstall it.
// Hook HandleProtocol instead.
//
if (!EFI_ERROR (Status)) {
mGop.ConsoleGop = Gop;
return EFI_SUCCESS;
}
}
if (!EFI_ERROR (Status)) {
gBS->UninstallProtocolInterface (
gST->ConsoleOutHandle,
&gEfiGraphicsOutputProtocolGuid,
OriginalGop
);
}
} else {
DEBUG ((DEBUG_INFO, "OCC: Installing GOP (%r) on ConsoleOutHandle...\n", Status));
Status = gBS->LocateProtocol (&gEfiGraphicsOutputProtocolGuid, NULL, (VOID **)&Gop);
}
if (!EFI_ERROR (Status)) {
Status = gBS->InstallMultipleProtocolInterfaces (
&gST->ConsoleOutHandle,
&gEfiGraphicsOutputProtocolGuid,
Gop,
NULL
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_WARN, "OCC: Failed to install GOP on ConsoleOutHandle - %r\n", Status));
}
mGop.ConsoleGop = Gop;
} else {
DEBUG ((DEBUG_WARN, "OCC: Missing compatible GOP - %r\n", Status));
}
return Status;
}
/**
Update current GOP mode to represent either custom (rotated)
or original mode. In general custom mode is used, but to
call the original functions it is safer to switch to original.
@param[in,out] This GOP protocol to update.
@param[in] Source Source mode.
**/
STATIC
VOID
SwitchMode (
IN OUT EFI_GRAPHICS_OUTPUT_PROTOCOL *This,
IN BOOLEAN UseCustom
)
{
EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *Source;
ASSERT (This != NULL);
ASSERT (This->Mode != NULL);
ASSERT (This->Mode->Info != NULL);
if (UseCustom) {
Source = &mGop.CustomModeInfo;
This->Mode->FrameBufferBase = 0;
This->Mode->FrameBufferSize = 0;
} else {
Source = &mGop.OriginalModeInfo;
This->Mode->FrameBufferBase = mGop.OriginalFrameBufferBase;
This->Mode->FrameBufferSize = mGop.OriginalFrameBufferSize;
}
This->Mode->Info->VerticalResolution = Source->VerticalResolution;
This->Mode->Info->HorizontalResolution = Source->HorizontalResolution;
This->Mode->Info->PixelsPerScanLine = Source->PixelsPerScanLine;
}
/**
Translate current GOP mode to custom (rotated) mode.
Both original and custom modes are saved.
@param[in,out] This GOP protocol to update.
@param[in] Rotation Rotation angle.
**/
STATIC
VOID
RotateMode (
IN OUT EFI_GRAPHICS_OUTPUT_PROTOCOL *This,
IN UINT32 Rotation
)
{
ASSERT (This != NULL);
ASSERT (This->Mode != NULL);
ASSERT (This->Mode->Info != NULL);
CopyMem (&mGop.OriginalModeInfo, This->Mode->Info, sizeof (mGop.OriginalModeInfo));
if ((Rotation == 90) || (Rotation == 270)) {
This->Mode->Info->HorizontalResolution = mGop.OriginalModeInfo.VerticalResolution;
This->Mode->Info->VerticalResolution = mGop.OriginalModeInfo.HorizontalResolution;
This->Mode->Info->PixelsPerScanLine = This->Mode->Info->HorizontalResolution;
}
mGop.OriginalFrameBufferBase = This->Mode->FrameBufferBase;
mGop.OriginalFrameBufferSize = This->Mode->FrameBufferSize;
if (Rotation != 0) {
//
// macOS requires FrameBufferBase to be 0 for rotation to work, which
// forces it inspect the AppleFramebufferInfo protocol.
// It also requires PixelFormat to be <= PixelBitMask, otherwise Apple
// logo will not show.
// REF: https://github.com/acidanthera/bugtracker/issues/1498#issuecomment-782822654
//
// Windows and Linux bootloaders only draw directly to the framebuffer.
//
This->Mode->FrameBufferBase = 0;
This->Mode->FrameBufferSize = 0;
}
CopyMem (&mGop.CustomModeInfo, This->Mode->Info, sizeof (mGop.CustomModeInfo));
}
STATIC
OC_BLIT_CONFIGURE *
EFIAPI
DirectGopFromTarget (
IN EFI_PHYSICAL_ADDRESS FramebufferBase,
IN EFI_GRAPHICS_OUTPUT_MODE_INFORMATION *Info,
OUT UINTN *PageCount
)
{
EFI_STATUS Status;
UINTN ConfigureSize;
OC_BLIT_CONFIGURE *Context;
ConfigureSize = 0;
Status = OcBlitConfigure (
(VOID *)(UINTN)FramebufferBase,
Info,
mGop.Rotation,
NULL,
&ConfigureSize
);
if (Status != EFI_BUFFER_TOO_SMALL) {
return NULL;
}
*PageCount = EFI_SIZE_TO_PAGES (ConfigureSize);
Context = AllocatePages (*PageCount);
if (Context == NULL) {
return NULL;
}
Status = OcBlitConfigure (
(VOID *)(UINTN)FramebufferBase,
Info,
mGop.Rotation,
Context,
&ConfigureSize
);
if (EFI_ERROR (Status)) {
FreePages (Context, *PageCount);
return NULL;
}
return Context;
}
STATIC
EFI_STATUS
EFIAPI
DirectGopSetMode (
IN EFI_GRAPHICS_OUTPUT_PROTOCOL *This,
IN UINT32 ModeNumber
)
{
EFI_STATUS Status;
EFI_TPL OldTpl;
OC_BLIT_CONFIGURE *Original;
if (ModeNumber == This->Mode->Mode) {
return EFI_SUCCESS;
}
OldTpl = gBS->RaiseTPL (TPL_NOTIFY);
//
// Protect from invalid Blt calls during SetMode.
//
Original = mGop.FramebufferContext;
mGop.FramebufferContext = NULL;
//
// Protect from mishandling of rotated info.
//
SwitchMode (This, FALSE);
Status = mGop.OriginalGopSetMode (This, ModeNumber);
if (EFI_ERROR (Status)) {
SwitchMode (This, TRUE);
mGop.FramebufferContext = Original;
gBS->RestoreTPL (OldTpl);
return Status;
}
if (Original != NULL) {
FreePages (Original, mGop.FramebufferContextPageCount);
}
RotateMode (This, mGop.Rotation);
mGop.FramebufferContext = DirectGopFromTarget (
mGop.OriginalFrameBufferBase,
&mGop.OriginalModeInfo,
&mGop.FramebufferContextPageCount
);
if (mGop.FramebufferContext == NULL) {
gBS->RestoreTPL (OldTpl);
return EFI_DEVICE_ERROR;
}
if (mGop.CachePolicy >= 0) {
MtrrSetMemoryAttribute (
mGop.OriginalFrameBufferBase,
mGop.OriginalFrameBufferSize,
mGop.CachePolicy
);
}
gBS->RestoreTPL (OldTpl);
return EFI_SUCCESS;
}
STATIC
EFI_STATUS
EFIAPI
DirectQueryMode (
IN EFI_GRAPHICS_OUTPUT_PROTOCOL *This,
IN UINT32 ModeNumber,
OUT UINTN *SizeOfInfo,
OUT EFI_GRAPHICS_OUTPUT_MODE_INFORMATION **Info
)
{
EFI_STATUS Status;
UINT32 HorizontalResolution;
SwitchMode (This, FALSE);
Status = mGop.OriginalGopQueryMode (This, ModeNumber, SizeOfInfo, Info);
if (EFI_ERROR (Status)) {
SwitchMode (This, TRUE);
return Status;
}
if ((mGop.Rotation == 90) || (mGop.Rotation == 270)) {
HorizontalResolution = (*Info)->HorizontalResolution;
(*Info)->HorizontalResolution = (*Info)->VerticalResolution;
(*Info)->VerticalResolution = HorizontalResolution;
(*Info)->PixelsPerScanLine = (*Info)->HorizontalResolution;
}
return Status;
}
STATIC
EFI_STATUS
EFIAPI
DirectGopBlt (
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 OPTIONAL
)
{
if (mGop.FramebufferContext != NULL) {
return OcBlitRender (
mGop.FramebufferContext,
BltBuffer,
BltOperation,
SourceX,
SourceY,
DestinationX,
DestinationY,
Width,
Height,
Delta
);
}
return EFI_DEVICE_ERROR;
}
VOID
OcReconnectConsole (
VOID
)
{
EFI_STATUS Status;
UINTN HandleCount;
EFI_HANDLE *HandleBuffer;
UINTN Index;
//
// When we change the GOP mode on some types of firmware, we need to reconnect the
// drivers that produce simple text out as otherwise, they will not produce text
// at the new resolution.
//
// Needy reports that boot.efi seems to work fine without this block of code.
// However, I believe that UEFI specification does not provide any standard way
// to inform TextOut protocol about resolution change, which means the firmware
// may not be aware of the change, especially when custom GOP is used.
// We can move this to quirks if it causes problems, but I believe the code below
// is legit.
//
// Note: this block of code may result in black screens on APTIO IV boards when
// launching OpenCore from the Shell. Hence it is optional.
//
Status = gBS->LocateHandleBuffer (
ByProtocol,
&gEfiSimpleTextOutProtocolGuid,
NULL,
&HandleCount,
&HandleBuffer
);
if (!EFI_ERROR (Status)) {
for (Index = 0; Index < HandleCount; ++Index) {
gBS->DisconnectController (HandleBuffer[Index], NULL, NULL);
}
for (Index = 0; Index < HandleCount; ++Index) {
gBS->ConnectController (HandleBuffer[Index], NULL, NULL, TRUE);
}
FreePool (HandleBuffer);
//
// It is implementation defined, which console mode is used by ConOut.
// Assume the implementation chooses most sensible value based on GOP resolution.
// If it does not, there is a separate ConsoleMode param, which expands to SetConsoleMode.
//
} else {
DEBUG ((DEBUG_WARN, "OCC: Failed to find any text output handles\n"));
}
}
EFI_STATUS
OcUseDirectGop (
IN INT32 CacheType
)
{
EFI_STATUS Status;
EFI_GRAPHICS_OUTPUT_PROTOCOL *Gop;
APPLE_EG2_INFO_PROTOCOL *Eg2;
UINT32 Rotation;
DEBUG ((DEBUG_INFO, "OCC: Switching to direct GOP renderer...\n"));
Status = gBS->HandleProtocol (
gST->ConsoleOutHandle,
&gEfiGraphicsOutputProtocolGuid,
(VOID **)&Gop
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OCC: Cannot find console GOP for direct GOP - %r\n", Status));
return Status;
}
if (Gop->Mode->Info->PixelFormat == PixelBltOnly) {
DEBUG ((DEBUG_INFO, "OCC: This GOP does not support direct rendering\n"));
return EFI_UNSUPPORTED;
}
Status = gBS->LocateProtocol (
&gAppleEg2InfoProtocolGuid,
NULL,
(VOID **)&Eg2
);
if (!EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OCC: Found EG2 support %X\n", Eg2->Revision));
if (Eg2->Revision >= APPLE_EG2_INFO_PROTOCOL_REVISION) {
Rotation = 0;
Status = Eg2->GetRotation (Eg2, &Rotation);
if (!EFI_ERROR (Status) && (Rotation < AppleDisplayRotateMax)) {
if (Rotation == AppleDisplayRotate90) {
mGop.Rotation = 90;
} else if (Rotation == AppleDisplayRotate180) {
mGop.Rotation = 180;
} else if (Rotation == AppleDisplayRotate270) {
mGop.Rotation = 270;
}
DEBUG ((DEBUG_INFO, "OCC: Got rotation %u degrees from EG2\n", mGop.Rotation));
} else {
DEBUG ((DEBUG_INFO, "OCC: Invalid rotation %u from EG2 - %r\n", Rotation, Status));
}
}
} else {
DEBUG ((DEBUG_INFO, "OCC: No Apple EG2 support - %r\n", Status));
}
RotateMode (Gop, mGop.Rotation);
mGop.FramebufferContext = DirectGopFromTarget (
mGop.OriginalFrameBufferBase,
&mGop.OriginalModeInfo,
&mGop.FramebufferContextPageCount
);
if (mGop.FramebufferContext == NULL) {
DEBUG ((DEBUG_INFO, "OCC: Delaying direct GOP configuration...\n"));
//
// This is possible at the start.
//
}
mGop.OriginalGopSetMode = Gop->SetMode;
mGop.OriginalGopQueryMode = Gop->QueryMode;
Gop->SetMode = DirectGopSetMode;
Gop->QueryMode = DirectQueryMode;
Gop->Blt = DirectGopBlt;
mGop.CachePolicy = -1;
if (CacheType >= 0) {
Status = MtrrSetMemoryAttribute (
mGop.OriginalFrameBufferBase,
mGop.OriginalFrameBufferSize,
CacheType
);
DEBUG ((
DEBUG_INFO,
"OCC: FB (%Lx, %Lx) MTRR (%x) - %r\n",
(UINT64)mGop.OriginalFrameBufferBase,
(UINT64)mGop.OriginalFrameBufferSize,
CacheType,
Status
));
if (!EFI_ERROR (Status)) {
mGop.CachePolicy = CacheType;
}
}
return EFI_SUCCESS;
}
CONST CONSOLE_GOP_CONTEXT *
InternalGetDirectGopContext (
VOID
)
{
if ((mGop.Rotation != 0) && (mGop.OriginalGopSetMode != NULL)) {
return &mGop;
}
return NULL;
}