Typed macros through syntax parse?

Working on some advanced trickery with syntax-parse to essentially get
typed-macros. Needs more work.
This commit is contained in:
William J. Bowman 2015-11-05 18:33:04 -05:00
parent 7624090e5a
commit 43e82910cb
No known key found for this signature in database
GPG Key ID: DDD48D26958F0D1A
2 changed files with 37 additions and 27 deletions

View File

@ -60,6 +60,8 @@
(for-syntax (all-from-out racket)) (for-syntax (all-from-out racket))
;; reflection ;; reflection
(for-syntax (for-syntax
gamma
extend-Γ/syn
cur->datum cur->datum
cur-expand cur-expand
type-infer/syn type-infer/syn

View File

@ -113,31 +113,39 @@
#`(elim D U P #,@(map rewrite-clause (syntax->list #'(clause* ...))) p ... e)])) #`(elim D U P #,@(map rewrite-clause (syntax->list #'(clause* ...))) p ... e)]))
(begin-for-syntax (begin-for-syntax
(define-syntax-class (gamma/cur-expr env)
(pattern e:expr
#:fail-unless (parameterize ([gamma env])
(type-infer/syn #'e))
(format "Could not infer a type for Cur term ~a"
(syntax->datum #'e))
#:attr type (parameterize ([gamma env])
(type-infer/syn #'e))))
(define-syntax-class cur-expr
(pattern e
#:declare e (gamma/cur-expr (gamma))
#:attr type #'e.type))
(define-syntax-class let-clause (define-syntax-class let-clause
(pattern (pattern
(~or (x:id e) ((x:id (~datum :) t) e)) (~or ((x:id (~commit (~datum :)) t:cur-expr) e) (x:id e:cur-expr))
#:attr id #'x #:fail-unless
#:attr expr #'e (or (not (attribute t))
#:attr type (cond (type-check/syn? #'e #'t))
[(attribute t) (format "Term ~a does not have expected type ~a. Inferred type was ~a"
;; TODO: Code duplication in :: (cur->datum #'e) (cur->datum #'t) (cur->datum (type-infer/syn #'e)))
(unless (type-check/syn? #'e #'t) #:attr id #'x
(raise-syntax-error #:attr expr #'e
'let #:attr type (if (attribute t) #'t #'e.type))))
(format "Term ~a does not have expected type ~a. Inferred type was ~a"
(cur->datum #'e) (cur->datum #'t) (cur->datum (type-infer/syn #'e)))
#'e (quasisyntax/loc #'x (x e))))
#'t]
[(type-infer/syn #'e)]
[else
(raise-syntax-error
'let
"Could not infer type of let bound expression"
#'e (quasisyntax/loc #'x (x e)))]))))
(define-syntax (let syn) (define-syntax (let syn)
(syntax-parse syn (syntax-parse syn
[(let (c:let-clause ...) body) [(let (c:let-clause ...) body)
#'((lambda* (c.id : c.type) ... body) c.e ...)])) #'((lambda* (c.id : c.type) ... body) c.expr ...)]))
#;(define-syntax (let syn)
(syntax-parse syn
[(let ([x:id e:cur-expr]) body)
#:declare body (gamma/cur-expr (extend-Γ/syn gamma #'x #'e.type))
#'((lambda (x : e.type) body) e)]))
;; Normally type checking will only happen if a term is actually used. This forces a term to be ;; Normally type checking will only happen if a term is actually used. This forces a term to be
;; checked against a particular type. ;; checked against a particular type.
@ -203,9 +211,8 @@
Type) Type)
(check-equal? (check-equal?
(let ([x Type] (let ([x Type])
[y (λ (x : (Type 1)) x)]) x)
(y x))
Type) Type)
(check-equal? (check-equal?
@ -216,11 +223,12 @@
;; check that raises decent syntax error ;; check that raises decent syntax error
;; Can't use this because (lambda () ...) and thunk are not things in Cur at runtime ;; Can't use this because (lambda () ...) and thunk are not things in Cur at runtime
(let ([x : (Type 1) Type]
[y (λ (x : (Type 1)) x)])
(y x))
#;(check-exn #;(check-exn
exn:fail:syntax? ;exn:fail:syntax?
(let ([x : (Type 1) Type] )
[y (λ (x : (Type 1)) x)])
(y x)))
;; check that raises type error ;; check that raises type error
#;(check-exn #;(check-exn