mirror of
https://github.com/crystalidea/qt6windows7.git
synced 2024-12-01 00:06:55 +08:00
328 lines
8.2 KiB
C++
328 lines
8.2 KiB
C++
|
// Copyright (C) 2022 The Qt Company Ltd.
|
||
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
||
|
|
||
|
#include <QtCore/QCoreApplication>
|
||
|
#include <QtCore/QEvent>
|
||
|
#include <QtCore/QMutex>
|
||
|
#include <QtCore/QObject>
|
||
|
#include <QtCore/QThread>
|
||
|
#include <QtCore/QTimer>
|
||
|
#include <QtCore/private/qstdweb_p.h>
|
||
|
|
||
|
#include <qtwasmtestlib.h>
|
||
|
|
||
|
#include "emscripten.h"
|
||
|
|
||
|
const int timerTimeout = 10;
|
||
|
|
||
|
class WasmEventDispatcherTest: public QObject
|
||
|
{
|
||
|
Q_OBJECT
|
||
|
private slots:
|
||
|
void postEventMainThread();
|
||
|
void timerMainThread();
|
||
|
void timerMainThreadMultiple();
|
||
|
|
||
|
#if QT_CONFIG(thread)
|
||
|
void postEventSecondaryThread();
|
||
|
void postEventSecondaryThreads();
|
||
|
void postEventToSecondaryThread();
|
||
|
void timerSecondaryThread();
|
||
|
#endif
|
||
|
|
||
|
void postEventAsyncify();
|
||
|
void timerAsyncify();
|
||
|
void postEventAsyncifyLoop();
|
||
|
|
||
|
private:
|
||
|
// Disabled test function: Asyncify wait on pthread_join is not supported,
|
||
|
// see https://github.com/emscripten-core/emscripten/issues/9910
|
||
|
#if QT_CONFIG(thread)
|
||
|
void threadAsyncifyWait();
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
class EventTarget : public QObject
|
||
|
{
|
||
|
Q_OBJECT
|
||
|
|
||
|
public:
|
||
|
static EventTarget *create(std::function<void()> callback)
|
||
|
{
|
||
|
return new EventTarget(callback);
|
||
|
}
|
||
|
|
||
|
static QEvent *createEvent()
|
||
|
{
|
||
|
return new QEvent(QEvent::User);
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
EventTarget(std::function<void()> callback)
|
||
|
: m_callback(callback) { }
|
||
|
|
||
|
bool event(QEvent *evt)
|
||
|
{
|
||
|
if (evt->type() == QEvent::User) {
|
||
|
m_callback();
|
||
|
deleteLater();
|
||
|
return true;
|
||
|
}
|
||
|
return QObject::event(evt);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
std::function<void()> m_callback;
|
||
|
};
|
||
|
|
||
|
class CompleteTestFunctionRefGuard {
|
||
|
public:
|
||
|
CompleteTestFunctionRefGuard(CompleteTestFunctionRefGuard const&) = delete;
|
||
|
CompleteTestFunctionRefGuard& operator=(CompleteTestFunctionRefGuard const&) = delete;
|
||
|
|
||
|
static CompleteTestFunctionRefGuard *create() {
|
||
|
return new CompleteTestFunctionRefGuard();
|
||
|
}
|
||
|
|
||
|
void ref() {
|
||
|
QMutexLocker lock(&mutex);
|
||
|
++m_counter;
|
||
|
}
|
||
|
|
||
|
void deref() {
|
||
|
const bool finalDeref = [this] {
|
||
|
QMutexLocker lock(&mutex);
|
||
|
return --m_counter == 0;
|
||
|
}();
|
||
|
|
||
|
if (finalDeref)
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
}
|
||
|
private:
|
||
|
CompleteTestFunctionRefGuard() { };
|
||
|
|
||
|
QMutex mutex;
|
||
|
int m_counter = 0;
|
||
|
};
|
||
|
|
||
|
#if QT_CONFIG(thread)
|
||
|
|
||
|
class TestThread : public QThread
|
||
|
{
|
||
|
public:
|
||
|
static QThread *create(std::function<void()> started, std::function<void()> finished)
|
||
|
{
|
||
|
TestThread *thread = new TestThread();
|
||
|
connect(thread, &QThread::started, [started]() {
|
||
|
started();
|
||
|
});
|
||
|
connect(thread, &QThread::finished, [thread, finished]() {
|
||
|
finished();
|
||
|
thread->deleteLater();
|
||
|
});
|
||
|
thread->start();
|
||
|
return thread;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#endif
|
||
|
|
||
|
// Post event to the main thread and verify that it is processed.
|
||
|
void WasmEventDispatcherTest::postEventMainThread()
|
||
|
{
|
||
|
QCoreApplication::postEvent(EventTarget::create([](){
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
}), EventTarget::createEvent());
|
||
|
}
|
||
|
|
||
|
// Create a timer on the main thread and verify that it fires
|
||
|
void WasmEventDispatcherTest::timerMainThread()
|
||
|
{
|
||
|
QTimer::singleShot(timerTimeout, [](){
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
void WasmEventDispatcherTest::timerMainThreadMultiple()
|
||
|
{
|
||
|
CompleteTestFunctionRefGuard *completeGuard = CompleteTestFunctionRefGuard::create();
|
||
|
int timers = 10;
|
||
|
for (int i = 0; i < timers; ++i) {
|
||
|
completeGuard->ref();
|
||
|
QTimer::singleShot(timerTimeout * i, [completeGuard](){
|
||
|
completeGuard->deref();
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if QT_CONFIG(thread)
|
||
|
|
||
|
// Post event on a secondary thread and verify that it is processed.
|
||
|
void WasmEventDispatcherTest::postEventSecondaryThread()
|
||
|
{
|
||
|
auto started = [](){
|
||
|
QCoreApplication::postEvent(EventTarget::create([](){
|
||
|
QThread::currentThread()->quit();
|
||
|
}), EventTarget::createEvent());
|
||
|
};
|
||
|
|
||
|
auto finished = [](){
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
};
|
||
|
|
||
|
TestThread::create(started, finished);
|
||
|
}
|
||
|
|
||
|
// Post event _to_ a secondary thread and verify that it is processed.
|
||
|
void WasmEventDispatcherTest::postEventToSecondaryThread()
|
||
|
{
|
||
|
auto started = [](){};
|
||
|
auto finished = [](){
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
};
|
||
|
|
||
|
QThread *t = TestThread::create(started, finished);
|
||
|
EventTarget *target = EventTarget::create([](){
|
||
|
QThread::currentThread()->quit();
|
||
|
});
|
||
|
target->moveToThread(t);
|
||
|
QCoreApplication::postEvent(target, EventTarget::createEvent());
|
||
|
}
|
||
|
|
||
|
// Post events to many secondary threads and verify that they are processed.
|
||
|
void WasmEventDispatcherTest::postEventSecondaryThreads()
|
||
|
{
|
||
|
// This test completes afer all threads has finished
|
||
|
CompleteTestFunctionRefGuard *completeGuard = CompleteTestFunctionRefGuard::create();
|
||
|
completeGuard->ref(); // including this thread
|
||
|
|
||
|
auto started = [](){
|
||
|
QCoreApplication::postEvent(EventTarget::create([](){
|
||
|
QThread::currentThread()->quit();
|
||
|
}), EventTarget::createEvent());
|
||
|
};
|
||
|
|
||
|
auto finished = [completeGuard](){
|
||
|
completeGuard->deref();
|
||
|
};
|
||
|
|
||
|
// Start a nymber of threads in parallel, keeping in mind that the browser
|
||
|
// has some max number of concurrent web workers (maybe 20), and that starting
|
||
|
// a new web worker requires completing a GET network request for the worker's JS.
|
||
|
const int numThreads = 10;
|
||
|
for (int i = 0; i < numThreads; ++i) {
|
||
|
completeGuard->ref();
|
||
|
TestThread::create(started, finished);
|
||
|
}
|
||
|
|
||
|
completeGuard->deref();
|
||
|
}
|
||
|
|
||
|
// Create a timer a secondary thread and verify that it fires
|
||
|
void WasmEventDispatcherTest::timerSecondaryThread()
|
||
|
{
|
||
|
auto started = [](){
|
||
|
QTimer::singleShot(timerTimeout, [](){
|
||
|
QThread::currentThread()->quit();
|
||
|
});
|
||
|
};
|
||
|
|
||
|
auto finished = [](){
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
};
|
||
|
|
||
|
TestThread::create(started, finished);
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
// Post an event to the main thread and asyncify wait for it
|
||
|
void WasmEventDispatcherTest::postEventAsyncify()
|
||
|
{
|
||
|
if (!qstdweb::haveAsyncify()) {
|
||
|
QtWasmTest::completeTestFunction(QtWasmTest::TestResult::Skip, "requires asyncify");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
QEventLoop loop;
|
||
|
QCoreApplication::postEvent(EventTarget::create([&loop](){
|
||
|
loop.quit();
|
||
|
}), EventTarget::createEvent());
|
||
|
loop.exec();
|
||
|
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
}
|
||
|
|
||
|
// Create a timer on the main thread and asyncify wait for it
|
||
|
void WasmEventDispatcherTest::timerAsyncify()
|
||
|
{
|
||
|
if (!qstdweb::haveAsyncify()) {
|
||
|
QtWasmTest::completeTestFunction(QtWasmTest::TestResult::Skip, "requires asyncify");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
QEventLoop loop;
|
||
|
QTimer::singleShot(timerTimeout, [&loop](){
|
||
|
loop.quit();
|
||
|
});
|
||
|
loop.exec();
|
||
|
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
}
|
||
|
|
||
|
// Asyncify wait in a loop
|
||
|
void WasmEventDispatcherTest::postEventAsyncifyLoop()
|
||
|
{
|
||
|
if (!qstdweb::haveAsyncify()) {
|
||
|
QtWasmTest::completeTestFunction(QtWasmTest::TestResult::Skip, "requires asyncify");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
for (int i = 0; i < 10; ++i) {
|
||
|
QEventLoop loop;
|
||
|
QCoreApplication::postEvent(EventTarget::create([&loop]() {
|
||
|
loop.quit();
|
||
|
}), EventTarget::createEvent());
|
||
|
loop.exec();
|
||
|
}
|
||
|
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
}
|
||
|
|
||
|
#if QT_CONFIG(thread)
|
||
|
// Asyncify wait for QThread::wait() / pthread_join()
|
||
|
void WasmEventDispatcherTest::threadAsyncifyWait()
|
||
|
{
|
||
|
if (!qstdweb::haveAsyncify())
|
||
|
QtWasmTest::completeTestFunction(QtWasmTest::TestResult::Skip, "requires asyncify");
|
||
|
|
||
|
const int threadCount = 15;
|
||
|
|
||
|
QVector<QThread *> threads;
|
||
|
threads.reserve(threadCount);
|
||
|
|
||
|
for (int i = 0; i < threadCount; ++i) {
|
||
|
QThread *thread = new QThread();
|
||
|
threads.push_back(thread);
|
||
|
thread->start();
|
||
|
}
|
||
|
|
||
|
for (int i = 0; i < threadCount; ++i) {
|
||
|
QThread *thread = threads[i];
|
||
|
thread->wait();
|
||
|
delete thread;
|
||
|
}
|
||
|
|
||
|
QtWasmTest::completeTestFunction();
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
auto testObject = std::make_shared<WasmEventDispatcherTest>();
|
||
|
QtWasmTest::initTestCase<QCoreApplication>(argc, argv, testObject);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#include "main.moc"
|