blob: e3da252b4018eea67b3a60ac5024e1f7549c5c97 [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
Armando Montanezfb3d3fb2020-06-09 18:12:12 -070015import("//build_overrides/pigweed.gni")
16
Alexei Frolov05d8ef22020-06-08 10:32:29 -070017import("$dir_pw_build/input_group.gni")
Wyatt Hepler51ded742020-10-19 14:45:27 -070018import("$dir_pw_build/python_action.gni")
Alexei Frolovedd2f142020-06-09 19:11:27 -070019import("$dir_pw_build/target_types.gni")
Alexei Frolovdd6fa5c2020-08-11 10:04:01 -070020import("$dir_pw_third_party/nanopb/nanopb.gni")
Wyatt Heplerd49f8fe2020-10-15 10:13:47 -070021
Alexei Frolov4c0428a2020-06-10 10:46:04 -070022declare_args() {
23 # Generators with which to compile protobuf code. These are used by the
24 # pw_proto_library template to determine which build targets to create.
25 #
26 # Supported generators:
Alexei Frolovc912ea72020-10-26 08:43:27 -070027 # "pwpb", "nanopb", "nanopb_rpc", "raw_rpc", "go"
Alexei Frolov4c0428a2020-06-10 10:46:04 -070028 pw_protobuf_GENERATORS = [
29 "pwpb",
Alexei Frolovc912ea72020-10-26 08:43:27 -070030 "raw_rpc",
Alexei Frolov4c0428a2020-06-10 10:46:04 -070031 "go",
32 ]
33}
Alexei Frolov942adf02019-12-11 17:07:28 -080034
35# Python script that invokes protoc.
36_gen_script_path =
37 "$dir_pw_protobuf_compiler/py/pw_protobuf_compiler/generate_protos.py"
38
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -070039_forwarded_vars = [
40 "testonly",
41 "visibility",
42]
43
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070044# Generates pw_protobuf C++ code for proto files, creating a source_set of the
45# generated files. This is internal and should not be used outside of this file.
46# Use pw_proto_library instead.
Alexei Frolov942adf02019-12-11 17:07:28 -080047#
48# Args:
49# protos: List of input .proto files.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070050template("_pw_pwpb_proto_library") {
Alexei Frolov942adf02019-12-11 17:07:28 -080051 _proto_gen_dir = "$root_gen_dir/protos"
Alexei Frolove19ebb82020-05-14 17:21:20 -070052 _module_path = get_path_info(".", "abspath")
53 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
54
55 _outputs = []
56 foreach(_proto, _relative_proto_paths) {
57 _output = string_replace(_proto, ".proto", ".pwpb.h")
58 _outputs += [ "$_proto_gen_dir/$_output" ]
59 }
Alexei Frolov942adf02019-12-11 17:07:28 -080060
61 _gen_target = "${target_name}_gen"
Wyatt Heplerc8e05a42020-10-19 14:49:39 -070062 pw_python_action(_gen_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -070063 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolov942adf02019-12-11 17:07:28 -080064 script = _gen_script_path
65 args = [
66 "--language",
67 "cc",
68 "--module-path",
Wyatt Hepler8224a642020-07-29 08:55:56 -070069 rebase_path(_module_path),
Alexei Frolove19ebb82020-05-14 17:21:20 -070070 "--include-file",
Wyatt Hepler8224a642020-07-29 08:55:56 -070071 rebase_path(invoker.include_file),
Alexei Frolov942adf02019-12-11 17:07:28 -080072 "--out-dir",
Wyatt Hepler8224a642020-07-29 08:55:56 -070073 rebase_path(_proto_gen_dir),
74 ] + rebase_path(invoker.protos)
Alexei Frolov942adf02019-12-11 17:07:28 -080075 inputs = invoker.protos
76 outputs = _outputs
Alexei Frolove19ebb82020-05-14 17:21:20 -070077 deps = invoker.deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -080078 if (defined(invoker.protoc_deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -070079 deps += invoker.protoc_deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -080080 }
Alexei Frolov942adf02019-12-11 17:07:28 -080081 }
82
83 # For C++ proto files, the generated proto directory is added as an include
84 # path for the code. This requires using "all_dependent_configs" to force the
85 # include on any code that transitively depends on the generated protos.
Alexei Frolov942adf02019-12-11 17:07:28 -080086 _include_config_target = "${target_name}_includes"
87 config(_include_config_target) {
Alexei Frolove19ebb82020-05-14 17:21:20 -070088 include_dirs = [ "$_proto_gen_dir" ]
Alexei Frolov942adf02019-12-11 17:07:28 -080089 }
90
91 # Create a library with the generated source files.
Alexei Frolovedd2f142020-06-09 19:11:27 -070092 pw_source_set(target_name) {
Alexei Frolov942adf02019-12-11 17:07:28 -080093 all_dependent_configs = [ ":$_include_config_target" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -070094 deps = [ ":$_gen_target" ]
95 public_deps = [ dir_pw_protobuf ] + invoker.gen_deps
Alexei Frolov942adf02019-12-11 17:07:28 -080096 sources = get_target_outputs(":$_gen_target")
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070097 public = filter_include(sources, [ "*.pwpb.h" ])
98 }
99}
100
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700101# Generates nanopb RPC code for proto files, creating a source_set of the
102# generated files. This is internal and should not be used outside of this file.
103# Use pw_proto_library instead.
104#
105# Args:
106# protos: List of input .proto files.
107#
108template("_pw_nanopb_rpc_proto_library") {
109 assert(defined(dir_pw_third_party_nanopb) && dir_pw_third_party_nanopb != "",
110 "\$dir_pw_third_party_nanopb must be set to compile nanopb protobufs")
111
112 _proto_gen_dir = "$root_gen_dir/protos"
113 _module_path = get_path_info(".", "abspath")
114 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
115
116 _outputs = []
117 foreach(_proto, _relative_proto_paths) {
Alexei Frolov3ab26ff2020-07-21 10:44:58 -0700118 _output_h = string_replace(_proto, ".proto", ".rpc.pb.h")
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700119 _outputs += [ "$_proto_gen_dir/$_output_h" ]
120 }
121
122 # Create a target which runs protoc configured with the nanopb_rpc plugin to
123 # generate the C++ proto RPC headers.
124 _gen_target = "${target_name}_gen"
Wyatt Heplerc8e05a42020-10-19 14:49:39 -0700125 pw_python_action(_gen_target) {
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700126 forward_variables_from(invoker, _forwarded_vars)
127 script = _gen_script_path
128 args = [
129 "--language",
130 "nanopb_rpc",
131 "--module-path",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700132 rebase_path(_module_path),
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700133 "--include-paths",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700134 rebase_path("$dir_pw_third_party_nanopb/generator/proto"),
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700135 "--include-file",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700136 rebase_path(invoker.include_file),
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700137 "--out-dir",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700138 rebase_path(_proto_gen_dir),
139 ] + rebase_path(invoker.protos)
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700140 inputs = invoker.protos
141 outputs = _outputs
142
143 deps = invoker.deps
144 if (defined(invoker.protoc_deps)) {
145 deps += invoker.protoc_deps
146 }
147 }
148
149 # For C++ proto files, the generated proto directory is added as an include
150 # path for the code. This requires using "all_dependent_configs" to force the
151 # include on any code that transitively depends on the generated protos.
152 _include_root = rebase_path(get_path_info(".", "abspath"), "//")
153 _include_config_target = "${target_name}_includes"
154 config(_include_config_target) {
155 include_dirs = [
156 "$_proto_gen_dir",
157 "$_proto_gen_dir/$_include_root",
158 ]
159 }
160
161 # Create a library with the generated source files.
162 pw_source_set(target_name) {
163 all_dependent_configs = [ ":$_include_config_target" ]
164 deps = [ ":$_gen_target" ]
165 public_deps = [
Wyatt Heplercbd09c22020-09-15 11:17:24 -0700166 "$dir_pw_rpc:server",
Alexei Frolova4d71502020-10-14 12:43:14 -0700167 "$dir_pw_rpc/nanopb:method_union",
Alexei Frolovdd6fa5c2020-08-11 10:04:01 -0700168 "$dir_pw_third_party/nanopb",
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700169 ] + invoker.gen_deps
170 public = get_target_outputs(":$_gen_target")
171 }
172}
173
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700174# Generates nanopb code for proto files, creating a source_set of the generated
175# files. This is internal and should not be used outside of this file. Use
176# pw_proto_library instead.
177#
178# Args:
179# protos: List of input .proto files.
180template("_pw_nanopb_proto_library") {
Alexei Frolov4c0428a2020-06-10 10:46:04 -0700181 assert(defined(dir_pw_third_party_nanopb) && dir_pw_third_party_nanopb != "",
182 "\$dir_pw_third_party_nanopb must be set to compile nanopb protobufs")
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700183
184 _proto_gen_dir = "$root_gen_dir/protos"
Alexei Frolove19ebb82020-05-14 17:21:20 -0700185 _module_path = get_path_info(".", "abspath")
186 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
187
188 _outputs = []
189 foreach(_proto, _relative_proto_paths) {
190 _output_h = string_replace(_proto, ".proto", ".pb.h")
191 _output_c = string_replace(_proto, ".proto", ".pb.c")
192 _outputs += [
193 "$_proto_gen_dir/$_output_h",
194 "$_proto_gen_dir/$_output_c",
195 ]
196 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700197
Alexei Frolov4c0428a2020-06-10 10:46:04 -0700198 _nanopb_plugin = "$dir_pw_third_party_nanopb/generator/protoc-gen-nanopb"
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700199 if (host_os == "win") {
200 _nanopb_plugin += ".bat"
201 }
202
203 # Create a target which runs protoc configured with the nanopb plugin to
204 # generate the C proto sources.
205 _gen_target = "${target_name}_gen"
Wyatt Heplerc8e05a42020-10-19 14:49:39 -0700206 pw_python_action(_gen_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700207 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700208 script = _gen_script_path
209 args = [
210 "--language",
211 "nanopb",
212 "--module-path",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700213 rebase_path(_module_path),
Alexei Frolovac5b6742020-05-01 13:28:14 -0700214 "--include-paths",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700215 rebase_path("$dir_pw_third_party_nanopb/generator/proto"),
Alexei Frolove19ebb82020-05-14 17:21:20 -0700216 "--include-file",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700217 rebase_path(invoker.include_file),
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700218 "--out-dir",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700219 rebase_path(_proto_gen_dir),
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700220 "--custom-plugin",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700221 rebase_path(_nanopb_plugin),
222 ] + rebase_path(invoker.protos)
223
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700224 inputs = invoker.protos
225 outputs = _outputs
226
Alexei Frolove19ebb82020-05-14 17:21:20 -0700227 deps = invoker.deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700228 if (defined(invoker.protoc_deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700229 deps += invoker.protoc_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700230 }
231 }
232
233 # For C++ proto files, the generated proto directory is added as an include
234 # path for the code. This requires using "all_dependent_configs" to force the
235 # include on any code that transitively depends on the generated protos.
236 _include_root = rebase_path(get_path_info(".", "abspath"), "//")
237 _include_config_target = "${target_name}_includes"
238 config(_include_config_target) {
239 include_dirs = [
240 "$_proto_gen_dir",
241 "$_proto_gen_dir/$_include_root",
242 ]
Ewout van Bekkume4d7b692020-10-15 13:12:30 -0700243
244 # Nanopb uses __cplusplus with the implicit default of 0.
245 cflags = [ "-Wno-undef" ]
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700246 }
247
248 # Create a library with the generated source files.
Alexei Frolovedd2f142020-06-09 19:11:27 -0700249 pw_source_set(target_name) {
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700250 all_dependent_configs = [ ":$_include_config_target" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -0700251 deps = [ ":$_gen_target" ]
Alexei Frolovdd6fa5c2020-08-11 10:04:01 -0700252 public_deps = [ "$dir_pw_third_party/nanopb" ] + invoker.gen_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700253 sources = get_target_outputs(":$_gen_target")
254 public = filter_include(sources, [ "*.pb.h" ])
Alexei Frolov942adf02019-12-11 17:07:28 -0800255 }
256}
257
Alexei Frolovc912ea72020-10-26 08:43:27 -0700258# Generates raw RPC code for proto files, creating a source_set of the generated
259# files. This is internal and should not be used outside of this file. Use
260# pw_proto_library instead.
261#
262# Args:
263# protos: List of input .proto files.
264#
265template("_pw_raw_rpc_proto_library") {
266 _proto_gen_dir = "$root_gen_dir/protos"
267 _module_path = get_path_info(".", "abspath")
268 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
269
270 _outputs = []
271 foreach(_proto, _relative_proto_paths) {
272 _output_h = string_replace(_proto, ".proto", ".raw_rpc.pb.h")
273 _outputs += [ "$_proto_gen_dir/$_output_h" ]
274 }
275
276 # Create a target which runs protoc configured with the nanopb_rpc plugin to
277 # generate the C++ proto RPC headers.
278 _gen_target = "${target_name}_gen"
279 pw_python_action(_gen_target) {
280 forward_variables_from(invoker, _forwarded_vars)
281 script = _gen_script_path
282 args = [
283 "--language",
284 "raw_rpc",
285 "--module-path",
286 rebase_path(_module_path),
287 "--include-file",
288 rebase_path(invoker.include_file),
289 "--out-dir",
290 rebase_path(_proto_gen_dir),
291 ] + rebase_path(invoker.protos)
292 inputs = invoker.protos
293 outputs = _outputs
294
295 deps = invoker.deps
296 if (defined(invoker.protoc_deps)) {
297 deps += invoker.protoc_deps
298 }
299 }
300
301 # For C++ proto files, the generated proto directory is added as an include
302 # path for the code.
303 _include_root = rebase_path(get_path_info(".", "abspath"), "//")
304 _include_config_target = "${target_name}_includes"
305 config(_include_config_target) {
306 include_dirs = [
307 "$_proto_gen_dir",
308 "$_proto_gen_dir/$_include_root",
309 ]
310 }
311
312 # Create a library with the generated source files.
313 pw_source_set(target_name) {
314 public_configs = [ ":$_include_config_target" ]
315 deps = [ ":$_gen_target" ]
316 public_deps = [
317 "$dir_pw_rpc:server",
318 "$dir_pw_rpc/raw:method_union",
319 ] + invoker.gen_deps
320 public = get_target_outputs(":$_gen_target")
321 }
322}
323
Alexei Frolovdef14712019-12-23 13:03:32 -0800324# Generates Go code for proto files, listing the proto output directory in the
325# metadata variable GOPATH. Internal use only.
326#
327# Args:
328# protos: List of input .proto files.
329template("_pw_go_proto_library") {
330 _proto_gopath = "$root_gen_dir/go"
331 _proto_gen_dir = "$_proto_gopath/src"
332 _rebased_gopath = rebase_path(_proto_gopath)
333
Wyatt Heplerc8e05a42020-10-19 14:49:39 -0700334 pw_python_action(target_name) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700335 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovdef14712019-12-23 13:03:32 -0800336 metadata = {
337 gopath = [ "GOPATH+=$_rebased_gopath" ]
Alexei Frolovc15a9882019-12-23 14:29:02 -0800338 external_deps = [
339 "github.com/golang/protobuf/proto",
340 "google.golang.org/grpc",
341 ]
Alexei Frolovdef14712019-12-23 13:03:32 -0800342 }
343 script = _gen_script_path
344 args = [
345 "--language",
346 "go",
347 "--module-path",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700348 rebase_path("//"),
Alexei Frolove19ebb82020-05-14 17:21:20 -0700349 "--include-file",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700350 rebase_path(invoker.include_file),
Alexei Frolovdef14712019-12-23 13:03:32 -0800351 "--out-dir",
Wyatt Hepler8224a642020-07-29 08:55:56 -0700352 rebase_path(_proto_gen_dir),
353 ] + rebase_path(invoker.protos)
Alexei Frolovdef14712019-12-23 13:03:32 -0800354 inputs = invoker.protos
Alexei Frolove19ebb82020-05-14 17:21:20 -0700355 deps = invoker.deps + invoker.gen_deps
Alexei Frolovdef14712019-12-23 13:03:32 -0800356 stamp = true
357 }
358}
359
Alexei Frolov942adf02019-12-11 17:07:28 -0800360# Generates protobuf code from .proto definitions for various languages.
361#
Alexei Frolov4c0428a2020-06-10 10:46:04 -0700362# The generators to use are defined in the pw_protobuf_GENERATORS build
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700363# variable. Each listed generator creates a generated code target called
Alexei Frolov942adf02019-12-11 17:07:28 -0800364#
Alexei Frolov8e30d462020-10-22 13:54:36 -0700365# <target_name>.<generator>
Alexei Frolov942adf02019-12-11 17:07:28 -0800366#
367# For example, with the following definitions:
368#
Alexei Frolov4c0428a2020-06-10 10:46:04 -0700369# pw_protobuf_GENERATORS = [ "pwpb", "py" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800370#
371# pw_proto_library("my_protos") {
372# sources = [ "foo.proto" ]
373# }
374#
375# Two build targets will be created for the declared "my_protos" target.
376#
Alexei Frolov8e30d462020-10-22 13:54:36 -0700377# "my_protos.pwpb" <-- C++ source_set containing generated proto code
378# "my_protos.py" <-- Python module containing generated proto code
Alexei Frolov942adf02019-12-11 17:07:28 -0800379#
380# Args:
381# sources: List of input .proto files.
382# deps: List of other pw_proto_library dependencies.
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700383# inputs: Other files on which the protos depend (e.g. nanopb .options files).
Alexei Frolov942adf02019-12-11 17:07:28 -0800384template("pw_proto_library") {
385 assert(defined(invoker.sources) && invoker.sources != [],
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700386 "pw_proto_library requires .proto source files")
Alexei Frolov942adf02019-12-11 17:07:28 -0800387
Alexei Frolove19ebb82020-05-14 17:21:20 -0700388 # For each proto target, create a file which collects the base directories of
389 # all of its dependencies to list as include paths to protoc.
390 _include_metadata_target = "${target_name}_include_paths"
391 _include_metadata_file = "${target_gen_dir}/${target_name}_includes.txt"
392 generated_file(_include_metadata_target) {
Alexei Frolov942adf02019-12-11 17:07:28 -0800393 if (defined(invoker.deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700394 # Collect metadata from the include path files of each dependency.
395 deps = process_file_template(invoker.deps, "{{source}}_include_paths")
Alexei Frolov942adf02019-12-11 17:07:28 -0800396 } else {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700397 deps = []
Alexei Frolov942adf02019-12-11 17:07:28 -0800398 }
Alexei Frolove19ebb82020-05-14 17:21:20 -0700399 data_keys = [ "protoc_includes" ]
400 outputs = [ _include_metadata_file ]
401
402 # Indicate this library's base directory for its dependents.
403 metadata = {
Alexei Frolovf79d2272020-06-18 13:37:38 -0700404 protoc_includes = [ rebase_path(".") ]
Alexei Frolove19ebb82020-05-14 17:21:20 -0700405 }
406 }
407
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700408 _deps = [ ":$_include_metadata_target" ]
409
410 if (defined(invoker.inputs)) {
411 # Toss any additional inputs into an input group dependency.
412 _input_target_name = "${target_name}_inputs"
413 pw_input_group(_input_target_name) {
414 inputs = invoker.inputs
415 }
416 _deps += [ ":$_input_target_name" ]
417 }
418
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700419 # If the nanopb_rpc generator is selected, make sure that nanopb is also
420 # selected.
421 has_nanopb_rpc = pw_protobuf_GENERATORS + [ "nanopb_rpc" ] -
422 [ "nanopb_rpc" ] != pw_protobuf_GENERATORS
423 if (has_nanopb_rpc) {
424 _generators =
425 pw_protobuf_GENERATORS + [ "nanopb" ] - [ "nanopb" ] + [ "nanopb" ]
426 } else {
427 _generators = pw_protobuf_GENERATORS
428 }
429
430 foreach(_gen, _generators) {
Alexei Frolov8e30d462020-10-22 13:54:36 -0700431 _lang_target = "${target_name}.${_gen}"
Alexei Frolov8185c822020-06-12 10:45:04 -0700432 _gen_deps = []
Alexei Frolovc4b62ec2020-07-13 08:35:10 -0700433
434 if (_gen == "nanopb_rpc") {
435 # Generated RPC code depends on the library's core protos.
Alexei Frolov8e30d462020-10-22 13:54:36 -0700436 _gen_deps += [ ":${target_name}.nanopb" ]
Alexei Frolovc4b62ec2020-07-13 08:35:10 -0700437 }
438
Alexei Frolov8185c822020-06-12 10:45:04 -0700439 if (defined(invoker.deps)) {
Alexei Frolov8e30d462020-10-22 13:54:36 -0700440 _gen_deps += process_file_template(invoker.deps, "{{source}}.${_gen}")
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700441
442 if (_gen == "nanopb_rpc") {
Alexei Frolovc4b62ec2020-07-13 08:35:10 -0700443 # RPC dependencies also depend on their core generated protos.
Alexei Frolov8e30d462020-10-22 13:54:36 -0700444 _gen_deps += process_file_template(invoker.deps, "{{source}}.nanopb")
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700445 }
Alexei Frolov8185c822020-06-12 10:45:04 -0700446 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800447
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700448 if (_gen == "pwpb") {
449 _pw_pwpb_proto_library(_lang_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700450 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolov942adf02019-12-11 17:07:28 -0800451 protos = invoker.sources
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700452 deps = _deps
Alexei Frolove19ebb82020-05-14 17:21:20 -0700453 include_file = _include_metadata_file
454 gen_deps = _gen_deps
Alexei Frolovc912ea72020-10-26 08:43:27 -0700455 protoc_deps = [ "$dir_pw_protobuf/py" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800456 }
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700457 } else if (_gen == "nanopb_rpc") {
458 _pw_nanopb_rpc_proto_library(_lang_target) {
459 forward_variables_from(invoker, _forwarded_vars)
460 protos = invoker.sources
461 deps = _deps
462 include_file = _include_metadata_file
463 gen_deps = _gen_deps
Alexei Frolovc912ea72020-10-26 08:43:27 -0700464 protoc_deps = [ "$dir_pw_rpc/py" ]
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700465 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700466 } else if (_gen == "nanopb") {
467 _pw_nanopb_proto_library(_lang_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700468 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700469 protos = invoker.sources
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700470 deps = _deps
Alexei Frolove19ebb82020-05-14 17:21:20 -0700471 include_file = _include_metadata_file
472 gen_deps = _gen_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700473 }
Alexei Frolovc912ea72020-10-26 08:43:27 -0700474 } else if (_gen == "raw_rpc") {
475 _pw_raw_rpc_proto_library(_lang_target) {
476 forward_variables_from(invoker, _forwarded_vars)
477 protos = invoker.sources
478 deps = _deps
479 include_file = _include_metadata_file
480 gen_deps = _gen_deps
481 protoc_deps = [ "$dir_pw_rpc/py" ]
482 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700483 } else if (_gen == "go") {
Alexei Frolovdef14712019-12-23 13:03:32 -0800484 _pw_go_proto_library(_lang_target) {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700485 forward_variables_from(invoker, _forwarded_vars)
Alexei Frolovdef14712019-12-23 13:03:32 -0800486 protos = invoker.sources
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700487 deps = _deps
Alexei Frolove19ebb82020-05-14 17:21:20 -0700488 include_file = _include_metadata_file
489 gen_deps = _gen_deps
Alexei Frolovdef14712019-12-23 13:03:32 -0800490 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800491 } else {
492 assert(false,
493 string_join(
494 " ",
495 [
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700496 "pw_proto_library doesn't know how to generate code for",
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700497 "generator '$_gen'. Please add support if you require it.",
Alexei Frolov942adf02019-12-11 17:07:28 -0800498 ]))
499 }
Alexei Frolov8e30d462020-10-22 13:54:36 -0700500
501 # TODO(frolv): Remove this after migrating existing code.
502 _legacy_target = "${target_name}_${_gen}"
503 group(_legacy_target) {
504 public_deps = [ ":$_lang_target" ]
505 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800506 }
507
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700508 # All supported pw_protobuf generators.
509 _protobuf_generators = [
510 "pwpb",
511 "nanopb",
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700512 "nanopb_rpc",
Alexei Frolovc912ea72020-10-26 08:43:27 -0700513 "raw_rpc",
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700514 "go",
515 ]
516
517 # Create stub versions of the proto library for other protobuf generators.
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700518 foreach(_gen, _protobuf_generators - _generators) {
Alexei Frolov8e30d462020-10-22 13:54:36 -0700519 pw_python_action("${target_name}.${_gen}") {
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -0700520 forward_variables_from(invoker, _forwarded_vars)
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700521 script = string_join("/",
522 [
523 dir_pw_protobuf_compiler,
524 "py",
525 "pw_protobuf_compiler",
526 "generator_not_selected.py",
527 ])
528 args = [
529 "--library",
Alexei Frolov8e30d462020-10-22 13:54:36 -0700530 target_name,
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700531 "--generator",
532 _gen,
533 ]
534 inputs = invoker.sources
535 stamp = true
536 }
537 }
538
Alexei Frolov942adf02019-12-11 17:07:28 -0800539 # If the user attempts to use the target directly instead of one of the
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700540 # generator targets, run a script which prints a nice error message.
Wyatt Heplerc8e05a42020-10-19 14:49:39 -0700541 pw_python_action(target_name) {
Alexei Frolov942adf02019-12-11 17:07:28 -0800542 script = string_join("/",
543 [
544 dir_pw_protobuf_compiler,
545 "py",
546 "pw_protobuf_compiler",
547 "proto_target_invalid.py",
548 ])
549 args = [
550 "--target",
551 target_name,
552 "--dir",
553 get_path_info(".", "abspath"),
554 "--root",
555 "//",
Alexei Frolov4c0428a2020-06-10 10:46:04 -0700556 ] + pw_protobuf_GENERATORS
Alexei Frolov942adf02019-12-11 17:07:28 -0800557 stamp = true
558 }
559}