2019-03-27 16:52:37 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2019 Linaro Limited
|
2020-07-16 18:01:02 +02:00
|
|
|
* Copyright (c) 2020 STMicroelectronics
|
2019-03-27 16:52:37 +01:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define LOG_DOMAIN flash_stm32wb
|
|
|
|
#define LOG_LEVEL CONFIG_FLASH_LOG_LEVEL
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/logging/log.h>
|
2019-03-27 16:52:37 +01:00
|
|
|
LOG_MODULE_REGISTER(LOG_DOMAIN);
|
|
|
|
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/kernel.h>
|
|
|
|
#include <zephyr/device.h>
|
2019-03-27 16:52:37 +01:00
|
|
|
#include <string.h>
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/drivers/flash.h>
|
|
|
|
#include <zephyr/init.h>
|
2019-03-27 16:52:37 +01:00
|
|
|
#include <soc.h>
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/sys/__assert.h>
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
#include "flash_stm32.h"
|
2020-07-16 18:01:02 +02:00
|
|
|
#include "stm32_hsem.h"
|
2020-09-03 21:34:39 +02:00
|
|
|
#if defined(CONFIG_BT)
|
2020-07-16 18:01:02 +02:00
|
|
|
#include "shci.h"
|
2020-09-03 21:34:39 +02:00
|
|
|
#endif
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
#define STM32WBX_PAGE_SHIFT 12
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Up to 255 4K pages
|
|
|
|
*/
|
2020-05-27 18:26:57 +02:00
|
|
|
static uint32_t get_page(off_t offset)
|
2019-03-27 16:52:37 +01:00
|
|
|
{
|
|
|
|
return offset >> STM32WBX_PAGE_SHIFT;
|
|
|
|
}
|
|
|
|
|
2021-04-06 11:30:51 +02:00
|
|
|
static inline void flush_cache(FLASH_TypeDef *regs)
|
|
|
|
{
|
|
|
|
if (regs->ACR & FLASH_ACR_DCEN) {
|
|
|
|
regs->ACR &= ~FLASH_ACR_DCEN;
|
|
|
|
/* Datasheet: DCRST: Data cache reset
|
2022-02-24 13:00:55 +01:00
|
|
|
* This bit can be written only when the data cache is disabled
|
2021-04-06 11:30:51 +02:00
|
|
|
*/
|
|
|
|
regs->ACR |= FLASH_ACR_DCRST;
|
|
|
|
regs->ACR &= ~FLASH_ACR_DCRST;
|
|
|
|
regs->ACR |= FLASH_ACR_DCEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (regs->ACR & FLASH_ACR_ICEN) {
|
|
|
|
regs->ACR &= ~FLASH_ACR_ICEN;
|
|
|
|
/* Datasheet: ICRST: Instruction cache reset :
|
|
|
|
* This bit can be written only when the instruction cache
|
|
|
|
* is disabled
|
|
|
|
*/
|
|
|
|
regs->ACR |= FLASH_ACR_ICRST;
|
|
|
|
regs->ACR &= ~FLASH_ACR_ICRST;
|
|
|
|
regs->ACR |= FLASH_ACR_ICEN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int write_dword(const struct device *dev, off_t offset, uint64_t val)
|
2019-03-27 16:52:37 +01:00
|
|
|
{
|
2024-01-23 10:39:22 +01:00
|
|
|
volatile uint32_t *flash = (uint32_t *)(offset + FLASH_STM32_BASE_ADDRESS);
|
2019-11-16 23:34:13 +01:00
|
|
|
FLASH_TypeDef *regs = FLASH_STM32_REGS(dev);
|
2020-05-27 18:26:57 +02:00
|
|
|
uint32_t tmp;
|
2019-03-27 16:52:37 +01:00
|
|
|
int ret, rc;
|
2020-07-16 18:01:02 +02:00
|
|
|
uint32_t cpu1_sem_status;
|
|
|
|
uint32_t cpu2_sem_status = 0;
|
|
|
|
uint32_t key;
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
/* if the control register is locked, do not fail silently */
|
2019-11-16 23:34:13 +01:00
|
|
|
if (regs->CR & FLASH_CR_LOCK) {
|
2020-07-08 16:43:13 +02:00
|
|
|
return -EIO;
|
2019-03-27 16:52:37 +01:00
|
|
|
}
|
|
|
|
|
2022-10-13 12:18:22 +02:00
|
|
|
/* Check if this double word is erased and value isn't 0.
|
|
|
|
*
|
|
|
|
* It is allowed to write only zeros over an already written dword
|
|
|
|
* See 3.3.8 in reference manual.
|
|
|
|
*/
|
|
|
|
if ((flash[0] != 0xFFFFFFFFUL ||
|
|
|
|
flash[1] != 0xFFFFFFFFUL) && val != 0UL) {
|
|
|
|
LOG_ERR("Word at offs %ld not erased", (long)offset);
|
2019-03-27 16:52:37 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = flash_stm32_check_status(dev);
|
|
|
|
if (ret < 0) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
/* Implementation of STM32 AN5289, proposed in STM32WB Cube Application
|
|
|
|
* BLE_RfWithFlash
|
|
|
|
* https://github.com/STMicroelectronics/STM32CubeWB/tree/master/Projects/P-NUCLEO-WB55.Nucleo/Applications/BLE/BLE_RfWithFlash
|
|
|
|
*/
|
2019-03-27 16:52:37 +01:00
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
do {
|
|
|
|
/**
|
|
|
|
* When the PESD bit mechanism is used by CPU2 to protect its
|
|
|
|
* timing, the PESD bit should be polled here.
|
|
|
|
* If the PESD is set, the CPU1 will be stalled when reading
|
|
|
|
* literals from an ISR that may occur after the flash
|
|
|
|
* processing has been requested but suspended due to the PESD
|
|
|
|
* bit.
|
|
|
|
*
|
|
|
|
* Note: This code is required only when the PESD mechanism is
|
|
|
|
* used to protect the CPU2 timing.
|
|
|
|
* However, keeping that code make it compatible with both
|
|
|
|
* mechanisms.
|
|
|
|
*/
|
2022-07-06 13:34:50 +02:00
|
|
|
while (LL_FLASH_IsActiveFlag_OperationSuspended()) {
|
2020-07-16 18:01:02 +02:00
|
|
|
;
|
2022-07-06 13:34:50 +02:00
|
|
|
}
|
2019-03-27 16:52:37 +01:00
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
/* Enter critical section */
|
|
|
|
key = irq_lock();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Depending on the application implementation, in case a
|
|
|
|
* multitasking is possible with an OS, it should be checked
|
|
|
|
* here if another task in the application disallowed flash
|
|
|
|
* processing to protect some latency in critical code
|
|
|
|
* execution.
|
|
|
|
* When flash processing is ongoing, the CPU cannot access the
|
|
|
|
* flash anymore.Trying to access the flash during that time
|
|
|
|
* stalls the CPU.
|
|
|
|
* The only way for CPU1 to disallow flash processing is to
|
|
|
|
* take CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID.
|
|
|
|
*/
|
|
|
|
cpu1_sem_status = LL_HSEM_GetStatus(HSEM,
|
|
|
|
CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID);
|
|
|
|
if (cpu1_sem_status == 0) {
|
|
|
|
/**
|
|
|
|
* Check now if the CPU2 disallows flash processing to
|
|
|
|
* protect its timing. If the semaphore is locked, the
|
|
|
|
* CPU2 does not allow flash processing
|
|
|
|
*
|
|
|
|
* Note: By default, the CPU2 uses the PESD mechanism
|
|
|
|
* to protect its timing, therefore, it is useless to
|
|
|
|
* get/release the semaphore.
|
|
|
|
*
|
|
|
|
* However, keeping that code make it compatible with
|
2022-02-24 13:00:55 +01:00
|
|
|
* both mechanisms.
|
2020-07-16 18:01:02 +02:00
|
|
|
* The protection by semaphore is enabled on CPU2 side
|
|
|
|
* with the command SHCI_C2_SetFlashActivityControl()
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
cpu2_sem_status = LL_HSEM_1StepLock(HSEM,
|
|
|
|
CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID);
|
|
|
|
if (cpu2_sem_status == 0) {
|
|
|
|
/**
|
|
|
|
* When CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID is
|
|
|
|
* taken, it is allowed to only write one
|
|
|
|
* single 64bits data.
|
|
|
|
* When several 64bits data need to be erased,
|
|
|
|
* the application shall first exit from the
|
|
|
|
* critical section and try again.
|
|
|
|
*/
|
|
|
|
/* Set the PG bit */
|
|
|
|
regs->CR |= FLASH_CR_PG;
|
|
|
|
|
|
|
|
/* Flush the register write */
|
|
|
|
tmp = regs->CR;
|
|
|
|
|
|
|
|
/* Perform the data write operation at desired
|
|
|
|
* memory address
|
|
|
|
*/
|
|
|
|
flash[0] = (uint32_t)val;
|
|
|
|
flash[1] = (uint32_t)(val >> 32);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Release the semaphore to give the
|
|
|
|
* opportunity to CPU2 to protect its timing
|
|
|
|
* versus the next flash operation by taking
|
|
|
|
* this semaphore.
|
|
|
|
* Note that the CPU2 is polling on this
|
|
|
|
* semaphore so CPU1 shall release it as fast
|
|
|
|
* as possible.
|
|
|
|
* This is why this code is protected by a
|
|
|
|
* critical section.
|
|
|
|
*/
|
|
|
|
LL_HSEM_ReleaseLock(HSEM,
|
|
|
|
CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Exit critical section */
|
|
|
|
irq_unlock(key);
|
|
|
|
|
|
|
|
} while (cpu2_sem_status || cpu1_sem_status);
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
/* Wait until the BSY bit is cleared */
|
|
|
|
rc = flash_stm32_wait_flash_idle(dev);
|
|
|
|
|
|
|
|
/* Clear the PG bit */
|
2019-11-16 23:34:13 +01:00
|
|
|
regs->CR &= (~FLASH_CR_PG);
|
2019-03-27 16:52:37 +01:00
|
|
|
|
2020-07-08 16:43:13 +02:00
|
|
|
return rc;
|
2019-03-27 16:52:37 +01:00
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int erase_page(const struct device *dev, uint32_t page)
|
2019-03-27 16:52:37 +01:00
|
|
|
{
|
2020-07-16 18:01:02 +02:00
|
|
|
uint32_t cpu1_sem_status;
|
|
|
|
uint32_t cpu2_sem_status = 0;
|
|
|
|
uint32_t key;
|
|
|
|
|
2019-11-16 23:34:13 +01:00
|
|
|
FLASH_TypeDef *regs = FLASH_STM32_REGS(dev);
|
2019-03-27 16:52:37 +01:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* if the control register is locked, do not fail silently */
|
2019-11-16 23:34:13 +01:00
|
|
|
if (regs->CR & FLASH_CR_LOCK) {
|
2019-03-27 16:52:37 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check that no Flash memory operation is ongoing */
|
|
|
|
rc = flash_stm32_wait_flash_idle(dev);
|
|
|
|
if (rc < 0) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2021-04-06 11:30:51 +02:00
|
|
|
/*
|
|
|
|
* If an erase operation in Flash memory also concerns data in the data
|
|
|
|
* or instruction cache, the user has to ensure that these data
|
|
|
|
* are rewritten before they are accessed during code execution.
|
|
|
|
*/
|
|
|
|
flush_cache(regs);
|
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
/* Implementation of STM32 AN5289, proposed in STM32WB Cube Application
|
|
|
|
* BLE_RfWithFlash
|
|
|
|
* https://github.com/STMicroelectronics/STM32CubeWB/tree/master/Projects/P-NUCLEO-WB55.Nucleo/Applications/BLE/BLE_RfWithFlash
|
|
|
|
*/
|
2019-03-27 16:52:37 +01:00
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
do {
|
|
|
|
/**
|
|
|
|
* When the PESD bit mechanism is used by CPU2 to protect its
|
|
|
|
* timing, the PESD bit should be polled here.
|
|
|
|
* If the PESD is set, the CPU1 will be stalled when reading
|
|
|
|
* literals from an ISR that may occur after the flash
|
|
|
|
* processing has been requested but suspended due to the PESD
|
|
|
|
* bit.
|
|
|
|
*
|
|
|
|
* Note: This code is required only when the PESD mechanism is
|
|
|
|
* used to protect the CPU2 timing.
|
|
|
|
* However, keeping that code make it compatible with both
|
|
|
|
* mechanisms.
|
|
|
|
*/
|
2022-07-06 13:34:50 +02:00
|
|
|
while (LL_FLASH_IsActiveFlag_OperationSuspended()) {
|
2020-07-16 18:01:02 +02:00
|
|
|
;
|
2022-07-06 13:34:50 +02:00
|
|
|
}
|
2020-07-16 18:01:02 +02:00
|
|
|
|
|
|
|
/* Enter critical section */
|
|
|
|
key = irq_lock();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Depending on the application implementation, in case a
|
|
|
|
* multitasking is possible with an OS, it should be checked
|
|
|
|
* here if another task in the application disallowed flash
|
|
|
|
* processing to protect some latency in critical code
|
|
|
|
* execution.
|
|
|
|
* When flash processing is ongoing, the CPU cannot access the
|
|
|
|
* flash anymore.Trying to access the flash during that time
|
|
|
|
* stalls the CPU.
|
|
|
|
* The only way for CPU1 to disallow flash processing is to
|
|
|
|
* take CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID.
|
|
|
|
*/
|
|
|
|
cpu1_sem_status = LL_HSEM_GetStatus(HSEM,
|
|
|
|
CFG_HW_BLOCK_FLASH_REQ_BY_CPU1_SEMID);
|
|
|
|
if (cpu1_sem_status == 0) {
|
|
|
|
/**
|
|
|
|
* Check now if the CPU2 disallows flash processing to
|
|
|
|
* protect its timing. If the semaphore is locked, the
|
|
|
|
* CPU2 does not allow flash processing
|
|
|
|
*
|
|
|
|
* Note: By default, the CPU2 uses the PESD mechanism
|
|
|
|
* to protect its timing, therefore, it is useless to
|
|
|
|
* get/release the semaphore.
|
|
|
|
*
|
|
|
|
* However, keeping that code make it compatible with
|
2022-02-24 13:00:55 +01:00
|
|
|
* both mechanisms.
|
2020-07-16 18:01:02 +02:00
|
|
|
* The protection by semaphore is enabled on CPU2 side
|
|
|
|
* with the command SHCI_C2_SetFlashActivityControl()
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
cpu2_sem_status = LL_HSEM_1StepLock(HSEM,
|
|
|
|
CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID);
|
|
|
|
if (cpu2_sem_status == 0) {
|
|
|
|
/**
|
|
|
|
* When CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID is
|
|
|
|
* taken, it is allowed to only erase one
|
|
|
|
* sector.
|
|
|
|
* When several sectors need to be erased,
|
|
|
|
* the application shall first exit from the
|
|
|
|
* critical section and try again.
|
|
|
|
*/
|
|
|
|
regs->CR |= FLASH_CR_PER;
|
|
|
|
regs->CR &= ~FLASH_CR_PNB_Msk;
|
|
|
|
regs->CR |= page << FLASH_CR_PNB_Pos;
|
|
|
|
|
|
|
|
regs->CR |= FLASH_CR_STRT;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Release the semaphore to give the
|
|
|
|
* opportunity to CPU2 to protect its timing
|
|
|
|
* versus the next flash operation by taking
|
|
|
|
* this semaphore.
|
|
|
|
* Note that the CPU2 is polling on this
|
|
|
|
* semaphore so CPU1 shall release it as fast
|
|
|
|
* as possible.
|
|
|
|
* This is why this code is protected by a
|
|
|
|
* critical section.
|
|
|
|
*/
|
|
|
|
LL_HSEM_ReleaseLock(HSEM,
|
|
|
|
CFG_HW_BLOCK_FLASH_REQ_BY_CPU2_SEMID,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Exit critical section */
|
|
|
|
irq_unlock(key);
|
|
|
|
|
|
|
|
} while (cpu2_sem_status || cpu1_sem_status);
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
|
|
|
|
/* Wait for the BSY bit */
|
|
|
|
rc = flash_stm32_wait_flash_idle(dev);
|
|
|
|
|
2020-07-16 14:00:53 +02:00
|
|
|
regs->CR &= ~FLASH_CR_PER;
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
int flash_stm32_block_erase_loop(const struct device *dev,
|
|
|
|
unsigned int offset,
|
2019-03-27 16:52:37 +01:00
|
|
|
unsigned int len)
|
|
|
|
{
|
|
|
|
int i, rc = 0;
|
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
#if defined(CONFIG_BT)
|
|
|
|
/**
|
|
|
|
* Notify the CPU2 that some flash erase activity may be executed
|
|
|
|
* On reception of this command, the CPU2 enables the BLE timing
|
|
|
|
* protection versus flash erase processing.
|
|
|
|
* The Erase flash activity will be executed only when the BLE RF is
|
|
|
|
* idle for at least 25ms.
|
|
|
|
* The CPU2 will prevent all flash activity (write or erase) in all
|
|
|
|
* cases when the BL RF Idle is shorter than 25ms.
|
|
|
|
*/
|
|
|
|
SHCI_C2_FLASH_EraseActivity(ERASE_ACTIVITY_ON);
|
|
|
|
#endif /* CONFIG_BT */
|
|
|
|
|
2019-03-27 16:52:37 +01:00
|
|
|
i = get_page(offset);
|
|
|
|
for (; i <= get_page(offset + len - 1) ; ++i) {
|
|
|
|
rc = erase_page(dev, i);
|
|
|
|
if (rc < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 18:01:02 +02:00
|
|
|
#if defined(CONFIG_BT)
|
|
|
|
/**
|
|
|
|
* Notify the CPU2 there will be no request anymore to erase the flash
|
|
|
|
* On reception of this command, the CPU2 disables the BLE timing
|
|
|
|
* protection versus flash erase processing
|
|
|
|
*/
|
|
|
|
SHCI_C2_FLASH_EraseActivity(ERASE_ACTIVITY_OFF);
|
|
|
|
#endif /* CONFIG_BT */
|
|
|
|
|
2019-03-27 16:52:37 +01:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
int flash_stm32_write_range(const struct device *dev, unsigned int offset,
|
2019-03-27 16:52:37 +01:00
|
|
|
const void *data, unsigned int len)
|
|
|
|
{
|
|
|
|
int i, rc = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < len; i += 8, offset += 8U) {
|
|
|
|
rc = write_dword(dev, offset,
|
2020-05-27 18:26:57 +02:00
|
|
|
UNALIGNED_GET((const uint64_t *) data + (i >> 3)));
|
2019-03-27 16:52:37 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
void flash_stm32_page_layout(const struct device *dev,
|
2019-03-27 16:52:37 +01:00
|
|
|
const struct flash_pages_layout **layout,
|
|
|
|
size_t *layout_size)
|
|
|
|
{
|
|
|
|
static struct flash_pages_layout stm32wb_flash_layout = {
|
|
|
|
.pages_count = 0,
|
|
|
|
.pages_size = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
ARG_UNUSED(dev);
|
|
|
|
|
|
|
|
if (stm32wb_flash_layout.pages_count == 0) {
|
|
|
|
stm32wb_flash_layout.pages_count = FLASH_SIZE / FLASH_PAGE_SIZE;
|
|
|
|
stm32wb_flash_layout.pages_size = FLASH_PAGE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*layout = &stm32wb_flash_layout;
|
|
|
|
*layout_size = 1;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
int flash_stm32_check_status(const struct device *dev)
|
2019-03-27 16:52:37 +01:00
|
|
|
{
|
2019-11-16 23:34:13 +01:00
|
|
|
FLASH_TypeDef *regs = FLASH_STM32_REGS(dev);
|
2020-05-27 18:26:57 +02:00
|
|
|
uint32_t error = 0;
|
2019-03-27 16:52:37 +01:00
|
|
|
|
|
|
|
/* Save Flash errors */
|
2020-01-29 16:34:45 +01:00
|
|
|
error = (regs->SR & FLASH_FLAG_SR_ERRORS);
|
|
|
|
error |= (regs->ECCR & FLASH_FLAG_ECCC);
|
2019-03-27 16:52:37 +01:00
|
|
|
|
2022-02-24 13:00:55 +01:00
|
|
|
/* Clear systematic Option and Engineering bits validity error */
|
2019-03-27 16:52:37 +01:00
|
|
|
if (error & FLASH_FLAG_OPTVERR) {
|
2020-01-29 16:34:45 +01:00
|
|
|
regs->SR |= FLASH_FLAG_SR_ERRORS;
|
2019-03-27 16:52:37 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (error) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|