2007-02-09 00:09:05 +02:00
/*
2009-01-29 01:16:49 +02:00
* Mersenne Twister PRNG algorithm
2009-01-19 17:46:40 +02:00
* Copyright ( c ) 2006 Ryan Martell
2009-01-29 01:03:17 +02:00
* Based on a C program for MT19937 , with initialization improved 2002 / 1 / 26.
* Coded by Takuji Nishimura and Makoto Matsumoto .
2007-02-09 00:09:05 +02:00
*
* This file is part of FFmpeg .
*
* FFmpeg is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2.1 of the License , or ( at your option ) any later version .
*
* FFmpeg 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
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
2008-08-31 10:39:47 +03:00
# ifndef AVUTIL_RANDOM_H
# define AVUTIL_RANDOM_H
2007-02-09 00:09:05 +02:00
# define AV_RANDOM_N 624
2009-01-19 01:07:30 +02:00
# include "avutil.h"
2009-01-19 00:50:57 +02:00
# include "common.h"
2007-02-09 00:09:05 +02:00
typedef struct {
unsigned int mt [ AV_RANDOM_N ] ; ///< the array for the state vector
2008-08-05 00:54:50 +03:00
int index ; ///< Current untempered value we use as the base.
2007-02-09 00:09:05 +02:00
} AVRandomState ;
2009-01-19 01:07:30 +02:00
# if LIBAVUTIL_VERSION_MAJOR < 50
2009-01-19 00:50:57 +02:00
attribute_deprecated void av_init_random ( unsigned int seed , AVRandomState * state ) ;
2009-01-19 01:07:30 +02:00
# endif
2009-01-19 00:50:57 +02:00
void av_random_init ( AVRandomState * state , unsigned int seed ) ; ///< To be inlined, the struct must be visible. So it does not make sense to try and keep it opaque with malloc/free-like calls.
2008-08-05 00:54:50 +03:00
void av_random_generate_untempered_numbers ( AVRandomState * state ) ; ///< Regenerate the untempered numbers (must be done every 624 iterations, or it will loop).
2007-02-09 00:09:05 +02:00
2008-07-28 18:44:00 +03:00
/**
2008-08-05 00:54:50 +03:00
* Generates a random number from the interval [ 0 , 0xffffffff ] .
2008-07-28 18:44:00 +03:00
*
2009-01-29 01:16:49 +02:00
* Please do NOT use the Mersenne Twister , it is slow . Use the random number
* generator from lfg . c / h or a simple LCG like state = state * 1664525 + 1013904223.
2008-07-28 18:44:00 +03:00
* If you still choose to use MT , expect that you will have to provide
* some evidence that it makes a difference for the case where you use it .
*/
2007-02-09 00:09:05 +02:00
static inline unsigned int av_random ( AVRandomState * state )
{
unsigned int y ;
2008-08-05 00:54:50 +03:00
// Regenerate the untempered numbers if we should...
2007-02-09 00:09:05 +02:00
if ( state - > index > = AV_RANDOM_N )
av_random_generate_untempered_numbers ( state ) ;
2008-08-05 00:54:50 +03:00
// Grab one...
2007-02-09 00:09:05 +02:00
y = state - > mt [ state - > index + + ] ;
2008-08-05 00:54:50 +03:00
/* Now temper (Mersenne Twister coefficients). The coefficients for MT19937 are.. */
2007-02-09 00:09:05 +02:00
y ^ = ( y > > 11 ) ;
y ^ = ( y < < 7 ) & 0x9d2c5680 ;
y ^ = ( y < < 15 ) & 0xefc60000 ;
y ^ = ( y > > 18 ) ;
return y ;
}
2009-01-29 01:16:49 +02:00
/** Returns a random number in the range [0-1] as double. */
2007-02-09 00:09:05 +02:00
static inline double av_random_real1 ( AVRandomState * state )
{
/* divided by 2^32-1 */
return av_random ( state ) * ( 1.0 / 4294967296.0 ) ;
}
2008-08-31 10:39:47 +03:00
# endif /* AVUTIL_RANDOM_H */