//Per-thread init flag
static QThreadStorage<bool> g_srand_flag;
+//32-Bit wrapper for qrand()
+#define QRAND() ((static_cast<quint32>(qrand()) & 0xFFFF) | (static_cast<quint32>(qrand()) << 16U))
+
//Robert Jenkins' 96 bit Mix Function
-static quint32 mix_function(const quint32 x, const quint32 y, const quint32 z)
+static quint32 mix_function(quint32 a, quint32 b, quint32 c)
{
- quint32 a = x;
- quint32 b = y;
- quint32 c = z;
-
a=a-b; a=a-c; a=a^(c >> 13);
b=b-c; b=b-a; b=b^(a << 8);
c=c-a; c=c-b; c=c^(b >> 13);
static void seed_rand(void)
{
QDateTime build(MUtils::Version::lib_build_date(), MUtils::Version::lib_build_time());
- const quint32 seed = mix_function(MUtils::OS::process_id(), MUtils::OS::thread_id(), build.toMSecsSinceEpoch());
- qsrand(mix_function(clock(), time(NULL), seed));
+ const quint32 seed_0 = mix_function(MUtils::OS::process_id(), MUtils::OS::thread_id(), build.toMSecsSinceEpoch());
+ qsrand(mix_function(clock(), time(NULL), seed_0));
}
static quint32 rand_fallback(void)
{
- Q_ASSERT(RAND_MAX >= 0xFFF);
+ Q_ASSERT(RAND_MAX >= 0x7FFF);
+
if (!(g_srand_flag.hasLocalData() && g_srand_flag.localData()))
{
seed_rand();
g_srand_flag.setLocalData(true);
}
- quint32 rnd = 0x32288EA3;
- for (size_t i = 0; i < 3; i++)
+
+ quint32 rnd_val = mix_function(0x32288EA3, clock(), time(NULL));
+
+ for (size_t i = 0; i < 42; i++)
{
- rnd = (rnd << 12) ^ qrand();
+ rnd_val = mix_function(rnd_val, QRAND(), QRAND());
+ rnd_val = mix_function(QRAND(), rnd_val, QRAND());
+ rnd_val = mix_function(QRAND(), QRAND(), rnd_val);
}
- return rnd;
+
+ return rnd_val;
}
quint32 MUtils::next_rand_u32(void)
bool MUtils::remove_file(const QString &fileName)
{
QFileInfo fileInfo(fileName);
- if(!(fileInfo.exists() && fileInfo.isFile()))
- {
- return true;
- }
- for(int i = 0; i < 32; i++)
+ for(size_t round = 0; round < 13; ++round)
{
- QFile file(fileName);
- file.setPermissions(FILE_PERMISSIONS_NONE);
- if((!(fileInfo.exists() && fileInfo.isFile())) || file.remove())
+ if (round > 0)
{
- return true;
+ MUtils::OS::sleep_ms(round);
+ fileInfo.refresh();
}
- MUtils::OS::sleep_ms(1);
- fileInfo.refresh();
- }
-
- qWarning("Could not delete \"%s\"", MUTILS_UTF8(fileName));
- return false;
-}
-
-static bool remove_directory_helper(const QDir &folder)
-{
- if(!folder.exists())
- {
- return true;
- }
- const QString dirName = folder.dirName();
- if(!dirName.isEmpty())
- {
- QDir parent(folder);
- if(parent.cdUp())
+ if (fileInfo.exists())
{
- QFile::setPermissions(folder.absolutePath(), FILE_PERMISSIONS_NONE);
- if(parent.rmdir(dirName))
+ QFile file(fileName);
+ if (round > 0)
{
- return true;
+ file.setPermissions(FILE_PERMISSIONS_NONE);
}
+ file.remove();
+ fileInfo.refresh();
+ }
+ if (!fileInfo.exists())
+ {
+ return true; /*success*/
}
}
+
+ qWarning("Could not delete \"%s\"", MUTILS_UTF8(fileName));
return false;
}
bool MUtils::remove_directory(const QString &folderPath, const bool &recursive)
{
- QDir folder(folderPath);
- if(!folder.exists())
- {
- return true;
- }
+ const QDir folder(folderPath);
- if(recursive)
+ if(recursive && folder.exists())
{
const QFileInfoList entryList = folder.entryInfoList(QDir::AllEntries | QDir::NoDotAndDotDot | QDir::Hidden);
for(QFileInfoList::ConstIterator iter = entryList.constBegin(); iter != entryList.constEnd(); iter++)
{
remove_directory(iter->canonicalFilePath(), true);
}
- else if(iter->isFile())
+ else
{
remove_file(iter->canonicalFilePath());
}
}
}
- for(int i = 0; i < 32; i++)
+ for(size_t round = 0; round < 13; ++round)
{
- if(remove_directory_helper(folder))
+ if(round > 0)
{
- return true;
+ MUtils::OS::sleep_ms(round);
+ folder.refresh();
+ }
+ if (folder.exists())
+ {
+ QDir parent = folder;
+ if (parent.cdUp())
+ {
+ if (round > 0)
+ {
+ QFile::setPermissions(folder.absolutePath(), FILE_PERMISSIONS_NONE);
+ }
+ parent.rmdir(folder.dirName());
+ folder.refresh();
+ }
+ }
+ if (!folder.exists())
+ {
+ return true; /*success*/
}
- MUtils::OS::sleep_ms(1);
- folder.refresh();
}
qWarning("Could not rmdir \"%s\"", MUTILS_UTF8(folderPath));
}
///////////////////////////////////////////////////////////////////////////////
+// FP MATH SUPPORT
+///////////////////////////////////////////////////////////////////////////////
+
+MUtils::fp_parts_t MUtils::break_fp(const double value)
+{
+ fp_parts_t result = { };
+ if (_finite(value))
+ {
+ result.parts[1] = modf(value, &result.parts[0]);
+ }
+ else
+ {
+ result.parts[0] = std::numeric_limits<double>::quiet_NaN();
+ result.parts[1] = std::numeric_limits<double>::quiet_NaN();
+ }
+ return result;
+}
+
+///////////////////////////////////////////////////////////////////////////////
// SELF-TEST
///////////////////////////////////////////////////////////////////////////////