rusefi/firmware/hw_layer/ports/stm32/stm32f7/mpu_util.cpp

225 lines
6.4 KiB
C++

/**
* @file mpu_util.cpp
*
* @date Jul 27, 2014
* @author Andrey Belomutskiy, (c) 2012-2020
*/
#include "pch.h"
#include "flash_int.h"
static bool isDualBank() {
// cleared bit indicates dual bank
return (FLASH->OPTCR & FLASH_OPTCR_nDBANK) == 0;
}
static uint16_t flashSize() {
return *reinterpret_cast<const volatile uint16_t*>(FLASHSIZE_BASE);
}
enum class DeviceType {
DualBank1MB,
DualBank2MB,
SingleBank1MB,
SingleBank2MB,
Unknown
};
static DeviceType determineDevice() {
bool db = isDualBank();
uint16_t fs = flashSize();
if (db) {
if (fs == 1024) {
return DeviceType::DualBank1MB;
} else if (fs == 2048) {
return DeviceType::DualBank2MB;
}
} else {
if (fs == 1024) {
// Unsupported scenario! Not enough space for program plus two config copies
firmwareError(OBD_PCM_Processor_Fault, "1MB single bank MCU detected: please clear nDBANK option bit and reinstall FW.");
return DeviceType::SingleBank1MB;
} else if (fs == 2048) {
return DeviceType::SingleBank2MB;
}
}
firmwareError(OBD_PCM_Processor_Fault, "Unrecognized flash memory layout db=%d, size=%d", db, fs);
return DeviceType::Unknown;
}
bool allowFlashWhileRunning() {
// Allow flash-while-running if dual bank mode is enabled, and we're a 2MB device (ie, no code located in second bank)
return determineDevice() == DeviceType::DualBank2MB;
}
// See ST AN4826
size_t flashSectorSize(flashsector_t sector) {
// 1MB devices have 8 sectors per bank
// 2MB devices have 12 sectors per bank
// However, the second bank always starts at index 12 (1MB devices have a 4 sector discontinuity between banks)
if (sector >= 12) {
// The second bank has the same structure as the first
return flashSectorSize(sector - 12);
}
// On 1MB devices, sectors 8-11 don't exist, therefore have zero size.
if (flashSize() == 1024) {
if (sector > 7 && sector < 12) {
return 0;
}
}
// Pages are twice the size when in single bank mode
size_t dbMul = isDualBank() ? 1 : 2;
if (sector <= 3)
return 16 * 1024 * dbMul;
else if (sector == 4)
return 64 * 1024 * dbMul;
else if (sector >= 5)
return 128 * 1024 * dbMul;
return 0;
}
uintptr_t getFlashAddrFirstCopy() {
switch (determineDevice()) {
case DeviceType::DualBank1MB:
// Sector 18, second to last 128K sector
return 0x080C0000;
case DeviceType::SingleBank1MB:
// Sector 7, last 256K sector
return 0x080C0000;
case DeviceType::DualBank2MB: /* falls thru */
case DeviceType::SingleBank2MB:
// Start of the second megabyte
// Sector 12 for dual bank
// Sector 8 for single bank
return 0x08100000;
default:
return 0;
}
}
uintptr_t getFlashAddrSecondCopy() {
switch (determineDevice()) {
case DeviceType::DualBank1MB:
// Sector 19, last 128K sector, 128K after the first copy
return 0x080E0000;
case DeviceType::DualBank2MB:
// Sector 14, 32K after the first copy
return 0x08108000;
case DeviceType::SingleBank2MB:
// Sector 9, 256K after the first copy
return 0x08140000;
case DeviceType::SingleBank1MB:
// We can't fit a second copy in this config, fall thru to failure case
default:
return 0;
}
}
#define FLASH_ACR (*(volatile uint32_t *)(FLASH_BASE + 0x00))
#define FLASH_KEYR (*(volatile uint32_t *)(FLASH_BASE + 0x04))
#define FLASH_OPTKEYR (*(volatile uint32_t *)(FLASH_BASE + 0x08))
#define FLASH_SR (*(volatile uint32_t *)(FLASH_BASE + 0x0C))
#define FLASH_CR (*(volatile uint32_t *)(FLASH_BASE + 0x10))
#define FLASH_OPTCR (*(volatile uint32_t *)(FLASH_BASE + 0x14))
#define FLASH_OPTCR_STRT (1 << 1)
#define FLASH_OPTKEY1 (0x08192A3B)
#define FLASH_OPTKEY2 (0x4C5D6E7F)
static void flash_wait_complete()
{
do { __DSB(); } while (FLASH->SR & FLASH_SR_BSY);
}
static void stm32f7_flash_mass_erase_dual_block()
{
FLASH_CR |= FLASH_CR_MER1 | FLASH_CR_MER2;
FLASH_CR |= FLASH_CR_STRT;
flash_wait_complete();
FLASH_CR &= ~(FLASH_CR_MER1 | FLASH_CR_MER2);
}
// todo: at the moment this does not work :(
// https://github.com/rusefi/rusefi/issues/2996
void sys_dual_bank(void) {
uint32_t reg;
efiPrintf("FLASH->SR before %x", FLASH->SR);
/* Unlock OPTCR */
FLASH_OPTKEYR = FLASH_OPTKEY1;
FLASH_OPTKEYR = FLASH_OPTKEY2;
flash_wait_complete();
/* Disable protection + Switch to dual bank */
reg = FLASH_OPTCR;
reg &= ~0x000FF00;
reg |= 0x0000AA00;
reg &= ~(FLASH_OPTCR_nDBANK);
FLASH_OPTCR = reg;
__DSB();
FLASH_OPTCR |= FLASH_OPTCR_STRT;
flash_wait_complete();
efiPrintf("FLASH->SR after %x", FLASH->SR);
/*
* see https://github.com/danielinux/stm32f7-dualbank-tool/issues/1
stm32f7_flash_mass_erase_dual_block();
*/
}
/*
STOP mode for F7 is needed for wakeup from multiple EXTI pins. For example PD0, which is CAN rx.
However, for F40X & F42X this may be useless. STOP in itself eats more current than standby.
With F4 only having PA0 available for wakeup, this negates its need.
*/
void stm32_stop() {
SysTick->CTRL = 0;
SCB->SCR |= SCB_SCR_SLEEPDEEP_Msk;
// Turn off the LEDs, those use some power
enginePins.errorLedPin.setValue(0);
enginePins.runningLedPin.setValue(0);
enginePins.communicationLedPin.setValue(0);
enginePins.warningLedPin.setValue(0);
// Do anything the board wants to prepare for stop mode - enabling wakeup sources!
boardPrepareForStop();
PWR->CSR1 |= PWR_CSR1_WUIF;
PWR->CR1 &= ~PWR_CR1_PDDS; // cleared PDDS means stop mode (not standby)
PWR->CR1 |= PWR_CR1_FPDS; // turn off flash in stop mode
PWR->CR1 |= PWR_CR1_UDEN; // regulator underdrive in stop mode
PWR->CR1 |= PWR_CR1_LPUDS; // low power regulator in under drive mode
PWR->CR1 |= PWR_CR1_LPDS; // regulator in low power mode
// enable Deepsleep mode
__disable_irq();
__WFI();
// Lastly, reboot
NVIC_SystemReset();
}
/*
Standby for both F4 & F7 works perfectly, with very little curent consumption. Downside is that theres a limited amount of pins that can wakeup F7, and only PA0 for F4XX.
Cannot be used for CAN wakeup without hardware modificatinos.
*/
void stm32_standby() {
SysTick->CTRL = 0;
SCB->SCR |= SCB_SCR_SLEEPDEEP_Msk;
PWR->CR1 |= PWR_CR1_PDDS; // PDDS = use standby mode (not stop mode)
PWR->CR1 |= PWR_CR1_CSBF; // Clear standby flag
// Do anything the board wants to prepare for standby mode - enabling wakeup sources!
boardPrepareForStandby();
__disable_irq();
__WFI();
}