Detect and smoketest data and ibuf registers.

Change-Id: I7ee4817ec63041a1577b83392d40b676fb67c207
This commit is contained in:
Tim Newsome 2017-02-08 20:40:37 -08:00
parent 8cac7d0cee
commit 2ad366e658
1 changed files with 69 additions and 28 deletions

View File

@ -171,7 +171,12 @@ struct memory_cache_line {
typedef struct { typedef struct {
/* Number of address bits in the dbus register. */ /* Number of address bits in the dbus register. */
uint8_t addrbits; unsigned abits;
/* Number of abstract command data registers. */
unsigned datacount;
/* Number of words in the Program Buffer. */
unsigned progsize;
/* Number of Program Buffer registers. */
/* Number of words in Debug RAM. */ /* Number of words in Debug RAM. */
unsigned int dramsize; // TODO: remove unsigned int dramsize; // TODO: remove
uint64_t dcsr; uint64_t dcsr;
@ -365,13 +370,13 @@ static void add_dbus_scan(const struct target *target, struct scan_field *field,
{ {
riscv013_info_t *info = get_info(target); riscv013_info_t *info = get_info(target);
field->num_bits = info->addrbits + DBUS_OP_SIZE + DBUS_DATA_SIZE; field->num_bits = info->abits + DBUS_OP_SIZE + DBUS_DATA_SIZE;
field->in_value = in_value; field->in_value = in_value;
field->out_value = out_value; field->out_value = out_value;
buf_set_u64(out_value, DBUS_OP_START, DBUS_OP_SIZE, op); buf_set_u64(out_value, DBUS_OP_START, DBUS_OP_SIZE, op);
buf_set_u64(out_value, DBUS_DATA_START, DBUS_DATA_SIZE, data); buf_set_u64(out_value, DBUS_DATA_START, DBUS_DATA_SIZE, data);
buf_set_u64(out_value, DBUS_ADDRESS_START, info->addrbits, address); buf_set_u64(out_value, DBUS_ADDRESS_START, info->abits, address);
jtag_add_dr_scan(target->tap, 1, field, TAP_IDLE); jtag_add_dr_scan(target->tap, 1, field, TAP_IDLE);
@ -394,26 +399,20 @@ static void dump_field(const struct scan_field *field)
return; return;
uint64_t out = buf_get_u64(field->out_value, 0, field->num_bits); uint64_t out = buf_get_u64(field->out_value, 0, field->num_bits);
unsigned int out_op = (out >> DBUS_OP_START) & ((1 << DBUS_OP_SIZE) - 1); unsigned int out_op = get_field(out, DTM_DBUS_OP);
char out_interrupt = ((out >> DBUS_DATA_START) & DMCONTROL_INTERRUPT) ? 'i' : '.'; unsigned int out_data = get_field(out, DTM_DBUS_DATA);
char out_haltnot = ((out >> DBUS_DATA_START) & DMCONTROL_HALTNOT) ? 'h' : '.'; unsigned int out_address = out >> DTM_DBUS_ADDRESS_OFFSET;
unsigned int out_data = out >> 2;
unsigned int out_address = out >> DBUS_ADDRESS_START;
uint64_t in = buf_get_u64(field->in_value, 0, field->num_bits); uint64_t in = buf_get_u64(field->in_value, 0, field->num_bits);
unsigned int in_op = (in >> DBUS_OP_START) & ((1 << DBUS_OP_SIZE) - 1); unsigned int in_op = get_field(in, DTM_DBUS_OP);
char in_interrupt = ((in >> DBUS_DATA_START) & DMCONTROL_INTERRUPT) ? 'i' : '.'; unsigned int in_data = get_field(in, DTM_DBUS_DATA);
char in_haltnot = ((in >> DBUS_DATA_START) & DMCONTROL_HALTNOT) ? 'h' : '.'; unsigned int in_address = in >> DTM_DBUS_ADDRESS_OFFSET;
unsigned int in_data = in >> 2;
unsigned int in_address = in >> DBUS_ADDRESS_START;
log_printf_lf(LOG_LVL_DEBUG, log_printf_lf(LOG_LVL_DEBUG,
__FILE__, __LINE__, "scan", __FILE__, __LINE__, "scan",
"%db %s %c%c:%08x @%02x -> %s %c%c:%08x @%02x", "%db %s %08x @%02x -> %s %08x @%02x",
field->num_bits, field->num_bits,
op_string[out_op], out_interrupt, out_haltnot, out_data, op_string[out_op], out_data, out_address,
out_address, status_string[in_op], in_data, in_address);
status_string[in_op], in_interrupt, in_haltnot, in_data,
in_address);
} }
static dbus_status_t dbus_scan(struct target *target, uint16_t *address_in, static dbus_status_t dbus_scan(struct target *target, uint16_t *address_in,
@ -423,16 +422,16 @@ static dbus_status_t dbus_scan(struct target *target, uint16_t *address_in,
uint8_t in[8] = {0}; uint8_t in[8] = {0};
uint8_t out[8]; uint8_t out[8];
struct scan_field field = { struct scan_field field = {
.num_bits = info->addrbits + DBUS_OP_SIZE + DBUS_DATA_SIZE, .num_bits = info->abits + DBUS_OP_SIZE + DBUS_DATA_SIZE,
.out_value = out, .out_value = out,
.in_value = in .in_value = in
}; };
assert(info->addrbits != 0); assert(info->abits != 0);
buf_set_u64(out, DBUS_OP_START, DBUS_OP_SIZE, op); buf_set_u64(out, DBUS_OP_START, DBUS_OP_SIZE, op);
buf_set_u64(out, DBUS_DATA_START, DBUS_DATA_SIZE, data_out); buf_set_u64(out, DBUS_DATA_START, DBUS_DATA_SIZE, data_out);
buf_set_u64(out, DBUS_ADDRESS_START, info->addrbits, address_out); buf_set_u64(out, DBUS_ADDRESS_START, info->abits, address_out);
/* Assume dbus is already selected. */ /* Assume dbus is already selected. */
jtag_add_dr_scan(target->tap, 1, &field, TAP_IDLE); jtag_add_dr_scan(target->tap, 1, &field, TAP_IDLE);
@ -454,7 +453,7 @@ static dbus_status_t dbus_scan(struct target *target, uint16_t *address_in,
} }
if (address_in) { if (address_in) {
*address_in = buf_get_u32(in, DBUS_ADDRESS_START, info->addrbits); *address_in = buf_get_u32(in, DBUS_ADDRESS_START, info->abits);
} }
dump_field(&field); dump_field(&field);
@ -478,7 +477,8 @@ static uint64_t dbus_read(struct target *target, uint16_t address)
i++ < 256); i++ < 256);
if (status != DBUS_STATUS_SUCCESS) { if (status != DBUS_STATUS_SUCCESS) {
LOG_ERROR("failed read from 0x%x; value=0x%" PRIx64 ", status=%d\n", address, value, status); LOG_ERROR("failed read from 0x%x; value=0x%" PRIx64 ", status=%d\n",
address, value, status);
} }
return value; return value;
@ -681,7 +681,7 @@ static bits_t read_bits(struct target *target)
do { do {
status = dbus_scan(target, &address_in, &value, DBUS_OP_READ, 0, 0); status = dbus_scan(target, &address_in, &value, DBUS_OP_READ, 0, 0);
if (status == DBUS_STATUS_BUSY) { if (status == DBUS_STATUS_BUSY) {
if (address_in == (1<<info->addrbits) - 1 && if (address_in == (1<<info->abits) - 1 &&
value == (1ULL<<DBUS_DATA_SIZE) - 1) { value == (1ULL<<DBUS_DATA_SIZE) - 1) {
LOG_ERROR("TDO seems to be stuck high."); LOG_ERROR("TDO seems to be stuck high.");
return err_result; return err_result;
@ -945,7 +945,7 @@ static int cache_write(struct target *target, unsigned int address, bool run)
} else { } else {
// We read a useful value in that last scan. // We read a useful value in that last scan.
unsigned int read_addr = scans_get_u32(scans, scans->next_scan-1, unsigned int read_addr = scans_get_u32(scans, scans->next_scan-1,
DBUS_ADDRESS_START, info->addrbits); DBUS_ADDRESS_START, info->abits);
if (read_addr != address) { if (read_addr != address) {
LOG_INFO("Got data from 0x%x but expected it from 0x%x", LOG_INFO("Got data from 0x%x but expected it from 0x%x",
read_addr, address); read_addr, address);
@ -1792,7 +1792,7 @@ static int examine(struct target *target)
} }
riscv013_info_t *info = get_info(target); riscv013_info_t *info = get_info(target);
info->addrbits = get_field(dtmcontrol, DTM_DTMCONTROL_ABITS); info->abits = get_field(dtmcontrol, DTM_DTMCONTROL_ABITS);
info->dtmcontrol_idle = get_field(dtmcontrol, DTM_DTMCONTROL_IDLE); info->dtmcontrol_idle = get_field(dtmcontrol, DTM_DTMCONTROL_IDLE);
if (info->dtmcontrol_idle == 0) { if (info->dtmcontrol_idle == 0) {
// Some old SiFive cores don't set idle but need it to be 1. // Some old SiFive cores don't set idle but need it to be 1.
@ -1807,7 +1807,6 @@ static int examine(struct target *target)
LOG_DEBUG(" reset=%d", get_field(dmcontrol, DMI_DMCONTROL_RESET)); LOG_DEBUG(" reset=%d", get_field(dmcontrol, DMI_DMCONTROL_RESET));
LOG_DEBUG(" dmactive=%d", get_field(dmcontrol, DMI_DMCONTROL_DMACTIVE)); LOG_DEBUG(" dmactive=%d", get_field(dmcontrol, DMI_DMCONTROL_DMACTIVE));
LOG_DEBUG(" hartid=0x%x", get_field(dmcontrol, DMI_DMCONTROL_HARTID)); LOG_DEBUG(" hartid=0x%x", get_field(dmcontrol, DMI_DMCONTROL_HARTID));
LOG_DEBUG(" haltsum=%d", get_field(dmcontrol, DMI_DMCONTROL_HALTSUM));
LOG_DEBUG(" authenticated=%d", get_field(dmcontrol, DMI_DMCONTROL_AUTHENTICATED)); LOG_DEBUG(" authenticated=%d", get_field(dmcontrol, DMI_DMCONTROL_AUTHENTICATED));
LOG_DEBUG(" authbusy=%d", get_field(dmcontrol, DMI_DMCONTROL_AUTHBUSY)); LOG_DEBUG(" authbusy=%d", get_field(dmcontrol, DMI_DMCONTROL_AUTHBUSY));
LOG_DEBUG(" authtype=%d", get_field(dmcontrol, DMI_DMCONTROL_AUTHTYPE)); LOG_DEBUG(" authtype=%d", get_field(dmcontrol, DMI_DMCONTROL_AUTHTYPE));
@ -1825,6 +1824,48 @@ static int examine(struct target *target)
return ERROR_FAIL; return ERROR_FAIL;
} }
// Check that abstract data registers are accessible.
uint32_t abstractcs = dbus_read(target, DMI_ABSTRACTCS);
info->datacount = get_field(abstractcs, DMI_ABSTRACTCS_DATACOUNT);
LOG_DEBUG("abstractcs=0x%x", abstractcs);
LOG_DEBUG(" datacount=%d", info->datacount);
uint32_t accesscs = dbus_read(target, DMI_ACCESSCS);
info->progsize = get_field(abstractcs, DMI_ACCESSCS_PROGSIZE);
LOG_DEBUG("accesscs=0x%x", accesscs);
LOG_DEBUG(" progsize=%d", info->progsize);
uint32_t value = 0x53467665;
for (unsigned i = 0; i < info->datacount; i++) {
dbus_write(target, DMI_DATA0 + i, value);
value += 0x52534335;
}
for (unsigned i = 0; i < info->progsize; i++) {
dbus_write(target, DMI_IBUF0 + i, value);
value += 0x52534335;
}
value = 0x53467665;
for (unsigned i = 0; i < info->datacount; i++) {
uint32_t check = dbus_read(target, DMI_DATA0 + i);
if (check != value) {
LOG_ERROR("Wrote 0x%x to dbus address 0x%x but got back 0x%x",
value, DMI_DATA0 + i, check);
return ERROR_FAIL;
}
value += 0x52534335;
}
for (unsigned i = 0; i < info->progsize; i++) {
uint32_t check = dbus_read(target, DMI_IBUF0 + i);
if (check != value) {
LOG_ERROR("Wrote 0x%x to dbus address 0x%x but got back 0x%x",
value, DMI_IBUF0 + i, check);
return ERROR_FAIL;
}
value += 0x52534335;
}
// Figure out XLEN, and test writing all of Debug RAM while we're at it. // Figure out XLEN, and test writing all of Debug RAM while we're at it.
cache_set32(target, 0, xori(S1, ZERO, -1)); cache_set32(target, 0, xori(S1, ZERO, -1));
// 0xffffffff 0xffffffff:ffffffff 0xffffffff:ffffffff:ffffffff:ffffffff // 0xffffffff 0xffffffff:ffffffff 0xffffffff:ffffffff:ffffffff:ffffffff
@ -1949,7 +1990,7 @@ static riscv_error_t handle_halt_routine(struct target *target)
DBUS_OP_SIZE); DBUS_OP_SIZE);
uint64_t data = scans_get_u64(scans, i, DBUS_DATA_START, DBUS_DATA_SIZE); uint64_t data = scans_get_u64(scans, i, DBUS_DATA_START, DBUS_DATA_SIZE);
uint32_t address = scans_get_u32(scans, i, DBUS_ADDRESS_START, uint32_t address = scans_get_u32(scans, i, DBUS_ADDRESS_START,
info->addrbits); info->abits);
switch (status) { switch (status) {
case DBUS_STATUS_SUCCESS: case DBUS_STATUS_SUCCESS:
break; break;