/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** ** GNU Lesser General Public License Usage ** ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** Other Usage ** ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include "qmlprofilerengine.h" #include "codaqmlprofilerrunner.h" #include "localqmlprofilerrunner.h" #include "remotelinuxqmlprofilerrunner.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Analyzer; using namespace ProjectExplorer; namespace QmlProfiler { namespace Internal { // // QmlProfilerEnginePrivate // class QmlProfilerEngine::QmlProfilerEnginePrivate { public: QmlProfilerEnginePrivate(QmlProfilerEngine *qq) : q(qq), m_runner(0) {} ~QmlProfilerEnginePrivate() { delete m_runner; } bool attach(const QString &address, uint port); static AbstractQmlProfilerRunner *createRunner(ProjectExplorer::RunConfiguration *runConfiguration, QObject *parent); QmlProfilerEngine *q; QmlProfilerStateManager *m_profilerState; AbstractQmlProfilerRunner *m_runner; QTimer m_noDebugOutputTimer; QmlDebug::QmlOutputParser m_outputParser; }; AbstractQmlProfilerRunner * QmlProfilerEngine::QmlProfilerEnginePrivate::createRunner(ProjectExplorer::RunConfiguration *runConfiguration, QObject *parent) { AbstractQmlProfilerRunner *runner = 0; if (!runConfiguration) // attaching return 0; if (QmlProjectManager::QmlProjectRunConfiguration *rc1 = qobject_cast(runConfiguration)) { // This is a "plain" .qmlproject. LocalQmlProfilerRunner::Configuration conf; conf.executable = rc1->observerPath(); conf.executableArguments = rc1->viewerArguments(); conf.workingDirectory = rc1->workingDirectory(); conf.environment = rc1->environment(); conf.port = rc1->debuggerAspect()->qmlDebugServerPort(); runner = new LocalQmlProfilerRunner(conf, parent); } else if (LocalApplicationRunConfiguration *rc2 = qobject_cast(runConfiguration)) { // FIXME: Check. LocalQmlProfilerRunner::Configuration conf; conf.executable = rc2->executable(); conf.executableArguments = rc2->commandLineArguments(); conf.workingDirectory = rc2->workingDirectory(); conf.environment = rc2->environment(); conf.port = rc2->debuggerAspect()->qmlDebugServerPort(); runner = new LocalQmlProfilerRunner(conf, parent); } else if (Qt4ProjectManager::S60DeviceRunConfiguration *s60Config = qobject_cast(runConfiguration)) { runner = new CodaQmlProfilerRunner(s60Config, parent); } else if (RemoteLinux::RemoteLinuxRunConfiguration *rmConfig = qobject_cast(runConfiguration)) { runner = new RemoteLinuxQmlProfilerRunner(rmConfig, parent); } else { QTC_CHECK(false); } return runner; } // // QmlProfilerEngine // QmlProfilerEngine::QmlProfilerEngine(IAnalyzerTool *tool, const Analyzer::AnalyzerStartParameters &sp, ProjectExplorer::RunConfiguration *runConfiguration) : IAnalyzerEngine(tool, sp, runConfiguration) , d(new QmlProfilerEnginePrivate(this)) { d->m_profilerState = 0; // Only wait 4 seconds for the 'Waiting for connection' on application ouput, then just try to connect // (application output might be redirected / blocked) d->m_noDebugOutputTimer.setSingleShot(true); d->m_noDebugOutputTimer.setInterval(4000); connect(&d->m_noDebugOutputTimer, SIGNAL(timeout()), this, SLOT(processIsRunning())); d->m_outputParser.setNoOutputText(ApplicationLauncher::msgWinCannotRetrieveDebuggingOutput()); connect(&d->m_outputParser, SIGNAL(waitingForConnectionOnPort(quint16)), this, SLOT(processIsRunning(quint16))); connect(&d->m_outputParser, SIGNAL(waitingForConnectionViaOst()), this, SLOT(processIsRunning())); connect(&d->m_outputParser, SIGNAL(noOutputMessage()), this, SLOT(processIsRunning())); connect(&d->m_outputParser, SIGNAL(errorMessage(QString)), this, SLOT(wrongSetupMessageBox(QString))); } QmlProfilerEngine::~QmlProfilerEngine() { if (d->m_profilerState && d->m_profilerState->currentState() == QmlProfilerStateManager::AppRunning) stop(); delete d; } bool QmlProfilerEngine::start() { QTC_ASSERT(d->m_profilerState, return false); if (d->m_runner) { delete d->m_runner; d->m_runner = 0; } d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppStarting); if (QmlProjectManager::QmlProjectRunConfiguration *rc = qobject_cast(runConfiguration())) { if (rc->observerPath().isEmpty()) { QmlProjectManager::QmlProjectPlugin::showQmlObserverToolWarning(); d->m_profilerState->setCurrentState(QmlProfilerStateManager::Idle); AnalyzerManager::stopTool(); return false; } } d->m_runner = QmlProfilerEnginePrivate::createRunner(runConfiguration(), this); if (LocalQmlProfilerRunner *qmlRunner = qobject_cast(d->m_runner)) { if (!qmlRunner->hasExecutable()) { showNonmodalWarning(tr("No executable file to launch.")); d->m_profilerState->setCurrentState(QmlProfilerStateManager::Idle); AnalyzerManager::stopTool(); return false; } } if (d->m_runner) { connect(d->m_runner, SIGNAL(stopped()), this, SLOT(processEnded())); connect(d->m_runner, SIGNAL(appendMessage(QString,Utils::OutputFormat)), this, SLOT(logApplicationMessage(QString,Utils::OutputFormat))); d->m_runner->start(); d->m_noDebugOutputTimer.start(); } else { emit processRunning(startParameters().connParams.port); } d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppRunning); emit starting(this); return true; } void QmlProfilerEngine::stop() { QTC_ASSERT(d->m_profilerState, return); switch (d->m_profilerState->currentState()) { case QmlProfilerStateManager::AppRunning : { d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppStopRequested); break; } case QmlProfilerStateManager::AppReadyToStop : { cancelProcess(); break; } case QmlProfilerStateManager::AppDying : // valid, but no further action is needed break; default: qDebug() << tr("Unexpected engine stop from state %1 in %2:%3").arg(d->m_profilerState->currentStateAsString(), QString(__FILE__), QString::number(__LINE__)); break; } } void QmlProfilerEngine::processEnded() { QTC_ASSERT(d->m_profilerState, return); switch (d->m_profilerState->currentState()) { case QmlProfilerStateManager::AppRunning : { d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppDying); AnalyzerManager::stopTool(); emit finished(); break; } case QmlProfilerStateManager::AppStopped : case QmlProfilerStateManager::AppKilled : d->m_profilerState->setCurrentState(QmlProfilerStateManager::Idle); break; default: qDebug() << tr("Process died unexpectedly from state %1 in %2:%3").arg(d->m_profilerState->currentStateAsString(), QString(__FILE__), QString::number(__LINE__)); break; } } void QmlProfilerEngine::cancelProcess() { QTC_ASSERT(d->m_profilerState, return); switch (d->m_profilerState->currentState()) { case QmlProfilerStateManager::AppReadyToStop : { d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppStopped); break; } case QmlProfilerStateManager::AppRunning : { d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppDying); break; } default: { qDebug() << tr("Unexpected process termination requested with state %1 in %2:%3").arg(d->m_profilerState->currentStateAsString(), QString(__FILE__), QString::number(__LINE__)); return; } } if (d->m_runner) d->m_runner->stop(); emit finished(); } void QmlProfilerEngine::logApplicationMessage(const QString &msg, Utils::OutputFormat format) { emit outputReceived(msg, format); d->m_outputParser.processOutput(msg); } void QmlProfilerEngine::wrongSetupMessageBox(const QString &errorMessage) { QMessageBox *infoBox = new QMessageBox(Core::ICore::mainWindow()); infoBox->setIcon(QMessageBox::Critical); infoBox->setWindowTitle(tr("Qt Creator")); //: %1 is detailed error message infoBox->setText(tr("Could not connect to the in-process QML debugger:\n%1") .arg(errorMessage)); infoBox->setStandardButtons(QMessageBox::Ok | QMessageBox::Help); infoBox->setDefaultButton(QMessageBox::Ok); infoBox->setModal(true); connect(infoBox, SIGNAL(finished(int)), this, SLOT(wrongSetupMessageBoxFinished(int))); infoBox->show(); // KILL d->m_profilerState->setCurrentState(QmlProfilerStateManager::AppDying); AnalyzerManager::stopTool(); emit finished(); } void QmlProfilerEngine::wrongSetupMessageBoxFinished(int button) { if (button == QMessageBox::Help) { Core::HelpManager *helpManager = Core::HelpManager::instance(); helpManager->handleHelpRequest("qthelp://com.nokia.qtcreator/doc/creator-debugging-qml.html" "#setting-up-qml-debugging"); } } void QmlProfilerEngine::showNonmodalWarning(const QString &warningMsg) { QMessageBox *noExecWarning = new QMessageBox(Core::ICore::mainWindow()); noExecWarning->setIcon(QMessageBox::Warning); noExecWarning->setWindowTitle(tr("QML Profiler")); noExecWarning->setText(warningMsg); noExecWarning->setStandardButtons(QMessageBox::Ok); noExecWarning->setDefaultButton(QMessageBox::Ok); noExecWarning->setModal(false); noExecWarning->show(); } void QmlProfilerEngine::processIsRunning(quint16 port) { d->m_noDebugOutputTimer.stop(); QTC_ASSERT(port == 0 || port == d->m_runner->debugPort(), qWarning() << "Port " << port << "from application output does not match" << startParameters().connParams.port << "from start parameters."); if (port > 0) emit processRunning(port); else emit processRunning(d->m_runner->debugPort()); } //////////////////////////////////////////////////////////////// // Profiler State void QmlProfilerEngine::registerProfilerStateManager( QmlProfilerStateManager *profilerState ) { // disconnect old if (d->m_profilerState) { disconnect(d->m_profilerState, SIGNAL(stateChanged()), this, SLOT(profilerStateChanged())); } d->m_profilerState = profilerState; // connect if (d->m_profilerState) { connect(d->m_profilerState, SIGNAL(stateChanged()), this, SLOT(profilerStateChanged())); } } void QmlProfilerEngine::profilerStateChanged() { switch (d->m_profilerState->currentState()) { case QmlProfilerStateManager::AppReadyToStop : { if (d->m_runner) cancelProcess(); break; } case QmlProfilerStateManager::Idle : { // When all the profiling is done, delete the profiler runner // (a new one will be created at start) if (d->m_runner) { delete d->m_runner; d->m_runner = 0; } break; } default: break; } } } // namespace Internal } // namespace QmlProfiler