diff --git a/include/util/RandGen.h b/include/util/RandGen.h index 733baca..55fdeb0 100644 --- a/include/util/RandGen.h +++ b/include/util/RandGen.h @@ -5,10 +5,13 @@ // James Turk (jpt2433@rit.edu) // // Version: -// $Id: RandGen.h,v 1.1 2005/01/27 03:35:23 cozman Exp $ +// $Id: RandGen.h,v 1.2 2005/02/04 21:56:39 cozman Exp $ // // Revisions: // $Log: RandGen.h,v $ +// Revision 1.2 2005/02/04 21:56:39 cozman +// RandGen implemented/tested +// // Revision 1.1 2005/01/27 03:35:23 cozman // initial import (exceptions,types, and logging,oh my!) // @@ -20,7 +23,100 @@ namespace photon { namespace util { +// Class: RandGen +// Psuedorandom number generator class which uses Mersenne Twister. +// MT19937 is described at . +class RandGen +{ +public: + // Group: (Con/De)structors + + // Function: RandGen + // Constructor for random generator, using time as seed. + RandGen(); + + // Function: RandGen + // Constructor for random generator, allowing specific seed. + // + // Parameters: + // seedVal - Seed for random generator, a given seed will always turn out + // the same string of random numbers. + // + // See Also: + // + RandGen(unsigned long seedVal); + + // Group: General + + // Function: seed + // Reseed random generator, a given seed will always turn out same string + // of random numbers. + // + // Parameters: + // seed - Seed for random generator. + void seed(unsigned long seedVal); + + // Function: genRand + // Obtain random number in range [0,max). + // + // Parameters: + // max - Boundary for random number. + // + // Returns: + // Random number from 0 to max-1. + unsigned long genRand(unsigned long max); + + // Function: genRand + // Obtain random integer in range [min,max]. + // + // Parameters: + // min - Minimum boundary for random number. + // max - Maximum boundary for random number. + // + // Returns: + // Random number from min to max. + int genRand(int min, int max); + + // Function: genRand + // Obtain random double in range [min,max]. + // + // Parameters: + // min - Minimum boundary for random number. + // max - Maximum boundary for random number. + // + // Returns: + // Random number from min to max. + double genRand(double min, double max); + + // Function: genRandSign + // Obtain random sign, positive or negative. + // + // Returns: + // Either -1.0 or +1.0 + double genRandSign(); + + // Function: genRand01 + // Obtain random double in range [0,1). + // + // Returns: + // Random number from 0 to under 1. + double genRand01(); + +private: // utilities + unsigned long genrand_int32(); //base of all generation + +private: // constants + static const unsigned long N=624; + static const unsigned long M=397; + static const unsigned long MATRIX_A=0x9908b0dfUL;; + static const unsigned long UPPER_MASK=0x80000000UL; + static const unsigned long LOWER_MASK=0x7fffffffUL; + +private: //state data + unsigned long stateVector_[N]; + unsigned long stateVectorIndex_; +}; } } diff --git a/src/util/RandGen.cpp b/src/util/RandGen.cpp index 2ba2599..e03eb40 100644 --- a/src/util/RandGen.cpp +++ b/src/util/RandGen.cpp @@ -5,21 +5,170 @@ // James Turk (jpt2433@rit.edu) // // Version: -// $Id: RandGen.cpp,v 1.1 2005/01/27 03:35:24 cozman Exp $ +// $Id: RandGen.cpp,v 1.2 2005/02/04 21:56:40 cozman Exp $ // // Revisions: // $Log: RandGen.cpp,v $ +// Revision 1.2 2005/02/04 21:56:40 cozman +// RandGen implemented/tested +// // Revision 1.1 2005/01/27 03:35:24 cozman // initial import (exceptions,types, and logging,oh my!) // // -#include "util/RandGen.h" +// The source in this file is based on MT19937, with much of the source +// replicated from mt19937ar.c, because of this the original license +// for that file is below. +// -James +// +// +//A C-program for MT19937, with initialization improved 2002/1/26. +//Coded by Takuji Nishimura and Makoto Matsumoto. +// +//Before using, initialize the state by using init_genrand(seed) +//or init_by_array(init_key, key_length). +// +//Copyright (C) 1997 - 2002, Makoto Matsumoto and Takuji Nishimura, +//All rights reserved. +// +//Redistribution and use in source and binary forms, with or without +//modification, are permitted provided that the following conditions +//are met: +// +// 1. Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// +// 2. Redistributions in binary form must reproduce the above copyright +// notice, this list of conditions and the following disclaimer in the +// documentation and/or other materials provided with the distribution. +// +// 3. The names of its contributors may not be used to endorse or promote +// products derived from this software without specific prior written +// permission. +// +//THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +//"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +//LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +//A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR +//CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +//EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +//PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +//PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +//LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +//NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +//SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +// +// +//Any feedback is very welcome. +//http://www.math.keio.ac.jp/matumoto/emt.html +//email: matumoto@math.keio.ac.jp +#include "util/RandGen.h" +#include namespace photon { namespace util { +//static consts +const unsigned long RandGen::N; +const unsigned long RandGen::M; +const unsigned long RandGen::MATRIX_A; +const unsigned long RandGen::UPPER_MASK; +const unsigned long RandGen::LOWER_MASK; + +RandGen::RandGen() +{ + seed(static_cast(std::time(0))); +} + +RandGen::RandGen(unsigned long seedVal) +{ + seed(seedVal); +} + +//adapted directly from mt19937ar : void init_genrand(unsigned long s) +void RandGen::seed(unsigned long seedVal) +{ + stateVector_[0]= seedVal & 0xffffffffUL; + for(stateVectorIndex_=1; stateVectorIndex_> 30)) + stateVectorIndex_); + /* See Knuth TAOCP Vol2. 3rd Ed. P.106 for multiplier. */ + /* In the previous versions, MSBs of the seed affect */ + /* only MSBs of the array stateVector_[]. */ + /* 2002/01/09 modified by Makoto Matsumoto */ + stateVector_[stateVectorIndex_] &= 0xffffffffUL; + /* for >32 bit machines */ + } +} + +unsigned long RandGen::genRand(unsigned long max) +{ + return genrand_int32()%max; +} + +int RandGen::genRand(int min, int max) +{ + return min + genrand_int32()%(max-min+1); +} + +//mostly adapted from mt19937ar : double genrand_real1(void) +double RandGen::genRand(double min, double max) +{ + return min + ((genrand_int32()*(1.0/4294967295.0))*(max-min)); +} + +double RandGen::genRandSign() +{ + return genrand_int32()%2 == 0 ? -1.0 : 1.0; +} + +//adapted directly from mt19937ar : double genrand_real2(void) +double RandGen::genRand01() +{ + return genrand_int32()*(1.0/4294967296.0); +} + +//adapted directly from mt19937ar : unsigned long genrand_int32(void) +unsigned long RandGen::genrand_int32() +{ + unsigned long y; + static unsigned long mag01[2]={0x0UL, MATRIX_A}; + /* mag01[x] = x * MATRIX_A for x=0,1 */ + + if (stateVectorIndex_ >= N) { /* generate N words at one time */ + unsigned int kk; + + //Removed uninitialized check, class initializes in constructor. -James + + for (kk=0;kk> 1) ^ mag01[y & 0x1UL]; + } + for (;kk> 1) ^ + mag01[y & 0x1UL]; + } + y = (stateVector_[N-1]&UPPER_MASK)|(stateVector_[0]&LOWER_MASK); + stateVector_[N-1] = stateVector_[M-1] ^ (y >> 1) ^ mag01[y & 0x1UL]; + + stateVectorIndex_ = 0; + } + + y = stateVector_[stateVectorIndex_++]; + + /* Tempering */ + y ^= (y >> 11); + y ^= (y << 7) & 0x9d2c5680UL; + y ^= (y << 15) & 0xefc60000UL; + y ^= (y >> 18); + + return y; +} }