ref: 569bdd00c2d3df235ed42abce09d286f30e0edbd
parent: ccfb9118a3bf19a0a943b3a7bbfddd82cf4ddc5b
author: cinap_lenrek <[email protected]>
date: Sun Nov 22 18:47:58 EST 2015
add mpc(1), extended precision code generator
--- /dev/null
+++ b/sys/man/1/mpc
@@ -1,0 +1,155 @@
+.TH MPC 1
+.SH NAME
+mpc \- extended precision arithmetic code generator
+.SH SYNOPSIS
+.B mpc
+[
+.I file ...
+]
+.SH DESCRIPTION
+.I Mpc
+generates C functions from a simple language that operates on
+extended precision integers using the
+.IR mp (2)
+library.
+.SH LANGUAGE
+The language consists of a series of function definitions of the form:
+.IP
+.I name
+(
+.I "parameter list"
+) {
+.I statements
+}
+.PP
+All variables and parameters are extended precision integers and are
+passed by reference. Statements are separated by semicolon and the
+following statemens are defined:
+.IP
+.I name
+.B =
+.I expression
+.IP
+.B if
+(
+.I condition
+) {
+.I statements
+}
+.B "else if"
+(
+.I condition
+) {
+.I statements
+}
+.B else
+{
+.I statements
+}
+.IP
+.B while
+(
+.I condition
+) {
+.I statements
+}
+.IP
+.B break
+.IP
+.I
+name
+(
+.I "parameter list"
+)
+.IP
+.B
+mod
+(
+.I modulus
+) {
+.I statements
+}
+.PP
+There is no distinction between input and output parameters, but
+conventionally, the outputs are put at the end of the
+.I "parameter list"
+and the language allows one to write
+.IP
+.I F
+(
+.IR X ,
+.IR Y ,
+.I Z
+)
+as
+.IR Y ,
+.I Z
+.B =
+.I F
+(
+.I X
+)
+.PP
+Expressions are composed out of the following arithmetic operations:
+.RS
+.TF _____________
+.TP
+.B +
+addition.
+.TP
+.B -
+subtraction.
+.TP
+.B *
+multiplication.
+.TP
+.B /
+division, or multiplicative inverse when enclosed in
+.B mod
+block.
+.TP
+.B %
+division remainder.
+.TP
+.B ^
+exponentiation.
+.TP
+.BI >> constant
+right shift by a constant.
+.TP
+.BI << constant
+left shift by a constant.
+.TP
+.IB condition ? a : b
+pick
+.I a
+when
+.I condition is true, otherwise
+.I b
+when false.
+.RE
+.PD
+.PP
+Conditions can use the following operations:
+.RS
+.TF _____________
+.TP
+.B ==
+equality.
+.TP
+.B !=
+inequality.
+.TP
+.B >
+bigger than.
+.TP
+.B <
+smaller than.
+.TP
+.BI ! condition
+negation.
+.RE
+.SH SOURCE
+.B /sys/src/cmd/mpc.y
+.SH "SEE ALSO"
+.IR mp (2)
--- a/sys/src/cmd/mkfile
+++ b/sys/src/cmd/mkfile
@@ -108,7 +108,7 @@
%.acid: %.$O $HFILES
$CC $CFLAGS -a $stem.c >$target
-(bc|units).c:R: \1.tab.c
+(bc|units|mpc).c:R: \1.tab.c
mv $stem1.tab.c $stem1.c
$BIN/init: $O.init
--- /dev/null
+++ b/sys/src/cmd/mpc.y
@@ -1,0 +1,1037 @@
+%{
+
+#include <u.h>
+#include <libc.h>
+#include <bio.h>
+#include <mp.h>
+
+typedef struct Sym Sym;
+typedef struct Node Node;
+
+enum {
+ FSET = 1,
+ FUSE = 2,
+ FARG = 4,
+ FLOC = 8,
+};
+
+struct Sym
+{
+ Sym* l;
+ int f;
+ char n[];
+};
+
+struct Node
+{
+ int c;
+ Node* l;
+ Node* r;
+ Sym* s;
+ int n;
+};
+
+#pragma varargck type "N" Node*
+
+int ntmp;
+Node *ftmps, *atmps;
+Node *modulo;
+
+Node* new(int, Node*, Node*);
+Sym* sym(char*);
+
+Biobuf bin;
+int goteof;
+int lineno;
+int clevel;
+char* filename;
+
+int getch(void);
+void ungetc(void);
+void yyerror(char*);
+int yyparse(void);
+void diag(Node*, char*, ...);
+void com(Node*);
+void fcom(Node*,Node*,Node*);
+
+#pragma varargck argpos cprint 1
+#pragma varargck argpos diag 2
+
+%}
+
+%union
+{
+ Sym* sval;
+ Node* node;
+ long lval;
+}
+
+%type <node> name num args expr bool block elif stmnt stmnts
+
+%left '{' '}' ';'
+%right '=' ','
+%right '?' ':'
+%left EQ NEQ '<' '>'
+%left LSH RSH
+%left '+' '-'
+%left '/' '%'
+%left '*'
+%left '^'
+%right '('
+
+%token <lval> MOD IF ELSE WHILE BREAK
+%token <sval> NAME NUM
+
+%%
+
+prog:
+ prog func
+| func
+
+func:
+ name args stmnt
+ {
+ fcom($1, $2, $3);
+ }
+
+args:
+ '(' expr ')'
+ {
+ $$ = $2;
+ }
+| '(' ')'
+ {
+ $$ = nil;
+ }
+
+name:
+ NAME
+ {
+ $$ = new(NAME,nil,nil);
+ $$->s = $1;
+ }
+num:
+ NUM
+ {
+ $$ = new(NUM,nil,nil);
+ $$->s = $1;
+ }
+
+elif:
+ ELSE IF '(' bool ')' stmnt
+ {
+ $$ = new('?', $4, new(':', $6, nil));
+ }
+| ELSE IF '(' bool ')' stmnt elif
+ {
+ $$ = new('?', $4, new(':', $6, $7));
+ }
+| ELSE stmnt
+ {
+ $$ = $2;
+ }
+
+sem:
+ sem ';'
+| ';'
+
+stmnt:
+ expr '=' expr sem
+ {
+ $$ = new('=', $1, $3);
+ }
+| MOD args stmnt
+ {
+ $$ = new('m', $2, $3);
+ }
+| IF '(' bool ')' stmnt
+ {
+ $$ = new('?', $3, new(':', $5, nil));
+ }
+| IF '(' bool ')' stmnt elif
+ {
+ $$ = new('?', $3, new(':', $5, $6));
+ }
+| WHILE '(' bool ')' stmnt
+ {
+ $$ = new('@', new('?', $3, new(':', $5, new('b', nil, nil))), nil);
+ }
+| BREAK sem
+ {
+ $$ = new('b', nil, nil);
+ }
+| expr sem
+ {
+ if($1->c == NAME)
+ $$ = new('e', $1, nil);
+ else
+ $$ = $1;
+ }
+| block
+
+block:
+ '{' stmnts '}'
+ {
+ $$ = $2;
+ }
+
+stmnts:
+ stmnts stmnt
+ {
+ $$ = new('\n', $1, $2);
+ }
+| stmnt
+
+expr:
+ '(' expr ')'
+ {
+ $$ = $2;
+ }
+| name
+ {
+ $$ = $1;
+ }
+| num
+ {
+ $$ = $1;
+ }
+| '-' expr
+ {
+ $$ = new(NUM, nil, nil);
+ $$->s = sym("0");
+ $$->s->f = 0;
+ $$ = new('-', $$, $2);
+ }
+| expr ',' expr
+ {
+ $$ = new(',', $1, $3);
+ }
+| expr '^' expr
+ {
+ $$ = new('^', $1, $3);
+ }
+| expr '*' expr
+ {
+ $$ = new('*', $1, $3);
+ }
+| expr '/' expr
+ {
+ $$ = new('/', $1, $3);
+ }
+| expr '%' expr
+ {
+ $$ = new('%', $1, $3);
+ }
+| expr '+' expr
+ {
+ $$ = new('+', $1, $3);
+ }
+| expr '-' expr
+ {
+ $$ = new('-', $1, $3);
+ }
+| bool '?' expr ':' expr
+ {
+ $$ = new('?', $1, new(':', $3, $5));
+ }
+| name args
+ {
+ $$ = new('e', $1, $2);
+ }
+| expr LSH num
+ {
+ $$ = new(LSH, $1, $3);
+ }
+| expr RSH num
+ {
+ $$ = new(RSH, $1, $3);
+ }
+
+bool:
+ '(' bool ')'
+ {
+ $$ = $2;
+ }
+| '!' bool
+ {
+ $$ = new('!', $2, nil);
+ }
+| expr EQ expr
+ {
+ $$ = new(EQ, $1, $3);
+ }
+| expr NEQ expr
+ {
+ $$ = new('!', new(EQ, $1, $3), nil);
+ }
+| expr '>' expr
+ {
+ $$ = new('>', $1, $3);
+ }
+| expr '<' expr
+ {
+ $$ = new('<', $1, $3);
+ }
+
+%%
+
+int
+yylex(void)
+{
+ static char buf[200];
+ char *p;
+ int c;
+
+Loop:
+ c = getch();
+ switch(c){
+ case -1:
+ return -1;
+ case ' ':
+ case '\t':
+ case '\n':
+ goto Loop;
+ case '#':
+ while((c = getch()) > 0)
+ if(c == '\n')
+ break;
+ goto Loop;
+ }
+
+ switch(c){
+ case '?': case ':':
+ case '+': case '-':
+ case '*': case '^':
+ case '/': case '%':
+ case '{': case '}':
+ case '(': case ')':
+ case ',': case ';':
+ return c;
+ case '<':
+ if(getch() == '<') return LSH;
+ ungetc();
+ return '<';
+ case '>':
+ if(getch() == '>') return RSH;
+ ungetc();
+ return '>';
+ case '=':
+ if(getch() == '=') return EQ;
+ ungetc();
+ return '=';
+ case '!':
+ if(getch() == '=') return NEQ;
+ ungetc();
+ return '!';
+ }
+
+ ungetc();
+ p = buf;
+ for(;;){
+ c = getch();
+ if((c >= Runeself)
+ || (c == '_')
+ || (c >= 'a' && c <= 'z')
+ || (c >= 'A' && c <= 'Z')
+ || (c >= '0' && c <= '9')){
+ *p++ = c;
+ continue;
+ }
+ ungetc();
+ break;
+ }
+ *p = '\0';
+
+ if(strcmp(buf, "mod") == 0)
+ return MOD;
+ if(strcmp(buf, "if") == 0)
+ return IF;
+ if(strcmp(buf, "else") == 0)
+ return ELSE;
+ if(strcmp(buf, "while") == 0)
+ return WHILE;
+ if(strcmp(buf, "break") == 0)
+ return BREAK;
+
+ yylval.sval = sym(buf);
+ yylval.sval->f = 0;
+ return (buf[0] >= '0' && buf[0] <= '9') ? NUM : NAME;
+}
+
+
+int
+getch(void)
+{
+ int c;
+
+ c = Bgetc(&bin);
+ if(c == Beof){
+ goteof = 1;
+ return -1;
+ }
+ if(c == '\n')
+ lineno++;
+ return c;
+}
+
+void
+ungetc(void)
+{
+ Bungetc(&bin);
+}
+
+Node*
+new(int c, Node *l, Node *r)
+{
+ Node *n;
+
+ n = malloc(sizeof(Node));
+ n->c = c;
+ n->l = l;
+ n->r = r;
+ n->s = nil;
+ n->n = lineno;
+ return n;
+}
+
+Sym*
+sym(char *n)
+{
+ static Sym *tab[128];
+ Sym *s;
+ ulong h, t;
+ int i;
+
+ h = 0;
+ for(i=0; n[i] != '\0'; i++){
+ t = h & 0xf8000000;
+ h <<= 5;
+ h ^= t>>27;
+ h ^= (ulong)n[i];
+ }
+ h %= nelem(tab);
+ for(s = tab[h]; s != nil; s = s->l)
+ if(strcmp(s->n, n) == 0)
+ return s;
+ s = malloc(sizeof(Sym)+i+1);
+ memmove(s->n, n, i+1);
+ s->f = 0;
+ s->l = tab[h];
+ tab[h] = s;
+ return s;
+}
+
+void
+yyerror(char *s)
+{
+ fprint(2, "%s:%d: %s\n", filename, lineno, s);
+ exits(s);
+}
+void
+cprint(char *fmt, ...)
+{
+ static char buf[1024], tabs[] = "\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t";
+ char *p, *x;
+ va_list a;
+
+ va_start(a, fmt);
+ vsnprint(buf, sizeof(buf), fmt, a);
+ va_end(a);
+
+ p = buf;
+ while((x = strchr(p, '\n')) != nil){
+ x++;
+ write(1, p, x-p);
+ p = &tabs[sizeof(tabs)-1 - clevel];
+ if(*p != '\0')
+ write(1, p, strlen(p));
+ p = x;
+ }
+ if(*p != '\0')
+ write(1, p, strlen(p));
+}
+
+Node*
+alloctmp(void)
+{
+ Node *t;
+
+ t = ftmps;
+ if(t != nil)
+ ftmps = t->l;
+ else {
+ char n[16];
+
+ snprint(n, sizeof(n), "tmp%d", ++ntmp);
+ t = new(NAME, nil, nil);
+ t->s = sym(n);
+
+ cprint("mpint *");
+ }
+ cprint("%N = mpnew(0);\n", t);
+ t->s->f &= ~(FSET|FUSE);
+ t->l = atmps;
+ atmps = t;
+ return t;
+}
+
+int
+isconst(Node *n)
+{
+ if(n->c == NUM)
+ return 1;
+ if(n->c == NAME){
+ return n->s == sym("mpzero") ||
+ n->s == sym("mpone") ||
+ n->s == sym("mptwo");
+ }
+ return 0;
+}
+
+int
+istmp(Node *n)
+{
+ Node *l;
+
+ if(n->c == NAME){
+ for(l = atmps; l != nil; l = l->l){
+ if(l->s == n->s)
+ return 1;
+ }
+ }
+ return 0;
+}
+
+
+void
+freetmp(Node *t)
+{
+ Node **ll, *l;
+
+ if(t == nil)
+ return;
+ if(t->c == ','){
+ freetmp(t->l);
+ freetmp(t->r);
+ return;
+ }
+ if(t->c != NAME)
+ return;
+
+ ll = &atmps;
+ for(l = atmps; l != nil; l = l->l){
+ if(l == t){
+ cprint("mpfree(%N);\n", t);
+ *ll = t->l;
+ t->l = ftmps;
+ ftmps = t;
+ return;
+ }
+ ll = &l->l;
+ }
+}
+
+int
+symref(Node *n, Sym *s)
+{
+ if(n == nil)
+ return 0;
+ if(n->c == NAME && n->s == s)
+ return 1;
+ return symref(n->l, s) || symref(n->r, s);
+}
+
+void
+nodeset(Node *n)
+{
+ if(n == nil)
+ return;
+ if(n->c == NAME){
+ n->s->f |= FSET;
+ return;
+ }
+ if(n->c == ','){
+ nodeset(n->l);
+ nodeset(n->r);
+ }
+}
+
+int
+complex(Node *n)
+{
+ if(n->c == NAME)
+ return 0;
+ if(n->c == NUM && strlen(n->s->n) == 1 && atoi(n->s->n) < 3)
+ return 0;
+ return 1;
+}
+
+void
+bcom(Node *n, Node *t);
+
+Node*
+ecom(Node *f, Node *t)
+{
+ Node *l, *r, *t2;
+ mpint *m;
+
+ if(f == nil)
+ return nil;
+
+ if(f->c == NUM){
+ m = strtomp(f->s->n, nil, 10, nil);
+ if(mpcmp(m, mpzero) == 0){
+ f->c = NAME;
+ f->s = sym("mpzero");
+ f->s->f = FSET;
+ return ecom(f, t);
+ }
+ if(mpcmp(m, mpone) == 0){
+ f->c = NAME;
+ f->s = sym("mpone");
+ f->s->f = FSET;
+ return ecom(f, t);
+ }
+ if(mpcmp(m, mptwo) == 0){
+ f->c = NAME;
+ f->s = sym("mptwo");
+ f->s->f = FSET;
+ return ecom(f, t);
+ }
+ mpfree(m);
+ }
+
+ if(f->c == ','){
+ if(t != nil)
+ diag(f, "cannot assign list to %N", t);
+ f->l = ecom(f->l, nil);
+ f->r = ecom(f->r, nil);
+ return f;
+ }
+
+ l = r = nil;
+ if(f->c == NAME){
+ if((f->s->f & FSET) == 0)
+ diag(f, "name used but not set");
+ f->s->f |= FUSE;
+ if(t == nil)
+ return f;
+ if(f->s != t->s)
+ cprint("mpassign(%N, %N);\n", f, t);
+ goto out;
+ }
+
+ if(t == nil)
+ t = alloctmp();
+
+ if(f->c == '?'){
+ bcom(f, t);
+ goto out;
+ }
+
+ if(f->c == 'e'){
+ r = ecom(f->r, nil);
+ if(r == nil)
+ cprint("%N(%N);\n", f->l, t);
+ else
+ cprint("%N(%N, %N);\n", f->l, r, t);
+ goto out;
+ }
+
+ if(t->c != NAME)
+ diag(f, "destination %N not a name", t);
+
+ switch(f->c){
+ case NUM:
+ m = strtomp(f->s->n, nil, 10, nil);
+ if(mpsignif(m) <= 32)
+ cprint("uitomp(%udUL, %N);\n", mptoui(m), t);
+ else if(mpsignif(m) <= 64)
+ cprint("uvtomp(%lludULL, %N);\n", mptouv(m), t);
+ else
+ cprint("strtomp(\"%.16B\", nil, 16, %N);\n", m, t);
+ mpfree(m);
+ goto out;
+ case LSH:
+ l = f->l->c == NAME ? f->l : ecom(f->l, t);
+ cprint("mpleft(%N, %N, %N);\n", l, f->r, t);
+ goto out;
+ case RSH:
+ l = f->l->c == NAME ? f->l : ecom(f->l, t);
+ cprint("mpright(%N, %N, %N);\n", l, f->r, t);
+ goto out;
+ case '*':
+ case '/':
+ l = ecom(f->l, nil);
+ r = ecom(f->r, nil);
+ break;
+ default:
+ l = ecom(f->l, complex(f->l) && !symref(f->r, t->s) ? t : nil);
+ r = ecom(f->r, complex(f->r) && l->s != t->s ? t : nil);
+ break;
+ }
+
+
+ if(modulo != nil){
+ switch(f->c){
+ case '+':
+ cprint("mpmodadd(%N, %N, %N, %N);\n", l, r, modulo, t);
+ goto out;
+ case '-':
+ cprint("mpmodsub(%N, %N, %N, %N);\n", l, r, modulo, t);
+ goto out;
+ case '*':
+ Modmul:
+ if(l->s == sym("mptwo") || r->s == sym("mptwo"))
+ cprint("mpmodadd(%N, %N, %N, %N); // 2*%N\n",
+ r->s == sym("mptwo") ? l : r,
+ r->s == sym("mptwo") ? l : r,
+ modulo, t,
+ r);
+ else
+ cprint("mpmodmul(%N, %N, %N, %N);\n", l, r, modulo, t);
+ goto out;
+ case '/':
+ if(l->s == sym("mpone")){
+ cprint("mpinvert(%N, %N, %N);\n", r, modulo, t);
+ goto out;
+ }
+ t2 = alloctmp();
+ cprint("mpinvert(%N, %N, %N);\n", r, modulo, t2);
+ cprint("mpmodmul(%N, %N, %N, %N);\n", l, t2, modulo, t);
+ freetmp(t2);
+ goto out;
+ case '^':
+ if(r->s == sym("mptwo")){
+ r = l;
+ goto Modmul;
+ }
+ cprint("mpexp(%N, %N, %N, %N);\n", l, r, modulo, t);
+ goto out;
+ }
+ }
+
+ switch(f->c){
+ case '+':
+ cprint("mpadd(%N, %N, %N);\n", l, r, t);
+ goto out;
+ case '-':
+ if(l->s == sym("mpzero")){
+ r = ecom(r, t);
+ cprint("%N->sign = -%N->sign;\n", t, t);
+ } else
+ cprint("mpsub(%N, %N, %N);\n", l, r, t);
+ goto out;
+ case '*':
+ Mul:
+ if(l->s == sym("mptwo") || r->s == sym("mptwo"))
+ cprint("mpleft(%N, 1, %N);\n", r->s == sym("mptwo") ? l : r, t);
+ else
+ cprint("mpmul(%N, %N, %N);\n", l, r, t);
+ goto out;
+ case '/':
+ cprint("mpdiv(%N, %N, %N, %N);\n", l, r, t, nil);
+ goto out;
+ case '%':
+ cprint("mpmod(%N, %N, %N);\n", l, r, t);
+ goto out;
+ case '^':
+ if(r->s == sym("mptwo")){
+ r = l;
+ goto Mul;
+ }
+ cprint("mpexp(%N, %N, nil, %N);\n", l, r, t);
+ goto out;
+ default:
+ diag(f, "unknown operation");
+ }
+
+out:
+ if(l != t)
+ freetmp(l);
+ if(r != t)
+ freetmp(r);
+ nodeset(t);
+ return t;
+}
+
+void
+bcom(Node *n, Node *t)
+{
+ Node *f, *l, *r;
+ int neg = 0;
+
+ l = r = nil;
+ f = n->l;
+Loop:
+ switch(f->c){
+ case '!':
+ neg = !neg;
+ f = f->l;
+ goto Loop;
+ case '>':
+ case '<':
+ case EQ:
+ l = ecom(f->l, nil);
+ r = ecom(f->r, nil);
+ if(t != nil) {
+ Node *b1, *b2;
+
+ b1 = ecom(n->r->l, nil);
+ b2 = ecom(n->r->r, nil);
+ cprint("mpsel(");
+
+ if(l->s == r->s)
+ cprint("0");
+ else {
+ if(f->c == '>')
+ cprint("-");
+ cprint("mpcmp(%N, %N)", l, r);
+ }
+ if(f->c == EQ)
+ neg = !neg;
+ else
+ cprint(" >> (sizeof(int)*8-1)");
+
+ cprint(", %N, %N, %N);\n", neg ? b2 : b1, neg ? b1 : b2, t);
+ freetmp(b1);
+ freetmp(b2);
+ } else {
+ cprint("if(");
+
+ if(l->s == r->s)
+ cprint("0");
+ else
+ cprint("mpcmp(%N, %N)", l, r);
+ if(f->c == EQ)
+ cprint(neg ? " != 0" : " == 0");
+ else if(f->c == '>')
+ cprint(neg ? " <= 0" : " > 0");
+ else
+ cprint(neg ? " >= 0" : " < 0");
+
+ cprint(")");
+ com(n->r);
+ }
+ break;
+ default:
+ diag(n, "saw %N in boolean expression", f);
+ }
+ freetmp(l);
+ freetmp(r);
+}
+
+void
+com(Node *n)
+{
+ Node *l, *r;
+
+Loop:
+ if(n != nil)
+ switch(n->c){
+ case '\n':
+ com(n->l);
+ n = n->r;
+ goto Loop;
+ case '?':
+ bcom(n, nil);
+ break;
+ case 'b':
+ for(l = atmps; l != nil; l = l->l)
+ cprint("mpfree(%N);\n", l);
+ cprint("break;\n");
+ break;
+ case '@':
+ cprint("for(;;)");
+ case ':':
+ clevel++;
+ cprint("{\n");
+ l = ftmps;
+ r = atmps;
+ if(n->c == '@')
+ atmps = nil;
+ ftmps = nil;
+ com(n->l);
+ if(n->r != nil){
+ cprint("}else{\n");
+ ftmps = nil;
+ com(n->r);
+ }
+ ftmps = l;
+ atmps = r;
+ clevel--;
+ cprint("}\n");
+ break;
+ case 'm':
+ l = modulo;
+ modulo = ecom(n->l, nil);
+ com(n->r);
+ freetmp(modulo);
+ modulo = l;
+ break;
+ case 'e':
+ if(n->r == nil)
+ cprint("%N();\n", n->l);
+ else {
+ r = ecom(n->r, nil);
+ cprint("%N(%N);\n", n->l, r);
+ freetmp(r);
+ }
+ break;
+ case '=':
+ ecom(n->r, n->l);
+ break;
+ }
+}
+
+Node*
+flocs(Node *n, Node *r)
+{
+Loop:
+ if(n != nil)
+ switch(n->c){
+ default:
+ r = flocs(n->l, r);
+ r = flocs(n->r, r);
+ n = n->r;
+ goto Loop;
+ case '=':
+ n = n->l;
+ if(n == nil)
+ diag(n, "lhs is nil");
+ while(n->c == ','){
+ n->c = '=';
+ r = flocs(n, r);
+ n->c = ',';
+ n = n->r;
+ if(n == nil)
+ return r;
+ }
+ if(n->c == NAME && (n->s->f & (FARG|FLOC)) == 0){
+ n->s->f = FLOC;
+ return new(',', n, r);
+ }
+ break;
+ }
+ return r;
+}
+
+void
+fcom(Node *f, Node *a, Node *b)
+{
+ Node *a0, *l0, *l;
+
+ ntmp = 0;
+ ftmps = atmps = modulo = nil;
+ clevel = 1;
+ cprint("void %N(", f);
+ a0 = a;
+ while(a != nil){
+ if(a != a0)
+ cprint(", ");
+ l = a->c == NAME ? a : a->l;
+ l->s->f = FARG|FSET;
+ cprint("mpint *%N", l);
+ a = a->r;
+ }
+ cprint("){\n");
+ l0 = flocs(b, nil);
+ for(a = l0; a != nil; a = a->r)
+ cprint("mpint *%N = mpnew(0);\n", a->l);
+ com(b);
+ for(a = l0; a != nil; a = a->r)
+ cprint("mpfree(%N);\n", a->l);
+ clevel = 0;
+ cprint("}\n");
+}
+
+void
+diag(Node *n, char *fmt, ...)
+{
+ static char buf[1024];
+ va_list a;
+
+ va_start(a, fmt);
+ vsnprint(buf, sizeof(buf), fmt, a);
+ va_end(a);
+
+ fprint(2, "%s:%d: for %N; %s\n", filename, n->n, n, buf);
+ exits("error");
+}
+
+int
+Nfmt(Fmt *f)
+{
+ Node *n = va_arg(f->args, Node*);
+
+ if(n == nil)
+ return fmtprint(f, "nil");
+
+ if(n->c == ',')
+ return fmtprint(f, "%N, %N", n->l, n->r);
+
+ switch(n->c){
+ case NAME:
+ case NUM:
+ return fmtprint(f, "%s", n->s->n);
+ case EQ:
+ return fmtprint(f, "==");
+ case IF:
+ return fmtprint(f, "if");
+ case ELSE:
+ return fmtprint(f, "else");
+ case MOD:
+ return fmtprint(f, "mod");
+ default:
+ return fmtprint(f, "%c", (char)n->c);
+ }
+}
+
+void
+parse(int fd, char *file)
+{
+ Binit(&bin, fd, OREAD);
+ filename = file;
+ clevel = 0;
+ lineno = 1;
+ goteof = 0;
+ while(!goteof)
+ yyparse();
+ Bterm(&bin);
+}
+
+void
+usage(void)
+{
+ fprint(2, "%s [file ...]\n", argv0);
+ exits("usage");
+}
+
+void
+main(int argc, char *argv[])
+{
+ fmtinstall('N', Nfmt);
+ fmtinstall('B', mpfmt);
+
+ ARGBEGIN {
+ default:
+ usage();
+ } ARGEND;
+
+ if(argc == 0){
+ parse(0, "<stdin>");
+ exits(nil);
+ }
+ while(*argv != nil){
+ int fd;
+
+ if((fd = open(*argv, OREAD)) < 0){
+ fprint(2, "%s: %r\n", *argv);
+ exits("error");
+ }
+ parse(fd, *argv);
+ close(fd);
+ argv++;
+ }
+ exits(nil);
+}