blob: 2f17853718ec747a81c85231aa622c470d8c92ba [file] [log] [blame]
/*
* Copyright (C) 2018 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.
*/
#ifndef AAPT2_LINK_H
#define AAPT2_LINK_H
#include <optional>
#include <regex>
#include <string>
#include <unordered_map>
#include <unordered_set>
#include <vector>
#include "Command.h"
#include "Resource.h"
#include "androidfw/IDiagnostics.h"
#include "cmd/Util.h"
#include "format/binary/TableFlattener.h"
#include "format/proto/ProtoSerialize.h"
#include "link/ManifestFixer.h"
#include "split/TableSplitter.h"
#include "trace/TraceBuffer.h"
namespace aapt {
enum class OutputFormat {
kApk,
kProto,
};
struct LinkOptions {
std::string output_path;
std::string manifest_path;
std::vector<std::string> include_paths;
std::vector<std::string> overlay_files;
std::vector<std::string> assets_dirs;
bool output_to_directory = false;
bool auto_add_overlay = false;
bool override_styles_instead_of_overlaying = false;
OutputFormat output_format = OutputFormat::kApk;
std::optional<std::string> rename_resources_package;
// Java/Proguard options.
std::optional<std::string> generate_java_class_path;
std::optional<std::string> custom_java_package;
std::set<std::string> extra_java_packages;
std::optional<std::string> generate_text_symbols_path;
std::optional<std::string> generate_proguard_rules_path;
std::optional<std::string> generate_main_dex_proguard_rules_path;
bool generate_conditional_proguard_rules = false;
bool generate_minimal_proguard_rules = false;
bool generate_non_final_ids = false;
bool no_proguard_location_reference = false;
std::vector<std::string> javadoc_annotations;
std::optional<std::string> private_symbols;
// Optimizations/features.
bool no_auto_version = false;
bool no_version_vectors = false;
bool no_version_transitions = false;
bool no_resource_deduping = false;
bool no_resource_removal = false;
bool no_xml_namespaces = false;
bool do_not_compress_anything = false;
bool use_sparse_encoding = false;
std::unordered_set<std::string> extensions_to_not_compress;
std::optional<std::regex> regex_to_not_compress;
FeatureFlagValues feature_flag_values;
// Static lib options.
bool no_static_lib_packages = false;
bool merge_only = false;
// AndroidManifest.xml massaging options.
ManifestFixerOptions manifest_fixer_options;
// Products to use/filter on.
std::unordered_set<std::string> products;
// Flattening options.
TableFlattenerOptions table_flattener_options;
SerializeTableOptions proto_table_flattener_options;
bool keep_raw_values = false;
// Split APK options.
TableSplitterOptions table_splitter_options;
std::vector<SplitConstraints> split_constraints;
std::vector<std::string> split_paths;
// Configurations to exclude
std::vector<std::string> exclude_configs_;
// Stable ID options.
std::unordered_map<ResourceName, ResourceId> stable_id_map;
std::optional<std::string> resource_id_map_path;
// When 'true', allow reserved package IDs to be used for applications. Pre-O, the platform
// treats negative resource IDs [those with a package ID of 0x80 or higher] as invalid.
// In order to work around this limitation, we allow the use of traditionally reserved
// resource IDs [those between 0x02 and 0x7E].
bool allow_reserved_package_id = false;
// Whether we should fail on definitions of a resource with conflicting visibility.
bool strict_visibility = false;
};
class LinkCommand : public Command {
public:
explicit LinkCommand(android::IDiagnostics* diag) : Command("link", "l"), diag_(diag) {
SetDescription("Links resources into an apk.");
AddRequiredFlag("-o", "Output path.", &options_.output_path, Command::kPath);
AddRequiredFlag("--manifest", "Path to the Android manifest to build.",
&options_.manifest_path, Command::kPath);
AddOptionalFlagList("-I", "Adds an Android APK to link against.", &options_.include_paths,
Command::kPath);
AddOptionalFlagList("-A", "An assets directory to include in the APK. These are unprocessed.",
&options_.assets_dirs, Command::kPath);
AddOptionalFlagList("-R", "Compilation unit to link, using `overlay` semantics.\n"
"The last conflicting resource given takes precedence.", &overlay_arg_list_,
Command::kPath);
AddOptionalFlag("--package-id",
"Specify the package ID to use for this app. Must be greater or equal to\n"
"0x7f and can't be used with --static-lib or --shared-lib.", &package_id_);
AddOptionalFlag("--java", "Directory in which to generate R.java.",
&options_.generate_java_class_path, Command::kPath);
AddOptionalFlag("--proguard", "Output file for generated Proguard rules.",
&options_.generate_proguard_rules_path, Command::kPath);
AddOptionalFlag("--proguard-main-dex",
"Output file for generated Proguard rules for the main dex.",
&options_.generate_main_dex_proguard_rules_path, Command::kPath);
AddOptionalSwitch("--proguard-conditional-keep-rules",
"Generate conditional Proguard keep rules.",
&options_.generate_conditional_proguard_rules);
AddOptionalSwitch("--proguard-minimal-keep-rules",
"Generate a minimal set of Proguard keep rules.",
&options_.generate_minimal_proguard_rules);
AddOptionalSwitch("--no-auto-version", "Disables automatic style and layout SDK versioning.",
&options_.no_auto_version);
AddOptionalSwitch("--no-version-vectors",
"Disables automatic versioning of vector drawables. Use this only\n"
"when building with vector drawable support library.",
&options_.no_version_vectors);
AddOptionalSwitch("--no-version-transitions",
"Disables automatic versioning of transition resources. Use this only\n"
"when building with transition support library.",
&options_.no_version_transitions);
AddOptionalSwitch("--no-resource-deduping", "Disables automatic deduping of resources with\n"
"identical values across compatible configurations.",
&options_.no_resource_deduping);
AddOptionalSwitch("--no-resource-removal", "Disables automatic removal of resources without\n"
"defaults. Use this only when building runtime resource overlay packages.",
&options_.no_resource_removal);
AddOptionalSwitch("--enable-sparse-encoding",
"This decreases APK size at the cost of resource retrieval performance.",
&options_.use_sparse_encoding);
AddOptionalSwitch("--enable-compact-entries",
"This decreases APK size by using compact resource entries for simple data types.",
&options_.table_flattener_options.use_compact_entries);
AddOptionalSwitch("-x", "Legacy flag that specifies to use the package identifier 0x01.",
&legacy_x_flag_);
AddOptionalSwitch("-z", "Require localization of strings marked 'suggested'.",
&require_localization_);
AddOptionalFlagList("-c",
"Comma separated list of configurations to include. The default\n"
"is all configurations.", &configs_);
AddOptionalFlag("--preferred-density",
"Selects the closest matching density and strips out all others.",
&preferred_density_);
AddOptionalFlag("--product", "Comma separated list of product names to keep", &product_list_);
AddOptionalSwitch("--output-to-dir", "Outputs the APK contents to a directory specified by -o.",
&options_.output_to_directory);
AddOptionalSwitch("--no-xml-namespaces", "Removes XML namespace prefix and URI information\n"
"from AndroidManifest.xml and XML binaries in res/*.",
&options_.no_xml_namespaces);
AddOptionalFlag("--min-sdk-version",
"Default minimum SDK version to use for AndroidManifest.xml.",
&options_.manifest_fixer_options.min_sdk_version_default);
AddOptionalFlag("--target-sdk-version",
"Default target SDK version to use for AndroidManifest.xml.",
&options_.manifest_fixer_options.target_sdk_version_default);
AddOptionalFlag("--version-code",
"Version code (integer) to inject into the AndroidManifest.xml if none is\n"
"present.", &options_.manifest_fixer_options.version_code_default);
AddOptionalFlag("--version-code-major",
"Version code major (integer) to inject into the AndroidManifest.xml if none is\n"
"present.", &options_.manifest_fixer_options.version_code_major_default);
AddOptionalFlag("--version-name",
"Version name to inject into the AndroidManifest.xml if none is present.",
&options_.manifest_fixer_options.version_name_default);
AddOptionalFlag("--revision-code",
"Revision code (integer) to inject into the AndroidManifest.xml if none is\n"
"present.", &options_.manifest_fixer_options.revision_code_default);
AddOptionalSwitch("--replace-version",
"If --version-code, --version-name, and/or --revision-code are specified, these\n"
"values will replace any value already in the manifest. By\n"
"default, nothing is changed if the manifest already defines\n"
"these attributes.",
&options_.manifest_fixer_options.replace_version);
AddOptionalFlag("--compile-sdk-version-code",
"Version code (integer) to inject into the AndroidManifest.xml if none is\n"
"present.",
&options_.manifest_fixer_options.compile_sdk_version);
AddOptionalFlag("--compile-sdk-version-name",
"Version name to inject into the AndroidManifest.xml if none is present.",
&options_.manifest_fixer_options.compile_sdk_version_codename);
AddOptionalSwitch(
"--no-compile-sdk-metadata",
"Suppresses output of compile SDK-related attributes in AndroidManifest.xml,\n"
"including android:compileSdkVersion and platformBuildVersion.",
&options_.manifest_fixer_options.no_compile_sdk_metadata);
AddOptionalFlagList("--fingerprint-prefix", "Fingerprint prefix to add to install constraints.",
&options_.manifest_fixer_options.fingerprint_prefixes);
AddOptionalSwitch("--shared-lib", "Generates a shared Android runtime library.",
&shared_lib_);
AddOptionalSwitch("--static-lib", "Generate a static Android library.", &static_lib_);
AddOptionalSwitch("--proto-format",
"Generates compiled resources in Protobuf format.\n"
"Suitable as input to the bundle tool for generating an App Bundle.",
&proto_format_);
AddOptionalSwitch("--no-static-lib-packages",
"Merge all library resources under the app's package.",
&options_.no_static_lib_packages);
AddOptionalSwitch("--non-final-ids",
"Generates R.java without the final modifier. This is implied when\n"
"--static-lib is specified.",
&options_.generate_non_final_ids);
AddOptionalSwitch("--no-proguard-location-reference",
"Keep proguard rules files from having a reference to the source file",
&options_.no_proguard_location_reference);
AddOptionalFlag("--stable-ids", "File containing a list of name to ID mapping.",
&stable_id_file_path_);
AddOptionalFlag("--emit-ids",
"Emit a file at the given path with a list of name to ID mappings,\n"
"suitable for use with --stable-ids.",
&options_.resource_id_map_path);
AddOptionalFlag("--private-symbols",
"Package name to use when generating R.java for private symbols.\n"
"If not specified, public and private symbols will use the application's\n"
"package name.",
&options_.private_symbols);
AddOptionalFlag("--custom-package", "Custom Java package under which to generate R.java.",
&options_.custom_java_package);
AddOptionalFlagList("--extra-packages",
"Generate the same R.java but with different package names.",
&extra_java_packages_);
AddOptionalFlagList("--add-javadoc-annotation",
"Adds a JavaDoc annotation to all generated Java classes.",
&options_.javadoc_annotations);
AddOptionalFlag("--output-text-symbols",
"Generates a text file containing the resource symbols of the R class in\n"
"the specified folder.",
&options_.generate_text_symbols_path);
AddOptionalSwitch("--allow-reserved-package-id",
"Allows the use of a reserved package ID. This should on be used for\n"
"packages with a pre-O min-sdk\n",
&options_.allow_reserved_package_id);
AddOptionalSwitch("--auto-add-overlay",
"Allows the addition of new resources in overlays without\n"
"<add-resource> tags.",
&options_.auto_add_overlay);
AddOptionalSwitch("--override-styles-instead-of-overlaying",
"Causes styles defined in -R resources to replace previous definitions\n"
"instead of merging into them\n",
&options_.override_styles_instead_of_overlaying);
AddOptionalFlag("--rename-manifest-package", "Renames the package in AndroidManifest.xml.",
&options_.manifest_fixer_options.rename_manifest_package);
AddOptionalFlag("--rename-resources-package", "Renames the package in resources table",
&options_.rename_resources_package);
AddOptionalFlag("--rename-instrumentation-target-package",
"Changes the name of the target package for instrumentation. Most useful\n"
"when used in conjunction with --rename-manifest-package.",
&options_.manifest_fixer_options.rename_instrumentation_target_package);
AddOptionalFlag("--rename-overlay-target-package",
"Changes the name of the target package for overlay. Most useful\n"
"when used in conjunction with --rename-manifest-package.",
&options_.manifest_fixer_options.rename_overlay_target_package);
AddOptionalFlag("--rename-overlay-category", "Changes the category for the overlay.",
&options_.manifest_fixer_options.rename_overlay_category);
AddOptionalFlagList("-0", "File suffix not to compress.",
&options_.extensions_to_not_compress);
AddOptionalSwitch("--no-compress", "Do not compress any resources.",
&options_.do_not_compress_anything);
AddOptionalSwitch("--keep-raw-values", "Preserve raw attribute values in xml files.",
&options_.keep_raw_values);
AddOptionalFlag("--no-compress-regex",
"Do not compress extensions matching the regular expression. Remember to\n"
"use the '$' symbol for end of line. Uses a case-sensitive ECMAScript"
"regular expression grammar.",
&no_compress_regex);
AddOptionalSwitch("--warn-manifest-validation",
"Treat manifest validation errors as warnings.",
&options_.manifest_fixer_options.warn_validation);
AddOptionalFlagList("--split",
"Split resources matching a set of configs out to a Split APK.\n"
"Syntax: path/to/output.apk:<config>[,<config>[...]].\n"
"On Windows, use a semicolon ';' separator instead.",
&split_args_);
AddOptionalFlagList("--exclude-configs",
"Excludes values of resources whose configs contain the specified qualifiers.",
&options_.exclude_configs_);
AddOptionalSwitch("--debug-mode",
"Inserts android:debuggable=\"true\" in to the application node of the\n"
"manifest, making the application debuggable even on production devices.",
&options_.manifest_fixer_options.debug_mode);
AddOptionalSwitch("--strict-visibility",
"Do not allow overlays with different visibility levels.",
&options_.strict_visibility);
AddOptionalSwitch("--exclude-sources",
"Do not serialize source file information when generating resources in\n"
"Protobuf format.",
&options_.proto_table_flattener_options.exclude_sources);
AddOptionalFlag("--trace-folder",
"Generate systrace json trace fragment to specified folder.",
&trace_folder_);
AddOptionalSwitch("--merge-only",
"Only merge the resources, without verifying resource references. This flag\n"
"should only be used together with the --static-lib flag.",
&options_.merge_only);
AddOptionalSwitch("-v", "Enables verbose logging.", &verbose_);
AddOptionalFlagList("--feature-flags",
"Specify the values of feature flags. The pairs in the argument\n"
"are separated by ',' the name is separated from the value by '='.\n"
"The name can have a suffix of ':ro' to indicate it is read only."
"Example: \"flag1=true,flag2:ro=false,flag3=\" (flag3 has no given value).",
&feature_flags_args_);
AddOptionalSwitch("--non-updatable-system",
"Mark the app as a non-updatable system app. This inserts\n"
"updatableSystem=\"false\" to the root manifest node, overwriting any\n"
"existing attribute. This is ignored if the manifest has a versionCode.",
&options_.manifest_fixer_options.non_updatable_system);
}
int Action(const std::vector<std::string>& args) override;
private:
android::IDiagnostics* diag_;
LinkOptions options_;
std::vector<std::string> overlay_arg_list_;
std::vector<std::string> extra_java_packages_;
std::optional<std::string> package_id_;
std::vector<std::string> configs_;
std::optional<std::string> preferred_density_;
std::optional<std::string> product_list_;
std::optional<std::string> no_compress_regex;
bool legacy_x_flag_ = false;
bool require_localization_ = false;
bool verbose_ = false;
bool shared_lib_ = false;
bool static_lib_ = false;
bool proto_format_ = false;
std::optional<std::string> stable_id_file_path_;
std::vector<std::string> split_args_;
std::optional<std::string> trace_folder_;
std::vector<std::string> feature_flags_args_;
};
}// namespace aapt
#endif //AAPT2_LINK_H