summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authormidipix <writeonce@midipix.org>2025-06-18 04:26:49 +0000
committermidipix <writeonce@midipix.org>2025-06-18 04:27:32 +0000
commitc5998808867bf8209722357c745508c592d337cf (patch)
treea1bb6922ea4df9e236bb095679bccda71cf74db6
parent2b1ceecd7c7440937ea975c379ddf1ca5d3b86a3 (diff)
downloadperk-c5998808867bf8209722357c745508c592d337cf.tar.bz2
perk-c5998808867bf8209722357c745508c592d337cf.tar.xz
output: pe_output_base_relocs(): initial implementation.
-rw-r--r--include/perk/perk.h1
-rw-r--r--project/common.mk1
-rw-r--r--src/output/pe_output_base_relocs.c149
3 files changed, 151 insertions, 0 deletions
diff --git a/include/perk/perk.h b/include/perk/perk.h
index 912387b..95f3b53 100644
--- a/include/perk/perk.h
+++ b/include/perk/perk.h
@@ -305,6 +305,7 @@ perk_api int pe_output_pecoff_strings (const struct pe_driver_ctx *, c
perk_api int pe_output_export_symbols (const struct pe_driver_ctx *, const struct pe_image_meta *);
perk_api int pe_output_import_libraries (const struct pe_driver_ctx *, const struct pe_image_meta *);
perk_api int pe_output_mdso_libraries (const struct pe_driver_ctx *, const struct pe_image_meta *);
+perk_api int pe_output_base_relocs (const struct pe_driver_ctx *, const struct pe_image_meta *);
/* error trace api */
perk_api int pe_output_error_record (const struct pe_driver_ctx *, const struct pe_error_info *);
diff --git a/project/common.mk b/project/common.mk
index 8bf1b0c..ba389c6 100644
--- a/project/common.mk
+++ b/project/common.mk
@@ -27,6 +27,7 @@ API_SRCS = \
src/logic/pe_image_meta_data.c \
src/logic/pe_image_raw_data.c \
src/output/pe_output_error.c \
+ src/output/pe_output_base_relocs.c \
src/output/pe_output_export_symbols.c \
src/output/pe_output_pecoff_category.c \
src/output/pe_output_pecoff_sections.c \
diff --git a/src/output/pe_output_base_relocs.c b/src/output/pe_output_base_relocs.c
new file mode 100644
index 0000000..ea398af
--- /dev/null
+++ b/src/output/pe_output_base_relocs.c
@@ -0,0 +1,149 @@
+/***************************************************************/
+/* perk: PE Resource Kit */
+/* Copyright (C) 2015--2025 SysDeer Technologies, LLC */
+/* Released under GPLv2 and GPLv3; see COPYING.PERK. */
+/***************************************************************/
+
+#include <stdio.h>
+
+#include <perk/perk.h>
+#include <perk/perk_output.h>
+#include <perk/perk_consts.h>
+#include <perk/perk_structs.h>
+#include "perk_driver_impl.h"
+#include "perk_endian_impl.h"
+#include "perk_reader_impl.h"
+#include "perk_dprintf_impl.h"
+#include "perk_errinfo_impl.h"
+
+static const char * pe_base_reloc_type_desc[0x10] = {
+ [PE_IMAGE_REL_BASED_ABSOLUTE] = "ABSOLUTE",
+ [PE_IMAGE_REL_BASED_HIGH] = "HIGH",
+ [PE_IMAGE_REL_BASED_LOW] = "LOW",
+ [PE_IMAGE_REL_BASED_HIGHLOW] = "HIGHLOW",
+ [PE_IMAGE_REL_BASED_HIGHADJ] = "HIGHADJ",
+ [PE_IMAGE_REL_BASED_RESERVED] = "RESERVED",
+ [PE_IMAGE_REL_BASED_DIR64] = "DIR64",
+};
+
+static int pe_output_base_reloc_records(
+ const struct pe_driver_ctx * dctx,
+ const struct pe_image_meta * meta,
+ int fdout)
+{
+ struct pe_raw_base_reloc_blk * r;
+ int i;
+ unsigned j;
+ uint32_t rva;
+ uint32_t size;
+ uint32_t nents;
+ uint16_t fixup;
+ uint32_t offset;
+ const char * desc;
+
+ for (i=0; i<meta->m_stats.t_nrelblks; i++) {
+ r = meta->r_reltbl[i];
+
+ rva = pe_read_long(r->blk_rva);
+ size = pe_read_long(r->blk_size);
+
+ nents = size - offsetof(struct pe_raw_base_reloc_blk,blk_data);
+ nents /= sizeof(uint16_t);
+
+ if (pe_dprintf(fdout,
+ "Virtual Address: %08x, "
+ "Chunk Size: %u (0x%x), "
+ "Number of Fixups: %u\n",
+ rva,size,size,nents) < 0)
+ return PERK_FILE_ERROR(dctx);
+
+ for (j=0; j<nents; j++) {
+ fixup = pe_read_short(r->blk_data[j]);
+ desc = pe_base_reloc_type_desc[fixup >> 12];
+ offset = fixup & 0xfff;
+
+ if (pe_dprintf(fdout,
+ "\t"
+ "reloc %4d "
+ "offset %4x [%x] %s\n",
+ j,offset,offset+rva,
+ desc) < 0)
+ return PERK_FILE_ERROR(dctx);
+ }
+
+ if (pe_dprintf(fdout,"\n") < 0)
+ return PERK_FILE_ERROR(dctx);
+ }
+
+ return 0;
+}
+
+static int pe_output_base_relocs_yaml(
+ const struct pe_driver_ctx * dctx,
+ const struct pe_image_meta * meta,
+ int fdout)
+{
+ struct pe_raw_base_reloc_blk * r;
+ int i;
+ unsigned j;
+ uint32_t rva;
+ uint32_t size;
+ uint32_t nents;
+ uint16_t fixup;
+ uint32_t offset;
+ const char * desc;
+
+ if (pe_dprintf(fdout," - Base relocation blocks:\n") < 0)
+ return PERK_FILE_ERROR(dctx);
+
+ for (i=0; i<meta->m_stats.t_nrelblks; i++) {
+ r = meta->r_reltbl[i];
+
+ rva = pe_read_long(r->blk_rva);
+ size = pe_read_long(r->blk_size);
+
+ nents = size - offsetof(struct pe_raw_base_reloc_blk,blk_data);
+ nents /= sizeof(uint16_t);
+
+ if (pe_dprintf(fdout,
+ " - base relocation block:\n"
+ " [ page-rva: 0x%08x ]\n"
+ " [ block-size: 0x%08x ]\n"
+ " [ fixup-entries: %u ]\n\n"
+ " - page fixups:\n",
+ rva,size,nents) < 0)
+ return PERK_FILE_ERROR(dctx);
+
+ for (j=0; j<nents; j++) {
+ fixup = pe_read_short(r->blk_data[j]);
+ desc = pe_base_reloc_type_desc[fixup >> 12];
+ offset = fixup & 0xfff;
+
+ if (pe_dprintf(fdout,
+ " [ type = %s ]\n"
+ " [ offset = 0x%03x ]\n\n",
+ desc,offset) < 0)
+ return PERK_FILE_ERROR(dctx);
+ }
+ }
+
+ return 0;
+}
+
+int pe_output_base_relocs(
+ const struct pe_driver_ctx * dctx,
+ const struct pe_image_meta * meta)
+{
+ int fdout = pe_driver_fdout(dctx);
+
+ if (dctx->cctx->fmtflags & PERK_PRETTY_YAML) {
+ if (pe_output_base_relocs_yaml(dctx,meta,fdout) < 0)
+ return PERK_NESTED_ERROR(dctx);
+
+ } else {
+ if (pe_output_base_reloc_records(dctx,meta,fdout) < 0)
+ return PERK_NESTED_ERROR(dctx);
+ }
+
+ return 0;
+}