Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
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.
151 lines
4.5 KiB
151 lines
4.5 KiB
// 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 |
|
|
|
#ifndef UPB_PROTOS_GENERATOR_OUTPUT_H |
|
#define UPB_PROTOS_GENERATOR_OUTPUT_H |
|
|
|
#include <vector> |
|
|
|
#include "absl/log/absl_log.h" |
|
#include "absl/strings/str_replace.h" |
|
#include "absl/strings/substitute.h" |
|
#include "google/protobuf/descriptor.h" |
|
#include "google/protobuf/io/zero_copy_stream.h" |
|
|
|
namespace protos_generator { |
|
|
|
class Output { |
|
public: |
|
Output(google::protobuf::io::ZeroCopyOutputStream* stream) : stream_(stream) {} |
|
~Output() { stream_->BackUp((int)buffer_size_); } |
|
|
|
template <class... Arg> |
|
void operator()(absl::string_view format, const Arg&... arg) { |
|
Write(absl::Substitute(format, arg...)); |
|
} |
|
|
|
// Indentation size in characters. |
|
static constexpr size_t kIndentationSize = 2; |
|
|
|
void Indent() { Indent(kIndentationSize); } |
|
void Indent(size_t size) { indent_ += size; } |
|
|
|
void Outdent() { Outdent(kIndentationSize); } |
|
void Outdent(size_t size) { |
|
if (indent_ < size) { |
|
ABSL_LOG(FATAL) << "mismatched Output indent/unindent calls"; |
|
} |
|
indent_ -= size; |
|
} |
|
|
|
private: |
|
void Write(absl::string_view data) { |
|
std::string stripped; |
|
if (absl::StartsWith(data, "\n ")) { |
|
size_t indent = data.substr(1).find_first_not_of(' '); |
|
if (indent > indent_) { |
|
indent -= indent_; |
|
} |
|
if (indent != absl::string_view::npos) { |
|
// Remove indentation from all lines. |
|
auto line_prefix = data.substr(0, indent + 1); |
|
// The final line has an extra newline and is indented two less, eg. |
|
// R"cc( |
|
// UPB_INLINE $0 $1_$2(const $1 *msg) { |
|
// return $1_has_$2(msg) ? *UPB_PTR_AT(msg, $3, $0) : $4; |
|
// } |
|
// )cc", |
|
std::string last_line_prefix = std::string(line_prefix); |
|
last_line_prefix.resize(last_line_prefix.size() - 2); |
|
data.remove_prefix(line_prefix.size()); |
|
stripped = absl::StrReplaceAll( |
|
data, {{line_prefix, "\n"}, {last_line_prefix, "\n"}}); |
|
data = stripped; |
|
} |
|
} else { |
|
WriteIndent(); |
|
} |
|
WriteRaw(data); |
|
} |
|
|
|
void WriteRaw(absl::string_view data) { |
|
while (!data.empty()) { |
|
RefreshOutput(); |
|
size_t to_write = std::min(data.size(), buffer_size_); |
|
memcpy(output_buffer_, data.data(), to_write); |
|
data.remove_prefix(to_write); |
|
output_buffer_ += to_write; |
|
buffer_size_ -= to_write; |
|
} |
|
} |
|
|
|
void WriteIndent() { |
|
if (indent_ == 0) { |
|
return; |
|
} |
|
size_t size = indent_; |
|
while (size > buffer_size_) { |
|
if (buffer_size_ > 0) { |
|
memset(output_buffer_, ' ', buffer_size_); |
|
} |
|
size -= buffer_size_; |
|
buffer_size_ = 0; |
|
RefreshOutput(); |
|
} |
|
memset(output_buffer_, ' ', size); |
|
output_buffer_ += size; |
|
buffer_size_ -= size; |
|
} |
|
|
|
void RefreshOutput() { |
|
while (buffer_size_ == 0) { |
|
void* void_buffer; |
|
int size; |
|
if (!stream_->Next(&void_buffer, &size)) { |
|
fprintf(stderr, "upb_generator: Failed to write to to output\n"); |
|
abort(); |
|
} |
|
output_buffer_ = static_cast<char*>(void_buffer); |
|
buffer_size_ = size; |
|
} |
|
} |
|
|
|
google::protobuf::io::ZeroCopyOutputStream* stream_; |
|
char* output_buffer_ = nullptr; |
|
size_t buffer_size_ = 0; |
|
// Current indentation size in characters. |
|
size_t indent_ = 0; |
|
friend class OutputIndenter; |
|
}; |
|
|
|
class OutputIndenter { |
|
public: |
|
OutputIndenter(Output& output) |
|
: OutputIndenter(output, Output::kIndentationSize) {} |
|
OutputIndenter(Output& output, size_t indent_size) |
|
: indent_size_(indent_size), output_(output) { |
|
output.Indent(indent_size); |
|
} |
|
~OutputIndenter() { output_.Outdent(indent_size_); } |
|
|
|
private: |
|
size_t indent_size_; |
|
Output& output_; |
|
}; |
|
|
|
std::string StripExtension(absl::string_view fname); |
|
std::string ToCIdent(absl::string_view str); |
|
std::string ToPreproc(absl::string_view str); |
|
void EmitFileWarning(const google::protobuf::FileDescriptor* file, Output& output); |
|
std::string MessageName(const google::protobuf::Descriptor* descriptor); |
|
std::string FileLayoutName(const google::protobuf::FileDescriptor* file); |
|
std::string CHeaderFilename(const google::protobuf::FileDescriptor* file); |
|
std::string CSourceFilename(const google::protobuf::FileDescriptor* file); |
|
|
|
} // namespace protos_generator |
|
|
|
#endif // UPB_PROTOS_GENERATOR_OUTPUT_H
|
|
|