/* * Copyright (C) 2021 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 "derive_classpath.h" bool ArgumentMatches(std::string_view argument, std::string_view prefix, std::string_view* value) { if (android::base::StartsWith(argument, prefix)) { *value = argument.substr(prefix.size()); return true; } return false; } // Command line flags need to be considered as a de facto API since there may be callers outside // of the SdkExtensions APEX, which needs to run on older Android versions. For example, otapreopt // currently executes derive_classpath with a single output file. When changing the flags, make sure // it won't break on older Android. bool ParseArgs(android::derive_classpath::Args& args, int argc, char** argv) { // Parse flags std::vector positional_args; for (int i = 1; i < argc; ++i) { const std::string_view arg = argv[i]; std::string_view value; if (ArgumentMatches(arg, "--bootclasspath-fragment=", &value)) { if (!args.system_bootclasspath_fragment.empty()) { LOG(ERROR) << "Duplicated flag --bootclasspath-fragment is specified"; return false; } args.system_bootclasspath_fragment = value; } else if (ArgumentMatches(arg, "--systemserverclasspath-fragment=", &value)) { if (!args.system_systemserverclasspath_fragment.empty()) { LOG(ERROR) << "Duplicated flag --systemserverclasspath-fragment is specified"; return false; } args.system_systemserverclasspath_fragment = value; } else if (ArgumentMatches(arg, "--scan-dirs=", &value)) { if (!args.scan_dirs.empty()) { LOG(ERROR) << "Duplicated flag --scan-dirs is specified"; return false; } args.scan_dirs = android::base::Split(std::string(value), ","); } else if (ArgumentMatches(arg, "--glob-pattern-prefix=", &value)) { if (!args.glob_pattern_prefix.empty()) { LOG(ERROR) << "Duplicated flag --glob-pattern-prefix is specified"; return false; } args.glob_pattern_prefix = value; } else if (ArgumentMatches(arg, "--override-device-sdk-version=", &value)) { if (args.override_device_sdk_version != 0) { LOG(ERROR) << "Duplicated flag --override-device-sdk-version is specified"; return false; } if (!android::base::ParseInt(std::string(value), &args.override_device_sdk_version, /*min=*/1, /*max=*/INT_MAX)) { PLOG(ERROR) << "Invalid value for --override-device-sdk-version \"" << value << "\""; return false; } } else if (ArgumentMatches(arg, "--override-device-codename=", &value)) { if (!args.override_device_codename.empty()) { LOG(ERROR) << "Duplicated flag --override-device-codename is specified"; return false; } args.override_device_codename = value; } else if (ArgumentMatches(arg, "--override-device-known-codenames=", &value)) { if (!args.override_device_known_codenames.empty()) { LOG(ERROR) << "Duplicated flag --override-device-known-codenames is specified"; return false; } std::vector known_codenames = android::base::Split(std::string(value), ","); std::move(known_codenames.begin(), known_codenames.end(), std::inserter(args.override_device_known_codenames, args.override_device_known_codenames.end())); } else { positional_args.emplace_back(arg); } } // Validate flag combinations if (!args.scan_dirs.empty() && (!args.system_bootclasspath_fragment.empty() || !args.system_systemserverclasspath_fragment.empty())) { LOG(ERROR) << "--scan-dirs should not be accompanied by --bootclasspath-fragment or " "--systemserverclasspath-fragment"; return false; } if (!args.glob_pattern_prefix.empty() && (!args.scan_dirs.empty() || !args.system_bootclasspath_fragment.empty() || !args.system_systemserverclasspath_fragment.empty())) { LOG(ERROR) << "--glob-pattern-prefix should not be accompanied by --scan-dirs, " "--bootclasspath-fragment or --systemserverclasspath-fragment"; return false; } if (args.override_device_sdk_version != 0 && args.override_device_codename.empty()) { LOG(ERROR) << "--override-device-sdk-version should be accompanied by --override-device-codename"; return false; } if (!args.override_device_codename.empty() && args.override_device_codename != "REL" && args.override_device_known_codenames.empty()) { LOG(ERROR) << "--override-device-codename should be accompanied by " "--override-device-known-codenames, unless it is set to \"REL\""; return false; } if (args.override_device_sdk_version == 0 && (!args.override_device_codename.empty() || !args.override_device_known_codenames.empty())) { LOG(ERROR) << "--override-device-codename and --override-device-known-codenames should not " "be specified without --override-device-sdk-version"; return false; } #ifndef SDKEXT_ANDROID if (args.glob_pattern_prefix.empty() && args.scan_dirs.empty()) { LOG(ERROR) << "Either --glob-pattern-prefix or --scan-dirs must be specified on host"; return false; } if (args.override_device_sdk_version == 0) { LOG(ERROR) << "--override-device-sdk-version and --override-device-codename must be specified on host"; return false; } #endif // Handle positional args if (positional_args.size() == 0) { #ifndef SDKEXT_ANDROID LOG(ERROR) << "Output path must be specified on host"; return false; #endif args.output_path = android::derive_classpath::kGeneratedClasspathExportsFilepath; } else if (positional_args.size() == 1) { args.output_path = positional_args[0]; } else { LOG(ERROR) << "Unrecognized positional arguments: " << android::base::Join(positional_args, ' '); return false; } return true; } int main(int argc, char** argv) { android::derive_classpath::Args args; if (!ParseArgs(args, argc, argv)) { return EXIT_FAILURE; } if (!android::derive_classpath::GenerateClasspathExports(args)) { return EXIT_FAILURE; } return EXIT_SUCCESS; }