mirror of
https://github.com/golang/go
synced 2024-11-25 21:57:57 -07:00
bug in stack size used in
extending segmented stack R=r OCL=27319 CL=27319
This commit is contained in:
parent
78edbfdcf4
commit
cb15bbe748
@ -37,7 +37,7 @@ argsize(void)
|
||||
int32 s;
|
||||
|
||||
//print("t=%T\n", thisfn);
|
||||
s = 0;
|
||||
s = align(0, thisfn->link, Aarg0);
|
||||
for(t=thisfn->down; t!=T; t=t->down) {
|
||||
switch(t->etype) {
|
||||
case TVOID:
|
||||
|
@ -1,550 +0,0 @@
|
||||
// Inferno utils/6c/sgen.c
|
||||
// http://code.google.com/p/inferno-os/source/browse/utils/6c/sgen.c
|
||||
//
|
||||
// Copyright © 1994-1999 Lucent Technologies Inc. All rights reserved.
|
||||
// Portions Copyright © 1995-1997 C H Forsyth (forsyth@terzarima.net)
|
||||
// Portions Copyright © 1997-1999 Vita Nuova Limited
|
||||
// Portions Copyright © 2000-2007 Vita Nuova Holdings Limited (www.vitanuova.com)
|
||||
// Portions Copyright © 2004,2006 Bruce Ellis
|
||||
// Portions Copyright © 2005-2007 C H Forsyth (forsyth@terzarima.net)
|
||||
// Revisions Copyright © 2000-2007 Lucent Technologies Inc. and others
|
||||
// Portions Copyright © 2009 The Go Authors. All rights reserved.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
// in the Software without restriction, including without limitation the rights
|
||||
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
// copies of the Software, and to permit persons to whom the Software is
|
||||
// furnished to do so, subject to the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be included in
|
||||
// all copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
// THE SOFTWARE.
|
||||
|
||||
#include "gc.h"
|
||||
|
||||
void
|
||||
codgen(Node *n, Node *nn)
|
||||
{
|
||||
Prog *sp;
|
||||
Node *n1, nod, nod1;
|
||||
|
||||
cursafe = 0;
|
||||
curarg = 0;
|
||||
maxargsafe = 0;
|
||||
|
||||
/*
|
||||
* isolate name
|
||||
*/
|
||||
for(n1 = nn;; n1 = n1->left) {
|
||||
if(n1 == Z) {
|
||||
diag(nn, "cant find function name");
|
||||
return;
|
||||
}
|
||||
if(n1->op == ONAME)
|
||||
break;
|
||||
}
|
||||
nearln = nn->lineno;
|
||||
gpseudo(ATEXT, n1->sym, nodconst(stkoff));
|
||||
sp = p;
|
||||
|
||||
/*
|
||||
* isolate first argument
|
||||
*/
|
||||
if(REGARG) {
|
||||
if(typecmplx[thisfn->link->etype]) {
|
||||
nod1 = *nodret->left;
|
||||
nodreg(&nod, &nod1, REGARG);
|
||||
gmove(&nod, &nod1);
|
||||
} else
|
||||
if(firstarg && typeword[firstargtype->etype]) {
|
||||
nod1 = *nodret->left;
|
||||
nod1.sym = firstarg;
|
||||
nod1.type = firstargtype;
|
||||
nod1.xoffset = align(0, firstargtype, Aarg1);
|
||||
nod1.etype = firstargtype->etype;
|
||||
nodreg(&nod, &nod1, REGARG);
|
||||
gmove(&nod, &nod1);
|
||||
}
|
||||
}
|
||||
|
||||
canreach = 1;
|
||||
warnreach = 1;
|
||||
gen(n);
|
||||
if(canreach && thisfn->link->etype != TVOID)
|
||||
warn(Z, "no return at end of function: %s", n1->sym->name);
|
||||
noretval(3);
|
||||
gbranch(ORETURN);
|
||||
|
||||
if(!debug['N'] || debug['R'] || debug['P'])
|
||||
regopt(sp);
|
||||
|
||||
if(thechar=='6' || thechar=='7') /* [sic] */
|
||||
maxargsafe = xround(maxargsafe, 8);
|
||||
sp->to.offset += maxargsafe;
|
||||
}
|
||||
|
||||
void
|
||||
supgen(Node *n)
|
||||
{
|
||||
int owarn;
|
||||
int32 spc;
|
||||
Prog *sp;
|
||||
|
||||
if(n == Z)
|
||||
return;
|
||||
suppress++;
|
||||
owarn = warnreach;
|
||||
warnreach = 0;
|
||||
spc = pc;
|
||||
sp = lastp;
|
||||
gen(n);
|
||||
lastp = sp;
|
||||
pc = spc;
|
||||
sp->link = nil;
|
||||
suppress--;
|
||||
warnreach = owarn;
|
||||
}
|
||||
|
||||
void
|
||||
gen(Node *n)
|
||||
{
|
||||
Node *l, nod;
|
||||
Prog *sp, *spc, *spb;
|
||||
Case *cn;
|
||||
int32 sbc, scc;
|
||||
int snbreak, sncontin;
|
||||
int f, o, oldreach;
|
||||
|
||||
loop:
|
||||
if(n == Z)
|
||||
return;
|
||||
nearln = n->lineno;
|
||||
o = n->op;
|
||||
if(debug['G'])
|
||||
if(o != OLIST)
|
||||
print("%L %O\n", nearln, o);
|
||||
|
||||
if(!canreach) {
|
||||
switch(o) {
|
||||
case OLABEL:
|
||||
case OCASE:
|
||||
case OLIST:
|
||||
case OBREAK:
|
||||
case OFOR:
|
||||
case OWHILE:
|
||||
case ODWHILE:
|
||||
/* all handled specially - see switch body below */
|
||||
break;
|
||||
default:
|
||||
if(warnreach) {
|
||||
warn(n, "unreachable code %O", o);
|
||||
warnreach = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
switch(o) {
|
||||
|
||||
default:
|
||||
complex(n);
|
||||
cgen(n, Z);
|
||||
break;
|
||||
|
||||
case OLIST:
|
||||
gen(n->left);
|
||||
|
||||
rloop:
|
||||
n = n->right;
|
||||
goto loop;
|
||||
|
||||
case ORETURN:
|
||||
canreach = 0;
|
||||
warnreach = !suppress;
|
||||
complex(n);
|
||||
if(n->type == T)
|
||||
break;
|
||||
l = n->left;
|
||||
if(l == Z) {
|
||||
noretval(3);
|
||||
gbranch(ORETURN);
|
||||
break;
|
||||
}
|
||||
if(typecmplx[n->type->etype]) {
|
||||
sugen(l, nodret, n->type->width);
|
||||
noretval(3);
|
||||
gbranch(ORETURN);
|
||||
break;
|
||||
}
|
||||
regret(&nod, n);
|
||||
cgen(l, &nod);
|
||||
regfree(&nod);
|
||||
if(typefd[n->type->etype])
|
||||
noretval(1);
|
||||
else
|
||||
noretval(2);
|
||||
gbranch(ORETURN);
|
||||
break;
|
||||
|
||||
case OLABEL:
|
||||
canreach = 1;
|
||||
l = n->left;
|
||||
if(l) {
|
||||
l->pc = pc;
|
||||
if(l->label)
|
||||
patch(l->label, pc);
|
||||
}
|
||||
gbranch(OGOTO); /* prevent self reference in reg */
|
||||
patch(p, pc);
|
||||
goto rloop;
|
||||
|
||||
case OGOTO:
|
||||
canreach = 0;
|
||||
warnreach = !suppress;
|
||||
n = n->left;
|
||||
if(n == Z)
|
||||
return;
|
||||
if(n->complex == 0) {
|
||||
diag(Z, "label undefined: %s", n->sym->name);
|
||||
return;
|
||||
}
|
||||
if(suppress)
|
||||
return;
|
||||
gbranch(OGOTO);
|
||||
if(n->pc) {
|
||||
patch(p, n->pc);
|
||||
return;
|
||||
}
|
||||
if(n->label)
|
||||
patch(n->label, pc-1);
|
||||
n->label = p;
|
||||
return;
|
||||
|
||||
case OCASE:
|
||||
canreach = 1;
|
||||
l = n->left;
|
||||
if(cases == C)
|
||||
diag(n, "case/default outside a switch");
|
||||
if(l == Z) {
|
||||
cas();
|
||||
cases->val = 0;
|
||||
cases->def = 1;
|
||||
cases->label = pc;
|
||||
cases->isv = 0;
|
||||
goto rloop;
|
||||
}
|
||||
complex(l);
|
||||
if(l->type == T)
|
||||
goto rloop;
|
||||
if(l->op == OCONST)
|
||||
if(typeword[l->type->etype] && l->type->etype != TIND) {
|
||||
cas();
|
||||
cases->val = l->vconst;
|
||||
cases->def = 0;
|
||||
cases->label = pc;
|
||||
cases->isv = typev[l->type->etype];
|
||||
goto rloop;
|
||||
}
|
||||
diag(n, "case expression must be integer constant");
|
||||
goto rloop;
|
||||
|
||||
case OSWITCH:
|
||||
l = n->left;
|
||||
complex(l);
|
||||
if(l->type == T)
|
||||
break;
|
||||
if(!typeword[l->type->etype] || l->type->etype == TIND) {
|
||||
diag(n, "switch expression must be integer");
|
||||
break;
|
||||
}
|
||||
|
||||
gbranch(OGOTO); /* entry */
|
||||
sp = p;
|
||||
|
||||
cn = cases;
|
||||
cases = C;
|
||||
cas();
|
||||
|
||||
sbc = breakpc;
|
||||
breakpc = pc;
|
||||
snbreak = nbreak;
|
||||
nbreak = 0;
|
||||
gbranch(OGOTO);
|
||||
spb = p;
|
||||
|
||||
gen(n->right); /* body */
|
||||
if(canreach){
|
||||
gbranch(OGOTO);
|
||||
patch(p, breakpc);
|
||||
nbreak++;
|
||||
}
|
||||
|
||||
patch(sp, pc);
|
||||
regalloc(&nod, l, Z);
|
||||
/* always signed */
|
||||
if(typev[l->type->etype])
|
||||
nod.type = types[TVLONG];
|
||||
else
|
||||
nod.type = types[TLONG];
|
||||
cgen(l, &nod);
|
||||
doswit(&nod);
|
||||
regfree(&nod);
|
||||
patch(spb, pc);
|
||||
|
||||
cases = cn;
|
||||
breakpc = sbc;
|
||||
canreach = nbreak!=0;
|
||||
if(canreach == 0)
|
||||
warnreach = !suppress;
|
||||
nbreak = snbreak;
|
||||
break;
|
||||
|
||||
case OWHILE:
|
||||
case ODWHILE:
|
||||
l = n->left;
|
||||
gbranch(OGOTO); /* entry */
|
||||
sp = p;
|
||||
|
||||
scc = continpc;
|
||||
continpc = pc;
|
||||
gbranch(OGOTO);
|
||||
spc = p;
|
||||
|
||||
sbc = breakpc;
|
||||
breakpc = pc;
|
||||
snbreak = nbreak;
|
||||
nbreak = 0;
|
||||
gbranch(OGOTO);
|
||||
spb = p;
|
||||
|
||||
patch(spc, pc);
|
||||
if(n->op == OWHILE)
|
||||
patch(sp, pc);
|
||||
bcomplex(l, Z); /* test */
|
||||
patch(p, breakpc);
|
||||
if(l->op != OCONST || vconst(l) == 0)
|
||||
nbreak++;
|
||||
|
||||
if(n->op == ODWHILE)
|
||||
patch(sp, pc);
|
||||
gen(n->right); /* body */
|
||||
gbranch(OGOTO);
|
||||
patch(p, continpc);
|
||||
|
||||
patch(spb, pc);
|
||||
continpc = scc;
|
||||
breakpc = sbc;
|
||||
canreach = nbreak!=0;
|
||||
if(canreach == 0)
|
||||
warnreach = !suppress;
|
||||
nbreak = snbreak;
|
||||
break;
|
||||
|
||||
case OFOR:
|
||||
l = n->left;
|
||||
if(!canreach && l->right->left && warnreach) {
|
||||
warn(n, "unreachable code FOR");
|
||||
warnreach = 0;
|
||||
}
|
||||
gen(l->right->left); /* init */
|
||||
gbranch(OGOTO); /* entry */
|
||||
sp = p;
|
||||
|
||||
/*
|
||||
* if there are no incoming labels in the
|
||||
* body and the top's not reachable, warn
|
||||
*/
|
||||
if(!canreach && warnreach && deadheads(n)) {
|
||||
warn(n, "unreachable code %O", o);
|
||||
warnreach = 0;
|
||||
}
|
||||
|
||||
scc = continpc;
|
||||
continpc = pc;
|
||||
gbranch(OGOTO);
|
||||
spc = p;
|
||||
|
||||
sbc = breakpc;
|
||||
breakpc = pc;
|
||||
snbreak = nbreak;
|
||||
nbreak = 0;
|
||||
sncontin = ncontin;
|
||||
ncontin = 0;
|
||||
gbranch(OGOTO);
|
||||
spb = p;
|
||||
|
||||
patch(spc, pc);
|
||||
gen(l->right->right); /* inc */
|
||||
patch(sp, pc);
|
||||
if(l->left != Z) { /* test */
|
||||
bcomplex(l->left, Z);
|
||||
patch(p, breakpc);
|
||||
if(l->left->op != OCONST || vconst(l->left) == 0)
|
||||
nbreak++;
|
||||
}
|
||||
canreach = 1;
|
||||
gen(n->right); /* body */
|
||||
if(canreach){
|
||||
gbranch(OGOTO);
|
||||
patch(p, continpc);
|
||||
ncontin++;
|
||||
}
|
||||
if(!ncontin && l->right->right && warnreach) {
|
||||
warn(l->right->right, "unreachable FOR inc");
|
||||
warnreach = 0;
|
||||
}
|
||||
|
||||
patch(spb, pc);
|
||||
continpc = scc;
|
||||
breakpc = sbc;
|
||||
canreach = nbreak!=0;
|
||||
if(canreach == 0)
|
||||
warnreach = !suppress;
|
||||
nbreak = snbreak;
|
||||
ncontin = sncontin;
|
||||
break;
|
||||
|
||||
case OCONTINUE:
|
||||
if(continpc < 0) {
|
||||
diag(n, "continue not in a loop");
|
||||
break;
|
||||
}
|
||||
gbranch(OGOTO);
|
||||
patch(p, continpc);
|
||||
ncontin++;
|
||||
canreach = 0;
|
||||
warnreach = !suppress;
|
||||
break;
|
||||
|
||||
case OBREAK:
|
||||
if(breakpc < 0) {
|
||||
diag(n, "break not in a loop");
|
||||
break;
|
||||
}
|
||||
/*
|
||||
* Don't complain about unreachable break statements.
|
||||
* There are breaks hidden in yacc's output and some people
|
||||
* write return; break; in their switch statements out of habit.
|
||||
* However, don't confuse the analysis by inserting an
|
||||
* unreachable reference to breakpc either.
|
||||
*/
|
||||
if(!canreach)
|
||||
break;
|
||||
gbranch(OGOTO);
|
||||
patch(p, breakpc);
|
||||
nbreak++;
|
||||
canreach = 0;
|
||||
warnreach = !suppress;
|
||||
break;
|
||||
|
||||
case OIF:
|
||||
l = n->left;
|
||||
if(bcomplex(l, n->right)) {
|
||||
if(typefd[l->type->etype])
|
||||
f = !l->fconst;
|
||||
else
|
||||
f = !l->vconst;
|
||||
if(debug['c'])
|
||||
print("%L const if %s\n", nearln, f ? "false" : "true");
|
||||
if(f) {
|
||||
canreach = 1;
|
||||
supgen(n->right->left);
|
||||
oldreach = canreach;
|
||||
canreach = 1;
|
||||
gen(n->right->right);
|
||||
/*
|
||||
* treat constant ifs as regular ifs for
|
||||
* reachability warnings.
|
||||
*/
|
||||
if(!canreach && oldreach && debug['w'] < 2)
|
||||
warnreach = 0;
|
||||
}
|
||||
else {
|
||||
canreach = 1;
|
||||
gen(n->right->left);
|
||||
oldreach = canreach;
|
||||
canreach = 1;
|
||||
supgen(n->right->right);
|
||||
/*
|
||||
* treat constant ifs as regular ifs for
|
||||
* reachability warnings.
|
||||
*/
|
||||
if(!oldreach && canreach && debug['w'] < 2)
|
||||
warnreach = 0;
|
||||
canreach = oldreach;
|
||||
}
|
||||
}
|
||||
else {
|
||||
sp = p;
|
||||
canreach = 1;
|
||||
if(n->right->left != Z)
|
||||
gen(n->right->left);
|
||||
oldreach = canreach;
|
||||
canreach = 1;
|
||||
if(n->right->right != Z) {
|
||||
gbranch(OGOTO);
|
||||
patch(sp, pc);
|
||||
sp = p;
|
||||
gen(n->right->right);
|
||||
}
|
||||
patch(sp, pc);
|
||||
canreach = canreach || oldreach;
|
||||
if(canreach == 0)
|
||||
warnreach = !suppress;
|
||||
}
|
||||
break;
|
||||
|
||||
case OSET:
|
||||
case OUSED:
|
||||
usedset(n->left, o);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
usedset(Node *n, int o)
|
||||
{
|
||||
if(n->op == OLIST) {
|
||||
usedset(n->left, o);
|
||||
usedset(n->right, o);
|
||||
return;
|
||||
}
|
||||
complex(n);
|
||||
switch(n->op) {
|
||||
case OADDR: /* volatile */
|
||||
gins(ANOP, n, Z);
|
||||
break;
|
||||
case ONAME:
|
||||
if(o == OSET)
|
||||
gins(ANOP, Z, n);
|
||||
else
|
||||
gins(ANOP, n, Z);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int
|
||||
bcomplex(Node *n, Node *c)
|
||||
{
|
||||
|
||||
complex(n);
|
||||
if(n->type != T)
|
||||
if(tcompat(n, T, n->type, tnot))
|
||||
n->type = T;
|
||||
if(n->type == T) {
|
||||
gbranch(OGOTO);
|
||||
return 0;
|
||||
}
|
||||
if(c != Z && n->op == OCONST && deadheads(c))
|
||||
return 1;
|
||||
bool64(n);
|
||||
boolgen(n, 1, Z);
|
||||
return 0;
|
||||
}
|
Loading…
Reference in New Issue
Block a user