/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** No Commercial Usage ** ** This file contains pre-release code and may not be distributed. ** You may use this file in accordance with the terms and conditions ** contained in the Technology Preview License Agreement accompanying ** this package. ** ** 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, 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. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include "s60devicerunconfiguration.h" #include "s60devicerunconfigurationwidget.h" #include "s60deployconfiguration.h" #include "qt4project.h" #include "qt4target.h" #include "s60manager.h" #include "s60devices.h" #include "s60runconfigbluetoothstarter.h" #include "qt4projectmanagerconstants.h" #include "qtoutputformatter.h" #include "qt4symbiantarget.h" #include "tcftrkdevice.h" #include "tcftrkmessage.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ProjectExplorer; using namespace Qt4ProjectManager; using namespace Qt4ProjectManager::Internal; using namespace tcftrk; namespace { const char * const S60_DEVICE_RC_ID("Qt4ProjectManager.S60DeviceRunConfiguration"); const char * const S60_DEVICE_RC_PREFIX("Qt4ProjectManager.S60DeviceRunConfiguration."); const char * const PRO_FILE_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.ProFile"); const char * const COMMUNICATION_TYPE_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.CommunicationType"); const char * const COMMAND_LINE_ARGUMENTS_KEY("Qt4ProjectManager.S60DeviceRunConfiguration.CommandLineArguments"); const int PROGRESS_MAX = 200; enum { debug = 0 }; // Format information about a file static inline QString msgListFile(const QString &f) { QString rc; const QFileInfo fi(f); QTextStream str(&rc); if (fi.exists()) str << fi.size() << ' ' << fi.lastModified().toString(Qt::ISODate) << ' ' << QDir::toNativeSeparators(fi.absoluteFilePath()); else str << " " << QDir::toNativeSeparators(fi.absoluteFilePath()); return rc; } QString pathFromId(const QString &id) { if (!id.startsWith(QLatin1String(S60_DEVICE_RC_PREFIX))) return QString(); return id.mid(QString::fromLatin1(S60_DEVICE_RC_PREFIX).size()); } QString pathToId(const QString &path) { return QString::fromLatin1(S60_DEVICE_RC_PREFIX) + path; } bool isProcessRunning(const TcfTrkCommandResult &result, const QString &processName) { if (result.values.size() && result.values.at(0).type() == JsonValue::Array) { foreach(const JsonValue &threadValue, result.values.at(0).children()) { for (int i = threadValue.children().count()-1; i >= 0; --i) { //Usually our process will be near the end of the list const JsonValue &value(threadValue.childAt(i)); if (value.hasName("p_name") && QString::fromLatin1(value.data()).startsWith(processName, Qt::CaseInsensitive)) return true; } } } return false; } } // anonymous namespace // ======== S60DeviceRunConfiguration S60DeviceRunConfiguration::S60DeviceRunConfiguration(Qt4BaseTarget *parent, const QString &proFilePath) : RunConfiguration(parent, QLatin1String(S60_DEVICE_RC_ID)), m_proFilePath(proFilePath), m_validParse(parent->qt4Project()->validParse(proFilePath)) { ctor(); } S60DeviceRunConfiguration::S60DeviceRunConfiguration(Qt4BaseTarget *target, S60DeviceRunConfiguration *source) : RunConfiguration(target, source), m_proFilePath(source->m_proFilePath), m_commandLineArguments(source->m_commandLineArguments), m_validParse(source->m_validParse) { ctor(); } void S60DeviceRunConfiguration::ctor() { if (!m_proFilePath.isEmpty()) //: S60 device runconfiguration default display name, %1 is base pro-File name setDefaultDisplayName(tr("%1 on Symbian Device").arg(QFileInfo(m_proFilePath).completeBaseName())); else //: S60 device runconfiguration default display name (no profile set) setDefaultDisplayName(tr("Run on Symbian device")); Qt4Project *pro = qt4Target()->qt4Project(); connect(pro, SIGNAL(proFileUpdated(Qt4ProjectManager::Internal::Qt4ProFileNode*,bool)), this, SLOT(proFileUpdate(Qt4ProjectManager::Internal::Qt4ProFileNode*,bool))); connect(pro, SIGNAL(proFileInvalidated(Qt4ProjectManager::Internal::Qt4ProFileNode *)), this, SLOT(proFileInvalidated(Qt4ProjectManager::Internal::Qt4ProFileNode *))); } void S60DeviceRunConfiguration::handleParserState(bool success) { bool enabled = isEnabled(); m_validParse = success; if (enabled != isEnabled()) emit isEnabledChanged(!enabled); } void S60DeviceRunConfiguration::proFileInvalidated(Qt4ProjectManager::Internal::Qt4ProFileNode *pro) { if (m_proFilePath != pro->path()) return; handleParserState(false); } void S60DeviceRunConfiguration::proFileUpdate(Qt4ProjectManager::Internal::Qt4ProFileNode *pro, bool success) { if (m_proFilePath != pro->path()) return; handleParserState(success); emit targetInformationChanged(); } S60DeviceRunConfiguration::~S60DeviceRunConfiguration() { } Qt4SymbianTarget *S60DeviceRunConfiguration::qt4Target() const { return static_cast(target()); } ProjectExplorer::ToolChainType S60DeviceRunConfiguration::toolChainType( ProjectExplorer::BuildConfiguration *configuration) const { if (Qt4BuildConfiguration *bc = qobject_cast(configuration)) return bc->toolChainType(); return ProjectExplorer::ToolChain_INVALID; } ProjectExplorer::ToolChainType S60DeviceRunConfiguration::toolChainType() const { if (Qt4BuildConfiguration *bc = qobject_cast(target()->activeBuildConfiguration())) return bc->toolChainType(); return ProjectExplorer::ToolChain_INVALID; } bool S60DeviceRunConfiguration::isEnabled(ProjectExplorer::BuildConfiguration *configuration) const { if (!m_validParse) return false; const Qt4BuildConfiguration *qt4bc = static_cast(configuration); switch (qt4bc->toolChainType()) { case ProjectExplorer::ToolChain_GCCE: case ProjectExplorer::ToolChain_RVCT2_ARMV5: case ProjectExplorer::ToolChain_RVCT2_ARMV6: case ProjectExplorer::ToolChain_RVCT4_ARMV5: case ProjectExplorer::ToolChain_RVCT4_ARMV6: case ProjectExplorer::ToolChain_GCCE_GNUPOC: case ProjectExplorer::ToolChain_RVCT_ARMV5_GNUPOC: return true; default: break; } return false; } QWidget *S60DeviceRunConfiguration::createConfigurationWidget() { return new S60DeviceRunConfigurationWidget(this); } ProjectExplorer::OutputFormatter *S60DeviceRunConfiguration::createOutputFormatter() const { return new QtOutputFormatter(qt4Target()->qt4Project()); } QVariantMap S60DeviceRunConfiguration::toMap() const { QVariantMap map = ProjectExplorer::RunConfiguration::toMap(); const QDir projectDir = QDir(target()->project()->projectDirectory()); map.insert(QLatin1String(PRO_FILE_KEY), projectDir.relativeFilePath(m_proFilePath)); map.insert(QLatin1String(COMMAND_LINE_ARGUMENTS_KEY), m_commandLineArguments); return map; } bool S60DeviceRunConfiguration::fromMap(const QVariantMap &map) { const QDir projectDir = QDir(target()->project()->projectDirectory()); m_proFilePath = projectDir.filePath(map.value(QLatin1String(PRO_FILE_KEY)).toString()); m_commandLineArguments = map.value(QLatin1String(COMMAND_LINE_ARGUMENTS_KEY)).toString(); if (m_proFilePath.isEmpty()) return false; if (!QFileInfo(m_proFilePath).exists()) return false; m_validParse = qt4Target()->qt4Project()->validParse(m_proFilePath); setDefaultDisplayName(tr("%1 on Symbian Device").arg(QFileInfo(m_proFilePath).completeBaseName())); return RunConfiguration::fromMap(map); } static inline QString fixBaseNameTarget(const QString &in) { if (in == QLatin1String("udeb")) return QLatin1String("debug"); if (in == QLatin1String("urel")) return QLatin1String("release"); return in; } QString S60DeviceRunConfiguration::targetName() const { TargetInformation ti = qt4Target()->qt4Project()->rootProjectNode()->targetInformation(projectFilePath()); if (!ti.valid) return QString(); return ti.target; } const QtVersion *S60DeviceRunConfiguration::qtVersion() const { if (const BuildConfiguration *bc = target()->activeBuildConfiguration()) if (const Qt4BuildConfiguration *qt4bc = qobject_cast(bc)) return qt4bc->qtVersion(); return 0; } bool S60DeviceRunConfiguration::isDebug() const { const Qt4BuildConfiguration *qt4bc = qt4Target()->activeBuildConfiguration(); return (qt4bc->qmakeBuildConfiguration() & QtVersion::DebugBuild); } QString S60DeviceRunConfiguration::symbianTarget() const { return isDebug() ? QLatin1String("udeb") : QLatin1String("urel"); } static inline QString symbianPlatformForToolChain(ProjectExplorer::ToolChainType t) { switch (t) { case ProjectExplorer::ToolChain_GCCE: case ProjectExplorer::ToolChain_GCCE_GNUPOC: return QLatin1String("gcce"); case ProjectExplorer::ToolChain_RVCT2_ARMV5: case ProjectExplorer::ToolChain_RVCT4_ARMV5: return QLatin1String("armv5"); default: // including ProjectExplorer::RVCT_ARMV6_GNUPOC: break; } return QLatin1String("armv6"); } /* Grep a package file for the '.exe' file. Currently for use on Linux only * as the '.pkg'-files on Windows do not contain drive letters, which is not * handled here. \code ; Executable and default resource files "./foo.exe" - "!:\sys\bin\foo.exe" \endcode */ static inline QString executableFromPackageUnix(const QString &packageFileName) { QFile packageFile(packageFileName); if (!packageFile.open(QIODevice::ReadOnly|QIODevice::Text)) return QString(); QRegExp pattern(QLatin1String("^\"(.*.exe)\" *- \"!:.*.exe\"$")); QTC_ASSERT(pattern.isValid(), return QString()); foreach(const QString &line, QString::fromLocal8Bit(packageFile.readAll()).split(QLatin1Char('\n'))) if (pattern.exactMatch(line)) { // Expand relative paths by package file paths QString rc = pattern.cap(1); if (rc.startsWith(QLatin1String("./"))) rc.remove(0, 2); const QFileInfo fi(rc); if (fi.isAbsolute()) return rc; return QFileInfo(packageFileName).absolutePath() + QLatin1Char('/') + rc; } return QString(); } // ABLD/Raptor: Return executable from device/EPOC static inline QString localExecutableFromDevice(const QtVersion *qtv, const QString &symbianTarget, /* udeb/urel */ const QString &targetName, ProjectExplorer::ToolChainType t) { QTC_ASSERT(qtv, return QString(); ) const S60Devices::Device device = S60Manager::instance()->deviceForQtVersion(qtv); QString localExecutable; QTextStream(&localExecutable) << device.epocRoot << "/epoc32/release/" << symbianPlatformForToolChain(t) << '/' << symbianTarget << '/' << targetName << ".exe"; return localExecutable; } QString S60DeviceRunConfiguration::localExecutableFileName() const { const ProjectExplorer::ToolChainType toolChain = toolChainType(); switch (toolChain) { case ProjectExplorer::ToolChain_GCCE_GNUPOC: case ProjectExplorer::ToolChain_RVCT_ARMV5_GNUPOC: { TargetInformation ti = qt4Target()->qt4Project()->rootProjectNode()->targetInformation(projectFilePath()); if (!ti.valid) return QString(); return executableFromPackageUnix(ti.buildDir + QLatin1Char('/') + ti.target + QLatin1String("_template.pkg")); } case ProjectExplorer::ToolChain_RVCT2_ARMV5: case ProjectExplorer::ToolChain_RVCT2_ARMV6: return localExecutableFromDevice(qtVersion(), symbianTarget(), targetName(), toolChain); break; case ProjectExplorer::ToolChain_GCCE: { // As of 4.7.1, qmake-gcce-Raptor builds were changed to put all executables into 'armv5' const QtVersion *qtv = qtVersion(); QTC_ASSERT(qtv, return QString(); ) return qtv->isBuildWithSymbianSbsV2() ? localExecutableFromDevice(qtv, symbianTarget(), targetName(), ProjectExplorer::ToolChain_RVCT2_ARMV5) : localExecutableFromDevice(qtv, symbianTarget(), targetName(), toolChain); } break; default: break; } return QString(); } quint32 S60DeviceRunConfiguration::executableUid() const { quint32 uid = 0; QString executablePath = localExecutableFileName(); if (!executablePath.isEmpty()) { QFile file(executablePath); if (file.open(QIODevice::ReadOnly)) { // executable's UID is 4 bytes starting at 8. const QByteArray data = file.read(12); if (data.size() == 12) { const unsigned char *d = reinterpret_cast(data.data() + 8); uid = *d++; uid += *d++ << 8; uid += *d++ << 16; uid += *d++ << 24; } } } return uid; } QString S60DeviceRunConfiguration::projectFilePath() const { return m_proFilePath; } QString S60DeviceRunConfiguration::commandLineArguments() const { return m_commandLineArguments; } void S60DeviceRunConfiguration::setCommandLineArguments(const QString &args) { m_commandLineArguments = args; } QString S60DeviceRunConfiguration::proFilePath() const { return m_proFilePath; } // ======== S60DeviceRunConfigurationFactory S60DeviceRunConfigurationFactory::S60DeviceRunConfigurationFactory(QObject *parent) : IRunConfigurationFactory(parent) { } S60DeviceRunConfigurationFactory::~S60DeviceRunConfigurationFactory() { } QStringList S60DeviceRunConfigurationFactory::availableCreationIds(Target *parent) const { Qt4SymbianTarget *target = qobject_cast(parent); if (!target || target->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID)) return QStringList(); return target->qt4Project()->applicationProFilePathes(QLatin1String(S60_DEVICE_RC_PREFIX)); } QString S60DeviceRunConfigurationFactory::displayNameForId(const QString &id) const { if (!pathFromId(id).isEmpty()) return tr("%1 on Symbian Device").arg(QFileInfo(pathFromId(id)).completeBaseName()); return QString(); } bool S60DeviceRunConfigurationFactory::canCreate(Target *parent, const QString &id) const { Qt4SymbianTarget *t = qobject_cast(parent); if (!t || t->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID)) return false; return t->qt4Project()->hasApplicationProFile(pathFromId(id)); } RunConfiguration *S60DeviceRunConfigurationFactory::create(Target *parent, const QString &id) { if (!canCreate(parent, id)) return 0; Qt4SymbianTarget *t = static_cast(parent); return new S60DeviceRunConfiguration(t, pathFromId(id)); } bool S60DeviceRunConfigurationFactory::canRestore(Target *parent, const QVariantMap &map) const { Qt4SymbianTarget *t = qobject_cast(parent); if (!t || t->id() != QLatin1String(Constants::S60_DEVICE_TARGET_ID)) return false; QString id = ProjectExplorer::idFromMap(map); return id == QLatin1String(S60_DEVICE_RC_ID); } RunConfiguration *S60DeviceRunConfigurationFactory::restore(Target *parent, const QVariantMap &map) { if (!canRestore(parent, map)) return 0; Qt4SymbianTarget *t = static_cast(parent); S60DeviceRunConfiguration *rc = new S60DeviceRunConfiguration(t, QString()); if (rc->fromMap(map)) return rc; delete rc; return 0; } bool S60DeviceRunConfigurationFactory::canClone(Target *parent, RunConfiguration *source) const { if (!qobject_cast(parent)) return false; return source->id() == QLatin1String(S60_DEVICE_RC_ID); } RunConfiguration *S60DeviceRunConfigurationFactory::clone(Target *parent, RunConfiguration *source) { if (!canClone(parent, source)) return 0; Qt4SymbianTarget *t = static_cast(parent); S60DeviceRunConfiguration *old = static_cast(source); return new S60DeviceRunConfiguration(t, old); } // ======== S60DeviceRunControlBase S60DeviceRunControl::S60DeviceRunControl(RunConfiguration *runConfiguration, QString mode) : RunControl(runConfiguration, mode), m_toolChain(ProjectExplorer::ToolChain_INVALID), m_tcfTrkDevice(0), m_launcher(0), m_state(StateUninit) { // connect for automatically reporting the "finished deploy" state to the progress manager connect(this, SIGNAL(finished()), this, SLOT(reportLaunchFinished())); S60DeviceRunConfiguration *s60runConfig = qobject_cast(runConfiguration); const Qt4BuildConfiguration *activeBuildConf = s60runConfig->qt4Target()->activeBuildConfiguration(); S60DeployConfiguration *activeDeployConf = qobject_cast(s60runConfig->qt4Target()->activeDeployConfiguration()); QTC_ASSERT(s60runConfig, return); m_toolChain = s60runConfig->toolChainType(); m_serialPortName = activeDeployConf->serialPortName(); m_serialPortFriendlyName = SymbianUtils::SymbianDeviceManager::instance()->friendlyNameForPort(m_serialPortName); m_executableUid = s60runConfig->executableUid(); m_targetName = s60runConfig->targetName(); m_commandLineArguments = s60runConfig->commandLineArguments(); m_qtDir = activeBuildConf->qtVersion()->versionInfo().value("QT_INSTALL_DATA"); m_installationDrive = activeDeployConf->installationDrive(); if (const QtVersion *qtv = activeDeployConf->qtVersion()) m_qtBinPath = qtv->versionInfo().value(QLatin1String("QT_INSTALL_BINS")); QTC_ASSERT(!m_qtBinPath.isEmpty(), return); m_executableFileName = s60runConfig->localExecutableFileName(); m_runSmartInstaller = activeDeployConf->runSmartInstaller(); switch (activeDeployConf->communicationChannel()) { case S60DeployConfiguration::CommunicationSerialConnection: break; case S60DeployConfiguration::CommunicationTcpConnection: m_address = activeDeployConf->deviceAddress(); m_port = activeDeployConf->devicePort().toInt(); } m_useOldTrk = activeDeployConf->communicationChannel() == S60DeployConfiguration::CommunicationSerialConnection; if (debug) qDebug() << "S60DeviceRunControl::CT" << m_targetName << ProjectExplorer::ToolChain::toolChainName(m_toolChain) << m_serialPortName << "Use old TRK" << m_useOldTrk; } S60DeviceRunControl::~S60DeviceRunControl() { if (m_tcfTrkDevice) { m_tcfTrkDevice->deleteLater(); m_tcfTrkDevice = 0; } if (m_launcher) { m_launcher->deleteLater(); m_launcher = 0; } } void S60DeviceRunControl::start() { m_launchProgress = new QFutureInterface; Core::ICore::instance()->progressManager()->addTask(m_launchProgress->future(), tr("Launching"), QLatin1String("Symbian.Launch")); m_launchProgress->setProgressRange(0, PROGRESS_MAX); m_launchProgress->setProgressValue(0); m_launchProgress->reportStarted(); emit started(); if (m_runSmartInstaller) { //Smart Installer does the running by itself appendMessage(tr("Please finalize the installation on your device."), NormalMessageFormat); emit finished(); return; } if (m_serialPortName.isEmpty() && m_address.isEmpty()) { m_launchProgress->reportCanceled(); QString msg = tr("No device is connected. Please connect a device and try again."); appendMessage(msg, NormalMessageFormat); emit finished(); return; } appendMessage(tr("Executable file: %1").arg(msgListFile(m_executableFileName)), NormalMessageFormat); QString errorMessage; QString settingsCategory; QString settingsPage; if (!checkConfiguration(&errorMessage, &settingsCategory, &settingsPage)) { m_launchProgress->reportCanceled(); appendMessage(errorMessage, ErrorMessageFormat); emit finished(); Core::ICore::instance()->showWarningWithOptions(tr("Debugger for Symbian Platform"), errorMessage, QString(), settingsCategory, settingsPage); return; } startLaunching(); } RunControl::StopResult S60DeviceRunControl::stop() { if (m_useOldTrk) { if (m_launcher) m_launcher->terminate(); } else { doStop(); } return AsynchronousStop; } bool S60DeviceRunControl::isRunning() const { if (m_useOldTrk) { return m_launcher && (m_launcher->state() == trk::Launcher::Connecting || m_launcher->state() == trk::Launcher::Connected || m_launcher->state() == trk::Launcher::WaitingForTrk); } else { return m_tcfTrkDevice && !m_tcfTrkDevice->device().isNull() && m_state >= StateConnecting; } } bool S60DeviceRunControl::promptToStop(bool *) const { // We override the settings prompt QTC_ASSERT(isRunning(), return true;) const QString question = tr("
%1 is still running on the device.
" "
Terminating it can leave the target in an inconsistent state.
" "
Would you still like to terminate it?
").arg(displayName()); return showPromptToStopDialog(tr("Application Still Running"), question, tr("Force Quit"), tr("Keep Running")); } void S60DeviceRunControl::startLaunching() { QString errorMessage; if (setupLauncher(errorMessage)) { if (m_launchProgress) m_launchProgress->setProgressValue(PROGRESS_MAX/2); } else { if (!errorMessage.isEmpty()) appendMessage(errorMessage, ErrorMessageFormat); stop(); emit finished(); } } bool S60DeviceRunControl::setupLauncher(QString &errorMessage) { connect(SymbianUtils::SymbianDeviceManager::instance(), SIGNAL(deviceRemoved(const SymbianUtils::SymbianDevice)), this, SLOT(deviceRemoved(SymbianUtils::SymbianDevice))); if(!m_useOldTrk) { //FIXME: Remove old TRK QTC_ASSERT(!m_tcfTrkDevice, return false); m_tcfTrkDevice = new TcfTrkDevice; if (debug) m_tcfTrkDevice->setVerbose(1); connect(m_tcfTrkDevice, SIGNAL(error(QString)), this, SLOT(slotError(QString))); connect(m_tcfTrkDevice, SIGNAL(logMessage(QString)), this, SLOT(slotTrkLogMessage(QString))); connect(m_tcfTrkDevice, SIGNAL(tcfEvent(tcftrk::TcfTrkEvent)), this, SLOT(slotTcftrkEvent(tcftrk::TcfTrkEvent))); connect(m_tcfTrkDevice, SIGNAL(serialPong(QString)), this, SLOT(slotSerialPong(QString))); const QSharedPointer tcfTrkSocket(new QTcpSocket); m_tcfTrkDevice->setDevice(tcfTrkSocket); tcfTrkSocket->connectToHost(m_address, m_port); m_state = StateConnecting; appendMessage(tr("Connecting to %1:%2...").arg(m_address).arg(m_port), NormalMessageFormat); } else { m_launcher = trk::Launcher::acquireFromDeviceManager(m_serialPortName, 0, &errorMessage); if (!m_launcher) return false; connect(m_launcher, SIGNAL(finished()), this, SLOT(launcherFinished())); connect(m_launcher, SIGNAL(canNotConnect(QString)), this, SLOT(printConnectFailed(QString))); connect(m_launcher, SIGNAL(stateChanged(int)), this, SLOT(slotLauncherStateChanged(int))); connect(m_launcher, SIGNAL(processStopped(uint,uint,uint,QString)), this, SLOT(processStopped(uint,uint,uint,QString))); if (!m_commandLineArguments.isEmpty()) m_launcher->setCommandLineArgs(m_commandLineArguments); const QString runFileName = QString::fromLatin1("%1:\\sys\\bin\\%2.exe").arg(m_installationDrive).arg(m_targetName); initLauncher(runFileName, m_launcher); const trk::PromptStartCommunicationResult src = S60RunConfigBluetoothStarter::startCommunication(m_launcher->trkDevice(), 0, &errorMessage); if (src != trk::PromptStartCommunicationConnected) return false; if (!m_launcher->startServer(&errorMessage)) { errorMessage = tr("Could not connect to phone on port '%1': %2\n" "Check if the phone is connected and App TRK is running.").arg(m_serialPortName, errorMessage); return false; } } return true; } void S60DeviceRunControl::doStop() { if (!m_tcfTrkDevice) { finishRunControl(); return; } switch (m_state) { case StateUninit: case StateConnecting: case StateConnected: finishRunControl(); break; case StateProcessRunning: QTC_ASSERT(!m_runningProcessId.isEmpty(), return); m_tcfTrkDevice->sendRunControlTerminateCommand(TcfTrkCallback(), m_runningProcessId.toAscii()); break; } } void S60DeviceRunControl::slotError(const QString &error) { appendMessage(tr("Error: %1").arg(error), ErrorMessageFormat); finishRunControl(); } void S60DeviceRunControl::slotTrkLogMessage(const QString &log) { if (debug) { qDebug("CODA log: %s", qPrintable(log.size()>200?log.left(200).append(QLatin1String(" ...")): log)); } } void S60DeviceRunControl::slotSerialPong(const QString &message) { if (debug) qDebug() << "CODA serial pong:" << message; } void S60DeviceRunControl::slotTcftrkEvent(const TcfTrkEvent &event) { if (debug) qDebug() << "CODA event:" << "Type:" << event.type() << "Message:" << event.toString(); switch (event.type()) { case TcfTrkEvent::LocatorHello: { // Commands accepted now m_state = StateConnected; appendMessage(tr("Connected."), NormalMessageFormat); if (m_launchProgress) m_launchProgress->setProgressValue(PROGRESS_MAX*0.80); initCommunication(); } break; case TcfTrkEvent::RunControlContextRemoved: handleContextRemoved(event); break; case TcfTrkEvent::RunControlContextAdded: m_state = StateProcessRunning; reportLaunchFinished(); handleContextAdded(event); break; case TcfTrkEvent::RunControlSuspended: handleContextSuspended(event); break; case TcfTrkEvent::RunControlModuleLoadSuspended: handleModuleLoadSuspended(event); break; case TcfTrkEvent::LoggingWriteEvent: handleLogging(event); break; default: if (debug) qDebug() << __FUNCTION__ << "Event not handled" << event.type(); break; } } void S60DeviceRunControl::initCommunication() { m_tcfTrkDevice->sendLoggingAddListenerCommand(TcfTrkCallback(this, &S60DeviceRunControl::handleAddListener)); } void S60DeviceRunControl::handleContextRemoved(const TcfTrkEvent &event) { const QVector removedItems = static_cast(event).ids(); if (!m_runningProcessId.isEmpty() && removedItems.contains(m_runningProcessId.toAscii())) { appendMessage(tr("Process has finished."), NormalMessageFormat); finishRunControl(); } } void S60DeviceRunControl::handleContextAdded(const TcfTrkEvent &event) { typedef TcfTrkRunControlContextAddedEvent TcfAddedEvent; const TcfAddedEvent &me = static_cast(event); foreach (const RunControlContext &context, me.contexts()) { if (context.parentId == "root") //is the created context a process m_runningProcessId = QLatin1String(context.id); } } void S60DeviceRunControl::handleContextSuspended(const TcfTrkEvent &event) { typedef TcfTrkRunControlContextSuspendedEvent TcfSuspendEvent; const TcfSuspendEvent &me = static_cast(event); switch (me.reason()) { case TcfSuspendEvent::Crash: appendMessage(tr("Process has crashed: %1").arg(QString::fromLatin1(me.message())), ErrorMessageFormat); m_tcfTrkDevice->sendRunControlResumeCommand(TcfTrkCallback(), me.id()); //TODO: Should I resume automaticly break; default: if (debug) qDebug() << "Context suspend not handled:" << "Reason:" << me.reason() << "Message:" << me.message(); break; } } void S60DeviceRunControl::handleModuleLoadSuspended(const TcfTrkEvent &event) { // Debug mode start: Continue: typedef TcfTrkRunControlModuleLoadContextSuspendedEvent TcfModuleLoadSuspendedEvent; const TcfModuleLoadSuspendedEvent &me = static_cast(event); if (me.info().requireResume) m_tcfTrkDevice->sendRunControlResumeCommand(TcfTrkCallback(), me.id()); } void S60DeviceRunControl::handleLogging(const TcfTrkEvent &event) { const TcfTrkLoggingWriteEvent &me = static_cast(event); appendMessage(me.message(), StdOutFormat); } void S60DeviceRunControl::handleAddListener(const TcfTrkCommandResult &result) { if (debug) qDebug() << __FUNCTION__ <<"Add log listener" << result.toString(); m_tcfTrkDevice->sendSymbianOsDataGetThreadsCommand(TcfTrkCallback(this, &S60DeviceRunControl::handleGetThreads)); } void S60DeviceRunControl::handleGetThreads(const TcfTrkCommandResult &result) { if (isProcessRunning(result, m_targetName)) { appendMessage(tr("The process is already running on the device. Please first close it."), ErrorMessageFormat); finishRunControl(); } else { if (m_launchProgress) m_launchProgress->setProgressValue(PROGRESS_MAX*0.90); const QString runFileName = QString::fromLatin1("%1.exe").arg(m_targetName); m_tcfTrkDevice->sendProcessStartCommand(TcfTrkCallback(this, &S60DeviceRunControl::handleCreateProcess), runFileName, m_executableUid, m_commandLineArguments.split(" "), QString(), true); appendMessage(tr("Launching: %1").arg(runFileName), NormalMessageFormat); } } void S60DeviceRunControl::handleCreateProcess(const TcfTrkCommandResult &result) { const bool ok = result.type == TcfTrkCommandResult::SuccessReply; if (ok) { if (m_launchProgress) m_launchProgress->setProgressValue(PROGRESS_MAX); appendMessage(tr("Launched."), NormalMessageFormat); } else { appendMessage(tr("Launch failed: %1").arg(result.toString()), ErrorMessageFormat); finishRunControl(); } } void S60DeviceRunControl::finishRunControl() { m_runningProcessId.clear(); if (m_tcfTrkDevice) m_tcfTrkDevice->deleteLater(); m_tcfTrkDevice = 0; m_state = StateUninit; handleRunFinished(); } //////// Launcher code - to be removed void S60DeviceRunControl::printConnectFailed(const QString &errorMessage) { appendMessage(tr("Could not connect to App TRK on device: %1. Restarting App TRK might help.").arg(errorMessage), ErrorMessageFormat); } void S60DeviceRunControl::launcherFinished() { trk::Launcher::releaseToDeviceManager(m_launcher); m_launcher->deleteLater(); m_launcher = 0; handleRunFinished(); } void S60DeviceRunControl::reportLaunchFinished() { if (m_launchProgress) { m_launchProgress->reportFinished(); delete m_launchProgress; m_launchProgress = 0; } } void S60DeviceRunControl::processStopped(uint pc, uint pid, uint tid, const QString &reason) { appendMessage(trk::Launcher::msgStopped(pid, tid, pc, reason), StdOutFormat); m_launcher->terminate(); } QMessageBox *S60DeviceRunControl::createTrkWaitingMessageBox(const QString &port, QWidget *parent) { const QString title = tr("Waiting for App TRK"); const QString text = tr("Qt Creator is waiting for the TRK application to connect.
" "Please make sure the application is running on " "your mobile phone and the right port is " "configured in the project settings.").arg(port); QMessageBox *rc = new QMessageBox(QMessageBox::Information, title, text, QMessageBox::Cancel, parent); return rc; } void S60DeviceRunControl::slotLauncherStateChanged(int s) { if (s == trk::Launcher::WaitingForTrk) { QMessageBox *mb = S60DeviceRunControl::createTrkWaitingMessageBox(m_launcher->trkServerName(), Core::ICore::instance()->mainWindow()); connect(m_launcher, SIGNAL(stateChanged(int)), mb, SLOT(close())); connect(mb, SIGNAL(finished(int)), this, SLOT(slotWaitingForTrkClosed())); mb->open(); } } void S60DeviceRunControl::slotWaitingForTrkClosed() { if (m_launcher && m_launcher->state() == trk::Launcher::WaitingForTrk) { stop(); appendMessage(tr("Canceled."), ErrorMessageFormat); emit finished(); } } void S60DeviceRunControl::printApplicationOutput(const QString &output) { printApplicationOutput(output, false); } void S60DeviceRunControl::printApplicationOutput(const QString &output, bool onStdErr) { appendMessage(output, onStdErr ? StdErrFormat : StdOutFormat); } void S60DeviceRunControl::deviceRemoved(const SymbianUtils::SymbianDevice &d) { if (m_launcher && d.portName() == m_serialPortName) { trk::Launcher::releaseToDeviceManager(m_launcher); m_launcher->deleteLater(); m_launcher = 0; QString msg = tr("The device '%1' has been disconnected").arg(d.friendlyName()); appendMessage(msg, ErrorMessageFormat); emit finished(); } } void S60DeviceRunControl::initLauncher(const QString &executable, trk::Launcher *launcher) { connect(launcher, SIGNAL(startingApplication()), this, SLOT(printStartingNotice())); connect(launcher, SIGNAL(applicationRunning(uint)), this, SLOT(applicationRunNotice(uint))); connect(launcher, SIGNAL(canNotRun(QString)), this, SLOT(applicationRunFailedNotice(QString))); connect(launcher, SIGNAL(applicationOutputReceived(QString)), this, SLOT(printApplicationOutput(QString))); launcher->addStartupActions(trk::Launcher::ActionRun); launcher->setFileName(executable); } void S60DeviceRunControl::printStartingNotice() { appendMessage(tr("Starting application..."), NormalMessageFormat); } void S60DeviceRunControl::applicationRunNotice(uint pid) { appendMessage(tr("Application running with pid %1.").arg(pid), NormalMessageFormat); if (m_launchProgress) m_launchProgress->setProgressValue(PROGRESS_MAX); } void S60DeviceRunControl::applicationRunFailedNotice(const QString &errorMessage) { appendMessage(tr("Could not start application: %1").arg(errorMessage), NormalMessageFormat); } // End of Launcher code - to be removed bool S60DeviceRunControl::checkConfiguration(QString * /* errorMessage */, QString * /* settingsCategory */, QString * /* settingsPage */) const { return true; } void S60DeviceRunControl::handleRunFinished() { emit finished(); appendMessage(tr("Finished."), NormalMessageFormat); } // ======== S60DeviceDebugRunControl // Return symbol file which should co-exist with the executable. // location in debug builds. This can be 'foo.sym' (ABLD) or 'foo.exe.sym' (Raptor) static inline QString symbolFileFromExecutable(const QString &executable) { // 'foo.exe.sym' (Raptor) const QFileInfo raptorSymFi(executable + QLatin1String(".sym")); if (raptorSymFi.isFile()) return raptorSymFi.absoluteFilePath(); // 'foo.sym' (ABLD) const int lastDotPos = executable.lastIndexOf(QLatin1Char('.')); if (lastDotPos != -1) { const QString symbolFileName = executable.mid(0, lastDotPos) + QLatin1String(".sym"); const QFileInfo symbolFileNameFi(symbolFileName); if (symbolFileNameFi.isFile()) return symbolFileNameFi.absoluteFilePath(); } return QString(); } // Create start parameters from run configuration static Debugger::DebuggerStartParameters s60DebuggerStartParams(const S60DeviceRunConfiguration *rc) { Debugger::DebuggerStartParameters sp; QTC_ASSERT(rc, return sp); const S60DeployConfiguration *activeDeployConf = qobject_cast(rc->qt4Target()->activeDeployConfiguration()); const QString debugFileName = QString::fromLatin1("%1:\\sys\\bin\\%2.exe") .arg(activeDeployConf->installationDrive()).arg(rc->targetName()); sp.remoteChannel = activeDeployConf->serialPortName(); sp.processArgs = rc->commandLineArguments(); sp.startMode = Debugger::StartInternal; sp.toolChainType = rc->toolChainType(); sp.executable = debugFileName; sp.executableUid = rc->executableUid(); sp.enabledEngines = Debugger::GdbEngineType; sp.serverAddress = activeDeployConf->deviceAddress(); sp.serverPort = activeDeployConf->devicePort().toInt(); //FIXME: there should be only one... trkAdapter sp.communicationChannel = activeDeployConf->communicationChannel() == S60DeployConfiguration::CommunicationSerialConnection? Debugger::DebuggerStartParameters::CommunicationChannelUsb: Debugger::DebuggerStartParameters::CommunicationChannelTcpIp; QTC_ASSERT(sp.executableUid, return sp); // Prefer the '*.sym' file over the '.exe', which should exist at the same // location in debug builds. This can be 'foo.exe' (ABLD) or 'foo.exe.sym' (Raptor) sp.symbolFileName = symbolFileFromExecutable(rc->localExecutableFileName()); return sp; } S60DeviceDebugRunControl::S60DeviceDebugRunControl(S60DeviceRunConfiguration *rc, const QString &) : Debugger::DebuggerRunControl(rc, s60DebuggerStartParams(rc)) { if (startParameters().symbolFileName.isEmpty()) { const QString msg = tr("Warning: Cannot locate the symbol file belonging to %1."). arg(rc->localExecutableFileName()); appendMessage(msg, ErrorMessageFormat); } } void S60DeviceDebugRunControl::start() { Debugger::ConfigurationCheck check = Debugger::checkDebugConfiguration(startParameters().toolChainType); if (!check) { appendMessage(check.errorMessage, ErrorMessageFormat); emit finished(); Core::ICore::instance()->showWarningWithOptions(tr("Debugger for Symbian Platform"), check.errorMessage, QString(), check.settingsCategory, check.settingsPage); return; } appendMessage(tr("Launching debugger..."), NormalMessageFormat); Debugger::DebuggerRunControl::start(); } bool S60DeviceDebugRunControl::promptToStop(bool *) const { // We override the settings prompt return Debugger::DebuggerRunControl::promptToStop(0); }