blob: 92b49d404a15567074ddafe6ef8e7a0e67ad1178 [file] [log] [blame]
Alexei Frolove19ebb82020-05-14 17:21:20 -07001# Copyright 2020 The Pigweed Authors
Alexei Frolov942adf02019-12-11 17:07:28 -08002#
3# Licensed under the Apache License, Version 2.0 (the "License"); you may not
4# use this file except in compliance with the License. You may obtain a copy of
5# the License at
6#
7# https://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12# License for the specific language governing permissions and limitations under
13# the License.
14
Alexei Frolov05d8ef22020-06-08 10:32:29 -070015import("$dir_pw_build/input_group.gni")
Alexei Frolov942adf02019-12-11 17:07:28 -080016import("$dir_pw_build/python_script.gni")
Alexei Frolovedd2f142020-06-09 19:11:27 -070017import("$dir_pw_build/target_types.gni")
Alexei Frolov942adf02019-12-11 17:07:28 -080018
19# Python script that invokes protoc.
20_gen_script_path =
21 "$dir_pw_protobuf_compiler/py/pw_protobuf_compiler/generate_protos.py"
22
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -070023_forwarded_vars = [
24 "testonly",
25 "visibility",
26]
27
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070028# Generates pw_protobuf C++ code for proto files, creating a source_set of the
29# generated files. This is internal and should not be used outside of this file.
30# Use pw_proto_library instead.
Alexei Frolov942adf02019-12-11 17:07:28 -080031#
32# Args:
33# protos: List of input .proto files.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070034template("_pw_pwpb_proto_library") {
Alexei Frolov942adf02019-12-11 17:07:28 -080035 _proto_gen_dir = "$root_gen_dir/protos"
Alexei Frolove19ebb82020-05-14 17:21:20 -070036 _module_path = get_path_info(".", "abspath")
37 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
38
39 _outputs = []
40 foreach(_proto, _relative_proto_paths) {
41 _output = string_replace(_proto, ".proto", ".pwpb.h")
42 _outputs += [ "$_proto_gen_dir/$_output" ]
43 }
Alexei Frolov942adf02019-12-11 17:07:28 -080044
45 _gen_target = "${target_name}_gen"
46 pw_python_script(_gen_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -070047 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolov942adf02019-12-11 17:07:28 -080048 script = _gen_script_path
49 args = [
50 "--language",
51 "cc",
52 "--module-path",
Alexei Frolove19ebb82020-05-14 17:21:20 -070053 _module_path,
54 "--include-file",
55 invoker.include_file,
Alexei Frolov942adf02019-12-11 17:07:28 -080056 "--out-dir",
57 _proto_gen_dir,
58 ] + get_path_info(invoker.protos, "abspath")
59 inputs = invoker.protos
60 outputs = _outputs
Alexei Frolove19ebb82020-05-14 17:21:20 -070061 deps = invoker.deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -080062 if (defined(invoker.protoc_deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -070063 deps += invoker.protoc_deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -080064 }
Alexei Frolov942adf02019-12-11 17:07:28 -080065 }
66
67 # For C++ proto files, the generated proto directory is added as an include
68 # path for the code. This requires using "all_dependent_configs" to force the
69 # include on any code that transitively depends on the generated protos.
Alexei Frolov942adf02019-12-11 17:07:28 -080070 _include_config_target = "${target_name}_includes"
71 config(_include_config_target) {
Alexei Frolove19ebb82020-05-14 17:21:20 -070072 include_dirs = [ "$_proto_gen_dir" ]
Alexei Frolov942adf02019-12-11 17:07:28 -080073 }
74
75 # Create a library with the generated source files.
Alexei Frolovedd2f142020-06-09 19:11:27 -070076 pw_source_set(target_name) {
Alexei Frolov942adf02019-12-11 17:07:28 -080077 all_dependent_configs = [ ":$_include_config_target" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -070078 deps = [ ":$_gen_target" ]
79 public_deps = [ dir_pw_protobuf ] + invoker.gen_deps
Alexei Frolov942adf02019-12-11 17:07:28 -080080 sources = get_target_outputs(":$_gen_target")
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070081 public = filter_include(sources, [ "*.pwpb.h" ])
82 }
83}
84
85# Generates nanopb code for proto files, creating a source_set of the generated
86# files. This is internal and should not be used outside of this file. Use
87# pw_proto_library instead.
88#
89# Args:
90# protos: List of input .proto files.
91template("_pw_nanopb_proto_library") {
92 assert(defined(dir_third_party_nanopb) && dir_third_party_nanopb != "",
93 "\$dir_third_party_nanopb must be set to compile nanopb protobufs")
94
95 _proto_gen_dir = "$root_gen_dir/protos"
Alexei Frolove19ebb82020-05-14 17:21:20 -070096 _module_path = get_path_info(".", "abspath")
97 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
98
99 _outputs = []
100 foreach(_proto, _relative_proto_paths) {
101 _output_h = string_replace(_proto, ".proto", ".pb.h")
102 _output_c = string_replace(_proto, ".proto", ".pb.c")
103 _outputs += [
104 "$_proto_gen_dir/$_output_h",
105 "$_proto_gen_dir/$_output_c",
106 ]
107 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700108
109 _nanopb_plugin = "$dir_third_party_nanopb/generator/protoc-gen-nanopb"
110 if (host_os == "win") {
111 _nanopb_plugin += ".bat"
112 }
113
114 # Create a target which runs protoc configured with the nanopb plugin to
115 # generate the C proto sources.
116 _gen_target = "${target_name}_gen"
117 pw_python_script(_gen_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700118 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700119 script = _gen_script_path
120 args = [
121 "--language",
122 "nanopb",
123 "--module-path",
Alexei Frolove19ebb82020-05-14 17:21:20 -0700124 _module_path,
Alexei Frolovac5b6742020-05-01 13:28:14 -0700125 "--include-paths",
126 "$dir_third_party_nanopb/generator/proto",
Alexei Frolove19ebb82020-05-14 17:21:20 -0700127 "--include-file",
128 invoker.include_file,
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700129 "--out-dir",
130 _proto_gen_dir,
131 "--custom-plugin",
132 get_path_info(_nanopb_plugin, "abspath"),
133 ] + get_path_info(invoker.protos, "abspath")
134 inputs = invoker.protos
135 outputs = _outputs
136
Alexei Frolove19ebb82020-05-14 17:21:20 -0700137 deps = invoker.deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700138 if (defined(invoker.protoc_deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700139 deps += invoker.protoc_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700140 }
141 }
142
143 # For C++ proto files, the generated proto directory is added as an include
144 # path for the code. This requires using "all_dependent_configs" to force the
145 # include on any code that transitively depends on the generated protos.
146 _include_root = rebase_path(get_path_info(".", "abspath"), "//")
147 _include_config_target = "${target_name}_includes"
148 config(_include_config_target) {
149 include_dirs = [
150 "$_proto_gen_dir",
151 "$_proto_gen_dir/$_include_root",
152 ]
153 }
154
155 # Create a library with the generated source files.
Alexei Frolovedd2f142020-06-09 19:11:27 -0700156 pw_source_set(target_name) {
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700157 all_dependent_configs = [ ":$_include_config_target" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -0700158 deps = [ ":$_gen_target" ]
159 public_deps = [ dir_third_party_nanopb ] + invoker.gen_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700160 sources = get_target_outputs(":$_gen_target")
161 public = filter_include(sources, [ "*.pb.h" ])
Alexei Frolov942adf02019-12-11 17:07:28 -0800162 }
163}
164
Alexei Frolovdef14712019-12-23 13:03:32 -0800165# Generates Go code for proto files, listing the proto output directory in the
166# metadata variable GOPATH. Internal use only.
167#
168# Args:
169# protos: List of input .proto files.
170template("_pw_go_proto_library") {
171 _proto_gopath = "$root_gen_dir/go"
172 _proto_gen_dir = "$_proto_gopath/src"
173 _rebased_gopath = rebase_path(_proto_gopath)
174
175 pw_python_script(target_name) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700176 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovdef14712019-12-23 13:03:32 -0800177 metadata = {
178 gopath = [ "GOPATH+=$_rebased_gopath" ]
Alexei Frolovc15a9882019-12-23 14:29:02 -0800179 external_deps = [
180 "github.com/golang/protobuf/proto",
181 "google.golang.org/grpc",
182 ]
Alexei Frolovdef14712019-12-23 13:03:32 -0800183 }
184 script = _gen_script_path
185 args = [
186 "--language",
187 "go",
188 "--module-path",
189 "//",
Alexei Frolove19ebb82020-05-14 17:21:20 -0700190 "--include-file",
191 invoker.include_file,
Alexei Frolovdef14712019-12-23 13:03:32 -0800192 "--out-dir",
193 _proto_gen_dir,
194 ] + get_path_info(invoker.protos, "abspath")
195 inputs = invoker.protos
Alexei Frolove19ebb82020-05-14 17:21:20 -0700196 deps = invoker.deps + invoker.gen_deps
Alexei Frolovdef14712019-12-23 13:03:32 -0800197 stamp = true
198 }
199}
200
Alexei Frolov942adf02019-12-11 17:07:28 -0800201# Generates protobuf code from .proto definitions for various languages.
202#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700203# The generators to use are defined in the pw_protobuf_generators build
204# variable. Each listed generator creates a generated code target called
Alexei Frolov942adf02019-12-11 17:07:28 -0800205#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700206# <target_name>_<generator>
Alexei Frolov942adf02019-12-11 17:07:28 -0800207#
208# For example, with the following definitions:
209#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700210# pw_protobuf_generators = [ "pwpb", "py" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800211#
212# pw_proto_library("my_protos") {
213# sources = [ "foo.proto" ]
214# }
215#
216# Two build targets will be created for the declared "my_protos" target.
217#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700218# "my_protos_pwpb" <-- C++ source_set containing generated proto code
219# "my_protos_py" <-- Python module containing generated proto code
Alexei Frolov942adf02019-12-11 17:07:28 -0800220#
221# Args:
222# sources: List of input .proto files.
223# deps: List of other pw_proto_library dependencies.
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700224# inputs: Other files on which the protos depend (e.g. nanopb .options files).
Alexei Frolov942adf02019-12-11 17:07:28 -0800225template("pw_proto_library") {
226 assert(defined(invoker.sources) && invoker.sources != [],
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700227 "pw_proto_library requires .proto source files")
Alexei Frolov942adf02019-12-11 17:07:28 -0800228
Alexei Frolove19ebb82020-05-14 17:21:20 -0700229 # For each proto target, create a file which collects the base directories of
230 # all of its dependencies to list as include paths to protoc.
231 _include_metadata_target = "${target_name}_include_paths"
232 _include_metadata_file = "${target_gen_dir}/${target_name}_includes.txt"
233 generated_file(_include_metadata_target) {
Alexei Frolov942adf02019-12-11 17:07:28 -0800234 if (defined(invoker.deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700235 # Collect metadata from the include path files of each dependency.
236 deps = process_file_template(invoker.deps, "{{source}}_include_paths")
Alexei Frolov942adf02019-12-11 17:07:28 -0800237 } else {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700238 deps = []
Alexei Frolov942adf02019-12-11 17:07:28 -0800239 }
Alexei Frolove19ebb82020-05-14 17:21:20 -0700240 data_keys = [ "protoc_includes" ]
241 outputs = [ _include_metadata_file ]
242
243 # Indicate this library's base directory for its dependents.
244 metadata = {
245 protoc_includes = [ rebase_path(".", root_out_dir) ]
246 }
247 }
248
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700249 _deps = [ ":$_include_metadata_target" ]
250
251 if (defined(invoker.inputs)) {
252 # Toss any additional inputs into an input group dependency.
253 _input_target_name = "${target_name}_inputs"
254 pw_input_group(_input_target_name) {
255 inputs = invoker.inputs
256 }
257 _deps += [ ":$_input_target_name" ]
258 }
259
Alexei Frolove19ebb82020-05-14 17:21:20 -0700260 foreach(_gen, pw_protobuf_generators) {
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700261 _lang_target = "${target_name}_${_gen}"
Alexei Frolov8185c822020-06-12 10:45:04 -0700262 _gen_deps = []
263 if (defined(invoker.deps)) {
264 _gen_deps = process_file_template(invoker.deps, "{{source}}_${_gen}")
265 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800266
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700267 if (_gen == "pwpb") {
268 _pw_pwpb_proto_library(_lang_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700269 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolov942adf02019-12-11 17:07:28 -0800270 protos = invoker.sources
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700271 deps = _deps
Alexei Frolove19ebb82020-05-14 17:21:20 -0700272 include_file = _include_metadata_file
273 gen_deps = _gen_deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -0800274
275 # List the pw_protobuf plugin's files as a dependency to recompile
276 # generated code if they are modified.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700277 protoc_deps = [ "$dir_pw_protobuf:codegen_protoc_plugin" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800278 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700279 } else if (_gen == "nanopb") {
280 _pw_nanopb_proto_library(_lang_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700281 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700282 protos = invoker.sources
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700283 deps = _deps
Alexei Frolove19ebb82020-05-14 17:21:20 -0700284 include_file = _include_metadata_file
285 gen_deps = _gen_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700286 }
287 } else if (_gen == "go") {
Alexei Frolovdef14712019-12-23 13:03:32 -0800288 _pw_go_proto_library(_lang_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700289 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovdef14712019-12-23 13:03:32 -0800290 protos = invoker.sources
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700291 deps = _deps
Alexei Frolove19ebb82020-05-14 17:21:20 -0700292 include_file = _include_metadata_file
293 gen_deps = _gen_deps
Alexei Frolovdef14712019-12-23 13:03:32 -0800294 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800295 } else {
296 assert(false,
297 string_join(
298 " ",
299 [
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700300 "pw_proto_library doesn't know how to generate code for",
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700301 "generator '$_gen'. Please add support if you require it.",
Alexei Frolov942adf02019-12-11 17:07:28 -0800302 ]))
303 }
304 }
305
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700306 # All supported pw_protobuf generators.
307 _protobuf_generators = [
308 "pwpb",
309 "nanopb",
310 "go",
311 ]
312
313 # Create stub versions of the proto library for other protobuf generators.
314 foreach(_gen, _protobuf_generators - pw_protobuf_generators) {
315 pw_python_script("${target_name}_${_gen}") {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700316 forward_variables_from(invoker, _forwarded_vars)
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700317 script = string_join("/",
318 [
319 dir_pw_protobuf_compiler,
320 "py",
321 "pw_protobuf_compiler",
322 "generator_not_selected.py",
323 ])
324 args = [
325 "--library",
326 "${target_name}_${_gen}",
327 "--generator",
328 _gen,
329 ]
330 inputs = invoker.sources
331 stamp = true
332 }
333 }
334
Alexei Frolov942adf02019-12-11 17:07:28 -0800335 # If the user attempts to use the target directly instead of one of the
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700336 # generator targets, run a script which prints a nice error message.
Alexei Frolov942adf02019-12-11 17:07:28 -0800337 pw_python_script(target_name) {
338 script = string_join("/",
339 [
340 dir_pw_protobuf_compiler,
341 "py",
342 "pw_protobuf_compiler",
343 "proto_target_invalid.py",
344 ])
345 args = [
346 "--target",
347 target_name,
348 "--dir",
349 get_path_info(".", "abspath"),
350 "--root",
351 "//",
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700352 ] + pw_protobuf_generators
Alexei Frolov942adf02019-12-11 17:07:28 -0800353 stamp = true
354 }
355}