/**************************************************************************** ** ** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 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. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt 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 #include #include #include class tst_Selftests: public QObject { Q_OBJECT private slots: void initTestCase(); void runSubTest_data(); void runSubTest(); void cleanupTestCase(); private: void doRunSubTest(QString const& subdir, QString const& logger, QStringList const& arguments ); }; struct BenchmarkResult { qint64 total; qint64 iterations; QString unit; inline QString toString() const { return QString("total:%1, unit:%2, iterations:%3").arg(total).arg(unit).arg(iterations); } static BenchmarkResult parse(QString const&, QString*); }; QT_BEGIN_NAMESPACE namespace QTest { template <> inline bool qCompare (BenchmarkResult const &r1, BenchmarkResult const &r2, const char* actual, const char* expected, const char* file, int line) { // First make sure the iterations and unit match. if (r1.iterations != r2.iterations || r1.unit != r2.unit) { /* Nope - compare whole string for best failure message */ return qCompare(r1.toString(), r2.toString(), actual, expected, file, line); } /* Now check the value. Some variance is allowed, and how much depends on the measured unit. */ qreal variance = 0.; if (r1.unit == "msec") { variance = 0.1; } else if (r1.unit == "instruction reads") { variance = 0.001; } else if (r1.unit == "ticks") { variance = 0.001; } if (variance == 0.) { /* No variance allowed - compare whole string */ return qCompare(r1.toString(), r2.toString(), actual, expected, file, line); } if (qAbs(qreal(r1.total) - qreal(r2.total)) <= qreal(r1.total)*variance) { return compare_helper(true, "COMPARE()", file, line); } /* Whoops, didn't match. Compare the whole string for the most useful failure message. */ return qCompare(r1.toString(), r2.toString(), actual, expected, file, line); } } QT_END_NAMESPACE static QList splitLines(QByteArray ba) { ba.replace('\r', ""); QList out = ba.split('\n'); // Replace any ` file="..."' or ` line="..."' in XML with a generic location. static const char *markers[][2] = { { " file=\"", " file=\"__FILE__\"" }, { " line=\"", " line=\"__LINE__\"" } }; static const int markerCount = sizeof markers / sizeof markers[0]; for (int i = 0; i < out.size(); ++i) { QByteArray& line = out[i]; for (int j = 0; j < markerCount; ++j) { int index = line.indexOf(markers[j][0]); if (index == -1) { continue; } int end = line.indexOf('"', index + strlen(markers[j][0])); if (end == -1) { continue; } line.replace(index, end-index + 1, markers[j][1]); } } return out; } static QList expectedResult(const QString &subdir, const QString &logger) { QString suffix = logger; if (suffix.isEmpty()) { suffix = "txt"; } QFile file(":/expected_" + subdir + "." + suffix); if (!file.open(QIODevice::ReadOnly)) return QList(); return splitLines(file.readAll()); } struct Logger { Logger(QString const&, QString const&, QStringList const&); QString name; QString testdata_suffix; QStringList arguments; }; Logger::Logger(QString const& _name, QString const& _testdata_suffix, QStringList const& _arguments) : name(_name) , testdata_suffix(_testdata_suffix) , arguments(_arguments) { } static QList allLoggers() { return QList() << Logger("plain", "txt", QStringList()) << Logger("xml", "xml", QStringList() << "-xml") << Logger("xml flush", "xml", QStringList() << "-xml" << "-flush") << Logger("xunitxml", "xunitxml", QStringList() << "-xunitxml") << Logger("lightxml", "lightxml", QStringList() << "-lightxml") ; } void tst_Selftests::initTestCase() { #ifdef Q_OS_WIN // cd up to be able to locate the binaries of the sub-processes. QDir workingDirectory = QDir::current(); if (workingDirectory.absolutePath().endsWith(QLatin1String("/debug"), Qt::CaseInsensitive) || workingDirectory.absolutePath().endsWith(QLatin1String("/release"), Qt::CaseInsensitive)) { QVERIFY(workingDirectory.cdUp()); QVERIFY(QDir::setCurrent(workingDirectory.absolutePath())); } #endif } void tst_Selftests::runSubTest_data() { QTest::addColumn("subdir"); QTest::addColumn("logger"); QTest::addColumn("arguments"); QStringList tests = QStringList() << "subtest" << "warnings" << "maxwarnings" << "cmptest" // << "alive" // timer dependent << "globaldata" << "skipglobal" << "skip" << "strcmp" << "expectfail" << "sleep" << "fetchbogus" << "crashes" << "multiexec" << "failinit" << "failinitdata" << "skipinit" << "skipinitdata" << "datetime" << "singleskip" //on windows assert does nothing in release mode and blocks execution with a popup window in debug mode #if !defined(Q_OS_WIN) << "assert" #endif << "waitwithoutgui" << "differentexec" #ifndef QT_NO_EXCEPTIONS // The machine that run the intel autotests will popup a dialog // with a warning that an uncaught exception was thrown. // This will time out and falsely fail, therefore we disable the test for that platform. # if !defined(Q_CC_INTEL) || !defined(Q_OS_WIN) << "exceptionthrow" # endif #endif << "qexecstringlist" << "datatable" << "commandlinedata" #if defined(__GNUC__) && defined(__i386) && defined(Q_OS_LINUX) << "benchlibcallgrind" #endif << "benchlibeventcounter" << "benchliboptions" //### These tests are affected by timing and whether the CPU tick counter is //### monotonically increasing. They won't work on some machines so leave them off by default. //### Feel free to uncomment for your own testing. #if 0 << "benchlibwalltime" << "benchlibtickcounter" #endif << "xunit" << "longstring" << "badxml" << "printdatatags" << "printdatatagswithglobaltags" ; foreach (Logger const& logger, allLoggers()) { QString rowSuffix; if (logger.name != "plain") { rowSuffix = QString(" %1").arg(logger.name); } foreach (QString const& subtest, tests) { QStringList arguments = logger.arguments; if (subtest == "commandlinedata") { arguments << QString("fiveTablePasses fiveTablePasses:fiveTablePasses_data1 -v2").split(' '); } else if (subtest == "benchlibcallgrind") { arguments << "-callgrind"; } else if (subtest == "benchlibeventcounter") { arguments << "-eventcounter"; } else if (subtest == "benchliboptions") { arguments << "-eventcounter"; } else if (subtest == "benchlibtickcounter") { arguments << "-tickcounter"; } else if (subtest == "badxml") { arguments << "-eventcounter"; } else if (subtest == "printdatatags") { arguments << "-datatags"; } else if (subtest == "printdatatagswithglobaltags") { arguments << "-datatags"; } // These tests don't work right with loggers other than plain, usually because // they internally supply arguments to themselves. if (logger.name != "plain") { if (subtest == "differentexec") { continue; } if (subtest == "qexecstringlist") { continue; } if (subtest == "benchliboptions") { continue; } if (subtest == "waitwithoutgui") { continue; } if (subtest == "printdatatags") { continue; } if (subtest == "printdatatagswithglobaltags") { continue; } // `crashes' will not output valid XML on platforms without a crash handler if (subtest == "crashes") { continue; } // this test prints out some floats in the testlog and the formatting is // platform-specific and hard to predict. if (subtest == "subtest") { continue; } } QTest::newRow(qPrintable(QString("%1%2").arg(subtest).arg(rowSuffix))) << subtest << logger.testdata_suffix << arguments ; } } } static inline QByteArray msgCannotStartProcess(const QString &binary, const QString &why) { return QString::fromLatin1("Cannot start '%1' from '%2': %3") .arg(QDir::toNativeSeparators(binary), QDir::toNativeSeparators(QDir::currentPath()), why).toLocal8Bit(); } void tst_Selftests::doRunSubTest(QString const& subdir, QString const& logger, QStringList const& arguments ) { // For the plain text logger, we'll read straight from standard output. // For all other loggers (XML), we'll tell testlib to redirect to a file. // The reason is that tests are allowed to print to standard output, and // that means the test log is no longer guaranteed to be valid XML. QStringList extraArguments; QString logfile; if (logger != "txt") { logfile = "test_output"; extraArguments << "-o" << logfile; } QProcess proc; proc.setEnvironment(QStringList("")); const QString binary = subdir + QLatin1Char('/') + subdir; proc.start(binary, QStringList() << arguments << extraArguments); QVERIFY2(proc.waitForStarted(), msgCannotStartProcess(binary, proc.errorString())); QVERIFY2(proc.waitForFinished(), qPrintable(proc.errorString())); QByteArray out; if (logfile.isEmpty()) { out = proc.readAllStandardOutput(); } else { QFile file(logfile); if (file.open(QIODevice::ReadOnly)) out = file.readAll(); } const QByteArray err(proc.readAllStandardError()); /* Some tests may output unpredictable strings to stderr, which we'll ignore. For instance, uncaught exceptions on Windows might say (depending on Windows version and JIT debugger settings): "This application has requested the Runtime to terminate it in an unusual way. Please contact the application's support team for more information." Also, tests which use valgrind may generate warnings if the toolchain is newer than the valgrind version, such that valgrind can't understand the debug information on the binary. */ if (subdir != QLatin1String("exceptionthrow") && subdir != QLatin1String("fetchbogus") && subdir != QLatin1String("xunit") && subdir != QLatin1String("benchlibcallgrind")) QVERIFY2(err.isEmpty(), err.constData()); QList res = splitLines(out); QList exp = expectedResult(subdir, logger); if (exp.count() == 0) { QList > expArr; int i = 1; do { exp = expectedResult(subdir + QString("_%1").arg(i++), logger); if (exp.count()) expArr += exp; } while(exp.count()); for (int j = 0; j < expArr.count(); ++j) { if (res.count() == expArr.at(j).count()) { exp = expArr.at(j); break; } } } else { QCOMPARE(res.count(), exp.count()); } if (logger == "xunitxml" || logger == "xml" || logger == "lightxml") { QByteArray xml(out); // lightxml intentionally skips the root element, which technically makes it // not valid XML. // We'll add that ourselves for the purpose of validation. if (logger == "lightxml") { xml.prepend(""); xml.append(""); } QXmlStreamReader reader(xml); while(!reader.atEnd()) reader.readNext(); QVERIFY2(!reader.error(), qPrintable(QString("line %1, col %2: %3") .arg(reader.lineNumber()) .arg(reader.columnNumber()) .arg(reader.errorString()) )); } bool benchmark = false; for (int i = 0; i < res.count(); ++i) { QByteArray line = res.at(i); if (line.startsWith("Config: Using QTest")) continue; // the __FILE__ __LINE__ output is compiler dependent, skip it if (line.startsWith(" Loc: [") && line.endsWith(")]")) continue; if (line.endsWith(" : failure location")) continue; const QString output(QString::fromLatin1(line)); const QString expected(QString::fromLatin1(exp.at(i)).replace("@INSERT_QT_VERSION_HERE@", QT_VERSION_STR)); if (line.contains("ASSERT") && output != expected) { QEXPECT_FAIL("assert", "QTestLib prints out the absolute path.", Continue); QEXPECT_FAIL("assert xml", "QTestLib prints out the absolute path.", Continue); QEXPECT_FAIL("assert xml flush","QTestLib prints out the absolute path.", Continue); QEXPECT_FAIL("assert lightxml", "QTestLib prints out the absolute path.", Continue); QEXPECT_FAIL("assert xunitxml", "QTestLib prints out the absolute path.", Continue); } /* On some platforms we compile without RTTI, and as a result we never throw an exception. */ if(expected.startsWith(QLatin1String("FAIL! : tst_Exception::throwException() Caught unhandled exce")) && expected != output) QCOMPARE(output.simplified(), QString::fromLatin1("tst_Exception::throwException()").simplified()); else { if(output != expected && qstrcmp(QTest::currentDataTag(), "subtest") == 0) { /* The floating point formatting differs between platforms, so let's just skip it. */ continue; } else { /* Are we expecting this line to be a benchmark result? If so, don't do a literal comparison, since results have some natural variance. */ if (benchmark || line.startsWith(" if (!line.endsWith("/>")) { if (error) *error = "unterminated XML"; return out; } QString unit = extractXmlAttribute(line, " metric=\""); QString sTotal = extractXmlAttribute(line, " value=\""); QString sIterations = extractXmlAttribute(line, " iterations=\""); if (unit.isNull() || sTotal.isNull() || sIterations.isNull()) { if (error) *error = "XML snippet did not contain all required values"; return out; } bool ok; double total = sTotal.toDouble(&ok); if (!ok) { if (error) *error = sTotal + " is not a valid number"; return out; } double iterations = sIterations.toDouble(&ok); if (!ok) { if (error) *error = sIterations + " is not a valid number"; return out; } out.unit = unit; out.total = total; out.iterations = iterations; return out; } // Text result /* This code avoids using a QRegExp because QRegExp might be broken. */ /* Sample format: 4,000 msec per iteration (total: 4,000, iterations: 1) */ QString sFirstNumber; while (!remaining.isEmpty() && !remaining.at(0).isSpace()) { sFirstNumber += remaining.at(0); remaining.remove(0,1); } remaining = remaining.trimmed(); /* 4,000 -> 4000 */ sFirstNumber.remove(','); /* Should now be parseable as floating point */ bool ok; double firstNumber = sFirstNumber.toDouble(&ok); if (!ok) { if (error) *error = sFirstNumber + " (at beginning of line) is not a valid number"; return out; } /* Remaining: msec per iteration (total: 4000, iterations: 1) */ static const char periterbit[] = " per iteration (total: "; QString unit; while (!remaining.startsWith(periterbit) && !remaining.isEmpty()) { unit += remaining.at(0); remaining.remove(0,1); } if (remaining.isEmpty()) { if (error) *error = "Could not find pattern: ' per iteration (total: '"; return out; } remaining = remaining.mid(sizeof(periterbit)-1); /* Remaining: 4,000, iterations: 1) */ static const char itersbit[] = ", iterations: "; QString sTotal; while (!remaining.startsWith(itersbit) && !remaining.isEmpty()) { sTotal += remaining.at(0); remaining.remove(0,1); } if (remaining.isEmpty()) { if (error) *error = "Could not find pattern: ', iterations: '"; return out; } remaining = remaining.mid(sizeof(itersbit)-1); /* 4,000 -> 4000 */ sTotal.remove(','); double total = sTotal.toDouble(&ok); if (!ok) { if (error) *error = sTotal + " (total) is not a valid number"; return out; } /* Remaining: 1) */ QString sIters; while (remaining != QLatin1String(")") && !remaining.isEmpty()) { sIters += remaining.at(0); remaining.remove(0,1); } if (remaining.isEmpty()) { if (error) *error = "Could not find pattern: ')'"; return out; } qint64 iters = sIters.toLongLong(&ok); if (!ok) { if (error) *error = sIters + " (iterations) is not a valid integer"; return out; } double calcFirstNumber = double(total)/double(iters); if (!qFuzzyCompare(firstNumber, calcFirstNumber)) { if (error) *error = QString("total/iters is %1, but benchlib output result as %2").arg(calcFirstNumber).arg(firstNumber); return out; } out.total = total; out.unit = unit; out.iterations = iters; return out; } void tst_Selftests::cleanupTestCase() { QFile::remove("test_output"); } QTEST_MAIN(tst_Selftests) #include "tst_selftests.moc"