org.redisson.misc.RandomXoshiro256PlusPlus Maven / Gradle / Ivy
Show all versions of redisson Show documentation
/**
* Copyright (c) 2013-2024 Nikita Koksharov
*
* Licensed 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.
*/
/*
* To the extent possible under law, the author has dedicated all copyright
* and related and neighboring rights to this software to the public domain
* worldwide. This software is distributed without any warranty.
*
* See
*/
package org.redisson.misc;
import java.nio.ByteBuffer;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.Random;
import java.util.concurrent.atomic.AtomicStampedReference;
/**
* Implementation of Random based on the xoshiro256** RNG. No-dependencies
* Java port of the original C code,
* which is public domain. This Java port is similarly dedicated to the public
* domain.
*
* This implementation is thread-safe.
*
* @see http://xoshiro.di.unimi.it/
* @author David Blackman and Sebastiano Vigna <[email protected]> (original C code)
* @author Una Thompson <[email protected]> (Java port)
* @author Nikita Koksharov
*/
public final class RandomXoshiro256PlusPlus extends Random {
private static final long serialVersionUID = -2837799889588687855L;
public static Random create() {
byte[] seed = SecureRandom.getSeed(32);
ByteBuffer bbw = ByteBuffer.wrap(seed);
return new RandomXoshiro256PlusPlus(bbw.getLong(), bbw.getLong(), bbw.getLong(), bbw.getLong());
}
RandomXoshiro256PlusPlus(long s1, long s2, long s3, long s4) {
setState(s1, s2, s3, s4);
}
// used to "stretch" seeds into a full 256-bit state; also makes
// it safe to pass in zero as a seed
////
// what generator is used here is unimportant, as long as it's
// from a different family, but splitmix64 happens to be an
// incredibly simple high-quality generator of a completely
// different family (and is recommended by the xoshiro authors)
private static final long SPLITMIX1_MAGIC = 0x9E3779B97F4A7C15L;
private static long splitmix64v1(long x) {
return (x + SPLITMIX1_MAGIC);
}
private static long splitmix64v2(long z) {
z = (z ^ (z >> 30)) * 0xBF58476D1CE4E5B9L;
z = (z ^ (z >> 27)) * 0x94D049BB133111EBL;
return z ^ (z >> 31);
}
@Override
public void setSeed(long seed) {
// update haveNextNextGaussian flag in super
super.setSeed(seed);
int[] stamp = {0};
stateRef = new AtomicStampedReference<>(new long[4], 0);
long[] oldState = stateRef.get(stamp);
long[] state = new long[4];
long sms = splitmix64v1(seed);
state[0] = splitmix64v2(sms);
sms = splitmix64v1(sms);
state[1] = splitmix64v2(sms);
sms = splitmix64v1(sms);
state[2] = splitmix64v2(sms);
sms = splitmix64v1(sms);
state[3] = splitmix64v2(sms);
if (!stateRef.compareAndSet(oldState, state, stamp[0], 1)) {
throw new IllegalStateException();
}
}
void setState(long s0, long s1, long s2, long s4) {
if (s0 == 0 && s1 == 0 && s2 == 0 && s4 == 0)
throw new IllegalArgumentException("xoshiro256++ state cannot be all zeroes");
int[] stamp = {0};
long[] oldState = stateRef.get(stamp);
long[] state = new long[4];
state[0] = s0;
state[1] = s1;
state[2] = s2;
state[3] = s4;
if (!stateRef.compareAndSet(oldState, state, stamp[0], 1)) {
throw new IllegalStateException();
}
}
// not called, implemented instead of just throwing for completeness
@Override
protected int next(int bits) {
return (int) (nextLong() & ((1L << bits) - 1));
}
@Override
public int nextInt() {
return (int) nextLong();
}
@Override
public int nextInt(int bound) {
return (int) nextLong(bound);
}
public long nextLong(long bound) {
if (bound <= 0) throw new IllegalArgumentException("bound must be positive");
// clear sign bit for positive-only, modulo to bound
return (nextLong() & Long.MAX_VALUE) % bound;
}
@Override
public double nextDouble() {
return (nextLong() >>> 11) * 0x1.0P-53;
}
@Override
public float nextFloat() {
return (nextLong() >>> 40) * 0x1.0P-24f;
}
@Override
public boolean nextBoolean() {
return (nextLong() & 1) != 0;
}
@Override
public void nextBytes(byte[] buf) {
nextBytes(buf, 0, buf.length);
}
public void nextBytes(byte[] buf, int ofs, int len) {
if (ofs < 0) throw new ArrayIndexOutOfBoundsException("Offset "+ofs+" is negative");
if (ofs >= buf.length) throw new ArrayIndexOutOfBoundsException("Offset "+ofs+" is greater than buffer length");
if (ofs+len > buf.length) throw new ArrayIndexOutOfBoundsException("Length "+len+" with offset "+ofs+" is past end of buffer");
int j = 8;
long l = 0;
for (int i = ofs; i < ofs+len; i++) {
if (j >= 8) {
l = nextLong();
j = 0;
}
buf[i] = (byte) (l&0xFF);
l = l >>> 8L;
j++;
}
}
AtomicStampedReference stateRef;
@Override
public long nextLong() {
while (true) {
int[] stamp = {0};
long[] oldState = stateRef.get(stamp);
long[] state = Arrays.copyOf(oldState, oldState.length);
long result = Long.rotateLeft(state[0] + state[3], 23) + state[0];
long t = state[1] << 17;
state[2] ^= state[0];
state[3] ^= state[1];
state[1] ^= state[2];
state[0] ^= state[3];
state[2] ^= t;
state[3] = Long.rotateLeft(state[3], 45);
if (stateRef.compareAndSet(oldState, state, stamp[0], stamp[0]+1)) {
return result;
}
}
}
}