blob: 64c0bc081a39f9bec22b0f78a0ff25700f9a8a95 [file] [log] [blame]
# Copyright 2020 The Fuchsia Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
# Declare data files to be accessible at runtime on the target device.
#
# A resource() target looks just like a copy() target but $outputs are
# relative target paths. Using $data_deps to this resource() target in
# each target whose code uses $outputs at runtime ensures that the files
# will be present on the system.
#
# If the `sources` list contains more than one file, the `outputs` should use
# source expansion template placeholders, such as `{{source_file_part}}`.
#
# For example:
#
# //some/dir/BUILD.gn:
# resource("testdata") {
# sources = [
# "testdata/input.json",
# "testdata/input_minified.json",
# ]
# outputs = [ "data/{{source_file_part}}" ]
# }
#
# The above `resource` target maps files in the subdirectory `testdata` to
# destination paths in a Fuchsia package as follows:
#
# //some/dir/testdata/input.json --> data/input.json
# //some/dir/testdata/input_minified.json --> data/input_minified.json
#
# For more details on using template placeholders, see the GN documentation:
#
# https://gn.googlesource.com/gn/+/HEAD/docs/reference.md#placeholders
#
# If the file is not in the source tree, it should be generated by another
# target in the build listed in $deps. If that would be a generated_file()
# target, then use generated_resource() instead of resource().
#
# Using resource() to install an ELF binary file into bin/ or test/ will
# break instrumented builds (with an error that only occurs very late in
# the build, during the packaging step). To solve this, use renamed_binary()
# to install them instead.
#
# Due to this, using resource() with an output directory of bin/ or test/
# is an error by default, but can be overridden for data files by setting
# allow_binary_output_dir to true!
#
# Parameters
#
# data_deps
# Optional: Additional dependencies for the runtime image. These are
# included in the image if this target is, but are not related to the
# $sources list.
# Type: list(label)
#
# deps
# Optional: Targets that produce $sources. Any files listed in
# $sources that are produced by the build should be produced by a
# target listed here. This is the only thing that guarantees those
# files will have been built by the time the image is being packed.
# Targets reached only via this $deps list will *not* contribute their
# own contents to the image directly. For that, list them in $data_deps.
# Targets listed here are used only to produce the $sources files.
# Type: list(label)
#
# outputs
# Required: List of one runtime path. This must be a relative path (no
# leading `/`). It can use placeholders based on $sources; see copy()
# and `gn help source_expansion`. When this resource() target is in
# the dependency graph of a zbi() target, then this is the path within
# the BOOTFS, which appears at /boot in the namespace of early-boot and
# standalone Zircon processes.
# Type: list(path)
#
# sources
# Required: List of files in the source tree or build that become $outputs.
# See copy() for details.
#
# Note: if sources are generated by an action then the resource() target
# should be a direct dependency of a fuchsia_package() or of a
# fuchsia_component() target. Do not add as an indirect dependency via an
# executable() target.
# Do this:
# fuchsia_package()/fuchsia_component() -> resource() -> action()
# Don't do this:
# fuchsia_package()/fuchsia_component() -> executable() -> resource() -> action()
#
# Type: list(file)
#
# allow_binary_output_dir
# Optional: Set to true to allow copying a resource into either
# bin/ or test/. Useful when installing non-ELF files there, while
# ELF files should be installed using renamed_binary() instead.
# Type: boolean
#
# See copy() for other parameters.
template("resource") {
if (invoker.sources != []) {
_label = get_label_info(":$target_name", "label_with_toolchain")
}
# Prevent binaries to be copied to bin/ or test/ with resource() to prevent
# build breaks with instrumented builds. See https://fxbug.dev/89436
#
_allow_binary_output_dir = defined(invoker.allow_binary_output_dir) &&
invoker.allow_binary_output_dir
if (!_allow_binary_output_dir) {
_outputs = invoker.outputs
# invoker.outputs can be a list or a string. If it is a string, turn
# it into a list here.
if ("${_outputs}" == _outputs) {
_outputs = [ _outputs ]
} else {
assert(_outputs == [ _outputs[0] ],
"outputs should be a string or a list of one string!")
}
_output_dir = get_path_info(_outputs[0], "dir")
assert(
_output_dir != "bin" && _output_dir != "test",
"Installing file to bin/ or test/ requires renamed_binary() or allow_binary_output_dir!")
}
group(target_name) {
forward_variables_from(invoker,
"*",
[
"metadata",
"outputs",
"sources",
])
metadata = {
# Used by the distribution_manifest() template.
distribution_entries_barrier = []
distribution_entries = []
# Used by the zbi() template.
zbi_input_barrier = []
if (defined(invoker.metadata)) {
forward_variables_from(invoker.metadata, "*")
}
# Stop *_manifest() and zbi_test() from picking up files or
# zbi_input() items from the deps, but let them reach the data_deps.
if (defined(data_deps)) {
distribution_entries_barrier += data_deps
zbi_input_barrier += data_deps
}
foreach(source, invoker.sources) {
foreach(target, process_file_template([ source ], invoker.outputs)) {
assert(rebase_path(target, "foo") != target,
"`outputs` in resource() cannot start with /")
distribution_entries += [
{
source = rebase_path(source, root_build_dir)
destination = target
label = _label
},
]
}
}
}
}
}
# Declare data files to be accessible at runtime on the target device.
#
# Similar to resource() but with a different interface that allows grouping
# multiple files into a single target when their packaged paths don't follow
# a common pattern.
#
# Parameters
#
# data_deps
# Optional: Additional dependencies for the runtime image. These are
# included in the image if this target is, but are not related to the
# $sources list.
# Type: list(label)
#
# deps
# Optional: Targets that produce $sources. Any files listed in
# $sources that are produced by the build should be produced by a
# target listed here. This is the only thing that guarantees those
# files will have been built by the time the image is being packed.
# Targets reached only via this $deps list will *not* contribute their
# own contents to the image directly. For that, list them in $data_deps.
# Targets listed here are used only to produce the $sources files.
# Type: list(label)
#
# files
# Required: List of scopes containing `source` and `dest` paths.
# `source` paths are local file paths.
# `dest` paths are packaged paths.
# For instance:
# files = [
# {
# source = "//path/to/file.txt"
# dest = "data/first.txt"
# },
# {
# source = "//path/to/other_file.txt",
# dest = "data/second.txt"
# },
# ]
# Type: list(scope)
#
# allow_binary_output_dir
# See resource()
#
# testonly, visibility
template("resource_group") {
if (invoker.files != []) {
_label = get_label_info(":$target_name", "label_with_toolchain")
}
if (!(defined(invoker.allow_binary_output_dir) &&
invoker.allow_binary_output_dir)) {
# Prevent binaries to be copied to bin/ or test/ with resource() to prevent
# build breaks with instrumented builds. See https://fxbug.dev/89436
#
# NOTE: This check cannot be performed in the metadata definition below
# because GN will complain that the _dest_with_prefix temporary variable
# is not a list (metadata scope variables can only be lists).
foreach(file, invoker.files) {
_dest_dir = get_path_info(file.dest, "dir")
assert(
_dest_dir != "bin" && _dest_dir != "test",
"Installing file to bin/ or test/ requires renamed_binary() or allow_binary_output_dir!")
}
}
group(target_name) {
forward_variables_from(invoker,
[
"deps",
"testonly",
"visibility",
])
metadata = {
# Used by the distribution_manifest() template.
distribution_entries_barrier = []
distribution_entries = []
# Used by the zbi() template.
zbi_input_barrier = []
if (defined(invoker.metadata)) {
forward_variables_from(invoker.metadata, "*")
}
# Stop *_manifest() and zbi_test() from picking up files or
# zbi_input() items from the deps, but let them reach the data_deps.
if (defined(data_deps)) {
distribution_entries_barrier += data_deps
zbi_input_barrier += data_deps
}
foreach(file, invoker.files) {
distribution_entries += [
{
source = rebase_path(file.source, root_build_dir)
destination = file.dest
label = _label
},
]
}
}
}
}
# resource_tree() is similar to resource() but makes it easy to replicate
# a tree of sources files, relative to a given sources path prefix, to a
# given destination directory.
#
# For example:
#
# //some/dir/BUILD.gn:
# resource_tree("my-resources") {
# sources_root = "res"
# sources = [
# foo.resource",
# bar/bar-1.resource",
# ]
# dest_dir = "data/resources"
# }
#
# Will declare the following installation requirements:
#
# //some/dir/res/foo.resource --> data/resources/foo.resource
# //some/dir/res/bar/bar.resource --> data/resources/bar/bar.resource
#
# This is difficult to do with resource() because GN source expansion
# cannot preserve the original sources input paths.
#
# This template is useful to avoid calling resource() multiple times
# in a loop when dealing with resource files laid out into different
# sub-directories. Note that the files cannot be renamed though!
#
# Parameters
#
# source_root
# Optional: A path prefix that is prepended to all items in the 'sources'
# list. If not specified, the current target's directory is used.
# Type: string(path)
#
# sources
# Required: List of files in the source tree or build that will be
# installed. Each 'file' item in this list is installed to
# '$dest_dir/$file', and its content taken from '$sources_root/$file'.
# Note that unlike resource(), there is no way to transform or expand
# source paths.
# Type: list(file)
#
# dest_dir
# Required: Destination path where all sources are installed.
# Cannot start with a "/". Use an empty string to install files directory
# to the package's top-level directory.
# Type: string(path)
#
# data_deps
# Optional: Additional dependencies for the runtime image. These are
# included in the image if this target is, but are not related to the
# $sources list.
# Type: list(label)
#
# allow_binary_output_dir
# See resource()
#
# deps
# Optional: Targets that produce $sources. Any files listed in
# $sources that are produced by the build should be produced by a
# target listed here. This is the only thing that guarantees those
# files will have been built by the time the image is being packed.
# Targets reached only via this $deps list will *not* contribute their
# own contents to the image directly. For that, list them in $data_deps.
# Targets listed here are used only to produce the $sources files.
# Type: list(label)
#
template("resource_tree") {
assert(defined(invoker.sources), "sources is required!")
assert(defined(invoker.dest_dir), "dest_dir is required!")
dest_dir = invoker.dest_dir
if (dest_dir != "") {
# Sanitize dest_dir and append a directory separator.
assert(rebase_path(dest_dir, "foo") != dest_dir,
"dest_dir cannot start with /: $dest_dir")
assert(dest_dir != "." && dest_dir != ".." &&
string_replace(dest_dir, "./", "") == dest_dir,
"dest_dir cannot contain . or .. path elements!: $dest_dir")
if (!(defined(invoker.allow_binary_output_dir) &&
invoker.allow_binary_output_dir)) {
assert(
dest_dir != "bin" && dest_dir != "test",
"Installing file to bin/ or test/ requires renamed_binary() or allow_binary_output_dir!")
}
dest_dir = string_replace(dest_dir + "/", "//", "/")
}
if (defined(invoker.sources_root)) {
assert(invoker.sources_root != "", "sources_root cannot be empty!")
sources_prefix = invoker.sources_root
# Append trailing separator if needed
if (string_replace(sources_prefix + "###", "/###", "") ==
sources_prefix + "###") {
sources_prefix += "/"
}
} else {
sources_prefix = ""
}
target_label = get_label_info(":$target_name", "label_with_toolchain")
if (invoker.sources == []) {
# Support resource_tree() targets with empty sources list.
not_needed(invoker,
[
"dest_dir",
"sources_root",
])
not_needed([
"sources_prefix",
"dest_dir",
"target_label",
])
}
group(target_name) {
forward_variables_from(invoker,
"*",
[
"dest_dir",
"metadata",
"sources",
"sources_root",
])
metadata = {
# Used by the distribution_manifest() template.
distribution_entries_barrier = []
distribution_entries = []
# Used by the zbi() template.
zbi_input_barrier = []
if (defined(invoker.metadata)) {
forward_variables_from(invoker.metadata, "*")
}
# Stop *_manifest() and zbi_test() from picking up files or
# zbi_input() items from the deps, but let them reach the data_deps.
if (defined(data_deps)) {
distribution_entries_barrier += data_deps
zbi_input_barrier += data_deps
}
foreach(_source, invoker.sources) {
distribution_entries += [
{
source = rebase_path(sources_prefix + _source, root_build_dir)
destination = dest_dir + _source
label = target_label
},
]
}
}
}
}