| 
									
										
										
										
											2009-01-07 14:57:49 -05:00
										 |  |  | ! Copyright (C) 2007-2009 Samuel Tardieu. | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  | USING: combinators combinators.short-circuit fry kernel locals | 
					
						
							|  |  |  | math math.bitwise math.functions math.order math.primes.erato | 
					
						
							| 
									
										
										
										
											2009-06-24 09:36:45 -04:00
										 |  |  | math.primes.erato.private math.primes.miller-rabin math.ranges | 
					
						
							|  |  |  | literals random sequences sets vectors ;
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | IN: math.primes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <PRIVATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 13:11:32 -05:00
										 |  |  | : look-in-bitmap ( n -- ? )
 | 
					
						
							| 
									
										
										
										
											2015-07-14 20:35:52 -04:00
										 |  |  |     integer>fixnum $[ 8,999,999 sieve ] marked-unsafe? ; inline
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-24 07:04:20 -04:00
										 |  |  | : (prime?) ( n -- ? )
 | 
					
						
							| 
									
										
										
										
											2015-07-14 20:35:52 -04:00
										 |  |  |     dup 8,999,999 <= [ look-in-bitmap ] [ miller-rabin ] if ;
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 02:02:39 -04:00
										 |  |  | : simple? ( n -- ? )
 | 
					
						
							|  |  |  |     { [ even? ] [ 3 divisor? ] [ 5 divisor? ] } 1|| ;
 | 
					
						
							| 
									
										
										
										
											2009-06-24 09:36:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-26 14:58:46 -05:00
										 |  |  | PRIVATE>
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : prime? ( n -- ? )
 | 
					
						
							| 
									
										
										
										
											2008-12-26 14:58:46 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-06-24 07:04:20 -04:00
										 |  |  |         { [ dup 7 < ] [ { 2 3 5 } member? ] } | 
					
						
							| 
									
										
										
										
											2009-06-24 09:36:45 -04:00
										 |  |  |         { [ dup simple? ] [ drop f ] } | 
					
						
							| 
									
										
										
										
											2009-06-24 07:04:20 -04:00
										 |  |  |         [ (prime?) ] | 
					
						
							| 
									
										
										
										
											2008-12-26 14:58:46 -05:00
										 |  |  |     } cond ; foldable
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : next-prime ( n -- p )
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  |     dup 2 < [ | 
					
						
							|  |  |  |         drop 2
 | 
					
						
							|  |  |  |     ] [ | 
					
						
							| 
									
										
										
										
											2009-06-24 07:04:20 -04:00
										 |  |  |         next-odd [ dup prime? ] [ 2 + ] until
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  |     ] if ; foldable
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-11 10:00:01 -04:00
										 |  |  | <PRIVATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  | : <primes-range> ( low high -- range )
 | 
					
						
							|  |  |  |     [ 3 max dup even? [ 1 + ] when ] dip 2 <range> ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-16 02:02:39 -04:00
										 |  |  | ! In order not to reallocate large vectors, we compute the upper | 
					
						
							|  |  |  | ! bound of the number of primes in a given interval. We use a | 
					
						
							|  |  |  | ! double inequality given by Pierre Dusart in | 
					
						
							|  |  |  | ! http://www.ams.org/mathscinet-getitem?mr=99d:11133 for x > | 
					
						
							|  |  |  | ! 598. Under this limit, we know that there are at most 108 | 
					
						
							|  |  |  | ! primes. | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  | : upper-pi ( x -- y )
 | 
					
						
							|  |  |  |     dup log [ / ] [ 1.2762 swap / 1 + ] bi * ceiling ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : lower-pi ( x -- y )
 | 
					
						
							|  |  |  |     dup log [ / ] [ 0.992 swap / 1 + ] bi * floor ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | :: <primes-vector> ( low high -- vector )
 | 
					
						
							|  |  |  |     high upper-pi low lower-pi - >integer
 | 
					
						
							|  |  |  |     108 10000 clamp <vector>
 | 
					
						
							|  |  |  |     low 3 < [ 2 suffix! ] when ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-11 10:00:01 -04:00
										 |  |  | : (primes-between) ( low high -- seq )
 | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  |     [ <primes-range> ] [ <primes-vector> ] 2bi
 | 
					
						
							|  |  |  |     [ '[ [ prime? ] _ push-if ] each ] keep ;
 | 
					
						
							| 
									
										
										
										
											2008-12-28 05:43:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-11 10:00:01 -04:00
										 |  |  | PRIVATE>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : primes-between ( low high -- seq )
 | 
					
						
							|  |  |  |     [ ceiling >integer ] [ floor >integer ] bi*
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         { [ 2dup > ] [ 2drop V{ } clone ] } | 
					
						
							|  |  |  |         { [ dup 2 = ] [ 2drop V{ 2 } clone ] } | 
					
						
							|  |  |  |         { [ dup 2 < ] [ 2drop V{ } clone ] } | 
					
						
							|  |  |  |         [ (primes-between) ] | 
					
						
							|  |  |  |     } cond ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  | : primes-upto ( n -- seq )
 | 
					
						
							|  |  |  |     2 swap primes-between ;
 | 
					
						
							| 
									
										
										
										
											2008-02-28 00:09:29 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  | : nprimes ( n -- seq )
 | 
					
						
							|  |  |  |     2 swap [ [ next-prime ] keep ] replicate nip ;
 | 
					
						
							| 
									
										
										
										
											2011-05-20 06:38:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-19 20:21:24 -04:00
										 |  |  | : coprime? ( a b -- ? ) simple-gcd 1 = ; foldable
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : random-prime ( numbits -- p )
 | 
					
						
							| 
									
										
										
										
											2009-08-29 15:42:15 -04:00
										 |  |  |     [ ] [ 2^ ] [ random-bits* next-prime ] tri
 | 
					
						
							|  |  |  |     2dup < [ 2drop random-prime ] [ 2nip ] if ;
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : estimated-primes ( m -- n )
 | 
					
						
							|  |  |  |     dup log / ; foldable
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: no-relative-prime n ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : find-relative-prime* ( n guess -- p )
 | 
					
						
							| 
									
										
										
										
											2015-08-13 19:13:05 -04:00
										 |  |  |     [ dup 1 <= [ no-relative-prime ] when ] | 
					
						
							| 
									
										
										
										
											2015-05-13 18:17:15 -04:00
										 |  |  |     [ >odd dup 1 <= [ drop 3 ] when ] bi*
 | 
					
						
							|  |  |  |     [ 2dup coprime? ] [ 2 + ] until nip ;
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : find-relative-prime ( n -- p )
 | 
					
						
							|  |  |  |     dup random find-relative-prime* ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: too-few-primes n numbits ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : unique-primes ( n numbits -- seq )
 | 
					
						
							| 
									
										
										
										
											2015-08-13 19:13:05 -04:00
										 |  |  |     2dup 2^ estimated-primes > [ too-few-primes ] when
 | 
					
						
							| 
									
										
										
										
											2009-06-24 07:04:20 -04:00
										 |  |  |     2dup [ random-prime ] curry replicate
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  |     dup all-unique? [ 2nip ] [ drop unique-primes ] if ;
 |