Compare commits

...

23 Commits

Author SHA1 Message Date
William J. Bowman
8429afbfcc Fixed latex output 2015-04-14 19:14:50 -04:00
William J. Bowman
a0803eb8b1 Fixed Coq generator to use eliminators
Not sure the resulting Coq code will actually type check, due to
differences in the types of Cur eliminators and Coq eliminators.
2015-04-14 18:58:54 -04:00
William J. Bowman
3be6730b1e Merge branch 'names' into actual-inductives
Conflicts:
	redex-curnel.rkt

With names finally fixed, many more tests can actually run.
* Changed uses of case to eliminators
* Fixed uses of new case* macro
* Fixed some assorted bugs in reduction of eliminators
2015-04-14 18:42:01 -04:00
William J. Bowman
f8a51e65ca Attempting to fix case typing checking
Maybe I shouldn't bother and should fix that after fixing inductives
2015-04-10 22:42:51 -04:00
William J. Bowman
9723870f14 Added capture-avoiding subst, α-equivalence
* Added capture-avoiding substitution
* Added equivalence during typing checking, including α-equivalence and
  limited β-equivalence
* Exposed better typing-check reflection features to allow typing
  checking modulo equivalence
* Tweaked QED macro to use new type-checking reflection feature.
* Fixed some test cases
2015-04-10 22:05:54 -04:00
William J. Bowman
b4090ae2cf Ahhh fucking substitution
I'm dumb
2015-03-27 21:49:03 -04:00
William J. Bowman
c4f0f723f5 Dynamic semantics for elim
elim now runs. Parts of the stdlib have been converted to use elim, but
it's complicated to program with. Need to implement case and fix in
terms of elim, I think.
2015-03-27 21:16:20 -04:00
William J. Bowman
aafd360541 Broken commit: work on dynamic semantics of elim 2015-03-26 17:46:59 -04:00
William J. Bowman
2928969691 A little clean up 2015-03-25 21:48:21 -04:00
William J. Bowman
a4862f3006 Elimination of false works! 2015-03-25 21:26:08 -04:00
William J. Bowman
7e489d52c8 Fixed typing inductive hypotheses
Fixed type checking of inductive hypotheses for elimination of inductive
types. Almost all elim tests type check. elimination of false still does
not.
2015-03-25 21:05:31 -04:00
William J. Bowman
0caeed6080 Cleaned up comments, added refernces 2015-03-25 20:29:00 -04:00
William J. Bowman
3304ed8531 First attempt at typing elim
elim now has a typing rule that works for some uses of elim, like on
"nat". However, eliminating things inductives such as "and" does not yet
work due to issue in typing methods.
2015-03-25 20:19:43 -04:00
William J. Bowman
90ba703d6f Adding well-formed constructor checking
* Added work to ensure inductive constructors actually return the type
  of the inductive they are defining. However, this currently fails for
  nats. Not sure why
2015-03-24 22:07:16 -04:00
William J. Bowman
f7ddeae5bc Fixed well-formedness checking for contexts 2015-03-24 20:36:09 -04:00
William J. Bowman
f5d387b689 Merge branch 'master' into actual-inductives 2015-03-24 20:14:00 -04:00
William J. Bowman
f4eeb3821a Merge branch 'master' into actual-inductives
Conflicts:
	redex-curnel.rkt
2015-03-24 20:12:22 -04:00
William J. Bowman
7641ee3d1c Sketch of elim semantics, got all tests running 2015-03-24 20:09:35 -04:00
William J. Bowman
97c1d144aa A complete sketch of typing for elim-D (case) 2015-03-23 19:29:56 -04:00
William J. Bowman
8a5bff1aec Merge branch 'master' into actual-inductives
Conflicts:
	redex-curnel.rkt
2015-03-23 17:37:04 -04:00
William J. Bowman
b185150195 More thoughts 2015-03-20 17:16:38 -04:00
William J. Bowman
2718aaec42 Musing about the correct implementation 2015-02-17 23:55:12 -05:00
William J. Bowman
e15b102a7f Starting to try to fix inductives 2015-02-17 19:09:56 -05:00
9 changed files with 902 additions and 479 deletions

68
oll.rkt
View File

@ -53,14 +53,14 @@
(when (attribute latex-file) (when (attribute latex-file)
(with-output-to-file (syntax->datum #'latex-file) (with-output-to-file (syntax->datum #'latex-file)
(thunk (thunk
(format "\\fbox{$~a$}$~n$\\begin{mathpar}~n~a~n\end{mathpar}$$" (printf (format "\\fbox{$~a$}$~n$\\begin{mathpar}~n~a~n\\end{mathpar}$$"
(syntax->datum #'(n types* ...)) (syntax->datum #'(n types* ...))
(string-trim (string-trim
(for/fold ([str ""]) (for/fold ([str ""])
([rule (attribute rules.latex)]) ([rule (attribute rules.latex)])
(format "~a~a\\and~n" str rule)) (format "~a~a\\and~n" str rule))
"\\and" "\\and"
#:left? #f))) #:left? #f))))
#:exists 'append)) #:exists 'append))
#`(begin #`(begin
#,@(if (attribute coq-file) #,@(if (attribute coq-file)
@ -187,7 +187,7 @@
#:left? #f))])))) #:left? #f))]))))
(define (generate-latex-bnf file-name vars clauses) (define (generate-latex-bnf file-name vars clauses)
(with-output-to-file file-name (with-output-to-file file-name
(thunk (output-latex-bnf vars clauses)) (thunk (printf (output-latex-bnf vars clauses)))
#:exists 'append))) #:exists 'append)))
(module+ test (module+ test
(require "stdlib/sugar.rkt") (require "stdlib/sugar.rkt")
@ -231,11 +231,19 @@
(data var : Type (avar : (-> nat var))) (data var : Type (avar : (-> nat var)))
(define (var-equal? (v1 : var) (v2 : var)) (define (var-equal? (v1 : var) (v2 : var))
(case* v1 (case* var v1 (lambda* (v : var) bool)
[(avar (n1 : nat)) [(avar (n1 : nat)) IH: ()
(case* v2 (case* var v2 (lambda* (v : var) bool)
[(avar (n2 : nat)) [(avar (n2 : nat)) IH: ()
(nat-equal? n1 n2)])])) (nat-equal? n1 n2)])]))
(module+ test
(require rackunit)
(check-equal?
(var-equal? (avar z) (avar z))
btrue)
(check-equal?
(var-equal? (avar z) (avar (s z)))
bfalse))
;; See stlc.rkt for examples ;; See stlc.rkt for examples
@ -263,7 +271,7 @@
#'begin) #'begin)
;; TODO: Need to add these to a literal set and export it ;; TODO: Need to add these to a literal set and export it
;; Or, maybe overwrite syntax-parse ;; Or, maybe overwrite syntax-parse
#:literals (lambda forall data real-app case define-theorem #:literals (lambda forall data real-app elim define-theorem
define qed begin Type) define qed begin Type)
[(begin e ...) [(begin e ...)
(for/fold ([str ""]) (for/fold ([str ""])
@ -303,7 +311,7 @@
(output-coq #'body))) (output-coq #'body)))
"")] "")]
[(lambda ~! (x:id (~datum :) t) body:expr) [(lambda ~! (x:id (~datum :) t) body:expr)
(format "(fun ~a : ~a => ~a)" (syntax-e #'x) (output-coq #'t) (format "(fun ~a : ~a => ~a)" (output-coq #'x) (output-coq #'t)
(output-coq #'body))] (output-coq #'body))]
[(forall ~! (x:id (~datum :) t) body:expr) [(forall ~! (x:id (~datum :) t) body:expr)
(format "(forall ~a : ~a, ~a)" (syntax-e #'x) (output-coq #'t) (format "(forall ~a : ~a, ~a)" (syntax-e #'x) (output-coq #'t)
@ -322,21 +330,14 @@
(output-coq #'t))])))) (output-coq #'t))]))))
"")] "")]
[(Type i) "Type"] [(Type i) "Type"]
[(case e (ec eb) ...) [(elim var e P m ...)
(format "(match ~a with~n~aend)" (format "(~a_rect ~a~a ~a)"
(output-coq #'e) (output-coq #'var)
(for/fold ([strs ""]) (output-coq #'P)
([con (syntax->list #'(ec ...))] (for/fold ([x ""])
[body (syntax->list #'(eb ...))]) ([m (syntax->list #'(m ...))])
(let* ([ids (generate-temporaries (constructor-args con))] (format "~a ~a" x (output-coq m)))
[names (map (compose ~a syntax->datum) ids)]) (output-coq #'e))]
(format "~a| (~a) => ~a~n" strs
(for/fold ([str (output-coq con)])
([n names])
(format "~a ~a" str n))
(for/fold ([body (output-coq body)])
([n names])
(format "(~a ~a)" body n))))))]
[(real-app e1 e2) [(real-app e1 e2)
(format "(~a ~a)" (output-coq #'e1) (output-coq #'e2))] (format "(~a ~a)" (output-coq #'e1) (output-coq #'e2))]
[e:id (sanitize-id (format "~a" (syntax->datum #'e)))]))) [e:id (sanitize-id (format "~a" (syntax->datum #'e)))])))
@ -377,14 +378,15 @@
(meow g e t)])) (meow g e t)]))
(coq-defns))]) (coq-defns))])
(check-regexp-match (check-regexp-match
"Inductive meow : \\(forall temp. : gamma, \\(forall temp. : term, \\(forall temp. : type, Type\\)\\)\\) :=" "Inductive meow : \\(forall .+ : gamma, \\(forall .+ : term, \\(forall .+ : type, Type\\)\\)\\) :="
(first (string-split t "\n"))) (first (string-split t "\n")))
(check-regexp-match (check-regexp-match
"\\| T-Bla : \\(forall g : gamma, \\(forall e : term, \\(forall t : type, \\(\\(\\(meow g\\) e\\) t\\)\\)\\)\\)\\." "\\| T-Bla : \\(forall g : gamma, \\(forall e : term, \\(forall t : type, \\(\\(\\(meow g\\) e\\) t\\)\\)\\)\\)\\."
(second (string-split t "\n")))) (second (string-split t "\n"))))
(let ([t (output-coq #'(case z (z z) (s (lambda (n : nat) (s n)))))]) (let ([t (output-coq #'(elim nat e (lambda (x : nat) nat) z
(lambda* (x : nat) (ih-x : nat) ih-x)))])
(check-regexp-match (check-regexp-match
"\\(match z with\n\\| \\(z\\) => z\n\\| \\(s .+\\) => \\(\\(fun n : nat => \\(s n\\)\\) .+\\)\nend\\)" "\\(nat_rect \\(fun x : nat => nat\\) z \\(fun x : nat => \\(fun ih_x : nat => ih_x\\)\\) e\\)"
t)) t))
(check-regexp-match (check-regexp-match
"Theorem thm_plus_commutes : \\(forall n : nat, \\(forall m : nat, \\(\\(\\(== nat\\) \\(\\(plus n\\) m\\)\\) \\(\\(plus m\\) n\\)\\)\\)\\).\n" "Theorem thm_plus_commutes : \\(forall n : nat, \\(forall m : nat, \\(\\(\\(== nat\\) \\(\\(plus n\\) m\\)\\) \\(\\(plus m\\) n\\)\\)\\)\\).\n"

View File

@ -77,6 +77,7 @@
(define (CPSf-relation (f1 : CPSf) (f2 : CPSf)) (define (CPSf-relation (f1 : CPSf) (f2 : CPSf))
;; TODO: Fix run so I can simply use (run CPSf) to perform ;; TODO: Fix run so I can simply use (run CPSf) to perform
;; substitution ;; substitution
(translate (run CPSf))
(translate (forall* (ans : Type) (k : (-> X ans)) ans))) (translate (forall* (ans : Type) (k : (-> X ans)) ans)))
#;(module+ test #;(module+ test
(require rackunit) (require rackunit)

File diff suppressed because it is too large Load Diff

View File

@ -8,9 +8,10 @@
(define-syntax (if syn) (define-syntax (if syn)
(syntax-case syn () (syntax-case syn ()
[(_ t s f) [(_ t s f)
#'(case t ;; Compute the motive
[btrue s] (let ([M #`(lambda (x : #,(type-infer/syn #'t))
[bfalse f])])) #,(type-infer/syn #'s))])
(quasisyntax/loc syn (elim bool t #,M s f)))]))
(define (bnot (x : bool)) (if x bfalse btrue)) (define (bnot (x : bool)) (if x bfalse btrue))
(module+ test (module+ test

View File

@ -8,9 +8,12 @@
(module+ test (module+ test
(require rackunit "bool.rkt") (require rackunit "bool.rkt")
;; TODO: Dependent pattern matching doesn't work yet #;(check-equal?
#;(check-equal? (case* (some bool btrue) (case* maybe (some bool btrue)
[(none (A : Type)) bfalse] (lambda (x : (maybe bool)) bool)
[(some (A : Type) (x : bool)) [(none (A : Type)) IH: ()
(if x btrue bfalse)]) bfalse]
btrue)) [(some (A : Type) (x : A)) IH: ()
;; TODO: Don't know how to use dependency yet
(if x btrue bfalse)])
btrue))

View File

@ -2,7 +2,7 @@
(require "sugar.rkt" "bool.rkt") (require "sugar.rkt" "bool.rkt")
;; TODO: override (all-defined-out) to enable exporting all these ;; TODO: override (all-defined-out) to enable exporting all these
;; properly. ;; properly.
(provide nat z s add1 sub1 plus nat-equal?) (provide nat z s add1 sub1 plus )
(module+ test (module+ test
(require rackunit)) (require rackunit))
@ -15,33 +15,37 @@
(check-equal? (add1 (s z)) (s (s z)))) (check-equal? (add1 (s z)) (s (s z))))
(define (sub1 (n : nat)) (define (sub1 (n : nat))
(case n (case* nat n (lambda (x : nat) nat)
[z z] [z z]
[s (lambda (x : nat) x)])) [(s (x : nat)) IH: ((ih-n : nat)) x]))
(module+ test (module+ test
(check-equal? (sub1 (s z)) z)) (check-equal? (sub1 (s z)) z))
(define plus (define (plus (n1 : nat) (n2 : nat))
(fix (plus : (forall* (n1 : nat) (n2 : nat) nat)) (case* nat n1 (lambda (x : nat) nat)
(lambda (n1 : nat) [z n2]
(lambda (n2 : nat) [(s (x : nat)) IH: ((ih-n1 : nat))
(case n1 (s ih-n1)]))
[z n2]
[s (λ (x : nat) (plus x (s n2)))])))))
(module+ test (module+ test
(check-equal? (plus z z) z) (check-equal? (plus z z) z)
(check-equal? (plus (s (s z)) (s (s z))) (s (s (s (s z)))))) (check-equal? (plus (s (s z)) (s (s z))) (s (s (s (s z))))))
(define-rec (nat-equal? (n1 : nat) (n2 : nat) : bool) ;; Credit to this function goes to Max
(case* n1 (define (nat-equal? (n1 : nat))
[z (case* n2 (elim nat n1 (lambda (x : nat) (-> nat bool))
[z btrue] (lambda (n2 : nat)
[(s (n3 : nat)) bfalse])] (elim nat n2 (lambda (x : nat) bool)
[(s (n3 : nat)) btrue
(case* n2 (lambda* (x : nat) (ih-n2 : bool) bfalse)))
[z btrue] (lambda* (x : nat) (ih : (-> nat bool))
[(s (n4 : nat)) (nat-equal? n3 n4)])])) (lambda (n2 : nat)
(elim nat n2 (lambda (x : nat) bool)
bfalse
(lambda* (x : nat) (ih-bla : bool)
(ih x)))))))
(module+ test (module+ test
(check-equal? btrue (nat-equal? z z)) (check-equal? (nat-equal? z z) btrue)
(check-equal? bfalse (nat-equal? z (s z))) (check-equal? (nat-equal? z (s z)) bfalse)
(check-equal? btrue (nat-equal? (s z) (s z)))) (check-equal? (nat-equal? (s z) (s z)) btrue))

View File

@ -18,7 +18,7 @@
(define-theorem thm:anything-implies-true (forall (P : Type) true)) (define-theorem thm:anything-implies-true (forall (P : Type) true))
(qed (run thm:anything-implies-true) (lambda (P : Type) T)) (qed thm:anything-implies-true (lambda (P : Type) T))
(data false : Type) (data false : Type)
@ -31,20 +31,23 @@
(define-theorem thm:and-is-symmetric (define-theorem thm:and-is-symmetric
(forall* (P : Type) (Q : Type) (ab : (and P Q)) (and Q P))) (forall* (P : Type) (Q : Type) (ab : (and P Q)) (and Q P)))
;; TODO: BAH! pattern matching on inductive families is still broken.
(define proof:and-is-symmetric (define proof:and-is-symmetric
(lambda* (P : Type) (Q : Type) (ab : (and P Q)) (lambda* (P : Type) (Q : Type) (ab : (and P Q))
(case* ab (case* and ab
((conj (P : Type) (Q : Type) (x : P) (y : Q)) (conj Q P y x))))) (lambda* (P : Type) (Q : Type) (ab : (and P Q))
(and Q P))
((conj (P : Type) (Q : Type) (x : P) (y : Q)) IH: () (conj Q P y x)))))
#;(qed thm:and-is-symmetric proof:and-is-symmetric) (qed thm:and-is-symmetric proof:and-is-symmetric)
(define-theorem thm:proj1 (define-theorem thm:proj1
(forall* (A : Type) (B : Type) (c : (and A B)) A)) (forall* (A : Type) (B : Type) (c : (and A B)) A))
(define proof:proj1 (define proof:proj1
(lambda* (A : Type) (B : Type) (c : (and A B)) (lambda* (A : Type) (B : Type) (c : (and A B))
(case c (conj (lambda* (A : Type) (B : Type) (a : A) (b : B) a))))) (case* and c
(lambda* (A : Type) (B : Type) (c : (and A B)) A)
((conj (A : Type) (B : Type) (a : A) (b : B)) IH: () a))))
(qed thm:proj1 proof:proj1) (qed thm:proj1 proof:proj1)
@ -53,7 +56,9 @@
(define proof:proj2 (define proof:proj2
(lambda* (A : Type) (B : Type) (c : (and A B)) (lambda* (A : Type) (B : Type) (c : (and A B))
(case c (conj (lambda* (A : Type) (B : Type) (a : A) (b : B) b))))) (case* and c
(lambda* (A : Type) (B : Type) (c : (and A B)) B)
((conj (A : Type) (B : Type) (a : A) (b : B)) IH: () b))))
(qed thm:proj2 proof:proj2) (qed thm:proj2 proof:proj2)

View File

@ -66,22 +66,30 @@
[(_ (name (a : t) ... : t_res) body) [(_ (name (a : t) ... : t_res) body)
#'(define name (fix (name : (forall* (a : t) ... t_res)) #'(define name (fix (name : (forall* (a : t) ... t_res))
(lambda* (a : t) ... body)))])) (lambda* (a : t) ... body)))]))
(begin-for-syntax (begin-for-syntax
(define (rewrite-clause clause) (define (rewrite-clause clause)
(syntax-case clause (:) (syntax-case clause (: IH:)
[((con (a : A) ...) body) #'(con (lambda* (a : A) ... body))] [((con (a : A) ...) IH: ((x : t) ...) body)
[(e body) #'(e body)]))) #'(lambda* (a : A) ... (x : t) ... body)]
[(e body) #'body])))
;; TODO: Expects clauses in same order as constructors as specified when
;; TODO: inductive D is defined.
(define-syntax (case* syn) (define-syntax (case* syn)
(syntax-case syn () (syntax-case syn ()
[(_ e clause* ...) [(_ D e P clause* ...)
#`(case e #,@(map rewrite-clause (syntax->list #'(clause* ...))))])) #`(elim D e P #,@(map rewrite-clause (syntax->list #'(clause* ...))))]))
(define-syntax-rule (define-theorem name prop) (define-syntax-rule (define-theorem name prop)
(define name prop)) (define name prop))
;; TODO: Current implementation doesn't do beta/eta while type-checking (define-syntax (qed stx)
;; because reasons. So manually insert a run in the type annotation (syntax-case stx ()
(define-syntax-rule (qed thm pf) [(_ t pf)
((lambda (x : (run thm)) Type) pf)) (begin
(unless (type-check/syn? #'pf #'t)
(raise-syntax-error 'qed "Invalid proof"
#'pf #'t))
#'pf)]))

View File

@ -17,13 +17,14 @@
(emp-gamma : gamma) (emp-gamma : gamma)
(extend-gamma : (->* gamma var stlc-type gamma))) (extend-gamma : (->* gamma var stlc-type gamma)))
(define-rec (lookup-gamma (g : gamma) (x : var) : (maybe stlc-type)) (define (lookup-gamma (g : gamma) (x : var))
(case* g (case* gamma g (lambda* (g : gamma) (maybe stlc-type))
[emp-gamma (none stlc-type)] [emp-gamma (none stlc-type)]
[(extend-gamma (g1 : gamma) (v1 : var) (t1 : stlc-type)) [(extend-gamma (g1 : gamma) (v1 : var) (t1 : stlc-type))
IH: ((ih-g1 : (maybe stlc-type)))
(if (var-equal? v1 x) (if (var-equal? v1 x)
(some stlc-type t1) (some stlc-type t1)
(lookup-gamma g1 x))])) ih-g1)]))
(define-relation (has-type gamma stlc-term stlc-type) (define-relation (has-type gamma stlc-term stlc-type)
#:output-coq "stlc.v" #:output-coq "stlc.v"