# fit a polynomial to a set of points # fit -dn [-v] # where n is the degree of the polynomial implement Fit; include "sys.m"; sys: Sys; include "draw.m"; include "math.m"; maths: Math; include "bufio.m"; bufio: Bufio; include "arg.m"; Fit: module { init: fn(nil: ref Draw->Context, argv: list of string); }; MAXPTS: con 512; MAXDEG: con 16; EPS: con 0.0000005; init(nil: ref Draw->Context, argv: list of string) { sys = load Sys Sys->PATH; maths = load Math Math->PATH; if(maths == nil) fatal(sys->sprint("cannot load maths library")); bufio = load Bufio Bufio->PATH; if(bufio == nil) fatal(sys->sprint("cannot load bufio")); main(argv); } isn(r: real, n: int): int { s := r - real n; if(s < 0.0) s = -s; return s < EPS; } fact(n: int): real { f := 1.0; for(i := 1; i <= n; i++) f *= real i; return f; } comb(n: int, r: int): real { f := 1.0; for(i := 0; i < r; i++) f *= real (n-i); return f/fact(r); } matalloc(n: int): array of array of real { mat := array[n] of array of real; for(i := 0; i < n; i++) mat[i] = array[n] of real; return mat; } matsalloc(n: int): array of array of array of real { mats := array[n+1] of array of array of real; for(i := 0; i <= n; i++) mats[i] = matalloc(i); return mats; } det(mat: array of array of real, n: int, mats: array of array of array of real): real { # easy cases first if(n == 0) return 1.0; if(n == 1) return mat[0][0]; if(n == 2) return mat[0][0]*mat[1][1]-mat[0][1]*mat[1][0]; d := 0.0; s := 1; m := mats[n-1]; for(k := 0; k < n; k++){ for(i := 0; i < n-1; i++){ for(j := 0; j < n-1; j++){ if(j < k) m[i][j] = mat[i+1][j]; else m[i][j] = mat[i+1][j+1]; } } d += (real s)*mat[0][k]*det(m, n-1, mats); s = -s; } return d; } get_data(fb: ref Bufio->Iobuf) : (int, array of real, array of real) { xd := array[MAXPTS] of real; yd := array[MAXPTS] of real; n := 0; while(1){ xs := ss(bufio->fb.gett(" \t\r\n")); if(xs == nil) break; ys := ss(bufio->fb.gett(" \t\r\n")); if(ys == nil) fatal(sys->sprint("missing value")); if(n >= MAXPTS) fatal(sys->sprint("too many points")); xd[n] = real xs; yd[n] = real ys; n++; } return (n, xd, yd); } get_powers(p, n : int, xd, yd : array of real) : (real, array of real) { i,j : int; x, y, z : real; a := array[p+1] of real; b := array[p+1] of real; sxy := array[p+1] of real; sx := array[2*p+1] of real; for(i = 0; i <= p; i++) sxy[i] = 0.0; for(i = 0; i <= 2*p; i++) sx[i] = 0.0; xbar := ybar := 0.0; for(i = 0; i < n; i++){ xbar += xd[i]; ybar += yd[i]; } xbar = xbar/(real n); ybar = ybar/(real n); for(i = 0; i < n; i++){ x = xd[i]-xbar; y = yd[i]-ybar; for(j = 0; j <= p; j++) sxy[j] += y*x**j; for(j = 0; j <= 2*p; j++) sx[j] += x**j; } mats := matsalloc(p+1); mat := mats[p+1]; for(i = 0; i <= p; i++) for(j = 0; j <= p; j++) mat[i][j] = sx[i+j]; d := det(mat, p+1, mats); if(isn(d, 0)) fatal(sys->sprint("points not independent")); for(j = 0; j <= p; j++){ for(i = 0; i <= p; i++) mat[i][j] = sxy[i]; a[j] = det(mat, p+1, mats)/d; for(i = 0; i <= p; i++) mat[i][j] = sx[i+j]; } # if(verbose) # sys->print("\npt actual x actual y predicted y\n"); e := 0.0; for(i = 0; i < n; i++){ x = xd[i]-xbar; y = yd[i]-ybar; z = 0.0; for(j = 0; j <= p; j++) z += a[j]*x**j; z += ybar; e += (z-yd[i])*(z-yd[i]); # if(verbose) # sys->print("%d. %f %f %f\n", i+1, xd[i], yd[i], z); } rms_error := maths->sqrt(e/(real n)); for(i = 0; i <= p; i++) b[i] = 0.0; b[0] += ybar; for(i = 0; i <= p; i++) for(j = 0; j <= i; j++) b[j] += a[i]*comb(i, j)*(-xbar)**(i-j); # pr := 0; # sys->print("y = "); # for(i = p; i >= 0; i--){ # if(!isn(b[i], 0) || (i == 0 && pr == 0)){ # if(b[i] < 0.0){ # sys->print("-"); # b[i] = -b[i]; # } # else if(pr) # sys->print("+"); # pr = 1; # if(i == 0) # sys->print("%f", b[i]); # else{ # if(!isn(b[i], 1)) # sys->print("%f*", b[i]); # sys->print("x"); # if(i > 1) # sys->print("^%d", i); # } # } # } return (rms_error, b); } main(argv: list of string) { fb: ref Bufio->Iobuf; p := 1; arg := load Arg Arg->PATH; if(arg == nil) fatal(sys->sprint("cannot load %s: %r", Arg->PATH)); arg->init(argv); verbose := 0; while((o := arg->opt()) != 0) case o{ 'd' => p = int arg->arg(); 'v' => verbose = 1; * => fatal(sys->sprint("bad option %c", o)); } args := arg->argv(); arg = nil; if(args != nil){ s := hd args; fb = bufio->open(s, bufio->OREAD); if(fb == nil) fatal(sys->sprint("cannot open %s", s)); } else{ fb = bufio->open("/dev/cons", bufio->OREAD); if(fb == nil) fatal(sys->sprint("missing data file name")); } (n, xd, yd) := get_data(fb); if(p < 0) fatal(sys->sprint("negative power")); if(p > MAXDEG) fatal(sys->sprint("power too large")); if(n < p+1) fatal(sys->sprint("not enough points")); # use x-xbar, y-ybar to avoid overflow (error, powers) := get_powers(p, n, xd, yd); sys->print("Error %f\n", error); for(i := 0; i <= p; i++) { sys->print("%d i -> %f powers[i]\n", i, powers[i]); } } ss(s: string): string { l := len s; while(l > 0 && (s[0] == ' ' || s[0] == '\t' || s[0] == '\r' || s[0] == '\n')){ s = s[1: ]; l--; } while(l > 0 && (s[l-1] == ' ' || s[l-1] == '\t' || s[l-1] == '\r' || s[l-1] == '\n')){ s = s[0: l-1]; l--; } return s; } fatal(s: string) { sys->fprint(sys->fildes(2), "fit: %s\n", s); exit; }