mirror of https://github.com/ARMmbed/mbed-os.git
262 lines
7.4 KiB
C
262 lines
7.4 KiB
C
/*
|
|
* Copyright (c) 2014-2015 ARM Limited. All rights reserved.
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
* 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
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* 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.
|
|
*/
|
|
#include "ns_types.h"
|
|
#include "ns_list.h"
|
|
#include "timer_sys.h"
|
|
#include "platform/arm_hal_interrupt.h"
|
|
#include "platform/arm_hal_timer.h"
|
|
#include "ns_timer.h"
|
|
#include "nsdynmemLIB.h"
|
|
#include "eventOS_event.h"
|
|
#include "eventOS_callback_timer.h"
|
|
|
|
#ifndef ST_MAX
|
|
#define ST_MAX 6
|
|
#endif
|
|
|
|
typedef struct sys_timer_struct_s {
|
|
uint32_t timer_sys_launch_time;
|
|
int8_t timer_sys_launch_receiver;
|
|
uint8_t timer_sys_launch_message;
|
|
uint8_t timer_event_type;
|
|
|
|
ns_list_link_t link;
|
|
} sys_timer_struct_s;
|
|
|
|
#define TIMER_SLOTS_PER_MS 20
|
|
#define TIMER_SYS_TICK_PERIOD 10 // milliseconds
|
|
|
|
static uint32_t run_time_tick_ticks = 0;
|
|
static NS_LIST_DEFINE(system_timer_free, sys_timer_struct_s, link);
|
|
static NS_LIST_DEFINE(system_timer_list, sys_timer_struct_s, link);
|
|
|
|
|
|
static sys_timer_struct_s *sys_timer_dynamically_allocate(void);
|
|
static void timer_sys_interrupt(void);
|
|
|
|
#ifndef NS_EVENTLOOP_USE_TICK_TIMER
|
|
static int8_t platform_tick_timer_start(uint32_t period_ms);
|
|
/* Implement platform tick timer using eventOS timer */
|
|
// platform tick timer callback function
|
|
static void (*tick_timer_callback)(void);
|
|
static int8_t tick_timer_id = -1; // eventOS timer id for tick timer
|
|
|
|
// EventOS timer callback function
|
|
static void tick_timer_eventOS_callback(int8_t timer_id, uint16_t slots)
|
|
{
|
|
// Not interested in timer id or slots
|
|
(void)slots;
|
|
// Call the tick timer callback
|
|
if (tick_timer_callback != NULL && timer_id == tick_timer_id) {
|
|
platform_tick_timer_start(TIMER_SYS_TICK_PERIOD);
|
|
tick_timer_callback();
|
|
}
|
|
}
|
|
|
|
static int8_t platform_tick_timer_register(void (*tick_timer_cb)(void))
|
|
{
|
|
tick_timer_callback = tick_timer_cb;
|
|
tick_timer_id = eventOS_callback_timer_register(tick_timer_eventOS_callback);
|
|
return tick_timer_id;
|
|
}
|
|
|
|
static int8_t platform_tick_timer_start(uint32_t period_ms)
|
|
{
|
|
return eventOS_callback_timer_start(tick_timer_id, TIMER_SLOTS_PER_MS * period_ms);
|
|
}
|
|
|
|
static int8_t platform_tick_timer_stop(void)
|
|
{
|
|
return eventOS_callback_timer_stop(tick_timer_id);
|
|
}
|
|
#endif // !NS_EVENTLOOP_USE_TICK_TIMER
|
|
|
|
/*
|
|
* Initializes timers and starts system timer
|
|
*/
|
|
void timer_sys_init(void)
|
|
{
|
|
run_time_tick_ticks = 0;
|
|
|
|
// Clear old timers
|
|
ns_list_foreach_safe(sys_timer_struct_s, temp, &system_timer_list) {
|
|
ns_list_remove(&system_timer_list, temp);
|
|
ns_dyn_mem_free(temp);
|
|
}
|
|
// Clear old free timer entrys
|
|
ns_list_foreach_safe(sys_timer_struct_s, temp, &system_timer_free) {
|
|
ns_list_remove(&system_timer_free, temp);
|
|
ns_dyn_mem_free(temp);
|
|
}
|
|
|
|
for (uint8_t i = 0; i < ST_MAX; i++) {
|
|
sys_timer_struct_s *temp = sys_timer_dynamically_allocate();
|
|
if (temp) {
|
|
ns_list_add_to_start(&system_timer_free, temp);
|
|
}
|
|
}
|
|
|
|
platform_tick_timer_register(timer_sys_interrupt);
|
|
platform_tick_timer_start(TIMER_SYS_TICK_PERIOD);
|
|
}
|
|
|
|
|
|
|
|
/*-------------------SYSTEM TIMER FUNCTIONS--------------------------*/
|
|
void timer_sys_disable(void)
|
|
{
|
|
platform_tick_timer_stop();
|
|
}
|
|
|
|
/*
|
|
* Starts ticking system timer interrupts every 10ms
|
|
*/
|
|
int8_t timer_sys_wakeup(void)
|
|
{
|
|
return platform_tick_timer_start(TIMER_SYS_TICK_PERIOD);
|
|
}
|
|
|
|
|
|
static void timer_sys_interrupt(void)
|
|
{
|
|
system_timer_tick_update(1);
|
|
}
|
|
|
|
|
|
|
|
/* * * * * * * * * */
|
|
|
|
static sys_timer_struct_s *sys_timer_dynamically_allocate(void)
|
|
{
|
|
return (sys_timer_struct_s*)ns_dyn_mem_alloc(sizeof(sys_timer_struct_s));
|
|
}
|
|
|
|
static sys_timer_struct_s *timer_struct_get(void)
|
|
{
|
|
sys_timer_struct_s *timer;
|
|
platform_enter_critical();
|
|
timer = ns_list_get_first(&system_timer_free);
|
|
if (timer) {
|
|
ns_list_remove(&system_timer_free, timer);
|
|
} else {
|
|
timer = sys_timer_dynamically_allocate();
|
|
}
|
|
platform_exit_critical();
|
|
return timer;
|
|
}
|
|
|
|
uint32_t timer_get_runtime_ticks(void) // only used in dev_stats_internal.c
|
|
{
|
|
uint32_t ret_val;
|
|
platform_enter_critical();
|
|
ret_val = run_time_tick_ticks;
|
|
platform_exit_critical();
|
|
return ret_val;
|
|
}
|
|
|
|
|
|
|
|
int8_t eventOS_event_timer_request(uint8_t snmessage, uint8_t event_type, int8_t tasklet_id, uint32_t time)
|
|
{
|
|
int8_t res = -1;
|
|
sys_timer_struct_s *timer = NULL;
|
|
|
|
platform_enter_critical();
|
|
// Note that someone wanting 20ms gets 2 ticks, thanks to this test. 30ms would be 4 ticks.
|
|
// And why shouldn't they be able to get a 1-tick callback?
|
|
if (time > 2 * TIMER_SYS_TICK_PERIOD) {
|
|
time /= TIMER_SYS_TICK_PERIOD;
|
|
// XXX Why this? Someone wanting 50ms shouldn't get 6 ticks. Round to nearest, maybe?
|
|
time++;
|
|
} else {
|
|
time = 2;
|
|
}
|
|
timer = timer_struct_get();
|
|
if (timer) {
|
|
timer->timer_sys_launch_message = snmessage;
|
|
timer->timer_sys_launch_receiver = tasklet_id;
|
|
timer->timer_event_type = event_type;
|
|
timer->timer_sys_launch_time = time;
|
|
ns_list_add_to_start(&system_timer_list, timer);
|
|
res = 0;
|
|
}
|
|
platform_exit_critical();
|
|
return res;
|
|
}
|
|
|
|
int8_t eventOS_event_timer_cancel(uint8_t snmessage, int8_t tasklet_id)
|
|
{
|
|
int8_t res = -1;
|
|
platform_enter_critical();
|
|
ns_list_foreach(sys_timer_struct_s, cur, &system_timer_list) {
|
|
if (cur->timer_sys_launch_receiver == tasklet_id && cur->timer_sys_launch_message == snmessage) {
|
|
ns_list_remove(&system_timer_list, cur);
|
|
ns_list_add_to_start(&system_timer_free, cur);
|
|
res = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
platform_exit_critical();
|
|
return res;
|
|
}
|
|
|
|
|
|
uint32_t eventOS_event_timer_shortest_active_timer(void)
|
|
{
|
|
uint32_t ret_val = 0;
|
|
|
|
platform_enter_critical();
|
|
ns_list_foreach(sys_timer_struct_s, cur, &system_timer_list) {
|
|
if (ret_val == 0 || cur->timer_sys_launch_time < ret_val) {
|
|
ret_val = cur->timer_sys_launch_time;
|
|
}
|
|
}
|
|
|
|
platform_exit_critical();
|
|
//Convert ticks to ms
|
|
ret_val *= TIMER_SYS_TICK_PERIOD;
|
|
return ret_val;
|
|
}
|
|
|
|
void system_timer_tick_update(uint32_t ticks)
|
|
{
|
|
platform_enter_critical();
|
|
//Keep runtime time
|
|
run_time_tick_ticks += ticks;
|
|
ns_list_foreach_safe(sys_timer_struct_s, cur, &system_timer_list) {
|
|
if (cur->timer_sys_launch_time <= ticks) {
|
|
arm_event_s event = {
|
|
.receiver = cur->timer_sys_launch_receiver,
|
|
.sender = 0, /**< Event sender Tasklet ID */
|
|
.data_ptr = NULL,
|
|
.event_type = cur->timer_event_type,
|
|
.event_id = cur->timer_sys_launch_message,
|
|
.event_data = 0,
|
|
.priority = ARM_LIB_MED_PRIORITY_EVENT,
|
|
};
|
|
eventOS_event_send(&event);
|
|
ns_list_remove(&system_timer_list, cur);
|
|
ns_list_add_to_start(&system_timer_free, cur);
|
|
} else {
|
|
cur->timer_sys_launch_time -= ticks;
|
|
}
|
|
}
|
|
|
|
platform_exit_critical();
|
|
}
|
|
|