blob: ee2e2dddefcb213e8e197a2091cb7cc0a8481db7 [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
15import("$dir_pw_build/python_script.gni")
16
17# Python script that invokes protoc.
18_gen_script_path =
19 "$dir_pw_protobuf_compiler/py/pw_protobuf_compiler/generate_protos.py"
20
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070021# Generates pw_protobuf C++ code for proto files, creating a source_set of the
22# generated files. This is internal and should not be used outside of this file.
23# Use pw_proto_library instead.
Alexei Frolov942adf02019-12-11 17:07:28 -080024#
25# Args:
26# protos: List of input .proto files.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070027template("_pw_pwpb_proto_library") {
Alexei Frolov942adf02019-12-11 17:07:28 -080028 _proto_gen_dir = "$root_gen_dir/protos"
Alexei Frolove19ebb82020-05-14 17:21:20 -070029 _module_path = get_path_info(".", "abspath")
30 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
31
32 _outputs = []
33 foreach(_proto, _relative_proto_paths) {
34 _output = string_replace(_proto, ".proto", ".pwpb.h")
35 _outputs += [ "$_proto_gen_dir/$_output" ]
36 }
Alexei Frolov942adf02019-12-11 17:07:28 -080037
38 _gen_target = "${target_name}_gen"
39 pw_python_script(_gen_target) {
40 script = _gen_script_path
41 args = [
42 "--language",
43 "cc",
44 "--module-path",
Alexei Frolove19ebb82020-05-14 17:21:20 -070045 _module_path,
46 "--include-file",
47 invoker.include_file,
Alexei Frolov942adf02019-12-11 17:07:28 -080048 "--out-dir",
49 _proto_gen_dir,
50 ] + get_path_info(invoker.protos, "abspath")
51 inputs = invoker.protos
52 outputs = _outputs
Alexei Frolove19ebb82020-05-14 17:21:20 -070053 deps = invoker.deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -080054 if (defined(invoker.protoc_deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -070055 deps += invoker.protoc_deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -080056 }
Alexei Frolov942adf02019-12-11 17:07:28 -080057 }
58
59 # For C++ proto files, the generated proto directory is added as an include
60 # path for the code. This requires using "all_dependent_configs" to force the
61 # include on any code that transitively depends on the generated protos.
Alexei Frolov942adf02019-12-11 17:07:28 -080062 _include_config_target = "${target_name}_includes"
63 config(_include_config_target) {
Alexei Frolove19ebb82020-05-14 17:21:20 -070064 include_dirs = [ "$_proto_gen_dir" ]
Alexei Frolov942adf02019-12-11 17:07:28 -080065 }
66
67 # Create a library with the generated source files.
Alexei Frolov942adf02019-12-11 17:07:28 -080068 source_set(target_name) {
69 all_dependent_configs = [ ":$_include_config_target" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -070070 deps = [ ":$_gen_target" ]
71 public_deps = [ dir_pw_protobuf ] + invoker.gen_deps
Alexei Frolov942adf02019-12-11 17:07:28 -080072 sources = get_target_outputs(":$_gen_target")
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070073 public = filter_include(sources, [ "*.pwpb.h" ])
74 }
75}
76
77# Generates nanopb code for proto files, creating a source_set of the generated
78# files. This is internal and should not be used outside of this file. Use
79# pw_proto_library instead.
80#
81# Args:
82# protos: List of input .proto files.
83template("_pw_nanopb_proto_library") {
84 assert(defined(dir_third_party_nanopb) && dir_third_party_nanopb != "",
85 "\$dir_third_party_nanopb must be set to compile nanopb protobufs")
86
87 _proto_gen_dir = "$root_gen_dir/protos"
Alexei Frolove19ebb82020-05-14 17:21:20 -070088 _module_path = get_path_info(".", "abspath")
89 _relative_proto_paths = rebase_path(invoker.protos, _module_path)
90
91 _outputs = []
92 foreach(_proto, _relative_proto_paths) {
93 _output_h = string_replace(_proto, ".proto", ".pb.h")
94 _output_c = string_replace(_proto, ".proto", ".pb.c")
95 _outputs += [
96 "$_proto_gen_dir/$_output_h",
97 "$_proto_gen_dir/$_output_c",
98 ]
99 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700100
101 _nanopb_plugin = "$dir_third_party_nanopb/generator/protoc-gen-nanopb"
102 if (host_os == "win") {
103 _nanopb_plugin += ".bat"
104 }
105
106 # Create a target which runs protoc configured with the nanopb plugin to
107 # generate the C proto sources.
108 _gen_target = "${target_name}_gen"
109 pw_python_script(_gen_target) {
110 script = _gen_script_path
111 args = [
112 "--language",
113 "nanopb",
114 "--module-path",
Alexei Frolove19ebb82020-05-14 17:21:20 -0700115 _module_path,
Alexei Frolovac5b6742020-05-01 13:28:14 -0700116 "--include-paths",
117 "$dir_third_party_nanopb/generator/proto",
Alexei Frolove19ebb82020-05-14 17:21:20 -0700118 "--include-file",
119 invoker.include_file,
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700120 "--out-dir",
121 _proto_gen_dir,
122 "--custom-plugin",
123 get_path_info(_nanopb_plugin, "abspath"),
124 ] + get_path_info(invoker.protos, "abspath")
125 inputs = invoker.protos
126 outputs = _outputs
127
Alexei Frolove19ebb82020-05-14 17:21:20 -0700128 deps = invoker.deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700129 if (defined(invoker.protoc_deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700130 deps += invoker.protoc_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700131 }
132 }
133
134 # For C++ proto files, the generated proto directory is added as an include
135 # path for the code. This requires using "all_dependent_configs" to force the
136 # include on any code that transitively depends on the generated protos.
137 _include_root = rebase_path(get_path_info(".", "abspath"), "//")
138 _include_config_target = "${target_name}_includes"
139 config(_include_config_target) {
140 include_dirs = [
141 "$_proto_gen_dir",
142 "$_proto_gen_dir/$_include_root",
143 ]
144 }
145
146 # Create a library with the generated source files.
147 source_set(target_name) {
148 all_dependent_configs = [ ":$_include_config_target" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -0700149 deps = [ ":$_gen_target" ]
150 public_deps = [ dir_third_party_nanopb ] + invoker.gen_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700151 sources = get_target_outputs(":$_gen_target")
152 public = filter_include(sources, [ "*.pb.h" ])
Alexei Frolov942adf02019-12-11 17:07:28 -0800153 }
154}
155
Alexei Frolovdef14712019-12-23 13:03:32 -0800156# Generates Go code for proto files, listing the proto output directory in the
157# metadata variable GOPATH. Internal use only.
158#
159# Args:
160# protos: List of input .proto files.
161template("_pw_go_proto_library") {
162 _proto_gopath = "$root_gen_dir/go"
163 _proto_gen_dir = "$_proto_gopath/src"
164 _rebased_gopath = rebase_path(_proto_gopath)
165
166 pw_python_script(target_name) {
167 metadata = {
168 gopath = [ "GOPATH+=$_rebased_gopath" ]
Alexei Frolovc15a9882019-12-23 14:29:02 -0800169 external_deps = [
170 "github.com/golang/protobuf/proto",
171 "google.golang.org/grpc",
172 ]
Alexei Frolovdef14712019-12-23 13:03:32 -0800173 }
174 script = _gen_script_path
175 args = [
176 "--language",
177 "go",
178 "--module-path",
179 "//",
Alexei Frolove19ebb82020-05-14 17:21:20 -0700180 "--include-file",
181 invoker.include_file,
Alexei Frolovdef14712019-12-23 13:03:32 -0800182 "--out-dir",
183 _proto_gen_dir,
184 ] + get_path_info(invoker.protos, "abspath")
185 inputs = invoker.protos
Alexei Frolove19ebb82020-05-14 17:21:20 -0700186 deps = invoker.deps + invoker.gen_deps
Alexei Frolovdef14712019-12-23 13:03:32 -0800187 stamp = true
188 }
189}
190
Alexei Frolov942adf02019-12-11 17:07:28 -0800191# Generates protobuf code from .proto definitions for various languages.
192#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700193# The generators to use are defined in the pw_protobuf_generators build
194# variable. Each listed generator creates a generated code target called
Alexei Frolov942adf02019-12-11 17:07:28 -0800195#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700196# <target_name>_<generator>
Alexei Frolov942adf02019-12-11 17:07:28 -0800197#
198# For example, with the following definitions:
199#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700200# pw_protobuf_generators = [ "pwpb", "py" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800201#
202# pw_proto_library("my_protos") {
203# sources = [ "foo.proto" ]
204# }
205#
206# Two build targets will be created for the declared "my_protos" target.
207#
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700208# "my_protos_pwpb" <-- C++ source_set containing generated proto code
209# "my_protos_py" <-- Python module containing generated proto code
Alexei Frolov942adf02019-12-11 17:07:28 -0800210#
211# Args:
212# sources: List of input .proto files.
213# deps: List of other pw_proto_library dependencies.
Alexei Frolov942adf02019-12-11 17:07:28 -0800214template("pw_proto_library") {
215 assert(defined(invoker.sources) && invoker.sources != [],
216 "pw_proto_codegen requires .proto source files")
217
Alexei Frolove19ebb82020-05-14 17:21:20 -0700218 # For each proto target, create a file which collects the base directories of
219 # all of its dependencies to list as include paths to protoc.
220 _include_metadata_target = "${target_name}_include_paths"
221 _include_metadata_file = "${target_gen_dir}/${target_name}_includes.txt"
222 generated_file(_include_metadata_target) {
Alexei Frolov942adf02019-12-11 17:07:28 -0800223 if (defined(invoker.deps)) {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700224 # Collect metadata from the include path files of each dependency.
225 deps = process_file_template(invoker.deps, "{{source}}_include_paths")
Alexei Frolov942adf02019-12-11 17:07:28 -0800226 } else {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700227 deps = []
Alexei Frolov942adf02019-12-11 17:07:28 -0800228 }
Alexei Frolove19ebb82020-05-14 17:21:20 -0700229 data_keys = [ "protoc_includes" ]
230 outputs = [ _include_metadata_file ]
231
232 # Indicate this library's base directory for its dependents.
233 metadata = {
234 protoc_includes = [ rebase_path(".", root_out_dir) ]
235 }
236 }
237
238 foreach(_gen, pw_protobuf_generators) {
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700239 _lang_target = "${target_name}_${_gen}"
Alexei Frolov942adf02019-12-11 17:07:28 -0800240
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700241 if (_gen == "pwpb") {
Alexei Frolove19ebb82020-05-14 17:21:20 -0700242 _gen_deps = []
243 if (defined(invoker.deps)) {
244 _gen_deps = process_file_template(invoker.deps, "{{source}}_${_gen}")
245 }
246
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700247 _pw_pwpb_proto_library(_lang_target) {
Alexei Frolov942adf02019-12-11 17:07:28 -0800248 protos = invoker.sources
Alexei Frolove19ebb82020-05-14 17:21:20 -0700249 deps = [ ":$_include_metadata_target" ]
250 include_file = _include_metadata_file
251 gen_deps = _gen_deps
Alexei Frolovbbf164c2019-12-16 12:51:59 -0800252
253 # List the pw_protobuf plugin's files as a dependency to recompile
254 # generated code if they are modified.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700255 protoc_deps = [ "$dir_pw_protobuf:codegen_protoc_plugin" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800256 }
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700257 } else if (_gen == "nanopb") {
258 _pw_nanopb_proto_library(_lang_target) {
259 protos = invoker.sources
Alexei Frolove19ebb82020-05-14 17:21:20 -0700260 deps = [ ":$_include_metadata_target" ]
261 include_file = _include_metadata_file
262 gen_deps = _gen_deps
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700263 }
264 } else if (_gen == "go") {
Alexei Frolovdef14712019-12-23 13:03:32 -0800265 _pw_go_proto_library(_lang_target) {
266 protos = invoker.sources
Alexei Frolove19ebb82020-05-14 17:21:20 -0700267 deps = [ ":$_include_metadata_target" ]
268 include_file = _include_metadata_file
269 gen_deps = _gen_deps
Alexei Frolovdef14712019-12-23 13:03:32 -0800270 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800271 } else {
272 assert(false,
273 string_join(
274 " ",
275 [
276 "pw_proto_codegen doesn't know how to generate code for",
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700277 "generator '$_gen'. Please add support if you require it.",
Alexei Frolov942adf02019-12-11 17:07:28 -0800278 ]))
279 }
280 }
281
282 # If the user attempts to use the target directly instead of one of the
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700283 # generator targets, run a script which prints a nice error message.
Alexei Frolov942adf02019-12-11 17:07:28 -0800284 pw_python_script(target_name) {
285 script = string_join("/",
286 [
287 dir_pw_protobuf_compiler,
288 "py",
289 "pw_protobuf_compiler",
290 "proto_target_invalid.py",
291 ])
292 args = [
293 "--target",
294 target_name,
295 "--dir",
296 get_path_info(".", "abspath"),
297 "--root",
298 "//",
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700299 ] + pw_protobuf_generators
Alexei Frolov942adf02019-12-11 17:07:28 -0800300 stamp = true
301 }
302}