debuggerplugin.cpp 49.5 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
hjk's avatar
hjk committed
26
** contact the sales department at http://qt.nokia.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
#include "debuggerconstants.h"
#include "debuggermanager.h"
#include "debuggerrunner.h"
38
#include "debuggerstringutils.h"
39
40
#include "debuggeruiswitcher.h"
#include "debuggermainwindow.h"
con's avatar
con committed
41

42
#include "ui_commonoptionspage.h"
hjk's avatar
hjk committed
43
#include "ui_dumperoptionpage.h"
44

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

con's avatar
con committed
61
#include <cplusplus/ExpressionUnderCursor.h>
hjk's avatar
hjk committed
62

con's avatar
con committed
63
#include <cppeditor/cppeditorconstants.h>
hjk's avatar
hjk committed
64

65
66
#include <extensionsystem/pluginmanager.h>

67
#include <coreplugin/manhattanstyle.h>
68
#include <projectexplorer/projectexplorer.h>
con's avatar
con committed
69
70
#include <projectexplorer/projectexplorerconstants.h>
#include <projectexplorer/session.h>
71
#include <projectexplorer/project.h>
hjk's avatar
hjk committed
72
73

#include <texteditor/basetexteditor.h>
con's avatar
con committed
74
75
76
#include <texteditor/basetextmark.h>
#include <texteditor/itexteditor.h>
#include <texteditor/texteditorconstants.h>
77
#include <texteditor/texteditorplugin.h>
hjk's avatar
hjk committed
78
79

#include <utils/qtcassert.h>
80
#include <utils/styledbar.h>
con's avatar
con committed
81
82
83
84
85

#include <QtCore/QDebug>
#include <QtCore/QObject>
#include <QtCore/QPoint>
#include <QtCore/QSettings>
86
#include <QtCore/QtPlugin>
87
#include <QtCore/QCoreApplication>
88
#include <QtCore/QTimer>
hjk's avatar
hjk committed
89

90
#include <QtGui/QLineEdit>
91
#include <QtGui/QDockWidget>
con's avatar
con committed
92
93
94
#include <QtGui/QPlainTextEdit>
#include <QtGui/QTextBlock>
#include <QtGui/QTextCursor>
95
#include <QtGui/QToolButton>
96
#include <QtGui/QMessageBox>
con's avatar
con committed
97

98
99
#include <climits>

con's avatar
con committed
100
using namespace Core;
101
using namespace Debugger;
hjk's avatar
hjk committed
102
103
using namespace Debugger::Constants;
using namespace Debugger::Internal;
con's avatar
con committed
104
using namespace ProjectExplorer;
hjk's avatar
hjk committed
105
using namespace TextEditor;
con's avatar
con committed
106
107
108
109
110


namespace Debugger {
namespace Constants {

111
112
const char * const M_DEBUG_START_DEBUGGING = "QtCreator.Menu.Debug.StartDebugging";

con's avatar
con committed
113
114
const char * const STARTEXTERNAL        = "Debugger.StartExternal";
const char * const ATTACHEXTERNAL       = "Debugger.AttachExternal";
115
const char * const ATTACHCORE           = "Debugger.AttachCore";
116
const char * const ATTACHREMOTE         = "Debugger.AttachRemote";
117
const char * const DETACH               = "Debugger.Detach";
con's avatar
con committed
118

119
120
const char * const RUN_TO_LINE1         = "Debugger.RunToLine1";
const char * const RUN_TO_LINE2         = "Debugger.RunToLine2";
con's avatar
con committed
121
const char * const RUN_TO_FUNCTION      = "Debugger.RunToFunction";
122
123
const char * const JUMP_TO_LINE1        = "Debugger.JumpToLine1";
const char * const JUMP_TO_LINE2        = "Debugger.JumpToLine2";
124
const char * const RETURN_FROM_FUNCTION = "Debugger.ReturnFromFunction";
125
const char * const SNAPSHOT             = "Debugger.Snapshot";
con's avatar
con committed
126
127
128
const char * const TOGGLE_BREAK         = "Debugger.ToggleBreak";
const char * const BREAK_BY_FUNCTION    = "Debugger.BreakByFunction";
const char * const BREAK_AT_MAIN        = "Debugger.BreakAtMain";
129
130
const char * const ADD_TO_WATCH1        = "Debugger.AddToWatch1";
const char * const ADD_TO_WATCH2        = "Debugger.AddToWatch2";
131
const char * const OPERATE_BY_INSTRUCTION  = "Debugger.OperateByInstruction";
con's avatar
con committed
132

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
133
#ifdef Q_WS_MAC
con's avatar
con committed
134
135
136
137
138
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";
139
const char * const REVERSE_KEY              = "";
con's avatar
con committed
140
141
142
143
144
145
146
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";
147
const char * const SNAPSHOT_KEY             = "Alt+D,Alt+S";
con's avatar
con committed
148
149
150
151
152
153
#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";
154
const char * const REVERSE_KEY              = "F12";
con's avatar
con committed
155
156
157
158
159
160
161
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";
162
const char * const SNAPSHOT_KEY             = "Alt+D,Alt+S";
con's avatar
con committed
163
164
165
166
167
168
#endif

} // namespace Constants
} // namespace Debugger


169
170
171
172
173
static ProjectExplorer::SessionManager *sessionManager()
{
    return ProjectExplorer::ProjectExplorerPlugin::instance()->session();
}

174
175
176
177
178
static QSettings *settings()
{
    return ICore::instance()->settings();
}

179
180
181
182
183
184
185
static QToolButton *toolButton(QAction *action)
{
    QToolButton *button = new QToolButton;
    button->setDefaultAction(action);
    return button;
}

186
187
188
189
190
191
///////////////////////////////////////////////////////////////////////
//
// DebugMode
//
///////////////////////////////////////////////////////////////////////

hjk's avatar
hjk committed
192
193
194
195
196
197
198
199
200
201
202
203
204
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
205
  : BaseMode(parent)
hjk's avatar
hjk committed
206
{
207
208
    setDisplayName(tr("Debug"));
    setId(Constants::MODE_DEBUG);
hjk's avatar
hjk committed
209
210
    setIcon(QIcon(":/fancyactionbar/images/mode_Debug.png"));
    setPriority(Constants::P_MODE_DEBUG);
hjk's avatar
hjk committed
211
212
213
214
215
216
217
218
}

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

219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
///////////////////////////////////////////////////////////////////////
//
// DebuggerListener: Close the debugging session if running.
//
///////////////////////////////////////////////////////////////////////

class DebuggerListener : public Core::ICoreListener {
    Q_OBJECT
public:
    explicit DebuggerListener(QObject *parent = 0);
    virtual bool coreAboutToClose();
};

DebuggerListener::DebuggerListener(QObject *parent) :
    Core::ICoreListener(parent)
{
}

bool DebuggerListener::coreAboutToClose()
{
    DebuggerManager *mgr = DebuggerManager::instance();
    if (!mgr)
        return true;
    // Ask to terminate the session.
    const QString title = tr("Close Debugging Session");
    bool cleanTermination = false;
    switch (mgr->state()) {
    case DebuggerNotReady:
        return true;
    case AdapterStarted:     // Most importantly, terminating a running
    case AdapterStartFailed: // debuggee can cause problems.
    case InferiorUnrunnable:
    case InferiorStartFailed:
    case InferiorStopped:
    case InferiorShutDown:
        cleanTermination = true;
        break;
    default:
        break;
    }
    const QString question = cleanTermination ?
260
        tr("A debugging session is still in progress.\nWould you like to terminate it?") :
261
262
        tr("A debugging session is still in progress. Terminating the session in the current"
           " state (%1) can leave the target in an inconsistent state."
263
           " Would you still like to terminate it?")
264
        .arg(QLatin1String(DebuggerManager::stateName(mgr->state())));
265
266
    QMessageBox::StandardButton answer = QMessageBox::question(DebuggerUISwitcher::instance()->mainWindow(),
                                         title, question,
267
268
                                         QMessageBox::Yes|QMessageBox::No, QMessageBox::Yes);
    if (answer != QMessageBox::Yes)
269
270
        return false;
    mgr->exitDebugger();
271
    QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents);
272
273
274
    return true;
}

275
276
277
} // namespace Internal
} // namespace Debugger

hjk's avatar
hjk committed
278

con's avatar
con committed
279
280
281
282
283
284
///////////////////////////////////////////////////////////////////////
//
// LocationMark
//
///////////////////////////////////////////////////////////////////////

285
286
287
namespace Debugger {
namespace Internal {

288
// Used in "real" editors
289
class LocationMark : public TextEditor::BaseTextMark
con's avatar
con committed
290
291
292
293
294
295
{
    Q_OBJECT

public:
    LocationMark(const QString &fileName, int linenumber)
        : BaseTextMark(fileName, linenumber)
296
    {}
con's avatar
con committed
297

298
    QIcon icon() const { return DebuggerManager::instance()->locationMarkIcon(); }
con's avatar
con committed
299
300
    void updateLineNumber(int /*lineNumber*/) {}
    void updateBlock(const QTextBlock & /*block*/) {}
301
    void removedFromEditor() {}
con's avatar
con committed
302
303
};

304
305
306
307
308
309
} // namespace Internal
} // namespace Debugger


///////////////////////////////////////////////////////////////////////
//
310
// CommonOptionsPage
311
312
313
314
315
316
//
///////////////////////////////////////////////////////////////////////

namespace Debugger {
namespace Internal {

317
class CommonOptionsPage : public Core::IOptionsPage
318
319
320
321
{
    Q_OBJECT

public:
322
    CommonOptionsPage() {}
323
324

    // IOptionsPage
325
    QString id() const
326
        { return QLatin1String(Debugger::Constants::DEBUGGER_COMMON_SETTINGS_ID); }
327
    QString displayName() const
328
        { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_COMMON_SETTINGS_NAME); }
329
330
    QString category() const
        { return QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY);  }
331
    QString displayCategory() const
332
        { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_SETTINGS_TR_CATEGORY); }
333
334

    QWidget *createPage(QWidget *parent);
335
    void apply() { m_group.apply(settings()); }
336
    void finish() { m_group.finish(); }
337
    virtual bool matches(const QString &s) const;
338
339

private:
340
    Ui::CommonOptionsPage m_ui;
341
    Utils::SavedActionSet m_group;
342
    QString m_searchKeywords;
343
344
};

345
QWidget *CommonOptionsPage::createPage(QWidget *parent)
346
347
348
{
    QWidget *w = new QWidget(parent);
    m_ui.setupUi(w);
349
    m_group.clear();
350

351
    m_group.insert(theDebuggerAction(ListSourceFiles),
352
        m_ui.checkBoxListSourceFiles);
353
    m_group.insert(theDebuggerAction(UseAlternatingRowColors),
354
        m_ui.checkBoxUseAlternatingRowColors);
355
356
    m_group.insert(theDebuggerAction(UseMessageBoxForSignals),
        m_ui.checkBoxUseMessageBoxForSignals);
357
    m_group.insert(theDebuggerAction(SkipKnownFrames),
358
        m_ui.checkBoxSkipKnownFrames);
359
    m_group.insert(theDebuggerAction(UseToolTipsInMainEditor),
360
        m_ui.checkBoxUseToolTipsInMainEditor);
361
    m_group.insert(theDebuggerAction(AutoDerefPointers), 0);
362
363
364
365
    m_group.insert(theDebuggerAction(UseToolTipsInLocalsView), 0);
    m_group.insert(theDebuggerAction(UseToolTipsInBreakpointsView), 0);
    m_group.insert(theDebuggerAction(UseAddressInBreakpointsView), 0);
    m_group.insert(theDebuggerAction(UseAddressInStackView), 0);
366
    m_group.insert(theDebuggerAction(EnableReverseDebugging),
367
        m_ui.checkBoxEnableReverseDebugging);
368
    m_group.insert(theDebuggerAction(MaximalStackDepth),
369
        m_ui.spinBoxMaximalStackDepth);
370
    m_group.insert(theDebuggerAction(GdbWatchdogTimeout), 0);
371
372
    m_group.insert(theDebuggerAction(ShowStdNamespace), 0);
    m_group.insert(theDebuggerAction(ShowQtNamespace), 0);
373
    m_group.insert(theDebuggerAction(LogTimeStamps), 0);
374
    m_group.insert(theDebuggerAction(VerboseLog), 0);
375
    m_group.insert(theDebuggerAction(UsePreciseBreakpoints), 0);
376
377
    m_group.insert(theDebuggerAction(BreakOnThrow), 0);
    m_group.insert(theDebuggerAction(BreakOnCatch), 0);
378

379
380
381
382
#ifdef USE_REVERSE_DEBUGGING
    m_ui.checkBoxEnableReverseDebugging->hide();
#endif

383
384
385
386
387
388
389
390
391
392
393
    if (m_searchKeywords.isEmpty()) {
        QTextStream(&m_searchKeywords) << ' ' << m_ui.checkBoxListSourceFiles->text()
                << ' ' << m_ui.checkBoxUseMessageBoxForSignals->text()
                << ' ' << m_ui.checkBoxUseAlternatingRowColors->text()
                << ' ' << m_ui.checkBoxUseToolTipsInMainEditor->text()
                << ' ' << m_ui.checkBoxSkipKnownFrames->text()
                << ' ' << m_ui.checkBoxEnableReverseDebugging->text()
                << ' ' << m_ui.labelMaximalStackDepth->text();
        m_searchKeywords.remove(QLatin1Char('&'));
    }

394
395
396
    return w;
}

397
398
399
400
401
bool CommonOptionsPage::matches(const QString &s) const
{
    return m_searchKeywords.contains(s, Qt::CaseInsensitive);
}

402
403
404
} // namespace Internal
} // namespace Debugger

hjk's avatar
hjk committed
405

hjk's avatar
hjk committed
406
407
///////////////////////////////////////////////////////////////////////
//
408
// DebuggingHelperOptionPage
hjk's avatar
hjk committed
409
410
411
//
///////////////////////////////////////////////////////////////////////

412
413
414
415
416
417
418
static inline bool oxygenStyle()
{
    if (const ManhattanStyle *ms = qobject_cast<const ManhattanStyle *>(qApp->style()))
        return !qstrcmp("OxygenStyle", ms->systemStyle()->metaObject()->className());
    return false;
}

hjk's avatar
hjk committed
419
420
421
namespace Debugger {
namespace Internal {

422
class DebuggingHelperOptionPage : public Core::IOptionsPage
hjk's avatar
hjk committed
423
424
425
426
{
    Q_OBJECT

public:
427
    DebuggingHelperOptionPage() {}
hjk's avatar
hjk committed
428
429

    // IOptionsPage
430
    QString id() const { return QLatin1String("B.DebuggingHelper"); }
431
    QString displayName() const { return tr("Debugging Helper"); }
432
    QString category() const { return QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY); }
433
    QString displayCategory() const { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_SETTINGS_TR_CATEGORY); }
hjk's avatar
hjk committed
434
435

    QWidget *createPage(QWidget *parent);
436
    void apply() { m_group.apply(settings()); }
437
    void finish() { m_group.finish(); }
438
    virtual bool matches(const QString &s) const;
hjk's avatar
hjk committed
439
440

private:
441
    Ui::DebuggingHelperOptionPage m_ui;
442
    Utils::SavedActionSet m_group;
443
    QString m_searchKeywords;
hjk's avatar
hjk committed
444
445
};

446
QWidget *DebuggingHelperOptionPage::createPage(QWidget *parent)
hjk's avatar
hjk committed
447
448
449
450
{
    QWidget *w = new QWidget(parent);
    m_ui.setupUi(w);

451
    m_ui.dumperLocationChooser->setExpectedKind(Utils::PathChooser::Command);
452
    m_ui.dumperLocationChooser->setPromptDialogTitle(tr("Choose DebuggingHelper Location"));
hjk's avatar
hjk committed
453
454
    m_ui.dumperLocationChooser->setInitialBrowsePathBackup(
        Core::ICore::instance()->resourcePath() + "../../lib");
hjk's avatar
hjk committed
455

456
    m_group.clear();
457
    m_group.insert(theDebuggerAction(UseDebuggingHelpers),
458
        m_ui.debuggingHelperGroupBox);
459
    m_group.insert(theDebuggerAction(UseCustomDebuggingHelperLocation),
460
461
462
463
464
        m_ui.customLocationGroupBox);
    // Suppress Oxygen style's giving flat group boxes bold titles
    if (oxygenStyle())
        m_ui.customLocationGroupBox->setStyleSheet(QLatin1String("QGroupBox::title { font: ; }"));

465
    m_group.insert(theDebuggerAction(CustomDebuggingHelperLocation),
466
        m_ui.dumperLocationChooser);
hjk's avatar
hjk committed
467

468
469
470
    m_group.insert(theDebuggerAction(UseCodeModel),
        m_ui.checkBoxUseCodeModel);

471
#ifdef QT_DEBUG
472
473
    m_group.insert(theDebuggerAction(DebugDebuggingHelpers),
        m_ui.checkBoxDebugDebuggingHelpers);
474
475
476
#else
    m_ui.checkBoxDebugDebuggingHelpers->hide();
#endif
hjk's avatar
hjk committed
477
478
479
480
481
482
483
484
485
486
487

#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

488
489
490
491
492
493
494
495
496
    if (m_searchKeywords.isEmpty()) {
        QTextStream(&m_searchKeywords)
                << ' ' << m_ui.debuggingHelperGroupBox->title()
                << ' ' << m_ui.customLocationGroupBox->title()
                << ' ' << m_ui.dumperLocationLabel->text()
                << ' ' << m_ui.checkBoxUseCodeModel->text()
                << ' ' << m_ui.checkBoxDebugDebuggingHelpers->text();
        m_searchKeywords.remove(QLatin1Char('&'));
    }
hjk's avatar
hjk committed
497
498
499
    return w;
}

500
bool DebuggingHelperOptionPage::matches(const QString &s) const
hjk's avatar
hjk committed
501
{
502
    return m_searchKeywords.contains(s, Qt::CaseInsensitive);
hjk's avatar
hjk committed
503
504
}

hjk's avatar
hjk committed
505
506
507
} // namespace Internal
} // namespace Debugger

508

con's avatar
con committed
509
510
511
512
513
514
///////////////////////////////////////////////////////////////////////
//
// DebuggerPlugin
//
///////////////////////////////////////////////////////////////////////

515
516
517
518
519
520
521

DebuggerPlugin::AttachRemoteParameters::AttachRemoteParameters() :
    attachPid(0),
    winCrashEvent(0)
{
}

522
523
DebuggerPlugin::DebuggerPlugin()
  : m_manager(0),
524
525
526
    m_debugMode(0),
    m_locationMark(0),
    m_gdbRunningContext(0),
527
    m_cmdLineEnabledEngines(AllEngineTypes)
528
{}
con's avatar
con committed
529
530
531
532
533
534

DebuggerPlugin::~DebuggerPlugin()
{}

void DebuggerPlugin::shutdown()
{
hjk's avatar
hjk committed
535
    QTC_ASSERT(m_manager, /**/);
con's avatar
con committed
536
537
538
    if (m_manager)
        m_manager->shutdown();

hjk's avatar
hjk committed
539
    writeSettings();
540
541
542
543

    if (m_uiSwitcher)
        m_uiSwitcher->shutdown();

dt's avatar
dt committed
544
    delete DebuggerSettings::instance();
hjk's avatar
hjk committed
545

con's avatar
con committed
546
547
548
549
550
551
552
553
554
555
    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;

ck's avatar
ck committed
556
    removeObject(m_manager);
con's avatar
con committed
557
558
    delete m_manager;
    m_manager = 0;
559
560
561
562

    removeObject(m_uiSwitcher);
    delete m_uiSwitcher;
    m_uiSwitcher = 0;
con's avatar
con committed
563
564
}

565
static QString msgParameterMissing(const QString &a)
566
567
568
569
{
    return DebuggerPlugin::tr("Option '%1' is missing the parameter.").arg(a);
}

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

// Parse arguments
576
577
578
579
static bool parseArgument(QStringList::const_iterator &it,
                          const QStringList::const_iterator &cend,
                          DebuggerPlugin::AttachRemoteParameters *attachRemoteParameters,
                          unsigned *enabledEngines, QString *errorMessage)
580
581
582
583
584
585
586
587
588
589
{
    const QString &option = *it;
    // '-debug <pid>'
    if (*it == QLatin1String("-debug")) {
        ++it;
        if (it == cend) {
            *errorMessage = msgParameterMissing(*it);
            return false;
        }
        bool ok;
590
        attachRemoteParameters->attachPid = it->toULongLong(&ok);
591
        if (!ok) {
592
593
            attachRemoteParameters->attachPid = 0;
            attachRemoteParameters->attachCore = *it;
594
595
596
        }
        return true;
    }
597
598
599
    // -wincrashevent <event-handle>. A handle used for
    // a handshake when attaching to a crashed Windows process.
    if (*it == QLatin1String("-wincrashevent")) {
600
601
602
603
604
605
        ++it;
        if (it == cend) {
            *errorMessage = msgParameterMissing(*it);
            return false;
        }
        bool ok;
606
        attachRemoteParameters->winCrashEvent = it->toULongLong(&ok);
607
608
609
610
611
612
613
614
        if (!ok) {
            *errorMessage = msgInvalidNumericParameter(option, *it);
            return false;
        }
        return true;
    }
    // engine disabling
    if (option == QLatin1String("-disable-cdb")) {
615
        *enabledEngines &= ~Debugger::CdbEngineType;
616
617
618
        return true;
    }
    if (option == QLatin1String("-disable-gdb")) {
619
        *enabledEngines &= ~Debugger::GdbEngineType;
620
621
622
        return true;
    }
    if (option == QLatin1String("-disable-sdb")) {
623
        *enabledEngines &= ~Debugger::ScriptEngineType;
624
625
626
        return true;
    }

627
    *errorMessage = DebuggerPlugin::tr("Invalid debugger option: %1").arg(option);
628
629
630
    return false;
}

631
632
633
static bool parseArguments(const QStringList &args,
                           DebuggerPlugin::AttachRemoteParameters *attachRemoteParameters,
                           unsigned *enabledEngines, QString *errorMessage)
634
635
636
{
    const QStringList::const_iterator cend = args.constEnd();
    for (QStringList::const_iterator it = args.constBegin(); it != cend; ++it)
637
        if (!parseArgument(it, cend, attachRemoteParameters, enabledEngines, errorMessage))
638
639
            return false;
    if (Debugger::Constants::Internal::debug)
640
        qDebug().nospace() << args << "engines=0x"
641
642
643
            << QString::number(*enabledEngines, 16)
            << " pid" << attachRemoteParameters->attachPid
            << " core" << attachRemoteParameters->attachCore << '\n';
644
645
646
    return true;
}

647
648
649
650
651
652
653
654
655
656
657
658
659
void DebuggerPlugin::remoteCommand(const QStringList &options, const QStringList &)
{
    QString errorMessage;
    AttachRemoteParameters parameters;
    unsigned dummy = 0;
    // Did we receive a request for debugging (unless it is ourselves)?
    if (parseArguments(options, &parameters, &dummy, &errorMessage)
        && parameters.attachPid != quint64(QCoreApplication::applicationPid())) {
        m_attachRemoteParameters = parameters;
        attachCmdLine();
    }
}

hjk's avatar
hjk committed
660
bool DebuggerPlugin::initialize(const QStringList &arguments, QString *errorMessage)
con's avatar
con committed
661
{
662
    // Do not fail the whole plugin if something goes wrong here
663
    if (!parseArguments(arguments, &m_attachRemoteParameters, &m_cmdLineEnabledEngines, errorMessage)) {
664
665
        *errorMessage = tr("Error evaluating command line arguments: %1")
            .arg(*errorMessage);
666
667
668
        qWarning("%s\n", qPrintable(*errorMessage));
        errorMessage->clear();
    }
con's avatar
con committed
669

670
671
672
673
674
675
676
    // Debug mode setup
    m_debugMode = new DebugMode(this);
    //addAutoReleasedObject(m_debugMode);
    m_uiSwitcher = new DebuggerUISwitcher(m_debugMode, this);
    ExtensionSystem::PluginManager::instance()->addObject(m_uiSwitcher);
    m_uiSwitcher->addLanguage(LANG_CPP);

677
    m_manager = new DebuggerManager;
ck's avatar
ck committed
678
    ExtensionSystem::PluginManager::instance()->addObject(m_manager);
679
680
    const QList<Core::IOptionsPage *> engineOptionPages =
        m_manager->initializeEngines(m_cmdLineEnabledEngines);
con's avatar
con committed
681

682
    ICore *core = ICore::instance();
hjk's avatar
hjk committed
683
    QTC_ASSERT(core, return false);
con's avatar
con committed
684

685
686
    Core::ActionManager *am = core->actionManager();
    QTC_ASSERT(am, return false);
con's avatar
con committed
687
688

    Core::UniqueIDManager *uidm = core->uniqueIDManager();
hjk's avatar
hjk committed
689
    QTC_ASSERT(uidm, return false);
con's avatar
con committed
690
691
692
693
694
695
696
697
698
699
700

    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;
701
    cppeditorcontext << uidm->uniqueIdentifier(CppEditor::Constants::C_CPPEDITOR);
con's avatar
con committed
702
703
704
705
706
707

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

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

708
709
710
711
712
713
714
715
716
717
    // register factory of DebuggerRunControl
    m_debuggerRunControlFactory = new DebuggerRunControlFactory(m_manager);
    addAutoReleasedObject(m_debuggerRunControlFactory);

    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);

718
    //Core::ActionContainer *mcppcontext =
719
    //    am->actionContainer(CppEditor::Constants::M_CONTEXT);
con's avatar
con committed
720

721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
    // 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()));

736
737
738
739
    m_startRemoteAction = new QAction(this);
    m_startRemoteAction->setText(tr("Start and Attach to Remote Application..."));
    connect(m_startRemoteAction, SIGNAL(triggered()),
        this, SLOT(startRemoteApplication()));
740

741
    m_detachAction = new QAction(this);
742
    m_detachAction->setText(tr("Detach Debugger"));
743
744
745
    connect(m_detachAction, SIGNAL(triggered()),
        m_manager, SLOT(detachDebugger()));

746
747
748
749
750
751
   // Core::ActionContainer *mdebug =
    //    am->actionContainer(ProjectExplorer::Constants::M_DEBUG);

    Core::Command *cmd = 0;
    const DebuggerManagerActions actions = m_manager->debuggerManagerActions();

con's avatar
con committed
752

753
754
755
    Core::ActionContainer *mstart =
        am->actionContainer(ProjectExplorer::Constants::M_DEBUG_STARTDEBUGGING);

756
    cmd = am->registerAction(actions.continueAction,
757
        ProjectExplorer::Constants::DEBUG, QList<int>() << m_gdbRunningContext);
758
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
759

760
    cmd = am->registerAction(m_startExternalAction,
con's avatar
con committed
761
        Constants::STARTEXTERNAL, globalcontext);
762
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
763

764
    cmd = am->registerAction(m_attachExternalAction,
con's avatar
con committed
765
        Constants::ATTACHEXTERNAL, globalcontext);
766
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
767

768
769
    cmd = am->registerAction(m_attachCoreAction,
        Constants::ATTACHCORE, globalcontext);
770
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
771

772
    cmd = am->registerAction(m_startRemoteAction,
773
        Constants::ATTACHREMOTE, globalcontext);
774
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
775

776
777
    cmd = am->registerAction(m_detachAction,
        Constants::DETACH, globalcontext);
778
    m_uiSwitcher->addMenuAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
779

780
    cmd = am->registerAction(actions.stopAction,
con's avatar
con committed
781
        Constants::INTERRUPT, globalcontext);
con's avatar
con committed
782
783
    cmd->setAttribute(Core::Command::CA_UpdateText);
    cmd->setAttribute(Core::Command::CA_UpdateIcon);
con's avatar
con committed
784
785
    cmd->setDefaultKeySequence(QKeySequence(Constants::INTERRUPT_KEY));
    cmd->setDefaultText(tr("Stop Debugger/Interrupt Debugger"));
786
    m_uiSwitcher->addMenuAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
787

788
    cmd = am->registerAction(actions.resetAction,
con's avatar
con committed
789
        Constants::RESET, globalcontext);
con's avatar
con committed
790
    cmd->setAttribute(Core::Command::CA_UpdateText);
791
    //cmd->setDefaultKeySequence(QKeySequence(Constants::RESET_KEY));
con's avatar
con committed
792
    cmd->setDefaultText(tr("Reset Debugger"));
793
    m_uiSwitcher->addMenuAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
794
795
796

    QAction *sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
797
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Step"), globalcontext);
798
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
799

800
    cmd = am->registerAction(actions.nextAction,
con's avatar
con committed
801
802
        Constants::NEXT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::NEXT_KEY));
803
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
804

805
    cmd = am->registerAction(actions.stepAction,
con's avatar
con committed
806
807
        Constants::STEP, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEP_KEY));
808
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
809

810
    cmd = am->registerAction(actions.stepOutAction,
con's avatar
con committed
811
812
        Constants::STEPOUT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEPOUT_KEY));
813
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
814

815
816
    cmd = am->registerAction(actions.runToLineAction1,
        Constants::RUN_TO_LINE1, debuggercontext);
con's avatar
con committed
817
    cmd->setDefaultKeySequence(QKeySequence(Constants::RUN_TO_LINE_KEY));
818
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
819

820
    cmd = am->registerAction(actions.runToFunctionAction,
con's avatar
con committed
821
822
        Constants::RUN_TO_FUNCTION, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::RUN_TO_FUNCTION_KEY));
823
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
824

825
826
    cmd = am->registerAction(actions.jumpToLineAction1,
        Constants::JUMP_TO_LINE1, debuggercontext);
827
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
828

829
830
    cmd = am->registerAction(actions.returnFromFunctionAction,
        Constants::RETURN_FROM_FUNCTION, debuggercontext);
831
    m_uiSwitcher->addMenuAction(cmd);
832

833
#ifdef USE_REVERSE_DEBUGGING
834
    cmd = am->registerAction(actions.reverseDirectionAction,
835
836
        Constants::REVERSE, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::REVERSE_KEY));
837
    m_uiSwitcher->addMenuAction(cmd);
838
#endif
839

con's avatar
con committed
840
841
    sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
842
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Break"), globalcontext);
843
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
844

845
846
847
    cmd = am->registerAction(actions.snapshotAction,
        Constants::SNAPSHOT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::SNAPSHOT_KEY));
848
    m_uiSwitcher->addMenuAction(cmd);
849

850
851
    cmd = am->registerAction(theDebuggerAction(OperateByInstruction),
        Constants::OPERATE_BY_INSTRUCTION, debuggercontext);
852
    m_uiSwitcher->addMenuAction(cmd);
853

854
    cmd = am->registerAction(actions.breakAction,
con's avatar
con committed
855
856
        Constants::TOGGLE_BREAK, cppeditorcontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::TOGGLE_BREAK_KEY));
857
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
858
859
860
861
    //mcppcontext->addAction(cmd);

    sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
862
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Watch"), globalcontext);
863
    m_uiSwitcher->addMenuAction(cmd);
con's avatar
con committed
864

865
866
    cmd = am->registerAction(actions.watchAction1,
        Constants::ADD_TO_WATCH1, cppeditorcontext);
867
    cmd->action()->setEnabled(true);
868
    //cmd->setDefaultKeySequence(QKeySequence(tr("ALT+D,ALT+W")));
869
    m_uiSwitcher->addMenuAction(cmd);
870

871
    // Editor context menu
872
873
874
875
876
877
    ActionContainer *editorContextMenu =
        am->actionContainer(CppEditor::Constants::M_CONTEXT);
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Views"),
        debuggercontext);
    editorContextMenu->addAction(cmd);
    cmd->setAttribute(Command::CA_Hide);
878

879
880
881
    cmd = am->registerAction(actions.watchAction2,
        Constants::ADD_TO_WATCH2, debuggercontext);
    cmd->action()->setEnabled(true);
882
883
884
885
886
887
888
889
890
891
892
893
    editorContextMenu->addAction(cmd);
    cmd->setAttribute(Command::CA_Hide);

    cmd = am->registerAction(actions.runToLineAction2,
        Constants::RUN_TO_LINE2, debuggercontext);
    cmd->action()->setEnabled(true);
    editorContextMenu->addAction(cmd);
    cmd->setAttribute(Command::CA_Hide);

    cmd = am->registerAction(actions.jumpToLineAction2,
        Constants::JUMP_TO_LINE2, debuggercontext);
    cmd->action()->setEnabled(true);
894
895
    editorContextMenu->addAction(cmd);
    cmd->setAttribute(Command::CA_Hide);
con's avatar
con committed
896

hjk's avatar
hjk committed
897
    // FIXME:
898
    addAutoReleasedObject(new CommonOptionsPage);
899
    addAutoReleasedObject(new DebuggingHelperOptionPage);
900
901
    foreach (Core::IOptionsPage* op, engineOptionPages)
        addAutoReleasedObject(op);
902
    addAutoReleasedObject(new DebuggerListener);
con's avatar
con committed
903
904
    m_locationMark = 0;

905
    m_manager->setSimpleDockWidgetArrangement(LANG_CPP);
hjk's avatar
hjk committed
906
907
    readSettings();

908
909
910
911
    m_uiSwitcher->setToolbar(LANG_CPP, createToolbar());
    connect(m_uiSwitcher, SIGNAL(dockArranged(QString)), m_manager,
            SLOT(setSimpleDockWidgetArrangement(QString)));

hjk's avatar
hjk committed
912
    connect(ModeManager::instance(), SIGNAL(currentModeChanged(Core::IMode*)),
913
            this, SLOT(onModeChanged(Core::IMode*)));
hjk's avatar
hjk committed
914
915
916
917
918
919
920
    m_debugMode->widget()->setFocusProxy(EditorManager::instance());
    addObject(m_debugMode);

    //
    //  Connections
    //

921
922
923
924
925
    // TextEditor
    connect(TextEditor::TextEditorPlugin::instance(),
        SIGNAL(fontSettingsChanged(TextEditor::FontSettings)),
        m_manager, SLOT(fontSettingsChanged(TextEditor::FontSettings)));

con's avatar
con committed
926
    // ProjectExplorer
927
    connect(sessionManager(), SIGNAL(sessionLoaded()),
con's avatar
con committed
928
       m_manager, SLOT(sessionLoaded()));
929
    connect(sessionManager(), SIGNAL(aboutToSaveSession()),
con's avatar
con committed
930
       m_manager, SLOT(aboutToSaveSession()));
931
932
    connect(sessionManager(), SIGNAL(aboutToUnloadSession()),
       m_manager, SLOT(aboutToUnloadSession()));
con's avatar
con committed
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951

    // 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()));
952
953
    connect(m_manager, SIGNAL(gotoLocationRequested(QString,int,bool)),
        this, SLOT(gotoLocation(QString,int,bool)));
hjk's avatar