Support for MSP-JTAG-TINY via USB bus (CP2102).
This commit is contained in:
parent
cf4ec417ff
commit
9dd216e604
1
AUTHORS
1
AUTHORS
|
@ -15,3 +15,4 @@ Hans Nieuwenhuis <vzzbx@xs4all.nl>:
|
|||
Peter Jansen <pwjansen@yahoo.com>:
|
||||
* Support for MSP430F169.
|
||||
* Testing and analysis for Olimex MSP430-JTAG-TINY.
|
||||
* libusb driver for Olimex MSP430-JTAG-TINY.
|
||||
|
|
4
Makefile
4
Makefile
|
@ -43,8 +43,8 @@ install: mspdebug mspdebug.man
|
|||
|
||||
.SUFFIXES: .c .o
|
||||
|
||||
mspdebug: main.o fet.o rf2500.o dis.o uif.o ihex.o elf32.o stab.o util.o \
|
||||
bsl.o sim.o symmap.o gdb.o btree.o rtools.o sym.o devcmd.o \
|
||||
mspdebug: main.o fet.o rf2500.o dis.o uif.o olimex.o ihex.o elf32.o stab.o \
|
||||
util.o bsl.o sim.o symmap.o gdb.o btree.o rtools.o sym.o devcmd.o \
|
||||
cproc.o vector.o cproc_util.o expr.o fet_error.o binfile.o fet_db.o
|
||||
$(CC) $(LDFLAGS) -o $@ $^ -lusb $(READLINE_LIBS)
|
||||
|
||||
|
|
19
main.c
19
main.c
|
@ -42,6 +42,7 @@
|
|||
#include "fet_db.h"
|
||||
|
||||
#include "uif.h"
|
||||
#include "olimex.h"
|
||||
#include "rf2500.h"
|
||||
|
||||
static void io_prefix(stab_t stab,
|
||||
|
@ -123,6 +124,7 @@ static void usage(const char *progname)
|
|||
"Usage: %s [options] -R [-v voltage] [command ...]\n"
|
||||
" %s [options] -u <device> [-j] [-v voltage] [command ...]\n"
|
||||
" %s [options] -O <device> [-j] [-v voltage] [command ...]\n"
|
||||
" %s [options] -l [-j] [-v voltage] [command ...]\n"
|
||||
" %s [options] -B <device> [command ...]\n"
|
||||
" %s [options] -s [command ...]\n"
|
||||
"\n"
|
||||
|
@ -132,6 +134,8 @@ static void usage(const char *progname)
|
|||
" Open the given tty device (FET430UIF compatible devices).\n"
|
||||
" -O device\n"
|
||||
" Open the given tty device (Olimex MSP430-JTAG-TINY).\n"
|
||||
" -l\n"
|
||||
" Open the first available Olimex MSP430-JTAG-TINY using libusb.\n"
|
||||
" -j\n"
|
||||
" Use JTAG, rather than Spy-Bi-Wire (UIF devices only).\n"
|
||||
" -v voltage\n"
|
||||
|
@ -153,7 +157,7 @@ static void usage(const char *progname)
|
|||
"\n"
|
||||
"If commands are given, they will be executed. Otherwise, an interactive\n"
|
||||
"command reader is started.\n",
|
||||
progname, progname, progname, progname, progname);
|
||||
progname, progname, progname, progname, progname, progname);
|
||||
}
|
||||
|
||||
static void process_rc_file(cproc_t cp)
|
||||
|
@ -174,6 +178,7 @@ static void process_rc_file(cproc_t cp)
|
|||
#define MODE_UIF_BSL 0x04
|
||||
#define MODE_SIM 0x08
|
||||
#define MODE_OLIMEX 0x10
|
||||
#define MODE_OLIMEX_USB 0x20
|
||||
|
||||
struct cmdline_args {
|
||||
const char *devpath;
|
||||
|
@ -229,7 +234,7 @@ static int parse_cmdline_args(int argc, char **argv,
|
|||
{NULL, 0, 0, 0}
|
||||
};
|
||||
|
||||
while ((opt = getopt_long(argc, argv, "u:jv:B:O:sR?n",
|
||||
while ((opt = getopt_long(argc, argv, "lu:jv:B:O:sR?n",
|
||||
longopts, NULL)) >= 0)
|
||||
switch (opt) {
|
||||
case 'O':
|
||||
|
@ -257,6 +262,10 @@ static int parse_cmdline_args(int argc, char **argv,
|
|||
args->mode |= MODE_UIF;
|
||||
break;
|
||||
|
||||
case 'l':
|
||||
args->mode |= MODE_OLIMEX_USB;
|
||||
break;
|
||||
|
||||
case 'v':
|
||||
args->vcc_mv = atoi(optarg);
|
||||
break;
|
||||
|
@ -323,6 +332,9 @@ device_t setup_device(const struct cmdline_args *args,
|
|||
flags |= FET_PROTO_OLIMEX;
|
||||
} else if (args->mode == MODE_UIF) {
|
||||
trans = uif_open(args->devpath, 0);
|
||||
} else if (args->mode == MODE_OLIMEX_USB) {
|
||||
trans = olimex_open();
|
||||
flags |= FET_PROTO_OLIMEX;
|
||||
} else {
|
||||
trans = rf2500_open();
|
||||
flags |= FET_PROTO_RF2500;
|
||||
|
@ -392,7 +404,8 @@ int main(int argc, char **argv)
|
|||
"MSPDebug version 0.8 - 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");
|
||||
"warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR "
|
||||
"PURPOSE.\n");
|
||||
|
||||
args.vcc_mv = 3000;
|
||||
if (parse_cmdline_args(argc, argv, &args) < 0)
|
||||
|
|
|
@ -8,6 +8,8 @@ MSPDebug - debugging tool for MSP430 MCUs
|
|||
.br
|
||||
\fBmspdebug\fR [options] \-O \fIdevice\fR [\-j] [\-v \fIvoltage\fR] [\fIcommand\fR ...]
|
||||
.br
|
||||
\fBmspdebug\fR [options] \-l [\-j] [\-v \fIvoltage\fR] [\fIcommand\fR ...]
|
||||
.br
|
||||
\fBmspdebug\fR [options] \-B \fIdevice\fR [\fIcommand\fR ...]
|
||||
.br
|
||||
\fBmspdebug\fR [options] \-s [\fIcommand\fR ...]
|
||||
|
@ -54,6 +56,9 @@ Linux kernel, and should appear as /dev/tty\fIXX\fR when attached.
|
|||
.IP "\-O \fIdevice\fR"
|
||||
Connect to an Olimex MSP430-JTAG-TINY device. The device argument
|
||||
should be the filename of the appropriate tty device.
|
||||
.IP "\-l"
|
||||
Connect to an Olimex MSP430-JTAG-TINY device using libusb. The first
|
||||
available device on the USB bus is claimed.
|
||||
.IP "\-j"
|
||||
Use JTAG instead of Spy-Bi-Wire to communicate with the MSP430. This
|
||||
option only works on FET430UIF devices.
|
||||
|
|
|
@ -0,0 +1,233 @@
|
|||
/* MSPDebug - debugging tool for the eZ430
|
||||
* Copyright (C) 2009, 2010 Daniel Beer
|
||||
* Copyright (C) 2010 Peter Jansen
|
||||
*
|
||||
* 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 <string.h>
|
||||
#include <usb.h>
|
||||
|
||||
#include "olimex.h"
|
||||
#include "util.h"
|
||||
|
||||
struct olimex_transport {
|
||||
struct transport base;
|
||||
|
||||
int int_number;
|
||||
struct usb_dev_handle *handle;
|
||||
|
||||
uint8_t buf[64];
|
||||
int len;
|
||||
int offset;
|
||||
};
|
||||
|
||||
/*********************************************************************
|
||||
* USB transport
|
||||
*
|
||||
* These functions handle the details of slicing data over USB
|
||||
* transfers. The interface presented is a continuous byte stream with
|
||||
* no slicing codes.
|
||||
*
|
||||
* Writes are unbuffered -- a single write translates to at least
|
||||
* one transfer.
|
||||
*/
|
||||
|
||||
#define USB_FET_VENDOR 0x15ba
|
||||
#define USB_FET_PRODUCT 0x0002
|
||||
#define USB_FET_INTERFACE_CLASS 255
|
||||
|
||||
#define USB_FET_IN_EP 0x81
|
||||
#define USB_FET_OUT_EP 0x01
|
||||
|
||||
#define CP210x_REQTYPE_HOST_TO_DEVICE 0x41
|
||||
|
||||
#define CP210X_IFC_ENABLE 0x00
|
||||
#define CP210X_SET_MHS 0x07
|
||||
|
||||
#define TIMEOUT 1000
|
||||
|
||||
static int open_interface(struct olimex_transport *tr,
|
||||
struct usb_device *dev, int ino)
|
||||
{
|
||||
#if !(defined (__APPLE__) || defined(WIN32))
|
||||
int drv;
|
||||
char drName[256];
|
||||
#endif
|
||||
|
||||
printf(__FILE__": Trying to open interface %d on %s\n",
|
||||
ino, dev->filename);
|
||||
|
||||
tr->int_number = ino;
|
||||
|
||||
tr->handle = usb_open(dev);
|
||||
if (!tr->handle) {
|
||||
perror(__FILE__": can't open device");
|
||||
return -1;
|
||||
}
|
||||
|
||||
#if !(defined(__APPLE__) || defined(WIN32))
|
||||
drv = usb_get_driver_np(tr->handle, tr->int_number, drName,
|
||||
sizeof(drName));
|
||||
printf(__FILE__" : driver %d\n", drv);
|
||||
if (drv >= 0) {
|
||||
if (usb_detach_kernel_driver_np(tr->handle,
|
||||
tr->int_number) < 0)
|
||||
perror(__FILE__": warning: can't detach "
|
||||
"kernel driver");
|
||||
}
|
||||
#endif
|
||||
|
||||
if (usb_claim_interface(tr->handle, tr->int_number) < 0) {
|
||||
perror(__FILE__": can't claim interface");
|
||||
usb_close(tr->handle);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int ret = usb_control_msg(tr->handle, CP210x_REQTYPE_HOST_TO_DEVICE,
|
||||
CP210X_IFC_ENABLE, 0x1, 0, NULL, 0, 300);
|
||||
#ifdef DEBUG_OLIMEX
|
||||
printf(__FILE__": %s : Sending control message ret %d\n",
|
||||
__FUNCTION__, ret);
|
||||
#endif
|
||||
ret = usb_control_msg(tr->handle, CP210x_REQTYPE_HOST_TO_DEVICE,
|
||||
CP210X_SET_MHS, 0x303, 0, NULL, 0, 300);
|
||||
#ifdef DEBUG_OLIMEX
|
||||
printf(__FILE__": %s : Sending control message ret %d\n",
|
||||
__FUNCTION__, ret);
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int open_device(struct olimex_transport *tr, struct usb_device *dev)
|
||||
{
|
||||
struct usb_config_descriptor *c = &dev->config[0];
|
||||
int i;
|
||||
|
||||
for (i = 0; i < c->bNumInterfaces; i++) {
|
||||
struct usb_interface *intf = &c->interface[i];
|
||||
struct usb_interface_descriptor *desc = &intf->altsetting[0];
|
||||
|
||||
if (desc->bInterfaceClass == USB_FET_INTERFACE_CLASS &&
|
||||
!open_interface(tr, dev, desc->bInterfaceNumber))
|
||||
return 0;
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int usbtr_send(transport_t tr_base, const uint8_t *data, int len)
|
||||
{
|
||||
struct olimex_transport *tr = (struct olimex_transport *)tr_base;
|
||||
|
||||
int sent;
|
||||
|
||||
while (len) {
|
||||
#ifdef DEBUG_OLIMEX
|
||||
debug_hexdump(__FILE__": USB transfer out", data, len);
|
||||
#endif
|
||||
sent = usb_bulk_write(tr->handle, USB_FET_OUT_EP,
|
||||
(char *)data, len, TIMEOUT);
|
||||
if (sent < 0) {
|
||||
perror(__FILE__": can't send data");
|
||||
return -1;
|
||||
}
|
||||
|
||||
len -= sent;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int usbtr_recv(transport_t tr_base, uint8_t *databuf, int max_len)
|
||||
{
|
||||
struct olimex_transport *tr = (struct olimex_transport *)tr_base;
|
||||
int rlen;
|
||||
|
||||
#ifdef DEBUG_OLIMEX
|
||||
printf(__FILE__": %s : read max %d\n", __FUNCTION__, max_len);
|
||||
#endif
|
||||
|
||||
rlen = usb_bulk_read(tr->handle, USB_FET_IN_EP, (char *)databuf,
|
||||
max_len, TIMEOUT);
|
||||
|
||||
#ifdef DEBUG_OLIMEX
|
||||
printf(__FILE__": %s : read %d\n", __FUNCTION__, rlen);
|
||||
#endif
|
||||
|
||||
if (rlen < 0) {
|
||||
perror(__FILE__": can't receive data");
|
||||
return -1;
|
||||
}
|
||||
|
||||
#ifdef DEBUG_OLIMEX
|
||||
debug_hexdump(__FILE__": USB transfer in", databuf, rlen);
|
||||
#endif
|
||||
|
||||
return rlen;
|
||||
}
|
||||
|
||||
static void usbtr_destroy(transport_t tr_base)
|
||||
{
|
||||
struct olimex_transport *tr = (struct olimex_transport *)tr_base;
|
||||
|
||||
usb_release_interface(tr->handle, tr->int_number);
|
||||
usb_close(tr->handle);
|
||||
free(tr);
|
||||
}
|
||||
|
||||
transport_t olimex_open(void)
|
||||
{
|
||||
struct olimex_transport *tr = malloc(sizeof(*tr));
|
||||
struct usb_bus *bus;
|
||||
|
||||
if (!tr) {
|
||||
perror(__FILE__": can't allocate memory");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
tr->base.destroy = usbtr_destroy;
|
||||
tr->base.send = usbtr_send;
|
||||
tr->base.recv = usbtr_recv;
|
||||
|
||||
usb_init();
|
||||
usb_find_busses();
|
||||
usb_find_devices();
|
||||
|
||||
for (bus = usb_get_busses(); bus; bus = bus->next) {
|
||||
struct usb_device *dev;
|
||||
|
||||
for (dev = bus->devices; dev; dev = dev->next) {
|
||||
if (dev->descriptor.idVendor == USB_FET_VENDOR &&
|
||||
dev->descriptor.idProduct == USB_FET_PRODUCT &&
|
||||
!open_device(tr, dev)) {
|
||||
char buf[64];
|
||||
|
||||
/* Flush out lingering data */
|
||||
while (usb_bulk_read(tr->handle, USB_FET_IN_EP,
|
||||
buf, sizeof(buf),
|
||||
100) >= 0);
|
||||
|
||||
return (transport_t)tr;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fprintf(stderr, __FILE__": no devices could be found\n");
|
||||
free(tr);
|
||||
return NULL;
|
||||
}
|
|
@ -0,0 +1,31 @@
|
|||
/* MSPDebug - debugging tool for MSP430 MCUs
|
||||
* Copyright (C) 2009, 2010 Daniel Beer
|
||||
* Copyright (C) 2010 Peter Jansen
|
||||
*
|
||||
* 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
|
||||
*/
|
||||
|
||||
#ifndef OLIMEX_H_
|
||||
#define OLIMEX_H_
|
||||
|
||||
#include "transport.h"
|
||||
|
||||
/* Search the USB bus for the first Olimex MSP-JTAG-TINY, and
|
||||
* initialize it. If successful, 0 is returned and the fet_* functions
|
||||
* are ready for use. If an error occurs, -1 is returned.
|
||||
*/
|
||||
transport_t olimex_open(void);
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue