1*84e33947SAndroid Build Coastguard Worker /*
2*84e33947SAndroid Build Coastguard Worker * Copyright (C) 2022 The Android Open Source Project
3*84e33947SAndroid Build Coastguard Worker *
4*84e33947SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*84e33947SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*84e33947SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*84e33947SAndroid Build Coastguard Worker *
8*84e33947SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*84e33947SAndroid Build Coastguard Worker *
10*84e33947SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*84e33947SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*84e33947SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*84e33947SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*84e33947SAndroid Build Coastguard Worker * limitations under the License.
15*84e33947SAndroid Build Coastguard Worker */
16*84e33947SAndroid Build Coastguard Worker
17*84e33947SAndroid Build Coastguard Worker #include "chre/platform/system_timer.h"
18*84e33947SAndroid Build Coastguard Worker #include "chre/platform/fatal_error.h"
19*84e33947SAndroid Build Coastguard Worker #include "chre/platform/log.h"
20*84e33947SAndroid Build Coastguard Worker #include "chre/util/time.h"
21*84e33947SAndroid Build Coastguard Worker
22*84e33947SAndroid Build Coastguard Worker namespace chre {
23*84e33947SAndroid Build Coastguard Worker
rtTimerCallback(struct rt_timer * rtTimer)24*84e33947SAndroid Build Coastguard Worker void SystemTimerBase::rtTimerCallback(struct rt_timer *rtTimer) {
25*84e33947SAndroid Build Coastguard Worker if (rtTimer != nullptr) {
26*84e33947SAndroid Build Coastguard Worker SystemTimer *systemTimer = static_cast<SystemTimer *>(rtTimer->private_ptr);
27*84e33947SAndroid Build Coastguard Worker BaseType_t xHigherPriorityTaskWoken = pdFALSE;
28*84e33947SAndroid Build Coastguard Worker vTaskNotifyGiveFromISR(systemTimer->mCallbackRunnerHandle,
29*84e33947SAndroid Build Coastguard Worker &xHigherPriorityTaskWoken);
30*84e33947SAndroid Build Coastguard Worker portYIELD_FROM_ISR(xHigherPriorityTaskWoken);
31*84e33947SAndroid Build Coastguard Worker }
32*84e33947SAndroid Build Coastguard Worker }
33*84e33947SAndroid Build Coastguard Worker
callbackRunner(void * context)34*84e33947SAndroid Build Coastguard Worker void SystemTimerBase::callbackRunner(void *context) {
35*84e33947SAndroid Build Coastguard Worker SystemTimer *systemTimer = static_cast<SystemTimer *>(context);
36*84e33947SAndroid Build Coastguard Worker if (systemTimer == nullptr) {
37*84e33947SAndroid Build Coastguard Worker FATAL_ERROR("Null System Timer");
38*84e33947SAndroid Build Coastguard Worker }
39*84e33947SAndroid Build Coastguard Worker while (true) {
40*84e33947SAndroid Build Coastguard Worker ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
41*84e33947SAndroid Build Coastguard Worker SystemTimerCallback *callback = systemTimer->mCallback;
42*84e33947SAndroid Build Coastguard Worker if (callback != nullptr) {
43*84e33947SAndroid Build Coastguard Worker callback(systemTimer->mData);
44*84e33947SAndroid Build Coastguard Worker }
45*84e33947SAndroid Build Coastguard Worker }
46*84e33947SAndroid Build Coastguard Worker }
47*84e33947SAndroid Build Coastguard Worker
SystemTimer()48*84e33947SAndroid Build Coastguard Worker SystemTimer::SystemTimer() {
49*84e33947SAndroid Build Coastguard Worker // Initialize the rtSystemTimer struct.
50*84e33947SAndroid Build Coastguard Worker // The timer's callback and the private data won't be changed through the
51*84e33947SAndroid Build Coastguard Worker // lifetime so init() should only be run once. The creation of the callback
52*84e33947SAndroid Build Coastguard Worker // runner thread is delayed to the call of init().
53*84e33947SAndroid Build Coastguard Worker rt_timer_init(&rtSystemTimer, /* func= */ rtTimerCallback, /* data= */ this);
54*84e33947SAndroid Build Coastguard Worker }
55*84e33947SAndroid Build Coastguard Worker
~SystemTimer()56*84e33947SAndroid Build Coastguard Worker SystemTimer::~SystemTimer() {
57*84e33947SAndroid Build Coastguard Worker // cancel an existing timer if any
58*84e33947SAndroid Build Coastguard Worker cancel();
59*84e33947SAndroid Build Coastguard Worker // Delete the callback runner thread if it was created
60*84e33947SAndroid Build Coastguard Worker if (mCallbackRunnerHandle != nullptr) {
61*84e33947SAndroid Build Coastguard Worker vTaskDelete(mCallbackRunnerHandle);
62*84e33947SAndroid Build Coastguard Worker mCallbackRunnerHandle = nullptr;
63*84e33947SAndroid Build Coastguard Worker }
64*84e33947SAndroid Build Coastguard Worker }
65*84e33947SAndroid Build Coastguard Worker
init()66*84e33947SAndroid Build Coastguard Worker bool SystemTimer::init() {
67*84e33947SAndroid Build Coastguard Worker if (mInitialized) {
68*84e33947SAndroid Build Coastguard Worker return true;
69*84e33947SAndroid Build Coastguard Worker }
70*84e33947SAndroid Build Coastguard Worker BaseType_t xReturned = xTaskCreate(
71*84e33947SAndroid Build Coastguard Worker callbackRunner, kTaskName, kStackDepthWords,
72*84e33947SAndroid Build Coastguard Worker /* pvParameters= */ this, kTaskPriority, &mCallbackRunnerHandle);
73*84e33947SAndroid Build Coastguard Worker if (xReturned == pdPASS) {
74*84e33947SAndroid Build Coastguard Worker mInitialized = true;
75*84e33947SAndroid Build Coastguard Worker return true;
76*84e33947SAndroid Build Coastguard Worker }
77*84e33947SAndroid Build Coastguard Worker LOGE("Failed to create the callback runner thread");
78*84e33947SAndroid Build Coastguard Worker return false;
79*84e33947SAndroid Build Coastguard Worker }
80*84e33947SAndroid Build Coastguard Worker
set(SystemTimerCallback * callback,void * data,Nanoseconds delay)81*84e33947SAndroid Build Coastguard Worker bool SystemTimer::set(SystemTimerCallback *callback, void *data,
82*84e33947SAndroid Build Coastguard Worker Nanoseconds delay) {
83*84e33947SAndroid Build Coastguard Worker if (!mInitialized) {
84*84e33947SAndroid Build Coastguard Worker LOGW("Timer is not initialized");
85*84e33947SAndroid Build Coastguard Worker return false;
86*84e33947SAndroid Build Coastguard Worker }
87*84e33947SAndroid Build Coastguard Worker cancel();
88*84e33947SAndroid Build Coastguard Worker mCallback = callback;
89*84e33947SAndroid Build Coastguard Worker mData = data;
90*84e33947SAndroid Build Coastguard Worker rt_timer_start(&rtSystemTimer, delay.toRawNanoseconds(),
91*84e33947SAndroid Build Coastguard Worker /* oneShot= */ true);
92*84e33947SAndroid Build Coastguard Worker return true;
93*84e33947SAndroid Build Coastguard Worker }
94*84e33947SAndroid Build Coastguard Worker
cancel()95*84e33947SAndroid Build Coastguard Worker bool SystemTimer::cancel() {
96*84e33947SAndroid Build Coastguard Worker // TODO(b/254708051): This usage of critical section is pending confirmation.
97*84e33947SAndroid Build Coastguard Worker taskENTER_CRITICAL();
98*84e33947SAndroid Build Coastguard Worker if (isActive()) {
99*84e33947SAndroid Build Coastguard Worker rt_timer_stop(&rtSystemTimer);
100*84e33947SAndroid Build Coastguard Worker }
101*84e33947SAndroid Build Coastguard Worker taskEXIT_CRITICAL();
102*84e33947SAndroid Build Coastguard Worker return true;
103*84e33947SAndroid Build Coastguard Worker }
104*84e33947SAndroid Build Coastguard Worker
isActive()105*84e33947SAndroid Build Coastguard Worker bool SystemTimer::isActive() {
106*84e33947SAndroid Build Coastguard Worker return rt_timer_active(&rtSystemTimer);
107*84e33947SAndroid Build Coastguard Worker }
108*84e33947SAndroid Build Coastguard Worker
109*84e33947SAndroid Build Coastguard Worker } // namespace chre