mirror of
https://github.com/pocoproject/poco.git
synced 2024-12-12 18:20:26 +01:00
555 lines
12 KiB
C++
555 lines
12 KiB
C++
//
|
|
// Thread_POSIX.cpp
|
|
//
|
|
// Library: Foundation
|
|
// Package: Threading
|
|
// Module: Thread
|
|
//
|
|
// Copyright (c) 2004-2007, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "Poco/Thread_POSIX.h"
|
|
#include "Poco/Thread.h"
|
|
#include "Poco/Exception.h"
|
|
#include "Poco/Error.h"
|
|
#include "Poco/ErrorHandler.h"
|
|
#include "Poco/Format.h"
|
|
#include "Poco/Error.h"
|
|
#include <signal.h>
|
|
#include <limits.h>
|
|
|
|
|
|
#if POCO_OS == POCO_OS_FREE_BSD
|
|
# include <sys/thr.h>
|
|
# include <pthread_np.h>
|
|
# include <osreldate.h>
|
|
#endif
|
|
|
|
|
|
#if defined(__sun) && defined(__SVR4)
|
|
# if !defined(__EXTENSIONS__)
|
|
# define __EXTENSIONS__
|
|
# endif
|
|
#endif
|
|
#if POCO_OS == POCO_OS_LINUX || POCO_OS == POCO_OS_ANDROID || POCO_OS == POCO_OS_MAC_OS_X || POCO_OS == POCO_OS_QNX
|
|
# include <time.h>
|
|
#endif
|
|
|
|
#if POCO_OS == POCO_OS_QNX
|
|
# include <sys/neutrino.h>
|
|
#endif
|
|
|
|
#if POCO_OS == POCO_OS_LINUX || POCO_OS == POCO_OS_ANDROID
|
|
# include <sys/prctl.h>
|
|
#endif
|
|
|
|
#if POCO_OS == POCO_OS_LINUX
|
|
#ifndef _GNU_SOURCE
|
|
#define _GNU_SOURCE /* See feature_test_macros(7) */
|
|
#endif
|
|
#include <unistd.h>
|
|
#include <sys/syscall.h> /* For SYS_xxx definitions */
|
|
#endif
|
|
|
|
|
|
//
|
|
// Block SIGPIPE in main thread.
|
|
//
|
|
#if defined(POCO_OS_FAMILY_UNIX) && !defined(POCO_VXWORKS)
|
|
namespace
|
|
{
|
|
class SignalBlocker
|
|
{
|
|
public:
|
|
SignalBlocker()
|
|
{
|
|
sigset_t sset;
|
|
sigemptyset(&sset);
|
|
sigaddset(&sset, SIGPIPE);
|
|
pthread_sigmask(SIG_BLOCK, &sset, 0);
|
|
}
|
|
~SignalBlocker()
|
|
{
|
|
}
|
|
};
|
|
|
|
static SignalBlocker signalBlocker;
|
|
}
|
|
#endif
|
|
|
|
|
|
namespace
|
|
{
|
|
std::string truncName(const std::string& name, int nameSize = POCO_MAX_THREAD_NAME_LEN)
|
|
{
|
|
if (name.size() > nameSize)
|
|
return name.substr(0, nameSize).append("~");
|
|
return name;
|
|
}
|
|
|
|
#ifndef POCO_NO_THREADNAME
|
|
void setThreadName(const std::string& threadName)
|
|
/// Sets thread name. Support for this feature varies
|
|
/// on platforms. Any errors are ignored.
|
|
{
|
|
#if (POCO_OS == POCO_OS_FREE_BSD)
|
|
pthread_setname_np(pthread_self(), truncName(threadName).c_str());
|
|
#elif (POCO_OS == POCO_OS_MAC_OS_X)
|
|
#ifdef __MAC_OS_X_VERSION_MIN_REQUIRED
|
|
#if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1060
|
|
pthread_setname_np(truncName(threadName).c_str()); // __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_2)
|
|
#endif
|
|
#endif // __MAC_OS_X_VERSION_MIN_REQUIRED
|
|
#elif (POCO_OS == POCO_OS_QNX)
|
|
pthread_setname_np(pthread_self(), truncName(threadName, _NTO_THREAD_NAME_MAX).c_str());
|
|
#else
|
|
prctl(PR_SET_NAME, truncName(threadName).c_str());
|
|
#endif
|
|
}
|
|
|
|
std::string getThreadName()
|
|
{
|
|
constexpr size_t nameSize =
|
|
#if (POCO_OS == POCO_OS_QNX)
|
|
_NTO_THREAD_NAME_MAX;
|
|
#else
|
|
POCO_MAX_THREAD_NAME_LEN;
|
|
#endif
|
|
char name[nameSize + 1]{'\0'};
|
|
#if (POCO_OS == POCO_OS_FREE_BSD)
|
|
pthread_getname_np(pthread_self(), name, nameSize + 1);
|
|
#elif (POCO_OS == POCO_OS_MAC_OS_X)
|
|
#ifdef __MAC_OS_X_VERSION_MIN_REQUIRED
|
|
#if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1060
|
|
pthread_getname_np(pthread_self(), name, nameSize + 1);
|
|
#endif
|
|
#endif // __MAC_OS_X_VERSION_MIN_REQUIRED
|
|
#elif (POCO_OS == POCO_OS_QNX)
|
|
pthread_getname_np(pthread_self(), name, nameSize);
|
|
#else
|
|
prctl(PR_GET_NAME, name);
|
|
#endif
|
|
return name;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
namespace Poco {
|
|
|
|
|
|
ThreadImpl::CurrentThreadHolder ThreadImpl::_currentThreadHolder;
|
|
|
|
|
|
ThreadImpl::ThreadImpl():
|
|
_pData(new ThreadData)
|
|
{
|
|
}
|
|
|
|
|
|
ThreadImpl::~ThreadImpl()
|
|
{
|
|
if (_pData->started && !_pData->joined)
|
|
{
|
|
pthread_detach(_pData->thread);
|
|
}
|
|
}
|
|
|
|
|
|
void ThreadImpl::setNameImpl(const std::string& threadName)
|
|
{
|
|
std::string realName = threadName;
|
|
#if (POCO_OS == POCO_OS_MAC_OS_X)
|
|
if (threadName.size() > POCO_MAX_THREAD_NAME_LEN)
|
|
{
|
|
int half = (POCO_MAX_THREAD_NAME_LEN - 1) / 2;
|
|
#else
|
|
if (threadName.size() > std::min(POCO_MAX_THREAD_NAME_LEN, 15))
|
|
{
|
|
int half = (std::min(POCO_MAX_THREAD_NAME_LEN, 15) - 1) / 2;
|
|
#endif
|
|
std::string truncName(threadName, 0, half);
|
|
truncName.append("~");
|
|
truncName.append(threadName, threadName.size() - half);
|
|
realName = truncName;
|
|
}
|
|
|
|
ScopedLock<FastMutex> lock(_pData->mutex);
|
|
if (realName != _pData->name)
|
|
{
|
|
_pData->name = realName;
|
|
}
|
|
}
|
|
|
|
|
|
std::string ThreadImpl::getNameImpl() const
|
|
{
|
|
ScopedLock<FastMutex> lock(_pData->mutex);
|
|
return _pData->name;
|
|
}
|
|
|
|
|
|
#ifndef POCO_NO_THREADNAME
|
|
std::string ThreadImpl::getOSThreadNameImpl()
|
|
{
|
|
return isRunningImpl() ? getThreadName() : "";
|
|
}
|
|
#endif
|
|
|
|
|
|
void ThreadImpl::setPriorityImpl(int prio)
|
|
{
|
|
if (prio != _pData->prio)
|
|
{
|
|
_pData->prio = prio;
|
|
_pData->policy = SCHED_OTHER;
|
|
if (isRunningImpl())
|
|
{
|
|
struct sched_param par;
|
|
par.sched_priority = mapPrio(_pData->prio, SCHED_OTHER);
|
|
int errorCode;
|
|
if ((errorCode = pthread_setschedparam(_pData->thread, SCHED_OTHER, &par)))
|
|
throw SystemException(Poco::format("cannot set thread priority (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void ThreadImpl::setOSPriorityImpl(int prio, int policy)
|
|
{
|
|
if (prio != _pData->osPrio || policy != _pData->policy)
|
|
{
|
|
if (_pData->pRunnableTarget)
|
|
{
|
|
struct sched_param par;
|
|
par.sched_priority = prio;
|
|
int errorCode;
|
|
if ((errorCode = pthread_setschedparam(_pData->thread, policy, &par)))
|
|
throw SystemException(Poco::format("cannot set thread priority (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
_pData->prio = reverseMapPrio(prio, policy);
|
|
_pData->osPrio = prio;
|
|
_pData->policy = policy;
|
|
}
|
|
}
|
|
|
|
|
|
int ThreadImpl::getMinOSPriorityImpl(int policy)
|
|
{
|
|
#if defined(POCO_THREAD_PRIORITY_MIN)
|
|
return POCO_THREAD_PRIORITY_MIN;
|
|
#elif defined(__digital__)
|
|
return PRI_OTHER_MIN;
|
|
#else
|
|
return sched_get_priority_min(policy);
|
|
#endif
|
|
}
|
|
|
|
|
|
int ThreadImpl::getMaxOSPriorityImpl(int policy)
|
|
{
|
|
#if defined(POCO_THREAD_PRIORITY_MAX)
|
|
return POCO_THREAD_PRIORITY_MAX;
|
|
#elif defined(__digital__)
|
|
return PRI_OTHER_MAX;
|
|
#else
|
|
return sched_get_priority_max(policy);
|
|
#endif
|
|
}
|
|
|
|
|
|
void ThreadImpl::setStackSizeImpl(int size)
|
|
{
|
|
#ifndef PTHREAD_STACK_MIN
|
|
_pData->stackSize = 0;
|
|
#else
|
|
if (size != 0)
|
|
{
|
|
#if defined(POCO_OS_FAMILY_BSD)
|
|
// we must round up to a multiple of the memory page size
|
|
const int STACK_PAGE_SIZE = 4096;
|
|
size = ((size + STACK_PAGE_SIZE - 1)/STACK_PAGE_SIZE)*STACK_PAGE_SIZE;
|
|
#endif
|
|
if (size < PTHREAD_STACK_MIN)
|
|
size = PTHREAD_STACK_MIN;
|
|
}
|
|
_pData->stackSize = size;
|
|
#endif
|
|
}
|
|
|
|
|
|
void ThreadImpl::setSignalMaskImpl(uint32_t sigMask)
|
|
{
|
|
sigset_t sset;
|
|
sigemptyset(&sset);
|
|
|
|
for (int sig = 0; sig < sizeof(uint32_t) * 8; ++sig)
|
|
{
|
|
if ((sigMask & (1 << sig)) != 0)
|
|
sigaddset(&sset, sig);
|
|
}
|
|
|
|
pthread_sigmask(SIG_BLOCK, &sset, 0);
|
|
}
|
|
|
|
|
|
void ThreadImpl::startImpl(SharedPtr<Runnable> pTarget)
|
|
{
|
|
{
|
|
FastMutex::ScopedLock l(_pData->mutex);
|
|
if (_pData->pRunnableTarget)
|
|
throw SystemException("thread already running");
|
|
}
|
|
|
|
pthread_attr_t attributes;
|
|
pthread_attr_init(&attributes);
|
|
|
|
if (_pData->stackSize != 0)
|
|
{
|
|
int errorCode;
|
|
if ((errorCode = pthread_attr_setstacksize(&attributes, _pData->stackSize)))
|
|
{
|
|
pthread_attr_destroy(&attributes);
|
|
throw SystemException(Poco::format("cannot set thread stack size: (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
}
|
|
|
|
{
|
|
FastMutex::ScopedLock l(_pData->mutex);
|
|
_pData->pRunnableTarget = pTarget;
|
|
int errorCode;
|
|
if ((errorCode = pthread_create(&_pData->thread, &attributes, runnableEntry, this)))
|
|
{
|
|
_pData->pRunnableTarget = 0;
|
|
pthread_attr_destroy(&attributes);
|
|
throw SystemException(Poco::format("cannot start thread (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
}
|
|
_pData->started = true;
|
|
_pData->joined = false;
|
|
pthread_attr_destroy(&attributes);
|
|
|
|
if (_pData->policy == SCHED_OTHER)
|
|
{
|
|
if (_pData->prio != PRIO_NORMAL_IMPL)
|
|
{
|
|
struct sched_param par;
|
|
par.sched_priority = mapPrio(_pData->prio, SCHED_OTHER);
|
|
int errorCode;
|
|
if ((errorCode = pthread_setschedparam(_pData->thread, SCHED_OTHER, &par)))
|
|
throw SystemException(Poco::format("cannot set thread priority (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
struct sched_param par;
|
|
par.sched_priority = _pData->osPrio;
|
|
int errorCode;
|
|
if ((errorCode = pthread_setschedparam(_pData->thread, _pData->policy, &par)))
|
|
throw SystemException(Poco::format("cannot set thread priority (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
}
|
|
|
|
|
|
void ThreadImpl::joinImpl()
|
|
{
|
|
if (!_pData->started) return;
|
|
_pData->done.wait();
|
|
if (!_pData->joined)
|
|
{
|
|
int errorCode;
|
|
if ((errorCode = pthread_join(_pData->thread, nullptr)))
|
|
{
|
|
throw SystemException(Poco::format("cannot join thread (%s)",
|
|
Error::getMessage(errorCode)));
|
|
}
|
|
_pData->joined = true;
|
|
}
|
|
}
|
|
|
|
|
|
bool ThreadImpl::joinImpl(long milliseconds)
|
|
{
|
|
if (_pData->started && _pData->done.tryWait(milliseconds) && !_pData->joined)
|
|
{
|
|
int errorCode;
|
|
if ((errorCode = pthread_join(_pData->thread, nullptr)))
|
|
throw SystemException(Poco::format("cannot join thread (%s)",
|
|
Error::getMessage(errorCode)));
|
|
_pData->joined = true;
|
|
return true;
|
|
}
|
|
else if (_pData->started) return false;
|
|
else return true;
|
|
}
|
|
|
|
|
|
ThreadImpl* ThreadImpl::currentImpl()
|
|
{
|
|
return _currentThreadHolder.get();
|
|
}
|
|
|
|
|
|
ThreadImpl::TIDImpl ThreadImpl::currentTidImpl()
|
|
{
|
|
return pthread_self();
|
|
}
|
|
|
|
|
|
long ThreadImpl::currentOsTidImpl()
|
|
{
|
|
long id = 0;
|
|
#if (POCO_OS == POCO_OS_LINUX) && !defined(POCO_EMSCRIPTEN)
|
|
id = ::syscall(SYS_gettid);
|
|
#elif POCO_OS == POCO_OS_MAC_OS_X
|
|
id = ::pthread_mach_thread_np(::pthread_self());
|
|
#elif POCO_OS == POCO_OS_FREE_BSD
|
|
if (0 != thr_self(&id)) id = 0;
|
|
#endif
|
|
return id;
|
|
}
|
|
|
|
|
|
void* ThreadImpl::runnableEntry(void* pThread)
|
|
{
|
|
_currentThreadHolder.set(reinterpret_cast<ThreadImpl*>(pThread));
|
|
|
|
#if defined(POCO_OS_FAMILY_UNIX)
|
|
sigset_t sset;
|
|
sigemptyset(&sset);
|
|
sigaddset(&sset, SIGQUIT);
|
|
sigaddset(&sset, SIGTERM);
|
|
sigaddset(&sset, SIGPIPE);
|
|
pthread_sigmask(SIG_BLOCK, &sset, 0);
|
|
#endif
|
|
|
|
ThreadImpl* pThreadImpl = reinterpret_cast<ThreadImpl*>(pThread);
|
|
#ifndef POCO_NO_THREADNAME
|
|
setThreadName(reinterpret_cast<Thread*>(pThread)->getName());
|
|
#endif
|
|
AutoPtr<ThreadData> pData = pThreadImpl->_pData;
|
|
|
|
#ifndef POCO_NO_THREADNAME
|
|
{
|
|
FastMutex::ScopedLock lock(pData->mutex);
|
|
setThreadName(pData->name);
|
|
}
|
|
#endif
|
|
|
|
try
|
|
{
|
|
pData->pRunnableTarget->run();
|
|
}
|
|
catch (Exception& exc)
|
|
{
|
|
ErrorHandler::handle(exc);
|
|
}
|
|
catch (std::exception& exc)
|
|
{
|
|
ErrorHandler::handle(exc);
|
|
}
|
|
catch (...)
|
|
{
|
|
ErrorHandler::handle();
|
|
}
|
|
|
|
FastMutex::ScopedLock l(pData->mutex);
|
|
pData->pRunnableTarget = 0;
|
|
pData->done.set();
|
|
return 0;
|
|
}
|
|
|
|
|
|
int ThreadImpl::mapPrio(int prio, int policy)
|
|
{
|
|
int pmin = getMinOSPriorityImpl(policy);
|
|
int pmax = getMaxOSPriorityImpl(policy);
|
|
|
|
switch (prio)
|
|
{
|
|
case PRIO_LOWEST_IMPL:
|
|
return pmin;
|
|
case PRIO_LOW_IMPL:
|
|
return pmin + (pmax - pmin)/4;
|
|
case PRIO_NORMAL_IMPL:
|
|
return pmin + (pmax - pmin)/2;
|
|
case PRIO_HIGH_IMPL:
|
|
return pmin + 3*(pmax - pmin)/4;
|
|
case PRIO_HIGHEST_IMPL:
|
|
return pmax;
|
|
default:
|
|
poco_bugcheck_msg("invalid thread priority");
|
|
}
|
|
return -1; // just to satisfy compiler - we'll never get here anyway
|
|
}
|
|
|
|
|
|
int ThreadImpl::reverseMapPrio(int prio, int policy)
|
|
{
|
|
if (policy == SCHED_OTHER)
|
|
{
|
|
int pmin = getMinOSPriorityImpl(policy);
|
|
int pmax = getMaxOSPriorityImpl(policy);
|
|
int normal = pmin + (pmax - pmin)/2;
|
|
if (prio == pmax)
|
|
return PRIO_HIGHEST_IMPL;
|
|
if (prio > normal)
|
|
return PRIO_HIGH_IMPL;
|
|
else if (prio == normal)
|
|
return PRIO_NORMAL_IMPL;
|
|
else if (prio > pmin)
|
|
return PRIO_LOW_IMPL;
|
|
else
|
|
return PRIO_LOWEST_IMPL;
|
|
}
|
|
else return PRIO_HIGHEST_IMPL;
|
|
}
|
|
|
|
|
|
bool ThreadImpl::setAffinityImpl(int coreID)
|
|
{
|
|
#if POCO_OS == POCO_OS_LINUX
|
|
int numCores = sysconf(_SC_NPROCESSORS_ONLN);
|
|
if (coreID < 0 || coreID >= numCores)
|
|
return false;
|
|
|
|
cpu_set_t cpuset;
|
|
CPU_ZERO(&cpuset);
|
|
CPU_SET(coreID, &cpuset);
|
|
|
|
return 0 == pthread_setaffinity_np(_pData->thread, sizeof(cpu_set_t), &cpuset);
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
|
|
int ThreadImpl::getAffinityImpl() const
|
|
{
|
|
#if POCO_OS == POCO_OS_LINUX
|
|
cpu_set_t cpuset;
|
|
CPU_ZERO(&cpuset);
|
|
if (0 == pthread_getaffinity_np(_pData->thread, sizeof(cpu_set_t), &cpuset))
|
|
{
|
|
for (int i = 0; i < CPU_SETSIZE; ++i)
|
|
{
|
|
if (CPU_ISSET(i, &cpuset)) return i;
|
|
}
|
|
}
|
|
#endif
|
|
return -1;
|
|
}
|
|
|
|
|
|
} // namespace Poco
|