2018-04-01 17:40:14 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Inria
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cpu_stm32_common
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level flash lock/unlock implementation
|
|
|
|
*
|
|
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
2019-03-27 09:49:00 +01:00
|
|
|
* @author Oleg Artamonov <oleg@unwds.com>
|
2018-04-01 17:40:14 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#if defined(CPU_FAM_STM32L0) || defined(CPU_FAM_STM32L1)
|
|
|
|
/* Data EEPROM and control register unlock keys */
|
|
|
|
#define FLASH_KEY1 ((uint32_t)0x89ABCDEF)
|
|
|
|
#define FLASH_KEY2 ((uint32_t)0x02030405)
|
|
|
|
#define CNTRL_REG (FLASH->PECR)
|
|
|
|
#define CNTRL_REG_LOCK (FLASH_PECR_PELOCK)
|
|
|
|
#define KEY_REG (FLASH->PEKEYR)
|
|
|
|
#else
|
2018-04-11 09:56:40 +02:00
|
|
|
#if defined(CPU_FAM_STM32L4)
|
|
|
|
#define FLASH_KEY1 ((uint32_t)0x45670123)
|
|
|
|
#define FLASH_KEY2 ((uint32_t)0xCDEF89AB)
|
|
|
|
#endif
|
2018-04-01 17:40:14 +02:00
|
|
|
#define CNTRL_REG (FLASH->CR)
|
|
|
|
#define CNTRL_REG_LOCK (FLASH_CR_LOCK)
|
|
|
|
#define KEY_REG (FLASH->KEYR)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void _unlock(void)
|
|
|
|
{
|
|
|
|
if (CNTRL_REG & CNTRL_REG_LOCK) {
|
2018-05-27 15:11:59 +02:00
|
|
|
DEBUG("[flash-common] unlocking the flash module\n");
|
2018-04-01 17:40:14 +02:00
|
|
|
KEY_REG = FLASH_KEY1;
|
|
|
|
KEY_REG = FLASH_KEY2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _lock(void)
|
|
|
|
{
|
2018-05-27 15:11:59 +02:00
|
|
|
if (!(CNTRL_REG & CNTRL_REG_LOCK)) {
|
|
|
|
DEBUG("[flash-common] locking the flash module\n");
|
|
|
|
CNTRL_REG |= CNTRL_REG_LOCK;
|
|
|
|
}
|
2018-04-01 17:40:14 +02:00
|
|
|
}
|
2019-03-27 09:49:00 +01:00
|
|
|
|
|
|
|
void _wait_for_pending_operations(void)
|
|
|
|
{
|
|
|
|
if (FLASH->SR & FLASH_SR_BSY) {
|
|
|
|
DEBUG("[flash-common] waiting for any pending operation to finish\n");
|
|
|
|
while (FLASH->SR & FLASH_SR_BSY) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clear 'end of operation' bit in status register */
|
|
|
|
if (FLASH->SR & FLASH_SR_EOP) {
|
|
|
|
FLASH->SR &= ~(FLASH_SR_EOP);
|
|
|
|
}
|
|
|
|
}
|