Reimplement arch alarm timer on top of oneshot driver framework
The benefit include: 1. Simplify the implementation 2. Support both tick and tickless automatically 3. No time drift in tickless mode 4. Support critmon arch API automatically
This commit is contained in:
parent
20a9a62fdf
commit
d0d4b7a531
@ -72,6 +72,8 @@ config ARCH_SIM
|
|||||||
select ARCH_HAVE_POWEROFF
|
select ARCH_HAVE_POWEROFF
|
||||||
select ARCH_HAVE_TESTSET
|
select ARCH_HAVE_TESTSET
|
||||||
select ARCH_NOINTC
|
select ARCH_NOINTC
|
||||||
|
select ALARM_ARCH
|
||||||
|
select ONESHOT
|
||||||
select SERIAL_CONSOLE
|
select SERIAL_CONSOLE
|
||||||
---help---
|
---help---
|
||||||
Linux/Cywgin user-mode simulation.
|
Linux/Cywgin user-mode simulation.
|
||||||
|
@ -74,16 +74,12 @@ CSRCS += up_allocateheap.c
|
|||||||
VPATH = sim
|
VPATH = sim
|
||||||
DEPPATH = $(patsubst %,--dep-path %,$(subst :, ,$(VPATH)))
|
DEPPATH = $(patsubst %,--dep-path %,$(subst :, ,$(VPATH)))
|
||||||
|
|
||||||
HOSTSRCS = up_hostusleep.c
|
HOSTSRCS = up_hosttime.c
|
||||||
|
|
||||||
ifeq ($(CONFIG_STACK_COLORATION),y)
|
ifeq ($(CONFIG_STACK_COLORATION),y)
|
||||||
CSRCS += up_checkstack.c
|
CSRCS += up_checkstack.c
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_SCHED_TICKLESS),y)
|
|
||||||
CSRCS += up_tickless.c
|
|
||||||
endif
|
|
||||||
|
|
||||||
ifeq ($(CONFIG_SPINLOCK),y)
|
ifeq ($(CONFIG_SPINLOCK),y)
|
||||||
HOSTSRCS += up_testset.c
|
HOSTSRCS += up_testset.c
|
||||||
endif
|
endif
|
||||||
@ -113,10 +109,6 @@ ifeq ($(CONFIG_ONESHOT),y)
|
|||||||
CSRCS += up_oneshot.c
|
CSRCS += up_oneshot.c
|
||||||
endif
|
endif
|
||||||
|
|
||||||
ifeq ($(CONFIG_SCHED_CRITMONITOR),y)
|
|
||||||
HOSTSRCS += up_critmon.c
|
|
||||||
endif
|
|
||||||
|
|
||||||
ifeq ($(CONFIG_NX_LCDDRIVER),y)
|
ifeq ($(CONFIG_NX_LCDDRIVER),y)
|
||||||
CSRCS += up_lcd.c
|
CSRCS += up_lcd.c
|
||||||
else
|
else
|
||||||
|
@ -1,114 +0,0 @@
|
|||||||
/************************************************************************************
|
|
||||||
* arch/sim/src/sim/up_critmon.c
|
|
||||||
*
|
|
||||||
* Copyright (C) 2018 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 <stdint.h>
|
|
||||||
#include <time.h>
|
|
||||||
|
|
||||||
/************************************************************************************
|
|
||||||
* Pre-processor Definitions
|
|
||||||
************************************************************************************/
|
|
||||||
|
|
||||||
#undef USE_CLOCK /* Too slow */
|
|
||||||
#define USE_CLOCK_GETTIME 1 /* Better */
|
|
||||||
|
|
||||||
/* From nuttx/clock.h */
|
|
||||||
|
|
||||||
#define NSEC_PER_SEC 1000000000
|
|
||||||
|
|
||||||
/* From fixedmath.h */
|
|
||||||
|
|
||||||
#define b32ONE 0x0000000100000000 /* 1 */
|
|
||||||
#define b32toi(a) ((a) >> 32) /* Conversion to integer */
|
|
||||||
#define itob32(i) (((b32_t)(i)) << 32) /* Conversion from integer */
|
|
||||||
#define b32frac(a) ((a) & 0x00000000ffffffff) /* Take fractional part */
|
|
||||||
|
|
||||||
/************************************************************************************
|
|
||||||
* Private Types
|
|
||||||
************************************************************************************/
|
|
||||||
|
|
||||||
/* From fixedmath.h */
|
|
||||||
|
|
||||||
typedef int64_t b32_t;
|
|
||||||
|
|
||||||
/************************************************************************************
|
|
||||||
* Private Data
|
|
||||||
************************************************************************************/
|
|
||||||
|
|
||||||
/************************************************************************************
|
|
||||||
* Public Functions
|
|
||||||
************************************************************************************/
|
|
||||||
|
|
||||||
/************************************************************************************
|
|
||||||
* Name: up_critmon_gettime
|
|
||||||
************************************************************************************/
|
|
||||||
|
|
||||||
#if defined(USE_CLOCK)
|
|
||||||
uint32_t up_critmon_gettime(void)
|
|
||||||
{
|
|
||||||
return (uint32_t)clock() + 1; /* Avoid returning zero which means clock-not-ready */
|
|
||||||
}
|
|
||||||
#else /* USE_CLOCK_GETTIME */
|
|
||||||
uint32_t up_critmon_gettime(void)
|
|
||||||
{
|
|
||||||
struct timespec ts;
|
|
||||||
clock_gettime(CLOCK_MONOTONIC, &ts);
|
|
||||||
return (uint32_t)ts.tv_nsec;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/************************************************************************************
|
|
||||||
* Name: up_critmon_gettime
|
|
||||||
************************************************************************************/
|
|
||||||
|
|
||||||
#if defined(USE_CLOCK)
|
|
||||||
void up_critmon_convert(uint32_t elapsed, struct timespec *ts)
|
|
||||||
{
|
|
||||||
b32_t b32elapsed;
|
|
||||||
|
|
||||||
b32elapsed = itob32(elapsed) / CLOCKS_PER_SEC;
|
|
||||||
ts->tv_sec = b32toi(b32elapsed);
|
|
||||||
ts->tv_nsec = NSEC_PER_SEC * b32frac(b32elapsed) / b32ONE;
|
|
||||||
}
|
|
||||||
#else /* USE_CLOCK_GETTIME */
|
|
||||||
void up_critmon_convert(uint32_t elapsed, struct timespec *ts)
|
|
||||||
{
|
|
||||||
ts->tv_sec = 0;
|
|
||||||
ts->tv_nsec = elapsed;
|
|
||||||
}
|
|
||||||
#endif
|
|
@ -1,5 +1,5 @@
|
|||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* arch/sim/src/sim/up_hostusleep.c
|
* arch/sim/src/sim/up_hosttime.c
|
||||||
*
|
*
|
||||||
* Copyright (C) 2008 Gregory Nutt. All rights reserved.
|
* Copyright (C) 2008 Gregory Nutt. All rights reserved.
|
||||||
* Author: Gregory Nutt <gnutt@nuttx.org>
|
* Author: Gregory Nutt <gnutt@nuttx.org>
|
||||||
@ -37,6 +37,9 @@
|
|||||||
* Included Files
|
* Included Files
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <time.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -44,10 +47,48 @@
|
|||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: up_hostusleep
|
* Name: host_gettime
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
int up_hostusleep(unsigned int usec)
|
uint64_t host_gettime(bool rtc)
|
||||||
{
|
{
|
||||||
return usleep(usec);
|
struct timespec tp;
|
||||||
|
|
||||||
|
clock_gettime(rtc ? CLOCK_REALTIME : CLOCK_MONOTONIC, &tp);
|
||||||
|
return 1000000000ull * tp.tv_sec + tp.tv_nsec;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: host_settime
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void host_settime(bool rtc, uint64_t nsec)
|
||||||
|
{
|
||||||
|
struct timespec tp;
|
||||||
|
|
||||||
|
tp.tv_sec = nsec / 1000000000;
|
||||||
|
tp.tv_nsec = nsec - 1000000000 * tp.tv_sec;
|
||||||
|
clock_settime(rtc ? CLOCK_REALTIME : CLOCK_MONOTONIC, &tp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: host_sleepuntil
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void host_sleepuntil(uint64_t nsec)
|
||||||
|
{
|
||||||
|
static uint64_t base;
|
||||||
|
uint64_t now;
|
||||||
|
|
||||||
|
now = host_gettime(false);
|
||||||
|
if (base == 0)
|
||||||
|
{
|
||||||
|
base = now;
|
||||||
|
}
|
||||||
|
now -= base;
|
||||||
|
|
||||||
|
if (nsec > now + 1000)
|
||||||
|
{
|
||||||
|
usleep((nsec - now) / 1000);
|
||||||
|
}
|
||||||
}
|
}
|
@ -72,16 +72,21 @@
|
|||||||
|
|
||||||
void up_idle(void)
|
void up_idle(void)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_SCHED_TICKLESS
|
#ifdef CONFIG_PM
|
||||||
/* Driver the simulated interval timer */
|
static enum pm_state_e state = PM_NORMAL;
|
||||||
|
enum pm_state_e newstate;
|
||||||
|
|
||||||
up_timer_update();
|
/* Fake some power management stuff for testing purposes */
|
||||||
#else
|
|
||||||
/* If the system is idle, then process "fake" timer interrupts.
|
|
||||||
* Hopefully, something will wake up.
|
|
||||||
*/
|
|
||||||
|
|
||||||
nxsched_process_timer();
|
newstate = pm_checkstate(PM_IDLE_DOMAIN);
|
||||||
|
if (newstate != state)
|
||||||
|
{
|
||||||
|
if (pm_changestate(PM_IDLE_DOMAIN, newstate) == OK)
|
||||||
|
{
|
||||||
|
state = newstate;
|
||||||
|
pwrinfo("IDLE: switching to new state %i\n", state);
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DEVCONSOLE
|
#ifdef USE_DEVCONSOLE
|
||||||
@ -106,30 +111,9 @@ void up_idle(void)
|
|||||||
up_rptun_loop();
|
up_rptun_loop();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef CONFIG_PM
|
#ifdef CONFIG_ONESHOT
|
||||||
/* Fake some power management stuff for testing purposes */
|
/* Driver the simulated interval timer */
|
||||||
|
|
||||||
{
|
up_timer_update();
|
||||||
static enum pm_state_e state = PM_NORMAL;
|
|
||||||
enum pm_state_e newstate;
|
|
||||||
|
|
||||||
newstate = pm_checkstate(PM_IDLE_DOMAIN);
|
|
||||||
if (newstate != state)
|
|
||||||
{
|
|
||||||
if (pm_changestate(PM_IDLE_DOMAIN, newstate) == OK)
|
|
||||||
{
|
|
||||||
state = newstate;
|
|
||||||
pwrinfo("IDLE: switching to new state %i\n", state);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef CONFIG_SIM_WALLTIME
|
|
||||||
/* Wait a bit so that the nxsched_process_timer() is called close to the
|
|
||||||
* correct rate.
|
|
||||||
*/
|
|
||||||
|
|
||||||
up_hostusleep(1000000 / CLK_TCK);
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -296,16 +296,3 @@ void up_initialize(void)
|
|||||||
up_init_smartfs();
|
up_init_smartfs();
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Function: up_timer_initialize
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* This function is called during start-up to initialize
|
|
||||||
* the timer hardware.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
void up_timer_initialize(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
@ -225,9 +225,11 @@ volatile spinlock_t g_cpu_paused[CONFIG_SMP_NCPUS] SP_SECTION;
|
|||||||
int up_setjmp(xcpt_reg_t *jb);
|
int up_setjmp(xcpt_reg_t *jb);
|
||||||
void up_longjmp(xcpt_reg_t *jb, int val) noreturn_function;
|
void up_longjmp(xcpt_reg_t *jb, int val) noreturn_function;
|
||||||
|
|
||||||
/* up_hostusleep.c **********************************************************/
|
/* up_hosttime.c ************************************************************/
|
||||||
|
|
||||||
int up_hostusleep(unsigned int usec);
|
uint64_t host_gettime(bool rtc);
|
||||||
|
void host_settime(bool rtc, uint64_t nsec);
|
||||||
|
void host_sleepuntil(uint64_t nsec);
|
||||||
|
|
||||||
/* up_simsmp.c **************************************************************/
|
/* up_simsmp.c **************************************************************/
|
||||||
|
|
||||||
@ -241,9 +243,9 @@ void sim_cpu0_start(void);
|
|||||||
void up_cpu_started(void);
|
void up_cpu_started(void);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* up_tickless.c ************************************************************/
|
/* up_oneshot.c *************************************************************/
|
||||||
|
|
||||||
#ifdef CONFIG_SCHED_TICKLESS
|
#ifdef CONFIG_ONESHOT
|
||||||
void up_timer_update(void);
|
void up_timer_update(void);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -39,16 +39,20 @@
|
|||||||
|
|
||||||
#include <nuttx/config.h>
|
#include <nuttx/config.h>
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <time.h>
|
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
|
#include <queue.h>
|
||||||
|
|
||||||
#include <nuttx/wdog.h>
|
#include <nuttx/nuttx.h>
|
||||||
|
#include <nuttx/arch.h>
|
||||||
|
#include <nuttx/clock.h>
|
||||||
#include <nuttx/kmalloc.h>
|
#include <nuttx/kmalloc.h>
|
||||||
#include <nuttx/timers/oneshot.h>
|
#include <nuttx/timers/oneshot.h>
|
||||||
|
#include <nuttx/timers/arch_alarm.h>
|
||||||
|
|
||||||
|
#include "up_internal.h"
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Private Types
|
* Private Types
|
||||||
@ -68,7 +72,9 @@ struct sim_oneshot_lowerhalf_s
|
|||||||
|
|
||||||
/* Private lower half data follows */
|
/* Private lower half data follows */
|
||||||
|
|
||||||
WDOG_ID wdog; /* Simulates oneshot timer */
|
sq_entry_t link;
|
||||||
|
struct timespec alarm;
|
||||||
|
|
||||||
oneshot_callback_t callback; /* internal handler that receives callback */
|
oneshot_callback_t callback; /* internal handler that receives callback */
|
||||||
FAR void *arg; /* Argument that is passed to the handler */
|
FAR void *arg; /* Argument that is passed to the handler */
|
||||||
};
|
};
|
||||||
@ -77,7 +83,7 @@ struct sim_oneshot_lowerhalf_s
|
|||||||
* Private Function Prototypes
|
* Private Function Prototypes
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void sim_oneshot_handler(int argc, wdparm_t arg1, ...);
|
static void sim_process_tick(sq_entry_t *entry);
|
||||||
|
|
||||||
static int sim_max_delay(FAR struct oneshot_lowerhalf_s *lower,
|
static int sim_max_delay(FAR struct oneshot_lowerhalf_s *lower,
|
||||||
FAR struct timespec *ts);
|
FAR struct timespec *ts);
|
||||||
@ -86,11 +92,16 @@ static int sim_start(FAR struct oneshot_lowerhalf_s *lower,
|
|||||||
FAR const struct timespec *ts);
|
FAR const struct timespec *ts);
|
||||||
static int sim_cancel(FAR struct oneshot_lowerhalf_s *lower,
|
static int sim_cancel(FAR struct oneshot_lowerhalf_s *lower,
|
||||||
FAR struct timespec *ts);
|
FAR struct timespec *ts);
|
||||||
|
static int sim_current(FAR struct oneshot_lowerhalf_s *lower,
|
||||||
|
FAR struct timespec *ts);
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Private Data
|
* Private Data
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
|
static struct timespec g_current;
|
||||||
|
static sq_queue_t g_oneshot_list;
|
||||||
|
|
||||||
/* Lower half operations */
|
/* Lower half operations */
|
||||||
|
|
||||||
static const struct oneshot_operations_s g_oneshot_ops =
|
static const struct oneshot_operations_s g_oneshot_ops =
|
||||||
@ -98,6 +109,7 @@ static const struct oneshot_operations_s g_oneshot_ops =
|
|||||||
.max_delay = sim_max_delay,
|
.max_delay = sim_max_delay,
|
||||||
.start = sim_start,
|
.start = sim_start,
|
||||||
.cancel = sim_cancel,
|
.cancel = sim_cancel,
|
||||||
|
.current = sim_current,
|
||||||
};
|
};
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -105,35 +117,35 @@ static const struct oneshot_operations_s g_oneshot_ops =
|
|||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* Name: sim_oneshot_handler
|
* Name: sim_process_tick
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Timer expiration handler
|
* Timer expiration handler
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameters:
|
||||||
* arg - Should be the same argument provided when sim_oneshot_start()
|
* entry - Point to the link field of sim_oneshot_lowerhalf_s.
|
||||||
* was called.
|
|
||||||
*
|
*
|
||||||
* Returned Value:
|
* Returned Value:
|
||||||
* None
|
* None
|
||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
static void sim_oneshot_handler(int argc, wdparm_t arg1, ...)
|
static void sim_process_tick(sq_entry_t *entry)
|
||||||
{
|
{
|
||||||
FAR struct sim_oneshot_lowerhalf_s *priv =
|
FAR struct sim_oneshot_lowerhalf_s *priv =
|
||||||
(FAR struct sim_oneshot_lowerhalf_s *)arg1;
|
container_of(entry, struct sim_oneshot_lowerhalf_s, link);
|
||||||
oneshot_callback_t callback;
|
oneshot_callback_t callback;
|
||||||
FAR void *cbarg;
|
FAR void *cbarg;
|
||||||
|
|
||||||
DEBUGASSERT(argc == 1 && priv != NULL);
|
DEBUGASSERT(priv != NULL);
|
||||||
|
|
||||||
/* Perhaps the callback was nullified in a race condition with
|
|
||||||
* sim_cancel?
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (priv->callback)
|
if (priv->callback)
|
||||||
{
|
{
|
||||||
|
if (clock_timespec_compare(&priv->alarm, &g_current) > 0)
|
||||||
|
{
|
||||||
|
return; /* Alarm doesn't expire yet */
|
||||||
|
}
|
||||||
|
|
||||||
/* Sample and nullify BEFORE executing callback (in case the callback
|
/* Sample and nullify BEFORE executing callback (in case the callback
|
||||||
* restarts the oneshot).
|
* restarts the oneshot).
|
||||||
*/
|
*/
|
||||||
@ -153,7 +165,7 @@ static void sim_oneshot_handler(int argc, wdparm_t arg1, ...)
|
|||||||
* Name: sim_max_delay
|
* Name: sim_max_delay
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Determine the maximum delay of the one-shot timer (in microseconds)
|
* Determine the maximum delay of the one-shot timer
|
||||||
*
|
*
|
||||||
* Input Parameters:
|
* Input Parameters:
|
||||||
* lower An instance of the lower-half oneshot state structure. This
|
* lower An instance of the lower-half oneshot state structure. This
|
||||||
@ -172,8 +184,8 @@ static int sim_max_delay(FAR struct oneshot_lowerhalf_s *lower,
|
|||||||
{
|
{
|
||||||
DEBUGASSERT(lower != NULL && ts != NULL);
|
DEBUGASSERT(lower != NULL && ts != NULL);
|
||||||
|
|
||||||
ts->tv_sec = INT_MAX;
|
ts->tv_sec = UINT_MAX;
|
||||||
ts->tv_nsec = 1000000000ul - 1;
|
ts->tv_nsec = NSEC_PER_SEC - 1;
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -203,24 +215,15 @@ static int sim_start(FAR struct oneshot_lowerhalf_s *lower,
|
|||||||
{
|
{
|
||||||
FAR struct sim_oneshot_lowerhalf_s *priv =
|
FAR struct sim_oneshot_lowerhalf_s *priv =
|
||||||
(FAR struct sim_oneshot_lowerhalf_s *)lower;
|
(FAR struct sim_oneshot_lowerhalf_s *)lower;
|
||||||
clock_t ticks;
|
|
||||||
int64_t nsec;
|
|
||||||
|
|
||||||
DEBUGASSERT(priv != NULL && callback != NULL && ts != NULL);
|
DEBUGASSERT(priv != NULL && callback != NULL && ts != NULL);
|
||||||
|
|
||||||
/* Convert time to ticks */
|
clock_timespec_add(&g_current, ts, &priv->alarm);
|
||||||
|
|
||||||
nsec = (int64_t)ts->tv_sec * NSEC_PER_SEC +
|
|
||||||
(int64_t)ts->tv_nsec;
|
|
||||||
ticks = (clock_t)((nsec + NSEC_PER_TICK - 1) / NSEC_PER_TICK);
|
|
||||||
|
|
||||||
/* Save the callback information and start the timer */
|
|
||||||
|
|
||||||
priv->callback = callback;
|
priv->callback = callback;
|
||||||
priv->arg = arg;
|
priv->arg = arg;
|
||||||
|
|
||||||
return wd_start(priv->wdog, ticks, (wdentry_t)sim_oneshot_handler,
|
return OK;
|
||||||
1, (wdparm_t)priv);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -252,17 +255,46 @@ static int sim_cancel(FAR struct oneshot_lowerhalf_s *lower,
|
|||||||
{
|
{
|
||||||
FAR struct sim_oneshot_lowerhalf_s *priv =
|
FAR struct sim_oneshot_lowerhalf_s *priv =
|
||||||
(FAR struct sim_oneshot_lowerhalf_s *)lower;
|
(FAR struct sim_oneshot_lowerhalf_s *)lower;
|
||||||
int ret;
|
|
||||||
|
|
||||||
DEBUGASSERT(priv != NULL);
|
DEBUGASSERT(priv != NULL && ts != NULL);
|
||||||
|
|
||||||
/* Cancel the timer */
|
clock_timespec_subtract(&priv->alarm, &g_current, ts);
|
||||||
|
|
||||||
ret = wd_cancel(priv->wdog);
|
|
||||||
priv->callback = NULL;
|
priv->callback = NULL;
|
||||||
priv->arg = NULL;
|
priv->arg = NULL;
|
||||||
|
|
||||||
return ret;
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: sim_current
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Get the current time.
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* lower Caller allocated instance of the oneshot state structure. This
|
||||||
|
* structure must have been previously initialized via a call to
|
||||||
|
* oneshot_initialize();
|
||||||
|
* ts The location in which to return the current time. A time of zero
|
||||||
|
* is returned for the initialization moment.
|
||||||
|
*
|
||||||
|
* Returned Value:
|
||||||
|
* Zero (OK) is returned on success, a negated errno value is returned on
|
||||||
|
* any failure.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
static int sim_current(FAR struct oneshot_lowerhalf_s *lower,
|
||||||
|
FAR struct timespec *ts)
|
||||||
|
{
|
||||||
|
FAR struct sim_oneshot_lowerhalf_s *priv =
|
||||||
|
(FAR struct sim_oneshot_lowerhalf_s *)lower;
|
||||||
|
|
||||||
|
DEBUGASSERT(priv != NULL && ts != NULL);
|
||||||
|
|
||||||
|
*ts = g_current;
|
||||||
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -306,17 +338,60 @@ FAR struct oneshot_lowerhalf_s *oneshot_initialize(int chan,
|
|||||||
|
|
||||||
/* Initialize the lower-half driver structure */
|
/* Initialize the lower-half driver structure */
|
||||||
|
|
||||||
|
sq_addlast(&priv->link, &g_oneshot_list);
|
||||||
priv->lh.ops = &g_oneshot_ops;
|
priv->lh.ops = &g_oneshot_ops;
|
||||||
|
|
||||||
/* Initialize the contained watchdog timer */
|
|
||||||
|
|
||||||
priv->wdog = wd_create();
|
|
||||||
if (priv->wdog == NULL)
|
|
||||||
{
|
|
||||||
tmrerr("ERROR: Failed to create wdog\n");
|
|
||||||
kmm_free(priv);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
return &priv->lh;
|
return &priv->lh;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Function: up_timer_initialize
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* This function is called during start-up to initialize
|
||||||
|
* the timer hardware.
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void up_timer_initialize(void)
|
||||||
|
{
|
||||||
|
up_alarm_set_lowerhalf(oneshot_initialize(0, 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* Name: up_timer_update
|
||||||
|
*
|
||||||
|
* Description:
|
||||||
|
* Called from the IDLE loop to fake one timer tick.
|
||||||
|
*
|
||||||
|
* Input Parameters:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
* Returned Value:
|
||||||
|
* None
|
||||||
|
*
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
void up_timer_update(void)
|
||||||
|
{
|
||||||
|
static const struct timespec tick =
|
||||||
|
{
|
||||||
|
.tv_sec = 0,
|
||||||
|
.tv_nsec = NSEC_PER_TICK,
|
||||||
|
};
|
||||||
|
FAR sq_entry_t *entry;
|
||||||
|
|
||||||
|
clock_timespec_add(&g_current, &tick, &g_current);
|
||||||
|
|
||||||
|
#ifdef CONFIG_SIM_WALLTIME
|
||||||
|
/* Wait a bit so that the timing is close to the correct rate. */
|
||||||
|
|
||||||
|
host_sleepuntil(g_current.tv_nsec +
|
||||||
|
(uint64_t)g_current.tv_sec * NSEC_PER_SEC);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for (entry = sq_peek(&g_oneshot_list); entry; entry = sq_next(entry))
|
||||||
|
{
|
||||||
|
sim_process_tick(entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,355 +0,0 @@
|
|||||||
/****************************************************************************
|
|
||||||
* arch/sim/src/sim/up_tickless.c
|
|
||||||
*
|
|
||||||
* Copyright (C) 2014, 2019 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.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Tickless OS Support.
|
|
||||||
*
|
|
||||||
* When CONFIG_SCHED_TICKLESS is enabled, all support for timer interrupts
|
|
||||||
* is suppressed and the platform specific code is expected to provide the
|
|
||||||
* following custom functions.
|
|
||||||
*
|
|
||||||
* void sim_timer_initialize(void): Initializes the timer facilities.
|
|
||||||
* Called early in the initialization sequence (by up_initialize()).
|
|
||||||
* int up_timer_gettime(FAR struct timespec *ts): Returns the current
|
|
||||||
* time from the platform specific time source.
|
|
||||||
* int up_timer_cancel(void): Cancels the interval timer.
|
|
||||||
* int up_timer_start(FAR const struct timespec *ts): Start (or re-starts)
|
|
||||||
* the interval timer.
|
|
||||||
*
|
|
||||||
* The RTOS will provide the following interfaces for use by the platform-
|
|
||||||
* specific interval timer implementation:
|
|
||||||
*
|
|
||||||
* void nxsched_timer_expiration(void): Called by the platform-specific
|
|
||||||
* logic when the interval timer expires.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Included Files
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#include <nuttx/config.h>
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <time.h>
|
|
||||||
|
|
||||||
#include <nuttx/arch.h>
|
|
||||||
#include <nuttx/clock.h>
|
|
||||||
|
|
||||||
#ifdef CONFIG_SCHED_TICKLESS
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Pre-processor Definitions
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#if defined(CONFIG_SIM_WALLTIME) || defined(CONFIG_SIM_X11FB)
|
|
||||||
/* TICK_USEC, the desired number of microseconds per clock tick (truncated) */
|
|
||||||
|
|
||||||
# define TICK_USEC (USEC_PER_SEC / CLK_TCK)
|
|
||||||
|
|
||||||
/* TICK_SEC, the integer number of seconds in the clock tick (truncated).
|
|
||||||
* Should be zero.
|
|
||||||
*/
|
|
||||||
|
|
||||||
# define TICK_SEC (TICK_USEC / USEC_PER_SEC)
|
|
||||||
|
|
||||||
/* TICK_NSEC, the integer number of nanoseconds (in addition to the number of
|
|
||||||
* seconds) in the clock tick.
|
|
||||||
*/
|
|
||||||
|
|
||||||
# define USEC_REMAINDER (TICK_USEC - (TICK_SEC * USEC_PER_SEC))
|
|
||||||
# define TICK_NSEC (USEC_REMAINDER * NSEC_PER_USEC)
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
# define TICK_SEC 0
|
|
||||||
# define TICK_NSEC NSEC_PER_TICK
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Private Data
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
static struct timespec g_elapsed_time;
|
|
||||||
static struct timespec g_interval_delay;
|
|
||||||
static bool g_timer_active;
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Public Functions
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: sim_timer_initialize
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Initializes all platform-specific timer facilities. This function is
|
|
||||||
* called early in the initialization sequence by up_initialize().
|
|
||||||
* On return, the current up-time should be available from
|
|
||||||
* up_timer_gettime() and the interval timer is ready for use (but not
|
|
||||||
* actively timing.
|
|
||||||
*
|
|
||||||
* Provided by platform-specific code and called from the architecture-
|
|
||||||
* specific logic.
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* None
|
|
||||||
*
|
|
||||||
* Returned Value:
|
|
||||||
* None
|
|
||||||
*
|
|
||||||
* Assumptions:
|
|
||||||
* Called early in the initialization sequence before any special
|
|
||||||
* concurrency protections are required.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
void sim_timer_initialize(void)
|
|
||||||
{
|
|
||||||
g_elapsed_time.tv_sec = 0;
|
|
||||||
g_elapsed_time.tv_nsec = 0;
|
|
||||||
g_timer_active = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: up_timer_gettime
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Return the elapsed time since power-up (or, more correctly, since
|
|
||||||
* sim_timer_initialize() was called). This function is functionally
|
|
||||||
* equivalent to:
|
|
||||||
*
|
|
||||||
* int clock_gettime(clockid_t clockid, FAR struct timespec *ts);
|
|
||||||
*
|
|
||||||
* when clockid is CLOCK_MONOTONIC.
|
|
||||||
*
|
|
||||||
* This function provides the basis for reporting the current time and
|
|
||||||
* also is used to eliminate error build-up from small erros in interval
|
|
||||||
* time calculations.
|
|
||||||
*
|
|
||||||
* Provided by platform-specific code and called from the RTOS base code.
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* ts - Provides the location in which to return the up-time.
|
|
||||||
*
|
|
||||||
* Returned Value:
|
|
||||||
* Zero (OK) is returned on success; a negated errno value is returned on
|
|
||||||
* any failure.
|
|
||||||
*
|
|
||||||
* Assumptions:
|
|
||||||
* Called from the normal tasking context. The implementation must
|
|
||||||
* provide whatever mutual exclusion is necessary for correct operation.
|
|
||||||
* This can include disabling interrupts in order to assure atomic register
|
|
||||||
* operations.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
int up_timer_gettime(FAR struct timespec *ts)
|
|
||||||
{
|
|
||||||
ts->tv_sec = g_elapsed_time.tv_sec;
|
|
||||||
ts->tv_nsec = g_elapsed_time.tv_nsec;
|
|
||||||
|
|
||||||
return OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: up_timer_cancel
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Cancel the interval timer and return the time remaining on the timer.
|
|
||||||
* These two steps need to be as nearly atomic as possible.
|
|
||||||
* nxsched_timer_expiration() will not be called unless the timer is
|
|
||||||
* restarted with up_timer_start().
|
|
||||||
*
|
|
||||||
* If, as a race condition, the timer has already expired when this
|
|
||||||
* function is called, then that pending interrupt must be cleared so
|
|
||||||
* that up_timer_start() and the remaining time of zero should be
|
|
||||||
* returned.
|
|
||||||
*
|
|
||||||
* Provided by platform-specific code and called from the RTOS base code.
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* ts - Location to return the remaining time. Zero should be returned
|
|
||||||
* if the timer is not active.
|
|
||||||
*
|
|
||||||
* Returned Value:
|
|
||||||
* Zero (OK) is returned on success; a negated errno value is returned on
|
|
||||||
* any failure.
|
|
||||||
*
|
|
||||||
* Assumptions:
|
|
||||||
* May be called from interrupt level handling or from the normal tasking
|
|
||||||
* level. Interrupts may need to be disabled internally to assure
|
|
||||||
* non-reentrancy.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#ifdef CONFIG_SCHED_TICKLESS
|
|
||||||
int up_timer_cancel(FAR struct timespec *ts)
|
|
||||||
{
|
|
||||||
/* Return the time remaining on the simulated timer */
|
|
||||||
|
|
||||||
if (g_timer_active)
|
|
||||||
{
|
|
||||||
ts->tv_sec = g_interval_delay.tv_sec;
|
|
||||||
ts->tv_nsec = g_interval_delay.tv_nsec;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ts->tv_sec = 0;
|
|
||||||
ts->tv_nsec = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Disable and reset the simulated timer */
|
|
||||||
|
|
||||||
g_interval_delay.tv_sec = 0;
|
|
||||||
g_interval_delay.tv_nsec = 0;
|
|
||||||
g_timer_active = false;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: up_timer_start
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Start the interval timer. nxsched_timer_expiration() will be
|
|
||||||
* called at the completion of the timeout (unless up_timer_cancel
|
|
||||||
* is called to stop the timing.
|
|
||||||
*
|
|
||||||
* Provided by platform-specific code and called from the RTOS base code.
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* ts - Provides the time interval until nxsched_timer_expiration() is
|
|
||||||
* called.
|
|
||||||
*
|
|
||||||
* Returned Value:
|
|
||||||
* Zero (OK) is returned on success; a negated errno value is returned on
|
|
||||||
* any failure.
|
|
||||||
*
|
|
||||||
* Assumptions:
|
|
||||||
* May be called from interrupt level handling or from the normal tasking
|
|
||||||
* level. Interrupts may need to be disabled internally to assure
|
|
||||||
* non-reentrancy.
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
#ifdef CONFIG_SCHED_TICKLESS
|
|
||||||
int up_timer_start(FAR const struct timespec *ts)
|
|
||||||
{
|
|
||||||
g_interval_delay.tv_sec = ts->tv_sec;
|
|
||||||
g_interval_delay.tv_nsec = ts->tv_nsec;
|
|
||||||
g_timer_active = true;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/****************************************************************************
|
|
||||||
* Name: up_timer_update
|
|
||||||
*
|
|
||||||
* Description:
|
|
||||||
* Called from the IDLE loop to fake one timer tick.
|
|
||||||
*
|
|
||||||
* Input Parameters:
|
|
||||||
* None
|
|
||||||
*
|
|
||||||
* Returned Value:
|
|
||||||
* None
|
|
||||||
*
|
|
||||||
****************************************************************************/
|
|
||||||
|
|
||||||
void up_timer_update(void)
|
|
||||||
{
|
|
||||||
/* Increment the elapsed time */
|
|
||||||
|
|
||||||
g_elapsed_time.tv_nsec += TICK_NSEC;
|
|
||||||
if (g_elapsed_time.tv_nsec >= NSEC_PER_SEC)
|
|
||||||
{
|
|
||||||
g_elapsed_time.tv_sec++;
|
|
||||||
g_elapsed_time.tv_nsec -= NSEC_PER_SEC;
|
|
||||||
}
|
|
||||||
|
|
||||||
g_elapsed_time.tv_sec += TICK_SEC;
|
|
||||||
|
|
||||||
/* Is the interval timer active? */
|
|
||||||
|
|
||||||
if (g_timer_active)
|
|
||||||
{
|
|
||||||
/* Yes... decrement the interval timer */
|
|
||||||
|
|
||||||
if (g_interval_delay.tv_sec < TICK_SEC)
|
|
||||||
{
|
|
||||||
/* No more seconds left... the timer has expired */
|
|
||||||
|
|
||||||
g_timer_active = false;
|
|
||||||
nxsched_timer_expiration();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* Decrement seconds. May decrement to zero */
|
|
||||||
|
|
||||||
g_interval_delay.tv_sec -= TICK_SEC;
|
|
||||||
|
|
||||||
/* Decrement nanoseconds */
|
|
||||||
|
|
||||||
if (g_interval_delay.tv_nsec >= TICK_NSEC)
|
|
||||||
{
|
|
||||||
g_interval_delay.tv_nsec -= TICK_NSEC;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Handle borrow from seconds */
|
|
||||||
|
|
||||||
else if (g_interval_delay.tv_sec > 0)
|
|
||||||
{
|
|
||||||
g_interval_delay.tv_nsec += NSEC_PER_SEC;
|
|
||||||
g_interval_delay.tv_sec--;
|
|
||||||
|
|
||||||
g_interval_delay.tv_nsec -= TICK_NSEC;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Otherwise the timer has expired */
|
|
||||||
|
|
||||||
else
|
|
||||||
{
|
|
||||||
g_timer_active = false;
|
|
||||||
nxsched_timer_expiration();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /* CONFIG_SCHED_TICKLESS */
|
|
Loading…
Reference in New Issue
Block a user