2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
**
|
2010-03-05 11:25:49 +01:00
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Nokia.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** If you are unsure which license is appropriate for your use, please
|
2009-08-14 09:30:56 +02:00
|
|
|
** contact the sales department at http://qt.nokia.com/contact.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 15:08:31 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "debuggerrunner.h"
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
#include "debuggeractions.h"
|
2010-11-10 11:39:01 +01:00
|
|
|
#include "debuggercore.h"
|
2010-06-16 11:08:54 +02:00
|
|
|
#include "debuggerengine.h"
|
|
|
|
#include "debuggerplugin.h"
|
|
|
|
#include "debuggerstringutils.h"
|
2010-08-20 14:19:25 +02:00
|
|
|
#include "debuggeruiswitcher.h"
|
2010-09-13 11:08:08 +02:00
|
|
|
#include "gdb/gdbengine.h"
|
|
|
|
#include "gdb/remotegdbserveradapter.h"
|
|
|
|
#include "gdb/remoteplaingdbadapter.h"
|
2010-11-10 16:33:11 +01:00
|
|
|
#include "gdb/gdboptionspage.h"
|
2010-09-16 17:08:07 +02:00
|
|
|
#include "qml/qmlengine.h"
|
2010-09-13 11:08:08 +02:00
|
|
|
#include "qml/qmlcppengine.h"
|
2010-10-08 12:14:51 +02:00
|
|
|
#include "lldb/lldbenginehost.h"
|
2010-06-15 09:13:22 +02:00
|
|
|
|
2010-06-22 12:28:05 +02:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
# include "peutils.h"
|
|
|
|
#endif
|
|
|
|
|
2009-11-09 16:25:24 +01:00
|
|
|
#include <projectexplorer/debugginghelper.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <projectexplorer/project.h>
|
2010-11-01 14:14:17 +01:00
|
|
|
#include <projectexplorer/toolchain.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
2010-02-08 15:50:06 +01:00
|
|
|
#include <projectexplorer/target.h>
|
2009-11-25 18:50:20 +01:00
|
|
|
#include <projectexplorer/buildconfiguration.h>
|
2010-06-11 16:31:54 +02:00
|
|
|
#include <projectexplorer/applicationrunconfiguration.h> // For LocalApplication*
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-09-29 10:32:41 +02:00
|
|
|
#include <utils/synchronousprocess.h>
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <utils/qtcassert.h>
|
2010-08-20 14:19:25 +02:00
|
|
|
#include <utils/fancymainwindow.h>
|
2009-10-08 17:23:27 +02:00
|
|
|
#include <coreplugin/icore.h>
|
2008-12-09 15:25:01 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <QtCore/QDir>
|
2010-08-20 14:19:25 +02:00
|
|
|
#include <QtGui/QMessageBox>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-06-11 16:16:14 +02:00
|
|
|
using namespace ProjectExplorer;
|
2010-06-14 17:23:25 +02:00
|
|
|
using namespace Debugger::Internal;
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
namespace Debugger {
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
DebuggerEngine *createGdbEngine(const DebuggerStartParameters &);
|
|
|
|
DebuggerEngine *createScriptEngine(const DebuggerStartParameters &);
|
|
|
|
DebuggerEngine *createPdbEngine(const DebuggerStartParameters &);
|
|
|
|
DebuggerEngine *createTcfEngine(const DebuggerStartParameters &);
|
|
|
|
DebuggerEngine *createQmlEngine(const DebuggerStartParameters &);
|
2010-08-18 13:54:12 +02:00
|
|
|
DebuggerEngine *createQmlCppEngine(const DebuggerStartParameters &);
|
2010-10-08 12:14:51 +02:00
|
|
|
DebuggerEngine *createLLDBEngine(const DebuggerStartParameters &);
|
2010-06-16 11:08:54 +02:00
|
|
|
|
2010-11-10 16:33:11 +01:00
|
|
|
extern QString msgNoBinaryForToolChain(int tc);
|
2010-06-16 11:08:54 +02:00
|
|
|
|
|
|
|
// FIXME: Outdated?
|
|
|
|
// The createCdbEngine function takes a list of options pages it can add to.
|
|
|
|
// This allows for having a "enabled" toggle on the page independently
|
|
|
|
// of the engine. That's good for not enabling the related ActiveX control
|
|
|
|
// unnecessarily.
|
|
|
|
|
2010-06-22 11:42:28 +02:00
|
|
|
#ifdef CDB_ENABLED
|
2010-10-05 14:30:26 +02:00
|
|
|
DebuggerEngine *createCdbEngine(const DebuggerStartParameters &, QString *errorMessage);
|
2010-06-16 11:08:54 +02:00
|
|
|
bool checkCdbConfiguration(int toolChain, QString *errorMsg, QString *settingsPage);
|
2010-10-05 14:30:26 +02:00
|
|
|
bool isCdbEngineEnabled(); // Check the configuration page
|
2010-06-22 11:42:28 +02:00
|
|
|
#else
|
2010-10-05 14:30:26 +02:00
|
|
|
DebuggerEngine *createCdbEngine(const DebuggerStartParameters &, QString *) { return 0; }
|
2010-06-22 11:42:28 +02:00
|
|
|
bool checkCdbConfiguration(int, QString *, QString *) { return false; }
|
|
|
|
#endif
|
2010-06-16 11:08:54 +02:00
|
|
|
|
2010-06-22 11:42:28 +02:00
|
|
|
} // namespace Internal
|
2010-06-16 11:08:54 +02:00
|
|
|
|
|
|
|
static QString toolChainName(int toolChainType)
|
|
|
|
{
|
2010-11-01 14:14:17 +01:00
|
|
|
return ToolChain::toolChainName(ProjectExplorer::ToolChainType(toolChainType));
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
|
|
|
|
2010-06-11 16:16:14 +02:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2009-09-25 11:35:44 +02:00
|
|
|
// DebuggerRunControlFactory
|
2008-12-02 12:01:29 +01:00
|
|
|
//
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
static QString msgEngineNotAvailable(const char *engine)
|
|
|
|
{
|
|
|
|
return DebuggerPlugin::tr("The application requires the debugger engine '%1', "
|
|
|
|
"which is disabled.").arg(QLatin1String(engine));
|
|
|
|
}
|
|
|
|
|
2009-05-05 17:48:54 +02:00
|
|
|
// A factory to create DebuggerRunControls
|
2010-06-16 11:08:54 +02:00
|
|
|
DebuggerRunControlFactory::DebuggerRunControlFactory(QObject *parent,
|
2010-10-05 14:30:26 +02:00
|
|
|
unsigned enabledEngines)
|
2010-06-16 11:08:54 +02:00
|
|
|
: IRunControlFactory(parent), m_enabledEngines(enabledEngines)
|
2008-12-02 12:01:29 +01:00
|
|
|
{}
|
|
|
|
|
2009-10-08 18:37:18 +02:00
|
|
|
bool DebuggerRunControlFactory::canRun(RunConfiguration *runConfiguration, const QString &mode) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-06-09 16:13:47 +02:00
|
|
|
// return mode == ProjectExplorer::Constants::DEBUGMODE;
|
2010-11-12 20:18:29 +01:00
|
|
|
return mode == Constants::DEBUGMODE
|
2009-10-08 18:37:18 +02:00
|
|
|
&& qobject_cast<LocalApplicationRunConfiguration *>(runConfiguration);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2009-09-25 11:35:44 +02:00
|
|
|
QString DebuggerRunControlFactory::displayName() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-02-27 20:06:08 +01:00
|
|
|
return tr("Debug");
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2010-09-29 10:32:41 +02:00
|
|
|
// Find Qt installation by running qmake
|
|
|
|
static inline QString findQtInstallPath(const QString &qmakePath)
|
|
|
|
{
|
|
|
|
QProcess proc;
|
|
|
|
QStringList args;
|
|
|
|
args.append(QLatin1String("-query"));
|
|
|
|
args.append(QLatin1String("QT_INSTALL_HEADERS"));
|
|
|
|
proc.start(qmakePath, args);
|
|
|
|
if (!proc.waitForStarted()) {
|
|
|
|
qWarning("%s: Cannot start '%s': %s", Q_FUNC_INFO, qPrintable(qmakePath),
|
|
|
|
qPrintable(proc.errorString()));
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
proc.closeWriteChannel();
|
|
|
|
if (!proc.waitForFinished()) {
|
|
|
|
Utils::SynchronousProcess::stopProcess(proc);
|
|
|
|
qWarning("%s: Timeout running '%s'.", Q_FUNC_INFO, qPrintable(qmakePath));
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
if (proc.exitStatus() != QProcess::NormalExit) {
|
|
|
|
qWarning("%s: '%s' crashed.", Q_FUNC_INFO, qPrintable(qmakePath));
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
const QByteArray ba = proc.readAllStandardOutput().trimmed();
|
|
|
|
QDir dir(QString::fromLocal8Bit(ba));
|
|
|
|
if (dir.exists() && dir.cdUp())
|
|
|
|
return dir.absolutePath();
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2010-06-14 08:57:15 +02:00
|
|
|
static DebuggerStartParameters localStartParameters(RunConfiguration *runConfiguration)
|
2009-07-28 09:10:35 +02:00
|
|
|
{
|
2010-06-14 08:57:15 +02:00
|
|
|
DebuggerStartParameters sp;
|
2010-06-11 16:06:13 +02:00
|
|
|
QTC_ASSERT(runConfiguration, return sp);
|
|
|
|
LocalApplicationRunConfiguration *rc =
|
|
|
|
qobject_cast<LocalApplicationRunConfiguration *>(runConfiguration);
|
|
|
|
QTC_ASSERT(rc, return sp);
|
|
|
|
|
2010-06-14 08:57:15 +02:00
|
|
|
sp.startMode = StartInternal;
|
|
|
|
sp.environment = rc->environment().toStringList();
|
2010-09-27 15:04:58 +01:00
|
|
|
sp.workingDirectory = rc->workingDirectory();
|
|
|
|
sp.executable = rc->executable();
|
|
|
|
sp.processArgs = rc->commandLineArguments();
|
2010-06-14 08:57:15 +02:00
|
|
|
sp.toolChainType = rc->toolChainType();
|
|
|
|
sp.useTerminal = rc->runMode() == LocalApplicationRunConfiguration::Console;
|
|
|
|
sp.dumperLibrary = rc->dumperLibrary();
|
|
|
|
sp.dumperLibraryLocations = rc->dumperLibraryLocations();
|
2010-08-18 13:54:12 +02:00
|
|
|
|
2010-11-10 11:39:01 +01:00
|
|
|
if (debuggerCore()->isActiveDebugLanguage(QmlLanguage)) {
|
2010-08-18 13:54:12 +02:00
|
|
|
sp.qmlServerAddress = QLatin1String("127.0.0.1");
|
2010-09-02 17:17:35 +02:00
|
|
|
sp.qmlServerPort = runConfiguration->qmlDebugServerPort();
|
2010-08-18 13:54:12 +02:00
|
|
|
|
2010-09-08 13:31:12 +02:00
|
|
|
sp.projectDir = runConfiguration->target()->project()->projectDirectory();
|
|
|
|
if (runConfiguration->target()->activeBuildConfiguration())
|
2010-11-10 11:39:01 +01:00
|
|
|
sp.projectBuildDir = runConfiguration->target()
|
|
|
|
->activeBuildConfiguration()->buildDirectory();
|
2010-09-08 13:31:12 +02:00
|
|
|
|
2010-11-10 11:39:01 +01:00
|
|
|
sp.processArgs.append(QLatin1String("-qmljsdebugger=port:")
|
|
|
|
+ QString::number(sp.qmlServerPort));
|
2010-08-18 13:54:12 +02:00
|
|
|
}
|
|
|
|
|
2010-07-21 12:04:20 +02:00
|
|
|
// FIXME: If it's not yet build this will be empty and not filled
|
|
|
|
// when rebuild as the runConfiguration is not stored and therefore
|
|
|
|
// cannot be used to retrieve the dumper location.
|
|
|
|
//qDebug() << "DUMPER: " << sp.dumperLibrary << sp.dumperLibraryLocations;
|
2010-06-14 08:57:15 +02:00
|
|
|
sp.displayName = rc->displayName();
|
2010-06-11 16:06:13 +02:00
|
|
|
|
2010-06-11 16:16:14 +02:00
|
|
|
// Find qtInstallPath.
|
|
|
|
QString qmakePath = DebuggingHelperLibrary::findSystemQt(rc->environment());
|
2010-09-29 10:32:41 +02:00
|
|
|
if (!qmakePath.isEmpty())
|
|
|
|
sp.qtInstallPath = findQtInstallPath(qmakePath);
|
2010-06-11 16:06:13 +02:00
|
|
|
return sp;
|
2009-07-28 09:10:35 +02:00
|
|
|
}
|
|
|
|
|
2010-06-15 08:22:02 +02:00
|
|
|
RunControl *DebuggerRunControlFactory::create
|
|
|
|
(RunConfiguration *runConfiguration, const QString &mode)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-11-12 20:18:29 +01:00
|
|
|
QTC_ASSERT(mode == Constants::DEBUGMODE, return 0);
|
2010-06-14 08:57:15 +02:00
|
|
|
DebuggerStartParameters sp = localStartParameters(runConfiguration);
|
2010-06-22 12:41:41 +02:00
|
|
|
return create(sp, runConfiguration);
|
2010-06-11 16:06:13 +02:00
|
|
|
}
|
|
|
|
|
2010-07-21 12:04:20 +02:00
|
|
|
DebuggerRunControl *DebuggerRunControlFactory::create(
|
|
|
|
const DebuggerStartParameters &sp,
|
2010-06-22 12:41:41 +02:00
|
|
|
RunConfiguration *runConfiguration)
|
2010-06-11 16:06:13 +02:00
|
|
|
{
|
2010-07-20 16:34:39 +02:00
|
|
|
DebuggerRunControl *runControl =
|
|
|
|
new DebuggerRunControl(runConfiguration, m_enabledEngines, sp);
|
2010-06-16 11:08:54 +02:00
|
|
|
if (!runControl->engine()) {
|
|
|
|
qDebug() << "FAILED TO CREATE ENGINE";
|
|
|
|
delete runControl;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return runControl;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2010-07-20 16:34:39 +02:00
|
|
|
QWidget *DebuggerRunControlFactory::createConfigurationWidget
|
|
|
|
(RunConfiguration *runConfiguration)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
// NBS TODO: Add GDB-specific configuration widget
|
2009-07-13 17:35:17 +02:00
|
|
|
Q_UNUSED(runConfiguration)
|
2008-12-02 12:01:29 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-15 08:22:02 +02:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2010-06-16 11:08:54 +02:00
|
|
|
// DebuggerRunControl
|
2010-06-15 08:22:02 +02:00
|
|
|
//
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2010-09-14 08:34:07 +02:00
|
|
|
struct DebuggerRunnerPrivate {
|
|
|
|
explicit DebuggerRunnerPrivate(RunConfiguration *runConfiguration,
|
2010-10-05 14:30:26 +02:00
|
|
|
unsigned enabledEngines);
|
|
|
|
|
|
|
|
unsigned enabledEngines() const;
|
2010-09-14 08:34:07 +02:00
|
|
|
|
|
|
|
DebuggerEngine *m_engine;
|
|
|
|
const QWeakPointer<RunConfiguration> m_myRunConfiguration;
|
|
|
|
bool m_running;
|
2010-10-05 14:30:26 +02:00
|
|
|
const unsigned m_cmdLineEnabledEngines;
|
2010-09-14 08:34:07 +02:00
|
|
|
QString m_errorMessage;
|
|
|
|
QString m_settingsIdHint;
|
|
|
|
};
|
|
|
|
|
2010-10-05 14:30:26 +02:00
|
|
|
unsigned DebuggerRunnerPrivate::enabledEngines() const
|
|
|
|
{
|
|
|
|
unsigned rc = m_cmdLineEnabledEngines;
|
|
|
|
#ifdef CDB_ENABLED
|
|
|
|
if (!Internal::isCdbEngineEnabled())
|
|
|
|
rc &= ~CdbEngineType;
|
|
|
|
#endif
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2010-09-14 08:34:07 +02:00
|
|
|
DebuggerRunnerPrivate::DebuggerRunnerPrivate(RunConfiguration *runConfiguration,
|
2010-10-05 14:30:26 +02:00
|
|
|
unsigned enabledEngines) :
|
2010-10-22 11:27:45 +02:00
|
|
|
m_engine(0)
|
|
|
|
, m_myRunConfiguration(runConfiguration)
|
2010-08-18 13:54:12 +02:00
|
|
|
, m_running(false)
|
2010-10-05 14:30:26 +02:00
|
|
|
, m_cmdLineEnabledEngines(enabledEngines)
|
2010-09-14 08:34:07 +02:00
|
|
|
{
|
|
|
|
}
|
2010-08-18 13:54:12 +02:00
|
|
|
|
2010-09-14 08:34:07 +02:00
|
|
|
DebuggerRunControl::DebuggerRunControl(RunConfiguration *runConfiguration,
|
2010-10-05 14:30:26 +02:00
|
|
|
unsigned enabledEngines, const DebuggerStartParameters &sp)
|
2010-11-12 20:18:29 +01:00
|
|
|
: RunControl(runConfiguration, Constants::DEBUGMODE),
|
2010-09-14 08:34:07 +02:00
|
|
|
d(new DebuggerRunnerPrivate(runConfiguration, enabledEngines))
|
2010-06-15 08:22:02 +02:00
|
|
|
{
|
2010-08-24 16:22:21 +02:00
|
|
|
connect(this, SIGNAL(finished()), this, SLOT(handleFinished()));
|
2010-08-24 17:17:54 +02:00
|
|
|
DebuggerStartParameters startParams = sp;
|
|
|
|
createEngine(startParams);
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
|
|
|
|
2010-06-22 15:14:44 +02:00
|
|
|
DebuggerRunControl::~DebuggerRunControl()
|
|
|
|
{
|
|
|
|
disconnect();
|
2010-10-05 14:30:26 +02:00
|
|
|
if (DebuggerEngine *engine = d->m_engine) {
|
|
|
|
d->m_engine = 0;
|
|
|
|
engine->disconnect();
|
|
|
|
delete engine;
|
|
|
|
}
|
2010-06-22 15:14:44 +02:00
|
|
|
}
|
|
|
|
|
2010-08-20 15:23:42 +02:00
|
|
|
const DebuggerStartParameters &DebuggerRunControl::startParameters() const
|
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, return *(new DebuggerStartParameters()));
|
|
|
|
return d->m_engine->startParameters();
|
2010-08-20 15:23:42 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
static DebuggerEngineType engineForToolChain(int toolChainType)
|
|
|
|
{
|
|
|
|
switch (toolChainType) {
|
2010-11-01 14:14:17 +01:00
|
|
|
case ProjectExplorer::ToolChain_LINUX_ICC:
|
|
|
|
case ProjectExplorer::ToolChain_MinGW:
|
|
|
|
case ProjectExplorer::ToolChain_GCC:
|
|
|
|
case ProjectExplorer::ToolChain_WINSCW: // S60
|
|
|
|
case ProjectExplorer::ToolChain_GCCE:
|
|
|
|
case ProjectExplorer::ToolChain_RVCT_ARMV5:
|
|
|
|
case ProjectExplorer::ToolChain_RVCT_ARMV6:
|
|
|
|
case ProjectExplorer::ToolChain_RVCT_ARMV5_GNUPOC:
|
|
|
|
case ProjectExplorer::ToolChain_GCCE_GNUPOC:
|
|
|
|
case ProjectExplorer::ToolChain_GCC_MAEMO:
|
2010-06-16 11:08:54 +02:00
|
|
|
return GdbEngineType;
|
|
|
|
|
2010-11-01 14:14:17 +01:00
|
|
|
case ProjectExplorer::ToolChain_MSVC:
|
|
|
|
case ProjectExplorer::ToolChain_WINCE:
|
2010-06-16 11:08:54 +02:00
|
|
|
return CdbEngineType;
|
|
|
|
|
2010-11-01 14:14:17 +01:00
|
|
|
case ProjectExplorer::ToolChain_OTHER:
|
|
|
|
case ProjectExplorer::ToolChain_UNKNOWN:
|
|
|
|
case ProjectExplorer::ToolChain_INVALID:
|
2010-06-16 11:08:54 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return NoEngineType;
|
2010-06-15 09:13:22 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
|
|
|
|
// Figure out the debugger type of an executable. Analyze executable
|
|
|
|
// unless the toolchain provides a hint.
|
2010-10-05 14:30:26 +02:00
|
|
|
DebuggerEngineType DebuggerRunControl::engineForExecutable(unsigned enabledEngineTypes, const QString &executable)
|
2010-06-15 09:13:22 +02:00
|
|
|
{
|
2010-09-14 13:39:32 +02:00
|
|
|
/*if (executable.endsWith(_("qmlviewer"))) {
|
2010-10-05 14:30:26 +02:00
|
|
|
if (enabledEngineTypes & QmlEngineType)
|
2010-06-16 11:08:54 +02:00
|
|
|
return QmlEngineType;
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_errorMessage = msgEngineNotAvailable("Qml Engine");
|
2010-09-14 13:39:32 +02:00
|
|
|
}*/
|
2010-06-15 08:22:02 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
if (executable.endsWith(_(".js"))) {
|
2010-10-05 14:30:26 +02:00
|
|
|
if (enabledEngineTypes & ScriptEngineType)
|
2010-06-16 11:08:54 +02:00
|
|
|
return ScriptEngineType;
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_errorMessage = msgEngineNotAvailable("Script Engine");
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
if (executable.endsWith(_(".py"))) {
|
2010-10-05 14:30:26 +02:00
|
|
|
if (enabledEngineTypes & PdbEngineType)
|
2010-06-16 11:08:54 +02:00
|
|
|
return PdbEngineType;
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_errorMessage = msgEngineNotAvailable("Pdb Engine");
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
// A remote executable?
|
|
|
|
if (!executable.endsWith(_(".exe")))
|
2010-06-22 12:28:05 +02:00
|
|
|
return GdbEngineType;
|
2010-06-16 11:08:54 +02:00
|
|
|
|
|
|
|
// If a file has PDB files, it has been compiled by VS.
|
|
|
|
QStringList pdbFiles;
|
2010-09-14 08:34:07 +02:00
|
|
|
if (!getPDBFiles(executable, &pdbFiles, &d->m_errorMessage)) {
|
2010-06-16 11:08:54 +02:00
|
|
|
qWarning("Cannot determine type of executable %s: %s",
|
2010-09-14 08:34:07 +02:00
|
|
|
qPrintable(executable), qPrintable(d->m_errorMessage));
|
2010-06-22 12:28:05 +02:00
|
|
|
return NoEngineType;
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
|
|
|
if (pdbFiles.empty())
|
|
|
|
return GdbEngineType;
|
2010-06-14 17:23:25 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
// We need the CDB debugger in order to be able to debug VS
|
|
|
|
// executables
|
2010-11-01 14:14:17 +01:00
|
|
|
if (checkDebugConfiguration(ProjectExplorer::ToolChain_MSVC, &d->m_errorMessage, 0, &d->m_settingsIdHint)) {
|
2010-10-05 14:30:26 +02:00
|
|
|
if (enabledEngineTypes & CdbEngineType)
|
|
|
|
return CdbEngineType;
|
|
|
|
d->m_errorMessage = msgEngineNotAvailable("Cdb Engine");
|
|
|
|
return NoEngineType;
|
|
|
|
}
|
2010-06-16 11:08:54 +02:00
|
|
|
#else
|
2010-10-05 14:30:26 +02:00
|
|
|
if (enabledEngineTypes & GdbEngineType)
|
2010-06-16 11:08:54 +02:00
|
|
|
return GdbEngineType;
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_errorMessage = msgEngineNotAvailable("Gdb Engine");
|
2010-06-16 11:08:54 +02:00
|
|
|
#endif
|
2009-10-08 17:36:07 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
return NoEngineType;
|
2010-06-15 08:22:02 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
// Debugger type for mode.
|
2010-10-05 14:30:26 +02:00
|
|
|
DebuggerEngineType DebuggerRunControl::engineForMode(unsigned enabledEngineTypes, DebuggerStartMode startMode)
|
2010-06-07 16:40:33 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
if (startMode == AttachTcf)
|
|
|
|
return TcfEngineType;
|
2010-06-07 16:40:33 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
// Preferably Windows debugger for attaching locally.
|
2010-10-05 14:30:26 +02:00
|
|
|
if (startMode != AttachToRemote && (enabledEngineTypes & CdbEngineType))
|
2010-06-16 11:08:54 +02:00
|
|
|
return CdbEngineType;
|
2010-10-22 11:31:33 +02:00
|
|
|
if (startMode == AttachCrashedExternal) {
|
|
|
|
d->m_errorMessage = tr("There is no debugging engine available for post-mortem debugging.");
|
|
|
|
return NoEngineType;
|
|
|
|
}
|
2010-06-22 12:28:05 +02:00
|
|
|
return GdbEngineType;
|
2010-06-16 11:08:54 +02:00
|
|
|
#else
|
|
|
|
Q_UNUSED(startMode)
|
2010-10-05 14:30:26 +02:00
|
|
|
Q_UNUSED(enabledEngineTypes)
|
2010-09-14 08:34:07 +02:00
|
|
|
// d->m_errorMessage = msgEngineNotAvailable("Gdb Engine");
|
2010-06-16 11:08:54 +02:00
|
|
|
return GdbEngineType;
|
|
|
|
#endif
|
2010-04-15 11:59:22 +02:00
|
|
|
}
|
|
|
|
|
2010-08-18 13:54:12 +02:00
|
|
|
void DebuggerRunControl::createEngine(const DebuggerStartParameters &startParams)
|
2009-09-09 18:35:25 +02:00
|
|
|
{
|
2010-08-18 13:54:12 +02:00
|
|
|
DebuggerStartParameters sp = startParams;
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
// Figure out engine according to toolchain, executable, attach or default.
|
|
|
|
DebuggerEngineType engineType = NoEngineType;
|
2010-11-10 11:39:01 +01:00
|
|
|
DebuggerLanguages activeLangs = debuggerCore()->activeLanguages();
|
2010-10-05 14:30:26 +02:00
|
|
|
const unsigned enabledEngineTypes = d->enabledEngines();
|
2010-09-27 17:22:57 +02:00
|
|
|
if (sp.executable.endsWith(_(".js")))
|
2010-06-16 11:08:54 +02:00
|
|
|
engineType = ScriptEngineType;
|
|
|
|
else if (sp.executable.endsWith(_(".py")))
|
|
|
|
engineType = PdbEngineType;
|
2010-10-05 14:30:26 +02:00
|
|
|
else {
|
2010-06-16 11:08:54 +02:00
|
|
|
engineType = engineForToolChain(sp.toolChainType);
|
2010-10-05 14:30:26 +02:00
|
|
|
if (engineType == CdbEngineType && !(enabledEngineTypes & CdbEngineType)) {
|
|
|
|
d->m_errorMessage = msgEngineNotAvailable("Cdb Engine");
|
|
|
|
engineType = NoEngineType;
|
|
|
|
}
|
|
|
|
}
|
2010-06-16 11:08:54 +02:00
|
|
|
|
2010-06-25 15:06:30 +02:00
|
|
|
// Fixme: 1 of 3 testing hacks.
|
|
|
|
if (sp.processArgs.size() >= 5 && sp.processArgs.at(0) == _("@tcf@"))
|
|
|
|
engineType = GdbEngineType;
|
|
|
|
|
2010-10-08 12:14:51 +02:00
|
|
|
if (sp.processArgs.contains( _("@lldb@")))
|
|
|
|
engineType = LLDBEngineType;
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
if (engineType == NoEngineType
|
|
|
|
&& sp.startMode != AttachToRemote
|
|
|
|
&& !sp.executable.isEmpty())
|
2010-10-05 14:30:26 +02:00
|
|
|
engineType = engineForExecutable(enabledEngineTypes, sp.executable);
|
2010-06-16 11:08:54 +02:00
|
|
|
|
2010-10-22 11:27:45 +02:00
|
|
|
if (engineType == NoEngineType)
|
2010-10-05 14:30:26 +02:00
|
|
|
engineType = engineForMode(enabledEngineTypes, sp.startMode);
|
2010-06-16 11:08:54 +02:00
|
|
|
|
2010-10-22 11:27:45 +02:00
|
|
|
if ((engineType != QmlEngineType && engineType != NoEngineType)
|
|
|
|
&& (activeLangs & QmlLanguage)) {
|
2010-08-24 17:17:54 +02:00
|
|
|
if (activeLangs & CppLanguage) {
|
2010-08-18 13:54:12 +02:00
|
|
|
sp.cppEngineType = engineType;
|
|
|
|
engineType = QmlCppEngineType;
|
|
|
|
} else {
|
|
|
|
engineType = QmlEngineType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-22 12:28:05 +02:00
|
|
|
// qDebug() << "USING ENGINE : " << engineType;
|
2010-06-16 11:08:54 +02:00
|
|
|
|
|
|
|
switch (engineType) {
|
|
|
|
case GdbEngineType:
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine = createGdbEngine(sp);
|
|
|
|
initGdbEngine(qobject_cast<Internal::GdbEngine *>(d->m_engine));
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case ScriptEngineType:
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine = Internal::createScriptEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case CdbEngineType:
|
2010-10-05 14:30:26 +02:00
|
|
|
d->m_engine = Internal::createCdbEngine(sp, &d->m_errorMessage);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case PdbEngineType:
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine = Internal::createPdbEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case TcfEngineType:
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine = Internal::createTcfEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case QmlEngineType:
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine = Internal::createQmlEngine(sp);
|
2010-09-16 17:08:07 +02:00
|
|
|
connect(qobject_cast<QmlEngine *>(d->m_engine),
|
|
|
|
SIGNAL(remoteStartupRequested()), this,
|
|
|
|
SIGNAL(engineRequestSetup()));
|
2010-06-16 11:08:54 +02:00
|
|
|
break;
|
2010-08-18 13:54:12 +02:00
|
|
|
case QmlCppEngineType:
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine = Internal::createQmlCppEngine(sp);
|
2010-09-13 11:08:08 +02:00
|
|
|
if (Internal::GdbEngine *embeddedGdbEngine = gdbEngine())
|
|
|
|
initGdbEngine(embeddedGdbEngine);
|
2010-08-18 13:54:12 +02:00
|
|
|
break;
|
2010-10-08 12:14:51 +02:00
|
|
|
case LLDBEngineType:
|
|
|
|
d->m_engine = Internal::createLLDBEngine(sp);
|
2010-10-05 14:30:26 +02:00
|
|
|
case NoEngineType:
|
|
|
|
case AllEngineTypes:
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-10-05 14:30:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!d->m_engine) {
|
|
|
|
// Could not find anything suitable.
|
|
|
|
debuggingFinished();
|
|
|
|
// Create Message box with possibility to go to settings
|
|
|
|
const QString msg = tr("Cannot debug '%1' (tool chain: '%2'): %3")
|
|
|
|
.arg(sp.executable, toolChainName(sp.toolChainType), d->m_errorMessage);
|
|
|
|
Core::ICore::instance()->showWarningWithOptions(tr("Warning"),
|
|
|
|
msg, QString(), QLatin1String(Constants::DEBUGGER_SETTINGS_CATEGORY),
|
|
|
|
d->m_settingsIdHint);
|
2010-06-14 18:19:02 +02:00
|
|
|
}
|
2010-04-19 14:21:33 +02:00
|
|
|
}
|
|
|
|
|
2010-09-13 11:08:08 +02:00
|
|
|
void DebuggerRunControl::initGdbEngine(Internal::GdbEngine *engine)
|
|
|
|
{
|
|
|
|
QTC_ASSERT(engine, return)
|
|
|
|
|
|
|
|
// Forward adapter signals.
|
|
|
|
Internal::AbstractGdbAdapter *adapter = engine->gdbAdapter();
|
|
|
|
if (RemotePlainGdbAdapter *rpga = qobject_cast<RemotePlainGdbAdapter *>(adapter)) {
|
|
|
|
connect(rpga, SIGNAL(requestSetup()), this,
|
2010-09-16 17:08:07 +02:00
|
|
|
SIGNAL(engineRequestSetup()));
|
2010-09-13 11:08:08 +02:00
|
|
|
} else if (RemoteGdbServerAdapter *rgsa = qobject_cast<RemoteGdbServerAdapter *>(adapter)) {
|
|
|
|
connect(rgsa, SIGNAL(requestSetup()),
|
2010-09-16 17:08:07 +02:00
|
|
|
this, SIGNAL(engineRequestSetup()));
|
2010-09-13 11:08:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
QString DebuggerRunControl::displayName() const
|
2010-06-15 08:22:02 +02:00
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, return QString());
|
|
|
|
return d->m_engine->startParameters().displayName;
|
2010-06-15 08:22:02 +02:00
|
|
|
}
|
|
|
|
|
2010-09-23 10:35:23 +02:00
|
|
|
void DebuggerRunControl::setCustomEnvironment(Utils::Environment env)
|
2010-06-15 08:22:02 +02:00
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_engine->startParameters().environment = env.toStringList();
|
2010-06-15 08:22:02 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
bool DebuggerRunControl::checkDebugConfiguration(int toolChain,
|
|
|
|
QString *errorMessage,
|
|
|
|
QString *settingsCategory /* = 0 */,
|
|
|
|
QString *settingsPage /* = 0 */)
|
|
|
|
{
|
|
|
|
errorMessage->clear();
|
|
|
|
if (settingsCategory)
|
|
|
|
settingsCategory->clear();
|
|
|
|
if (settingsPage)
|
|
|
|
settingsPage->clear();
|
|
|
|
|
|
|
|
bool success = true;
|
|
|
|
|
2010-11-10 11:39:01 +01:00
|
|
|
if (!(debuggerCore()->activeLanguages() & CppLanguage))
|
2010-10-15 13:28:46 +02:00
|
|
|
return success;
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
switch(toolChain) {
|
2010-11-01 14:14:17 +01:00
|
|
|
case ProjectExplorer::ToolChain_GCC:
|
|
|
|
case ProjectExplorer::ToolChain_LINUX_ICC:
|
|
|
|
case ProjectExplorer::ToolChain_MinGW:
|
|
|
|
case ProjectExplorer::ToolChain_WINCE: // S60
|
|
|
|
case ProjectExplorer::ToolChain_WINSCW:
|
|
|
|
case ProjectExplorer::ToolChain_GCCE:
|
|
|
|
case ProjectExplorer::ToolChain_RVCT_ARMV5:
|
|
|
|
case ProjectExplorer::ToolChain_RVCT_ARMV6:
|
2010-11-10 16:33:11 +01:00
|
|
|
if (debuggerCore()->gdbBinaryForToolChain(toolChain).isEmpty()) {
|
|
|
|
*errorMessage = msgNoBinaryForToolChain(toolChain);
|
|
|
|
*settingsPage = GdbOptionsPage::settingsId();
|
2010-09-10 17:21:22 +02:00
|
|
|
*errorMessage += msgEngineNotAvailable("Gdb");
|
2010-11-10 16:33:11 +01:00
|
|
|
success = false;
|
|
|
|
} else {
|
|
|
|
success = true;
|
|
|
|
}
|
2010-06-16 11:08:54 +02:00
|
|
|
break;
|
2010-11-01 14:14:17 +01:00
|
|
|
case ProjectExplorer::ToolChain_MSVC:
|
2010-06-16 11:08:54 +02:00
|
|
|
success = checkCdbConfiguration(toolChain, errorMessage, settingsPage);
|
|
|
|
if (!success) {
|
2010-09-10 17:21:22 +02:00
|
|
|
*errorMessage += msgEngineNotAvailable("Cdb");
|
2010-06-16 11:08:54 +02:00
|
|
|
if (settingsPage)
|
|
|
|
*settingsPage = QLatin1String("Cdb");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!success && settingsCategory && settingsPage && !settingsPage->isEmpty())
|
|
|
|
*settingsCategory = QLatin1String(Constants::DEBUGGER_SETTINGS_CATEGORY);
|
2009-09-11 08:55:16 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
return success;
|
2010-06-15 09:13:22 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
void DebuggerRunControl::start()
|
2010-06-15 09:13:22 +02:00
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, return);
|
|
|
|
const DebuggerStartParameters &sp = d->m_engine->startParameters();
|
2010-06-15 09:13:22 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
QString errorMessage;
|
|
|
|
QString settingsCategory;
|
|
|
|
QString settingsPage;
|
2010-06-15 09:13:22 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
if (!checkDebugConfiguration(sp.toolChainType,
|
|
|
|
&errorMessage, &settingsCategory, &settingsPage)) {
|
|
|
|
emit appendMessage(this, errorMessage, true);
|
|
|
|
emit finished();
|
|
|
|
Core::ICore::instance()->showWarningWithOptions(tr("Debugger"),
|
|
|
|
errorMessage, QString(), settingsCategory, settingsPage);
|
2010-06-15 12:47:37 +02:00
|
|
|
return;
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
2010-06-15 11:42:49 +02:00
|
|
|
|
2010-11-10 11:39:01 +01:00
|
|
|
debuggerCore()->runControlStarted(this);
|
2010-08-27 16:22:51 +02:00
|
|
|
|
2010-09-29 10:32:41 +02:00
|
|
|
// We might get a synchronous startFailed() notification on Windows,
|
|
|
|
// when launching the process fails. Emit a proper finished() sequence.
|
2010-06-16 11:08:54 +02:00
|
|
|
emit started();
|
2010-09-29 10:32:41 +02:00
|
|
|
d->m_running = true;
|
|
|
|
|
|
|
|
engine()->startDebugger(this);
|
|
|
|
|
|
|
|
if (d->m_running) {
|
|
|
|
emit addToOutputWindowInline(this, tr("Debugging starts"), false);
|
|
|
|
emit addToOutputWindowInline(this, "\n", false);
|
|
|
|
}
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-11-10 11:39:01 +01:00
|
|
|
QString DebuggerRunControl::idString() const
|
|
|
|
{
|
|
|
|
return tr("Starting debugger '%1' for tool chain '%2'...")
|
|
|
|
.arg(d->m_engine->objectName())
|
|
|
|
.arg(toolChainName(d->m_engine->startParameters().toolChainType));
|
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
void DebuggerRunControl::startFailed()
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-08-27 16:07:42 +02:00
|
|
|
emit addToOutputWindowInline(this, tr("Debugging has failed"), false);
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_running = false;
|
2010-06-16 11:08:54 +02:00
|
|
|
emit finished();
|
2010-07-20 18:31:35 +02:00
|
|
|
engine()->handleStartFailed();
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
void DebuggerRunControl::handleFinished()
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-08-27 16:07:42 +02:00
|
|
|
emit addToOutputWindowInline(this, tr("Debugging has finished"), false);
|
2010-10-05 14:30:26 +02:00
|
|
|
if (engine())
|
|
|
|
engine()->handleFinished();
|
2010-11-10 11:39:01 +01:00
|
|
|
debuggerCore()->runControlFinished(this);
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
void DebuggerRunControl::showMessage(const QString &msg, int channel)
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
switch (channel) {
|
|
|
|
case AppOutput:
|
|
|
|
emit addToOutputWindowInline(this, msg, false);
|
|
|
|
break;
|
|
|
|
case AppError:
|
|
|
|
emit addToOutputWindowInline(this, msg, true);
|
|
|
|
break;
|
|
|
|
case AppStuff:
|
|
|
|
emit appendMessage(this, msg, true);
|
|
|
|
break;
|
|
|
|
}
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-08-20 14:19:25 +02:00
|
|
|
bool DebuggerRunControl::aboutToStop() const
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-08-20 14:19:25 +02:00
|
|
|
QTC_ASSERT(isRunning(), return true;)
|
|
|
|
|
2010-09-29 12:48:00 +02:00
|
|
|
const QString question = tr("A debugging session is still in progress. "
|
2010-08-20 14:19:25 +02:00
|
|
|
"Terminating the session in the current"
|
|
|
|
" state can leave the target in an inconsistent state."
|
|
|
|
" Would you still like to terminate it?");
|
|
|
|
|
|
|
|
const QMessageBox::StandardButton answer =
|
2010-11-10 11:39:01 +01:00
|
|
|
QMessageBox::question(debuggerCore()->mainWindow(),
|
2010-08-20 14:19:25 +02:00
|
|
|
tr("Close Debugging Session"), question,
|
|
|
|
QMessageBox::Yes|QMessageBox::No);
|
|
|
|
return answer == QMessageBox::Yes;
|
|
|
|
}
|
|
|
|
|
|
|
|
RunControl::StopResult DebuggerRunControl::stop()
|
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, return StoppedSynchronously);
|
|
|
|
d->m_engine->quitDebugger();
|
2010-08-20 14:19:25 +02:00
|
|
|
return AsynchronousStop;
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
void DebuggerRunControl::debuggingFinished()
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
d->m_running = false;
|
2010-06-16 11:08:54 +02:00
|
|
|
emit finished();
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
bool DebuggerRunControl::isRunning() const
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
return d->m_running;
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-06-22 12:28:05 +02:00
|
|
|
DebuggerState DebuggerRunControl::state() const
|
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, return DebuggerNotReady);
|
|
|
|
return d->m_engine->state();
|
2010-06-22 12:28:05 +02:00
|
|
|
}
|
|
|
|
|
2010-09-13 13:30:35 +02:00
|
|
|
DebuggerEngine *DebuggerRunControl::engine()
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, /**/);
|
|
|
|
return d->m_engine;
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-09-13 11:08:08 +02:00
|
|
|
Internal::GdbEngine *DebuggerRunControl::gdbEngine() const
|
|
|
|
{
|
2010-09-14 08:34:07 +02:00
|
|
|
QTC_ASSERT(d->m_engine, return 0);
|
|
|
|
if (GdbEngine *gdbEngine = qobject_cast<GdbEngine *>(d->m_engine))
|
2010-09-13 11:08:08 +02:00
|
|
|
return gdbEngine;
|
2010-09-14 08:34:07 +02:00
|
|
|
if (QmlCppEngine * const qmlEngine = qobject_cast<QmlCppEngine *>(d->m_engine))
|
2010-09-13 13:30:35 +02:00
|
|
|
if (Internal::GdbEngine *embeddedGdbEngine = qobject_cast<GdbEngine *>(qmlEngine->cppEngine()))
|
2010-09-13 11:08:08 +02:00
|
|
|
return embeddedGdbEngine;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Internal::AbstractGdbAdapter *DebuggerRunControl::gdbAdapter() const
|
|
|
|
{
|
|
|
|
GdbEngine *engine = gdbEngine();
|
|
|
|
QTC_ASSERT(engine, return 0)
|
|
|
|
return engine->gdbAdapter();
|
|
|
|
}
|
|
|
|
|
2010-11-02 10:24:50 +01:00
|
|
|
void DebuggerRunControl::handleRemoteSetupDone(int gdbServerPort, int qmlPort)
|
2010-09-13 11:08:08 +02:00
|
|
|
{
|
2010-09-16 17:08:07 +02:00
|
|
|
if (QmlEngine *qmlEngine = qobject_cast<QmlEngine *>(d->m_engine)) {
|
2010-11-02 10:24:50 +01:00
|
|
|
qmlEngine->handleRemoteSetupDone(qmlPort);
|
2010-09-16 17:08:07 +02:00
|
|
|
} else if (Internal::AbstractGdbAdapter *adapter = gdbAdapter()) {
|
|
|
|
if (RemotePlainGdbAdapter *rpga = qobject_cast<RemotePlainGdbAdapter *>(adapter)) {
|
2010-11-02 10:24:50 +01:00
|
|
|
rpga->handleSetupDone(qmlPort);
|
2010-09-16 17:08:07 +02:00
|
|
|
} else if (RemoteGdbServerAdapter *rgsa = qobject_cast<RemoteGdbServerAdapter *>(adapter)) {
|
2010-11-02 10:24:50 +01:00
|
|
|
rgsa->handleSetupDone(gdbServerPort, qmlPort);
|
2010-09-16 17:08:07 +02:00
|
|
|
} else {
|
|
|
|
QTC_ASSERT(false, /* */ );
|
|
|
|
}
|
2010-09-13 11:08:08 +02:00
|
|
|
} else {
|
|
|
|
QTC_ASSERT(false, /* */ );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-16 17:08:07 +02:00
|
|
|
void DebuggerRunControl::handleRemoteSetupFailed(const QString &message)
|
2010-09-13 11:08:08 +02:00
|
|
|
{
|
2010-09-16 17:08:07 +02:00
|
|
|
if (QmlEngine *qmlEngine = qobject_cast<QmlEngine *>(d->m_engine)) {
|
|
|
|
qmlEngine->handleRemoteSetupFailed(message);
|
|
|
|
} else if (Internal::AbstractGdbAdapter *adapter = gdbAdapter()) {
|
|
|
|
if (RemotePlainGdbAdapter *rpga = qobject_cast<RemotePlainGdbAdapter *>(adapter)) {
|
|
|
|
rpga->handleSetupFailed(message);
|
|
|
|
} else if (RemoteGdbServerAdapter *rgsa = qobject_cast<RemoteGdbServerAdapter *>(adapter)) {
|
|
|
|
rgsa->handleSetupFailed(message);
|
|
|
|
} else {
|
|
|
|
QTC_ASSERT(false, /* */ );
|
|
|
|
}
|
2010-09-13 11:08:08 +02:00
|
|
|
} else {
|
|
|
|
QTC_ASSERT(false, /* */ );
|
|
|
|
}
|
|
|
|
}
|
2010-09-13 13:30:35 +02:00
|
|
|
|
|
|
|
void DebuggerRunControl::emitAddToOutputWindow(const QString &line, bool onStdErr)
|
|
|
|
{
|
|
|
|
emit addToOutputWindow(this, line, onStdErr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebuggerRunControl::emitAppendMessage(const QString &m, bool isError)
|
|
|
|
{
|
|
|
|
emit appendMessage(this, m, isError);
|
|
|
|
}
|
2010-09-14 08:34:07 +02:00
|
|
|
|
|
|
|
RunConfiguration *DebuggerRunControl::runConfiguration() const
|
|
|
|
{
|
|
|
|
return d->m_myRunConfiguration.data();
|
|
|
|
}
|
2010-11-10 11:39:01 +01:00
|
|
|
|
2009-09-11 08:55:16 +02:00
|
|
|
} // namespace Debugger
|