/* * Copyright (C) 2018 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 <thread> #include <getopt.h> #include <unistd.h> #include <android-base/file.h> #include <android-base/logging.h> #include <sys/types.h> #include "perfmgr/HintManager.h" namespace android { namespace perfmgr { class NodeVerifier : public HintManager { public: static bool VerifyNodes(const std::string& config_path) { std::string json_doc; if (!android::base::ReadFileToString(config_path, &json_doc)) { LOG(ERROR) << "Failed to read JSON config from " << config_path; return false; } std::vector<std::unique_ptr<Node>> nodes = ParseNodes(json_doc); if (nodes.empty()) { LOG(ERROR) << "Failed to parse Nodes section from " << config_path; return false; } for (const auto& node : nodes) { std::vector<std::string> values = node->GetValues(); std::string default_value = values[node->GetDefaultIndex()]; // Always set to default first values.insert(values.begin(), default_value); // And reset to default after test values.push_back(default_value); for (const auto& value : values) { if (!android::base::WriteStringToFile(value, node->GetPath())) { LOG(ERROR) << "Failed to write to node: " << node->GetPath() << " with value: " << value; return false; } LOG(VERBOSE) << "Wrote to node: " << node->GetPath() << " with value: " << value; } } return true; } private: NodeVerifier(sp<NodeLooperThread> nm, const std::map<std::string, std::vector<NodeAction>>& actions) : HintManager(std::move(nm), actions) {} }; } // namespace perfmgr } // namespace android static void printUsage(const char* exec_name) { std::string usage = exec_name; usage = usage + " is a command-line tool to verify Nodes in Json config are writable.\n" "Usages:\n" " [su system] " + exec_name + " [options]\n" "\n" "Options:\n" " --config [PATH], -c\n" " path to Json config file\n\n" " --exec_hint, -e\n" " do hints in Json config\n\n" " --help, -h\n" " print this message\n\n" " --verbose, -v\n" " print verbose log during execution\n\n"; LOG(INFO) << usage; } static void execConfig(const std::string& json_file) { std::unique_ptr<android::perfmgr::HintManager> hm = android::perfmgr::HintManager::GetFromJSON(json_file); if (!hm.get() || !hm->IsRunning()) { LOG(ERROR) << "Failed to Parse JSON config"; } std::vector<std::string> hints = hm->GetHints(); for (const auto& hint : hints) { LOG(INFO) << "Do hint: " << hint; hm->DoHint(hint); std::this_thread::yield(); std::this_thread::sleep_for(std::chrono::milliseconds(50)); LOG(INFO) << "End hint: " << hint; hm->EndHint(hint); std::this_thread::yield(); std::this_thread::sleep_for(std::chrono::milliseconds(50)); } } int main(int argc, char* argv[]) { android::base::InitLogging(argv, android::base::StderrLogger); if (getuid() == 0) { LOG(WARNING) << "Running as root might mask node permission"; } std::string config_path; bool exec_hint = false; while (true) { static struct option opts[] = { {"config", required_argument, nullptr, 'c'}, {"exec_hint", no_argument, nullptr, 'e'}, {"help", no_argument, nullptr, 'h'}, {"verbose", no_argument, nullptr, 'v'}, {0, 0, 0, 0} // termination of the option list }; int option_index = 0; int c = getopt_long(argc, argv, "c:ehv", opts, &option_index); if (c == -1) { break; } switch (c) { case 'c': config_path = optarg; break; case 'e': exec_hint = true; break; case 'v': android::base::SetMinimumLogSeverity(android::base::VERBOSE); break; case 'h': printUsage(argv[0]); return 0; default: // getopt already prints "invalid option -- %c" for us. return 1; } } if (config_path.empty()) { LOG(ERROR) << "Need specify JSON config"; printUsage(argv[0]); return 1; } if (exec_hint) { execConfig(config_path); return 0; } if (android::perfmgr::NodeVerifier::VerifyNodes(config_path)) { LOG(INFO) << "Verified writing to JSON config"; return 0; } else { LOG(ERROR) << "Failed to verify nodes in JSON config"; return 1; } }