/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QQMLPROFILERSERVICE_P_H #define QQMLPROFILERSERVICE_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE struct Q_AUTOTEST_EXPORT QQmlProfilerData { qint64 time; int messageType; int detailType; //### QString detailData; //used by RangeData and RangeLocation int line; //used by RangeLocation int column; //used by RangeLocation int framerate; //used by animation events int animationcount; //used by animation events int bindingType; QByteArray toByteArray() const; }; Q_DECLARE_TYPEINFO(QQmlProfilerData, Q_MOVABLE_TYPE); class QUrl; class QQmlEngine; class Q_QML_PRIVATE_EXPORT QQmlProfilerService : public QQmlDebugService { public: enum Message { Event, RangeStart, RangeData, RangeLocation, RangeEnd, Complete, // end of transmission MaximumMessage }; enum EventType { FramePaint, Mouse, Key, AnimationFrame, EndTrace, StartTrace, MaximumEventType }; enum RangeType { Painting, Compiling, Creating, Binding, //running a binding HandlingSignal, //running a signal handler MaximumRangeType }; enum BindingType { QmlBinding, V8Binding, V4Binding, MaximumBindingType }; static void initialize(); static bool startProfiling(); static bool stopProfiling(); static void sendStartedProfilingMessage(); static void addEvent(EventType); static void animationFrame(qint64); static void sendProfilingData(); QQmlProfilerService(); ~QQmlProfilerService(); protected: virtual void stateAboutToBeChanged(State state); virtual void messageReceived(const QByteArray &); private: bool startProfilingImpl(); bool stopProfilingImpl(); void sendStartedProfilingMessageImpl(); void addEventImpl(EventType); void animationFrameImpl(qint64); void startRange(RangeType, BindingType bindingType = QmlBinding); void rangeData(RangeType, const QString &); void rangeData(RangeType, const QUrl &); void rangeLocation(RangeType, const QString &, int, int); void rangeLocation(RangeType, const QUrl &, int, int); void endRange(RangeType); bool profilingEnabled(); void setProfilingEnabled(bool enable); void sendMessages(); void processMessage(const QQmlProfilerData &); private: QElapsedTimer m_timer; bool m_enabled; QVector m_data; QMutex m_dataMutex; QMutex m_initializeMutex; QWaitCondition m_initializeCondition; static QQmlProfilerService *instance; friend struct QQmlBindingProfiler; friend struct QQmlHandlingSignalProfiler; friend struct QQmlObjectCreatingProfiler; friend struct QQmlCompilingProfiler; }; // // RAII helper structs // struct QQmlBindingProfiler { QQmlBindingProfiler(const QString &url, int line, int column, QQmlProfilerService::BindingType bindingType) { QQmlProfilerService *instance = QQmlProfilerService::instance; enabled = instance ? instance->profilingEnabled() : false; if (enabled) { instance->startRange(QQmlProfilerService::Binding, bindingType); instance->rangeLocation(QQmlProfilerService::Binding, url, line, column); } } ~QQmlBindingProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::Binding); } bool enabled; }; struct QQmlHandlingSignalProfiler { QQmlHandlingSignalProfiler(QQmlBoundSignalExpression *expression) { enabled = QQmlProfilerService::instance ? QQmlProfilerService::instance->profilingEnabled() : false; if (enabled) { QQmlProfilerService *service = QQmlProfilerService::instance; service->startRange(QQmlProfilerService::HandlingSignal); service->rangeLocation(QQmlProfilerService::HandlingSignal, expression->sourceFile(), expression->lineNumber(), expression->columnNumber()); } } ~QQmlHandlingSignalProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::HandlingSignal); } bool enabled; }; struct QQmlObjectCreatingProfiler { QQmlObjectCreatingProfiler() { enabled = QQmlProfilerService::instance ? QQmlProfilerService::instance->profilingEnabled() : false; if (enabled) { QQmlProfilerService *service = QQmlProfilerService::instance; service->startRange(QQmlProfilerService::Creating); } } void setTypeName(const QString &typeName) { Q_ASSERT_X(enabled, Q_FUNC_INFO, "method called although profiler is not enabled."); QQmlProfilerService::instance->rangeData(QQmlProfilerService::Creating, typeName); } void setLocation(const QUrl &url, int line, int column) { Q_ASSERT_X(enabled, Q_FUNC_INFO, "method called although profiler is not enabled."); if (enabled) QQmlProfilerService::instance->rangeLocation( QQmlProfilerService::Creating, url, line, column); } ~QQmlObjectCreatingProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::Creating); } bool enabled; }; struct QQmlCompilingProfiler { QQmlCompilingProfiler(const QString &name) { QQmlProfilerService *instance = QQmlProfilerService::instance; enabled = instance ? instance->profilingEnabled() : false; if (enabled) { instance->startRange(QQmlProfilerService::Compiling); instance->rangeLocation(QQmlProfilerService::Compiling, name, 1, 1); instance->rangeData(QQmlProfilerService::Compiling, name); } } ~QQmlCompilingProfiler() { if (enabled) QQmlProfilerService::instance->endRange(QQmlProfilerService::Compiling); } bool enabled; }; QT_END_NAMESPACE #endif // QQMLPROFILERSERVICE_P_H