1*d57664e9SAndroid Build Coastguard Worker /* 2*d57664e9SAndroid Build Coastguard Worker * Copyright (C) 2017 The Android Open Source Project 3*d57664e9SAndroid Build Coastguard Worker * 4*d57664e9SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*d57664e9SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*d57664e9SAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*d57664e9SAndroid Build Coastguard Worker * 8*d57664e9SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*d57664e9SAndroid Build Coastguard Worker * 10*d57664e9SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*d57664e9SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*d57664e9SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*d57664e9SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*d57664e9SAndroid Build Coastguard Worker * limitations under the License. 15*d57664e9SAndroid Build Coastguard Worker */ 16*d57664e9SAndroid Build Coastguard Worker 17*d57664e9SAndroid Build Coastguard Worker #ifndef AAPT2_CONFIGURATION_H 18*d57664e9SAndroid Build Coastguard Worker #define AAPT2_CONFIGURATION_H 19*d57664e9SAndroid Build Coastguard Worker 20*d57664e9SAndroid Build Coastguard Worker #include <optional> 21*d57664e9SAndroid Build Coastguard Worker #include <set> 22*d57664e9SAndroid Build Coastguard Worker #include <string> 23*d57664e9SAndroid Build Coastguard Worker #include <unordered_map> 24*d57664e9SAndroid Build Coastguard Worker #include <vector> 25*d57664e9SAndroid Build Coastguard Worker 26*d57664e9SAndroid Build Coastguard Worker #include "androidfw/ConfigDescription.h" 27*d57664e9SAndroid Build Coastguard Worker #include "androidfw/IDiagnostics.h" 28*d57664e9SAndroid Build Coastguard Worker 29*d57664e9SAndroid Build Coastguard Worker namespace aapt { 30*d57664e9SAndroid Build Coastguard Worker 31*d57664e9SAndroid Build Coastguard Worker namespace configuration { 32*d57664e9SAndroid Build Coastguard Worker 33*d57664e9SAndroid Build Coastguard Worker /** Enumeration of currently supported ABIs. */ 34*d57664e9SAndroid Build Coastguard Worker enum class Abi { 35*d57664e9SAndroid Build Coastguard Worker kArmeV6, 36*d57664e9SAndroid Build Coastguard Worker kArmV7a, 37*d57664e9SAndroid Build Coastguard Worker kArm64V8a, 38*d57664e9SAndroid Build Coastguard Worker kX86, 39*d57664e9SAndroid Build Coastguard Worker kX86_64, 40*d57664e9SAndroid Build Coastguard Worker kMips, 41*d57664e9SAndroid Build Coastguard Worker kMips64, 42*d57664e9SAndroid Build Coastguard Worker kUniversal 43*d57664e9SAndroid Build Coastguard Worker }; 44*d57664e9SAndroid Build Coastguard Worker 45*d57664e9SAndroid Build Coastguard Worker /** Helper method to convert an ABI to a string representing the path within the APK. */ 46*d57664e9SAndroid Build Coastguard Worker android::StringPiece AbiToString(Abi abi); 47*d57664e9SAndroid Build Coastguard Worker 48*d57664e9SAndroid Build Coastguard Worker /** 49*d57664e9SAndroid Build Coastguard Worker * Represents an individual locale. When a locale is included, it must be 50*d57664e9SAndroid Build Coastguard Worker * declared from least specific to most specific, as a region does not make 51*d57664e9SAndroid Build Coastguard Worker * sense without a language. If neither the language or region are specified it 52*d57664e9SAndroid Build Coastguard Worker * acts as a special case for catch all. This can allow all locales to be kept, 53*d57664e9SAndroid Build Coastguard Worker * or compressed. 54*d57664e9SAndroid Build Coastguard Worker */ 55*d57664e9SAndroid Build Coastguard Worker struct Locale { 56*d57664e9SAndroid Build Coastguard Worker /** The ISO<?> standard locale language code. */ 57*d57664e9SAndroid Build Coastguard Worker std::optional<std::string> lang; 58*d57664e9SAndroid Build Coastguard Worker /** The ISO<?> standard locale region code. */ 59*d57664e9SAndroid Build Coastguard Worker std::optional<std::string> region; 60*d57664e9SAndroid Build Coastguard Worker 61*d57664e9SAndroid Build Coastguard Worker inline friend bool operator==(const Locale& lhs, const Locale& rhs) { 62*d57664e9SAndroid Build Coastguard Worker return lhs.lang == rhs.lang && lhs.region == rhs.region; 63*d57664e9SAndroid Build Coastguard Worker } 64*d57664e9SAndroid Build Coastguard Worker }; 65*d57664e9SAndroid Build Coastguard Worker 66*d57664e9SAndroid Build Coastguard Worker // TODO: Encapsulate manifest modifications from the configuration file. 67*d57664e9SAndroid Build Coastguard Worker struct AndroidManifest { 68*d57664e9SAndroid Build Coastguard Worker inline friend bool operator==(const AndroidManifest& lhs, const AndroidManifest& rhs) { 69*d57664e9SAndroid Build Coastguard Worker return true; // nothing to compare yet. 70*d57664e9SAndroid Build Coastguard Worker } 71*d57664e9SAndroid Build Coastguard Worker }; 72*d57664e9SAndroid Build Coastguard Worker 73*d57664e9SAndroid Build Coastguard Worker struct AndroidSdk { 74*d57664e9SAndroid Build Coastguard Worker std::string label; 75*d57664e9SAndroid Build Coastguard Worker int min_sdk_version; // min_sdk_version is mandatory if splitting by SDK. 76*d57664e9SAndroid Build Coastguard Worker std::optional<int> target_sdk_version; 77*d57664e9SAndroid Build Coastguard Worker std::optional<int> max_sdk_version; 78*d57664e9SAndroid Build Coastguard Worker std::optional<AndroidManifest> manifest; 79*d57664e9SAndroid Build Coastguard Worker ForMinSdkAndroidSdk80*d57664e9SAndroid Build Coastguard Worker static AndroidSdk ForMinSdk(int min_sdk) { 81*d57664e9SAndroid Build Coastguard Worker AndroidSdk sdk; 82*d57664e9SAndroid Build Coastguard Worker sdk.min_sdk_version = min_sdk; 83*d57664e9SAndroid Build Coastguard Worker return sdk; 84*d57664e9SAndroid Build Coastguard Worker } 85*d57664e9SAndroid Build Coastguard Worker 86*d57664e9SAndroid Build Coastguard Worker inline friend bool operator==(const AndroidSdk& lhs, const AndroidSdk& rhs) { 87*d57664e9SAndroid Build Coastguard Worker return lhs.min_sdk_version == rhs.min_sdk_version && 88*d57664e9SAndroid Build Coastguard Worker lhs.target_sdk_version == rhs.target_sdk_version && 89*d57664e9SAndroid Build Coastguard Worker lhs.max_sdk_version == rhs.max_sdk_version && 90*d57664e9SAndroid Build Coastguard Worker lhs.manifest == rhs.manifest; 91*d57664e9SAndroid Build Coastguard Worker } 92*d57664e9SAndroid Build Coastguard Worker }; 93*d57664e9SAndroid Build Coastguard Worker 94*d57664e9SAndroid Build Coastguard Worker // TODO: Make device features more than just an arbitrary string? 95*d57664e9SAndroid Build Coastguard Worker using DeviceFeature = std::string; 96*d57664e9SAndroid Build Coastguard Worker 97*d57664e9SAndroid Build Coastguard Worker /** Represents a mapping of texture paths to a GL texture format. */ 98*d57664e9SAndroid Build Coastguard Worker struct GlTexture { 99*d57664e9SAndroid Build Coastguard Worker std::string name; 100*d57664e9SAndroid Build Coastguard Worker std::vector<std::string> texture_paths; 101*d57664e9SAndroid Build Coastguard Worker 102*d57664e9SAndroid Build Coastguard Worker inline friend bool operator==(const GlTexture& lhs, const GlTexture& rhs) { 103*d57664e9SAndroid Build Coastguard Worker return lhs.name == rhs.name && lhs.texture_paths == rhs.texture_paths; 104*d57664e9SAndroid Build Coastguard Worker } 105*d57664e9SAndroid Build Coastguard Worker }; 106*d57664e9SAndroid Build Coastguard Worker 107*d57664e9SAndroid Build Coastguard Worker /** An artifact with all the details pulled from the PostProcessingConfiguration. */ 108*d57664e9SAndroid Build Coastguard Worker struct OutputArtifact { 109*d57664e9SAndroid Build Coastguard Worker std::string name; 110*d57664e9SAndroid Build Coastguard Worker int version; 111*d57664e9SAndroid Build Coastguard Worker std::vector<Abi> abis; 112*d57664e9SAndroid Build Coastguard Worker std::vector<android::ConfigDescription> screen_densities; 113*d57664e9SAndroid Build Coastguard Worker std::vector<android::ConfigDescription> locales; 114*d57664e9SAndroid Build Coastguard Worker std::optional<AndroidSdk> android_sdk; 115*d57664e9SAndroid Build Coastguard Worker std::vector<DeviceFeature> features; 116*d57664e9SAndroid Build Coastguard Worker std::vector<GlTexture> textures; 117*d57664e9SAndroid Build Coastguard Worker 118*d57664e9SAndroid Build Coastguard Worker inline int GetMinSdk(int default_value = -1) const { 119*d57664e9SAndroid Build Coastguard Worker if (!android_sdk) { 120*d57664e9SAndroid Build Coastguard Worker return default_value; 121*d57664e9SAndroid Build Coastguard Worker } 122*d57664e9SAndroid Build Coastguard Worker return android_sdk.value().min_sdk_version; 123*d57664e9SAndroid Build Coastguard Worker } 124*d57664e9SAndroid Build Coastguard Worker }; 125*d57664e9SAndroid Build Coastguard Worker 126*d57664e9SAndroid Build Coastguard Worker } // namespace configuration 127*d57664e9SAndroid Build Coastguard Worker 128*d57664e9SAndroid Build Coastguard Worker /** 129*d57664e9SAndroid Build Coastguard Worker * XML configuration file parser for the split and optimize commands. 130*d57664e9SAndroid Build Coastguard Worker */ 131*d57664e9SAndroid Build Coastguard Worker class ConfigurationParser { 132*d57664e9SAndroid Build Coastguard Worker public: 133*d57664e9SAndroid Build Coastguard Worker 134*d57664e9SAndroid Build Coastguard Worker /** Returns a ConfigurationParser for the file located at the provided path. */ 135*d57664e9SAndroid Build Coastguard Worker static std::optional<ConfigurationParser> ForPath(const std::string& path); 136*d57664e9SAndroid Build Coastguard Worker 137*d57664e9SAndroid Build Coastguard Worker /** Returns a ConfigurationParser for the configuration in the provided file contents. */ ForContents(const std::string & contents,const std::string & path)138*d57664e9SAndroid Build Coastguard Worker static ConfigurationParser ForContents(const std::string& contents, const std::string& path) { 139*d57664e9SAndroid Build Coastguard Worker ConfigurationParser parser{contents, path}; 140*d57664e9SAndroid Build Coastguard Worker return parser; 141*d57664e9SAndroid Build Coastguard Worker } 142*d57664e9SAndroid Build Coastguard Worker 143*d57664e9SAndroid Build Coastguard Worker /** Sets the diagnostics context to use when parsing. */ WithDiagnostics(android::IDiagnostics * diagnostics)144*d57664e9SAndroid Build Coastguard Worker ConfigurationParser& WithDiagnostics(android::IDiagnostics* diagnostics) { 145*d57664e9SAndroid Build Coastguard Worker diag_ = diagnostics; 146*d57664e9SAndroid Build Coastguard Worker return *this; 147*d57664e9SAndroid Build Coastguard Worker } 148*d57664e9SAndroid Build Coastguard Worker 149*d57664e9SAndroid Build Coastguard Worker /** 150*d57664e9SAndroid Build Coastguard Worker * Parses the configuration file and returns the results. If the configuration could not be parsed 151*d57664e9SAndroid Build Coastguard Worker * the result is empty and any errors will be displayed with the provided diagnostics context. 152*d57664e9SAndroid Build Coastguard Worker */ 153*d57664e9SAndroid Build Coastguard Worker std::optional<std::vector<configuration::OutputArtifact>> Parse(android::StringPiece apk_path); 154*d57664e9SAndroid Build Coastguard Worker 155*d57664e9SAndroid Build Coastguard Worker protected: 156*d57664e9SAndroid Build Coastguard Worker /** 157*d57664e9SAndroid Build Coastguard Worker * Instantiates a new ConfigurationParser with the provided configuration file and a no-op 158*d57664e9SAndroid Build Coastguard Worker * diagnostics context. The default diagnostics context can be overridden with a call to 159*d57664e9SAndroid Build Coastguard Worker * WithDiagnostics(IDiagnostics *). 160*d57664e9SAndroid Build Coastguard Worker */ 161*d57664e9SAndroid Build Coastguard Worker ConfigurationParser(std::string contents, const std::string& config_path); 162*d57664e9SAndroid Build Coastguard Worker 163*d57664e9SAndroid Build Coastguard Worker /** Returns the current diagnostics context to any subclasses. */ diagnostics()164*d57664e9SAndroid Build Coastguard Worker android::IDiagnostics* diagnostics() { 165*d57664e9SAndroid Build Coastguard Worker return diag_; 166*d57664e9SAndroid Build Coastguard Worker } 167*d57664e9SAndroid Build Coastguard Worker 168*d57664e9SAndroid Build Coastguard Worker private: 169*d57664e9SAndroid Build Coastguard Worker /** The contents of the configuration file to parse. */ 170*d57664e9SAndroid Build Coastguard Worker const std::string contents_; 171*d57664e9SAndroid Build Coastguard Worker /** Path to the input configuration. */ 172*d57664e9SAndroid Build Coastguard Worker const std::string config_path_; 173*d57664e9SAndroid Build Coastguard Worker /** The diagnostics context to send messages to. */ 174*d57664e9SAndroid Build Coastguard Worker android::IDiagnostics* diag_; 175*d57664e9SAndroid Build Coastguard Worker }; 176*d57664e9SAndroid Build Coastguard Worker 177*d57664e9SAndroid Build Coastguard Worker } // namespace aapt 178*d57664e9SAndroid Build Coastguard Worker 179*d57664e9SAndroid Build Coastguard Worker #endif // AAPT2_CONFIGURATION_H 180