| // 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 <cstdint> | 
 | #include <string> | 
 | #include <utility> | 
 | #include <vector> | 
 |  | 
 | #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<const google::protobuf::FileDescriptor*> files{file}; | 
 |     return GenerateAll(files, parameter, generator_context, error); | 
 |   } | 
 |  | 
 |   bool GenerateAll(const std::vector<const google::protobuf::FileDescriptor*>& 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<std::string> 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_2023; | 
 |   } | 
 | }; | 
 |  | 
 | }  // namespace generator | 
 | }  // namespace upb | 
 |  | 
 | int main(int argc, char** argv) { | 
 |   upb::generator::MiniTableGenerator generator; | 
 |   return google::protobuf::compiler::PluginMain(argc, argv, &generator); | 
 | } |