aboutsummaryrefslogtreecommitdiffstats
path: root/emulator
diff options
context:
space:
mode:
Diffstat (limited to 'emulator')
-rw-r--r--emulator/qtools/Android.mk157
-rw-r--r--emulator/qtools/armdis.cpp905
-rw-r--r--emulator/qtools/armdis.h45
-rw-r--r--emulator/qtools/bb2sym.cpp140
-rw-r--r--emulator/qtools/bb_dump.cpp47
-rw-r--r--emulator/qtools/bbprof.cpp222
-rw-r--r--emulator/qtools/bitvector.h40
-rw-r--r--emulator/qtools/callstack.h775
-rw-r--r--emulator/qtools/check_stack.cpp270
-rw-r--r--emulator/qtools/check_trace.cpp61
-rw-r--r--emulator/qtools/coverage.cpp153
-rw-r--r--emulator/qtools/decoder.cpp278
-rw-r--r--emulator/qtools/decoder.h28
-rw-r--r--emulator/qtools/dmtrace.cpp255
-rw-r--r--emulator/qtools/dmtrace.h61
-rw-r--r--emulator/qtools/dump_regions.cpp59
-rw-r--r--emulator/qtools/exc_dump.cpp28
-rw-r--r--emulator/qtools/gtrace.cpp152
-rw-r--r--emulator/qtools/gtrace.h69
-rw-r--r--emulator/qtools/hash_table.h219
-rw-r--r--emulator/qtools/hist_trace.cpp64
-rw-r--r--emulator/qtools/opcode.cpp204
-rw-r--r--emulator/qtools/opcode.h166
-rw-r--r--emulator/qtools/parse_options-inl.h155
-rw-r--r--emulator/qtools/parse_options.cpp119
-rw-r--r--emulator/qtools/parse_options.h32
-rw-r--r--emulator/qtools/post_trace.cpp151
-rw-r--r--emulator/qtools/profile_pid.cpp94
-rw-r--r--emulator/qtools/profile_trace.cpp131
-rw-r--r--emulator/qtools/q2dm.cpp274
-rw-r--r--emulator/qtools/q2g.cpp108
-rw-r--r--emulator/qtools/read_addr.cpp29
-rw-r--r--emulator/qtools/read_elf.cpp210
-rw-r--r--emulator/qtools/read_elf.h20
-rw-r--r--emulator/qtools/read_method.cpp137
-rw-r--r--emulator/qtools/read_pid.cpp71
-rw-r--r--emulator/qtools/read_trace.cpp165
-rw-r--r--emulator/qtools/stack_dump.cpp156
-rw-r--r--emulator/qtools/tests/common_head.mk25
-rw-r--r--emulator/qtools/tests/common_tail.mk3
-rw-r--r--emulator/qtools/tests/gtrace/Makefile18
-rw-r--r--emulator/qtools/tests/gtrace/test.c201
-rw-r--r--emulator/qtools/tests/macros.h93
-rw-r--r--emulator/qtools/tests/tests.ld10
-rw-r--r--emulator/qtools/thumbdis.cpp503
-rw-r--r--emulator/qtools/trace_reader.cpp1201
-rw-r--r--emulator/qtools/trace_reader.h1559
-rw-r--r--emulator/qtools/trace_reader_base.h332
48 files changed, 0 insertions, 10195 deletions
diff --git a/emulator/qtools/Android.mk b/emulator/qtools/Android.mk
deleted file mode 100644
index 8b2c25b..0000000
--- a/emulator/qtools/Android.mk
+++ /dev/null
@@ -1,157 +0,0 @@
-#
-# Copyright 2006 The Android Open Source Project
-#
-# Java method trace dump tool
-#
-
-LOCAL_PATH:= $(call my-dir)
-
-common_includes := external/qemu/include external/qemu
-common_cflags := -O0 -g
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := post_trace.cpp trace_reader.cpp decoder.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := post_trace
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := read_trace.cpp trace_reader.cpp decoder.cpp armdis.cpp \
- thumbdis.cpp opcode.cpp read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := read_trace
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := check_trace.cpp trace_reader.cpp decoder.cpp \
- opcode.cpp read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := check_trace
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := bb_dump.cpp trace_reader.cpp decoder.cpp \
- read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := bb_dump
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := bb2sym.cpp trace_reader.cpp decoder.cpp \
- read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := bb2sym
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := profile_trace.cpp trace_reader.cpp decoder.cpp \
- opcode.cpp read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := profile_trace
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := bbprof.cpp trace_reader.cpp decoder.cpp armdis.cpp \
- thumbdis.cpp opcode.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := bbprof
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := q2g.cpp trace_reader.cpp decoder.cpp \
- opcode.cpp read_elf.cpp parse_options.cpp gtrace.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := q2g
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := q2dm.cpp trace_reader.cpp decoder.cpp armdis.cpp \
- thumbdis.cpp opcode.cpp read_elf.cpp parse_options.cpp dmtrace.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := q2dm
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := coverage.cpp trace_reader.cpp decoder.cpp armdis.cpp \
- thumbdis.cpp opcode.cpp read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := coverage
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := stack_dump.cpp trace_reader.cpp decoder.cpp armdis.cpp \
- thumbdis.cpp opcode.cpp read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := stack_dump
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := check_stack.cpp trace_reader.cpp decoder.cpp armdis.cpp \
- thumbdis.cpp opcode.cpp read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := check_stack
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := hist_trace.cpp trace_reader.cpp decoder.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := hist_trace
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := read_addr.cpp trace_reader.cpp decoder.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := read_addr
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := read_pid.cpp trace_reader.cpp decoder.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := read_pid
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := exc_dump.cpp trace_reader.cpp decoder.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := exc_dump
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := read_method.cpp trace_reader.cpp decoder.cpp \
- read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := read_method
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := profile_pid.cpp trace_reader.cpp decoder.cpp \
- read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := profile_pid
-include $(BUILD_HOST_EXECUTABLE)
-
-include $(CLEAR_VARS)
-LOCAL_SRC_FILES := dump_regions.cpp trace_reader.cpp decoder.cpp \
- read_elf.cpp parse_options.cpp
-LOCAL_C_INCLUDES += $(common_includes)
-LOCAL_CFLAGS += $(common_cflags)
-LOCAL_MODULE := dump_regions
-include $(BUILD_HOST_EXECUTABLE)
diff --git a/emulator/qtools/armdis.cpp b/emulator/qtools/armdis.cpp
deleted file mode 100644
index 1f35867..0000000
--- a/emulator/qtools/armdis.cpp
+++ /dev/null
@@ -1,905 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <string.h>
-#include "armdis.h"
-#include "opcode.h"
-
-static const char *cond_names[] = {
- "eq",
- "ne",
- "cs",
- "cc",
- "mi",
- "pl",
- "vs",
- "vc",
- "hi",
- "ls",
- "ge",
- "lt",
- "gt",
- "le",
- "",
- "RESERVED"
-};
-
-// Indexed by the shift type (bits 6-5)
-static const char *shift_names[] = {
- "LSL",
- "LSR",
- "ASR",
- "ROR"
-};
-
-static const char* cond_to_str(int cond) {
- return cond_names[cond];
-}
-
-char *Arm::disasm(uint32_t addr, uint32_t insn, char *result)
-{
- static char buf[80];
- char *ptr;
-
- ptr = result ? result : buf;
- Opcode opcode = decode(insn);
- switch (opcode) {
- case OP_INVALID:
- sprintf(ptr, "Invalid");
- return ptr;
- case OP_UNDEFINED:
- sprintf(ptr, "Undefined");
- return ptr;
- case OP_ADC:
- case OP_ADD:
- case OP_AND:
- case OP_BIC:
- case OP_CMN:
- case OP_CMP:
- case OP_EOR:
- case OP_MOV:
- case OP_MVN:
- case OP_ORR:
- case OP_RSB:
- case OP_RSC:
- case OP_SBC:
- case OP_SUB:
- case OP_TEQ:
- case OP_TST:
- return disasm_alu(opcode, insn, ptr);
- case OP_B:
- case OP_BL:
- return disasm_branch(addr, opcode, insn, ptr);
- case OP_BKPT:
- return disasm_bkpt(insn, ptr);
- case OP_BLX:
- // not supported yet
- break;
- case OP_BX:
- return disasm_bx(insn, ptr);
- case OP_CDP:
- sprintf(ptr, "cdp");
- return ptr;
- case OP_CLZ:
- return disasm_clz(insn, ptr);
- case OP_LDC:
- sprintf(ptr, "ldc");
- return ptr;
- case OP_LDM:
- case OP_STM:
- return disasm_memblock(opcode, insn, ptr);
- case OP_LDR:
- case OP_LDRB:
- case OP_LDRBT:
- case OP_LDRT:
- case OP_STR:
- case OP_STRB:
- case OP_STRBT:
- case OP_STRT:
- return disasm_mem(insn, ptr);
- case OP_LDRH:
- case OP_LDRSB:
- case OP_LDRSH:
- case OP_STRH:
- return disasm_memhalf(insn, ptr);
- case OP_MCR:
- case OP_MRC:
- return disasm_mcr(opcode, insn, ptr);
- case OP_MLA:
- return disasm_mla(opcode, insn, ptr);
- case OP_MRS:
- return disasm_mrs(insn, ptr);
- case OP_MSR:
- return disasm_msr(insn, ptr);
- case OP_MUL:
- return disasm_mul(opcode, insn, ptr);
- case OP_PLD:
- return disasm_pld(insn, ptr);
- case OP_STC:
- sprintf(ptr, "stc");
- return ptr;
- case OP_SWI:
- return disasm_swi(insn, ptr);
- case OP_SWP:
- case OP_SWPB:
- return disasm_swp(opcode, insn, ptr);
- case OP_UMLAL:
- case OP_UMULL:
- case OP_SMLAL:
- case OP_SMULL:
- return disasm_umlal(opcode, insn, ptr);
- default:
- sprintf(ptr, "Error");
- return ptr;
- }
- return NULL;
-}
-
-char *Arm::disasm_alu(Opcode opcode, uint32_t insn, char *ptr)
-{
- static const uint8_t kNoOperand1 = 1;
- static const uint8_t kNoDest = 2;
- static const uint8_t kNoSbit = 4;
-
- char rn_str[20];
- char rd_str[20];
- uint8_t flags = 0;
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t is_immed = (insn >> 25) & 0x1;
- uint8_t bit_s = (insn >> 20) & 1;
- uint8_t rn = (insn >> 16) & 0xf;
- uint8_t rd = (insn >> 12) & 0xf;
- uint8_t immed = insn & 0xff;
-
- const char *opname = opcode_names[opcode];
- switch (opcode) {
- case OP_CMN:
- case OP_CMP:
- case OP_TEQ:
- case OP_TST:
- flags = kNoDest | kNoSbit;
- break;
- case OP_MOV:
- case OP_MVN:
- flags = kNoOperand1;
- break;
- default:
- break;
- }
-
- // The "mov" instruction ignores the first operand (rn).
- rn_str[0] = 0;
- if ((flags & kNoOperand1) == 0) {
- sprintf(rn_str, "r%d, ", rn);
- }
-
- // The following instructions do not write the result register (rd):
- // tst, teq, cmp, cmn.
- rd_str[0] = 0;
- if ((flags & kNoDest) == 0) {
- sprintf(rd_str, "r%d, ", rd);
- }
-
- const char *sbit_str = "";
- if (bit_s && !(flags & kNoSbit))
- sbit_str = "s";
-
- if (is_immed) {
- sprintf(ptr, "%s%s%s\t%s%s#%u ; 0x%x",
- opname, cond_to_str(cond), sbit_str, rd_str, rn_str, immed, immed);
- return ptr;
- }
-
- uint8_t shift_is_reg = (insn >> 4) & 1;
- uint8_t rotate = (insn >> 8) & 0xf;
- uint8_t rm = insn & 0xf;
- uint8_t shift_type = (insn >> 5) & 0x3;
- uint8_t rs = (insn >> 8) & 0xf;
- uint8_t shift_amount = (insn >> 7) & 0x1f;
- uint32_t rotated_val = immed;
- uint8_t rotate2 = rotate << 1;
- rotated_val = (rotated_val >> rotate2) | (rotated_val << (32 - rotate2));
-
- if (!shift_is_reg && shift_type == 0 && shift_amount == 0) {
- sprintf(ptr, "%s%s%s\t%s%sr%d",
- opname, cond_to_str(cond), sbit_str, rd_str, rn_str, rm);
- return ptr;
- }
-
- const char *shift_name = shift_names[shift_type];
- if (shift_is_reg) {
- sprintf(ptr, "%s%s%s\t%s%sr%d, %s r%d",
- opname, cond_to_str(cond), sbit_str, rd_str, rn_str, rm,
- shift_name, rs);
- return ptr;
- }
- if (shift_amount == 0) {
- if (shift_type == 3) {
- sprintf(ptr, "%s%s%s\t%s%sr%d, RRX",
- opname, cond_to_str(cond), sbit_str, rd_str, rn_str, rm);
- return ptr;
- }
- shift_amount = 32;
- }
- sprintf(ptr, "%s%s%s\t%s%sr%d, %s #%u",
- opname, cond_to_str(cond), sbit_str, rd_str, rn_str, rm,
- shift_name, shift_amount);
- return ptr;
-}
-
-char *Arm::disasm_branch(uint32_t addr, Opcode opcode, uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint32_t offset = insn & 0xffffff;
- // Sign-extend the 24-bit offset
- if ((offset >> 23) & 1)
- offset |= 0xff000000;
-
- // Pre-compute the left-shift and the prefetch offset
- offset <<= 2;
- offset += 8;
- addr += offset;
- const char *opname = opcode_names[opcode];
- sprintf(ptr, "%s%s\t0x%x", opname, cond_to_str(cond), addr);
- return ptr;
-}
-
-char *Arm::disasm_bx(uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rn = insn & 0xf;
- sprintf(ptr, "bx%s\tr%d", cond_to_str(cond), rn);
- return ptr;
-}
-
-char *Arm::disasm_bkpt(uint32_t insn, char *ptr)
-{
- uint32_t immed = (((insn >> 8) & 0xfff) << 4) | (insn & 0xf);
- sprintf(ptr, "bkpt\t#%d", immed);
- return ptr;
-}
-
-char *Arm::disasm_clz(uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rd = (insn >> 12) & 0xf;
- uint8_t rm = insn & 0xf;
- sprintf(ptr, "clz%s\tr%d, r%d", cond_to_str(cond), rd, rm);
- return ptr;
-}
-
-char *Arm::disasm_memblock(Opcode opcode, uint32_t insn, char *ptr)
-{
- char tmp_reg[10], tmp_list[80];
-
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t write_back = (insn >> 21) & 0x1;
- uint8_t bit_s = (insn >> 22) & 0x1;
- uint8_t is_up = (insn >> 23) & 0x1;
- uint8_t is_pre = (insn >> 24) & 0x1;
- uint8_t rn = (insn >> 16) & 0xf;
- uint16_t reg_list = insn & 0xffff;
-
- const char *opname = opcode_names[opcode];
-
- const char *bang = "";
- if (write_back)
- bang = "!";
-
- const char *carret = "";
- if (bit_s)
- carret = "^";
-
- const char *comma = "";
- tmp_list[0] = 0;
- for (int ii = 0; ii < 16; ++ii) {
- if (reg_list & (1 << ii)) {
- sprintf(tmp_reg, "%sr%d", comma, ii);
- strcat(tmp_list, tmp_reg);
- comma = ",";
- }
- }
-
- const char *addr_mode = "";
- if (is_pre) {
- if (is_up) {
- addr_mode = "ib";
- } else {
- addr_mode = "db";
- }
- } else {
- if (is_up) {
- addr_mode = "ia";
- } else {
- addr_mode = "da";
- }
- }
-
- sprintf(ptr, "%s%s%s\tr%d%s, {%s}%s",
- opname, cond_to_str(cond), addr_mode, rn, bang, tmp_list, carret);
- return ptr;
-}
-
-char *Arm::disasm_mem(uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t is_reg = (insn >> 25) & 0x1;
- uint8_t is_load = (insn >> 20) & 0x1;
- uint8_t write_back = (insn >> 21) & 0x1;
- uint8_t is_byte = (insn >> 22) & 0x1;
- uint8_t is_up = (insn >> 23) & 0x1;
- uint8_t is_pre = (insn >> 24) & 0x1;
- uint8_t rn = (insn >> 16) & 0xf;
- uint8_t rd = (insn >> 12) & 0xf;
- uint16_t offset = insn & 0xfff;
-
- const char *opname = "ldr";
- if (!is_load)
- opname = "str";
-
- const char *bang = "";
- if (write_back)
- bang = "!";
-
- const char *minus = "";
- if (is_up == 0)
- minus = "-";
-
- const char *byte = "";
- if (is_byte)
- byte = "b";
-
- if (is_reg == 0) {
- if (is_pre) {
- if (offset == 0) {
- sprintf(ptr, "%s%s%s\tr%d, [r%d]",
- opname, cond_to_str(cond), byte, rd, rn);
- } else {
- sprintf(ptr, "%s%s%s\tr%d, [r%d, #%s%u]%s",
- opname, cond_to_str(cond), byte, rd, rn, minus, offset, bang);
- }
- } else {
- const char *transfer = "";
- if (write_back)
- transfer = "t";
- sprintf(ptr, "%s%s%s%s\tr%d, [r%d], #%s%u",
- opname, cond_to_str(cond), byte, transfer, rd, rn, minus, offset);
- }
- return ptr;
- }
-
- uint8_t rm = insn & 0xf;
- uint8_t shift_type = (insn >> 5) & 0x3;
- uint8_t shift_amount = (insn >> 7) & 0x1f;
-
- const char *shift_name = shift_names[shift_type];
-
- if (is_pre) {
- if (shift_amount == 0) {
- if (shift_type == 0) {
- sprintf(ptr, "%s%s%s\tr%d, [r%d, %sr%d]%s",
- opname, cond_to_str(cond), byte, rd, rn, minus, rm, bang);
- return ptr;
- }
- if (shift_type == 3) {
- sprintf(ptr, "%s%s%s\tr%d, [r%d, %sr%d, RRX]%s",
- opname, cond_to_str(cond), byte, rd, rn, minus, rm, bang);
- return ptr;
- }
- shift_amount = 32;
- }
- sprintf(ptr, "%s%s%s\tr%d, [r%d, %sr%d, %s #%u]%s",
- opname, cond_to_str(cond), byte, rd, rn, minus, rm,
- shift_name, shift_amount, bang);
- return ptr;
- }
-
- const char *transfer = "";
- if (write_back)
- transfer = "t";
-
- if (shift_amount == 0) {
- if (shift_type == 0) {
- sprintf(ptr, "%s%s%s%s\tr%d, [r%d], %sr%d",
- opname, cond_to_str(cond), byte, transfer, rd, rn, minus, rm);
- return ptr;
- }
- if (shift_type == 3) {
- sprintf(ptr, "%s%s%s%s\tr%d, [r%d], %sr%d, RRX",
- opname, cond_to_str(cond), byte, transfer, rd, rn, minus, rm);
- return ptr;
- }
- shift_amount = 32;
- }
-
- sprintf(ptr, "%s%s%s%s\tr%d, [r%d], %sr%d, %s #%u",
- opname, cond_to_str(cond), byte, transfer, rd, rn, minus, rm,
- shift_name, shift_amount);
- return ptr;
-}
-
-char *Arm::disasm_memhalf(uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t is_load = (insn >> 20) & 0x1;
- uint8_t write_back = (insn >> 21) & 0x1;
- uint8_t is_immed = (insn >> 22) & 0x1;
- uint8_t is_up = (insn >> 23) & 0x1;
- uint8_t is_pre = (insn >> 24) & 0x1;
- uint8_t rn = (insn >> 16) & 0xf;
- uint8_t rd = (insn >> 12) & 0xf;
- uint8_t bits_65 = (insn >> 5) & 0x3;
- uint8_t rm = insn & 0xf;
- uint8_t offset = (((insn >> 8) & 0xf) << 4) | (insn & 0xf);
-
- const char *opname = "ldr";
- if (is_load == 0)
- opname = "str";
-
- const char *width = "";
- if (bits_65 == 1)
- width = "h";
- else if (bits_65 == 2)
- width = "sb";
- else
- width = "sh";
-
- const char *bang = "";
- if (write_back)
- bang = "!";
- const char *minus = "";
- if (is_up == 0)
- minus = "-";
-
- if (is_immed) {
- if (is_pre) {
- if (offset == 0) {
- sprintf(ptr, "%s%sh\tr%d, [r%d]", opname, cond_to_str(cond), rd, rn);
- } else {
- sprintf(ptr, "%s%sh\tr%d, [r%d, #%s%u]%s",
- opname, cond_to_str(cond), rd, rn, minus, offset, bang);
- }
- } else {
- sprintf(ptr, "%s%sh\tr%d, [r%d], #%s%u",
- opname, cond_to_str(cond), rd, rn, minus, offset);
- }
- return ptr;
- }
-
- if (is_pre) {
- sprintf(ptr, "%s%sh\tr%d, [r%d, %sr%d]%s",
- opname, cond_to_str(cond), rd, rn, minus, rm, bang);
- } else {
- sprintf(ptr, "%s%sh\tr%d, [r%d], %sr%d",
- opname, cond_to_str(cond), rd, rn, minus, rm);
- }
- return ptr;
-}
-
-char *Arm::disasm_mcr(Opcode opcode, uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t crn = (insn >> 16) & 0xf;
- uint8_t crd = (insn >> 12) & 0xf;
- uint8_t cpnum = (insn >> 8) & 0xf;
- uint8_t opcode2 = (insn >> 5) & 0x7;
- uint8_t crm = insn & 0xf;
-
- const char *opname = opcode_names[opcode];
- sprintf(ptr, "%s%s\t%d, 0, r%d, cr%d, cr%d, {%d}",
- opname, cond_to_str(cond), cpnum, crd, crn, crm, opcode2);
- return ptr;
-}
-
-char *Arm::disasm_mla(Opcode opcode, uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rd = (insn >> 16) & 0xf;
- uint8_t rn = (insn >> 12) & 0xf;
- uint8_t rs = (insn >> 8) & 0xf;
- uint8_t rm = insn & 0xf;
- uint8_t bit_s = (insn >> 20) & 1;
-
- const char *opname = opcode_names[opcode];
- sprintf(ptr, "%s%s%s\tr%d, r%d, r%d, r%d",
- opname, cond_to_str(cond), bit_s ? "s" : "", rd, rm, rs, rn);
- return ptr;
-}
-
-char *Arm::disasm_umlal(Opcode opcode, uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rdhi = (insn >> 16) & 0xf;
- uint8_t rdlo = (insn >> 12) & 0xf;
- uint8_t rs = (insn >> 8) & 0xf;
- uint8_t rm = insn & 0xf;
- uint8_t bit_s = (insn >> 20) & 1;
-
- const char *opname = opcode_names[opcode];
- sprintf(ptr, "%s%s%s\tr%d, r%d, r%d, r%d",
- opname, cond_to_str(cond), bit_s ? "s" : "", rdlo, rdhi, rm, rs);
- return ptr;
-}
-
-char *Arm::disasm_mul(Opcode opcode, uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rd = (insn >> 16) & 0xf;
- uint8_t rs = (insn >> 8) & 0xf;
- uint8_t rm = insn & 0xf;
- uint8_t bit_s = (insn >> 20) & 1;
-
- const char *opname = opcode_names[opcode];
- sprintf(ptr, "%s%s%s\tr%d, r%d, r%d",
- opname, cond_to_str(cond), bit_s ? "s" : "", rd, rm, rs);
- return ptr;
-}
-
-char *Arm::disasm_mrs(uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rd = (insn >> 12) & 0xf;
- uint8_t ps = (insn >> 22) & 1;
-
- sprintf(ptr, "mrs%s\tr%d, %s", cond_to_str(cond), rd, ps ? "spsr" : "cpsr");
- return ptr;
-}
-
-char *Arm::disasm_msr(uint32_t insn, char *ptr)
-{
- char flags[8];
- int flag_index = 0;
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t is_immed = (insn >> 25) & 0x1;
- uint8_t pd = (insn >> 22) & 1;
- uint8_t mask = (insn >> 16) & 0xf;
-
- if (mask & 1)
- flags[flag_index++] = 'c';
- if (mask & 2)
- flags[flag_index++] = 'x';
- if (mask & 4)
- flags[flag_index++] = 's';
- if (mask & 8)
- flags[flag_index++] = 'f';
- flags[flag_index] = 0;
-
- if (is_immed) {
- uint32_t immed = insn & 0xff;
- uint8_t rotate = (insn >> 8) & 0xf;
- uint8_t rotate2 = rotate << 1;
- uint32_t rotated_val = (immed >> rotate2) | (immed << (32 - rotate2));
- sprintf(ptr, "msr%s\t%s_%s, #0x%x",
- cond_to_str(cond), pd ? "spsr" : "cpsr", flags, rotated_val);
- return ptr;
- }
-
- uint8_t rm = insn & 0xf;
-
- sprintf(ptr, "msr%s\t%s_%s, r%d",
- cond_to_str(cond), pd ? "spsr" : "cpsr", flags, rm);
- return ptr;
-}
-
-char *Arm::disasm_pld(uint32_t insn, char *ptr)
-{
- uint8_t is_reg = (insn >> 25) & 0x1;
- uint8_t is_up = (insn >> 23) & 0x1;
- uint8_t rn = (insn >> 16) & 0xf;
-
- const char *minus = "";
- if (is_up == 0)
- minus = "-";
-
- if (is_reg) {
- uint8_t rm = insn & 0xf;
- sprintf(ptr, "pld\t[r%d, %sr%d]", rn, minus, rm);
- return ptr;
- }
-
- uint16_t offset = insn & 0xfff;
- if (offset == 0) {
- sprintf(ptr, "pld\t[r%d]", rn);
- } else {
- sprintf(ptr, "pld\t[r%d, #%s%u]", rn, minus, offset);
- }
- return ptr;
-}
-
-char *Arm::disasm_swi(uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint32_t sysnum = insn & 0x00ffffff;
-
- sprintf(ptr, "swi%s 0x%x", cond_to_str(cond), sysnum);
- return ptr;
-}
-
-char *Arm::disasm_swp(Opcode opcode, uint32_t insn, char *ptr)
-{
- uint8_t cond = (insn >> 28) & 0xf;
- uint8_t rn = (insn >> 16) & 0xf;
- uint8_t rd = (insn >> 12) & 0xf;
- uint8_t rm = insn & 0xf;
-
- const char *opname = opcode_names[opcode];
- sprintf(ptr, "%s%s\tr%d, r%d, [r%d]", opname, cond_to_str(cond), rd, rm, rn);
- return ptr;
-}
-
-Opcode Arm::decode(uint32_t insn) {
- uint32_t bits27_26 = (insn >> 26) & 0x3;
- switch (bits27_26) {
- case 0x0:
- return decode00(insn);
- case 0x1:
- return decode01(insn);
- case 0x2:
- return decode10(insn);
- case 0x3:
- return decode11(insn);
- }
- return OP_INVALID;
-}
-
-Opcode Arm::decode00(uint32_t insn) {
- uint8_t bit25 = (insn >> 25) & 0x1;
- uint8_t bit4 = (insn >> 4) & 0x1;
- if (bit25 == 0 && bit4 == 1) {
- if ((insn & 0x0ffffff0) == 0x012fff10) {
- // Bx instruction
- return OP_BX;
- }
- if ((insn & 0x0ff000f0) == 0x01600010) {
- // Clz instruction
- return OP_CLZ;
- }
- if ((insn & 0xfff000f0) == 0xe1200070) {
- // Bkpt instruction
- return OP_BKPT;
- }
- uint32_t bits7_4 = (insn >> 4) & 0xf;
- if (bits7_4 == 0x9) {
- if ((insn & 0x0ff00ff0) == 0x01000090) {
- // Swp instruction
- uint8_t bit22 = (insn >> 22) & 0x1;
- if (bit22)
- return OP_SWPB;
- return OP_SWP;
- }
- // One of the multiply instructions
- return decode_mul(insn);
- }
-
- uint8_t bit7 = (insn >> 7) & 0x1;
- if (bit7 == 1) {
- // One of the load/store halfword/byte instructions
- return decode_ldrh(insn);
- }
- }
-
- // One of the data processing instructions
- return decode_alu(insn);
-}
-
-Opcode Arm::decode01(uint32_t insn) {
- uint8_t is_reg = (insn >> 25) & 0x1;
- uint8_t bit4 = (insn >> 4) & 0x1;
- if (is_reg == 1 && bit4 == 1)
- return OP_UNDEFINED;
- uint8_t is_load = (insn >> 20) & 0x1;
- uint8_t is_byte = (insn >> 22) & 0x1;
- if ((insn & 0xfd70f000) == 0xf550f000) {
- // Pre-load
- return OP_PLD;
- }
- if (is_load) {
- if (is_byte) {
- // Load byte
- return OP_LDRB;
- }
- // Load word
- return OP_LDR;
- }
- if (is_byte) {
- // Store byte
- return OP_STRB;
- }
- // Store word
- return OP_STR;
-}
-
-Opcode Arm::decode10(uint32_t insn) {
- uint8_t bit25 = (insn >> 25) & 0x1;
- if (bit25 == 0) {
- // LDM/STM
- uint8_t is_load = (insn >> 20) & 0x1;
- if (is_load)
- return OP_LDM;
- return OP_STM;
- }
- // Branch or Branch with link
- uint8_t is_link = (insn >> 24) & 1;
- uint32_t offset = insn & 0xffffff;
-
- // Sign-extend the 24-bit offset
- if ((offset >> 23) & 1)
- offset |= 0xff000000;
-
- // Pre-compute the left-shift and the prefetch offset
- offset <<= 2;
- offset += 8;
- if (is_link == 0)
- return OP_B;
- return OP_BL;
-}
-
-Opcode Arm::decode11(uint32_t insn) {
- uint8_t bit25 = (insn >> 25) & 0x1;
- if (bit25 == 0) {
- // LDC, SDC
- uint8_t is_load = (insn >> 20) & 0x1;
- if (is_load) {
- // LDC
- return OP_LDC;
- }
- // STC
- return OP_STC;
- }
-
- uint8_t bit24 = (insn >> 24) & 0x1;
- if (bit24 == 0x1) {
- // SWI
- return OP_SWI;
- }
-
- uint8_t bit4 = (insn >> 4) & 0x1;
- uint8_t cpnum = (insn >> 8) & 0xf;
-
- if (cpnum == 15) {
- // Special case for coprocessor 15
- uint8_t opcode = (insn >> 21) & 0x7;
- if (bit4 == 0 || opcode != 0) {
- // This is an unexpected bit pattern. Create an undefined
- // instruction in case this is ever executed.
- return OP_UNDEFINED;
- }
-
- // MRC, MCR
- uint8_t is_mrc = (insn >> 20) & 0x1;
- if (is_mrc)
- return OP_MRC;
- return OP_MCR;
- }
-
- if (bit4 == 0) {
- // CDP
- return OP_CDP;
- }
- // MRC, MCR
- uint8_t is_mrc = (insn >> 20) & 0x1;
- if (is_mrc)
- return OP_MRC;
- return OP_MCR;
-}
-
-Opcode Arm::decode_mul(uint32_t insn) {
- uint8_t bit24 = (insn >> 24) & 0x1;
- if (bit24 != 0) {
- // This is an unexpected bit pattern. Create an undefined
- // instruction in case this is ever executed.
- return OP_UNDEFINED;
- }
- uint8_t bit23 = (insn >> 23) & 0x1;
- uint8_t bit22_U = (insn >> 22) & 0x1;
- uint8_t bit21_A = (insn >> 21) & 0x1;
- if (bit23 == 0) {
- // 32-bit multiply
- if (bit22_U != 0) {
- // This is an unexpected bit pattern. Create an undefined
- // instruction in case this is ever executed.
- return OP_UNDEFINED;
- }
- if (bit21_A == 0)
- return OP_MUL;
- return OP_MLA;
- }
- // 64-bit multiply
- if (bit22_U == 0) {
- // Unsigned multiply long
- if (bit21_A == 0)
- return OP_UMULL;
- return OP_UMLAL;
- }
- // Signed multiply long
- if (bit21_A == 0)
- return OP_SMULL;
- return OP_SMLAL;
-}
-
-Opcode Arm::decode_ldrh(uint32_t insn) {
- uint8_t is_load = (insn >> 20) & 0x1;
- uint8_t bits_65 = (insn >> 5) & 0x3;
- if (is_load) {
- if (bits_65 == 0x1) {
- // Load unsigned halfword
- return OP_LDRH;
- } else if (bits_65 == 0x2) {
- // Load signed byte
- return OP_LDRSB;
- }
- // Signed halfword
- if (bits_65 != 0x3) {
- // This is an unexpected bit pattern. Create an undefined
- // instruction in case this is ever executed.
- return OP_UNDEFINED;
- }
- // Load signed halfword
- return OP_LDRSH;
- }
- // Store halfword
- if (bits_65 != 0x1) {
- // This is an unexpected bit pattern. Create an undefined
- // instruction in case this is ever executed.
- return OP_UNDEFINED;
- }
- // Store halfword
- return OP_STRH;
-}
-
-Opcode Arm::decode_alu(uint32_t insn) {
- uint8_t is_immed = (insn >> 25) & 0x1;
- uint8_t opcode = (insn >> 21) & 0xf;
- uint8_t bit_s = (insn >> 20) & 1;
- uint8_t shift_is_reg = (insn >> 4) & 1;
- uint8_t bit7 = (insn >> 7) & 1;
- if (!is_immed && shift_is_reg && (bit7 != 0)) {
- // This is an unexpected bit pattern. Create an undefined
- // instruction in case this is ever executed.
- return OP_UNDEFINED;
- }
- switch (opcode) {
- case 0x0:
- return OP_AND;
- case 0x1:
- return OP_EOR;
- case 0x2:
- return OP_SUB;
- case 0x3:
- return OP_RSB;
- case 0x4:
- return OP_ADD;
- case 0x5:
- return OP_ADC;
- case 0x6:
- return OP_SBC;
- case 0x7:
- return OP_RSC;
- case 0x8:
- if (bit_s)
- return OP_TST;
- return OP_MRS;
- case 0x9:
- if (bit_s)
- return OP_TEQ;
- return OP_MSR;
- case 0xa:
- if (bit_s)
- return OP_CMP;
- return OP_MRS;
- case 0xb:
- if (bit_s)
- return OP_CMN;
- return OP_MSR;
- case 0xc:
- return OP_ORR;
- case 0xd:
- return OP_MOV;
- case 0xe:
- return OP_BIC;
- case 0xf:
- return OP_MVN;
- }
- // Unreachable
- return OP_INVALID;
-}
diff --git a/emulator/qtools/armdis.h b/emulator/qtools/armdis.h
deleted file mode 100644
index 230f833..0000000
--- a/emulator/qtools/armdis.h
+++ /dev/null
@@ -1,45 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef ARMDIS_H
-#define ARMDIS_H
-
-#include <inttypes.h>
-#include "opcode.h"
-
-class Arm {
- public:
- static char *disasm(uint32_t addr, uint32_t insn, char *buffer);
- static Opcode decode(uint32_t insn);
-
- private:
- static Opcode decode00(uint32_t insn);
- static Opcode decode01(uint32_t insn);
- static Opcode decode10(uint32_t insn);
- static Opcode decode11(uint32_t insn);
- static Opcode decode_mul(uint32_t insn);
- static Opcode decode_ldrh(uint32_t insn);
- static Opcode decode_alu(uint32_t insn);
-
- static char *disasm_alu(Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_branch(uint32_t addr, Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_bx(uint32_t insn, char *ptr);
- static char *disasm_bkpt(uint32_t insn, char *ptr);
- static char *disasm_clz(uint32_t insn, char *ptr);
- static char *disasm_memblock(Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_mem(uint32_t insn, char *ptr);
- static char *disasm_memhalf(uint32_t insn, char *ptr);
- static char *disasm_mcr(Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_mla(Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_umlal(Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_mul(Opcode opcode, uint32_t insn, char *ptr);
- static char *disasm_mrs(uint32_t insn, char *ptr);
- static char *disasm_msr(uint32_t insn, char *ptr);
- static char *disasm_pld(uint32_t insn, char *ptr);
- static char *disasm_swi(uint32_t insn, char *ptr);
- static char *disasm_swp(Opcode opcode, uint32_t insn, char *ptr);
-};
-
-extern char *disasm_insn_thumb(uint32_t pc, uint32_t insn1, uint32_t insn2, char *result);
-extern Opcode decode_insn_thumb(uint32_t given);
-
-#endif /* ARMDIS_H */
diff --git a/emulator/qtools/bb2sym.cpp b/emulator/qtools/bb2sym.cpp
deleted file mode 100644
index 8a18b67..0000000
--- a/emulator/qtools/bb2sym.cpp
+++ /dev/null
@@ -1,140 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-struct MyStaticRec {
- StaticRec bb;
- symbol_type *sym;
- MyStaticRec *inner; // pointer to an inner basic block
- int is_thumb;
-};
-
-MyStaticRec **assign_inner_blocks(int num_blocks, MyStaticRec *blocks);
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file elf_file\n", program);
- OptionsUsage();
-}
-
-// This function is called from quicksort to compare addresses of basic
-// blocks.
-int cmp_inc_addr(const void *a, const void *b) {
- MyStaticRec *bb1, *bb2;
-
- bb1 = *(MyStaticRec**)a;
- bb2 = *(MyStaticRec**)b;
- if (bb1->bb.bb_addr < bb2->bb.bb_addr)
- return -1;
- if (bb1->bb.bb_addr > bb2->bb.bb_addr)
- return 1;
- return bb1->bb.bb_num - bb2->bb.bb_num;
-}
-
-int main(int argc, char **argv) {
- uint32_t insns[kMaxInsnPerBB];
-
- // Parse the options
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReader<> *trace = new TraceReader<>;
- trace->Open(trace_filename);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- TraceHeader *header = trace->GetHeader();
- uint32_t num_static_bb = header->num_static_bb;
-
- // Allocate space for all of the static blocks
- MyStaticRec *blocks = new MyStaticRec[num_static_bb];
-
- // Read in all the static blocks
- for (uint32_t ii = 0; ii < num_static_bb; ++ii) {
- trace->ReadStatic(&blocks[ii].bb);
- blocks[ii].is_thumb = blocks[ii].bb.bb_addr & 1;
- blocks[ii].bb.bb_addr &= ~1;
- blocks[ii].sym = NULL;
- blocks[ii].inner = NULL;
- trace->ReadStaticInsns(blocks[ii].bb.num_insns, insns);
- }
-
- MyStaticRec **sorted = assign_inner_blocks(num_static_bb, blocks);
-
- while (1) {
- symbol_type *sym;
- BBEvent event;
- BBEvent ignored;
-
- if (GetNextValidEvent(trace, &event, &ignored, &sym))
- break;
-
- uint64_t bb_num = event.bb_num;
- blocks[bb_num].sym = sym;
- }
-
- printf("# bb num_insns bb_addr file symbol\n");
- for (uint32_t ii = 0; ii < num_static_bb; ++ii) {
- if (sorted[ii]->bb.bb_addr == 0 || sorted[ii]->bb.num_insns == 0
- || sorted[ii]->sym == NULL)
- continue;
-
- printf("%8lld %3d 0x%08x %s %s\n",
- sorted[ii]->bb.bb_num, sorted[ii]->bb.num_insns,
- sorted[ii]->bb.bb_addr, sorted[ii]->sym->region->path,
- sorted[ii]->sym->name);
- }
- return 0;
-}
-
-// Find the basic blocks that are subsets of other basic blocks.
-MyStaticRec **assign_inner_blocks(int num_blocks, MyStaticRec *blocks)
-{
- int ii;
- uint32_t addr_end, addr_diff;
-
- // Create a list of pointers to the basic blocks that we can sort.
- MyStaticRec **sorted = new MyStaticRec*[num_blocks];
- for (ii = 0; ii < num_blocks; ++ii) {
- sorted[ii] = &blocks[ii];
- }
-
- // Sort the basic blocks into increasing address order
- qsort(sorted, num_blocks, sizeof(MyStaticRec*), cmp_inc_addr);
-
- // Create pointers to inner blocks and break up the enclosing block
- // so that there is no overlap.
- for (ii = 0; ii < num_blocks - 1; ++ii) {
- int num_bytes;
- if (sorted[ii]->is_thumb)
- num_bytes = sorted[ii]->bb.num_insns << 1;
- else
- num_bytes = sorted[ii]->bb.num_insns << 2;
- addr_end = sorted[ii]->bb.bb_addr + num_bytes;
- if (addr_end > sorted[ii + 1]->bb.bb_addr) {
- sorted[ii]->inner = sorted[ii + 1];
- addr_diff = sorted[ii + 1]->bb.bb_addr - sorted[ii]->bb.bb_addr;
- uint32_t num_insns;
- if (sorted[ii]->is_thumb)
- num_insns = addr_diff >> 1;
- else
- num_insns = addr_diff >> 2;
- sorted[ii]->bb.num_insns = num_insns;
- }
- }
-
- return sorted;
-}
diff --git a/emulator/qtools/bb_dump.cpp b/emulator/qtools/bb_dump.cpp
deleted file mode 100644
index de241fb..0000000
--- a/emulator/qtools/bb_dump.cpp
+++ /dev/null
@@ -1,47 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file elf_file\n", program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv) {
- // Parse the options
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReader<> *trace = new TraceReader<>;
- trace->Open(trace_filename);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- printf("# time bb pid num_insns bb_addr\n");
- while (1) {
- symbol_type *sym;
- BBEvent event;
- BBEvent ignored;
-
- if (GetNextValidEvent(trace, &event, &ignored, &sym))
- break;
- printf("%7lld %4lld %5d %3d 0x%08x %s\n",
- event.time, event.bb_num, event.pid, event.num_insns,
- event.bb_addr, sym->name);
- }
- return 0;
-}
diff --git a/emulator/qtools/bbprof.cpp b/emulator/qtools/bbprof.cpp
deleted file mode 100644
index 36d0941..0000000
--- a/emulator/qtools/bbprof.cpp
+++ /dev/null
@@ -1,222 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-#include "armdis.h"
-
-struct MyStaticRec {
- StaticRec bb;
- uint32_t *insns;
- uint32_t *cycles; // number of cycles for each insn
- uint32_t elapsed; // number of cycles for basic block
- int freq; // execution frequency
- MyStaticRec *inner; // pointer to an inner basic block
- int is_thumb;
-};
-
-MyStaticRec **assign_inner_blocks(int num_blocks, MyStaticRec *blocks);
-
-// This function is called from quicksort to compare addresses of basic
-// blocks.
-int cmp_inc_addr(const void *a, const void *b) {
- MyStaticRec *bb1, *bb2;
-
- bb1 = *(MyStaticRec**)a;
- bb2 = *(MyStaticRec**)b;
- if (bb1->bb.bb_addr < bb2->bb.bb_addr)
- return -1;
- if (bb1->bb.bb_addr > bb2->bb.bb_addr)
- return 1;
- return bb1->bb.bb_num - bb2->bb.bb_num;
-}
-
-// This function is called from quicksort to compare the elapsed time
-// of basic blocks.
-int cmp_dec_elapsed(const void *a, const void *b) {
- MyStaticRec *bb1, *bb2;
-
- bb1 = *(MyStaticRec**)a;
- bb2 = *(MyStaticRec**)b;
- if (bb1->elapsed < bb2->elapsed)
- return 1;
- if (bb1->elapsed > bb2->elapsed)
- return -1;
- return bb1->bb.bb_num - bb2->bb.bb_num;
-}
-
-// This function is called from quicksort to compare frequencies of
-// basic blocks.
-int cmp_dec_freq(const void *a, const void *b) {
- MyStaticRec *bb1, *bb2;
-
- bb1 = *(MyStaticRec**)a;
- bb2 = *(MyStaticRec**)b;
- if (bb1->freq < bb2->freq)
- return 1;
- if (bb1->freq > bb2->freq)
- return -1;
- return bb1->bb.bb_num - bb2->bb.bb_num;
-}
-
-int main(int argc, char **argv)
-{
- if (argc != 2) {
- fprintf(stderr, "Usage: %s trace_file\n", argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[1];
- TraceReaderBase *trace = new TraceReaderBase;
- trace->Open(trace_filename);
- TraceHeader *header = trace->GetHeader();
- uint32_t num_static_bb = header->num_static_bb;
-
- // Allocate space for all of the static blocks
- MyStaticRec *blocks = new MyStaticRec[num_static_bb];
-
- // Read in all the static blocks
- for (uint32_t ii = 0; ii < num_static_bb; ++ii) {
- trace->ReadStatic(&blocks[ii].bb);
- blocks[ii].is_thumb = blocks[ii].bb.bb_addr & 1;
- blocks[ii].bb.bb_addr &= ~1;
- uint32_t num_insns = blocks[ii].bb.num_insns;
- blocks[ii].insns = new uint32_t[num_insns];
- blocks[ii].cycles = new uint32_t[num_insns];
- memset(blocks[ii].cycles, 0, num_insns * sizeof(uint32_t));
- trace->ReadStaticInsns(num_insns, blocks[ii].insns);
- blocks[ii].elapsed = 0;
- blocks[ii].freq = 0;
- blocks[ii].inner = NULL;
- }
-
- MyStaticRec **sorted = assign_inner_blocks(num_static_bb, blocks);
-
- uint32_t prev_time = 0;
- uint32_t elapsed = 0;
- uint32_t dummy;
- uint32_t *cycle_ptr = &dummy;
- uint32_t *bb_elapsed_ptr = &dummy;
- while (1) {
- BBEvent event;
-
- if (trace->ReadBB(&event))
- break;
- // Assign frequencies to each basic block
- uint64_t bb_num = event.bb_num;
- int num_insns = event.num_insns;
- blocks[bb_num].freq += 1;
- for (MyStaticRec *bptr = blocks[bb_num].inner; bptr; bptr = bptr->inner)
- bptr->freq += 1;
-
- // Assign simulation time to each instruction
- for (MyStaticRec *bptr = &blocks[bb_num]; bptr; bptr = bptr->inner) {
- uint32_t bb_num_insns = bptr->bb.num_insns;
- for (uint32_t ii = 0; num_insns && ii < bb_num_insns; ++ii, --num_insns) {
- uint32_t sim_time = trace->ReadInsnTime(event.time);
- elapsed = sim_time - prev_time;
- prev_time = sim_time;
-
- // Attribute the elapsed time to the previous instruction and
- // basic block.
- *cycle_ptr += elapsed;
- *bb_elapsed_ptr += elapsed;
- cycle_ptr = &bptr->cycles[ii];
- bb_elapsed_ptr = &bptr->elapsed;
- }
- }
- }
- *cycle_ptr += 1;
- *bb_elapsed_ptr += 1;
-
- // Sort the basic blocks into decreasing elapsed time
- qsort(sorted, num_static_bb, sizeof(MyStaticRec*), cmp_dec_elapsed);
-
- char spaces[80];
- memset(spaces, ' ', 79);
- spaces[79] = 0;
- for (uint32_t ii = 0; ii < num_static_bb; ++ii) {
- printf("bb %lld addr: 0x%x, insns: %d freq: %u elapsed: %u\n",
- sorted[ii]->bb.bb_num, sorted[ii]->bb.bb_addr,
- sorted[ii]->bb.num_insns, sorted[ii]->freq,
- sorted[ii]->elapsed);
- int num_insns = sorted[ii]->bb.num_insns;
- uint32_t addr = sorted[ii]->bb.bb_addr;
- for (int jj = 0; jj < num_insns; ++jj) {
- uint32_t elapsed = sorted[ii]->cycles[jj];
- uint32_t insn = sorted[ii]->insns[jj];
- if (insn_is_thumb(insn)) {
- insn = insn_unwrap_thumb(insn);
-
- // thumb_pair is true if this is the first of a pair of
- // thumb instructions (BL or BLX).
- bool thumb_pair = ((insn & 0xf800) == 0xf000);
-
- // Get the next thumb instruction (if any) because we may need
- // it for the case where insn is BL or BLX.
- uint32_t insn2 = 0;
- if (thumb_pair && (jj + 1 < num_insns)) {
- insn2 = sorted[ii]->insns[jj + 1];
- insn2 = insn_unwrap_thumb(insn2);
- jj += 1;
- }
- char *disasm = disasm_insn_thumb(addr, insn, insn2, NULL);
- if (thumb_pair) {
- printf(" %4u %08x %04x %04x %s\n", elapsed, addr, insn,
- insn2, disasm);
- addr += 2;
- } else {
- printf(" %4u %08x %04x %s\n", elapsed, addr, insn,
- disasm);
- }
- addr += 2;
- } else {
- char *disasm = Arm::disasm(addr, insn, NULL);
- printf(" %4u %08x %08x %s\n", elapsed, addr, insn, disasm);
- addr += 4;
- }
- }
- }
-
- delete[] sorted;
- return 0;
-}
-
-// Find the basic blocks that are subsets of other basic blocks.
-MyStaticRec **assign_inner_blocks(int num_blocks, MyStaticRec *blocks)
-{
- int ii;
- uint32_t addr_end, addr_diff;
-
- // Create a list of pointers to the basic blocks that we can sort.
- MyStaticRec **sorted = new MyStaticRec*[num_blocks];
- for (ii = 0; ii < num_blocks; ++ii) {
- sorted[ii] = &blocks[ii];
- }
-
- // Sort the basic blocks into increasing address order
- qsort(sorted, num_blocks, sizeof(MyStaticRec*), cmp_inc_addr);
-
- // Create pointers to inner blocks and break up the enclosing block
- // so that there is no overlap.
- for (ii = 0; ii < num_blocks - 1; ++ii) {
- int num_bytes;
- if (sorted[ii]->is_thumb)
- num_bytes = sorted[ii]->bb.num_insns << 1;
- else
- num_bytes = sorted[ii]->bb.num_insns << 2;
- addr_end = sorted[ii]->bb.bb_addr + num_bytes;
- if (addr_end > sorted[ii + 1]->bb.bb_addr) {
- sorted[ii]->inner = sorted[ii + 1];
- addr_diff = sorted[ii + 1]->bb.bb_addr - sorted[ii]->bb.bb_addr;
- uint32_t num_insns;
- if (sorted[ii]->is_thumb)
- num_insns = addr_diff >> 1;
- else
- num_insns = addr_diff >> 2;
- sorted[ii]->bb.num_insns = num_insns;
- }
- }
-
- return sorted;
-}
diff --git a/emulator/qtools/bitvector.h b/emulator/qtools/bitvector.h
deleted file mode 100644
index d3f5cf1..0000000
--- a/emulator/qtools/bitvector.h
+++ /dev/null
@@ -1,40 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef BITVECTOR_H
-#define BITVECTOR_H
-
-#include <inttypes.h>
-#include <assert.h>
-
-class Bitvector {
- public:
- explicit Bitvector(int num_bits) {
- num_bits_ = num_bits;
-
- // Round up to a multiple of 32
- num_bits = (num_bits + 31) & ~31;
- vector_ = new uint32_t[num_bits >> 5];
- }
- ~Bitvector() {
- delete[] vector_;
- }
-
- void SetBit(int bitnum) {
- assert(bitnum < num_bits_);
- vector_[bitnum >> 5] |= 1 << (bitnum & 31);
- }
- void ClearBit(int bitnum) {
- assert(bitnum < num_bits_);
- vector_[bitnum >> 5] &= ~(1 << (bitnum & 31));
- }
- bool GetBit(int bitnum) {
- assert(bitnum < num_bits_);
- return (vector_[bitnum >> 5] >> (bitnum & 31)) & 1;
- }
-
- private:
- int num_bits_;
- uint32_t *vector_;
-};
-
-#endif // BITVECTOR_H
diff --git a/emulator/qtools/callstack.h b/emulator/qtools/callstack.h
deleted file mode 100644
index 8982330..0000000
--- a/emulator/qtools/callstack.h
+++ /dev/null
@@ -1,775 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef CALL_STACK_H
-#define CALL_STACK_H
-
-#include "opcode.h"
-#include "armdis.h"
-
-class CallStackBase {
- public:
- int getId() { return mId; }
- void setId(int id) { mId = id; }
-
- private:
- int mId;
-};
-
-// Define a template class for the stack frame. The template parameter
-// SYM is the symbol_type from the TraceReader<> template class. To
-// use the CallStack class, the user derives a subclass of StackFrame
-// and defines push() and pop() methods. This derived class is then
-// passed as a template parameter to CallStack.
-template <class SYM>
-class StackFrame {
- public:
-
- virtual ~StackFrame() {};
-
- virtual void push(int stackLevel, uint64_t time, CallStackBase *base) {};
- virtual void pop(int stackLevel, uint64_t time, CallStackBase *base) {};
-
- typedef SYM symbol_type;
- static const uint32_t kCausedException = 0x01;
- static const uint32_t kInterpreted = 0x02;
- static const uint32_t kStartNative = 0x04;
- static const uint32_t kPopBarrier = (kCausedException | kInterpreted
- | kStartNative);
-
- symbol_type *function; // the symbol for the function we entered
- uint32_t addr; // return address when this function returns
- uint32_t flags;
- uint32_t time; // for debugging when a problem occurred
- uint32_t global_time; // for debugging when a problem occurred
-};
-
-template <class FRAME, class BASE = CallStackBase>
-class CallStack : public BASE {
-public:
- typedef FRAME frame_type;
- typedef typename FRAME::symbol_type symbol_type;
- typedef typename FRAME::symbol_type::region_type region_type;
- typedef BASE base_type;
-
- CallStack(int id, int numFrames, TraceReaderType *trace);
- ~CallStack();
-
- void updateStack(BBEvent *event, symbol_type *function);
- void popAll(uint64_t time);
- void threadStart(uint64_t time);
- void threadStop(uint64_t time);
-
- // Set to true if you don't want to see any Java methods ever
- void setNativeOnly(bool nativeOnly) {
- mNativeOnly = nativeOnly;
- }
-
- int getStackLevel() { return mTop; }
-
- uint64_t getGlobalTime(uint64_t time) { return time + mSkippedTime; }
- void showStack(FILE *stream);
-
- int mNumFrames;
- FRAME *mFrames;
- int mTop; // index of the next stack frame to write
-
-private:
- enum Action { NONE, PUSH, POP, NATIVE_PUSH };
-
- Action getAction(BBEvent *event, symbol_type *function);
- void doMethodAction(BBEvent *event, symbol_type *function);
- void doMethodPop(BBEvent *event, uint32_t addr, const uint32_t flags);
- void doSimplePush(symbol_type *function, uint32_t addr,
- uint64_t time, int flags);
- void doSimplePop(uint64_t time);
- void doPush(BBEvent *event, symbol_type *function);
- void doPop(BBEvent *event, symbol_type *function, Action methodAction);
-
- TraceReaderType *mTrace;
-
- // This is a global switch that disables Java methods from appearing
- // on the stack.
- bool mNativeOnly;
-
- // This keeps track of whether native frames are currently allowed on the
- // stack.
- bool mAllowNativeFrames;
-
- symbol_type mDummyFunction;
- region_type mDummyRegion;
-
- symbol_type *mPrevFunction;
- BBEvent mPrevEvent;
-
- symbol_type *mUserFunction;
- BBEvent mUserEvent; // the previous user-mode event
-
- uint64_t mSkippedTime;
- uint64_t mLastRunTime;
-
- static MethodRec sCurrentMethod;
- static MethodRec sNextMethod;
-};
-
-template<class FRAME, class BASE>
-MethodRec CallStack<FRAME, BASE>::sCurrentMethod;
-template<class FRAME, class BASE>
-MethodRec CallStack<FRAME, BASE>::sNextMethod;
-
-template<class FRAME, class BASE>
-CallStack<FRAME, BASE>::CallStack(int id, int numFrames, TraceReaderType *trace)
-{
- mNativeOnly = false;
- mTrace = trace;
- BASE::setId(id);
- mNumFrames = numFrames;
- mFrames = new FRAME[mNumFrames];
- mTop = 0;
- mAllowNativeFrames = true;
-
- memset(&mDummyFunction, 0, sizeof(symbol_type));
- memset(&mDummyRegion, 0, sizeof(region_type));
- mDummyFunction.region = &mDummyRegion;
- mPrevFunction = &mDummyFunction;
- memset(&mPrevEvent, 0, sizeof(BBEvent));
- mUserFunction = &mDummyFunction;
- memset(&mUserEvent, 0, sizeof(BBEvent));
- mSkippedTime = 0;
- mLastRunTime = 0;
-
- // Read the first two methods from the trace if we haven't already read
- // from the method trace yet.
- if (sCurrentMethod.time == 0) {
- if (mTrace->ReadMethod(&sCurrentMethod)) {
- sCurrentMethod.time = ~0ull;
- sNextMethod.time = ~0ull;
- }
- if (sNextMethod.time != ~0ull && mTrace->ReadMethod(&sNextMethod)) {
- sNextMethod.time = ~0ull;
- }
- }
-}
-
-template<class FRAME, class BASE>
-CallStack<FRAME, BASE>::~CallStack()
-{
- delete mFrames;
-}
-
-template<class FRAME, class BASE>
-void
-CallStack<FRAME, BASE>::updateStack(BBEvent *event, symbol_type *function)
-{
- if (mNativeOnly) {
- // If this is an interpreted function, then use the native VM function
- // instead.
- if (function->vm_sym != NULL)
- function = function->vm_sym;
- } else {
- doMethodAction(event, function);
- }
-
- Action action = getAction(event, function);
-
- // Allow native frames if we are executing in the kernel.
- if (!mAllowNativeFrames
- && (function->region->flags & region_type::kIsKernelRegion) == 0) {
- action = NONE;
- }
-
- if (function->vm_sym != NULL) {
- function = function->vm_sym;
- function->vm_sym = NULL;
- }
- if (action == PUSH) {
- doPush(event, function);
- } else if (action == POP) {
- doPop(event, function, NONE);
- }
-
-#if 0
- // Pop off native functions before pushing or popping Java methods.
- if (action == POP && mPrevFunction->vm_sym == NULL) {
- // Pop off the previous function first.
- doPop(event, function, NONE);
- if (methodAction == POP) {
- doPop(event, function, POP);
- } else if (methodAction == PUSH) {
- doPush(event, function);
- }
- } else {
- if (methodAction != NONE) {
- // If the method trace has a push or pop, then do it.
- action = methodAction;
- } else if (function->vm_sym != NULL) {
- // This function is a Java method. Don't push or pop the
- // Java method without a corresponding method trace record.
- action = NONE;
- }
- if (action == POP) {
- doPop(event, function, methodAction);
- } else if (action == PUSH) {
- doPush(event, function);
- }
- }
-#endif
-
- // If the stack is now empty, then push the current function.
- if (mTop == 0) {
- uint64_t time = event->time - mSkippedTime;
- int flags = 0;
- if (function->vm_sym != NULL) {
- flags = FRAME::kInterpreted;
- }
- doSimplePush(function, 0, time, 0);
- }
-
- mPrevFunction = function;
- mPrevEvent = *event;
-}
-
-template<class FRAME, class BASE>
-void
-CallStack<FRAME, BASE>::threadStart(uint64_t time)
-{
- mSkippedTime += time - mLastRunTime;
-}
-
-template<class FRAME, class BASE>
-void
-CallStack<FRAME, BASE>::threadStop(uint64_t time)
-{
- mLastRunTime = time;
-}
-
-template<class FRAME, class BASE>
-typename CallStack<FRAME, BASE>::Action
-CallStack<FRAME, BASE>::getAction(BBEvent *event, symbol_type *function)
-{
- Action action;
- uint32_t offset;
-
- // Compute the offset from the start of the function to this basic
- // block address.
- offset = event->bb_addr - function->addr - function->region->base_addr;
-
- // Get the previously executed instruction
- Opcode op = OP_INVALID;
- int numInsns = mPrevEvent.num_insns;
- uint32_t insn = 0;
- if (numInsns > 0) {
- insn = mPrevEvent.insns[numInsns - 1];
- if (mPrevEvent.is_thumb) {
- insn = insn_unwrap_thumb(insn);
- op = decode_insn_thumb(insn);
- } else {
- op = Arm::decode(insn);
- }
- }
-
- // The number of bytes in the previous basic block depends on
- // whether the basic block was ARM or THUMB instructions.
- int numBytes;
- if (mPrevEvent.is_thumb) {
- numBytes = numInsns << 1;
- } else {
- numBytes = numInsns << 2;
- }
-
- // If this basic block follows the previous one, then return NONE.
- // If we don't do this, then we may be fooled into thinking this
- // is a POP if the previous block ended with a conditional
- // (non-executed) ldmia instruction. We do this check before
- // checking if we are in a different function because we otherwise
- // we might be fooled into thinking this is a PUSH to a new function
- // when it is really just a fall-thru into a local kernel symbol
- // that just looks like a new function.
- uint32_t prev_end_addr = mPrevEvent.bb_addr + numBytes;
- if (prev_end_addr == event->bb_addr) {
- return NONE;
- }
-
- // If this basic block is in the same function as the last basic block,
- // then just return NONE (but see the exceptions below).
- // Exception 1: if the function calls itself (offset == 0) then we
- // want to push this function.
- // Exception 2: if the function returns to itself, then we want
- // to pop this function. We detect this case by checking if the last
- // instruction in the previous basic block was a load-multiple (ldm)
- // and included r15 as one of the loaded registers.
- if (function == mPrevFunction) {
- if (numInsns > 0) {
- // If this is the beginning of the function and the previous
- // instruction was not a branch, then it's a PUSH.
- if (offset == 0 && op != OP_B && op != OP_THUMB_B)
- return PUSH;
-
- // If the previous instruction was an ldm that loaded r15,
- // then it's a POP.
- if (offset != 0 && ((op == OP_LDM && (insn & 0x8000))
- || (op == OP_THUMB_POP && (insn & 0x100)))) {
- return POP;
- }
- }
-
- return NONE;
- }
-
- // We have to figure out if this new function is a call or a
- // return. We don't necessarily have a complete call stack (since
- // we could have started tracing at any point), so we have to use
- // heuristics. If the address we are jumping to is the beginning
- // of a function, or if the instruction that took us there was
- // either "bl" or "blx" then this is a PUSH. Also, if the
- // function offset is non-zero and the previous instruction is a
- // branch instruction, we will call it a PUSH. This happens in
- // the kernel a lot when there is a branch to an offset from a
- // label. A couple more special cases:
- //
- // - entering a .plt section ("procedure linkage table") is a PUSH,
- // - an exception that jumps into the kernel vector entry point
- // is also a push.
- //
- // If the function offset is non-zero and the previous instruction
- // is a bx or some non-branch instruction, then it's a POP.
- //
- // There's another special case that comes up. The user code
- // might execute an instruction that returns but before the pc
- // starts executing in the caller, a kernel interrupt occurs.
- // But it may be hard to tell if this is a return until after
- // the kernel interrupt code is done and returns to user space.
- // So we save the last user basic block and look at it when
- // we come back into user space.
-
- const uint32_t kIsKernelRegion = region_type::kIsKernelRegion;
-
- if (((mPrevFunction->region->flags & kIsKernelRegion) == 0)
- && (function->region->flags & kIsKernelRegion)) {
- // We just switched into the kernel. Save the previous
- // user-mode basic block and function.
- mUserEvent = mPrevEvent;
- mUserFunction = mPrevFunction;
- } else if ((mPrevFunction->region->flags & kIsKernelRegion)
- && ((function->region->flags & kIsKernelRegion) == 0)) {
- // We just switched from kernel to user mode.
- return POP;
- }
-
- action = PUSH;
- if (offset != 0 && mPrevFunction != &mDummyFunction) {
- // We are jumping into the middle of a function, so this is
- // probably a return, not a function call. But look at the
- // previous instruction first to see if it was a branch-and-link.
-
- // If the previous instruction was not a branch (and not a
- // branch-and-link) then POP; or if it is a "bx" instruction
- // then POP because that is used to return from functions.
- if (!isBranch(op) || op == OP_BX || op == OP_THUMB_BX) {
- action = POP;
- } else if (isBranch(op) && !isBranchLink(op)) {
- // If the previous instruction was a normal branch to a
- // local symbol then don't count it as a push or a pop.
- action = NONE;
- }
-
- if (function->flags & symbol_type::kIsVectorTable)
- action = PUSH;
- }
- return action;
-}
-
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::doPush(BBEvent *event, symbol_type *function)
-{
- uint64_t time = event->time - mSkippedTime;
-
- // Check for stack overflow
- if (mTop >= mNumFrames) {
- // Don't show the stack by default because this generates a lot
- // of output and this is seen by users if there is an error when
- // post-processing the trace. But this is useful for debugging.
-#if 0
- showStack(stderr);
-#endif
- fprintf(stderr, "Error: stack overflow (%d frames)\n", mTop);
- exit(1);
- }
-
- // Compute the return address here because we may need to change
- // it if we are popping off a frame for a vector table.
- int numBytes;
- if (mPrevEvent.is_thumb) {
- numBytes = mPrevEvent.num_insns << 1;
- } else {
- numBytes = mPrevEvent.num_insns << 2;
- }
- uint32_t retAddr = mPrevEvent.bb_addr + numBytes;
-
- // If this is a Java method then set the return address to zero.
- // We won't be using it for popping the method and it may lead
- // to false matches when searching the stack.
- if (function->vm_sym != NULL) {
- retAddr = 0;
- }
-
-#if 0
- // For debugging only. Show the stack before entering the kernel
- // exception-handling code.
- if (function->flags & symbol_type::kIsVectorStart) {
- printf("stack before entering exception\n");
- showStack(stderr);
- }
-#endif
-
- // If the top of stack is a vector table, then pop it
- // off before pushing on the new function. Also, change the
- // return address for the new function to the return address
- // from the vector table.
- if (mTop > 0
- && (mFrames[mTop - 1].function->flags & symbol_type::kIsVectorTable)) {
- retAddr = mFrames[mTop - 1].addr;
- doSimplePop(time);
- }
-
- const uint32_t kIsKernelRegion = region_type::kIsKernelRegion;
-
- // The following code handles the case where one function, F1,
- // calls another function, F2, but the before F2 can start
- // executing, it takes a page fault (on the first instruction
- // in F2). The kernel is entered, handles the page fault, and
- // then returns to the called function. The problem is that
- // this looks like a new function call to F2 from the kernel.
- // The following code cleans up the stack by popping the
- // kernel frames back to F1 (but not including F1). The
- // return address for F2 also has to be fixed up to point to
- // F1 instead of the kernel.
- //
- // We detect this case by checking if the previous basic block
- // was in the kernel and the current basic block is not.
- if ((mPrevFunction->region->flags & kIsKernelRegion)
- && ((function->region->flags & kIsKernelRegion) == 0)
- && mTop > 0) {
- // We are switching from kernel mode to user mode.
-#if 0
- // For debugging.
- printf(" doPush(): popping to user mode, bb_addr: 0x%08x\n",
- event->bb_addr);
- showStack(stderr);
-#endif
- do {
- // Pop off the kernel frames until we reach the one that
- // caused the exception.
- doSimplePop(time);
-
- // If the next stack frame is the one that caused an
- // exception then stop popping frames.
- if (mTop > 0
- && (mFrames[mTop - 1].flags & FRAME::kCausedException)) {
- mFrames[mTop - 1].flags &= ~FRAME::kCausedException;
- retAddr = mFrames[mTop].addr;
- break;
- }
- } while (mTop > 0);
-#if 0
- // For debugging
- printf(" doPush() popping to level %d, using retAddr 0x%08x\n",
- mTop, retAddr);
-#endif
- }
-
- // If we are starting an exception handler, then mark the previous
- // stack frame so that we know where to return when the exception
- // handler finishes.
- if ((function->flags & symbol_type::kIsVectorStart) && mTop > 0)
- mFrames[mTop - 1].flags |= FRAME::kCausedException;
-
- // If the function being pushed is a Java method, then mark it on
- // the stack so that we don't pop it off until we get a matching
- // trace record from the method trace file.
- int flags = 0;
- if (function->vm_sym != NULL) {
- flags = FRAME::kInterpreted;
- }
- doSimplePush(function, retAddr, time, flags);
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::doSimplePush(symbol_type *function, uint32_t addr,
- uint64_t time, int flags)
-{
- // Check for stack overflow
- if (mTop >= mNumFrames) {
- showStack(stderr);
- fprintf(stderr, "too many stack frames (%d)\n", mTop);
- exit(1);
- }
-
- mFrames[mTop].addr = addr;
- mFrames[mTop].function = function;
- mFrames[mTop].flags = flags;
- mFrames[mTop].time = time;
- mFrames[mTop].global_time = time + mSkippedTime;
-
- mFrames[mTop].push(mTop, time, this);
- mTop += 1;
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::doSimplePop(uint64_t time)
-{
- if (mTop <= 0) {
- return;
- }
-
- mTop -= 1;
- mFrames[mTop].pop(mTop, time, this);
-
- if (mNativeOnly)
- return;
-
- // If the stack is empty, then allow more native frames.
- // Otherwise, if we are transitioning from Java to native, then allow
- // more native frames.
- // Otherwise, if we are transitioning from native to Java, then disallow
- // more native frames.
- if (mTop == 0) {
- mAllowNativeFrames = true;
- } else {
- bool newerIsJava = (mFrames[mTop].flags & FRAME::kInterpreted) != 0;
- bool olderIsJava = (mFrames[mTop - 1].flags & FRAME::kInterpreted) != 0;
- if (newerIsJava && !olderIsJava) {
- // We are transitioning from Java to native
- mAllowNativeFrames = true;
- } else if (!newerIsJava && olderIsJava) {
- // We are transitioning from native to Java
- mAllowNativeFrames = false;
- }
- }
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::doPop(BBEvent *event, symbol_type *function,
- Action methodAction)
-{
- uint64_t time = event->time - mSkippedTime;
-
- // Search backward on the stack for a matching return address.
- // The most common case is that we pop one stack frame, but
- // sometimes we pop more than one.
- int stackLevel;
- bool allowMethodPop = (methodAction == POP);
- for (stackLevel = mTop - 1; stackLevel >= 0; --stackLevel) {
- if (event->bb_addr == mFrames[stackLevel].addr) {
- // We found a matching return address on the stack.
- break;
- }
-
- // If this stack frame caused an exception, then do not pop
- // this stack frame.
- if (mFrames[stackLevel].flags & FRAME::kPopBarrier) {
- // If this is a Java method, then allow a pop only if we
- // have a matching trace record.
- if (mFrames[stackLevel].flags & FRAME::kInterpreted) {
- if (allowMethodPop) {
- // Allow at most one method pop
- allowMethodPop = false;
- continue;
- }
- }
- stackLevel += 1;
- break;
- }
- }
-
- // If we didn't find a matching return address then search the stack
- // again for a matching function.
- if (stackLevel < 0 || event->bb_addr != mFrames[stackLevel].addr) {
- bool allowMethodPop = (methodAction == POP);
- for (stackLevel = mTop - 1; stackLevel >= 0; --stackLevel) {
- // Compare the function with the one in the stack frame.
- if (function == mFrames[stackLevel].function) {
- // We found a matching function. We want to pop up to but not
- // including this frame. But allow popping this frame if this
- // method called itself and we have a method pop.
- if (allowMethodPop && function == mPrevFunction) {
- // pop this frame
- break;
- }
- // do not pop this frame
- stackLevel += 1;
- break;
- }
-
- // If this stack frame caused an exception, then do not pop
- // this stack frame.
- if (mFrames[stackLevel].flags & FRAME::kPopBarrier) {
- // If this is a Java method, then allow a pop only if we
- // have a matching trace record.
- if (mFrames[stackLevel].flags & FRAME::kInterpreted) {
- if (allowMethodPop) {
- // Allow at most one method pop
- allowMethodPop = false;
- continue;
- }
- }
- stackLevel += 1;
- break;
- }
- }
- if (stackLevel < 0)
- stackLevel = 0;
- }
-
- // Note that if we didn't find a matching stack frame, we will pop
- // the whole stack (unless there is a Java method or exception
- // frame on the stack). This is intentional because we may have
- // started the trace in the middle of an executing program that is
- // returning up the stack and we do not know the whole stack. So
- // the right thing to do is to empty the stack.
-
- // If we are emptying the stack, then add the current function
- // on top. If the current function is the same as the top of
- // stack, then avoid an extraneous pop and push.
- if (stackLevel == 0 && mFrames[0].function == function)
- stackLevel = 1;
-
-#if 0
- // If we are popping off a large number of stack frames, then
- // we might have a bug.
- if (mTop - stackLevel > 7) {
- printf("popping thru level %d\n", stackLevel);
- showStack(stderr);
- }
-#endif
-
- // Pop the stack frames
- for (int ii = mTop - 1; ii >= stackLevel; --ii)
- doSimplePop(time);
-
- // Clear the "caused exception" bit on the current stack frame
- if (mTop > 0) {
- mFrames[mTop - 1].flags &= ~FRAME::kCausedException;
- }
-
- // Also handle the case where F1 calls F2 and F2 returns to
- // F1, but before we can execute any instructions in F1, we
- // switch to the kernel. Then when we return from the kernel
- // we want to pop off F2 from the stack instead of pushing F1
- // on top of F2. To handle this case, we saved the last
- // user-mode basic block when we entered the kernel (in
- // the getAction() function) and now we can check to see if
- // that was a return to F1 instead of a call. We use the
- // getAction() function to determine this.
- const uint32_t kIsKernelRegion = region_type::kIsKernelRegion;
- if ((mPrevFunction->region->flags & kIsKernelRegion)
- && ((function->region->flags & kIsKernelRegion) == 0)) {
- mPrevEvent = mUserEvent;
- mPrevFunction = mUserFunction;
- if (getAction(event, function) == POP) {
- // We may need to pop more than one frame, so just
- // call doPop() again. This won't be an infinite loop
- // here because we changed mPrevEvent to the last
- // user-mode event.
- doPop(event, function, methodAction);
- return;
- }
- }
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::popAll(uint64_t time)
-{
- time -= mSkippedTime;
- while (mTop != 0) {
- doSimplePop(time);
- }
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::doMethodPop(BBEvent *event, uint32_t addr,
- const uint32_t flags)
-{
- uint64_t time = event->time - mSkippedTime;
-
- // Search the stack from the top down for a frame that contains a
- // matching method.
- int stackLevel;
- for (stackLevel = mTop - 1; stackLevel >= 0; --stackLevel) {
- if (mFrames[stackLevel].flags & flags) {
- // If we are searching for a native method, then don't bother trying
- // to match the address.
- if (flags == FRAME::kStartNative)
- break;
- symbol_type *func = mFrames[stackLevel].function;
- uint32_t methodAddr = func->region->base_addr + func->addr;
- if (methodAddr == addr) {
- break;
- }
- }
- }
-
- // If we found a matching frame then pop the stack up to and including
- // that frame.
- if (stackLevel >= 0) {
- // Pop the stack frames
- for (int ii = mTop - 1; ii >= stackLevel; --ii)
- doSimplePop(time);
- }
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::doMethodAction(BBEvent *event, symbol_type *function)
-{
- // If the events get ahead of the method trace, then read ahead until we
- // sync up again. This can happen if there is a pop of a method in the
- // method trace for which we don't have a previous push. Such an unmatched
- // pop can happen because the user can start tracing at any time and so
- // there might already be a stack when we start tracing.
- while (event->time >= sNextMethod.time) {
- sCurrentMethod = sNextMethod;
- if (mTrace->ReadMethod(&sNextMethod)) {
- sNextMethod.time = ~0ull;
- }
- }
-
- if (event->time >= sCurrentMethod.time && event->pid == sCurrentMethod.pid) {
- uint64_t time = event->time - mSkippedTime;
- int flags = sCurrentMethod.flags;
- if (flags == kMethodEnter) {
- doSimplePush(function, 0, time, FRAME::kInterpreted);
- mAllowNativeFrames = false;
- } else if (flags == kNativeEnter) {
- doSimplePush(function, 0, time, FRAME::kStartNative);
- mAllowNativeFrames = true;
- } else if (flags == kMethodExit || flags == kMethodException) {
- doMethodPop(event, sCurrentMethod.addr, FRAME::kInterpreted);
- } else if (flags == kNativeExit || flags == kNativeException) {
- doMethodPop(event, sCurrentMethod.addr, FRAME::kStartNative);
- }
-
- // We found a match, so read the next record. When we get to the end
- // of the trace, we set the time to the maximum value (~0).
- sCurrentMethod = sNextMethod;
- if (sNextMethod.time != ~0ull && mTrace->ReadMethod(&sNextMethod)) {
- sNextMethod.time = ~0ull;
- }
- }
-}
-
-template<class FRAME, class BASE>
-void CallStack<FRAME, BASE>::showStack(FILE *stream)
-{
- fprintf(stream, "mTop: %d skippedTime: %llu\n", mTop, mSkippedTime);
- for (int ii = 0; ii < mTop; ++ii) {
- uint32_t addr = mFrames[ii].function->addr;
- addr += mFrames[ii].function->region->vstart;
- fprintf(stream, " %d: t %d gt %d f %x 0x%08x 0x%08x %s\n",
- ii, mFrames[ii].time, mFrames[ii].global_time,
- mFrames[ii].flags,
- mFrames[ii].addr, addr,
- mFrames[ii].function->name);
- }
-}
-
-#endif /* CALL_STACK_H */
diff --git a/emulator/qtools/check_stack.cpp b/emulator/qtools/check_stack.cpp
deleted file mode 100644
index b4d14d3..0000000
--- a/emulator/qtools/check_stack.cpp
+++ /dev/null
@@ -1,270 +0,0 @@
-// Copyright 2009 The Android Open Source Project
-
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "bitvector.h"
-#include "parse_options.h"
-#include "armdis.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-#include "callstack.h"
-
-typedef CallStack<StackFrame<symbol_type> > CallStackType;
-
-void compareStacks(uint64_t time, int pid);
-void dumpStacks(int pid);
-
-static uint64_t debugTime;
-static const int kNumStackFrames = 500;
-static const int kMaxThreads = (32 * 1024);
-CallStackType *eStacks[kMaxThreads];
-
-int numErrors;
-static const int kMaxErrors = 3;
-
-struct frame {
- uint64_t time;
- uint32_t addr;
- const char *name;
- bool isNative;
-
- frame(uint64_t time, uint32_t addr, const char *name, bool isNative) {
- this->time = time;
- this->addr = addr;
- this->name = name;
- this->isNative = isNative;
- }
-};
-
-class Stack {
-public:
- static const int kMaxFrames = 1000;
- int top;
- frame *frames[kMaxFrames];
-
- Stack() {
- top = 0;
- }
-
- void push(frame *pframe);
- frame* pop();
- void dump();
-};
-
-void Stack::push(frame *pframe) {
- if (top == kMaxFrames) {
- fprintf(stderr, "Error: stack overflow\n");
- exit(1);
- }
- frames[top] = pframe;
- top += 1;
-}
-
-frame *Stack::pop() {
- if (top <= 0)
- return NULL;
- top -= 1;
- return frames[top];
-}
-
-Stack *mStacks[kMaxThreads];
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_name elf_file\n",
- program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv)
-{
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *qemu_trace_file = argv[optind++];
- char *elf_file = argv[optind++];
-
- TraceReaderType *etrace = new TraceReaderType;
- etrace->Open(qemu_trace_file);
- etrace->ReadKernelSymbols(elf_file);
- etrace->SetRoot(root);
-
- TraceReaderType *mtrace = new TraceReaderType;
- mtrace->Open(qemu_trace_file);
- mtrace->ReadKernelSymbols(elf_file);
- mtrace->SetRoot(root);
-
- BBEvent event;
- while (1) {
- BBEvent ignored;
- symbol_type *function;
- MethodRec method_record;
- symbol_type *sym;
- TraceReaderType::ProcessState *proc;
- frame *pframe;
-
- if (mtrace->ReadMethodSymbol(&method_record, &sym, &proc))
- break;
-
- if (!IsValidPid(proc->pid))
- continue;
-
- // Get the stack for the current thread
- Stack *mStack = mStacks[proc->pid];
-
- // If the stack does not exist, then allocate a new one.
- if (mStack == NULL) {
- mStack = new Stack();
- mStacks[proc->pid] = mStack;
- }
-
- int flags = method_record.flags;
- if (flags == kMethodEnter || flags == kNativeEnter) {
- pframe = new frame(method_record.time, method_record.addr,
- sym == NULL ? NULL: sym->name,
- method_record.flags == kNativeEnter);
- mStack->push(pframe);
- } else {
- pframe = mStack->pop();
- delete pframe;
- }
-
- do {
- if (GetNextValidEvent(etrace, &event, &ignored, &function))
- break;
- if (event.bb_num == 0)
- break;
-
- // Get the stack for the current thread
- CallStackType *eStack = eStacks[event.pid];
-
- // If the stack does not exist, then allocate a new one.
- if (eStack == NULL) {
- eStack = new CallStackType(event.pid, kNumStackFrames, etrace);
- eStacks[event.pid] = eStack;
- }
- if (debugTime != 0 && event.time >= debugTime)
- printf("time: %llu debug time: %lld\n", event.time, debugTime);
-
- // Update the stack
- eStack->updateStack(&event, function);
- } while (event.time < method_record.time);
-
- compareStacks(event.time, event.pid);
- }
-
- for (int ii = 0; ii < kMaxThreads; ++ii) {
- if (eStacks[ii])
- eStacks[ii]->popAll(event.time);
- }
-
- delete etrace;
- delete mtrace;
- return 0;
-}
-
-void compareStacks(uint64_t time, int pid) {
- CallStackType *eStack = eStacks[pid];
- Stack *mStack = mStacks[pid];
- frame **mFrames = mStack->frames;
- frame *mframe;
-
- int mTop = mStack->top;
- int eTop = eStack->mTop;
- CallStackType::frame_type *eFrames = eStack->mFrames;
-
- // Count the number of non-native methods (ie, Java methods) on the
- // Java method stack
- int numNonNativeMethods = 0;
- for (int ii = 0; ii < mTop; ++ii) {
- if (!mFrames[ii]->isNative) {
- numNonNativeMethods += 1;
- }
- }
-
- // Count the number of Java methods on the native stack
- int numMethods = 0;
- for (int ii = 0; ii < eTop; ++ii) {
- if (eFrames[ii].flags & CallStackType::frame_type::kInterpreted) {
- numMethods += 1;
- }
- }
-
- // Verify that the number of Java methods on both stacks are the same.
- // Allow the native stack to have one less Java method because the
- // native stack might be pushing a native function first.
- if (numNonNativeMethods != numMethods && numNonNativeMethods != numMethods + 1) {
- printf("\nDiff at time %llu pid %d: non-native %d numMethods %d\n",
- time, pid, numNonNativeMethods, numMethods);
- dumpStacks(pid);
- numErrors += 1;
- if (numErrors >= kMaxErrors)
- exit(1);
- }
-
- // Verify that the Java methods on the method stack are the same
- // as the Java methods on the native stack.
- int mIndex = 0;
- for (int ii = 0; ii < eTop; ++ii) {
- // Ignore native functions on the native stack.
- if ((eFrames[ii].flags & CallStackType::frame_type::kInterpreted) == 0)
- continue;
- uint32_t addr = eFrames[ii].function->addr;
- addr += eFrames[ii].function->region->vstart;
- while (mIndex < mTop && mFrames[mIndex]->isNative) {
- mIndex += 1;
- }
- if (mIndex >= mTop)
- break;
- if (addr != mFrames[mIndex]->addr) {
- printf("\nDiff at time %llu pid %d: frame %d\n", time, pid, ii);
- dumpStacks(pid);
- exit(1);
- }
- mIndex += 1;
- }
-}
-
-void dumpStacks(int pid) {
- CallStackType *eStack = eStacks[pid];
- Stack *mStack = mStacks[pid];
- frame *mframe;
-
- int mTop = mStack->top;
- printf("\nJava method stack\n");
- for (int ii = 0; ii < mTop; ii++) {
- mframe = mStack->frames[ii];
- const char *native = mframe->isNative ? "n" : " ";
- printf(" %s %d: %llu 0x%x %s\n",
- native, ii, mframe->time, mframe->addr,
- mframe->name == NULL ? "" : mframe->name);
- }
-
- int eTop = eStack->mTop;
- CallStackType::frame_type *eFrames = eStack->mFrames;
- int mIndex = 0;
- printf("\nNative stack\n");
- for (int ii = 0; ii < eTop; ++ii) {
- uint32_t addr = eFrames[ii].function->addr;
- addr += eFrames[ii].function->region->vstart;
- const char *marker = " ";
- if (eFrames[ii].flags & CallStackType::frame_type::kInterpreted) {
- if (mIndex >= mTop || addr != mStack->frames[mIndex]->addr) {
- marker = "*";
- }
- mIndex += 1;
- }
- printf(" %s %d: %d f %d 0x%08x %s\n",
- marker, ii, eFrames[ii].time, eFrames[ii].flags, addr,
- eFrames[ii].function->name);
- }
-}
diff --git a/emulator/qtools/check_trace.cpp b/emulator/qtools/check_trace.cpp
deleted file mode 100644
index d933a87..0000000
--- a/emulator/qtools/check_trace.cpp
+++ /dev/null
@@ -1,61 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "armdis.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-static const uint32_t kOffsetThreshold = 0x100000;
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file elf_file\n", program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv) {
- // Parse the options
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReader<> *trace = new TraceReader<>;
- trace->Open(trace_filename);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- while (1) {
- symbol_type *sym;
- BBEvent event;
- BBEvent ignored;
-
- if (GetNextValidEvent(trace, &event, &ignored, &sym))
- break;
- if (event.bb_num == 0)
- break;
- //printf("t%llu bb %lld %d\n", event.time, event.bb_num, event.num_insns);
- uint64_t insn_time = trace->ReadInsnTime(event.time);
- if (insn_time != event.time) {
- printf("time: %llu insn time: %llu bb: %llu addr: 0x%x num_insns: %d, pid: %d\n",
- event.time, insn_time, event.bb_num, event.bb_addr,
- event.num_insns, event.pid);
- exit(1);
- }
- for (int ii = 1; ii < event.num_insns; ++ii) {
- trace->ReadInsnTime(event.time);
- }
- }
-
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/coverage.cpp b/emulator/qtools/coverage.cpp
deleted file mode 100644
index 790f721..0000000
--- a/emulator/qtools/coverage.cpp
+++ /dev/null
@@ -1,153 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <string.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-#include "opcode.h"
-
-const int kMillion = 1000000;
-const int kMHz = 200 * kMillion;
-
-struct symbol {
- int numCalls; // number of times this function is called
-};
-
-typedef TraceReader<symbol> TraceReaderType;
-
-#include "parse_options-inl.h"
-#include "callstack.h"
-
-class MyFrame : public StackFrame<symbol_type> {
- public:
- void push(int stackLevel, uint64_t time, CallStackBase *base) {
- function->numCalls += 1;
- }
- void pop(int stackLevel, uint64_t time, CallStackBase *base) {
- }
-};
-
-typedef CallStack<MyFrame> CallStackType;
-
-static const int kNumStackFrames = 500;
-static const int kMaxThreads = (32 * 1024);
-CallStackType *stacks[kMaxThreads];
-
-// This comparison function is called from qsort() to sort symbols
-// into decreasing number of calls.
-int cmp_sym_calls(const void *a, const void *b) {
- const symbol_type *syma, *symb;
- uint64_t calls1, calls2;
-
- syma = static_cast<symbol_type const *>(a);
- symb = static_cast<symbol_type const *>(b);
- calls1 = syma->numCalls;
- calls2 = symb->numCalls;
- if (calls1 < calls2)
- return 1;
- if (calls1 == calls2) {
- int cmp = strcmp(syma->name, symb->name);
- if (cmp == 0)
- cmp = strcmp(syma->region->path, symb->region->path);
- return cmp;
- }
- return -1;
-}
-
-// This comparison function is called from qsort() to sort symbols
-// into alphabetical order.
-int cmp_sym_names(const void *a, const void *b) {
- const symbol_type *syma, *symb;
-
- syma = static_cast<symbol_type const *>(a);
- symb = static_cast<symbol_type const *>(b);
- int cmp = strcmp(syma->region->path, symb->region->path);
- if (cmp == 0)
- cmp = strcmp(syma->name, symb->name);
- return cmp;
-}
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file elf_file\n", program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv)
-{
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReader<symbol> *trace = new TraceReader<symbol>;
- trace->Open(trace_filename);
- trace->SetDemangle(demangle);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- BBEvent event;
- while (1) {
- BBEvent ignored;
- symbol_type *function;
-
- if (GetNextValidEvent(trace, &event, &ignored, &function))
- break;
- if (event.bb_num == 0)
- break;
-
- // Get the stack for the current thread
- CallStackType *pStack = stacks[event.pid];
-
- // If the stack does not exist, then allocate a new one.
- if (pStack == NULL) {
- pStack = new CallStackType(event.pid, kNumStackFrames, trace);
- stacks[event.pid] = pStack;
- }
-
- // Update the stack
- pStack->updateStack(&event, function);
- }
-
- for (int ii = 0; ii < kMaxThreads; ++ii) {
- if (stacks[ii])
- stacks[ii]->popAll(event.time);
- }
-
- int nsyms;
- symbol_type *syms = trace->GetSymbols(&nsyms);
-
- // Sort the symbols into decreasing number of calls
- qsort(syms, nsyms, sizeof(symbol_type), cmp_sym_names);
-
- symbol_type *psym = syms;
- for (int ii = 0; ii < nsyms; ++ii, ++psym) {
- // Ignore functions with non-zero calls
- if (psym->numCalls)
- continue;
-
- // Ignore some symbols
- if (strcmp(psym->name, "(end)") == 0)
- continue;
- if (strcmp(psym->name, "(unknown)") == 0)
- continue;
- if (strcmp(psym->name, ".plt") == 0)
- continue;
- const char *ksym = " ";
- if (psym->region->flags & region_type::kIsKernelRegion)
- ksym = "k";
- printf("%s %s %s\n", ksym, psym->name, psym->region->path);
-#if 0
- printf("#%d %5d %s %s %s\n", ii + 1, psym->numCalls, ksym, psym->name,
- psym->region->path);
-#endif
- }
- delete[] syms;
- delete trace;
-
- return 0;
-}
diff --git a/emulator/qtools/decoder.cpp b/emulator/qtools/decoder.cpp
deleted file mode 100644
index ec53181..0000000
--- a/emulator/qtools/decoder.cpp
+++ /dev/null
@@ -1,278 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <assert.h>
-#include "decoder.h"
-#include "trace_common.h"
-
-// This array provides a fast conversion from the initial byte in
-// a varint-encoded object to the length (in bytes) of that object.
-int prefix_to_len[] = {
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
- 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
- 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
- 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
- 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
- 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
- 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
- 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4,
- 5, 5, 5, 5, 5, 5, 5, 5, 6, 6, 6, 6, 9, 9, 17, 17
-};
-
-// This array provides a fast conversion from the initial byte in
-// a varint-encoded object to the initial data bits for that object.
-int prefix_to_data[] = {
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
- 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
- 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47,
- 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63,
- 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79,
- 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95,
- 96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111,
- 112, 113, 114, 115, 116, 117, 118, 119,
- 120, 121, 122, 123, 124, 125, 126, 127,
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
- 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
- 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47,
- 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63,
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
- 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
- 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
- 0, 1, 2, 3, 4, 5, 6, 7, 0, 1, 2, 3, 0, 0, 0, 0
-};
-
-signed char prefix_to_signed_data[] = {
- 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
- 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
- 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17,
- 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
- 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27,
- 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
- 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37,
- 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
- 0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7,
- 0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf,
- 0xd0, 0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7,
- 0xd8, 0xd9, 0xda, 0xdb, 0xdc, 0xdd, 0xde, 0xdf,
- 0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7,
- 0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef,
- 0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7,
- 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff,
- 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
- 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
- 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17,
- 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
- 0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7,
- 0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef,
- 0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7,
- 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff,
- 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
- 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
- 0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7,
- 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff,
- 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
- 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff,
- 0x00, 0x01, 0x02, 0x03, 0xfc, 0xfd, 0xfe, 0xff,
- 0x00, 0x01, 0xfe, 0xff, 0x00, 0xff, 0x00, 0xff,
-};
-
-Decoder::Decoder()
-{
- filename_ = NULL;
- fstream_ = NULL;
- next_ = NULL;
- end_ = NULL;
-}
-
-Decoder::~Decoder()
-{
- Close();
- delete[] filename_;
-}
-
-void Decoder::Close()
-{
- if (fstream_) {
- fclose(fstream_);
- fstream_ = NULL;
- }
-}
-
-void Decoder::Open(char *filename)
-{
- if (filename_ != NULL) {
- delete[] filename_;
- }
- filename_ = new char[strlen(filename) + 1];
- strcpy(filename_, filename);
- fstream_ = fopen(filename_, "r");
- if (fstream_ == NULL) {
- perror(filename_);
- exit(1);
- }
-
- int rval = fread(buf_, 1, kBufSize, fstream_);
- if (rval != kBufSize) {
- if (ferror(fstream_)) {
- perror(filename_);
- exit(1);
- }
- if (!feof(fstream_)) {
- fprintf(stderr, "Unexpected short fread() before eof\n");
- exit(1);
- }
- }
- next_ = buf_;
- end_ = buf_ + rval;
-}
-
-void Decoder::FillBuffer()
-{
- assert(next_ <= end_);
-
- if (end_ - next_ < kDecodingSpace && end_ == &buf_[kBufSize]) {
- // Copy the unused bytes left at the end to the beginning of the
- // buffer.
- int len = end_ - next_;
- if (len > 0)
- memcpy(buf_, next_, len);
-
- // Read enough bytes to fill up the buffer, if possible.
- int nbytes = kBufSize - len;
- int rval = fread(buf_ + len, 1, nbytes, fstream_);
- if (rval < nbytes) {
- if (ferror(fstream_)) {
- perror(filename_);
- exit(1);
- }
- if (!feof(fstream_)) {
- fprintf(stderr, "Unexpected short fread() before eof\n");
- exit(1);
- }
- }
- end_ = &buf_[len + rval];
- next_ = buf_;
- }
-}
-
-void Decoder::Read(char *dest, int len)
-{
- while (len > 0) {
- int nbytes = end_ - next_;
- if (nbytes == 0) {
- FillBuffer();
- nbytes = end_ - next_;
- if (nbytes == 0)
- break;
- }
- if (nbytes > len)
- nbytes = len;
- memcpy(dest, next_, nbytes);
- dest += nbytes;
- len -= nbytes;
- next_ += nbytes;
- }
-}
-
-// Decode a varint-encoded object starting at the current position in
-// the array "buf_" and return the decoded value as a 64-bit integer.
-// A varint-encoded object has an initial prefix that specifies how many
-// data bits follow. If the first bit is zero, for example, then there
-// are 7 data bits that follow. The table below shows the prefix values
-// and corresponding data bits.
-//
-// Prefix Bytes Data bits
-// 0 1 7
-// 10 2 14
-// 110 3 21
-// 1110 4 28
-// 11110 5 35
-// 111110 6 42
-// 11111100 9 64
-// 11111101 reserved
-// 11111110 reserved
-// 11111111 reserved
-int64_t Decoder::Decode(bool is_signed)
-{
- int64_t val64;
-
- if (end_ - next_ < kDecodingSpace)
- FillBuffer();
-
-#if BYTE_ORDER == BIG_ENDIAN
- uint8_t byte0 = *next_;
-
- // Get the number of bytes to decode based on the first byte.
- int len = prefix_to_len[byte0];
-
- if (next_ + len > end_) {
- fprintf(stderr, "%s: decoding past end of file.\n", filename_);
- exit(1);
- }
-
- // Get the first data byte.
- if (is_signed)
- val64 = prefix_to_signed_data[byte0];
- else
- val64 = prefix_to_data[byte0];
-
- next_ += 1;
- for (int ii = 1; ii < len; ++ii) {
- val64 = (val64 << 8) | *next_++;
- }
-#else
- // If we are on a little-endian machine, then use large, unaligned loads.
- uint64_t data = *(reinterpret_cast<uint64_t*>(next_));
- uint8_t byte0 = data;
- data = bswap64(data);
-
- // Get the number of bytes to decode based on the first byte.
- int len = prefix_to_len[byte0];
-
- if (next_ + len > end_) {
- fprintf(stderr, "%s: decoding past end of file.\n", filename_);
- exit(1);
- }
-
- // Get the first data byte.
- if (is_signed)
- val64 = prefix_to_signed_data[byte0];
- else
- val64 = prefix_to_data[byte0];
-
- switch (len) {
- case 1:
- break;
- case 2:
- val64 = (val64 << 8) | ((data >> 48) & 0xffull);
- break;
- case 3:
- val64 = (val64 << 16) | ((data >> 40) & 0xffffull);
- break;
- case 4:
- val64 = (val64 << 24) | ((data >> 32) & 0xffffffull);
- break;
- case 5:
- val64 = (val64 << 32) | ((data >> 24) & 0xffffffffull);
- break;
- case 6:
- val64 = (val64 << 40) | ((data >> 16) & 0xffffffffffull);
- break;
- case 9:
- data = *(reinterpret_cast<uint64_t*>(&next_[1]));
- val64 = bswap64(data);
- break;
- }
- next_ += len;
-#endif
- return val64;
-}
diff --git a/emulator/qtools/decoder.h b/emulator/qtools/decoder.h
deleted file mode 100644
index 44905fd..0000000
--- a/emulator/qtools/decoder.h
+++ /dev/null
@@ -1,28 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <inttypes.h>
-
-class Decoder {
- public:
- Decoder();
- ~Decoder();
-
- void Open(char *filename);
- void Close();
- int64_t Decode(bool is_signed);
- void Read(char *dest, int len);
- bool IsEOF() { return (end_ == next_) && feof(fstream_); }
-
- private:
- static const int kBufSize = 4096;
- static const int kDecodingSpace = 9;
-
- void FillBuffer();
-
- char *filename_;
- FILE *fstream_;
- uint8_t buf_[kBufSize];
- uint8_t *next_;
- uint8_t *end_;
-};
diff --git a/emulator/qtools/dmtrace.cpp b/emulator/qtools/dmtrace.cpp
deleted file mode 100644
index c486c5f..0000000
--- a/emulator/qtools/dmtrace.cpp
+++ /dev/null
@@ -1,255 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <unistd.h>
-#include <inttypes.h>
-#include <string.h>
-#include <unistd.h>
-#include "dmtrace.h"
-
-static const short kVersion = 2;
-
-const DmTrace::Header DmTrace::header = {
- 0x574f4c53, kVersion, sizeof(DmTrace::Header), 0LL
-};
-
-static char *keyHeader = "*version\n" "2\n" "clock=thread-cpu\n";
-static char *keyThreadHeader = "*threads\n";
-static char *keyFunctionHeader = "*methods\n";
-static char *keyEnd = "*end\n";
-
-DmTrace::DmTrace() {
- fData = NULL;
- fTrace = NULL;
- threads = new std::vector<ThreadRecord*>;
- functions = new std::vector<FunctionRecord*>;
-}
-
-DmTrace::~DmTrace() {
- delete threads;
- delete functions;
-}
-
-void DmTrace::open(const char *dmtrace_file, uint64_t start_time)
-{
- fTrace = fopen(dmtrace_file, "w");
- if (fTrace == NULL) {
- perror(dmtrace_file);
- exit(1);
- }
-
- // Make a temporary file to write the data into.
- char tmpData[32];
- strcpy(tmpData, "/tmp/dmtrace-data-XXXXXX");
- int data_fd = mkstemp(tmpData);
- if (data_fd < 0) {
- perror("Cannot create temporary file");
- exit(1);
- }
-
- // Ensure it goes away on exit.
- unlink(tmpData);
- fData = fdopen(data_fd, "w+");
- if (fData == NULL) {
- perror("Can't make temp data file");
- exit(1);
- }
-
- writeHeader(fData, start_time);
-}
-
-void DmTrace::close()
-{
- if (fTrace == NULL)
- return;
- writeKeyFile(fTrace);
-
- // Take down how much data we wrote to the temp data file.
- long size = ftell(fData);
- // Rewind the data file and append its contents to the trace file.
- rewind(fData);
- char *data = (char *)malloc(size);
- fread(data, size, 1, fData);
- fwrite(data, size, 1, fTrace);
- free(data);
- fclose(fData);
- fclose(fTrace);
-}
-
-/*
- * Write values to the binary data file.
- */
-void DmTrace::write2LE(FILE* fstream, unsigned short val)
-{
- putc(val & 0xff, fstream);
- putc(val >> 8, fstream);
-}
-
-void DmTrace::write4LE(FILE* fstream, unsigned int val)
-{
- putc(val & 0xff, fstream);
- putc((val >> 8) & 0xff, fstream);
- putc((val >> 16) & 0xff, fstream);
- putc((val >> 24) & 0xff, fstream);
-}
-
-void DmTrace::write8LE(FILE* fstream, unsigned long long val)
-{
- putc(val & 0xff, fstream);
- putc((val >> 8) & 0xff, fstream);
- putc((val >> 16) & 0xff, fstream);
- putc((val >> 24) & 0xff, fstream);
- putc((val >> 32) & 0xff, fstream);
- putc((val >> 40) & 0xff, fstream);
- putc((val >> 48) & 0xff, fstream);
- putc((val >> 56) & 0xff, fstream);
-}
-
-void DmTrace::writeHeader(FILE *fstream, uint64_t startTime)
-{
- write4LE(fstream, header.magic);
- write2LE(fstream, header.version);
- write2LE(fstream, header.offset);
- write8LE(fstream, startTime);
-}
-
-void DmTrace::writeDataRecord(FILE *fstream, int threadId,
- unsigned int methodVal,
- unsigned int elapsedTime)
-{
- write2LE(fstream, threadId);
- write4LE(fstream, methodVal);
- write4LE(fstream, elapsedTime);
-}
-
-void DmTrace::addFunctionEntry(int functionId, uint32_t cycle, uint32_t pid)
-{
- writeDataRecord(fData, pid, functionId, cycle);
-}
-
-void DmTrace::addFunctionExit(int functionId, uint32_t cycle, uint32_t pid)
-{
- writeDataRecord(fData, pid, functionId | 1, cycle);
-}
-
-void DmTrace::addFunction(int functionId, const char *name)
-{
- FunctionRecord *rec = new FunctionRecord;
- rec->id = functionId;
- rec->name = name;
- functions->push_back(rec);
-}
-
-void DmTrace::addFunction(int functionId, const char *clazz,
- const char *method, const char *sig)
-{
- // Allocate space for all the strings, plus 2 tab separators plus null byte.
- // We currently don't reclaim this space.
- int len = strlen(clazz) + strlen(method) + strlen(sig) + 3;
- char *name = new char[len];
- sprintf(name, "%s\t%s\t%s", clazz, method, sig);
-
- addFunction(functionId, name);
-}
-
-void DmTrace::parseAndAddFunction(int functionId, const char *name)
-{
- // Parse the "name" string into "class", "method" and "signature".
- // The "name" string should look something like this:
- // name = "java.util.LinkedList.size()I"
- // and it will be parsed into this:
- // clazz = "java.util.LinkedList"
- // method = "size"
- // sig = "()I"
-
- // Find the first parenthesis, the start of the signature.
- char *paren = (char*)strchr(name, '(');
-
- // If not found, then add the original name.
- if (paren == NULL) {
- addFunction(functionId, name);
- return;
- }
-
- // Copy the signature
- int len = strlen(paren) + 1;
- char *sig = new char[len];
- strcpy(sig, paren);
-
- // Zero the parenthesis so that we can search backwards from the signature
- *paren = 0;
-
- // Search for the last period, the start of the method name
- char *dot = (char*)strrchr(name, '.');
-
- // If not found, then add the original name.
- if (dot == NULL || dot == name) {
- delete[] sig;
- *paren = '(';
- addFunction(functionId, name);
- return;
- }
-
- // Copy the method, not including the dot
- len = strlen(dot + 1) + 1;
- char *method = new char[len];
- strcpy(method, dot + 1);
-
- // Zero the dot to delimit the class name
- *dot = 0;
-
- addFunction(functionId, name, method, sig);
-
- // Free the space we allocated.
- delete[] sig;
- delete[] method;
-}
-
-void DmTrace::addThread(int threadId, const char *name)
-{
- ThreadRecord *rec = new ThreadRecord;
- rec->id = threadId;
- rec->name = name;
- threads->push_back(rec);
-}
-
-void DmTrace::updateName(int threadId, const char *name)
-{
- std::vector<ThreadRecord*>::iterator iter;
-
- for (iter = threads->begin(); iter != threads->end(); ++iter) {
- if ((*iter)->id == threadId) {
- (*iter)->name = name;
- return;
- }
- }
-}
-
-void DmTrace::writeKeyFile(FILE *fstream)
-{
- fwrite(keyHeader, strlen(keyHeader), 1, fstream);
- writeThreads(fstream);
- writeFunctions(fstream);
- fwrite(keyEnd, strlen(keyEnd), 1, fstream);
-}
-
-void DmTrace::writeThreads(FILE *fstream)
-{
- std::vector<ThreadRecord*>::iterator iter;
-
- fwrite(keyThreadHeader, strlen(keyThreadHeader), 1, fstream);
- for (iter = threads->begin(); iter != threads->end(); ++iter) {
- fprintf(fstream, "%d\t%s\n", (*iter)->id, (*iter)->name);
- }
-}
-
-void DmTrace::writeFunctions(FILE *fstream)
-{
- std::vector<FunctionRecord*>::iterator iter;
-
- fwrite(keyFunctionHeader, strlen(keyFunctionHeader), 1, fstream);
- for (iter = functions->begin(); iter != functions->end(); ++iter) {
- fprintf(fstream, "0x%x\t%s\n", (*iter)->id, (*iter)->name);
- }
-}
diff --git a/emulator/qtools/dmtrace.h b/emulator/qtools/dmtrace.h
deleted file mode 100644
index 6e20921..0000000
--- a/emulator/qtools/dmtrace.h
+++ /dev/null
@@ -1,61 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef DMTRACE_H
-#define DMTRACE_H
-
-#include <vector>
-
-class DmTrace {
- public:
- struct Header {
- uint32_t magic;
- uint16_t version;
- uint16_t offset;
- uint64_t date_time;
- };
-
- DmTrace();
- ~DmTrace();
-
- void open(const char *dmtrace_file, uint64_t startTime);
- void close();
- void addFunctionEntry(int methodId, uint32_t cycle, uint32_t pid);
- void addFunctionExit(int methodId, uint32_t cycle, uint32_t pid);
- void addFunction(int functionId, const char *name);
- void addFunction(int functionId, const char *clazz, const char *method,
- const char *sig);
- void parseAndAddFunction(int functionId, const char *name);
- void addThread(int threadId, const char *name);
- void updateName(int threadId, const char *name);
-
- private:
- static const Header header;
-
- struct ThreadRecord {
- int id;
- const char *name;
- };
-
- struct FunctionRecord {
- int id;
- const char *name;
- };
-
- void write2LE(FILE* fstream, unsigned short val);
- void write4LE(FILE* fstream, unsigned int val);
- void write8LE(FILE* fstream, unsigned long long val);
- void writeHeader(FILE *fstream, uint64_t startTime);
- void writeDataRecord(FILE *fstream, int threadId,
- unsigned int methodVal,
- unsigned int elapsedTime);
- void writeKeyFile(FILE *fstream);
- void writeThreads(FILE *fstream);
- void writeFunctions(FILE *fstream);
-
- FILE *fData;
- FILE *fTrace;
- std::vector<ThreadRecord*> *threads;
- std::vector<FunctionRecord*> *functions;
-};
-
-#endif // DMTRACE_H
diff --git a/emulator/qtools/dump_regions.cpp b/emulator/qtools/dump_regions.cpp
deleted file mode 100644
index 57389f9..0000000
--- a/emulator/qtools/dump_regions.cpp
+++ /dev/null
@@ -1,59 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <string.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file\n", program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv) {
- // Parse the options
- ParseOptions(argc, argv);
- if (argc - optind != 1) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind];
- TraceReader<> *trace = new TraceReader<>;
- trace->Open(trace_filename);
- trace->SetRoot(root);
-
- while (1) {
- BBEvent event, ignored;
- symbol_type *dummy_sym;
-
- if (GetNextValidEvent(trace, &event, &ignored, &dummy_sym))
- break;
- }
-
- int num_procs;
- ProcessState *processes = trace->GetProcesses(&num_procs);
-
- ProcessState *pstate = &processes[0];
- for (int ii = 0; ii < num_procs; ++ii, ++pstate) {
- if (pstate->name == NULL)
- pstate->name = "";
- ProcessState *manager = pstate->addr_manager;
- printf("pid %d regions: %d %s",
- pstate->pid, manager->nregions, pstate->name);
- for (int jj = 1; jj < pstate->argc; ++jj) {
- printf(" %s", pstate->argv[jj]);
- }
- printf("\n");
- trace->DumpRegions(stdout, pstate);
- }
-
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/exc_dump.cpp b/emulator/qtools/exc_dump.cpp
deleted file mode 100644
index 166586f..0000000
--- a/emulator/qtools/exc_dump.cpp
+++ /dev/null
@@ -1,28 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include "trace_reader_base.h"
-
-int main(int argc, char **argv) {
- if (argc != 2) {
- fprintf(stderr, "Usage: %s trace_file\n", argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[1];
- TraceReaderBase *trace = new TraceReaderBase;
- trace->Open(trace_filename);
-
- while (1) {
- uint64_t time, recnum, bb_num, bb_start_time;
- uint32_t pc, target_pc;
- int num_insns;
-
- if (trace->ReadExc(&time, &pc, &recnum, &target_pc, &bb_num,
- &bb_start_time, &num_insns))
- break;
- printf("time: %lld rec: %llu pc: %08x target: %08x bb: %llu bb_start: %llu insns: %d\n",
- time, recnum, pc, target_pc, bb_num, bb_start_time, num_insns);
- }
- return 0;
-}
diff --git a/emulator/qtools/gtrace.cpp b/emulator/qtools/gtrace.cpp
deleted file mode 100644
index 673d8a4..0000000
--- a/emulator/qtools/gtrace.cpp
+++ /dev/null
@@ -1,152 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include "gtrace.h"
-
-// A buffer of zeros
-static char zeros[Gtrace::kGtraceEntriesPerBlock * sizeof(Gtrace::trace_entry)];
-
-Gtrace::Gtrace() {
- gtrace_file_ = NULL;
- ftrace_ = NULL;
- fnames_ = NULL;
- start_sec_ = 0;
- pdate_ = 0;
- ptime_ = 0;
- num_entries_ = 0;
- blockno_ = 1;
- current_pid_ = 0;
-}
-
-Gtrace::~Gtrace() {
- if (ftrace_) {
- // Extend the trace file to a multiple of 8k. Otherwise gtracepost64
- // complains.
- long pos = ftell(ftrace_);
- long pos_end = (pos + 0x1fff) & ~0x1fff;
- if (pos_end > pos) {
- char ch = 0;
- fseek(ftrace_, pos_end - 1, SEEK_SET);
- fwrite(&ch, 1, 1, ftrace_);
- }
- fclose(ftrace_);
- }
- if (fnames_)
- fclose(fnames_);
-}
-
-void Gtrace::Open(const char *gtrace_file, uint32_t pdate, uint32_t ptime)
-{
- ftrace_ = fopen(gtrace_file, "w");
- if (ftrace_ == NULL) {
- perror(gtrace_file);
- exit(1);
- }
- gtrace_file_ = gtrace_file;
-
- pdate_ = pdate;
- ptime_ = ptime;
- sprintf(gname_file_, "gname_%x_%06x.txt", pdate, ptime);
- fnames_ = fopen(gname_file_, "w");
- if (fnames_ == NULL) {
- perror(gname_file_);
- exit(1);
- }
- fprintf(fnames_, "# File# Proc# Line# Name\n");
-}
-
-void Gtrace::WriteFirstHeader(uint32_t start_sec, uint32_t pid)
-{
- first_header fh;
- current_pid_ = pid;
- start_sec_ = start_sec;
- FillFirstHeader(start_sec, pid, &fh);
- fwrite(&fh, sizeof(fh), 1, ftrace_);
- num_entries_ = 8;
-}
-
-void Gtrace::FillFirstHeader(uint32_t start_sec, uint32_t pid,
- first_header *fh) {
- int cpu = 0;
- int max_files = 16;
- int max_procedures = 12;
-
- fh->common.blockno = 0;
- fh->common.entry_width = 8;
- fh->common.block_tic = kBaseTic;
- fh->common.block_time = start_sec;
- //fh->common.usec_cpu = (start_usec << 8) | (cpu & 0xff);
- fh->common.usec_cpu = cpu & 0xff;
- fh->common.pid = pid;
- fh->common.bug_count = 0;
- fh->common.zero_count = 0;
-
- fh->tic = kBaseTic + 1;
- fh->one = 1;
- fh->tics_per_second = kTicsPerSecond;
- fh->trace_time = start_sec;
- fh->version = 5;
- fh->file_proc = (max_files << 8) | max_procedures;
- fh->pdate = pdate_;
- fh->ptime = ptime_;
-}
-
-void Gtrace::WriteBlockHeader(uint32_t cycle, uint32_t pid)
-{
- int cpu = 0;
- block_header bh;
-
- bh.blockno = blockno_++;
- bh.entry_width = 8;
- bh.block_tic = cycle + kBaseTic;
- bh.block_time = start_sec_ + cycle / kTicsPerSecond;
- //bh.usec_cpu = (start_usec << 8) | (cpu & 0xff);
- bh.usec_cpu = cpu & 0xff;
- bh.pid = pid;
- bh.bug_count = 0;
- bh.zero_count = 0;
- fwrite(&bh, sizeof(bh), 1, ftrace_);
-}
-
-void Gtrace::AddGtraceRecord(int filenum, int procnum, uint32_t cycle, uint32_t pid,
- int is_exit)
-{
- trace_entry entry;
-
- if (current_pid_ != pid) {
- current_pid_ = pid;
-
- // We are switching to a new process id, so pad the current block
- // with zeros.
- int num_zeros = (kGtraceEntriesPerBlock - num_entries_) * sizeof(entry);
- fwrite(zeros, num_zeros, 1, ftrace_);
- WriteBlockHeader(cycle, pid);
- num_entries_ = 4;
- }
-
- // If the current block is full, write out a new block header
- if (num_entries_ == kGtraceEntriesPerBlock) {
- WriteBlockHeader(cycle, pid);
- num_entries_ = 4;
- }
-
- entry.cycle = cycle + kBaseTic;
- entry.event = (filenum << 13) | (procnum << 1) | is_exit;
- fwrite(&entry, sizeof(entry), 1, ftrace_);
- num_entries_ += 1;
-}
-
-void Gtrace::AddProcEntry(int filenum, int procnum, uint32_t cycle, uint32_t pid)
-{
- AddGtraceRecord(filenum, procnum, cycle, pid, 0);
-}
-
-void Gtrace::AddProcExit(int filenum, int procnum, uint32_t cycle, uint32_t pid)
-{
- AddGtraceRecord(filenum, procnum, cycle, pid, 1);
-}
-
-void Gtrace::AddProcedure(int filenum, int procnum, const char *proc_name)
-{
- fprintf(fnames_, "%d %d %d %s\n", filenum, procnum, procnum, proc_name);
-}
diff --git a/emulator/qtools/gtrace.h b/emulator/qtools/gtrace.h
deleted file mode 100644
index 542adc2..0000000
--- a/emulator/qtools/gtrace.h
+++ /dev/null
@@ -1,69 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef GTRACE_H
-#define GTRACE_H
-
-class Gtrace {
- public:
- static const int kGtraceEntriesPerBlock = 1024;
- static const uint32_t kMillion = 1000000;
- static const uint32_t kTicsPerSecond = 200 * kMillion;
- static const int kBaseTic = 0x1000;
-
- struct trace_entry {
- uint32_t cycle;
- uint32_t event;
- };
-
- struct block_header {
- uint32_t blockno;
- uint32_t entry_width;
- uint32_t block_tic;
- uint32_t block_time;
- uint32_t usec_cpu;
- uint32_t pid;
- uint32_t bug_count;
- uint32_t zero_count;
- };
-
- struct first_header {
- block_header common;
- uint32_t tic;
- uint32_t one;
- uint32_t tics_per_second;
- uint32_t trace_time;
- uint32_t version;
- uint32_t file_proc;
- uint32_t pdate;
- uint32_t ptime;
- };
-
- Gtrace();
- ~Gtrace();
-
- void Open(const char *gtrace_file, uint32_t pdate, uint32_t ptime);
- void WriteFirstHeader(uint32_t start_sec, uint32_t pid);
- void AddProcedure(int filenum, int procnum, const char *proc_name);
- void AddProcEntry(int filenum, int procnum, uint32_t cycle, uint32_t pid);
- void AddProcExit(int filenum, int procnum, uint32_t cycle, uint32_t pid);
-
- private:
- void AddGtraceRecord(int filenum, int procnum, uint32_t cycle, uint32_t pid,
- int is_exit);
- void FillFirstHeader(uint32_t start_sec, uint32_t pid,
- first_header *fh);
- void WriteBlockHeader(uint32_t cycle, uint32_t pid);
-
- const char *gtrace_file_;
- char gname_file_[100];
- FILE *ftrace_;
- FILE *fnames_;
- uint32_t start_sec_;
- uint32_t pdate_;
- uint32_t ptime_;
- int num_entries_;
- int blockno_;
- uint32_t current_pid_;
-};
-
-#endif // GTRACE_H
diff --git a/emulator/qtools/hash_table.h b/emulator/qtools/hash_table.h
deleted file mode 100644
index 4ea9ed5..0000000
--- a/emulator/qtools/hash_table.h
+++ /dev/null
@@ -1,219 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef HASH_TABLE_H
-#define HASH_TABLE_H
-
-#include <string.h>
-#include <inttypes.h>
-
-template<class T>
-class HashTable {
- public:
- HashTable(int size, T default_value = T());
- ~HashTable();
-
- typedef struct entry {
- entry *next;
- char *key;
- T value;
- } entry_type;
-
- typedef T value_type;
-
- void Update(const char *key, T value);
- bool Remove(const char *key);
- T Find(const char *key);
- entry_type* GetFirst();
- entry_type* GetNext();
-
- private:
- uint32_t HashFunction(const char *key);
-
- int size_;
- int mask_;
- T default_value_;
- entry_type **table_;
- int num_entries_;
- int current_index_;
- entry_type *current_ptr_;
-};
-
-template<class T>
-HashTable<T>::HashTable(int size, T default_value)
-{
- int pow2;
-
- // Round up size to a power of two
- for (pow2 = 2; pow2 < size; pow2 <<= 1)
- ; // empty body
-
- size_ = pow2;
- mask_ = pow2 - 1;
- default_value_ = default_value;
-
- // Allocate a table of pointers and initialize them all to NULL.
- table_ = new entry_type*[size_];
- for (int ii = 0; ii < size_; ++ii)
- table_[ii] = NULL;
- num_entries_ = 0;
- current_index_ = 0;
- current_ptr_ = NULL;
-}
-
-template<class T>
-HashTable<T>::~HashTable()
-{
- for (int ii = 0; ii < size_; ++ii) {
- entry_type *ptr, *next;
-
- // Delete all the pointers in the chain at this table position.
- // Save the next pointer before deleting each entry so that we
- // don't dereference part of a deallocated object.
- for (ptr = table_[ii]; ptr; ptr = next) {
- next = ptr->next;
- delete[] ptr->key;
- delete ptr;
- }
- }
- delete[] table_;
-}
-
-// Professor Daniel J. Bernstein's hash function. See
-// http://www.partow.net/programming/hashfunctions/
-template<class T>
-uint32_t HashTable<T>::HashFunction(const char *key)
-{
- uint32_t hash = 5381;
-
- int len = strlen(key);
- for(int ii = 0; ii < len; ++key, ++ii)
- hash = ((hash << 5) + hash) + *key;
-
- return hash;
-}
-
-template<class T>
-void HashTable<T>::Update(const char *key, T value)
-{
- // Hash the key to get the table position
- int len = strlen(key);
- int pos = HashFunction(key) & mask_;
-
- // Search the chain for a matching key
- for (entry_type *ptr = table_[pos]; ptr; ptr = ptr->next) {
- if (strcmp(ptr->key, key) == 0) {
- ptr->value = value;
- return;
- }
- }
-
- // Create a new hash entry and fill in the values
- entry_type *ptr = new entry_type;
-
- // Copy the string
- ptr->key = new char[len + 1];
- strcpy(ptr->key, key);
- ptr->value = value;
-
- // Insert the new entry at the beginning of the list
- ptr->next = table_[pos];
- table_[pos] = ptr;
- num_entries_ += 1;
-}
-
-template<class T>
-bool HashTable<T>::Remove(const char *key)
-{
- // Hash the key to get the table position
- int len = strlen(key);
- int pos = HashFunction(key) & mask_;
-
- // Search the chain for a matching key and keep track of the previous
- // element in the chain.
- entry_type *prev = NULL;
- for (entry_type *ptr = table_[pos]; ptr; prev = ptr, ptr = ptr->next) {
- if (strcmp(ptr->key, key) == 0) {
- if (prev == NULL) {
- table_[pos] = ptr->next;
- } else {
- prev->next = ptr->next;
- }
- delete ptr->key;
- delete ptr;
- return true;
- }
- }
- return false;
-}
-
-template<class T>
-typename HashTable<T>::value_type HashTable<T>::Find(const char *key)
-{
- // Hash the key to get the table position
- int pos = HashFunction(key) & mask_;
-
- // Search the chain for a matching key
- for (entry_type *ptr = table_[pos]; ptr; ptr = ptr->next) {
- if (strcmp(ptr->key, key) == 0)
- return ptr->value;
- }
-
- // If we get here, then we didn't find the key
- return default_value_;
-}
-
-template<class T>
-typename HashTable<T>::entry_type* HashTable<T>::GetFirst()
-{
- // Find the first non-NULL table entry.
- for (current_index_ = 0; current_index_ < size_; ++current_index_) {
- if (table_[current_index_])
- break;
- }
-
- // If there are no table entries, then return NULL.
- if (current_index_ == size_)
- return NULL;
-
- // Remember and return the current element.
- current_ptr_ = table_[current_index_];
- return current_ptr_;
-}
-
-template<class T>
-typename HashTable<T>::entry_type* HashTable<T>::GetNext()
-{
- // If we already iterated part way through the hash table, then continue
- // to the next element.
- if (current_ptr_) {
- current_ptr_ = current_ptr_->next;
-
- // If we are pointing to a valid element, then return it.
- if (current_ptr_)
- return current_ptr_;
-
- // Otherwise, start searching at the next table index.
- current_index_ += 1;
- }
-
- // Find the next non-NULL table entry.
- for (; current_index_ < size_; ++current_index_) {
- if (table_[current_index_])
- break;
- }
-
- // If there are no more non-NULL table entries, then return NULL.
- if (current_index_ == size_) {
- // Reset the current index so that we will start over from the
- // beginning on the next call to GetNext().
- current_index_ = 0;
- return NULL;
- }
-
- // Remember and return the current element.
- current_ptr_ = table_[current_index_];
- return current_ptr_;
-}
-
-
-#endif // HASH_TABLE_H
diff --git a/emulator/qtools/hist_trace.cpp b/emulator/qtools/hist_trace.cpp
deleted file mode 100644
index d2c6a90..0000000
--- a/emulator/qtools/hist_trace.cpp
+++ /dev/null
@@ -1,64 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-
-static const int kMaxHistEntries = 256;
-static const int kMaxHistEntries2 = kMaxHistEntries / 2;
-int hist[kMaxHistEntries];
-int underflow, overflow;
-
-int main(int argc, char **argv) {
- if (argc != 2) {
- fprintf(stderr, "Usage: %s trace_file\n", argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[1];
- TraceReaderBase *trace = new TraceReaderBase;
- trace->Open(trace_filename);
-
- uint64_t prev_bb_num = 0;
- uint64_t prev_time = 0;
- int total = 0;
-
- while (1) {
- BBEvent event;
-
- if (trace->ReadBB(&event))
- break;
- int bb_diff = event.bb_num - prev_bb_num;
- //int time_diff = event.time - prev_time;
- //printf("bb_num: %llu prev: %llu, diff: %d\n",
- // event.bb_num, prev_bb_num, bb_diff);
- prev_bb_num = event.bb_num;
- prev_time = event.time;
-
- bb_diff += kMaxHistEntries2;
- if (bb_diff < 0)
- underflow += 1;
- else if (bb_diff >= kMaxHistEntries)
- overflow += 1;
- else
- hist[bb_diff] += 1;
- total += 1;
- }
-
- int sum = 0;
- double sum_per = 0;
- double per = 0;
- for (int ii = 0; ii < kMaxHistEntries; ++ii) {
- if (hist[ii] == 0)
- continue;
- per = 100.0 * hist[ii] / total;
- sum += hist[ii];
- sum_per = 100.0 * sum / total;
- printf(" %4d: %6d %6.2f %6.2f\n", ii - kMaxHistEntries2, hist[ii], per, sum_per);
- }
- per = 100.0 * underflow / total;
- printf("under: %6d %6.2f\n", underflow, per);
- per = 100.0 * overflow / total;
- printf("over: %6d %6.2f\n", overflow, per);
- printf("total: %6d\n", total);
- return 0;
-}
diff --git a/emulator/qtools/opcode.cpp b/emulator/qtools/opcode.cpp
deleted file mode 100644
index 41bef3a..0000000
--- a/emulator/qtools/opcode.cpp
+++ /dev/null
@@ -1,204 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <inttypes.h>
-#include "opcode.h"
-
-// Note: this array depends on the Opcode enum defined in opcode.h
-uint32_t opcode_flags[] = {
- 0, // OP_INVALID
- 0, // OP_UNDEFINED
- kCatAlu, // OP_ADC
- kCatAlu, // OP_ADD
- kCatAlu, // OP_AND
- kCatBranch, // OP_B
- kCatBranch | kCatBranchLink, // OP_BL
- kCatAlu, // OP_BIC
- 0, // OP_BKPT
- kCatBranch | kCatBranchLink | kCatBranchExch, // OP_BLX
- kCatBranch | kCatBranchExch, // OP_BX
- kCatCoproc, // OP_CDP
- kCatAlu, // OP_CLZ
- kCatAlu, // OP_CMN
- kCatAlu, // OP_CMP
- kCatAlu, // OP_EOR
- kCatCoproc | kCatLoad, // OP_LDC
- kCatLoad | kCatMultiple, // OP_LDM
- kCatLoad | kCatWord, // OP_LDR
- kCatLoad | kCatByte, // OP_LDRB
- kCatLoad | kCatByte, // OP_LDRBT
- kCatLoad | kCatHalf, // OP_LDRH
- kCatLoad | kCatByte | kCatSigned, // OP_LDRSB
- kCatLoad | kCatHalf | kCatSigned, // OP_LDRSH
- kCatLoad | kCatWord, // OP_LDRT
- kCatCoproc, // OP_MCR
- kCatAlu, // OP_MLA
- kCatAlu, // OP_MOV
- kCatCoproc, // OP_MRC
- 0, // OP_MRS
- 0, // OP_MSR
- kCatAlu, // OP_MUL
- kCatAlu, // OP_MVN
- kCatAlu, // OP_ORR
- 0, // OP_PLD
- kCatAlu, // OP_RSB
- kCatAlu, // OP_RSC
- kCatAlu, // OP_SBC
- kCatAlu, // OP_SMLAL
- kCatAlu, // OP_SMULL
- kCatCoproc | kCatStore, // OP_STC
- kCatStore | kCatMultiple, // OP_STM
- kCatStore | kCatWord, // OP_STR
- kCatStore | kCatByte, // OP_STRB
- kCatStore | kCatByte, // OP_STRBT
- kCatStore | kCatHalf, // OP_STRH
- kCatStore | kCatWord, // OP_STRT
- kCatAlu, // OP_SUB
- 0, // OP_SWI
- kCatLoad | kCatStore, // OP_SWP
- kCatLoad | kCatStore | kCatByte, // OP_SWPB
- kCatAlu, // OP_TEQ
- kCatAlu, // OP_TST
- kCatAlu, // OP_UMLAL
- kCatAlu, // OP_UMULL
-
- 0, // OP_THUMB_UNDEFINED,
- kCatAlu, // OP_THUMB_ADC,
- kCatAlu, // OP_THUMB_ADD,
- kCatAlu, // OP_THUMB_AND,
- kCatAlu, // OP_THUMB_ASR,
- kCatBranch, // OP_THUMB_B,
- kCatAlu, // OP_THUMB_BIC,
- 0, // OP_THUMB_BKPT,
- kCatBranch | kCatBranchLink, // OP_THUMB_BL,
- kCatBranch | kCatBranchLink | kCatBranchExch, // OP_THUMB_BLX,
- kCatBranch | kCatBranchExch, // OP_THUMB_BX,
- kCatAlu, // OP_THUMB_CMN,
- kCatAlu, // OP_THUMB_CMP,
- kCatAlu, // OP_THUMB_EOR,
- kCatLoad | kCatMultiple, // OP_THUMB_LDMIA,
- kCatLoad | kCatWord, // OP_THUMB_LDR,
- kCatLoad | kCatByte, // OP_THUMB_LDRB,
- kCatLoad | kCatHalf, // OP_THUMB_LDRH,
- kCatLoad | kCatByte | kCatSigned, // OP_THUMB_LDRSB,
- kCatLoad | kCatHalf | kCatSigned, // OP_THUMB_LDRSH,
- kCatAlu, // OP_THUMB_LSL,
- kCatAlu, // OP_THUMB_LSR,
- kCatAlu, // OP_THUMB_MOV,
- kCatAlu, // OP_THUMB_MUL,
- kCatAlu, // OP_THUMB_MVN,
- kCatAlu, // OP_THUMB_NEG,
- kCatAlu, // OP_THUMB_ORR,
- kCatLoad | kCatMultiple, // OP_THUMB_POP,
- kCatStore | kCatMultiple, // OP_THUMB_PUSH,
- kCatAlu, // OP_THUMB_ROR,
- kCatAlu, // OP_THUMB_SBC,
- kCatStore | kCatMultiple, // OP_THUMB_STMIA,
- kCatStore | kCatWord, // OP_THUMB_STR,
- kCatStore | kCatByte, // OP_THUMB_STRB,
- kCatStore | kCatHalf, // OP_THUMB_STRH,
- kCatAlu, // OP_THUMB_SUB,
- 0, // OP_THUMB_SWI,
- kCatAlu, // OP_THUMB_TST,
-
- 0, // OP_END
-};
-
-const char *opcode_names[] = {
- "invalid",
- "undefined",
- "adc",
- "add",
- "and",
- "b",
- "bl",
- "bic",
- "bkpt",
- "blx",
- "bx",
- "cdp",
- "clz",
- "cmn",
- "cmp",
- "eor",
- "ldc",
- "ldm",
- "ldr",
- "ldrb",
- "ldrbt",
- "ldrh",
- "ldrsb",
- "ldrsh",
- "ldrt",
- "mcr",
- "mla",
- "mov",
- "mrc",
- "mrs",
- "msr",
- "mul",
- "mvn",
- "orr",
- "pld",
- "rsb",
- "rsc",
- "sbc",
- "smlal",
- "smull",
- "stc",
- "stm",
- "str",
- "strb",
- "strbt",
- "strh",
- "strt",
- "sub",
- "swi",
- "swp",
- "swpb",
- "teq",
- "tst",
- "umlal",
- "umull",
-
- "undefined",
- "adc",
- "add",
- "and",
- "asr",
- "b",
- "bic",
- "bkpt",
- "bl",
- "blx",
- "bx",
- "cmn",
- "cmp",
- "eor",
- "ldmia",
- "ldr",
- "ldrb",
- "ldrh",
- "ldrsb",
- "ldrsh",
- "lsl",
- "lsr",
- "mov",
- "mul",
- "mvn",
- "neg",
- "orr",
- "pop",
- "push",
- "ror",
- "sbc",
- "stmia",
- "str",
- "strb",
- "strh",
- "sub",
- "swi",
- "tst",
-
- NULL
-};
diff --git a/emulator/qtools/opcode.h b/emulator/qtools/opcode.h
deleted file mode 100644
index c8b67a6..0000000
--- a/emulator/qtools/opcode.h
+++ /dev/null
@@ -1,166 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef OPCODE_H
-#define OPCODE_H
-
-#include <inttypes.h>
-
-// Note: this list of opcodes must match the list used to initialize
-// the opflags[] array in opcode.cpp.
-enum Opcode {
- OP_INVALID,
- OP_UNDEFINED,
- OP_ADC,
- OP_ADD,
- OP_AND,
- OP_B,
- OP_BL,
- OP_BIC,
- OP_BKPT,
- OP_BLX,
- OP_BX,
- OP_CDP,
- OP_CLZ,
- OP_CMN,
- OP_CMP,
- OP_EOR,
- OP_LDC,
- OP_LDM,
- OP_LDR,
- OP_LDRB,
- OP_LDRBT,
- OP_LDRH,
- OP_LDRSB,
- OP_LDRSH,
- OP_LDRT,
- OP_MCR,
- OP_MLA,
- OP_MOV,
- OP_MRC,
- OP_MRS,
- OP_MSR,
- OP_MUL,
- OP_MVN,
- OP_ORR,
- OP_PLD,
- OP_RSB,
- OP_RSC,
- OP_SBC,
- OP_SMLAL,
- OP_SMULL,
- OP_STC,
- OP_STM,
- OP_STR,
- OP_STRB,
- OP_STRBT,
- OP_STRH,
- OP_STRT,
- OP_SUB,
- OP_SWI,
- OP_SWP,
- OP_SWPB,
- OP_TEQ,
- OP_TST,
- OP_UMLAL,
- OP_UMULL,
-
- // Define thumb opcodes
- OP_THUMB_UNDEFINED,
- OP_THUMB_ADC,
- OP_THUMB_ADD,
- OP_THUMB_AND,
- OP_THUMB_ASR,
- OP_THUMB_B,
- OP_THUMB_BIC,
- OP_THUMB_BKPT,
- OP_THUMB_BL,
- OP_THUMB_BLX,
- OP_THUMB_BX,
- OP_THUMB_CMN,
- OP_THUMB_CMP,
- OP_THUMB_EOR,
- OP_THUMB_LDMIA,
- OP_THUMB_LDR,
- OP_THUMB_LDRB,
- OP_THUMB_LDRH,
- OP_THUMB_LDRSB,
- OP_THUMB_LDRSH,
- OP_THUMB_LSL,
- OP_THUMB_LSR,
- OP_THUMB_MOV,
- OP_THUMB_MUL,
- OP_THUMB_MVN,
- OP_THUMB_NEG,
- OP_THUMB_ORR,
- OP_THUMB_POP,
- OP_THUMB_PUSH,
- OP_THUMB_ROR,
- OP_THUMB_SBC,
- OP_THUMB_STMIA,
- OP_THUMB_STR,
- OP_THUMB_STRB,
- OP_THUMB_STRH,
- OP_THUMB_SUB,
- OP_THUMB_SWI,
- OP_THUMB_TST,
-
- OP_END // must be last
-};
-
-extern uint32_t opcode_flags[];
-extern const char *opcode_names[];
-
-// Define bit flags for the opcode categories
-static const uint32_t kCatByte = 0x0001;
-static const uint32_t kCatHalf = 0x0002;
-static const uint32_t kCatWord = 0x0004;
-static const uint32_t kCatLong = 0x0008;
-static const uint32_t kCatNumBytes = (kCatByte | kCatHalf | kCatWord | kCatLong);
-static const uint32_t kCatMultiple = 0x0010;
-static const uint32_t kCatSigned = 0x0020;
-static const uint32_t kCatLoad = 0x0040;
-static const uint32_t kCatStore = 0x0080;
-static const uint32_t kCatMemoryRef = (kCatLoad | kCatStore);
-static const uint32_t kCatAlu = 0x0100;
-static const uint32_t kCatBranch = 0x0200;
-static const uint32_t kCatBranchLink = 0x0400;
-static const uint32_t kCatBranchExch = 0x0800;
-static const uint32_t kCatCoproc = 0x1000;
-static const uint32_t kCatLoadMultiple = (kCatLoad | kCatMultiple);
-static const uint32_t kCatStoreMultiple = (kCatStore | kCatMultiple);
-
-inline bool isALU(Opcode op) { return (opcode_flags[op] & kCatAlu) != 0; }
-inline bool isBranch(Opcode op) { return (opcode_flags[op] & kCatBranch) != 0; }
-inline bool isBranchLink(Opcode op) {
- return (opcode_flags[op] & kCatBranchLink) != 0;
-}
-inline bool isBranchExch(Opcode op) {
- return (opcode_flags[op] & kCatBranchExch) != 0;
-}
-inline bool isLoad(Opcode op) { return (opcode_flags[op] & kCatLoad) != 0; }
-inline bool isLoadMultiple(Opcode op) {
- return (opcode_flags[op] & kCatLoadMultiple) == kCatLoadMultiple;
-}
-inline bool isStoreMultiple(Opcode op) {
- return (opcode_flags[op] & kCatStoreMultiple) == kCatStoreMultiple;
-}
-inline bool isStore(Opcode op) { return (opcode_flags[op] & kCatStore) != 0; }
-inline bool isSigned(Opcode op) { return (opcode_flags[op] & kCatSigned) != 0; }
-inline bool isMemoryRef(Opcode op) {
- return (opcode_flags[op] & kCatMemoryRef) != 0;
-}
-inline int getAccessSize(Opcode op) { return opcode_flags[op] & kCatNumBytes; }
-inline bool isCoproc(Opcode op) { return (opcode_flags[op] & kCatCoproc) != 0; }
-inline int getNumAccesses(Opcode op, uint32_t binary) {
- extern int num_one_bits[];
- int num_accesses = 0;
- if (opcode_flags[op] & kCatNumBytes)
- num_accesses = 1;
- else if (opcode_flags[op] & kCatMultiple) {
- num_accesses = num_one_bits[(binary >> 8) & 0xff]
- + num_one_bits[binary & 0xff];
- }
- return num_accesses;
-}
-
-#endif // OPCODE_H
diff --git a/emulator/qtools/parse_options-inl.h b/emulator/qtools/parse_options-inl.h
deleted file mode 100644
index beb9df4..0000000
--- a/emulator/qtools/parse_options-inl.h
+++ /dev/null
@@ -1,155 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef PARSE_OPTIONS_INL_H
-#define PARSE_OPTIONS_INL_H
-
-// Define a typedef for TraceReaderType and include "parse_options.h"
-// before including this header file in a C++ source file.
-//
-// For example:
-//
-// struct symbol {
-// int elapsed;
-// };
-//
-// typedef TraceReader<symbol> TraceReaderType;
-
-
-typedef TraceReaderType::symbol_type symbol_type;
-typedef TraceReaderType::region_type region_type;
-typedef TraceReaderType::ProcessState ProcessState;
-
-symbol_type *kernel_sym;
-symbol_type *library_sym;
-
-// Returns true if the given event is included (or not excluded)
-// from the list of valid events specified by the user on the
-// command line.
-inline bool IsValidEvent(BBEvent *event, symbol_type *sym)
-{
- if (include_some_pids && pid_include_vector.GetBit(event->pid) == 0)
- return false;
- if (exclude_some_pids && pid_exclude_vector.GetBit(event->pid))
- return false;
- if (include_some_procedures) {
- if (sym == NULL || included_procedures.Find(sym->name) == 0)
- return false;
- }
- if (exclude_some_procedures) {
- if (sym == NULL || excluded_procedures.Find(sym->name))
- return false;
- }
- return true;
-}
-
-inline bool IsValidPid(int pid) {
- if (include_some_pids && pid_include_vector.GetBit(pid) == 0)
- return false;
- if (exclude_some_pids && pid_exclude_vector.GetBit(pid))
- return false;
- return true;
-}
-
-inline symbol_type *GetSymbol(TraceReaderType *trace, int pid, uint32_t addr,
- uint64_t time)
-{
- symbol_type *sym = trace->LookupFunction(pid, addr, time);
-
- if (lump_kernel && (sym->region->flags & region_type::kIsKernelRegion)) {
- if (kernel_sym == NULL) {
- kernel_sym = sym;
- sym->name = ":kernel";
- } else {
- sym = kernel_sym;
- }
- }
-
- if (lump_libraries && (sym->region->flags & region_type::kIsLibraryRegion)) {
- if (library_sym == NULL) {
- library_sym = sym;
- sym->name = ":libs";
- } else {
- sym = library_sym;
- }
- }
-
- return sym;
-}
-
-inline bool IsIncludedProcedure(symbol_type *sym)
-{
- if (include_kernel_syms && (sym->region->flags & region_type::kIsKernelRegion))
- return true;
- if (include_library_syms && (sym->region->flags & region_type::kIsLibraryRegion))
- return true;
- return included_procedures.Find(sym->name);
-}
-
-inline bool IsExcludedProcedure(symbol_type *sym)
-{
- if (exclude_kernel_syms && (sym->region->flags & region_type::kIsKernelRegion))
- return true;
- if (exclude_library_syms && (sym->region->flags & region_type::kIsLibraryRegion))
- return true;
- return excluded_procedures.Find(sym->name);
-}
-
-// Returns true on end-of-file.
-inline bool GetNextValidEvent(TraceReaderType *trace,
- BBEvent *event,
- BBEvent *first_ignored_event,
- symbol_type **sym_ptr)
-{
- symbol_type *sym = NULL;
- first_ignored_event->time = 0;
- if (trace->ReadBB(event))
- return true;
- bool recheck = true;
- while (recheck) {
- recheck = false;
- if (include_some_pids) {
- while (pid_include_vector.GetBit(event->pid) == 0) {
- if (first_ignored_event->time == 0)
- *first_ignored_event = *event;
- if (trace->ReadBB(event))
- return true;
- }
- } else if (exclude_some_pids) {
- while (pid_exclude_vector.GetBit(event->pid)) {
- if (first_ignored_event->time == 0)
- *first_ignored_event = *event;
- if (trace->ReadBB(event))
- return true;
- }
- }
-
- if (include_some_procedures) {
- sym = GetSymbol(trace, event->pid, event->bb_addr, event->time);
- while (!IsIncludedProcedure(sym)) {
- if (first_ignored_event->time == 0)
- *first_ignored_event = *event;
- if (trace->ReadBB(event))
- return true;
- recheck = true;
- sym = GetSymbol(trace, event->pid, event->bb_addr, event->time);
- }
- } else if (exclude_some_procedures) {
- sym = GetSymbol(trace, event->pid, event->bb_addr, event->time);
- while (IsExcludedProcedure(sym)) {
- if (first_ignored_event->time == 0)
- *first_ignored_event = *event;
- if (trace->ReadBB(event))
- return true;
- recheck = true;
- sym = GetSymbol(trace, event->pid, event->bb_addr, event->time);
- }
- }
- }
- if (sym == NULL)
- sym = GetSymbol(trace, event->pid, event->bb_addr, event->time);
-
- *sym_ptr = sym;
- return false;
-}
-
-#endif // PARSE_OPTIONS_INL_H
diff --git a/emulator/qtools/parse_options.cpp b/emulator/qtools/parse_options.cpp
deleted file mode 100644
index 395e9a1..0000000
--- a/emulator/qtools/parse_options.cpp
+++ /dev/null
@@ -1,119 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <unistd.h>
-#include <ctype.h>
-#include "bitvector.h"
-#include "parse_options.h"
-#include "hash_table.h"
-
-const char *root = "";
-bool lump_kernel;
-bool lump_libraries;
-Bitvector pid_include_vector(32768);
-Bitvector pid_exclude_vector(32768);
-bool include_some_pids;
-bool exclude_some_pids;
-
-HashTable<int> excluded_procedures(2000);
-HashTable<int> included_procedures(2000);
-bool exclude_some_procedures;
-bool include_some_procedures;
-
-bool exclude_kernel_syms;
-bool exclude_library_syms;
-bool include_kernel_syms;
-bool include_library_syms;
-bool demangle = true;
-
-static const char *OptionsUsageStr =
- " -e :kernel exclude all kernel symbols\n"
- " -e :libs exclude all library symbols\n"
- " -e <func> exclude function <func>\n"
- " -e <pid> exclude process <pid>\n"
- " -i :kernel include all kernel symbols\n"
- " -i :libs include all library symbols\n"
- " -i <func> include function <func>\n"
- " -i <pid> include process <pid>\n"
- " -l :kernel lump all the kernel symbols together\n"
- " -l :libs lump all the library symbols together\n"
- " -m do not demangle C++ symbols (m for 'mangle')\n"
- " -r <root> use <root> as the path for finding ELF executables\n"
- ;
-
-void OptionsUsage()
-{
- fprintf(stderr, "%s", OptionsUsageStr);
-}
-
-void ParseOptions(int argc, char **argv)
-{
- bool err = false;
- while (!err) {
- int opt = getopt(argc, argv, "+e:i:l:mr:");
- if (opt == -1)
- break;
- switch (opt) {
- case 'e':
- if (*optarg == ':') {
- if (strcmp(optarg, ":kernel") == 0)
- exclude_kernel_syms = true;
- else if (strcmp(optarg, ":libs") == 0)
- exclude_library_syms = true;
- else
- err = true;
- excluded_procedures.Update(optarg, 1);
- exclude_some_procedures = true;
- } else if (isdigit(*optarg)) {
- int bitnum = atoi(optarg);
- pid_exclude_vector.SetBit(bitnum);
- exclude_some_pids = true;
- } else {
- excluded_procedures.Update(optarg, 1);
- exclude_some_procedures = true;
- }
- break;
- case 'i':
- if (*optarg == ':') {
- if (strcmp(optarg, ":kernel") == 0)
- include_kernel_syms = true;
- else if (strcmp(optarg, ":libs") == 0)
- include_library_syms = true;
- else
- err = true;
- included_procedures.Update(optarg, 1);
- include_some_procedures = true;
- } else if (isdigit(*optarg)) {
- int bitnum = atoi(optarg);
- pid_include_vector.SetBit(bitnum);
- include_some_pids = true;
- } else {
- included_procedures.Update(optarg, 1);
- include_some_procedures = true;
- }
- break;
- case 'l':
- if (strcmp(optarg, ":kernel") == 0)
- lump_kernel = true;
- else if (strcmp(optarg, ":libs") == 0)
- lump_libraries = true;
- else
- err = true;
- break;
- case 'm':
- demangle = false;
- break;
- case 'r':
- root = optarg;
- break;
- default:
- err = true;
- break;
- }
- }
-
- if (err) {
- Usage(argv[0]);
- exit(1);
- }
-}
diff --git a/emulator/qtools/parse_options.h b/emulator/qtools/parse_options.h
deleted file mode 100644
index aacbb9e..0000000
--- a/emulator/qtools/parse_options.h
+++ /dev/null
@@ -1,32 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef PARSE_OPTIONS_H
-#define PARSE_OPTIONS_H
-
-#include "bitvector.h"
-#include "hash_table.h"
-
-extern const char *root;
-extern bool lump_kernel;
-extern bool lump_libraries;
-extern Bitvector pid_include_vector;
-extern Bitvector pid_exclude_vector;
-extern bool include_some_pids;
-extern bool exclude_some_pids;
-
-extern HashTable<int> excluded_procedures;
-extern HashTable<int> included_procedures;
-extern bool exclude_some_procedures;
-extern bool include_some_procedures;
-
-extern bool exclude_kernel_syms;
-extern bool exclude_library_syms;
-extern bool include_kernel_syms;
-extern bool include_library_syms;
-extern bool demangle;
-
-extern void Usage(const char *program);
-extern void ParseOptions(int argc, char **argv);
-extern void OptionsUsage();
-
-#endif // PARSE_OPTIONS_H
diff --git a/emulator/qtools/post_trace.cpp b/emulator/qtools/post_trace.cpp
deleted file mode 100644
index becfc2b..0000000
--- a/emulator/qtools/post_trace.cpp
+++ /dev/null
@@ -1,151 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-
-typedef struct MyStaticRec {
- StaticRec bb;
- uint32_t *insns;
-} MyStaticRec;
-
-const int kNumPids = 32768;
-char usedPids[kNumPids];
-
-int main(int argc, char **argv) {
- uint32_t insns[kMaxInsnPerBB];
-
- if (argc != 2) {
- fprintf(stderr, "Usage: %s trace_file\n", argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[1];
- TraceReaderBase *trace = new TraceReaderBase;
- trace->SetPostProcessing(true);
- trace->Open(trace_filename);
-
- // Count the number of static basic blocks and instructions.
- uint64_t num_static_bb = 0;
- uint64_t num_static_insn = 0;
- while (1) {
- StaticRec static_rec;
-
- if (trace->ReadStatic(&static_rec))
- break;
- if (static_rec.bb_num != num_static_bb) {
- fprintf(stderr,
- "Error: basic block numbers out of order; expected %lld, got %lld\n",
- num_static_bb, static_rec.bb_num);
- exit(1);
- }
- num_static_bb += 1;
- num_static_insn += static_rec.num_insns;
- trace->ReadStaticInsns(static_rec.num_insns, insns);
- }
- trace->Close();
-
- // Allocate space for all of the static blocks
- MyStaticRec *blocks = new MyStaticRec[num_static_bb];
-
- // Read the static blocks again and save pointers to them
- trace->Open(trace_filename);
- for (uint32_t ii = 0; ii < num_static_bb; ++ii) {
- trace->ReadStatic(&blocks[ii].bb);
- uint32_t num_insns = blocks[ii].bb.num_insns;
- if (num_insns > 0) {
- blocks[ii].insns = new uint32_t[num_insns];
- trace->ReadStaticInsns(num_insns, blocks[ii].insns);
- }
- }
-
- // Check the last basic block. If it contains a special undefined
- // instruction, then truncate the basic block at that point.
- uint32_t num_insns = blocks[num_static_bb - 1].bb.num_insns;
- uint32_t *insn_ptr = blocks[num_static_bb - 1].insns;
- for (uint32_t ii = 0; ii < num_insns; ++ii, ++insn_ptr) {
- if (*insn_ptr == 0xe6c00110) {
- uint32_t actual_num_insns = ii + 1;
- blocks[num_static_bb - 1].bb.num_insns = actual_num_insns;
- num_static_insn -= (num_insns - actual_num_insns);
-
- // Write the changes back to the trace file
- trace->TruncateLastBlock(actual_num_insns);
- break;
- }
- }
- TraceHeader *header = trace->GetHeader();
- strcpy(header->ident, TRACE_IDENT);
- header->num_static_bb = num_static_bb;
- header->num_dynamic_bb = 0;
- header->num_static_insn = num_static_insn;
- header->num_dynamic_insn = 0;
- trace->WriteHeader(header);
-
- // Reopen the trace file in order to force the trace manager to reread
- // the static blocks now that we have written that information to the
- // header.
- trace->Close();
- trace->Open(trace_filename);
-
- // Count the number of dynamic executions of basic blocks and instructions.
- // Also keep track of which process ids are used.
- uint64_t num_dynamic_bb = 0;
- uint64_t num_dynamic_insn = 0;
- while (1) {
- BBEvent event;
-
- if (trace->ReadBB(&event))
- break;
- if (event.bb_num >= num_static_bb) {
- fprintf(stderr,
- "Error: basic block number (%lld) too large (num blocks: %lld)\n",
- event.bb_num, num_static_bb);
- exit(1);
- }
- usedPids[event.pid] = 1;
- num_dynamic_bb += 1;
- num_dynamic_insn += event.num_insns;
- }
-
- // Count the number of process ids that are used and remember the first
- // unused pid.
- int numUsedPids = 0;
- int unusedPid = -1;
- for (int pid = 0; pid < kNumPids; pid++) {
- if (usedPids[pid] == 1) {
- numUsedPids += 1;
- } else if (unusedPid == -1) {
- unusedPid = pid;
- }
- }
-
- // Rewrite the header with the dynamic counts
- header->num_dynamic_bb = num_dynamic_bb;
- header->num_dynamic_insn = num_dynamic_insn;
- header->num_used_pids = numUsedPids;
- header->first_unused_pid = unusedPid;
- trace->WriteHeader(header);
- trace->Close();
-
- printf("Static basic blocks: %llu, Dynamic basic blocks: %llu\n",
- num_static_bb, num_dynamic_bb);
- printf("Static instructions: %llu, Dynamic instructions: %llu\n",
- num_static_insn, num_dynamic_insn);
-
- double elapsed_secs = header->elapsed_usecs / 1000000.0;
- double insn_per_sec = 0;
- if (elapsed_secs != 0)
- insn_per_sec = num_dynamic_insn / elapsed_secs;
- const char *suffix = "";
- if (insn_per_sec >= 1000000) {
- insn_per_sec /= 1000000.0;
- suffix = "M";
- } else if (insn_per_sec > 1000) {
- insn_per_sec /= 1000.0;
- suffix = "K";
- }
- printf("Elapsed seconds: %.2f, simulated instructions/sec: %.1f%s\n",
- elapsed_secs, insn_per_sec, suffix);
- return 0;
-}
diff --git a/emulator/qtools/profile_pid.cpp b/emulator/qtools/profile_pid.cpp
deleted file mode 100644
index 11acbf9..0000000
--- a/emulator/qtools/profile_pid.cpp
+++ /dev/null
@@ -1,94 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <string.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-// This function is called from quicksort to compare the cpu time
-// of processes and sort into decreasing order.
-int cmp_dec_cpu_time(const void *a, const void *b) {
- ProcessState *proc1, *proc2;
-
- proc1 = (ProcessState*)a;
- proc2 = (ProcessState*)b;
- if (proc1 == NULL) {
- if (proc2 == NULL)
- return 0;
- return 1;
- }
- if (proc2 == NULL)
- return -1;
- if (proc1->cpu_time < proc2->cpu_time)
- return 1;
- if (proc1->cpu_time > proc2->cpu_time)
- return -1;
- // If the cpu_time times are the same, then sort into increasing
- // order of pid.
- return proc1->pid - proc2->pid;
-}
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file\n", program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv) {
- // Parse the options
- ParseOptions(argc, argv);
- if (argc - optind != 1) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind];
- TraceReader<> *trace = new TraceReader<>;
- trace->Open(trace_filename);
- trace->SetRoot(root);
-
- while (1) {
- BBEvent event, ignored;
- symbol_type *dummy_sym;
-
- if (GetNextValidEvent(trace, &event, &ignored, &dummy_sym))
- break;
- }
-
- int num_procs;
- ProcessState *processes = trace->GetProcesses(&num_procs);
- qsort(processes, num_procs, sizeof(ProcessState), cmp_dec_cpu_time);
-
- uint64_t total_time = 0;
- for (int ii = 0; ii < num_procs; ++ii) {
- total_time += processes[ii].cpu_time;
- }
-
- uint64_t sum_time = 0;
- printf(" pid parent cpu_time %% %% flags argv\n");
- ProcessState *pstate = &processes[0];
- for (int ii = 0; ii < num_procs; ++ii, ++pstate) {
- sum_time += pstate->cpu_time;
- double per = 100.0 * pstate->cpu_time / total_time;
- double sum_per = 100.0 * sum_time / total_time;
- const char *print_flags = "";
- if ((pstate->flags & ProcessState::kCalledExec) == 0)
- print_flags = "T";
- if (pstate->name == NULL)
- pstate->name = "";
- printf("%5d %5d %10llu %6.2f %6.2f %5s %s",
- pstate->pid, pstate->parent_pid, pstate->cpu_time,
- per, sum_per, print_flags, pstate->name);
- for (int jj = 1; jj < pstate->argc; ++jj) {
- printf(" %s", pstate->argv[jj]);
- }
- printf("\n");
- }
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/profile_trace.cpp b/emulator/qtools/profile_trace.cpp
deleted file mode 100644
index 0b056cc..0000000
--- a/emulator/qtools/profile_trace.cpp
+++ /dev/null
@@ -1,131 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <string.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-
-const int kMillion = 1000000;
-const int kMHz = 200 * kMillion;
-
-struct symbol {
- int count; // number of times a function is executed
- uint64_t elapsed; // elapsed time for this function
-};
-
-typedef TraceReader<symbol> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-static const uint32_t kOffsetThreshold = 0x100000;
-
-// This comparison function is called from qsort() to sort
-// symbols into decreasing elapsed time.
-int cmp_sym_elapsed(const void *a, const void *b) {
- const symbol_type *syma, *symb;
- uint64_t elapsed1, elapsed2;
-
- syma = static_cast<symbol_type const *>(a);
- symb = static_cast<symbol_type const *>(b);
- elapsed1 = syma->elapsed;
- elapsed2 = symb->elapsed;
- if (elapsed1 < elapsed2)
- return 1;
- if (elapsed1 == elapsed2)
- return strcmp(syma->name, symb->name);
- return -1;
-}
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file elf_file\n", program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv)
-{
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReader<symbol> *trace = new TraceReader<symbol>;
- trace->Open(trace_filename);
- trace->SetDemangle(demangle);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- symbol_type dummy;
- dummy.count = 0;
- dummy.elapsed = 0;
- symbol_type *prev_sym = &dummy;
- uint64_t prev_bb_time = 0;
- while (1) {
- symbol_type *sym;
- BBEvent event;
- BBEvent first_ignored_event;
-
- bool eof = GetNextValidEvent(trace, &event, &first_ignored_event, &sym);
-
- // Assign the elapsed time to the previous function symbol
- uint64_t elapsed = 0;
- if (first_ignored_event.time != 0)
- elapsed = first_ignored_event.time - prev_bb_time;
- else if (!eof)
- elapsed = event.time - prev_bb_time;
- prev_sym->elapsed += elapsed;
-
- if (eof)
- break;
-
- prev_bb_time = event.time;
- sym->count += 1;
- prev_sym = sym;
-#if 0
- printf("t%lld bb_num: %d, bb_addr: 0x%x func: %s, addr: 0x%x, count: %d\n",
- bb_time, bb_num, bb_addr, sym->name, sym->addr, sym->count);
-#endif
- }
-
- int nsyms;
- symbol_type *syms = trace->GetSymbols(&nsyms);
-
- // Sort the symbols into decreasing order of elapsed time
- qsort(syms, nsyms, sizeof(symbol_type), cmp_sym_elapsed);
-
- // Add up all the cycles
- uint64_t total = 0;
- symbol_type *sym = syms;
- for (int ii = 0; ii < nsyms; ++ii, ++sym) {
- total += sym->elapsed;
- }
-
- double secs = 1.0 * total / kMHz;
- printf("Total seconds: %.2f, total cycles: %lld, MHz: %d\n\n",
- secs, total, kMHz / kMillion);
-
- uint64_t sum = 0;
- printf("Elapsed secs Elapsed cyc %% %% Function\n");
- sym = syms;
- for (int ii = 0; ii < nsyms; ++ii, ++sym) {
- if (sym->elapsed == 0)
- break;
- sum += sym->elapsed;
- double per = 100.0 * sym->elapsed / total;
- double sum_per = 100.0 * sum / total;
- double secs = 1.0 * sym->elapsed / kMHz;
- const char *ksym = " ";
- if (sym->region->flags & region_type::kIsKernelRegion)
- ksym = "k";
- printf("%12.2f %11lld %6.2f %6.2f %s %s\n",
- secs, sym->elapsed, per, sum_per, ksym, sym->name);
- }
- delete[] syms;
- delete trace;
-
- return 0;
-}
diff --git a/emulator/qtools/q2dm.cpp b/emulator/qtools/q2dm.cpp
deleted file mode 100644
index 74dbaeb..0000000
--- a/emulator/qtools/q2dm.cpp
+++ /dev/null
@@ -1,274 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "bitvector.h"
-#include "parse_options.h"
-#include "dmtrace.h"
-#include "armdis.h"
-
-struct symbol {
- uint32_t id;
-};
-
-typedef TraceReader<symbol> TraceReaderType;
-
-#include "parse_options-inl.h"
-#include "callstack.h"
-
-DmTrace *dmtrace;
-
-class MyFrame : public StackFrame<symbol_type> {
- public:
- void push(int stackLevel, uint64_t time, CallStackBase *base);
- void pop(int stackLevel, uint64_t time, CallStackBase *base);
-};
-
-typedef CallStack<MyFrame> CallStackType;
-
-static const int kNumStackFrames = 500;
-static const int kMaxThreads = (32 * 1024);
-uint64_t thread_time[kMaxThreads];
-
-class FunctionStack {
- public:
- FunctionStack() {
- top = 0;
- }
- void push(symbol_type *sym) {
- if (top >= kNumStackFrames)
- return;
- frames[top] = sym;
- top += 1;
- }
-
- symbol_type* pop() {
- if (top <= 0) {
- return NULL;
- }
- top -= 1;
- return frames[top];
- }
-
- void showStack() {
- fprintf(stderr, "top %d\n", top);
- for (int ii = 0; ii < top; ii++) {
- fprintf(stderr, " %d: %s\n", ii, frames[ii]->name);
- }
- }
-
- private:
- int top;
- symbol_type *frames[kNumStackFrames];
-};
-
-FunctionStack *dmtrace_stack[kMaxThreads];
-
-void MyFrame::push(int stackLevel, uint64_t time, CallStackBase *base)
-{
- int pid = base->getId();
- CallStackType *stack = (CallStackType *) base;
-
-#if 0
- fprintf(stderr, "native push t %llu p %d s %d fid %d 0x%x %s\n",
- stack->getGlobalTime(time), pid, stackLevel,
- function->id, function->addr, function->name);
-#endif
-
- FunctionStack *fstack = dmtrace_stack[pid];
- if (fstack == NULL) {
- fstack = new FunctionStack();
- dmtrace_stack[pid] = fstack;
- }
-
- fstack->push(function);
- thread_time[pid] = time;
- dmtrace->addFunctionEntry(function->id, time, pid);
-}
-
-void MyFrame::pop(int stackLevel, uint64_t time, CallStackBase *base)
-{
- int pid = base->getId();
- CallStackType *stack = (CallStackType *) base;
-
-#if 0
- fprintf(stderr, "native pop t %llu p %d s %d fid %d 0x%x %s\n",
- stack->getGlobalTime(time), pid, stackLevel,
- function->id, function->addr, function->name);
-#endif
-
- FunctionStack *fstack = dmtrace_stack[pid];
- if (fstack == NULL) {
- fstack = new FunctionStack();
- dmtrace_stack[pid] = fstack;
- }
-
- symbol_type *sym = fstack->pop();
- if (sym != NULL && sym != function) {
- fprintf(stderr, "Error: q2dm function mismatch at time %llu pid %d sym %s\n",
- stack->getGlobalTime(time), pid, sym->name);
- fstack->showStack();
- exit(1);
- }
-
- thread_time[pid] = time;
- dmtrace->addFunctionExit(function->id, time, pid);
-}
-
-uint32_t nextFunctionId = 4;
-CallStackType *stacks[kMaxThreads];
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_name elf_file dmtrace_name\n",
- program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv)
-{
- bool useKernelStack = true;
-
- ParseOptions(argc, argv);
- if (argc - optind != 3) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *qemu_trace_file = argv[optind++];
- char *elf_file = argv[optind++];
- char *dmtrace_file = argv[optind++];
- TraceReaderType *trace = new TraceReaderType;
- trace->Open(qemu_trace_file);
- trace->SetDemangle(demangle);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
- TraceHeader *qheader = trace->GetHeader();
- uint64_t startTime = qheader->start_sec;
- startTime = (startTime << 32) | qheader->start_usec;
- int kernelPid = qheader->first_unused_pid;
-
- dmtrace = new DmTrace;
- dmtrace->open(dmtrace_file, startTime);
-
- bool inKernel = false;
- CallStackType *kernelStack = NULL;
- if (useKernelStack) {
- // Create a fake kernel thread stack where we will put all the kernel
- // code.
- kernelStack = new CallStackType(kernelPid, kNumStackFrames, trace);
- dmtrace->addThread(kernelPid, "(kernel)");
- }
-
- CallStackType *prevStack = NULL;
- BBEvent event;
- while (1) {
- BBEvent ignored;
- symbol_type *function;
-
- if (GetNextValidEvent(trace, &event, &ignored, &function))
- break;
- if (event.bb_num == 0)
- break;
-#if 0
- fprintf(stderr, "event t %llu p %d %s\n",
- event.time, event.pid, function->name);
-#endif
-
- CallStackType *pStack;
- if (useKernelStack) {
- uint32_t flags = function->region->flags;
- uint32_t region_mask = region_type::kIsKernelRegion
- | region_type::kIsUserMappedRegion;
- if ((flags & region_mask) == region_type::kIsKernelRegion) {
- // Use the kernel stack
- pStack = kernelStack;
- inKernel = true;
- } else {
- // If we were just in the kernel then pop off all of the
- // stack frames for the kernel thread.
- if (inKernel == true) {
- inKernel = false;
- kernelStack->popAll(event.time);
- }
-
- // Get the stack for the current thread
- pStack = stacks[event.pid];
- }
- } else {
- // Get the stack for the current thread
- pStack = stacks[event.pid];
- }
-
- // If the stack does not exist, then allocate a new one.
- if (pStack == NULL) {
- pStack = new CallStackType(event.pid, kNumStackFrames, trace);
- stacks[event.pid] = pStack;
- const char *name = trace->GetProcessName(event.pid);
- dmtrace->addThread(event.pid, name);
- }
-
- if (prevStack != pStack) {
- pStack->threadStart(event.time);
- if (prevStack)
- prevStack->threadStop(event.time);
- }
- prevStack = pStack;
-
- // If we have never seen this function before, then add it to the
- // list of known functions.
- if (function->id == 0) {
- function->id = nextFunctionId;
- nextFunctionId += 4;
- uint32_t flags = function->region->flags;
- const char *name = function->name;
- if (flags & region_type::kIsKernelRegion) {
- // To distinguish kernel function names from user library
- // names, add a marker to the name.
- int len = strlen(name) + strlen(" [kernel]") + 1;
- char *kernelName = new char[len];
- strcpy(kernelName, name);
- strcat(kernelName, " [kernel]");
- name = kernelName;
- }
- dmtrace->parseAndAddFunction(function->id, name);
- }
-
- // Update the stack
- pStack->updateStack(&event, function);
- }
-
- if (prevStack == NULL) {
- fprintf(stderr, "Error: no events in trace.\n");
- exit(1);
- }
- prevStack->threadStop(event.time);
- for (int ii = 0; ii < kMaxThreads; ++ii) {
- if (stacks[ii]) {
- stacks[ii]->threadStart(event.time);
- stacks[ii]->popAll(event.time);
- }
- }
- if (useKernelStack) {
- kernelStack->popAll(event.time);
- }
-
- // Read the pid events to find the names of the processes
- while (1) {
- PidEvent pid_event;
- if (trace->ReadPidEvent(&pid_event))
- break;
- if (pid_event.rec_type == kPidName) {
- dmtrace->updateName(pid_event.pid, pid_event.path);
- }
- }
-
- dmtrace->close();
- delete dmtrace;
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/q2g.cpp b/emulator/qtools/q2g.cpp
deleted file mode 100644
index 6b2ae92..0000000
--- a/emulator/qtools/q2g.cpp
+++ /dev/null
@@ -1,108 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "gtrace.h"
-#include "bitvector.h"
-#include "parse_options.h"
-
-struct symbol {
- int filenum; // the file number (for gtrace)
- int procnum; // the procedure number (for gtrace)
-};
-
-typedef TraceReader<symbol> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-const int kMaxProcNum = 4095;
-int next_filenum = 1;
-int next_procnum = 1;
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_file elf_file gtrace_file\n",
- program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv)
-{
- ParseOptions(argc, argv);
- if (argc - optind != 3) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *qemu_trace_file = argv[optind++];
- char *elf_file = argv[optind++];
- char *gtrace_file = argv[optind++];
- TraceReader<symbol> *trace = new TraceReader<symbol>;
- trace->Open(qemu_trace_file);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
- TraceHeader *qheader = trace->GetHeader();
-
- // Get the first valid event to get the process id for the gtrace header.
- BBEvent event;
- BBEvent ignored;
- symbol_type *sym;
- if (GetNextValidEvent(trace, &event, &ignored, &sym))
- return 0;
-
- Gtrace *gtrace = new Gtrace;
- gtrace->Open(gtrace_file, qheader->pdate, qheader->ptime);
- gtrace->WriteFirstHeader(qheader->start_sec, event.pid);
-
- symbol_type *prev_sym = NULL;
- bool eof = false;
- while (!eof) {
- if (sym != prev_sym) {
- // This procedure is different from the previous procedure.
-
- // If we have never seen this symbol before, then add it to the
- // list of known procedures.
- if (sym->filenum == 0) {
- sym->filenum = next_filenum;
- sym->procnum = next_procnum;
- gtrace->AddProcedure(sym->filenum, sym->procnum, sym->name);
- next_procnum += 1;
- if (next_procnum > kMaxProcNum) {
- next_filenum += 1;
- next_procnum = 1;
- }
- }
-
- // If we haven't yet recorded the procedure exit for the previous
- // procedure, then do it now.
- if (prev_sym) {
- gtrace->AddProcExit(prev_sym->filenum, prev_sym->procnum, event.time,
- event.pid);
- }
-
- // If this is not the terminating record, then record a procedure
- // entry.
- if (event.bb_num != 0) {
- gtrace->AddProcEntry(sym->filenum, sym->procnum, event.time, event.pid);
- prev_sym = sym;
- }
- }
-
- eof = GetNextValidEvent(trace, &event, &ignored, &sym);
- if (ignored.time != 0 && prev_sym) {
- // We read an event that we are ignoring.
- // If we haven't already recorded a procedure exit, then do so.
- gtrace->AddProcExit(prev_sym->filenum, prev_sym->procnum, ignored.time,
- ignored.pid);
- prev_sym = NULL;
- }
- }
-
- delete gtrace;
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/read_addr.cpp b/emulator/qtools/read_addr.cpp
deleted file mode 100644
index 1c8c20f..0000000
--- a/emulator/qtools/read_addr.cpp
+++ /dev/null
@@ -1,29 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-
-int main(int argc, char **argv) {
- if (argc != 2) {
- fprintf(stderr, "Usage: %s trace_file\n", argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[1];
- TraceReaderBase *trace = new TraceReaderBase;
- trace->Open(trace_filename);
-
- while (1) {
- uint64_t time;
- uint32_t addr;
- int flags;
-
- if (trace->ReadAddr(&time, &addr, &flags))
- break;
- const char *op = "ld";
- if (flags == 1)
- op = "st";
- printf("%lld 0x%08x %s\n", time, addr, op);
- }
- return 0;
-}
diff --git a/emulator/qtools/read_elf.cpp b/emulator/qtools/read_elf.cpp
deleted file mode 100644
index 1d1a74a..0000000
--- a/emulator/qtools/read_elf.cpp
+++ /dev/null
@@ -1,210 +0,0 @@
-/*************************************************************************
- Copyright (C) 2002,2003,2004,2005 Wei Qin
- See file COPYING for more information.
-
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 2 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-*************************************************************************/
-
-#include <stdlib.h>
-#include <string.h>
-#include <errno.h>
-#include <assert.h>
-#include "read_elf.h"
-
-#define SwapHalf(a) (((a & 0x00ff) << 8) | ((a & 0xff00) >> 8))
-#define SwapWord(a) (((a & 0xff000000) >> 24) | ((a & 0x00ff0000) >> 8) | ((a & 0x0000ff00) << 8) | ((a & 0x000000ff) << 24))
-#define SwapAddr(a) SwapWord(a)
-#define SwapOff(a) SwapWord(a)
-#define SwapSection(a) SwapHalf(a)
-
-int LittleEndian()
-{
- Elf32_Word a = 0x01020304;
- return *(char *) &a == 0x04;
-}
-
-void SwapElfHeader(Elf32_Ehdr *hdr)
-{
- hdr->e_type = SwapHalf(hdr->e_type);
- hdr->e_machine = SwapHalf(hdr->e_machine);
- hdr->e_version = SwapWord(hdr->e_version);
- hdr->e_entry = SwapAddr(hdr->e_entry);
- hdr->e_phoff = SwapOff(hdr->e_phoff);
- hdr->e_shoff = SwapOff(hdr->e_shoff);
- hdr->e_flags = SwapWord(hdr->e_flags);
- hdr->e_ehsize = SwapHalf(hdr->e_ehsize);
- hdr->e_phentsize = SwapHalf(hdr->e_phentsize);
- hdr->e_phnum = SwapHalf(hdr->e_phnum);
- hdr->e_shentsize = SwapHalf(hdr->e_shentsize);
- hdr->e_shnum = SwapHalf(hdr->e_shnum);
- hdr->e_shstrndx = SwapHalf(hdr->e_shstrndx);
-}
-
-void SwapSectionHeader(Elf32_Shdr *shdr)
-{
- shdr->sh_name = SwapWord(shdr->sh_name);
- shdr->sh_type = SwapWord(shdr->sh_type);
- shdr->sh_flags = SwapWord(shdr->sh_flags);
- shdr->sh_addr = SwapAddr(shdr->sh_addr);
- shdr->sh_offset = SwapOff(shdr->sh_offset);
- shdr->sh_size = SwapWord(shdr->sh_size);
- shdr->sh_link = SwapWord(shdr->sh_link);
- shdr->sh_info = SwapWord(shdr->sh_info);
- shdr->sh_addralign = SwapWord(shdr->sh_addralign);
- shdr->sh_entsize = SwapWord(shdr->sh_entsize);
-}
-
-void SwapElfSymbol(Elf32_Sym *sym)
-{
- sym->st_name = SwapWord(sym->st_name);
- sym->st_value = SwapAddr(sym->st_value);
- sym->st_size = SwapWord(sym->st_size);
- sym->st_shndx = SwapSection(sym->st_shndx);
-}
-
-void AdjustElfHeader(Elf32_Ehdr *hdr)
-{
- switch(hdr->e_ident[EI_DATA])
- {
- case ELFDATA2LSB:
- if (!LittleEndian())
- SwapElfHeader(hdr);
- break;
- case ELFDATA2MSB:
- if (LittleEndian())
- SwapElfHeader(hdr);
- break;
- }
-}
-
-void AdjustSectionHeader(Elf32_Ehdr *hdr, Elf32_Shdr *shdr)
-{
- switch(hdr->e_ident[EI_DATA])
- {
- case ELFDATA2LSB:
- if (!LittleEndian())
- SwapSectionHeader(shdr);
- break;
- case ELFDATA2MSB:
- if (LittleEndian())
- SwapSectionHeader(shdr);
- break;
- }
-}
-
-void AdjustElfSymbols(Elf32_Ehdr *hdr, Elf32_Sym *elf_symbols, int num_entries)
-{
- if (hdr->e_ident[EI_DATA] == ELFDATA2LSB && LittleEndian())
- return;
- if (hdr->e_ident[EI_DATA] == ELFDATA2MSB && !LittleEndian())
- return;
- for (int ii = 0; ii < num_entries; ++ii) {
- SwapElfSymbol(&elf_symbols[ii]);
- }
-}
-
-Elf32_Ehdr *ReadElfHeader(FILE *fobj)
-{
- Elf32_Ehdr *hdr = new Elf32_Ehdr;
- int rval = fread(hdr, sizeof(Elf32_Ehdr), 1, fobj);
- if (rval != 1) {
- delete hdr;
- return NULL;
- }
- if (hdr->e_ident[EI_MAG0] != 0x7f || hdr->e_ident[EI_MAG1] != 'E' ||
- hdr->e_ident[EI_MAG2] != 'L' || hdr->e_ident[EI_MAG3] != 'F') {
- delete hdr;
- return NULL;
- }
- AdjustElfHeader(hdr);
- return hdr;
-}
-
-Elf32_Shdr *ReadSectionHeaders(Elf32_Ehdr *hdr, FILE *f)
-{
- int i;
- unsigned long sz = hdr->e_shnum * hdr->e_shentsize;
- assert(sizeof(Elf32_Shdr) == hdr->e_shentsize);
- Elf32_Shdr *shdr = new Elf32_Shdr[hdr->e_shnum];
-
- if (fseek(f, hdr->e_shoff, SEEK_SET) != 0)
- {
- delete[] shdr;
- return NULL;
- }
- if (fread(shdr, sz, 1, f) != 1)
- {
- delete[] shdr;
- return NULL;
- }
-
- for(i = 0; i < hdr->e_shnum; i++)
- AdjustSectionHeader(hdr, shdr + i);
-
- return shdr;
-}
-
-
-char *ReadStringTable(Elf32_Ehdr *hdr, Elf32_Shdr *shdr_table, FILE *f)
-{
- Elf32_Shdr *shdr = shdr_table + hdr->e_shstrndx;
- char *string_table;
-
- string_table = new char[shdr->sh_size];
- fseek(f, shdr->sh_offset, SEEK_SET);
- fread(string_table, shdr->sh_size, 1, f);
-
- return string_table;
-}
-
-int ReadSection(Elf32_Shdr *shdr, void *buffer, FILE *f)
-{
- if (fseek(f, shdr->sh_offset, SEEK_SET) != 0)
- return -1;
- if (fread(buffer, shdr->sh_size, 1, f) != 1)
- return -1;
- return 0;
-}
-
-char *GetSymbolName(Elf32_Half index, char *string_table)
-{
- return string_table + index;
-}
-
-Elf32_Shdr *FindSymbolTableSection(Elf32_Ehdr *hdr,
- Elf32_Shdr *shdr,
- char *string_table)
-{
- for(int ii = 0; ii < hdr->e_shnum; ii++) {
- if (shdr[ii].sh_type == SHT_SYMTAB &&
- strcmp(GetSymbolName(shdr[ii].sh_name, string_table),
- ".symtab") == 0)
- {
- return &shdr[ii];
- }
- }
- return NULL;
-}
-
-Elf32_Shdr *FindSymbolStringTableSection(Elf32_Ehdr *hdr,
- Elf32_Shdr *shdr,
- char *string_table)
-{
- for(int ii = 0; ii < hdr->e_shnum; ii++) {
- if (shdr[ii].sh_type == SHT_STRTAB &&
- strcmp(GetSymbolName(shdr[ii].sh_name, string_table),
- ".strtab") == 0)
- {
- return &shdr[ii];
- }
- }
- return NULL;
-}
diff --git a/emulator/qtools/read_elf.h b/emulator/qtools/read_elf.h
deleted file mode 100644
index 72266c3..0000000
--- a/emulator/qtools/read_elf.h
+++ /dev/null
@@ -1,20 +0,0 @@
-#ifndef READ_ELF_H
-#define READ_ELF_H
-
-#include <stdio.h>
-#include <elf.h>
-
-Elf32_Ehdr *ReadElfHeader(FILE *fobj);
-Elf32_Shdr *ReadSectionHeaders(Elf32_Ehdr *hdr, FILE *fobj);
-char *ReadStringTable(Elf32_Ehdr *hdr, Elf32_Shdr *shdr, FILE *fobj);
-Elf32_Shdr *FindSymbolTableSection(Elf32_Ehdr *hdr,
- Elf32_Shdr *shdr,
- char *string_table);
-Elf32_Shdr *FindSymbolStringTableSection(Elf32_Ehdr *hdr,
- Elf32_Shdr *shdr,
- char *string_table);
-int ReadSection(Elf32_Shdr *shdr, void *buffer, FILE *f);
-void AdjustElfSymbols(Elf32_Ehdr *hdr, Elf32_Sym *elf_symbols,
- int num_entries);
-
-#endif /* READ_ELF_H */
diff --git a/emulator/qtools/read_method.cpp b/emulator/qtools/read_method.cpp
deleted file mode 100644
index 6aa0c1a..0000000
--- a/emulator/qtools/read_method.cpp
+++ /dev/null
@@ -1,137 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-struct frame {
- uint64_t time;
- uint32_t addr;
- const char *name;
- bool isNative;
-
- frame(uint64_t time, uint32_t addr, const char *name, bool isNative) {
- this->time = time;
- this->addr = addr;
- this->name = name;
- this->isNative = isNative;
- }
-};
-
-class Stack {
- static const int kMaxFrames = 1000;
- int top;
- frame *frames[kMaxFrames];
-
-public:
- Stack() {
- top = 0;
- }
-
- void push(frame *pframe);
- frame* pop();
- void dump();
-};
-
-void Stack::push(frame *pframe) {
- if (top == kMaxFrames) {
- fprintf(stderr, "Error: stack overflow\n");
- exit(1);
- }
- frames[top] = pframe;
- top += 1;
-}
-
-frame *Stack::pop() {
- if (top <= 0)
- return NULL;
- top -= 1;
- return frames[top];
-}
-
-void Stack::dump() {
- frame *pframe;
-
- for (int ii = 0; ii < top; ii++) {
- pframe = frames[ii];
- const char *native = pframe->isNative ? "n" : " ";
- printf(" %s %d: %llu 0x%x %s\n",
- native, ii, pframe->time, pframe->addr,
- pframe->name == NULL ? "" : pframe->name);
- }
-}
-
-static const int kMaxThreads = (32 * 1024);
-Stack *stacks[kMaxThreads];
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] trace_name elf_file\n",
- program);
- OptionsUsage();
-}
-
-int main(int argc, char **argv) {
- ParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *qemu_trace_file = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReaderType *trace = new TraceReaderType;
- trace->Open(qemu_trace_file);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- while (1) {
- MethodRec method_record;
- symbol_type *sym;
- TraceReaderType::ProcessState *proc;
- frame *pframe;
-
- if (trace->ReadMethodSymbol(&method_record, &sym, &proc))
- break;
-
- if (!IsValidPid(proc->pid))
- continue;
-
- if (sym != NULL) {
- printf("%lld p %d 0x%x %d %s\n",
- method_record.time, proc->pid, method_record.addr,
- method_record.flags, sym->name);
- } else {
- printf("%lld p %d 0x%x %d\n",
- method_record.time, proc->pid, method_record.addr,
- method_record.flags);
- }
-
- // Get the stack for the current thread
- Stack *pStack = stacks[proc->pid];
-
- // If the stack does not exist, then allocate a new one.
- if (pStack == NULL) {
- pStack = new Stack();
- stacks[proc->pid] = pStack;
- }
-
- int flags = method_record.flags;
- if (flags == kMethodEnter || flags == kNativeEnter) {
- pframe = new frame(method_record.time, method_record.addr,
- sym == NULL ? NULL: sym->name,
- method_record.flags == kNativeEnter);
- pStack->push(pframe);
- } else {
- pframe = pStack->pop();
- delete pframe;
- }
- pStack->dump();
- }
- return 0;
-}
diff --git a/emulator/qtools/read_pid.cpp b/emulator/qtools/read_pid.cpp
deleted file mode 100644
index a2d69d4..0000000
--- a/emulator/qtools/read_pid.cpp
+++ /dev/null
@@ -1,71 +0,0 @@
-#include <stdio.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include "trace_reader.h"
-
-int main(int argc, char **argv) {
- if (argc != 2) {
- fprintf(stderr, "Usage: %s trace_file\n", argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[1];
- TraceReaderBase *trace = new TraceReaderBase;
- trace->Open(trace_filename);
-
- while (1) {
- PidEvent event;
- if (trace->ReadPidEvent(&event))
- break;
- switch (event.rec_type) {
- case kPidFork:
- printf("t%lld fork tgid %d pid %d\n", event.time, event.tgid, event.pid);
- break;
- case kPidClone:
- printf("t%lld clone tgid %d pid %d\n", event.time, event.tgid, event.pid);
- break;
- case kPidSwitch:
- printf("t%lld switch %d\n", event.time, event.pid);
- break;
- case kPidExit:
- printf("t%lld exit %d\n", event.time, event.pid);
- break;
- case kPidMmap:
- printf("t%lld mmap %08x - %08x, offset %08x '%s'\n",
- event.time, event.vstart, event.vend, event.offset, event.path);
- delete[] event.path;
- break;
- case kPidMunmap:
- printf("t%lld munmap %08x - %08x\n",
- event.time, event.vstart, event.vend);
- break;
- case kPidSymbolAdd:
- printf("t%lld add sym %08x '%s'\n",
- event.time, event.vstart, event.path);
- delete[] event.path;
- break;
- case kPidSymbolRemove:
- printf("t%lld remove %08x\n", event.time, event.vstart);
- break;
- case kPidExec:
- printf("t%lld argc: %d\n", event.time, event.argc);
- for (int ii = 0; ii < event.argc; ++ii) {
- printf(" argv[%d]: %s\n", ii, event.argv[ii]);
- delete[] event.argv[ii];
- }
- delete[] event.argv;
- break;
- case kPidKthreadName:
- printf("t%lld kthread tgid %d pid %d %s\n",
- event.time, event.tgid, event.pid, event.path);
- delete[] event.path;
- break;
- case kPidName:
- printf("t%lld name %d %s\n",
- event.time, event.pid, event.path);
- delete[] event.path;
- break;
- }
- }
- return 0;
-}
diff --git a/emulator/qtools/read_trace.cpp b/emulator/qtools/read_trace.cpp
deleted file mode 100644
index fb4917c..0000000
--- a/emulator/qtools/read_trace.cpp
+++ /dev/null
@@ -1,165 +0,0 @@
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "armdis.h"
-#include "parse_options.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-
-static const uint32_t kOffsetThreshold = 0x100000;
-static uint64_t startTime = 0;
-
-void Usage(const char *program)
-{
- fprintf(stderr,
- "Usage: %s [options] [-- -s start_time] trace_file elf_file\n",
- program);
- OptionsUsage();
-}
-
-
-bool localParseOptions(int argc, char **argv)
-{
- bool err = false;
- while (!err) {
- int opt = getopt(argc, argv, "+s:");
- if (opt == -1)
- break;
- switch (opt) {
- case 's':
- startTime = strtoull(optarg, NULL, 0);
- break;
- default:
- err = true;
- break;
- }
- }
- return err;
-}
-
-int main(int argc, char **argv) {
- // Parse the options
- ParseOptions(argc, argv);
- localParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *trace_filename = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReader<> *trace = new TraceReader<>;
- trace->Open(trace_filename);
- trace->SetDemangle(demangle);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- while (1) {
- symbol_type *sym;
- char buf[1024];
- BBEvent event;
- BBEvent ignored;
-
- if (GetNextValidEvent(trace, &event, &ignored, &sym))
- break;
-#if 0
- fprintf(stderr, "t%llu bb %lld %d\n",
- event.time, event.bb_num, event.num_insns);
-#endif
-
- uint32_t *insns = event.insns;
- uint32_t addr = event.bb_addr;
- uint32_t offset = addr - sym->addr - sym->region->base_addr;
- symbol_type *vm_sym = sym->vm_sym;
- const char *vm_name = NULL;
- if (vm_sym != NULL) {
- vm_name = vm_sym->name;
- offset = addr - vm_sym->addr - vm_sym->region->base_addr;
- }
-#if 0
- if (strcmp(sym->name, "(unknown)") == 0 || offset > kOffsetThreshold) {
- ProcessState *process = trace->GetCurrentProcess();
- ProcessState *manager = process->addr_manager;
- for (int ii = 0; ii < manager->nregions; ++ii) {
- printf(" %2d: %08x - %08x base: %08x offset: %u nsyms: %4d flags: 0x%x %s\n",
- ii,
- manager->regions[ii]->vstart,
- manager->regions[ii]->vend,
- manager->regions[ii]->base_addr,
- manager->regions[ii]->file_offset,
- manager->regions[ii]->nsymbols,
- manager->regions[ii]->flags,
- manager->regions[ii]->path);
- int nsymbols = manager->regions[ii]->nsymbols;
- for (int jj = 0; jj < 10 && jj < nsymbols; ++jj) {
- printf(" %08x %s\n",
- manager->regions[ii]->symbols[jj].addr,
- manager->regions[ii]->symbols[jj].name);
- }
- }
- }
-#endif
-#if 1
- for (int ii = 0; ii < event.num_insns; ++ii) {
- uint64_t sim_time = trace->ReadInsnTime(event.time);
- if (sim_time < startTime)
- continue;
-
- uint32_t insn = insns[ii];
- char *disasm;
- int bytes;
- if (vm_name != NULL) {
- sprintf(buf, "%s+%02x: %s", vm_name, offset, sym->name);
- } else {
- sprintf(buf, "%s+%02x", sym->name, offset);
- }
-
- if (insn_is_thumb(insn)) {
- bytes = 2;
- insn = insn_unwrap_thumb(insn);
-
- // thumb_pair is true if this is the first of a pair of
- // thumb instructions (BL or BLX).
- bool thumb_pair = ((insn & 0xf800) == 0xf000);
-
- // Get the next thumb instruction (if any) because we may need
- // it for the case where insn is BL or BLX.
- uint32_t insn2 = 0;
- if (thumb_pair && (ii + 1 < event.num_insns)) {
- insn2 = insns[ii + 1];
- insn2 = insn_unwrap_thumb(insn2);
- bytes = 4;
- ii += 1;
- }
- disasm = disasm_insn_thumb(addr, insn, insn2, NULL);
- if (thumb_pair) {
- printf("%llu p%-4d %08x %04x %04x %-30s %s\n",
- sim_time, event.pid, addr, insn, insn2, buf, disasm);
- } else {
- printf("%llu p%-4d %08x %04x %-30s %s\n",
- sim_time, event.pid, addr, insn, buf, disasm);
- }
- } else {
- bytes = 4;
- disasm = Arm::disasm(addr, insn, NULL);
- printf("%llu p%-4d %08x %08x %-30s %s\n",
- sim_time, event.pid, addr, insn, buf, disasm);
- }
- //printf("t%llu \t%08x\n", sim_time, addr);
- addr += bytes;
- offset += bytes;
- }
-#endif
-#if 0
- assert(offset < kOffsetThreshold);
-#endif
- }
-
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/stack_dump.cpp b/emulator/qtools/stack_dump.cpp
deleted file mode 100644
index f685cd0..0000000
--- a/emulator/qtools/stack_dump.cpp
+++ /dev/null
@@ -1,156 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <unistd.h>
-#include <stdlib.h>
-#include <inttypes.h>
-#include <assert.h>
-#include "trace_reader.h"
-#include "bitvector.h"
-#include "parse_options.h"
-#include "armdis.h"
-
-typedef TraceReader<> TraceReaderType;
-
-#include "parse_options-inl.h"
-#include "callstack.h"
-
-static uint64_t debugTime;
-static uint64_t dumpTime = 0;
-
-class MyFrame : public StackFrame<symbol_type> {
- public:
- void push(int stackLevel, uint64_t time, CallStackBase *base);
- void pop(int stackLevel, uint64_t time, CallStackBase *base);
- void getFrameType(char *type);
-};
-
-typedef CallStack<MyFrame> CallStackType;
-
-void MyFrame::getFrameType(char *type)
-{
- strcpy(type, "----");
- if (flags & kCausedException)
- type[0] = 'e';
- if (flags & kInterpreted)
- type[1] = 'm';
- if (function->region->flags & region_type::kIsKernelRegion)
- type[2] = 'k';
- if (function->flags & symbol_type::kIsVectorTable)
- type[3] = 'v';
-}
-
-void MyFrame::push(int stackLevel, uint64_t time, CallStackBase *base)
-{
- char type[5];
-
- if (dumpTime > 0)
- return;
-
- getFrameType(type);
- printf("%llu en thr %d %s %3d", time, base->getId(), type, stackLevel);
- for (int ii = 0; ii < stackLevel; ++ii)
- printf(".");
- printf(" 0x%08x %s\n", addr, function->name);
-}
-
-void MyFrame::pop(int stackLevel, uint64_t time, CallStackBase *base)
-{
- char type[5];
-
- if (dumpTime > 0)
- return;
-
- getFrameType(type);
- printf("%llu x thr %d %s %3d", time, base->getId(), type, stackLevel);
- for (int ii = 0; ii < stackLevel; ++ii)
- printf(".");
- printf(" 0x%08x %s\n", addr, function->name);
-}
-
-static const int kNumStackFrames = 500;
-static const int kMaxThreads = (32 * 1024);
-CallStackType *stacks[kMaxThreads];
-
-void Usage(const char *program)
-{
- fprintf(stderr, "Usage: %s [options] [-- -d dumpTime] trace_name elf_file\n",
- program);
- OptionsUsage();
-}
-
-bool localParseOptions(int argc, char **argv)
-{
- bool err = false;
- while (!err) {
- int opt = getopt(argc, argv, "+d:");
- if (opt == -1)
- break;
- switch (opt) {
- case 'd':
- dumpTime = strtoull(optarg, NULL, 0);
- break;
- default:
- err = true;
- break;
- }
- }
- return err;
-}
-
-int main(int argc, char **argv)
-{
- ParseOptions(argc, argv);
- localParseOptions(argc, argv);
- if (argc - optind != 2) {
- Usage(argv[0]);
- exit(1);
- }
-
- char *qemu_trace_file = argv[optind++];
- char *elf_file = argv[optind++];
- TraceReaderType *trace = new TraceReaderType;
- trace->Open(qemu_trace_file);
- trace->ReadKernelSymbols(elf_file);
- trace->SetRoot(root);
-
- BBEvent event;
- while (1) {
- BBEvent ignored;
- symbol_type *function;
-
- if (GetNextValidEvent(trace, &event, &ignored, &function))
- break;
- if (event.bb_num == 0)
- break;
-
- // Get the stack for the current thread
- CallStackType *pStack = stacks[event.pid];
-
- // If the stack does not exist, then allocate a new one.
- if (pStack == NULL) {
- pStack = new CallStackType(event.pid, kNumStackFrames, trace);
- stacks[event.pid] = pStack;
- }
- if (debugTime != 0 && event.time >= debugTime)
- printf("debug time: %lld\n", debugTime);
-
- // Update the stack
- pStack->updateStack(&event, function);
-
- // If the user requested a stack dump at a certain time,
- // and we are at that time, then dump the stack and exit.
- if (dumpTime > 0 && event.time >= dumpTime) {
- pStack->showStack(stdout);
- break;
- }
- }
-
- for (int ii = 0; ii < kMaxThreads; ++ii) {
- if (stacks[ii])
- stacks[ii]->popAll(event.time);
- }
-
- delete trace;
- return 0;
-}
diff --git a/emulator/qtools/tests/common_head.mk b/emulator/qtools/tests/common_head.mk
deleted file mode 100644
index e8170e9..0000000
--- a/emulator/qtools/tests/common_head.mk
+++ /dev/null
@@ -1,25 +0,0 @@
-CC := arm-elf-gcc
-LD := arm-elf-ld
-AS := arm-elf-as
-OBJCOPY := arm-elf-objcopy
-OBJDUMP := arm-elf-objdump
-
-OPT := -g
-CFLAGS := $(OPT) -mcpu=arm9
-
-.SUFFIXES: .dis .bin .elf
-
-.c.elf:
- $(CC) $(CFLAGS) -Xlinker --script ../tests.ld -o $@ $< -nostdlib
-
-.c.s:
- $(CC) $(CFLAGS) -static -S $<
-
-.S.elf:
- $(CC) $(CFLAGS) -Xlinker --script ../tests.ld -nostdlib -o $@ $<
-
-.elf.dis:
- $(OBJDUMP) -adx $< > $@
-
-.elf.bin:
- $(OBJCOPY) -O binary $< $@
diff --git a/emulator/qtools/tests/common_tail.mk b/emulator/qtools/tests/common_tail.mk
deleted file mode 100644
index be1c141..0000000
--- a/emulator/qtools/tests/common_tail.mk
+++ /dev/null
@@ -1,3 +0,0 @@
-
-clean:
- rm -f *.elf *.dis *.bin *.o *~ a.out
diff --git a/emulator/qtools/tests/gtrace/Makefile b/emulator/qtools/tests/gtrace/Makefile
deleted file mode 100644
index 1d2050c..0000000
--- a/emulator/qtools/tests/gtrace/Makefile
+++ /dev/null
@@ -1,18 +0,0 @@
-include ../common_head.mk
-
-P4ROOT=/work/android
-QEMU=$(P4ROOT)/device/tools/qemu/arm-softmmu/qemu-system-arm
-QTOOLS=$(P4ROOT)/device/tools/qtools
-
-all: test.elf test.bin test.dis
-
-trace: test.elf test.bin
- $(QEMU) -QEMU -kernel test.bin -trace foo
- $(QTOOLS)/post_trace foo
- $(QTOOLS)/read_trace foo test.elf > t1
-
-gtrace: trace
- $(QTOOLS)/q2g -r $(P4ROOT)/device/out/linux-arm-release/symbols foo test.elf foo.gtrace
- gtracepost64 foo.gtrace > t2
-
-include ../common_tail.mk
diff --git a/emulator/qtools/tests/gtrace/test.c b/emulator/qtools/tests/gtrace/test.c
deleted file mode 100644
index e56a0f1..0000000
--- a/emulator/qtools/tests/gtrace/test.c
+++ /dev/null
@@ -1,201 +0,0 @@
-#include "../macros.h"
-
-int foo1();
-int foo2();
-void bar();
-int child1();
-int child2();
-int child3();
-int child4();
-int child5();
-
-int global;
-
-void start()
-{
- // Set the stack pointer
- asm(" mov r13,#0x200000");
- PRINT_STR("hello\n");
- TRACE_INIT_NAME(701, "proc_foo");
- TRACE_INIT_NAME(702, "proc_bar");
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
- TRACE_SWITCH(702);
- if (global++ > 0)
- global++;
- bar();
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo2();
- TRACE_SWITCH(703);
- if (global++ > 0)
- global++;
- foo1();
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(704);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(705);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(706);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(707);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(708);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(709);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(701);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_SWITCH(710);
- if (global++ > 0)
- global++;
- foo1();
-
- TRACE_STOP_EMU();
-}
-
-int foo1()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 3; ++ii) {
- a += child1();
- a += child2();
- a += child3();
- }
- return a;
-}
-
-int foo2()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 2; ++ii) {
- a += child3();
- a += child4();
- a += child5();
- }
- return a;
-}
-
-#define kStride 64
-void bar()
-{
- int a = 0;
-
- static char mem[1000 * kStride];
-
- int ii, jj;
-
- for (ii = 0; ii < 4; ++ii) {
- for (jj = 0; jj < 10; ++jj)
- a += mem[jj * kStride];
- foo1();
- foo2();
- }
-}
-
-int child1()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 2; ++ii)
- a += ii;
- return a;
-}
-
-int child2()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 4; ++ii)
- a += ii;
- return a;
-}
-
-int child3()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 6; ++ii)
- a += ii;
- return a;
-}
-
-int child4()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 8; ++ii)
- a += ii;
- return a;
-}
-
-int child5()
-{
- int a = 0;
-
- int ii;
- for (ii = 0; ii < 10; ++ii)
- a += ii;
- return a;
-}
diff --git a/emulator/qtools/tests/macros.h b/emulator/qtools/tests/macros.h
deleted file mode 100644
index 066374b..0000000
--- a/emulator/qtools/tests/macros.h
+++ /dev/null
@@ -1,93 +0,0 @@
-#ifndef _TEST_TRACE_C_H_
-#define _TEST_TRACE_C_H_
-
-/* the base address of trace device */
-#define TRACE_DEV_BASE_ADDR 0x21000000
-
-/*the register addresses of the trace device */
-#define TRACE_DEV_REG_SWITCH 0
-#define TRACE_DEV_REG_FORK 1
-#define TRACE_DEV_REG_EXECVE_PID 2
-#define TRACE_DEV_REG_EXECVE_VMSTART 3
-#define TRACE_DEV_REG_EXECVE_VMEND 4
-#define TRACE_DEV_REG_EXECVE_OFFSET 5
-#define TRACE_DEV_REG_EXECVE_EXEPATH 6
-#define TRACE_DEV_REG_EXIT 7
-#define TRACE_DEV_REG_CMDLINE 8
-#define TRACE_DEV_REG_CMDLINE_LEN 9
-#define TRACE_DEV_REG_MMAP_EXEPATH 10
-#define TRACE_DEV_REG_INIT_PID 11
-#define TRACE_DEV_REG_INIT_NAME 12
-#define TRACE_DEV_REG_CLONE 13
-#define TRACE_DEV_REG_DYN_SYM 50
-#define TRACE_DEV_REG_DYN_SYM_ADDR 51
-#define TRACE_DEV_REG_PRINT_STR 60
-#define TRACE_DEV_REG_PRINT_NUM_DEC 61
-#define TRACE_DEV_REG_PRINT_NUM_HEX 62
-#define TRACE_DEV_REG_STOP_EMU 90
-#define TRACE_DEV_REG_ENABLE 100
-#define TRACE_DEV_REG_DISABLE 101
-
-/* write a word to a trace device register */
-#define DEV_WRITE_WORD(addr,value)\
- (*(volatile unsigned long *)(TRACE_DEV_BASE_ADDR + ((addr) << 2)) = (value))
-
-/*************************************************************/
-/* generates test events */
-
-/* context switch */
-#define TRACE_SWITCH(pid) DEV_WRITE_WORD(TRACE_DEV_REG_SWITCH, (pid))
-/* fork */
-#define TRACE_FORK(pid) DEV_WRITE_WORD(TRACE_DEV_REG_FORK, (pid))
-/* clone */
-#define TRACE_CLONE(pid) DEV_WRITE_WORD(TRACE_DEV_REG_CLONE, (pid))
-/* dump name and path of threads executed before trace device created */
-#define TRACE_INIT_NAME(pid,path)\
-do {\
- DEV_WRITE_WORD(TRACE_DEV_REG_INIT_PID, (pid));\
- DEV_WRITE_WORD(TRACE_DEV_REG_INIT_NAME, (unsigned long)(path));\
-}while(0)
-/* dump exec mapping of threads executed before trace device created */
-#define TRACE_INIT_EXEC(vstart,vend,eoff,path)\
-do {\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_VMSTART, (vstart));\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_VMEND, (vend));\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_OFFSET, (eoff));\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_EXEPATH, (unsigned long)(path));\
-}while(0)
-/* mmap */
-#define TRACE_MMAP(vstart,vend,eoff,path)\
-do {\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_VMSTART, (vstart));\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_VMEND, (vend));\
- DEV_WRITE_WORD(TRACE_DEV_REG_EXECVE_OFFSET, (eoff));\
- DEV_WRITE_WORD(TRACE_DEV_REG_MMAP_EXEPATH, (unsigned long)(path));\
-}while(0)
-/* execve */
-#define TRACE_EXECVE(cmdlen,cmd)\
-do {\
- DEV_WRITE_WORD(TRACE_DEV_REG_CMDLINE_LEN, (cmdlen));\
- DEV_WRITE_WORD(TRACE_DEV_REG_CMDLINE, (unsigned long)(cmd));\
-}while(0)
-/* exit */
-#define TRACE_EXIT(retv) DEV_WRITE_WORD(TRACE_DEV_REG_EXIT, (retv))
-
-/* other commands */
-
-/* stop emulation */
-#define TRACE_STOP_EMU() DEV_WRITE_WORD(TRACE_DEV_REG_STOP_EMU, 1)
-/* enable/disable tracing */
-#define TRACE_ENABLE_TRACING() DEV_WRITE_WORD(TRACE_DEV_REG_ENABLE, 1)
-#define TRACE_DISABLE_TRACING() DEV_WRITE_WORD(TRACE_DEV_REG_DISABLE, 1)
-/* dynamic symbols */
-#define TRACE_DYN_SYM(addr,sym)\
-do {\
- DEV_WRITE_WORD(TRACE_DEV_REG_DYN_SYM_ADDR, (addr));\
- DEV_WRITE_WORD(TRACE_DEV_REG_DYN_SYM, (unsigned long)(sym));\
-}while(0)
-/* prints */
-#define PRINT_STR(str) DEV_WRITE_WORD(TRACE_DEV_REG_PRINT_STR, (unsigned long)(str))
-#define PRINT_NUM_DEC(num) DEV_WRITE_WORD(TRACE_DEV_REG_PRINT_NUM_DEC, (num))
-#define PRINT_NUM_HEX(num) DEV_WRITE_WORD(TRACE_DEV_REG_PRINT_NUM_HEX, (num))
-
-#endif
diff --git a/emulator/qtools/tests/tests.ld b/emulator/qtools/tests/tests.ld
deleted file mode 100644
index 05fe41b..0000000
--- a/emulator/qtools/tests/tests.ld
+++ /dev/null
@@ -1,10 +0,0 @@
-SECTIONS {
- TEXT_START = 0x00010000 ;
- DATA_START = 0x00200000 ;
- handlers 0x0 : { *(handlers) }
- .text TEXT_START : { *(.text) }
- .data DATA_START : { *(.data) }
- .bss : { *(.bss) *(COMMON) }
- p00300000 0x00300000 : { *(p00300000) }
- p00400000 0x00400000 : { *(p00400000) }
-}
diff --git a/emulator/qtools/thumbdis.cpp b/emulator/qtools/thumbdis.cpp
deleted file mode 100644
index 07c482f..0000000
--- a/emulator/qtools/thumbdis.cpp
+++ /dev/null
@@ -1,503 +0,0 @@
-/* Instruction printing code for the ARM
- Copyright 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002
- Free Software Foundation, Inc.
- Contributed by Richard Earnshaw (rwe@pegasus.esprit.ec.org)
- Modification by James G. Smith (jsmith@cygnus.co.uk)
-
-This file is part of libopcodes.
-
-This program is free software; you can redistribute it and/or modify it under
-the terms of the GNU General Public License as published by the Free
-Software Foundation; either version 2 of the License, or (at your option)
-any later version.
-
-This program is distributed in the hope that it will be useful, but WITHOUT
-ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
-FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
-more details.
-
-You should have received a copy of the GNU General Public License
-along with this program; if not, write to the Free Software
-Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
-
-/* Modified to fit into the qtools framework. The main differences are:
- *
- * - The disassembly function returns a string instead of writing it to a
- * file stream.
- *
- * - All the references to the struct "disassemble_info" have been removed.
- *
- * - A set of enums for the thumb opcodes have been defined, along with a
- * "decode()" function that maps a thumb instruction to an opcode enum.
- *
- * - Eliminated uses of the special characters ', `, and ? from the
- * thumb_opcodes[] table so that we can easily specify separate opcodes
- * for distinct instructions.
- */
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <inttypes.h>
-#include "opcode.h"
-
-
-struct thumb_opcode
-{
- unsigned short value, mask; /* recognise instruction if (op&mask)==value */
- Opcode opcode;
- const char * assembler; /* how to disassemble this instruction */
-};
-
-/* format of the assembler string :
-
- %% %
- %<bitfield>d print the bitfield in decimal
- %<bitfield>x print the bitfield in hex
- %<bitfield>X print the bitfield as 1 hex digit without leading "0x"
- %<bitfield>r print as an ARM register
- %<bitfield>f print a floating point constant if >7 else a
- floating point register
- %<code>y print a single precision VFP reg.
- Codes: 0=>Sm, 1=>Sd, 2=>Sn, 3=>multi-list, 4=>Sm pair
- %<code>z print a double precision VFP reg
- Codes: 0=>Dm, 1=>Dd, 2=>Dn, 3=>multi-list
- %c print condition code (always bits 28-31)
- %P print floating point precision in arithmetic insn
- %Q print floating point precision in ldf/stf insn
- %R print floating point rounding mode
- %<bitnum>'c print specified char iff bit is one
- %<bitnum>`c print specified char iff bit is zero
- %<bitnum>?ab print a if bit is one else print b
- %p print 'p' iff bits 12-15 are 15
- %t print 't' iff bit 21 set and bit 24 clear
- %o print operand2 (immediate or register + shift)
- %a print address for ldr/str instruction
- %s print address for ldr/str halfword/signextend instruction
- %b print branch destination
- %B print arm BLX(1) destination
- %A print address for ldc/stc/ldf/stf instruction
- %m print register mask for ldm/stm instruction
- %C print the PSR sub type.
- %F print the COUNT field of a LFM/SFM instruction.
-Thumb specific format options:
- %D print Thumb register (bits 0..2 as high number if bit 7 set)
- %S print Thumb register (bits 3..5 as high number if bit 6 set)
- %<bitfield>I print bitfield as a signed decimal
- (top bit of range being the sign bit)
- %M print Thumb register mask
- %N print Thumb register mask (with LR)
- %O print Thumb register mask (with PC)
- %T print Thumb condition code (always bits 8-11)
- %I print cirrus signed shift immediate: bits 0..3|4..6
- %<bitfield>B print Thumb branch destination (signed displacement)
- %<bitfield>W print (bitfield * 4) as a decimal
- %<bitfield>H print (bitfield * 2) as a decimal
- %<bitfield>a print (bitfield * 4) as a pc-rel offset + decoded symbol
-*/
-
-
-static struct thumb_opcode thumb_opcodes[] =
-{
- /* Thumb instructions. */
-
- /* ARM V5 ISA extends Thumb. */
- {0xbe00, 0xff00, OP_THUMB_BKPT, "bkpt\t%0-7x"},
- {0x4780, 0xff87, OP_THUMB_BLX, "blx\t%3-6r"}, /* note: 4 bit register number. */
- /* Format 5 instructions do not update the PSR. */
- {0x1C00, 0xFFC0, OP_THUMB_MOV, "mov\t%0-2r, %3-5r"},
- /* Format 4. */
- {0x4000, 0xFFC0, OP_THUMB_AND, "and\t%0-2r, %3-5r"},
- {0x4040, 0xFFC0, OP_THUMB_EOR, "eor\t%0-2r, %3-5r"},
- {0x4080, 0xFFC0, OP_THUMB_LSL, "lsl\t%0-2r, %3-5r"},
- {0x40C0, 0xFFC0, OP_THUMB_LSR, "lsr\t%0-2r, %3-5r"},
- {0x4100, 0xFFC0, OP_THUMB_ASR, "asr\t%0-2r, %3-5r"},
- {0x4140, 0xFFC0, OP_THUMB_ADC, "adc\t%0-2r, %3-5r"},
- {0x4180, 0xFFC0, OP_THUMB_SBC, "sbc\t%0-2r, %3-5r"},
- {0x41C0, 0xFFC0, OP_THUMB_ROR, "ror\t%0-2r, %3-5r"},
- {0x4200, 0xFFC0, OP_THUMB_TST, "tst\t%0-2r, %3-5r"},
- {0x4240, 0xFFC0, OP_THUMB_NEG, "neg\t%0-2r, %3-5r"},
- {0x4280, 0xFFC0, OP_THUMB_CMP, "cmp\t%0-2r, %3-5r"},
- {0x42C0, 0xFFC0, OP_THUMB_CMN, "cmn\t%0-2r, %3-5r"},
- {0x4300, 0xFFC0, OP_THUMB_ORR, "orr\t%0-2r, %3-5r"},
- {0x4340, 0xFFC0, OP_THUMB_MUL, "mul\t%0-2r, %3-5r"},
- {0x4380, 0xFFC0, OP_THUMB_BIC, "bic\t%0-2r, %3-5r"},
- {0x43C0, 0xFFC0, OP_THUMB_MVN, "mvn\t%0-2r, %3-5r"},
- /* format 13 */
- {0xB000, 0xFF80, OP_THUMB_ADD, "add\tsp, #%0-6W"},
- {0xB080, 0xFF80, OP_THUMB_SUB, "sub\tsp, #%0-6W"},
- /* format 5 */
- {0x4700, 0xFF80, OP_THUMB_BX, "bx\t%S"},
- {0x4400, 0xFF00, OP_THUMB_ADD, "add\t%D, %S"},
- {0x4500, 0xFF00, OP_THUMB_CMP, "cmp\t%D, %S"},
- {0x4600, 0xFF00, OP_THUMB_MOV, "mov\t%D, %S"},
- /* format 14 */
- {0xB400, 0xFE00, OP_THUMB_PUSH, "push\t%N"},
- {0xBC00, 0xFE00, OP_THUMB_POP, "pop\t%O"},
- /* format 2 */
- {0x1800, 0xFE00, OP_THUMB_ADD, "add\t%0-2r, %3-5r, %6-8r"},
- {0x1A00, 0xFE00, OP_THUMB_SUB, "sub\t%0-2r, %3-5r, %6-8r"},
- {0x1C00, 0xFE00, OP_THUMB_ADD, "add\t%0-2r, %3-5r, #%6-8d"},
- {0x1E00, 0xFE00, OP_THUMB_SUB, "sub\t%0-2r, %3-5r, #%6-8d"},
- /* format 8 */
- {0x5200, 0xFE00, OP_THUMB_STRH, "strh\t%0-2r, [%3-5r, %6-8r]"},
- {0x5A00, 0xFE00, OP_THUMB_LDRH, "ldrh\t%0-2r, [%3-5r, %6-8r]"},
- {0x5600, 0xFE00, OP_THUMB_LDRSB, "ldrsb\t%0-2r, [%3-5r, %6-8r]"},
- {0x5E00, 0xFE00, OP_THUMB_LDRSH, "ldrsh\t%0-2r, [%3-5r, %6-8r]"},
- /* format 7 */
- {0x5000, 0xFE00, OP_THUMB_STR, "str\t%0-2r, [%3-5r, %6-8r]"},
- {0x5400, 0xFE00, OP_THUMB_STRB, "strb\t%0-2r, [%3-5r, %6-8r]"},
- {0x5800, 0xFE00, OP_THUMB_LDR, "ldr\t%0-2r, [%3-5r, %6-8r]"},
- {0x5C00, 0xFE00, OP_THUMB_LDRB, "ldrb\t%0-2r, [%3-5r, %6-8r]"},
- /* format 1 */
- {0x0000, 0xF800, OP_THUMB_LSL, "lsl\t%0-2r, %3-5r, #%6-10d"},
- {0x0800, 0xF800, OP_THUMB_LSR, "lsr\t%0-2r, %3-5r, #%6-10d"},
- {0x1000, 0xF800, OP_THUMB_ASR, "asr\t%0-2r, %3-5r, #%6-10d"},
- /* format 3 */
- {0x2000, 0xF800, OP_THUMB_MOV, "mov\t%8-10r, #%0-7d"},
- {0x2800, 0xF800, OP_THUMB_CMP, "cmp\t%8-10r, #%0-7d"},
- {0x3000, 0xF800, OP_THUMB_ADD, "add\t%8-10r, #%0-7d"},
- {0x3800, 0xF800, OP_THUMB_SUB, "sub\t%8-10r, #%0-7d"},
- /* format 6 */
- /* TODO: Disassemble PC relative "LDR rD,=<symbolic>" */
- {0x4800, 0xF800, OP_THUMB_LDR, "ldr\t%8-10r, [pc, #%0-7W]\t(%0-7a)"},
- /* format 9 */
- {0x6000, 0xF800, OP_THUMB_STR, "str\t%0-2r, [%3-5r, #%6-10W]"},
- {0x6800, 0xF800, OP_THUMB_LDR, "ldr\t%0-2r, [%3-5r, #%6-10W]"},
- {0x7000, 0xF800, OP_THUMB_STRB, "strb\t%0-2r, [%3-5r, #%6-10d]"},
- {0x7800, 0xF800, OP_THUMB_LDRB, "ldrb\t%0-2r, [%3-5r, #%6-10d]"},
- /* format 10 */
- {0x8000, 0xF800, OP_THUMB_STRH, "strh\t%0-2r, [%3-5r, #%6-10H]"},
- {0x8800, 0xF800, OP_THUMB_LDRH, "ldrh\t%0-2r, [%3-5r, #%6-10H]"},
- /* format 11 */
- {0x9000, 0xF800, OP_THUMB_STR, "str\t%8-10r, [sp, #%0-7W]"},
- {0x9800, 0xF800, OP_THUMB_LDR, "ldr\t%8-10r, [sp, #%0-7W]"},
- /* format 12 */
- {0xA000, 0xF800, OP_THUMB_ADD, "add\t%8-10r, pc, #%0-7W\t(adr %8-10r,%0-7a)"},
- {0xA800, 0xF800, OP_THUMB_ADD, "add\t%8-10r, sp, #%0-7W"},
- /* format 15 */
- {0xC000, 0xF800, OP_THUMB_STMIA, "stmia\t%8-10r!,%M"},
- {0xC800, 0xF800, OP_THUMB_LDMIA, "ldmia\t%8-10r!,%M"},
- /* format 18 */
- {0xE000, 0xF800, OP_THUMB_B, "b\t%0-10B"},
- /* format 19 */
- /* special processing required in disassembler */
- {0xF000, 0xF800, OP_THUMB_BL, ""},
- {0xF800, 0xF800, OP_THUMB_BL, "second half of BL instruction %0-15x"},
- {0xE800, 0xF800, OP_THUMB_BLX, "second half of BLX instruction %0-15x"},
- /* format 16 */
- {0xD000, 0xFF00, OP_THUMB_B, "beq\t%0-7B"},
- {0xD100, 0xFF00, OP_THUMB_B, "bne\t%0-7B"},
- {0xD200, 0xFF00, OP_THUMB_B, "bcs\t%0-7B"},
- {0xD300, 0xFF00, OP_THUMB_B, "bcc\t%0-7B"},
- {0xD400, 0xFF00, OP_THUMB_B, "bmi\t%0-7B"},
- {0xD500, 0xFF00, OP_THUMB_B, "bpl\t%0-7B"},
- {0xD600, 0xFF00, OP_THUMB_B, "bvs\t%0-7B"},
- {0xD700, 0xFF00, OP_THUMB_B, "bvc\t%0-7B"},
- {0xD800, 0xFF00, OP_THUMB_B, "bhi\t%0-7B"},
- {0xD900, 0xFF00, OP_THUMB_B, "bls\t%0-7B"},
- {0xDA00, 0xFF00, OP_THUMB_B, "bge\t%0-7B"},
- {0xDB00, 0xFF00, OP_THUMB_B, "blt\t%0-7B"},
- {0xDC00, 0xFF00, OP_THUMB_B, "bgt\t%0-7B"},
- {0xDD00, 0xFF00, OP_THUMB_B, "ble\t%0-7B"},
- /* format 17 */
- {0xDE00, 0xFF00, OP_THUMB_UNDEFINED, "undefined"},
- {0xDF00, 0xFF00, OP_THUMB_SWI, "swi\t%0-7d"},
- /* format 9 */
- {0x6000, 0xF800, OP_THUMB_STR, "str\t%0-2r, [%3-5r, #%6-10W]"},
- {0x6800, 0xF800, OP_THUMB_LDR, "ldr\t%0-2r, [%3-5r, #%6-10W]"},
- {0x7000, 0xF800, OP_THUMB_STRB, "strb\t%0-2r, [%3-5r, #%6-10d]"},
- {0x7800, 0xF800, OP_THUMB_LDRB, "ldrb\t%0-2r, [%3-5r, #%6-10d]"},
- /* the rest */
- {0x0000, 0x0000, OP_THUMB_UNDEFINED, "undefined instruction %0-15x"},
- {0x0000, 0x0000, OP_END, 0}
-};
-
-#define BDISP23(x,y) ((((((x) & 0x07ff) << 11) | ((y) & 0x07ff)) \
- ^ 0x200000) - 0x200000) /* 23bit */
-
-static const char * arm_conditional[] =
-{"eq", "ne", "cs", "cc", "mi", "pl", "vs", "vc",
- "hi", "ls", "ge", "lt", "gt", "le", "", "nv"};
-
-typedef struct
-{
- const char * name;
- const char * description;
- const char * reg_names[16];
-}
-arm_regname;
-
-static arm_regname regnames[] =
-{
- { "raw" , "Select raw register names",
- { "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"}},
- { "gcc", "Select register names used by GCC",
- { "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "sl", "fp", "ip", "sp", "lr", "pc" }},
- { "std", "Select register names used in ARM's ISA documentation",
- { "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11", "r12", "sp", "lr", "pc" }},
- { "apcs", "Select register names used in the APCS",
- { "a1", "a2", "a3", "a4", "v1", "v2", "v3", "v4", "v5", "v6", "sl", "fp", "ip", "sp", "lr", "pc" }},
- { "atpcs", "Select register names used in the ATPCS",
- { "a1", "a2", "a3", "a4", "v1", "v2", "v3", "v4", "v5", "v6", "v7", "v8", "IP", "SP", "LR", "PC" }},
- { "special-atpcs", "Select special register names used in the ATPCS",
- { "a1", "a2", "a3", "a4", "v1", "v2", "v3", "WR", "v5", "SB", "SL", "FP", "IP", "SP", "LR", "PC" }}
-};
-
-/* Default to STD register name set. */
-static unsigned int regname_selected = 2;
-
-#define NUM_ARM_REGNAMES NUM_ELEM (regnames)
-#define arm_regnames regnames[regname_selected].reg_names
-
-Opcode decode_insn_thumb(uint32_t given)
-{
- struct thumb_opcode * insn;
-
- for (insn = thumb_opcodes; insn->assembler; insn++) {
- if ((given & insn->mask) == insn->value)
- return insn->opcode;
- }
- return OP_THUMB_UNDEFINED;
-}
-
-// Generates the disassembly string for the thumb instruction "insn1".
-// If "insn1" is a BL or BLX instruction that is the first of two Thumb
-// instructions, then insn2 is the second of two instructions. Otherwise,
-// insn2 is ignored.
-char *disasm_insn_thumb(uint32_t pc, uint32_t insn1, uint32_t insn2, char *result)
-{
- struct thumb_opcode * insn;
- static char buf[80];
- char *ptr;
- uint32_t addr;
- int len;
-
- if (result == NULL)
- result = buf;
- ptr = result;
-
- for (insn = thumb_opcodes; insn->assembler; insn++) {
- if ((insn1 & insn->mask) != insn->value)
- continue;
-
- const char * c = insn->assembler;
-
- /* Special processing for Thumb 2-instruction BL sequence: */
- if (!*c) { /* Check for empty (not NULL) assembler string. */
- uint32_t offset;
-
- offset = BDISP23 (insn1, insn2);
- offset = offset * 2 + pc + 4;
-
- if ((insn2 & 0x1000) == 0) {
- len = sprintf(ptr, "blx\t");
- offset &= 0xfffffffc;
- } else {
- len = sprintf(ptr, "bl\t");
- }
- ptr += len;
-
- sprintf(ptr, "0x%x", offset);
- return result;
- }
-
- insn1 &= 0xffff;
-
- for (; *c; c++) {
- if (*c != '%') {
- len = sprintf(ptr, "%c", *c);
- ptr += len;
- continue;
- }
-
- int domaskpc = 0;
- int domasklr = 0;
-
- switch (*++c) {
- case '%':
- len = sprintf(ptr, "%%");
- ptr += len;
- break;
-
- case 'S': {
- uint32_t reg;
-
- reg = (insn1 >> 3) & 0x7;
- if (insn1 & (1 << 6))
- reg += 8;
-
- len = sprintf(ptr, "%s", arm_regnames[reg]);
- ptr += len;
- break;
- }
-
- case 'D': {
- uint32_t reg;
-
- reg = insn1 & 0x7;
- if (insn1 & (1 << 7))
- reg += 8;
-
- len = sprintf(ptr, "%s", arm_regnames[reg]);
- ptr += len;
- break;
- }
-
- case 'T':
- len = sprintf(ptr, "%s",
- arm_conditional [(insn1 >> 8) & 0xf]);
- ptr += len;
- break;
-
- case 'N':
- if (insn1 & (1 << 8))
- domasklr = 1;
- /* Fall through. */
- case 'O':
- if (*c == 'O' && (insn1 & (1 << 8)))
- domaskpc = 1;
- /* Fall through. */
- case 'M': {
- int started = 0;
- int reg;
-
- len = sprintf(ptr, "{");
- ptr += len;
-
- /* It would be nice if we could spot
- ranges, and generate the rS-rE format: */
- for (reg = 0; (reg < 8); reg++)
- if ((insn1 & (1 << reg)) != 0) {
- if (started) {
- len = sprintf(ptr, ", ");
- ptr += len;
- }
- started = 1;
- len = sprintf(ptr, "%s", arm_regnames[reg]);
- ptr += len;
- }
-
- if (domasklr) {
- if (started) {
- len = sprintf(ptr, ", ");
- ptr += len;
- }
- started = 1;
- len = sprintf(ptr, arm_regnames[14] /* "lr" */);
- ptr += len;
- }
-
- if (domaskpc) {
- if (started) {
- len = sprintf(ptr, ", ");
- ptr += len;
- }
- len = sprintf(ptr, arm_regnames[15] /* "pc" */);
- ptr += len;
- }
-
- len = sprintf(ptr, "}");
- ptr += len;
- break;
- }
-
- case '0': case '1': case '2': case '3': case '4':
- case '5': case '6': case '7': case '8': case '9': {
- int bitstart = *c++ - '0';
- int bitend = 0;
-
- while (*c >= '0' && *c <= '9')
- bitstart = (bitstart * 10) + *c++ - '0';
-
- switch (*c) {
- case '-': {
- uint32_t reg;
-
- c++;
- while (*c >= '0' && *c <= '9')
- bitend = (bitend * 10) + *c++ - '0';
- if (!bitend)
- abort ();
- reg = insn1 >> bitstart;
- reg &= (2 << (bitend - bitstart)) - 1;
- switch (*c) {
- case 'r':
- len = sprintf(ptr, "%s", arm_regnames[reg]);
- break;
-
- case 'd':
- len = sprintf(ptr, "%d", reg);
- break;
-
- case 'H':
- len = sprintf(ptr, "%d", reg << 1);
- break;
-
- case 'W':
- len = sprintf(ptr, "%d", reg << 2);
- break;
-
- case 'a':
- /* PC-relative address -- the bottom two
- bits of the address are dropped
- before the calculation. */
- addr = ((pc + 4) & ~3) + (reg << 2);
- len = sprintf(ptr, "0x%x", addr);
- break;
-
- case 'x':
- len = sprintf(ptr, "0x%04x", reg);
- break;
-
- case 'I':
- reg = ((reg ^ (1 << bitend)) - (1 << bitend));
- len = sprintf(ptr, "%d", reg);
- break;
-
- case 'B':
- reg = ((reg ^ (1 << bitend)) - (1 << bitend));
- addr = reg * 2 + pc + 4;
- len = sprintf(ptr, "0x%x", addr);
- break;
-
- default:
- abort ();
- }
- ptr += len;
- break;
- }
-
- case '\'':
- c++;
- if ((insn1 & (1 << bitstart)) != 0) {
- len = sprintf(ptr, "%c", *c);
- ptr += len;
- }
- break;
-
- case '?':
- ++c;
- if ((insn1 & (1 << bitstart)) != 0)
- len = sprintf(ptr, "%c", *c++);
- else
- len = sprintf(ptr, "%c", *++c);
- ptr += len;
- break;
-
- default:
- abort ();
- }
- break;
- }
-
- default:
- abort ();
- }
- }
- return result;
- }
-
- /* No match. */
- abort ();
-}
diff --git a/emulator/qtools/trace_reader.cpp b/emulator/qtools/trace_reader.cpp
deleted file mode 100644
index 47b5d93..0000000
--- a/emulator/qtools/trace_reader.cpp
+++ /dev/null
@@ -1,1201 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <limits.h>
-#include <inttypes.h>
-#include <assert.h>
-#include <unistd.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <elf.h>
-#include "trace_reader.h"
-#include "decoder.h"
-
-// A struct for creating temporary linked-lists of DexSym structs
-struct DexSymList {
- DexSymList *next;
- DexSym sym;
-};
-
-// Declare static functions used in this file
-static char *ExtractDexPathFromMmap(const char *mmap_path);
-static void CopyDexSymbolsToArray(DexFileList *dexfile,
- DexSymList *head, int num_symbols);
-
-// This function creates the pathname to the a specific trace file. The
-// string space is allocated in this routine and must be freed by the
-// caller.
-static char *CreateTracePath(const char *filename, const char *ext)
-{
- char *fname;
- const char *base_start, *base_end;
- int ii, len, base_len, dir_len, path_len, qtrace_len;
-
- // Handle error cases
- if (filename == NULL || *filename == 0 || strcmp(filename, "/") == 0)
- return NULL;
-
- // Ignore a trailing slash, if any
- len = strlen(filename);
- if (filename[len - 1] == '/')
- len -= 1;
-
- // Find the basename. We don't use basename(3) because there are
- // different behaviors for GNU and Posix in the case where the
- // last character is a slash.
- base_start = base_end = &filename[len];
- for (ii = 0; ii < len; ++ii) {
- base_start -= 1;
- if (*base_start == '/') {
- base_start += 1;
- break;
- }
- }
- base_len = base_end - base_start;
- dir_len = len - base_len;
- qtrace_len = strlen("/qtrace");
-
- // Create space for the pathname: "/dir/basename/qtrace.ext"
- // The "ext" string already contains the dot, so just add a byte
- // for the terminating zero.
- path_len = dir_len + base_len + qtrace_len + strlen(ext) + 1;
- fname = new char[path_len];
- if (dir_len > 0)
- strncpy(fname, filename, dir_len);
- fname[dir_len] = 0;
- strncat(fname, base_start, base_len);
- strcat(fname, "/qtrace");
- strcat(fname, ext);
- return fname;
-}
-
-inline BBReader::Future *BBReader::AllocFuture()
-{
- Future *future = free_;
- free_ = free_->next;
- return future;
-}
-
-inline void BBReader::FreeFuture(Future *future)
-{
- future->next = free_;
- free_ = future;
-}
-
-inline void BBReader::InsertFuture(Future *future)
-{
- uint64_t future_time = future->bb.next_time;
- Future *prev = NULL;
- Future *ptr;
- for (ptr = head_; ptr; prev = ptr, ptr = ptr->next) {
- if (future_time <= ptr->bb.next_time)
- break;
- }
- if (prev == NULL) {
- // link it at the front
- future->next = head_;
- head_ = future;
- } else {
- // link it after "prev"
- future->next = prev->next;
- prev->next = future;
- }
-}
-
-// Decodes the next basic block record from the file. Returns 1
-// at end-of-file, otherwise returns 0.
-inline int BBReader::DecodeNextRec()
-{
- int64_t bb_diff = decoder_->Decode(true);
- uint64_t time_diff = decoder_->Decode(false);
- nextrec_.bb_rec.repeat = decoder_->Decode(false);
- if (time_diff == 0)
- return 1;
- if (nextrec_.bb_rec.repeat)
- nextrec_.bb_rec.time_diff = decoder_->Decode(false);
- nextrec_.bb_rec.bb_num += bb_diff;
- nextrec_.bb_rec.start_time += time_diff;
- return 0;
-}
-
-BBReader::BBReader(TraceReaderBase *trace)
-{
- trace_ = trace;
- decoder_ = new Decoder;
-}
-
-BBReader::~BBReader()
-{
- delete decoder_;
-}
-
-void BBReader::Open(const char *filename)
-{
- // Initialize the class variables
- memset(&nextrec_, 0, sizeof(TimeRec));
- memset(futures_, 0, sizeof(Future) * kMaxNumBasicBlocks);
- head_ = NULL;
-
- // Link all of the futures_[] array elements on the free list.
- for (int ii = 0; ii < kMaxNumBasicBlocks - 1; ++ii) {
- futures_[ii].next = &futures_[ii + 1];
- }
- futures_[kMaxNumBasicBlocks - 1].next = 0;
- free_ = &futures_[0];
-
- // Open the trace.bb file
- char *fname = CreateTracePath(filename, ".bb");
- decoder_->Open(fname);
- is_eof_ = DecodeNextRec();
- delete[] fname;
-}
-
-void BBReader::Close()
-{
- decoder_->Close();
-}
-
-// Returns true at end of file.
-bool BBReader::ReadBB(BBEvent *event)
-{
- if (is_eof_ && head_ == NULL) {
- return true;
- }
-
-#if 0
- if (nextrec_) {
- printf("nextrec: buffer[%d], bb_num: %lld start: %d diff %d repeat %d next %u\n",
- nextrec_ - &buffer_[0],
- nextrec_->bb_rec.bb_num, nextrec_->bb_rec.start_time,
- nextrec_->bb_rec.time_diff, nextrec_->bb_rec.repeat,
- nextrec_->next_time);
- }
- if (head_) {
- printf("head: 0x%x, bb_num: %lld start: %d diff %d repeat %d next %u\n",
- head_,
- head_->bb->bb_rec.bb_num, head_->bb->bb_rec.start_time,
- head_->bb->bb_rec.time_diff, head_->bb->bb_rec.repeat,
- head_->bb->next_time);
- }
-#endif
- if (!is_eof_) {
- if (head_) {
- TimeRec *bb = &head_->bb;
- if (bb->next_time < nextrec_.bb_rec.start_time) {
- // The head is earlier.
- event->time = bb->next_time;
- event->bb_num = bb->bb_rec.bb_num;
- event->bb_addr = trace_->GetBBAddr(event->bb_num);
- event->insns = trace_->GetInsns(event->bb_num);
- event->num_insns = trace_->FindNumInsns(event->bb_num, event->time);
- event->pid = trace_->FindCurrentPid(event->time);
- event->is_thumb = trace_->GetIsThumb(event->bb_num);
-
- // Remove the head element from the list
- Future *future = head_;
- head_ = head_->next;
- if (bb->bb_rec.repeat > 0) {
- // there are more repetitions of this bb
- bb->bb_rec.repeat -= 1;
- bb->next_time += bb->bb_rec.time_diff;
-
- // Insert this future into the sorted list
- InsertFuture(future);
- } else {
- // Add this future to the free list
- FreeFuture(future);
- }
- return false;
- }
- }
- // The nextrec is earlier (or there was no head)
- event->time = nextrec_.bb_rec.start_time;
- event->bb_num = nextrec_.bb_rec.bb_num;
- event->bb_addr = trace_->GetBBAddr(event->bb_num);
- event->insns = trace_->GetInsns(event->bb_num);
- event->num_insns = trace_->FindNumInsns(event->bb_num, event->time);
- event->pid = trace_->FindCurrentPid(event->time);
- event->is_thumb = trace_->GetIsThumb(event->bb_num);
- if (nextrec_.bb_rec.repeat > 0) {
- Future *future = AllocFuture();
- future->bb.bb_rec = nextrec_.bb_rec;
- future->bb.bb_rec.repeat -= 1;
- future->bb.next_time = nextrec_.bb_rec.start_time + nextrec_.bb_rec.time_diff;
- InsertFuture(future);
- }
-
- is_eof_ = DecodeNextRec();
- return false;
- }
-
- //printf("using head_ 0x%x\n", head_);
- assert(head_);
- TimeRec *bb = &head_->bb;
- event->time = bb->next_time;
- event->bb_num = bb->bb_rec.bb_num;
- event->bb_addr = trace_->GetBBAddr(event->bb_num);
- event->insns = trace_->GetInsns(event->bb_num);
- event->num_insns = trace_->FindNumInsns(event->bb_num, event->time);
- event->pid = trace_->FindCurrentPid(event->time);
- event->is_thumb = trace_->GetIsThumb(event->bb_num);
-
- // Remove the head element from the list
- Future *future = head_;
- head_ = head_->next;
- if (bb->bb_rec.repeat > 0) {
- // there are more repetitions of this bb
- bb->bb_rec.repeat -= 1;
- bb->next_time += bb->bb_rec.time_diff;
-
- // Insert this future into the sorted list
- InsertFuture(future);
- } else {
- // Add this future to the free list
- FreeFuture(future);
- }
- return false;
-}
-
-InsnReader::InsnReader()
-{
- decoder_ = new Decoder;
-}
-
-InsnReader::~InsnReader()
-{
- delete decoder_;
-}
-
-void InsnReader::Open(const char *filename)
-{
- prev_time_ = 0;
- time_diff_ = 0;
- repeat_ = -1;
-
- // Open the trace.insn file
- char *fname = CreateTracePath(filename, ".insn");
- decoder_->Open(fname);
- delete[] fname;
-}
-
-void InsnReader::Close()
-{
- decoder_->Close();
-}
-
-uint64_t InsnReader::ReadInsnTime(uint64_t min_time)
-{
- do {
- if (repeat_ == -1) {
- time_diff_ = decoder_->Decode(false);
- repeat_ = decoder_->Decode(false);
- }
- prev_time_ += time_diff_;
- repeat_ -= 1;
- } while (prev_time_ < min_time);
- return prev_time_;
-}
-
-AddrReader::AddrReader()
-{
- decoder_ = new Decoder;
- opened_ = false;
-}
-
-AddrReader::~AddrReader()
-{
- delete decoder_;
-}
-
-// Returns true if there is an error opening the file
-bool AddrReader::Open(const char *filename, const char *suffix)
-{
- struct stat stat_buf;
-
- prev_addr_ = 0;
- prev_time_ = 0;
-
- // Open the trace.addr file
- char *fname = CreateTracePath(filename, suffix);
- int rval = stat(fname, &stat_buf);
- if (rval == -1) {
- // The file does not exist
- delete[] fname;
- return true;
- }
- decoder_->Open(fname);
- opened_ = true;
- delete[] fname;
- return false;
-}
-
-void AddrReader::Close()
-{
- decoder_->Close();
-}
-
-// Returns true at end of file.
-bool AddrReader::ReadAddr(uint64_t *time, uint32_t *addr)
-{
- if (!opened_) {
- fprintf(stderr, "Cannot read address trace\n");
- exit(1);
- }
- uint32_t addr_diff = decoder_->Decode(true);
- uint64_t time_diff = decoder_->Decode(false);
- if (time_diff == 0 && addr_diff == 0) {
- *addr = 0;
- *time = 0;
- return true;
- }
- prev_addr_ += addr_diff;
- prev_time_ += time_diff;
- *addr = prev_addr_;
- *time = prev_time_;
- return false;
-}
-
-ExcReader::ExcReader()
-{
- decoder_ = new Decoder;
-}
-
-ExcReader::~ExcReader()
-{
- delete decoder_;
-}
-
-void ExcReader::Open(const char *filename)
-{
- prev_time_ = 0;
- prev_recnum_ = 0;
-
- // Open the trace.exc file
- char *fname = CreateTracePath(filename, ".exc");
- decoder_->Open(fname);
- delete[] fname;
-}
-
-void ExcReader::Close()
-{
- decoder_->Close();
-}
-
-// Returns true at end of file.
-bool ExcReader::ReadExc(uint64_t *time, uint32_t *current_pc, uint64_t *recnum,
- uint32_t *target_pc, uint64_t *bb_num,
- uint64_t *bb_start_time, int *num_insns)
-{
- uint64_t time_diff = decoder_->Decode(false);
- uint32_t pc = decoder_->Decode(false);
- if ((time_diff | pc) == 0) {
- decoder_->Decode(false);
- decoder_->Decode(false);
- decoder_->Decode(false);
- decoder_->Decode(false);
- decoder_->Decode(false);
- return true;
- }
- uint64_t recnum_diff = decoder_->Decode(false);
- prev_time_ += time_diff;
- prev_recnum_ += recnum_diff;
- *time = prev_time_;
- *current_pc = pc;
- *recnum = prev_recnum_;
- *target_pc = decoder_->Decode(false);
- *bb_num = decoder_->Decode(false);
- *bb_start_time = decoder_->Decode(false);
- *num_insns = decoder_->Decode(false);
- return false;
-}
-
-PidReader::PidReader()
-{
- decoder_ = new Decoder;
-}
-
-PidReader::~PidReader()
-{
- delete decoder_;
-}
-
-void PidReader::Open(const char *filename)
-{
- prev_time_ = 0;
-
- // Open the trace.pid file
- char *fname = CreateTracePath(filename, ".pid");
- decoder_->Open(fname);
- delete[] fname;
-}
-
-void PidReader::Close()
-{
- decoder_->Close();
-}
-
-// Returns true at end of file.
-bool PidReader::ReadPidEvent(PidEvent *event)
-{
- uint64_t time_diff = decoder_->Decode(false);
- int rec_type = decoder_->Decode(false);
- prev_time_ += time_diff;
- event->time = prev_time_;
- event->rec_type = rec_type;
- switch(rec_type) {
- case kPidEndOfFile:
- return true;
- case kPidSwitch:
- case kPidExit:
- event->pid = decoder_->Decode(false);
- break;
- case kPidFork:
- case kPidClone:
- event->tgid = decoder_->Decode(false);
- event->pid = decoder_->Decode(false);
- break;
- case kPidMmap:
- {
- event->vstart = decoder_->Decode(false);
- event->vend = decoder_->Decode(false);
- event->offset = decoder_->Decode(false);
- int len = decoder_->Decode(false);
- char *path = new char[len + 1];
- decoder_->Read(path, len);
- path[len] = 0;
- event->path = path;
- event->mmap_path = path;
- char *dexfile = ExtractDexPathFromMmap(path);
- if (dexfile != NULL) {
- delete[] event->path;
- event->path = dexfile;
- }
- }
- break;
- case kPidMunmap:
- {
- event->vstart = decoder_->Decode(false);
- event->vend = decoder_->Decode(false);
- }
- break;
- case kPidSymbolAdd:
- {
- event->vstart = decoder_->Decode(false);
- int len = decoder_->Decode(false);
- char *path = new char[len + 1];
- decoder_->Read(path, len);
- path[len] = 0;
- event->path = path;
- }
- break;
- case kPidSymbolRemove:
- event->vstart = decoder_->Decode(false);
- break;
- case kPidExec:
- {
- int argc = decoder_->Decode(false);
- event->argc = argc;
- char **argv = new char*[argc];
- event->argv = argv;
- for (int ii = 0; ii < argc; ++ii) {
- int alen = decoder_->Decode(false);
- argv[ii] = new char[alen + 1];
- decoder_->Read(argv[ii], alen);
- argv[ii][alen] = 0;
- }
- }
- break;
- case kPidName:
- case kPidKthreadName:
- {
- if (rec_type == kPidKthreadName) {
- event->tgid = decoder_->Decode(false);
- }
- event->pid = decoder_->Decode(false);
- int len = decoder_->Decode(false);
- char *path = new char[len + 1];
- decoder_->Read(path, len);
- path[len] = 0;
- event->path = path;
- }
- break;
- }
- return false;
-}
-
-// Frees the memory that might have been allocated for the given event.
-void PidReader::Dispose(PidEvent *event)
-{
- switch(event->rec_type) {
- case kPidMmap:
- case kPidSymbolAdd:
- case kPidName:
- case kPidKthreadName:
- delete[] event->path;
- event->path = NULL;
- event->mmap_path = NULL;
- break;
-
- case kPidExec:
- for (int ii = 0; ii < event->argc; ++ii) {
- delete[] event->argv[ii];
- }
- delete[] event->argv;
- event->argv = NULL;
- event->argc = 0;
- break;
- }
-}
-
-
-MethodReader::MethodReader()
-{
- decoder_ = new Decoder;
- opened_ = false;
-}
-
-MethodReader::~MethodReader()
-{
- delete decoder_;
-}
-
-bool MethodReader::Open(const char *filename)
-{
- struct stat stat_buf;
-
- prev_time_ = 0;
- prev_addr_ = 0;
- prev_pid_ = 0;
-
- // Open the trace.method file
- char *fname = CreateTracePath(filename, ".method");
- int rval = stat(fname, &stat_buf);
- if (rval == -1) {
- // The file does not exist
- delete[] fname;
- return true;
- }
- decoder_->Open(fname);
- delete[] fname;
- opened_ = true;
- return false;
-}
-
-void MethodReader::Close()
-{
- decoder_->Close();
-}
-
-// Returns true at end of file.
-bool MethodReader::ReadMethod(MethodRec *method_record)
-{
- if (!opened_)
- return true;
- uint64_t time_diff = decoder_->Decode(false);
- int32_t addr_diff = decoder_->Decode(true);
- if (time_diff == 0) {
- method_record->time = 0;
- method_record->addr = 0;
- method_record->flags = 0;
- return true;
- }
- int32_t pid_diff = decoder_->Decode(true);
- prev_time_ += time_diff;
- prev_addr_ += addr_diff;
- prev_pid_ += pid_diff;
- method_record->time = prev_time_;
- method_record->addr = prev_addr_;
- method_record->pid = prev_pid_;
- method_record->flags = decoder_->Decode(false);
- return false;
-}
-
-TraceReaderBase::TraceReaderBase()
-{
- static_filename_ = NULL;
- static_fstream_ = NULL;
- header_ = new TraceHeader;
- bb_reader_ = new BBReader(this);
- insn_reader_ = new InsnReader;
- load_addr_reader_ = new AddrReader;
- store_addr_reader_ = new AddrReader;
- exc_reader_ = new ExcReader;
- pid_reader_ = new PidReader;
- method_reader_ = new MethodReader;
- internal_exc_reader_ = new ExcReader;
- internal_pid_reader_ = new PidReader;
- internal_method_reader_ = new MethodReader;
- blocks_ = NULL;
- bb_recnum_ = 0;
- exc_recnum_ = 0;
- exc_end_ = false;
- exc_bb_num_ = 0;
- exc_time_ = 0;
- exc_num_insns_ = 0;
- current_pid_ = 0;
- next_pid_ = 0;
- next_pid_switch_time_ = 0;
- post_processing_ = false;
- dex_hash_ = NULL;
- load_eof_ = false;
- load_time_ = 0;
- load_addr_ = 0;
- store_eof_ = false;
- store_time_ = 0;
- store_addr_ = 0;
-}
-
-TraceReaderBase::~TraceReaderBase()
-{
- Close();
- delete bb_reader_;
- delete insn_reader_;
- delete load_addr_reader_;
- delete store_addr_reader_;
- delete exc_reader_;
- delete pid_reader_;
- delete method_reader_;
- delete internal_exc_reader_;
- delete internal_pid_reader_;
- delete internal_method_reader_;
- if (blocks_) {
- int num_static_bb = header_->num_static_bb;
- for (int ii = 0; ii < num_static_bb; ++ii) {
- delete[] blocks_[ii].insns;
- }
- delete[] blocks_;
- }
- delete header_;
- if (dex_hash_ != NULL) {
- HashTable<DexFileList*>::entry_type *ptr;
- for (ptr = dex_hash_->GetFirst(); ptr; ptr = dex_hash_->GetNext()) {
- DexFileList *dexfile = ptr->value;
- delete[] dexfile->path;
- int nsymbols = dexfile->nsymbols;
- DexSym *symbols = dexfile->symbols;
- for (int ii = 0; ii < nsymbols; ii++) {
- delete[] symbols[ii].name;
- }
- delete[] dexfile->symbols;
- delete dexfile;
- }
- }
- delete dex_hash_;
- delete[] static_filename_;
-}
-
-void TraceReaderBase::ReadTraceHeader(FILE *fstream, const char *filename,
- const char *tracename, TraceHeader *header)
-{
- int rval = fread(header, sizeof(TraceHeader), 1, fstream);
- if (rval != 1) {
- perror(filename);
- exit(1);
- }
-
- if (!post_processing_ && strcmp(header->ident, TRACE_IDENT) != 0) {
- fprintf(stderr, "%s: missing trace header; run 'post_trace %s' first\n",
- filename, tracename);
- exit(1);
- }
-
- if (header->version != TRACE_VERSION) {
- fprintf(stderr,
- "%s: trace header version (%d) does not match compiled tools version (%d)\n",
- tracename, header->version, TRACE_VERSION);
- exit(1);
- }
-
- convert32(header->version);
- convert32(header->start_sec);
- convert32(header->start_usec);
- convert32(header->pdate);
- convert32(header->ptime);
- convert64(header->num_static_bb);
- convert64(header->num_static_insn);
- convert64(header->num_dynamic_bb);
- convert64(header->num_dynamic_insn);
- convert64(header->elapsed_usecs);
-}
-
-
-void TraceReaderBase::Open(const char *filename)
-{
- char *fname;
- FILE *fstream;
-
- // Open the qtrace.bb file
- bb_reader_->Open(filename);
-
- // Open the qtrace.insn file
- insn_reader_->Open(filename);
-
- // Open the qtrace.load file and read the first line
- load_eof_ = load_addr_reader_->Open(filename, ".load");
- if (!load_eof_)
- load_eof_ = load_addr_reader_->ReadAddr(&load_time_, &load_addr_);
-
- // Open the qtrace.store file and read the first line
- store_eof_ = store_addr_reader_->Open(filename, ".store");
- if (!store_eof_)
- store_eof_ = store_addr_reader_->ReadAddr(&store_time_, &store_addr_);
-
- // Open the qtrace.exc file
- exc_reader_->Open(filename);
-
- // Open another file stream to the qtrace.exc file for internal reads.
- // This allows the caller to also read from the qtrace.exc file.
- internal_exc_reader_->Open(filename);
-
- // Open the qtrace.pid file
- pid_reader_->Open(filename);
- internal_pid_reader_->Open(filename);
-
- // Open the qtrace.method file
- method_reader_->Open(filename);
- internal_method_reader_->Open(filename);
-
- // Open the qtrace.static file
- fname = CreateTracePath(filename, ".static");
- static_filename_ = fname;
-
- fstream = fopen(fname, "r");
- if (fstream == NULL) {
- perror(fname);
- exit(1);
- }
- static_fstream_ = fstream;
-
- // Read the header
- ReadTraceHeader(fstream, fname, filename, header_);
-
- // Allocate space for all of the static blocks
- int num_static_bb = header_->num_static_bb;
- if (num_static_bb) {
- blocks_ = new StaticBlock[num_static_bb];
-
- // Read in all the static blocks
- for (int ii = 0; ii < num_static_bb; ++ii) {
- ReadStatic(&blocks_[ii].rec);
- int num_insns = blocks_[ii].rec.num_insns;
- if (num_insns > 0) {
- blocks_[ii].insns = new uint32_t[num_insns];
- ReadStaticInsns(num_insns, blocks_[ii].insns);
- } else {
- blocks_[ii].insns = NULL;
- }
- }
- fseek(static_fstream_, sizeof(TraceHeader), SEEK_SET);
- }
-
- ParseDexList(filename);
-
- // If the dex_hash_ is NULL, then assign it a small hash table
- // so that we can simply do a Find() operation without having
- // to check for NULL first.
- if (dex_hash_ == NULL) {
- dex_hash_ = new HashTable<DexFileList*>(1, NULL);
- }
-}
-
-// Reads the list of pid events looking for an mmap of a dex file.
-PidEvent * TraceReaderBase::FindMmapDexFileEvent()
-{
- static PidEvent event;
-
- while (!pid_reader_->ReadPidEvent(&event)) {
- if (event.rec_type == kPidMmap && event.path != event.mmap_path) {
- return &event;
- }
- pid_reader_->Dispose(&event);
- }
- return NULL;
-}
-
-static void CopyDexSymbolsToArray(DexFileList *dexfile,
- DexSymList *head, int num_symbols)
-{
- if (dexfile == NULL)
- return;
-
- DexSym *symbols = NULL;
- if (num_symbols > 0) {
- symbols = new DexSym[num_symbols];
- }
- dexfile->nsymbols = num_symbols;
- dexfile->symbols = symbols;
-
- // Copy the linked-list to the array.
- DexSymList *next_sym = NULL;
- int next_index = 0;
- for (DexSymList *sym = head; sym; sym = next_sym) {
- next_sym = sym->next;
- symbols[next_index].addr = sym->sym.addr;
- symbols[next_index].len = sym->sym.len;
- symbols[next_index].name = sym->sym.name;
- next_index += 1;
- delete sym;
- }
-}
-
-void TraceReaderBase::ParseDexList(const char *filename)
-{
- struct stat stat_buf;
- static const int kBufSize = 4096;
- char buf[kBufSize];
- char current_file[kBufSize];
-
- // Find an example dex file in the list of mmaps
- PidEvent *event = FindMmapDexFileEvent();
-
- // Reset the pid_reader to the beginning of the file.
- pid_reader_->Close();
- pid_reader_->Open(filename);
-
- // If there were no mmapped dex files, then there is no need to parse
- // the dexlist.
- if (event == NULL)
- return;
- char *mmap_dexfile = event->path;
-
- // Check if the dexlist file exists. It should have the name
- // "qtrace.dexlist"
- char *fname = CreateTracePath(filename, ".dexlist");
- int rval = stat(fname, &stat_buf);
- if (rval == -1) {
- // The file does not exist
- delete[] fname;
- return;
- }
-
- // Open the qtrace.dexlist file
- FILE *fstream = fopen(fname, "r");
- if (fstream == NULL) {
- perror(fname);
- exit(1);
- }
-
- // First pass: read all the filenames, looking for a match for the
- // example mmap dex filename. Also count the files so that we
- // know how big to make the hash table.
- char *match = NULL;
- int num_files = 0;
- while (fgets(buf, kBufSize, fstream)) {
- if (buf[0] != '#')
- continue;
- num_files += 1;
- match = strstr(buf + 1, mmap_dexfile);
-
- // Check that the dexlist file ends with the string mmap_dexfile.
- // We add one to the length of the mmap_dexfile because buf[]
- // ends with a newline. The strlen(mmap_dexfile) computation
- // could be moved above the loop but it should only ever be
- // executed once.
- if (match != NULL && strlen(match) == strlen(mmap_dexfile) + 1)
- break;
- }
-
- // Count the rest of the files
- while (fgets(buf, kBufSize, fstream)) {
- if (buf[0] == '#')
- num_files += 1;
- }
-
- if (match == NULL) {
- fprintf(stderr,
- "Cannot find the mmapped dex file '%s' in the dexlist\n",
- mmap_dexfile);
- exit(1);
- }
- delete[] mmap_dexfile;
-
- // The prefix length includes the leading '#'.
- int prefix_len = match - buf;
-
- // Allocate a hash table
- dex_hash_ = new HashTable<DexFileList*>(4 * num_files, NULL);
-
- // Reset the file stream to the beginning
- rewind(fstream);
-
- // Second pass: read the filenames, stripping off the common prefix.
- // And read all the (address, method) mappings. When we read a new
- // filename, create a new DexFileList and add it to the hash table.
- // Add new symbol mappings to a linked list until we have the whole
- // list and then create an array for them so that we can use binary
- // search on the address to find the symbol name quickly.
-
- // Use a linked list for storing the symbols
- DexSymList *head = NULL;
- DexSymList *prev = NULL;
- int num_symbols = 0;
-
- DexFileList *dexfile = NULL;
- int linenum = 0;
- while (fgets(buf, kBufSize, fstream)) {
- linenum += 1;
- if (buf[0] == '#') {
- // Everything after the '#' is a filename.
- // Ignore the common prefix.
-
- // First, save all the symbols from the previous file (if any).
- CopyDexSymbolsToArray(dexfile, head, num_symbols);
-
- dexfile = new DexFileList;
- // Subtract one because buf[] contains a trailing newline
- int pathlen = strlen(buf) - prefix_len - 1;
- char *path = new char[pathlen + 1];
- strncpy(path, buf + prefix_len, pathlen);
- path[pathlen] = 0;
- dexfile->path = path;
- dexfile->nsymbols = 0;
- dexfile->symbols = NULL;
- dex_hash_->Update(path, dexfile);
- num_symbols = 0;
- head = NULL;
- prev = NULL;
- continue;
- }
-
- uint32_t addr;
- int len, line;
- char clazz[kBufSize], method[kBufSize], sig[kBufSize], file[kBufSize];
- if (sscanf(buf, "0x%x %d %s %s %s %s %d",
- &addr, &len, clazz, method, sig, file, &line) != 7) {
- fprintf(stderr, "Cannot parse line %d of file %s:\n%s",
- linenum, fname, buf);
- exit(1);
- }
-
- // Concatenate the class name, method name, and signature
- // plus one for the period separating the class and method.
- int nchars = strlen(clazz) + strlen(method) + strlen(sig) + 1;
- char *name = new char[nchars + 1];
- strcpy(name, clazz);
- strcat(name, ".");
- strcat(name, method);
- strcat(name, sig);
-
- DexSymList *symbol = new DexSymList;
- symbol->sym.addr = addr;
- symbol->sym.len = len;
- symbol->sym.name = name;
- symbol->next = NULL;
-
- // Keep the list in the same order as the file
- if (head == NULL)
- head = symbol;
- if (prev != NULL)
- prev->next = symbol;
- prev = symbol;
- num_symbols += 1;
- }
- fclose(fstream);
-
- // Copy the symbols from the last file.
- CopyDexSymbolsToArray(dexfile, head, num_symbols);
- delete[] fname;
-}
-
-// Extracts the pathname to a jar file (or .apk file) from the mmap pathname.
-// An example mmap pathname looks something like this:
-// /data/dalvik-cache/system@app@TestHarness.apk@classes.dex
-// We want to convert that to this:
-// /system/app/TestHarness.apk
-// If the pathname is not of the expected form, then NULL is returned.
-// The space for the extracted path is allocated in this routine and should
-// be freed by the caller after it is no longer needed.
-static char *ExtractDexPathFromMmap(const char *mmap_path)
-{
- const char *end = rindex(mmap_path, '@');
- if (end == NULL)
- return NULL;
- const char *start = rindex(mmap_path, '/');
- if (start == NULL)
- return NULL;
- int len = end - start;
- char *path = new char[len + 1];
- strncpy(path, start, len);
- path[len] = 0;
-
- // Replace all the occurrences of '@' with '/'
- for (int ii = 0; ii < len; ii++) {
- if (path[ii] == '@')
- path[ii] = '/';
- }
- return path;
-}
-
-void TraceReaderBase::Close()
-{
- bb_reader_->Close();
- insn_reader_->Close();
- load_addr_reader_->Close();
- store_addr_reader_->Close();
- exc_reader_->Close();
- pid_reader_->Close();
- method_reader_->Close();
- internal_exc_reader_->Close();
- internal_pid_reader_->Close();
- internal_method_reader_->Close();
- fclose(static_fstream_);
- static_fstream_ = NULL;
-}
-
-void TraceReaderBase::WriteHeader(TraceHeader *header)
-{
- TraceHeader swappedHeader;
-
- freopen(static_filename_, "r+", static_fstream_);
- fseek(static_fstream_, 0, SEEK_SET);
-
- memcpy(&swappedHeader, header, sizeof(TraceHeader));
-
- convert32(swappedHeader.version);
- convert32(swappedHeader.start_sec);
- convert32(swappedHeader.start_usec);
- convert32(swappedHeader.pdate);
- convert32(swappedHeader.ptime);
- convert64(swappedHeader.num_static_bb);
- convert64(swappedHeader.num_static_insn);
- convert64(swappedHeader.num_dynamic_bb);
- convert64(swappedHeader.num_dynamic_insn);
- convert64(swappedHeader.elapsed_usecs);
-
- fwrite(&swappedHeader, sizeof(TraceHeader), 1, static_fstream_);
-}
-
-// Reads the next StaticRec from the trace file (not including the list
-// of instructions). On end-of-file, this function returns true.
-int TraceReaderBase::ReadStatic(StaticRec *rec)
-{
- int rval = fread(rec, sizeof(StaticRec), 1, static_fstream_);
- if (rval != 1) {
- if (feof(static_fstream_)) {
- return true;
- }
- perror(static_filename_);
- exit(1);
- }
- convert64(rec->bb_num);
- convert32(rec->bb_addr);
- convert32(rec->num_insns);
- return false;
-}
-
-// Reads "num" instructions into the array "insns" which must be large
-// enough to hold the "num" instructions.
-// Returns the actual number of instructions read. This will usually
-// be "num" but may be less if end-of-file occurred.
-int TraceReaderBase::ReadStaticInsns(int num, uint32_t *insns)
-{
- if (num == 0)
- return 0;
- int rval = fread(insns, sizeof(uint32_t), num, static_fstream_);
-
- // Convert from little-endian, if necessary
- for (int ii = 0; ii < num; ++ii)
- convert32(insns[ii]);
-
- if (rval != num) {
- if (feof(static_fstream_)) {
- return rval;
- }
- perror(static_filename_);
- exit(1);
- }
- return rval;
-}
-
-void TraceReaderBase::TruncateLastBlock(uint32_t num_insns)
-{
- uint32_t insns[kMaxInsnPerBB];
- StaticRec static_rec;
- long loc = 0, prev_loc = 0;
-
- freopen(static_filename_, "r+", static_fstream_);
- fseek(static_fstream_, sizeof(TraceHeader), SEEK_SET);
-
- // Find the last record
- while (1) {
- prev_loc = loc;
- loc = ftell(static_fstream_);
-
- // We don't need to byte-swap static_rec here because we are just
- // reading the records until we get to the last one.
- int rval = fread(&static_rec, sizeof(StaticRec), 1, static_fstream_);
- if (rval != 1)
- break;
- ReadStaticInsns(static_rec.num_insns, insns);
- }
- if (prev_loc != 0) {
- fseek(static_fstream_, prev_loc, SEEK_SET);
- static_rec.num_insns = num_insns;
-
- // Now we need to byte-swap, but just the field that we changed.
- convert32(static_rec.num_insns);
- fwrite(&static_rec, sizeof(StaticRec), 1, static_fstream_);
- int fd = fileno(static_fstream_);
- long len = ftell(static_fstream_);
- len += num_insns * sizeof(uint32_t);
- ftruncate(fd, len);
- }
-}
-
-int TraceReaderBase::FindNumInsns(uint64_t bb_num, uint64_t bb_start_time)
-{
- int num_insns;
-
- // Read the exception trace file. "bb_recnum_" is the number of
- // basic block records that have been read so far, and "exc_recnum_"
- // is the record number from the exception trace.
- while (!exc_end_ && exc_recnum_ < bb_recnum_) {
- uint32_t current_pc, target_pc;
- uint64_t time;
-
- exc_end_ = internal_exc_reader_->ReadExc(&time, &current_pc, &exc_recnum_,
- &target_pc, &exc_bb_num_,
- &exc_time_, &exc_num_insns_);
- }
-
- // If an exception occurred in this basic block, then use the
- // number of instructions specified in the exception record.
- if (!exc_end_ && exc_recnum_ == bb_recnum_) {
- num_insns = exc_num_insns_;
- } else {
- // Otherwise, use the number of instructions specified in the
- // static basic block.
- num_insns = blocks_[bb_num].rec.num_insns;
- }
- return num_insns;
-}
-
-// Finds the current pid for the given time. This routine reads the pid
-// trace file and assumes that the "time" parameter is monotonically
-// increasing.
-int TraceReaderBase::FindCurrentPid(uint64_t time)
-{
- PidEvent event;
-
- if (time < next_pid_switch_time_)
- return current_pid_;
-
- current_pid_ = next_pid_;
- while (1) {
- if (internal_pid_reader_->ReadPidEvent(&event)) {
- next_pid_switch_time_ = ~0ull;
- break;
- }
- if (event.rec_type != kPidSwitch)
- continue;
- if (event.time > time) {
- next_pid_ = event.pid;
- next_pid_switch_time_ = event.time;
- break;
- }
- current_pid_ = event.pid;
- }
- return current_pid_;
-}
diff --git a/emulator/qtools/trace_reader.h b/emulator/qtools/trace_reader.h
deleted file mode 100644
index b91cb1b..0000000
--- a/emulator/qtools/trace_reader.h
+++ /dev/null
@@ -1,1559 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef TRACE_READER_H
-#define TRACE_READER_H
-
-#include <string.h>
-#include <inttypes.h>
-#include <elf.h>
-#include <assert.h>
-#include <cxxabi.h>
-#include "read_elf.h"
-#include "trace_reader_base.h"
-#include "hash_table.h"
-
-struct TraceReaderEmptyStruct {
-};
-
-template <class T = TraceReaderEmptyStruct>
-class TraceReader : public TraceReaderBase {
- public:
-
- struct region_entry;
- typedef struct symbol_entry : public T {
- typedef region_entry region_type;
-
- // Define flag values
- static const uint32_t kIsPlt = 0x01;
- static const uint32_t kIsVectorStart = 0x02;
- static const uint32_t kIsVectorTable = (kIsPlt | kIsVectorStart);
- static const uint32_t kIsInterpreter = 0x04;
- static const uint32_t kIsMethod = 0x08;
-
- uint32_t addr;
-
- // This may hold the name of the interpreted method instead of
- // the name of the native function if the native function is a
- // virtual machine interpreter.
- const char *name;
-
- // The symbol for the virtual machine interpreter, or NULL
- symbol_entry *vm_sym;
- region_type *region;
- uint32_t flags;
- } symbol_type;
-
- typedef struct region_entry {
- // Define flag values
- static const uint32_t kIsKernelRegion = 0x01;
- static const uint32_t kSharedSymbols = 0x02;
- static const uint32_t kIsLibraryRegion = 0x04;
- static const uint32_t kIsUserMappedRegion = 0x08;
-
- region_entry() : refs(0), path(NULL), vstart(0), vend(0), base_addr(0),
- file_offset(0), flags(0), nsymbols(0), symbols(NULL) {}
-
- symbol_type *LookupFunctionByName(char *name) {
- // Just do a linear search
- for (int ii = 0; ii < nsymbols; ++ii) {
- if (strcmp(symbols[ii].name, name) == 0)
- return &symbols[ii];
- }
- return NULL;
- }
-
- region_entry *MakePrivateCopy(region_entry *dest) {
- dest->refs = 0;
- dest->path = Strdup(path);
- dest->vstart = vstart;
- dest->vend = vend;
- dest->base_addr = base_addr;
- dest->file_offset = file_offset;
- dest->flags = flags;
- dest->nsymbols = nsymbols;
- dest->symbols = symbols;
- return dest;
- }
-
- int refs; // reference count
- char *path;
- uint32_t vstart;
- uint32_t vend;
- uint32_t base_addr;
- uint32_t file_offset;
- uint32_t flags;
- int nsymbols;
- symbol_type *symbols;
- } region_type;
-
- typedef typename HashTable<region_type*>::entry_type hash_entry_type;
-
- class ProcessState {
- public:
-
- // The "regions" array below is a pointer to array of pointers to
- // regions. The size of the pointer array is kInitialNumRegions,
- // but grows if needed. There is a separate region for each mmap
- // call which includes shared libraries as well as .dex and .jar
- // files. In addition, there is a region for the main executable
- // for this process, as well as a few regions for the kernel.
- //
- // If a child process is a clone of a parent process, the
- // regions array is unused. Instead, the "addr_manager" pointer is
- // used to find the process that is the address space manager for
- // both the parent and child processes.
- static const int kInitialNumRegions = 10;
-
- static const int kMaxMethodStackSize = 1000;
-
- // Define values for the ProcessState flag bits
- static const int kCalledExec = 0x01;
- static const int kCalledExit = 0x02;
- static const int kIsClone = 0x04;
- static const int kHasKernelRegion = 0x08;
- static const int kHasFirstMmap = 0x10;
-
- struct methodFrame {
- uint32_t addr;
- bool isNative;
- };
-
- ProcessState() {
- cpu_time = 0;
- tgid = 0;
- pid = 0;
- parent_pid = 0;
- exit_val = 0;
- flags = 0;
- argc = 0;
- argv = NULL;
- name = NULL;
- nregions = 0;
- max_regions = 0;
- // Don't allocate space yet until we know if we are a clone.
- regions = NULL;
- parent = NULL;
- addr_manager = this;
- next = NULL;
- current_method_sym = NULL;
- method_stack_top = 0;
- }
-
- ~ProcessState() {
- delete[] name;
- if ((flags & kIsClone) != 0) {
- return;
- }
-
- // Free the regions. We must be careful not to free the symbols
- // within each region because the symbols are sometimes shared
- // between multiple regions. The TraceReader class has a hash
- // table containing all the unique regions and it will free the
- // region symbols in its destructor. We need to free only the
- // regions and the array of region pointers.
- //
- // Each region is also reference-counted. The count is zero
- // if no other processes are sharing this region.
- for (int ii = 0; ii < nregions; ii++) {
- if (regions[ii]->refs > 0) {
- regions[ii]->refs -= 1;
- continue;
- }
-
- delete regions[ii];
- }
-
- delete[] regions;
-
- for (int ii = 0; ii < argc; ++ii)
- delete[] argv[ii];
- delete[] argv;
- }
-
- // Dumps the stack contents to standard output. For debugging.
- void DumpStack(FILE *stream);
-
- uint64_t cpu_time;
- uint64_t start_time;
- uint64_t end_time;
- int tgid;
- int pid;
- int parent_pid;
- int exit_val;
- uint32_t flags;
- int argc;
- char **argv;
- const char *name;
- int nregions; // num regions in use
- int max_regions; // max regions allocated
- region_type **regions;
- ProcessState *parent;
- ProcessState *addr_manager; // the address space manager process
- ProcessState *next;
- int method_stack_top;
- methodFrame method_stack[kMaxMethodStackSize];
- symbol_type *current_method_sym;
- };
-
- TraceReader();
- ~TraceReader();
-
- void ReadKernelSymbols(const char *kernel_file);
- void CopyKernelRegion(ProcessState *pstate);
- void ClearRegions(ProcessState *pstate);
- void CopyRegions(ProcessState *parent, ProcessState *child);
- void DumpRegions(FILE *stream, ProcessState *pstate);
- symbol_type *LookupFunction(int pid, uint32_t addr, uint64_t time);
- symbol_type *GetSymbols(int *num_syms);
- ProcessState *GetCurrentProcess() { return current_; }
- ProcessState *GetProcesses(int *num_procs);
- ProcessState *GetNextProcess();
- const char *GetProcessName(int pid);
- void SetRoot(const char *root) { root_ = root; }
- void SetDemangle(bool demangle) { demangle_ = demangle; }
- bool ReadMethodSymbol(MethodRec *method_record,
- symbol_type **psym,
- ProcessState **pproc);
-
- protected:
- virtual int FindCurrentPid(uint64_t time);
-
- private:
-
- static const int kNumPids = 32768;
- static const uint32_t kIncludeLocalSymbols = 0x1;
-
- void AddPredefinedRegion(region_type *region, const char *path,
- uint32_t vstart, uint32_t vend,
- uint32_t base);
- void InitRegionSymbols(region_type *region, int nsymbols);
- void AddRegionSymbol(region_type *region, int idx,
- uint32_t addr, const char *name,
- uint32_t flags);
- void AddPredefinedRegions(ProcessState *pstate);
- void demangle_names(int nfuncs, symbol_type *functions);
- bool ReadElfSymbols(region_type *region, uint32_t flags);
- void AddRegion(ProcessState *pstate, region_type *region);
- region_type *FindRegion(uint32_t addr, int nregions,
- region_type **regions);
- int FindRegionIndex(uint32_t addr, int nregions,
- region_type **regions);
- void FindAndRemoveRegion(ProcessState *pstate,
- uint32_t vstart, uint32_t vend);
- symbol_type *FindFunction(uint32_t addr, int nsyms,
- symbol_type *symbols, bool exact_match);
- symbol_type *FindCurrentMethod(int pid, uint64_t time);
- void PopulateSymbolsFromDexFile(const DexFileList *dexfile,
- region_type *region);
- void HandlePidEvent(PidEvent *event);
- void HandleMethodRecord(ProcessState *pstate,
- MethodRec *method_rec);
-
- int cached_pid_;
- symbol_type *cached_func_;
- symbol_type unknown_;
- int next_pid_;
-
- PidEvent next_pid_event_;
- ProcessState *processes_[kNumPids];
- ProcessState *current_;
- MethodRec next_method_;
- uint64_t function_start_time_;
- const char *root_;
- HashTable<region_type*> *hash_;
- bool demangle_;
-};
-
-template<class T>
-TraceReader<T>::TraceReader()
-{
- static PidEvent event_no_action;
-
- cached_pid_ = -1;
- cached_func_ = NULL;
-
- memset(&unknown_, 0, sizeof(symbol_type));
- unknown_.name = "(unknown)";
- next_pid_ = 0;
-
- memset(&event_no_action, 0, sizeof(PidEvent));
- event_no_action.rec_type = kPidNoAction;
- next_pid_event_ = event_no_action;
- for (int ii = 1; ii < kNumPids; ++ii)
- processes_[ii] = NULL;
- current_ = new ProcessState;
- processes_[0] = current_;
- next_method_.time = 0;
- next_method_.addr = 0;
- next_method_.flags = 0;
- function_start_time_ = 0;
- root_ = "";
- hash_ = new HashTable<region_type*>(512);
- AddPredefinedRegions(current_);
- demangle_ = true;
-}
-
-template<class T>
-TraceReader<T>::~TraceReader()
-{
- hash_entry_type *ptr;
- for (ptr = hash_->GetFirst(); ptr; ptr = hash_->GetNext()) {
- region_type *region = ptr->value;
- // If the symbols are not shared with another region, then delete them.
- if ((region->flags & region_type::kSharedSymbols) == 0) {
- int nsymbols = region->nsymbols;
- for (int ii = 0; ii < nsymbols; ii++) {
- delete[] region->symbols[ii].name;
- }
- delete[] region->symbols;
- }
- delete[] region->path;
-
- // Do not delete the region itself here. Each region
- // is reference-counted and deleted by the ProcessState
- // object that owns it.
- }
- delete hash_;
-
- // Delete the ProcessState objects after the region symbols in
- // the hash table above so that we still have valid region pointers
- // when deleting the region symbols.
- for (int ii = 0; ii < kNumPids; ++ii) {
- delete processes_[ii];
- }
-}
-
-// This function is used by the qsort() routine to sort symbols
-// into increasing address order.
-template<class T>
-int cmp_symbol_addr(const void *a, const void *b) {
- typedef typename TraceReader<T>::symbol_type stype;
-
- const stype *syma = static_cast<stype const *>(a);
- const stype *symb = static_cast<stype const *>(b);
- uint32_t addr1 = syma->addr;
- uint32_t addr2 = symb->addr;
- if (addr1 < addr2)
- return -1;
- if (addr1 > addr2)
- return 1;
-
- // The addresses are the same, sort the symbols into
- // increasing alphabetical order. But put symbols that
- // that start with "_" last.
- if (syma->name[0] == '_' || symb->name[0] == '_') {
- // Count the number of leading underscores and sort the
- // symbol with the most underscores last.
- int aCount = 0;
- while (syma->name[aCount] == '_')
- aCount += 1;
- int bCount = 0;
- while (symb->name[bCount] == '_')
- bCount += 1;
- if (aCount < bCount) {
- return -1;
- }
- if (aCount > bCount) {
- return 1;
- }
- // If the symbols have the same number of underscores, then
- // fall through and sort by the whole name.
- }
- return strcmp(syma->name, symb->name);
-}
-
-// This function is used by the qsort() routine to sort region entries
-// into increasing address order.
-template<class T>
-int cmp_region_addr(const void *a, const void *b) {
- typedef typename TraceReader<T>::region_type rtype;
-
- const rtype *ma = *static_cast<rtype* const *>(a);
- const rtype *mb = *static_cast<rtype* const *>(b);
- uint32_t addr1 = ma->vstart;
- uint32_t addr2 = mb->vstart;
- if (addr1 < addr2)
- return -1;
- if (addr1 == addr2)
- return 0;
- return 1;
-}
-
-// This routine returns a new array containing all the symbols.
-template<class T>
-typename TraceReader<T>::symbol_type*
-TraceReader<T>::GetSymbols(int *num_syms)
-{
- // Count the symbols
- int nsyms = 0;
- for (hash_entry_type *ptr = hash_->GetFirst(); ptr; ptr = hash_->GetNext()) {
- region_type *region = ptr->value;
- nsyms += region->nsymbols;
- }
- *num_syms = nsyms;
-
- // Allocate space
- symbol_type *syms = new symbol_type[nsyms];
- symbol_type *next_sym = syms;
-
- // Copy the symbols
- for (hash_entry_type *ptr = hash_->GetFirst(); ptr; ptr = hash_->GetNext()) {
- region_type *region = ptr->value;
- memcpy(next_sym, region->symbols, region->nsymbols * sizeof(symbol_type));
- next_sym += region->nsymbols;
- }
-
- return syms;
-}
-
-// This routine returns all the valid processes.
-template<class T>
-typename TraceReader<T>::ProcessState*
-TraceReader<T>::GetProcesses(int *num_procs)
-{
- // Count the valid processes
- int nprocs = 0;
- for (int ii = 0; ii < kNumPids; ++ii) {
- if (processes_[ii])
- nprocs += 1;
- }
-
- // Allocate a new array to hold the valid processes.
- ProcessState *procs = new ProcessState[nprocs];
-
- // Copy the processes to the new array.
- ProcessState *pstate = procs;
- for (int ii = 0; ii < kNumPids; ++ii) {
- if (processes_[ii])
- memcpy(pstate++, processes_[ii], sizeof(ProcessState));
- }
-
- *num_procs = nprocs;
- return procs;
-}
-
-// This routine returns the next valid process, or NULL if there are no
-// more valid processes.
-template<class T>
-typename TraceReader<T>::ProcessState*
-TraceReader<T>::GetNextProcess()
-{
- while (next_pid_ < kNumPids) {
- if (processes_[next_pid_])
- return processes_[next_pid_++];
- next_pid_ += 1;
- }
- next_pid_ = 0;
- return NULL;
-}
-
-template<class T>
-const char* TraceReader<T>::GetProcessName(int pid)
-{
- if (pid < 0 || pid >= kNumPids || processes_[pid] == NULL)
- return "(unknown)";
- return processes_[pid]->name;
-}
-
-template<class T>
-void TraceReader<T>::AddPredefinedRegion(region_type *region, const char *path,
- uint32_t vstart, uint32_t vend,
- uint32_t base)
-{
- // Copy the path to make it easy to delete later.
- int len = strlen(path);
- region->path = new char[len + 1];
- strcpy(region->path, path);
- region->vstart = vstart;
- region->vend = vend;
- region->base_addr = base;
- region->flags = region_type::kIsKernelRegion;
-}
-
-template<class T>
-void TraceReader<T>::InitRegionSymbols(region_type *region, int nsymbols)
-{
- region->nsymbols = nsymbols;
- region->symbols = new symbol_type[nsymbols];
- memset(region->symbols, 0, nsymbols * sizeof(symbol_type));
-}
-
-template<class T>
-void TraceReader<T>::AddRegionSymbol(region_type *region, int idx,
- uint32_t addr, const char *name,
- uint32_t flags)
-{
- region->symbols[idx].addr = addr;
- region->symbols[idx].name = Strdup(name);
- region->symbols[idx].vm_sym = NULL;
- region->symbols[idx].region = region;
- region->symbols[idx].flags = flags;
-}
-
-template<class T>
-void TraceReader<T>::AddPredefinedRegions(ProcessState *pstate)
-{
- region_type *region = new region_type;
- AddPredefinedRegion(region, "(bootloader)", 0, 0x14, 0);
- InitRegionSymbols(region, 2);
- AddRegionSymbol(region, 0, 0, "(bootloader_start)", 0);
- AddRegionSymbol(region, 1, 0x14, "(bootloader_end)", 0);
- AddRegion(pstate, region);
- hash_->Update(region->path, region);
-
- region = new region_type;
- AddPredefinedRegion(region, "(exception vectors)", 0xffff0000, 0xffff0500,
- 0xffff0000);
- InitRegionSymbols(region, 2);
- AddRegionSymbol(region, 0, 0x0, "(vector_start)",
- symbol_type::kIsVectorStart);
- AddRegionSymbol(region, 1, 0x500, "(vector_end)", 0);
- AddRegion(pstate, region);
- hash_->Update(region->path, region);
-
- region = new region_type;
- AddPredefinedRegion(region, "(atomic ops)", 0xffff0f80, 0xffff1000,
- 0xffff0f80);
- // Mark this region as also being mapped in user-space.
- // This isn't used anywhere in this code but client code can test for
- // this flag and decide whether to treat this as kernel or user code.
- region->flags |= region_type::kIsUserMappedRegion;
-
- InitRegionSymbols(region, 4);
- AddRegionSymbol(region, 0, 0x0, "(kuser_atomic_inc)", 0);
- AddRegionSymbol(region, 1, 0x20, "(kuser_atomic_dec)", 0);
- AddRegionSymbol(region, 2, 0x40, "(kuser_cmpxchg)", 0);
- AddRegionSymbol(region, 3, 0x80, "(kuser_end)", 0);
- AddRegion(pstate, region);
- hash_->Update(region->path, region);
-}
-
-template<class T>
-void TraceReader<T>::ReadKernelSymbols(const char *kernel_file)
-{
- region_type *region = new region_type;
- // Copy the path to make it easy to delete later.
- int len = strlen(kernel_file);
- region->path = new char[len + 1];
- strcpy(region->path, kernel_file);
- region->flags = region_type::kIsKernelRegion;
- ReadElfSymbols(region, kIncludeLocalSymbols);
- region->vend = 0xffff0000;
- AddRegion(processes_[0], region);
- processes_[0]->flags |= ProcessState::kHasKernelRegion;
- hash_->Update(region->path, region);
-}
-
-template<class T>
-void TraceReader<T>::demangle_names(int nfuncs, symbol_type *functions)
-{
- char *demangled;
- int status;
-
- for (int ii = 0; ii < nfuncs; ++ii) {
- demangled = NULL;
- int len = strlen(functions[ii].name);
-
- // If we don't check for "len > 1" then the demangler will incorrectly
- // expand 1-letter function names. For example, "b" becomes "bool",
- // "c" becomes "char" and "d" becomes "double". Also check that the
- // first character is an underscore. Otherwise, on some strings
- // the demangler will try to read past the end of the string (because
- // the string is not really a C++ mangled name) and valgrind will
- // complain.
- if (demangle_ && len > 1 && functions[ii].name[0] == '_') {
- demangled = abi::__cxa_demangle(functions[ii].name, 0, NULL,
- &status);
- }
-
- if (demangled != NULL) {
- delete[] functions[ii].name;
- functions[ii].name = Strdup(demangled);
- free(demangled);
- }
- }
-}
-
-// Adds the symbols from the given ELF file to the given process.
-// Returns false if the file was not an ELF file or if there was an
-// error trying to read the sections of the ELF file.
-template<class T>
-bool TraceReader<T>::ReadElfSymbols(region_type *region, uint32_t flags)
-{
- static char full_path[4096];
- Elf32_Shdr *symtab, *symstr;
- Elf32_Ehdr *hdr;
- Elf32_Shdr *shdr;
-
- full_path[0] = 0;
- if (root_ && strcmp(root_, "/")) {
- strcpy(full_path, root_);
- }
- strcat(full_path, region->path);
- FILE *fobj = fopen(full_path, "r");
- if(fobj == NULL) {
- EmptyRegion:
- // we need to create an (unknown) symbol with address 0, otherwise some
- // other parts of the trace reader will simply crash when dealing with
- // an empty region
- region->vstart = 0;
- region->nsymbols = 1;
- region->symbols = new symbol_type[1];
- memset(region->symbols, 0, sizeof(symbol_type));
-
- region->symbols[0].addr = 0;
- region->symbols[0].name = Strdup("(unknown)");
- region->symbols[0].vm_sym = NULL;
- region->symbols[0].region = region;
- region->symbols[0].flags = 0;
-
- if (fobj != NULL)
- fclose(fobj);
- return false;
- }
-
- hdr = ReadElfHeader(fobj);
- if (hdr == NULL) {
- fprintf(stderr, "Cannot read ELF header from '%s'\n", full_path);
- goto EmptyRegion;
- }
-
- shdr = ReadSectionHeaders(hdr, fobj);
- if(shdr == NULL) {
- fprintf(stderr, "Can't read section headers from executable\n");
- goto EmptyRegion;
- }
- char *section_names = ReadStringTable(hdr, shdr, fobj);
-
- // Get the symbol table section
- symtab = FindSymbolTableSection(hdr, shdr, section_names);
- if (symtab == NULL || symtab->sh_size == 0) {
- fprintf(stderr, "Can't read symbol table from '%s'\n", full_path);
- goto EmptyRegion;
- }
-
- // Get the symbol string table section
- symstr = FindSymbolStringTableSection(hdr, shdr, section_names);
- if (symstr == NULL || symstr->sh_size == 0) {
- fprintf(stderr, "Can't read symbol string table from '%s'\n", full_path);
- goto EmptyRegion;
- }
-
- // Load the symbol string table data
- char *symbol_names = new char[symstr->sh_size];
- ReadSection(symstr, symbol_names, fobj);
-
- int num_entries = symtab->sh_size / symtab->sh_entsize;
- Elf32_Sym *elf_symbols = new Elf32_Sym[num_entries];
- ReadSection(symtab, elf_symbols, fobj);
- AdjustElfSymbols(hdr, elf_symbols, num_entries);
-#if 0
- printf("size: %d, ent_size: %d, num_entries: %d\n",
- symtab->sh_size, symtab->sh_entsize, num_entries);
-#endif
- int nfuncs = 0;
-
- // Allocate space for all of the symbols for now. We will
- // reallocate space for just the function symbols after we
- // know how many there are. Also, make sure there is room
- // for some extra symbols, including the text section names.
- int num_alloc = num_entries + hdr->e_shnum + 1;
- symbol_type *func_symbols = new symbol_type[num_alloc];
- memset(func_symbols, 0, num_alloc * sizeof(symbol_type));
-
- // If this is the shared library for a virtual machine, then
- // set the IsInterpreter flag for all symbols in that shared library.
- // This will allow us to replace the symbol names with the name of
- // the currently executing method on the virtual machine.
- int symbol_flags = 0;
- char *cp = strrchr(region->path, '/');
- if (cp != NULL) {
- // Move past the '/'
- cp += 1;
- } else {
- // There was no '/', so use the whole path
- cp = region->path;
- }
- if (strcmp(cp, "libdvm.so") == 0) {
- symbol_flags = symbol_type::kIsInterpreter;
- }
-
- bool zero_found = false;
- for (int ii = 1; ii < num_entries; ++ii) {
- int idx = elf_symbols[ii].st_name;
-
- // If the symbol does not have a name, or if the name starts with a
- // dollar sign ($), then skip it.
- if (idx == 0 || symbol_names[idx] == 0 || symbol_names[idx] == '$')
- continue;
-
- // If the section index is not executable, then skip it.
- uint32_t section = elf_symbols[ii].st_shndx;
- if (section == 0 || section >= hdr->e_shnum)
- continue;
- if ((shdr[section].sh_flags & SHF_EXECINSTR) == 0)
- continue;
-
- uint8_t sym_type = ELF32_ST_TYPE(elf_symbols[ii].st_info);
- uint8_t sym_bind = ELF32_ST_BIND(elf_symbols[ii].st_info);
-
- // Allow the caller to decide if we want local non-function
- // symbols to be included. We currently include these symbols
- // only for the kernel, where it is useful because the kernel
- // has lots of assembly language labels that have meaningful names.
- if ((flags & kIncludeLocalSymbols) == 0 && sym_bind == STB_LOCAL
- && sym_type != STT_FUNC) {
- continue;
- }
-#if 0
- printf("%08x %x %x %s\n",
- elf_symbols[ii].st_value,
- sym_bind,
- sym_type,
- &symbol_names[idx]);
-#endif
- if (sym_type != STT_FUNC && sym_type != STT_NOTYPE)
- continue;
-
- if (elf_symbols[ii].st_value == 0)
- zero_found = true;
-
- // The address of thumb functions seem to have the low bit set,
- // even though the instructions are really at an even address.
- uint32_t addr = elf_symbols[ii].st_value & ~0x1;
- func_symbols[nfuncs].addr = addr;
- func_symbols[nfuncs].name = Strdup(&symbol_names[idx]);
- func_symbols[nfuncs].flags = symbol_flags;
-
- nfuncs += 1;
- }
-
- // Add a [0, "(unknown)"] symbol pair if there is not already a
- // symbol with the address zero. We don't need to reallocate space
- // because we already have more than we need.
- if (!zero_found) {
- func_symbols[nfuncs].addr = 0;
- func_symbols[nfuncs].name = Strdup("(0 unknown)");
- nfuncs += 1;
- }
-
- // Add another entry at the end
- func_symbols[nfuncs].addr = 0xffffffff;
- func_symbols[nfuncs].name = Strdup("(end)");
- nfuncs += 1;
-
- // Add in the names of the text sections, but only if there
- // are no symbols with that address already.
- for (int section = 0; section < hdr->e_shnum; ++section) {
- if ((shdr[section].sh_flags & SHF_EXECINSTR) == 0)
- continue;
-
- uint32_t addr = shdr[section].sh_addr;
- // Search for a symbol with a matching address. The symbols aren't
- // sorted yet so we just search the whole list.
- int ii;
- for (ii = 0; ii < nfuncs; ++ii) {
- if (addr == func_symbols[ii].addr)
- break;
- }
- if (ii == nfuncs) {
- // Symbol at address "addr" does not exist, so add the text
- // section name. This will usually add the ".plt" section
- // (procedure linkage table).
- int idx = shdr[section].sh_name;
- func_symbols[nfuncs].addr = addr;
- func_symbols[nfuncs].name = Strdup(&section_names[idx]);
- if (strcmp(func_symbols[nfuncs].name, ".plt") == 0) {
- func_symbols[nfuncs].flags |= symbol_type::kIsPlt;
- // Change the name of the symbol to include the
- // name of the library. Otherwise we will have lots
- // of ".plt" symbols.
- int len = strlen(region->path);
- len += strlen(":.plt");
- char *name = new char[len + 1];
- strcpy(name, region->path);
- strcat(name, ":.plt");
- delete[] func_symbols[nfuncs].name;
- func_symbols[nfuncs].name = name;
-
- // Check if this is part of the virtual machine interpreter
- char *cp = strrchr(region->path, '/');
- if (cp != NULL) {
- // Move past the '/'
- cp += 1;
- } else {
- // There was no '/', so use the whole path
- cp = region->path;
- }
- if (strcmp(cp, "libdvm.so") == 0) {
- func_symbols[nfuncs].flags |= symbol_type::kIsInterpreter;
- }
- }
- nfuncs += 1;
- }
- }
-
- // Allocate just the space we need now that we know exactly
- // how many symbols we have.
- symbol_type *functions = new symbol_type[nfuncs];
-
- // Copy the symbols to the functions array
- memcpy(functions, func_symbols, nfuncs * sizeof(symbol_type));
- delete[] func_symbols;
-
- // Assign the region pointers
- for (int ii = 0; ii < nfuncs; ++ii) {
- functions[ii].region = region;
- }
-
- // Sort the symbols into increasing address order
- qsort(functions, nfuncs, sizeof(symbol_type), cmp_symbol_addr<T>);
-
- // If there are multiple symbols with the same address, then remove
- // the duplicates. First, count the number of duplicates.
- uint32_t prev_addr = ~0;
- int num_duplicates = 0;
- for (int ii = 0; ii < nfuncs; ++ii) {
- if (prev_addr == functions[ii].addr)
- num_duplicates += 1;
- prev_addr = functions[ii].addr;
- }
-
- if (num_duplicates > 0) {
- int num_uniq = nfuncs - num_duplicates;
-
- // Allocate space for the unique functions
- symbol_type *uniq_functions = new symbol_type[num_uniq];
-
- // Copy the unique functions
- prev_addr = ~0;
- int next_uniq = 0;
- for (int ii = 0; ii < nfuncs; ++ii) {
- if (prev_addr == functions[ii].addr) {
- delete[] functions[ii].name;
- continue;
- }
- memcpy(&uniq_functions[next_uniq++], &functions[ii],
- sizeof(symbol_type));
- prev_addr = functions[ii].addr;
- }
- assert(next_uniq == num_uniq);
-
- delete[] functions;
- functions = uniq_functions;
- nfuncs = num_uniq;
- }
-
- // Finally, demangle all of the symbol names
- demangle_names(nfuncs, functions);
-
- uint32_t min_addr = 0;
- if (!zero_found)
- min_addr = functions[1].addr;
- if (region->vstart == 0)
- region->vstart = min_addr;
- region->nsymbols = nfuncs;
- region->symbols = functions;
-
-#if 0
- printf("%s num symbols: %d min_addr: 0x%x\n", region->path, nfuncs, min_addr);
- for (int ii = 0; ii < nfuncs; ++ii) {
- printf("0x%08x %s\n", functions[ii].addr, functions[ii].name);
- }
-#endif
- delete[] elf_symbols;
- delete[] symbol_names;
- delete[] section_names;
- delete[] shdr;
- delete hdr;
- fclose(fobj);
-
- return true;
-}
-
-template<class T>
-void TraceReader<T>::CopyKernelRegion(ProcessState *pstate)
-{
- ProcessState *manager = pstate->addr_manager;
- if (manager->flags & ProcessState::kHasKernelRegion)
- return;
-
- int nregions = processes_[0]->nregions;
- region_type **regions = processes_[0]->regions;
- for (int ii = 0; ii < nregions; ii++) {
- if (regions[ii]->flags & region_type::kIsKernelRegion) {
- AddRegion(manager, regions[ii]);
- regions[ii]->refs += 1;
- }
- }
- manager->flags |= ProcessState::kHasKernelRegion;
-}
-
-template<class T>
-void TraceReader<T>::ClearRegions(ProcessState *pstate)
-{
- assert(pstate->pid != 0);
- int nregions = pstate->nregions;
- region_type **regions = pstate->regions;
-
- // Decrement the reference count on all the regions
- for (int ii = 0; ii < nregions; ii++) {
- if (regions[ii]->refs > 0) {
- regions[ii]->refs -= 1;
- continue;
- }
-
- delete regions[ii];
- }
- delete[] pstate->regions;
- pstate->regions = NULL;
- pstate->nregions = 0;
- pstate->max_regions = 0;
- pstate->addr_manager = pstate;
- pstate->flags &= ~ProcessState::kIsClone;
- pstate->flags &= ~ProcessState::kHasKernelRegion;
- CopyKernelRegion(pstate);
-}
-
-template<class T>
-void TraceReader<T>::AddRegion(ProcessState *pstate, region_type *region)
-{
- ProcessState *manager = pstate->addr_manager;
- if (manager->regions == NULL) {
- manager->max_regions = ProcessState::kInitialNumRegions;
- manager->regions = new region_type*[manager->max_regions];
- manager->nregions = 0;
- }
-
- // Check if we need to grow the array
- int nregions = manager->nregions;
- int max_regions = manager->max_regions;
- if (nregions >= max_regions) {
- max_regions <<= 1;
- manager->max_regions = max_regions;
- region_type **regions = new region_type*[max_regions];
- for (int ii = 0; ii < nregions; ii++) {
- regions[ii] = manager->regions[ii];
- }
- delete[] manager->regions;
- manager->regions = regions;
- }
-
- // Add the new region to the end of the array and resort
- manager->regions[nregions] = region;
- nregions += 1;
- manager->nregions = nregions;
-
- // Resort the regions into increasing start address
- qsort(manager->regions, nregions, sizeof(region_type*), cmp_region_addr<T>);
-}
-
-template<class T>
-void TraceReader<T>::FindAndRemoveRegion(ProcessState *pstate, uint32_t vstart,
- uint32_t vend)
-{
- ProcessState *manager = pstate->addr_manager;
- int nregions = manager->nregions;
- int index = FindRegionIndex(vstart, nregions, manager->regions);
- region_type *region = manager->regions[index];
-
- // If the region does not contain [vstart,vend], then return.
- if (vstart < region->vstart || vend > region->vend)
- return;
-
- // If the existing region exactly matches the address range [vstart,vend]
- // then remove the whole region.
- if (vstart == region->vstart && vend == region->vend) {
- // The regions are reference-counted.
- if (region->refs == 0) {
- // Free the region
- hash_->Remove(region->path);
- delete region;
- } else {
- region->refs -= 1;
- }
-
- if (nregions > 1) {
- // Assign the region at the end of the array to this empty slot
- manager->regions[index] = manager->regions[nregions - 1];
-
- // Resort the regions into increasing start address
- qsort(manager->regions, nregions - 1, sizeof(region_type*),
- cmp_region_addr<T>);
- }
- manager->nregions = nregions - 1;
- return;
- }
-
- // If the existing region contains the given range and ends at the
- // end of the given range (a common case for some reason), then
- // truncate the existing region so that it ends at vstart (because
- // we are deleting the range [vstart,vend]).
- if (vstart > region->vstart && vend == region->vend) {
- region_type *truncated;
-
- if (region->refs == 0) {
- // This region is not shared, so truncate it directly
- truncated = region;
- } else {
- // This region is shared, so make a copy that we can truncate
- region->refs -= 1;
- truncated = region->MakePrivateCopy(new region_type);
- }
- truncated->vend = vstart;
- manager->regions[index] = truncated;
- }
-}
-
-template<class T>
-void TraceReader<T>::CopyRegions(ProcessState *parent, ProcessState *child)
-{
- // Copy the parent's address space
- ProcessState *manager = parent->addr_manager;
- int nregions = manager->nregions;
- child->nregions = nregions;
- child->max_regions = manager->max_regions;
- region_type **regions = new region_type*[manager->max_regions];
- child->regions = regions;
- memcpy(regions, manager->regions, nregions * sizeof(region_type*));
-
- // Increment the reference count on all the regions
- for (int ii = 0; ii < nregions; ii++) {
- regions[ii]->refs += 1;
- }
-}
-
-template<class T>
-void TraceReader<T>::DumpRegions(FILE *stream, ProcessState *pstate) {
- ProcessState *manager = pstate->addr_manager;
- for (int ii = 0; ii < manager->nregions; ++ii) {
- fprintf(stream, " %08x - %08x offset: %5x nsyms: %4d refs: %d %s\n",
- manager->regions[ii]->vstart,
- manager->regions[ii]->vend,
- manager->regions[ii]->file_offset,
- manager->regions[ii]->nsymbols,
- manager->regions[ii]->refs,
- manager->regions[ii]->path);
- }
-}
-
-template<class T>
-typename TraceReader<T>::region_type *
-TraceReader<T>::FindRegion(uint32_t addr, int nregions, region_type **regions)
-{
- int high = nregions;
- int low = -1;
- while (low + 1 < high) {
- int middle = (high + low) / 2;
- uint32_t middle_addr = regions[middle]->vstart;
- if (middle_addr == addr)
- return regions[middle];
- if (middle_addr > addr)
- high = middle;
- else
- low = middle;
- }
-
- // If we get here then we did not find an exact address match. So use
- // the closest region address that is less than the given address.
- if (low < 0)
- low = 0;
- return regions[low];
-}
-
-template<class T>
-int TraceReader<T>::FindRegionIndex(uint32_t addr, int nregions,
- region_type **regions)
-{
- int high = nregions;
- int low = -1;
- while (low + 1 < high) {
- int middle = (high + low) / 2;
- uint32_t middle_addr = regions[middle]->vstart;
- if (middle_addr == addr)
- return middle;
- if (middle_addr > addr)
- high = middle;
- else
- low = middle;
- }
-
- // If we get here then we did not find an exact address match. So use
- // the closest region address that is less than the given address.
- if (low < 0)
- low = 0;
- return low;
-}
-
-template<class T>
-typename TraceReader<T>::symbol_type *
-TraceReader<T>::FindFunction(uint32_t addr, int nsyms, symbol_type *symbols,
- bool exact_match)
-{
- int high = nsyms;
- int low = -1;
- while (low + 1 < high) {
- int middle = (high + low) / 2;
- uint32_t middle_addr = symbols[middle].addr;
- if (middle_addr == addr)
- return &symbols[middle];
- if (middle_addr > addr)
- high = middle;
- else
- low = middle;
- }
-
- // If we get here then we did not find an exact address match. So use
- // the closest function address that is less than the given address.
- // We added a symbol with address zero so if there is no known
- // function containing the given address, then we will return the
- // "(unknown)" symbol.
- if (low >= 0 && !exact_match)
- return &symbols[low];
- return NULL;
-}
-
-template<class T>
-typename TraceReader<T>::symbol_type *
-TraceReader<T>::LookupFunction(int pid, uint32_t addr, uint64_t time)
-{
- // Check if the previous match is still a good match.
- if (cached_pid_ == pid) {
- uint32_t vstart = cached_func_->region->vstart;
- uint32_t vend = cached_func_->region->vend;
- if (addr >= vstart && addr < vend) {
- uint32_t sym_addr = addr - cached_func_->region->base_addr;
- if (sym_addr >= cached_func_->addr
- && sym_addr < (cached_func_ + 1)->addr) {
-
- // Check if there is a Java method on the method trace.
- symbol_type *sym = FindCurrentMethod(pid, time);
- if (sym != NULL) {
- sym->vm_sym = cached_func_;
- return sym;
- }
- return cached_func_;
- }
- }
- }
-
- ProcessState *pstate = processes_[pid];
- if (pstate == NULL) {
- // There is no process state for the specified pid.
- // This should never happen.
- cached_pid_ = -1;
- cached_func_ = NULL;
- return NULL;
- }
- ProcessState *manager = pstate->addr_manager;
- cached_pid_ = pid;
- region_type *region = FindRegion(addr, manager->nregions, manager->regions);
- uint32_t sym_addr = addr - region->base_addr;
-
- cached_func_ = FindFunction(sym_addr, region->nsymbols, region->symbols,
- false /* no exact match */);
- if (cached_func_ != NULL) {
- cached_func_->region = region;
-
- // Check if there is a Java method on the method trace.
- symbol_type *sym = FindCurrentMethod(pid, time);
- if (sym != NULL) {
- sym->vm_sym = cached_func_;
- return sym;
- }
- }
-
- return cached_func_;
-}
-
-template <class T>
-void TraceReader<T>::HandlePidEvent(PidEvent *event)
-{
- switch (event->rec_type) {
- case kPidFork:
- case kPidClone:
- // event->pid is the process id of the child
- if (event->pid >= kNumPids) {
- fprintf(stderr, "Error: pid (%d) too large\n", event->pid);
- exit(1);
- }
- // Create a new ProcessState struct for the child
- // and link it in at the front of the list for that
- // pid.
- {
- ProcessState *child = new ProcessState;
- processes_[event->pid] = child;
- child->pid = event->pid;
- child->tgid = event->tgid;
-
- // Link the new child at the front of the list (only needed if
- // pids wrap around, which will probably never happen when
- // tracing because it would take so long).
- child->next = processes_[event->pid];
- child->parent_pid = current_->pid;
- child->parent = current_;
- child->start_time = event->time;
- child->name = Strdup(current_->name);
- if (event->rec_type == kPidFork) {
- CopyRegions(current_, child);
- } else {
- // Share the parent's address space
- child->flags |= ProcessState::kIsClone;
-
- // The address space manager for the clone is the same
- // as the address space manager for the parent. This works
- // even if the child later clones itself.
- child->addr_manager = current_->addr_manager;
- }
- }
- break;
- case kPidSwitch:
- // event->pid is the process id of the process we are
- // switching to.
- {
- uint64_t elapsed = event->time - function_start_time_;
- function_start_time_ = event->time;
- current_->cpu_time += elapsed;
- }
- if (current_->flags & ProcessState::kCalledExit)
- current_->end_time = event->time;
-
- if (event->pid >= kNumPids) {
- fprintf(stderr, "Error: pid (%d) too large\n", event->pid);
- exit(1);
- }
-
- // If the process we are switching to does not exist, then
- // create one. This can happen because the tracing code does
- // not start tracing from the very beginning of the kernel.
- current_ = processes_[event->pid];
- if (current_ == NULL) {
- current_ = new ProcessState;
- processes_[event->pid] = current_;
- current_->pid = event->pid;
- current_->start_time = event->time;
- CopyKernelRegion(current_);
- }
-#if 0
- {
- printf("switching to p%d\n", current_->pid);
- ProcessState *manager = current_->addr_manager;
- for (int ii = 0; ii < manager->nregions; ++ii) {
- printf(" %08x - %08x offset: %d nsyms: %4d %s\n",
- manager->regions[ii]->vstart,
- manager->regions[ii]->vend,
- manager->regions[ii]->file_offset,
- manager->regions[ii]->nsymbols,
- manager->regions[ii]->path);
- }
- }
-#endif
- break;
- case kPidExit:
- current_->exit_val = event->pid;
- current_->flags |= ProcessState::kCalledExit;
- break;
- case kPidMunmap:
- FindAndRemoveRegion(current_, event->vstart, event->vend);
- break;
- case kPidMmap:
- {
- region_type *region;
- region_type *existing_region = hash_->Find(event->path);
- if (existing_region == NULL
- || existing_region->vstart != event->vstart
- || existing_region->vend != event->vend
- || existing_region->file_offset != event->offset) {
- // Create a new region and add it to the current process'
- // address space.
- region = new region_type;
-
- // The event->path is allocated by ReadPidEvent() and owned
- // by us.
- region->path = event->path;
- region->vstart = event->vstart;
- region->vend = event->vend;
- region->file_offset = event->offset;
- if (existing_region == NULL) {
- DexFileList *dexfile = dex_hash_->Find(event->path);
- if (dexfile != NULL) {
- PopulateSymbolsFromDexFile(dexfile, region);
- } else {
- ReadElfSymbols(region, 0);
- }
- hash_->Update(region->path, region);
- } else {
- region->nsymbols = existing_region->nsymbols;
- region->symbols = existing_region->symbols;
- region->flags |= region_type::kSharedSymbols;
- }
-
- // The base_addr is subtracted from an address before the
- // symbol name lookup and is either zero or event->vstart.
- // HACK: Determine if base_addr is non-zero by looking at the
- // second symbol address (skip the first symbol because that is
- // the special symbol "(unknown)" with an address of zero).
- if (region->nsymbols > 2 && region->symbols[1].addr < event->vstart)
- region->base_addr = event->vstart;
-
- // Treat all mmapped regions after the first as "libraries".
- // Profiling tools can test for this property.
- if (current_->flags & ProcessState::kHasFirstMmap)
- region->flags |= region_type::kIsLibraryRegion;
- else
- current_->flags |= ProcessState::kHasFirstMmap;
-#if 0
- printf("%s vstart: 0x%x vend: 0x%x offset: 0x%x\n",
- region->path, region->vstart, region->vend, region->file_offset);
-#endif
- } else {
- region = existing_region;
- region->refs += 1;
- delete[] event->path;
- }
- AddRegion(current_, region);
- }
- break;
- case kPidExec:
- if (current_->argc > 0) {
- for (int ii = 0; ii < current_->argc; ii++) {
- delete[] current_->argv[ii];
- }
- delete[] current_->argv;
- }
- delete[] current_->name;
-
- current_->argc = event->argc;
- current_->argv = event->argv;
- current_->name = Strdup(current_->argv[0]);
- current_->flags |= ProcessState::kCalledExec;
- ClearRegions(current_);
- break;
- case kPidName:
- case kPidKthreadName:
- {
- ProcessState *pstate = processes_[event->pid];
- if (pstate == NULL) {
- pstate = new ProcessState;
- if (event->rec_type == kPidKthreadName) {
- pstate->tgid = event->tgid;
- }
- pstate->pid = event->pid;
- pstate->start_time = event->time;
- processes_[event->pid] = pstate;
- CopyKernelRegion(pstate);
- } else {
- delete[] pstate->name;
- }
- pstate->name = event->path;
- }
- break;
- case kPidNoAction:
- break;
- case kPidSymbolAdd:
- delete[] event->path;
- break;
- case kPidSymbolRemove:
- break;
- }
-}
-
-// Finds the current pid for the given time. This routine reads the pid
-// trace file and assumes that the "time" parameter is monotonically
-// increasing.
-template <class T>
-int TraceReader<T>::FindCurrentPid(uint64_t time)
-{
- if (time < next_pid_event_.time)
- return current_->pid;
-
- while (1) {
- HandlePidEvent(&next_pid_event_);
-
- if (internal_pid_reader_->ReadPidEvent(&next_pid_event_)) {
- next_pid_event_.time = ~0ull;
- break;
- }
- if (next_pid_event_.time > time)
- break;
- }
- return current_->pid;
-}
-
-template <class T>
-void TraceReader<T>::ProcessState::DumpStack(FILE *stream)
-{
- const char *native;
- for (int ii = 0; ii < method_stack_top; ii++) {
- native = method_stack[ii].isNative ? "n" : " ";
- fprintf(stream, "%2d: %s 0x%08x\n", ii, native, method_stack[ii].addr);
- }
-}
-
-template <class T>
-void TraceReader<T>::HandleMethodRecord(ProcessState *pstate,
- MethodRec *method_rec)
-{
- uint32_t addr;
- int top = pstate->method_stack_top;
- int flags = method_rec->flags;
- bool isNative;
- if (flags == kMethodEnter || flags == kNativeEnter) {
- // Push this method on the stack
- if (top >= pstate->kMaxMethodStackSize) {
- fprintf(stderr, "Stack overflow at time %llu\n", method_rec->time);
- exit(1);
- }
- pstate->method_stack[top].addr = method_rec->addr;
- isNative = (flags == kNativeEnter);
- pstate->method_stack[top].isNative = isNative;
- pstate->method_stack_top = top + 1;
- addr = method_rec->addr;
- } else {
- if (top <= 0) {
- // If the stack underflows, then set the current method to NULL.
- pstate->current_method_sym = NULL;
- return;
- }
- top -= 1;
- addr = pstate->method_stack[top].addr;
-
- // If this is a non-native method then the address we are popping should
- // match the top-of-stack address. Native pops don't always match the
- // address of the native push for some reason.
- if (addr != method_rec->addr && !pstate->method_stack[top].isNative) {
- fprintf(stderr,
- "Stack method (0x%x) at index %d does not match trace record (0x%x) at time %llu\n",
- addr, top, method_rec->addr, method_rec->time);
- pstate->DumpStack(stderr);
- exit(1);
- }
-
- // If we are popping a native method, then the top-of-stack should also
- // be a native method.
- bool poppingNative = (flags == kNativeExit) || (flags == kNativeException);
- if (poppingNative != pstate->method_stack[top].isNative) {
- fprintf(stderr,
- "Popping native vs. non-native mismatch at index %d time %llu\n",
- top, method_rec->time);
- pstate->DumpStack(stderr);
- exit(1);
- }
-
- pstate->method_stack_top = top;
- if (top == 0) {
- // When we empty the stack, set the current method to NULL
- pstate->current_method_sym = NULL;
- return;
- }
- addr = pstate->method_stack[top - 1].addr;
- isNative = pstate->method_stack[top - 1].isNative;
- }
-
- // If the top-of-stack is a native method, then set the current method
- // to NULL.
- if (isNative) {
- pstate->current_method_sym = NULL;
- return;
- }
-
- ProcessState *manager = pstate->addr_manager;
- region_type *region = FindRegion(addr, manager->nregions, manager->regions);
- uint32_t sym_addr = addr - region->base_addr;
- symbol_type *sym = FindFunction(sym_addr, region->nsymbols,
- region->symbols, true /* exact match */);
-
- pstate->current_method_sym = sym;
- if (sym != NULL) {
- sym->region = region;
- }
-}
-
-// Returns the current top-of-stack Java method, if any, for the given pid
-// at the given time. The "time" parameter must be monotonically increasing
-// across successive calls to this method.
-// If the Java method stack is empty or if a native JNI method is on the
-// top of the stack, then this method returns NULL.
-template <class T>
-typename TraceReader<T>::symbol_type*
-TraceReader<T>::FindCurrentMethod(int pid, uint64_t time)
-{
- ProcessState *procState = processes_[pid];
-
- if (time < next_method_.time) {
- return procState->current_method_sym;
- }
-
- while (1) {
- if (next_method_.time != 0) {
- // We may have to process methods from a different pid so use
- // a local variable here so that we don't overwrite procState.
- ProcessState *pState = processes_[next_method_.pid];
- HandleMethodRecord(pState, &next_method_);
- }
-
- if (internal_method_reader_->ReadMethod(&next_method_)) {
- next_method_.time = ~0ull;
- break;
- }
- if (next_method_.time > time)
- break;
- }
- return procState->current_method_sym;
-}
-
-template <class T>
-void TraceReader<T>::PopulateSymbolsFromDexFile(const DexFileList *dexfile,
- region_type *region)
-
-{
- int nsymbols = dexfile->nsymbols;
- DexSym *dexsyms = dexfile->symbols;
- region->nsymbols = nsymbols + 1;
- symbol_type *symbols = new symbol_type[nsymbols + 1];
- memset(symbols, 0, (nsymbols + 1) * sizeof(symbol_type));
- region->symbols = symbols;
- for (int ii = 0; ii < nsymbols; ii++) {
- symbols[ii].addr = dexsyms[ii].addr;
- symbols[ii].name = Strdup(dexsyms[ii].name);
- symbols[ii].vm_sym = NULL;
- symbols[ii].region = region;
- symbols[ii].flags = symbol_type::kIsMethod;
- }
-
- // Add an entry at the end with an address of 0xffffffff. This
- // is required for LookupFunction() to work.
- symbol_type *symbol = &symbols[nsymbols];
- symbol->addr = 0xffffffff;
- symbol->name = Strdup("(end)");
- symbol->vm_sym = NULL;
- symbol->region = region;
- symbol->flags = symbol_type::kIsMethod;
-}
-
-template <class T>
-bool TraceReader<T>::ReadMethodSymbol(MethodRec *method_record,
- symbol_type **psym,
- ProcessState **pproc)
-{
- if (internal_method_reader_->ReadMethod(&next_method_)) {
- return true;
- }
-
- // Copy the whole MethodRec struct
- *method_record = next_method_;
-
- uint64_t time = next_method_.time;
-
- // Read the pid trace file up to this point to make sure the
- // process state is valid.
- FindCurrentPid(time);
-
- ProcessState *pstate = processes_[next_method_.pid];
- *pproc = pstate;
- HandleMethodRecord(pstate, &next_method_);
- *psym = pstate->current_method_sym;
- return false;
-}
-
-#endif /* TRACE_READER_H */
diff --git a/emulator/qtools/trace_reader_base.h b/emulator/qtools/trace_reader_base.h
deleted file mode 100644
index 416c3d1..0000000
--- a/emulator/qtools/trace_reader_base.h
+++ /dev/null
@@ -1,332 +0,0 @@
-// Copyright 2006 The Android Open Source Project
-
-#ifndef TRACE_READER_BASE_H
-#define TRACE_READER_BASE_H
-
-#include <inttypes.h>
-#include "trace_common.h"
-#include "hash_table.h"
-
-class BBReader;
-class InsnReader;
-class AddrReader;
-class ExcReader;
-class PidReader;
-class MethodReader;
-
-struct StaticRec {
- uint64_t bb_num;
- uint32_t bb_addr;
- uint32_t num_insns;
-};
-
-struct StaticBlock {
- StaticRec rec;
- uint32_t *insns;
-};
-
-struct BBEvent {
- uint64_t time;
- uint64_t bb_num;
- uint32_t bb_addr;
- uint32_t *insns;
- int num_insns;
- int pid;
- int is_thumb;
-};
-
-struct PidEvent {
- uint64_t time;
- int rec_type; // record type: fork, context switch, exit ...
- int tgid; // thread group id
- int pid; // for fork: child pid; for switch: next pid;
- // for exit: exit value
- uint32_t vstart; // virtual start address (only used with mmap)
- uint32_t vend; // virtual end address (only used with mmap)
- uint32_t offset; // virtual file offset (only used with mmap)
-
- // Dynamically allocated path to executable (or lib). In the case of
- // an mmapped dex file, the path is modified to be more useful for
- // comparing against the output of dexlist. For example, instead of this:
- // /data/dalvik-cache/system@app@TestHarness.apk@classes.dex
- // We convert to this:
- // /system/app/TestHarness.apk
- char *path;
- char *mmap_path; // unmodified mmap path
- int argc; // number of args
- char **argv; // dynamically allocated array of args
-};
-
-struct MethodRec {
- uint64_t time;
- uint32_t addr;
- int pid;
- int flags;
-};
-
-struct DexSym {
- uint32_t addr;
- int len;
- char *name;
-};
-
-struct DexFileList {
- char *path;
- int nsymbols;
- DexSym *symbols;
-};
-
-class TraceReaderBase {
- public:
- TraceReaderBase();
- virtual ~TraceReaderBase();
-
- friend class BBReader;
-
- void Open(const char *filename);
- void Close();
- void WriteHeader(TraceHeader *header);
- inline bool ReadBB(BBEvent *event);
- int ReadStatic(StaticRec *rec);
- int ReadStaticInsns(int num, uint32_t *insns);
- TraceHeader *GetHeader() { return header_; }
- inline uint64_t ReadInsnTime(uint64_t min_time);
- void TruncateLastBlock(uint32_t num_insns);
- inline bool ReadAddr(uint64_t *time, uint32_t *addr, int *flags);
- inline bool ReadExc(uint64_t *time, uint32_t *current_pc,
- uint64_t *recnum, uint32_t *target_pc,
- uint64_t *bb_num, uint64_t *bb_start_time,
- int *num_insns);
- inline bool ReadPidEvent(PidEvent *event);
- inline bool ReadMethod(MethodRec *method_record);
- StaticBlock *GetStaticBlock(uint64_t bb_num) { return &blocks_[bb_num]; }
- uint32_t *GetInsns(uint64_t bb_num) { return blocks_[bb_num].insns; }
- uint32_t GetBBAddr(uint64_t bb_num) {
- return blocks_[bb_num].rec.bb_addr & ~1;
- }
- int GetIsThumb(uint64_t bb_num) {
- return blocks_[bb_num].rec.bb_addr & 1;
- }
- void SetPostProcessing(bool val) { post_processing_ = val; }
-
- protected:
- virtual int FindCurrentPid(uint64_t time);
- int current_pid_;
- int next_pid_;
- uint64_t next_pid_switch_time_;
- PidReader *internal_pid_reader_;
- MethodReader *internal_method_reader_;
- HashTable<DexFileList*> *dex_hash_;
-
- private:
- int FindNumInsns(uint64_t bb_num, uint64_t bb_start_time);
- void ReadTraceHeader(FILE *fstream, const char *filename,
- const char *tracename, TraceHeader *header);
- PidEvent *FindMmapDexFileEvent();
- void ParseDexList(const char *filename);
-
- char *static_filename_;
- FILE *static_fstream_;
- TraceHeader *header_;
- BBReader *bb_reader_;
- InsnReader *insn_reader_;
- AddrReader *load_addr_reader_;
- AddrReader *store_addr_reader_;
- ExcReader *exc_reader_;
- PidReader *pid_reader_;
- MethodReader *method_reader_;
- ExcReader *internal_exc_reader_;
- StaticBlock *blocks_;
- bool exc_end_;
- uint64_t bb_recnum_;
- uint64_t exc_recnum_;
- uint64_t exc_bb_num_;
- uint64_t exc_time_;
- int exc_num_insns_;
- bool post_processing_;
-
- bool load_eof_;
- uint64_t load_time_;
- uint32_t load_addr_;
- bool store_eof_;
- uint64_t store_time_;
- uint32_t store_addr_;
-};
-
-class Decoder;
-
-class BBReader {
- public:
- explicit BBReader(TraceReaderBase *trace);
- ~BBReader();
- void Open(const char *filename);
- void Close();
- bool ReadBB(BBEvent *event);
-
- private:
- struct TimeRec {
- BBRec bb_rec;
- uint64_t next_time;
- };
-
- struct Future {
- Future *next;
- TimeRec bb;
- };
-
- inline Future *AllocFuture();
- inline void FreeFuture(Future *future);
- inline void InsertFuture(Future *future);
- inline int DecodeNextRec();
-
- TimeRec nextrec_;
- Future futures_[kMaxNumBasicBlocks];
- Future *head_;
- Future *free_;
- Decoder *decoder_;
- bool is_eof_;
- TraceReaderBase *trace_;
-};
-
-class InsnReader {
- public:
- InsnReader();
- ~InsnReader();
-
- void Open(const char *filename);
- void Close();
- uint64_t ReadInsnTime(uint64_t min_time);
-
- private:
- Decoder *decoder_;
- uint64_t prev_time_;
- uint64_t time_diff_;
- int repeat_;
-};
-
-class AddrReader {
- public:
- AddrReader();
- ~AddrReader();
-
- bool Open(const char *filename, const char *suffix);
- void Close();
- bool ReadAddr(uint64_t *time, uint32_t *addr);
-
- private:
- Decoder *decoder_;
- uint32_t prev_addr_;
- uint64_t prev_time_;
- bool opened_; // true after file is opened
-};
-
-class ExcReader {
- public:
- ExcReader();
- ~ExcReader();
-
- void Open(const char *filename);
- void Close();
- bool ReadExc(uint64_t *time, uint32_t *current_pc,
- uint64_t *recnum, uint32_t *target_pc,
- uint64_t *bb_num, uint64_t *bb_start_time,
- int *num_insns);
-
- private:
- Decoder *decoder_;
- uint64_t prev_time_;
- uint64_t prev_recnum_;
-};
-
-class PidReader {
- public:
- PidReader();
- ~PidReader();
-
- void Open(const char *filename);
- void Close();
- bool ReadPidEvent(struct PidEvent *event);
- void Dispose(struct PidEvent *event);
-
- private:
- Decoder *decoder_;
- uint64_t prev_time_;
-};
-
-class MethodReader {
- public:
- MethodReader();
- ~MethodReader();
-
- bool Open(const char *filename);
- void Close();
- bool ReadMethod(MethodRec *method_record);
-
- private:
- Decoder *decoder_;
- uint64_t prev_time_;
- uint32_t prev_addr_;
- int32_t prev_pid_;
- bool opened_; // true after file is opened
-};
-
-// Reads the next dynamic basic block from the trace.
-// Returns true on end-of-file.
-inline bool TraceReaderBase::ReadBB(BBEvent *event)
-{
- bb_recnum_ += 1;
- return bb_reader_->ReadBB(event);
-}
-
-inline uint64_t TraceReaderBase::ReadInsnTime(uint64_t min_time)
-{
- return insn_reader_->ReadInsnTime(min_time);
-}
-
-inline bool TraceReaderBase::ReadAddr(uint64_t *time, uint32_t *addr, int *flags)
-{
- if (load_eof_ && store_eof_)
- return true;
-
- if (store_eof_ || (!load_eof_ && load_time_ <= store_time_)) {
- *time = load_time_;
- *addr = load_addr_;
- *flags = 0;
- load_eof_ = load_addr_reader_->ReadAddr(&load_time_, &load_addr_);
- } else {
- *time = store_time_;
- *addr = store_addr_;
- *flags = 1;
- store_eof_ = store_addr_reader_->ReadAddr(&store_time_, &store_addr_);
- }
- return false;
-}
-
-inline bool TraceReaderBase::ReadExc(uint64_t *time, uint32_t *current_pc,
- uint64_t *recnum, uint32_t *target_pc,
- uint64_t *bb_num, uint64_t *bb_start_time,
- int *num_insns)
-{
- return exc_reader_->ReadExc(time, current_pc, recnum, target_pc, bb_num,
- bb_start_time, num_insns);
-}
-
-inline bool TraceReaderBase::ReadPidEvent(PidEvent *event)
-{
- return pid_reader_->ReadPidEvent(event);
-}
-
-inline bool TraceReaderBase::ReadMethod(MethodRec *method_record)
-{
- return method_reader_->ReadMethod(method_record);
-}
-
-// Duplicates a string, allocating space using new[].
-inline char * Strdup(const char *src) {
- int len = strlen(src);
- char *copy = new char[len + 1];
- strcpy(copy, src);
- return copy;
-}
-
-#endif /* TRACE_READER_BASE_H */