mirror of https://github.com/grpc/grpc.git
[event_engine] Thread pool that can handle deletion in a callback (#30763)
* [event_engine] Thread pool that can handle deletion in a callback * missed file * Automated change: Fix sanity tests * simplify memory model * review feedback * Automated change: Fix sanity tests * detect-stuckness * fix * Automated change: Fix sanity tests Co-authored-by: ctiller <ctiller@users.noreply.github.com>pull/30964/head
parent
cb3d7a9e9e
commit
f6e1cf1dc0
8 changed files with 341 additions and 117 deletions
@ -0,0 +1,98 @@ |
|||||||
|
// Copyright 2022 The 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 "src/core/lib/event_engine/thread_pool.h" |
||||||
|
|
||||||
|
#include <atomic> |
||||||
|
#include <chrono> |
||||||
|
#include <thread> |
||||||
|
|
||||||
|
#include <gtest/gtest.h> |
||||||
|
|
||||||
|
#include "absl/synchronization/notification.h" |
||||||
|
#include "gtest/gtest.h" |
||||||
|
|
||||||
|
#include <grpc/support/log.h> |
||||||
|
|
||||||
|
namespace grpc_event_engine { |
||||||
|
namespace experimental { |
||||||
|
|
||||||
|
TEST(ThreadPoolTest, CanRunClosure) { |
||||||
|
ThreadPool p(1); |
||||||
|
absl::Notification n; |
||||||
|
p.Add([&n] { n.Notify(); }); |
||||||
|
n.WaitForNotification(); |
||||||
|
} |
||||||
|
|
||||||
|
TEST(ThreadPoolTest, CanDestroyInsideClosure) { |
||||||
|
auto p = std::make_shared<ThreadPool>(1); |
||||||
|
p->Add([p]() { std::this_thread::sleep_for(std::chrono::seconds(1)); }); |
||||||
|
} |
||||||
|
|
||||||
|
TEST(ThreadPoolTest, CanSurviveFork) { |
||||||
|
ThreadPool p(1); |
||||||
|
absl::Notification n; |
||||||
|
gpr_log(GPR_INFO, "add callback 1"); |
||||||
|
p.Add([&n, &p] { |
||||||
|
std::this_thread::sleep_for(std::chrono::seconds(1)); |
||||||
|
gpr_log(GPR_INFO, "add callback 2"); |
||||||
|
p.Add([&n] { |
||||||
|
std::this_thread::sleep_for(std::chrono::seconds(1)); |
||||||
|
gpr_log(GPR_INFO, "notify"); |
||||||
|
n.Notify(); |
||||||
|
}); |
||||||
|
}); |
||||||
|
gpr_log(GPR_INFO, "prepare fork"); |
||||||
|
p.PrepareFork(); |
||||||
|
gpr_log(GPR_INFO, "wait for notification"); |
||||||
|
n.WaitForNotification(); |
||||||
|
gpr_log(GPR_INFO, "postfork child"); |
||||||
|
p.PostforkChild(); |
||||||
|
absl::Notification n2; |
||||||
|
gpr_log(GPR_INFO, "add callback 3"); |
||||||
|
p.Add([&n2] { |
||||||
|
gpr_log(GPR_INFO, "notify"); |
||||||
|
n2.Notify(); |
||||||
|
}); |
||||||
|
gpr_log(GPR_INFO, "wait for notification"); |
||||||
|
n2.WaitForNotification(); |
||||||
|
} |
||||||
|
|
||||||
|
void ScheduleSelf(ThreadPool* p) { |
||||||
|
p->Add([p] { ScheduleSelf(p); }); |
||||||
|
} |
||||||
|
|
||||||
|
TEST(ThreadPoolDeathTest, CanDetectStucknessAtFork) { |
||||||
|
ASSERT_DEATH_IF_SUPPORTED( |
||||||
|
[] { |
||||||
|
gpr_set_log_verbosity(GPR_LOG_SEVERITY_ERROR); |
||||||
|
ThreadPool p(1); |
||||||
|
ScheduleSelf(&p); |
||||||
|
std::thread terminator([] { |
||||||
|
std::this_thread::sleep_for(std::chrono::seconds(10)); |
||||||
|
abort(); |
||||||
|
}); |
||||||
|
p.PrepareFork(); |
||||||
|
}(), |
||||||
|
"Waiting for thread pool to idle before forking"); |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace experimental
|
||||||
|
} // namespace grpc_event_engine
|
||||||
|
|
||||||
|
int main(int argc, char** argv) { |
||||||
|
gpr_log_verbosity_init(); |
||||||
|
::testing::InitGoogleTest(&argc, argv); |
||||||
|
return RUN_ALL_TESTS(); |
||||||
|
} |
Loading…
Reference in new issue