debuggerplugin.cpp 50.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"
con's avatar
con committed
39

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

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

con's avatar
con committed
59
#include <cplusplus/ExpressionUnderCursor.h>
hjk's avatar
hjk committed
60

con's avatar
con committed
61
#include <cppeditor/cppeditorconstants.h>
hjk's avatar
hjk committed
62

63
64
#include <extensionsystem/pluginmanager.h>

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

#include <texteditor/basetexteditor.h>
con's avatar
con committed
72
73
74
#include <texteditor/basetextmark.h>
#include <texteditor/itexteditor.h>
#include <texteditor/texteditorconstants.h>
hjk's avatar
hjk committed
75
76

#include <utils/qtcassert.h>
77
#include <utils/styledbar.h>
78
#include <utils/fancymainwindow.h>
con's avatar
con committed
79
80
81
82
83

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

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

96
97
#include <climits>

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


namespace Debugger {
namespace Constants {

108
109
const char * const M_DEBUG_START_DEBUGGING = "QtCreator.Menu.Debug.StartDebugging";

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

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";
122
123
const char * const ADD_TO_WATCH1        = "Debugger.AddToWatch1";
const char * const ADD_TO_WATCH2        = "Debugger.AddToWatch2";
124
const char * const OPERATE_BY_INSTRUCTION  = "Debugger.OperateByInstruction";
con's avatar
con committed
125

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
126
#ifdef Q_WS_MAC
con's avatar
con committed
127
128
129
130
131
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";
132
const char * const REVERSE_KEY              = "";
con's avatar
con committed
133
134
135
136
137
138
139
140
141
142
143
144
145
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";
146
const char * const REVERSE_KEY              = "F12";
con's avatar
con committed
147
148
149
150
151
152
153
154
155
156
157
158
159
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


160
161
162
163
164
static ProjectExplorer::SessionManager *sessionManager()
{
    return ProjectExplorer::ProjectExplorerPlugin::instance()->session();
}

165
166
167
168
169
static QSettings *settings()
{
    return ICore::instance()->settings();
}

170
171
172
173
174
175
176
static QToolButton *toolButton(QAction *action)
{
    QToolButton *button = new QToolButton;
    button->setDefaultAction(action);
    return button;
}

177
178
179
180
181
182
///////////////////////////////////////////////////////////////////////
//
// DebugMode
//
///////////////////////////////////////////////////////////////////////

hjk's avatar
hjk committed
183
184
185
186
187
188
189
190
191
192
193
194
195
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
196
  : BaseMode(parent)
hjk's avatar
hjk committed
197
{
198
199
    setDisplayName(tr("Debug"));
    setId(Constants::MODE_DEBUG);
hjk's avatar
hjk committed
200
201
    setIcon(QIcon(":/fancyactionbar/images/mode_Debug.png"));
    setPriority(Constants::P_MODE_DEBUG);
hjk's avatar
hjk committed
202
203
204
205
206
207
208
209
}

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

210
211
212
213
214
215
216
217
218
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
///////////////////////////////////////////////////////////////////////
//
// 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 ?
251
        tr("A debugging session is still in progress.\nWould you like to terminate it?") :
252
253
        tr("A debugging session is still in progress. Terminating the session in the current"
           " state (%1) can leave the target in an inconsistent state."
254
           " Would you still like to terminate it?")
255
        .arg(QLatin1String(DebuggerManager::stateName(mgr->state())));
256
257
258
    QMessageBox::StandardButton answer = QMessageBox::question(mgr->mainWindow(), title, question,
                                         QMessageBox::Yes|QMessageBox::No, QMessageBox::Yes);
    if (answer != QMessageBox::Yes)
259
260
        return false;
    mgr->exitDebugger();
261
    QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents);
262
263
264
    return true;
}

265
266
267
} // namespace Internal
} // namespace Debugger

hjk's avatar
hjk committed
268

con's avatar
con committed
269
270
271
272
273
274
///////////////////////////////////////////////////////////////////////
//
// LocationMark
//
///////////////////////////////////////////////////////////////////////

275
276
277
namespace Debugger {
namespace Internal {

278
279
280
281
282
283
284
static QIcon locationMarkIcon()
{
    static const QIcon icon(":/debugger/images/location.svg");
    return icon;
}

// Used in "real" editors
285
class LocationMark : public TextEditor::BaseTextMark
con's avatar
con committed
286
287
288
289
290
291
{
    Q_OBJECT

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

294
    QIcon icon() const { return locationMarkIcon(); }
con's avatar
con committed
295
296
    void updateLineNumber(int /*lineNumber*/) {}
    void updateBlock(const QTextBlock & /*block*/) {}
297
    void removedFromEditor() {}
con's avatar
con committed
298
299
};

300
301
302
303
304
305
} // namespace Internal
} // namespace Debugger


///////////////////////////////////////////////////////////////////////
//
306
// CommonOptionsPage
307
308
309
310
311
312
//
///////////////////////////////////////////////////////////////////////

namespace Debugger {
namespace Internal {

313
class CommonOptionsPage : public Core::IOptionsPage
314
315
316
317
{
    Q_OBJECT

public:
318
    CommonOptionsPage() {}
319
320

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

    QWidget *createPage(QWidget *parent);
331
    void apply() { m_group.apply(settings()); }
332
    void finish() { m_group.finish(); }
333
    virtual bool matches(const QString &s) const;
334
335

private:
336
    Ui::CommonOptionsPage m_ui;
337
    Utils::SavedActionSet m_group;
338
    QString m_searchKeywords;
339
340
};

341
QWidget *CommonOptionsPage::createPage(QWidget *parent)
342
343
344
{
    QWidget *w = new QWidget(parent);
    m_ui.setupUi(w);
345
    m_group.clear();
346

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

373
374
375
376
#ifdef USE_REVERSE_DEBUGGING
    m_ui.checkBoxEnableReverseDebugging->hide();
#endif

377
378
379
380
381
382
383
384
385
386
387
    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('&'));
    }

388
389
390
    return w;
}

391
392
393
394
395
bool CommonOptionsPage::matches(const QString &s) const
{
    return m_searchKeywords.contains(s, Qt::CaseInsensitive);
}

396
397
398
} // namespace Internal
} // namespace Debugger

hjk's avatar
hjk committed
399

hjk's avatar
hjk committed
400
401
///////////////////////////////////////////////////////////////////////
//
402
// DebuggingHelperOptionPage
hjk's avatar
hjk committed
403
404
405
//
///////////////////////////////////////////////////////////////////////

406
407
408
409
410
411
412
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
413
414
415
namespace Debugger {
namespace Internal {

416
class DebuggingHelperOptionPage : public Core::IOptionsPage
hjk's avatar
hjk committed
417
418
419
420
{
    Q_OBJECT

public:
421
    DebuggingHelperOptionPage() {}
hjk's avatar
hjk committed
422
423

    // IOptionsPage
424
    QString id() const { return QLatin1String("B.DebuggingHelper"); }
425
    QString displayName() const { return tr("Debugging Helper"); }
426
    QString category() const { return QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY); }
427
    QString displayCategory() const { return QCoreApplication::translate("Debugger", Debugger::Constants::DEBUGGER_SETTINGS_TR_CATEGORY); }
hjk's avatar
hjk committed
428
429

    QWidget *createPage(QWidget *parent);
430
    void apply() { m_group.apply(settings()); }
431
    void finish() { m_group.finish(); }
432
    virtual bool matches(const QString &s) const;
hjk's avatar
hjk committed
433
434

private:
435
    Ui::DebuggingHelperOptionPage m_ui;
436
    Utils::SavedActionSet m_group;
437
    QString m_searchKeywords;
hjk's avatar
hjk committed
438
439
};

440
QWidget *DebuggingHelperOptionPage::createPage(QWidget *parent)
hjk's avatar
hjk committed
441
442
443
444
{
    QWidget *w = new QWidget(parent);
    m_ui.setupUi(w);

445
    m_ui.dumperLocationChooser->setExpectedKind(Utils::PathChooser::Command);
446
    m_ui.dumperLocationChooser->setPromptDialogTitle(tr("Choose DebuggingHelper Location"));
hjk's avatar
hjk committed
447
448
    m_ui.dumperLocationChooser->setInitialBrowsePathBackup(
        Core::ICore::instance()->resourcePath() + "../../lib");
hjk's avatar
hjk committed
449

450
    m_group.clear();
451
    m_group.insert(theDebuggerAction(UseDebuggingHelpers),
452
        m_ui.debuggingHelperGroupBox);
453
    m_group.insert(theDebuggerAction(UseCustomDebuggingHelperLocation),
454
455
456
457
458
        m_ui.customLocationGroupBox);
    // Suppress Oxygen style's giving flat group boxes bold titles
    if (oxygenStyle())
        m_ui.customLocationGroupBox->setStyleSheet(QLatin1String("QGroupBox::title { font: ; }"));

459
    m_group.insert(theDebuggerAction(CustomDebuggingHelperLocation),
460
        m_ui.dumperLocationChooser);
hjk's avatar
hjk committed
461

462
463
464
    m_group.insert(theDebuggerAction(UseCodeModel),
        m_ui.checkBoxUseCodeModel);

465
#ifdef QT_DEBUG
466
467
    m_group.insert(theDebuggerAction(DebugDebuggingHelpers),
        m_ui.checkBoxDebugDebuggingHelpers);
468
469
470
#else
    m_ui.checkBoxDebugDebuggingHelpers->hide();
#endif
hjk's avatar
hjk committed
471
472
473
474
475
476
477
478
479
480
481

#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

482
483
484
485
486
487
488
489
490
    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
491
492
493
    return w;
}

494
bool DebuggingHelperOptionPage::matches(const QString &s) const
hjk's avatar
hjk committed
495
{
496
    return m_searchKeywords.contains(s, Qt::CaseInsensitive);
hjk's avatar
hjk committed
497
498
}

hjk's avatar
hjk committed
499
500
501
} // namespace Internal
} // namespace Debugger

502

con's avatar
con committed
503
504
505
506
507
508
///////////////////////////////////////////////////////////////////////
//
// DebuggerPlugin
//
///////////////////////////////////////////////////////////////////////

509
510
511
512
513
514
515

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

516
517
DebuggerPlugin::DebuggerPlugin()
  : m_manager(0),
518
519
520
    m_debugMode(0),
    m_locationMark(0),
    m_gdbRunningContext(0),
521
    m_cmdLineEnabledEngines(AllEngineTypes),
522
523
    m_toggleLockedAction(0)
{}
con's avatar
con committed
524
525
526
527
528
529

DebuggerPlugin::~DebuggerPlugin()
{}

void DebuggerPlugin::shutdown()
{
hjk's avatar
hjk committed
530
    QTC_ASSERT(m_manager, /**/);
con's avatar
con committed
531
532
533
    if (m_manager)
        m_manager->shutdown();

hjk's avatar
hjk committed
534
    writeSettings();
dt's avatar
dt committed
535
    delete DebuggerSettings::instance();
hjk's avatar
hjk committed
536

con's avatar
con committed
537
538
539
540
541
542
543
544
545
546
547
    //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;

ck's avatar
ck committed
548
    removeObject(m_manager);
con's avatar
con committed
549
550
551
552
    delete m_manager;
    m_manager = 0;
}

553
static QString msgParameterMissing(const QString &a)
554
555
556
557
{
    return DebuggerPlugin::tr("Option '%1' is missing the parameter.").arg(a);
}

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

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

615
    *errorMessage = DebuggerPlugin::tr("Invalid debugger option: %1").arg(option);
616
617
618
    return false;
}

619
620
621
static bool parseArguments(const QStringList &args,
                           DebuggerPlugin::AttachRemoteParameters *attachRemoteParameters,
                           unsigned *enabledEngines, QString *errorMessage)
622
623
624
{
    const QStringList::const_iterator cend = args.constEnd();
    for (QStringList::const_iterator it = args.constBegin(); it != cend; ++it)
625
        if (!parseArgument(it, cend, attachRemoteParameters, enabledEngines, errorMessage))
626
627
            return false;
    if (Debugger::Constants::Internal::debug)
628
        qDebug().nospace() << args << "engines=0x"
629
630
631
            << QString::number(*enabledEngines, 16)
            << " pid" << attachRemoteParameters->attachPid
            << " core" << attachRemoteParameters->attachCore << '\n';
632
633
634
    return true;
}

635
636
637
638
639
640
641
642
643
644
645
646
647
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
648
bool DebuggerPlugin::initialize(const QStringList &arguments, QString *errorMessage)
con's avatar
con committed
649
{
650
    // Do not fail the whole plugin if something goes wrong here
651
    if (!parseArguments(arguments, &m_attachRemoteParameters, &m_cmdLineEnabledEngines, errorMessage)) {
652
653
        *errorMessage = tr("Error evaluating command line arguments: %1")
            .arg(*errorMessage);
654
655
656
        qWarning("%s\n", qPrintable(*errorMessage));
        errorMessage->clear();
    }
con's avatar
con committed
657

658
    m_manager = new DebuggerManager;
ck's avatar
ck committed
659
    ExtensionSystem::PluginManager::instance()->addObject(m_manager);
660
661
    const QList<Core::IOptionsPage *> engineOptionPages =
        m_manager->initializeEngines(m_cmdLineEnabledEngines);
con's avatar
con committed
662

663
    ICore *core = ICore::instance();
hjk's avatar
hjk committed
664
    QTC_ASSERT(core, return false);
con's avatar
con committed
665

666
667
    Core::ActionManager *am = core->actionManager();
    QTC_ASSERT(am, return false);
con's avatar
con committed
668
669

    Core::UniqueIDManager *uidm = core->uniqueIDManager();
hjk's avatar
hjk committed
670
    QTC_ASSERT(uidm, return false);
con's avatar
con committed
671
672
673
674
675
676
677
678
679
680
681

    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;
682
    cppeditorcontext << uidm->uniqueIdentifier(CppEditor::Constants::C_CPPEDITOR);
con's avatar
con committed
683
684
685
686
687
688

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

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

689
    //Core::ActionContainer *mcppcontext =
690
    //    am->actionContainer(CppEditor::Constants::M_CONTEXT);
con's avatar
con committed
691

692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
    // 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()));

707

708
709
710
711
    m_startRemoteAction = new QAction(this);
    m_startRemoteAction->setText(tr("Start and Attach to Remote Application..."));
    connect(m_startRemoteAction, SIGNAL(triggered()),
        this, SLOT(startRemoteApplication()));
712
713


714
    m_detachAction = new QAction(this);
715
    m_detachAction->setText(tr("Detach Debugger"));
716
717
718
    connect(m_detachAction, SIGNAL(triggered()),
        m_manager, SLOT(detachDebugger()));

719
    Core::ActionContainer *mdebug =
720
        am->actionContainer(ProjectExplorer::Constants::M_DEBUG);
con's avatar
con committed
721

722
723
724
    Core::ActionContainer *mstart =
        am->actionContainer(ProjectExplorer::Constants::M_DEBUG_STARTDEBUGGING);

con's avatar
con committed
725
    Core::Command *cmd = 0;
726
727
    const DebuggerManagerActions actions = m_manager->debuggerManagerActions();
    cmd = am->registerAction(actions.continueAction,
728
        ProjectExplorer::Constants::DEBUG, QList<int>() << m_gdbRunningContext);
729
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
730

731
    cmd = am->registerAction(m_startExternalAction,
con's avatar
con committed
732
        Constants::STARTEXTERNAL, globalcontext);
733
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
734

735
    cmd = am->registerAction(m_attachExternalAction,
con's avatar
con committed
736
        Constants::ATTACHEXTERNAL, globalcontext);
737
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
738

739
740
    cmd = am->registerAction(m_attachCoreAction,
        Constants::ATTACHCORE, globalcontext);
741
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
742

743
    cmd = am->registerAction(m_startRemoteAction,
744
        Constants::ATTACHREMOTE, globalcontext);
745
    mstart->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
746

747
748
749
    cmd = am->registerAction(m_detachAction,
        Constants::DETACH, globalcontext);
    mdebug->addAction(cmd, Core::Constants::G_DEFAULT_ONE);
con's avatar
con committed
750

751
    cmd = am->registerAction(actions.stopAction,
con's avatar
con committed
752
        Constants::INTERRUPT, globalcontext);
con's avatar
con committed
753
754
    cmd->setAttribute(Core::Command::CA_UpdateText);
    cmd->setAttribute(Core::Command::CA_UpdateIcon);
con's avatar
con committed
755
756
757
758
    cmd->setDefaultKeySequence(QKeySequence(Constants::INTERRUPT_KEY));
    cmd->setDefaultText(tr("Stop Debugger/Interrupt Debugger"));
    mdebug->addAction(cmd, Core::Constants::G_DEFAULT_ONE);

759
    cmd = am->registerAction(actions.resetAction,
con's avatar
con committed
760
        Constants::RESET, globalcontext);
con's avatar
con committed
761
    cmd->setAttribute(Core::Command::CA_UpdateText);
con's avatar
con committed
762
763
764
765
766
767
    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
768
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Step"), globalcontext);
con's avatar
con committed
769
770
    mdebug->addAction(cmd);

771
    cmd = am->registerAction(actions.nextAction,
con's avatar
con committed
772
773
774
775
        Constants::NEXT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::NEXT_KEY));
    mdebug->addAction(cmd);

776
    cmd = am->registerAction(actions.stepAction,
con's avatar
con committed
777
778
779
780
        Constants::STEP, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEP_KEY));
    mdebug->addAction(cmd);

781
    cmd = am->registerAction(actions.stepOutAction,
con's avatar
con committed
782
783
784
785
        Constants::STEPOUT, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::STEPOUT_KEY));
    mdebug->addAction(cmd);

786
    cmd = am->registerAction(actions.runToLineAction,
con's avatar
con committed
787
788
789
790
        Constants::RUN_TO_LINE, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::RUN_TO_LINE_KEY));
    mdebug->addAction(cmd);

791
    cmd = am->registerAction(actions.runToFunctionAction,
con's avatar
con committed
792
793
794
795
        Constants::RUN_TO_FUNCTION, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::RUN_TO_FUNCTION_KEY));
    mdebug->addAction(cmd);

796
    cmd = am->registerAction(actions.jumpToLineAction,
con's avatar
con committed
797
798
799
        Constants::JUMP_TO_LINE, debuggercontext);
    mdebug->addAction(cmd);

800
#ifdef USE_REVERSE_DEBUGGING
801
    cmd = am->registerAction(actions.reverseDirectionAction,
802
803
804
        Constants::REVERSE, debuggercontext);
    cmd->setDefaultKeySequence(QKeySequence(Constants::REVERSE_KEY));
    mdebug->addAction(cmd);
805
#endif
806

con's avatar
con committed
807
808
    sep = new QAction(this);
    sep->setSeparator(true);
con's avatar
con committed
809
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Break"), globalcontext);
con's avatar
con committed
810
811
    mdebug->addAction(cmd);

812
813
814
815
    cmd = am->registerAction(theDebuggerAction(OperateByInstruction),
        Constants::OPERATE_BY_INSTRUCTION, debuggercontext);
    mdebug->addAction(cmd);

816
    cmd = am->registerAction(actions.breakAction,
con's avatar
con committed
817
818
819
820
821
822
823
        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
824
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Watch"), globalcontext);
con's avatar
con committed
825
826
    mdebug->addAction(cmd);

827
828
    cmd = am->registerAction(actions.watchAction1,
        Constants::ADD_TO_WATCH1, cppeditorcontext);
829
    cmd->action()->setEnabled(true);
830
    //cmd->setDefaultKeySequence(QKeySequence(tr("ALT+D,ALT+W")));
con's avatar
con committed
831
    mdebug->addAction(cmd);
832
833
834
835
836
837
838
839
840
841
842
843
844

    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);
    cmd = am->registerAction(actions.watchAction2,
        Constants::ADD_TO_WATCH2, debuggercontext);
    cmd->action()->setEnabled(true);
    //cmd->setDefaultKeySequence(QKeySequence(tr("ALT+D,ALT+W")));
    editorContextMenu->addAction(cmd);
    cmd->setAttribute(Command::CA_Hide);
con's avatar
con committed
845

846
    // Views menu
con's avatar
con committed
847
    cmd = am->registerAction(sep, QLatin1String("Debugger.Sep.Views"), globalcontext);
848
    mdebug->addAction(cmd);
849
    ActionContainer *viewsMenu = am->createMenu(Constants::M_DEBUG_VIEWS);
850
851
852
853
854
855
856
857
858
    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)),
859
860
        m_manager->mainWindow(), SLOT(setLocked(bool)));
    foreach (QDockWidget *dockWidget, m_manager->mainWindow()->dockWidgets()) {
861
862
863
864
865
866
867
868
869
870
871
872
873
        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()));

hjk's avatar
hjk committed
874
    // FIXME:
875
    addAutoReleasedObject(new CommonOptionsPage);
876
    addAutoReleasedObject(new DebuggingHelperOptionPage);
877
878
    foreach (Core::IOptionsPage* op, engineOptionPages)
        addAutoReleasedObject(op);
879
    addAutoReleasedObject(new DebuggerListener);
con's avatar
con committed
880
881
    m_locationMark = 0;

hjk's avatar
hjk committed
882
883
884
885
886

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

889
    // register factory of DebuggerRunControl
890
891
    m_debuggerRunControlFactory = new DebuggerRunControlFactory(m_manager);
    addAutoReleasedObject(m_debuggerRunControlFactory);
con's avatar
con committed
892

hjk's avatar
hjk committed
893
894
895
896
897
898
899
900
901
902
903
904
    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);

    QWidget *editorAndFindWidget = new QWidget;
    editorAndFindWidget->setLayout(editorHolderLayout);
905
906
    editorHolderLayout->addWidget(new EditorManagerPlaceHolder(m_debugMode));
    editorHolderLayout->addWidget(new FindToolBarPlaceHolder(editorAndFindWidget));
hjk's avatar
hjk committed
907
908
909
910
911
912
913
914
915
916
917
918
919

    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());
920
    splitter->addWidget(new OutputPanePlaceHolder(m_debugMode, splitter));
hjk's avatar
hjk committed
921
922
923
924
925
926
927
928
929
930
931
932
    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);

933
    Utils::StyledBar *debugToolBar = new Utils::StyledBar;
934
    debugToolBar->setProperty("topBorder", true);
935
936
937
938
939
940
941
942
    QHBoxLayout *debugToolBarLayout = new QHBoxLayout(debugToolBar);
    debugToolBarLayout->setMargin(0);
    debugToolBarLayout->setSpacing(0);
    debugToolBarLayout->addWidget(toolButton(am->command(ProjectExplorer::Constants::DEBUG)->action()));
    debugToolBarLayout->addWidget(toolButton(am->command(Constants::INTERRUPT)->action()));
    debugToolBarLayout->addWidget(toolButton(am->command(Constants::NEXT)->action()));
    debugToolBarLayout->addWidget(toolButton(am->command(Constants::STEP)->action()));
    debugToolBarLayout->addWidget(toolButton(am->command(Constants::STEPOUT)->action()));
943
    debugToolBarLayout->addWidget(toolButton(am->command(Constants::OPERATE_BY_INSTRUCTION)->action()));
944
#ifdef USE_REVERSE_DEBUGGING
945
    debugToolBarLayout->addWidget(new Utils::StyledSeparator);
946
    debugToolBarLayout->addWidget(toolButton(am->command(Constants::REVERSE)->action()));
947
#endif
948
    debugToolBarLayout->addWidget(new Utils::StyledSeparator);
949
    debugToolBarLayout->addWidget(new QLabel(tr("Threads:")));
hjk's avatar
hjk committed
950
951
952
953
954

    QComboBox *threadBox = new QComboBox;
    threadBox->setModel(m_manager->threadsModel());
    connect(threadBox, SIGNAL(activated(int)),
        m_manager->threadsWindow(), SIGNAL(threadSelected(int)));
955
956
    debugToolBarLayout->addWidget(threadBox);
    debugToolBarLayout->addWidget(m_manager->statusLabel(), 10);
hjk's avatar
hjk committed
957
958
959
960
961
962
963
964
965
966
967

    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*)),
968
            this, SLOT(onModeChanged(Core::IMode*)));
hjk's avatar
hjk committed
969
970
971
972
973
974
975
    m_debugMode->widget()->setFocusProxy(EditorManager::instance());
    addObject(m_debugMode);

    //
    //  Connections
    //

con's avatar
con committed
976
    // ProjectExplorer
977
    connect(sessionManager(), SIGNAL(sessionLoaded()),
con's avatar
con committed
978
       m_manager, SLOT(sessionLoaded()));
979
    connect(sessionManager(), SIGNAL(aboutToSaveSession()),
con's avatar
con committed
980
       m_manager, SLOT(aboutToSaveSession()));
981
982
    connect(sessionManager(), SIGNAL(aboutToUnloadSession()),
       m_manager, SLOT(aboutToUnloadSession()));
con's avatar
con committed
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001

    // 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()));
1002
1003
    connect(m_manager, SIGNAL(gotoLocationRequested(QString,int,bool)),
        this, SLOT(gotoLocation(QString,int,bool)));
hjk's avatar
hjk committed
1004
1005
    connect(m_manager, SIGNAL(stateChanged(int)),
        this, SLOT(handleStateChanged(int)));
con's avatar
con committed
1006
1007
1008
1009
1010
    connect(m_manager, SIGNAL(previousModeRequested()),
        this, SLOT(activatePreviousMode()));
    connect(m_manager, SIGNAL(debugModeRequested()),
        this, SLOT(activateDebugMode()));

1011
    connect(theDebuggerAction(SettingsDialog), SIGNAL(triggered()),
1012
1013
        this, SLOT(showSettingsDialog()));

hjk's avatar
hjk committed
1014
    handleStateChanged(DebuggerNotReady);
con's avatar
con committed
1015
1016
1017
1018
1019
    return true;
}

void DebuggerPlugin::extensionsInitialized()
{
hjk's avatar
hjk committed
1020
    // time gdb -i mi -ex 'debuggerplugin.cpp:800' -ex r -ex q bin/qtcreator.bin
1021
    const QByteArray env = qgetenv("QTC_DEBUGGER_TEST");
hjk's avatar
hjk committed
1022
    //qDebug() << "EXTENSIONS INITIALIZED:" << env;