/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ ** 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 The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qmlprofilerapplication.h" #include "constants.h" #include #include #include #include #include #include #include #include #include static const char commandTextC[] = "The following commands are available:\n" "'r', 'record'\n" " Switch recording on or off.\n" "'o [file]', 'output [file]'\n" " Output profiling data to . If no \n" " parameter is given, output to whatever was given\n" " with --output, or standard output.\n" "'c', 'clear'\n" " Clear profiling data recorded so far from memory.\n" "'f [file]', 'flush [file]'\n" " Stop recording if it is running, then output the\n" " data, and finally clear it from memory.\n" "'q', 'quit'\n" " Terminate the program if started from qmlprofiler,\n" " and qmlprofiler itself."; static const char *features[] = { "javascript", "memory", "pixmapcache", "scenegraph", "animations", "painting", "compiling", "creating", "binding", "handlingsignal", "inputevents", "debugmessages" }; Q_STATIC_ASSERT(sizeof(features) == MaximumProfileFeature * sizeof(char *)); QmlProfilerApplication::QmlProfilerApplication(int &argc, char **argv) : QCoreApplication(argc, argv), m_runMode(LaunchMode), m_process(nullptr), m_hostName(QLatin1String("127.0.0.1")), m_port(0), m_pendingRequest(REQUEST_NONE), m_verbose(false), m_recording(true), m_interactive(false), m_connectionAttempts(0) { m_connection.reset(new QQmlDebugConnection); m_profilerData.reset(new QmlProfilerData); m_qmlProfilerClient.reset(new QmlProfilerClient(m_connection.data(), m_profilerData.data())); m_connectTimer.setInterval(1000); connect(&m_connectTimer, &QTimer::timeout, this, &QmlProfilerApplication::tryToConnect); connect(m_connection.data(), &QQmlDebugConnection::connected, this, &QmlProfilerApplication::connected); connect(m_connection.data(), &QQmlDebugConnection::disconnected, this, &QmlProfilerApplication::disconnected); connect(m_qmlProfilerClient.data(), &QmlProfilerClient::enabledChanged, this, &QmlProfilerApplication::traceClientEnabledChanged); connect(m_qmlProfilerClient.data(), &QmlProfilerClient::traceStarted, this, &QmlProfilerApplication::notifyTraceStarted); connect(m_qmlProfilerClient.data(), &QmlProfilerClient::error, this, &QmlProfilerApplication::logError); connect(m_profilerData.data(), &QmlProfilerData::error, this, &QmlProfilerApplication::logError); connect(m_profilerData.data(), &QmlProfilerData::dataReady, this, &QmlProfilerApplication::traceFinished); } QmlProfilerApplication::~QmlProfilerApplication() { if (!m_process) return; logStatus("Terminating process ..."); m_process->disconnect(); m_process->terminate(); if (!m_process->waitForFinished(1000)) { logStatus("Killing process ..."); m_process->kill(); } delete m_process; } void QmlProfilerApplication::parseArguments() { setApplicationName(QLatin1String("qmlprofiler")); setApplicationVersion(QLatin1String(qVersion())); QCommandLineParser parser; parser.setSingleDashWordOptionMode(QCommandLineParser::ParseAsLongOptions); parser.setOptionsAfterPositionalArgumentsMode(QCommandLineParser::ParseAsPositionalArguments); parser.setApplicationDescription(QChar::LineFeed + tr( "The QML Profiler retrieves QML tracing data from an application. The data\n" "collected can then be visualized in Qt Creator. The application to be profiled\n" "has to enable QML debugging. See the Qt Creator documentation on how to do\n" "this for different Qt versions.")); QCommandLineOption attach(QStringList() << QLatin1String("a") << QLatin1String("attach"), tr("Attach to an application already running on , " "instead of starting it locally."), QLatin1String("hostname")); parser.addOption(attach); QCommandLineOption port(QStringList() << QLatin1String("p") << QLatin1String("port"), tr("Connect to the TCP port . The default is 3768."), QLatin1String("port"), QLatin1String("3768")); parser.addOption(port); QCommandLineOption output(QStringList() << QLatin1String("o") << QLatin1String("output"), tr("Save tracing data in . By default the data is sent to the " "standard output."), QLatin1String("file"), QString()); parser.addOption(output); QCommandLineOption record(QLatin1String("record"), tr("If set to 'off', don't immediately start recording data when the " "QML engine starts, but instead either start the recording " "interactively or with the JavaScript console.profile() function. " "By default the recording starts immediately."), QLatin1String("on|off"), QLatin1String("on")); parser.addOption(record); QStringList featureList; for (int i = 0; i < MaximumProfileFeature; ++i) featureList << QLatin1String(features[i]); QCommandLineOption include(QLatin1String("include"), tr("Comma-separated list of features to record. By default all " "features supported by the QML engine are recorded. If --include " "is specified, only the given features will be recorded. " "The following features are unserstood by qmlprofiler: %1").arg( featureList.join(", ")), QLatin1String("feature,...")); parser.addOption(include); QCommandLineOption exclude(QLatin1String("exclude"), tr("Comma-separated list of features to exclude when recording. By " "default all features supported by the QML engine are recorded. " "See --include for the features understood by qmlprofiler."), QLatin1String("feature,...")); parser.addOption(exclude); QCommandLineOption interactive(QLatin1String("interactive"), tr("Manually control the recording from the command line. The " "profiler will not terminate itself when the application " "does so in this case.") + QChar::Space + tr(commandTextC)); parser.addOption(interactive); QCommandLineOption verbose(QStringList() << QLatin1String("verbose"), tr("Print debugging output.")); parser.addOption(verbose); parser.addHelpOption(); parser.addVersionOption(); parser.addPositionalArgument(QLatin1String("program"), tr("The program to be started and profiled."), QLatin1String("[program]")); parser.addPositionalArgument(QLatin1String("parameters"), tr("Parameters for the program to be started."), QLatin1String("[parameters...]")); parser.process(*this); if (parser.isSet(attach)) { m_hostName = parser.value(attach); m_runMode = AttachMode; m_port = 3768; } if (parser.isSet(port)) { bool isNumber; m_port = parser.value(port).toUShort(&isNumber); if (!isNumber) { logError(tr("'%1' is not a valid port.").arg(parser.value(port))); parser.showHelp(1); } } else if (m_port == 0) { QTemporaryFile file; if (file.open()) m_socketFile = file.fileName(); } m_outputFile = parser.value(output); m_recording = (parser.value(record) == QLatin1String("on")); m_interactive = parser.isSet(interactive); quint64 features = std::numeric_limits::max(); if (parser.isSet(include)) { if (parser.isSet(exclude)) { logError(tr("qmlprofiler can only process either --include or --exclude, not both.")); parser.showHelp(4); } features = parseFeatures(featureList, parser.value(include), false); } if (parser.isSet(exclude)) features = parseFeatures(featureList, parser.value(exclude), true); if (features == 0) parser.showHelp(4); m_qmlProfilerClient->setRequestedFeatures(features); if (parser.isSet(verbose)) m_verbose = true; m_programArguments = parser.positionalArguments(); if (!m_programArguments.isEmpty()) m_programPath = m_programArguments.takeFirst(); if (m_runMode == LaunchMode && m_programPath.isEmpty()) { logError(tr("You have to specify either --attach or a program to start.")); parser.showHelp(2); } if (m_runMode == AttachMode && !m_programPath.isEmpty()) { logError(tr("--attach cannot be used when starting a program.")); parser.showHelp(3); } } int QmlProfilerApplication::exec() { QTimer::singleShot(0, this, &QmlProfilerApplication::run); return QCoreApplication::exec(); } bool QmlProfilerApplication::isInteractive() const { return m_interactive; } quint64 QmlProfilerApplication::parseFeatures(const QStringList &featureList, const QString &values, bool exclude) { quint64 features = exclude ? std::numeric_limits::max() : 0; const QStringList givenFeatures = values.split(QLatin1Char(',')); for (const QString &f : givenFeatures) { int index = featureList.indexOf(f); if (index < 0) { logError(tr("Unknown feature '%1'").arg(f)); return 0; } quint64 flag = static_cast(1) << index; features = (exclude ? (features ^ flag) : (features | flag)); } if (features == 0) { logError(exclude ? tr("No features remaining to record after processing --exclude.") : tr("No features specified for --include.")); } return features; } void QmlProfilerApplication::flush() { if (m_recording) { m_pendingRequest = REQUEST_FLUSH; m_qmlProfilerClient->sendRecordingStatus(false); } else { if (m_profilerData->save(m_interactiveOutputFile)) { m_profilerData->clear(); if (!m_interactiveOutputFile.isEmpty()) prompt(tr("Data written to %1.").arg(m_interactiveOutputFile)); else prompt(); } else { prompt(tr("Saving failed.")); } m_interactiveOutputFile.clear(); m_pendingRequest = REQUEST_NONE; } } void QmlProfilerApplication::output() { if (m_profilerData->save(m_interactiveOutputFile)) { if (!m_interactiveOutputFile.isEmpty()) prompt(tr("Data written to %1.").arg(m_interactiveOutputFile)); else prompt(); } else { prompt(tr("Saving failed")); } m_interactiveOutputFile.clear(); m_pendingRequest = REQUEST_NONE; } bool QmlProfilerApplication::checkOutputFile(PendingRequest pending) { if (m_interactiveOutputFile.isEmpty()) return true; QFileInfo file(m_interactiveOutputFile); if (file.exists()) { if (!file.isFile()) { prompt(tr("Cannot overwrite %1.").arg(m_interactiveOutputFile)); m_interactiveOutputFile.clear(); } else { prompt(tr("%1 exists. Overwrite (y/n)?").arg(m_interactiveOutputFile)); m_pendingRequest = pending; } return false; } else { return true; } } void QmlProfilerApplication::userCommand(const QString &command) { auto args = command.splitRef(QChar::Space, QString::SkipEmptyParts); if (args.isEmpty()) { prompt(); return; } QByteArray cmd = args.takeFirst().trimmed().toLatin1(); if (m_pendingRequest == REQUEST_QUIT) { if (cmd == Constants::CMD_YES || cmd == Constants::CMD_YES2) { quit(); } else if (cmd == Constants::CMD_NO || cmd == Constants::CMD_NO2) { m_pendingRequest = REQUEST_NONE; prompt(); } else { prompt(tr("The application is still generating data. Really quit (y/n)?")); } return; } if (m_pendingRequest == REQUEST_OUTPUT_FILE || m_pendingRequest == REQUEST_FLUSH_FILE) { if (cmd == Constants::CMD_YES || cmd == Constants::CMD_YES2) { if (m_pendingRequest == REQUEST_OUTPUT_FILE) output(); else flush(); } else if (cmd == Constants::CMD_NO || cmd == Constants::CMD_NO2) { m_pendingRequest = REQUEST_NONE; m_interactiveOutputFile.clear(); prompt(); } else { prompt(tr("%1 exists. Overwrite (y/n)?")); } return; } if (cmd == Constants::CMD_RECORD || cmd == Constants::CMD_RECORD2) { m_pendingRequest = REQUEST_TOGGLE_RECORDING; m_qmlProfilerClient->sendRecordingStatus(!m_recording); } else if (cmd == Constants::CMD_QUIT || cmd == Constants::CMD_QUIT2) { m_pendingRequest = REQUEST_QUIT; if (m_recording) { prompt(tr("The application is still generating data. Really quit (y/n)?")); } else if (!m_profilerData->isEmpty()) { prompt(tr("There is still trace data in memory. Really quit (y/n)?")); } else { quit(); } } else if (cmd == Constants::CMD_OUTPUT || cmd == Constants::CMD_OUTPUT2) { if (m_recording) { prompt(tr("Cannot output while recording data.")); } else if (m_profilerData->isEmpty()) { prompt(tr("No data was recorded so far.")); } else { m_interactiveOutputFile = args.length() > 0 ? args.at(0).toString() : m_outputFile; if (checkOutputFile(REQUEST_OUTPUT_FILE)) output(); } } else if (cmd == Constants::CMD_CLEAR || cmd == Constants::CMD_CLEAR2) { if (m_recording) { prompt(tr("Cannot clear data while recording.")); } else if (m_profilerData->isEmpty()) { prompt(tr("No data was recorded so far.")); } else { m_profilerData->clear(); prompt(tr("Trace data cleared.")); } } else if (cmd == Constants::CMD_FLUSH || cmd == Constants::CMD_FLUSH2) { if (!m_recording && m_profilerData->isEmpty()) { prompt(tr("No data was recorded so far.")); } else { m_interactiveOutputFile = args.length() > 0 ? args.at(0).toString() : m_outputFile; if (checkOutputFile(REQUEST_FLUSH_FILE)) flush(); } } else { prompt(tr(commandTextC)); } } void QmlProfilerApplication::notifyTraceStarted() { // Synchronize to server state. It doesn't hurt to do this multiple times in a row for // different traces. There is no symmetric event to "Complete" after all. m_recording = true; if (m_pendingRequest == REQUEST_TOGGLE_RECORDING) { m_pendingRequest = REQUEST_NONE; prompt(tr("Recording started")); } else { prompt(tr("Application started recording"), false); } } void QmlProfilerApplication::outputData() { if (!m_profilerData->isEmpty()) { m_profilerData->save(m_outputFile); m_profilerData->clear(); } } void QmlProfilerApplication::run() { if (m_runMode == LaunchMode) { if (!m_socketFile.isEmpty()) { logStatus(QString::fromLatin1("Listening on %1 ...").arg(m_socketFile)); m_connection->startLocalServer(m_socketFile); } m_process = new QProcess(this); QStringList arguments; arguments << QString::fromLatin1("-qmljsdebugger=%1:%2,block,services:CanvasFrameRate") .arg(QLatin1String(m_socketFile.isEmpty() ? "port" : "file")) .arg(m_socketFile.isEmpty() ? QString::number(m_port) : m_socketFile); arguments << m_programArguments; m_process->setProcessChannelMode(QProcess::MergedChannels); connect(m_process, &QIODevice::readyRead, this, &QmlProfilerApplication::processHasOutput); connect(m_process, static_cast(&QProcess::finished), this, [this](int){ processFinished(); }); logStatus(QString("Starting '%1 %2' ...").arg(m_programPath, arguments.join(QLatin1Char(' ')))); m_process->start(m_programPath, arguments); if (!m_process->waitForStarted()) { logError(QString("Could not run '%1': %2").arg(m_programPath, m_process->errorString())); exit(1); } } m_connectTimer.start(); } void QmlProfilerApplication::tryToConnect() { Q_ASSERT(!m_connection->isConnected()); ++ m_connectionAttempts; if (!m_verbose && !(m_connectionAttempts % 5)) {// print every 5 seconds if (m_verbose) { if (m_socketFile.isEmpty()) logError(QString::fromLatin1("Could not connect to %1:%2 for %3 seconds ...") .arg(m_hostName).arg(m_port).arg(m_connectionAttempts)); else logError(QString::fromLatin1("No connection received on %1 for %2 seconds ...") .arg(m_socketFile).arg(m_connectionAttempts)); } } if (m_socketFile.isEmpty()) { logStatus(QString::fromLatin1("Connecting to %1:%2 ...").arg(m_hostName).arg(m_port)); m_connection->connectToHost(m_hostName, m_port); } } void QmlProfilerApplication::connected() { m_connectTimer.stop(); QString endpoint = m_socketFile.isEmpty() ? QString::fromLatin1("%1:%2").arg(m_hostName).arg(m_port) : m_socketFile; prompt(tr("Connected to %1. Wait for profile data or type a command (type 'help' to show list " "of commands).\nRecording Status: %2") .arg(endpoint).arg(m_recording ? tr("on") : tr("off"))); } void QmlProfilerApplication::disconnected() { if (m_runMode == AttachMode) { int exitCode = 0; if (m_recording) { logError("Connection dropped while recording, last trace is damaged!"); exitCode = 2; } if (!m_interactive ) exit(exitCode); else m_qmlProfilerClient->clearAll(); } } void QmlProfilerApplication::processHasOutput() { Q_ASSERT(m_process); while (m_process->bytesAvailable()) { QTextStream out(stderr); out << m_process->readAll(); } } void QmlProfilerApplication::processFinished() { Q_ASSERT(m_process); int exitCode = 0; if (m_process->exitStatus() == QProcess::NormalExit) { logStatus(QString("Process exited (%1).").arg(m_process->exitCode())); if (m_recording) { logError("Process exited while recording, last trace is damaged!"); exitCode = 2; } } else { logError("Process crashed!"); exitCode = 3; } if (!m_interactive) exit(exitCode); else m_qmlProfilerClient->clearAll(); } void QmlProfilerApplication::traceClientEnabledChanged(bool enabled) { if (enabled) { logStatus("Trace client is attached."); // blocked server is waiting for recording message from both clients // once the last one is connected, both messages should be sent m_qmlProfilerClient->setRecording(m_recording); } } void QmlProfilerApplication::traceFinished() { m_recording = false; // only on "Complete" we know that the trace is really finished. if (m_pendingRequest == REQUEST_FLUSH) { flush(); } else if (m_pendingRequest == REQUEST_TOGGLE_RECORDING) { m_pendingRequest = REQUEST_NONE; prompt(tr("Recording stopped.")); } else { prompt(tr("Application stopped recording."), false); } m_qmlProfilerClient->clearEvents(); } void QmlProfilerApplication::prompt(const QString &line, bool ready) { if (m_interactive) { QTextStream err(stderr); if (!line.isEmpty()) err << line << endl; err << QLatin1String("> "); if (ready) emit readyForCommand(); } } void QmlProfilerApplication::logError(const QString &error) { QTextStream err(stderr); err << "Error: " << error << endl; } void QmlProfilerApplication::logStatus(const QString &status) { if (!m_verbose) return; QTextStream err(stderr); err << status << endl; }