aboutsummaryrefslogtreecommitdiff
path: root/frida_mode/src/lib
diff options
context:
space:
mode:
authorvan Hauser <vh@thc.org>2021-05-10 13:57:47 +0200
committerGitHub <noreply@github.com>2021-05-10 13:57:47 +0200
commit8b7a7b29c60f11cdf6226b3e418e87a5c3f5caac (patch)
tree6ce9d90644f161d21d802e9cbe48eb38467684e9 /frida_mode/src/lib
parentd0225c2c4d465968660a08c93857fed354e539b1 (diff)
downloadafl++-8b7a7b29c60f11cdf6226b3e418e87a5c3f5caac.tar.gz
Push to stable (#895)
* sync (#886) * Create FUNDING.yml * Update FUNDING.yml * moved custom_mutator examples * unicorn speedtest makefile cleanup * fixed example location * fix qdbi * update util readme * Frida persistent (#880) * Added x64 support for persistent mode (function call only), in-memory teest cases and complog * Review changes, fix NeverZero and code to parse the .text section of the main executable. Excluded ranges TBC * Various minor fixes and finished support for AFL_INST_LIBS * Review changes Co-authored-by: Your Name <you@example.com> * nits * fix frida mode * Integer overflow/underflow fixes in libdislocator (#889) * libdislocator: fixing integer overflow in 'max_mem' variable and setting 'max_mem' type to 'size_t' * libdislocator: fixing potential integer underflow in 'total_mem' variable due to its different values in different threads * Bumped warnings up to the max and fixed remaining issues (#890) Co-authored-by: Your Name <you@example.com> * nits * frida mode - support non-pie * nits * nit * update grammar mutator * Fixes for aarch64, OSX and other minor issues (#891) Co-authored-by: Your Name <you@example.com> * nits * nits * fix PCGUARD, build aflpp_driver with fPIC * Added representative fuzzbench test and test for libxml (#893) * Added representative fuzzbench test and test for libxml * Added support for building FRIDA from source with FRIDA_SOURCE=1 Co-authored-by: Your Name <you@example.com> * nits * update changelog * typos * fixed potential double free in custom trim (#881) * error handling, freeing mem * frida: complog -> cmplog * fix statsd writing * let aflpp_qemu_driver_hook.so build fail gracefully * fix stdin trimming * Support for AFL_ENTRYPOINT (#898) Co-authored-by: Your Name <you@example.com> * remove the input file .cur_input at the end of the fuzzing, if AFL_TMPDIR is used * reverse push (#901) * Create FUNDING.yml * Update FUNDING.yml * disable QEMU static pie Co-authored-by: Andrea Fioraldi <andreafioraldi@gmail.com> * clarify that no modifications are required. * add new test for frida_mode (please review) * typos * fix persistent mode (64-bit) * set ARCH for linux intel 32-bit for frida-gum-devkit * prepare for 32-bit support (later) * not on qemu 3 anymore * unicorn mips fixes * instrumentation further move to C++11 (#900) * unicorn fixes * more unicorn fixes * Fix memory errors when trim causes testcase growth (#881) (#903) * Revert "fixed potential double free in custom trim (#881)" This reverts commit e9d2f72382cab75832721d859c3e731da071435d. * Revert "fix custom trim for increasing data" This reverts commit 86a8ef168dda766d2f25f15c15c4d3ecf21d0667. * Fix memory errors when trim causes testcase growth Modify trim_case_custom to avoid writing into in_buf because some custom mutators can cause the testcase to grow rather than shrink. Instead of modifying in_buf directly, we write the update out to the disk when trimming is complete, and then the caller is responsible for refreshing the in-memory buffer from the file. This is still a bit sketchy because it does need to modify q->len in order to notify the upper layers that something changed, and it could end up telling upper layer code that the q->len is *bigger* than the buffer (q->testcase_buf) that contains it, which is asking for trouble down the line somewhere... * Fix an unlikely situation Put back some `unlikely()` calls that were in the e9d2f72382cab75832721d859c3e731da071435d commit that was reverted. * typo * Exit on time (#904) * Variable AFL_EXIT_ON_TIME description has been added. Variables AFL_EXIT_ON_TIME and afl_exit_on_time has been added. afl->exit_on_time variable initialization has been added. The asignment of a value to the afl->afl_env.afl_exit_on_time variable from environment variables has been added. Code to exit on timeout if new path not found has been added. * Type of afl_exit_on_time variable has been changed. Variable exit_on_time has been added to the afl_state_t structure. * Command `export AFL_EXIT_WHEN_DONE=1` has been added. * Millisecond to second conversion has been added. Call get_cur_time() has been added. * Revert to using the saved current time value. * Useless check has been removed. * fix new path to custom-mutators * ensure crashes/README.txt exists * fix * Changes to bump FRIDA version and to clone FRIDA repo in to build directory rather than use a submodule as the FRIDA build scripts don't like it (#906) Co-authored-by: Your Name <you@example.com> * Fix numeric overflow in cmplog implementation (#907) Co-authored-by: Your Name <you@example.com> * testcase fixes for unicorn * remove merge conflict artifacts * fix afl-plot * Changes to remove binaries from frida_mode (#913) Co-authored-by: Your Name <you@example.com> * Frida cmplog fail fast (#914) * Changes to remove binaries from frida_mode * Changes to make cmplog fail fast Co-authored-by: Your Name <you@example.com> * afl-plot: relative time * arch linux and mac os support for afl-system-config * typo * code-format * update documentation Co-authored-by: Dominik Maier <domenukk@gmail.com> Co-authored-by: WorksButNotTested <62701594+WorksButNotTested@users.noreply.github.com> Co-authored-by: Your Name <you@example.com> Co-authored-by: Dmitry Zheregelya <zheregelya.d@gmail.com> Co-authored-by: hexcoder <hexcoder-@users.noreply.github.com> Co-authored-by: hexcoder- <heiko@hexco.de> Co-authored-by: Andrea Fioraldi <andreafioraldi@gmail.com> Co-authored-by: David CARLIER <devnexen@gmail.com> Co-authored-by: realmadsci <71108352+realmadsci@users.noreply.github.com> Co-authored-by: Roman M. Iudichev <SecNotice@ya.ru>
Diffstat (limited to 'frida_mode/src/lib')
-rw-r--r--frida_mode/src/lib/lib.c176
-rw-r--r--frida_mode/src/lib/lib_apple.c82
2 files changed, 258 insertions, 0 deletions
diff --git a/frida_mode/src/lib/lib.c b/frida_mode/src/lib/lib.c
new file mode 100644
index 00000000..c5045533
--- /dev/null
+++ b/frida_mode/src/lib/lib.c
@@ -0,0 +1,176 @@
+#ifndef __APPLE__
+ #include <elf.h>
+ #include <fcntl.h>
+ #include <limits.h>
+ #include <stdio.h>
+ #include <sys/mman.h>
+ #include <unistd.h>
+
+ #include "frida-gum.h"
+
+ #include "debug.h"
+
+ #include "lib.h"
+
+ #if defined(__arm__) || defined(__i386__)
+ #define ELFCLASS ELFCLASS32
+typedef Elf32_Ehdr Elf_Ehdr;
+typedef Elf32_Phdr Elf_Phdr;
+typedef Elf32_Shdr Elf_Shdr;
+typedef Elf32_Addr Elf_Addr;
+ #elif defined(__aarch64__) || defined(__x86_64__)
+ #define ELFCLASS ELFCLASS64
+typedef Elf64_Ehdr Elf_Ehdr;
+typedef Elf64_Phdr Elf_Phdr;
+typedef Elf64_Shdr Elf_Shdr;
+typedef Elf64_Addr Elf_Addr;
+ #else
+ #error "Unsupported platform"
+ #endif
+
+typedef struct {
+
+ gchar name[PATH_MAX + 1];
+ gchar path[PATH_MAX + 1];
+ GumAddress base_address;
+ gsize size;
+
+} lib_details_t;
+
+static guint64 text_base = 0;
+static guint64 text_limit = 0;
+
+static gboolean lib_find_exe(const GumModuleDetails *details,
+ gpointer user_data) {
+
+ lib_details_t *lib_details = (lib_details_t *)user_data;
+
+ memcpy(lib_details->name, details->name, PATH_MAX);
+ memcpy(lib_details->path, details->path, PATH_MAX);
+ lib_details->base_address = details->range->base_address;
+ lib_details->size = details->range->size;
+ return FALSE;
+
+}
+
+static void lib_validate_hdr(Elf_Ehdr *hdr) {
+
+ if (hdr->e_ident[0] != ELFMAG0) FATAL("Invalid e_ident[0]");
+ if (hdr->e_ident[1] != ELFMAG1) FATAL("Invalid e_ident[1]");
+ if (hdr->e_ident[2] != ELFMAG2) FATAL("Invalid e_ident[2]");
+ if (hdr->e_ident[3] != ELFMAG3) FATAL("Invalid e_ident[3]");
+ if (hdr->e_ident[4] != ELFCLASS) FATAL("Invalid class");
+
+}
+
+static void lib_read_text_section(lib_details_t *lib_details, Elf_Ehdr *hdr) {
+
+ Elf_Phdr *phdr;
+ gboolean found_preferred_base = FALSE;
+ Elf_Addr preferred_base;
+ Elf_Shdr *shdr;
+ Elf_Shdr *shstrtab;
+ char * shstr;
+ char * section_name;
+ Elf_Shdr *curr;
+ char text_name[] = ".text";
+
+ phdr = (Elf_Phdr *)((char *)hdr + hdr->e_phoff);
+ for (size_t i = 0; i < hdr->e_phnum; i++) {
+
+ if (phdr[i].p_type == PT_LOAD) {
+
+ preferred_base = phdr[i].p_vaddr;
+ found_preferred_base = TRUE;
+ break;
+
+ }
+
+ }
+
+ if (!found_preferred_base) { FATAL("Failed to find preferred load address"); }
+
+ OKF("Image preferred load address 0x%016lx", preferred_base);
+
+ shdr = (Elf_Shdr *)((char *)hdr + hdr->e_shoff);
+ shstrtab = &shdr[hdr->e_shstrndx];
+ shstr = (char *)hdr + shstrtab->sh_offset;
+
+ OKF("shdr: %p", shdr);
+ OKF("shstrtab: %p", shstrtab);
+ OKF("shstr: %p", shstr);
+
+ for (size_t i = 0; i < hdr->e_shnum; i++) {
+
+ curr = &shdr[i];
+
+ if (curr->sh_name == 0) continue;
+
+ section_name = &shstr[curr->sh_name];
+ OKF("Section: %2lu - base: 0x%016lX size: 0x%016lX %s", i, curr->sh_addr,
+ curr->sh_size, section_name);
+ if (memcmp(section_name, text_name, sizeof(text_name)) == 0 &&
+ text_base == 0) {
+
+ text_base = lib_details->base_address + curr->sh_addr - preferred_base;
+ text_limit = text_base + curr->sh_size;
+ OKF("> text_addr: 0x%016lX", text_base);
+ OKF("> text_limit: 0x%016lX", text_limit);
+
+ }
+
+ }
+
+}
+
+static void lib_get_text_section(lib_details_t *details) {
+
+ int fd = -1;
+ off_t len;
+ Elf_Ehdr *hdr;
+
+ fd = open(details->path, O_RDONLY);
+ if (fd < 0) { FATAL("Failed to open %s", details->path); }
+
+ len = lseek(fd, 0, SEEK_END);
+
+ if (len == (off_t)-1) { FATAL("Failed to lseek %s", details->path); }
+
+ OKF("len: %ld", len);
+
+ hdr = (Elf_Ehdr *)mmap(NULL, len, PROT_READ, MAP_PRIVATE, fd, 0);
+ if (hdr == MAP_FAILED) { FATAL("Failed to map %s", details->path); }
+
+ lib_validate_hdr(hdr);
+ lib_read_text_section(details, hdr);
+
+ munmap(hdr, len);
+ close(fd);
+
+}
+
+void lib_init(void) {
+
+ lib_details_t lib_details;
+ gum_process_enumerate_modules(lib_find_exe, &lib_details);
+ OKF("Executable: 0x%016lx - %s", lib_details.base_address, lib_details.path);
+ lib_get_text_section(&lib_details);
+
+}
+
+guint64 lib_get_text_base(void) {
+
+ if (text_base == 0) FATAL("Lib not initialized");
+ return text_base;
+
+}
+
+guint64 lib_get_text_limit(void) {
+
+ if (text_limit == 0) FATAL("Lib not initialized");
+ return text_limit;
+
+}
+
+#endif
+
diff --git a/frida_mode/src/lib/lib_apple.c b/frida_mode/src/lib/lib_apple.c
new file mode 100644
index 00000000..8f863861
--- /dev/null
+++ b/frida_mode/src/lib/lib_apple.c
@@ -0,0 +1,82 @@
+#ifdef __APPLE__
+ #include "frida-gum.h"
+
+ #include "debug.h"
+
+ #include "lib.h"
+ #include "util.h"
+
+extern mach_port_t mach_task_self();
+extern void gum_darwin_enumerate_modules(mach_port_t task,
+ GumFoundModuleFunc func,
+ gpointer user_data);
+
+static guint64 text_base = 0;
+static guint64 text_limit = 0;
+
+static gboolean lib_get_main_module(const GumModuleDetails *details,
+ gpointer user_data) {
+
+ GumDarwinModule **ret = (GumDarwinModule **)user_data;
+ GumDarwinModule * module = gum_darwin_module_new_from_memory(
+ details->path, mach_task_self(), details->range->base_address,
+ GUM_DARWIN_MODULE_FLAGS_NONE, NULL);
+
+ OKF("Found main module: %s", module->name);
+
+ *ret = module;
+
+ return FALSE;
+
+}
+
+gboolean lib_get_text_section(const GumDarwinSectionDetails *details,
+ gpointer user_data) {
+
+ UNUSED_PARAMETER(user_data);
+ static size_t idx = 0;
+ char text_name[] = "__text";
+
+ OKF("Section: %2lu - base: 0x%016" G_GINT64_MODIFIER
+ "X size: 0x%016" G_GINT64_MODIFIER "X %s",
+ idx++, details->vm_address, details->vm_address + details->size,
+ details->section_name);
+
+ if (memcmp(details->section_name, text_name, sizeof(text_name)) == 0 &&
+ text_base == 0) {
+
+ text_base = details->vm_address;
+ text_limit = details->vm_address + details->size;
+ OKF("> text_addr: 0x%016" G_GINT64_MODIFIER "X", text_base);
+ OKF("> text_limit: 0x%016" G_GINT64_MODIFIER "X", text_limit);
+
+ }
+
+ return TRUE;
+
+}
+
+void lib_init(void) {
+
+ GumDarwinModule *module = NULL;
+ gum_darwin_enumerate_modules(mach_task_self(), lib_get_main_module, &module);
+ gum_darwin_module_enumerate_sections(module, lib_get_text_section, NULL);
+
+}
+
+guint64 lib_get_text_base(void) {
+
+ if (text_base == 0) FATAL("Lib not initialized");
+ return text_base;
+
+}
+
+guint64 lib_get_text_limit(void) {
+
+ if (text_limit == 0) FATAL("Lib not initialized");
+ return text_limit;
+
+}
+
+#endif
+