123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
- module records; % A record package for RLISP using MSTRUCT.
- % Author: Bruce Florman.
- % Copyright: (c) 1989 The RAND Corporation. All rights reserved.
- % Revision History:
- % 01/26/89 BAF -- Added this file header.
- % Sat Apr 24 12:38:32 1993 - Remove non-RLISP'88 functions (first,
- % etc.).
- % BothTimes Load MSTRUCT;
- %-----------------------------------------------------------------------
- % RECORD Declaration
- %-----------------------------------------------------------------------
- Expr PROCEDURE RecordStat();
- % RECORD <struct-name>
- % { /* <annotation> */ }
- % { WITH <field> := <expression> { , <field> := <expression> }... }
- % { HAS <option> { , <option> }... } ;
- begin scalar f, stat;
- f := FlagP('HAS,'DELIM);
- Flag('(HAS),'DELIM);
- stat := Errorset('(RecordStat1),NIL,nil);
- if not f then RemFlag('(HAS),'DELIM);
- if errorp stat THEN while cursym!* neq '!*SEMICOL!* do scan()
- else return car stat
- end;
- expr procedure recordstat1();
- begin scalar structname, annotation, fields, options;
- structname := Scan();
- if not idp structname then symerr('RECORD, T);
- if eqcar(scan(), '!*COMMENT!*) then
- <<annotation := cadr cursym!*; Scan()>>;
- if cursym!* eq 'WITH then fields := remcomma xread nil;
- if cursym!* eq 'HAS then options := remcomma xread NIL;
- if cursym!* eq '!*SEMICOL!* then
- return {'RECORD, structname, annotation, fields, options}
- else symerr('RECORD, T)
- END;
- Put('RECORD,'STAT,'RecordStat);
- expr procedure formrecord(u, vars, mode);
- apply(form_function, cdr u)
- where form_function =
- function(lambda(record_name, annotation, fields, options);
- begin scalar structspec, fieldspecs, constructor, form;
- structspec := Form_structure_specification(record_name, options);
- fieldspecs := Form_field_specifications(fields);
- constructor := Cdr Atsoc('CONSTRUCTOR,
- Get_defstruct_options structspec);
- form := {NIL};
- tconc(form, 'PROGN);
- if constructor then
- << tconc(form,
- {'put, mkquote constructor,
- '(QUOTE FORMFN),
- '(QUOTE FORM_RECORD_CONSTRUCTOR)});
- put(constructor, 'FORMFN, 'FORM_RECORD_CONSTRUCTOR) >>;
- if annotation then
- tconc(form, {'PUT, mkquote record_name,
- '(QUOTE ANNOTATION),
- annotation});
- tconc(form, 'DEFSTRUCT . structspec . fieldspecs);
- return Car form
- end);
- Put('RECORD, 'FORMFN, 'FormRecord);
- expr procedure tconc(ptr,elem);
- % ACONC with pointer to end of list. Ptr is (list . last CDR of
- % list). Returns updated Ptr. Ptr should be initialized to
- % (NIL . NIL) before calling the first time.
- <<elem := list elem;
- if not pairp ptr then elem . elem
- else if null cdr ptr then rplaca(rplacd(ptr,elem),elem)
- else <<rplacd(cdr ptr,elem); rplacd(ptr,elem)>>>>;
- expr procedure Form_structure_specification(record_name, options);
- append(defaults,
- for each entry in options
- collect if atom entry then entry
- else if eqcar(entry, 'NO) and length entry=2 then
- {cadr entry, NIL}
- else if car entry eq 'EQUAL and length entry=3 then
- {cadr entry, caddr entry}
- else error(0, {"Bad RECORD option:", entry}))
- where defaults = {record_name,{'CONSTRUCTOR, record_name},
- 'predicate};
- expr procedure form_field_specifications field_list;
- for each entry in field_list
- join
- if eqcar(entry, 'SETQ)
- then {{cadr(entry), form1(caddr entry, NIL, 'SYMBOLIC)}}
- else nil;
- expr procedure form_record_constructor(u, vars, mode);
- begin scalar constructor, arglist;
- constructor := car u;
- arglist := {NIL};
- for each arg in cdr u
- do if eqcar(arg, 'SETQ) then
- << tconc(arglist, cadr arg);
- tconc(arglist, form1(caddr arg, vars, mode)) >>
- else rederr {arg, "is not a proper initialization form for",
- constructor};
- return constructor . car arglist;
- end;
- endmodule;
- end;
|