summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-01-18 06:27:52 -0800
committerKaz Kylheku <kaz@kylheku.com>2016-01-18 06:27:52 -0800
commit4a9d763799c3f0b3c17182b44e8497e174547120 (patch)
treefbf68b87168a68846c94e35a3941b0ab815ef52e
parentcc8f11bf43842e38f0a515b8070f4a7afe9a716d (diff)
downloadtxr-4a9d763799c3f0b3c17182b44e8497e174547120.tar.gz
txr-4a9d763799c3f0b3c17182b44e8497e174547120.tar.bz2
txr-4a9d763799c3f0b3c17182b44e8497e174547120.zip
random: wrong mask width for power-of-two moduli.
This mistake causes wasteful behavior for power-of-two moduli in the random function, in both the bignum and fixnum cases. For instance, the modulus 16 is taken to be 17 bits wide. But we really want the width 16: the number of bits needed for values in the range [0, 16). The result isn't wrong, but the loop generates 17-bit random numbers, and then throws away those which equal or exceed the modulus, which is wasteful. * mpi/mpi.c (mp_is_pow_two): New function. * mpi/mpi.h (mp_is_pow_two): Declared. * rand.c (random): In bignum case, after counting bits in the modulus, subtract 1 if the modulus is a power of two. In the fixnum case, subtract 1 from the modulus and then count the bits in the reduced value. * tests/013/maze.expected: regenerate due to different prng behavior.
-rw-r--r--mpi/mpi.c5
-rw-r--r--mpi/mpi.h1
-rw-r--r--rand.c4
-rw-r--r--tests/013/maze.expected118
4 files changed, 67 insertions, 61 deletions
diff --git a/mpi/mpi.c b/mpi/mpi.c
index 24df7c30..20a48a12 100644
--- a/mpi/mpi.c
+++ b/mpi/mpi.c
@@ -2998,6 +2998,11 @@ int mp_count_bits(mp_int *mp)
/* }}} */
+int mp_is_pow_two(mp_int *mp)
+{
+ return s_mp_ispow2(mp) >= 0;
+}
+
/* {{{ mp_read_radix(mp, str, radix) */
/*
diff --git a/mpi/mpi.h b/mpi/mpi.h
index 49ccd294..fb9e47e8 100644
--- a/mpi/mpi.h
+++ b/mpi/mpi.h
@@ -223,6 +223,7 @@ int mp_unsigned_bin_size(mp_int *mp);
mp_err mp_to_unsigned_bin(mp_int *mp, unsigned char *str);
int mp_count_bits(mp_int *mp);
+int mp_is_pow_two(mp_int *mp);
#if MP_COMPAT_MACROS
#define mp_read_raw(mp, str, len) mp_read_signed_bin((mp), (str), (len))
diff --git a/rand.c b/rand.c
index a2215124..fe3b6044 100644
--- a/rand.c
+++ b/rand.c
@@ -185,7 +185,7 @@ val random(val state, val modulus)
if (bignump(modulus)) {
mp_int *m = mp(modulus);
- int bits = mp_count_bits(m);
+ int bits = mp_count_bits(m) - mp_is_pow_two(m);
int rands_needed = (bits + 32 - 1) / 32;
int msb_rand_bits = bits % 32;
rand32_t msb_rand_mask = convert(rand32_t, -1) >> (32 - msb_rand_bits);
@@ -222,10 +222,10 @@ val random(val state, val modulus)
return normalize(out);
} else if (fixnump(modulus)) {
cnum m = c_num(modulus);
- int bits = highest_bit(m);
if (m == 1) {
return zero;
} else if (m > 1) {
+ int bits = highest_bit(m - 1);
#if SIZEOF_PTR >= 8
int rands_needed = (bits + 32 - 1) / 32;
#endif
diff --git a/tests/013/maze.expected b/tests/013/maze.expected
index f0a09770..e3cbbf80 100644
--- a/tests/013/maze.expected
+++ b/tests/013/maze.expected
@@ -1,61 +1,61 @@
+ +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+
-| | | | | | |
-| | | | | | |
-+ + + + +----+----+----+----+ + + + + +----+----+----+ + +----+ +
-| | | | | | | | | | |
-| | | | | | | | | | |
-+----+ + + +----+ +----+ +----+----+ +----+----+ +----+ +----+----+ +----+
-| | | | | | | | | | |
-| | | | | | | | | | |
-+----+----+ + + +----+ +----+----+ + +----+----+----+----+----+ +----+----+ +
-| | | | | | | | | | |
-| | | | | | | | | | |
-+ + + +----+----+----+ + +----+----+ + +----+ + + + + +----+----+
-| | | | | | | | | | | | |
-| | | | | | | | | | | | |
-+ +----+----+ +----+ +----+----+----+ + + + +----+ +----+----+ + + +
-| | | | | | | | | | |
-| | | | | | | | | | |
-+----+----+ + + +----+ +----+ + +----+ + + +----+----+----+----+ + +
-| | | | | | | | | | | | |
-| | | | | | | | | | | | |
-+ +----+----+----+ +----+----+ + +----+ + + + + + +----+ + + +
-| | | | | | | | | | | | |
-| | | | | | | | | | | | |
-+ + + + +----+ + +----+----+----+----+ +----+ + + + +----+----+----+
-| | | | | | | | | |
-| | | | | | | | | |
-+ +----+----+----+----+----+----+----+----+ + +----+----+----+ +----+ +----+----+ +
-| | | | | | | |
-| | | | | | | |
-+ + + +----+----+----+----+ +----+ +----+ +----+ +----+----+----+ +----+ +
-| | | | | | | | | | | | |
-| | | | | | | | | | | | |
-+ +----+----+----+ + + +----+ +----+ +----+ + + + + +----+ + +
-| | | | | | | | | | |
-| | | | | | | | | | |
-+----+ + + +----+----+ +----+ + +----+ +----+----+----+----+----+ +----+ +
-| | | | | | | | | | | |
-| | | | | | | | | | | |
-+ + +----+ + +----+----+ +----+ + +----+ + + + +----+----+ + +
-| | | | | | | | |
-| | | | | | | | |
-+ + +----+----+----+----+----+----+ +----+----+----+----+----+----+----+ + +----+ +
-| | | | | | | | |
-| | | | | | | | |
-+ + +----+----+ +----+ +----+----+ + + + + +----+ + +----+----+ +
-| | | | | | | | | | | | | | |
-| | | | | | | | | | | | | | |
-+----+ + + +----+ +----+ + + +----+----+ +----+ +----+----+ + + +
-| | | | | | | | | | |
-| | | | | | | | | | |
-+ +----+ +----+ +----+----+ + +----+----+ +----+----+----+----+ +----+----+ +
-| | | | | | | | | |
-| | | | | | | | | |
-+----+----+ + +----+ + +----+----+ +----+----+----+----+ + +----+ + +----+
-| | | | | | | | | | | |
-| | | | | | | | | | | |
-+ + + + + +----+----+----+ +----+----+----+----+ + +----+ + + + +
-| | | | | | |
-| | | | | | |
+| | | | | | | | |
+| | | | | | | | |
++ + + + +----+ + + + +----+ + +----+----+ + + +----+ +----+
+| | | | | | | | | | | |
+| | | | | | | | | | | |
++----+----+----+----+ + +----+----+----+----+----+ +----+ + +----+ +----+----+ +
+| | | | | | | | |
+| | | | | | | | |
++ +----+ + + + +----+ +----+----+ +----+ + +----+ +----+ +----+ +
+| | | | | | | | | | | | |
+| | | | | | | | | | | | |
++ +----+----+----+----+ + +----+----+ +----+ + + + +----+----+----+ +----+
+| | | | | | | | | | |
+| | | | | | | | | | |
++----+----+ + + +----+ + +----+----+ + + +----+----+----+ + + + +
+| | | | | | | | | | | | |
+| | | | | | | | | | | | |
++ +----+ + +----+----+----+ + + + +----+ +----+ + +----+ + + +
+| | | | | | | | | | | | |
+| | | | | | | | | | | | |
++ + +----+ + +----+ +----+----+ +----+ + + +----+----+ +----+----+----+
+| | | | | | | | | | | |
+| | | | | | | | | | | |
++ +----+ +----+----+ +----+ + + + +----+ + +----+----+----+ + + +
+| | | | | | | | | | | |
+| | | | | | | | | | | |
++ + +----+ + +----+ + +----+----+----+ +----+----+ +----+ + + + +
+| | | | | | | | | | | | | |
+| | | | | | | | | | | | | |
++----+ + + +----+ +----+----+ +----+ + + + +----+----+ +----+----+ +
+| | | | | | | | | | |
+| | | | | | | | | | |
++ +----+----+----+ +----+----+ +----+ + + + +----+----+ + + +----+----+
+| | | | | | | |
+| | | | | | | |
++ +----+ +----+----+----+----+----+ +----+----+ + +----+----+----+----+----+----+ +
+| | | | | | | | | |
+| | | | | | | | | |
++ + +----+ + +----+ + +----+----+----+----+ + +----+ +----+ + +----+
+| | | | | | | | | | |
+| | | | | | | | | | |
++ +----+ +----+----+ +----+ +----+----+ + +----+----+ +----+ +----+----+ +
+| | | | | | | | | | |
+| | | | | | | | | | |
++----+----+ + + + + +----+ +----+----+----+ + + +----+----+----+ +----+
+| | | | | | | | | |
+| | | | | | | | | |
++ +----+----+ +----+----+----+----+----+----+ + +----+----+----+ +----+ + +----+
+| | | | | | | | |
+| | | | | | | | |
++----+ + + +----+ + +----+----+----+ +----+ + + +----+----+ +----+ +
+| | | | | | | | | | |
+| | | | | | | | | | |
++ +----+ +----+----+----+----+----+----+ + +----+----+----+ + +----+----+----+ +
+| | | | | | | | | | |
+| | | | | | | | | | |
++ +----+----+----+ + +----+ +----+ +----+----+ + + + + + + + +
+| | | | | | |
+| | | | | | |
+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+ +