/* MSPDebug - debugging tool for the eZ430 * Copyright (C) 2009, 2010 Daniel Beer * Copyright (C) 2010 Andrew Armenia * * 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 #include #include #include #include #include #include "flash_bsl.h" #include "util.h" #include "output.h" #include "fet_error.h" #include "sport.h" #include "bsllib.h" struct flash_bsl_device { struct device base; sport_t serial_fd; int long_password; const struct device_args *args; const char *seq; }; #define MAX_BLOCK 256 /* This should be at least MAX_BLOCK + 4 */ #define MAX_PACKET 512 /* adapted from TI's published BSL source code */ #define CRC_INIT 0xffff static uint16_t crc_ccitt(const uint8_t *data, int len) { uint16_t crc = CRC_INIT; uint16_t temp; int i; for (i = 0; i < len; ++i) { temp = ((crc >> 8) ^ data[i]) & 0xff; temp ^= (temp >> 4); crc = (crc << 8) ^ (temp << 12) ^ (temp << 5) ^ temp; } return crc; } static void crc_selftest(void) { /* These test vectors are from page 30 of TI doc SLAU319A */ uint16_t crc_expected = 0x5590; uint16_t crc_actual = crc_ccitt((uint8_t *)"\x52\x02", 2); if (crc_expected != crc_actual) { printc_err("flash_bsl: CRC malfunction " "(expected 0x%04x got 0x%04x)\n", crc_expected, crc_actual); } crc_expected = 0x121d; crc_actual = crc_ccitt((uint8_t *)"\x3a\x04\x01", 3); if (crc_expected != crc_actual) { printc_err("flash_bsl: CRC malfunction " "(expected 0x%04x got 0x%04x)\n", crc_expected, crc_actual); } crc_expected = 0x528b; crc_actual = crc_ccitt((uint8_t *)"\x1a", 1); if (crc_expected != crc_actual) { printc_err("flash_bsl: CRC malfunction " "(expected 0x%04x got 0x%04x)\n", crc_expected, crc_actual); } } #define RX_DATA_BLOCK 0x10 #define RX_DATA_BLOCK_FAST 0x1b #define RX_PASSWORD 0x11 #define ERASE_SEGMENT 0x12 #define UNLOCK_LOCK_INFO 0x13 #define MASS_ERASE 0x15 #define CRC_CHECK 0x16 #define LOAD_PC 0x17 #define TX_DATA_BLOCK 0x18 #define TX_BSL_VERSION 0x19 #define TX_BUFFER_SIZE 0x1a static int flash_bsl_send(struct flash_bsl_device *dev, const uint8_t *data, int len) { uint16_t crc; uint8_t cmd_buf[MAX_PACKET + 5]; uint8_t response; #if defined(FLASH_BSL_VERBOSE) debug_hexdump("flash_bsl: sending", data, len); #endif crc = crc_ccitt(data, len); if (len > MAX_PACKET) { printc_err("flash_bsl: attempted to transmit " "long packet (len=%d)\n", len); return -1; } cmd_buf[0] = 0x80; cmd_buf[1] = (len & 0xff); cmd_buf[2] = (len >> 8) & 0xff; memcpy(cmd_buf + 3, data, len); cmd_buf[len + 3] = (crc & 0xff); cmd_buf[len + 4] = (crc >> 8) & 0xff; if (sport_write_all(dev->serial_fd, cmd_buf, len + 5) < 0) { printc_err("flash_bsl: serial write failed: %s\n", last_error()); return -1; } if (sport_read_all(dev->serial_fd, &response, 1) < 0) { printc_err("flash_bsl: serial read failed: %s\n", last_error()); return -1; } if (response != 0) { switch (response) { case 0x51: printc_err("flash_bsl: BSL reports incorrect " "packet header\n"); break; case 0x52: printc_err("flash_bsl: BSL reports checksum " "incorrect\n"); break; case 0x53: printc_err("flash_bsl: BSL got zero-size packet\n"); break; case 0x54: printc_err("flash_bsl: BSL receive buffer " "overflowed\n"); break; case 0x55: printc_err("flash_bsl: (known-)unknown error\n"); break; case 0x56: printc_err("flash_bsl: unknown baud rate\n"); break; default: printc_err("flash_bsl: unknown unknown error\n"); break; } return -1; } return 0; } static int flash_bsl_recv(struct flash_bsl_device *dev, uint8_t *recv_buf, int buf_len) { uint8_t header[3]; uint8_t crc_bytes[2]; uint16_t recv_len; uint16_t crc_value; if (sport_read_all(dev->serial_fd, header, 3) < 0) { printc_err("flash_bsl: read response failed: %s\n", last_error()); return -1; } if (header[0] != 0x80) { printc_err("flash_bsl: incorrect response header received\n"); return -1; } recv_len = header[2]; recv_len <<= 8; recv_len |= header[1]; #if defined(FLASH_BSL_VERBOSE) printc_dbg("flash_bsl: incoming message length %d\n", recv_len); #endif if (recv_len > buf_len) { printc_err("flash_bsl: insufficient buffer to receive data\n"); return -1; } if (sport_read_all(dev->serial_fd, recv_buf, recv_len) < 0) { perror("receive message"); printc_err("flash_bsl: error receiving message\n"); return -1; } if (sport_read_all(dev->serial_fd, crc_bytes, 2) < 0) { perror("receive message CRC"); printc_err("flash_bsl: error receiving message CRC\n"); return -1; } crc_value = crc_bytes[1]; crc_value <<= 8; crc_value |= crc_bytes[0]; if (crc_ccitt(recv_buf, recv_len) != crc_value) { printc_err("flash_bsl: received message with bad CRC\n"); return -1; } #if defined(FLASH_BSL_VERBOSE) debug_hexdump("received message", recv_buf, recv_len); #endif delay_ms(10); return recv_len; } static void flash_bsl_perror(uint8_t code) { switch (code) { case 0x00: printc_err("flash_bsl: success\n"); break; case 0x01: printc_err("flash_bsl: FLASH verify failed\n"); break; case 0x02: printc_err("flash_bsl: FLASH operation failed\n"); break; case 0x03: printc_err("flash_bsl: voltage not constant during program\n"); break; case 0x04: printc_err("flash_bsl: BSL is locked\n"); break; case 0x05: printc_err("flash_bsl: incorrect password\n"); break; case 0x06: printc_err("flash_bsl: attempted byte write to FLASH\n"); break; case 0x07: printc_err("flash_bsl: unrecognized command\n"); break; case 0x08: printc_err("flash_bsl: command was too long\n"); break; default: printc_err("flash_bsl: unknown status message\n"); break; } } static int flash_bsl_readmem(device_t dev_base, address_t addr, uint8_t *mem, address_t len) { struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base; uint8_t recv_buf[MAX_BLOCK*2]; uint8_t send_buf[64]; uint16_t read_size; int ret; if (addr > 0xfffff || addr + len > 0x100000) { printc_err("flash_bsl: read exceeds possible range\n"); return -1; } while (len > 0) { if (len > MAX_BLOCK) { read_size = MAX_BLOCK; } else { read_size = len; } /* build command */ send_buf[0] = TX_DATA_BLOCK; /* command: transmit data block */ send_buf[1] = addr & 0xff; send_buf[2] = (addr >> 8) & 0xff; send_buf[3] = (addr >> 16) & 0xff; send_buf[4] = read_size & 0xff; send_buf[5] = (read_size >> 8) & 0xff; if (flash_bsl_send(dev, send_buf, 6) < 0) { printc_err("flash_bsl readmem: send failed\n"); return -1; } ret = flash_bsl_recv(dev, recv_buf, read_size + 1); if (ret < 0) { printc_err("flash_bsl readmem: receive failed\n"); return -1; } else if (ret < read_size) { printc_err("flash_bsl readmem: warning: not all requested data received\n"); } if (recv_buf[0] == 0x3a) { memcpy(mem, recv_buf + 1, ret - 1); addr += ret - 1; len -= ret - 1; mem += ret - 1; } else if (recv_buf[0] == 0x3b) { flash_bsl_perror(recv_buf[1]); } else { printc_err("flash_bsl readmem: invalid response\n"); return -1; } } return 0; } static int flash_bsl_erase(device_t dev_base, device_erase_type_t type, address_t addr) { struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base; uint8_t erase_cmd[4]; uint8_t response_buffer[16]; int ret; if (type == DEVICE_ERASE_ALL) { printc_err("flash_bsl_erase: simultaneous code/info erase not supported\n"); return -1; } else if (type == DEVICE_ERASE_MAIN) { erase_cmd[0] = MASS_ERASE; if (flash_bsl_send(dev, erase_cmd, 1) < 0) { printc_err("flash_bsl_erase: failed to send erase command\n"); return -1; } } else if (type == DEVICE_ERASE_SEGMENT) { erase_cmd[0] = ERASE_SEGMENT; erase_cmd[1] = addr & 0xff; erase_cmd[2] = (addr >> 8) & 0xff; erase_cmd[3] = (addr >> 16) & 0xff; if (flash_bsl_send(dev, erase_cmd, 4) < 0) { printc_err("flash_bsl_erase: failed to send erase command\n"); return -1; } } else { printc_err("flash_bsl_erase: unsupported erase type\n"); return -1; } ret = flash_bsl_recv(dev, response_buffer, sizeof(response_buffer)); if (ret < 2) { printc_err("flash_bsl_erase: no response\n"); return -1; } if (response_buffer[0] != 0x3b) { printc_err("flash_bsl_erase: incorrect response\n"); return -1; } if (response_buffer[1] != 0) { flash_bsl_perror(response_buffer[1]); printc_err("flash_bsl_erase: erase failed\n"); return -1; } else { #if defined(FLASH_BSL_VERBOSE) printc_dbg("flash_bsl_erase: success\n"); #endif } return 0; } static int flash_bsl_unlock(struct flash_bsl_device *dev, const uint8_t *password) { /* * after erase, the password will be 0xff * (16 or 32) * (an empty interrupt vector table) */ uint8_t rx_password_cmd[33] = "\x11" "\xff\xff\xff\xff\xff\xff\xff\xff" "\xff\xff\xff\xff\xff\xff\xff\xff" "\xff\xff\xff\xff\xff\xff\xff\xff" "\xff\xff\xff\xff\xff\xff\xff\xff"; uint8_t response_buffer[16]; int ret; memcpy(rx_password_cmd + 1, password, 32); /* mass erase - this might wipe Information Memory on some devices */ /* (according to the documentation it should not) */ if (flash_bsl_erase((device_t)dev, DEVICE_ERASE_MAIN, 0) < 0) { printc_err("flash_bsl_unlock: warning: erase failed\n"); } /* send password (which is now erased FLASH) */ if (dev->long_password) { #if defined(FLASH_BSL_VERBOSE) printc_dbg("flash_bsl_unlock: using long password\n"); #endif } if (flash_bsl_send(dev, rx_password_cmd, dev->long_password ? 33 : 17) < 0) { printc_err("flash_bsl_unlock: send password failed\n"); return -1; } ret = flash_bsl_recv(dev, response_buffer, sizeof(response_buffer)); if (ret < 2) { printc_err("flash_bsl_unlock: error receiving password response\n"); return -1; } if (response_buffer[0] != 0x3b) { printc_err("flash_bsl_unlock: received invalid password response\n"); return -1; } if (response_buffer[1] != 0x00) { flash_bsl_perror(response_buffer[1]); printc_err("flash_bsl_unlock: password error\n"); return -1; } return 0; } static int flash_bsl_ctl(device_t dev_base, device_ctl_t type) { (void)dev_base; switch (type) { case DEVICE_CTL_HALT: /* Ignore halt requests */ return 0; case DEVICE_CTL_RESET: /* Ignore reset requests */ return 0; default: printc_err("flash_bsl: CPU control is not possible\n"); } return -1; } static device_status_t flash_bsl_poll(device_t dev_base) { (void)dev_base; return DEVICE_STATUS_HALTED; } static int flash_bsl_getregs(device_t dev_base, address_t *regs) { (void)dev_base; (void)regs; printc_err("flash_bsl: register fetch is not implemented\n"); return -1; } static int flash_bsl_setregs(device_t dev_base, const address_t *regs) { (void)dev_base; (void)regs; printc_err("flash_bsl: register store is not implemented\n"); return -1; } static int flash_bsl_writemem(device_t dev_base, address_t addr, const uint8_t *mem, address_t len) { struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base; uint8_t send_buf[2*MAX_BLOCK]; uint8_t recv_buf[16]; uint16_t write_size; int n_recv; if (addr > 0xfffff || addr + len > 0x100000) { printc_err("flash_bsl: write exceeds possible range\n"); return -1; } while (len > 0) { /* compute size of this write operation */ if (len > MAX_BLOCK) { write_size = MAX_BLOCK; } else { write_size = len; } /* build write command */ /* command */ send_buf[0] = RX_DATA_BLOCK; /* address */ send_buf[1] = addr & 0xff; send_buf[2] = (addr >> 8) & 0xff; send_buf[3] = (addr >> 16) & 0xff; /* data */ memcpy(&send_buf[4], mem, write_size); addr += write_size; mem += write_size; len -= write_size; /* send command */ if (flash_bsl_send(dev, send_buf, write_size + 4) < 0) { printc_err("flash_bsl: send failed\n"); return -1; } /* receive and check response */ n_recv = flash_bsl_recv(dev, recv_buf, sizeof(recv_buf)); if (n_recv < 0) { printc_err("flash_bsl write: error occurred receiving response\n"); return -1; } else if (n_recv < 2) { printc_err("flash_bsl write: response too short\n"); return -1; } else if (recv_buf[0] != 0x3b) { printc_err("flash_bsl write: invalid response received\n"); return -1; } else if (recv_buf[1] != 0x00) { printc_err("flash_bsl write: BSL reported write error: "); flash_bsl_perror(recv_buf[1]); return -1; } /* else success! */ } return 0; } static void flash_bsl_destroy(device_t dev_base) { struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base; if ( dev->args->bsl_gpio_used ) { bsllib_seq_do_gpio(dev->args->bsl_gpio_rts, dev->args->bsl_gpio_dtr, bsllib_seq_next(dev->seq)); } else { bsllib_seq_do(dev->serial_fd, bsllib_seq_next(dev->seq)); } sport_close(dev->serial_fd); free(dev); } static device_t flash_bsl_open(const struct device_args *args) { struct flash_bsl_device *dev; uint8_t tx_bsl_version_command[] = { TX_BSL_VERSION }; uint8_t tx_bsl_version_response[5]; if (!(args->flags & DEVICE_FLAG_TTY)) { printc_err("This driver does not support raw USB access.\n"); return NULL; } dev = malloc(sizeof(*dev)); if (!dev) { pr_error("flash_bsl: can't allocate memory"); return NULL; } crc_selftest( ); memset(dev, 0, sizeof(*dev)); dev->base.type = &device_flash_bsl; dev->args = args; dev->serial_fd = sport_open(args->path, 9600, SPORT_EVEN_PARITY); if (SPORT_ISERR(dev->serial_fd)) { printc_err("flash_bsl: can't open %s: %s\n", args->path, last_error()); free(dev); return NULL; } dev->seq = args->bsl_entry_seq; if (!dev->seq) dev->seq = "dR,r,R,r,R,D:dR,DR"; dev->long_password = args->flags & DEVICE_FLAG_LONG_PW; /* enter bootloader */ if ( args->bsl_gpio_used ) { if (bsllib_seq_do_gpio(args->bsl_gpio_rts, args->bsl_gpio_dtr, dev->seq) < 0) { printc_err("BSL entry sequence failed\n"); goto fail; } } else { if (bsllib_seq_do(dev->serial_fd, dev->seq) < 0) { printc_err("BSL entry sequence failed\n"); goto fail; } } delay_ms(500); /* unlock device (erase then send password) */ if (flash_bsl_unlock(dev, args->bsl_entry_password) < 0) { goto fail; } if (flash_bsl_send(dev, tx_bsl_version_command, sizeof(tx_bsl_version_command)) < 0) { printc_err("flash_bsl: failed to read BSL version"); goto fail; } if (flash_bsl_recv(dev, tx_bsl_version_response, sizeof(tx_bsl_version_response)) < sizeof(tx_bsl_version_response)) { printc_err("flash_bsl: BSL responded with invalid version"); goto fail; } debug_hexdump("BSL version", tx_bsl_version_response, sizeof(tx_bsl_version_response)); return (device_t)dev; fail: sport_close(dev->serial_fd); free(dev); return NULL; } const struct device_class device_flash_bsl = { .name = "flash-bsl", .help = "TI generic flash-based bootloader via RS-232", .open = flash_bsl_open, .destroy = flash_bsl_destroy, .readmem = flash_bsl_readmem, .writemem = flash_bsl_writemem, .getregs = flash_bsl_getregs, .setregs = flash_bsl_setregs, .ctl = flash_bsl_ctl, .poll = flash_bsl_poll, .erase = flash_bsl_erase, .getconfigfuses = NULL };