2019-03-17 00:59:46 +00:00
|
|
|
/*
|
|
|
|
* This file is part of OpenTTD.
|
|
|
|
* OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
|
|
|
|
* OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
* See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** @file thread.h Base of all threads. */
|
|
|
|
|
|
|
|
#ifndef THREAD_H
|
|
|
|
#define THREAD_H
|
|
|
|
|
|
|
|
#include "debug.h"
|
2021-03-13 20:34:51 +00:00
|
|
|
#include "crashlog.h"
|
2019-03-17 00:59:46 +00:00
|
|
|
#include <system_error>
|
|
|
|
#include <thread>
|
2021-06-17 18:10:40 +00:00
|
|
|
#include <mutex>
|
2019-03-17 00:59:46 +00:00
|
|
|
|
2019-03-17 10:05:53 +00:00
|
|
|
/**
|
|
|
|
* Sleep on the current thread for a defined time.
|
|
|
|
* @param milliseconds Time to sleep for in milliseconds.
|
|
|
|
*/
|
|
|
|
inline void CSleep(int milliseconds)
|
|
|
|
{
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(milliseconds));
|
|
|
|
}
|
|
|
|
|
2019-03-17 00:59:46 +00:00
|
|
|
/**
|
|
|
|
* Name the thread this function is called on for the debugger.
|
|
|
|
* @param name Name to set for the thread..
|
|
|
|
*/
|
|
|
|
void SetCurrentThreadName(const char *name);
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start a new thread.
|
|
|
|
* @tparam TFn Type of the function to call on the thread.
|
|
|
|
* @tparam TArgs Type of the parameters of the thread function.
|
2019-04-10 21:07:06 +00:00
|
|
|
* @param thr Pointer to a thread object; may be \c nullptr if a detached thread is wanted.
|
2019-03-17 00:59:46 +00:00
|
|
|
* @param name Name of the thread.
|
|
|
|
* @param _Fx Function to call on the thread.
|
|
|
|
* @param _Ax Arguments for the thread function.
|
|
|
|
* @return True if the thread was successfully started, false otherwise.
|
|
|
|
*/
|
|
|
|
template<class TFn, class... TArgs>
|
|
|
|
inline bool StartNewThread(std::thread *thr, const char *name, TFn&& _Fx, TArgs&&... _Ax)
|
|
|
|
{
|
|
|
|
#ifndef NO_THREADS
|
|
|
|
try {
|
2021-06-17 18:10:40 +00:00
|
|
|
static std::mutex thread_startup_mutex;
|
|
|
|
std::lock_guard<std::mutex> lock(thread_startup_mutex);
|
|
|
|
|
2019-03-17 00:59:46 +00:00
|
|
|
std::thread t([] (const char *name, TFn&& F, TArgs&&... A) {
|
2021-06-17 18:10:40 +00:00
|
|
|
/* Delay starting the thread till the main thread is finished
|
|
|
|
* with the administration. This prevent race-conditions on
|
|
|
|
* startup. */
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock(thread_startup_mutex);
|
|
|
|
}
|
|
|
|
|
2019-03-17 00:59:46 +00:00
|
|
|
SetCurrentThreadName(name);
|
2021-03-13 20:34:51 +00:00
|
|
|
CrashLog::InitThread();
|
2019-03-17 00:59:46 +00:00
|
|
|
try {
|
|
|
|
/* Call user function with the given arguments. */
|
|
|
|
F(A...);
|
|
|
|
} catch (...) {
|
|
|
|
NOT_REACHED();
|
|
|
|
}
|
|
|
|
}, name, std::forward<TFn>(_Fx), std::forward<TArgs>(_Ax)...);
|
|
|
|
|
2019-04-10 21:07:06 +00:00
|
|
|
if (thr != nullptr) {
|
2019-03-17 00:59:46 +00:00
|
|
|
*thr = std::move(t);
|
|
|
|
} else {
|
|
|
|
t.detach();
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} catch (const std::system_error& e) {
|
|
|
|
/* Something went wrong, the system we are running on might not support threads. */
|
2021-06-12 07:10:17 +00:00
|
|
|
Debug(misc, 1, "Can't create thread '{}': {}", name, e.what());
|
2019-03-17 00:59:46 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* THREAD_H */
|