/* * fw_update.c * * Update bootloader for main firmware. * * Written & released by Keir Fraser * * This is free and unencumbered software released into the public domain. * See the file COPYING for more details, or visit . */ #if STM32F == 1 /* 8kB-64kB (56kB total) */ #define FIRMWARE_START 0x08002000 #define FIRMWARE_END 0x08010000 #elif STM32F == 7 /* 16kB-64KB (48kB total) */ #define FIRMWARE_START 0x08004000 #define FIRMWARE_END 0x08010000 #endif int EXC_reset(void) __attribute__((alias("main"))); static struct { time_t time; bool_t state; } blink; static enum { ST_inactive, ST_command_wait, ST_update, } state = ST_inactive; static uint8_t u_buf[2048] aligned(4); static uint32_t u_prod; static bool_t upd_strapped; struct gw_info gw_info = { /* Max Index == 0 signals that this is the Bootloader. */ .max_index = 0, .max_cmd = CMD_MAX, .hw_model = STM32F }; static void blink_init(void) { blink.time = time_now(); blink.state = FALSE; act_led(FALSE); } static void update_reset(void) { blink_init(); state = ST_inactive; } static void update_configure(void) { blink_init(); state = ST_command_wait; u_prod = 0; } const struct usb_class_ops usb_cdc_acm_ops = { .reset = update_reset, .configure = update_configure }; static void end_command(void *ack, unsigned int ack_len) { if (state == ST_command_wait) blink_init(); usb_write(EP_TX, ack, ack_len); u_prod = 0; } static struct { uint32_t len; uint32_t cur; } update; static void erase_old_firmware(void) { uint32_t p; for (p = FIRMWARE_START; p < FIRMWARE_END; p += FLASH_PAGE_SIZE) fpec_page_erase(p); } static void update_prep(uint32_t len) { fpec_init(); erase_old_firmware(); state = ST_update; update.cur = 0; update.len = len; printk("Update: %u bytes\n", len); } static void update_continue(void) { int len; if ((len = ep_rx_ready(EP_RX)) >= 0) { usb_read(EP_RX, &u_buf[u_prod], len); u_prod += len; } if ((len = u_prod) >= 2) { int nr = len & ~1; fpec_write(u_buf, nr, FIRMWARE_START + update.cur); update.cur += nr; u_prod -= nr; memcpy(u_buf, &u_buf[nr], u_prod); } if ((update.cur >= update.len) && ep_tx_ready(EP_TX)) { uint16_t crc = crc16_ccitt((void *)FIRMWARE_START, update.len, 0xffff); printk("Final CRC: %04x (%s)\n", crc, crc ? "FAIL" : "OK"); u_buf[0] = !!crc; state = ST_command_wait; end_command(u_buf, 1); if (crc) erase_old_firmware(); } } static void process_command(void) { uint8_t cmd = u_buf[0]; uint8_t len = u_buf[1]; uint8_t resp_sz = 2; act_led(TRUE); switch (cmd) { case CMD_GET_INFO: { uint8_t idx = u_buf[2]; if ((len != 3) || (idx != 0)) goto bad_command; memset(&u_buf[2], 0, 32); gw_info.fw_major = fw_major; gw_info.fw_minor = fw_minor; /* sample_freq is used as flags: bit 0 indicates if we entered * the bootloader because the update jumper is strapped. */ gw_info.sample_freq = upd_strapped; memcpy(&u_buf[2], &gw_info, sizeof(gw_info)); resp_sz += 32; break; } case CMD_UPDATE: { uint32_t u_len = *(uint32_t *)&u_buf[2]; if (len != 6) goto bad_command; if (u_len & 3) goto bad_command; update_prep(u_len); break; } #if STM32F == 7 case CMD_SWITCH_FW_MODE: { uint8_t mode = u_buf[2]; if ((len != 3) || (mode & ~1)) goto bad_command; if (mode == FW_MODE_NORMAL) { usb_deinit(); delay_ms(500); system_reset(); } break; } #endif default: goto bad_command; } u_buf[1] = ACK_OKAY; out: end_command(u_buf, resp_sz); return; bad_command: u_buf[1] = ACK_BAD_COMMAND; goto out; } static void update_process(void) { int len; time_t now = time_now(); switch (state) { case ST_command_wait: if (time_diff(blink.time, now) > time_ms(200)) { blink.time = now; blink.state ^= 1; act_led(blink.state); } len = ep_rx_ready(EP_RX); if ((len >= 0) && (len < (sizeof(u_buf)-u_prod))) { usb_read(EP_RX, &u_buf[u_prod], len); u_prod += len; } if ((u_prod >= 2) && (u_prod >= u_buf[1]) && ep_tx_ready(EP_TX)) { process_command(); } break; case ST_update: update_continue(); break; default: break; } } #if STM32F == 1 static bool_t enter_bootloader(void) { /* Turn on AFIO and GPIOA clocks. */ rcc->apb2enr = RCC_APB2ENR_IOPAEN | RCC_APB2ENR_AFIOEN; /* Turn off serial-wire JTAG and reclaim the GPIOs. */ afio->mapr = AFIO_MAPR_SWJ_CFG_DISABLED; /* Enable GPIOA, set all pins as floating, except PA14 = weak pull-up. */ gpioa->odr = 0xffffu; gpioa->crh = 0x48444444u; gpioa->crl = 0x44444444u; /* Wait for PA14 to be pulled HIGH. */ cpu_relax(); cpu_relax(); /* Enter update mode only if PA14 (DCLK) is strapped to GND. */ upd_strapped = !gpio_read_pin(gpioa, 14); return upd_strapped; } #elif STM32F == 7 static bool_t check_update_requested(void) { /* Check-and-clear a magic value poked into SRAM1 by the main firmware. */ bool_t match = (*(volatile uint32_t *)0x20010000 == 0xdeadbeef); *(volatile uint32_t *)0x20010000 = 0; return match; } static bool_t check_update_strapped(void) { int i, j; /* Check whether the Serial TX/RX lines (PA9/PA10) are strapped. */ rcc->ahb1enr |= RCC_AHB1ENR_GPIOAEN; (void)rcc->ahb1enr; gpio_configure_pin(gpioa, 9, GPO_pushpull(IOSPD_LOW, HIGH)); gpio_configure_pin(gpioa, 10, GPI_pull_up); for (i = 0; i < 10; i++) { gpio_write_pin(gpioa, 9, i&1); for (j = 0; j < 10000; j++) cpu_relax(); if (gpio_read_pin(gpioa, 10) != (i&1)) return FALSE; } return TRUE; } static bool_t enter_bootloader(void) { bool_t upd_requested = check_update_requested(); upd_strapped = check_update_strapped(); return upd_requested || upd_strapped; } #endif int main(void) { /* Relocate DATA. Initialise BSS. */ if (_sdat != _ldat) memcpy(_sdat, _ldat, _edat-_sdat); memset(_sbss, 0, _ebss-_sbss); if (!enter_bootloader()) { /* Nope, so jump straight at the main firmware. */ uint32_t sp = *(uint32_t *)FIRMWARE_START; uint32_t pc = *(uint32_t *)(FIRMWARE_START + 4); if (sp != ~0u) { /* only if firmware is apparently not erased */ asm volatile ( "mov sp,%0 ; blx %1" :: "r" (sp), "r" (pc)); } } stm32_init(); time_init(); console_init(); console_crash_on_input(); board_init(); printk("\n** Greaseweazle Update Bootloader v%u.%u\n", fw_major, fw_minor); printk("** Keir Fraser \n"); printk("** https://github.com/keirf/Greaseweazle\n\n"); usb_init(); for (;;) { usb_process(); update_process(); } return 0; } /* * Local variables: * mode: C * c-file-style: "Linux" * c-basic-offset: 4 * tab-width: 4 * indent-tabs-mode: nil * End: */