extensible-parser-specifica.../private/define-syntax+simple-api.rkt

85 lines
3.5 KiB
Racket

#lang racket/base
(provide define-syntax/parse+simple
(for-syntax define/syntax-parse+simple))
(require phc-toolkit/untyped
syntax/parse/define
(for-syntax racket/base
syntax/parse
racket/stxparam
racket/syntax)
(for-meta 2
racket/base
syntax/parse
racket/syntax
phc-toolkit/untyped
(prefix-in syntax/parse: syntax/parse/private/residual-ct)))
(define-simple-macro (define-syntax/parse+simple [name . args] . body)
#:with name-forward (format-id #'name "~a-forward-attributes" #'name)
#:with tmp-forward (format-id #'tmp "~a-forward-attributes" #'tmp)
(begin
(begin-for-syntax
(define/syntax-parse+simple [tmp . args] . body)
(define-syntax name-forward (make-rename-transformer #'tmp-forward)))
(define-syntax name tmp)))
(begin-for-syntax
(define-syntax (define/syntax-parse+simple stx)
(syntax-case stx ()
[(_ [name . args] . body)
(let ()
(define introducer (make-syntax-introducer))
(define/with-syntax args-stxclass
(introducer (datum->syntax #'args 'args-stxclass) 'add))
(define/with-syntax body-introduced
(introducer #'body 'add))
#'(begin
(define-syntax-class args-stxclass
#:auto-nested-attributes
(pattern args))
(define/syntax-parse+simple/stxclass [name args-stxclass]
. body-introduced)))]))
(define-syntax define/syntax-parse+simple/stxclass
(syntax-parser
[(_ [name (~var cls (static syntax/parse:stxclass? "a syntax class"))]
. body)
#:with colon-stxclass (format-id #'cls ":~a" #'cls)
#:with name-forward (format-id #'name "~a-forward-attributes" #'name)
(with-disappeared-uses
(let ()
(define c (syntax-local-value/record #'cls syntax/parse:stxclass?))
(define attrs (syntax/parse:stxclass-attrs c))
(define/with-syntax (attr-name ) (map syntax/parse:attr-name attrs))
(define/with-syntax (attr-name/ctx )
(stx-map (λ (a) (datum->syntax #'body (syntax-e a)))
#'(attr-name )))
(define-temp-ids "~a/arg" (attr-name ))
(define/with-syntax (attr-depth )
(map syntax/parse:attr-depth attrs))
#'(begin
(define (name stx2)
(syntax-parameterize ([stx (make-rename-transformer #'stx2)])
(syntax-parse stx2
[(_ . colon-stxclass) . body])))
(define (private-simple-api stx/arg attr-name/arg )
(syntax-parameterize ([stx (make-rename-transformer #'stx/arg)])
(syntax-parse #'nothing
[(~bind [(attr-name/ctx attr-depth) attr-name/arg] )
. body])))
(define-syntax (name-forward stx3)
(syntax-case stx3 ()
[(_)
#`(private-simple-api
stx
(attribute #,(datum->syntax stx3 'attr-name))
)]
[(_ forward-args-prefix)
#`(private-simple-api
stx
(attribute #,(format-id stx3 "~a.~a"
#'forward-args-prefix
'attr-name))
)])))))])))