Wrote a preexpander for a simplified variant of syntax-case.

This commit is contained in:
Georges Dupéron 2016-04-07 18:56:06 +02:00
parent abe7438d7d
commit 984f861a85
3 changed files with 112 additions and 0 deletions

17
and.rkt Normal file
View File

@ -0,0 +1,17 @@
#lang racket
(provide preexpanded-and)
(require syntax/parse
(for-template racket/base))
(define (preexpanded-and stx)
(syntax-parse stx
[(clause)
#'clause]
[(#t . rest)
(preexpanded-and #`rest)]
[(clause . rest)
#`(if clause
#,(preexpanded-and #`rest)
#f)]))

65
syntax-case.rkt Normal file
View File

@ -0,0 +1,65 @@
#lang racket
(provide preexpanded-syntax-case/no-bind)
(require (for-syntax preexpanded/and
racket/pretty
syntax/stx
syntax/parse
syntax/parse/experimental/template)
syntax/stx)
(begin-for-syntax
(define-syntax-class (pat part)
#:attributes (test)
(pattern (~literal _)
#:with test #'#t)
(pattern ()
#:with test #`(null? #,part))
(pattern ((~literal ~literal) identifier:id)
#:when (syntax-pattern-variable?
(syntax-local-value #'identifier
(λ _ #f)))
#:with test #`(free-identifier=? #,part (quote-syntax identifier)))
(pattern ((~literal ~literal) identifier:id)
#:with test #`(free-identifier=? #,part (quote-syntax identifier)))
(pattern ((~literal ~datum) identifier:id)
#:with test #`(eq? (syntax-e #,part) 'identifier))
(pattern k:keyword
#:with test #`(eq? (syntax-e #,part) 'k))
(pattern ((~var sub (pat #'car-part)) . (~var rest (pat #'cdr-part)))
;; TODO: optimize the #t case.
#:with test (preexpanded-and
#`((stx-pair? #,part)
(let-values ([(car-part) (stx-car #,part)]
[(cdr-part) (stx-cdr #,part)])
#,(preexpanded-and
#'(sub.test rest.test))))))))
(begin-for-syntax
(define-splicing-syntax-class (clause-maybe-dotted whole)
(pattern (~seq [(~var pat (pat whole)) body]
(~optional (~seq (patvar ...)
(~and ddd (~literal ...)))))
#:with test #'pat.test
;#:with (patvar ...) #`#,(attribute pat.patvar)
#:with expanded
(if (attribute ddd)
#'(map (lambda (patvar ...)
(with-syntax ([patvar patvar] ...)
#'[test body]))
(syntax->list #'(patvar (... ...)))
...)
#'(list #'[test body])))))
(define-syntax (preexpanded-syntax-case/no-bind stx)
(syntax-parse stx
[(_ name stx2 (~var clause (clause-maybe-dotted #'whole)) ...)
((λ (x)
;(pretty-write (syntax->datum x))
x)
#'#`(let-values ([(whole) stx2])
(cond #,@clause.expanded
...
[else (raise-syntax-error 'name "Invalid syntax" whole)])))]))

30
test/test-syntax-case.rkt Normal file
View File

@ -0,0 +1,30 @@
#lang racket
(require (for-syntax preexpanded/syntax-case))
(define-syntax (define-a*-b stx)
(syntax-case stx ()
[(_ name [foo val] ...)
#`(define-syntax (name stx2)
#,(preexpanded-syntax-case/no-bind define-a*-b stx2
[(_ #:a (~literal foo)) #'val]
(foo val)
...
[(_ #:b) #'2]))]))
(define-a*-b myab [a 10] [b 20] [c 30])
(myab #:a a)
(myab #:a b)
(myab #:a c)
;(myab #:a d) ;; Invalid syntax, as expected
;(myab #:a) ;; Invalid syntax, as expected
;(myab #:a a e) ;; Invalid syntax, as expected
(myab #:b)
#;(preexpanded-syntax-case/no-bind mymacro #'(mymacro #:a +)
[(_ #:a (~literal +)) 1]
[(_ #:b) 2])
#;(preexpanded-syntax-case/no-bind mymacro #'(mymacro #:b foo)
[(_ #:a (~literal +)) 1]
[(_ #:b (~datum foo)) 2])