// Protocol Buffers - Google's data interchange format // Copyright 2023 Google LLC. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file or at // https://developers.google.com/open-source/licenses/bsd #include #include #include #include #include "absl/container/flat_hash_set.h" #include "absl/log/absl_check.h" #include "absl/memory/memory.h" #include "absl/strings/cord.h" #include "absl/strings/string_view.h" #include "absl/strings/substitute.h" #include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/plugin.h" #include "upb/mem/arena.hpp" #include "upb/reflection/def.hpp" #include "upb_generator/common.h" #include "upb_generator/common/names.h" #include "upb_generator/file_layout.h" #include "upb_generator/minitable/generator.h" #include "upb_generator/minitable/names_internal.h" #include "upb_generator/plugin.h" // Must be last. #include "upb/port/def.inc" namespace upb { namespace generator { std::string SourceFilename(upb::FileDefPtr file) { return StripExtension(file.name()) + ".upb_minitable.c"; } void GenerateFile(const DefPoolPair& pools, upb::FileDefPtr file, const MiniTableOptions& options, google::protobuf::compiler::GeneratorContext* context) { Output h_output; WriteMiniTableHeader(pools, file, options, h_output); { auto stream = absl::WrapUnique( context->Open(MiniTableHeaderFilename(file.name(), false))); ABSL_CHECK(stream->WriteCord(absl::Cord(h_output.output()))); } Output c_output; WriteMiniTableSource(pools, file, options, c_output); { auto stream = absl::WrapUnique(context->Open(SourceFilename(file))); ABSL_CHECK(stream->WriteCord(absl::Cord(c_output.output()))); } if (options.one_output_per_message) { WriteMiniTableMultipleSources(pools, file, options, context); } } bool ParseOptions(MiniTableOptions* options, absl::string_view parameter, std::string* error) { for (const auto& pair : ParseGeneratorParameter(parameter)) { if (pair.first == "bootstrap_stage") { options->bootstrap = true; } else if (pair.first == "experimental_strip_nonfunctional_codegen") { options->strip_nonfunctional_codegen = true; } else if (pair.first == "one_output_per_message") { options->one_output_per_message = true; } else { *error = absl::Substitute("Unknown parameter: $0", pair.first); return false; } } return true; } class MiniTableGenerator : public google::protobuf::compiler::CodeGenerator { bool Generate(const google::protobuf::FileDescriptor* file, const std::string& parameter, google::protobuf::compiler::GeneratorContext* generator_context, std::string* error) const override { std::vector files{file}; return GenerateAll(files, parameter, generator_context, error); } bool GenerateAll(const std::vector& files, const std::string& parameter, google::protobuf::compiler::GeneratorContext* generator_context, std::string* error) const override { MiniTableOptions options; if (!ParseOptions(&options, parameter, error)) { return false; } upb::Arena arena; DefPoolPair pools; absl::flat_hash_set files_seen; for (const auto* file : files) { PopulateDefPool(file, &arena, &pools, &files_seen); upb::FileDefPtr upb_file = pools.GetFile(file->name()); GenerateFile(pools, upb_file, options, generator_context); } return true; } uint64_t GetSupportedFeatures() const override { return FEATURE_PROTO3_OPTIONAL | FEATURE_SUPPORTS_EDITIONS; } google::protobuf::Edition GetMinimumEdition() const override { return google::protobuf::Edition::EDITION_PROTO2; } google::protobuf::Edition GetMaximumEdition() const override { return google::protobuf::Edition::EDITION_2024; } }; } // namespace generator } // namespace upb int main(int argc, char** argv) { upb::generator::MiniTableGenerator generator; return google::protobuf::compiler::PluginMain(argc, argv, &generator); }