1 /*
2  * Copyright (c) 2023 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include <gtest/gtest.h>
17 #ifndef WITH_NO_MOCKER
18 #include <mockcpp/mockcpp.hpp>
19 #endif
20 #include <thread>
21 #include <climits>
22 #include <cstring>
23 #define private public
24 #define protected public
25 #include "eu/worker_manager.h"
26 #include "eu/scpuworker_manager.h"
27 #include "sched/task_scheduler.h"
28 #include "tm/scpu_task.h"
29 #include "sched/scheduler.h"
30 #undef private
31 #undef protected
32 #include "../common.h"
33 
34 using namespace testing;
35 #ifdef HWTEST_TESTING_EXT_ENABLE
36 using namespace testing::ext;
37 #endif
38 using namespace ffrt;
39 
40 class WorkerManagerTest : public testing::Test {
41 protected:
SetUpTestCase()42     static void SetUpTestCase()
43     {
44     }
45 
TearDownTestCase()46     static void TearDownTestCase()
47     {
48     }
49 
SetUp()50     virtual void SetUp()
51     {
52     }
53 
TearDown()54     virtual void TearDown()
55     {
56     }
57 };
58 
HWTEST_F(WorkerManagerTest, JoinRtgTest, TestSize.Level1)59 HWTEST_F(WorkerManagerTest, JoinRtgTest, TestSize.Level1)
60 {
61     CPUWorkerManager* cm = new SCPUWorkerManager();
62     QoS* qos = new QoS();
63     cm->IncWorker(*qos);
64     cm->JoinRtg(*qos);
65 
66     delete qos;
67     delete cm;
68 }
69 
HWTEST_F(WorkerManagerTest, IncWorkerTest, TestSize.Level1)70 HWTEST_F(WorkerManagerTest, IncWorkerTest, TestSize.Level1)
71 {
72     CPUWorkerManager* cm = new SCPUWorkerManager();
73     QoS* qos = new QoS(-1);
74     cm->IncWorker(*qos);
75 
76     delete qos;
77     delete cm;
78 }
79 
HWTEST_F(WorkerManagerTest, GetWorkerCountTest, TestSize.Level1)80 HWTEST_F(WorkerManagerTest, GetWorkerCountTest, TestSize.Level1)
81 {
82     CPUWorkerManager* cm = new SCPUWorkerManager();
83     QoS* qos = new QoS(2);
84     cm->GetWorkerCount(*qos);
85 
86     delete qos;
87     delete cm;
88 }
89 
HWTEST_F(WorkerManagerTest, JoinTGTest, TestSize.Level1)90 HWTEST_F(WorkerManagerTest, JoinTGTest, TestSize.Level1)
91 {
92     CPUWorkerManager* cm = new SCPUWorkerManager();
93     QoS* qos = new QoS(ffrt::qos_deadline_request);
94     ThreadGroup* tg = cm->JoinTG(*qos);
95     EXPECT_NE(tg, nullptr);
96 
97     QoS* qos1 = new QoS(ffrt::qos_user_interactive);
98     ThreadGroup* tg1 = cm->JoinTG(*qos1);
99     EXPECT_EQ(tg1, nullptr);
100 }
101 
HWTEST_F(WorkerManagerTest, LeaveTGTest, TestSize.Level1)102 HWTEST_F(WorkerManagerTest, LeaveTGTest, TestSize.Level1)
103 {
104     CPUWorkerManager* cm = new SCPUWorkerManager();
105     QoS* qos = new QoS(ffrt::qos_deadline_request);
106     cm->IncWorker(*qos);
107 #ifndef WITH_NO_MOCKER
108     MOCKER_CPP(&RTGCtrl::GetThreadGroup).stubs().will(returnValue(1));
109     MOCKER_CPP(&RTGCtrl::PutThreadGroup).stubs().will(returnValue(true));
110     MOCKER_CPP(&RTGCtrl::JoinThread).stubs().will(returnValue(true));
111     MOCKER_CPP(&RTGCtrl::RemoveThread).stubs().will(returnValue(true));
112 #endif
113     cm->JoinTG(*qos);
114     cm->LeaveTG(*qos);
115 
116     delete qos;
117     delete cm;
118 #ifndef WITH_NO_MOCKER
119     GlobalMockObject::verify();
120 #endif
121 }
122 
123 #ifdef FFRT_GITEE
HWTEST_F(WorkerManagerTest, CPUManagerStrategyApiTest, TestSize.Level1)124 HWTEST_F(WorkerManagerTest, CPUManagerStrategyApiTest, TestSize.Level1)
125 {
126     WorkerManager* manager = new SCPUWorkerManager();
127 
128     CPUMonitor* monitor = CPUManagerStrategy::CreateCPUMonitor(QoS(2), manager);
129     EXPECT_NE(monitor, nullptr);
130     delete monitor;
131 
132     WorkerThread* worker = CPUManagerStrategy::CreateCPUWorker(QoS(2), manager);
133     EXPECT_NE(worker, nullptr);
134 
135     delete manager;
136     worker->Join();
137     delete worker;
138 }
139 #endif
140 
HWTEST_F(WorkerManagerTest, CPUWorkerStandardLoopTest, TestSize.Level1)141 HWTEST_F(WorkerManagerTest, CPUWorkerStandardLoopTest, TestSize.Level1)
142 {
143     CPUWorkerManager* manager = new SCPUWorkerManager();
144 
145     CpuWorkerOps ops {
146         CPUWorker::WorkerLooperStandard,
147         std::bind(&CPUWorkerManager::PickUpTaskFromGlobalQueue, manager, std::placeholders::_1),
148         std::bind(&CPUWorkerManager::NotifyTaskPicked, manager, std::placeholders::_1),
149         std::bind(&CPUWorkerManager::WorkerIdleActionSimplified, manager, std::placeholders::_1),
150         std::bind(&CPUWorkerManager::WorkerRetired, manager, std::placeholders::_1),
151         std::bind(&CPUWorkerManager::WorkerPrepare, manager, std::placeholders::_1),
152         std::bind(&CPUWorkerManager::TryPoll, manager, std::placeholders::_1, std::placeholders::_2),
153         std::bind(&CPUWorkerManager::StealTaskBatch, manager, std::placeholders::_1),
154         std::bind(&CPUWorkerManager::PickUpTaskBatch, manager, std::placeholders::_1),
155 #ifdef FFRT_WORKERS_DYNAMIC_SCALING
156         std::bind(&CPUWorkerManager::IsExceedRunningThreshold, manager, std::placeholders::_1),
157         std::bind(&CPUWorkerManager::IsBlockAwareInit, manager),
158 #endif
159     };
160     WorkerThread* worker = new CPUWorker(QoS(2), std::move(ops), manager);
161     EXPECT_NE(worker, nullptr);
162     sleep(1); // wait worker into wait action
163     manager->NotifyTaskAdded(QoS(2)); // wake worker
164 
165     delete manager;
166     worker->Join();
167     delete worker;
168 }
169 
HWTEST_F(WorkerManagerTest, CPUMonitorHandleNotifyConservativeTest, TestSize.Level1)170 HWTEST_F(WorkerManagerTest, CPUMonitorHandleNotifyConservativeTest, TestSize.Level1)
171 {
172     SCPUWorkerManager* manager = new SCPUWorkerManager();
173 
174     CpuMonitorOps monitorOps { // change monitor's notify handle strategy
175         std::bind(&SCPUWorkerManager::IncWorker, manager, std::placeholders::_1),
176         std::bind(&SCPUWorkerManager::WakeupWorkers, manager, std::placeholders::_1),
177         std::bind(&SCPUWorkerManager::GetTaskCount, manager, std::placeholders::_1),
178         std::bind(&SCPUWorkerManager::GetWorkerCount, manager, std::placeholders::_1),
179         CPUMonitor::HandleTaskNotifyConservative,
180     };
181     manager->monitor->ops = std::move(monitorOps);
182 
183     manager->NotifyTaskAdded(QoS(2)); // task notify event
184 }
185 
GetTaskCountStub(const QoS& qos)186 int GetTaskCountStub(const QoS& qos)
187 {
188     return 1;
189 }
190 
191 /*
192  * 测试用例名称:CPUMonitorHandleTaskNotifyUltraConservativeTest
193  * 测试用例描述:ffrt保守调度策略
194  * 预置条件    :创建SCPUWorkerManager,策略设置为HandleTaskNotifyUltraConservative,GetTaskCount方法打桩为GetTaskCountStub
195  * 操作步骤    :调用SCPUWorkerManager的Notify方法
196  * 预期结果    :成功执行HandleTaskNotifyUltraConservative方法
197  */
HWTEST_F(WorkerManagerTest, CPUMonitorHandleTaskNotifyUltraConservativeTest, TestSize.Level1)198 HWTEST_F(WorkerManagerTest, CPUMonitorHandleTaskNotifyUltraConservativeTest, TestSize.Level1)
199 {
200     SCPUWorkerManager* manager = new SCPUWorkerManager();
201 
202     CpuMonitorOps monitorOps { // change monitor's notify handle strategy
203         std::bind(&SCPUWorkerManager::IncWorker, manager, std::placeholders::_1),
204         std::bind(&SCPUWorkerManager::WakeupWorkers, manager, std::placeholders::_1),
205         std::bind(&GetTaskCountStub, std::placeholders::_1),
206         std::bind(&SCPUWorkerManager::GetWorkerCount, manager, std::placeholders::_1),
207         CPUMonitor::HandleTaskNotifyUltraConservative,
208     };
209     manager->monitor->ops = std::move(monitorOps);
210 
211     manager->NotifyTaskAdded(QoS(2)); // task notify event
212 
213     manager->monitor->ctrlQueue[2].sleepingWorkerNum = 1;
214     manager->NotifyTaskAdded(QoS(2)); // task notify event
215 }
216 
HWTEST_F(WorkerManagerTest, PickUpTaskFromGlobalQueue, TestSize.Level1)217 HWTEST_F(WorkerManagerTest, PickUpTaskFromGlobalQueue, TestSize.Level1)
218 {
219     CPUWorkerManager* manager = new SCPUWorkerManager();
220     CPUManagerStrategy* strategy = new CPUManagerStrategy();
221     SCPUEUTask* task = new SCPUEUTask(nullptr, nullptr, 0, QoS(qos(0)));
222 
223     auto worker = strategy->CreateCPUWorker(QoS(qos(0)), manager);
224     auto& sched = FFRTScheduler::Instance()->GetScheduler(worker->GetQos());
225 
226     int ret = sched.WakeupTask(reinterpret_cast<CPUEUTask*>(task));
227     EXPECT_EQ(ret, 1);
228 
229     auto pickTask = manager->PickUpTaskFromGlobalQueue(worker);
230     EXPECT_NE(pickTask, nullptr);
231 
232     delete worker;
233     delete task;
234     delete strategy;
235     delete manager;
236 }
237 
HWTEST_F(WorkerManagerTest, PickUpTaskBatch, TestSize.Level1)238 HWTEST_F(WorkerManagerTest, PickUpTaskBatch, TestSize.Level1)
239 {
240     CPUWorkerManager* manager = new SCPUWorkerManager();
241     CPUManagerStrategy* strategy = new CPUManagerStrategy();
242     SCPUEUTask* task1 = new SCPUEUTask(nullptr, nullptr, 0, QoS(qos(0)));
243     SCPUEUTask* task2 = new SCPUEUTask(nullptr, nullptr, 0, QoS(qos(0)));
244     CPUMonitor* monitor = manager->GetCPUMonitor();
245 
246     auto worker1 = strategy->CreateCPUWorker(QoS(qos(0)), manager);
247     auto worker2 = strategy->CreateCPUWorker(QoS(qos(0)), manager);
248 
249     monitor->WakeupDeepSleep(QoS(qos(0)), false);
250     monitor->WakeupDeepSleep(QoS(qos(0)), false);
251 
252     auto& sched1 = FFRTScheduler::Instance()->GetScheduler(worker1->GetQos());
253     auto& sched2 = FFRTScheduler::Instance()->GetScheduler(worker2->GetQos());
254 
255     EXPECT_EQ(sched1.WakeupTask(reinterpret_cast<CPUEUTask*>(task1)), 1);
256     EXPECT_EQ(sched2.WakeupTask(reinterpret_cast<CPUEUTask*>(task2)), 1);
257 
258     EXPECT_NE(manager->PickUpTaskBatch(worker1), nullptr);
259     EXPECT_NE(manager->PickUpTaskBatch(worker2), nullptr);
260 
261     delete worker1;
262     delete worker2;
263     delete task1;
264     delete task2;
265     delete strategy;
266     delete manager;
267 }