forked from auracaster/openocd
8dea2908b7
This is a major rewrite of the RISC-V v0.13 OpenOCD port. This shouldn't have any meaningful effect on the v0.11 support, but it does add generic versions of many functions that will allow me to later refactor the v0.11 support so it's easier to maintain both ports. This started as an emergency feature branch and went on for a long time, so it's all been squashed down into one commit so there isn't a big set of broken commits lying around. The changes are: * You can pass "-rtos riscv" to the target in OpenOCD's configuration file, which enables multi-hart mode. This uses OpenOCD's RTOS support to control all the harts from the debug module using commands like "info threads" in GDB. This support is still expermental. * There is support for RV64I, but due to OpenOCD limitations we only support 32-bit physical addresses. I hope to remedy this by rebasing onto the latest OpenOCD release, which I've heard should fix this. * This matches the latest draft version of the RISC-V debug spec, as of April 26th. This version fixes a number of spec bugs and should be close to the final debug spec.
316 lines
8.7 KiB
C
316 lines
8.7 KiB
C
#ifdef HAVE_CONFIG_H
|
||
#include "config.h"
|
||
#endif
|
||
|
||
#include "riscv_debug.h"
|
||
#include "target/target.h"
|
||
#include "target/riscv/riscv.h"
|
||
#include "rtos.h"
|
||
#include "server/gdb_server.h"
|
||
|
||
static int riscv_update_threads(struct rtos *rtos);
|
||
static int riscv_gdb_thread_packet(struct connection *connection, const char *packet, int packet_size);
|
||
static int riscv_gdb_v_packet(struct connection *connection, const char *packet, int packet_size);
|
||
|
||
static int riscv_detect_rtos(struct target *target)
|
||
{
|
||
LOG_ERROR("riscv_detect_rtos() unimplemented");
|
||
return -1;
|
||
}
|
||
|
||
static int riscv_create_rtos(struct target *target)
|
||
{
|
||
LOG_DEBUG("RISC-V Debug 'RTOS' created: this doesn't mean you're running an RTOS, just that you have multi-hart support on RISC-V");
|
||
|
||
struct riscv_rtos *r = calloc(1, sizeof(*r));
|
||
target->rtos->rtos_specific_params = r;
|
||
#if 0
|
||
r->target_hartid = 0;
|
||
r->target_any_hart = true;
|
||
r->target_every_hart = true;
|
||
#endif
|
||
|
||
target->rtos->current_threadid = 1;
|
||
target->rtos->current_thread = 1;
|
||
riscv_update_threads(target->rtos);
|
||
|
||
target->rtos->gdb_thread_packet = riscv_gdb_thread_packet;
|
||
target->rtos->gdb_v_packet = riscv_gdb_v_packet;
|
||
|
||
return JIM_OK;
|
||
}
|
||
|
||
static int riscv_update_threads(struct rtos *rtos)
|
||
{
|
||
LOG_DEBUG("Updating the RISC-V Hart List");
|
||
|
||
/* Figures out how many harts there are on the system. */
|
||
int hart_count = riscv_count_harts(rtos->target);
|
||
if (rtos->thread_count != hart_count) {
|
||
rtos_free_threadlist(rtos);
|
||
rtos->thread_count = hart_count;
|
||
rtos->thread_details = calloc(rtos->thread_count, sizeof(*rtos->thread_details));
|
||
for (int i = 0; i < rtos->thread_count; ++i) {
|
||
LOG_DEBUG(" Setting up Hart %d", i);
|
||
rtos->thread_details[i].threadid = i + 1;
|
||
rtos->thread_details[i].exists = true;
|
||
if (asprintf(&rtos->thread_details[i].thread_name_str, "Hart %d", i) < 0)
|
||
LOG_ERROR("riscv_update_threads() failed asprintf");
|
||
if (asprintf(&rtos->thread_details[i].extra_info_str, "RV64") < 0)
|
||
LOG_ERROR("riscv_update_threads() failed asprintf");
|
||
}
|
||
}
|
||
return JIM_OK;
|
||
}
|
||
|
||
static int riscv_gdb_thread_packet(struct connection *connection, const char *packet, int packet_size)
|
||
{
|
||
struct target *target = get_target_from_connection(connection);
|
||
struct rtos *rtos = target->rtos;
|
||
struct riscv_rtos *r = (struct riscv_rtos *)(target->rtos->rtos_specific_params);
|
||
|
||
char *packet_stttrr = malloc(packet_size + 1);
|
||
memset(packet_stttrr, '\0', packet_size + 1);
|
||
memcpy(packet_stttrr, packet, packet_size);
|
||
LOG_DEBUG("handling packet '%s'", packet_stttrr);
|
||
|
||
switch (packet[0]) {
|
||
case 'q':
|
||
if (strncmp(packet, "qfThreadInfo", 12) == 0) {
|
||
riscv_update_threads(target->rtos);
|
||
r->qs_thread_info_offset = 1;
|
||
|
||
char m[16];
|
||
snprintf(m, 16, "m%08x", (int)rtos->thread_details[0].threadid);
|
||
gdb_put_packet(connection, m, strlen(m));
|
||
return ERROR_OK;
|
||
}
|
||
|
||
if (strncmp(packet, "qsThreadInfo", 12) == 0) {
|
||
if (r->qs_thread_info_offset >= rtos->thread_count) {
|
||
gdb_put_packet(connection, "l", 1);
|
||
return ERROR_OK;
|
||
}
|
||
|
||
int tid = r->qs_thread_info_offset++;
|
||
char m[16];
|
||
snprintf(m, 16, "m%08x", (int)rtos->thread_details[tid].threadid);
|
||
gdb_put_packet(connection, m, strlen(m));
|
||
return ERROR_OK;
|
||
}
|
||
|
||
if (strncmp(packet, "qAttached", 9) == 0) {
|
||
gdb_put_packet(connection, "1", 1);
|
||
return ERROR_OK;
|
||
}
|
||
|
||
if (strncmp(packet, "qThreadExtraInfo", 16) == 0) {
|
||
char tid_str[32];
|
||
memcpy(tid_str, packet + 17, packet_size - 17);
|
||
tid_str[packet_size - 17] = '\0';
|
||
char *end;
|
||
int tid = strtol(tid_str, &end, 16);
|
||
if (*end != '\0') {
|
||
LOG_ERROR("Got qThreadExtraInfo with non-numeric TID: '%s'", tid_str);
|
||
gdb_put_packet(connection, NULL, 0);
|
||
return ERROR_FAIL;
|
||
}
|
||
|
||
char m[16];
|
||
snprintf(m, 16, "hart %d", tid);
|
||
char h[33];
|
||
h[0] = '\0';
|
||
for (size_t i = 0; i < strlen(m); ++i) {
|
||
char byte[3];
|
||
snprintf(byte, 3, "%02x", m[i]);
|
||
strncat(h, byte, 32);
|
||
}
|
||
gdb_put_packet(connection, h, strlen(h));
|
||
return ERROR_OK;
|
||
}
|
||
|
||
return GDB_THREAD_PACKET_NOT_CONSUMED;
|
||
|
||
case 'Q':
|
||
return GDB_THREAD_PACKET_NOT_CONSUMED;
|
||
|
||
case 'H':
|
||
/* ‘H op thread-id’
|
||
*
|
||
* Set thread for subsequent operations (‘m’, ‘M’, ‘g’, ‘G’,
|
||
* et.al.). Depending on the operation to be performed, op
|
||
* should be ‘c’ for step and continue operations (note that
|
||
* this is deprecated, supporting the ‘vCont’ command is a
|
||
* better option), and ‘g’ for other operations. The thread
|
||
* designator thread-id has the format and interpretation
|
||
* described in thread-id syntax.
|
||
*
|
||
* Reply:
|
||
* ‘OK’ for success
|
||
* ‘E NN’ for an error
|
||
*/
|
||
{
|
||
char tid_str[32];
|
||
memcpy(tid_str, packet + 2, packet_size - 2);
|
||
tid_str[packet_size - 2] = '\0';
|
||
char *entptr;
|
||
int tid = strtol(tid_str, &entptr, 16);
|
||
if (*entptr != '\0') {
|
||
LOG_ERROR("Got H packet, but without integer: %s", tid_str);
|
||
return GDB_THREAD_PACKET_NOT_CONSUMED;
|
||
}
|
||
|
||
switch (tid) {
|
||
case 0:
|
||
case -1:
|
||
riscv_set_all_rtos_harts(target);
|
||
break;
|
||
default:
|
||
riscv_set_rtos_hartid(target, tid - 1);
|
||
break;
|
||
}
|
||
|
||
switch (packet[1]) {
|
||
case 'g':
|
||
case 'c':
|
||
gdb_put_packet(connection, "OK", 2);
|
||
return ERROR_OK;
|
||
default:
|
||
LOG_ERROR("Unknown H packet subtype %2x\n", packet[1]);
|
||
gdb_put_packet(connection, NULL, 0);
|
||
return ERROR_FAIL;
|
||
}
|
||
}
|
||
|
||
case 'T':
|
||
{
|
||
char tid_str[32];
|
||
memcpy(tid_str, packet + 1, packet_size - 1);
|
||
tid_str[packet_size - 1] = '\0';
|
||
char *end;
|
||
int tid = strtol(tid_str, &end, 16);
|
||
if (*end != '\0') {
|
||
LOG_ERROR("T packet with non-numeric tid %s", tid_str);
|
||
gdb_put_packet(connection, NULL, 0);
|
||
return ERROR_FAIL;
|
||
}
|
||
|
||
riscv_update_threads(target->rtos);
|
||
if (tid <= target->rtos->thread_count) {
|
||
gdb_put_packet(connection, "OK", 2);
|
||
return ERROR_OK;
|
||
} else {
|
||
gdb_put_packet(connection, "E00", 3);
|
||
return ERROR_OK;
|
||
}
|
||
}
|
||
|
||
case 'c':
|
||
case 's':
|
||
target->state = TARGET_HALTED;
|
||
return JIM_OK;
|
||
|
||
case 'R':
|
||
gdb_put_packet(connection, "E00", 3);
|
||
return JIM_OK;
|
||
|
||
default:
|
||
LOG_ERROR("Unknown packet of type 0x%2.2x", packet[0]);
|
||
gdb_put_packet(connection, NULL, 0);
|
||
return JIM_OK;
|
||
}
|
||
}
|
||
|
||
static int riscv_gdb_v_packet(struct connection *connection, const char *packet, int packet_size)
|
||
{
|
||
char *packet_stttrr = malloc(packet_size + 1);
|
||
memset(packet_stttrr, '\0', packet_size + 1);
|
||
memcpy(packet_stttrr, packet, packet_size);
|
||
LOG_DEBUG("handling packet '%s'", packet_stttrr);
|
||
|
||
struct target *target = get_target_from_connection(connection);
|
||
|
||
if (strcmp(packet_stttrr, "vCont?") == 0) {
|
||
static const char *message = "OK";
|
||
gdb_put_packet(connection, (char *)message, strlen(message));
|
||
return JIM_OK;
|
||
}
|
||
|
||
int threadid;
|
||
if (sscanf(packet_stttrr, "vCont;s:%d;c", &threadid) == 1) {
|
||
riscv_set_rtos_hartid(target, threadid - 1);
|
||
riscv_step_rtos_hart(target);
|
||
|
||
gdb_put_packet(connection, "S02", 3);
|
||
return JIM_OK;
|
||
}
|
||
|
||
if (strcmp(packet_stttrr, "vCont;c") == 0) {
|
||
target->state = TARGET_RUNNING;
|
||
gdb_set_frontend_state_running(connection);
|
||
target_call_event_callbacks(target, TARGET_EVENT_GDB_START);
|
||
target_call_event_callbacks(target, TARGET_EVENT_RESUME_START);
|
||
riscv_resume_all_harts(target);
|
||
target_call_event_callbacks(target, TARGET_EVENT_RESUME_END);
|
||
return JIM_OK;
|
||
}
|
||
|
||
if (strncmp(packet_stttrr, "vCont", 5) == 0)
|
||
LOG_ERROR("Got unknown vCont-type packet");
|
||
|
||
return GDB_THREAD_PACKET_NOT_CONSUMED;
|
||
}
|
||
|
||
static int riscv_get_thread_reg_list(struct rtos *rtos, int64_t thread_id, char **hex_reg_list)
|
||
{
|
||
LOG_DEBUG("Updating RISC-V regiser list for hart %d", (int)(thread_id - 1));
|
||
|
||
#if 0
|
||
LOG_ERROR(" Not actually updating");
|
||
*hex_reg_list = 0;
|
||
return JIM_OK;
|
||
#endif
|
||
|
||
size_t n_regs = 32;
|
||
size_t xlen = 64;
|
||
size_t reg_chars = xlen / 8 * 2;
|
||
|
||
ssize_t hex_reg_list_length = n_regs * reg_chars + 2;
|
||
*hex_reg_list = malloc(hex_reg_list_length);
|
||
*hex_reg_list[0] = '\0';
|
||
for (size_t i = 0; i < n_regs; ++i) {
|
||
if (riscv_has_register(rtos->target, thread_id, i)) {
|
||
uint64_t reg_value = riscv_get_register_on_hart(rtos->target, thread_id - 1, i);
|
||
for (size_t byte = 0; byte < xlen / 8; ++byte) {
|
||
uint8_t reg_byte = reg_value >> (byte * 8);
|
||
char hex[3] = {'x', 'x', 'x'};
|
||
snprintf(hex, 3, "%02x", reg_byte);
|
||
strncat(*hex_reg_list, hex, hex_reg_list_length);
|
||
}
|
||
} else {
|
||
for (size_t byte = 0; byte < xlen / 8; ++byte)
|
||
strncat(*hex_reg_list, "xx", hex_reg_list_length);
|
||
}
|
||
}
|
||
LOG_DEBUG("%s", *hex_reg_list);
|
||
return JIM_OK;
|
||
}
|
||
|
||
static int riscv_get_symbol_list_to_lookup(symbol_table_elem_t *symbol_list[])
|
||
{
|
||
*symbol_list = calloc(1, sizeof(symbol_table_elem_t));
|
||
(*symbol_list)[0].symbol_name = NULL;
|
||
(*symbol_list)[0].optional = false;
|
||
return JIM_OK;
|
||
}
|
||
|
||
const struct rtos_type riscv_rtos =
|
||
{
|
||
.name = "riscv",
|
||
.detect_rtos = riscv_detect_rtos,
|
||
.create = riscv_create_rtos,
|
||
.update_threads = riscv_update_threads,
|
||
.get_thread_reg_list = riscv_get_thread_reg_list,
|
||
.get_symbol_list_to_lookup = riscv_get_symbol_list_to_lookup,
|
||
};
|