This commit is contained in:
Evgeniy Naydanov 2025-03-07 12:23:40 +03:00 committed by GitHub
commit 45260bb18d
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
30 changed files with 799 additions and 348 deletions

View File

@ -9271,7 +9271,7 @@ various operations. The current target may be changed
by using @command{targets} command with the name of the by using @command{targets} command with the name of the
target which should become current. target which should become current.
@deffn {Command} {reg} [(number|name) [(value|'force')]] @deffn {Command} {reg} [(number|name)] [value] ['force']
Access a single register by @var{number} or by its @var{name}. Access a single register by @var{number} or by its @var{name}.
The target must generally be halted before access to CPU core The target must generally be halted before access to CPU core
registers is allowed. Depending on the hardware, some other registers is allowed. Depending on the hardware, some other
@ -9285,15 +9285,17 @@ which are also dirty (and will be written back later)
are flagged as such. are flagged as such.
@emph{With number/name}: display that register's value. @emph{With number/name}: display that register's value.
Use @var{force} argument to read directly from the target, Use @var{force} argument to read directly from the target
bypassing any internal cache. (as opposed to OpenOCD's internal register cache).
@emph{With both number/name and value}: set register's value. @emph{With both number/name and value}: set register's value.
Writes may be held in a writeback cache internal to OpenOCD, Writes may be held in a writeback cache internal to OpenOCD,
so that setting the value marks the register as dirty instead so that setting the value marks the register as dirty instead
of immediately flushing that value. Resuming CPU execution of immediately flushing that value. Resuming CPU execution
(including by single stepping) or otherwise activating the (including by single stepping) or otherwise activating the
relevant module will flush such values. relevant module will flush such values. The use of @var{force}
causes the register value to be written to the target
immediately.
Cores may have surprisingly many registers in their Cores may have surprisingly many registers in their
Debug and trace infrastructure: Debug and trace infrastructure:
@ -9310,10 +9312,13 @@ Debug and trace infrastructure:
@end example @end example
@end deffn @end deffn
@deffn {Command} {set_reg} dict @deffn {Command} {set_reg} ['-force'] dict
Set register values of the target. Set register values of the target. The new register values may be
kept in OpenOCD's cache and not written to the target immediately
(unless @var{-force} is used).
@itemize @itemize
@item @option{-force} ... Force immediate write of the new register values.
@item @var{dict} ... Tcl dictionary with pairs of register names and values. @item @var{dict} ... Tcl dictionary with pairs of register names and values.
@end itemize @end itemize
@ -9329,7 +9334,7 @@ set_reg @{pc 0 sp 0x1000@}
Get register values from the target and return them as Tcl dictionary with pairs Get register values from the target and return them as Tcl dictionary with pairs
of register names and values. of register names and values.
If option "-force" is set, the register values are read directly from the If option "-force" is set, the register values are read directly from the
target, bypassing any caching. target, not from OpenOCD's internal cache.
@itemize @itemize
@item @var{list} ... List of register names @item @var{list} ... List of register names
@ -9343,6 +9348,13 @@ get_reg @{pc sp@}
@end example @end example
@end deffn @end deffn
@deffn {Command} {flush_reg_cache} [-invalidate]
Flush the internal OpenOCD's register cache - write back the dirty register values to the target.
If @option{-invalidate} is set, also invalidate (forget) the OpenOCD's cached register values;
therefore the next call to get_reg is guaranteed to read the fresh register value directly
from the target.
@end deffn
@deffn {Command} {write_memory} address width data ['phys'] @deffn {Command} {write_memory} address width data ['phys']
This function provides an efficient way to write to the target memory from a Tcl This function provides an efficient way to write to the target memory from a Tcl
script. script.

View File

@ -293,6 +293,7 @@ static int arc_set_register(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type arc_reg_type = { static const struct reg_arch_type arc_reg_type = {
.get = arc_get_register, .get = arc_get_register,
.set = arc_set_register, .set = arc_set_register,
.flush = NULL,
}; };
/* GDB register groups. For now we support only general and "empty" */ /* GDB register groups. For now we support only general and "empty" */

View File

@ -655,6 +655,7 @@ static int armv4_5_set_core_reg(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type arm_reg_type = { static const struct reg_arch_type arm_reg_type = {
.get = armv4_5_get_core_reg, .get = armv4_5_get_core_reg,
.set = armv4_5_set_core_reg, .set = armv4_5_set_core_reg,
.flush = NULL,
}; };
struct reg_cache *arm_build_reg_cache(struct target *target, struct arm *arm) struct reg_cache *arm_build_reg_cache(struct target *target, struct arm *arm)

View File

@ -786,6 +786,7 @@ int armv7m_arch_state(struct target *target)
static const struct reg_arch_type armv7m_reg_type = { static const struct reg_arch_type armv7m_reg_type = {
.get = armv7m_get_core_reg, .get = armv7m_get_core_reg,
.set = armv7m_set_core_reg, .set = armv7m_set_core_reg,
.flush = NULL,
}; };
/** Builds cache of architecturally defined registers. */ /** Builds cache of architecturally defined registers. */

View File

@ -1714,6 +1714,7 @@ static int armv8_set_core_reg(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type armv8_reg_type = { static const struct reg_arch_type armv8_reg_type = {
.get = armv8_get_core_reg, .get = armv8_get_core_reg,
.set = armv8_set_core_reg, .set = armv8_set_core_reg,
.flush = NULL,
}; };
static int armv8_get_core_reg32(struct reg *reg) static int armv8_get_core_reg32(struct reg *reg)
@ -1775,6 +1776,7 @@ static int armv8_set_core_reg32(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type armv8_reg32_type = { static const struct reg_arch_type armv8_reg32_type = {
.get = armv8_get_core_reg32, .get = armv8_get_core_reg32,
.set = armv8_set_core_reg32, .set = armv8_set_core_reg32,
.flush = NULL,
}; };
/** Builds cache of architecturally defined registers. */ /** Builds cache of architecturally defined registers. */

View File

@ -157,6 +157,7 @@ static int avr32_set_core_reg(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type avr32_reg_type = { static const struct reg_arch_type avr32_reg_type = {
.get = avr32_get_core_reg, .get = avr32_get_core_reg,
.set = avr32_set_core_reg, .set = avr32_set_core_reg,
.flush = NULL,
}; };
static struct reg_cache *avr32_build_reg_cache(struct target *target) static struct reg_cache *avr32_build_reg_cache(struct target *target)

View File

@ -2431,6 +2431,7 @@ static const struct dwt_reg dwt_comp[] = {
static const struct reg_arch_type dwt_reg_type = { static const struct reg_arch_type dwt_reg_type = {
.get = cortex_m_dwt_get_reg, .get = cortex_m_dwt_get_reg,
.set = cortex_m_dwt_set_reg, .set = cortex_m_dwt_set_reg,
.flush = NULL,
}; };
static void cortex_m_dwt_addreg(struct target *t, struct reg *r, const struct dwt_reg *d) static void cortex_m_dwt_addreg(struct target *t, struct reg *r, const struct dwt_reg *d)

View File

@ -430,6 +430,7 @@ static int dsp563xx_set_core_reg(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type dsp563xx_reg_type = { static const struct reg_arch_type dsp563xx_reg_type = {
.get = dsp563xx_get_core_reg, .get = dsp563xx_get_core_reg,
.set = dsp563xx_set_core_reg, .set = dsp563xx_set_core_reg,
.flush = NULL,
}; };
static void dsp563xx_build_reg_cache(struct target *target) static void dsp563xx_build_reg_cache(struct target *target)

View File

@ -152,6 +152,7 @@ static int embeddedice_get_reg(struct reg *reg)
static const struct reg_arch_type eice_reg_type = { static const struct reg_arch_type eice_reg_type = {
.get = embeddedice_get_reg, .get = embeddedice_get_reg,
.set = embeddedice_set_reg_w_exec, .set = embeddedice_set_reg_w_exec,
.flush = NULL,
}; };
/** /**

View File

@ -1419,6 +1419,7 @@ static int esirisc_set_reg(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type esirisc_reg_type = { static const struct reg_arch_type esirisc_reg_type = {
.get = esirisc_get_reg, .get = esirisc_get_reg,
.set = esirisc_set_reg, .set = esirisc_set_reg,
.flush = NULL,
}; };
static struct reg_cache *esirisc_build_reg_cache(struct target *target) static struct reg_cache *esirisc_build_reg_cache(struct target *target)

View File

@ -108,6 +108,7 @@ static int etb_get_reg(struct reg *reg)
static const struct reg_arch_type etb_reg_type = { static const struct reg_arch_type etb_reg_type = {
.get = etb_get_reg, .get = etb_get_reg,
.set = etb_set_reg_w_exec, .set = etb_set_reg_w_exec,
.flush = NULL,
}; };
struct reg_cache *etb_build_reg_cache(struct etb *etb) struct reg_cache *etb_build_reg_cache(struct etb *etb)

View File

@ -215,6 +215,7 @@ static int etm_write_reg(struct reg *reg, uint32_t value);
static const struct reg_arch_type etm_scan6_type = { static const struct reg_arch_type etm_scan6_type = {
.get = etm_get_reg, .get = etm_get_reg,
.set = etm_set_reg_w_exec, .set = etm_set_reg_w_exec,
.flush = NULL,
}; };
/* Look up register by ID ... most ETM instances only /* Look up register by ID ... most ETM instances only

View File

@ -357,6 +357,7 @@ static const struct reg_arch_type lakemont_reg_type = {
*/ */
.get = lakemont_get_core_reg, .get = lakemont_get_core_reg,
.set = lakemont_set_core_reg, .set = lakemont_set_core_reg,
.flush = NULL,
}; };
struct reg_cache *lakemont_build_reg_cache(struct target *t) struct reg_cache *lakemont_build_reg_cache(struct target *t)

View File

@ -179,6 +179,7 @@ static int mem_ap_reg_set(struct reg *reg, uint8_t *buf)
static struct reg_arch_type mem_ap_reg_arch_type = { static struct reg_arch_type mem_ap_reg_arch_type = {
.get = mem_ap_reg_get, .get = mem_ap_reg_get,
.set = mem_ap_reg_set, .set = mem_ap_reg_set,
.flush = NULL,
}; };
static const char *mem_ap_get_gdb_arch(const struct target *target) static const char *mem_ap_get_gdb_arch(const struct target *target)

View File

@ -496,6 +496,7 @@ int mips32_arch_state(struct target *target)
static const struct reg_arch_type mips32_reg_type = { static const struct reg_arch_type mips32_reg_type = {
.get = mips32_get_core_reg, .get = mips32_get_core_reg,
.set = mips32_set_core_reg, .set = mips32_set_core_reg,
.flush = NULL,
}; };
struct reg_cache *mips32_build_reg_cache(struct target *target) struct reg_cache *mips32_build_reg_cache(struct target *target)

View File

@ -370,6 +370,7 @@ int mips64_arch_state(struct target *target)
static const struct reg_arch_type mips64_reg_type = { static const struct reg_arch_type mips64_reg_type = {
.get = mips64_get_core_reg, .get = mips64_get_core_reg,
.set = mips64_set_core_reg, .set = mips64_set_core_reg,
.flush = NULL,
}; };
int mips64_build_reg_cache(struct target *target) int mips64_build_reg_cache(struct target *target)

View File

@ -494,6 +494,7 @@ static int or1k_set_core_reg(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type or1k_reg_type = { static const struct reg_arch_type or1k_reg_type = {
.get = or1k_get_core_reg, .get = or1k_get_core_reg,
.set = or1k_set_core_reg, .set = or1k_set_core_reg,
.flush = NULL,
}; };
static struct reg_cache *or1k_build_reg_cache(struct target *target) static struct reg_cache *or1k_build_reg_cache(struct target *target)

View File

@ -14,6 +14,8 @@
#include "register.h" #include "register.h"
#include <helper/log.h> #include <helper/log.h>
#include <target/target.h>
#include <target/target_type.h>
/** /**
* @file * @file
@ -115,12 +117,72 @@ static int register_set_dummy_core_reg(struct reg *reg, uint8_t *buf)
return ERROR_OK; return ERROR_OK;
} }
static int register_flush_dummy(struct reg *reg)
{
reg->dirty = false;
return ERROR_OK;
}
static const struct reg_arch_type dummy_type = { static const struct reg_arch_type dummy_type = {
.get = register_get_dummy_core_reg, .get = register_get_dummy_core_reg,
.set = register_set_dummy_core_reg, .set = register_set_dummy_core_reg,
.flush = register_flush_dummy,
}; };
void register_init_dummy(struct reg *reg) void register_init_dummy(struct reg *reg)
{ {
reg->type = &dummy_type; reg->type = &dummy_type;
} }
int register_flush(const struct target *target, struct reg *reg, bool invalidate)
{
if (!reg) {
LOG_ERROR("BUG: %s called with NULL", __func__);
return ERROR_FAIL;
}
if (!reg->exist) {
LOG_ERROR("BUG: %s called with non-existent register", __func__);
return ERROR_FAIL;
}
if (!reg->dirty) {
LOG_TARGET_DEBUG(target, "Register '%s' is not dirty, nothing to flush", reg->name);
if (reg->valid && invalidate) {
LOG_TARGET_DEBUG(target, "Invalidating register '%s'", reg->name);
reg->valid = false;
}
return ERROR_OK;
}
if (!reg->type->flush) {
LOG_TARGET_ERROR(target, "Unable to flush dirty register '%s' - operation not yet supported "
"by %s implementation in OpenOCD", reg->name, target->type->name);
return ERROR_NOT_IMPLEMENTED;
}
if (!reg->valid) {
LOG_ERROR("BUG: Register '%s' is not valid, but flush attempted", reg->name);
return ERROR_FAIL;
}
LOG_TARGET_DEBUG(target, "Flushing register '%s'", reg->name);
int result = reg->type->flush(reg);
if (result != ERROR_OK) {
LOG_TARGET_ERROR(target, "Failed to flush register '%s'", reg->name);
return result;
}
if (reg->dirty) {
LOG_ERROR("BUG: Register '%s' remained dirty after flushing", reg->name);
return ERROR_FAIL;
}
if (reg->valid && invalidate) {
LOG_TARGET_DEBUG(target, "Invalidating register '%s' after flush", reg->name);
reg->valid = false;
}
return ERROR_OK;
}

View File

@ -151,6 +151,7 @@ struct reg_cache {
struct reg_arch_type { struct reg_arch_type {
int (*get)(struct reg *reg); int (*get)(struct reg *reg);
int (*set)(struct reg *reg, uint8_t *buf); int (*set)(struct reg *reg, uint8_t *buf);
int (*flush)(struct reg *reg);
}; };
struct reg *register_get_by_number(struct reg_cache *first, struct reg *register_get_by_number(struct reg_cache *first,
@ -163,4 +164,7 @@ void register_cache_invalidate(struct reg_cache *cache);
void register_init_dummy(struct reg *reg); void register_init_dummy(struct reg *reg);
/* Flushes the register. Also invalidates the cached register value if invalidate == true */
int register_flush(const struct target *target, struct reg *reg, bool invalidate);
#endif /* OPENOCD_TARGET_REGISTER_H */ #endif /* OPENOCD_TARGET_REGISTER_H */

View File

@ -39,11 +39,10 @@ int riscv_program_init(struct riscv_program *p, struct target *t);
/* Write the program to the program buffer. */ /* Write the program to the program buffer. */
int riscv_program_write(struct riscv_program *program); int riscv_program_write(struct riscv_program *program);
/* Executes a program, returning 0 if the program successfully executed. Note /* Executes the RISC-V Program Buffer and returns ERROR_OK if the program
* that this may cause registers to be saved or restored, which could result to * buffer got successfully executed. In case of failure, more detailed error reason
* calls to things like riscv013_reg_save which itself could require a * can be found in p->execution_result.
* program to execute. That's OK, just make sure this eventually terminates. */
* */
int riscv_program_exec(struct riscv_program *p, struct target *t); int riscv_program_exec(struct riscv_program *p, struct target *t);
/* A lower level interface, you shouldn't use this unless you have a reason. */ /* A lower level interface, you shouldn't use this unless you have a reason. */

View File

@ -986,7 +986,7 @@ static int examine_progbuf(struct target *target)
return ERROR_OK; return ERROR_OK;
} }
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -1318,7 +1318,7 @@ static int fpr_read_progbuf(struct target *target, uint64_t *value,
const unsigned int freg = number - GDB_REGNO_FPR0; const unsigned int freg = number - GDB_REGNO_FPR0;
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -1348,7 +1348,7 @@ static int csr_read_progbuf(struct target *target, uint64_t *value,
assert(target->state == TARGET_HALTED); assert(target->state == TARGET_HALTED);
assert(number >= GDB_REGNO_CSR0 && number <= GDB_REGNO_CSR4095); assert(number >= GDB_REGNO_CSR0 && number <= GDB_REGNO_CSR4095);
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -1416,7 +1416,7 @@ static int fpr_write_progbuf(struct target *target, enum gdb_regno number,
assert(number >= GDB_REGNO_FPR0 && number <= GDB_REGNO_FPR31); assert(number >= GDB_REGNO_FPR0 && number <= GDB_REGNO_FPR31);
const unsigned int freg = number - GDB_REGNO_FPR0; const unsigned int freg = number - GDB_REGNO_FPR0;
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -1446,11 +1446,11 @@ static int vtype_write_progbuf(struct target *target, riscv_reg_t value)
{ {
assert(target->state == TARGET_HALTED); assert(target->state == TARGET_HALTED);
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (register_write_abstract(target, GDB_REGNO_S0, value) != ERROR_OK) if (register_write_abstract(target, GDB_REGNO_S0, value) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (riscv013_reg_save(target, GDB_REGNO_S1) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S1) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -1467,11 +1467,11 @@ static int vl_write_progbuf(struct target *target, riscv_reg_t value)
{ {
assert(target->state == TARGET_HALTED); assert(target->state == TARGET_HALTED);
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (register_write_abstract(target, GDB_REGNO_S0, value) != ERROR_OK) if (register_write_abstract(target, GDB_REGNO_S0, value) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (riscv013_reg_save(target, GDB_REGNO_S1) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S1) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -1490,7 +1490,7 @@ static int csr_write_progbuf(struct target *target, enum gdb_regno number,
assert(target->state == TARGET_HALTED); assert(target->state == TARGET_HALTED);
assert(number >= GDB_REGNO_CSR0 && number <= GDB_REGNO_CSR4095); assert(number >= GDB_REGNO_CSR0 && number <= GDB_REGNO_CSR4095);
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (register_write_abstract(target, GDB_REGNO_S0, value) != ERROR_OK) if (register_write_abstract(target, GDB_REGNO_S0, value) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
@ -2296,7 +2296,7 @@ int riscv013_get_register_buf(struct target *target, uint8_t *value,
&debug_vl, &debug_vsew) != ERROR_OK) &debug_vl, &debug_vsew) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
unsigned int vnum = regno - GDB_REGNO_V0; unsigned int vnum = regno - GDB_REGNO_V0;
@ -2351,7 +2351,7 @@ int riscv013_set_register_buf(struct target *target, enum gdb_regno regno,
&debug_vl, &debug_vsew) != ERROR_OK) &debug_vl, &debug_vsew) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
unsigned int vnum = regno - GDB_REGNO_V0; unsigned int vnum = regno - GDB_REGNO_V0;
@ -4278,11 +4278,11 @@ static int read_memory_progbuf_inner_fill_progbuf(struct target *target,
{ {
const bool is_repeated_read = increment == 0; const bool is_repeated_read = increment == 0;
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (riscv013_reg_save(target, GDB_REGNO_S1) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S1) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (is_repeated_read && riscv013_reg_save(target, GDB_REGNO_A0) != ERROR_OK) if (is_repeated_read && riscv013_reg_save_gpr(target, GDB_REGNO_A0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -4376,7 +4376,7 @@ read_memory_progbuf_inner_one(struct target *target, const riscv_mem_access_args
{ {
assert(riscv_mem_access_is_read(args)); assert(riscv_mem_access_is_read(args));
if (riscv013_reg_save(target, GDB_REGNO_S1) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S1) != ERROR_OK)
return mem_access_result(MEM_ACCESS_SKIPPED_REG_SAVE_FAILED); return mem_access_result(MEM_ACCESS_SKIPPED_REG_SAVE_FAILED);
struct riscv_program program; struct riscv_program program;
@ -4951,9 +4951,9 @@ static int write_memory_progbuf_try_to_write(struct target *target,
static int write_memory_progbuf_fill_progbuf(struct target *target, uint32_t size) static int write_memory_progbuf_fill_progbuf(struct target *target, uint32_t size)
{ {
if (riscv013_reg_save(target, GDB_REGNO_S0) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S0) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
if (riscv013_reg_save(target, GDB_REGNO_S1) != ERROR_OK) if (riscv013_reg_save_gpr(target, GDB_REGNO_S1) != ERROR_OK)
return ERROR_FAIL; return ERROR_FAIL;
struct riscv_program program; struct riscv_program program;
@ -5058,19 +5058,8 @@ struct target_type riscv013_target = {
int riscv013_get_register(struct target *target, int riscv013_get_register(struct target *target,
riscv_reg_t *value, enum gdb_regno rid) riscv_reg_t *value, enum gdb_regno rid)
{ {
/* It would be beneficial to move this redirection to the assert(rid != GDB_REGNO_PC && "'pc' should be read through 'dpc'");
* version-independent section, but there is a conflict: assert(rid != GDB_REGNO_PRIV && "'priv' should be read through 'dcsr'");
* `dcsr[5]` is `dcsr.v` in current spec, but it is `dcsr.debugint` in 0.11.
*/
if (rid == GDB_REGNO_PRIV) {
uint64_t dcsr;
if (riscv_reg_get(target, &dcsr, GDB_REGNO_DCSR) != ERROR_OK)
return ERROR_FAIL;
*value = set_field(0, VIRT_PRIV_V, get_field(dcsr, CSR_DCSR_V));
*value = set_field(*value, VIRT_PRIV_PRV, get_field(dcsr, CSR_DCSR_PRV));
return ERROR_OK;
}
LOG_TARGET_DEBUG(target, "reading register %s", riscv_reg_gdb_regno_name(target, rid)); LOG_TARGET_DEBUG(target, "reading register %s", riscv_reg_gdb_regno_name(target, rid));
if (dm013_select_target(target) != ERROR_OK) if (dm013_select_target(target) != ERROR_OK)
@ -5087,6 +5076,8 @@ int riscv013_get_register(struct target *target,
int riscv013_set_register(struct target *target, enum gdb_regno rid, int riscv013_set_register(struct target *target, enum gdb_regno rid,
riscv_reg_t value) riscv_reg_t value)
{ {
assert(rid != GDB_REGNO_PC && "'pc' should be written through 'dpc'");
assert(rid != GDB_REGNO_PRIV && "'priv' should be written through 'dcsr'");
LOG_TARGET_DEBUG(target, "writing 0x%" PRIx64 " to register %s", LOG_TARGET_DEBUG(target, "writing 0x%" PRIx64 " to register %s",
value, riscv_reg_gdb_regno_name(target, rid)); value, riscv_reg_gdb_regno_name(target, rid));

View File

@ -13,73 +13,455 @@
#include "debug_defines.h" #include "debug_defines.h"
#include <helper/time_support.h> #include <helper/time_support.h>
static int riscv013_reg_get(struct reg *reg) /* Start of register fetch/send definitions.
* - "reg_fetcher" (named "fetch_*") -- loads the value from target into reg
* cache entry. Modifies only "reg->value".
* - "reg_sender" (named "send_*") -- stores the value from "buf" to the
* target. Does not modify "reg" at all.
*/
typedef int (*reg_fetcher)(struct reg *reg);
typedef int (*reg_sender)(const struct reg *reg, const uint8_t *buf);
/* TODO: Introduce separate "fetch_*" and "send_*" functions for register
* classes (xregs, fregs, csrs).
*/
static int fetch_riscv013_reg(struct reg *reg)
{ {
struct target *target = riscv_reg_impl_get_target(reg); riscv_reg_t value;
int res = riscv013_get_register(riscv_reg_impl_get_target(reg),
&value, reg->number);
if (res != ERROR_OK)
return res;
buf_set_u64(reg->value, 0, reg->size, value);
return res;
}
/* TODO: Hack to deal with gdb that thinks these registers still exist. */ static int send_riscv013_reg(const struct reg *reg, const uint8_t *buf)
if (reg->number > GDB_REGNO_XPR15 && reg->number <= GDB_REGNO_XPR31 && {
riscv_supports_extension(target, 'E')) { assert(reg->size <= 64);
buf_set_u64(reg->value, 0, reg->size, 0); const riscv_reg_t value = buf_get_u64(buf, 0, reg->size);
return ERROR_OK; return riscv013_set_register(riscv_reg_impl_get_target(reg),
reg->number, value);
}
/* TODO: Inline "riscv013_*_register_buf" into "vreg" accessors.
*/
static int fetch_vreg(struct reg *reg)
{
return riscv013_get_register_buf(riscv_reg_impl_get_target(reg),
reg->value, reg->number);
}
static int send_vreg(const struct reg *reg, const uint8_t *buf)
{
return riscv013_set_register_buf(riscv_reg_impl_get_target(reg),
reg->number, reg->value);
}
/* End of register fetch/send definitions. */
/* Start of cache entry utils. */
static void set_cache_value(struct reg *reg, const uint8_t *buf)
{
assert(riscv_reg_impl_is_existing(reg));
buf_cpy(buf, reg->value, reg->size);
}
/**
* Note: There are only 3 reachable states for an item in the register cache:
* |: state :|: "reg->valid" :|: "reg->dirty" :|
* | "invalid" | false | false |
* | "valid" | true | false |
* | "dirty" | true | true |
*
* The state "reg->valid == dirty && reg->valid == false" should be unreacheable.
* To achieve this, "reg->valid" and "reg->dirty" are not assigned directly
* outside of "mark_<state>()" functions.
*/
static void mark_invalid(struct reg *reg)
{
assert(riscv_reg_impl_is_existing(reg));
reg->valid = false;
reg->dirty = false;
}
static void mark_clean(struct reg *reg)
{
assert(riscv_reg_impl_is_existing(reg));
assert(riscv_reg_impl_get_target(reg)->state == TARGET_HALTED);
reg->valid = true;
reg->dirty = false;
}
static void mark_dirty(struct reg *reg)
{
assert(riscv_reg_impl_is_existing(reg));
assert(riscv_reg_impl_get_target(reg)->state == TARGET_HALTED);
reg->valid = true;
reg->dirty = true;
}
static int access_prologue(const struct reg *reg, bool write)
{
assert(riscv_reg_impl_is_initialized(reg));
const struct target * const target = riscv_reg_impl_get_target(reg);
if (!reg->exist) {
LOG_TARGET_DEBUG(target, "Register %s does not exist",
reg->name);
return ERROR_FAIL;
} }
LOG_TARGET_DEBUG(target, "%s %s (valid=%s, dirty=%s)",
if (reg->number >= GDB_REGNO_V0 && reg->number <= GDB_REGNO_V31) { write ? "Writing" : "Reading",
if (riscv013_get_register_buf(target, reg->value, reg->number) != ERROR_OK) reg->name,
return ERROR_FAIL; reg->valid ? "true" : "false",
reg->dirty ? "true" : "false");
reg->valid = riscv_reg_impl_gdb_regno_cacheable(reg->number, /* is write? */ false);
} else {
uint64_t value;
int result = riscv_reg_get(target, &value, reg->number);
if (result != ERROR_OK)
return result;
buf_set_u64(reg->value, 0, reg->size, value);
}
char *str = buf_to_hex_str(reg->value, reg->size);
LOG_TARGET_DEBUG(target, "Read 0x%s from %s (valid=%d).", str, reg->name,
reg->valid);
free(str);
return ERROR_OK; return ERROR_OK;
} }
static int riscv013_reg_set(struct reg *reg, uint8_t *buf) static void access_epilogue(const struct reg *reg, bool write)
{ {
struct target *target = riscv_reg_impl_get_target(reg); assert(riscv_reg_impl_is_existing(reg));
if (debug_level < LOG_LVL_DEBUG)
char *str = buf_to_hex_str(buf, reg->size); return;
LOG_TARGET_DEBUG(target, "Write 0x%s to %s (valid=%d).", str, reg->name, char *value_hex = buf_to_hex_str(reg->value, reg->size);
reg->valid); LOG_TARGET_DEBUG(riscv_reg_impl_get_target(reg),
free(str); "%s 0x%s %s %s (valid=%s, dirty=%s)",
write ? "Wrote" : "Read",
/* TODO: Hack to deal with gdb that thinks these registers still exist. */ value_hex,
if (reg->number > GDB_REGNO_XPR15 && reg->number <= GDB_REGNO_XPR31 && write ? "to" : "from",
riscv_supports_extension(target, 'E') && reg->name,
buf_get_u64(buf, 0, reg->size) == 0) reg->valid ? "true" : "false",
return ERROR_OK; reg->dirty ? "true" : "false");
free(value_hex);
if (reg->number >= GDB_REGNO_V0 && reg->number <= GDB_REGNO_V31) { }
if (riscv013_set_register_buf(target, reg->number, buf) != ERROR_OK)
return ERROR_FAIL;
memcpy(reg->value, buf, DIV_ROUND_UP(reg->size, 8));
reg->valid = riscv_reg_impl_gdb_regno_cacheable(reg->number, /* is write? */ true);
} else {
const riscv_reg_t value = buf_get_u64(buf, 0, reg->size);
if (riscv_reg_set(target, reg->number, value) != ERROR_OK)
return ERROR_FAIL;
}
static int noncaching_get(struct reg *reg, reg_fetcher fetch_reg)
{
assert(riscv_reg_impl_is_initialized(reg));
int res = access_prologue(reg, /*write*/ false);
if (res != ERROR_OK)
return res;
res = fetch_reg(reg);
mark_invalid(reg);
if (res != ERROR_OK)
return res;
access_epilogue(reg, /*write*/ false);
return ERROR_OK; return ERROR_OK;
} }
static int caching_get(struct reg *reg, reg_fetcher fetch_reg)
{
assert(riscv_reg_impl_is_initialized(reg));
int res = access_prologue(reg, /*write*/ false);
if (res != ERROR_OK)
return res;
if (reg->valid) {
LOG_TARGET_DEBUG(riscv_reg_impl_get_target(reg),
"Reading %s from cache", reg->name);
return ERROR_OK;
}
res = fetch_reg(reg);
if (res != ERROR_OK) {
mark_invalid(reg);
return res;
}
if (riscv_reg_impl_get_target(reg)->state == TARGET_HALTED)
mark_clean(reg);
else
mark_invalid(reg);
access_epilogue(reg, /*write*/ false);
return ERROR_OK;
}
static int noncaching_set(struct reg *reg, const uint8_t *buf,
reg_sender send_reg)
{
int res = access_prologue(reg, /*write*/ true);
if (res != ERROR_OK)
return res;
res = send_reg(reg, buf);
mark_invalid(reg);
if (res != ERROR_OK)
return res;
set_cache_value(reg, buf);
access_epilogue(reg, /*write*/ true);
return ERROR_OK;
}
static int caching_set(struct reg *reg, const uint8_t *buf,
reg_sender send_reg)
{
assert(riscv_reg_impl_is_initialized(reg));
const struct target * const target = riscv_reg_impl_get_target(reg);
if (target->state != TARGET_HALTED) {
LOG_TARGET_DEBUG(target, "Not caching the write to %s", reg->name);
return noncaching_set(reg, buf, send_reg);
}
int res = access_prologue(reg, /*write*/ true);
if (res != ERROR_OK)
return res;
if (reg->valid && buf_eq(reg->value, buf, reg->size)) {
LOG_TARGET_DEBUG(riscv_reg_impl_get_target(reg),
"Writing the same value to %s", reg->name);
} else {
LOG_TARGET_DEBUG(target, "Caching the write to %s", reg->name);
set_cache_value(reg, buf);
mark_dirty(reg);
}
access_epilogue(reg, /*write*/ true);
return ERROR_OK;
}
static int flush_reg(struct reg *reg, reg_sender send_reg)
{
if (!reg->dirty)
return ERROR_OK;
int res = access_prologue(reg, /*write*/ true);
if (res != ERROR_OK)
return res;
assert(reg->dirty);
assert(reg->valid);
const struct target * const target = riscv_reg_impl_get_target(reg);
if (target->state != TARGET_HALTED) {
LOG_TARGET_ERROR(target,
"BUG: register %s is dirty while the target is not halted",
reg->name);
return ERROR_TARGET_NOT_HALTED;
}
res = send_reg(reg, reg->value);
if (res != ERROR_OK) {
/* Register is not marked "invalid" here (like it's done on failure in
* "caching_set") since it is "dirty", i.e. the most recent value
* is currently in the cache. */
return res;
}
mark_clean(reg);
access_epilogue(reg, /*write*/ true);
return res;
}
/* End of cache entry utils. */
/* Start of "reg_arch_type" method definitions. */
static int riscv013_noncaching_get(struct reg *reg)
{
return noncaching_get(reg, fetch_riscv013_reg);
}
static int riscv013_caching_get(struct reg *reg)
{
return caching_get(reg, fetch_riscv013_reg);
}
static int riscv013_noncaching_set(struct reg *reg, uint8_t *buf)
{
return noncaching_set(reg, buf, send_riscv013_reg);
}
static int riscv013_caching_set(struct reg *reg, uint8_t *buf)
{
return caching_set(reg, buf, send_riscv013_reg);
}
static int riscv013_reg_flush(struct reg *reg)
{
return flush_reg(reg, send_riscv013_reg);
}
static int flush_noncacheable(struct reg *reg)
{
assert(!reg->dirty && "Non-cacheable register must never get dirty");
return ERROR_OK;
}
static int get_vreg(struct reg *reg)
{
assert(reg->number >= GDB_REGNO_V0 && reg->number <= GDB_REGNO_V31);
return caching_get(reg, fetch_vreg);
}
static int set_vreg(struct reg *reg, uint8_t *buf)
{
assert(reg->number >= GDB_REGNO_V0 && reg->number <= GDB_REGNO_V31);
return caching_set(reg, buf, send_vreg);
}
static int flush_vreg(struct reg *reg)
{
assert(reg->number >= GDB_REGNO_V0 && reg->number <= GDB_REGNO_V31);
return flush_reg(reg, send_vreg);
}
static int get_pc(struct reg *pc)
{
assert(pc->number == GDB_REGNO_PC);
assert(!pc->valid);
assert(!pc->dirty);
struct reg * const dpc =
riscv_reg_impl_cache_entry(riscv_reg_impl_get_target(pc),
GDB_REGNO_DPC);
int res = dpc->type->get(dpc);
if (res == ERROR_OK) {
/* "pc" is just an alias for "dpc". "dpc" just got updated, sync up "pc" */
set_cache_value(pc, dpc->value);
}
return res;
}
static int set_pc(struct reg *pc, uint8_t *buf)
{
assert(pc->number == GDB_REGNO_PC);
assert(!pc->valid);
assert(!pc->dirty);
struct reg * const dpc =
riscv_reg_impl_cache_entry(riscv_reg_impl_get_target(pc),
GDB_REGNO_DPC);
int res = dpc->type->set(dpc, buf);
if (res == ERROR_OK) {
/* "pc" is just an alias for "dpc". "dpc" just got updated, sync up "pc" */
set_cache_value(pc, dpc->value);
}
return res;
}
static int get_priv(struct reg *priv)
{
assert(priv->number == GDB_REGNO_PRIV);
assert(!priv->valid);
assert(!priv->dirty);
struct reg * const dcsr =
riscv_reg_impl_cache_entry(riscv_reg_impl_get_target(priv),
GDB_REGNO_DCSR);
assert(dcsr->size == 32);
int res = dcsr->type->get(dcsr);
if (res != ERROR_OK)
return res;
buf_set_u32(priv->value, VIRT_PRIV_PRV_OFFSET, VIRT_PRIV_PRV_LENGTH,
buf_get_u32(dcsr->value, CSR_DCSR_PRV_OFFSET, CSR_DCSR_PRV_LENGTH));
buf_set_u32(priv->value, VIRT_PRIV_V_OFFSET, VIRT_PRIV_V_LENGTH,
buf_get_u32(dcsr->value, CSR_DCSR_V_OFFSET, CSR_DCSR_V_LENGTH));
return ERROR_OK;
}
static int set_priv(struct reg *priv, uint8_t *priv_buf)
{
assert(priv->number == GDB_REGNO_PRIV);
assert(!priv->valid);
assert(!priv->dirty);
struct reg * const dcsr = riscv_reg_impl_cache_entry(riscv_reg_impl_get_target(priv),
GDB_REGNO_DCSR);
int res = dcsr->type->get(dcsr);
if (res != ERROR_OK)
return res;
assert(dcsr->size == 32);
uint8_t dcsr_buf[32 / 8];
buf_cpy(dcsr->value, dcsr_buf, dcsr->size);
buf_set_u32(dcsr_buf, CSR_DCSR_PRV_OFFSET, CSR_DCSR_PRV_LENGTH,
buf_get_u32(priv_buf, VIRT_PRIV_PRV_OFFSET, VIRT_PRIV_PRV_LENGTH));
buf_set_u32(dcsr_buf, CSR_DCSR_V_OFFSET, CSR_DCSR_V_LENGTH,
buf_get_u32(priv_buf, VIRT_PRIV_V_OFFSET, VIRT_PRIV_V_LENGTH));
res = dcsr->type->set(dcsr, dcsr_buf);
if (res == ERROR_OK)
set_cache_value(priv, priv_buf);
return res;
}
/* End of "reg_arch_type" method definitions. */
static const struct reg_arch_type zero_type = {
.get = riscv013_caching_get,
/* "zero" is read-only, but there is nothig wrong with attempting to
* write it (e.g. for the purpose of HW testing). */
.set = riscv013_noncaching_set,
.flush = riscv013_reg_flush,
};
static const struct reg_arch_type xreg_type = {
.get = riscv013_caching_get,
.set = riscv013_caching_set,
.flush = riscv013_reg_flush,
};
static const struct reg_arch_type freg_type = {
.get = riscv013_caching_get,
.set = riscv013_caching_set,
.flush = riscv013_reg_flush,
};
static const struct reg_arch_type vreg_type = {
.get = get_vreg,
.set = set_vreg,
.flush = flush_vreg
};
static const struct reg_arch_type pc_type = {
.get = get_pc,
.set = set_pc,
.flush = flush_noncacheable
};
static const struct reg_arch_type priv_type = {
.get = get_priv,
.set = set_priv,
.flush = flush_noncacheable
};
static const struct reg_arch_type warl_csr_type = {
.get = riscv013_caching_get,
.set = riscv013_noncaching_set,
.flush = riscv013_reg_flush
};
static const struct reg_arch_type noncacheable_type = {
.get = riscv013_noncaching_get,
.set = riscv013_noncaching_set,
.flush = flush_noncacheable
};
static const struct reg_arch_type *riscv013_gdb_regno_reg_type(uint32_t regno) static const struct reg_arch_type *riscv013_gdb_regno_reg_type(uint32_t regno)
{ {
static const struct reg_arch_type riscv013_reg_type = { if (regno != GDB_REGNO_ZERO && regno <= GDB_REGNO_XPR31)
.get = riscv013_reg_get, return &xreg_type;
.set = riscv013_reg_set if (regno >= GDB_REGNO_FPR0 && regno <= GDB_REGNO_FPR31) {
}; /* For now "freg_type" is the same as "xreg_type", but it will be
return &riscv013_reg_type; * changed soon */
return &freg_type;
}
if (regno >= GDB_REGNO_V0 && regno <= GDB_REGNO_V31)
return &vreg_type;
switch (regno) {
case GDB_REGNO_ZERO:
return &zero_type;
case GDB_REGNO_PC:
return &pc_type;
case GDB_REGNO_PRIV:
return &priv_type;
case GDB_REGNO_VLENB:
/* "vlenb" is read-only, but there is nothig wrong with attempting to
* write it (e.g. for the purpose of HW testing). */
case GDB_REGNO_DPC:
case GDB_REGNO_VSTART:
case GDB_REGNO_VXSAT:
case GDB_REGNO_VXRM:
case GDB_REGNO_VL:
case GDB_REGNO_VTYPE:
case GDB_REGNO_MISA:
case GDB_REGNO_DCSR:
case GDB_REGNO_DSCRATCH0:
case GDB_REGNO_MEPC:
case GDB_REGNO_SATP:
return &warl_csr_type;
/* TODO: https://github.com/riscv-collab/riscv-openocd/issues/1219 */
case GDB_REGNO_MSTATUS:
case GDB_REGNO_MCAUSE:
/* TODO: Sdtrig register should be WARL. Note that "tdata*" and "tinfo"
* values must get invalidated when "tselect" changes. */
case GDB_REGNO_TSELECT:
case GDB_REGNO_TDATA1:
case GDB_REGNO_TDATA2:
default:
return &noncacheable_type;
}
} }
static int init_cache_entry(struct target *target, uint32_t regno) static int init_cache_entry(struct target *target, uint32_t regno)
@ -345,42 +727,30 @@ int riscv013_reg_examine_all(struct target *target)
return ERROR_OK; return ERROR_OK;
} }
/** int riscv013_reg_save_gpr(struct target *target, enum gdb_regno regid)
* This function is used to save the value of a register in cache. The register
* is marked as dirty, and writeback is delayed for as long as possible.
*/
int riscv013_reg_save(struct target *target, enum gdb_regno regid)
{ {
if (target->state != TARGET_HALTED) { assert(regid >= GDB_REGNO_ZERO && regid <= GDB_REGNO_XPR31);
LOG_TARGET_ERROR(target, "Can't save register %s on a hart that is not halted.", struct reg *reg = riscv_reg_impl_cache_entry(target, regid);
riscv_reg_gdb_regno_name(target, regid)); assert(riscv_reg_impl_is_initialized(reg));
return ERROR_FAIL;
}
assert(riscv_reg_impl_gdb_regno_cacheable(regid, /* is write? */ false) &&
"Only cacheable registers can be saved.");
RISCV_INFO(r); RISCV_INFO(r);
riscv_reg_t value; if (target->state != TARGET_HALTED) {
if (!target->reg_cache) { LOG_TARGET_ERROR(target, "Can't save register %s on a hart that is not halted",
assert(!target_was_examined(target)); reg->name);
/* To create register cache it is needed to examine the target first, return ERROR_TARGET_NOT_HALTED;
* therefore during examine, any changed register needs to be saved
* and restored manually.
*/
return ERROR_OK;
} }
struct reg *reg = riscv_reg_impl_cache_entry(target, regid);
LOG_TARGET_DEBUG(target, "Saving %s", reg->name); LOG_TARGET_DEBUG(target, "Saving %s", reg->name);
if (riscv_reg_get(target, &value, regid) != ERROR_OK) int res = reg->type->get(reg);
return ERROR_FAIL; if (res != ERROR_OK)
return res;
assert(reg->valid && assert(reg->valid &&
"The register is cacheable, so the cache entry must be valid now."); "The register is cacheable, so the cache entry must be valid now.");
/* Mark the register dirty. We assume that this function is called /* Mark the register as dirty so that its value is written back to the target
* during the next register cache flush. We assume that this function is called
* because the caller is about to mess with the underlying value of the * because the caller is about to mess with the underlying value of the
* register. */ * register, and wants to make sure the value gets restored later. */
reg->dirty = true; reg->dirty = true;
r->last_activity = timeval_ms(); r->last_activity = timeval_ms();

View File

@ -20,13 +20,10 @@
int riscv013_reg_examine_all(struct target *target); int riscv013_reg_examine_all(struct target *target);
/** /**
* This function is used to save the value of a register in cache. The register * This function is used to save the value of a GPR in cache. The register is
* is marked as dirty, and writeback is delayed for as long as possible. * marked as dirty, and writeback is delayed for as long as possible.
* Generally used to save registers before program buffer execution. * Generally used to save registers before program buffer execution.
*
* TODO: The interface should be restricted in such a way that only GPRs can be
* saved.
*/ */
int riscv013_reg_save(struct target *target, enum gdb_regno regid); int riscv013_reg_save_gpr(struct target *target, enum gdb_regno regid);
#endif /* OPENOCD_TARGET_RISCV_RISCV_013_REG_H */ #endif /* OPENOCD_TARGET_RISCV_RISCV_013_REG_H */

View File

@ -8,19 +8,6 @@
#include "riscv.h" #include "riscv.h"
#include "riscv_reg.h" #include "riscv_reg.h"
#include "riscv_reg_impl.h" #include "riscv_reg_impl.h"
/**
* TODO: Currently `reg->get/set` is implemented in terms of
* `riscv_get/set_register`. However, the intention behind
* `riscv_get/set_register` is to work with the cache, therefore it accesses
* and modifyes register cache directly. The idea is to implement
* `riscv_get/set_register` in terms of `riscv_reg_impl_cache_entry` and
* `reg->get/set`.
* Once this is done, the following includes should be removed.
*/
#include "debug_defines.h"
#include "riscv-011.h"
#include "riscv-013.h"
#include "field_helpers.h"
static const char * const default_reg_names[GDB_REGNO_COUNT] = { static const char * const default_reg_names[GDB_REGNO_COUNT] = {
[GDB_REGNO_ZERO] = "zero", [GDB_REGNO_ZERO] = "zero",
@ -586,6 +573,7 @@ static int resize_reg(const struct target *target, uint32_t regno, bool exist,
reg->size = size; reg->size = size;
reg->exist = exist; reg->exist = exist;
if (reg->exist) { if (reg->exist) {
assert(size > 0);
reg->value = malloc(DIV_ROUND_UP(reg->size, 8)); reg->value = malloc(DIV_ROUND_UP(reg->size, 8));
if (!reg->value) { if (!reg->value) {
LOG_ERROR("Failed to allocate memory."); LOG_ERROR("Failed to allocate memory.");
@ -784,97 +772,19 @@ int riscv_reg_flush_all(struct target *target)
* may become dirty in the process (e.g. S0, S1). For that reason, flush * may become dirty in the process (e.g. S0, S1). For that reason, flush
* registers in reverse order, so that GPRs are flushed last. * registers in reverse order, so that GPRs are flushed last.
*/ */
int res = ERROR_OK;
for (unsigned int number = target->reg_cache->num_regs; number-- > 0; ) { for (unsigned int number = target->reg_cache->num_regs; number-- > 0; ) {
struct reg *reg = riscv_reg_impl_cache_entry(target, number); struct reg *reg = riscv_reg_impl_cache_entry(target, number);
if (reg->valid && reg->dirty) { assert(reg->type->flush
riscv_reg_t value = buf_get_u64(reg->value, 0, reg->size); && "All RISC-V registers must have the 'flush' method defined");
if (reg->type->flush(reg) != ERROR_OK)
LOG_TARGET_DEBUG(target, "%s is dirty; write back 0x%" PRIx64, res = ERROR_FAIL;
reg->name, value);
if (riscv_reg_write(target, number, value) != ERROR_OK)
return ERROR_FAIL;
}
} }
LOG_TARGET_DEBUG(target, "Flush of register cache completed"); if (res == ERROR_OK)
return ERROR_OK; LOG_TARGET_DEBUG(target, "Flush of register cache completed successfully");
} else
LOG_TARGET_ERROR(target, "Flush of register cache failed");
/** return res;
* This function is used internally by functions that change register values.
* If `write_through` is true, it is ensured that the value of the target's
* register is set to be equal to the `value` argument. The cached value is
* updated if the register is cacheable.
* TODO: Currently `reg->get/set` is implemented in terms of
* `riscv_get/set_register`. However, the intention behind
* `riscv_get/set_register` is to work with the cache, therefore it accesses
* and modifyes register cache directly. The idea is to implement
* `riscv_get/set_register` in terms of `riscv_reg_impl_cache_entry` and
* `reg->get/set`.
*/
static int riscv_set_or_write_register(struct target *target,
enum gdb_regno regid, riscv_reg_t value, bool write_through)
{
RISCV_INFO(r);
assert(r);
if (r->dtm_version == DTM_DTMCS_VERSION_0_11)
return riscv011_set_register(target, regid, value);
keep_alive();
if (regid == GDB_REGNO_PC) {
return riscv_set_or_write_register(target, GDB_REGNO_DPC, value, write_through);
} else if (regid == GDB_REGNO_PRIV) {
riscv_reg_t dcsr;
if (riscv_reg_get(target, &dcsr, GDB_REGNO_DCSR) != ERROR_OK)
return ERROR_FAIL;
dcsr = set_field(dcsr, CSR_DCSR_PRV, get_field(value, VIRT_PRIV_PRV));
dcsr = set_field(dcsr, CSR_DCSR_V, get_field(value, VIRT_PRIV_V));
return riscv_set_or_write_register(target, GDB_REGNO_DCSR, dcsr, write_through);
}
struct reg *reg = riscv_reg_impl_cache_entry(target, regid);
assert(riscv_reg_impl_is_initialized(reg));
if (!reg->exist) {
LOG_TARGET_DEBUG(target, "Register %s does not exist.", reg->name);
return ERROR_FAIL;
}
if (target->state != TARGET_HALTED) {
LOG_TARGET_DEBUG(target,
"Target not halted, writing to target: %s <- 0x%" PRIx64,
reg->name, value);
return riscv013_set_register(target, regid, value);
}
const bool need_to_write = !reg->valid || reg->dirty ||
value != buf_get_u64(reg->value, 0, reg->size);
const bool cacheable = riscv_reg_impl_gdb_regno_cacheable(regid, need_to_write);
if (!cacheable || (write_through && need_to_write)) {
LOG_TARGET_DEBUG(target,
"Writing to target: %s <- 0x%" PRIx64 " (cacheable=%s, valid=%s, dirty=%s)",
reg->name, value, cacheable ? "true" : "false",
reg->valid ? "true" : "false",
reg->dirty ? "true" : "false");
if (riscv013_set_register(target, regid, value) != ERROR_OK)
return ERROR_FAIL;
reg->dirty = false;
} else {
reg->dirty = need_to_write;
}
buf_set_u64(reg->value, 0, reg->size, value);
reg->valid = cacheable;
LOG_TARGET_DEBUG(target,
"Wrote 0x%" PRIx64 " to %s (cacheable=%s, valid=%s, dirty=%s)",
value, reg->name, cacheable ? "true" : "false",
reg->valid ? "true" : "false",
reg->dirty ? "true" : "false");
return ERROR_OK;
} }
bool riscv_reg_cache_any_dirty(const struct target *target, int log_level) bool riscv_reg_cache_any_dirty(const struct target *target, int log_level)
@ -908,83 +818,49 @@ void riscv_reg_cache_invalidate_all(struct target *target)
/** /**
* This function is used to change the value of a register. The new value may * This function is used to change the value of a register. The new value may
* be cached, and may not be written until the hart is resumed. * be cached, and may not be written until the hart is resumed.
* TODO: Currently `reg->get/set` is implemented in terms of
* `riscv_get/set_register`. However, the intention behind
* `riscv_get/set_register` is to work with the cache, therefore it accesses
* and modifyes register cache directly. The idea is to implement
* `riscv_get/set_register` in terms of `riscv_reg_impl_cache_entry` and
* `reg->get/set`.
*/ */
int riscv_reg_set(struct target *target, enum gdb_regno regid, int riscv_reg_set(struct target *target, enum gdb_regno regid,
riscv_reg_t value) riscv_reg_t value)
{ {
return riscv_set_or_write_register(target, regid, value, struct reg * const reg = riscv_reg_impl_cache_entry(target, regid);
/* write_through */ false); assert(riscv_reg_impl_is_initialized(reg));
assert(reg->size <= sizeof(riscv_reg_t) * CHAR_BIT);
assert(reg->size <= 64);
uint8_t buf[64 / 8];
buf_set_u64(buf, 0, reg->size, value);
return reg->type->set(reg, buf);
} }
/** /**
* This function is used to change the value of a register. The new value may * This function is used to change the value of a register. The new value may
* be cached, but it will be written to hart immediately. * be cached, but it will be written to hart immediately.
* TODO: Currently `reg->get/set` is implemented in terms of
* `riscv_get/set_register`. However, the intention behind
* `riscv_get/set_register` is to work with the cache, therefore it accesses
* and modifyes register cache directly. The idea is to implement
* `riscv_get/set_register` in terms of `riscv_reg_impl_cache_entry` and
* `reg->get/set`.
*/ */
int riscv_reg_write(struct target *target, enum gdb_regno regid, int riscv_reg_write(struct target *target, enum gdb_regno regid,
riscv_reg_t value) riscv_reg_t value)
{ {
return riscv_set_or_write_register(target, regid, value, int res = riscv_reg_set(target, regid, value);
/* write_through */ true); if (res != ERROR_OK)
return res;
struct reg * const reg = riscv_reg_impl_cache_entry(target, regid);
return reg->type->flush(reg);
} }
/** /**
* This function is used to get the value of a register. If possible, the value * This function is used to get the value of a register. If possible, the value
* in cache will be updated. * in cache will be updated.
* TODO: Currently `reg->get/set` is implemented in terms of
* `riscv_get/set_register`. However, the intention behind
* `riscv_get/set_register` is to work with the cache, therefore it accesses
* and modifyes register cache directly. The idea is to implement
* `riscv_get/set_register` in terms of `riscv_reg_impl_cache_entry` and
* `reg->get/set`.
*/ */
int riscv_reg_get(struct target *target, riscv_reg_t *value, int riscv_reg_get(struct target *target, riscv_reg_t *value,
enum gdb_regno regid) enum gdb_regno regid)
{ {
RISCV_INFO(r); assert(value);
assert(r);
if (r->dtm_version == DTM_DTMCS_VERSION_0_11)
return riscv011_get_register(target, value, regid);
keep_alive(); struct reg * const reg = riscv_reg_impl_cache_entry(target, regid);
if (regid == GDB_REGNO_PC)
return riscv_reg_get(target, value, GDB_REGNO_DPC);
struct reg *reg = riscv_reg_impl_cache_entry(target, regid);
assert(riscv_reg_impl_is_initialized(reg)); assert(riscv_reg_impl_is_initialized(reg));
if (!reg->exist) { assert(reg->size <= sizeof(riscv_reg_t) * CHAR_BIT);
LOG_TARGET_DEBUG(target, "Register %s does not exist.", reg->name); assert(reg->size <= 64);
return ERROR_FAIL; int res = reg->type->get(reg);
} if (res != ERROR_OK)
return res;
if (reg->valid) { *value = buf_get_u64(reg->value, 0, reg->size);
*value = buf_get_u64(reg->value, 0, reg->size);
LOG_TARGET_DEBUG(target, "Read %s: 0x%" PRIx64 " (cached)", reg->name,
*value);
return ERROR_OK;
}
LOG_TARGET_DEBUG(target, "Reading %s from target", reg->name);
if (riscv013_get_register(target, value, regid) != ERROR_OK)
return ERROR_FAIL;
buf_set_u64(reg->value, 0, reg->size, *value);
reg->valid = riscv_reg_impl_gdb_regno_cacheable(regid, /* is write? */ false) &&
target->state == TARGET_HALTED;
reg->dirty = false;
LOG_TARGET_DEBUG(target, "Read %s: 0x%" PRIx64, reg->name, *value);
return ERROR_OK; return ERROR_OK;
} }

View File

@ -30,6 +30,16 @@ static inline bool riscv_reg_impl_is_initialized(const struct reg *reg)
assert(reg->valid || !reg->dirty); assert(reg->valid || !reg->dirty);
return true; return true;
} }
static inline bool riscv_reg_impl_is_existing(const struct reg *reg)
{
assert(riscv_reg_impl_is_initialized(reg));
if (!reg->exist)
return false;
assert(reg->size > 0);
assert(reg->value);
return true;
}
/** /**
* Initialize register cache. Note, that each specific register cache entry is * Initialize register cache. Note, that each specific register cache entry is
* not initialized by this function. * not initialized by this function.
@ -171,52 +181,4 @@ int riscv_reg_impl_expose_csrs(const struct target *target);
/** Hide additional CSRs, as specified by `riscv_info_t::hide_csr` list. */ /** Hide additional CSRs, as specified by `riscv_info_t::hide_csr` list. */
void riscv_reg_impl_hide_csrs(const struct target *target); void riscv_reg_impl_hide_csrs(const struct target *target);
/**
* If write is true:
* return true iff we are guaranteed that the register will contain exactly
* the value we just wrote when it's read.
* If write is false:
* return true iff we are guaranteed that the register will read the same
* value in the future as the value we just read.
*/
static inline bool riscv_reg_impl_gdb_regno_cacheable(enum gdb_regno regno,
bool is_write)
{
if (regno == GDB_REGNO_ZERO)
return !is_write;
/* GPRs, FPRs, vector registers are just normal data stores. */
if (regno <= GDB_REGNO_XPR31 ||
(regno >= GDB_REGNO_FPR0 && regno <= GDB_REGNO_FPR31) ||
(regno >= GDB_REGNO_V0 && regno <= GDB_REGNO_V31))
return true;
/* Most CSRs won't change value on us, but we can't assume it about arbitrary
* CSRs. */
switch (regno) {
case GDB_REGNO_DPC:
case GDB_REGNO_VSTART:
case GDB_REGNO_VXSAT:
case GDB_REGNO_VXRM:
case GDB_REGNO_VLENB:
case GDB_REGNO_VL:
case GDB_REGNO_VTYPE:
case GDB_REGNO_MISA:
case GDB_REGNO_DCSR:
case GDB_REGNO_DSCRATCH0:
case GDB_REGNO_MEPC:
case GDB_REGNO_SATP:
/*
* WARL registers might not contain the value we just wrote, but
* these ones won't spontaneously change their value either. *
*/
return !is_write;
case GDB_REGNO_TSELECT: /* I think this should be above, but then it doesn't work. */
case GDB_REGNO_TDATA1: /* Changes value when tselect is changed. */
case GDB_REGNO_TDATA2: /* Changes value when tselect is changed. */
default:
return false;
}
}
#endif /* OPENOCD_TARGET_RISCV_RISCV_REG_IMPL_H */ #endif /* OPENOCD_TARGET_RISCV_RISCV_REG_IMPL_H */

View File

@ -1180,6 +1180,7 @@ static int stm8_get_gdb_reg_list(struct target *target, struct reg **reg_list[],
static const struct reg_arch_type stm8_reg_type = { static const struct reg_arch_type stm8_reg_type = {
.get = stm8_get_core_reg, .get = stm8_get_core_reg,
.set = stm8_set_core_reg, .set = stm8_set_core_reg,
.flush = NULL,
}; };
static struct reg_cache *stm8_build_reg_cache(struct target *target) static struct reg_cache *stm8_build_reg_cache(struct target *target)

View File

@ -1470,6 +1470,86 @@ unsigned int target_data_bits(struct target *target)
return 32; return 32;
} }
static bool target_implements_reg_flush(const struct reg_cache *cache)
{
while (cache) {
for (unsigned int i = 0; i < cache->num_regs; i++) {
struct reg *reg = &cache->reg_list[i];
if (reg->type->flush)
return true;
}
cache = cache->next;
}
return false;
}
static int target_flush_all_regs_default(struct target *target, bool invalidate)
{
struct reg_cache *cache = target->reg_cache;
/* In targets where per-register flushing isn't implemented yet, this function
* would result in many errors (one for each dirty register). The check below is
* to prevent that, and just a single clear error message will be printed. */
if (!target_implements_reg_flush(cache)) {
LOG_TARGET_ERROR(target, "Unable to flush register cache - operation not yet supported "
"by %s implementation in OpenOCD", target->type->name);
return ERROR_FAIL;
}
unsigned int total_count = 0;
unsigned int dirty_count = 0;
unsigned int failed_count = 0;
while (cache) {
for (unsigned int i = 0; i < cache->num_regs; i++) {
total_count++;
struct reg *reg = &cache->reg_list[i];
if (!reg->exist || !reg->valid)
continue;
if (reg->dirty) {
dirty_count++;
if (!reg->type->flush) {
LOG_TARGET_WARNING(target, "Flushing register '%s' is not implemented", reg->name);
failed_count++;
continue;
}
LOG_TARGET_DEBUG(target, "Flushing register %s", reg->name);
int result = reg->type->flush(reg);
if (result != ERROR_OK) {
LOG_TARGET_ERROR(target, "Failed to flush register '%s'", reg->name);
failed_count++;
continue;
}
if (reg->dirty) {
LOG_TARGET_ERROR(target, "BUG: Register '%s' remains dirty after flushing", reg->name);
return ERROR_FAIL;
}
}
if (invalidate) {
LOG_TARGET_DEBUG(target, "Invalidating register '%s'", reg->name);
reg->valid = false;
}
}
cache = cache->next;
}
if (dirty_count == 0) {
LOG_TARGET_INFO(target, "No registers dirty, nothing to flush");
return ERROR_OK;
}
LOG_TARGET_DEBUG(target, "Found %u dirty registers out of %u registers", dirty_count, total_count);
if (failed_count > 0) {
LOG_TARGET_ERROR(target, "Failed to flush %u out of %u dirty registers",
failed_count, dirty_count);
return ERROR_FAIL;
}
LOG_TARGET_DEBUG(target, "All dirty registers flushed");
return ERROR_OK;
}
static int target_profiling(struct target *target, uint32_t *samples, static int target_profiling(struct target *target, uint32_t *samples,
uint32_t max_num_samples, uint32_t *num_samples, uint32_t seconds) uint32_t max_num_samples, uint32_t *num_samples, uint32_t seconds)
{ {
@ -1536,6 +1616,9 @@ static int target_init_one(struct command_context *cmd_ctx,
if (!target->type->profiling) if (!target->type->profiling)
target->type->profiling = target_profiling_default; target->type->profiling = target_profiling_default;
if (!target->type->flush_all_regs)
target->type->flush_all_regs = target_flush_all_regs_default;
return ERROR_OK; return ERROR_OK;
} }
@ -3106,16 +3189,24 @@ COMMAND_HANDLER(handle_reg_command)
/* display a register */ /* display a register */
if ((CMD_ARGC == 1) || ((CMD_ARGC == 2) && !((CMD_ARGV[1][0] >= '0') if ((CMD_ARGC == 1) || ((CMD_ARGC == 2) && !((CMD_ARGV[1][0] >= '0')
&& (CMD_ARGV[1][0] <= '9')))) { && (CMD_ARGV[1][0] <= '9')))) {
if ((CMD_ARGC == 2) && (strcmp(CMD_ARGV[1], "force") == 0)) if (CMD_ARGC == 2
reg->valid = false; && strcmp(CMD_ARGV[1], "force") == 0
&& reg->valid) {
int retval = register_flush(target, reg, true);
if (retval != ERROR_OK) {
LOG_TARGET_ERROR(target, "Failed to flush register '%s' before force-reading", reg->name);
return retval;
}
}
if (!reg->valid) { if (!reg->valid) {
int retval = reg->type->get(reg); int retval = reg->type->get(reg);
if (retval != ERROR_OK) { if (retval != ERROR_OK) {
LOG_ERROR("Could not read register '%s'", reg->name); LOG_TARGET_ERROR(target, "Could not read register '%s'", reg->name);
return retval; return retval;
} }
} }
char *value = buf_to_hex_str(reg->value, reg->size); char *value = buf_to_hex_str(reg->value, reg->size);
command_print(CMD, "%s (/%i): 0x%s", reg->name, (int)(reg->size), value); command_print(CMD, "%s (/%i): 0x%s", reg->name, (int)(reg->size), value);
free(value); free(value);
@ -3123,7 +3214,14 @@ COMMAND_HANDLER(handle_reg_command)
} }
/* set register value */ /* set register value */
if (CMD_ARGC == 2) { if (CMD_ARGC >= 2 && CMD_ARGC <= 3) {
bool flush = false;
if (CMD_ARGC >= 3 && !strcmp(CMD_ARGV[2], "force"))
flush = true;
else if (CMD_ARGC >= 3)
return ERROR_COMMAND_SYNTAX_ERROR;
uint8_t *buf = malloc(DIV_ROUND_UP(reg->size, 8)); uint8_t *buf = malloc(DIV_ROUND_UP(reg->size, 8));
if (!buf) { if (!buf) {
LOG_ERROR("Failed to allocate memory"); LOG_ERROR("Failed to allocate memory");
@ -3137,17 +3235,21 @@ COMMAND_HANDLER(handle_reg_command)
} }
retval = reg->type->set(reg, buf); retval = reg->type->set(reg, buf);
free(buf);
if (retval != ERROR_OK) { if (retval != ERROR_OK) {
LOG_ERROR("Could not write to register '%s'", reg->name); LOG_ERROR("Could not write to register '%s'", reg->name);
return retval;
} else { } else {
char *value = buf_to_hex_str(reg->value, reg->size); char *value = buf_to_hex_str(reg->value, reg->size);
command_print(CMD, "%s (/%i): 0x%s", reg->name, (int)(reg->size), value); command_print(CMD, "%s (/%i): 0x%s", reg->name, (int)(reg->size), value);
free(value); free(value);
} }
free(buf); if (reg->dirty && flush)
return register_flush(target, reg, false);
return retval; return ERROR_OK;
} }
return ERROR_COMMAND_SYNTAX_ERROR; return ERROR_COMMAND_SYNTAX_ERROR;
@ -4742,6 +4844,14 @@ static int target_jim_get_reg(Jim_Interp *interp, int argc,
return JIM_ERR; return JIM_ERR;
} }
if (force && reg->valid && reg->dirty) {
int retval = register_flush(target, reg, false);
if (retval != ERROR_OK) {
Jim_SetResultFormatted(interp, "failed to flush dirty register '%s' before force-reading", reg->name);
return retval;
}
}
if (force || !reg->valid) { if (force || !reg->valid) {
int retval = reg->type->get(reg); int retval = reg->type->get(reg);
@ -4781,18 +4891,30 @@ static int target_jim_get_reg(Jim_Interp *interp, int argc,
COMMAND_HANDLER(handle_set_reg_command) COMMAND_HANDLER(handle_set_reg_command)
{ {
if (CMD_ARGC != 1) if (CMD_ARGC < 1 || CMD_ARGC > 2)
return ERROR_COMMAND_SYNTAX_ERROR; return ERROR_COMMAND_SYNTAX_ERROR;
bool flush = false;
/* Check for the "-force" argument */
if (CMD_ARGC == 2) {
if (!strcmp(CMD_ARGV[0], "-force")) {
flush = true;
} else {
command_print(CMD, "unknown argument '%s'", CMD_ARGV[0]);
return ERROR_COMMAND_ARGUMENT_INVALID;
}
}
int tmp; int tmp;
#if JIM_VERSION >= 80 #if JIM_VERSION >= 80
Jim_Obj **dict = Jim_DictPairs(CMD_CTX->interp, CMD_JIMTCL_ARGV[0], &tmp); Jim_Obj **dict = Jim_DictPairs(CMD_CTX->interp, CMD_JIMTCL_ARGV[CMD_ARGC - 1], &tmp);
if (!dict) if (!dict)
return ERROR_FAIL; return ERROR_FAIL;
#else #else
Jim_Obj **dict; Jim_Obj **dict;
int ret = Jim_DictPairs(CMD_CTX->interp, CMD_JIMTCL_ARGV[0], &dict, &tmp); int ret = Jim_DictPairs(CMD_CTX->interp, CMD_JIMTCL_ARGV[CMD_ARGC - 1], &dict, &tmp);
if (ret != JIM_OK) if (ret != JIM_OK)
return ERROR_FAIL; return ERROR_FAIL;
@ -4833,6 +4955,14 @@ COMMAND_HANDLER(handle_set_reg_command)
reg_value, reg_name); reg_value, reg_name);
return retval; return retval;
} }
if (flush && reg->valid && reg->dirty) {
retval = register_flush(target, reg, false);
if (retval != ERROR_OK) {
Jim_SetResultFormatted(CMD_CTX->interp, "Failed to flush register '%s'", reg->name);
return retval;
}
}
} }
return ERROR_OK; return ERROR_OK;
@ -5570,14 +5700,14 @@ static const struct command_registration target_instance_command_handlers[] = {
.mode = COMMAND_EXEC, .mode = COMMAND_EXEC,
.jim_handler = target_jim_get_reg, .jim_handler = target_jim_get_reg,
.help = "Get register values from the target", .help = "Get register values from the target",
.usage = "list", .usage = "['-force'] list",
}, },
{ {
.name = "set_reg", .name = "set_reg",
.mode = COMMAND_EXEC, .mode = COMMAND_EXEC,
.handler = handle_set_reg_command, .handler = handle_set_reg_command,
.help = "Set target register values", .help = "Set target register values",
.usage = "dict", .usage = "['-force'] dict",
}, },
{ {
.name = "read_memory", .name = "read_memory",
@ -6495,6 +6625,24 @@ nextw:
return retval; return retval;
} }
COMMAND_HANDLER(handle_flush_reg_cache_command)
{
struct target *target = get_current_target(CMD_CTX);
if (CMD_ARGC > 1)
return ERROR_COMMAND_SYNTAX_ERROR;
bool invalidate = false;
if (CMD_ARGC == 1) {
if (strcmp(CMD_ARGV[0], "-invalidate"))
return ERROR_COMMAND_SYNTAX_ERROR;
invalidate = true;
}
return target->type->flush_all_regs(target, invalidate);
}
static const struct command_registration target_exec_command_handlers[] = { static const struct command_registration target_exec_command_handlers[] = {
{ {
.name = "fast_load_image", .name = "fast_load_image",
@ -6532,9 +6680,9 @@ static const struct command_registration target_exec_command_handlers[] = {
.name = "reg", .name = "reg",
.handler = handle_reg_command, .handler = handle_reg_command,
.mode = COMMAND_EXEC, .mode = COMMAND_EXEC,
.help = "display (reread from target with \"force\") or set a register; " .help = "display (reread from target with \"force\") or set a register "
"with no arguments, displays all registers and their values", "(write to the target immediately with \"force\"); ",
.usage = "[(register_number|register_name) [(value|'force')]]", .usage = "[(register_number|register_name) [value] ['force']]",
}, },
{ {
.name = "poll", .name = "poll",
@ -6708,14 +6856,21 @@ static const struct command_registration target_exec_command_handlers[] = {
.mode = COMMAND_EXEC, .mode = COMMAND_EXEC,
.jim_handler = target_jim_get_reg, .jim_handler = target_jim_get_reg,
.help = "Get register values from the target", .help = "Get register values from the target",
.usage = "list", .usage = "['-force'] list",
}, },
{ {
.name = "set_reg", .name = "set_reg",
.mode = COMMAND_EXEC, .mode = COMMAND_EXEC,
.handler = handle_set_reg_command, .handler = handle_set_reg_command,
.help = "Set target register values", .help = "Set target register values",
.usage = "dict", .usage = "['-force'] dict",
},
{
.name = "flush_reg_cache",
.handler = handle_flush_reg_cache_command,
.mode = COMMAND_EXEC,
.help = "Flush register cache",
.usage = "['-invalidate']",
}, },
{ {
.name = "read_memory", .name = "read_memory",
@ -6760,7 +6915,6 @@ static const struct command_registration target_exec_command_handlers[] = {
.help = "Test the target's memory access functions", .help = "Test the target's memory access functions",
.usage = "size", .usage = "size",
}, },
COMMAND_REGISTRATION_DONE COMMAND_REGISTRATION_DONE
}; };
static int target_register_user_commands(struct command_context *cmd_ctx) static int target_register_user_commands(struct command_context *cmd_ctx)

View File

@ -309,6 +309,10 @@ struct target_type {
* will typically be 32 for 32-bit targets, and 64 for 64-bit targets. If * will typically be 32 for 32-bit targets, and 64 for 64-bit targets. If
* not implemented, it's assumed to be 32. */ * not implemented, it's assumed to be 32. */
unsigned int (*data_bits)(struct target *target); unsigned int (*data_bits)(struct target *target);
/* Flush the whole register cache.
* Also invalidate flushed register values if invalidate == true */
int (*flush_all_regs)(struct target *target, bool invalidate);
}; };
extern struct target_type aarch64_target; extern struct target_type aarch64_target;

View File

@ -2854,6 +2854,7 @@ static int xscale_analyze_trace(struct target *target, struct command_invocation
static const struct reg_arch_type xscale_reg_type = { static const struct reg_arch_type xscale_reg_type = {
.get = xscale_get_reg, .get = xscale_get_reg,
.set = xscale_set_reg, .set = xscale_set_reg,
.flush = NULL,
}; };
static void xscale_build_reg_cache(struct target *target) static void xscale_build_reg_cache(struct target *target)

View File

@ -490,6 +490,7 @@ static int xtensa_core_reg_set(struct reg *reg, uint8_t *buf)
static const struct reg_arch_type xtensa_reg_type = { static const struct reg_arch_type xtensa_reg_type = {
.get = xtensa_core_reg_get, .get = xtensa_core_reg_get,
.set = xtensa_core_reg_set, .set = xtensa_core_reg_set,
.flush = NULL,
}; };
/* Convert a register index that's indexed relative to windowbase, to the real address. */ /* Convert a register index that's indexed relative to windowbase, to the real address. */