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

unix.superglobalmegacorp.com

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