// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause #include "frequencymonitor.h" #include #include #include #include #include //#define VERBOSE_TRACE inline QDebug qtTrace() { return qDebug() << "[frequencymonitor]"; } #ifdef VERBOSE_TRACE inline QDebug qtVerboseTrace() { return qtTrace(); } #else inline QNoDebug qtVerboseTrace() { return QNoDebug(); } #endif static const int DefaultSamplingInterval = 100; static const int DefaultTraceInterval = 0; class FrequencyMonitorPrivate : public QObject { Q_OBJECT public: FrequencyMonitorPrivate(FrequencyMonitor *parent); void calculateInstantaneousFrequency(); private slots: void calculateAverageFrequency(); void stalled(); public: FrequencyMonitor *const q_ptr; QString m_label; bool m_active; qreal m_instantaneousFrequency; QElapsedTimer m_instantaneousElapsed; QTimer *m_averageTimer; QElapsedTimer m_averageElapsed; int m_count; qreal m_averageFrequency; QTimer *m_traceTimer; QTimer *m_stalledTimer; }; FrequencyMonitorPrivate::FrequencyMonitorPrivate(FrequencyMonitor *parent) : QObject(parent) , q_ptr(parent) , m_active(false) , m_instantaneousFrequency(0) , m_averageTimer(new QTimer(this)) , m_count(0) , m_averageFrequency(0) , m_traceTimer(new QTimer(this)) , m_stalledTimer(new QTimer(this)) { m_instantaneousElapsed.start(); connect(m_averageTimer, &QTimer::timeout, this, &FrequencyMonitorPrivate::calculateAverageFrequency); if (DefaultSamplingInterval) m_averageTimer->start(DefaultSamplingInterval); m_averageElapsed.start(); connect(m_traceTimer, &QTimer::timeout, q_ptr, &FrequencyMonitor::trace); if (DefaultTraceInterval) m_traceTimer->start(DefaultTraceInterval); m_stalledTimer->setSingleShot(true); connect(m_stalledTimer, &QTimer::timeout, this, &FrequencyMonitorPrivate::stalled); } void FrequencyMonitorPrivate::calculateInstantaneousFrequency() { const qint64 ms = m_instantaneousElapsed.restart(); m_instantaneousFrequency = ms ? qreal(1000) / ms : 0; m_stalledTimer->start(3 * ms); if (m_instantaneousFrequency) q_ptr->setActive(true); emit q_ptr->instantaneousFrequencyChanged(m_instantaneousFrequency); emit q_ptr->frequencyChanged(); } void FrequencyMonitorPrivate::calculateAverageFrequency() { const qint64 ms = m_averageElapsed.restart(); m_averageFrequency = qreal(m_count * 1000) / ms; emit q_ptr->averageFrequencyChanged(m_averageFrequency); emit q_ptr->frequencyChanged(); m_count = 0; } void FrequencyMonitorPrivate::stalled() { if (m_instantaneousFrequency) { m_instantaneousFrequency = 0; emit q_ptr->instantaneousFrequencyChanged(m_instantaneousFrequency); emit q_ptr->frequencyChanged(); } } FrequencyMonitor::FrequencyMonitor(QObject *parent) : QObject(parent) { d_ptr = new FrequencyMonitorPrivate(this); } FrequencyMonitor::~FrequencyMonitor() = default; QString FrequencyMonitor::label() const { return d_func()->m_label; } bool FrequencyMonitor::active() const { return d_func()->m_active; } int FrequencyMonitor::samplingInterval() const { return d_ptr->m_averageTimer->isActive() ? d_ptr->m_averageTimer->interval() : 0; } int FrequencyMonitor::traceInterval() const { return d_ptr->m_traceTimer->isActive() ? d_ptr->m_traceTimer->interval() : 0; } qreal FrequencyMonitor::instantaneousFrequency() const { return d_func()->m_instantaneousFrequency; } qreal FrequencyMonitor::averageFrequency() const { return d_func()->m_averageFrequency; } void FrequencyMonitor::notify() { Q_D(FrequencyMonitor); ++(d->m_count); d->calculateInstantaneousFrequency(); } void FrequencyMonitor::trace() { Q_D(FrequencyMonitor); const QString value = QString::fromLatin1("instant %1 average %2") .arg(d->m_instantaneousFrequency, 0, 'f', 2) .arg(d->m_averageFrequency, 0, 'f', 2); if (d->m_label.isEmpty()) qtTrace() << "FrequencyMonitor::trace" << value; else qtTrace() << "FrequencyMonitor::trace" << "label" << d->m_label << value; } void FrequencyMonitor::setLabel(const QString &value) { Q_D(FrequencyMonitor); if (d->m_label != value) { d->m_label = value; emit labelChanged(d->m_label); } } void FrequencyMonitor::setActive(bool value) { Q_D(FrequencyMonitor); if (d->m_active != value) { d->m_active = value; emit activeChanged(d->m_active); } } void FrequencyMonitor::setSamplingInterval(int value) { Q_D(FrequencyMonitor); if (samplingInterval() != value) { if (value) { d->m_averageTimer->setInterval(value); d->m_averageTimer->start(); } else { d->m_averageTimer->stop(); } emit samplingIntervalChanged(value); } } void FrequencyMonitor::setTraceInterval(int value) { Q_D(FrequencyMonitor); if (traceInterval() != value) { if (value) { d->m_traceTimer->setInterval(value); d->m_traceTimer->start(); } else { d->m_traceTimer->stop(); } emit traceIntervalChanged(value); } } #include "frequencymonitor.moc"