mspdebug/main.c

415 lines
8.6 KiB
C

/* MSPDebug - debugging tool for MSP430 MCUs
* Copyright (C) 2009, 2010 Daniel Beer
*
* 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 2 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, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <stdio.h>
#include <ctype.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <getopt.h>
#include "dis.h"
#include "device.h"
#include "binfile.h"
#include "stab.h"
#include "util.h"
#include "gdb.h"
#include "rtools.h"
#include "sym.h"
#include "devcmd.h"
#include "expr.h"
#include "sim.h"
#include "bsl.h"
#include "fet.h"
#include "vector.h"
#include "fet_db.h"
#include "uif.h"
#include "rf2500.h"
static void io_prefix(stab_t stab,
const char *prefix, uint16_t pc,
uint16_t addr, int is_byte)
{
char name[64];
if (!stab_nearest(stab, pc, name, sizeof(name), &pc)) {
printf("%s", name);
if (pc)
printf("+0x%x", pc);
} else {
printf("0x%04x", pc);
}
printf(": IO %s.%c: 0x%04x", prefix, is_byte ? 'B' : 'W', addr);
if (!stab_nearest(stab, addr, name, sizeof(name), &addr)) {
printf(" (%s", name);
if (addr)
printf("+0x%x", addr);
printf(")");
}
}
static int fetch_io(void *user_data, uint16_t pc,
uint16_t addr, int is_byte, uint16_t *data_ret)
{
stab_t stab = (stab_t)user_data;
io_prefix(stab, "READ", pc, addr, is_byte);
for (;;) {
char text[128];
int len;
int data;
printf("? ");
fflush(stdout);
if (!fgets(text, sizeof(text), stdin)) {
printf("\nAborted IO request\n");
return -1;
}
len = strlen(text);
while (len && isspace(text[len - 1]))
len--;
text[len] = 0;
if (!len)
return 0;
if (!expr_eval(stab, text, &data)) {
if (data_ret)
*data_ret = data;
return 0;
}
}
return 0;
}
static void store_io(void *user_data, uint16_t pc,
uint16_t addr, int is_byte, uint16_t data)
{
stab_t stab = (stab_t)user_data;
io_prefix(stab, "WRITE", pc, addr, is_byte);
if (is_byte)
printf(" => 0x%02x\n", data & 0xff);
else
printf(" => 0x%04x\n", data);
}
static void usage(const char *progname)
{
fprintf(stderr,
"Usage: %s [options] -R [-v voltage] [command ...]\n"
" %s [options] -u <device> [-j] [-v voltage] [command ...]\n"
" %s [options] -B <device> [command ...]\n"
" %s [options] -s [command ...]\n"
"\n"
" -R\n"
" Open the first available RF2500 device on the USB bus.\n"
" -u device\n"
" Open the given tty device (MSP430 UIF compatible devices).\n"
" -j\n"
" Use JTAG, rather than spy-bi-wire (UIF devices only).\n"
" -v voltage\n"
" Set the supply voltage, in millivolts.\n"
" -B device\n"
" Debug the FET itself through the bootloader.\n"
" -s\n"
" Start in simulation mode.\n"
" -n\n"
" Do not read ~/.mspdebug on startup.\n"
" --help\n"
" Show this help text.\n"
" --fet-list\n"
" Show a list of devices supported by the FET driver.\n"
" --fet-force-id string\n"
" Override the device ID returned by the FET.\n"
"\n"
"By default, the first RF2500 device on the USB bus is opened.\n"
"\n"
"If commands are given, they will be executed. Otherwise, an interactive\n"
"command reader is started.\n",
progname, progname, progname, progname);
}
static void process_rc_file(cproc_t cp)
{
const char *home = getenv("HOME");
char text[256];
if (!home)
return;
snprintf(text, sizeof(text), "%s/.mspdebug", home);
if (!access(text, F_OK))
cproc_process_file(cp, text);
}
#define MODE_RF2500 0x01
#define MODE_UIF 0x02
#define MODE_UIF_BSL 0x04
#define MODE_SIM 0x08
struct cmdline_args {
const char *uif_device;
const char *bsl_device;
const char *fet_force_id;
int mode;
int want_jtag;
int no_rc;
int vcc_mv;
};
static int add_fet_device(void *user_data, const struct fet_db_record *r)
{
struct vector *v = (struct vector *)user_data;
return vector_push(v, &r->name, 1);
}
static int cmp_char_ptr(const void *a, const void *b)
{
return strcmp(*(const char **)a, *(const char **)b);
}
static int list_devices(void)
{
struct vector v;
int i;
vector_init(&v, sizeof(const char *));
if (fet_db_enum(add_fet_device, &v) < 0) {
perror("couldn't allocate memory");
vector_destroy(&v);
return -1;
}
qsort(v.ptr, v.size, v.elemsize, cmp_char_ptr);
printf("Devices supported by FET driver:\n");
for (i = 0; i < v.size; i++)
printf(" %s\n", VECTOR_AT(v, i, const char *));
vector_destroy(&v);
return 0;
}
static int parse_cmdline_args(int argc, char **argv,
struct cmdline_args *args)
{
int opt;
const static struct option longopts[] = {
{"help", 0, 0, 'H'},
{"fet-list", 0, 0, 'L'},
{"fet-force-id", 1, 0, 'F'},
{NULL, 0, 0, 0}
};
while ((opt = getopt_long(argc, argv, "u:jv:B:sR?n",
longopts, NULL)) >= 0)
switch (opt) {
case 'L':
exit(list_devices());
case 'F':
args->fet_force_id = optarg;
break;
case 'H':
usage(argv[0]);
exit(0);
case 'R':
args->mode |= MODE_RF2500;
break;
case 'u':
args->uif_device = optarg;
args->mode |= MODE_UIF;
break;
case 'v':
args->vcc_mv = atoi(optarg);
break;
case 'j':
args->want_jtag = 1;
break;
case 'B':
args->bsl_device = optarg;
args->mode |= MODE_UIF_BSL;
break;
case 's':
args->mode |= MODE_SIM;
break;
case 'n':
args->no_rc = 1;
break;
case '?':
return -1;
default:
fprintf(stderr, "Invalid argument: %c\n"
"Try --help for help.\n", opt);
return -1;
}
/* Check for incompatible arguments */
if (args->mode & (args->mode - 1)) {
fprintf(stderr, "Multiple incompatible options specified.\n"
"Try --help for help.\n");
return -1;
}
if (!args->mode) {
fprintf(stderr, "You need to specify an operating mode.\n"
"Try --help for help.\n");
return -1;
}
return 0;
}
device_t setup_device(const struct cmdline_args *args,
stab_t stab)
{
device_t msp430_dev = NULL;
transport_t trans = NULL;
/* Open a device */
if (args->mode == MODE_SIM) {
msp430_dev = sim_open(fetch_io, store_io, stab);
} else if (args->mode == MODE_UIF_BSL) {
msp430_dev = bsl_open(args->bsl_device);
} else if (args->mode == MODE_RF2500 || args->mode == MODE_UIF) {
int flags = 0;
/* Open the appropriate transport */
if (args->mode == MODE_UIF) {
trans = uif_open(args->uif_device);
} else {
trans = rf2500_open();
flags |= FET_PROTO_RF2500;
}
if (!trans)
return NULL;
/* Then initialize the device */
if (!args->want_jtag)
flags |= FET_PROTO_SPYBIWIRE;
msp430_dev = fet_open(trans, flags, args->vcc_mv,
args->fet_force_id);
}
if (!msp430_dev) {
if (trans)
trans->destroy(trans);
return NULL;
}
return msp430_dev;
}
cproc_t setup_cproc(const struct cmdline_args *args)
{
device_t msp430_dev;
stab_t stab;
cproc_t cp;
stab = stab_new();
if (!stab)
return NULL;
msp430_dev = setup_device(args, stab);
if (!msp430_dev) {
stab_destroy(stab);
return NULL;
}
cp = cproc_new(msp430_dev, stab);
if (!cp) {
msp430_dev->destroy(msp430_dev);
stab_destroy(stab);
return NULL;
}
if (sym_register(cp) < 0 ||
devcmd_register(cp) < 0 ||
gdb_register(cp) < 0 ||
rtools_register(cp) < 0) {
cproc_destroy(cp);
return NULL;
}
return cp;
}
int main(int argc, char **argv)
{
struct cmdline_args args = {0};
cproc_t cp;
int ret = 0;
puts(
"MSPDebug version 0.7 - debugging tool for MSP430 MCUs\n"
"Copyright (C) 2009, 2010 Daniel Beer <daniel@tortek.co.nz>\n"
"This is free software; see the source for copying conditions. There is NO\n"
"warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n");
args.vcc_mv = 3000;
if (parse_cmdline_args(argc, argv, &args) < 0)
return -1;
ctrlc_init();
cp = setup_cproc(&args);
if (!cp)
return -1;
if (!args.no_rc)
process_rc_file(cp);
/* Process commands */
if (optind < argc) {
while (optind < argc) {
if (cproc_process_command(cp, argv[optind++]) < 0) {
ret = -1;
break;
}
}
} else {
cproc_reader_loop(cp);
}
cproc_destroy(cp);
return ret;
}