blob: eb23cd96879fb1a7ab21719d1a448f088806dedd [file] [log] [blame]
/**************************************************************************
*
* random.c -- pseudo random number generator
* Copyright (C) 1994 Chris Wallace (csw@bruce.cs.monash.edu.au)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
*
**************************************************************************/
#include <sys/types.h>
/*
* modified by dxm@sgi.com so that this file acts as a drop in replacement
* for srandom and random.
*/
/*
* A random number generator called as a function by
* random (iseed) or irandm (iseed)
* The parameter should be a pointer to a 2-element int32_t vector.
* The first function returns a double uniform in 0 .. 1.
* The second returns a int32_t integer uniform in 0 .. 2**31-1
* Both update iseed[] in exactly the same way.
* iseed[] must be a 2-element integer vector.
* The initial value of the second element may be anything.
*
* The period of the random sequence is 2**32 * (2**32-1)
* The table mt[0:127] is defined by mt[i] = 69069 ** (128-i)
*/
#define MASK ((int32_t) 593970775)
/* or in hex, 23674657 */
#define SCALE ((double) 1.0 / (1024.0 * 1024.0 * 1024.0 * 2.0))
/* i.e. 2 to power -31 */
static int32_t mt [128] = {
902906369,
2030498053,
-473499623,
1640834941,
723406961,
1993558325,
-257162999,
-1627724755,
913952737,
278845029,
1327502073,
-1261253155,
981676113,
-1785280363,
1700077033,
366908557,
-1514479167,
-682799163,
141955545,
-830150595,
317871153,
1542036469,
-946413879,
-1950779155,
985397153,
626515237,
530871481,
783087261,
-1512358895,
1031357269,
-2007710807,
-1652747955,
-1867214463,
928251525,
1243003801,
-2132510467,
1874683889,
-717013323,
218254473,
-1628774995,
-2064896159,
69678053,
281568889,
-2104168611,
-165128239,
1536495125,
-39650967,
546594317,
-725987007,
1392966981,
1044706649,
687331773,
-2051306575,
1544302965,
-758494647,
-1243934099,
-75073759,
293132965,
-1935153095,
118929437,
807830417,
-1416222507,
-1550074071,
-84903219,
1355292929,
-380482555,
-1818444007,
-204797315,
170442609,
-1636797387,
868931593,
-623503571,
1711722209,
381210981,
-161547783,
-272740131,
-1450066095,
2116588437,
1100682473,
358442893,
-1529216831,
2116152005,
-776333095,
1265240893,
-482278607,
1067190005,
333444553,
86502381,
753481377,
39000101,
1779014585,
219658653,
-920253679,
2029538901,
1207761577,
-1515772851,
-236195711,
442620293,
423166617,
-1763648515,
-398436623,
-1749358155,
-538598519,
-652439379,
430550625,
-1481396507,
2093206905,
-1934691747,
-962631983,
1454463253,
-1877118871,
-291917555,
-1711673279,
201201733,
-474645415,
-96764739,
-1587365199,
1945705589,
1303896393,
1744831853,
381957665,
2135332261,
-55996615,
-1190135011,
1790562961,
-1493191723,
475559465,
69069
};
double
_random (int32_t is [2])
{
int32_t it, leh, nit;
it = is [0];
leh = is [1];
if (it <= 0)
it = (it + it) ^ MASK;
else
it = it + it;
nit = it - 1;
/* to ensure all-ones pattern omitted */
leh = leh * mt[nit & 127] + nit;
is [0] = it; is [1] = leh;
if (leh < 0) leh = ~leh;
return (SCALE * ((int32_t) (leh | 1)));
}
int32_t
_irandm (int32_t is [2])
{
int32_t it, leh, nit;
it = is [0];
leh = is [1];
if (it <= 0)
it = (it + it) ^ MASK;
else
it = it + it;
nit = it - 1;
/* to ensure all-ones pattern omitted */
leh = leh * mt[nit & 127] + nit;
is [0] = it; is [1] = leh;
if (leh < 0) leh = ~leh;
return (leh);
}
/*
* make this a drop in replacement for random and srandom
*
* XXX not thread safe I guess.
*/
static int32_t saved_seed[2];
long random(void)
{
return _irandm(saved_seed);
}
void srandom(unsigned seed)
{
saved_seed[0]=seed;
saved_seed[1]=0;
_irandm(saved_seed);
}