1
0
mirror of https://github.com/golang/go synced 2024-10-05 16:01:22 -06:00
go/src/cmd/gc/walk.c

3738 lines
63 KiB
C
Raw Normal View History

// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
#include "go.h"
static Node* curfn;
enum
{
Inone,
I2T,
2008-11-05 15:27:07 -07:00
I2T2,
I2I,
2008-11-05 15:27:07 -07:00
I2I2,
T2I,
2009-01-27 19:21:03 -07:00
I2Isame,
};
// can this code branch reach the end
// without an undcontitional RETURN
// this is hard, so it is conservative
int
walkret(Node *n)
{
loop:
if(n != N)
switch(n->op) {
case OLIST:
if(n->right == N) {
n = n->left;
goto loop;
}
n = n->right;
goto loop;
// at this point, we have the last
// statement of the function
case OGOTO:
case OPANIC:
case ORETURN:
return 0;
}
// all other statements
// will flow to the end
return 1;
}
void
walk(Node *fn)
{
char s[50];
curfn = fn;
if(debug['W']) {
snprint(s, sizeof(s), "\nbefore %S", curfn->nname->sym);
dump(s, curfn->nbody);
}
if(curfn->type->outtuple)
if(walkret(curfn->nbody))
yyerror("function ends without a return statement");
if(addtop != N) {
dump("addtop", addtop);
fatal("addtop in walk");
}
walkstate(curfn->nbody);
if(debug['W']) {
snprint(s, sizeof(s), "after walk %S", curfn->nname->sym);
dump(s, curfn->nbody);
}
heapmoves();
if(debug['W'] && curfn->enter != N) {
snprint(s, sizeof(s), "enter %S", curfn->nname->sym);
dump(s, curfn->enter);
}
}
void
addtotop(Node *n)
{
Node *l;
while(addtop != N) {
l = addtop;
addtop = N;
walktype(l, Etop);
n->ninit = list(n->ninit, l);
}
}
void
gettype(Node *n, Node *a)
{
if(debug['W'])
dump("\nbefore gettype", n);
walktype(n, Erv);
if(a == N && addtop != N)
fatal("gettype: addtop");
addtotop(a);
if(debug['W'])
dump("after gettype", n);
}
void
2008-09-04 13:21:10 -06:00
walkstate(Node *n)
{
Node *more;
2008-09-04 13:21:10 -06:00
loop:
if(n == N)
return;
more = N;
switch(n->op) {
case OLIST:
walkstate(n->left);
more = n->right;
break;
default:
yyerror("walkstate: %O not a top level statement", n->op);
case OASOP:
case OAS:
case OCALLMETH:
case OCALLINTER:
case OCALL:
case ODCL:
2008-09-04 13:21:10 -06:00
case OSEND:
case ORECV:
case OPRINT:
case OPRINTN:
2008-09-04 13:21:10 -06:00
case OPANIC:
case OPANICN:
2008-09-04 13:21:10 -06:00
case OFOR:
case OIF:
case OSWITCH:
case OSELECT:
case OEMPTY:
case OBREAK:
case OCONTINUE:
case OGOTO:
case OLABEL:
case OFALL:
case OXCASE:
case OCASE:
case OXFALL:
case ORETURN:
case OPROC:
2009-01-27 13:03:53 -07:00
case ODEFER:
2008-09-04 13:21:10 -06:00
walktype(n, Etop);
break;
}
addtotop(n);
2008-09-04 13:21:10 -06:00
if(more != N) {
n = more;
goto loop;
}
}
void
indir(Node *nl, Node *nr)
{
if(nr != N && nl != nr)
*nl = *nr;
}
void
implicitstar(Node **nn)
{
Type *t;
Node *n;
// insert implicit * if needed
n = *nn;
t = n->type;
if(t == T || !isptr[t->etype])
return;
t = t->type;
if(t == T)
return;
switch(t->etype) {
case TMAP:
case TSTRING:
case TARRAY:
break;
default:
return;
}
n = nod(OIND, n, N);
walktype(n, Elv);
*nn = n;
}
2008-09-04 13:21:10 -06:00
void
walktype(Node *n, int top)
{
Node *r, *l;
Type *t;
Sym *s;
2008-06-22 22:02:06 -06:00
int et, cl, cr;
int32 lno;
2008-06-28 18:27:39 -06:00
2008-10-29 21:25:34 -06:00
if(n == N)
return;
2008-06-28 18:27:39 -06:00
lno = setlineno(n);
/*
* walk the whole tree of the body of a function.
* the types expressions are calculated.
* compile-time constants are evaluated.
*/
loop:
if(n == N)
goto ret;
2008-06-28 18:27:39 -06:00
2008-06-26 18:54:44 -06:00
setlineno(n);
2008-06-17 19:07:40 -06:00
if(debug['w'] > 1 && top == Etop && n->op != OLIST)
dump("walk-before", n);
t = T;
et = Txxx;
switch(n->op) {
default:
fatal("walktype: switch 1 unknown op %N", n);
goto ret;
case ODCL:
goto ret;
case OLIST:
2008-09-03 15:40:22 -06:00
case OKEY:
walktype(n->left, top);
n = n->right;
goto loop;
case OPRINT:
if(top != Etop)
goto nottop;
walktype(n->left, Erv);
indir(n, prcompat(n->left, 0));
goto ret;
case OPRINTN:
if(top != Etop)
goto nottop;
walktype(n->left, Erv);
indir(n, prcompat(n->left, 1));
goto ret;
case OPANIC:
if(top != Etop)
goto nottop;
walktype(n->left, Erv);
indir(n, list(prcompat(n->left, 0), nodpanic(n->lineno)));
goto ret;
case OPANICN:
if(top != Etop)
goto nottop;
walktype(n->left, Erv);
indir(n, list(prcompat(n->left, 1), nodpanic(n->lineno)));
goto ret;
case OLITERAL:
if(top != Erv)
goto nottop;
n->addable = 1;
goto ret;
case ONONAME:
s = n->sym;
if(s->undef == 0) {
s->undef = 1;
yyerror("%S: undefined", s);
goto ret;
}
if(top == Etop)
goto nottop;
goto ret;
case ONAME:
if(top == Etop)
goto nottop;
if(!(n->class & PHEAP) && n->class != PPARAMREF)
n->addable = 1;
if(n->type == T) {
s = n->sym;
if(s->undef == 0) {
yyerror("walktype: %S undeclared", s);
s->undef = 1;
}
}
goto ret;
case OFOR:
if(top != Etop)
goto nottop;
2008-09-04 13:21:10 -06:00
walkstate(n->ninit);
walkbool(n->ntest);
2008-09-04 13:21:10 -06:00
walkstate(n->nincr);
walkstate(n->nbody);
goto ret;
case OSWITCH:
if(top != Etop)
goto nottop;
walkswitch(n);
goto ret;
case OSELECT:
if(top != Etop)
goto nottop;
walkselect(n);
goto ret;
case OIF:
if(top != Etop)
goto nottop;
2008-09-04 13:21:10 -06:00
walkstate(n->ninit);
walkbool(n->ntest);
2008-09-04 13:21:10 -06:00
walkstate(n->nbody);
walkstate(n->nelse);
2008-09-04 13:21:10 -06:00
goto ret;
2009-01-27 13:03:53 -07:00
case ODEFER:
hasdefer = 1;
2008-07-07 13:29:26 -06:00
case OPROC:
if(top != Etop)
goto nottop;
walktype(n->left, Etop);
2008-07-07 13:29:26 -06:00
goto ret;
case OCALLMETH:
case OCALLINTER:
case OCALL:
if(top == Elv)
goto nottop;
if(n->type != T)
goto ret;
if(n->left == N)
goto ret;
2009-02-16 18:44:05 -07:00
walktype(n->left, Erv);
convlit(n->left, types[TFUNC]);
t = n->left->type;
if(t == T)
goto ret;
if(n->left->op == ODOTMETH)
n->op = OCALLMETH;
if(n->left->op == ODOTINTER)
n->op = OCALLINTER;
if(isptr[t->etype])
t = t->type;
if(t->etype != TFUNC) {
yyerror("call of a non-function %T", t);
goto ret;
}
dowidth(t);
n->type = *getoutarg(t);
switch(t->outtuple) {
case 0:
if(top == Erv) {
yyerror("function requires a return type");
n->type = types[TINT];
}
break;
case 1:
n->type = n->type->type->type;
break;
}
walktype(n->right, Erv);
switch(n->op) {
default:
fatal("walk: op: %O", n->op);
case OCALLINTER:
l = ascompatte(n->op, getinarg(t), &n->right, 0);
2008-06-10 22:29:57 -06:00
n->right = reorder1(l);
break;
case OCALL:
l = ascompatte(n->op, getinarg(t), &n->right, 0);
2008-06-10 22:29:57 -06:00
n->right = reorder1(l);
if(isselect(n)) {
// clear output bool - special prob with selectsend
r = ascompatte(n->op, getoutarg(t), &boolfalse, 0);
n->right = list(n->right, r);
}
break;
case OCALLMETH:
l = ascompatte(n->op, getinarg(t), &n->right, 0);
r = ascompatte(n->op, getthis(t), &n->left->left, 0);
l = list(r, l);
n->left->left = N;
ullmancalc(n->left);
n->right = reorder1(l);
break;
}
goto ret;
case OAS:
if(top != Etop)
goto nottop;
addtop = list(addtop, n->ninit);
n->ninit = N;
l = n->left;
r = n->right;
walktype(l, Elv);
2008-06-22 22:02:06 -06:00
if(l == N || r == N)
goto ret;
2008-06-22 22:02:06 -06:00
cl = listcount(l);
cr = listcount(r);
if(cl == cr) {
walktype(r, Erv);
l = ascompatee(n->op, &n->left, &n->right);
if(l != N)
indir(n, reorder3(l));
2008-06-22 22:02:06 -06:00
goto ret;
}
switch(r->op) {
2008-07-13 15:29:46 -06:00
2008-06-22 22:02:06 -06:00
case OCALLMETH:
case OCALLINTER:
case OCALL:
if(cr == 1) {
// a,b,... = fn()
walktype(r, Erv);
2009-02-16 18:44:05 -07:00
convlit(r, types[TFUNC]);
l = ascompatet(n->op, &n->left, &r->type, 0);
if(l != N)
indir(n, list(r, reorder2(l)));
goto ret;
2008-06-10 22:29:57 -06:00
}
2008-06-22 22:02:06 -06:00
break;
2008-06-17 19:07:40 -06:00
2008-06-22 22:02:06 -06:00
case OINDEX:
if(cl == 2 && cr == 1) {
// a,b = map[] - mapaccess2
2008-09-11 20:09:25 -06:00
walktype(r->left, Erv);
if(!istype(r->left->type, TMAP))
break;
l = mapop(n, top);
if(l == N)
break;
indir(n, l);
goto ret;
}
break;
2008-07-13 15:29:46 -06:00
case ORECV:
if(cl == 2 && cr == 1) {
// a,b = <chan - chanrecv2
2008-07-15 22:07:59 -06:00
walktype(r->left, Erv);
if(!istype(r->left->type, TCHAN))
2008-07-13 15:29:46 -06:00
break;
l = chanop(n, top);
if(l == N)
break;
indir(n, l);
2008-07-13 15:29:46 -06:00
goto ret;
}
break;
2008-11-05 15:27:07 -07:00
case ODOTTYPE:
2008-11-05 15:27:07 -07:00
if(cl == 2 && cr == 1) {
// a,b = i.(T)
walktype(r->left, Erv);
2008-11-05 15:27:07 -07:00
if(r->left == N)
break;
et = ifaceas1(r->type, r->left->type, 1);
2008-11-05 15:27:07 -07:00
switch(et) {
case I2T:
et = I2T2;
break;
2009-01-27 19:21:03 -07:00
case I2Isame:
2008-11-05 15:27:07 -07:00
case I2I:
et = I2I2;
break;
default:
et = Inone;
break;
}
if(et == Inone)
break;
r = ifaceop(r->type, r->left, et);
l = ascompatet(n->op, &n->left, &r->type, 0);
if(l != N)
indir(n, list(r, reorder2(l)));
goto ret;
}
break;
}
switch(l->op) {
case OINDEX:
if(cl == 1 && cr == 2) {
// map[] = a,b - mapassign2
if(!istype(l->left->type, TMAP))
break;
l = mapop(n, top);
if(l == N)
break;
indir(n, l);
goto ret;
}
2008-06-22 22:02:06 -06:00
break;
}
if(l->diag == 0) {
l->diag = 1;
yyerror("assignment count mismatch: %d = %d", cl, cr);
}
goto ret;
case OBREAK:
case OCONTINUE:
case OGOTO:
case OLABEL:
if(top != Etop)
goto nottop;
goto ret;
case OXCASE:
if(top != Etop)
goto nottop;
yyerror("case statement out of place");
n->op = OCASE;
case OCASE:
if(top != Etop)
goto nottop;
2008-09-04 13:21:10 -06:00
walkstate(n->right);
goto ret;
case OXFALL:
if(top != Etop)
goto nottop;
yyerror("fallthrough statement out of place");
n->op = OFALL;
case OFALL:
case OINDREG:
2008-09-29 21:33:51 -06:00
case OEMPTY:
goto ret;
case OCONV:
case ODOTTYPE:
if(top != Erv)
goto nottop;
walkconv(n);
goto ret;
case OCOMPOS:
t = n->type;
if(t == T)
goto ret;
l = n->left;
if(l == N)
goto ret;
walktype(l, Erv);
// structure literal
if(t->etype == TSTRUCT) {
indir(n, structlit(n, N));
goto ret;
}
// array literal
if(t->etype == TARRAY) {
indir(n, arraylit(n, N));
goto ret;
}
// map literal
if(t->etype == TMAP) {
indir(n, maplit(n, N));
goto ret;
}
yyerror("invalid type for composite literal: %T", t);
goto ret;
case ORETURN:
if(top != Etop)
goto nottop;
walktype(n->left, Erv);
if(curfn->type->outnamed && n->left == N) {
// print("special return\n");
goto ret;
}
l = ascompatte(n->op, getoutarg(curfn->type), &n->left, 1);
if(l != N)
2008-06-10 22:29:57 -06:00
n->left = reorder4(l);
goto ret;
case ONOT:
if(top != Erv)
goto nottop;
walktype(n->left, Erv);
if(n->left == N || n->left->type == T)
goto ret;
et = n->left->type->etype;
break;
case OASOP:
if(top != Etop)
goto nottop;
walktype(n->left, Elv);
l = n->left;
if(l->op != OINDEX) {
if(n->etype == OLSH || n->etype == ORSH)
goto shft;
goto com;
}
if(!istype(l->left->type, TMAP))
goto com;
indir(n, mapop(n, top));
2008-09-04 13:21:10 -06:00
goto ret;
case OLSH:
case ORSH:
if(top != Erv)
goto nottop;
walktype(n->left, Erv);
shft:
walktype(n->right, Erv);
if(n->left == N || n->right == N)
goto ret;
evconst(n);
if(n->op == OLITERAL)
goto ret;
2008-10-29 21:25:34 -06:00
if(n->left->type == T)
convlit(n->left, types[TINT]);
if(n->right->type == T)
convlit(n->right, types[TUINT]);
if(n->left->type == T || n->right->type == T)
goto ret;
if(issigned[n->right->type->etype])
goto badt;
break;
case OMOD:
case OAND:
case OANDNOT:
case OOR:
case OXOR:
case OANDAND:
case OOROR:
case OEQ:
case ONE:
case OLT:
case OLE:
case OGE:
case OGT:
case OADD:
case OSUB:
case OMUL:
case ODIV:
if(top != Erv)
goto nottop;
walktype(n->left, Erv);
com:
walktype(n->right, Erv);
if(n->left == N || n->right == N)
goto ret;
evconst(n);
if(n->op == OLITERAL)
goto ret;
2008-07-05 18:43:25 -06:00
convlit(n->left, n->right->type);
convlit(n->right, n->left->type);
if(n->left->type == T || n->right->type == T)
goto ret;
if(!eqtype(n->left->type, n->right->type, 0))
goto badt;
switch(n->op) {
case OANDNOT:
n->op = OAND;
n->right = nod(OCOM, n->right, N);
n->right->type = n->right->left->type;
break;
case OASOP:
if(n->etype == OANDNOT) {
n->etype = OAND;
n->right = nod(OCOM, n->right, N);
n->right->type = n->right->left->type;
break;
}
case OEQ:
case ONE:
case OLT:
case OLE:
case OGE:
case OGT:
case OADD:
if(istype(n->left->type, TSTRING)) {
indir(n, stringop(n, top));
goto ret;
}
break;
}
break;
case OMINUS:
case OPLUS:
case OCOM:
if(top != Erv)
goto nottop;
walktype(n->left, Erv);
if(n->left == N)
goto ret;
evconst(n);
if(n->op == OLITERAL)
goto ret;
break;
case OLEN:
if(top != Erv)
goto nottop;
walktype(n->left, Erv);
implicitstar(&n->left);
t = n->left->type;
if(t == T)
goto ret;
switch(t->etype) {
default:
goto badt;
case TSTRING:
2009-03-10 20:16:31 -06:00
if(whatis(n->left) == Wlitstr)
nodconst(n, types[TINT], n->left->val.u.sval->len);
break;
case TMAP:
2008-07-18 12:59:35 -06:00
break;
case TARRAY:
if(t->bound >= 0)
nodconst(n, types[TINT], t->bound);
break;
}
n->type = types[TINT];
goto ret;
2008-08-27 18:28:30 -06:00
case OCAP:
if(top != Erv)
goto nottop;
walktype(n->left, Erv);
implicitstar(&n->left);
2008-08-27 18:28:30 -06:00
evconst(n);
t = n->left->type;
if(t == T)
goto ret;
switch(t->etype) {
default:
goto badt;
case TARRAY:
if(t->bound >= 0)
nodconst(n, types[TINT], t->bound);
2008-08-27 18:28:30 -06:00
break;
}
n->type = types[TINT];
2008-08-27 18:28:30 -06:00
goto ret;
case OINDEX:
if(top == Etop)
goto nottop;
2008-06-27 17:30:20 -06:00
walktype(n->left, Erv);
walktype(n->right, Erv);
if(n->left == N || n->right == N)
goto ret;
2008-06-16 23:34:50 -06:00
defaultlit(n->left);
implicitstar(&n->left);
t = n->left->type;
if(t == T)
goto ret;
switch(t->etype) {
default:
goto badt;
case TSTRING:
2008-07-05 18:43:25 -06:00
// right side must be an int
if(top != Erv)
goto nottop;
if(n->right->type == T) {
convlit(n->right, types[TINT]);
2008-07-05 18:43:25 -06:00
if(n->right->type == T)
break;
2008-07-05 18:43:25 -06:00
}
if(!isint[n->right->type->etype])
goto badt;
indir(n, stringop(n, top));
break;
2008-06-16 23:34:50 -06:00
case TMAP:
2008-07-13 15:29:46 -06:00
// right side must be map type
2008-06-16 23:34:50 -06:00
if(n->right->type == T) {
convlit(n->right, t->down);
if(n->right->type == T)
break;
}
if(!eqtype(n->right->type, t->down, 0))
goto badt;
n->type = t->type;
if(top == Erv)
indir(n, mapop(n, top));
2008-06-16 23:34:50 -06:00
break;
2008-08-27 18:28:30 -06:00
case TARRAY:
2008-06-16 23:34:50 -06:00
// right side must be an int
if(n->right->type == T) {
convlit(n->right, types[TINT]);
2008-06-16 23:34:50 -06:00
if(n->right->type == T)
break;
}
if(!isint[n->right->type->etype])
goto badt;
2008-06-16 23:34:50 -06:00
n->type = t->type;
break;
}
goto ret;
2008-07-13 15:29:46 -06:00
case OSEND:
2008-07-15 22:07:59 -06:00
if(top == Elv)
2008-07-13 15:29:46 -06:00
goto nottop;
walktype(n->left, Erv); // chan
walktype(n->right, Erv); // e
indir(n, chanop(n, top));
2008-07-13 15:29:46 -06:00
goto ret;
case ORECV:
2008-07-15 22:07:59 -06:00
if(top == Elv)
2008-07-13 15:29:46 -06:00
goto nottop;
2008-07-15 22:07:59 -06:00
if(n->right == N) {
walktype(n->left, Erv); // chan
indir(n, chanop(n, top)); // returns e blocking
2008-07-15 22:07:59 -06:00
goto ret;
}
walktype(n->left, Elv); // e
walktype(n->right, Erv); // chan
indir(n, chanop(n, top)); // returns bool non-blocking
2008-07-13 15:29:46 -06:00
goto ret;
case OSLICE:
if(top == Etop)
goto nottop;
walktype(n->left, top);
walktype(n->right, Erv);
if(n->left == N || n->right == N)
goto ret;
convlit(n->left, types[TSTRING]);
implicitstar(&n->left);
2008-08-27 18:28:30 -06:00
t = n->left->type;
if(t == T)
goto ret;
2008-08-27 18:28:30 -06:00
if(t->etype == TSTRING) {
indir(n, stringop(n, top));
goto ret;
}
if(t->etype == TARRAY) {
indir(n, arrayop(n, top));
2008-08-27 18:28:30 -06:00
goto ret;
}
badtype(OSLICE, n->left->type, T);
goto ret;
case ODOT:
case ODOTPTR:
case ODOTMETH:
case ODOTINTER:
if(top == Etop)
goto nottop;
walkdot(n);
goto ret;
case OADDR:
if(top != Erv)
goto nottop;
if(n->left->op == OCOMPOS && n->left->type != T) {
// turn &Point(1, 2) or &[]int(1, 2) or &[...]int(1, 2) into allocation.
// initialize with
// nvar := new(*Point);
// *nvar = Point(1, 2);
// and replace expression with nvar
; // stupid c syntax - case label must be on stmt, not decl
Node *nvar, *nas, *nstar;
nvar = nod(OXXX, N, N);
tempname(nvar, ptrto(n->left->type));
nas = nod(OAS, nvar, callnew(n->left->type));
addtop = list(addtop, nas);
nstar = nod(OIND, nvar, N);
nstar->type = n->left->type;
switch(n->left->type->etype) {
case TSTRUCT:
structlit(n->left, nstar);
break;
case TARRAY:
arraylit(n->left, nstar);
break;
case TMAP:
maplit(n->left, nstar);
break;
default:
goto badlit;
}
indir(n, nvar);
goto ret;
}
badlit:
if(istype(n->left->type, TFUNC) && n->left->class == PFUNC) {
if(!n->diag) {
n->diag = 1;
yyerror("cannot take address of function");
}
}
walktype(n->left, Elv);
addrescapes(n->left);
if(n->left == N)
goto ret;
t = n->left->type;
if(t == T)
goto ret;
n->type = ptrto(t);
goto ret;
case OIND:
if(top == Etop)
goto nottop;
if(top == Elv) // even if n is lvalue, n->left is rvalue
top = Erv;
walktype(n->left, top);
if(n->left == N)
goto ret;
t = n->left->type;
if(t == T)
goto ret;
if(!isptr[t->etype])
goto badt;
n->type = t->type;
goto ret;
case OMAKE:
if(top != Erv)
goto nottop;
indir(n, makecompat(n));
goto ret;
case ONEW:
if(top != Erv)
goto nottop;
if(n->left != N) {
yyerror("cannot new(%T, expr)", t);
goto ret;
}
t = n->type;
if(t == T)
goto ret;
indir(n, callnew(t));
goto ret;
}
/*
* ======== second switch ========
*/
switch(n->op) {
default:
fatal("walktype: switch 2 unknown op %N", n);
goto ret;
case OASOP:
break;
case ONOT:
case OANDAND:
case OOROR:
if(n->left->type == T)
goto ret;
et = n->left->type->etype;
if(et != TBOOL)
goto badt;
t = types[TBOOL];
break;
case OEQ:
case ONE:
if(n->left->type == T)
goto ret;
if(isslice(n->left->type)) {
2008-12-18 22:33:45 -07:00
t = types[TBOOL];
break;
}
et = n->left->type->etype;
if(!okforeq[et])
goto badt;
if(isinter(n->left->type)) {
indir(n, ifaceop(T, n, n->op));
goto ret;
}
t = types[TBOOL];
break;
case OLT:
case OLE:
case OGE:
case OGT:
if(n->left->type == T)
goto ret;
et = n->left->type->etype;
if(!okforadd[et] && et != TSTRING)
goto badt;
t = types[TBOOL];
break;
case OADD:
case OSUB:
case OMUL:
case ODIV:
case OPLUS:
if(n->left->type == T)
goto ret;
et = n->left->type->etype;
if(!okforadd[et])
goto badt;
break;
case OMINUS:
if(n->left->type == T)
goto ret;
et = n->left->type->etype;
if(!okforadd[et])
goto badt;
if(!isfloat[et])
break;
l = nod(OLITERAL, N, N);
l->val.u.fval = mal(sizeof(*l->val.u.fval));
l->val.ctype = CTFLT;
mpmovecflt(l->val.u.fval, 0.0);
l = nod(OSUB, l, n->left);
indir(n, l);
walktype(n, Erv);
goto ret;
case OLSH:
case ORSH:
case OAND:
case OANDNOT:
case OOR:
case OXOR:
case OMOD:
case OCOM:
if(n->left->type == T)
goto ret;
et = n->left->type->etype;
if(!okforand[et])
goto badt;
break;
}
if(t == T)
t = n->left->type;
n->type = t;
goto ret;
nottop:
2008-11-22 18:58:53 -07:00
switch(top) {
default:
yyerror("didn't expect %O here", n->op);
break;
case Etop:
yyerror("operation %O not allowed in statement context", n->op);
break;
case Elv:
yyerror("operation %O not allowed in assignment context", n->op);
break;
case Erv:
yyerror("operation %O not allowed in expression context", n->op);
break;
}
2008-06-10 22:29:57 -06:00
goto ret;
badt:
if(n->right == N) {
if(n->left == N) {
badtype(n->op, T, T);
goto ret;
}
badtype(n->op, n->left->type, T);
goto ret;
}
badtype(n->op, n->left->type, n->right->type);
goto ret;
ret:
2008-06-17 19:07:40 -06:00
if(debug['w'] && top == Etop && n != N)
dump("walk", n);
2008-06-16 23:34:50 -06:00
ullmancalc(n);
2008-06-26 18:54:44 -06:00
lineno = lno;
}
void
walkbool(Node *n)
{
walktype(n, Erv);
addtotop(n);
if(n != N && n->type != T)
if(!eqtype(n->type, types[TBOOL], 0))
yyerror("IF and FOR require a boolean type");
}
void
walkconv(Node *n)
{
int et;
Type *t;
Node *l;
t = n->type;
if(t == T)
return;
l = n->left;
if(l == N)
return;
walktype(l, Erv);
convlit1(l, t, 1);
// if using .(T), interface assertion.
if(n->op == ODOTTYPE) {
// interface conversion
et = ifaceas(n->type, l->type, 1);
if(et != Inone) {
indir(n, ifaceop(n->type, l, et));
return;
}
goto bad;
}
// otherwise, conversion.
// nil conversion
if(eqtype(t, l->type, 0)) {
if(l->op != ONAME) {
indir(n, l);
n->type = t;
}
return;
}
// simple fix-float
if(l->type != T)
if(isint[l->type->etype] || isfloat[l->type->etype])
if(isint[t->etype] || isfloat[t->etype]) {
evconst(n);
return;
}
// to string
if(l->type != T)
if(istype(t, TSTRING)) {
et = l->type->etype;
if(isint[et]) {
indir(n, stringop(n, Erv));
return;
}
if(et == TARRAY)
if(istype(l->type->type, TUINT8)) {
n->op = OARRAY;
indir(n, stringop(n, Erv));
return;
}
}
// convert dynamic to static generated by ONEW/OMAKE
if(isfixedarray(t) && isslice(l->type))
return;
// convert static array to dynamic array
if(isslice(t) && isfixedarray(l->type)) {
if(eqtype(t->type->type, l->type->type->type, 0)) {
indir(n, arrayop(n, Erv));
return;
}
}
// convert to unsafe.pointer
if(isptrto(n->type, TANY)) {
if(isptr[l->type->etype])
return;
if(l->type->etype == TUINTPTR)
return;
}
// convert from unsafe.pointer
if(isptrto(l->type, TANY)) {
if(isptr[n->type->etype])
return;
if(n->type->etype == TUINTPTR)
return;
}
bad:
if(l->type != T)
yyerror("invalid conversion: %T to %T", l->type, t);
else
if(n->left->op == OLIST)
yyerror("invalid type for composite literal: %T", t);
}
Node*
selcase(Node *n, Node *var)
{
Node *a, *r, *on, *c;
Type *t;
2008-11-05 22:50:28 -07:00
if(n->left == N)
goto dflt;
c = n->left;
if(c->op == ORECV)
goto recv;
walktype(c->left, Erv); // chan
walktype(c->right, Erv); // elem
t = fixchan(c->left->type);
if(t == T)
return N;
convlit(c->right, t->type);
if(!ascompat(t->type, c->right->type)) {
badtype(c->op, t->type, c->right->type);
return N;
}
// selectsend(sel *byte, hchan *chan any, elem any) (selected bool);
on = syslook("selectsend", 1);
argtype(on, t->type);
argtype(on, t->type);
a = c->right; // elem
r = a;
a = c->left; // chan
r = list(a, r);
a = var; // sel-var
r = list(a, r);
goto out;
recv:
if(c->right != N)
goto recv2;
walktype(c->left, Erv); // chan
t = fixchan(c->left->type);
if(t == T)
return N;
// selectrecv(sel *byte, hchan *chan any, elem *any) (selected bool);
on = syslook("selectrecv", 1);
argtype(on, t->type);
argtype(on, t->type);
a = c->left; // nil elem
a = nod(OLITERAL, N, N);
a->val.ctype = CTNIL;
r = a;
a = c->left; // chan
r = list(a, r);
a = var; // sel-var
r = list(a, r);
goto out;
recv2:
walktype(c->right, Erv); // chan
t = fixchan(c->right->type);
if(t == T)
return N;
walktype(c->left, Elv); // elem
convlit(c->left, t->type);
if(!ascompat(t->type, c->left->type)) {
badtype(c->op, t->type, c->left->type);
return N;
}
// selectrecv(sel *byte, hchan *chan any, elem *any) (selected bool);
on = syslook("selectrecv", 1);
argtype(on, t->type);
argtype(on, t->type);
a = c->left; // elem
a = nod(OADDR, a, N);
r = a;
a = c->right; // chan
r = list(a, r);
a = var; // sel-var
r = list(a, r);
2008-11-05 22:50:28 -07:00
goto out;
dflt:
// selectdefault(sel *byte);
on = syslook("selectdefault", 0);
a = var;
r = a; // sel-var
goto out;
out:
a = nod(OCALL, on, r);
r = nod(OIF, N, N);
r->ntest = a;
return r;
}
/*
* enumerate the special cases
* of the case statement:
* case v := <-chan // select and switch
* case v := map[] // switch
* case v := interface.(TYPE) // switch
*/
Node*
selectas(Node *name, Node *expr)
{
Node *a;
Type *t;
if(expr == N)
goto bad;
switch(expr->op) {
default:
//dump("case", expr);
goto bad;
case ORECV:
walktype(expr->left, Erv);
t = expr->left->type;
if(t == T)
goto bad;
if(t->etype != TCHAN)
goto bad;
t = t->type;
break;
case OINDEX:
walktype(expr->left, Erv);
walktype(expr->right, Erv);
t = expr->left->type;
if(t == T)
goto bad;
if(t->etype != TMAP)
goto bad;
t = t->type;
break;
case ODOTTYPE:
walktype(expr->left, Erv);
t = expr->left->type;
if(t == T)
goto bad;
if(t->etype != TINTER)
goto bad;
t = expr->type;
break;
}
a = old2new(name, t);
return a;
bad:
return name;
}
void
walkselect(Node *sel)
{
Iter iter;
Node *n, *l, *oc, *on, *r;
Node *var, *bod, *nbod, *res, *def;
2008-11-05 22:50:28 -07:00
int count, op;
int32 lno;
lno = setlineno(sel);
// generate sel-struct
var = nod(OXXX, N, N);
tempname(var, ptrto(types[TUINT8]));
n = listfirst(&iter, &sel->left);
if(n == N || n->op != OXCASE)
yyerror("first select statement must be a case");
count = 0; // number of cases
res = N; // entire select body
bod = N; // body of each case
oc = N; // last case
2008-11-05 22:50:28 -07:00
def = N; // default case
for(; n!=N; n=listnext(&iter)) {
setlineno(n);
switch(n->op) {
default:
bod = list(bod, n);
break;
case OXCASE:
2008-11-05 22:50:28 -07:00
if(n->left == N) {
op = ORECV; // actual value not used
if(def != N)
yyerror("only one default select allowed");
def = n;
} else
op = n->left->op;
nbod = N;
2008-11-05 22:50:28 -07:00
switch(op) {
default:
2008-11-05 22:50:28 -07:00
yyerror("select cases must be send, recv or default");
break;
case OAS:
// convert new syntax (a=recv(chan)) to (recv(a,chan))
l = n->left;
if(l->right == N || l->right->op != ORECV) {
2008-11-05 22:50:28 -07:00
yyerror("select cases must be send, recv or default");
break;
}
r = l->right; // rcv
r->right = r->left;
r->left = l->left;
n->left = r;
// convert case x := foo: body
// to case tmp := foo: x := tmp; body.
// if x escapes and must be allocated
// on the heap, this delays the allocation
// until after the select has chosen this branch.
if(n->ninit != N && n->ninit->op == ODCL) {
on = nod(OXXX, N, N);
tempname(on, l->left->type);
on->sym = lookup("!tmpselect!");
r->left = on;
nbod = nod(OAS, l->left, on);
nbod->ninit = n->ninit;
n->ninit = N;
}
// fall through
case OSEND:
case ORECV:
if(oc != N) {
bod = list(bod, nod(OBREAK, N, N));
oc->nbody = rev(bod);
}
oc = selcase(n, var);
res = list(res, oc);
break;
}
bod = nbod;
count++;
break;
}
}
if(oc != N) {
bod = list(bod, nod(OBREAK, N, N));
oc->nbody = rev(bod);
}
setlineno(sel);
// selectgo(sel *byte);
on = syslook("selectgo", 0);
r = nod(OCALL, on, var); // sel-var
res = list(res, r);
// newselect(size uint32) (sel *byte);
on = syslook("newselect", 0);
r = nod(OXXX, N, N);
nodconst(r, types[TINT], count); // count
r = nod(OCALL, on, r);
r = nod(OAS, var, r);
sel->ninit = r;
sel->nbody = rev(res);
sel->left = N;
2008-09-04 13:21:10 -06:00
walkstate(sel->ninit);
walkstate(sel->nbody);
//dump("sel", sel);
lineno = lno;
}
Type*
lookdot1(Sym *s, Type *t, Type *f)
{
Type *r;
r = T;
for(; f!=T; f=f->down) {
if(f->sym == S)
continue;
if(f->sym != s)
continue;
if(r != T) {
yyerror("ambiguous DOT reference %T.%S", t, s);
break;
}
r = f;
}
return r;
}
int
lookdot(Node *n, Type *t)
{
Type *f1, *f2, *tt;
int op;
Sym *s;
s = n->right->sym;
f1 = T;
if(t->etype == TSTRUCT || t->etype == TINTER)
f1 = lookdot1(s, t, t->type);
f2 = methtype(n->left->type);
if(f2 != T)
f2 = lookdot1(s, f2, f2->method);
if(f1 != T) {
if(f2 != T)
yyerror("ambiguous DOT reference %S as both field and method",
n->right->sym);
n->right = f1->nname; // substitute real name
n->xoffset = f1->width;
n->type = f1->type;
if(t->etype == TINTER)
n->op = ODOTINTER;
return 1;
}
if(f2 != T) {
tt = n->left->type;
if((op = methconv(tt)) != 0) {
switch(op) {
case OADDR:
walktype(n->left, Elv);
addrescapes(n->left);
n->left = nod(OADDR, n->left, N);
n->left->type = ptrto(tt);
break;
case OIND:
n->left = nod(OIND, n->left, N);
n->left->type = tt->type;
break;
}
}
n->right = methodname(n->right, n->left->type);
n->xoffset = f2->width;
n->type = f2->type;
n->op = ODOTMETH;
return 1;
}
return 0;
}
void
walkdot(Node *n)
{
Type *t;
addtop = list(addtop, n->ninit);
n->ninit = N;
if(n->left == N || n->right == N)
return;
switch(n->op) {
case ODOTINTER:
case ODOTMETH:
return; // already done
}
walktype(n->left, Erv);
if(n->right->op != ONAME) {
yyerror("rhs of . must be a name");
return;
}
t = n->left->type;
if(t == T)
return;
// as a structure field or pointer to structure field
if(isptr[t->etype]) {
t = t->type;
if(t == T)
return;
n->op = ODOTPTR;
}
if(!lookdot(n, t))
yyerror("undefined DOT %S on %T", n->right->sym, n->left->type);
}
Node*
ascompatee(int op, Node **nl, Node **nr)
{
Node *l, *r, *nn, *a;
Iter savel, saver;
2008-06-10 22:29:57 -06:00
/*
* check assign expression list to
* a expression list. called in
* expr-list = expr-list
*/
l = listfirst(&savel, nl);
r = listfirst(&saver, nr);
nn = N;
loop:
if(l == N || r == N) {
if(l != r)
2008-07-05 18:43:25 -06:00
yyerror("error in shape across %O", op);
2008-06-10 22:29:57 -06:00
return rev(nn);
}
convlit(r, l->type);
if(!ascompat(l->type, r->type)) {
badtype(op, l->type, r->type);
return N;
}
if(l->op == ONAME && l->class == PFUNC)
yyerror("cannot assign to function");
a = nod(OAS, l, r);
a = convas(a);
2008-11-01 17:36:46 -06:00
nn = list(a, nn);
l = listnext(&savel);
r = listnext(&saver);
goto loop;
}
Node*
ascompatet(int op, Node **nl, Type **nr, int fp)
{
Node *l, *nn, *a;
Type *r;
Iter savel, saver;
2008-06-10 22:29:57 -06:00
/*
* check assign type list to
* a expression list. called in
* expr-list = func()
*/
l = listfirst(&savel, nl);
r = structfirst(&saver, nr);
nn = N;
loop:
if(l == N || r == T) {
if(l != N || r != T)
2008-07-05 18:43:25 -06:00
yyerror("error in shape across %O", op);
2008-06-10 22:29:57 -06:00
return rev(nn);
}
if(!ascompat(l->type, r->type)) {
badtype(op, l->type, r->type);
return N;
}
a = nod(OAS, l, nodarg(r, fp));
a = convas(a);
2008-11-01 17:36:46 -06:00
nn = list(a, nn);
l = listnext(&savel);
r = structnext(&saver);
goto loop;
}
2008-11-01 17:36:46 -06:00
/*
* make a tsig for the structure
* carrying the ... arguments
*/
Type*
sigtype(Type *st)
{
Dcl *x;
Sym *s;
Type *t;
static int sigdddgen;
dowidth(st);
sigdddgen++;
snprint(namebuf, sizeof(namebuf), "dsigddd_%d", sigdddgen);
s = lookup(namebuf);
t = newtype(s);
t = dodcltype(t);
updatetype(t, st);
t->local = 1;
2008-11-01 17:36:46 -06:00
// record internal type for signature generation
x = mal(sizeof(*x));
x->op = OTYPE;
x->dsym = s;
x->dtype = s->otype;
x->forw = signatlist;
x->block = block;
signatlist = x;
return s->otype;
}
/*
* package all the arguments that
* match a ... parameter into an
* automatic structure.
* then call the ... arg (interface)
* with a pointer to the structure.
2008-11-01 17:36:46 -06:00
*/
Node*
mkdotargs(Node *r, Node *rr, Iter *saver, Node *nn, Type *l, int fp)
2008-11-01 17:36:46 -06:00
{
Type *t, *st, *ft;
Node *a, *n, *var;
Iter saven;
n = N; // list of assignments
st = typ(TSTRUCT); // generated structure
ft = T; // last field
while(r != N) {
defaultlit(r);
// generate the next structure field
t = typ(TFIELD);
t->type = r->type;
if(ft == T)
st->type = t;
else
ft->down = t;
ft = t;
a = nod(OAS, N, r);
n = list(n, a);
if(rr != N) {
r = rr;
rr = N;
} else
r = listnext(saver);
2008-11-01 17:36:46 -06:00
}
// make a named type for the struct
st = sigtype(st);
dowidth(st);
2008-11-01 17:36:46 -06:00
// now we have the size, make the struct
var = nod(OXXX, N, N);
tempname(var, st);
var->sym = lookup(".ddd");
2008-11-01 17:36:46 -06:00
// assign the fields to the struct.
// use addtop so that reorder1 doesn't reorder
// these assignments after the interface conversion
// below.
2008-11-01 17:36:46 -06:00
n = rev(n);
r = listfirst(&saven, &n);
t = st->type;
while(r != N) {
r->left = nod(OXXX, N, N);
*r->left = *var;
r->left->type = r->right->type;
r->left->xoffset += t->width;
addtop = list(addtop, r);
2008-11-01 17:36:46 -06:00
r = listnext(&saven);
t = t->down;
}
// last thing is to put assignment
// of the structure to the DDD parameter
a = nod(OAS, nodarg(l, fp), var);
nn = list(convas(a), nn);
2008-11-01 17:36:46 -06:00
return nn;
}
/*
* check assign expression list to
* a type list. called in
* return expr-list
* func(expr-list)
*/
Node*
ascompatte(int op, Type **nl, Node **nr, int fp)
{
Type *l, *ll;
Node *r, *rr, *nn, *a;
Iter savel, saver, peekl, peekr;
l = structfirst(&savel, nl);
r = listfirst(&saver, nr);
nn = N;
2008-11-01 17:36:46 -06:00
// 1 to many
peekl = savel;
peekr = saver;
if(l != T && r != N
&& structnext(&peekl) != T
2009-01-08 15:31:11 -07:00
&& listnext(&peekr) == N
&& eqtypenoname(r->type, *nl)) {
// clumsy check for differently aligned structs.
// need to handle eventually, but this keeps us
// from inserting bugs
if(r->type->width != (*nl)->width)
yyerror("misaligned multiple return (6g's fault)");
a = nodarg(*nl, fp);
a->type = r->type;
return convas(nod(OAS, a, r));
}
loop:
2008-11-01 17:36:46 -06:00
if(l != T && isddd(l->type)) {
// the ddd parameter must be last
ll = structnext(&savel);
if(ll != T)
yyerror("... must be last argument");
// special case --
// only if we are assigning a single ddd
// argument to a ddd parameter then it is
// passed thru unencapsulated
rr = listnext(&saver);
if(r != N && rr == N && isddd(r->type)) {
a = nod(OAS, nodarg(l, fp), r);
a = convas(a);
nn = list(a, nn);
return rev(nn);
2008-11-01 17:36:46 -06:00
}
// normal case -- make a structure of all
// remaining arguments and pass a pointer to
// it to the ddd parameter (empty interface)
nn = mkdotargs(r, rr, &saver, nn, l, fp);
2008-11-01 17:36:46 -06:00
return rev(nn);
}
if(l == T || r == N) {
if(l != T || r != N)
2008-07-05 18:43:25 -06:00
yyerror("error in shape across %O", op);
2008-06-10 22:29:57 -06:00
return rev(nn);
}
convlit(r, l->type);
if(!ascompat(l->type, r->type)) {
badtype(op, l->type, r->type);
return N;
}
a = nod(OAS, nodarg(l, fp), r);
a = convas(a);
2008-11-01 17:36:46 -06:00
nn = list(a, nn);
l = structnext(&savel);
r = listnext(&saver);
goto loop;
}
/*
* can we assign var of type src to var of type dst
*/
int
ascompat(Type *dst, Type *src)
{
if(eqtype(dst, src, 0))
return 1;
if(isslice(dst) && isfixedarray(src))
return 1;
if(isnilinter(dst) || isnilinter(src))
return 1;
if(isinter(dst) && isinter(src))
return 1;
if(isinter(dst) && methtype(src))
return 1;
if(isinter(src) && methtype(dst))
return 1;
return 0;
}
Node*
prcompat(Node *n, int fmt)
{
Node *l, *r;
Node *on;
Type *t;
Iter save;
int w, notfirst;
r = N;
l = listfirst(&save, &n);
notfirst = 0;
loop:
if(l == N) {
if(fmt) {
on = syslook("printnl", 0);
r = list(r, nod(OCALL, on, N));
}
walktype(r, Etop);
return r;
}
if(notfirst) {
on = syslook("printsp", 0);
r = list(r, nod(OCALL, on, N));
}
w = whatis(l);
switch(w) {
default:
if(l->type == T)
goto out;
2008-09-22 17:58:30 -06:00
if(isinter(l->type)) {
on = syslook("printinter", 1);
argtype(on, l->type); // any-1
break;
}
if(isptr[l->type->etype]
|| l->type->etype == TCHAN
|| l->type->etype == TMAP
|| l->type->etype == TFUNC) {
2008-09-22 17:58:30 -06:00
on = syslook("printpointer", 1);
argtype(on, l->type); // any-1
2008-09-22 17:58:30 -06:00
break;
}
if(isslice(l->type)) {
2008-12-18 23:17:05 -07:00
on = syslook("printarray", 1);
argtype(on, l->type); // any-1
break;
}
badtype(OPRINT, l->type, T);
2008-09-22 17:58:30 -06:00
l = listnext(&save);
goto loop;
case Wlitint:
case Wtint:
on = syslook("printint", 0);
break;
case Wlitfloat:
case Wtfloat:
on = syslook("printfloat", 0);
break;
case Wlitbool:
case Wtbool:
on = syslook("printbool", 0);
break;
case Wlitstr:
case Wtstr:
on = syslook("printstring", 0);
break;
}
t = *getinarg(on->type);
if(t != nil)
t = t->type;
if(t != nil)
t = t->type;
if(!eqtype(t, l->type, 0)) {
l = nod(OCONV, l, N);
l->type = t;
}
r = list(r, nod(OCALL, on, l));
out:
notfirst = fmt;
l = listnext(&save);
goto loop;
}
Node*
nodpanic(int32 lineno)
{
Node *n, *on;
on = syslook("panicl", 0);
n = nodintconst(lineno);
n = nod(OCALL, on, n);
walktype(n, Etop);
return n;
}
Node*
makecompat(Node *n)
{
Type *t;
2008-12-18 21:06:28 -07:00
t = n->type;
if(t != T)
switch(t->etype) {
case TARRAY:
return arrayop(n, Erv);
case TMAP:
return mapop(n, Erv);
case TCHAN:
return chanop(n, Erv);
}
/*
* ken had code to malloc here,
* but rsc cut it out so that make(int)
* is diagnosed as an error (probably meant new).
* might come back once we know the
* language semantics for make(int).
*/
yyerror("cannot make(%T)", t);
return n;
}
Node*
callnew(Type *t)
{
Node *r, *on;
dowidth(t);
on = syslook("mal", 1);
argtype(on, t);
r = nodintconst(t->width);
r = nod(OCALL, on, r);
walktype(r, Erv);
return r;
}
Node*
stringop(Node *n, int top)
{
Node *r, *c, *on;
switch(n->op) {
default:
2008-07-13 15:29:46 -06:00
fatal("stringop: unknown op %O", n->op);
case OEQ:
case ONE:
case OGE:
case OGT:
case OLE:
case OLT:
// sys_cmpstring(s1, s2) :: 0
on = syslook("cmpstring", 0);
r = list(n->left, n->right);
r = nod(OCALL, on, r);
c = nodintconst(0);
r = nod(n->op, r, c);
break;
case OADD:
// sys_catstring(s1, s2)
on = syslook("catstring", 0);
r = list(n->left, n->right);
r = nod(OCALL, on, r);
break;
case OASOP:
// sys_catstring(s1, s2)
switch(n->etype) {
default:
2008-07-13 15:29:46 -06:00
fatal("stringop: unknown op %O-%O", n->op, n->etype);
case OADD:
// s1 = sys_catstring(s1, s2)
if(n->etype != OADD)
fatal("stringop: not cat");
r = list(n->left, n->right);
on = syslook("catstring", 0);
r = nod(OCALL, on, r);
r = nod(OAS, n->left, r);
break;
}
break;
case OSLICE:
// sys_slicestring(s, lb, hb)
r = nod(OCONV, n->right->left, N);
r->type = types[TINT];
c = nod(OCONV, n->right->right, N);
c->type = types[TINT];
r = list(r, c);
r = list(n->left, r);
on = syslook("slicestring", 0);
r = nod(OCALL, on, r);
break;
2008-07-05 18:43:25 -06:00
case OINDEX:
// sys_indexstring(s, i)
r = nod(OCONV, n->right, N);
r->type = types[TINT];
r = list(n->left, r);
on = syslook("indexstring", 0);
r = nod(OCALL, on, r);
break;
case OCONV:
// sys_intstring(v)
r = nod(OCONV, n->left, N);
r->type = types[TINT64];
on = syslook("intstring", 0);
r = nod(OCALL, on, r);
break;
case OARRAY:
2008-12-18 21:06:28 -07:00
// arraystring([]byte) string;
r = n->left;
on = syslook("arraystring", 0);
r = nod(OCALL, on, r);
break;
}
walktype(r, top);
return r;
}
Type*
fixmap(Type *t)
{
if(t == T)
goto bad;
if(t->etype != TMAP)
goto bad;
if(t->down == T || t->type == T)
goto bad;
dowidth(t->down);
dowidth(t->type);
return t;
bad:
yyerror("not a map: %lT", t);
return T;
}
Type*
fixchan(Type *t)
{
if(t == T)
goto bad;
if(t->etype != TCHAN)
goto bad;
if(t->type == T)
goto bad;
dowidth(t->type);
return t;
bad:
yyerror("not a channel: %lT", t);
return T;
}
Node*
mapop(Node *n, int top)
{
Node *r, *a;
Type *t;
Node *on;
int cl, cr;
r = n;
switch(n->op) {
default:
2008-07-13 15:29:46 -06:00
fatal("mapop: unknown op %O", n->op);
case OMAKE:
2009-03-11 18:37:04 -06:00
cl = listcount(n->left);
if(cl > 1)
yyerror("too many arguments to make map");
2008-06-16 23:34:50 -06:00
if(top != Erv)
goto nottop;
// newmap(keysize int, valsize int,
// keyalg int, valalg int,
// hint int) (hmap map[any-1]any-2);
t = fixmap(n->type);
if(t == T)
break;
a = n->left; // hint
2009-03-11 18:37:04 -06:00
if(cl != 1)
a = nodintconst(0);
2009-03-11 18:37:04 -06:00
r = a;
a = nodintconst(algtype(t->type)); // val algorithm
r = list(a, r);
a = nodintconst(algtype(t->down)); // key algorithm
r = list(a, r);
a = nodintconst(t->type->width); // val width
r = list(a, r);
a = nodintconst(t->down->width); // key width
r = list(a, r);
on = syslook("newmap", 1);
argtype(on, t->down); // any-1
argtype(on, t->type); // any-2
r = nod(OCALL, on, r);
walktype(r, top);
r->type = n->type;
break;
case OINDEX:
2008-06-16 23:34:50 -06:00
if(top != Erv)
goto nottop;
// mapaccess1(hmap map[any]any, key any) (val any);
t = fixmap(n->left->type);
if(t == T)
break;
convlit(n->right, t->down);
if(!eqtype(n->right->type, t->down, 0)) {
badtype(n->op, n->right->type, t->down);
break;
}
a = n->right; // key
r = a;
a = n->left; // map
r = list(a, r);
2008-06-24 12:10:47 -06:00
on = syslook("mapaccess1", 1);
argtype(on, t->down); // any-1
argtype(on, t->type); // any-2
argtype(on, t->down); // any-3
argtype(on, t->type); // any-4
r = nod(OCALL, on, r);
walktype(r, Erv);
r->type = t->type;
break;
case OAS:
cl = listcount(n->left);
cr = listcount(n->right);
2008-06-16 23:34:50 -06:00
if(cl == 1 && cr == 2)
goto assign2;
if(cl == 2 && cr == 1)
goto access2;
if(cl != 1 || cr != 1)
goto shape;
// mapassign1(hmap map[any-1]any-2, key any-3, val any-4);
if(n->left->op != OINDEX)
goto shape;
t = fixmap(n->left->left->type);
2008-06-16 23:34:50 -06:00
if(t == T)
break;
a = n->right; // val
2008-06-16 23:34:50 -06:00
r = a;
a = n->left->right; // key
r = list(a, r);
a = n->left->left; // map
r = list(a, r);
2008-06-16 23:34:50 -06:00
on = syslook("mapassign1", 1);
2008-06-16 23:34:50 -06:00
argtype(on, t->down); // any-1
argtype(on, t->type); // any-2
argtype(on, t->down); // any-3
argtype(on, t->type); // any-4
r = nod(OCALL, on, r);
walktype(r, Etop);
2008-06-16 23:34:50 -06:00
break;
assign2:
// mapassign2(hmap map[any]any, key any, val any, pres bool);
if(n->left->op != OINDEX)
goto shape;
2008-06-16 23:34:50 -06:00
t = fixmap(n->left->left->type);
if(t == T)
break;
a = n->right->right; // pres
2008-06-16 23:34:50 -06:00
r = a;
a = n->right->left; // val
r =list(a, r);
2008-06-16 23:34:50 -06:00
a = n->left->right; // key
r = list(a, r);
2008-06-16 23:34:50 -06:00
a = n->left->left; // map
r = list(a, r);
2008-06-16 23:34:50 -06:00
on = syslook("mapassign2", 1);
2008-06-16 23:34:50 -06:00
argtype(on, t->down); // any-1
argtype(on, t->type); // any-2
argtype(on, t->down); // any-3
argtype(on, t->type); // any-4
r = nod(OCALL, on, r);
walktype(r, Etop);
2008-06-16 23:34:50 -06:00
break;
access2:
// mapaccess2(hmap map[any-1]any-2, key any-3) (val-4 any, pres bool);
2008-06-16 23:34:50 -06:00
//dump("access2", n);
if(n->right->op != OINDEX)
goto shape;
2008-06-16 23:34:50 -06:00
t = fixmap(n->right->left->type);
2008-06-16 23:34:50 -06:00
if(t == T)
break;
a = n->right->right; // key
2008-06-16 23:34:50 -06:00
r = a;
a = n->right->left; // map
r = list(a, r);
2008-06-16 23:34:50 -06:00
on = syslook("mapaccess2", 1);
2008-06-16 23:34:50 -06:00
argtype(on, t->down); // any-1
argtype(on, t->type); // any-2
argtype(on, t->down); // any-3
argtype(on, t->type); // any-4
n->right = nod(OCALL, on, r);
walktype(n, Etop);
r = n;
2008-06-16 23:34:50 -06:00
break;
case OASOP:
// rewrite map[index] op= right
// into tmpi := index; map[tmpi] = map[tmpi] op right
t = n->left->left->type;
a = nod(OXXX, N, N);
tempname(a, t->down); // tmpi
r = nod(OAS, a, n->left->right); // tmpi := index
n->left->right = a; // m[tmpi]
a = nod(OXXX, N, N);
indir(a, n->left); // copy of map[tmpi]
a = nod(n->etype, a, n->right); // m[tmpi] op right
a = nod(OAS, n->left, a); // map[tmpi] = map[tmpi] op right
r = nod(OLIST, r, a);
2008-10-04 18:11:01 -06:00
walktype(r, Etop);
break;
2008-06-16 23:34:50 -06:00
}
return r;
2008-06-16 23:34:50 -06:00
shape:
dump("shape", n);
fatal("mapop: cl=%d cr=%d, %O", top, n->op);
return N;
2008-06-16 23:34:50 -06:00
nottop:
2008-11-14 17:24:27 -07:00
yyerror("didn't expect %O here", n->op);
2008-06-16 23:34:50 -06:00
return N;
}
Node*
chanop(Node *n, int top)
{
Node *r, *a;
Type *t;
Node *on;
int cl, cr;
//dump("chanop", n);
r = n;
switch(n->op) {
default:
2008-07-13 15:29:46 -06:00
fatal("chanop: unknown op %O", n->op);
case OMAKE:
2009-03-11 18:37:04 -06:00
cl = listcount(n->left);
if(cl > 1)
yyerror("too many arguments to make chan");
// newchan(elemsize int, elemalg int,
// hint int) (hmap *chan[any-1]);
t = fixchan(n->type);
if(t == T)
break;
2009-03-11 18:37:04 -06:00
a = nodintconst(0);
if(cl == 1) {
// async buf size
a = nod(OCONV, n->left, N);
2008-10-29 14:18:21 -06:00
a->type = types[TINT];
2009-03-11 18:37:04 -06:00
}
r = a;
a = nodintconst(algtype(t->type)); // elem algorithm
r = list(a, r);
a = nodintconst(t->type->width); // elem width
r = list(a, r);
on = syslook("newchan", 1);
argtype(on, t->type); // any-1
r = nod(OCALL, on, r);
walktype(r, top);
r->type = n->type;
break;
2008-07-13 15:29:46 -06:00
case OAS:
2008-07-13 17:20:27 -06:00
cl = listcount(n->left);
cr = listcount(n->right);
2008-07-13 15:29:46 -06:00
2008-07-15 22:07:59 -06:00
if(cl != 2 || cr != 1 || n->right->op != ORECV)
2008-07-13 15:29:46 -06:00
goto shape;
2008-07-15 22:07:59 -06:00
// chanrecv2(hchan *chan any) (elem any, pres bool);
t = fixchan(n->right->left->type);
2008-07-13 15:29:46 -06:00
if(t == T)
break;
2008-07-15 22:07:59 -06:00
a = n->right->left; // chan
2008-07-13 15:29:46 -06:00
r = a;
2008-07-15 22:07:59 -06:00
on = syslook("chanrecv2", 1);
2008-07-13 15:29:46 -06:00
argtype(on, t->type); // any-1
argtype(on, t->type); // any-2
r = nod(OCALL, on, r);
2008-07-15 22:07:59 -06:00
n->right = r;
r = n;
walktype(r, Etop);
2008-07-13 15:29:46 -06:00
break;
2008-07-13 17:20:27 -06:00
case ORECV:
2008-07-15 22:07:59 -06:00
if(n->right != N)
goto recv2;
2008-07-13 17:20:27 -06:00
// chanrecv1(hchan *chan any) (elem any);
t = fixchan(n->left->type);
if(t == T)
break;
a = n->left; // chan
r = a;
on = syslook("chanrecv1", 1);
argtype(on, t->type); // any-1
argtype(on, t->type); // any-2
r = nod(OCALL, on, r);
walktype(r, Erv);
break;
recv2:
// chanrecv3(hchan *chan any, *elem any) (pres bool);
2008-07-15 22:07:59 -06:00
t = fixchan(n->right->type);
2008-07-13 17:20:27 -06:00
if(t == T)
break;
2008-07-15 22:07:59 -06:00
a = n->right; // chan
2008-07-13 17:20:27 -06:00
r = a;
a = n->left; // elem
if(a == N) {
a = nod(OLITERAL, N, N);
a->val.ctype = CTNIL;
} else
a = nod(OADDR, a, N);
2008-07-13 17:20:27 -06:00
on = syslook("chanrecv3", 1);
2008-07-13 17:20:27 -06:00
argtype(on, t->type); // any-1
argtype(on, t->type); // any-2
2008-07-13 17:20:27 -06:00
r = nod(OCALL, on, r);
n->right = r;
r = n;
walktype(r, Etop);
break;
2008-07-15 22:07:59 -06:00
case OSEND:
t = fixchan(n->left->type);
if(t == T)
break;
if(top != Etop)
goto send2;
// chansend1(hchan *chan any, elem any);
t = fixchan(n->left->type);
if(t == T)
break;
a = n->right; // e
r = a;
a = n->left; // chan
r = list(a, r);
2008-07-15 22:07:59 -06:00
on = syslook("chansend1", 1);
argtype(on, t->type); // any-1
argtype(on, t->type); // any-2
r = nod(OCALL, on, r);
2008-09-04 13:21:10 -06:00
walktype(r, Etop);
2008-07-15 22:07:59 -06:00
break;
send2:
// chansend2(hchan *chan any, val any) (pres bool);
t = fixchan(n->left->type);
if(t == T)
break;
a = n->right; // e
r = a;
a = n->left; // chan
r = list(a, r);
2008-07-15 22:07:59 -06:00
on = syslook("chansend2", 1);
argtype(on, t->type); // any-1
argtype(on, t->type); // any-2
r = nod(OCALL, on, r);
2008-09-04 13:21:10 -06:00
walktype(r, Etop);
2008-07-15 22:07:59 -06:00
break;
}
return r;
2008-07-13 15:29:46 -06:00
shape:
fatal("chanop: %O", n->op);
return N;
}
2008-08-27 18:28:30 -06:00
Type*
2008-12-18 21:06:28 -07:00
fixarray(Type *t)
2008-08-27 18:28:30 -06:00
{
if(t == T)
goto bad;
if(t->etype != TARRAY)
goto bad;
if(t->type == T)
goto bad;
2008-12-18 21:06:28 -07:00
dowidth(t);
2008-08-27 18:28:30 -06:00
return t;
bad:
2008-12-18 21:06:28 -07:00
yyerror("not an array: %lT", t);
return T;
2008-08-27 18:28:30 -06:00
}
Node*
arrayop(Node *n, int top)
{
Node *r, *a;
2008-12-18 21:06:28 -07:00
Type *t, *tl;
2008-08-27 18:28:30 -06:00
Node *on;
Iter save;
2009-03-11 18:37:04 -06:00
int cl;
2008-08-27 18:28:30 -06:00
r = n;
switch(n->op) {
default:
fatal("darrayop: unknown op %O", n->op);
case OCONV:
// arrays2d(old *any, nel int) (ary []any)
t = fixarray(n->left->type);
tl = fixarray(n->type);
if(t == T || tl == T)
break;
a = nodintconst(t->bound); // nel
a = nod(OCONV, a, N);
a->type = types[TINT];
r = a;
a = nod(OADDR, n->left, N); // old
addrescapes(n->left);
r = list(a, r);
on = syslook("arrays2d", 1);
argtype(on, t); // any-1
argtype(on, tl->type); // any-2
r = nod(OCALL, on, r);
walktype(r, top);
2009-01-07 16:54:08 -07:00
n->left = r;
return n;
2008-12-18 21:06:28 -07:00
case OAS:
// arrays2d(old *any, nel int) (ary []any)
t = fixarray(n->right->type);
tl = fixarray(n->left->type);
if(t == T || tl == T)
break;
2008-12-18 21:06:28 -07:00
a = nodintconst(t->bound); // nel
a = nod(OCONV, a, N);
a->type = types[TINT];
r = a;
a = nod(OADDR, n->right, N); // old
addrescapes(n->right);
2008-12-18 21:06:28 -07:00
r = list(a, r);
on = syslook("arrays2d", 1);
argtype(on, t); // any-1
argtype(on, tl->type); // any-2
r = nod(OCALL, on, r);
walktype(r, top);
n->right = r;
return n;
case OMAKE:
2009-03-11 18:37:04 -06:00
cl = listcount(n->left);
if(cl > 2)
yyerror("too many arguments to make array");
2008-12-18 21:06:28 -07:00
// newarray(nel int, max int, width int) (ary []any)
2008-08-27 18:28:30 -06:00
t = fixarray(n->type);
if(t == T)
break;
2008-08-27 18:28:30 -06:00
a = nodintconst(t->type->width); // width
a = nod(OCONV, a, N);
a->type = types[TINT];
2008-08-27 18:28:30 -06:00
r = a;
a = listfirst(&save, &n->left); // max
a = listnext(&save);
2008-08-27 18:28:30 -06:00
if(a == N)
a = nodintconst(0);
a = nod(OCONV, a, N);
a->type = types[TINT];
2008-08-27 18:28:30 -06:00
r = list(a, r);
a = listfirst(&save, &n->left); // nel
if(a == N) {
if(t->bound < 0)
yyerror("new open array must have size");
2008-08-27 18:28:30 -06:00
a = nodintconst(t->bound);
}
2008-08-27 18:28:30 -06:00
a = nod(OCONV, a, N);
a->type = types[TINT];
2008-08-27 18:28:30 -06:00
r = list(a, r);
on = syslook("newarray", 1);
argtype(on, t->type); // any-1
r = nod(OCALL, on, r);
walktype(r, top);
r->type = t; // if t had a name, going through newarray lost it
2008-08-27 18:28:30 -06:00
break;
case OSLICE:
2008-12-18 21:06:28 -07:00
// arrayslices(old any, nel int, lb int, hb int, width int) (ary []any)
// arraysliced(old []any, lb int, hb int, width int) (ary []any)
2008-08-27 18:28:30 -06:00
t = fixarray(n->left->type);
if(t == T)
break;
2008-08-27 18:28:30 -06:00
a = nodintconst(t->type->width); // width
a = nod(OCONV, a, N);
a->type = types[TINT];
2008-08-27 18:28:30 -06:00
r = a;
a = nod(OCONV, n->right->right, N); // hb
a->type = types[TINT];
2008-08-27 18:28:30 -06:00
r = list(a, r);
a = nod(OCONV, n->right->left, N); // lb
a->type = types[TINT];
2008-08-27 18:28:30 -06:00
r = list(a, r);
2008-12-18 21:06:28 -07:00
if(t->bound >= 0) {
// static slice
a = nodintconst(t->bound); // nel
a = nod(OCONV, a, N);
a->type = types[TINT];
r = list(a, r);
a = nod(OADDR, n->left, N); // old
r = list(a, r);
on = syslook("arrayslices", 1);
argtype(on, t); // any-1
argtype(on, t->type); // any-2
} else {
// dynamic slice
a = n->left; // old
r = list(a, r);
on = syslook("arraysliced", 1);
argtype(on, t->type); // any-1
argtype(on, t->type); // any-2
}
2008-08-27 18:28:30 -06:00
r = nod(OCALL, on, r);
walktype(r, top);
break;
}
return r;
}
/*
* assigning src to dst involving interfaces?
* return op to use.
*/
int
ifaceas1(Type *dst, Type *src, int explicit)
{
if(src == T || dst == T)
return Inone;
if(explicit && !isinter(src))
yyerror("cannot use .(T) on non-interface type %T", src);
if(isinter(dst)) {
if(isinter(src)) {
if(eqtype(dst, src, 0))
2009-01-27 19:21:03 -07:00
return I2Isame;
if(!isnilinter(dst))
ifacecheck(dst, src, lineno, explicit);
return I2I;
}
if(isnilinter(dst))
return T2I;
ifacecheck(dst, src, lineno, explicit);
return T2I;
}
if(isinter(src)) {
ifacecheck(dst, src, lineno, explicit);
return I2T;
}
return Inone;
}
2009-01-27 19:21:03 -07:00
/*
* treat convert T to T as noop
*/
int
ifaceas(Type *dst, Type *src, int explicit)
2009-01-27 19:21:03 -07:00
{
int et;
et = ifaceas1(dst, src, explicit);
2009-01-27 19:21:03 -07:00
if(et == I2Isame)
et = Inone;
return et;
}
2008-11-05 15:27:07 -07:00
static char*
ifacename[] =
{
2009-01-27 19:21:03 -07:00
[I2T] = "ifaceI2T",
[I2T2] = "ifaceI2T2",
[I2I] = "ifaceI2I",
[I2I2] = "ifaceI2I2",
[I2Isame] = "ifaceI2Isame",
2008-11-05 15:27:07 -07:00
};
Node*
ifaceop(Type *tl, Node *n, int op)
{
Type *tr;
Node *r, *a, *on;
Sym *s;
tr = n->type;
switch(op) {
default:
2008-11-05 15:27:07 -07:00
fatal("ifaceop: unknown op %O\n", op);
case T2I:
2008-09-22 17:58:30 -06:00
// ifaceT2I(sigi *byte, sigt *byte, elem any) (ret any);
a = n; // elem
r = a;
s = signame(tr); // sigt
if(s == S)
fatal("ifaceop: signame-1 T2I: %lT", tr);
a = s->oname;
a = nod(OADDR, a, N);
r = list(a, r);
s = signame(tl); // sigi
if(s == S) {
fatal("ifaceop: signame-2 T2I: %lT", tl);
}
a = s->oname;
a = nod(OADDR, a, N);
r = list(a, r);
on = syslook("ifaceT2I", 1);
argtype(on, tr);
2008-09-22 17:58:30 -06:00
argtype(on, tl);
break;
2008-11-05 15:27:07 -07:00
case I2T:
case I2T2:
case I2I:
2008-11-05 15:27:07 -07:00
case I2I2:
// iface[IT]2[IT][2](sigt *byte, iface any) (ret any[, ok bool]);
a = n; // interface
r = a;
2008-09-22 17:58:30 -06:00
s = signame(tl); // sigi
if(s == S)
2008-11-05 15:27:07 -07:00
fatal("ifaceop: signame %d", op);
a = s->oname;
a = nod(OADDR, a, N);
r = list(a, r);
2008-09-22 17:58:30 -06:00
2008-11-05 15:27:07 -07:00
on = syslook(ifacename[op], 1);
argtype(on, tr);
2008-09-22 17:58:30 -06:00
argtype(on, tl);
2008-11-05 15:27:07 -07:00
break;
case OEQ:
case ONE:
// ifaceeq(i1 any-1, i2 any-2) (ret bool);
a = n->right; // i2
r = a;
a = n->left; // i1
r = list(a, r);
on = syslook("ifaceeq", 1);
argtype(on, n->right->type);
argtype(on, n->left->type);
r = nod(OCALL, on, r);
if(op == ONE)
r = nod(ONOT, r, N);
walktype(r, Erv);
return r;
}
r = nod(OCALL, on, r);
walktype(r, Erv);
return r;
}
Node*
convas(Node *n)
{
Node *l, *r;
Type *lt, *rt;
int et;
if(n->op != OAS)
fatal("convas: not OAS %O", n->op);
2008-11-01 17:36:46 -06:00
lt = T;
rt = T;
l = n->left;
r = n->right;
if(l == N || r == N)
goto out;
lt = l->type;
rt = r->type;
if(lt == T || rt == T)
goto out;
2008-06-16 23:34:50 -06:00
if(n->left->op == OINDEX)
if(istype(n->left->left->type, TMAP)) {
indir(n, mapop(n, Elv));
goto out;
2008-06-16 23:34:50 -06:00
}
2008-07-13 15:29:46 -06:00
if(n->left->op == OSEND)
if(n->left->type != T) {
indir(n, chanop(n, Elv));
goto out;
2008-07-13 15:29:46 -06:00
}
if(eqtype(lt, rt, 0))
goto out;
et = ifaceas(lt, rt, 0);
if(et != Inone) {
n->right = ifaceop(lt, r, et);
goto out;
}
if(isslice(lt) && isfixedarray(rt)) {
2008-08-27 18:28:30 -06:00
if(!eqtype(lt->type->type, rt->type->type, 0))
goto bad;
indir(n, arrayop(n, Etop));
goto out;
2008-08-27 18:28:30 -06:00
}
if(ascompat(lt, rt))
goto out;
2008-08-27 18:28:30 -06:00
bad:
badtype(n->op, lt, rt);
out:
ullmancalc(n);
return n;
}
Node*
old2new(Node *n, Type *t)
{
Node *l;
if(n->op != ONAME && n->op != ONONAME) {
yyerror("left side of := must be a name");
return n;
}
l = newname(n->sym);
dodclvar(l, t);
return l;
}
Node*
colas(Node *nl, Node *nr)
{
Iter savel, saver;
Node *l, *r, *a, *n;
Type *t;
int cl, cr;
/* nl is an expression list.
* nr is an expression list.
* return a newname-list from
2008-06-22 22:02:06 -06:00
* types derived from the rhs.
*/
n = N;
cr = listcount(nr);
cl = listcount(nl);
if(cl != cr) {
if(cr == 1)
goto multi;
goto badt;
}
l = listfirst(&savel, &nl);
r = listfirst(&saver, &nr);
2008-06-22 22:02:06 -06:00
while(l != N) {
walktype(r, Erv);
defaultlit(r);
a = old2new(l, r->type);
2008-11-01 17:36:46 -06:00
n = list(n, a);
2008-06-22 22:02:06 -06:00
l = listnext(&savel);
r = listnext(&saver);
}
n = rev(n);
2008-06-22 22:02:06 -06:00
return n;
multi:
/*
* there is a list on the left
* and a mono on the right.
* go into the right to get
* individual types for the left.
*/
switch(nr->op) {
default:
goto badt;
case OCALLMETH:
case OCALLINTER:
case OCALL:
walktype(nr->left, Erv);
2009-02-16 18:44:05 -07:00
convlit(nr->left, types[TFUNC]);
t = nr->left->type;
if(t != T && t->etype == tptr)
t = t->type;
if(t == T || t->etype != TFUNC)
goto badt;
if(t->outtuple != cl)
goto badt;
l = listfirst(&savel, &nl);
t = structfirst(&saver, getoutarg(t));
while(l != N) {
2008-07-03 18:41:05 -06:00
a = old2new(l, t->type);
2008-11-01 17:36:46 -06:00
n = list(n, a);
l = listnext(&savel);
t = structnext(&saver);
}
break;
case OINDEX:
// check if rhs is a map index.
// if so, types are valuetype,bool
if(cl != 2)
goto badt;
walktype(nr->left, Elv);
t = nr->left->type;
if(!istype(t, TMAP))
goto badt;
2008-06-22 22:02:06 -06:00
a = old2new(nl->left, t->type);
n = a;
2008-06-22 22:02:06 -06:00
a = old2new(nl->right, types[TBOOL]);
n = list(n, a);
break;
2008-07-13 17:20:27 -06:00
case ODOTTYPE:
// a,b := i.(T)
if(cl != 2)
goto badt;
walktype(nr->left, Erv);
if(!isinter(nr->left->type))
goto badt;
// a,b = iface
a = old2new(nl->left, nr->type);
n = a;
a = old2new(nl->right, types[TBOOL]);
n = list(n, a);
break;
2008-07-13 17:20:27 -06:00
case ORECV:
if(cl != 2)
goto badt;
walktype(nr->left, Erv);
t = nr->left->type;
if(!istype(t, TCHAN))
2008-07-13 17:20:27 -06:00
goto badt;
a = old2new(nl->left, t->type);
2008-07-13 17:20:27 -06:00
n = a;
a = old2new(nl->right, types[TBOOL]);
n = list(n, a);
break;
}
n = rev(n);
return n;
badt:
if(nl->diag == 0) {
nl->diag = 1;
yyerror("assignment count mismatch: %d = %d", cl, cr);
}
return nl;
}
/*
* rewrite a range statement
* k and v are names/new_names
* m is an array or map
* local is =/0 or :=/1
*/
2008-12-05 19:24:05 -07:00
Node*
dorange(Node *nn)
2008-12-05 19:24:05 -07:00
{
Node *k, *v, *m;
2008-12-05 19:24:05 -07:00
Node *n, *hk, *on, *r, *a;
Type *t, *th;
int local;
2008-12-05 19:24:05 -07:00
if(nn->op != ORANGE)
fatal("dorange not ORANGE");
implicitstar(&nn->right);
k = nn->left;
m = nn->right;
local = nn->etype;
v = N;
if(k->op == OLIST) {
v = k->right;
k = k->left;
}
2008-12-05 19:24:05 -07:00
n = nod(OFOR, N, N);
walktype(m, Erv);
t = m->type;
if(t == T)
goto out;
if(t->etype == TARRAY)
goto ary;
if(t->etype == TMAP)
goto map;
yyerror("range must be over map/array");
goto out;
ary:
hk = nod(OXXX, N, N); // hidden key
tempname(hk, types[TINT]); // maybe TINT32
n->ninit = nod(OAS, hk, literal(0));
n->ntest = nod(OLT, hk, nod(OLEN, m, N));
n->nincr = nod(OASOP, hk, literal(1));
n->nincr->etype = OADD;
if(local)
k = old2new(k, hk->type);
2008-12-05 19:24:05 -07:00
n->nbody = nod(OAS, k, hk);
if(v != N) {
if(local)
v = old2new(v, t->type);
2008-12-05 19:24:05 -07:00
n->nbody = list(n->nbody,
nod(OAS, v, nod(OINDEX, m, hk)) );
}
addtotop(n);
2008-12-05 19:24:05 -07:00
goto out;
map:
th = typ(TARRAY);
th->type = ptrto(types[TUINT8]);
th->bound = (sizeof(struct Hiter) + types[tptr]->width - 1) /
types[tptr]->width;
hk = nod(OXXX, N, N); // hidden iterator
tempname(hk, th); // hashmap hash_iter
on = syslook("mapiterinit", 1);
argtype(on, t->down);
argtype(on, t->type);
argtype(on, th);
r = nod(OADDR, hk, N);
r = list(m, r);
r = nod(OCALL, on, r);
n->ninit = r;
r = nod(OINDEX, hk, literal(0));
a = nod(OLITERAL, N, N);
a->val.ctype = CTNIL;
r = nod(ONE, r, a);
n->ntest = r;
on = syslook("mapiternext", 1);
argtype(on, th);
r = nod(OADDR, hk, N);
r = nod(OCALL, on, r);
n->nincr = r;
if(local)
k = old2new(k, t->down);
2008-12-05 19:24:05 -07:00
if(v == N) {
on = syslook("mapiter1", 1);
argtype(on, th);
argtype(on, t->down);
r = nod(OADDR, hk, N);
r = nod(OCALL, on, r);
n->nbody = nod(OAS, k, r);
goto out;
}
if(local)
v = old2new(v, t->type);
2008-12-05 19:24:05 -07:00
on = syslook("mapiter2", 1);
argtype(on, th);
argtype(on, t->down);
argtype(on, t->type);
r = nod(OADDR, hk, N);
r = nod(OCALL, on, r);
n->nbody = nod(OAS, nod(OLIST, k, v), r);
goto out;
out:
return n;
}
/*
* from ascompat[te]
* evaluating actual function arguments.
* f(a,b)
* if there is exactly one function expr,
* then it is done first. otherwise must
* make temp variables
*/
2008-06-10 22:29:57 -06:00
Node*
reorder1(Node *n)
{
Iter save;
2008-06-11 22:06:26 -06:00
Node *l, *r, *f, *a, *g;
2008-06-10 22:29:57 -06:00
int c, t;
l = listfirst(&save, &n);
c = 0; // function calls
t = 0; // total parameters
loop1:
if(l == N) {
if(c == 0 || t == 1)
return n;
goto pass2;
}
if(l->op == OLIST)
fatal("reorder1 OLIST");
t++;
ullmancalc(l);
2008-06-10 22:29:57 -06:00
if(l->ullman >= UINF)
c++;
l = listnext(&save);
goto loop1;
pass2:
l = listfirst(&save, &n);
2008-06-11 22:06:26 -06:00
g = N; // fncalls assigned to tempnames
f = N; // one fncall assigned to stack
r = N; // non fncalls and tempnames assigned to stack
2008-06-10 22:29:57 -06:00
loop2:
if(l == N) {
r = rev(r);
2008-06-11 22:06:26 -06:00
g = rev(g);
2008-11-01 17:36:46 -06:00
f = list(g, f);
r = list(f, r);
return r;
2008-06-10 22:29:57 -06:00
}
ullmancalc(l);
2008-06-11 22:06:26 -06:00
if(l->ullman < UINF) {
2008-11-01 17:36:46 -06:00
r = list(l, r);
2008-06-11 22:06:26 -06:00
goto more;
}
if(f == N) {
2008-06-10 22:29:57 -06:00
f = l;
2008-06-11 22:06:26 -06:00
goto more;
}
// make assignment of fncall to tempname
a = nod(OXXX, N, N);
tempname(a, l->right->type);
a = nod(OAS, a, l->right);
2008-11-01 17:36:46 -06:00
g = list(a, g);
2008-06-11 22:06:26 -06:00
// put normal arg assignment on list
// with fncall replaced by tempname
l->right = a->left;
2008-11-01 17:36:46 -06:00
r = list(l, r);
2008-06-10 22:29:57 -06:00
2008-06-11 22:06:26 -06:00
more:
2008-06-10 22:29:57 -06:00
l = listnext(&save);
goto loop2;
}
/*
* from ascompat[et]
* a,b = f()
* return of a multi.
* there can be no function calls at all,
* or they will over-write the return values.
*/
2008-06-10 22:29:57 -06:00
Node*
reorder2(Node *n)
{
Iter save;
Node *l;
int c;
l = listfirst(&save, &n);
c = 0;
loop1:
if(l == N) {
if(c > 0)
yyerror("reorder2: too many funcation calls evaluating parameters");
return n;
}
if(l->op == OLIST)
fatal("reorder2 OLIST");
ullmancalc(l);
2008-06-10 22:29:57 -06:00
if(l->ullman >= UINF)
c++;
l = listnext(&save);
goto loop1;
}
/*
* from ascompat[ee]
* a,b = c,d
2008-06-25 12:35:06 -06:00
* simultaneous assignment. there cannot
* be later use of an earlier lvalue.
*/
int
2008-06-25 12:35:06 -06:00
vmatch2(Node *l, Node *r)
{
2008-06-25 12:35:06 -06:00
loop:
/*
* isolate all right sides
*/
if(r == N)
return 0;
switch(r->op) {
case ONAME:
// match each right given left
if(l == r)
return 1;
case OLITERAL:
return 0;
}
if(vmatch2(l, r->right))
return 1;
r = r->left;
goto loop;
}
int
vmatch1(Node *l, Node *r)
{
loop:
/*
* isolate all left sides
*/
if(l == N)
return 0;
switch(l->op) {
case ONAME:
// match each left with all rights
return vmatch2(l, r);
case OLITERAL:
return 0;
}
if(vmatch1(l->right, r))
return 1;
l = l->left;
goto loop;
}
2008-06-10 22:29:57 -06:00
Node*
reorder3(Node *n)
{
Iter save1, save2;
2008-06-25 12:35:06 -06:00
Node *l1, *l2, *q, *r;
int c1, c2;
2008-06-25 12:35:06 -06:00
r = N;
l1 = listfirst(&save1, &n);
c1 = 0;
while(l1 != N) {
2008-06-25 12:35:06 -06:00
l2 = listfirst(&save2, &n);
c2 = 0;
while(l2 != N) {
if(c2 > c1) {
2008-06-25 12:35:06 -06:00
if(vmatch1(l1->left, l2->right)) {
q = nod(OXXX, N, N);
tempname(q, l1->right->type);
2008-06-25 12:35:06 -06:00
q = nod(OAS, l1->left, q);
l1->left = q->right;
r = list(r, q);
2008-06-25 12:35:06 -06:00
break;
}
}
2008-06-25 12:35:06 -06:00
l2 = listnext(&save2);
c2++;
}
l1 = listnext(&save1);
c1++;
}
2008-06-25 12:35:06 -06:00
if(r == N)
return n;
q = N;
l1 = listfirst(&save1, &n);
while(l1 != N) {
2008-11-01 17:36:46 -06:00
q = list(q, l1);
2008-06-25 12:35:06 -06:00
l1 = listnext(&save1);
}
r = rev(r);
l1 = listfirst(&save1, &r);
while(l1 != N) {
2008-11-01 17:36:46 -06:00
q = list(q, l1);
2008-06-25 12:35:06 -06:00
l1 = listnext(&save1);
}
q = rev(q);
//dump("res", q);
return q;
2008-06-10 22:29:57 -06:00
}
Node*
reorder4(Node *n)
{
/*
* from ascompat[te]
* return c,d
* return expression assigned to output
* parameters. there may be no problems.
*/
return n;
}
Node*
structlit(Node *n, Node *var)
{
Iter savel, saver;
Type *l, *t;
Node *r, *a;
t = n->type;
if(t->etype != TSTRUCT)
fatal("structlit: not struct");
if(var == N) {
var = nod(OXXX, N, N);
tempname(var, t);
}
l = structfirst(&savel, &n->type);
r = listfirst(&saver, &n->left);
2008-09-29 21:33:51 -06:00
if(r != N && r->op == OEMPTY)
r = N;
loop:
if(l == T || r == N) {
if(l != T)
yyerror("struct literal expect expr of type %T", l);
if(r != N)
yyerror("struct literal too many expressions");
return var;
}
// build list of var.field = expr
a = nod(ODOT, var, newname(l->sym));
a = nod(OAS, a, r);
walktype(a, Etop); // add any assignments in r to addtop
addtop = list(addtop, a);
l = structnext(&savel);
r = listnext(&saver);
goto loop;
}
2008-09-03 15:09:29 -06:00
Node*
arraylit(Node *n, Node *var)
{
Iter saver;
Type *t;
Node *r, *a, *nnew;
2009-01-07 14:20:10 -07:00
int idx, ninit, b;
t = n->type;
if(t->etype != TARRAY)
fatal("arraylit: not array");
2009-01-07 14:20:10 -07:00
// count initializers
ninit = 0;
r = listfirst(&saver, &n->left);
if(r != N && r->op == OEMPTY)
r = N;
while(r != N) {
ninit++;
r = listnext(&saver);
}
b = t->bound;
if(b == -100) {
// flag for [...]
b = ninit;
t = shallow(t);
t->bound = b;
}
if(var == N) {
var = nod(OXXX, N, N);
tempname(var, t);
}
nnew = nil;
2009-01-07 14:20:10 -07:00
if(b < 0) {
// slice
nnew = nod(OMAKE, N, N);
nnew->type = t;
a = nod(OAS, var, nnew);
addtop = list(addtop, a);
2009-01-07 14:20:10 -07:00
} else {
// if entire array isnt initialized,
// then clear the array
2009-01-07 14:20:10 -07:00
if(ninit < b) {
a = nod(OAS, var, N);
addtop = list(addtop, a);
}
}
idx = 0;
r = listfirst(&saver, &n->left);
if(r != N && r->op == OEMPTY)
r = N;
while(r != N) {
// build list of var[c] = expr
a = nodintconst(idx);
a = nod(OINDEX, var, a);
a = nod(OAS, a, r);
addtop = list(addtop, a);
idx++;
r = listnext(&saver);
}
if(b < 0)
nnew->left = nodintconst(idx);
return var;
}
2008-09-03 15:40:22 -06:00
Node*
maplit(Node *n, Node *var)
2008-09-03 15:40:22 -06:00
{
Iter saver;
Type *t;
Node *r, *a;
2008-09-03 15:40:22 -06:00
t = n->type;
if(t->etype != TMAP)
fatal("maplit: not map");
2008-09-03 15:40:22 -06:00
if(var == N) {
var = nod(OXXX, N, N);
tempname(var, t);
}
2008-09-03 15:40:22 -06:00
a = nod(OMAKE, N, N);
a->type = t;
2008-09-03 15:40:22 -06:00
a = nod(OAS, var, a);
addtop = list(addtop, a);
r = listfirst(&saver, &n->left);
2008-09-29 21:33:51 -06:00
if(r != N && r->op == OEMPTY)
r = N;
2008-09-03 15:40:22 -06:00
loop:
if(r == N) {
return var;
}
if(r->op != OKEY) {
yyerror("map literal must have key:value pairs");
return var;
}
// build list of var[c] = expr
a = nod(OINDEX, var, r->left);
a = nod(OAS, a, r->right);
addtop = list(addtop, a);
r = listnext(&saver);
goto loop;
}
/*
* the address of n has been taken and might be used after
* the current function returns. mark any local vars
* as needing to move to the heap.
*/
static char *pnames[] = {
[PAUTO] "auto",
[PPARAM] "param",
[PPARAMOUT] "param_out",
};
void
addrescapes(Node *n)
{
char buf[100];
switch(n->op) {
default:
dump("addrescapes", n);
break;
case ONAME:
if(n->noescape)
break;
switch(n->class) {
case PPARAMOUT:
yyerror("cannot take address of out parameter %s", n->sym->name);
break;
case PAUTO:
case PPARAM:
if(debug['E'])
print("%L %s %S escapes %p\n", n->lineno, pnames[n->class], n->sym, n);
// if func param, need separate temporary
// to hold heap pointer.
if(n->class == PPARAM) {
// expression to refer to stack copy
n->stackparam = nod(OPARAM, n, N);
n->stackparam->type = n->type;
n->stackparam->addable = 1;
n->stackparam->xoffset = n->xoffset;
}
n->class |= PHEAP;
n->addable = 0;
n->ullman = 2;
n->alloc = callnew(n->type);
n->xoffset = 0;
// create stack variable to hold pointer to heap
n->heapaddr = nod(0, N, N);
tempname(n->heapaddr, ptrto(n->type));
snprint(buf, sizeof buf, "&%S", n->sym);
n->heapaddr->sym = lookup(buf);
break;
}
break;
case OIND:
case ODOTPTR:
break;
case ODOT:
case OINDEX:
// ODOTPTR has already been introduced,
// so these are the non-pointer ODOT and OINDEX.
// In &x[0], if x is a slice, then x does not
// escape--the pointer inside x does, but that
// is always a heap pointer anyway.
if(!isslice(n->left->type))
addrescapes(n->left);
break;
}
}
/*
* walk through argin parameters.
* generate and return code to allocate
* copies of escaped parameters to the heap.
*/
Node*
paramstoheap(Type **argin)
{
Type *t;
Iter savet;
Node *v, *nn;
nn = N;
for(t = structfirst(&savet, argin); t != T; t = structnext(&savet)) {
v = t->nname;
if(v == N || !(v->class & PHEAP))
continue;
// generate allocation & copying code
nn = list(nn, nod(OAS, v->heapaddr, v->alloc));
nn = list(nn, nod(OAS, v, v->stackparam));
}
return nn;
}
/*
* take care of migrating any function in/out args
* between the stack and the heap. adds code to
* curfn's before and after lists.
*/
void
heapmoves(void)
{
Node *nn;
nn = paramstoheap(getthis(curfn->type));
nn = list(nn, paramstoheap(getinarg(curfn->type)));
curfn->enter = list(curfn->enter, nn);
}