Annotation of researchv9/jtools/src/pi/expr.c, revision 1.1

1.1     ! root        1: #include "expr.pri"
        !             2: #include "gram.h"
        !             3: #include "symbol.h"
        !             4: #include "format.pub"
        !             5: #include "frame.pri"
        !             6: #include "phrase.pub"
        !             7: #include "symtab.pub"
        !             8: #include "core.pub"
        !             9: #include <CC/signal.h>         /* floating point exceptions */
        !            10: SRCFILE("expr.c")
        !            11: 
        !            12: #ifdef V9
        !            13: #define SIG_TYP SIG_ARG_TYP
        !            14: #endif
        !            15: 
        !            16: char *OpName(Op op)
        !            17: {
        !            18:        switch(op){
        !            19:        case O_DEREF:   return "*";
        !            20:        case O_REF:     return "&";
        !            21:        case O_INDEX:   return "[]";
        !            22:        case O_DOT:     return ".";
        !            23:        case O_ARROW:   return "->";
        !            24:        case O_PLUS:    return "+";
        !            25:        case O_MINUS:   return "-";
        !            26:        case O_MULT:    return "*";
        !            27:        case O_DIV:     return "/";
        !            28:        case O_CALL:    return "()";
        !            29:        case O_MOD:     return "%";
        !            30:        case O_ASSIGN:  return "=";
        !            31:        case O_COMMA:   return ",";
        !            32:        case O_SIZEOF:  return "sizeof";
        !            33:        case O_TYPEOF:  return "typeof";
        !            34:        case O_QINDEX:  return "[?]";
        !            35:        case O_CAST:    return "(cast)";
        !            36:        case O_QCAST:   return " ?";
        !            37:        case O_QSTRUCT: return "struct ?";
        !            38:        case O_QENUM:   return "enum ?";
        !            39:        case O_EQ:      return "==";
        !            40:        case O_NE:      return "!=";
        !            41:        case O_LT:      return "<";
        !            42:        case O_GT:      return ">";
        !            43:        case O_LE:      return "<=";
        !            44:        case O_GE:      return ">=";
        !            45:        case O_LOGAND:  return "&&";
        !            46:        case O_LOGOR:   return "||";
        !            47:        case O_LOGNOT:  return "!";
        !            48:        case O_BITAND:  return "&";
        !            49:        case O_BITOR:   return "|";
        !            50:        case O_BITXOR:  return "^";
        !            51:        case O_1SCOMP:  return "~";
        !            52:        case O_FABS:    return "fabs";
        !            53:        case O_ENV:     return "{}";
        !            54:        case O_RANGE:   return "..";
        !            55:        case O_SPECIAL: return "special";
        !            56:        case O_LSHIFT:  return "<<";
        !            57:        case O_RSHIFT:  return ">>";
        !            58:        default:        return Name( "Op=%d", op );
        !            59:        }
        !            60: }
        !            61: 
        !            62: int Prec( Op op )              /* for Expr::text() - not gram.y */
        !            63: {
        !            64:        switch(op){
        !            65:        case O_ENV:     return 1;
        !            66:        case O_CALL:
        !            67:        case O_INDEX:
        !            68:        case O_DOT:
        !            69:        case O_ARROW:   return 2;
        !            70:        case O_MULT:
        !            71:        case O_DIV:
        !            72:        case O_MOD:     return 3;
        !            73:        case O_PLUS:
        !            74:        case O_MINUS:   return 4;
        !            75:        case O_LSHIFT:
        !            76:        case O_RSHIFT:  return 5;
        !            77:        case O_LT:
        !            78:        case O_GT:
        !            79:        case O_LE:
        !            80:        case O_GE:      return 6;
        !            81:        case O_EQ:
        !            82:        case O_NE:      return 7;
        !            83:        case O_BITAND:  return 8;
        !            84:        case O_BITXOR:  return 9;
        !            85:        case O_BITOR:   return 10;
        !            86:        case O_LOGAND:  return 11;
        !            87:        case O_LOGOR:   return 12;
        !            88:        case O_ASSIGN:  return 13;
        !            89:        case O_COMMA:   return 14;
        !            90:        case O_RANGE:   return 15;
        !            91:        default:        return 16;
        !            92:        }
        !            93: }
        !            94: 
        !            95: Expr *E_Id(char *id ) { return new Expr(E_ID,0,0,0,0,id,0,0); }
        !            96: 
        !            97: Expr *E_Sym(Symbol *s ) { return new Expr(E_SYMNODE,0,0,0,0,0,0,s); }
        !            98:  
        !            99: Expr *E_Unary(Op unary, Expr *sub) { return new Expr(E_UNARY,unary,sub,0,0,0,0,0); }
        !           100: 
        !           101: Expr *E_Binary(Expr*l,Op binary,Expr*r)
        !           102:                { return new Expr(E_BINARY,binary,l,r,0,0,0,0); }
        !           103: 
        !           104: Expr *E_IConst(long i) { return new Expr(E_LCONST,0,0,0,i,Token,LONG,0); }
        !           105: 
        !           106: Expr *E_DConst(double d) { return new Expr(E_DCONST,0,0,0,d,Token,DOUBLE,0); }
        !           107: 
        !           108: char *Expr::floaterror()
        !           109: {
        !           110:        return val.floaterror();
        !           111: }
        !           112: 
        !           113: char *Expr::under(Expr *e)
        !           114: {
        !           115:        if( e->edisc != E_BINARY || Prec(e->op) < Prec(op) ) return e->text();
        !           116:        return sf( "(%s)", e->text() );
        !           117: }
        !           118: 
        !           119: Expr::Expr() {}
        !           120: 
        !           121: Expr::Expr( EDisc d, Op o, Expr* s1, Expr* s2, Cslfd c, char* s, int t, Symbol *v)
        !           122: {
        !           123:        type.pcc = t;
        !           124:        edisc = d;
        !           125:        op = o;
        !           126:        sub1 = s1;
        !           127:        sub2 = s2;
        !           128:        val = c;
        !           129:        id = s ? sf("%s",s) : "";
        !           130:        sym = v;
        !           131:        evalerr = 0;
        !           132:        spy = 0;
        !           133:        addr = 0;
        !           134:        bitaddr = 0;
        !           135:        trace( "Expr(edisc=%d,op=%s,sub1=%d,sub2=%d,val.lng=%d,id=%s,symbol=%d",
        !           136:                edisc, OpName(op), sub1, sub2, val.lng, id?id:"", v); 
        !           137: }
        !           138: 
        !           139: char *Expr::left(Expr *e)
        !           140: {
        !           141:        switch( e->edisc ){
        !           142:        case E_BINARY:  switch( e->op ){
        !           143:                        case O_INDEX:
        !           144:                        case O_DOT:
        !           145:                        case O_ARROW:   return e->text();
        !           146:                        }
        !           147:        case E_UNARY:   return sf( "(%s)", e->text() );
        !           148:        default:        return e->text();
        !           149:        }
        !           150: }
        !           151: 
        !           152: char *Expr::textunary()
        !           153: {
        !           154:        trace( "%d.textunary()", this );        OK("textunary");
        !           155:        switch( op ){
        !           156:        case O_QCAST:
        !           157:        case O_QSTRUCT:
        !           158:        case O_QENUM:
        !           159:                return sf("(%0.*s%s )%s",val.lng,"*******",OpName(op),sub1->text());
        !           160:        case O_CAST:
        !           161:                return sf("(%s)%s",type.text(),sub1->text());
        !           162:        case O_QINDEX:
        !           163:                return sf( "%s[ ? ]", left(sub1) );
        !           164:        case O_SIZEOF:
        !           165:        case O_TYPEOF:
        !           166:        case O_FABS:
        !           167:        case O_SPECIAL:
        !           168:                return sf( "%s(%s)", OpName(op), sub1->text() );
        !           169:        }
        !           170:        if( sub1->edisc==E_BINARY && Prec(sub1->op)>=Prec(O_MULT) )
        !           171:                return sf( "%s(%s)", OpName(op), sub1->text() );
        !           172:        return sf( "%s%s", OpName(op), sub1->text() );
        !           173: }
        !           174: 
        !           175: char *Expr::textbinary()
        !           176: {
        !           177:        trace( "%d.textbinary()", this );       OK("textbinary");
        !           178:        switch( op ){
        !           179:        case O_ENV:
        !           180:                return sf( "{%s}%s", sub1->text(), sub2->text() );
        !           181:        case O_COMMA:                                                   /* ??? */
        !           182:                return sf( "%s,%s", sub1->text(), sub2->text() );
        !           183:        case O_CALL:
        !           184:                return sf( "%s(%s)", sub1->text(), sub2?sub2->text():"" );
        !           185:        case O_ASSIGN:
        !           186:                return sf("(%s:=%s)", left(sub1), sub2->text() );
        !           187:        case O_INDEX:
        !           188:                return sf("%s[%s]", left(sub1), sub2->text() );
        !           189:        case O_DOT:
        !           190:        case O_ARROW:
        !           191:                return sf("%s%s%s", left(sub1), OpName(op), sub2->text());
        !           192:        }
        !           193:        if( sub1->edisc==E_BINARY && Prec(sub1->op) == Prec(op) )
        !           194:                return sf("%s%s%s", sub1->text(), OpName(op), under(sub2));
        !           195:        return sf("%s%s%s", under(sub1), OpName(op), under(sub2));
        !           196: }
        !           197: 
        !           198: char *Expr::text()
        !           199: {
        !           200:        trace( "%d.text()", this );     OK("text");
        !           201:        switch( edisc ){
        !           202:        case E_DCONST:  if( !id || !*id ) return sf( "%g", val.dbl );
        !           203:        case E_LCONST:  if( !id || !*id ) return sf( "%d", val.lng );
        !           204:        case E_ID:      return id;
        !           205:        case E_UNARY:   return textunary();
        !           206:        case E_BINARY:  return textbinary();
        !           207:        case E_SYMNODE:
        !           208:                if( !sym || !sym->_text ) return "<symbol>";
        !           209:                return sym->_text;
        !           210:        }
        !           211:        return "<expr>";
        !           212: }
        !           213: 
        !           214: int Expr::format()
        !           215: {
        !           216:        trace( "%d.format() %s %s", this, text(), type.text() ); OK(F_HEX);
        !           217:        return type.format();
        !           218: }
        !           219: 
        !           220: void Expr::reformat(int over, int stars)
        !           221: {
        !           222:        trace("%d.reformat(0x%X,%d) %s %s",this,over,stars,text(),type.text()); VOK;
        !           223:        switch( edisc ){
        !           224:        case E_ID: break;               // shut up cfront
        !           225:        case E_LCONST:
        !           226:        case E_DCONST:
        !           227:                type.reformat(over);
        !           228:                return;
        !           229:        case E_SYMNODE:
        !           230:                IF_LIVE(!sym) return;
        !           231:                ((Var*)sym)->type.reformat(over,stars);
        !           232:                return;
        !           233:        case E_UNARY:
        !           234:                switch( op ){
        !           235:                case O_REF:
        !           236:                case O_SIZEOF:
        !           237:                case O_CAST:
        !           238:                        type.reformat(over,stars);
        !           239:                        return;
        !           240:                case O_DEREF:
        !           241:                        sub1->reformat(over,stars+1);
        !           242:                        return;
        !           243:                case O_MINUS:
        !           244:                case O_1SCOMP:
        !           245:                case O_LOGNOT:
        !           246:                        sub1->reformat(over,stars);
        !           247:                        return;
        !           248:                }
        !           249:        case E_BINARY:
        !           250:                switch( op ){
        !           251:                case O_DIV:
        !           252:                case O_MULT:
        !           253:                case O_MOD:
        !           254:                case O_PLUS:
        !           255:                case O_MINUS:
        !           256:                case O_BITAND:
        !           257:                case O_BITOR:
        !           258:                case O_BITXOR:
        !           259:                        sub1->reformat(over,stars);             /* fall thru */
        !           260:                case O_ARROW:
        !           261:                case O_DOT:
        !           262:                case O_COMMA:
        !           263:                        sub2->reformat(over,stars);
        !           264:                        return;
        !           265:                case O_ASSIGN:
        !           266:                case O_ENV:
        !           267:                case O_LSHIFT:
        !           268:                case O_RSHIFT:
        !           269:                        sub1->reformat(over,stars);
        !           270:                        return;
        !           271:                case O_INDEX:
        !           272:                        sub1->reformat(over,stars+1);
        !           273:                        return;
        !           274:                case O_CALL:
        !           275:                        Func *func = (Func*) sub1->sym;
        !           276:                        if( func->ok() )
        !           277:                                func->type.decref()->reformat(over,stars);
        !           278:                        return;
        !           279:                }
        !           280:        }
        !           281: }
        !           282: 
        !           283: char *Expr::ascii(Frame *frame, int limit )
        !           284: {
        !           285:        static char buf[256];
        !           286:        char *raw, *fail = "fail";              // fail is unique
        !           287:        Bls esc( "\"" );
        !           288:        int i;
        !           289: 
        !           290:        trace("%d.ascii(%d)", this, frame);     OK("ascii");
        !           291:        if( !val.lng ) return "0";
        !           292:        raw = frame->peekstring(val.lng, fail);
        !           293:        if( raw == fail ) return "<string error>";
        !           294:        for( i = 0; raw[i] && i<limit; ++i )
        !           295:                esc.af( "%s",  FmtByte(raw[i]) );
        !           296:        if( raw[i] ) esc.af("...");
        !           297:        sprintf(buf, "%s\"", esc.text);
        !           298:        return buf;
        !           299: }
        !           300: 
        !           301: char *Expr::enumformat()
        !           302: {
        !           303:        Var *m;
        !           304: 
        !           305:        trace( "%d.enumformat()", this ); OK("enumformat");
        !           306:        TypMems tm(type.utype());
        !           307:        while( m = tm.gen() )
        !           308:                if( m->range.lo == val.lng )
        !           309:                        return m->text();
        !           310:        return sf( "(enum %s)%d", type.utype()?type.utype()->text():"?", val.lng );
        !           311: }
        !           312: 
        !           313: char *Expr::utypeformat(Frame *frame, Bls &build)
        !           314: {
        !           315:        Var *m, g(0,0,0,0,"$pi.Expr::utypeformat");
        !           316:        int i = 0;
        !           317:        Expr e;         /* use constructor to guarantee zeros? */
        !           318: 
        !           319:        trace("%d.utypeformat(%d,%d)", this, frame, &build); OK("utypeformat");
        !           320:        build.af("{");
        !           321:        TypMems tm(type.utype());
        !           322:        while( m = tm.gen() ) if( m->showorhide==SHOW ){
        !           323:                Bls local;
        !           324:                g = *m;
        !           325:                g._disc = U_GLB;
        !           326:                DType t = g.type;
        !           327:                e.bitaddr = 0;
        !           328:                if( t.pcc==BITS || t.pcc==UBITS ){
        !           329:                        e.bitaddr = g.range.lo;
        !           330:                        g.range.lo = 0;
        !           331:                }
        !           332:                g.range.lo += addr;
        !           333:                e.edisc = E_SYMNODE;
        !           334:                e.sym = &g;
        !           335:                e.spy = 0;
        !           336:                e.op = 0;
        !           337:                build.af( "%s%s", i++?",":"", e.evaltext(frame,local) );
        !           338:        }
        !           339:        build.af( "}" );
        !           340:        return build.text;
        !           341: }
        !           342: 
        !           343: char *Expr::evaltextcomma(Frame *frame, Bls &build)
        !           344: {
        !           345:        trace("%d.evaltextcomma(%d,%d)",this,frame,&build);OK("evaltextcomma");
        !           346:        sub1->evaltext(frame,build);
        !           347:        if( evalerr = sub1->evalerr ) return build.text;
        !           348:        build.af(", ");
        !           349:        sub2->evaltext(frame,build);
        !           350:        evalerr = sub2->evalerr;
        !           351:        type = sub2->type;
        !           352:        val = sub2->val;
        !           353:        return build.text;
        !           354: }
        !           355: 
        !           356: char *Expr::evaltext(Frame *frame, Bls &build)
        !           357: {
        !           358:        trace( "%d.evaltext(%d,%d)", this, frame, &build ); OK("evaltext");
        !           359:        evalerr = 0;
        !           360:        doevaltext(frame, build);
        !           361:        if( evalerr ) setspy(0);
        !           362:        if( spy ) spy->b = build.text;
        !           363:        return build.text;
        !           364: }
        !           365: 
        !           366: char *Expr::doevaltext(Frame *frame, Bls &build)
        !           367: {
        !           368:        char *error, *t;
        !           369:        Format form(0, frame->core->symtab());
        !           370: 
        !           371:        trace( "%d.doevaltext(%d,%d)", this, frame, &build ); OK("doevaltext");
        !           372:        switch( op ){
        !           373:                case O_QINDEX:
        !           374:                case O_QCAST:
        !           375:                case O_QENUM:
        !           376:                case O_QSTRUCT:
        !           377:                        build.af( "%s", text() );
        !           378:                        return build.text;
        !           379:        }
        !           380:        if( op == O_COMMA ) return evaltextcomma(frame,build);
        !           381:        if( error = eval(frame) ){
        !           382:                build.af( "%s: %s", text(), error );
        !           383:                evalerr = 1;
        !           384:                return build.text;
        !           385:        }
        !           386:        if( spy ) build.af(">>> ");     
        !           387:        build.af( "%s", text() );
        !           388:        if( type.pcc == VOID ){
        !           389:                build.af( "=void" );
        !           390:                return build.text;
        !           391:        }
        !           392:        form.format = format();
        !           393:        if( form.format&(F_DOUBLE|F_FLOAT) && val.floaterror() ){
        !           394:                build.af( "=<%s>", val.floaterror() );
        !           395:                form.format &= ~(F_DOUBLE|F_FLOAT);
        !           396:                form.format |= F_NONE;
        !           397:        }
        !           398:        if( form.format&F_ARY ) val.lng = addr;
        !           399:        if( *(t = form.f(val.lng,val.dbl)) )
        !           400:                build.af( "=%s", t );
        !           401:        if( form.format&F_UTYPE ){
        !           402:                build.af( "=");
        !           403:                utypeformat(frame,build);
        !           404:        }       
        !           405:        if( form.format&F_ENUM )
        !           406:                build.af( "=%s", enumformat() );        
        !           407:        if( form.format&F_STRING )
        !           408:                build.af( "=%s", ascii(frame, form.format&F_LONGSTRING?200:20) );
        !           409:        return build.text;
        !           410: }
        !           411: 
        !           412: char *Expr::getval( Frame *frame )             /* addr already fixed */
        !           413: {
        !           414:        trace( "%d.getval(%d)", this, frame );  OK("getval");
        !           415:        if( sym ){
        !           416:                switch( sym->disc() ){
        !           417:                case U_FUNC:
        !           418:                        type = ((Func*)sym)->type;
        !           419:                        break;
        !           420:                default:
        !           421:                        type = ((Var*)sym)->type;
        !           422:                        ((Var*)sym)->show(SHOW);
        !           423:                }
        !           424:        }
        !           425:        if( !addr ) return "addressing error: 0";
        !           426:        addr += (bitaddr>>5)*4;
        !           427:        if( type.isscalar() ){
        !           428:                Cslfd *m = frame->peek(addr,0);
        !           429:                if( !m ) return sf( "addressing error: 0x%X", addr );
        !           430:                val = *m;
        !           431:        }
        !           432:        if( type.isscalar() ) switch( type.pcc ){
        !           433:                case BITS:
        !           434:                case UBITS:     val.lng = (val.lng>>(bitaddr&31));
        !           435:                                int power = 1<<type.dim;
        !           436:                                val.lng &= power-1;
        !           437:                                if( type.pcc==BITS && val.lng&(power>>1) )
        !           438:                                        val.lng |= ~(power-1);
        !           439:                                addr = 0; break;
        !           440:                case CHAR:      val.lng = (char)  val.chr;              break;
        !           441:                case UCHAR:     val.lng = (unsigned char) val.chr;      break;
        !           442:                case SHORT:     val.lng = (short) val.sht;              break;
        !           443:                case USHORT:    val.lng = (unsigned short) val.sht;     break;
        !           444:                case FLOAT:     val.dbl = val.flt;                      break;
        !           445:        }
        !           446:        trace( "%s %d %d %s", sym?sym->_text:"", addr, val.lng, type.text() );
        !           447:        return 0;
        !           448: }
        !           449: 
        !           450: char *Expr::invalidoperands(char *more)
        !           451: {
        !           452:        trace( "%d.invalidoperands()", this );  OK("invalidoperands");
        !           453:        char *colon = more ? ": " : "";
        !           454:        if( !more ) more = "";
        !           455:        return sf( "invalid operand(s) of %s%s%s", OpName(op), colon, more );
        !           456: }
        !           457: 
        !           458: char *Expr::eval(Frame *frame)
        !           459: {
        !           460:        trace( "%d.eval(%d)", this, frame );    OK("eval");
        !           461:        switch(op){
        !           462:                case O_QINDEX:
        !           463:                case O_QCAST:
        !           464:                case O_QSTRUCT:
        !           465:                case O_QENUM:
        !           466:                        return "? stops evaluation";
        !           467:        }
        !           468:        switch( edisc ){
        !           469:        case E_DCONST:
        !           470:        case E_LCONST:
        !           471:                return 0;                               /* should all be there! */
        !           472:        case E_UNARY:
        !           473:                return evalunary(frame);
        !           474:        case E_BINARY:
        !           475:                return evalbinary(frame);
        !           476:        case E_ID:
        !           477:                if( !(sym = frame->idtosym(id)) )
        !           478:                        return enumid(frame);
        !           479:                edisc = E_SYMNODE;                              /* fall thru  */
        !           480:        case E_SYMNODE:
        !           481:                if( !sym ) return "symbol table error";
        !           482:                addr = frame->locate((Var*)sym);
        !           483:                return getval(frame);
        !           484:        default:
        !           485:                return "not an expression";
        !           486:        }
        !           487: }
        !           488: 
        !           489: char *Expr::enumid(Frame *frame)
        !           490: {
        !           491:        trace("%d.enumid(%s)", this, id); OK("enumid");
        !           492:        UType *u; Var *v;
        !           493:        for( u = frame->symtab()->utypelist(); u; u = (UType*) u->rsib ){
        !           494:                if( u->type.pcc != ENUMTY ) continue;
        !           495:                TypMems tm(u);
        !           496:                while( v = tm.gen() ) if( !strcmp(id, v->_text) ){
        !           497:                        edisc = E_LCONST;
        !           498:                        val.lng = v->range.lo;
        !           499:                        addr = 0;
        !           500:                        type.pcc = LONG;
        !           501:                        return 0;
        !           502:                }
        !           503:        }
        !           504:        return sf("not found: %s", id);
        !           505: }
        !           506: 
        !           507: char *Expr::evalindex(Frame *frame, Expr *ap, long i)
        !           508: {
        !           509:        long size = 0;
        !           510: 
        !           511:        trace( "%d.index(%d,%d,%d)", this, frame, ap, i); OK("evalindex");
        !           512:        if( ap && ap->type.decref() ){ 
        !           513:                type = *ap->type.decref();
        !           514:                size = type.size_of();
        !           515:        }
        !           516:        if( !size ) return "cannot determine size for []";
        !           517:        addr = ap->type.isptr() ? ap->val.lng : ap->addr;
        !           518:        if( !addr ) return "zero pointer for []";
        !           519:        addr += size * i;
        !           520:        return getval(frame);
        !           521: }
        !           522: 
        !           523: char *Expr::evaldotarrow(Frame *frame)
        !           524: {
        !           525:        DType t;
        !           526: 
        !           527:        trace( "%d.evaldotarrow(%d)", this, frame ); OK("evaldotarrow");
        !           528:        addr = 0;
        !           529:        t = sub1->type;
        !           530:        if( op == O_DOT )
        !           531:                addr = sub1->addr;
        !           532:        else {
        !           533:                if( !t.isptr() ) return invalidoperands("not a pointer");
        !           534:                t = *t.decref();
        !           535:                addr = sub1->val.lng;
        !           536:        }
        !           537:        if( !t.isstrun() ) return invalidoperands("not a struct/union");
        !           538:        if( !addr ) return invalidoperands("zero address");
        !           539:        if( sub2->edisc == E_ID && t.utype() ){
        !           540:                TypMems tm(t.utype());
        !           541:                while( sub2->sym = tm.gen() )
        !           542:                        if( !strcmp(sub2->sym->text(), sub2->id) ) break;
        !           543:                if( !sub2->sym ) return sf("not found: %s", sub2->id);
        !           544:                sub2->edisc = E_SYMNODE;
        !           545:        }
        !           546:        if( sub2->edisc != E_SYMNODE ) return invalidoperands();
        !           547:        sym = sub2->sym;
        !           548:        t = ((Var*)sym)->type;
        !           549:        if( t.pcc==BITS || t.pcc==UBITS )
        !           550:                bitaddr = sym->range.lo;
        !           551:        else
        !           552:                addr += sym->range.lo;
        !           553:        return getval(frame);
        !           554: }
        !           555: 
        !           556: Expr *Expr::actual(int a)
        !           557: {
        !           558:        int m;
        !           559:        trace( "%d.actual(%d)", this, a );
        !           560:        if( !this ) return 0;
        !           561:        if( op != O_COMMA ) return a == 1 ? this : 0;
        !           562:        for( m = 0; sub1->actual(m+1); ++m ) {}
        !           563:        return a<=m ? sub1->actual(a) : sub2->actual(a-m);
        !           564: }
        !           565: 
        !           566: char *Expr::evalcall(Frame *frame)
        !           567: {
        !           568:        char *error = 0;
        !           569:        long i, nargs = 1, argwords = 0, regloc;
        !           570:        Var *formal;
        !           571:        Func *func;
        !           572:        Frame *called = 0;
        !           573: 
        !           574:        trace( "%d.evalcall(%d)", this, frame ); OK("evalcall");
        !           575:        if( !sub1 )return "<fcn call>";
        !           576:        switch( sub1->edisc ){
        !           577:        default: return "<fcn call>";
        !           578:        case E_ID:
        !           579:                sub1->sym = frame->symtab()->idtosym( U_FUNC, sub1->id );
        !           580:        case E_SYMNODE:
        !           581:                func = (Func*) sub1->sym;
        !           582:        }
        !           583:        if( !func ) return sf( "not a function: %s", sub1->id );
        !           584:        while( sub2->actual(nargs) )
        !           585:                if( !(formal = func->argument(nargs)) )
        !           586:                        return "too many args in function call";
        !           587:                else {
        !           588:                        argwords += (formal->type.size_of()+3)/4;   /* vax & 32 */
        !           589:                        ++nargs;
        !           590:                }
        !           591:        if( func->argument(nargs) )
        !           592:                return "too few args in function call";
        !           593:        --nargs;
        !           594:        Core *core = frame->core;
        !           595:        if( !core->online() )
        !           596:                return "cannot call function in dump";
        !           597:        Context *cc = core->newContext();
        !           598:        if( cc->error ){
        !           599:                delete cc;
        !           600:                return cc->error;
        !           601:        }
        !           602:        called = new Frame(core);
        !           603:        called->ap = core->apforcall(argwords*4);
        !           604:        if( !called->ap )
        !           605:                { error = "function calling is broken"; goto Restore; }
        !           606:        for( i = 1; i <= nargs; ++i ){
        !           607:                Expr *e;
        !           608:                e = E_Binary(E_Sym(func->argument(i)), O_ASSIGN, sub2->actual(i));
        !           609:                if( error = e->sub1->eval(called) )
        !           610:                        { error = sf("formal arg: %s", error); goto Restore; }
        !           611:                if( error = e->sub2->eval(frame) )
        !           612:                        { error = sf("actual arg: %s", error); goto Restore; }
        !           613:                if( error = e->evalassign(called) )
        !           614:                        { error = sf("arg assign: %s", error); goto Restore; }
        !           615:        }
        !           616:        if( error = core->docall(func->range.lo, argwords) )
        !           617:                goto Restore;
        !           618:        type = *func->type.decref();
        !           619:        regloc = core->returnregloc();
        !           620:        if( !regloc)
        !           621:                { error = "function return error"; goto Restore; }
        !           622:        if( type.isstrun() )
        !           623:                switch( type.size_of() ){
        !           624:                case 1:
        !           625:                case 2:
        !           626:                case 4: addr = regloc;
        !           627:                        break;
        !           628:                default:
        !           629:                        addr = core->peek(regloc)->lng;
        !           630:                }
        !           631:        else
        !           632:                val = *core->peek(regloc);              /* doubles? */
        !           633: Restore:
        !           634:        if( called ) delete called;
        !           635:        cc->restore();
        !           636:        if( cc->error && !error )
        !           637:                error = cc->error;
        !           638:        delete cc;
        !           639:        return error;
        !           640: }
        !           641: 
        !           642: char *Expr::evalenv(Frame *frame)
        !           643: {
        !           644:        Frame *env = frame ? frame->caller() : 0;
        !           645:        char *error;
        !           646: 
        !           647:        trace( "%d.evalenv(%d)", this, frame ); OK("evalenv");
        !           648:        for( ; env; env = env->caller() )
        !           649:                if( !strcmp(env->func->_text,sub2->text()) )
        !           650:                        break;
        !           651:        if( !env ){
        !           652:                Frame sta(frame->core);
        !           653:                sta.func = (Func*) sta.core->symtab()->idtosym(U_FUNC,sub2->text());
        !           654:                if( sta.func ) env = &sta;
        !           655:        }
        !           656:        if( !env ) return sf( "not found: %s()", sub2->text() );
        !           657:        if( error = sub1->eval(env) ) return error;
        !           658:        type = sub1->type;
        !           659:        val = sub1->val;
        !           660:        addr = sub1->addr;
        !           661:        return 0;
        !           662: }
        !           663: 
        !           664: char *Expr::evalrange()
        !           665: {
        !           666:        const int range = 32;
        !           667:        trace( "%d.evalrange()", this ); OK("evalrange");
        !           668:        if( !sub1->type.isintegral()
        !           669:         || !sub2->type.isintegral()
        !           670:         ||  sub2->val.lng < sub1->val.lng )
        !           671:                return invalidoperands();
        !           672:        if( sub2->val.lng > sub1->val.lng+range )
        !           673:                return sf( "range may not exceed %d", range );
        !           674:        return 0;
        !           675: }
        !           676: 
        !           677: void Expr::catchfpe()          /* VAX: put down operands that must succeed */
        !           678: {                              /*      and restart the instruction         */
        !           679:        *fp1 = *fp2 = 1.0;
        !           680:        fpe = "floating point exception";
        !           681: }
        !           682: 
        !           683: char *Expr::evalflop()         /* VAX host */
        !           684: {
        !           685:        char *error;
        !           686:        trace( "%d.evalflop()", this ); OK("evalflop");
        !           687:        if( !sub1->type.isreal() || !sub2->type.isreal() )
        !           688:                return invalidoperands();
        !           689:        if( (error = sub1->floaterror())
        !           690:         || (error = sub2->floaterror()) )
        !           691:                         return invalidoperands(error);
        !           692:        type.pcc = DOUBLE;
        !           693:        double l = sub1->val.dbl; double r = sub2->val.dbl;
        !           694:        fp1 = &l;
        !           695:        fp2 = &r;
        !           696:        fpe = 0;
        !           697:        signal(SIGFPE, (SIG_TYP)&Expr::catchfpe);
        !           698:        switch( op ){
        !           699:                case O_MULT:    val.dbl = l*r;  break;
        !           700:                case O_DIV:     val.dbl = l/r;  break;
        !           701:                case O_PLUS:    val.dbl = l+r;  break;
        !           702:                case O_MINUS:   val.dbl = l-r;  break;
        !           703:        }
        !           704:        signal(SIGFPE, (SIG_TYP)SIG_DFL);
        !           705:        return fpe;
        !           706: }
        !           707: 
        !           708: char *Expr::evalbinary(Frame *frame)
        !           709: {
        !           710:        char *error;
        !           711:        long size;
        !           712: 
        !           713:        trace( "%d.evalbinary(%d)", this, frame ); OK("evalbinary");
        !           714:        IF_LIVE( edisc!=E_BINARY ) return "<binary expr>";
        !           715:        switch( op ){
        !           716:                case O_CALL: return evalcall(frame);
        !           717:                case O_ENV:  return evalenv(frame);
        !           718:        }
        !           719:        if( error = sub1->eval(frame) ) return error;
        !           720:        type.pcc = LONG;
        !           721:        type.over = sub1->type.over;
        !           722:        if( op==O_DOT || op==O_ARROW ) return evaldotarrow(frame);
        !           723:        if( (sub1->type.isintegral() && op==O_LOGAND && !sub1->val.lng)
        !           724:         || (sub1->type.isintegral() && op==O_LOGOR  &&  sub1->val.lng) ){
        !           725:                val = sub1->val.lng != 0;
        !           726:                return 0;
        !           727:        }
        !           728:        if( error = sub2->eval(frame) ) return error;
        !           729:        type.over |= sub2->type.over;
        !           730:        switch( op ){
        !           731:        case O_RANGE:
        !           732:                return evalrange();
        !           733:        case O_COMMA:
        !           734:                type = sub2->type;
        !           735:                val = sub2->val;
        !           736:                return 0;
        !           737:        case O_ASSIGN:
        !           738:                return evalassign(frame);
        !           739:        case O_MULT:
        !           740:        case O_DIV:
        !           741:                if( sub1->type.isreal() || sub2->type.isreal() )
        !           742:                        return evalflop();
        !           743:        case O_MOD:
        !           744:                if( sub1->type.isintegral() && sub2->type.isintegral() ){
        !           745:                        long l = sub1->val.lng, r = sub2->val.lng;
        !           746:                        if( op==O_MULT ){
        !           747:                                val.lng = l * r;
        !           748:                                return 0;
        !           749:                        }
        !           750:                        if( r == 0 ) return "zero divide";
        !           751:                        if( op == O_MOD ) val.lng = l % r;
        !           752:                        if( op == O_DIV ) val.lng = l / r;
        !           753:                        return 0;
        !           754:                }
        !           755:                return invalidoperands();
        !           756:        case O_LOGAND:
        !           757:        case O_LOGOR:
        !           758:        case O_BITAND:
        !           759:        case O_BITOR:
        !           760:        case O_BITXOR:
        !           761:        case O_LSHIFT:
        !           762:        case O_RSHIFT:
        !           763:                if( sub1->type.isscalar() && sub2->type.isscalar() ){
        !           764:                        long l = sub1->val.lng, r = sub2->val.lng;
        !           765:                        switch( op ){
        !           766:                                case O_LOGAND:  val.lng = l && r; return 0;
        !           767:                                case O_LOGOR:   val.lng = l || r; return 0;
        !           768:                                case O_BITAND:  val.lng = l &  r; return 0;
        !           769:                                case O_BITOR:   val.lng = l |  r; return 0;
        !           770:                                case O_BITXOR:  val.lng = l ^  r; return 0;
        !           771:                                case O_LSHIFT:  val.lng = l << r; return 0;
        !           772:                                case O_RSHIFT:  val.lng = l >> r; return 0;
        !           773:                        }
        !           774:                        return "internal error: && || & | ^ >> <<";
        !           775:                }
        !           776:                return invalidoperands();
        !           777:        case O_EQ:
        !           778:        case O_NE:
        !           779:        case O_LT:
        !           780:        case O_GT:
        !           781:        case O_LE:
        !           782:        case O_GE:
        !           783:                type.pcc = LONG;
        !           784:                if( sub1->type.isreal() || sub2->type.isreal() ){
        !           785:                        if( !sub1->type.isreal() || sub1->floaterror()
        !           786:                         || !sub2->type.isreal() || sub2->floaterror() )
        !           787:                                return invalidoperands();
        !           788:                        double l = sub1->val.dbl; double r = sub2->val.dbl; /*C++*/
        !           789:                        switch( op ){
        !           790:                                case O_EQ: val.lng = l == r; return 0;
        !           791:                                case O_NE: val.lng = l != r; return 0;
        !           792:                                case O_LT: val.lng = l <  r; return 0;
        !           793:                                case O_GT: val.lng = l >  r; return 0;
        !           794:                                case O_LE: val.lng = l <= r; return 0;
        !           795:                                case O_GE: val.lng = l >= r; return 0;
        !           796:                        }
        !           797:                        return "internal floating relation error";
        !           798:                }
        !           799:                if( sub1->type.isscalar() && sub2->type.isscalar() ){
        !           800:                        long l = sub1->val.lng, r = sub2->val.lng;
        !           801:                        switch( op ){
        !           802:                                case O_EQ: val.lng = l == r; return 0;
        !           803:                                case O_NE: val.lng = l != r; return 0;
        !           804:                                case O_LT: val.lng = l <  r; return 0;
        !           805:                                case O_GT: val.lng = l >  r; return 0;
        !           806:                                case O_LE: val.lng = l <= r; return 0;
        !           807:                                case O_GE: val.lng = l >= r; return 0;
        !           808:                        }
        !           809:                        return "internal fixed point relation error";
        !           810:                }
        !           811:                return invalidoperands();
        !           812:        case O_PLUS:
        !           813:        case O_MINUS:
        !           814:                if( sub1->type.isreal() || sub2->type.isreal() )
        !           815:                        return evalflop();
        !           816:                if( sub1->type.isary() ){
        !           817:                        sub1->val.lng = sub1->addr;
        !           818:                        sub1->type = sub1->type.decref()->incref();
        !           819:                }
        !           820:                if( sub2->type.isary() ){
        !           821:                        sub2->val.lng = sub2->addr;
        !           822:                        sub2->type = sub2->type.decref()->incref();
        !           823:                }
        !           824:                if( sub1->type.isptr() && sub2->type.isintegral() ){
        !           825:                        size = sub1->type.decref()->size_of();
        !           826:                        if( !size ) return "pointer arithmetic error";
        !           827:                        if( op == O_MINUS ) size = -size;
        !           828:                        val.lng = sub1->val.lng + size*sub2->val.lng;
        !           829:                        type = sub1->type;
        !           830:                        return 0;
        !           831:                }
        !           832:                if( sub1->type.isintegral() && sub2->type.isintegral() ){
        !           833:                        if( op==O_PLUS ) val.lng = sub1->val.lng+sub2->val.lng;
        !           834:                        if( op==O_MINUS) val.lng = sub1->val.lng-sub2->val.lng;
        !           835:                        return 0;
        !           836:                }
        !           837:                if( op == O_PLUS && sub2->type.isptr() && sub1->type.isintegral() ){
        !           838:                        size = sub2->type.decref()->size_of();
        !           839:                        if( !size ) return "pointer arithmetic error";
        !           840:                        val.lng = sub2->val.lng + size*sub1->val.lng;
        !           841:                        type = sub2->type;
        !           842:                        return 0;
        !           843:                }
        !           844:                if( op == O_MINUS && sub1->type.isptr() && sub2->type.isptr() ){
        !           845:                        size = sub1->type.decref()->size_of();
        !           846:                        if( !size || size!=sub2->type.decref()->size_of() )
        !           847:                                return "pointer-pointer size error";
        !           848:                        val.lng = (sub1->val.lng - sub2->val.lng)/size;
        !           849:                        type.over |= sub2->type.over;
        !           850:                        return 0;
        !           851:                }
        !           852:                return invalidoperands();
        !           853:        case O_INDEX:
        !           854:                trace( "%s %s", sub1->type.text(), sub2->type.text() );
        !           855:                if( sub1->type.isaryorptr() && sub2->type.isintegral() )
        !           856:                        return evalindex(frame,sub1,sub2->val.lng);
        !           857:                if( sub2->type.isaryorptr() && sub1->type.isintegral() )
        !           858:                        return evalindex(frame,sub2,sub1->val.lng);
        !           859:                return invalidoperands();
        !           860:        default:
        !           861:                return "binary operator not implemented";
        !           862:        }
        !           863: }
        !           864: 
        !           865: char *Expr::evalcast()
        !           866: {
        !           867:        trace( "%d.evalcast()", this,  ); OK("<cast>");
        !           868:        IF_LIVE( !type.isscalar() || !sub1->type.isscalar() ) return "cast error";
        !           869:        if( type.isptr() ){
        !           870:                if( sub1->type.isreal() ) return "can't cast float to ptr";
        !           871:                val.lng = sub1->val.lng;
        !           872:                return 0;
        !           873:        }
        !           874:        if( type.isreal() ){
        !           875:                if( sub1->floaterror() ) return sub1->floaterror();
        !           876:                val.dbl = sub1->val.dbl;
        !           877:                if( !sub1->type.isreal() ) val.dbl = sub1->val.lng;
        !           878:                return 0;
        !           879:        }
        !           880:        val.lng = sub1->val.lng;
        !           881:        switch( sub1->type.pcc ){
        !           882:                case CHAR:      val.lng = sub1->val.chr; break;
        !           883:                case SHORT:     val.lng = sub1->val.sht; break;
        !           884:        }
        !           885:        addr = sub1->addr;
        !           886:        if( sub1->type.isreal() ){
        !           887:                if( sub1->floaterror() ) return sub1->floaterror();
        !           888:                val.lng = (long) sub1->val.dbl;
        !           889:        }
        !           890:        return 0;
        !           891: }
        !           892: 
        !           893: char *Expr::evalunary(Frame *frame)
        !           894: {
        !           895:        char *error;
        !           896: 
        !           897:        trace( "%d.evalunary(%d)", this, frame ); OK("<unary expr>");
        !           898:        IF_LIVE( edisc!=E_UNARY ) return "<unary expr>";
        !           899:        if( error = sub1->eval(frame) ) return error;
        !           900:        switch( op ){
        !           901:        case O_SPECIAL:
        !           902:                type = sub1->type;
        !           903:                addr = sub1->addr;
        !           904:                val = sub1->val;
        !           905:                if( type.isptr() ){
        !           906:                        type = *type.decref();
        !           907:                        addr = val.lng;
        !           908:                }
        !           909:                if( !type.pcc == STRTY || !type.utype() || !addr )
        !           910:                        return invalidoperands();
        !           911:                error = frame->special( type.utype()->text(), addr );
        !           912:                {                       // cfront bug
        !           913:                DType nulltype;
        !           914:                type = nulltype;
        !           915:                }                       // cfront bug
        !           916:                type.pcc = VOID;
        !           917:                addr = 0;
        !           918:                return error;
        !           919:        case O_1SCOMP:
        !           920:        case O_LOGNOT:
        !           921:                if( !sub1->type.isscalar() || sub1->type.isreal() )
        !           922:                        return invalidoperands();
        !           923:                type.pcc = LONG;
        !           924:                if( op == O_1SCOMP ) val.lng = ~sub1->val.lng;
        !           925:                if( op == O_LOGNOT ) val.lng = !sub1->val.lng;
        !           926:                return 0;
        !           927:        case O_FABS:
        !           928:                {                               // cfront bug
        !           929:                double fabs(double);
        !           930:                if( !sub1->type.isreal() || sub1->floaterror() )
        !           931:                        return invalidoperands();
        !           932:                type.pcc = DOUBLE;
        !           933:                val.dbl = fabs(sub1->val.dbl);          /* cannot fail? */
        !           934:                return 0;
        !           935:                }                               // cfront bug
        !           936:        case O_MINUS:
        !           937:                if( sub1->type.isintegral() ){
        !           938:                        type.pcc = LONG;
        !           939:                        val.lng = -sub1->val.lng;
        !           940:                        return 0;
        !           941:                }
        !           942:                if( sub1->type.isreal() && !sub1->floaterror() ){
        !           943:                        type.pcc = DOUBLE;
        !           944:                        val.dbl = -sub1->val.dbl;
        !           945:                        return 0;
        !           946:                }
        !           947:                return invalidoperands();
        !           948:        case O_CAST:
        !           949:                return evalcast();
        !           950:        case O_SIZEOF:
        !           951:                addr = 0;
        !           952:                if( !(val.lng = sub1->type.size_of())) return "sizeof error";
        !           953:                type.pcc = UNSIGNED;
        !           954:                return 0;
        !           955:        case O_TYPEOF:
        !           956:                return sub1->type.text();
        !           957:        case O_DEREF:
        !           958:                if( !sub1->type.isptr() ) return "unary * applied to non-pointer";
        !           959:                addr = sub1->val.lng;
        !           960:                type = *sub1->type.decref();
        !           961:                return getval(frame);
        !           962:        case O_REF:
        !           963:                if( !sub1->addr ) return "unary & applied to non-lvalue";
        !           964:                addr = 0;
        !           965:                int o = type.over;
        !           966:                type = sub1->type.incref();
        !           967:                type.over = o;
        !           968:                val.lng = sub1->addr;
        !           969:                return 0;
        !           970:        default:
        !           971:                return "unary operator not implemented";
        !           972:        }
        !           973: }
        !           974: 
        !           975: Index Expr::castcarte()
        !           976: {
        !           977:        static Index *ix;
        !           978:        static short bt[] = { DOUBLE, FLOAT, LONG, SHORT, CHAR, 0 };
        !           979:        Menu m;
        !           980:        DType *base, *ptr;
        !           981: 
        !           982:        trace( "%d.castcarte()", this );        OK(ZIndex);
        !           983:        if (!ix)
        !           984:                ix = new Index(0,0);
        !           985:        if( !ix->null() ) return *ix;
        !           986:        Action a = (Action)&Phrase::applycast;
        !           987:        int t;
        !           988:        for( t = 0; bt[t]; ++t ){
        !           989:                base = new DType();
        !           990:                base->pcc = bt[t];
        !           991:                ptr = new DType;
        !           992:                *ptr = base->incref();
        !           993:                m.first( sf( "%s\240",  ptr->text() ), a, (long) ptr  );
        !           994:                m.first( sf( " %s\240", base->text()), a, (long) base );
        !           995:        }
        !           996:        Action i = (Action)&Phrase::increfcast;
        !           997:        Action e = (Action)&Phrase::enumcast;
        !           998:        Action s = (Action)&Phrase::strcast;
        !           999:        m.last( "        * ? ", i, 1 );
        !          1000:        m.last( "     enum ? ", e, 0 );
        !          1001:        m.last( "  *struct ? ", s, 1 );
        !          1002:        return *ix = m.index("cast $");
        !          1003: }
        !          1004: 
        !          1005: Index Expr::carte(Frame *f)
        !          1006: {
        !          1007:        Menu m;
        !          1008: 
        !          1009:        trace( "%d.carte()", this );    OK(ZIndex);
        !          1010:        m.last( "eval $", (Action)&Phrase::evaluate );
        !          1011:        if( evalerr || op==O_TYPEOF )
        !          1012:                return m.index();
        !          1013:        switch( op ){
        !          1014:        case O_QCAST:
        !          1015:                return castcarte();
        !          1016:        case O_QSTRUCT:
        !          1017:                return f ? f->symtab()->utypecarte(STRTY)  : ZIndex;
        !          1018:        case O_QENUM:
        !          1019:                return f ? f->symtab()->utypecarte(ENUMTY) : ZIndex;
        !          1020:        case O_QINDEX:
        !          1021:                return NumericRange( -2, 20 );
        !          1022:        }
        !          1023:        if( !spy )
        !          1024:                m.first( "spy on $", (Action) &Phrase::setspy, 1 );
        !          1025:        else
        !          1026:                m.first( "unspy $",   (Action) &Phrase::setspy, 0 );
        !          1027:        if( addr ) m.last( "& $", (Action)&Phrase::applyunary, (long)O_REF );
        !          1028:        if( val.lng )
        !          1029:                m.last( "mem .=$ ", (Action)&Phrase::memory );
        !          1030:        if( type.isscalar() ) m.last(castcarte());
        !          1031:        m.last(type.carte());
        !          1032:        return m.index();
        !          1033: }
        !          1034: 
        !          1035: void Expr::setspy(long s)
        !          1036: {
        !          1037:        trace( "%d.setspy(%d)", this, s );      VOK;
        !          1038:        if( !s && spy ){
        !          1039:                delete spy;
        !          1040:                spy = 0;
        !          1041:        } else if( s )
        !          1042:                spy = new Spy;
        !          1043: }
        !          1044: 
        !          1045: char *Expr::evalassign(Frame *frame)
        !          1046: {
        !          1047:        char *error;
        !          1048: 
        !          1049:        trace( "%d.evalassign(%d)", this, frame ); OK("evalassign");
        !          1050:        if( !sub1->addr ) return "lhs of = does not yield an lvalue";
        !          1051:        if( sub2->type.isary() && sub2->addr ){
        !          1052:                sub2->type.pcc = PTR;
        !          1053:                sub2->val.lng = sub2->addr;
        !          1054:        }
        !          1055:        if( sub1->type.isscalar() && sub2->type.isscalar() ){
        !          1056:                DType type1 = sub1->type, type2 = sub2->type;
        !          1057:                long addr1 = sub1->addr, size1 = type1.size_of();
        !          1058:                Cslfd *val2 = &sub2->val;
        !          1059:                if( type1.isreal() || type1.isreal() ){
        !          1060:                        if( !type1.isreal() || !type2.isreal() )
        !          1061:                                return invalidoperands("mixed mode");
        !          1062:                        error = frame->pokedbl(addr1, val2->dbl, type1.size_of());
        !          1063:                } else
        !          1064:                        error = frame->poke(addr1, val2->lng, type1.size_of());
        !          1065:                if( error )
        !          1066:                        return sf( "%s: 0x%X", error, sub1->addr );
        !          1067:                if( error = sub1->eval(frame) )
        !          1068:                        return error;
        !          1069:                type = sub1->type;
        !          1070:                val = sub1->val;
        !          1071:                addr = 0;
        !          1072:                return 0;
        !          1073:        }
        !          1074:        if( sub1->type.isstrun() && sub2->type.isstrun() ){
        !          1075:                UType *u1 = sub1->type.utype(), *u2 = sub2->type.utype();
        !          1076:                if( !u1
        !          1077:                 || !u2
        !          1078:                 || u1->type.pcc != u2->type.pcc
        !          1079:                 || strcmp(u1->type.text(), u2->type.text()) )
        !          1080:                        return "incompatible struct/union =";
        !          1081:                if( !sub2->addr )
        !          1082:                        return "rhs of struct/union = does not yield an lvalue";
        !          1083:                error = frame->blockmove(sub2->addr,sub1->addr,u1->type.size_of());
        !          1084:                type = sub1->type;
        !          1085:                val = 0;
        !          1086:                addr = sub1->addr;
        !          1087:                return error;
        !          1088:        }
        !          1089:        return "invalid asssignment";
        !          1090: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.