aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/compiler/cpp_generator.cc
diff options
context:
space:
mode:
authorGravatar nnoble <nnoble@google.com>2014-12-10 16:31:01 -0800
committerGravatar Jan Tattermusch <jtattermusch@google.com>2014-12-11 15:11:44 -0800
commitebebb7e21c976b5887a24795cf1435bed58a2906 (patch)
tree6ff3d70312bb60e422b1b8621a6dd47de091f992 /src/compiler/cpp_generator.cc
parent0005b778ee2b18026d3c47329ea61437c853de56 (diff)
First pass at making the compilers open-sourcable.
Change on 2014/12/10 by nnoble <nnoble@google.com> ------------- Created by MOE: http://code.google.com/p/moe-java MOE_MIGRATED_REVID=81829127
Diffstat (limited to 'src/compiler/cpp_generator.cc')
-rw-r--r--src/compiler/cpp_generator.cc451
1 files changed, 451 insertions, 0 deletions
diff --git a/src/compiler/cpp_generator.cc b/src/compiler/cpp_generator.cc
new file mode 100644
index 0000000000..1eccbf7e09
--- /dev/null
+++ b/src/compiler/cpp_generator.cc
@@ -0,0 +1,451 @@
+/*
+ *
+ * Copyright 2014, Google Inc.
+ * 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 Inc. 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 "src/compiler/cpp_generator.h"
+
+#include "src/compiler/cpp_generator_helpers.h"
+#include <google/protobuf/descriptor.h>
+#include <google/protobuf/descriptor.pb.h>
+#include <google/protobuf/io/printer.h>
+#include <google/protobuf/io/zero_copy_stream_impl_lite.h>
+
+namespace grpc_cpp_generator {
+namespace {
+
+bool NoStreaming(const google::protobuf::MethodDescriptor* method) {
+ return !method->options().has_client_streaming() &&
+ !method->options().has_server_streaming();
+}
+
+bool ClientOnlyStreaming(const google::protobuf::MethodDescriptor* method) {
+ return method->options().has_client_streaming() &&
+ !method->options().has_server_streaming();
+}
+
+bool ServerOnlyStreaming(const google::protobuf::MethodDescriptor* method) {
+ return !method->options().has_client_streaming() &&
+ method->options().has_server_streaming();
+}
+
+bool BidiStreaming(const google::protobuf::MethodDescriptor* method) {
+ return method->options().has_client_streaming() &&
+ method->options().has_server_streaming();
+}
+
+bool HasClientOnlyStreaming(const google::protobuf::FileDescriptor* file) {
+ for (int i = 0; i < file->service_count(); i++) {
+ for (int j = 0; j < file->service(i)->method_count(); j++) {
+ if (ClientOnlyStreaming(file->service(i)->method(j))) {
+ return true;
+ }
+ }
+ }
+ return false;
+}
+
+bool HasServerOnlyStreaming(const google::protobuf::FileDescriptor* file) {
+ for (int i = 0; i < file->service_count(); i++) {
+ for (int j = 0; j < file->service(i)->method_count(); j++) {
+ if (ServerOnlyStreaming(file->service(i)->method(j))) {
+ return true;
+ }
+ }
+ }
+ return false;
+}
+
+bool HasBidiStreaming(const google::protobuf::FileDescriptor* file) {
+ for (int i = 0; i < file->service_count(); i++) {
+ for (int j = 0; j < file->service(i)->method_count(); j++) {
+ if (BidiStreaming(file->service(i)->method(j))) {
+ return true;
+ }
+ }
+ }
+ return false;
+}
+} // namespace
+
+string GetHeaderIncludes(const google::protobuf::FileDescriptor* file) {
+ string temp =
+ "#include \"net/grpc/cpp/internal/client/internal_stub.h\"\n"
+ "#include \"net/grpc/cpp/public/status.h\"\n"
+ "\n"
+ "namespace grpc {\n"
+ "class ChannelInterface;\n"
+ "class RpcService;\n";
+ if (HasClientOnlyStreaming(file)) {
+ temp.append("template <class OutMessage> class ClientWriter;\n");
+ temp.append("template <class InMessage> class ServerReader;\n");
+ }
+ if (HasServerOnlyStreaming(file)) {
+ temp.append("template <class InMessage> class ClientReader;\n");
+ temp.append("template <class OutMessage> class ServerWriter;\n");
+ }
+ if (HasBidiStreaming(file)) {
+ temp.append(
+ "template <class OutMessage, class InMessage>\n"
+ "class ClientReaderWriter;\n");
+ temp.append(
+ "template <class OutMessage, class InMessage>\n"
+ "class ServerReaderWriter;\n");
+ }
+ temp.append("} // namespace grpc\n");
+ return temp;
+}
+
+string GetSourceIncludes() {
+ return "#include \"net/grpc/cpp/internal/rpc_method.h\"\n"
+ "#include \"net/grpc/cpp/internal/server/rpc_service_method.h\"\n"
+ "#include \"net/grpc/cpp/public/channel_interface.h\"\n"
+ "#include \"net/grpc/cpp/public/stream.h\"\n";
+}
+
+void PrintHeaderClientMethod(google::protobuf::io::Printer* printer,
+ const google::protobuf::MethodDescriptor* method,
+ map<string, string>* vars) {
+ (*vars)["Method"] = method->name();
+ (*vars)["Request"] =
+ grpc_cpp_generator::ClassName(method->input_type(), true);
+ (*vars)["Response"] =
+ grpc_cpp_generator::ClassName(method->output_type(), true);
+ if (NoStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::Status $Method$(::grpc::ClientContext* context, "
+ "const $Request$& request, $Response$* response);\n\n");
+ } else if (ClientOnlyStreaming(method)) {
+ printer->Print(
+ *vars,
+ "::grpc::ClientWriter<$Request$>* $Method$("
+ "::grpc::ClientContext* context, $Response$* response);\n\n");
+ } else if (ServerOnlyStreaming(method)) {
+ printer->Print(
+ *vars,
+ "::grpc::ClientReader<$Response$>* $Method$("
+ "::grpc::ClientContext* context, const $Request$* request);\n\n");
+ } else if (BidiStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::ClientReaderWriter<$Request$, $Response$>* "
+ "$Method$(::grpc::ClientContext* context);\n\n");
+ }
+}
+
+void PrintHeaderServerMethod(google::protobuf::io::Printer* printer,
+ const google::protobuf::MethodDescriptor* method,
+ map<string, string>* vars) {
+ (*vars)["Method"] = method->name();
+ (*vars)["Request"] =
+ grpc_cpp_generator::ClassName(method->input_type(), true);
+ (*vars)["Response"] =
+ grpc_cpp_generator::ClassName(method->output_type(), true);
+ if (NoStreaming(method)) {
+ printer->Print(*vars,
+ "virtual ::grpc::Status $Method$(const $Request$* request, "
+ "$Response$* response);\n");
+ } else if (ClientOnlyStreaming(method)) {
+ printer->Print(*vars,
+ "virtual ::grpc::Status $Method$("
+ "::grpc::ServerReader<$Request$>* reader, "
+ "$Response$* response);\n");
+ } else if (ServerOnlyStreaming(method)) {
+ printer->Print(*vars,
+ "virtual ::grpc::Status $Method$(const $Request$* request, "
+ "::grpc::ServerWriter<$Response$>* writer);\n");
+ } else if (BidiStreaming(method)) {
+ printer->Print(*vars,
+ "virtual ::grpc::Status $Method$("
+ "::grpc::ServerReaderWriter<$Response$, $Request$>* stream);"
+ "\n");
+ }
+}
+
+void PrintHeaderService(google::protobuf::io::Printer* printer,
+ const google::protobuf::ServiceDescriptor* service,
+ map<string, string>* vars) {
+ (*vars)["Service"] = service->name();
+
+ printer->Print(*vars,
+ "class $Service$ {\n"
+ " public:\n");
+ printer->Indent();
+
+ // Client side
+ printer->Print("class Stub : public ::grpc::InternalStub {\n"
+ " public:\n");
+ printer->Indent();
+ for (int i = 0; i < service->method_count(); ++i) {
+ PrintHeaderClientMethod(printer, service->method(i), vars);
+ }
+ printer->Outdent();
+ printer->Print("};\n");
+ printer->Print(
+ "static Stub* NewStub(const std::shared_ptr<::grpc::ChannelInterface>& "
+ "channel);\n");
+
+ printer->Print("\n");
+
+ // Server side
+ printer->Print("class Service {\n"
+ " public:\n");
+ printer->Indent();
+ printer->Print("Service() : service_(nullptr) {}\n");
+ printer->Print("virtual ~Service();\n");
+ for (int i = 0; i < service->method_count(); ++i) {
+ PrintHeaderServerMethod(printer, service->method(i), vars);
+ }
+ printer->Print("::grpc::RpcService* service();\n");
+ printer->Outdent();
+ printer->Print(" private:\n"
+ " ::grpc::RpcService* service_;\n");
+ printer->Print("};\n");
+
+ printer->Outdent();
+ printer->Print("};\n");
+}
+
+string GetHeaderServices(const google::protobuf::FileDescriptor* file) {
+ string output;
+ google::protobuf::io::StringOutputStream output_stream(&output);
+ google::protobuf::io::Printer printer(&output_stream, '$');
+ map<string, string> vars;
+
+ for (int i = 0; i < file->service_count(); ++i) {
+ PrintHeaderService(&printer, file->service(0), &vars);
+ printer.Print("\n");
+ }
+ return output;
+}
+
+void PrintSourceClientMethod(google::protobuf::io::Printer* printer,
+ const google::protobuf::MethodDescriptor* method,
+ map<string, string>* vars) {
+ (*vars)["Method"] = method->name();
+ (*vars)["Request"] =
+ grpc_cpp_generator::ClassName(method->input_type(), true);
+ (*vars)["Response"] =
+ grpc_cpp_generator::ClassName(method->output_type(), true);
+ if (NoStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::Status $Service$::Stub::$Method$("
+ "::grpc::ClientContext* context, "
+ "const $Request$& request, $Response$* response) {\n");
+ printer->Print(*vars,
+ " return channel()->StartBlockingRpc("
+ "::grpc::RpcMethod(\"/$Service$/$Method$\"), "
+ "context, request, response);\n"
+ "}\n\n");
+ } else if (ClientOnlyStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::ClientWriter<$Request$>* $Service$::Stub::$Method$("
+ "::grpc::ClientContext* context, $Response$* response) {\n");
+ printer->Print(
+ *vars,
+ " return new ::grpc::ClientWriter<$Request$>("
+ "channel()->CreateStream(::grpc::RpcMethod(\"/$Service$/$Method$\", "
+ "::grpc::RpcMethod::RpcType::CLIENT_STREAMING), "
+ "context, nullptr, response));\n"
+ "}\n\n");
+ } else if (ServerOnlyStreaming(method)) {
+ printer->Print(
+ *vars,
+ "::grpc::ClientReader<$Response$>* $Service$::Stub::$Method$("
+ "::grpc::ClientContext* context, const $Request$* request) {\n");
+ printer->Print(
+ *vars,
+ " return new ::grpc::ClientReader<$Response$>("
+ "channel()->CreateStream(::grpc::RpcMethod(\"/$Service$/$Method$\", "
+ "::grpc::RpcMethod::RpcType::SERVER_STREAMING), "
+ "context, request, nullptr));\n"
+ "}\n\n");
+ } else if (BidiStreaming(method)) {
+ printer->Print(
+ *vars,
+ "::grpc::ClientReaderWriter<$Request$, $Response$>* "
+ "$Service$::Stub::$Method$(::grpc::ClientContext* context) {\n");
+ printer->Print(
+ *vars,
+ " return new ::grpc::ClientReaderWriter<$Request$, $Response$>("
+ "channel()->CreateStream(::grpc::RpcMethod(\"/$Service$/$Method$\", "
+ "::grpc::RpcMethod::RpcType::BIDI_STREAMING), "
+ "context, nullptr, nullptr));\n"
+ "}\n\n");
+ }
+}
+
+void PrintSourceServerMethod(google::protobuf::io::Printer* printer,
+ const google::protobuf::MethodDescriptor* method,
+ map<string, string>* vars) {
+ (*vars)["Method"] = method->name();
+ (*vars)["Request"] =
+ grpc_cpp_generator::ClassName(method->input_type(), true);
+ (*vars)["Response"] =
+ grpc_cpp_generator::ClassName(method->output_type(), true);
+ if (NoStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::Status $Service$::Service::$Method$("
+ "const $Request$* request, $Response$* response) {\n");
+ printer->Print(
+ " return ::grpc::Status("
+ "::grpc::StatusCode::UNIMPLEMENTED);\n");
+ printer->Print("}\n\n");
+ } else if (ClientOnlyStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::Status $Service$::Service::$Method$("
+ "::grpc::ServerReader<$Request$>* reader, "
+ "$Response$* response) {\n");
+ printer->Print(
+ " return ::grpc::Status("
+ "::grpc::StatusCode::UNIMPLEMENTED);\n");
+ printer->Print("}\n\n");
+ } else if (ServerOnlyStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::Status $Service$::Service::$Method$("
+ "const $Request$* request, "
+ "::grpc::ServerWriter<$Response$>* writer) {\n");
+ printer->Print(
+ " return ::grpc::Status("
+ "::grpc::StatusCode::UNIMPLEMENTED);\n");
+ printer->Print("}\n\n");
+ } else if (BidiStreaming(method)) {
+ printer->Print(*vars,
+ "::grpc::Status $Service$::Service::$Method$("
+ "::grpc::ServerReaderWriter<$Response$, $Request$>* "
+ "stream) {\n");
+ printer->Print(
+ " return ::grpc::Status("
+ "::grpc::StatusCode::UNIMPLEMENTED);\n");
+ printer->Print("}\n\n");
+ }
+}
+
+void PrintSourceService(google::protobuf::io::Printer* printer,
+ const google::protobuf::ServiceDescriptor* service,
+ map<string, string>* vars) {
+ (*vars)["Service"] = service->name();
+ printer->Print(*vars,
+ "$Service$::Stub* $Service$::NewStub("
+ "const std::shared_ptr<::grpc::ChannelInterface>& channel) {\n"
+ " $Service$::Stub* stub = new $Service$::Stub();\n"
+ " stub->set_channel(channel);\n"
+ " return stub;\n"
+ "};\n\n");
+ for (int i = 0; i < service->method_count(); ++i) {
+ PrintSourceClientMethod(printer, service->method(i), vars);
+ }
+
+ printer->Print(*vars,
+ "$Service$::Service::~Service() {\n"
+ " delete service_;\n"
+ "}\n\n");
+ for (int i = 0; i < service->method_count(); ++i) {
+ PrintSourceServerMethod(printer, service->method(i), vars);
+ }
+ printer->Print(*vars,
+ "::grpc::RpcService* $Service$::Service::service() {\n");
+ printer->Indent();
+ printer->Print("if (service_ != nullptr) {\n"
+ " return service_;\n"
+ "}\n");
+ printer->Print("service_ = new ::grpc::RpcService();\n");
+ for (int i = 0; i < service->method_count(); ++i) {
+ const google::protobuf::MethodDescriptor* method = service->method(i);
+ (*vars)["Method"] = method->name();
+ (*vars)["Request"] =
+ grpc_cpp_generator::ClassName(method->input_type(), true);
+ (*vars)["Response"] =
+ grpc_cpp_generator::ClassName(method->output_type(), true);
+ if (NoStreaming(method)) {
+ printer->Print(
+ *vars,
+ "service_->AddMethod(new ::grpc::RpcServiceMethod(\n"
+ " \"/$Service$/$Method$\", ::grpc::RpcMethod::NORMAL_RPC,\n"
+ " new ::grpc::RpcMethodHandler<$Service$::Service, $Request$, "
+ "$Response$>(\n"
+ " std::function<::grpc::Status($Service$::Service*, "
+ "const $Request$*, $Response$*)>("
+ "&$Service$::Service::$Method$), this),\n"
+ " new $Request$, new $Response$));\n");
+ } else if (ClientOnlyStreaming(method)) {
+ printer->Print(
+ *vars,
+ "service_->AddMethod(new ::grpc::RpcServiceMethod(\n"
+ " \"/$Service$/$Method$\", ::grpc::RpcMethod::CLIENT_STREAMING,\n"
+ " new ::grpc::ClientStreamingHandler<"
+ "$Service$::Service, $Request$, $Response$>(\n"
+ " std::function<::grpc::Status($Service$::Service*, "
+ "::grpc::ServerReader<$Request$>*, $Response$*)>("
+ "&$Service$::Service::$Method$), this),\n"
+ " new $Request$, new $Response$));\n");
+ } else if (ServerOnlyStreaming(method)) {
+ printer->Print(
+ *vars,
+ "service_->AddMethod(new ::grpc::RpcServiceMethod(\n"
+ " \"/$Service$/$Method$\", ::grpc::RpcMethod::SERVER_STREAMING,\n"
+ " new ::grpc::ServerStreamingHandler<"
+ "$Service$::Service, $Request$, $Response$>(\n"
+ " std::function<::grpc::Status($Service$::Service*, "
+ "const $Request$*, ::grpc::ServerWriter<$Response$>*)>("
+ "&$Service$::Service::$Method$), this),\n"
+ " new $Request$, new $Response$));\n");
+ } else if (BidiStreaming(method)) {
+ printer->Print(
+ *vars,
+ "service_->AddMethod(new ::grpc::RpcServiceMethod(\n"
+ " \"/$Service$/$Method$\", ::grpc::RpcMethod::BIDI_STREAMING,\n"
+ " new ::grpc::BidiStreamingHandler<"
+ "$Service$::Service, $Request$, $Response$>(\n"
+ " std::function<::grpc::Status($Service$::Service*, "
+ "::grpc::ServerReaderWriter<$Response$, $Request$>*)>("
+ "&$Service$::Service::$Method$), this),\n"
+ " new $Request$, new $Response$));\n");
+ }
+ }
+ printer->Print("return service_;\n");
+ printer->Outdent();
+ printer->Print("}\n\n");
+}
+
+string GetSourceServices(const google::protobuf::FileDescriptor* file) {
+ string output;
+ google::protobuf::io::StringOutputStream output_stream(&output);
+ google::protobuf::io::Printer printer(&output_stream, '$');
+ map<string, string> vars;
+
+ for (int i = 0; i < file->service_count(); ++i) {
+ PrintSourceService(&printer, file->service(0), &vars);
+ printer.Print("\n");
+ }
+ return output;
+}
+
+} // namespace grpc_cpp_generator