1996-09-27 06:29:02 -04:00
|
|
|
|
/*
|
|
|
|
|
* m a c r o s . c -- Simple statically scoped macros
|
|
|
|
|
*
|
1999-09-05 07:16:41 -04:00
|
|
|
|
* Copyright <EFBFBD> 1993-1999 Erick Gallesio - I3S-CNRS/ESSI <eg@unice.fr>
|
1996-09-27 06:29:02 -04:00
|
|
|
|
*
|
|
|
|
|
*
|
1999-09-05 07:16:41 -04:00
|
|
|
|
* Permission to use, copy, modify, distribute,and license this
|
|
|
|
|
* software and its documentation for any purpose is hereby granted,
|
|
|
|
|
* provided that existing copyright notices are retained in all
|
|
|
|
|
* copies and that this notice is included verbatim in any
|
|
|
|
|
* distributions. No written agreement, license, or royalty fee is
|
|
|
|
|
* required for any of the authorized uses.
|
|
|
|
|
* This software is provided ``AS IS'' without express or implied
|
|
|
|
|
* warranty.
|
1996-09-27 06:29:02 -04:00
|
|
|
|
*
|
|
|
|
|
* Author: Erick Gallesio [eg@unice.fr]
|
|
|
|
|
* Creation date: ??-Oct-1993 ??:??
|
1999-09-05 07:16:41 -04:00
|
|
|
|
* Last file update: 3-Sep-1999 20:21 (eg)
|
1996-09-27 06:29:02 -04:00
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "stk.h"
|
|
|
|
|
|
|
|
|
|
PRIMITIVE STk_macro(SCM args, SCM env, int len)
|
|
|
|
|
{
|
|
|
|
|
SCM z, code;
|
|
|
|
|
|
1998-04-10 06:59:06 -04:00
|
|
|
|
if (len != 2) Err("macro: Bad definition", args);
|
1996-09-27 06:29:02 -04:00
|
|
|
|
|
|
|
|
|
code = Cons(Sym_lambda, args); /* Create code before to avoid GC problems */
|
|
|
|
|
NEWCELL(z, tc_macro);
|
|
|
|
|
z->storage_as.macro.code = EVAL(code);
|
1998-04-10 06:59:06 -04:00
|
|
|
|
z->storage_as.macro.env = Ntruth;
|
1996-09-27 06:29:02 -04:00
|
|
|
|
return z;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-10 06:59:06 -04:00
|
|
|
|
PRIMITIVE STk_macro_R5(SCM args, SCM env, int len)
|
|
|
|
|
{
|
|
|
|
|
SCM z = STk_macro(args, env, len);
|
|
|
|
|
|
|
|
|
|
z->storage_as.macro.env = env;
|
|
|
|
|
return z;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1996-09-27 06:29:02 -04:00
|
|
|
|
/*
|
|
|
|
|
* macro_expand and macro_expand_1 are defined as fsubr rather than subr_1.
|
|
|
|
|
* This provision permits to carry the environment with the form to expand
|
|
|
|
|
* (this permits to corerctly expanse macros which are not are global level,
|
|
|
|
|
* since we do a varlookup instead of grabbing the macro in a global var).
|
|
|
|
|
* But are there people which define macros at a level which are not global?
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
static SCM expand(SCM form, SCM env, int just_1)
|
|
|
|
|
{
|
|
|
|
|
if (CONSP(form)) {
|
|
|
|
|
register SCM tmp, car = CAR(form);
|
|
|
|
|
|
|
|
|
|
if (SYMBOLP(car) && MACROP(tmp = *STk_varlookup(car, env, FALSE))) {
|
|
|
|
|
tmp = Apply(tmp->storage_as.macro.code, form);
|
|
|
|
|
return just_1 ? tmp : expand(tmp, env, FALSE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return form;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PRIMITIVE STk_macro_expand_1(SCM form, SCM env, int len)
|
|
|
|
|
{
|
|
|
|
|
if (len != 1) Err("macro-expand-1: bad number of parameters", form);
|
|
|
|
|
return expand(EVAL(CAR(form)), env, TRUE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PRIMITIVE STk_macro_expand(SCM form, SCM env, int len)
|
|
|
|
|
{
|
|
|
|
|
if (len != 1) Err("macro-expand: bad number of parameters", form);
|
|
|
|
|
return expand(EVAL(CAR(form)), env, FALSE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PRIMITIVE STk_macro_body(SCM form)
|
|
|
|
|
{
|
|
|
|
|
if (NMACROP(form)) Err("macro-body: bad macro", form);
|
|
|
|
|
return Cons(Intern("macro"), CDR(form->storage_as.macro.code));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PRIMITIVE STk_macrop(SCM obj)
|
|
|
|
|
{
|
|
|
|
|
return MACROP(obj)? Truth: Ntruth;
|
|
|
|
|
}
|
|
|
|
|
|