Prepare v2023.10
[platform/kernel/u-boot.git] / arch / arm / mach-imx / imx_bootaux.c
index 30fb45d..f7b14ca 100644 (file)
 #include <linux/compiler.h>
 #include <cpu_func.h>
 
-#ifndef CONFIG_IMX8M
-const __weak struct rproc_att hostmap[] = { };
+#ifndef CONFIG_IMX8
+/* Just to avoid build error */
+#if IS_ENABLED(CONFIG_IMX8M)
+#define SRC_M4C_NON_SCLR_RST_MASK      BIT(0)
+#define SRC_M4_ENABLE_MASK             BIT(0)
+#define SRC_M4_REG_OFFSET              0
+#endif
+
+__weak const struct rproc_att *imx_bootaux_get_hostmap(void)
+{
+       return NULL;
+}
 
 static const struct rproc_att *get_host_mapping(unsigned long auxcore)
 {
-       const struct rproc_att *mmap = hostmap;
+       const struct rproc_att *mmap = imx_bootaux_get_hostmap();
 
        while (mmap && mmap->size) {
                if (mmap->da <= auxcore &&
@@ -36,10 +46,11 @@ static const struct rproc_att *get_host_mapping(unsigned long auxcore)
  * is valid, returns the entry point address.
  * Translates load addresses in the elf file to the U-Boot address space.
  */
-static unsigned long load_elf_image_m_core_phdr(unsigned long addr)
+static u32 load_elf_image_m_core_phdr(unsigned long addr, u32 *stack)
 {
        Elf32_Ehdr *ehdr; /* ELF header structure pointer */
        Elf32_Phdr *phdr; /* Program header structure pointer */
+       int num = 0;
        int i;
 
        ehdr = (Elf32_Ehdr *)addr;
@@ -54,19 +65,24 @@ static unsigned long load_elf_image_m_core_phdr(unsigned long addr)
                        continue;
 
                if (!mmap) {
-                       printf("Invalid aux core address: %08x",
+                       printf("Invalid aux core address: %08x\n",
                               phdr->p_paddr);
                        return 0;
                }
 
-               dst = (void *)(phdr->p_paddr - mmap->da) + mmap->sa;
+               dst = (void *)(ulong)(phdr->p_paddr - mmap->da) + mmap->sa;
                src = (void *)addr + phdr->p_offset;
 
                debug("Loading phdr %i to 0x%p (%i bytes)\n",
                      i, dst, phdr->p_filesz);
 
-               if (phdr->p_filesz)
+               if (phdr->p_filesz) {
                        memcpy(dst, src, phdr->p_filesz);
+                       /* Stack in __isr_vector is the first section/word */
+                       if (!num)
+                               *stack = *(uint32_t *)src;
+                       num++;
+               }
                if (phdr->p_filesz != phdr->p_memsz)
                        memset(dst + phdr->p_filesz, 0x00,
                               phdr->p_memsz - phdr->p_filesz);
@@ -77,29 +93,25 @@ static unsigned long load_elf_image_m_core_phdr(unsigned long addr)
 
        return ehdr->e_entry;
 }
-#endif
 
 int arch_auxiliary_core_up(u32 core_id, ulong addr)
 {
-       ulong stack, pc;
+       u32 stack, pc;
 
        if (!addr)
                return -EINVAL;
 
-#ifdef CONFIG_IMX8M
-       stack = *(u32 *)addr;
-       pc = *(u32 *)(addr + 4);
-#else
        /*
         * handling ELF64 binaries
         * isn't supported yet.
         */
        if (valid_elf_image(addr)) {
-               stack = 0x0;
-               pc = load_elf_image_m_core_phdr(addr);
+               pc = load_elf_image_m_core_phdr(addr, &stack);
                if (!pc)
                        return CMD_RET_FAILURE;
 
+               if (!IS_ENABLED(CONFIG_ARM64))
+                       stack = 0x0;
        } else {
                /*
                 * Assume binary file with vector table at the beginning.
@@ -109,83 +121,83 @@ int arch_auxiliary_core_up(u32 core_id, ulong addr)
                stack = *(u32 *)addr;
                pc = *(u32 *)(addr + 4);
        }
-#endif
-       printf("## Starting auxiliary core stack = 0x%08lX, pc = 0x%08lX...\n",
+
+       printf("## Starting auxiliary core stack = 0x%08X, pc = 0x%08X...\n",
               stack, pc);
 
-       /* Set the stack and pc to M4 bootROM */
-       writel(stack, M4_BOOTROM_BASE_ADDR);
-       writel(pc, M4_BOOTROM_BASE_ADDR + 4);
+       /* Set the stack and pc to MCU bootROM */
+       writel(stack, MCU_BOOTROM_BASE_ADDR);
+       writel(pc, MCU_BOOTROM_BASE_ADDR + 4);
 
        flush_dcache_all();
 
-       /* Enable M4 */
-#ifdef CONFIG_IMX8M
-       arm_smccc_smc(IMX_SIP_SRC, IMX_SIP_SRC_M4_START, 0, 0,
-                     0, 0, 0, 0, NULL);
-#else
-       clrsetbits_le32(SRC_BASE_ADDR + SRC_M4_REG_OFFSET,
-                       SRC_M4C_NON_SCLR_RST_MASK, SRC_M4_ENABLE_MASK);
-#endif
+       /* Enable MCU */
+       if (IS_ENABLED(CONFIG_IMX8M)) {
+               arm_smccc_smc(IMX_SIP_SRC, IMX_SIP_SRC_MCU_START, 0, 0, 0, 0, 0, 0, NULL);
+       } else {
+               clrsetbits_le32(SRC_BASE_ADDR + SRC_M4_REG_OFFSET,
+                               SRC_M4C_NON_SCLR_RST_MASK, SRC_M4_ENABLE_MASK);
+       }
 
        return 0;
 }
 
 int arch_auxiliary_core_check_up(u32 core_id)
 {
-#ifdef CONFIG_IMX8M
        struct arm_smccc_res res;
-
-       arm_smccc_smc(IMX_SIP_SRC, IMX_SIP_SRC_M4_STARTED, 0, 0,
-                     0, 0, 0, 0, &res);
-
-       return res.a0;
-#else
        unsigned int val;
 
+       if (IS_ENABLED(CONFIG_IMX8M)) {
+               arm_smccc_smc(IMX_SIP_SRC, IMX_SIP_SRC_MCU_STARTED, 0, 0, 0, 0, 0, 0, &res);
+               return res.a0;
+       }
+
        val = readl(SRC_BASE_ADDR + SRC_M4_REG_OFFSET);
 
        if (val & SRC_M4C_NON_SCLR_RST_MASK)
                return 0;  /* assert in reset */
 
        return 1;
-#endif
 }
-
+#endif
 /*
  * To i.MX6SX and i.MX7D, the image supported by bootaux needs
  * the reset vector at the head for the image, with SP and PC
  * as the first two words.
  *
- * Per the cortex-M reference manual, the reset vector of M4 needs
- * to exist at 0x0 (TCMUL). The PC and SP are the first two addresses
- * of that vector.  So to boot M4, the A core must build the M4's reset
+ * Per the cortex-M reference manual, the reset vector of M4/M7 needs
+ * to exist at 0x0 (TCMUL/IDTCM). The PC and SP are the first two addresses
+ * of that vector.  So to boot M4/M7, the A core must build the M4/M7's reset
  * vector with getting the PC and SP from image and filling them to
- * TCMUL. When M4 is kicked, it will load the PC and SP by itself.
- * The TCMUL is mapped to (M4_BOOTROM_BASE_ADDR) at A core side for
- * accessing the M4 TCMUL.
+ * TCMUL/IDTCM. When M4/M7 is kicked, it will load the PC and SP by itself.
+ * The TCMUL/IDTCM is mapped to (MCU_BOOTROM_BASE_ADDR) at A core side for
+ * accessing the M4/M7 TCMUL/IDTCM.
  */
 static int do_bootaux(struct cmd_tbl *cmdtp, int flag, int argc,
                      char *const argv[])
 {
        ulong addr;
        int ret, up;
+       u32 core = 0;
 
        if (argc < 2)
                return CMD_RET_USAGE;
 
-       up = arch_auxiliary_core_check_up(0);
+       if (argc > 2)
+               core = simple_strtoul(argv[2], NULL, 10);
+
+       up = arch_auxiliary_core_check_up(core);
        if (up) {
                printf("## Auxiliary core is already up\n");
                return CMD_RET_SUCCESS;
        }
 
-       addr = simple_strtoul(argv[1], NULL, 16);
+       addr = hextoul(argv[1], NULL);
 
        if (!addr)
                return CMD_RET_FAILURE;
 
-       ret = arch_auxiliary_core_up(0, addr);
+       ret = arch_auxiliary_core_up(core, addr);
        if (ret)
                return CMD_RET_FAILURE;
 
@@ -195,5 +207,7 @@ static int do_bootaux(struct cmd_tbl *cmdtp, int flag, int argc,
 U_BOOT_CMD(
        bootaux, CONFIG_SYS_MAXARGS, 1, do_bootaux,
        "Start auxiliary core",
-       ""
+       "<address> [<core>]\n"
+       "   - start auxiliary core [<core>] (default 0),\n"
+       "     at address <address>\n"
 );