sksp2024-mcu/ucw-stm32lib/lib/modbus-bootloader.c

428 lines
10 KiB
C

/*
* MODBUS Bootloader
*
* (c) 2023 Martin Mareš <mj@ucw.cz>
*
* Based on example code from the libopencm3 project, which is
* Copyright (C) 2010 Gareth McMullin <gareth@blacksphere.co.nz>
*
* Licensed under the GNU LGPL v3 or any later version.
*/
#include "util.h"
#include "modbus.h"
#include "modbus-bootloader-proto.h"
#include <libopencm3/cm3/cortex.h>
#include <libopencm3/cm3/nvic.h>
#include <libopencm3/cm3/scb.h>
#include <libopencm3/cm3/systick.h>
#include <libopencm3/stm32/rcc.h>
#include <libopencm3/stm32/crc.h>
#include <libopencm3/stm32/gpio.h>
#include <libopencm3/stm32/flash.h>
#include <libopencm3/stm32/usart.h>
#include <libopencm3/stm32/desig.h>
#include <string.h>
#ifdef BOOTLOADER_DEBUG
#define DEBUG(x...) debug_printf(x)
#else
#define DEBUG(x...) do { } while (0)
#endif
// Offsets to firmware header fields (see tools/dfu-sign.c)
#define HDR_LENGTH 0x1c
#define HDR_FLASH_IN_PROGRESS 0x20
// Block size should be equal to erase block of the flash memory
#define BLOCK_SIZE 1024
static byte current_block[BLOCK_SIZE];
static uint current_block_number;
static uint timeout;
#define DEFAULT_TIMEOUT 5000 // ms
static char usb_serial_number[13];
static uint flash_size_kb;
static uint status;
static inline u32 get_u32(u32 addr)
{
return *(u32*)addr;
}
static inline u16 get_u16(u32 addr)
{
return *(u16*)addr;
}
static bool verify_firmware(void)
{
u32 len = get_u32(BOOTLOADER_APP_START + HDR_LENGTH);
u16 flash_in_progress = get_u16(BOOTLOADER_APP_START + HDR_FLASH_IN_PROGRESS);
// Just to be sure
len = MIN(len, flash_size_kb * 1024);
crc_reset();
u32 crc = crc_calculate_block((u32 *)BOOTLOADER_APP_START, len/4);
u32 want_crc = get_u32(BOOTLOADER_APP_START + len);
DEBUG("BOOT: fip=%04x crc=%08x/%08x len=%u\n", (uint) flash_in_progress, (uint) crc, (uint) want_crc, (uint) len);
if (flash_in_progress || crc != want_crc) {
DEBUG("BOOT: Bad firmware\n");
return 0;
}
return 1;
}
static bool flash_block(void)
{
if (current_block_number >= flash_size_kb * 1024 / BLOCK_SIZE) {
DEBUG("BOOT: Bad block nr\n");
return false;
}
if (current_block_number == 0) {
// The "flash in progress" word is programmed as 0xffff first and reset later
*(u16*)(current_block + HDR_FLASH_IN_PROGRESS) = 0xffff;
}
u32 baseaddr = BOOTLOADER_APP_START + current_block_number * BLOCK_SIZE;
DEBUG("BOOT: Block %u -> %08x\n", current_block_number, (uint) baseaddr);
flash_unlock();
flash_erase_page(baseaddr);
for (uint i = 0; i < BLOCK_SIZE; i += 2)
flash_program_half_word(baseaddr + i, *(u16*)(current_block + i));
flash_lock();
for (uint i = 0; i < BLOCK_SIZE; i++) {
if (*(byte *)(baseaddr + i) != current_block[i]) {
DEBUG("BOOT: Verification failed\n");
return false;
}
}
return true;
}
static bool flash_end(void)
{
flash_unlock();
flash_program_half_word(BOOTLOADER_APP_START + 0x20, 0);
flash_lock();
return verify_firmware();
}
/*
* This is a modified version of rcc_clock_setup_in_hsi_out_48mhz(),
* which properly turns off the PLL before setting its parameters.
*/
static void my_rcc_clock_setup_in_hsi_out_48mhz(void)
{
/* Enable internal high-speed oscillator. */
rcc_osc_on(RCC_HSI);
rcc_wait_for_osc_ready(RCC_HSI);
/* Select HSI as SYSCLK source. */
rcc_set_sysclk_source(RCC_CFGR_SW_SYSCLKSEL_HSICLK);
// XXX: Disable PLL
rcc_osc_off(RCC_PLL);
/*
* Set prescalers for AHB, ADC, ABP1, ABP2.
* Do this before touching the PLL (TODO: why?).
*/
rcc_set_hpre(RCC_CFGR_HPRE_SYSCLK_NODIV); /*Set.48MHz Max.72MHz */
rcc_set_adcpre(RCC_CFGR_ADCPRE_PCLK2_DIV8); /*Set. 6MHz Max.14MHz */
rcc_set_ppre1(RCC_CFGR_PPRE1_HCLK_DIV2); /*Set.24MHz Max.36MHz */
rcc_set_ppre2(RCC_CFGR_PPRE2_HCLK_NODIV); /*Set.48MHz Max.72MHz */
rcc_set_usbpre(RCC_CFGR_USBPRE_PLL_CLK_NODIV); /*Set.48MHz Max.48MHz */
/*
* Sysclk runs with 48MHz -> 1 waitstates.
* 0WS from 0-24MHz
* 1WS from 24-48MHz
* 2WS from 48-72MHz
*/
flash_set_ws(FLASH_ACR_LATENCY_1WS);
/*
* Set the PLL multiplication factor to 12.
* 8MHz (internal) * 12 (multiplier) / 2 (PLLSRC_HSI_CLK_DIV2) = 48MHz
*/
rcc_set_pll_multiplication_factor(RCC_CFGR_PLLMUL_PLL_CLK_MUL12);
/* Select HSI/2 as PLL source. */
rcc_set_pll_source(RCC_CFGR_PLLSRC_HSI_CLK_DIV2);
/* Enable PLL oscillator and wait for it to stabilize. */
rcc_osc_on(RCC_PLL);
rcc_wait_for_osc_ready(RCC_PLL);
/* Select PLL as SYSCLK source. */
rcc_set_sysclk_source(RCC_CFGR_SW_SYSCLKSEL_PLLCLK);
/* Set the peripheral clock frequencies used */
rcc_ahb_frequency = 48000000;
rcc_apb1_frequency = 24000000;
rcc_apb2_frequency = 48000000;
}
static void clock_plain_hsi(void)
{
// Select HSI as SYSCLK source
rcc_set_sysclk_source(RCC_CFGR_SW_SYSCLKSEL_HSICLK);
// Disable PLL
rcc_osc_off(RCC_PLL);
// Set prescalers for AHB, ADC, ABP1, ABP2, USB to defaults
rcc_set_hpre(RCC_CFGR_HPRE_SYSCLK_NODIV);
rcc_set_adcpre(RCC_CFGR_ADCPRE_PCLK2_DIV2);
rcc_set_ppre1(RCC_CFGR_PPRE1_HCLK_NODIV);
rcc_set_ppre2(RCC_CFGR_PPRE2_HCLK_NODIV);
rcc_set_usbpre(RCC_CFGR_USBPRE_PLL_VCO_CLK_DIV3);
}
static void reset_peripherals(void)
{
// Turn off clock to all peripherals and reset them
RCC_AHBENR = 0x00000014;
RCC_APB1ENR = 0;
RCC_APB2ENR = 0;
RCC_APB1RSTR = 0x22fec9ff;
RCC_APB2RSTR = 0x0038fffd;
RCC_APB1RSTR = 0;
RCC_APB2RSTR = 0;
}
static void configure_hardware(void)
{
rcc_periph_clock_enable(RCC_GPIOA);
rcc_periph_clock_enable(RCC_GPIOB);
rcc_periph_clock_enable(RCC_GPIOC);
rcc_periph_clock_enable(RCC_CRC);
#ifdef DEBUG_USART
#if DEBUG_USART == USART1
rcc_periph_clock_enable(RCC_USART1);
gpio_set_mode(GPIOA, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_ALTFN_PUSHPULL, GPIO9);
#elif DEBUG_USART == USART2
rcc_periph_clock_enable(RCC_USART2);
gpio_set_mode(GPIOA, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_ALTFN_PUSHPULL, GPIO2);
#elif DEBUG_USART == USART3
rcc_periph_clock_enable(RCC_USART3);
gpio_set_mode(GPIOB, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_ALTFN_PUSHPULL, GPIO10);
#else
#error "Unknown USART for debugging"
#endif
usart_set_baudrate(DEBUG_USART, 115200);
usart_set_databits(DEBUG_USART, 8);
usart_set_stopbits(DEBUG_USART, USART_STOPBITS_1);
usart_set_mode(DEBUG_USART, USART_MODE_TX);
usart_set_parity(DEBUG_USART, USART_PARITY_NONE);
usart_set_flow_control(DEBUG_USART, USART_FLOWCONTROL_NONE);
usart_enable(DEBUG_USART);
#endif
#ifdef DEBUG_LED_BLUEPILL
// BluePill LED
gpio_set_mode(GPIOC, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_PUSHPULL, GPIO13);
debug_led(1);
#endif
// Systick: set to overflow in 1 ms, will use only the overflow flag, no interrupts
systick_set_frequency(1000, CPU_CLOCK_MHZ * 1000000);
systick_clear();
systick_counter_enable();
}
/*** Modbus callbacks ***/
bool modbus_check_discrete_input(u16 addr UNUSED)
{
return false;
}
bool modbus_get_discrete_input(u16 addr UNUSED)
{
return false;
}
bool modbus_check_coil(u16 addr UNUSED)
{
return false;
}
bool modbus_get_coil(u16 addr UNUSED)
{
return false;
}
void modbus_set_coil(u16 addr UNUSED, bool value UNUSED)
{
}
bool modbus_check_input_register(u16 addr)
{
return addr >= BL_INPUT_MAGIC_HI && addr < BL_INPUT_MAX;
}
u16 modbus_get_input_register(u16 addr)
{
switch (addr) {
case BL_INPUT_MAGIC_HI:
return BL_MAGIC_HI;
case BL_INPUT_MAGIC_LO:
return BL_MAGIC_LO;
case BL_INPUT_LOADER_VERSION:
return BL_LOADER_VERSION;
case BL_INPUT_STATUS:
return status;
case BL_INPUT_VENDOR_ID:
return BOOTLOADER_VENDOR_ID;
case BL_INPUT_DEVICE_ID:
return BOOTLOADER_DEVICE_ID;
case BL_INPUT_BLOCK_SIZE:
return BLOCK_SIZE;
case BL_INPUT_FLASH_SIZE:
return flash_size_kb;
default:
return get_u16_le((byte *)usb_serial_number + 2*(addr - BL_INPUT_SERIAL_NUMBER));
}
}
bool modbus_check_holding_register(u16 addr)
{
return addr >= BL_HOLD_COMMAND && addr < BL_HOLD_MAX
|| addr >= BL_HOLD_BLOCK_DATA && addr < BL_HOLD_BLOCK_DATA + BLOCK_SIZE / 2;
}
u16 modbus_get_holding_register(u16 addr UNUSED)
{
// Reading of holding registers is not supported
return 0;
}
static uint bl_command(uint value)
{
switch (value) {
case BL_COMMAND_BOOT:
if (status == BL_STATUS_READY)
return BL_STATUS_BOOTING;
return BL_STATUS_ERROR;
case BL_COMMAND_FLASH_START:
return BL_STATUS_FLASHING;
case BL_COMMAND_FLASH_END:
if (flash_end())
return BL_STATUS_READY;
else
return BL_STATUS_CORRUPTED;
case BL_COMMAND_FLASH_BLOCK:
if (flash_block())
return BL_STATUS_FLASHING;
else
return BL_STATUS_ERROR;
default:
return BL_STATUS_ERROR;
}
}
void modbus_set_holding_register(u16 addr, u16 value)
{
switch (addr) {
case BL_HOLD_COMMAND:
status = bl_command(value);
DEBUG("BOOT: cmd=%u status=%u\n", value, status);
timeout = DEFAULT_TIMEOUT;
break;
case BL_HOLD_BLOCK_NUMBER:
current_block_number = value;
break;
default:
put_u16_le(current_block + 2*(addr - BL_HOLD_BLOCK_DATA), value);
}
}
// These should be implemented by board-specific code
// void modbus_ready_hook(void);
// void modbus_frame_start_hook(void);
// const char * const modbus_id_strings[MODBUS_ID_MAX];
static void delay_ms(uint ms)
{
for (uint j=0; j<ms; j++)
while (!systick_get_countflag())
;
}
int main(void)
{
reset_peripherals();
// Flash programming requires running on the internal oscillator
my_rcc_clock_setup_in_hsi_out_48mhz();
configure_hardware();
custom_hw_init();
desig_get_unique_id_as_dfu(usb_serial_number);
flash_size_kb = desig_get_flash_size();
// Allow ST-link to attach before we initialize the rest of hardware
for (int i=0; i<100; i++) {
debug_led_toggle();
delay_ms(20);
}
DEBUG("BOOT: Started (SN %s, fs=%u)\n", usb_serial_number, flash_size_kb);
modbus_init();
DEBUG("BOOT: Ready\n");
debug_led(0);
if (verify_firmware())
status = BL_STATUS_READY;
else
status = BL_STATUS_CORRUPTED;
timeout = DEFAULT_TIMEOUT;
byte led_counter = 0;
while (status != BL_STATUS_READY || timeout) {
modbus_loop();
if (status == BL_STATUS_BOOTING && modbus_is_idle())
break;
if (systick_get_countflag()) {
if (timeout)
timeout--;
if (!(led_counter++ & 0x3f))
debug_led_toggle();
}
}
u32 sp = get_u32(BOOTLOADER_APP_START);
u32 pc = get_u32(BOOTLOADER_APP_START + 4);
DEBUG("BOOT: Start (sp=%08x pc=%08x)\n", (uint) sp, (uint) pc);
#ifdef DEBUG_USART
debug_flush();
#endif
debug_led(0);
reset_peripherals();
clock_plain_hsi();
/* Set vector table base address. */
SCB_VTOR = BOOTLOADER_APP_START;
/* Initialize master stack pointer. */
asm volatile("msr msp, %0"::"g" (sp));
/* Jump to application. */
((void (*)(void)) pc)();
}