aboutsummaryrefslogtreecommitdiff
path: root/frida_mode/src
diff options
context:
space:
mode:
authorvan Hauser <vh@thc.org>2021-12-09 11:55:36 +0100
committerGitHub <noreply@github.com>2021-12-09 11:55:36 +0100
commit08ca4d54a55fe73e64a994c41a12af61f52e497e (patch)
tree2d0f060cf98afbe80f5bf810fd6b167a5152be81 /frida_mode/src
parent773baf9391ff5f1793deb7968366819e7fa07adc (diff)
parent4c6d94ea5f854071277ed9729de2d4ef7d07cc84 (diff)
downloadafl++-08ca4d54a55fe73e64a994c41a12af61f52e497e.tar.gz
Merge pull request #1101 from AFLplusplus/dev
Dev
Diffstat (limited to 'frida_mode/src')
-rw-r--r--frida_mode/src/asan/asan.c28
-rw-r--r--frida_mode/src/asan/asan_arm32.c6
-rw-r--r--frida_mode/src/asan/asan_arm64.c6
-rw-r--r--frida_mode/src/asan/asan_x64.c6
-rw-r--r--frida_mode/src/asan/asan_x86.c6
-rw-r--r--frida_mode/src/cmplog/cmplog.c40
-rw-r--r--frida_mode/src/cmplog/cmplog_arm32.c4
-rw-r--r--frida_mode/src/cmplog/cmplog_arm64.c61
-rw-r--r--frida_mode/src/cmplog/cmplog_x64.c60
-rw-r--r--frida_mode/src/cmplog/cmplog_x86.c63
-rw-r--r--frida_mode/src/ctx/ctx_arm32.c5
-rw-r--r--frida_mode/src/ctx/ctx_arm64.c13
-rw-r--r--frida_mode/src/ctx/ctx_x64.c5
-rw-r--r--frida_mode/src/ctx/ctx_x86.c5
-rw-r--r--frida_mode/src/entry.c44
-rw-r--r--frida_mode/src/instrument/instrument.c61
-rw-r--r--frida_mode/src/instrument/instrument_arm32.c20
-rw-r--r--frida_mode/src/instrument/instrument_arm64.c5
-rw-r--r--frida_mode/src/instrument/instrument_coverage.c123
-rw-r--r--frida_mode/src/instrument/instrument_debug.c32
-rw-r--r--frida_mode/src/instrument/instrument_x64.c427
-rw-r--r--frida_mode/src/instrument/instrument_x86.c208
-rw-r--r--frida_mode/src/intercept.c5
-rw-r--r--frida_mode/src/js/api.js29
-rw-r--r--frida_mode/src/js/js.c21
-rw-r--r--frida_mode/src/js/js_api.c35
-rw-r--r--frida_mode/src/lib/lib.c55
-rw-r--r--frida_mode/src/lib/lib_apple.c20
-rw-r--r--frida_mode/src/main.c57
-rw-r--r--frida_mode/src/output.c13
-rw-r--r--frida_mode/src/persistent/persistent.c39
-rw-r--r--frida_mode/src/persistent/persistent_arm32.c6
-rw-r--r--frida_mode/src/persistent/persistent_arm64.c3
-rw-r--r--frida_mode/src/persistent/persistent_x64.c3
-rw-r--r--frida_mode/src/persistent/persistent_x86.c4
-rw-r--r--frida_mode/src/prefetch.c22
-rw-r--r--frida_mode/src/ranges.c138
-rw-r--r--frida_mode/src/seccomp/seccomp.c140
-rw-r--r--frida_mode/src/seccomp/seccomp_atomic.c12
-rw-r--r--frida_mode/src/seccomp/seccomp_callback.c145
-rw-r--r--frida_mode/src/seccomp/seccomp_child.c31
-rw-r--r--frida_mode/src/seccomp/seccomp_event.c31
-rw-r--r--frida_mode/src/seccomp/seccomp_filter.c87
-rw-r--r--frida_mode/src/seccomp/seccomp_print.c30
-rw-r--r--frida_mode/src/seccomp/seccomp_socket.c35
-rw-r--r--frida_mode/src/seccomp/seccomp_syscall.c15
-rw-r--r--frida_mode/src/stalker.c57
-rw-r--r--frida_mode/src/stats/stats.c21
-rw-r--r--frida_mode/src/stats/stats_arm32.c8
-rw-r--r--frida_mode/src/stats/stats_arm64.c6
-rw-r--r--frida_mode/src/stats/stats_x86_64.c16
-rw-r--r--frida_mode/src/util.c73
52 files changed, 1629 insertions, 756 deletions
diff --git a/frida_mode/src/asan/asan.c b/frida_mode/src/asan/asan.c
index b2e763ca..884bec53 100644
--- a/frida_mode/src/asan/asan.c
+++ b/frida_mode/src/asan/asan.c
@@ -1,8 +1,8 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "asan.h"
+#include "ranges.h"
+#include "util.h"
static gboolean asan_enabled = FALSE;
gboolean asan_initialized = FALSE;
@@ -11,12 +11,12 @@ void asan_config(void) {
if (getenv("AFL_USE_FASAN") != NULL) {
- OKF("Frida ASAN mode enabled");
+ FOKF("Frida ASAN mode enabled");
asan_enabled = TRUE;
} else {
- OKF("Frida ASAN mode disabled");
+ FOKF("Frida ASAN mode disabled");
}
@@ -33,3 +33,23 @@ void asan_init(void) {
}
+static gboolean asan_exclude_module(const GumModuleDetails *details,
+ gpointer user_data) {
+
+ gchar * symbol_name = (gchar *)user_data;
+ GumAddress address;
+
+ address = gum_module_find_export_by_name(details->name, symbol_name);
+ if (address == 0) { return TRUE; }
+
+ ranges_add_exclude((GumMemoryRange *)details->range);
+ return FALSE;
+
+}
+
+void asan_exclude_module_by_symbol(gchar *symbol_name) {
+
+ gum_process_enumerate_modules(asan_exclude_module, symbol_name);
+
+}
+
diff --git a/frida_mode/src/asan/asan_arm32.c b/frida_mode/src/asan/asan_arm32.c
index f5fa4713..21400881 100644
--- a/frida_mode/src/asan/asan_arm32.c
+++ b/frida_mode/src/asan/asan_arm32.c
@@ -1,7 +1,5 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "asan.h"
#include "util.h"
@@ -12,7 +10,7 @@ void asan_instrument(const cs_insn *instr, GumStalkerIterator *iterator) {
UNUSED_PARAMETER(iterator);
if (asan_initialized) {
- FATAL("ASAN mode not supported on this architecture");
+ FFATAL("ASAN mode not supported on this architecture");
}
@@ -20,7 +18,7 @@ void asan_instrument(const cs_insn *instr, GumStalkerIterator *iterator) {
void asan_arch_init(void) {
- FATAL("ASAN mode not supported on this architecture");
+ FFATAL("ASAN mode not supported on this architecture");
}
diff --git a/frida_mode/src/asan/asan_arm64.c b/frida_mode/src/asan/asan_arm64.c
index 65524e03..b2adfa52 100644
--- a/frida_mode/src/asan/asan_arm64.c
+++ b/frida_mode/src/asan/asan_arm64.c
@@ -1,8 +1,6 @@
#include <dlfcn.h>
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "asan.h"
#include "ctx.h"
#include "util.h"
@@ -86,10 +84,12 @@ void asan_arch_init(void) {
asan_storeN = (asan_loadN_t)dlsym(RTLD_DEFAULT, "__asan_storeN");
if (asan_loadN == NULL || asan_storeN == NULL) {
- FATAL("Frida ASAN failed to find '__asan_loadN' or '__asan_storeN'");
+ FFATAL("Frida ASAN failed to find '__asan_loadN' or '__asan_storeN'");
}
+ asan_exclude_module_by_symbol("__asan_loadN");
+
}
#endif
diff --git a/frida_mode/src/asan/asan_x64.c b/frida_mode/src/asan/asan_x64.c
index 5c12669f..a287ea34 100644
--- a/frida_mode/src/asan/asan_x64.c
+++ b/frida_mode/src/asan/asan_x64.c
@@ -1,8 +1,6 @@
#include <dlfcn.h>
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "asan.h"
#include "ctx.h"
#include "util.h"
@@ -83,10 +81,12 @@ void asan_arch_init(void) {
asan_storeN = (asan_loadN_t)dlsym(RTLD_DEFAULT, "__asan_storeN");
if (asan_loadN == NULL || asan_storeN == NULL) {
- FATAL("Frida ASAN failed to find '__asan_loadN' or '__asan_storeN'");
+ FFATAL("Frida ASAN failed to find '__asan_loadN' or '__asan_storeN'");
}
+ asan_exclude_module_by_symbol("__asan_loadN");
+
}
#endif
diff --git a/frida_mode/src/asan/asan_x86.c b/frida_mode/src/asan/asan_x86.c
index 6d2f9e2b..331d026b 100644
--- a/frida_mode/src/asan/asan_x86.c
+++ b/frida_mode/src/asan/asan_x86.c
@@ -1,8 +1,6 @@
#include <dlfcn.h>
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "asan.h"
#include "ctx.h"
#include "util.h"
@@ -83,10 +81,12 @@ void asan_arch_init(void) {
asan_storeN = (asan_loadN_t)dlsym(RTLD_DEFAULT, "__asan_storeN");
if (asan_loadN == NULL || asan_storeN == NULL) {
- FATAL("Frida ASAN failed to find '__asan_loadN' or '__asan_storeN'");
+ FFATAL("Frida ASAN failed to find '__asan_loadN' or '__asan_storeN'");
}
+ asan_exclude_module_by_symbol("__asan_loadN");
+
}
#endif
diff --git a/frida_mode/src/cmplog/cmplog.c b/frida_mode/src/cmplog/cmplog.c
index ae3116eb..443baa1d 100644
--- a/frida_mode/src/cmplog/cmplog.c
+++ b/frida_mode/src/cmplog/cmplog.c
@@ -7,8 +7,6 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "util.h"
#define DEFAULT_MMAP_MIN_ADDR (32UL << 10)
@@ -35,14 +33,28 @@ static gboolean cmplog_range(const GumRangeDetails *details,
static gint cmplog_sort(gconstpointer a, gconstpointer b) {
- return ((GumMemoryRange *)b)->base_address -
- ((GumMemoryRange *)a)->base_address;
+ GumMemoryRange *ra = (GumMemoryRange *)a;
+ GumMemoryRange *rb = (GumMemoryRange *)b;
+
+ if (ra->base_address < rb->base_address) {
+
+ return -1;
+
+ } else if (ra->base_address > rb->base_address) {
+
+ return 1;
+
+ } else {
+
+ return 0;
+
+ }
}
static void cmplog_get_ranges(void) {
- OKF("CMPLOG - Collecting ranges");
+ FOKF("CMPLOG - Collecting ranges");
cmplog_ranges = g_array_sized_new(false, false, sizeof(GumMemoryRange), 100);
gum_process_enumerate_ranges(GUM_PAGE_READ, cmplog_range, cmplog_ranges);
@@ -56,7 +68,7 @@ void cmplog_config(void) {
void cmplog_init(void) {
- OKF("CMPLOG - Enabled [%c]", __afl_cmp_map == NULL ? ' ' : 'X');
+ FOKF("CMPLOG - Enabled [%c]", __afl_cmp_map == NULL ? ' ' : 'X');
if (__afl_cmp_map == NULL) { return; }
@@ -65,9 +77,9 @@ void cmplog_init(void) {
for (guint i = 0; i < cmplog_ranges->len; i++) {
GumMemoryRange *range = &g_array_index(cmplog_ranges, GumMemoryRange, i);
- OKF("CMPLOG Range - %3u: 0x%016" G_GINT64_MODIFIER
- "X - 0x%016" G_GINT64_MODIFIER "X",
- i, range->base_address, range->base_address + range->size);
+ FOKF("CMPLOG Range - %3u: 0x%016" G_GINT64_MODIFIER
+ "X - 0x%016" G_GINT64_MODIFIER "X",
+ i, range->base_address, range->base_address + range->size);
}
@@ -78,14 +90,14 @@ void cmplog_init(void) {
hash_yes = g_hash_table_new(g_direct_hash, g_direct_equal);
if (hash_yes == NULL) {
- FATAL("Failed to g_hash_table_new, errno: %d", errno);
+ FFATAL("Failed to g_hash_table_new, errno: %d", errno);
}
hash_no = g_hash_table_new(g_direct_hash, g_direct_equal);
if (hash_no == NULL) {
- FATAL("Failed to g_hash_table_new, errno: %d", errno);
+ FFATAL("Failed to g_hash_table_new, errno: %d", errno);
}
@@ -117,7 +129,7 @@ gboolean cmplog_test_addr(guint64 addr, size_t size) {
if (!g_hash_table_add(hash_no, GSIZE_TO_POINTER(addr))) {
- FATAL("Failed - g_hash_table_add");
+ FFATAL("Failed - g_hash_table_add");
}
@@ -127,7 +139,7 @@ gboolean cmplog_test_addr(guint64 addr, size_t size) {
if (!g_hash_table_add(hash_yes, GSIZE_TO_POINTER(addr))) {
- FATAL("Failed - g_hash_table_add");
+ FFATAL("Failed - g_hash_table_add");
}
@@ -139,7 +151,7 @@ gboolean cmplog_test_addr(guint64 addr, size_t size) {
gboolean cmplog_is_readable(guint64 addr, size_t size) {
- if (cmplog_ranges == NULL) FATAL("CMPLOG not initialized");
+ if (cmplog_ranges == NULL) FFATAL("CMPLOG not initialized");
/*
* The Linux kernel prevents mmap from allocating from the very bottom of the
diff --git a/frida_mode/src/cmplog/cmplog_arm32.c b/frida_mode/src/cmplog/cmplog_arm32.c
index ac703408..106baa52 100644
--- a/frida_mode/src/cmplog/cmplog_arm32.c
+++ b/frida_mode/src/cmplog/cmplog_arm32.c
@@ -1,7 +1,5 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "frida_cmplog.h"
#include "util.h"
@@ -11,7 +9,7 @@ void cmplog_instrument(const cs_insn *instr, GumStalkerIterator *iterator) {
UNUSED_PARAMETER(instr);
UNUSED_PARAMETER(iterator);
if (__afl_cmp_map == NULL) { return; }
- FATAL("CMPLOG mode not supported on this architecture");
+ FFATAL("CMPLOG mode not supported on this architecture");
}
diff --git a/frida_mode/src/cmplog/cmplog_arm64.c b/frida_mode/src/cmplog/cmplog_arm64.c
index dd97f38d..515a6256 100644
--- a/frida_mode/src/cmplog/cmplog_arm64.c
+++ b/frida_mode/src/cmplog/cmplog_arm64.c
@@ -5,6 +5,7 @@
#include "ctx.h"
#include "frida_cmplog.h"
+#include "instrument.h"
#include "util.h"
#if defined(__aarch64__)
@@ -66,7 +67,7 @@ static gboolean cmplog_read_mem(GumCpuContext *ctx, uint8_t size,
*val = *((guint64 *)GSIZE_TO_POINTER(address));
return TRUE;
default:
- FATAL("Invalid operand size: %d\n", size);
+ FFATAL("Invalid operand size: %d\n", size);
}
@@ -88,7 +89,7 @@ static gboolean cmplog_get_operand_value(GumCpuContext *context,
case ARM64_OP_MEM:
return cmplog_read_mem(context, ctx->size, &ctx->mem, val);
default:
- FATAL("Invalid operand type: %d\n", ctx->type);
+ FFATAL("Invalid operand type: %d\n", ctx->type);
}
@@ -104,30 +105,45 @@ static void cmplog_call_callout(GumCpuContext *context, gpointer user_data) {
gsize x0 = ctx_read_reg(context, ARM64_REG_X0);
gsize x1 = ctx_read_reg(context, ARM64_REG_X1);
- if (((G_MAXULONG - x0) < 32) || ((G_MAXULONG - x1) < 32)) return;
+ if (((G_MAXULONG - x0) < 31) || ((G_MAXULONG - x1) < 31)) return;
- if (!cmplog_is_readable(x0, 32) || !cmplog_is_readable(x1, 32)) return;
+ if (!cmplog_is_readable(x0, 31) || !cmplog_is_readable(x1, 31)) return;
void *ptr1 = GSIZE_TO_POINTER(x0);
void *ptr2 = GSIZE_TO_POINTER(x1);
- uintptr_t k = address;
+ guint64 k = instrument_get_offset_hash(GUM_ADDRESS(address));
- k = (k >> 4) ^ (k << 8);
- k &= CMP_MAP_W - 1;
+ if (__afl_cmp_map->headers[k].type != CMP_TYPE_RTN) {
+
+ __afl_cmp_map->headers[k].type = CMP_TYPE_RTN;
+ __afl_cmp_map->headers[k].hits = 0;
+
+ }
- __afl_cmp_map->headers[k].type = CMP_TYPE_RTN;
+ u32 hits = 0;
+
+ if (__afl_cmp_map->headers[k].hits == 0) {
+
+ __afl_cmp_map->headers[k].shape = 30;
+
+ } else {
+
+ hits = __afl_cmp_map->headers[k].hits;
+
+ }
- u32 hits = __afl_cmp_map->headers[k].hits;
__afl_cmp_map->headers[k].hits = hits + 1;
- __afl_cmp_map->headers[k].shape = 31;
+ __afl_cmp_map->headers[k].shape = 30;
hits &= CMP_MAP_RTN_H - 1;
+ ((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v0_len = 31;
+ ((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v1_len = 31;
gum_memcpy(((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v0, ptr1,
- 32);
+ 31);
gum_memcpy(((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v1, ptr2,
- 32);
+ 31);
}
@@ -147,7 +163,7 @@ static void cmplog_instrument_put_operand(cmplog_ctx_t *ctx,
gum_memcpy(&ctx->mem, &operand->mem, sizeof(arm64_op_mem));
break;
default:
- FATAL("Invalid operand type: %d\n", operand->type);
+ FFATAL("Invalid operand type: %d\n", operand->type);
}
@@ -193,12 +209,23 @@ static void cmplog_handle_cmp_sub(GumCpuContext *context, gsize operand1,
k = (k >> 4) ^ (k << 8);
k &= CMP_MAP_W - 1;
- __afl_cmp_map->headers[k].type = CMP_TYPE_INS;
+ if (__afl_cmp_map->headers[k].type != CMP_TYPE_INS)
+ __afl_cmp_map->headers[k].hits = 0;
- u32 hits = __afl_cmp_map->headers[k].hits;
- __afl_cmp_map->headers[k].hits = hits + 1;
+ u32 hits = 0;
+
+ if (__afl_cmp_map->headers[k].hits == 0) {
+
+ __afl_cmp_map->headers[k].type = CMP_TYPE_INS;
+ __afl_cmp_map->headers[k].shape = (size - 1);
+
+ } else {
- __afl_cmp_map->headers[k].shape = (size - 1);
+ hits = __afl_cmp_map->headers[k].hits;
+
+ }
+
+ __afl_cmp_map->headers[k].hits = hits + 1;
hits &= CMP_MAP_H - 1;
__afl_cmp_map->log[k][hits].v0 = operand1;
diff --git a/frida_mode/src/cmplog/cmplog_x64.c b/frida_mode/src/cmplog/cmplog_x64.c
index 0d18767a..7d515336 100644
--- a/frida_mode/src/cmplog/cmplog_x64.c
+++ b/frida_mode/src/cmplog/cmplog_x64.c
@@ -5,6 +5,7 @@
#include "ctx.h"
#include "frida_cmplog.h"
+#include "instrument.h"
#include "util.h"
#if defined(__x86_64__)
@@ -61,7 +62,7 @@ static gboolean cmplog_read_mem(GumCpuContext *ctx, uint8_t size,
*val = *((guint64 *)GSIZE_TO_POINTER(address));
return TRUE;
default:
- FATAL("Invalid operand size: %d\n", size);
+ FFATAL("Invalid operand size: %d\n", size);
}
@@ -83,7 +84,7 @@ static gboolean cmplog_get_operand_value(GumCpuContext *context,
case X86_OP_MEM:
return cmplog_read_mem(context, ctx->size, &ctx->mem, val);
default:
- FATAL("Invalid operand type: %d\n", ctx->type);
+ FFATAL("Invalid operand type: %d\n", ctx->type);
}
@@ -99,30 +100,43 @@ static void cmplog_call_callout(GumCpuContext *context, gpointer user_data) {
gsize rdi = ctx_read_reg(context, X86_REG_RDI);
gsize rsi = ctx_read_reg(context, X86_REG_RSI);
- if (((G_MAXULONG - rdi) < 32) || ((G_MAXULONG - rsi) < 32)) return;
+ if (((G_MAXULONG - rdi) < 31) || ((G_MAXULONG - rsi) < 31)) return;
- if (!cmplog_is_readable(rdi, 32) || !cmplog_is_readable(rsi, 32)) return;
+ if (!cmplog_is_readable(rdi, 31) || !cmplog_is_readable(rsi, 31)) return;
void *ptr1 = GSIZE_TO_POINTER(rdi);
void *ptr2 = GSIZE_TO_POINTER(rsi);
- uintptr_t k = address;
+ guint64 k = instrument_get_offset_hash(GUM_ADDRESS(address));
- k = (k >> 4) ^ (k << 8);
- k &= CMP_MAP_W - 1;
+ if (__afl_cmp_map->headers[k].type != CMP_TYPE_RTN) {
- __afl_cmp_map->headers[k].type = CMP_TYPE_RTN;
+ __afl_cmp_map->headers[k].type = CMP_TYPE_RTN;
+ __afl_cmp_map->headers[k].hits = 0;
- u32 hits = __afl_cmp_map->headers[k].hits;
- __afl_cmp_map->headers[k].hits = hits + 1;
+ }
+
+ u32 hits = 0;
+
+ if (__afl_cmp_map->headers[k].hits == 0) {
+
+ __afl_cmp_map->headers[k].shape = 30;
+
+ } else {
+
+ hits = __afl_cmp_map->headers[k].hits;
+
+ }
- __afl_cmp_map->headers[k].shape = 31;
+ __afl_cmp_map->headers[k].hits = hits + 1;
hits &= CMP_MAP_RTN_H - 1;
+ ((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v0_len = 31;
+ ((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v1_len = 31;
gum_memcpy(((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v0, ptr1,
- 32);
+ 31);
gum_memcpy(((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v1, ptr2,
- 32);
+ 31);
}
@@ -143,7 +157,7 @@ static void cmplog_instrument_put_operand(cmplog_ctx_t *ctx,
gum_memcpy(&ctx->mem, &operand->mem, sizeof(x86_op_mem));
break;
default:
- FATAL("Invalid operand type: %d\n", operand->type);
+ FFATAL("Invalid operand type: %d\n", operand->type);
}
@@ -179,13 +193,23 @@ static void cmplog_handle_cmp_sub(GumCpuContext *context, gsize operand1,
k = (k >> 4) ^ (k << 8);
k &= CMP_MAP_W - 7;
- __afl_cmp_map->headers[k].type = CMP_TYPE_INS;
+ if (__afl_cmp_map->headers[k].type != CMP_TYPE_INS)
+ __afl_cmp_map->headers[k].hits = 0;
- u32 hits = __afl_cmp_map->headers[k].hits;
- __afl_cmp_map->headers[k].hits = hits + 1;
+ u32 hits = 0;
+
+ if (__afl_cmp_map->headers[k].hits == 0) {
+
+ __afl_cmp_map->headers[k].type = CMP_TYPE_INS;
+ __afl_cmp_map->headers[k].shape = (size - 1);
- __afl_cmp_map->headers[k].shape = (size - 1);
+ } else {
+ hits = __afl_cmp_map->headers[k].hits;
+
+ }
+
+ __afl_cmp_map->headers[k].hits = hits + 1;
hits &= CMP_MAP_H - 1;
__afl_cmp_map->log[k][hits].v0 = operand1;
__afl_cmp_map->log[k][hits].v1 = operand2;
diff --git a/frida_mode/src/cmplog/cmplog_x86.c b/frida_mode/src/cmplog/cmplog_x86.c
index dd666c34..4a747417 100644
--- a/frida_mode/src/cmplog/cmplog_x86.c
+++ b/frida_mode/src/cmplog/cmplog_x86.c
@@ -5,6 +5,7 @@
#include "ctx.h"
#include "frida_cmplog.h"
+#include "instrument.h"
#include "util.h"
#if defined(__i386__)
@@ -58,7 +59,7 @@ static gboolean cmplog_read_mem(GumCpuContext *ctx, uint8_t size,
*val = *((guint32 *)GSIZE_TO_POINTER(address));
return TRUE;
default:
- FATAL("Invalid operand size: %d\n", size);
+ FFATAL("Invalid operand size: %d\n", size);
}
@@ -80,7 +81,7 @@ static gboolean cmplog_get_operand_value(GumCpuContext *context,
case X86_OP_MEM:
return cmplog_read_mem(context, ctx->size, &ctx->mem, val);
default:
- FATAL("Invalid operand type: %d\n", ctx->type);
+ FFATAL("Invalid operand type: %d\n", ctx->type);
}
@@ -104,30 +105,43 @@ static void cmplog_call_callout(GumCpuContext *context, gpointer user_data) {
gsize arg1 = esp[0];
gsize arg2 = esp[1];
- if (((G_MAXULONG - arg1) < 32) || ((G_MAXULONG - arg2) < 32)) return;
+ if (((G_MAXULONG - arg1) < 31) || ((G_MAXULONG - arg2) < 31)) return;
- if (!cmplog_is_readable(arg1, 32) || !cmplog_is_readable(arg2, 32)) return;
+ if (!cmplog_is_readable(arg1, 31) || !cmplog_is_readable(arg2, 31)) return;
void *ptr1 = GSIZE_TO_POINTER(arg1);
void *ptr2 = GSIZE_TO_POINTER(arg2);
- uintptr_t k = address;
+ guint64 k = instrument_get_offset_hash(GUM_ADDRESS(address));
- k = (k >> 4) ^ (k << 8);
- k &= CMP_MAP_W - 1;
+ if (__afl_cmp_map->headers[k].type != CMP_TYPE_RTN) {
- __afl_cmp_map->headers[k].type = CMP_TYPE_RTN;
+ __afl_cmp_map->headers[k].type = CMP_TYPE_RTN;
+ __afl_cmp_map->headers[k].hits = 0;
- u32 hits = __afl_cmp_map->headers[k].hits;
- __afl_cmp_map->headers[k].hits = hits + 1;
+ }
+
+ u32 hits = 0;
+
+ if (__afl_cmp_map->headers[k].hits == 0) {
+
+ __afl_cmp_map->headers[k].shape = 30;
+
+ } else {
+
+ hits = __afl_cmp_map->headers[k].hits;
+
+ }
- __afl_cmp_map->headers[k].shape = 31;
+ __afl_cmp_map->headers[k].hits = hits + 1;
hits &= CMP_MAP_RTN_H - 1;
+ ((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v0_len = 31;
+ ((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v1_len = 31;
gum_memcpy(((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v0, ptr1,
- 32);
+ 31);
gum_memcpy(((struct cmpfn_operands *)__afl_cmp_map->log[k])[hits].v1, ptr2,
- 32);
+ 31);
}
@@ -148,7 +162,7 @@ static void cmplog_instrument_put_operand(cmplog_ctx_t *ctx,
gum_memcpy(&ctx->mem, &operand->mem, sizeof(x86_op_mem));
break;
default:
- FATAL("Invalid operand type: %d\n", operand->type);
+ FFATAL("Invalid operand type: %d\n", operand->type);
}
@@ -184,12 +198,23 @@ static void cmplog_handle_cmp_sub(GumCpuContext *context, gsize operand1,
k = (k >> 4) ^ (k << 8);
k &= CMP_MAP_W - 1;
- __afl_cmp_map->headers[k].type = CMP_TYPE_INS;
+ if (__afl_cmp_map->headers[k].type != CMP_TYPE_INS)
+ __afl_cmp_map->headers[k].hits = 0;
- u32 hits = __afl_cmp_map->headers[k].hits;
- __afl_cmp_map->headers[k].hits = hits + 1;
+ u32 hits = 0;
+
+ if (__afl_cmp_map->headers[k].hits == 0) {
+
+ __afl_cmp_map->headers[k].type = CMP_TYPE_INS;
+ __afl_cmp_map->headers[k].shape = (size - 1);
- __afl_cmp_map->headers[k].shape = (size - 1);
+ } else {
+
+ hits = __afl_cmp_map->headers[k].hits;
+
+ }
+
+ __afl_cmp_map->headers[k].hits = hits + 1;
hits &= CMP_MAP_H - 1;
__afl_cmp_map->log[k][hits].v0 = operand1;
@@ -203,7 +228,7 @@ static void cmplog_cmp_sub_callout(GumCpuContext *context, gpointer user_data) {
gsize operand1;
gsize operand2;
- if (ctx->operand1.size != ctx->operand2.size) FATAL("Operand size mismatch");
+ if (ctx->operand1.size != ctx->operand2.size) FFATAL("Operand size mismatch");
if (!cmplog_get_operand_value(context, &ctx->operand1, &operand1)) { return; }
if (!cmplog_get_operand_value(context, &ctx->operand2, &operand2)) { return; }
diff --git a/frida_mode/src/ctx/ctx_arm32.c b/frida_mode/src/ctx/ctx_arm32.c
index 9fc70fb4..28fc706b 100644
--- a/frida_mode/src/ctx/ctx_arm32.c
+++ b/frida_mode/src/ctx/ctx_arm32.c
@@ -1,14 +1,13 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "ctx.h"
+#include "util.h"
#if defined(__arm__)
gsize ctx_read_reg(GumArmCpuContext *ctx, arm_reg reg) {
- FATAL("ctx_read_reg unimplemented for this architecture");
+ FFATAL("ctx_read_reg unimplemented for this architecture");
}
diff --git a/frida_mode/src/ctx/ctx_arm64.c b/frida_mode/src/ctx/ctx_arm64.c
index a735401b..63b6cf09 100644
--- a/frida_mode/src/ctx/ctx_arm64.c
+++ b/frida_mode/src/ctx/ctx_arm64.c
@@ -1,8 +1,7 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "ctx.h"
+#include "util.h"
#if defined(__aarch64__)
@@ -174,7 +173,7 @@ gsize ctx_read_reg(GumArm64CpuContext *ctx, arm64_reg reg) {
ARM64_REG_64(ARM64_REG_SP, ctx->sp)
default:
- FATAL("Failed to read register: %d", reg);
+ FFATAL("Failed to read register: %d", reg);
return 0;
}
@@ -206,7 +205,7 @@ size_t ctx_get_size(const cs_insn *instr, cs_arm64_op *operand) {
}
mnemonic_len = strlen(instr->mnemonic);
- if (mnemonic_len == 0) { FATAL("No mnemonic found"); };
+ if (mnemonic_len == 0) { FFATAL("No mnemonic found"); };
char last = instr->mnemonic[mnemonic_len - 1];
switch (last) {
@@ -252,14 +251,14 @@ size_t ctx_get_size(const cs_insn *instr, cs_arm64_op *operand) {
if (mnemonic_len < 3) {
- FATAL("VAS Mnemonic too short: %s\n", instr->mnemonic);
+ FFATAL("VAS Mnemonic too short: %s\n", instr->mnemonic);
}
vas_digit = instr->mnemonic[2];
if (vas_digit < '0' || vas_digit > '9') {
- FATAL("VAS Mnemonic digit out of range: %s\n", instr->mnemonic);
+ FFATAL("VAS Mnemonic digit out of range: %s\n", instr->mnemonic);
}
@@ -293,7 +292,7 @@ size_t ctx_get_size(const cs_insn *instr, cs_arm64_op *operand) {
case ARM64_VAS_16B:
return 16 * count_byte;
default:
- FATAL("Unexpected VAS type: %s %d", instr->mnemonic, operand->vas);
+ FFATAL("Unexpected VAS type: %s %d", instr->mnemonic, operand->vas);
}
diff --git a/frida_mode/src/ctx/ctx_x64.c b/frida_mode/src/ctx/ctx_x64.c
index da5cb13a..02423416 100644
--- a/frida_mode/src/ctx/ctx_x64.c
+++ b/frida_mode/src/ctx/ctx_x64.c
@@ -1,8 +1,7 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "ctx.h"
+#include "util.h"
#if defined(__x86_64__)
@@ -121,7 +120,7 @@ gsize ctx_read_reg(GumX64CpuContext *ctx, x86_reg reg) {
X86_REG_64(X86_REG_RIP, ctx->rip)
default:
- FATAL("Failed to read register: %d", reg);
+ FFATAL("Failed to read register: %d", reg);
return 0;
}
diff --git a/frida_mode/src/ctx/ctx_x86.c b/frida_mode/src/ctx/ctx_x86.c
index 1a587702..438e1fde 100644
--- a/frida_mode/src/ctx/ctx_x86.c
+++ b/frida_mode/src/ctx/ctx_x86.c
@@ -1,8 +1,7 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "ctx.h"
+#include "util.h"
#if defined(__i386__)
@@ -72,7 +71,7 @@ gsize ctx_read_reg(GumIA32CpuContext *ctx, x86_reg reg) {
X86_REG_32(X86_REG_EIP, ctx->eip)
default:
- FATAL("Failed to read register: %d", reg);
+ FFATAL("Failed to read register: %d", reg);
return 0;
}
diff --git a/frida_mode/src/entry.c b/frida_mode/src/entry.c
index 186ddd3a..de645fdb 100644
--- a/frida_mode/src/entry.c
+++ b/frida_mode/src/entry.c
@@ -1,8 +1,10 @@
#include <dlfcn.h>
-#include "frida-gumjs.h"
+#if defined(__linux__) && !defined(__ANDROID__)
+ #include <sys/prctl.h>
+#endif
-#include "debug.h"
+#include "frida-gumjs.h"
#include "entry.h"
#include "instrument.h"
@@ -16,33 +18,61 @@
extern void __afl_manual_init();
guint64 entry_point = 0;
+gboolean traceable = FALSE;
gboolean entry_compiled = FALSE;
gboolean entry_run = FALSE;
static void entry_launch(void) {
- OKF("Entry point reached");
+ FOKF("Entry point reached");
__afl_manual_init();
/* Child here */
entry_run = TRUE;
+ entry_on_fork();
instrument_on_fork();
seccomp_on_fork();
stats_on_fork();
}
+#if defined(__linux__) && defined(PR_SET_PTRACER) && !defined(__ANDROID__)
+void entry_on_fork(void) {
+
+ if (traceable) {
+
+ if (prctl(PR_SET_PTRACER, PR_SET_PTRACER_ANY) < 0) {
+
+ FFATAL("Failed to PR_SET_PTRACER");
+
+ }
+
+ }
+
+}
+
+#else
+void entry_on_fork(void) {
+
+ if (traceable) { FWARNF("AFL_FRIDA_TRACEABLE unsupported"); }
+
+}
+
+#endif
+
void entry_config(void) {
- entry_point = util_read_address("AFL_ENTRYPOINT");
+ entry_point = util_read_address("AFL_ENTRYPOINT", 0);
+ if (getenv("AFL_FRIDA_TRACEABLE") != NULL) { traceable = TRUE; }
}
void entry_init(void) {
- OKF("entry_point: 0x%016" G_GINT64_MODIFIER "X", entry_point);
+ FOKF("entry_point: 0x%016" G_GINT64_MODIFIER "X", entry_point);
+ FOKF("dumpable: [%c]", traceable ? 'X' : ' ');
- if (dlopen(NULL, RTLD_NOW) == NULL) { FATAL("Failed to dlopen: %d", errno); }
+ if (dlopen(NULL, RTLD_NOW) == NULL) { FFATAL("Failed to dlopen: %d", errno); }
}
@@ -64,7 +94,7 @@ static void entry_callout(GumCpuContext *cpu_context, gpointer user_data) {
void entry_prologue(GumStalkerIterator *iterator, GumStalkerOutput *output) {
UNUSED_PARAMETER(output);
- OKF("AFL_ENTRYPOINT reached");
+ FOKF("AFL_ENTRYPOINT reached");
if (persistent_start == 0) {
diff --git a/frida_mode/src/instrument/instrument.c b/frida_mode/src/instrument/instrument.c
index fd0982f8..8ee21f5b 100644
--- a/frida_mode/src/instrument/instrument.c
+++ b/frida_mode/src/instrument/instrument.c
@@ -6,7 +6,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "hash.h"
#include "asan.h"
@@ -69,7 +68,8 @@ guint64 instrument_get_offset_hash(GumAddress current_rip) {
guint64 area_offset = hash64((unsigned char *)&current_rip,
sizeof(GumAddress), instrument_hash_seed);
- return area_offset &= MAP_SIZE - 1;
+ gsize map_size_pow2 = util_log2(__afl_map_size);
+ return area_offset &= ((1 << map_size_pow2) - 1);
}
@@ -135,8 +135,8 @@ __attribute__((hot)) static void on_basic_block(GumCpuContext *context,
previous_rip = current_rip;
previous_end = current_end;
- instrument_previous_pc = ((current_pc & (MAP_SIZE - 1) >> 1)) |
- ((current_pc & 0x1) << (MAP_SIZE_POW2 - 1));
+ gsize map_size_pow2 = util_log2(__afl_map_size);
+ instrument_previous_pc = util_rotate(current_pc, 1, map_size_pow2);
}
@@ -193,7 +193,20 @@ static void instrument_basic_block(GumStalkerIterator *iterator,
instrument_debug_start(instr->address, output);
instrument_coverage_start(instr->address);
+#if defined(__arm__)
+ if (output->encoding == GUM_INSTRUCTION_SPECIAL) {
+
+ prefetch_write(GSIZE_TO_POINTER(instr->address + 1));
+
+ } else {
+
+ prefetch_write(GSIZE_TO_POINTER(instr->address));
+
+ }
+
+#else
prefetch_write(GSIZE_TO_POINTER(instr->address));
+#endif
if (likely(!excluded)) {
@@ -213,7 +226,7 @@ static void instrument_basic_block(GumStalkerIterator *iterator,
}
- instrument_debug_instruction(instr->address, instr->size);
+ instrument_debug_instruction(instr->address, instr->size, output);
if (likely(!excluded)) {
@@ -246,7 +259,7 @@ void instrument_config(void) {
instrument_tracing = (getenv("AFL_FRIDA_INST_TRACE") != NULL);
instrument_unique = (getenv("AFL_FRIDA_INST_TRACE_UNIQUE") != NULL);
instrument_use_fixed_seed = (getenv("AFL_FRIDA_INST_SEED") != NULL);
- instrument_fixed_seed = util_read_num("AFL_FRIDA_INST_SEED");
+ instrument_fixed_seed = util_read_num("AFL_FRIDA_INST_SEED", 0);
instrument_coverage_unstable_filename =
(getenv("AFL_FRIDA_INST_UNSTABLE_COVERAGE_FILE"));
@@ -261,14 +274,14 @@ void instrument_init(void) {
if (!instrument_is_coverage_optimize_supported()) instrument_optimize = false;
- OKF("Instrumentation - optimize [%c]", instrument_optimize ? 'X' : ' ');
- OKF("Instrumentation - tracing [%c]", instrument_tracing ? 'X' : ' ');
- OKF("Instrumentation - unique [%c]", instrument_unique ? 'X' : ' ');
- OKF("Instrumentation - fixed seed [%c] [0x%016" G_GINT64_MODIFIER "x]",
- instrument_use_fixed_seed ? 'X' : ' ', instrument_fixed_seed);
- OKF("Instrumentation - unstable coverage [%c] [%s]",
- instrument_coverage_unstable_filename == NULL ? ' ' : 'X',
- instrument_coverage_unstable_filename);
+ FOKF("Instrumentation - optimize [%c]", instrument_optimize ? 'X' : ' ');
+ FOKF("Instrumentation - tracing [%c]", instrument_tracing ? 'X' : ' ');
+ FOKF("Instrumentation - unique [%c]", instrument_unique ? 'X' : ' ');
+ FOKF("Instrumentation - fixed seed [%c] [0x%016" G_GINT64_MODIFIER "x]",
+ instrument_use_fixed_seed ? 'X' : ' ', instrument_fixed_seed);
+ FOKF("Instrumentation - unstable coverage [%c] [%s]",
+ instrument_coverage_unstable_filename == NULL ? ' ' : 'X',
+ instrument_coverage_unstable_filename);
if (instrument_tracing && instrument_optimize) {
@@ -304,7 +317,8 @@ void instrument_init(void) {
if (instrument_unique) {
- int shm_id = shmget(IPC_PRIVATE, MAP_SIZE, IPC_CREAT | IPC_EXCL | 0600);
+ int shm_id =
+ shmget(IPC_PRIVATE, __afl_map_size, IPC_CREAT | IPC_EXCL | 0600);
if (shm_id < 0) { FATAL("shm_id < 0 - errno: %d\n", errno); }
edges_notified = shmat(shm_id, NULL, 0);
@@ -321,7 +335,7 @@ void instrument_init(void) {
}
/* Clear it, not sure it's necessary, just seems like good practice */
- memset(edges_notified, '\0', MAP_SIZE);
+ memset(edges_notified, '\0', __afl_map_size);
}
@@ -341,15 +355,22 @@ void instrument_init(void) {
* parallel fuzzing. The seed itself, doesn't have to be random, it
* just needs to be different for each instance.
*/
- instrument_hash_seed = g_get_monotonic_time() ^
- (((guint64)getpid()) << 32) ^ syscall(SYS_gettid);
+ guint64 tid;
+#if defined(__APPLE__)
+ pthread_threadid_np(NULL, &tid);
+#else
+ tid = syscall(SYS_gettid);
+#endif
+ instrument_hash_seed =
+ g_get_monotonic_time() ^ (((guint64)getpid()) << 32) ^ tid;
}
- OKF("Instrumentation - seed [0x%016" G_GINT64_MODIFIER "x]",
- instrument_hash_seed);
+ FOKF("Instrumentation - seed [0x%016" G_GINT64_MODIFIER "x]",
+ instrument_hash_seed);
instrument_hash_zero = instrument_get_offset_hash(0);
+ instrument_coverage_optimize_init();
instrument_debug_init();
instrument_coverage_init();
asan_init();
diff --git a/frida_mode/src/instrument/instrument_arm32.c b/frida_mode/src/instrument/instrument_arm32.c
index 0e15940a..16e8eaab 100644
--- a/frida_mode/src/instrument/instrument_arm32.c
+++ b/frida_mode/src/instrument/instrument_arm32.c
@@ -1,7 +1,5 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "instrument.h"
#include "util.h"
@@ -18,13 +16,27 @@ void instrument_coverage_optimize(const cs_insn * instr,
UNUSED_PARAMETER(instr);
UNUSED_PARAMETER(output);
- FATAL("Optimized coverage not supported on this architecture");
+ FFATAL("Optimized coverage not supported on this architecture");
+
+}
+
+void instrument_coverage_optimize_init(void) {
+
+ FWARNF("Optimized coverage not supported on this architecture");
}
void instrument_flush(GumStalkerOutput *output) {
- gum_arm_writer_flush(output->writer.arm);
+ if (output->encoding == GUM_INSTRUCTION_SPECIAL) {
+
+ gum_thumb_writer_flush(output->writer.thumb);
+
+ } else {
+
+ gum_arm_writer_flush(output->writer.arm);
+
+ }
}
diff --git a/frida_mode/src/instrument/instrument_arm64.c b/frida_mode/src/instrument/instrument_arm64.c
index cf37e048..0f635458 100644
--- a/frida_mode/src/instrument/instrument_arm64.c
+++ b/frida_mode/src/instrument/instrument_arm64.c
@@ -1,7 +1,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "instrument.h"
@@ -95,6 +94,10 @@ void instrument_coverage_optimize(const cs_insn * instr,
}
+void instrument_coverage_optimize_init(void) {
+
+}
+
void instrument_flush(GumStalkerOutput *output) {
gum_arm64_writer_flush(output->writer.arm64);
diff --git a/frida_mode/src/instrument/instrument_coverage.c b/frida_mode/src/instrument/instrument_coverage.c
index 46c816bc..c1984eb2 100644
--- a/frida_mode/src/instrument/instrument_coverage.c
+++ b/frida_mode/src/instrument/instrument_coverage.c
@@ -5,8 +5,6 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "instrument.h"
#include "util.h"
@@ -239,7 +237,7 @@ static void instrument_coverage_mark(void *key, void *value, void *user_data) {
}
-static void coverage_write(void *data, size_t size) {
+static void coverage_write(int fd, void *data, size_t size) {
ssize_t written;
size_t remain = size;
@@ -247,11 +245,11 @@ static void coverage_write(void *data, size_t size) {
for (char *cursor = (char *)data; remain > 0;
remain -= written, cursor += written) {
- written = write(normal_coverage_fd, cursor, remain);
+ written = write(fd, cursor, remain);
if (written < 0) {
- FATAL("Coverage - Failed to write: %s (%d)\n", (char *)data, errno);
+ FFATAL("Coverage - Failed to write: %s (%d)\n", (char *)data, errno);
}
@@ -259,7 +257,7 @@ static void coverage_write(void *data, size_t size) {
}
-static void coverage_format(char *format, ...) {
+static void coverage_format(int fd, char *format, ...) {
va_list ap;
char buffer[4096] = {0};
@@ -274,11 +272,11 @@ static void coverage_format(char *format, ...) {
len = strnlen(buffer, sizeof(buffer));
- coverage_write(buffer, len);
+ coverage_write(fd, buffer, len);
}
-static void coverage_write_modules(GArray *coverage_modules) {
+static void coverage_write_modules(int fd, GArray *coverage_modules) {
guint emitted = 0;
for (guint i = 0; i < coverage_modules->len; i++) {
@@ -287,16 +285,16 @@ static void coverage_write_modules(GArray *coverage_modules) {
&g_array_index(coverage_modules, coverage_range_t, i);
if (module->count == 0) continue;
- coverage_format("%3u, ", emitted);
- coverage_format("%016" G_GINT64_MODIFIER "X, ", module->base_address);
- coverage_format("%016" G_GINT64_MODIFIER "X, ", module->limit);
+ coverage_format(fd, "%3u, ", emitted);
+ coverage_format(fd, "%016" G_GINT64_MODIFIER "X, ", module->base_address);
+ coverage_format(fd, "%016" G_GINT64_MODIFIER "X, ", module->limit);
/* entry */
- coverage_format("%016" G_GINT64_MODIFIER "X, ", 0);
+ coverage_format(fd, "%016" G_GINT64_MODIFIER "X, ", 0);
/* checksum */
- coverage_format("%016" G_GINT64_MODIFIER "X, ", 0);
+ coverage_format(fd, "%016" G_GINT64_MODIFIER "X, ", 0);
/* timestamp */
- coverage_format("%08" G_GINT32_MODIFIER "X, ", 0);
- coverage_format("%s\n", module->path);
+ coverage_format(fd, "%08" G_GINT32_MODIFIER "X, ", 0);
+ coverage_format(fd, "%s\n", module->path);
emitted++;
}
@@ -306,7 +304,7 @@ static void coverage_write_modules(GArray *coverage_modules) {
static void coverage_write_events(void *key, void *value, void *user_data) {
UNUSED_PARAMETER(key);
- UNUSED_PARAMETER(user_data);
+ int fd = *((int *)user_data);
normal_coverage_data_t *val = (normal_coverage_data_t *)value;
if (val->module == NULL) { return; }
@@ -319,20 +317,20 @@ static void coverage_write_events(void *key, void *value, void *user_data) {
};
- coverage_write(&evt, sizeof(coverage_event_t));
+ coverage_write(fd, &evt, sizeof(coverage_event_t));
}
-static void coverage_write_header(guint coverage_marked_modules) {
+static void coverage_write_header(int fd, guint coverage_marked_modules) {
char version[] = "DRCOV VERSION: 2\n";
char flavour[] = "DRCOV FLAVOR: frida\n";
char columns[] = "Columns: id, base, end, entry, checksum, timestamp, path\n";
- coverage_write(version, sizeof(version) - 1);
- coverage_write(flavour, sizeof(flavour) - 1);
- coverage_format("Module Table: version 2, count %u\n",
+ coverage_write(fd, version, sizeof(version) - 1);
+ coverage_write(fd, flavour, sizeof(flavour) - 1);
+ coverage_format(fd, "Module Table: version 2, count %u\n",
coverage_marked_modules);
- coverage_write(columns, sizeof(columns) - 1);
+ coverage_write(fd, columns, sizeof(columns) - 1);
}
@@ -371,7 +369,7 @@ static void instrument_coverage_normal_run() {
if (close(normal_coverage_pipes[STDOUT_FILENO]) != 0) {
- FATAL("Failed to close parent read pipe");
+ FFATAL("Failed to close parent read pipe");
}
@@ -379,7 +377,7 @@ static void instrument_coverage_normal_run() {
g_hash_table_new_full(g_direct_hash, g_direct_equal, NULL, g_free);
if (coverage_hash == NULL) {
- FATAL("Failed to g_hash_table_new, errno: %d", errno);
+ FFATAL("Failed to g_hash_table_new, errno: %d", errno);
}
@@ -396,7 +394,7 @@ static void instrument_coverage_normal_run() {
}
- if (bytes != 0) { FATAL("Coverage data truncated"); }
+ if (bytes != 0) { FFATAL("Coverage data truncated"); }
instrument_coverage_print("Coverage - Preparing\n");
@@ -414,10 +412,11 @@ static void instrument_coverage_normal_run() {
instrument_coverage_print("Coverage - Marked Modules: %u\n",
coverage_marked_modules);
- coverage_write_header(coverage_marked_modules);
- coverage_write_modules(coverage_modules);
- coverage_format("BB Table: %u bbs\n", ctx.count);
- g_hash_table_foreach(coverage_hash, coverage_write_events, NULL);
+ coverage_write_header(normal_coverage_fd, coverage_marked_modules);
+ coverage_write_modules(normal_coverage_fd, coverage_modules);
+ coverage_format(normal_coverage_fd, "BB Table: %u bbs\n", ctx.count);
+ g_hash_table_foreach(coverage_hash, coverage_write_events,
+ &normal_coverage_fd);
g_hash_table_unref(coverage_hash);
@@ -435,7 +434,7 @@ static GArray *instrument_coverage_unstable_read_unstable_ids(void) {
if (!g_file_get_contents(unstable_coverage_fuzzer_stats, &contents, &length,
NULL)) {
- FATAL("Failed to read fuzzer_stats");
+ FFATAL("Failed to read fuzzer_stats");
}
@@ -526,7 +525,7 @@ static GHashTable *instrument_collect_unstable_blocks(
GHashTable *child =
(GHashTable *)g_hash_table_lookup(unstable_coverage_hash, *id);
- if (child == NULL) { FATAL("Failed to find edge ID"); }
+ if (child == NULL) { FFATAL("Failed to find edge ID"); }
GHashTableIter iter = {0};
gpointer value;
@@ -565,7 +564,7 @@ static void instrument_coverage_unstable_run(void) {
if (close(unstable_coverage_pipes[STDOUT_FILENO]) != 0) {
- FATAL("Failed to close parent read pipe");
+ FFATAL("Failed to close parent read pipe");
}
@@ -573,7 +572,7 @@ static void instrument_coverage_unstable_run(void) {
g_direct_hash, g_direct_equal, NULL, (GDestroyNotify)g_hash_table_unref);
if (unstable_coverage_hash == NULL) {
- FATAL("Failed to g_hash_table_new, errno: %d", errno);
+ FFATAL("Failed to g_hash_table_new, errno: %d", errno);
}
@@ -599,7 +598,7 @@ static void instrument_coverage_unstable_run(void) {
if (!g_hash_table_insert(unstable_coverage_hash,
GSIZE_TO_POINTER(value->edge), hash_value)) {
- FATAL("Entry already in hashtable");
+ FFATAL("Entry already in hashtable");
}
@@ -613,7 +612,7 @@ static void instrument_coverage_unstable_run(void) {
}
- if (bytes != 0) { FATAL("Unstable coverage data truncated"); }
+ if (bytes != 0) { FFATAL("Unstable coverage data truncated"); }
instrument_coverage_print("Coverage - Preparing\n");
@@ -638,10 +637,11 @@ static void instrument_coverage_unstable_run(void) {
instrument_coverage_print("Coverage - Marked Modules: %u\n",
coverage_marked_modules);
- coverage_write_header(coverage_marked_modules);
- coverage_write_modules(coverage_modules);
- coverage_format("BB Table: %u bbs\n", ctx.count);
- g_hash_table_foreach(unstable_blocks, coverage_write_events, NULL);
+ coverage_write_header(unstable_coverage_fd, coverage_marked_modules);
+ coverage_write_modules(unstable_coverage_fd, coverage_modules);
+ coverage_format(unstable_coverage_fd, "BB Table: %u bbs\n", ctx.count);
+ g_hash_table_foreach(unstable_blocks, coverage_write_events,
+ &unstable_coverage_fd);
g_hash_table_unref(unstable_blocks);
g_array_free(unstable_edge_ids, TRUE);
@@ -659,33 +659,33 @@ void instrument_coverage_config(void) {
void instrument_coverage_normal_init(void) {
- OKF("Coverage - enabled [%c]",
- instrument_coverage_filename == NULL ? ' ' : 'X');
+ FOKF("Coverage - enabled [%c]",
+ instrument_coverage_filename == NULL ? ' ' : 'X');
if (instrument_coverage_filename == NULL) { return; }
- OKF("Coverage - file [%s]", instrument_coverage_filename);
+ FOKF("Coverage - file [%s]", instrument_coverage_filename);
char *path = g_canonicalize_filename(instrument_coverage_filename,
g_get_current_dir());
- OKF("Coverage - path [%s]", path);
+ FOKF("Coverage - path [%s]", path);
normal_coverage_fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
if (normal_coverage_fd < 0) {
- FATAL("Failed to open coverage file '%s'", path);
+ FFATAL("Failed to open coverage file '%s'", path);
}
g_free(path);
- if (pipe(normal_coverage_pipes) != 0) { FATAL("Failed to create pipes"); }
+ if (pipe(normal_coverage_pipes) != 0) { FFATAL("Failed to create pipes"); }
pid_t pid = fork();
- if (pid == -1) { FATAL("Failed to start coverage process"); }
+ if (pid == -1) { FFATAL("Failed to start coverage process"); }
if (pid == 0) {
@@ -697,13 +697,13 @@ void instrument_coverage_normal_init(void) {
if (close(normal_coverage_fd) < 0) {
- FATAL("Failed to close coverage output file");
+ FFATAL("Failed to close coverage output file");
}
if (close(normal_coverage_pipes[STDIN_FILENO]) != 0) {
- FATAL("Failed to close parent read pipe");
+ FFATAL("Failed to close parent read pipe");
}
@@ -711,15 +711,14 @@ void instrument_coverage_normal_init(void) {
void instrument_coverage_unstable_find_output(void) {
- pid_t parent = getpid();
gchar *fds_name = g_strdup_printf("/proc/%d/fd/", getppid());
gchar *root = g_file_read_link("/proc/self/root", NULL);
- if (root == NULL) { FATAL("Failed to read link"); }
+ if (root == NULL) { FFATAL("Failed to read link"); }
GDir *dir = g_dir_open(fds_name, 0, NULL);
- OKF("Coverage Unstable - fds: %s", fds_name);
+ FOKF("Coverage Unstable - fds: %s", fds_name);
for (const gchar *filename = g_dir_read_name(dir); filename != NULL;
filename = g_dir_read_name(dir)) {
@@ -727,7 +726,7 @@ void instrument_coverage_unstable_find_output(void) {
gchar *fullname = g_build_path("/", fds_name, filename, NULL);
gchar *link = g_file_read_link(fullname, NULL);
- if (link == NULL) { FATAL("Failed to read link: %s", fullname); }
+ if (link == NULL) { FFATAL("Failed to read link: %s", fullname); }
gchar *basename = g_path_get_basename(link);
if (g_strcmp0(basename, "default") != 0) {
@@ -779,11 +778,11 @@ void instrument_coverage_unstable_find_output(void) {
if (unstable_coverage_fuzzer_stats == NULL) {
- FATAL("Failed to find fuzzer stats");
+ FFATAL("Failed to find fuzzer stats");
}
- OKF("Fuzzer stats: %s", unstable_coverage_fuzzer_stats);
+ FOKF("Fuzzer stats: %s", unstable_coverage_fuzzer_stats);
}
@@ -794,14 +793,14 @@ void instrument_coverage_unstable_init(void) {
char *path = g_canonicalize_filename(instrument_coverage_unstable_filename,
g_get_current_dir());
- OKF("Coverage - unstable path [%s]", instrument_coverage_unstable_filename);
+ FOKF("Coverage - unstable path [%s]", instrument_coverage_unstable_filename);
unstable_coverage_fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
if (unstable_coverage_fd < 0) {
- FATAL("Failed to open unstable coverage file '%s'", path);
+ FFATAL("Failed to open unstable coverage file '%s'", path);
}
@@ -811,12 +810,12 @@ void instrument_coverage_unstable_init(void) {
if (pipe(unstable_coverage_pipes) != 0) {
- FATAL("Failed to create unstable pipes");
+ FFATAL("Failed to create unstable pipes");
}
pid_t pid = fork();
- if (pid == -1) { FATAL("Failed to start coverage process"); }
+ if (pid == -1) { FFATAL("Failed to start coverage process"); }
if (pid == 0) {
@@ -828,13 +827,13 @@ void instrument_coverage_unstable_init(void) {
if (close(unstable_coverage_fd) < 0) {
- FATAL("Failed to close unstable coverage output file");
+ FFATAL("Failed to close unstable coverage output file");
}
if (close(unstable_coverage_pipes[STDIN_FILENO]) != 0) {
- FATAL("Failed to close parent read pipe");
+ FFATAL("Failed to close parent read pipe");
}
@@ -866,7 +865,7 @@ void instrument_coverage_end(uint64_t address) {
if (write(normal_coverage_pipes[STDOUT_FILENO], &data,
sizeof(normal_coverage_data_t)) != sizeof(normal_coverage_data_t)) {
- FATAL("Coverage I/O error");
+ FFATAL("Coverage I/O error");
}
@@ -889,7 +888,7 @@ void instrument_coverage_unstable(guint64 edge, guint64 previous_rip,
sizeof(unstable_coverage_data_t)) !=
sizeof(unstable_coverage_data_t)) {
- FATAL("Unstable coverage I/O error");
+ FFATAL("Unstable coverage I/O error");
}
diff --git a/frida_mode/src/instrument/instrument_debug.c b/frida_mode/src/instrument/instrument_debug.c
index b8cca634..9c95857f 100644
--- a/frida_mode/src/instrument/instrument_debug.c
+++ b/frida_mode/src/instrument/instrument_debug.c
@@ -5,8 +5,6 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "instrument.h"
#include "util.h"
@@ -34,18 +32,27 @@ static void instrument_debug(char *format, ...) {
}
-static void instrument_disasm(guint8 *start, guint8 *end) {
+static void instrument_disasm(guint8 *start, guint8 *end,
+ GumStalkerOutput *output) {
csh capstone;
cs_err err;
+ cs_mode mode;
uint16_t size;
cs_insn *insn;
size_t count = 0;
size_t i;
uint16_t len;
+ mode = GUM_DEFAULT_CS_MODE | GUM_DEFAULT_CS_ENDIAN;
+
+#if defined(__arm__)
+ if (output->encoding == GUM_INSTRUCTION_SPECIAL) { mode |= CS_MODE_THUMB; }
+#endif
+
err = cs_open(GUM_DEFAULT_CS_ARCH,
- GUM_DEFAULT_CS_MODE | GUM_DEFAULT_CS_ENDIAN, &capstone);
+ CS_MODE_THUMB | GUM_DEFAULT_CS_MODE | GUM_DEFAULT_CS_ENDIAN,
+ &capstone);
g_assert(err == CS_ERR_OK);
size = GPOINTER_TO_SIZE(end) - GPOINTER_TO_SIZE(start);
@@ -89,24 +96,24 @@ void instrument_debug_config(void) {
void instrument_debug_init(void) {
- OKF("Instrumentation debugging - enabled [%c]",
- instrument_debug_filename == NULL ? ' ' : 'X');
+ FOKF("Instrumentation debugging - enabled [%c]",
+ instrument_debug_filename == NULL ? ' ' : 'X');
if (instrument_debug_filename == NULL) { return; }
- OKF("Instrumentation debugging - file [%s]", instrument_debug_filename);
+ FOKF("Instrumentation debugging - file [%s]", instrument_debug_filename);
if (instrument_debug_filename == NULL) { return; }
char *path =
g_canonicalize_filename(instrument_debug_filename, g_get_current_dir());
- OKF("Instrumentation debugging - path [%s]", path);
+ FOKF("Instrumentation debugging - path [%s]", path);
debugging_fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
- if (debugging_fd < 0) { FATAL("Failed to open stats file '%s'", path); }
+ if (debugging_fd < 0) { FFATAL("Failed to open stats file '%s'", path); }
g_free(path);
@@ -123,11 +130,12 @@ void instrument_debug_start(uint64_t address, GumStalkerOutput *output) {
}
-void instrument_debug_instruction(uint64_t address, uint16_t size) {
+void instrument_debug_instruction(uint64_t address, uint16_t size,
+ GumStalkerOutput *output) {
if (likely(debugging_fd < 0)) { return; }
uint8_t *start = (uint8_t *)GSIZE_TO_POINTER(address);
- instrument_disasm(start, start + size);
+ instrument_disasm(start, start + size, output);
}
@@ -138,7 +146,7 @@ void instrument_debug_end(GumStalkerOutput *output) {
instrument_debug("\nGenerated block %p-%p\n", instrument_gen_start,
instrument_gen_end);
- instrument_disasm(instrument_gen_start, instrument_gen_end);
+ instrument_disasm(instrument_gen_start, instrument_gen_end, output);
}
diff --git a/frida_mode/src/instrument/instrument_x64.c b/frida_mode/src/instrument/instrument_x64.c
index fec8afbb..41162f2a 100644
--- a/frida_mode/src/instrument/instrument_x64.c
+++ b/frida_mode/src/instrument/instrument_x64.c
@@ -1,85 +1,343 @@
+#include <fcntl.h>
+#include <stddef.h>
+#include <sys/mman.h>
+#include <sys/shm.h>
+
+#if defined(__linux__)
+ #if !defined(__ANDROID__)
+ #include <sys/prctl.h>
+ #include <sys/syscall.h>
+ #else
+ #include <linux/ashmem.h>
+ #endif
+#endif
+
#include "frida-gumjs.h"
#include "config.h"
#include "instrument.h"
+#include "ranges.h"
+#include "stalker.h"
+#include "util.h"
#if defined(__x86_64__)
-static GumAddress current_log_impl = GUM_ADDRESS(0);
+ #ifndef MAP_FIXED_NOREPLACE
+ #ifdef MAP_EXCL
+ #define MAP_FIXED_NOREPLACE MAP_EXCL | MAP_FIXED
+ #else
+ #define MAP_FIXED_NOREPLACE MAP_FIXED
+ #endif
+ #endif
-static const guint8 afl_log_code[] = {
+static GHashTable *coverage_blocks = NULL;
- 0x9c, /* pushfq */
- 0x51, /* push rcx */
- 0x52, /* push rdx */
+gboolean instrument_is_coverage_optimize_supported(void) {
- 0x48, 0x8b, 0x0d, 0x26,
- 0x00, 0x00, 0x00, /* mov rcx, sym.&previous_pc */
- 0x48, 0x8b, 0x11, /* mov rdx, qword [rcx] */
- 0x48, 0x31, 0xfa, /* xor rdx, rdi */
+ return true;
- 0x48, 0x03, 0x15, 0x11,
- 0x00, 0x00, 0x00, /* add rdx, sym._afl_area_ptr_ptr */
+}
- 0x80, 0x02, 0x01, /* add byte ptr [rdx], 1 */
- 0x80, 0x12, 0x00, /* adc byte ptr [rdx], 0 */
- 0x66, 0xd1, 0xcf, /* ror di, 1 */
- 0x48, 0x89, 0x39, /* mov qword [rcx], rdi */
+static gboolean instrument_coverage_in_range(gssize offset) {
- 0x5a, /* pop rdx */
- 0x59, /* pop rcx */
- 0x9d, /* popfq */
+ return (offset >= G_MININT32 && offset <= G_MAXINT32);
- 0xc3, /* ret */
+}
- 0x90
+ #pragma pack(push, 1)
+typedef struct {
- /* Read-only data goes here: */
- /* uint8_t* __afl_area_ptr */
- /* uint64_t* &previous_pc */
+ // cur_location = (block_address >> 4) ^ (block_address << 8);
+ // shared_mem[cur_location ^ prev_location]++;
+ // prev_location = cur_location >> 1;
-};
+ // mov QWORD PTR [rsp-0x80],rax
+ // lahf
+ // mov QWORD PTR [rsp-0x88],rax
+ // mov QWORD PTR [rsp-0x90],rbx
+ // mov eax,DWORD PTR [rip+0x333d5a] # 0x7ffff6ff2740
+ // mov DWORD PTR [rip+0x333d3c],0x9fbb # 0x7ffff6ff2740
+ // xor eax,0x103f77
+ // mov bl,BYTE PTR [rax]
+ // add bl,0x1
+ // adc bl,0x0
+ // mov BYTE PTR [rax],bl
+ // mov rbx,QWORD PTR [rsp-0x90]
+ // mov rax,QWORD PTR [rsp-0x88]
+ // sahf
+ // mov rax,QWORD PTR [rsp-0x80]
-gboolean instrument_is_coverage_optimize_supported(void) {
+ uint8_t mov_rax_rsp_88[8];
+ uint8_t lahf;
+ uint8_t mov_rax_rsp_90[8];
+ uint8_t mov_rbx_rsp_98[8];
- return true;
+ uint8_t mov_eax_prev_loc[6];
+ uint8_t mov_prev_loc_curr_loc_shr1[10];
+
+ uint8_t xor_eax_curr_loc[5];
+
+ uint8_t mov_rbx_ptr_rax[2];
+ uint8_t add_bl_1[3];
+ uint8_t adc_bl_0[3];
+ uint8_t mov_ptr_rax_rbx[2];
+
+ uint8_t mov_rsp_98_rbx[8];
+ uint8_t mov_rsp_90_rax[8];
+ uint8_t sahf;
+ uint8_t mov_rsp_88_rax[8];
+
+} afl_log_code_asm_t;
+
+ #pragma pack(pop)
+
+typedef union {
+
+ afl_log_code_asm_t code;
+ uint8_t bytes[0];
+
+} afl_log_code;
+
+static const afl_log_code_asm_t template =
+ {
+
+ .mov_rax_rsp_88 = {0x48, 0x89, 0x84, 0x24, 0x78, 0xFF, 0xFF, 0xFF},
+ .lahf = 0x9f,
+ .mov_rax_rsp_90 = {0x48, 0x89, 0x84, 0x24, 0x70, 0xFF, 0xFF, 0xFF},
+ .mov_rbx_rsp_98 = {0x48, 0x89, 0x9C, 0x24, 0x68, 0xFF, 0xFF, 0xFF},
+
+ .mov_eax_prev_loc = {0x8b, 0x05},
+ .mov_prev_loc_curr_loc_shr1 = {0xc7, 0x05},
+
+ .xor_eax_curr_loc = {0x35},
+ .mov_rbx_ptr_rax = {0x8a, 0x18},
+ .add_bl_1 = {0x80, 0xc3, 0x01},
+ .adc_bl_0 = {0x80, 0xd3, 0x00},
+ .mov_ptr_rax_rbx = {0x88, 0x18},
+
+ .mov_rsp_98_rbx = {0x48, 0x8B, 0x9C, 0x24, 0x68, 0xFF, 0xFF, 0xFF},
+ .mov_rsp_90_rax = {0x48, 0x8B, 0x84, 0x24, 0x70, 0xFF, 0xFF, 0xFF},
+ .sahf = 0x9e,
+ .mov_rsp_88_rax = {0x48, 0x8B, 0x84, 0x24, 0x78, 0xFF, 0xFF, 0xFF},
}
-static guint8 align_pad[] = {0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90, 0x90};
+;
-static void instrument_coverate_write_function(GumStalkerOutput *output) {
+static gboolean instrument_coverage_find_low(const GumRangeDetails *details,
+ gpointer user_data) {
- guint64 misalign = 0;
- GumX86Writer *cw = output->writer.x86;
+ static GumAddress last_limit = (64ULL << 10);
+ gpointer * address = (gpointer *)user_data;
+
+ if ((details->range->base_address - last_limit) > __afl_map_size) {
+
+ *address = GSIZE_TO_POINTER(last_limit);
+ return FALSE;
+
+ }
+
+ if (details->range->base_address > ((2ULL << 30) - __afl_map_size)) {
+
+ return FALSE;
+
+ }
+
+ /*
+ * Align our buffer on a 64k boundary so that the low 16-bits of the address
+ * are zero, then we can just XOR the base address in, when we XOR with the
+ * current block ID.
+ */
+ last_limit = GUM_ALIGN_SIZE(
+ details->range->base_address + details->range->size, (64ULL << 10));
+ return TRUE;
+
+}
+
+static void instrument_coverage_optimize_map_mmap_anon(gpointer address) {
+
+ __afl_area_ptr =
+ mmap(address, __afl_map_size, PROT_READ | PROT_WRITE,
+ MAP_FIXED_NOREPLACE | MAP_SHARED | MAP_ANONYMOUS, -1, 0);
+ if (__afl_area_ptr != address) {
+
+ FATAL("Failed to map mmap __afl_area_ptr: %d", errno);
+
+ }
+
+}
+
+static void instrument_coverage_optimize_map_mmap(char * shm_file_path,
+ gpointer address) {
+
+ int shm_fd = -1;
+
+ if (munmap(__afl_area_ptr, __afl_map_size) != 0) {
+
+ FATAL("Failed to unmap previous __afl_area_ptr");
+
+ }
+
+ __afl_area_ptr = NULL;
+
+ #if !defined(__ANDROID__)
+ shm_fd = shm_open(shm_file_path, O_RDWR, DEFAULT_PERMISSION);
+ if (shm_fd == -1) { FATAL("shm_open() failed\n"); }
+ #else
+ shm_fd = open("/dev/ashmem", O_RDWR);
+ if (shm_fd == -1) { FATAL("open() failed\n"); }
+ if (ioctl(shm_fd, ASHMEM_SET_NAME, shm_file_path) == -1) {
+
+ FATAL("ioctl(ASHMEM_SET_NAME) failed");
+
+ }
+
+ if (ioctl(shm_fd, ASHMEM_SET_SIZE, __afl_map_size) == -1) {
+
+ FATAL("ioctl(ASHMEM_SET_SIZE) failed");
+
+ }
+
+ #endif
+
+ __afl_area_ptr = mmap(address, __afl_map_size, PROT_READ | PROT_WRITE,
+ MAP_FIXED_NOREPLACE | MAP_SHARED, shm_fd, 0);
+ if (__afl_area_ptr != address) {
+
+ FATAL("Failed to map mmap __afl_area_ptr: %d", errno);
+
+ }
+
+ if (close(shm_fd) != 0) { FATAL("Failed to close shm_fd"); }
+
+}
+
+static void instrument_coverage_optimize_map_shm(guint64 shm_env_val,
+ gpointer address) {
+
+ if (shmdt(__afl_area_ptr) != 0) {
+
+ FATAL("Failed to detach previous __afl_area_ptr");
+
+ }
- if (current_log_impl == 0 ||
- !gum_x86_writer_can_branch_directly_between(cw->pc, current_log_impl) ||
- !gum_x86_writer_can_branch_directly_between(cw->pc + 128,
- current_log_impl)) {
+ __afl_area_ptr = shmat(shm_env_val, address, 0);
+ if (__afl_area_ptr != address) {
- gconstpointer after_log_impl = cw->code + 1;
+ FATAL("Failed to map shm __afl_area_ptr: %d", errno);
- gum_x86_writer_put_jmp_near_label(cw, after_log_impl);
+ }
+
+}
+
+static void instrument_coverage_switch(GumStalkerObserver *self,
+ gpointer start_address,
+ const cs_insn * from_insn,
+ gpointer * target) {
+
+ UNUSED_PARAMETER(self);
+ UNUSED_PARAMETER(start_address);
+
+ cs_x86 * x86;
+ cs_x86_op *op;
+ if (from_insn == NULL) { return; }
+
+ x86 = &from_insn->detail->x86;
+ op = x86->operands;
+
+ if (!g_hash_table_contains(coverage_blocks, GSIZE_TO_POINTER(*target))) {
+
+ return;
+
+ }
+
+ switch (from_insn->id) {
+
+ case X86_INS_CALL:
+ case X86_INS_JMP:
+ if (x86->op_count != 1) {
+
+ FATAL("Unexpected operand count: %d", x86->op_count);
+
+ }
+
+ if (op[0].type != X86_OP_IMM) { return; }
+
+ break;
+ case X86_INS_RET:
+ break;
+ default:
+ return;
+
+ }
+
+ *target = (guint8 *)*target + sizeof(afl_log_code);
+
+}
+
+void instrument_coverage_optimize_init(void) {
+
+ gpointer low_address = NULL;
+
+ gum_process_enumerate_ranges(GUM_PAGE_NO_ACCESS, instrument_coverage_find_low,
+ &low_address);
+
+ FOKF("Low address: %p", low_address);
+
+ if (low_address == 0 ||
+ GPOINTER_TO_SIZE(low_address) > ((2UL << 20) - __afl_map_size)) {
+
+ FATAL("Invalid low_address: %p", low_address);
+
+ }
+
+ ranges_print_debug_maps();
+
+ char *shm_env = getenv(SHM_ENV_VAR);
+ FOKF("SHM_ENV_VAR: %s", shm_env);
+
+ if (shm_env == NULL) {
+
+ FWARNF("SHM_ENV_VAR not set, using anonymous map for debugging purposes");
- misalign = (cw->pc & 0x7);
- if (misalign != 0) {
+ instrument_coverage_optimize_map_mmap_anon(low_address);
- gum_x86_writer_put_bytes(cw, align_pad, 8 - misalign);
+ } else {
+
+ guint64 shm_env_val = g_ascii_strtoull(shm_env, NULL, 10);
+
+ if (shm_env_val == 0) {
+
+ instrument_coverage_optimize_map_mmap(shm_env, low_address);
+
+ } else {
+
+ instrument_coverage_optimize_map_shm(shm_env_val, low_address);
}
- current_log_impl = cw->pc;
- gum_x86_writer_put_bytes(cw, afl_log_code, sizeof(afl_log_code));
+ }
+
+ FOKF("__afl_area_ptr: %p", __afl_area_ptr);
+ FOKF("instrument_previous_pc: %p", &instrument_previous_pc);
+
+}
- uint64_t *afl_prev_loc_ptr = &instrument_previous_pc;
- gum_x86_writer_put_bytes(cw, (const guint8 *)&__afl_area_ptr,
- sizeof(__afl_area_ptr));
- gum_x86_writer_put_bytes(cw, (const guint8 *)&afl_prev_loc_ptr,
- sizeof(afl_prev_loc_ptr));
+static void instrument_coverage_suppress_init(void) {
- gum_x86_writer_put_label(cw, after_log_impl);
+ static gboolean initialized = false;
+ if (initialized) { return; }
+ initialized = true;
+
+ GumStalkerObserver * observer = stalker_get_observer();
+ GumStalkerObserverInterface *iface = GUM_STALKER_OBSERVER_GET_IFACE(observer);
+ iface->switch_callback = instrument_coverage_switch;
+
+ coverage_blocks = g_hash_table_new(g_direct_hash, g_direct_equal);
+ if (coverage_blocks == NULL) {
+
+ FATAL("Failed to g_hash_table_new, errno: %d", errno);
}
@@ -88,18 +346,73 @@ static void instrument_coverate_write_function(GumStalkerOutput *output) {
void instrument_coverage_optimize(const cs_insn * instr,
GumStalkerOutput *output) {
+ afl_log_code code = {0};
GumX86Writer *cw = output->writer.x86;
guint64 area_offset = instrument_get_offset_hash(GUM_ADDRESS(instr->address));
- instrument_coverate_write_function(output);
-
- gum_x86_writer_put_lea_reg_reg_offset(cw, GUM_REG_RSP, GUM_REG_RSP,
- -GUM_RED_ZONE_SIZE);
- gum_x86_writer_put_push_reg(cw, GUM_REG_RDI);
- gum_x86_writer_put_mov_reg_address(cw, GUM_REG_RDI, area_offset);
- gum_x86_writer_put_call_address(cw, current_log_impl);
- gum_x86_writer_put_pop_reg(cw, GUM_REG_RDI);
- gum_x86_writer_put_lea_reg_reg_offset(cw, GUM_REG_RSP, GUM_REG_RSP,
- GUM_RED_ZONE_SIZE);
+ gsize map_size_pow2;
+ gsize area_offset_ror;
+ GumAddress code_addr = 0;
+
+ instrument_coverage_suppress_init();
+
+ // gum_x86_writer_put_breakpoint(cw);
+ code_addr = cw->pc;
+ if (!g_hash_table_add(coverage_blocks, GSIZE_TO_POINTER(cw->code))) {
+
+ FATAL("Failed - g_hash_table_add");
+
+ }
+
+ code.code = template;
+
+ gssize curr_loc_shr_1_offset =
+ offsetof(afl_log_code, code.mov_prev_loc_curr_loc_shr1) +
+ sizeof(code.code.mov_prev_loc_curr_loc_shr1) - sizeof(guint32);
+
+ map_size_pow2 = util_log2(__afl_map_size);
+ area_offset_ror = util_rotate(area_offset, 1, map_size_pow2);
+
+ *((guint32 *)&code.bytes[curr_loc_shr_1_offset]) = (guint32)(area_offset_ror);
+
+ gssize prev_loc_value =
+ GPOINTER_TO_SIZE(&instrument_previous_pc) -
+ (code_addr + offsetof(afl_log_code, code.mov_prev_loc_curr_loc_shr1) +
+ sizeof(code.code.mov_prev_loc_curr_loc_shr1));
+ gssize prev_loc_value_offset =
+ offsetof(afl_log_code, code.mov_prev_loc_curr_loc_shr1) +
+ sizeof(code.code.mov_prev_loc_curr_loc_shr1) - sizeof(gint) -
+ sizeof(guint32);
+ if (!instrument_coverage_in_range(prev_loc_value)) {
+
+ FATAL("Patch out of range (current_pc_value1): 0x%016lX", prev_loc_value);
+
+ }
+
+ *((gint *)&code.bytes[prev_loc_value_offset]) = (gint)prev_loc_value;
+
+ gssize prev_loc_value2 =
+ GPOINTER_TO_SIZE(&instrument_previous_pc) -
+ (code_addr + offsetof(afl_log_code, code.mov_eax_prev_loc) +
+ sizeof(code.code.mov_eax_prev_loc));
+ gssize prev_loc_value_offset2 =
+ offsetof(afl_log_code, code.mov_eax_prev_loc) +
+ sizeof(code.code.mov_eax_prev_loc) - sizeof(gint);
+ if (!instrument_coverage_in_range(prev_loc_value)) {
+
+ FATAL("Patch out of range (current_pc_value1): 0x%016lX", prev_loc_value2);
+
+ }
+
+ *((gint *)&code.bytes[prev_loc_value_offset2]) = (gint)prev_loc_value2;
+
+ gssize xor_curr_loc_offset = offsetof(afl_log_code, code.xor_eax_curr_loc) +
+ sizeof(code.code.xor_eax_curr_loc) -
+ sizeof(guint32);
+
+ *((guint32 *)&code.bytes[xor_curr_loc_offset]) =
+ (guint32)(GPOINTER_TO_SIZE(__afl_area_ptr) | area_offset);
+
+ gum_x86_writer_put_bytes(cw, code.bytes, sizeof(afl_log_code));
}
diff --git a/frida_mode/src/instrument/instrument_x86.c b/frida_mode/src/instrument/instrument_x86.c
index 7bf48f96..ad837e2d 100644
--- a/frida_mode/src/instrument/instrument_x86.c
+++ b/frida_mode/src/instrument/instrument_x86.c
@@ -1,69 +1,144 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "instrument.h"
+#include "stalker.h"
#include "util.h"
#if defined(__i386__)
-static GumAddress current_log_impl = GUM_ADDRESS(0);
+static GHashTable *coverage_blocks = NULL;
+
+ #pragma pack(push, 1)
+typedef struct {
+
+ // cur_location = (block_address >> 4) ^ (block_address << 8);
+ // shared_mem[cur_location ^ prev_location]++;
+ // prev_location = cur_location >> 1;
+
+ uint8_t mov_eax_esp_4[4];
+ uint8_t lahf;
+ uint8_t mov_eax_esp_8[4];
+ uint8_t mov_ebx_esp_c[4];
+
+ uint8_t mov_eax_prev_loc[5];
+ uint8_t mov_prev_loc_curr_loc_shr1[10];
+
+ uint8_t xor_eax_curr_loc[5];
+ uint8_t add_eax_area_ptr[5];
+
+ uint8_t mov_ebx_ptr_eax[2];
+ uint8_t add_bl_1[3];
+ uint8_t adc_bl_0[3];
+ uint8_t mov_ptr_eax_ebx[2];
-static void instrument_coverage_function(GumX86Writer *cw) {
+ uint8_t mov_esp_c_ebx[4];
+ uint8_t mov_esp_8_eax[4];
+ uint8_t sahf;
+ uint8_t mov_esp_4_eax[4];
- gum_x86_writer_put_pushfx(cw);
- gum_x86_writer_put_push_reg(cw, GUM_REG_ECX);
- gum_x86_writer_put_push_reg(cw, GUM_REG_EDX);
+} afl_log_code_asm_t;
- gum_x86_writer_put_mov_reg_address(cw, GUM_REG_ECX,
- GUM_ADDRESS(&instrument_previous_pc));
- gum_x86_writer_put_mov_reg_reg_ptr(cw, GUM_REG_EDX, GUM_REG_ECX);
- gum_x86_writer_put_xor_reg_reg(cw, GUM_REG_EDX, GUM_REG_EDI);
+ #pragma pack(pop)
- gum_x86_writer_put_add_reg_imm(cw, GUM_REG_EDX, GUM_ADDRESS(__afl_area_ptr));
+typedef union {
- /* add byte ptr [edx], 1 */
- uint8_t add_byte_ptr_edx_1[] = {0x80, 0x02, 0x01};
- gum_x86_writer_put_bytes(cw, add_byte_ptr_edx_1, sizeof(add_byte_ptr_edx_1));
+ afl_log_code_asm_t code;
+ uint8_t bytes[0];
- /* adc byte ptr [edx], 0 */
- uint8_t adc_byte_ptr_edx_0[] = {0x80, 0x12, 0x00};
- gum_x86_writer_put_bytes(cw, adc_byte_ptr_edx_0, sizeof(adc_byte_ptr_edx_0));
+} afl_log_code;
- uint8_t ror_di_1[] = {0x66, 0xd1, 0xcf};
- gum_x86_writer_put_bytes(cw, ror_di_1, sizeof(ror_di_1));
- gum_x86_writer_put_mov_reg_ptr_reg(cw, GUM_REG_ECX, GUM_REG_EDI);
+static const afl_log_code_asm_t template =
+ {
- gum_x86_writer_put_pop_reg(cw, GUM_REG_EDX);
- gum_x86_writer_put_pop_reg(cw, GUM_REG_ECX);
- gum_x86_writer_put_popfx(cw);
- gum_x86_writer_put_ret(cw);
+ .mov_eax_esp_4 = {0x89, 0x44, 0x24, 0xFC},
+ .lahf = 0x9f,
+ .mov_eax_esp_8 = {0x89, 0x44, 0x24, 0xF8},
+ .mov_ebx_esp_c = {0x89, 0x5C, 0x24, 0xF4},
+
+ .mov_eax_prev_loc = {0xA1},
+ .mov_prev_loc_curr_loc_shr1 = {0xc7, 0x05},
+
+ .xor_eax_curr_loc = {0x35},
+ .add_eax_area_ptr = {0x05},
+ .mov_ebx_ptr_eax = {0x8a, 0x18},
+ .add_bl_1 = {0x80, 0xc3, 0x01},
+ .adc_bl_0 = {0x80, 0xd3, 0x00},
+ .mov_ptr_eax_ebx = {0x88, 0x18},
+
+ .mov_esp_c_ebx = {0x8B, 0x5C, 0x24, 0xF4},
+ .mov_esp_8_eax = {0x8B, 0x44, 0x24, 0xF8},
+ .sahf = 0x9e,
+ .mov_esp_4_eax = {0x8B, 0x44, 0x24, 0xFC},
}
+;
+
gboolean instrument_is_coverage_optimize_supported(void) {
return true;
}
-static void instrument_coverate_write_function(GumStalkerOutput *output) {
+static void instrument_coverage_switch(GumStalkerObserver *self,
+ gpointer start_address,
+ const cs_insn * from_insn,
+ gpointer * target) {
- GumX86Writer *cw = output->writer.x86;
+ UNUSED_PARAMETER(self);
+ UNUSED_PARAMETER(start_address);
+
+ cs_x86 * x86;
+ cs_x86_op *op;
+ if (from_insn == NULL) { return; }
+
+ x86 = &from_insn->detail->x86;
+ op = x86->operands;
+
+ if (!g_hash_table_contains(coverage_blocks, GSIZE_TO_POINTER(*target))) {
+
+ return;
+
+ }
+
+ switch (from_insn->id) {
+
+ case X86_INS_CALL:
+ case X86_INS_JMP:
+ if (x86->op_count != 1) {
+
+ FATAL("Unexpected operand count: %d", x86->op_count);
+
+ }
+
+ if (op[0].type != X86_OP_IMM) { return; }
+
+ break;
+ case X86_INS_RET:
+ break;
+ default:
+ return;
+
+ }
+
+ *target = (guint8 *)*target + sizeof(afl_log_code);
+
+}
- if (current_log_impl == 0 ||
- !gum_x86_writer_can_branch_directly_between(cw->pc, current_log_impl) ||
- !gum_x86_writer_can_branch_directly_between(cw->pc + 128,
- current_log_impl)) {
+static void instrument_coverage_suppress_init(void) {
- gconstpointer after_log_impl = cw->code + 1;
+ static gboolean initialized = false;
+ if (initialized) { return; }
+ initialized = true;
- gum_x86_writer_put_jmp_near_label(cw, after_log_impl);
+ GumStalkerObserver * observer = stalker_get_observer();
+ GumStalkerObserverInterface *iface = GUM_STALKER_OBSERVER_GET_IFACE(observer);
+ iface->switch_callback = instrument_coverage_switch;
- current_log_impl = cw->pc;
- instrument_coverage_function(cw);
+ coverage_blocks = g_hash_table_new(g_direct_hash, g_direct_equal);
+ if (coverage_blocks == NULL) {
- gum_x86_writer_put_label(cw, after_log_impl);
+ FATAL("Failed to g_hash_table_new, errno: %d", errno);
}
@@ -72,14 +147,65 @@ static void instrument_coverate_write_function(GumStalkerOutput *output) {
void instrument_coverage_optimize(const cs_insn * instr,
GumStalkerOutput *output) {
+ afl_log_code code = {0};
GumX86Writer *cw = output->writer.x86;
guint64 area_offset = instrument_get_offset_hash(GUM_ADDRESS(instr->address));
- instrument_coverate_write_function(output);
+ gsize map_size_pow2;
+ gsize area_offset_ror;
+
+ code.code = template;
+
+ instrument_coverage_suppress_init();
+
+ // gum_x86_writer_put_breakpoint(cw);
+
+ if (!g_hash_table_add(coverage_blocks, GSIZE_TO_POINTER(cw->code))) {
+
+ FATAL("Failed - g_hash_table_add");
+
+ }
+
+ gssize prev_loc_value_offset2 =
+ offsetof(afl_log_code, code.mov_eax_prev_loc) +
+ sizeof(code.code.mov_eax_prev_loc) - sizeof(gint);
+
+ *((gint *)&code.bytes[prev_loc_value_offset2]) =
+ (gint)GPOINTER_TO_SIZE(&instrument_previous_pc);
+
+ gssize curr_loc_shr_1_offset =
+ offsetof(afl_log_code, code.mov_prev_loc_curr_loc_shr1) +
+ sizeof(code.code.mov_prev_loc_curr_loc_shr1) - sizeof(guint32);
+
+ map_size_pow2 = util_log2(__afl_map_size);
+ area_offset_ror = util_rotate(area_offset, 1, map_size_pow2);
+
+ *((guint32 *)&code.bytes[curr_loc_shr_1_offset]) = (guint32)(area_offset_ror);
+
+ gssize prev_loc_value_offset =
+ offsetof(afl_log_code, code.mov_prev_loc_curr_loc_shr1) +
+ sizeof(code.code.mov_prev_loc_curr_loc_shr1) - sizeof(gint) -
+ sizeof(guint32);
+
+ *((gint *)&code.bytes[prev_loc_value_offset]) =
+ (gint)GPOINTER_TO_SIZE(&instrument_previous_pc);
+
+ gssize xor_curr_loc_offset = offsetof(afl_log_code, code.xor_eax_curr_loc) +
+ sizeof(code.code.xor_eax_curr_loc) -
+ sizeof(guint32);
+
+ *((guint32 *)&code.bytes[xor_curr_loc_offset]) = (guint32)area_offset;
+
+ gssize add_area_ptr_offset = offsetof(afl_log_code, code.add_eax_area_ptr) +
+ sizeof(code.code.add_eax_area_ptr) -
+ sizeof(guint32);
+
+ *((guint32 *)&code.bytes[add_area_ptr_offset]) = (guint32)__afl_area_ptr;
+
+ gum_x86_writer_put_bytes(cw, code.bytes, sizeof(afl_log_code));
+
+}
- gum_x86_writer_put_push_reg(cw, GUM_REG_EDI);
- gum_x86_writer_put_mov_reg_address(cw, GUM_REG_EDI, area_offset);
- gum_x86_writer_put_call_address(cw, current_log_impl);
- gum_x86_writer_put_pop_reg(cw, GUM_REG_EDI);
+void instrument_coverage_optimize_init(void) {
}
diff --git a/frida_mode/src/intercept.c b/frida_mode/src/intercept.c
index ed8d27bd..26d20c50 100644
--- a/frida_mode/src/intercept.c
+++ b/frida_mode/src/intercept.c
@@ -1,8 +1,7 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "intercept.h"
+#include "util.h"
void intercept_hook(void *address, gpointer replacement, gpointer user_data) {
@@ -10,7 +9,7 @@ void intercept_hook(void *address, gpointer replacement, gpointer user_data) {
gum_interceptor_begin_transaction(interceptor);
GumReplaceReturn ret =
gum_interceptor_replace(interceptor, address, replacement, user_data);
- if (ret != GUM_REPLACE_OK) { FATAL("gum_interceptor_attach: %d", ret); }
+ if (ret != GUM_REPLACE_OK) { FFATAL("gum_interceptor_attach: %d", ret); }
gum_interceptor_end_transaction(interceptor);
}
diff --git a/frida_mode/src/js/api.js b/frida_mode/src/js/api.js
index 40bb4a16..215fbdaf 100644
--- a/frida_mode/src/js/api.js
+++ b/frida_mode/src/js/api.js
@@ -63,6 +63,12 @@ class Afl {
Afl.jsApiWrite(STDOUT_FILENO, buf, log.length);
}
/**
+ * See `AFL_FRIDA_INST_NO_BACKPATCH`.
+ */
+ static setBackpatchDisable() {
+ Afl.jsApiSetBackpatchDisable();
+ }
+ /**
* See `AFL_FRIDA_DEBUG_MAPS`.
*/
static setDebugMaps() {
@@ -145,6 +151,13 @@ class Afl {
const buf = Memory.allocUtf8String(file);
Afl.jsApiSetInstrumentUnstableCoverageFile(buf);
}
+ /*
+ * Set a callback to be called in place of the usual `main` function. This see
+ * `Scripting.md` for details.
+ */
+ static setJsMainHook(address) {
+ Afl.jsApiSetJsMainHook(address);
+ }
/**
* This is equivalent to setting `AFL_FRIDA_PERSISTENT_ADDR`, again a
* `NativePointer` should be provided as it's argument.
@@ -199,6 +212,12 @@ class Afl {
const buf = Memory.allocUtf8String(file);
Afl.jsApiSetSeccompFile(buf);
}
+ /**
+ * See `AFL_FRIDA_STALKER_ADJACENT_BLOCKS`.
+ */
+ static setStalkerAdjacentBlocks(val) {
+ Afl.jsApiSetStalkerAdjacentBlocks(val);
+ }
/*
* Set a function to be called for each instruction which is instrumented
* by AFL FRIDA mode.
@@ -243,6 +262,12 @@ class Afl {
const buf = Memory.allocUtf8String(file);
Afl.jsApiSetStdOut(buf);
}
+ /**
+ * See `AFL_FRIDA_TRACEABLE`.
+ */
+ static setTraceable() {
+ Afl.jsApiSetTraceable();
+ }
static jsApiGetFunction(name, retType, argTypes) {
const addr = Afl.module.getExportByName(name);
return new NativeFunction(addr, retType, argTypes);
@@ -261,6 +286,7 @@ Afl.jsApiAddIncludeRange = Afl.jsApiGetFunction("js_api_add_include_range", "voi
Afl.jsApiAflSharedMemFuzzing = Afl.jsApiGetSymbol("__afl_sharedmem_fuzzing");
Afl.jsApiDone = Afl.jsApiGetFunction("js_api_done", "void", []);
Afl.jsApiError = Afl.jsApiGetFunction("js_api_error", "void", ["pointer"]);
+Afl.jsApiSetBackpatchDisable = Afl.jsApiGetFunction("js_api_set_backpatch_disable", "void", []);
Afl.jsApiSetDebugMaps = Afl.jsApiGetFunction("js_api_set_debug_maps", "void", []);
Afl.jsApiSetEntryPoint = Afl.jsApiGetFunction("js_api_set_entrypoint", "void", ["pointer"]);
Afl.jsApiSetInstrumentCoverageFile = Afl.jsApiGetFunction("js_api_set_instrument_coverage_file", "void", ["pointer"]);
@@ -272,6 +298,7 @@ Afl.jsApiSetInstrumentSeed = Afl.jsApiGetFunction("js_api_set_instrument_seed",
Afl.jsApiSetInstrumentTrace = Afl.jsApiGetFunction("js_api_set_instrument_trace", "void", []);
Afl.jsApiSetInstrumentTraceUnique = Afl.jsApiGetFunction("js_api_set_instrument_trace_unique", "void", []);
Afl.jsApiSetInstrumentUnstableCoverageFile = Afl.jsApiGetFunction("js_api_set_instrument_unstable_coverage_file", "void", ["pointer"]);
+Afl.jsApiSetJsMainHook = Afl.jsApiGetFunction("js_api_set_js_main_hook", "void", ["pointer"]);
Afl.jsApiSetPersistentAddress = Afl.jsApiGetFunction("js_api_set_persistent_address", "void", ["pointer"]);
Afl.jsApiSetPersistentCount = Afl.jsApiGetFunction("js_api_set_persistent_count", "void", ["uint64"]);
Afl.jsApiSetPersistentDebug = Afl.jsApiGetFunction("js_api_set_persistent_debug", "void", []);
@@ -280,12 +307,14 @@ Afl.jsApiSetPersistentReturn = Afl.jsApiGetFunction("js_api_set_persistent_retur
Afl.jsApiSetPrefetchBackpatchDisable = Afl.jsApiGetFunction("js_api_set_prefetch_backpatch_disable", "void", []);
Afl.jsApiSetPrefetchDisable = Afl.jsApiGetFunction("js_api_set_prefetch_disable", "void", []);
Afl.jsApiSetSeccompFile = Afl.jsApiGetFunction("js_api_set_seccomp_file", "void", ["pointer"]);
+Afl.jsApiSetStalkerAdjacentBlocks = Afl.jsApiGetFunction("js_api_set_stalker_adjacent_blocks", "void", ["uint32"]);
Afl.jsApiSetStalkerCallback = Afl.jsApiGetFunction("js_api_set_stalker_callback", "void", ["pointer"]);
Afl.jsApiSetStalkerIcEntries = Afl.jsApiGetFunction("js_api_set_stalker_ic_entries", "void", ["uint32"]);
Afl.jsApiSetStatsFile = Afl.jsApiGetFunction("js_api_set_stats_file", "void", ["pointer"]);
Afl.jsApiSetStatsInterval = Afl.jsApiGetFunction("js_api_set_stats_interval", "void", ["uint64"]);
Afl.jsApiSetStdErr = Afl.jsApiGetFunction("js_api_set_stderr", "void", ["pointer"]);
Afl.jsApiSetStdOut = Afl.jsApiGetFunction("js_api_set_stdout", "void", ["pointer"]);
+Afl.jsApiSetTraceable = Afl.jsApiGetFunction("js_api_set_traceable", "void", []);
Afl.jsApiWrite = new NativeFunction(
/* tslint:disable-next-line:no-null-keyword */
Module.getExportByName(null, "write"), "int", ["int", "pointer", "int"]);
diff --git a/frida_mode/src/js/js.c b/frida_mode/src/js/js.c
index e3cd4933..5f477388 100644
--- a/frida_mode/src/js/js.c
+++ b/frida_mode/src/js/js.c
@@ -1,14 +1,13 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "js.h"
#include "util.h"
-static char * js_script = NULL;
gboolean js_done = FALSE;
js_api_stalker_callback_t js_user_callback = NULL;
+js_main_hook_t js_main_hook = NULL;
+static char * js_script = NULL;
static gchar * filename = "afl.js";
static gchar * contents;
static GumScriptBackend * backend;
@@ -25,7 +24,7 @@ static void js_msg(GumScript *script, const gchar *message, GBytes *data,
UNUSED_PARAMETER(script);
UNUSED_PARAMETER(data);
UNUSED_PARAMETER(user_data);
- OKF("%s", message);
+ FOKF("%s", message);
}
@@ -50,14 +49,14 @@ static gchar *js_get_script() {
} else {
- FATAL("Could not load script file: %s", filename);
+ FFATAL("Could not load script file: %s", filename);
}
} else {
- OKF("Loaded AFL script: %s, %" G_GSIZE_MODIFIER "d bytes", filename,
- length);
+ FOKF("Loaded AFL script: %s, %" G_GSIZE_MODIFIER "d bytes", filename,
+ length);
gchar *source = g_malloc0(api_js_len + length + 1);
memcpy(source, api_js, api_js_len);
@@ -75,7 +74,7 @@ static void js_print_script(gchar *source) {
for (size_t i = 0; split[i] != NULL; i++) {
- OKF("%3" G_GSIZE_MODIFIER "d. %s", i + 1, split[i]);
+ FOKF("%3" G_GSIZE_MODIFIER "d. %s", i + 1, split[i]);
}
@@ -89,7 +88,7 @@ static void load_cb(GObject *source_object, GAsyncResult *result,
UNUSED_PARAMETER(source_object);
UNUSED_PARAMETER(user_data);
gum_script_load_finish(script, result);
- if (error != NULL) { FATAL("Failed to load script - %s", error->message); }
+ if (error != NULL) { FFATAL("Failed to load script - %s", error->message); }
}
@@ -99,7 +98,7 @@ static void create_cb(GObject *source_object, GAsyncResult *result,
UNUSED_PARAMETER(source_object);
UNUSED_PARAMETER(user_data);
script = gum_script_backend_create_finish(backend, result, &error);
- if (error != NULL) { FATAL("Failed to create script: %s", error->message); }
+ if (error != NULL) { FFATAL("Failed to create script: %s", error->message); }
gum_script_set_message_handler(script, js_msg, NULL, NULL);
@@ -128,7 +127,7 @@ void js_start(void) {
while (g_main_context_pending(context))
g_main_context_iteration(context, FALSE);
- if (!js_done) { FATAL("Script didn't call Afl.done()"); }
+ if (!js_done) { FFATAL("Script didn't call Afl.done()"); }
}
diff --git a/frida_mode/src/js/js_api.c b/frida_mode/src/js/js_api.c
index 9dba79aa..5021b531 100644
--- a/frida_mode/src/js/js_api.c
+++ b/frida_mode/src/js/js_api.c
@@ -1,4 +1,3 @@
-#include "debug.h"
#include "entry.h"
#include "instrument.h"
@@ -12,6 +11,10 @@
#include "stats.h"
#include "util.h"
+typedef uint8_t u8;
+
+extern void __afl_set_persistent_mode(u8 mode);
+
__attribute__((visibility("default"))) void js_api_done() {
js_done = TRUE;
@@ -20,7 +23,7 @@ __attribute__((visibility("default"))) void js_api_done() {
__attribute__((visibility("default"))) void js_api_error(char *msg) {
- FATAL("%s", msg);
+ FFATAL("%s", msg);
}
@@ -48,6 +51,8 @@ __attribute__((visibility("default"))) void js_api_set_persistent_address(
persistent_start = GPOINTER_TO_SIZE(address);
+ __afl_set_persistent_mode(1);
+
}
__attribute__((visibility("default"))) void js_api_set_persistent_return(
@@ -231,3 +236,29 @@ __attribute__((visibility("default"))) void js_api_set_stalker_ic_entries(
}
+__attribute__((visibility("default"))) void js_api_set_traceable(void) {
+
+ traceable = TRUE;
+
+}
+
+__attribute__((visibility("default"))) void js_api_set_backpatch_disable(void) {
+
+ backpatch_enable = FALSE;
+
+}
+
+__attribute__((visibility("default"))) void js_api_set_stalker_adjacent_blocks(
+ guint val) {
+
+ stalker_adjacent_blocks = val;
+
+}
+
+__attribute__((visibility("default"))) void js_api_set_js_main_hook(
+ const js_main_hook_t hook) {
+
+ js_main_hook = hook;
+
+}
+
diff --git a/frida_mode/src/lib/lib.c b/frida_mode/src/lib/lib.c
index 59a3fcf9..48d2ea2a 100644
--- a/frida_mode/src/lib/lib.c
+++ b/frida_mode/src/lib/lib.c
@@ -8,9 +8,8 @@
#include "frida-gumjs.h"
- #include "debug.h"
-
#include "lib.h"
+ #include "util.h"
#if defined(__arm__) || defined(__i386__)
#define ELFCLASS ELFCLASS32
@@ -55,11 +54,11 @@ static gboolean lib_find_exe(const GumModuleDetails *details,
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");
+ if (hdr->e_ident[0] != ELFMAG0) FFATAL("Invalid e_ident[0]");
+ if (hdr->e_ident[1] != ELFMAG1) FFATAL("Invalid e_ident[1]");
+ if (hdr->e_ident[2] != ELFMAG2) FFATAL("Invalid e_ident[2]");
+ if (hdr->e_ident[3] != ELFMAG3) FFATAL("Invalid e_ident[3]");
+ if (hdr->e_ident[4] != ELFCLASS) FFATAL("Invalid class");
}
@@ -88,18 +87,22 @@ static void lib_read_text_section(lib_details_t *lib_details, Elf_Ehdr *hdr) {
}
- if (!found_preferred_base) { FATAL("Failed to find preferred load address"); }
+ if (!found_preferred_base) {
+
+ FFATAL("Failed to find preferred load address");
+
+ }
- OKF("Image preferred load address 0x%016" G_GSIZE_MODIFIER "x",
- preferred_base);
+ FOKF("Image preferred load address 0x%016" G_GSIZE_MODIFIER "x",
+ 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);
+ FOKF("shdr: %p", shdr);
+ FOKF("shstrtab: %p", shstrtab);
+ FOKF("shstr: %p", shstr);
for (size_t i = 0; i < hdr->e_shnum; i++) {
@@ -108,16 +111,16 @@ static void lib_read_text_section(lib_details_t *lib_details, Elf_Ehdr *hdr) {
if (curr->sh_name == 0) continue;
section_name = &shstr[curr->sh_name];
- OKF("Section: %2" G_GSIZE_MODIFIER "u - base: 0x%016" G_GSIZE_MODIFIER
- "X size: 0x%016" G_GSIZE_MODIFIER "X %s",
- i, curr->sh_addr, curr->sh_size, section_name);
+ FOKF("Section: %2" G_GSIZE_MODIFIER "u - base: 0x%016" G_GSIZE_MODIFIER
+ "X size: 0x%016" G_GSIZE_MODIFIER "X %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%016" G_GINT64_MODIFIER "X", text_base);
- OKF("> text_limit: 0x%016" G_GINT64_MODIFIER "X", text_limit);
+ FOKF("> text_addr: 0x%016" G_GINT64_MODIFIER "X", text_base);
+ FOKF("> text_limit: 0x%016" G_GINT64_MODIFIER "X", text_limit);
}
@@ -132,16 +135,16 @@ static void lib_get_text_section(lib_details_t *details) {
Elf_Ehdr *hdr;
fd = open(details->path, O_RDONLY);
- if (fd < 0) { FATAL("Failed to open %s", details->path); }
+ if (fd < 0) { FFATAL("Failed to open %s", details->path); }
len = lseek(fd, 0, SEEK_END);
- if (len == (off_t)-1) { FATAL("Failed to lseek %s", details->path); }
+ if (len == (off_t)-1) { FFATAL("Failed to lseek %s", details->path); }
- OKF("len: %ld", len);
+ FOKF("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); }
+ if (hdr == MAP_FAILED) { FFATAL("Failed to map %s", details->path); }
lib_validate_hdr(hdr);
lib_read_text_section(details, hdr);
@@ -159,22 +162,22 @@ void lib_init(void) {
lib_details_t lib_details;
gum_process_enumerate_modules(lib_find_exe, &lib_details);
- OKF("Executable: 0x%016" G_GINT64_MODIFIER "x - %s", lib_details.base_address,
- lib_details.path);
+ FOKF("Executable: 0x%016" G_GINT64_MODIFIER "x - %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");
+ if (text_base == 0) FFATAL("Lib not initialized");
return text_base;
}
guint64 lib_get_text_limit(void) {
- if (text_limit == 0) FATAL("Lib not initialized");
+ if (text_limit == 0) FFATAL("Lib not initialized");
return text_limit;
}
diff --git a/frida_mode/src/lib/lib_apple.c b/frida_mode/src/lib/lib_apple.c
index 2aa48a13..3bdb8c10 100644
--- a/frida_mode/src/lib/lib_apple.c
+++ b/frida_mode/src/lib/lib_apple.c
@@ -1,8 +1,6 @@
#ifdef __APPLE__
#include "frida-gumjs.h"
- #include "debug.h"
-
#include "lib.h"
#include "util.h"
@@ -22,7 +20,7 @@ static gboolean lib_get_main_module(const GumModuleDetails *details,
details->path, mach_task_self(), details->range->base_address,
GUM_DARWIN_MODULE_FLAGS_NONE, NULL);
- OKF("Found main module: %s", module->name);
+ FOKF("Found main module: %s", module->name);
*ret = module;
@@ -37,18 +35,18 @@ gboolean lib_get_text_section(const GumDarwinSectionDetails *details,
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);
+ FOKF("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);
+ FOKF("> text_addr: 0x%016" G_GINT64_MODIFIER "X", text_base);
+ FOKF("> text_limit: 0x%016" G_GINT64_MODIFIER "X", text_limit);
}
@@ -70,14 +68,14 @@ void lib_init(void) {
guint64 lib_get_text_base(void) {
- if (text_base == 0) FATAL("Lib not initialized");
+ if (text_base == 0) FFATAL("Lib not initialized");
return text_base;
}
guint64 lib_get_text_limit(void) {
- if (text_limit == 0) FATAL("Lib not initialized");
+ if (text_limit == 0) FFATAL("Lib not initialized");
return text_limit;
}
diff --git a/frida_mode/src/main.c b/frida_mode/src/main.c
index c0de9c6b..1be63bc4 100644
--- a/frida_mode/src/main.c
+++ b/frida_mode/src/main.c
@@ -6,6 +6,7 @@
#ifdef __APPLE__
#include <mach/mach.h>
#include <mach-o/dyld_images.h>
+ #include <crt_externs.h>
#else
#include <sys/wait.h>
#include <sys/personality.h>
@@ -14,7 +15,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "entry.h"
#include "instrument.h"
@@ -36,13 +36,13 @@
extern mach_port_t mach_task_self();
extern GumAddress gum_darwin_find_entrypoint(mach_port_t task);
#else
-extern int __libc_start_main(int *(main)(int, char **, char **), int argc,
+extern int __libc_start_main(int (*main)(int, char **, char **), int argc,
char **ubp_av, void (*init)(void),
void (*fini)(void), void (*rtld_fini)(void),
void(*stack_end));
#endif
-typedef int *(*main_fn_t)(int argc, char **argv, char **envp);
+typedef int (*main_fn_t)(int argc, char **argv, char **envp);
static main_fn_t main_fn = NULL;
@@ -62,7 +62,7 @@ static void on_main_os(int argc, char **argv, char **envp) {
/* Personality doesn't affect the current process, it only takes effect on
* evec */
int persona = personality(ADDR_NO_RANDOMIZE);
- if (persona == -1) { WARNF("Failed to set ADDR_NO_RANDOMIZE: %d", errno); }
+ if (persona == -1) { FWARNF("Failed to set ADDR_NO_RANDOMIZE: %d", errno); }
if ((persona & ADDR_NO_RANDOMIZE) == 0) { execvpe(argv[0], argv, envp); }
GumInterceptor *interceptor = gum_interceptor_obtain();
@@ -90,13 +90,14 @@ static void embedded_init(void) {
static void afl_print_cmdline(void) {
+#if defined(__linux__)
char * buffer = g_malloc0(PROC_MAX);
gchar *fname = g_strdup_printf("/proc/%d/cmdline", getppid());
int fd = open(fname, O_RDONLY);
if (fd < 0) {
- WARNF("Failed to open /proc/self/cmdline, errno: (%d)", errno);
+ FWARNF("Failed to open /proc/self/cmdline, errno: (%d)", errno);
return;
}
@@ -104,7 +105,7 @@ static void afl_print_cmdline(void) {
ssize_t bytes_read = read(fd, buffer, PROC_MAX - 1);
if (bytes_read < 0) {
- FATAL("Failed to read /proc/self/cmdline, errno: (%d)", errno);
+ FFATAL("Failed to read /proc/self/cmdline, errno: (%d)", errno);
}
@@ -114,7 +115,7 @@ static void afl_print_cmdline(void) {
if (i == 0 || buffer[i - 1] == '\0') {
- OKF("AFL - COMMANDLINE: argv[%d] = %s", idx++, &buffer[i]);
+ FOKF("AFL - COMMANDLINE: argv[%d] = %s", idx++, &buffer[i]);
}
@@ -123,6 +124,18 @@ static void afl_print_cmdline(void) {
close(fd);
g_free(fname);
g_free(buffer);
+#elif defined(__APPLE__)
+ int idx;
+ char **argv = *_NSGetArgv();
+ int nargv = *_NSGetArgc();
+
+ for (idx = 0; idx < nargv; idx++) {
+
+ FOKF("AFL - COMMANDLINE: argv[%d] = %s", idx, argv[idx]);
+
+ }
+
+#endif
}
@@ -134,7 +147,7 @@ static void afl_print_env(void) {
if (fd < 0) {
- WARNF("Failed to open /proc/self/cmdline, errno: (%d)", errno);
+ FWARNF("Failed to open /proc/self/cmdline, errno: (%d)", errno);
return;
}
@@ -142,7 +155,7 @@ static void afl_print_env(void) {
ssize_t bytes_read = read(fd, buffer, PROC_MAX - 1);
if (bytes_read < 0) {
- FATAL("Failed to read /proc/self/cmdline, errno: (%d)", errno);
+ FFATAL("Failed to read /proc/self/cmdline, errno: (%d)", errno);
}
@@ -152,7 +165,7 @@ static void afl_print_env(void) {
if (i == 0 || buffer[i - 1] == '\0') {
- OKF("AFL - ENVIRONMENT %3d: %s", idx++, &buffer[i]);
+ FOKF("AFL - ENVIRONMENT %3d: %s", idx++, &buffer[i]);
}
@@ -204,7 +217,9 @@ __attribute__((visibility("default"))) void afl_frida_start(void) {
}
-static int *on_main(int argc, char **argv, char **envp) {
+static int on_main(int argc, char **argv, char **envp) {
+
+ int ret;
on_main_os(argc, argv, envp);
@@ -212,12 +227,22 @@ static int *on_main(int argc, char **argv, char **envp) {
afl_frida_start();
- return main_fn(argc, argv, envp);
+ if (js_main_hook != NULL) {
+
+ ret = js_main_hook(argc, argv, envp);
+
+ } else {
+
+ ret = main_fn(argc, argv, envp);
+
+ }
+
+ return ret;
}
#if defined(EMBEDDED)
-extern int *main(int argc, char **argv, char **envp);
+extern int main(int argc, char **argv, char **envp);
static void intercept_main(void) {
@@ -230,9 +255,9 @@ static void intercept_main(void) {
static void intercept_main(void) {
mach_port_t task = mach_task_self();
- OKF("Task Id: %u", task);
+ FOKF("Task Id: %u", task);
GumAddress entry = gum_darwin_find_entrypoint(task);
- OKF("Entry Point: 0x%016" G_GINT64_MODIFIER "x", entry);
+ FOKF("Entry Point: 0x%016" G_GINT64_MODIFIER "x", entry);
void *main = GSIZE_TO_POINTER(entry);
main_fn = main;
intercept_hook(main, on_main, NULL);
@@ -240,7 +265,7 @@ static void intercept_main(void) {
}
#else
-static int on_libc_start_main(int *(main)(int, char **, char **), int argc,
+static int on_libc_start_main(int (*main)(int, char **, char **), int argc,
char **ubp_av, void (*init)(void),
void (*fini)(void), void (*rtld_fini)(void),
void(*stack_end)) {
diff --git a/frida_mode/src/output.c b/frida_mode/src/output.c
index e2b744e7..f570fe91 100644
--- a/frida_mode/src/output.c
+++ b/frida_mode/src/output.c
@@ -4,9 +4,8 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "output.h"
+#include "util.h"
char *output_stdout = NULL;
char *output_stderr = NULL;
@@ -19,18 +18,18 @@ static void output_redirect(int fd, char *filename) {
path = g_canonicalize_filename(filename, g_get_current_dir());
- OKF("Redirect %d -> '%s'", fd, path);
+ FOKF("Redirect %d -> '%s'", fd, path);
int output_fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
g_free(path);
- if (output_fd < 0) { FATAL("Failed to open fd(%d) error %d", fd, errno); }
+ if (output_fd < 0) { FFATAL("Failed to open fd(%d) error %d", fd, errno); }
if (dup2(output_fd, fd) < 0) {
- FATAL("Failed to set fd(%d) error %d", fd, errno);
+ FFATAL("Failed to set fd(%d) error %d", fd, errno);
}
@@ -47,8 +46,8 @@ void output_config(void) {
void output_init(void) {
- OKF("Output - StdOut: %s", output_stdout);
- OKF("Output - StdErr: %s", output_stderr);
+ FOKF("Output - StdOut: %s", output_stdout);
+ FOKF("Output - StdErr: %s", output_stderr);
output_redirect(STDOUT_FILENO, output_stdout);
output_redirect(STDERR_FILENO, output_stderr);
diff --git a/frida_mode/src/persistent/persistent.c b/frida_mode/src/persistent/persistent.c
index b2915a2f..817d9925 100644
--- a/frida_mode/src/persistent/persistent.c
+++ b/frida_mode/src/persistent/persistent.c
@@ -3,7 +3,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "entry.h"
#include "persistent.h"
@@ -23,15 +22,15 @@ gboolean persistent_debug = FALSE;
void persistent_config(void) {
hook_name = getenv("AFL_FRIDA_PERSISTENT_HOOK");
- persistent_start = util_read_address("AFL_FRIDA_PERSISTENT_ADDR");
- persistent_count = util_read_num("AFL_FRIDA_PERSISTENT_CNT");
- persistent_ret = util_read_address("AFL_FRIDA_PERSISTENT_RET");
+ persistent_start = util_read_address("AFL_FRIDA_PERSISTENT_ADDR", 0);
+ persistent_count = util_read_num("AFL_FRIDA_PERSISTENT_CNT", 0);
+ persistent_ret = util_read_address("AFL_FRIDA_PERSISTENT_RET", 0);
if (getenv("AFL_FRIDA_PERSISTENT_DEBUG") != NULL) { persistent_debug = TRUE; }
if (persistent_count != 0 && persistent_start == 0) {
- FATAL(
+ FFATAL(
"AFL_FRIDA_PERSISTENT_ADDR must be specified if "
"AFL_FRIDA_PERSISTENT_CNT is");
@@ -40,11 +39,11 @@ void persistent_config(void) {
if (persistent_start != 0 && persistent_count == 0) persistent_count = 1000;
if (persistent_start != 0 && !persistent_is_supported())
- FATAL("Persistent mode not supported on this architecture");
+ FFATAL("Persistent mode not supported on this architecture");
if (persistent_ret != 0 && persistent_start == 0) {
- FATAL(
+ FFATAL(
"AFL_FRIDA_PERSISTENT_ADDR must be specified if "
"AFL_FRIDA_PERSISTENT_RET is");
@@ -54,33 +53,33 @@ void persistent_config(void) {
void *hook_obj = dlopen(hook_name, RTLD_NOW);
if (hook_obj == NULL)
- FATAL("Failed to load AFL_FRIDA_PERSISTENT_HOOK (%s)", hook_name);
+ FFATAL("Failed to load AFL_FRIDA_PERSISTENT_HOOK (%s)", hook_name);
int (*afl_persistent_hook_init_ptr)(void) =
dlsym(hook_obj, "afl_persistent_hook_init");
if (afl_persistent_hook_init_ptr == NULL)
- FATAL("Failed to find afl_persistent_hook_init in %s", hook_name);
+ FFATAL("Failed to find afl_persistent_hook_init in %s", hook_name);
if (afl_persistent_hook_init_ptr() == 0)
- FATAL("afl_persistent_hook_init returned a failure");
+ FFATAL("afl_persistent_hook_init returned a failure");
persistent_hook =
(afl_persistent_hook_fn)dlsym(hook_obj, "afl_persistent_hook");
if (persistent_hook == NULL)
- FATAL("Failed to find afl_persistent_hook in %s", hook_name);
+ FFATAL("Failed to find afl_persistent_hook in %s", hook_name);
}
void persistent_init(void) {
- OKF("Instrumentation - persistent mode [%c] (0x%016" G_GINT64_MODIFIER "X)",
- persistent_start == 0 ? ' ' : 'X', persistent_start);
- OKF("Instrumentation - persistent count [%c] (%" G_GINT64_MODIFIER "d)",
- persistent_start == 0 ? ' ' : 'X', persistent_count);
- OKF("Instrumentation - hook [%s]", hook_name);
+ FOKF("Instrumentation - persistent mode [%c] (0x%016" G_GINT64_MODIFIER "X)",
+ persistent_start == 0 ? ' ' : 'X', persistent_start);
+ FOKF("Instrumentation - persistent count [%c] (%" G_GINT64_MODIFIER "d)",
+ persistent_start == 0 ? ' ' : 'X', persistent_count);
+ FOKF("Instrumentation - hook [%s]", hook_name);
- OKF("Instrumentation - persistent ret [%c] (0x%016" G_GINT64_MODIFIER "X)",
- persistent_ret == 0 ? ' ' : 'X', persistent_ret);
+ FOKF("Instrumentation - persistent ret [%c] (0x%016" G_GINT64_MODIFIER "X)",
+ persistent_ret == 0 ? ' ' : 'X', persistent_ret);
if (persistent_hook != NULL) { __afl_sharedmem_fuzzing = 1; }
@@ -88,7 +87,7 @@ void persistent_init(void) {
void persistent_prologue(GumStalkerOutput *output) {
- OKF("AFL_FRIDA_PERSISTENT_ADDR reached");
+ FOKF("AFL_FRIDA_PERSISTENT_ADDR reached");
entry_compiled = TRUE;
ranges_exclude();
stalker_trust();
@@ -98,7 +97,7 @@ void persistent_prologue(GumStalkerOutput *output) {
void persistent_epilogue(GumStalkerOutput *output) {
- OKF("AFL_FRIDA_PERSISTENT_RET reached");
+ FOKF("AFL_FRIDA_PERSISTENT_RET reached");
persistent_epilogue_arch(output);
}
diff --git a/frida_mode/src/persistent/persistent_arm32.c b/frida_mode/src/persistent/persistent_arm32.c
index 769f1505..b4e50897 100644
--- a/frida_mode/src/persistent/persistent_arm32.c
+++ b/frida_mode/src/persistent/persistent_arm32.c
@@ -1,7 +1,5 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "persistent.h"
#include "util.h"
@@ -64,14 +62,14 @@ gboolean persistent_is_supported(void) {
void persistent_prologue_arch(GumStalkerOutput *output) {
UNUSED_PARAMETER(output);
- FATAL("Persistent mode not supported on this architecture");
+ FFATAL("Persistent mode not supported on this architecture");
}
void persistent_epilogue_arch(GumStalkerOutput *output) {
UNUSED_PARAMETER(output);
- FATAL("Persistent mode not supported on this architecture");
+ FFATAL("Persistent mode not supported on this architecture");
}
diff --git a/frida_mode/src/persistent/persistent_arm64.c b/frida_mode/src/persistent/persistent_arm64.c
index 3cd61cd5..c9159ca1 100644
--- a/frida_mode/src/persistent/persistent_arm64.c
+++ b/frida_mode/src/persistent/persistent_arm64.c
@@ -2,7 +2,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "instrument.h"
#include "persistent.h"
@@ -325,7 +324,7 @@ void persistent_prologue_arch(GumStalkerOutput *output) {
gconstpointer loop = cw->code + 1;
- OKF("Persistent loop reached");
+ FOKF("Persistent loop reached");
instrument_persitent_save_regs(cw, &saved_regs);
diff --git a/frida_mode/src/persistent/persistent_x64.c b/frida_mode/src/persistent/persistent_x64.c
index c0bd9a09..8cbde633 100644
--- a/frida_mode/src/persistent/persistent_x64.c
+++ b/frida_mode/src/persistent/persistent_x64.c
@@ -2,7 +2,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "instrument.h"
#include "persistent.h"
@@ -270,7 +269,7 @@ void persistent_prologue_arch(GumStalkerOutput *output) {
gconstpointer loop = cw->code + 1;
- OKF("Persistent loop reached");
+ FOKF("Persistent loop reached");
/* Pop the return value */
gum_x86_writer_put_lea_reg_reg_offset(cw, GUM_REG_RSP, GUM_REG_RSP, 8);
diff --git a/frida_mode/src/persistent/persistent_x86.c b/frida_mode/src/persistent/persistent_x86.c
index b911676a..5425b01b 100644
--- a/frida_mode/src/persistent/persistent_x86.c
+++ b/frida_mode/src/persistent/persistent_x86.c
@@ -1,10 +1,10 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "instrument.h"
#include "persistent.h"
+#include "util.h"
#if defined(__i386__)
@@ -210,7 +210,7 @@ void persistent_prologue_arch(GumStalkerOutput *output) {
gconstpointer loop = cw->code + 1;
- OKF("Persistent loop reached");
+ FOKF("Persistent loop reached");
/* Pop the return value */
gum_x86_writer_put_lea_reg_reg_offset(cw, GUM_REG_ESP, GUM_REG_ESP, 4);
diff --git a/frida_mode/src/prefetch.c b/frida_mode/src/prefetch.c
index 0efbc9bf..8c9ce94d 100644
--- a/frida_mode/src/prefetch.c
+++ b/frida_mode/src/prefetch.c
@@ -4,8 +4,6 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "entry.h"
#include "intercept.h"
#include "prefetch.h"
@@ -44,8 +42,9 @@ static void gum_afl_stalker_backpatcher_notify(GumStalkerObserver *self,
sizeof(prefetch_data->backpatch_data) - prefetch_data->backpatch_size;
if (sizeof(gsize) + size > remaining) { return; }
- *(gsize *)(&prefetch_data->backpatch_data[prefetch_data->backpatch_size]) =
- size;
+ gsize *dst_backpatch_size =
+ (gsize *)&prefetch_data->backpatch_data[prefetch_data->backpatch_size];
+ *dst_backpatch_size = size;
prefetch_data->backpatch_size += sizeof(gsize);
memcpy(&prefetch_data->backpatch_data[prefetch_data->backpatch_size],
@@ -115,12 +114,13 @@ static void prefetch_read_patches(void) {
remaining > sizeof(gsize);
remaining = prefetch_data->backpatch_size - offset) {
- gsize size = *(gsize *)(&prefetch_data->backpatch_data[offset]);
+ gsize *src_backpatch_data = (gsize *)&prefetch_data->backpatch_data[offset];
+ gsize size = *src_backpatch_data;
offset += sizeof(gsize);
if (prefetch_data->backpatch_size - offset < size) {
- FATAL("Incomplete backpatch entry");
+ FFATAL("Incomplete backpatch entry");
}
@@ -178,9 +178,9 @@ static void prefetch_hook_fork(void) {
void prefetch_init(void) {
- OKF("Instrumentation - prefetch [%c]", prefetch_enable ? 'X' : ' ');
- OKF("Instrumentation - prefetch_backpatch [%c]",
- prefetch_backpatch ? 'X' : ' ');
+ FOKF("Instrumentation - prefetch [%c]", prefetch_enable ? 'X' : ' ');
+ FOKF("Instrumentation - prefetch_backpatch [%c]",
+ prefetch_backpatch ? 'X' : ' ');
if (!prefetch_enable) { return; }
/*
@@ -192,7 +192,7 @@ void prefetch_init(void) {
shmget(IPC_PRIVATE, sizeof(prefetch_data_t), IPC_CREAT | IPC_EXCL | 0600);
if (prefetch_shm_id < 0) {
- FATAL("prefetch_shm_id < 0 - errno: %d\n", errno);
+ FFATAL("prefetch_shm_id < 0 - errno: %d\n", errno);
}
@@ -204,7 +204,7 @@ void prefetch_init(void) {
*/
if (shmctl(prefetch_shm_id, IPC_RMID, NULL) < 0) {
- FATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
+ FFATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
}
diff --git a/frida_mode/src/ranges.c b/frida_mode/src/ranges.c
index 5b6eb462..9844c74c 100644
--- a/frida_mode/src/ranges.c
+++ b/frida_mode/src/ranges.c
@@ -1,7 +1,5 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "lib.h"
#include "ranges.h"
#include "stalker.h"
@@ -37,8 +35,8 @@ static void convert_address_token(gchar *token, GumMemoryRange *range) {
if (token_count != 2) {
- FATAL("Invalid range (should have two addresses seperated by a '-'): %s\n",
- token);
+ FFATAL("Invalid range (should have two addresses seperated by a '-'): %s\n",
+ token);
}
@@ -47,15 +45,15 @@ static void convert_address_token(gchar *token, GumMemoryRange *range) {
if (!g_str_has_prefix(from_str, "0x")) {
- FATAL("Invalid range: %s - Start address should have 0x prefix: %s\n",
- token, from_str);
+ FFATAL("Invalid range: %s - Start address should have 0x prefix: %s\n",
+ token, from_str);
}
if (!g_str_has_prefix(to_str, "0x")) {
- FATAL("Invalid range: %s - End address should have 0x prefix: %s\n", token,
- to_str);
+ FFATAL("Invalid range: %s - End address should have 0x prefix: %s\n", token,
+ to_str);
}
@@ -66,8 +64,8 @@ static void convert_address_token(gchar *token, GumMemoryRange *range) {
if (!g_ascii_isxdigit(*c)) {
- FATAL("Invalid range: %s - Start address not formed of hex digits: %s\n",
- token, from_str);
+ FFATAL("Invalid range: %s - Start address not formed of hex digits: %s\n",
+ token, from_str);
}
@@ -77,8 +75,8 @@ static void convert_address_token(gchar *token, GumMemoryRange *range) {
if (!g_ascii_isxdigit(*c)) {
- FATAL("Invalid range: %s - End address not formed of hex digits: %s\n",
- token, to_str);
+ FFATAL("Invalid range: %s - End address not formed of hex digits: %s\n",
+ token, to_str);
}
@@ -87,24 +85,25 @@ static void convert_address_token(gchar *token, GumMemoryRange *range) {
guint64 from = g_ascii_strtoull(from_str, NULL, 16);
if (from == 0) {
- FATAL("Invalid range: %s - Start failed hex conversion: %s\n", token,
- from_str);
+ FFATAL("Invalid range: %s - Start failed hex conversion: %s\n", token,
+ from_str);
}
guint64 to = g_ascii_strtoull(to_str, NULL, 16);
if (to == 0) {
- FATAL("Invalid range: %s - End failed hex conversion: %s\n", token, to_str);
+ FFATAL("Invalid range: %s - End failed hex conversion: %s\n", token,
+ to_str);
}
if (from >= to) {
- FATAL("Invalid range: %s - Start (0x%016" G_GINT64_MODIFIER
- "x) must be less than end "
- "(0x%016" G_GINT64_MODIFIER "x)\n",
- token, from, to);
+ FFATAL("Invalid range: %s - Start (0x%016" G_GINT64_MODIFIER
+ "x) must be less than end "
+ "(0x%016" G_GINT64_MODIFIER "x)\n",
+ token, from, to);
}
@@ -123,10 +122,10 @@ static gboolean convert_name_token_for_module(const GumModuleDetails *details,
if (!g_str_has_suffix(details->path, ctx->suffix)) { return true; };
- OKF("Found module - prefix: %s, 0x%016" G_GINT64_MODIFIER
- "x-0x%016" G_GINT64_MODIFIER "x %s",
- ctx->suffix, details->range->base_address,
- details->range->base_address + details->range->size, details->path);
+ FOKF("Found module - prefix: %s, 0x%016" G_GINT64_MODIFIER
+ "x-0x%016" G_GINT64_MODIFIER "x %s",
+ ctx->suffix, details->range->base_address,
+ details->range->base_address + details->range->size, details->path);
*ctx->range = *details->range;
ctx->done = true;
@@ -140,7 +139,7 @@ static void convert_name_token(gchar *token, GumMemoryRange *range) {
convert_name_ctx_t ctx = {.suffix = suffix, .range = range, .done = false};
gum_process_enumerate_modules(convert_name_token_for_module, &ctx);
- if (!ctx.done) { FATAL("Failed to resolve module: %s\n", token); }
+ if (!ctx.done) { FFATAL("Failed to resolve module: %s\n", token); }
g_free(suffix);
}
@@ -159,16 +158,30 @@ static void convert_token(gchar *token, GumMemoryRange *range) {
}
- OKF("Converted token: %s -> 0x%016" G_GINT64_MODIFIER
- "x-0x%016" G_GINT64_MODIFIER "x\n",
- token, range->base_address, range->base_address + range->size);
+ FOKF("Converted token: %s -> 0x%016" G_GINT64_MODIFIER
+ "x-0x%016" G_GINT64_MODIFIER "x\n",
+ token, range->base_address, range->base_address + range->size);
}
gint range_sort(gconstpointer a, gconstpointer b) {
- return ((GumMemoryRange *)a)->base_address -
- ((GumMemoryRange *)b)->base_address;
+ GumMemoryRange *ra = (GumMemoryRange *)a;
+ GumMemoryRange *rb = (GumMemoryRange *)b;
+
+ if (ra->base_address < rb->base_address) {
+
+ return -1;
+
+ } else if (ra->base_address > rb->base_address) {
+
+ return 1;
+
+ } else {
+
+ return 0;
+
+ }
}
@@ -179,24 +192,24 @@ static gboolean print_ranges_callback(const GumRangeDetails *details,
if (details->file == NULL) {
- OKF("MAP - 0x%016" G_GINT64_MODIFIER "x - 0x%016" G_GINT64_MODIFIER
- "X %c%c%c",
- details->range->base_address,
- details->range->base_address + details->range->size,
- details->protection & GUM_PAGE_READ ? 'R' : '-',
- details->protection & GUM_PAGE_WRITE ? 'W' : '-',
- details->protection & GUM_PAGE_EXECUTE ? 'X' : '-');
+ FOKF("MAP - 0x%016" G_GINT64_MODIFIER "x - 0x%016" G_GINT64_MODIFIER
+ "X %c%c%c",
+ details->range->base_address,
+ details->range->base_address + details->range->size,
+ details->protection & GUM_PAGE_READ ? 'R' : '-',
+ details->protection & GUM_PAGE_WRITE ? 'W' : '-',
+ details->protection & GUM_PAGE_EXECUTE ? 'X' : '-');
} else {
- OKF("MAP - 0x%016" G_GINT64_MODIFIER "x - 0x%016" G_GINT64_MODIFIER
- "X %c%c%c %s(0x%016" G_GINT64_MODIFIER "x)",
- details->range->base_address,
- details->range->base_address + details->range->size,
- details->protection & GUM_PAGE_READ ? 'R' : '-',
- details->protection & GUM_PAGE_WRITE ? 'W' : '-',
- details->protection & GUM_PAGE_EXECUTE ? 'X' : '-', details->file->path,
- details->file->offset);
+ FOKF("MAP - 0x%016" G_GINT64_MODIFIER "x - 0x%016" G_GINT64_MODIFIER
+ "X %c%c%c %s(0x%016" G_GINT64_MODIFIER "x)",
+ details->range->base_address,
+ details->range->base_address + details->range->size,
+ details->protection & GUM_PAGE_READ ? 'R' : '-',
+ details->protection & GUM_PAGE_WRITE ? 'W' : '-',
+ details->protection & GUM_PAGE_EXECUTE ? 'X' : '-',
+ details->file->path, details->file->offset);
}
@@ -206,14 +219,14 @@ static gboolean print_ranges_callback(const GumRangeDetails *details,
static void print_ranges(char *key, GArray *ranges) {
- OKF("Range: %s Length: %d", key, ranges->len);
+ FOKF("Range: %s Length: %d", key, ranges->len);
for (guint i = 0; i < ranges->len; i++) {
GumMemoryRange *curr = &g_array_index(ranges, GumMemoryRange, i);
GumAddress curr_limit = curr->base_address + curr->size;
- OKF("Range: %s Idx: %3d - 0x%016" G_GINT64_MODIFIER
- "x-0x%016" G_GINT64_MODIFIER "x",
- key, i, curr->base_address, curr_limit);
+ FOKF("Range: %s Idx: %3d - 0x%016" G_GINT64_MODIFIER
+ "x-0x%016" G_GINT64_MODIFIER "x",
+ key, i, curr->base_address, curr_limit);
}
@@ -250,10 +263,10 @@ static void check_for_overlaps(GArray *array) {
GumAddress curr_limit = curr->base_address + curr->size;
if (prev_limit > curr->base_address) {
- FATAL("OVerlapping ranges 0x%016" G_GINT64_MODIFIER
- "x-0x%016" G_GINT64_MODIFIER "x 0x%016" G_GINT64_MODIFIER
- "x-0x%016" G_GINT64_MODIFIER "x",
- prev->base_address, prev_limit, curr->base_address, curr_limit);
+ FFATAL("Overlapping ranges 0x%016" G_GINT64_MODIFIER
+ "x-0x%016" G_GINT64_MODIFIER "x 0x%016" G_GINT64_MODIFIER
+ "x-0x%016" G_GINT64_MODIFIER "x",
+ prev->base_address, prev_limit, curr->base_address, curr_limit);
}
@@ -549,18 +562,19 @@ static GArray *merge_ranges(GArray *a) {
}
+void ranges_print_debug_maps(void) {
+
+ gum_process_enumerate_ranges(GUM_PAGE_NO_ACCESS, print_ranges_callback, NULL);
+
+}
+
void ranges_config(void) {
if (getenv("AFL_FRIDA_DEBUG_MAPS") != NULL) { ranges_debug_maps = TRUE; }
if (getenv("AFL_INST_LIBS") != NULL) { ranges_inst_libs = TRUE; }
if (getenv("AFL_FRIDA_INST_JIT") != NULL) { ranges_inst_jit = TRUE; }
- if (ranges_debug_maps) {
-
- gum_process_enumerate_ranges(GUM_PAGE_NO_ACCESS, print_ranges_callback,
- NULL);
-
- }
+ if (ranges_debug_maps) { ranges_print_debug_maps(); }
include_ranges = collect_ranges("AFL_FRIDA_INST_RANGES");
exclude_ranges = collect_ranges("AFL_FRIDA_EXCLUDE_RANGES");
@@ -576,13 +590,13 @@ void ranges_init(void) {
GArray * step4;
GArray * step5;
- OKF("Ranges - Instrument jit [%c]", ranges_inst_jit ? 'X' : ' ');
- OKF("Ranges - Instrument libraries [%c]", ranges_inst_libs ? 'X' : ' ');
+ FOKF("Ranges - Instrument jit [%c]", ranges_inst_jit ? 'X' : ' ');
+ FOKF("Ranges - Instrument libraries [%c]", ranges_inst_libs ? 'X' : ' ');
print_ranges("AFL_FRIDA_INST_RANGES", include_ranges);
print_ranges("AFL_FRIDA_EXCLUDE_RANGES", exclude_ranges);
- OKF("Ranges - Instrument libraries [%c]", ranges_inst_libs ? 'X' : ' ');
+ FOKF("Ranges - Instrument libraries [%c]", ranges_inst_libs ? 'X' : ' ');
print_ranges("AFL_FRIDA_INST_RANGES", include_ranges);
print_ranges("AFL_FRIDA_EXCLUDE_RANGES", exclude_ranges);
@@ -659,7 +673,7 @@ void ranges_exclude() {
GumMemoryRange *r;
GumStalker * stalker = stalker_get();
- OKF("Excluding ranges");
+ FOKF("Excluding ranges");
for (guint i = 0; i < ranges->len; i++) {
diff --git a/frida_mode/src/seccomp/seccomp.c b/frida_mode/src/seccomp/seccomp.c
index 7683cd71..9d8fdd5d 100644
--- a/frida_mode/src/seccomp/seccomp.c
+++ b/frida_mode/src/seccomp/seccomp.c
@@ -1,123 +1,19 @@
-#include <execinfo.h>
-#include <fcntl.h>
-#include <linux/seccomp.h>
-#include <stdio.h>
-#include <unistd.h>
-
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "seccomp.h"
#include "util.h"
char *seccomp_filename = NULL;
-static void seccomp_vprint(int fd, char *format, va_list ap) {
-
- char buffer[4096] = {0};
- int len;
-
- if (vsnprintf(buffer, sizeof(buffer) - 1, format, ap) < 0) { return; }
-
- len = strnlen(buffer, sizeof(buffer));
- IGNORED_RETURN(write(fd, buffer, len));
-
-}
-
-void seccomp_print(char *format, ...) {
-
- va_list ap;
- va_start(ap, format);
- seccomp_vprint(SECCOMP_OUTPUT_FILE_FD, format, ap);
- va_end(ap);
-
-}
-
-static void seccomp_filter_callback(struct seccomp_notif * req,
- struct seccomp_notif_resp *resp,
- GumReturnAddressArray * frames) {
-
- GumDebugSymbolDetails details = {0};
- if (req->data.nr == SYS_OPENAT) {
-
- seccomp_print("SYS_OPENAT: (%s)\n", (char *)req->data.args[1]);
-
- }
-
- seccomp_print(
- "\nID (%#llx) for PID %d - %d (%s) [0x%llx 0x%llx 0x%llx 0x%llx 0x%llx "
- "0x%llx ]\n",
- req->id, req->pid, req->data.nr, seccomp_syscall_lookup(req->data.nr),
- req->data.args[0], req->data.args[1], req->data.args[2],
- req->data.args[3], req->data.args[4], req->data.args[5]);
-
- seccomp_print("FRAMES: (%u)\n", frames->len);
- char **syms = backtrace_symbols(frames->items, frames->len);
- if (syms == NULL) { FATAL("Failed to get symbols"); }
-
- for (guint i = 0; i < frames->len; i++) {
-
- if (gum_symbol_details_from_address(frames->items[i], &details)) {
-
- seccomp_print("\t%3d. %s!%s\n", i, details.module_name,
- details.symbol_name);
-
- } else {
-
- seccomp_print("\t%3d. %s\n", i, syms[i]);
-
- }
-
- }
-
- free(syms);
-
- resp->error = 0;
- resp->val = 0;
- resp->id = req->id;
- resp->flags = SECCOMP_USER_NOTIF_FLAG_CONTINUE;
-
-}
-
-static void seccomp_child(int signal_parent, void *ctx) {
-
- int sock_fd = *((int *)ctx);
- int fd = seccomp_socket_recv(sock_fd);
-
- if (close(sock_fd) < 0) { FATAL("child - close"); }
-
- seccomp_event_signal(signal_parent);
- seccomp_filter_child_install();
- seccomp_filter_run(fd, seccomp_filter_callback);
-
-}
-
void seccomp_on_fork(void) {
- int sock[2] = {-1, -1};
- pid_t child = -1;
- int child_fd = -1;
-
if (seccomp_filename == NULL) { return; }
- seccomp_socket_create(sock);
- seccomp_child_run(seccomp_child, sock, &child, &child_fd);
-
- if (dup2(child_fd, SECCOMP_PARENT_EVENT_FD) < 0) { FATAL("dup2"); }
-
- if (close(child_fd) < 0) { FATAL("seccomp_on_fork - close (1)"); }
-
- if (close(sock[STDIN_FILENO]) < 0) { FATAL("grandparent - close (2)"); }
-
- int fd = seccomp_filter_install(child);
- seccomp_socket_send(sock[STDOUT_FILENO], fd);
-
- if (close(sock[STDOUT_FILENO]) < 0) { FATAL("grandparent - close (3)"); }
-
- if (close(fd) < 0) { FATAL("grandparent - close (4)"); }
-
- seccomp_child_wait(SECCOMP_PARENT_EVENT_FD);
+#ifdef __APPLE__
+ FFATAL("Seccomp not supported on OSX");
+#else
+ seccomp_callback_parent();
+#endif
}
@@ -129,29 +25,15 @@ void seccomp_config(void) {
void seccomp_init(void) {
- char *path = NULL;
- int fd;
-
- OKF("Seccomp - file [%s]", seccomp_filename);
+ FOKF("Seccomp - file [%s]", seccomp_filename);
if (seccomp_filename == NULL) { return; }
- path = g_canonicalize_filename(seccomp_filename, g_get_current_dir());
-
- OKF("Seccomp - path [%s]", path);
-
- fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
- S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
-
- if (dup2(fd, SECCOMP_OUTPUT_FILE_FD) < 0) {
-
- FATAL("Failed to duplicate seccomp output file");
-
- }
-
- if (close(fd) < 0) { FATAL("Failed to close seccomp output file fd"); }
-
- g_free(path);
+#ifdef __APPLE__
+ FFATAL("Seccomp not supported on OSX");
+#else
+ seccomp_callback_initialize();
+#endif
}
diff --git a/frida_mode/src/seccomp/seccomp_atomic.c b/frida_mode/src/seccomp/seccomp_atomic.c
index 1720a726..18cb6724 100644
--- a/frida_mode/src/seccomp/seccomp_atomic.c
+++ b/frida_mode/src/seccomp/seccomp_atomic.c
@@ -1,13 +1,15 @@
-#include <stdbool.h>
-#include <stdio.h>
+#if defined(__linux__) && !defined(__ANDROID__)
-#include "debug.h"
+ #include <stdbool.h>
+ #include <stdio.h>
+
+ #include "util.h"
void seccomp_atomic_set(volatile bool *ptr, bool val) {
if (!__sync_bool_compare_and_swap(ptr, !val, val)) {
- FATAL("Failed to set event");
+ FFATAL("Failed to set event");
}
@@ -26,3 +28,5 @@ void seccomp_atomic_wait(volatile bool *ptr, bool val) {
}
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_callback.c b/frida_mode/src/seccomp/seccomp_callback.c
new file mode 100644
index 00000000..f7aaf78b
--- /dev/null
+++ b/frida_mode/src/seccomp/seccomp_callback.c
@@ -0,0 +1,145 @@
+#if defined(__linux__) && !defined(__ANDROID__)
+
+ #if !defined(__MUSL__)
+ #include <execinfo.h>
+ #endif
+ #include <fcntl.h>
+
+ #include "seccomp.h"
+ #include "util.h"
+
+static void seccomp_callback_filter(struct seccomp_notif * req,
+ struct seccomp_notif_resp *resp,
+ GumReturnAddressArray * frames) {
+
+ GumDebugSymbolDetails details = {0};
+ if (req->data.nr == SYS_OPENAT) {
+
+ #if UINTPTR_MAX == 0xffffffffffffffffu
+ seccomp_print("SYS_OPENAT: (%s)\n", (char *)req->data.args[1]);
+ #endif
+ #if UINTPTR_MAX == 0xffffffff
+ seccomp_print("SYS_OPENAT: (%s)\n", (char *)(__u32)req->data.args[1]);
+ #endif
+
+ }
+
+ seccomp_print(
+ "\nID (%#llx) for PID %d - %d (%s) [0x%llx 0x%llx 0x%llx 0x%llx 0x%llx "
+ "0x%llx ]\n",
+ req->id, req->pid, req->data.nr, seccomp_syscall_lookup(req->data.nr),
+ req->data.args[0], req->data.args[1], req->data.args[2],
+ req->data.args[3], req->data.args[4], req->data.args[5]);
+
+ #if !defined(__MUSL__)
+ seccomp_print("FRAMES: (%u)\n", frames->len);
+ char **syms = backtrace_symbols(frames->items, frames->len);
+ if (syms == NULL) { FFATAL("Failed to get symbols"); }
+
+ for (guint i = 0; i < frames->len; i++) {
+
+ if (gum_symbol_details_from_address(frames->items[i], &details)) {
+
+ seccomp_print("\t%3d. %s!%s\n", i, details.module_name,
+ details.symbol_name);
+
+ } else {
+
+ seccomp_print("\t%3d. %s\n", i, syms[i]);
+
+ }
+
+ }
+
+ free(syms);
+ #else
+ void **syms = (void **)__builtin_frame_address(0);
+ void * framep = __builtin_frame_address(1);
+ int i = 0;
+
+ syms = framep;
+ while (syms) {
+
+ framep = *syms;
+ syms = framep;
+
+ if (!syms) break;
+
+ seccomp_print("\%3d. %s\n", i++, (char *)framep);
+
+ }
+
+ #endif
+
+ resp->error = 0;
+ resp->val = 0;
+ resp->id = req->id;
+ resp->flags = SECCOMP_USER_NOTIF_FLAG_CONTINUE;
+
+}
+
+static void seccomp_callback_child(int signal_parent, void *ctx) {
+
+ int sock_fd = *((int *)ctx);
+ int fd = seccomp_socket_recv(sock_fd);
+
+ if (close(sock_fd) < 0) { FFATAL("child - close"); }
+
+ seccomp_event_signal(signal_parent);
+ seccomp_filter_child_install();
+ seccomp_filter_run(fd, seccomp_callback_filter);
+
+}
+
+void seccomp_callback_parent(void) {
+
+ int sock[2] = {-1, -1};
+ pid_t child = -1;
+ int child_fd = -1;
+
+ seccomp_socket_create(sock);
+ seccomp_child_run(seccomp_callback_child, sock, &child, &child_fd);
+
+ if (dup2(child_fd, SECCOMP_PARENT_EVENT_FD) < 0) { FFATAL("dup2"); }
+
+ if (close(child_fd) < 0) { FFATAL("seccomp_on_fork - close (1)"); }
+
+ if (close(sock[STDIN_FILENO]) < 0) { FFATAL("grandparent - close (2)"); }
+
+ int fd = seccomp_filter_install(child);
+ seccomp_socket_send(sock[STDOUT_FILENO], fd);
+
+ if (close(sock[STDOUT_FILENO]) < 0) { FFATAL("grandparent - close (3)"); }
+
+ if (close(fd) < 0) { FFATAL("grandparent - close (4)"); }
+
+ seccomp_child_wait(SECCOMP_PARENT_EVENT_FD);
+
+}
+
+void seccomp_callback_initialize(void) {
+
+ char *path = NULL;
+ int fd;
+
+ path = g_canonicalize_filename(seccomp_filename, g_get_current_dir());
+
+ FOKF("Seccomp - path [%s]", path);
+
+ fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
+ S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
+
+ if (dup2(fd, SECCOMP_OUTPUT_FILE_FD) < 0) {
+
+ FFATAL("Failed to duplicate seccomp output file");
+
+ }
+
+ if (close(fd) < 0) { FFATAL("Failed to close seccomp output file fd"); }
+
+ g_free(path);
+
+}
+
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_child.c b/frida_mode/src/seccomp/seccomp_child.c
index 4d494137..c02ef67c 100644
--- a/frida_mode/src/seccomp/seccomp_child.c
+++ b/frida_mode/src/seccomp/seccomp_child.c
@@ -1,18 +1,19 @@
-#include <fcntl.h>
-#include <sched.h>
-#include <signal.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <sys/mman.h>
-#include <sys/prctl.h>
-#include <sys/types.h>
-#include <unistd.h>
+#if defined(__linux__) && !defined(__ANDROID__)
-#include "debug.h"
+ #include <fcntl.h>
+ #include <sched.h>
+ #include <signal.h>
+ #include <stdio.h>
+ #include <stdlib.h>
+ #include <sys/mman.h>
+ #include <sys/prctl.h>
+ #include <sys/types.h>
+ #include <unistd.h>
-#include "seccomp.h"
+ #include "seccomp.h"
+ #include "util.h"
-#define SECCOMP_CHILD_STACK_SIZE (1UL << 20)
+ #define SECCOMP_CHILD_STACK_SIZE (1UL << 20)
typedef void (*seccomp_child_func_t)(int event_fd, void *ctx);
@@ -49,11 +50,11 @@ void seccomp_child_run(seccomp_child_func_t child_func, void *ctx, pid_t *child,
char *stack =
(char *)mmap(NULL, SECCOMP_CHILD_STACK_SIZE, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
- if (stack == MAP_FAILED) { FATAL("mmap"); }
+ if (stack == MAP_FAILED) { FFATAL("mmap"); }
pid_t child_pid = clone(seccomp_child_func, &stack[SECCOMP_CHILD_STACK_SIZE],
flags, child_ctx, NULL, NULL, NULL);
- if (child_pid < 0) { FATAL("clone"); }
+ if (child_pid < 0) { FFATAL("clone"); }
if (child != NULL) { *child = child_pid; }
if (event_fd != NULL) { *event_fd = fd; }
@@ -67,3 +68,5 @@ void seccomp_child_wait(int event_fd) {
}
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_event.c b/frida_mode/src/seccomp/seccomp_event.c
index ecb9be32..e6585f1d 100644
--- a/frida_mode/src/seccomp/seccomp_event.c
+++ b/frida_mode/src/seccomp/seccomp_event.c
@@ -1,16 +1,23 @@
-#include <stdint.h>
-#include <stdio.h>
-#include <sys/eventfd.h>
-#include <unistd.h>
+#if defined(__linux__) && !defined(__ANDROID__)
-#include "debug.h"
+ #include <stdint.h>
+ #include <stdio.h>
+ #include <sys/syscall.h>
+ #include <unistd.h>
-#include "seccomp.h"
+ #include "seccomp.h"
+ #include "util.h"
int seccomp_event_create(void) {
- int fd = eventfd(0, 0);
- if (fd < 0) { FATAL("seccomp_event_create"); }
+ #ifdef SYS_eventfd
+ int fd = syscall(SYS_eventfd, 0, 0);
+ #else
+ #ifdef SYS_eventfd2
+ int fd = syscall(SYS_eventfd2, 0, 0);
+ #endif
+ #endif
+ if (fd < 0) { FFATAL("seccomp_event_create"); }
return fd;
}
@@ -20,7 +27,7 @@ void seccomp_event_signal(int fd) {
uint64_t val = 1;
if (write(fd, &val, sizeof(uint64_t)) != sizeof(uint64_t)) {
- FATAL("seccomp_event_signal");
+ FFATAL("seccomp_event_signal");
}
@@ -31,7 +38,7 @@ void seccomp_event_wait(int fd) {
uint64_t val = 1;
if (read(fd, &val, sizeof(uint64_t)) != sizeof(uint64_t)) {
- FATAL("seccomp_event_wait");
+ FFATAL("seccomp_event_wait");
}
@@ -39,7 +46,9 @@ void seccomp_event_wait(int fd) {
void seccomp_event_destroy(int fd) {
- if (close(fd) < 0) { FATAL("seccomp_event_destroy"); }
+ if (close(fd) < 0) { FFATAL("seccomp_event_destroy"); }
}
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_filter.c b/frida_mode/src/seccomp/seccomp_filter.c
index c16e7ebd..1d050303 100644
--- a/frida_mode/src/seccomp/seccomp_filter.c
+++ b/frida_mode/src/seccomp/seccomp_filter.c
@@ -1,27 +1,28 @@
-#include <alloca.h>
-#include <errno.h>
-#include <execinfo.h>
-#include <linux/filter.h>
-#include <linux/seccomp.h>
-#include <sys/ioctl.h>
-#include <sys/prctl.h>
-#include <sys/syscall.h>
-#include <signal.h>
-#include <stdbool.h>
-#include <stddef.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <unistd.h>
-
-#include "debug.h"
-
-#include "frida-gumjs.h"
-
-#include "seccomp.h"
-#include "util.h"
-
-#define SECCOMP_FILTER_NUM_FRAMES 512
+#if defined(__linux__) && !defined(__ANDROID__)
+
+ #include <alloca.h>
+ #include <errno.h>
+ #if !defined(__MUSL__)
+ #include <execinfo.h>
+ #endif
+ #include <linux/filter.h>
+ #include <sys/ioctl.h>
+ #include <sys/prctl.h>
+ #include <sys/syscall.h>
+ #include <signal.h>
+ #include <stdbool.h>
+ #include <stddef.h>
+ #include <stdio.h>
+ #include <stdlib.h>
+ #include <string.h>
+ #include <unistd.h>
+
+ #include "frida-gumjs.h"
+
+ #include "seccomp.h"
+ #include "util.h"
+
+ #define SECCOMP_FILTER_NUM_FRAMES 512
extern void gum_linux_parse_ucontext(const ucontext_t *uc, GumCpuContext *ctx);
@@ -71,7 +72,13 @@ static struct sock_filter filter[] = {
/* Allow us to make anonymous maps */
BPF_STMT(BPF_LD | BPF_W | BPF_ABS, (offsetof(struct seccomp_data, nr))),
+ #ifdef __NR_mmap
BPF_JUMP(BPF_JMP | BPF_JEQ | BPF_K, __NR_mmap, 0, 3),
+ #else
+ #ifdef __NR_mmap2
+ BPF_JUMP(BPF_JMP | BPF_JEQ | BPF_K, __NR_mmap2, 0, 3),
+ #endif
+ #endif
BPF_STMT(BPF_LD | BPF_W | BPF_ABS,
(offsetof(struct seccomp_data, args[4]))),
BPF_JUMP(BPF_JMP | BPF_JEQ | BPF_K, -1, 0, 1),
@@ -127,7 +134,10 @@ static GumBacktracer * seccomp_filter_backtracer = NULL;
static void seccomp_filter_child_handler(int sig, siginfo_t *info,
void *ucontext) {
- GumCpuContext cpu_context;
+ UNUSED_PARAMETER(sig);
+ UNUSED_PARAMETER(info);
+ UNUSED_PARAMETER(ucontext);
+
if (seccomp_filter_backtracer == NULL) {
seccomp_filter_backtracer = gum_backtracer_make_fuzzy();
@@ -150,9 +160,10 @@ static void seccomp_filter_parent_handler(int sig, siginfo_t *info,
ucontext_t *uc = (ucontext_t *)ucontext;
gum_linux_parse_ucontext(uc, &seccomp_filter_cpu_context);
- if (tgkill(seccomp_filter_child, seccomp_filter_child, SIGUSR1) < 0) {
+ if (syscall(SYS_tgkill, seccomp_filter_child, seccomp_filter_child, SIGUSR1) <
+ 0) {
- FATAL("kill");
+ FFATAL("kill");
}
@@ -165,7 +176,7 @@ void seccomp_filter_child_install(void) {
const struct sigaction sa = {.sa_sigaction = seccomp_filter_child_handler,
.sa_flags = SA_SIGINFO | SA_RESTART};
- if (sigaction(SIGUSR1, &sa, NULL) < 0) { FATAL("sigaction"); }
+ if (sigaction(SIGUSR1, &sa, NULL) < 0) { FFATAL("sigaction"); }
}
@@ -180,17 +191,17 @@ int seccomp_filter_install(pid_t child) {
.len = sizeof(filter) / sizeof(struct sock_filter), .filter = filter};
- if (sigaction(SIGUSR1, &sa, NULL) < 0) { FATAL("sigaction"); }
+ if (sigaction(SIGUSR1, &sa, NULL) < 0) { FFATAL("sigaction"); }
if (prctl(PR_SET_NO_NEW_PRIVS, 1, 0, 0, 0)) {
- FATAL("PR_SET_NO_NEW_PRIVS %d", errno);
+ FFATAL("PR_SET_NO_NEW_PRIVS %d", errno);
}
int fd = syscall(SYS_seccomp, SECCOMP_SET_MODE_FILTER,
SECCOMP_FILTER_FLAG_NEW_LISTENER, &filter_prog);
- if (fd < 0) { FATAL("SYS_seccomp %d", fd); }
+ if (fd < 0) { FFATAL("SYS_seccomp %d", fd); }
return fd;
@@ -204,19 +215,19 @@ void seccomp_filter_run(int fd, seccomp_filter_callback_t callback) {
if (syscall(SYS_seccomp, SECCOMP_GET_NOTIF_SIZES, 0, &sizes) == -1) {
- FATAL("seccomp-SECCOMP_GET_NOTIF_SIZES");
+ FFATAL("seccomp-SECCOMP_GET_NOTIF_SIZES");
}
if (sizes.seccomp_notif != sizeof(struct seccomp_notif)) {
- FATAL("size - seccomp_notif");
+ FFATAL("size - seccomp_notif");
}
if (sizes.seccomp_notif_resp != sizeof(struct seccomp_notif_resp)) {
- FATAL("size - seccomp_notif");
+ FFATAL("size - seccomp_notif");
}
@@ -230,7 +241,7 @@ void seccomp_filter_run(int fd, seccomp_filter_callback_t callback) {
if (ioctl(fd, SECCOMP_IOCTL_NOTIF_RECV, req) < 0) {
if (errno == EINTR) { continue; }
- FATAL("SECCOMP_IOCTL_NOTIF_RECV: %d\n", fd);
+ FFATAL("SECCOMP_IOCTL_NOTIF_RECV: %d\n", fd);
}
@@ -240,14 +251,14 @@ void seccomp_filter_run(int fd, seccomp_filter_callback_t callback) {
} else {
- if (kill(req->pid, SIGUSR1) < 0) { FATAL("kill"); }
+ if (kill(req->pid, SIGUSR1) < 0) { FFATAL("kill"); }
}
if (ioctl(fd, SECCOMP_IOCTL_NOTIF_SEND, resp) < 0) {
if (errno == ENOENT) { continue; }
- OKF("SECCOMP_IOCTL_NOTIF_SEND");
+ FOKF("SECCOMP_IOCTL_NOTIF_SEND");
continue;
}
@@ -256,3 +267,5 @@ void seccomp_filter_run(int fd, seccomp_filter_callback_t callback) {
}
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_print.c b/frida_mode/src/seccomp/seccomp_print.c
new file mode 100644
index 00000000..3cea1239
--- /dev/null
+++ b/frida_mode/src/seccomp/seccomp_print.c
@@ -0,0 +1,30 @@
+#if defined(__linux__) && !defined(__ANDROID__)
+
+ #include <stdarg.h>
+
+ #include "seccomp.h"
+ #include "util.h"
+
+static void seccomp_print_v(int fd, char *format, va_list ap) {
+
+ char buffer[4096] = {0};
+ int len;
+
+ if (vsnprintf(buffer, sizeof(buffer) - 1, format, ap) < 0) { return; }
+
+ len = strnlen(buffer, sizeof(buffer));
+ IGNORED_RETURN(write(fd, buffer, len));
+
+}
+
+void seccomp_print(char *format, ...) {
+
+ va_list ap;
+ va_start(ap, format);
+ seccomp_print_v(SECCOMP_OUTPUT_FILE_FD, format, ap);
+ va_end(ap);
+
+}
+
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_socket.c b/frida_mode/src/seccomp/seccomp_socket.c
index ca42e158..a01e88ee 100644
--- a/frida_mode/src/seccomp/seccomp_socket.c
+++ b/frida_mode/src/seccomp/seccomp_socket.c
@@ -1,11 +1,12 @@
-#include <stdio.h>
-#include <string.h>
-#include <sys/socket.h>
-#include <unistd.h>
+#if defined(__linux__) && !defined(__ANDROID__)
-#include "debug.h"
+ #include <stdio.h>
+ #include <string.h>
+ #include <sys/socket.h>
+ #include <unistd.h>
-#include "seccomp.h"
+ #include "seccomp.h"
+ #include "util.h"
union cmsg {
@@ -19,31 +20,31 @@ void seccomp_socket_create(int *sock) {
int tmp_sock[2] = {-1, -1};
if (socketpair(AF_UNIX, SOCK_STREAM, 0, tmp_sock) < 0) {
- FATAL("socketpair");
+ FFATAL("socketpair");
}
if (dup2(tmp_sock[STDIN_FILENO], SECCOMP_SOCKET_RECV_FD) < 0) {
- FATAL("seccomp_socket_create - dup2 (1)");
+ FFATAL("seccomp_socket_create - dup2 (1)");
}
if (dup2(tmp_sock[STDOUT_FILENO], SECCOMP_SOCKET_SEND_FD) < 0) {
- FATAL("seccomp_socket_create - dup2 (1)");
+ FFATAL("seccomp_socket_create - dup2 (1)");
}
if (close(tmp_sock[STDIN_FILENO]) < 0) {
- FATAL("seccomp_socket_create - close (1)");
+ FFATAL("seccomp_socket_create - close (1)");
}
if (close(tmp_sock[STDOUT_FILENO]) < 0) {
- FATAL("seccomp_socket_create - close (2)");
+ FFATAL("seccomp_socket_create - close (2)");
}
@@ -74,7 +75,7 @@ void seccomp_socket_send(int sockfd, int fd) {
memcpy(CMSG_DATA(&control_msg.hdr), &fd, sizeof(int));
- if (sendmsg(sockfd, &message, 0) == -1) { FATAL("sendmsg"); }
+ if (sendmsg(sockfd, &message, 0) == -1) { FFATAL("sendmsg"); }
}
@@ -93,23 +94,23 @@ int seccomp_socket_recv(int sockfd) {
int fd;
- if (recvmsg(sockfd, &message, 0) < 0) { FATAL("recvmsg"); }
+ if (recvmsg(sockfd, &message, 0) < 0) { FFATAL("recvmsg"); }
if (control_msg.hdr.cmsg_len != CMSG_LEN(sizeof(int))) {
- FATAL("control_msg.hdr.cmsg_len");
+ FFATAL("control_msg.hdr.cmsg_len");
}
if (control_msg.hdr.cmsg_level != SOL_SOCKET) {
- FATAL("control_msg.hdr.cmsg_level");
+ FFATAL("control_msg.hdr.cmsg_level");
}
if (control_msg.hdr.cmsg_type != SCM_RIGHTS) {
- FATAL("control_msg.hdr.cmsg_type");
+ FFATAL("control_msg.hdr.cmsg_type");
}
@@ -119,3 +120,5 @@ int seccomp_socket_recv(int sockfd) {
}
+#endif
+
diff --git a/frida_mode/src/seccomp/seccomp_syscall.c b/frida_mode/src/seccomp/seccomp_syscall.c
index b2c084c8..2eac1af3 100644
--- a/frida_mode/src/seccomp/seccomp_syscall.c
+++ b/frida_mode/src/seccomp/seccomp_syscall.c
@@ -1,9 +1,10 @@
-#include <limits.h>
-#include <stdio.h>
+#if defined(__linux__) && !defined(__ANDROID__)
-#include "debug.h"
+ #include <limits.h>
+ #include <stdio.h>
-#include "seccomp.h"
+ #include "seccomp.h"
+ #include "util.h"
typedef struct {
@@ -322,10 +323,10 @@ static syscall_entry_t seccomp_syscall_table[] = {
char *seccomp_syscall_lookup(int id) {
- if (id < 0) { FATAL("Invalid id: %d", id); }
+ if (id < 0) { FFATAL("Invalid id: %d", id); }
if ((uint32_t)id >= sizeof(seccomp_syscall_table) / sizeof(syscall_entry_t)) {
- FATAL("Invalid id: %d", id);
+ FFATAL("Invalid id: %d", id);
}
@@ -333,3 +334,5 @@ char *seccomp_syscall_lookup(int id) {
}
+#endif
+
diff --git a/frida_mode/src/stalker.c b/frida_mode/src/stalker.c
index 814aaeb3..caa16b3f 100644
--- a/frida_mode/src/stalker.c
+++ b/frida_mode/src/stalker.c
@@ -1,4 +1,3 @@
-#include "debug.h"
#include "instrument.h"
#include "prefetch.h"
@@ -6,7 +5,9 @@
#include "stats.h"
#include "util.h"
-guint stalker_ic_entries = 0;
+guint stalker_ic_entries = 0;
+gboolean backpatch_enable = TRUE;
+guint stalker_adjacent_blocks = 0;
static GumStalker *stalker = NULL;
@@ -56,9 +57,14 @@ static void gum_afl_stalker_observer_init(GumAflStalkerObserver *self) {
void stalker_config(void) {
- if (!gum_stalker_is_supported()) { FATAL("Failed to initialize embedded"); }
+ if (!gum_stalker_is_supported()) { FFATAL("Failed to initialize embedded"); }
- stalker_ic_entries = util_read_num("AFL_FRIDA_STALKER_IC_ENTRIES");
+ backpatch_enable = (getenv("AFL_FRIDA_INST_NO_BACKPATCH") == NULL);
+
+ stalker_ic_entries = util_read_num("AFL_FRIDA_STALKER_IC_ENTRIES", 32);
+
+ stalker_adjacent_blocks =
+ util_read_num("AFL_FRIDA_STALKER_ADJACENT_BLOCKS", 32);
observer = g_object_new(GUM_TYPE_AFL_STALKER_OBSERVER, NULL);
@@ -87,27 +93,50 @@ static gboolean stalker_exclude_self(const GumRangeDetails *details,
void stalker_init(void) {
- OKF("Stalker - ic_entries [%u]", stalker_ic_entries);
+ FOKF("Instrumentation - backpatch [%c]", backpatch_enable ? 'X' : ' ');
+
+ FOKF("Stalker - ic_entries [%u]", stalker_ic_entries);
+ FOKF("Stalker - adjacent_blocks [%u]", stalker_adjacent_blocks);
#if !(defined(__x86_64__) || defined(__i386__))
- if (stalker_ic_entries != 0) {
+ if (getenv("AFL_FRIDA_STALKER_IC_ENTRIES") != NULL) {
+
+ FFATAL("AFL_FRIDA_STALKER_IC_ENTRIES not supported");
+
+ }
+
+ if (getenv("AFL_FRIDA_STALKER_ADJACENT_BLOCKS") != NULL) {
- FATAL("AFL_FRIDA_STALKER_IC_ENTRIES not supported");
+ FFATAL("AFL_FRIDA_STALKER_ADJACENT_BLOCKS not supported");
}
#endif
- if (stalker_ic_entries == 0) { stalker_ic_entries = 32; }
+ if (instrument_coverage_filename != NULL) {
+
+ if (getenv("AFL_FRIDA_STALKER_ADJACENT_BLOCKS") != NULL) {
+
+ FFATAL(
+ "AFL_FRIDA_STALKER_ADJACENT_BLOCKS and AFL_FRIDA_INST_COVERAGE_FILE "
+ "are incompatible");
+
+ } else {
+
+ stalker_adjacent_blocks = 0;
+
+ }
+
+ }
#if defined(__x86_64__) || defined(__i386__)
- stalker =
- g_object_new(GUM_TYPE_STALKER, "ic-entries", stalker_ic_entries, NULL);
+ stalker = g_object_new(GUM_TYPE_STALKER, "ic-entries", stalker_ic_entries,
+ "adjacent-blocks", stalker_adjacent_blocks, NULL);
#else
stalker = gum_stalker_new();
#endif
- if (stalker == NULL) { FATAL("Failed to initialize stalker"); }
+ if (stalker == NULL) { FFATAL("Failed to initialize stalker"); }
gum_stalker_set_trust_threshold(stalker, -1);
@@ -118,7 +147,7 @@ void stalker_init(void) {
GumStalker *stalker_get(void) {
- if (stalker == NULL) { FATAL("Stalker uninitialized"); }
+ if (stalker == NULL) { FFATAL("Stalker uninitialized"); }
return stalker;
}
@@ -134,13 +163,13 @@ void stalker_start(void) {
void stalker_trust(void) {
- gum_stalker_set_trust_threshold(stalker, 0);
+ if (backpatch_enable) { gum_stalker_set_trust_threshold(stalker, 0); }
}
GumStalkerObserver *stalker_get_observer(void) {
- if (observer == NULL) { FATAL("Stalker not yet initialized"); }
+ if (observer == NULL) { FFATAL("Stalker not yet initialized"); }
return GUM_STALKER_OBSERVER(observer);
}
diff --git a/frida_mode/src/stats/stats.c b/frida_mode/src/stats/stats.c
index 7972b881..83ecf89a 100644
--- a/frida_mode/src/stats/stats.c
+++ b/frida_mode/src/stats/stats.c
@@ -8,7 +8,6 @@
#include "frida-gumjs.h"
#include "config.h"
-#include "debug.h"
#include "util.h"
#include "entry.h"
@@ -324,42 +323,42 @@ static void stats_observer_init(GumStalkerObserver *observer) {
void stats_config(void) {
stats_filename = getenv("AFL_FRIDA_STATS_FILE");
- stats_interval = util_read_num("AFL_FRIDA_STATS_INTERVAL");
+ stats_interval = util_read_num("AFL_FRIDA_STATS_INTERVAL", 10);
}
void stats_init(void) {
- OKF("Stats - file [%s]", stats_filename);
- OKF("Stats - interval [%" G_GINT64_MODIFIER "u]", stats_interval);
+ FOKF("Stats - file [%s]", stats_filename);
+ FOKF("Stats - interval [%" G_GINT64_MODIFIER "u]", stats_interval);
- if (stats_interval != 0 && stats_filename == NULL) {
+ if (getenv("AFL_FRIDA_STATS_INTERVAL") != NULL &&
+ getenv("AFL_FRIDA_STATS_FILE") == NULL) {
- FATAL(
+ FFATAL(
"AFL_FRIDA_STATS_FILE must be specified if "
"AFL_FRIDA_STATS_INTERVAL is");
}
- if (stats_interval == 0) { stats_interval = 10; }
stats_interval_us = stats_interval * MICRO_TO_SEC;
if (stats_filename == NULL) { return; }
char *path = g_canonicalize_filename(stats_filename, g_get_current_dir());
- OKF("Stats - path [%s]", path);
+ FOKF("Stats - path [%s]", path);
stats_fd = open(path, O_RDWR | O_CREAT | O_TRUNC,
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP);
- if (stats_fd < 0) { FATAL("Failed to open stats file '%s'", path); }
+ if (stats_fd < 0) { FFATAL("Failed to open stats file '%s'", path); }
g_free(path);
int shm_id =
shmget(IPC_PRIVATE, sizeof(stats_data_t), IPC_CREAT | IPC_EXCL | 0600);
- if (shm_id < 0) { FATAL("shm_id < 0 - errno: %d\n", errno); }
+ if (shm_id < 0) { FFATAL("shm_id < 0 - errno: %d\n", errno); }
stats_data = shmat(shm_id, NULL, 0);
g_assert(stats_data != MAP_FAILED);
@@ -372,7 +371,7 @@ void stats_init(void) {
*/
if (shmctl(shm_id, IPC_RMID, NULL) < 0) {
- FATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
+ FFATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
}
diff --git a/frida_mode/src/stats/stats_arm32.c b/frida_mode/src/stats/stats_arm32.c
index 5860d33b..bd652aa3 100644
--- a/frida_mode/src/stats/stats_arm32.c
+++ b/frida_mode/src/stats/stats_arm32.c
@@ -1,7 +1,5 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "stats.h"
#include "util.h"
@@ -9,13 +7,13 @@
void starts_arch_init(void) {
- FATAL("Stats not supported on this architecture");
+ FFATAL("Stats not supported on this architecture");
}
void stats_write_arch(stats_data_t *data) {
- FATAL("Stats not supported on this architecture");
+ FFATAL("Stats not supported on this architecture");
}
@@ -23,7 +21,7 @@ void stats_collect_arch(const cs_insn *instr, gboolean begin) {
UNUSED_PARAMETER(instr);
UNUSED_PARAMETER(begin);
- FATAL("Stats not supported on this architecture");
+ FFATAL("Stats not supported on this architecture");
}
diff --git a/frida_mode/src/stats/stats_arm64.c b/frida_mode/src/stats/stats_arm64.c
index 54b3faf1..313ab47a 100644
--- a/frida_mode/src/stats/stats_arm64.c
+++ b/frida_mode/src/stats/stats_arm64.c
@@ -3,8 +3,6 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "ranges.h"
#include "stats.h"
#include "util.h"
@@ -48,7 +46,7 @@ void starts_arch_init(void) {
int shm_id = shmget(IPC_PRIVATE, sizeof(stats_data_arch_t),
IPC_CREAT | IPC_EXCL | 0600);
- if (shm_id < 0) { FATAL("shm_id < 0 - errno: %d\n", errno); }
+ if (shm_id < 0) { FFATAL("shm_id < 0 - errno: %d\n", errno); }
stats_data_arch = shmat(shm_id, NULL, 0);
g_assert(stats_data_arch != MAP_FAILED);
@@ -58,7 +56,7 @@ void starts_arch_init(void) {
*/
if (shmctl(shm_id, IPC_RMID, NULL) < 0) {
- FATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
+ FFATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
}
diff --git a/frida_mode/src/stats/stats_x86_64.c b/frida_mode/src/stats/stats_x86_64.c
index ab914951..0bfe3baa 100644
--- a/frida_mode/src/stats/stats_x86_64.c
+++ b/frida_mode/src/stats/stats_x86_64.c
@@ -3,8 +3,6 @@
#include "frida-gumjs.h"
-#include "debug.h"
-
#include "ranges.h"
#include "stats.h"
#include "util.h"
@@ -50,7 +48,7 @@ void starts_arch_init(void) {
int shm_id = shmget(IPC_PRIVATE, sizeof(stats_data_arch_t),
IPC_CREAT | IPC_EXCL | 0600);
- if (shm_id < 0) { FATAL("shm_id < 0 - errno: %d\n", errno); }
+ if (shm_id < 0) { FFATAL("shm_id < 0 - errno: %d\n", errno); }
stats_data_arch = shmat(shm_id, NULL, 0);
g_assert(stats_data_arch != MAP_FAILED);
@@ -60,7 +58,7 @@ void starts_arch_init(void) {
*/
if (shmctl(shm_id, IPC_RMID, NULL) < 0) {
- FATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
+ FFATAL("shmctl (IPC_RMID) < 0 - errno: %d\n", errno);
}
@@ -255,8 +253,8 @@ static x86_op_type stats_get_operand_type(const cs_insn *instr) {
if (x86->op_count != 1) {
- FATAL("Unexpected operand count (%d): %s %s\n", x86->op_count,
- instr->mnemonic, instr->op_str);
+ FFATAL("Unexpected operand count (%d): %s %s\n", x86->op_count,
+ instr->mnemonic, instr->op_str);
}
@@ -295,7 +293,7 @@ static void stats_collect_call_arch(const cs_insn *instr) {
stats_data_arch->num_call_mem++;
break;
default:
- FATAL("Invalid operand type: %s %s\n", instr->mnemonic, instr->op_str);
+ FFATAL("Invalid operand type: %s %s\n", instr->mnemonic, instr->op_str);
}
@@ -316,7 +314,7 @@ static void stats_collect_jump_arch(const cs_insn *instr) {
stats_data_arch->num_jmp_mem++;
break;
default:
- FATAL("Invalid operand type: %s %s\n", instr->mnemonic, instr->op_str);
+ FFATAL("Invalid operand type: %s %s\n", instr->mnemonic, instr->op_str);
}
@@ -337,7 +335,7 @@ static void stats_collect_jump_cond_arch(const cs_insn *instr) {
stats_data_arch->num_jmp_cond_mem++;
break;
default:
- FATAL("Invalid operand type: %s %s\n", instr->mnemonic, instr->op_str);
+ FFATAL("Invalid operand type: %s %s\n", instr->mnemonic, instr->op_str);
}
diff --git a/frida_mode/src/util.c b/frida_mode/src/util.c
index 09e8a58b..6f52b6cb 100644
--- a/frida_mode/src/util.c
+++ b/frida_mode/src/util.c
@@ -1,12 +1,11 @@
#include "util.h"
-#include "debug.h"
-
-guint64 util_read_address(char *key) {
+guint64 util_read_address(char *key, guint64 default_value) {
char *value_str = getenv(key);
+ char *end_ptr;
- if (value_str == NULL) { return 0; }
+ if (value_str == NULL) { return default_value; }
if (!g_str_has_prefix(value_str, "0x")) {
@@ -27,8 +26,17 @@ guint64 util_read_address(char *key) {
}
- guint64 value = g_ascii_strtoull(value_str2, NULL, 16);
- if (value == 0) {
+ errno = 0;
+
+ guint64 value = g_ascii_strtoull(value_str2, &end_ptr, 16);
+
+ if (errno != 0) {
+
+ FATAL("Error (%d) during conversion: %s", errno, value_str);
+
+ }
+
+ if (value == 0 && end_ptr == value_str2) {
FATAL("Invalid address failed hex conversion: %s=%s\n", key, value_str2);
@@ -38,11 +46,12 @@ guint64 util_read_address(char *key) {
}
-guint64 util_read_num(char *key) {
+guint64 util_read_num(char *key, guint64 default_value) {
char *value_str = getenv(key);
+ char *end_ptr;
- if (value_str == NULL) { return 0; }
+ if (value_str == NULL) { return default_value; }
for (char *c = value_str; *c != '\0'; c++) {
@@ -55,8 +64,17 @@ guint64 util_read_num(char *key) {
}
+ errno = 0;
+
guint64 value = g_ascii_strtoull(value_str, NULL, 10);
- if (value == 0) {
+
+ if (errno != 0) {
+
+ FATAL("Error (%d) during conversion: %s", errno, value_str);
+
+ }
+
+ if (value == 0 && end_ptr == value_str) {
FATAL("Invalid address failed numeric conversion: %s=%s\n", key, value_str);
@@ -66,3 +84,40 @@ guint64 util_read_num(char *key) {
}
+gboolean util_output_enabled(void) {
+
+ static gboolean initialized = FALSE;
+ static gboolean enabled = TRUE;
+
+ if (!initialized) {
+
+ initialized = TRUE;
+ if (getenv("AFL_DEBUG_CHILD") == NULL) { enabled = FALSE; }
+
+ }
+
+ return enabled;
+
+}
+
+gsize util_rotate(gsize val, gsize shift, gsize size) {
+
+ if (shift == 0) { return val; }
+ gsize result = ((val >> shift) | (val << (size - shift)));
+ result = result & ((1 << size) - 1);
+ return result;
+
+}
+
+gsize util_log2(gsize val) {
+
+ for (gsize i = 0; i < 64; i++) {
+
+ if (((gsize)1 << i) == val) { return i; }
+
+ }
+
+ FFATAL("Not a power of two");
+
+}
+