aboutsummaryrefslogtreecommitdiff
path: root/EfiGuardDxe/util.c
diff options
context:
space:
mode:
Diffstat (limited to 'EfiGuardDxe/util.c')
-rw-r--r--EfiGuardDxe/util.c380
1 files changed, 380 insertions, 0 deletions
diff --git a/EfiGuardDxe/util.c b/EfiGuardDxe/util.c
new file mode 100644
index 0000000..861c81d
--- /dev/null
+++ b/EfiGuardDxe/util.c
@@ -0,0 +1,380 @@
+#include "EfiGuardDxe.h"
+#include "util.h"
+
+#include <Library/UefiLib.h>
+#include <Library/DevicePathLib.h>
+#include <Library/PrintLib.h>
+#include <Library/UefiBootServicesTableLib.h>
+
+#ifndef ZYDIS_DISABLE_FORMATTER
+
+#include <Library/PrintLib.h>
+#include <Zycore/Format.h>
+
+STATIC ZydisFormatterFunc DefaultInstructionFormatter;
+
+#endif
+
+//
+// When debugging, we can choose between poor debugging facilities (VirtualBox) or poor performance and Windows compatibility (QEMU).
+// (I guess there is also the closed source thing with the horrible user interface that installs 50 drivers on the host (VMware))
+// This is a bandaid to make Print() calls readable ...for a while... when using VirtualBox or a live machine with no debugger
+//
+EFI_STATUS
+EFIAPI
+RtlSleep(
+ IN UINTN Milliseconds
+ )
+{
+ ASSERT(gBS != NULL);
+ ASSERT(gBS->Stall != NULL);
+
+ return gBS->Stall(Milliseconds * 1000);
+}
+
+VOID
+EFIAPI
+PrintLoadedImageInfo(
+ IN EFI_LOADED_IMAGE *ImageInfo
+ )
+{
+ CHAR16* PathString = ConvertDevicePathToText(ImageInfo->FilePath, TRUE, TRUE);
+ Print(L"\r\n[+] %s\r\n", PathString);
+ Print(L" -> ImageBase = %llx\r\n", ImageInfo->ImageBase);
+ Print(L" -> ImageSize = %llx\r\n", ImageInfo->ImageSize);
+ if (PathString != NULL)
+ FreePool(PathString);
+}
+
+VOID
+EFIAPI
+AppendKernelPatchMessage(
+ IN CONST CHAR16 *Format,
+ ...
+ )
+{
+ ASSERT(gKernelPatchInfo.BufferSize % sizeof(CHAR16) == 0);
+ ASSERT(gKernelPatchInfo.BufferSize < sizeof(gKernelPatchInfo.Buffer));
+
+ VA_LIST VaList;
+ VA_START(VaList, Format);
+ CONST UINTN NumCharsPrinted = UnicodeVSPrint(gKernelPatchInfo.Buffer + (gKernelPatchInfo.BufferSize / sizeof(CHAR16)),
+ sizeof(gKernelPatchInfo.Buffer) - gKernelPatchInfo.BufferSize,
+ Format,
+ VaList);
+ VA_END(VaList);
+
+ ASSERT(gKernelPatchInfo.BufferSize + (NumCharsPrinted * sizeof(CHAR16)) < sizeof(gKernelPatchInfo.Buffer));
+ gKernelPatchInfo.BufferSize += (NumCharsPrinted * sizeof(CHAR16));
+
+ // Paranoid null terminator (UnicodeVSPrint should do this)
+ *(CHAR16*)(gKernelPatchInfo.Buffer + (gKernelPatchInfo.BufferSize / sizeof(CHAR16))) = CHAR_NULL;
+
+ // Separate the next message using the null terminator. This is because most Print() implementations crap out
+ // after ~4 lines (depending on PCDs), so we will print the final buffer using multiple calls to Print()
+ gKernelPatchInfo.BufferSize += sizeof(CHAR16);
+}
+
+VOID
+EFIAPI
+PrintKernelPatchInfo(
+ )
+{
+ ASSERT(gST->ConOut != NULL);
+
+ UINTN NumChars = gKernelPatchInfo.BufferSize / sizeof(CHAR16);
+ if (NumChars * sizeof(CHAR16) >= sizeof(gKernelPatchInfo.Buffer) - sizeof(CHAR16))
+ NumChars = sizeof(gKernelPatchInfo.Buffer) - (2 * sizeof(CHAR16)); // Avoid buffer overrun
+
+ CHAR16* String = gKernelPatchInfo.Buffer;
+ String[NumChars] = String[NumChars + 1] = CHAR_NULL; // Ensure we have a double null terminator at the end
+ UINTN Length;
+
+ // A double null terminator marks the end. It's just like that lovely Win32 getenv API that makes me want to kill myself every time I see it
+ while ((Length = StrLen(String)) != 0)
+ {
+ gST->ConOut->OutputString(gST->ConOut, String);
+ String += Length + 1;
+ }
+}
+
+BOOLEAN
+EFIAPI
+WaitForKey(
+ )
+{
+ // Hack: because we call this at TPL_NOTIFY in ExitBootServices, we cannot use WaitForEvent()
+ // in that scenario because it requires TPL == TPL_APPLICATION. So check the TPL
+ CONST EFI_TPL Tpl = EfiGetCurrentTpl();
+
+ EFI_INPUT_KEY Key = { 0, 0 };
+ EFI_STATUS Status = EFI_NOT_READY;
+
+ while (Status == EFI_NOT_READY)
+ {
+ // Can we call WaitForEvent()?
+ UINTN Index = 0;
+ if (Tpl == TPL_APPLICATION)
+ gBS->WaitForEvent(1, &gST->ConIn->WaitForKey, &Index); // Yep
+ else
+ RtlSleep(1); // Nope; burn CPU. // TODO: find a way to parallelize this to achieve GeForce FX 5800 temperatures
+
+ // At TPL_APPLICATION, we will always get EFI_SUCCESS (barring hardware failures). At higher TPLs we may also get EFI_NOT_READY
+ Status = gST->ConIn->ReadKeyStroke(gST->ConIn, &Key);
+ }
+
+ ASSERT_EFI_ERROR(Status);
+ return (BOOLEAN)(Key.ScanCode != SCAN_ESC);
+}
+
+INT32
+EFIAPI
+SetConsoleTextColour(
+ IN UINTN TextColour,
+ IN BOOLEAN ClearScreen
+ )
+{
+ CONST INT32 OriginalAttribute = gST->ConOut->Mode->Attribute;
+ CONST UINTN BackgroundColour = (UINTN)((OriginalAttribute >> 4) & 0x7);
+
+ gST->ConOut->SetAttribute(gST->ConOut, (TextColour | BackgroundColour));
+ if (ClearScreen)
+ gST->ConOut->ClearScreen(gST->ConOut);
+
+ return OriginalAttribute;
+}
+
+// TODO: #ifdef EFI_DEBUG, this should keep a match count and continue until the end of the buffer, then ASSERT(MatchCount == 1)
+EFI_STATUS
+EFIAPI
+FindPattern(
+ IN CONST UINT8* Pattern,
+ IN UINT8 Wildcard,
+ IN UINT32 PatternLength,
+ IN VOID* Base,
+ IN UINT32 Size,
+ OUT VOID **Found
+ )
+{
+ if (Found == NULL || Pattern == NULL || Base == NULL)
+ return EFI_INVALID_PARAMETER;
+
+ *Found = NULL;
+
+ for (UINT8 *Address = (UINT8*)Base; Address < (UINT8*)((UINTN)Base + Size - PatternLength); ++Address)
+ {
+ UINT32 i;
+ for (i = 0; i < PatternLength; ++i)
+ {
+ if (Pattern[i] != Wildcard && (*(Address + i) != Pattern[i]))
+ break;
+ }
+
+ if (i == PatternLength)
+ {
+ *Found = (VOID*)Address;
+ return EFI_SUCCESS;
+ }
+ }
+
+ return EFI_NOT_FOUND;
+}
+
+// For debugging non-working signatures. Not that I would ever need to do such a thing of course. Ha ha... ha
+// TODO: #ifdef EFI_DEBUG, this should keep a match count and continue until the end of the buffer, then ASSERT(MatchCount == 1)
+EFI_STATUS
+EFIAPI
+FindPatternVerbose(
+ IN CONST UINT8* Pattern,
+ IN UINT8 Wildcard,
+ IN UINT32 PatternLength,
+ IN VOID* Base,
+ IN UINT32 Size,
+ OUT VOID **Found
+ )
+{
+ if (Found == NULL || Pattern == NULL || Base == NULL)
+ return EFI_INVALID_PARAMETER;
+
+ *Found = NULL;
+
+ CONST UINTN Start = (UINTN)Base;
+ CONST UINTN End = Start + Size - PatternLength;
+ EFI_STATUS Status = EFI_NOT_FOUND;
+
+ UINT32 Max = 0;
+ UINT8 *AddrOfMax = NULL;
+
+ for (UINT8 *Address = (UINT8*)Start; Address < (UINT8*)End; ++Address)
+ {
+ UINT32 i;
+ for (i = 0; i < PatternLength; ++i)
+ {
+ if (Pattern[i] != Wildcard && (*(Address + i) != Pattern[i]))
+ break;
+ }
+
+ if (i > Max)
+ {
+ Max = i;
+ AddrOfMax = Address;
+ }
+
+ if (i == PatternLength)
+ {
+ *Found = (VOID*)Address;
+ Status = EFI_SUCCESS;
+ }
+ }
+
+ Print(L"\r\nBest match: %lu/%lu matched at 0x%p\r\n", Max, PatternLength, (VOID*)AddrOfMax);
+
+ for (UINT32 i = 0; i < PatternLength && AddrOfMax != NULL; ++i)
+ {
+ if (Pattern[i] != Wildcard && (*(AddrOfMax + i) != Pattern[i]))
+ Print(L"[%lu] [X] %02X != %02X\r\n", i, (*(AddrOfMax + i)), Pattern[i]); // Mismatch
+ else if (Pattern[i] == Wildcard)
+ Print(L"[%lu] [ ] %02X\r\n", i, (*(AddrOfMax + i))); // Matched wildcard byte
+ else
+ Print(L"[%lu] [v] %02X\r\n", i, Pattern[i]); // Matched exact byte
+ }
+
+ return Status;
+}
+
+#ifndef ZYDIS_DISABLE_FORMATTER
+
+// Formatter hook to prefix the opcode bytes to the output
+STATIC
+ZyanStatus
+ZydisInstructionBytesFormatter(
+ IN CONST ZydisFormatter* Formatter,
+ IN OUT ZydisFormatterBuffer* Buffer,
+ IN ZydisFormatterContext* Context
+ )
+{
+ CONST ZyanU8 MaxOpcodeBytes = 12; // Print at most 10 bytes (so 20 characters), with room for ellip.. ses
+
+ ZyanString *String;
+ ZYAN_CHECK(ZydisFormatterBufferGetString(Buffer, &String));
+
+ // We cannot use ZyanStringAppendFormat() because at the moment it may use dynamic memory allocation
+ // to resize the string buffer, with no way to disable this behaviour. Therefore call AsciiSPrint
+ for (ZyanU8 i = 0; i < MaxOpcodeBytes; ++i)
+ {
+ CONST ZyanUSize Length = String->vector.size;
+ UINTN N;
+
+ if (i < Context->instruction->length && i < MaxOpcodeBytes - 2)
+ {
+ // Print one byte of the instruction
+ N = AsciiSPrint((CHAR8*)(String->vector.data) + Length - 1,
+ String->vector.capacity - Length + 1,
+ "%02X",
+ *(UINT8*)(Context->runtime_address + i));
+ }
+ else if (i < Context->instruction->length && i == MaxOpcodeBytes - 2)
+ {
+ // This is a huge instruction; truncate remaining bytes with ellipses
+ N = AsciiSPrint((CHAR8*)(String->vector.data) + Length - 1,
+ String->vector.capacity - Length + 1,
+ "%a",
+ ".. ");
+ }
+ else
+ {
+ // Print an empty string for alignment padding
+ N = AsciiSPrint((CHAR8*)(String->vector.data) + Length - 1,
+ String->vector.capacity - Length + 1,
+ "%a",
+ " ");
+ }
+
+ // Do bounds check. According to docs, an ASSERT() should have already happened
+ // if we went OOB, but debug asserts may be disabled on this platform
+ if ((INTN)N < 0 || N > (UINTN)(String->vector.capacity - Length))
+ return ZYAN_STATUS_FAILED;
+
+ String->vector.size += (ZyanUSize)N;
+ }
+
+ // Call the default formatter to print the actual instruction text
+ return DefaultInstructionFormatter(Formatter, Buffer, Context);
+}
+
+#endif
+
+ZyanStatus
+EFIAPI
+ZydisInit(
+ IN PEFI_IMAGE_NT_HEADERS NtHeaders,
+ OUT ZydisDecoder *Decoder,
+ OUT ZydisFormatter *Formatter OPTIONAL
+ )
+{
+ ZyanStatus Status;
+ if (!ZYAN_SUCCESS((Status = ZydisDecoderInit(Decoder,
+ IMAGE64(NtHeaders) ? ZYDIS_MACHINE_MODE_LONG_64 : ZYDIS_MACHINE_MODE_LONG_COMPAT_32,
+ IMAGE64(NtHeaders) ? ZYDIS_ADDRESS_WIDTH_64 : ZYDIS_ADDRESS_WIDTH_32))))
+ return Status;
+
+#ifdef ZYDIS_DISABLE_FORMATTER
+ ASSERT(Formatter == NULL);
+#else
+ if (!ZYAN_SUCCESS((Status = ZydisFormatterInit(Formatter, ZYDIS_FORMATTER_STYLE_INTEL))))
+ return Status;
+ if (!ZYAN_SUCCESS((Status = ZydisFormatterSetProperty(Formatter, ZYDIS_FORMATTER_PROP_FORCE_SIZE, ZYAN_TRUE))))
+ return Status;
+
+ DefaultInstructionFormatter = (ZydisFormatterFunc)&ZydisInstructionBytesFormatter;
+ if (!ZYAN_SUCCESS((Status = ZydisFormatterSetHook(Formatter,
+ ZYDIS_FORMATTER_FUNC_FORMAT_INSTRUCTION,
+ (CONST VOID**)&DefaultInstructionFormatter))))
+ return Status;
+#endif
+
+ return ZYAN_STATUS_SUCCESS;
+}
+
+UINT8*
+EFIAPI
+BacktrackToFunctionStart(
+ IN CONST UINT8* StartAddress,
+ IN CONST UINT8* LowerBound
+ )
+{
+ // Test for null. This allows callers to do 'FindPattern(..., &Address); X = Backtrack(Address, ...)' with a single failure branch
+ if (StartAddress == NULL)
+ return NULL;
+
+ ASSERT(StartAddress > LowerBound);
+
+ UINT8 *Address;
+ BOOLEAN Found = FALSE;
+ for (Address = (UINT8*)StartAddress; Address >= LowerBound; --Address)
+ {
+ if ((*(Address - 1) == 0xCC || // Previous byte is int 3 padding, or
+ (*(Address - 2) == 0x90 && *(Address - 1) == 0x90) || // Previous 2 bytes are nop padding, or
+ (*(Address - 4) == 0x00 && *(Address - 3) == 0x00 && // Previous 4+ bytes are 00 padding (rare, only happens at start of a section), or
+ *(Address - 2) == 0x00 && *(Address - 1) == 0x00) ||
+ (*(Address - 1) == 0xC3 && *(Address - 3) != 0x8D) // Previous byte is 'ret', or
+#if defined(MDE_CPU_IA32) || defined(_M_IX86)
+ || (*(Address - 3) == 0xC2 && *(Address - 1) == 0x00) // Previous 3 bytes are 'ret XX' (x86)
+#endif
+ )
+ && // *and*
+ (*Address == 0x40 || *Address == 0x55 || // Current byte is either 'push [ebp|ebx|rbp|rbx]', 'mov REG, XX' or 'sub REG, XX'
+ (Address < StartAddress && *Address == 0x44 && *(Address + 1) == 0x89) ||
+ (Address < StartAddress && *Address == 0x48 && *(Address + 1) == 0x83) ||
+ (Address < StartAddress && *Address == 0x48 && *(Address + 1) == 0x89) ||
+ (Address < StartAddress && *Address == 0x48 && *(Address + 1) == 0x8B) ||
+ (Address < StartAddress && *Address == 0x49 && *(Address + 1) == 0x89) ||
+ (Address < StartAddress && *Address == 0x4C && *(Address + 1) == 0x8B)))
+ {
+ Found = TRUE;
+ break;
+ }
+ }
+
+ return Found ? Address : NULL;
+}