From: Karl Heuer Date: Wed, 16 Mar 1994 06:48:19 +0000 (+0000) Subject: (Frandom): Eliminate bias in random number generator. X-Git-Tag: emacs-19.34~9476 X-Git-Url: http://git.eshelyaron.com/gitweb/?a=commitdiff_plain;h=78217ef17f31062ff4ea7de57af278acfec12df3;p=emacs.git (Frandom): Eliminate bias in random number generator. --- diff --git a/src/fns.c b/src/fns.c index 43ff91d183e..555cae98afb 100644 --- a/src/fns.c +++ b/src/fns.c @@ -55,24 +55,29 @@ With argument t, set the random number seed from the current time and pid.") Lisp_Object limit; { int val; + unsigned long denominator; extern long random (); extern srandom (); extern long time (); if (EQ (limit, Qt)) srandom (getpid () + time (0)); - val = random (); - if (XTYPE (limit) == Lisp_Int && XINT (limit) != 0) + if (XTYPE (limit) == Lisp_Int && XINT (limit) > 0) { /* Try to take our random number from the higher bits of VAL, not the lower, since (says Gentzel) the low bits of `random' - are less random than the higher ones. */ - val &= 0xfffffff; /* Ensure positive. */ - val >>= 5; - if (XINT (limit) < 10000) - val >>= 6; - val %= XINT (limit); + are less random than the higher ones. We do this by using the + quotient rather than the remainder. At the high end of the RNG + it's possible to get a quotient larger than limit; discarding + these values eliminates the bias that would otherwise appear + when using a large limit. */ + denominator = (unsigned long)0x80000000 / XFASTINT (limit); + do + val = (random () & 0x7fffffff) / denominator; + while (val >= limit); } + else + val = random (); return make_number (val); }