2012-04-29 20:52:25 +02:00
|
|
|
//
|
|
|
|
// Thread_POSIX.h
|
|
|
|
//
|
|
|
|
// Library: Foundation
|
|
|
|
// Package: Threading
|
|
|
|
// Module: Thread
|
|
|
|
//
|
|
|
|
// Definition of the ThreadImpl class for POSIX Threads.
|
|
|
|
//
|
|
|
|
// Copyright (c) 2004-2007, Applied Informatics Software Engineering GmbH.
|
|
|
|
// and Contributors.
|
|
|
|
//
|
2014-05-04 21:02:42 +02:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
2012-04-29 20:52:25 +02:00
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef Foundation_Thread_POSIX_INCLUDED
|
|
|
|
#define Foundation_Thread_POSIX_INCLUDED
|
|
|
|
|
|
|
|
|
|
|
|
#include "Poco/Foundation.h"
|
2024-07-29 08:16:50 +02:00
|
|
|
#include "Poco/Mutex.h"
|
2012-04-29 20:52:25 +02:00
|
|
|
#include "Poco/Runnable.h"
|
|
|
|
#include "Poco/SignalHandler.h"
|
|
|
|
#include "Poco/Event.h"
|
|
|
|
#include "Poco/RefCountedObject.h"
|
|
|
|
#include "Poco/AutoPtr.h"
|
2014-11-16 20:43:19 +01:00
|
|
|
#include "Poco/SharedPtr.h"
|
2012-04-29 20:52:25 +02:00
|
|
|
#include <pthread.h>
|
|
|
|
// must be limits.h (not <climits>) for PTHREAD_STACK_MIN on Solaris
|
|
|
|
#include <limits.h>
|
|
|
|
#if !defined(POCO_NO_SYS_SELECT_H)
|
|
|
|
#include <sys/select.h>
|
|
|
|
#endif
|
|
|
|
#if defined(POCO_VXWORKS)
|
|
|
|
#include <cstring>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace Poco {
|
|
|
|
|
|
|
|
class Foundation_API ThreadImpl
|
|
|
|
{
|
2017-09-29 01:12:31 +02:00
|
|
|
public:
|
2024-07-29 08:16:50 +02:00
|
|
|
using TIDImpl = pthread_t;
|
|
|
|
using Callable = void (*)(void *);
|
2012-04-29 20:52:25 +02:00
|
|
|
|
|
|
|
enum Priority
|
|
|
|
{
|
|
|
|
PRIO_LOWEST_IMPL,
|
|
|
|
PRIO_LOW_IMPL,
|
|
|
|
PRIO_NORMAL_IMPL,
|
|
|
|
PRIO_HIGH_IMPL,
|
|
|
|
PRIO_HIGHEST_IMPL
|
|
|
|
};
|
2017-09-29 01:12:31 +02:00
|
|
|
|
2012-05-19 05:04:51 +02:00
|
|
|
enum Policy
|
|
|
|
{
|
|
|
|
POLICY_DEFAULT_IMPL = SCHED_OTHER
|
|
|
|
};
|
2022-07-07 11:18:20 +02:00
|
|
|
|
2012-11-13 03:45:16 +01:00
|
|
|
ThreadImpl();
|
2012-04-29 20:52:25 +02:00
|
|
|
~ThreadImpl();
|
2012-11-13 03:45:16 +01:00
|
|
|
|
2012-04-29 20:52:25 +02:00
|
|
|
TIDImpl tidImpl() const;
|
2022-07-18 12:21:33 +02:00
|
|
|
void setNameImpl(const std::string& threadName);
|
|
|
|
std::string getNameImpl() const;
|
2024-10-22 12:36:03 +02:00
|
|
|
#ifndef POCO_NO_THREADNAME
|
2022-07-18 12:21:33 +02:00
|
|
|
std::string getOSThreadNameImpl();
|
|
|
|
/// Returns the thread's name, expressed as an operating system
|
|
|
|
/// specific name value. Return empty string if thread is not running.
|
|
|
|
/// For test used only.
|
2024-10-22 12:36:03 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
void setPriorityImpl(int prio);
|
|
|
|
int getPriorityImpl() const;
|
2012-05-19 05:04:51 +02:00
|
|
|
void setOSPriorityImpl(int prio, int policy = SCHED_OTHER);
|
2012-04-29 20:52:25 +02:00
|
|
|
int getOSPriorityImpl() const;
|
2012-05-19 05:04:51 +02:00
|
|
|
static int getMinOSPriorityImpl(int policy);
|
|
|
|
static int getMaxOSPriorityImpl(int policy);
|
2012-04-29 20:52:25 +02:00
|
|
|
void setStackSizeImpl(int size);
|
|
|
|
int getStackSizeImpl() const;
|
2023-11-24 20:22:01 +01:00
|
|
|
void setSignalMaskImpl(uint32_t sigMask);
|
2014-11-16 20:43:19 +01:00
|
|
|
void startImpl(SharedPtr<Runnable> pTarget);
|
2012-04-29 20:52:25 +02:00
|
|
|
void joinImpl();
|
|
|
|
bool joinImpl(long milliseconds);
|
|
|
|
bool isRunningImpl() const;
|
|
|
|
static void yieldImpl();
|
|
|
|
static ThreadImpl* currentImpl();
|
|
|
|
static TIDImpl currentTidImpl();
|
2022-05-19 00:23:16 +02:00
|
|
|
static long currentOsTidImpl();
|
2022-07-26 13:54:56 +02:00
|
|
|
bool setAffinityImpl(int coreID);
|
|
|
|
int getAffinityImpl() const;
|
2012-04-29 20:52:25 +02:00
|
|
|
|
|
|
|
protected:
|
|
|
|
static void* runnableEntry(void* pThread);
|
2012-05-19 05:04:51 +02:00
|
|
|
static int mapPrio(int prio, int policy = SCHED_OTHER);
|
|
|
|
static int reverseMapPrio(int osPrio, int policy = SCHED_OTHER);
|
2012-04-29 20:52:25 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
class CurrentThreadHolder
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CurrentThreadHolder()
|
|
|
|
{
|
2024-07-29 08:16:50 +02:00
|
|
|
if (pthread_key_create(&_key, nullptr))
|
2012-04-29 20:52:25 +02:00
|
|
|
throw SystemException("cannot allocate thread context key");
|
|
|
|
}
|
|
|
|
~CurrentThreadHolder()
|
|
|
|
{
|
|
|
|
pthread_key_delete(_key);
|
|
|
|
}
|
|
|
|
ThreadImpl* get() const
|
|
|
|
{
|
|
|
|
return reinterpret_cast<ThreadImpl*>(pthread_getspecific(_key));
|
|
|
|
}
|
|
|
|
void set(ThreadImpl* pThread)
|
|
|
|
{
|
|
|
|
pthread_setspecific(_key, pThread);
|
|
|
|
}
|
2017-09-29 01:12:31 +02:00
|
|
|
|
2012-04-29 20:52:25 +02:00
|
|
|
private:
|
|
|
|
pthread_key_t _key;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ThreadData: public RefCountedObject
|
|
|
|
{
|
|
|
|
ThreadData():
|
|
|
|
thread(0),
|
|
|
|
prio(PRIO_NORMAL_IMPL),
|
2016-09-26 18:15:43 +02:00
|
|
|
osPrio(),
|
2012-05-19 05:04:51 +02:00
|
|
|
policy(SCHED_OTHER),
|
2024-07-29 08:16:50 +02:00
|
|
|
done(Event::EVENT_MANUALRESET),
|
2013-03-18 03:13:04 +01:00
|
|
|
stackSize(POCO_THREAD_STACK_SIZE),
|
2013-03-29 20:02:53 +01:00
|
|
|
started(false),
|
2013-03-18 03:13:04 +01:00
|
|
|
joined(false)
|
2012-04-29 20:52:25 +02:00
|
|
|
{
|
|
|
|
#if defined(POCO_VXWORKS)
|
|
|
|
// This workaround is for VxWorks 5.x where
|
|
|
|
// pthread_init() won't properly initialize the thread.
|
|
|
|
std::memset(&thread, 0, sizeof(thread));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-11-16 20:43:19 +01:00
|
|
|
SharedPtr<Runnable> pRunnableTarget;
|
2012-04-29 20:52:25 +02:00
|
|
|
pthread_t thread;
|
|
|
|
int prio;
|
|
|
|
int osPrio;
|
2012-05-19 05:04:51 +02:00
|
|
|
int policy;
|
2012-04-29 20:52:25 +02:00
|
|
|
Event done;
|
|
|
|
std::size_t stackSize;
|
2024-01-31 22:07:07 +01:00
|
|
|
std::atomic<bool> started;
|
|
|
|
std::atomic<bool> joined;
|
2022-07-18 12:21:33 +02:00
|
|
|
std::string name;
|
2022-07-26 13:54:56 +02:00
|
|
|
int affinity;
|
2022-06-02 06:47:26 +02:00
|
|
|
mutable FastMutex mutex;
|
2012-04-29 20:52:25 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
AutoPtr<ThreadData> _pData;
|
|
|
|
static CurrentThreadHolder _currentThreadHolder;
|
2022-07-07 11:18:20 +02:00
|
|
|
|
2012-04-29 20:52:25 +02:00
|
|
|
#if defined(POCO_OS_FAMILY_UNIX) && !defined(POCO_VXWORKS)
|
|
|
|
SignalHandler::JumpBufferVec _jumpBufferVec;
|
|
|
|
friend class SignalHandler;
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// inlines
|
|
|
|
//
|
|
|
|
inline int ThreadImpl::getPriorityImpl() const
|
|
|
|
{
|
|
|
|
return _pData->prio;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline int ThreadImpl::getOSPriorityImpl() const
|
|
|
|
{
|
|
|
|
return _pData->osPrio;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool ThreadImpl::isRunningImpl() const
|
|
|
|
{
|
2022-06-02 06:47:26 +02:00
|
|
|
FastMutex::ScopedLock l(_pData->mutex);
|
2014-11-16 20:51:11 +01:00
|
|
|
return !_pData->pRunnableTarget.isNull();
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline void ThreadImpl::yieldImpl()
|
|
|
|
{
|
|
|
|
sched_yield();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline int ThreadImpl::getStackSizeImpl() const
|
|
|
|
{
|
2012-09-06 05:41:21 +02:00
|
|
|
return static_cast<int>(_pData->stackSize);
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline ThreadImpl::TIDImpl ThreadImpl::tidImpl() const
|
|
|
|
{
|
|
|
|
return _pData->thread;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace Poco
|
|
|
|
|
|
|
|
|
|
|
|
#endif // Foundation_Thread_POSIX_INCLUDED
|