Disable hash seeding for bootstrapped tools

Any bootstrapped tool is a development tool, by definition. So the
effects of seeding the hash with a random number can cause the same
source input to produce different binary results, which can throw some
caching tools into disarray (like the Open Build System).

There should be minimal fall out from the reduced protection against
DoS. Since those are only development tools, "specially crafted" input
implies the developer is DoS'ing him/herself.

Note: the change to qhash.cpp applies to moc and rcc, which are always
bootstrapped.

Change-Id: I061ab52036e40627c0703f1bf881455cbf848f43
Reviewed-by: Oswald Buddenhagen <oswald.buddenhagen@digia.com>
Reviewed-by: hjk <hjk121@nokiamail.com>
This commit is contained in:
Thiago Macieira 2014-05-19 16:53:34 -07:00 committed by The Qt Project
parent 1b031630de
commit 5283a6c87b
3 changed files with 9 additions and 4 deletions

View File

@ -222,12 +222,13 @@ uint qHash(QLatin1String key, uint seed) Q_DECL_NOTHROW
*/ */
static uint qt_create_qhash_seed() static uint qt_create_qhash_seed()
{ {
uint seed = 0;
#ifndef QT_BOOTSTRAPPED
QByteArray envSeed = qgetenv("QT_HASH_SEED"); QByteArray envSeed = qgetenv("QT_HASH_SEED");
if (!envSeed.isNull()) if (!envSeed.isNull())
return envSeed.toUInt(); return envSeed.toUInt();
uint seed = 0;
#ifdef Q_OS_UNIX #ifdef Q_OS_UNIX
int randomfd = qt_safe_open("/dev/urandom", O_RDONLY); int randomfd = qt_safe_open("/dev/urandom", O_RDONLY);
if (randomfd == -1) if (randomfd == -1)
@ -254,17 +255,16 @@ static uint qt_create_qhash_seed()
seed ^= timestamp; seed ^= timestamp;
seed ^= (timestamp >> 32); seed ^= (timestamp >> 32);
#ifndef QT_BOOTSTRAPPED
quint64 pid = QCoreApplication::applicationPid(); quint64 pid = QCoreApplication::applicationPid();
seed ^= pid; seed ^= pid;
seed ^= (pid >> 32); seed ^= (pid >> 32);
#endif // QT_BOOTSTRAPPED
quintptr seedPtr = reinterpret_cast<quintptr>(&seed); quintptr seedPtr = reinterpret_cast<quintptr>(&seed);
seed ^= seedPtr; seed ^= seedPtr;
#if QT_POINTER_SIZE == 8 #if QT_POINTER_SIZE == 8
seed ^= (seedPtr >> 32); seed ^= (seedPtr >> 32);
#endif #endif
#endif // QT_BOOTSTRAPPED
return seed; return seed;
} }

View File

@ -542,6 +542,7 @@ static void processQdocconfFile(const QString &fileName)
Generator::debugSegfault("qdoc finished!"); Generator::debugSegfault("qdoc finished!");
} }
extern Q_CORE_EXPORT QBasicAtomicInt qt_qhash_seed;
QT_END_NAMESPACE QT_END_NAMESPACE
int main(int argc, char **argv) int main(int argc, char **argv)
@ -549,6 +550,7 @@ int main(int argc, char **argv)
QT_USE_NAMESPACE QT_USE_NAMESPACE
#ifndef QT_BOOTSTRAPPED #ifndef QT_BOOTSTRAPPED
qt_qhash_seed.testAndSetRelaxed(-1, 0); // set the hash seed to 0 if it wasn't set yet
QCoreApplication app(argc, argv); QCoreApplication app(argc, argv);
#endif #endif

View File

@ -52,9 +52,12 @@
#include <qcommandlineparser.h> #include <qcommandlineparser.h>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
extern Q_CORE_EXPORT QBasicAtomicInt qt_qhash_seed;
int runUic(int argc, char *argv[]) int runUic(int argc, char *argv[])
{ {
qt_qhash_seed.testAndSetRelaxed(-1, 0); // set the hash seed to 0 if it wasn't set yet
QCoreApplication app(argc, argv); QCoreApplication app(argc, argv);
QCoreApplication::setApplicationVersion(QString::fromLatin1(QT_VERSION_STR)); QCoreApplication::setApplicationVersion(QString::fromLatin1(QT_VERSION_STR));