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