Line data Source code
1 : /* Copyright (C) 2000 The PARI group.
2 :
3 : This file is part of the PARI/GP package.
4 :
5 : PARI/GP is free software; you can redistribute it and/or modify it under the
6 : terms of the GNU General Public License as published by the Free Software
7 : Foundation; either version 2 of the License, or (at your option) any later
8 : version. It is distributed in the hope that it will be useful, but WITHOUT
9 : ANY WARRANTY WHATSOEVER.
10 :
11 : Check the License for details. You should have received a copy of it, along
12 : with the package; see the file 'COPYING'. If not, write to the Free Software
13 : Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
14 : /********************************************************************/
15 : /* */
16 : /* PSEUDO-RANDOM INTEGERS */
17 : /* */
18 : /********************************************************************/
19 : #include "pari.h"
20 : #include "paripriv.h"
21 : /********************************************************************/
22 : /* XORGEN (Richard P. Brent) */
23 : /* http://wwwmaths.anu.edu.au/~brent/random.html */
24 : /* (initial adaptation to PARI/GP by Randall Rathbun) */
25 : /********************************************************************/
26 : /* Adapted from xorgens.c version 3.04, Richard P. Brent, 20060628 (GPL).
27 : * 32-bit or 64-bit integer random number generator with period at
28 : * least 2**4096-1. It is assumed that "ulong" is a 32-bit or 64-bit integer */
29 :
30 : #ifdef LONG_IS_64BIT
31 : typedef ulong u64;
32 : #else
33 : typedef unsigned long long u64;
34 : static u64
35 448241 : _32to64(ulong a, ulong b) { u64 v = a; return (v<<32)|b; }
36 : static void
37 13017092 : _64to32(u64 v, ulong *a, ulong *b) { *a = v>>32; *b = v&0xFFFFFFFF; }
38 : #endif
39 : static THREAD u64 state[64];
40 : static THREAD u64 xorgen_w;
41 : static THREAD int xorgen_i;
42 : /* weyl = odd approximation to 2^64*(sqrt(5)-1)/2. */
43 : static const u64 weyl = (((u64)0x61c88646U)<<32)|((u64)0x80b583ebU);
44 :
45 : static u64
46 167151312 : block(void)
47 : {
48 167151312 : const int r = 64;
49 167151312 : const int a = 33, b = 26, c = 27, d = 29, s = 53;
50 : u64 t, v, w;
51 167151312 : xorgen_i = (xorgen_i+1)&(r-1);
52 167151312 : t = state[xorgen_i];
53 167151312 : v = state[(xorgen_i+(r-s))&(r-1)]; /* index is (i-s) mod r */
54 167151312 : t ^= t<<a; t ^= t>>b; /* (I + L^a)(I + R^b) */
55 167151312 : v ^= v<<c; v ^= v>>d; /* (I + L^c)(I + R^d) */
56 167151312 : w = t^v;
57 167151312 : return state[xorgen_i] = w; /* update circular array */
58 : }
59 :
60 : /* v > 0 */
61 : static void
62 372044 : init_xor4096i(u64 v)
63 : {
64 372044 : const int r = 64;
65 : int k;
66 :
67 24173133 : for (k = r; k > 0; k--) {/* avoid correlations for close seeds */
68 23801089 : v ^= v<<10; v ^= v>>15; /* recurrence has period 2**64-1 */
69 23801089 : v ^= v<<4; v ^= v>>13;
70 : }
71 24160623 : for (xorgen_w = v, k = 0; k < r; k++) { /* initialise circular array */
72 23788579 : v ^= v<<10; v ^= v>>15;
73 23788579 : v ^= v<<4; v ^= v>>13;
74 23788579 : state[k] = v + (xorgen_w+=weyl);
75 : }
76 : /* discard first 4*r results */
77 94530198 : for (xorgen_i = r-1, k = 4*r; k > 0; k--) (void)block();
78 371733 : }
79 :
80 316690 : void pari_init_rand(void) { init_xor4096i(1UL); }
81 :
82 : static u64
83 72993713 : rand64(void)
84 : {
85 72993713 : u64 v = block();
86 73009789 : xorgen_w += weyl; /* update Weyl generator */
87 73009789 : return v + (xorgen_w ^ (xorgen_w>>27));
88 : }
89 :
90 : /* One random number uniformly distributed in [0..2**BIL) is returned, where
91 : * BIL = 8*sizeof(ulong) = 32 or 64. */
92 : ulong
93 68733 : pari_rand(void) { return rand64(); }
94 :
95 : void
96 103647 : setrand(GEN x)
97 : {
98 103647 : const int r2 = numberof(state);
99 : long i, lx;
100 : u64 v;
101 : GEN xp;
102 103647 : if (typ(x)!=t_INT) pari_err_TYPE("setrand",x);
103 103647 : if (signe(x) <= 0) pari_err_DOMAIN("setrand","n", "<=", gen_0, x);
104 103640 : lx = lgefint(x);
105 103640 : if (lx == 3) { v = x[2]; init_xor4096i(v); return; }
106 : #ifndef LONG_IS_64BIT
107 6898 : if (lx == 4)
108 : {
109 1 : v = _32to64(*int_W(x,1),*int_W(x,0));
110 1 : init_xor4096i(v); return;
111 : }
112 : #endif
113 48285 : xp = int_LSW(x);
114 : #ifdef LONG_IS_64BIT
115 41388 : if (lx != 2 + r2+2)
116 6 : pari_err_DOMAIN("setrand", "n", "!=", strtoGENstr("getrand()"), x);
117 2689830 : for (i = 0; i < r2; i++, xp = int_nextW(xp)) state[i] = *xp;
118 41382 : xorgen_w = *xp; xp = int_nextW(xp);
119 : #else
120 6897 : if (lx != 2 + 2*r2+3)
121 1 : pari_err_DOMAIN("setrand", "n", "!=", strtoGENstr("getrand()"), x);
122 448240 : for (i = 0; i < r2; i++, xp = int_nextW(int_nextW(xp)))
123 441344 : state[i] = _32to64(*int_nextW(xp), *xp);
124 6896 : xorgen_w = _32to64(*int_nextW(xp), *xp); xp = int_nextW(int_nextW(xp));
125 : #endif
126 48278 : xorgen_i = (*xp) & 63;
127 : }
128 :
129 : GEN
130 1386507 : getrand(void)
131 : {
132 1386507 : const int r2 = numberof(state);
133 : GEN x;
134 : ulong *xp;
135 : long i;
136 1386507 : if (xorgen_i < 0) init_xor4096i(1UL);
137 :
138 : #ifdef LONG_IS_64BIT
139 1188658 : x = cgetipos(2+r2+2); xp = (ulong *) int_LSW(x);
140 77262645 : for (i = 0; i < r2; i++, xp = int_nextW(xp)) *xp = state[i];
141 1188658 : *xp = xorgen_w; xp = int_nextW(xp);
142 : #else
143 197849 : x = cgetipos(2+2*r2+3); xp = (ulong *) int_LSW(x);
144 12860185 : for (i = 0; i < r2; i++, xp = int_nextW(int_nextW(xp)))
145 12662336 : _64to32(state[i], int_nextW(xp), xp);
146 197849 : _64to32(xorgen_w, int_nextW(xp), xp); xp = int_nextW(int_nextW(xp));
147 : #endif
148 1386507 : *xp = xorgen_i? xorgen_i: 64; return x;
149 : }
150 :
151 : /* assume 0 <= k <= BITS_IN_LONG. Return uniform random 0 <= x < (1<<k) */
152 : long
153 17087546 : random_bits(long k) { return rand64() >> (64-k); }
154 :
155 : /********************************************************************/
156 : /* */
157 : /* GENERIC ROUTINES */
158 : /* */
159 : /********************************************************************/
160 :
161 : /* assume n > 0 */
162 : ulong
163 38495637 : random_Fl(ulong n)
164 : {
165 : ulong d;
166 : int shift;
167 : #ifdef LONG_IS_64BIT
168 32430403 : int SHIFT = 0;
169 : #else
170 6065234 : int SHIFT = 32;
171 : #endif
172 :
173 38495637 : if (n == 1) return 0;
174 :
175 38224044 : shift = bfffo(n); /* 2^(BIL-shift) > n >= 2^(BIL-shift-1)*/
176 : /* if N a power of 2, increment shift. No reject */
177 38224044 : if ((n << shift) == HIGHBIT) return rand64() >> (SHIFT+shift+1);
178 : for (;;) {
179 53114854 : d = rand64() >> (SHIFT+shift); /* d < 2^(64-shift) uniformly distributed */
180 : /* reject strategy: proba success = n 2^(shift-64), in [1/2, 1[ */
181 53129851 : if (d < n) return d;
182 : }
183 : }
184 :
185 : /* assume N > 0, see random_Fl() for algorithm. Make sure that 32-bit and
186 : * 64-bit architectures produce the same integers (consuming random bits
187 : * by packets of 64) */
188 : GEN
189 6381536 : randomi(GEN N)
190 : {
191 6381536 : long lx = lgefint(N);
192 : GEN x, d;
193 : int shift;
194 :
195 6381536 : if (lx == 3) return utoi( random_Fl(N[2]) );
196 :
197 196982 : shift = bfffo(*int_MSW(N));
198 : /* if N a power of 2, increment shift */
199 196982 : if (Z_ispow2(N) && ++shift == BITS_IN_LONG) { shift = 0; lx--; }
200 196982 : x = cgetipos(lx);
201 147678 : for (;;) {
202 344660 : GEN y, MSW = int_MSW(x), STOP = MSW;
203 : #ifdef LONG_IS_64BIT
204 816312 : for (d = int_LSW(x); d != STOP; d = int_nextW(d)) *d = rand64();
205 276417 : *d = rand64() >> shift;
206 : #else
207 68243 : if (!odd(lx)) STOP = int_precW(STOP);
208 : /* STOP points to where MSW would in 64-bit */
209 156907 : for (d = int_LSW(x); d != STOP; d = int_nextW(d))
210 : {
211 88664 : ulong a, b; _64to32(rand64(), &a,&b);
212 88664 : *d = b; d = int_nextW(d);
213 88664 : *d = a;
214 : }
215 : {
216 68243 : ulong a, b; _64to32(rand64() >> shift, &a,&b);
217 68243 : if (d == MSW) /* 32 bits needed */
218 37889 : *d = a;
219 : else
220 : { /* 64 bits needed */
221 30354 : *d = b; d = int_nextW(d);
222 30354 : *d = a;
223 : }
224 : }
225 : #endif
226 344660 : y = int_normalize(x, 0);
227 344660 : if (abscmpii(y, N) < 0) return y;
228 : }
229 : }
230 :
231 : GEN
232 543642 : random_F2x(long d, long vs)
233 : {
234 543642 : ulong db, dl = dvmduBIL(d,&db);
235 543642 : long i, l = 2 + dl + !!db;
236 543642 : GEN y = cgetg(l,t_VECSMALL); y[1] = vs;
237 : #ifdef LONG_IS_64BIT
238 932358 : for (i=2; i<l; i++) uel(y,i) = rand64();
239 : #else
240 77901 : for (i=2; i<l-1; i+=2)
241 : {
242 135 : u64 v = rand64();
243 135 : uel(y,i) = (ulong) v;
244 135 : uel(y,i+1) = (ulong) (v>>32);
245 : }
246 77766 : if (i<l) uel(y,i) = (ulong) rand64();
247 : #endif
248 543642 : if (db) uel(y,l-1) &= ((1UL<<db)-1UL);
249 543642 : return F2x_renormalize(y,l);
250 : }
251 :
252 : GEN
253 46 : random_zv(long n)
254 : {
255 46 : GEN y = cgetg(n+1, t_VECSMALL);
256 : long i;
257 68717 : for (i=1; i<=n; i++) uel(y,i) = pari_rand();
258 46 : return y;
259 : }
260 :
261 : GEN
262 4473 : randomr(long prec)
263 : {
264 : pari_sp av;
265 : long b;
266 : GEN x, y;
267 4473 : if (prec <= 2) return real_0_bit(0);
268 4445 : x = cgetr(prec); av = avma;
269 4445 : b = prec2nbits(prec);
270 4445 : y = randomi(int2n(b));
271 4445 : if (!signe(y)) return real_0_bit(b);
272 4445 : affir(y, x); shiftr_inplace(x, - b);
273 4445 : set_avma(av); return x;
274 : }
275 :
276 : static GEN
277 156478 : polrandom(GEN N) /* assume N!=0 */
278 : {
279 156478 : long i, d = lg(N);
280 156478 : GEN z = leading_coeff(N);
281 156478 : GEN y = cgetg(d,t_POL);
282 156478 : y[1] = evalsigne(1) | evalvarn(varn(N));
283 714532 : for (i=2; i<d; i++) gel(y,i) = genrand(z);
284 156478 : return normalizepol_lg(y,d);
285 : }
286 :
287 : GEN
288 5584152 : genrand(GEN N)
289 : {
290 : GEN z;
291 5584152 : if (!N) return utoi( random_bits(31) );
292 5583473 : switch(typ(N))
293 : {
294 737408 : case t_INT:
295 737408 : switch(signe(N))
296 : {
297 : pari_sp av;
298 : GEN d;
299 283087 : case 1:
300 283087 : return randomi(N);
301 454314 : case -1:
302 454314 : av = avma; N = addiu(N, 1); d = subui(1, shifti(N, 1));
303 454314 : return gerepileuptoint(av, addii(N, randomi(d)));
304 7 : default: pari_err_DOMAIN("random","N","=",gen_0,gen_0);
305 : }
306 7 : case t_REAL:
307 7 : return randomr(realprec(N));
308 680421 : case t_INTMOD:
309 680421 : z = cgetg(3, t_INTMOD);
310 680421 : gel(z,1) = icopy(gel(N,1));
311 680421 : gel(z,2) = randomi(gel(N,1)); return z;
312 187481 : case t_FFELT:
313 187481 : return ffrandom(N);
314 156478 : case t_POL:
315 156478 : if (signe(N)==0) return pol_0(varn(N));
316 156478 : return polrandom(N);
317 3821671 : case t_VEC:
318 3821671 : if (lg(N) == 3)
319 : {
320 3575824 : pari_sp av = avma;
321 3575824 : GEN a = gel(N,1), b = gel(N,2), d;
322 3575824 : if (typ(a) != t_INT) a = gceil(a);
323 3575824 : if (typ(b) != t_INT) b = gfloor(b);
324 3575824 : if (typ(a) != t_INT || typ(b) != t_INT) pari_err_TYPE("random", N);
325 3575824 : d = subii(b,a);
326 3575824 : if (signe(d) < 0) pari_err_TYPE("random([a,b]) (a > b)", N);
327 3575824 : return gerepileuptoint(av, addii(a, randomi(addiu(d,1))));
328 : }
329 245847 : return ellrandom(N);
330 7 : default:
331 7 : pari_err_TYPE("genrand",N);
332 : return NULL;/*LCOV_EXCL_LINE*/
333 : }
334 : }
|