Xtensa: Add CPU1 start logic
This commit is contained in:
parent
c1334048c5
commit
51fc3de40b
3 changed files with 236 additions and 9 deletions
|
@ -35,7 +35,7 @@
|
|||
|
||||
# The start-up, "head", file
|
||||
|
||||
HEAD_ASRC =
|
||||
HEAD_ASRC = esp32_start.c
|
||||
|
||||
# Common XTENSA files (arch/xtensa/src/common)
|
||||
|
||||
|
@ -50,8 +50,9 @@ CMN_CSRCS += xtensa_stackframe.c xtensa_udelay.c xtensa_usestack.c
|
|||
# Configuration-dependent common XTENSA files
|
||||
|
||||
# Use of common/xtensa_etherstub.c is deprecated. The preferred mechanism
|
||||
# is to use CONFIG_NETDEV_LATEINIT=y to suppress the call to up_netinitialize()
|
||||
# in xtensa_initialize.c. Then this stub would not be needed.
|
||||
# is to use CONFIG_NETDEV_LATEINIT=y to suppress the call to
|
||||
# up_netinitialize() in xtensa_initialize.c. Then this stub would not be
|
||||
# needed.
|
||||
|
||||
ifneq ($(CONFIG_LX6_ETHERNET),y)
|
||||
ifeq ($(CONFIG_NET),y)
|
||||
|
@ -59,9 +60,15 @@ ifeq ($(CONFIG_NET),y)
|
|||
endif
|
||||
endif
|
||||
|
||||
# Required LX6 files (arch/xtensa/src/lx6)
|
||||
# Required ESP32 files (arch/xtensa/src/lx6)
|
||||
|
||||
CHIP_ASRCS =
|
||||
CHIP_CSRCS =
|
||||
|
||||
# Configuration-dependent LX6 files
|
||||
# Configuration-dependent ESP32 files
|
||||
|
||||
ifeq ($(CONFIG_SMP),y)
|
||||
#CMN_CSRCS += esp32_cpuindex.c esp32_cpustart.c esp32_cpupause.c
|
||||
#CMN_CSRCS += esp32_cpuidlestack.c
|
||||
CMN_CSRCS += esp32_cpustart.c
|
||||
endif
|
||||
|
|
220
arch/xtensa/src/esp32/esp32_cpustart.c
Normal file
220
arch/xtensa/src/esp32/esp32_cpustart.c
Normal file
|
@ -0,0 +1,220 @@
|
|||
/****************************************************************************
|
||||
* arch/xtensa/src/esp32/esp32_cpustart.c
|
||||
*
|
||||
* Copyright (C) 2016 Gregory Nutt. All rights reserved.
|
||||
* Author: Gregory Nutt <gnutt@nuttx.org>
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
*
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in
|
||||
* the documentation and/or other materials provided with the
|
||||
* distribution.
|
||||
* 3. Neither the name NuttX nor the names of its contributors may be
|
||||
* used to endorse or promote products derived from this software
|
||||
* without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||||
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
||||
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||||
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||||
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||
* POSSIBILITY OF SUCH DAMAGE.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include <nuttx/config.h>
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include <semaphore.h>
|
||||
#include <debug.h>
|
||||
|
||||
#include <nuttx/arch.h>
|
||||
#include <nuttx/sched.h>
|
||||
|
||||
#include "sched/sched.h"
|
||||
#include "xtensa.h"
|
||||
|
||||
#ifdef CONFIG_SMP
|
||||
|
||||
#warning REVISIT Need cpu_configure_region_protection() prototype
|
||||
void cpu_configure_region_protection(void);
|
||||
#warning REVISIT Need ets_set_appcpu_boot_addr() prototype
|
||||
void ets_set_appcpu_boot_addr(uint32_t);
|
||||
|
||||
/****************************************************************************
|
||||
* Private Data
|
||||
****************************************************************************/
|
||||
|
||||
static bool g_appcpu_started;
|
||||
static sem_t g_appcpu_interlock;
|
||||
|
||||
/****************************************************************************
|
||||
* Private Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: xtensa_registerdump
|
||||
****************************************************************************/
|
||||
|
||||
#if 0 /* Was useful in solving some startup problems */
|
||||
static inline void xtensa_registerdump(FAR struct tcb_s *tcb)
|
||||
{
|
||||
_info("CPU%d:\n", up_cpu_index());
|
||||
|
||||
/* Dump the startup registers */
|
||||
/* To be provided */
|
||||
}
|
||||
#else
|
||||
# define xtensa_registerdump(tcb)
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: xtensa_start_handler
|
||||
*
|
||||
* Description:
|
||||
* This is the handler for SGI1. This handler simply returns from the
|
||||
* interrupt, restoring the state of the new task at the head of the ready
|
||||
* to run list.
|
||||
*
|
||||
* Input Parameters:
|
||||
* Standard interrupt handling
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero on success; a negated errno value on failure.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
int xtensa_start_handler(int irq, FAR void *context)
|
||||
{
|
||||
FAR struct tcb_s *tcb;
|
||||
|
||||
sinfo("CPU%d Started\n", up_cpu_index());
|
||||
|
||||
/* Handle interlock*/
|
||||
|
||||
g_appcpu_started = true;
|
||||
sem_post(&g_appcpu_interlock);
|
||||
|
||||
/* Reset scheduler parameters */
|
||||
|
||||
tcb = this_task();
|
||||
sched_resume_scheduler(tcb);
|
||||
|
||||
/* Move CPU0 exception vectors to IRAM */
|
||||
|
||||
asm volatile ("wsr %0, vecbase\n"::"r" (&_init_start));
|
||||
|
||||
/* Make page 0 access raise an exception */
|
||||
|
||||
cpu_configure_region_protection();
|
||||
|
||||
/* Dump registers so that we can see what is going to happen on return */
|
||||
|
||||
xtensa_registerdump(tcb);
|
||||
|
||||
/* Then switch contexts. This instantiates the exception context of the
|
||||
* tcb at the head of the assigned task list. In this case, this should
|
||||
* be the CPUs NULL task.
|
||||
*/
|
||||
|
||||
up_restorestate(tcb->xcp.regs);
|
||||
return OK;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: up_cpu_start
|
||||
*
|
||||
* Description:
|
||||
* In an SMP configution, only one CPU is initially active (CPU 0). System
|
||||
* initialization occurs on that single thread. At the completion of the
|
||||
* initialization of the OS, just before beginning normal multitasking,
|
||||
* the additional CPUs would be started by calling this function.
|
||||
*
|
||||
* Each CPU is provided the entry point to is IDLE task when started. A
|
||||
* TCB for each CPU's IDLE task has been initialized and placed in the
|
||||
* CPU's g_assignedtasks[cpu] list. Not stack has been alloced or
|
||||
* initialized.
|
||||
*
|
||||
* The OS initialization logic calls this function repeatedly until each
|
||||
* CPU has been started, 1 through (CONFIG_SMP_NCPUS-1).
|
||||
*
|
||||
* Input Parameters:
|
||||
* cpu - The index of the CPU being started. This will be a numeric
|
||||
* value in the range of from one to (CONFIG_SMP_NCPUS-1). (CPU
|
||||
* 0 is already active)
|
||||
*
|
||||
* Returned Value:
|
||||
* Zero on success; a negated errno value on failure.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
int up_cpu_start(int cpu)
|
||||
{
|
||||
DEBUGASSERT(cpu >= 0 && cpu < CONFIG_SMP_NCPUS && cpu != this_cpu());
|
||||
|
||||
if (!g_appcpu_started)
|
||||
{
|
||||
uint32_t regval;
|
||||
int ret;
|
||||
|
||||
/* Start CPU1 */
|
||||
|
||||
sinfo("Starting CPU%d\n", cpu);
|
||||
sem_init(&g_appcpu_interlock, 0, 0)
|
||||
|
||||
regval = getreg32(DPORT_APPCPU_CTRL_B_REG);
|
||||
regval |= DPORT_APPCPU_CLKGATE_EN;
|
||||
putreg32(regval, DPORT_APPCPU_CTRL_B_REG);
|
||||
|
||||
regval = getreg32(DPORT_APPCPU_CTRL_C_REG);
|
||||
regval ~= DPORT_APPCPU_RUNSTALL;
|
||||
putreg32(regval, DPORT_APPCPU_CTRL_C_REG);
|
||||
|
||||
regval = getreg32(DPORT_APPCPU_CTRL_A_REG);
|
||||
regval |= DPORT_APPCPU_RESETTING;
|
||||
putreg32(regval, DPORT_APPCPU_CTRL_A_REG);
|
||||
|
||||
regval = getreg32(DPORT_APPCPU_CTRL_A_REG);
|
||||
regval &= ~DPORT_APPCPU_RESETTING;
|
||||
putreg32(regval, DPORT_APPCPU_CTRL_A_REG);
|
||||
|
||||
/* Set the CPU1 start address */
|
||||
|
||||
ets_set_appcpu_boot_addr((uint32_t)xtensa_start_handler);
|
||||
|
||||
/* And way for the initial task to run on CPU1 */
|
||||
|
||||
while (!app_cpu_started)
|
||||
{
|
||||
ret = sem_wait(&g_appcpu_interlock);
|
||||
if (ret < 0)
|
||||
{
|
||||
DEBUGASSERT(errno == EINTR);
|
||||
}
|
||||
}
|
||||
|
||||
sem_destroy(&g_appcpu_interlock);
|
||||
}
|
||||
}
|
||||
|
||||
#endif /* CONFIG_SMP */
|
|
@ -66,20 +66,20 @@ void IRAM_ATTR __start()
|
|||
/* Kill the watchdog timer */
|
||||
|
||||
regval = getreg32(RTC_CNTL_WDTCONFIG0_REG);
|
||||
regval ~= RTC_CNTL_WDT_FLASHBOOT_MOD_EN;
|
||||
regval &= ~RTC_CNTL_WDT_FLASHBOOT_MOD_EN;
|
||||
putreg32(regval, RTC_CNTL_WDTCONFIG0_REG);
|
||||
|
||||
regval = getreg32(0x6001f048); /* DR_REG_BB_BASE+48 */
|
||||
regval ~= (1 << 14);
|
||||
regval &= ~(1 << 14);
|
||||
putreg32(regval, 0x6001f048);
|
||||
|
||||
/* Make page 0 access raise an exception */
|
||||
|
||||
cpu_configure_region_protection();
|
||||
|
||||
/* Move exception vectors to IRAM */
|
||||
/* Move CPU0 exception vectors to IRAM */
|
||||
|
||||
asm volatile ("wsr %0, vecbase\n"::"r" (&_init_start));
|
||||
asm volatile ("wsr %0, vecbase\n"::"r" (&_init_start));
|
||||
|
||||
/* Set .bss to zero */
|
||||
|
||||
|
|
Loading…
Reference in a new issue