| 
									
										
										
										
											2009-03-25 13:42:47 +01:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Contact:  Qt Software Information (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 qt-sales@nokia.com. | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "watchutils.h"
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | #include "watchhandler.h"
 | 
					
						
							|  |  |  | #include <utils/qtcassert.h>
 | 
					
						
							| 
									
										
										
										
											2009-03-25 13:42:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <QtCore/QDebug>
 | 
					
						
							|  |  |  | #include <QtCore/QTime>
 | 
					
						
							|  |  |  | #include <QtCore/QStringList>
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | #include <QtCore/QCoreApplication>
 | 
					
						
							|  |  |  | #include <QtCore/QTextStream>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <ctype.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum { debug = 0 }; | 
					
						
							| 
									
										
										
										
											2009-03-25 13:42:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Debugger { | 
					
						
							|  |  |  | namespace Internal { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString dotEscape(QString str) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-03-26 08:42:27 +01:00
										 |  |  |     const QChar dot = QLatin1Char('.'); | 
					
						
							| 
									
										
										
										
											2009-03-25 13:42:47 +01:00
										 |  |  |     str.replace(QLatin1Char(' '), dot); | 
					
						
							|  |  |  |     str.replace(QLatin1Char('\\'), dot); | 
					
						
							|  |  |  |     str.replace(QLatin1Char('/'), dot); | 
					
						
							|  |  |  |     return str; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString currentTime() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return QTime::currentTime().toString(QLatin1String("hh:mm:ss.zzz")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isSkippableFunction(const QString &funcName, const QString &fileName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/qobject.cpp"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/moc_qobject.cpp"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/qmetaobject.cpp"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String(".moc"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (funcName.endsWith("::qt_metacall")) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isLeavableFunction(const QString &funcName, const QString &fileName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (funcName.endsWith(QLatin1String("QObjectPrivate::setCurrentSender"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/qmetaobject.cpp")) | 
					
						
							|  |  |  |             && funcName.endsWith(QLatin1String("QMetaObject::methodOffset"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/qobject.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/qobject.cpp")) | 
					
						
							|  |  |  |             && funcName.endsWith(QLatin1String("QObjectConnectionListVector::at"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("kernel/qobject.cpp")) | 
					
						
							|  |  |  |             && funcName.endsWith(QLatin1String("~QObject"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("thread/qmutex.cpp"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("thread/qthread.cpp"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("thread/qthread_unix.cpp"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("thread/qmutex.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.contains(QLatin1String("thread/qbasicatomic"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.contains(QLatin1String("thread/qorderedmutexlocker_p"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.contains(QLatin1String("arch/qatomic"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("tools/qvector.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("tools/qlist.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("tools/qhash.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("tools/qmap.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("tools/qstring.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     if (fileName.endsWith(QLatin1String("global/qglobal.h"))) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool hasLetterOrNumber(const QString &exp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QChar underscore = QLatin1Char('_'); | 
					
						
							|  |  |  |     for (int i = exp.size(); --i >= 0; ) | 
					
						
							|  |  |  |         if (exp.at(i).isLetterOrNumber() || exp.at(i) == underscore) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool hasSideEffects(const QString &exp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // FIXME: complete?
 | 
					
						
							|  |  |  |     return exp.contains(QLatin1String("-=")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("+=")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("/=")) | 
					
						
							| 
									
										
										
										
											2009-05-05 20:33:31 +02:00
										 |  |  |         || exp.contains(QLatin1String("%=")) | 
					
						
							| 
									
										
										
										
											2009-03-25 13:42:47 +01:00
										 |  |  |         || exp.contains(QLatin1String("*=")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("&=")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("|=")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("^=")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("--")) | 
					
						
							|  |  |  |         || exp.contains(QLatin1String("++")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isKeyWord(const QString &exp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // FIXME: incomplete
 | 
					
						
							|  |  |  |     return exp == QLatin1String("class") | 
					
						
							|  |  |  |         || exp == QLatin1String("const") | 
					
						
							|  |  |  |         || exp == QLatin1String("do") | 
					
						
							|  |  |  |         || exp == QLatin1String("if") | 
					
						
							|  |  |  |         || exp == QLatin1String("return") | 
					
						
							|  |  |  |         || exp == QLatin1String("struct") | 
					
						
							|  |  |  |         || exp == QLatin1String("template") | 
					
						
							|  |  |  |         || exp == QLatin1String("void") | 
					
						
							|  |  |  |         || exp == QLatin1String("volatile") | 
					
						
							|  |  |  |         || exp == QLatin1String("while"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isPointerType(const QString &type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return type.endsWith(QLatin1Char('*')) || type.endsWith(QLatin1String("* const")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isAccessSpecifier(const QString &str) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  |     static const QStringList items = QStringList() | 
					
						
							|  |  |  |         << QLatin1String("private") | 
					
						
							|  |  |  |         << QLatin1String("protected") | 
					
						
							|  |  |  |         << QLatin1String("public"); | 
					
						
							| 
									
										
										
										
											2009-03-25 13:42:47 +01:00
										 |  |  |     return items.contains(str); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool startsWithDigit(const QString &str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return !str.isEmpty() && str.at(0).isDigit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString stripPointerType(QString type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (type.endsWith(QLatin1Char('*'))) | 
					
						
							|  |  |  |         type.chop(1); | 
					
						
							|  |  |  |     if (type.endsWith(QLatin1String("* const"))) | 
					
						
							|  |  |  |         type.chop(7); | 
					
						
							|  |  |  |     if (type.endsWith(QLatin1Char(' '))) | 
					
						
							|  |  |  |         type.chop(1); | 
					
						
							|  |  |  |     return type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString gdbQuoteTypes(const QString &type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // gdb does not understand sizeof(Core::IFile*).
 | 
					
						
							|  |  |  |     // "sizeof('Core::IFile*')" is also not acceptable,
 | 
					
						
							|  |  |  |     // it needs to be "sizeof('Core::IFile'*)"
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     // We never will have a perfect solution here (even if we had a full blown
 | 
					
						
							|  |  |  |     // C++ parser as we do not have information on what is a type and what is
 | 
					
						
							|  |  |  |     // a variable name. So "a<b>::c" could either be two comparisons of values
 | 
					
						
							|  |  |  |     // 'a', 'b' and '::c', or a nested type 'c' in a template 'a<b>'. We
 | 
					
						
							|  |  |  |     // assume here it is the latter.
 | 
					
						
							|  |  |  |     //return type;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // (*('myns::QPointer<myns::QObject>*'*)0x684060)" is not acceptable
 | 
					
						
							|  |  |  |     // (*('myns::QPointer<myns::QObject>'**)0x684060)" is acceptable
 | 
					
						
							|  |  |  |     if (isPointerType(type)) | 
					
						
							|  |  |  |         return gdbQuoteTypes(stripPointerType(type)) + QLatin1Char('*'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QString accu; | 
					
						
							|  |  |  |     QString result; | 
					
						
							|  |  |  |     int templateLevel = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QChar colon = QLatin1Char(':'); | 
					
						
							|  |  |  |     const QChar singleQuote = QLatin1Char('\''); | 
					
						
							|  |  |  |     const QChar lessThan = QLatin1Char('<'); | 
					
						
							|  |  |  |     const QChar greaterThan = QLatin1Char('>'); | 
					
						
							|  |  |  |     for (int i = 0; i != type.size(); ++i) { | 
					
						
							|  |  |  |         const QChar c = type.at(i); | 
					
						
							|  |  |  |         if (c.isLetterOrNumber() || c == QLatin1Char('_') || c == colon || c == QLatin1Char(' ')) { | 
					
						
							|  |  |  |             accu += c; | 
					
						
							|  |  |  |         } else if (c == lessThan) { | 
					
						
							|  |  |  |             ++templateLevel; | 
					
						
							|  |  |  |             accu += c; | 
					
						
							|  |  |  |         } else if (c == greaterThan) { | 
					
						
							|  |  |  |             --templateLevel; | 
					
						
							|  |  |  |             accu += c; | 
					
						
							|  |  |  |         } else if (templateLevel > 0) { | 
					
						
							|  |  |  |             accu += c; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             if (accu.contains(colon) || accu.contains(lessThan)) | 
					
						
							|  |  |  |                 result += singleQuote + accu + singleQuote; | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 result += accu; | 
					
						
							|  |  |  |             accu.clear(); | 
					
						
							|  |  |  |             result += c; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (accu.contains(colon) || accu.contains(lessThan)) | 
					
						
							|  |  |  |         result += singleQuote + accu + singleQuote; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         result += accu; | 
					
						
							|  |  |  |     //qDebug() << "GDB_QUOTING" << type << " TO " << result;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool extractTemplate(const QString &type, QString *tmplate, QString *inner) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Input "Template<Inner1,Inner2,...>::Foo" will return "Template::Foo" in
 | 
					
						
							|  |  |  |     // 'tmplate' and "Inner1@Inner2@..." etc in 'inner'. Result indicates
 | 
					
						
							|  |  |  |     // whether parsing was successful
 | 
					
						
							|  |  |  |     int level = 0; | 
					
						
							|  |  |  |     bool skipSpace = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 0; i != type.size(); ++i) { | 
					
						
							|  |  |  |         const QChar c = type.at(i); | 
					
						
							|  |  |  |         if (c == QLatin1Char(' ') && skipSpace) { | 
					
						
							|  |  |  |             skipSpace = false; | 
					
						
							|  |  |  |         } else if (c == QLatin1Char('<')) { | 
					
						
							|  |  |  |             *(level == 0 ? tmplate : inner) += c; | 
					
						
							|  |  |  |             ++level; | 
					
						
							|  |  |  |         } else if (c == QLatin1Char('>')) { | 
					
						
							|  |  |  |             --level; | 
					
						
							|  |  |  |             *(level == 0 ? tmplate : inner) += c; | 
					
						
							|  |  |  |         } else if (c == QLatin1Char(',')) { | 
					
						
							|  |  |  |             *inner += (level == 1) ? QLatin1Char('@') : QLatin1Char(','); | 
					
						
							|  |  |  |             skipSpace = true; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             *(level == 0 ? tmplate : inner) += c; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     *tmplate = tmplate->trimmed(); | 
					
						
							|  |  |  |     *tmplate = tmplate->remove(QLatin1String("<>")); | 
					
						
							|  |  |  |     *inner = inner->trimmed(); | 
					
						
							|  |  |  |     //qDebug() << "EXTRACT TEMPLATE: " << *tmplate << *inner << " FROM " << type;
 | 
					
						
							|  |  |  |     return !inner->isEmpty(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString extractTypeFromPTypeOutput(const QString &str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int pos0 = str.indexOf(QLatin1Char('=')); | 
					
						
							|  |  |  |     int pos1 = str.indexOf(QLatin1Char('{')); | 
					
						
							|  |  |  |     int pos2 = str.lastIndexOf(QLatin1Char('}')); | 
					
						
							|  |  |  |     QString res = str; | 
					
						
							|  |  |  |     if (pos0 != -1 && pos1 != -1 && pos2 != -1) | 
					
						
							|  |  |  |         res = str.mid(pos0 + 2, pos1 - 1 - pos0) | 
					
						
							|  |  |  |             + QLatin1String(" ... ") + str.right(str.size() - pos2); | 
					
						
							|  |  |  |     return res.simplified(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isIntOrFloatType(const QString &type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     static const QStringList types = QStringList() | 
					
						
							|  |  |  |         << QLatin1String("char") << QLatin1String("int") << QLatin1String("short") | 
					
						
							|  |  |  |         << QLatin1String("float") << QLatin1String("double") << QLatin1String("long") | 
					
						
							|  |  |  |         << QLatin1String("bool") << QLatin1String("signed char") << QLatin1String("unsigned") | 
					
						
							|  |  |  |         << QLatin1String("unsigned char") | 
					
						
							|  |  |  |         << QLatin1String("unsigned int") << QLatin1String("unsigned long") | 
					
						
							|  |  |  |         << QLatin1String("long long")  << QLatin1String("unsigned long long"); | 
					
						
							|  |  |  |     return types.contains(type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString sizeofTypeExpression(const QString &type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (type.endsWith(QLatin1Char('*'))) | 
					
						
							|  |  |  |         return QLatin1String("sizeof(void*)"); | 
					
						
							|  |  |  |     if (type.endsWith(QLatin1Char('>'))) | 
					
						
							|  |  |  |         return QLatin1String("sizeof(") + type + QLatin1Char(')'); | 
					
						
							|  |  |  |     return QLatin1String("sizeof(") + gdbQuoteTypes(type) + QLatin1Char(')'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | // Utilities to decode string data returned by the dumper helpers.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-08 15:37:41 +02:00
										 |  |  | QString quoteUnprintableLatin1(const QByteArray &ba) | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     QString res; | 
					
						
							|  |  |  |     char buf[10]; | 
					
						
							|  |  |  |     for (int i = 0, n = ba.size(); i != n; ++i) { | 
					
						
							|  |  |  |         const unsigned char c = ba.at(i); | 
					
						
							|  |  |  |         if (isprint(c)) { | 
					
						
							|  |  |  |             res += c; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             qsnprintf(buf, sizeof(buf) - 1, "\\%x", int(c)); | 
					
						
							|  |  |  |             res += buf; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  | QString decodeData(const QByteArray &ba, int encoding) | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (encoding) { | 
					
						
							|  |  |  |         case 0: // unencoded 8 bit data
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  |             return quoteUnprintableLatin1(ba); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         case 1: { //  base64 encoded 8 bit data, used for QByteArray
 | 
					
						
							|  |  |  |             const QChar doubleQuote(QLatin1Char('"')); | 
					
						
							|  |  |  |             QString rc = doubleQuote; | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  |             rc += quoteUnprintableLatin1(QByteArray::fromBase64(ba)); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             rc += doubleQuote; | 
					
						
							|  |  |  |             return rc; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case 2: { //  base64 encoded 16 bit data, used for QString
 | 
					
						
							|  |  |  |             const QChar doubleQuote(QLatin1Char('"')); | 
					
						
							| 
									
										
										
										
											2009-05-04 12:12:45 +02:00
										 |  |  |             const QByteArray decodedBa = QByteArray::fromBase64(ba); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             QString rc = doubleQuote; | 
					
						
							| 
									
										
										
										
											2009-05-04 12:12:45 +02:00
										 |  |  |             rc += QString::fromUtf16(reinterpret_cast<const ushort *>(decodedBa.data()), decodedBa.size() / 2); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             rc += doubleQuote; | 
					
						
							|  |  |  |             return rc; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case 3: { //  base64 encoded 32 bit data
 | 
					
						
							| 
									
										
										
										
											2009-05-04 12:12:45 +02:00
										 |  |  |             const QByteArray decodedBa = QByteArray::fromBase64(ba); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             const QChar doubleQuote(QLatin1Char('"')); | 
					
						
							|  |  |  |             QString rc = doubleQuote; | 
					
						
							| 
									
										
										
										
											2009-05-04 12:12:45 +02:00
										 |  |  |             rc += QString::fromUcs4(reinterpret_cast<const uint *>(decodedBa.data()), decodedBa.size() / 4); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             rc += doubleQuote; | 
					
						
							|  |  |  |             return rc; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case 4: { //  base64 encoded 16 bit data, without quotes (see 2)
 | 
					
						
							| 
									
										
										
										
											2009-05-04 12:12:45 +02:00
										 |  |  |             const QByteArray decodedBa = QByteArray::fromBase64(ba); | 
					
						
							|  |  |  |             return QString::fromUtf16(reinterpret_cast<const ushort *>(decodedBa.data()), decodedBa.size() / 2); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return QCoreApplication::translate("Debugger", "<Encoding error>"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // --------------- QtDumperResult
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QtDumperResult::Child::Child() : | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |    valueEncoded(0), | 
					
						
							|  |  |  |    childCount(0), | 
					
						
							|  |  |  |    valuedisabled(false) | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QtDumperResult::QtDumperResult() : | 
					
						
							|  |  |  |     valueEncoded(0), | 
					
						
							|  |  |  |     valuedisabled(false), | 
					
						
							|  |  |  |     childCount(0), | 
					
						
							|  |  |  |     internal(false) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void QtDumperResult::clear() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     iname.clear(); | 
					
						
							|  |  |  |     value.clear(); | 
					
						
							|  |  |  |     address.clear(); | 
					
						
							|  |  |  |     type.clear(); | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     displayedType.clear(); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     valueEncoded = 0; | 
					
						
							|  |  |  |     valuedisabled = false; | 
					
						
							|  |  |  |     childCount = 0; | 
					
						
							|  |  |  |     internal = false; | 
					
						
							|  |  |  |     childType.clear(); | 
					
						
							|  |  |  |     children.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QList<WatchData> QtDumperResult::toWatchData(int source) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QList<WatchData> rc; | 
					
						
							|  |  |  |     rc.push_back(WatchData()); | 
					
						
							|  |  |  |     WatchData &root = rc.front(); | 
					
						
							|  |  |  |     root.iname = iname; | 
					
						
							|  |  |  |     const QChar dot = QLatin1Char('.'); | 
					
						
							|  |  |  |     const int lastDotIndex = root.iname.lastIndexOf(dot); | 
					
						
							|  |  |  |     root.exp = root.name = lastDotIndex == -1 ? iname : iname.mid(lastDotIndex + 1); | 
					
						
							|  |  |  |     root.setValue(decodeData(value, valueEncoded)); | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     root.setType(displayedType.isEmpty() ? type : displayedType); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     root.valuedisabled = valuedisabled; | 
					
						
							|  |  |  |     root.setAddress(address); | 
					
						
							|  |  |  |     root.source = source; | 
					
						
							|  |  |  |     root.setChildCount(childCount); | 
					
						
							|  |  |  |     // Children
 | 
					
						
							|  |  |  |     if (childCount > 0) { | 
					
						
							|  |  |  |         if (children.size() == childCount) { | 
					
						
							|  |  |  |             for (int c = 0; c < childCount; c++) { | 
					
						
							|  |  |  |                 const Child &dchild = children.at(c); | 
					
						
							|  |  |  |                 rc.push_back(WatchData()); | 
					
						
							|  |  |  |                 WatchData &wchild = rc.back(); | 
					
						
							|  |  |  |                 wchild.source = source; | 
					
						
							|  |  |  |                 wchild.iname = iname; | 
					
						
							|  |  |  |                 wchild.iname += dot; | 
					
						
							|  |  |  |                 wchild.iname += dchild.name;                 | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |                 wchild.name = dchild.name; | 
					
						
							|  |  |  |                 wchild.exp = dchild.exp; | 
					
						
							|  |  |  |                 wchild.valuedisabled = dchild.valuedisabled; | 
					
						
							|  |  |  |                 wchild.setType(dchild.type.isEmpty() ? childType : dchild.type); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                 wchild.setAddress(dchild.address); | 
					
						
							|  |  |  |                 wchild.setValue(decodeData(dchild.value, dchild.valueEncoded)); | 
					
						
							|  |  |  |                 wchild.setChildCount(0); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             root.setChildrenUnneeded(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             root.setChildrenNeeded(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QDebug operator<<(QDebug in, const QtDumperResult &d) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QDebug nospace = in.nospace(); | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     nospace << " iname=" << d.iname << " type=" << d.type << " displayed=" << d.displayedType | 
					
						
							|  |  |  |             << " address=" << d.address | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             << " value="  << d.value | 
					
						
							|  |  |  |             << " disabled=" << d.valuedisabled | 
					
						
							|  |  |  |             << " encoded=" << d.valueEncoded << " internal=" << d.internal; | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     const int realChildCount = d.children.size(); | 
					
						
							|  |  |  |     if (d.childCount || realChildCount) { | 
					
						
							|  |  |  |         nospace << " childCount=" << d.childCount << '/' << realChildCount | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                 << " childType=" << d.childType << '\n'; | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |         for (int i = 0; i < realChildCount; i++) { | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             const QtDumperResult::Child &c = d.children.at(i); | 
					
						
							|  |  |  |             nospace << "   #" << i << " addr=" << c.address | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |                     << " disabled=" << c.valuedisabled | 
					
						
							| 
									
										
										
										
											2009-05-06 09:22:05 +02:00
										 |  |  |                     << " type=" << c.type << " exp=" << c.exp | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                     << " name=" << c.name << " encoded=" << c.valueEncoded | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |                     << " value=" << c.value | 
					
						
							|  |  |  |                     << "childcount=" << c.childCount << '\n'; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return in; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ----------------- QtDumperHelper::TypeData
 | 
					
						
							|  |  |  | QtDumperHelper::TypeData::TypeData() : | 
					
						
							|  |  |  |     type(UnknownType), | 
					
						
							|  |  |  |     isTemplate(false) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void QtDumperHelper::TypeData::clear() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     isTemplate = false; | 
					
						
							|  |  |  |     type = UnknownType; | 
					
						
							|  |  |  |     tmplate.clear(); | 
					
						
							|  |  |  |     inner.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ----------------- QtDumperHelper
 | 
					
						
							|  |  |  | QtDumperHelper::QtDumperHelper() : | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     m_stdAllocatorPrefix(QLatin1String("std::allocator")), | 
					
						
							|  |  |  |     m_intSize(0), | 
					
						
							|  |  |  |     m_pointerSize(0), | 
					
						
							|  |  |  |     m_stdAllocatorSize(0), | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     m_qtVersion(0) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void QtDumperHelper::clear() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_nameTypeMap.clear(); | 
					
						
							|  |  |  |     m_qtVersion = 0; | 
					
						
							|  |  |  |     m_qtNamespace.clear(); | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     m_sizeCache.clear(); | 
					
						
							|  |  |  |     m_intSize = 0; | 
					
						
							|  |  |  |     m_pointerSize = 0; | 
					
						
							|  |  |  |     m_stdAllocatorSize = 0; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void formatQtVersion(int v, QTextStream &str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     str  << ((v >> 16) & 0xFF) << '.' << ((v >> 8) & 0xFF) << '.' << (v & 0xFF); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString QtDumperHelper::toString(bool debug) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug)  { | 
					
						
							|  |  |  |         QString rc; | 
					
						
							|  |  |  |         QTextStream str(&rc); | 
					
						
							|  |  |  |         str << "version="; | 
					
						
							|  |  |  |         formatQtVersion(m_qtVersion, str); | 
					
						
							|  |  |  |         str << " namespace='" << m_qtNamespace << "'," << m_nameTypeMap.size() << " known types: "; | 
					
						
							|  |  |  |         const NameTypeMap::const_iterator cend = m_nameTypeMap.constEnd(); | 
					
						
							|  |  |  |         for (NameTypeMap::const_iterator it = m_nameTypeMap.constBegin(); it != cend; ++it) { | 
					
						
							|  |  |  |             str <<",[" << it.key() << ',' << it.value() << ']'; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |         str << "Sizes: intsize=" << m_intSize << " pointer size=" << m_pointerSize | 
					
						
							|  |  |  |                 << " allocatorsize=" << m_stdAllocatorSize; | 
					
						
							|  |  |  |         const SizeCache::const_iterator scend = m_sizeCache.constEnd(); | 
					
						
							|  |  |  |         for (SizeCache::const_iterator it = m_sizeCache.constBegin(); it != scend; ++it) { | 
					
						
							|  |  |  |             str << ' ' << it.key() << '=' << it.value(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         return rc; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const QString nameSpace = m_qtNamespace.isEmpty() ? QCoreApplication::translate("QtDumperHelper", "<none>") : m_qtNamespace; | 
					
						
							|  |  |  |     return QCoreApplication::translate("QtDumperHelper", | 
					
						
							|  |  |  |                                        "%n known types, Qt version: %1, Qt namespace: %2", | 
					
						
							|  |  |  |                                        0, QCoreApplication::CodecForTr, | 
					
						
							|  |  |  |                                        m_nameTypeMap.size()).arg(qtVersionString(), nameSpace); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QtDumperHelper::Type QtDumperHelper::simpleType(const QString &simpleType) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_nameTypeMap.value(simpleType, UnknownType); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int QtDumperHelper::qtVersion() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_qtVersion; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString QtDumperHelper::qtNamespace() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_qtNamespace; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void QtDumperHelper::setQtNamespace(const QString &qtNamespace) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_qtNamespace = qtNamespace; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int QtDumperHelper::typeCount() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_nameTypeMap.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Look up unnamespaced 'std' types.
 | 
					
						
							|  |  |  | static inline QtDumperHelper::Type stdType(const QString &s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (s == QLatin1String("vector")) | 
					
						
							|  |  |  |         return QtDumperHelper::StdVectorType; | 
					
						
							|  |  |  |     if (s == QLatin1String("deque")) | 
					
						
							|  |  |  |         return QtDumperHelper::StdDequeType; | 
					
						
							|  |  |  |     if (s == QLatin1String("set")) | 
					
						
							|  |  |  |         return QtDumperHelper::StdSetType; | 
					
						
							|  |  |  |     if (s == QLatin1String("stack")) | 
					
						
							|  |  |  |         return QtDumperHelper::StdStackType; | 
					
						
							|  |  |  |     if (s == QLatin1String("map")) | 
					
						
							|  |  |  |         return QtDumperHelper::StdMapType; | 
					
						
							|  |  |  |     if (s == QLatin1String("basic_string")) | 
					
						
							|  |  |  |         return QtDumperHelper::StdStringType; | 
					
						
							|  |  |  |     return QtDumperHelper::UnknownType; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QtDumperHelper::Type QtDumperHelper::specialType(QString s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Std classes.
 | 
					
						
							|  |  |  |     if (s.startsWith(QLatin1String("std::"))) | 
					
						
							|  |  |  |         return stdType(s.mid(5)); | 
					
						
							|  |  |  |     // Strip namespace
 | 
					
						
							|  |  |  |     const int namespaceIndex = s.lastIndexOf(QLatin1String("::")); | 
					
						
							|  |  |  |     if (namespaceIndex == -1) { | 
					
						
							|  |  |  |         // None ... check for std..
 | 
					
						
							|  |  |  |         const Type sType = stdType(s); | 
					
						
							|  |  |  |         if (sType != UnknownType) | 
					
						
							|  |  |  |             return sType; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         s.remove(namespaceIndex + 2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (s == QLatin1String("QObject")) | 
					
						
							|  |  |  |         return QObjectType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QWidget")) | 
					
						
							|  |  |  |         return QWidgetType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QObjectSlot")) | 
					
						
							|  |  |  |         return QObjectSlotType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QObjectSignal")) | 
					
						
							|  |  |  |         return QObjectSignalType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QVector")) | 
					
						
							|  |  |  |         return QVectorType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QMap")) | 
					
						
							|  |  |  |         return QMapType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QMultiMap")) | 
					
						
							|  |  |  |         return QMultiMapType; | 
					
						
							|  |  |  |     if (s == QLatin1String("QMapNode")) | 
					
						
							|  |  |  |         return QMapNodeType; | 
					
						
							|  |  |  |     return UnknownType; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool QtDumperHelper::needsExpressionSyntax(Type t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (t) { | 
					
						
							|  |  |  |         case QObjectSlotType: | 
					
						
							|  |  |  |         case QObjectSignalType: | 
					
						
							|  |  |  |         case QMapType: | 
					
						
							|  |  |  |         case QVectorType: | 
					
						
							|  |  |  |         case QMultiMapType: | 
					
						
							|  |  |  |         case QMapNodeType: | 
					
						
							|  |  |  |         case StdMapType: | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString QtDumperHelper::qtVersionString() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QString rc; | 
					
						
							|  |  |  |     QTextStream str(&rc); | 
					
						
							|  |  |  |     formatQtVersion(m_qtVersion, str); | 
					
						
							|  |  |  |     return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void QtDumperHelper::setQtVersion(int v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_qtVersion = v; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void QtDumperHelper::setQtVersion(const QString &v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_qtVersion = 0; | 
					
						
							|  |  |  |     const QStringList vl = v.split(QLatin1Char('.')); | 
					
						
							|  |  |  |     if (vl.size() == 3) { | 
					
						
							|  |  |  |         const int major = vl.at(0).toInt(); | 
					
						
							|  |  |  |         const int minor = vl.at(1).toInt(); | 
					
						
							|  |  |  |         const int patch = vl.at(2).toInt(); | 
					
						
							|  |  |  |         m_qtVersion = (major << 16) | (minor << 8) | patch; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse a list of types.
 | 
					
						
							|  |  |  | void QtDumperHelper::parseQueryTypes(const QStringList &l, Debugger debugger) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_nameTypeMap.clear(); | 
					
						
							|  |  |  |     const int count = l.count(); | 
					
						
							|  |  |  |     for (int i = 0; i < count; i++) { | 
					
						
							|  |  |  |         const Type t = specialType(l.at(i)); | 
					
						
							|  |  |  |         if (t != UnknownType) { | 
					
						
							|  |  |  |             // Exclude types that require expression syntax for CDB
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  |             if (debugger == GdbDebugger || !needsExpressionSyntax(t)) | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                 m_nameTypeMap.insert(l.at(i), t); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             m_nameTypeMap.insert(l.at(i), SupportedType); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*  A parse for dumper output:
 | 
					
						
							|  |  |  |  * "iname="local.sl",addr="0x0012BA84",value="<3 items>",valuedisabled="true", | 
					
						
							|  |  |  |  * numchild="3",childtype="QString",childnumchild="0",children=[{name="0",value="<binhex>", | 
					
						
							|  |  |  |  * valueencoded="2"},{name="1",value="dAB3AG8A",valueencoded="2"},{name="2", | 
					
						
							|  |  |  |  * value="dABoAHIAZQBlAA==",valueencoded="2"}]" | 
					
						
							|  |  |  |  * Default implementation can be used for debugging purposes. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  | class DumperParser | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | public: | 
					
						
							|  |  |  |     explicit DumperParser(const char *s) : m_s(s) {} | 
					
						
							|  |  |  |     bool run(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     // handle 'key="value"'
 | 
					
						
							|  |  |  |     virtual bool handleKeyword(const char *k, int size); | 
					
						
							|  |  |  |     virtual bool handleListStart(); | 
					
						
							|  |  |  |     virtual bool handleListEnd(); | 
					
						
							|  |  |  |     virtual bool handleHashStart(); | 
					
						
							|  |  |  |     virtual bool handleHashEnd(); | 
					
						
							|  |  |  |     virtual bool handleValue(const char *k, int size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     bool parseHash(int level, const char *&pos); | 
					
						
							|  |  |  |     bool parseValue(int level, const char *&pos); | 
					
						
							|  |  |  |     bool parseStringValue(const char *&ptr, int &size, const char *&pos) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const char *m_s; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // get a string value with pos at the opening double quote
 | 
					
						
							|  |  |  | bool DumperParser::parseStringValue(const char *&ptr, int &size, const char *&pos) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     pos++; | 
					
						
							|  |  |  |     const char *endValuePtr = strchr(pos, '"'); | 
					
						
							|  |  |  |     if (!endValuePtr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     size = endValuePtr - pos; | 
					
						
							|  |  |  |     ptr = pos; | 
					
						
							|  |  |  |     pos = endValuePtr + 1; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::run() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char *ptr = m_s; | 
					
						
							|  |  |  |     const bool rc = parseHash(0, ptr); | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO << '\n' << m_s << rc; | 
					
						
							|  |  |  |     return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse a non-empty hash with pos at the first keyword.
 | 
					
						
							|  |  |  | // Curly braces are present at level 0 only.
 | 
					
						
							|  |  |  | // '{a="X", b="X"}'
 | 
					
						
							|  |  |  | bool DumperParser::parseHash(int level, const char *&pos) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (true) { | 
					
						
							|  |  |  |         switch (*pos) { | 
					
						
							|  |  |  |         case '\0': // EOS is acceptable at level 0 only
 | 
					
						
							|  |  |  |             return level == 0; | 
					
						
							|  |  |  |         case '}': | 
					
						
							|  |  |  |             pos++; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const char *equalsPtr = strchr(pos, '='); | 
					
						
							|  |  |  |         if (!equalsPtr) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         const int keywordLen = equalsPtr - pos; | 
					
						
							|  |  |  |         if (!handleKeyword(pos, keywordLen)) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         pos = equalsPtr + 1; | 
					
						
							|  |  |  |         if (!*pos) | 
					
						
							|  |  |  |             return false;         | 
					
						
							|  |  |  |         if (!parseValue(level + 1, pos)) | 
					
						
							|  |  |  |             return false;     | 
					
						
							|  |  |  |         if (*pos == ',') | 
					
						
							|  |  |  |             pos++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::parseValue(int level, const char *&pos) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Simple string literal
 | 
					
						
							|  |  |  |     switch (*pos) { | 
					
						
							|  |  |  |     case '"': { | 
					
						
							|  |  |  |             const char *valuePtr; | 
					
						
							|  |  |  |             int valueSize; | 
					
						
							|  |  |  |             return parseStringValue(valuePtr, valueSize, pos) && handleValue(valuePtr, valueSize); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // A List. Note that it has a trailing comma '["a",]'
 | 
					
						
							|  |  |  |     case '[': { | 
					
						
							|  |  |  |             if (!handleListStart()) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             pos++; | 
					
						
							|  |  |  |             while (true) { | 
					
						
							|  |  |  |                 switch (*pos) { | 
					
						
							|  |  |  |                 case ']': | 
					
						
							|  |  |  |                     pos++; | 
					
						
							|  |  |  |                     return handleListEnd(); | 
					
						
							|  |  |  |                 case '\0': | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (!parseValue(level + 1, pos)) | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 if (*pos == ',') | 
					
						
							|  |  |  |                     pos++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }         | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |         // A hash '{a="b",b="c"}'
 | 
					
						
							|  |  |  |     case '{': { | 
					
						
							|  |  |  |             if (!handleHashStart()) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             pos++; | 
					
						
							|  |  |  |             if (!parseHash(level + 1, pos)) | 
					
						
							|  |  |  |                 return false;             | 
					
						
							|  |  |  |             return handleHashEnd(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::handleKeyword(const char *k, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO << '\n' << QByteArray(k, size); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::handleListStart() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::handleListEnd() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::handleHashStart() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::handleHashEnd() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DumperParser::handleValue(const char *k, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << Q_FUNC_INFO << '\n' << QByteArray(k, size); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-22 17:28:26 +02:00
										 |  |  | /* Parse 'query' (1) protocol response of the custom dumpers:
 | 
					
						
							|  |  |  |  * "'dumpers=["QByteArray","QDateTime",..."std::basic_string",], | 
					
						
							|  |  |  |  * qtversion=["4","5","1"],namespace="""' */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | class QueryDumperParser : public DumperParser { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     typedef QPair<QString, int> SizeEntry; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     explicit QueryDumperParser(const char *s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct Data { | 
					
						
							|  |  |  |         Data() : qtVersion(0) {} | 
					
						
							|  |  |  |         QString qtNameSpace; | 
					
						
							|  |  |  |         QString qtVersion; | 
					
						
							|  |  |  |         QStringList types; | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |         QList<SizeEntry> sizes; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     inline Data data() const { return m_data; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     virtual bool handleKeyword(const char *k, int size); | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     virtual bool handleListStart();     | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     virtual bool handleListEnd(); | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     virtual bool handleHashEnd(); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     virtual bool handleValue(const char *k, int size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     enum Mode { None, ExpectingDumpers, ExpectingVersion, ExpectingNameSpace, ExpectingSizes }; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     Mode m_mode; | 
					
						
							|  |  |  |     Data m_data; | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     QString m_lastSizeType; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QueryDumperParser::QueryDumperParser(const char *s) : | 
					
						
							|  |  |  |     DumperParser(s), | 
					
						
							|  |  |  |     m_mode(None) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool QueryDumperParser::handleKeyword(const char *k, int size)         | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  | {     | 
					
						
							|  |  |  |     if (m_mode == ExpectingSizes) { | 
					
						
							|  |  |  |         m_lastSizeType = QString::fromLatin1(k, size); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     if (!qstrncmp(k, "dumpers", size)) { | 
					
						
							|  |  |  |         m_mode = ExpectingDumpers; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!qstrncmp(k, "qtversion", size)) { | 
					
						
							|  |  |  |         m_mode = ExpectingVersion; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!qstrncmp(k, "namespace", size)) { | 
					
						
							|  |  |  |         m_mode = ExpectingNameSpace; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     if (!qstrncmp(k, "sizes", size)) { | 
					
						
							|  |  |  |         m_mode = ExpectingSizes; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     qWarning("%s Unexpected keyword %s.\n", Q_FUNC_INFO, QByteArray(k, size).constData()); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool QueryDumperParser::handleListStart() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_mode == ExpectingDumpers || m_mode == ExpectingVersion; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool QueryDumperParser::handleListEnd() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_mode = None; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  | bool QueryDumperParser::handleHashEnd() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_mode = None; // Size hash
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | bool QueryDumperParser::handleValue(const char *k, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (m_mode) { | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     case None: | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     case ExpectingDumpers: | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         m_data.types.push_back(QString::fromLatin1(k, size)); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     case ExpectingNameSpace: | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         m_data.qtNameSpace = QString::fromLatin1(k, size); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingVersion: // ["4","1","5"]
 | 
					
						
							|  |  |  |         if (!m_data.qtVersion.isEmpty()) | 
					
						
							|  |  |  |             m_data.qtVersion += QLatin1Char('.'); | 
					
						
							|  |  |  |         m_data.qtVersion += QString::fromLatin1(k, size); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     case ExpectingSizes: | 
					
						
							|  |  |  |         m_data.sizes.push_back(SizeEntry(m_lastSizeType, QString::fromLatin1(k, size).toInt())); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // parse a query
 | 
					
						
							|  |  |  | bool QtDumperHelper::parseQuery(const char *data, Debugger debugger) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QueryDumperParser parser(data); | 
					
						
							|  |  |  |     if (!parser.run()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     clear(); | 
					
						
							|  |  |  |     m_qtNamespace = parser.data().qtNameSpace; | 
					
						
							|  |  |  |     setQtVersion(parser.data().qtVersion); | 
					
						
							|  |  |  |     parseQueryTypes(parser.data().types, debugger); | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |     foreach (const QueryDumperParser::SizeEntry &se, parser.data().sizes) | 
					
						
							|  |  |  |         addSize(se.first, se.second); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  | void QtDumperHelper::addSize(const QString &name, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Special interest cases
 | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         if (name == QLatin1String("char*")) { | 
					
						
							|  |  |  |             m_pointerSize = size; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (name == QLatin1String("int")) { | 
					
						
							|  |  |  |             m_intSize = size; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (name.startsWith(m_stdAllocatorPrefix)) { | 
					
						
							|  |  |  |             m_stdAllocatorSize = size; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (name == QLatin1String("std::string")) { | 
					
						
							|  |  |  |             m_sizeCache.insert(QLatin1String("std::basic_string<char,std::char_traits<char>,std::allocator<char>>"), size); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (name == QLatin1String("std::wstring")) { | 
					
						
							|  |  |  |             m_sizeCache.insert(QLatin1String("std::basic_string<unsigned short,std::char_traits<unsignedshort>,std::allocator<unsignedshort> >"), size); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } while (false); | 
					
						
							|  |  |  |     m_sizeCache.insert(name, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | QtDumperHelper::Type QtDumperHelper::type(const QString &typeName) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QtDumperHelper::TypeData td = typeData(typeName); | 
					
						
							|  |  |  |     return td.type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QtDumperHelper::TypeData QtDumperHelper::typeData(const QString &typeName) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     TypeData td; | 
					
						
							|  |  |  |     td.type = UnknownType; | 
					
						
							|  |  |  |     const Type st = simpleType(typeName); | 
					
						
							|  |  |  |     if (st != UnknownType) { | 
					
						
							|  |  |  |         td.isTemplate = false; | 
					
						
							|  |  |  |         td.type =st; | 
					
						
							|  |  |  |         return td; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // Try template
 | 
					
						
							|  |  |  |     td.isTemplate = extractTemplate(typeName, &td.tmplate, &td.inner); | 
					
						
							|  |  |  |     if (!td.isTemplate) | 
					
						
							|  |  |  |         return td; | 
					
						
							|  |  |  |     // Check the template type QMap<X,Y> -> 'QMap'
 | 
					
						
							|  |  |  |     td.type = simpleType(td.tmplate); | 
					
						
							|  |  |  |     return td; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  | // Format an expression to have the debugger query the
 | 
					
						
							|  |  |  | // size. Use size cache if possible
 | 
					
						
							|  |  |  | QString QtDumperHelper::evaluationSizeofTypeExpression(const QString &typeName, | 
					
						
							|  |  |  |                                                        Debugger /* debugger */) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Look up fixed types
 | 
					
						
							|  |  |  |     if (m_pointerSize && isPointerType(typeName)) | 
					
						
							|  |  |  |         return QString::number(m_pointerSize); | 
					
						
							|  |  |  |     if (m_stdAllocatorSize && typeName.startsWith(m_stdAllocatorPrefix)) | 
					
						
							|  |  |  |         return QString::number(m_stdAllocatorSize); | 
					
						
							|  |  |  |     const SizeCache::const_iterator sit = m_sizeCache.constFind(typeName); | 
					
						
							|  |  |  |     if (sit != m_sizeCache.constEnd()) | 
					
						
							|  |  |  |         return QString::number(sit.value()); | 
					
						
							|  |  |  |     // Finally have the debugger evaluate
 | 
					
						
							|  |  |  |     return sizeofTypeExpression(typeName); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | void QtDumperHelper::evaluationParameters(const WatchData &data, | 
					
						
							|  |  |  |                                           const TypeData &td, | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |                                           Debugger debugger, | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                                           QByteArray *inBuffer, | 
					
						
							|  |  |  |                                           QStringList *extraArgsIn) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     enum { maxExtraArgCount = 4 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QStringList &extraArgs = *extraArgsIn; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // See extractTemplate for parameters
 | 
					
						
							|  |  |  |     QStringList inners = td.inner.split(QLatin1Char('@')); | 
					
						
							|  |  |  |     if (inners.at(0).isEmpty()) | 
					
						
							|  |  |  |         inners.clear(); | 
					
						
							|  |  |  |     for (int i = 0; i != inners.size(); ++i) | 
					
						
							|  |  |  |         inners[i] = inners[i].simplified(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QString outertype = td.isTemplate ? td.tmplate : data.type; | 
					
						
							|  |  |  |     // adjust the data extract
 | 
					
						
							|  |  |  |     if (outertype == m_qtNamespace + QLatin1String("QWidget")) | 
					
						
							|  |  |  |         outertype = m_qtNamespace + QLatin1String("QObject"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     extraArgs.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!inners.empty()) { | 
					
						
							|  |  |  |         // "generic" template dumpers: passing sizeof(argument)
 | 
					
						
							|  |  |  |         // gives already most information the dumpers need
 | 
					
						
							|  |  |  |         const int count = qMin(int(maxExtraArgCount), inners.size()); | 
					
						
							|  |  |  |         for (int i = 0; i < count; i++) | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |             extraArgs.push_back(evaluationSizeofTypeExpression(inners.at(i), debugger)); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     int extraArgCount = extraArgs.size(); | 
					
						
							|  |  |  |     // Pad with zeros
 | 
					
						
							|  |  |  |     const QString zero = QString(QLatin1Char('0')); | 
					
						
							|  |  |  |     const int extraPad = maxExtraArgCount - extraArgCount; | 
					
						
							|  |  |  |     for (int i = 0; i < extraPad; i++) | 
					
						
							|  |  |  |         extraArgs.push_back(zero); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // in rare cases we need more or less:
 | 
					
						
							|  |  |  |     switch (td.type) { | 
					
						
							|  |  |  |     case QObjectType: | 
					
						
							|  |  |  |     case QWidgetType: | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |         if (debugger == GdbDebugger) { | 
					
						
							|  |  |  |             extraArgs[0] = QLatin1String("(char*)&((('"); | 
					
						
							|  |  |  |             extraArgs[0] += m_qtNamespace; | 
					
						
							|  |  |  |             extraArgs[0] += QLatin1String("QObjectPrivate'*)&"); | 
					
						
							|  |  |  |             extraArgs[0] += data.exp; | 
					
						
							|  |  |  |             extraArgs[0] += QLatin1String(")->children)-(char*)&"); | 
					
						
							|  |  |  |             extraArgs[0] += data.exp; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case QVectorType: | 
					
						
							|  |  |  |         extraArgs[1] = QLatin1String("(char*)&(("); | 
					
						
							|  |  |  |         extraArgs[1] += data.exp; | 
					
						
							|  |  |  |         extraArgs[1] += QLatin1String(").d->array)-(char*)"); | 
					
						
							|  |  |  |         extraArgs[1] += data.exp; | 
					
						
							|  |  |  |         extraArgs[1] +=  QLatin1String(".d"); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case QObjectSlotType: | 
					
						
							|  |  |  |     case QObjectSignalType: { | 
					
						
							|  |  |  |             // we need the number out of something like
 | 
					
						
							|  |  |  |             // iname="local.ob.slots.2" // ".deleteLater()"?
 | 
					
						
							|  |  |  |             const int pos = data.iname.lastIndexOf('.'); | 
					
						
							|  |  |  |             const QString slotNumber = data.iname.mid(pos + 1); | 
					
						
							|  |  |  |             QTC_ASSERT(slotNumber.toInt() != -1, /**/); | 
					
						
							|  |  |  |             extraArgs[0] = slotNumber; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case QMapType: | 
					
						
							|  |  |  |     case QMultiMapType: { | 
					
						
							|  |  |  |             QString nodetype; | 
					
						
							|  |  |  |             if (m_qtVersion >= (4 << 16) + (5 << 8) + 0) { | 
					
						
							|  |  |  |                 nodetype = m_qtNamespace + QLatin1String("QMapNode"); | 
					
						
							|  |  |  |                 nodetype += data.type.mid(outertype.size()); | 
					
						
							| 
									
										
										
										
											2009-04-22 17:28:26 +02:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                 // FIXME: doesn't work for QMultiMap
 | 
					
						
							|  |  |  |                 nodetype  = data.type + QLatin1String("::Node"); | 
					
						
							| 
									
										
										
										
											2009-04-22 17:28:26 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             //qDebug() << "OUTERTYPE: " << outertype << " NODETYPE: " << nodetype
 | 
					
						
							|  |  |  |             //    << "QT VERSION" << m_qtVersion << ((4 << 16) + (5 << 8) + 0);
 | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |             extraArgs[2] = evaluationSizeofTypeExpression(nodetype, debugger); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             extraArgs[3] = QLatin1String("(size_t)&(('"); | 
					
						
							|  |  |  |             extraArgs[3] += nodetype; | 
					
						
							|  |  |  |             extraArgs[3] += QLatin1String("'*)0)->value"); | 
					
						
							| 
									
										
										
										
											2009-04-22 17:28:26 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |             case QMapNodeType: | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |         extraArgs[2] = evaluationSizeofTypeExpression(data.type, debugger); | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         extraArgs[3] = QLatin1String("(size_t)&(('"); | 
					
						
							|  |  |  |         extraArgs[3] += data.type; | 
					
						
							|  |  |  |         extraArgs[3] += QLatin1String("'*)0)->value"); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case StdVectorType: | 
					
						
							|  |  |  |         //qDebug() << "EXTRACT TEMPLATE: " << outertype << inners;
 | 
					
						
							|  |  |  |         if (inners.at(0) == QLatin1String("bool")) { | 
					
						
							|  |  |  |             outertype = QLatin1String("std::vector::bool"); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  |             //extraArgs[extraArgCount++] = evaluationSizeofTypeExpression(data.type, debugger);
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |             //extraArgs[extraArgCount++] = "(size_t)&(('" + data.type + "'*)0)->value";
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case StdDequeType: | 
					
						
							|  |  |  |         extraArgs[1] = zero; | 
					
						
							|  |  |  |     case StdStackType: | 
					
						
							|  |  |  |         // remove 'std::allocator<...>':
 | 
					
						
							|  |  |  |         extraArgs[1] = zero; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case StdSetType: | 
					
						
							|  |  |  |         // remove 'std::less<...>':
 | 
					
						
							|  |  |  |         extraArgs[1] = zero; | 
					
						
							|  |  |  |         // remove 'std::allocator<...>':
 | 
					
						
							|  |  |  |         extraArgs[2] = zero; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case StdMapType: { | 
					
						
							|  |  |  |             // We don't want the comparator and the allocator confuse gdb.
 | 
					
						
							|  |  |  |             // But we need the offset of the second item in the value pair.
 | 
					
						
							|  |  |  |             // We read the type of the pair from the allocator argument because
 | 
					
						
							|  |  |  |             // that gets the constness "right" (in the sense that gdb can
 | 
					
						
							|  |  |  |             // read it back;
 | 
					
						
							|  |  |  |             QString pairType = inners.at(3); | 
					
						
							|  |  |  |             // remove 'std::allocator<...>':
 | 
					
						
							|  |  |  |             pairType = pairType.mid(15, pairType.size() - 15 - 2); | 
					
						
							|  |  |  |             extraArgs[2] = QLatin1String("(size_t)&(('"); | 
					
						
							|  |  |  |             extraArgs[2] += pairType; | 
					
						
							|  |  |  |             extraArgs[2] += QLatin1String("'*)0)->second"); | 
					
						
							|  |  |  |             extraArgs[3] = zero; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case StdStringType: | 
					
						
							|  |  |  |         //qDebug() << "EXTRACT TEMPLATE: " << outertype << inners;
 | 
					
						
							|  |  |  |         if (inners.at(0) == QLatin1String("char")) { | 
					
						
							|  |  |  |             outertype = QLatin1String("std::string"); | 
					
						
							|  |  |  |         } else if (inners.at(0) == QLatin1String("wchar_t")) { | 
					
						
							|  |  |  |             outertype = QLatin1String("std::wstring"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         qFill(extraArgs, zero); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case UnknownType: | 
					
						
							|  |  |  |         qWarning("Unknown type encountered in %s.\n", Q_FUNC_INFO); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case SupportedType: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     inBuffer->clear(); | 
					
						
							|  |  |  |     inBuffer->append(outertype.toUtf8()); | 
					
						
							|  |  |  |     inBuffer->append('\0'); | 
					
						
							|  |  |  |     inBuffer->append(data.iname.toUtf8()); | 
					
						
							|  |  |  |     inBuffer->append('\0'); | 
					
						
							|  |  |  |     inBuffer->append(data.exp.toUtf8()); | 
					
						
							|  |  |  |     inBuffer->append('\0'); | 
					
						
							|  |  |  |     inBuffer->append(td.inner.toUtf8()); | 
					
						
							|  |  |  |     inBuffer->append('\0'); | 
					
						
							|  |  |  |     inBuffer->append(data.iname.toUtf8()); | 
					
						
							|  |  |  |     inBuffer->append('\0'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << '\n' << Q_FUNC_INFO << '\n' << data.toString() << "\n-->" << outertype << td.type << extraArgs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Parse value:
 | 
					
						
							|  |  |  |  * "iname="local.sl",addr="0x0012BA84",value="<3 items>",valuedisabled="true", | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  |  * numchild="3",childtype="QString",childnumchild="0", | 
					
						
							|  |  |  |  * children=[{name="0",value="<binhex>",valueencoded="2"}, | 
					
						
							|  |  |  |  * {name="1",value="dAB3AG8A",valueencoded="2"}, | 
					
						
							|  |  |  |  * {name="2",value="dABoAHIAZQBlAA==",valueencoded="2"}]" */ | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:03:35 +02:00
										 |  |  | class ValueDumperParser : public DumperParser | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | public: | 
					
						
							|  |  |  |     explicit ValueDumperParser(const char *s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     inline QtDumperResult result() const { return m_result; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     virtual bool handleKeyword(const char *k, int size); | 
					
						
							|  |  |  |     virtual bool handleHashStart(); | 
					
						
							|  |  |  |     virtual bool handleValue(const char *k, int size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     enum Mode { None, ExpectingIName, ExpectingAddress, ExpectingValue, | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |                 ExpectingType, ExpectingDisplayedType, ExpectingInternal, | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                 ExpectingValueDisabled,  ExpectingValueEncoded, | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |                 ExpectingCommonChildType, ExpectingChildCount, | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |                 IgnoreNext, | 
					
						
							|  |  |  |                 ChildModeStart, | 
					
						
							|  |  |  |                 ExpectingChildren,ExpectingChildName, ExpectingChildAddress, | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |                 ExpectingChildExpression, ExpectingChildType, | 
					
						
							|  |  |  |                 ExpectingChildValue, ExpectingChildValueEncoded, | 
					
						
							|  |  |  |                 ExpectingChildValueDisabled, ExpectingChildChildCount | 
					
						
							|  |  |  |               }; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static inline Mode nextMode(Mode in, const char *keyword, int size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Mode m_mode; | 
					
						
							|  |  |  |     QtDumperResult m_result; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ValueDumperParser::ValueDumperParser(const char *s) : | 
					
						
							|  |  |  |    DumperParser(s), | 
					
						
							|  |  |  |    m_mode(None) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Check key words
 | 
					
						
							|  |  |  | ValueDumperParser::Mode ValueDumperParser::nextMode(Mode in, const char *keyword, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Careful with same prefix
 | 
					
						
							|  |  |  |     switch (size) { | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     case 3: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "exp", size)) | 
					
						
							|  |  |  |             return ExpectingChildExpression; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     case 4: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "addr", size)) | 
					
						
							|  |  |  |             return in > ChildModeStart ? ExpectingChildAddress : ExpectingAddress; | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "type", size)) | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |             return in > ChildModeStart ? ExpectingChildType : ExpectingType; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         if (!qstrncmp(keyword, "name", size)) | 
					
						
							|  |  |  |             return ExpectingChildName; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case 5: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "iname", size)) | 
					
						
							|  |  |  |             return ExpectingIName; | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "value", size)) | 
					
						
							|  |  |  |             return in > ChildModeStart ? ExpectingChildValue : ExpectingValue; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case 8: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "children", size)) | 
					
						
							|  |  |  |             return ExpectingChildren; | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "numchild", size)) | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |             return in > ChildModeStart ?  ExpectingChildChildCount : ExpectingChildCount; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         if (!qstrncmp(keyword, "internal", size)) | 
					
						
							|  |  |  |             return ExpectingInternal; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case 9: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "childtype", size)) | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |             return ExpectingCommonChildType; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         break;     | 
					
						
							|  |  |  |     case 12: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "valueencoded", size)) | 
					
						
							|  |  |  |             return in > ChildModeStart ? ExpectingChildValueEncoded : ExpectingValueEncoded; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case 13: | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "valuedisabled", size)) | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |             return in > ChildModeStart ? ExpectingChildValueDisabled : ExpectingValueDisabled; | 
					
						
							|  |  |  |         if (!qstrncmp(keyword, "displayedtype", size)) | 
					
						
							|  |  |  |             return ExpectingDisplayedType; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         if (!qstrncmp(keyword, "childnumchild", size)) | 
					
						
							|  |  |  |             return IgnoreNext; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IgnoreNext; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ValueDumperParser::handleKeyword(const char *k, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const Mode newMode = nextMode(m_mode, k, size); | 
					
						
							|  |  |  |     if (debug && newMode == IgnoreNext) | 
					
						
							|  |  |  |         qWarning("%s Unexpected keyword %s.\n", Q_FUNC_INFO, QByteArray(k, size).constData()); | 
					
						
							|  |  |  |     m_mode = newMode; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ValueDumperParser::handleHashStart() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_result.children.push_back(QtDumperResult::Child()); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ValueDumperParser::handleValue(const char *k, int size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QByteArray valueBA(k, size); | 
					
						
							|  |  |  |     switch (m_mode) { | 
					
						
							|  |  |  |     case None: | 
					
						
							|  |  |  |     case ChildModeStart: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     case ExpectingIName: | 
					
						
							|  |  |  |         m_result.iname = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingAddress: | 
					
						
							|  |  |  |         m_result.address = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingValue: | 
					
						
							|  |  |  |         m_result.value = valueBA; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingValueDisabled: | 
					
						
							|  |  |  |         m_result.valuedisabled = valueBA == "true"; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingValueEncoded: | 
					
						
							|  |  |  |         m_result.valueEncoded = QString::fromLatin1(valueBA).toInt(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingType: | 
					
						
							|  |  |  |         m_result.type = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     case ExpectingDisplayedType: | 
					
						
							|  |  |  |         m_result.displayedType = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     case ExpectingInternal: | 
					
						
							|  |  |  |         m_result.internal = valueBA == "true"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     case ExpectingCommonChildType: | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |         m_result.childType = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildCount: | 
					
						
							|  |  |  |         m_result.childCount = QString::fromLatin1(valueBA).toInt(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildren: | 
					
						
							|  |  |  |     case IgnoreNext: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildName: | 
					
						
							|  |  |  |         m_result.children.back().name = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildAddress: | 
					
						
							|  |  |  |         m_result.children.back().address = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildValue: | 
					
						
							|  |  |  |         m_result.children.back().value = valueBA; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     case ExpectingChildExpression: | 
					
						
							|  |  |  |         m_result.children.back().exp = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     case ExpectingChildValueEncoded: | 
					
						
							|  |  |  |         m_result.children.back().valueEncoded = QString::fromLatin1(valueBA).toInt(); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     case ExpectingChildValueDisabled: | 
					
						
							|  |  |  |         m_result.children.back().valuedisabled = valueBA == "true"; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildType: | 
					
						
							|  |  |  |         m_result.children.back().type = QString::fromLatin1(valueBA); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case ExpectingChildChildCount: | 
					
						
							|  |  |  |         m_result.children.back().childCount = QString::fromLatin1(valueBA).toInt(); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-04-22 17:28:26 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  | bool QtDumperHelper::parseValue(const char *data, QtDumperResult *r) | 
					
						
							|  |  |  | {     | 
					
						
							|  |  |  |     ValueDumperParser parser(data); | 
					
						
							|  |  |  |     if (!parser.run()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     *r = parser.result(); | 
					
						
							| 
									
										
										
										
											2009-05-05 16:39:51 +02:00
										 |  |  |     // Sanity
 | 
					
						
							|  |  |  |     if (r->childCount < r->children.size()) | 
					
						
							|  |  |  |         r->childCount  = r->children.size(); | 
					
						
							| 
									
										
										
										
											2009-05-06 09:22:05 +02:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << '\n' << data << *r; | 
					
						
							| 
									
										
										
										
											2009-04-29 14:15:09 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 16:52:14 +02:00
										 |  |  | QDebug operator<<(QDebug in, const QtDumperHelper::TypeData &d) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QDebug nsp = in.nospace(); | 
					
						
							|  |  |  |     nsp << " type=" << d.type << " tpl=" << d.isTemplate; | 
					
						
							|  |  |  |     if (d.isTemplate) | 
					
						
							|  |  |  |         nsp << d.tmplate << '<' << d.inner << '>'; | 
					
						
							|  |  |  |     return in; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-05 10:22:44 +02:00
										 |  |  | } // namespace Internal
 | 
					
						
							|  |  |  | } // namespace Debugger
 |