/* $NetBSD: xoshiro128starstar.c,v 1.2 2019/01/09 16:55:14 christos Exp $ */ /* * Portions Copyright (C) Internet Systems Consortium, Inc. ("ISC") * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * See the COPYRIGHT file distributed with this work for additional * information regarding copyright ownership. */ /* * Written in 2018 by David Blackman and Sebastiano Vigna (vigna@acm.org) * * 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 . */ #include #include /* * This is xoshiro128** 1.0, our 32-bit all-purpose, rock-solid generator. * It has excellent (sub-ns) speed, a state size (128 bits) that is large * enough for mild parallelism, and it passes all tests we are aware of. * * For generating just single-precision (i.e., 32-bit) floating-point * numbers, xoshiro128+ is even faster. * * The state must be seeded so that it is not everywhere zero. */ #if defined(HAVE_TLS) #define _LOCK() {}; #define _UNLOCK() {}; #if defined(HAVE_THREAD_LOCAL) #include static thread_local uint32_t seed[4]; #elif defined(HAVE___THREAD) static __thread uint32_t seed[4]; #elif defined(HAVE___DECLSPEC_THREAD) static __declspec( thread ) uint32_t seed[4]; #else #error "Unknown method for defining a TLS variable!" #endif #else #if defined(_WIN32) || defined(_WIN64) #include static volatile HANDLE _mutex = NULL; /* * Initialize the mutex on the first lock attempt. On collision, each thread * will attempt to allocate a mutex and compare-and-swap it into place as the * global mutex. On failure to swap in the global mutex, the mutex is closed. */ #define _LOCK() \ do { \ if (!_mutex) { \ HANDLE p = CreateMutex(NULL, FALSE, NULL); \ if (InterlockedCompareExchangePointer \ ((void **)&_mutex, (void *)p, NULL)) { \ CloseHandle(p); \ } \ } \ WaitForSingleObject(_mutex, INFINITE); \ } while (0) #define _UNLOCK() ReleaseMutex(_mutex) #else /* defined(_WIN32) || defined(_WIN64) */ #include static pthread_mutex_t _mutex = PTHREAD_MUTEX_INITIALIZER; #define _LOCK() RUNTIME_CHECK(pthread_mutex_lock(&_mutex)==0) #define _UNLOCK() RUNTIME_CHECK(pthread_mutex_unlock(&_mutex)==0) #endif /* defined(_WIN32) || defined(_WIN64) */ static uint32_t seed[4]; #endif /* defined(HAVE_TLS) */ static inline uint32_t rotl(const uint32_t x, int k) { return (x << k) | (x >> (32 - k)); } static inline uint32_t next(void) { uint32_t result_starstar, t; _LOCK(); result_starstar = rotl(seed[0] * 5, 7) * 9; t = seed[1] << 9; seed[2] ^= seed[0]; seed[3] ^= seed[1]; seed[1] ^= seed[2]; seed[0] ^= seed[3]; seed[2] ^= t; seed[3] = rotl(seed[3], 11); _UNLOCK(); return (result_starstar); }