ref: b5af480daae055412fcf901cd97d884a764cbf0d
author: Benjamin Scher Purcell <ben@cbza.org>
date: Tue Aug 25 09:41:58 EDT 2015
first commit
--- /dev/null
+++ b/fifo.c
@@ -1,0 +1,63 @@
+#include <u.h>
+#include <libc.h>
+#include "rat.h"
+#include "fifo.h"
+
+Fifo*
+mkfifo(void)
+{
+ Fifo *q;
+
+ q = malloc(sizeof(*q));
+ q->front = nil;
+ return q;
+}
+
+int
+isempty(Fifo *q)
+{
+ return q->front == nil;
+}
+
+void
+frontinsert(Fifo *q, Rat val)
+{
+ Node *n;
+
+ n = malloc(sizeof(*n));
+ if(q->front == nil)
+ q->rear = &n->link;
+ n->val = val;
+ n->link = q->front;
+ q->front = n;
+ return;
+}
+
+void
+insert(Fifo *q, Rat val)
+{
+ Node *n;
+
+ n = malloc(sizeof(*n));
+ n->val = val;
+ n->link = nil;
+ if(q->front == nil)
+ q->front = n;
+ else
+ *q->rear = n;
+ q->rear = &n->link;
+ return;
+}
+
+Rat
+delete(Fifo *q)
+{
+ Rat r;
+ Node *n;
+
+ n = q->front;
+ r = n->val;
+ q->front = n->link;
+ free(n);
+ return r;
+}
--- /dev/null
+++ b/fifo.h
@@ -1,0 +1,19 @@
+typedef struct Node Node;
+typedef struct Fifo Fifo;
+
+struct Node
+{
+ Node *link;
+ Rat val;
+};
+
+struct Fifo
+{
+ Node *front;
+ Node **rear;
+};
+
+Fifo *mkfifo(void);
+void insert(Fifo*, Rat);
+void frontinsert(Fifo*, Rat);
+Rat delete(Fifo*);
--- /dev/null
+++ b/mkfile
@@ -1,0 +1,8 @@
+</$objtype/mkfile
+
+TARG= squint
+OFILES= squint.$O fifo.$O rat.$O
+HFILES= fifo.h rat.h
+BIN= .
+
+</sys/src/cmd/mkone
--- /dev/null
+++ b/rat.c
@@ -1,0 +1,111 @@
+#include <u.h>
+#include <libc.h>
+#include "rat.h"
+
+static int
+gcd(vlong a, vlong b)
+{
+ vlong t;
+
+ while(b != 0){
+ t = b;
+ b = a % b;
+ a = t;
+ }
+ return abs(a);
+}
+
+Rat
+ratmk(vlong n, vlong d)
+{
+ Rat r;
+ vlong g;
+
+ g = gcd(n, d);
+ if(d > 0){
+ r.num = n/g;
+ r.den = d/g;
+ }else{
+ r.num = -n/g;
+ r.den = -d/g;
+ }
+ return r;
+}
+
+Rat
+ratrecip(Rat a)
+{
+ vlong t;
+
+ t = a.num;
+ a.num = a.den;
+ a.den = t;
+ if(a.den < 0){
+ a.num = -a.num;
+ a.den = -a.den;
+ }
+ return a;
+}
+
+Rat
+ratneg(Rat a)
+{
+ a.num = -a.num;
+ return a;
+}
+
+Rat
+ratadd(Rat a, Rat b)
+{
+ vlong aden, bden, g;
+
+ g = gcd(a.den, b.den);
+ aden = a.den / g;
+ bden = b.den / g;
+ return ratmk(a.num*bden + b.num*aden, a.den*bden);
+}
+
+Rat
+ratsub(Rat r, Rat s)
+{
+ return ratadd(r, ratneg(s));
+}
+
+Rat
+ratmul(Rat r, Rat s)
+{
+ Rat t;
+ vlong g, h;
+
+ g = gcd(r.num, s.den);
+ h = gcd(s.num, r.den);
+ t.num = (r.num/g) * (s.num/h);
+ t.den = (r.den/h) * (s.den/g);
+ return t;
+}
+
+Rat
+ratpow(Rat r, int e)
+{
+ r.num = pow(r.num, e);
+ r.den = pow(r.den, e);
+ return r;
+}
+
+int
+Rfmt(Fmt *f)
+{
+ Rat r;
+
+ r = va_arg(f->args, Rat);
+ if(r.den == 1)
+ return fmtprint(f, "%lld", r.num);
+ else
+ return fmtprint(f, "%lld/%lld", r.num, r.den);
+}
+
+void
+ratfmtinstall(void)
+{
+ fmtinstall('R', Rfmt);
+}
--- /dev/null
+++ b/rat.h
@@ -1,0 +1,17 @@
+typedef struct Rat Rat;
+struct Rat
+{
+ vlong num;
+ vlong den;
+};
+
+#pragma varargck type "R" Rat
+
+Rat ratmk(vlong, vlong);
+Rat ratrecip(Rat);
+Rat ratneg(Rat);
+Rat ratadd(Rat, Rat);
+Rat ratsub(Rat, Rat);
+Rat ratmul(Rat, Rat);
+Rat ratpow(Rat, int);
+void ratfmtinstall(void);
--- /dev/null
+++ b/readme
@@ -1,0 +1,24 @@
+Squinting at Power Series
+=========================
+
+This code is a translation of the code in
+[Squinting at Power Series](http://swtch.com/~rsc/thread/squint.pdf)
+into C using libthread.
+
+There are a few differences:
+
+* The paper uses a queue of threads to split a series. This code uses
+two threads each with a queue data structure.
+* The paper uses queues of threads to multiply two series. This code
+maintains two queues of values in one thread.
+* The code for power series reversion is based on the simpler formula
+given in McIlroy's later paper [Power Series, Power
+Serious](http://www.cs.dartmouth.edu/~doug/pearl.ps.gz).
+* Newsqueak is presumably garbage collected. In this implementation
+each power series has an extra channel `ps->close` which is
+used to signal to the thread providing the values for the series to
+free all its data structures and exit. The handling needed for that
+channel likely doubles the size of the code.
+
+I have implementations of the original method of handling
+multiplication and splitting, email me if you are interested.
--- /dev/null
+++ b/squint.c
@@ -1,0 +1,873 @@
+#include <u.h>
+#include <libc.h>
+#include <thread.h>
+#include "rat.h"
+#include "fifo.h"
+
+int debug;
+int threadcount;
+
+enum
+{
+ TMAX = 1000,
+ STK = 2048
+};
+
+typedef struct Ps Ps;
+struct Ps
+{
+ Channel *close;
+ Channel *req;
+ Channel *dat;
+};
+
+Rat
+getr(Ps *f)
+{
+ Rat r;
+
+ send(f->req, nil);
+ recv(f->dat, &r);
+ return r;
+}
+
+Ps*
+psmk(int nel)
+{
+ Ps *d;
+
+ d = malloc(sizeof(*d));
+ d->close = chancreate(1, 0);
+ d->req = chancreate(1, 0);
+ d->dat = chancreate(sizeof(Rat), nel);
+ return d;
+}
+
+void
+psfree(Ps *d)
+{
+ chanfree(d->close);
+ chanfree(d->req);
+ chanfree(d->dat);
+ free(d);
+}
+
+static void
+_split(void *a)
+{
+ void **argv;
+ Ps *f, *s;
+ Channel *qchan, *close;
+ Fifo q;
+ Rat r;
+ enum{
+ REQ,
+ QCHAN,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ s = argv[1];
+ qchan = argv[2];
+ close = argv[3];
+
+ alts[REQ].c = s->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[QCHAN].c = qchan;
+ alts[QCHAN].v = &r;
+ alts[QCHAN].op = CHANRCV;
+ alts[CLOSE].c = s->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ q.front = nil;
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ if(q.front == nil){
+ r = getr(f);
+ send(qchan, &r);
+ send(s->dat, &r);
+ }else{
+ r = delete(&q);
+ send(s->dat, &r);
+ }
+ break;
+ case QCHAN:
+ insert(&q, r);
+ break;
+ case CLOSE:
+ while(q.front != nil)
+ delete(&q);
+ psfree(s);
+ if(chanclosing(qchan) == -1){
+ if(close != nil){
+ send(close, nil);
+ chanfree(close);
+ }
+ chanclose(qchan);
+ }else{
+ chanfree(qchan);
+ send(f->close, nil);
+ }
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+void
+split(Ps *f, Ps *s[2], Channel *splitclose)
+{
+ void *argv[2][4];
+ Channel *q;
+ int i;
+
+ q = chancreate(sizeof(Rat), 0);
+ for(i = 0; i < 2; i++){
+ s[i] = psmk(0);
+ argv[i][0] = f;
+ argv[i][1] = s[i];
+ argv[i][2] = q;
+ argv[i][3] = splitclose;
+ threadcreate(_split, argv[i], STK);
+ if(debug) threadcount++;
+ yield();
+ }
+}
+
+static void
+_mkconst(void *a)
+{
+ void **argv;
+ Ps *o;
+ Rat c;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ c = *(Rat*)argv[0];
+ o = argv[1];
+
+ alts[REQ].c = o->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = o->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ send(o->dat, &c);
+ break;
+ case CLOSE:
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+Ps*
+mkconst(Rat c)
+{
+ void *argv[2];
+ Ps *o;
+
+ o = psmk(0);
+ argv[0] = &c;
+ argv[1] = o;
+ threadcreate(_mkconst, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return o;
+}
+
+Ps*
+binop(void (*oper)(void*), Ps *f, Ps *g)
+{
+ Ps *argv[3], *o;
+
+ o = psmk(0);
+ argv[0] = f;
+ argv[1] = g;
+ argv[2] = o;
+ threadcreate(oper, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return o;
+}
+
+static void
+_psadd(void *a)
+{
+ Ps **argv, *f, *g, *s;
+ Rat r;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ g = argv[1];
+ s = argv[2];
+
+ alts[REQ].c = s->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = s->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ r = ratadd(getr(f), getr(g));
+ send(s->dat, &r);
+ break;
+ case CLOSE:
+ psfree(s);
+ send(f->close, nil);
+ send(g->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+Ps*
+psadd(Ps *f, Ps *g)
+{
+ return binop(_psadd, f, g);
+}
+
+static void
+_psderiv(void *a)
+{
+ Ps **argv, *f, *d;
+ Rat t, r;
+ int i;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ d = argv[1];
+
+ alts[REQ].c = d->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = d->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ getr(f);
+ for(i = 1;; i++){
+ switch(alt(alts)){
+ case REQ:
+ r = getr(f);
+ t = ratmk(i * r.num, r.den);
+ send(d->dat, &t);
+ break;
+ case CLOSE:
+ psfree(d);
+ send(f->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+Ps*
+psderiv(Ps *f)
+{
+ Ps *argv[2], *d = psmk(0);
+
+ argv[0] = f;
+ argv[1] = d;
+ threadcreate(_psderiv, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return d;
+}
+
+void
+psprint(Ps *ps, int n)
+{
+ int i;
+
+ for(i = 0; i < n - 1; i++)
+ print("%R ", getr(ps));
+ print("%R\n", getr(ps));
+
+}
+
+static void
+_psinteg(void *a)
+{
+ void **argv = a;
+ Ps *f, *i;
+ Rat c, out;
+ int j;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ f = argv[0];
+ c = *(Rat*)argv[1];
+ i = argv[2];
+
+ alts[REQ].c = i->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = i->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ switch(alt(alts)){
+ case REQ:
+ send(i->dat, &c);
+ break;
+ case CLOSE:
+ goto End;
+ }
+ for(j = 1;; j++){
+ switch(alt(alts)){
+ case REQ:
+ out = ratmul(getr(f), ratmk(1, j));
+ send(i->dat, &out);
+ break;
+ case CLOSE:
+ goto End;
+ }
+ }
+End:
+ psfree(i);
+ send(f->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+}
+
+
+Ps*
+psinteg(Ps *ps, Rat c)
+{
+ void *argv[3];
+ Ps *i = psmk(0);
+
+ argv[0] = ps;
+ argv[1] = &c;
+ argv[2] = i;
+ threadcreate(_psinteg, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return i;
+}
+
+static void
+_pscmul(void *a)
+{
+ void **argv = a;
+ Ps *f, *o;
+ Rat c, p;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ f = argv[0];
+ c = *(Rat*)argv[1];
+ o = argv[2];
+
+ alts[REQ].c = o->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = o->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ p = ratmul(c, getr(f));
+ send(o->dat, &p);
+ break;
+ case CLOSE:
+ psfree(o);
+ send(f->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+Ps*
+pscmul(Rat c, Ps* f)
+{
+ void *argv[3];
+ Ps *o = psmk(0);
+
+ argv[0] = f;
+ argv[1] = &c;
+ argv[2] = o;
+ threadcreate(_pscmul, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return o;
+}
+
+static void
+_psmul(void *a)
+{
+ Ps **argv, *f, *g, *p;
+ Fifo fq, gq;
+ Node *fnode, *gnode;
+ Rat sum;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ g = argv[1];
+ p = argv[2];
+
+ alts[REQ].c = p->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = p->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ fq.front = gq.front = nil;
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ insert(&fq, getr(f));
+ frontinsert(&gq, getr(g));
+ fnode = fq.front;
+ gnode = gq.front;
+ sum = ratmk(0, 1);
+ while(fnode != nil){
+ sum = ratadd(sum, ratmul(fnode->val, gnode->val));
+ fnode = fnode->link;
+ gnode = gnode->link;
+ }
+ send(p->dat, &sum);
+ break;
+ case CLOSE:
+ psfree(p);
+ while(fq.front != nil)
+ delete(&fq);
+ while(gq.front != nil)
+ delete(&gq);
+ send(f->close, nil);
+ send(g->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+Ps*
+psmul(Ps *f, Ps *g)
+{
+ return binop(_psmul, f, g);
+}
+
+static void
+_psrecip(void *a)
+{
+ void **argv;
+ Ps *f, *r, *rr, *recip;
+ Channel *close;
+ Rat g;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ r = argv[1];
+ rr = argv[2];
+ close = argv[3];
+
+ alts[REQ].c = r->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ switch(alt(alts)){
+ case REQ:
+ g = ratrecip(getr(f));
+ send(r->dat, &g);
+ break;
+ case CLOSE:
+ send(rr->close, nil);
+ send(f->close, nil);
+ goto End;
+ }
+ recip = pscmul(ratneg(g), psmul(f, rr));
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ g = getr(recip);
+ send(r->dat, &g);
+ break;
+ case CLOSE:
+ send(recip->close, nil);
+ goto End;
+ }
+ }
+End:
+ recv(r->close, nil);
+ psfree(r);
+ if(debug) threadcount--;
+ threadexits(0);
+}
+
+Ps*
+psrecip(Ps *f)
+{
+ void *argv[4];
+ Ps *rr[2], *r = psmk(0);
+ Channel *close = chancreate(1, 0);
+
+ split(r, rr, close);
+ argv[0] = f;
+ argv[1] = r;
+ argv[2] = rr[0];
+ argv[3] = close;
+ threadcreate(_psrecip, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return rr[1];
+}
+
+static void
+_psmsubst(void *a)
+{
+ void **argv;
+ Ps *f, *m;
+ Rat c, zero, r;
+ int deg, i, j;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ c = *(Rat*)argv[1];
+ deg = *(int*)argv[2];
+ m = argv[3];
+
+ alts[REQ].c = m->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = m->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ zero = ratmk(0, 1);
+ for(i = 0;; i++){
+ switch(alt(alts)){
+ case REQ:
+ r = ratmul(getr(f), ratpow(c, i));
+ send(m->dat, &r);
+ for(j = 0; j < deg - 1; j++){
+ switch(alt(alts)){
+ case REQ:
+ send(m->dat, &zero);
+ break;
+ case CLOSE:
+ goto End;
+ }
+ }
+ break;
+ case CLOSE:
+ goto End;
+ }
+ }
+End:
+ psfree(m);
+ send(f->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+}
+
+Ps*
+psmsubst(Ps *f, Rat c, int deg)
+{
+ void *argv[4];
+ Ps *m;
+
+ m = psmk(0);
+ argv[0] = f;
+ argv[1] = &c;
+ argv[2] = °
+ argv[3] = m;
+ threadcreate(_psmsubst, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return m;
+}
+
+Ps *pssubst(Ps*, Ps*);
+
+static void
+_pssubst(void *a)
+{
+ Ps **argv, *gg[2], *f, *g, *o, *subst;
+ Rat r;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ g = argv[1];
+ o = argv[2];
+
+ alts[REQ].c = o->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = o->close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ switch(alt(alts)){
+ case REQ:
+ r = getr(f);
+ send(o->dat, &r);
+ break;
+ case CLOSE:
+ psfree(o);
+ send(f->close, nil);
+ send(g->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ split(g, gg, nil);
+ getr(gg[0]);
+ subst = psmul(gg[0], pssubst(f, gg[1]));
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ r = getr(subst);
+ send(o->dat, &r);
+ break;
+ case CLOSE:
+ psfree(o);
+ send(subst->close, nil);
+ if(debug) threadcount--;
+ threadexits(0);
+ }
+ }
+}
+
+Ps*
+pssubst(Ps *f, Ps *g)
+{
+ return binop(_pssubst, f, g);
+}
+
+static void
+_psrev(void *a)
+{
+ void **argv;
+ Ps *f, *r, *rr, *rever;
+ Channel *close;
+ Rat g;
+ enum{
+ REQ,
+ CLOSE,
+ NALT
+ };
+ Alt alts[NALT+1];
+
+ argv = a;
+ f = argv[0];
+ r = argv[1];
+ rr = argv[2];
+ close = argv[3];
+
+ alts[REQ].c = r->req;
+ alts[REQ].v = nil;
+ alts[REQ].op = CHANRCV;
+ alts[CLOSE].c = close;
+ alts[CLOSE].v = nil;
+ alts[CLOSE].op = CHANRCV;
+ alts[NALT].op = CHANEND;
+
+ g = ratmk(0, 1);
+ switch(alt(alts)){
+ case REQ:
+ send(r->dat, &g);
+ break;
+ case CLOSE:
+ send(rr->close, nil);
+ send(f->close, nil);
+ goto End;
+ }
+ getr(f);
+ rever = psrecip(pssubst(f, rr));
+ for(;;){
+ switch(alt(alts)){
+ case REQ:
+ g = getr(rever);
+ send(r->dat, &g);
+ break;
+ case CLOSE:
+ send(rever->close, nil);
+ goto End;
+ }
+ }
+End:
+ recv(r->close, nil);
+ psfree(r);
+ if(debug) threadcount--;
+ threadexits(0);
+}
+
+Ps*
+psrev(Ps *f)
+{
+ void *argv[4];
+ Ps *rr[2], *r;
+ Channel *close = chancreate(1,0);
+
+ r = psmk(0);
+ split(r, rr, close);
+ argv[0] = f;
+ argv[1] = r;
+ argv[2] = rr[0];
+ argv[3] = close;
+ threadcreate(_psrev, argv, STK);
+ if(debug) threadcount++;
+ yield();
+ return rr[1];
+}
+
+void
+threadmain(int argc, char *argv[])
+{
+ Ps *ps1, *ps2, *pssum, *ints, *tanx, *pspair[2];
+ Rat one, zero;
+
+ ratfmtinstall();
+
+ zero = ratmk(0, 1);
+ one = ratmk(1, 1);
+
+ ps1 = mkconst(one);
+ print("1 1 1 1 1 1 1 1 1 1\n");
+ psprint(ps1, 10);
+ print("\n");
+
+ ps2 = mkconst(one);
+ pssum = psadd(ps1, ps2);
+ print("2 2 2 2 2 2 2 2 2 2\n");
+ psprint(pssum, 10);
+ print("\n");
+ send(pssum->close, nil);
+
+ ps1 = mkconst(one);
+ ints = psderiv(ps1);
+ print("1 2 3 4 5 6 7 8 9 10\n");
+ psprint(ints, 10);
+ print("\n");
+ send(ints->close, nil);
+
+ ps1 = mkconst(one);
+ ps2 = psinteg(ps1, one);
+ print("1 1 1/2 1/3 1/4 1/5 1/6 1/7 1/8 1/9\n");
+ psprint(ps2, 10);
+ print("\n");
+ send(ps2->close, nil);
+
+ ps1 = mkconst(one);
+ ints = psderiv(ps1);
+ split(ints, pspair, nil);
+ print("1 2 3 4 5 6 7 8 9 10\n");
+ print("1 2 3 4 5 6 7 8 9 10\n");
+ psprint(pspair[0], 10);
+ psprint(pspair[1], 10);
+ print("\n");
+ send(pspair[0]->close, nil);
+ send(pspair[1]->close, nil);
+
+ ps1 = mkconst(one);
+ ps2 = pscmul(ratmk(2, 1), ps1);
+ print("2 2 2 2 2 2 2 2 2 2\n");
+ psprint(ps2, 10);
+ print("\n");
+ send(ps2->close, nil);
+
+ ps1 = mkconst(one);
+ split(ps1, pspair, nil);
+ ps2 = psmul(pspair[0], pspair[1]);
+ print("1 2 3 4 5 6 7 8 9 10\n");
+ psprint(ps2, 10);
+ print("\n");
+ send(ps2->close, nil);
+
+ ps1 = mkconst(one);
+ ps2 = psrecip(ps1);
+ print("1 -1 0 0 0 0 0 0 0 0\n");
+ psprint(ps2, 10);
+ print("\n");
+ send(ps2->close, nil);
+
+ ps1 = mkconst(one);
+ tanx = psrev(psinteg(psmsubst(ps1, ratmk(-1, 1), 2), zero));
+ print("0 1 0 1/3 0 2/15 0 17/315 0 62/2835 0 1382/155925\n");
+ psprint(tanx, 12);
+ print("\n");
+ send(tanx->close, nil);
+
+ threadexitsall(0);
+}
--- /dev/null
+++ b/squint.h
@@ -1,0 +1,31 @@
+enum
+{
+ TMAX = 1000,
+ STK = 2048
+};
+
+int threadcount;
+int verbose;
+
+typedef struct Rat Rat;
+typedef struct Node Node;
+typedef struct Lifo Lifo;
+
+struct Rat{
+ vlong num;
+ vlong den;
+};
+
+struct Node{
+ Rat val;
+ Node *link;
+};
+
+struct Lifo{
+ Node *front;
+ Node *rear;
+};
+
+Lifo *mklifo(void);
+int insert(Lifo*, Rat);
+int delete(Lifo*, Rat*);