1 /* 2 * Copyright (C) 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 18 #ifndef ANALYZER_PSEUDORANDOM_H 19 #define ANALYZER_PSEUDORANDOM_H 20 21 #include <cctype> 22 23 class PseudoRandom { 24 public: 25 PseudoRandom(int64_t seed = 99887766) mSeed(seed)26 : mSeed(seed) 27 {} 28 29 /** 30 * Returns the next random double from -1.0 to 1.0 31 * 32 * @return value from -1.0 to 1.0 33 */ nextRandomDouble()34 double nextRandomDouble() { 35 return nextRandomInteger() * (0.5 / (((int32_t)1) << 30)); 36 } 37 38 /** Calculate random 32 bit number using linear-congruential method 39 * with known real-time performance. 40 */ nextRandomInteger()41 int32_t nextRandomInteger() { 42 #if __has_builtin(__builtin_mul_overflow) && __has_builtin(__builtin_add_overflow) 43 int64_t prod; 44 // Use values for 64-bit sequence from MMIX by Donald Knuth. 45 __builtin_mul_overflow(mSeed, (int64_t)6364136223846793005, &prod); 46 __builtin_add_overflow(prod, (int64_t)1442695040888963407, &mSeed); 47 #else 48 mSeed = (mSeed * (int64_t)6364136223846793005) + (int64_t)1442695040888963407; 49 #endif 50 return (int32_t) (mSeed >> 32); // The higher bits have a longer sequence. 51 } 52 53 private: 54 int64_t mSeed; 55 }; 56 57 #endif //ANALYZER_PSEUDORANDOM_H 58