Index: gdb/arm-tdep.c |
diff --git a/gdb/arm-tdep.c b/gdb/arm-tdep.c |
index 18204514a05a7c147a6e8fac2ea78949f34ff65d..568ace59396b53c3dacdc26e3fcf9628bbe6dfd0 100644 |
--- a/gdb/arm-tdep.c |
+++ b/gdb/arm-tdep.c |
@@ -40,6 +40,7 @@ |
#include "dwarf2-frame.h" |
#include "gdbtypes.h" |
#include "prologue-value.h" |
+#include "remote.h" |
#include "target-descriptions.h" |
#include "user-regs.h" |
#include "observer.h" |
@@ -54,7 +55,11 @@ |
#include "gdb_assert.h" |
#include "vec.h" |
+#include "record.h" |
+ |
#include "features/arm-with-m.c" |
+#include "features/arm-with-m-fpa-layout.c" |
+#include "features/arm-with-m-vfp-d16.c" |
#include "features/arm-with-iwmmxt.c" |
#include "features/arm-with-vfpv2.c" |
#include "features/arm-with-vfpv3.c" |
@@ -97,7 +102,7 @@ static struct cmd_list_element *showarmcmdlist = NULL; |
/* The type of floating-point to use. Keep this in sync with enum |
arm_float_model, and the help string in _initialize_arm_tdep. */ |
-static const char *fp_model_strings[] = |
+static const char *const fp_model_strings[] = |
{ |
"auto", |
"softfpa", |
@@ -112,7 +117,7 @@ static enum arm_float_model arm_fp_model = ARM_FLOAT_AUTO; |
static const char *current_fp_model = "auto"; |
/* The ABI to use. Keep this in sync with arm_abi_kind. */ |
-static const char *arm_abi_strings[] = |
+static const char *const arm_abi_strings[] = |
{ |
"auto", |
"APCS", |
@@ -125,7 +130,7 @@ static enum arm_abi_kind arm_abi_global = ARM_ABI_AUTO; |
static const char *arm_abi_string = "auto"; |
/* The execution mode to assume. */ |
-static const char *arm_mode_strings[] = |
+static const char *const arm_mode_strings[] = |
{ |
"auto", |
"arm", |
@@ -375,7 +380,6 @@ arm_find_mapping_symbol (CORE_ADDR memaddr, CORE_ADDR *start) |
int |
arm_pc_is_thumb (struct gdbarch *gdbarch, CORE_ADDR memaddr) |
{ |
- struct obj_section *sec; |
struct minimal_symbol *sym; |
char type; |
struct displaced_step_closure* dsc |
@@ -1280,7 +1284,7 @@ static CORE_ADDR |
arm_skip_stack_protector(CORE_ADDR pc, struct gdbarch *gdbarch) |
{ |
enum bfd_endian byte_order_for_code = gdbarch_byte_order_for_code (gdbarch); |
- unsigned int address, basereg; |
+ unsigned int basereg; |
struct minimal_symbol *stack_chk_guard; |
int offset; |
int is_thumb = arm_pc_is_thumb (gdbarch, pc); |
@@ -1372,7 +1376,6 @@ arm_skip_prologue (struct gdbarch *gdbarch, CORE_ADDR pc) |
unsigned long inst; |
CORE_ADDR skip_pc; |
CORE_ADDR func_addr, limit_pc; |
- struct symtab_and_line sal; |
/* See if we can determine the end of the prologue via the symbol table. |
If so, then return either PC, or the PC after the prologue, whichever |
@@ -1531,7 +1534,6 @@ thumb_scan_prologue (struct gdbarch *gdbarch, CORE_ADDR prev_pc, |
{ |
CORE_ADDR prologue_start; |
CORE_ADDR prologue_end; |
- CORE_ADDR current_pc; |
if (find_pc_partial_function (block_addr, NULL, &prologue_start, |
&prologue_end)) |
@@ -3215,6 +3217,9 @@ arm_in_function_epilogue_p (struct gdbarch *gdbarch, CORE_ADDR pc) |
else if ((insn & 0x0fff0000) == 0x08bd0000) |
/* POP (LDMIA). */ |
found_stack_adjust = 1; |
+ else if ((insn & 0x0fff0000) == 0x049d0000) |
+ /* POP of a single register. */ |
+ found_stack_adjust = 1; |
} |
if (found_stack_adjust) |
@@ -5127,7 +5132,7 @@ static gdb_byte * |
extend_buffer_earlier (gdb_byte *buf, CORE_ADDR endaddr, |
int old_len, int new_len) |
{ |
- gdb_byte *new_buf, *middle; |
+ gdb_byte *new_buf; |
int bytes_to_read = new_len - old_len; |
new_buf = xmalloc (new_len); |
@@ -5160,7 +5165,7 @@ arm_adjust_breakpoint_address (struct gdbarch *gdbarch, CORE_ADDR bpaddr) |
gdb_byte *buf; |
char map_type; |
CORE_ADDR boundary, func_start; |
- int buf_len, buf2_len; |
+ int buf_len; |
enum bfd_endian order = gdbarch_byte_order_for_code (gdbarch); |
int i, any, last_it, last_it_count; |
@@ -6860,7 +6865,7 @@ cleanup_block_load_pc (struct gdbarch *gdbarch, |
struct displaced_step_closure *dsc) |
{ |
uint32_t status = displaced_read_reg (regs, dsc, ARM_PS_REGNUM); |
- int load_executed = condition_true (dsc->u.block.cond, status), i; |
+ int load_executed = condition_true (dsc->u.block.cond, status); |
unsigned int mask = dsc->u.block.regmask, write_reg = ARM_PC_REGNUM; |
unsigned int regs_loaded = bitcount (mask); |
unsigned int num_to_shuffle = regs_loaded, clobbered; |
@@ -8702,8 +8707,6 @@ static void |
arm_remote_breakpoint_from_pc (struct gdbarch *gdbarch, CORE_ADDR *pcptr, |
int *kindptr) |
{ |
- struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch); |
- |
arm_breakpoint_from_pc (gdbarch, pcptr, kindptr); |
if (arm_pc_is_thumb (gdbarch, *pcptr) && *kindptr == 4) |
@@ -9006,11 +9009,12 @@ arm_store_return_value (struct type *type, struct regcache *regs, |
/* Handle function return values. */ |
static enum return_value_convention |
-arm_return_value (struct gdbarch *gdbarch, struct type *func_type, |
+arm_return_value (struct gdbarch *gdbarch, struct value *function, |
struct type *valtype, struct regcache *regcache, |
gdb_byte *readbuf, const gdb_byte *writebuf) |
{ |
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch); |
+ struct type *func_type = function ? value_type (function) : NULL; |
enum arm_vfp_cprc_base_type vfp_base_type; |
int vfp_base_count; |
@@ -9101,7 +9105,7 @@ arm_get_longjmp_target (struct frame_info *frame, CORE_ADDR *pc) |
CORE_ADDR |
arm_skip_stub (struct frame_info *frame, CORE_ADDR pc) |
{ |
- char *name; |
+ const char *name; |
int namelen; |
CORE_ADDR start_addr; |
@@ -9277,8 +9281,6 @@ static void |
arm_show_fallback_mode (struct ui_file *file, int from_tty, |
struct cmd_list_element *c, const char *value) |
{ |
- struct gdbarch_tdep *tdep = gdbarch_tdep (target_gdbarch); |
- |
fprintf_filtered (file, |
_("The current execution mode assumed " |
"(when symbols are unavailable) is \"%s\".\n"), |
@@ -9568,7 +9570,6 @@ arm_neon_quad_write (struct gdbarch *gdbarch, struct regcache *regcache, |
int regnum, const gdb_byte *buf) |
{ |
char name_buf[4]; |
- gdb_byte reg_buf[8]; |
int offset, double_regnum; |
sprintf (name_buf, "d%d", regnum << 1); |
@@ -9665,6 +9666,49 @@ arm_register_reggroup_p (struct gdbarch *gdbarch, int regnum, |
} |
+/* For backward-compatibility we allow two 'g' packet lengths with |
+ the remote protocol depending on whether FPA registers are |
+ supplied. M-profile targets do not have FPA registers, but some |
+ stubs already exist in the wild which use a 'g' packet which |
+ supplies them albeit with dummy values. The packet format which |
+ includes FPA registers should be considered deprecated for |
+ M-profile targets. */ |
+ |
+static void |
+arm_register_g_packet_guesses (struct gdbarch *gdbarch) |
+{ |
+ if (gdbarch_tdep (gdbarch)->is_m) |
+ { |
+ /* If we know from the executable this is an M-profile target, |
+ cater for remote targets whose register set layout is the |
+ same as the FPA layout. */ |
+ register_remote_g_packet_guess (gdbarch, |
+ /* r0-r12,sp,lr,pc; f0-f7; fps,xpsr */ |
+ (16 * INT_REGISTER_SIZE) |
+ + (8 * FP_REGISTER_SIZE) |
+ + (2 * INT_REGISTER_SIZE), |
+ tdesc_arm_with_m_fpa_layout); |
+ |
+ /* The regular M-profile layout. */ |
+ register_remote_g_packet_guess (gdbarch, |
+ /* r0-r12,sp,lr,pc; xpsr */ |
+ (16 * INT_REGISTER_SIZE) |
+ + INT_REGISTER_SIZE, |
+ tdesc_arm_with_m); |
+ |
+ /* M-profile plus M4F VFP. */ |
+ register_remote_g_packet_guess (gdbarch, |
+ /* r0-r12,sp,lr,pc; d0-d15; fpscr,xpsr */ |
+ (16 * INT_REGISTER_SIZE) |
+ + (16 * VFP_REGISTER_SIZE) |
+ + (2 * INT_REGISTER_SIZE), |
+ tdesc_arm_with_m_vfp_d16); |
+ } |
+ |
+ /* Otherwise we don't have a useful guess. */ |
+} |
+ |
+ |
/* Initialize the current architecture based on INFO. If possible, |
re-use an architecture from ARCHES, which is a list of |
architectures already created during this debugging session. |
@@ -9798,7 +9842,7 @@ arm_gdbarch_init (struct gdbarch_info info, struct gdbarch_list *arches) |
&& (attr_arch == TAG_CPU_ARCH_V6_M |
|| attr_arch == TAG_CPU_ARCH_V6S_M |
|| attr_profile == 'M')) |
- tdesc = tdesc_arm_with_m; |
+ is_m = 1; |
#endif |
} |
@@ -10055,6 +10099,8 @@ arm_gdbarch_init (struct gdbarch_info info, struct gdbarch_list *arches) |
tdep->have_neon_pseudos = have_neon_pseudos; |
tdep->have_neon = have_neon; |
+ arm_register_g_packet_guesses (gdbarch); |
+ |
/* Breakpoints. */ |
switch (info.byte_order_for_code) |
{ |
@@ -10291,6 +10337,8 @@ _initialize_arm_tdep (void) |
/* Initialize the standard target descriptions. */ |
initialize_tdesc_arm_with_m (); |
+ initialize_tdesc_arm_with_m_fpa_layout (); |
+ initialize_tdesc_arm_with_m_vfp_d16 (); |
initialize_tdesc_arm_with_iwmmxt (); |
initialize_tdesc_arm_with_vfpv2 (); |
initialize_tdesc_arm_with_vfpv3 (); |
@@ -10404,3 +10452,2062 @@ vfp - VFP co-processor."), |
NULL, /* FIXME: i18n: "ARM debugging is %s. */ |
&setdebuglist, &showdebuglist); |
} |
+ |
+/* ARM-reversible process record data structures. */ |
+ |
+#define ARM_INSN_SIZE_BYTES 4 |
+#define THUMB_INSN_SIZE_BYTES 2 |
+#define THUMB2_INSN_SIZE_BYTES 4 |
+ |
+ |
+#define INSN_S_L_BIT_NUM 20 |
+ |
+#define REG_ALLOC(REGS, LENGTH, RECORD_BUF) \ |
+ do \ |
+ { \ |
+ unsigned int reg_len = LENGTH; \ |
+ if (reg_len) \ |
+ { \ |
+ REGS = XNEWVEC (uint32_t, reg_len); \ |
+ memcpy(®S[0], &RECORD_BUF[0], sizeof(uint32_t)*LENGTH); \ |
+ } \ |
+ } \ |
+ while (0) |
+ |
+#define MEM_ALLOC(MEMS, LENGTH, RECORD_BUF) \ |
+ do \ |
+ { \ |
+ unsigned int mem_len = LENGTH; \ |
+ if (mem_len) \ |
+ { \ |
+ MEMS = XNEWVEC (struct arm_mem_r, mem_len); \ |
+ memcpy(&MEMS->len, &RECORD_BUF[0], \ |
+ sizeof(struct arm_mem_r) * LENGTH); \ |
+ } \ |
+ } \ |
+ while (0) |
+ |
+/* Checks whether insn is already recorded or yet to be decoded. (boolean expression). */ |
+#define INSN_RECORDED(ARM_RECORD) \ |
+ (0 != (ARM_RECORD)->reg_rec_count || 0 != (ARM_RECORD)->mem_rec_count) |
+ |
+/* ARM memory record structure. */ |
+struct arm_mem_r |
+{ |
+ uint32_t len; /* Record length. */ |
+ CORE_ADDR addr; /* Memory address. */ |
+}; |
+ |
+/* ARM instruction record contains opcode of current insn |
+ and execution state (before entry to decode_insn()), |
+ contains list of to-be-modified registers and |
+ memory blocks (on return from decode_insn()). */ |
+ |
+typedef struct insn_decode_record_t |
+{ |
+ struct gdbarch *gdbarch; |
+ struct regcache *regcache; |
+ CORE_ADDR this_addr; /* Address of the insn being decoded. */ |
+ uint32_t arm_insn; /* Should accommodate thumb. */ |
+ uint32_t cond; /* Condition code. */ |
+ uint32_t opcode; /* Insn opcode. */ |
+ uint32_t decode; /* Insn decode bits. */ |
+ uint32_t mem_rec_count; /* No of mem records. */ |
+ uint32_t reg_rec_count; /* No of reg records. */ |
+ uint32_t *arm_regs; /* Registers to be saved for this record. */ |
+ struct arm_mem_r *arm_mems; /* Memory to be saved for this record. */ |
+} insn_decode_record; |
+ |
+ |
+/* Checks ARM SBZ and SBO mandatory fields. */ |
+ |
+static int |
+sbo_sbz (uint32_t insn, uint32_t bit_num, uint32_t len, uint32_t sbo) |
+{ |
+ uint32_t ones = bits (insn, bit_num - 1, (bit_num -1) + (len - 1)); |
+ |
+ if (!len) |
+ return 1; |
+ |
+ if (!sbo) |
+ ones = ~ones; |
+ |
+ while (ones) |
+ { |
+ if (!(ones & sbo)) |
+ { |
+ return 0; |
+ } |
+ ones = ones >> 1; |
+ } |
+ return 1; |
+} |
+ |
+typedef enum |
+{ |
+ ARM_RECORD_STRH=1, |
+ ARM_RECORD_STRD |
+} arm_record_strx_t; |
+ |
+typedef enum |
+{ |
+ ARM_RECORD=1, |
+ THUMB_RECORD, |
+ THUMB2_RECORD |
+} record_type_t; |
+ |
+ |
+static int |
+arm_record_strx (insn_decode_record *arm_insn_r, uint32_t *record_buf, |
+ uint32_t *record_buf_mem, arm_record_strx_t str_type) |
+{ |
+ |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ ULONGEST u_regval[2]= {0}; |
+ |
+ uint32_t reg_src1 = 0, reg_src2 = 0; |
+ uint32_t immed_high = 0, immed_low = 0,offset_8 = 0, tgt_mem_addr = 0; |
+ uint32_t opcode1 = 0; |
+ |
+ arm_insn_r->opcode = bits (arm_insn_r->arm_insn, 21, 24); |
+ arm_insn_r->decode = bits (arm_insn_r->arm_insn, 4, 7); |
+ opcode1 = bits (arm_insn_r->arm_insn, 20, 24); |
+ |
+ |
+ if (14 == arm_insn_r->opcode || 10 == arm_insn_r->opcode) |
+ { |
+ /* 1) Handle misc store, immediate offset. */ |
+ immed_low = bits (arm_insn_r->arm_insn, 0, 3); |
+ immed_high = bits (arm_insn_r->arm_insn, 8, 11); |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, |
+ &u_regval[0]); |
+ if (ARM_PC_REGNUM == reg_src1) |
+ { |
+ /* If R15 was used as Rn, hence current PC+8. */ |
+ u_regval[0] = u_regval[0] + 8; |
+ } |
+ offset_8 = (immed_high << 4) | immed_low; |
+ /* Calculate target store address. */ |
+ if (14 == arm_insn_r->opcode) |
+ { |
+ tgt_mem_addr = u_regval[0] + offset_8; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval[0] - offset_8; |
+ } |
+ if (ARM_RECORD_STRH == str_type) |
+ { |
+ record_buf_mem[0] = 2; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ } |
+ else if (ARM_RECORD_STRD == str_type) |
+ { |
+ record_buf_mem[0] = 4; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ record_buf_mem[2] = 4; |
+ record_buf_mem[3] = tgt_mem_addr + 4; |
+ arm_insn_r->mem_rec_count = 2; |
+ } |
+ } |
+ else if (12 == arm_insn_r->opcode || 8 == arm_insn_r->opcode) |
+ { |
+ /* 2) Store, register offset. */ |
+ /* Get Rm. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 0, 3); |
+ /* Get Rn. */ |
+ reg_src2 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ regcache_raw_read_unsigned (reg_cache, reg_src2, &u_regval[1]); |
+ if (15 == reg_src2) |
+ { |
+ /* If R15 was used as Rn, hence current PC+8. */ |
+ u_regval[0] = u_regval[0] + 8; |
+ } |
+ /* Calculate target store address, Rn +/- Rm, register offset. */ |
+ if (12 == arm_insn_r->opcode) |
+ { |
+ tgt_mem_addr = u_regval[0] + u_regval[1]; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval[1] - u_regval[0]; |
+ } |
+ if (ARM_RECORD_STRH == str_type) |
+ { |
+ record_buf_mem[0] = 2; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ } |
+ else if (ARM_RECORD_STRD == str_type) |
+ { |
+ record_buf_mem[0] = 4; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ record_buf_mem[2] = 4; |
+ record_buf_mem[3] = tgt_mem_addr + 4; |
+ arm_insn_r->mem_rec_count = 2; |
+ } |
+ } |
+ else if (11 == arm_insn_r->opcode || 15 == arm_insn_r->opcode |
+ || 2 == arm_insn_r->opcode || 6 == arm_insn_r->opcode) |
+ { |
+ /* 3) Store, immediate pre-indexed. */ |
+ /* 5) Store, immediate post-indexed. */ |
+ immed_low = bits (arm_insn_r->arm_insn, 0, 3); |
+ immed_high = bits (arm_insn_r->arm_insn, 8, 11); |
+ offset_8 = (immed_high << 4) | immed_low; |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ /* Calculate target store address, Rn +/- Rm, register offset. */ |
+ if (15 == arm_insn_r->opcode || 6 == arm_insn_r->opcode) |
+ { |
+ tgt_mem_addr = u_regval[0] + offset_8; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval[0] - offset_8; |
+ } |
+ if (ARM_RECORD_STRH == str_type) |
+ { |
+ record_buf_mem[0] = 2; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ } |
+ else if (ARM_RECORD_STRD == str_type) |
+ { |
+ record_buf_mem[0] = 4; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ record_buf_mem[2] = 4; |
+ record_buf_mem[3] = tgt_mem_addr + 4; |
+ arm_insn_r->mem_rec_count = 2; |
+ } |
+ /* Record Rn also as it changes. */ |
+ *(record_buf) = bits (arm_insn_r->arm_insn, 16, 19); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (9 == arm_insn_r->opcode || 13 == arm_insn_r->opcode |
+ || 0 == arm_insn_r->opcode || 4 == arm_insn_r->opcode) |
+ { |
+ /* 4) Store, register pre-indexed. */ |
+ /* 6) Store, register post -indexed. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 0, 3); |
+ reg_src2 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ regcache_raw_read_unsigned (reg_cache, reg_src2, &u_regval[1]); |
+ /* Calculate target store address, Rn +/- Rm, register offset. */ |
+ if (13 == arm_insn_r->opcode || 4 == arm_insn_r->opcode) |
+ { |
+ tgt_mem_addr = u_regval[0] + u_regval[1]; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval[1] - u_regval[0]; |
+ } |
+ if (ARM_RECORD_STRH == str_type) |
+ { |
+ record_buf_mem[0] = 2; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ } |
+ else if (ARM_RECORD_STRD == str_type) |
+ { |
+ record_buf_mem[0] = 4; |
+ record_buf_mem[1] = tgt_mem_addr; |
+ record_buf_mem[2] = 4; |
+ record_buf_mem[3] = tgt_mem_addr + 4; |
+ arm_insn_r->mem_rec_count = 2; |
+ } |
+ /* Record Rn also as it changes. */ |
+ *(record_buf) = bits (arm_insn_r->arm_insn, 16, 19); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ return 0; |
+} |
+ |
+/* Handling ARM extension space insns. */ |
+ |
+static int |
+arm_record_extension_space (insn_decode_record *arm_insn_r) |
+{ |
+ uint32_t ret = 0; /* Return value: -1:record failure ; 0:success */ |
+ uint32_t opcode1 = 0, opcode2 = 0, insn_op1 = 0; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ uint32_t reg_src1 = 0; |
+ uint32_t immed_high = 0, immed_low = 0,offset_8 = 0, tgt_mem_addr = 0; |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ ULONGEST u_regval = 0; |
+ |
+ gdb_assert (!INSN_RECORDED(arm_insn_r)); |
+ /* Handle unconditional insn extension space. */ |
+ |
+ opcode1 = bits (arm_insn_r->arm_insn, 20, 27); |
+ opcode2 = bits (arm_insn_r->arm_insn, 4, 7); |
+ if (arm_insn_r->cond) |
+ { |
+ /* PLD has no affect on architectural state, it just affects |
+ the caches. */ |
+ if (5 == ((opcode1 & 0xE0) >> 5)) |
+ { |
+ /* BLX(1) */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ /* STC2, LDC2, MCR2, MRC2, CDP2: <TBD>, co-processor insn. */ |
+ } |
+ |
+ |
+ opcode1 = bits (arm_insn_r->arm_insn, 25, 27); |
+ if (3 == opcode1 && bit (arm_insn_r->arm_insn, 4)) |
+ { |
+ ret = -1; |
+ /* Undefined instruction on ARM V5; need to handle if later |
+ versions define it. */ |
+ } |
+ |
+ opcode1 = bits (arm_insn_r->arm_insn, 24, 27); |
+ opcode2 = bits (arm_insn_r->arm_insn, 4, 7); |
+ insn_op1 = bits (arm_insn_r->arm_insn, 20, 23); |
+ |
+ /* Handle arithmetic insn extension space. */ |
+ if (!opcode1 && 9 == opcode2 && 1 != arm_insn_r->cond |
+ && !INSN_RECORDED(arm_insn_r)) |
+ { |
+ /* Handle MLA(S) and MUL(S). */ |
+ if (0 <= insn_op1 && 3 >= insn_op1) |
+ { |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (4 <= insn_op1 && 15 >= insn_op1) |
+ { |
+ /* Handle SMLAL(S), SMULL(S), UMLAL(S), UMULL(S). */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 16, 19); |
+ record_buf[1] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[2] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 3; |
+ } |
+ } |
+ |
+ opcode1 = bits (arm_insn_r->arm_insn, 26, 27); |
+ opcode2 = bits (arm_insn_r->arm_insn, 23, 24); |
+ insn_op1 = bits (arm_insn_r->arm_insn, 21, 22); |
+ |
+ /* Handle control insn extension space. */ |
+ |
+ if (!opcode1 && 2 == opcode2 && !bit (arm_insn_r->arm_insn, 20) |
+ && 1 != arm_insn_r->cond && !INSN_RECORDED(arm_insn_r)) |
+ { |
+ if (!bit (arm_insn_r->arm_insn,25)) |
+ { |
+ if (!bits (arm_insn_r->arm_insn, 4, 7)) |
+ { |
+ if ((0 == insn_op1) || (2 == insn_op1)) |
+ { |
+ /* MRS. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (1 == insn_op1) |
+ { |
+ /* CSPR is going to be changed. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (3 == insn_op1) |
+ { |
+ /* SPSR is going to be changed. */ |
+ /* We need to get SPSR value, which is yet to be done. */ |
+ printf_unfiltered (_("Process record does not support " |
+ "instruction 0x%0x at address %s.\n"), |
+ arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, |
+ arm_insn_r->this_addr)); |
+ return -1; |
+ } |
+ } |
+ else if (1 == bits (arm_insn_r->arm_insn, 4, 7)) |
+ { |
+ if (1 == insn_op1) |
+ { |
+ /* BX. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (3 == insn_op1) |
+ { |
+ /* CLZ. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ } |
+ else if (3 == bits (arm_insn_r->arm_insn, 4, 7)) |
+ { |
+ /* BLX. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (5 == bits (arm_insn_r->arm_insn, 4, 7)) |
+ { |
+ /* QADD, QSUB, QDADD, QDSUB */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (7 == bits (arm_insn_r->arm_insn, 4, 7)) |
+ { |
+ /* BKPT. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ |
+ /* Save SPSR also;how? */ |
+ printf_unfiltered (_("Process record does not support " |
+ "instruction 0x%0x at address %s.\n"), |
+ arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, arm_insn_r->this_addr)); |
+ return -1; |
+ } |
+ else if(8 == bits (arm_insn_r->arm_insn, 4, 7) |
+ || 10 == bits (arm_insn_r->arm_insn, 4, 7) |
+ || 12 == bits (arm_insn_r->arm_insn, 4, 7) |
+ || 14 == bits (arm_insn_r->arm_insn, 4, 7) |
+ ) |
+ { |
+ if (0 == insn_op1 || 1 == insn_op1) |
+ { |
+ /* SMLA<x><y>, SMLAW<y>, SMULW<y>. */ |
+ /* We dont do optimization for SMULW<y> where we |
+ need only Rd. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (2 == insn_op1) |
+ { |
+ /* SMLAL<x><y>. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[1] = bits (arm_insn_r->arm_insn, 16, 19); |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (3 == insn_op1) |
+ { |
+ /* SMUL<x><y>. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ } |
+ } |
+ else |
+ { |
+ /* MSR : immediate form. */ |
+ if (1 == insn_op1) |
+ { |
+ /* CSPR is going to be changed. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (3 == insn_op1) |
+ { |
+ /* SPSR is going to be changed. */ |
+ /* we need to get SPSR value, which is yet to be done */ |
+ printf_unfiltered (_("Process record does not support " |
+ "instruction 0x%0x at address %s.\n"), |
+ arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, |
+ arm_insn_r->this_addr)); |
+ return -1; |
+ } |
+ } |
+ } |
+ |
+ opcode1 = bits (arm_insn_r->arm_insn, 25, 27); |
+ opcode2 = bits (arm_insn_r->arm_insn, 20, 24); |
+ insn_op1 = bits (arm_insn_r->arm_insn, 5, 6); |
+ |
+ /* Handle load/store insn extension space. */ |
+ |
+ if (!opcode1 && bit (arm_insn_r->arm_insn, 7) |
+ && bit (arm_insn_r->arm_insn, 4) && 1 != arm_insn_r->cond |
+ && !INSN_RECORDED(arm_insn_r)) |
+ { |
+ /* SWP/SWPB. */ |
+ if (0 == insn_op1) |
+ { |
+ /* These insn, changes register and memory as well. */ |
+ /* SWP or SWPB insn. */ |
+ /* Get memory address given by Rn. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval); |
+ /* SWP insn ?, swaps word. */ |
+ if (8 == arm_insn_r->opcode) |
+ { |
+ record_buf_mem[0] = 4; |
+ } |
+ else |
+ { |
+ /* SWPB insn, swaps only byte. */ |
+ record_buf_mem[0] = 1; |
+ } |
+ record_buf_mem[1] = u_regval; |
+ arm_insn_r->mem_rec_count = 1; |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (1 == insn_op1 && !bit (arm_insn_r->arm_insn, 20)) |
+ { |
+ /* STRH. */ |
+ arm_record_strx(arm_insn_r, &record_buf[0], &record_buf_mem[0], |
+ ARM_RECORD_STRH); |
+ } |
+ else if (2 == insn_op1 && !bit (arm_insn_r->arm_insn, 20)) |
+ { |
+ /* LDRD. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[1] = record_buf[0] + 1; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (3 == insn_op1 && !bit (arm_insn_r->arm_insn, 20)) |
+ { |
+ /* STRD. */ |
+ arm_record_strx(arm_insn_r, &record_buf[0], &record_buf_mem[0], |
+ ARM_RECORD_STRD); |
+ } |
+ else if (bit (arm_insn_r->arm_insn, 20) && insn_op1 <= 3) |
+ { |
+ /* LDRH, LDRSB, LDRSH. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ |
+ } |
+ |
+ opcode1 = bits (arm_insn_r->arm_insn, 23, 27); |
+ if (24 == opcode1 && bit (arm_insn_r->arm_insn, 21) |
+ && !INSN_RECORDED(arm_insn_r)) |
+ { |
+ ret = -1; |
+ /* Handle coprocessor insn extension space. */ |
+ } |
+ |
+ /* To be done for ARMv5 and later; as of now we return -1. */ |
+ if (-1 == ret) |
+ printf_unfiltered (_("Process record does not support instruction x%0x " |
+ "at address %s.\n"),arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, arm_insn_r->this_addr)); |
+ |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (arm_insn_r->arm_mems, arm_insn_r->mem_rec_count, record_buf_mem); |
+ |
+ return ret; |
+} |
+ |
+/* Handling opcode 000 insns. */ |
+ |
+static int |
+arm_record_data_proc_misc_ld_str (insn_decode_record *arm_insn_r) |
+{ |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ ULONGEST u_regval[2] = {0}; |
+ |
+ uint32_t reg_src1 = 0, reg_src2 = 0, reg_dest = 0; |
+ uint32_t immed_high = 0, immed_low = 0, offset_8 = 0, tgt_mem_addr = 0; |
+ uint32_t opcode1 = 0; |
+ |
+ arm_insn_r->opcode = bits (arm_insn_r->arm_insn, 21, 24); |
+ arm_insn_r->decode = bits (arm_insn_r->arm_insn, 4, 7); |
+ opcode1 = bits (arm_insn_r->arm_insn, 20, 24); |
+ |
+ /* Data processing insn /multiply insn. */ |
+ if (9 == arm_insn_r->decode |
+ && ((4 <= arm_insn_r->opcode && 7 >= arm_insn_r->opcode) |
+ || (0 == arm_insn_r->opcode || 1 == arm_insn_r->opcode))) |
+ { |
+ /* Handle multiply instructions. */ |
+ /* MLA, MUL, SMLAL, SMULL, UMLAL, UMULL. */ |
+ if (0 == arm_insn_r->opcode || 1 == arm_insn_r->opcode) |
+ { |
+ /* Handle MLA and MUL. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 16, 19); |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else if (4 <= arm_insn_r->opcode && 7 >= arm_insn_r->opcode) |
+ { |
+ /* Handle SMLAL, SMULL, UMLAL, UMULL. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 16, 19); |
+ record_buf[1] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[2] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 3; |
+ } |
+ } |
+ else if (bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM) |
+ && (11 == arm_insn_r->decode || 13 == arm_insn_r->decode)) |
+ { |
+ /* Handle misc load insns, as 20th bit (L = 1). */ |
+ /* LDR insn has a capability to do branching, if |
+ MOV LR, PC is precceded by LDR insn having Rn as R15 |
+ in that case, it emulates branch and link insn, and hence we |
+ need to save CSPR and PC as well. I am not sure this is right |
+ place; as opcode = 010 LDR insn make this happen, if R15 was |
+ used. */ |
+ reg_dest = bits (arm_insn_r->arm_insn, 12, 15); |
+ if (15 != reg_dest) |
+ { |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ record_buf[0] = reg_dest; |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ } |
+ else if ((9 == arm_insn_r->opcode || 11 == arm_insn_r->opcode) |
+ && sbo_sbz (arm_insn_r->arm_insn, 5, 12, 0) |
+ && sbo_sbz (arm_insn_r->arm_insn, 13, 4, 1) |
+ && 2 == bits (arm_insn_r->arm_insn, 20, 21)) |
+ { |
+ /* Handle MSR insn. */ |
+ if (9 == arm_insn_r->opcode) |
+ { |
+ /* CSPR is going to be changed. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ /* SPSR is going to be changed. */ |
+ /* How to read SPSR value? */ |
+ printf_unfiltered (_("Process record does not support instruction " |
+ "0x%0x at address %s.\n"), |
+ arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, arm_insn_r->this_addr)); |
+ return -1; |
+ } |
+ } |
+ else if (9 == arm_insn_r->decode |
+ && (8 == arm_insn_r->opcode || 10 == arm_insn_r->opcode) |
+ && !bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM)) |
+ { |
+ /* Handling SWP, SWPB. */ |
+ /* These insn, changes register and memory as well. */ |
+ /* SWP or SWPB insn. */ |
+ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ /* SWP insn ?, swaps word. */ |
+ if (8 == arm_insn_r->opcode) |
+ { |
+ record_buf_mem[0] = 4; |
+ } |
+ else |
+ { |
+ /* SWPB insn, swaps only byte. */ |
+ record_buf_mem[0] = 1; |
+ } |
+ record_buf_mem[1] = u_regval[0]; |
+ arm_insn_r->mem_rec_count = 1; |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (3 == arm_insn_r->decode && 0x12 == opcode1 |
+ && sbo_sbz (arm_insn_r->arm_insn, 9, 12, 1)) |
+ { |
+ /* Handle BLX, branch and link/exchange. */ |
+ if (9 == arm_insn_r->opcode) |
+ { |
+ /* Branch is chosen by setting T bit of CSPR, bitp[0] of Rm, |
+ and R14 stores the return address. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ } |
+ else if (7 == arm_insn_r->decode && 0x12 == opcode1) |
+ { |
+ /* Handle enhanced software breakpoint insn, BKPT. */ |
+ /* CPSR is changed to be executed in ARM state, disabling normal |
+ interrupts, entering abort mode. */ |
+ /* According to high vector configuration PC is set. */ |
+ /* user hit breakpoint and type reverse, in |
+ that case, we need to go back with previous CPSR and |
+ Program Counter. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ |
+ /* Save SPSR also; how? */ |
+ printf_unfiltered (_("Process record does not support instruction " |
+ "0x%0x at address %s.\n"),arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, |
+ arm_insn_r->this_addr)); |
+ return -1; |
+ } |
+ else if (11 == arm_insn_r->decode |
+ && !bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM)) |
+ { |
+ /* Handle enhanced store insns and DSP insns (e.g. LDRD). */ |
+ |
+ /* Handle str(x) insn */ |
+ arm_record_strx(arm_insn_r, &record_buf[0], &record_buf_mem[0], |
+ ARM_RECORD_STRH); |
+ } |
+ else if (1 == arm_insn_r->decode && 0x12 == opcode1 |
+ && sbo_sbz (arm_insn_r->arm_insn, 9, 12, 1)) |
+ { |
+ /* Handle BX, branch and link/exchange. */ |
+ /* Branch is chosen by setting T bit of CSPR, bitp[0] of Rm. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (1 == arm_insn_r->decode && 0x16 == opcode1 |
+ && sbo_sbz (arm_insn_r->arm_insn, 9, 4, 1) |
+ && sbo_sbz (arm_insn_r->arm_insn, 17, 4, 1)) |
+ { |
+ /* Count leading zeros: CLZ. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (!bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM) |
+ && (8 == arm_insn_r->opcode || 10 == arm_insn_r->opcode) |
+ && sbo_sbz (arm_insn_r->arm_insn, 17, 4, 1) |
+ && sbo_sbz (arm_insn_r->arm_insn, 1, 12, 0) |
+ ) |
+ { |
+ /* Handle MRS insn. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else if (arm_insn_r->opcode <= 15) |
+ { |
+ /* Normal data processing insns. */ |
+ /* Out of 11 shifter operands mode, all the insn modifies destination |
+ register, which is specified by 13-16 decode. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else |
+ { |
+ return -1; |
+ } |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (arm_insn_r->arm_mems, arm_insn_r->mem_rec_count, record_buf_mem); |
+ return 0; |
+} |
+ |
+/* Handling opcode 001 insns. */ |
+ |
+static int |
+arm_record_data_proc_imm (insn_decode_record *arm_insn_r) |
+{ |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ |
+ arm_insn_r->opcode = bits (arm_insn_r->arm_insn, 21, 24); |
+ arm_insn_r->decode = bits (arm_insn_r->arm_insn, 4, 7); |
+ |
+ if ((9 == arm_insn_r->opcode || 11 == arm_insn_r->opcode) |
+ && 2 == bits (arm_insn_r->arm_insn, 20, 21) |
+ && sbo_sbz (arm_insn_r->arm_insn, 13, 4, 1) |
+ ) |
+ { |
+ /* Handle MSR insn. */ |
+ if (9 == arm_insn_r->opcode) |
+ { |
+ /* CSPR is going to be changed. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ /* SPSR is going to be changed. */ |
+ } |
+ } |
+ else if (arm_insn_r->opcode <= 15) |
+ { |
+ /* Normal data processing insns. */ |
+ /* Out of 11 shifter operands mode, all the insn modifies destination |
+ register, which is specified by 13-16 decode. */ |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ else |
+ { |
+ return -1; |
+ } |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (arm_insn_r->arm_mems, arm_insn_r->mem_rec_count, record_buf_mem); |
+ return 0; |
+} |
+ |
+/* Handling opcode 010 insns. */ |
+ |
+static int |
+arm_record_ld_st_imm_offset (insn_decode_record *arm_insn_r) |
+{ |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ |
+ uint32_t reg_src1 = 0 , reg_dest = 0; |
+ uint32_t offset_12 = 0, tgt_mem_addr = 0; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ |
+ ULONGEST u_regval = 0; |
+ |
+ arm_insn_r->opcode = bits (arm_insn_r->arm_insn, 21, 24); |
+ arm_insn_r->decode = bits (arm_insn_r->arm_insn, 4, 7); |
+ |
+ if (bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM)) |
+ { |
+ reg_dest = bits (arm_insn_r->arm_insn, 12, 15); |
+ /* LDR insn has a capability to do branching, if |
+ MOV LR, PC is precedded by LDR insn having Rn as R15 |
+ in that case, it emulates branch and link insn, and hence we |
+ need to save CSPR and PC as well. */ |
+ if (ARM_PC_REGNUM != reg_dest) |
+ { |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ record_buf[0] = reg_dest; |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ } |
+ else |
+ { |
+ /* Store, immediate offset, immediate pre-indexed, |
+ immediate post-indexed. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ offset_12 = bits (arm_insn_r->arm_insn, 0, 11); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval); |
+ /* U == 1 */ |
+ if (bit (arm_insn_r->arm_insn, 23)) |
+ { |
+ tgt_mem_addr = u_regval + offset_12; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval - offset_12; |
+ } |
+ |
+ switch (arm_insn_r->opcode) |
+ { |
+ /* STR. */ |
+ case 8: |
+ case 12: |
+ /* STR. */ |
+ case 9: |
+ case 13: |
+ /* STRT. */ |
+ case 1: |
+ case 5: |
+ /* STR. */ |
+ case 4: |
+ case 0: |
+ record_buf_mem[0] = 4; |
+ break; |
+ |
+ /* STRB. */ |
+ case 10: |
+ case 14: |
+ /* STRB. */ |
+ case 11: |
+ case 15: |
+ /* STRBT. */ |
+ case 3: |
+ case 7: |
+ /* STRB. */ |
+ case 2: |
+ case 6: |
+ record_buf_mem[0] = 1; |
+ break; |
+ |
+ default: |
+ gdb_assert_not_reached ("no decoding pattern found"); |
+ break; |
+ } |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ |
+ if (9 == arm_insn_r->opcode || 11 == arm_insn_r->opcode |
+ || 13 == arm_insn_r->opcode || 15 == arm_insn_r->opcode |
+ || 0 == arm_insn_r->opcode || 2 == arm_insn_r->opcode |
+ || 4 == arm_insn_r->opcode || 6 == arm_insn_r->opcode |
+ || 1 == arm_insn_r->opcode || 3 == arm_insn_r->opcode |
+ || 5 == arm_insn_r->opcode || 7 == arm_insn_r->opcode |
+ ) |
+ { |
+ /* We are handling pre-indexed mode; post-indexed mode; |
+ where Rn is going to be changed. */ |
+ record_buf[0] = reg_src1; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ } |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (arm_insn_r->arm_mems, arm_insn_r->mem_rec_count, record_buf_mem); |
+ return 0; |
+} |
+ |
+/* Handling opcode 011 insns. */ |
+ |
+static int |
+arm_record_ld_st_reg_offset (insn_decode_record *arm_insn_r) |
+{ |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ |
+ uint32_t shift_imm = 0; |
+ uint32_t reg_src1 = 0, reg_src2 = 0, reg_dest = 0; |
+ uint32_t offset_12 = 0, tgt_mem_addr = 0; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ |
+ LONGEST s_word; |
+ ULONGEST u_regval[2]; |
+ |
+ arm_insn_r->opcode = bits (arm_insn_r->arm_insn, 21, 24); |
+ arm_insn_r->decode = bits (arm_insn_r->arm_insn, 4, 7); |
+ |
+ /* Handle enhanced store insns and LDRD DSP insn, |
+ order begins according to addressing modes for store insns |
+ STRH insn. */ |
+ |
+ /* LDR or STR? */ |
+ if (bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM)) |
+ { |
+ reg_dest = bits (arm_insn_r->arm_insn, 12, 15); |
+ /* LDR insn has a capability to do branching, if |
+ MOV LR, PC is precedded by LDR insn having Rn as R15 |
+ in that case, it emulates branch and link insn, and hence we |
+ need to save CSPR and PC as well. */ |
+ if (15 != reg_dest) |
+ { |
+ record_buf[0] = bits (arm_insn_r->arm_insn, 12, 15); |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ record_buf[0] = reg_dest; |
+ record_buf[1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = 2; |
+ } |
+ } |
+ else |
+ { |
+ if (! bits (arm_insn_r->arm_insn, 4, 11)) |
+ { |
+ /* Store insn, register offset and register pre-indexed, |
+ register post-indexed. */ |
+ /* Get Rm. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 0, 3); |
+ /* Get Rn. */ |
+ reg_src2 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1 |
+ , &u_regval[0]); |
+ regcache_raw_read_unsigned (reg_cache, reg_src2 |
+ , &u_regval[1]); |
+ if (15 == reg_src2) |
+ { |
+ /* If R15 was used as Rn, hence current PC+8. */ |
+ /* Pre-indexed mode doesnt reach here ; illegal insn. */ |
+ u_regval[0] = u_regval[0] + 8; |
+ } |
+ /* Calculate target store address, Rn +/- Rm, register offset. */ |
+ /* U == 1. */ |
+ if (bit (arm_insn_r->arm_insn, 23)) |
+ { |
+ tgt_mem_addr = u_regval[0] + u_regval[1]; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval[1] - u_regval[0]; |
+ } |
+ |
+ switch (arm_insn_r->opcode) |
+ { |
+ /* STR. */ |
+ case 8: |
+ case 12: |
+ /* STR. */ |
+ case 9: |
+ case 13: |
+ /* STRT. */ |
+ case 1: |
+ case 5: |
+ /* STR. */ |
+ case 0: |
+ case 4: |
+ record_buf_mem[0] = 4; |
+ break; |
+ |
+ /* STRB. */ |
+ case 10: |
+ case 14: |
+ /* STRB. */ |
+ case 11: |
+ case 15: |
+ /* STRBT. */ |
+ case 3: |
+ case 7: |
+ /* STRB. */ |
+ case 2: |
+ case 6: |
+ record_buf_mem[0] = 1; |
+ break; |
+ |
+ default: |
+ gdb_assert_not_reached ("no decoding pattern found"); |
+ break; |
+ } |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ |
+ if (9 == arm_insn_r->opcode || 11 == arm_insn_r->opcode |
+ || 13 == arm_insn_r->opcode || 15 == arm_insn_r->opcode |
+ || 0 == arm_insn_r->opcode || 2 == arm_insn_r->opcode |
+ || 4 == arm_insn_r->opcode || 6 == arm_insn_r->opcode |
+ || 1 == arm_insn_r->opcode || 3 == arm_insn_r->opcode |
+ || 5 == arm_insn_r->opcode || 7 == arm_insn_r->opcode |
+ ) |
+ { |
+ /* Rn is going to be changed in pre-indexed mode and |
+ post-indexed mode as well. */ |
+ record_buf[0] = reg_src2; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ } |
+ else |
+ { |
+ /* Store insn, scaled register offset; scaled pre-indexed. */ |
+ offset_12 = bits (arm_insn_r->arm_insn, 5, 6); |
+ /* Get Rm. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 0, 3); |
+ /* Get Rn. */ |
+ reg_src2 = bits (arm_insn_r->arm_insn, 16, 19); |
+ /* Get shift_imm. */ |
+ shift_imm = bits (arm_insn_r->arm_insn, 7, 11); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ regcache_raw_read_signed (reg_cache, reg_src1, &s_word); |
+ regcache_raw_read_unsigned (reg_cache, reg_src2, &u_regval[1]); |
+ /* Offset_12 used as shift. */ |
+ switch (offset_12) |
+ { |
+ case 0: |
+ /* Offset_12 used as index. */ |
+ offset_12 = u_regval[0] << shift_imm; |
+ break; |
+ |
+ case 1: |
+ offset_12 = (!shift_imm)?0:u_regval[0] >> shift_imm; |
+ break; |
+ |
+ case 2: |
+ if (!shift_imm) |
+ { |
+ if (bit (u_regval[0], 31)) |
+ { |
+ offset_12 = 0xFFFFFFFF; |
+ } |
+ else |
+ { |
+ offset_12 = 0; |
+ } |
+ } |
+ else |
+ { |
+ /* This is arithmetic shift. */ |
+ offset_12 = s_word >> shift_imm; |
+ } |
+ break; |
+ |
+ case 3: |
+ if (!shift_imm) |
+ { |
+ regcache_raw_read_unsigned (reg_cache, ARM_PS_REGNUM, |
+ &u_regval[1]); |
+ /* Get C flag value and shift it by 31. */ |
+ offset_12 = (((bit (u_regval[1], 29)) << 31) \ |
+ | (u_regval[0]) >> 1); |
+ } |
+ else |
+ { |
+ offset_12 = (u_regval[0] >> shift_imm) \ |
+ | (u_regval[0] << |
+ (sizeof(uint32_t) - shift_imm)); |
+ } |
+ break; |
+ |
+ default: |
+ gdb_assert_not_reached ("no decoding pattern found"); |
+ break; |
+ } |
+ |
+ regcache_raw_read_unsigned (reg_cache, reg_src2, &u_regval[1]); |
+ /* bit U set. */ |
+ if (bit (arm_insn_r->arm_insn, 23)) |
+ { |
+ tgt_mem_addr = u_regval[1] + offset_12; |
+ } |
+ else |
+ { |
+ tgt_mem_addr = u_regval[1] - offset_12; |
+ } |
+ |
+ switch (arm_insn_r->opcode) |
+ { |
+ /* STR. */ |
+ case 8: |
+ case 12: |
+ /* STR. */ |
+ case 9: |
+ case 13: |
+ /* STRT. */ |
+ case 1: |
+ case 5: |
+ /* STR. */ |
+ case 0: |
+ case 4: |
+ record_buf_mem[0] = 4; |
+ break; |
+ |
+ /* STRB. */ |
+ case 10: |
+ case 14: |
+ /* STRB. */ |
+ case 11: |
+ case 15: |
+ /* STRBT. */ |
+ case 3: |
+ case 7: |
+ /* STRB. */ |
+ case 2: |
+ case 6: |
+ record_buf_mem[0] = 1; |
+ break; |
+ |
+ default: |
+ gdb_assert_not_reached ("no decoding pattern found"); |
+ break; |
+ } |
+ record_buf_mem[1] = tgt_mem_addr; |
+ arm_insn_r->mem_rec_count = 1; |
+ |
+ if (9 == arm_insn_r->opcode || 11 == arm_insn_r->opcode |
+ || 13 == arm_insn_r->opcode || 15 == arm_insn_r->opcode |
+ || 0 == arm_insn_r->opcode || 2 == arm_insn_r->opcode |
+ || 4 == arm_insn_r->opcode || 6 == arm_insn_r->opcode |
+ || 1 == arm_insn_r->opcode || 3 == arm_insn_r->opcode |
+ || 5 == arm_insn_r->opcode || 7 == arm_insn_r->opcode |
+ ) |
+ { |
+ /* Rn is going to be changed in register scaled pre-indexed |
+ mode,and scaled post indexed mode. */ |
+ record_buf[0] = reg_src2; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ } |
+ } |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (arm_insn_r->arm_mems, arm_insn_r->mem_rec_count, record_buf_mem); |
+ return 0; |
+} |
+ |
+/* Handling opcode 100 insns. */ |
+ |
+static int |
+arm_record_ld_st_multiple (insn_decode_record *arm_insn_r) |
+{ |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ |
+ uint32_t register_list[16] = {0}, register_count = 0, register_bits = 0; |
+ uint32_t reg_src1 = 0, addr_mode = 0, no_of_regs = 0; |
+ uint32_t start_address = 0, index = 0; |
+ uint32_t record_buf[24], record_buf_mem[48]; |
+ |
+ ULONGEST u_regval[2] = {0}; |
+ |
+ /* This mode is exclusively for load and store multiple. */ |
+ /* Handle incremenrt after/before and decrment after.before mode; |
+ Rn is changing depending on W bit, but as of now we store Rn too |
+ without optimization. */ |
+ |
+ if (bit (arm_insn_r->arm_insn, INSN_S_L_BIT_NUM)) |
+ { |
+ /* LDM (1,2,3) where LDM (3) changes CPSR too. */ |
+ |
+ if (bit (arm_insn_r->arm_insn, 20) && !bit (arm_insn_r->arm_insn, 22)) |
+ { |
+ register_bits = bits (arm_insn_r->arm_insn, 0, 15); |
+ no_of_regs = 15; |
+ } |
+ else |
+ { |
+ register_bits = bits (arm_insn_r->arm_insn, 0, 14); |
+ no_of_regs = 14; |
+ } |
+ /* Get Rn. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ while (register_bits) |
+ { |
+ if (register_bits & 0x00000001) |
+ register_list[register_count++] = 1; |
+ register_bits = register_bits >> 1; |
+ } |
+ |
+ /* Extra space for Base Register and CPSR; wihtout optimization. */ |
+ record_buf[register_count] = reg_src1; |
+ record_buf[register_count + 1] = ARM_PS_REGNUM; |
+ arm_insn_r->reg_rec_count = register_count + 2; |
+ |
+ for (register_count = 0; register_count < no_of_regs; register_count++) |
+ { |
+ if (register_list[register_count]) |
+ { |
+ /* Register_count gives total no of registers |
+ and dually working as reg number. */ |
+ record_buf[index] = register_count; |
+ index++; |
+ } |
+ } |
+ |
+ } |
+ else |
+ { |
+ /* It handles both STM(1) and STM(2). */ |
+ addr_mode = bits (arm_insn_r->arm_insn, 23, 24); |
+ |
+ register_bits = bits (arm_insn_r->arm_insn, 0, 15); |
+ /* Get Rn. */ |
+ reg_src1 = bits (arm_insn_r->arm_insn, 16, 19); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ while (register_bits) |
+ { |
+ if (register_bits & 0x00000001) |
+ register_count++; |
+ register_bits = register_bits >> 1; |
+ } |
+ |
+ switch (addr_mode) |
+ { |
+ /* Decrement after. */ |
+ case 0: |
+ start_address = (u_regval[0]) - (register_count * 4) + 4; |
+ arm_insn_r->mem_rec_count = register_count; |
+ while (register_count) |
+ { |
+ record_buf_mem[(register_count * 2) - 1] = start_address; |
+ record_buf_mem[(register_count * 2) - 2] = 4; |
+ start_address = start_address + 4; |
+ register_count--; |
+ } |
+ break; |
+ |
+ /* Increment after. */ |
+ case 1: |
+ start_address = u_regval[0]; |
+ arm_insn_r->mem_rec_count = register_count; |
+ while (register_count) |
+ { |
+ record_buf_mem[(register_count * 2) - 1] = start_address; |
+ record_buf_mem[(register_count * 2) - 2] = 4; |
+ start_address = start_address + 4; |
+ register_count--; |
+ } |
+ break; |
+ |
+ /* Decrement before. */ |
+ case 2: |
+ |
+ start_address = (u_regval[0]) - (register_count * 4); |
+ arm_insn_r->mem_rec_count = register_count; |
+ while (register_count) |
+ { |
+ record_buf_mem[(register_count * 2) - 1] = start_address; |
+ record_buf_mem[(register_count * 2) - 2] = 4; |
+ start_address = start_address + 4; |
+ register_count--; |
+ } |
+ break; |
+ |
+ /* Increment before. */ |
+ case 3: |
+ start_address = u_regval[0] + 4; |
+ arm_insn_r->mem_rec_count = register_count; |
+ while (register_count) |
+ { |
+ record_buf_mem[(register_count * 2) - 1] = start_address; |
+ record_buf_mem[(register_count * 2) - 2] = 4; |
+ start_address = start_address + 4; |
+ register_count--; |
+ } |
+ break; |
+ |
+ default: |
+ gdb_assert_not_reached ("no decoding pattern found"); |
+ break; |
+ } |
+ |
+ /* Base register also changes; based on condition and W bit. */ |
+ /* We save it anyway without optimization. */ |
+ record_buf[0] = reg_src1; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (arm_insn_r->arm_mems, arm_insn_r->mem_rec_count, record_buf_mem); |
+ return 0; |
+} |
+ |
+/* Handling opcode 101 insns. */ |
+ |
+static int |
+arm_record_b_bl (insn_decode_record *arm_insn_r) |
+{ |
+ uint32_t record_buf[8]; |
+ |
+ /* Handle B, BL, BLX(1) insns. */ |
+ /* B simply branches so we do nothing here. */ |
+ /* Note: BLX(1) doesnt fall here but instead it falls into |
+ extension space. */ |
+ if (bit (arm_insn_r->arm_insn, 24)) |
+ { |
+ record_buf[0] = ARM_LR_REGNUM; |
+ arm_insn_r->reg_rec_count = 1; |
+ } |
+ |
+ REG_ALLOC (arm_insn_r->arm_regs, arm_insn_r->reg_rec_count, record_buf); |
+ |
+ return 0; |
+} |
+ |
+/* Handling opcode 110 insns. */ |
+ |
+static int |
+arm_record_coproc (insn_decode_record *arm_insn_r) |
+{ |
+ printf_unfiltered (_("Process record does not support instruction " |
+ "0x%0x at address %s.\n"),arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, arm_insn_r->this_addr)); |
+ |
+ return -1; |
+} |
+ |
+/* Handling opcode 111 insns. */ |
+ |
+static int |
+arm_record_coproc_data_proc (insn_decode_record *arm_insn_r) |
+{ |
+ struct gdbarch_tdep *tdep = gdbarch_tdep (arm_insn_r->gdbarch); |
+ struct regcache *reg_cache = arm_insn_r->regcache; |
+ uint32_t ret = 0; /* function return value: -1:record failure ; 0:success */ |
+ |
+ /* Handle SWI insn; system call would be handled over here. */ |
+ |
+ arm_insn_r->opcode = bits (arm_insn_r->arm_insn, 24, 27); |
+ if (15 == arm_insn_r->opcode) |
+ { |
+ /* Handle arm syscall insn. */ |
+ if (tdep->arm_swi_record != NULL) |
+ { |
+ ret = tdep->arm_swi_record(reg_cache); |
+ } |
+ else |
+ { |
+ printf_unfiltered (_("no syscall record support\n")); |
+ ret = -1; |
+ } |
+ } |
+ |
+ printf_unfiltered (_("Process record does not support instruction " |
+ "0x%0x at address %s.\n"),arm_insn_r->arm_insn, |
+ paddress (arm_insn_r->gdbarch, arm_insn_r->this_addr)); |
+ return ret; |
+} |
+ |
+/* Handling opcode 000 insns. */ |
+ |
+static int |
+thumb_record_shift_add_sub (insn_decode_record *thumb_insn_r) |
+{ |
+ uint32_t record_buf[8]; |
+ uint32_t reg_src1 = 0; |
+ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 0, 2); |
+ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 2; |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ |
+ return 0; |
+} |
+ |
+ |
+/* Handling opcode 001 insns. */ |
+ |
+static int |
+thumb_record_add_sub_cmp_mov (insn_decode_record *thumb_insn_r) |
+{ |
+ uint32_t record_buf[8]; |
+ uint32_t reg_src1 = 0; |
+ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 2; |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ |
+ return 0; |
+} |
+ |
+/* Handling opcode 010 insns. */ |
+ |
+static int |
+thumb_record_ld_st_reg_offset (insn_decode_record *thumb_insn_r) |
+{ |
+ struct regcache *reg_cache = thumb_insn_r->regcache; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ |
+ uint32_t reg_src1 = 0, reg_src2 = 0; |
+ uint32_t opcode1 = 0, opcode2 = 0, opcode3 = 0; |
+ |
+ ULONGEST u_regval[2] = {0}; |
+ |
+ opcode1 = bits (thumb_insn_r->arm_insn, 10, 12); |
+ |
+ if (bit (thumb_insn_r->arm_insn, 12)) |
+ { |
+ /* Handle load/store register offset. */ |
+ opcode2 = bits (thumb_insn_r->arm_insn, 9, 10); |
+ if (opcode2 >= 12 && opcode2 <= 15) |
+ { |
+ /* LDR(2), LDRB(2) , LDRH(2), LDRSB, LDRSH. */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn,0, 2); |
+ record_buf[0] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (opcode2 >= 8 && opcode2 <= 10) |
+ { |
+ /* STR(2), STRB(2), STRH(2) . */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 3, 5); |
+ reg_src2 = bits (thumb_insn_r->arm_insn, 6, 8); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval[0]); |
+ regcache_raw_read_unsigned (reg_cache, reg_src2, &u_regval[1]); |
+ if (8 == opcode2) |
+ record_buf_mem[0] = 4; /* STR (2). */ |
+ else if (10 == opcode2) |
+ record_buf_mem[0] = 1; /* STRB (2). */ |
+ else if (9 == opcode2) |
+ record_buf_mem[0] = 2; /* STRH (2). */ |
+ record_buf_mem[1] = u_regval[0] + u_regval[1]; |
+ thumb_insn_r->mem_rec_count = 1; |
+ } |
+ } |
+ else if (bit (thumb_insn_r->arm_insn, 11)) |
+ { |
+ /* Handle load from literal pool. */ |
+ /* LDR(3). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ record_buf[0] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (opcode1) |
+ { |
+ opcode2 = bits (thumb_insn_r->arm_insn, 8, 9); |
+ opcode3 = bits (thumb_insn_r->arm_insn, 0, 2); |
+ if ((3 == opcode2) && (!opcode3)) |
+ { |
+ /* Branch with exchange. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ /* Format 8; special data processing insns. */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 0, 2); |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 2; |
+ } |
+ } |
+ else |
+ { |
+ /* Format 5; data processing insns. */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 0, 2); |
+ if (bit (thumb_insn_r->arm_insn, 7)) |
+ { |
+ reg_src1 = reg_src1 + 8; |
+ } |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 2; |
+ } |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (thumb_insn_r->arm_mems, thumb_insn_r->mem_rec_count, |
+ record_buf_mem); |
+ |
+ return 0; |
+} |
+ |
+/* Handling opcode 001 insns. */ |
+ |
+static int |
+thumb_record_ld_st_imm_offset (insn_decode_record *thumb_insn_r) |
+{ |
+ struct regcache *reg_cache = thumb_insn_r->regcache; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ |
+ uint32_t reg_src1 = 0; |
+ uint32_t opcode = 0, immed_5 = 0; |
+ |
+ ULONGEST u_regval = 0; |
+ |
+ opcode = bits (thumb_insn_r->arm_insn, 11, 12); |
+ |
+ if (opcode) |
+ { |
+ /* LDR(1). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 0, 2); |
+ record_buf[0] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else |
+ { |
+ /* STR(1). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 3, 5); |
+ immed_5 = bits (thumb_insn_r->arm_insn, 6, 10); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval); |
+ record_buf_mem[0] = 4; |
+ record_buf_mem[1] = u_regval + (immed_5 * 4); |
+ thumb_insn_r->mem_rec_count = 1; |
+ } |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (thumb_insn_r->arm_mems, thumb_insn_r->mem_rec_count, |
+ record_buf_mem); |
+ |
+ return 0; |
+} |
+ |
+/* Handling opcode 100 insns. */ |
+ |
+static int |
+thumb_record_ld_st_stack (insn_decode_record *thumb_insn_r) |
+{ |
+ struct regcache *reg_cache = thumb_insn_r->regcache; |
+ uint32_t record_buf[8], record_buf_mem[8]; |
+ |
+ uint32_t reg_src1 = 0; |
+ uint32_t opcode = 0, immed_8 = 0, immed_5 = 0; |
+ |
+ ULONGEST u_regval = 0; |
+ |
+ opcode = bits (thumb_insn_r->arm_insn, 11, 12); |
+ |
+ if (3 == opcode) |
+ { |
+ /* LDR(4). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ record_buf[0] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (1 == opcode) |
+ { |
+ /* LDRH(1). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 0, 2); |
+ record_buf[0] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (2 == opcode) |
+ { |
+ /* STR(3). */ |
+ immed_8 = bits (thumb_insn_r->arm_insn, 0, 7); |
+ regcache_raw_read_unsigned (reg_cache, ARM_SP_REGNUM, &u_regval); |
+ record_buf_mem[0] = 4; |
+ record_buf_mem[1] = u_regval + (immed_8 * 4); |
+ thumb_insn_r->mem_rec_count = 1; |
+ } |
+ else if (0 == opcode) |
+ { |
+ /* STRH(1). */ |
+ immed_5 = bits (thumb_insn_r->arm_insn, 6, 10); |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 3, 5); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval); |
+ record_buf_mem[0] = 2; |
+ record_buf_mem[1] = u_regval + (immed_5 * 2); |
+ thumb_insn_r->mem_rec_count = 1; |
+ } |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (thumb_insn_r->arm_mems, thumb_insn_r->mem_rec_count, |
+ record_buf_mem); |
+ |
+ return 0; |
+} |
+ |
+/* Handling opcode 101 insns. */ |
+ |
+static int |
+thumb_record_misc (insn_decode_record *thumb_insn_r) |
+{ |
+ struct regcache *reg_cache = thumb_insn_r->regcache; |
+ |
+ uint32_t opcode = 0, opcode1 = 0, opcode2 = 0; |
+ uint32_t register_bits = 0, register_count = 0; |
+ uint32_t register_list[8] = {0}, index = 0, start_address = 0; |
+ uint32_t record_buf[24], record_buf_mem[48]; |
+ uint32_t reg_src1; |
+ |
+ ULONGEST u_regval = 0; |
+ |
+ opcode = bits (thumb_insn_r->arm_insn, 11, 12); |
+ opcode1 = bits (thumb_insn_r->arm_insn, 8, 12); |
+ opcode2 = bits (thumb_insn_r->arm_insn, 9, 12); |
+ |
+ if (14 == opcode2) |
+ { |
+ /* POP. */ |
+ register_bits = bits (thumb_insn_r->arm_insn, 0, 7); |
+ while (register_bits) |
+ { |
+ if (register_bits & 0x00000001) |
+ register_list[register_count++] = 1; |
+ register_bits = register_bits >> 1; |
+ } |
+ record_buf[register_count] = ARM_PS_REGNUM; |
+ record_buf[register_count + 1] = ARM_SP_REGNUM; |
+ thumb_insn_r->reg_rec_count = register_count + 2; |
+ for (register_count = 0; register_count < 8; register_count++) |
+ { |
+ if (register_list[register_count]) |
+ { |
+ record_buf[index] = register_count; |
+ index++; |
+ } |
+ } |
+ } |
+ else if (10 == opcode2) |
+ { |
+ /* PUSH. */ |
+ register_bits = bits (thumb_insn_r->arm_insn, 0, 7); |
+ regcache_raw_read_unsigned (reg_cache, ARM_PC_REGNUM, &u_regval); |
+ while (register_bits) |
+ { |
+ if (register_bits & 0x00000001) |
+ register_count++; |
+ register_bits = register_bits >> 1; |
+ } |
+ start_address = u_regval - \ |
+ (4 * (bit (thumb_insn_r->arm_insn, 8) + register_count)); |
+ thumb_insn_r->mem_rec_count = register_count; |
+ while (register_count) |
+ { |
+ record_buf_mem[(register_count * 2) - 1] = start_address; |
+ record_buf_mem[(register_count * 2) - 2] = 4; |
+ start_address = start_address + 4; |
+ register_count--; |
+ } |
+ record_buf[0] = ARM_SP_REGNUM; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (0x1E == opcode1) |
+ { |
+ /* BKPT insn. */ |
+ /* Handle enhanced software breakpoint insn, BKPT. */ |
+ /* CPSR is changed to be executed in ARM state, disabling normal |
+ interrupts, entering abort mode. */ |
+ /* According to high vector configuration PC is set. */ |
+ /* User hits breakpoint and type reverse, in that case, we need to go back with |
+ previous CPSR and Program Counter. */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ thumb_insn_r->reg_rec_count = 2; |
+ /* We need to save SPSR value, which is not yet done. */ |
+ printf_unfiltered (_("Process record does not support instruction " |
+ "0x%0x at address %s.\n"), |
+ thumb_insn_r->arm_insn, |
+ paddress (thumb_insn_r->gdbarch, |
+ thumb_insn_r->this_addr)); |
+ return -1; |
+ } |
+ else if ((0 == opcode) || (1 == opcode)) |
+ { |
+ /* ADD(5), ADD(6). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ record_buf[0] = reg_src1; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (2 == opcode) |
+ { |
+ /* ADD(7), SUB(4). */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ record_buf[0] = ARM_SP_REGNUM; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (thumb_insn_r->arm_mems, thumb_insn_r->mem_rec_count, |
+ record_buf_mem); |
+ |
+ return 0; |
+} |
+ |
+/* Handling opcode 110 insns. */ |
+ |
+static int |
+thumb_record_ldm_stm_swi (insn_decode_record *thumb_insn_r) |
+{ |
+ struct gdbarch_tdep *tdep = gdbarch_tdep (thumb_insn_r->gdbarch); |
+ struct regcache *reg_cache = thumb_insn_r->regcache; |
+ |
+ uint32_t ret = 0; /* function return value: -1:record failure ; 0:success */ |
+ uint32_t reg_src1 = 0; |
+ uint32_t opcode1 = 0, opcode2 = 0, register_bits = 0, register_count = 0; |
+ uint32_t register_list[8] = {0}, index = 0, start_address = 0; |
+ uint32_t record_buf[24], record_buf_mem[48]; |
+ |
+ ULONGEST u_regval = 0; |
+ |
+ opcode1 = bits (thumb_insn_r->arm_insn, 8, 12); |
+ opcode2 = bits (thumb_insn_r->arm_insn, 11, 12); |
+ |
+ if (1 == opcode2) |
+ { |
+ |
+ /* LDMIA. */ |
+ register_bits = bits (thumb_insn_r->arm_insn, 0, 7); |
+ /* Get Rn. */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ while (register_bits) |
+ { |
+ if (register_bits & 0x00000001) |
+ register_list[register_count++] = 1; |
+ register_bits = register_bits >> 1; |
+ } |
+ record_buf[register_count] = reg_src1; |
+ thumb_insn_r->reg_rec_count = register_count + 1; |
+ for (register_count = 0; register_count < 8; register_count++) |
+ { |
+ if (register_list[register_count]) |
+ { |
+ record_buf[index] = register_count; |
+ index++; |
+ } |
+ } |
+ } |
+ else if (0 == opcode2) |
+ { |
+ /* It handles both STMIA. */ |
+ register_bits = bits (thumb_insn_r->arm_insn, 0, 7); |
+ /* Get Rn. */ |
+ reg_src1 = bits (thumb_insn_r->arm_insn, 8, 10); |
+ regcache_raw_read_unsigned (reg_cache, reg_src1, &u_regval); |
+ while (register_bits) |
+ { |
+ if (register_bits & 0x00000001) |
+ register_count++; |
+ register_bits = register_bits >> 1; |
+ } |
+ start_address = u_regval; |
+ thumb_insn_r->mem_rec_count = register_count; |
+ while (register_count) |
+ { |
+ record_buf_mem[(register_count * 2) - 1] = start_address; |
+ record_buf_mem[(register_count * 2) - 2] = 4; |
+ start_address = start_address + 4; |
+ register_count--; |
+ } |
+ } |
+ else if (0x1F == opcode1) |
+ { |
+ /* Handle arm syscall insn. */ |
+ if (tdep->arm_swi_record != NULL) |
+ { |
+ ret = tdep->arm_swi_record(reg_cache); |
+ } |
+ else |
+ { |
+ printf_unfiltered (_("no syscall record support\n")); |
+ return -1; |
+ } |
+ } |
+ |
+ /* B (1), conditional branch is automatically taken care in process_record, |
+ as PC is saved there. */ |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ MEM_ALLOC (thumb_insn_r->arm_mems, thumb_insn_r->mem_rec_count, |
+ record_buf_mem); |
+ |
+ return ret; |
+} |
+ |
+/* Handling opcode 111 insns. */ |
+ |
+static int |
+thumb_record_branch (insn_decode_record *thumb_insn_r) |
+{ |
+ uint32_t record_buf[8]; |
+ uint32_t bits_h = 0; |
+ |
+ bits_h = bits (thumb_insn_r->arm_insn, 11, 12); |
+ |
+ if (2 == bits_h || 3 == bits_h) |
+ { |
+ /* BL */ |
+ record_buf[0] = ARM_LR_REGNUM; |
+ thumb_insn_r->reg_rec_count = 1; |
+ } |
+ else if (1 == bits_h) |
+ { |
+ /* BLX(1). */ |
+ record_buf[0] = ARM_PS_REGNUM; |
+ record_buf[1] = ARM_LR_REGNUM; |
+ thumb_insn_r->reg_rec_count = 2; |
+ } |
+ |
+ /* B(2) is automatically taken care in process_record, as PC is |
+ saved there. */ |
+ |
+ REG_ALLOC (thumb_insn_r->arm_regs, thumb_insn_r->reg_rec_count, record_buf); |
+ |
+ return 0; |
+} |
+ |
+ |
+/* Extracts arm/thumb/thumb2 insn depending on the size, and returns 0 on success |
+and positive val on fauilure. */ |
+ |
+static int |
+extract_arm_insn (insn_decode_record *insn_record, uint32_t insn_size) |
+{ |
+ gdb_byte buf[insn_size]; |
+ |
+ memset (&buf[0], 0, insn_size); |
+ |
+ if (target_read_memory (insn_record->this_addr, &buf[0], insn_size)) |
+ return 1; |
+ insn_record->arm_insn = (uint32_t) extract_unsigned_integer (&buf[0], |
+ insn_size, |
+ gdbarch_byte_order (insn_record->gdbarch)); |
+ return 0; |
+} |
+ |
+typedef int (*sti_arm_hdl_fp_t) (insn_decode_record*); |
+ |
+/* Decode arm/thumb insn depending on condition cods and opcodes; and |
+ dispatch it. */ |
+ |
+static int |
+decode_insn (insn_decode_record *arm_record, record_type_t record_type, |
+ uint32_t insn_size) |
+{ |
+ |
+ /* (Starting from numerical 0); bits 25, 26, 27 decodes type of arm instruction. */ |
+ static const sti_arm_hdl_fp_t const arm_handle_insn[8] = |
+ { |
+ arm_record_data_proc_misc_ld_str, /* 000. */ |
+ arm_record_data_proc_imm, /* 001. */ |
+ arm_record_ld_st_imm_offset, /* 010. */ |
+ arm_record_ld_st_reg_offset, /* 011. */ |
+ arm_record_ld_st_multiple, /* 100. */ |
+ arm_record_b_bl, /* 101. */ |
+ arm_record_coproc, /* 110. */ |
+ arm_record_coproc_data_proc /* 111. */ |
+ }; |
+ |
+ /* (Starting from numerical 0); bits 13,14,15 decodes type of thumb instruction. */ |
+ static const sti_arm_hdl_fp_t const thumb_handle_insn[8] = |
+ { \ |
+ thumb_record_shift_add_sub, /* 000. */ |
+ thumb_record_add_sub_cmp_mov, /* 001. */ |
+ thumb_record_ld_st_reg_offset, /* 010. */ |
+ thumb_record_ld_st_imm_offset, /* 011. */ |
+ thumb_record_ld_st_stack, /* 100. */ |
+ thumb_record_misc, /* 101. */ |
+ thumb_record_ldm_stm_swi, /* 110. */ |
+ thumb_record_branch /* 111. */ |
+ }; |
+ |
+ uint32_t ret = 0; /* return value: negative:failure 0:success. */ |
+ uint32_t insn_id = 0; |
+ |
+ if (extract_arm_insn (arm_record, insn_size)) |
+ { |
+ if (record_debug) |
+ { |
+ printf_unfiltered (_("Process record: error reading memory at " |
+ "addr %s len = %d.\n"), |
+ paddress (arm_record->gdbarch, arm_record->this_addr), insn_size); |
+ } |
+ return -1; |
+ } |
+ else if (ARM_RECORD == record_type) |
+ { |
+ arm_record->cond = bits (arm_record->arm_insn, 28, 31); |
+ insn_id = bits (arm_record->arm_insn, 25, 27); |
+ ret = arm_record_extension_space (arm_record); |
+ /* If this insn has fallen into extension space |
+ then we need not decode it anymore. */ |
+ if (ret != -1 && !INSN_RECORDED(arm_record)) |
+ { |
+ ret = arm_handle_insn[insn_id] (arm_record); |
+ } |
+ } |
+ else if (THUMB_RECORD == record_type) |
+ { |
+ /* As thumb does not have condition codes, we set negative. */ |
+ arm_record->cond = -1; |
+ insn_id = bits (arm_record->arm_insn, 13, 15); |
+ ret = thumb_handle_insn[insn_id] (arm_record); |
+ } |
+ else if (THUMB2_RECORD == record_type) |
+ { |
+ printf_unfiltered (_("Process record doesnt support thumb32 instruction " |
+ "0x%0x at address %s.\n"),arm_record->arm_insn, |
+ paddress (arm_record->gdbarch, |
+ arm_record->this_addr)); |
+ ret = -1; |
+ } |
+ else |
+ { |
+ /* Throw assertion. */ |
+ gdb_assert_not_reached ("not a valid instruction, could not decode"); |
+ } |
+ |
+ return ret; |
+} |
+ |
+ |
+/* Cleans up local record registers and memory allocations. */ |
+ |
+static void |
+deallocate_reg_mem (insn_decode_record *record) |
+{ |
+ xfree (record->arm_regs); |
+ xfree (record->arm_mems); |
+} |
+ |
+ |
+/* Parse the current instruction and record the values of the registers and |
+ memory that will be changed in current instruction to record_arch_list". |
+ Return -1 if something is wrong. */ |
+ |
+int |
+arm_process_record (struct gdbarch *gdbarch, struct regcache *regcache, |
+ CORE_ADDR insn_addr) |
+{ |
+ |
+ enum bfd_endian byte_order = gdbarch_byte_order (gdbarch); |
+ uint32_t no_of_rec = 0; |
+ uint32_t ret = 0; /* return value: -1:record failure ; 0:success */ |
+ ULONGEST t_bit = 0, insn_id = 0; |
+ |
+ ULONGEST u_regval = 0; |
+ |
+ insn_decode_record arm_record; |
+ |
+ memset (&arm_record, 0, sizeof (insn_decode_record)); |
+ arm_record.regcache = regcache; |
+ arm_record.this_addr = insn_addr; |
+ arm_record.gdbarch = gdbarch; |
+ |
+ |
+ if (record_debug > 1) |
+ { |
+ fprintf_unfiltered (gdb_stdlog, "Process record: arm_process_record " |
+ "addr = %s\n", |
+ paddress (gdbarch, arm_record.this_addr)); |
+ } |
+ |
+ if (extract_arm_insn (&arm_record, 2)) |
+ { |
+ if (record_debug) |
+ { |
+ printf_unfiltered (_("Process record: error reading memory at " |
+ "addr %s len = %d.\n"), |
+ paddress (arm_record.gdbarch, |
+ arm_record.this_addr), 2); |
+ } |
+ return -1; |
+ } |
+ |
+ /* Check the insn, whether it is thumb or arm one. */ |
+ |
+ t_bit = arm_psr_thumb_bit (arm_record.gdbarch); |
+ regcache_raw_read_unsigned (arm_record.regcache, ARM_PS_REGNUM, &u_regval); |
+ |
+ |
+ if (!(u_regval & t_bit)) |
+ { |
+ /* We are decoding arm insn. */ |
+ ret = decode_insn (&arm_record, ARM_RECORD, ARM_INSN_SIZE_BYTES); |
+ } |
+ else |
+ { |
+ insn_id = bits (arm_record.arm_insn, 11, 15); |
+ /* is it thumb2 insn? */ |
+ if ((0x1D == insn_id) || (0x1E == insn_id) || (0x1F == insn_id)) |
+ { |
+ ret = decode_insn (&arm_record, THUMB2_RECORD, |
+ THUMB2_INSN_SIZE_BYTES); |
+ } |
+ else |
+ { |
+ /* We are decoding thumb insn. */ |
+ ret = decode_insn (&arm_record, THUMB_RECORD, THUMB_INSN_SIZE_BYTES); |
+ } |
+ } |
+ |
+ if (0 == ret) |
+ { |
+ /* Record registers. */ |
+ record_arch_list_add_reg (arm_record.regcache, ARM_PC_REGNUM); |
+ if (arm_record.arm_regs) |
+ { |
+ for (no_of_rec = 0; no_of_rec < arm_record.reg_rec_count; no_of_rec++) |
+ { |
+ if (record_arch_list_add_reg (arm_record.regcache , |
+ arm_record.arm_regs[no_of_rec])) |
+ ret = -1; |
+ } |
+ } |
+ /* Record memories. */ |
+ if (arm_record.arm_mems) |
+ { |
+ for (no_of_rec = 0; no_of_rec < arm_record.mem_rec_count; no_of_rec++) |
+ { |
+ if (record_arch_list_add_mem |
+ ((CORE_ADDR)arm_record.arm_mems[no_of_rec].addr, |
+ arm_record.arm_mems[no_of_rec].len)) |
+ ret = -1; |
+ } |
+ } |
+ |
+ if (record_arch_list_add_end ()) |
+ ret = -1; |
+ } |
+ |
+ |
+ deallocate_reg_mem (&arm_record); |
+ |
+ return ret; |
+} |
+ |