diff --git a/src/include/stm32lx-nvm.h b/src/include/stm32lx-nvm.h deleted file mode 100644 index 2e9d8c5..0000000 --- a/src/include/stm32lx-nvm.h +++ /dev/null @@ -1,190 +0,0 @@ -/* @file stm32lx-nvm.h - * - * This file is part of the Black Magic Debug project. - * - * Copyright (C) 2014 Woollysoft - * Written by Marc Singer - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -#if !defined (STM32Lx_NVM_H_INCLUDED) -# define STM32Lx_NVM_H_INCLUDED - -/* ----- Includes */ - -#include - -/* ----- Macros */ - -/* ----- Types */ - -enum { - STM32Lx_STUB_PHYS = 0x20000000ul, - STM32Lx_STUB_INFO_PHYS = 0x20000004ul, - STM32Lx_STUB_DATA_PHYS = (0x20000000ul + 1024), - STM32Lx_STUB_DATA_MAX = 2048, - - STM32Lx_NVM_OPT_PHYS = 0x1ff80000ul, - STM32Lx_NVM_EEPROM_PHYS = 0x08080000ul, - - STM32L0_NVM_PHYS = 0x40022000ul, - STM32L0_NVM_PROG_PAGE_SIZE = 128, - STM32L0_NVM_DATA_PAGE_SIZE = 4, - STM32L0_NVM_OPT_SIZE = 12, - STM32L0_NVM_EEPROM_SIZE = 2*1024, - - STM32L1_NVM_PHYS = 0x40023c00ul, - STM32L1_NVM_PROG_PAGE_SIZE = 256, - STM32L1_NVM_DATA_PAGE_SIZE = 4, - STM32L1_NVM_OPT_SIZE = 32, - STM32L1_NVM_EEPROM_SIZE = 16*1024, - - STM32Lx_NVM_PEKEY1 = 0x89abcdeful, - STM32Lx_NVM_PEKEY2 = 0x02030405ul, - STM32Lx_NVM_PRGKEY1 = 0x8c9daebful, - STM32Lx_NVM_PRGKEY2 = 0x13141516ul, - STM32Lx_NVM_OPTKEY1 = 0xfbead9c8ul, - STM32Lx_NVM_OPTKEY2 = 0x24252627ul, - - STM32Lx_NVM_PECR_OBL_LAUNCH = (1<<18), - STM32Lx_NVM_PECR_ERRIE = (1<<17), - STM32Lx_NVM_PECR_EOPIE = (1<<16), - STM32Lx_NVM_PECR_FPRG = (1<<10), - STM32Lx_NVM_PECR_ERASE = (1<< 9), - STM32Lx_NVM_PECR_FIX = (1<< 8), /* FTDW */ - STM32Lx_NVM_PECR_DATA = (1<< 4), - STM32Lx_NVM_PECR_PROG = (1<< 3), - STM32Lx_NVM_PECR_OPTLOCK = (1<< 2), - STM32Lx_NVM_PECR_PRGLOCK = (1<< 1), - STM32Lx_NVM_PECR_PELOCK = (1<< 0), - - STM32Lx_NVM_SR_FWWERR = (1<<17), - STM32Lx_NVM_SR_NOTZEROERR = (1<<16), - STM32Lx_NVM_SR_RDERR = (1<<13), - STM32Lx_NVM_SR_OPTVER = (1<<11), - STM32Lx_NVM_SR_SIZERR = (1<<10), - STM32Lx_NVM_SR_PGAERR = (1<<9), - STM32Lx_NVM_SR_WRPERR = (1<<8), - STM32Lx_NVM_SR_READY = (1<<3), - STM32Lx_NVM_SR_HWOFF = (1<<2), - STM32Lx_NVM_SR_EOP = (1<<1), - STM32Lx_NVM_SR_BSY = (1<<0), - STM32Lx_NVM_SR_ERR_M = ( STM32Lx_NVM_SR_WRPERR - | STM32Lx_NVM_SR_PGAERR - | STM32Lx_NVM_SR_SIZERR - | STM32Lx_NVM_SR_NOTZEROERR), - - STM32L0_NVM_OPTR_BOOT1 = (1<<31), - STM32L0_NVM_OPTR_WDG_SW = (1<<20), - STM32L0_NVM_OPTR_WPRMOD = (1<<8), - STM32L0_NVM_OPTR_RDPROT_S = (0), - STM32L0_NVM_OPTR_RDPROT_M = (0xff), - STM32L0_NVM_OPTR_RDPROT_0 = (0xaa), - STM32L0_NVM_OPTR_RDPROT_2 = (0xcc), - - STM32L1_NVM_OPTR_nBFB2 = (1<<23), - STM32L1_NVM_OPTR_nRST_STDBY = (1<<22), - STM32L1_NVM_OPTR_nRST_STOP = (1<<21), - STM32L1_NVM_OPTR_WDG_SW = (1<<20), - STM32L1_NVM_OPTR_BOR_LEV_S = (16), - STM32L1_NVM_OPTR_BOR_LEV_M = (0xf), - STM32L1_NVM_OPTR_SPRMOD = (1<<8), - STM32L1_NVM_OPTR_RDPROT_S = (0), - STM32L1_NVM_OPTR_RDPROT_M = (0xff), - STM32L1_NVM_OPTR_RDPROT_0 = (0xaa), - STM32L1_NVM_OPTR_RDPROT_2 = (0xcc), - -}; - -#if defined (__cplusplus) - -namespace STM32 { - struct NVM { - volatile uint32_t acr; - volatile uint32_t pecr; - volatile uint32_t pdkeyr; - volatile uint32_t pkeyr; - volatile uint32_t prgkeyr; - volatile uint32_t optkeyr; - volatile uint32_t sr; - volatile uint32_t optr; - volatile uint32_t wrprot; - - static constexpr uint32_t PKEY1 = 0x89abcdef; - static constexpr uint32_t PKEY2 = 0x02030405; - static constexpr uint32_t PRGKEY1 = 0x8c9daebf; - static constexpr uint32_t PRGKEY2 = 0x13141516; - static constexpr uint32_t OPTKEY1 = 0xfbead9c8; - static constexpr uint32_t OPTKEY2 = 0x24252627; - static constexpr uint32_t PDKEY1 = 0x04152637; - static constexpr uint32_t PDKEY2 = 0xfafbfcfd; - }; - - static_assert(sizeof (NVM) == 9*4, "NVM size error"); -} -using stm32lx_stub_pointer_t = uint32_t*; - -#define Nvm(nvm) (*reinterpret_cast(nvm)) -#define Info (*reinterpret_cast(STM32Lx_STUB_INFO_PHYS)) - -namespace { - inline __attribute((always_inline)) bool unlock (STM32::NVM& nvm) { - // Lock guarantees unlock - nvm.pecr = STM32Lx_NVM_PECR_PELOCK; - - nvm.pkeyr = STM32::NVM::PKEY1; - nvm.pkeyr = STM32::NVM::PKEY2; - nvm.prgkeyr = STM32::NVM::PRGKEY1; - nvm.prgkeyr = STM32::NVM::PRGKEY2; - return !(nvm.pecr & STM32Lx_NVM_PECR_PRGLOCK); - } - inline __attribute((always_inline)) void lock (STM32::NVM& nvm) { - nvm.pecr = STM32Lx_NVM_PECR_PELOCK; } - -} - -#else - -typedef uint32_t stm32lx_stub_pointer_t; - -#define STM32Lx_NVM_PECR(p) ((p) + 0x04) -#define STM32Lx_NVM_PEKEYR(p) ((p) + 0x0C) -#define STM32Lx_NVM_PRGKEYR(p) ((p) + 0x10) -#define STM32Lx_NVM_OPTKEYR(p) ((p) + 0x14) -#define STM32Lx_NVM_SR(p) ((p) + 0x18) -#define STM32Lx_NVM_OPTR(p) ((p) + 0x1C) - -#endif - -enum { - OPT_STM32L1 = 1<<1, -}; - -struct stm32lx_nvm_stub_info { - stm32lx_stub_pointer_t destination; - int32_t size; - stm32lx_stub_pointer_t source; - uint32_t nvm; - uint16_t page_size; - uint16_t options; -} __attribute__((packed)); - -/* ----- Globals */ - -/* ----- Prototypes */ - - - -#endif /* STM32Lx_NVM_H_INCLUDED */ diff --git a/src/stm32l0.c b/src/stm32l0.c index 0c35577..b5ee548 100644 --- a/src/stm32l0.c +++ b/src/stm32l0.c @@ -80,7 +80,68 @@ #include "gdb_packet.h" #include "cortexm.h" -#include "stm32lx-nvm.h" +#define STM32Lx_NVM_PECR(p) ((p) + 0x04) +#define STM32Lx_NVM_PEKEYR(p) ((p) + 0x0C) +#define STM32Lx_NVM_PRGKEYR(p) ((p) + 0x10) +#define STM32Lx_NVM_OPTKEYR(p) ((p) + 0x14) +#define STM32Lx_NVM_SR(p) ((p) + 0x18) +#define STM32Lx_NVM_OPTR(p) ((p) + 0x1C) + +#define STM32L0_NVM_PHYS (0x40022000ul) +#define STM32L0_NVM_OPT_SIZE (12) +#define STM32L0_NVM_EEPROM_SIZE (2*1024) + +#define STM32L1_NVM_PHYS (0x40023c00ul) +#define STM32L1_NVM_OPT_SIZE (32) +#define STM32L1_NVM_EEPROM_SIZE (16*1024) + +#define STM32Lx_NVM_OPT_PHYS 0x1ff80000ul +#define STM32Lx_NVM_EEPROM_PHYS 0x08080000ul + +#define STM32Lx_NVM_PEKEY1 (0x89abcdeful) +#define STM32Lx_NVM_PEKEY2 (0x02030405ul) +#define STM32Lx_NVM_PRGKEY1 (0x8c9daebful) +#define STM32Lx_NVM_PRGKEY2 (0x13141516ul) +#define STM32Lx_NVM_OPTKEY1 (0xfbead9c8ul) +#define STM32Lx_NVM_OPTKEY2 (0x24252627ul) + +#define STM32Lx_NVM_PECR_OBL_LAUNCH (1<<18) +#define STM32Lx_NVM_PECR_ERRIE (1<<17) +#define STM32Lx_NVM_PECR_EOPIE (1<<16) +#define STM32Lx_NVM_PECR_FPRG (1<<10) +#define STM32Lx_NVM_PECR_ERASE (1<< 9) +#define STM32Lx_NVM_PECR_FIX (1<< 8) /* FTDW */ +#define STM32Lx_NVM_PECR_DATA (1<< 4) +#define STM32Lx_NVM_PECR_PROG (1<< 3) +#define STM32Lx_NVM_PECR_OPTLOCK (1<< 2) +#define STM32Lx_NVM_PECR_PRGLOCK (1<< 1) +#define STM32Lx_NVM_PECR_PELOCK (1<< 0) + +#define STM32Lx_NVM_SR_NOTZEROERR (1<<16) +#define STM32Lx_NVM_SR_SIZERR (1<<10) +#define STM32Lx_NVM_SR_PGAERR (1<<9) +#define STM32Lx_NVM_SR_WRPERR (1<<8) +#define STM32Lx_NVM_SR_EOP (1<<1) +#define STM32Lx_NVM_SR_BSY (1<<0) +#define STM32Lx_NVM_SR_ERR_M (STM32Lx_NVM_SR_WRPERR | \ + STM32Lx_NVM_SR_PGAERR | \ + STM32Lx_NVM_SR_SIZERR | \ + STM32Lx_NVM_SR_NOTZEROERR) + +#define STM32L0_NVM_OPTR_BOOT1 (1<<31) +#define STM32Lx_NVM_OPTR_WDG_SW (1<<20) +#define STM32L0_NVM_OPTR_WPRMOD (1<<8) +#define STM32Lx_NVM_OPTR_RDPROT_S (0) +#define STM32Lx_NVM_OPTR_RDPROT_M (0xff) +#define STM32Lx_NVM_OPTR_RDPROT_0 (0xaa) +#define STM32Lx_NVM_OPTR_RDPROT_2 (0xcc) + +#define STM32L1_NVM_OPTR_nBFB2 (1<<23) +#define STM32L1_NVM_OPTR_nRST_STDBY (1<<22) +#define STM32L1_NVM_OPTR_nRST_STOP (1<<21) +#define STM32L1_NVM_OPTR_BOR_LEV_S (16) +#define STM32L1_NVM_OPTR_BOR_LEV_M (0xf) +#define STM32L1_NVM_OPTR_SPRMOD (1<<8) static int stm32lx_nvm_prog_erase(struct target_flash* f, uint32_t addr, size_t len); @@ -566,11 +627,11 @@ static bool stm32lx_cmd_option(target* t, int argc, char** argv) if (stm32lx_is_stm32l1(t)) { uint32_t optr = target_mem_read32(t, STM32Lx_NVM_OPTR(nvm)); - uint8_t rdprot = (optr >> STM32L1_NVM_OPTR_RDPROT_S) - & STM32L1_NVM_OPTR_RDPROT_M; - if (rdprot == STM32L1_NVM_OPTR_RDPROT_0) + uint8_t rdprot = (optr >> STM32Lx_NVM_OPTR_RDPROT_S) + & STM32Lx_NVM_OPTR_RDPROT_M; + if (rdprot == STM32Lx_NVM_OPTR_RDPROT_0) rdprot = 0; - else if (rdprot == STM32L1_NVM_OPTR_RDPROT_2) + else if (rdprot == STM32Lx_NVM_OPTR_RDPROT_2) rdprot = 2; else rdprot = 1; @@ -581,18 +642,18 @@ static bool stm32lx_cmd_option(target* t, int argc, char** argv) (optr & STM32L1_NVM_OPTR_SPRMOD) ? 1 : 0, (optr >> STM32L1_NVM_OPTR_BOR_LEV_S) & STM32L1_NVM_OPTR_BOR_LEV_M, - (optr & STM32L1_NVM_OPTR_WDG_SW) ? 1 : 0, + (optr & STM32Lx_NVM_OPTR_WDG_SW) ? 1 : 0, (optr & STM32L1_NVM_OPTR_nRST_STOP) ? 1 : 0, (optr & STM32L1_NVM_OPTR_nRST_STDBY) ? 1 : 0, (optr & STM32L1_NVM_OPTR_nBFB2) ? 1 : 0); } else { uint32_t optr = target_mem_read32(t, STM32Lx_NVM_OPTR(nvm)); - uint8_t rdprot = (optr >> STM32L0_NVM_OPTR_RDPROT_S) - & STM32L0_NVM_OPTR_RDPROT_M; - if (rdprot == STM32L0_NVM_OPTR_RDPROT_0) + uint8_t rdprot = (optr >> STM32Lx_NVM_OPTR_RDPROT_S) + & STM32Lx_NVM_OPTR_RDPROT_M; + if (rdprot == STM32Lx_NVM_OPTR_RDPROT_0) rdprot = 0; - else if (rdprot == STM32L0_NVM_OPTR_RDPROT_2) + else if (rdprot == STM32Lx_NVM_OPTR_RDPROT_2) rdprot = 2; else rdprot = 1; @@ -600,7 +661,7 @@ static bool stm32lx_cmd_option(target* t, int argc, char** argv) "BOOT1 %d\n", optr, rdprot, (optr & STM32L0_NVM_OPTR_WPRMOD) ? 1 : 0, - (optr & STM32L0_NVM_OPTR_WDG_SW) ? 1 : 0, + (optr & STM32Lx_NVM_OPTR_WDG_SW) ? 1 : 0, (optr & STM32L0_NVM_OPTR_BOOT1) ? 1 : 0); }