Align flash writes to word boundaries.
The FET silently drops flash memory write requests which are not address or length aligned to a word boundary. Based on a patch by Sören Höckner <soeren.hoeckner@gmx.de>.
This commit is contained in:
parent
40fa5de592
commit
479b0b2a9f
3
AUTHORS
3
AUTHORS
|
@ -32,3 +32,6 @@ Robert Spanton <rspanton@zepler.net>:
|
||||||
|
|
||||||
Stephen Kench <stephen@pmssystems.com>
|
Stephen Kench <stephen@pmssystems.com>
|
||||||
* Support for MSP430F47197.
|
* Support for MSP430F47197.
|
||||||
|
|
||||||
|
Sören Höckner <soeren.hoeckner@gmx.de>:
|
||||||
|
* Testing and debugging of flash memory word-alignment bug.
|
||||||
|
|
2
Makefile
2
Makefile
|
@ -58,7 +58,7 @@ 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 \
|
util.o bsl.o sim.o symmap.o gdb.o btree.o rtools.o sym.o devcmd.o \
|
||||||
reader.o vector.o output_util.o expr.o fet_error.o binfile.o \
|
reader.o vector.o output_util.o expr.o fet_error.o binfile.o \
|
||||||
fet_db.o usbutil.o titext.o srec.o device.o coff.o opdb.o output.o \
|
fet_db.o usbutil.o titext.o srec.o device.o coff.o opdb.o output.o \
|
||||||
cmddb.o stdcmd.o
|
cmddb.o stdcmd.o prog.o
|
||||||
$(CC) $(LDFLAGS) $(MACPORTS_LDFLAGS) -o $@ $^ -lusb $(READLINE_LIBS)
|
$(CC) $(LDFLAGS) $(MACPORTS_LDFLAGS) -o $@ $^ -lusb $(READLINE_LIBS)
|
||||||
|
|
||||||
.c.o:
|
.c.o:
|
||||||
|
|
76
devcmd.c
76
devcmd.c
|
@ -29,6 +29,7 @@
|
||||||
#include "reader.h"
|
#include "reader.h"
|
||||||
#include "output_util.h"
|
#include "output_util.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
#include "prog.h"
|
||||||
#include "dis.h"
|
#include "dis.h"
|
||||||
|
|
||||||
int cmd_regs(char **arg)
|
int cmd_regs(char **arg)
|
||||||
|
@ -450,75 +451,10 @@ fail:
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct prog_data {
|
static int cmd_prog_feed(void *user_data, address_t addr,
|
||||||
uint8_t buf[128];
|
const uint8_t *data, int len)
|
||||||
address_t addr;
|
|
||||||
int len;
|
|
||||||
int have_erased;
|
|
||||||
};
|
|
||||||
|
|
||||||
static int prog_flush(struct prog_data *prog)
|
|
||||||
{
|
{
|
||||||
while (prog->len) {
|
return prog_feed((struct prog_data *)user_data, addr, data, len);
|
||||||
int wlen = prog->len;
|
|
||||||
|
|
||||||
if (!prog->have_erased) {
|
|
||||||
printc("Erasing...\n");
|
|
||||||
if (device_default->ctl(device_default,
|
|
||||||
DEVICE_CTL_ERASE) < 0)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
printc("Programming...\n");
|
|
||||||
prog->have_erased = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
printc_dbg("Writing %3d bytes to %04x...\n", wlen, prog->addr);
|
|
||||||
if (device_default->writemem(device_default, prog->addr,
|
|
||||||
prog->buf, wlen) < 0)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
memmove(prog->buf, prog->buf + wlen, prog->len - wlen);
|
|
||||||
prog->len -= wlen;
|
|
||||||
prog->addr += wlen;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int prog_feed(void *user_data,
|
|
||||||
address_t addr, const uint8_t *data, int len)
|
|
||||||
{
|
|
||||||
struct prog_data *prog = (struct prog_data *)user_data;
|
|
||||||
|
|
||||||
/* Flush if this section is discontiguous */
|
|
||||||
if (prog->len && prog->addr + prog->len != addr &&
|
|
||||||
prog_flush(prog) < 0)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
if (!prog->len)
|
|
||||||
prog->addr = addr;
|
|
||||||
|
|
||||||
/* Add the buffer in piece by piece, flushing when it gets
|
|
||||||
* full.
|
|
||||||
*/
|
|
||||||
while (len) {
|
|
||||||
int count = sizeof(prog->buf) - prog->len;
|
|
||||||
|
|
||||||
if (count > len)
|
|
||||||
count = len;
|
|
||||||
|
|
||||||
if (!count) {
|
|
||||||
if (prog_flush(prog) < 0)
|
|
||||||
return -1;
|
|
||||||
} else {
|
|
||||||
memcpy(prog->buf + prog->len, data, count);
|
|
||||||
prog->len += count;
|
|
||||||
data += count;
|
|
||||||
len -= count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int cmd_prog(char **arg)
|
int cmd_prog(char **arg)
|
||||||
|
@ -540,9 +476,9 @@ int cmd_prog(char **arg)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(&prog, 0, sizeof(prog));
|
prog_init(&prog, PROG_WANT_ERASE);
|
||||||
|
|
||||||
if (binfile_extract(in, prog_feed, &prog) < 0) {
|
if (binfile_extract(in, cmd_prog_feed, &prog) < 0) {
|
||||||
fclose(in);
|
fclose(in);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
8
gdb.c
8
gdb.c
|
@ -28,6 +28,7 @@
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include "device.h"
|
#include "device.h"
|
||||||
|
#include "prog.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
#include "opdb.h"
|
#include "opdb.h"
|
||||||
#include "gdb.h"
|
#include "gdb.h"
|
||||||
|
@ -341,6 +342,7 @@ static int write_memory(struct gdb_data *data, char *text)
|
||||||
address_t length, addr;
|
address_t length, addr;
|
||||||
uint8_t buf[MAX_MEM_XFER];
|
uint8_t buf[MAX_MEM_XFER];
|
||||||
int buflen = 0;
|
int buflen = 0;
|
||||||
|
struct prog_data prog;
|
||||||
|
|
||||||
if (!(data_text && length_text)) {
|
if (!(data_text && length_text)) {
|
||||||
printc_err("gdb: malformed memory write request\n");
|
printc_err("gdb: malformed memory write request\n");
|
||||||
|
@ -364,9 +366,9 @@ static int write_memory(struct gdb_data *data, char *text)
|
||||||
return gdb_send(data, "E00");
|
return gdb_send(data, "E00");
|
||||||
}
|
}
|
||||||
|
|
||||||
printc("Writing %d bytes to 0x%04x\n", buflen, addr);
|
prog_init(&prog, 0);
|
||||||
|
if (prog_feed(&prog, addr, buf, buflen) < 0 ||
|
||||||
if (device_default->writemem(device_default, addr, buf, buflen) < 0)
|
prog_flush(&prog) < 0)
|
||||||
return gdb_send(data, "E00");
|
return gdb_send(data, "E00");
|
||||||
|
|
||||||
return gdb_send(data, "OK");
|
return gdb_send(data, "OK");
|
||||||
|
|
|
@ -0,0 +1,105 @@
|
||||||
|
/* MSPDebug - debugging tool for the eZ430
|
||||||
|
* 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 <string.h>
|
||||||
|
#include "device.h"
|
||||||
|
#include "prog.h"
|
||||||
|
#include "output.h"
|
||||||
|
|
||||||
|
void prog_init(struct prog_data *prog, int flags)
|
||||||
|
{
|
||||||
|
memset(prog, 0, sizeof(*prog));
|
||||||
|
prog->flags = flags;
|
||||||
|
}
|
||||||
|
|
||||||
|
int prog_flush(struct prog_data *prog)
|
||||||
|
{
|
||||||
|
if (!prog->len)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!prog->have_erased && (prog->flags & PROG_WANT_ERASE)) {
|
||||||
|
printc("Erasing...\n");
|
||||||
|
if (device_default->ctl(device_default,
|
||||||
|
DEVICE_CTL_ERASE) < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
prog->have_erased = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If writing an odd number of bytes to flash memory, add a
|
||||||
|
* trailing pad byte.
|
||||||
|
*/
|
||||||
|
if (prog->addr + prog->len >= device_default->code_start &&
|
||||||
|
(prog->len & 1)) {
|
||||||
|
printc_dbg("prog: adding trailing pad byte\n");
|
||||||
|
prog->buf[prog->len++] = 0xff;
|
||||||
|
}
|
||||||
|
|
||||||
|
printc_dbg("Writing %3d bytes to %04x...\n", prog->len, prog->addr);
|
||||||
|
if (device_default->writemem(device_default, prog->addr,
|
||||||
|
prog->buf, prog->len) < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
prog->addr += prog->len;
|
||||||
|
prog->len = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int prog_feed(struct prog_data *prog, address_t addr,
|
||||||
|
const uint8_t *data, int len)
|
||||||
|
{
|
||||||
|
/* Flush if this section is discontiguous */
|
||||||
|
if (prog->len && prog->addr + prog->len != addr &&
|
||||||
|
prog_flush(prog) < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (!prog->len) {
|
||||||
|
/* If starting at an odd address, add a leading pad byte. */
|
||||||
|
if (prog->addr + len >= device_default->code_start &&
|
||||||
|
(addr & 1)) {
|
||||||
|
printc_dbg("prog: adding initial pad byte\n");
|
||||||
|
prog->addr = addr - 1;
|
||||||
|
prog->len = 1;
|
||||||
|
prog->buf[0] = 0xff;
|
||||||
|
} else {
|
||||||
|
prog->addr = addr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Add the buffer in piece by piece, flushing when it gets
|
||||||
|
* full.
|
||||||
|
*/
|
||||||
|
while (len) {
|
||||||
|
int count = sizeof(prog->buf) - prog->len;
|
||||||
|
|
||||||
|
if (count > len)
|
||||||
|
count = len;
|
||||||
|
|
||||||
|
if (!count) {
|
||||||
|
if (prog_flush(prog) < 0)
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
memcpy(prog->buf + prog->len, data, count);
|
||||||
|
prog->len += count;
|
||||||
|
data += count;
|
||||||
|
len -= count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
|
@ -0,0 +1,39 @@
|
||||||
|
/* 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
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PROG_H_
|
||||||
|
#define PROG_H_
|
||||||
|
|
||||||
|
#define PROG_BUFSIZE 128
|
||||||
|
|
||||||
|
struct prog_data {
|
||||||
|
uint8_t buf[PROG_BUFSIZE];
|
||||||
|
address_t addr;
|
||||||
|
int len;
|
||||||
|
int flags;
|
||||||
|
int have_erased;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define PROG_WANT_ERASE 0x01
|
||||||
|
|
||||||
|
void prog_init(struct prog_data *data, int flags);
|
||||||
|
int prog_feed(struct prog_data *data, address_t addr,
|
||||||
|
const uint8_t *buffer, int count);
|
||||||
|
int prog_flush(struct prog_data *data);
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue