Files correlati : cg0.exe cg0700a.msk cg0700b.msk cg3.exe cg4.exe Bug : Commento: Merge 1.0 libraries
251 lines
8.3 KiB
C++
251 lines
8.3 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
** Contact: http://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the test suite of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see http://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at http://www.qt.io/contact-us.
|
|
**
|
|
** 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 or version 3 as published by the Free
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
** following information to ensure the GNU Lesser General Public License
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3.0 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.GPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU General Public License version 3.0 requirements will be
|
|
** met: http://www.gnu.org/copyleft/gpl.html.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
|
|
#include "qbic.h"
|
|
|
|
#include "QtCore/qfile.h"
|
|
#include "QtCore/qdebug.h"
|
|
|
|
void QBic::addBlacklistedClass(const QString &wildcard)
|
|
{
|
|
blackList.append(QRegExp(wildcard, Qt::CaseSensitive, QRegExp::Wildcard));
|
|
}
|
|
|
|
void QBic::removeBlacklistedClass(const QString &wildcard)
|
|
{
|
|
blackList.removeAll(QRegExp(wildcard, Qt::CaseSensitive, QRegExp::Wildcard));
|
|
}
|
|
|
|
bool QBic::isBlacklisted(const QString &className) const
|
|
{
|
|
// all templates are blacklisted
|
|
if (className.contains('<'))
|
|
return true;
|
|
|
|
for (int i = 0; i < blackList.count(); ++i)
|
|
if (blackList.at(i).exactMatch(className))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
static QStringList normalizedVTable(const QStringList &entry)
|
|
{
|
|
QStringList normalized;
|
|
|
|
for (int i = 2; i < entry.count(); ++i) {
|
|
const QString line = entry.at(i).simplified();
|
|
bool isOk = false;
|
|
int num = line.left(line.indexOf(QLatin1Char(' '))).toInt(&isOk);
|
|
if (!isOk) {
|
|
qWarning("unrecognized line: %s", qPrintable(line));
|
|
continue;
|
|
}
|
|
|
|
QString sym = line.mid(line.indexOf(QLatin1Char(' ')) + 1);
|
|
if (sym.startsWith(QLatin1Char('('))) {
|
|
if (sym.endsWith(QLatin1Char(')'))) {
|
|
sym = sym.mid(sym.lastIndexOf('(') + 1);
|
|
sym.chop(1);
|
|
} else {
|
|
sym = sym.mid(sym.lastIndexOf(QLatin1Char(')')) + 1);
|
|
}
|
|
} else {
|
|
sym = sym.left(sym.indexOf(QLatin1Char('(')));
|
|
}
|
|
|
|
if (sym.startsWith(QLatin1String("& ")))
|
|
sym.remove(1, 1);
|
|
|
|
if (sym.startsWith(QLatin1String("-0")) || sym.startsWith(QLatin1String("0"))) {
|
|
if (sym.endsWith('u'))
|
|
sym.chop(1);
|
|
|
|
bool isOk = false;
|
|
qint64 num = sym.toLongLong(&isOk, 16);
|
|
if (!isOk) {
|
|
qWarning("unrecognized token: %s", qPrintable(sym));
|
|
continue;
|
|
}
|
|
if (sizeof(void*) == 4)
|
|
sym = QString::number(int(num));
|
|
else
|
|
sym = QString::number(num);
|
|
}
|
|
|
|
normalized << QString::number(num) + QLatin1Char(' ') + sym;
|
|
}
|
|
|
|
return normalized;
|
|
}
|
|
|
|
QBic::Info QBic::parseOutput(const QByteArray &ba) const
|
|
{
|
|
Info info;
|
|
const QStringList source = QString::fromLatin1(ba).split("\n\n");
|
|
|
|
foreach(QString str, source) {
|
|
QStringList entry = str.split('\n');
|
|
if (entry.count() < 2)
|
|
continue;
|
|
if (entry.at(0).startsWith("Class ")) {
|
|
const QString className = entry.at(0).mid(6);
|
|
if (isBlacklisted(className))
|
|
continue;
|
|
QRegExp rx("size=(\\d+)");
|
|
if (rx.indexIn(entry.at(1)) == -1) {
|
|
qWarning("Could not parse class information for className %s", className.toLatin1().constData());
|
|
continue;
|
|
}
|
|
info.classSizes[className] = rx.cap(1).toInt();
|
|
} else if (entry.at(0).startsWith("Vtable for ")) {
|
|
const QString className = entry.at(0).mid(11);
|
|
if (isBlacklisted(className))
|
|
continue;
|
|
info.classVTables[className] = normalizedVTable(entry);
|
|
}
|
|
}
|
|
|
|
return info;
|
|
}
|
|
|
|
QBic::Info QBic::parseFile(const QString &fileName) const
|
|
{
|
|
QFile f(fileName);
|
|
if (!f.open(QIODevice::ReadOnly | QIODevice::Text))
|
|
return Info();
|
|
|
|
QByteArray ba = f.readAll();
|
|
f.close();
|
|
|
|
return parseOutput(ba);
|
|
}
|
|
|
|
enum VTableDiffResult { Match, Mismatch, Reimp };
|
|
static VTableDiffResult diffVTableEntry(const QString &v1, const QString &v2)
|
|
{
|
|
if (v1 == v2)
|
|
return Match;
|
|
if (v2.endsWith(QLatin1String("__cxa_pure_virtual")))
|
|
return Reimp;
|
|
if (!v1.contains(QLatin1String("::")) || !v2.contains(QLatin1String("::")))
|
|
return Mismatch;
|
|
|
|
const QString sym1 = v1.mid(v1.lastIndexOf(QLatin1String("::")) + 2);
|
|
const QString sym2 = v2.mid(v2.lastIndexOf(QLatin1String("::")) + 2);
|
|
|
|
if (sym1 == sym2)
|
|
return Reimp;
|
|
|
|
return Mismatch;
|
|
}
|
|
|
|
QBic::VTableDiff QBic::diffVTables(const Info &oldLib, const Info &newLib) const
|
|
{
|
|
VTableDiff result;
|
|
|
|
for (QHash<QString, QStringList>::const_iterator it = newLib.classVTables.constBegin();
|
|
it != newLib.classVTables.constEnd(); ++it) {
|
|
if (!oldLib.classVTables.contains(it.key())) {
|
|
result.addedVTables.append(it.key());
|
|
continue;
|
|
}
|
|
const QStringList oldVTable = oldLib.classVTables.value(it.key());
|
|
const QStringList vTable = it.value();
|
|
if (vTable.count() != oldVTable.count()) {
|
|
result.modifiedVTables.append(QPair<QString, QString>(it.key(),
|
|
QLatin1String("size mismatch")));
|
|
continue;
|
|
}
|
|
|
|
for (int i = 0; i < vTable.count(); ++i) {
|
|
VTableDiffResult diffResult = diffVTableEntry(vTable.at(i), oldVTable.at(i));
|
|
switch (diffResult) {
|
|
case Match:
|
|
// do nothing
|
|
break;
|
|
case Mismatch:
|
|
result.modifiedVTables.append(QPair<QString, QString>(oldVTable.at(i),
|
|
vTable.at(i)));
|
|
break;
|
|
case Reimp:
|
|
result.reimpMethods.append(QPair<QString, QString>(oldVTable.at(i), vTable.at(i)));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (QHash<QString, QStringList>::const_iterator it = oldLib.classVTables.constBegin();
|
|
it != oldLib.classVTables.constEnd(); ++it) {
|
|
if (!newLib.classVTables.contains(it.key()))
|
|
result.removedVTables.append(it.key());
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
QBic::SizeDiff QBic::diffSizes(const Info &oldLib, const Info &newLib) const
|
|
{
|
|
QBic::SizeDiff result;
|
|
|
|
for (QHash<QString, int>::const_iterator it = newLib.classSizes.constBegin();
|
|
it != newLib.classSizes.constEnd(); ++it) {
|
|
if (!oldLib.classSizes.contains(it.key())) {
|
|
result.added.append(it.key());
|
|
continue;
|
|
}
|
|
int oldSize = oldLib.classSizes.value(it.key());
|
|
int newSize = it.value();
|
|
|
|
if (oldSize != newSize)
|
|
result.mismatch.append(it.key());
|
|
}
|
|
|
|
for (QHash<QString, int>::const_iterator it = oldLib.classSizes.constBegin();
|
|
it != oldLib.classSizes.constEnd(); ++it) {
|
|
if (!newLib.classSizes.contains(it.key()))
|
|
result.removed.append(it.key());
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|