2010-07-13 17:16:31 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2011-01-11 16:28:15 +01:00
|
|
|
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
2011-11-02 15:59:12 +01:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
|
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
|
**
|
2011-04-13 08:42:33 +02:00
|
|
|
** 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.
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
2011-04-13 08:42:33 +02:00
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
2010-12-17 16:01:08 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
2011-04-13 08:42:33 +02:00
|
|
|
** Other Usage
|
|
|
|
|
**
|
|
|
|
|
** Alternatively, this file may be used in accordance with the terms and
|
|
|
|
|
** conditions contained in a signed written agreement between you and Nokia.
|
|
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** If you have questions regarding the use of this file, please contact
|
2011-11-02 15:59:12 +01:00
|
|
|
** Nokia at qt-info@nokia.com.
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
2010-10-25 14:00:19 +02:00
|
|
|
#include "breakpoint.h"
|
2010-07-13 17:16:31 +02:00
|
|
|
|
2011-06-15 14:02:26 +02:00
|
|
|
#include "utils/qtcassert.h"
|
|
|
|
|
|
2010-07-13 17:16:31 +02:00
|
|
|
#include <QtCore/QByteArray>
|
|
|
|
|
#include <QtCore/QDebug>
|
2011-07-04 14:24:42 +02:00
|
|
|
#include <QtCore/QFileInfo>
|
2010-07-13 17:16:31 +02:00
|
|
|
|
|
|
|
|
namespace Debugger {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2011-06-15 14:02:26 +02:00
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
2011-06-24 16:25:30 +02:00
|
|
|
// BreakpointModelId
|
2011-06-15 14:02:26 +02:00
|
|
|
//
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
/*!
|
|
|
|
|
\class Debugger::Internal::ModelId
|
|
|
|
|
|
|
|
|
|
This identifies a breakpoint in the \c BreakHandler. The
|
|
|
|
|
major parts are strictly increasing over time.
|
2011-06-30 11:01:10 +02:00
|
|
|
|
|
|
|
|
The minor part identifies a multiple breakpoint
|
|
|
|
|
set for example by gdb in constructors.
|
2011-06-24 16:25:30 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
QDebug operator<<(QDebug d, const BreakpointModelId &id)
|
|
|
|
|
{
|
|
|
|
|
d << qPrintable(id.toString());
|
|
|
|
|
return d;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QByteArray BreakpointModelId::toByteArray() const
|
|
|
|
|
{
|
|
|
|
|
if (!isValid())
|
|
|
|
|
return "<invalid bkpt>";
|
|
|
|
|
QByteArray ba = QByteArray::number(m_majorPart);
|
|
|
|
|
if (isMinor()) {
|
|
|
|
|
ba.append('.');
|
|
|
|
|
ba.append(QByteArray::number(m_minorPart));
|
|
|
|
|
}
|
|
|
|
|
return ba;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString BreakpointModelId::toString() const
|
|
|
|
|
{
|
|
|
|
|
if (!isValid())
|
2011-12-21 14:02:52 +01:00
|
|
|
return QLatin1String("<invalid bkpt>");
|
2011-06-24 16:25:30 +02:00
|
|
|
if (isMinor())
|
2011-12-21 14:02:52 +01:00
|
|
|
return QString::fromLatin1("%1.%2").arg(m_majorPart).arg(m_minorPart);
|
2011-06-24 16:25:30 +02:00
|
|
|
return QString::number(m_majorPart);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BreakpointModelId BreakpointModelId::parent() const
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(isMinor(), return BreakpointModelId());
|
|
|
|
|
return BreakpointModelId(m_majorPart, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BreakpointModelId BreakpointModelId::child(int row) const
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(isMajor(), return BreakpointModelId());
|
|
|
|
|
return BreakpointModelId(m_majorPart, row + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// BreakpointResponseId
|
|
|
|
|
//
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\class Debugger::Internal::BreakpointResponseId
|
|
|
|
|
|
|
|
|
|
This is what the external debuggers use to identify a breakpoint.
|
|
|
|
|
It is only valid for one debugger run.
|
2011-06-30 11:01:10 +02:00
|
|
|
|
|
|
|
|
In gdb, the breakpoint number is used, which is constant
|
|
|
|
|
during a session. CDB's breakpoint numbers vary if breakpoints
|
|
|
|
|
are deleted, so, the ID is used.
|
2011-06-24 16:25:30 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
BreakpointResponseId::BreakpointResponseId(const QByteArray &ba)
|
2011-06-21 16:45:23 +02:00
|
|
|
{
|
|
|
|
|
int pos = ba.indexOf('.');
|
|
|
|
|
if (pos == -1) {
|
|
|
|
|
m_majorPart = ba.toInt();
|
|
|
|
|
m_minorPart = 0;
|
|
|
|
|
} else {
|
|
|
|
|
m_majorPart = ba.left(pos).toInt();
|
|
|
|
|
m_minorPart = ba.mid(pos + 1).toInt();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
QDebug operator<<(QDebug d, const BreakpointResponseId &id)
|
2011-06-15 14:02:26 +02:00
|
|
|
{
|
|
|
|
|
d << qPrintable(id.toString());
|
|
|
|
|
return d;
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
QByteArray BreakpointResponseId::toByteArray() const
|
2011-06-21 16:45:23 +02:00
|
|
|
{
|
|
|
|
|
if (!isValid())
|
|
|
|
|
return "<invalid bkpt>";
|
|
|
|
|
QByteArray ba = QByteArray::number(m_majorPart);
|
|
|
|
|
if (isMinor()) {
|
|
|
|
|
ba.append('.');
|
|
|
|
|
ba.append(QByteArray::number(m_minorPart));
|
|
|
|
|
}
|
|
|
|
|
return ba;
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
QString BreakpointResponseId::toString() const
|
2011-06-15 14:02:26 +02:00
|
|
|
{
|
|
|
|
|
if (!isValid())
|
2011-12-21 14:02:52 +01:00
|
|
|
return QLatin1String("<invalid bkpt>");
|
2011-06-15 14:02:26 +02:00
|
|
|
if (isMinor())
|
2011-12-21 14:02:52 +01:00
|
|
|
return QString::fromLatin1("%1.%2").arg(m_majorPart).arg(m_minorPart);
|
2011-06-15 14:02:26 +02:00
|
|
|
return QString::number(m_majorPart);
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
BreakpointResponseId BreakpointResponseId::parent() const
|
2011-06-15 14:02:26 +02:00
|
|
|
{
|
2011-06-24 16:25:30 +02:00
|
|
|
QTC_ASSERT(isMinor(), return BreakpointResponseId());
|
|
|
|
|
return BreakpointResponseId(m_majorPart, 0);
|
2011-06-15 14:02:26 +02:00
|
|
|
}
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
BreakpointResponseId BreakpointResponseId::child(int row) const
|
2011-06-15 14:02:26 +02:00
|
|
|
{
|
2011-06-24 16:25:30 +02:00
|
|
|
QTC_ASSERT(isMajor(), return BreakpointResponseId());
|
|
|
|
|
return BreakpointResponseId(m_majorPart, row + 1);
|
2011-06-15 14:02:26 +02:00
|
|
|
}
|
|
|
|
|
|
2010-07-13 17:16:31 +02:00
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
2010-11-16 11:48:17 +01:00
|
|
|
// BreakpointParameters
|
2010-07-13 17:16:31 +02:00
|
|
|
//
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2011-02-04 15:08:31 +01:00
|
|
|
/*!
|
|
|
|
|
\class Debugger::Internal::BreakpointParameters
|
|
|
|
|
|
|
|
|
|
Data type holding the parameters of a breakpoint.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-11-16 11:48:17 +01:00
|
|
|
BreakpointParameters::BreakpointParameters(BreakpointType t)
|
2011-02-17 13:00:11 +01:00
|
|
|
: type(t), enabled(true), pathUsage(BreakpointPathUsageEngineDefault),
|
2011-03-01 19:16:24 +01:00
|
|
|
ignoreCount(0), lineNumber(0), address(0), size(0),
|
|
|
|
|
bitpos(0), bitsize(0), threadSpec(-1),
|
2010-12-16 13:02:59 +01:00
|
|
|
tracepoint(false)
|
2010-11-16 11:55:48 +01:00
|
|
|
{}
|
2010-07-13 17:16:31 +02:00
|
|
|
|
2011-03-29 12:55:36 +02:00
|
|
|
BreakpointParts BreakpointParameters::differencesTo
|
|
|
|
|
(const BreakpointParameters &rhs) const
|
|
|
|
|
{
|
|
|
|
|
BreakpointParts parts = BreakpointParts();
|
|
|
|
|
if (type != rhs.type)
|
|
|
|
|
parts |= TypePart;
|
|
|
|
|
if (enabled != rhs.enabled)
|
|
|
|
|
parts |= EnabledPart;
|
|
|
|
|
if (pathUsage != rhs.pathUsage)
|
|
|
|
|
parts |= PathUsagePart;
|
|
|
|
|
if (fileName != rhs.fileName)
|
|
|
|
|
parts |= FileAndLinePart;
|
2011-05-02 11:43:41 +02:00
|
|
|
if (!conditionsMatch(rhs.condition))
|
2011-03-29 12:55:36 +02:00
|
|
|
parts |= ConditionPart;
|
|
|
|
|
if (ignoreCount != rhs.ignoreCount)
|
|
|
|
|
parts |= IgnoreCountPart;
|
|
|
|
|
if (lineNumber != rhs.lineNumber)
|
|
|
|
|
parts |= FileAndLinePart;
|
|
|
|
|
if (address != rhs.address)
|
|
|
|
|
parts |= AddressPart;
|
|
|
|
|
if (threadSpec != rhs.threadSpec)
|
|
|
|
|
parts |= ThreadSpecPart;
|
|
|
|
|
if (functionName != rhs.functionName)
|
|
|
|
|
parts |= FunctionPart;
|
|
|
|
|
if (tracepoint != rhs.tracepoint)
|
|
|
|
|
parts |= TracePointPart;
|
|
|
|
|
if (module != rhs.module)
|
|
|
|
|
parts |= ModulePart;
|
|
|
|
|
if (command != rhs.command)
|
|
|
|
|
parts |= CommandPart;
|
2011-06-27 10:37:57 +02:00
|
|
|
if (message != rhs.message)
|
|
|
|
|
parts |= MessagePart;
|
2011-03-29 12:55:36 +02:00
|
|
|
return parts;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-20 10:45:59 +02:00
|
|
|
bool BreakpointParameters::isValid() const
|
|
|
|
|
{
|
|
|
|
|
switch (type) {
|
|
|
|
|
case Debugger::Internal::BreakpointByFileAndLine:
|
|
|
|
|
return !fileName.isEmpty() && lineNumber > 0;
|
|
|
|
|
case Debugger::Internal::BreakpointByFunction:
|
|
|
|
|
return !functionName.isEmpty();
|
|
|
|
|
case Debugger::Internal::WatchpointAtAddress:
|
|
|
|
|
case Debugger::Internal::BreakpointByAddress:
|
|
|
|
|
return address != 0;
|
|
|
|
|
case Debugger::Internal::BreakpointAtThrow:
|
|
|
|
|
case Debugger::Internal::BreakpointAtCatch:
|
|
|
|
|
case Debugger::Internal::BreakpointAtMain:
|
|
|
|
|
case Debugger::Internal::BreakpointAtFork:
|
|
|
|
|
case Debugger::Internal::BreakpointAtExec:
|
|
|
|
|
case Debugger::Internal::BreakpointAtSysCall:
|
|
|
|
|
case Debugger::Internal::BreakpointOnQmlSignalHandler:
|
|
|
|
|
case Debugger::Internal::BreakpointAtJavaScriptThrow:
|
|
|
|
|
break;
|
|
|
|
|
case Debugger::Internal::WatchpointAtExpression:
|
|
|
|
|
return !expression.isEmpty();
|
|
|
|
|
case Debugger::Internal::UnknownType:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-15 17:04:29 +01:00
|
|
|
bool BreakpointParameters::equals(const BreakpointParameters &rhs) const
|
|
|
|
|
{
|
2011-03-29 12:55:36 +02:00
|
|
|
return !differencesTo(rhs);
|
2010-11-15 17:04:29 +01:00
|
|
|
}
|
|
|
|
|
|
2010-11-16 11:06:09 +01:00
|
|
|
bool BreakpointParameters::conditionsMatch(const QByteArray &other) const
|
2010-07-13 17:16:31 +02:00
|
|
|
{
|
|
|
|
|
// Some versions of gdb "beautify" the passed condition.
|
2010-11-16 11:06:09 +01:00
|
|
|
QByteArray s1 = condition;
|
2010-11-15 16:22:51 +01:00
|
|
|
s1.replace(' ', "");
|
|
|
|
|
QByteArray s2 = other;
|
|
|
|
|
s2.replace(' ', "");
|
2010-07-13 17:16:31 +02:00
|
|
|
return s1 == s2;
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-04 14:24:42 +02:00
|
|
|
void BreakpointParameters::updateLocation(const QByteArray &location)
|
2011-06-24 14:17:16 +02:00
|
|
|
{
|
|
|
|
|
if (location.size()) {
|
|
|
|
|
int pos = location.indexOf(':');
|
|
|
|
|
lineNumber = location.mid(pos + 1).toInt();
|
2011-07-04 14:24:42 +02:00
|
|
|
QString file = QString::fromUtf8(location.left(pos));
|
|
|
|
|
if (file.startsWith(QLatin1Char('"')) && file.endsWith(QLatin1Char('"')))
|
|
|
|
|
file = file.mid(1, file.size() - 2);
|
|
|
|
|
QFileInfo fi(file);
|
|
|
|
|
if (fi.isReadable())
|
|
|
|
|
fileName = fi.absoluteFilePath();
|
2011-06-24 14:17:16 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-16 11:48:17 +01:00
|
|
|
QString BreakpointParameters::toString() const
|
2010-11-10 16:33:11 +01:00
|
|
|
{
|
|
|
|
|
QString result;
|
|
|
|
|
QTextStream ts(&result);
|
2011-03-16 13:39:29 +01:00
|
|
|
ts << "Type: " << type;
|
|
|
|
|
switch (type) {
|
2011-03-22 10:24:33 +01:00
|
|
|
case BreakpointByFileAndLine:
|
2011-03-16 13:39:29 +01:00
|
|
|
ts << " FileName: " << fileName << ':' << lineNumber
|
|
|
|
|
<< " PathUsage: " << pathUsage;
|
|
|
|
|
break;
|
2011-03-22 10:24:33 +01:00
|
|
|
case BreakpointByFunction:
|
2011-09-28 17:55:48 +02:00
|
|
|
case BreakpointOnQmlSignalHandler:
|
2011-03-16 13:39:29 +01:00
|
|
|
ts << " FunctionName: " << functionName;
|
|
|
|
|
break;
|
2011-03-22 10:24:33 +01:00
|
|
|
case BreakpointByAddress:
|
2011-05-09 08:35:58 +02:00
|
|
|
case WatchpointAtAddress:
|
2011-03-16 13:39:29 +01:00
|
|
|
ts << " Address: " << address;
|
|
|
|
|
break;
|
2011-05-09 08:35:58 +02:00
|
|
|
case WatchpointAtExpression:
|
|
|
|
|
ts << " Expression: " << expression;
|
|
|
|
|
break;
|
2011-03-22 10:24:33 +01:00
|
|
|
case BreakpointAtThrow:
|
|
|
|
|
case BreakpointAtCatch:
|
|
|
|
|
case BreakpointAtMain:
|
|
|
|
|
case BreakpointAtFork:
|
|
|
|
|
case BreakpointAtExec:
|
2011-03-29 09:58:21 +02:00
|
|
|
//case BreakpointAtVFork:
|
2011-03-22 10:24:33 +01:00
|
|
|
case BreakpointAtSysCall:
|
2011-10-10 15:25:18 +02:00
|
|
|
case BreakpointAtJavaScriptThrow:
|
2011-03-22 10:24:33 +01:00
|
|
|
case UnknownType:
|
2011-03-16 13:39:29 +01:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
ts << (enabled ? " [enabled]" : " [disabled]");
|
|
|
|
|
if (!condition.isEmpty())
|
|
|
|
|
ts << " Condition: " << condition;
|
|
|
|
|
if (ignoreCount)
|
|
|
|
|
ts << " IgnoreCount: " << ignoreCount;
|
|
|
|
|
if (tracepoint)
|
|
|
|
|
ts << " [tracepoint]";
|
|
|
|
|
if (!module.isEmpty())
|
|
|
|
|
ts << " Module: " << module;
|
|
|
|
|
if (!command.isEmpty())
|
|
|
|
|
ts << " Command: " << command;
|
2011-06-27 10:37:57 +02:00
|
|
|
if (!message.isEmpty())
|
|
|
|
|
ts << " Message: " << message;
|
2010-11-10 16:33:11 +01:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-16 11:55:48 +01:00
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
2011-02-04 15:08:31 +01:00
|
|
|
// BreakpointResponse
|
2010-11-16 11:55:48 +01:00
|
|
|
//
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2011-02-04 15:08:31 +01:00
|
|
|
/*!
|
|
|
|
|
\class Debugger::Internal::BreakpointResponse
|
|
|
|
|
|
|
|
|
|
This is what debuggers produce in response to the attempt to
|
|
|
|
|
insert a breakpoint. The data might differ from the requested bits.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-11-16 11:55:48 +01:00
|
|
|
BreakpointResponse::BreakpointResponse()
|
2011-06-15 14:02:26 +02:00
|
|
|
{
|
|
|
|
|
pending = true;
|
2011-06-21 16:45:23 +02:00
|
|
|
hitCount = 0;
|
2011-06-15 14:02:26 +02:00
|
|
|
multiple = false;
|
|
|
|
|
correctedLineNumber = 0;
|
|
|
|
|
}
|
2010-11-16 11:55:48 +01:00
|
|
|
|
2010-11-10 16:33:11 +01:00
|
|
|
QString BreakpointResponse::toString() const
|
|
|
|
|
{
|
2011-03-16 13:39:29 +01:00
|
|
|
QString result = BreakpointParameters::toString();
|
2010-11-10 16:33:11 +01:00
|
|
|
QTextStream ts(&result);
|
2011-06-21 16:45:23 +02:00
|
|
|
ts << " Number: " << id.toString();
|
2011-03-16 13:39:29 +01:00
|
|
|
if (pending)
|
|
|
|
|
ts << " [pending]";
|
2011-06-15 14:02:26 +02:00
|
|
|
if (!functionName.isEmpty())
|
|
|
|
|
ts << " Function: " << functionName;
|
2011-03-16 13:39:29 +01:00
|
|
|
if (multiple)
|
|
|
|
|
ts << " Multiple: " << multiple;
|
|
|
|
|
if (correctedLineNumber)
|
|
|
|
|
ts << " CorrectedLineNumber: " << correctedLineNumber;
|
2011-06-21 16:45:23 +02:00
|
|
|
ts << " Hit: " << hitCount << " times";
|
2011-06-15 14:02:26 +02:00
|
|
|
ts << ' ';
|
2010-11-16 17:53:08 +01:00
|
|
|
return result + BreakpointParameters::toString();
|
2010-11-10 16:33:11 +01:00
|
|
|
}
|
|
|
|
|
|
2010-11-16 10:23:20 +01:00
|
|
|
void BreakpointResponse::fromParameters(const BreakpointParameters &p)
|
|
|
|
|
{
|
|
|
|
|
BreakpointParameters::operator=(p);
|
2011-06-24 16:25:30 +02:00
|
|
|
id = BreakpointResponseId();
|
2010-11-16 10:23:20 +01:00
|
|
|
multiple = false;
|
2010-11-18 12:30:56 +01:00
|
|
|
correctedLineNumber = 0;
|
2011-06-21 16:45:23 +02:00
|
|
|
hitCount = 0;
|
2010-11-16 10:23:20 +01:00
|
|
|
}
|
|
|
|
|
|
2010-07-13 17:16:31 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|