mirror of https://github.com/grpc/grpc.git
Merge pull request #18856 from vjpai/unary_reactor_server
Revamp server-side callback API and message allocator APIpull/21267/head
commit
2c6a7e1f19
47 changed files with 2390 additions and 1608 deletions
@ -0,0 +1,814 @@ |
|||||||
|
/*
|
||||||
|
* |
||||||
|
* Copyright 2019 gRPC authors. |
||||||
|
* |
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
* you may not use this file except in compliance with the License. |
||||||
|
* You may obtain a copy of the License at |
||||||
|
* |
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* |
||||||
|
* Unless required by applicable law or agreed to in writing, software |
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
* See the License for the specific language governing permissions and |
||||||
|
* limitations under the License. |
||||||
|
*/ |
||||||
|
|
||||||
|
#ifndef GRPCPP_IMPL_CODEGEN_SERVER_CALLBACK_HANDLERS_H |
||||||
|
#define GRPCPP_IMPL_CODEGEN_SERVER_CALLBACK_HANDLERS_H |
||||||
|
|
||||||
|
#include <grpcpp/impl/codegen/message_allocator.h> |
||||||
|
#include <grpcpp/impl/codegen/rpc_service_method.h> |
||||||
|
#include <grpcpp/impl/codegen/server_callback_impl.h> |
||||||
|
#include <grpcpp/impl/codegen/server_context_impl.h> |
||||||
|
#include <grpcpp/impl/codegen/status.h> |
||||||
|
|
||||||
|
namespace grpc_impl { |
||||||
|
namespace internal { |
||||||
|
|
||||||
|
template <class RequestType, class ResponseType> |
||||||
|
class CallbackUnaryHandler : public ::grpc::internal::MethodHandler { |
||||||
|
public: |
||||||
|
explicit CallbackUnaryHandler( |
||||||
|
std::function<experimental::ServerUnaryReactor*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*, const RequestType*, |
||||||
|
ResponseType*)> |
||||||
|
get_reactor) |
||||||
|
: get_reactor_(std::move(get_reactor)) {} |
||||||
|
|
||||||
|
void SetMessageAllocator( |
||||||
|
::grpc::experimental::MessageAllocator<RequestType, ResponseType>* |
||||||
|
allocator) { |
||||||
|
allocator_ = allocator; |
||||||
|
} |
||||||
|
|
||||||
|
void RunHandler(const HandlerParameter& param) final { |
||||||
|
// Arena allocate a controller structure (that includes request/response)
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_ref(param.call->call()); |
||||||
|
auto* allocator_state = static_cast< |
||||||
|
::grpc::experimental::MessageHolder<RequestType, ResponseType>*>( |
||||||
|
param.internal_data); |
||||||
|
|
||||||
|
auto* call = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(ServerCallbackUnaryImpl))) |
||||||
|
ServerCallbackUnaryImpl( |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
param.call, allocator_state, std::move(param.call_requester)); |
||||||
|
param.server_context->BeginCompletionOp( |
||||||
|
param.call, [call](bool) { call->MaybeDone(); }, call); |
||||||
|
|
||||||
|
experimental::ServerUnaryReactor* reactor = nullptr; |
||||||
|
if (param.status.ok()) { |
||||||
|
reactor = ::grpc::internal::CatchingReactorGetter< |
||||||
|
experimental::ServerUnaryReactor>( |
||||||
|
get_reactor_, |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
call->request(), call->response()); |
||||||
|
} |
||||||
|
|
||||||
|
if (reactor == nullptr) { |
||||||
|
// if deserialization or reactor creator failed, we need to fail the call
|
||||||
|
reactor = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(UnimplementedUnaryReactor))) |
||||||
|
UnimplementedUnaryReactor( |
||||||
|
::grpc::Status(::grpc::StatusCode::UNIMPLEMENTED, "")); |
||||||
|
} |
||||||
|
|
||||||
|
/// Invoke SetupReactor as the last part of the handler
|
||||||
|
call->SetupReactor(reactor); |
||||||
|
} |
||||||
|
|
||||||
|
void* Deserialize(grpc_call* call, grpc_byte_buffer* req, |
||||||
|
::grpc::Status* status, void** handler_data) final { |
||||||
|
::grpc::ByteBuffer buf; |
||||||
|
buf.set_buffer(req); |
||||||
|
RequestType* request = nullptr; |
||||||
|
::grpc::experimental::MessageHolder<RequestType, ResponseType>* |
||||||
|
allocator_state = nullptr; |
||||||
|
if (allocator_ != nullptr) { |
||||||
|
allocator_state = allocator_->AllocateMessages(); |
||||||
|
} else { |
||||||
|
allocator_state = |
||||||
|
new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
call, sizeof(DefaultMessageHolder<RequestType, ResponseType>))) |
||||||
|
DefaultMessageHolder<RequestType, ResponseType>(); |
||||||
|
} |
||||||
|
*handler_data = allocator_state; |
||||||
|
request = allocator_state->request(); |
||||||
|
*status = |
||||||
|
::grpc::SerializationTraits<RequestType>::Deserialize(&buf, request); |
||||||
|
buf.Release(); |
||||||
|
if (status->ok()) { |
||||||
|
return request; |
||||||
|
} |
||||||
|
// Clean up on deserialization failure.
|
||||||
|
allocator_state->Release(); |
||||||
|
return nullptr; |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
std::function<experimental::ServerUnaryReactor*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*, const RequestType*, |
||||||
|
ResponseType*)> |
||||||
|
get_reactor_; |
||||||
|
::grpc::experimental::MessageAllocator<RequestType, ResponseType>* |
||||||
|
allocator_ = nullptr; |
||||||
|
|
||||||
|
class ServerCallbackUnaryImpl : public experimental::ServerCallbackUnary { |
||||||
|
public: |
||||||
|
void Finish(::grpc::Status s) override { |
||||||
|
finish_tag_.Set( |
||||||
|
call_.call(), [this](bool) { MaybeDone(); }, &finish_ops_, |
||||||
|
reactor_.load(std::memory_order_relaxed)->InternalInlineable()); |
||||||
|
finish_ops_.set_core_cq_tag(&finish_tag_); |
||||||
|
|
||||||
|
if (!ctx_->sent_initial_metadata_) { |
||||||
|
finish_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
finish_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
} |
||||||
|
// The response is dropped if the status is not OK.
|
||||||
|
if (s.ok()) { |
||||||
|
finish_ops_.ServerSendStatus(&ctx_->trailing_metadata_, |
||||||
|
finish_ops_.SendMessagePtr(response())); |
||||||
|
} else { |
||||||
|
finish_ops_.ServerSendStatus(&ctx_->trailing_metadata_, s); |
||||||
|
} |
||||||
|
finish_ops_.set_core_cq_tag(&finish_tag_); |
||||||
|
call_.PerformOps(&finish_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void SendInitialMetadata() override { |
||||||
|
GPR_CODEGEN_ASSERT(!ctx_->sent_initial_metadata_); |
||||||
|
this->Ref(); |
||||||
|
meta_tag_.Set(call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed) |
||||||
|
->OnSendInitialMetadataDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&meta_ops_, false); |
||||||
|
meta_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
meta_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
meta_ops_.set_core_cq_tag(&meta_tag_); |
||||||
|
call_.PerformOps(&meta_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
friend class CallbackUnaryHandler<RequestType, ResponseType>; |
||||||
|
|
||||||
|
ServerCallbackUnaryImpl( |
||||||
|
::grpc_impl::experimental::CallbackServerContext* ctx, |
||||||
|
::grpc::internal::Call* call, |
||||||
|
::grpc::experimental::MessageHolder<RequestType, ResponseType>* |
||||||
|
allocator_state, |
||||||
|
std::function<void()> call_requester) |
||||||
|
: ctx_(ctx), |
||||||
|
call_(*call), |
||||||
|
allocator_state_(allocator_state), |
||||||
|
call_requester_(std::move(call_requester)) { |
||||||
|
ctx_->set_message_allocator_state(allocator_state); |
||||||
|
} |
||||||
|
|
||||||
|
/// SetupReactor binds the reactor (which also releases any queued
|
||||||
|
/// operations), maybe calls OnCancel if possible/needed, and maybe marks
|
||||||
|
/// the completion of the RPC. This should be the last component of the
|
||||||
|
/// handler.
|
||||||
|
void SetupReactor(experimental::ServerUnaryReactor* reactor) { |
||||||
|
reactor_.store(reactor, std::memory_order_relaxed); |
||||||
|
this->BindReactor(reactor); |
||||||
|
this->MaybeCallOnCancel(reactor); |
||||||
|
this->MaybeDone(); |
||||||
|
} |
||||||
|
|
||||||
|
const RequestType* request() { return allocator_state_->request(); } |
||||||
|
ResponseType* response() { return allocator_state_->response(); } |
||||||
|
|
||||||
|
void MaybeDone() override { |
||||||
|
if (GPR_UNLIKELY(this->Unref() == 1)) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnDone(); |
||||||
|
grpc_call* call = call_.call(); |
||||||
|
auto call_requester = std::move(call_requester_); |
||||||
|
allocator_state_->Release(); |
||||||
|
this->~ServerCallbackUnaryImpl(); // explicitly call destructor
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_unref(call); |
||||||
|
call_requester(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ServerReactor* reactor() override { |
||||||
|
return reactor_.load(std::memory_order_relaxed); |
||||||
|
} |
||||||
|
|
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata> |
||||||
|
meta_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag meta_tag_; |
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata, |
||||||
|
::grpc::internal::CallOpSendMessage, |
||||||
|
::grpc::internal::CallOpServerSendStatus> |
||||||
|
finish_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag finish_tag_; |
||||||
|
|
||||||
|
::grpc_impl::experimental::CallbackServerContext* const ctx_; |
||||||
|
::grpc::internal::Call call_; |
||||||
|
::grpc::experimental::MessageHolder<RequestType, ResponseType>* const |
||||||
|
allocator_state_; |
||||||
|
std::function<void()> call_requester_; |
||||||
|
// reactor_ can always be loaded/stored with relaxed memory ordering because
|
||||||
|
// its value is only set once, independently of other data in the object,
|
||||||
|
// and the loads that use it will always actually come provably later even
|
||||||
|
// though they are from different threads since they are triggered by
|
||||||
|
// actions initiated only by the setting up of the reactor_ variable. In
|
||||||
|
// a sense, it's a delayed "const": it gets its value from the SetupReactor
|
||||||
|
// method (not the constructor, so it's not a true const), but it doesn't
|
||||||
|
// change after that and it only gets used by actions caused, directly or
|
||||||
|
// indirectly, by that setup. This comment also applies to the reactor_
|
||||||
|
// variables of the other streaming objects in this file.
|
||||||
|
std::atomic<experimental::ServerUnaryReactor*> reactor_; |
||||||
|
// callbacks_outstanding_ follows a refcount pattern
|
||||||
|
std::atomic<intptr_t> callbacks_outstanding_{ |
||||||
|
3}; // reserve for start, Finish, and CompletionOp
|
||||||
|
}; |
||||||
|
}; |
||||||
|
|
||||||
|
template <class RequestType, class ResponseType> |
||||||
|
class CallbackClientStreamingHandler : public ::grpc::internal::MethodHandler { |
||||||
|
public: |
||||||
|
explicit CallbackClientStreamingHandler( |
||||||
|
std::function<experimental::ServerReadReactor<RequestType>*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*, ResponseType*)> |
||||||
|
get_reactor) |
||||||
|
: get_reactor_(std::move(get_reactor)) {} |
||||||
|
void RunHandler(const HandlerParameter& param) final { |
||||||
|
// Arena allocate a reader structure (that includes response)
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_ref(param.call->call()); |
||||||
|
|
||||||
|
auto* reader = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(ServerCallbackReaderImpl))) |
||||||
|
ServerCallbackReaderImpl( |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
param.call, std::move(param.call_requester)); |
||||||
|
param.server_context->BeginCompletionOp( |
||||||
|
param.call, [reader](bool) { reader->MaybeDone(); }, reader); |
||||||
|
|
||||||
|
experimental::ServerReadReactor<RequestType>* reactor = nullptr; |
||||||
|
if (param.status.ok()) { |
||||||
|
reactor = ::grpc::internal::CatchingReactorGetter< |
||||||
|
experimental::ServerReadReactor<RequestType>>( |
||||||
|
get_reactor_, |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
reader->response()); |
||||||
|
} |
||||||
|
|
||||||
|
if (reactor == nullptr) { |
||||||
|
// if deserialization or reactor creator failed, we need to fail the call
|
||||||
|
reactor = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(UnimplementedReadReactor<RequestType>))) |
||||||
|
UnimplementedReadReactor<RequestType>( |
||||||
|
::grpc::Status(::grpc::StatusCode::UNIMPLEMENTED, "")); |
||||||
|
} |
||||||
|
|
||||||
|
reader->SetupReactor(reactor); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
std::function<experimental::ServerReadReactor<RequestType>*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*, ResponseType*)> |
||||||
|
get_reactor_; |
||||||
|
|
||||||
|
class ServerCallbackReaderImpl |
||||||
|
: public experimental::ServerCallbackReader<RequestType> { |
||||||
|
public: |
||||||
|
void Finish(::grpc::Status s) override { |
||||||
|
finish_tag_.Set(call_.call(), [this](bool) { MaybeDone(); }, &finish_ops_, |
||||||
|
false); |
||||||
|
if (!ctx_->sent_initial_metadata_) { |
||||||
|
finish_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
finish_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
} |
||||||
|
// The response is dropped if the status is not OK.
|
||||||
|
if (s.ok()) { |
||||||
|
finish_ops_.ServerSendStatus(&ctx_->trailing_metadata_, |
||||||
|
finish_ops_.SendMessagePtr(&resp_)); |
||||||
|
} else { |
||||||
|
finish_ops_.ServerSendStatus(&ctx_->trailing_metadata_, s); |
||||||
|
} |
||||||
|
finish_ops_.set_core_cq_tag(&finish_tag_); |
||||||
|
call_.PerformOps(&finish_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void SendInitialMetadata() override { |
||||||
|
GPR_CODEGEN_ASSERT(!ctx_->sent_initial_metadata_); |
||||||
|
this->Ref(); |
||||||
|
meta_tag_.Set(call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed) |
||||||
|
->OnSendInitialMetadataDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&meta_ops_, false); |
||||||
|
meta_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
meta_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
meta_ops_.set_core_cq_tag(&meta_tag_); |
||||||
|
call_.PerformOps(&meta_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void Read(RequestType* req) override { |
||||||
|
this->Ref(); |
||||||
|
read_ops_.RecvMessage(req); |
||||||
|
call_.PerformOps(&read_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
friend class CallbackClientStreamingHandler<RequestType, ResponseType>; |
||||||
|
|
||||||
|
ServerCallbackReaderImpl( |
||||||
|
::grpc_impl::experimental::CallbackServerContext* ctx, |
||||||
|
::grpc::internal::Call* call, std::function<void()> call_requester) |
||||||
|
: ctx_(ctx), call_(*call), call_requester_(std::move(call_requester)) {} |
||||||
|
|
||||||
|
void SetupReactor(experimental::ServerReadReactor<RequestType>* reactor) { |
||||||
|
reactor_.store(reactor, std::memory_order_relaxed); |
||||||
|
read_tag_.Set(call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnReadDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&read_ops_, false); |
||||||
|
read_ops_.set_core_cq_tag(&read_tag_); |
||||||
|
this->BindReactor(reactor); |
||||||
|
this->MaybeCallOnCancel(reactor); |
||||||
|
this->MaybeDone(); |
||||||
|
} |
||||||
|
|
||||||
|
~ServerCallbackReaderImpl() {} |
||||||
|
|
||||||
|
ResponseType* response() { return &resp_; } |
||||||
|
|
||||||
|
void MaybeDone() override { |
||||||
|
if (GPR_UNLIKELY(this->Unref() == 1)) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnDone(); |
||||||
|
grpc_call* call = call_.call(); |
||||||
|
auto call_requester = std::move(call_requester_); |
||||||
|
this->~ServerCallbackReaderImpl(); // explicitly call destructor
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_unref(call); |
||||||
|
call_requester(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ServerReactor* reactor() override { |
||||||
|
return reactor_.load(std::memory_order_relaxed); |
||||||
|
} |
||||||
|
|
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata> |
||||||
|
meta_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag meta_tag_; |
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata, |
||||||
|
::grpc::internal::CallOpSendMessage, |
||||||
|
::grpc::internal::CallOpServerSendStatus> |
||||||
|
finish_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag finish_tag_; |
||||||
|
::grpc::internal::CallOpSet< |
||||||
|
::grpc::internal::CallOpRecvMessage<RequestType>> |
||||||
|
read_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag read_tag_; |
||||||
|
|
||||||
|
::grpc_impl::experimental::CallbackServerContext* const ctx_; |
||||||
|
::grpc::internal::Call call_; |
||||||
|
ResponseType resp_; |
||||||
|
std::function<void()> call_requester_; |
||||||
|
// The memory ordering of reactor_ follows ServerCallbackUnaryImpl.
|
||||||
|
std::atomic<experimental::ServerReadReactor<RequestType>*> reactor_; |
||||||
|
// callbacks_outstanding_ follows a refcount pattern
|
||||||
|
std::atomic<intptr_t> callbacks_outstanding_{ |
||||||
|
3}; // reserve for OnStarted, Finish, and CompletionOp
|
||||||
|
}; |
||||||
|
}; |
||||||
|
|
||||||
|
template <class RequestType, class ResponseType> |
||||||
|
class CallbackServerStreamingHandler : public ::grpc::internal::MethodHandler { |
||||||
|
public: |
||||||
|
explicit CallbackServerStreamingHandler( |
||||||
|
std::function<experimental::ServerWriteReactor<ResponseType>*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*, |
||||||
|
const RequestType*)> |
||||||
|
get_reactor) |
||||||
|
: get_reactor_(std::move(get_reactor)) {} |
||||||
|
void RunHandler(const HandlerParameter& param) final { |
||||||
|
// Arena allocate a writer structure
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_ref(param.call->call()); |
||||||
|
|
||||||
|
auto* writer = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(ServerCallbackWriterImpl))) |
||||||
|
ServerCallbackWriterImpl( |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
param.call, static_cast<RequestType*>(param.request), |
||||||
|
std::move(param.call_requester)); |
||||||
|
param.server_context->BeginCompletionOp( |
||||||
|
param.call, [writer](bool) { writer->MaybeDone(); }, writer); |
||||||
|
|
||||||
|
experimental::ServerWriteReactor<ResponseType>* reactor = nullptr; |
||||||
|
if (param.status.ok()) { |
||||||
|
reactor = ::grpc::internal::CatchingReactorGetter< |
||||||
|
experimental::ServerWriteReactor<ResponseType>>( |
||||||
|
get_reactor_, |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
writer->request()); |
||||||
|
} |
||||||
|
if (reactor == nullptr) { |
||||||
|
// if deserialization or reactor creator failed, we need to fail the call
|
||||||
|
reactor = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(UnimplementedWriteReactor<ResponseType>))) |
||||||
|
UnimplementedWriteReactor<ResponseType>( |
||||||
|
::grpc::Status(::grpc::StatusCode::UNIMPLEMENTED, "")); |
||||||
|
} |
||||||
|
|
||||||
|
writer->SetupReactor(reactor); |
||||||
|
} |
||||||
|
|
||||||
|
void* Deserialize(grpc_call* call, grpc_byte_buffer* req, |
||||||
|
::grpc::Status* status, void** /*handler_data*/) final { |
||||||
|
::grpc::ByteBuffer buf; |
||||||
|
buf.set_buffer(req); |
||||||
|
auto* request = |
||||||
|
new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
call, sizeof(RequestType))) RequestType(); |
||||||
|
*status = |
||||||
|
::grpc::SerializationTraits<RequestType>::Deserialize(&buf, request); |
||||||
|
buf.Release(); |
||||||
|
if (status->ok()) { |
||||||
|
return request; |
||||||
|
} |
||||||
|
request->~RequestType(); |
||||||
|
return nullptr; |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
std::function<experimental::ServerWriteReactor<ResponseType>*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*, const RequestType*)> |
||||||
|
get_reactor_; |
||||||
|
|
||||||
|
class ServerCallbackWriterImpl |
||||||
|
: public experimental::ServerCallbackWriter<ResponseType> { |
||||||
|
public: |
||||||
|
void Finish(::grpc::Status s) override { |
||||||
|
finish_tag_.Set(call_.call(), [this](bool) { MaybeDone(); }, &finish_ops_, |
||||||
|
false); |
||||||
|
finish_ops_.set_core_cq_tag(&finish_tag_); |
||||||
|
|
||||||
|
if (!ctx_->sent_initial_metadata_) { |
||||||
|
finish_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
finish_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
} |
||||||
|
finish_ops_.ServerSendStatus(&ctx_->trailing_metadata_, s); |
||||||
|
call_.PerformOps(&finish_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void SendInitialMetadata() override { |
||||||
|
GPR_CODEGEN_ASSERT(!ctx_->sent_initial_metadata_); |
||||||
|
this->Ref(); |
||||||
|
meta_tag_.Set(call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed) |
||||||
|
->OnSendInitialMetadataDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&meta_ops_, false); |
||||||
|
meta_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
meta_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
meta_ops_.set_core_cq_tag(&meta_tag_); |
||||||
|
call_.PerformOps(&meta_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void Write(const ResponseType* resp, |
||||||
|
::grpc::WriteOptions options) override { |
||||||
|
this->Ref(); |
||||||
|
if (options.is_last_message()) { |
||||||
|
options.set_buffer_hint(); |
||||||
|
} |
||||||
|
if (!ctx_->sent_initial_metadata_) { |
||||||
|
write_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
write_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
} |
||||||
|
// TODO(vjpai): don't assert
|
||||||
|
GPR_CODEGEN_ASSERT(write_ops_.SendMessagePtr(resp, options).ok()); |
||||||
|
call_.PerformOps(&write_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void WriteAndFinish(const ResponseType* resp, ::grpc::WriteOptions options, |
||||||
|
::grpc::Status s) override { |
||||||
|
// This combines the write into the finish callback
|
||||||
|
// Don't send any message if the status is bad
|
||||||
|
if (s.ok()) { |
||||||
|
// TODO(vjpai): don't assert
|
||||||
|
GPR_CODEGEN_ASSERT(finish_ops_.SendMessagePtr(resp, options).ok()); |
||||||
|
} |
||||||
|
Finish(std::move(s)); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
friend class CallbackServerStreamingHandler<RequestType, ResponseType>; |
||||||
|
|
||||||
|
ServerCallbackWriterImpl( |
||||||
|
::grpc_impl::experimental::CallbackServerContext* ctx, |
||||||
|
::grpc::internal::Call* call, const RequestType* req, |
||||||
|
std::function<void()> call_requester) |
||||||
|
: ctx_(ctx), |
||||||
|
call_(*call), |
||||||
|
req_(req), |
||||||
|
call_requester_(std::move(call_requester)) {} |
||||||
|
|
||||||
|
void SetupReactor(experimental::ServerWriteReactor<ResponseType>* reactor) { |
||||||
|
reactor_.store(reactor, std::memory_order_relaxed); |
||||||
|
write_tag_.Set( |
||||||
|
call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnWriteDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&write_ops_, false); |
||||||
|
write_ops_.set_core_cq_tag(&write_tag_); |
||||||
|
this->BindReactor(reactor); |
||||||
|
this->MaybeCallOnCancel(reactor); |
||||||
|
this->MaybeDone(); |
||||||
|
} |
||||||
|
~ServerCallbackWriterImpl() { req_->~RequestType(); } |
||||||
|
|
||||||
|
const RequestType* request() { return req_; } |
||||||
|
|
||||||
|
void MaybeDone() override { |
||||||
|
if (GPR_UNLIKELY(this->Unref() == 1)) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnDone(); |
||||||
|
grpc_call* call = call_.call(); |
||||||
|
auto call_requester = std::move(call_requester_); |
||||||
|
this->~ServerCallbackWriterImpl(); // explicitly call destructor
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_unref(call); |
||||||
|
call_requester(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ServerReactor* reactor() override { |
||||||
|
return reactor_.load(std::memory_order_relaxed); |
||||||
|
} |
||||||
|
|
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata> |
||||||
|
meta_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag meta_tag_; |
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata, |
||||||
|
::grpc::internal::CallOpSendMessage, |
||||||
|
::grpc::internal::CallOpServerSendStatus> |
||||||
|
finish_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag finish_tag_; |
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata, |
||||||
|
::grpc::internal::CallOpSendMessage> |
||||||
|
write_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag write_tag_; |
||||||
|
|
||||||
|
::grpc_impl::experimental::CallbackServerContext* const ctx_; |
||||||
|
::grpc::internal::Call call_; |
||||||
|
const RequestType* req_; |
||||||
|
std::function<void()> call_requester_; |
||||||
|
// The memory ordering of reactor_ follows ServerCallbackUnaryImpl.
|
||||||
|
std::atomic<experimental::ServerWriteReactor<ResponseType>*> reactor_; |
||||||
|
// callbacks_outstanding_ follows a refcount pattern
|
||||||
|
std::atomic<intptr_t> callbacks_outstanding_{ |
||||||
|
3}; // reserve for OnStarted, Finish, and CompletionOp
|
||||||
|
}; |
||||||
|
}; |
||||||
|
|
||||||
|
template <class RequestType, class ResponseType> |
||||||
|
class CallbackBidiHandler : public ::grpc::internal::MethodHandler { |
||||||
|
public: |
||||||
|
explicit CallbackBidiHandler( |
||||||
|
std::function<experimental::ServerBidiReactor<RequestType, ResponseType>*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*)> |
||||||
|
get_reactor) |
||||||
|
: get_reactor_(std::move(get_reactor)) {} |
||||||
|
void RunHandler(const HandlerParameter& param) final { |
||||||
|
::grpc::g_core_codegen_interface->grpc_call_ref(param.call->call()); |
||||||
|
|
||||||
|
auto* stream = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), sizeof(ServerCallbackReaderWriterImpl))) |
||||||
|
ServerCallbackReaderWriterImpl( |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context), |
||||||
|
param.call, std::move(param.call_requester)); |
||||||
|
param.server_context->BeginCompletionOp( |
||||||
|
param.call, [stream](bool) { stream->MaybeDone(); }, stream); |
||||||
|
|
||||||
|
experimental::ServerBidiReactor<RequestType, ResponseType>* reactor = |
||||||
|
nullptr; |
||||||
|
if (param.status.ok()) { |
||||||
|
reactor = ::grpc::internal::CatchingReactorGetter< |
||||||
|
experimental::ServerBidiReactor<RequestType, ResponseType>>( |
||||||
|
get_reactor_, |
||||||
|
static_cast<::grpc_impl::experimental::CallbackServerContext*>( |
||||||
|
param.server_context)); |
||||||
|
} |
||||||
|
|
||||||
|
if (reactor == nullptr) { |
||||||
|
// if deserialization or reactor creator failed, we need to fail the call
|
||||||
|
reactor = new (::grpc::g_core_codegen_interface->grpc_call_arena_alloc( |
||||||
|
param.call->call(), |
||||||
|
sizeof(UnimplementedBidiReactor<RequestType, ResponseType>))) |
||||||
|
UnimplementedBidiReactor<RequestType, ResponseType>( |
||||||
|
::grpc::Status(::grpc::StatusCode::UNIMPLEMENTED, "")); |
||||||
|
} |
||||||
|
|
||||||
|
stream->SetupReactor(reactor); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
std::function<experimental::ServerBidiReactor<RequestType, ResponseType>*( |
||||||
|
::grpc_impl::experimental::CallbackServerContext*)> |
||||||
|
get_reactor_; |
||||||
|
|
||||||
|
class ServerCallbackReaderWriterImpl |
||||||
|
: public experimental::ServerCallbackReaderWriter<RequestType, |
||||||
|
ResponseType> { |
||||||
|
public: |
||||||
|
void Finish(::grpc::Status s) override { |
||||||
|
finish_tag_.Set(call_.call(), [this](bool) { MaybeDone(); }, &finish_ops_, |
||||||
|
false); |
||||||
|
finish_ops_.set_core_cq_tag(&finish_tag_); |
||||||
|
|
||||||
|
if (!ctx_->sent_initial_metadata_) { |
||||||
|
finish_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
finish_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
} |
||||||
|
finish_ops_.ServerSendStatus(&ctx_->trailing_metadata_, s); |
||||||
|
call_.PerformOps(&finish_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void SendInitialMetadata() override { |
||||||
|
GPR_CODEGEN_ASSERT(!ctx_->sent_initial_metadata_); |
||||||
|
this->Ref(); |
||||||
|
meta_tag_.Set(call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed) |
||||||
|
->OnSendInitialMetadataDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&meta_ops_, false); |
||||||
|
meta_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
meta_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
meta_ops_.set_core_cq_tag(&meta_tag_); |
||||||
|
call_.PerformOps(&meta_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void Write(const ResponseType* resp, |
||||||
|
::grpc::WriteOptions options) override { |
||||||
|
this->Ref(); |
||||||
|
if (options.is_last_message()) { |
||||||
|
options.set_buffer_hint(); |
||||||
|
} |
||||||
|
if (!ctx_->sent_initial_metadata_) { |
||||||
|
write_ops_.SendInitialMetadata(&ctx_->initial_metadata_, |
||||||
|
ctx_->initial_metadata_flags()); |
||||||
|
if (ctx_->compression_level_set()) { |
||||||
|
write_ops_.set_compression_level(ctx_->compression_level()); |
||||||
|
} |
||||||
|
ctx_->sent_initial_metadata_ = true; |
||||||
|
} |
||||||
|
// TODO(vjpai): don't assert
|
||||||
|
GPR_CODEGEN_ASSERT(write_ops_.SendMessagePtr(resp, options).ok()); |
||||||
|
call_.PerformOps(&write_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
void WriteAndFinish(const ResponseType* resp, ::grpc::WriteOptions options, |
||||||
|
::grpc::Status s) override { |
||||||
|
// Don't send any message if the status is bad
|
||||||
|
if (s.ok()) { |
||||||
|
// TODO(vjpai): don't assert
|
||||||
|
GPR_CODEGEN_ASSERT(finish_ops_.SendMessagePtr(resp, options).ok()); |
||||||
|
} |
||||||
|
Finish(std::move(s)); |
||||||
|
} |
||||||
|
|
||||||
|
void Read(RequestType* req) override { |
||||||
|
this->Ref(); |
||||||
|
read_ops_.RecvMessage(req); |
||||||
|
call_.PerformOps(&read_ops_); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
friend class CallbackBidiHandler<RequestType, ResponseType>; |
||||||
|
|
||||||
|
ServerCallbackReaderWriterImpl( |
||||||
|
::grpc_impl::experimental::CallbackServerContext* ctx, |
||||||
|
::grpc::internal::Call* call, std::function<void()> call_requester) |
||||||
|
: ctx_(ctx), call_(*call), call_requester_(std::move(call_requester)) {} |
||||||
|
|
||||||
|
void SetupReactor( |
||||||
|
experimental::ServerBidiReactor<RequestType, ResponseType>* reactor) { |
||||||
|
reactor_.store(reactor, std::memory_order_relaxed); |
||||||
|
write_tag_.Set( |
||||||
|
call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnWriteDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&write_ops_, false); |
||||||
|
write_ops_.set_core_cq_tag(&write_tag_); |
||||||
|
read_tag_.Set(call_.call(), |
||||||
|
[this](bool ok) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnReadDone(ok); |
||||||
|
MaybeDone(); |
||||||
|
}, |
||||||
|
&read_ops_, false); |
||||||
|
read_ops_.set_core_cq_tag(&read_tag_); |
||||||
|
this->BindReactor(reactor); |
||||||
|
this->MaybeCallOnCancel(reactor); |
||||||
|
this->MaybeDone(); |
||||||
|
} |
||||||
|
|
||||||
|
void MaybeDone() override { |
||||||
|
if (GPR_UNLIKELY(this->Unref() == 1)) { |
||||||
|
reactor_.load(std::memory_order_relaxed)->OnDone(); |
||||||
|
grpc_call* call = call_.call(); |
||||||
|
auto call_requester = std::move(call_requester_); |
||||||
|
this->~ServerCallbackReaderWriterImpl(); // explicitly call destructor
|
||||||
|
::grpc::g_core_codegen_interface->grpc_call_unref(call); |
||||||
|
call_requester(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ServerReactor* reactor() override { |
||||||
|
return reactor_.load(std::memory_order_relaxed); |
||||||
|
} |
||||||
|
|
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata> |
||||||
|
meta_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag meta_tag_; |
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata, |
||||||
|
::grpc::internal::CallOpSendMessage, |
||||||
|
::grpc::internal::CallOpServerSendStatus> |
||||||
|
finish_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag finish_tag_; |
||||||
|
::grpc::internal::CallOpSet<::grpc::internal::CallOpSendInitialMetadata, |
||||||
|
::grpc::internal::CallOpSendMessage> |
||||||
|
write_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag write_tag_; |
||||||
|
::grpc::internal::CallOpSet< |
||||||
|
::grpc::internal::CallOpRecvMessage<RequestType>> |
||||||
|
read_ops_; |
||||||
|
::grpc::internal::CallbackWithSuccessTag read_tag_; |
||||||
|
|
||||||
|
::grpc_impl::experimental::CallbackServerContext* const ctx_; |
||||||
|
::grpc::internal::Call call_; |
||||||
|
std::function<void()> call_requester_; |
||||||
|
// The memory ordering of reactor_ follows ServerCallbackUnaryImpl.
|
||||||
|
std::atomic<experimental::ServerBidiReactor<RequestType, ResponseType>*> |
||||||
|
reactor_; |
||||||
|
// callbacks_outstanding_ follows a refcount pattern
|
||||||
|
std::atomic<intptr_t> callbacks_outstanding_{ |
||||||
|
3}; // reserve for OnStarted, Finish, and CompletionOp
|
||||||
|
}; |
||||||
|
}; |
||||||
|
|
||||||
|
} // namespace internal
|
||||||
|
} // namespace grpc_impl
|
||||||
|
|
||||||
|
#endif // GRPCPP_IMPL_CODEGEN_SERVER_CALLBACK_HANDLERS_H
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,55 @@ |
|||||||
|
/*
|
||||||
|
* |
||||||
|
* Copyright 2019 gRPC authors. |
||||||
|
* |
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
* you may not use this file except in compliance with the License. |
||||||
|
* You may obtain a copy of the License at |
||||||
|
* |
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* |
||||||
|
* Unless required by applicable law or agreed to in writing, software |
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
* See the License for the specific language governing permissions and |
||||||
|
* limitations under the License. |
||||||
|
* |
||||||
|
*/ |
||||||
|
|
||||||
|
#ifndef GRPCPP_TEST_DEFAULT_REACTOR_TEST_PEER_H |
||||||
|
#define GRPCPP_TEST_DEFAULT_REACTOR_TEST_PEER_H |
||||||
|
|
||||||
|
#include <grpcpp/server_context.h> |
||||||
|
#include <grpcpp/support/server_callback.h> |
||||||
|
|
||||||
|
namespace grpc { |
||||||
|
namespace testing { |
||||||
|
|
||||||
|
/// A test-only class to monitor the behavior of the ServerContext's
|
||||||
|
/// DefaultReactor. It is intended for allow unit-testing of a callback API
|
||||||
|
/// service via direct invocation of the service methods rather than through
|
||||||
|
/// RPCs. It is only applicable for unary RPC methods that use the
|
||||||
|
/// DefaultReactor rather than any user-defined reactor.
|
||||||
|
class DefaultReactorTestPeer { |
||||||
|
public: |
||||||
|
explicit DefaultReactorTestPeer(experimental::CallbackServerContext* ctx) |
||||||
|
: DefaultReactorTestPeer(ctx, [](::grpc::Status) {}) {} |
||||||
|
DefaultReactorTestPeer(experimental::CallbackServerContext* ctx, |
||||||
|
std::function<void(::grpc::Status)> finish_func) |
||||||
|
: ctx_(ctx) { |
||||||
|
ctx->SetupTestDefaultReactor(std::move(finish_func)); |
||||||
|
} |
||||||
|
::grpc::experimental::ServerUnaryReactor* reactor() const { |
||||||
|
return &ctx_->default_reactor_; |
||||||
|
} |
||||||
|
bool test_status_set() const { return ctx_->test_status_set(); } |
||||||
|
Status test_status() const { return ctx_->test_status(); } |
||||||
|
|
||||||
|
private: |
||||||
|
experimental::CallbackServerContext* const ctx_; // not owned
|
||||||
|
}; |
||||||
|
|
||||||
|
} // namespace testing
|
||||||
|
} // namespace grpc
|
||||||
|
|
||||||
|
#endif // GRPCPP_TEST_DEFAULT_REACTOR_TEST_PEER_H
|
@ -0,0 +1,52 @@ |
|||||||
|
/*
|
||||||
|
* Copyright 2019 gRPC authors. |
||||||
|
* |
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
* you may not use this file except in compliance with the License. |
||||||
|
* You may obtain a copy of the License at |
||||||
|
* |
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* |
||||||
|
* Unless required by applicable law or agreed to in writing, software |
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
* See the License for the specific language governing permissions and |
||||||
|
* limitations under the License. |
||||||
|
* |
||||||
|
*/ |
||||||
|
|
||||||
|
#include <grpcpp/impl/codegen/server_callback_impl.h> |
||||||
|
|
||||||
|
#include "src/core/lib/iomgr/closure.h" |
||||||
|
#include "src/core/lib/iomgr/exec_ctx.h" |
||||||
|
#include "src/core/lib/iomgr/executor.h" |
||||||
|
|
||||||
|
namespace grpc_impl { |
||||||
|
namespace internal { |
||||||
|
|
||||||
|
void ServerCallbackCall::CallOnCancel(ServerReactor* reactor) { |
||||||
|
if (reactor->InternalInlineable()) { |
||||||
|
reactor->OnCancel(); |
||||||
|
} else { |
||||||
|
Ref(); |
||||||
|
grpc_core::ExecCtx exec_ctx; |
||||||
|
struct ClosureArg { |
||||||
|
ServerCallbackCall* call; |
||||||
|
ServerReactor* reactor; |
||||||
|
}; |
||||||
|
ClosureArg* arg = new ClosureArg{this, reactor}; |
||||||
|
grpc_core::Executor::Run(GRPC_CLOSURE_CREATE( |
||||||
|
[](void* void_arg, grpc_error*) { |
||||||
|
ClosureArg* arg = |
||||||
|
static_cast<ClosureArg*>(void_arg); |
||||||
|
arg->reactor->OnCancel(); |
||||||
|
arg->call->MaybeDone(); |
||||||
|
delete arg; |
||||||
|
}, |
||||||
|
arg, nullptr), |
||||||
|
GRPC_ERROR_NONE); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace internal
|
||||||
|
} // namespace grpc_impl
|
Loading…
Reference in new issue