2009-03-25 13:42:47 +01: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).
|
2009-03-25 13:42:47 +01:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2009-03-25 13:42:47 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** No Commercial Usage
|
2009-03-25 13:42:47 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** This file contains pre-release code and may not be distributed.
|
|
|
|
|
** You may use this file in accordance with the terms and conditions
|
|
|
|
|
** contained in the Technology Preview License Agreement accompanying
|
|
|
|
|
** this package.
|
2009-03-25 13:42:47 +01:00
|
|
|
**
|
|
|
|
|
** 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.
|
|
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please contact
|
|
|
|
|
** Nokia at qt-info@nokia.com.
|
2009-03-25 13:42:47 +01:00
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
|
|
#ifndef WATCHUTILS_H
|
|
|
|
|
#define WATCHUTILS_H
|
|
|
|
|
|
2010-04-26 18:54:08 +02:00
|
|
|
#include <QtCore/QSet>
|
2009-03-25 13:42:47 +01:00
|
|
|
#include <QtCore/QString>
|
2009-04-29 14:15:09 +02:00
|
|
|
#include <QtCore/QMap>
|
2009-03-25 13:42:47 +01:00
|
|
|
|
|
|
|
|
QT_BEGIN_NAMESPACE
|
2009-04-29 14:15:09 +02:00
|
|
|
class QDebug;
|
2009-03-25 13:42:47 +01:00
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
|
2009-05-14 14:29:37 +02:00
|
|
|
namespace TextEditor {
|
|
|
|
|
class ITextEditor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace Core {
|
|
|
|
|
class IEditor;
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-16 16:26:28 +02:00
|
|
|
namespace CPlusPlus {
|
|
|
|
|
class Snapshot;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-25 13:42:47 +01:00
|
|
|
namespace Debugger {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2009-04-29 14:15:09 +02:00
|
|
|
class WatchData;
|
2009-09-18 16:55:17 +02:00
|
|
|
class GdbMi;
|
2009-04-29 14:15:09 +02:00
|
|
|
|
2010-01-05 16:51:55 +01:00
|
|
|
QByteArray dotEscape(QByteArray str);
|
2009-03-25 13:42:47 +01:00
|
|
|
QString currentTime();
|
|
|
|
|
bool isSkippableFunction(const QString &funcName, const QString &fileName);
|
|
|
|
|
bool isLeavableFunction(const QString &funcName, const QString &fileName);
|
|
|
|
|
|
|
|
|
|
inline bool isNameChar(char c)
|
|
|
|
|
{
|
|
|
|
|
// could be 'stopped' or 'shlibs-added'
|
|
|
|
|
return (c >= 'a' && c <= 'z') || c == '-';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool hasLetterOrNumber(const QString &exp);
|
|
|
|
|
bool hasSideEffects(const QString &exp);
|
|
|
|
|
bool isKeyWord(const QString &exp);
|
2010-09-01 17:36:09 +02:00
|
|
|
bool isPointerType(const QByteArray &type);
|
|
|
|
|
bool isCharPointerType(const QByteArray &type);
|
2009-03-25 13:42:47 +01:00
|
|
|
bool startsWithDigit(const QString &str);
|
2010-09-01 17:36:09 +02:00
|
|
|
QByteArray stripPointerType(QByteArray type);
|
2010-09-01 18:01:08 +02:00
|
|
|
QByteArray gdbQuoteTypes(const QByteArray &type);
|
2009-03-25 13:42:47 +01:00
|
|
|
bool extractTemplate(const QString &type, QString *tmplate, QString *inner);
|
|
|
|
|
QString extractTypeFromPTypeOutput(const QString &str);
|
2010-09-22 17:30:22 +02:00
|
|
|
bool isFloatType(const QByteArray &type);
|
2010-09-01 17:36:09 +02:00
|
|
|
bool isIntOrFloatType(const QByteArray &type);
|
|
|
|
|
bool isIntType(const QByteArray &type);
|
|
|
|
|
bool isSymbianIntType(const QByteArray &type);
|
2009-07-14 11:21:52 +02:00
|
|
|
|
|
|
|
|
enum GuessChildrenResult { HasChildren, HasNoChildren, HasPossiblyChildren };
|
2010-09-01 17:36:09 +02:00
|
|
|
GuessChildrenResult guessChildren(const QByteArray &type);
|
2009-07-14 11:21:52 +02:00
|
|
|
|
2009-05-08 15:37:41 +02:00
|
|
|
QString quoteUnprintableLatin1(const QByteArray &ba);
|
2009-03-25 13:42:47 +01:00
|
|
|
|
2009-05-14 14:29:37 +02:00
|
|
|
// Editor tooltip support
|
|
|
|
|
bool isCppEditor(Core::IEditor *editor);
|
|
|
|
|
QString cppExpressionAt(TextEditor::ITextEditor *editor, int pos,
|
|
|
|
|
int *line, int *column, QString *function = 0);
|
2010-11-04 11:46:16 +01:00
|
|
|
// Editor helpers
|
|
|
|
|
TextEditor::ITextEditor *currentTextEditor();
|
|
|
|
|
bool currentTextEditorPosition(QString *fileNameIn = 0,
|
|
|
|
|
int *lineNumberIn = 0);
|
2009-05-14 14:29:37 +02:00
|
|
|
|
2009-04-29 14:15:09 +02:00
|
|
|
// Decode string data as returned by the dumper helpers.
|
|
|
|
|
QString decodeData(const QByteArray &baIn, int encoding);
|
|
|
|
|
|
2009-10-16 16:26:28 +02:00
|
|
|
// Get variables that are not initialized at a certain line
|
|
|
|
|
// of a function from the code model. Shadowed variables will
|
|
|
|
|
// be reported using the debugger naming conventions '<shadowed n>'
|
|
|
|
|
bool getUninitializedVariables(const CPlusPlus::Snapshot &snapshot,
|
|
|
|
|
const QString &function,
|
|
|
|
|
const QString &file,
|
|
|
|
|
int line,
|
|
|
|
|
QStringList *uninitializedVariables);
|
|
|
|
|
|
2009-04-29 14:15:09 +02:00
|
|
|
/* Attempt to put common code of the dumper handling into a helper
|
|
|
|
|
* class.
|
|
|
|
|
* "Custom dumper" is a library compiled against the current
|
|
|
|
|
* Qt containing functions to evaluate values of Qt classes
|
|
|
|
|
* (such as QString, taking pointers to their addresses).
|
|
|
|
|
* The library must be loaded into the debuggee.
|
|
|
|
|
* It provides a function that takes input from an input buffer
|
|
|
|
|
* and some parameters and writes output into an output buffer.
|
|
|
|
|
* Parameter 1 is the protocol:
|
|
|
|
|
* 1) Query. Fills output buffer with known types, Qt version and namespace.
|
|
|
|
|
* This information is parsed and stored by this class (special type
|
|
|
|
|
* enumeration).
|
|
|
|
|
* 2) Evaluate symbol, taking address and some additional parameters
|
|
|
|
|
* depending on type. */
|
|
|
|
|
|
2009-04-29 18:03:35 +02:00
|
|
|
class QtDumperHelper
|
|
|
|
|
{
|
2009-04-29 14:15:09 +02:00
|
|
|
public:
|
|
|
|
|
enum Type {
|
|
|
|
|
UnknownType,
|
|
|
|
|
SupportedType, // A type that requires no special handling by the dumper
|
|
|
|
|
// Below types require special handling
|
2009-06-03 12:46:55 +02:00
|
|
|
QAbstractItemType,
|
2009-04-29 14:15:09 +02:00
|
|
|
QObjectType, QWidgetType, QObjectSlotType, QObjectSignalType,
|
2009-08-28 09:44:11 +02:00
|
|
|
QVectorType, QMapType, QMultiMapType, QMapNodeType, QStackType,
|
2009-04-29 18:03:35 +02:00
|
|
|
StdVectorType, StdDequeType, StdSetType, StdMapType, StdStackType,
|
2009-04-29 14:15:09 +02:00
|
|
|
StdStringType
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Type/Parameter struct required for building a value query
|
|
|
|
|
struct TypeData {
|
|
|
|
|
TypeData();
|
|
|
|
|
void clear();
|
|
|
|
|
|
|
|
|
|
Type type;
|
|
|
|
|
bool isTemplate;
|
2010-09-01 17:36:09 +02:00
|
|
|
QByteArray tmplate;
|
|
|
|
|
QByteArray inner;
|
2009-04-29 14:15:09 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
QtDumperHelper();
|
|
|
|
|
void clear();
|
|
|
|
|
|
2009-07-07 16:00:45 +02:00
|
|
|
double dumperVersion() const { return m_dumperVersion; }
|
|
|
|
|
|
2009-04-29 14:15:09 +02:00
|
|
|
int typeCount() const;
|
|
|
|
|
// Look up a simple, non-template type
|
2010-09-01 17:36:09 +02:00
|
|
|
Type simpleType(const QByteArray &simpleType) const;
|
2009-04-29 14:15:09 +02:00
|
|
|
// Look up a (potentially) template type and fill parameter struct
|
2010-09-01 17:36:09 +02:00
|
|
|
TypeData typeData(const QByteArray &typeName) const;
|
|
|
|
|
Type type(const QByteArray &typeName) const;
|
2009-04-29 14:15:09 +02:00
|
|
|
|
|
|
|
|
int qtVersion() const;
|
2010-09-01 17:36:09 +02:00
|
|
|
QByteArray qtVersionString() const;
|
|
|
|
|
QByteArray qtNamespace() const;
|
2011-01-17 15:11:11 +01:00
|
|
|
void setQtNamespace(const QByteArray &ba)
|
|
|
|
|
{ if (!ba.isEmpty()) m_qtNamespace = ba; }
|
2009-04-29 14:15:09 +02:00
|
|
|
|
|
|
|
|
// Complete parse of "query" (protocol 1) response from debuggee buffer.
|
|
|
|
|
// 'data' excludes the leading indicator character.
|
2010-08-27 17:24:13 +02:00
|
|
|
bool parseQuery(const GdbMi &data);
|
2009-09-18 16:55:17 +02:00
|
|
|
// Sizes can be added as the debugger determines them
|
2010-09-01 17:36:09 +02:00
|
|
|
void addSize(const QByteArray &type, int size);
|
2009-04-29 14:15:09 +02:00
|
|
|
|
|
|
|
|
// Determine the parameters required for an "evaluate" (protocol 2) call
|
|
|
|
|
void evaluationParameters(const WatchData &data,
|
|
|
|
|
const TypeData &td,
|
|
|
|
|
QByteArray *inBuffer,
|
2010-09-01 17:36:09 +02:00
|
|
|
QList<QByteArray> *extraParameters) const;
|
2009-04-29 14:15:09 +02:00
|
|
|
|
|
|
|
|
QString toString(bool debug = false) const;
|
|
|
|
|
|
2009-07-07 16:00:45 +02:00
|
|
|
static QString msgDumperOutdated(double requiredVersion, double currentVersion);
|
2009-04-29 16:52:14 +02:00
|
|
|
|
2009-07-07 16:00:45 +02:00
|
|
|
private:
|
2009-04-29 14:15:09 +02:00
|
|
|
typedef QMap<QString, Type> NameTypeMap;
|
2010-09-01 17:36:09 +02:00
|
|
|
typedef QMap<QByteArray, int> SizeCache;
|
2009-04-29 14:15:09 +02:00
|
|
|
|
|
|
|
|
// Look up a simple (namespace) type
|
2011-01-10 10:14:23 +01:00
|
|
|
QByteArray evaluationSizeofTypeExpression(const QByteArray &typeName) const;
|
2009-04-29 16:52:14 +02:00
|
|
|
|
2009-04-29 14:15:09 +02:00
|
|
|
NameTypeMap m_nameTypeMap;
|
2009-04-29 16:52:14 +02:00
|
|
|
SizeCache m_sizeCache;
|
2009-07-02 16:38:15 +02:00
|
|
|
|
|
|
|
|
// The initial dumper query function returns sizes of some special
|
|
|
|
|
// types to aid CDB since it cannot determine the size of classes.
|
|
|
|
|
// They are not complete (std::allocator<X>).
|
|
|
|
|
enum SpecialSizeType { IntSize, PointerSize, StdAllocatorSize,
|
|
|
|
|
QSharedPointerSize, QSharedDataPointerSize,
|
2009-09-21 14:55:39 +02:00
|
|
|
QWeakPointerSize, QPointerSize,
|
|
|
|
|
QListSize, QLinkedListSize, QVectorSize, QQueueSize,
|
|
|
|
|
SpecialSizeCount };
|
2009-07-02 16:38:15 +02:00
|
|
|
|
|
|
|
|
// Resolve name to enumeration or SpecialSizeCount (invalid)
|
2010-09-01 17:36:09 +02:00
|
|
|
SpecialSizeType specialSizeType(const QByteArray &type) const;
|
2009-07-02 16:38:15 +02:00
|
|
|
|
|
|
|
|
int m_specialSizes[SpecialSizeCount];
|
|
|
|
|
|
2010-09-01 17:36:09 +02:00
|
|
|
typedef QMap<QByteArray, QByteArray> ExpressionCache;
|
|
|
|
|
ExpressionCache m_expressionCache;
|
2009-04-29 14:15:09 +02:00
|
|
|
int m_qtVersion;
|
2009-07-07 16:00:45 +02:00
|
|
|
double m_dumperVersion;
|
2010-09-01 17:36:09 +02:00
|
|
|
QByteArray m_qtNamespace;
|
|
|
|
|
|
|
|
|
|
void setQClassPrefixes(const QByteArray &qNamespace);
|
|
|
|
|
|
|
|
|
|
QByteArray m_qPointerPrefix;
|
|
|
|
|
QByteArray m_qSharedPointerPrefix;
|
|
|
|
|
QByteArray m_qSharedDataPointerPrefix;
|
|
|
|
|
QByteArray m_qWeakPointerPrefix;
|
|
|
|
|
QByteArray m_qListPrefix;
|
|
|
|
|
QByteArray m_qLinkedListPrefix;
|
|
|
|
|
QByteArray m_qVectorPrefix;
|
|
|
|
|
QByteArray m_qQueuePrefix;
|
2009-04-29 14:15:09 +02:00
|
|
|
};
|
2009-03-25 13:42:47 +01:00
|
|
|
|
2009-04-29 16:52:14 +02:00
|
|
|
QDebug operator<<(QDebug in, const QtDumperHelper::TypeData &d);
|
2009-04-29 18:03:35 +02:00
|
|
|
|
2009-06-23 10:24:25 +02:00
|
|
|
// remove the default template argument in std:: containers
|
|
|
|
|
QString removeDefaultTemplateArguments(QString type);
|
|
|
|
|
|
2010-04-26 18:54:08 +02:00
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// GdbMi interaction
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
void setWatchDataValue(WatchData &data, const GdbMi &item);
|
|
|
|
|
void setWatchDataValueToolTip(WatchData &data, const GdbMi &mi,
|
|
|
|
|
int encoding);
|
|
|
|
|
void setWatchDataChildCount(WatchData &data, const GdbMi &mi);
|
|
|
|
|
void setWatchDataValueEnabled(WatchData &data, const GdbMi &mi);
|
|
|
|
|
void setWatchDataValueEditable(WatchData &data, const GdbMi &mi);
|
|
|
|
|
void setWatchDataExpression(WatchData &data, const GdbMi &mi);
|
|
|
|
|
void setWatchDataAddress(WatchData &data, const GdbMi &mi);
|
|
|
|
|
void setWatchDataAddressHelper(WatchData &data, const QByteArray &addr);
|
|
|
|
|
void setWatchDataType(WatchData &data, const GdbMi &mi);
|
|
|
|
|
void setWatchDataDisplayedType(WatchData &data, const GdbMi &mi);
|
|
|
|
|
|
|
|
|
|
void parseWatchData(const QSet<QByteArray> &expandedINames,
|
2010-10-08 14:55:57 +02:00
|
|
|
const WatchData &parent, const GdbMi &child,
|
2010-04-26 18:54:08 +02:00
|
|
|
QList<WatchData> *insertions);
|
|
|
|
|
|
2009-03-25 13:42:47 +01:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|
|
|
|
|
|
|
|
|
|
#endif // WATCHUTILS_H
|