[Untested] Fixed advanced version of match.

Need to start testing/converting stdlib, replacing case and case*
This commit is contained in:
William J. Bowman 2016-01-15 18:29:10 -05:00
parent 6820c07cd1
commit ceb2a1aefc
No known key found for this signature in database
GPG Key ID: DDD48D26958F0D1A
2 changed files with 46 additions and 29 deletions

View File

@ -35,17 +35,21 @@
(define square (run (exp (s (s z)))))
;; Credit to this function goes to Max
(define nat-equal?
(elim Nat Type (lambda (x : Nat) (-> Nat Bool))
(elim Nat Type (lambda (x : Nat) Bool)
true
(lambda (x : Nat) (ih-n2 : Bool) false))
(lambda (x : Nat) (ih : (-> Nat Bool))
(elim Nat Type (lambda (x : Nat) Bool)
false
(lambda (x : Nat) (ih-bla : Bool)
(ih x))))))
(define (zero? (n : Nat))
(match n
[z true]
[(s (n : Nat))
false]))
(define (nat-equal? (n : Nat))
(match n
[z zero?]
[(s (n-1 : Nat))
(lambda (m : Nat)
(match m
[z false]
[(s (m-1 : Nat))
((recur n-1) m-1)]))]))
(define (even? (n : Nat))
(match n

View File

@ -157,7 +157,7 @@
#:attr indices
'()
#:attr decls
'(#`(#,(gensym) : x)))
(list #`(#,(gensym) : x)))
(pattern
(x:id e:expr ...)
@ -173,15 +173,17 @@
(for/list ([e (attribute e)])
(type-infer/syn e))
#:attr decls
`(,@(for/list ([name (attribute names)]
[type (attribute types)])
#`(name : type))
#`(#,(gensym) : ((x e ...) #,@(attribute names))))))
(append
(for/list ([name (attribute names)]
[type (attribute types)])
#`(name : type))
;; TODO: quasisyntax/loc
(list #`(#,(gensym) : ((x e ...) #,@(attribute names)))))))
;; todo: Support just names, inferring types
(define-syntax-class match-declaration
(pattern
;; TODO: Use parameter-declaration defined earlier
(name:id (~datum :) type:expr)
#:attr decl
#'(name : type)))
@ -202,7 +204,10 @@
(pattern
(x:id d:match-declaration ...)
#:attr local-env
(apply dict-set* (make-immutable-hash) (attribute d.name) (attribute d.type))
(for/fold ([d (make-immutable-hash)])
([name (attribute d.name)]
[type (attribute d.type)])
(dict-set d name type))
#:attr decls
(attribute d.decl)
#:attr names
@ -221,9 +226,10 @@
[name-syn (attribute d.names)]
;; NB: Non-hygenic
#:when (cur-equal? type-syn D))
(let ([ih-name (format-id name-syn "ih-~a" name-syn)]
(let ([ih-name (quasisyntax/loc d #,(format-id name-syn "ih-~a" name-syn))]
;; BUG TODO: This is broken; motive must be applied to indices and such, too
[ih-type #`(#,motive #,name-syn)])
(dict-set! ih-dict (syntax-e name-syn) ih-name)
(dict-set! ih-dict (syntax->datum name-syn) ih-name)
(append decls (list #`(#,ih-name : #,ih-type)))))))
(define-syntax-class (match-preclause maybe-return-type)
@ -231,16 +237,21 @@
(p:match-prepattern b:expr)
#:attr return-type
;; TODO: Check that the infered type matches maybe-return-type, if it is provied
(or maybe-return-type (type-infer/syn #:local-env (attribute p.local-env) #'b))))
(or maybe-return-type
;; Ignore errors when trying to infer this type; other attempt might succeed
(with-handlers ([values (lambda _ #f)])
(type-infer/syn #:local-env (attribute p.local-env) #'b)))))
(define-syntax-class (match-clause src D motive)
(define-syntax-class (match-clause D motive)
(pattern
((~var p (match-pattern D motive))
;; TODO: nothing more advanced?
b:expr)
#:attr method
(quasisyntax/loc src
(lambda #,@(attribute p.decls) b)))))
(quasisyntax/loc #'p
#,(if (null? (attribute p.decls))
#'b
#`(lambda #,@(attribute p.decls) b))))))
(define-syntax (recur syn)
(syntax-case syn ()
@ -254,6 +265,7 @@
;; TODO: Detect when inside a match to provide better error
(format
"Cannot recur on ~a. Ether not inside a match or ~a is not an inductive argument."
(syntax->datum #'id)
(syntax->datum #'id))
syn)))]))
@ -261,13 +273,14 @@
(syntax-parse syn
[(_ d:expr
(~optional
(~seq #:in t:inductive-type-declaration)
(~seq #:in t)
#:defaults
([t (or (type-infer/syn #'d)
(raise-syntax-error
'match
"Could not infer discrimnant's type. Try using #:in to declare it."
syn))]))
(~parse D:inductive-type-declaration (attribute t))
(~optional (~seq #:return maybe-return-type:expr))
(~peek (~seq (~var prec (match-preclause (attribute maybe-return-type))) ...))
(~bind (return-type (ormap values (attribute prec.return-type))))
@ -277,13 +290,13 @@
"Could not infer return type. Try using #:return to declare it."
syn)))
(~bind (motive (quasisyntax/loc syn
(lambda #,@(attribute t.decls) return-type))))
(~var c (match-clause syn (attribute t) (attribute motive))) ...)
(lambda #,@(attribute D.decls) return-type))))
(~var c (match-clause (attribute D) (attribute motive))) ...)
(quasisyntax/loc syn
((elim t.inductive-name #,(type-infer/syn (attribute return-type)))
(elim D.inductive-name #,(type-infer/syn (attribute return-type))
motive
c.method ...
#,@(attribute t.indices)
#,@(attribute D.indices)
d))]))
(begin-for-syntax