247 lines
6.7 KiB
C++
247 lines
6.7 KiB
C++
/*
|
|
* IThreadPoolTest.actor.cpp
|
|
*
|
|
* This source file is part of the FoundationDB open source project
|
|
*
|
|
* Copyright 2013-2024 Apple Inc. and the FoundationDB project 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.
|
|
*/
|
|
|
|
// Thread naming only works on Linux.
|
|
#if defined(__linux__)
|
|
|
|
#include "flow/IThreadPool.h"
|
|
|
|
#include <pthread.h>
|
|
#include <iostream>
|
|
|
|
#include "flow/UnitTest.h"
|
|
#include "flow/actorcompiler.h" // has to be last include
|
|
|
|
void forceLinkIThreadPoolTests() {}
|
|
|
|
struct ThreadNameReceiver final : IThreadPoolReceiver {
|
|
void init() override {}
|
|
|
|
struct GetNameAction final : TypedAction<ThreadNameReceiver, GetNameAction> {
|
|
ThreadReturnPromise<std::string> name;
|
|
|
|
double getTimeEstimate() const override { return 3.; }
|
|
};
|
|
|
|
void action(GetNameAction& a) {
|
|
pthread_t t = pthread_self();
|
|
const size_t arrayLen = 16;
|
|
char name[arrayLen];
|
|
int err = pthread_getname_np(t, name, arrayLen);
|
|
if (err != 0) {
|
|
std::cout << "Get name failed with error code: " << err << std::endl;
|
|
a.name.sendError(platform_error());
|
|
return;
|
|
}
|
|
std::string s = name;
|
|
ASSERT(a.name.isValid());
|
|
a.name.send(std::move(s));
|
|
ASSERT(!a.name.isValid());
|
|
}
|
|
};
|
|
|
|
TEST_CASE("/flow/IThreadPool/NamedThread") {
|
|
noUnseed = true;
|
|
|
|
state Reference<IThreadPool> pool = createGenericThreadPool();
|
|
pool->addThread(new ThreadNameReceiver(), "thread-foo");
|
|
|
|
// Warning: this action is a little racy with the call to `pthread_setname_np`. In practice,
|
|
// ~nothing should depend on the thread name being set instantaneously. If this test ever
|
|
// flakes, we can make `startThread` in platform a little bit more complex to clearly order
|
|
// the actions.
|
|
auto* a = new ThreadNameReceiver::GetNameAction();
|
|
auto fut = a->name.getFuture();
|
|
pool->post(a);
|
|
|
|
std::string name = wait(fut);
|
|
if (name != "thread-foo") {
|
|
std::cout << "Incorrect thread name: " << name << std::endl;
|
|
ASSERT(false);
|
|
}
|
|
|
|
wait(pool->stop());
|
|
|
|
return Void();
|
|
}
|
|
|
|
struct ThreadSafePromiseStreamSender final : IThreadPoolReceiver {
|
|
ThreadSafePromiseStreamSender(ThreadReturnPromiseStream<std::string>* notifications)
|
|
: notifications(notifications) {}
|
|
void init() override {}
|
|
|
|
struct GetNameAction final : TypedAction<ThreadSafePromiseStreamSender, GetNameAction> {
|
|
double getTimeEstimate() const override { return 3.; }
|
|
};
|
|
|
|
void action(GetNameAction& a) {
|
|
pthread_t t = pthread_self();
|
|
const size_t arrayLen = 16;
|
|
char name[arrayLen];
|
|
int err = pthread_getname_np(t, name, arrayLen);
|
|
if (err != 0) {
|
|
std::cout << "Get name failed with error code: " << err << std::endl;
|
|
notifications->sendError(platform_error());
|
|
return;
|
|
}
|
|
notifications->send(name);
|
|
}
|
|
|
|
struct FaultyAction final : TypedAction<ThreadSafePromiseStreamSender, FaultyAction> {
|
|
double getTimeEstimate() const override { return 3.; }
|
|
};
|
|
|
|
void action(FaultyAction& a) { notifications->sendError(platform_error().asInjectedFault()); }
|
|
|
|
private:
|
|
ThreadReturnPromiseStream<std::string>* notifications;
|
|
};
|
|
|
|
TEST_CASE("/flow/IThreadPool/ThreadReturnPromiseStream") {
|
|
noUnseed = true;
|
|
|
|
state std::unique_ptr<ThreadReturnPromiseStream<std::string>> notifications(
|
|
new ThreadReturnPromiseStream<std::string>());
|
|
|
|
state Reference<IThreadPool> pool = createGenericThreadPool();
|
|
pool->addThread(new ThreadSafePromiseStreamSender(notifications.get()), "thread-foo");
|
|
|
|
// Warning: this action is a little racy with the call to `pthread_setname_np`. In practice,
|
|
// ~nothing should depend on the thread name being set instantaneously. If this test ever
|
|
// flakes, we can make `startThread` in platform a little bit more complex to clearly order
|
|
// the actions.
|
|
state int num = 3;
|
|
for (int i = 0; i < num; ++i) {
|
|
auto* a = new ThreadSafePromiseStreamSender::GetNameAction();
|
|
pool->post(a);
|
|
}
|
|
|
|
state FutureStream<std::string> futs = notifications->getFuture();
|
|
|
|
state int n = 0;
|
|
while (n < num) {
|
|
std::string name = waitNext(futs);
|
|
if (name != "thread-foo") {
|
|
std::cout << "Incorrect thread name: " << name << std::endl;
|
|
ASSERT(false);
|
|
}
|
|
++n;
|
|
}
|
|
|
|
ASSERT(n == num);
|
|
|
|
auto* faultyAction = new ThreadSafePromiseStreamSender::FaultyAction();
|
|
pool->post(faultyAction);
|
|
|
|
try {
|
|
std::string name = waitNext(futs);
|
|
ASSERT(false);
|
|
} catch (Error& e) {
|
|
ASSERT(e.isInjectedFault());
|
|
}
|
|
|
|
wait(pool->stop());
|
|
|
|
return Void();
|
|
}
|
|
|
|
TEST_CASE("/flow/IThreadPool/ThreadReturnPromiseStream_DestroyPromise") {
|
|
{
|
|
std::cout << "ThreadReturnPromiseStream with future > 0, promise == 0, end_of_stream sent\n";
|
|
// After all references to PromiseStream are gone, FutureStream should still be able to get
|
|
// all the values if PromiseStream had reached end_of_stream.
|
|
state FutureStream<int> fs1 = ([]() {
|
|
ThreadReturnPromiseStream<int> p;
|
|
auto ret = p.getFuture();
|
|
for (int i = 0; i < 10; i++) {
|
|
p.send(i);
|
|
}
|
|
p.sendError(end_of_stream());
|
|
ASSERT(p.getFutureReferenceCount() == 1);
|
|
return ret;
|
|
})();
|
|
|
|
wait(delay(1));
|
|
|
|
state int recvd = 0;
|
|
try {
|
|
while (true) {
|
|
int _ = waitNext(fs1);
|
|
recvd += 1;
|
|
}
|
|
} catch (Error& err) {
|
|
if (err.code() == error_code_end_of_stream) {
|
|
ASSERT_EQ(recvd, 10);
|
|
} else {
|
|
ASSERT(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
std::cout << "ThreadReturnPromiseStream with future > 0, promise == 0, no end_of_stream\n";
|
|
{
|
|
// After all references to PromiseStream are gone, but the stream was not ended cleanly
|
|
// by sending end_of_stream, we should instead get broken_promise.
|
|
state FutureStream<int> fs2 = ([]() {
|
|
ThreadReturnPromiseStream<int> p;
|
|
auto ret = p.getFuture();
|
|
for (int i = 0; i < 10; i++) {
|
|
p.send(i);
|
|
}
|
|
ASSERT(p.getFutureReferenceCount() == 1);
|
|
return ret;
|
|
})();
|
|
|
|
wait(delay(1));
|
|
|
|
try {
|
|
while (true) {
|
|
choose {
|
|
when(int _ = waitNext(fs2)) {}
|
|
when(wait(delay(1))) {
|
|
ASSERT(false); // shouldn't hangup if end_of_stream is not sent.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
} catch (Error& err) {
|
|
ASSERT(err.code() == error_code_broken_promise);
|
|
}
|
|
}
|
|
|
|
return Void();
|
|
}
|
|
|
|
// FIXME: Someday fix ThreadReturnPromise for this testcase.
|
|
// TEST_CASE("/flow/IThreadPool/ThreadReturnPromiseGetFutureAfterSend") {
|
|
// auto get_f = [&]() {
|
|
// ThreadReturnPromise<int> p;
|
|
// p.send(1234);
|
|
// return p.getFuture();
|
|
// };
|
|
// int res = wait(get_f());
|
|
// ASSERT_EQ(res, 1234);
|
|
// return Void();
|
|
// }
|
|
|
|
#else
|
|
void forceLinkIThreadPoolTests() {}
|
|
#endif
|