xref: /aosp_15_r20/system/chre/test/simulation/timer_test.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_api/chre/re.h"
18*84e33947SAndroid Build Coastguard Worker 
19*84e33947SAndroid Build Coastguard Worker #include <cstdint>
20*84e33947SAndroid Build Coastguard Worker 
21*84e33947SAndroid Build Coastguard Worker #include "chre/core/event_loop_manager.h"
22*84e33947SAndroid Build Coastguard Worker #include "chre/core/settings.h"
23*84e33947SAndroid Build Coastguard Worker #include "chre/platform/log.h"
24*84e33947SAndroid Build Coastguard Worker #include "chre/util/time.h"
25*84e33947SAndroid Build Coastguard Worker #include "chre_api/chre/event.h"
26*84e33947SAndroid Build Coastguard Worker 
27*84e33947SAndroid Build Coastguard Worker #include "gtest/gtest.h"
28*84e33947SAndroid Build Coastguard Worker #include "inc/test_util.h"
29*84e33947SAndroid Build Coastguard Worker #include "test_base.h"
30*84e33947SAndroid Build Coastguard Worker #include "test_event.h"
31*84e33947SAndroid Build Coastguard Worker #include "test_event_queue.h"
32*84e33947SAndroid Build Coastguard Worker #include "test_util.h"
33*84e33947SAndroid Build Coastguard Worker 
34*84e33947SAndroid Build Coastguard Worker namespace chre {
35*84e33947SAndroid Build Coastguard Worker 
36*84e33947SAndroid Build Coastguard Worker // TestTimer is required to access private members of the TimerPool.
37*84e33947SAndroid Build Coastguard Worker class TestTimer : public TestBase {
38*84e33947SAndroid Build Coastguard Worker  protected:
hasNanoappTimers(TimerPool & pool,uint16_t instanceId)39*84e33947SAndroid Build Coastguard Worker   bool hasNanoappTimers(TimerPool &pool, uint16_t instanceId) {
40*84e33947SAndroid Build Coastguard Worker     return pool.hasNanoappTimers(instanceId);
41*84e33947SAndroid Build Coastguard Worker   }
42*84e33947SAndroid Build Coastguard Worker };
43*84e33947SAndroid Build Coastguard Worker 
44*84e33947SAndroid Build Coastguard Worker namespace {
TEST_F(TestTimer,SetupAndCancelPeriodicTimer)45*84e33947SAndroid Build Coastguard Worker TEST_F(TestTimer, SetupAndCancelPeriodicTimer) {
46*84e33947SAndroid Build Coastguard Worker   CREATE_CHRE_TEST_EVENT(START_TIMER, 0);
47*84e33947SAndroid Build Coastguard Worker   CREATE_CHRE_TEST_EVENT(STOP_TIMER, 1);
48*84e33947SAndroid Build Coastguard Worker 
49*84e33947SAndroid Build Coastguard Worker   class App : public TestNanoapp {
50*84e33947SAndroid Build Coastguard Worker    public:
51*84e33947SAndroid Build Coastguard Worker     void handleEvent(uint32_t, uint16_t eventType,
52*84e33947SAndroid Build Coastguard Worker                      const void *eventData) override {
53*84e33947SAndroid Build Coastguard Worker       switch (eventType) {
54*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TIMER: {
55*84e33947SAndroid Build Coastguard Worker           auto data = static_cast<const uint32_t *>(eventData);
56*84e33947SAndroid Build Coastguard Worker           if (*data == mCookie) {
57*84e33947SAndroid Build Coastguard Worker             mCount++;
58*84e33947SAndroid Build Coastguard Worker             if (mCount == 3) {
59*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(CHRE_EVENT_TIMER);
60*84e33947SAndroid Build Coastguard Worker             }
61*84e33947SAndroid Build Coastguard Worker           }
62*84e33947SAndroid Build Coastguard Worker           break;
63*84e33947SAndroid Build Coastguard Worker         }
64*84e33947SAndroid Build Coastguard Worker 
65*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TEST_EVENT: {
66*84e33947SAndroid Build Coastguard Worker           auto event = static_cast<const TestEvent *>(eventData);
67*84e33947SAndroid Build Coastguard Worker           switch (event->type) {
68*84e33947SAndroid Build Coastguard Worker             case START_TIMER: {
69*84e33947SAndroid Build Coastguard Worker               uint32_t handle = chreTimerSet(10 * kOneMillisecondInNanoseconds,
70*84e33947SAndroid Build Coastguard Worker                                              &mCookie, false /*oneShot*/);
71*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(START_TIMER, handle);
72*84e33947SAndroid Build Coastguard Worker               break;
73*84e33947SAndroid Build Coastguard Worker             }
74*84e33947SAndroid Build Coastguard Worker             case STOP_TIMER: {
75*84e33947SAndroid Build Coastguard Worker               auto handle = static_cast<const uint32_t *>(event->data);
76*84e33947SAndroid Build Coastguard Worker               bool success = chreTimerCancel(*handle);
77*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(STOP_TIMER, success);
78*84e33947SAndroid Build Coastguard Worker               break;
79*84e33947SAndroid Build Coastguard Worker             }
80*84e33947SAndroid Build Coastguard Worker           }
81*84e33947SAndroid Build Coastguard Worker         }
82*84e33947SAndroid Build Coastguard Worker       }
83*84e33947SAndroid Build Coastguard Worker     }
84*84e33947SAndroid Build Coastguard Worker 
85*84e33947SAndroid Build Coastguard Worker    protected:
86*84e33947SAndroid Build Coastguard Worker     const uint32_t mCookie = 123;
87*84e33947SAndroid Build Coastguard Worker     int mCount = 0;
88*84e33947SAndroid Build Coastguard Worker   };
89*84e33947SAndroid Build Coastguard Worker 
90*84e33947SAndroid Build Coastguard Worker   uint64_t appId = loadNanoapp(MakeUnique<App>());
91*84e33947SAndroid Build Coastguard Worker 
92*84e33947SAndroid Build Coastguard Worker   TimerPool &timerPool =
93*84e33947SAndroid Build Coastguard Worker       EventLoopManagerSingleton::get()->getEventLoop().getTimerPool();
94*84e33947SAndroid Build Coastguard Worker 
95*84e33947SAndroid Build Coastguard Worker   uint16_t instanceId;
96*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(EventLoopManagerSingleton::get()
97*84e33947SAndroid Build Coastguard Worker                   ->getEventLoop()
98*84e33947SAndroid Build Coastguard Worker                   .findNanoappInstanceIdByAppId(appId, &instanceId));
99*84e33947SAndroid Build Coastguard Worker 
100*84e33947SAndroid Build Coastguard Worker   uint32_t handle;
101*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, START_TIMER);
102*84e33947SAndroid Build Coastguard Worker   waitForEvent(START_TIMER, &handle);
103*84e33947SAndroid Build Coastguard Worker   EXPECT_NE(handle, CHRE_TIMER_INVALID);
104*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(hasNanoappTimers(timerPool, instanceId));
105*84e33947SAndroid Build Coastguard Worker 
106*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_TIMER);
107*84e33947SAndroid Build Coastguard Worker 
108*84e33947SAndroid Build Coastguard Worker   bool success;
109*84e33947SAndroid Build Coastguard Worker 
110*84e33947SAndroid Build Coastguard Worker   // Cancelling an active timer should be successful.
111*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, STOP_TIMER, handle);
112*84e33947SAndroid Build Coastguard Worker   waitForEvent(STOP_TIMER, &success);
113*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(success);
114*84e33947SAndroid Build Coastguard Worker   EXPECT_FALSE(hasNanoappTimers(timerPool, instanceId));
115*84e33947SAndroid Build Coastguard Worker 
116*84e33947SAndroid Build Coastguard Worker   // Cancelling an inactive time should return false.
117*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, STOP_TIMER, handle);
118*84e33947SAndroid Build Coastguard Worker   waitForEvent(STOP_TIMER, &success);
119*84e33947SAndroid Build Coastguard Worker   EXPECT_FALSE(success);
120*84e33947SAndroid Build Coastguard Worker }
121*84e33947SAndroid Build Coastguard Worker 
TEST_F(TestTimer,CancelPeriodicTimerOnUnload)122*84e33947SAndroid Build Coastguard Worker TEST_F(TestTimer, CancelPeriodicTimerOnUnload) {
123*84e33947SAndroid Build Coastguard Worker   CREATE_CHRE_TEST_EVENT(START_TIMER, 0);
124*84e33947SAndroid Build Coastguard Worker 
125*84e33947SAndroid Build Coastguard Worker   class App : public TestNanoapp {
126*84e33947SAndroid Build Coastguard Worker    public:
127*84e33947SAndroid Build Coastguard Worker     void handleEvent(uint32_t, uint16_t eventType, const void *eventData) {
128*84e33947SAndroid Build Coastguard Worker       switch (eventType) {
129*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TIMER: {
130*84e33947SAndroid Build Coastguard Worker           auto data = static_cast<const uint32_t *>(eventData);
131*84e33947SAndroid Build Coastguard Worker           if (*data == mCookie) {
132*84e33947SAndroid Build Coastguard Worker             mCount++;
133*84e33947SAndroid Build Coastguard Worker             if (mCount == 3) {
134*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(CHRE_EVENT_TIMER);
135*84e33947SAndroid Build Coastguard Worker             }
136*84e33947SAndroid Build Coastguard Worker           }
137*84e33947SAndroid Build Coastguard Worker           break;
138*84e33947SAndroid Build Coastguard Worker         }
139*84e33947SAndroid Build Coastguard Worker 
140*84e33947SAndroid Build Coastguard Worker         case CHRE_EVENT_TEST_EVENT: {
141*84e33947SAndroid Build Coastguard Worker           auto event = static_cast<const TestEvent *>(eventData);
142*84e33947SAndroid Build Coastguard Worker           switch (event->type) {
143*84e33947SAndroid Build Coastguard Worker             case START_TIMER: {
144*84e33947SAndroid Build Coastguard Worker               uint32_t handle = chreTimerSet(10 * kOneMillisecondInNanoseconds,
145*84e33947SAndroid Build Coastguard Worker                                              &mCookie, false /*oneShot*/);
146*84e33947SAndroid Build Coastguard Worker               TestEventQueueSingleton::get()->pushEvent(START_TIMER, handle);
147*84e33947SAndroid Build Coastguard Worker               break;
148*84e33947SAndroid Build Coastguard Worker             }
149*84e33947SAndroid Build Coastguard Worker           }
150*84e33947SAndroid Build Coastguard Worker         }
151*84e33947SAndroid Build Coastguard Worker       }
152*84e33947SAndroid Build Coastguard Worker     }
153*84e33947SAndroid Build Coastguard Worker 
154*84e33947SAndroid Build Coastguard Worker    protected:
155*84e33947SAndroid Build Coastguard Worker     const uint32_t mCookie = 123;
156*84e33947SAndroid Build Coastguard Worker     int mCount = 0;
157*84e33947SAndroid Build Coastguard Worker   };
158*84e33947SAndroid Build Coastguard Worker 
159*84e33947SAndroid Build Coastguard Worker   uint64_t appId = loadNanoapp(MakeUnique<App>());
160*84e33947SAndroid Build Coastguard Worker 
161*84e33947SAndroid Build Coastguard Worker   TimerPool &timerPool =
162*84e33947SAndroid Build Coastguard Worker       EventLoopManagerSingleton::get()->getEventLoop().getTimerPool();
163*84e33947SAndroid Build Coastguard Worker 
164*84e33947SAndroid Build Coastguard Worker   uint16_t instanceId;
165*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(EventLoopManagerSingleton::get()
166*84e33947SAndroid Build Coastguard Worker                   ->getEventLoop()
167*84e33947SAndroid Build Coastguard Worker                   .findNanoappInstanceIdByAppId(appId, &instanceId));
168*84e33947SAndroid Build Coastguard Worker 
169*84e33947SAndroid Build Coastguard Worker   uint32_t handle;
170*84e33947SAndroid Build Coastguard Worker   sendEventToNanoapp(appId, START_TIMER);
171*84e33947SAndroid Build Coastguard Worker   waitForEvent(START_TIMER, &handle);
172*84e33947SAndroid Build Coastguard Worker   EXPECT_NE(handle, CHRE_TIMER_INVALID);
173*84e33947SAndroid Build Coastguard Worker   EXPECT_TRUE(hasNanoappTimers(timerPool, instanceId));
174*84e33947SAndroid Build Coastguard Worker 
175*84e33947SAndroid Build Coastguard Worker   waitForEvent(CHRE_EVENT_TIMER);
176*84e33947SAndroid Build Coastguard Worker 
177*84e33947SAndroid Build Coastguard Worker   unloadNanoapp(appId);
178*84e33947SAndroid Build Coastguard Worker   EXPECT_FALSE(hasNanoappTimers(timerPool, instanceId));
179*84e33947SAndroid Build Coastguard Worker }
180*84e33947SAndroid Build Coastguard Worker 
181*84e33947SAndroid Build Coastguard Worker }  // namespace
182*84e33947SAndroid Build Coastguard Worker }  // namespace chre
183