blob: 995a6f7f60d98eb63cf740ad8967d0f5f3ea05bb [file] [log] [blame] [edit]
# 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.
import("//build/cpp/sdk_shared_library.gni")
import("//build/cpp/sdk_source_set.gni")
import("//build/cpp/sdk_static_library.gni")
import("//build/unification/global_variables.gni")
# Build a zircon-specific library, possibly in several ways.
#
# This template should be used by Zircon build rules exclusively, and follows
# most of the conventions of the Zircon build `library()` template, which
# differ significantly from those of `library()` in the Fuchsia build.
#
# IMPORTANT NOTE: Due to historical reasons, its behaviour is very different,
# depending on the value of the `is_kernel` global variable.
#
# * If `is_kernel` is true, which means when building part of the kernel:
# The template will assert if the `kernel` argument is not set to true.
# It will create a source_set() if building part of the kernel, or
# a static_library() for any other Zircon artefact. The values of `sdk`,
# `sdk_publishable` and `sdk_headers` will be ignored entirely.
#
# This will also create a :headers sub-target, a group used to give access
# to the library's include directory from any dependent. Note that this
# does *not* make the library's headers, listed in `sdk_headers` public.
#
# This behaviour is needed to replicate the Zircon build's behaviour when
# compiling Zircon artefacts.
#
# * If `is`_kernel` is false, which means that the library is built either as a
# Fuchsia user binary or a host binary, then the `kernel` parameter is
# ignored, and `sdk` must be set to determine the shape of the library,
# as well as `sdk_headers to list the library's public headers, relative to
# its `include` sub-directory.
#
# If `sdk_publishable` is true, the library will be part of the Fuchsia SDK.
#
# Any :headers sub-targets that appear in public_deps will be rewritten into
# a dependency to the library itself, e.g.:
#
# public_deps = [ "//zircon/system/ulib/foo:headers" ]
#
# will be replaced by:
#
# public_deps = [ "//zircon/system/ulib/foo" ]
#
# Because the Fuchsia build doesn't support :headers sub-targets in
# non-kernel libraries for several technical reasons.
#
# Similarly, any deps or public_deps entry to
# //zircon/system/ulib/zircon will be replaced by the proper dependency
# required to access Zircon headers and, for Fuchsia user binaries, the
# link-time VDSO. This feature will be removed once the VDSO is built
# with the Fuchsia build.
#
# Parameters
#
# kernel
# Optional: True if this library can be used in the kernel.
# Type: bool
# Default: false
#
# sdk
# Optional: A value that indicates whether to build this library as a
# source set, static library or shared library with the Fuchsia build.
# This is required, except if the global `is_kernel` is true. Note that
# this name is confusing for historical reasons, because using
# this parameter does not make this library exported to the Fuchsia SDK
# (see `sdk_publishable` below for this).
# Type: "static" or "shared" or "source"
#
# sdk_publishable
# Optional: Indicates that this library can be added to the Fuchsia SDK.
# Type: bool
# Default: false
#
# sdk_headers
# Optional: Required, except if the global `is_kernel` is true.
# This must list all the public header files in the library's `include/`
# directory; names should be relative to `include/`.
# Type: list(string)
#
# sdk_name
# Optional: Name under which the library is published. This can be used
# when $target_name is already used by another target in the Fuchsia build.
# Type: string
# Default: target_name
#
# See source_set() for other parameters.
#
# The following parameters are ignored by this implementation. They are here
# to support zx_library() calls that are shared by BUILD.zircon.gn and BUILD.gn
# file. Support for these may be removed after build unification completes.
#
# host
# Optional: Was set to true if this library can be used in
# host-side code, e.g. to be a dependency of a host_tool().
# Type: bool
# Default: false
#
# static
# Optional: True if this library can be statically linked in user code.
# Type: bool
# Default: !$kernel
#
# shared
# Optional: True if this library can be made a shared library for user code.
# Type: bool
# Default: false
#
# sdk_migrated
# Optional: Indicates that this library is built directly by the Fuchsia
# build, and no longer needs to be exported as a legacy target. This is
# used to ease build unification, and will disappear once it is completed.
# Type: bool
# Default: false
#
template("zx_library") {
template_params = [
"host",
"kernel",
"sdk",
"sdk_headers",
"sdk_publishable",
"shared",
"static",
]
# `sdk_migrated` is only used by the Zircon build system.
not_needed(invoker, [ "sdk_migrated" ])
if (zircon_toolchain != false) {
_library_name = target_name
assert(
!is_kernel || (defined(invoker.kernel) && invoker.kernel),
"This zx_library() cannot be built with a Zircon-specific toolchain! " +
"Consider adding `kernel = true` to its declaration.")
if (defined(invoker.shared) && invoker.shared) {
kernel_library_target_type = "shared_library"
} else if (toolchain.environment == "kernel") {
# In the kernel proper, zx_library() is always a source_set().
# Everything goes into the kernel and anything unused gets linker GC.
kernel_library_target_type = "source_set"
} else {
kernel_library_target_type = "static_library"
}
target(kernel_library_target_type, _library_name) {
if (kernel_library_target_type == "static_library") {
complete_static_lib = true
}
forward_variables_from(invoker, "*", template_params)
if (!defined(public_deps)) {
public_deps = []
}
public_deps += [ ":headers" ]
}
# If the library has the same name as its directory, just create
# :headers and :headers.config sub-targets. Otherwise, create
# :foo.headers and :foo.headers.config
if (get_label_info(":$_library_name", "name") ==
get_path_info(get_label_info(":$_library_name", "dir"), "file")) {
_headers_target = "headers"
_headers_config_target = "headers.config"
} else {
_headers_target = "$_library_name.headers"
_headers_config_target = "$_library_name.headers.config"
}
group(_headers_target) {
# Used to represent header dependencies.
# Direct us of public_configs should be rare but is sometimes needed.
forward_variables_from(invoker,
[
"public_configs",
"public_deps",
"testonly",
"visibility",
])
if (!defined(public_configs)) {
public_configs = []
}
public_configs += [ ":${_headers_config_target}" ]
}
config(_headers_config_target) {
include_dirs = [ "include" ]
}
} else {
# The following parameters are just ignored, as in the GN build the shape of
# a library is driven by the value of the "sdk" parameter.
not_needed(invoker,
[
"kernel",
"host",
"static",
"shared",
])
assert(
defined(invoker.sdk),
"The `sdk` argument is needed to build a zx_library() with a Fuchsia or host toolchain")
shape = invoker.sdk
sdkable = defined(invoker.sdk_publishable) && invoker.sdk_publishable
extra_target_args = {
}
if (shape == "source") {
if (sdkable) {
target_type = "sdk_source_set"
extra_target_args = {
category = "partner"
api = "//sdk/lib/$target_name/$target_name.api"
sdk_name = target_name
build_as_static = true
}
} else {
target_type = "static_library"
}
} else if (shape == "static") {
if (sdkable) {
target_type = "sdk_static_library"
extra_target_args = {
category = "partner"
api = "//sdk/lib/$target_name/$target_name.api"
sdk_name = target_name
libcxx_linkage = "static"
}
} else {
target_type = "static_library"
}
} else if (shape == "shared") {
if (sdkable) {
target_type = "sdk_shared_library"
extra_target_args = {
category = "partner"
api = "//sdk/lib/$target_name/$target_name.api"
symbols_api = "//sdk/lib/$target_name/$target_name.symbols.api"
sdk_name = target_name
libcxx_linkage = "static"
}
} else {
target_type = "shared_library"
}
} else {
assert(false, "Unknown library type: $shape")
}
main_target_name = target_name
if (defined(invoker.sdk_name)) {
main_target_name = invoker.sdk_name
}
config_target_name = "$target_name.config"
config(config_target_name) {
include_dirs = [ "include" ]
}
# IMPORTANT: Rewrite header dependencies in public_deps
#
# For each zx_library(), the Zircon build used to produce a 'headers'
# group that adds a 'headers.config' public config. This is done to allow
# other targets to depend on the library's headers, but not the library
# itself.
#
# In practice, this means it is common for targets to list these headers
# groups in their public_deps, as in:
#
# public_deps = [ "$zx/system/ulib/fit:headers" ]
#
# However, these groups do not mix well with the Fuchsia build, and
# especially with sdk_source_set() or sdk_static_library().
#
# To work around this, rewrite the headers public dependencies into
# regular dependencies into the libraries themselves.
#
lib_deps = []
if (defined(invoker.deps)) {
lib_deps = invoker.deps
}
lib_public_deps = []
if (defined(invoker.public_deps)) {
foreach(dep, invoker.public_deps) {
if (get_label_info(dep, "name") == "headers") {
# Format //zircon/.../foo:headers -> //zircon/.../foo
dep = get_label_info(dep, "dir")
} else if (get_path_info(get_label_info(dep, "name"), "extension") ==
"headers") {
# Format //zircon/.../foo:bar.headers -> //zircon/.../foo:bar
dep = get_label_info(dep, "dir") + ":" +
get_path_info(get_label_info(dep, "name"), "name")
}
lib_public_deps += [ dep ]
}
}
# Remove references to a libzircon dependency, as it is provided by the
# sysroot in the present build.
lib_libs = []
if (defined(invoker.libs)) {
lib_dirs = invoker.libs
}
if (lib_deps + [ "//zircon/system/ulib/zircon" ] -
[ "//zircon/system/ulib/zircon" ] != lib_deps) {
lib_deps -= [ "//zircon/system/ulib/zircon" ]
if (is_fuchsia) {
lib_deps += [ "//src/zircon/lib/zircon" ]
} else {
lib_public_deps += [
"//zircon/system/public",
"//zircon/system/ulib/zircon:zircon-headers",
]
}
}
if (lib_public_deps + [ "//zircon/system/ulib/zircon" ] -
[ "//zircon/system/ulib/zircon" ] != lib_public_deps) {
lib_public_deps -= [ "//zircon/system/ulib/zircon" ]
if (is_fuchsia) {
lib_deps += [ "//src/zircon/lib/zircon" ]
} else {
lib_public_deps += [
"//zircon/system/public",
"//zircon/system/ulib/zircon:zircon-headers",
]
}
}
if (!is_fuchsia) {
lib_public_deps += [ "//zircon/system/public" ]
}
_has_public_headers =
defined(invoker.sdk_headers) && invoker.sdk_headers != []
target(target_type, main_target_name) {
forward_variables_from(invoker,
"*",
template_params + [
"deps",
"libs",
"public_deps",
])
deps = lib_deps
libs = lib_libs
public_deps = lib_public_deps
forward_variables_from(extra_target_args, "*")
if (_has_public_headers) {
public = []
foreach(header, invoker.sdk_headers) {
public += [ "include/$header" ]
}
} else if (defined(invoker.sdk_headers) && sdkable) {
# Add the headers to `sources` to ensure sdk_atom checks the API properly.
extra_headers = []
foreach(header, invoker.sdk_headers) {
extra_headers += [ "include/$header" ]
}
}
if (!defined(public_configs)) {
public_configs = []
}
public_configs += [ ":$config_target_name" ]
if (!defined(defines)) {
defines = []
}
defines += [ "_ALL_SOURCE" ]
}
}
}
set_defaults("zx_library") {
configs = default_common_binary_configs
}