// Copyright (c) 2009-2021, Google LLC // All rights reserved. // // 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 Google LLC 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); }