primes.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. #include <u.h>
  2. #include <libc.h>
  3. #include <bio.h>
  4. double big = 9.007199254740992e15;
  5. int pt[] =
  6. {
  7. 2, 3, 5, 7, 11, 13, 17, 19, 23, 29,
  8. 31, 37, 41, 43, 47, 53, 59, 61, 67, 71,
  9. 73, 79, 83, 89, 97,101,103,107,109,113,
  10. 127,131,137,139,149,151,157,163,167,173,
  11. 179,181,191,193,197,199,211,223,227,229,
  12. };
  13. double wheel[] =
  14. {
  15. 10, 2, 4, 2, 4, 6, 2, 6, 4, 2,
  16. 4, 6, 6, 2, 6, 4, 2, 6, 4, 6,
  17. 8, 4, 2, 4, 2, 4, 8, 6, 4, 6,
  18. 2, 4, 6, 2, 6, 6, 4, 2, 4, 6,
  19. 2, 6, 4, 2, 4, 2,10, 2,
  20. };
  21. uchar table[1000];
  22. uchar bittab[] =
  23. {
  24. 1, 2, 4, 8, 16, 32, 64, 128,
  25. };
  26. enum {
  27. ptsiz = nelem(pt),
  28. whsiz = nelem(wheel),
  29. tabsiz = nelem(table),
  30. tsiz8 = tabsiz*8,
  31. };
  32. void mark(double nn, long k);
  33. void
  34. usage(void)
  35. {
  36. fprint(2, "usage: %s [start [finish]]\n", argv0);
  37. exits("limits");
  38. }
  39. void
  40. ouch(void)
  41. {
  42. fprint(2, "limits exceeded\n");
  43. exits("limits");
  44. }
  45. void
  46. main(int argc, char *argv[])
  47. {
  48. int i;
  49. double k, temp, v, limit, nn;
  50. char *l;
  51. Biobuf bin;
  52. ARGBEGIN{
  53. default:
  54. usage();
  55. break;
  56. }ARGEND;
  57. nn = 0;
  58. limit = big;
  59. switch (argc) {
  60. case 0:
  61. Binit(&bin, 0, OREAD);
  62. while ((l = Brdline(&bin, '\n')) != nil) {
  63. if (*l == '\n')
  64. continue;
  65. nn = atof(l);
  66. if(nn < 0)
  67. sysfatal("negative start");
  68. break;
  69. }
  70. Bterm(&bin);
  71. break;
  72. case 2:
  73. limit = atof(argv[1]);
  74. if(limit < nn)
  75. exits(0);
  76. if(limit > big)
  77. ouch();
  78. /* fallthrough */
  79. case 1:
  80. nn = atof(argv[0]);
  81. break;
  82. default:
  83. usage();
  84. break;
  85. }
  86. if(nn < 0 || nn > big)
  87. ouch();
  88. if(nn == 0)
  89. nn = 1;
  90. if(nn < 230) {
  91. for(i=0; i<ptsiz; i++) {
  92. if(pt[i] < nn)
  93. continue;
  94. if(pt[i] > limit)
  95. exits(0);
  96. print("%d\n", pt[i]);
  97. // if(limit >= big)
  98. // exits(0);
  99. }
  100. nn = 230;
  101. }
  102. modf(nn/2, &temp);
  103. nn = 2*temp + 1;
  104. /*
  105. * clear the sieve table.
  106. */
  107. for(;;) {
  108. for(i = 0; i < tabsiz; i++)
  109. table[i] = 0;
  110. /*
  111. * run the sieve.
  112. */
  113. v = sqrt(nn+tsiz8);
  114. mark(nn, 3);
  115. mark(nn, 5);
  116. mark(nn, 7);
  117. for(i = 0, k = 11; k <= v; k += wheel[i]) {
  118. mark(nn, k);
  119. i++;
  120. if(i >= whsiz)
  121. i = 0;
  122. }
  123. /*
  124. * now get the primes from the table
  125. * and print them.
  126. */
  127. for(i = 0; i < tsiz8; i += 2) {
  128. if(table[i>>3] & bittab[i&07])
  129. continue;
  130. temp = nn + i;
  131. if(temp > limit)
  132. exits(0);
  133. print("%.0f\n", temp);
  134. // if(limit >= big)
  135. // exits(0);
  136. }
  137. nn += tsiz8;
  138. }
  139. }
  140. void
  141. mark(double nn, long k)
  142. {
  143. double t1;
  144. long j;
  145. modf(nn/k, &t1);
  146. j = k*t1 - nn;
  147. if(j < 0)
  148. j += k;
  149. for(; j < tsiz8; j += k)
  150. table[j>>3] |= bittab[j&07];
  151. }