xref: /aosp_15_r20/system/chre/platform/tinysys/system_timer.cc (revision 84e339476a462649f82315436d70fd732297a399)
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