statemachine: Really fix signal transition handling in multi-threaded setup
Commit f9a17d7f0f
fixed it for the case
where the sender object is in a different thread at transition setup
time. However, it still didn't work if either the sender object or the
state machine was moved to a different thread at some later time,
before the machine was started.
Therefore: Bite the sour grape and traverse all the machine's
transitions when the machine is being started, registering those
signal transitions whose sender objects are in other threads.
This will increase the machine's startup time (proportional to the
number of transitions), but at least it works in all known scenarios,
meaning we don't have to document weird restrictions regarding the
order in which the user's operations have to be done.
Task-number: QTBUG-19789
Change-Id: I5f1dd1321994e49635f52be65cf56d2678ed1253
Reviewed-by: Eskil Abrahamsen Blomfeldt <eskil.abrahamsen-blomfeldt@nokia.com>
This commit is contained in:
parent
f7272b4587
commit
8efcfda41f
@ -1335,6 +1335,28 @@ void QStateMachinePrivate::clearHistory()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*!
|
||||||
|
\internal
|
||||||
|
|
||||||
|
Registers all signal transitions whose sender object lives in another thread.
|
||||||
|
|
||||||
|
Normally, signal transitions are lazily registered (when a state becomes
|
||||||
|
active). But if the sender is in a different thread, the transition must be
|
||||||
|
registered early to keep the state machine from "dropping" signals; e.g.,
|
||||||
|
a second (transition-bound) signal could be emitted on the sender thread
|
||||||
|
before the state machine gets to process the first signal.
|
||||||
|
*/
|
||||||
|
void QStateMachinePrivate::registerMultiThreadedSignalTransitions()
|
||||||
|
{
|
||||||
|
Q_Q(QStateMachine);
|
||||||
|
QList<QSignalTransition*> transitions = rootState()->findChildren<QSignalTransition*>();
|
||||||
|
for (int i = 0; i < transitions.size(); ++i) {
|
||||||
|
QSignalTransition *t = transitions.at(i);
|
||||||
|
if ((t->machine() == q) && t->senderObject() && (t->senderObject()->thread() != q->thread()))
|
||||||
|
registerSignalTransition(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void QStateMachinePrivate::_q_start()
|
void QStateMachinePrivate::_q_start()
|
||||||
{
|
{
|
||||||
Q_Q(QStateMachine);
|
Q_Q(QStateMachine);
|
||||||
@ -1346,6 +1368,8 @@ void QStateMachinePrivate::_q_start()
|
|||||||
externalEventQueue.clear();
|
externalEventQueue.clear();
|
||||||
clearHistory();
|
clearHistory();
|
||||||
|
|
||||||
|
registerMultiThreadedSignalTransitions();
|
||||||
|
|
||||||
#ifdef QSTATEMACHINE_DEBUG
|
#ifdef QSTATEMACHINE_DEBUG
|
||||||
qDebug() << q << ": starting";
|
qDebug() << q << ": starting";
|
||||||
#endif
|
#endif
|
||||||
@ -1703,8 +1727,8 @@ void QStateMachinePrivate::unregisterTransition(QAbstractTransition *transition)
|
|||||||
void QStateMachinePrivate::maybeRegisterSignalTransition(QSignalTransition *transition)
|
void QStateMachinePrivate::maybeRegisterSignalTransition(QSignalTransition *transition)
|
||||||
{
|
{
|
||||||
Q_Q(QStateMachine);
|
Q_Q(QStateMachine);
|
||||||
if (((state == Running) && configuration.contains(transition->sourceState()))
|
if ((state == Running) && (configuration.contains(transition->sourceState())
|
||||||
|| (transition->senderObject() && (transition->senderObject()->thread() != q->thread()))) {
|
|| (transition->senderObject() && (transition->senderObject()->thread() != q->thread())))) {
|
||||||
registerSignalTransition(transition);
|
registerSignalTransition(transition);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1771,14 +1795,10 @@ void QStateMachinePrivate::registerSignalTransition(QSignalTransition *transitio
|
|||||||
|
|
||||||
void QStateMachinePrivate::unregisterSignalTransition(QSignalTransition *transition)
|
void QStateMachinePrivate::unregisterSignalTransition(QSignalTransition *transition)
|
||||||
{
|
{
|
||||||
Q_Q(QStateMachine);
|
|
||||||
int signalIndex = QSignalTransitionPrivate::get(transition)->signalIndex;
|
int signalIndex = QSignalTransitionPrivate::get(transition)->signalIndex;
|
||||||
if (signalIndex == -1)
|
if (signalIndex == -1)
|
||||||
return; // not registered
|
return; // not registered
|
||||||
const QObject *sender = QSignalTransitionPrivate::get(transition)->sender;
|
const QObject *sender = QSignalTransitionPrivate::get(transition)->sender;
|
||||||
// Don't unregister if the sender is on another thread
|
|
||||||
if (!sender || (sender->thread() != q->thread()))
|
|
||||||
return;
|
|
||||||
QSignalTransitionPrivate::get(transition)->signalIndex = -1;
|
QSignalTransitionPrivate::get(transition)->signalIndex = -1;
|
||||||
|
|
||||||
connectionsMutex.lock();
|
connectionsMutex.lock();
|
||||||
|
@ -175,6 +175,7 @@ public:
|
|||||||
void maybeRegisterSignalTransition(QSignalTransition *transition);
|
void maybeRegisterSignalTransition(QSignalTransition *transition);
|
||||||
void registerSignalTransition(QSignalTransition *transition);
|
void registerSignalTransition(QSignalTransition *transition);
|
||||||
void unregisterSignalTransition(QSignalTransition *transition);
|
void unregisterSignalTransition(QSignalTransition *transition);
|
||||||
|
void registerMultiThreadedSignalTransitions();
|
||||||
#ifndef QT_NO_STATEMACHINE_EVENTFILTER
|
#ifndef QT_NO_STATEMACHINE_EVENTFILTER
|
||||||
void maybeRegisterEventTransition(QEventTransition *transition);
|
void maybeRegisterEventTransition(QEventTransition *transition);
|
||||||
void registerEventTransition(QEventTransition *transition);
|
void registerEventTransition(QEventTransition *transition);
|
||||||
|
@ -209,6 +209,7 @@ private slots:
|
|||||||
void createSignalTransitionWhenRunning();
|
void createSignalTransitionWhenRunning();
|
||||||
void createEventTransitionWhenRunning();
|
void createEventTransitionWhenRunning();
|
||||||
void signalTransitionSenderInDifferentThread();
|
void signalTransitionSenderInDifferentThread();
|
||||||
|
void signalTransitionSenderInDifferentThread2();
|
||||||
void signalTransitionRegistrationThreadSafety();
|
void signalTransitionRegistrationThreadSafety();
|
||||||
void childModeConstructor();
|
void childModeConstructor();
|
||||||
};
|
};
|
||||||
@ -4895,6 +4896,43 @@ void tst_QStateMachine::signalTransitionSenderInDifferentThread()
|
|||||||
QTRY_VERIFY(thread.wait());
|
QTRY_VERIFY(thread.wait());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void tst_QStateMachine::signalTransitionSenderInDifferentThread2()
|
||||||
|
{
|
||||||
|
QStateMachine machine;
|
||||||
|
QState *s1 = new QState(&machine);
|
||||||
|
machine.setInitialState(s1);
|
||||||
|
|
||||||
|
QState *s2 = new QState(&machine);
|
||||||
|
SignalEmitter emitter;
|
||||||
|
// At the time of the transition creation, the machine and the emitter
|
||||||
|
// are both in the same thread.
|
||||||
|
s1->addTransition(&emitter, SIGNAL(signalWithNoArg()), s2);
|
||||||
|
|
||||||
|
QFinalState *s3 = new QFinalState(&machine);
|
||||||
|
s2->addTransition(&emitter, SIGNAL(signalWithDefaultArg()), s3);
|
||||||
|
|
||||||
|
QThread thread;
|
||||||
|
// Move the machine and its states to a secondary thread, but let the
|
||||||
|
// SignalEmitter stay in the main thread.
|
||||||
|
machine.moveToThread(&thread);
|
||||||
|
|
||||||
|
thread.start();
|
||||||
|
QTRY_VERIFY(thread.isRunning());
|
||||||
|
|
||||||
|
QSignalSpy startedSpy(&machine, SIGNAL(started()));
|
||||||
|
QSignalSpy finishedSpy(&machine, SIGNAL(finished()));
|
||||||
|
machine.start();
|
||||||
|
QTRY_COMPARE(startedSpy.count(), 1);
|
||||||
|
|
||||||
|
emitter.emitSignalWithNoArg();
|
||||||
|
// The second emission should not get "lost".
|
||||||
|
emitter.emitSignalWithDefaultArg();
|
||||||
|
QTRY_COMPARE(finishedSpy.count(), 1);
|
||||||
|
|
||||||
|
thread.quit();
|
||||||
|
QTRY_VERIFY(thread.wait());
|
||||||
|
}
|
||||||
|
|
||||||
class SignalTransitionMutatorThread : public QThread
|
class SignalTransitionMutatorThread : public QThread
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
Loading…
Reference in New Issue
Block a user