Search code examples
macroslispcommon-lisp

How do you compile macros in a Lisp compiler?


In a Lisp interpreter, there can easily be a branch in eval that can expand a macro, and in the process of expanding it, call functions to build up the expanded expression. I've done this before using low-level macros, it's easily concieved.

But, in a compiler there aren't any functions to call to build up the expanded code: The issue can be seen quite simply in the following example:

(defmacro cube (n)
    (let ((x (gensym)))
      `(let ((,x ,n))
          (* ,x ,x ,x))))

When the macro is expanded by an interpreter, it calls gensym and does what you expect. When expanded by a compiler, you'd generate the code for a let which binds x to (gensym) but the gensymmed symbol is only necessary for the compiler to do the right thing. And since gensym isn't actually called before the macro is compiled, it's not very useful.

This gets even more strange to me when a macro builds up a list to be used as the expansion using map or filter.

So how does this work? Surely the compiled code isn't compiled to (eval *macro-code*) because that'd be horribly inefficient. Is there a well written Lisp compiler where this is clear?


Solution

  • How this works is very different in various Lisp dialects. For Common Lisp it is standardized in the ANSI Common Lisp standard and the various Common Lisp implementations differ mostly whether they use a compiler, an interpreter or both.

    The following assumes Common Lisp.

    EVAL is not the interpreter. EVAL can be implemented with a compiler. Some Common Lisp implementations even don't have an interpreter. Then EVAL is a call to the compiler to compile the code and then calls the compiled code. These implementations use an incremental compiler, which can compile also simple expressions like 2, (+ 2 3), (gensym), and so on.

    Macroexpansion is done with the functions MACROEXPANDand MACROEXPAND-1.

    A macro in Common Lisp is a function that expects some forms and returns another form. DEFMACRO registers this function as a macro.

    Your macro

    (defmacro cube (n)
      (let ((x (gensym)))
        `(let ((,x ,n))
            (* ,x ,x ,x))))
    

    is nothing but a Lisp function, which is registered as a macro.

    The effect is similar to this:

    (defun cube-internal (form environment)
      (destructuring-bind (name n) form   ; the name would be CUBE
        (let ((x (gensym)))
          `(let ((,x ,n))
             (* ,x ,x ,x)))))
    
    (setf (macro-function 'my-cube) #'cube-internal)
    

    In a real CL implementation DEFMACRO expands differently and does not use a name like CUBE-INTERNAL. But conceptually it is defining a macro function and registering it.

    When the Lisp compiler sees a macro definition, it usually compiles the macro function and stores it in the current so-called environment. If the environment is the runtime environment, it is remembered at runtime. If the environment is the compiler environment while compiling a file, the macro is forgotten after the file is compiled. The compiled file needs to be loaded so that Lisp then knows the macro.

    So, there is a side effect in defining a macro and compiling it. The compiler remembers the compiled macro and stores its code.

    When the compiler now sees some code which uses the macro (cube 10), then the compiler just calls the macro function which is stored in the current environment under the name CUBE, calls this macro function which 10 as an argument, and then compiles the generated form. As mentioned above, it is not done directly, but via the MACROEXPAND functions.

    Here is the Macro definition:

    CL-USER 5 > (defmacro cube (n)
                  (let ((x (gensym)))
                    `(let ((,x ,n))
                       (* ,x ,x ,x))))
    CUBE
    

    We compile the macro:

    CL-USER 6 > (compile 'cube)
    CUBE
    NIL
    NIL
    

    MACRO-FUNCTION returns the function for a macro. We can call it like any other function with FUNCALL. It expects two arguments: a whole form like (cube 10) and an environment (here NIL).

    CL-USER 7 > (funcall (macro-function 'cube) '(cube 10) nil)
    (LET ((#:G2251 10)) (* #:G2251 #:G2251 #:G2251))
    

    It is also possible to take a function (which accepts two arguments: a form and an environment) and store it using SETF as a macro function.

    Summary

    When the Common Lisp compiler runs, it simply knows the macro functions and calls them when necessary to expand code via the built-in macro expander. The macro functions are simply Lisp code themselves. When the Lisp compiler sees a macro definition, it compiles the macro function, stores it in the current environment and uses it to expand subsequent uses of the macro.

    Note: This makes it necessary in Common Lisp that a macro is defined before it can be used by the compiler.