mersenne.c
author Darkvater
Fri, 15 Jul 2005 21:17:38 +0000
changeset 2074 f23dcde246da
parent 2073 899419c9b997
child 2075 7f0ca01392db
permissions -rw-r--r--
(svn r2584) - Fix: Game no longer asserts when you use the remove-road tool on a town building or industry.
2073
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     1
#include "stdafx.h"
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     2
#include "openttd.h"
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     3
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     4
#ifdef MERSENNE_TWISTER
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     5
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     6
// Source code for Mersenne Twister.
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     7
// A Random number generator with much higher quality random numbers.
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     8
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
     9
#define N              (624)                 // length of _mt_state vector
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    10
#define M              (397)                 // a period parameter
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    11
#define K              (0x9908B0DFU)         // a magic constant
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    12
#define hiBit(u)       ((u) & 0x80000000U)   // mask all but highest   bit of u
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    13
#define loBit(u)       ((u) & 0x00000001U)   // mask all but lowest    bit of u
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    14
#define loBits(u)      ((u) & 0x7FFFFFFFU)   // mask     the highest   bit of u
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    15
#define mixBits(u, v)  (hiBit(u)|loBits(v))  // move hi bit of u to hi bit of v
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    16
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    17
static uint32   _mt_state[N+1];     // _mt_state vector + 1 extra to not violate ANSI C
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    18
static uint32   *_mt_next;          // _mt_next random value is computed from here
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    19
static int      _mt_left = -1;      // can *_mt_next++ this many times before reloading
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    20
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    21
void SeedMT(uint32 seed)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    22
{
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    23
    register uint32 x = (seed | 1U) & 0xFFFFFFFFU, *s = _mt_state;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    24
    register int    j;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    25
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    26
    for(_mt_left=0, *s++=x, j=N; --j;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    27
        *s++ = (x*=69069U) & 0xFFFFFFFFU);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    28
 }
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    29
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    30
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    31
static uint32 ReloadMT(void)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    32
 {
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    33
    register uint32 *p0=_mt_state, *p2=_mt_state+2, *pM=_mt_state+M, s0, s1;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    34
    register int    j;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    35
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    36
    if(_mt_left < -1)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    37
        SeedMT(4357U);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    38
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    39
    _mt_left=N-1, _mt_next=_mt_state+1;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    40
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    41
    for(s0=_mt_state[0], s1=_mt_state[1], j=N-M+1; --j; s0=s1, s1=*p2++)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    42
        *p0++ = *pM++ ^ (mixBits(s0, s1) >> 1) ^ (loBit(s1) ? K : 0U);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    43
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    44
    for(pM=_mt_state, j=M; --j; s0=s1, s1=*p2++)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    45
        *p0++ = *pM++ ^ (mixBits(s0, s1) >> 1) ^ (loBit(s1) ? K : 0U);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    46
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    47
    s1=_mt_state[0], *p0 = *pM ^ (mixBits(s0, s1) >> 1) ^ (loBit(s1) ? K : 0U);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    48
    s1 ^= (s1 >> 11);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    49
    s1 ^= (s1 <<  7) & 0x9D2C5680U;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    50
    s1 ^= (s1 << 15) & 0xEFC60000U;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    51
    return(s1 ^ (s1 >> 18));
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    52
 }
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    53
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    54
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    55
uint32 RandomMT(void)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    56
{
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    57
	uint32 y;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    58
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    59
	if(--_mt_left < 0)
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    60
		return ReloadMT();
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    61
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    62
	y  = *_mt_next++;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    63
	y ^= (y >> 11);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    64
	y ^= (y <<  7) & 0x9D2C5680U;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    65
	y ^= (y << 15) & 0xEFC60000U;
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    66
	return y ^ (y >> 18);
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    67
}
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    68
#else
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    69
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    70
void SeedMT(uint32 seed) {}
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    71
899419c9b997 (svn r2583) Move OS specific code out of misc.c
ludde
parents:
diff changeset
    72
#endif