{- This module implements a (good) random number generator. The June 1988 (v31 #6) issue of the Communications of the ACM has an article by Pierre L'Ecuyer called, "Efficient and Portable Combined Random Number Generators". Here is the Portable Combined Generator of L'Ecuyer for 32-bit computers. It has a period of roughly 2.30584e18. Transliterator: Lennart Augustsson -} module Random(randomInts, randomDoubles) where -- Use seeds s1 in 1..2147483562 and s2 in 1..2147483398 to generate -- an infinite list of random Ints. randomInts :: Int -> Int -> [Int] randomInts s1 s2 = if 1 <= s1 && s1 <= 2147483562 then if 1 <= s2 && s2 <= 2147483398 then rands s1 s2 else error "randomInts: Bad second seed." else error "randomInts: Bad first seed." rands :: Int -> Int -> [Int] rands s1 s2 = let k = s1 `div` 53668 s1' = 40014 * (s1 - k * 53668) - k * 12211 s1'' = if s1' < 0 then s1' + 2147483563 else s1' k' = s2 `div` 52774 s2' = 40692 * (s2 - k' * 52774) - k' * 3791 s2'' = if s2' < 0 then s2' + 2147483399 else s2' z = s1'' - s2'' {- z' = if z < 1 then z + 2147483562 else z in z' : rands s1'' s2'' -} -- Use this instead; it is a little stricter and generates much better code in if z < 1 then z + 2147483562 : rands s1'' s2'' else z : rands s1'' s2'' -- For those of you who don't have fromInt fromInt = fromInteger . toInteger -- Same values for s1 and s2 as above, generates an infinite -- list of Doubles uniformly distibuted in (0,1). randomDoubles :: Int -> Int -> [Double] randomDoubles s1 s2 = map (\x -> fromInt x * 4.6566130638969828e-10) (randomInts s1 s2)