# Copyright 2014 The Chromium Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import("//base/android/linker/config.gni") import("//build/config/android/config.gni") import("//build/config/android/internal_rules.gni") import("//build/config/sanitizers/sanitizers.gni") import("//build/toolchain/toolchain.gni") import("//third_party/android_platform/config.gni") import("//tools/grit/grit_rule.gni") assert(is_android) # Declare a jni target # # This target generates the native jni bindings for a set of .java files. # # See base/android/jni_generator/jni_generator.py for more info about the # format of generating JNI bindings. # # Variables # sources: list of .java files to generate jni for # jni_package: subdirectory path for generated bindings # # Example # generate_jni("foo_jni") { # sources = [ # "android/java/src/org/chromium/foo/Foo.java", # "android/java/src/org/chromium/foo/FooUtil.java", # ] # jni_package = "foo" # } template("generate_jni") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.sources)) assert(defined(invoker.jni_package)) jni_package = invoker.jni_package base_output_dir = "${target_gen_dir}/${target_name}" package_output_dir = "${base_output_dir}/${jni_package}" jni_output_dir = "${package_output_dir}/jni" jni_generator_include = "//base/android/jni_generator/jni_generator_helper.h" foreach_target_name = "${target_name}__jni_gen" action_foreach(foreach_target_name) { script = "//base/android/jni_generator/jni_generator.py" depfile = "$target_gen_dir/$target_name.{{source_name_part}}.d" sources = invoker.sources outputs = [ depfile, "${jni_output_dir}/{{source_name_part}}_jni.h", ] args = [ "--depfile", rebase_path(depfile, root_build_dir), "--input_file={{source}}", "--optimize_generation=1", "--ptr_type=long", "--output_dir", rebase_path(jni_output_dir, root_build_dir), "--includes", rebase_path(jni_generator_include, jni_output_dir), "--native_exports_optional", ] } config("jni_includes_${target_name}") { # TODO(cjhopman): #includes should probably all be relative to # base_output_dir. Remove that from this config once the includes are # updated. include_dirs = [ base_output_dir, package_output_dir, ] } group(target_name) { forward_variables_from(invoker, [ "deps", "public_deps", "visibility", ]) if (!defined(public_deps)) { public_deps = [] } public_deps += [ ":$foreach_target_name" ] public_configs = [ ":jni_includes_${target_name}" ] } } # Declare a jni target for a prebuilt jar # # This target generates the native jni bindings for a set of classes in a .jar. # # See base/android/jni_generator/jni_generator.py for more info about the # format of generating JNI bindings. # # Variables # classes: list of .class files in the jar to generate jni for. These should # include the full path to the .class file. # jni_package: subdirectory path for generated bindings # jar_file: the path to the .jar. If not provided, will default to the sdk's # android.jar # # deps, public_deps: As normal # # Example # generate_jar_jni("foo_jni") { # classes = [ # "android/view/Foo.class", # ] # jni_package = "foo" # } template("generate_jar_jni") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.classes)) assert(defined(invoker.jni_package)) if (defined(invoker.jar_file)) { jar_file = invoker.jar_file } else { jar_file = android_sdk_jar } jni_package = invoker.jni_package base_output_dir = "${root_gen_dir}/${target_name}/${jni_package}" jni_output_dir = "${base_output_dir}/jni" jni_generator_include = "//base/android/jni_generator/jni_generator_helper.h" # TODO(cjhopman): make jni_generator.py support generating jni for multiple # .class files from a .jar. jni_actions = [] foreach(class, invoker.classes) { _classname_list = [] _classname_list = process_file_template([ class ], "{{source_name_part}}") classname = _classname_list[0] jni_target_name = "${target_name}__jni_${classname}" jni_actions += [ ":$jni_target_name" ] action(jni_target_name) { # The sources aren't compiled so don't check their dependencies. check_includes = false depfile = "$target_gen_dir/$target_name.d" script = "//base/android/jni_generator/jni_generator.py" sources = [ jar_file, ] outputs = [ depfile, "${jni_output_dir}/${classname}_jni.h", ] args = [ "--depfile", rebase_path(depfile, root_build_dir), "--jar_file", rebase_path(jar_file, root_build_dir), "--input_file", class, "--optimize_generation=1", "--ptr_type=long", "--output_dir", rebase_path(jni_output_dir, root_build_dir), "--includes", rebase_path(jni_generator_include, jni_output_dir), "--native_exports_optional", ] } } config("jni_includes_${target_name}") { include_dirs = [ base_output_dir ] } group(target_name) { public_deps = [] forward_variables_from(invoker, [ "deps", "public_deps", "visibility", ]) public_deps += jni_actions public_configs = [ ":jni_includes_${target_name}" ] } } # Declare a target for c-preprocessor-generated java files # # NOTE: For generating Java conterparts to enums prefer using the java_cpp_enum # rule instead. # # This target generates java files using the host C pre-processor. Each file in # sources will be compiled using the C pre-processor. If include_path is # specified, it will be passed (with --I) to the pre-processor. # # This target will create a single .srcjar. Adding this target to an # android_library target's srcjar_deps will make the generated java files be # included in that library's final outputs. # # Variables # sources: list of files to be processed by the C pre-processor. For each # file in sources, there will be one .java file in the final .srcjar. For a # file named FooBar.template, a java file will be created with name # FooBar.java. # inputs: additional compile-time dependencies. Any files # `#include`-ed in the templates should be listed here. # package_name: this will be the subdirectory for each .java file in the # .srcjar. # # Example # java_cpp_template("foo_generated_enum") { # sources = [ # "android/java/templates/Foo.template", # ] # inputs = [ # "android/java/templates/native_foo_header.h", # ] # # package_name = "org/chromium/base/library_loader" # include_path = "android/java/templates" # } template("java_cpp_template") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.sources)) package_name = invoker.package_name + "" if (defined(invoker.include_path)) { include_path = invoker.include_path + "" } else { include_path = "//" } apply_gcc_target_name = "${target_name}__apply_gcc" zip_srcjar_target_name = "${target_name}__zip_srcjar" final_target_name = target_name action_foreach(apply_gcc_target_name) { forward_variables_from(invoker, [ "deps", "public_deps", "data_deps", ]) visibility = [ ":$zip_srcjar_target_name" ] script = "//build/android/gyp/gcc_preprocess.py" if (defined(invoker.inputs)) { inputs = invoker.inputs + [] } depfile = "${target_gen_dir}/${target_name}_{{source_name_part}}.d" sources = invoker.sources gen_dir = "${target_gen_dir}/${target_name}/java_cpp_template/${package_name}" gcc_template_output_pattern = "${gen_dir}/{{source_name_part}}.java" outputs = [ depfile, gcc_template_output_pattern, ] args = [ "--depfile", rebase_path(depfile, root_build_dir), "--include-path", rebase_path(include_path, root_build_dir), "--output", rebase_path(gen_dir, root_build_dir) + "/{{source_name_part}}.java", "--template={{source}}", ] if (defined(invoker.defines)) { foreach(def, invoker.defines) { args += [ "--defines", def, ] } } } apply_gcc_outputs = get_target_outputs(":$apply_gcc_target_name") base_gen_dir = get_label_info(":$apply_gcc_target_name", "target_gen_dir") srcjar_path = "${target_gen_dir}/${target_name}.srcjar" zip(zip_srcjar_target_name) { visibility = [ ":$final_target_name" ] inputs = apply_gcc_outputs output = srcjar_path base_dir = base_gen_dir deps = [ ":$apply_gcc_target_name", ] } group(final_target_name) { forward_variables_from(invoker, [ "visibility" ]) public_deps = [ ":$zip_srcjar_target_name", ] } } # Declare a target for generating Java classes from C++ enums. # # This target generates Java files from C++ enums using a script. # # This target will create a single .srcjar. Adding this target to an # android_library target's srcjar_deps will make the generated java files be # included in that library's final outputs. # # Variables # sources: list of files to be processed by the script. For each annotated # enum contained in the sources files the script will generate a .java # file with the same name as the name of the enum. # # Example # java_cpp_enum("foo_generated_enum") { # sources = [ # "src/native_foo_header.h", # ] # } template("java_cpp_enum") { action(target_name) { # The sources aren't compiled so don't check their dependencies. check_includes = false set_sources_assignment_filter([]) assert(defined(invoker.sources)) forward_variables_from(invoker, [ "sources", "testonly", "visibility", ]) script = "//build/android/gyp/java_cpp_enum.py" depfile = "$target_gen_dir/$target_name.d" _srcjar_path = "${target_gen_dir}/${target_name}.srcjar" _rebased_srcjar_path = rebase_path(_srcjar_path, root_build_dir) _rebased_sources = rebase_path(invoker.sources, root_build_dir) args = [ "--depfile", rebase_path(depfile, root_build_dir), "--srcjar=$_rebased_srcjar_path", ] + _rebased_sources outputs = [ depfile, _srcjar_path, ] } } # Declare a target for processing a Jinja template. # # Variables # input: The template file to be processed. # output: Where to save the result. # variables: (Optional) A list of variables to make available to the template # processing environment, e.g. ["name=foo", "color=red"]. # # Example # jinja_template("chrome_public_manifest") { # input = "java/AndroidManifest.xml" # output = "$target_gen_dir/AndroidManifest.xml" # } template("jinja_template") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.input)) assert(defined(invoker.output)) action(target_name) { forward_variables_from(invoker, [ "visibility", "deps", ]) sources = [ invoker.input, ] script = "//build/android/gyp/jinja_template.py" depfile = "$target_gen_dir/$target_name.d" outputs = [ depfile, invoker.output, ] args = [ "--inputs", rebase_path(invoker.input, root_build_dir), "--output", rebase_path(invoker.output, root_build_dir), "--depfile", rebase_path(depfile, root_build_dir), ] if (defined(invoker.variables)) { variables = invoker.variables args += [ "--variables=${variables}" ] } } } # Declare a target for processing Android resources as Jinja templates. # # This takes an Android resource directory where each resource is a Jinja # template, processes each template, then packages the results in a zip file # which can be consumed by an android resources, library, or apk target. # # If this target is included in the deps of an android resources/library/apk, # the resources will be included with that target. # # Variables # resources: The list of resources files to process. # res_dir: The resource directory containing the resources. # variables: (Optional) A list of variables to make available to the template # processing environment, e.g. ["name=foo", "color=red"]. # # Example # jinja_template_resources("chrome_public_template_resources") { # res_dir = "res_template" # resources = ["res_template/xml/syncable.xml"] # variables = ["color=red"] # } template("jinja_template_resources") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.resources)) assert(defined(invoker.res_dir)) _base_path = "$target_gen_dir/$target_name" _resources_zip = _base_path + ".resources.zip" _build_config = _base_path + ".build_config" write_build_config("${target_name}__build_config") { build_config = _build_config resources_zip = _resources_zip type = "android_resources" } action("${target_name}__template") { sources = invoker.resources script = "//build/android/gyp/jinja_template.py" depfile = "$target_gen_dir/$target_name.d" outputs = [ depfile, _resources_zip, ] rebased_resources = rebase_path(invoker.resources, root_build_dir) args = [ "--inputs=${rebased_resources}", "--inputs-base-dir", rebase_path(invoker.res_dir, root_build_dir), "--outputs-zip", rebase_path(_resources_zip, root_build_dir), "--depfile", rebase_path(depfile, root_build_dir), ] if (defined(invoker.variables)) { variables = invoker.variables args += [ "--variables=${variables}" ] } } group(target_name) { public_deps = [ ":${target_name}__build_config", ":${target_name}__template", ] } } # Creates a resources.zip with locale.pak files placed into appropriate # resource configs (e.g. en-GB.pak -> res/raw-en/en_gb.pak). Also generates # a locale_paks TypedArray so that resource files can be enumerated at runtime. # # If this target is included in the deps of an android resources/library/apk, # the resources will be included with that target. # # Variables: # sources: List of .pak files. Names must be of the form "en.pak" or # "en-US.pak". # deps: (optional) List of dependencies that might be needed to generate # the .pak files. # # Example # locale_pak_resources("locale_paks") { # sources = [ "path/en-US.pak", "path/fr.pak", ... ] # } template("locale_pak_resources") { set_sources_assignment_filter([]) assert(defined(invoker.sources)) _base_path = "$target_gen_dir/$target_name" _resources_zip = _base_path + ".resources.zip" _build_config = _base_path + ".build_config" write_build_config("${target_name}__build_config") { build_config = _build_config resources_zip = _resources_zip type = "android_resources" is_locale_resource = true } action("${target_name}__create_resources_zip") { forward_variables_from(invoker, [ "deps", "sources", ]) script = "//build/android/gyp/locale_pak_resources.py" depfile = "$target_gen_dir/$target_name.d" outputs = [ depfile, _resources_zip, ] _rebased_sources = rebase_path(sources, root_build_dir) args = [ "--locale-paks=${_rebased_sources}", "--resources-zip", rebase_path(_resources_zip, root_build_dir), "--depfile", rebase_path(depfile, root_build_dir), ] } group(target_name) { public_deps = [ ":${target_name}__build_config", ":${target_name}__create_resources_zip", ] } } # Declare an Android resources target # # This creates a resources zip file that will be used when building an Android # library or apk and included into a final apk. # # To include these resources in a library/apk, this target should be listed in # the library's deps. A library/apk will also include any resources used by its # own dependencies. # # Variables # deps: Specifies the dependencies of this target. Any Android resources # listed in deps will be included by libraries/apks that depend on this # target. # resource_dirs: List of directories containing resources for this target. # generated_resource_dirs: List of directories containing resources for this # target which are *generated* by a dependency. |generated_resource_files| # must be specified if |generated_resource_dirs| is specified. # generated_resource_files: List of all files in |generated_resource_dirs|. # |generated_resource_dirs| must be specified in |generated_resource_files| # is specified. # android_manifest: AndroidManifest.xml for this target. Defaults to # //build/android/AndroidManifest.xml. # custom_package: java package for generated .java files. # v14_skip: If true, don't run v14 resource generator on this. Defaults to # false. (see build/android/gyp/generate_v14_compatible_resources.py) # shared_resources: If true make a resource package that can be loaded by a # different application at runtime to access the package's resources. # app_as_shared_lib: If true make a resource package that can be loaded as # both shared_resources and normal application. # Example: # android_resources("foo_resources") { # deps = [":foo_strings_grd"] # resource_dirs = ["res"] # custom_package = "org.chromium.foo" # } # # android_resources("foo_resources_overrides") { # deps = [":foo_resources"] # resource_dirs = ["res_overrides"] # } template("android_resources") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.resource_dirs)) base_path = "$target_gen_dir/$target_name" zip_path = base_path + ".resources.zip" srcjar_path = base_path + ".srcjar" r_text_path = base_path + "_R.txt" build_config = base_path + ".build_config" build_config_target_name = "${target_name}__build_config" process_resources_target_name = "${target_name}__process_resources" final_target_name = target_name write_build_config(build_config_target_name) { forward_variables_from(invoker, [ "android_manifest", "custom_package", "deps", ]) # No package means resources override their deps. if (defined(custom_package) || defined(android_manifest)) { r_text = r_text_path } else { assert(defined(invoker.deps), "Must specify deps when custom_package is omitted.") } visibility = [ ":$process_resources_target_name" ] type = "android_resources" resources_zip = zip_path srcjar = srcjar_path } process_resources(process_resources_target_name) { visibility = [ ":$final_target_name" ] forward_variables_from(invoker, [ "app_as_shared_lib", "android_manifest", "custom_package", "deps", "generated_resource_dirs", "generated_resource_files", "resource_dirs", "shared_resources", "v14_skip", ]) if (!defined(deps)) { deps = [] } deps += [ ":$build_config_target_name" ] # Always generate R.onResourcesLoaded() method, it is required for # compiling ResourceRewriter, there is no side effect because the # generated R.class isn't used in final apk. shared_resources = true if (!defined(android_manifest)) { android_manifest = "//build/android/AndroidManifest.xml" } } group(final_target_name) { forward_variables_from(invoker, [ "visibility" ]) public_deps = [ ":${target_name}__process_resources", ] } } # Declare an Android assets target. # # Defines a set of files to include as assets in a dependent apk. # # To include these assets in an apk, this target should be listed in # the apk's deps, or in the deps of a library target used by an apk. # # Variables # deps: Specifies the dependencies of this target. Any Android assets # listed in deps will be included by libraries/apks that depend on this # target. # sources: List of files to include as assets. # renaming_sources: List of files to include as assets and be renamed. # renaming_destinations: List of asset paths for files in renaming_sources. # disable_compression: Whether to disable compression for files that are # known to be compressable (default: false). # # Example: # android_assets("content_shell_assets") { # deps = [ # ":generates_foo", # ":other_assets", # ] # sources = [ # "//path/asset1.png", # "//path/asset2.png", # "$target_gen_dir/foo.dat", # ] # } # # android_assets("overriding_content_shell_assets") { # deps = [ ":content_shell_assets" ] # # Override foo.dat from content_shell_assets. # sources = [ "//custom/foo.dat" ] # renaming_sources = [ "//path/asset2.png" ] # renaming_destinations = [ "renamed/asset2.png" ] # } template("android_assets") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) _build_config = "$target_gen_dir/$target_name.build_config" _build_config_target_name = "${target_name}__build_config" write_build_config(_build_config_target_name) { forward_variables_from(invoker, [ "deps", "disable_compression", ]) type = "android_assets" build_config = _build_config if (defined(invoker.sources)) { asset_sources = invoker.sources } if (defined(invoker.renaming_sources)) { assert(defined(invoker.renaming_destinations)) _source_count = 0 foreach(_, invoker.renaming_sources) { _source_count += 1 } _dest_count = 0 foreach(_, invoker.renaming_destinations) { _dest_count += 1 } assert( _source_count == _dest_count, "android_assets() renaming_sources.length != renaming_destinations.length") asset_renaming_sources = invoker.renaming_sources asset_renaming_destinations = invoker.renaming_destinations } } group(target_name) { forward_variables_from(invoker, [ "visibility" ]) public_deps = [ ":$_build_config_target_name", ] } } # Declare a group() that supports forwarding java dependency information. # # Example # java_group("conditional_deps") { # if (enable_foo) { # deps = [":foo_java"] # } # } template("java_group") { write_build_config("${target_name}__build_config") { forward_variables_from(invoker, [ "deps" ]) type = "group" build_config = "$target_gen_dir/${invoker.target_name}.build_config" } group(target_name) { deps = [] forward_variables_from(invoker, "*") deps += [ ":${target_name}__build_config" ] } } # Declare a target that generates localized strings.xml from a .grd file. # # If this target is included in the deps of an android resources/library/apk, # the strings.xml will be included with that target. # # Variables # deps: Specifies the dependencies of this target. # grd_file: Path to the .grd file to generate strings.xml from. # outputs: Expected grit outputs (see grit rule). # # Example # java_strings_grd("foo_strings_grd") { # grd_file = "foo_strings.grd" # } template("java_strings_grd") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) base_path = "$target_gen_dir/$target_name" resources_zip = base_path + ".resources.zip" build_config = base_path + ".build_config" write_build_config("${target_name}__build_config") { forward_variables_from(invoker, [ "deps" ]) type = "android_resources" } # Put grit files into this subdirectory of target_gen_dir. extra_output_path = target_name + "_grit_output" grit_target_name = "${target_name}__grit" grit_output_dir = "$target_gen_dir/$extra_output_path" grit(grit_target_name) { grit_flags = [ "-E", "ANDROID_JAVA_TAGGED_ONLY=false", ] output_dir = grit_output_dir resource_ids = "" source = invoker.grd_file outputs = invoker.outputs } # This needs to get outputs from grit's internal target, not the final # source_set. generate_strings_outputs = get_target_outputs(":${grit_target_name}_grit") zip("${target_name}__zip") { base_dir = grit_output_dir inputs = generate_strings_outputs output = resources_zip deps = [ ":$grit_target_name", ] } group(target_name) { public_deps = [ ":${target_name}__build_config", ":${target_name}__zip", ] } } # Declare a target that packages strings.xml generated from a grd file. # # If this target is included in the deps of an android resources/library/apk, # the strings.xml will be included with that target. # # Variables # grit_output_dir: directory containing grit-generated files. # generated_files: list of android resource files to package. # # Example # java_strings_grd_prebuilt("foo_strings_grd") { # grit_output_dir = "$root_gen_dir/foo/grit" # generated_files = [ # "values/strings.xml" # ] # } template("java_strings_grd_prebuilt") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) base_path = "$target_gen_dir/$target_name" resources_zip = base_path + ".resources.zip" build_config = base_path + ".build_config" build_config_target_name = "${target_name}__build_config" zip_target_name = "${target_name}__zip" final_target_name = target_name write_build_config(build_config_target_name) { visibility = [ ":$zip_target_name" ] type = "android_resources" } zip(zip_target_name) { visibility = [ ":$final_target_name" ] base_dir = invoker.grit_output_dir inputs = rebase_path(invoker.generated_files, ".", base_dir) output = resources_zip deps = [ ":$build_config_target_name", ] if (defined(invoker.deps)) { deps += invoker.deps } } group(final_target_name) { forward_variables_from(invoker, [ "visibility" ]) public_deps = [ ":$zip_target_name", ] } } # Declare a Java executable target # # This target creates an executable from java code and libraries. The executable # will be in the output folder's /bin/ directory. # # Variables # deps: Specifies the dependencies of this target. Java targets in this list # will be included in the executable (and the javac classpath). # java_files: List of .java files included in this library. # srcjar_deps: List of srcjar dependencies. The .java files in the srcjars # will be added to java_files and be included in this library. # srcjars: List of srcjars to be included in this library, together with the # ones obtained from srcjar_deps. # bypass_platform_checks: Disables checks about cross-platform (Java/Android) # dependencies for this target. This will allow depending on an # android_library target, for example. # chromium_code: If true, extra analysis warning/errors will be enabled. # enable_errorprone: If true, enables the errorprone compiler. # enable_incremental_javac_override: Overrides the # global enable_incremental_javac. # main_class: When specified, a wrapper script is created within # $root_build_dir/bin to launch the binary with the given class as the # entrypoint. # wrapper_script_args: List of additional arguments for the wrapper script. # # data_deps, testonly # # Example # java_binary("foo") { # java_files = [ "org/chromium/foo/FooMain.java" ] # deps = [ ":bar_java" ] # main_class = "org.chromium.foo.FooMain" # } template("java_binary") { set_sources_assignment_filter([]) java_library_impl(target_name) { forward_variables_from(invoker, "*") supports_android = false main_class = invoker.main_class is_java_binary = true } } # Declare a Junit executable target # # This target creates an executable from java code for running as a junit test # suite. The executable will be in the output folder's /bin/ directory. # # Variables # deps: Specifies the dependencies of this target. Java targets in this list # will be included in the executable (and the javac classpath). # # java_files: List of .java files included in this library. # srcjar_deps: List of srcjar dependencies. The .java files in the srcjars # will be added to java_files and be included in this library. # srcjars: List of srcjars to be included in this library, together with the # ones obtained from srcjar_deps. # # chromium_code: If true, extra analysis warning/errors will be enabled. # # Example # junit_binary("foo") { # java_files = [ "org/chromium/foo/FooTest.java" ] # deps = [ ":bar_java" ] # } template("junit_binary") { set_sources_assignment_filter([]) testonly = true _java_binary_target_name = "${target_name}__java_binary" _test_runner_target_name = "${target_name}__test_runner_script" test_runner_script(_test_runner_target_name) { test_name = invoker.target_name test_suite = invoker.target_name test_type = "junit" } java_binary(_java_binary_target_name) { deps = [] jar_name = invoker.target_name forward_variables_from(invoker, "*") testonly = true bypass_platform_checks = true main_class = "org.chromium.testing.local.JunitTestMain" wrapper_script_name = "helper/$target_name" deps += [ "//testing/android/junit:junit_test_support", "//third_party/junit", "//third_party/mockito:mockito_java", "//third_party/robolectric:android-all-4.3_r2-robolectric-0", "//third_party/robolectric:robolectric_java", ] } group(target_name) { public_deps = [ ":$_java_binary_target_name", ":$_test_runner_target_name", ] } } # Declare a java library target # # Variables # deps: Specifies the dependencies of this target. Java targets in this list # will be added to the javac classpath. # # java_files: List of .java files included in this library. # srcjar_deps: List of srcjar dependencies. The .java files in the srcjars # will be added to java_files and be included in this library. # srcjars: List of srcjars to be included in this library, together with the # ones obtained from srcjar_deps. # # chromium_code: If true, extra analysis warning/errors will be enabled. # enable_errorprone: If true, enables the errorprone compiler. # enable_incremental_javac_override: Overrides the global # enable_incremental_javac. # # jar_excluded_patterns: List of patterns of .class files to exclude from the # final jar. # # proguard_preprocess: If true, proguard preprocessing will be run. This can # be used to remove unwanted parts of the library. # proguard_config: Path to the proguard config for preprocessing. # # supports_android: If true, Android targets (android_library, android_apk) # may depend on this target. Note: if true, this target must only use the # subset of Java available on Android. # bypass_platform_checks: Disables checks about cross-platform (Java/Android) # dependencies for this target. This will allow depending on an # android_library target, for example. # # data_deps, testonly # # Example # java_library("foo_java") { # java_files = [ # "org/chromium/foo/Foo.java", # "org/chromium/foo/FooInterface.java", # "org/chromium/foo/FooService.java", # ] # deps = [ # ":bar_java" # ] # srcjar_deps = [ # ":foo_generated_enum" # ] # jar_excluded_patterns = [ # "*/FooService.class", "*/FooService##*.class" # ] # } template("java_library") { set_sources_assignment_filter([]) java_library_impl(target_name) { forward_variables_from(invoker, "*") } } # Declare a java library target for a prebuilt jar # # Variables # deps: Specifies the dependencies of this target. Java targets in this list # will be added to the javac classpath. # jar_path: Path to the prebuilt jar. # jar_dep: Target that builds jar_path (optional). # proguard_preprocess: If true, proguard preprocessing will be run. This can # be used to remove unwanted parts of the library. # proguard_config: Path to the proguard config for preprocessing. # supports_android: If true, Android targets (android_library, android_apk) # may depend on this target. Note: if true, this target must only use the # subset of Java available on Android. # # Example # java_prebuilt("foo_java") { # jar_path = "foo.jar" # deps = [ # ":foo_resources", # ":bar_java" # ] # } template("java_prebuilt") { set_sources_assignment_filter([]) java_prebuilt_impl(target_name) { forward_variables_from(invoker, "*") } } # Declare an Android library target # # This target creates an Android library containing java code and Android # resources. # # Variables # deps: Specifies the dependencies of this target. Java targets in this list # will be added to the javac classpath. Android resources in dependencies # will be used when building this library. # # java_files: List of .java files included in this library. # srcjar_deps: List of srcjar dependencies. The .java files in the srcjars # will be added to java_files and be included in this library. # srcjars: List of srcjars to be included in this library, together with the # ones obtained from srcjar_deps. # # chromium_code: If true, extra analysis warning/errors will be enabled. # enable_errorprone: If true, enables the errorprone compiler. # enable_incremental_javac_override: Overrides the global # enable_incremental_javac. # # jar_excluded_patterns: List of patterns of .class files to exclude from the # final jar. # # proguard_preprocess: If true, proguard preprocessing will be run. This can # be used to remove unwanted parts of the library. # proguard_config: Path to the proguard config for preprocessing. # # dex_path: If set, the resulting .dex.jar file will be placed under this # path. # # alternative_android_sdk_ijar: if set, the given android_sdk_ijar file # replaces the default android_sdk_ijar. # # alternative_android_sdk_ijar_dep: the target that generates # alternative_android_sdk_ijar, must be set if alternative_android_sdk_ijar # is used. # # emma_never_instrument: Disables EMMA Java code coverage for this target. # # Example # android_library("foo_java") { # java_files = [ # "android/org/chromium/foo/Foo.java", # "android/org/chromium/foo/FooInterface.java", # "android/org/chromium/foo/FooService.java", # ] # deps = [ # ":bar_java" # ] # srcjar_deps = [ # ":foo_generated_enum" # ] # jar_excluded_patterns = [ # "*/FooService.class", "*/FooService##*.class" # ] # } template("android_library") { set_sources_assignment_filter([]) assert(!defined(invoker.jar_path), "android_library does not support a custom jar path") if (defined(invoker.alternative_android_sdk_ijar)) { assert(defined(invoker.alternative_android_sdk_ijar_dep)) } java_library_impl(target_name) { forward_variables_from(invoker, "*") supports_android = true requires_android = true if (!defined(jar_excluded_patterns)) { jar_excluded_patterns = [] } jar_excluded_patterns += [ "*/R.class", "*/R##*.class", "*/Manifest.class", "*/Manifest##*.class", ] } } # Declare a target that packages a set of Java dependencies into a standalone # .dex.jar. # # Variables # deps: specifies the dependencies of this target. Android libraries in deps # will be packaged into the resulting .dex.jar file. # dex_path: location at which the output file will be put template("android_standalone_library") { set_sources_assignment_filter([]) deps_dex(target_name) { forward_variables_from(invoker, [ "deps", "dex_path", "excluded_jars", ]) } } # Declare an Android library target for a prebuilt jar # # This target creates an Android library containing java code and Android # resources. # # Variables # deps: Specifies the dependencies of this target. Java targets in this list # will be added to the javac classpath. Android resources in dependencies # will be used when building this library. # jar_path: Path to the prebuilt jar. # proguard_preprocess: If true, proguard preprocessing will be run. This can # be used to remove unwanted parts of the library. # proguard_config: Path to the proguard config for preprocessing. # # Example # android_java_prebuilt("foo_java") { # jar_path = "foo.jar" # deps = [ # ":foo_resources", # ":bar_java" # ] # } template("android_java_prebuilt") { set_sources_assignment_filter([]) java_prebuilt_impl(target_name) { forward_variables_from(invoker, "*") supports_android = true requires_android = true strip_resource_classes = true } } # Declare an Android apk target # # This target creates an Android APK containing java code, resources, assets, # and (possibly) native libraries. # # Variables # alternative_android_sdk_jar: The alternative android sdk jar used in # proguard. # android_aapt_path: Android aapt tool to replace default one to build # resource. # android_manifest: Path to AndroidManifest.xml. # android_manifest_dep: Target that generates AndroidManifest (if applicable) # chromium_code: If true, extra analysis warning/errors will be enabled. # create_dist_ijar: Whether to define the "${target_name}_dist_ijar" target # (used by instrumentation_test_apk). # data_deps: List of dependencies needed at runtime. These will be built but # won't change the generated .apk in any way (in fact they may be built # after the .apk is). # deps: List of dependencies. All Android java resources and libraries in the # "transitive closure" of these dependencies will be included in the apk. # Note: this "transitive closure" actually only includes such targets if # they are depended on through android_library or android_resources targets # (and so not through builtin targets like 'action', 'group', etc). # install_script_name: Name of wrapper script (default=target_name). # java_files: List of .java files to include in the apk. # srcjar_deps: List of srcjar dependencies. The .java files in the srcjars # will be added to java_files and be included in this apk. # apk_name: Name for final apk. # final_apk_path: Path to final built apk. Default is # $root_out_dir/apks/$apk_name.apk. Setting this will override apk_name. # loadable_modules: List of paths to native libraries to include. Different # from |native_libs| in that: # * dependencies of this .so are not automatically included # * ".cr.so" is never added # * they are not side-loaded for _incremental targets. # * load_library_from_apk, use_chromium_linker, # and enable_relocation_packing do not apply # Use this instead of native_libs when you are going to load the library # conditionally, and only when native_libs doesn't work for you. # native_libs: List paths of native libraries to include in this apk. If these # libraries depend on other shared_library targets, those dependencies will # also be included in the apk. When building with is_component_build, # The extension is automatically changed to ".cr.so". # native_lib_placeholders: List of placeholder filenames to add to the apk # (optional). # apk_under_test: For an instrumentation test apk, this is the target of the # tested apk. # include_all_resources - If true include all resource IDs in all generated # R.java files. # testonly: Marks this target as "test-only". # write_asset_list: Adds an extra file to the assets, which contains a list of # all other asset files. # alternative_locale_resource_dep: The locale resource target which overrides # any exsting locale resources in dep graph. # requires_sdk_api_level_23: If defined and true, the apk is intended for # installation only on Android M or later. In these releases the system # linker does relocation unpacking, so we can enable it unconditionally. # secondary_native_libs: the path of native libraries for secondary app abi. # run_findbugs_override: Forces run_findbugs on or off. If undefined, the # default will use the build arg run_findbugs. # # Example # android_apk("foo_apk") { # android_manifest = "AndroidManifest.xml" # java_files = [ # "android/org/chromium/foo/FooApplication.java", # "android/org/chromium/foo/FooActivity.java", # ] # deps = [ # ":foo_support_java" # ":foo_resources" # ] # srcjar_deps = [ # ":foo_generated_enum" # ] # native_libs = [ # native_lib_path # ] # } template("android_apk") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) assert(defined(invoker.final_apk_path) || defined(invoker.apk_name)) assert(defined(invoker.android_manifest)) gen_dir = "$target_gen_dir/$target_name" base_path = "$gen_dir/$target_name" _build_config = "$target_gen_dir/$target_name.build_config" resources_zip_path = "$base_path.resources.zip" _all_resources_zip_path = "$base_path.resources.all.zip" _jar_path = "$base_path.jar" _lib_dex_path = "$base_path.dex.jar" _rebased_lib_dex_path = rebase_path(_lib_dex_path, root_build_dir) _template_name = target_name enable_multidex = defined(invoker.enable_multidex) && invoker.enable_multidex if (enable_multidex) { final_dex_path = "$gen_dir/classes.dex.zip" } else { final_dex_path = "$gen_dir/classes.dex" } final_dex_target_name = "${_template_name}__final_dex" _final_apk_path = "" if (defined(invoker.final_apk_path)) { _final_apk_path = invoker.final_apk_path } else if (defined(invoker.apk_name)) { _final_apk_path = "$root_build_dir/apks/" + invoker.apk_name + ".apk" } _final_apk_path_no_ext_list = process_file_template([ _final_apk_path ], "{{source_dir}}/{{source_name_part}}") _final_apk_path_no_ext = _final_apk_path_no_ext_list[0] assert(_final_apk_path_no_ext != "") # Mark as used. _install_script_name = "install_$_template_name" if (defined(invoker.install_script_name)) { _install_script_name = invoker.install_script_name } _incremental_install_script_path = "${root_out_dir}/bin/${_install_script_name}_incremental" _native_libs = [] _version_code = android_default_version_code if (defined(invoker.version_code)) { _version_code = invoker.version_code } _version_name = android_default_version_name if (defined(invoker.version_name)) { _version_name = invoker.version_name } _keystore_path = android_keystore_path _keystore_name = android_keystore_name _keystore_password = android_keystore_password if (defined(invoker.keystore_path)) { _keystore_path = invoker.keystore_path _keystore_name = invoker.keystore_name _keystore_password = invoker.keystore_password } _srcjar_deps = [] if (defined(invoker.srcjar_deps)) { _srcjar_deps += invoker.srcjar_deps } _use_chromium_linker = defined(invoker.use_chromium_linker) && invoker.use_chromium_linker _enable_relocation_packing = defined(invoker.enable_relocation_packing) && invoker.enable_relocation_packing _load_library_from_apk = defined(invoker.load_library_from_apk) && invoker.load_library_from_apk _requires_sdk_api_level_23 = defined(invoker.requires_sdk_api_level_23) && invoker.requires_sdk_api_level_23 assert(_use_chromium_linker || true) # Mark as used. assert(_requires_sdk_api_level_23 || true) if (_enable_relocation_packing) { assert(_use_chromium_linker || _requires_sdk_api_level_23, "enable_relocation_packing requires either use_chromium_linker " + "or requires_sdk_api_level_23") } if (_load_library_from_apk) { assert(_use_chromium_linker || _requires_sdk_api_level_23, "load_library_from_apk requires use_chromium_linker " + "or requires_sdk_api_level_23") } # The dependency that makes the chromium linker, if any is needed. _native_libs_deps = [] if (defined(invoker.native_libs) && invoker.native_libs != []) { if (is_component_build || is_asan) { _native_libs += [ "$root_shlib_dir/libc++_shared.so" ] _native_libs_deps += [ "//build/android:cpplib_stripped" ] } # Allow native_libs to be in the form "foo.so" or "foo.cr.so" _first_ext_removed = process_file_template(invoker.native_libs, "{{source_name_part}}") _native_libs += process_file_template( _first_ext_removed, "$root_shlib_dir/{{source_name_part}}$shlib_extension") _native_lib_version_rule = "" if (defined(invoker.native_lib_version_rule)) { _native_lib_version_rule = invoker.native_lib_version_rule } _native_lib_version_arg = "\"\"" if (defined(invoker.native_lib_version_arg)) { _native_lib_version_arg = invoker.native_lib_version_arg } } _android_manifest_deps = [] if (defined(invoker.android_manifest_dep)) { _android_manifest_deps = [ invoker.android_manifest_dep ] } _android_manifest = invoker.android_manifest _rebased_build_config = rebase_path(_build_config, root_build_dir) _create_abi_split = defined(invoker.create_abi_split) && invoker.create_abi_split _create_density_splits = defined(invoker.create_density_splits) && invoker.create_density_splits _create_language_splits = defined(invoker.language_splits) && invoker.language_splits != [] # Help GN understand that _create_abi_split is not unused (bug in GN). assert(_create_abi_split || true) _proguard_enabled = defined(invoker.proguard_enabled) && invoker.proguard_enabled if (_proguard_enabled) { _proguard_jar_path = "$base_path.proguard.jar" } _emma_never_instrument = defined(invoker.testonly) && invoker.testonly build_config_target = "${_template_name}__build_config" write_build_config(build_config_target) { forward_variables_from(invoker, [ "apk_under_test" ]) type = "android_apk" jar_path = _jar_path dex_path = final_dex_path apk_path = _final_apk_path incremental_apk_path = "${_final_apk_path_no_ext}_incremental.apk" incremental_install_script_path = _incremental_install_script_path resources_zip = resources_zip_path build_config = _build_config android_manifest = _android_manifest deps = _native_libs_deps + _android_manifest_deps if (defined(invoker.deps)) { deps += invoker.deps } if (defined(invoker.alternative_locale_resource_dep)) { deps += [ invoker.alternative_locale_resource_dep ] has_alternative_locale_resource = true } # Added emma to the target's classpath via its .build_config. if (emma_coverage && !_emma_never_instrument) { deps += [ "//third_party/android_tools:emma_device" ] } proguard_enabled = _proguard_enabled if (_proguard_enabled) { proguard_info = "$_proguard_jar_path.info" } native_libs = _native_libs } _final_deps = [] _generated_proguard_config = "$base_path.resources.proguard.txt" process_resources_target = "${_template_name}__process_resources" process_resources(process_resources_target) { forward_variables_from(invoker, [ "alternative_android_sdk_jar", "android_aapt_path", "app_as_shared_lib", "include_all_resources", "shared_resources", ]) srcjar_path = "${target_gen_dir}/${target_name}.srcjar" r_text_path = "${target_gen_dir}/${target_name}_R.txt" android_manifest = _android_manifest resource_dirs = [ "//build/android/ant/empty/res" ] zip_path = resources_zip_path all_resources_zip_path = _all_resources_zip_path generate_constant_ids = true proguard_file = _generated_proguard_config build_config = _build_config deps = _android_manifest_deps + [ ":$build_config_target" ] if (defined(invoker.deps)) { deps += invoker.deps } } _srcjar_deps += [ ":$process_resources_target" ] if (_native_libs != []) { _enable_chromium_linker_tests = false if (defined(invoker.enable_chromium_linker_tests)) { _enable_chromium_linker_tests = invoker.enable_chromium_linker_tests } java_cpp_template("${_template_name}__native_libraries_java") { package_name = "org/chromium/base/library_loader" sources = [ "//base/android/java/templates/NativeLibraries.template", ] inputs = [ _build_config, ] deps = [ ":$build_config_target", ] if (_native_lib_version_rule != "") { deps += [ _native_lib_version_rule ] } defines = [ "NATIVE_LIBRARIES_LIST=" + "@FileArg($_rebased_build_config:native:java_libraries_list)", "NATIVE_LIBRARIES_VERSION_NUMBER=$_native_lib_version_arg", ] if (_use_chromium_linker) { defines += [ "ENABLE_CHROMIUM_LINKER" ] } if (_load_library_from_apk) { defines += [ "ENABLE_CHROMIUM_LINKER_LIBRARY_IN_ZIP_FILE" ] } if (_enable_chromium_linker_tests) { defines += [ "ENABLE_CHROMIUM_LINKER_TESTS" ] } } _srcjar_deps += [ ":${_template_name}__native_libraries_java" ] } if (!defined(invoker.apk_under_test)) { java_cpp_template("${_template_name}__build_config_java") { package_name = "org/chromium/base" sources = [ "//base/android/java/templates/BuildConfig.template", ] defines = [] if (enable_multidex) { defines += [ "ENABLE_MULTIDEX" ] } } _srcjar_deps += [ ":${_template_name}__build_config_java" ] } java_target = "${_template_name}__java" java_library_impl(java_target) { forward_variables_from(invoker, [ "chromium_code", "java_files", "run_findbugs_override", ]) supports_android = true requires_android = true override_build_config = _build_config deps = _android_manifest_deps + [ ":$build_config_target" ] android_manifest = _android_manifest srcjar_deps = _srcjar_deps jar_path = _jar_path dex_path = _lib_dex_path emma_never_instrument = _emma_never_instrument if (defined(invoker.deps)) { deps += invoker.deps } if (defined(invoker.apk_under_test)) { deps += [ "${invoker.apk_under_test}__java" ] } } # TODO(cjhopman): This is only ever needed to calculate the list of tests to # run. See build/android/pylib/instrumentation/test_jar.py. We should be # able to just do that calculation at build time instead. if (defined(invoker.create_dist_ijar) && invoker.create_dist_ijar) { _dist_ijar_path = "$root_build_dir/test.lib.java/" + get_path_info(_final_apk_path, "name") + ".jar" action("${_template_name}_dist_ijar") { script = "//build/android/gyp/create_dist_jar.py" depfile = "$target_gen_dir/$target_name.d" inputs = [ _build_config, ] outputs = [ depfile, "${_dist_ijar_path}", ] data = [ _dist_ijar_path, ] args = [ "--depfile", rebase_path(depfile, root_build_dir), "--output", rebase_path("${_dist_ijar_path}", root_build_dir), "--inputs=@FileArg($_rebased_build_config:dist_jar:all_interface_jars)", ] deps = [ ":$build_config_target", # Generates the build config file. ":$java_target", # Generates the jar file. ] } } if (_proguard_enabled) { _proguard_configs = [ _generated_proguard_config ] if (defined(invoker.proguard_configs)) { _proguard_configs += invoker.proguard_configs } _proguard_target = "${_template_name}__proguard" proguard(_proguard_target) { forward_variables_from(invoker, [ "alternative_android_sdk_jar" ]) deps = [ ":$build_config_target", ":$java_target", ":$process_resources_target", ] inputs = [ _build_config, _jar_path, ] + _proguard_configs output_jar_path = _proguard_jar_path _rebased_proguard_configs = rebase_path(_proguard_configs, root_build_dir) args = [ "--proguard-configs=$_rebased_proguard_configs", "--input-paths=@FileArg($_rebased_build_config:proguard:input_paths)", ] if (defined(invoker.apk_under_test)) { deps += [ "${invoker.apk_under_test}__build_config", "${invoker.apk_under_test}__proguard", ] _apk_under_test_build_config = get_label_info(invoker.apk_under_test, "target_gen_dir") + "/" + get_label_info(invoker.apk_under_test, "name") + ".build_config" _rebased_apk_under_test_build_config = rebase_path(_apk_under_test_build_config, root_build_dir) args += [ "--tested-apk-info=@FileArg($_rebased_apk_under_test_build_config:deps_info:proguard_info)" ] } } _dex_sources = [ _proguard_jar_path ] _dex_deps = [ ":$_proguard_target" ] } else { if (enable_multidex) { _dex_sources = [ _jar_path ] } else { _dex_sources = [ _lib_dex_path ] } _dex_deps = [ ":$java_target" ] } dex("$final_dex_target_name") { deps = _dex_deps + [ ":$build_config_target" ] inputs = [ _build_config, ] sources = _dex_sources output = final_dex_path # All deps are already included in _dex_sources when proguard is used. if (!_proguard_enabled) { if (enable_multidex) { _dex_arg_key = "${_rebased_build_config}:dist_jar:dependency_jars" } else { _dex_arg_key = "${_rebased_build_config}:final_dex:dependency_dex_files" } args = [ "--inputs=@FileArg($_dex_arg_key)" ] } } _native_libs_file_arg_dep = ":$build_config_target" _native_libs_file_arg = "@FileArg($_rebased_build_config:native:libraries)" if (_native_libs != [] && _enable_relocation_packing) { _prepare_native_target_name = "${_template_name}__prepare_native" _native_libs_dir = "$gen_dir/packed-libs" _native_libs_json = "$gen_dir/packed-libs/filelist.json" _rebased_native_libs_json = rebase_path(_native_libs_json, root_build_dir) _native_libs_file_arg_dep = ":$_prepare_native_target_name" _native_libs_file_arg = "@FileArg($_rebased_native_libs_json:files)" action(_prepare_native_target_name) { forward_variables_from(invoker, [ "deps", "public_deps", ]) script = "//build/android/gyp/pack_relocations.py" depfile = "$target_gen_dir/$target_name.d" outputs = [ depfile, _native_libs_json, ] inputs = _native_libs + [ _build_config ] deps += _native_libs_deps deps += [ ":$build_config_target", relocation_packer_target, ] args = [ "--depfile", rebase_path(depfile, root_build_dir), "--enable-packing=1", "--android-pack-relocations", rebase_path(relocation_packer_exe, root_build_dir), "--stripped-libraries-dir", rebase_path(root_build_dir, root_build_dir), "--packed-libraries-dir", rebase_path(_native_libs_dir, root_build_dir), "--libraries=@FileArg(${_rebased_build_config}:native:libraries)", "--filelistjson=$_rebased_native_libs_json", ] } } _extra_native_libs = [] _extra_native_libs_deps = [] _extra_native_libs_even_when_incremental = [] _extra_native_libs_even_when_incremental_deps = [] assert(_extra_native_libs_even_when_incremental_deps == []) # Mark as used. if (_native_libs != []) { if (is_debug) { _extra_native_libs_even_when_incremental = [ android_gdbserver ] } if (_use_chromium_linker) { _extra_native_libs = [ "$root_shlib_dir/libchromium_android_linker$shlib_extension" ] _extra_native_libs_deps += [ "//base/android/linker:chromium_android_linker" ] } } if (defined(invoker.loadable_modules) && invoker.loadable_modules != []) { _extra_native_libs_even_when_incremental += invoker.loadable_modules } _final_deps += [ ":${_template_name}__create" ] create_apk("${_template_name}__create") { forward_variables_from(invoker, [ "alternative_android_sdk_jar", "android_aapt_path", "app_as_shared_lib", "deps", "extensions_to_not_compress", "language_splits", "page_align_shared_libraries", "public_deps", "secondary_native_libs", "shared_resources", "uncompress_shared_libraries", "write_asset_list", ]) if (!defined(deps)) { deps = [] } apk_path = _final_apk_path android_manifest = _android_manifest assets_build_config = _build_config resources_zip = _all_resources_zip_path dex_path = final_dex_path load_library_from_apk = _load_library_from_apk create_density_splits = _create_density_splits emma_instrument = emma_coverage && !_emma_never_instrument if (!defined(extensions_to_not_compress)) { # Allow icu data, v8 snapshots, and pak files to be loaded directly from # the .apk. # Note: These are actually suffix matches, not necessarily extensions. extensions_to_not_compress = ".dat,.bin,.pak" } version_code = _version_code version_name = _version_name keystore_name = _keystore_name keystore_path = _keystore_path keystore_password = _keystore_password # Incremental apk does not use native libs nor final dex. incremental_deps = deps + _android_manifest_deps + [ ":$build_config_target", ":$process_resources_target", ] # This target generates the input file _all_resources_zip_path. deps += _android_manifest_deps + [ ":$build_config_target", ":$process_resources_target", ":$final_dex_target_name", ] if ((_native_libs != [] || _extra_native_libs_even_when_incremental != []) && !_create_abi_split) { deps += _native_libs_deps + _extra_native_libs_deps + _extra_native_libs_even_when_incremental_deps + [ _native_libs_file_arg_dep ] native_libs_filearg = _native_libs_file_arg native_libs = _extra_native_libs native_libs_even_when_incremental = _extra_native_libs_even_when_incremental } # Placeholders necessary for some older devices. # http://crbug.com/395038 forward_variables_from(invoker, [ "native_lib_placeholders" ]) } if ((_native_libs != [] || _extra_native_libs_even_when_incremental != []) && _create_abi_split) { _manifest_rule = "${_template_name}__split_manifest_abi_${android_app_abi}" generate_split_manifest(_manifest_rule) { main_manifest = _android_manifest out_manifest = "$gen_dir/split-manifests/${android_app_abi}/AndroidManifest.xml" split_name = "abi_${android_app_abi}" deps = _android_manifest_deps } _apk_rule = "${_template_name}__split_apk_abi_${android_app_abi}" _final_deps += [ ":$_apk_rule" ] create_apk(_apk_rule) { apk_path = "${_final_apk_path_no_ext}-abi-${android_app_abi}.apk" base_path = "$gen_dir/$_apk_rule" manifest_outputs = get_target_outputs(":${_manifest_rule}") android_manifest = manifest_outputs[1] load_library_from_apk = _load_library_from_apk version_code = _version_code version_name = _version_name keystore_name = _keystore_name keystore_path = _keystore_path keystore_password = _keystore_password # Placeholders necessary for some older devices. # http://crbug.com/395038 deps = [] forward_variables_from(invoker, [ "alternative_android_sdk_jar", "android_aapt_path", "deps", "native_lib_placeholders", "public_deps", ]) incremental_deps = deps + _extra_native_libs_even_when_incremental_deps + [ ":$_manifest_rule" ] deps = [] deps = incremental_deps + _native_libs_deps + _extra_native_libs_deps + [ _native_libs_file_arg_dep ] native_libs_filearg = _native_libs_file_arg native_libs = _extra_native_libs native_libs_even_when_incremental = _extra_native_libs_even_when_incremental } } _create_incremental_script_rule_name = "${_template_name}__incremental_script" action(_create_incremental_script_rule_name) { script = "//build/android/incremental_install/create_install_script.py" depfile = "$target_gen_dir/$target_name.d" deps = [ _native_libs_file_arg_dep, ] outputs = [ depfile, _incremental_install_script_path, ] _rebased_apk_path_no_ext = rebase_path(_final_apk_path_no_ext, root_build_dir) _rebased_incremental_install_script_path = rebase_path(_incremental_install_script_path, root_build_dir) _rebased_depfile = rebase_path(depfile, root_build_dir) _dex_arg_key = "${_rebased_build_config}:final_dex:dependency_dex_files" args = [ "--apk-path=${_rebased_apk_path_no_ext}_incremental.apk", "--script-output-path=$_rebased_incremental_install_script_path", "--dex-file=$_rebased_lib_dex_path", "--dex-file-list=@FileArg($_dex_arg_key)", "--depfile=$_rebased_depfile", ] if (_proguard_enabled) { args += [ "--show-proguard-warning" ] } if (defined(_native_libs_file_arg)) { args += [ "--native-libs=$_native_libs_file_arg" ] } if (_extra_native_libs != []) { # Don't pass in _extra_native_libs_even_when_incremental, since these are # end up in the apk and are not side-loaded. _rebased_extra_native_libs = rebase_path(_extra_native_libs, root_build_dir) args += [ "--native-libs=$_rebased_extra_native_libs" ] } if (_create_density_splits) { args += [ "--split=${_rebased_apk_path_no_ext}-density-*.apk" ] } if (_create_language_splits) { args += [ "--split=${_rebased_apk_path_no_ext}-language-*.apk" ] } if (_load_library_from_apk) { args += [ "--dont-even-try=Incremental builds do not work with load_library_from_apk. Try setting is_component_build=true in your GN args." ] } } group(target_name) { forward_variables_from(invoker, [ "data", "data_deps", ]) public_deps = _final_deps } group("${target_name}_incremental") { forward_variables_from(invoker, [ "data", "data_deps", ]) if (!defined(data_deps)) { data_deps = [] } # device/commands is used by the installer script to push files via .zip. data_deps += [ "//build/android/pylib/device/commands" ] + _native_libs_deps + _extra_native_libs_deps # Since the _incremental.apk does not include use .so nor .dex from the # actual target, but instead loads them at runtime, we need to explicitly # depend on them here. public_deps = [ ":${_create_incremental_script_rule_name}", ":${_template_name}__create_incremental", ":${java_target}", ] } } # Declare an Android instrumentation test apk # # This target creates an Android instrumentation test apk. # # Variables # android_manifest: Path to AndroidManifest.xml. # data_deps: List of dependencies needed at runtime. These will be built but # won't change the generated .apk in any way (in fact they may be built # after the .apk is). # deps: List of dependencies. All Android java resources and libraries in the # "transitive closure" of these dependencies will be included in the apk. # Note: this "transitive closure" actually only includes such targets if # they are depended on through android_library or android_resources targets # (and so not through builtin targets like 'action', 'group', etc). # java_files: List of .java files to include in the apk. # srcjar_deps: List of srcjar dependencies. The .java files in the srcjars # will be added to java_files and be included in this apk. # apk_name: Name for final apk. # final_apk_path: Path to final built apk. Default is # $root_out_dir/apks/$apk_name.apk. Setting this will override apk_name. # native_libs: List paths of native libraries to include in this apk. If these # libraries depend on other shared_library targets, those dependencies will # also be included in the apk. # apk_under_test: The apk being tested. # isolate_file: Isolate file containing the list of test data dependencies. # # Example # instrumentation_test_apk("foo_test_apk") { # android_manifest = "AndroidManifest.xml" # apk_name = "FooTest" # apk_under_test = "Foo" # java_files = [ # "android/org/chromium/foo/FooTestCase.java", # "android/org/chromium/foo/FooExampleTest.java", # ] # deps = [ # ":foo_test_support_java" # ] # } template("instrumentation_test_apk") { testonly = true _apk_target_name = "${target_name}__apk" _test_runner_target_name = "${target_name}__test_runner_script" _install_script_name = "install_$target_name" test_runner_script(_test_runner_target_name) { forward_variables_from(invoker, [ "additional_apks", "apk_under_test", "isolate_file", ]) test_name = invoker.target_name test_type = "instrumentation" apk_target = ":$_apk_target_name" } test_runner_script("${_test_runner_target_name}_incremental") { forward_variables_from(invoker, [ "additional_apks", "apk_under_test", "isolate_file", ]) test_name = "${invoker.target_name}_incremental" test_type = "instrumentation" apk_target = ":$_apk_target_name" incremental_install = true } android_apk(_apk_target_name) { deps = [] data_deps = [] forward_variables_from(invoker, "*") install_script_name = _install_script_name deps += [ "//testing/android/broker:broker_java" ] data_deps += [ "//testing/android/driver:driver_apk", "//tools/android/forwarder2", "//tools/android/md5sum", ] if (defined(invoker.additional_apks)) { data_deps += invoker.additional_apks } create_dist_ijar = true if (defined(invoker.run_findbugs_override)) { # Only allow findbugs when there are java files. run_findbugs_override = invoker.run_findbugs_override && defined(invoker.java_files) } } group(target_name) { public_deps = [ ":$_apk_target_name", ":$_test_runner_target_name", # Required by test runner to enumerate test list. ":${_apk_target_name}_dist_ijar", ] if (defined(invoker.apk_under_test)) { public_deps += [ invoker.apk_under_test ] } if (defined(invoker.isolate_file)) { isolate_values = exec_script("//build/gypi_to_gn.py", [ rebase_path(invoker.isolate_file), "--replace", "<(DEPTH)=/", ], "scope", [ invoker.isolate_file ]) data = isolate_values.files } } # TODO: Delete once recipes no longer use this target. group("${target_name}_run") { public_deps = [ ":${invoker.target_name}", ] } group("${target_name}_incremental") { public_deps = [ ":${_apk_target_name}_dist_ijar", ":${_apk_target_name}_incremental", ":${_test_runner_target_name}_incremental", ] if (defined(invoker.apk_under_test)) { public_deps += [ "${invoker.apk_under_test}_incremental" ] } } } # Declare an Android gtest apk # # This target creates an Android apk for running gtest-based unittests. # # Variables # deps: Specifies the dependencies of this target. These will be passed to # the underlying android_apk invocation and should include the java and # resource dependencies of the apk. # unittests_dep: This should be the label of the gtest native target. This # target must be defined previously in the same file. # unittests_binary: The basename of the library produced by the unittests_dep # target. If unspecified, it assumes the name of the unittests_dep target # (which will be correct unless that target specifies an "output_name". # TODO(brettw) make this automatic by allowing get_target_outputs to # support executables. # apk_name: The name of the produced apk. If unspecified, it uses the name # of the unittests_dep target postfixed with "_apk" # use_default_launcher: Whether the default activity (NativeUnitTestActivity) # should be used for launching tests. # # Example # unittest_apk("foo_unittests_apk") { # deps = [ ":foo_java", ":foo_resources" ] # unittests_dep = ":foo_unittests" # } template("unittest_apk") { android_apk(target_name) { set_sources_assignment_filter([]) data_deps = [] deps = [] forward_variables_from(invoker, "*") testonly = true assert(defined(unittests_dep), "Need unittests_dep for $target_name") test_suite_name = get_label_info(unittests_dep, "name") # This trivial assert is needed in case both unittests_binary and apk_name # are defined, as otherwise test_suite_name would not be used. assert(test_suite_name != "") if (!defined(apk_name)) { apk_name = test_suite_name } if (!defined(android_manifest)) { android_manifest = "//testing/android/native_test/java/AndroidManifest.xml" } if (!defined(unittests_binary)) { unittests_binary = "lib${test_suite_name}${shlib_extension}" } final_apk_path = "$root_build_dir/${apk_name}_apk/${apk_name}-debug.apk" if (!defined(use_default_launcher) || use_default_launcher) { deps += [ "//testing/android/native_test:native_test_java" ] } native_libs = [ unittests_binary ] deps += [ "//base:base_java", "//testing/android/appurify_support:appurify_support_java", "//testing/android/reporter:reporter_java", ] data_deps += [ "//build/android/pylib/remote/device/dummy:remote_device_dummy_apk", "//tools/android/md5sum", ] if (host_os == "linux") { data_deps += [ "//tools/android/forwarder2" ] } } } # Generate .java files from .aidl files. # # This target will store the .java files in a srcjar and should be included in # an android_library or android_apk's srcjar_deps. # # Variables # sources: Paths to .aidl files to compile. # import_include: Path to directory containing .java files imported by the # .aidl files. # interface_file: Preprocessed aidl file to import. # # Example # android_aidl("foo_aidl") { # import_include = "java/src" # sources = [ # "java/src/com/foo/bar/FooBarService.aidl", # "java/src/com/foo/bar/FooBarServiceCallback.aidl", # ] # } template("android_aidl") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) srcjar_path = "${target_gen_dir}/${target_name}.srcjar" aidl_path = "${android_sdk_build_tools}/aidl" framework_aidl = "$android_sdk/framework.aidl" action(target_name) { script = "//build/android/gyp/aidl.py" sources = invoker.sources imports = [ framework_aidl ] if (defined(invoker.interface_file)) { assert(invoker.interface_file != "") imports += [ invoker.interface_file ] } inputs = [ aidl_path ] + imports depfile = "${target_gen_dir}/${target_name}.d" outputs = [ depfile, srcjar_path, ] rebased_imports = rebase_path(imports, root_build_dir) args = [ "--depfile", rebase_path(depfile, root_build_dir), "--aidl-path", rebase_path(aidl_path, root_build_dir), "--imports=$rebased_imports", "--srcjar", rebase_path(srcjar_path, root_build_dir), ] if (defined(invoker.import_include) && invoker.import_include != "") { # TODO(cjhopman): aidl supports creating a depfile. We should be able to # switch to constructing a depfile for the overall action from that # instead of having all the .java files in the include paths as inputs. rebased_import_includes = rebase_path([ invoker.import_include ], root_build_dir) args += [ "--includes=$rebased_import_includes" ] _java_files_build_rel = exec_script("//build/android/gyp/find.py", rebase_path([ invoker.import_include ], root_build_dir), "list lines") _java_files = rebase_path(_java_files_build_rel, ".", root_build_dir) inputs += _java_files } args += rebase_path(sources, root_build_dir) } } # Creates a dist directory for a native executable. # # Running a native executable on a device requires all the shared library # dependencies of that executable. To make it easier to install and run such an # executable, this will create a directory containing the native exe and all # it's library dependencies. # # Note: It's usually better to package things as an APK than as a native # executable. # # Variables # dist_dir: Directory for the exe and libraries. Everything in this directory # will be deleted before copying in the exe and libraries. # binary: Path to (stripped) executable. # extra_files: List of extra files to copy in (optional). # # Example # create_native_executable_dist("foo_dist") { # dist_dir = "$root_build_dir/foo_dist" # binary = "$root_build_dir/foo" # deps = [ ":the_thing_that_makes_foo" ] # } template("create_native_executable_dist") { forward_variables_from(invoker, [ "testonly" ]) _libraries_list = "${target_gen_dir}/${target_name}_library_dependencies.list" _find_deps_target_name = "${target_name}__find_library_dependencies" # TODO(agrieve): Extract dependent libs from GN rather than readelf. action(_find_deps_target_name) { forward_variables_from(invoker, [ "deps" ]) script = "//build/android/gyp/write_ordered_libraries.py" depfile = "$target_gen_dir/$target_name.d" inputs = [ invoker.binary, android_readelf, ] outputs = [ depfile, _libraries_list, ] rebased_binaries = rebase_path([ invoker.binary ], root_build_dir) args = [ "--depfile", rebase_path(depfile, root_build_dir), "--input-libraries=$rebased_binaries", "--libraries-dir", rebase_path(root_shlib_dir, root_build_dir), "--output", rebase_path(_libraries_list, root_build_dir), "--readelf", rebase_path(android_readelf, root_build_dir), ] } copy_ex(target_name) { clear_dir = true inputs = [ _libraries_list, invoker.binary, ] dest = invoker.dist_dir data = [ "${invoker.dist_dir}/", ] _rebased_libraries_list = rebase_path(_libraries_list, root_build_dir) _rebased_binaries_list = rebase_path([ invoker.binary ], root_build_dir) args = [ "--files=@FileArg($_rebased_libraries_list:lib_paths)", "--files=$_rebased_binaries_list", ] if (defined(invoker.extra_files)) { _rebased_extra_files = rebase_path(invoker.extra_files, root_build_dir) args += [ "--files=$_rebased_extra_files" ] } deps = [ ":$_find_deps_target_name", ] if (defined(invoker.deps)) { deps += invoker.deps } } } # Compile a protocol buffer to java. # # This generates java files from protocol buffers and creates an Android library # containing the classes. # # Variables # sources: Paths to .proto files to compile. # proto_path: Root directory of .proto files. # # Example: # proto_java_library("foo_proto_java") { # proto_path = "src/foo" # sources = [ "$proto_path/foo.proto" ] # } template("proto_java_library") { set_sources_assignment_filter([]) forward_variables_from(invoker, [ "testonly" ]) _protoc_dep = "//third_party/android_protobuf:android_protoc($host_toolchain)" _protoc_out_dir = get_label_info(_protoc_dep, "root_out_dir") _protoc_bin = "$_protoc_out_dir/android_protoc" _proto_path = invoker.proto_path _template_name = target_name action("${_template_name}__protoc_java") { srcjar_path = "$target_gen_dir/$target_name.srcjar" script = "//build/protoc_java.py" deps = [ _protoc_dep, ] if (defined(invoker.deps)) { deps += invoker.deps } sources = invoker.sources depfile = "$target_gen_dir/$target_name.d" outputs = [ depfile, srcjar_path, ] args = [ "--depfile", rebase_path(depfile, root_build_dir), "--protoc", rebase_path(_protoc_bin, root_build_dir), "--proto-path", rebase_path(_proto_path, root_build_dir), "--srcjar", rebase_path(srcjar_path, root_build_dir), ] + rebase_path(sources, root_build_dir) } android_library(target_name) { chromium_code = false java_files = [] srcjar_deps = [ ":${_template_name}__protoc_java" ] deps = [ "//third_party/android_protobuf:protobuf_nano_javalib", ] } } # Writes a script to root_out_dir/bin that passes --output-directory to the # wrapped script, in addition to forwarding arguments. Most / all of these # wrappers should be made deps of //tools/android:android_tools. # # Variables # target: Script to wrap. # flag_name: Default is "--output-directory" # # Example # wrapper_script("foo_wrapper") { # target = "//pkg/foo.py" # } template("wrapper_script") { action(target_name) { _name = get_path_info(invoker.target, "name") _output = "$root_out_dir/bin/$_name" script = "//build/android/gyp/create_tool_wrapper.py" outputs = [ _output, ] # The target isn't actually used by the script, but it's nice to have GN # check that it exists. inputs = [ invoker.target, ] args = [ "--output", rebase_path(_output, root_build_dir), "--target", rebase_path(invoker.target, root_build_dir), "--output-directory", rebase_path(root_out_dir, root_build_dir), ] if (defined(invoker.flag_name)) { args += [ "--flag-name=${invoker.flag_name}" ] } } }