forked from qt-creator/qt-creator
		
	
		
			
				
	
	
		
			214 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			214 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/**************************************************************************
 | 
						|
**
 | 
						|
** This file is part of Qt Creator
 | 
						|
**
 | 
						|
** Copyright (c) 2009 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 <QtCore/QStringList>
 | 
						|
#include <QtCore/QTimer>
 | 
						|
 | 
						|
#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::syncCommand(int command, const QString &arg)
 | 
						|
{
 | 
						|
    QString errorMessage;
 | 
						|
    switch (command) {
 | 
						|
    case Sync_EvalExpression: {
 | 
						|
            QString value;
 | 
						|
            QString type;
 | 
						|
            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 Unknown:
 | 
						|
        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(), false,
 | 
						|
                                                           &errorMessage);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        break;
 | 
						|
    case Execution_Go:
 | 
						|
        std::printf("Go\n");
 | 
						|
        ok = m_engine->setExecutionStatus(DEBUG_STATUS_GO, &errorMessage);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
    if (ok) {
 | 
						|
        m_engine->startWatchTimer();
 | 
						|
    } else {
 | 
						|
        std::fprintf(stderr, "%s\n", qPrintable(errorMessage));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void CdbApplication::debugEvent()
 | 
						|
{
 | 
						|
    std::printf("Debug event\n");
 | 
						|
}
 | 
						|
 | 
						|
void CdbApplication::processAttached(void *handle)
 | 
						|
{
 | 
						|
    std::printf("pe\n");
 | 
						|
    m_processHandle = handle;
 | 
						|
}
 |