// @(#)root/mathcore:$Id$ // Author: L. Moneta Tue Aug 4 2015 /********************************************************************** * * * Copyright (c) 2015 LCG ROOT Math Team, CERN/PH-SFT * * * * * **********************************************************************/ // random engines based on ROOT #ifndef ROOT_Math_MixMaxEngine #define ROOT_Math_MixMaxEngine #include #include #include #include "Math/TRandomEngine.h" // struct rng_state_st; /// forward declare generator state // typedef struct rng_state_st rng_state_t; // namespace mixmax { // template // class mixmax_engine; // } namespace ROOT { namespace Math { template class MixMaxEngineImpl; /** MixMaxEngine is a wrapper class for the MIXMAX Random number generator. MIXMAX is a matrix-recursive random number generator introduced by G. Savvidy. The real implementation of the generator, written in C, is in the mixmax.h and mixmax.cxx files. This generator code is available also at hepforge: http://mixmax.hepforge.org The MIXMAX code has been created and developed by Konstantin Savvidy and it is released under GNU Lesser General Public License v3. This wrapper class provides 3 different variants of MIXMAX according to the template para extra parameter N. The extra parameter, `SkipNumber`, is used to perform additional iterations of the generator before returning the random numbers. For example, when `SkipNumber = 2`, the generator will have two extra iterations that will be discarder. - MIXMAX with N = 240. This is a new version of the generator (version 2.0beta) described in the 2016 paper (3rd reference), with special number \f$s=487013230256099140\f$, \f$m=2^{51}+1\f$ and having a period of \f$10^{4389}\f$. - MIXMAX with N = 17, from the 2.0 beta version with \f$s=0\f$ and \f$m=2^{36}+1\f$. The period of the generator is \f$10^{294}\f$. - MIXMAX with N = 256 from the 1.0 version. The period is (for `SkipNumber=0`) \f$10^{4682}\f$. For this generator we recommend in ROOT using a default value of `SkipNumber=2, while for the previous two generators skipping is not needed. This table describes the properties of the MIXMAX generators. MIXMAX is a genuine 61 bit generator on the Galois field GF[p], where \f$p=2^{61}-1\f$ is the Mersenne prime number. The MIXMAX generators with these parameters pass all of the BigCrush tests in the [TestU01 suite](http://simul.iro.umontreal.ca/testu01/tu01.html) | Dimension | Entropy | Decorrelation Time | Iteration Time | Relaxation Time | Period q | |-----------|-------------|----------------------------------|-----------------|---------------------------------------------------|-------------------------------------| | N | \f$~h(T)\f$ | \f$\tau_0 = {1\over h(T) 2N }\f$ | t | \f$\tau ={1\over h(T) \ln {1\over \delta v_0}}\f$ | \f$\log_{10} (q)\f$ | | 256 | 194 | 0.000012 | 1 | 95.00 | 4682 (full period is not confirmed) | | 8 | 220 | 0.00028 | 1 | 1.54 | 129 | | 17 | 374 | 0.000079 | 1 | 1.92 | 294 | | 240 | 8679 | 0.00000024 | 1 | 1.17 | 4389 | The entropy \f$h(T)\f$, decorrelation time \f$\tau_0\f$ decorrelation time, relaxation time \f$\tau\f$ and period of the MIXMAX generator, expressed in units of the iteration time \f$t\f$, which is normalised to 1. Clearly \f$\tau_0~ < t ~< \tau\f$. #### The References for MIXMAX are: - G.K.Savvidy and N.G.Ter-Arutyunian, *On the Monte Carlo simulation of physical systems, J.Comput.Phys. 97, 566 (1991)*; Preprint EPI-865-16-86, Yerevan, Jan. 1986 - K.Savvidy, *The MIXMAX random number generator*, Comp. Phys. Commun. 196 (2015), pp 161–165 http://dx.doi.org/10.1016/j.cpc.2015.06.003 - K.Savvidy and G.Savvidy, *Spectrum and Entropy of C-systems MIXMAX Random Number Generator*, Chaos, Solitons & Fractals, Volume 91, (2016) pp. 33–38 http://dx.doi.org/10.1016/j.chaos.2016.05.003 @ingroup Random */ template class MixMaxEngine : public TRandomEngine { public: typedef TRandomEngine BaseType; // this should be changed for WINDOWS #ifndef __LP64__ typedef uint64_t StateInt_t; #else typedef unsigned long long StateInt_t; #endif typedef uint64_t Result_t; MixMaxEngine(uint64_t seed=1); virtual ~MixMaxEngine(); /// Get the size of the generator static int Size(); /// maximum integer that can be generated. For MIXMAX is 2^61-1 static uint64_t MaxInt(); /// minimum integer that can be generated. For MIXMAX is 0 static uint64_t MinInt(); /// set the generator seed void SetSeed(Result_t seed); // generate a random number (virtual interface) virtual double Rndm() { return Rndm_impl(); } /// generate a double random number (faster interface) inline double operator() () { return Rndm_impl(); } /// generate an array of random numbers void RndmArray (int n, double * array); /// generate a 64 bit integer number Result_t IntRndm(); /// get name of the generator static const char *Name(); protected: // protected functions used for tesing the generator /// get the state of the generator void GetState(std::vector & state) const; ///set the full initial generator state void SetState(const std::vector & state); /// Get the counter (between 0 and Size-1) int Counter() const; private: /// implementation function to generate the random number double Rndm_impl(); //rng_state_t * fRngState; // mix-max generator state //mixmax::mixmax_engine * fRng; // mixmax internal engine class MixMaxEngineImpl * fRng; // mixmax internal engine class }; typedef MixMaxEngine<240,0> MixMaxEngine240; typedef MixMaxEngine<256,2> MixMaxEngine256; typedef MixMaxEngine<17,0> MixMaxEngine17; extern template class MixMaxEngine<240,0>; extern template class MixMaxEngine<256,0>; extern template class MixMaxEngine<256,2>; extern template class MixMaxEngine<256,4>; extern template class MixMaxEngine<17,0>; extern template class MixMaxEngine<17,1>; extern template class MixMaxEngine<17,2>; } // end namespace Math } // end namespace ROOT #include "Math/MixMaxEngine.icc" #endif /* ROOT_Math_MixMaxEngine */