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

#include "coregdbadapter.h"

#include "debuggeractions.h"
#include "gdbengine.h"
34
#include "debuggerstringutils.h"
35
36
37

#include <utils/qtcassert.h>

38
#include <QtCore/QDir>
39
#include <QtCore/QFileInfo>
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
40
#include <QtGui/QMessageBox>
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61

namespace Debugger {
namespace Internal {

#define CB(callback) \
    static_cast<GdbEngine::AdapterCallback>(&CoreGdbAdapter::callback), \
    STRINGIFY(callback)

///////////////////////////////////////////////////////////////////////
//
// CoreGdbAdapter
//
///////////////////////////////////////////////////////////////////////

CoreGdbAdapter::CoreGdbAdapter(GdbEngine *engine, QObject *parent)
    : AbstractGdbAdapter(engine, parent)
{
}

void CoreGdbAdapter::startAdapter()
{
hjk's avatar
hjk committed
62
    QTC_ASSERT(state() == EngineStarting, qDebug() << state());
63
    showMessage(_("TRYING TO START ADAPTER"));
64

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
65
66
    if (!m_engine->startGdb())
        return;
67
68
69
70
71
72

    emit adapterStarted();
}

void CoreGdbAdapter::startInferior()
{
hjk's avatar
hjk committed
73
    QTC_ASSERT(state() == InferiorStarting, qDebug() << state());
hjk's avatar
hjk committed
74
75
    m_executable = startParameters().executable;
    if (m_executable.isEmpty()) {
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
76
#ifdef EXE_FROM_CORE
hjk's avatar
hjk committed
77
78
79
        // Extra round trip to get executable name from core file.
        // This is sometimes not the full name, so it can't be used
        // as the generic solution.
80

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
81
82
83
84
85
86
87
        m_round = 1;
        loadCoreFile();
#else
        showMessageBox(QMessageBox::Warning, tr("Error Loading Symbols"),
                       tr("No executable to load symbols from specified."));
#endif
        return;
hjk's avatar
hjk committed
88
    }
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
89
90
91
92
#ifdef EXE_FROM_CORE
    m_round = 2;
#endif
    loadExeAndSyms();
hjk's avatar
hjk committed
93
94
}

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
95
void CoreGdbAdapter::loadExeAndSyms()
hjk's avatar
hjk committed
96
{
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
97
98
    // Do that first, otherwise no symbols are loaded.
    QFileInfo fi(m_executable);
99
100
101
    QByteArray path = fi.absoluteFilePath().toLocal8Bit();
    m_engine->postCommand("-file-exec-and-symbols \"" + path + '"',
         CB(handleFileExecAndSymbols));
102
103
}

104
105
106
107
void CoreGdbAdapter::handleFileExecAndSymbols(const GdbResponse &response)
{
    QTC_ASSERT(state() == InferiorStarting, qDebug() << state());
    if (response.resultClass == GdbResultDone) {
108
        showMessage(tr("Symbols found."), StatusBar);
109
    } else {
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
110
        QString msg = tr("Loading symbols from \"%1\" failed:\n").arg(m_executable)
111
            + QString::fromLocal8Bit(response.data.findChild("msg").data());
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
112
        showMessageBox(QMessageBox::Warning, tr("Error Loading Symbols"), msg);
113
    }
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
114
    loadCoreFile();
115
116
}

Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
117
118
119
120
void CoreGdbAdapter::loadCoreFile()
{
    // Quoting core name below fails in gdb 6.8-debian.
    QFileInfo fi(startParameters().coreFile);
121
122
    QByteArray coreName = fi.absoluteFilePath().toLocal8Bit();
    m_engine->postCommand("target core " + coreName, CB(handleTargetCore));
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
123
124
125
}

void CoreGdbAdapter::handleTargetCore(const GdbResponse &response)
hjk's avatar
hjk committed
126
127
128
{
    QTC_ASSERT(state() == InferiorStarting, qDebug() << state());
    if (response.resultClass == GdbResultDone) {
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
129
130
131
132
133
134
135
136
137
138
139
140
141
#ifdef EXE_FROM_CORE
        if (m_round == 1) {
            m_round = 2;
            GdbMi console = response.data.findChild("consolestreamoutput");
            int pos1 = console.data().indexOf('`');
            int pos2 = console.data().indexOf('\'');
            if (pos1 != -1 && pos2 != -1) {
                m_executable = console.data().mid(pos1 + 1, pos2 - pos1 - 1);
                // Strip off command line arguments. FIXME: make robust.
                int idx = m_executable.indexOf(_c(' '));
                if (idx >= 0)
                    m_executable.truncate(idx);
                if (!m_executable.isEmpty()) {
142
143
144
145
                    m_executable = QFileInfo(startParameters().coreFile).absoluteDir()
                                   .absoluteFilePath(m_executable);
                    if (QFile::exists(m_executable)) {
                        // Finish extra round ...
146
                        showMessage(tr("Attached to core temporarily."), StatusBar);
147
                        m_engine->postCommand("detach");
148
149
150
151
                        // ... and retry.
                        loadExeAndSyms();
                        return;
                    }
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
152
153
154
155
156
157
                }
            }
            showMessageBox(QMessageBox::Warning, tr("Error Loading Symbols"),
                           tr("Unable to determine executable from core file."));
        }
#endif
158
        showMessage(tr("Attached to core."), StatusBar);
hjk's avatar
hjk committed
159
160
        setState(InferiorUnrunnable);
        m_engine->updateAll();
161
    } else {
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
162
        QString msg = tr("Attach to core \"%1\" failed:\n").arg(startParameters().coreFile)
163
            + QString::fromLocal8Bit(response.data.findChild("msg").data());
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
164
        emit inferiorStartFailed(msg);
hjk's avatar
hjk committed
165
166
    }
}
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
167

168
169
170
171
172
173
174
175
void CoreGdbAdapter::interruptInferior()
{
    // A core should never 'run'
    QTC_ASSERT(false, /**/);
}

} // namespace Internal
} // namespace Debugger