xref: /aosp_15_r20/frameworks/base/tools/aapt2/configuration/ConfigurationParser.h (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
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