This is prog.c in view mode; [Download] [Up]
/* Copyright (C) 1994 M. Hagiya, W. Schelter, T. Yuasa This file is part of GNU Common Lisp, herein referred to as GCL GCL is free software; you can redistribute it and/or modify it under the terms of the GNU LIBRARY GENERAL PUBLIC LICENSE as published by the Free Software Foundation; either version 2, or (at your option) any later version. GCL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with GCL; see the file COPYING. If not, write to the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ /* prog.c */ #include "include.h" /* use of VS in tagbody: old_top -> | id | | lex0 | | lex1 | | lex2 | tinf_base -> | tag1 | where 'bodyi' is the part of tag-body | body1 | that follows the tag 'tagi'. | : | : i.e. | : | tag-body | tagn | = (...tag1..........tagn.............) | bodyn | | |<- bodyn ->| new_top -> | | | | VS |<-------- body1 -------->| */ Ftagbody(body) object body; { object endp_temp; object *old_top = vs_top; object *new_top; object *tinf; object *tinf_base; object *env = lex_env; object id = alloc_frame_id(); object bodysv = body; object label; enum type item_type; vs_push(id); lex_copy(); tinf_base = vs_top; while (!endp(body)) { label = MMcar(body); item_type = type_of(label); if (item_type == t_symbol || item_type == t_fixnum || item_type == t_bignum) { lex_tag_bind(label, id); vs_push(label); vs_push(MMcdr(body)); } body = MMcdr(body); } new_top = vs_top; frs_push(FRS_CATCH, id); body = bodysv; if (nlj_active) { label = cdr(nlj_tag); nlj_active = FALSE; for(tinf = tinf_base; tinf < new_top && !eql(tinf[0],label); tinf += 2) ; if (tinf >= new_top) FEerror("Someone tried to RETURN-FROM a TAGBODY.",0); body = tinf[1]; } while (body != Cnil) { vs_top = new_top; item_type = type_of(MMcar(body)); if (item_type != t_symbol && item_type != t_fixnum && item_type != t_bignum) eval(MMcar(body)); body = MMcdr(body); } frs_pop(); lex_env = env; vs_base = old_top; vs_top = old_top+1; vs_base[0] = Cnil; } Fprog(arg) object arg; { object endp_temp; object *oldlex = lex_env; struct bind_temp *start; object body; bds_ptr old_bds_top = bds_top; if (endp(arg)) FEtoo_few_argumentsF(arg); make_nil_block(); if (nlj_active) { nlj_active = FALSE; goto END; } start = (struct bind_temp *)vs_top; let_var_list(arg->c.c_car); body = let_bind(arg->c.c_cdr, start, (struct bind_temp *)vs_top); vs_top = (object *)start; vs_push(body); Ftagbody(body); END: bds_unwind(old_bds_top); frs_pop(); lex_env = oldlex; } FprogA(arg) object arg; { object endp_temp; object *oldlex = lex_env; object *top; struct bind_temp *start; object body; bds_ptr old_bds_top = bds_top; if (endp(arg)) FEtoo_few_argumentsF(arg); make_nil_block(); if (nlj_active) { nlj_active = FALSE; goto END; } start = (struct bind_temp *) vs_top; let_var_list(arg->c.c_car); body = letA_bind(arg->c.c_cdr, start, (struct bind_temp *)vs_top); vs_top = (object *)start; vs_push(body); Ftagbody(body); END: bds_unwind(old_bds_top); frs_pop(); lex_env = oldlex; } Fgo(args) object args; { object endp_temp; object lex_tag; frame_ptr fr; if (endp(args)) FEtoo_few_argumentsF(args); if (!endp(MMcdr(args))) FEtoo_many_argumentsF(args); lex_tag = lex_tag_sch(MMcar(args)); if (MMnull(lex_tag)) FEerror("~S is an undefined tag.", 1, MMcar(args)); fr = frs_sch(MMcaddr(lex_tag)); if (fr == NULL) FEerror("The tag ~S is missing.", 1, MMcar(args)); vs_push(MMcons(MMcaddr(lex_tag), MMcar(lex_tag))); vs_base = vs_top; unwind(fr,vs_top[-1]); /* never reached */ } Fprogv(args) object args; { object endp_temp; object *top; object symbols; object values; bds_ptr old_bds_top; object var; if (endp(args) || endp(MMcdr(args))) FEtoo_few_argumentsF(args); old_bds_top=bds_top; top=vs_top; eval(MMcar(args)); vs_top=top; symbols=vs_base[0]; vs_push(symbols); eval(MMcadr(args)); vs_top=top+1; values=vs_base[0]; vs_push(values); while (!endp(symbols)) { var = MMcar(symbols); if (type_of(var)!=t_symbol) not_a_symbol(var); if ((enum stype)var->s.s_stype == stp_constant) FEerror("Cannot bind the constant ~S.", 1, var); if (endp(values)) { bds_bind(var, OBJNULL); } else { bds_bind(var, MMcar(values)); values=MMcdr(values); } symbols=MMcdr(symbols); } Fprogn(MMcddr(args)); bds_unwind(old_bds_top); } Fprogn(body) object body; { object endp_temp; if(endp(body)) { vs_base=vs_top; vs_push(Cnil); } else { object *top=vs_top; do { vs_top=top; eval(MMcar(body)); body=MMcdr(body); } while (!endp(body)); } } Fprog1(arg) object arg; { object endp_temp; object *top = vs_top; if(endp(arg)) FEtoo_few_argumentsF(arg); eval(MMcar(arg)); vs_top = top; vs_push(vs_base[0]); for(arg = MMcdr(arg); !endp(arg); vs_top = top+1, arg = MMcdr(arg)) eval(MMcar(arg)); vs_base = top; vs_top = top + 1; } Fprog2(arg) object arg; { object endp_temp; object *top = vs_top; if(endp(arg) || endp(MMcdr(arg))) FEtoo_few_argumentsF(arg); eval(MMcar(arg)); vs_top = top; arg = MMcdr(arg); eval(MMcar(arg)); vs_top = top; vs_push(vs_base[0]); for(arg = MMcdr(arg); !endp(arg); vs_top = top+1, arg = MMcdr(arg)) eval(MMcar(arg)); vs_base = top; vs_top = top+1; } init_prog() { make_special_form("TAGBODY", Ftagbody); make_special_form("PROG", Fprog); make_special_form("PROG*", FprogA); make_special_form("GO", Fgo); make_special_form("PROGV", Fprogv); make_special_form("PROGN",Fprogn); make_special_form("PROG1",Fprog1); make_special_form("PROG2",Fprog2); }
These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.