blob: 59637ee2052636349e248f81ee547de89a955a9c [file] [log] [blame]
# Copyright 2019 The Pigweed Authors
#
# 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
#
# https://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("$dir_pw_build/python_script.gni")
# Python script that invokes protoc.
_gen_script_path =
"$dir_pw_protobuf_compiler/py/pw_protobuf_compiler/generate_protos.py"
# Generates C++ code for proto files, creating a source_set of the generated
# files. This is internal and should not be used outside of this file. Use
# pw_proto_library instead.
#
# Args:
# protos: List of input .proto files.
template("_pw_cc_proto_library") {
_proto_gen_dir = "$root_gen_dir/protos"
_outputs = process_file_template(
invoker.protos,
"$_proto_gen_dir/{{source_root_relative_dir}}/{{source_name_part}}.pb.h")
_gen_target = "${target_name}_gen"
pw_python_script(_gen_target) {
script = _gen_script_path
args = [
"--language",
"cc",
"--module-path",
"//",
"--out-dir",
_proto_gen_dir,
] + get_path_info(invoker.protos, "abspath")
inputs = invoker.protos
outputs = _outputs
if (defined(invoker.protoc_deps)) {
deps = invoker.protoc_deps
}
}
# For C++ proto files, the generated proto directory is added as an include
# path for the code. This requires using "all_dependent_configs" to force the
# include on any code that transitively depends on the generated protos.
_include_root = rebase_path(get_path_info(".", "abspath"), "//")
_include_config_target = "${target_name}_includes"
config(_include_config_target) {
include_dirs = [ "$_proto_gen_dir/$_include_root" ]
}
# Create a library with the generated source files.
# TODO(frolv): This currently only supports pw_protobuf, which is header-only.
# Figure out how to support .cc files.
source_set(target_name) {
all_dependent_configs = [ ":$_include_config_target" ]
deps = [ ":$_gen_target" ] + invoker.deps
sources = get_target_outputs(":$_gen_target")
}
}
# Generates Go code for proto files, listing the proto output directory in the
# metadata variable GOPATH. Internal use only.
#
# Args:
# protos: List of input .proto files.
template("_pw_go_proto_library") {
_proto_gopath = "$root_gen_dir/go"
_proto_gen_dir = "$_proto_gopath/src"
_rebased_gopath = rebase_path(_proto_gopath)
pw_python_script(target_name) {
metadata = {
gopath = [ "GOPATH+=$_rebased_gopath" ]
external_deps = [
"github.com/golang/protobuf/proto",
"google.golang.org/grpc",
]
}
script = _gen_script_path
args = [
"--language",
"go",
"--module-path",
"//",
"--out-dir",
_proto_gen_dir,
] + get_path_info(invoker.protos, "abspath")
inputs = invoker.protos
deps = invoker.deps
stamp = true
}
}
# Generates protobuf code from .proto definitions for various languages.
#
# The languages to generate are defined in the pw_protobuf_langs build variable.
# Each listed language creates a generated code target called
#
# <target_name>_<language>
#
# For example, with the following definitions:
#
# pw_protobuf_langs = [ "cc", "py" ]
#
# pw_proto_library("my_protos") {
# sources = [ "foo.proto" ]
# }
#
# Two build targets will be created for the declared "my_protos" target.
#
# "my_protos_cc" <-- C++ source_set containing generated proto code
# "my_protos_py" <-- Python module containing generated proto code
#
# Args:
# sources: List of input .proto files.
# deps: List of other pw_proto_library dependencies.
#
# TODO(frolv): Provide a way to set the protoc plugin for different languages.
template("pw_proto_library") {
assert(defined(invoker.sources) && invoker.sources != [],
"pw_proto_codegen requires .proto source files")
foreach(lang, pw_protobuf_langs) {
if (defined(invoker.deps)) {
_lang_deps = process_file_template(invoker.deps, "{{source}}_${lang}")
} else {
_lang_deps = []
}
_lang_target = "${target_name}_${lang}"
if (lang == "cc") {
_pw_cc_proto_library(_lang_target) {
protos = invoker.sources
deps = _lang_deps
# List the pw_protobuf plugin's files as a dependency to recompile
# generated code if they are modified.
#
# TODO(frolv): This check is currently true as pw_protobuf is the
# only supported plugin. It should be updated to support other proto
# libraries such as nanopb.
if (true) {
protoc_deps = [ "$dir_pw_protobuf:codegen_protoc_plugin" ]
}
}
} else if (lang == "go") {
_pw_go_proto_library(_lang_target) {
protos = invoker.sources
deps = _lang_deps
}
} else {
assert(false,
string_join(
" ",
[
"pw_proto_codegen doesn't know how to generate code for",
"language '$lang'. Please add support if you require it.",
]))
}
}
# If the user attempts to use the target directly instead of one of the
# language targets, run a script which prints a nice error message.
pw_python_script(target_name) {
script = string_join("/",
[
dir_pw_protobuf_compiler,
"py",
"pw_protobuf_compiler",
"proto_target_invalid.py",
])
args = [
"--target",
target_name,
"--dir",
get_path_info(".", "abspath"),
"--root",
"//",
] + pw_protobuf_langs
stamp = true
}
}