commit
6ec35bc9ff
|
@ -28,6 +28,7 @@ stm32f4_flash_write_stub(uint32_t *dest, uint32_t *src, uint32_t size)
|
||||||
for (int i = 0; i < size; i += 4) {
|
for (int i = 0; i < size; i += 4) {
|
||||||
FLASH_CR = FLASH_CR_PROGRAM_X32 | FLASH_CR_PG;
|
FLASH_CR = FLASH_CR_PROGRAM_X32 | FLASH_CR_PG;
|
||||||
*dest++ = *src++;
|
*dest++ = *src++;
|
||||||
|
__asm("dsb");
|
||||||
while (FLASH_SR & FLASH_SR_BSY)
|
while (FLASH_SR & FLASH_SR_BSY)
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
0x2300, 0x4C09, 0x4293, 0xD209, 0x4D08, 0x4E09, 0x602E, 0x58CD, 0x50C5, 0x6825, 0x03ED, 0xD4FC, 0x3304, 0xE7F2, 0x23F2, 0x6822, 0x421A, 0xD000, 0xBE01, 0xBE00, 0x3C0C, 0x4002, 0x3C10, 0x4002, 0x0201, 0x0000,
|
0x2300, 0x4C0A, 0x4293, 0xD20B, 0x4E09, 0x4D0A, 0x602E, 0x58CD, 0x50C5, 0xF3BF, 0x8F4F, 0x6825, 0x03ED, 0xD4FC, 0x3304, 0xE7F0, 0x23F2, 0x6822, 0x421A, 0xD000, 0xBE01, 0xBE00, 0x3C0C, 0x4002, 0x0201, 0x0000, 0x3C10, 0x4002,
|
|
@ -89,6 +89,7 @@ struct cortexm_priv {
|
||||||
uint8_t type;
|
uint8_t type;
|
||||||
uint8_t size;
|
uint8_t size;
|
||||||
} hw_watchpoint[CORTEXM_MAX_WATCHPOINTS];
|
} hw_watchpoint[CORTEXM_MAX_WATCHPOINTS];
|
||||||
|
unsigned flash_patch_revision;
|
||||||
unsigned hw_watchpoint_max;
|
unsigned hw_watchpoint_max;
|
||||||
/* Breakpoint unit status */
|
/* Breakpoint unit status */
|
||||||
uint32_t hw_breakpoint[CORTEXM_MAX_BREAKPOINTS];
|
uint32_t hw_breakpoint[CORTEXM_MAX_BREAKPOINTS];
|
||||||
|
@ -283,6 +284,7 @@ bool cortexm_attach(target *t)
|
||||||
r = target_mem_read32(t, CORTEXM_FPB_CTRL);
|
r = target_mem_read32(t, CORTEXM_FPB_CTRL);
|
||||||
if (((r >> 4) & 0xf) < priv->hw_breakpoint_max) /* only look at NUM_COMP1 */
|
if (((r >> 4) & 0xf) < priv->hw_breakpoint_max) /* only look at NUM_COMP1 */
|
||||||
priv->hw_breakpoint_max = (r >> 4) & 0xf;
|
priv->hw_breakpoint_max = (r >> 4) & 0xf;
|
||||||
|
priv->flash_patch_revision = (r >> 28);
|
||||||
priv->hw_watchpoint_max = CORTEXM_MAX_WATCHPOINTS;
|
priv->hw_watchpoint_max = CORTEXM_MAX_WATCHPOINTS;
|
||||||
r = target_mem_read32(t, CORTEXM_DWT_CTRL);
|
r = target_mem_read32(t, CORTEXM_DWT_CTRL);
|
||||||
if ((r >> 28) > priv->hw_watchpoint_max)
|
if ((r >> 28) > priv->hw_watchpoint_max)
|
||||||
|
@ -637,10 +639,13 @@ static int cortexm_set_hw_bp(target *t, uint32_t addr)
|
||||||
{
|
{
|
||||||
ADIv5_AP_t *ap = adiv5_target_ap(t);
|
ADIv5_AP_t *ap = adiv5_target_ap(t);
|
||||||
struct cortexm_priv *priv = ap->priv;
|
struct cortexm_priv *priv = ap->priv;
|
||||||
uint32_t val = addr & 0x1FFFFFFC;
|
uint32_t val = addr;
|
||||||
unsigned i;
|
unsigned i;
|
||||||
|
|
||||||
|
if (priv->flash_patch_revision == 0) {
|
||||||
|
val = addr & 0x1FFFFFFC;
|
||||||
val |= (addr & 2)?0x80000000:0x40000000;
|
val |= (addr & 2)?0x80000000:0x40000000;
|
||||||
|
}
|
||||||
val |= 1;
|
val |= 1;
|
||||||
|
|
||||||
for(i = 0; i < priv->hw_breakpoint_max; i++)
|
for(i = 0; i < priv->hw_breakpoint_max; i++)
|
||||||
|
|
|
@ -52,6 +52,7 @@ static int stm32f4_flash_write(struct target_flash *f,
|
||||||
uint32_t dest, const void *src, size_t len);
|
uint32_t dest, const void *src, size_t len);
|
||||||
|
|
||||||
static const char stm32f4_driver_str[] = "STM32F4xx";
|
static const char stm32f4_driver_str[] = "STM32F4xx";
|
||||||
|
static const char stm32f7_driver_str[] = "STM32F7xx";
|
||||||
|
|
||||||
/* Flash Program ad Erase Controller Register Map */
|
/* Flash Program ad Erase Controller Register Map */
|
||||||
#define FPEC_BASE 0x40023C00
|
#define FPEC_BASE 0x40023C00
|
||||||
|
@ -92,7 +93,16 @@ static const char stm32f4_driver_str[] = "STM32F4xx";
|
||||||
|
|
||||||
#define DBGMCU_IDCODE 0xE0042000
|
#define DBGMCU_IDCODE 0xE0042000
|
||||||
|
|
||||||
/* This routine is uses word access. Only usable on target voltage >2.7V */
|
#define DBGMCU_CR 0xE0042004
|
||||||
|
#define DBG_STANDBY (1 << 0)
|
||||||
|
#define DBG_STOP (1 << 1)
|
||||||
|
#define DBG_SLEEP (1 << 2)
|
||||||
|
|
||||||
|
#define DBGMCU_APB1_FZ 0xE0042008
|
||||||
|
#define DBG_WWDG_STOP (1 << 11)
|
||||||
|
#define DBG_IWDG_STOP (1 << 12)
|
||||||
|
|
||||||
|
/* This routine uses word access. Only usable on target voltage >2.7V */
|
||||||
static const uint16_t stm32f4_flash_write_stub[] = {
|
static const uint16_t stm32f4_flash_write_stub[] = {
|
||||||
#include "../flashstub/stm32f4.stub"
|
#include "../flashstub/stm32f4.stub"
|
||||||
};
|
};
|
||||||
|
@ -127,7 +137,8 @@ bool stm32f4_probe(target *t)
|
||||||
uint32_t idcode;
|
uint32_t idcode;
|
||||||
|
|
||||||
idcode = target_mem_read32(t, DBGMCU_IDCODE);
|
idcode = target_mem_read32(t, DBGMCU_IDCODE);
|
||||||
switch(idcode & 0xFFF) {
|
idcode &= 0xFFF;
|
||||||
|
switch(idcode) {
|
||||||
case 0x419: /* 427/437 */
|
case 0x419: /* 427/437 */
|
||||||
/* Second bank for 2M parts. */
|
/* Second bank for 2M parts. */
|
||||||
stm32f4_add_flash(t, 0x8100000, 0x10000, 0x4000, 12);
|
stm32f4_add_flash(t, 0x8100000, 0x10000, 0x4000, 12);
|
||||||
|
@ -147,10 +158,22 @@ bool stm32f4_probe(target *t)
|
||||||
stm32f4_add_flash(t, 0x8010000, 0x10000, 0x10000, 4);
|
stm32f4_add_flash(t, 0x8010000, 0x10000, 0x10000, 4);
|
||||||
stm32f4_add_flash(t, 0x8020000, 0xE0000, 0x20000, 5);
|
stm32f4_add_flash(t, 0x8020000, 0xE0000, 0x20000, 5);
|
||||||
target_add_commands(t, stm32f4_cmd_list, "STM32F4");
|
target_add_commands(t, stm32f4_cmd_list, "STM32F4");
|
||||||
return true;
|
break;
|
||||||
}
|
case 0x449: /* F7x6 RM0385 Rev.2 */
|
||||||
|
t->driver = stm32f7_driver_str;
|
||||||
|
target_add_ram(t, 0x00000000, 0x4000);
|
||||||
|
target_add_ram(t, 0x20000000, 0x50000);
|
||||||
|
stm32f4_add_flash(t, 0x8000000, 0x20000, 0x8000, 0);
|
||||||
|
stm32f4_add_flash(t, 0x8020000, 0x20000, 0x20000, 4);
|
||||||
|
stm32f4_add_flash(t, 0x8040000, 0xC0000, 0x40000, 5);
|
||||||
|
target_add_commands(t, stm32f4_cmd_list, "STM32F7");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
t->idcode = idcode;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static void stm32f4_flash_unlock(target *t)
|
static void stm32f4_flash_unlock(target *t)
|
||||||
{
|
{
|
||||||
|
@ -258,7 +281,17 @@ static bool stm32f4_option_write(target *t, uint32_t value)
|
||||||
|
|
||||||
static bool stm32f4_cmd_option(target *t, int argc, char *argv[])
|
static bool stm32f4_cmd_option(target *t, int argc, char *argv[])
|
||||||
{
|
{
|
||||||
uint32_t addr, val;
|
uint32_t start, val;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
if (t->idcode == 0x449) {
|
||||||
|
start = 0x1FFF0000;
|
||||||
|
len = 0x20;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
start = 0x1FFFC000;
|
||||||
|
len = 0x10;
|
||||||
|
}
|
||||||
|
|
||||||
if ((argc == 2) && !strcmp(argv[1], "erase")) {
|
if ((argc == 2) && !strcmp(argv[1], "erase")) {
|
||||||
stm32f4_option_write(t, 0x0fffaaed);
|
stm32f4_option_write(t, 0x0fffaaed);
|
||||||
|
@ -271,8 +304,8 @@ static bool stm32f4_cmd_option(target *t, int argc, char *argv[])
|
||||||
gdb_out("usage: monitor option write <value>\n");
|
gdb_out("usage: monitor option write <value>\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; i < 0xf; i += 8) {
|
for (int i = 0; i < len; i += 8) {
|
||||||
addr = 0x1fffC000 + i;
|
uint32_t addr = start + i;
|
||||||
val = target_mem_read32(t, addr);
|
val = target_mem_read32(t, addr);
|
||||||
gdb_outf("0x%08X: 0x%04X\n", addr, val & 0xFFFF);
|
gdb_outf("0x%08X: 0x%04X\n", addr, val & 0xFFFF);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue