Added inductive types; assorted fixes

* Fixed variable typing; weakening is no longer a rule but built into
  lookup logic.
* Fixed substitution.
This commit is contained in:
William J. Bowman 2014-07-16 18:51:02 +02:00
parent 944e9690dc
commit 176f08dd92

View File

@ -1,10 +1,12 @@
#lang racket/base
(require
(only-in racket/set set=?)
(only-in racket curry)
redex/reduction-semantics)
#;(provide
define-inductive-type
define-constructor-for
match
define-fun
define-rec
@ -24,10 +26,19 @@
(x ::= variable-not-otherwise-mentioned)
;; TODO: Having 2 binders is stupid.
(v ::= (Π (x : t) t) (λ (x : t) t) x U)
(t e ::= v (t t)))
(t e ::= (case e (x e) (x e)...) v (t t)))
(module+ test
(require (except-in rackunit check))
(require rackunit)
(check-true (redex-match? dtracketL x (term T)))
(check-true (redex-match? dtracketL x (term truth)))
(check-true (redex-match? dtracketL x (term zero)))
(check-true (redex-match? dtracketL x (term s)))
(check-true (redex-match? dtracketL t (term zero)))
(check-true (redex-match? dtracketL t (term s)))
(check-true (redex-match? dtracketL t (term Type)))
(check-true (redex-match? dtracketL x (term nat)))
(check-true (redex-match? dtracketL t (term nat)))
(check-true (redex-match? dtracketL U (term Type)))
(check-true (redex-match? dtracketL U (term (Unv 0))))
(check-true (redex-match? dtracketL e (term (λ (x_0 : (Unv 0)) x_0))))
@ -70,12 +81,14 @@
(define-metafunction dtracketL
subst : t x t -> t
[(subst x x t) t]
[(subst x_0 x t) x]
[(subst x_0 x t) x_0]
[(subst (Π (x : t_0) t_1) x t) (Π (x : t_0) t_1)]
[(subst (λ (x : t_0) t_1) x t) (λ (x : t_0) t_1)]
[(subst (Π (x_0 : t_0) t_1) x t) (Π (x_0 : t_0) (subst t_1 x t))]
[(subst (λ (x_0 : t_0) t_1) x t) (λ (x_0 : t_0) (subst t_1 x t))]
[(subst (e_0 e_1) x t) ((subst e_0 x t) (subst e_1 x t))])
;; NB:
;; TODO: Why do I not have tests for substitutions?!?!?!?!?!?!?!!?!?!?!?!?!?!!??!?!
(define-extended-language dtracket-redL dtracketL
(E hole (E t) (λ (x : t) E) (Π (x : t) E)))
@ -107,7 +120,7 @@
;; http://www.cs.ox.ac.uk/ralf.hinze/WG2.8/31/slides/stephanie.pdf
(define-extended-language dtracket-typingL dtracketL
(Γ (Γ x : t)))
(Σ Γ ::= (Γ x : t)))
;; NB: Depends on clause order
(define-metafunction dtracket-typingL
@ -123,38 +136,139 @@
[(remove (Γ x : t) x) Γ]
[(remove (Γ x_0 : t_0) x_1) (remove Γ x_1)])
(define-metafunction dtracket-typingL
result-type : Σ t -> t or #f
[(result-type Σ (Π (x : t) e)) (result-type Σ e)]
[(result-type Σ x) ,(and (term (lookup Σ x)) (term x))]
[(result-type Σ t) #f])
(module+ test
(define Σ (term ((( nat : Type) zero : nat) s : (Π (x : nat) nat))))
(check-equal? (term nat) (term (result-type ,Σ (Π (x : nat) nat))))
(check-equal? (term nat) (term (result-type ,Σ nat))))
(define-judgment-form dtracket-typingL
#:mode (types I I O)
#:contract (types Γ e t)
#:mode (constructor-for I I O)
#:contract (constructor-for Σ t x)
[(where t_0 (result-type Σ t))
-------------
(constructor-for (Σ x : t) t_0 x)]
[(constructor-for Σ t_1 x)
-------------
(constructor-for (Σ x_0 : t_0) t_1 x)])
(module+ test
(check-true
(judgment-holds (constructor-for (( truth : Type) T : truth) truth T)))
(check-true
(judgment-holds
(constructor-for (( nat : Type) zero : nat)
nat zero)))
(check set=?
(judgment-holds
(constructor-for ((( nat : Type) zero : nat) s : (Π (x : nat) nat))
nat x) x)
(list (term zero) (term s))))
(define-metafunction dtracket-typingL
constructors-for : Σ t (x ...) -> #t or #f
[(constructors-for Σ t (x ...))
,((lambda (x y) (and (set=? x y) (= (length x) (length y))))
(term (x ...))
(judgment-holds (constructor-for Σ t x_00) x_00))])
(module+ test
(check-true
(term (constructors-for ((( nat : Type) zero : nat) s : (Π (x : nat) nat))
nat (zero s))))
(check-false
(term (constructors-for ((( nat : Type) zero : nat) s : (Π (x : nat) nat))
nat (zero)))))
(define-metafunction dtracketL
branch-type : t t t -> t
[(branch-type t_ind (Π (x_0 : t) e_0) (Π (x_1 : t) e_1))
(branch-type t_ind e_0 e_1)]
[(branch-type t_ind t_ind t) t])
(module+ test
(check-equal? (term Type) (term (branch-type nat (lookup ,Σ zero) Type)))
(check-equal? (term nat) (term (branch-type nat nat nat)))
(check-equal? (term Type) (term (branch-type nat (lookup ,Σ s) (Π (x : nat) Type)))))
(define-metafunction dtracket-typingL
branch-types-match : Σ (x ...) (t ...) t t -> #t or #f
[(branch-types-match Σ (x ...) (t_11 ...) t t_1)
,(andmap (curry equal? (term t)) (term ((branch-type t_1 (lookup Σ x) t_11) ...)))])
(module+ test
(check-true
(term (branch-types-match (( truth : Type) T : truth) () () Type nat)))
(check-true
(term (branch-types-match ,Σ (zero s) (Type (Π (x : nat) Type)) Type nat)))
(check-true
(term (branch-types-match ,Σ (zero s) (nat (Π (x : nat) nat)) nat nat))))
(define-judgment-form dtracket-typingL
#:mode (wf I I)
#:contract (wf Σ Γ)
[-----------------
(wf )]
[(types Σ Γ t U)
-----------------
(wf Σ (Γ x : t))]
[(types Σ t U)
-----------------
(wf (Σ x : t) )])
(module+ test
(check-true (judgment-holds (wf )))
(check-true (judgment-holds (wf ( truth : Type) )))
(check-true (judgment-holds (wf ( x : Type))))
(check-true (judgment-holds (wf ( nat : Type) ( x : nat))))
(check-true (judgment-holds (wf ( nat : Type) ( x : (Π (x : nat) nat))))))
(define-judgment-form dtracket-typingL
#:mode (types I I I O)
#:contract (types Σ Γ e t)
[(unv-ok U_0 U_1)
(wf Σ Γ)
----------------- "DTR-Axiom"
(types U_0 U_1)]
(types Σ Γ U_0 U_1)]
[(where t (lookup Σ x))
----------------- "DTR-Inductive"
(types Σ Γ x t)]
[(where t (lookup Γ x))
(types (remove Γ x) t U)
----------------- "DTR-Start"
(types Γ x t)]
(types Σ Γ x t)]
[(types Γ t t_1) (types Γ t_0 U)
----------------- "DTR-Weakening"
(types (Γ x : t_0) t t_1)]
[(types Γ t_0 U_1)
(types (Γ x : t_0) t U_2)
[(types Σ Γ t_0 U_1)
(types Σ (Γ x : t_0) t U_2)
(unv-kind U_1 U_2 U)
----------------- "DTR-Product"
(types Γ (Π (x : t_0) t) U)]
(types Σ Γ (Π (x : t_0) t) U)]
[(types Γ e_0 (Π (x : t_0) t_1))
(types Γ e_1 t_0)
[(types Σ Γ e_0 (Π (x : t_0) t_1))
(types Σ Γ e_1 t_0)
----------------- "DTR-Application"
(types Γ (e_0 e_1) (subst t_1 x e_1))]
(types Σ Γ (e_0 e_1) (subst t_1 x e_1))]
[(types (Γ x : t_0) e t_1)
(types Γ (Π (x : t_0) t_1) U)
[(types Σ (Γ x : t_0) e t_1)
(types Σ Γ (Π (x : t_0) t_1) U)
----------------- "DTR-Abstraction"
(types Γ (λ (x : t_0) e) (Π (x : t_0) t_1))]
(types Σ Γ (λ (x : t_0) e) (Π (x : t_0) t_1))]
[(types Σ Γ e t_1)
(side-condition (constructors-for Σ t_1 (x_0 x_1 ...)))
(types Σ Γ e_0 t_00)
(types Σ Γ e_1 t_11) ...
;; TODO Some of these meta-functions aren't very consistent in terms
;; of interface
(where t (branch-type t_1 (lookup Σ x_0) t_00))
(side-condition (branch-types-match Σ (x_1 ...) (t_11 ...) t t_1))
----------------- "DTR-Case"
(types Σ Γ (case e (x_0 e_0) (x_1 e_1) ...) t)]
;; TODO: beta-equiv
;; This rule is no good for algorithmic checking; Redex infinitly
@ -168,29 +282,71 @@
----------------- "DTR-Conversion"
(types Γ e t_1)])
(module+ test
(check-true (judgment-holds (types Type (Unv 0))))
(check-true (judgment-holds (types ( x : Type) Type (Unv 0))))
(check-true (judgment-holds (types ( x : Type) x Type)))
(check-true (judgment-holds (types (( x_0 : Type) x_1 : Type)
(check-true (judgment-holds (types Type (Unv 0))))
(check-true (judgment-holds (types ( x : Type) Type (Unv 0))))
(check-true (judgment-holds (types ( x : Type) x Type)))
(check-true (judgment-holds (types (( x_0 : Type) x_1 : Type)
(Π (x_3 : x_0) x_1) Type)))
(check-true (judgment-holds (types (λ (x : Type) x) (Π (x : Type) Type))))
(check-true (judgment-holds (types (λ (y : Type) (λ (x : y) x))
(Π (y : Type) (Π (x : y) y))))))
(check-true (judgment-holds (types (λ (x : Type) x) (Π (x : Type) Type))))
(check-true (judgment-holds (types (λ (y : Type) (λ (x : y) x))
(Π (y : Type) (Π (x : y) y)))))
(check-true
(judgment-holds (types (( truth : Type) T : truth)
T
truth)))
(check-true
(judgment-holds (types (( truth : Type) T : truth)
Type
(Unv 0))))
(check-true
(judgment-holds (types (( truth : Type) T : truth)
(case T (T Type))
(Unv 0))))
(check-false
(judgment-holds (types (( truth : Type) T : truth)
(case T (T Type) (T Type))
(Unv 0))))
(define-syntax-rule (nat-test syn ...)
(check-true (judgment-holds
(types ((( nat : Type) zero : nat) s : (Π (x : nat) nat))
syn ...))))
(nat-test (Π (x : nat) nat) Type)
(nat-test (λ (x : nat) x) (Π (x : nat) nat))
(nat-test (case zero (zero zero) (s (λ (x : nat) x)))
nat)
(nat-test nat Type)
(nat-test zero nat)
(nat-test s (Π (x : nat) nat))
(nat-test (s zero) nat)
(nat-test (case zero (zero (s zero)) (s (λ (x : nat) (s (s x)))))
nat)
(nat-test (case zero (zero (s zero)) (s (λ (x : nat) (s (s x)))))
nat)
(check-false (judgment-holds
(types ((( nat : Type) zero : nat) s : (Π (x : nat) nat))
(case zero (zero (s zero)))
nat))))
(define-judgment-form dtracket-typingL
#:mode (type-check I I I)
#:contract (type-check Γ e t)
[(types Γ e t)
[(types Σ e t)
---------------
(type-check Γ e (reduce t))])
(type-check Σ e (reduce t))])
;; Infer-core Language
;; A relaxed core where annotation are optional.
(define-extended-language dtracket-surfaceL dtracketL
(v ::= .... (λ x e) (Π t e))
(t e ::= .... (e : t)))
(t e ::= .... (data x (x : t) (x : t) ...) (case e ([e e] ...)) (e : t)))
;; http://www.cs.ox.ac.uk/ralf.hinze/WG2.8/31/slides/stephanie.pdf
#;(define-judgment-form dtracket-typingL