Annotation of 43BSDTahoe/ucb/pascal/src/func.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[] = "@(#)func.c     5.1 (Berkeley) 6/5/85";
        !             9: #endif not lint
        !            10: 
        !            11: 
        !            12: #include "whoami.h"
        !            13: #ifdef OBJ
        !            14:     /*
        !            15:      * the rest of the file
        !            16:      */
        !            17: #include "0.h"
        !            18: #include "tree.h"
        !            19: #include "opcode.h"
        !            20: #include "tree_ty.h"
        !            21: 
        !            22: /*
        !            23:  * Funccod generates code for
        !            24:  * built in function calls and calls
        !            25:  * call to generate calls to user
        !            26:  * defined functions and procedures.
        !            27:  */
        !            28: struct nl
        !            29: *funccod(r)
        !            30:        struct tnode *r;
        !            31: {
        !            32:        struct nl *p;
        !            33:        register struct nl *p1;
        !            34:        struct nl *tempnlp;
        !            35:        register struct tnode *al;
        !            36:        register op;
        !            37:        int argc;
        !            38:        struct tnode *argv, tr, tr2;
        !            39: 
        !            40:        /*
        !            41:         * Verify that the given name
        !            42:         * is defined and the name of
        !            43:         * a function.
        !            44:         */
        !            45:        p = lookup(r->pcall_node.proc_id);
        !            46:        if (p == NLNIL) {
        !            47:                rvlist(r->pcall_node.arg);
        !            48:                return (NLNIL);
        !            49:        }
        !            50:        if (p->class != FUNC && p->class != FFUNC) {
        !            51:                error("%s is not a function", p->symbol);
        !            52:                rvlist(r->pcall_node.arg);
        !            53:                return (NLNIL);
        !            54:        }
        !            55:        argv = r->pcall_node.arg;
        !            56:        /*
        !            57:         * Call handles user defined
        !            58:         * procedures and functions
        !            59:         */
        !            60:        if (bn != 0)
        !            61:                return (call(p, argv, FUNC, bn));
        !            62:        /*
        !            63:         * Count the arguments
        !            64:         */
        !            65:        argc = 0;
        !            66:        for (al = argv; al != TR_NIL; al = al->list_node.next)
        !            67:                argc++;
        !            68:        /*
        !            69:         * Built-in functions have
        !            70:         * their interpreter opcode
        !            71:         * associated with them.
        !            72:         */
        !            73:        op = p->value[0] &~ NSTAND;
        !            74:        if (opt('s') && (p->value[0] & NSTAND)) {
        !            75:                standard();
        !            76:                error("%s is a nonstandard function", p->symbol);
        !            77:        }
        !            78:        switch (op) {
        !            79:                /*
        !            80:                 * Parameterless functions
        !            81:                 */
        !            82:                case O_CLCK:
        !            83:                case O_SCLCK:
        !            84:                case O_WCLCK:
        !            85:                case O_ARGC:
        !            86:                        if (argc != 0) {
        !            87:                                error("%s takes no arguments", p->symbol);
        !            88:                                rvlist(argv);
        !            89:                                return (NLNIL);
        !            90:                        }
        !            91:                        (void) put(1, op);
        !            92:                        return (nl+T4INT);
        !            93:                case O_EOF:
        !            94:                case O_EOLN:
        !            95:                        if (argc == 0) {
        !            96:                                argv = (&tr);
        !            97:                                tr.list_node.list = (&tr2);
        !            98:                                tr2.tag = T_VAR;
        !            99:                                tr2.var_node.cptr = input->symbol;
        !           100:                                tr2.var_node.line_no = NIL;
        !           101:                                tr2.var_node.qual = TR_NIL;
        !           102:                                argc = 1;
        !           103:                        } else if (argc != 1) {
        !           104:                                error("%s takes either zero or one argument", p->symbol);
        !           105:                                rvlist(argv);
        !           106:                                return (NLNIL);
        !           107:                        }
        !           108:                }
        !           109:        /*
        !           110:         * All other functions take
        !           111:         * exactly one argument.
        !           112:         */
        !           113:        if (argc != 1) {
        !           114:                error("%s takes exactly one argument", p->symbol);
        !           115:                rvlist(argv);
        !           116:                return (NLNIL);
        !           117:        }
        !           118:        /*
        !           119:         * Evaluate the argmument
        !           120:         */
        !           121:        if (op == O_EOF || op == O_EOLN)
        !           122:                p1 = stklval(argv->list_node.list, NIL );
        !           123:        else
        !           124:                p1 = stkrval(argv->list_node.list, NLNIL , (long) RREQ );
        !           125:        if (p1 == NLNIL)
        !           126:                return (NLNIL);
        !           127:        switch (op) {
        !           128:                case 0:
        !           129:                        error("%s is an unimplemented 6000-3.4 extension", p->symbol);
        !           130:                default:
        !           131:                        panic("func1");
        !           132:                case O_EXP:
        !           133:                case O_SIN:
        !           134:                case O_COS:
        !           135:                case O_ATAN:
        !           136:                case O_LN:
        !           137:                case O_SQRT:
        !           138:                case O_RANDOM:
        !           139:                case O_EXPO:
        !           140:                case O_UNDEF:
        !           141:                        if (isa(p1, "i"))
        !           142:                                convert( nl+T4INT , nl+TDOUBLE);
        !           143:                        else if (isnta(p1, "d")) {
        !           144:                                error("%s's argument must be integer or real, not %s", p->symbol, nameof(p1));
        !           145:                                return (NLNIL);
        !           146:                        }
        !           147:                        (void) put(1, op);
        !           148:                        if (op == O_UNDEF)
        !           149:                                return (nl+TBOOL);
        !           150:                        else if (op == O_EXPO)
        !           151:                                return (nl+T4INT);
        !           152:                        else
        !           153:                                return (nl+TDOUBLE);
        !           154:                case O_SEED:
        !           155:                        if (isnta(p1, "i")) {
        !           156:                                error("seed's argument must be an integer, not %s", nameof(p1));
        !           157:                                return (NLNIL);
        !           158:                        }
        !           159:                        (void) put(1, op);
        !           160:                        return (nl+T4INT);
        !           161:                case O_ROUND:
        !           162:                case O_TRUNC:
        !           163:                        if (isnta(p1, "d"))  {
        !           164:                                error("%s's argument must be a real, not %s", p->symbol, nameof(p1));
        !           165:                                return (NLNIL);
        !           166:                        }
        !           167:                        (void) put(1, op);
        !           168:                        return (nl+T4INT);
        !           169:                case O_ABS2:
        !           170:                case O_SQR2:
        !           171:                        if (isa(p1, "d")) {
        !           172:                                (void) put(1, op + O_ABS8-O_ABS2);
        !           173:                                return (nl+TDOUBLE);
        !           174:                        }
        !           175:                        if (isa(p1, "i")) {
        !           176:                                (void) put(1, op + (width(p1) >> 2));
        !           177:                                return (nl+T4INT);
        !           178:                        }
        !           179:                        error("%s's argument must be an integer or real, not %s", p->symbol, nameof(p1));
        !           180:                        return (NLNIL);
        !           181:                case O_ORD2:
        !           182:                        if (isa(p1, "bcis")) {
        !           183:                                return (nl+T4INT);
        !           184:                        }
        !           185:                        if (classify(p1) == TPTR) {
        !           186:                            if (!opt('s')) {
        !           187:                                return (nl+T4INT);
        !           188:                            }
        !           189:                            standard();
        !           190:                        }
        !           191:                        error("ord's argument must be of scalar type, not %s",
        !           192:                                nameof(p1));
        !           193:                        return (NLNIL);
        !           194:                case O_SUCC2:
        !           195:                case O_PRED2:
        !           196:                        if (isa(p1, "d")) {
        !           197:                                error("%s is forbidden for reals", p->symbol);
        !           198:                                return (NLNIL);
        !           199:                        }
        !           200:                        if ( isnta( p1 , "bcsi" ) ) {
        !           201:                                error("%s's argument must be of scalar type, not %s", p->symbol, nameof(p1));
        !           202:                                return NIL;
        !           203:                        }
        !           204:                        tempnlp = p1 -> class == TYPE ? p1 -> type : p1;
        !           205:                        if (isa(p1, "i")) {
        !           206:                                if (width(p1) <= 2) {
        !           207:                                        op += O_PRED24 - O_PRED2;
        !           208:                                        (void) put(3, op, (int)tempnlp->range[0],
        !           209:                                                (int)tempnlp->range[1]);
        !           210:                                } else {
        !           211:                                        op++;
        !           212:                                        (void) put(3, op, tempnlp->range[0],
        !           213:                                                tempnlp->range[1]);
        !           214:                                }
        !           215:                                return nl + T4INT;
        !           216:                        } else {
        !           217:                                (void) put(3, op, (int)tempnlp->range[0],
        !           218:                                        (int)tempnlp->range[1]);
        !           219:                                return p1;
        !           220:                        }
        !           221:                case O_ODD2:
        !           222:                        if (isnta(p1, "i")) {
        !           223:                                error("odd's argument must be an integer, not %s", nameof(p1));
        !           224:                                return (NLNIL);
        !           225:                        }
        !           226:                        (void) put(1, op + (width(p1) >> 2));
        !           227:                        return (nl+TBOOL);
        !           228:                case O_CHR2:
        !           229:                        if (isnta(p1, "i")) {
        !           230:                                error("chr's argument must be an integer, not %s", nameof(p1));
        !           231:                                return (NLNIL);
        !           232:                        }
        !           233:                        (void) put(1, op + (width(p1) >> 2));
        !           234:                        return (nl+TCHAR);
        !           235:                case O_CARD:
        !           236:                        if (isnta(p1, "t")) {
        !           237:                            error("Argument to card must be a set, not %s", nameof(p1));
        !           238:                            return (NLNIL);
        !           239:                        }
        !           240:                        (void) put(2, O_CARD, width(p1));
        !           241:                        return (nl+T2INT);
        !           242:                case O_EOLN:
        !           243:                        if (!text(p1)) {
        !           244:                                error("Argument to eoln must be a text file, not %s", nameof(p1));
        !           245:                                return (NLNIL);
        !           246:                        }
        !           247:                        (void) put(1, op);
        !           248:                        return (nl+TBOOL);
        !           249:                case O_EOF:
        !           250:                        if (p1->class != FILET) {
        !           251:                                error("Argument to eof must be file, not %s", nameof(p1));
        !           252:                                return (NLNIL);
        !           253:                        }
        !           254:                        (void) put(1, op);
        !           255:                        return (nl+TBOOL);
        !           256:        }
        !           257: }
        !           258: #endif OBJ

unix.superglobalmegacorp.com

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