2020-09-30 16:53:55 +02:00

1207 lines
50 KiB
C

#include <private.h>
#if defined(use_SplitSymbolsX)
# include <splitsymx.h>
# define SymMalloc(a) malloc(a)
# define SymFree(a) { if (a != NULL) { free(a); a=NULL; } }
#else
# define SymMalloc(a) MemAlloc(a)
# define SymFree(a) { if (a != NULL) { MemFree(a); a=NULL; } }
#endif // use_SplitSymbolsX
#include <strsafe.h>
#define CLEAN_PD(addr) ((addr) & ~0x3)
#define CLEAN_PD64(addr) ((addr) & ~0x3UI64)
#if defined(use_SplitSymbolsX)
BOOL
SplitSymbolsX(
LPSTR ImageName,
LPSTR SymbolsPath,
LPSTR SymbolFilePath,
DWORD SizeOfSymbolFilePath,
ULONG Flags,
PCHAR RSDSDllToLoad,
LPSTR CreatedPdb,
DWORD SizeOfPdbBuffer
)
#else
BOOL
IMAGEAPI
SplitSymbols(
LPSTR ImageName,
LPSTR SymbolsPath,
LPSTR SymbolFilePath,
ULONG Flags
)
#endif // use_SplitSymbolsX
{
// UnSafe...
HANDLE FileHandle = INVALID_HANDLE_VALUE, SymbolFileHandle = INVALID_HANDLE_VALUE;
HANDLE hMappedFile;
LPVOID ImageBase = NULL;
PIMAGE_NT_HEADERS32 NtHeaders;
LPSTR ImageFileName;
DWORD SizeOfSymbols;
ULONG_PTR ImageNameOffset;
ULONG_PTR DebugSectionStart;
PIMAGE_SECTION_HEADER DebugSection = NULL;
DWORD SectionNumber, BytesWritten, NewFileSize, HeaderSum, CheckSum;
PIMAGE_DEBUG_DIRECTORY DebugDirectory, DebugDirectories, DbgDebugDirectories = NULL;
IMAGE_DEBUG_DIRECTORY MiscDebugDirectory = {0};
IMAGE_DEBUG_DIRECTORY FpoDebugDirectory = {0};
IMAGE_DEBUG_DIRECTORY FunctionTableDir;
PIMAGE_DEBUG_DIRECTORY pFpoDebugDirectory = NULL;
DWORD DebugDirectorySize, DbgFileHeaderSize, NumberOfDebugDirectories;
IMAGE_SEPARATE_DEBUG_HEADER DbgFileHeader;
PIMAGE_EXPORT_DIRECTORY ExportDirectory;
DWORD ExportedNamesSize;
LPDWORD pp;
LPSTR ExportedNames = NULL, Src, Dst;
DWORD i, j, RvaOffset, ExportDirectorySize;
PFPO_DATA FpoTable = NULL;
DWORD FpoTableSize;
PIMAGE_ALPHA_RUNTIME_FUNCTION_ENTRY RuntimeFunctionTable, pSrc;
DWORD RuntimeFunctionTableSize;
PIMAGE_FUNCTION_ENTRY FunctionTable = NULL, pDst;
DWORD FunctionTableSize;
ULONG NumberOfFunctionTableEntries, DbgOffset;
DWORD SavedErrorCode;
BOOL InsertExtensionSubDir;
LPSTR ImageFilePathToSaveInImage;
BOOL MiscInRdata = FALSE;
BOOL DiscardFPO = Flags & SPLITSYM_EXTRACT_ALL;
BOOL MiscDebugFound, OtherDebugFound, PdbDebugFound;
BOOL fNewCvData = FALSE;
PCHAR NewDebugData = NULL;
CHAR AltPdbPath[_MAX_PATH+1];
PIMAGE_FILE_HEADER FileHeader;
PIMAGE_OPTIONAL_HEADER32 OptionalHeader;
PIMAGE_SECTION_HEADER Sections;
PCVDD pDebugCV;
#if !defined(use_SplitSymbolsX) // SplitSymbolsX recieves this as a param
DWORD SizeOfSymbolFilePath = MAX_PATH+1; // size of SymbolFilePath is assumed
#endif
LPSTR tempPtr;
HRESULT hrRetCode;
BOOL bSplitSymRetValue = FALSE; // by default, assume failure
DWORD dwLastError = 0;
__try { // SymbolsPath is manipulated within this function, but the size of the string
// is not known so wrap everything in order to catch possible exceptions.
// I still need to verify if this can possibly lead to leaked resources.
//
// verify the two required parameters are not NULL
//
if (ImageName==NULL || SymbolFilePath==NULL) {
dwLastError = ERROR_INVALID_PARAMETER;
__leave;
}
if (Flags & SPLITSYM_SYMBOLPATH_IS_SRC) {
if ( S_OK != (hrRetCode = StringCbCopy(AltPdbPath, sizeof(AltPdbPath), SymbolFilePath)) ) {
dwLastError = hrRetCode;
__leave;
}
}
//
// make ImageFileName point to the first char after the last directory in ImageName (first char of filename)
//
ImageFileName = ImageName + strlen( ImageName );
while (ImageFileName > ImageName) {
if (*ImageFileName == '\\' ||
*ImageFileName == '/' ||
*ImageFileName == ':' )
{
ImageFileName = CharNext(ImageFileName);
break;
} else {
ImageFileName = CharPrev(ImageName, ImageFileName);
}
}
//
// Put the initial symbols path into SymbolFilePath
//
if (SymbolsPath == NULL ||
SymbolsPath[ 0 ] == '\0' ||
SymbolsPath[ 0 ] == '.' )
{
CHAR TempChar;
//
// copy everything except the filename from ImageName to SymbolFilePath
//
TempChar = *ImageFileName; // temporarily null terminate ImageName immediately prior to the
*ImageFileName = '\0'; // filename
if ( S_OK != (hrRetCode = StringCchCopy(SymbolFilePath, SizeOfSymbolFilePath, ImageName)) ) {
dwLastError = hrRetCode;
__leave;
}
*ImageFileName = TempChar; // replace the character we temporarily removed
InsertExtensionSubDir = FALSE;
} else {
//
// Use the provided SymbolsPath
//
if ( S_OK != (hrRetCode = StringCchCopy(SymbolFilePath, SizeOfSymbolFilePath, SymbolsPath)) ) {
dwLastError = hrRetCode;
__leave;
}
InsertExtensionSubDir = TRUE;
}
//
// make sure SymbolFilePath ends in '\\' and that Dst points to the end of the string
//
Dst = SymbolFilePath + strlen( SymbolFilePath );
tempPtr = CharPrev(SymbolFilePath, Dst); // clean up PreFast warning
if (Dst > SymbolFilePath &&
*tempPtr != '\\' &&
*tempPtr != '/' &&
*tempPtr != ':')
{
if ( S_OK != (hrRetCode = StringCchCat(SymbolFilePath, SizeOfSymbolFilePath, "\\")) ) {
dwLastError = hrRetCode;
__leave;
}
Dst++; // fix prefast warning
}
// ImageFilePathToSaveInImage points to the end of SymbolFilePath
ImageFilePathToSaveInImage = Dst;
//
// If there is a filename extension and InsertExtensionSubDir is TRUE, copy the extension to SymbolFilePath
// as a subdirectory
//
Src = strrchr( ImageFileName, '.' );
if (Src != NULL && InsertExtensionSubDir) {
Src++; // Skip past '.'
if ( S_OK != (hrRetCode = StringCchCat(SymbolFilePath, SizeOfSymbolFilePath, Src)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCchCat(SymbolFilePath, SizeOfSymbolFilePath, "\\")) ) {
dwLastError = hrRetCode;
__leave;
}
}
//
// add the filename to SymbolFilePath
//
if ( S_OK != (hrRetCode = StringCchCat(SymbolFilePath, SizeOfSymbolFilePath, ImageFileName)) ) {
dwLastError = hrRetCode;
__leave;
}
//
// change the extension to ".dbg" (or just append ".dbg" ??)
//
Dst = strrchr(Dst, '.');
if (Dst == NULL) {
Dst = SymbolFilePath + strlen( SymbolFilePath );
} else {
*Dst = '\0';
}
if ( S_OK != (hrRetCode = StringCchCat( SymbolFilePath, SizeOfSymbolFilePath, ".dbg" )) ) {
dwLastError = hrRetCode;
__leave;
}
// Now, open and map the input file.
FileHandle = CreateFile( ImageName,
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
0,
NULL
);
if (FileHandle == INVALID_HANDLE_VALUE) {
__leave;
}
hMappedFile = CreateFileMapping( FileHandle,
NULL,
PAGE_READWRITE,
0,
0,
NULL
);
if (!hMappedFile) {
__leave;
}
ImageBase = MapViewOfFile( hMappedFile,
FILE_MAP_WRITE,
0,
0,
0
);
CloseHandle( hMappedFile );
if (!ImageBase) {
__leave;
}
//
// Everything is mapped. Now check the image and find nt image headers
//
#ifndef _WIN64
NtHeaders = ImageNtHeader( ImageBase );
if (NtHeaders == NULL) {
FileHeader = (PIMAGE_FILE_HEADER)ImageBase;
OptionalHeader = ((PIMAGE_OPTIONAL_HEADER32)((ULONG_PTR)FileHeader+IMAGE_SIZEOF_FILE_HEADER));
// One last check
if (OptionalHeader->Magic == IMAGE_NT_OPTIONAL_HDR32_MAGIC)
goto HeaderOk;
HeaderBad:
#endif
dwLastError = ERROR_BAD_EXE_FORMAT;
__leave;
#ifndef _WIN64
} else {
FileHeader = &NtHeaders->FileHeader;
OptionalHeader = &NtHeaders->OptionalHeader;
if (OptionalHeader->Magic != IMAGE_NT_OPTIONAL_HDR32_MAGIC)
goto HeaderBad;
}
HeaderOk:
if ((OptionalHeader->MajorLinkerVersion < 3) &&
(OptionalHeader->MinorLinkerVersion < 5) )
{
dwLastError = ERROR_BAD_EXE_FORMAT;
__leave;
}
{
DWORD dwDataSize;
PVOID pData;
pData = ImageDirectoryEntryToData(ImageBase, FALSE, IMAGE_DIRECTORY_ENTRY_SECURITY, &dwDataSize);
if (pData || dwDataSize) {
// This image has been signed. Can't strip the symbols w/o invalidating the certificate.
dwLastError = ERROR_BAD_EXE_FORMAT;
__leave;
}
pData = ImageDirectoryEntryToData(
ImageBase,
FALSE,
IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR,
&dwDataSize
);
if (pData) {
// COR header found - see if it's strong signed
if (((IMAGE_COR20_HEADER *)pData)->Flags & COMIMAGE_FLAGS_STRONGNAMESIGNED)
{
// This image has been strong signed. Can't strip the symbols w/o invalidating the certificate.
dwLastError = ERROR_BAD_EXE_FORMAT;
__leave;
}
}
}
if (FileHeader->Characteristics & IMAGE_FILE_DEBUG_STRIPPED)
{
// The symbols have already been stripped. No need to continue.
dwLastError = ERROR_ALREADY_ASSIGNED;
__leave;
}
DebugDirectories = (PIMAGE_DEBUG_DIRECTORY) ImageDirectoryEntryToData( ImageBase,
FALSE,
IMAGE_DIRECTORY_ENTRY_DEBUG,
&DebugDirectorySize
);
if (!DebugDirectoryIsUseful(DebugDirectories, DebugDirectorySize)) {
dwLastError = ERROR_BAD_EXE_FORMAT;
__leave;
}
NumberOfDebugDirectories = DebugDirectorySize / sizeof( IMAGE_DEBUG_DIRECTORY );
// See if there's a MISC debug dir and if not, there s/b ONLY a CV data or it's an error.
MiscDebugFound = FALSE;
OtherDebugFound = FALSE;
for (i=0,DebugDirectory=DebugDirectories; i<NumberOfDebugDirectories; i++,DebugDirectory++) {
switch (DebugDirectory->Type) {
case IMAGE_DEBUG_TYPE_MISC:
MiscDebugFound = TRUE;
break;
case IMAGE_DEBUG_TYPE_CODEVIEW:
pDebugCV = ( PCVDD ) (DebugDirectory->PointerToRawData + (PCHAR)ImageBase);
if (pDebugCV->dwSig == '01BN') {
PdbDebugFound = TRUE;
}
#if defined(use_SplitSymbolsX)
if (pDebugCV->dwSig == 'SDSR') {
PdbDebugFound = TRUE;
}
#endif
break;
default:
OtherDebugFound = TRUE;
break;
}
}
if (OtherDebugFound && !MiscDebugFound) {
dwLastError = ERROR_BAD_EXE_FORMAT;
__leave;
}
if (PdbDebugFound && !OtherDebugFound && (OptionalHeader->MajorLinkerVersion >= 6)) {
// This is a VC6 generated image. Don't create a .dbg file.
MiscDebugFound = FALSE;
}
// Make sure we can open the .dbg file before we continue...
if (!MakeSureDirectoryPathExists( SymbolFilePath )) {
__leave;
}
if (MiscDebugFound) {
// Try to open the symbol file
SymbolFileHandle = CreateFile( SymbolFilePath,
GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
0,
NULL
);
if (SymbolFileHandle == INVALID_HANDLE_VALUE) {
goto nosyms;
}
}
// The entire file is mapped so we don't have to care if the rva's
// are correct. It is interesting to note if there's a debug section
// we need to whack before terminating, though.
{
if (NtHeaders) {
Sections = IMAGE_FIRST_SECTION( NtHeaders );
} else {
Sections = (PIMAGE_SECTION_HEADER)
((ULONG_PTR)ImageBase +
((PIMAGE_FILE_HEADER)ImageBase)->SizeOfOptionalHeader +
IMAGE_SIZEOF_FILE_HEADER );
}
for (SectionNumber = 0;
SectionNumber < FileHeader->NumberOfSections;
SectionNumber++ ) {
if (Sections[ SectionNumber ].PointerToRawData != 0 &&
!_stricmp( (char *) Sections[ SectionNumber ].Name, ".debug" )) {
DebugSection = &Sections[ SectionNumber ];
}
}
}
FpoTable = NULL;
ExportedNames = NULL;
DebugSectionStart = 0xffffffff;
//
// Find the size of the debug section.
//
SizeOfSymbols = 0;
for (i=0,DebugDirectory=DebugDirectories; i<NumberOfDebugDirectories; i++,DebugDirectory++) {
switch (DebugDirectory->Type) {
case IMAGE_DEBUG_TYPE_MISC :
// Save it away.
MiscDebugDirectory = *DebugDirectory;
// check to see if the misc debug data is in some other section.
// If Address Of Raw Data is cleared, it must be in .debug (there's no such thing as not-mapped rdata)
// If it's set and there's no debug section, it must be somewhere else.
// If it's set and there's a debug section, check the range.
if ((DebugDirectory->AddressOfRawData != 0) &&
((DebugSection == NULL) ||
(((DebugDirectory->PointerToRawData < DebugSection->PointerToRawData) ||
(DebugDirectory->PointerToRawData >= DebugSection->PointerToRawData + DebugSection->SizeOfRawData)
)
)
)
)
{
MiscInRdata = TRUE;
} else {
if (DebugDirectory->PointerToRawData < DebugSectionStart) {
DebugSectionStart = DebugDirectory->PointerToRawData;
}
}
break;
case IMAGE_DEBUG_TYPE_FPO:
if (DebugDirectory->PointerToRawData < DebugSectionStart) {
DebugSectionStart = DebugDirectory->PointerToRawData;
}
// Save it away.
FpoDebugDirectory = *DebugDirectory;
pFpoDebugDirectory = DebugDirectory;
break;
case IMAGE_DEBUG_TYPE_CODEVIEW:
{
ULONG NewDebugSize;
if (DebugDirectory->PointerToRawData < DebugSectionStart) {
DebugSectionStart = DebugDirectory->PointerToRawData;
}
// If private's are removed do so to the static CV data and save the new size...
pDebugCV = ( PCVDD ) (DebugDirectory->PointerToRawData + (PCHAR)ImageBase);
if (pDebugCV->dwSig == '01BN') {
// Got a PDB. The name immediately follows the signature.
CHAR PdbName[_MAX_PATH + 1];
CHAR NewPdbName[_MAX_PATH + 1];
CHAR Drive[_MAX_DRIVE + 1];
CHAR Dir[_MAX_DIR + 1];
CHAR Filename[_MAX_FNAME + 1];
CHAR FileExt[_MAX_EXT + 1];
BOOL rc;
memset(PdbName, 0, sizeof(PdbName));
memcpy(PdbName, ((PCHAR)pDebugCV)+ sizeof(NB10IH), DebugDirectory->SizeOfData - sizeof(NB10IH));
_splitpath(PdbName, NULL, NULL, Filename, FileExt);
_splitpath(SymbolFilePath, Drive, Dir, NULL, NULL);
_makepath(NewPdbName, Drive, Dir, Filename, FileExt);
#if defined(use_SplitSymbolsX)
rc = CopyPdbX(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE, NULL);
if (rc) {
StringCchCopy(CreatedPdb, SizeOfPdbBuffer, NewPdbName);
}
#else
rc = CopyPdb(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE);
#endif
if (!rc) {
if (Flags & SPLITSYM_SYMBOLPATH_IS_SRC) {
// Try the AltPdbPath.
if ( S_OK != (hrRetCode = StringCbCopy(PdbName, sizeof(PdbName), AltPdbPath)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( PdbName, sizeof(PdbName), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( PdbName, sizeof(PdbName), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
#if defined(use_SplitSymbolsX)
rc = CopyPdbX(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE, NULL);
if (rc) {
StringCchCopy(CreatedPdb, SizeOfPdbBuffer, NewPdbName);
}
#else
rc = CopyPdb(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE);
#endif
}
if ( !rc) {
// It's possible the name in the pdb isn't in the same location as it was when built. See if we can
// find it in the same dir as the image...
_splitpath(ImageName, Drive, Dir, NULL, NULL);
_makepath(PdbName, Drive, Dir, Filename, FileExt);
#if defined(use_SplitSymbolsX)
rc = CopyPdbX(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE, NULL);
if (rc) {
StringCchCopy(CreatedPdb, SizeOfPdbBuffer, NewPdbName);
}
#else
rc = CopyPdb(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE);
#endif
}
}
if (rc) {
SetFileAttributes(NewPdbName, FILE_ATTRIBUTE_NORMAL);
// Change the data so only the pdb name is in the .dbg file (no path).
if (MiscDebugFound) {
NewDebugSize = sizeof(NB10IH) + strlen(Filename) + strlen(FileExt) + 1;
NewDebugData = (PCHAR) SymMalloc( NewDebugSize );
((PCVDD)NewDebugData)->nb10ih = pDebugCV->nb10ih;
if ( S_OK != (hrRetCode = StringCbCopy(NewDebugData + sizeof(NB10IH), NewDebugSize-sizeof(NB10IH), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( NewDebugData + sizeof(NB10IH), NewDebugSize-sizeof(NB10IH), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
DebugDirectory->PointerToRawData = (ULONG) (NewDebugData - (PCHAR)ImageBase);
DebugDirectory->SizeOfData = NewDebugSize;
} else {
if ( S_OK != (hrRetCode = StringCbCopy( ((PCHAR)pDebugCV) + sizeof(NB10IH), DebugDirectory->SizeOfData - sizeof(NB10IH), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( ((PCHAR)pDebugCV)+ sizeof(NB10IH), DebugDirectory->SizeOfData - sizeof(NB10IH), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
}
} else {
// Replace <Path>\<filename>.<ext> with just <filename>.<ext> in the debug data
if ( S_OK != (hrRetCode = StringCbCopy(((PCHAR)pDebugCV) + sizeof(NB10IH), DebugDirectory->SizeOfData - sizeof(NB10IH), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( ((PCHAR)pDebugCV) + sizeof(NB10IH), DebugDirectory->SizeOfData - sizeof(NB10IH), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
DebugDirectory->SizeOfData = sizeof(NB10IH) + strlen(Filename) + strlen(FileExt) + 1;
}
#if defined(use_SplitSymbolsX)
} else if ( pDebugCV->dwSig == 'SDSR') {
// Got a PDB. The name immediately follows the signature.
CHAR PdbName[sizeof(((PRSDSI)(0))->szPdb)];
CHAR NewPdbName[_MAX_PATH+1];
CHAR Drive[_MAX_DRIVE+1];
CHAR Dir[_MAX_DIR+1];
CHAR Filename[_MAX_FNAME+1];
CHAR FileExt[_MAX_EXT+1];
BOOL rc;
ZeroMemory(PdbName, sizeof(PdbName));
memcpy(PdbName, ((PCHAR)pDebugCV)+ sizeof(RSDSIH), __min(DebugDirectory->SizeOfData - sizeof(RSDSIH), sizeof(PdbName)));
_splitpath(PdbName, NULL, NULL, Filename, FileExt);
_splitpath(SymbolFilePath, Drive, Dir, NULL, NULL);
_makepath(NewPdbName, Drive, Dir, Filename, FileExt);
rc = CopyPdbX(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE, RSDSDllToLoad);
if (rc) {
StringCchCopy(CreatedPdb, SizeOfPdbBuffer, NewPdbName);
}
if (!rc) {
if (Flags & SPLITSYM_SYMBOLPATH_IS_SRC) {
// Try the AltPdbPath.
if ( S_OK != (hrRetCode = StringCbCopy(PdbName, sizeof(PdbName), AltPdbPath)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( PdbName, sizeof(PdbName), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( PdbName, sizeof(PdbName), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
rc = CopyPdbX(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE, RSDSDllToLoad);
if (rc) {
StringCchCopy(CreatedPdb, SizeOfPdbBuffer, NewPdbName);
}
}
if ( !rc) {
// It's possible the name in the pdb isn't in the same location as it was when built. See if we can
// find it in the same dir as the image...
_splitpath(ImageName, Drive, Dir, NULL, NULL);
_makepath(PdbName, Drive, Dir, Filename, FileExt);
rc = CopyPdbX(PdbName, NewPdbName, Flags & SPLITSYM_REMOVE_PRIVATE, RSDSDllToLoad);
if (rc) {
StringCchCopy(CreatedPdb, SizeOfPdbBuffer, NewPdbName);
}
}
}
if (rc) {
SetFileAttributes(NewPdbName, FILE_ATTRIBUTE_NORMAL);
// Change the data so only the pdb name is in the .dbg file (no path).
if (MiscDebugFound) {
NewDebugSize = sizeof(RSDSIH) + strlen(Filename) + strlen(FileExt) + 1;
NewDebugData = (PCHAR) SymMalloc( NewDebugSize );
((PCVDD)NewDebugData)->rsdsih = pDebugCV->rsdsih;
if ( S_OK != (hrRetCode = StringCbCopy(NewDebugData + sizeof(RSDSIH), NewDebugSize-sizeof(RSDSIH), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( NewDebugData + sizeof(RSDSIH), NewDebugSize-sizeof(RSDSIH), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
DebugDirectory->PointerToRawData = (ULONG) (NewDebugData - (PCHAR)ImageBase);
DebugDirectory->SizeOfData = NewDebugSize;
} else {
if ( S_OK != (hrRetCode = StringCbCopy(((PCHAR)pDebugCV) + sizeof(RSDSIH), DebugDirectory->SizeOfData - sizeof(RSDSIH), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( ((PCHAR)pDebugCV) + sizeof(RSDSIH), DebugDirectory->SizeOfData - sizeof(RSDSIH), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
}
} else {
// Replace <Path>\<filename>.<ext> with just <filename>.<ext> in the debug data
if ( S_OK != (hrRetCode = StringCbCopy(((PCHAR)pDebugCV) + sizeof(RSDSIH), DebugDirectory->SizeOfData - sizeof(RSDSIH), Filename)) ) {
dwLastError = hrRetCode;
__leave;
}
if ( S_OK != (hrRetCode = StringCbCat( ((PCHAR)pDebugCV) + sizeof(RSDSIH), DebugDirectory->SizeOfData - sizeof(RSDSIH), FileExt)) ) {
dwLastError = hrRetCode;
__leave;
}
DebugDirectory->SizeOfData = sizeof(RSDSIH) + strlen(Filename) + strlen(FileExt) + 1;
}
#endif
} else {
if (Flags & SPLITSYM_REMOVE_PRIVATE) {
if (RemovePrivateCvSymbolicEx(DebugDirectory->PointerToRawData + (PCHAR)ImageBase,
DebugDirectory->SizeOfData,
&NewDebugData,
&NewDebugSize)) {
if (DebugDirectory->PointerToRawData != (ULONG) (NewDebugData - (PCHAR)ImageBase))
{
DebugDirectory->PointerToRawData = (ULONG) (NewDebugData - (PCHAR)ImageBase);
DebugDirectory->SizeOfData = NewDebugSize;
} else {
NewDebugData = NULL;
}
}
}
}
}
break;
case IMAGE_DEBUG_TYPE_OMAP_TO_SRC:
case IMAGE_DEBUG_TYPE_OMAP_FROM_SRC:
if (DebugDirectory->PointerToRawData < DebugSectionStart) {
DebugSectionStart = DebugDirectory->PointerToRawData;
}
// W/o the OMAP, FPO is useless.
DiscardFPO = TRUE;
break;
case IMAGE_DEBUG_TYPE_FIXUP:
if (DebugDirectory->PointerToRawData < DebugSectionStart) {
DebugSectionStart = DebugDirectory->PointerToRawData;
}
// If all PRIVATE debug is removed, don't send FIXUP along.
if (Flags & SPLITSYM_REMOVE_PRIVATE) {
DebugDirectory->SizeOfData = 0;
}
break;
default:
if (DebugDirectory->SizeOfData &&
(DebugDirectory->PointerToRawData < DebugSectionStart))
{
DebugSectionStart = DebugDirectory->PointerToRawData;
}
// Nothing else to special case...
break;
}
SizeOfSymbols += (DebugDirectory->SizeOfData + 3) & ~3; // Minimally align it all.
}
if (!MiscDebugFound) {
NewFileSize = GetFileSize(FileHandle, NULL);
CheckSumMappedFile( ImageBase,
NewFileSize,
&HeaderSum,
&CheckSum
);
OptionalHeader->CheckSum = CheckSum;
goto nomisc;
}
if (DiscardFPO) {
pFpoDebugDirectory = NULL;
}
if (pFpoDebugDirectory) {
// If FPO stays here, make a copy so we don't need to worry about stomping on it.
FpoTableSize = pFpoDebugDirectory->SizeOfData;
FpoTable = (PFPO_DATA) SymMalloc( FpoTableSize );
if ( FpoTable == NULL ) {
goto nosyms;
}
RtlMoveMemory( FpoTable,
(PCHAR) ImageBase + pFpoDebugDirectory->PointerToRawData,
FpoTableSize );
}
ExportDirectory = (PIMAGE_EXPORT_DIRECTORY)
ImageDirectoryEntryToData( ImageBase,
FALSE,
IMAGE_DIRECTORY_ENTRY_EXPORT,
&ExportDirectorySize
);
if (ExportDirectory) {
//
// This particular piece of magic gets us the RVA of the
// EXPORT section. Dont ask.
//
RvaOffset = (ULONG_PTR)
ImageDirectoryEntryToData( ImageBase,
TRUE,
IMAGE_DIRECTORY_ENTRY_EXPORT,
&ExportDirectorySize
) - (ULONG_PTR)ImageBase;
pp = (LPDWORD)((ULONG_PTR)ExportDirectory +
(ULONG_PTR)ExportDirectory->AddressOfNames - RvaOffset
);
ExportedNamesSize = 1;
for (i=0; i<ExportDirectory->NumberOfNames; i++) {
Src = (LPSTR)((ULONG_PTR)ExportDirectory + *pp++ - RvaOffset);
ExportedNamesSize += strlen( Src ) + 1;
}
ExportedNamesSize = (ExportedNamesSize + 16) & ~15;
Dst = (LPSTR) SymMalloc( ExportedNamesSize );
if (Dst != NULL) {
ExportedNames = Dst;
pp = (LPDWORD)((ULONG_PTR)ExportDirectory +
(ULONG_PTR)ExportDirectory->AddressOfNames - RvaOffset
);
for (i=0; i<ExportDirectory->NumberOfNames; i++) {
Src = (LPSTR)((ULONG_PTR)ExportDirectory + *pp++ - RvaOffset);
while (*Dst++ = *Src++) {
;
}
}
}
} else {
ExportedNamesSize = 0;
}
RuntimeFunctionTable = (PIMAGE_ALPHA_RUNTIME_FUNCTION_ENTRY)
ImageDirectoryEntryToData( ImageBase,
FALSE,
IMAGE_DIRECTORY_ENTRY_EXCEPTION,
&RuntimeFunctionTableSize
);
if (RuntimeFunctionTable == NULL) {
RuntimeFunctionTableSize = 0;
FunctionTableSize = 0;
FunctionTable = NULL;
}
else {
NumberOfFunctionTableEntries = RuntimeFunctionTableSize / sizeof( IMAGE_ALPHA_RUNTIME_FUNCTION_ENTRY );
FunctionTableSize = NumberOfFunctionTableEntries * sizeof( IMAGE_FUNCTION_ENTRY );
FunctionTable = (PIMAGE_FUNCTION_ENTRY) SymMalloc( FunctionTableSize );
if (FunctionTable == NULL) {
goto nosyms;
}
pSrc = RuntimeFunctionTable;
pDst = FunctionTable;
for (i=0; i<NumberOfFunctionTableEntries; i++) {
//
// Make .pdata entries in .DBG file relative.
//
pDst->StartingAddress = CLEAN_PD(pSrc->BeginAddress) - OptionalHeader->ImageBase;
pDst->EndingAddress = CLEAN_PD(pSrc->EndAddress) - OptionalHeader->ImageBase;
pDst->EndOfPrologue = CLEAN_PD(pSrc->PrologEndAddress) - OptionalHeader->ImageBase;
pSrc += 1;
pDst += 1;
}
}
DbgFileHeaderSize = sizeof( DbgFileHeader ) +
((FileHeader->NumberOfSections - (DebugSection ? 1 : 0)) *
sizeof( IMAGE_SECTION_HEADER )) +
ExportedNamesSize +
FunctionTableSize +
DebugDirectorySize;
if (FunctionTable != NULL) {
DbgFileHeaderSize += sizeof( IMAGE_DEBUG_DIRECTORY );
memset( &FunctionTableDir, 0, sizeof( IMAGE_DEBUG_DIRECTORY ) );
FunctionTableDir.Type = IMAGE_DEBUG_TYPE_EXCEPTION;
FunctionTableDir.SizeOfData = FunctionTableSize;
FunctionTableDir.PointerToRawData = DbgFileHeaderSize - FunctionTableSize;
}
DbgFileHeaderSize = ((DbgFileHeaderSize + 15) & ~15);
BytesWritten = 0;
if (SetFilePointer( SymbolFileHandle,
DbgFileHeaderSize,
NULL,
FILE_BEGIN
) == DbgFileHeaderSize ) {
for (i=0, DebugDirectory=DebugDirectories;
i < NumberOfDebugDirectories;
i++, DebugDirectory++) {
DWORD WriteCount;
if (DebugDirectory->SizeOfData) {
WriteFile( SymbolFileHandle,
(PCHAR) ImageBase + DebugDirectory->PointerToRawData,
(DebugDirectory->SizeOfData +3) & ~3,
&WriteCount,
NULL );
BytesWritten += WriteCount;
}
}
}
if (BytesWritten == SizeOfSymbols) {
FileHeader->PointerToSymbolTable = 0;
FileHeader->NumberOfSymbols = 0;
FileHeader->Characteristics |= IMAGE_FILE_DEBUG_STRIPPED;
if (DebugSection != NULL) {
OptionalHeader->SizeOfImage = DebugSection->VirtualAddress;
OptionalHeader->SizeOfInitializedData -= DebugSection->SizeOfRawData;
FileHeader->NumberOfSections--;
// NULL out that section
memset(DebugSection, 0, IMAGE_SIZEOF_SECTION_HEADER);
}
NewFileSize = DebugSectionStart; // Start with no symbolic
//
// Now that the data has moved to the .dbg file, rebuild the original
// with MISC debug first and FPO second.
//
if (MiscDebugDirectory.SizeOfData) {
if (MiscInRdata) {
// Just store the new name in the existing misc field...
ImageNameOffset = (ULONG_PTR) ((PCHAR)ImageBase +
MiscDebugDirectory.PointerToRawData +
FIELD_OFFSET( IMAGE_DEBUG_MISC, Data ));
RtlCopyMemory( (LPVOID) ImageNameOffset,
ImageFilePathToSaveInImage,
strlen(ImageFilePathToSaveInImage) + 1 );
} else {
if (DebugSectionStart != MiscDebugDirectory.PointerToRawData) {
RtlMoveMemory((PCHAR) ImageBase + DebugSectionStart,
(PCHAR) ImageBase + MiscDebugDirectory.PointerToRawData,
MiscDebugDirectory.SizeOfData);
}
ImageNameOffset = (ULONG_PTR) ((PCHAR)ImageBase + DebugSectionStart +
FIELD_OFFSET( IMAGE_DEBUG_MISC, Data ));
RtlCopyMemory( (LPVOID)ImageNameOffset,
ImageFilePathToSaveInImage,
strlen(ImageFilePathToSaveInImage) + 1 );
NewFileSize += MiscDebugDirectory.SizeOfData;
NewFileSize = (NewFileSize + 3) & ~3;
}
}
if (FpoTable) {
RtlCopyMemory( (PCHAR) ImageBase + NewFileSize,
FpoTable,
FpoTableSize );
NewFileSize += FpoTableSize;
NewFileSize = (NewFileSize + 3) & ~3;
}
// Make a copy of the Debug directory that we can write into the .dbg file
DbgDebugDirectories = (PIMAGE_DEBUG_DIRECTORY) SymMalloc( NumberOfDebugDirectories * sizeof(IMAGE_DEBUG_DIRECTORY) );
RtlMoveMemory(DbgDebugDirectories,
DebugDirectories,
sizeof(IMAGE_DEBUG_DIRECTORY) * NumberOfDebugDirectories);
// Then write the MISC and (perhaps) FPO data to the image.
FpoDebugDirectory.PointerToRawData = DebugSectionStart;
DebugDirectorySize = 0;
if (MiscDebugDirectory.SizeOfData != 0) {
if (!MiscInRdata) {
MiscDebugDirectory.PointerToRawData = DebugSectionStart;
FpoDebugDirectory.PointerToRawData += MiscDebugDirectory.SizeOfData;
MiscDebugDirectory.AddressOfRawData = 0;
}
DebugDirectories[0] = MiscDebugDirectory;
DebugDirectorySize += sizeof(IMAGE_DEBUG_DIRECTORY);
}
if (pFpoDebugDirectory) {
FpoDebugDirectory.AddressOfRawData = 0;
DebugDirectories[DebugDirectorySize / sizeof(IMAGE_DEBUG_DIRECTORY)] = FpoDebugDirectory;
DebugDirectorySize += sizeof(IMAGE_DEBUG_DIRECTORY);
}
// Zero out remaining slots in image.
if (NumberOfDebugDirectories < (DebugDirectorySize / sizeof(IMAGE_DEBUG_DIRECTORY))) {
ZeroMemory(&DebugDirectories[DebugDirectorySize / sizeof(IMAGE_DEBUG_DIRECTORY)],
NumberOfDebugDirectories * sizeof(IMAGE_DEBUG_DIRECTORY) -
DebugDirectorySize);
}
OptionalHeader->DataDirectory[IMAGE_DIRECTORY_ENTRY_DEBUG].Size = DebugDirectorySize;
DbgOffset = DbgFileHeaderSize;
for (i = 0, j=0, DebugDirectory=DbgDebugDirectories;
i < NumberOfDebugDirectories; i++) {
if (DebugDirectory[i].SizeOfData) {
DebugDirectory[j] = DebugDirectory[i];
DebugDirectory[j].AddressOfRawData = 0;
DebugDirectory[j].PointerToRawData = DbgOffset;
DbgOffset += (DebugDirectory[j].SizeOfData + 3 )& ~3;
j++;
}
}
if (FunctionTable) {
FunctionTableDir.PointerToRawData -= sizeof(IMAGE_DEBUG_DIRECTORY) * (NumberOfDebugDirectories - j);
}
NumberOfDebugDirectories = j;
CheckSumMappedFile( ImageBase,
NewFileSize,
&HeaderSum,
&CheckSum
);
OptionalHeader->CheckSum = CheckSum;
DbgFileHeader.Signature = IMAGE_SEPARATE_DEBUG_SIGNATURE;
DbgFileHeader.Flags = 0;
DbgFileHeader.Machine = FileHeader->Machine;
DbgFileHeader.Characteristics = FileHeader->Characteristics;
DbgFileHeader.TimeDateStamp = FileHeader->TimeDateStamp;
DbgFileHeader.CheckSum = CheckSum;
DbgFileHeader.ImageBase = OptionalHeader->ImageBase;
DbgFileHeader.SizeOfImage = OptionalHeader->SizeOfImage;
DbgFileHeader.ExportedNamesSize = ExportedNamesSize;
DbgFileHeader.DebugDirectorySize = NumberOfDebugDirectories * sizeof(IMAGE_DEBUG_DIRECTORY);
if (FunctionTable) {
DbgFileHeader.DebugDirectorySize += sizeof (IMAGE_DEBUG_DIRECTORY);
}
DbgFileHeader.NumberOfSections = FileHeader->NumberOfSections;
memset( DbgFileHeader.Reserved, 0, sizeof( DbgFileHeader.Reserved ) );
DbgFileHeader.SectionAlignment = OptionalHeader->SectionAlignment;
SetFilePointer( SymbolFileHandle, 0, NULL, FILE_BEGIN );
WriteFile( SymbolFileHandle,
&DbgFileHeader,
sizeof( DbgFileHeader ),
&BytesWritten,
NULL
);
if (NtHeaders) {
Sections = IMAGE_FIRST_SECTION( NtHeaders );
} else {
Sections = (PIMAGE_SECTION_HEADER)
((ULONG_PTR)ImageBase +
((PIMAGE_FILE_HEADER)ImageBase)->SizeOfOptionalHeader +
IMAGE_SIZEOF_FILE_HEADER );
}
WriteFile( SymbolFileHandle,
(PVOID)Sections,
sizeof( IMAGE_SECTION_HEADER ) * FileHeader->NumberOfSections,
&BytesWritten,
NULL
);
if (ExportedNamesSize) {
WriteFile( SymbolFileHandle,
ExportedNames,
ExportedNamesSize,
&BytesWritten,
NULL
);
}
WriteFile( SymbolFileHandle,
DbgDebugDirectories,
sizeof (IMAGE_DEBUG_DIRECTORY) * NumberOfDebugDirectories,
&BytesWritten,
NULL );
if (FunctionTable) {
WriteFile( SymbolFileHandle,
&FunctionTableDir,
sizeof (IMAGE_DEBUG_DIRECTORY),
&BytesWritten,
NULL );
WriteFile( SymbolFileHandle,
FunctionTable,
FunctionTableSize,
&BytesWritten,
NULL
);
}
SetFilePointer( SymbolFileHandle, 0, NULL, FILE_END );
nomisc:
FlushViewOfFile( ImageBase, NewFileSize );
UnmapViewOfFile( ImageBase );
SetFilePointer( FileHandle, NewFileSize, NULL, FILE_BEGIN );
SetEndOfFile( FileHandle );
TouchFileTimes( FileHandle, NULL );
bSplitSymRetValue=TRUE;
__leave;
} else {
CloseHandle( SymbolFileHandle );
DeleteFile( SymbolFilePath );
}
nosyms:
dwLastError = GetLastError();
#endif
} __except(EXCEPTION_EXECUTE_HANDLER) {
dwLastError = GetExceptionCode();
}
// SymFree checks that the pointer passed isn't null and
// sets the pointer to null after freeing the memory
SymFree(DbgDebugDirectories);
SymFree(ExportedNames);
SymFree(FpoTable);
SymFree(FunctionTable);
SymFree(NewDebugData);
if (FileHandle != INVALID_HANDLE_VALUE) {
CloseHandle(FileHandle);
}
if (SymbolFileHandle != INVALID_HANDLE_VALUE) {
CloseHandle(SymbolFileHandle);
}
if (ImageBase) {
UnmapViewOfFile(ImageBase);
}
if (dwLastError != 0) {
SetLastError(dwLastError);
}
return(bSplitSymRetValue);
}
#if defined(use_SplitSymbolsX)
LPSTR CharNext(
LPCSTR lpCurrentChar)
{
if (IsDBCSLeadByte(*lpCurrentChar)) {
lpCurrentChar++;
}
/*
* if we have only DBCS LeadingByte, we will point string-terminaler.
*/
if (*lpCurrentChar) {
lpCurrentChar++;
}
return (LPSTR)lpCurrentChar;
}
LPSTR CharPrev(
LPCSTR lpStart,
LPCSTR lpCurrentChar)
{
if (lpCurrentChar > lpStart) {
LPCSTR lpChar;
BOOL bDBC = FALSE;
for (lpChar = --lpCurrentChar - 1 ; lpChar >= lpStart ; lpChar--) {
if (!IsDBCSLeadByte(*lpChar))
break;
bDBC = !bDBC;
}
if (bDBC)
lpCurrentChar--;
}
return (LPSTR)lpCurrentChar;
}
#endif