2022-08-30 10:01:12 -05:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2022-06-26 18:24:07 -05:00
|
|
|
|
2010-06-15 16:18:44 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/***************************************************************************
|
|
|
|
* Copyright (C) 2005 by Dominic Rath *
|
|
|
|
* Dominic.Rath@gmx.de *
|
|
|
|
***************************************************************************/
|
2012-02-05 06:03:04 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "arm920t.h"
|
2009-12-03 06:14:29 -06:00
|
|
|
#include <helper/time_support.h>
|
2009-05-31 07:38:28 -05:00
|
|
|
#include "target_type.h"
|
2009-11-16 02:35:14 -06:00
|
|
|
#include "register.h"
|
2009-12-07 16:54:12 -06:00
|
|
|
#include "arm_opcodes.h"
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
/*
|
|
|
|
* For information about the ARM920T, see ARM DDI 0151C especially
|
|
|
|
* Chapter 9 about debug support, which shows how to manipulate each
|
|
|
|
* of the different scan chains:
|
|
|
|
*
|
|
|
|
* 0 ... ARM920 signals, e.g. to rest of SOC (unused here)
|
|
|
|
* 1 ... debugging; watchpoint and breakpoint status, etc; also
|
|
|
|
* MMU and cache access in conjunction with scan chain 15
|
|
|
|
* 2 ... EmbeddedICE
|
|
|
|
* 3 ... external boundary scan (SoC-specific, unused here)
|
|
|
|
* 4 ... access to cache tag RAM
|
|
|
|
* 6 ... ETM9
|
|
|
|
* 15 ... access coprocessor 15, "physical" or "interpreted" modes
|
|
|
|
* "interpreted" works with a few actual MRC/MCR instructions
|
2010-02-16 20:50:16 -06:00
|
|
|
* "physical" provides register-like behaviors. Section 9.6.7
|
|
|
|
* covers these details.
|
2009-11-05 22:35:37 -06:00
|
|
|
*
|
|
|
|
* The ARM922T is similar, but with smaller caches (8K each, vs 16K).
|
|
|
|
*/
|
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
#if 0
|
|
|
|
#define _DEBUG_INSTRUCTION_EXECUTION_
|
|
|
|
#endif
|
|
|
|
|
2010-02-16 20:50:16 -06:00
|
|
|
/* Table 9-8 shows scan chain 15 format during physical access mode, using a
|
|
|
|
* dedicated 6-bit address space (encoded in bits 33:38). Writes use one
|
|
|
|
* JTAG scan, while reads use two.
|
|
|
|
*
|
|
|
|
* Table 9-9 lists the thirteen registers which support physical access.
|
|
|
|
* ARM920T_CP15_PHYS_ADDR() constructs the 6-bit reg_addr parameter passed
|
|
|
|
* to arm920t_read_cp15_physical() and arm920t_write_cp15_physical().
|
|
|
|
*
|
|
|
|
* x == bit[38]
|
|
|
|
* y == bits[37:34]
|
|
|
|
* z == bit[33]
|
|
|
|
*/
|
2008-02-25 11:48:04 -06:00
|
|
|
#define ARM920T_CP15_PHYS_ADDR(x, y, z) ((x << 5) | (y << 1) << (z))
|
|
|
|
|
2010-02-16 20:50:16 -06:00
|
|
|
/* Registers supporting physical Read access (from table 9-9) */
|
2012-02-05 06:03:04 -06:00
|
|
|
#define CP15PHYS_CACHETYPE ARM920T_CP15_PHYS_ADDR(0, 0x0, 1)
|
|
|
|
#define CP15PHYS_ICACHE_IDX ARM920T_CP15_PHYS_ADDR(1, 0xd, 1)
|
|
|
|
#define CP15PHYS_DCACHE_IDX ARM920T_CP15_PHYS_ADDR(1, 0xe, 1)
|
2010-02-16 20:50:16 -06:00
|
|
|
/* NOTE: several more registers support only physical read access */
|
|
|
|
|
|
|
|
/* Registers supporting physical Read/Write access (from table 9-9) */
|
2012-02-05 06:03:04 -06:00
|
|
|
#define CP15PHYS_CTRL ARM920T_CP15_PHYS_ADDR(0, 0x1, 0)
|
|
|
|
#define CP15PHYS_PID ARM920T_CP15_PHYS_ADDR(0, 0xd, 0)
|
|
|
|
#define CP15PHYS_TESTSTATE ARM920T_CP15_PHYS_ADDR(0, 0xf, 0)
|
|
|
|
#define CP15PHYS_ICACHE ARM920T_CP15_PHYS_ADDR(1, 0x1, 1)
|
|
|
|
#define CP15PHYS_DCACHE ARM920T_CP15_PHYS_ADDR(1, 0x2, 1)
|
2010-02-16 20:50:16 -06:00
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_read_cp15_physical(struct target *target,
|
2012-02-05 06:03:04 -06:00
|
|
|
int reg_addr, uint32_t *value)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-13 10:41:00 -06:00
|
|
|
struct arm_jtag *jtag_info;
|
2009-11-13 05:28:03 -06:00
|
|
|
struct scan_field fields[4];
|
2009-06-18 02:04:08 -05:00
|
|
|
uint8_t access_type_buf = 1;
|
|
|
|
uint8_t reg_addr_buf = reg_addr & 0x3f;
|
|
|
|
uint8_t nr_w_buf = 0;
|
2010-07-19 07:37:45 -05:00
|
|
|
int retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2009-11-17 03:09:50 -06:00
|
|
|
jtag_info = &arm920t->arm7_9_common.jtag_info;
|
2009-11-06 00:03:13 -06:00
|
|
|
|
2010-07-20 01:23:59 -05:00
|
|
|
retval = arm_jtag_scann(jtag_info, 0xf, TAP_IDLE);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2015-11-13 17:30:28 -06:00
|
|
|
retval = arm_jtag_set_instr(jtag_info->tap, jtag_info->intest_instr, NULL, TAP_IDLE);
|
2010-07-19 07:37:45 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
fields[0].num_bits = 1;
|
|
|
|
fields[0].out_value = &access_type_buf;
|
|
|
|
fields[0].in_value = NULL;
|
|
|
|
|
|
|
|
fields[1].num_bits = 32;
|
|
|
|
fields[1].out_value = NULL;
|
|
|
|
fields[1].in_value = NULL;
|
|
|
|
|
|
|
|
fields[2].num_bits = 6;
|
|
|
|
fields[2].out_value = ®_addr_buf;
|
|
|
|
fields[2].in_value = NULL;
|
|
|
|
|
|
|
|
fields[3].num_bits = 1;
|
|
|
|
fields[3].out_value = &nr_w_buf;
|
|
|
|
fields[3].in_value = NULL;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2010-03-16 08:13:03 -05:00
|
|
|
jtag_add_dr_scan(jtag_info->tap, 4, fields, TAP_IDLE);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2009-06-18 02:04:08 -05:00
|
|
|
fields[1].in_value = (uint8_t *)value;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2010-03-16 08:13:03 -05:00
|
|
|
jtag_add_dr_scan(jtag_info->tap, 4, fields, TAP_IDLE);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2009-06-19 03:18:36 -05:00
|
|
|
jtag_add_callback(arm_le_to_h_u32, (jtag_callback_data_t)value);
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
#ifdef _DEBUG_INSTRUCTION_EXECUTION_
|
2008-02-25 11:48:04 -06:00
|
|
|
jtag_execute_queue();
|
2008-03-25 10:45:17 -05:00
|
|
|
LOG_DEBUG("addr: 0x%x value: %8.8x", reg_addr, *value);
|
2008-02-25 11:48:04 -06:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_write_cp15_physical(struct target *target,
|
2012-02-05 06:03:04 -06:00
|
|
|
int reg_addr, uint32_t value)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-13 10:41:00 -06:00
|
|
|
struct arm_jtag *jtag_info;
|
2009-11-13 05:28:03 -06:00
|
|
|
struct scan_field fields[4];
|
2009-06-18 02:04:08 -05:00
|
|
|
uint8_t access_type_buf = 1;
|
|
|
|
uint8_t reg_addr_buf = reg_addr & 0x3f;
|
|
|
|
uint8_t nr_w_buf = 1;
|
|
|
|
uint8_t value_buf[4];
|
2010-07-19 07:37:45 -05:00
|
|
|
int retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2009-11-17 03:09:50 -06:00
|
|
|
jtag_info = &arm920t->arm7_9_common.jtag_info;
|
2009-11-06 00:03:13 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
buf_set_u32(value_buf, 0, 32, value);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2010-07-20 01:23:59 -05:00
|
|
|
retval = arm_jtag_scann(jtag_info, 0xf, TAP_IDLE);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2015-11-13 17:30:28 -06:00
|
|
|
retval = arm_jtag_set_instr(jtag_info->tap, jtag_info->intest_instr, NULL, TAP_IDLE);
|
2010-07-19 07:37:45 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
fields[0].num_bits = 1;
|
|
|
|
fields[0].out_value = &access_type_buf;
|
|
|
|
fields[0].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fields[1].num_bits = 32;
|
|
|
|
fields[1].out_value = value_buf;
|
|
|
|
fields[1].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fields[2].num_bits = 6;
|
|
|
|
fields[2].out_value = ®_addr_buf;
|
|
|
|
fields[2].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fields[3].num_bits = 1;
|
|
|
|
fields[3].out_value = &nr_w_buf;
|
|
|
|
fields[3].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2010-03-16 08:13:03 -05:00
|
|
|
jtag_add_dr_scan(jtag_info->tap, 4, fields, TAP_IDLE);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
#ifdef _DEBUG_INSTRUCTION_EXECUTION_
|
2008-03-25 10:45:17 -05:00
|
|
|
LOG_DEBUG("addr: 0x%x value: %8.8x", reg_addr, value);
|
2008-02-25 11:48:04 -06:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2010-02-16 20:50:16 -06:00
|
|
|
/* See table 9-10 for scan chain 15 format during interpreted access mode.
|
|
|
|
* If the TESTSTATE register is set for interpreted access, certain CP15
|
|
|
|
* MRC and MCR instructions may be executed through scan chain 15.
|
|
|
|
*
|
|
|
|
* Tables 9-11, 9-12, and 9-13 show which MRC and MCR instructions can be
|
|
|
|
* executed using scan chain 15 interpreted mode.
|
|
|
|
*/
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_execute_cp15(struct target *target, uint32_t cp15_opcode,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t arm_opcode)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2008-10-15 06:44:36 -05:00
|
|
|
int retval;
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-13 10:41:00 -06:00
|
|
|
struct arm_jtag *jtag_info;
|
2009-11-13 05:28:03 -06:00
|
|
|
struct scan_field fields[4];
|
2009-06-18 02:04:08 -05:00
|
|
|
uint8_t access_type_buf = 0; /* interpreted access */
|
|
|
|
uint8_t reg_addr_buf = 0x0;
|
|
|
|
uint8_t nr_w_buf = 0;
|
|
|
|
uint8_t cp15_opcode_buf[4];
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2009-11-17 03:09:50 -06:00
|
|
|
jtag_info = &arm920t->arm7_9_common.jtag_info;
|
2009-11-06 00:03:13 -06:00
|
|
|
|
2010-07-20 01:23:59 -05:00
|
|
|
retval = arm_jtag_scann(jtag_info, 0xf, TAP_IDLE);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2015-11-13 17:30:28 -06:00
|
|
|
retval = arm_jtag_set_instr(jtag_info->tap, jtag_info->intest_instr, NULL, TAP_IDLE);
|
2010-07-19 07:37:45 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
buf_set_u32(cp15_opcode_buf, 0, 32, cp15_opcode);
|
|
|
|
|
|
|
|
fields[0].num_bits = 1;
|
|
|
|
fields[0].out_value = &access_type_buf;
|
|
|
|
fields[0].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fields[1].num_bits = 32;
|
|
|
|
fields[1].out_value = cp15_opcode_buf;
|
|
|
|
fields[1].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fields[2].num_bits = 6;
|
|
|
|
fields[2].out_value = ®_addr_buf;
|
|
|
|
fields[2].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fields[3].num_bits = 1;
|
|
|
|
fields[3].out_value = &nr_w_buf;
|
|
|
|
fields[3].in_value = NULL;
|
2009-05-07 08:31:34 -05:00
|
|
|
|
2010-03-16 08:13:03 -05:00
|
|
|
jtag_add_dr_scan(jtag_info->tap, 4, fields, TAP_IDLE);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
arm9tdmi_clock_out(jtag_info, arm_opcode, 0, NULL, 0);
|
|
|
|
arm9tdmi_clock_out(jtag_info, ARMV4_5_NOP, 0, NULL, 1);
|
2008-11-04 11:31:23 -06:00
|
|
|
retval = arm7_9_execute_sys_speed(target);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK) {
|
2009-11-06 00:03:13 -06:00
|
|
|
LOG_ERROR("failed executing JTAG queue");
|
2008-10-15 06:44:36 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_read_cp15_interpreted(struct target *target,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t cp15_opcode, uint32_t address, uint32_t *value)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2012-01-19 04:06:37 -06:00
|
|
|
struct arm *arm = target_to_arm(target);
|
2021-03-12 11:38:02 -06:00
|
|
|
uint32_t *regs_p[16];
|
|
|
|
uint32_t regs[16];
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15c15 = 0x0;
|
2012-01-19 04:06:37 -06:00
|
|
|
struct reg *r = arm->core_cache->reg_list;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* load address into R1 */
|
|
|
|
regs[1] = address;
|
2008-12-13 00:25:50 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
|
|
|
|
|
|
|
/* read-modify-write CP15 test state register
|
2008-02-25 11:48:04 -06:00
|
|
|
* to enable interpreted access mode */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_TESTSTATE, &cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
jtag_execute_queue();
|
|
|
|
cp15c15 |= 1; /* set interpret mode */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* execute CP15 instruction and ARM load (reading from coprocessor) */
|
|
|
|
arm920t_execute_cp15(target, cp15_opcode, ARMV4_5_LDR(0, 1));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* disable interpreted access mode */
|
|
|
|
cp15c15 &= ~1U; /* clear interpret mode */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* retrieve value from R0 */
|
|
|
|
regs_p[0] = value;
|
|
|
|
arm9tdmi_read_core_regs(target, 0x1, regs_p);
|
|
|
|
jtag_execute_queue();
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
#ifdef _DEBUG_INSTRUCTION_EXECUTION_
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("cp15_opcode: %8.8x, address: %8.8x, value: %8.8x",
|
2012-02-05 06:03:04 -06:00
|
|
|
cp15_opcode, address, *value);
|
2008-02-25 11:48:04 -06:00
|
|
|
#endif
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (!is_arm_mode(arm->core_mode)) {
|
2010-08-11 03:58:49 -05:00
|
|
|
LOG_ERROR("not a valid arm core mode - communication failure?");
|
2008-05-19 14:02:36 -05:00
|
|
|
return ERROR_FAIL;
|
2010-08-11 03:58:49 -05:00
|
|
|
}
|
2008-05-19 14:02:36 -05:00
|
|
|
|
2019-02-26 07:05:43 -06:00
|
|
|
r[0].dirty = true;
|
|
|
|
r[1].dirty = true;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
static
|
2009-11-13 12:11:13 -06:00
|
|
|
int arm920t_write_cp15_interpreted(struct target *target,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t cp15_opcode, uint32_t value, uint32_t address)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15c15 = 0x0;
|
2012-01-19 04:06:37 -06:00
|
|
|
struct arm *arm = target_to_arm(target);
|
2021-03-12 11:38:02 -06:00
|
|
|
uint32_t regs[16];
|
2012-01-19 04:06:37 -06:00
|
|
|
struct reg *r = arm->core_cache->reg_list;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* load value, address into R0, R1 */
|
|
|
|
regs[0] = value;
|
|
|
|
regs[1] = address;
|
|
|
|
arm9tdmi_write_core_regs(target, 0x3, regs);
|
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
/* read-modify-write CP15 test state register
|
2008-02-25 11:48:04 -06:00
|
|
|
* to enable interpreted access mode */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_TESTSTATE, &cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
jtag_execute_queue();
|
|
|
|
cp15c15 |= 1; /* set interpret mode */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* execute CP15 instruction and ARM store (writing to coprocessor) */
|
|
|
|
arm920t_execute_cp15(target, cp15_opcode, ARMV4_5_STR(0, 1));
|
|
|
|
|
|
|
|
/* disable interpreted access mode */
|
|
|
|
cp15c15 &= ~1U; /* set interpret mode */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
#ifdef _DEBUG_INSTRUCTION_EXECUTION_
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("cp15_opcode: %8.8x, value: %8.8x, address: %8.8x",
|
2012-02-05 06:03:04 -06:00
|
|
|
cp15_opcode, value, address);
|
2008-02-25 11:48:04 -06:00
|
|
|
#endif
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (!is_arm_mode(arm->core_mode)) {
|
2010-08-11 03:58:49 -05:00
|
|
|
LOG_ERROR("not a valid arm core mode - communication failure?");
|
2008-05-19 14:02:36 -05:00
|
|
|
return ERROR_FAIL;
|
2010-08-11 03:58:49 -05:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2019-02-26 07:05:43 -06:00
|
|
|
r[0].dirty = true;
|
|
|
|
r[1].dirty = true;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* EXPORTED to FA256 */
|
2010-07-19 01:45:45 -05:00
|
|
|
int arm920t_get_ttb(struct target *target, uint32_t *result)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
|
|
|
int retval;
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t ttb = 0x0;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = arm920t_read_cp15_interpreted(target,
|
2010-02-20 22:47:38 -06:00
|
|
|
/* FIXME use opcode macro */
|
2012-02-05 06:03:04 -06:00
|
|
|
0xeebf0f51, 0x0, &ttb);
|
|
|
|
if (retval != ERROR_OK)
|
2008-02-25 11:48:04 -06:00
|
|
|
return retval;
|
|
|
|
|
2010-07-19 01:45:45 -05:00
|
|
|
*result = ttb;
|
|
|
|
return ERROR_OK;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* EXPORTED to FA256 */
|
2010-07-19 03:58:07 -05:00
|
|
|
int arm920t_disable_mmu_caches(struct target *target, int mmu,
|
2012-02-05 06:03:04 -06:00
|
|
|
int d_u_cache, int i_cache)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15_control;
|
2010-07-19 03:58:07 -05:00
|
|
|
int retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* read cp15 control register */
|
2010-07-19 03:58:07 -05:00
|
|
|
retval = arm920t_read_cp15_physical(target, CP15PHYS_CTRL, &cp15_control);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
if (mmu)
|
|
|
|
cp15_control &= ~0x1U;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
if (d_u_cache)
|
|
|
|
cp15_control &= ~0x4U;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
if (i_cache)
|
|
|
|
cp15_control &= ~0x1000U;
|
|
|
|
|
2010-07-19 03:58:07 -05:00
|
|
|
retval = arm920t_write_cp15_physical(target, CP15PHYS_CTRL, cp15_control);
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* EXPORTED to FA256 */
|
2010-07-19 03:58:07 -05:00
|
|
|
int arm920t_enable_mmu_caches(struct target *target, int mmu,
|
2012-02-05 06:03:04 -06:00
|
|
|
int d_u_cache, int i_cache)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15_control;
|
2010-07-19 03:58:07 -05:00
|
|
|
int retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* read cp15 control register */
|
2010-07-19 03:58:07 -05:00
|
|
|
retval = arm920t_read_cp15_physical(target, CP15PHYS_CTRL, &cp15_control);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
if (mmu)
|
|
|
|
cp15_control |= 0x1U;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
if (d_u_cache)
|
|
|
|
cp15_control |= 0x4U;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
if (i_cache)
|
|
|
|
cp15_control |= 0x1000U;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2010-07-19 03:58:07 -05:00
|
|
|
retval = arm920t_write_cp15_physical(target, CP15PHYS_CTRL, cp15_control);
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* EXPORTED to FA256 */
|
2010-07-19 05:34:54 -05:00
|
|
|
int arm920t_post_debug_entry(struct target *target)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15c15;
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2010-07-19 05:34:54 -05:00
|
|
|
int retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* examine cp15 control reg */
|
2010-07-19 05:34:54 -05:00
|
|
|
retval = arm920t_read_cp15_physical(target,
|
2010-02-16 20:50:16 -06:00
|
|
|
CP15PHYS_CTRL, &arm920t->cp15_control_reg);
|
2010-07-19 05:34:54 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("cp15_control_reg: %8.8" PRIx32, arm920t->cp15_control_reg);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (arm920t->armv4_5_mmu.armv4_5_cache.ctype == -1) {
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cache_type_reg;
|
2008-02-25 11:48:04 -06:00
|
|
|
/* identify caches */
|
2010-07-19 05:34:54 -05:00
|
|
|
retval = arm920t_read_cp15_physical(target,
|
2010-02-16 20:50:16 -06:00
|
|
|
CP15PHYS_CACHETYPE, &cache_type_reg);
|
2010-07-19 05:34:54 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2010-02-20 22:47:38 -06:00
|
|
|
armv4_5_identify_cache(cache_type_reg,
|
2012-02-05 06:03:04 -06:00
|
|
|
&arm920t->armv4_5_mmu.armv4_5_cache);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t->armv4_5_mmu.mmu_enabled =
|
2012-02-05 06:03:04 -06:00
|
|
|
(arm920t->cp15_control_reg & 0x1U) ? 1 : 0;
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t->armv4_5_mmu.armv4_5_cache.d_u_cache_enabled =
|
2012-02-05 06:03:04 -06:00
|
|
|
(arm920t->cp15_control_reg & 0x4U) ? 1 : 0;
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t->armv4_5_mmu.armv4_5_cache.i_cache_enabled =
|
2012-02-05 06:03:04 -06:00
|
|
|
(arm920t->cp15_control_reg & 0x1000U) ? 1 : 0;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* save i/d fault status and address register
|
|
|
|
* FIXME use opcode macros */
|
2010-07-19 05:34:54 -05:00
|
|
|
retval = arm920t_read_cp15_interpreted(target, 0xee150f10, 0x0, &arm920t->d_fsr);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = arm920t_read_cp15_interpreted(target, 0xee150f30, 0x0, &arm920t->i_fsr);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = arm920t_read_cp15_interpreted(target, 0xee160f10, 0x0, &arm920t->d_far);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = arm920t_read_cp15_interpreted(target, 0xee160f30, 0x0, &arm920t->i_far);
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("D FSR: 0x%8.8" PRIx32 ", D FAR: 0x%8.8" PRIx32
|
|
|
|
", I FSR: 0x%8.8" PRIx32 ", I FAR: 0x%8.8" PRIx32,
|
2008-12-13 00:25:50 -06:00
|
|
|
arm920t->d_fsr, arm920t->d_far, arm920t->i_fsr, arm920t->i_far);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (arm920t->preserve_cache) {
|
2008-12-13 00:25:50 -06:00
|
|
|
/* read-modify-write CP15 test state register
|
2008-02-25 11:48:04 -06:00
|
|
|
* to disable I/D-cache linefills */
|
2010-07-19 05:34:54 -05:00
|
|
|
retval = arm920t_read_cp15_physical(target,
|
2010-02-16 20:50:16 -06:00
|
|
|
CP15PHYS_TESTSTATE, &cp15c15);
|
2010-07-19 05:34:54 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
cp15c15 |= 0x600;
|
2010-07-19 05:34:54 -05:00
|
|
|
retval = arm920t_write_cp15_physical(target,
|
2010-02-16 20:50:16 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2010-07-19 05:34:54 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2010-07-19 05:34:54 -05:00
|
|
|
return ERROR_OK;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* EXPORTED to FA256 */
|
2009-11-13 12:11:13 -06:00
|
|
|
void arm920t_pre_restore_context(struct target *target)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2019-12-20 16:43:13 -06:00
|
|
|
uint32_t cp15c15 = 0;
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* restore i/d fault status and address register */
|
|
|
|
arm920t_write_cp15_interpreted(target, 0xee050f10, arm920t->d_fsr, 0x0);
|
|
|
|
arm920t_write_cp15_interpreted(target, 0xee050f30, arm920t->i_fsr, 0x0);
|
|
|
|
arm920t_write_cp15_interpreted(target, 0xee060f10, arm920t->d_far, 0x0);
|
|
|
|
arm920t_write_cp15_interpreted(target, 0xee060f30, arm920t->i_far, 0x0);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
|
|
|
/* read-modify-write CP15 test state register
|
2008-02-25 11:48:04 -06:00
|
|
|
* to reenable I/D-cache linefills */
|
2012-02-05 06:03:04 -06:00
|
|
|
if (arm920t->preserve_cache) {
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, &cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
jtag_execute_queue();
|
|
|
|
cp15c15 &= ~0x600U;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-06 00:03:13 -06:00
|
|
|
static const char arm920_not[] = "target is not an ARM920";
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2019-03-31 21:32:12 -05:00
|
|
|
static int arm920t_verify_pointer(struct command_invocation *cmd,
|
2012-02-05 06:03:04 -06:00
|
|
|
struct arm920t_common *arm920t)
|
2009-11-06 00:03:13 -06:00
|
|
|
{
|
|
|
|
if (arm920t->common_magic != ARM920T_COMMON_MAGIC) {
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(cmd, arm920_not);
|
2009-11-06 00:03:13 -06:00
|
|
|
return ERROR_TARGET_INVALID;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
/** Logs summary of ARM920 state for a halted target. */
|
2009-11-13 12:11:13 -06:00
|
|
|
int arm920t_arch_state(struct target *target)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2012-02-05 06:03:04 -06:00
|
|
|
static const char *state[] = {
|
2008-02-25 11:48:04 -06:00
|
|
|
"disabled", "enabled"
|
|
|
|
};
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-06 00:03:13 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (arm920t->common_magic != ARM920T_COMMON_MAGIC) {
|
2009-11-06 00:03:13 -06:00
|
|
|
LOG_ERROR("BUG: %s", arm920_not);
|
|
|
|
return ERROR_TARGET_INVALID;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2009-12-07 16:55:08 -06:00
|
|
|
arm_arch_state(target);
|
|
|
|
LOG_USER("MMU: %s, D-Cache: %s, I-Cache: %s",
|
2012-02-05 06:03:04 -06:00
|
|
|
state[arm920t->armv4_5_mmu.mmu_enabled],
|
|
|
|
state[arm920t->armv4_5_mmu.armv4_5_cache.d_u_cache_enabled],
|
|
|
|
state[arm920t->armv4_5_mmu.armv4_5_cache.i_cache_enabled]);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920_mmu(struct target *target, int *enabled)
|
2009-11-10 03:36:59 -06:00
|
|
|
{
|
|
|
|
if (target->state != TARGET_HALTED) {
|
|
|
|
LOG_ERROR("%s: target not halted", __func__);
|
|
|
|
return ERROR_TARGET_INVALID;
|
|
|
|
}
|
|
|
|
|
|
|
|
*enabled = target_to_arm920(target)->armv4_5_mmu.mmu_enabled;
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920_virt2phys(struct target *target,
|
2013-09-23 03:27:03 -05:00
|
|
|
target_addr_t virt, target_addr_t *phys)
|
2009-11-10 03:36:59 -06:00
|
|
|
{
|
2010-02-12 01:23:44 -06:00
|
|
|
uint32_t cb;
|
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
|
|
|
|
2010-06-10 09:18:14 -05:00
|
|
|
uint32_t ret;
|
|
|
|
int retval = armv4_5_mmu_translate_va(target,
|
2010-06-12 05:35:06 -05:00
|
|
|
&arm920t->armv4_5_mmu, virt, &cb, &ret);
|
2010-06-10 09:18:14 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2010-02-12 01:23:44 -06:00
|
|
|
*phys = ret;
|
|
|
|
return ERROR_OK;
|
2009-11-10 03:36:59 -06:00
|
|
|
}
|
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
/** Reads a buffer, in the specified word size, with current MMU settings. */
|
2013-09-23 03:27:03 -05:00
|
|
|
int arm920t_read_memory(struct target *target, target_addr_t address,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t size, uint32_t count, uint8_t *buffer)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
|
|
|
int retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
retval = arm7_9_read_memory(target, address, size, count, buffer);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2009-10-21 08:32:29 -05:00
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_read_phys_memory(struct target *target,
|
2013-09-23 03:27:03 -05:00
|
|
|
target_addr_t address, uint32_t size,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t count, uint8_t *buffer)
|
2009-10-21 08:32:29 -05:00
|
|
|
{
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-10-21 08:32:29 -05:00
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
return armv4_5_mmu_read_physical(target, &arm920t->armv4_5_mmu,
|
2012-02-05 06:03:04 -06:00
|
|
|
address, size, count, buffer);
|
2009-10-21 08:32:29 -05:00
|
|
|
}
|
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_write_phys_memory(struct target *target,
|
2013-09-23 03:27:03 -05:00
|
|
|
target_addr_t address, uint32_t size,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t count, const uint8_t *buffer)
|
2009-10-21 08:32:29 -05:00
|
|
|
{
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-10-21 08:32:29 -05:00
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
return armv4_5_mmu_write_physical(target, &arm920t->armv4_5_mmu,
|
2012-02-05 06:03:04 -06:00
|
|
|
address, size, count, buffer);
|
2009-10-21 08:32:29 -05:00
|
|
|
}
|
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
/** Writes a buffer, in the specified word size, with current MMU settings. */
|
2013-09-23 03:27:03 -05:00
|
|
|
int arm920t_write_memory(struct target *target, target_addr_t address,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t size, uint32_t count, const uint8_t *buffer)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
|
|
|
int retval;
|
2012-02-05 06:03:04 -06:00
|
|
|
const uint32_t cache_mask = ~0x1f; /* cache line size : 32 byte */
|
2010-02-16 03:08:18 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2010-02-16 03:08:18 -06:00
|
|
|
/* FIX!!!! this should be cleaned up and made much more general. The
|
|
|
|
* plan is to write up and test on arm920t specifically and
|
2010-03-21 13:20:26 -05:00
|
|
|
* then generalize and clean up afterwards.
|
|
|
|
*
|
|
|
|
* Also it should be moved to the callbacks that handle breakpoints
|
|
|
|
* specifically and not the generic memory write fn's. See XScale code.
|
|
|
|
*/
|
2010-02-20 22:47:38 -06:00
|
|
|
if (arm920t->armv4_5_mmu.mmu_enabled && (count == 1) &&
|
2012-02-05 06:03:04 -06:00
|
|
|
((size == 2) || (size == 4))) {
|
2010-02-20 22:47:38 -06:00
|
|
|
/* special case the handling of single word writes to
|
|
|
|
* bypass MMU, to allow implementation of breakpoints
|
|
|
|
* in memory marked read only
|
|
|
|
* by MMU
|
|
|
|
*/
|
2010-02-16 03:08:18 -06:00
|
|
|
uint32_t cb;
|
|
|
|
uint32_t pa;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We need physical address and cb
|
|
|
|
*/
|
2010-06-11 22:01:24 -05:00
|
|
|
retval = armv4_5_mmu_translate_va(target, &arm920t->armv4_5_mmu,
|
2010-06-12 05:35:06 -05:00
|
|
|
address, &cb, &pa);
|
2010-06-10 09:18:14 -05:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2009-11-06 00:03:13 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (arm920t->armv4_5_mmu.armv4_5_cache.d_u_cache_enabled) {
|
|
|
|
if (cb & 0x1) {
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("D-Cache buffered, "
|
2012-02-05 06:03:04 -06:00
|
|
|
"drain write buffer");
|
2010-02-16 03:08:18 -06:00
|
|
|
/*
|
|
|
|
* Buffered ?
|
|
|
|
* Drain write buffer - MCR p15,0,Rd,c7,c10,4
|
|
|
|
*/
|
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
retval = arm920t_write_cp15_interpreted(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 7, 10, 4),
|
|
|
|
0x0, 0);
|
2010-02-16 03:08:18 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (cb == 0x3) {
|
2010-02-16 03:08:18 -06:00
|
|
|
/*
|
|
|
|
* Write back memory ? -> clean cache
|
|
|
|
*
|
2010-02-20 22:47:38 -06:00
|
|
|
* There is no way to clean cache lines using
|
|
|
|
* cp15 scan chain, so copy the full cache
|
|
|
|
* line from cache to physical memory.
|
2010-02-16 03:08:18 -06:00
|
|
|
*/
|
|
|
|
uint8_t data[32];
|
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("D-Cache in 'write back' mode, "
|
2012-02-05 06:03:04 -06:00
|
|
|
"flush cache line");
|
2010-02-16 03:08:18 -06:00
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
retval = target_read_memory(target,
|
|
|
|
address & cache_mask, 1,
|
|
|
|
sizeof(data), &data[0]);
|
2010-02-16 03:08:18 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
retval = armv4_5_mmu_write_physical(target,
|
|
|
|
&arm920t->armv4_5_mmu,
|
|
|
|
pa & cache_mask, 1,
|
|
|
|
sizeof(data), &data[0]);
|
2010-02-16 03:08:18 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Cached ? */
|
2012-02-05 06:03:04 -06:00
|
|
|
if (cb & 0x2) {
|
2010-02-16 03:08:18 -06:00
|
|
|
/*
|
|
|
|
* Cached ? -> Invalidate data cache using MVA
|
|
|
|
*
|
|
|
|
* MCR p15,0,Rd,c7,c6,1
|
|
|
|
*/
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("D-Cache enabled, "
|
|
|
|
"invalidate cache line");
|
2010-02-16 03:08:18 -06:00
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
retval = arm920t_write_cp15_interpreted(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 7, 6, 1), 0x0,
|
|
|
|
address & cache_mask);
|
2010-02-16 03:08:18 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
}
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
/* write directly to physical memory,
|
|
|
|
* bypassing any read only MMU bits, etc.
|
|
|
|
*/
|
|
|
|
retval = armv4_5_mmu_write_physical(target,
|
|
|
|
&arm920t->armv4_5_mmu, pa, size,
|
|
|
|
count, buffer);
|
2010-02-16 03:08:18 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2012-02-05 06:03:04 -06:00
|
|
|
} else {
|
|
|
|
retval = arm7_9_write_memory(target, address, size, count, buffer);
|
|
|
|
if (retval != ERROR_OK)
|
2010-02-16 03:08:18 -06:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If ICache is enabled, we have to invalidate affected ICache lines
|
2010-02-20 22:47:38 -06:00
|
|
|
* the DCache is forced to write-through,
|
|
|
|
* so we don't have to clean it here
|
2010-02-16 03:08:18 -06:00
|
|
|
*/
|
2012-02-05 06:03:04 -06:00
|
|
|
if (arm920t->armv4_5_mmu.armv4_5_cache.i_cache_enabled) {
|
|
|
|
if (count <= 1) {
|
2010-02-16 03:08:18 -06:00
|
|
|
/* invalidate ICache single entry with MVA
|
2010-02-20 22:47:38 -06:00
|
|
|
* mcr 15, 0, r0, cr7, cr5, {1}
|
2010-02-16 03:08:18 -06:00
|
|
|
*/
|
2010-02-20 22:47:38 -06:00
|
|
|
LOG_DEBUG("I-Cache enabled, "
|
|
|
|
"invalidating affected I-Cache line");
|
|
|
|
retval = arm920t_write_cp15_interpreted(target,
|
|
|
|
ARMV4_5_MCR(15, 0, 0, 7, 5, 1),
|
|
|
|
0x0, address & cache_mask);
|
2010-02-16 03:08:18 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2012-02-05 06:03:04 -06:00
|
|
|
} else {
|
2010-02-16 03:08:18 -06:00
|
|
|
/* invalidate ICache
|
2010-02-20 22:47:38 -06:00
|
|
|
* mcr 15, 0, r0, cr7, cr5, {0}
|
|
|
|
*/
|
|
|
|
retval = arm920t_write_cp15_interpreted(target,
|
|
|
|
ARMV4_5_MCR(15, 0, 0, 7, 5, 0),
|
|
|
|
0x0, 0x0);
|
2009-11-03 05:28:00 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-11 22:07:47 -05:00
|
|
|
return ERROR_OK;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
/* EXPORTED to FA256 */
|
2009-11-13 12:11:13 -06:00
|
|
|
int arm920t_soft_reset_halt(struct target *target)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2008-10-14 06:06:30 -05:00
|
|
|
int retval = ERROR_OK;
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-13 10:40:03 -06:00
|
|
|
struct arm7_9_common *arm7_9 = target_to_arm7_9(target);
|
2012-01-19 04:06:37 -06:00
|
|
|
struct arm *arm = &arm7_9->arm;
|
2009-11-13 11:55:49 -06:00
|
|
|
struct reg *dbg_stat = &arm7_9->eice_cache->reg_list[EICE_DBG_STAT];
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = target_halt(target);
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2016-05-21 21:34:04 -05:00
|
|
|
int64_t then = timeval_ms();
|
|
|
|
bool timeout;
|
2012-02-05 06:03:04 -06:00
|
|
|
while (!(timeout = ((timeval_ms()-then) > 1000))) {
|
|
|
|
if (buf_get_u32(dbg_stat->value, EICE_DBG_STATUS_DBGACK, 1) == 0) {
|
2008-04-02 01:37:08 -05:00
|
|
|
embeddedice_read_reg(dbg_stat);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-04-02 01:37:08 -05:00
|
|
|
} else
|
|
|
|
break;
|
2012-02-05 06:03:04 -06:00
|
|
|
if (debug_level >= 3) {
|
2008-08-20 02:14:45 -05:00
|
|
|
/* do not eat all CPU, time out after 1 se*/
|
|
|
|
alive_sleep(100);
|
|
|
|
} else
|
|
|
|
keep_alive();
|
2008-04-02 01:37:08 -05:00
|
|
|
}
|
2012-02-05 06:03:04 -06:00
|
|
|
if (timeout) {
|
2008-04-02 01:37:08 -05:00
|
|
|
LOG_ERROR("Failed to halt CPU after 1 sec");
|
|
|
|
return ERROR_TARGET_TIMEOUT;
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
target->state = TARGET_HALTED;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* SVC, ARM state, IRQ and FIQ disabled */
|
2009-11-22 05:38:34 -06:00
|
|
|
uint32_t cpsr;
|
|
|
|
|
2012-01-19 04:06:37 -06:00
|
|
|
cpsr = buf_get_u32(arm->cpsr->value, 0, 32);
|
2009-11-22 05:38:34 -06:00
|
|
|
cpsr &= ~0xff;
|
|
|
|
cpsr |= 0xd3;
|
2012-01-19 04:06:37 -06:00
|
|
|
arm_set_cpsr(arm, cpsr);
|
2019-02-26 07:05:43 -06:00
|
|
|
arm->cpsr->dirty = true;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* start fetching from 0x0 */
|
2012-01-19 04:06:37 -06:00
|
|
|
buf_set_u32(arm->pc->value, 0, 32, 0x0);
|
2019-02-26 07:05:43 -06:00
|
|
|
arm->pc->dirty = true;
|
|
|
|
arm->pc->valid = true;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
arm920t_disable_mmu_caches(target, 1, 1, 1);
|
|
|
|
arm920t->armv4_5_mmu.mmu_enabled = 0;
|
|
|
|
arm920t->armv4_5_mmu.armv4_5_cache.d_u_cache_enabled = 0;
|
|
|
|
arm920t->armv4_5_mmu.armv4_5_cache.i_cache_enabled = 0;
|
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
return target_call_event_callbacks(target, TARGET_EVENT_HALTED);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2009-12-01 02:48:53 -06:00
|
|
|
/* FIXME remove forward decls */
|
|
|
|
static int arm920t_mrc(struct target *target, int cpnum,
|
|
|
|
uint32_t op1, uint32_t op2,
|
2021-04-27 08:58:26 -05:00
|
|
|
uint32_t crn, uint32_t crm,
|
2009-12-01 02:48:53 -06:00
|
|
|
uint32_t *value);
|
|
|
|
static int arm920t_mcr(struct target *target, int cpnum,
|
|
|
|
uint32_t op1, uint32_t op2,
|
2021-04-27 08:58:26 -05:00
|
|
|
uint32_t crn, uint32_t crm,
|
2009-12-01 02:48:53 -06:00
|
|
|
uint32_t value);
|
|
|
|
|
2010-04-10 04:01:06 -05:00
|
|
|
static int arm920t_init_arch_info(struct target *target,
|
2012-02-05 06:03:04 -06:00
|
|
|
struct arm920t_common *arm920t, struct jtag_tap *tap)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-11-17 03:09:50 -06:00
|
|
|
struct arm7_9_common *arm7_9 = &arm920t->arm7_9_common;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-01-19 04:06:37 -06:00
|
|
|
arm7_9->arm.mrc = arm920t_mrc;
|
|
|
|
arm7_9->arm.mcr = arm920t_mcr;
|
2009-12-01 02:48:53 -06:00
|
|
|
|
2009-11-17 03:09:50 -06:00
|
|
|
/* initialize arm7/arm9 specific info (including armv4_5) */
|
|
|
|
arm9tdmi_init_arch_info(target, arm7_9, tap);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
arm920t->common_magic = ARM920T_COMMON_MAGIC;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
arm7_9->post_debug_entry = arm920t_post_debug_entry;
|
|
|
|
arm7_9->pre_restore_context = arm920t_pre_restore_context;
|
2013-11-26 09:46:09 -06:00
|
|
|
arm7_9->write_memory = arm920t_write_memory;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
arm920t->armv4_5_mmu.armv4_5_cache.ctype = -1;
|
|
|
|
arm920t->armv4_5_mmu.get_ttb = arm920t_get_ttb;
|
|
|
|
arm920t->armv4_5_mmu.read_memory = arm7_9_read_memory;
|
|
|
|
arm920t->armv4_5_mmu.write_memory = arm7_9_write_memory;
|
|
|
|
arm920t->armv4_5_mmu.disable_mmu_caches = arm920t_disable_mmu_caches;
|
|
|
|
arm920t->armv4_5_mmu.enable_mmu_caches = arm920t_enable_mmu_caches;
|
|
|
|
arm920t->armv4_5_mmu.has_tiny_pages = 1;
|
|
|
|
arm920t->armv4_5_mmu.mmu_enabled = 0;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* disabling linefills leads to lockups, so keep them enabled for now
|
|
|
|
* this doesn't affect correctness, but might affect timing issues, if
|
|
|
|
* important data is evicted from the cache during the debug session
|
|
|
|
* */
|
|
|
|
arm920t->preserve_cache = 0;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* override hw single-step capability from ARM9TDMI */
|
|
|
|
arm7_9->has_single_step = 1;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-13 12:11:13 -06:00
|
|
|
static int arm920t_target_create(struct target *target, Jim_Interp *interp)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2010-02-20 22:47:38 -06:00
|
|
|
struct arm920t_common *arm920t;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
arm920t = calloc(1, sizeof(struct arm920t_common));
|
2009-11-05 22:35:37 -06:00
|
|
|
return arm920t_init_arch_info(target, arm920t, target->tap);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2020-05-25 04:28:22 -05:00
|
|
|
static void arm920t_deinit_target(struct target *target)
|
|
|
|
{
|
|
|
|
struct arm *arm = target_to_arm(target);
|
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
|
|
|
|
|
|
|
arm7_9_deinit(target);
|
|
|
|
arm_free_reg_cache(arm);
|
|
|
|
free(arm920t);
|
|
|
|
}
|
|
|
|
|
2009-11-10 01:56:52 -06:00
|
|
|
COMMAND_HANDLER(arm920t_handle_read_cache_command)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2008-10-14 06:06:30 -05:00
|
|
|
int retval = ERROR_OK;
|
2009-11-15 07:57:37 -06:00
|
|
|
struct target *target = get_current_target(CMD_CTX);
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-13 10:40:03 -06:00
|
|
|
struct arm7_9_common *arm7_9 = target_to_arm7_9(target);
|
2012-01-19 04:06:37 -06:00
|
|
|
struct arm *arm = &arm7_9->arm;
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15c15;
|
|
|
|
uint32_t cp15_ctrl, cp15_ctrl_saved;
|
|
|
|
uint32_t regs[16];
|
|
|
|
uint32_t *regs_p[16];
|
2021-04-27 08:58:26 -05:00
|
|
|
uint32_t c15_c_d_ind, c15_c_i_ind;
|
2008-02-25 11:48:04 -06:00
|
|
|
int i;
|
|
|
|
FILE *output;
|
2010-06-15 16:18:44 -05:00
|
|
|
int segment, index_t;
|
2009-11-22 12:21:48 -06:00
|
|
|
struct reg *r;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2019-03-31 21:32:12 -05:00
|
|
|
retval = arm920t_verify_pointer(CMD, arm920t);
|
2009-11-06 00:03:13 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
|
2009-11-15 06:57:12 -06:00
|
|
|
if (CMD_ARGC != 1)
|
2011-12-16 00:48:39 -06:00
|
|
|
return ERROR_COMMAND_SYNTAX_ERROR;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
output = fopen(CMD_ARGV[0], "w");
|
2021-07-03 11:51:20 -05:00
|
|
|
if (!output) {
|
2008-03-25 10:45:17 -05:00
|
|
|
LOG_DEBUG("error opening cache content file");
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
regs_p[i] = ®s[i];
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* disable MMU and Caches */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_CTRL, &cp15_ctrl);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
cp15_ctrl_saved = cp15_ctrl;
|
2010-02-20 22:47:38 -06:00
|
|
|
cp15_ctrl &= ~(ARMV4_5_MMU_ENABLED
|
2012-02-05 06:03:04 -06:00
|
|
|
| ARMV4_5_D_U_CACHE_ENABLED | ARMV4_5_I_CACHE_ENABLED);
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_CTRL, cp15_ctrl);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
/* read CP15 test state register */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_TESTSTATE, &cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
jtag_execute_queue();
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read DCache content */
|
|
|
|
fprintf(output, "DCache:\n");
|
2008-12-13 00:25:50 -06:00
|
|
|
|
|
|
|
/* go through segments 0 to nsets (8 on ARM920T, 4 on ARM922T) */
|
2010-02-20 22:47:38 -06:00
|
|
|
for (segment = 0;
|
2012-02-05 06:03:04 -06:00
|
|
|
segment < arm920t->armv4_5_mmu.armv4_5_cache.d_u_size.nsets;
|
|
|
|
segment++) {
|
2008-02-25 11:48:04 -06:00
|
|
|
fprintf(output, "\nsegment: %i\n----------", segment);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Ra: r0 = SBZ(31:8):segment(7:5):SBZ(4:0) */
|
|
|
|
regs[0] = 0x0 | (segment << 5);
|
|
|
|
arm9tdmi_write_core_regs(target, 0x1, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* D CAM Read, loads current victim into C15.C.D.Ind */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 2, 0, 15, 6, 2), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read current victim */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target,
|
2021-04-27 08:58:26 -05:00
|
|
|
CP15PHYS_DCACHE_IDX, &c15_c_d_ind);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (index_t = 0; index_t < 64; index_t++) {
|
2010-02-20 22:47:38 -06:00
|
|
|
/* Ra:
|
|
|
|
* r0 = index(31:26):SBZ(25:8):segment(7:5):SBZ(4:0)
|
|
|
|
*/
|
2010-06-15 16:18:44 -05:00
|
|
|
regs[0] = 0x0 | (segment << 5) | (index_t << 26);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x1, regs);
|
|
|
|
|
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write DCache victim */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 9, 1, 0), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read D RAM */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 2, 0, 15, 10, 2),
|
2010-02-20 22:47:38 -06:00
|
|
|
ARMV4_5_LDMIA(0, 0x1fe, 0, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read D CAM */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 2, 0, 15, 6, 2),
|
2010-02-20 22:47:38 -06:00
|
|
|
ARMV4_5_LDR(9, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* read D RAM and CAM content */
|
|
|
|
arm9tdmi_read_core_regs(target, 0x3fe, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* mask LFSR[6] */
|
|
|
|
regs[9] &= 0xfffffffe;
|
2010-02-20 22:47:38 -06:00
|
|
|
fprintf(output, "\nsegment: %i, index: %i, CAM: 0x%8.8"
|
|
|
|
PRIx32 ", content (%s):\n",
|
2010-06-15 16:18:44 -05:00
|
|
|
segment, index_t, regs[9],
|
2010-02-20 22:47:38 -06:00
|
|
|
(regs[9] & 0x10) ? "valid" : "invalid");
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (i = 1; i < 9; i++) {
|
|
|
|
fprintf(output, "%i: 0x%8.8" PRIx32 "\n",
|
|
|
|
i-1, regs[i]);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Ra: r0 = index(31:26):SBZ(25:8):segment(7:5):SBZ(4:0) */
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[0] = 0x0 | (segment << 5) | (c15_c_d_ind << 26);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x1, regs);
|
|
|
|
|
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write DCache victim */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 9, 1, 0), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/* read ICache content */
|
|
|
|
fprintf(output, "ICache:\n");
|
2008-12-13 00:25:50 -06:00
|
|
|
|
|
|
|
/* go through segments 0 to nsets (8 on ARM920T, 4 on ARM922T) */
|
2010-02-20 22:47:38 -06:00
|
|
|
for (segment = 0;
|
2012-02-05 06:03:04 -06:00
|
|
|
segment < arm920t->armv4_5_mmu.armv4_5_cache.d_u_size.nsets;
|
|
|
|
segment++) {
|
2008-02-25 11:48:04 -06:00
|
|
|
fprintf(output, "segment: %i\n----------", segment);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Ra: r0 = SBZ(31:8):segment(7:5):SBZ(4:0) */
|
|
|
|
regs[0] = 0x0 | (segment << 5);
|
|
|
|
arm9tdmi_write_core_regs(target, 0x1, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* I CAM Read, loads current victim into C15.C.I.Ind */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 2, 0, 15, 5, 2), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read current victim */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_ICACHE_IDX,
|
2021-04-27 08:58:26 -05:00
|
|
|
&c15_c_i_ind);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (index_t = 0; index_t < 64; index_t++) {
|
2010-02-20 22:47:38 -06:00
|
|
|
/* Ra:
|
|
|
|
* r0 = index(31:26):SBZ(25:8):segment(7:5):SBZ(4:0)
|
|
|
|
*/
|
2010-06-15 16:18:44 -05:00
|
|
|
regs[0] = 0x0 | (segment << 5) | (index_t << 26);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x1, regs);
|
|
|
|
|
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write ICache victim */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 9, 1, 1), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read I RAM */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 2, 0, 15, 9, 2),
|
2010-02-20 22:47:38 -06:00
|
|
|
ARMV4_5_LDMIA(0, 0x1fe, 0, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read I CAM */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 2, 0, 15, 5, 2),
|
2010-02-20 22:47:38 -06:00
|
|
|
ARMV4_5_LDR(9, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* read I RAM and CAM content */
|
|
|
|
arm9tdmi_read_core_regs(target, 0x3fe, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* mask LFSR[6] */
|
|
|
|
regs[9] &= 0xfffffffe;
|
2010-02-20 22:47:38 -06:00
|
|
|
fprintf(output, "\nsegment: %i, index: %i, "
|
|
|
|
"CAM: 0x%8.8" PRIx32 ", content (%s):\n",
|
2010-06-15 16:18:44 -05:00
|
|
|
segment, index_t, regs[9],
|
2010-02-20 22:47:38 -06:00
|
|
|
(regs[9] & 0x10) ? "valid" : "invalid");
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (i = 1; i < 9; i++) {
|
|
|
|
fprintf(output, "%i: 0x%8.8" PRIx32 "\n",
|
|
|
|
i-1, regs[i]);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Ra: r0 = index(31:26):SBZ(25:8):segment(7:5):SBZ(4:0) */
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[0] = 0x0 | (segment << 5) | (c15_c_d_ind << 26);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x1, regs);
|
|
|
|
|
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write ICache victim */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 9, 1, 1), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* restore CP15 MMU and Cache settings */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_CTRL, cp15_ctrl_saved);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD, "cache content successfully output to %s",
|
2012-02-05 06:03:04 -06:00
|
|
|
CMD_ARGV[0]);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fclose(output);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (!is_arm_mode(arm->core_mode)) {
|
2010-08-11 03:58:49 -05:00
|
|
|
LOG_ERROR("not a valid arm core mode - communication failure?");
|
2008-05-19 14:02:36 -05:00
|
|
|
return ERROR_FAIL;
|
2010-08-11 03:58:49 -05:00
|
|
|
}
|
2008-05-19 14:02:36 -05:00
|
|
|
|
2009-11-22 12:21:48 -06:00
|
|
|
/* force writeback of the valid data */
|
2012-01-19 04:06:37 -06:00
|
|
|
r = arm->core_cache->reg_list;
|
2009-11-22 12:21:48 -06:00
|
|
|
r[0].dirty = r[0].valid;
|
|
|
|
r[1].dirty = r[1].valid;
|
|
|
|
r[2].dirty = r[2].valid;
|
|
|
|
r[3].dirty = r[3].valid;
|
|
|
|
r[4].dirty = r[4].valid;
|
|
|
|
r[5].dirty = r[5].valid;
|
|
|
|
r[6].dirty = r[6].valid;
|
|
|
|
r[7].dirty = r[7].valid;
|
|
|
|
|
2012-01-19 04:06:37 -06:00
|
|
|
r = arm_reg_current(arm, 8);
|
2009-11-22 12:21:48 -06:00
|
|
|
r->dirty = r->valid;
|
|
|
|
|
2012-01-19 04:06:37 -06:00
|
|
|
r = arm_reg_current(arm, 9);
|
2009-11-22 12:21:48 -06:00
|
|
|
r->dirty = r->valid;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-10 01:56:52 -06:00
|
|
|
COMMAND_HANDLER(arm920t_handle_read_mmu_command)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2008-10-14 06:06:30 -05:00
|
|
|
int retval = ERROR_OK;
|
2009-11-15 07:57:37 -06:00
|
|
|
struct target *target = get_current_target(CMD_CTX);
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2009-11-13 10:40:03 -06:00
|
|
|
struct arm7_9_common *arm7_9 = target_to_arm7_9(target);
|
2012-01-19 04:06:37 -06:00
|
|
|
struct arm *arm = &arm7_9->arm;
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t cp15c15;
|
|
|
|
uint32_t cp15_ctrl, cp15_ctrl_saved;
|
|
|
|
uint32_t regs[16];
|
|
|
|
uint32_t *regs_p[16];
|
2008-02-25 11:48:04 -06:00
|
|
|
int i;
|
|
|
|
FILE *output;
|
2021-04-27 08:58:26 -05:00
|
|
|
uint32_t d_lockdown, i_lockdown;
|
2009-11-13 10:40:12 -06:00
|
|
|
struct arm920t_tlb_entry d_tlb[64], i_tlb[64];
|
2008-02-25 11:48:04 -06:00
|
|
|
int victim;
|
2009-11-22 12:21:48 -06:00
|
|
|
struct reg *r;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2019-03-31 21:32:12 -05:00
|
|
|
retval = arm920t_verify_pointer(CMD, arm920t);
|
2009-11-06 00:03:13 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
|
|
|
|
2009-11-15 06:57:12 -06:00
|
|
|
if (CMD_ARGC != 1)
|
2011-12-16 00:48:39 -06:00
|
|
|
return ERROR_COMMAND_SYNTAX_ERROR;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
output = fopen(CMD_ARGV[0], "w");
|
2021-07-03 11:51:20 -05:00
|
|
|
if (!output) {
|
2008-03-25 10:45:17 -05:00
|
|
|
LOG_DEBUG("error opening mmu content file");
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
regs_p[i] = ®s[i];
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* disable MMU and Caches */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_CTRL, &cp15_ctrl);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
cp15_ctrl_saved = cp15_ctrl;
|
2010-02-20 22:47:38 -06:00
|
|
|
cp15_ctrl &= ~(ARMV4_5_MMU_ENABLED
|
|
|
|
| ARMV4_5_D_U_CACHE_ENABLED | ARMV4_5_I_CACHE_ENABLED);
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_CTRL, cp15_ctrl);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
/* read CP15 test state register */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_read_cp15_physical(target, CP15PHYS_TESTSTATE, &cp15c15);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
/* prepare reading D TLB content
|
2008-02-25 11:48:04 -06:00
|
|
|
* */
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read D TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MRC(15, 0, 0, 10, 0, 0), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read D TLB lockdown stored to r1 */
|
|
|
|
arm9tdmi_read_core_regs(target, 0x2, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2021-04-27 08:58:26 -05:00
|
|
|
d_lockdown = regs[1];
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (victim = 0; victim < 64; victim += 8) {
|
2008-12-13 00:25:50 -06:00
|
|
|
/* new lockdown value: base[31:26]:victim[25:20]:SBZ[19:1]:p[0]
|
2010-02-20 22:47:38 -06:00
|
|
|
* base remains unchanged, victim goes through entries 0 to 63
|
|
|
|
*/
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[1] = (d_lockdown & 0xfc000000) | (victim << 20);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write D TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 10, 0, 0),
|
2010-02-20 22:47:38 -06:00
|
|
|
ARMV4_5_STR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read D TLB CAM */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 4, 0, 15, 6, 4),
|
2010-02-20 22:47:38 -06:00
|
|
|
ARMV4_5_LDMIA(0, 0x3fc, 0, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read D TLB CAM content stored to r2-r9 */
|
|
|
|
arm9tdmi_read_core_regs(target, 0x3fc, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
for (i = 0; i < 8; i++)
|
2008-12-13 00:25:50 -06:00
|
|
|
d_tlb[victim + i].cam = regs[i + 2];
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (victim = 0; victim < 64; victim++) {
|
2008-12-13 00:25:50 -06:00
|
|
|
/* new lockdown value: base[31:26]:victim[25:20]:SBZ[19:1]:p[0]
|
2010-02-20 22:47:38 -06:00
|
|
|
* base remains unchanged, victim goes through entries 0 to 63
|
|
|
|
*/
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[1] = (d_lockdown & 0xfc000000) | (victim << 20);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write D TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 10, 0, 0), ARMV4_5_STR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read D TLB RAM1 */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 4, 0, 15, 10, 4), ARMV4_5_LDR(2, 0));
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* Read D TLB RAM2 */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 4, 0, 15, 2, 5), ARMV4_5_LDR(3, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read D TLB RAM content stored to r2 and r3 */
|
|
|
|
arm9tdmi_read_core_regs(target, 0xc, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
d_tlb[victim].ram1 = regs[2];
|
|
|
|
d_tlb[victim].ram2 = regs[3];
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* restore D TLB lockdown */
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[1] = d_lockdown;
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write D TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 10, 0, 0), ARMV4_5_STR(1, 0));
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
/* prepare reading I TLB content
|
2008-02-25 11:48:04 -06:00
|
|
|
* */
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read I TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MRC(15, 0, 0, 10, 0, 1), ARMV4_5_LDR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read I TLB lockdown stored to r1 */
|
|
|
|
arm9tdmi_read_core_regs(target, 0x2, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2021-04-27 08:58:26 -05:00
|
|
|
i_lockdown = regs[1];
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (victim = 0; victim < 64; victim += 8) {
|
2008-12-13 00:25:50 -06:00
|
|
|
/* new lockdown value: base[31:26]:victim[25:20]:SBZ[19:1]:p[0]
|
2010-02-20 22:47:38 -06:00
|
|
|
* base remains unchanged, victim goes through entries 0 to 63
|
|
|
|
*/
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[1] = (i_lockdown & 0xfc000000) | (victim << 20);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write I TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 10, 0, 1),
|
|
|
|
ARMV4_5_STR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read I TLB CAM */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 4, 0, 15, 5, 4),
|
|
|
|
ARMV4_5_LDMIA(0, 0x3fc, 0, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read I TLB CAM content stored to r2-r9 */
|
|
|
|
arm9tdmi_read_core_regs(target, 0x3fc, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
for (i = 0; i < 8; i++)
|
2008-12-13 00:25:50 -06:00
|
|
|
i_tlb[i + victim].cam = regs[i + 2];
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
for (victim = 0; victim < 64; victim++) {
|
2008-12-13 00:25:50 -06:00
|
|
|
/* new lockdown value: base[31:26]:victim[25:20]:SBZ[19:1]:p[0]
|
2010-02-20 22:47:38 -06:00
|
|
|
* base remains unchanged, victim goes through entries 0 to 63
|
|
|
|
*/
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[1] = (d_lockdown & 0xfc000000) | (victim << 20);
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* set interpret mode */
|
|
|
|
cp15c15 |= 0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write I TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 10, 0, 1), ARMV4_5_STR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Read I TLB RAM1 */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 4, 0, 15, 9, 4), ARMV4_5_LDR(2, 0));
|
2008-02-25 11:48:04 -06:00
|
|
|
|
|
|
|
/* Read I TLB RAM2 */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 4, 0, 15, 1, 5), ARMV4_5_LDR(3, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* clear interpret mode */
|
|
|
|
cp15c15 &= ~0x1;
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
CP15PHYS_TESTSTATE, cp15c15);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* read I TLB RAM content stored to r2 and r3 */
|
|
|
|
arm9tdmi_read_core_regs(target, 0xc, regs_p);
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
i_tlb[victim].ram1 = regs[2];
|
|
|
|
i_tlb[victim].ram2 = regs[3];
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* restore I TLB lockdown */
|
2021-04-27 08:58:26 -05:00
|
|
|
regs[1] = i_lockdown;
|
2008-02-25 11:48:04 -06:00
|
|
|
arm9tdmi_write_core_regs(target, 0x2, regs);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* Write I TLB lockdown */
|
2010-02-20 22:47:38 -06:00
|
|
|
arm920t_execute_cp15(target,
|
2012-02-05 06:03:04 -06:00
|
|
|
ARMV4_5_MCR(15, 0, 0, 10, 0, 1), ARMV4_5_STR(1, 0));
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
/* restore CP15 MMU and Cache settings */
|
2010-02-16 20:50:16 -06:00
|
|
|
arm920t_write_cp15_physical(target, CP15PHYS_CTRL, cp15_ctrl_saved);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2008-12-13 00:25:50 -06:00
|
|
|
/* output data to file */
|
2008-02-25 11:48:04 -06:00
|
|
|
fprintf(output, "D TLB content:\n");
|
2012-02-05 06:03:04 -06:00
|
|
|
for (i = 0; i < 64; i++) {
|
2010-02-20 22:47:38 -06:00
|
|
|
fprintf(output, "%i: 0x%8.8" PRIx32 " 0x%8.8" PRIx32
|
|
|
|
" 0x%8.8" PRIx32 " %s\n",
|
|
|
|
i, d_tlb[i].cam, d_tlb[i].ram1, d_tlb[i].ram2,
|
|
|
|
(d_tlb[i].cam & 0x20) ? "(valid)" : "(invalid)");
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(output, "\n\nI TLB content:\n");
|
2012-02-05 06:03:04 -06:00
|
|
|
for (i = 0; i < 64; i++) {
|
2010-02-20 22:47:38 -06:00
|
|
|
fprintf(output, "%i: 0x%8.8" PRIx32 " 0x%8.8" PRIx32
|
|
|
|
" 0x%8.8" PRIx32 " %s\n",
|
|
|
|
i, i_tlb[i].cam, i_tlb[i].ram1, i_tlb[i].ram2,
|
|
|
|
(i_tlb[i].cam & 0x20) ? "(valid)" : "(invalid)");
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2008-12-13 00:25:50 -06:00
|
|
|
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD, "mmu content successfully output to %s",
|
2012-02-05 06:03:04 -06:00
|
|
|
CMD_ARGV[0]);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
fclose(output);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (!is_arm_mode(arm->core_mode)) {
|
2010-08-11 03:58:49 -05:00
|
|
|
LOG_ERROR("not a valid arm core mode - communication failure?");
|
2008-05-19 14:02:36 -05:00
|
|
|
return ERROR_FAIL;
|
2010-08-11 03:58:49 -05:00
|
|
|
}
|
2008-05-19 14:02:36 -05:00
|
|
|
|
2009-11-22 12:21:48 -06:00
|
|
|
/* force writeback of the valid data */
|
2012-01-19 04:06:37 -06:00
|
|
|
r = arm->core_cache->reg_list;
|
2009-11-22 12:21:48 -06:00
|
|
|
r[0].dirty = r[0].valid;
|
|
|
|
r[1].dirty = r[1].valid;
|
|
|
|
r[2].dirty = r[2].valid;
|
|
|
|
r[3].dirty = r[3].valid;
|
|
|
|
r[4].dirty = r[4].valid;
|
|
|
|
r[5].dirty = r[5].valid;
|
|
|
|
r[6].dirty = r[6].valid;
|
|
|
|
r[7].dirty = r[7].valid;
|
|
|
|
|
2012-01-19 04:06:37 -06:00
|
|
|
r = arm_reg_current(arm, 8);
|
2009-11-22 12:21:48 -06:00
|
|
|
r->dirty = r->valid;
|
|
|
|
|
2012-01-19 04:06:37 -06:00
|
|
|
r = arm_reg_current(arm, 9);
|
2009-11-22 12:21:48 -06:00
|
|
|
r->dirty = r->valid;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
2009-11-05 22:35:37 -06:00
|
|
|
|
2009-11-10 01:56:52 -06:00
|
|
|
COMMAND_HANDLER(arm920t_handle_cp15_command)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
|
|
|
int retval;
|
2009-11-15 07:57:37 -06:00
|
|
|
struct target *target = get_current_target(CMD_CTX);
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2019-03-31 21:32:12 -05:00
|
|
|
retval = arm920t_verify_pointer(CMD, arm920t);
|
2009-11-06 00:03:13 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (target->state != TARGET_HALTED) {
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD, "target must be stopped for "
|
2010-02-20 22:47:38 -06:00
|
|
|
"\"%s\" command", CMD_NAME);
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2010-02-20 22:47:38 -06:00
|
|
|
/* one argument, read a register.
|
|
|
|
* two arguments, write it.
|
|
|
|
*/
|
2012-02-05 06:03:04 -06:00
|
|
|
if (CMD_ARGC >= 1) {
|
2009-10-24 08:36:05 -05:00
|
|
|
int address;
|
2009-11-15 10:15:59 -06:00
|
|
|
COMMAND_PARSE_NUMBER(int, CMD_ARGV[0], address);
|
2008-02-25 11:48:04 -06:00
|
|
|
|
2012-02-05 06:03:04 -06:00
|
|
|
if (CMD_ARGC == 1) {
|
2009-06-18 02:08:52 -05:00
|
|
|
uint32_t value;
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = arm920t_read_cp15_physical(target, address, &value);
|
|
|
|
if (retval != ERROR_OK) {
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD,
|
2010-02-20 22:47:38 -06:00
|
|
|
"couldn't access reg %i", address);
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
2012-02-05 06:03:04 -06:00
|
|
|
retval = jtag_execute_queue();
|
|
|
|
if (retval != ERROR_OK)
|
2008-10-14 06:06:30 -05:00
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD, "%i: %8.8" PRIx32,
|
2012-02-05 06:03:04 -06:00
|
|
|
address, value);
|
|
|
|
} else if (CMD_ARGC == 2) {
|
2009-10-24 08:36:05 -05:00
|
|
|
uint32_t value;
|
2009-11-15 10:15:59 -06:00
|
|
|
COMMAND_PARSE_NUMBER(u32, CMD_ARGV[1], value);
|
2010-02-20 22:47:38 -06:00
|
|
|
retval = arm920t_write_cp15_physical(target,
|
|
|
|
address, value);
|
2012-02-05 06:03:04 -06:00
|
|
|
if (retval != ERROR_OK) {
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD,
|
2010-02-20 22:47:38 -06:00
|
|
|
"couldn't access reg %i", address);
|
|
|
|
/* REVISIT why lie? "return retval"? */
|
2008-02-25 11:48:04 -06:00
|
|
|
return ERROR_OK;
|
|
|
|
}
|
helper/command: change prototype of command_print/command_print_sameline
To prepare for handling TCL return values consistently, all calls
to command_print/command_print_sameline should switch to CMD as
first parameter.
Change prototype of command_print() and command_print_sameline()
to pass CMD instead of CMD_CTX.
Since the first parameter is currently not used, the change can be
done though scripts without manual coding.
This patch is created using the command:
sed -i PATTERN $(find src/ doc/ -type f)
with all the following patters:
's/\(command_print(cmd\)->ctx,/\1,/'
's/\(command_print(CMD\)_CTX,/\1,/'
's/\(command_print(struct command_\)context \*context,/\1invocation *cmd,/'
's/\(command_print_sameline(cmd\)->ctx,/\1,/'
's/\(command_print_sameline(CMD\)_CTX,/\1,/'
's/\(command_print_sameline(struct command_\)context \*context,/\1invocation *cmd,/'
This change is inspired by http://openocd.zylin.com/1815 from Paul
Fertser but is now done through scripting.
Change-Id: I3386d8f96cdc477e7a2308dd18269de3bed04385
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Signed-off-by: Paul Fertser <fercerpav@gmail.com>
Reviewed-on: http://openocd.zylin.com/5081
Tested-by: jenkins
Reviewed-by: Tomas Vanek <vanekt@fbl.cz>
2019-04-03 03:37:24 -05:00
|
|
|
command_print(CMD, "%i: %8.8" PRIx32,
|
2012-02-05 06:03:04 -06:00
|
|
|
address, value);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2009-11-10 01:56:52 -06:00
|
|
|
COMMAND_HANDLER(arm920t_handle_cache_info_command)
|
2008-02-25 11:48:04 -06:00
|
|
|
{
|
2009-11-06 00:03:13 -06:00
|
|
|
int retval;
|
2009-11-15 07:57:37 -06:00
|
|
|
struct target *target = get_current_target(CMD_CTX);
|
2009-11-13 10:36:02 -06:00
|
|
|
struct arm920t_common *arm920t = target_to_arm920(target);
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2019-03-31 21:32:12 -05:00
|
|
|
retval = arm920t_verify_pointer(CMD, arm920t);
|
2009-11-06 00:03:13 -06:00
|
|
|
if (retval != ERROR_OK)
|
|
|
|
return retval;
|
2008-12-13 00:25:50 -06:00
|
|
|
|
2019-03-31 21:42:23 -05:00
|
|
|
return armv4_5_handle_cache_info_command(CMD,
|
2012-02-05 06:03:04 -06:00
|
|
|
&arm920t->armv4_5_mmu.armv4_5_cache);
|
2008-02-25 11:48:04 -06:00
|
|
|
}
|
2009-11-03 05:54:26 -06:00
|
|
|
|
|
|
|
|
2009-12-07 16:54:12 -06:00
|
|
|
static int arm920t_mrc(struct target *target, int cpnum,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t op1, uint32_t op2,
|
2021-04-27 08:58:26 -05:00
|
|
|
uint32_t crn, uint32_t crm,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t *value)
|
2009-11-03 05:54:26 -06:00
|
|
|
{
|
2012-02-05 06:03:04 -06:00
|
|
|
if (cpnum != 15) {
|
2009-11-03 05:54:26 -06:00
|
|
|
LOG_ERROR("Only cp15 is supported");
|
|
|
|
return ERROR_FAIL;
|
|
|
|
}
|
|
|
|
|
2009-12-07 16:54:12 -06:00
|
|
|
/* read "to" r0 */
|
|
|
|
return arm920t_read_cp15_interpreted(target,
|
2021-04-27 08:58:26 -05:00
|
|
|
ARMV4_5_MRC(cpnum, op1, 0, crn, crm, op2),
|
2012-02-05 06:03:04 -06:00
|
|
|
0, value);
|
2009-11-03 05:54:26 -06:00
|
|
|
}
|
|
|
|
|
2009-12-07 16:54:12 -06:00
|
|
|
static int arm920t_mcr(struct target *target, int cpnum,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t op1, uint32_t op2,
|
2021-04-27 08:58:26 -05:00
|
|
|
uint32_t crn, uint32_t crm,
|
2012-02-05 06:03:04 -06:00
|
|
|
uint32_t value)
|
2009-11-03 05:54:26 -06:00
|
|
|
{
|
2012-02-05 06:03:04 -06:00
|
|
|
if (cpnum != 15) {
|
2009-11-03 05:54:26 -06:00
|
|
|
LOG_ERROR("Only cp15 is supported");
|
|
|
|
return ERROR_FAIL;
|
|
|
|
}
|
|
|
|
|
2009-12-07 16:54:12 -06:00
|
|
|
/* write "from" r0 */
|
|
|
|
return arm920t_write_cp15_interpreted(target,
|
2021-04-27 08:58:26 -05:00
|
|
|
ARMV4_5_MCR(cpnum, op1, 0, crn, crm, op2),
|
2012-02-05 06:03:04 -06:00
|
|
|
0, value);
|
2009-11-03 05:54:26 -06:00
|
|
|
}
|
2009-11-05 22:35:37 -06:00
|
|
|
|
2009-11-23 09:43:05 -06:00
|
|
|
static const struct command_registration arm920t_exec_command_handlers[] = {
|
|
|
|
{
|
|
|
|
.name = "cp15",
|
2010-01-07 18:30:09 -06:00
|
|
|
.handler = arm920t_handle_cp15_command,
|
2009-11-23 09:43:05 -06:00
|
|
|
.mode = COMMAND_EXEC,
|
|
|
|
.help = "display/modify cp15 register",
|
2010-01-07 18:30:09 -06:00
|
|
|
.usage = "regnum [value]",
|
2009-11-23 09:43:05 -06:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "cache_info",
|
2010-01-07 18:30:09 -06:00
|
|
|
.handler = arm920t_handle_cache_info_command,
|
2009-11-23 09:43:05 -06:00
|
|
|
.mode = COMMAND_EXEC,
|
2012-01-16 07:35:23 -06:00
|
|
|
.usage = "",
|
2009-11-23 09:43:05 -06:00
|
|
|
.help = "display information about target caches",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "read_cache",
|
2010-01-07 18:30:09 -06:00
|
|
|
.handler = arm920t_handle_read_cache_command,
|
2009-11-23 09:43:05 -06:00
|
|
|
.mode = COMMAND_EXEC,
|
2010-01-07 18:30:09 -06:00
|
|
|
.help = "dump I/D cache content to file",
|
|
|
|
.usage = "filename",
|
2009-11-23 09:43:05 -06:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "read_mmu",
|
2010-01-07 18:30:09 -06:00
|
|
|
.handler = arm920t_handle_read_mmu_command,
|
2009-11-23 09:43:05 -06:00
|
|
|
.mode = COMMAND_EXEC,
|
2010-01-07 18:30:09 -06:00
|
|
|
.help = "dump I/D mmu content to file",
|
|
|
|
.usage = "filename",
|
2009-11-23 09:43:05 -06:00
|
|
|
},
|
|
|
|
COMMAND_REGISTRATION_DONE
|
|
|
|
};
|
2009-11-23 10:17:01 -06:00
|
|
|
const struct command_registration arm920t_command_handlers[] = {
|
|
|
|
{
|
|
|
|
.chain = arm9tdmi_command_handlers,
|
|
|
|
},
|
2009-11-23 09:43:05 -06:00
|
|
|
{
|
|
|
|
.name = "arm920t",
|
|
|
|
.mode = COMMAND_ANY,
|
|
|
|
.help = "arm920t command group",
|
2012-01-16 07:35:23 -06:00
|
|
|
.usage = "",
|
2009-11-23 09:43:05 -06:00
|
|
|
.chain = arm920t_exec_command_handlers,
|
|
|
|
},
|
|
|
|
COMMAND_REGISTRATION_DONE
|
|
|
|
};
|
|
|
|
|
2009-11-05 22:35:37 -06:00
|
|
|
/** Holds methods for ARM920 targets. */
|
2012-02-05 06:03:04 -06:00
|
|
|
struct target_type arm920t_target = {
|
2009-11-05 22:35:37 -06:00
|
|
|
.name = "arm920t",
|
|
|
|
|
|
|
|
.poll = arm7_9_poll,
|
|
|
|
.arch_state = arm920t_arch_state,
|
|
|
|
|
|
|
|
.target_request_data = arm7_9_target_request_data,
|
|
|
|
|
|
|
|
.halt = arm7_9_halt,
|
|
|
|
.resume = arm7_9_resume,
|
|
|
|
.step = arm7_9_step,
|
|
|
|
|
|
|
|
.assert_reset = arm7_9_assert_reset,
|
|
|
|
.deassert_reset = arm7_9_deassert_reset,
|
|
|
|
.soft_reset_halt = arm920t_soft_reset_halt,
|
|
|
|
|
target/arm: add support for multi-architecture gdb
GDB can be built for multi-architecture through the command
./configure --enable-targets=all && make
Such multi-architecture GDB requires the target's architecture to
be selected either manually by the user through the GDB command
"set architecture" or automatically by the target description sent
by the remote target (i.e. OpenOCD).
Commit e65acd889c61a424c7bd72fdee5d6a3aee1d8504 ("gdb_server: add
support for architecture element") already provides the required
infrastructure to support multi-architecture gdb.
arm-none-eabi-gdb 8.2 uses "arm" as default architecture, but also
supports the following values: "arm_any", "armv2", "armv2a",
"armv3", "armv3m", "armv4", "armv4t", "armv5", "armv5t", "armv5te",
"armv5tej", "armv6", "armv6k", "armv6kz", "armv6-m", "armv6s-m",
"armv6t2", "armv7", "armv7e-m", "armv8-a", "armv8-m.base",
"armv8-m.main", "armv8-r", "ep9312", "iwmmxt", "iwmmxt2", "xscale".
These values can be displayed on arm gdb prompt by typing
"set architecture " followed by a TAB for autocompletion.
Set the gdb architecture value for all arm targets to "arm".
Change-Id: I176cb89878606e1febd546ce26543b3e7849500a
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Reviewed-on: http://openocd.zylin.com/4754
Tested-by: jenkins
Reviewed-by: Spencer Oliver <spen@spen-soft.co.uk>
2018-11-01 08:50:27 -05:00
|
|
|
.get_gdb_arch = arm_get_gdb_arch,
|
2009-12-07 16:54:13 -06:00
|
|
|
.get_gdb_reg_list = arm_get_gdb_reg_list,
|
2009-11-05 22:35:37 -06:00
|
|
|
|
|
|
|
.read_memory = arm920t_read_memory,
|
2013-10-04 17:19:08 -05:00
|
|
|
.write_memory = arm7_9_write_memory_opt,
|
2009-11-05 22:35:37 -06:00
|
|
|
.read_phys_memory = arm920t_read_phys_memory,
|
|
|
|
.write_phys_memory = arm920t_write_phys_memory,
|
2009-11-10 03:36:59 -06:00
|
|
|
.mmu = arm920_mmu,
|
|
|
|
.virt2phys = arm920_virt2phys,
|
|
|
|
|
2009-11-15 12:35:34 -06:00
|
|
|
.checksum_memory = arm_checksum_memory,
|
|
|
|
.blank_check_memory = arm_blank_check_memory,
|
2009-11-05 22:35:37 -06:00
|
|
|
|
|
|
|
.run_algorithm = armv4_5_run_algorithm,
|
|
|
|
|
|
|
|
.add_breakpoint = arm7_9_add_breakpoint,
|
|
|
|
.remove_breakpoint = arm7_9_remove_breakpoint,
|
|
|
|
.add_watchpoint = arm7_9_add_watchpoint,
|
|
|
|
.remove_watchpoint = arm7_9_remove_watchpoint,
|
|
|
|
|
2009-11-23 10:17:01 -06:00
|
|
|
.commands = arm920t_command_handlers,
|
2009-11-05 22:35:37 -06:00
|
|
|
.target_create = arm920t_target_create,
|
|
|
|
.init_target = arm9tdmi_init_target,
|
2020-05-25 04:28:22 -05:00
|
|
|
.deinit_target = arm920t_deinit_target,
|
2009-11-13 18:26:39 -06:00
|
|
|
.examine = arm7_9_examine,
|
2010-01-11 08:30:22 -06:00
|
|
|
.check_reset = arm7_9_check_reset,
|
2009-11-05 22:35:37 -06:00
|
|
|
};
|