6#include "QskObjectCounter.h"
7#include "QskInternalMacros.h"
12#include <qcoreapplication.h>
15#include <private/qhooks_p.h>
16#include <private/qquickitem_p.h>
19#define QSK_OBJECT_INFO 0
25static inline bool qskIsItem(
const QObject*
object )
34 auto o_p = QObjectPrivate::get(
const_cast< QObject*
>(
object ) );
35 return dynamic_cast< QQuickItemPrivate*
>( o_p ) !=
nullptr;
50 created = destroyed = current = maximum = 0;
58 if ( current > maximum )
80 QSet< const QObject* > objectTable;
90 void registerCounters( CounterData*,
bool on );
91 bool isCountersRegistered(
const CounterData* )
const;
93 bool isActive()
const;
96 void addObject( QObject* );
97 void removeObject( QObject* );
99 static void cleanupHook();
102 static void startupHook();
103 static void addObjectHook( QObject* );
104 static void removeObjectHook( QObject* );
106 QSet< CounterData* > m_counterDataSet;
108 quintptr m_otherStartup;
109 quintptr m_otherAddObject;
110 quintptr m_otherRemoveObject;
114static bool qskAutoDeleteHook =
false;
115static CounterHook* qskCounterHook =
nullptr;
117CounterHook::CounterHook()
119 m_otherStartup = qtHookData[ QHooks::Startup ];
120 m_otherAddObject = qtHookData[ QHooks::AddQObject ];
121 m_otherRemoveObject = qtHookData[ QHooks::RemoveQObject ];
123 qtHookData[ QHooks::Startup ] =
reinterpret_cast< quintptr
>( &startupHook );
124 qtHookData[ QHooks::AddQObject ] =
reinterpret_cast< quintptr
>( &addObjectHook );
125 qtHookData[ QHooks::RemoveQObject ] =
reinterpret_cast< quintptr
>( &removeObjectHook );
128CounterHook::~CounterHook()
130 qtHookData[ QHooks::Startup ] = m_otherStartup;
131 qtHookData[ QHooks::AddQObject ] = m_otherAddObject;
132 qtHookData[ QHooks::RemoveQObject ] = m_otherRemoveObject;
135void CounterHook::registerCounters( CounterData* data,
bool on )
138 m_counterDataSet.insert( data );
140 m_counterDataSet.remove( data );
143bool CounterHook::isCountersRegistered(
const CounterData* data )
const
145 return m_counterDataSet.contains(
const_cast< CounterData*
>( data ) );
148bool CounterHook::isActive()
const
150 return !m_counterDataSet.isEmpty();
153void CounterHook::startup()
156 qDebug() <<
"** QskObjectCounterHook enabled";
158 if ( m_otherStartup )
159 reinterpret_cast< QHooks::StartupCallback
>( m_otherStartup )();
162void CounterHook::addObject( QObject*
object )
164 const bool isItem = qskIsItem(
object );
166 for (
auto counterData : std::as_const( m_counterDataSet ) )
168 counterData->counter[ QskObjectCounter::Objects ].increment();
171 counterData->counter[ QskObjectCounter::Items ].increment();
174 counterData->objectTable.insert(
object );
178 if ( m_otherAddObject )
179 reinterpret_cast< QHooks::AddQObjectCallback
>( m_otherAddObject )(
object );
182void CounterHook::removeObject( QObject*
object )
184 const bool isItem = qskIsItem(
object );
186 for (
auto counterData : std::as_const( m_counterDataSet ) )
188 counterData->counter[ QskObjectCounter::Objects ].decrement();
191 counterData->counter[ QskObjectCounter::Items ].decrement();
194 counterData->objectTable.remove(
object );
198 if ( m_otherRemoveObject )
199 reinterpret_cast< QHooks::RemoveQObjectCallback
>( m_otherRemoveObject )(
object );
202void CounterHook::startupHook()
204 if ( qskCounterHook )
205 qskCounterHook->startup();
208void CounterHook::addObjectHook( QObject*
object )
210 if ( qskCounterHook )
211 qskCounterHook->addObject(
object );
214void CounterHook::removeObjectHook( QObject*
object )
216 if ( qskCounterHook )
217 qskCounterHook->removeObject(
object );
220void CounterHook::cleanupHook()
222 if ( qskCounterHook && !qskCounterHook->isActive() )
224 delete qskCounterHook;
225 qskCounterHook =
nullptr;
229 qskAutoDeleteHook =
true;
232static void qskInstallCleanupHookHandler()
234 qAddPostRoutine( CounterHook::cleanupHook );
237Q_COREAPP_STARTUP_FUNCTION( qskInstallCleanupHookHandler )
239class QskObjectCounter::PrivateData
242 PrivateData(
bool debugAtDestruction )
243 : debugAtDestruction( debugAtDestruction )
247 CounterData counterData;
248 const bool debugAtDestruction;
251QskObjectCounter::QskObjectCounter(
bool debugAtDestruction )
252 : m_data( new PrivateData( debugAtDestruction ) )
257QskObjectCounter::~QskObjectCounter()
261 if ( m_data->debugAtDestruction )
265void QskObjectCounter::setActive(
bool on )
269 if ( qskCounterHook ==
nullptr )
270 qskCounterHook =
new CounterHook();
272 qskCounterHook->registerCounters( &m_data->counterData, on );
276 qskCounterHook->registerCounters( &m_data->counterData, on );
277 if ( !qskCounterHook->isActive() )
279 if ( qskAutoDeleteHook )
281 delete qskCounterHook;
282 qskCounterHook =
nullptr;
288bool QskObjectCounter::isActive()
const
290 return qskCounterHook && qskCounterHook->isCountersRegistered( &m_data->counterData );
293void QskObjectCounter::reset()
295 auto& counters = m_data->counterData.counter;
297 counters[ Objects ].reset();
298 counters[ Items ].reset();
301int QskObjectCounter::created( ObjectType objectType )
const
303 return m_data->counterData.counter[ objectType ].created;
306int QskObjectCounter::destroyed( ObjectType objectType )
const
308 return m_data->counterData.counter[ objectType ].destroyed;
311int QskObjectCounter::current( ObjectType objectType )
const
313 return m_data->counterData.counter[ objectType ].current;
316int QskObjectCounter::maximum( ObjectType objectType )
const
318 return m_data->counterData.counter[ objectType ].maximum;
321void QskObjectCounter::debugStatistics( QDebug debug, ObjectType objectType )
const
323 const auto& c = m_data->counterData.counter[ objectType ];
325 QDebugStateSaver saver( debug );
328 debug <<
"created: " << c.created
329 <<
", destroyed: " << c.destroyed
330 <<
", current: " << c.current
331 <<
", maximum: " << c.maximum;
335 if ( objectType == Objects )
337 const auto& objectTable = m_data->counterData.objectTable;
339 if ( !objectTable.isEmpty() )
341 debug <<
"\n\t=== Leaks ===\n";
342 for (
const auto object : objectTable )
344 debug <<
"\tClass: " <<
object->metaObject()->className();
345 if ( !object->objectName().isEmpty() )
346 debug <<
" Name: " <<
object->objectName();
354void QskObjectCounter::dump()
const
356 QDebug debug = qDebug();
358 QDebugStateSaver saver( debug );
362 debug <<
"* Statistics\n";
363 debug <<
" Objects: ";
364 debugStatistics( debug, Objects );
366 debug <<
"\n Items: ";
367 debugStatistics( debug, Items );
370#ifndef QT_NO_DEBUG_STREAM
374 counter.debugStatistics( debug, QskObjectCounter::Objects );