b2e6fbd0ae
script generated to consistently point to the top-level copyright. Signed-off-by: Pavan Balaji <balaji@anl.gov>
141 linhas
5.3 KiB
C++
141 linhas
5.3 KiB
C++
/*
|
|
* Copyright (C) by Argonne National Laboratory
|
|
* See COPYRIGHT in top-level directory
|
|
*/
|
|
|
|
#include <windows.h>
|
|
#include <stdio.h>
|
|
#include <time.h>
|
|
|
|
/* This function returns a pointer to the section header containing the RVA. Return NULL on error */
|
|
template < typename NTHeaderType > static inline PIMAGE_SECTION_HEADER GetSecHeader(DWORD rva,
|
|
NTHeaderType *
|
|
pNTHeader)
|
|
{
|
|
PIMAGE_SECTION_HEADER section;
|
|
section = IMAGE_FIRST_SECTION(pNTHeader);
|
|
for (int i = 0; i < pNTHeader->FileHeader.NumberOfSections; i++, section++) {
|
|
DWORD size = section->Misc.VirtualSize;
|
|
DWORD secRVA = section->VirtualAddress;
|
|
if ((secRVA <= rva) && (rva < (secRVA + size))) {
|
|
return section;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* This function returns the real address from the RVA. Returns NULL on error */
|
|
template < typename NTHeaderType > static inline LPVOID GetRealAddrFromRVA(DWORD rva,
|
|
NTHeaderType * pNTHeader,
|
|
PBYTE imageBase)
|
|
{
|
|
PIMAGE_SECTION_HEADER pSectionHdr = GetSecHeader(rva, pNTHeader);
|
|
if (!pSectionHdr) {
|
|
return NULL;
|
|
}
|
|
/* Since we map the DLL manually we need to calculate mapShift to get the real addr */
|
|
INT mapShift = (INT) (pSectionHdr->VirtualAddress - pSectionHdr->PointerToRawData);
|
|
|
|
return (PVOID) (imageBase + rva - mapShift);
|
|
}
|
|
|
|
/* This function reads the exported symbols from a dll/exe and prints a DLL definition file, compatible with gcc,
|
|
with those symbols.
|
|
INPUT:
|
|
pImageBase : Pointer to the base of the dll image, offset = 0
|
|
pNTHeader : Pointer to the NT header (32-bit/64-bit) section of image.
|
|
*/
|
|
template < typename NTHeaderType > void PrintDefFileWithExpSymbols(PBYTE pImageBase,
|
|
NTHeaderType * pNTHeader)
|
|
{
|
|
/* Export Section RVA */
|
|
DWORD expSecRVA =
|
|
pNTHeader->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT].VirtualAddress;
|
|
|
|
/* From RVA get the real addrs for the various components of the exports section */
|
|
PIMAGE_EXPORT_DIRECTORY pExportDir =
|
|
(PIMAGE_EXPORT_DIRECTORY) GetRealAddrFromRVA(expSecRVA, pNTHeader, pImageBase);
|
|
PDWORD pszFuncNames =
|
|
(PDWORD) GetRealAddrFromRVA(pExportDir->AddressOfNames, pNTHeader, pImageBase);
|
|
|
|
printf("EXPORTS\n");
|
|
|
|
for (unsigned int i = 0; i < pExportDir->NumberOfFunctions; i++) {
|
|
/* FIXME: Assumes that the current DLL does not have ordinal nums */
|
|
if (i < pExportDir->NumberOfNames) {
|
|
/* Ordinal numbers start at 1 - not 0 */
|
|
printf("\t%s @ %d ;\n", GetRealAddrFromRVA(pszFuncNames[i], pNTHeader, pImageBase),
|
|
i + 1);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void HelpUsage(char *exe_name)
|
|
{
|
|
printf("Usage: %s <DLLNAME>\n", exe_name);
|
|
printf
|
|
("\t\t The tool reads the export section from <DLLNAME> and prints the definition file for gcc to stdout\n");
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
HANDLE hFile, hFileMapping;
|
|
LPVOID lpFileBase;
|
|
PIMAGE_DOS_HEADER pImgHeader;
|
|
PIMAGE_NT_HEADERS pImgNTHeader;
|
|
PIMAGE_NT_HEADERS64 pImgNTHeader64;
|
|
|
|
if ((argc < 2) || (strcmp(argv[1], "/?") == 0) || (strcmp(argv[1], "--help") == 0)) {
|
|
HelpUsage(argv[0]);
|
|
exit(1);
|
|
}
|
|
|
|
/* Open the DLL and map it to memory */
|
|
hFile =
|
|
CreateFile(argv[1], GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL, 0);
|
|
if (hFile == INVALID_HANDLE_VALUE) {
|
|
printf("Unable to open dll, %s\n", argv[1]);
|
|
exit(-1);
|
|
}
|
|
|
|
hFileMapping = CreateFileMapping(hFile, NULL, PAGE_READONLY, 0, 0, NULL);
|
|
if (hFileMapping == 0) {
|
|
CloseHandle(hFile);
|
|
printf("Cannot open file mapping for dll, %s\n", argv[1]);
|
|
exit(-1);
|
|
}
|
|
|
|
lpFileBase = MapViewOfFile(hFileMapping, FILE_MAP_READ, 0, 0, 0);
|
|
if (lpFileBase == 0) {
|
|
CloseHandle(hFileMapping);
|
|
CloseHandle(hFile);
|
|
printf("Cannot map view of dll, %s\n", argv[1]);
|
|
exit(-1);
|
|
}
|
|
|
|
/* pImgHeader points to the MS-DOS HEADER section, offset=0, of dll */
|
|
pImgHeader = (PIMAGE_DOS_HEADER) lpFileBase;
|
|
|
|
/* pImgNTHeader points to the NT HEADER section of the dll, pImgHeader + <File address of new exe header> */
|
|
pImgNTHeader = (PIMAGE_NT_HEADERS) ((DWORD_PTR) pImgHeader + (DWORD_PTR) pImgHeader->e_lfanew);
|
|
|
|
/* In the case of the 64-bit dlls, the Optional header section within NT header has 64-bit fields */
|
|
pImgNTHeader64 = (PIMAGE_NT_HEADERS64) pImgNTHeader;
|
|
|
|
/* The Magic number is always a WORD. This number is used to determine if the dll is 32-bit or 64-bit */
|
|
if (pImgNTHeader->OptionalHeader.Magic == IMAGE_NT_OPTIONAL_HDR64_MAGIC) {
|
|
/* 64-bit dll */
|
|
PrintDefFileWithExpSymbols((PBYTE) pImgHeader, pImgNTHeader64);
|
|
} else {
|
|
/* 32-bit dll */
|
|
PrintDefFileWithExpSymbols((PBYTE) pImgHeader, pImgNTHeader);
|
|
}
|
|
|
|
UnmapViewOfFile(lpFileBase);
|
|
CloseHandle(hFileMapping);
|
|
CloseHandle(hFile);
|
|
}
|