1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "dex2oat_options.h"
18 
19 #include <memory>
20 
21 #include "cmdline_parser.h"
22 #include "driver/compiler_options_map-inl.h"
23 
24 namespace art {
25 
26 template<>
27 struct CmdlineType<InstructionSet> : CmdlineTypeParser<InstructionSet> {
Parseart::CmdlineType28   Result Parse(const std::string& option) {
29     InstructionSet set = GetInstructionSetFromString(option.c_str());
30     if (set == InstructionSet::kNone) {
31       return Result::Failure(std::string("Not a valid instruction set: '") + option + "'");
32     }
33     return Result::Success(set);
34   }
35 
Nameart::CmdlineType36   static const char* Name() { return "InstructionSet"; }
DescribeTypeart::CmdlineType37   static const char* DescribeType() { return "arm|arm64|riscv64|x86|x86_64|none"; }
38 };
39 
40 #define COMPILER_OPTIONS_MAP_TYPE Dex2oatArgumentMap
41 #define COMPILER_OPTIONS_MAP_KEY_TYPE Dex2oatArgumentMapKey
42 #include "driver/compiler_options_map-storage.h"
43 
44 // Specify storage for the Dex2oatOptions keys.
45 
46 #define DEX2OAT_OPTIONS_KEY(Type, Name, ...) \
47   const Dex2oatArgumentMap::Key<Type> Dex2oatArgumentMap::Name {__VA_ARGS__};
48 #include "dex2oat_options.def"
49 
50 #pragma GCC diagnostic push
51 #pragma GCC diagnostic ignored "-Wframe-larger-than="
52 
53 using M = Dex2oatArgumentMap;
54 using Parser = CmdlineParser<Dex2oatArgumentMap, Dex2oatArgumentMap::Key>;
55 using Builder = Parser::Builder;
56 
AddInputMappings(Builder & builder)57 static void AddInputMappings(Builder& builder) {
58   // clang-format off
59   builder.
60       Define("--dex-file=_")
61           .WithType<std::vector<std::string>>().AppendValues()
62           .WithHelp("Specifies a .dex, .jar, or .apk file to compile.\n"
63                     "Eg: --dex-file=/system/framework/core.jar")
64           .WithMetavar("<dex-file>")
65           .IntoKey(M::DexFiles)
66       .Define("--dex-location=_")
67           .WithType<std::vector<std::string>>().AppendValues()
68           .WithMetavar("<dex-location>")
69           .WithHelp("specifies an alternative dex location to encode in the oat file for the\n"
70                     "corresponding --dex-file argument. The first --dex-location corresponds to\n"
71                     "the first --dex-file, the second to the second and so on.\n"
72                     "Eg: --dex-file=/home/build/out/system/framework/core.jar\n"
73                     "    --dex-location=/system/framework/core.jar")
74           .IntoKey(M::DexLocations)
75       .Define("--dex-fd=_")
76           .WithType<std::vector<int>>().AppendValues()
77           .WithHelp("Specifies a file descriptor of a dex file. It can be specified for multiple\n"
78                     "times, but the number must match the number of --dex-file. Eg: --dex-fd=5")
79           .IntoKey(M::DexFds)
80       .Define("--zip-fd=_")
81           .WithType<int>()
82           .WithHelp("specifies a file descriptor of a zip file containing a classes.dex file to\n"
83                     "compile. Eg: --zip-fd=5")
84           .IntoKey(M::ZipFd)
85       .Define("--zip-location=_")
86           .WithType<std::string>()
87           .WithHelp("Specifies a symbolic name for the file corresponding to the FD given by\n"
88                     "--zip-fd.")
89           .IntoKey(M::ZipLocation)
90       .Define("--boot-image=_")
91           .WithType<std::string>()
92           .WithHelp("provide the image file for the boot class path.\n"
93                     "Do not include the arch as part of the name, it is added automatically.\n"
94                     "Example: --boot-image=/system/framework/boot.art\n"
95                     "         (specifies /system/framework/<arch>/boot.art as the image file)\n"
96                     "Example: --boot-image=boot.art:boot-framework.art\n"
97                     "         (specifies <bcp-path1>/<arch>/boot.art as the image file and\n"
98                     "         <bcp-path2>/<arch>/boot-framework.art as the image extension file\n"
99                     "         with paths taken from corresponding boot class path components)\n"
100                     "Example: --boot-image=/apex/com.android.art/boot.art:/system/framework/*:*\n"
101                     "         (specifies /apex/com.android.art/<arch>/boot.art as the image\n"
102                     "         file and search for extensions in /framework/system and boot\n"
103                     "         class path components' paths)\n"
104                     "Default: $ANDROID_ROOT/system/framework/boot.art")
105           .IntoKey(M::BootImage);
106   // clang-format on
107 }
108 
AddGeneratedArtifactMappings(Builder & builder)109 static void AddGeneratedArtifactMappings(Builder& builder) {
110   // clang-format off
111   builder.
112       Define("--input-vdex-fd=_")
113           .WithType<int>()
114           .WithHelp("specifies the vdex input source via a file descriptor.")
115           .IntoKey(M::InputVdexFd)
116       .Define("--input-vdex=_")
117           .WithType<std::string>()
118           .WithHelp("specifies the vdex input source via a filename.")
119           .IntoKey(M::InputVdex)
120       .Define("--output-vdex-fd=_")
121           .WithHelp("specifies the vdex output destination via a file descriptor.")
122           .WithType<int>()
123           .IntoKey(M::OutputVdexFd)
124       .Define("--output-vdex=_")
125           .WithType<std::string>()
126           .WithHelp("specifies the vdex output destination via a filename.")
127           .IntoKey(M::OutputVdex)
128       .Define("--dm-fd=_")
129           .WithType<int>()
130           .WithHelp("specifies the dm output destination via a file descriptor.")
131           .IntoKey(M::DmFd)
132       .Define("--dm-file=_")
133           .WithType<std::string>()
134           .WithHelp("specifies the dm output destination via a filename.")
135           .IntoKey(M::DmFile)
136       .Define("--oat-file=_")
137           .WithType<std::string>()
138           .WithHelp(" Specifies an oat output destination via a filename.\n"
139                     "Eg: --oat-file=/system/framework/boot.oat")
140           .IntoKey(M::OatFile)
141       .Define("--oat-symbols=_")
142           .WithType<std::string>()
143           .WithHelp("Specifies a symbolized oat output destination.\n"
144                     "Eg: --oat-symbols=symbols/system/framework/boot.oat")
145           .IntoKey(M::OatSymbols)
146       .Define("--strip")
147           .WithHelp("remove all debugging sections at the end (but keep mini-debug-info).\n"
148                     "This is equivalent to the \"strip\" command as build post-processing step.\n"
149                     "It is intended to be used with --oat-symbols and it happens after it.\n"
150                     "Eg: --oat-symbols=/symbols/system/framework/boot.oat")
151           .IntoKey(M::Strip)
152       .Define("--oat-fd=_")
153           .WithType<int>()
154           .WithHelp("Specifies the oat output destination via a file descriptor. Eg: --oat-fd=5")
155           .IntoKey(M::OatFd)
156       .Define("--oat-location=_")
157           .WithType<std::string>()
158           .WithHelp("specifies a symbolic name for the file corresponding to the file descriptor\n"
159                     "specified by --oat-fd.\n"
160                     "Eg: --oat-location=/data/dalvik-cache/system@app@Calculator.apk.oat")
161           .IntoKey(M::OatLocation);
162   // clang-format on
163 }
164 
AddImageMappings(Builder & builder)165 static void AddImageMappings(Builder& builder) {
166   // clang-format off
167   builder.
168       Define("--image=_")
169           .WithType<std::string>()
170           .WithHelp("specifies an output image filename. Eg: --image=/system/framework/boot.art")
171           .IntoKey(M::ImageFilename)
172       .Define("--image-fd=_")
173           .WithType<int>()
174           .WithHelp("specifies an output image file descriptor. Cannot be used with --image.\n"
175                     "Eg: --image-fd=7")
176           .IntoKey(M::ImageFd)
177       .Define("--base=_")
178           .WithType<std::string>()
179           .WithHelp("Specifies the base address when creating a boot image. Eg: --base=0x50000000")
180           .WithMetavar("{hex address}")
181           .IntoKey(M::Base)
182       .Define("--app-image-file=_")
183           .WithType<std::string>()
184           .WithHelp("Specify a file name for app image. Only used if a profile is passed in.")
185           .IntoKey(M::AppImageFile)
186       .Define("--app-image-fd=_")
187           .WithType<int>()
188           .WithHelp("Specify a file descriptor for app image. Only used if a profile is passed in.")
189           .IntoKey(M::AppImageFileFd)
190       .Define({"--multi-image", "--single-image"})
191           .WithValues({true, false})
192           .WithHelp("Specifies if separate oat and image files should be generated for each dex\n"
193                     "file. --multi-image is default for boot image and --single-image for app\n"
194                     "images.")
195           .IntoKey(M::MultiImage)
196       .Define("--dirty-image-objects=_")
197           .WithType<std::string>()
198           .WithHelp("list of known dirty objects in the image. The image writer will group them"
199                     " together")
200           .IntoKey(M::DirtyImageObjects)
201       .Define("--dirty-image-objects-fd=_")
202           .WithType<int>()
203           .WithHelp("Specify a file descriptor for reading the list of known dirty objects in\n"
204                     "the image. The image writer will group them together")
205           .IntoKey(M::DirtyImageObjectsFd)
206       .Define("--updatable-bcp-packages-file=_")
207           .WithType<std::string>()
208           .WithHelp("Deprecated. No longer takes effect.")
209           .IntoKey(M::UpdatableBcpPackagesFile)
210       .Define("--updatable-bcp-packages-fd=_")
211           .WithType<int>()
212           .WithHelp("Deprecated. No longer takes effect.")
213           .IntoKey(M::UpdatableBcpPackagesFd)
214       .Define("--image-format=_")
215           .WithType<ImageHeader::StorageMode>()
216           .WithValueMap({{"lz4", ImageHeader::kStorageModeLZ4},
217                          {"lz4hc", ImageHeader::kStorageModeLZ4HC},
218                          {"uncompressed", ImageHeader::kStorageModeUncompressed}})
219           .WithHelp("Which format to store the image Defaults to uncompressed. Eg:"
220                     " --image-format=lz4")
221           .IntoKey(M::ImageFormat);
222   // clang-format on
223 }
224 
AddSwapMappings(Builder & builder)225 static void AddSwapMappings(Builder& builder) {
226   // clang-format off
227   builder.
228       Define("--swap-file=_")
229           .WithType<std::string>()
230           .WithHelp("Specify a file to use for swap. Eg: --swap-file=/data/tmp/swap.001")
231           .IntoKey(M::SwapFile)
232       .Define("--swap-fd=_")
233           .WithType<int>()
234           .WithHelp("Specify a file to use for swap by file-descriptor. Eg: --swap-fd=3")
235           .IntoKey(M::SwapFileFd)
236       .Define("--swap-dex-size-threshold=_")
237           .WithType<unsigned int>()
238           .WithHelp("specifies the minimum total dex file size in bytes to allow the use of swap.")
239           .IntoKey(M::SwapDexSizeThreshold)
240       .Define("--swap-dex-count-threshold=_")
241           .WithType<unsigned int>()
242           .WithHelp("specifies the minimum number of dex file to allow the use of swap.")
243           .IntoKey(M::SwapDexCountThreshold);
244   // clang-format on
245 }
246 
AddCompilerMappings(Builder & builder)247 static void AddCompilerMappings(Builder& builder) {
248   // clang-format off
249   builder.
250       Define("--run-passes=_")
251           .WithType<std::string>()
252           .IntoKey(M::Passes)
253       .Define("--profile-file=_")
254           .WithType<std::vector<std::string>>().AppendValues()
255           .WithHelp("Specify profiler output file to use for compilation using a filename.\n"
256                     "When multiple profiles are used, the order matters: If multiple profiles \n"
257                     "contain classes and methods of the same dex file with different checksums, \n"
258                     "only the classes and methods from the first profile will be used for that \n"
259                     "particular dex file.")
260           .IntoKey(M::Profile)
261       .Define("--profile-file-fd=_")
262           .WithType<std::vector<int>>().AppendValues()
263           .WithHelp("Specify profiler output file to use for compilation using a file-descriptor.")
264           .IntoKey(M::ProfileFd)
265       .Define("--no-inline-from=_")
266           .WithType<std::string>()
267           .IntoKey(M::NoInlineFrom)
268       .Define("--preloaded-classes=_")
269           .WithType<std::vector<std::string>>().AppendValues()
270           .WithHelp("Specify files containing list of classes preloaded in the zygote.")
271           .IntoKey(M::PreloadedClasses)
272       .Define("--preloaded-classes-fds=_")
273           .WithType<std::vector<int>>().AppendValues()
274           .WithHelp("Specify files containing list of classes preloaded in the zygote.")
275           .IntoKey(M::PreloadedClassesFds);
276   // clang-format on
277 }
278 
AddTargetMappings(Builder & builder)279 static void AddTargetMappings(Builder& builder) {
280   // clang-format off
281   builder.
282       Define("--instruction-set=_")
283           .WithType<InstructionSet>()
284           .WithHelp("Compile for a particular instruction set.")
285           .IntoKey(M::TargetInstructionSet)
286       .Define("--instruction-set-variant=_")
287           .WithType<std::string>()
288           .WithHelp("Specify instruction set features using variant name.\n"
289                     "Eg: --instruction-set-variant=silvermont")
290           .WithMetavar("{Variant Name}")
291           .IntoKey(M::TargetInstructionSetVariant)
292       .Define("--instruction-set-features=_")
293           .WithType<std::string>()
294           .WithHelp("Specify instruction set features.\n"
295                     "On target the value 'runtime' can be used to detect features at run time.\n"
296                     "If target does not support run-time detection the value 'runtime'\n"
297                     "has the same effect as the value 'default'.\n"
298                     "Note: the value 'runtime' has no effect if it is used on host.\n"
299                     "Example: --instruction-set-features=div\n"
300                     "Default: default")
301           .IntoKey(M::TargetInstructionSetFeatures);
302   // clang-format on
303 }
304 
CreateDex2oatArgumentParser()305 Parser CreateDex2oatArgumentParser() {
306   std::unique_ptr<Builder> parser_builder = std::make_unique<Builder>();
307 
308   AddInputMappings(*parser_builder);
309   AddGeneratedArtifactMappings(*parser_builder);
310   AddImageMappings(*parser_builder);
311   AddSwapMappings(*parser_builder);
312   AddCompilerMappings(*parser_builder);
313   AddTargetMappings(*parser_builder);
314 
315   // clang-format off
316   parser_builder->
317       Define({"--watch-dog", "--no-watch-dog"})
318           .WithHelp("Enable or disable the watchdog timer.")
319           .WithValues({true, false})
320           .IntoKey(M::Watchdog)
321       .Define("--watchdog-timeout=_")
322           .WithType<int>()
323           .WithHelp("Set the watchdog timeout value in milliseconds.")
324           .IntoKey(M::WatchdogTimeout)
325       .Define("-j_")
326           .WithType<unsigned int>()
327           .WithHelp("specifies the number of threads used for compilation. Default is the number\n"
328                     "of detected hardware threads available on the host system.")
329           .IntoKey(M::Threads)
330       .Define("--cpu-set=_")
331           .WithType<ParseIntList<','>>()
332           .WithHelp("sets the cpu affinitiy to the given <set>. The <set> is a comma separated\n"
333                     "list of cpus. Eg: --cpu-set=0,1,2,3")
334           .WithMetavar("<set>")
335           .IntoKey(M::CpuSet)
336       .Define("--android-root=_")
337           .WithType<std::string>()
338           .WithHelp("Used to locate libraries for portable linking.\n"
339                     "Eg: --android-root=out/host/linux-x86\n"
340                     "Default: $ANDROID_ROOT")
341           .IntoKey(M::AndroidRoot)
342       .Define("--host")
343           .WithHelp("Run in host mode")
344           .IntoKey(M::Host)
345       .Define("--avoid-storing-invocation")
346           .WithHelp("Avoid storing the invocation args in the key-value store. Used to test\n"
347                     "determinism with different args.")
348           .IntoKey(M::AvoidStoringInvocation)
349       .Define("--very-large-app-threshold=_")
350           .WithType<unsigned int>()
351           .WithHelp("Specifies the minimum total dex file size in bytes to consider the input\n"
352                     "\"very large\" and reduce compilation done.")
353           .IntoKey(M::VeryLargeAppThreshold)
354       .Define("--force-determinism")
355           .WithHelp("Force the compiler to emit a deterministic output")
356           .IntoKey(M::ForceDeterminism)
357       .Define("--check-linkage-conditions")
358           .IntoKey(M::CheckLinkageConditions)
359       .Define("--crash-on-linkage-violation")
360           .IntoKey(M::CrashOnLinkageViolation)
361       .Define("--copy-dex-files=_")
362           .WithType<linker::CopyOption>()
363           .WithValueMap({{"true", linker::CopyOption::kOnlyIfCompressed},
364                          {"false", linker::CopyOption::kNever},
365                          {"always", linker::CopyOption::kAlways}})
366           .WithHelp("enable|disable copying the dex files into the output vdex.")
367           .IntoKey(M::CopyDexFiles)
368       .Define("--force-allow-oj-inlines")
369           .WithHelp("Disables automatic no-inline for core-oj on host. Has no effect on target."
370                     " FOR TESTING USE ONLY! DO NOT DISTRIBUTE BINARIES BUILT WITH THIS OPTION!")
371           .IntoKey(M::ForceAllowOjInlines)
372       .Define("--write-invocation-to=_")
373           .WithHelp("Write the invocation commandline to the given file for later use. Used to\n"
374                     "test determinism with different args.")
375           .WithType<std::string>()
376           .IntoKey(M::InvocationFile)
377       .Define("--classpath-dir=_")
378           .WithType<std::string>()
379           .WithHelp("Directory used to resolve relative class paths.")
380           .IntoKey(M::ClasspathDir)
381       .Define("--class-loader-context=_")
382           .WithType<std::string>()
383           .WithHelp("a string specifying the intended runtime loading context for the compiled\n"
384                     "dex files.")
385           .IntoKey(M::ClassLoaderContext)
386       .Define("--class-loader-context-fds=_")
387           .WithType<std::string>()
388           .WithHelp("a colon-separated list of file descriptors for dex files in\n"
389                     "--class-loader-context. Their order must be the same as dex files in a\n"
390                     "flattened class loader context")
391           .IntoKey(M::ClassLoaderContextFds)
392       .Define("--stored-class-loader-context=_")
393           .WithType<std::string>()
394           .WithHelp("a string specifying the intended runtime loading context that is stored\n"
395                     "in the oat file. Overrides --class-loader-context. Note that this ignores\n"
396                     "the classpath_dir arg.\n"
397                     "\n"
398                     "It describes how the class loader chain should be built in order to ensure\n"
399                     "classes are resolved during dex2aot as they would be resolved at runtime.\n"
400                     "This spec will be encoded in the oat file. If at runtime the dex file is\n"
401                     "loaded in a different context, the oat file will be rejected.\n"
402                     "\n"
403                     "The chain is interpreted in the natural 'parent order', meaning that class\n"
404                     "loader 'i+1' will be the parent of class loader 'i'.\n"
405                     "The compilation sources will be appended to the classpath of the first class\n"
406                     "loader.\n"
407                     "\n"
408                     "E.g. if the context is 'PCL[lib1.dex];DLC[lib2.dex]' and \n"
409                     "--dex-file=src.dex then dex2oat will setup a PathClassLoader with classpath \n"
410                     "'lib1.dex:src.dex' and set its parent to a DelegateLastClassLoader with \n"
411                     "classpath 'lib2.dex'.\n"
412                     "\n"
413                     "Note that the compiler will be tolerant if the source dex files specified\n"
414                     "with --dex-file are found in the classpath. The source dex files will be\n"
415                     "removed from any class loader's classpath possibly resulting in empty\n"
416                     "class loaders.\n"
417                     "\n"
418                     "Example: --class-loader-context=PCL[lib1.dex:lib2.dex];DLC[lib3.dex]")
419           .IntoKey(M::StoredClassLoaderContext)
420       // TODO(b/325430813): Obsolete argument that only prints a warning if used. Delete altogether.
421       .Define("--compact-dex-level=_")
422           .WithType<std::string>()
423           .IntoKey(M::CompactDexLevel)
424       .Define("--runtime-arg _")
425           .WithType<std::vector<std::string>>().AppendValues()
426           .WithMetavar("{dalvikvm-arg}")
427           .WithHelp("used to specify various arguments for the runtime, such as initial heap\n"
428                     "size, maximum heap size, and verbose output. Use a separate --runtime-arg\n"
429                     "switch for each argument.\n"
430                     "Example: --runtime-arg -Xms256m")
431           .IntoKey(M::RuntimeOptions)
432       .Define("--compilation-reason=_")
433           .WithType<std::string>()
434           .WithHelp("optional metadata specifying the reason for compiling the apk. If specified,\n"
435                     "the string will be embedded verbatim in the key value store of the oat file.\n"
436                     "Example: --compilation-reason=install")
437           .IntoKey(M::CompilationReason)
438       .Define("--compile-individually")
439           .WithHelp("Compiles dex files individually, unloading classes in between compiling each"
440                     " file.")
441           .IntoKey(M::CompileIndividually)
442       .Define("--public-sdk=_")
443           .WithType<std::string>()
444           .IntoKey(M::PublicSdk)
445       .Define("--apex-versions=_")
446           .WithType<std::string>()
447           .WithHelp("Versions of apexes in the boot classpath, separated by '/'")
448           .IntoKey(M::ApexVersions)
449       .Define("--force-jit-zygote")
450           .WithHelp("Optimizes the app to be executed in an environment that uses JIT Zygote.")
451           .IntoKey(M::ForceJitZygote)
452       .Define("--force-palette-compilation-hooks")
453           .WithHelp("Force PaletteNotify{Start,End}Dex2oatCompilation calls.")
454           .IntoKey(M::ForcePaletteCompilationHooks)
455       .Ignore({
456         "--comments=_",
457         "--cache-info-fd=_",  // Handled in mark_compact.cc.
458         "--compiler-backend",
459       });
460   // clang-format on
461 
462   AddCompilerOptionsArgumentParserOptions<Dex2oatArgumentMap>(*parser_builder);
463 
464   parser_builder->IgnoreUnrecognized(false);
465 
466   return parser_builder->Build();
467 }
468 
Parse(int argc,const char ** argv,std::string * error_msg)469 std::unique_ptr<Dex2oatArgumentMap> Dex2oatArgumentMap::Parse(int argc,
470                                                               const char** argv,
471                                                               std::string* error_msg) {
472   Parser parser = CreateDex2oatArgumentParser();
473   CmdlineResult parse_result = parser.Parse(argv, argc);
474   if (!parse_result.IsSuccess()) {
475     *error_msg = parse_result.GetMessage();
476     return nullptr;
477   }
478 
479   return std::make_unique<Dex2oatArgumentMap>(parser.ReleaseArgumentsMap());
480 }
481 
482 #pragma GCC diagnostic pop
483 }  // namespace art
484