aboutsummaryrefslogtreecommitdiffstats
path: root/hw
diff options
context:
space:
mode:
authorVladimir Chtchetkine <vchtchetkine@google.com>2010-02-16 10:38:35 -0800
committerVladimir Chtchetkine <vchtchetkine@google.com>2010-02-18 15:22:07 -0800
commit5389aa19033153c09556d1362a8b8a56abccb8f5 (patch)
tree5d731effe5bd5d2f162f06aadec7212045eaef3d /hw
parent76dbca0489ab98a46f2954bc7b77c3df6f9d8264 (diff)
downloadexternal_qemu-5389aa19033153c09556d1362a8b8a56abccb8f5.zip
external_qemu-5389aa19033153c09556d1362a8b8a56abccb8f5.tar.gz
external_qemu-5389aa19033153c09556d1362a8b8a56abccb8f5.tar.bz2
Merge memory checking from sandbox
Change-id: Ibce845d0
Diffstat (limited to 'hw')
-rw-r--r--hw/android_arm.c17
-rw-r--r--hw/goldfish_trace.c203
2 files changed, 179 insertions, 41 deletions
diff --git a/hw/android_arm.c b/hw/android_arm.c
index 4ef4a10..d656b81 100644
--- a/hw/android_arm.c
+++ b/hw/android_arm.c
@@ -20,6 +20,9 @@
#include "audio/audio.h"
#include "arm-misc.h"
#include "console.h"
+#ifdef CONFIG_MEMCHECK
+#include "memcheck/memcheck_api.h"
+#endif // CONFIG_MEMCHECK
#define ARM_CPU_SAVE_VERSION 1
@@ -63,7 +66,7 @@ uint32_t switch_test_write(void *opaque, uint32_t state)
#endif
static void android_arm_init_(ram_addr_t ram_size,
const char *boot_device,
- const char *kernel_filename,
+ const char *kernel_filename,
const char *kernel_cmdline,
const char *initrd_filename,
const char *cpu_model)
@@ -142,9 +145,19 @@ static void android_arm_init_(ram_addr_t ram_size,
#endif
#ifdef CONFIG_TRACE
extern const char *trace_filename;
- if (trace_filename != NULL) {
+ /* Init trace device if either tracing, or memory checking is enabled. */
+ if (trace_filename != NULL
+#ifdef CONFIG_MEMCHECK
+ || memcheck_enabled
+#endif // CONFIG_MEMCHECK
+ ) {
trace_dev_init();
}
+ if (trace_filename != NULL) {
+ dprint( "Trace file name is set to %s\n", trace_filename );
+ } else {
+ dprint("Trace file name is not set\n");
+ }
#endif
#if TEST_SWITCH
diff --git a/hw/goldfish_trace.c b/hw/goldfish_trace.c
index c4f2f92..742ac63 100644
--- a/hw/goldfish_trace.c
+++ b/hw/goldfish_trace.c
@@ -15,12 +15,16 @@
*/
#include "qemu_file.h"
#include "goldfish_trace.h"
+#ifdef CONFIG_MEMCHECK
+#include "memcheck/memcheck.h"
+#endif // CONFIG_MEMCHECK
//#define DEBUG 1
extern void cpu_loop_exit(void);
extern int tracing;
+extern const char *trace_filename;
/* for execve */
static char path[CLIENT_PAGE_SIZE];
@@ -44,28 +48,51 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
switch (offset >> 2) {
case TRACE_DEV_REG_SWITCH: // context switch, switch to pid
- trace_switch(value);
+ if (trace_filename != NULL) {
+ trace_switch(value);
#ifdef DEBUG
- printf("QEMU.trace: kernel, context switch %u\n", value);
+ printf("QEMU.trace: kernel, context switch %u\n", value);
#endif
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_switch(value);
+ }
+#endif // CONFIG_MEMCHECK
break;
case TRACE_DEV_REG_TGID: // save the tgid for the following fork/clone
tgid = value;
#ifdef DEBUG
- printf("QEMU.trace: kernel, tgid %u\n", value);
+ if (trace_filename != NULL) {
+ printf("QEMU.trace: kernel, tgid %u\n", value);
+ }
#endif
break;
case TRACE_DEV_REG_FORK: // fork, fork new pid
- trace_fork(tgid, value);
+ if (trace_filename != NULL) {
+ trace_fork(tgid, value);
#ifdef DEBUG
- printf("QEMU.trace: kernel, fork %u\n", value);
+ printf("QEMU.trace: kernel, fork %u\n", value);
#endif
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_fork(tgid, value);
+ }
+#endif // CONFIG_MEMCHECK
break;
case TRACE_DEV_REG_CLONE: // fork, clone new pid (i.e. thread)
- trace_clone(tgid, value);
+ if (trace_filename != NULL) {
+ trace_clone(tgid, value);
#ifdef DEBUG
- printf("QEMU.trace: kernel, clone %u\n", value);
+ printf("QEMU.trace: kernel, clone %u\n", value);
#endif
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_clone(tgid, value);
+ }
+#endif // CONFIG_MEMCHECK
break;
case TRACE_DEV_REG_EXECVE_VMSTART: // execve, vstart
vstart = value;
@@ -78,10 +105,23 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
break;
case TRACE_DEV_REG_EXECVE_EXEPATH: // init exec, path of EXE
vstrcpy(value, path, CLIENT_PAGE_SIZE);
- trace_init_exec(vstart, vend, eoff, path);
+ if (trace_filename != NULL) {
+ trace_init_exec(vstart, vend, eoff, path);
#ifdef DEBUG
- printf("QEMU.trace: kernel, init exec [%lx,%lx]@%lx [%s]\n", vstart, vend, eoff, path);
+ printf("QEMU.trace: kernel, init exec [%lx,%lx]@%lx [%s]\n",
+ vstart, vend, eoff, path);
#endif
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ if (path[0] == '\0') {
+ // vstrcpy may fail to copy path. In this case lets do it
+ // differently.
+ memcheck_get_guest_kernel_string(path, value, CLIENT_PAGE_SIZE);
+ }
+ memcheck_mmap_exepath(vstart, vend, eoff, path);
+ }
+#endif // CONFIG_MEMCHECK
path[0] = 0;
break;
case TRACE_DEV_REG_CMDLINE_LEN: // execve, process cmdline length
@@ -89,23 +129,37 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
break;
case TRACE_DEV_REG_CMDLINE: // execve, process cmdline
cpu_memory_rw_debug(cpu_single_env, value, arg, cmdlen, 0);
- trace_execve(arg, cmdlen);
+ if (trace_filename != NULL) {
+ trace_execve(arg, cmdlen);
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_set_cmd_line(arg, cmdlen);
+ }
+#endif // CONFIG_MEMCHECK
#ifdef DEBUG
- {
+ if (trace_filename != NULL) {
int i;
for (i = 0; i < cmdlen; i ++)
if (i != cmdlen - 1 && arg[i] == 0)
arg[i] = ' ';
printf("QEMU.trace: kernel, execve %s[%d]\n", arg, cmdlen);
+ arg[0] = 0;
}
#endif
- arg[0] = 0;
break;
case TRACE_DEV_REG_EXIT: // exit, exit current process with exit code
- trace_exit(value);
+ if (trace_filename != NULL) {
+ trace_exit(value);
#ifdef DEBUG
- printf("QEMU.trace: kernel, exit %x\n", value);
+ printf("QEMU.trace: kernel, exit %x\n", value);
#endif
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_exit(value);
+ }
+#endif // CONFIG_MEMCHECK
break;
case TRACE_DEV_REG_NAME: // record thread name
vstrcpy(value, path, CLIENT_PAGE_SIZE);
@@ -115,28 +169,49 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
if (path[len - 1] == '\n') {
path[len - 1] = 0;
}
- trace_name(path);
+ if (trace_filename != NULL) {
+ trace_name(path);
#ifdef DEBUG
- printf("QEMU.trace: kernel, name %s\n", path);
+ printf("QEMU.trace: kernel, name %s\n", path);
#endif
+ }
break;
case TRACE_DEV_REG_MMAP_EXEPATH: // mmap, path of EXE, the others are same as execve
vstrcpy(value, path, CLIENT_PAGE_SIZE);
- trace_mmap(vstart, vend, eoff, path);
+ if (trace_filename != NULL) {
+ trace_mmap(vstart, vend, eoff, path);
#ifdef DEBUG
- printf("QEMU.trace: kernel, mmap [%lx,%lx]@%lx [%s]\n", vstart, vend, eoff, path);
+ printf("QEMU.trace: kernel, mmap [%lx,%lx]@%lx [%s]\n", vstart, vend, eoff, path);
#endif
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ if (path[0] == '\0') {
+ // vstrcpy may fail to copy path. In this case lets do it
+ // differently.
+ memcheck_get_guest_kernel_string(path, value, CLIENT_PAGE_SIZE);
+ }
+ memcheck_mmap_exepath(vstart, vend, eoff, path);
+ }
+#endif // CONFIG_MEMCHECK
path[0] = 0;
break;
case TRACE_DEV_REG_INIT_PID: // init, name the pid that starts before device registered
pid = value;
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_init_pid(value);
+ }
+#endif // CONFIG_MEMCHECK
break;
case TRACE_DEV_REG_INIT_NAME: // init, the comm of the init pid
vstrcpy(value, path, CLIENT_PAGE_SIZE);
- trace_init_name(tgid, pid, path);
+ if (trace_filename != NULL) {
+ trace_init_name(tgid, pid, path);
#ifdef DEBUG
- printf("QEMU.trace: kernel, init name %u [%s]\n", pid, path);
+ printf("QEMU.trace: kernel, init name %u [%s]\n", pid, path);
#endif
+ }
path[0] = 0;
break;
@@ -145,18 +220,21 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
break;
case TRACE_DEV_REG_DYN_SYM: // add dynamic symbol
vstrcpy(value, arg, CLIENT_PAGE_SIZE);
- trace_dynamic_symbol_add(dsaddr, arg);
+ if (trace_filename != NULL) {
+ trace_dynamic_symbol_add(dsaddr, arg);
#ifdef DEBUG
- printf("QEMU.trace: dynamic symbol %lx:%s\n", dsaddr, arg);
+ printf("QEMU.trace: dynamic symbol %lx:%s\n", dsaddr, arg);
#endif
+ }
arg[0] = 0;
break;
case TRACE_DEV_REG_REMOVE_ADDR: // remove dynamic symbol addr
- trace_dynamic_symbol_remove(value);
+ if (trace_filename != NULL) {
+ trace_dynamic_symbol_remove(value);
#ifdef DEBUG
- printf("QEMU.trace: dynamic symbol remove %lx\n", dsaddr);
+ printf("QEMU.trace: dynamic symbol remove %lx\n", dsaddr);
#endif
- arg[0] = 0;
+ }
break;
case TRACE_DEV_REG_PRINT_STR: // print string
@@ -172,10 +250,11 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
break;
case TRACE_DEV_REG_STOP_EMU: // stop the VM execution
- // To ensure that the number of instructions executed in this
- // block is correct, we pretend that there was an exception.
- trace_exception(0);
-
+ if (trace_filename != NULL) {
+ // To ensure that the number of instructions executed in this
+ // block is correct, we pretend that there was an exception.
+ trace_exception(0);
+ }
cpu_single_env->exception_index = EXCP_HLT;
cpu_single_env->halted = 1;
qemu_system_shutdown_request();
@@ -183,14 +262,19 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
break;
case TRACE_DEV_REG_ENABLE: // tracing enable: 0 = stop, 1 = start
- if (value == 1)
- start_tracing();
+ if (value == 1) {
+ if (trace_filename != NULL) {
+ start_tracing();
+ }
+ }
else if (value == 0) {
- stop_tracing();
+ if (trace_filename != NULL) {
+ stop_tracing();
- // To ensure that the number of instructions executed in this
- // block is correct, we pretend that there was an exception.
- trace_exception(0);
+ // To ensure that the number of instructions executed in this
+ // block is correct, we pretend that there was an exception.
+ trace_exception(0);
+ }
}
break;
@@ -198,7 +282,14 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
unmap_start = value;
break;
case TRACE_DEV_REG_UNMAP_END:
- trace_munmap(unmap_start, value);
+ if (trace_filename != NULL) {
+ trace_munmap(unmap_start, value);
+ }
+#ifdef CONFIG_MEMCHECK
+ if (memcheck_enabled) {
+ memcheck_unmap(unmap_start, value);
+ }
+#endif // CONFIG_MEMCHECK
break;
case TRACE_DEV_REG_METHOD_ENTRY:
@@ -207,11 +298,45 @@ static void trace_dev_write(void *opaque, target_phys_addr_t offset, uint32_t va
case TRACE_DEV_REG_NATIVE_ENTRY:
case TRACE_DEV_REG_NATIVE_EXIT:
case TRACE_DEV_REG_NATIVE_EXCEPTION:
- if (tracing) {
- int call_type = (offset - 4096) >> 2;
- trace_interpreted_method(value, call_type);
+ if (trace_filename != NULL) {
+ if (tracing) {
+ int call_type = (offset - 4096) >> 2;
+ trace_interpreted_method(value, call_type);
+ }
+ }
+ break;
+
+#ifdef CONFIG_MEMCHECK
+ case TRACE_DEV_REG_MALLOC:
+ if (memcheck_enabled) {
+ memcheck_guest_alloc(value);
+ }
+ break;
+
+ case TRACE_DEV_REG_FREE_PTR:
+ if (memcheck_enabled) {
+ memcheck_guest_free(value);
+ }
+ break;
+
+ case TRACE_DEV_REG_QUERY_MALLOC:
+ if (memcheck_enabled) {
+ memcheck_guest_query_malloc(value);
+ }
+ break;
+
+ case TRACE_DEV_REG_LIBC_INIT:
+ if (memcheck_enabled) {
+ memcheck_guest_libc_initialized(value);
+ }
+ break;
+
+ case TRACE_DEV_REG_PRINT_USER_STR:
+ if (memcheck_enabled) {
+ memcheck_guest_print_str(value);
}
break;
+#endif // CONFIG_MEMCHECK
default:
if (offset < 4096) {