From dfc9e894966ceedf99c17a2fdcd60d284006076d Mon Sep 17 00:00:00 2001 From: Roman Nozdrin Date: Wed, 1 Nov 2023 18:19:34 +0000 Subject: [PATCH] fix(rowstorage): SplitMix64 PRNG implementation to replace stdlib MT PRNG that uses /dev/urandom guarded by spinlock --- utils/rowgroup/rowstorage.cpp | 7 ++----- utils/rowgroup/rowstorage.h | 19 ++++++++++++++++--- 2 files changed, 18 insertions(+), 8 deletions(-) diff --git a/utils/rowgroup/rowstorage.cpp b/utils/rowgroup/rowstorage.cpp index 5d997a6dca..5507889737 100644 --- a/utils/rowgroup/rowstorage.cpp +++ b/utils/rowgroup/rowstorage.cpp @@ -1495,9 +1495,6 @@ RowAggStorage::RowAggStorage(const std::string& tmpDir, RowGroup* rowGroupOut, R , fTmpDir(tmpDir) , fRowGroupOut(rowGroupOut) , fKeysRowGroup(keysRowGroup) - , fRD() - , fRandGen(fRD()) - , fRandDistr(0, 100) { char suffix[PATH_MAX]; snprintf(suffix, sizeof(suffix), "/p%u-t%p/", getpid(), this); @@ -1683,7 +1680,7 @@ void RowAggStorage::dump() break; } - int64_t totalMem = fMM->getConfigured(); + const int64_t totalMem = fMM->getConfigured(); // If the generations are allowed and there are less than half of // rowgroups in memory, then we start a new generation if (fAllowGenerations && fStorage->fLRU->size() < fStorage->fRGDatas.size() / 2 && @@ -1691,7 +1688,7 @@ void RowAggStorage::dump() { startNewGeneration(); } - else if (fAllowGenerations && freeMem < totalMem / 10 * 3 && fRandDistr(fRandGen) < 30) + else if (fAllowGenerations && freeMem < totalMem / 10 * 3 && nextRandDistib() < 30) { startNewGeneration(); } diff --git a/utils/rowgroup/rowstorage.h b/utils/rowgroup/rowstorage.h index 8b230cf1cb..50151dc613 100644 --- a/utils/rowgroup/rowstorage.h +++ b/utils/rowgroup/rowstorage.h @@ -311,6 +311,21 @@ class RowAggStorage static constexpr uint8_t INIT_INFO_HASH_SHIFT{0}; static constexpr uint16_t MAX_INMEMORY_GENS{4}; + // This is SplitMix64 implementation borrowed from here + // https://thompsonsed.co.uk/random-number-generators-for-c-performance-tested + inline uint64_t nextRandom() + { + uint64_t z = (fRandom += UINT64_C(0x9E3779B97F4A7C15)); + z = (z ^ (z >> 30)) * UINT64_C(0xBF58476D1CE4E5B9); + z = (z ^ (z >> 27)) * UINT64_C(0x94D049BB133111EB); + return z ^ (z >> 31); + } + + inline uint64_t nextRandDistib() + { + return nextRandom() % 100; + } + struct Data { RowPosHashStoragePtr fHashes; @@ -349,9 +364,7 @@ class RowAggStorage bool fInitialized{false}; rowgroup::RowGroup* fRowGroupOut; rowgroup::RowGroup* fKeysRowGroup; - std::random_device fRD; - std::mt19937 fRandGen; - std::uniform_int_distribution fRandDistr; + uint64_t fRandom = 0xc4ceb9fe1a85ec53ULL; // initial integer to set PRNG up }; } // namespace rowgroup