blob: 94d0c6c5addbfac4e419fed8bcd2c599bed9f0c5 [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 Frolovb499d3f2020-10-28 13:00:08 -070017import("$dir_pw_build/error.gni")
Alexei Frolov05d8ef22020-06-08 10:32:29 -070018import("$dir_pw_build/input_group.gni")
Alexei Frolova4c0aee2020-12-01 13:48:48 -080019import("$dir_pw_build/python.gni")
Wyatt Hepler51ded742020-10-19 14:45:27 -070020import("$dir_pw_build/python_action.gni")
Alexei Frolovedd2f142020-06-09 19:11:27 -070021import("$dir_pw_build/target_types.gni")
Alexei Frolovdd6fa5c2020-08-11 10:04:01 -070022import("$dir_pw_third_party/nanopb/nanopb.gni")
Wyatt Heplerd49f8fe2020-10-15 10:13:47 -070023
Wyatt Heplerd9336a42020-11-10 09:47:30 -080024# Variables forwarded from the public pw_proto_library template to the final
25# pw_source_set.
Wyatt Heplere0c4fdc2020-05-29 17:26:19 -070026_forwarded_vars = [
27 "testonly",
28 "visibility",
29]
30
Wyatt Heplerd9336a42020-11-10 09:47:30 -080031# Internal template that invokes protoc with a pw_python_action. This should not
32# be used outside of this file; use pw_proto_library instead.
33#
34# This creates the internal GN target $target_name.$language._gen that compiles
35# proto files with protoc.
36template("_pw_invoke_protoc") {
37 _output = rebase_path(get_target_outputs(":${invoker.base_target}._metadata"))
38
39 pw_python_action("$target_name._gen") {
40 forward_variables_from(invoker, [ "metadata" ])
41 script =
42 "$dir_pw_protobuf_compiler/py/pw_protobuf_compiler/generate_protos.py"
43
Wyatt Hepler438caa02021-01-15 17:13:11 -080044 python_deps = [ "$dir_pw_protobuf_compiler/py" ]
45 if (defined(invoker.python_deps)) {
46 python_deps += invoker.python_deps
47 }
48
Wyatt Heplerd9336a42020-11-10 09:47:30 -080049 deps = [
50 ":${invoker.base_target}._metadata",
51 ":${invoker.base_target}._inputs",
52 ] + invoker.deps
53
54 args = [
55 "--language",
56 invoker.language,
57 "--module-path",
58 rebase_path("."),
59 "--include-file",
60 _output[0],
61 "--out-dir",
62 rebase_path(invoker.gen_dir),
63 ] + rebase_path(invoker.sources)
64
65 inputs = invoker.sources
66
67 if (defined(invoker.plugin)) {
68 inputs += [ invoker.plugin ]
69 args += [ "--plugin-path=" + rebase_path(invoker.plugin) ]
70 }
71
72 if (defined(invoker.include_paths)) {
73 args += [
74 "--include-paths",
75 string_join(";", rebase_path(invoker.include_paths)),
76 ]
77 }
78
79 outputs = []
80 foreach(extension, invoker.output_extensions) {
81 foreach(proto,
82 rebase_path(invoker.sources, get_path_info(".", "abspath"))) {
83 _output = string_replace(proto, ".proto", extension)
84 outputs += [ "${invoker.gen_dir}/$_output" ]
85 }
86 }
87
88 if (outputs == []) {
89 stamp = true
90 }
91
92 visibility = [ ":*" ]
93 }
94}
95
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070096# Generates pw_protobuf C++ code for proto files, creating a source_set of the
97# generated files. This is internal and should not be used outside of this file.
98# Use pw_proto_library instead.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -070099template("_pw_pwpb_proto_library") {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800100 _pw_invoke_protoc(target_name) {
101 forward_variables_from(invoker, "*", _forwarded_vars)
102 language = "pwpb"
103 plugin = "$dir_pw_protobuf/py/pw_protobuf/plugin.py"
Wyatt Hepler438caa02021-01-15 17:13:11 -0800104 python_deps = [ "$dir_pw_protobuf/py" ]
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800105 output_extensions = [ ".pwpb.h" ]
Alexei Frolov942adf02019-12-11 17:07:28 -0800106 }
107
108 # Create a library with the generated source files.
Alexei Frolovedd2f142020-06-09 19:11:27 -0700109 pw_source_set(target_name) {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800110 forward_variables_from(invoker, _forwarded_vars)
111 public_configs = [ ":${invoker.base_target}._include_path" ]
112 deps = [ ":$target_name._gen" ]
113 public_deps = [ dir_pw_protobuf ] + invoker.deps
114 sources = get_target_outputs(":$target_name._gen")
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700115 public = filter_include(sources, [ "*.pwpb.h" ])
116 }
117}
118
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700119# Generates nanopb RPC code for proto files, creating a source_set of the
120# generated files. This is internal and should not be used outside of this file.
121# Use pw_proto_library instead.
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700122template("_pw_nanopb_rpc_proto_library") {
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700123 # Create a target which runs protoc configured with the nanopb_rpc plugin to
124 # generate the C++ proto RPC headers.
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800125 _pw_invoke_protoc(target_name) {
126 forward_variables_from(invoker, "*", _forwarded_vars)
127 language = "nanopb_rpc"
128 plugin = "$dir_pw_rpc/py/pw_rpc/plugin_nanopb.py"
Wyatt Hepler438caa02021-01-15 17:13:11 -0800129 python_deps = [ "$dir_pw_rpc/py" ]
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800130 include_paths = [ "$dir_pw_third_party_nanopb/generator/proto" ]
131 output_extensions = [ ".rpc.pb.h" ]
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700132 }
133
134 # Create a library with the generated source files.
135 pw_source_set(target_name) {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800136 forward_variables_from(invoker, _forwarded_vars)
137 public_configs = [ ":${invoker.base_target}._include_path" ]
138 deps = [ ":$target_name._gen" ]
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700139 public_deps = [
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800140 ":${invoker.base_target}.nanopb",
Wyatt Heplercbd09c22020-09-15 11:17:24 -0700141 "$dir_pw_rpc:server",
Alexei Frolova4d71502020-10-14 12:43:14 -0700142 "$dir_pw_rpc/nanopb:method_union",
Alexei Frolovdd6fa5c2020-08-11 10:04:01 -0700143 "$dir_pw_third_party/nanopb",
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800144 ] + invoker.deps
145 public = get_target_outputs(":$target_name._gen")
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700146 }
147}
148
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700149# Generates nanopb code for proto files, creating a source_set of the generated
150# files. This is internal and should not be used outside of this file. Use
151# pw_proto_library instead.
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700152template("_pw_nanopb_proto_library") {
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700153 # Create a target which runs protoc configured with the nanopb plugin to
154 # generate the C proto sources.
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800155 _pw_invoke_protoc(target_name) {
156 forward_variables_from(invoker, "*", _forwarded_vars)
157 language = "nanopb"
158 plugin = "$dir_pw_third_party_nanopb/generator/protoc-gen-nanopb"
159 include_paths = [ "$dir_pw_third_party_nanopb/generator/proto" ]
160 output_extensions = [
161 ".pb.h",
162 ".pb.c",
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700163 ]
164 }
165
166 # Create a library with the generated source files.
Alexei Frolovedd2f142020-06-09 19:11:27 -0700167 pw_source_set(target_name) {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800168 forward_variables_from(invoker, _forwarded_vars)
169 public_configs = [ ":${invoker.base_target}._include_path" ]
170 deps = [ ":$target_name._gen" ]
171 public_deps = [ "$dir_pw_third_party/nanopb" ] + invoker.deps
172 sources = get_target_outputs(":$target_name._gen")
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700173 public = filter_include(sources, [ "*.pb.h" ])
Alexei Frolov942adf02019-12-11 17:07:28 -0800174 }
175}
176
Alexei Frolovc912ea72020-10-26 08:43:27 -0700177# Generates raw RPC code for proto files, creating a source_set of the generated
178# files. This is internal and should not be used outside of this file. Use
179# pw_proto_library instead.
Alexei Frolovc912ea72020-10-26 08:43:27 -0700180template("_pw_raw_rpc_proto_library") {
Alexei Frolovc912ea72020-10-26 08:43:27 -0700181 # Create a target which runs protoc configured with the nanopb_rpc plugin to
182 # generate the C++ proto RPC headers.
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800183 _pw_invoke_protoc(target_name) {
184 forward_variables_from(invoker, "*", _forwarded_vars)
185 language = "raw_rpc"
186 plugin = "$dir_pw_rpc/py/pw_rpc/plugin_raw.py"
Wyatt Hepler438caa02021-01-15 17:13:11 -0800187 python_deps = [ "$dir_pw_rpc/py" ]
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800188 output_extensions = [ ".raw_rpc.pb.h" ]
Alexei Frolovc912ea72020-10-26 08:43:27 -0700189 }
190
191 # Create a library with the generated source files.
192 pw_source_set(target_name) {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800193 forward_variables_from(invoker, _forwarded_vars)
194 public_configs = [ ":${invoker.base_target}._include_path" ]
195 deps = [ ":$target_name._gen" ]
Alexei Frolovc912ea72020-10-26 08:43:27 -0700196 public_deps = [
197 "$dir_pw_rpc:server",
198 "$dir_pw_rpc/raw:method_union",
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800199 ] + invoker.deps
200 public = get_target_outputs(":$target_name._gen")
Alexei Frolovc912ea72020-10-26 08:43:27 -0700201 }
202}
203
Alexei Frolovdef14712019-12-23 13:03:32 -0800204# Generates Go code for proto files, listing the proto output directory in the
205# metadata variable GOPATH. Internal use only.
Alexei Frolovdef14712019-12-23 13:03:32 -0800206template("_pw_go_proto_library") {
207 _proto_gopath = "$root_gen_dir/go"
Alexei Frolovdef14712019-12-23 13:03:32 -0800208
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800209 _pw_invoke_protoc(target_name) {
210 forward_variables_from(invoker, "*")
211 language = "go"
Alexei Frolovdef14712019-12-23 13:03:32 -0800212 metadata = {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800213 gopath = [ "GOPATH+=" + rebase_path(_proto_gopath) ]
Alexei Frolovc15a9882019-12-23 14:29:02 -0800214 external_deps = [
215 "github.com/golang/protobuf/proto",
216 "google.golang.org/grpc",
217 ]
Alexei Frolovdef14712019-12-23 13:03:32 -0800218 }
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800219 output_extensions = [] # Don't enumerate the generated .go files.
220 gen_dir = "$_proto_gopath/src"
221 }
222
223 group(target_name) {
224 deps = [ ":$target_name._gen" ]
Alexei Frolovdef14712019-12-23 13:03:32 -0800225 }
226}
227
Alexei Frolova4c0aee2020-12-01 13:48:48 -0800228# Generates Python code for proto files, creating a pw_python_package containing
229# the generated files. This is internal and should not be used outside of this
230# file. Use pw_proto_library instead.
231template("_pw_python_proto_library") {
232 _target = target_name
233 _package_dir = ""
234
235 foreach(_rebased_proto_path, rebase_path(invoker.sources, ".")) {
236 _path_components = []
237 _path_components = string_split(_rebased_proto_path, "/")
238
239 assert(_path_components != [ _rebased_proto_path ] &&
240 _path_components[0] != "..",
241 "Sources in a pw_proto_library must live in subdirectories " +
242 "of where it is defined")
243
244 if (_package_dir == "") {
245 _package_dir = _path_components[0]
246 } else {
247 assert(_path_components[0] == _package_dir,
248 "All .proto sources in a pw_proto_library must live " +
249 "in the same directory tree")
250 }
251 }
252
253 _pw_invoke_protoc(target_name) {
254 forward_variables_from(invoker, "*", _forwarded_vars)
255 language = "python"
256 output_extensions = [ "_pb2.py" ]
257 }
258
259 _setup_py = "${invoker.gen_dir}/setup.py"
260
261 # Create the setup and init files for the Python package.
262 pw_python_action(target_name + "._package_gen") {
263 script = "$dir_pw_protobuf_compiler/py/pw_protobuf_compiler/generate_python_package.py"
264 args = [
265 "--setup",
266 rebase_path(_setup_py),
267 "--package",
268 _package_dir,
269 ] + rebase_path(get_path_info(invoker.sources, "dir"), ".")
270 public_deps = [ ":$_target._gen" ]
271 stamp = true
272 }
273
274 # Create a Python package with the generated source files.
275 pw_python_package(target_name) {
276 forward_variables_from(invoker, _forwarded_vars)
277 setup = [ _setup_py ]
278 sources = get_target_outputs(":$target_name._gen")
279 python_deps = invoker.deps
280 other_deps = [ ":$_target._package_gen" ]
281 _pw_generated = true
282 }
283}
284
Alexei Frolov942adf02019-12-11 17:07:28 -0800285# Generates protobuf code from .proto definitions for various languages.
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700286# For each supported generator, creates a sub-target named:
Alexei Frolov942adf02019-12-11 17:07:28 -0800287#
Alexei Frolov8e30d462020-10-22 13:54:36 -0700288# <target_name>.<generator>
Alexei Frolov942adf02019-12-11 17:07:28 -0800289#
Alexei Frolov942adf02019-12-11 17:07:28 -0800290# Args:
291# sources: List of input .proto files.
292# deps: List of other pw_proto_library dependencies.
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700293# inputs: Other files on which the protos depend (e.g. nanopb .options files).
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700294#
Alexei Frolov942adf02019-12-11 17:07:28 -0800295template("pw_proto_library") {
296 assert(defined(invoker.sources) && invoker.sources != [],
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700297 "pw_proto_library requires .proto source files")
Alexei Frolov942adf02019-12-11 17:07:28 -0800298
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800299 _common = {
300 base_target = target_name
301 gen_dir = "$target_gen_dir/protos"
302 sources = invoker.sources
303 }
304
305 if (defined(invoker.deps)) {
306 _deps = invoker.deps
307 } else {
308 _deps = []
309 }
310
Alexei Frolove19ebb82020-05-14 17:21:20 -0700311 # For each proto target, create a file which collects the base directories of
312 # all of its dependencies to list as include paths to protoc.
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800313 generated_file("$target_name._metadata") {
314 # Collect metadata from the include path files of each dependency.
315 deps = process_file_template(_deps, "{{source}}._metadata")
316
Alexei Frolove19ebb82020-05-14 17:21:20 -0700317 data_keys = [ "protoc_includes" ]
Wyatt Hepler7c61caf2020-11-24 11:29:05 -0800318 outputs = [ "$target_gen_dir/${_common.base_target}_includes.txt" ]
Alexei Frolove19ebb82020-05-14 17:21:20 -0700319
320 # Indicate this library's base directory for its dependents.
321 metadata = {
Alexei Frolovf79d2272020-06-18 13:37:38 -0700322 protoc_includes = [ rebase_path(".") ]
Alexei Frolove19ebb82020-05-14 17:21:20 -0700323 }
324 }
325
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800326 # Toss any additional inputs into an input group dependency.
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700327 if (defined(invoker.inputs)) {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800328 pw_input_group("$target_name._inputs") {
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700329 inputs = invoker.inputs
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800330 visibility = [ ":*" ]
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700331 }
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800332 } else {
333 group("$target_name._inputs") {
334 visibility = [ ":*" ]
335 }
Alexei Frolov05d8ef22020-06-08 10:32:29 -0700336 }
337
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800338 # Create a config with the generated proto directory, which is used for C++.
339 config("$target_name._include_path") {
340 include_dirs = [ _common.gen_dir ]
341 visibility = [ ":*" ]
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700342 }
343
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700344 # Enumerate all of the protobuf generator targets.
Alexei Frolovc4b62ec2020-07-13 08:35:10 -0700345
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800346 _pw_pwpb_proto_library("$target_name.pwpb") {
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700347 forward_variables_from(invoker, _forwarded_vars)
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800348 forward_variables_from(_common, "*")
349 deps = process_file_template(_deps, "{{source}}.pwpb")
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700350 }
351
352 if (dir_pw_third_party_nanopb != "") {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800353 _pw_nanopb_rpc_proto_library("$target_name.nanopb_rpc") {
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700354 forward_variables_from(invoker, _forwarded_vars)
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800355 forward_variables_from(_common, "*")
356 deps = process_file_template(_deps, "{{source}}.nanopb_rpc")
Alexei Frolovc4b62ec2020-07-13 08:35:10 -0700357 }
358
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800359 _pw_nanopb_proto_library("$target_name.nanopb") {
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700360 forward_variables_from(invoker, _forwarded_vars)
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800361 forward_variables_from(_common, "*")
362 deps = process_file_template(_deps, "{{source}}.nanopb")
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700363 }
364 } else {
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800365 pw_error("$target_name.nanopb_rpc") {
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700366 message =
367 "\$dir_pw_third_party_nanopb must be set to generate nanopb RPC code."
Alexei Frolov8185c822020-06-12 10:45:04 -0700368 }
Alexei Frolov942adf02019-12-11 17:07:28 -0800369
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800370 pw_error("$target_name.nanopb") {
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700371 message =
372 "\$dir_pw_third_party_nanopb must be set to compile nanopb protobufs."
Alexei Frolov942adf02019-12-11 17:07:28 -0800373 }
374 }
375
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800376 _pw_raw_rpc_proto_library("$target_name.raw_rpc") {
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700377 forward_variables_from(invoker, _forwarded_vars)
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800378 forward_variables_from(_common, "*", [ "deps" ])
379 deps = process_file_template(_deps, "{{source}}.raw_rpc")
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700380 }
381
Wyatt Heplerd9336a42020-11-10 09:47:30 -0800382 _pw_go_proto_library("$target_name.go") {
383 sources = invoker.sources
384 deps = process_file_template(_deps, "{{source}}.go")
385 base_target = _common.base_target
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700386 }
387
Alexei Frolov38cad0c2020-12-03 12:36:24 -0800388 _pw_python_proto_library("$target_name.python") {
389 sources = invoker.sources
390 forward_variables_from(_common, "*")
391 deps = process_file_template(_deps, "{{source}}.python")
392 base_target = _common.base_target
Alexei Frolova4c0aee2020-12-01 13:48:48 -0800393 }
394
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700395 # All supported pw_protobuf generators.
396 _protobuf_generators = [
397 "pwpb",
398 "nanopb",
Alexei Frolov79b7cb02020-07-06 13:51:43 -0700399 "nanopb_rpc",
Alexei Frolovc912ea72020-10-26 08:43:27 -0700400 "raw_rpc",
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700401 "go",
Alexei Frolova4c0aee2020-12-01 13:48:48 -0800402 "python",
Wyatt Heplerb4b73a62020-05-27 15:17:27 -0700403 ]
404
Alexei Frolov942adf02019-12-11 17:07:28 -0800405 # If the user attempts to use the target directly instead of one of the
Alexei Frolovf39cd8b2020-04-13 17:59:20 -0700406 # generator targets, run a script which prints a nice error message.
Wyatt Heplerc8e05a42020-10-19 14:49:39 -0700407 pw_python_action(target_name) {
Alexei Frolov942adf02019-12-11 17:07:28 -0800408 script = string_join("/",
409 [
410 dir_pw_protobuf_compiler,
411 "py",
412 "pw_protobuf_compiler",
413 "proto_target_invalid.py",
414 ])
415 args = [
416 "--target",
417 target_name,
418 "--dir",
419 get_path_info(".", "abspath"),
420 "--root",
421 "//",
Alexei Frolovb499d3f2020-10-28 13:00:08 -0700422 ] + _protobuf_generators
Alexei Frolov942adf02019-12-11 17:07:28 -0800423 stamp = true
424 }
425}