implement Pi; include "sys.m"; sys: Sys; include "draw.m"; include "math.m"; math: Math; log: import math; include "daytime.m"; daytime: Daytime; LBASE: con 3; # 4 BASE: con 1000; # 10000 stderr: ref Sys->FD; # spawn process for each series ? Pi: module { init: fn(nil: ref Draw->Context, argv: list of string); }; init(nil: ref Draw->Context, argv: list of string) { sys = load Sys Sys->PATH; math = load Math Math->PATH; daytime = load Daytime Daytime->PATH; stderr = sys->fildes(2); dp := 1000; argv = tl argv; if(argv != nil){ if(tl argv != nil){ picmp(hd argv, hd tl argv); exit; } dp = int hd argv; } if(dp <= 0) exit; # t1 := daytime->now(); p2 := pi2(dp+1); # t2 := daytime->now(); prpi(p2); p1 := pi1(dp+1); # t3 := daytime->now(); # sys->print("%d %d\n", t2-t1, t3-t2); if(p1 == nil && p2 == nil) fatal("too many dp: reduce dp or source base"); else if(p1 == nil) p1 = p2; else if(p2 == nil) p2 = p1; n1 := len p1; n2 := len p2; if(n1 != n2) fatal(sys->sprint("lens differ %d %d", n1, n2)); f := array[10] of { * => 0 }; for(i := 0; i < n1; i++){ if(p1[i] != p2[i]) fatal(sys->sprint("arrays differ %d/%d: %d %d", i, n1, p1[i], p2[i])); if(p1[i] < 0 || p1[i] >= BASE) fatal(sys->sprint("bad array element %d: %d", i, p1[i])); if(0){ p := p1[i]; for(j := 0; j < LBASE; j++){ f[p%10]++; p /= 10; } } } # prpi(p1); if(0){ t := 0; for(i = 0; i < 10; i++){ sys->print("%d %d\n", i, f[i]); t += f[i]; } sys->print("T %d\n", t); } } terms(dp: int, f: int, v: int): (int, int) { p := dp; t := 0; for(;;){ t = 2 + int ((real p*log(real 10)+log(real v))/log(real f)); if(!(t&1)) t++; e := int (log(real (v*(t+1)/2))/log(real 10))+1; if(dp <= p-e) break; p += e; } # sys->fprint(stderr, "dp=%d p=%d f=%d v=%d terms=%d\n", dp, p, f, v, t); if(t < f*f) k := f*f; else k = t; m := BASE*k; if(m < 0 || m < BASE || m < k || m/BASE != k || m/k != BASE) return (-1, -1); return (t, p); } prpi(p: array of int) { n := len p; sys->print("π ≅ "); m := BASE/10; sys->print("%d.%.*d", p[0]/m, LBASE-1, p[0]%m); for(i := 1; i < n; i++) sys->print("%.*d", LBASE, p[i]); sys->print("\n"); } memcmp(b1: array of byte, b2: array of byte, n: int): (int, int, int) { for(i := 0; i < n; i++) if(b1[i] != b2[i]) return (i, int b1[i], int b2[i]); return (-1, 0, 0); } picmp(f1: string, f2: string) { fd1 := sys->open(f1, Sys->OREAD); fd2 := sys->open(f2, Sys->OREAD); if(fd1 == nil || fd2 == nil) fatal(sys->sprint("cannot open %s or %s", f1, f2)); b1 := array[Sys->ATOMICIO] of byte; b2 := array[Sys->ATOMICIO] of byte; t := 0; shouldexit := 0; for(;;){ n1 := sys->read(fd1, b1, len b1); n2 := sys->read(fd2, b2, len b2); if(n1 <= 0 || n2 <= 0) return; if(shouldexit) fatal("bad picmp"); if(n1 < n2) (d, v1, v2) := memcmp(b1, b2, n1); else (d, v1, v2) = memcmp(b1, b2, n2); if(d >= 0){ if(v1 == '\n' || v2 == '\n') shouldexit = 1; else fatal(sys->sprint("%s %s differ at byte %d(%c %c)", f1, f2, t+d, v1, v2)); } t += n1; if(n1 != n2) shouldexit = 1; } } roundup(n: int, m: int): (int, int) { r := m*((n+m-1)/m); return (r, r/m); } pi1(dp: int): array of int { fs := array[2] of { 5, 239 }; vs := array[2] of { 16, 4 }; ss := array[2] of { 1, -1 }; # sys->fprint(stderr, "π1\n"); return pi(dp, fs, vs, ss); } pi2(dp: int): array of int { fs := array[3] of { 18, 57, 239 }; vs := array[3] of { 48, 32, 20 }; ss := array[3] of { 1, 1, -1 }; # sys->fprint(stderr, "π2\n"); return pi(dp, fs, vs, ss); } pi3(dp: int): array of int { fs := array[4] of { 57, 239, 682, 12943 }; vs := array[4] of { 176, 28, 48, 96 }; ss := array[4] of { 1, 1, -1, 1 }; # sys->fprint(stderr, "π3\n"); return pi(dp, fs, vs, ss); } pi(dp: int, fs: array of int, vs: array of int, ss: array of int): array of int { k := len fs; n := cn := adp := 0; (dp, n) = roundup(dp, LBASE); cdp := dp; m := array[k] of int; for(i := 0; i < k; i++){ (m[i], adp) = terms(dp+1, fs[i], vs[i]); if(m[i] < 0) return nil; if(adp > cdp) cdp = adp; } (cdp, cn) = roundup(cdp, LBASE); a := array[cn] of int; p := array[cn] of int; for(i = 0; i < cn; i++) p[i] = 0; for(i = 0; i < k; i++){ series(m[i], cn, fs[i], (vs[i]*BASE)/10, ss[i], a, p); # sys->fprint(stderr, "term %d done\n", i+1); } return p[0: n]; } series(m: int, n: int, f: int, v: int, s: int, a: array of int, p: array of int) { i, j, k, q, r, r1, r2, n0: int; v *= f; f *= f; for(j = 0; j < n; j++) a[j] = 0; a[0] = v; if(s == 1) series1(m, n, f, v, a, p); else series2(m, n, f, v, a, p); return; # following code now split n0 = 0; # reaches n when very close to m so no check needed for(i = 1; i <= m; i += 2){ r1 = r2 = 0; for(j = n0; j < n; j++){ v = a[j]+r1; q = v/f; r1 = (v-q*f)*BASE; a[j] = q; v = q+r2; q = v/i; r2 = (v-q*i)*BASE; for(k = j; q > 0; k--){ r = p[k]+s*q; if(r >= BASE){ p[k] = r-BASE; q = 1; } else if(r < 0){ p[k] = r+BASE; q = 1; } else{ p[k] = r; q = 0; } } } for(j = n0; j < n; j++){ if(a[j] == 0) n0++; else break; } s = -s; } } series1(m: int, n: int, f: int, v: int, a: array of int, p: array of int) { i, j, k, q, r, r1, r2, n0: int; n0 = 0; for(i = 1; i <= m; i += 2){ r1 = r2 = 0; for(j = n0; j < n; j++){ v = a[j]+r1; q = v/f; r1 = (v-q*f)*BASE; a[j] = q; v = q+r2; q = v/i; r2 = (v-q*i)*BASE; for(k = j; q > 0; k--){ r = p[k]+q; if(r >= BASE){ p[k] = r-BASE; q = 1; } else{ p[k] = r; q = 0; } } } for(j = n0; j < n; j++){ if(a[j] == 0) n0++; else break; } i += 2; r1 = r2 = 0; for(j = n0; j < n; j++){ v = a[j]+r1; q = v/f; r1 = (v-q*f)*BASE; a[j] = q; v = q+r2; q = v/i; r2 = (v-q*i)*BASE; for(k = j; q > 0; k--){ r = p[k]-q; if(r < 0){ p[k] = r+BASE; q = 1; } else{ p[k] = r; q = 0; } } } for(j = n0; j < n; j++){ if(a[j] == 0) n0++; else break; } } } series2(m: int, n: int, f: int, v: int, a: array of int, p: array of int) { i, j, k, q, r, r1, r2, n0: int; n0 = 0; for(i = 1; i <= m; i += 2){ r1 = r2 = 0; for(j = n0; j < n; j++){ v = a[j]+r1; q = v/f; r1 = (v-q*f)*BASE; a[j] = q; v = q+r2; q = v/i; r2 = (v-q*i)*BASE; for(k = j; q > 0; k--){ r = p[k]-q; if(r < 0){ p[k] = r+BASE; q = 1; } else{ p[k] = r; q = 0; } } } for(j = n0; j < n; j++){ if(a[j] == 0) n0++; else break; } i += 2; r1 = r2 = 0; for(j = n0; j < n; j++){ v = a[j]+r1; q = v/f; r1 = (v-q*f)*BASE; a[j] = q; v = q+r2; q = v/i; r2 = (v-q*i)*BASE; for(k = j; q > 0; k--){ r = p[k]+q; if(r >= BASE){ p[k] = r-BASE; q = 1; } else{ p[k] = r; q = 0; } } } for(j = n0; j < n; j++){ if(a[j] == 0) n0++; else break; } } } fatal(e: string) { sys->print("%s\n", e); exit; }