/*
 * m a c r o s . c			-- Simple statically scoped macros
 *
 * Copyright (C) 1993,1994,1995 Erick Gallesio - I3S-CNRS/ESSI <eg@unice.fr>
 * 
 *
 * Permission to use, copy, and/or distribute this software and its
 * documentation for any purpose and without fee is hereby granted, provided
 * that both the above copyright notice and this permission notice appear in
 * all copies and derived works.  Fees for distribution or use of this
 * software or derived works may only be charged with express written
 * permission of the copyright holder.  
 * This software is provided ``as is'' without express or implied warranty.
 *
 * This software is a derivative work of other copyrighted softwares; the
 * copyright notices of these softwares are placed in the file COPYRIGHTS
 *
 *           Author: Erick Gallesio [eg@unice.fr]
 *    Creation date: ??-Oct-1993 ??:?? 
 * Last file update:  2-Jun-1995 22:42
 *
 */


#include "stk.h"

PRIMITIVE STk_macro(SCM args, SCM env, int len)
{
  SCM z, code;
  
  if (len != 2) Err("macro: Bad parameter list", args);

  code = Cons(Sym_lambda, args); /* Create code before to avoid GC problems */
  NEWCELL(z, tc_macro);
  z->storage_as.macro.code = EVAL(code);
  return z;
}

/*
 * 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;
}

