DCCL v4
Loading...
Searching...
No Matches
pb_plugin.cpp
1// Copyright 2015-2023:
2// GobySoft, LLC (2013-)
3// Community contributors (see AUTHORS file)
4// File authors:
5// Toby Schneider <toby@gobysoft.org>
6// Stephanie Petillo <stephanie@gobysoft.org>
7// Nathan Knotts <nknotts@gmail.com>
8//
9//
10// This file is part of the Dynamic Compact Control Language Library
11// ("DCCL").
12//
13// DCCL is free software: you can redistribute it and/or modify
14// it under the terms of the GNU Lesser General Public License as published by
15// the Free Software Foundation, either version 2.1 of the License, or
16// (at your option) any later version.
17//
18// DCCL is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU Lesser General Public License for more details.
22//
23// You should have received a copy of the GNU Lesser General Public License
24// along with DCCL. If not, see <http://www.gnu.org/licenses/>.
25#include "gen_units_class_plugin.h"
26#include "option_extensions.pb.h"
27#include <boost/algorithm/string.hpp>
28#include <fstream>
29#include <google/protobuf/compiler/cpp/cpp_generator.h>
30#include <google/protobuf/compiler/plugin.h>
31#include <google/protobuf/descriptor.h>
32#include <google/protobuf/io/printer.h>
33#include <google/protobuf/io/zero_copy_stream.h>
34#include <iostream>
35#include <memory>
36#include <set>
37#include <sstream>
38
39std::set<std::string> systems_to_include_;
40std::set<std::string> base_units_to_include_;
41std::set<std::string> custom_unit_headers_to_include_;
42std::string filename_h_;
43std::string load_file_cpp_;
44std::shared_ptr<std::fstream> load_file_output_;
45
46std::string load_file_base_{
47 R"DEL(#include <dccl/codec.h>
48
49// DO NOT REMOVE: required by loader code
50std::vector<const google::protobuf::Descriptor*> descriptors;
51template<typename PB> struct DCCLLoader { DCCLLoader() { descriptors.push_back(PB::descriptor()); }};
52// END: required by loader code
53
54
55extern "C"
56{
57 void dccl3_load(dccl::Codec* dccl)
58 {
59 // DO NOT REMOVE: required by loader code
60 for(auto d : descriptors)
61 dccl->load(d);
62 // END: required by loader code
63 }
64
65 void dccl3_unload(dccl::Codec* dccl)
66 {
67 // DO NOT REMOVE: required by loader code
68 for(auto d : descriptors)
69 dccl->unload(d);
70 // END: required by loader code
71 }
72}
73
74// BEGIN (TO END OF FILE): AUTOGENERATED LOADERS
75)DEL"};
76
77class DCCLGenerator : public google::protobuf::compiler::cpp::CppGenerator
78{
79 public:
80 DCCLGenerator() = default;
81 ~DCCLGenerator() override = default;
82
83 // implements CodeGenerator ----------------------------------------
84 bool Generate(const google::protobuf::FileDescriptor* file, const std::string& parameter,
85 google::protobuf::compiler::GeneratorContext* generator_context,
86 std::string* error) const override;
87
88 private:
89 void generate_message(
90 const google::protobuf::Descriptor* desc,
91 google::protobuf::compiler::GeneratorContext* generator_context,
92 std::shared_ptr<std::string> message_unit_system = std::shared_ptr<std::string>()) const;
93 void generate_field(const google::protobuf::FieldDescriptor* field,
94 google::protobuf::io::Printer* printer,
95 std::shared_ptr<std::string> message_unit_system) const;
96 bool check_field_type(const google::protobuf::FieldDescriptor* field) const;
97
98 void generate_load_file_headers() const;
99 void generate_load_file_message_loader(const google::protobuf::Descriptor* desc) const;
100};
101
102bool DCCLGenerator::check_field_type(const google::protobuf::FieldDescriptor* field) const
103{
104 bool is_integer = field->cpp_type() == google::protobuf::FieldDescriptor::CPPTYPE_INT32 ||
105 field->cpp_type() == google::protobuf::FieldDescriptor::CPPTYPE_INT64 ||
106 field->cpp_type() == google::protobuf::FieldDescriptor::CPPTYPE_UINT32 ||
107 field->cpp_type() == google::protobuf::FieldDescriptor::CPPTYPE_UINT64;
108
109 bool is_float = field->cpp_type() == google::protobuf::FieldDescriptor::CPPTYPE_DOUBLE ||
110 field->cpp_type() == google::protobuf::FieldDescriptor::CPPTYPE_FLOAT;
111
112 if (!is_float && !is_integer)
113 {
114 throw std::runtime_error("Can only use (dccl.field).base_dimensions on numeric fields");
115 }
116 return is_integer;
117}
118
119bool DCCLGenerator::Generate(const google::protobuf::FileDescriptor* file,
120 const std::string& parameter,
121 google::protobuf::compiler::GeneratorContext* generator_context,
122 std::string* error) const
123{
124#ifdef INCLUDE_CPP
125 google::protobuf::compiler::cpp::CppGenerator::Generate(file, parameter, generator_context,
126 error);
127#endif
128
129 std::vector<std::pair<std::string, std::string>> options;
130 google::protobuf::compiler::ParseGeneratorParameter(parameter, &options);
131
132 for (const auto& option : options)
133 {
134 const auto& key = option.first;
135 const auto& value = option.second;
136 if (key == "dccl3_load_file")
137 {
138 load_file_cpp_ = value;
139 load_file_output_ = std::make_shared<std::fstream>(
140 load_file_cpp_, std::ios::in | std::ios::out | std::ios::app);
141
142 if (!load_file_output_->is_open())
143 {
144 *error = "Failed to open dccl3_load_file: " + load_file_cpp_;
145 return false;
146 }
147 }
148 else
149 {
150 *error = "Unknown parameter: " + key;
151 return false;
152 }
153 }
154
155 try
156 {
157 const std::string& filename = file->name();
158 filename_h_ = filename.substr(0, filename.find(".proto")) + ".pb.h";
159 // std::string filename_cc = filename.substr(0, filename.find(".proto")) + ".pb.cc";
160
161 if (load_file_output_)
162 generate_load_file_headers();
163
164 for (int message_i = 0, message_n = file->message_type_count(); message_i < message_n;
165 ++message_i)
166 {
167 try
168 {
169 generate_message(file->message_type(message_i), generator_context);
170 }
171 catch (std::exception& e)
172 {
173 throw(
174 std::runtime_error(std::string("Failed to generate DCCL code: \n") + e.what()));
175 }
176 }
177
178 std::shared_ptr<google::protobuf::io::ZeroCopyOutputStream> include_output(
179 generator_context->OpenForInsert(filename_h_, "includes"));
180 google::protobuf::io::Printer include_printer(include_output.get(), '$');
181 std::stringstream includes_ss;
182
183 includes_ss << "#include <boost/units/quantity.hpp>" << std::endl;
184 includes_ss << "#include <boost/units/absolute.hpp>" << std::endl;
185 includes_ss << "#include <boost/units/dimensionless_type.hpp>" << std::endl;
186 includes_ss << "#include <boost/units/make_scaled_unit.hpp>" << std::endl;
187
188 for (const auto& it : systems_to_include_) include_units_headers(it, includes_ss);
189 for (const auto& it : base_units_to_include_) include_base_unit_headers(it, includes_ss);
190 for (const auto& it : custom_unit_headers_to_include_)
191 include_custom_unit_headers(it, includes_ss);
192
193 include_printer.Print(includes_ss.str().c_str());
194
195 return true;
196 }
197 catch (std::exception& e)
198 {
199 *error = e.what();
200 return false;
201 }
202}
203
204void DCCLGenerator::generate_message(
205 const google::protobuf::Descriptor* desc,
206 google::protobuf::compiler::GeneratorContext* generator_context,
207 std::shared_ptr<std::string> message_unit_system) const
208{
209 try
210 {
211 {
212 std::shared_ptr<google::protobuf::io::ZeroCopyOutputStream> output(
213 generator_context->OpenForInsert(filename_h_, "class_scope:" + desc->full_name()));
214 google::protobuf::io::Printer printer(output.get(), '$');
215
216 if (desc->options().HasExtension(dccl::msg))
217 {
218 if (desc->options().GetExtension(dccl::msg).id() != 0)
219 {
220 std::stringstream id_enum;
221 id_enum << "enum DCCLParameters { DCCL_ID = "
222 << desc->options().GetExtension(dccl::msg).id() << ", "
223 << " DCCL_MAX_BYTES = "
224 << desc->options().GetExtension(dccl::msg).max_bytes() << " };\n";
225 printer.Print(id_enum.str().c_str());
226
227 if (load_file_output_)
228 generate_load_file_message_loader(desc);
229 }
230
231 // set message level unit system - used if fields do not specify
232 const dccl::DCCLMessageOptions& dccl_msg_options =
233 desc->options().GetExtension(dccl::msg);
234 if (dccl_msg_options.has_unit_system())
235 {
236 message_unit_system.reset(new std::string(dccl_msg_options.unit_system()));
237 systems_to_include_.insert(dccl_msg_options.unit_system());
238 }
239 }
240
241 for (int field_i = 0, field_n = desc->field_count(); field_i < field_n; ++field_i)
242 {
243 generate_field(desc->field(field_i), &printer, message_unit_system);
244 }
245
246 for (int nested_type_i = 0, nested_type_n = desc->nested_type_count();
247 nested_type_i < nested_type_n; ++nested_type_i)
248 generate_message(desc->nested_type(nested_type_i), generator_context,
249 message_unit_system);
250 }
251 }
252 catch (std::exception& e)
253 {
254 throw(std::runtime_error(std::string("Message: \n" + desc->full_name() + "\n" + e.what())));
255 }
256}
257
258void DCCLGenerator::generate_field(const google::protobuf::FieldDescriptor* field,
259 google::protobuf::io::Printer* printer,
260 std::shared_ptr<std::string> message_unit_system) const
261{
262 try
263 {
264 const dccl::DCCLFieldOptions& dccl_field_options =
265 field->options().GetExtension(dccl::field);
266
267 if (!dccl_field_options.has_units())
268 {
269 return;
270 }
271
272 if ((dccl_field_options.units().has_base_dimensions() &&
273 dccl_field_options.units().has_derived_dimensions()) ||
274 (dccl_field_options.units().has_base_dimensions() &&
275 dccl_field_options.units().has_unit()) ||
276 (dccl_field_options.units().has_unit() &&
277 dccl_field_options.units().has_derived_dimensions()))
278 {
279 throw(std::runtime_error("May define either (dccl.field).units.base_dimensions or "
280 "(dccl.field).units.derived_dimensions or "
281 "(dccl.field).units.unit, but not more than one."));
282 }
283 else if (dccl_field_options.units().has_unit())
284 {
285 std::stringstream new_methods;
286
287 construct_units_typedef_from_base_unit(
288 field->name(), dccl_field_options.units().unit(),
289 dccl_field_options.units().relative_temperature(),
290 dccl_field_options.units().prefix(), new_methods);
291 construct_field_class_plugin(field->name(), new_methods,
292 dccl::units::get_field_type_name(field->cpp_type()),
293 field->is_repeated());
294 printer->Print(new_methods.str().c_str());
295 base_units_to_include_.insert(dccl_field_options.units().unit());
296 }
297 else if (dccl_field_options.units().has_custom())
298 {
299 std::stringstream new_methods;
300
301 construct_units_typedef_from_custom_unit(
302 field->name(), dccl_field_options.units().custom().unit(),
303 dccl_field_options.units().relative_temperature(),
304 dccl_field_options.units().prefix(), new_methods);
305 construct_field_class_plugin(field->name(), new_methods,
306 dccl::units::get_field_type_name(field->cpp_type()),
307 field->is_repeated());
308 printer->Print(new_methods.str().c_str());
309
310 if (dccl_field_options.units().custom().has_header())
311 custom_unit_headers_to_include_.insert(
312 dccl_field_options.units().custom().header());
313 }
314 else if (dccl_field_options.units().has_base_dimensions())
315 {
316 std::stringstream new_methods;
317
318 std::vector<double> powers;
319 std::vector<std::string> short_dimensions;
320 std::vector<std::string> dimensions;
321 if (dccl::units::parse_base_dimensions(
322 dccl_field_options.units().base_dimensions().begin(),
323 dccl_field_options.units().base_dimensions().end(), powers, short_dimensions,
324 dimensions))
325 {
326 if (!dccl_field_options.units().has_system() && !message_unit_system)
327 throw(std::runtime_error(
328 std::string("Field must have 'system' defined or message must have "
329 "'unit_system' defined when using 'base_dimensions'.")));
330
331 // default to system set in the field, otherwise use the system set at the message level
332 const std::string& unit_system =
333 (!dccl_field_options.units().has_system() && message_unit_system)
334 ? *message_unit_system
335 : dccl_field_options.units().system();
336
337 construct_base_dims_typedef(dimensions, powers, field->name(), unit_system,
338 dccl_field_options.units().relative_temperature(),
339 dccl_field_options.units().prefix(), new_methods);
340
341 construct_field_class_plugin(field->name(), new_methods,
342 dccl::units::get_field_type_name(field->cpp_type()),
343 field->is_repeated());
344 printer->Print(new_methods.str().c_str());
345 systems_to_include_.insert(unit_system);
346 }
347 else
348 {
349 throw(std::runtime_error(std::string("Failed to parse base_dimensions string: \"" +
350 dccl_field_options.units().base_dimensions() +
351 "\"")));
352 }
353 }
354 else if (dccl_field_options.units().has_derived_dimensions())
355 {
356 std::stringstream new_methods;
357
358 std::vector<std::string> operators;
359 std::vector<std::string> dimensions;
360 if (dccl::units::parse_derived_dimensions(
361 dccl_field_options.units().derived_dimensions().begin(),
362 dccl_field_options.units().derived_dimensions().end(), operators, dimensions))
363 {
364 if (!dccl_field_options.units().has_system() && !message_unit_system)
365 throw(std::runtime_error(
366 std::string("Field must have 'system' defined or message must have "
367 "'unit_system' defined when using 'derived_dimensions'.")));
368 const std::string& unit_system =
369 (!dccl_field_options.units().has_system() && message_unit_system)
370 ? *message_unit_system
371 : dccl_field_options.units().system();
372
373 construct_derived_dims_typedef(dimensions, operators, field->name(), unit_system,
374 dccl_field_options.units().relative_temperature(),
375 dccl_field_options.units().prefix(), new_methods);
376
377 construct_field_class_plugin(field->name(), new_methods,
378 dccl::units::get_field_type_name(field->cpp_type()),
379 field->is_repeated());
380 printer->Print(new_methods.str().c_str());
381 systems_to_include_.insert(unit_system);
382 }
383 else
384 {
385 throw(std::runtime_error(
386 std::string("Failed to parse derived_dimensions string: \"" +
387 dccl_field_options.units().derived_dimensions() + "\"")));
388 }
389 }
390 }
391 catch (std::exception& e)
392 {
393 throw(
394 std::runtime_error(std::string("Field: \n" + field->DebugString() + "\n" + e.what())));
395 }
396}
397
398void DCCLGenerator::generate_load_file_headers() const
399{
400 bool file_is_empty = (load_file_output_->peek() == std::ifstream::traits_type::eof());
401 load_file_output_->clear();
402
403 bool header_already_written = false;
404 if (file_is_empty)
405 {
406 *load_file_output_ << load_file_base_ << std::flush;
407 }
408 else
409 {
410 for (std::string line; getline(*load_file_output_, line);)
411 {
412 if (line.find("\"" + filename_h_ + "\"") != std::string::npos)
413 {
414 header_already_written = true;
415 break;
416 }
417 }
418 }
419 // clear EOF
420 load_file_output_->clear();
421 // seek to the end
422 load_file_output_->seekp(0, std::ios_base::end);
423
424 if (!header_already_written)
425 *load_file_output_ << "#include \"" << filename_h_ << "\"" << std::endl;
426}
427
428void DCCLGenerator::generate_load_file_message_loader(
429 const google::protobuf::Descriptor* desc) const
430{
431 // seek to the beginning
432 load_file_output_->seekp(0, std::ios_base::beg);
433
434 // cpp class name
435 std::string cpp_name = desc->full_name();
436 boost::algorithm::replace_all(cpp_name, ".", "::");
437
438 // lower case class name with underscores
439 std::string loader_name = desc->full_name() + "_loader";
440 boost::algorithm::replace_all(loader_name, ".", "_");
441 boost::algorithm::to_lower(loader_name);
442
443 bool loader_already_written = false;
444 for (std::string line; getline(*load_file_output_, line);)
445 {
446 if (line.find("<" + cpp_name + ">") != std::string::npos)
447 {
448 loader_already_written = true;
449 break;
450 }
451 }
452 load_file_output_->clear();
453 load_file_output_->seekp(0, std::ios_base::end);
454
455 if (!loader_already_written)
456 *load_file_output_ << "DCCLLoader<" << cpp_name << "> " << loader_name << ";" << std::endl;
457}
458
459int main(int argc, char* argv[])
460{
461 DCCLGenerator generator;
462 return google::protobuf::compiler::PluginMain(argc, argv, &generator);
463}