srfi-27.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. /* 54-BIT (double) IMPLEMENTATION IN C OF THE "MRG32K3A" GENERATOR
  2. ===============================================================
  3. Sebastian.Egner@philips.com, Mar-2002, in ANSI-C and Scheme 48 0.57
  4. This code is a C-implementation of Pierre L'Ecuyer's MRG32k3a generator.
  5. The code uses (double)-arithmetics, assuming that it covers the range
  6. {-2^53..2^53-1} exactly (!). The code of the generator is based on the
  7. L'Ecuyer's own implementation of the generator. Please refer to the
  8. file 'mrg32k3a.scm' for more information about the method.
  9. The method provides the following functions via the C/Scheme
  10. interface of Scheme 48 0.57 to 'mrg32k3a-b.scm':
  11. s48_value mrg32k3a_pack_state1(s48_value state);
  12. s48_value mrg32k3a_unpack_state1(s48_value state);
  13. s48_value mrg32k3a_random_range();
  14. s48_value mrg32k3a_random_integer(s48_value state, s48_value range);
  15. s48_value mrg32k3a_random_real(s48_value state);
  16. As Scheme48 FIXNUMs cannot cover the range {0..m1-1}, we break up
  17. all values x in the state into x0+x1*w, where w = 2^16 = 65536.
  18. The procedures in Scheme correct for that.
  19. compile this file with:
  20. gcc -c -I $SCHEME48 mrg32k3a-b.c
  21. history of this file:
  22. SE, 18-Mar-2002: initial version
  23. SE, 22-Mar-2002: interface changed
  24. SE, 25-Mar-2002: tested with Scheme 48 0.57 in c/srfi-27
  25. SE, 27-Mar-2002: cleaned
  26. SE, 13-May-2002: bug found by Shiro Kawai removed
  27. */
  28. #include "scheme48.h" /* $SCHEME48/c/scheme48.h */
  29. #include <sys/time.h>
  30. #ifndef NULL
  31. #define NULL 0
  32. #endif
  33. /* maximum value for random_integer: min(S48_MAX_FIXNUM_VALUE, m1) */
  34. #define m_max (((long)1 << 29) - 1)
  35. /* The Generator
  36. =============
  37. */
  38. /* moduli of the components */
  39. #define m1 4294967087.0
  40. #define m2 4294944443.0
  41. /* representation of the state in C */
  42. typedef struct {
  43. double
  44. x10, x11, x12,
  45. x20, x21, x22;
  46. } state_t;
  47. /* recursion coefficients of the components */
  48. #define a12 1403580.0
  49. #define a13n 810728.0
  50. #define a21 527612.0
  51. #define a23n 1370589.0
  52. /* normalization factor 1/(m1 + 1) */
  53. #define norm 2.328306549295728e-10
  54. /* the actual generator */
  55. static double mrg32k3a(state_t *s) { /* (double), in {0..m1-1} */
  56. double x10, x20, y;
  57. long k10, k20;
  58. /* #define debug 1 */
  59. #if defined(debug)
  60. printf(
  61. "state = {%g %g %g %g %g %g};\n",
  62. s->x10, s->x11, s->x12,
  63. s->x20, s->x21, s->x22
  64. );
  65. #endif
  66. /* component 1 */
  67. x10 = a12*(s->x11) - a13n*(s->x12);
  68. k10 = x10 / m1;
  69. x10 -= k10 * m1;
  70. if (x10 < 0.0)
  71. x10 += m1;
  72. s->x12 = s->x11;
  73. s->x11 = s->x10;
  74. s->x10 = x10;
  75. /* component 2 */
  76. x20 = a21*(s->x20) - a23n*(s->x22);
  77. k20 = x20 / m2;
  78. x20 -= k20 * m2;
  79. if (x20 < 0.0)
  80. x20 += m2;
  81. s->x22 = s->x21;
  82. s->x21 = s->x20;
  83. s->x20 = x20;
  84. /* combination of component */
  85. y = x10 - x20;
  86. if (y < 0.0)
  87. y += m1;
  88. return y;
  89. }
  90. /* Exported Interface
  91. ==================
  92. */
  93. s48_value mrg32k3a_pack_state1(s48_value state) {
  94. s48_value result;
  95. state_t s;
  96. S48_DECLARE_GC_PROTECT(1);
  97. S48_GC_PROTECT_1(state); /* s48_extract_integer may GC */
  98. #define REF(i) (double)s48_extract_integer(S48_VECTOR_REF(state, (long)(i)))
  99. /* copy the numbers from state into s */
  100. s.x10 = REF( 0) + 65536.0 * REF( 1);
  101. s.x11 = REF( 2) + 65536.0 * REF( 3);
  102. s.x12 = REF( 4) + 65536.0 * REF( 5);
  103. s.x20 = REF( 6) + 65536.0 * REF( 7);
  104. s.x21 = REF( 8) + 65536.0 * REF( 9);
  105. s.x22 = REF(10) + 65536.0 * REF(11);
  106. #undef REF
  107. S48_GC_UNPROTECT();
  108. /* box s into a Scheme object */
  109. result = S48_MAKE_VALUE(state_t);
  110. S48_SET_VALUE(result, state_t, s);
  111. return result;
  112. }
  113. s48_value mrg32k3a_unpack_state1(s48_value state) {
  114. s48_value result = S48_UNSPECIFIC;
  115. state_t s;
  116. S48_DECLARE_GC_PROTECT(1);
  117. S48_GC_PROTECT_1(result);
  118. /* unbox s from the Scheme object */
  119. s = S48_EXTRACT_VALUE(state, state_t);
  120. /* make and fill a Scheme vector with the numbers */
  121. result = s48_make_vector((long)12, S48_FALSE);
  122. #define SET(i, x) { \
  123. long x1 = (long)((x) / 65536.0); \
  124. long x0 = (long)((x) - 65536.0 * (double)x1); \
  125. S48_VECTOR_SET(result, (long)(i+0), s48_enter_integer(x0)); \
  126. S48_VECTOR_SET(result, (long)(i+1), s48_enter_integer(x1)); }
  127. SET( 0, s.x10);
  128. SET( 2, s.x11);
  129. SET( 4, s.x12);
  130. SET( 6, s.x20);
  131. SET( 8, s.x21);
  132. SET(10, s.x22);
  133. #undef SET
  134. S48_GC_UNPROTECT();
  135. return result;
  136. }
  137. s48_value mrg32k3a_random_range(void) {
  138. return s48_enter_fixnum(m_max);
  139. }
  140. s48_value mrg32k3a_random_integer(s48_value state, s48_value range) {
  141. long result;
  142. state_t s;
  143. long n;
  144. double x, q, qn, xq;
  145. s = S48_EXTRACT_VALUE(state, state_t);
  146. n = s48_extract_integer(range);
  147. if (!( ((long)1 <= n) && (n <= m_max) ))
  148. s48_raise_range_error(n, (long)1, m_max);
  149. /* generate result in {0..n-1} using the rejection method */
  150. q = (double)( (unsigned long)(m1 / (double)n) );
  151. qn = q * n;
  152. do {
  153. x = mrg32k3a(&s);
  154. } while (x >= qn);
  155. xq = x / q;
  156. /* check the range */
  157. if (!( (0.0 <= xq) && (xq < (double)m_max) ))
  158. s48_raise_range_error((long)xq, (long)0, m_max);
  159. /* return result */
  160. result = (long)xq;
  161. S48_SET_VALUE(state, state_t, s);
  162. return s48_enter_fixnum(result);
  163. }
  164. s48_value mrg32k3a_random_real(s48_value state) {
  165. state_t s;
  166. double x;
  167. s = S48_EXTRACT_VALUE(state, state_t);
  168. x = (mrg32k3a(&s) + 1.0) * norm;
  169. S48_SET_VALUE(state, state_t, s);
  170. return s48_enter_double(x);
  171. }
  172. /* Kludge for scsh */
  173. static s48_value current_time(void){
  174. struct timeval tv;
  175. gettimeofday(&tv, NULL);
  176. return s48_enter_integer(tv.tv_sec);
  177. }
  178. /* Exporting the C values to Scheme
  179. ================================
  180. */
  181. void s48_init_srfi_27(void) {
  182. S48_EXPORT_FUNCTION(mrg32k3a_pack_state1);
  183. S48_EXPORT_FUNCTION(mrg32k3a_unpack_state1);
  184. S48_EXPORT_FUNCTION(mrg32k3a_random_range);
  185. S48_EXPORT_FUNCTION(mrg32k3a_random_integer);
  186. S48_EXPORT_FUNCTION(mrg32k3a_random_real);
  187. S48_EXPORT_FUNCTION(current_time);
  188. }