xref: /aosp_15_r20/frameworks/base/tools/split-select/Main.cpp (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker  * Copyright (C) 2014 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 #include <algorithm>
18*d57664e9SAndroid Build Coastguard Worker #include <cstdio>
19*d57664e9SAndroid Build Coastguard Worker 
20*d57664e9SAndroid Build Coastguard Worker #include "aapt/AaptUtil.h"
21*d57664e9SAndroid Build Coastguard Worker 
22*d57664e9SAndroid Build Coastguard Worker #include "Grouper.h"
23*d57664e9SAndroid Build Coastguard Worker #include "Rule.h"
24*d57664e9SAndroid Build Coastguard Worker #include "RuleGenerator.h"
25*d57664e9SAndroid Build Coastguard Worker #include "SplitDescription.h"
26*d57664e9SAndroid Build Coastguard Worker #include "SplitSelector.h"
27*d57664e9SAndroid Build Coastguard Worker 
28*d57664e9SAndroid Build Coastguard Worker #include <androidfw/AssetManager.h>
29*d57664e9SAndroid Build Coastguard Worker #include <androidfw/ResourceTypes.h>
30*d57664e9SAndroid Build Coastguard Worker #include <utils/KeyedVector.h>
31*d57664e9SAndroid Build Coastguard Worker #include <utils/Vector.h>
32*d57664e9SAndroid Build Coastguard Worker 
33*d57664e9SAndroid Build Coastguard Worker using namespace android;
34*d57664e9SAndroid Build Coastguard Worker 
35*d57664e9SAndroid Build Coastguard Worker namespace split {
36*d57664e9SAndroid Build Coastguard Worker 
usage()37*d57664e9SAndroid Build Coastguard Worker static void usage() {
38*d57664e9SAndroid Build Coastguard Worker     fprintf(stderr,
39*d57664e9SAndroid Build Coastguard Worker             "split-select --help\n"
40*d57664e9SAndroid Build Coastguard Worker             "split-select --target <config> --base <path/to/apk> [--split <path/to/apk> [...]]\n"
41*d57664e9SAndroid Build Coastguard Worker             "split-select --generate --base <path/to/apk> [--split <path/to/apk> [...]]\n"
42*d57664e9SAndroid Build Coastguard Worker             "\n"
43*d57664e9SAndroid Build Coastguard Worker             "  --help                   Displays more information about this program.\n"
44*d57664e9SAndroid Build Coastguard Worker             "  --target <config>        Performs the Split APK selection on the given configuration.\n"
45*d57664e9SAndroid Build Coastguard Worker             "  --generate               Generates the logic for selecting the Split APK, in JSON format.\n"
46*d57664e9SAndroid Build Coastguard Worker             "  --base <path/to/apk>     Specifies the base APK, from which all Split APKs must be based off.\n"
47*d57664e9SAndroid Build Coastguard Worker             "  --split <path/to/apk>    Includes a Split APK in the selection process.\n"
48*d57664e9SAndroid Build Coastguard Worker             "\n"
49*d57664e9SAndroid Build Coastguard Worker             "  Where <config> is an extended AAPT resource qualifier of the form\n"
50*d57664e9SAndroid Build Coastguard Worker             "  'resource-qualifiers:extended-qualifiers', where 'resource-qualifiers' is an AAPT resource\n"
51*d57664e9SAndroid Build Coastguard Worker             "  qualifier (ex: en-rUS-sw600dp-xhdpi), and 'extended-qualifiers' is an ordered list of one\n"
52*d57664e9SAndroid Build Coastguard Worker             "  qualifier (or none) from each category:\n"
53*d57664e9SAndroid Build Coastguard Worker             "    Architecture: armeabi, armeabi-v7a, arm64-v8a, x86, x86_64, mips\n");
54*d57664e9SAndroid Build Coastguard Worker }
55*d57664e9SAndroid Build Coastguard Worker 
help()56*d57664e9SAndroid Build Coastguard Worker static void help() {
57*d57664e9SAndroid Build Coastguard Worker     usage();
58*d57664e9SAndroid Build Coastguard Worker     fprintf(stderr, "\n"
59*d57664e9SAndroid Build Coastguard Worker             "  Generates the logic for selecting a Split APK given some target Android device configuration.\n"
60*d57664e9SAndroid Build Coastguard Worker             "  Using the flag --generate will emit a JSON encoded tree of rules that must be satisfied in order\n"
61*d57664e9SAndroid Build Coastguard Worker             "  to install the given Split APK. Using the flag --target along with the device configuration\n"
62*d57664e9SAndroid Build Coastguard Worker             "  will emit the set of Split APKs to install, following the same logic that would have been emitted\n"
63*d57664e9SAndroid Build Coastguard Worker             "  via JSON.\n");
64*d57664e9SAndroid Build Coastguard Worker }
65*d57664e9SAndroid Build Coastguard Worker 
select(const SplitDescription & target,const Vector<SplitDescription> & splits)66*d57664e9SAndroid Build Coastguard Worker Vector<SplitDescription> select(const SplitDescription& target, const Vector<SplitDescription>& splits) {
67*d57664e9SAndroid Build Coastguard Worker     const SplitSelector selector(splits);
68*d57664e9SAndroid Build Coastguard Worker     return selector.getBestSplits(target);
69*d57664e9SAndroid Build Coastguard Worker }
70*d57664e9SAndroid Build Coastguard Worker 
generate(const KeyedVector<String8,Vector<SplitDescription>> & splits,const String8 & base)71*d57664e9SAndroid Build Coastguard Worker void generate(const KeyedVector<String8, Vector<SplitDescription> >& splits, const String8& base) {
72*d57664e9SAndroid Build Coastguard Worker     Vector<SplitDescription> allSplits;
73*d57664e9SAndroid Build Coastguard Worker     const size_t apkSplitCount = splits.size();
74*d57664e9SAndroid Build Coastguard Worker     for (size_t i = 0; i < apkSplitCount; i++) {
75*d57664e9SAndroid Build Coastguard Worker         allSplits.appendVector(splits[i]);
76*d57664e9SAndroid Build Coastguard Worker     }
77*d57664e9SAndroid Build Coastguard Worker     const SplitSelector selector(allSplits);
78*d57664e9SAndroid Build Coastguard Worker     KeyedVector<SplitDescription, sp<Rule> > rules(selector.getRules());
79*d57664e9SAndroid Build Coastguard Worker 
80*d57664e9SAndroid Build Coastguard Worker     bool first = true;
81*d57664e9SAndroid Build Coastguard Worker     fprintf(stdout, "[\n");
82*d57664e9SAndroid Build Coastguard Worker     for (size_t i = 0; i < apkSplitCount; i++) {
83*d57664e9SAndroid Build Coastguard Worker         if (splits.keyAt(i) == base) {
84*d57664e9SAndroid Build Coastguard Worker             // Skip the base.
85*d57664e9SAndroid Build Coastguard Worker             continue;
86*d57664e9SAndroid Build Coastguard Worker         }
87*d57664e9SAndroid Build Coastguard Worker 
88*d57664e9SAndroid Build Coastguard Worker         if (!first) {
89*d57664e9SAndroid Build Coastguard Worker             fprintf(stdout, ",\n");
90*d57664e9SAndroid Build Coastguard Worker         }
91*d57664e9SAndroid Build Coastguard Worker         first = false;
92*d57664e9SAndroid Build Coastguard Worker 
93*d57664e9SAndroid Build Coastguard Worker         sp<Rule> masterRule = new Rule();
94*d57664e9SAndroid Build Coastguard Worker         masterRule->op = Rule::OR_SUBRULES;
95*d57664e9SAndroid Build Coastguard Worker         const Vector<SplitDescription>& splitDescriptions = splits[i];
96*d57664e9SAndroid Build Coastguard Worker         const size_t splitDescriptionCount = splitDescriptions.size();
97*d57664e9SAndroid Build Coastguard Worker         for (size_t j = 0; j < splitDescriptionCount; j++) {
98*d57664e9SAndroid Build Coastguard Worker             masterRule->subrules.add(rules.valueFor(splitDescriptions[j]));
99*d57664e9SAndroid Build Coastguard Worker         }
100*d57664e9SAndroid Build Coastguard Worker         masterRule = Rule::simplify(masterRule);
101*d57664e9SAndroid Build Coastguard Worker         fprintf(stdout, "  {\n    \"path\": \"%s\",\n    \"rules\": %s\n  }",
102*d57664e9SAndroid Build Coastguard Worker                 splits.keyAt(i).c_str(), masterRule->toJson(2).c_str());
103*d57664e9SAndroid Build Coastguard Worker     }
104*d57664e9SAndroid Build Coastguard Worker     fprintf(stdout, "\n]\n");
105*d57664e9SAndroid Build Coastguard Worker }
106*d57664e9SAndroid Build Coastguard Worker 
removeRuntimeQualifiers(ConfigDescription * outConfig)107*d57664e9SAndroid Build Coastguard Worker static void removeRuntimeQualifiers(ConfigDescription* outConfig) {
108*d57664e9SAndroid Build Coastguard Worker     outConfig->imsi = 0;
109*d57664e9SAndroid Build Coastguard Worker     outConfig->orientation = ResTable_config::ORIENTATION_ANY;
110*d57664e9SAndroid Build Coastguard Worker     outConfig->screenWidth = ResTable_config::SCREENWIDTH_ANY;
111*d57664e9SAndroid Build Coastguard Worker     outConfig->screenHeight = ResTable_config::SCREENHEIGHT_ANY;
112*d57664e9SAndroid Build Coastguard Worker     outConfig->uiMode &= ResTable_config::UI_MODE_NIGHT_ANY;
113*d57664e9SAndroid Build Coastguard Worker }
114*d57664e9SAndroid Build Coastguard Worker 
115*d57664e9SAndroid Build Coastguard Worker struct AppInfo {
116*d57664e9SAndroid Build Coastguard Worker     int versionCode;
117*d57664e9SAndroid Build Coastguard Worker     int minSdkVersion;
118*d57664e9SAndroid Build Coastguard Worker     bool multiArch;
119*d57664e9SAndroid Build Coastguard Worker };
120*d57664e9SAndroid Build Coastguard Worker 
getAppInfo(const String8 & path,AppInfo & outInfo)121*d57664e9SAndroid Build Coastguard Worker static bool getAppInfo(const String8& path, AppInfo& outInfo) {
122*d57664e9SAndroid Build Coastguard Worker     memset(&outInfo, 0, sizeof(outInfo));
123*d57664e9SAndroid Build Coastguard Worker 
124*d57664e9SAndroid Build Coastguard Worker     AssetManager assetManager;
125*d57664e9SAndroid Build Coastguard Worker     int32_t cookie = 0;
126*d57664e9SAndroid Build Coastguard Worker     if (!assetManager.addAssetPath(path, &cookie)) {
127*d57664e9SAndroid Build Coastguard Worker         return false;
128*d57664e9SAndroid Build Coastguard Worker     }
129*d57664e9SAndroid Build Coastguard Worker 
130*d57664e9SAndroid Build Coastguard Worker     Asset* asset = assetManager.openNonAsset(cookie, "AndroidManifest.xml", Asset::ACCESS_BUFFER);
131*d57664e9SAndroid Build Coastguard Worker     if (asset == NULL) {
132*d57664e9SAndroid Build Coastguard Worker         return false;
133*d57664e9SAndroid Build Coastguard Worker     }
134*d57664e9SAndroid Build Coastguard Worker 
135*d57664e9SAndroid Build Coastguard Worker     ResXMLTree xml;
136*d57664e9SAndroid Build Coastguard Worker     if (xml.setTo(asset->getBuffer(true), asset->getLength(), false) != NO_ERROR) {
137*d57664e9SAndroid Build Coastguard Worker         delete asset;
138*d57664e9SAndroid Build Coastguard Worker         return false;
139*d57664e9SAndroid Build Coastguard Worker     }
140*d57664e9SAndroid Build Coastguard Worker 
141*d57664e9SAndroid Build Coastguard Worker     const String16 kAndroidNamespace("http://schemas.android.com/apk/res/android");
142*d57664e9SAndroid Build Coastguard Worker     const String16 kManifestTag("manifest");
143*d57664e9SAndroid Build Coastguard Worker     const String16 kApplicationTag("application");
144*d57664e9SAndroid Build Coastguard Worker     const String16 kUsesSdkTag("uses-sdk");
145*d57664e9SAndroid Build Coastguard Worker     const String16 kVersionCodeAttr("versionCode");
146*d57664e9SAndroid Build Coastguard Worker     const String16 kMultiArchAttr("multiArch");
147*d57664e9SAndroid Build Coastguard Worker     const String16 kMinSdkVersionAttr("minSdkVersion");
148*d57664e9SAndroid Build Coastguard Worker 
149*d57664e9SAndroid Build Coastguard Worker     ResXMLParser::event_code_t event;
150*d57664e9SAndroid Build Coastguard Worker     while ((event = xml.next()) != ResXMLParser::BAD_DOCUMENT &&
151*d57664e9SAndroid Build Coastguard Worker             event != ResXMLParser::END_DOCUMENT) {
152*d57664e9SAndroid Build Coastguard Worker         if (event != ResXMLParser::START_TAG) {
153*d57664e9SAndroid Build Coastguard Worker             continue;
154*d57664e9SAndroid Build Coastguard Worker         }
155*d57664e9SAndroid Build Coastguard Worker 
156*d57664e9SAndroid Build Coastguard Worker         size_t len;
157*d57664e9SAndroid Build Coastguard Worker         const char16_t* name = xml.getElementName(&len);
158*d57664e9SAndroid Build Coastguard Worker         String16 name16(name, len);
159*d57664e9SAndroid Build Coastguard Worker         if (name16 == kManifestTag) {
160*d57664e9SAndroid Build Coastguard Worker             ssize_t idx = xml.indexOfAttribute(kAndroidNamespace.c_str(), kAndroidNamespace.size(),
161*d57664e9SAndroid Build Coastguard Worker                                                kVersionCodeAttr.c_str(), kVersionCodeAttr.size());
162*d57664e9SAndroid Build Coastguard Worker             if (idx >= 0) {
163*d57664e9SAndroid Build Coastguard Worker                 outInfo.versionCode = xml.getAttributeData(idx);
164*d57664e9SAndroid Build Coastguard Worker             }
165*d57664e9SAndroid Build Coastguard Worker 
166*d57664e9SAndroid Build Coastguard Worker         } else if (name16 == kApplicationTag) {
167*d57664e9SAndroid Build Coastguard Worker             ssize_t idx = xml.indexOfAttribute(kAndroidNamespace.c_str(), kAndroidNamespace.size(),
168*d57664e9SAndroid Build Coastguard Worker                                                kMultiArchAttr.c_str(), kMultiArchAttr.size());
169*d57664e9SAndroid Build Coastguard Worker             if (idx >= 0) {
170*d57664e9SAndroid Build Coastguard Worker                 outInfo.multiArch = xml.getAttributeData(idx) != 0;
171*d57664e9SAndroid Build Coastguard Worker             }
172*d57664e9SAndroid Build Coastguard Worker 
173*d57664e9SAndroid Build Coastguard Worker         } else if (name16 == kUsesSdkTag) {
174*d57664e9SAndroid Build Coastguard Worker             ssize_t idx =
175*d57664e9SAndroid Build Coastguard Worker                     xml.indexOfAttribute(kAndroidNamespace.c_str(), kAndroidNamespace.size(),
176*d57664e9SAndroid Build Coastguard Worker                                          kMinSdkVersionAttr.c_str(), kMinSdkVersionAttr.size());
177*d57664e9SAndroid Build Coastguard Worker             if (idx >= 0) {
178*d57664e9SAndroid Build Coastguard Worker                 uint16_t type = xml.getAttributeDataType(idx);
179*d57664e9SAndroid Build Coastguard Worker                 if (type >= Res_value::TYPE_FIRST_INT && type <= Res_value::TYPE_LAST_INT) {
180*d57664e9SAndroid Build Coastguard Worker                     outInfo.minSdkVersion = xml.getAttributeData(idx);
181*d57664e9SAndroid Build Coastguard Worker                 } else if (type == Res_value::TYPE_STRING) {
182*d57664e9SAndroid Build Coastguard Worker                     auto minSdk8 = xml.getStrings().string8ObjectAt(idx);
183*d57664e9SAndroid Build Coastguard Worker                     if (!minSdk8.has_value()) {
184*d57664e9SAndroid Build Coastguard Worker                         fprintf(stderr, "warning: failed to retrieve android:minSdkVersion.\n");
185*d57664e9SAndroid Build Coastguard Worker                     } else {
186*d57664e9SAndroid Build Coastguard Worker                         char *endPtr;
187*d57664e9SAndroid Build Coastguard Worker                         int minSdk = strtol(minSdk8->c_str(), &endPtr, 10);
188*d57664e9SAndroid Build Coastguard Worker                         if (endPtr != minSdk8->c_str() + minSdk8->size()) {
189*d57664e9SAndroid Build Coastguard Worker                             fprintf(stderr, "warning: failed to parse android:minSdkVersion '%s'\n",
190*d57664e9SAndroid Build Coastguard Worker                                     minSdk8->c_str());
191*d57664e9SAndroid Build Coastguard Worker                         } else {
192*d57664e9SAndroid Build Coastguard Worker                             outInfo.minSdkVersion = minSdk;
193*d57664e9SAndroid Build Coastguard Worker                         }
194*d57664e9SAndroid Build Coastguard Worker                     }
195*d57664e9SAndroid Build Coastguard Worker                 } else {
196*d57664e9SAndroid Build Coastguard Worker                     fprintf(stderr, "warning: unrecognized value for android:minSdkVersion.\n");
197*d57664e9SAndroid Build Coastguard Worker                 }
198*d57664e9SAndroid Build Coastguard Worker             }
199*d57664e9SAndroid Build Coastguard Worker         }
200*d57664e9SAndroid Build Coastguard Worker     }
201*d57664e9SAndroid Build Coastguard Worker 
202*d57664e9SAndroid Build Coastguard Worker     delete asset;
203*d57664e9SAndroid Build Coastguard Worker     return true;
204*d57664e9SAndroid Build Coastguard Worker }
205*d57664e9SAndroid Build Coastguard Worker 
extractSplitDescriptionsFromApk(const String8 & path)206*d57664e9SAndroid Build Coastguard Worker static Vector<SplitDescription> extractSplitDescriptionsFromApk(const String8& path) {
207*d57664e9SAndroid Build Coastguard Worker     AssetManager assetManager;
208*d57664e9SAndroid Build Coastguard Worker     Vector<SplitDescription> splits;
209*d57664e9SAndroid Build Coastguard Worker     int32_t cookie = 0;
210*d57664e9SAndroid Build Coastguard Worker     if (!assetManager.addAssetPath(path, &cookie)) {
211*d57664e9SAndroid Build Coastguard Worker         return splits;
212*d57664e9SAndroid Build Coastguard Worker     }
213*d57664e9SAndroid Build Coastguard Worker 
214*d57664e9SAndroid Build Coastguard Worker     const ResTable& res = assetManager.getResources(false);
215*d57664e9SAndroid Build Coastguard Worker     if (res.getError() == NO_ERROR) {
216*d57664e9SAndroid Build Coastguard Worker         Vector<ResTable_config> configs;
217*d57664e9SAndroid Build Coastguard Worker         res.getConfigurations(&configs, true);
218*d57664e9SAndroid Build Coastguard Worker         const size_t configCount = configs.size();
219*d57664e9SAndroid Build Coastguard Worker         for (size_t i = 0; i < configCount; i++) {
220*d57664e9SAndroid Build Coastguard Worker             splits.add();
221*d57664e9SAndroid Build Coastguard Worker             splits.editTop().config = configs[i];
222*d57664e9SAndroid Build Coastguard Worker         }
223*d57664e9SAndroid Build Coastguard Worker     }
224*d57664e9SAndroid Build Coastguard Worker 
225*d57664e9SAndroid Build Coastguard Worker     AssetDir* dir = assetManager.openNonAssetDir(cookie, "lib");
226*d57664e9SAndroid Build Coastguard Worker     if (dir != NULL) {
227*d57664e9SAndroid Build Coastguard Worker         const size_t fileCount = dir->getFileCount();
228*d57664e9SAndroid Build Coastguard Worker         for (size_t i = 0; i < fileCount; i++) {
229*d57664e9SAndroid Build Coastguard Worker             splits.add();
230*d57664e9SAndroid Build Coastguard Worker             Vector<String8> parts = AaptUtil::splitAndLowerCase(dir->getFileName(i), '-');
231*d57664e9SAndroid Build Coastguard Worker             if (parseAbi(parts, 0, &splits.editTop()) < 0) {
232*d57664e9SAndroid Build Coastguard Worker                 fprintf(stderr, "Malformed library %s\n", dir->getFileName(i).c_str());
233*d57664e9SAndroid Build Coastguard Worker                 splits.pop();
234*d57664e9SAndroid Build Coastguard Worker             }
235*d57664e9SAndroid Build Coastguard Worker         }
236*d57664e9SAndroid Build Coastguard Worker         delete dir;
237*d57664e9SAndroid Build Coastguard Worker     }
238*d57664e9SAndroid Build Coastguard Worker     return splits;
239*d57664e9SAndroid Build Coastguard Worker }
240*d57664e9SAndroid Build Coastguard Worker 
main(int argc,char ** argv)241*d57664e9SAndroid Build Coastguard Worker static int main(int argc, char** argv) {
242*d57664e9SAndroid Build Coastguard Worker     // Skip over the first argument.
243*d57664e9SAndroid Build Coastguard Worker     argc--;
244*d57664e9SAndroid Build Coastguard Worker     argv++;
245*d57664e9SAndroid Build Coastguard Worker 
246*d57664e9SAndroid Build Coastguard Worker     bool generateFlag = false;
247*d57664e9SAndroid Build Coastguard Worker     String8 targetConfigStr;
248*d57664e9SAndroid Build Coastguard Worker     Vector<String8> splitApkPaths;
249*d57664e9SAndroid Build Coastguard Worker     String8 baseApkPath;
250*d57664e9SAndroid Build Coastguard Worker     while (argc > 0) {
251*d57664e9SAndroid Build Coastguard Worker         const String8 arg(*argv);
252*d57664e9SAndroid Build Coastguard Worker         if (arg == "--target") {
253*d57664e9SAndroid Build Coastguard Worker             argc--;
254*d57664e9SAndroid Build Coastguard Worker             argv++;
255*d57664e9SAndroid Build Coastguard Worker             if (argc < 1) {
256*d57664e9SAndroid Build Coastguard Worker                 fprintf(stderr, "error: missing parameter for --target.\n");
257*d57664e9SAndroid Build Coastguard Worker                 usage();
258*d57664e9SAndroid Build Coastguard Worker                 return 1;
259*d57664e9SAndroid Build Coastguard Worker             }
260*d57664e9SAndroid Build Coastguard Worker             targetConfigStr = *argv;
261*d57664e9SAndroid Build Coastguard Worker         } else if (arg == "--split") {
262*d57664e9SAndroid Build Coastguard Worker             argc--;
263*d57664e9SAndroid Build Coastguard Worker             argv++;
264*d57664e9SAndroid Build Coastguard Worker             if (argc < 1) {
265*d57664e9SAndroid Build Coastguard Worker                 fprintf(stderr, "error: missing parameter for --split.\n");
266*d57664e9SAndroid Build Coastguard Worker                 usage();
267*d57664e9SAndroid Build Coastguard Worker                 return 1;
268*d57664e9SAndroid Build Coastguard Worker             }
269*d57664e9SAndroid Build Coastguard Worker             splitApkPaths.add(String8(*argv));
270*d57664e9SAndroid Build Coastguard Worker         } else if (arg == "--base") {
271*d57664e9SAndroid Build Coastguard Worker             argc--;
272*d57664e9SAndroid Build Coastguard Worker             argv++;
273*d57664e9SAndroid Build Coastguard Worker             if (argc < 1) {
274*d57664e9SAndroid Build Coastguard Worker                 fprintf(stderr, "error: missing parameter for --base.\n");
275*d57664e9SAndroid Build Coastguard Worker                 usage();
276*d57664e9SAndroid Build Coastguard Worker                 return 1;
277*d57664e9SAndroid Build Coastguard Worker             }
278*d57664e9SAndroid Build Coastguard Worker 
279*d57664e9SAndroid Build Coastguard Worker             if (baseApkPath.size() > 0) {
280*d57664e9SAndroid Build Coastguard Worker                 fprintf(stderr, "error: multiple --base flags not allowed.\n");
281*d57664e9SAndroid Build Coastguard Worker                 usage();
282*d57664e9SAndroid Build Coastguard Worker                 return 1;
283*d57664e9SAndroid Build Coastguard Worker             }
284*d57664e9SAndroid Build Coastguard Worker             baseApkPath = *argv;
285*d57664e9SAndroid Build Coastguard Worker         } else if (arg == "--generate") {
286*d57664e9SAndroid Build Coastguard Worker             generateFlag = true;
287*d57664e9SAndroid Build Coastguard Worker         } else if (arg == "--help") {
288*d57664e9SAndroid Build Coastguard Worker             help();
289*d57664e9SAndroid Build Coastguard Worker             return 0;
290*d57664e9SAndroid Build Coastguard Worker         } else {
291*d57664e9SAndroid Build Coastguard Worker             fprintf(stderr, "error: unknown argument '%s'.\n", arg.c_str());
292*d57664e9SAndroid Build Coastguard Worker             usage();
293*d57664e9SAndroid Build Coastguard Worker             return 1;
294*d57664e9SAndroid Build Coastguard Worker         }
295*d57664e9SAndroid Build Coastguard Worker         argc--;
296*d57664e9SAndroid Build Coastguard Worker         argv++;
297*d57664e9SAndroid Build Coastguard Worker     }
298*d57664e9SAndroid Build Coastguard Worker 
299*d57664e9SAndroid Build Coastguard Worker     if (!generateFlag && targetConfigStr == "") {
300*d57664e9SAndroid Build Coastguard Worker         usage();
301*d57664e9SAndroid Build Coastguard Worker         return 1;
302*d57664e9SAndroid Build Coastguard Worker     }
303*d57664e9SAndroid Build Coastguard Worker 
304*d57664e9SAndroid Build Coastguard Worker     if (baseApkPath.size() == 0) {
305*d57664e9SAndroid Build Coastguard Worker         fprintf(stderr, "error: missing --base argument.\n");
306*d57664e9SAndroid Build Coastguard Worker         usage();
307*d57664e9SAndroid Build Coastguard Worker         return 1;
308*d57664e9SAndroid Build Coastguard Worker     }
309*d57664e9SAndroid Build Coastguard Worker 
310*d57664e9SAndroid Build Coastguard Worker     // Find out some details about the base APK.
311*d57664e9SAndroid Build Coastguard Worker     AppInfo baseAppInfo;
312*d57664e9SAndroid Build Coastguard Worker     if (!getAppInfo(baseApkPath, baseAppInfo)) {
313*d57664e9SAndroid Build Coastguard Worker         fprintf(stderr, "error: unable to read base APK: '%s'.\n", baseApkPath.c_str());
314*d57664e9SAndroid Build Coastguard Worker         return 1;
315*d57664e9SAndroid Build Coastguard Worker     }
316*d57664e9SAndroid Build Coastguard Worker 
317*d57664e9SAndroid Build Coastguard Worker     SplitDescription targetSplit;
318*d57664e9SAndroid Build Coastguard Worker     if (!generateFlag) {
319*d57664e9SAndroid Build Coastguard Worker         if (!SplitDescription::parse(targetConfigStr, &targetSplit)) {
320*d57664e9SAndroid Build Coastguard Worker             fprintf(stderr, "error: invalid --target config: '%s'.\n", targetConfigStr.c_str());
321*d57664e9SAndroid Build Coastguard Worker             usage();
322*d57664e9SAndroid Build Coastguard Worker             return 1;
323*d57664e9SAndroid Build Coastguard Worker         }
324*d57664e9SAndroid Build Coastguard Worker 
325*d57664e9SAndroid Build Coastguard Worker         // We don't want to match on things that will change at run-time
326*d57664e9SAndroid Build Coastguard Worker         // (orientation, w/h, etc.).
327*d57664e9SAndroid Build Coastguard Worker         removeRuntimeQualifiers(&targetSplit.config);
328*d57664e9SAndroid Build Coastguard Worker     }
329*d57664e9SAndroid Build Coastguard Worker 
330*d57664e9SAndroid Build Coastguard Worker     splitApkPaths.add(baseApkPath);
331*d57664e9SAndroid Build Coastguard Worker 
332*d57664e9SAndroid Build Coastguard Worker     KeyedVector<String8, Vector<SplitDescription> > apkPathSplitMap;
333*d57664e9SAndroid Build Coastguard Worker     KeyedVector<SplitDescription, String8> splitApkPathMap;
334*d57664e9SAndroid Build Coastguard Worker     Vector<SplitDescription> splitConfigs;
335*d57664e9SAndroid Build Coastguard Worker     const size_t splitCount = splitApkPaths.size();
336*d57664e9SAndroid Build Coastguard Worker     for (size_t i = 0; i < splitCount; i++) {
337*d57664e9SAndroid Build Coastguard Worker         Vector<SplitDescription> splits = extractSplitDescriptionsFromApk(splitApkPaths[i]);
338*d57664e9SAndroid Build Coastguard Worker         if (splits.isEmpty()) {
339*d57664e9SAndroid Build Coastguard Worker             fprintf(stderr, "error: invalid --split path: '%s'. No splits found.\n",
340*d57664e9SAndroid Build Coastguard Worker                     splitApkPaths[i].c_str());
341*d57664e9SAndroid Build Coastguard Worker             usage();
342*d57664e9SAndroid Build Coastguard Worker             return 1;
343*d57664e9SAndroid Build Coastguard Worker         }
344*d57664e9SAndroid Build Coastguard Worker         apkPathSplitMap.replaceValueFor(splitApkPaths[i], splits);
345*d57664e9SAndroid Build Coastguard Worker         const size_t apkSplitDescriptionCount = splits.size();
346*d57664e9SAndroid Build Coastguard Worker         for (size_t j = 0; j < apkSplitDescriptionCount; j++) {
347*d57664e9SAndroid Build Coastguard Worker             splitApkPathMap.replaceValueFor(splits[j], splitApkPaths[i]);
348*d57664e9SAndroid Build Coastguard Worker         }
349*d57664e9SAndroid Build Coastguard Worker         splitConfigs.appendVector(splits);
350*d57664e9SAndroid Build Coastguard Worker     }
351*d57664e9SAndroid Build Coastguard Worker 
352*d57664e9SAndroid Build Coastguard Worker     if (!generateFlag) {
353*d57664e9SAndroid Build Coastguard Worker         Vector<SplitDescription> matchingConfigs = select(targetSplit, splitConfigs);
354*d57664e9SAndroid Build Coastguard Worker         const size_t matchingConfigCount = matchingConfigs.size();
355*d57664e9SAndroid Build Coastguard Worker         SortedVector<String8> matchingSplitPaths;
356*d57664e9SAndroid Build Coastguard Worker         for (size_t i = 0; i < matchingConfigCount; i++) {
357*d57664e9SAndroid Build Coastguard Worker             matchingSplitPaths.add(splitApkPathMap.valueFor(matchingConfigs[i]));
358*d57664e9SAndroid Build Coastguard Worker         }
359*d57664e9SAndroid Build Coastguard Worker 
360*d57664e9SAndroid Build Coastguard Worker         const size_t matchingSplitApkPathCount = matchingSplitPaths.size();
361*d57664e9SAndroid Build Coastguard Worker         for (size_t i = 0; i < matchingSplitApkPathCount; i++) {
362*d57664e9SAndroid Build Coastguard Worker             if (matchingSplitPaths[i] != baseApkPath) {
363*d57664e9SAndroid Build Coastguard Worker                 fprintf(stdout, "%s\n", matchingSplitPaths[i].c_str());
364*d57664e9SAndroid Build Coastguard Worker             }
365*d57664e9SAndroid Build Coastguard Worker         }
366*d57664e9SAndroid Build Coastguard Worker     } else {
367*d57664e9SAndroid Build Coastguard Worker         generate(apkPathSplitMap, baseApkPath);
368*d57664e9SAndroid Build Coastguard Worker     }
369*d57664e9SAndroid Build Coastguard Worker     return 0;
370*d57664e9SAndroid Build Coastguard Worker }
371*d57664e9SAndroid Build Coastguard Worker 
372*d57664e9SAndroid Build Coastguard Worker } // namespace split
373*d57664e9SAndroid Build Coastguard Worker 
main(int argc,char ** argv)374*d57664e9SAndroid Build Coastguard Worker int main(int argc, char** argv) {
375*d57664e9SAndroid Build Coastguard Worker     return split::main(argc, argv);
376*d57664e9SAndroid Build Coastguard Worker }
377