mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2024-12-12 14:36:56 +01:00
337 lines
8.2 KiB
C
337 lines
8.2 KiB
C
|
/** @file
|
||
|
Default instance of VideoBiosPatchLib library for video bios patches.
|
||
|
|
||
|
Ported from Chameleon's Resolution module (created by Evan Lojewski)
|
||
|
which is a version of 915resolution (created by steve tomljenovic).
|
||
|
|
||
|
Ported to UEFI by usr-sse2, tweaked and added as VideoBiosPatchLib by dmazar.
|
||
|
|
||
|
**/
|
||
|
|
||
|
#include "VideoBiosPatchLibInternal.h"
|
||
|
|
||
|
|
||
|
//
|
||
|
// Internal pointers to LegacyRegion protocols
|
||
|
//
|
||
|
EFI_LEGACY_REGION_PROTOCOL *mLegacyRegion = NULL;
|
||
|
EFI_LEGACY_REGION2_PROTOCOL *mLegacyRegion2 = NULL;
|
||
|
|
||
|
//
|
||
|
// Temp var for passing Edid to readEDID() in edid.c
|
||
|
//
|
||
|
UINT8 *mEdid = NULL;
|
||
|
|
||
|
|
||
|
/**
|
||
|
Searches Source for Search pattern of size SearchSize
|
||
|
and replaces it with Replace up to MaxReplaces times.
|
||
|
|
||
|
@param Source Source bytes that will be searched.
|
||
|
@param SourceSize Number of bytes in Source region.
|
||
|
@param Search Bytes to search for.
|
||
|
@param SearchSize Number of bytes in Search.
|
||
|
@param Replace Bytes that will replace found bytes in Source (size is SearchSize).
|
||
|
@param MaxReplaces Maximum number of replaces. If MaxReplaces <= 0, then there is no restriction.
|
||
|
|
||
|
@retval Number of replaces done.
|
||
|
|
||
|
**/
|
||
|
UINTN
|
||
|
VideoBiosPatchSearchAndReplace (
|
||
|
IN UINT8 *Source,
|
||
|
IN UINTN SourceSize,
|
||
|
IN UINT8 *Search,
|
||
|
IN UINTN SearchSize,
|
||
|
IN UINT8 *Replace,
|
||
|
IN INTN MaxReplaces
|
||
|
)
|
||
|
{
|
||
|
UINTN NumReplaces = 0;
|
||
|
BOOLEAN NoReplacesRestriction = MaxReplaces <= 0;
|
||
|
UINT8 *End = Source + SourceSize;
|
||
|
|
||
|
while (Source < End && (NoReplacesRestriction || MaxReplaces > 0)) {
|
||
|
if (CompareMem(Source, Search, SearchSize) == 0) {
|
||
|
CopyMem(Source, Replace, SearchSize);
|
||
|
NumReplaces++;
|
||
|
MaxReplaces--;
|
||
|
Source += SearchSize;
|
||
|
} else {
|
||
|
Source++;
|
||
|
}
|
||
|
}
|
||
|
return NumReplaces;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
Inits mLegacyRegion or mLegacyRegion2 protocols.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
VideoBiosPatchInit (
|
||
|
VOID
|
||
|
)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
|
||
|
//
|
||
|
// Return if we are already inited
|
||
|
//
|
||
|
if (mLegacyRegion != NULL || mLegacyRegion2 != NULL) {
|
||
|
return EFI_SUCCESS;
|
||
|
}
|
||
|
|
||
|
DBG (" VideoBiosPatchInit(");
|
||
|
//
|
||
|
// Check for EfiLegacyRegionProtocol and/or EfiLegacyRegion2Protocol
|
||
|
//
|
||
|
Status = gBS->LocateProtocol (&gEfiLegacyRegionProtocolGuid, NULL, (VOID **) &mLegacyRegion);
|
||
|
DBG ("LegacyRegion = %r", Status);
|
||
|
if (EFI_ERROR (Status)) {
|
||
|
mLegacyRegion = NULL;
|
||
|
Status = gBS->LocateProtocol (&gEfiLegacyRegion2ProtocolGuid, NULL, (VOID **) &mLegacyRegion2);
|
||
|
DBG (", LegacyRegion2 = %r", Status);
|
||
|
if (EFI_ERROR (Status)) {
|
||
|
mLegacyRegion2 = NULL;
|
||
|
}
|
||
|
}
|
||
|
DBG (") = %r\n", Status);
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
Unlocks video bios area for writing.
|
||
|
|
||
|
@retval EFI_SUCCESS If area is unlocked.
|
||
|
@retval other In case of error.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
VideoBiosUnlock (
|
||
|
VOID
|
||
|
)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
UINT32 Granularity;
|
||
|
UINT8 *TstPtr;
|
||
|
UINT8 TstVar;
|
||
|
|
||
|
Status = VideoBiosPatchInit ();
|
||
|
if (EFI_ERROR (Status)) {
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
Status = EFI_NOT_FOUND;
|
||
|
|
||
|
DBG(" VideoBiosUnlock: ");
|
||
|
if (mLegacyRegion != NULL) {
|
||
|
Status = mLegacyRegion->UnLock (mLegacyRegion, VBIOS_START, VBIOS_SIZE, &Granularity);
|
||
|
} else if (mLegacyRegion2 != NULL) {
|
||
|
Status = mLegacyRegion2->UnLock (mLegacyRegion2, VBIOS_START, VBIOS_SIZE, &Granularity);
|
||
|
}
|
||
|
//DBG("%r\n", Status);
|
||
|
|
||
|
//
|
||
|
// Test some vbios address for writing
|
||
|
//
|
||
|
TstPtr = (UINT8*)(UINTN)(VBIOS_START + 0xA110);
|
||
|
TstVar = *TstPtr;
|
||
|
*TstPtr = *TstPtr + 1;
|
||
|
if (TstVar == *TstPtr) {
|
||
|
DBG(" Test unlock: Error, not unlocked!\n");
|
||
|
Status = EFI_DEVICE_ERROR;
|
||
|
} else {
|
||
|
DBG(" unlocked\n");
|
||
|
}
|
||
|
*TstPtr = TstVar;
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
Locks video bios area for writing.
|
||
|
|
||
|
@retval EFI_SUCCESS If area is locked.
|
||
|
@retval other In case of error.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
VideoBiosLock (
|
||
|
VOID
|
||
|
)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
UINT32 Granularity;
|
||
|
|
||
|
Status = VideoBiosPatchInit ();
|
||
|
if (EFI_ERROR (Status)) {
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
Status = EFI_NOT_FOUND;
|
||
|
|
||
|
DBG(" VideoBiosLock: ");
|
||
|
if (mLegacyRegion != NULL) {
|
||
|
Status = mLegacyRegion->Lock (mLegacyRegion, VBIOS_START, VBIOS_SIZE, &Granularity);
|
||
|
} else if (mLegacyRegion2 != NULL) {
|
||
|
Status = mLegacyRegion2->Lock (mLegacyRegion2, VBIOS_START, VBIOS_SIZE, &Granularity);
|
||
|
}
|
||
|
DBG("%r\n", Status);
|
||
|
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
Performs mutltiple Search&Replace operations on the video bios memory.
|
||
|
|
||
|
@param FindAndReplace Pointer to array of VBIOS_PATCH_BYTES.
|
||
|
@param FindAndReplaceCount Number of VBIOS_PATCH_BYTES elements in a FindAndReplace array.
|
||
|
|
||
|
@retval EFI_SUCCESS If no error occured.
|
||
|
@retval other In case of error.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
VideoBiosPatchBytes (
|
||
|
IN VBIOS_PATCH_BYTES *FindAndReplace,
|
||
|
IN UINTN FindAndReplaceCount
|
||
|
)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
UINTN Index;
|
||
|
UINTN NumReplaces;
|
||
|
UINTN NumReplacesTotal;
|
||
|
|
||
|
if (FindAndReplace == NULL || FindAndReplaceCount == 0) {
|
||
|
return EFI_INVALID_PARAMETER;
|
||
|
}
|
||
|
|
||
|
DBG ("VideoBiosPatchBytes(%d patches):\n", FindAndReplaceCount);
|
||
|
Status = VideoBiosUnlock ();
|
||
|
if (EFI_ERROR (Status)) {
|
||
|
DBG (" = not done.\n");
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
NumReplaces = 0;
|
||
|
NumReplacesTotal = 0;
|
||
|
for (Index = 0; Index < FindAndReplaceCount; Index++) {
|
||
|
NumReplaces = VideoBiosPatchSearchAndReplace (
|
||
|
(UINT8*)(UINTN)VBIOS_START,
|
||
|
VBIOS_SIZE,
|
||
|
FindAndReplace[Index].Find,
|
||
|
FindAndReplace[Index].NumberOfBytes,
|
||
|
FindAndReplace[Index].Replace,
|
||
|
-1
|
||
|
);
|
||
|
NumReplacesTotal += NumReplaces;
|
||
|
DBG (" patch %d: patched %d time(s)\n", Index, NumReplaces);
|
||
|
}
|
||
|
|
||
|
DBG (" patched %d time(s)\n", NumReplacesTotal);
|
||
|
|
||
|
VideoBiosLock ();
|
||
|
|
||
|
return EFI_SUCCESS;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
Reads and returns Edid from EFI_EDID_ACTIVE_PROTOCOL.
|
||
|
|
||
|
@retval Edid If Edid found.
|
||
|
@retval NULL If Edid not found.
|
||
|
|
||
|
**/
|
||
|
UINT8* VideoBiosPatchGetEdid (VOID)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
EFI_EDID_ACTIVE_PROTOCOL *EdidProtocol;
|
||
|
UINT8 *Edid;
|
||
|
|
||
|
DBG (" Edid:");
|
||
|
Edid = NULL;
|
||
|
Status = gBS->LocateProtocol (&gEfiEdidActiveProtocolGuid, NULL, (VOID**)&EdidProtocol);
|
||
|
if (!EFI_ERROR (Status)) {
|
||
|
DBG(" size=%d", EdidProtocol->SizeOfEdid);
|
||
|
if (EdidProtocol->SizeOfEdid > 0) {
|
||
|
Edid = AllocateCopyPool (EdidProtocol->SizeOfEdid, EdidProtocol->Edid);
|
||
|
}
|
||
|
}
|
||
|
DBG(" %a\n", Edid != NULL ? "found" : "not found");
|
||
|
|
||
|
return Edid;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
Determines "native" resolution from Edid detail timing descriptor
|
||
|
and patches first video mode with that timing/resolution info.
|
||
|
|
||
|
@param Edid Edid to use. If NULL, then Edid will be read from EFI_EDID_ACTIVE_PROTOCOL
|
||
|
|
||
|
@retval EFI_SUCCESS If no error occured.
|
||
|
@retval other In case of error.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
VideoBiosPatchNativeFromEdid (
|
||
|
IN UINT8 *Edid OPTIONAL
|
||
|
)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
BOOLEAN ReleaseEdid;
|
||
|
vbios_map *map;
|
||
|
|
||
|
DBG ("VideoBiosPatchNativeFromEdid:\n");
|
||
|
|
||
|
ReleaseEdid = FALSE;
|
||
|
if (Edid == NULL) {
|
||
|
Edid = VideoBiosPatchGetEdid ();
|
||
|
if (Edid == NULL) {
|
||
|
return EFI_UNSUPPORTED;
|
||
|
}
|
||
|
ReleaseEdid = TRUE;
|
||
|
}
|
||
|
|
||
|
map = open_vbios(CT_UNKNOWN);
|
||
|
if (map == NULL) {
|
||
|
DBG (" = unknown video bios.\n");
|
||
|
if (ReleaseEdid) {
|
||
|
FreePool (Edid);
|
||
|
}
|
||
|
return EFI_UNSUPPORTED;
|
||
|
}
|
||
|
|
||
|
Status = VideoBiosUnlock ();
|
||
|
if (EFI_ERROR (Status)) {
|
||
|
DBG (" = not done.\n");
|
||
|
return Status;
|
||
|
}
|
||
|
|
||
|
mEdid = Edid;
|
||
|
set_mode (map, 0, 0, 0, 0, 0);
|
||
|
mEdid = NULL;
|
||
|
if (ReleaseEdid) {
|
||
|
FreePool (Edid);
|
||
|
}
|
||
|
|
||
|
VideoBiosLock ();
|
||
|
|
||
|
close_vbios (map);
|
||
|
|
||
|
return EFI_SUCCESS;
|
||
|
|
||
|
}
|
||
|
|