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