mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2025-01-11 19:22:14 +01:00
621 lines
14 KiB
C
621 lines
14 KiB
C
/** @file
|
|
Provides services for C++ symbols.
|
|
|
|
Copyright (c) 2018, Download-Fritz. All rights reserved.<BR>
|
|
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 <Base.h>
|
|
|
|
#include <IndustryStandard/AppleMachoImage.h>
|
|
|
|
#include <Library/BaseLib.h>
|
|
#include <Library/BaseMemoryLib.h>
|
|
#include <Library/DebugLib.h>
|
|
#include <Library/OcGuardLib.h>
|
|
#include <Library/MachoLib.h>
|
|
#include <Library/OcStringLib.h>
|
|
|
|
#define CXX_PREFIX "__Z"
|
|
#define VTABLE_PREFIX CXX_PREFIX "TV"
|
|
#define OSOBJ_PREFIX CXX_PREFIX "N"
|
|
#define RESERVED_TOKEN "_RESERVED"
|
|
#define METACLASS_TOKEN "10gMetaClassE"
|
|
#define SMCP_TOKEN "10superClassE"
|
|
#define METACLASS_VTABLE_PREFIX VTABLE_PREFIX "N"
|
|
#define METACLASS_VTABLE_SUFFIX "9MetaClassE"
|
|
#define CXX_PURE_VIRTUAL "___cxa_pure_virtual"
|
|
#define FINAL_CLASS_TOKEN "14__OSFinalClassEv"
|
|
|
|
#define VTABLE_ENTRY_SIZE_64 8U
|
|
#define VTABLE_HEADER_LEN_64 2U
|
|
#define VTABLE_HEADER_SIZE_64 (VTABLE_HEADER_LEN_64 * VTABLE_ENTRY_SIZE_64)
|
|
|
|
#define SYM_MAX_NAME_LEN 256U
|
|
|
|
/**
|
|
Returns whether Name is pure virtual.
|
|
|
|
@param[in] Name The name to evaluate.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoSymbolNameIsPureVirtual (
|
|
IN CONST CHAR8 *Name
|
|
)
|
|
{
|
|
ASSERT (Name != NULL);
|
|
return (AsciiStrCmp (Name, CXX_PURE_VIRTUAL) == 0);
|
|
}
|
|
|
|
/**
|
|
Returns whether Name is a Padslot.
|
|
|
|
@param[in] Name The name to evaluate.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoSymbolNameIsPadslot (
|
|
IN CONST CHAR8 *Name
|
|
)
|
|
{
|
|
ASSERT (Name != NULL);
|
|
return (AsciiStrStr (Name, RESERVED_TOKEN) != NULL);
|
|
}
|
|
|
|
/**
|
|
Returns whether SymbolName defines a Super Metaclass Pointer.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] SymbolName The symbol name to check.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoSymbolNameIsSmcp64 (
|
|
IN OUT OC_MACHO_CONTEXT *Context,
|
|
IN CONST CHAR8 *SymbolName
|
|
)
|
|
{
|
|
CONST CHAR8 *Suffix;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SymbolName != NULL);
|
|
//
|
|
// Verify the symbol has...
|
|
// 1) The C++ prefix.
|
|
// 2) The SMCP suffix.
|
|
// 3) At least one character between the prefix and the suffix.
|
|
//
|
|
Suffix = AsciiStrStr (SymbolName, SMCP_TOKEN);
|
|
|
|
if ((Suffix == NULL)
|
|
|| (AsciiStrnCmp (SymbolName, OSOBJ_PREFIX, L_STR_LEN (OSOBJ_PREFIX)) != 0)
|
|
|| ((UINTN)(Suffix - SymbolName) <= L_STR_LEN (OSOBJ_PREFIX))) {
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Returns whether SymbolName defines a Super Metaclass Pointer.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] SymbolName The symbol name to check.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoSymbolNameIsMetaclassPointer64 (
|
|
IN OUT OC_MACHO_CONTEXT *Context,
|
|
IN CONST CHAR8 *SymbolName
|
|
)
|
|
{
|
|
CONST CHAR8 *Suffix;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SymbolName != NULL);
|
|
//
|
|
// Verify the symbol has...
|
|
// 1) The C++ prefix.
|
|
// 2) The MetaClass suffix.
|
|
// 3) At least one character between the prefix and the suffix.
|
|
//
|
|
Suffix = AsciiStrStr (SymbolName, METACLASS_TOKEN);
|
|
|
|
if ((Suffix == NULL)
|
|
|| (AsciiStrnCmp (SymbolName, OSOBJ_PREFIX, L_STR_LEN (OSOBJ_PREFIX)) != 0)
|
|
|| ((UINTN)(Suffix - SymbolName) <= L_STR_LEN (OSOBJ_PREFIX))) {
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Retrieves the class name of a Super Meta Class Pointer.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] SmcpName SMCP Symbol name to get the class name of.
|
|
@param[in] ClassNameSize The size of ClassName.
|
|
@param[out] ClassName The output buffer for the class name.
|
|
|
|
@returns Whether the name has been retrieved successfully.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetClassNameFromSuperMetaClassPointer (
|
|
IN OUT OC_MACHO_CONTEXT *Context,
|
|
IN CONST CHAR8 *SmcpName,
|
|
IN UINTN ClassNameSize,
|
|
OUT CHAR8 *ClassName
|
|
)
|
|
{
|
|
UINTN PrefixSize;
|
|
UINTN SuffixSize;
|
|
UINTN OutputSize;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SmcpName != NULL);
|
|
ASSERT (ClassNameSize > 0);
|
|
ASSERT (ClassName != NULL);
|
|
|
|
ASSERT (Context->StringTable != NULL);
|
|
|
|
ASSERT (MachoSymbolNameIsSmcp64 (Context, SmcpName));
|
|
|
|
PrefixSize = L_STR_LEN (OSOBJ_PREFIX);
|
|
SuffixSize = L_STR_LEN (SMCP_TOKEN);
|
|
|
|
OutputSize = (AsciiStrLen (SmcpName) - PrefixSize - SuffixSize);
|
|
|
|
if ((OutputSize + 1) > ClassNameSize) {
|
|
return FALSE;
|
|
}
|
|
|
|
CopyMem (ClassName, &SmcpName[PrefixSize], OutputSize);
|
|
ClassName[OutputSize] = '\0';
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Retrieves the class name of a VTable.
|
|
|
|
@param[out] VtableName The VTable's name.
|
|
|
|
**/
|
|
CONST CHAR8 *
|
|
MachoGetClassNameFromVtableName (
|
|
IN CONST CHAR8 *VtableName
|
|
)
|
|
{
|
|
ASSERT (VtableName != NULL);
|
|
ASSERT (MachoSymbolNameIsVtable64 (VtableName));
|
|
//
|
|
// As there is no suffix, just return a pointer from within VtableName.
|
|
//
|
|
return &VtableName[L_STR_LEN (VTABLE_PREFIX)];
|
|
}
|
|
|
|
/**
|
|
Retrieves the function prefix of a class name.
|
|
|
|
@param[in] ClassName The class name to evaluate.
|
|
@param[in] FunctionPrefixSize The size of FunctionPrefix.
|
|
@param[out] FunctionPrefix The output buffer for the function prefix.
|
|
|
|
@returns Whether the name has been retrieved successfully.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetFunctionPrefixFromClassName (
|
|
IN CONST CHAR8 *ClassName,
|
|
IN UINTN FunctionPrefixSize,
|
|
OUT CHAR8 *FunctionPrefix
|
|
)
|
|
{
|
|
UINTN Index;
|
|
UINTN BodySize;
|
|
BOOLEAN Result;
|
|
UINTN TotalSize;
|
|
|
|
ASSERT (ClassName != NULL);
|
|
ASSERT (FunctionPrefixSize > 0);
|
|
ASSERT (FunctionPrefix != NULL);
|
|
|
|
BodySize = AsciiStrSize (ClassName);
|
|
Result = OcOverflowAddUN (L_STR_LEN (OSOBJ_PREFIX), BodySize, &TotalSize);
|
|
if (Result || (FunctionPrefixSize < TotalSize)) {
|
|
return FALSE;
|
|
}
|
|
|
|
Index = 0;
|
|
CopyMem (
|
|
&FunctionPrefix[Index],
|
|
OSOBJ_PREFIX,
|
|
L_STR_LEN (OSOBJ_PREFIX)
|
|
);
|
|
|
|
Index += L_STR_LEN (OSOBJ_PREFIX);
|
|
CopyMem (
|
|
&FunctionPrefix[Index],
|
|
ClassName,
|
|
BodySize
|
|
);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Retrieves the class name of a Meta Class Pointer.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] MetaClassName MCP Symbol name to get the class name of.
|
|
@param[in] ClassNameSize The size of ClassName.
|
|
@param[out] ClassName The output buffer for the class name.
|
|
|
|
@returns Whether the name has been retrieved successfully.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetClassNameFromMetaClassPointer (
|
|
IN OUT OC_MACHO_CONTEXT *Context,
|
|
IN CONST CHAR8 *MetaClassName,
|
|
IN UINTN ClassNameSize,
|
|
OUT CHAR8 *ClassName
|
|
)
|
|
{
|
|
UINTN PrefixSize;
|
|
UINTN SuffixSize;
|
|
UINTN ClassNameLength;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (MetaClassName != NULL);
|
|
ASSERT (ClassNameSize > 0);
|
|
ASSERT (ClassName != NULL);
|
|
|
|
ASSERT (Context->StringTable != NULL);
|
|
|
|
ASSERT (MachoSymbolNameIsMetaclassPointer64 (Context, MetaClassName));
|
|
|
|
PrefixSize = L_STR_LEN (OSOBJ_PREFIX);
|
|
SuffixSize = L_STR_LEN (METACLASS_TOKEN);
|
|
|
|
ClassNameLength = (AsciiStrLen (MetaClassName) - PrefixSize - SuffixSize);
|
|
if ((ClassNameLength + 1) > ClassNameSize) {
|
|
return FALSE;
|
|
}
|
|
|
|
CopyMem (ClassName, &MetaClassName[PrefixSize], ClassNameLength);
|
|
ClassName[ClassNameLength] = '\0';
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Retrieves the VTable name of a class name.
|
|
|
|
@param[in] ClassName Class name to get the VTable name of.
|
|
@param[in] VtableNameSize The size of VtableName.
|
|
@param[out] VtableName The output buffer for the VTable name.
|
|
|
|
@returns Whether the name has been retrieved successfully.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetVtableNameFromClassName (
|
|
IN CONST CHAR8 *ClassName,
|
|
IN UINTN VtableNameSize,
|
|
OUT CHAR8 *VtableName
|
|
)
|
|
{
|
|
UINTN Index;
|
|
UINTN BodySize;
|
|
BOOLEAN Result;
|
|
UINTN TotalSize;
|
|
|
|
ASSERT (ClassName != NULL);
|
|
ASSERT (VtableNameSize > 0);
|
|
ASSERT (VtableName != NULL);
|
|
|
|
BodySize = AsciiStrSize (ClassName);
|
|
|
|
Result = OcOverflowAddUN (
|
|
L_STR_LEN (VTABLE_PREFIX),
|
|
BodySize,
|
|
&TotalSize
|
|
);
|
|
if (Result || (VtableNameSize < TotalSize)) {
|
|
return FALSE;
|
|
}
|
|
|
|
Index = 0;
|
|
CopyMem (
|
|
&VtableName[Index],
|
|
VTABLE_PREFIX,
|
|
L_STR_LEN (VTABLE_PREFIX)
|
|
);
|
|
|
|
Index += L_STR_LEN (VTABLE_PREFIX);
|
|
CopyMem (&VtableName[Index], ClassName, BodySize);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Retrieves the Meta VTable name of a class name.
|
|
|
|
@param[in] ClassName Class name to get the Meta VTable name of.
|
|
@param[in] VtableNameSize The size of VtableName.
|
|
@param[out] VtableName The output buffer for the VTable name.
|
|
|
|
@returns Whether the name has been retrieved successfully.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetMetaVtableNameFromClassName (
|
|
IN CONST CHAR8 *ClassName,
|
|
IN UINTN VtableNameSize,
|
|
OUT CHAR8 *VtableName
|
|
)
|
|
{
|
|
UINTN BodyLength;
|
|
BOOLEAN Result;
|
|
UINTN TotalSize;
|
|
UINTN Index;
|
|
|
|
ASSERT (ClassName != NULL);
|
|
ASSERT (VtableNameSize > 0);
|
|
ASSERT (VtableName != NULL);
|
|
|
|
BodyLength = AsciiStrLen (ClassName);
|
|
|
|
Result = OcOverflowTriAddUN (
|
|
L_STR_LEN (METACLASS_VTABLE_PREFIX),
|
|
BodyLength,
|
|
L_STR_SIZE (METACLASS_VTABLE_SUFFIX),
|
|
&TotalSize
|
|
);
|
|
if (Result || (VtableNameSize < TotalSize)) {
|
|
return FALSE;
|
|
}
|
|
|
|
Index = 0;
|
|
CopyMem (
|
|
&VtableName[Index],
|
|
METACLASS_VTABLE_PREFIX,
|
|
L_STR_LEN (METACLASS_VTABLE_PREFIX)
|
|
);
|
|
|
|
Index += L_STR_LEN (METACLASS_VTABLE_PREFIX);
|
|
CopyMem (&VtableName[Index], ClassName, BodyLength);
|
|
|
|
Index += BodyLength;
|
|
CopyMem (
|
|
&VtableName[Index],
|
|
METACLASS_VTABLE_SUFFIX,
|
|
L_STR_SIZE (METACLASS_VTABLE_SUFFIX)
|
|
);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Retrieves the final symbol name of a class name.
|
|
|
|
@param[in] ClassName Class name to get the final symbol name of.
|
|
@param[in] FinalSymbolNameSize The size of FinalSymbolName.
|
|
@param[out] FinalSymbolName The output buffer for the final symbol name.
|
|
|
|
@returns Whether the name has been retrieved successfully.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetFinalSymbolNameFromClassName (
|
|
IN CONST CHAR8 *ClassName,
|
|
IN UINTN FinalSymbolNameSize,
|
|
OUT CHAR8 *FinalSymbolName
|
|
)
|
|
{
|
|
UINTN BodyLength;
|
|
BOOLEAN Result;
|
|
UINTN TotalSize;
|
|
UINTN Index;
|
|
|
|
ASSERT (ClassName != NULL);
|
|
ASSERT (FinalSymbolNameSize > 0);
|
|
ASSERT (FinalSymbolName != NULL);
|
|
|
|
BodyLength = AsciiStrLen (ClassName);
|
|
|
|
Result = OcOverflowTriAddUN (
|
|
L_STR_LEN (OSOBJ_PREFIX),
|
|
BodyLength,
|
|
L_STR_SIZE (FINAL_CLASS_TOKEN),
|
|
&TotalSize
|
|
);
|
|
if (Result || (FinalSymbolNameSize < TotalSize)) {
|
|
return FALSE;
|
|
}
|
|
|
|
Index = 0;
|
|
CopyMem (
|
|
&FinalSymbolName[Index],
|
|
OSOBJ_PREFIX,
|
|
L_STR_LEN (OSOBJ_PREFIX)
|
|
);
|
|
|
|
Index += L_STR_LEN (OSOBJ_PREFIX);
|
|
CopyMem (
|
|
&FinalSymbolName[Index],
|
|
ClassName,
|
|
BodyLength
|
|
);
|
|
|
|
Index += BodyLength;
|
|
CopyMem (
|
|
&FinalSymbolName[Index],
|
|
FINAL_CLASS_TOKEN,
|
|
L_STR_SIZE (FINAL_CLASS_TOKEN)
|
|
);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Returns whether SymbolName defines a VTable.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] SymbolName The symbol name to check.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoSymbolNameIsVtable64 (
|
|
IN CONST CHAR8 *SymbolName
|
|
)
|
|
{
|
|
ASSERT (SymbolName != NULL);
|
|
//
|
|
// Implicitely checks for METACLASS_VTABLE_PREFIX.
|
|
//
|
|
return AsciiStrnCmp (SymbolName, VTABLE_PREFIX, L_STR_LEN (VTABLE_PREFIX)) == 0;
|
|
}
|
|
|
|
/**
|
|
Returns whether the symbol name describes a C++ symbol.
|
|
|
|
@param[in] Name The name to evaluate.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoSymbolNameIsCxx (
|
|
IN CONST CHAR8 *Name
|
|
)
|
|
{
|
|
ASSERT (Name != NULL);
|
|
return AsciiStrnCmp (Name, CXX_PREFIX, L_STR_LEN (CXX_PREFIX)) == 0;
|
|
}
|
|
|
|
/**
|
|
Retrieves Metaclass symbol of a SMCP.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] Smcp The SMCP to evaluate.
|
|
|
|
@retval NULL NULL is returned on failure.
|
|
|
|
**/
|
|
MACH_NLIST_64 *
|
|
MachoGetMetaclassSymbolFromSmcpSymbol64 (
|
|
IN OUT OC_MACHO_CONTEXT *Context,
|
|
IN CONST MACH_NLIST_64 *Smcp
|
|
)
|
|
{
|
|
MACH_NLIST_64 *Symbol;
|
|
BOOLEAN Result;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (Smcp != NULL);
|
|
|
|
Result = MachoGetSymbolByRelocationOffset64 (
|
|
Context,
|
|
Smcp->Value,
|
|
&Symbol
|
|
);
|
|
if (Result && (Symbol != NULL)) {
|
|
Result = MachoSymbolNameIsMetaclassPointer64 (
|
|
Context,
|
|
MachoGetSymbolName64 (Context, Symbol)
|
|
);
|
|
if (Result) {
|
|
return Symbol;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
Retrieves VTable and Meta VTable of a SMCP.
|
|
Logically matches XNU's get_vtable_syms_from_smcp.
|
|
|
|
@param[in,out] Context Context of the Mach-O.
|
|
@param[in] SmcpName SMCP Symbol mame to retrieve the VTables from.
|
|
@param[out] Vtable Output buffer for the VTable symbol pointer.
|
|
@param[out] MetaVtable Output buffer for the Meta VTable symbol pointer.
|
|
|
|
**/
|
|
BOOLEAN
|
|
MachoGetVtableSymbolsFromSmcp64 (
|
|
IN OUT OC_MACHO_CONTEXT *Context,
|
|
IN CONST CHAR8 *SmcpName,
|
|
OUT CONST MACH_NLIST_64 **Vtable,
|
|
OUT CONST MACH_NLIST_64 **MetaVtable
|
|
)
|
|
{
|
|
CHAR8 ClassName[SYM_MAX_NAME_LEN];
|
|
CHAR8 VtableName[SYM_MAX_NAME_LEN];
|
|
CHAR8 MetaVtableName[SYM_MAX_NAME_LEN];
|
|
BOOLEAN Result;
|
|
MACH_NLIST_64 *VtableSymbol;
|
|
MACH_NLIST_64 *MetaVtableSymbol;
|
|
|
|
ASSERT (Context != NULL);
|
|
ASSERT (SmcpName != NULL);
|
|
ASSERT (Vtable != NULL);
|
|
ASSERT (MetaVtable != NULL);
|
|
|
|
Result = MachoGetClassNameFromSuperMetaClassPointer (
|
|
Context,
|
|
SmcpName,
|
|
sizeof (ClassName),
|
|
ClassName
|
|
);
|
|
if (!Result) {
|
|
return FALSE;
|
|
}
|
|
|
|
Result = MachoGetVtableNameFromClassName (
|
|
ClassName,
|
|
sizeof (VtableName),
|
|
VtableName
|
|
);
|
|
if (!Result) {
|
|
return FALSE;
|
|
}
|
|
|
|
VtableSymbol = MachoGetLocalDefinedSymbolByName (Context, VtableName);
|
|
if (VtableSymbol == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
Result = MachoGetMetaVtableNameFromClassName (
|
|
ClassName,
|
|
sizeof (MetaVtableName),
|
|
MetaVtableName
|
|
);
|
|
if (!Result) {
|
|
return FALSE;
|
|
}
|
|
|
|
MetaVtableSymbol = MachoGetLocalDefinedSymbolByName (
|
|
Context,
|
|
MetaVtableName
|
|
);
|
|
if (MetaVtableSymbol == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
*Vtable = VtableSymbol;
|
|
*MetaVtable = MetaVtableSymbol;
|
|
|
|
return TRUE;
|
|
}
|