⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pe_image.c

📁 君正早期ucos系统(只有早期的才不没有打包成库),MPLAYER,文件系统,图片解码,浏览,电子书,录音,想学ucos,识货的人就下吧 russblock fmradio explore set
💻 C
📖 第 1 页 / 共 3 页
字号:
		    int ordinal = IMAGE_ORDINAL(thunk_list->u1.Ordinal);		    TRACE("--- Ordinal %s.%d\n",name,ordinal);		    thunk_list->u1.Function=LookupExternal(		      name, ordinal);		} else {		    pe_name=(PIMAGE_IMPORT_BY_NAME) RVA(thunk_list->u1.AddressOfData);		    TRACE("--- %s %s.%d\n",		   		  pe_name->Name,name,pe_name->Hint);		    thunk_list->u1.Function=LookupExternalByName(		      name, pe_name->Name);		}		thunk_list++;	    }	}    }    return 0;}static int calc_vma_size( HMODULE hModule ){    int i,vma_size = 0;    IMAGE_SECTION_HEADER *pe_seg = PE_SECTIONS(hModule);    TRACE("Dump of segment table\n");    TRACE("   Name    VSz  Vaddr     SzRaw   Fileadr  *Reloc *Lineum #Reloc #Linum Char\n");    for (i = 0; i< PE_HEADER(hModule)->FileHeader.NumberOfSections; i++)    {        TRACE("%8s: %4.4lx %8.8lx %8.8lx %8.8lx %8.8lx %8.8lx %4.4x %4.4x %8.8lx\n",                       pe_seg->Name,                       pe_seg->Misc.VirtualSize,                      pe_seg->VirtualAddress,                      pe_seg->SizeOfRawData,                      pe_seg->PointerToRawData,                      pe_seg->PointerToRelocations,                      pe_seg->PointerToLinenumbers,                      pe_seg->NumberOfRelocations,                      pe_seg->NumberOfLinenumbers,                      pe_seg->Characteristics);        vma_size=max(vma_size, pe_seg->VirtualAddress+pe_seg->SizeOfRawData);        vma_size=max(vma_size, pe_seg->VirtualAddress+pe_seg->Misc.VirtualSize);        pe_seg++;    }    return vma_size;}static void do_relocations( unsigned int load_addr, IMAGE_BASE_RELOCATION *r ){    int delta = load_addr - PE_HEADER(load_addr)->OptionalHeader.ImageBase;    int	hdelta = (delta >> 16) & 0xFFFF;    int	ldelta = delta & 0xFFFF;	if(delta == 0)				return;	while(r->VirtualAddress)	{		char *page = (char*) RVA(r->VirtualAddress);		int count = (r->SizeOfBlock - 8)/2;		int i;		TRACE_(fixup)("%x relocations for page %lx\n",			count, r->VirtualAddress);				for(i=0;i<count;i++)		{			int offset = r->TypeOffset[i] & 0xFFF;			int type = r->TypeOffset[i] >> 12;//			TRACE_(fixup)("patching %x type %x\n", offset, type);			switch(type)			{			case IMAGE_REL_BASED_ABSOLUTE: break;			case IMAGE_REL_BASED_HIGH:				*(short*)(page+offset) += hdelta;				break;			case IMAGE_REL_BASED_LOW:				*(short*)(page+offset) += ldelta;				break;			case IMAGE_REL_BASED_HIGHLOW:				*(int*)(page+offset) += delta;								break;			case IMAGE_REL_BASED_HIGHADJ:				FIXME("Don't know what to do with IMAGE_REL_BASED_HIGHADJ\n");				break;			case IMAGE_REL_BASED_MIPS_JMPADDR:				FIXME("Is this a MIPS machine ???\n");				break;			default:				FIXME("Unknown fixup type\n");				break;			}		}		r = (IMAGE_BASE_RELOCATION*)((char*)r + r->SizeOfBlock);	}}				/********************************************************************** *			PE_LoadImage * Load one PE format DLL/EXE into memory *  * Unluckily we can't just mmap the sections where we want them, for  * (at least) Linux does only support offsets which are page-aligned. * * BUT we have to map the whole image anyway, for Win32 programs sometimes * want to access them. (HMODULE32 point to the start of it) */HMODULE PE_LoadImage( int handle, LPCSTR filename, WORD *version ){    HMODULE	hModule;    HANDLE	mapping;    IMAGE_NT_HEADERS *nt;    IMAGE_SECTION_HEADER *pe_sec;    IMAGE_DATA_DIRECTORY *dir;//    BY_HANDLE_FILE_INFORMATION bhfi;    int	i, rawsize, lowest_va, vma_size, file_size = 0;    DWORD load_addr = 0, aoep, reloc = 0;//    struct get_read_fd_request *req = get_req_buffer();    int unix_handle = handle;    int page_size = getpagesize();    //    if ( GetFileInformationByHandle( hFile, &bhfi ) ) //    	file_size = bhfi.nFileSizeLow;     file_size=lseek(handle, 0, SEEK_END);    lseek(handle, 0, SEEK_SET);//#warning fix CreateFileMappingA    mapping = CreateFileMappingA( handle, NULL, PAGE_READONLY | SEC_COMMIT,                                    0, 0, NULL );    if (!mapping)    {        WARN("CreateFileMapping error %ld\n", GetLastError() );        return 0;    }//    hModule = (HMODULE)MapViewOfFile( mapping, FILE_MAP_READ, 0, 0, 0 );    hModule=(HMODULE)mapping;//    CloseHandle( mapping );    if (!hModule)    {        WARN("MapViewOfFile error %ld\n", GetLastError() );        return 0;    }    if ( *(WORD*)hModule !=IMAGE_DOS_SIGNATURE)    {        WARN("%s image doesn't have DOS signature, but 0x%04x\n", filename,*(WORD*)hModule);        goto error;    }    nt = PE_HEADER( hModule );        if ( nt->Signature != IMAGE_NT_SIGNATURE )    {        WARN("%s image doesn't have PE signature, but 0x%08lx\n", filename, nt->Signature );        goto error;    }        if ( nt->FileHeader.Machine != IMAGE_FILE_MACHINE_I386 )    {        MESSAGE("Trying to load PE image for unsupported architecture (");        switch (nt->FileHeader.Machine)        {        case IMAGE_FILE_MACHINE_UNKNOWN: MESSAGE("Unknown"); break;        case IMAGE_FILE_MACHINE_I860:    MESSAGE("I860"); break;        case IMAGE_FILE_MACHINE_R3000:   MESSAGE("R3000"); break;        case IMAGE_FILE_MACHINE_R4000:   MESSAGE("R4000"); break;        case IMAGE_FILE_MACHINE_R10000:  MESSAGE("R10000"); break;        case IMAGE_FILE_MACHINE_ALPHA:   MESSAGE("Alpha"); break;        case IMAGE_FILE_MACHINE_POWERPC: MESSAGE("PowerPC"); break;        default: MESSAGE("Unknown-%04x", nt->FileHeader.Machine); break;        }        MESSAGE(")\n");        goto error;    }        pe_sec = PE_SECTIONS( hModule );    rawsize = 0; lowest_va = 0x10000;    for (i = 0; i < nt->FileHeader.NumberOfSections; i++)     {        if (lowest_va > pe_sec[i].VirtualAddress)           lowest_va = pe_sec[i].VirtualAddress;    	if (pe_sec[i].Characteristics & IMAGE_SCN_CNT_UNINITIALIZED_DATA)	    continue;	if (pe_sec[i].PointerToRawData+pe_sec[i].SizeOfRawData > rawsize)	    rawsize = pe_sec[i].PointerToRawData+pe_sec[i].SizeOfRawData;    }         if ( file_size && file_size < rawsize )    {        ERR("PE module is too small (header: %d, filesize: %d), "                    "probably truncated download?\n",                     rawsize, file_size );        goto error;    }        aoep = nt->OptionalHeader.AddressOfEntryPoint;    if (aoep && (aoep < lowest_va))        FIXME("VIRUS WARNING: '%s' has an invalid entrypoint (0x%08lx) "                      "below the first virtual address (0x%08x) "                      "(possibly infected by Tchernobyl/SpaceFiller virus)!\n",                       filename, aoep, lowest_va );    /* FIXME:  Hack!  While we don't really support shared sections yet,     *         this checks for those special cases where the whole DLL     *         consists only of shared sections and is mapped into the     *         shared address space > 2GB.  In this case, we assume that     *         the module got mapped at its base address. Thus we simply     *         check whether the module has actually been mapped there     *         and use it, if so.  This is needed to get Win95 USER32.DLL     *         to work (until we support shared sections properly).     */    if ( nt->OptionalHeader.ImageBase & 0x80000000 &&        !strstr(filename, "xanlib.dll"))    {        HMODULE sharedMod = (HMODULE)nt->OptionalHeader.ImageBase;         IMAGE_NT_HEADERS *sharedNt = (PIMAGE_NT_HEADERS)               ( (LPBYTE)sharedMod + ((LPBYTE)nt - (LPBYTE)hModule) );        /* Well, this check is not really comprehensive,            but should be good enough for now ... */        if (    !IsBadReadPtr( (LPBYTE)sharedMod, sizeof(IMAGE_DOS_HEADER) )             && memcmp( (LPBYTE)sharedMod, (LPBYTE)hModule, sizeof(IMAGE_DOS_HEADER) ) == 0             && !IsBadReadPtr( sharedNt, sizeof(IMAGE_NT_HEADERS) )             && memcmp( sharedNt, nt, sizeof(IMAGE_NT_HEADERS) ) == 0 )        {            UnmapViewOfFile( (LPVOID)hModule );            return sharedMod;        }    }        load_addr = nt->OptionalHeader.ImageBase;    vma_size = calc_vma_size( hModule );    load_addr = (DWORD)VirtualAlloc( (void*)load_addr, vma_size,                                     MEM_RESERVE | MEM_COMMIT,                                     PAGE_EXECUTE_READWRITE );    if (load_addr == 0)     {                FIXME("We need to perform base relocations for %s\n", filename);	dir = nt->OptionalHeader.DataDirectory+IMAGE_DIRECTORY_ENTRY_BASERELOC;        if (dir->Size)            reloc = dir->VirtualAddress;        else         {            FIXME( "FATAL: Need to relocate %s, but no relocation records present (%s). Try to run that file directly !\n",                   filename,                   (nt->FileHeader.Characteristics&IMAGE_FILE_RELOCS_STRIPPED)?                   "stripped during link" : "unknown reason" );            goto error;        }        /* FIXME: If we need to relocate a system DLL (base > 2GB) we should         *        really make sure that the *new* base address is also > 2GB.         *        Some DLLs really check the MSB of the module handle :-/         */        if ( nt->OptionalHeader.ImageBase & 0x80000000 )            ERR( "Forced to relocate system DLL (base > 2GB). This is not good.\n" );        load_addr = (DWORD)VirtualAlloc( NULL, vma_size,					 MEM_RESERVE | MEM_COMMIT,					 PAGE_EXECUTE_READWRITE );	if (!load_addr) {            FIXME_(win32)(                   "FATAL: Couldn't load module %s (out of memory, %d needed)!\n", filename, vma_size);            goto error;	}    }    TRACE("Load addr is %lx (base %lx), range %x\n",          load_addr, nt->OptionalHeader.ImageBase, vma_size );    TRACE_(segment)("Loading %s at %lx, range %x\n",                    filename, load_addr, vma_size );#if 0        *(PIMAGE_DOS_HEADER)load_addr = *(PIMAGE_DOS_HEADER)hModule;    *PE_HEADER( load_addr ) = *nt;    memcpy( PE_SECTIONS(load_addr), PE_SECTIONS(hModule),            sizeof(IMAGE_SECTION_HEADER) * nt->FileHeader.NumberOfSections );        memcpy( load_addr, hModule, lowest_fa );#endif    if ((void*)FILE_dommap( handle, (void *)load_addr, 0, nt->OptionalHeader.SizeOfHeaders,                     0, 0, PROT_EXEC | PROT_WRITE | PROT_READ,                     MAP_PRIVATE | MAP_FIXED ) != (void*)load_addr)    {        ERR_(win32)( "Critical Error: failed to map PE header to necessary address.\n");	        goto error;    }        pe_sec = PE_SECTIONS( hModule );    for (i = 0; i < nt->FileHeader.NumberOfSections; i++, pe_sec++)    {        if (!pe_sec->SizeOfRawData || !pe_sec->PointerToRawData) continue;        TRACE("%s: mmaping section %s at %p off %lx size %lx/%lx\n",              filename, pe_sec->Name, (void*)RVA(pe_sec->VirtualAddress),              pe_sec->PointerToRawData, pe_sec->SizeOfRawData, pe_sec->Misc.VirtualSize );        if ((void*)FILE_dommap( unix_handle, (void*)RVA(pe_sec->VirtualAddress),                         0, pe_sec->SizeOfRawData, 0, pe_sec->PointerToRawData,                         PROT_EXEC | PROT_WRITE | PROT_READ,                         MAP_PRIVATE | MAP_FIXED ) != (void*)RVA(pe_sec->VirtualAddress))        {                        ERR_(win32)( "Critical Error: failed to map PE section to necessary address.\n");            goto error;        }

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -