mirror of https://github.com/grpc/grpc.git
The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#)
https://grpc.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
184 lines
5.4 KiB
184 lines
5.4 KiB
4 years ago
|
|
||
|
#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());
|
||
|
}
|