diff options
Diffstat (limited to 'src/cmds')
-rw-r--r-- | src/cmds/pe_cmd_ar.c | 134 | ||||
-rw-r--r-- | src/cmds/pe_cmd_perk.c | 97 |
2 files changed, 231 insertions, 0 deletions
diff --git a/src/cmds/pe_cmd_ar.c b/src/cmds/pe_cmd_ar.c new file mode 100644 index 0000000..97e94c7 --- /dev/null +++ b/src/cmds/pe_cmd_ar.c @@ -0,0 +1,134 @@ +/***************************************************************/ +/* perk: PE Resource Kit */ +/* Copyright (C) 2015--2025 SysDeer Technologies, LLC */ +/* Released under GPLv2 and GPLv3; see COPYING.PERK. */ +/***************************************************************/ + +#include <perk/perk.h> +#include "perk_driver_impl.h" +#include "perk_errinfo_impl.h" +#include "perk_ar_impl.h" + +int pe_ar_list_members(const struct pe_archive_meta *, const char **); + +static int pe_cmd_ar_perform_unit_actions( + const struct pe_driver_ctx * dctx, + const char * arname, + const char ** members, + uint64_t action) +{ + struct pe_unit_ctx * arctx = 0; + int (*pe_ar_fn)(const struct pe_archive_meta *,const char **); + + if (action == PERK_DRIVER_AR_LIST_MEMBERS) { + pe_ar_fn = pe_ar_list_members; + + } else if (action == PERK_DRIVER_AR_PRINT_MEMBERS) { + pe_ar_fn = pe_ar_print_members; + + } else { + return 0; + } + + if (pe_lib_get_unit_ctx(dctx,arname,&arctx) < 0) + return PERK_NESTED_ERROR(dctx); + + if (arctx->armeta == 0) + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_NON_ARCHIVE_IMAGE); + + if (pe_ar_fn(arctx->armeta,members) < 0) { + pe_lib_free_unit_ctx(arctx); + return PERK_NESTED_ERROR(dctx); + } + + pe_lib_free_unit_ctx(arctx); + + return 0; +} + +static int pe_cmd_ar_verify_cmdline( + const struct pe_driver_ctx * dctx, + uint64_t flags, + const char * posname, + const char * arname, + const char ** members) +{ + uint64_t action; + uint64_t poscmd; + uint64_t vercmd; + + action = (flags & AR_ACTION_MASK); + poscmd = (flags & AR_POSNAME_MASK); + vercmd = (flags & PERK_DRIVER_VERSION); + + if (vercmd && !posname && !arname && !members) + return 0; + + switch (action) { + case 0: + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_MISSING_ACTION); + + case PERK_DRIVER_AR_LIST_MEMBERS: + case PERK_DRIVER_AR_DELETE_MEMBERS: + case PERK_DRIVER_AR_APPEND_MEMBERS: + case PERK_DRIVER_AR_EXTRACT_MEMBERS: + case PERK_DRIVER_AR_PRINT_MEMBERS: + if (poscmd || posname) + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_INVALID_ANCHORS); + + break; + + case AR_UPDATE_MASK: + case PERK_DRIVER_AR_REPLACE_MEMBERS: + case PERK_DRIVER_AR_MOVE_MEMBERS: + switch (poscmd) { + case 0: + if (posname) + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_MISSING_ANCHOR); + break; + + case PERK_DRIVER_AR_POSITION_AFTER: + case PERK_DRIVER_AR_POSITION_BEFORE: + if (!posname) + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_NULL_POSNAME); + break; + + default: + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_MULTIPLE_ANCHORS); + } + + default: + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_MULTIPLE_ACTIONS); + } + + if (!arname) + return PERK_CUSTOM_ERROR(dctx, + PERK_ERR_AR_NULL_ARNAME); + + return 0; +} + +int pe_cmd_ar( + const struct pe_driver_ctx * dctx, + uint64_t flags, + const char * posname, + const char * arname, + const char ** members) +{ + uint64_t action = (flags & AR_ACTION_MASK); + + if (pe_cmd_ar_verify_cmdline(dctx,flags,posname,arname,members) < 0) + return PERK_NESTED_ERROR(dctx); + + if (pe_cmd_ar_perform_unit_actions(dctx,arname,members,action) < 0) + return PERK_NESTED_ERROR(dctx); + + return 0; +} diff --git a/src/cmds/pe_cmd_perk.c b/src/cmds/pe_cmd_perk.c new file mode 100644 index 0000000..6fe6112 --- /dev/null +++ b/src/cmds/pe_cmd_perk.c @@ -0,0 +1,97 @@ +/***************************************************************/ +/* perk: PE Resource Kit */ +/* Copyright (C) 2015--2025 SysDeer Technologies, LLC */ +/* Released under GPLv2 and GPLv3; see COPYING.PERK. */ +/***************************************************************/ + +#include <stdio.h> +#include <unistd.h> +#include <perk/perk.h> +#include <perk/perk_output.h> +#include "perk_driver_impl.h" +#include "perk_dprintf_impl.h" + +static void pe_perform_hdrdump_actions( + const struct pe_driver_ctx * dctx, + const struct pe_image_meta * meta) +{ + if (dctx->cctx->hdrdump & PERK_HDRDUMP_IMAGE_DOS_HEADER) + pe_hexdump_image_dos_hdr(dctx,meta); + + if (dctx->cctx->hdrdump & PERK_HDRDUMP_COFF_IMAGE_HEADER) + pe_hexdump_coff_image_hdr(dctx,meta); + + if (dctx->cctx->hdrdump & PERK_HDRDUMP_COFF_OBJECT_HEADER) + pe_hexdump_coff_object_hdr(dctx,meta); + + if (dctx->cctx->hdrdump & PERK_HDRDUMP_COFF_OPT_HEADER) + pe_hexdump_coff_opt_hdr(dctx,meta); + + if (dctx->cctx->hdrdump & PERK_HDRDUMP_SECTION_TABLE) + pe_hexdump_sec_tbl(dctx,meta); + + if (dctx->cctx->hdrdump & PERK_HDRDUMP_EXPORT_HEADER) + pe_hexdump_export_hdr(dctx,meta); + + if (dctx->cctx->hdrdump & PERK_HDRDUMP_IMPORT_TABLE) + pe_hexdump_import_tbl(dctx,meta); +} + +static void pe_perform_unit_actions_impl( + const struct pe_driver_ctx * dctx, + const struct pe_image_meta * meta) +{ + uint64_t flags = dctx->cctx->fmtflags; + + if (flags & PERK_OUTPUT_IMAGE_CATEGORY) + pe_output_image_category(dctx,meta); + + if (flags & PERK_OUTPUT_IMAGE_SECTIONS) + pe_output_image_sections(dctx,meta); + + if (flags & PERK_OUTPUT_IMAGE_SYMBOLS) + pe_output_image_symbols(dctx,meta); + + if (flags & PERK_OUTPUT_IMAGE_STRINGS) + pe_output_image_strings(dctx,meta); + + if (flags & PERK_OUTPUT_EXPORT_SYMS) + pe_output_export_symbols(dctx,meta); + + if ((flags & PERK_OUTPUT_IMPORT_LIBS) || (flags & PERK_OUTPUT_IMPORT_SYMS)) + pe_output_import_libraries(dctx,meta); + + if ((flags & PERK_OUTPUT_MDSO_LIBS) || (flags & PERK_OUTPUT_MDSO_SYMS)) + pe_output_mdso_libraries(dctx,meta); + + if (dctx->cctx->hdrdump) + pe_perform_hdrdump_actions(dctx,meta); +} + +static void pe_perform_unit_actions( + const struct pe_driver_ctx * dctx, + const struct pe_unit_ctx * uctx) +{ + const struct pe_image_meta ** objmeta; + + if (uctx->meta) + pe_perform_unit_actions_impl(dctx,uctx->meta); + + if ((objmeta = uctx->objmeta)) + for (; *objmeta; ) + pe_perform_unit_actions_impl(dctx,*objmeta++); +} + +int pe_cmd_perk(const struct pe_driver_ctx * dctx, const char * path) +{ + int ret; + struct pe_unit_ctx * uctx; + + if ((ret = pe_lib_get_unit_ctx(dctx,path,&uctx)) < 0) + return ret; + + pe_perform_unit_actions(dctx,uctx); + pe_lib_free_unit_ctx(uctx); + + return 0; +} |