• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2023 gRPC authors.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #include "src/core/lib/event_engine/thready_event_engine/thready_event_engine.h"
16 
17 #include <grpc/support/port_platform.h>
18 
19 #include <memory>
20 #include <string>
21 #include <type_traits>
22 #include <vector>
23 
24 #include "src/core/util/crash.h"
25 #include "src/core/util/sync.h"
26 #include "src/core/util/thd.h"
27 
28 namespace grpc_event_engine {
29 namespace experimental {
30 
Asynchronously(absl::AnyInvocable<void ()> fn)31 void ThreadyEventEngine::Asynchronously(absl::AnyInvocable<void()> fn) {
32   grpc_core::Thread t("thready_event_engine", std::move(fn), nullptr,
33                       grpc_core::Thread::Options().set_joinable(false));
34   t.Start();
35 }
36 
37 absl::StatusOr<std::unique_ptr<EventEngine::Listener>>
CreateListener(Listener::AcceptCallback on_accept,absl::AnyInvocable<void (absl::Status)> on_shutdown,const EndpointConfig & config,std::unique_ptr<MemoryAllocatorFactory> memory_allocator_factory)38 ThreadyEventEngine::CreateListener(
39     Listener::AcceptCallback on_accept,
40     absl::AnyInvocable<void(absl::Status)> on_shutdown,
41     const EndpointConfig& config,
42     std::unique_ptr<MemoryAllocatorFactory> memory_allocator_factory) {
43   struct AcceptState {
44     grpc_core::Mutex mu_;
45     grpc_core::CondVar cv_;
46     int pending_accepts_ ABSL_GUARDED_BY(mu_) = 0;
47   };
48   auto accept_state = std::make_shared<AcceptState>();
49   return impl_->CreateListener(
50       [this, accept_state,
51        on_accept = std::make_shared<Listener::AcceptCallback>(
52            std::move(on_accept))](std::unique_ptr<Endpoint> endpoint,
53                                   MemoryAllocator memory_allocator) {
54         {
55           grpc_core::MutexLock lock(&accept_state->mu_);
56           ++accept_state->pending_accepts_;
57         }
58         Asynchronously(
59             [on_accept, accept_state, endpoint = std::move(endpoint),
60              memory_allocator = std::move(memory_allocator)]() mutable {
61               (*on_accept)(std::move(endpoint), std::move(memory_allocator));
62               {
63                 grpc_core::MutexLock lock(&accept_state->mu_);
64                 --accept_state->pending_accepts_;
65                 if (accept_state->pending_accepts_ == 0) {
66                   accept_state->cv_.Signal();
67                 }
68               }
69             });
70       },
71       [this, accept_state,
72        on_shutdown = std::move(on_shutdown)](absl::Status status) mutable {
73         Asynchronously([accept_state, on_shutdown = std::move(on_shutdown),
74                         status = std::move(status)]() mutable {
75           while (true) {
76             grpc_core::MutexLock lock(&accept_state->mu_);
77             if (accept_state->pending_accepts_ == 0) {
78               break;
79             }
80             accept_state->cv_.Wait(&accept_state->mu_);
81           }
82           on_shutdown(std::move(status));
83         });
84       },
85       config, std::move(memory_allocator_factory));
86 }
87 
Connect(OnConnectCallback on_connect,const ResolvedAddress & addr,const EndpointConfig & args,MemoryAllocator memory_allocator,Duration timeout)88 EventEngine::ConnectionHandle ThreadyEventEngine::Connect(
89     OnConnectCallback on_connect, const ResolvedAddress& addr,
90     const EndpointConfig& args, MemoryAllocator memory_allocator,
91     Duration timeout) {
92   return impl_->Connect(
93       [this, on_connect = std::move(on_connect)](
94           absl::StatusOr<std::unique_ptr<Endpoint>> c) mutable {
95         Asynchronously(
96             [on_connect = std::move(on_connect), c = std::move(c)]() mutable {
97               on_connect(std::move(c));
98             });
99       },
100       addr, args, std::move(memory_allocator), timeout);
101 }
102 
CancelConnect(ConnectionHandle handle)103 bool ThreadyEventEngine::CancelConnect(ConnectionHandle handle) {
104   return impl_->CancelConnect(handle);
105 }
106 
IsWorkerThread()107 bool ThreadyEventEngine::IsWorkerThread() {
108   grpc_core::Crash("we should remove this");
109 }
110 
111 absl::StatusOr<std::unique_ptr<EventEngine::DNSResolver>>
GetDNSResolver(const DNSResolver::ResolverOptions & options)112 ThreadyEventEngine::GetDNSResolver(
113     const DNSResolver::ResolverOptions& options) {
114   return std::make_unique<ThreadyDNSResolver>(
115       *impl_->GetDNSResolver(options),
116       std::static_pointer_cast<ThreadyEventEngine>(shared_from_this()));
117 }
118 
Run(Closure * closure)119 void ThreadyEventEngine::Run(Closure* closure) {
120   Run([closure]() { closure->Run(); });
121 }
122 
Run(absl::AnyInvocable<void ()> closure)123 void ThreadyEventEngine::Run(absl::AnyInvocable<void()> closure) {
124   Asynchronously(std::move(closure));
125 }
126 
RunAfter(Duration when,Closure * closure)127 EventEngine::TaskHandle ThreadyEventEngine::RunAfter(Duration when,
128                                                      Closure* closure) {
129   return RunAfter(when, [closure]() { closure->Run(); });
130 }
131 
RunAfter(Duration when,absl::AnyInvocable<void ()> closure)132 EventEngine::TaskHandle ThreadyEventEngine::RunAfter(
133     Duration when, absl::AnyInvocable<void()> closure) {
134   return impl_->RunAfter(when, [this, closure = std::move(closure)]() mutable {
135     Asynchronously(std::move(closure));
136   });
137 }
138 
Cancel(TaskHandle handle)139 bool ThreadyEventEngine::Cancel(TaskHandle handle) {
140   return impl_->Cancel(handle);
141 }
142 
LookupHostname(LookupHostnameCallback on_resolve,absl::string_view name,absl::string_view default_port)143 void ThreadyEventEngine::ThreadyDNSResolver::LookupHostname(
144     LookupHostnameCallback on_resolve, absl::string_view name,
145     absl::string_view default_port) {
146   return impl_->LookupHostname(
147       [engine = engine_, on_resolve = std::move(on_resolve)](
148           absl::StatusOr<std::vector<ResolvedAddress>> addresses) mutable {
149         engine->Asynchronously([on_resolve = std::move(on_resolve),
150                                 addresses = std::move(addresses)]() mutable {
151           on_resolve(std::move(addresses));
152         });
153       },
154       name, default_port);
155 }
156 
LookupSRV(LookupSRVCallback on_resolve,absl::string_view name)157 void ThreadyEventEngine::ThreadyDNSResolver::LookupSRV(
158     LookupSRVCallback on_resolve, absl::string_view name) {
159   return impl_->LookupSRV(
160       [engine = engine_, on_resolve = std::move(on_resolve)](
161           absl::StatusOr<std::vector<SRVRecord>> records) mutable {
162         return engine->Asynchronously([on_resolve = std::move(on_resolve),
163                                        records = std::move(records)]() mutable {
164           on_resolve(std::move(records));
165         });
166       },
167       name);
168 }
169 
LookupTXT(LookupTXTCallback on_resolve,absl::string_view name)170 void ThreadyEventEngine::ThreadyDNSResolver::LookupTXT(
171     LookupTXTCallback on_resolve, absl::string_view name) {
172   return impl_->LookupTXT(
173       [engine = engine_, on_resolve = std::move(on_resolve)](
174           absl::StatusOr<std::vector<std::string>> record) mutable {
175         return engine->Asynchronously([on_resolve = std::move(on_resolve),
176                                        record = std::move(record)]() mutable {
177           on_resolve(std::move(record));
178         });
179       },
180       name);
181 }
182 
183 }  // namespace experimental
184 }  // namespace grpc_event_engine
185