From b087cb7f9adffab22a9cdb5e7f5b4525d6b10c33 Mon Sep 17 00:00:00 2001 From: Gregory Nutt Date: Thu, 11 Feb 2016 12:16:19 -0600 Subject: [PATCH] SIM: Add SMP idle thread setup logic --- arch/sim/include/spinlock.h | 2 ++ arch/sim/src/Makefile | 9 +++++++++ arch/sim/src/up_idle.c | 27 +++++++++++++++++++++++++-- arch/sim/src/up_internal.h | 4 ++++ arch/sim/src/up_simsmp.c | 18 ++++++++++-------- 5 files changed, 50 insertions(+), 10 deletions(-) diff --git a/arch/sim/include/spinlock.h b/arch/sim/include/spinlock.h index f6ea21e13c..3c20ffe22a 100644 --- a/arch/sim/include/spinlock.h +++ b/arch/sim/include/spinlock.h @@ -46,6 +46,7 @@ /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ +/* Must match definitions in up_testset.c */ #define SP_UNLOCKED false /* The Un-locked state */ #define SP_LOCKED true /* The Locked state */ @@ -53,6 +54,7 @@ /**************************************************************************** * Public Types ****************************************************************************/ +/* Must match definitions in up_testset.c */ typedef bool spinlock_t; diff --git a/arch/sim/src/Makefile b/arch/sim/src/Makefile index 94169fa1ba..12d8794a7d 100644 --- a/arch/sim/src/Makefile +++ b/arch/sim/src/Makefile @@ -65,7 +65,12 @@ ifeq ($(CONFIG_SCHED_TICKLESS),y) CSRCS += up_tickless.c endif +ifeq ($(CONFIG_SPINLOCK),y) + HOSTSRCS += up_testset.c +endif + ifeq ($(CONFIG_SMP),y) + CSRCS += up_idlesetup.c HOSTSRCS += up_simsmp.c endif @@ -115,6 +120,10 @@ else endif endif +ifeq ($(CONFIG_SMP),y) + HOSTCFLAGS += -DCONFIG_SMP=1 +endif + ifeq ($(CONFIG_FS_HOSTFS),y) HOSTSRCS += up_hostfs.c diff --git a/arch/sim/src/up_idle.c b/arch/sim/src/up_idle.c index 3cc7875b99..b81fd4fc6c 100644 --- a/arch/sim/src/up_idle.c +++ b/arch/sim/src/up_idle.c @@ -1,7 +1,7 @@ /**************************************************************************** * arch/sim/src/up_idle.c * - * Copyright (C) 2007-2009, 2011-2012, 2014 Gregory Nutt. All rights reserved. + * Copyright (C) 2007-2009, 2011-2012, 2014, 2016 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * Redistribution and use in source and binary forms, with or without @@ -42,6 +42,7 @@ #include #include +#include #include #include "up_internal.h" @@ -94,6 +95,23 @@ extern void up_x11update(void); void up_idle(void) { +#ifdef CONFIG_SMP + /* In the SMP configuration, only one CPU should do these operations. It + * should not matter which, however. + */ + + static volatile spinlock_t lock; + + /* The one that gets the lock is the one that executes the IDLE operations */ + + if (up_testset(&lock) != SP_UNLOCKED) + { + /* We didn't get it... just return and try again later */ + + return; + } +#endif + #ifdef CONFIG_SCHED_TICKLESS /* Driver the simulated interval timer */ @@ -172,5 +190,10 @@ void up_idle(void) } #endif #endif -} +#ifdef CONFIG_SMP + /* Release the spinlock and return */ + + lock = SP_UNLOCKED; +#endif +} diff --git a/arch/sim/src/up_internal.h b/arch/sim/src/up_internal.h index e0cdcbe917..4f723c1599 100644 --- a/arch/sim/src/up_internal.h +++ b/arch/sim/src/up_internal.h @@ -49,6 +49,9 @@ # include # include +# ifdef CONFIG_SMP +# include +# endif #endif /**************************************************************************** @@ -214,6 +217,7 @@ void up_longjmp(xcpt_reg_t *jb, int val) noreturn_function; #ifdef CONFIG_SMP int sim_cpu0initialize(void); +int sim_cpustart(int cpu, main_t idletask); #endif /* up_tickless.c **********************************************************/ diff --git a/arch/sim/src/up_simsmp.c b/arch/sim/src/up_simsmp.c index 735e3a9168..bf8f9c48c2 100644 --- a/arch/sim/src/up_simsmp.c +++ b/arch/sim/src/up_simsmp.c @@ -69,7 +69,7 @@ static pthread_key_t g_cpukey; * Description: * This is a pthread task entry point. A (host) pthread is used to * simulate a CPU. Multiple pthreads is a good analog to tasks running on - * multiple CPUs + * multiple CPUs * * This function is simply a wrapper that sets the pthread specific data * that presents the CPU number and then calls into the IDLE task entry @@ -171,7 +171,7 @@ int up_cpundx(void) } /**************************************************************************** - * Name: up_cpustart + * Name: sim_cpustart * * Description: * In an SMP configution, only one CPU is initially active (CPU 0). System @@ -194,7 +194,7 @@ int up_cpundx(void) * ****************************************************************************/ -int up_cpustart(int cpu, main_t idletask) +int sim_cpustart(int cpu, main_t idletask) { struct sim_cpuinfo_s cpuinfo; pthread_t thread; @@ -207,7 +207,7 @@ int up_cpustart(int cpu, main_t idletask) ret = pthread_mutex_init(&cpuinfo.mutex, NULL); if (ret != 0) { - return -ret; + return -ret; /* REVISIT: That is a host errno value. */ } /* Lock the mutex */ @@ -215,16 +215,18 @@ int up_cpustart(int cpu, main_t idletask) ret = pthread_mutex_lock(&cpuinfo.mutex); if (ret != 0) { - ret = -ret; + ret = -ret; /* REVISIT: This is a host errno value. */ goto errout_with_mutex; } - /* Start the CPU emulation thread */ + /* Start the CPU emulation thread. This is analogous to starting the CPU + * in a multi-CPU hardware model. + */ ret = pthread_create(&thread, NULL, sim_idle_trampoline, &cpuinfo); if (ret != 0) { - ret = -ret; + ret = -ret; /* REVISIT: That is a host errno value. */ goto errout_with_lock; } @@ -235,7 +237,7 @@ int up_cpustart(int cpu, main_t idletask) ret = pthread_mutex_lock(&cpuinfo.mutex); if (ret != 0) { - ret = -ret; + ret = -ret; /* REVISIT: That is a host errno value. */ } errout_with_lock: