1.2.1 Phases and Reusable Syntax Classes

1.2 Examples
top
up

1.2.1Phases and Reusable Syntax ClassesπŸ”— i

As demonstrated in the Introduction, the simplest place to define a syntax class is within the macro definition that uses it. But that limits the scope of the syntax class to the one client macro, and it makes for very large macro definitions. Creating reusable syntax classes requires some awareness of the Racket phase level separation. A syntax class defined immediately within a module cannot be used by macros in the same module; it is defined at the wrong phase.

> (module phase-mismatch-modracket
(require syntax/parse(for-syntax syntax/parse))
(pattern (abc)))
(define-syntax (macrostx)
[(_ f:foo)#'(+ f.af.bf.c)])))

syntax-parse: not defined as syntax class

at: foo

in: (syntax-parse stx ((_ f:foo) (syntax (+ f.a f.b

f.c))))

In the module above, the syntax class foo is defined at phase level 0. The reference to foo within macro, however, is at phase level 1, being the implementation of a macro transformer. (Needing to require syntax/parse twice, once normally and once for-syntax is a common warning sign of phase level incompatibility.)

The phase level mismatch is easily remedied by putting the syntax class definition within a begin-for-syntax block:

> (module phase-ok-modracket
(require (for-syntax syntax/parse))
(pattern (abc))))
(define-syntax (macrostx)
[(_ f:foo)#'(+ f.af.bf.c)])))

In the revised module above, foo is defined at phase 1, so it can be used in the implementation of the macro.

An alternative to begin-for-syntax is to define the syntax class in a separate module and require that module for-syntax .

> (module stxclass-modracket
(require syntax/parse)
(pattern (abc)))
(provide foo))
> (module macro-modracket
(require (for-syntax syntax/parse
'stxclass-mod))
(define-syntax (macrostx)
[(_ f:foo)#'(+ f.af.bf.c)]))
(provide macro))
> (require 'macro-mod)
> (macro(123))

6

If a syntax class refers to literal identifiers, or if it computes expressions via syntax templates, then the module containing the syntax class must generally require for-template the bindings referred to in the patterns and templates.

> (module arith-keywords-modracket
(provide plustimes))
> (module arith-stxclass-modracket
(require syntax/parse
(for-template 'arith-keywords-mod
racket))
#:literals(plustimes)
(pattern n:nat
#:withexpr #'n)
(pattern (plusa:arithb:arith)
#:withexpr #'(+ a.exprb.expr))
(pattern (timesa:arithb:arith)
#:withexpr #'(* a.exprb.expr)))
(provide arith))
> (module arith-macro-modracket
(require (for-syntax syntax/parse
'arith-stxclass-mod)
'arith-keywords-mod)
(define-syntax (arith-macrostx)
[(_ a:arith)
#'(values 'a.expra.expr)]))
(provide arith-macro
(all-from-out 'arith-keywords-mod)))
> (require 'arith-macro-mod)
> (arith-macro(plus1(times23)))

'(+ 1 (* 2 3))

7

In 'arith-stxclass-mod, the module 'arith-keywords-mod must be required for-template because the keywords are used in phase-0 expressions. Likewise, the module racket must be required for-template because the syntax class contains syntax templates involving + and * (and, in fact, the implicit #%app syntax). All of these identifiers (the keywords plus and times; the procedures + and * ; and the implicit syntax #%app ) must be bound at “absolute” phase level 0. Since the module 'arith-stxclass-mod is required with a phase level offset of 1 (that is, for-syntax ), it must compensate with a phase level offset of -1, or for-template .

top
up

AltStyle γ«γ‚ˆγ£γ¦ε€‰ζ›γ•γ‚ŒγŸγƒšγƒΌγ‚Έ (->γ‚ͺγƒͺγ‚ΈγƒŠγƒ«) /