module Main import Data.Vect import Data.Fin data Ty = TyInt | TyBool| TyFun Ty Ty interpTy : Ty -> Type interpTy TyInt = Int interpTy TyBool = Bool interpTy (TyFun s t) = interpTy s -> interpTy t using (G : Vect n Ty) data Env : Vect n Ty -> Type where Nil : Env Nil (::) : interpTy a -> Env G -> Env (a :: G) data HasType : (i : Fin n) -> Vect n Ty -> Ty -> Type where Stop : HasType FZ (t :: G) t Pop : HasType k G t -> HasType (FS k) (u :: G) t lookup : HasType i G t -> Env G -> interpTy t lookup Stop (x :: xs) = x lookup (Pop k) (x :: xs) = lookup k xs data Expr : Vect n Ty -> Ty -> Type where Var : HasType i G t -> Expr G t Val : (x : Int) -> Expr G TyInt Lam : Expr (a :: G) t -> Expr G (TyFun a t) App : Expr G (TyFun a t) -> Expr G a -> Expr G t Op : (interpTy a -> interpTy b -> interpTy c) -> Expr G a -> Expr G b -> Expr G c If : Expr G TyBool -> Lazy (Expr G a) -> Lazy (Expr G a) -> Expr G a interp : Env G -> (e : Expr G t) -> interpTy t interp env (Var i) = lookup i env interp env (Val x) = x interp env (Lam sc) = \x => interp (x :: env) sc interp env (App f s) = interp env f (interp env s) interp env (Op op x y) = op (interp env x) (interp env y) interp env (If x t e) = if interp env x then interp env t else interp env e eId : Expr G (TyFun TyInt TyInt) eId = Lam (Var Stop) eAdd : Expr G (TyFun TyInt (TyFun TyInt TyInt)) eAdd = Lam (Lam (Op (+) (Var Stop) (Var (Pop Stop)))) eEq : Expr G (TyFun TyInt (TyFun TyInt TyBool)) eEq = Lam (Lam (Op (==) (Var Stop) (Var (Pop Stop)))) eDouble : Expr G (TyFun TyInt TyInt) eDouble = Lam (App (App eAdd (Var Stop)) (Var Stop)) fact : Expr G (TyFun TyInt TyInt) fact = Lam (If (Op (==) (Var Stop) (Val 0)) (Val 1) (Op (*) (App fact (Op (-) (Var Stop) (Val 1))) (Var Stop))) testFac : Int testFac = interp [] fact 4 -- unitTestFac : so (interp [] fact 4 == 24) -- unitTestFac = oh main : IO () main = do putStr "Enter a number: " x <- getLine printLn (interp [] fact (cast x))