2015-04-11 07:44:37 +08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2011-2014 Wind River Systems, Inc.
|
|
|
|
*
|
2015-10-07 00:00:37 +08:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2015-04-11 07:44:37 +08:00
|
|
|
*
|
2015-10-07 00:00:37 +08:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2015-04-11 07:44:37 +08:00
|
|
|
*
|
2015-10-07 00:00:37 +08:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
2015-04-11 07:44:37 +08:00
|
|
|
*/
|
|
|
|
|
2015-12-04 23:09:39 +08:00
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* @brief Nanokernel idle support
|
|
|
|
*
|
2015-10-21 00:42:33 +08:00
|
|
|
* This module provides routines to set the idle field in the nanokernel
|
|
|
|
* data structure.
|
2015-07-02 05:22:39 +08:00
|
|
|
*/
|
2015-04-11 07:44:37 +08:00
|
|
|
|
|
|
|
|
|
|
|
#include <nanokernel.h>
|
2015-06-19 23:07:02 +08:00
|
|
|
#include <nano_private.h>
|
2015-04-11 07:44:37 +08:00
|
|
|
#include <toolchain.h>
|
|
|
|
#include <sections.h>
|
2015-10-05 22:19:39 +08:00
|
|
|
#include <drivers/system_timer.h>
|
|
|
|
#include <wait_q.h>
|
2015-04-11 07:44:37 +08:00
|
|
|
|
2015-07-02 05:22:39 +08:00
|
|
|
/**
|
|
|
|
*
|
2015-07-02 05:51:40 +08:00
|
|
|
* @brief Indicate that nanokernel is idling in tickless mode
|
2015-07-02 05:22:39 +08:00
|
|
|
*
|
|
|
|
* Sets the nanokernel data structure idle field to a non-zero value.
|
|
|
|
*
|
2015-10-21 00:42:33 +08:00
|
|
|
* @param ticks the number of ticks to idle
|
|
|
|
*
|
2015-07-02 05:29:04 +08:00
|
|
|
* @return N/A
|
2015-07-02 05:22:39 +08:00
|
|
|
*/
|
2015-04-17 03:25:10 +08:00
|
|
|
void nano_cpu_set_idle(int32_t ticks)
|
2015-04-11 07:44:37 +08:00
|
|
|
{
|
2015-05-09 06:12:45 +08:00
|
|
|
extern tNANO _nanokernel;
|
2015-04-15 06:15:52 +08:00
|
|
|
|
2015-05-09 06:12:45 +08:00
|
|
|
_nanokernel.idle = ticks;
|
2015-04-11 07:44:37 +08:00
|
|
|
}
|
|
|
|
|
2015-10-05 22:19:39 +08:00
|
|
|
#if defined(CONFIG_NANOKERNEL) && defined(CONFIG_TICKLESS_IDLE)
|
|
|
|
|
|
|
|
int32_t _sys_idle_ticks_threshold = CONFIG_TICKLESS_IDLE_THRESH;
|
|
|
|
|
2016-03-11 00:18:05 +08:00
|
|
|
#if defined(CONFIG_NANO_TIMEOUTS) || defined(CONFIG_NANO_TIMERS)
|
|
|
|
static inline int32_t get_next_tick_expiry(void)
|
2015-10-05 22:19:39 +08:00
|
|
|
{
|
|
|
|
return _nano_get_earliest_timeouts_deadline();
|
|
|
|
}
|
|
|
|
#else
|
2016-03-11 00:18:05 +08:00
|
|
|
#define get_next_tick_expiry(void) TICKS_UNLIMITED
|
2015-10-05 22:19:39 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static inline int was_in_tickless_idle(void)
|
|
|
|
{
|
|
|
|
return (_nanokernel.idle == TICKS_UNLIMITED) ||
|
|
|
|
(_nanokernel.idle >= _sys_idle_ticks_threshold);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int must_enter_tickless_idle(void)
|
|
|
|
{
|
|
|
|
/* uses same logic as was_in_tickless_idle() */
|
|
|
|
return was_in_tickless_idle();
|
|
|
|
}
|
|
|
|
|
|
|
|
void _power_save_idle(void)
|
|
|
|
{
|
|
|
|
_nanokernel.idle = get_next_tick_expiry();
|
|
|
|
|
|
|
|
if (must_enter_tickless_idle()) {
|
|
|
|
_timer_idle_enter((uint32_t)_nanokernel.idle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _power_save_idle_exit(void)
|
|
|
|
{
|
|
|
|
if (was_in_tickless_idle()) {
|
|
|
|
_timer_idle_exit();
|
|
|
|
_nanokernel.idle = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_NANOKERNEL && CONFIG_TICKLESS_IDLE */
|