Support for flashing or verifying raw binary files.
This commit is contained in:
parent
239fbb7f94
commit
c66e885899
1
AUTHORS
1
AUTHORS
|
@ -68,3 +68,4 @@ Tamas Tevesz <ice@extreme.hu>:
|
|||
|
||||
Ingo van Lil <inguin@gmx.de>:
|
||||
* Support for demangling of C++ function names.
|
||||
* load_raw, verify_raw and save_raw commands.
|
||||
|
|
1
Makefile
1
Makefile
|
@ -146,6 +146,7 @@ OBJ=\
|
|||
ui/rtools.o \
|
||||
ui/sym.o \
|
||||
ui/devcmd.o \
|
||||
ui/flatfile.o \
|
||||
ui/reader.o \
|
||||
ui/cmddb.o \
|
||||
ui/stdcmd.o \
|
||||
|
|
13
mspdebug.man
13
mspdebug.man
|
@ -333,6 +333,10 @@ Program the device under test using the binary file supplied. This
|
|||
command is like \fBprog\fR, but it does not load symbols or erase
|
||||
the device before programming.
|
||||
|
||||
The CPU is reset and halted before and after programming.
|
||||
.IP "\fBload_raw\fR \fIfilename\fR \fIaddress\fR"
|
||||
Write the data contained in a raw binary file to the given memory address.
|
||||
|
||||
The CPU is reset and halted before and after programming.
|
||||
.IP "\fBlocka\fR [\fBset\fR|\fBclear\fR]"
|
||||
Show or change the status of the LOCKA bit in the chip's memory
|
||||
|
@ -403,6 +407,11 @@ is pressed by the user.
|
|||
After the CPU halts, the current register values are shown as well as
|
||||
a disassembly of the first few instructions at the address selected
|
||||
by the program counter.
|
||||
.IP "\fBsave_raw\fR \fIaddress\fR \fIlength\fR \fIfilename\fR"
|
||||
Save a region of memory to a raw binary file. The address and length
|
||||
arguments may both be address expressions.
|
||||
|
||||
If the specified file already exists, then it will be overwritten.
|
||||
.IP "\fBset\fR \fIregister\fR \fIvalue\fR"
|
||||
Alter the value of a register. Registers are specified as numbers from
|
||||
0 through 15. Any leading non-numeric characters are ignored (so a
|
||||
|
@ -497,6 +506,10 @@ renamed.
|
|||
Compare the contents of the given binary file to the chip memory. If any
|
||||
differences are found, a message is printed for the first mismatched
|
||||
byte.
|
||||
.IP "\fBverify_raw \fIfilename\fR \fIaddress\fR"
|
||||
Compare the contents of a raw binary file to the device memory at the given
|
||||
address. If any differences are found, a message is printed for the first
|
||||
mismatched byte.
|
||||
.SH SUPPORTED CHIPS
|
||||
The following chips are supported when using FET-compatible drivers:
|
||||
.PP
|
||||
|
|
24
ui/cmddb.c
24
ui/cmddb.c
|
@ -22,6 +22,7 @@
|
|||
#include "util.h"
|
||||
|
||||
#include "devcmd.h"
|
||||
#include "flatfile.h"
|
||||
#include "gdb.h"
|
||||
#include "rtools.h"
|
||||
#include "sym.h"
|
||||
|
@ -127,6 +128,29 @@ const struct cmddb_record commands[] = {
|
|||
.help =
|
||||
"verify <filename>\n"
|
||||
" Compare the contents of the given binary file to the device memory.\n"
|
||||
},
|
||||
{
|
||||
.name = "load_raw",
|
||||
.func = cmd_load_raw,
|
||||
.help =
|
||||
"load_raw <filename> <address>\n"
|
||||
" Write the data contained in a raw binary file to the given memory\n"
|
||||
" address.\n"
|
||||
},
|
||||
{
|
||||
.name = "verify_raw",
|
||||
.func = cmd_verify_raw,
|
||||
.help =
|
||||
"verify_raw <filename> <address>\n"
|
||||
" Compare the contents of a raw binary file to the device memory at\n"
|
||||
" the given address.\n"
|
||||
},
|
||||
{
|
||||
.name = "save_raw",
|
||||
.func = cmd_save_raw,
|
||||
.help =
|
||||
"save_raw <address> <length> <filename>\n"
|
||||
" Save a region of memory to a raw binary file.\n"
|
||||
},
|
||||
{
|
||||
.name = "md",
|
||||
|
|
|
@ -0,0 +1,243 @@
|
|||
/* MSPDebug - debugging tool for MSP430 MCUs
|
||||
* Copyright (C) 2009-2012 Daniel Beer
|
||||
* Copyright (C) 2012 Ingo van Lil
|
||||
*
|
||||
* 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 "flatfile.h"
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <ctype.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "device.h"
|
||||
#include "expr.h"
|
||||
#include "output_util.h"
|
||||
|
||||
enum operation {
|
||||
LOAD,
|
||||
SAVE,
|
||||
VERIFY
|
||||
};
|
||||
|
||||
static int read_flatfile(const char *path, uint8_t **buf, address_t *len)
|
||||
{
|
||||
FILE *in;
|
||||
char *fullpath = expand_tilde(path);
|
||||
size_t count;
|
||||
|
||||
if (!fullpath)
|
||||
return -1;
|
||||
|
||||
in = fopen(fullpath, "rb");
|
||||
free(fullpath);
|
||||
|
||||
if (!in) {
|
||||
printc_err("%s: %s\n", path, last_error());
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (fseek(in, 0, SEEK_END) < 0) {
|
||||
printc_err("%s: can't seek to end: %s\n", path, last_error());
|
||||
fclose(in);
|
||||
return -1;
|
||||
}
|
||||
|
||||
*len = ftell(in);
|
||||
rewind(in);
|
||||
|
||||
*buf = malloc(*len);
|
||||
if (!*buf) {
|
||||
printc_err("flatfile: can't allocate memory\n");
|
||||
fclose(in);
|
||||
return -1;
|
||||
}
|
||||
|
||||
count = fread(*buf, *len, 1, in);
|
||||
fclose(in);
|
||||
|
||||
if (count != 1) {
|
||||
printc_err("%s: failed to read: %s\n", path, last_error());
|
||||
free(*buf);
|
||||
*buf = NULL;
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int write_flatfile(const char *path, uint8_t *buf, address_t len)
|
||||
{
|
||||
FILE *out;
|
||||
char *fullpath = expand_tilde(path);
|
||||
const char *errmsg = NULL;
|
||||
|
||||
if (!fullpath)
|
||||
return -1;
|
||||
|
||||
out = fopen(fullpath, "wb");
|
||||
free(fullpath);
|
||||
|
||||
if (!out) {
|
||||
printc_err("%s: %s\n", path, last_error());
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (fwrite(buf, len, 1, out) != 1)
|
||||
errmsg = last_error();
|
||||
|
||||
if (fclose(out) != 0 && !errmsg)
|
||||
errmsg = last_error();
|
||||
|
||||
if (errmsg) {
|
||||
printc_err("%s: failed to write: %s\n", path, errmsg);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int do_flatfile(enum operation op, const char *path, address_t addr, address_t len)
|
||||
{
|
||||
uint8_t *in_buf = NULL;
|
||||
uint8_t *out_buf = NULL;
|
||||
|
||||
int ret = -1;
|
||||
|
||||
if (op == LOAD || op == VERIFY) {
|
||||
ret = read_flatfile(path, &in_buf, &len);
|
||||
if (ret != 0)
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (device_ctl(DEVICE_CTL_HALT) < 0)
|
||||
goto out;
|
||||
|
||||
if (op == LOAD) {
|
||||
if (device_writemem(addr, in_buf, len) != 0)
|
||||
goto out;
|
||||
} else {
|
||||
out_buf = malloc(len);
|
||||
if (!out_buf) {
|
||||
printc_err("flatfile: can't allocate memory\n");
|
||||
goto out;
|
||||
}
|
||||
if (device_readmem(addr, out_buf, len) != 0)
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (device_ctl(DEVICE_CTL_RESET) < 0)
|
||||
printc_err("warning: flatfile: "
|
||||
"failed to reset after programming\n");
|
||||
|
||||
if (op == VERIFY) {
|
||||
int i;
|
||||
for (i = 0; i < len; i++) {
|
||||
if (out_buf[i] != in_buf[i]) {
|
||||
printc("\x1b[1mERROR:\x1b[0m "
|
||||
"mismatch at %04x (read %02x, "
|
||||
"expected %02x)\n",
|
||||
addr + i,
|
||||
out_buf[i], in_buf[i]);
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
ret = 0;
|
||||
} else if (op == SAVE) {
|
||||
ret = write_flatfile(path, out_buf, len);
|
||||
}
|
||||
|
||||
if (ret == 0)
|
||||
printc("Done, %d bytes total\n", len);
|
||||
|
||||
out:
|
||||
free(in_buf);
|
||||
free(out_buf);
|
||||
return ret;
|
||||
}
|
||||
|
||||
int cmd_load_raw(char **arg)
|
||||
{
|
||||
const char *path, *addr_text;
|
||||
address_t addr;
|
||||
|
||||
path = get_arg(arg);
|
||||
if (!path) {
|
||||
printc_err("load_raw: need file name argument\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
addr_text = get_arg(arg);
|
||||
if (!addr_text) {
|
||||
printc_err("load_raw: need flash address argument\n");
|
||||
return -1;
|
||||
} else if (expr_eval(addr_text, &addr) < 0) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
return do_flatfile(LOAD, path, addr, 0);
|
||||
}
|
||||
|
||||
int cmd_verify_raw(char **arg)
|
||||
{
|
||||
const char *path, *addr_text;
|
||||
address_t addr;
|
||||
|
||||
path = get_arg(arg);
|
||||
if (!path) {
|
||||
printc_err("verify_raw: need file name argument\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
addr_text = get_arg(arg);
|
||||
if (!addr_text) {
|
||||
printc_err("verify_raw: need flash address argument\n");
|
||||
return -1;
|
||||
} else if (expr_eval(addr_text, &addr) < 0) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
return do_flatfile(VERIFY, path, addr, 0);
|
||||
}
|
||||
|
||||
int cmd_save_raw(char **arg)
|
||||
{
|
||||
const char *addr_text, *len_text, *path;
|
||||
address_t addr, len;
|
||||
|
||||
addr_text = get_arg(arg);
|
||||
if (!addr_text) {
|
||||
printc_err("save_raw: need flash address argument\n");
|
||||
return -1;
|
||||
} else if (expr_eval(addr_text, &addr) < 0) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
len_text = get_arg(arg);
|
||||
if (!len_text) {
|
||||
printc_err("save_raw: need length argument\n");
|
||||
return -1;
|
||||
} else if (expr_eval(len_text, &len) < 0)
|
||||
return -1;
|
||||
|
||||
path = get_arg(arg);
|
||||
if (!path) {
|
||||
printc_err("save_raw: need file name argument\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
return do_flatfile(SAVE, path, addr, len);
|
||||
}
|
|
@ -0,0 +1,27 @@
|
|||
/* MSPDebug - debugging tool for MSP430 MCUs
|
||||
* Copyright (C) 2009-2012 Daniel Beer
|
||||
* Copyright (C) 2012 Ingo van Lil
|
||||
*
|
||||
* 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 FLATFILE_H_
|
||||
#define FLATFILE_H_
|
||||
|
||||
int cmd_load_raw(char **arg);
|
||||
int cmd_verify_raw(char **arg);
|
||||
int cmd_save_raw(char **arg);
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue