From a9c3ea9cd84aa4e9c79f6999ff098fd89154c502 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Sun, 15 Oct 2023 22:29:12 +0100 Subject: [PATCH 1/9] stm32mp15: Added a stub probe routine for the Cortex-A7 cores so at least the driver name is set properly --- src/target/cortexar.c | 6 ++++++ src/target/stm32mp15.c | 11 +++++++++++ src/target/target_probe.c | 1 + src/target/target_probe.h | 1 + 4 files changed, 19 insertions(+) diff --git a/src/target/cortexar.c b/src/target/cortexar.c index 3abe83cbeb2..a88ba6063c1 100644 --- a/src/target/cortexar.c +++ b/src/target/cortexar.c @@ -808,6 +808,12 @@ bool cortexa_probe(adiv5_access_port_s *const ap, const target_addr_t base_addre if (!target) return false; + switch (target->designer_code) { + case JEP106_MANUFACTURER_STM: + PROBE(stm32mp15_ca7_probe); + break; + } + #if PC_HOSTED == 0 gdb_outf("Please report unknown device with Designer 0x%x Part ID 0x%x\n", target->designer_code, target->part_id); #else diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index ae11884486c..20a44a81f2a 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -111,6 +111,17 @@ bool stm32mp15_cm4_probe(target_s *target) return true; } +#ifdef ENABLE_CORTEXAR +bool stm32mp15_ca7_probe(target_s *target) +{ + if (target->part_id != ID_STM32MP15x && target->part_id != ID_STM32MP15x_ERRATA) + return false; + + target->driver = "STM32MP15"; + return true; +} +#endif + /* * Print the Unique device ID. * Can be reused for other STM32 devices with uid as parameter. diff --git a/src/target/target_probe.c b/src/target/target_probe.c index 4c3e13790e3..741c686b1e2 100644 --- a/src/target/target_probe.c +++ b/src/target/target_probe.c @@ -94,6 +94,7 @@ TARGET_PROBE_WEAK_NOP(stm32f4_probe) TARGET_PROBE_WEAK_NOP(stm32h5_probe) TARGET_PROBE_WEAK_NOP(stm32h7_probe) TARGET_PROBE_WEAK_NOP(stm32mp15_cm4_probe) +TARGET_PROBE_WEAK_NOP(stm32mp15_ca7_probe) TARGET_PROBE_WEAK_NOP(stm32l0_probe) TARGET_PROBE_WEAK_NOP(stm32l1_probe) TARGET_PROBE_WEAK_NOP(stm32l4_probe) diff --git a/src/target/target_probe.h b/src/target/target_probe.h index 23faa2e6b77..dafda11a3be 100644 --- a/src/target/target_probe.h +++ b/src/target/target_probe.h @@ -64,6 +64,7 @@ bool stm32f4_probe(target_s *target); bool stm32h5_probe(target_s *target); bool stm32h7_probe(target_s *target); bool stm32mp15_cm4_probe(target_s *target); +bool stm32mp15_ca7_probe(target_s *target); bool stm32l0_probe(target_s *target); bool stm32l1_probe(target_s *target); bool stm32l4_probe(target_s *target); From 3dbffc5017d0c733861584befda4291c2e4cf539 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Sun, 22 Oct 2023 06:11:08 +0100 Subject: [PATCH 2/9] stm32mp15: Implemented a check for the private storage allocation succeeding --- src/target/stm32mp15.c | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index 20a44a81f2a..6b29efb6188 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -102,6 +102,10 @@ bool stm32mp15_cm4_probe(target_s *target) /* Allocate private storage */ stm32mp15_priv_s *priv = calloc(1, sizeof(*priv)); + if (!priv) { /* calloc failed: heap exhaustion */ + DEBUG_ERROR("calloc: failed in %s\n", __func__); + return false; + } target->target_storage = priv; /* Figure 4. Memory map from §2.5.2 in RM0436 rev 6, pg158 */ From e7f20dd250ac080794e982a396750fc4ba6d229c Mon Sep 17 00:00:00 2001 From: dragonmux Date: Sun, 22 Oct 2023 06:12:40 +0100 Subject: [PATCH 3/9] stm32mp15: Implemented RAM mappings for the Cortex-A7 processors --- src/target/stm32mp15.c | 29 ++++++++++++++++++++++++----- 1 file changed, 24 insertions(+), 5 deletions(-) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index 6b29efb6188..5ff29a881d6 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -3,6 +3,7 @@ * * Copyright (C) 2023 1BitSquared * Written by ALTracer + * Modified by Rachel Mant * * 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 @@ -33,10 +34,16 @@ #include "cortexm.h" /* Memory map constants for STM32MP15x */ -#define STM32MP15_CM4_RETRAM_BASE 0x00000000U -#define STM32MP15_RETRAM_SIZE 0x00001000U /* RETRAM, 64 KiB */ -#define STM32MP15_CM4_AHBSRAM_BASE 0x10000000U -#define STM32MP15_AHBSRAM_SIZE 0x00060000U /* AHB SRAM 1+2+3+4, 128+128+64+64 KiB */ +#define STM32MP15_CM4_RETRAM_BASE 0x00000000U +#define STM32MP15_CA7_RETRAM_BASE 0x38000000U +#define STM32MP15_RETRAM_SIZE 0x00001000U /* RETRAM, 64 KiB */ +#define STM32MP15_AHBSRAM_BASE 0x10000000U +#define STM32MP15_CA7_AHBSRAM_ALIAS_BASE 0x30000000U +#define STM32MP15_AHBSRAM_SIZE 0x00060000U /* AHB SRAM 1+2+3+4, 128+128+64+64 KiB */ +#define STM32MP15_SYSRAM_BASE 0x2ffc0000U +#define STM32MP15_SYSRAM_SIZE 0x00040000U +#define STM32MP15_CAN_SRAM_BASE 0x44011000U +#define STM32MP15_CAN_SRAM_SIZE 0x00002800U /* Access from processor address space. * Access via the debug APB is at 0xe0081000 over AP1. */ @@ -110,7 +117,7 @@ bool stm32mp15_cm4_probe(target_s *target) /* Figure 4. Memory map from §2.5.2 in RM0436 rev 6, pg158 */ target_add_ram(target, STM32MP15_CM4_RETRAM_BASE, STM32MP15_RETRAM_SIZE); - target_add_ram(target, STM32MP15_CM4_AHBSRAM_BASE, STM32MP15_AHBSRAM_SIZE); + target_add_ram(target, STM32MP15_AHBSRAM_BASE, STM32MP15_AHBSRAM_SIZE); return true; } @@ -122,6 +129,18 @@ bool stm32mp15_ca7_probe(target_s *target) return false; target->driver = "STM32MP15"; + + /* Figure 4. Memory map from §2.5.2 in RM0436 rev 6, pg158 */ + target_add_ram(target, STM32MP15_CA7_RETRAM_BASE, STM32MP15_RETRAM_SIZE); + target_add_ram(target, STM32MP15_AHBSRAM_BASE, STM32MP15_AHBSRAM_SIZE); + /* + * The SRAM appears twice in the map as it's mapped to both the main SRAM + * window and the alias window on the Cortex-A7 cores. + * (Unlike the RETRAM which only appears in the alias window) + */ + target_add_ram(target, STM32MP15_CA7_AHBSRAM_ALIAS_BASE, STM32MP15_AHBSRAM_SIZE); + target_add_ram(target, STM32MP15_SYSRAM_BASE, STM32MP15_SYSRAM_SIZE); + target_add_ram(target, STM32MP15_CAN_SRAM_BASE, STM32MP15_CAN_SRAM_SIZE); return true; } #endif From eee89034ee36396a135cfd1416f6d97df21b6962 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Mon, 30 Oct 2023 16:36:11 +0000 Subject: [PATCH 4/9] stm32mp15: Fixed a small mistake in the RETRAM size definition --- src/target/stm32mp15.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index 5ff29a881d6..00e761319d6 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -36,7 +36,7 @@ /* Memory map constants for STM32MP15x */ #define STM32MP15_CM4_RETRAM_BASE 0x00000000U #define STM32MP15_CA7_RETRAM_BASE 0x38000000U -#define STM32MP15_RETRAM_SIZE 0x00001000U /* RETRAM, 64 KiB */ +#define STM32MP15_RETRAM_SIZE 0x00010000U /* RETRAM, 64 KiB */ #define STM32MP15_AHBSRAM_BASE 0x10000000U #define STM32MP15_CA7_AHBSRAM_ALIAS_BASE 0x30000000U #define STM32MP15_AHBSRAM_SIZE 0x00060000U /* AHB SRAM 1+2+3+4, 128+128+64+64 KiB */ From a7bd566506313a79beef6ea95d5bc73bba4ffda1 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Thu, 2 Nov 2023 21:13:02 +0000 Subject: [PATCH 5/9] stm32f1: Fixed the mass_erase target member getting set but not cleaned up after a failed device identification during `stm32f1_probe()` --- src/target/stm32f1.c | 1 + 1 file changed, 1 insertion(+) diff --git a/src/target/stm32f1.c b/src/target/stm32f1.c index 46f651959cd..ca539b3a547 100644 --- a/src/target/stm32f1.c +++ b/src/target/stm32f1.c @@ -537,6 +537,7 @@ bool stm32f1_probe(target_s *target) break; default: /* NONE */ + target->mass_erase = NULL; return false; } From 4b324b150402d733fd9254458723843613789ce8 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Thu, 2 Nov 2023 21:17:54 +0000 Subject: [PATCH 6/9] stm32mp15: Slight code re-organisation to make the reading order more consistent with other targets --- src/target/stm32mp15.c | 49 ++++++++++++++++++++++-------------------- 1 file changed, 26 insertions(+), 23 deletions(-) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index 00e761319d6..fe799626e47 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -50,8 +50,8 @@ #define STM32MP15_DBGMCU_BASE 0x50081000U #define STM32MP15_UID_BASE 0x5c005234U -#define DBGMCU_IDCODE (STM32MP15_DBGMCU_BASE + 0U) -#define DBGMCU_CTRL (STM32MP15_DBGMCU_BASE + 4U) +#define DBGMCU_IDCODE (STM32MP15_DBGMCU_BASE + 0x000U) +#define DBGMCU_CTRL (STM32MP15_DBGMCU_BASE + 0x004U) #define DBGMCU_CTRL_DBGSLEEP (1U << 0U) #define DBGMCU_CTRL_DBGSTOP (1U << 1U) #define DBGMCU_CTRL_DBGSTBY (1U << 2U) @@ -75,27 +75,8 @@ const command_s stm32mp15_cmd_list[] = { {NULL, NULL, NULL}, }; -static bool stm32mp15_attach(target_s *target) -{ - if (!cortexm_attach(target)) - return false; - - /* Save DBGMCU_CR to restore it when detaching */ - stm32mp15_priv_s *const priv = (stm32mp15_priv_s *)target->target_storage; - priv->dbgmcu_ctrl = target_mem_read32(target, DBGMCU_CTRL); - - /* Disable C-Sleep, C-Stop, C-Standby for debugging */ - target_mem_write32(target, DBGMCU_CTRL, DBGMCU_CTRL_DBGSLEEP | DBGMCU_CTRL_DBGSTOP | DBGMCU_CTRL_DBGSTBY); - - return true; -} - -static void stm32mp15_detach(target_s *target) -{ - stm32mp15_priv_s *priv = (stm32mp15_priv_s *)target->target_storage; - target_mem_write32(target, DBGMCU_CTRL, priv->dbgmcu_ctrl); - cortexm_detach(target); -} +static bool stm32mp15_attach(target_s *target); +static void stm32mp15_detach(target_s *target); bool stm32mp15_cm4_probe(target_s *target) { @@ -145,6 +126,28 @@ bool stm32mp15_ca7_probe(target_s *target) } #endif +static bool stm32mp15_attach(target_s *target) +{ + if (!cortexm_attach(target)) + return false; + + /* Save DBGMCU_CR to restore it when detaching */ + stm32mp15_priv_s *const priv = (stm32mp15_priv_s *)target->target_storage; + priv->dbgmcu_ctrl = target_mem_read32(target, DBGMCU_CTRL); + + /* Disable C-Sleep, C-Stop, C-Standby for debugging */ + target_mem_write32(target, DBGMCU_CTRL, DBGMCU_CTRL_DBGSLEEP | DBGMCU_CTRL_DBGSTOP | DBGMCU_CTRL_DBGSTBY); + + return true; +} + +static void stm32mp15_detach(target_s *target) +{ + stm32mp15_priv_s *priv = (stm32mp15_priv_s *)target->target_storage; + target_mem_write32(target, DBGMCU_CTRL, priv->dbgmcu_ctrl); + cortexm_detach(target); +} + /* * Print the Unique device ID. * Can be reused for other STM32 devices with uid as parameter. From da85aef870edadbdbaecd9a7c9ccad9c5d645b69 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Thu, 2 Nov 2023 21:19:51 +0000 Subject: [PATCH 7/9] stm32mp15: Fixed some missing `const` on the function parameters --- src/target/stm32mp15.c | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index fe799626e47..cb641d57622 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -78,7 +78,7 @@ const command_s stm32mp15_cmd_list[] = { static bool stm32mp15_attach(target_s *target); static void stm32mp15_detach(target_s *target); -bool stm32mp15_cm4_probe(target_s *target) +bool stm32mp15_cm4_probe(target_s *const target) { if (target->part_id != ID_STM32MP15x && target->part_id != ID_STM32MP15x_ERRATA) return false; @@ -104,7 +104,7 @@ bool stm32mp15_cm4_probe(target_s *target) } #ifdef ENABLE_CORTEXAR -bool stm32mp15_ca7_probe(target_s *target) +bool stm32mp15_ca7_probe(target_s *const target) { if (target->part_id != ID_STM32MP15x && target->part_id != ID_STM32MP15x_ERRATA) return false; @@ -126,7 +126,7 @@ bool stm32mp15_ca7_probe(target_s *target) } #endif -static bool stm32mp15_attach(target_s *target) +static bool stm32mp15_attach(target_s *const target) { if (!cortexm_attach(target)) return false; @@ -141,7 +141,7 @@ static bool stm32mp15_attach(target_s *target) return true; } -static void stm32mp15_detach(target_s *target) +static void stm32mp15_detach(target_s *const target) { stm32mp15_priv_s *priv = (stm32mp15_priv_s *)target->target_storage; target_mem_write32(target, DBGMCU_CTRL, priv->dbgmcu_ctrl); @@ -152,7 +152,7 @@ static void stm32mp15_detach(target_s *target) * Print the Unique device ID. * Can be reused for other STM32 devices with uid as parameter. */ -static bool stm32mp15_uid(target_s *target, int argc, const char **argv) +static bool stm32mp15_uid(target_s *const target, const int argc, const char **const argv) { (void)argc; (void)argv; @@ -175,7 +175,7 @@ static const struct { {0x2001U, 'Z'}, }; -static bool stm32mp15_cmd_rev(target_s *target, int argc, const char **argv) +static bool stm32mp15_cmd_rev(target_s *const target, const int argc, const char **const argv) { (void)argc; (void)argv; From 1ba980f4ba845b0d439ff5d459b929716af5d1e8 Mon Sep 17 00:00:00 2001 From: dragonmux Date: Thu, 2 Nov 2023 21:31:19 +0000 Subject: [PATCH 8/9] stm32mp15: Fixed up identification of the parts based on the AP part number. --- src/target/stm32mp15.c | 24 +++++++++++++++++++++--- 1 file changed, 21 insertions(+), 3 deletions(-) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index cb641d57622..69f9536337e 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -58,7 +58,7 @@ /* Taken from DP_TARGETID.TPARTNO = 0x5000 in §66.8.3 of RM0436 rev 6, pg3669 */ /* Taken from DBGMCU_IDC.DEV_ID = 0x500 in §66.10.9 of RM0436 rev 6, pg3825 */ -#define ID_STM32MP15x 0x5000U +#define ID_STM32MP15x 0x500U /* Taken from CM4ROM_PIDRx in 2.3.21 of ES0438 rev 7, pg18 */ #define ID_STM32MP15x_ERRATA 0x450U @@ -78,9 +78,27 @@ const command_s stm32mp15_cmd_list[] = { static bool stm32mp15_attach(target_s *target); static void stm32mp15_detach(target_s *target); +static bool stm32mp15_ident(target_s *const target, const bool cortexm) +{ + const adiv5_access_port_s *const ap = cortex_ap(target); + /* Check if the part's a STM32MP15 */ + if (ap->partno != ID_STM32MP15x) { + /* If it's not a Cortex-M core or it doesn't match the errata ID code, return false */ + if (!cortexm || ap->partno != ID_STM32MP15x_ERRATA) + return false; + } + /* + * We now know the part is either a Cortex-M core with the errata code, or matched the main ID code. + * Copy the correct (AP) part number over to the target structure to handle the difference between + * JTAG and SWD as ST has a different ID in the DP TARGETID register vs the ROM tables, which needs ignoring. + */ + target->part_id = ap->partno; + return true; +} + bool stm32mp15_cm4_probe(target_s *const target) { - if (target->part_id != ID_STM32MP15x && target->part_id != ID_STM32MP15x_ERRATA) + if (!stm32mp15_ident(target, true)) return false; target->driver = "STM32MP15"; @@ -106,7 +124,7 @@ bool stm32mp15_cm4_probe(target_s *const target) #ifdef ENABLE_CORTEXAR bool stm32mp15_ca7_probe(target_s *const target) { - if (target->part_id != ID_STM32MP15x && target->part_id != ID_STM32MP15x_ERRATA) + if (!stm32mp15_ident(target, false)) return false; target->driver = "STM32MP15"; From 22249b4ca1266792328ade8f67cc79a604bd426c Mon Sep 17 00:00:00 2001 From: dragonmux Date: Thu, 2 Nov 2023 22:00:50 +0000 Subject: [PATCH 9/9] stm32mp15: Register the target's commands for the Cortex-A cores too --- src/target/stm32mp15.c | 1 + 1 file changed, 1 insertion(+) diff --git a/src/target/stm32mp15.c b/src/target/stm32mp15.c index 69f9536337e..32e8d732e35 100644 --- a/src/target/stm32mp15.c +++ b/src/target/stm32mp15.c @@ -128,6 +128,7 @@ bool stm32mp15_ca7_probe(target_s *const target) return false; target->driver = "STM32MP15"; + target_add_commands(target, stm32mp15_cmd_list, target->driver); /* Figure 4. Memory map from §2.5.2 in RM0436 rev 6, pg158 */ target_add_ram(target, STM32MP15_CA7_RETRAM_BASE, STM32MP15_RETRAM_SIZE);