QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskObjectCounter.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskObjectCounter.h"
7#include "QskInternalMacros.h"
8
9#include <qdebug.h>
10#include <qset.h>
11#include <qobject.h>
12#include <qcoreapplication.h>
13
14QSK_QT_PRIVATE_BEGIN
15#include <private/qhooks_p.h>
16#include <private/qquickitem_p.h>
17QSK_QT_PRIVATE_END
18
19#define QSK_OBJECT_INFO 0
20
21#if QSK_OBJECT_INFO
22#include <qset.h>
23#endif
24
25static inline bool qskIsItem( const QObject* object )
26{
27 /*
28 The addObject hook is called from the constructor of QObject,
29 where we don't have the derived class constructed yet.
30 So we can't cast the object itself and also have to rely on
31 RTTI being enabled. TODO ...
32 */
33
34 auto o_p = QObjectPrivate::get( const_cast< QObject* >( object ) );
35 return dynamic_cast< QQuickItemPrivate* >( o_p ) != nullptr;
36}
37
38namespace
39{
40 class Counter
41 {
42 public:
43 Counter()
44 {
45 reset();
46 }
47
48 void reset()
49 {
50 created = destroyed = current = maximum = 0;
51 }
52
53 void increment()
54 {
55 created++;
56 current++;
57
58 if ( current > maximum )
59 maximum = current;
60 }
61
62 void decrement()
63 {
64 destroyed++;
65 current--;
66 }
67
68 int created;
69 int destroyed;
70 int current;
71 int maximum;
72 };
73
74 class CounterData
75 {
76 public:
77 Counter counter[ 2 ];
78
79#if QSK_OBJECT_INFO
80 QSet< const QObject* > objectTable;
81#endif
82 };
83
84 class CounterHook
85 {
86 public:
87 CounterHook();
88 ~CounterHook();
89
90 void registerCounters( CounterData*, bool on );
91 bool isCountersRegistered( const CounterData* ) const;
92
93 bool isActive() const;
94
95 void startup();
96 void addObject( QObject* );
97 void removeObject( QObject* );
98
99 static void cleanupHook();
100
101 private:
102 static void startupHook();
103 static void addObjectHook( QObject* );
104 static void removeObjectHook( QObject* );
105
106 QSet< CounterData* > m_counterDataSet;
107
108 quintptr m_otherStartup;
109 quintptr m_otherAddObject;
110 quintptr m_otherRemoveObject;
111 };
112}
113
114static bool qskAutoDeleteHook = false;
115static CounterHook* qskCounterHook = nullptr;
116
117CounterHook::CounterHook()
118{
119 m_otherStartup = qtHookData[ QHooks::Startup ];
120 m_otherAddObject = qtHookData[ QHooks::AddQObject ];
121 m_otherRemoveObject = qtHookData[ QHooks::RemoveQObject ];
122
123 qtHookData[ QHooks::Startup ] = reinterpret_cast< quintptr >( &startupHook );
124 qtHookData[ QHooks::AddQObject ] = reinterpret_cast< quintptr >( &addObjectHook );
125 qtHookData[ QHooks::RemoveQObject ] = reinterpret_cast< quintptr >( &removeObjectHook );
126}
127
128CounterHook::~CounterHook()
129{
130 qtHookData[ QHooks::Startup ] = m_otherStartup;
131 qtHookData[ QHooks::AddQObject ] = m_otherAddObject;
132 qtHookData[ QHooks::RemoveQObject ] = m_otherRemoveObject;
133}
134
135void CounterHook::registerCounters( CounterData* data, bool on )
136{
137 if ( on )
138 m_counterDataSet.insert( data );
139 else
140 m_counterDataSet.remove( data );
141}
142
143bool CounterHook::isCountersRegistered( const CounterData* data ) const
144{
145 return m_counterDataSet.contains( const_cast< CounterData* >( data ) );
146}
147
148bool CounterHook::isActive() const
149{
150 return !m_counterDataSet.isEmpty();
151}
152
153void CounterHook::startup()
154{
155#if 0
156 qDebug() << "** QskObjectCounterHook enabled";
157#endif
158 if ( m_otherStartup )
159 reinterpret_cast< QHooks::StartupCallback >( m_otherStartup )();
160}
161
162void CounterHook::addObject( QObject* object )
163{
164 const bool isItem = qskIsItem( object );
165
166 for ( auto counterData : std::as_const( m_counterDataSet ) )
167 {
168 counterData->counter[ QskObjectCounter::Objects ].increment();
169
170 if ( isItem )
171 counterData->counter[ QskObjectCounter::Items ].increment();
172
173#if QSK_OBJECT_INFO
174 counterData->objectTable.insert( object );
175#endif
176 }
177
178 if ( m_otherAddObject )
179 reinterpret_cast< QHooks::AddQObjectCallback >( m_otherAddObject )( object );
180}
181
182void CounterHook::removeObject( QObject* object )
183{
184 const bool isItem = qskIsItem( object );
185
186 for ( auto counterData : std::as_const( m_counterDataSet ) )
187 {
188 counterData->counter[ QskObjectCounter::Objects ].decrement();
189
190 if ( isItem )
191 counterData->counter[ QskObjectCounter::Items ].decrement();
192
193#if QSK_OBJECT_INFO
194 counterData->objectTable.remove( object );
195#endif
196 }
197
198 if ( m_otherRemoveObject )
199 reinterpret_cast< QHooks::RemoveQObjectCallback >( m_otherRemoveObject )( object );
200}
201
202void CounterHook::startupHook()
203{
204 if ( qskCounterHook )
205 qskCounterHook->startup();
206}
207
208void CounterHook::addObjectHook( QObject* object )
209{
210 if ( qskCounterHook )
211 qskCounterHook->addObject( object );
212}
213
214void CounterHook::removeObjectHook( QObject* object )
215{
216 if ( qskCounterHook )
217 qskCounterHook->removeObject( object );
218}
219
220void CounterHook::cleanupHook()
221{
222 if ( qskCounterHook && !qskCounterHook->isActive() )
223 {
224 delete qskCounterHook;
225 qskCounterHook = nullptr;
226 }
227
228 // From now on we remove the hooks as soon as there are no counters
229 qskAutoDeleteHook = true;
230}
231
232static void qskInstallCleanupHookHandler()
233{
234 qAddPostRoutine( CounterHook::cleanupHook );
235}
236
237Q_COREAPP_STARTUP_FUNCTION( qskInstallCleanupHookHandler )
238
239class QskObjectCounter::PrivateData
240{
241 public:
242 PrivateData( bool debugAtDestruction )
243 : debugAtDestruction( debugAtDestruction )
244 {
245 }
246
247 CounterData counterData;
248 const bool debugAtDestruction;
249};
250
251QskObjectCounter::QskObjectCounter( bool debugAtDestruction )
252 : m_data( new PrivateData( debugAtDestruction ) )
253{
254 setActive( true );
255}
256
257QskObjectCounter::~QskObjectCounter()
258{
259 setActive( false );
260
261 if ( m_data->debugAtDestruction )
262 dump();
263}
264
265void QskObjectCounter::setActive( bool on )
266{
267 if ( on )
268 {
269 if ( qskCounterHook == nullptr )
270 qskCounterHook = new CounterHook();
271
272 qskCounterHook->registerCounters( &m_data->counterData, on );
273 }
274 else
275 {
276 qskCounterHook->registerCounters( &m_data->counterData, on );
277 if ( !qskCounterHook->isActive() )
278 {
279 if ( qskAutoDeleteHook )
280 {
281 delete qskCounterHook;
282 qskCounterHook = nullptr;
283 }
284 }
285 }
286}
287
288bool QskObjectCounter::isActive() const
289{
290 return qskCounterHook && qskCounterHook->isCountersRegistered( &m_data->counterData );
291}
292
293void QskObjectCounter::reset()
294{
295 auto& counters = m_data->counterData.counter;
296
297 counters[ Objects ].reset();
298 counters[ Items ].reset();
299}
300
301int QskObjectCounter::created( ObjectType objectType ) const
302{
303 return m_data->counterData.counter[ objectType ].created;
304}
305
306int QskObjectCounter::destroyed( ObjectType objectType ) const
307{
308 return m_data->counterData.counter[ objectType ].destroyed;
309}
310
311int QskObjectCounter::current( ObjectType objectType ) const
312{
313 return m_data->counterData.counter[ objectType ].current;
314}
315
316int QskObjectCounter::maximum( ObjectType objectType ) const
317{
318 return m_data->counterData.counter[ objectType ].maximum;
319}
320
321void QskObjectCounter::debugStatistics( QDebug debug, ObjectType objectType ) const
322{
323 const auto& c = m_data->counterData.counter[ objectType ];
324
325 QDebugStateSaver saver( debug );
326 debug.nospace();
327 debug << '(';
328 debug << "created: " << c.created
329 << ", destroyed: " << c.destroyed
330 << ", current: " << c.current
331 << ", maximum: " << c.maximum;
332 debug << ')';
333
334#if QSK_OBJECT_INFO
335 if ( objectType == Objects )
336 {
337 const auto& objectTable = m_data->counterData.objectTable;
338
339 if ( !objectTable.isEmpty() )
340 {
341 debug << "\n\t=== Leaks ===\n";
342 for ( const auto object : objectTable )
343 {
344 debug << "\tClass: " << object->metaObject()->className();
345 if ( !object->objectName().isEmpty() )
346 debug << " Name: " << object->objectName();
347 debug << '\n';
348 }
349 }
350 }
351#endif
352}
353
354void QskObjectCounter::dump() const
355{
356 QDebug debug = qDebug();
357
358 QDebugStateSaver saver( debug );
359
360 debug.nospace();
361
362 debug << "* Statistics\n";
363 debug << " Objects: ";
364 debugStatistics( debug, Objects );
365
366 debug << "\n Items: ";
367 debugStatistics( debug, Items );
368}
369
370#ifndef QT_NO_DEBUG_STREAM
371
372QDebug operator<<( QDebug debug, const QskObjectCounter& counter )
373{
374 counter.debugStatistics( debug, QskObjectCounter::Objects );
375 return debug;
376}
377
378#endif