-
-
Notifications
You must be signed in to change notification settings - Fork 374
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
1. Everything is based off of Microsoft's PE format documentation [found here](https://learn.microsoft.com/en-us/windows/win32/debug/pe-format) 2. Up until now, imports were used for relocations in the PE & MDMP format, which is wrong. PE uses base relocations. 3. PE relocations do not have symbols associated with them, and thus have no name. In order to still have flags in relocs flagspace, dummy symbols were allocated with generic names composed of the relocation's virtual address space. * Remove old code of setting import entries to PE relocations * Parse PE base relocations * Implement converting from the specific RzBinPeRelocEnt type to the general RzBinReloc type * Implement corrent type naming of PE relocations * Add missing `PE_IMAGE_FILE_MACHINE` types * Add PE base relocation types * Fix PE format 'has_canary' function * Remove 'RzBinPEObj.endian' and use 'RzBinPEObj.big_endian' instead * Removed a.exe and b.exe tests, as they don't contain base relocations and swapped them with correct tests * Corrected old PE relocation tests * Add new PE base relocation tests
- Loading branch information
Showing
16 changed files
with
1,658 additions
and
324 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
// SPDX-FileCopyrightText: 2024 Roee Toledano <[email protected]> | ||
// SPDX-License-Identifier: LGPL-3.0-only | ||
|
||
#define RZ_BIN_PE64 1 | ||
#include "pe_relocs.c" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
// SPDX-FileCopyrightText: 2024 Roee Toledano <[email protected]> | ||
// SPDX-License-Identifier: LGPL-3.0-only | ||
|
||
#include "pe.h" | ||
|
||
bool PE_(bin_pe_has_base_relocs)(RZ_NONNULL RzBinPEObj *bin) { | ||
rz_return_val_if_fail(bin, false); | ||
|
||
return bin->relocs && (rz_vector_len(bin->relocs) > 0); | ||
} | ||
|
||
static bool read_reloc_ent_from_block(RZ_NONNULL RzVector /*<RzBinPeRelocEnt>*/ *relocs, | ||
RzBuffer *b, RzBinPeRelocBlock *block, ut64 *offset, const int big_endian) { | ||
const ut32 reloc_block_end = *offset + block->block_size - 8; // block size includes the size of the next blocks entry, which is 8 bytes long | ||
do { | ||
RzBinPeRelocEnt reloc = { 0 }; | ||
if (!rz_buf_read_ble16_offset(b, offset, &reloc.raw_val, big_endian)) { | ||
return false; | ||
} | ||
reloc.page_rva = block->page_rva; | ||
|
||
rz_vector_push(relocs, &reloc); | ||
|
||
} while (*offset < reloc_block_end); | ||
|
||
return true; | ||
} | ||
|
||
static bool get_relocs_from_data_dir(RZ_NONNULL RzBinPEObj *bin, RZ_BORROW RZ_NONNULL RzVector /*<RzBinPeRelocEnt>*/ *relocs) { | ||
rz_return_val_if_fail(bin->b && bin->nt_headers && bin->optional_header, false); | ||
RzBuffer *b = bin->b; | ||
const st64 o_addr = rz_buf_tell(b); | ||
|
||
// get offset in file of first reloc block | ||
ut64 offset = PE_(bin_pe_rva_to_paddr)(bin, bin->optional_header->DataDirectory[PE_IMAGE_DIRECTORY_ENTRY_BASERELOC].VirtualAddress); | ||
const ut64 relocs_end_offset = offset + bin->optional_header->DataDirectory[PE_IMAGE_DIRECTORY_ENTRY_BASERELOC].Size; | ||
|
||
do { | ||
RzBinPeRelocBlock block = { 0 }; | ||
if (!rz_buf_read_ble32_offset(b, &offset, &block.page_rva, bin->big_endian) || | ||
!rz_buf_read_ble32_offset(b, &offset, &block.block_size, bin->big_endian) || | ||
!read_reloc_ent_from_block(relocs, b, &block, &offset, bin->big_endian)) { | ||
return false; | ||
} | ||
|
||
} while (offset < relocs_end_offset); | ||
|
||
rz_buf_seek(b, o_addr, RZ_BUF_SET); | ||
|
||
return true; | ||
} | ||
|
||
int PE_(bin_pe_init_relocs)(RZ_NONNULL RzBinPEObj *bin) { | ||
rz_return_val_if_fail(bin, false); | ||
|
||
RzVector *ret = rz_vector_new(sizeof(RzBinPeRelocEnt), NULL, NULL); | ||
if (!ret) { | ||
bin->relocs = NULL; | ||
return false; | ||
} | ||
|
||
if (PE_(rz_bin_pe_is_stripped_relocs)(bin) || !get_relocs_from_data_dir(bin, ret)) { | ||
rz_vector_free(bin->relocs); | ||
bin->relocs = NULL; | ||
return false; | ||
} | ||
|
||
bin->relocs = ret; | ||
|
||
return true; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.