Welcome to Expanders!! :D
The project lets you define expanders. An expander is like a namespace for forms that should transform in a specific way.
It easier to understand with an example. Suppose we have the following form (op (+ 3 4) b). This form will be used in two different macros named plus-macro and minus-macro. Each macro receives a form. That form could start with op. In that case, op is substituted by + or - respectively.
Let's define 2 different expanders:
;; We assume (use-package #:expanders)
(defexpander plus-expander)
(defexpander minus-expander)minus-expanderNow we have two different expanders. The function exp:expanderp can tell us if a symbol denotes an expander:
(expanderp 'plus-expander)t(expanderp 'hey)nil(expanderp 'minus-expander)tNow it is time to define the expansion op for each expander using exp:defexpansion:
(defexpansion plus-expander op (a b)
"OP to + expansion"
`(+ ,a ,b))
(defexpansion minus-expander op (a b)
"OP to - expansion"
`(- ,a ,b))opWe can check if a symbol is an expansion for a given expander using exp:expansionp:
(expansionp 'plus-expander 'op)t(expansionp 'plus-expander 'hey)nil(expansionp 'minus-expander 'op)tAlso, we can retrieve or set the docstring using documentation:
(documentation 'op 'plus-expander)"OP to + expansion"(let ((old-docstring (documentation 'op 'minus-expander)))
(setf (documentation 'op 'minus-expander) "Another docstring")
(let ((new-docstring (documentation 'op 'minus-expander)))
(format t "Old: ~s~%New: ~s" old-docstring new-docstring)))Old: "OP to - expansion"
New: "Another docstring"
nilWe can expand a form using exp:expand. Note that the form must be a list starting with the name of an expansion:
(expand 'plus-expander '(op 3 (+ 5 6)))(+ 3 (+ 5 6))(expand 'minus-expander '(op 3 (+ 5 6)))(- 3 (+ 5 6))Finally, let's define the macros plus-macro and minus-macro:
(defmacro plus-macro (form)
(if (and (consp form)
(expansionp 'plus-expander (car form)))
(expand 'plus-expander form)
form))plus-macro(defmacro minus-macro (form)
(if (and (consp form)
(expansionp 'minus-expander (car form)))
(expand 'minus-expander form)
form))minus-macroIf we use the form (op 5 4) we will see that each macro will expand to (+ 5 4) or (- 5 4) respectively.
(plus-macro (op 5 4))9(minus-macro (op 5 4))1- It is common: I have noticed that having expanders is a relatively common pattern in macros. The best example is
setfand itssetf-expanders. Another project using expanders is CFFI and its type parsers. In my own projects I ended up using the same techniques (Clith). - Duality of syntax: We can increase the duality of syntax using expanders. The best example is
setf. Thanks tosetfwe don't need names for setters because they come for free with the getter.
Defines an expander represented by the symbol SYM.
Defines an expansion for the expander EXPANDER. NAME must be a symbol denoting
the new expansion. ARGS is a destructuring lambda list. This must return the desired
expansion for NAME and EXPANDER.
Expands a form using an expander. FORM must be a list starting with a valid
expansion symbol for the expander EXPANDER.
Check if a symbol denotes an expander.
Checks if EXPANSION is a valid expansion for the expander EXPANDER.
EXPANDER must be a valid expander.