/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specic language governing permissions and * limitations under the License. */ #include <algorithm> #include <thread> #include <android-base/file.h> #include <gtest/gtest.h> #include "perfmgr/FileNode.h" #include "perfmgr/NodeLooperThread.h" namespace android { namespace perfmgr { using namespace std::chrono_literals; constexpr auto kSLEEP_TOLERANCE_MS = 50ms; class NodeLooperThreadTest : public ::testing::Test { protected: virtual void SetUp() { std::unique_ptr<TemporaryFile> tf = std::make_unique<TemporaryFile>(); nodes_.emplace_back(new FileNode( "n0", tf->path, {{"n0_value0"}, {"n0_value1"}, {"n0_value2"}}, 2, false)); files_.emplace_back(std::move(tf)); tf = std::make_unique<TemporaryFile>(); nodes_.emplace_back(new FileNode( "n1", tf->path, {{"n1_value0"}, {"n1_value1"}, {"n1_value2"}}, 2, true)); files_.emplace_back(std::move(tf)); } virtual void TearDown() { nodes_.clear(); files_.clear(); } std::vector<std::unique_ptr<Node>> nodes_; std::vector<std::unique_ptr<TemporaryFile>> files_; }; static inline void _VerifyPathValue(const std::string& path, const std::string& value) { std::string s; EXPECT_TRUE(android::base::ReadFileToString(path, &s)) << strerror(errno); EXPECT_EQ(value, s); } // Test default value init TEST_F(NodeLooperThreadTest, InitRunTest) { sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_)); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); EXPECT_TRUE(th->isRunning()); _VerifyPathValue(files_[0]->path, ""); _VerifyPathValue(files_[1]->path, "n1_value2"); th->Stop(); EXPECT_FALSE(th->isRunning()); } // Test add request TEST_F(NodeLooperThreadTest, AddRequest) { sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_)); EXPECT_TRUE(th->isRunning()); // Dummy LAUNCH boost actions: // Node0, value0, 200ms // Node1, value1, 400ms std::vector<NodeAction> actions{{0, 0, 200ms}, {1, 1, 400ms}}; EXPECT_TRUE(th->Request(actions, "LAUNCH")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); _VerifyPathValue(files_[0]->path, "n0_value0"); _VerifyPathValue(files_[1]->path, "n1_value1"); std::this_thread::sleep_for(200ms); _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value1"); std::this_thread::sleep_for(200ms); _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value2"); th->Stop(); EXPECT_FALSE(th->isRunning()); } // Test request to override expire time TEST_F(NodeLooperThreadTest, AddRequestOverride) { sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_)); EXPECT_TRUE(th->isRunning()); // Dummy LAUNCH boost actions: // Node0, value0, 200ms // Node1, value1, 500ms std::vector<NodeAction> actions{{0, 0, 200ms}, {1, 1, 500ms}}; EXPECT_TRUE(th->Request(actions, "LAUNCH")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); _VerifyPathValue(files_[0]->path, "n0_value0"); _VerifyPathValue(files_[1]->path, "n1_value1"); // Dummy LAUNCH boost actions: // Node0, value0, 300ms will extend // Node1, value1, 100ms will not extend actions = std::vector<NodeAction>{{0, 0, 300ms}, {1, 1, 100ms}}; EXPECT_TRUE(th->Request(actions, "LAUNCH")); std::this_thread::sleep_for(200ms); _VerifyPathValue(files_[0]->path, "n0_value0"); _VerifyPathValue(files_[1]->path, "n1_value1"); std::this_thread::sleep_for(150ms); // Node0 value0 expired _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value1"); std::this_thread::sleep_for(150ms); _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value2"); th->Stop(); EXPECT_FALSE(th->isRunning()); } // Test cancel request TEST_F(NodeLooperThreadTest, CancelRequest) { sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_)); EXPECT_TRUE(th->isRunning()); // Dummy LAUNCH boost actions: // Node0, value0, forever // Node1, value1, forever std::vector<NodeAction> actions{{0, 0, 0ms}, {1, 1, 0ms}}; EXPECT_TRUE(th->Request(actions, "LAUNCH")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); _VerifyPathValue(files_[0]->path, "n0_value0"); _VerifyPathValue(files_[1]->path, "n1_value1"); EXPECT_TRUE(th->Cancel(actions, "LAUNCH")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value2"); th->Stop(); EXPECT_FALSE(th->isRunning()); } // Test multiple request TEST_F(NodeLooperThreadTest, MultipleRequest) { sp<NodeLooperThread> th = new NodeLooperThread(std::move(nodes_)); EXPECT_TRUE(th->isRunning()); // Dummy LAUNCH boost actions: // Node0, value1, 800ms // Node1, value1, forever std::vector<NodeAction> actions_interaction{{0, 1, 800ms}, {1, 1, 0ms}}; EXPECT_TRUE(th->Request(actions_interaction, "INTERACTION")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); _VerifyPathValue(files_[0]->path, "n0_value1"); _VerifyPathValue(files_[1]->path, "n1_value1"); // Dummy LAUNCH boost actions: // Node0, value0, forever // Node1, value0, 400ms std::vector<NodeAction> actions_launch{{0, 0, 0ms}, {1, 0, 400ms}}; EXPECT_TRUE(th->Request(actions_launch, "LAUNCH")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); _VerifyPathValue(files_[0]->path, "n0_value0"); _VerifyPathValue(files_[1]->path, "n1_value0"); std::this_thread::sleep_for(400ms); // "LAUNCH" node1 expired _VerifyPathValue(files_[0]->path, "n0_value0"); _VerifyPathValue(files_[1]->path, "n1_value1"); EXPECT_TRUE(th->Cancel(actions_launch, "LAUNCH")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); // "LAUNCH" canceled _VerifyPathValue(files_[0]->path, "n0_value1"); _VerifyPathValue(files_[1]->path, "n1_value1"); std::this_thread::sleep_for(400ms); // "INTERACTION" node0 expired _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value1"); EXPECT_TRUE(th->Cancel(actions_interaction, "INTERACTION")); std::this_thread::sleep_for(kSLEEP_TOLERANCE_MS); // "INTERACTION" canceled _VerifyPathValue(files_[0]->path, "n0_value2"); _VerifyPathValue(files_[1]->path, "n1_value2"); th->Stop(); EXPECT_FALSE(th->isRunning()); } } // namespace perfmgr } // namespace android