shithub: squint

Download patch

ref: 0c8f9f3968acc8b277f5aea3ac3c568fde1a3631
parent: 09411555a2a9a5600efeec449722ea362d1de25e
author: Benjamin Purcell <bpurcell@backstopsolutions.com>
date: Thu Aug 4 10:36:40 EDT 2016

Use concurrent C extension

--- a/fifo.c
+++ /dev/null
@@ -1,63 +1,0 @@
-#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;
-}
--- a/fifo.h
+++ /dev/null
@@ -1,19 +1,0 @@
-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*);
--- a/mkfile
+++ b/mkfile
@@ -1,8 +1,8 @@
 </$objtype/mkfile
 
 TARG=	squint
-OFILES=	squint.$O fifo.$O rat.$O
-HFILES=	fifo.h rat.h
+OFILES=	squint.$O queue.$O rat.$O
+HFILES=	queue.h rat.h
 BIN=	.
 
 </sys/src/cmd/mkone
--- /dev/null
+++ b/queue.c
@@ -1,0 +1,69 @@
+#include <u.h>
+#include <libc.h>
+#include "rat.h"
+#include "queue.h"
+
+Queue*
+mkqueue(void)
+{
+	Queue *q;
+
+	q = malloc(sizeof(*q));
+	queueset(q);
+	return q;
+}
+
+void
+queueset(Queue *q)
+{
+	q->front = nil;
+	q->rear = &q->front;
+}
+
+int
+isempty(Queue *q)
+{
+	return q->front == nil;
+}
+
+void
+frontinsert(Queue *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(Queue *q, Rat val)
+{
+	Node	*n;
+
+	n = malloc(sizeof(*n));
+	n->val = val;
+	n->link = nil;
+	*q->rear = n;
+	q->rear = &n->link;
+	return;
+}
+
+Rat
+delete(Queue *q)
+{
+	Rat	r;
+	Node	*n;
+
+	n = q->front;
+	r = n->val;
+	q->front = n->link;
+	free(n);
+	if(q->front == nil)
+		q->rear = &q->front;
+	return r;
+}
--- /dev/null
+++ b/queue.h
@@ -1,0 +1,20 @@
+typedef struct Node Node;
+typedef struct Queue Queue;
+
+struct Node
+{
+	Node *link;
+	Rat	val;
+};
+
+struct Queue
+{
+	Node *front;
+	Node **rear;
+};
+
+Queue *mkqueue(void);
+void queueset(Queue*);
+void insert(Queue*, Rat);
+void frontinsert(Queue*, Rat);
+Rat delete(Queue*);
--- a/rat.c
+++ b/rat.c
@@ -7,7 +7,7 @@
 {
 	vlong t;
 
-	while(b != 0){
+	while(b != 0) {
 		t = b;
 		b = a % b;
 		a = t;
@@ -22,10 +22,10 @@
 	vlong	g;
 
 	g = gcd(n, d);
-	if(d > 0){
+	if(d > 0) {
 		r.num = n/g;
 		r.den = d/g;
-	}else{
+	} else {
 		r.num = -n/g;
 		r.den = -d/g;
 	}
@@ -40,7 +40,7 @@
 	t = a.num;
 	a.num = a.den;
 	a.den = t;
-	if(a.den < 0){
+	if(a.den < 0) {
 		a.num = -a.num;
 		a.den = -a.den;
 	}
--- a/rat.h
+++ b/rat.h
@@ -1,4 +1,5 @@
-typedef struct Rat	Rat;
+typedef struct Rat Rat;
+
 struct Rat
 {
 	vlong num;
@@ -7,11 +8,11 @@
 
 #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);
+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);
--- a/readme.md
+++ b/readme.md
@@ -3,7 +3,7 @@
 
 This code is a translation of the code in
 [Squinting at Power Series](http://swtch.com/~rsc/thread/squint.pdf)
-into C using libthread.
+into C using the concurrent extensions to KenCC.
 
 There are a few differences:
 
@@ -11,14 +11,14 @@
 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
+* The implementation of 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
+* Newsqueak is garbage collected.  This implementation instead
+gives each power series a 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.
+free all its data structures and exit.
 
-I have implementations of the original method of handling
-multiplication and splitting, email me if you are interested.
+I had implementations of the original method of handling
+multiplication and splitting, email me if you are interested I might
+be able to find them.
--- a/squint.c
+++ b/squint.c
@@ -2,32 +2,30 @@
 #include <libc.h>
 #include <thread.h>
 #include "rat.h"
-#include "fifo.h"
+#include "queue.h"
 
-int debug;
-int threadcount;
-
 enum
 {
-	TMAX = 1000,
-	STK = 2048
+	STK = 8192
 };
 
 typedef struct Ps Ps;
 struct Ps
 {
-	Channel *close;
-	Channel *req;
-	Channel	*dat;
+	char @close;
+	char @req;
+	Rat @dat;
 };
 
+int debug;
+int threadcount;
+
 Rat
 getr(Ps *f)
 {
-	Rat	r;
-
-	send(f->req, nil);
-	recv(f->dat, &r);
+	Rat r;
+	f->req @= 0;
+	r = @f->dat;
 	return r;
 }
 
@@ -37,9 +35,9 @@
 	Ps	*d;
 
 	d = malloc(sizeof(*d));
-	d->close = chancreate(1, 0);
-	d->req = chancreate(1, 0);
-	d->dat = chancreate(sizeof(Rat), nel);
+	chanset(d->close, 0);
+	chanset(d->req, 0);
+	chanset(d->dat, nel);
 	return d;
 }
 
@@ -52,129 +50,69 @@
 	free(d);
 }
 
+
 static void
-_split(void *a)
+_split(Ps *f, Ps *s, Rat @qchan, char @close)
 {
-	void	**argv;
-	Ps	*f, *s;
-	Channel	*qchan, *close;
-	Fifo	q;
-	Rat	r;
-	enum{
-		REQ,
-		QCHAN,
-		CLOSE,
-		NALT
-	};
-	Alt	alts[NALT+1];
+	Queue q;
+	Rat r;
 
-	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);
+	queueset(&q);
+	for(;;) switch @{
+	alt @s->req:
+		if(q.front == nil) {
+			r = getr(f);
+			qchan @= r;
+			s->dat @= r;
+		} else
+			s->dat @= delete(&q);
+		break;
+	alt r = @qchan:
+		insert(&q, r);
+		break;
+	alt @s->close:
+		while(q.front != nil)
+			delete(&q);
+		psfree(s);
+		if(chanclosing(qchan) == -1) {
+			if(close != nil) {
+				close @= 0;
+				chanfree(close);
 			}
-			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);
+			chanclose(qchan);
+		} else {
+			chanfree(qchan);
+			f->close @= 0;
 		}
+		if(debug) threadcount--;
+		threadexits(0);
 	}
 }
 
 void
-split(Ps *f, Ps *s[2], Channel *splitclose)
+split(Ps *f, Ps *s[2], char @splitclose)
 {
-	void	*argv[2][4];
-	Channel	*q;
-	int	i;
+	Rat @q;
+	int i;
 
-	q = chancreate(sizeof(Rat), 0);
-	for(i = 0; i < 2; i++){
+	chanset(q, 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);
+		cothread(_split(f, s[i], q, splitclose), STK);
 		if(debug) threadcount++;
-		yield();
 	}
 }
 
 static void
-_mkconst(void *a)
+_mkconst(Rat c, Ps *o)
 {
-	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);
-		}
+	for(;;) switch @{
+	alt @o->req:
+		o->dat @= c;
+		break;
+	alt @o->close:
+		if(debug) threadcount--;
+		threadexits(0);
 	}
 }
 
@@ -181,71 +119,38 @@
 Ps*
 mkconst(Rat c)
 {
-	void	*argv[2];
-	Ps	*o;
+	Ps *o;
 
 	o = psmk(0);
-	argv[0] = &c;
-	argv[1] = o;
-	threadcreate(_mkconst, argv, STK);
+	cothread(_mkconst(c, o), STK);
 	if(debug) threadcount++;
-	yield();
 	return o;
 }
 
 Ps*
-binop(void (*oper)(void*), Ps *f, Ps *g)
+binop(void (*oper)(Ps*, Ps*, Ps*), Ps *f, Ps *g)
 {
-	Ps *argv[3], *o;
+	Ps *o;
 
 	o = psmk(0);
-	argv[0] = f;
-	argv[1] = g;
-	argv[2] = o;
-	threadcreate(oper, argv, STK);
+	cothread(oper(f, g, o), STK);
 	if(debug) threadcount++;
-	yield();
 	return o;
 }
 
 static void
-_psadd(void *a)
+_psadd(Ps *f, Ps *g, Ps *s)
 {
-	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);
-		}
+	for(;;) switch @{
+	alt @s->req:
+		s->dat @= ratadd(getr(f), getr(g));
+		break;
+	alt @s->close:
+		psfree(s);
+		f->close @= 0;
+		g->close @= 0;
+		if(debug) threadcount--;
+		threadexits(0);
 	}
 }
 
@@ -256,41 +161,21 @@
 }
 
 static void
-_psderiv(void *a)
+_psderiv(Ps *f, Ps *d)
 {
-	Ps	**argv, *f, *d;
-	Rat	t, r;
-	int	i;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt	alts[NALT+1];
+	Rat r;
+	int i;
 
-	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:
+	for(i = 1;; i++) {
+		switch @{
+		alt @d->req:
 			r = getr(f);
-			t = ratmk(i * r.num, r.den);
-			send(d->dat, &t);
+			d->dat @= ratmk(i * r.num, r.den);
 			break;
-		case CLOSE:
+		alt @d->close:
 			psfree(d);
-			send(f->close, nil);
+			f->close @= 0;
 			if(debug) threadcount--;
 			threadexits(0);
 		}
@@ -300,13 +185,11 @@
 Ps*
 psderiv(Ps *f)
 {
-	Ps	*argv[2], *d = psmk(0);
+	Ps *d;
 
-	argv[0] = f;
-	argv[1] = d;
-	threadcreate(_psderiv, argv, STK);
+	d = psmk(0);
+	cothread(_psderiv(f, d), STK);
 	if(debug) threadcount++;
-	yield();
 	return d;
 }
 
@@ -322,51 +205,29 @@
 }
 
 static void
-_psinteg(void *a)
+_psinteg(Ps *f, Rat c, Ps *i)
 {
-	void	**argv = a;
-	Ps	*f, *i;
-	Rat	c, out;
-	int	j;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt alts[NALT+1];
+	int j;
 
-	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);
+	switch @{
+	alt @i->req:
+		i->dat @= c;
 		break;
-	case CLOSE:
+	alt @i->close:
 		goto End;
 	}
-	for(j = 1;; j++){
-		switch(alt(alts)){
-		case REQ:
-			out = ratmul(getr(f), ratmk(1, j));
-			send(i->dat, &out);
+	for(j = 1;; j++) {
+		switch @{
+		alt @i->req:
+			i->dat @= ratmul(getr(f), ratmk(1, j));
 			break;
-		case CLOSE:
+		alt @i->close:
 			goto End;
 		}
 	}
 End:
 	psfree(i);
-	send(f->close, nil);
+	f->close @= 0;
 	if(debug) threadcount--;
 	threadexits(0);
 }
@@ -375,55 +236,26 @@
 Ps*
 psinteg(Ps *ps, Rat c)
 {
-	void	*argv[3];
-	Ps	*i = psmk(0);
+	Ps *i;
 
-	argv[0] = ps;
-	argv[1] = &c;
-	argv[2] = i;
-	threadcreate(_psinteg, argv, STK);
+	i = psmk(0);
+	cothread(_psinteg(ps, c, i), STK);
 	if(debug) threadcount++;
-	yield();
 	return i;
 }
 
 static void
-_pscmul(void *a)
+_pscmul(Ps *f, Rat c, Ps *o)
 {
-	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);
-		}
+	for(;;) switch @{
+	alt @o->req:
+		o->dat @= ratmul(c, getr(f));
+		break;
+	alt @o->close:
+		psfree(o);
+		f->close @= 0;
+		if(debug) threadcount--;
+		threadexits(0);
 	}
 }
 
@@ -430,72 +262,47 @@
 Ps*
 pscmul(Rat c, Ps* f)
 {
-	void	*argv[3];
-	Ps	*o = psmk(0);
+	Ps *o;
 
-	argv[0] = f;
-	argv[1] = &c;
-	argv[2] = o;
-	threadcreate(_pscmul, argv, STK);
+	o = psmk(0);
+	cothread(_pscmul(f, c, o), STK);
 	if(debug) threadcount++;
-	yield();
 	return o;
 }
 
 static void
-_psmul(void *a)
+_psmul(Ps *f, Ps *g, Ps *p)
 {
-	Ps	**argv, *f, *g, *p;
-	Fifo	fq, gq;
-	Node	*fnode, *gnode;
-	Rat	sum;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt 	alts[NALT+1];
+	Queue fq, gq;
+	Node *fnode, *gnode;
+	Rat sum;
 
-	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);
+	queueset(&fq);
+	queueset(&gq);
+	for(;;) switch @{
+	alt @p->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;	
 		}
+		p->dat @= sum;
+		break;
+	alt @p->close:
+		psfree(p);
+		while(fq.front != nil)
+			delete(&fq);
+		while(gq.front != nil)
+			delete(&gq);
+		f->close @= 0;
+		g->close @= 0;
+		if(debug) threadcount--;
+		threadexits(0);
 	}
 }
 
@@ -506,57 +313,32 @@
 }
 
 static void
-_psrecip(void *a)
+_psrecip(Ps *f, Ps *r, Ps *rr, char @close)
 {
-	void	**argv;
-	Ps	*f, *r, *rr, *recip;
-	Channel	*close;
-	Rat	g;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt alts[NALT+1];
+	Ps *recip;
+	Rat g;
 
-	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:
+	switch @{
+	alt @r->req:
 		g = ratrecip(getr(f));
-		send(r->dat, &g);
+		r->dat @= g;
 		break;
-	case CLOSE:
-		send(rr->close, nil);
-		send(f->close, nil);
+	alt @close:
+		rr->close @= 0;
+		f->close @= 0;
 		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;
-		}
+	for(;;) switch @{
+	alt @r->req:
+		r->dat @= getr(recip);
+		break;
+	alt @close:
+		recip->close @= 0;
+		goto End;
 	}
 End:
-	recv(r->close, nil);
+	@r->close;
 	psfree(r);
 	if(debug) threadcount--;
 	threadexits(0);
@@ -565,72 +347,46 @@
 Ps*
 psrecip(Ps *f)
 {
-	void	*argv[4];
-	Ps	*rr[2], *r = psmk(0);
-	Channel	*close = chancreate(1, 0);
+	Ps *rr[2], *r;
+	char @close;
 
+	r = psmk(0);
+	chanset(close, 0);
+
 	split(r, rr, close);
-	argv[0] = f;
-	argv[1] = r;
-	argv[2] = rr[0];
-	argv[3] = close;
-	threadcreate(_psrecip, argv, STK);
+	cothread(_psrecip(f, r, rr[0], close), STK);
 	if(debug) threadcount++;
-	yield();
 	return rr[1];
 }
 
 static void
-_psmsubst(void *a)
+_psmsubst(Ps *f, Rat c, int deg, Ps *m)
 {
-	void	**argv;
-	Ps	*f, *m;
-	Rat	c, zero, r;
-	int	deg, i, j;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt	alts[NALT+1];
+	Rat zero;
+	int i, j;
 
-	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(i = 0;; i++) {
+		switch @{
+		alt @m->req:
+			m->dat @= ratmul(getr(f), ratpow(c, i));
 			for(j = 0; j < deg - 1; j++){
-				switch(alt(alts)){
-				case REQ:
-					send(m->dat, &zero);
+				switch @{
+				alt @m->req:
+					m->dat @= zero;
 					break;
-				case CLOSE:
+				alt @m->close:
 					goto End;
 				}
 			}
 			break;
-		case CLOSE:
+		alt @m->close:
 			goto End;
 		}
 	}
 End:
 	psfree(m);
-	send(f->close, nil);
+	f->close @= 0;
 	if(debug) threadcount--;
 	threadexits(0);
 }
@@ -638,17 +394,11 @@
 Ps*
 psmsubst(Ps *f, Rat c, int deg)
 {
-	void	*argv[4];
-	Ps	*m;
+	Ps *m;
 
 	m = psmk(0);
-	argv[0] = f;
-	argv[1] = &c;
-	argv[2] = &deg;
-	argv[3] = m;
-	threadcreate(_psmsubst, argv, STK);
+	cothread(_psmsubst(f, c, deg, m), STK);
 	if(debug) threadcount++;
-	yield();
 	return m;
 }
 
@@ -655,39 +405,18 @@
 Ps *pssubst(Ps*, Ps*);
 
 static void
-_pssubst(void *a)
+_pssubst(Ps *f, Ps *g, Ps *o)
 {
-	Ps	**argv, *gg[2], *f, *g, *o, *subst;
-	Rat	r;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt	alts[NALT+1];
+	Ps *gg[2], *subst;
 
-	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);
+	switch @{
+	alt @o->req:
+		o->dat @= getr(f);
 		break;
-	case CLOSE:
+	alt @o->close:
 		psfree(o);
-		send(f->close, nil);
-		send(g->close, nil);
+		f->close @= 0;
+		g->close @= 0;
 		if(debug) threadcount--;
 		threadexits(0);
 	}
@@ -694,18 +423,15 @@
 	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);
-		}
+	for(;;) switch @{
+	alt @o->req:
+		o->dat @= getr(subst);
+		break;
+	alt @o->close:
+		psfree(o);
+		subst->close @= 0;
+		if(debug) threadcount--;
+		threadexits(0);
 	}
 }
 
@@ -716,58 +442,33 @@
 }
 
 static void
-_psrev(void *a)
+_psrev(Ps *f, Ps *r, Ps *rr, char @close)
 {
-	void	**argv;
-	Ps	*f, *r, *rr, *rever;
-	Channel	*close;
-	Rat	g;
-	enum{
-		REQ,
-		CLOSE,
-		NALT
-	};
-	Alt alts[NALT+1];
+	Ps *rever;
+	Rat g;
 
-	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);
+	switch @{
+	alt @r->req:
+		r->dat @= g;
 		break;
-	case CLOSE:
-		send(rr->close, nil);
-		send(f->close, nil);
+	alt @close:
+		rr->close @= 0;
+		f->close @= 0;
 		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;
-		}
+	for(;;) switch @{
+	alt @r->req:
+		r->dat @= getr(rever);
+		break;
+	alt @close:
+		rever->close @= 0;
+		goto End;
 	}
 End:
-	recv(r->close, nil);
+	@r->close;
 	psfree(r);
 	if(debug) threadcount--;
 	threadexits(0);
@@ -776,28 +477,24 @@
 Ps*
 psrev(Ps *f)
 {
-	void	*argv[4];
-	Ps	*rr[2], *r;
-	Channel	*close = chancreate(1,0);
+	Ps *rr[2], *r;
+	char @close;
 
+	chanset(close, 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);
+	cothread(_psrev(f, r, rr[0], close), STK);
 	if(debug) threadcount++;
-	yield();
 	return rr[1];
 }
 
 void
-threadmain(int argc, char *argv[])
+threadmain(int, char **)
 {
 	Ps	*ps1, *ps2, *pssum, *ints, *tanx, *pspair[2];
 	Rat	one, zero;
 
+	debug = 1;
 	ratfmtinstall();
 
 	zero = ratmk(0, 1);
@@ -804,6 +501,7 @@
 	one = ratmk(1, 1);
 
 	ps1 = mkconst(one);
+	print("1/1-x\n");
 	print("1 1 1 1 1 1 1 1 1 1\n");
 	psprint(ps1, 10);
 	print("\n");
@@ -810,6 +508,7 @@
 
 	ps2 = mkconst(one);
 	pssum = psadd(ps1, ps2);
+	print("1/1-x + 1/1-x\n");
 	print("2 2 2 2 2 2 2 2 2 2\n");
 	psprint(pssum, 10);
 	print("\n");
@@ -817,6 +516,7 @@
 
 	ps1 = mkconst(one);
 	ints = psderiv(ps1);
+	print("d/dx(1/1-x)\n");
 	print("1 2 3 4 5 6 7 8 9 10\n");
 	psprint(ints, 10);
 	print("\n");
@@ -824,6 +524,7 @@
 
 	ps1 = mkconst(one);
 	ps2 = psinteg(ps1, one);
+	print("integral of 1/1-x\n");
 	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");
@@ -832,6 +533,7 @@
 	ps1 = mkconst(one);
 	ints = psderiv(ps1);
 	split(ints, pspair, nil);
+	print("d/dx(1/1-x) split into two streams\n");
 	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);
@@ -842,6 +544,7 @@
 
 	ps1 = mkconst(one);
 	ps2 = pscmul(ratmk(2, 1), ps1);
+	print("2*(1/1-x)\n");
 	print("2 2 2 2 2 2 2 2 2 2\n");
 	psprint(ps2, 10);
 	print("\n");
@@ -850,6 +553,7 @@
 	ps1 = mkconst(one);
 	split(ps1, pspair, nil);
 	ps2 = psmul(pspair[0], pspair[1]);
+	print("1/1-x split into two streams and then multiplied\n");
 	print("1 2 3 4 5 6 7 8 9 10\n");
 	psprint(ps2, 10);
 	print("\n");
@@ -857,6 +561,7 @@
 
 	ps1 = mkconst(one);
 	ps2 = psrecip(ps1);
+	print("1/(1/1-x)\n");
 	print("1 -1 0 0 0 0 0 0 0 0\n");
 	psprint(ps2, 10);
 	print("\n");
@@ -864,10 +569,9 @@
 
 	ps1 = mkconst(one);
 	tanx = psrev(psinteg(psmsubst(ps1, ratmk(-1, 1), 2), zero));
+	print("tan(x)\n");
 	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);
 }