| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 | 
							
- #include <memory>
 
- #include "google/protobuf/compiler/code_generator.h"
 
- #include "google/protobuf/compiler/plugin.h"
 
- #include "google/protobuf/descriptor.h"
 
- #include "google/protobuf/descriptor.pb.h"
 
- #include "upbc/common.h"
 
- namespace upbc {
 
- namespace {
 
- namespace protoc = ::google::protobuf::compiler;
 
- namespace protobuf = ::google::protobuf;
 
- std::string DefInitSymbol(const protobuf::FileDescriptor *file) {
 
-   return ToCIdent(file->name()) + "_upbdefinit";
 
- }
 
- static std::string DefHeaderFilename(std::string proto_filename) {
 
-   return StripExtension(proto_filename) + ".upbdefs.h";
 
- }
 
- static std::string DefSourceFilename(std::string proto_filename) {
 
-   return StripExtension(proto_filename) + ".upbdefs.c";
 
- }
 
- void GenerateMessageDefAccessor(const protobuf::Descriptor* d, Output& output) {
 
-   output("UPB_INLINE const upb_msgdef *$0_getmsgdef(upb_symtab *s) {\n",
 
-          ToCIdent(d->full_name()));
 
-   output("  _upb_symtab_loaddefinit(s, &$0);\n", DefInitSymbol(d->file()));
 
-   output("  return upb_symtab_lookupmsg(s, \"$0\");\n", d->full_name());
 
-   output("}\n");
 
-   output("\n");
 
-   for (int i = 0; i < d->nested_type_count(); i++) {
 
-     GenerateMessageDefAccessor(d->nested_type(i), output);
 
-   }
 
- }
 
- void WriteDefHeader(const protobuf::FileDescriptor* file, Output& output) {
 
-   EmitFileWarning(file, output);
 
-   output(
 
-       "#ifndef $0_UPBDEFS_H_\n"
 
-       "#define $0_UPBDEFS_H_\n\n"
 
-       "#include \"upb/def.h\"\n"
 
-       "#include \"upb/port_def.inc\"\n"
 
-       "#ifdef __cplusplus\n"
 
-       "extern \"C\" {\n"
 
-       "#endif\n\n",
 
-       ToPreproc(file->name()));
 
-   output("#include \"upb/def.h\"\n");
 
-   output("\n");
 
-   output("#include \"upb/port_def.inc\"\n");
 
-   output("\n");
 
-   output("extern upb_def_init $0;\n", DefInitSymbol(file));
 
-   output("\n");
 
-   for (int i = 0; i < file->message_type_count(); i++) {
 
-     GenerateMessageDefAccessor(file->message_type(i), output);
 
-   }
 
-   output(
 
-       "#ifdef __cplusplus\n"
 
-       "}  /* extern \"C\" */\n"
 
-       "#endif\n"
 
-       "\n"
 
-       "#include \"upb/port_undef.inc\"\n"
 
-       "\n"
 
-       "#endif  /* $0_UPBDEFS_H_ */\n",
 
-       ToPreproc(file->name()));
 
- }
 
- void WriteDefSource(const protobuf::FileDescriptor* file, Output& output) {
 
-   EmitFileWarning(file, output);
 
-   output("#include \"upb/def.h\"\n");
 
-   output("#include \"$0\"\n", DefHeaderFilename(file->name()));
 
-   output("\n");
 
-   for (int i = 0; i < file->dependency_count(); i++) {
 
-     output("extern upb_def_init $0;\n", DefInitSymbol(file->dependency(i)));
 
-   }
 
-   std::vector<const protobuf::Descriptor*> file_messages =
 
-       SortedMessages(file);
 
-   for (auto message : file_messages) {
 
-     output("extern const upb_msglayout $0;\n", MessageInit(message));
 
-   }
 
-   output("\n");
 
-   if (!file_messages.empty()) {
 
-     output("static const upb_msglayout *layouts[$0] = {\n", file_messages.size());
 
-     for (auto message : file_messages) {
 
-       output("  &$0,\n", MessageInit(message));
 
-     }
 
-     output("};\n");
 
-     output("\n");
 
-   }
 
-   protobuf::FileDescriptorProto file_proto;
 
-   file->CopyTo(&file_proto);
 
-   std::string file_data;
 
-   file_proto.SerializeToString(&file_data);
 
-   output("static const char descriptor[$0] = {", file_data.size());
 
-   // C90 only guarantees that strings can be up to 509 characters, and some
 
-   // implementations have limits here (for example, MSVC only allows 64k:
 
-   // https://docs.microsoft.com/en-us/cpp/error-messages/compiler-errors-1/fatal-error-c1091.
 
-   // So we always emit an array instead of a string.
 
-   for (size_t i = 0; i < file_data.size();) {
 
-     for (size_t j = 0; j < 25 && i < file_data.size(); ++i, ++j) {
 
-       output("'$0', ", absl::CEscape(file_data.substr(i, 1)));
 
-     }
 
-     output("\n");
 
-   }
 
-   output("};\n\n");
 
-   output("static upb_def_init *deps[$0] = {\n", file->dependency_count() + 1);
 
-   for (int i = 0; i < file->dependency_count(); i++) {
 
-     output("  &$0,\n", DefInitSymbol(file->dependency(i)));
 
-   }
 
-   output("  NULL\n");
 
-   output("};\n");
 
-   output("\n");
 
-   output("upb_def_init $0 = {\n", DefInitSymbol(file));
 
-   output("  deps,\n");
 
-   if (file_messages.empty()) {
 
-     output("  NULL,\n");
 
-   } else {
 
-     output("  layouts,\n");
 
-   }
 
-   output("  \"$0\",\n", file->name());
 
-   output("  UPB_STRVIEW_INIT(descriptor, $0)\n", file_data.size());
 
-   output("};\n");
 
- }
 
- class Generator : public protoc::CodeGenerator {
 
-   ~Generator() override {}
 
-   bool Generate(const protobuf::FileDescriptor* file,
 
-                 const std::string& parameter, protoc::GeneratorContext* context,
 
-                 std::string* error) const override;
 
-   uint64_t GetSupportedFeatures() const override {
 
-     return FEATURE_PROTO3_OPTIONAL;
 
-   }
 
- };
 
- bool Generator::Generate(const protobuf::FileDescriptor* file,
 
-                          const std::string& parameter,
 
-                          protoc::GeneratorContext* context,
 
-                          std::string* error) const {
 
-   std::vector<std::pair<std::string, std::string>> params;
 
-   google::protobuf::compiler::ParseGeneratorParameter(parameter, ¶ms);
 
-   for (const auto& pair : params) {
 
-     *error = "Unknown parameter: " + pair.first;
 
-     return false;
 
-   }
 
-   Output h_def_output(context->Open(DefHeaderFilename(file->name())));
 
-   WriteDefHeader(file, h_def_output);
 
-   Output c_def_output(context->Open(DefSourceFilename(file->name())));
 
-   WriteDefSource(file, c_def_output);
 
-   return true;
 
- }
 
- }  // namespace
 
- }  // namespace upbc
 
- int main(int argc, char** argv) {
 
-   std::unique_ptr<google::protobuf::compiler::CodeGenerator> generator(
 
-       new upbc::Generator());
 
-   return google::protobuf::compiler::PluginMain(argc, argv, generator.get());
 
- }
 
 
  |