Skip to content

grumply/pure-random-pcg

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

41 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

pure-random-pcg

Implementation of the RXS-M-XS variant of the permuted-congruential-generator suite. RXS-M-XS was chosen for best browser performance.

Features

pure-random-pcg comes with a convenient set of combinators.

Generators

Simple generators.

newtype Generator a = Generator { generate :: Seed -> (Seed,a) }

A variate type-class for uniform bounded and unbounded random variate generation.

class Variate a where
  uniform :: Generator a
  uniformR :: a -> a -> Generator a

Applicative and Monadic generation

data C = Double :+ Double

randomC :: Generator C
randomC = pure (:+) <*> uniform <*> uniform

randomC' :: Generator C
randomC' = do
  d1 <- uniform
  d2 <- uniform
  return (d1 :+ d2)

Num instance

inc :: Num a => Generator a -> Generator a
inc = (1 +)

Reproducibility

PCG includes the ability to very quickly walk a Seed forwards or backwards via advance and retract.

retract n . advance n == id

main = do
  seed0 <- newSeed
  let seed1 = advance 1000 seed0
      seed2 = retract 1000 seed1
  print (seed0 == seed2) -- True
main = do
  seed0 <- newSeed
  let (seed1,_) = generate uniform seed0
      seed2 = retract 1 seed1
  print (seed0 == seed2) -- True

Sampling

Sample from a distribution uniformly at random.

names = sample ["Alice","Bob","Charlie","Dan","Elaine"]

main = do
  seed <- newSeed
  print (generate names seed)

Generate arbitrary bounded enumerables.

data Dir = North | South | East | West deriving (Bounded,Enum,Show)

main = do
  seed <- newSeed
  let dirs :: [Dir]
      dirs = list choose seed
  print $ take 10 dirs

Shuffling

Knuth Fisher-Yates shuffle.

main = do
  seed <- newSeed
  print $ shuffle [1..10] seed

Important Notes

Keep in mind that pcg is not cryptographically secure.

Performance

On an i9-9880H, this implementation achieves throughput of 2 bytes per cycle (64Gb/s for 64-bit types).

On the same machine in Chrome v93, this implementation achieves thoughput of 1Gb/s for 32-bit types.

Note that RXS-M-XS has a much smaller period than MWC8222 or Mersenne Twister, but is extremely performant. The major benefit of this implementation is the pure implementation that is compatible with GHCJS.

Thanks

This library was inspired by Max Goldstein's elm-random-pcg.

Much of the Distribution and range generation code comes from Bryan O'Sullivan's mwc-random.

About

Pseudorandom number generation for pure

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages