blob: af540c2b42e6050b837cfe3ddf9e8bed32d8d5b0 [file] [log] [blame]
# Copyright (C) 2017 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import("perfetto.gni")
import("perfetto_component.gni")
# This gni file defines rules for proto generation. There are various types of
# proto targets that can be defined in our codebase:
# "lite" targets: these use the standard libprotobuf library. They are used
# mainly for tests and readback.
# "zero" targets: these use the protozero library and its protoc plugin. They
# are used pretty much everywhere.
# "descriptor" targets: they are used to generate a proto-encoded reflection
# descriptor that describes the schema of the proto using protobuf itself.
# All these targets can be generated using the perfetto_proto_library rule. It
# wraps the instantiation of several proto targets using a convenience template.
#
# For instance:
# perfetto_proto_library("xxx_@TYPE@") {
# proto_generators = [ "lite", "zero" ] # lite+zero+cpp is the default value.
# sources = [ "one.proto", "two.proto" ]
# deps = [ "dep:@TYPE@" ]
# }
#
# Is the equivalent of:
# proto_library("xxx_lite") { sources = [...], deps = [ "dep:lite"] }
# protozero_library("xxx_zero") { sources = [...], deps = [ "dep:zero"] }
# Load the protobuf's proto_library() definition.
if (!defined(perfetto_protobuf_target_prefix)) {
if (perfetto_root_path == "//") {
perfetto_protobuf_target_prefix = "//buildtools"
} else {
perfetto_protobuf_target_prefix = "//third_party/protobuf"
}
}
if (!defined(perfetto_protobuf_gni)) {
if (perfetto_root_path == "//") {
perfetto_protobuf_gni = "//gn/standalone/proto_library.gni"
} else {
perfetto_protobuf_gni = "//third_party/protobuf/proto_library.gni"
}
}
if (!defined(perfetto_protobuf_src_dir)) {
if (perfetto_root_path == "//") {
perfetto_protobuf_src_dir = "//buildtools/protobuf/src"
} else {
perfetto_protobuf_src_dir = "//third_party/protobuf/src"
}
}
import(perfetto_protobuf_gni)
# Equivalent to proto_library (generation of .h/.cc from .proto files) but
# enables also generation using the protozero plugin.
# The generated files will have the .pbzero.{cc,h} suffix, as opposed to the
# .pb.{cc,h} of the official proto library.
# DO NOT use this target directly, use perfetto_proto_library() below.
template("protozero_library") {
proto_library(target_name) {
perfetto_root_path = invoker.perfetto_root_path
generate_cc = false
generate_python = false
generator_plugin_label =
perfetto_root_path + "src/protozero/protoc_plugin:protozero_plugin"
generator_plugin_suffix = ".pbzero"
if (build_with_chromium) {
component_build_force_source_set = true
}
# Convert deps to link_deps: the proto_library rule requires that C++ files
# are passed in as link_deps wheras in Perfetto, we always works with deps.
link_deps = []
if (defined(invoker.deps)) {
link_deps += invoker.deps
}
# omit_protozero_dep is intended to be used when protozero_library
# is used in Chrome (for generation of code for proto extensions)
# to avoid ODR violations in case of component builds. The embedder
# (Chrome) is then responsible for adding the appropriate transitive
# dependency on Protozero.
#
# TODO(b/173041866): use fine-grained components instead when available
if (!(defined(invoker.omit_protozero_dep) && invoker.omit_protozero_dep)) {
link_deps += [ perfetto_root_path + "src/protozero" ]
}
if (defined(invoker.link_deps)) {
link_deps += invoker.link_deps
}
forward_variables_from(invoker,
[
"defines",
"generator_plugin_options",
"include_dirs",
"proto_data_sources",
"proto_deps",
"proto_in_dir",
"proto_out_dir",
"sources",
"testonly",
"visibility",
"generate_descriptor",
"propagate_imports_configs",
"import_dirs",
])
}
}
# This template generates .gen.cc/h files from .proto files. The generated
# sources are actual C++ classes that can be moved and copied around, very
# similar to the libprotobuf generated ones API-wise, but use protozero under
# the hoods, without any zero-copy benefit though.
# They are mainly used for the perfetto IPC layer and tests.
template("protozero_cpp_library") {
proto_library(target_name) {
perfetto_root_path = invoker.perfetto_root_path
generate_cc = false
generate_python = false
generator_plugin_label =
perfetto_root_path + "src/protozero/protoc_plugin:cppgen_plugin"
generator_plugin_suffix = ".gen"
if (build_with_chromium) {
component_build_force_source_set = true
}
link_deps = [
"$perfetto_root_path/gn:default_deps",
"$perfetto_root_path/include/perfetto/base",
"$perfetto_root_path/src/protozero",
]
# Convert deps to link_deps: the proto_library rule requires that C++ files
# are passed in as link_deps wheras in Perfetto, we always works with deps.
if (defined(invoker.deps)) {
link_deps += invoker.deps
}
forward_variables_from(invoker,
[
"defines",
"generator_plugin_options",
"include_dirs",
"proto_deps",
"proto_in_dir",
"proto_out_dir",
"sources",
"testonly",
"visibility",
"generate_descriptor",
"propagate_imports_configs",
"import_dirs",
])
}
}
# Generates .ipc.{h,cc} stubs for IPC services defined in .proto files.
template("ipc_library") {
proto_library(target_name) {
perfetto_root_path = invoker.perfetto_root_path
generate_cc = false
generate_python = false
generator_plugin_label =
"$perfetto_root_path/src/ipc/protoc_plugin:ipc_plugin"
generator_plugin_suffix = ".ipc"
link_deps = [ "$perfetto_root_path/gn:default_deps" ]
if (perfetto_component_type == "static_library") {
link_deps += [ "$perfetto_root_path/src/ipc:perfetto_ipc" ]
} else {
link_deps += [ "$perfetto_root_path/src/ipc:common" ]
}
if (is_win) {
# TODO(primiano): investigate this. In Windows standalone builds, some
# executable targets end up in a state where no code pulls a dep on the
# ipc:client (in turn that seems a subtle consequence of not having
# traced_probes on Windows). This dep here is effectively needed because
# the client-side code in the generated .ipc.cc effectively depends on the
# client-side IPC library. Perhaps we just should do this unconditionally
# on all platforms?
if (perfetto_component_type == "static_library") {
link_deps += [ "$perfetto_root_path/src/ipc:perfetto_ipc" ]
} else {
link_deps += [ "$perfetto_root_path/src/ipc:client" ]
}
}
# Convert deps to link_deps: the proto_library rule requires that C++ files
# are passed in as link_deps wheras in Perfetto, we always works with deps.
if (defined(invoker.deps)) {
link_deps += invoker.deps
}
forward_variables_from(invoker,
[
"defines",
"extra_configs",
"include_dirs",
"proto_deps",
"proto_in_dir",
"proto_out_dir",
"generator_plugin_options",
"sources",
"testonly",
"visibility",
"propagate_imports_configs",
"import_dirs",
])
}
}
# Generates .grpc.{h,cc} stubs for services defined in .proto files.
# We require explicit opt-in as gRPC is very heavyweight so we do not
# want accidental dependencies on this.
if (enable_perfetto_grpc) {
template("perfetto_grpc_library") {
proto_library(target_name) {
proto_in_dir = perfetto_root_path
proto_out_dir = perfetto_root_path
propagate_imports_configs = false
perfetto_root_path = perfetto_root_path
generate_cc = false
generate_python = false
generator_plugin_label =
"$perfetto_root_path/buildtools/grpc:grpc_cpp_plugin"
generator_plugin_suffix = ".grpc.pb"
link_deps = [ "$perfetto_root_path/buildtools/grpc:grpc++" ]
public_configs = [ "$perfetto_root_path/buildtools:grpc_gen_config" ]
# Convert deps to link_deps: the proto_library rule requires that C++
# files are passed in as link_deps wheras in Perfetto, we always works
# with deps.
if (defined(invoker.deps)) {
link_deps += invoker.deps
}
forward_variables_from(invoker,
[
"defines",
"extra_configs",
"include_dirs",
"sources",
"testonly",
"visibility",
])
}
}
}
# The template used everywhere in the codebase.
template("perfetto_proto_library") {
if (defined(invoker.proto_generators)) {
proto_generators = invoker.proto_generators
} else {
proto_generators = [
"zero",
"lite",
"cpp",
]
}
# proto imports and C++ #includes are relative to this path.
if (defined(invoker.proto_path)) {
proto_path = invoker.proto_path
} else {
proto_path = perfetto_root_path
}
if (defined(invoker.import_dirs)) {
import_dirs_ = invoker.import_dirs
} else {
import_dirs_ = []
}
expansion_token = "@TYPE@"
# The source set target should always be generated as it is used by the
# build generators and for generating descriptors.
source_set_target_name =
string_replace(target_name, expansion_token, "source_set")
# This config is necessary for Chrome proto_library build rule to work
# correctly.
source_set_input_config_name = "${source_set_target_name}_input_config"
config(source_set_input_config_name) {
inputs = invoker.sources
}
group(source_set_target_name) {
# To propagate indirect inputs dependencies to descendant tareget, we use
# public_deps and public_configs in this target.
public_deps = []
exports_ = []
if (defined(invoker.public_deps)) {
foreach(dep, invoker.public_deps) {
# Get the absolute target path
mapped_dep = string_replace(dep, expansion_token, "source_set")
public_deps += [ mapped_dep ]
exports_ += [ get_label_info(mapped_dep, "dir") + ":" +
get_label_info(mapped_dep, "name") ]
}
}
if (defined(invoker.deps)) {
foreach(dep, invoker.deps) {
mapped_dep = string_replace(dep, expansion_token, "source_set")
public_deps += [ mapped_dep ]
}
}
sources = []
foreach(source, invoker.sources) {
sources += [ get_path_info(source, "abspath") ]
}
public_configs = [ ":${source_set_input_config_name}" ]
metadata = {
proto_library_sources = sources
proto_import_dirs = import_dirs_
exports = exports_
}
}
# Generate the descriptor if the option is set.
if (defined(invoker.generate_descriptor)) {
target_name_ = string_replace(target_name, expansion_token, "descriptor")
proto_library(target_name_) {
proto_in_dir = proto_path
proto_out_dir = proto_path
generate_python = false
generate_cc = false
generate_descriptor = rebase_path(invoker.generate_descriptor, proto_path)
sources = [ invoker.descriptor_root_source ]
import_dirs = import_dirs_
deps = [ ":${source_set_target_name}" ]
forward_variables_from(invoker,
[
"visibility",
"testonly",
"exclude_imports",
])
}
}
foreach(gen_type, proto_generators) {
target_name_ = string_replace(target_name, expansion_token, gen_type)
# Translate deps from xxx:@TYPE@ to xxx:lite/zero.
all_deps_ = []
if (defined(invoker.deps)) {
foreach(dep, invoker.deps) {
all_deps_ += [ string_replace(dep, expansion_token, gen_type) ]
}
}
# The distinction between deps and public_deps does not matter for GN
# but Bazel cares about the difference so we distinguish between the two.
if (defined(invoker.public_deps)) {
foreach(dep, invoker.public_deps) {
all_deps_ += [ string_replace(dep, expansion_token, gen_type) ]
}
}
# gn:public_config propagates the gen dir as include directory. We
# disable the proto_library's public_config to avoid duplicate include
# directory command line flags (crbug.com/1043279, crbug.com/gn/142).
propagate_imports_configs_ = false
vars_to_forward = []
vars_to_forward += [
"sources",
"visibility",
"testonly",
]
if (gen_type == "zero") {
protozero_library(target_name_) {
proto_in_dir = proto_path
proto_out_dir = proto_path
generator_plugin_options = "wrapper_namespace=pbzero"
deps = all_deps_
proto_deps = [ ":$source_set_target_name" ]
propagate_imports_configs = propagate_imports_configs_
import_dirs = import_dirs_
forward_variables_from(invoker, vars_to_forward)
}
} else if (gen_type == "cpp") {
protozero_cpp_library(target_name_) {
proto_in_dir = proto_path
proto_out_dir = proto_path
generator_plugin_options = "wrapper_namespace=gen"
deps = all_deps_
proto_deps = [ ":$source_set_target_name" ]
propagate_imports_configs = propagate_imports_configs_
import_dirs = import_dirs_
forward_variables_from(invoker, vars_to_forward)
}
} else if (gen_type == "ipc") {
cpp_target_name_ = string_replace(target_name, expansion_token, "cpp")
ipc_library(target_name_) {
proto_in_dir = proto_path
proto_out_dir = proto_path
generator_plugin_options = "wrapper_namespace=gen"
proto_deps = [ ":$source_set_target_name" ]
deps = all_deps_ + [ ":$cpp_target_name_" ]
propagate_imports_configs = propagate_imports_configs_
import_dirs = import_dirs_
forward_variables_from(invoker, vars_to_forward)
}
} else if (gen_type == "lite") {
proto_library(target_name_) {
proto_in_dir = proto_path
proto_out_dir = proto_path
generate_python = false
link_deps = all_deps_
cc_generator_options = "lite=true:"
propagate_imports_configs = propagate_imports_configs_
import_dirs = import_dirs_
proto_deps = [ ":${source_set_target_name}" ]
forward_variables_from(invoker, vars_to_forward)
}
} else {
assert(false, "Invalid 'proto_generators' value.")
}
}
}