/* * This file is part of the Black Magic Debug project. * * Copyright (C) 2011 Black Sphere Technologies Ltd. * Written by Gareth McMullin * Copyright (C) 2021 Uwe Bonnes * (bon@elektron.ikp.physik.tu-darmstadt.de) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ /* This file implements a basic command interpreter for GDB 'monitor' * commands. */ #include "general.h" #include "exception.h" #include "command.h" #include "gdb_packet.h" #include "target.h" #include "target_internal.h" #include "morse.h" #include "version.h" #include "serialno.h" #include "jtagtap.h" #ifdef ENABLE_RTT #include "rtt.h" #endif #ifdef PLATFORM_HAS_TRACESWO #include "traceswo.h" #endif #include static bool cmd_version(target *t, int argc, const char **argv); static bool cmd_help(target *t, int argc, const char **argv); static bool cmd_jtag_scan(target *t, int argc, const char **argv); static bool cmd_swdp_scan(target *t, int argc, const char **argv); static bool cmd_auto_scan(target *t, int argc, const char **argv); static bool cmd_frequency(target *t, int argc, const char **argv); static bool cmd_targets(target *t, int argc, const char **argv); static bool cmd_morse(target *t, int argc, const char **argv); static bool cmd_halt_timeout(target *t, int argc, const char **argv); static bool cmd_connect_reset(target *t, int argc, const char **argv); static bool cmd_reset(target *t, int argc, const char **argv); static bool cmd_tdi_low_reset(target *t, int argc, const char **argv); #ifdef PLATFORM_HAS_POWER_SWITCH static bool cmd_target_power(target *t, int argc, const char **argv); #endif #ifdef PLATFORM_HAS_TRACESWO static bool cmd_traceswo(target *t, int argc, const char **argv); #endif static bool cmd_heapinfo(target *t, int argc, const char **argv); #ifdef ENABLE_RTT static bool cmd_rtt(target *t, int argc, const char **argv); #endif #if defined(PLATFORM_HAS_DEBUG) && (PC_HOSTED == 0) static bool cmd_debug_bmp(target *t, int argc, const char **argv); #endif const command_t cmd_list[] = { {"version", cmd_version, "Display firmware version info"}, {"help", cmd_help, "Display help for monitor commands"}, {"jtag_scan", cmd_jtag_scan, "Scan JTAG chain for devices"}, {"swdp_scan", cmd_swdp_scan, "Scan SW-DP for devices"}, {"auto_scan", cmd_auto_scan, "Automatically scan all chain types for devices"}, {"frequency", cmd_frequency, "set minimum high and low times"}, {"targets", cmd_targets, "Display list of available targets"}, {"morse", cmd_morse, "Display morse error message"}, {"halt_timeout", cmd_halt_timeout, "Timeout (ms) to wait until Cortex-M is halted: (Default 2000)"}, {"connect_rst", cmd_connect_reset, "Configure connect under reset: (enable|disable)"}, {"reset", cmd_reset, "Pulse the nRST line - disconnects target"}, {"tdi_low_reset", cmd_tdi_low_reset, "Pulse nRST with TDI set low to attempt to wake certain targets up (eg LPC82x)"}, #ifdef PLATFORM_HAS_POWER_SWITCH {"tpwr", cmd_target_power, "Supplies power to the target: (enable|disable)"}, #endif #ifdef ENABLE_RTT {"rtt", cmd_rtt, "enable|disable|status|channel 0..15|ident (str)|cblock|poll maxms minms maxerr"}, #endif #ifdef PLATFORM_HAS_TRACESWO #if defined TRACESWO_PROTOCOL && TRACESWO_PROTOCOL == 2 {"traceswo", cmd_traceswo, "Start trace capture, NRZ mode: (baudrate) (decode channel ...)"}, #else {"traceswo", cmd_traceswo, "Start trace capture, Manchester mode: (decode channel ...)"}, #endif #endif {"heapinfo", cmd_heapinfo, "Set semihosting heapinfo"}, #if defined(PLATFORM_HAS_DEBUG) && (PC_HOSTED == 0) {"debug_bmp", cmd_debug_bmp, "Output BMP \"debug\" strings to the second vcom: (enable|disable)"}, #endif {NULL, NULL, NULL}, }; bool connect_assert_nrst; #if defined(PLATFORM_HAS_DEBUG) && (PC_HOSTED == 0) bool debug_bmp; #endif unsigned cortexm_wait_timeout = 2000; /* Timeout to wait for Cortex to react on halt command. */ int command_process(target *t, char *cmd) { /* Initial estimate for argc */ size_t argc = 1; for (size_t i = 0; i < strlen(cmd); ++i) { if (cmd[i] == ' ' || cmd[i] == '\t') ++argc; } /* This needs replacing with something more sensible. * It should be pinging -Wvla among other things, and it failing is straight-up UB */ const char **const argv = alloca(sizeof(const char *) * argc); /* Tokenize cmd to find argv */ argc = 0; for (const char *part = strtok(cmd, " \t"); part; part = strtok(NULL, " \t")) argv[argc++] = part; /* Look for match and call handler */ for (const command_t *cmd = cmd_list; cmd->cmd; ++cmd) { /* Accept a partial match as GDB does. * So 'mon ver' will match 'monitor version' */ if ((argc == 0) || !strncmp(argv[0], cmd->cmd, strlen(argv[0]))) return !cmd->handler(t, argc, argv); } if (!t) return -1; return target_command(t, argc, argv); } #define BOARD_IDENT "Black Magic Probe" PLATFORM_IDENT FIRMWARE_VERSION bool cmd_version(target *t, int argc, const char **argv) { (void)t; (void)argc; (void)argv; #if PC_HOSTED == 1 char ident[256]; gdb_ident(ident, sizeof(ident)); DEBUG_WARN("%s\n", ident); #else gdb_out(BOARD_IDENT); gdb_outf(", Hardware Version %d\n", platform_hwversion()); gdb_out("Copyright (C) 2022 Black Magic Debug Project\n"); gdb_out("License GPLv3+: GNU GPL version 3 or later " "\n\n"); #endif return true; } bool cmd_help(target *t, int argc, const char **argv) { (void)argc; (void)argv; if (!t || t->tc->destroy_callback) { gdb_out("General commands:\n"); for (const command_t *cmd = cmd_list; cmd->cmd; cmd++) gdb_outf("\t%s -- %s\n", cmd->cmd, cmd->help); if (!t) return true; } target_command_help(t); return true; } static bool cmd_jtag_scan(target *t, int argc, const char **argv) { (void)t; uint8_t irlens[argc]; if (platform_target_voltage()) gdb_outf("Target voltage: %s\n", platform_target_voltage()); if (argc > 1) { /* Accept a list of IR lengths on command line */ for (size_t i = 1; i < (size_t)argc; i++) irlens[i - 1] = strtoul(argv[i], NULL, 0); irlens[argc - 1] = 0; } if (connect_assert_nrst) platform_nrst_set_val(true); /* will be deasserted after attach */ int devs = -1; volatile struct exception e; TRY_CATCH (e, EXCEPTION_ALL) { #if PC_HOSTED == 1 devs = platform_jtag_scan(argc > 1 ? irlens : NULL); #else devs = jtag_scan(argc > 1 ? irlens : NULL); #endif } switch (e.type) { case EXCEPTION_TIMEOUT: gdb_outf("Timeout during scan. Is target stuck in WFI?\n"); break; case EXCEPTION_ERROR: gdb_outf("Exception: %s\n", e.msg); break; } if (devs == 0) { platform_nrst_set_val(false); gdb_out("JTAG device scan failed!\n"); return false; } cmd_targets(NULL, 0, NULL); morse(NULL, false); return true; } bool cmd_swdp_scan(target *t, int argc, const char **argv) { (void)t; volatile uint32_t targetid = 0; if (argc > 1) targetid = strtol(argv[1], NULL, 0); if (platform_target_voltage()) gdb_outf("Target voltage: %s\n", platform_target_voltage()); if (connect_assert_nrst) platform_nrst_set_val(true); /* will be deasserted after attach */ int devs = -1; volatile struct exception e; TRY_CATCH (e, EXCEPTION_ALL) { #if PC_HOSTED == 1 devs = platform_adiv5_swdp_scan(targetid); #else devs = adiv5_swdp_scan(targetid); #endif } switch (e.type) { case EXCEPTION_TIMEOUT: gdb_outf("Timeout during scan. Is target stuck in WFI?\n"); break; case EXCEPTION_ERROR: gdb_outf("Exception: %s\n", e.msg); break; } if (devs <= 0) { platform_nrst_set_val(false); gdb_out("SW-DP scan failed!\n"); return false; } cmd_targets(NULL, 0, NULL); morse(NULL, false); return true; } bool cmd_auto_scan(target *t, int argc, const char **argv) { (void)t; (void)argc; (void)argv; if (platform_target_voltage()) gdb_outf("Target voltage: %s\n", platform_target_voltage()); if (connect_assert_nrst) platform_nrst_set_val(true); /* will be deasserted after attach */ int devs = -1; volatile struct exception e; TRY_CATCH (e, EXCEPTION_ALL) { #if PC_HOSTED == 1 devs = platform_jtag_scan(NULL); #else devs = jtag_scan(NULL); #endif if (devs > 0) break; gdb_out("JTAG scan found no devices, trying SWD!\n"); #if PC_HOSTED == 1 devs = platform_adiv5_swdp_scan(0); #else devs = adiv5_swdp_scan(0); #endif if (devs > 0) break; platform_nrst_set_val(false); gdb_out("SW-DP scan failed!\n"); return false; } switch (e.type) { case EXCEPTION_TIMEOUT: gdb_outf("Timeout during scan. Is target stuck in WFI?\n"); break; case EXCEPTION_ERROR: gdb_outf("Exception: %s\n", e.msg); break; } if (devs <= 0) { platform_nrst_set_val(false); gdb_out("auto scan failed!\n"); return false; } cmd_targets(NULL, 0, NULL); morse(NULL, false); return true; } bool cmd_frequency(target *t, int argc, const char **argv) { (void)t; if (argc == 2) { char *multiplier = NULL; uint32_t frequency = strtoul(argv[1], &multiplier, 10); if (!multiplier) { gdb_outf("Frequency must be an integral value possibly followed by 'k' or 'M'"); return false; } switch (*multiplier) { case 'k': frequency *= 1000U; break; case 'M': frequency *= 1000U * 1000U; break; } platform_max_frequency_set(frequency); } const uint32_t freq = platform_max_frequency_get(); if (freq == FREQ_FIXED) gdb_outf("SWJ freq fixed\n"); else gdb_outf("Current SWJ freq %" PRIu32 "Hz\n", freq); return true; } static void display_target(int i, target *t, void *context) { (void)context; const char attached = target_attached(t) ? '*' : ' '; const char *const core_name = target_core_name(t); if (!strcmp(target_driver_name(t), "ARM Cortex-M")) gdb_outf("***%2d %c Unknown %s Designer 0x%x Part ID 0x%x %s\n", i, attached, target_driver_name(t), target_designer(t), target_part_id(t), core_name ? core_name : ""); else gdb_outf("%2d %c %s %s\n", i, attached, target_driver_name(t), core_name ? core_name : ""); } bool cmd_targets(target *t, int argc, const char **argv) { (void)t; (void)argc; (void)argv; gdb_out("Available Targets:\n"); gdb_out("No. Att Driver\n"); if (!target_foreach(display_target, NULL)) { gdb_out("No usable targets found.\n"); return false; } return true; } bool cmd_morse(target *t, int argc, const char **argv) { (void)t; (void)argc; (void)argv; if (morse_msg) { gdb_outf("%s\n", morse_msg); DEBUG_WARN("%s\n", morse_msg); } else gdb_out("No message\n"); return true; } bool parse_enable_or_disable(const char *value, bool *out) { const size_t value_len = strlen(value); if (value_len && !strncmp(value, "enable", value_len)) *out = true; else if (value_len && !strncmp(value, "disable", value_len)) *out = false; else { gdb_out("'enable' or 'disable' argument must be provided\n"); return false; } return true; } static bool cmd_connect_reset(target *t, int argc, const char **argv) { (void)t; bool print_status = false; if (argc == 1) print_status = true; else if (argc == 2) { if (parse_enable_or_disable(argv[1], &connect_assert_nrst)) print_status = true; } else gdb_out("Unrecognized command format\n"); if (print_status) { gdb_outf("Assert nRST during connect: %s\n", connect_assert_nrst ? "enabled" : "disabled"); } return true; } static bool cmd_halt_timeout(target *t, int argc, const char **argv) { (void)t; if (argc > 1) cortexm_wait_timeout = strtoul(argv[1], NULL, 0); gdb_outf("Cortex-M timeout to wait for device halts: %d\n", cortexm_wait_timeout); return true; } static bool cmd_reset(target *t, int argc, const char **argv) { (void)t; (void)argc; (void)argv; target_list_free(); platform_nrst_set_val(true); platform_nrst_set_val(false); return true; } static bool cmd_tdi_low_reset(target *t, int argc, const char **argv) { (void)t; (void)argc; (void)argv; jtag_proc.jtagtap_next(true, false); cmd_reset(NULL, 0, NULL); return true; } #ifdef PLATFORM_HAS_POWER_SWITCH static bool cmd_target_power(target *t, int argc, const char **argv) { (void)t; if (argc == 1) gdb_outf("Target Power: %s\n", platform_target_get_power() ? "enabled" : "disabled"); else if (argc == 2) { bool want_enable = false; if (parse_enable_or_disable(argv[1], &want_enable)) { if (want_enable && !platform_target_get_power() && platform_target_voltage_sense() > POWER_CONFLICT_THRESHOLD) { /* want to enable target power, but VREF > 0.5V sensed -> cancel */ gdb_outf("Target already powered (%s)\n", platform_target_voltage()); } else { platform_target_set_power(want_enable); gdb_outf("%s target power\n", want_enable ? "Enabling" : "Disabling"); } } } else gdb_outf("Unrecognized command format\n"); return true; } #endif #ifdef ENABLE_RTT static const char *on_or_off(const bool value) { return value ? "on" : "off"; } static bool cmd_rtt(target *t, int argc, const char **argv) { (void)t; const size_t command_len = strlen(argv[1]); if (argc == 1 || (argc == 2 && !strncmp(argv[1], "enabled", command_len))) { rtt_enabled = true; rtt_found = false; } else if ((argc == 2) && !strncmp(argv[1], "disabled", command_len)) { rtt_enabled = false; rtt_found = false; } else if ((argc == 2) && !strncmp(argv[1], "status", command_len)) { gdb_outf("rtt: %s found: %s ident: \"%s\"", on_or_off(rtt_enabled), rtt_found ? "yes" : "no", rtt_ident[0] == '\0' ? "off" : rtt_ident); gdb_outf(" halt: %s", on_or_off(target_no_background_memory_access(t))); gdb_out(" channels: "); if (rtt_auto_channel) gdb_out("auto "); for (size_t i = 0; i < MAX_RTT_CHAN; i++) { if (rtt_channel[i].is_enabled) gdb_outf("%d ", i); } gdb_outf( "\nmax poll ms: %u min poll ms: %u max errs: %u\n", rtt_max_poll_ms, rtt_min_poll_ms, rtt_max_poll_errs); } else if (argc >= 2 && !strncmp(argv[1], "channel", command_len)) { /* mon rtt channel switches to auto rtt channel selection mon rtt channel number... selects channels given */ for (size_t i = 0; i < MAX_RTT_CHAN; i++) rtt_channel[i].is_enabled = false; if (argc == 2) rtt_auto_channel = true; else { rtt_auto_channel = false; for (size_t i = 2; i < (size_t)argc; ++i) { const uint32_t channel = strtoul(argv[i], NULL, 0); if (channel < MAX_RTT_CHAN) rtt_channel[channel].is_enabled = true; } } } else if (argc == 2 && !strncmp(argv[1], "ident", command_len)) rtt_ident[0] = '\0'; else if (argc == 2 && !strncmp(argv[1], "poll", command_len)) gdb_outf("%u %u %u\n", rtt_max_poll_ms, rtt_min_poll_ms, rtt_max_poll_errs); else if (argc == 2 && !strncmp(argv[1], "cblock", command_len)) { gdb_outf("cbaddr: 0x%x\n", rtt_cbaddr); gdb_out("ch ena cfg i/o buf@ size head@ tail@ flg\n"); for (size_t i = 0; i < MAX_RTT_CHAN; ++i) { gdb_outf("%2zu %c %c %s 0x%08x %5d 0x%08x 0x%08x %d\n", i, rtt_channel[i].is_enabled ? 'y' : 'n', rtt_channel[i].is_configured ? 'y' : 'n', rtt_channel[i].is_output ? "out" : "in ", rtt_channel[i].buf_addr, rtt_channel[i].buf_size, rtt_channel[i].head_addr, rtt_channel[i].tail_addr, rtt_channel[i].flag); } } else if (argc == 3 && !strncmp(argv[1], "ident", command_len)) { strncpy(rtt_ident, argv[2], sizeof(rtt_ident)); rtt_ident[sizeof(rtt_ident) - 1] = '\0'; for (size_t i = 0; i < sizeof(rtt_ident); i++) { if (rtt_ident[i] == '_') rtt_ident[i] = ' '; } } else if (argc == 5 && !strncmp(argv[1], "poll", command_len)) { /* set polling params */ rtt_max_poll_ms = strtoul(argv[2], NULL, 0); rtt_min_poll_ms = strtoul(argv[3], NULL, 0); rtt_max_poll_errs = strtoul(argv[4], NULL, 0); } else gdb_out("what?\n"); return true; } #endif #ifdef PLATFORM_HAS_TRACESWO static bool cmd_traceswo(target *t, int argc, const char **argv) { (void)t; #if TRACESWO_PROTOCOL == 2 uint32_t baudrate = SWO_DEFAULT_BAUD; #endif uint32_t swo_channelmask = 0; /* swo decoding off */ uint8_t decode_arg = 1; #if TRACESWO_PROTOCOL == 2 /* argument: optional baud rate for async mode */ if (argc > 1 && argv[1][0] >= '0' && argv[1][0] <= '9') { baudrate = strtoul(argv[1], NULL, 0); if (baudrate == 0) baudrate = SWO_DEFAULT_BAUD; decode_arg = 2; } #endif /* argument: 'decode' literal */ if (argc > decode_arg && !strncmp(argv[decode_arg], "decode", strlen(argv[decode_arg]))) { swo_channelmask = 0xFFFFFFFFU; /* decoding all channels */ /* arguments: channels to decode */ if (argc > decode_arg + 1) { swo_channelmask = 0U; for (size_t i = decode_arg + 1; i < (size_t)argc; ++i) { /* create bitmask of channels to decode */ const uint32_t channel = strtoul(argv[i], NULL, 0); if (channel < 32) swo_channelmask |= 1U << channel; } } } #if TRACESWO_PROTOCOL == 2 gdb_outf("Baudrate: %lu ", baudrate); #endif gdb_outf("Channel mask: "); for (size_t i = 0; i < 32; ++i) { const uint32_t bit = (swo_channelmask >> (31U - i)) & 1U; gdb_outf("%" PRIu32, bit); } gdb_outf("\n"); #if TRACESWO_PROTOCOL == 2 traceswo_init(baudrate, swo_channelmask); #else traceswo_init(swo_channelmask); #endif char serial_no[DFU_SERIAL_LENGTH]; serial_no_read(serial_no); gdb_outf("Trace enabled for BMP serial %s, USB EP 5\n", serial_no); return true; } #endif #if defined(PLATFORM_HAS_DEBUG) && (PC_HOSTED == 0) static bool cmd_debug_bmp(target *t, int argc, const char **argv) { (void)t; if (argc == 2) { if (!parse_enable_or_disable(argv[1], &debug_bmp)) return false; } else if (argc > 2) { gdb_outf("usage: monitor debug [enable|disable]\n"); return false; } gdb_outf("Debug mode is %s\n", debug_bmp ? "enabled" : "disabled"); return true; } #endif static bool cmd_heapinfo(target *t, int argc, const char **argv) { if (t == NULL) gdb_out("not attached\n"); else if (argc == 5) { target_addr heap_base = strtoul(argv[1], NULL, 16); target_addr heap_limit = strtoul(argv[2], NULL, 16); target_addr stack_base = strtoul(argv[3], NULL, 16); target_addr stack_limit = strtoul(argv[4], NULL, 16); gdb_outf("heapinfo heap_base: %p heap_limit: %p stack_base: %p stack_limit: %p\n", heap_base, heap_limit, stack_base, stack_limit); target_set_heapinfo(t, heap_base, heap_limit, stack_base, stack_limit); } else gdb_outf("heapinfo heap_base heap_limit stack_base stack_limit\n"); return true; }