Perceptual Color

asyncimagerenderthread.cpp
1// SPDX-FileCopyrightText: Lukas Sommer <sommerluk@gmail.com>
2// SPDX-License-Identifier: BSD-2-Clause OR MIT
3
4// Own headers
5// First the interface, which forces the header to be self-contained.
6#include "asyncimagerenderthread.h"
7
8#include <qglobal.h>
9#include <qmetatype.h>
10
11class QObject;
12
13namespace PerceptualColor
14{
15/** @brief The constructor.
16 *
17 * @param renderFunction Pointer to the render function that will be used.
18 * @param parent The widget’s parent widget. This parameter will be passed
19 * to the base class’s constructor. */
20AsyncImageRenderThread::AsyncImageRenderThread(const pointerToRenderFunction &renderFunction, QObject *parent)
21 : QThread(parent)
22 , m_renderFunction(renderFunction)
23{
24 qRegisterMetaType<PerceptualColor::AsyncImageRenderCallback::InterlacingState>();
25}
26
27/** @brief The destructor.
28 *
29 * This destructor might takes a little while because he has to
30 * stop the associated thread before destroying it: Possibly running
31 * rendering operations are aborted. */
32AsyncImageRenderThread::~AsyncImageRenderThread()
33{
34 m_loopMutex.lock();
35 m_loopAbort = true;
36 m_loopCondition.wakeOne();
37 m_loopMutex.unlock();
38
39 wait(); // Wait for the thread to terminate.
40
41 // We make sure no thread will stay blocked when this object is
42 // destroyed. However, given that this class itself is NOT thread-safe,
43 // anyway it isn’t allowed to execute the destructor and waitForIdle()
44 // in parallel. Therefore, this should be a no-operation. We stays
45 // here just to feel safe.
46 m_syncCondition.wakeAll();
47}
48
49/** @brief Asynchronously start rendering.
50 *
51 * As this function is asynchronous, it will return very fast.
52 *
53 * @param parameters The parameters of the requested rendering.
54 *
55 * @post If the <tt>parameters</tt> are different from those at the last
56 * call, a new rendering of the new parameters will be started. (If there
57 * is currently a rendering of other parameters in progress, this rendering
58 * will be requested to stop as soon as possible.) If the <tt>parameters</tt>
59 * are identical to those at the last call, nothing happens.
60 *
61 * The rendering will emit the signal @ref interlacingPassCompleted(). */
62void AsyncImageRenderThread::startRenderingAsync(const QVariant &parameters)
63{
64#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
65 QMutexLocker<QMutex> loopLocker(&m_loopMutex);
66#else
67 QMutexLocker loopLocker(&m_loopMutex);
68#endif
69
70 if (m_imageParameters == parameters) {
71 // Nothing to do here.
72 return;
73 }
74
75 m_imageParameters = parameters;
76
77 {
78#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
79 QMutexLocker<QMutex> syncLocker(&m_syncMutex);
80#else
81 QMutexLocker syncLocker(&m_syncMutex);
82#endif
83 m_syncIsIdle = false;
84 }
85 if (!isRunning()) {
86#if (QT_VERSION >= QT_VERSION_CHECK(6, 9, 0))
87 // The scheduler should run this thread to a high-performance CPU core
88 // to get the image as fast as possible.
89 setServiceLevel(QThread::QualityOfService::High);
90#endif
91 // But avoid blocking other threads:
92 start(LowPriority); // One priority level lower than normal priority.
93 } else {
94 m_loopRestart = true;
95 m_loopCondition.wakeOne();
96 }
97}
98
99/** @brief The code that will run within the thread.
100 *
101 * Reimplemented from base class.
102 *
103 * This is a wrapper that provides the thread-control (loops and so on).
104 * The actual rendering is done by calling @ref m_renderFunction. */
105void AsyncImageRenderThread::run()
106{
107 Q_FOREVER {
108 m_loopMutex.lock();
109 const QVariant parameters = m_imageParameters;
110 m_loopMutex.unlock();
111
112 if (m_loopAbort) {
113 return;
114 }
115
116 // From Qt Example’s documentation:
117 //
118 // “If we discover inside […] [this function call] that restart
119 // has been set to true (by render()), this function will return
120 // immediately, so that the control quickly returns to the very
121 // top of […] the forever loop […] and we fetch the new rendering
122 // parameters. Similarly, if we discover that abort has been set
123 // to true (by the RenderThread destructor), we return from the
124 // function immediately, terminating the thread.”
125 //
126 // Here, this is done by passing m_abortRun and m_restart (in form of
127 // shouldAbort())to the render function, which is supposed to return
128 // as fast as possible if indicated.
129 m_renderFunction(parameters, *this);
130
131 // cppcheck-suppress identicalConditionAfterEarlyExit // false positive
132 if (m_loopAbort) {
133 return;
134 }
135
136 // From Qt’s examples:
137 // “Once we're done with all the iterations, we call
138 // QWaitCondition::wait() to put the thread to sleep, unless
139 // restart is true. There's no use in keeping a worker thread
140 // looping indefinitely while there's nothing to do.”
141 m_loopMutex.lock();
142 if (!m_loopRestart && !m_loopAbort) {
143#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
144 QMutexLocker<QMutex> syncLocker(&m_syncMutex);
145#else
146 QMutexLocker syncLocker(&m_syncMutex);
147#endif
148 m_syncIsIdle = true;
149 m_syncCondition.wakeOne();
150 }
151 while (!m_loopRestart && !m_loopAbort) {
152 // QWaitCondition::wait() does the following things:
153 // 1.) Unlock mutex.
154 // 2.) Wait until another thread calls QWaitCondition::wakeOne()
155 // or QWaitCondition::wakeAll().
156 // 3.) Lock mutex again.
157 //
158 // As explained on the StackOverflow webpage at
159 // https://stackoverflow.com/questions/40445629
160 // using QWaitCondition::wait() alone can do spurious
161 // wake-up (wake-up without a reason). To prevent the
162 // rendering from continuing in this case, we put the
163 // QWaitCondition::wait() call into a while loop. This way
164 // we can go back to sleep if the wake-up was without
165 // reason (this is checked by the condition in the while loop).
166 m_loopCondition.wait(&m_loopMutex);
167 }
168 m_loopRestart = false;
169 m_loopMutex.unlock();
170 }
171}
172
173/** @brief Deliver the result of an interlacing pass of
174 * the <em>rendering</em> operation.
175 *
176 * This function is thread-safe.
177 *
178 * @param image The image
179 * @param parameters The parameters of the image
180 * @param state The interlacing state of the image. A render function
181 * must first return zero or more images with intermediate state. After
182 * that, it must return exactly one image with final state (unless it
183 * was aborted). After that, it must not return any more images. */
184void AsyncImageRenderThread::deliverInterlacingPass(const QImage &image, const QVariant &parameters, const AsyncImageRenderCallback::InterlacingState state)
185{
186 // interlacingPassCompleted() is documented as being possibly emitted
187 // by different threads, so this call is thread-safe within the
188 // restrictions mentioned in the documentation.
189 Q_EMIT interlacingPassCompleted(image, parameters, state);
190}
191
192/** @brief If the render function should abort.
193 *
194 * This function is thread-safe.
195 *
196 * @returns <tt>true</tt> if the render function should abort (and
197 * return). <tt>false</tt> otherwise.
198 *
199 * @internal
200 *
201 * @sa @ref m_renderFunction */
202bool AsyncImageRenderThread::shouldAbort() const
203{
204 // m_abortRun and m_restart are atomic, so this call is thread-safe.
205 return (m_loopAbort || m_loopRestart);
206}
207
208/** @brief Wait until the render thread is idle. */
209void AsyncImageRenderThread::waitForIdle()
210{
211 m_syncMutex.lock();
212 while (!m_syncIsIdle) {
213 // QWaitCondition::wait() does the following things:
214 // 1.) Unlock mutex.
215 // 2.) Wait until another thread calls QWaitCondition::wakeOne()
216 // or QWaitCondition::wakeAll().
217 // 3.) Lock mutex again.
218 //
219 // As explained on the StackOverflow webpage at
220 // https://stackoverflow.com/questions/40445629
221 // using QWaitCondition::wait() alone can do spurious
222 // wake-up (wake-up without a reason). To prevent the
223 // rendering from continuing in this case, we put the
224 // QWaitCondition::wait() call into a while loop. This way
225 // we can go back to sleep if the wake-up was without
226 // reason (this is checked by the condition in the while loop).
227 m_syncCondition.wait(&m_syncMutex);
228 }
229 m_syncMutex.unlock();
230}
231
232} // namespace PerceptualColor
Q_SCRIPTABLE Q_NOREPLY void start()
The namespace of this library.
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 11:54:42 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.