2009-04-09 16:51:13 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
|
|
|
|
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
|
|
|
|
|
**
|
|
|
|
|
** Contact: Qt Software Information (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 qt-sales@nokia.com.
|
|
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "cdbmodules.h"
|
|
|
|
|
#include "moduleshandler.h"
|
|
|
|
|
#include "cdbdebugengine_p.h"
|
|
|
|
|
|
2009-04-15 12:01:58 +02:00
|
|
|
#include <QtCore/QFileInfo>
|
|
|
|
|
|
2009-04-09 16:51:13 +02:00
|
|
|
namespace Debugger {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
bool getModuleList(IDebugSymbols3 *syms, QList<Module> *modules, QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
modules->clear();
|
|
|
|
|
ULONG loadedCount, unloadedCount;
|
|
|
|
|
HRESULT hr = syms->GetNumberModules(&loadedCount, &unloadedCount);
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
|
*errorMessage= msgComFailed("GetNumberModules", hr);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// retrieve array of parameters
|
|
|
|
|
const ULONG count = loadedCount + unloadedCount;
|
|
|
|
|
QVector<DEBUG_MODULE_PARAMETERS> parameters(count);
|
|
|
|
|
DEBUG_MODULE_PARAMETERS *parmPtr = &(*parameters.begin());
|
|
|
|
|
memset(parmPtr, 0, sizeof(DEBUG_MODULE_PARAMETERS) * count);
|
|
|
|
|
hr = syms->GetModuleParameters(count, 0, 0u, parmPtr);
|
|
|
|
|
// E_INVALIDARG indicates 'Partial results' according to docu
|
|
|
|
|
if (FAILED(hr) && hr != E_INVALIDARG) {
|
|
|
|
|
*errorMessage= msgComFailed("GetModuleParameters", hr);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// fill array
|
|
|
|
|
const QString hexPrefix = QLatin1String("0x");
|
|
|
|
|
WCHAR wszBuf[MAX_PATH];
|
|
|
|
|
for (ULONG m = 0; m < count; m++) {
|
|
|
|
|
const DEBUG_MODULE_PARAMETERS &p = parameters.at(m);
|
|
|
|
|
if (p.Base != DEBUG_INVALID_OFFSET) { // Partial results?
|
|
|
|
|
Module module;
|
|
|
|
|
module.symbolsRead = (p.Flags & DEBUG_MODULE_USER_MODE)
|
|
|
|
|
&& (p.SymbolType != DEBUG_SYMTYPE_NONE);
|
|
|
|
|
module.startAddress = hexPrefix + QString::number(p.Base, 16);
|
|
|
|
|
module.endAddress = hexPrefix + QString::number((p.Base + p.Size), 16);
|
|
|
|
|
hr = syms ->GetModuleNameStringWide(DEBUG_MODNAME_IMAGE, m, 0, wszBuf, MAX_PATH - 1, 0);
|
|
|
|
|
if (FAILED(hr) && hr != E_INVALIDARG) {
|
|
|
|
|
*errorMessage= msgComFailed("GetModuleNameStringWide", hr);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
module.moduleName = QString::fromUtf16(wszBuf);
|
|
|
|
|
modules->push_back(module);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2009-04-15 10:05:40 +02:00
|
|
|
// Search symbols matching a pattern
|
|
|
|
|
bool searchSymbols(IDebugSymbols3 *syms, const QString &pattern,
|
|
|
|
|
QStringList *matches, QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
matches->clear();
|
2009-04-15 12:01:58 +02:00
|
|
|
ULONG64 handle = 0;
|
|
|
|
|
// E_NOINTERFACE means "no match". Apparently, it does not always
|
|
|
|
|
// set handle.
|
2009-04-15 10:05:40 +02:00
|
|
|
HRESULT hr = syms->StartSymbolMatchWide(pattern.utf16(), &handle);
|
|
|
|
|
if (hr == E_NOINTERFACE) {
|
2009-04-15 12:01:58 +02:00
|
|
|
if (handle)
|
|
|
|
|
syms->EndSymbolMatch(handle);
|
2009-04-15 10:05:40 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
|
*errorMessage= msgComFailed("StartSymbolMatchWide", hr);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
WCHAR wszBuf[MAX_PATH];
|
|
|
|
|
while (true) {
|
|
|
|
|
hr = syms->GetNextSymbolMatchWide(handle, wszBuf, MAX_PATH - 1, 0, 0);
|
|
|
|
|
if (hr == E_NOINTERFACE)
|
|
|
|
|
break;
|
|
|
|
|
if (hr == S_OK)
|
|
|
|
|
matches->push_back(QString::fromUtf16(wszBuf));
|
|
|
|
|
}
|
|
|
|
|
syms->EndSymbolMatch(handle);
|
|
|
|
|
if (matches->empty())
|
|
|
|
|
*errorMessage = QString::fromLatin1("No symbol matches '%1'.").arg(pattern);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add missing the module specifier: "main" -> "project!main"
|
|
|
|
|
|
|
|
|
|
ResolveSymbolResult resolveSymbol(IDebugSymbols3 *syms, QString *symbol,
|
|
|
|
|
QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
// Is it an incomplete symbol?
|
|
|
|
|
if (symbol->contains(QLatin1Char('!')))
|
|
|
|
|
return ResolveSymbolOk;
|
|
|
|
|
// 'main' is a #define for gdb, but not for VS
|
|
|
|
|
if (*symbol == QLatin1String("qMain"))
|
|
|
|
|
*symbol = QLatin1String("main");
|
|
|
|
|
// resolve
|
|
|
|
|
QStringList matches;
|
|
|
|
|
if (!searchSymbols(syms, *symbol, &matches, errorMessage))
|
|
|
|
|
return ResolveSymbolError;
|
|
|
|
|
if (matches.empty())
|
|
|
|
|
return ResolveSymbolNotFound;
|
|
|
|
|
*symbol = matches.front();
|
|
|
|
|
if (matches.size() > 1) {
|
|
|
|
|
*errorMessage = QString::fromLatin1("Ambiguous symbol '%1': %2").
|
|
|
|
|
arg(*symbol, matches.join(QString(QLatin1Char(' '))));
|
|
|
|
|
return ResolveSymbolAmbiguous;
|
|
|
|
|
}
|
|
|
|
|
return ResolveSymbolOk;
|
|
|
|
|
}
|
|
|
|
|
|
2009-04-15 12:01:58 +02:00
|
|
|
// List symbols of a module
|
|
|
|
|
bool getModuleSymbols(IDebugSymbols3 *syms, const QString &moduleName,
|
|
|
|
|
QList<Symbol> *symbols, QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
// Search all symbols and retrieve addresses
|
|
|
|
|
symbols->clear();
|
|
|
|
|
QStringList matches;
|
|
|
|
|
const QString pattern = QFileInfo(moduleName).baseName() + QLatin1String("!*");
|
|
|
|
|
if (!searchSymbols(syms, pattern, &matches, errorMessage))
|
|
|
|
|
return false;
|
|
|
|
|
const QString hexPrefix = QLatin1String("0x");
|
|
|
|
|
foreach (const QString &name, matches) {
|
|
|
|
|
Symbol symbol;
|
|
|
|
|
symbol.name = name;
|
|
|
|
|
ULONG64 offset = 0;
|
|
|
|
|
if (SUCCEEDED(syms->GetOffsetByNameWide(name.utf16(), &offset)))
|
|
|
|
|
symbol.address = hexPrefix + QString::number(offset, 16);
|
|
|
|
|
symbols->push_back(symbol);
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2009-04-15 14:26:08 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|