What is Macro Hygiene?
One important, though surprisingly uncommon, feature of macro systems is that of hygiene. I mentioned in a previous post that I would eventually say something about hygiene. It turns out macro hygiene is somewhat tricky to define precisely, and I know a couple of people who are actively working on a formal definition of hygiene. The intuition behind hygiene isn’t too bad though. Basically, we want our macros to not break our code. So how can macros break code?
Recall that macros are basically programs that transform your program’s code, rather than runtime values. In doing so, they may introduce new variable bindings. If we’re not careful, these new bindings can end up capturing variables in your own code. That is, the new binding might shadow a variable you as the programmer have already created. All of a sudden, the variable you thought you were referring to is no longer the same thing, and because all of this code is hidden in a macro expansion, it will be very hard to figure out what’s going on.
Consider the following Scheme macro for or
.
(define-syntax or
(syntax-rules ()
((_ e1 e2)
(let ((t e1))
(if t t e2)))))
Like a good macro, this uses a temporary variable, t
, to avoid
calculating e1
twice and potentially duplicating any side effects in
that expression. Unfortunately, if we’re not careful, this binding can
capture an existing binding of t
. Consider the following program.
(let ((t 5))
(or #f t))
If you run this in the Scheme REPL, you should get 5
. Let’s see what
happens if we blindly expand the or
macro without regard for
hygiene. We would end up with the following program.
(let ((t 5))
(let ((t #f))
(if t t t)))
This program evaluates to #f
, which is the exact opposite of what we
were supposed to get! Expanding our macro has shadowed the binding of
t
to 5
with a binding of t
to #f
.
One way to work around this, which was a common trick for LISP
programmers of yore is to choose variable names that a programmer is
unlikely to guess. We could rewrite our or
macro like this:
(define-syntax or
(syntax-rules ()
((_ e1 e2)
(let ((this-is-my-super-secret-name-which-you-will-never-guess e1))
(if this-is-my-super-secret-name-which-you-will-never-guess
this-is-my-super-secret-name-which-you-will-never-guess
e2)))))
This works okay, except it’s a lot more typing. Eventually, some
overly clever programmer is going to name their own variable
this-is-my-super-secret-name-which-you-will-never-guess
, and then
their program will break in really unexpected ways. It’d really be
great if our macro expander could take care of these issues on its
own. That way, the macro writers could type less and use names they
like, and macro users don’t have to worry about their variables being
captured unexpectedly.
We could modify the macro expander to automatically rename any
variables bound by a macro expansion. In this case, our simple test
program would expand as follows, using our first definition of or
.
(let ((t 5))
(let ((t.1 #f))
(if t.1 t.1 t)))
This program evaluates as expected, so things are looking good. But, what if someone writes this program?
(let ((if (lambda (a b c) b)))
(or #f 5))
We’ll use our variable-renaming expander and see that we end up with the following program:
(let ((if (lambda (a b c) b)))
(let ((t.1 #f))
(if t.1 t.1 5)))
This program once again evaluates to #f
instead of 5
like we’d
like it too. This illustrates the second, and more subtle, class of
hygiene error. The problem is that the programmer’s definition of if
has captured the if
used by the expansion of or
. Now, many
languages treat keywords like if
specially and don’t let you name
your variables after them. Enforcing a rule like that in Scheme would
solve this particular case, but at the cost of a lot of the power that
Scheme programmers love. The proper solution is to find some way of
tracking what if
was bound to when the or
macro was defined and
using that version in the expansion of or
.
Properly maintaining hygiene in macro systems turns out to be really tricky. The reward is worth it, however, as programmers can then reason much more easily about the behavior of macros and start to rely on them in large projects. Macros are a powerful feature of programming languages, and many newer language have some form of macro system. Sadly, these are often not hygienic, or they so “Oh, we’ll add hygiene later.” Given how important hygiene is, and how tricky it is to get right, language designers should really implemented hygiene in their macro systems from the very beginning.