Annotation of 42BSD/ucb/pascal/src/pclval.c, revision 1.1

1.1     ! root        1: /* Copyright (c) 1979 Regents of the University of California */
        !             2: 
        !             3: static char sccsid[] = "@(#)pclval.c 1.4 6/1/81";
        !             4: 
        !             5: #include "whoami.h"
        !             6: #include "0.h"
        !             7: #include "tree.h"
        !             8: #include "opcode.h"
        !             9: #include "objfmt.h"
        !            10: #ifdef PC
        !            11:        /*
        !            12:         *      and the rest of the file
        !            13:         */
        !            14: #   include    "pc.h"
        !            15: #   include    "pcops.h"
        !            16: 
        !            17: extern int flagwas;
        !            18: /*
        !            19:  * pclvalue computes the address
        !            20:  * of a qualified name and
        !            21:  * leaves it on the stack.
        !            22:  * for pc, it can be asked for either an lvalue or an rvalue.
        !            23:  * the semantics are the same, only the code is different.
        !            24:  * for putting out calls to check for nil and fnil,
        !            25:  * we have to traverse the list of qualifications twice:
        !            26:  * once to put out the calls and once to put out the address to be checked.
        !            27:  */
        !            28: struct nl *
        !            29: pclvalue( r , modflag , required )
        !            30:        int     *r;
        !            31:        int     modflag;
        !            32:        int     required;
        !            33: {
        !            34:        register struct nl      *p;
        !            35:        register                *c, *co;
        !            36:        int                     f, o;
        !            37:        int                     tr[2], trp[3];
        !            38:        struct nl               *firstp;
        !            39:        struct nl               *lastp;
        !            40:        char                    *firstsymbol;
        !            41:        char                    firstextra_flags;
        !            42:        int                     firstbn;
        !            43: 
        !            44:        if ( r == NIL ) {
        !            45:                return NIL;
        !            46:        }
        !            47:        if ( nowexp( r ) ) {
        !            48:                return NIL;
        !            49:        }
        !            50:        if ( r[0] != T_VAR ) {
        !            51:                error("Variable required");     /* Pass mesgs down from pt of call ? */
        !            52:                return NIL;
        !            53:        }
        !            54:        firstp = p = lookup( r[2] );
        !            55:        if ( p == NIL ) {
        !            56:                return NIL;
        !            57:        }
        !            58:        firstsymbol = p -> symbol;
        !            59:        firstbn = bn;
        !            60:        firstextra_flags = p -> extra_flags;
        !            61:        c = r[3];
        !            62:        if ( ( modflag & NOUSE ) && ! lptr( c ) ) {
        !            63:                p -> nl_flags = flagwas;
        !            64:        }
        !            65:        if ( modflag & MOD ) {
        !            66:                p -> nl_flags |= NMOD;
        !            67:        }
        !            68:        /*
        !            69:         * Only possibilities for p -> class here
        !            70:         * are the named classes, i.e. CONST, TYPE
        !            71:         * VAR, PROC, FUNC, REF, or a WITHPTR.
        !            72:         */
        !            73:        if ( p -> class == WITHPTR ) {
        !            74:                /*
        !            75:                 * Construct the tree implied by
        !            76:                 * the with statement
        !            77:                 */
        !            78:            trp[0] = T_LISTPP;
        !            79:            trp[1] = tr;
        !            80:            trp[2] = r[3];
        !            81:            tr[0] = T_FIELD;
        !            82:            tr[1] = r[2];
        !            83:            c = trp;
        !            84:        }
        !            85:            /*
        !            86:             *  this not only puts out the names of functions to call
        !            87:             *  but also does all the semantic checking of the qualifications.
        !            88:             */
        !            89:        if ( ! nilfnil( p , c , modflag , firstp , r[2] ) ) {
        !            90:            return NIL;
        !            91:        }
        !            92:        switch (p -> class) {
        !            93:                case WITHPTR:
        !            94:                case REF:
        !            95:                        /*
        !            96:                         * Obtain the indirect word
        !            97:                         * of the WITHPTR or REF
        !            98:                         * as the base of our lvalue
        !            99:                         */
        !           100:                        putRV( firstsymbol , firstbn , p -> value[ 0 ] ,
        !           101:                                firstextra_flags , p2type( p ) );
        !           102:                        firstsymbol = 0;
        !           103:                        f = 0;          /* have an lv on stack */
        !           104:                        o = 0;
        !           105:                        break;
        !           106:                case VAR:
        !           107:                        f = 1;          /* no lv on stack yet */
        !           108:                        o = p -> value[0];
        !           109:                        break;
        !           110:                default:
        !           111:                        error("%s %s found where variable required", classes[p -> class], p -> symbol);
        !           112:                        return (NIL);
        !           113:        }
        !           114:        /*
        !           115:         * Loop and handle each
        !           116:         * qualification on the name
        !           117:         */
        !           118:        if ( c == NIL &&
        !           119:            ( modflag & ASGN ) &&
        !           120:            ( p -> value[ NL_FORV ] & FORVAR ) ) {
        !           121:                error("Can't modify the for variable %s in the range of the loop", p -> symbol);
        !           122:                return (NIL);
        !           123:        }
        !           124:        for ( ; c != NIL ; c = c[2] ) {
        !           125:                co = c[1];
        !           126:                if ( co == NIL ) {
        !           127:                        return NIL;
        !           128:                }
        !           129:                lastp = p;
        !           130:                p = p -> type;
        !           131:                if ( p == NIL ) {
        !           132:                        return NIL;
        !           133:                }
        !           134:                switch ( co[0] ) {
        !           135:                        case T_PTR:
        !           136:                                /*
        !           137:                                 * Pointer qualification.
        !           138:                                 */
        !           139:                                if ( f ) {
        !           140:                                        putLV( firstsymbol , firstbn , o ,
        !           141:                                            firstextra_flags , p2type( p ) );
        !           142:                                        firstsymbol = 0;
        !           143:                                } else {
        !           144:                                        if (o) {
        !           145:                                            putleaf( P2ICON , o , 0 , P2INT
        !           146:                                                    , 0 );
        !           147:                                            putop( P2PLUS , P2PTR | P2CHAR );
        !           148:                                        }
        !           149:                                }
        !           150:                                    /*
        !           151:                                     * Pointer cannot be
        !           152:                                     * nil and file cannot
        !           153:                                     * be at end-of-file.
        !           154:                                     * the appropriate function name is 
        !           155:                                     * already out there from nilfnil.
        !           156:                                     */
        !           157:                                if ( p -> class == PTR ) {
        !           158:                                        /*
        !           159:                                         * this is the indirection from
        !           160:                                         * the address of the pointer 
        !           161:                                         * to the pointer itself.
        !           162:                                         * kirk sez:
        !           163:                                         * fnil doesn't want this.
        !           164:                                         * and does it itself for files
        !           165:                                         * since only it knows where the
        !           166:                                         * actual window is.
        !           167:                                         * but i have to do this for
        !           168:                                         * regular pointers.
        !           169:                                         */
        !           170:                                    putop( P2UNARY P2MUL , p2type( p ) );
        !           171:                                    if ( opt( 't' ) ) {
        !           172:                                        putop( P2CALL , P2INT );
        !           173:                                    }
        !           174:                                } else {
        !           175:                                    putop( P2CALL , P2INT );
        !           176:                                }
        !           177:                                f = o = 0;
        !           178:                                continue;
        !           179:                        case T_ARGL:
        !           180:                        case T_ARY:
        !           181:                                if ( f ) {
        !           182:                                        putLV( firstsymbol , firstbn , o ,
        !           183:                                            firstextra_flags , p2type( p ) );
        !           184:                                        firstsymbol = 0;
        !           185:                                } else {
        !           186:                                        if (o) {
        !           187:                                            putleaf( P2ICON , o , 0 , P2INT
        !           188:                                                    , 0 );
        !           189:                                            putop( P2PLUS , P2INT );
        !           190:                                        }
        !           191:                                }
        !           192:                                arycod( p , co[1] );
        !           193:                                f = o = 0;
        !           194:                                continue;
        !           195:                        case T_FIELD:
        !           196:                                /*
        !           197:                                 * Field names are just
        !           198:                                 * an offset with some 
        !           199:                                 * semantic checking.
        !           200:                                 */
        !           201:                                p = reclook(p, co[1]);
        !           202:                                o += p -> value[0];
        !           203:                                continue;
        !           204:                        default:
        !           205:                                panic("lval2");
        !           206:                }
        !           207:        }
        !           208:        if (f) {
        !           209:                if ( required == LREQ ) {
        !           210:                    putLV( firstsymbol , firstbn , o ,
        !           211:                            firstextra_flags , p2type( p -> type ) );
        !           212:                } else {
        !           213:                    putRV( firstsymbol , firstbn , o ,
        !           214:                            firstextra_flags , p2type( p -> type ) );
        !           215:                }
        !           216:        } else {
        !           217:                if (o) {
        !           218:                    putleaf( P2ICON , o , 0 , P2INT , 0 );
        !           219:                    putop( P2PLUS , P2INT );
        !           220:                }
        !           221:                if ( required == RREQ ) {
        !           222:                    putop( P2UNARY P2MUL , p2type( p -> type ) );
        !           223:                }
        !           224:        }
        !           225:        return ( p -> type );
        !           226: }
        !           227: 
        !           228:     /*
        !           229:      * this recursively follows done a list of qualifications
        !           230:      * and puts out the beginnings of calls to fnil for files
        !           231:      * or nil for pointers (if checking is on) on the way back.
        !           232:      * this returns true or false.
        !           233:      */
        !           234: nilfnil( p , c , modflag , firstp , r2 )
        !           235:     struct nl  *p;
        !           236:     int                *c;
        !           237:     int                modflag;
        !           238:     struct nl  *firstp;
        !           239:     char       *r2;            /* no, not r2-d2 */
        !           240:     {
        !           241:        int             *co;
        !           242:        struct nl       *lastp;
        !           243:        int             t;
        !           244: 
        !           245:        if ( c == NIL ) {
        !           246:            return TRUE;
        !           247:        }
        !           248:        co = (int *) ( c[1] );
        !           249:        if ( co == NIL ) {
        !           250:                return FALSE;
        !           251:        }
        !           252:        lastp = p;
        !           253:        p = p -> type;
        !           254:        if ( p == NIL ) {
        !           255:                return FALSE;
        !           256:        }
        !           257:        switch ( co[0] ) {
        !           258:            case T_PTR:
        !           259:                    /*
        !           260:                     * Pointer qualification.
        !           261:                     */
        !           262:                    lastp -> nl_flags |= NUSED;
        !           263:                    if ( p -> class != PTR && p -> class != FILET) {
        !           264:                            error("^ allowed only on files and pointers, not on %ss", nameof(p));
        !           265:                            goto bad;
        !           266:                    }
        !           267:                    break;
        !           268:            case T_ARGL:
        !           269:                    if ( p -> class != ARRAY ) {
        !           270:                            if ( lastp == firstp ) {
        !           271:                                    error("%s is a %s, not a function", r2, classes[firstp -> class]);
        !           272:                            } else {
        !           273:                                    error("Illegal function qualificiation");
        !           274:                            }
        !           275:                            return FALSE;
        !           276:                    }
        !           277:                    recovered();
        !           278:                    error("Pascal uses [] for subscripting, not ()");
        !           279:                    /* and fall through */
        !           280:            case T_ARY:
        !           281:                    if ( p -> class != ARRAY ) {
        !           282:                            error("Subscripting allowed only on arrays, not on %ss", nameof(p));
        !           283:                            goto bad;
        !           284:                    }
        !           285:                    codeoff();
        !           286:                    t = arycod( p , co[1] );
        !           287:                    codeon();
        !           288:                    switch ( t ) {
        !           289:                            case 0:
        !           290:                                    return FALSE;
        !           291:                            case -1:
        !           292:                                    goto bad;
        !           293:                    }
        !           294:                    break;
        !           295:            case T_FIELD:
        !           296:                    /*
        !           297:                     * Field names are just
        !           298:                     * an offset with some 
        !           299:                     * semantic checking.
        !           300:                     */
        !           301:                    if ( p -> class != RECORD ) {
        !           302:                            error(". allowed only on records, not on %ss", nameof(p));
        !           303:                            goto bad;
        !           304:                    }
        !           305:                    if ( co[1] == NIL ) {
        !           306:                            return FALSE;
        !           307:                    }
        !           308:                    p = reclook( p , co[1] );
        !           309:                    if ( p == NIL ) {
        !           310:                            error("%s is not a field in this record", co[1]);
        !           311:                            goto bad;
        !           312:                    }
        !           313:                    if ( modflag & MOD ) {
        !           314:                            p -> nl_flags |= NMOD;
        !           315:                    }
        !           316:                    if ( ( modflag & NOUSE ) == 0 || lptr( c[2] ) ) {
        !           317:                            p -> nl_flags |= NUSED;
        !           318:                    }
        !           319:                    break;
        !           320:            default:
        !           321:                    panic("nilfnil");
        !           322:        }
        !           323:            /*
        !           324:             *  recursive call, check the rest of the qualifications.
        !           325:             */
        !           326:        if ( ! nilfnil( p , c[2] , modflag , firstp , r2 ) ) {
        !           327:            return FALSE;
        !           328:        }
        !           329:            /*
        !           330:             *  the point of all this.
        !           331:             */
        !           332:        if ( co[0] == T_PTR ) {
        !           333:            if ( p -> class == PTR ) {
        !           334:                    if ( opt( 't' ) ) {
        !           335:                        putleaf( P2ICON , 0 , 0
        !           336:                            , ADDTYPE( P2FTN | P2INT , P2PTR )
        !           337:                            , "_NIL" );
        !           338:                    }
        !           339:            } else {
        !           340:                    putleaf( P2ICON , 0 , 0
        !           341:                        , ADDTYPE( P2FTN | P2INT , P2PTR )
        !           342:                        , "_FNIL" );
        !           343:            }
        !           344:        }
        !           345:        return TRUE;
        !           346: bad:
        !           347:        cerror("Error occurred on qualification of %s", r2);
        !           348:        return FALSE;
        !           349:     }
        !           350: #endif PC

unix.superglobalmegacorp.com

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