| // 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 "google/protobuf/descriptor.pb.h" |
| #include "absl/strings/str_replace.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 "google/protobuf/descriptor.h" |
| #include "google/protobuf/io/printer.h" |
| |
| namespace protoc = ::google::protobuf::compiler; |
| namespace protobuf = ::google::protobuf; |
| |
| class LuaGenerator : public protoc::CodeGenerator { |
| bool Generate(const protobuf::FileDescriptor* file, |
| const std::string& parameter, protoc::GeneratorContext* context, |
| std::string* error) const override; |
| }; |
| |
| static std::string StripExtension(absl::string_view fname) { |
| size_t lastdot = fname.find_last_of('.'); |
| if (lastdot == std::string::npos) { |
| return std::string(fname); |
| } |
| return std::string(fname.substr(0, lastdot)); |
| } |
| |
| static std::string Filename(const protobuf::FileDescriptor* file) { |
| return StripExtension(file->name()) + "_pb.lua"; |
| } |
| |
| static std::string ModuleName(const protobuf::FileDescriptor* file) { |
| std::string ret = StripExtension(file->name()) + "_pb"; |
| return absl::StrReplaceAll(ret, {{"/", "."}}); |
| } |
| |
| static void PrintHexDigit(char digit, protobuf::io::Printer* printer) { |
| char text; |
| if (digit < 10) { |
| text = '0' + digit; |
| } else { |
| text = 'A' + (digit - 10); |
| } |
| printer->WriteRaw(&text, 1); |
| } |
| |
| static bool IsPrint(int ch) { |
| // isprint(ch) with negative values is UB. |
| return ch < 0 ? false : isprint(ch); |
| } |
| |
| static void PrintString(int max_cols, absl::string_view* str, |
| protobuf::io::Printer* printer) { |
| printer->Print("\'"); |
| while (max_cols > 0 && !str->empty()) { |
| char ch = (*str)[0]; |
| if (ch == '\\') { |
| printer->PrintRaw("\\\\"); |
| max_cols--; |
| } else if (ch == '\'') { |
| printer->PrintRaw("\\'"); |
| max_cols--; |
| } else if (IsPrint(ch)) { |
| printer->WriteRaw(&ch, 1); |
| max_cols--; |
| } else { |
| unsigned char byte = ch; |
| printer->PrintRaw("\\x"); |
| PrintHexDigit(byte >> 4, printer); |
| PrintHexDigit(byte & 15, printer); |
| max_cols -= 4; |
| } |
| str->remove_prefix(1); |
| } |
| printer->Print("\'"); |
| } |
| |
| bool LuaGenerator::Generate(const protobuf::FileDescriptor* file, |
| const std::string& /* parameter */, |
| protoc::GeneratorContext* context, |
| std::string* /* error */) const { |
| std::string filename = Filename(file); |
| protobuf::io::ZeroCopyOutputStream* out = context->Open(filename); |
| protobuf::io::Printer printer(out, '$'); |
| |
| for (int i = 0; i < file->dependency_count(); i++) { |
| const protobuf::FileDescriptor* dep = file->dependency(i); |
| printer.Print("require('$name$')\n", "name", ModuleName(dep)); |
| } |
| |
| printer.Print("local upb = require('upb')\n"); |
| |
| protobuf::FileDescriptorProto file_proto; |
| file->CopyTo(&file_proto); |
| std::string file_data; |
| file_proto.SerializeToString(&file_data); |
| |
| printer.Print("local descriptor = table.concat({\n"); |
| absl::string_view data(file_data); |
| while (!data.empty()) { |
| printer.Print(" "); |
| PrintString(72, &data, &printer); |
| printer.Print(",\n"); |
| } |
| printer.Print("})\n"); |
| |
| printer.Print("return upb._generated_module(descriptor)\n"); |
| |
| return true; |
| } |
| |
| int main(int argc, char** argv) { |
| LuaGenerator generator; |
| return google::protobuf::compiler::PluginMain(argc, argv, &generator); |
| } |