/* * Copyright (C) 2019 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 specific language governing permissions and * limitations under the License. */ #include #include #include #include #include #include #include #include #include "edify/expr.h" #include "otautil/error_code.h" #include "otautil/paths.h" #include "updater/blockimg.h" #include "updater/build_info.h" #include "updater/dynamic_partitions.h" #include "updater/install.h" #include "updater/simulator_runtime.h" #include "updater/updater.h" using namespace std::string_literals; void Usage(std::string_view name) { LOG(INFO) << "Usage: " << name << "[--oem_settings ]" << "[--skip_functions ]" << " --source " << " --ota_package "; } Value* SimulatorPlaceHolderFn(const char* name, State* /* state */, const std::vector>& /* argv */) { LOG(INFO) << "Skip function " << name << " in host simulation"; return StringValue("t"); } int main(int argc, char** argv) { // Write the logs to stdout. android::base::InitLogging(argv, &android::base::StderrLogger); std::string oem_settings; std::string skip_function_file; std::string source_target_file; std::string package_name; std::string work_dir; bool keep_images = false; constexpr struct option OPTIONS[] = { { "keep_images", no_argument, nullptr, 0 }, { "oem_settings", required_argument, nullptr, 0 }, { "ota_package", required_argument, nullptr, 0 }, { "skip_functions", required_argument, nullptr, 0 }, { "source", required_argument, nullptr, 0 }, { "work_dir", required_argument, nullptr, 0 }, { nullptr, 0, nullptr, 0 }, }; int arg; int option_index; while ((arg = getopt_long(argc, argv, "", OPTIONS, &option_index)) != -1) { if (arg != 0) { LOG(ERROR) << "Invalid command argument"; Usage(argv[0]); return EXIT_FAILURE; } auto option_name = OPTIONS[option_index].name; // The same oem property file used during OTA generation. It's needed for file_getprop() to // return the correct value for the source build. if (option_name == "oem_settings"s) { oem_settings = optarg; } else if (option_name == "skip_functions"s) { skip_function_file = optarg; } else if (option_name == "source"s) { source_target_file = optarg; } else if (option_name == "ota_package"s) { package_name = optarg; } else if (option_name == "keep_images"s) { keep_images = true; } else if (option_name == "work_dir"s) { work_dir = optarg; } else { Usage(argv[0]); return EXIT_FAILURE; } } if (source_target_file.empty() || package_name.empty()) { Usage(argv[0]); return EXIT_FAILURE; } // Configure edify's functions. RegisterBuiltins(); RegisterInstallFunctions(); RegisterBlockImageFunctions(); RegisterDynamicPartitionsFunctions(); if (!skip_function_file.empty()) { std::string content; if (!android::base::ReadFileToString(skip_function_file, &content)) { PLOG(ERROR) << "Failed to read " << skip_function_file; return EXIT_FAILURE; } auto lines = android::base::Split(content, "\n"); for (const auto& line : lines) { if (line.empty() || android::base::StartsWith(line, "#")) { continue; } RegisterFunction(line, SimulatorPlaceHolderFn); } } TemporaryFile temp_saved_source; TemporaryFile temp_last_command; TemporaryDir temp_stash_base; Paths::Get().set_cache_temp_source(temp_saved_source.path); Paths::Get().set_last_command_file(temp_last_command.path); Paths::Get().set_stash_directory_base(temp_stash_base.path); TemporaryFile cmd_pipe; TemporaryDir source_temp_dir; if (work_dir.empty()) { work_dir = source_temp_dir.path; } BuildInfo source_build_info(work_dir, keep_images); if (!source_build_info.ParseTargetFile(source_target_file, false)) { LOG(ERROR) << "Failed to parse the target file " << source_target_file; return EXIT_FAILURE; } if (!oem_settings.empty()) { CHECK_EQ(0, access(oem_settings.c_str(), R_OK)); source_build_info.SetOemSettings(oem_settings); } Updater updater(std::make_unique(&source_build_info)); if (!updater.Init(cmd_pipe.release(), package_name, false)) { return EXIT_FAILURE; } if (!updater.RunUpdate()) { return EXIT_FAILURE; } LOG(INFO) << "\nscript succeeded, result: " << updater.GetResult(); return 0; }