9#include <private/qcoreapplication_p.h>
10#include <private/qeventdispatcher_win_p.h>
15#include <qt_windows.h>
29 _In_ PCWSTR lpThreadDescription
33#ifndef THREAD_POWER_THROTTLING_EXECUTION_SPEED
34#define THREAD_POWER_THROTTLING_EXECUTION_SPEED 0x1
35#define THREAD_POWER_THROTTLING_CURRENT_VERSION 1
41} THREAD_POWER_THROTTLING_STATE;
51Q_CONSTINIT
static thread_local QThreadData *currentThreadData =
nullptr;
53static void destroy_current_thread_data(
void *p)
55 QThreadData *data =
static_cast<QThreadData *>(p);
56 QThread *thread = data->thread.loadAcquire();
58 if (data->isAdopted) {
62 QThreadPrivate *thread_p =
static_cast<QThreadPrivate *>(QObjectPrivate::get(thread));
74 currentThreadData =
nullptr;
77static QThreadData *get_thread_data()
79 return currentThreadData;
82static void set_thread_data(QThreadData *data)
noexcept
86 Cleanup() { QThreadStoragePrivate::init(); }
87 ~Cleanup() { destroy_current_thread_data(currentThreadData); }
89 static thread_local Cleanup currentThreadCleanup;
91 currentThreadData = data;
95
96
97void QThreadData::clearCurrentThreadData()
99 set_thread_data(
nullptr);
102QThreadData *QThreadData::currentThreadData()
noexcept
104 return get_thread_data();
107QThreadData *QThreadData::createCurrentThreadData()
109 Q_ASSERT(!currentThreadData());
110 std::unique_ptr data = std::make_unique<QThreadData>();
114 set_thread_data(data.get());
117 data->thread.storeRelease(
new QAdoptedThread(data.get()));
119 clearCurrentThreadData();
122 return data.release();
125void QAdoptedThread::init()
127 d_func()->handle = GetCurrentThread();
131
132
139 return new QEventDispatcherWin32;
144unsigned int __stdcall QT_ENSURE_STACK_ALIGNED_FOR_SSE QThreadPrivate::start(
void *arg)
noexcept
146 QThread *thr =
reinterpret_cast<QThread *>(arg);
147 QThreadData *data = QThreadData::get2(thr);
150 set_thread_data(data);
151 data->threadId.storeRelaxed(QThread::currentThreadId());
154 data->reuseBindingStatusForNewNativeThread();
156 QThread::setTerminationEnabled(
false);
159 QMutexLocker locker(&thr->d_func()->mutex);
160 data->quitNow = thr->d_func()->exited;
162 if (thr->d_func()->serviceLevel != QThread::QualityOfService::Auto)
163 thr->d_func()->setQualityOfServiceLevel(thr->d_func()->serviceLevel);
166 data->ensureEventDispatcher();
167 data->eventDispatcher.loadRelaxed()->startingUp();
170 QString threadName = std::exchange(thr->d_func()->objectName, {});
171 if (Q_LIKELY(threadName.isEmpty()))
172 threadName = QString::fromUtf8(thr->metaObject()->className());
173#ifndef QT_WIN_SERVER_2016_COMPAT
174 SetThreadDescription(GetCurrentThread(),
reinterpret_cast<
const wchar_t *>(threadName.utf16()));
176 HMODULE kernelbase = GetModuleHandleW(L"kernelbase.dll");
177 if (kernelbase != NULL) {
178 typedef HRESULT (WINAPI *DESCFUNC)(HANDLE, PCWSTR);
180 DESCFUNC setThreadDescription =
181 (DESCFUNC)GetProcAddress(kernelbase,
"SetThreadDescription");
182 if (setThreadDescription != NULL) {
183 setThreadDescription(GetCurrentThread(),
184 reinterpret_cast<
const wchar_t *>(threadName.utf16()));
189 emit thr->started(QThread::QPrivateSignal());
190 QThread::setTerminationEnabled(
true);
193 thr->d_func()->finish();
197void QThreadPrivate::setQualityOfServiceLevel(QThread::QualityOfService qosLevel)
200 serviceLevel = qosLevel;
202#if (_WIN32_WINNT >= _WIN32_WINNT_WIN10_RS3)
203 qCDebug(lcQThread) <<
"Setting thread QoS class to" << qosLevel <<
"for thread" << q;
205 THREAD_POWER_THROTTLING_STATE state;
206 memset(&state, 0,
sizeof(state));
207 state.Version = THREAD_POWER_THROTTLING_CURRENT_VERSION;
210 case QThread::QualityOfService::Auto:
211 state.ControlMask = 0;
214 case QThread::QualityOfService::Eco:
215 state.ControlMask = THREAD_POWER_THROTTLING_EXECUTION_SPEED;
216 state.StateMask = THREAD_POWER_THROTTLING_EXECUTION_SPEED;
218 case QThread::QualityOfService::High:
219 state.ControlMask = THREAD_POWER_THROTTLING_EXECUTION_SPEED;
223 if (!SetThreadInformation(::GetCurrentThread(), THREAD_INFORMATION_CLASS::ThreadPowerThrottling,
224 &state,
sizeof(state))) {
225 qErrnoWarning(
"Failed to set thread power throttling state");
231
232
233
234
235
236
237
238
239
240
241void QThreadPrivate::finish(
bool lockAnyway)
noexcept
243 QThreadPrivate *d =
this;
244 QThread *thr = q_func();
246 QMutexLocker locker(lockAnyway ? &d->mutex :
nullptr);
247 d->threadState = QThreadPrivate::Finishing;
248 d->priority = QThread::InheritPriority;
251 emit thr->finished(QThread::QPrivateSignal());
252 QCoreApplicationPrivate::sendPostedEvents(
nullptr, QEvent::DeferredDelete, d->data);
253 QThreadStoragePrivate::finish(&d->data->tls);
257 QAbstractEventDispatcher *eventDispatcher = d->data->eventDispatcher.loadRelaxed();
258 if (eventDispatcher) {
259 d->data->eventDispatcher = 0;
262 eventDispatcher->closingDown();
263 delete eventDispatcher;
268 d->threadState = QThreadPrivate::Finished;
269 d->interruptionRequested.store(
false, std::memory_order_relaxed);
272 CloseHandle(d->handle);
278
279
281Qt::HANDLE QThread::currentThreadIdImpl()
noexcept
283 return reinterpret_cast<Qt::HANDLE>(quintptr(GetCurrentThreadId()));
286int QThread::idealThreadCount()
noexcept
289 GetSystemInfo(&sysinfo);
290 return sysinfo.dwNumberOfProcessors;
293void QThread::yieldCurrentThread()
300void QThread::sleep(std::chrono::nanoseconds nsecs)
302 using namespace std::chrono;
303 ::Sleep(DWORD(duration_cast<milliseconds>(nsecs).count()));
306void QThread::sleep(
unsigned long secs)
308 ::Sleep(secs * 1000);
311void QThread::msleep(
unsigned long msecs)
316void QThread::usleep(
unsigned long usecs)
318 ::Sleep((usecs / 1000) + 1);
323void QThread::start(Priority priority)
326 QMutexLocker locker(&d->mutex);
328 if (d->threadState == QThreadPrivate::Finishing) {
334 if (d->threadState == QThreadPrivate::Running)
338 d->objectName = d->extraData ? d->extraData->objectName.valueBypassingBindings()
340 d->threadState = QThreadPrivate::Running;
343 d->interruptionRequested.store(
false, std::memory_order_relaxed);
346
347
348
349
350
351
352
353
354
355#if defined(Q_CC_MSVC) && !defined(_DLL)
357 d->handle = (Qt::HANDLE) _beginthreadex(NULL, d->stackSize, QThreadPrivate::start,
358 this, CREATE_SUSPENDED,
nullptr);
361 d->handle = CreateThread(
nullptr, d->stackSize,
362 reinterpret_cast<LPTHREAD_START_ROUTINE>(QThreadPrivate::start),
363 this, CREATE_SUSPENDED,
nullptr);
367 qErrnoWarning(
"QThread::start: Failed to create thread");
368 d->threadState = QThreadPrivate::NotStarted;
373 d->priority = priority;
376 prio = THREAD_PRIORITY_IDLE;
380 prio = THREAD_PRIORITY_LOWEST;
384 prio = THREAD_PRIORITY_BELOW_NORMAL;
388 prio = THREAD_PRIORITY_NORMAL;
392 prio = THREAD_PRIORITY_ABOVE_NORMAL;
395 case HighestPriority:
396 prio = THREAD_PRIORITY_HIGHEST;
399 case TimeCriticalPriority:
400 prio = THREAD_PRIORITY_TIME_CRITICAL;
403 case InheritPriority:
405 prio = GetThreadPriority(GetCurrentThread());
409 if (!SetThreadPriority(d->handle, prio)) {
410 qErrnoWarning(
"QThread::start: Failed to set thread priority");
413 if (ResumeThread(d->handle) == (DWORD) -1) {
414 qErrnoWarning(
"QThread::start: Failed to resume new thread");
418void QThread::terminate()
421 QMutexLocker locker(&d->mutex);
422 if (d->threadState != QThreadPrivate::Running)
424 if (!d->terminationEnabled) {
425 d->terminatePending =
true;
429 TerminateThread(d->handle, 0);
433bool QThreadPrivate::wait(QMutexLocker<QMutex> &locker, QDeadlineTimer deadline)
435 Q_ASSERT(threadState != QThreadPrivate::Finished);
436 Q_ASSERT(locker.isLocked());
437 QThreadPrivate *d =
this;
440 locker.mutex()->unlock();
443 switch (WaitForSingleObject(d->handle, deadline.remainingTime())) {
448 qErrnoWarning(
"QThread::wait: Thread wait failure");
456 locker.mutex()->lock();
459 if (ret && d->threadState < QThreadPrivate::Finished) {
465 if (d->threadState == QThreadPrivate::Finished && !d->waiters) {
466 CloseHandle(d->handle);
473void QThread::setTerminationEnabled(
bool enabled)
475 QThread *thr = currentThread();
476 Q_ASSERT_X(thr != 0,
"QThread::setTerminationEnabled()",
477 "Current thread was not started with QThread.");
478 QThreadPrivate *d = thr->d_func();
479 QMutexLocker locker(&d->mutex);
480 d->terminationEnabled = enabled;
481 if (enabled && d->terminatePending) {
489void QThreadPrivate::setPriority(QThread::Priority threadPriority)
494 priority = threadPriority;
495 switch (threadPriority) {
496 case QThread::IdlePriority:
497 prio = THREAD_PRIORITY_IDLE;
500 case QThread::LowestPriority:
501 prio = THREAD_PRIORITY_LOWEST;
504 case QThread::LowPriority:
505 prio = THREAD_PRIORITY_BELOW_NORMAL;
508 case QThread::NormalPriority:
509 prio = THREAD_PRIORITY_NORMAL;
512 case QThread::HighPriority:
513 prio = THREAD_PRIORITY_ABOVE_NORMAL;
516 case QThread::HighestPriority:
517 prio = THREAD_PRIORITY_HIGHEST;
520 case QThread::TimeCriticalPriority:
521 prio = THREAD_PRIORITY_TIME_CRITICAL;
528 if (!SetThreadPriority(handle, prio)) {
529 qErrnoWarning(
"QThread::setPriority: Failed to set thread priority");
static QAbstractEventDispatcher * createEventDispatcher(QThreadData *data)
#define Q_STATIC_LOGGING_CATEGORY(name,...)