forked from qt-creator/qt-creator
344 lines
11 KiB
C++
344 lines
11 KiB
C++
/**************************************************************************
|
|
**
|
|
** This file is part of Qt Creator
|
|
**
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
|
**
|
|
** 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 "cdbapplication.h"
|
|
#include "coreengine.h"
|
|
#include "cdbdebugoutput.h"
|
|
#include "cdbpromptthread.h"
|
|
#include "debugeventcallback.h"
|
|
#include "symbolgroupcontext.h"
|
|
#include "stacktracecontext.h"
|
|
|
|
#include <QtCore/QStringList>
|
|
#include <QtCore/QTimer>
|
|
#include <QtCore/QDebug>
|
|
|
|
#include <cstdio>
|
|
|
|
const char usage[] =
|
|
"CDB command line test tool\n\n"
|
|
"ccdb <Options>\n"
|
|
"Options: -p engine path\n";
|
|
|
|
class PrintfOutputHandler : public CdbCore::DebugOutputBase
|
|
{
|
|
public:
|
|
PrintfOutputHandler() {}
|
|
|
|
protected:
|
|
virtual void output(ULONG mask, const QString &message)
|
|
{ std::printf("%10s: %s\n", maskDescription(mask), qPrintable(message)); }
|
|
};
|
|
|
|
|
|
// -------------- CdbApplication
|
|
CdbApplication::CdbApplication(int argc, char *argv[]) :
|
|
QCoreApplication(argc, argv),
|
|
m_engine(new CdbCore::CoreEngine),
|
|
m_promptThread(0),
|
|
m_processHandle(0)
|
|
{
|
|
}
|
|
|
|
CdbApplication::~CdbApplication()
|
|
{
|
|
}
|
|
|
|
CdbApplication::InitResult CdbApplication::init()
|
|
{
|
|
if (!parseOptions()) {
|
|
printf(usage);
|
|
return InitUsageShown;
|
|
}
|
|
QString errorMessage;
|
|
std::printf("Initializing engine %s...\n", qPrintable(m_engineDll));
|
|
if (!m_engine->init(m_engineDll, &errorMessage)) {
|
|
std::fprintf(stderr, "Failed: %s\n", qPrintable(errorMessage));
|
|
return InitFailed;
|
|
}
|
|
m_engine->setDebugOutput(CdbCore::CoreEngine::DebugOutputBasePtr(new PrintfOutputHandler));
|
|
DebugEventCallback *evt = new DebugEventCallback;
|
|
connect(evt, SIGNAL(processAttached(void*)), this, SLOT(processAttached(void*)));
|
|
m_engine->setDebugEventCallback(CdbCore::CoreEngine::DebugEventCallbackBasePtr(evt));
|
|
m_engine->setExpressionSyntax(CdbCore::CoreEngine::CppExpressionSyntax);
|
|
m_engine->setCodeLevel(CdbCore::CoreEngine::CodeLevelSource);
|
|
connect(m_engine.data(), SIGNAL(watchTimerDebugEvent()), this, SLOT(debugEvent()));
|
|
std::printf("Succeded.\n");
|
|
// Prompt
|
|
m_promptThread = new CdbPromptThread(this);
|
|
connect(m_promptThread, SIGNAL(finished()), this, SLOT(promptThreadTerminated()));
|
|
connect(m_promptThread, SIGNAL(asyncCommand(int,QString)),
|
|
this, SLOT(asyncCommand(int,QString)), Qt::QueuedConnection);
|
|
connect(m_promptThread, SIGNAL(syncCommand(int,QString)),
|
|
this, SLOT(syncCommand(int,QString)), Qt::BlockingQueuedConnection);
|
|
connect(m_promptThread, SIGNAL(executionCommand(int,QString)),
|
|
this, SLOT(executionCommand(int,QString)), Qt::BlockingQueuedConnection);
|
|
|
|
m_promptThread->start();
|
|
return InitOk;
|
|
}
|
|
|
|
void CdbApplication::promptThreadTerminated()
|
|
{
|
|
QString errorMessage;
|
|
m_engine->endSession(&errorMessage);
|
|
std::printf("Terminating.\n");
|
|
m_promptThread->wait();
|
|
quit();
|
|
}
|
|
|
|
bool CdbApplication::parseOptions()
|
|
{
|
|
const QStringList args = QCoreApplication::arguments();
|
|
const QStringList::const_iterator cend = args.constEnd();
|
|
QStringList::const_iterator it = args.constBegin();
|
|
for (++it; it != cend ; ++it) {
|
|
const QString &a = *it;
|
|
if (a == QLatin1String("-p")) {
|
|
++it;
|
|
if (it == cend) {
|
|
std::fprintf(stderr, "Option -p is missing an argument.\n");
|
|
return false;
|
|
}
|
|
m_engineDll = *it;
|
|
} else {
|
|
std::fprintf(stderr, "Invalid option %s\n", qPrintable(a));
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void CdbApplication::asyncCommand(int command, const QString &arg)
|
|
{
|
|
Q_UNUSED(arg)
|
|
QString errorMessage;
|
|
switch (command) {
|
|
case Async_Interrupt:
|
|
if (m_processHandle) {
|
|
if (m_engine->debugBreakProcess(m_processHandle, &errorMessage)) {
|
|
std::printf("Stopped\n");
|
|
} else {
|
|
std::printf("%s\n", qPrintable(errorMessage));
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CdbApplication::printFrame(const QString &arg)
|
|
{
|
|
QString errorMessage;
|
|
do {
|
|
if (m_stackTrace.isNull()) {
|
|
errorMessage = QLatin1String("No trace.");
|
|
break;
|
|
}
|
|
bool ok;
|
|
const int frame = arg.toInt(&ok);
|
|
if (!ok || frame < 0 || frame >= m_stackTrace->frameCount()) {
|
|
errorMessage = QLatin1String("Invalid or out of range.");
|
|
break;
|
|
}
|
|
CdbCore::SymbolGroupContext *ctx = m_stackTrace->symbolGroupContextAt(frame, &errorMessage);
|
|
if (!ctx)
|
|
break;
|
|
printf("%s\n", qPrintable(ctx->toString()));
|
|
} while (false);
|
|
if (!errorMessage.isEmpty())
|
|
printf("%s\n", qPrintable(errorMessage));
|
|
}
|
|
|
|
// Return address or 0 on failure
|
|
quint64 CdbApplication::addQueuedBreakPoint(const QString &arg, QString *errorMessage)
|
|
{
|
|
// Queue file:line
|
|
const int cpos = arg.lastIndexOf(QLatin1Char(':'));
|
|
if (cpos == -1) {
|
|
*errorMessage = QString::fromLatin1("Syntax error in '%1': No colon.").arg(arg);
|
|
return 0;
|
|
}
|
|
|
|
const QString fileName = arg.left(cpos);
|
|
bool ok;
|
|
const int lineNumber = arg.mid(cpos + 1).toInt(&ok);
|
|
if (!ok || lineNumber < 1) {
|
|
*errorMessage = QString::fromLatin1("Syntax error in '%1': No line number.").arg(arg);
|
|
return 0;
|
|
}
|
|
CdbCore::BreakPoint bp;
|
|
bp.address = m_engine->getSourceLineAddress(fileName, lineNumber, errorMessage);
|
|
if (!bp.address)
|
|
return 0;
|
|
if (!bp.add(m_engine->interfaces().debugControl, errorMessage))
|
|
return 0;
|
|
return bp.address;
|
|
}
|
|
|
|
void CdbApplication::syncCommand(int command, const QString &arg)
|
|
{
|
|
QString errorMessage;
|
|
switch (command) {
|
|
case Sync_EvalExpression: {
|
|
QString value;
|
|
QString type;
|
|
std::printf("Evaluating '%s' in code level %d, syntax %d\n",
|
|
qPrintable(arg), m_engine->codeLevel(), m_engine->expressionSyntax());
|
|
if (m_engine->evaluateExpression(arg, &value, &type, &errorMessage)) {
|
|
std::printf("[%s] %s\n", qPrintable(type), qPrintable(value));
|
|
} else {
|
|
std::printf("%s\n", qPrintable(errorMessage));
|
|
}
|
|
}
|
|
break;
|
|
case Sync_Queue: {
|
|
const QString targs = arg.trimmed();
|
|
if (targs.isEmpty()) {
|
|
std::printf("Queue cleared\n");
|
|
m_queuedCommands.clear();
|
|
} else {
|
|
std::printf("Queueing %s\n", qPrintable(targs));
|
|
m_queuedCommands.push_back(targs);
|
|
}
|
|
}
|
|
break;
|
|
case Sync_QueueBreakPoint: {
|
|
const QString targs = arg.trimmed();
|
|
if (targs.isEmpty()) {
|
|
std::printf("Breakpoint queue cleared\n");
|
|
m_queuedBreakPoints.clear();
|
|
} else {
|
|
m_queuedBreakPoints.push_back(targs);
|
|
std::printf("Queueing breakpoint %s\n", qPrintable(targs));
|
|
}
|
|
}
|
|
break;
|
|
case Sync_ListBreakPoints: {
|
|
QList<CdbCore::BreakPoint> bps;
|
|
if (CdbCore::BreakPoint::getBreakPoints(m_engine->interfaces().debugControl, &bps, &errorMessage)) {
|
|
foreach (const CdbCore::BreakPoint &bp, bps)
|
|
std::printf("%s\n", qPrintable(bp.expression()));
|
|
} else {
|
|
std::printf("BREAKPOINT LIST FAILED: %s\n", qPrintable(errorMessage));
|
|
}
|
|
}
|
|
break;
|
|
case Sync_PrintFrame:
|
|
printFrame(arg);
|
|
break;
|
|
case Unknown:
|
|
std::printf("Executing '%s' in code level %d, syntax %d\n",
|
|
qPrintable(arg), m_engine->codeLevel(), m_engine->expressionSyntax());
|
|
if (!m_engine->executeDebuggerCommand(arg, &errorMessage))
|
|
std::printf("%s\n", qPrintable(errorMessage));
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CdbApplication::executionCommand(int command, const QString &arg)
|
|
{
|
|
bool ok = false;
|
|
QString errorMessage;
|
|
switch (command) {
|
|
case Execution_StartBinary: {
|
|
QStringList args = arg.split(QLatin1Char(' '), QString::SkipEmptyParts);
|
|
if (args.isEmpty()) {
|
|
errorMessage = QLatin1String("Specify executable.");
|
|
} else {
|
|
std::printf("Starting\n");
|
|
const QString binary = args.front();
|
|
args.pop_front();
|
|
ok = m_engine->startDebuggerWithExecutable(QString(), binary, args,
|
|
QStringList(), &errorMessage);
|
|
}
|
|
}
|
|
break;
|
|
case Execution_Go:
|
|
std::printf("Go\n");
|
|
ok = m_engine->setExecutionStatus(DEBUG_STATUS_GO, &errorMessage);
|
|
break;
|
|
}
|
|
if (ok) {
|
|
m_engine->startWatchTimer();
|
|
m_stackTrace.reset();
|
|
} else {
|
|
std::fprintf(stderr, "%s\n", qPrintable(errorMessage));
|
|
}
|
|
}
|
|
|
|
void CdbApplication::debugEvent()
|
|
{
|
|
QString errorMessage;
|
|
std::printf("Debug event\n");
|
|
|
|
CdbCore::StackTraceContext::ThreadIdFrameMap threads;
|
|
ULONG currentThreadId;
|
|
if (CdbCore::StackTraceContext::getThreads(m_engine->interfaces(), &threads,
|
|
¤tThreadId, &errorMessage)) {
|
|
printf("%s\n", qPrintable(CdbCore::StackTraceContext::formatThreads(threads)));
|
|
} else {
|
|
std::fprintf(stderr, "%s\n", qPrintable(errorMessage));
|
|
}
|
|
|
|
CdbCore::StackTraceContext *trace =
|
|
CdbCore::StackTraceContext::create(&m_engine->interfaces(),
|
|
0xFFFF, &errorMessage);
|
|
if (trace) {
|
|
m_stackTrace.reset(trace);
|
|
printf("%s\n", qPrintable(m_stackTrace->toString()));
|
|
} else {
|
|
std::fprintf(stderr, "%s\n", qPrintable(errorMessage));
|
|
}
|
|
}
|
|
|
|
void CdbApplication::processAttached(void *handle)
|
|
{
|
|
std::printf("### Process attached\n");
|
|
m_processHandle = handle;
|
|
QString errorMessage;
|
|
// Commands
|
|
foreach(const QString &qc, m_queuedCommands) {
|
|
if (m_engine->executeDebuggerCommand(qc, &errorMessage)) {
|
|
std::printf("'%s' [ok]\n", qPrintable(qc));
|
|
} else {
|
|
std::printf("%s\n", qPrintable(errorMessage));
|
|
}
|
|
}
|
|
// Breakpoints
|
|
foreach(const QString &bp, m_queuedBreakPoints) {
|
|
if (const quint64 address = addQueuedBreakPoint(bp, &errorMessage)) {
|
|
std::printf("'%s' 0x%lx [ok]\n", qPrintable(bp), address);
|
|
} else {
|
|
std::fprintf(stderr, "%s: %s\n",
|
|
qPrintable(bp),
|
|
qPrintable(errorMessage));
|
|
}
|
|
}
|
|
}
|