| 
									
										
										
										
											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. | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  | USING: combinators kernel math math.bitwise math.functions | 
					
						
							|  |  |  | math.order math.primes.erato math.primes.miller-rabin | 
					
						
							|  |  |  | math.ranges random sequences sets fry ;
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | IN: math.primes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <PRIVATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-26 14:58:46 -05:00
										 |  |  | : look-in-bitmap ( n -- ? ) >index 4999999 sieve nth ;
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-26 14:58:46 -05:00
										 |  |  | : really-prime? ( n -- ? )
 | 
					
						
							|  |  |  |     dup 5000000 < [ look-in-bitmap ] [ miller-rabin ] if ; foldable
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05: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
										 |  |  |     { | 
					
						
							|  |  |  |         { [ dup 2 < ] [ drop f ] } | 
					
						
							|  |  |  |         { [ dup even? ] [ 2 = ] } | 
					
						
							|  |  |  |         [ really-prime? ] | 
					
						
							|  |  |  |     } cond ; foldable
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : next-prime ( n -- p )
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  |     dup 2 < [ | 
					
						
							|  |  |  |         drop 2
 | 
					
						
							|  |  |  |     ] [ | 
					
						
							|  |  |  |         next-odd [ dup really-prime? ] [ 2 + ] until
 | 
					
						
							|  |  |  |     ] if ; foldable
 | 
					
						
							| 
									
										
										
										
											2007-12-26 21:59:39 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : primes-between ( low high -- seq )
 | 
					
						
							| 
									
										
										
										
											2008-12-28 05:43:13 -05:00
										 |  |  |     [ dup 3 max dup even? [ 1 + ] when ] dip
 | 
					
						
							|  |  |  |     2 <range> [ prime? ] filter
 | 
					
						
							|  |  |  |     swap 3 < [ 2 prefix ] when ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : primes-upto ( n -- seq ) 2 swap primes-between ;
 | 
					
						
							| 
									
										
										
										
											2008-02-28 00:09:29 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : coprime? ( a b -- ? ) gcd nip 1 = ; foldable
 | 
					
						
							| 
									
										
										
										
											2009-05-10 14:47:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : random-prime ( numbits -- p )
 | 
					
						
							|  |  |  |     random-bits* next-prime ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : estimated-primes ( m -- n )
 | 
					
						
							|  |  |  |     dup log / ; foldable
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: no-relative-prime n ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <PRIVATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : (find-relative-prime) ( n guess -- p )
 | 
					
						
							|  |  |  |     over 1 <= [ over no-relative-prime ] when
 | 
					
						
							|  |  |  |     dup 1 <= [ drop 3 ] when
 | 
					
						
							|  |  |  |     2dup gcd nip 1 > [ 2 + (find-relative-prime) ] [ nip ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PRIVATE>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : find-relative-prime* ( n guess -- p )
 | 
					
						
							|  |  |  |     #! find a prime relative to n with initial guess | 
					
						
							|  |  |  |     >odd (find-relative-prime) ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : find-relative-prime ( n -- p )
 | 
					
						
							|  |  |  |     dup random find-relative-prime* ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: too-few-primes n numbits ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : unique-primes ( n numbits -- seq )
 | 
					
						
							|  |  |  |     2dup 2^ estimated-primes > [ too-few-primes ] when
 | 
					
						
							|  |  |  |     2dup '[ _ random-prime ] replicate
 | 
					
						
							|  |  |  |     dup all-unique? [ 2nip ] [ drop unique-primes ] if ;
 |