arm_adi_v5: fix mem_ap_read_buf_u32() JTAG nastiness..
Moved JTAG code out of transport-neutral file (arm_adi_v5.c) into transport specific file (adi_v5_jtag.c). Added ap_block_read to dap_ops interface (arm_adi_v5.h) to support the move. Change-Id: I796d3984f138aad052b97c77ac9c12ffd1158f74 Signed-off-by: mike brown <mike@theshedworks.org.uk> Reviewed-on: http://openocd.zylin.com/1277 Tested-by: jenkins Reviewed-by: Michel JAOUEN <michel.jaouen@stericsson.com> Reviewed-by: Spencer Oliver <spen@spen-soft.co.uk>
This commit is contained in:
parent
0875e64ddb
commit
2a8a89edcb
|
@ -75,10 +75,7 @@
|
||||||
* @param ack points to where the three bit JTAG_ACK_* code will be stored
|
* @param ack points to where the three bit JTAG_ACK_* code will be stored
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* FIXME don't export ... this is a temporary workaround for the
|
static int adi_jtag_dp_scan(struct adiv5_dap *dap,
|
||||||
* mem_ap_read_buf_u32() mess, until it's no longer JTAG-specific.
|
|
||||||
*/
|
|
||||||
int adi_jtag_dp_scan(struct adiv5_dap *dap,
|
|
||||||
uint8_t instr, uint8_t reg_addr, uint8_t RnW,
|
uint8_t instr, uint8_t reg_addr, uint8_t RnW,
|
||||||
uint8_t *outvalue, uint8_t *invalue, uint8_t *ack)
|
uint8_t *outvalue, uint8_t *invalue, uint8_t *ack)
|
||||||
{
|
{
|
||||||
|
@ -417,6 +414,40 @@ static int jtag_ap_q_write(struct adiv5_dap *dap, unsigned reg,
|
||||||
return adi_jtag_ap_write_check(dap, reg, out_value_buf);
|
return adi_jtag_ap_write_check(dap, reg, out_value_buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int jtag_ap_q_read_block(struct adiv5_dap *dap, unsigned reg,
|
||||||
|
uint32_t blocksize, uint8_t *buffer)
|
||||||
|
{
|
||||||
|
uint32_t readcount;
|
||||||
|
int retval = ERROR_OK;
|
||||||
|
|
||||||
|
/* Scan out first read */
|
||||||
|
retval = adi_jtag_dp_scan(dap, JTAG_DP_APACC, reg,
|
||||||
|
DPAP_READ, 0, NULL, NULL);
|
||||||
|
if (retval != ERROR_OK)
|
||||||
|
return retval;
|
||||||
|
|
||||||
|
for (readcount = 0; readcount < blocksize - 1; readcount++) {
|
||||||
|
/* Scan out next read; scan in posted value for the
|
||||||
|
* previous one. Assumes read is acked "OK/FAULT",
|
||||||
|
* and CTRL_STAT says that meant "OK".
|
||||||
|
*/
|
||||||
|
retval = adi_jtag_dp_scan(dap, JTAG_DP_APACC, reg,
|
||||||
|
DPAP_READ, 0, buffer + 4 * readcount,
|
||||||
|
&dap->ack);
|
||||||
|
if (retval != ERROR_OK)
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scan in last posted value; RDBUFF has no other effect,
|
||||||
|
* assuming ack is OK/FAULT and CTRL_STAT says "OK".
|
||||||
|
*/
|
||||||
|
retval = adi_jtag_dp_scan(dap, JTAG_DP_DPACC, DP_RDBUFF,
|
||||||
|
DPAP_READ, 0, buffer + 4 * readcount,
|
||||||
|
&dap->ack);
|
||||||
|
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
static int jtag_ap_q_abort(struct adiv5_dap *dap, uint8_t *ack)
|
static int jtag_ap_q_abort(struct adiv5_dap *dap, uint8_t *ack)
|
||||||
{
|
{
|
||||||
/* for JTAG, this is the only valid ABORT register operation */
|
/* for JTAG, this is the only valid ABORT register operation */
|
||||||
|
@ -433,13 +464,14 @@ static int jtag_dp_run(struct adiv5_dap *dap)
|
||||||
* part of DAP setup
|
* part of DAP setup
|
||||||
*/
|
*/
|
||||||
const struct dap_ops jtag_dp_ops = {
|
const struct dap_ops jtag_dp_ops = {
|
||||||
.queue_idcode_read = jtag_idcode_q_read,
|
.queue_idcode_read = jtag_idcode_q_read,
|
||||||
.queue_dp_read = jtag_dp_q_read,
|
.queue_dp_read = jtag_dp_q_read,
|
||||||
.queue_dp_write = jtag_dp_q_write,
|
.queue_dp_write = jtag_dp_q_write,
|
||||||
.queue_ap_read = jtag_ap_q_read,
|
.queue_ap_read = jtag_ap_q_read,
|
||||||
.queue_ap_write = jtag_ap_q_write,
|
.queue_ap_write = jtag_ap_q_write,
|
||||||
.queue_ap_abort = jtag_ap_q_abort,
|
.queue_ap_read_block = jtag_ap_q_read_block,
|
||||||
.run = jtag_dp_run,
|
.queue_ap_abort = jtag_ap_q_abort,
|
||||||
|
.run = jtag_dp_run,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -539,13 +539,6 @@ int mem_ap_write_buf_u8(struct adiv5_dap *dap, const uint8_t *buffer, int count,
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* FIXME don't import ... this is a temporary workaround for the
|
|
||||||
* mem_ap_read_buf_u32() mess, until it's no longer JTAG-specific.
|
|
||||||
*/
|
|
||||||
extern int adi_jtag_dp_scan(struct adiv5_dap *dap,
|
|
||||||
uint8_t instr, uint8_t reg_addr, uint8_t RnW,
|
|
||||||
uint8_t *outvalue, uint8_t *invalue, uint8_t *ack);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Synchronously read a block of 32-bit words into a buffer
|
* Synchronously read a block of 32-bit words into a buffer
|
||||||
* @param dap The DAP connected to the MEM-AP.
|
* @param dap The DAP connected to the MEM-AP.
|
||||||
|
@ -588,38 +581,7 @@ int mem_ap_read_buf_u32(struct adiv5_dap *dap, uint8_t *buffer,
|
||||||
if (retval != ERROR_OK)
|
if (retval != ERROR_OK)
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
/* FIXME remove these three calls to adi_jtag_dp_scan(),
|
retval = dap_queue_ap_read_block(dap, AP_REG_DRW, blocksize, buffer);
|
||||||
* so this routine becomes transport-neutral. Be careful
|
|
||||||
* not to cause performance problems with JTAG; would it
|
|
||||||
* suffice to loop over dap_queue_ap_read(), or would that
|
|
||||||
* be slower when JTAG is the chosen transport?
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* Scan out first read */
|
|
||||||
retval = adi_jtag_dp_scan(dap, JTAG_DP_APACC, AP_REG_DRW,
|
|
||||||
DPAP_READ, 0, NULL, NULL);
|
|
||||||
if (retval != ERROR_OK)
|
|
||||||
return retval;
|
|
||||||
for (readcount = 0; readcount < blocksize - 1; readcount++) {
|
|
||||||
/* Scan out next read; scan in posted value for the
|
|
||||||
* previous one. Assumes read is acked "OK/FAULT",
|
|
||||||
* and CTRL_STAT says that meant "OK".
|
|
||||||
*/
|
|
||||||
retval = adi_jtag_dp_scan(dap, JTAG_DP_APACC, AP_REG_DRW,
|
|
||||||
DPAP_READ, 0, buffer + 4 * readcount,
|
|
||||||
&dap->ack);
|
|
||||||
if (retval != ERROR_OK)
|
|
||||||
return retval;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Scan in last posted value; RDBUFF has no other effect,
|
|
||||||
* assuming ack is OK/FAULT and CTRL_STAT says "OK".
|
|
||||||
*/
|
|
||||||
retval = adi_jtag_dp_scan(dap, JTAG_DP_DPACC, DP_RDBUFF,
|
|
||||||
DPAP_READ, 0, buffer + 4 * readcount,
|
|
||||||
&dap->ack);
|
|
||||||
if (retval != ERROR_OK)
|
|
||||||
return retval;
|
|
||||||
|
|
||||||
retval = dap_run(dap);
|
retval = dap_run(dap);
|
||||||
if (retval != ERROR_OK) {
|
if (retval != ERROR_OK) {
|
||||||
|
|
|
@ -216,6 +216,9 @@ struct dap_ops {
|
||||||
/** AP register write. */
|
/** AP register write. */
|
||||||
int (*queue_ap_write)(struct adiv5_dap *dap, unsigned reg,
|
int (*queue_ap_write)(struct adiv5_dap *dap, unsigned reg,
|
||||||
uint32_t data);
|
uint32_t data);
|
||||||
|
/** AP read block. */
|
||||||
|
int (*queue_ap_read_block)(struct adiv5_dap *dap, unsigned reg,
|
||||||
|
uint32_t blocksize, uint8_t *buffer);
|
||||||
|
|
||||||
/** AP operation abort. */
|
/** AP operation abort. */
|
||||||
int (*queue_ap_abort)(struct adiv5_dap *dap, uint8_t *ack);
|
int (*queue_ap_abort)(struct adiv5_dap *dap, uint8_t *ack);
|
||||||
|
@ -321,6 +324,24 @@ static inline int dap_queue_ap_write(struct adiv5_dap *dap,
|
||||||
return dap->ops->queue_ap_write(dap, reg, data);
|
return dap->ops->queue_ap_write(dap, reg, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Queue an AP block read.
|
||||||
|
*
|
||||||
|
* @param dap The DAP used for reading.
|
||||||
|
* @param reg The number of the AP register being read.
|
||||||
|
* @param blocksize The number of the AP register being read.
|
||||||
|
* @param buffer Pointer saying where to store the data
|
||||||
|
* (in host endianness).
|
||||||
|
*
|
||||||
|
* @return ERROR_OK for success, else a fault code.
|
||||||
|
*/
|
||||||
|
static inline int dap_queue_ap_read_block(struct adiv5_dap *dap,
|
||||||
|
unsigned reg, unsigned blocksize, uint8_t *buffer)
|
||||||
|
{
|
||||||
|
assert(dap->ops != NULL);
|
||||||
|
return dap->ops->queue_ap_read_block(dap, reg, blocksize, buffer);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queue an AP abort operation. The current AP transaction is aborted,
|
* Queue an AP abort operation. The current AP transaction is aborted,
|
||||||
* including any update of the transaction counter. The AP is left in
|
* including any update of the transaction counter. The AP is left in
|
||||||
|
|
Loading…
Reference in New Issue