| // Protocol Buffers - Google's data interchange format |
| // Copyright 2023 Google LLC. All rights reserved. |
| // https://developers.google.com/protocol-buffers/ |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are |
| // met: |
| // |
| // * Redistributions of source code must retain the above copyright |
| // notice, this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above |
| // copyright notice, this list of conditions and the following disclaimer |
| // in the documentation and/or other materials provided with the |
| // distribution. |
| // * Neither the name of Google LLC nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #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 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); |
| } |