Annotation of 43BSDReno/pgrm/f77/pass1.tahoe/putpcc.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1980 Regents of the University of California.
        !             3:  * All rights reserved.  The Berkeley software License Agreement
        !             4:  * specifies the terms and conditions for redistribution.
        !             5:  */
        !             6: 
        !             7: #ifndef lint
        !             8: static char sccsid[] = "@(#)putpcc.c   5.1 (Berkeley) 6/7/85";
        !             9: #endif not lint
        !            10: 
        !            11: /*
        !            12:  * putpcc.c
        !            13:  *
        !            14:  * Intermediate code generation for S. C. Johnson C compilers
        !            15:  * New version using binary polish postfix intermediate
        !            16:  *
        !            17:  * University of Utah CS Dept modification history:
        !            18:  *
        !            19:  * $Header: putpcc.c,v 3.2 85/03/25 09:35:57 root Exp $
        !            20:  * $Log:       putpcc.c,v $
        !            21:  * Revision 3.2  85/03/25  09:35:57  root
        !            22:  * fseek return -1 on error.
        !            23:  * 
        !            24:  * Revision 3.1  85/02/27  19:06:55  donn
        !            25:  * Changed to use pcc.h instead of pccdefs.h.
        !            26:  * 
        !            27:  * Revision 2.12  85/02/22  01:05:54  donn
        !            28:  * putaddr() didn't know about intrinsic functions...
        !            29:  * 
        !            30:  * Revision 2.11  84/11/28  21:28:49  donn
        !            31:  * Hacked putop() to handle any character expression being converted to int,
        !            32:  * not just function calls.  Previously it bombed on concatenations.
        !            33:  * 
        !            34:  * Revision 2.10  84/11/01  22:07:07  donn
        !            35:  * Yet another try at getting putop() to work right.  It appears that the
        !            36:  * second pass can't abide certain explicit conversions (e.g. short to long)
        !            37:  * so the conversion code in putop() tries to remove them.  I think this
        !            38:  * version (finally) works.
        !            39:  * 
        !            40:  * Revision 2.9  84/10/29  02:30:57  donn
        !            41:  * Earlier fix to putop() for conversions was insufficient -- we NEVER want to
        !            42:  * see the type of the left operand of the thing left over from stripping off
        !            43:  * conversions...
        !            44:  * 
        !            45:  * Revision 2.8  84/09/18  03:09:21  donn
        !            46:  * Fixed bug in putop() where the left operand of an addrblock was being
        !            47:  * extracted...  This caused an extremely obscure conversion error when
        !            48:  * an array of longs was subscripted by a short.
        !            49:  * 
        !            50:  * Revision 2.7  84/08/19  20:10:19  donn
        !            51:  * Removed stuff in putbranch that treats STGARG parameters specially -- the
        !            52:  * bug in the code generation pass that motivated it has been fixed.
        !            53:  * 
        !            54:  * Revision 2.6  84/08/07  21:32:23  donn
        !            55:  * Bumped the size of the buffer for the intermediate code file from 0.5K
        !            56:  * to 4K on a VAX.
        !            57:  * 
        !            58:  * Revision 2.5  84/08/04  20:26:43  donn
        !            59:  * Fixed a goof in the new putbranch() -- it now calls mkaltemp instead of
        !            60:  * mktemp().  Correction due to Jerry Berkman.
        !            61:  * 
        !            62:  * Revision 2.4  84/07/24  19:07:15  donn
        !            63:  * Fixed bug reported by Craig Leres in which putmnmx() mistakenly assumed
        !            64:  * that mkaltemp() returns tempblocks, and tried to free them with frtemp().
        !            65:  * 
        !            66:  * Revision 2.3  84/07/19  17:22:09  donn
        !            67:  * Changed putch1() so that OPPAREN expressions of type CHARACTER are legal.
        !            68:  * 
        !            69:  * Revision 2.2  84/07/19  12:30:38  donn
        !            70:  * Fixed a type clash in Bob Corbett's new putbranch().
        !            71:  * 
        !            72:  * Revision 2.1  84/07/19  12:04:27  donn
        !            73:  * Changed comment headers for UofU.
        !            74:  * 
        !            75:  * Revision 1.8  84/07/19  11:38:23  donn
        !            76:  * Replaced putbranch() routine so that you can ASSIGN into argument variables.
        !            77:  * The code is from Bob Corbett, donated by Jerry Berkman.
        !            78:  * 
        !            79:  * Revision 1.7  84/05/31  00:48:32  donn
        !            80:  * Fixed an extremely obscure bug dealing with the comparison of CHARACTER*1
        !            81:  * expressions -- a foulup in the order of COMOP and the comparison caused
        !            82:  * one operand of the comparison to be garbage.
        !            83:  * 
        !            84:  * Revision 1.6  84/04/16  09:54:19  donn
        !            85:  * Backed out earlier fix for bug where items in the argtemplist were
        !            86:  * (incorrectly) being given away; this is now fixed in mkargtemp().
        !            87:  * 
        !            88:  * Revision 1.5  84/03/23  22:49:48  donn
        !            89:  * Took out the initialization of the subroutine argument temporary list in
        !            90:  * putcall() -- it needs to be done once per statement instead of once per call.
        !            91:  * 
        !            92:  * Revision 1.4  84/03/01  06:48:05  donn
        !            93:  * Fixed bug in Bob Corbett's code for argument temporaries that caused an
        !            94:  * addrblock to get thrown out inadvertently when it was needed for recycling
        !            95:  * purposes later on.
        !            96:  * 
        !            97:  * Revision 1.3  84/02/26  06:32:38  donn
        !            98:  * Added Berkeley changes to move data definitions around and reduce offsets.
        !            99:  * 
        !           100:  * Revision 1.2  84/02/26  06:27:45  donn
        !           101:  * Added code to catch TTEMP values passed to putx().
        !           102:  * 
        !           103:  */
        !           104: 
        !           105: #if FAMILY != PCC
        !           106:        WRONG put FILE !!!!
        !           107: #endif
        !           108: 
        !           109: #include "defs.h"
        !           110: #include <pcc.h>
        !           111: 
        !           112: Addrp putcall(), putcxeq(), putcx1(), realpart();
        !           113: expptr imagpart();
        !           114: ftnint lencat();
        !           115: 
        !           116: #define FOUR 4
        !           117: extern int ops2[];
        !           118: extern int types2[];
        !           119: 
        !           120: #if HERE==VAX || HERE == TAHOE
        !           121: #define PCC_BUFFMAX 1024
        !           122: #else
        !           123: #define PCC_BUFFMAX 128
        !           124: #endif
        !           125: static long int p2buff[PCC_BUFFMAX];
        !           126: static long int *p2bufp                = &p2buff[0];
        !           127: static long int *p2bufend      = &p2buff[PCC_BUFFMAX];
        !           128: 
        !           129: 
        !           130: puthead(s, class)
        !           131: char *s;
        !           132: int class;
        !           133: {
        !           134: char buff[100];
        !           135: #if TARGET == VAX || TARGET == TAHOE
        !           136:        if(s)
        !           137:                p2ps("\t.globl\t_%s", s);
        !           138: #endif
        !           139: /* put out fake copy of left bracket line, to be redone later */
        !           140: if( ! headerdone )
        !           141:        {
        !           142: #if FAMILY == PCC
        !           143:        p2flush();
        !           144: #endif
        !           145:        headoffset = ftell(textfile);
        !           146:        prhead(textfile);
        !           147:        headerdone = YES;
        !           148:        p2triple(PCCF_FEXPR, (strlen(infname)+ALILONG-1)/ALILONG, 0);
        !           149:        p2str(infname);
        !           150: #if TARGET == PDP11
        !           151:        /* fake jump to start the optimizer */
        !           152:        if(class != CLBLOCK)
        !           153:                putgoto( fudgelabel = newlabel() );
        !           154: #endif
        !           155: 
        !           156: #if TARGET == VAX || TARGET == TAHOE
        !           157:        /* jump from top to bottom */
        !           158:        if(s!=CNULL && class!=CLBLOCK)
        !           159:                {
        !           160:                int proflab = newlabel();
        !           161:                p2pass("\t.align\t1");
        !           162:                p2ps("_%s:", s);
        !           163:                p2pi("\t.word\tLWM%d", procno);
        !           164:                prsave(proflab);
        !           165: #if TARGET == VAX
        !           166:                p2pi("\tjbr\tL%d",
        !           167: #else
        !           168:                putgoto(
        !           169: #endif
        !           170:                 fudgelabel = newlabel());
        !           171:                }
        !           172: #endif
        !           173:        }
        !           174: }
        !           175: 
        !           176: 
        !           177: 
        !           178: 
        !           179: 
        !           180: /* It is necessary to precede each procedure with a "left bracket"
        !           181:  * line that tells pass 2 how many register variables and how
        !           182:  * much automatic space is required for the function.  This compiler
        !           183:  * does not know how much automatic space is needed until the
        !           184:  * entire procedure has been processed.  Therefore, "puthead"
        !           185:  * is called at the begining to record the current location in textfile,
        !           186:  * then to put out a placeholder left bracket line.  This procedure
        !           187:  * repositions the file and rewrites that line, then puts the
        !           188:  * file pointer back to the end of the file.
        !           189:  */
        !           190: 
        !           191: putbracket()
        !           192: {
        !           193: long int hereoffset;
        !           194: 
        !           195: #if FAMILY == PCC
        !           196:        p2flush();
        !           197: #endif
        !           198: hereoffset = ftell(textfile);
        !           199: if(fseek(textfile, headoffset, 0) == -1)
        !           200:        fatal("fseek failed");
        !           201: prhead(textfile);
        !           202: if(fseek(textfile, hereoffset, 0) == -1)
        !           203:        fatal("fseek failed 2");
        !           204: }
        !           205: 
        !           206: 
        !           207: 
        !           208: 
        !           209: putrbrack(k)
        !           210: int k;
        !           211: {
        !           212: p2op(PCCF_FRBRAC, k);
        !           213: }
        !           214: 
        !           215: 
        !           216: 
        !           217: putnreg()
        !           218: {
        !           219: }
        !           220: 
        !           221: 
        !           222: 
        !           223: 
        !           224: 
        !           225: 
        !           226: puteof()
        !           227: {
        !           228: p2op(PCCF_FEOF, 0);
        !           229: p2flush();
        !           230: }
        !           231: 
        !           232: 
        !           233: 
        !           234: putstmt()
        !           235: {
        !           236: p2triple(PCCF_FEXPR, 0, lineno);
        !           237: }
        !           238: 
        !           239: 
        !           240: 
        !           241: 
        !           242: /* put out code for if( ! p) goto l  */
        !           243: putif(p,l)
        !           244: register expptr p;
        !           245: int l;
        !           246: {
        !           247: register int k;
        !           248: 
        !           249: if( ( k = (p = fixtype(p))->headblock.vtype) != TYLOGICAL)
        !           250:        {
        !           251:        if(k != TYERROR)
        !           252:                err("non-logical expression in IF statement");
        !           253:        frexpr(p);
        !           254:        }
        !           255: else
        !           256:        {
        !           257:        putex1(p);
        !           258:        p2icon( (long int) l , PCCT_INT);
        !           259:        p2op(PCC_CBRANCH, 0);
        !           260:        putstmt();
        !           261:        }
        !           262: }
        !           263: 
        !           264: 
        !           265: 
        !           266: 
        !           267: 
        !           268: /* put out code for  goto l   */
        !           269: putgoto(label)
        !           270: int label;
        !           271: {
        !           272: p2triple(PCC_GOTO, 1, label);
        !           273: putstmt();
        !           274: }
        !           275: 
        !           276: 
        !           277: /* branch to address constant or integer variable */
        !           278: putbranch(p)
        !           279: register Addrp p;
        !           280: {
        !           281:   putex1((expptr) p);
        !           282:   p2op(PCC_GOTO, PCCT_INT);
        !           283:   putstmt();
        !           284: }
        !           285: 
        !           286: 
        !           287: 
        !           288: /* put out label  l:     */
        !           289: putlabel(label)
        !           290: int label;
        !           291: {
        !           292: p2op(PCCF_FLABEL, label);
        !           293: }
        !           294: 
        !           295: 
        !           296: 
        !           297: 
        !           298: putexpr(p)
        !           299: expptr p;
        !           300: {
        !           301: putex1(p);
        !           302: putstmt();
        !           303: }
        !           304: 
        !           305: 
        !           306: 
        !           307: 
        !           308: putcmgo(index, nlab, labs)
        !           309: expptr index;
        !           310: int nlab;
        !           311: struct Labelblock *labs[];
        !           312: {
        !           313: int i, labarray, skiplabel;
        !           314: 
        !           315: if(! ISINT(index->headblock.vtype) )
        !           316:        {
        !           317:        execerr("computed goto index must be integer", CNULL);
        !           318:        return;
        !           319:        }
        !           320: 
        !           321: #if TARGET == VAX || TARGET == TAHOE
        !           322:        /* use special case instruction */
        !           323:        casegoto(index, nlab, labs);
        !           324: #else
        !           325:        labarray = newlabel();
        !           326:        preven(ALIADDR);
        !           327:        prlabel(asmfile, labarray);
        !           328:        prcona(asmfile, (ftnint) (skiplabel = newlabel()) );
        !           329:        for(i = 0 ; i < nlab ; ++i)
        !           330:                if( labs[i] )
        !           331:                        prcona(asmfile, (ftnint)(labs[i]->labelno) );
        !           332:        prcmgoto(index, nlab, skiplabel, labarray);
        !           333:        putlabel(skiplabel);
        !           334: #endif
        !           335: }
        !           336: 
        !           337: putx(p)
        !           338: expptr p;
        !           339: {
        !           340: char *memname();
        !           341: int opc;
        !           342: int ncomma;
        !           343: int type, k;
        !           344: 
        !           345: if (!p)
        !           346:        return;
        !           347: 
        !           348: switch(p->tag)
        !           349:        {
        !           350:        case TERROR:
        !           351:                free( (charptr) p );
        !           352:                break;
        !           353: 
        !           354:        case TCONST:
        !           355:                switch(type = p->constblock.vtype)
        !           356:                        {
        !           357:                        case TYLOGICAL:
        !           358:                                type = tyint;
        !           359:                        case TYLONG:
        !           360:                        case TYSHORT:
        !           361:                                p2icon(p->constblock.const.ci, types2[type]);
        !           362:                                free( (charptr) p );
        !           363:                                break;
        !           364: 
        !           365:                        case TYADDR:
        !           366:                                p2triple(PCC_ICON, 1, PCCT_INT|PCCTM_PTR);
        !           367:                                p2word(0L);
        !           368:                                p2name(memname(STGCONST,
        !           369:                                        (int) p->constblock.const.ci) );
        !           370:                                free( (charptr) p );
        !           371:                                break;
        !           372: 
        !           373:                        default:
        !           374:                                putx( putconst(p) );
        !           375:                                break;
        !           376:                        }
        !           377:                break;
        !           378: 
        !           379:        case TEXPR:
        !           380:                switch(opc = p->exprblock.opcode)
        !           381:                        {
        !           382:                        case OPCALL:
        !           383:                        case OPCCALL:
        !           384:                                if( ISCOMPLEX(p->exprblock.vtype) )
        !           385:                                        putcxop(p);
        !           386:                                else    putcall(p);
        !           387:                                break;
        !           388: 
        !           389:                        case OPMIN:
        !           390:                        case OPMAX:
        !           391:                                putmnmx(p);
        !           392:                                break;
        !           393: 
        !           394: 
        !           395:                        case OPASSIGN:
        !           396:                                if(ISCOMPLEX(p->exprblock.leftp->headblock.vtype)
        !           397:                                || ISCOMPLEX(p->exprblock.rightp->headblock.vtype) )
        !           398:                                        frexpr( putcxeq(p) );
        !           399:                                else if( ISCHAR(p) )
        !           400:                                        putcheq(p);
        !           401:                                else
        !           402:                                        goto putopp;
        !           403:                                break;
        !           404: 
        !           405:                        case OPEQ:
        !           406:                        case OPNE:
        !           407:                                if( ISCOMPLEX(p->exprblock.leftp->headblock.vtype) ||
        !           408:                                    ISCOMPLEX(p->exprblock.rightp->headblock.vtype) )
        !           409:                                        {
        !           410:                                        putcxcmp(p);
        !           411:                                        break;
        !           412:                                        }
        !           413:                        case OPLT:
        !           414:                        case OPLE:
        !           415:                        case OPGT:
        !           416:                        case OPGE:
        !           417:                                if(ISCHAR(p->exprblock.leftp))
        !           418:                                        {
        !           419:                                        putchcmp(p);
        !           420:                                        break;
        !           421:                                        }
        !           422:                                goto putopp;
        !           423: 
        !           424:                        case OPPOWER:
        !           425:                                putpower(p);
        !           426:                                break;
        !           427: 
        !           428:                        case OPSTAR:
        !           429: #if FAMILY == PCC
        !           430:                                /*   m * (2**k) -> m<<k   */
        !           431:                                if(INT(p->exprblock.leftp->headblock.vtype) &&
        !           432:                                   ISICON(p->exprblock.rightp) &&
        !           433:                                   ( (k = log2(p->exprblock.rightp->constblock.const.ci))>0) )
        !           434:                                        {
        !           435:                                        p->exprblock.opcode = OPLSHIFT;
        !           436:                                        frexpr(p->exprblock.rightp);
        !           437:                                        p->exprblock.rightp = ICON(k);
        !           438:                                        goto putopp;
        !           439:                                        }
        !           440: #endif
        !           441: 
        !           442:                        case OPMOD:
        !           443:                                goto putopp;
        !           444:                        case OPPLUS:
        !           445:                        case OPMINUS:
        !           446:                        case OPSLASH:
        !           447:                        case OPNEG:
        !           448:                                if( ISCOMPLEX(p->exprblock.vtype) )
        !           449:                                        putcxop(p);
        !           450:                                else    goto putopp;
        !           451:                                break;
        !           452: 
        !           453:                        case OPCONV:
        !           454:                                if( ISCOMPLEX(p->exprblock.vtype) )
        !           455:                                        putcxop(p);
        !           456:                                else if( ISCOMPLEX(p->exprblock.leftp->headblock.vtype) )
        !           457:                                        {
        !           458:                                        ncomma = 0;
        !           459:                                        putx( mkconv(p->exprblock.vtype,
        !           460:                                                realpart(putcx1(p->exprblock.leftp,
        !           461:                                                        &ncomma))));
        !           462:                                        putcomma(ncomma, p->exprblock.vtype, NO);
        !           463:                                        free( (charptr) p );
        !           464:                                        }
        !           465:                                else    goto putopp;
        !           466:                                break;
        !           467: 
        !           468:                        case OPNOT:
        !           469:                        case OPOR:
        !           470:                        case OPAND:
        !           471:                        case OPEQV:
        !           472:                        case OPNEQV:
        !           473:                        case OPADDR:
        !           474:                        case OPPLUSEQ:
        !           475:                        case OPSTAREQ:
        !           476:                        case OPCOMMA:
        !           477:                        case OPQUEST:
        !           478:                        case OPCOLON:
        !           479:                        case OPBITOR:
        !           480:                        case OPBITAND:
        !           481:                        case OPBITXOR:
        !           482:                        case OPBITNOT:
        !           483:                        case OPLSHIFT:
        !           484:                        case OPRSHIFT:
        !           485:                putopp:
        !           486:                                putop(p);
        !           487:                                break;
        !           488: 
        !           489:                        case OPPAREN:
        !           490:                                putx (p->exprblock.leftp);
        !           491:                                break;
        !           492:                        default:
        !           493:                                badop("putx", opc);
        !           494:                        }
        !           495:                break;
        !           496: 
        !           497:        case TADDR:
        !           498:                putaddr(p, YES);
        !           499:                break;
        !           500: 
        !           501:        case TTEMP:
        !           502:                /*
        !           503:                 * This type is sometimes passed to putx when errors occur
        !           504:                 *      upstream, I don't know why.
        !           505:                 */
        !           506:                frexpr(p);
        !           507:                break;
        !           508: 
        !           509:        default:
        !           510:                badtag("putx", p->tag);
        !           511:        }
        !           512: }
        !           513: 
        !           514: 
        !           515: 
        !           516: LOCAL putop(p)
        !           517: expptr p;
        !           518: {
        !           519: int k;
        !           520: expptr lp, tp;
        !           521: int pt, lt, tt;
        !           522: int comma;
        !           523: Addrp putch1();
        !           524: 
        !           525: switch(p->exprblock.opcode)    /* check for special cases and rewrite */
        !           526:        {
        !           527:        case OPCONV:
        !           528:                tt = pt = p->exprblock.vtype;
        !           529:                lp = p->exprblock.leftp;
        !           530:                lt = lp->headblock.vtype;
        !           531: #if TARGET == VAX
        !           532:                if (pt == TYREAL && lt == TYDREAL)
        !           533:                        {
        !           534:                        putx(lp);
        !           535:                        p2op(PCC_SCONV, PCCT_FLOAT);
        !           536:                        return;
        !           537:                        }
        !           538: #endif
        !           539:                while(p->tag==TEXPR && p->exprblock.opcode==OPCONV && (
        !           540: #if TARGET != TAHOE
        !           541:                       (ISREAL(pt)&&ISREAL(lt)) ||
        !           542: #endif
        !           543:                        (INT(pt)&&(ONEOF(lt,MSKINT|MSKADDR|MSKCHAR|M(TYSUBR)))) ))
        !           544:                        {
        !           545: #if SZINT < SZLONG
        !           546:                        if(lp->tag != TEXPR)
        !           547:                                {
        !           548:                                if(pt==TYINT && lt==TYLONG)
        !           549:                                        break;
        !           550:                                if(lt==TYINT && pt==TYLONG)
        !           551:                                        break;
        !           552:                                }
        !           553: #endif
        !           554: 
        !           555: #if TARGET == VAX
        !           556:                        if(pt==TYDREAL && lt==TYREAL)
        !           557:                                {
        !           558:                                if(lp->tag==TEXPR &&
        !           559:                                   lp->exprblock.opcode==OPCONV &&
        !           560:                                   lp->exprblock.leftp->headblock.vtype==TYDREAL)
        !           561:                                        {
        !           562:                                        putx(lp->exprblock.leftp);
        !           563:                                        p2op(PCC_SCONV, PCCT_FLOAT);
        !           564:                                        p2op(PCC_SCONV, PCCT_DOUBLE);
        !           565:                                        free( (charptr) p );
        !           566:                                        return;
        !           567:                                        }
        !           568:                                else break;
        !           569:                                }
        !           570: #endif
        !           571:                        if(lt==TYCHAR && lp->tag==TEXPR)
        !           572:                                {
        !           573:                                int ncomma = 0;
        !           574:                                p->exprblock.leftp = (expptr) putch1(lp, &ncomma);
        !           575:                                putop(p);
        !           576:                                putcomma(ncomma, pt, NO);
        !           577:                                free( (charptr) p );
        !           578:                                return;
        !           579:                                }
        !           580:                        free( (charptr) p );
        !           581:                        p = lp;
        !           582:                        pt = lt;
        !           583:                        if (p->tag == TEXPR)
        !           584:                                {
        !           585:                                lp = p->exprblock.leftp;
        !           586:                                lt = lp->headblock.vtype;
        !           587:                                }
        !           588:                        }
        !           589:                if(p->tag==TEXPR && p->exprblock.opcode==OPCONV)
        !           590:                        break;
        !           591:                putx(p);
        !           592:                if (types2[tt] != types2[pt] &&
        !           593:                    ! ( (ISREAL(tt)&&ISREAL(pt)) ||
        !           594:                        (INT(tt)&&(ONEOF(pt,MSKINT|MSKADDR|MSKCHAR|M(TYSUBR)))) ))
        !           595:                        p2op(PCC_SCONV,types2[tt]);
        !           596:                return;
        !           597: 
        !           598:        case OPADDR:
        !           599:                comma = NO;
        !           600:                lp = p->exprblock.leftp;
        !           601:                if(lp->tag != TADDR)
        !           602:                        {
        !           603:                        tp = (expptr) mkaltemp
        !           604:                                (lp->headblock.vtype,lp->headblock.vleng);
        !           605:                        putx( mkexpr(OPASSIGN, cpexpr(tp), lp) );
        !           606:                        lp = tp;
        !           607:                        comma = YES;
        !           608:                        }
        !           609:                putaddr(lp, NO);
        !           610:                if(comma)
        !           611:                        putcomma(1, TYINT, NO);
        !           612:                free( (charptr) p );
        !           613:                return;
        !           614: #if TARGET == VAX || TARGET == TAHOE
        !           615: /* take advantage of a glitch in the code generator that does not check
        !           616:    the type clash in an assignment or comparison of an integer zero and
        !           617:    a floating left operand, and generates optimal code for the correct
        !           618:    type.  (The PCC has no floating-constant node to encode this correctly.)
        !           619: */
        !           620:        case OPASSIGN:
        !           621:        case OPLT:
        !           622:        case OPLE:
        !           623:        case OPGT:
        !           624:        case OPGE:
        !           625:        case OPEQ:
        !           626:        case OPNE:
        !           627:                if(ISREAL(p->exprblock.leftp->headblock.vtype) &&
        !           628:                   ISREAL(p->exprblock.rightp->headblock.vtype) &&
        !           629:                   ISCONST(p->exprblock.rightp) &&
        !           630:                   p->exprblock.rightp->constblock.const.cd[0]==0)
        !           631:                        {
        !           632:                        p->exprblock.rightp->constblock.vtype = TYINT;
        !           633:                        p->exprblock.rightp->constblock.const.ci = 0;
        !           634:                        }
        !           635: #endif
        !           636:        }
        !           637: 
        !           638: if( (k = ops2[p->exprblock.opcode]) <= 0)
        !           639:        badop("putop", p->exprblock.opcode);
        !           640: putx(p->exprblock.leftp);
        !           641: if(p->exprblock.rightp)
        !           642:        putx(p->exprblock.rightp);
        !           643: p2op(k, types2[p->exprblock.vtype]);
        !           644: 
        !           645: if(p->exprblock.vleng)
        !           646:        frexpr(p->exprblock.vleng);
        !           647: free( (charptr) p );
        !           648: }
        !           649: 
        !           650: putforce(t, p)
        !           651: int t;
        !           652: expptr p;
        !           653: {
        !           654: p = mkconv(t, fixtype(p));
        !           655: putx(p);
        !           656: p2op(PCC_FORCE,
        !           657: #if TARGET == TAHOE
        !           658:        (t==TYLONG ? PCCT_LONG : (t==TYREAL ? PCCT_FLOAT : PCCT_DOUBLE)) );
        !           659: #else
        !           660:        (t==TYSHORT ? PCCT_SHORT : (t==TYLONG ? PCCT_LONG : PCCT_DOUBLE)) );
        !           661: #endif
        !           662: putstmt();
        !           663: }
        !           664: 
        !           665: 
        !           666: 
        !           667: LOCAL putpower(p)
        !           668: expptr p;
        !           669: {
        !           670: expptr base;
        !           671: Addrp t1, t2;
        !           672: ftnint k;
        !           673: int type;
        !           674: int ncomma;
        !           675: 
        !           676: if(!ISICON(p->exprblock.rightp) ||
        !           677:     (k = p->exprblock.rightp->constblock.const.ci)<2)
        !           678:        fatal("putpower: bad call");
        !           679: base = p->exprblock.leftp;
        !           680: type = base->headblock.vtype;
        !           681: 
        !           682: if ((k == 2) && base->tag == TADDR && ISCONST(base->addrblock.memoffset))
        !           683: {
        !           684:        putx( mkexpr(OPSTAR,cpexpr(base),cpexpr(base)));
        !           685:        
        !           686:        return;
        !           687: }
        !           688: t1 = mkaltemp(type, PNULL);
        !           689: t2 = NULL;
        !           690: ncomma = 1;
        !           691: putassign(cpexpr(t1), cpexpr(base) );
        !           692: 
        !           693: for( ; (k&1)==0 && k>2 ; k>>=1 )
        !           694:        {
        !           695:        ++ncomma;
        !           696:        putsteq(t1, t1);
        !           697:        }
        !           698: 
        !           699: if(k == 2)
        !           700:        putx( mkexpr(OPSTAR, cpexpr(t1), cpexpr(t1)) );
        !           701: else
        !           702:        {
        !           703:        t2 = mkaltemp(type, PNULL);
        !           704:        ++ncomma;
        !           705:        putassign(cpexpr(t2), cpexpr(t1));
        !           706:        
        !           707:        for(k>>=1 ; k>1 ; k>>=1)
        !           708:                {
        !           709:                ++ncomma;
        !           710:                putsteq(t1, t1);
        !           711:                if(k & 1)
        !           712:                        {
        !           713:                        ++ncomma;
        !           714:                        putsteq(t2, t1);
        !           715:                        }
        !           716:                }
        !           717:        putx( mkexpr(OPSTAR, cpexpr(t2),
        !           718:                mkexpr(OPSTAR, cpexpr(t1), cpexpr(t1)) ));
        !           719:        }
        !           720: putcomma(ncomma, type, NO);
        !           721: frexpr(t1);
        !           722: if(t2)
        !           723:        frexpr(t2);
        !           724: frexpr(p);
        !           725: }
        !           726: 
        !           727: 
        !           728: 
        !           729: 
        !           730: LOCAL Addrp intdouble(p, ncommap)
        !           731: Addrp p;
        !           732: int *ncommap;
        !           733: {
        !           734: register Addrp t;
        !           735: 
        !           736: t = mkaltemp(TYDREAL, PNULL);
        !           737: ++*ncommap;
        !           738: putassign(cpexpr(t), p);
        !           739: return(t);
        !           740: }
        !           741: 
        !           742: 
        !           743: 
        !           744: 
        !           745: 
        !           746: LOCAL Addrp putcxeq(p)
        !           747: register expptr p;
        !           748: {
        !           749: register Addrp lp, rp;
        !           750: int ncomma;
        !           751: 
        !           752: if(p->tag != TEXPR)
        !           753:        badtag("putcxeq", p->tag);
        !           754: 
        !           755: ncomma = 0;
        !           756: lp = putcx1(p->exprblock.leftp, &ncomma);
        !           757: rp = putcx1(p->exprblock.rightp, &ncomma);
        !           758: putassign(realpart(lp), realpart(rp));
        !           759: if( ISCOMPLEX(p->exprblock.vtype) )
        !           760:        {
        !           761:        ++ncomma;
        !           762:        putassign(imagpart(lp), imagpart(rp));
        !           763:        }
        !           764: putcomma(ncomma, TYREAL, NO);
        !           765: frexpr(rp);
        !           766: free( (charptr) p );
        !           767: return(lp);
        !           768: }
        !           769: 
        !           770: 
        !           771: 
        !           772: LOCAL putcxop(p)
        !           773: expptr p;
        !           774: {
        !           775: Addrp putcx1();
        !           776: int ncomma;
        !           777: 
        !           778: ncomma = 0;
        !           779: putaddr( putcx1(p, &ncomma), NO);
        !           780: putcomma(ncomma, TYINT, NO);
        !           781: }
        !           782: 
        !           783: 
        !           784: 
        !           785: LOCAL Addrp putcx1(p, ncommap)
        !           786: register expptr p;
        !           787: int *ncommap;
        !           788: {
        !           789: expptr q;
        !           790: Addrp lp, rp;
        !           791: register Addrp resp;
        !           792: int opcode;
        !           793: int ltype, rtype;
        !           794: expptr mkrealcon();
        !           795: 
        !           796: if(p == NULL)
        !           797:        return(NULL);
        !           798: 
        !           799: switch(p->tag)
        !           800:        {
        !           801:        case TCONST:
        !           802:                if( ISCOMPLEX(p->constblock.vtype) )
        !           803:                        p = (expptr) putconst(p);
        !           804:                return( (Addrp) p );
        !           805: 
        !           806:        case TADDR:
        !           807:                if( ! addressable(p) )
        !           808:                        {
        !           809:                        ++*ncommap;
        !           810:                        resp = mkaltemp(tyint, PNULL);
        !           811:                        putassign( cpexpr(resp), p->addrblock.memoffset );
        !           812:                        p->addrblock.memoffset = (expptr)resp;
        !           813:                        }
        !           814:                return( (Addrp) p );
        !           815: 
        !           816:        case TEXPR:
        !           817:                if( ISCOMPLEX(p->exprblock.vtype) )
        !           818:                        break;
        !           819:                ++*ncommap;
        !           820:                resp = mkaltemp(TYDREAL, NO);
        !           821:                putassign( cpexpr(resp), p);
        !           822:                return(resp);
        !           823: 
        !           824:        default:
        !           825:                badtag("putcx1", p->tag);
        !           826:        }
        !           827: 
        !           828: opcode = p->exprblock.opcode;
        !           829: if(opcode==OPCALL || opcode==OPCCALL)
        !           830:        {
        !           831:        ++*ncommap;
        !           832:        return( putcall(p) );
        !           833:        }
        !           834: else if(opcode == OPASSIGN)
        !           835:        {
        !           836:        ++*ncommap;
        !           837:        return( putcxeq(p) );
        !           838:        }
        !           839: resp = mkaltemp(p->exprblock.vtype, PNULL);
        !           840: if(lp = putcx1(p->exprblock.leftp, ncommap) )
        !           841:        ltype = lp->vtype;
        !           842: if(rp = putcx1(p->exprblock.rightp, ncommap) )
        !           843:        rtype = rp->vtype;
        !           844: 
        !           845: switch(opcode)
        !           846:        {
        !           847:        case OPPAREN:
        !           848:                frexpr (resp);
        !           849:                resp = lp;
        !           850:                lp = NULL;
        !           851:                break;
        !           852: 
        !           853:        case OPCOMMA:
        !           854:                frexpr(resp);
        !           855:                resp = rp;
        !           856:                rp = NULL;
        !           857:                break;
        !           858: 
        !           859:        case OPNEG:
        !           860:                putassign( realpart(resp), mkexpr(OPNEG, realpart(lp), ENULL) );
        !           861:                putassign( imagpart(resp), mkexpr(OPNEG, imagpart(lp), ENULL) );
        !           862:                *ncommap += 2;
        !           863:                break;
        !           864: 
        !           865:        case OPPLUS:
        !           866:        case OPMINUS:
        !           867:                putassign( realpart(resp),
        !           868:                        mkexpr(opcode, realpart(lp), realpart(rp) ));
        !           869:                if(rtype < TYCOMPLEX)
        !           870:                        putassign( imagpart(resp), imagpart(lp) );
        !           871:                else if(ltype < TYCOMPLEX)
        !           872:                        {
        !           873:                        if(opcode == OPPLUS)
        !           874:                                putassign( imagpart(resp), imagpart(rp) );
        !           875:                        else    putassign( imagpart(resp),
        !           876:                                        mkexpr(OPNEG, imagpart(rp), ENULL) );
        !           877:                        }
        !           878:                else
        !           879:                        putassign( imagpart(resp),
        !           880:                                mkexpr(opcode, imagpart(lp), imagpart(rp) ));
        !           881: 
        !           882:                *ncommap += 2;
        !           883:                break;
        !           884: 
        !           885:        case OPSTAR:
        !           886:                if(ltype < TYCOMPLEX)
        !           887:                        {
        !           888:                        if( ISINT(ltype) )
        !           889:                                lp = intdouble(lp, ncommap);
        !           890:                        putassign( realpart(resp),
        !           891:                                mkexpr(OPSTAR, cpexpr(lp), realpart(rp) ));
        !           892:                        putassign( imagpart(resp),
        !           893:                                mkexpr(OPSTAR, cpexpr(lp), imagpart(rp) ));
        !           894:                        }
        !           895:                else if(rtype < TYCOMPLEX)
        !           896:                        {
        !           897:                        if( ISINT(rtype) )
        !           898:                                rp = intdouble(rp, ncommap);
        !           899:                        putassign( realpart(resp),
        !           900:                                mkexpr(OPSTAR, cpexpr(rp), realpart(lp) ));
        !           901:                        putassign( imagpart(resp),
        !           902:                                mkexpr(OPSTAR, cpexpr(rp), imagpart(lp) ));
        !           903:                        }
        !           904:                else    {
        !           905:                        putassign( realpart(resp), mkexpr(OPMINUS,
        !           906:                                mkexpr(OPSTAR, realpart(lp), realpart(rp)),
        !           907:                                mkexpr(OPSTAR, imagpart(lp), imagpart(rp)) ));
        !           908:                        putassign( imagpart(resp), mkexpr(OPPLUS,
        !           909:                                mkexpr(OPSTAR, realpart(lp), imagpart(rp)),
        !           910:                                mkexpr(OPSTAR, imagpart(lp), realpart(rp)) ));
        !           911:                        }
        !           912:                *ncommap += 2;
        !           913:                break;
        !           914: 
        !           915:        case OPSLASH:
        !           916:                /* fixexpr has already replaced all divisions
        !           917:                 * by a complex by a function call
        !           918:                 */
        !           919:                if( ISINT(rtype) )
        !           920:                        rp = intdouble(rp, ncommap);
        !           921:                putassign( realpart(resp),
        !           922:                        mkexpr(OPSLASH, realpart(lp), cpexpr(rp)) );
        !           923:                putassign( imagpart(resp),
        !           924:                        mkexpr(OPSLASH, imagpart(lp), cpexpr(rp)) );
        !           925:                *ncommap += 2;
        !           926:                break;
        !           927: 
        !           928:        case OPCONV:
        !           929:                putassign( realpart(resp), realpart(lp) );
        !           930:                if( ISCOMPLEX(lp->vtype) )
        !           931:                        q = imagpart(lp);
        !           932:                else if(rp != NULL)
        !           933:                        q = (expptr) realpart(rp);
        !           934:                else
        !           935:                        q = mkrealcon(TYDREAL, 0.0);
        !           936:                putassign( imagpart(resp), q);
        !           937:                *ncommap += 2;
        !           938:                break;
        !           939: 
        !           940:        default:
        !           941:                badop("putcx1", opcode);
        !           942:        }
        !           943: 
        !           944: frexpr(lp);
        !           945: frexpr(rp);
        !           946: free( (charptr) p );
        !           947: return(resp);
        !           948: }
        !           949: 
        !           950: 
        !           951: 
        !           952: 
        !           953: LOCAL putcxcmp(p)
        !           954: register expptr p;
        !           955: {
        !           956: int opcode;
        !           957: int ncomma;
        !           958: register Addrp lp, rp;
        !           959: expptr q;
        !           960: 
        !           961: if(p->tag != TEXPR)
        !           962:        badtag("putcxcmp", p->tag);
        !           963: 
        !           964: ncomma = 0;
        !           965: opcode = p->exprblock.opcode;
        !           966: lp = putcx1(p->exprblock.leftp, &ncomma);
        !           967: rp = putcx1(p->exprblock.rightp, &ncomma);
        !           968: 
        !           969: q = mkexpr( opcode==OPEQ ? OPAND : OPOR ,
        !           970:        mkexpr(opcode, realpart(lp), realpart(rp)),
        !           971:        mkexpr(opcode, imagpart(lp), imagpart(rp)) );
        !           972: putx( fixexpr(q) );
        !           973: putcomma(ncomma, TYINT, NO);
        !           974: 
        !           975: free( (charptr) lp);
        !           976: free( (charptr) rp);
        !           977: free( (charptr) p );
        !           978: }
        !           979: 
        !           980: LOCAL Addrp putch1(p, ncommap)
        !           981: register expptr p;
        !           982: int * ncommap;
        !           983: {
        !           984: register Addrp t;
        !           985: 
        !           986: switch(p->tag)
        !           987:        {
        !           988:        case TCONST:
        !           989:                return( putconst(p) );
        !           990: 
        !           991:        case TADDR:
        !           992:                return( (Addrp) p );
        !           993: 
        !           994:        case TEXPR:
        !           995:                ++*ncommap;
        !           996: 
        !           997:                switch(p->exprblock.opcode)
        !           998:                        {
        !           999:                        expptr q;
        !          1000: 
        !          1001:                        case OPCALL:
        !          1002:                        case OPCCALL:
        !          1003:                                t = putcall(p);
        !          1004:                                break;
        !          1005: 
        !          1006:                        case OPPAREN:
        !          1007:                                --*ncommap;
        !          1008:                                t = putch1(p->exprblock.leftp, ncommap);
        !          1009:                                break;
        !          1010: 
        !          1011:                        case OPCONCAT:
        !          1012:                                t = mkaltemp(TYCHAR, ICON(lencat(p)) );
        !          1013:                                q = (expptr) cpexpr(p->headblock.vleng);
        !          1014:                                putcat( cpexpr(t), p );
        !          1015:                                /* put the correct length on the block */
        !          1016:                                frexpr(t->vleng);
        !          1017:                                t->vleng = q;
        !          1018: 
        !          1019:                                break;
        !          1020: 
        !          1021:                        case OPCONV:
        !          1022:                                if(!ISICON(p->exprblock.vleng)
        !          1023:                                   || p->exprblock.vleng->constblock.const.ci!=1
        !          1024:                                   || ! INT(p->exprblock.leftp->headblock.vtype) )
        !          1025:                                        fatal("putch1: bad character conversion");
        !          1026:                                t = mkaltemp(TYCHAR, ICON(1) );
        !          1027:                                putop( mkexpr(OPASSIGN, cpexpr(t), p) );
        !          1028:                                break;
        !          1029:                        default:
        !          1030:                                badop("putch1", p->exprblock.opcode);
        !          1031:                        }
        !          1032:                return(t);
        !          1033: 
        !          1034:        default:
        !          1035:                badtag("putch1", p->tag);
        !          1036:        }
        !          1037: /* NOTREACHED */
        !          1038: }
        !          1039: 
        !          1040: 
        !          1041: 
        !          1042: 
        !          1043: LOCAL putchop(p)
        !          1044: expptr p;
        !          1045: {
        !          1046: int ncomma;
        !          1047: 
        !          1048: ncomma = 0;
        !          1049: putaddr( putch1(p, &ncomma) , NO );
        !          1050: putcomma(ncomma, TYCHAR, YES);
        !          1051: }
        !          1052: 
        !          1053: 
        !          1054: 
        !          1055: 
        !          1056: LOCAL putcheq(p)
        !          1057: register expptr p;
        !          1058: {
        !          1059: int ncomma;
        !          1060: expptr lp, rp;
        !          1061: 
        !          1062: if(p->tag != TEXPR)
        !          1063:        badtag("putcheq", p->tag);
        !          1064: 
        !          1065: ncomma = 0;
        !          1066: lp = p->exprblock.leftp;
        !          1067: rp = p->exprblock.rightp;
        !          1068: if( rp->tag==TEXPR && rp->exprblock.opcode==OPCONCAT )
        !          1069:        putcat(lp, rp);
        !          1070: else if( ISONE(lp->headblock.vleng) && ISONE(rp->headblock.vleng) )
        !          1071:        {
        !          1072:        putaddr( putch1(lp, &ncomma) , YES );
        !          1073:        putaddr( putch1(rp, &ncomma) , YES );
        !          1074:        putcomma(ncomma, TYINT, NO);
        !          1075:        p2op(PCC_ASSIGN, PCCT_CHAR);
        !          1076:        }
        !          1077: else
        !          1078:        {
        !          1079:        putx( call2(TYINT, "s_copy", lp, rp) );
        !          1080:        putcomma(ncomma, TYINT, NO);
        !          1081:        }
        !          1082: 
        !          1083: frexpr(p->exprblock.vleng);
        !          1084: free( (charptr) p );
        !          1085: }
        !          1086: 
        !          1087: 
        !          1088: 
        !          1089: 
        !          1090: LOCAL putchcmp(p)
        !          1091: register expptr p;
        !          1092: {
        !          1093: int ncomma;
        !          1094: expptr lp, rp;
        !          1095: 
        !          1096: if(p->tag != TEXPR)
        !          1097:        badtag("putchcmp", p->tag);
        !          1098: 
        !          1099: ncomma = 0;
        !          1100: lp = p->exprblock.leftp;
        !          1101: rp = p->exprblock.rightp;
        !          1102: 
        !          1103: if(ISONE(lp->headblock.vleng) && ISONE(rp->headblock.vleng) )
        !          1104:        {
        !          1105:        putaddr( putch1(lp, &ncomma) , YES );
        !          1106:        putcomma(ncomma, TYINT, NO);
        !          1107:        ncomma = 0;
        !          1108:        putaddr( putch1(rp, &ncomma) , YES );
        !          1109:        putcomma(ncomma, TYINT, NO);
        !          1110:        p2op(ops2[p->exprblock.opcode], PCCT_CHAR);
        !          1111:        free( (charptr) p );
        !          1112:        }
        !          1113: else
        !          1114:        {
        !          1115:        p->exprblock.leftp = call2(TYINT,"s_cmp", lp, rp);
        !          1116:        p->exprblock.rightp = ICON(0);
        !          1117:        putop(p);
        !          1118:        }
        !          1119: }
        !          1120: 
        !          1121: 
        !          1122: 
        !          1123: 
        !          1124: 
        !          1125: LOCAL putcat(lhs, rhs)
        !          1126: register Addrp lhs;
        !          1127: register expptr rhs;
        !          1128: {
        !          1129: int n, ncomma;
        !          1130: Addrp lp, cp;
        !          1131: 
        !          1132: ncomma = 0;
        !          1133: n = ncat(rhs);
        !          1134: lp = mkaltmpn(n, TYLENG, PNULL);
        !          1135: cp = mkaltmpn(n, TYADDR, PNULL);
        !          1136: 
        !          1137: n = 0;
        !          1138: putct1(rhs, lp, cp, &n, &ncomma);
        !          1139: 
        !          1140: putx( call4(TYSUBR, "s_cat", lhs, cp, lp, mkconv(TYLONG, ICON(n)) ) );
        !          1141: putcomma(ncomma, TYINT, NO);
        !          1142: }
        !          1143: 
        !          1144: 
        !          1145: 
        !          1146: 
        !          1147: 
        !          1148: LOCAL putct1(q, lp, cp, ip, ncommap)
        !          1149: register expptr q;
        !          1150: register Addrp lp, cp;
        !          1151: int *ip, *ncommap;
        !          1152: {
        !          1153: int i;
        !          1154: Addrp lp1, cp1;
        !          1155: 
        !          1156: if(q->tag==TEXPR && q->exprblock.opcode==OPCONCAT)
        !          1157:        {
        !          1158:        putct1(q->exprblock.leftp, lp, cp, ip, ncommap);
        !          1159:        putct1(q->exprblock.rightp, lp, cp , ip, ncommap);
        !          1160:        frexpr(q->exprblock.vleng);
        !          1161:        free( (charptr) q );
        !          1162:        }
        !          1163: else
        !          1164:        {
        !          1165:        i = (*ip)++;
        !          1166:        lp1 = (Addrp) cpexpr(lp);
        !          1167:        lp1->memoffset = mkexpr(OPPLUS,lp1->memoffset, ICON(i*SZLENG));
        !          1168:        cp1 = (Addrp) cpexpr(cp);
        !          1169:        cp1->memoffset = mkexpr(OPPLUS, cp1->memoffset, ICON(i*SZADDR));
        !          1170:        putassign( lp1, cpexpr(q->headblock.vleng) );
        !          1171:        putassign( cp1, addrof(putch1(q,ncommap)) );
        !          1172:        *ncommap += 2;
        !          1173:        }
        !          1174: }
        !          1175: 
        !          1176: LOCAL putaddr(p, indir)
        !          1177: register Addrp p;
        !          1178: int indir;
        !          1179: {
        !          1180: int type, type2, funct;
        !          1181: ftnint offset, simoffset();
        !          1182: expptr offp, shorten();
        !          1183: 
        !          1184: if( p->tag==TERROR || (p->memoffset!=NULL && ISERROR(p->memoffset)) )
        !          1185:        {
        !          1186:        frexpr(p);
        !          1187:        return;
        !          1188:        }
        !          1189: if (p->tag != TADDR) badtag ("putaddr",p->tag);
        !          1190: 
        !          1191: type = p->vtype;
        !          1192: type2 = types2[type];
        !          1193: funct = (p->vclass==CLPROC ? PCCTM_FTN<<2 : 0);
        !          1194: 
        !          1195: offp = (p->memoffset ? (expptr) cpexpr(p->memoffset) : (expptr)NULL );
        !          1196: 
        !          1197: 
        !          1198: #if (FUDGEOFFSET != 1)
        !          1199: if(offp)
        !          1200:        offp = mkexpr(OPSTAR, ICON(FUDGEOFFSET), offp);
        !          1201: #endif
        !          1202: 
        !          1203: offset = simoffset( &offp );
        !          1204: #if SZINT < SZLONG
        !          1205:        if(offp)
        !          1206:                if(shortsubs)
        !          1207:                        offp = shorten(offp);
        !          1208:                else
        !          1209:                        offp = mkconv(TYINT, offp);
        !          1210: #else
        !          1211:        if(offp)
        !          1212:                offp = mkconv(TYINT, offp);
        !          1213: #endif
        !          1214: 
        !          1215: if (p->vclass == CLVAR
        !          1216:     && (p->vstg == STGBSS || p->vstg == STGEQUIV)
        !          1217:     && SMALLVAR(p->varsize)
        !          1218:     && offset >= -32768 && offset <= 32767)
        !          1219:   {
        !          1220:     anylocals = YES;
        !          1221:     if (indir && !offp)
        !          1222:       p2ldisp(offset, memname(p->vstg, p->memno), type2);
        !          1223:     else
        !          1224:       {
        !          1225:        p2reg(LVARREG, type2 | PCCTM_PTR);
        !          1226:        p2triple(PCC_ICON, 1, PCCT_INT);
        !          1227:        p2word(offset);
        !          1228:        p2ndisp(memname(p->vstg, p->memno));
        !          1229:        p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1230:        if (offp)
        !          1231:          {
        !          1232:            putx(offp);
        !          1233:            p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1234:          }
        !          1235:        if (indir)
        !          1236:          p2op(PCC_DEREF, type2);
        !          1237:       }
        !          1238:     frexpr((tagptr) p);
        !          1239:     return;
        !          1240:   }
        !          1241: 
        !          1242: switch(p->vstg)
        !          1243:        {
        !          1244:        case STGAUTO:
        !          1245:                if(indir && !offp)
        !          1246:                        {
        !          1247:                        p2oreg(offset, AUTOREG, type2);
        !          1248:                        break;
        !          1249:                        }
        !          1250: 
        !          1251:                if(!indir && !offp && !offset)
        !          1252:                        {
        !          1253:                        p2reg(AUTOREG, type2 | PCCTM_PTR);
        !          1254:                        break;
        !          1255:                        }
        !          1256: 
        !          1257:                p2reg(AUTOREG, type2 | PCCTM_PTR);
        !          1258:                if(offp)
        !          1259:                        {
        !          1260:                        putx(offp);
        !          1261:                        if(offset)
        !          1262:                                p2icon(offset, PCCT_INT);
        !          1263:                        }
        !          1264:                else
        !          1265:                        p2icon(offset, PCCT_INT);
        !          1266:                if(offp && offset)
        !          1267:                        p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1268:                p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1269:                if(indir)
        !          1270:                        p2op(PCC_DEREF, type2);
        !          1271:                break;
        !          1272: 
        !          1273:        case STGARG:
        !          1274:                p2oreg(
        !          1275: #ifdef ARGOFFSET
        !          1276:                        ARGOFFSET +
        !          1277: #endif
        !          1278:                        (ftnint) (FUDGEOFFSET*p->memno),
        !          1279:                        ARGREG,   type2 | PCCTM_PTR | funct );
        !          1280: 
        !          1281:        based:
        !          1282:                if(offset)
        !          1283:                        {
        !          1284:                        p2icon(offset, PCCT_INT);
        !          1285:                        p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1286:                        }
        !          1287:                if(offp)
        !          1288:                        {
        !          1289:                        putx(offp);
        !          1290:                        p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1291:                        }
        !          1292:                if(indir)
        !          1293:                        p2op(PCC_DEREF, type2);
        !          1294:                break;
        !          1295: 
        !          1296:        case STGLENG:
        !          1297:                if(indir)
        !          1298:                        {
        !          1299:                        p2oreg(
        !          1300: #ifdef ARGOFFSET
        !          1301:                                ARGOFFSET +
        !          1302: #endif
        !          1303:                                (ftnint) (FUDGEOFFSET*p->memno),
        !          1304:                                ARGREG,   type2 );
        !          1305:                        }
        !          1306:                else    {
        !          1307:                        p2reg(ARGREG, type2 | PCCTM_PTR );
        !          1308:                        p2icon(
        !          1309: #ifdef ARGOFFSET
        !          1310:                                ARGOFFSET +
        !          1311: #endif
        !          1312:                                (ftnint) (FUDGEOFFSET*p->memno), PCCT_INT);
        !          1313:                        p2op(PCC_PLUS, type2 | PCCTM_PTR );
        !          1314:                        }
        !          1315:                break;
        !          1316: 
        !          1317: 
        !          1318:        case STGBSS:
        !          1319:        case STGINIT:
        !          1320:        case STGEXT:
        !          1321:        case STGINTR:
        !          1322:        case STGCOMMON:
        !          1323:        case STGEQUIV:
        !          1324:        case STGCONST:
        !          1325:                if(offp)
        !          1326:                        {
        !          1327:                        putx(offp);
        !          1328:                        putmem(p, PCC_ICON, offset);
        !          1329:                        p2op(PCC_PLUS, type2 | PCCTM_PTR);
        !          1330:                        if(indir)
        !          1331:                                p2op(PCC_DEREF, type2);
        !          1332:                        }
        !          1333:                else
        !          1334:                        putmem(p, (indir ? PCC_NAME : PCC_ICON), offset);
        !          1335: 
        !          1336:                break;
        !          1337: 
        !          1338:        case STGREG:
        !          1339:                if(indir)
        !          1340:                        p2reg(p->memno, type2);
        !          1341:                else
        !          1342:                        fatal("attempt to take address of a register");
        !          1343:                break;
        !          1344: 
        !          1345:        case STGPREG:
        !          1346:                if(indir && !offp)
        !          1347:                        p2oreg(offset, p->memno, type2);
        !          1348:                else
        !          1349:                        {
        !          1350:                        p2reg(p->memno, type2 | PCCTM_PTR);
        !          1351:                        goto based;
        !          1352:                        }
        !          1353:                break;
        !          1354: 
        !          1355:        default:
        !          1356:                badstg("putaddr", p->vstg);
        !          1357:        }
        !          1358: frexpr(p);
        !          1359: }
        !          1360: 
        !          1361: 
        !          1362: 
        !          1363: 
        !          1364: LOCAL putmem(p, class, offset)
        !          1365: expptr p;
        !          1366: int class;
        !          1367: ftnint offset;
        !          1368: {
        !          1369: int type2;
        !          1370: int funct;
        !          1371: char *name,  *memname();
        !          1372: 
        !          1373: funct = (p->headblock.vclass==CLPROC ? PCCTM_FTN<<2 : 0);
        !          1374: type2 = types2[p->headblock.vtype];
        !          1375: if(p->headblock.vclass == CLPROC)
        !          1376:        type2 |= (PCCTM_FTN<<2);
        !          1377: name = memname(p->addrblock.vstg, p->addrblock.memno);
        !          1378: if(class == PCC_ICON)
        !          1379:        {
        !          1380:        p2triple(PCC_ICON, name[0]!='\0', type2|PCCTM_PTR);
        !          1381:        p2word(offset);
        !          1382:        if(name[0])
        !          1383:                p2name(name);
        !          1384:        }
        !          1385: else
        !          1386:        {
        !          1387:        p2triple(PCC_NAME, offset!=0, type2);
        !          1388:        if(offset != 0)
        !          1389:                p2word(offset);
        !          1390:        p2name(name);
        !          1391:        }
        !          1392: }
        !          1393: 
        !          1394: 
        !          1395: 
        !          1396: LOCAL Addrp putcall(p)
        !          1397: register Exprp p;
        !          1398: {
        !          1399: chainp arglist, charsp, cp;
        !          1400: int n, first;
        !          1401: Addrp t;
        !          1402: register expptr q;
        !          1403: Addrp fval, mkargtemp();
        !          1404: int type, type2, ctype, qtype, indir;
        !          1405: 
        !          1406: type2 = types2[type = p->vtype];
        !          1407: charsp = NULL;
        !          1408: indir =  (p->opcode == OPCCALL);
        !          1409: n = 0;
        !          1410: first = YES;
        !          1411: 
        !          1412: if(p->rightp)
        !          1413:        {
        !          1414:        arglist = p->rightp->listblock.listp;
        !          1415:        free( (charptr) (p->rightp) );
        !          1416:        }
        !          1417: else
        !          1418:        arglist = NULL;
        !          1419: 
        !          1420: for(cp = arglist ; cp ; cp = cp->nextp)
        !          1421:        {
        !          1422:        q = (expptr) cp->datap;
        !          1423:        if(indir)
        !          1424:                ++n;
        !          1425:        else    {
        !          1426:                q = (expptr) (cp->datap);
        !          1427:                if( ISCONST(q) )
        !          1428:                        {
        !          1429:                        q = (expptr) putconst(q);
        !          1430:                        cp->datap = (tagptr) q;
        !          1431:                        }
        !          1432:                if( ISCHAR(q) && q->headblock.vclass!=CLPROC )
        !          1433:                        {
        !          1434:                        charsp = hookup(charsp,
        !          1435:                                        mkchain(cpexpr(q->headblock.vleng),
        !          1436:                                                CHNULL));
        !          1437:                        n += 2;
        !          1438:                        }
        !          1439:                else
        !          1440:                        n += 1;
        !          1441:                }
        !          1442:        }
        !          1443: 
        !          1444: if(type == TYCHAR)
        !          1445:        {
        !          1446:        if( ISICON(p->vleng) )
        !          1447:                {
        !          1448:                fval = mkargtemp(TYCHAR, p->vleng);
        !          1449:                n += 2;
        !          1450:                }
        !          1451:        else    {
        !          1452:                err("adjustable character function");
        !          1453:                return;
        !          1454:                }
        !          1455:        }
        !          1456: else if( ISCOMPLEX(type) )
        !          1457:        {
        !          1458:        fval = mkargtemp(type, PNULL);
        !          1459:        n += 1;
        !          1460:        }
        !          1461: else
        !          1462:        fval = NULL;
        !          1463: 
        !          1464: ctype = (fval ? PCCT_INT : type2);
        !          1465: putaddr(p->leftp, NO);
        !          1466: 
        !          1467: if(fval)
        !          1468:        {
        !          1469:        first = NO;
        !          1470:        putaddr( cpexpr(fval), NO);
        !          1471:        if(type==TYCHAR)
        !          1472:                {
        !          1473:                putx( mkconv(TYLENG,p->vleng) );
        !          1474:                p2op(PCC_CM, type2);
        !          1475:                }
        !          1476:        }
        !          1477: 
        !          1478: for(cp = arglist ; cp ; cp = cp->nextp)
        !          1479:        {
        !          1480:        q = (expptr) (cp->datap);
        !          1481:        if(q->tag==TADDR && (indir || q->addrblock.vstg!=STGREG) )
        !          1482:                putaddr(q, indir && q->addrblock.vtype!=TYCHAR);
        !          1483:        else if( ISCOMPLEX(q->headblock.vtype) )
        !          1484:                putcxop(q);
        !          1485:        else if (ISCHAR(q) )
        !          1486:                putchop(q);
        !          1487:        else if( ! ISERROR(q) )
        !          1488:                {
        !          1489:                if(indir)
        !          1490:                        putx(q);
        !          1491:                else    {
        !          1492:                        t = mkargtemp(qtype = q->headblock.vtype,
        !          1493:                                q->headblock.vleng);
        !          1494:                        putassign( cpexpr(t), q );
        !          1495:                        putaddr(t, NO);
        !          1496:                        putcomma(1, qtype, YES);
        !          1497:                        }
        !          1498:                }
        !          1499:        if(first)
        !          1500:                first = NO;
        !          1501:        else
        !          1502:                p2op(PCC_CM, type2);
        !          1503:        }
        !          1504: 
        !          1505: if(arglist)
        !          1506:        frchain(&arglist);
        !          1507: for(cp = charsp ; cp ; cp = cp->nextp)
        !          1508:        {
        !          1509:        putx( mkconv(TYLENG,cp->datap) );
        !          1510:        p2op(PCC_CM, type2);
        !          1511:        }
        !          1512: frchain(&charsp);
        !          1513: #if TARGET == TAHOE
        !          1514: if(indir && ctype==PCCT_FLOAT) /* function opcodes */
        !          1515:        p2op(PCC_FORTCALL, ctype);
        !          1516: else
        !          1517: #endif
        !          1518: p2op(n>0 ? PCC_CALL : PCC_UCALL , ctype);
        !          1519: free( (charptr) p );
        !          1520: return(fval);
        !          1521: }
        !          1522: 
        !          1523: 
        !          1524: 
        !          1525: LOCAL putmnmx(p)
        !          1526: register expptr p;
        !          1527: {
        !          1528: int op, type;
        !          1529: int ncomma;
        !          1530: expptr qp;
        !          1531: chainp p0, p1;
        !          1532: Addrp sp, tp;
        !          1533: 
        !          1534: if(p->tag != TEXPR)
        !          1535:        badtag("putmnmx", p->tag);
        !          1536: 
        !          1537: type = p->exprblock.vtype;
        !          1538: op = (p->exprblock.opcode==OPMIN ? OPLT : OPGT );
        !          1539: p0 = p->exprblock.leftp->listblock.listp;
        !          1540: free( (charptr) (p->exprblock.leftp) );
        !          1541: free( (charptr) p );
        !          1542: 
        !          1543: sp = mkaltemp(type, PNULL);
        !          1544: tp = mkaltemp(type, PNULL);
        !          1545: qp = mkexpr(OPCOLON, cpexpr(tp), cpexpr(sp));
        !          1546: qp = mkexpr(OPQUEST, mkexpr(op, cpexpr(tp),cpexpr(sp)), qp);
        !          1547: qp = fixexpr(qp);
        !          1548: 
        !          1549: ncomma = 1;
        !          1550: putassign( cpexpr(sp), p0->datap );
        !          1551: 
        !          1552: for(p1 = p0->nextp ; p1 ; p1 = p1->nextp)
        !          1553:        {
        !          1554:        ++ncomma;
        !          1555:        putassign( cpexpr(tp), p1->datap );
        !          1556:        if(p1->nextp)
        !          1557:                {
        !          1558:                ++ncomma;
        !          1559:                putassign( cpexpr(sp), cpexpr(qp) );
        !          1560:                }
        !          1561:        else
        !          1562:                putx(qp);
        !          1563:        }
        !          1564: 
        !          1565: putcomma(ncomma, type, NO);
        !          1566: frexpr(sp);
        !          1567: frexpr(tp);
        !          1568: frchain( &p0 );
        !          1569: }
        !          1570: 
        !          1571: 
        !          1572: 
        !          1573: 
        !          1574: LOCAL putcomma(n, type, indir)
        !          1575: int n, type, indir;
        !          1576: {
        !          1577: type = types2[type];
        !          1578: if(indir)
        !          1579:        type |= PCCTM_PTR;
        !          1580: while(--n >= 0)
        !          1581:        p2op(PCC_COMOP, type);
        !          1582: }
        !          1583: 
        !          1584: 
        !          1585: 
        !          1586: 
        !          1587: ftnint simoffset(p0)
        !          1588: expptr *p0;
        !          1589: {
        !          1590: ftnint offset, prod;
        !          1591: register expptr p, lp, rp;
        !          1592: 
        !          1593: offset = 0;
        !          1594: p = *p0;
        !          1595: if(p == NULL)
        !          1596:        return(0);
        !          1597: 
        !          1598: if( ! ISINT(p->headblock.vtype) )
        !          1599:        return(0);
        !          1600: 
        !          1601: if(p->tag==TEXPR && p->exprblock.opcode==OPSTAR)
        !          1602:        {
        !          1603:        lp = p->exprblock.leftp;
        !          1604:        rp = p->exprblock.rightp;
        !          1605:        if(ISICON(rp) && lp->tag==TEXPR &&
        !          1606:           lp->exprblock.opcode==OPPLUS && ISICON(lp->exprblock.rightp))
        !          1607:                {
        !          1608:                p->exprblock.opcode = OPPLUS;
        !          1609:                lp->exprblock.opcode = OPSTAR;
        !          1610:                prod = rp->constblock.const.ci *
        !          1611:                        lp->exprblock.rightp->constblock.const.ci;
        !          1612:                lp->exprblock.rightp->constblock.const.ci = rp->constblock.const.ci;
        !          1613:                rp->constblock.const.ci = prod;
        !          1614:                }
        !          1615:        }
        !          1616: 
        !          1617: if(p->tag==TEXPR && p->exprblock.opcode==OPPLUS &&
        !          1618:     ISICON(p->exprblock.rightp))
        !          1619:        {
        !          1620:        rp = p->exprblock.rightp;
        !          1621:        lp = p->exprblock.leftp;
        !          1622:        offset += rp->constblock.const.ci;
        !          1623:        frexpr(rp);
        !          1624:        free( (charptr) p );
        !          1625:        *p0 = lp;
        !          1626:        }
        !          1627: 
        !          1628: if( ISCONST(p) )
        !          1629:        {
        !          1630:        offset += p->constblock.const.ci;
        !          1631:        frexpr(p);
        !          1632:        *p0 = NULL;
        !          1633:        }
        !          1634: 
        !          1635: return(offset);
        !          1636: }
        !          1637: 
        !          1638: 
        !          1639: 
        !          1640: 
        !          1641: 
        !          1642: p2op(op, type)
        !          1643: int op, type;
        !          1644: {
        !          1645: p2triple(op, 0, type);
        !          1646: }
        !          1647: 
        !          1648: p2icon(offset, type)
        !          1649: ftnint offset;
        !          1650: int type;
        !          1651: {
        !          1652: p2triple(PCC_ICON, 0, type);
        !          1653: p2word(offset);
        !          1654: }
        !          1655: 
        !          1656: 
        !          1657: 
        !          1658: 
        !          1659: p2oreg(offset, reg, type)
        !          1660: ftnint offset;
        !          1661: int reg, type;
        !          1662: {
        !          1663: p2triple(PCC_OREG, reg, type);
        !          1664: p2word(offset);
        !          1665: p2name("");
        !          1666: }
        !          1667: 
        !          1668: 
        !          1669: 
        !          1670: 
        !          1671: p2reg(reg, type)
        !          1672: int reg, type;
        !          1673: {
        !          1674: p2triple(PCC_REG, reg, type);
        !          1675: }
        !          1676: 
        !          1677: 
        !          1678: 
        !          1679: p2pi(s, i)
        !          1680: char *s;
        !          1681: int i;
        !          1682: {
        !          1683: char buff[100];
        !          1684: sprintf(buff, s, i);
        !          1685: p2pass(buff);
        !          1686: }
        !          1687: 
        !          1688: 
        !          1689: 
        !          1690: p2pij(s, i, j)
        !          1691: char *s;
        !          1692: int i, j;
        !          1693: {
        !          1694: char buff[100];
        !          1695: sprintf(buff, s, i, j);
        !          1696: p2pass(buff);
        !          1697: }
        !          1698: 
        !          1699: 
        !          1700: 
        !          1701: 
        !          1702: p2ps(s, t)
        !          1703: char *s, *t;
        !          1704: {
        !          1705: char buff[100];
        !          1706: sprintf(buff, s, t);
        !          1707: p2pass(buff);
        !          1708: }
        !          1709: 
        !          1710: 
        !          1711: 
        !          1712: 
        !          1713: p2pass(s)
        !          1714: char *s;
        !          1715: {
        !          1716: p2triple(PCCF_FTEXT, (strlen(s) + ALILONG-1)/ALILONG, 0);
        !          1717: p2str(s);
        !          1718: }
        !          1719: 
        !          1720: 
        !          1721: 
        !          1722: 
        !          1723: p2str(s)
        !          1724: register char *s;
        !          1725: {
        !          1726: union { long int word; char str[SZLONG]; } u;
        !          1727: register int i;
        !          1728: 
        !          1729: i = 0;
        !          1730: u.word = 0;
        !          1731: while(*s)
        !          1732:        {
        !          1733:        u.str[i++] = *s++;
        !          1734:        if(i == SZLONG)
        !          1735:                {
        !          1736:                p2word(u.word);
        !          1737:                u.word = 0;
        !          1738:                i = 0;
        !          1739:                }
        !          1740:        }
        !          1741: if(i > 0)
        !          1742:        p2word(u.word);
        !          1743: }
        !          1744: 
        !          1745: 
        !          1746: 
        !          1747: 
        !          1748: p2triple(op, var, type)
        !          1749: int op, var, type;
        !          1750: {
        !          1751: register long word;
        !          1752: word = PCCM_TRIPLE(op, var, type);
        !          1753: p2word(word);
        !          1754: }
        !          1755: 
        !          1756: 
        !          1757: 
        !          1758: 
        !          1759: 
        !          1760: p2name(s)
        !          1761: register char *s;
        !          1762: {
        !          1763: register int i;
        !          1764: 
        !          1765: #ifdef UCBPASS2
        !          1766:        /* arbitrary length names, terminated by a null,
        !          1767:           padded to a full word */
        !          1768: 
        !          1769: #      define WL   sizeof(long int)
        !          1770:        union { long int word; char str[WL]; } w;
        !          1771:        
        !          1772:        w.word = 0;
        !          1773:        i = 0;
        !          1774:        while(w.str[i++] = *s++)
        !          1775:                if(i == WL)
        !          1776:                        {
        !          1777:                        p2word(w.word);
        !          1778:                        w.word = 0;
        !          1779:                        i = 0;
        !          1780:                        }
        !          1781:        if(i > 0)
        !          1782:                p2word(w.word);
        !          1783: #else
        !          1784:        /* standard intermediate, names are 8 characters long */
        !          1785: 
        !          1786:        union  { long int word[2];  char str[8]; } u;
        !          1787:        
        !          1788:        u.word[0] = u.word[1] = 0;
        !          1789:        for(i = 0 ; i<8 && *s ; ++i)
        !          1790:                u.str[i] = *s++;
        !          1791:        p2word(u.word[0]);
        !          1792:        p2word(u.word[1]);
        !          1793: 
        !          1794: #endif
        !          1795: 
        !          1796: }
        !          1797: 
        !          1798: 
        !          1799: 
        !          1800: 
        !          1801: p2word(w)
        !          1802: long int w;
        !          1803: {
        !          1804: *p2bufp++ = w;
        !          1805: if(p2bufp >= p2bufend)
        !          1806:        p2flush();
        !          1807: }
        !          1808: 
        !          1809: 
        !          1810: 
        !          1811: p2flush()
        !          1812: {
        !          1813: if(p2bufp > p2buff)
        !          1814:        write(fileno(textfile), p2buff, (p2bufp-p2buff)*sizeof(long int));
        !          1815: p2bufp = p2buff;
        !          1816: }
        !          1817: 
        !          1818: 
        !          1819: 
        !          1820: LOCAL
        !          1821: p2ldisp(offset, vname, type)
        !          1822: ftnint offset;
        !          1823: char *vname;
        !          1824: int type;
        !          1825: {
        !          1826:   char buff[100];
        !          1827: 
        !          1828:   sprintf(buff, "%s-v.%d", vname, bsslabel);
        !          1829:   p2triple(PCC_OREG, LVARREG, type);
        !          1830:   p2word(offset);
        !          1831:   p2name(buff);
        !          1832: }
        !          1833: 
        !          1834: 
        !          1835: 
        !          1836: p2ndisp(vname)
        !          1837: char *vname;
        !          1838: {
        !          1839:   char buff[100];
        !          1840: 
        !          1841:   sprintf(buff, "%s-v.%d", vname, bsslabel);
        !          1842:   p2name(buff);
        !          1843: }

unix.superglobalmegacorp.com

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