/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of Qt Creator. ** ** 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. ** ****************************************************************************/ #include "iosdebugsupport.h" #include "iosrunner.h" #include "iosmanager.h" #include "iosdevice.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef Q_OS_UNIX #include #else #include #endif using namespace Debugger; using namespace ProjectExplorer; using namespace QmakeProjectManager; namespace Ios { namespace Internal { RunControl *IosDebugSupport::createDebugRunControl(IosRunConfiguration *runConfig, QString *errorMessage) { Target *target = runConfig->target(); if (!target) return 0; ProjectExplorer::IDevice::ConstPtr device = DeviceKitInformation::device(target->kit()); if (device.isNull()) return 0; QmakeProject *project = static_cast(target->project()); Kit *kit = target->kit(); DebuggerStartParameters params; if (device->type() == Core::Id(Ios::Constants::IOS_DEVICE_TYPE)) { params.startMode = AttachToRemoteProcess; params.platform = QLatin1String("remote-ios"); IosDevice::ConstPtr iosDevice = device.dynamicCast(); if (iosDevice.isNull()) return 0; QString osVersion = iosDevice->osVersion(); Utils::FileName deviceSdk1 = Utils::FileName::fromString(QDir::homePath() + QLatin1String("/Library/Developer/Xcode/iOS DeviceSupport/") + osVersion + QLatin1String("/Symbols")); QString deviceSdk; if (deviceSdk1.toFileInfo().isDir()) { deviceSdk = deviceSdk1.toString(); } else { Utils::FileName deviceSdk2 = IosConfigurations::developerPath() .appendPath(QLatin1String("Platforms/iPhoneOS.platform/DeviceSupport/")) .appendPath(osVersion).appendPath(QLatin1String("Symbols")); if (deviceSdk2.toFileInfo().isDir()) { deviceSdk = deviceSdk2.toString(); } else { TaskHub::addTask(Task::Warning, tr( "Could not find device specific debug symbols at %1. " "Debugging initialization will be slow until you open the Organizer window of " "Xcode with the device connected to have the symbols generated.") .arg(deviceSdk1.toUserOutput()), ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT); } } params.deviceSymbolsRoot = deviceSdk; } else { params.startMode = AttachExternal; params.platform = QLatin1String("ios-simulator"); } params.displayName = runConfig->appName(); params.remoteSetupNeeded = true; if (!params.breakOnMain) params.continueAfterAttach = true; Debugger::DebuggerRunConfigurationAspect *aspect = runConfig->extraAspect(); bool cppDebug = aspect->useCppDebugger(); bool qmlDebug = aspect->useQmlDebugger(); if (cppDebug) { params.languages |= CppLanguage; params.sysRoot = SysRootKitInformation::sysRoot(kit).toString(); params.debuggerCommand = DebuggerKitInformation::debuggerCommand(kit).toString(); if (ToolChain *tc = ToolChainKitInformation::toolChain(kit)) params.toolChainAbi = tc->targetAbi(); params.executable = runConfig->exePath().toString(); params.remoteChannel = QLatin1String("connect://localhost:0"); Utils::FileName xcodeInfo = IosConfigurations::developerPath().parentDir() .appendPath(QLatin1String("Info.plist")); bool buggyLldb = false; if (xcodeInfo.toFileInfo().exists()) { QSettings settings(xcodeInfo.toString(), QSettings::NativeFormat); QStringList version = settings.value(QLatin1String("CFBundleShortVersionString")).toString() .split(QLatin1Char('.')); if (version.value(0).toInt() == 5 && version.value(1, QString::number(1)).toInt() == 0) buggyLldb = true; } QString bundlePath = runConfig->bundleDir().toString(); bundlePath.chop(4); Utils::FileName dsymPath = Utils::FileName::fromString( bundlePath.append(QLatin1String(".dSYM"))); if (!dsymPath.toFileInfo().exists()) { if (buggyLldb) TaskHub::addTask(Task::Warning, tr("Debugging with Xcode 5.0.x can be unreliable without a dSYM. " "To create one, add a dsymutil deploystep."), ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT); } else if (dsymPath.toFileInfo().lastModified() < QFileInfo(runConfig->exePath().toUserOutput()).lastModified()) { TaskHub::addTask(Task::Warning, tr("The dSYM %1 seems to be outdated, it might confuse the debugger.") .arg(dsymPath.toUserOutput()), ProjectExplorer::Constants::TASK_CATEGORY_DEPLOYMENT); } } if (qmlDebug) { params.languages |= QmlLanguage; params.projectSourceDirectory = project->projectDirectory(); params.projectSourceFiles = project->files(QmakeProject::ExcludeGeneratedFiles); params.projectBuildDirectory = project->rootQmakeProjectNode()->buildDir(); if (!cppDebug) params.startMode = AttachToRemoteServer; } DebuggerRunControl * const debuggerRunControl = DebuggerPlugin::createDebugger(params, runConfig, errorMessage); if (debuggerRunControl) new IosDebugSupport(runConfig, debuggerRunControl, cppDebug, qmlDebug); return debuggerRunControl; } IosDebugSupport::IosDebugSupport(IosRunConfiguration *runConfig, DebuggerRunControl *runControl, bool cppDebug, bool qmlDebug) : QObject(runControl), m_runControl(runControl), m_runner(new IosRunner(this, runConfig, cppDebug, qmlDebug)) { connect(m_runControl->engine(), SIGNAL(requestRemoteSetup()), m_runner, SLOT(start())); connect(m_runControl, SIGNAL(finished()), m_runner, SLOT(stop())); connect(m_runner, SIGNAL(gotServerPorts(int,int)), SLOT(handleServerPorts(int,int))); connect(m_runner, SIGNAL(gotInferiorPid(Q_PID, int)), SLOT(handleGotInferiorPid(Q_PID, int))); connect(m_runner, SIGNAL(finished(bool)), SLOT(handleRemoteProcessFinished(bool))); connect(m_runner, SIGNAL(errorMsg(QString)), SLOT(handleRemoteErrorOutput(QString))); connect(m_runner, SIGNAL(appOutput(QString)), SLOT(handleRemoteOutput(QString))); } IosDebugSupport::~IosDebugSupport() { } void IosDebugSupport::handleServerPorts(int gdbServerPort, int qmlPort) { if (gdbServerPort > 0 || (m_runner && !m_runner->cppDebug() && qmlPort > 0)) { m_runControl->engine()->notifyEngineRemoteSetupDone(gdbServerPort, qmlPort); } else { m_runControl->engine()->notifyEngineRemoteSetupFailed( tr("Could not get debug server file descriptor.")); } } void IosDebugSupport::handleGotInferiorPid(Q_PID pid, int qmlPort) { if (pid > 0) { //m_runControl->engine()->notifyInferiorPid(pid); m_runControl->engine()->notifyEngineRemoteSetupDone(int(Utils::qPidToPid(pid)), qmlPort); } else { m_runControl->engine()->notifyEngineRemoteSetupFailed( tr("Got an invalid process id.")); } } void IosDebugSupport::handleRemoteProcessFinished(bool cleanEnd) { if (!cleanEnd && m_runControl) m_runControl->showMessage(tr("Run failed unexpectedly."), AppStuff); //m_runControl->engine()->notifyInferiorIll(); m_runControl->engine()->abortDebugger(); } void IosDebugSupport::handleRemoteOutput(const QString &output) { if (m_runControl) { if (m_runControl->engine()) m_runControl->engine()->showMessage(output, AppOutput); else m_runControl->showMessage(output, AppOutput); } } void IosDebugSupport::handleRemoteErrorOutput(const QString &output) { if (m_runControl) { if (m_runControl->engine()) m_runControl->engine()->showMessage(output, AppError); else m_runControl->showMessage(output, AppError); } } } // namespace Internal } // namespace Ios