2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2016-01-15 14:57:40 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2010-07-13 17:16:31 +02:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2010-12-17 16:01:08 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
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
|
|
|
|
2013-09-02 14:04:12 +02:00
|
|
|
#include <utils/qtcassert.h>
|
2011-06-15 14:02:26 +02:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
|
|
|
|
#include <QFileInfo>
|
2010-07-13 17:16:31 +02:00
|
|
|
|
|
|
|
|
namespace Debugger {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2011-06-24 16:25:30 +02:00
|
|
|
/*!
|
2015-06-11 18:05:43 +02:00
|
|
|
\class Debugger::Internal::BreakpointIdBase
|
2011-06-24 16:25:30 +02:00
|
|
|
|
2015-06-11 18:05:43 +02:00
|
|
|
Convenience base class for BreakpointModelId and
|
|
|
|
|
BreakpointResponseId.
|
2011-06-24 16:25:30 +02:00
|
|
|
*/
|
|
|
|
|
|
2015-06-11 18:05:43 +02:00
|
|
|
QDebug operator<<(QDebug d, const BreakpointIdBase &id)
|
2011-06-24 16:25:30 +02:00
|
|
|
{
|
|
|
|
|
d << qPrintable(id.toString());
|
|
|
|
|
return d;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-11 18:05:43 +02:00
|
|
|
QByteArray BreakpointIdBase::toByteArray() const
|
2011-06-24 16:25:30 +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;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-11 18:05:43 +02:00
|
|
|
QString BreakpointIdBase::toString() const
|
2011-06-24 16:25:30 +02:00
|
|
|
{
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-06-11 18:05:43 +02:00
|
|
|
/*!
|
|
|
|
|
\class Debugger::Internal::BreakpointModelId
|
|
|
|
|
|
|
|
|
|
This identifies a breakpoint in the \c BreakHandler. The
|
|
|
|
|
major parts are strictly increasing over time.
|
|
|
|
|
|
|
|
|
|
The minor part identifies a multiple breakpoint
|
|
|
|
|
set for example by gdb in constructors.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
BreakpointModelId::BreakpointModelId(const QByteArray &ba)
|
|
|
|
|
{
|
|
|
|
|
int pos = ba.indexOf('\'');
|
|
|
|
|
if (pos == -1) {
|
|
|
|
|
m_majorPart = ba.toUShort();
|
|
|
|
|
m_minorPart = 0;
|
|
|
|
|
} else {
|
|
|
|
|
m_majorPart = ba.left(pos).toUShort();
|
|
|
|
|
m_minorPart = ba.mid(pos + 1).toUShort();
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-06-24 16:25:30 +02:00
|
|
|
|
|
|
|
|
/*!
|
|
|
|
|
\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();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
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),
|
2012-08-21 14:16:07 +02:00
|
|
|
tracepoint(false), oneShot(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;
|
2012-08-21 14:16:07 +02:00
|
|
|
if (oneShot != rhs.oneShot)
|
|
|
|
|
parts |= OneShotPart;
|
2011-03-29 12:55:36 +02:00
|
|
|
return parts;
|
|
|
|
|
}
|
|
|
|
|
|
2011-10-20 10:45:59 +02:00
|
|
|
bool BreakpointParameters::isValid() const
|
|
|
|
|
{
|
|
|
|
|
switch (type) {
|
2012-08-21 14:16:07 +02:00
|
|
|
case BreakpointByFileAndLine:
|
2011-10-20 10:45:59 +02:00
|
|
|
return !fileName.isEmpty() && lineNumber > 0;
|
2012-08-21 14:16:07 +02:00
|
|
|
case BreakpointByFunction:
|
2011-10-20 10:45:59 +02:00
|
|
|
return !functionName.isEmpty();
|
2012-08-21 14:16:07 +02:00
|
|
|
case WatchpointAtAddress:
|
|
|
|
|
case BreakpointByAddress:
|
2011-10-20 10:45:59 +02:00
|
|
|
return address != 0;
|
2012-08-21 14:16:07 +02:00
|
|
|
case BreakpointAtThrow:
|
|
|
|
|
case BreakpointAtCatch:
|
|
|
|
|
case BreakpointAtMain:
|
|
|
|
|
case BreakpointAtFork:
|
|
|
|
|
case BreakpointAtExec:
|
|
|
|
|
case BreakpointAtSysCall:
|
|
|
|
|
case BreakpointOnQmlSignalEmit:
|
|
|
|
|
case BreakpointAtJavaScriptThrow:
|
2011-10-20 10:45:59 +02:00
|
|
|
break;
|
2012-08-21 14:16:07 +02:00
|
|
|
case WatchpointAtExpression:
|
2011-10-20 10:45:59 +02:00
|
|
|
return !expression.isEmpty();
|
2013-04-09 10:59:36 +02:00
|
|
|
case UnknownBreakpointType:
|
|
|
|
|
case LastBreakpointType:
|
2011-10-20 10:45:59 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-24 16:41:30 +01:00
|
|
|
bool BreakpointParameters::isCppBreakpoint() const
|
|
|
|
|
{
|
|
|
|
|
// Qml specific breakpoint types.
|
|
|
|
|
if (type == BreakpointAtJavaScriptThrow
|
2012-04-16 12:00:23 +02:00
|
|
|
|| type == BreakpointOnQmlSignalEmit)
|
2012-01-24 16:41:30 +01:00
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// Qml is currently only file.
|
2015-09-23 13:14:30 +02:00
|
|
|
if (type == BreakpointByFileAndLine) {
|
|
|
|
|
auto qmlExtensionString = QString::fromLocal8Bit(qgetenv("QTC_QMLDEBUGGER_FILEEXTENSIONS"));
|
|
|
|
|
if (qmlExtensionString.isEmpty())
|
|
|
|
|
qmlExtensionString = QLatin1Literal(".qml;.js");
|
|
|
|
|
|
|
|
|
|
auto qmlFileExtensions = qmlExtensionString.split(QLatin1Literal(";"), QString::SkipEmptyParts);
|
|
|
|
|
foreach (QString extension, qmlFileExtensions) {
|
|
|
|
|
if (fileName.endsWith(extension, Qt::CaseInsensitive))
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-01-24 16:41:30 +01:00
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
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:
|
2012-04-16 12:00:23 +02:00
|
|
|
case BreakpointOnQmlSignalEmit:
|
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:
|
|
|
|
|
case BreakpointAtSysCall:
|
2011-10-10 15:25:18 +02:00
|
|
|
case BreakpointAtJavaScriptThrow:
|
2013-04-09 10:59:36 +02:00
|
|
|
case UnknownBreakpointType:
|
|
|
|
|
case LastBreakpointType:
|
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
|