coregdbadapter.cpp 6.23 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() == EngineSetupRequested, 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
    m_engine->handleAdapterStarted();
69 70
}

hjk's avatar
hjk committed
71
void CoreGdbAdapter::setupInferior()
72
{
hjk's avatar
hjk committed
73
    QTC_ASSERT(state() == InferiorSetupRequested, 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
void CoreGdbAdapter::handleFileExecAndSymbols(const GdbResponse &response)
{
hjk's avatar
hjk committed
106
    QTC_ASSERT(state() == InferiorSetupRequested, qDebug() << state());
107
    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
{
hjk's avatar
hjk committed
127
    QTC_ASSERT(state() == InferiorSetupRequested, qDebug() << state());
hjk's avatar
hjk committed
128
    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
        m_engine->handleInferiorPrepared();
160
    } else {
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
161
        QString msg = tr("Attach to core \"%1\" failed:\n").arg(startParameters().coreFile)
162
            + QString::fromLocal8Bit(response.data.findChild("msg").data());
hjk's avatar
hjk committed
163
        m_engine->notifyInferiorSetupFailed(msg);
hjk's avatar
hjk committed
164 165
    }
}
Oswald Buddenhagen's avatar
Oswald Buddenhagen committed
166

hjk's avatar
hjk committed
167
void CoreGdbAdapter::runEngine()
hjk's avatar
hjk committed
168
{
hjk's avatar
hjk committed
169 170
    QTC_ASSERT(state() == EngineRunRequested, qDebug() << state());
    m_engine->notifyInferiorUnrunnable();
hjk's avatar
hjk committed
171 172 173
    m_engine->updateAll();
}

174 175
void CoreGdbAdapter::interruptInferior()
{
hjk's avatar
hjk committed
176
    // A core never runs, so this cannot be called.
177 178 179
    QTC_ASSERT(false, /**/);
}

hjk's avatar
hjk committed
180 181 182 183 184 185 186 187 188 189
void CoreGdbAdapter::shutdownInferior()
{
    m_engine->notifyInferiorShutdownOk();
}

void CoreGdbAdapter::shutdownAdapter()
{
    m_engine->notifyAdapterShutdownOk();
}

190 191
} // namespace Internal
} // namespace Debugger