{-# LANGUAGE MultiParamTypeClasses, FlexibleInstances, DeriveFunctor, TypeSynonymInstances, PatternGuards #-} module Idris.AbsSyntaxTree where import Core.TT import Core.Evaluate import Core.Elaborate hiding (Tactic(..)) import Core.Typecheck import IRTS.Lang import IRTS.CodegenCommon import Util.Pretty import Util.DynamicLinker import Paths_idris import System.Console.Haskeline import Control.Monad.Trans.State.Strict import Data.List import Data.Char import Data.Either import Debug.Trace data IOption = IOption { opt_logLevel :: Int, opt_typecase :: Bool, opt_typeintype :: Bool, opt_coverage :: Bool, opt_showimp :: Bool, opt_errContext :: Bool, opt_repl :: Bool, opt_verbose :: Bool, opt_quiet :: Bool, opt_codegen :: Codegen, opt_outputTy :: OutputType, opt_ibcsubdir :: FilePath, opt_importdirs :: [FilePath], opt_triple :: String, opt_cpu :: String, opt_optLevel :: Int, opt_cmdline :: [Opt] -- remember whole command line } deriving (Show, Eq) defaultOpts = IOption { opt_logLevel = 0 , opt_typecase = False , opt_typeintype = False , opt_coverage = True , opt_showimp = False , opt_errContext = False , opt_repl = True , opt_verbose = True , opt_quiet = False , opt_codegen = ViaC , opt_outputTy = Executable , opt_ibcsubdir = "" , opt_importdirs = [] , opt_triple = "" , opt_cpu = "" , opt_optLevel = 2 , opt_cmdline = [] } data LanguageExt = TypeProviders deriving (Show, Eq, Read, Ord) -- | The output mode in use data OutputMode = RawOutput | IdeSlave Integer deriving Show -- TODO: Add 'module data' to IState, which can be saved out and reloaded quickly (i.e -- without typechecking). -- This will include all the functions and data declarations, plus fixity declarations -- and syntax macros. -- | The global state used in the Idris monad data IState = IState { tt_ctxt :: Context, -- ^ All the currently defined names and their terms idris_constraints :: [(UConstraint, FC)], -- ^ A list of universe constraints and their corresponding source locations idris_infixes :: [FixDecl], -- ^ Currently defined infix operators idris_implicits :: Ctxt [PArg], idris_statics :: Ctxt [Bool], idris_classes :: Ctxt ClassInfo, idris_dsls :: Ctxt DSL, idris_optimisation :: Ctxt OptInfo, idris_datatypes :: Ctxt TypeInfo, idris_patdefs :: Ctxt ([([Name], Term, Term)], [PTerm]), -- not exported -- ^ list of lhs/rhs, and a list of missing clauses idris_flags :: Ctxt [FnOpt], idris_callgraph :: Ctxt CGInfo, -- name, args used in each pos idris_calledgraph :: Ctxt [Name], idris_docstrings :: Ctxt String, idris_totcheck :: [(FC, Name)], idris_log :: String, idris_options :: IOption, idris_name :: Int, idris_metavars :: [Name], -- ^ The currently defined but not proven metavariables idris_coercions :: [Name], idris_transforms :: [(Term, Term)], syntax_rules :: [Syntax], syntax_keywords :: [String], imported :: [FilePath], -- ^ The imported modules idris_scprims :: [(Name, (Int, PrimFn))], idris_objs :: [(Codegen, FilePath)], idris_libs :: [(Codegen, String)], idris_hdrs :: [(Codegen, String)], proof_list :: [(Name, [String])], errLine :: Maybe Int, lastParse :: Maybe Name, indent_stack :: [Int], brace_stack :: [Maybe Int], hide_list :: [(Name, Maybe Accessibility)], default_access :: Accessibility, default_total :: Bool, ibc_write :: [IBCWrite], compiled_so :: Maybe String, idris_dynamic_libs :: [DynamicLib], idris_language_extensions :: [LanguageExt], idris_outputmode :: OutputMode } data SizeChange = Smaller | Same | Bigger | Unknown deriving (Show, Eq) {-! deriving instance Binary SizeChange !-} type SCGEntry = (Name, [Maybe (Int, SizeChange)]) data CGInfo = CGInfo { argsdef :: [Name], calls :: [(Name, [[Name]])], scg :: [SCGEntry], argsused :: [Name], unusedpos :: [Int] } deriving Show {-! deriving instance Binary CGInfo !-} primDefs = [UN "unsafePerformIO", UN "mkLazyForeign", UN "mkForeign", UN "FalseElim"] -- information that needs writing for the current module's .ibc file data IBCWrite = IBCFix FixDecl | IBCImp Name | IBCStatic Name | IBCClass Name | IBCInstance Bool Name Name | IBCDSL Name | IBCData Name | IBCOpt Name | IBCSyntax Syntax | IBCKeyword String | IBCImport FilePath | IBCObj Codegen FilePath | IBCLib Codegen String | IBCDyLib String | IBCHeader Codegen String | IBCAccess Name Accessibility | IBCTotal Name Totality | IBCFlags Name [FnOpt] | IBCTrans (Term, Term) | IBCCG Name | IBCDoc Name | IBCCoercion Name | IBCDef Name -- i.e. main context deriving Show idrisInit = IState initContext [] [] emptyContext emptyContext emptyContext emptyContext emptyContext emptyContext emptyContext emptyContext emptyContext emptyContext emptyContext [] "" defaultOpts 6 [] [] [] [] [] [] [] [] [] [] [] Nothing Nothing [] [] [] Hidden False [] Nothing [] [] RawOutput -- | The monad for the main REPL - reading and processing files and updating -- global state (hence the IO inner monad). --type Idris = WriterT [Either String (IO ())] (State IState a)) type Idris = StateT IState IO -- Commands in the REPL data Codegen = ViaC | ViaJava | ViaNode | ViaJavaScript | ViaLLVM | Bytecode deriving (Show, Eq) -- | REPL commands data Command = Quit | Help | Eval PTerm | Check PTerm | DocStr Name | TotCheck Name | Reload | Load FilePath | ChangeDirectory FilePath | ModImport String | Edit | Compile Codegen String | Execute | ExecVal PTerm | NewCompile String | Metavars | Prove Name | AddProof (Maybe Name) | RmProof Name | ShowProof Name | Proofs | Universes | TTShell | LogLvl Int | Spec PTerm | HNF PTerm | Defn Name | Info Name | Missing Name | DynamicLink FilePath | ListDynamic | Pattelab PTerm | DebugInfo Name | Search PTerm | SetOpt Opt | UnsetOpt Opt | NOP data Opt = Filename String | Ver | Usage | Quiet | Ideslave | ShowLibs | ShowLibdir | ShowIncs | NoPrelude | NoREPL | OLogging Int | Output String | NewOutput String | TypeCase | TypeInType | DefaultTotal | DefaultPartial | WarnPartial | NoCoverage | ErrContext | ShowImpl | Verbose | IBCSubDir String | ImportDir String | PkgBuild String | PkgInstall String | PkgClean String | WarnOnly | Pkg String | BCAsm String | DumpDefun String | DumpCases String | FOVM String | UseCodegen Codegen | OutputTy OutputType | Extension LanguageExt | InterpretScript String | TargetTriple String | TargetCPU String | OptLevel Int deriving (Show, Eq) -- Parsed declarations data Fixity = Infixl { prec :: Int } | Infixr { prec :: Int } | InfixN { prec :: Int } | PrefixN { prec :: Int } deriving Eq {-! deriving instance Binary Fixity !-} instance Show Fixity where show (Infixl i) = "infixl " ++ show i show (Infixr i) = "infixr " ++ show i show (InfixN i) = "infix " ++ show i show (PrefixN i) = "prefix " ++ show i data FixDecl = Fix Fixity String deriving Eq instance Show FixDecl where show (Fix f s) = show f ++ " " ++ s {-! deriving instance Binary FixDecl !-} instance Ord FixDecl where compare (Fix x _) (Fix y _) = compare (prec x) (prec y) data Static = Static | Dynamic deriving (Show, Eq) {-! deriving instance Binary Static !-} -- Mark bindings with their explicitness, and laziness data Plicity = Imp { plazy :: Bool, pstatic :: Static, pdocstr :: String } | Exp { plazy :: Bool, pstatic :: Static, pdocstr :: String } | Constraint { plazy :: Bool, pstatic :: Static, pdocstr :: String } | TacImp { plazy :: Bool, pstatic :: Static, pscript :: PTerm, pdocstr :: String } deriving (Show, Eq) {-! deriving instance Binary Plicity !-} impl = Imp False Dynamic "" expl = Exp False Dynamic "" constraint = Constraint False Dynamic "" tacimpl t = TacImp False Dynamic t "" data FnOpt = Inlinable | TotalFn | PartialFn | Coinductive | AssertTotal | TCGen | Implicit -- implicit coercion | CExport String -- export, with a C name | Specialise [(Name, Maybe Int)] -- specialise it, freeze these names deriving (Show, Eq) {-! deriving instance Binary FnOpt !-} type FnOpts = [FnOpt] inlinable :: FnOpts -> Bool inlinable = elem Inlinable -- | Top-level declarations such as compiler directives, definitions, -- datatypes and typeclasses. data PDecl' t = PFix FC Fixity [String] -- ^ Fixity declaration | PTy String SyntaxInfo FC FnOpts Name t -- ^ Type declaration | PPostulate String SyntaxInfo FC FnOpts Name t -- ^ Postulate | PClauses FC FnOpts Name [PClause' t] -- ^ Pattern clause | PCAF FC Name t -- ^ Top level constant | PData String SyntaxInfo FC Bool (PData' t) -- ^ Data declaration. The Bool argument is True for codata. | PParams FC [(Name, t)] [PDecl' t] -- ^ Params block | PNamespace String [PDecl' t] -- ^ New namespace | PRecord String SyntaxInfo FC Name t String Name t -- ^ Record declaration | PClass String SyntaxInfo FC [t] -- constraints Name [(Name, t)] -- parameters [PDecl' t] -- declarations -- ^ Type class: arguments are documentation, syntax info, source location, constraints, -- class name, parameters, method declarations | PInstance SyntaxInfo FC [t] -- constraints Name -- class [t] -- parameters t -- full instance type (Maybe Name) -- explicit name [PDecl' t] -- ^ Instance declaration: arguments are syntax info, source location, constraints, -- class name, parameters, full instance type, optional explicit name, and definitions | PDSL Name (DSL' t) -- ^ DSL declaration | PSyntax FC Syntax -- ^ Syntax definition | PMutual FC [PDecl' t] -- ^ Mutual block | PDirective (Idris ()) -- ^ Compiler directive. The parser inserts the corresponding action in the Idris monad. | PProvider SyntaxInfo FC Name t t -- ^ Type provider. The first t is the type, the second is the term | PTransform FC Bool t t -- ^ Source-to-source transformation rule. If -- bool is True, lhs and rhs must be convertible | PReflection FC Name -- type to reflect to t -- type of variables [PClause' t] -- ^ Pattern clauses deriving Functor {-! deriving instance Binary PDecl' !-} -- | One clause of a top-level definition. Term arguments to constructors are: -- -- 1. The whole application (missing for PClauseR and PWithR because they're within a "with" clause) -- -- 2. The list of patterns -- -- 3. The right-hand side -- -- 4. The where block (PDecl' t) data PClause' t = PClause FC Name t [t] t [PDecl' t] -- ^ A normal top-level definition. | PWith FC Name t [t] t [PDecl' t] | PClauseR FC [t] t [PDecl' t] | PWithR FC [t] t [PDecl' t] deriving Functor {-! deriving instance Binary PClause' !-} -- | Data declaration data PData' t = PDatadecl { d_name :: Name, -- ^ The name of the datatype d_tcon :: t, -- ^ Type constructor d_cons :: [(String, Name, t, FC)] -- ^ Constructors } -- ^ Data declaration | PLaterdecl { d_name :: Name, d_tcon :: t } -- ^ "Placeholder" for data whose constructors are defined later deriving Functor {-! deriving instance Binary PData' !-} -- Handy to get a free function for applying PTerm -> PTerm functions -- across a program, by deriving Functor type PDecl = PDecl' PTerm type PData = PData' PTerm type PClause = PClause' PTerm -- get all the names declared in a decl declared :: PDecl -> [Name] declared (PFix _ _ _) = [] declared (PTy _ _ _ _ n t) = [n] declared (PPostulate _ _ _ _ n t) = [n] declared (PClauses _ _ n _) = [] -- not a declaration declared (PCAF _ n _) = [n] declared (PData _ _ _ _ (PDatadecl n _ ts)) = n : map fstt ts where fstt (_, a, _, _) = a declared (PData _ _ _ _ (PLaterdecl n _)) = [n] declared (PParams _ _ ds) = concatMap declared ds declared (PNamespace _ ds) = concatMap declared ds declared (PRecord _ _ _ n _ _ c _) = [n, c] declared (PClass _ _ _ _ n _ ms) = n : concatMap declared ms declared (PInstance _ _ _ _ _ _ _ _) = [] declared (PDSL n _) = [n] declared (PSyntax _ _) = [] declared (PMutual _ ds) = concatMap declared ds declared (PDirective _) = [] -- get the names declared, not counting nested parameter blocks tldeclared :: PDecl -> [Name] tldeclared (PFix _ _ _) = [] tldeclared (PTy _ _ _ _ n t) = [n] tldeclared (PPostulate _ _ _ _ n t) = [n] tldeclared (PClauses _ _ n _) = [] -- not a declaration tldeclared (PRecord _ _ _ n _ _ c _) = [n, c] tldeclared (PData _ _ _ _ (PDatadecl n _ ts)) = n : map fstt ts where fstt (_, a, _, _) = a tldeclared (PParams _ _ ds) = [] tldeclared (PMutual _ ds) = concatMap tldeclared ds tldeclared (PNamespace _ ds) = concatMap tldeclared ds defined :: PDecl -> [Name] defined (PFix _ _ _) = [] defined (PTy _ _ _ _ n t) = [] defined (PPostulate _ _ _ _ n t) = [] defined (PClauses _ _ n _) = [n] -- not a declaration defined (PCAF _ n _) = [n] defined (PData _ _ _ _ (PDatadecl n _ ts)) = n : map fstt ts where fstt (_, a, _, _) = a defined (PData _ _ _ _ (PLaterdecl n _)) = [] defined (PParams _ _ ds) = concatMap defined ds defined (PNamespace _ ds) = concatMap defined ds defined (PRecord _ _ _ n _ _ c _) = [n, c] defined (PClass _ _ _ _ n _ ms) = n : concatMap defined ms defined (PInstance _ _ _ _ _ _ _ _) = [] defined (PDSL n _) = [n] defined (PSyntax _ _) = [] defined (PMutual _ ds) = concatMap defined ds defined (PDirective _) = [] --defined _ = [] updateN :: [(Name, Name)] -> Name -> Name updateN ns n | Just n' <- lookup n ns = n' updateN _ n = n updateNs :: [(Name, Name)] -> PTerm -> PTerm updateNs [] t = t updateNs ns t = mapPT updateRef t where updateRef (PRef fc f) = PRef fc (updateN ns f) updateRef t = t -- updateDNs :: [(Name, Name)] -> PDecl -> PDecl -- updateDNs [] t = t -- updateDNs ns (PTy s f n t) | Just n' <- lookup n ns = PTy s f n' t -- updateDNs ns (PClauses f n c) | Just n' <- lookup n ns = PClauses f n' (map updateCNs c) -- where updateCNs ns (PClause n l ts r ds) -- = PClause (updateN ns n) (fmap (updateNs ns) l) -- (map (fmap (updateNs ns)) ts) -- (fmap (updateNs ns) r) -- (map (updateDNs ns) ds) -- updateDNs ns c = c -- | High level language terms data PTerm = PQuote Raw | PRef FC Name | PInferRef FC Name -- ^ A name to be defined later | PPatvar FC Name | PLam Name PTerm PTerm | PPi Plicity Name PTerm PTerm | PLet Name PTerm PTerm PTerm | PTyped PTerm PTerm -- ^ Term with explicit type | PApp FC PTerm [PArg] | PMatchApp FC Name -- ^ Make an application by type matching | PCase FC PTerm [(PTerm, PTerm)] | PTrue FC | PFalse FC | PRefl FC PTerm | PResolveTC FC | PEq FC PTerm PTerm | PRewrite FC PTerm PTerm (Maybe PTerm) | PPair FC PTerm PTerm | PDPair FC PTerm PTerm PTerm | PAlternative Bool [PTerm] -- True if only one may work | PHidden PTerm -- ^ Irrelevant or hidden pattern | PType | PConstant Const | Placeholder | PDoBlock [PDo] | PIdiom FC PTerm | PReturn FC | PMetavar Name | PProof [PTactic] -- ^ Proof script | PTactics [PTactic] -- ^ As PProof, but no auto solving | PElabError Err -- ^ Error to report on elaboration | PImpossible -- ^ Special case for declaring when an LHS can't typecheck | PCoerced PTerm -- ^ To mark a coerced argument, so as not to coerce twice | PUnifyLog PTerm -- ^ dump a trace of unifications when building term deriving Eq {-! deriving instance Binary PTerm !-} mapPT :: (PTerm -> PTerm) -> PTerm -> PTerm mapPT f t = f (mpt t) where mpt (PLam n t s) = PLam n (mapPT f t) (mapPT f s) mpt (PPi p n t s) = PPi p n (mapPT f t) (mapPT f s) mpt (PLet n ty v s) = PLet n (mapPT f ty) (mapPT f v) (mapPT f s) mpt (PRewrite fc t s g) = PRewrite fc (mapPT f t) (mapPT f s) (fmap (mapPT f) g) mpt (PApp fc t as) = PApp fc (mapPT f t) (map (fmap (mapPT f)) as) mpt (PCase fc c os) = PCase fc (mapPT f c) (map (pmap (mapPT f)) os) mpt (PEq fc l r) = PEq fc (mapPT f l) (mapPT f r) mpt (PTyped l r) = PTyped (mapPT f l) (mapPT f r) mpt (PPair fc l r) = PPair fc (mapPT f l) (mapPT f r) mpt (PDPair fc l t r) = PDPair fc (mapPT f l) (mapPT f t) (mapPT f r) mpt (PAlternative a as) = PAlternative a (map (mapPT f) as) mpt (PHidden t) = PHidden (mapPT f t) mpt (PDoBlock ds) = PDoBlock (map (fmap (mapPT f)) ds) mpt (PProof ts) = PProof (map (fmap (mapPT f)) ts) mpt (PTactics ts) = PTactics (map (fmap (mapPT f)) ts) mpt (PUnifyLog tm) = PUnifyLog (mapPT f tm) mpt x = x data PTactic' t = Intro [Name] | Intros | Focus Name | Refine Name [Bool] | Rewrite t | Equiv t | MatchRefine Name | LetTac Name t | LetTacTy Name t t | Exact t | Compute | Trivial | Solve | Attack | ProofState | ProofTerm | Undo | Try (PTactic' t) (PTactic' t) | TSeq (PTactic' t) (PTactic' t) | ApplyTactic t -- see Language.Reflection module | Reflect t | Fill t | GoalType String (PTactic' t) | Qed | Abandon deriving (Show, Eq, Functor) {-! deriving instance Binary PTactic' !-} instance Sized a => Sized (PTactic' a) where size (Intro nms) = 1 + size nms size Intros = 1 size (Focus nm) = 1 + size nm size (Refine nm bs) = 1 + size nm + length bs size (Rewrite t) = 1 + size t size (LetTac nm t) = 1 + size nm + size t size (Exact t) = 1 + size t size Compute = 1 size Trivial = 1 size Solve = 1 size Attack = 1 size ProofState = 1 size ProofTerm = 1 size Undo = 1 size (Try l r) = 1 + size l + size r size (TSeq l r) = 1 + size l + size r size (ApplyTactic t) = 1 + size t size (Reflect t) = 1 + size t size (Fill t) = 1 + size t size Qed = 1 size Abandon = 1 type PTactic = PTactic' PTerm data PDo' t = DoExp FC t | DoBind FC Name t | DoBindP FC t t | DoLet FC Name t t | DoLetP FC t t deriving (Eq, Functor) {-! deriving instance Binary PDo' !-} instance Sized a => Sized (PDo' a) where size (DoExp fc t) = 1 + size fc + size t size (DoBind fc nm t) = 1 + size fc + size nm + size t size (DoBindP fc l r) = 1 + size fc + size l + size r size (DoLet fc nm l r) = 1 + size fc + size nm + size l + size r size (DoLetP fc l r) = 1 + size fc + size l + size r type PDo = PDo' PTerm -- The priority gives a hint as to elaboration order. Best to elaborate -- things early which will help give a more concrete type to other -- variables, e.g. a before (interpTy a). -- TODO: priority no longer serves any purpose, drop it! data PArg' t = PImp { priority :: Int, lazyarg :: Bool, pname :: Name, getTm :: t, pargdoc :: String } | PExp { priority :: Int, lazyarg :: Bool, getTm :: t, pargdoc :: String } | PConstraint { priority :: Int, lazyarg :: Bool, getTm :: t, pargdoc :: String } | PTacImplicit { priority :: Int, lazyarg :: Bool, pname :: Name, getScript :: t, getTm :: t, pargdoc :: String } deriving (Show, Eq, Functor) instance Sized a => Sized (PArg' a) where size (PImp p l nm trm _) = 1 + size nm + size trm size (PExp p l trm _) = 1 + size trm size (PConstraint p l trm _) = 1 + size trm size (PTacImplicit p l nm scr trm _) = 1 + size nm + size scr + size trm {-! deriving instance Binary PArg' !-} pimp n t = PImp 0 True n t "" pexp t = PExp 0 False t "" pconst t = PConstraint 0 False t "" ptacimp n s t = PTacImplicit 0 True n s t "" type PArg = PArg' PTerm -- Type class data data ClassInfo = CI { instanceName :: Name, class_methods :: [(Name, (FnOpts, PTerm))], class_defaults :: [(Name, (Name, PDecl))], -- method name -> default impl class_params :: [Name], class_instances :: [Name] } deriving Show {-! deriving instance Binary ClassInfo !-} data OptInfo = Optimise { collapsible :: Bool, forceable :: [Int], -- argument positions recursive :: [Int] } deriving Show {-! deriving instance Binary OptInfo !-} data TypeInfo = TI { con_names :: [Name], codata :: Bool, param_pos :: [Int] } deriving Show {-! deriving instance Binary TypeInfo !-} -- Syntactic sugar info data DSL' t = DSL { dsl_bind :: t, dsl_return :: t, dsl_apply :: t, dsl_pure :: t, dsl_var :: Maybe t, index_first :: Maybe t, index_next :: Maybe t, dsl_lambda :: Maybe t, dsl_let :: Maybe t } deriving (Show, Functor) {-! deriving instance Binary DSL' !-} type DSL = DSL' PTerm data SynContext = PatternSyntax | TermSyntax | AnySyntax deriving Show {-! deriving instance Binary SynContext !-} data Syntax = Rule [SSymbol] PTerm SynContext deriving Show {-! deriving instance Binary Syntax !-} data SSymbol = Keyword Name | Symbol String | Binding Name | Expr Name | SimpleExpr Name deriving Show {-! deriving instance Binary SSymbol !-} initDSL = DSL (PRef f (UN ">>=")) (PRef f (UN "return")) (PRef f (UN "<$>")) (PRef f (UN "pure")) Nothing Nothing Nothing Nothing Nothing where f = FC "(builtin)" 0 data Using = UImplicit Name PTerm | UConstraint Name [Name] deriving (Show, Eq) {-! deriving instance Binary Using !-} data SyntaxInfo = Syn { using :: [Using], syn_params :: [(Name, PTerm)], syn_namespace :: [String], no_imp :: [Name], decoration :: Name -> Name, inPattern :: Bool, implicitAllowed :: Bool, dsl_info :: DSL } deriving Show {-! deriving instance Binary SyntaxInfo !-} defaultSyntax = Syn [] [] [] [] id False False initDSL expandNS :: SyntaxInfo -> Name -> Name expandNS syn n@(NS _ _) = n expandNS syn n = case syn_namespace syn of [] -> n xs -> NS n xs --- Pretty printing declarations and terms instance Show PTerm where show tm = showImp False tm instance Pretty PTerm where pretty = prettyImp False instance Show PDecl where show d = showDeclImp False d instance Show PClause where show c = showCImp True c instance Show PData where show d = showDImp False d showDeclImp _ (PFix _ f ops) = show f ++ " " ++ showSep ", " ops showDeclImp t (PTy _ _ _ _ n ty) = show n ++ " : " ++ showImp t ty showDeclImp t (PPostulate _ _ _ _ n ty) = show n ++ " : " ++ showImp t ty showDeclImp _ (PClauses _ _ n c) = showSep "\n" (map show c) showDeclImp _ (PData _ _ _ _ d) = show d showDeclImp _ (PParams f ns ps) = "parameters " ++ show ns ++ "\n" ++ showSep "\n" (map show ps) showCImp :: Bool -> PClause -> String showCImp impl (PClause _ n l ws r w) = showImp impl l ++ showWs ws ++ " = " ++ showImp impl r ++ " where " ++ show w where showWs [] = "" showWs (x : xs) = " | " ++ showImp impl x ++ showWs xs showCImp impl (PWith _ n l ws r w) = showImp impl l ++ showWs ws ++ " with " ++ showImp impl r ++ " { " ++ show w ++ " } " where showWs [] = "" showWs (x : xs) = " | " ++ showImp impl x ++ showWs xs showDImp :: Bool -> PData -> String showDImp impl (PDatadecl n ty cons) = "data " ++ show n ++ " : " ++ showImp impl ty ++ " where\n\t" ++ showSep "\n\t| " (map (\ (_, n, t, _) -> show n ++ " : " ++ showImp impl t) cons) getImps :: [PArg] -> [(Name, PTerm)] getImps [] = [] getImps (PImp _ _ n tm _ : xs) = (n, tm) : getImps xs getImps (_ : xs) = getImps xs getExps :: [PArg] -> [PTerm] getExps [] = [] getExps (PExp _ _ tm _ : xs) = tm : getExps xs getExps (_ : xs) = getExps xs getConsts :: [PArg] -> [PTerm] getConsts [] = [] getConsts (PConstraint _ _ tm _ : xs) = tm : getConsts xs getConsts (_ : xs) = getConsts xs getAll :: [PArg] -> [PTerm] getAll = map getTm prettyImp :: Bool -> PTerm -> Doc prettyImp impl = prettySe 10 where prettySe p (PQuote r) = if size r > breakingSize then text "![" $$ pretty r <> text "]" else text "![" <> pretty r <> text "]" prettySe p (PPatvar fc n) = pretty n prettySe p (PRef fc n) = if impl then pretty n else prettyBasic n where prettyBasic n@(UN _) = pretty n prettyBasic (MN _ s) = text s prettyBasic (NS n s) = (foldr (<>) empty (intersperse (text ".") (map text $ reverse s))) <> prettyBasic n prettySe p (PLam n ty sc) = bracket p 2 $ if size sc > breakingSize then text "λ" <> pretty n <+> text "=>" $+$ pretty sc else text "λ" <> pretty n <+> text "=>" <+> pretty sc prettySe p (PLet n ty v sc) = bracket p 2 $ if size sc > breakingSize then text "let" <+> pretty n <+> text "=" <+> prettySe 10 v <+> text "in" $+$ nest nestingSize (prettySe 10 sc) else text "let" <+> pretty n <+> text "=" <+> prettySe 10 v <+> text "in" <+> prettySe 10 sc prettySe p (PPi (Exp l s _) n ty sc) | n `elem` allNamesIn sc || impl = let open = if l then text "|" <> lparen else lparen in bracket p 2 $ if size sc > breakingSize then open <> pretty n <+> colon <+> prettySe 10 ty <> rparen <+> st <+> text "->" $+$ prettySe 10 sc else open <> pretty n <+> colon <+> prettySe 10 ty <> rparen <+> st <+> text "->" <+> prettySe 10 sc | otherwise = bracket p 2 $ if size sc > breakingSize then prettySe 0 ty <+> st <+> text "->" $+$ prettySe 10 sc else prettySe 0 ty <+> st <+> text "->" <+> prettySe 10 sc where st = case s of Static -> text "[static]" _ -> empty prettySe p (PPi (Imp l s _) n ty sc) | impl = let open = if l then text "|" <> lbrace else lbrace in bracket p 2 $ if size sc > breakingSize then open <> pretty n <+> colon <+> prettySe 10 ty <> rbrace <+> st <+> text "->" <+> prettySe 10 sc else open <> pretty n <+> colon <+> prettySe 10 ty <> rbrace <+> st <+> text "->" <+> prettySe 10 sc | otherwise = prettySe 10 sc where st = case s of Static -> text $ "[static]" _ -> empty prettySe p (PPi (Constraint _ _ _) n ty sc) = bracket p 2 $ if size sc > breakingSize then prettySe 10 ty <+> text "=>" <+> prettySe 10 sc else prettySe 10 ty <+> text "=>" $+$ prettySe 10 sc prettySe p (PPi (TacImp _ _ s _) n ty sc) = bracket p 2 $ if size sc > breakingSize then lbrace <> text "tacimp" <+> pretty n <+> colon <+> prettySe 10 ty <> rbrace <+> text "->" $+$ prettySe 10 sc else lbrace <> text "tacimp" <+> pretty n <+> colon <+> prettySe 10 ty <> rbrace <+> text "->" <+> prettySe 10 sc prettySe p (PApp _ (PRef _ f) []) | not impl = pretty f prettySe p (PApp _ (PRef _ op@(UN (f:_))) args) | length (getExps args) == 2 && (not impl) && (not $ isAlpha f) = let [l, r] = getExps args in bracket p 1 $ if size r > breakingSize then prettySe 1 l <+> pretty op $+$ prettySe 0 r else prettySe 1 l <+> pretty op <+> prettySe 0 r prettySe p (PApp _ f as) = let args = getExps as in bracket p 1 $ prettySe 1 f <+> if impl then foldl' fS empty as -- foldr (<+>) empty $ map prettyArgS as else foldl' fSe empty args -- foldr (<+>) empty $ map prettyArgSe args where fS l r = if size r > breakingSize then l $+$ nest nestingSize (prettyArgS r) else l <+> prettyArgS r fSe l r = if size r > breakingSize then l $+$ nest nestingSize (prettyArgSe r) else l <+> prettyArgSe r prettySe p (PCase _ scr opts) = text "case" <+> prettySe 10 scr <+> text "of" $+$ nest nestingSize prettyBody where prettyBody = foldr ($$) empty $ intersperse (text "|") $ map sc opts sc (l, r) = prettySe 10 l <+> text "=>" <+> prettySe 10 r prettySe p (PHidden tm) = text "." <> prettySe 0 tm prettySe p (PRefl _ _) = text "refl" prettySe p (PResolveTC _) = text "resolvetc" prettySe p (PTrue _) = text "()" prettySe p (PFalse _) = text "_|_" prettySe p (PEq _ l r) = bracket p 2 $ if size r > breakingSize then prettySe 10 l <+> text "=" $$ nest nestingSize (prettySe 10 r) else prettySe 10 l <+> text "=" <+> prettySe 10 r prettySe p (PRewrite _ l r _) = bracket p 2 $ if size r > breakingSize then text "rewrite" <+> prettySe 10 l <+> text "in" $$ nest nestingSize (prettySe 10 r) else text "rewrite" <+> prettySe 10 l <+> text "in" <+> prettySe 10 r prettySe p (PTyped l r) = lparen <> prettySe 10 l <+> colon <+> prettySe 10 r <> rparen prettySe p (PPair _ l r) = if size r > breakingSize then lparen <> prettySe 10 l <> text "," $+$ prettySe 10 r <> rparen else lparen <> prettySe 10 l <> text "," <+> prettySe 10 r <> rparen prettySe p (PDPair _ l t r) = if size r > breakingSize then lparen <> prettySe 10 l <+> text "**" $+$ prettySe 10 r <> rparen else lparen <> prettySe 10 l <+> text "**" <+> prettySe 10 r <> rparen prettySe p (PAlternative a as) = lparen <> text "|" <> prettyAs <> text "|" <> rparen where prettyAs = foldr (\l -> \r -> l <+> text "," <+> r) empty $ map (prettySe 10) as prettySe p PType = text "Type" prettySe p (PConstant c) = pretty c -- XXX: add pretty for tactics prettySe p (PProof ts) = text "proof" <+> lbrace $+$ nest nestingSize (text . show $ ts) $+$ rbrace prettySe p (PTactics ts) = text "tactics" <+> lbrace $+$ nest nestingSize (text . show $ ts) $+$ rbrace prettySe p (PMetavar n) = text "?" <> pretty n prettySe p (PReturn f) = text "return" prettySe p PImpossible = text "impossible" prettySe p Placeholder = text "_" prettySe p (PDoBlock _) = text "do block pretty not implemented" prettySe p (PElabError s) = pretty s prettySe p _ = text "test" prettyArgS (PImp _ _ n tm _) = prettyArgSi (n, tm) prettyArgS (PExp _ _ tm _) = prettyArgSe tm prettyArgS (PConstraint _ _ tm _) = prettyArgSc tm prettyArgS (PTacImplicit _ _ n _ tm _) = prettyArgSti (n, tm) prettyArgSe arg = prettySe 0 arg prettyArgSi (n, val) = lbrace <> pretty n <+> text "=" <+> prettySe 10 val <> rbrace prettyArgSc val = lbrace <> lbrace <> prettySe 10 val <> rbrace <> rbrace prettyArgSti (n, val) = lbrace <> text "auto" <+> pretty n <+> text "=" <+> prettySe 10 val <> rbrace bracket outer inner doc | inner > outer = lparen <> doc <> rparen | otherwise = doc showImp :: Bool -> PTerm -> String showImp impl tm = se 10 tm where se p (PQuote r) = "![" ++ show r ++ "]" se p (PPatvar fc n) = if impl then show n ++ "[p]" else show n se p (PInferRef fc n) = "!" ++ show n -- ++ "[" ++ show fc ++ "]" se p (PRef fc n) = if impl then show n -- ++ "[" ++ show fc ++ "]" else showbasic n where showbasic n@(UN _) = show n showbasic (MN _ s) = s showbasic (NS n s) = showSep "." (reverse s) ++ "." ++ showbasic n se p (PLam n ty sc) = bracket p 2 $ "\\ " ++ show n ++ (if impl then " : " ++ se 10 ty else "") ++ " => " ++ se 10 sc se p (PLet n ty v sc) = bracket p 2 $ "let " ++ show n ++ " = " ++ se 10 v ++ " in " ++ se 10 sc se p (PPi (Exp l s _) n ty sc) | n `elem` allNamesIn sc || impl = bracket p 2 $ (if l then "|(" else "(") ++ show n ++ " : " ++ se 10 ty ++ ") " ++ st ++ "-> " ++ se 10 sc | otherwise = bracket p 2 $ se 0 ty ++ " " ++ st ++ "-> " ++ se 10 sc where st = case s of Static -> "[static] " _ -> "" se p (PPi (Imp l s _) n ty sc) | impl = bracket p 2 $ (if l then "|{" else "{") ++ show n ++ " : " ++ se 10 ty ++ "} " ++ st ++ "-> " ++ se 10 sc | otherwise = se 10 sc where st = case s of Static -> "[static] " _ -> "" se p (PPi (Constraint _ _ _) n ty sc) = bracket p 2 $ se 10 ty ++ " => " ++ se 10 sc se p (PPi (TacImp _ _ s _) n ty sc) = bracket p 2 $ "{tacimp " ++ show n ++ " : " ++ se 10 ty ++ "} -> " ++ se 10 sc se p e | Just str <- slist p e = str | Just num <- snat p e = show num se p (PMatchApp _ f) = "match " ++ show f se p (PApp _ (PRef _ f) []) | not impl = show f se p (PApp _ (PRef _ op@(UN (f:_))) args) | length (getExps args) == 2 && not impl && not (isAlpha f) = let [l, r] = getExps args in bracket p 1 $ se 1 l ++ " " ++ show op ++ " " ++ se 0 r se p (PApp _ f as) = let args = getExps as in bracket p 1 $ se 1 f ++ if impl then concatMap sArg as else concatMap seArg args se p (PCase _ scr opts) = "case " ++ se 10 scr ++ " of " ++ showSep " | " (map sc opts) where sc (l, r) = se 10 l ++ " => " ++ se 10 r se p (PHidden tm) = "." ++ se 0 tm se p (PRefl _ t) | not impl = "refl" | otherwise = "refl {" ++ se 10 t ++ "}" se p (PResolveTC _) = "resolvetc" se p (PTrue _) = "()" se p (PFalse _) = "_|_" se p (PEq _ l r) = bracket p 2 $ se 10 l ++ " = " ++ se 10 r se p (PRewrite _ l r _) = bracket p 2 $ "rewrite " ++ se 10 l ++ " in " ++ se 10 r se p (PTyped l r) = "(" ++ se 10 l ++ " : " ++ se 10 r ++ ")" se p (PPair _ l r) = "(" ++ se 10 l ++ ", " ++ se 10 r ++ ")" se p (PDPair _ l t r) = "(" ++ se 10 l ++ " ** " ++ se 10 r ++ ")" se p (PAlternative a as) = "(|" ++ showSep " , " (map (se 10) as) ++ "|)" se p PType = "Type" se p (PConstant c) = show c se p (PProof ts) = "proof { " ++ show ts ++ "}" se p (PTactics ts) = "tactics { " ++ show ts ++ "}" se p (PMetavar n) = "?" ++ show n se p (PReturn f) = "return" se p PImpossible = "impossible" se p Placeholder = "_" se p (PDoBlock _) = "do block show not implemented" se p (PElabError s) = show s se p (PCoerced t) = se p t se p (PUnifyLog t) = "%unifyLog " ++ se p t -- se p x = "Not implemented" slist' p (PApp _ (PRef _ nil) _) | nsroot nil == UN "Nil" = Just [] slist' p (PApp _ (PRef _ cons) args) | nsroot cons == UN "::", (PExp {getTm=tl}):(PExp {getTm=hd}):imps <- reverse args, all isImp imps, Just tl' <- slist' p tl = Just (hd:tl') where isImp (PImp {}) = True isImp _ = False slist' _ _ = Nothing slist p e | Just es <- slist' p e = Just $ case es of [] -> "[]" [x] -> "[" ++ se p x ++ "]" xs -> "[" ++ intercalate "," (map (se p) xs) ++ "]" slist _ _ = Nothing -- since Prelude is always imported, S & Z are unqualified iff they're the -- Nat ones. snat p (PRef _ o) | show o == (natns++"Z") || show o == "Z" = Just 0 snat p (PApp _ s [PExp {getTm=n}]) | show s == (natns++"S") || show s == "S", Just n' <- snat p n = Just $ 1 + n' snat _ _ = Nothing natns = "Prelude.Nat." sArg (PImp _ _ n tm _) = siArg (n, tm) sArg (PExp _ _ tm _) = seArg tm sArg (PConstraint _ _ tm _) = scArg tm sArg (PTacImplicit _ _ n _ tm _) = stiArg (n, tm) seArg arg = " " ++ se 0 arg siArg (n, val) = " {" ++ show n ++ " = " ++ se 10 val ++ "}" scArg val = " {{" ++ se 10 val ++ "}}" stiArg (n, val) = " {auto " ++ show n ++ " = " ++ se 10 val ++ "}" bracket outer inner str | inner > outer = "(" ++ str ++ ")" | otherwise = str instance Sized PTerm where size (PQuote rawTerm) = size rawTerm size (PRef fc name) = size name size (PLam name ty bdy) = 1 + size ty + size bdy size (PPi plicity name ty bdy) = 1 + size ty + size bdy size (PLet name ty def bdy) = 1 + size ty + size def + size bdy size (PTyped trm ty) = 1 + size trm + size ty size (PApp fc name args) = 1 + size args size (PCase fc trm bdy) = 1 + size trm + size bdy size (PTrue fc) = 1 size (PFalse fc) = 1 size (PRefl fc _) = 1 size (PResolveTC fc) = 1 size (PEq fc left right) = 1 + size left + size right size (PRewrite fc left right _) = 1 + size left + size right size (PPair fc left right) = 1 + size left + size right size (PDPair fs left ty right) = 1 + size left + size ty + size right size (PAlternative a alts) = 1 + size alts size (PHidden hidden) = size hidden size (PUnifyLog tm) = size tm size PType = 1 size (PConstant const) = 1 + size const size Placeholder = 1 size (PDoBlock dos) = 1 + size dos size (PIdiom fc term) = 1 + size term size (PReturn fc) = 1 size (PMetavar name) = 1 size (PProof tactics) = size tactics size (PElabError err) = size err size PImpossible = 1 getPArity :: PTerm -> Int getPArity (PPi _ _ _ sc) = 1 + getPArity sc getPArity _ = 0 -- Return all names, free or globally bound, in the given term. allNamesIn :: PTerm -> [Name] allNamesIn tm = nub $ ni [] tm where ni env (PRef _ n) | not (n `elem` env) = [n] ni env (PApp _ f as) = ni env f ++ concatMap (ni env) (map getTm as) ni env (PCase _ c os) = ni env c ++ concatMap (ni env) (map snd os) ni env (PLam n ty sc) = ni env ty ++ ni (n:env) sc ni env (PPi _ n ty sc) = ni env ty ++ ni (n:env) sc ni env (PHidden tm) = ni env tm ni env (PEq _ l r) = ni env l ++ ni env r ni env (PRewrite _ l r _) = ni env l ++ ni env r ni env (PTyped l r) = ni env l ++ ni env r ni env (PPair _ l r) = ni env l ++ ni env r ni env (PDPair _ (PRef _ n) t r) = ni env t ++ ni (n:env) r ni env (PDPair _ l t r) = ni env l ++ ni env t ++ ni env r ni env (PAlternative a ls) = concatMap (ni env) ls ni env (PUnifyLog tm) = ni env tm ni env _ = [] -- Return names which are free in the given term. namesIn :: [(Name, PTerm)] -> IState -> PTerm -> [Name] namesIn uvars ist tm = nub $ ni [] tm where ni env (PRef _ n) | not (n `elem` env) = case lookupTy n (tt_ctxt ist) of [] -> [n] _ -> if n `elem` (map fst uvars) then [n] else [] ni env (PApp _ f as) = ni env f ++ concatMap (ni env) (map getTm as) ni env (PCase _ c os) = ni env c ++ concatMap (ni env) (map snd os) ni env (PLam n ty sc) = ni env ty ++ ni (n:env) sc ni env (PPi _ n ty sc) = ni env ty ++ ni (n:env) sc ni env (PEq _ l r) = ni env l ++ ni env r ni env (PRewrite _ l r _) = ni env l ++ ni env r ni env (PTyped l r) = ni env l ++ ni env r ni env (PPair _ l r) = ni env l ++ ni env r ni env (PDPair _ (PRef _ n) t r) = ni env t ++ ni (n:env) r ni env (PDPair _ l t r) = ni env l ++ ni env t ++ ni env r ni env (PAlternative a as) = concatMap (ni env) as ni env (PHidden tm) = ni env tm ni env (PUnifyLog tm) = ni env tm ni env _ = [] -- Return which of the given names are used in the given term. usedNamesIn :: [Name] -> IState -> PTerm -> [Name] usedNamesIn vars ist tm = nub $ ni [] tm where ni env (PRef _ n) | n `elem` vars && not (n `elem` env) = case lookupTy n (tt_ctxt ist) of [] -> [n] _ -> [] ni env (PApp _ f as) = ni env f ++ concatMap (ni env) (map getTm as) ni env (PCase _ c os) = ni env c ++ concatMap (ni env) (map snd os) ni env (PLam n ty sc) = ni env ty ++ ni (n:env) sc ni env (PPi _ n ty sc) = ni env ty ++ ni (n:env) sc ni env (PEq _ l r) = ni env l ++ ni env r ni env (PRewrite _ l r _) = ni env l ++ ni env r ni env (PTyped l r) = ni env l ++ ni env r ni env (PPair _ l r) = ni env l ++ ni env r ni env (PDPair _ (PRef _ n) t r) = ni env t ++ ni (n:env) r ni env (PDPair _ l t r) = ni env l ++ ni env t ++ ni env r ni env (PAlternative a as) = concatMap (ni env) as ni env (PHidden tm) = ni env tm ni env (PUnifyLog tm) = ni env tm ni env _ = []