debuggerplugin.cpp 47.2 KB
Newer Older
1
/**************************************************************************
con's avatar
con committed
2 3 4
**
** This file is part of Qt Creator
**
5
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
con's avatar
con committed
6
**
7
** Contact: Nokia Corporation (qt-info@nokia.com)
con's avatar
con committed
8
**
9
** Commercial Usage
10
**
11 12 13 14
** 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.
15
**
16
** GNU Lesser General Public License Usage
17
**
18 19 20 21 22 23
** 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.
24
**
25
** If you are unsure which license is appropriate for your use, please
26
** contact the sales department at http://www.qtsoftware.com/contact.
con's avatar
con committed
27
**
28
**************************************************************************/
hjk's avatar
hjk committed
29

con's avatar
con committed
30 31
#include "debuggerplugin.h"

32
#include "breakhandler.h"
33
#include "debuggeractions.h"
34
#include "debuggerdialogs.h"
con's avatar
con committed
35 36 37 38
#include "debuggerconstants.h"
#include "debuggermanager.h"
#include "debuggerrunner.h"

39
#include "ui_commonoptionspage.h"
hjk's avatar
hjk committed
40
#include "ui_dumperoptionpage.h"
41

42
#include <coreplugin/actionmanager/actionmanager.h>
hjk's avatar
hjk committed
43
#include <coreplugin/basemode.h>
con's avatar
con committed
44
#include <coreplugin/coreconstants.h>
45
#include <coreplugin/dialogs/ioptionspage.h>
con's avatar
con committed
46
#include <coreplugin/editormanager/editormanager.h>
hjk's avatar
hjk committed
47
#include <coreplugin/findplaceholder.h>
con's avatar
con committed
48 49
#include <coreplugin/icore.h>
#include <coreplugin/messagemanager.h>
hjk's avatar
hjk committed
50
#include <coreplugin/minisplitter.h>
con's avatar
con committed
51
#include <coreplugin/modemanager.h>
hjk's avatar
hjk committed
52 53 54
#include <coreplugin/navigationwidget.h>
#include <coreplugin/outputpane.h>
#include <coreplugin/rightpane.h>
con's avatar
con committed
55
#include <coreplugin/uniqueidmanager.h>
hjk's avatar
hjk committed
56

con's avatar
con committed
57
#include <cplusplus/ExpressionUnderCursor.h>
hjk's avatar
hjk committed
58

con's avatar
con committed
59
#include <cppeditor/cppeditorconstants.h>
hjk's avatar
hjk committed
60

61 62
#include <extensionsystem/pluginmanager.h>

63
#include <projectexplorer/projectexplorer.h>
con's avatar
con committed
64 65
#include <projectexplorer/projectexplorerconstants.h>
#include <projectexplorer/session.h>
hjk's avatar
hjk committed
66 67

#include <texteditor/basetexteditor.h>
con's avatar
con committed
68 69 70
#include <texteditor/basetextmark.h>
#include <texteditor/itexteditor.h>
#include <texteditor/texteditorconstants.h>
hjk's avatar
hjk committed
71 72

#include <utils/qtcassert.h>
con's avatar
con committed
73 74 75 76 77

#include <QtCore/QDebug>
#include <QtCore/QObject>
#include <QtCore/QPoint>
#include <QtCore/QSettings>
78
#include <QtCore/QtPlugin>
79
#include <QtCore/QCoreApplication>
80
#include <QtCore/QTimer>
hjk's avatar
hjk committed
81

82
#include <QtGui/QLineEdit>
83
#include <QtGui/QDockWidget>
hjk's avatar
hjk committed
84
#include <QtGui/QMainWindow>
con's avatar
con committed
85 86 87
#include <QtGui/QPlainTextEdit>
#include <QtGui/QTextBlock>
#include <QtGui/QTextCursor>
88
#include <QtGui/QMessageBox>
con's avatar
con committed
89 90

using namespace Core;
hjk's avatar
hjk committed
91 92
using namespace Debugger::Constants;
using namespace Debugger::Internal;
con's avatar
con committed
93
using namespace ProjectExplorer;
hjk's avatar
hjk committed
94
using namespace TextEditor;
con's avatar
con committed
95 96 97 98 99


namespace Debugger {
namespace Constants {

100 101
const char * const M_DEBUG_START_DEBUGGING = "QtCreator.Menu.Debug.StartDebugging";

con's avatar
con committed
102 103
const char * const STARTEXTERNAL        = "Debugger.StartExternal";
const char * const ATTACHEXTERNAL       = "Debugger.AttachExternal";
104
const char * const ATTACHCORE           = "Debugger.AttachCore";
105
const char * const ATTACHTCF            = "Debugger.AttachTcf";
106
const char * const ATTACHREMOTE         = "Debugger.AttachRemote";
107
const char * const DETACH               = "Debugger.Detach";
con's avatar
con committed
108 109 110 111 112 113 114 115 116

const char * const RUN_TO_LINE          = "Debugger.RunToLine";
const char * const RUN_TO_FUNCTION      = "Debugger.RunToFunction";
const char * const JUMP_TO_LINE         = "Debugger.JumpToLine";
const char * const TOGGLE_BREAK         = "Debugger.ToggleBreak";
const char * const BREAK_BY_FUNCTION    = "Debugger.BreakByFunction";
const char * const BREAK_AT_MAIN        = "Debugger.BreakAtMain";
const char * const ADD_TO_WATCH         = "Debugger.AddToWatch";

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
117
#ifdef Q_WS_MAC
con's avatar
con committed
118 119 120 121 122 123 124
const char * const INTERRUPT_KEY            = "Shift+F5";
const char * const RESET_KEY                = "Ctrl+Shift+F5";
const char * const STEP_KEY                 = "F7";
const char * const STEPOUT_KEY              = "Shift+F7";
const char * const NEXT_KEY                 = "F6";
const char * const STEPI_KEY                = "Shift+F9";
const char * const NEXTI_KEY                = "Shift+F6";
125
const char * const REVERSE_KEY              = "";
con's avatar
con committed
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
const char * const RUN_TO_LINE_KEY          = "Shift+F8";
const char * const RUN_TO_FUNCTION_KEY      = "Ctrl+F6";
const char * const JUMP_TO_LINE_KEY         = "Alt+D,Alt+L";
const char * const TOGGLE_BREAK_KEY         = "F8";
const char * const BREAK_BY_FUNCTION_KEY    = "Alt+D,Alt+F";
const char * const BREAK_AT_MAIN_KEY        = "Alt+D,Alt+M";
const char * const ADD_TO_WATCH_KEY         = "Alt+D,Alt+W";
#else
const char * const INTERRUPT_KEY            = "Shift+F5";
const char * const RESET_KEY                = "Ctrl+Shift+F5";
const char * const STEP_KEY                 = "F11";
const char * const STEPOUT_KEY              = "Shift+F11";
const char * const NEXT_KEY                 = "F10";
const char * const STEPI_KEY                = "";
const char * const NEXTI_KEY                = "";
141
const char * const REVERSE_KEY              = "F12";
con's avatar
con committed
142 143 144 145 146 147 148 149 150 151 152 153 154
const char * const RUN_TO_LINE_KEY          = "";
const char * const RUN_TO_FUNCTION_KEY      = "";
const char * const JUMP_TO_LINE_KEY         = "";
const char * const TOGGLE_BREAK_KEY         = "F9";
const char * const BREAK_BY_FUNCTION_KEY    = "";
const char * const BREAK_AT_MAIN_KEY        = "";
const char * const ADD_TO_WATCH_KEY         = "Ctrl+Alt+Q";
#endif

} // namespace Constants
} // namespace Debugger


155 156 157 158 159
static ProjectExplorer::SessionManager *sessionManager()
{
    return ProjectExplorer::ProjectExplorerPlugin::instance()->session();
}

160 161 162 163 164
static QSettings *settings()
{
    return ICore::instance()->settings();
}

165 166 167 168 169 170
///////////////////////////////////////////////////////////////////////
//
// DebugMode
//
///////////////////////////////////////////////////////////////////////

hjk's avatar
hjk committed
171 172 173 174 175 176 177 178 179 180 181 182 183
namespace Debugger {
namespace Internal {

class DebugMode : public Core::BaseMode
{
    Q_OBJECT

public:
    DebugMode(QObject *parent = 0);
    ~DebugMode();
};

DebugMode::DebugMode(QObject *parent)
hjk's avatar
hjk committed
184
  : BaseMode(parent)
hjk's avatar
hjk committed
185
{
hjk's avatar
hjk committed
186 187 188 189
    setName(tr("Debug"));
    setUniqueModeName(Constants::MODE_DEBUG);
    setIcon(QIcon(":/fancyactionbar/images/mode_Debug.png"));
    setPriority(Constants::P_MODE_DEBUG);
hjk's avatar
hjk committed
190 191 192 193 194 195 196 197
}

DebugMode::~DebugMode()
{
    // Make sure the editor manager does not get deleted
    EditorManager::instance()->setParent(0);
}

198 199 200
} // namespace Internal
} // namespace Debugger

hjk's avatar
hjk committed
201

con's avatar
con committed
202 203 204 205 206 207
///////////////////////////////////////////////////////////////////////
//
// LocationMark
//
///////////////////////////////////////////////////////////////////////

208 209 210 211
namespace Debugger {
namespace Internal {

class LocationMark : public TextEditor::BaseTextMark
con's avatar
con committed
212 213 214 215 216 217
{
    Q_OBJECT

public:
    LocationMark(const QString &fileName, int linenumber)
        : BaseTextMark(fileName, linenumber)
218
    {}
con's avatar
con committed
219 220 221 222 223
    ~LocationMark();

    QIcon icon() const;
    void updateLineNumber(int /*lineNumber*/) {}
    void updateBlock(const QTextBlock & /*block*/) {}
224
    void removedFromEditor() {}
con's avatar
con committed
225 226 227 228
};

LocationMark::~LocationMark()
{
229
    //qDebug() << "LOCATIONMARK DESTRUCTOR";
con's avatar
con committed
230 231 232 233
}

QIcon LocationMark::icon() const
{
234
    static const QIcon icon(":/debugger/images/location.svg");
con's avatar
con committed
235 236 237
    return icon;
}

238 239 240 241 242 243
} // namespace Internal
} // namespace Debugger


///////////////////////////////////////////////////////////////////////
//
244
// CommonOptionsPage
245 246 247 248 249 250
//
///////////////////////////////////////////////////////////////////////

namespace Debugger {
namespace Internal {

251
class CommonOptionsPage : public Core::IOptionsPage
252 253 254 255
{
    Q_OBJECT

public:
256
    CommonOptionsPage() {}
257 258

    // IOptionsPage
259 260 261 262 263 264 265 266
    QString id() const
        { return QLatin1String(Debugger::Constants::DEBUGGER_COMMON_SETTINGS_PAGE); }
    QString trName() const
        { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_COMMON_SETTINGS_PAGE); }
    QString category() const
        { return QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY);  }
    QString trCategory() const
        { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY); }
267 268

    QWidget *createPage(QWidget *parent);
269
    void apply() { m_group.apply(settings()); }
270
    void finish() { m_group.finish(); }
271 272

private:
273
    Ui::CommonOptionsPage m_ui;
274
    Core::Utils::SavedActionSet m_group;
275 276
};

277
QWidget *CommonOptionsPage::createPage(QWidget *parent)
278 279 280
{
    QWidget *w = new QWidget(parent);
    m_ui.setupUi(w);
281
    m_group.clear();
282

283
    m_group.insert(theDebuggerAction(ListSourceFiles),
284
        m_ui.checkBoxListSourceFiles);
285
    m_group.insert(theDebuggerAction(UseAlternatingRowColors),
286
        m_ui.checkBoxUseAlternatingRowColors);
287
    m_group.insert(theDebuggerAction(SkipKnownFrames),
288
        m_ui.checkBoxSkipKnownFrames);
289
    m_group.insert(theDebuggerAction(UseToolTips),
290
        m_ui.checkBoxUseToolTips);
291 292 293
    m_group.insert(theDebuggerAction(EnableReverseDebugging), 
        m_ui.checkBoxEnableReverseDebugging);
    m_group.insert(theDebuggerAction(MaximalStackDepth), 
294
        m_ui.spinBoxMaximalStackDepth);
295

296 297 298 299
#ifdef USE_REVERSE_DEBUGGING
    m_ui.checkBoxEnableReverseDebugging->hide();
#endif

300 301 302 303 304 305
    return w;
}

} // namespace Internal
} // namespace Debugger

hjk's avatar
hjk committed
306

hjk's avatar
hjk committed
307 308
///////////////////////////////////////////////////////////////////////
//
309
// DebuggingHelperOptionPage
hjk's avatar
hjk committed
310 311 312 313 314 315
//
///////////////////////////////////////////////////////////////////////

namespace Debugger {
namespace Internal {

316
class DebuggingHelperOptionPage : public Core::IOptionsPage
hjk's avatar
hjk committed
317 318 319 320
{
    Q_OBJECT

public:
321
    DebuggingHelperOptionPage() {}
hjk's avatar
hjk committed
322 323

    // IOptionsPage
324 325
    QString id() const { return QLatin1String("DebuggingHelper"); }
    QString trName() const { return tr("Debugging Helper"); }
326 327
    QString category() const { return QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY); }
    QString trCategory() const { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY); }
hjk's avatar
hjk committed
328 329

    QWidget *createPage(QWidget *parent);
330
    void apply() { m_group.apply(settings()); }
331
    void finish() { m_group.finish(); }
hjk's avatar
hjk committed
332 333

private:
hjk's avatar
hjk committed
334 335
    Q_SLOT void updateState();

hjk's avatar
hjk committed
336
    friend class DebuggerPlugin;
337
    Ui::DebuggingHelperOptionPage m_ui;
hjk's avatar
hjk committed
338

339
    Core::Utils::SavedActionSet m_group;
hjk's avatar
hjk committed
340 341
};

342
QWidget *DebuggingHelperOptionPage::createPage(QWidget *parent)
hjk's avatar
hjk committed
343 344 345 346 347
{
    QWidget *w = new QWidget(parent);
    m_ui.setupUi(w);

    m_ui.dumperLocationChooser->setExpectedKind(Core::Utils::PathChooser::Command);
348
    m_ui.dumperLocationChooser->setPromptDialogTitle(tr("Choose DebuggingHelper Location"));
hjk's avatar
hjk committed
349 350
    m_ui.dumperLocationChooser->setInitialBrowsePathBackup(
        Core::ICore::instance()->resourcePath() + "../../lib");
hjk's avatar
hjk committed
351

352
    connect(m_ui.checkBoxUseDebuggingHelpers, SIGNAL(toggled(bool)),
hjk's avatar
hjk committed
353
        this, SLOT(updateState()));
354
    connect(m_ui.checkBoxUseCustomDebuggingHelperLocation, SIGNAL(toggled(bool)),
hjk's avatar
hjk committed
355
        this, SLOT(updateState()));
356

357
    m_group.clear();
358
#ifdef QT_DEBUG
359 360
    m_group.insert(theDebuggerAction(UseDebuggingHelpers),
        m_ui.checkBoxUseDebuggingHelpers);
361
#endif
362 363 364
    m_group.insert(theDebuggerAction(UseCustomDebuggingHelperLocation),
        m_ui.checkBoxUseCustomDebuggingHelperLocation);
    m_group.insert(theDebuggerAction(CustomDebuggingHelperLocation),
365
        m_ui.dumperLocationChooser);
hjk's avatar
hjk committed
366

367 368
    m_group.insert(theDebuggerAction(DebugDebuggingHelpers),
        m_ui.checkBoxDebugDebuggingHelpers);
hjk's avatar
hjk committed
369 370

    m_ui.dumperLocationChooser->
371
        setEnabled(theDebuggerAction(UseCustomDebuggingHelperLocation)->value().toBool());
hjk's avatar
hjk committed
372 373 374 375 376 377 378 379 380 381

#ifndef QT_DEBUG
#if 0
    cmd = am->registerAction(m_manager->m_dumpLogAction,
        Constants::DUMP_LOG, globalcontext);
    //cmd->setDefaultKeySequence(QKeySequence(tr("Ctrl+D,Ctrl+L")));
    cmd->setDefaultKeySequence(QKeySequence(tr("Ctrl+Shift+F11")));
    mdebug->addAction(cmd);
#endif
#endif
382
    updateState();
hjk's avatar
hjk committed
383 384 385 386

    return w;
}

387
void DebuggingHelperOptionPage::updateState()
hjk's avatar
hjk committed
388
{
389 390
    m_ui.checkBoxUseCustomDebuggingHelperLocation->setEnabled(
        m_ui.checkBoxUseDebuggingHelpers->isChecked());
391 392 393 394
    bool locationEnabled = m_ui.checkBoxUseDebuggingHelpers->isChecked()
         && m_ui.checkBoxUseCustomDebuggingHelperLocation->isChecked();
    m_ui.dumperLocationChooser->setEnabled(locationEnabled);
    m_ui.dumperLocationLabel->setEnabled(locationEnabled);
hjk's avatar
hjk committed
395 396
}

hjk's avatar
hjk committed
397 398 399
} // namespace Internal
} // namespace Debugger

con's avatar
con committed
400 401 402 403 404 405
///////////////////////////////////////////////////////////////////////
//
// DebuggerPlugin
//
///////////////////////////////////////////////////////////////////////

406 407
DebuggerPlugin::DebuggerPlugin()
  : m_manager(0),
408 409 410
    m_debugMode(0),
    m_locationMark(0),
    m_gdbRunningContext(0),
411 412
    m_cmdLineEnabledEngines(AllEngineTypes),
    m_cmdLineAttachPid(0),
413
    m_cmdLineWinCrashEvent(0),
414 415
    m_toggleLockedAction(0)
{}
con's avatar
con committed
416 417 418 419 420 421

DebuggerPlugin::~DebuggerPlugin()
{}

void DebuggerPlugin::shutdown()
{
hjk's avatar
hjk committed
422
    QTC_ASSERT(m_manager, /**/);
con's avatar
con committed
423 424 425
    if (m_manager)
        m_manager->shutdown();

hjk's avatar
hjk committed
426
    writeSettings();
dt's avatar
dt committed
427
    delete DebuggerSettings::instance();
hjk's avatar
hjk committed
428

con's avatar
con committed
429 430 431 432 433 434 435 436 437 438 439 440 441 442 443
    //qDebug() << "DebuggerPlugin::~DebuggerPlugin";
    removeObject(m_debugMode);

    // FIXME: when using the line below, BreakWindow etc gets deleted twice.
    // so better leak for now...
    delete m_debugMode;
    m_debugMode = 0;

    delete m_locationMark;
    m_locationMark = 0;

    delete m_manager;
    m_manager = 0;
}

444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474
static inline QString msgParameterMissing(const QString &a)
{
    return DebuggerPlugin::tr("Option '%1' is missing the parameter.").arg(a);
}

static inline QString msgInvalidNumericParameter(const QString &a, const QString &number)
{
    return DebuggerPlugin::tr("The parameter '%1' of option '%2' is not a number.").arg(number, a);
}

// Parse arguments
bool DebuggerPlugin::parseArgument(QStringList::const_iterator &it,
                                   const QStringList::const_iterator& cend,
                                   QString *errorMessage)
{
    const QString &option = *it;
    // '-debug <pid>'
    if (*it == QLatin1String("-debug")) {
        ++it;
        if (it == cend) {
            *errorMessage = msgParameterMissing(*it);
            return false;
        }
        bool ok;
        m_cmdLineAttachPid = it->toULongLong(&ok);
        if (!ok) {
            *errorMessage = msgInvalidNumericParameter(option, *it);
            return false;
        }
        return true;
    }
475 476 477
    // -wincrashevent <event-handle>. A handle used for
    // a handshake when attaching to a crashed Windows process.
    if (*it == QLatin1String("-wincrashevent")) {
478 479 480 481 482 483
        ++it;
        if (it == cend) {
            *errorMessage = msgParameterMissing(*it);
            return false;
        }
        bool ok;
484
        m_cmdLineWinCrashEvent = it->toULongLong(&ok);
485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523
        if (!ok) {
            *errorMessage = msgInvalidNumericParameter(option, *it);
            return false;
        }
        return true;
    }
    // engine disabling
    if (option == QLatin1String("-disable-cdb")) {
        m_cmdLineEnabledEngines &= ~CdbEngineType;
        return true;
    }
    if (option == QLatin1String("-disable-gdb")) {
        m_cmdLineEnabledEngines &= ~GdbEngineType;
        return true;
    }
    if (option == QLatin1String("-disable-sdb")) {
        m_cmdLineEnabledEngines &= ~ScriptEngineType;
        return true;
    }
    if (option == QLatin1String("-disable-tcf")) {
        m_cmdLineEnabledEngines &= ~TcfEngineType;
        return true;
    }

    *errorMessage = tr("Invalid debugger option: %1").arg(option);
    return false;
}

bool DebuggerPlugin::parseArguments(const QStringList &args, QString *errorMessage)
{
    const QStringList::const_iterator cend = args.constEnd();
    for (QStringList::const_iterator it = args.constBegin(); it != cend; ++it)
        if (!parseArgument(it, cend, errorMessage))
            return false;
    if (Debugger::Constants::Internal::debug)
        qDebug().nospace() << args << "engines=0x" << QString::number(m_cmdLineEnabledEngines, 16) << " pid" << m_cmdLineAttachPid << '\n';
    return true;
}

hjk's avatar
hjk committed
524
bool DebuggerPlugin::initialize(const QStringList &arguments, QString *errorMessage)
con's avatar
con committed
525
{
526 527 528 529 530 531
    // Do not fail the whole plugin if something goes wrong here
    if (!parseArguments(arguments, errorMessage)) {
        *errorMessage = tr("Error evaluating command line arguments: %1").arg(*errorMessage);
        qWarning("%s\n", qPrintable(*errorMessage));
        errorMessage->clear();
    }
con's avatar
con committed
532

533
    m_manager = new DebuggerManager;
534
    const QList<Core::IOptionsPage *> engineOptionPages = m_manager->initializeEngines(m_cmdLineEnabledEngines);
con's avatar
con committed
535

536
    ICore *core = ICore::instance();
hjk's avatar
hjk committed
537
    QTC_ASSERT(core, return false);
con's avatar
con committed
538

539 540
    Core::ActionManager *am = core->actionManager();
    QTC_ASSERT(am, return false);
con's avatar
con committed
541 542

    Core::UniqueIDManager *uidm = core->uniqueIDManager();
hjk's avatar
hjk committed
543
    QTC_ASSERT(uidm, return false);
con's avatar
con committed
544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561

    QList<int> globalcontext;
    globalcontext << Core::Constants::C_GLOBAL_ID;

    QList<int> cppcontext;
    cppcontext << uidm->uniqueIdentifier(ProjectExplorer::Constants::LANG_CXX);

    QList<int> debuggercontext;
    debuggercontext << uidm->uniqueIdentifier(C_GDBDEBUGGER);

    QList<int> cppeditorcontext;
    cppeditorcontext << uidm->uniqueIdentifier(CppEditor::Constants::C_CPPEDITOR);

    QList<int> texteditorcontext;
    texteditorcontext << uidm->uniqueIdentifier(TextEditor::Constants::C_TEXTEDITOR);

    m_gdbRunningContext = uidm->uniqueIdentifier(Constants::GDBRUNNING);

562
    //Core::ActionContainer *mcppcontext =
563
    //    am->actionContainer(CppEditor::Constants::M_CONTEXT);
con's avatar
con committed
564

565 566 567 568 569 570 571 572 573 574 575 576 577 578 579
    // External apps
    m_startExternalAction = new QAction(this);
    m_startExternalAction->setText(tr("Start and Debug External Application..."));
    connect(m_startExternalAction, SIGNAL(triggered()),
        this, SLOT(startExternalApplication()));

    m_attachExternalAction = new QAction(this);
    m_attachExternalAction->setText(tr("Attach to Running External Application..."));
    connect(m_attachExternalAction, SIGNAL(triggered()),
        this, SLOT(attachExternalApplication()));

    m_attachCoreAction = new QAction(this);
    m_attachCoreAction->setText(tr("Attach to Core..."));
    connect(m_attachCoreAction, SIGNAL(triggered()), this, SLOT(attachCore()));

580
    m_attachTcfAction = new QAction(this);
581
    m_attachTcfAction->setText(tr("Attach to Running Tcf Agent..."));
582 583 584 585 586 587
    m_attachTcfAction->setToolTip(tr("This attaches to a running "
        "'Target Communication Framework' agent."));
    connect(m_attachTcfAction, SIGNAL(triggered()),
        this, SLOT(attachRemoteTcf()));


588 589 590 591
    m_startRemoteAction = new QAction(this);
    m_startRemoteAction->setText(tr("Start and Attach to Remote Application..."));
    connect(m_startRemoteAction, SIGNAL(triggered()),
        this, SLOT(startRemoteApplication()));
592 593


594
    m_detachAction = new QAction(this);
595
    m_detachAction->setText(tr("Detach Debugger"));
596 597 598
    connect(m_detachAction, SIGNAL(triggered()),
        m_manager, SLOT(detachDebugger()));

599
    Core::ActionContainer *mdebug =
600
        am->actionContainer(ProjectExplorer::Constants::M_DEBUG);
con's avatar
con committed
601

602 603 604
    Core::ActionContainer *mstart =
        am->actionContainer(ProjectExplorer::Constants::M_DEBUG_STARTDEBUGGING);

con's avatar
con committed
605
    Core::Command *cmd = 0;
606 607
    cmd = am->registerAction(m_manager->m_continueAction,
        ProjectExplorer::Constants::DEBUG, QList<int>() << m_gdbRunningContext);
608
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
609

610
    cmd = am->registerAction(m_startExternalAction,
con's avatar
con committed
611
        Constants::STARTEXTERNAL, globalcontext);
612
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
613

614
    cmd = am->registerAction(m_attachExternalAction,
con's avatar
con committed
615
        Constants::ATTACHEXTERNAL, globalcontext);
616
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
617

618 619
    cmd = am->registerAction(m_attachCoreAction,
        Constants::ATTACHCORE, globalcontext);
620
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
621
/*
622 623 624
    cmd = am->registerAction(m_attachTcfAction,
        Constants::ATTACHTCF, globalcontext);
    mdebug->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
625
*/
626

627
    cmd = am->registerAction(m_startRemoteAction,
628
        Constants::ATTACHREMOTE, globalcontext);
629
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
630

631 632 633
    cmd = am->registerAction(m_detachAction,
        Constants::DETACH, globalcontext);
    mdebug->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
634

635
    cmd = am->registerAction(m_manager->m_stopAction,
con's avatar
con committed
636
        Constants::INTERRUPT, globalcontext);
con's avatar
con committed
637 638
    cmd->setAttribute(Core::Command::CA_UpdateText);
    cmd->setAttribute(Core::Command::CA_UpdateIcon);
con's avatar
con committed
639 640 641 642
    cmd->setDefaultKeySequence(QKeySequence(Constants::INTERRUPT_KEY));
    cmd->setDefaultText(tr("Stop Debugger/Interrupt Debugger"));
    mdebug->addAction(cmd, Core::Constants::G_DEFAULT_ONE);

643
    cmd = am->registerAction(m_manager->m_resetAction,
con's avatar
con committed
644
        Constants::RESET, globalcontext);
con's avatar
con committed
645
    cmd->setAttribute(Core::Command::CA_UpdateText);
con's avatar
con committed
646 647 648 649 650 651
    cmd->setDefaultKeySequence(QKeySequence(Constants::RESET_KEY));
    cmd->setDefaultText(tr("Reset Debugger"));
    //disabled mdebug->addAction(cmd, Core::Constants::G_DEFAULT_ONE);

    QAction *sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
652
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Step"), globalcontext);
con's avatar
con committed
653 654
    mdebug->addAction(cmd);

655
    cmd = am->registerAction(m_manager->m_nextAction,
con's avatar
con committed
656 657 658 659
        Constants::NEXT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::NEXT_KEY));
    mdebug->addAction(cmd);

660
    cmd = am->registerAction(m_manager->m_stepAction,
con's avatar
con committed
661 662 663 664
        Constants::STEP, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEP_KEY));
    mdebug->addAction(cmd);

665
    cmd = am->registerAction(m_manager->m_stepOutAction,
con's avatar
con committed
666 667 668 669
        Constants::STEPOUT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEPOUT_KEY));
    mdebug->addAction(cmd);

670
    cmd = am->registerAction(m_manager->m_nextIAction,
con's avatar
con committed
671 672 673 674
        Constants::NEXTI, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::NEXTI_KEY));
    mdebug->addAction(cmd);

675
    cmd = am->registerAction(m_manager->m_stepIAction,
con's avatar
con committed
676 677 678 679
        Constants::STEPI, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEPI_KEY));
    mdebug->addAction(cmd);

680
    cmd = am->registerAction(m_manager->m_runToLineAction,
con's avatar
con committed
681 682 683 684
        Constants::RUN_TO_LINE, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::RUN_TO_LINE_KEY));
    mdebug->addAction(cmd);

685
    cmd = am->registerAction(m_manager->m_runToFunctionAction,
con's avatar
con committed
686 687 688 689
        Constants::RUN_TO_FUNCTION, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::RUN_TO_FUNCTION_KEY));
    mdebug->addAction(cmd);

690
    cmd = am->registerAction(m_manager->m_jumpToLineAction,
con's avatar
con committed
691 692 693
        Constants::JUMP_TO_LINE, debuggercontext);
    mdebug->addAction(cmd);

694
#ifdef USE_REVERSE_DEBUGGING
695 696 697 698
    cmd = am->registerAction(m_manager->m_reverseDirectionAction,
        Constants::REVERSE, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::REVERSE_KEY));
    mdebug->addAction(cmd);
699
#endif
700

con's avatar
con committed
701 702
    sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
703
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Break"), globalcontext);
con's avatar
con committed
704 705
    mdebug->addAction(cmd);

706
    cmd = am->registerAction(m_manager->m_breakAction,
con's avatar
con committed
707 708 709 710 711 712 713
        Constants::TOGGLE_BREAK, cppeditorcontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::TOGGLE_BREAK_KEY));
    mdebug->addAction(cmd);
    //mcppcontext->addAction(cmd);

    sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
714
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Watch"), globalcontext);
con's avatar
con committed
715 716
    mdebug->addAction(cmd);

717
    cmd = am->registerAction(m_manager->m_watchAction,
con's avatar
con committed
718 719 720 721
        Constants::ADD_TO_WATCH, cppeditorcontext);
    //cmd->setDefaultKeySequence(QKeySequence(tr("ALT+D,ALT+W")));
    mdebug->addAction(cmd);

722
    // Views menu
con's avatar
con committed
723
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Views"), globalcontext);
724
    mdebug->addAction(cmd);
725
    ActionContainer *viewsMenu = am->createMenu(Constants::M_DEBUG_VIEWS);
726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749
    QMenu *m = viewsMenu->menu();
    m->setEnabled(true);
    m->setTitle(tr("&Views"));
    mdebug->addMenu(viewsMenu, Core::Constants::G_DEFAULT_THREE);

    m_toggleLockedAction = new QAction(tr("Locked"), this);
    m_toggleLockedAction->setCheckable(true);
    m_toggleLockedAction->setChecked(true);
    connect(m_toggleLockedAction, SIGNAL(toggled(bool)),
        m_manager, SLOT(setLocked(bool)));
    foreach (QDockWidget *dockWidget, m_manager->dockWidgets()) {
        cmd = am->registerAction(dockWidget->toggleViewAction(),
            "Debugger." + dockWidget->objectName(), debuggercontext);
        viewsMenu->addAction(cmd);
        //m->addAction(dockWidget->toggleViewAction());
    }
    m->addSeparator();
    m->addAction(m_toggleLockedAction);
    m->addSeparator();

    QAction *resetToSimpleAction = viewsMenu->menu()->addAction(tr("Reset to default layout"));
    connect(resetToSimpleAction, SIGNAL(triggered()),
        m_manager, SLOT(setSimpleDockWidgetArrangement()));

750 751 752 753 754 755 756 757 758 759 760 761 762
    connect(theDebuggerAction(FormatHexadecimal), SIGNAL(triggered()),
        m_manager, SLOT(reloadRegisters()));
    connect(theDebuggerAction(FormatDecimal), SIGNAL(triggered()),
        m_manager, SLOT(reloadRegisters()));
    connect(theDebuggerAction(FormatOctal), SIGNAL(triggered()),
        m_manager, SLOT(reloadRegisters()));
    connect(theDebuggerAction(FormatBinary), SIGNAL(triggered()),
        m_manager, SLOT(reloadRegisters()));
    connect(theDebuggerAction(FormatRaw), SIGNAL(triggered()),
        m_manager, SLOT(reloadRegisters()));
    connect(theDebuggerAction(FormatNatural), SIGNAL(triggered()),
        m_manager, SLOT(reloadRegisters()));

763
   // FIXME:
764
    addAutoReleasedObject(new CommonOptionsPage);
765
    addAutoReleasedObject(new DebuggingHelperOptionPage);
766 767
    foreach (Core::IOptionsPage* op, engineOptionPages)
        addAutoReleasedObject(op);
con's avatar
con committed
768 769 770

    m_locationMark = 0;

hjk's avatar
hjk committed
771 772 773 774 775

    //
    // Debug mode setup
    //
    m_debugMode = new DebugMode(this);
con's avatar
con committed
776 777
    //addAutoReleasedObject(m_debugMode);

778 779 780
    // register factory of DebuggerRunControl
    m_debuggerRunner = new DebuggerRunner(m_manager);
    addAutoReleasedObject(m_debuggerRunner);
con's avatar
con committed
781

hjk's avatar
hjk committed
782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822
    QList<int> context;
    context.append(uidm->uniqueIdentifier(Core::Constants::C_EDITORMANAGER));
    context.append(uidm->uniqueIdentifier(Debugger::Constants::C_GDBDEBUGGER));
    context.append(uidm->uniqueIdentifier(Core::Constants::C_NAVIGATION_PANE));
    m_debugMode->setContext(context);

    QBoxLayout *editorHolderLayout = new QVBoxLayout;
    editorHolderLayout->setMargin(0);
    editorHolderLayout->setSpacing(0);
    editorHolderLayout->addWidget(new EditorManagerPlaceHolder(m_debugMode));
    editorHolderLayout->addWidget(new FindToolBarPlaceHolder(m_debugMode));

    QWidget *editorAndFindWidget = new QWidget;
    editorAndFindWidget->setLayout(editorHolderLayout);

    MiniSplitter *rightPaneSplitter = new MiniSplitter;
    rightPaneSplitter->addWidget(editorAndFindWidget);
    rightPaneSplitter->addWidget(new RightPanePlaceHolder(m_debugMode));
    rightPaneSplitter->setStretchFactor(0, 1);
    rightPaneSplitter->setStretchFactor(1, 0);

    QWidget *centralWidget = new QWidget;

    m_manager->mainWindow()->setCentralWidget(centralWidget);

    MiniSplitter *splitter = new MiniSplitter;
    splitter->addWidget(m_manager->mainWindow());
    splitter->addWidget(new OutputPanePlaceHolder(m_debugMode));
    splitter->setStretchFactor(0, 10);
    splitter->setStretchFactor(1, 0);
    splitter->setOrientation(Qt::Vertical);

    MiniSplitter *splitter2 = new MiniSplitter;
    splitter2->addWidget(new NavigationWidgetPlaceHolder(m_debugMode));
    splitter2->addWidget(splitter);
    splitter2->setStretchFactor(0, 0);
    splitter2->setStretchFactor(1, 1);

    m_debugMode->setWidget(splitter2);

    QToolBar *debugToolBar = new QToolBar;
823
    debugToolBar->setProperty("topBorder", true);
hjk's avatar
hjk committed
824 825 826 827 828 829 830 831
    debugToolBar->addAction(am->command(ProjectExplorer::Constants::DEBUG)->action());
    debugToolBar->addAction(am->command(Constants::INTERRUPT)->action());
    debugToolBar->addAction(am->command(Constants::NEXT)->action());
    debugToolBar->addAction(am->command(Constants::STEP)->action());
    debugToolBar->addAction(am->command(Constants::STEPOUT)->action());
    debugToolBar->addSeparator();
    debugToolBar->addAction(am->command(Constants::STEPI)->action());
    debugToolBar->addAction(am->command(Constants::NEXTI)->action());
832
#ifdef USE_REVERSE_DEBUGGING
hjk's avatar
hjk committed
833
    debugToolBar->addSeparator();
834
    debugToolBar->addAction(am->command(Constants::REVERSE)->action());
835
#endif
836
    debugToolBar->addSeparator();
hjk's avatar
hjk committed
837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859
    debugToolBar->addWidget(new QLabel(tr("Threads:")));

    QComboBox *threadBox = new QComboBox;
    threadBox->setModel(m_manager->threadsModel());
    connect(threadBox, SIGNAL(activated(int)),
        m_manager->threadsWindow(), SIGNAL(threadSelected(int)));
    debugToolBar->addWidget(threadBox);
    debugToolBar->addWidget(m_manager->statusLabel());

    QWidget *stretch = new QWidget;
    stretch->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Minimum);
    debugToolBar->addWidget(stretch);

    QBoxLayout *toolBarAddingLayout = new QVBoxLayout(centralWidget);
    toolBarAddingLayout->setMargin(0);
    toolBarAddingLayout->setSpacing(0);
    toolBarAddingLayout->addWidget(rightPaneSplitter);
    toolBarAddingLayout->addWidget(debugToolBar);

    m_manager->setSimpleDockWidgetArrangement();
    readSettings();

    connect(ModeManager::instance(), SIGNAL(currentModeChanged(Core::IMode*)),
860
            this, SLOT(onModeChanged(Core::IMode*)));
hjk's avatar
hjk committed
861 862 863 864 865 866 867
    m_debugMode->widget()->setFocusProxy(EditorManager::instance());
    addObject(m_debugMode);

    //
    //  Connections
    //

con's avatar
con committed
868
    // ProjectExplorer
869
    connect(sessionManager(), SIGNAL(sessionLoaded()),
con's avatar
con committed
870
       m_manager, SLOT(sessionLoaded()));
871
    connect(sessionManager(), SIGNAL(aboutToSaveSession()),
con's avatar
con committed
872
       m_manager, SLOT(aboutToSaveSession()));
873 874
    connect(sessionManager(), SIGNAL(sessionUnloaded()),
       m_manager, SLOT(sessionUnloaded()));
con's avatar
con committed
875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901

    // EditorManager
    QObject *editorManager = core->editorManager();
    connect(editorManager, SIGNAL(editorAboutToClose(Core::IEditor*)),
        this, SLOT(editorAboutToClose(Core::IEditor*)));
    connect(editorManager, SIGNAL(editorOpened(Core::IEditor*)),
        this, SLOT(editorOpened(Core::IEditor*)));

    // Application interaction
    connect(m_manager, SIGNAL(currentTextEditorRequested(QString*,int*,QObject**)),
        this, SLOT(queryCurrentTextEditor(QString*,int*,QObject**)));

    connect(m_manager, SIGNAL(setSessionValueRequested(QString,QVariant)),
        this, SLOT(setSessionValue(QString,QVariant)));
    connect(m_manager, SIGNAL(sessionValueRequested(QString,QVariant*)),
        this, SLOT(querySessionValue(QString,QVariant*)));

    connect(m_manager, SIGNAL(resetLocationRequested()),
        this, SLOT(resetLocation()));
    connect(m_manager, SIGNAL(gotoLocationRequested(QString,int,bool)),
        this, SLOT(gotoLocation(QString,int,bool)));
    connect(m_manager, SIGNAL(statusChanged(int)),
        this, SLOT(changeStatus(int)));
    connect(m_manager, SIGNAL(previousModeRequested()),
        this, SLOT(activatePreviousMode()));
    connect(m_manager, SIGNAL(debugModeRequested()),
        this, SLOT(activateDebugMode()));
902 903 904
    connect(m_manager, SIGNAL(statusChanged(int)),
        this, SLOT(updateActions(int)));

con's avatar
con committed
905

906
    connect(theDebuggerAction(SettingsDialog), SIGNAL(triggered()),
907 908
        this, SLOT(showSettingsDialog()));

con's avatar
con committed
909 910 911 912 913
    return true;
}

void DebuggerPlugin::extensionsInitialized()
{
hjk's avatar
hjk committed
914
    // time gdb -i mi -ex 'debuggerplugin.cpp:800' -ex r -ex q bin/qtcreator.bin
915
    const QByteArray env = qgetenv("QTC_DEBUGGER_TEST");
hjk's avatar
hjk committed
916
    //qDebug() << "EXTENSIONS INITIALIZED:" << env;
hjk's avatar
hjk committed
917 918
    if (!env.isEmpty())
        m_manager->runTest(QString::fromLocal8Bit(env));
919 920 921 922 923 924
    if (m_cmdLineAttachPid)
        QTimer::singleShot(0, this, SLOT(attachCmdLinePid()));
}

void DebuggerPlugin::attachCmdLinePid()
{
925 926 927
    m_manager->showStatusMessage(tr("Attaching to PID %1.").arg(m_cmdLineAttachPid));
    const QString crashParameter = m_cmdLineWinCrashEvent ? QString::number(m_cmdLineWinCrashEvent) : QString();
    attachExternalApplication(m_cmdLineAttachPid, crashParameter);
con's avatar
con committed
928 929 930 931 932
}

/*! Activates the previous mode when the current mode is the debug mode. */
void DebuggerPlugin::activatePreviousMode()
{
933
    Core::ModeManager *const modeManager = ICore::instance()->modeManager();
con's avatar
con committed
934 935 936 937 938 939 940 941 942 943

    if (modeManager->currentMode() == modeManager->mode(Constants::MODE_DEBUG)
            && !m_previousMode.isEmpty()) {
        modeManager->activateMode(m_previousMode);
        m_previousMode.clear();
    }
}

void DebuggerPlugin::activateDebugMode()
{
944
    ModeManager *modeManager = ModeManager::instance();
con's avatar
con committed
945 946 947 948 949 950
    m_previousMode = QLatin1String(modeManager->currentMode()->uniqueModeName());
    modeManager->activateMode(QLatin1String(MODE_DEBUG));
}

void DebuggerPlugin::queryCurrentTextEditor(QString *fileName, int *lineNumber, QObject **object)
{
951 952
    EditorManager *editorManager = EditorManager::instance();
    if (!editorManager)
con's avatar
con committed
953
        return;
954
    Core::IEditor *editor = editorManager->currentEditor();
con's avatar
con committed
955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972
    ITextEditor *textEditor = qobject_cast<ITextEditor*>(editor);
    if (!textEditor)
        return;
    if (fileName)
        *fileName = textEditor->file()->fileName();
    if (lineNumber)
        *lineNumber = textEditor->currentLine();
    if (object)
        *object = textEditor->widget();
}

void DebuggerPlugin::editorOpened(Core::IEditor *editor)
{
    if (ITextEditor *textEditor = qobject_cast<ITextEditor *>(editor)) {
        connect(textEditor, SIGNAL(markRequested(TextEditor::ITextEditor*,int)),
            this, SLOT(requestMark(TextEditor::ITextEditor*,int)));
        connect(editor, SIGNAL(tooltipRequested(TextEditor::ITextEditor*,QPoint,int)),
            this, SLOT(showToolTip(TextEditor::ITextEditor*,QPoint,int)));
hjk's avatar
hjk committed
973 974
        connect(textEditor, SIGNAL(markContextMenuRequested(TextEditor::ITextEditor*,int,QMenu*)),
            this, SLOT(requestContextMenu(TextEditor::ITextEditor*,int,QMenu*)));
con's avatar
con committed
975 976 977 978 979 980 981 982 983 984
    }
}

void DebuggerPlugin::editorAboutToClose(Core::IEditor *editor)
{
    if (ITextEditor *textEditor = qobject_cast<ITextEditor *>(editor)) {
        disconnect(textEditor, SIGNAL(markRequested(TextEditor::ITextEditor*,int)),
            this, SLOT(requestMark(TextEditor::ITextEditor*,int)));
        disconnect(editor, SIGNAL(tooltipRequested(TextEditor::ITextEditor*,QPoint,int)),
            this, SLOT(showToolTip(TextEditor::ITextEditor*,QPoint,int)));
hjk's avatar
hjk committed
985 986
        disconnect(textEditor, SIGNAL(markContextMenuRequested(TextEditor::ITextEditor*,int,QMenu*)),
            this, SLOT(requestContextMenu(TextEditor::ITextEditor*,int,QMenu*)));
con's avatar
con committed
987 988 989
    }
}

hjk's avatar
hjk committed
990 991 992
void DebuggerPlugin::requestContextMenu(TextEditor::ITextEditor *editor,
    int lineNumber, QMenu *menu)
{
993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030
    QString fileName = editor->file()->fileName();
    QString position = fileName + QString(":%1").arg(lineNumber);
    BreakpointData *data = m_manager->findBreakpoint(fileName, lineNumber);

    if (data) {
        // existing breakpoint
        QAction *act = new QAction(tr