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"
|
|
|
|
#include "debuggerconstants.h"
|
|
|
|
#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"
|
|
|
|
#include "qml/qmlcppengine.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/environment.h>
|
|
|
|
#include <projectexplorer/project.h>
|
2010-08-18 13:54:12 +02:00
|
|
|
#include <projectexplorer/projectexplorer.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
|
|
|
|
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/QDebug>
|
|
|
|
#include <QtCore/QDir>
|
|
|
|
#include <QtCore/QFileInfo>
|
2010-06-15 12:47:37 +02:00
|
|
|
#include <QtCore/QTimer>
|
2010-06-22 12:28:05 +02:00
|
|
|
#include <QtCore/QStringList>
|
2008-12-09 15:25:01 +01:00
|
|
|
|
2010-06-15 09:13:22 +02:00
|
|
|
#include <QtGui/QAbstractItemView>
|
2008-12-03 10:55:18 +01:00
|
|
|
#include <QtGui/QTextDocument>
|
2010-06-15 11:42:49 +02:00
|
|
|
#include <QtGui/QTreeWidget>
|
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-06-16 11:08:54 +02:00
|
|
|
|
2010-06-22 11:42:28 +02:00
|
|
|
bool checkGdbConfiguration(int toolChain, QString *errorMsg, QString *settingsPage);
|
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
|
|
|
|
DebuggerEngine *createCdbEngine(const DebuggerStartParameters &);
|
2010-06-16 11:08:54 +02:00
|
|
|
bool checkCdbConfiguration(int toolChain, QString *errorMsg, QString *settingsPage);
|
2010-06-22 11:42:28 +02:00
|
|
|
#else
|
|
|
|
DebuggerEngine *createCdbEngine(const DebuggerStartParameters &) { return 0; }
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
return ToolChain::toolChainName(ToolChain::ToolChainType(toolChainType));
|
|
|
|
}
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
|
|
|
static DebuggerPlugin *plugin() { return DebuggerPlugin::instance(); }
|
2010-06-15 08:22:02 +02:00
|
|
|
|
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,
|
|
|
|
DebuggerEngineType enabledEngines)
|
|
|
|
: 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;
|
2009-05-25 16:22:11 +02:00
|
|
|
return mode == ProjectExplorer::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-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.executable = rc->executable();
|
|
|
|
sp.environment = rc->environment().toStringList();
|
|
|
|
sp.workingDirectory = rc->workingDirectory();
|
|
|
|
sp.processArgs = rc->commandLineArguments();
|
|
|
|
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-08-27 16:22:51 +02:00
|
|
|
DebuggerLanguages activeLangs = DebuggerUISwitcher::instance()->activeDebugLanguages();
|
|
|
|
if (activeLangs & 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())
|
|
|
|
sp.projectBuildDir = runConfiguration->target()->activeBuildConfiguration()->buildDirectory();
|
|
|
|
|
2010-08-18 13:54:12 +02:00
|
|
|
sp.environment << QString(Constants::E_QML_DEBUG_SERVER_PORT)
|
|
|
|
+ QLatin1Char('=') + QString::number(sp.qmlServerPort);
|
|
|
|
}
|
|
|
|
|
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-06-11 16:06:13 +02:00
|
|
|
if (!qmakePath.isEmpty()) {
|
|
|
|
QProcess proc;
|
|
|
|
QStringList args;
|
|
|
|
args.append(QLatin1String("-query"));
|
|
|
|
args.append(QLatin1String("QT_INSTALL_HEADERS"));
|
|
|
|
proc.start(qmakePath, args);
|
|
|
|
proc.waitForFinished();
|
|
|
|
QByteArray ba = proc.readAllStandardOutput().trimmed();
|
|
|
|
QFileInfo fi(QString::fromLocal8Bit(ba) + "/..");
|
2010-06-14 08:57:15 +02:00
|
|
|
sp.qtInstallPath = fi.absoluteFilePath();
|
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
|
|
|
{
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(mode == ProjectExplorer::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-07-20 16:34:39 +02:00
|
|
|
DebuggerRunControl::DebuggerRunControl(RunConfiguration *runConfiguration,
|
2010-08-18 13:54:12 +02:00
|
|
|
DebuggerEngineType enabledEngines, const DebuggerStartParameters &sp)
|
|
|
|
: RunControl(runConfiguration, ProjectExplorer::Constants::DEBUGMODE)
|
|
|
|
, m_myRunConfiguration(runConfiguration)
|
|
|
|
, m_running(false)
|
|
|
|
, m_started(false)
|
2010-08-25 09:59:24 +02:00
|
|
|
, m_enabledEngines(enabledEngines)
|
2010-08-18 13:54:12 +02:00
|
|
|
|
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();
|
|
|
|
DebuggerEngine *engine = m_engine;
|
|
|
|
m_engine = 0;
|
|
|
|
engine->disconnect();
|
|
|
|
delete engine;
|
|
|
|
}
|
|
|
|
|
2010-08-20 15:23:42 +02:00
|
|
|
const DebuggerStartParameters &DebuggerRunControl::startParameters() const
|
|
|
|
{
|
|
|
|
QTC_ASSERT(m_engine, return *(new DebuggerStartParameters()));
|
|
|
|
return m_engine->startParameters();
|
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
static DebuggerEngineType engineForToolChain(int toolChainType)
|
|
|
|
{
|
|
|
|
switch (toolChainType) {
|
|
|
|
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:
|
2010-09-08 14:25:33 +02:00
|
|
|
case ProjectExplorer::ToolChain::GCC_MAEMO:
|
2010-06-16 11:08:54 +02:00
|
|
|
return GdbEngineType;
|
|
|
|
|
|
|
|
case ProjectExplorer::ToolChain::MSVC:
|
|
|
|
case ProjectExplorer::ToolChain::WINCE:
|
|
|
|
return CdbEngineType;
|
|
|
|
|
|
|
|
case ProjectExplorer::ToolChain::OTHER:
|
|
|
|
case ProjectExplorer::ToolChain::UNKNOWN:
|
|
|
|
case ProjectExplorer::ToolChain::INVALID:
|
|
|
|
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.
|
|
|
|
DebuggerEngineType DebuggerRunControl::engineForExecutable(const QString &executable)
|
2010-06-15 09:13:22 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
if (executable.endsWith(_("qmlviewer"))) {
|
|
|
|
if (m_enabledEngines & QmlEngineType)
|
|
|
|
return QmlEngineType;
|
|
|
|
m_errorMessage = msgEngineNotAvailable("Qml Engine");
|
|
|
|
}
|
2010-06-15 08:22:02 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
if (executable.endsWith(_(".js"))) {
|
|
|
|
if (m_enabledEngines & ScriptEngineType)
|
|
|
|
return ScriptEngineType;
|
|
|
|
m_errorMessage = msgEngineNotAvailable("Script Engine");
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
if (executable.endsWith(_(".py"))) {
|
|
|
|
if (m_enabledEngines & PdbEngineType)
|
|
|
|
return PdbEngineType;
|
|
|
|
m_errorMessage = msgEngineNotAvailable("Pdb Engine");
|
|
|
|
}
|
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-06-22 12:28:05 +02:00
|
|
|
if (!getPDBFiles(executable, &pdbFiles, &m_errorMessage)) {
|
2010-06-16 11:08:54 +02:00
|
|
|
qWarning("Cannot determine type of executable %s: %s",
|
|
|
|
qPrintable(executable), qPrintable(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-06-22 12:28:05 +02:00
|
|
|
if (checkDebugConfiguration(ToolChain::MSVC, &m_errorMessage, 0, &m_settingsIdHint))
|
2010-06-16 11:08:54 +02:00
|
|
|
return CdbEngineType;
|
|
|
|
#else
|
|
|
|
if (m_enabledEngines & GdbEngineType)
|
|
|
|
return GdbEngineType;
|
|
|
|
m_errorMessage = msgEngineNotAvailable("Gdb Engine");
|
|
|
|
#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.
|
|
|
|
DebuggerEngineType DebuggerRunControl::engineForMode(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-06-22 12:28:05 +02:00
|
|
|
if (startMode != AttachToRemote)
|
2010-06-16 11:08:54 +02:00
|
|
|
return CdbEngineType;
|
2010-06-22 12:28:05 +02:00
|
|
|
return GdbEngineType;
|
2010-06-16 11:08:54 +02:00
|
|
|
m_errorMessage = msgEngineNotAvailable("Gdb Engine");
|
|
|
|
return NoEngineType;
|
|
|
|
#else
|
|
|
|
Q_UNUSED(startMode)
|
|
|
|
// m_errorMessage = msgEngineNotAvailable("Gdb Engine");
|
|
|
|
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-08-18 13:54:12 +02:00
|
|
|
DebuggerLanguages activeLangs = DebuggerPlugin::instance()->activeLanguages();
|
2010-08-13 14:56:05 +02:00
|
|
|
bool isQmlExecutable = sp.executable.endsWith(_("qmlviewer")) || sp.executable.endsWith(_("qmlobserver"));
|
|
|
|
#ifdef Q_OS_MAC
|
|
|
|
isQmlExecutable = sp.executable.endsWith(_("QMLViewer.app")) || sp.executable.endsWith(_("QMLObserver.app"));
|
|
|
|
#endif
|
|
|
|
if (isQmlExecutable)
|
2010-06-16 11:08:54 +02:00
|
|
|
engineType = QmlEngineType;
|
|
|
|
else if (sp.executable.endsWith(_(".js")))
|
|
|
|
engineType = ScriptEngineType;
|
|
|
|
else if (sp.executable.endsWith(_(".py")))
|
|
|
|
engineType = PdbEngineType;
|
|
|
|
else
|
|
|
|
engineType = engineForToolChain(sp.toolChainType);
|
|
|
|
|
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-06-16 11:08:54 +02:00
|
|
|
if (engineType == NoEngineType
|
|
|
|
&& sp.startMode != AttachToRemote
|
|
|
|
&& !sp.executable.isEmpty())
|
|
|
|
engineType = engineForExecutable(sp.executable);
|
|
|
|
|
|
|
|
if (!engineType)
|
|
|
|
engineType = engineForMode(sp.startMode);
|
|
|
|
|
2010-08-27 16:22:51 +02:00
|
|
|
if (engineType != QmlEngineType && (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:
|
|
|
|
m_engine = createGdbEngine(sp);
|
2010-09-13 11:08:08 +02:00
|
|
|
initGdbEngine(qobject_cast<Internal::GdbEngine *>(m_engine));
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case ScriptEngineType:
|
|
|
|
m_engine = createScriptEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case CdbEngineType:
|
|
|
|
m_engine = createCdbEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case PdbEngineType:
|
|
|
|
m_engine = createPdbEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case TcfEngineType:
|
|
|
|
m_engine = createTcfEngine(sp);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
case QmlEngineType:
|
|
|
|
m_engine = createQmlEngine(sp);
|
|
|
|
break;
|
2010-08-18 13:54:12 +02:00
|
|
|
case QmlCppEngineType:
|
|
|
|
m_engine = 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-06-16 11:08:54 +02:00
|
|
|
default: {
|
|
|
|
// Could not find anything suitable.
|
2010-06-23 11:55:35 +02:00
|
|
|
debuggingFinished();
|
2010-06-16 11:08:54 +02:00
|
|
|
// 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), m_errorMessage);
|
|
|
|
Core::ICore::instance()->showWarningWithOptions(tr("Warning"),
|
|
|
|
msg, QString(), QLatin1String(Constants::DEBUGGER_SETTINGS_CATEGORY),
|
|
|
|
m_settingsIdHint);
|
2010-06-14 18:19:02 +02:00
|
|
|
break;
|
2010-06-16 11:08:54 +02:00
|
|
|
}
|
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,
|
|
|
|
SIGNAL(gdbAdapterRequestSetup()));
|
|
|
|
} else if (RemoteGdbServerAdapter *rgsa = qobject_cast<RemoteGdbServerAdapter *>(adapter)) {
|
|
|
|
connect(rgsa, SIGNAL(requestSetup()),
|
|
|
|
this, SIGNAL(gdbAdapterRequestSetup()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
QString DebuggerRunControl::displayName() const
|
2010-06-15 08:22:02 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
QTC_ASSERT(m_engine, return QString());
|
|
|
|
return m_engine->startParameters().displayName;
|
2010-06-15 08:22:02 +02:00
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
void DebuggerRunControl::setCustomEnvironment(ProjectExplorer::Environment env)
|
2010-06-15 08:22:02 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
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;
|
|
|
|
|
|
|
|
switch(toolChain) {
|
|
|
|
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:
|
|
|
|
success = checkGdbConfiguration(toolChain, errorMessage, settingsPage);
|
|
|
|
if (!success)
|
2010-09-10 17:21:22 +02:00
|
|
|
*errorMessage += msgEngineNotAvailable("Gdb");
|
2010-06-16 11:08:54 +02:00
|
|
|
break;
|
|
|
|
case ProjectExplorer::ToolChain::MSVC:
|
|
|
|
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-06-16 11:08:54 +02:00
|
|
|
QTC_ASSERT(m_engine, return);
|
|
|
|
const DebuggerStartParameters &sp = 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-08-31 14:05:33 +02:00
|
|
|
plugin()->activateDebugMode();
|
2010-08-27 16:22:51 +02:00
|
|
|
DebuggerUISwitcher::instance()->aboutToStartDebugger();
|
2010-08-24 15:35:46 +02:00
|
|
|
|
|
|
|
const QString message = tr("Starting debugger '%1' for tool chain '%2'...").
|
|
|
|
arg(m_engine->objectName(), toolChainName(sp.toolChainType));
|
|
|
|
plugin()->showMessage(message, StatusBar);
|
2010-06-30 13:15:44 +02:00
|
|
|
plugin()->showMessage(DebuggerSettings::instance()->dump(), LogDebug);
|
2010-08-24 16:22:21 +02:00
|
|
|
plugin()->runControlStarted(this);
|
2010-08-27 16:22:51 +02:00
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
engine()->startDebugger(this);
|
|
|
|
m_running = true;
|
2010-08-27 16:07:42 +02:00
|
|
|
emit addToOutputWindowInline(this, tr("Debugging starts"), false);
|
|
|
|
emit addToOutputWindowInline(this, "\n", false);
|
2010-06-16 11:08:54 +02:00
|
|
|
emit started();
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
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-06-16 11:08:54 +02:00
|
|
|
m_running = false;
|
|
|
|
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-06-22 15:14:44 +02:00
|
|
|
engine()->handleFinished();
|
2010-06-16 11:08:54 +02:00
|
|
|
plugin()->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;)
|
|
|
|
|
|
|
|
const QString question = tr("A debugging session are still in progress. "
|
|
|
|
"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 =
|
|
|
|
QMessageBox::question(DebuggerUISwitcher::instance()->mainWindow(),
|
|
|
|
tr("Close Debugging Session"), question,
|
|
|
|
QMessageBox::Yes|QMessageBox::No);
|
|
|
|
return answer == QMessageBox::Yes;
|
|
|
|
}
|
|
|
|
|
|
|
|
RunControl::StopResult DebuggerRunControl::stop()
|
|
|
|
{
|
|
|
|
QTC_ASSERT(m_engine, return StoppedSynchronously);
|
2010-07-02 12:20:26 +02:00
|
|
|
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-06-16 11:08:54 +02:00
|
|
|
m_running = false;
|
|
|
|
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-06-16 11:08:54 +02:00
|
|
|
return m_running;
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-06-22 12:28:05 +02:00
|
|
|
DebuggerState DebuggerRunControl::state() const
|
|
|
|
{
|
|
|
|
QTC_ASSERT(m_engine, return DebuggerNotReady);
|
|
|
|
return m_engine->state();
|
|
|
|
}
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
Internal::DebuggerEngine *DebuggerRunControl::engine()
|
2010-06-15 11:42:49 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
QTC_ASSERT(m_engine, /**/);
|
|
|
|
return m_engine;
|
2010-06-15 11:42:49 +02:00
|
|
|
}
|
|
|
|
|
2010-09-13 11:08:08 +02:00
|
|
|
Internal::GdbEngine *DebuggerRunControl::gdbEngine() const
|
|
|
|
{
|
|
|
|
QTC_ASSERT(m_engine, return 0);
|
|
|
|
if (GdbEngine *gdbEngine = qobject_cast<GdbEngine *>(m_engine))
|
|
|
|
return gdbEngine;
|
|
|
|
if (QmlCppEngine * const qmlEngine = qobject_cast<QmlCppEngine *>(m_engine))
|
|
|
|
if (GdbEngine *embeddedGdbEngine = qobject_cast<GdbEngine *>(qmlEngine->cppEngine()))
|
|
|
|
return embeddedGdbEngine;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Internal::AbstractGdbAdapter *DebuggerRunControl::gdbAdapter() const
|
|
|
|
{
|
|
|
|
GdbEngine *engine = gdbEngine();
|
|
|
|
QTC_ASSERT(engine, return 0)
|
|
|
|
return engine->gdbAdapter();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebuggerRunControl::remoteGdbHandleSetupDone()
|
|
|
|
{
|
|
|
|
Internal::AbstractGdbAdapter *adapter = gdbAdapter();
|
|
|
|
QTC_ASSERT(adapter, return);
|
|
|
|
if (RemotePlainGdbAdapter *rpga = qobject_cast<RemotePlainGdbAdapter *>(adapter)) {
|
|
|
|
rpga->handleSetupDone();
|
|
|
|
} else if (RemoteGdbServerAdapter *rgsa = qobject_cast<RemoteGdbServerAdapter *>(adapter)) {
|
|
|
|
rgsa->handleSetupDone();
|
|
|
|
} else {
|
|
|
|
QTC_ASSERT(false, /* */ );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebuggerRunControl::remoteGdbHandleSetupFailed(const QString &message)
|
|
|
|
{
|
|
|
|
Internal::AbstractGdbAdapter *adapter = gdbAdapter();
|
|
|
|
QTC_ASSERT(adapter, return);
|
|
|
|
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, /* */ );
|
|
|
|
}
|
|
|
|
}
|
2009-09-11 08:55:16 +02:00
|
|
|
} // namespace Debugger
|