Added a fancy let form

This commit is contained in:
William J. Bowman 2015-10-02 21:14:23 -04:00
parent e84b7d7f2e
commit a80c4a162f
No known key found for this signature in database
GPG Key ID: DDD48D26958F0D1A
2 changed files with 70 additions and 1 deletions

View File

@ -145,6 +145,24 @@ Necessary for more advanced types, like @racket[And], because @racket[case] is n
a])]
}
@defform[(let (clause ...) body)
#:grammar
[(clause
(code:line (id expr))
(code:line ((id : type) expr)))]]{
Evaluates the expressions @racket[expr] from each clause, left to right, and binds them to each
@racket[id]. If a @racket[type] is not given for the @racket[id], attempts to infer the type of the
corresponding @racket[expr], raising a syntax error if no type can be inferred.
@examples[#:eval curnel-eval
(let ([x Type]
[y (λ (x : (Type 1)) x)])
(y x))
(let ([x uninferrable-expr]
[y (λ (x : (Type 1)) x)])
(y x))]
}
@defform[(run syn)]{
Like @racket[normalize/syn], but is a syntactic form which allows a Cur term to be written by
computing part of the term from another Cur term.

View File

@ -15,6 +15,9 @@
define-type
case
case*
let
;; reflection in syntax
run
step
step-n
@ -109,6 +112,25 @@
[(_ D U e (p ...) P clause* ...)
#`(elim D U P #,@(map rewrite-clause (syntax->list #'(clause* ...))) p ... e)]))
(begin-for-syntax
(define-syntax-class let-clause
(pattern
(~or (x:id e) ((x:id (~datum :) t) e))
#:attr id #'x
#:attr expr #'e
#:attr type (cond
[(attribute t) #'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)
(syntax-parse syn
[(let (c:let-clause ...) body)
#'((lambda* (c.id : c.type) ... body) c.e ...)]))
(define-syntax-rule (define-theorem name prop)
(define name prop))
@ -167,4 +189,33 @@
((λ* (x : (Type 1)) (y : ( (Type 1) (Type 1))) (y x))
Type
(λ (x : (Type 1)) x))
Type))
Type)
(check-equal?
(let ([x Type]
[y (λ (x : (Type 1)) x)])
(y x))
Type)
(check-equal?
(let ([(x : (Type 1)) Type]
[y (λ (x : (Type 1)) x)])
(y x))
Type)
;; check that raises decent syntax error
;; Can't use this because (lambda () ...) and thunk are not things in Cur at runtime
#;(check-exn
exn:fail:syntax?
(let ([x : (Type 1) Type]
[y (λ (x : (Type 1)) x)])
(y x)))
;; check that raises type error
#;(check-exn
exn:fail:syntax?
(let ([x uninferrable-expr]
[y (λ (x : (Type 1)) x)])
(y x)))
)