blob: 0df657437c81719576ad6cdec6ce96e27c001e64 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "Random.h"
#include <decaf/util/Date.h>
using namespace decaf;
using namespace decaf::util;
using namespace decaf::lang;
unsigned long long Random::multiplier = 0x5deece66dLL;
////////////////////////////////////////////////////////////////////////////////
Random::Random() {
setSeed(Date::getCurrentTimeMilliseconds());
}
////////////////////////////////////////////////////////////////////////////////
Random::Random( unsigned long long seed ) {
setSeed(seed);
}
////////////////////////////////////////////////////////////////////////////////
bool Random::nextBoolean() {
return next(1) != 0;
}
////////////////////////////////////////////////////////////////////////////////
void Random::nextBytes( std::vector<unsigned char>& buf ) {
int rand = 0;
std::size_t count = 0, loop = 0;
while (count < buf.size()) {
if (loop == 0) {
rand = nextInt();
loop = 3;
} else {
loop--;
}
buf[count++] = (unsigned char) rand;
rand >>= 8;
}
}
////////////////////////////////////////////////////////////////////////////////
double Random::nextDouble() {
// was: return ((((long long) next(26) << 27) + next(27)) / (double) (1L << 53));
long long divisor = 1LL;
divisor <<= 31;
divisor <<= 22;
return ((((long long) next(26) << 27) + next(27)) / (double) divisor);
}
////////////////////////////////////////////////////////////////////////////////
float Random::nextFloat() {
return (next(24) / 16777216.0f);
}
////////////////////////////////////////////////////////////////////////////////
double Random::nextGaussian() {
if (haveNextNextGaussian) {
// if X1 has been returned, return the second Gaussian
haveNextNextGaussian = false;
return nextNextGaussian;
}
double v1, v2, s;
do {
// Generates two independent random variables U1, U2
v1 = 2 * nextDouble() - 1;
v2 = 2 * nextDouble() - 1;
s = v1 * v1 + v2 * v2;
} while (s >= 1);
double norm = std::sqrt(-2 * std::log(s) / s);
// should that not be norm instead of multiplier ?
nextNextGaussian = v2 * norm;
haveNextNextGaussian = true;
// should that not be norm instead of multiplier ?
return v1 * norm;
}
////////////////////////////////////////////////////////////////////////////////
int Random::nextInt() {
return next(32);
}
////////////////////////////////////////////////////////////////////////////////
int Random::nextInt( int n ) throw( exceptions::IllegalArgumentException ) {
if (n > 0) {
if ((n & -n) == n) {
return (int) ((n * (long long) next(31)) >> 31);
}
int bits, val;
do {
bits = next(31);
val = bits % n;
} while (bits - val + (n - 1) < 0);
return val;
}
throw exceptions::IllegalArgumentException();
}
////////////////////////////////////////////////////////////////////////////////
long long Random::nextLong() {
return ((long long) next(32) << 32) + next(32);
}
////////////////////////////////////////////////////////////////////////////////
void Random::setSeed( unsigned long long seed ) {
// was this->seed = (seed ^ multiplier) & ((1L << 48) - 1);
unsigned long long mask = 1ULL;
mask <<= 31;
mask <<= 17;
this->seed = (seed ^ multiplier) & (mask - 1);
haveNextNextGaussian = false;
}
////////////////////////////////////////////////////////////////////////////////
int Random::next( int bits ) {
// was: seed = (seed * multiplier + 0xbL) & ((1L << 48) - 1);
long long mask = 1L;
mask <<= 31;
mask <<= 17;
seed = (seed * multiplier + 0xbL) & (mask - 1);
// was: return (int) (seed >>> (48 - bits));
return (int) (seed >> (48 - bits));
}