arch, include, sched : Refactor ARCH_GLOBAL_IRQDISABLE related code
Summary: - ARCH_GLOBAL_IRQDISABLE was initially introduced for LC823450 SMP - At that time, i.MX6 (quad Cortex-A9) did not use this config - However, this option is now used for all CPUs which support SMP - So it's good timing for refactoring the code Impact: - Should have no impact because the logic is the same for SMP Testing: - Tested with board: spresense:smp, spresense:wifi_smp - Tested with qemu: esp32-core:smp, maix-bit:smp, sabre-6quad:smp - Build only: lc823450-xgevk:rndis, sam4cmp-db:nsh Signed-off-by: Masayuki Ishikawa <Masayuki.Ishikawa@jp.sony.com>
This commit is contained in:
parent
74aec1c424
commit
08c4376606
@ -277,15 +277,6 @@ config ARCH_HAVE_RTC_SUBSECONDS
|
|||||||
bool
|
bool
|
||||||
default n
|
default n
|
||||||
|
|
||||||
config ARCH_GLOBAL_IRQDISABLE
|
|
||||||
bool
|
|
||||||
default n
|
|
||||||
---help---
|
|
||||||
Indicates that disabling interrupts on one CPU will either (1) disable
|
|
||||||
all interrupts globally on all CPUs, or (2) will disable interprocessor
|
|
||||||
interrupts as well so that no context switches can occur on the CPU
|
|
||||||
that disabled "local" interrupts.
|
|
||||||
|
|
||||||
config ARCH_HAVE_SYSCALL_HOOKS
|
config ARCH_HAVE_SYSCALL_HOOKS
|
||||||
bool
|
bool
|
||||||
default n
|
default n
|
||||||
|
@ -89,7 +89,6 @@ config ARCH_CHIP_IMX6
|
|||||||
select BOOT_RUNFROMSDRAM
|
select BOOT_RUNFROMSDRAM
|
||||||
select ARCH_HAVE_ADDRENV
|
select ARCH_HAVE_ADDRENV
|
||||||
select ARCH_NEED_ADDRENV_MAPPING
|
select ARCH_NEED_ADDRENV_MAPPING
|
||||||
select ARCH_GLOBAL_IRQDISABLE
|
|
||||||
---help---
|
---help---
|
||||||
Freescale iMX.6 architectures (Cortex-A9)
|
Freescale iMX.6 architectures (Cortex-A9)
|
||||||
|
|
||||||
@ -132,7 +131,6 @@ config ARCH_CHIP_LC823450
|
|||||||
select ARCH_HAVE_HEAPCHECK
|
select ARCH_HAVE_HEAPCHECK
|
||||||
select ARCH_HAVE_MULTICPU
|
select ARCH_HAVE_MULTICPU
|
||||||
select ARCH_HAVE_I2CRESET
|
select ARCH_HAVE_I2CRESET
|
||||||
select ARCH_GLOBAL_IRQDISABLE
|
|
||||||
---help---
|
---help---
|
||||||
ON Semiconductor LC823450 architectures (ARM dual Cortex-M3)
|
ON Semiconductor LC823450 architectures (ARM dual Cortex-M3)
|
||||||
|
|
||||||
@ -428,7 +426,6 @@ config ARCH_CHIP_CXD56XX
|
|||||||
select ARCH_HAVE_FPU
|
select ARCH_HAVE_FPU
|
||||||
select ARCH_HAVE_HEAPCHECK
|
select ARCH_HAVE_HEAPCHECK
|
||||||
select ARCH_HAVE_MULTICPU
|
select ARCH_HAVE_MULTICPU
|
||||||
select ARCH_GLOBAL_IRQDISABLE
|
|
||||||
select ARCH_HAVE_SDIO if MMCSD
|
select ARCH_HAVE_SDIO if MMCSD
|
||||||
select ARCH_HAVE_MATH_H
|
select ARCH_HAVE_MATH_H
|
||||||
---help---
|
---help---
|
||||||
|
@ -247,7 +247,6 @@ config ARCH_CHIP_SAM4CM
|
|||||||
default n
|
default n
|
||||||
select ARCH_HAVE_MULTICPU
|
select ARCH_HAVE_MULTICPU
|
||||||
select ARCH_HAVE_TICKLESS
|
select ARCH_HAVE_TICKLESS
|
||||||
select ARCH_GLOBAL_IRQDISABLE
|
|
||||||
|
|
||||||
config ARCH_CHIP_SAM4L
|
config ARCH_CHIP_SAM4L
|
||||||
bool
|
bool
|
||||||
|
@ -22,7 +22,6 @@ config ARCH_CHIP_K210
|
|||||||
select ARCH_HAVE_MPU
|
select ARCH_HAVE_MPU
|
||||||
select ARCH_HAVE_TESTSET
|
select ARCH_HAVE_TESTSET
|
||||||
select ARCH_HAVE_MULTICPU
|
select ARCH_HAVE_MULTICPU
|
||||||
select ARCH_GLOBAL_IRQDISABLE
|
|
||||||
---help---
|
---help---
|
||||||
Kendryte K210 processor (RISC-V 64bit core with GC extensions)
|
Kendryte K210 processor (RISC-V 64bit core with GC extensions)
|
||||||
|
|
||||||
|
@ -16,7 +16,6 @@ config ARCH_CHIP_ESP32
|
|||||||
select ARCH_HAVE_MULTICPU
|
select ARCH_HAVE_MULTICPU
|
||||||
select ARCH_HAVE_MODULE_TEXT
|
select ARCH_HAVE_MODULE_TEXT
|
||||||
select ARCH_TOOLCHAIN_GNU
|
select ARCH_TOOLCHAIN_GNU
|
||||||
select ARCH_GLOBAL_IRQDISABLE
|
|
||||||
---help---
|
---help---
|
||||||
The ESP32 is a dual-core system from Espressif with two Harvard
|
The ESP32 is a dual-core system from Espressif with two Harvard
|
||||||
architecture Xtensa LX6 CPUs. All embedded memory, external memory
|
architecture Xtensa LX6 CPUs. All embedded memory, external memory
|
||||||
|
@ -260,8 +260,7 @@ void leave_critical_section(irqstate_t flags);
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#if defined(CONFIG_SMP) && defined(CONFIG_SPINLOCK_IRQ) && \
|
#if defined(CONFIG_SMP) && defined(CONFIG_SPINLOCK_IRQ)
|
||||||
defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
irqstate_t spin_lock_irqsave(void);
|
irqstate_t spin_lock_irqsave(void);
|
||||||
#else
|
#else
|
||||||
# define spin_lock_irqsave() enter_critical_section()
|
# define spin_lock_irqsave() enter_critical_section()
|
||||||
@ -289,8 +288,7 @@ irqstate_t spin_lock_irqsave(void);
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
#if defined(CONFIG_SMP) && defined(CONFIG_SPINLOCK_IRQ) && \
|
#if defined(CONFIG_SMP) && defined(CONFIG_SPINLOCK_IRQ)
|
||||||
defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
void spin_unlock_irqrestore(irqstate_t flags);
|
void spin_unlock_irqrestore(irqstate_t flags);
|
||||||
#else
|
#else
|
||||||
# define spin_unlock_irqrestore(f) leave_critical_section(f)
|
# define spin_unlock_irqrestore(f) leave_critical_section(f)
|
||||||
|
@ -244,7 +244,6 @@ config SPINLOCK
|
|||||||
config SPINLOCK_IRQ
|
config SPINLOCK_IRQ
|
||||||
bool "Support Spinlocks with IRQ control"
|
bool "Support Spinlocks with IRQ control"
|
||||||
default n
|
default n
|
||||||
depends on ARCH_GLOBAL_IRQDISABLE
|
|
||||||
---help---
|
---help---
|
||||||
Enables support for spinlocks with IRQ control. This feature can be
|
Enables support for spinlocks with IRQ control. This feature can be
|
||||||
used to protect data in SMP mode.
|
used to protect data in SMP mode.
|
||||||
|
@ -37,11 +37,9 @@ CSRCS += irq_initialize.c irq_attach.c irq_dispatch.c irq_unexpectedisr.c
|
|||||||
|
|
||||||
ifeq ($(CONFIG_SMP),y)
|
ifeq ($(CONFIG_SMP),y)
|
||||||
ifeq ($(CONFIG_SPINLOCK_IRQ),y)
|
ifeq ($(CONFIG_SPINLOCK_IRQ),y)
|
||||||
ifeq ($(CONFIG_ARCH_GLOBAL_IRQDISABLE),y)
|
|
||||||
CSRCS += irq_spinlock.c
|
CSRCS += irq_spinlock.c
|
||||||
endif
|
endif
|
||||||
endif
|
endif
|
||||||
endif
|
|
||||||
|
|
||||||
ifeq ($(CONFIG_IRQCOUNT),y)
|
ifeq ($(CONFIG_IRQCOUNT),y)
|
||||||
CSRCS += irq_csection.c
|
CSRCS += irq_csection.c
|
||||||
|
@ -646,15 +646,6 @@ bool irq_cpu_locked(int cpu)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_HAVE_FETCHADD) && !defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
/* If the global lockcount has been incremented then simply return true */
|
|
||||||
|
|
||||||
if (g_global_lockcount > 0)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Test if g_cpu_irqlock is locked. We don't really need to use check
|
/* Test if g_cpu_irqlock is locked. We don't really need to use check
|
||||||
* g_cpu_irqlock to do this, we can use the g_cpu_set.
|
* g_cpu_irqlock to do this, we can use the g_cpu_set.
|
||||||
*
|
*
|
||||||
|
@ -30,8 +30,7 @@
|
|||||||
|
|
||||||
#include "sched/sched.h"
|
#include "sched/sched.h"
|
||||||
|
|
||||||
#if defined(CONFIG_SMP) && defined (CONFIG_SPINLOCK_IRQ) && \
|
#if defined(CONFIG_SMP) && defined (CONFIG_SPINLOCK_IRQ)
|
||||||
defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Public Data
|
* Public Data
|
||||||
@ -127,4 +126,4 @@ void spin_unlock_irqrestore(irqstate_t flags)
|
|||||||
up_irq_restore(flags);
|
up_irq_restore(flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* CONFIG_SMP && CONFIG_SPINLOCK_IRQ && CONFIG_ARCH_GLOBAL_IRQDISABLE */
|
#endif /* CONFIG_SMP && CONFIG_SPINLOCK_IRQ */
|
||||||
|
@ -86,11 +86,7 @@ endif
|
|||||||
|
|
||||||
ifeq ($(CONFIG_SMP),y)
|
ifeq ($(CONFIG_SMP),y)
|
||||||
CSRCS += sched_tasklistlock.c
|
CSRCS += sched_tasklistlock.c
|
||||||
ifeq ($(CONFIG_ARCH_GLOBAL_IRQDISABLE),y)
|
|
||||||
CSRCS += sched_thistask.c
|
CSRCS += sched_thistask.c
|
||||||
else ifeq ($(CONFIG_ARCH_HAVE_FETCHADD),y)
|
|
||||||
CSRCS += sched_thistask.c
|
|
||||||
endif
|
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_SCHED_INSTRUMENTATION_BUFFER),y)
|
ifeq ($(CONFIG_SCHED_INSTRUMENTATION_BUFFER),y)
|
||||||
|
@ -57,17 +57,12 @@
|
|||||||
|
|
||||||
/* These are macros to access the current CPU and the current task on a CPU.
|
/* These are macros to access the current CPU and the current task on a CPU.
|
||||||
* These macros are intended to support a future SMP implementation.
|
* These macros are intended to support a future SMP implementation.
|
||||||
* NOTE: this_task() for SMP is implemented in sched_thistask.c if the CPU
|
* NOTE: this_task() for SMP is implemented in sched_thistask.c
|
||||||
* supports disabling of inter-processor interrupts or if it supports the
|
|
||||||
* atomic fetch add operation.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
# define current_task(cpu) ((FAR struct tcb_s *)g_assignedtasks[cpu].head)
|
# define current_task(cpu) ((FAR struct tcb_s *)g_assignedtasks[cpu].head)
|
||||||
# define this_cpu() up_cpu_index()
|
# define this_cpu() up_cpu_index()
|
||||||
# if !defined(CONFIG_ARCH_GLOBAL_IRQDISABLE) && !defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
# define this_task() (current_task(this_cpu()))
|
|
||||||
# endif
|
|
||||||
#else
|
#else
|
||||||
# define current_task(cpu) ((FAR struct tcb_s *)g_readytorun.head)
|
# define current_task(cpu) ((FAR struct tcb_s *)g_readytorun.head)
|
||||||
# define this_cpu() (0)
|
# define this_cpu() (0)
|
||||||
@ -353,14 +348,6 @@ extern volatile cpu_set_t g_cpu_lockset SP_SECTION;
|
|||||||
|
|
||||||
extern volatile spinlock_t g_cpu_tasklistlock SP_SECTION;
|
extern volatile spinlock_t g_cpu_tasklistlock SP_SECTION;
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_HAVE_FETCHADD) && !defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
/* This is part of the sched_lock() logic to handle atomic operations when
|
|
||||||
* locking the scheduler.
|
|
||||||
*/
|
|
||||||
|
|
||||||
extern volatile int16_t g_global_lockcount;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif /* CONFIG_SMP */
|
#endif /* CONFIG_SMP */
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -425,10 +412,7 @@ void nxsched_continue(FAR struct tcb_s *tcb);
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE) || \
|
|
||||||
defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
FAR struct tcb_s *this_task(void);
|
FAR struct tcb_s *this_task(void);
|
||||||
#endif
|
|
||||||
|
|
||||||
int nxsched_select_cpu(cpu_set_t affinity);
|
int nxsched_select_cpu(cpu_set_t affinity);
|
||||||
int nxsched_pause_cpu(FAR struct tcb_s *tcb);
|
int nxsched_pause_cpu(FAR struct tcb_s *tcb);
|
||||||
@ -436,15 +420,7 @@ int nxsched_pause_cpu(FAR struct tcb_s *tcb);
|
|||||||
irqstate_t nxsched_lock_tasklist(void);
|
irqstate_t nxsched_lock_tasklist(void);
|
||||||
void nxsched_unlock_tasklist(irqstate_t lock);
|
void nxsched_unlock_tasklist(irqstate_t lock);
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_HAVE_FETCHADD) && \
|
# define nxsched_islocked_global() spin_islocked(&g_cpu_schedlock)
|
||||||
!defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
# define nxsched_islocked_global() \
|
|
||||||
(spin_islocked(&g_cpu_schedlock) || g_global_lockcount > 0)
|
|
||||||
#else
|
|
||||||
# define nxsched_islocked_global() \
|
|
||||||
spin_islocked(&g_cpu_schedlock)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
# define nxsched_islocked_tcb(tcb) nxsched_islocked_global()
|
# define nxsched_islocked_tcb(tcb) nxsched_islocked_global()
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
@ -104,13 +104,6 @@ volatile spinlock_t g_cpu_schedlock SP_SECTION = SP_UNLOCKED;
|
|||||||
volatile spinlock_t g_cpu_locksetlock SP_SECTION;
|
volatile spinlock_t g_cpu_locksetlock SP_SECTION;
|
||||||
volatile cpu_set_t g_cpu_lockset SP_SECTION;
|
volatile cpu_set_t g_cpu_lockset SP_SECTION;
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_HAVE_FETCHADD) && !defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
/* This is part of the sched_lock() logic to handle atomic operations when
|
|
||||||
* locking the scheduler.
|
|
||||||
*/
|
|
||||||
|
|
||||||
volatile int16_t g_global_lockcount;
|
|
||||||
#endif
|
|
||||||
#endif /* CONFIG_SMP */
|
#endif /* CONFIG_SMP */
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -140,16 +133,9 @@ volatile int16_t g_global_lockcount;
|
|||||||
int sched_lock(void)
|
int sched_lock(void)
|
||||||
{
|
{
|
||||||
FAR struct tcb_s *rtcb;
|
FAR struct tcb_s *rtcb;
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
irqstate_t flags;
|
irqstate_t flags;
|
||||||
#endif
|
|
||||||
int cpu;
|
int cpu;
|
||||||
|
|
||||||
/* The following operation is non-atomic unless
|
|
||||||
* CONFIG_ARCH_GLOBAL_IRQDISABLE or CONFIG_ARCH_HAVE_FETCHADD is defined.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
/* If the CPU supports suppression of interprocessor interrupts, then
|
/* If the CPU supports suppression of interprocessor interrupts, then
|
||||||
* simple disabling interrupts will provide sufficient protection for
|
* simple disabling interrupts will provide sufficient protection for
|
||||||
* the following operation.
|
* the following operation.
|
||||||
@ -157,19 +143,6 @@ int sched_lock(void)
|
|||||||
|
|
||||||
flags = up_irq_save();
|
flags = up_irq_save();
|
||||||
|
|
||||||
#elif defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
/* If the CPU supports an atomic fetch add operation, then we can use the
|
|
||||||
* global lockcount to assure that the following operation is atomic.
|
|
||||||
*/
|
|
||||||
|
|
||||||
DEBUGASSERT((uint16_t)g_global_lockcount < INT16_MAX); /* Not atomic! */
|
|
||||||
up_fetchadd16(&g_global_lockcount, 1);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* This operation is safe if CONFIG_ARCH_HAVE_FETCHADD is defined. NOTE
|
|
||||||
* we cannot use this_task() because it calls sched_lock().
|
|
||||||
*/
|
|
||||||
|
|
||||||
cpu = this_cpu();
|
cpu = this_cpu();
|
||||||
rtcb = current_task(cpu);
|
rtcb = current_task(cpu);
|
||||||
|
|
||||||
@ -180,12 +153,7 @@ int sched_lock(void)
|
|||||||
|
|
||||||
if (rtcb == NULL || up_interrupt_context())
|
if (rtcb == NULL || up_interrupt_context())
|
||||||
{
|
{
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
up_irq_restore(flags);
|
up_irq_restore(flags);
|
||||||
#elif defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
DEBUGASSERT(g_global_lockcount > 0);
|
|
||||||
up_fetchsub16(&g_global_lockcount, 1);
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -228,12 +196,7 @@ int sched_lock(void)
|
|||||||
|
|
||||||
rtcb->lockcount++;
|
rtcb->lockcount++;
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
up_irq_restore(flags);
|
up_irq_restore(flags);
|
||||||
#elif defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
DEBUGASSERT(g_global_lockcount > 0);
|
|
||||||
up_fetchsub16(&g_global_lockcount, 1);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(CONFIG_SCHED_INSTRUMENTATION_PREEMPTION) || \
|
#if defined(CONFIG_SCHED_INSTRUMENTATION_PREEMPTION) || \
|
||||||
defined(CONFIG_SCHED_CRITMONITOR)
|
defined(CONFIG_SCHED_CRITMONITOR)
|
||||||
|
@ -54,7 +54,6 @@
|
|||||||
FAR struct tcb_s *this_task(void)
|
FAR struct tcb_s *this_task(void)
|
||||||
{
|
{
|
||||||
FAR struct tcb_s *tcb;
|
FAR struct tcb_s *tcb;
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
irqstate_t flags;
|
irqstate_t flags;
|
||||||
|
|
||||||
/* If the CPU supports suppression of interprocessor interrupts, then
|
/* If the CPU supports suppression of interprocessor interrupts, then
|
||||||
@ -63,20 +62,6 @@ FAR struct tcb_s *this_task(void)
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
flags = up_irq_save();
|
flags = up_irq_save();
|
||||||
#elif defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
/* Global locking is supported and, hence, sched_lock() will provide the
|
|
||||||
* necessary protection.
|
|
||||||
*/
|
|
||||||
|
|
||||||
sched_lock();
|
|
||||||
#else
|
|
||||||
/* REVISIT: Otherwise, there is no protection available. sched_lock() and
|
|
||||||
* enter_critical section are not viable options here (because both depend
|
|
||||||
* on this_task()).
|
|
||||||
*/
|
|
||||||
|
|
||||||
# warning "Missing critical section"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Obtain the TCB which is currently running on this CPU */
|
/* Obtain the TCB which is currently running on this CPU */
|
||||||
|
|
||||||
@ -84,11 +69,7 @@ FAR struct tcb_s *this_task(void)
|
|||||||
|
|
||||||
/* Enable local interrupts */
|
/* Enable local interrupts */
|
||||||
|
|
||||||
#if defined(CONFIG_ARCH_GLOBAL_IRQDISABLE)
|
|
||||||
up_irq_restore(flags);
|
up_irq_restore(flags);
|
||||||
#elif defined(CONFIG_ARCH_HAVE_FETCHADD)
|
|
||||||
sched_unlock();
|
|
||||||
#endif
|
|
||||||
return tcb;
|
return tcb;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user