2025-03-10 20:43:14 +03:00
|
|
|
|
#include <Arduino.h>
|
|
|
|
|
#include <STM32_CAN.h>
|
|
|
|
|
#include "flash.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
STM32_CAN Can(CAN2, DEF);
|
2025-03-19 18:56:00 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
volatile bool fw_update = false;
|
|
|
|
|
volatile uint32_t fw_size = 0;
|
|
|
|
|
volatile uint32_t fw_crc = 0;
|
2025-03-21 14:00:00 +03:00
|
|
|
|
volatile uint32_t jump;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
static FLASH_RECORD flash_record = {0};
|
2025-03-19 18:56:00 +03:00
|
|
|
|
static uint32_t ptr_flash;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
// Прототипы функций
|
|
|
|
|
void jump_to_app();
|
|
|
|
|
void process_can_message(const CAN_message_t &msg);
|
|
|
|
|
void erase_flash_pages();
|
|
|
|
|
bool verify_firmware();
|
|
|
|
|
void send_ack(uint8_t status);
|
2025-03-10 20:43:14 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
void setup() {
|
|
|
|
|
// Инициализация периферии
|
2025-03-19 18:56:00 +03:00
|
|
|
|
Serial.setRx(HARDWARE_SERIAL_RX_PIN);
|
|
|
|
|
Serial.setTx(HARDWARE_SERIAL_TX_PIN);
|
2025-03-17 15:43:34 +03:00
|
|
|
|
Serial.begin(115200);
|
2025-03-19 18:56:00 +03:00
|
|
|
|
Can.begin();
|
|
|
|
|
Can.setBaudRate(1000000);
|
|
|
|
|
TIM_TypeDef *Instance = TIM2;
|
|
|
|
|
HardwareTimer *SendTimer = new HardwareTimer(Instance);
|
|
|
|
|
SendTimer->setOverflow(100, HERTZ_FORMAT); // 50 Hz
|
|
|
|
|
// SendTimer->attachInterrupt(process_can_message);
|
|
|
|
|
SendTimer->resume();
|
|
|
|
|
// Разрешить все ID (маска 0x00000000)
|
2025-03-21 14:00:00 +03:00
|
|
|
|
Can.setFilter(0, 0, STD);
|
2025-03-17 15:43:34 +03:00
|
|
|
|
|
|
|
|
|
// Настройка GPIO
|
|
|
|
|
RCC->AHB1ENR |= RCC_AHB1ENR_GPIOCEN;
|
|
|
|
|
GPIOC->MODER |= GPIO_MODER_MODE10_0 | GPIO_MODER_MODE11_0;
|
|
|
|
|
GPIOC->ODR |= GPIO_ODR_OD11;
|
|
|
|
|
|
|
|
|
|
// Проверка флага обновления
|
2025-03-21 14:00:00 +03:00
|
|
|
|
// erase_sector(6);
|
|
|
|
|
// flash_program_word(FLAG_BOOT,0xDEADBEEF,0);
|
|
|
|
|
// flash_record.data_id = addr_id;
|
|
|
|
|
// flash_record.crc = 0x6933;
|
|
|
|
|
// flash_record.value = 0x69;
|
|
|
|
|
// flash_record.write_ptr_now = SECTOR_6;
|
|
|
|
|
// flash_write(SECTOR_6, &flash_record);
|
2025-03-19 18:56:00 +03:00
|
|
|
|
flash_record = load_params();
|
2025-03-17 15:43:34 +03:00
|
|
|
|
/* Добавить проверку адреса, т.е во время отправки запроса прошивки по CAN
|
|
|
|
|
мы сохраняем как флаг, так и аддрес устройства к которому будет обращатлься во
|
|
|
|
|
время прошивки */
|
2025-03-19 18:56:00 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
if(*(volatile uint32_t*)(FLAG_BOOT) == UPDATE_FLAG) {
|
|
|
|
|
fw_update = true;
|
|
|
|
|
GPIOC->ODR |= GPIO_ODR_OD10; // Индикация обновления
|
|
|
|
|
erase_flash_pages();
|
|
|
|
|
} else {
|
|
|
|
|
jump_to_app();
|
|
|
|
|
}
|
2025-03-10 20:43:14 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
}
|
2025-03-10 20:43:14 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
void loop() {
|
|
|
|
|
if(fw_update) {
|
2025-03-19 18:56:00 +03:00
|
|
|
|
GPIOC->ODR ^= GPIO_ODR_OD10;
|
2025-03-21 14:00:00 +03:00
|
|
|
|
// HAL_Delay(100);
|
2025-03-17 15:43:34 +03:00
|
|
|
|
CAN_message_t msg;
|
2025-03-21 14:00:00 +03:00
|
|
|
|
while(Can.read(msg))
|
2025-03-17 15:43:34 +03:00
|
|
|
|
process_can_message(msg);
|
|
|
|
|
}
|
2025-03-10 20:43:14 +03:00
|
|
|
|
}
|
2025-03-19 18:56:00 +03:00
|
|
|
|
|
2025-03-21 14:00:00 +03:00
|
|
|
|
|
2025-03-10 20:43:14 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
void process_can_message(const CAN_message_t &msg) {
|
|
|
|
|
switch(msg.id) {
|
|
|
|
|
case BOOT_CAN_ID:
|
|
|
|
|
if(msg.buf[0] == 0x01) { // Старт передачи
|
|
|
|
|
fw_size = *(uint32_t*)&msg.buf[1]; //размер прошивки тип 4 байта
|
2025-03-19 18:56:00 +03:00
|
|
|
|
fw_crc = *(uint16_t*)&msg.buf[5]; //crc
|
|
|
|
|
ptr_flash = APP_ADDRESS;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
send_ack(0x01);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case DATA_CAN_ID: // Пакет данных
|
2025-03-19 18:56:00 +03:00
|
|
|
|
if(ptr_flash < (APP_ADDRESS + fw_size)) {
|
2025-03-17 15:43:34 +03:00
|
|
|
|
write_flash_page((const uint8_t*)msg.buf, msg.len);
|
2025-03-19 18:56:00 +03:00
|
|
|
|
ptr_flash += msg.len;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
send_ack(0x02);
|
2025-03-21 14:00:00 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
}
|
|
|
|
|
break;
|
2025-03-21 14:00:00 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
|
|
|
|
|
case BOOT_CAN_END: // Завершение передачи
|
|
|
|
|
if(verify_firmware()) {
|
|
|
|
|
erase_sector(7); // Сброс флага
|
|
|
|
|
send_ack(0xAA);
|
|
|
|
|
NVIC_SystemReset();
|
|
|
|
|
} else {
|
|
|
|
|
send_ack(0x55);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2025-03-10 20:43:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
2025-03-21 14:00:00 +03:00
|
|
|
|
|
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
void jump_to_app() {
|
2025-03-21 14:00:00 +03:00
|
|
|
|
__disable_irq();
|
|
|
|
|
jump = *(volatile uint32_t*)(APP_ADDRESS + 4);
|
|
|
|
|
void (*app_entry)(void);
|
|
|
|
|
app_entry = (void (*)(void))jump;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < 3; i++) {
|
|
|
|
|
NVIC->ICPR[i] = 0xFFFFFFFF;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__set_MSP(*(volatile uint32_t*)APP_ADDRESS);
|
|
|
|
|
SCB->VTOR = APP_ADDRESS;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
app_entry();
|
2025-03-10 20:43:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
void erase_flash_pages() {
|
|
|
|
|
FLASH_EraseInitTypeDef erase;
|
|
|
|
|
erase.TypeErase = FLASH_TYPEERASE_SECTORS;
|
2025-03-19 18:56:00 +03:00
|
|
|
|
erase.Sector = FLASH_SECTOR_2;
|
|
|
|
|
erase.NbSectors = 4;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
erase.VoltageRange = FLASH_VOLTAGE_RANGE_3;
|
|
|
|
|
|
|
|
|
|
uint32_t error;
|
|
|
|
|
flash_unlock();
|
|
|
|
|
HAL_FLASHEx_Erase(&erase, &error);
|
|
|
|
|
flash_lock();
|
|
|
|
|
}
|
2025-03-10 20:43:14 +03:00
|
|
|
|
|
|
|
|
|
|
2025-03-21 14:00:00 +03:00
|
|
|
|
|
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
uint16_t CalculateCRC16(const uint8_t* data, uint32_t length) {
|
2025-03-19 18:56:00 +03:00
|
|
|
|
uint16_t crc = 0xFFFF; // Начальное значение
|
2025-03-17 15:43:34 +03:00
|
|
|
|
while (length--) {
|
2025-03-19 18:56:00 +03:00
|
|
|
|
crc ^= *data++; // Обрабатываем LSB первым
|
|
|
|
|
for (uint8_t i = 0; i < 8; i++) {
|
|
|
|
|
if (crc & 0x0001) { // Проверяем младший бит
|
|
|
|
|
crc = (crc >> 1) ^ 0xA001; // Полином 0x8005 (reverse)
|
|
|
|
|
} else {
|
|
|
|
|
crc >>= 1;
|
|
|
|
|
}
|
2025-03-17 15:43:34 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
2025-03-19 18:56:00 +03:00
|
|
|
|
return crc; // Финальный XOR = 0x0000 (не требуется)
|
2025-03-17 15:43:34 +03:00
|
|
|
|
}
|
2025-03-19 18:56:00 +03:00
|
|
|
|
|
2025-03-21 14:00:00 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
bool verify_firmware() {
|
|
|
|
|
uint32_t calculated_crc = 0;
|
2025-03-19 18:56:00 +03:00
|
|
|
|
calculated_crc = CalculateCRC16((uint8_t*)APP_ADDRESS,fw_size);
|
2025-03-17 15:43:34 +03:00
|
|
|
|
if(calculated_crc != (uint16_t)fw_crc)
|
|
|
|
|
return false;
|
|
|
|
|
// Реализация проверки CRC
|
|
|
|
|
// ...
|
|
|
|
|
// return (calculated_crc == fw_crc);
|
2025-03-19 18:56:00 +03:00
|
|
|
|
else
|
|
|
|
|
return true;
|
2025-03-10 20:43:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
void send_ack(uint8_t status) {
|
2025-03-19 18:56:00 +03:00
|
|
|
|
|
2025-03-17 15:43:34 +03:00
|
|
|
|
CAN_message_t ack;
|
2025-03-19 18:56:00 +03:00
|
|
|
|
ack.id = ACK_CAN_ID;
|
2025-03-17 15:43:34 +03:00
|
|
|
|
ack.len = 1;
|
|
|
|
|
ack.buf[0] = status;
|
|
|
|
|
Can.write(ack);
|
2025-03-19 18:56:00 +03:00
|
|
|
|
}
|
|
|
|
|
|