mirror of
https://github.com/acidanthera/OpenCorePkg.git
synced 2025-12-08 19:25:01 +00:00
As of macOS 13 Developer Beta 3, the Kernel Collection's inner kernel references a segment that precedes itself. The current model is that a Kernel Collection is a container format and the included files are (mostly) separate. Hence, this was treated as an out-of-bounds issue. Kernel Collections apparently are rather an unconventional composite format, where the sub-files are still part of the whole. Redesign OcMachoLib to treat the Kernel Collection as the reference file. Patches still use only the inner file, while parsing considers the whole file.
345 lines
9.5 KiB
C
345 lines
9.5 KiB
C
/** @file
|
|
Copyright (C) 2018, 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 <Library/BaseMemoryLib.h>
|
|
#include <Library/DebugLib.h>
|
|
#include <Library/MemoryAllocationLib.h>
|
|
#include <Library/OcMachoLib.h>
|
|
#include <Library/OcMiscLib.h>
|
|
|
|
#include <string.h>
|
|
#include <sys/time.h>
|
|
|
|
#include <UserFile.h>
|
|
|
|
MACH_HEADER_64 mHeader;
|
|
MACH_SECTION_64 mSect;
|
|
MACH_SEGMENT_COMMAND_64 mSeg;
|
|
MACH_UUID_COMMAND mUuid;
|
|
|
|
STATIC
|
|
int
|
|
FeedMacho (
|
|
IN OUT VOID *File,
|
|
IN UINT32 Size
|
|
)
|
|
{
|
|
OC_MACHO_CONTEXT Context;
|
|
|
|
if (!MachoInitializeContext64 (&Context, File, Size, 0, Size)) {
|
|
return -1;
|
|
}
|
|
|
|
int Code = 0;
|
|
|
|
MACH_HEADER_64 *Hdr = MachoGetMachHeader64 (&Context);
|
|
|
|
if ((Hdr != NULL) && (MachoGetFileSize (&Context) > 10) && (MachoGetLastAddress (&Context) != 10)) {
|
|
CopyMem (&mHeader, Hdr, sizeof (mHeader));
|
|
++Code;
|
|
}
|
|
|
|
MACH_UUID_COMMAND *Cmd = MachoGetUuid (&Context);
|
|
|
|
if (Cmd != NULL) {
|
|
CopyMem (&mUuid, Cmd, sizeof (mUuid));
|
|
++Code;
|
|
}
|
|
|
|
MACH_SEGMENT_COMMAND_64 *Segment = MachoGetSegmentByName64 (&Context, "__LINKEDIT");
|
|
MACH_SECTION_64 *Section;
|
|
|
|
if (Segment != NULL) {
|
|
CopyMem (&mSeg, Segment, sizeof (mSeg));
|
|
Section = MachoGetSectionByName64 (&Context, Segment, "__objc");
|
|
if (Section != NULL) {
|
|
CopyMem (&mSect, Section, sizeof (mSect));
|
|
++Code;
|
|
}
|
|
}
|
|
|
|
UINT32 Index = 0;
|
|
|
|
while ((Section = MachoGetSectionByIndex64 (&Context, Index)) != NULL) {
|
|
CopyMem (&mSect, Section, sizeof (mSect));
|
|
++Index;
|
|
}
|
|
|
|
if ((Section = MachoGetSectionByAddress64 (&Context, Index)) != NULL) {
|
|
CopyMem (&mSect, Section, sizeof (mSect));
|
|
++Code;
|
|
}
|
|
|
|
MACH_NLIST_64 *Symbol = NULL;
|
|
|
|
for (Index = 0; (Symbol = MachoGetSymbolByIndex64 (&Context, Index)) != NULL; ++Index) {
|
|
CONST CHAR8 *Indirect = MachoGetIndirectSymbolName64 (&Context, Symbol);
|
|
if ( (AsciiStrCmp (MachoGetSymbolName64 (&Context, Symbol), "__hack") == 0)
|
|
|| ((Indirect != NULL) && (AsciiStrCmp (Indirect, "__hack") == 0)))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolIsSection64 (Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolIsDefined64 (Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolIsLocalDefined64 (&Context, Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoIsSymbolValueInRange64 (&Context, Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
UINT32 Offset;
|
|
if (MachoSymbolGetFileOffset64 (&Context, Symbol, &Offset, NULL)) {
|
|
Code += Offset;
|
|
}
|
|
|
|
if (MachoSymbolNameIsPureVirtual (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsPadslot (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsSmcp (&Context, MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsMetaclassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
char Out[64];
|
|
if ( MachoSymbolNameIsSmcp (&Context, MachoGetSymbolName64 (&Context, Symbol))
|
|
&& MachoGetClassNameFromSuperMetaClassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsVtable (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
if (AsciiStrCmp (MachoGetClassNameFromVtableName (MachoGetSymbolName64 (&Context, Symbol)), "sym") != 0) {
|
|
++Code;
|
|
}
|
|
}
|
|
|
|
if ( MachoGetFunctionPrefixFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoSymbolNameIsMetaclassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol))
|
|
&& MachoGetClassNameFromMetaClassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoGetVtableNameFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoGetMetaVtableNameFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoGetFinalSymbolNameFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsCxx (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
MACH_NLIST_64 *SCMP = MachoGetMetaclassSymbolFromSmcpSymbol64 (&Context, Symbol);
|
|
if (SCMP != NULL) {
|
|
if (AsciiStrCmp (MachoGetSymbolName64 (&Context, SCMP), "__hack") == 0) {
|
|
++Code;
|
|
}
|
|
|
|
CONST MACH_NLIST_64 *Vtable;
|
|
CONST MACH_NLIST_64 *MetaVtable;
|
|
if (MachoGetVtableSymbolsFromSmcp64 (&Context, MachoGetSymbolName64 (&Context, SCMP), &Vtable, &MetaVtable)) {
|
|
if (AsciiStrCmp (MachoGetSymbolName64 (&Context, Vtable), "__hack") == 0) {
|
|
++Code;
|
|
}
|
|
|
|
if (AsciiStrCmp (MachoGetSymbolName64 (&Context, MetaVtable), "__hack") == 0) {
|
|
++Code;
|
|
}
|
|
}
|
|
}
|
|
|
|
MACH_NLIST_64 SSSS = *Symbol;
|
|
MachoRelocateSymbol64 (&Context, 0x100000000, &SSSS);
|
|
}
|
|
|
|
Symbol = MachoGetLocalDefinedSymbolByName64 (&Context, "_Assert");
|
|
if (Symbol != NULL) {
|
|
CONST CHAR8 *Indirect = MachoGetIndirectSymbolName64 (&Context, Symbol);
|
|
if ( (AsciiStrCmp (MachoGetSymbolName64 (&Context, Symbol), "__hack") == 0)
|
|
|| ((Indirect != NULL) && (AsciiStrCmp (Indirect, "__hack") == 0)))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolIsSection64 (Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolIsDefined64 (Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolIsLocalDefined64 (&Context, Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoIsSymbolValueInRange64 (&Context, Symbol)) {
|
|
++Code;
|
|
}
|
|
|
|
UINT32 Offset;
|
|
if (MachoSymbolGetFileOffset64 (&Context, Symbol, &Offset, NULL)) {
|
|
Code += Offset;
|
|
}
|
|
|
|
if (MachoSymbolNameIsPureVirtual (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsPadslot (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsSmcp (&Context, MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsMetaclassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
|
|
CHAR8 Out[64];
|
|
if ( MachoSymbolNameIsSmcp (&Context, MachoGetSymbolName64 (&Context, Symbol))
|
|
&& MachoGetClassNameFromSuperMetaClassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsVtable (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
if (AsciiStrCmp (MachoGetClassNameFromVtableName (MachoGetSymbolName64 (&Context, Symbol)), "sym") != 0) {
|
|
++Code;
|
|
}
|
|
}
|
|
|
|
if ( MachoGetFunctionPrefixFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoSymbolNameIsMetaclassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol))
|
|
&& MachoGetClassNameFromMetaClassPointer (&Context, MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoGetVtableNameFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoGetMetaVtableNameFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if ( MachoGetFinalSymbolNameFromClassName (MachoGetSymbolName64 (&Context, Symbol), sizeof (Out), Out)
|
|
&& (AsciiStrCmp ("SomeReallyLongStringJustInCaseToCheckIt", Out) == 0))
|
|
{
|
|
++Code;
|
|
}
|
|
|
|
if (MachoSymbolNameIsCxx (MachoGetSymbolName64 (&Context, Symbol))) {
|
|
++Code;
|
|
}
|
|
}
|
|
|
|
for (UINTN i = 0x1000000; i < MAX_UINTN; i += 0x1000000) {
|
|
if (MachoGetSymbolByRelocationOffset64 (&Context, i, &Symbol)) {
|
|
if (AsciiStrCmp (MachoGetSymbolName64 (&Context, Symbol), "__hack") == 0) {
|
|
++Code;
|
|
}
|
|
}
|
|
}
|
|
|
|
return Code != 963;
|
|
}
|
|
|
|
int
|
|
ENTRY_POINT (
|
|
int argc,
|
|
char *argv[]
|
|
)
|
|
{
|
|
UINT32 FileSize;
|
|
UINT8 *Buffer;
|
|
|
|
if ((Buffer = UserReadFile ((argc > 1) ? argv[1] : "kernel", &FileSize)) == NULL) {
|
|
DEBUG ((DEBUG_ERROR, "Read fail\n"));
|
|
return -1;
|
|
}
|
|
|
|
return FeedMacho (Buffer, FileSize);
|
|
}
|
|
|
|
int
|
|
LLVMFuzzerTestOneInput (
|
|
const uint8_t *Data,
|
|
size_t Size
|
|
)
|
|
{
|
|
VOID *NewData;
|
|
|
|
if (Size > 0) {
|
|
NewData = AllocatePool (Size);
|
|
if (NewData != NULL) {
|
|
CopyMem (NewData, Data, Size);
|
|
FeedMacho (NewData, (UINT32)Size);
|
|
FreePool (NewData);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|