From c315479a049e5bd0080b9ff0c32ddd1441acb6fa Mon Sep 17 00:00:00 2001 From: Gregory Nutt Date: Thu, 11 Feb 2016 14:11:26 -0600 Subject: [PATCH] SMP: Move IDLE thread init logic from sim to OS; need to assign unique IDs to each IDLE task --- arch | 2 +- include/nuttx/arch.h | 3 +- sched/init/os_smpstart.c | 190 +++++++++++++++++++++++++++++++++++++-- sched/init/os_start.c | 6 +- sched/task/task_create.c | 16 ---- sched/task/task_setup.c | 6 -- 6 files changed, 190 insertions(+), 33 deletions(-) diff --git a/arch b/arch index cf3c8382fa..df987248a7 160000 --- a/arch +++ b/arch @@ -1 +1 @@ -Subproject commit cf3c8382fa36d28b1f153536368cfb07adfbb286 +Subproject commit df987248a73f078f04d532f2b6d2b9ace867926d diff --git a/include/nuttx/arch.h b/include/nuttx/arch.h index 488cbb9815..1eb455ee3d 100644 --- a/include/nuttx/arch.h +++ b/include/nuttx/arch.h @@ -1700,6 +1700,7 @@ int up_cpundx(void); * value in the range of from one to (CONFIG_SMP_NCPUS-1). (CPU * 0 is already active) * idletask - The entry point to the IDLE task. + * pid - Task ID of the IDLE task * * Returned Value: * Zero on success; a negated errno value on failure. @@ -1707,7 +1708,7 @@ int up_cpundx(void); ****************************************************************************/ #ifdef CONFIG_SMP -int up_cpustart(int cpu, main_t idletask); +int up_cpustart(int cpu, main_t idletask, pid_t pid); #endif /**************************************************************************** diff --git a/sched/init/os_smpstart.c b/sched/init/os_smpstart.c index 351e90324a..63164845a3 100644 --- a/sched/init/os_smpstart.c +++ b/sched/init/os_smpstart.c @@ -40,15 +40,39 @@ #include #include +#include +#include +#include #include #include #include +#include +#include "group/group.h" +#include "sched/sched.h" #include "init/init.h" #ifdef CONFIG_SMP +/**************************************************************************** + * Private Types + ****************************************************************************/ + +struct os_tcballoc_s +{ + struct task_tcb_s tcb; /* IDLE task TCB */ + FAR char *idleargv[2]; /* Argument list */ +}; + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +#if CONFIG_TASK_NAME_SIZE < 1 +static const char g_idlename[] = "CPUn Idle" +#endif + /**************************************************************************** * Private Functions ****************************************************************************/ @@ -107,6 +131,130 @@ int os_idletask(int argc, FAR char *argv[]) } } +/**************************************************************************** + * Name: os_idletcb_setup + * + * Description: + * Initialize the IDLE task TCB for this CPU and add it to the correct OS + * task list. The calling sequence here is: + * + * 1. os_start() - Initializes the system and calls os_smpstart() + * 2. os_smpstart() - Calls up_cpustart() for each CPU 1..(CONFIG_SMP_NCPUS-1) + * 3. up_cpustart() - Calls os_idletcb_setup() when appropriate to configure + * the OS data structures. + * 4. up_cpustart() - may also call os_idletcb_teardown() to recover from + * from any errors. + * + * Input Parameters: + * cpu - The CPU to undel + * idle - Memory allocated by os_idletcb_setup + * pid - Task ID of the IDLE task + * + * Returned Value: + * Memory allocated by os_idletcb_setup (for error recovery). NULL is + * returned only a a failure to allocate memory. + * + ****************************************************************************/ + +static FAR void *os_idletcb_setup(int cpu, main_t idletask, pid_t pid) +{ + FAR struct os_tcballoc_s *alloc; + FAR struct task_tcb_s *itcb; + dq_queue_t *tasklist; + + /* IDLE TCB Initialization ************************************************/ + /* Allocate and clear the IDLE TCB */ + + alloc = (FAR struct os_tcballoc_s *)kmm_zalloc(sizeof(struct os_tcballoc_s)); + if (alloc == NULL) + { + return NULL; + } + + /* Initialize the TCB for the IDLE task on the stack. + * REVISIT: We should be able to use task_schedsetup() to do most of this + */ + + itcb = &alloc->tcb; + itcb->cmn.task_state = TSTATE_TASK_RUNNING; + itcb->cmn.entry.main = idletask; + itcb->cmn.flags = (TCB_FLAG_TTYPE_KERNEL | TCB_FLAG_CPU_ASSIGNED); + itcb->cmn.cpu = cpu; + +#if CONFIG_TASK_NAME_SIZE > 0 + snprintf(itcb->cmn.name, CONFIG_TASK_NAME_SIZE, "CPU%d IDLE", cpu); + alloc->idleargv[0] = itcb->cmn.name; +#else + alloc->idleargv[0] = (FAR char *)g_idlename; +#endif + alloc->idleargv[1] = NULL; + itcb->argv = alloc->idleargv; + + /* Add the IDLE task TCB to the end of the assigned task list */ + + tasklist = TLIST_HEAD(TSTATE_TASK_RUNNING, cpu); + dq_addfirst((FAR dq_entry_t *)itcb, tasklist); + + /* Initialize the processor-specific portion of the TCB */ + + up_initial_state(&itcb->cmn); + + /* PID assignment *********************************************************/ + + g_pidhash[PIDHASH(pid)].tcb = &itcb->cmn; + + /* IDLE Group Initialization **********************************************/ +#ifdef HAVE_TASK_GROUP + /* Allocate the IDLE group */ + + DEBUGVERIFY(group_allocate(itcb, itcb->cmn.flags)); +#endif + +#if CONFIG_NFILE_DESCRIPTORS > 0 || CONFIG_NSOCKET_DESCRIPTORS > 0 + /* Create stdout, stderr, stdin on the IDLE task. These will be + * inherited by all of the threads created by the IDLE task. + */ + + DEBUGVERIFY(group_setupidlefiles(itcb)); +#endif + +#ifdef HAVE_TASK_GROUP + /* Complete initialization of the IDLE group. Suppress retention + * of child status in the IDLE group. + */ + + DEBUGVERIFY(group_initialize(itcb)); + itcb->cmn.group->tg_flags = GROUP_FLAG_NOCLDWAIT; +#endif + + return alloc; +} + +/**************************************************************************** + * Name: os_idletcb_teardown + * + * Description: + * Undo what os_idletcb_setup() did. Necessary only for error recovery. + * + * Input Parameters: + * cpu - The CPU to undel + * alloc - Memory allocated by os_idletcb_setup + * + * Returned Value: + * None + * + ****************************************************************************/ + +static void os_idletcb_teardown(int cpu, FAR void *alloc) +{ + /* Undo what os_idletcb_setup() did */ + + FAR dq_queue_t *tasklist= TLIST_HEAD(TSTATE_TASK_RUNNING, 0); + dq_init(tasklist); + + kmm_free(alloc); +} + /**************************************************************************** * Public Functions ****************************************************************************/ @@ -126,25 +274,51 @@ int os_idletask(int argc, FAR char *argv[]) * Returned Value: * Zero on success; a negater errno value on failure. * + * Assumption: + * Runs before the full initialization sequence has completed. Runs after + * all OS facilities are set up, but before multi-tasking has been started. + * ****************************************************************************/ int os_smpstart(void) { + FAR void *alloc; + pid_t pid; int ret; - int i; + int cpu; - for (i = 1; i < CONFIG_SMP_NCPUS; i++) + /* Reserve PIDs for IDLE tasks 1..(CONFIG_SMP_NCPUS-1). We do not have to + * be careful here because so far because there is only one CPU running and + * we have not yet started multi-tasking. + */ + + pid = g_lastpid + 1; /* Should be 1 */ + g_lastpid += CONFIG_SMP_NCPUS; /* Should be CONFIG_SMP_NCPUS */ + + /* CPU0 is already running. Start the remaining CPUs */ + + for (cpu = 1; cpu < CONFIG_SMP_NCPUS; cpu++, pid++) { - ret = up_cpustart(i, os_idletask); -#ifdef CONFIG_DEBUG + /* Assign a PID to the IDLE task and set up the IDLE thread TCB */ + + alloc = os_idletcb_setup(cpu, os_idletask, pid); + if (alloc == NULL) + { + return -ENOMEM; + } + + /* And start the CPU. */ + + ret = up_cpustart(cpu, os_idletask, pid); if (ret < 0) { - sdbg("ERROR: Failed to start CPU%d: %d\n", i, ret); + sdbg("ERROR: Failed to start CPU%d: %d\n", cpu, ret); + + /* Undo what os_idletcb_setup() did and return the failure*/ + + os_idletcb_teardown(cpu, alloc); return ret; } -#else - UNUSED(ret); -#endif } return OK; diff --git a/sched/init/os_start.c b/sched/init/os_start.c index c0caabc8f7..4a5b776d93 100644 --- a/sched/init/os_start.c +++ b/sched/init/os_start.c @@ -304,7 +304,11 @@ static struct task_tcb_s g_idletcb; /* This is the name of the idle task */ +#ifdef CONFIG_SMP +static const char g_idlename[] = "CPU0 Idle"; +#else static const char g_idlename[] = "Idle Task"; +#endif /* This the IDLE idle threads argument list. */ @@ -650,7 +654,7 @@ void os_start(void) #ifdef CONFIG_SMP /* Start all CPUs *********************************************************/ - DEBUGASSERT(this_cpu() == 0); + DEBUGASSERT(this_cpu() == 0 && CONFIG_MAX_TASKS > CONFIG_SMP_NCPUS); DEBUGVERIFY(os_smpstart()); #endif /* CONFIG_SMP */ diff --git a/sched/task/task_create.c b/sched/task/task_create.c index e853b7d521..1ced3b9515 100644 --- a/sched/task/task_create.c +++ b/sched/task/task_create.c @@ -52,22 +52,6 @@ #include "group/group.h" #include "task/task.h" -/**************************************************************************** - * Pre-processor Definitions - ****************************************************************************/ - -/**************************************************************************** - * Private Type Declarations - ****************************************************************************/ - -/**************************************************************************** - * Public Data - ****************************************************************************/ - -/**************************************************************************** - * Private Variables - ****************************************************************************/ - /**************************************************************************** * Private Functions ****************************************************************************/ diff --git a/sched/task/task_setup.c b/sched/task/task_setup.c index d7730c6509..ee87fd93a9 100644 --- a/sched/task/task_setup.c +++ b/sched/task/task_setup.c @@ -70,12 +70,6 @@ static const char g_noname[] = ""; -/**************************************************************************** - * Private Function Prototypes - ****************************************************************************/ - -static int task_assignpid(FAR struct tcb_s * tcb); - /**************************************************************************** * Private Functions ****************************************************************************/