----------------------------------------------------------------------------- -- -- Module : Language.PureScript.TypeChecker.Types -- Copyright : (c) Phil Freeman 2013 -- License : MIT -- -- Maintainer : Phil Freeman -- Stability : experimental -- Portability : -- -- | -- This module implements the type checker -- ----------------------------------------------------------------------------- {-# OPTIONS_GHC -fno-warn-orphans #-} {-# LANGUAGE DeriveDataTypeable #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE MultiParamTypeClasses #-} module Language.PureScript.TypeChecker.Types ( typesOf ) where {- The following functions represent the corresponding type checking judgements: infer Synthesize a type for a value check Check a value has a given type checkProperties Check an object with a given type contains specified properties checkFunctionApplication Check a function of a given type returns a value of another type when applied to its arguments subsumes Check a type subsumes another type -} import Data.List import Data.Maybe (maybeToList, isNothing, isJust, fromMaybe) import qualified Data.Data as D import Data.Generics (mkM, everywhereM, everything, mkT, something, everywhere, mkQ) import Language.PureScript.Values import Language.PureScript.Types import Language.PureScript.Kinds import Language.PureScript.Names import Language.PureScript.TypeChecker.Monad import Language.PureScript.TypeChecker.Kinds import Language.PureScript.TypeChecker.Synonyms import Language.PureScript.Pretty import Control.Monad.State import Control.Monad.Error import Control.Monad.Reader import Control.Monad.Unify import Control.Applicative import Control.Arrow (Arrow(..)) import qualified Data.Map as M import Data.Function (on) instance Unifiable Check Type where unknown = TUnknown isUnknown (TUnknown u) = Just u isUnknown _ = Nothing (?=) = unifyTypes -- | -- Unify two types, updating the current substitution -- unifyTypes :: Type -> Type -> UnifyT Check () unifyTypes t1 t2 = rethrow (\e -> "Error unifying type " ++ prettyPrintType t1 ++ " with type " ++ prettyPrintType t2 ++ ":\n" ++ e) $ do unifyTypes' t1 t2 where unifyTypes' (TUnknown u1) (TUnknown u2) | u1 == u2 = return () unifyTypes' (TUnknown u) t = replace u t unifyTypes' t (TUnknown u) = replace u t unifyTypes' (SaturatedTypeSynonym name args) ty = do ty1 <- expandTypeSynonym name args ty1 `unifyTypes` ty unifyTypes' ty s@(SaturatedTypeSynonym _ _) = s `unifyTypes` ty unifyTypes' (ForAll ident1 ty1) (ForAll ident2 ty2) = do sk <- skolemize ident1 ty1 replaced <- replaceVarWithUnknown ident2 ty2 sk `unifyTypes` replaced unifyTypes' (ForAll ident ty1) ty2 = do sk <- skolemize ident ty1 sk `unifyTypes` ty2 unifyTypes' ty f@(ForAll _ _) = f `unifyTypes` ty unifyTypes' (Object row1) (Object row2) = row1 ?= row2 unifyTypes' (TypeVar v1) (TypeVar v2) | v1 == v2 = return () unifyTypes' (TypeConstructor c1) (TypeConstructor c2) = do env <- getEnv Just moduleName <- checkCurrentModule <$> get guardWith ("Cannot unify " ++ show c1 ++ " with " ++ show c2 ++ ".") (typeConstructorsAreEqual env moduleName c1 c2) unifyTypes' (TypeApp t3 t4) (TypeApp t5 t6) = do t3 `unifyTypes` t5 t4 `unifyTypes` t6 unifyTypes' (Skolem s1) (Skolem s2) | s1 == s2 = return () unifyTypes' r1@(RCons _ _ _) r2 = unifyRows r1 r2 unifyTypes' r1 r2@(RCons _ _ _) = unifyRows r1 r2 unifyTypes' r1@REmpty r2 = unifyRows r1 r2 unifyTypes' r1 r2@REmpty = unifyRows r1 r2 unifyTypes' t3 t4 = throwError $ "Cannot unify " ++ prettyPrintType t3 ++ " with " ++ prettyPrintType t4 ++ "." -- | -- Unify two rows, updating the current substitution -- -- Common labels are first identified, and unified. Remaining labels and types are unified with a -- trailing row unification variable, if appropriate, otherwise leftover labels result in a unification -- error. -- unifyRows :: Type -> Type -> UnifyT Check () unifyRows r1 r2 = let (s1, r1') = rowToList r1 (s2, r2') = rowToList r2 int = [ (t1, t2) | (name, t1) <- s1, (name', t2) <- s2, name == name' ] sd1 = [ (name, t1) | (name, t1) <- s1, name `notElem` map fst s2 ] sd2 = [ (name, t2) | (name, t2) <- s2, name `notElem` map fst s1 ] in do forM_ int (uncurry (?=)) unifyRows' sd1 r1' sd2 r2' where unifyRows' :: [(String, Type)] -> Type -> [(String, Type)] -> Type -> UnifyT Check () unifyRows' [] (TUnknown u) sd r = replace u (rowFromList (sd, r)) unifyRows' sd r [] (TUnknown u) = replace u (rowFromList (sd, r)) unifyRows' ((name, ty):row) r others u@(TUnknown un) = do occursCheck un ty forM_ row $ \(_, t) -> occursCheck un t u' <- fresh u ?= RCons name ty u' unifyRows' row r others u' unifyRows' [] REmpty [] REmpty = return () unifyRows' [] (TypeVar v1) [] (TypeVar v2) | v1 == v2 = return () unifyRows' [] (Skolem s1) [] (Skolem s2) | s1 == s2 = return () unifyRows' sd3 r3 sd4 r4 = throwError $ "Cannot unify " ++ prettyPrintRow (rowFromList (sd3, r3)) ++ " with " ++ prettyPrintRow (rowFromList (sd4, r4)) ++ "." -- | -- Ensure type constructors are equal after canonicalization -- typeConstructorsAreEqual :: Environment -> ModuleName -> Qualified ProperName -> Qualified ProperName -> Bool typeConstructorsAreEqual env moduleName = (==) `on` canonicalizeType moduleName env -- | -- Infer the types of multiple mutually-recursive values, and return elaborated values including -- type class dictionaries and type annotations. -- typesOf :: ModuleName -> [(Ident, Value)] -> Check [(Ident, (Value, Type))] typesOf moduleName vals = do tys <- liftUnify $ do let es = map isTyped vals typed = filter (isJust . snd . snd) es untyped = filter (isNothing . snd . snd) es typedDict = map (\(ident, (_, Just ty)) -> (ident, ty)) typed untypedNames <- replicateM (length untyped) fresh let untypedDict = zip (map fst untyped) untypedNames dict = M.fromList (map (\(ident, ty) -> ((moduleName, ident), (ty, LocalVariable))) $ (map (id *** fst) typedDict) ++ untypedDict) forM es $ \e -> do triple@(_, (val, ty)) <- case e of (ident, (val, Just (ty, checkType))) -> do kind <- liftCheck $ kindOf moduleName ty guardWith ("Expected type of kind *, was " ++ prettyPrintKind kind) $ kind == Star ty' <- replaceAllTypeSynonyms ty val' <- bindNames dict $ if checkType then check val ty' else return val return (ident, (val', ty')) (ident, (val, Nothing)) -> do TypedValue _ val' ty <- bindNames dict $ infer val ty ?= fromMaybe (error "name not found in dictionary") (lookup ident untypedDict) return (ident, (val', ty)) when (moduleName == ModuleName (ProperName "Main") && fst e == Ident "main") $ do [eff, a] <- replicateM 2 fresh ty ?= TypeApp (TypeApp (TypeConstructor (Qualified (Just (ModuleName (ProperName "Eff"))) (ProperName "Eff"))) eff) a escapeCheck val ty return triple forM_ tys $ skolemEscapeCheck . snd . snd forM tys $ \(ident, (val, ty)) -> do val' <- replaceTypeClassDictionaries moduleName val return (ident, (overTypes (desaturateAllTypeSynonyms . setifyAll) $ val' , varIfUnknown . desaturateAllTypeSynonyms . setifyAll $ ty)) -- | -- Check if a value contains a type annotation -- isTyped :: (Ident, Value) -> (Ident, (Value, Maybe (Type, Bool))) isTyped (name, TypedValue checkType value ty) = (name, (value, Just (ty, checkType))) isTyped (name, value) = (name, (value, Nothing)) -- | -- Map a function over type annotations appearing inside a value -- overTypes :: (Type -> Type) -> Value -> Value overTypes f = everywhere (mkT f) -- | -- Replace type class dictionary placeholders with inferred type class dictionaries -- replaceTypeClassDictionaries :: ModuleName -> Value -> Check Value replaceTypeClassDictionaries mn = everywhereM (mkM go) where go (TypeClassDictionary constraint dicts) = entails mn dicts constraint go other = return other -- | -- Check that the current set of type class dictionaries entail the specified type class goal, and, if so, -- return a type class dictionary reference. -- entails :: ModuleName -> [TypeClassDictionaryInScope] -> (Qualified ProperName, Type) -> Check Value entails moduleName context goal@(className, ty) = do env <- getEnv case go env goal of [] -> throwError $ "No " ++ show className ++ " instance found for " ++ prettyPrintType ty (dict : _) -> return dict where go env (className', ty') = [ mkDictionary (canonicalizeDictionary tcd) args | tcd <- context , filterModule tcd , typeConstructorsAreEqual env moduleName className' (tcdClassName tcd) , subst <- maybeToList $ typeHeadsAreEqual moduleName env ty' (tcdInstanceType tcd) , args <- solveSubgoals env subst (tcdDependencies tcd) ] solveSubgoals :: Environment -> [(String, Type)] -> Maybe [(Qualified ProperName, Type)] -> [Maybe [Value]] solveSubgoals _ _ Nothing = return Nothing solveSubgoals env subst (Just subgoals) = do dict <- mapM (go env) (replaceAllTypeVars subst subgoals) return $ Just dict mkDictionary :: Qualified Ident -> Maybe [Value] -> Value mkDictionary fnName Nothing = Var fnName mkDictionary fnName (Just dicts) = foldr (flip App) (Var fnName) dicts filterModule :: TypeClassDictionaryInScope -> Bool filterModule (TypeClassDictionaryInScope { tcdName = Qualified (Just mn) _ }) | mn == moduleName = True filterModule (TypeClassDictionaryInScope { tcdName = Qualified Nothing _ }) = True filterModule _ = False canonicalizeDictionary :: TypeClassDictionaryInScope -> Qualified Ident canonicalizeDictionary (TypeClassDictionaryInScope { tcdType = TCDRegular, tcdName = nm }) = nm canonicalizeDictionary (TypeClassDictionaryInScope { tcdType = TCDAlias nm }) = nm -- | -- Check whether the type heads of two types are equal (for the purposes of type class dictionary lookup), -- and return a substitution from type variables to types which makes the type heads unify. -- typeHeadsAreEqual :: ModuleName -> Environment -> Type -> Type -> Maybe [(String, Type)] typeHeadsAreEqual _ _ (Skolem s1) (Skolem s2) | s1 == s2 = Just [] typeHeadsAreEqual m e (TypeConstructor c1) (TypeConstructor c2) | typeConstructorsAreEqual e m c1 c2 = Just [] typeHeadsAreEqual m e (TypeApp h1 (TypeVar v)) (TypeApp h2 arg) = (:) (v, arg) <$> typeHeadsAreEqual m e h1 h2 typeHeadsAreEqual m e t1@(TypeApp _ _) t2@(TypeApp _ (TypeVar _)) = typeHeadsAreEqual m e t2 t1 typeHeadsAreEqual _ _ _ _ = Nothing -- | -- Ensure unsolved unification variables do not escape -- escapeCheck :: Value -> Type -> UnifyT Check () escapeCheck value ty = do subst <- unifyCurrentSubstitution <$> UnifyT get let visibleUnknowns = nub $ unknowns $ runSubstitution subst ty let allUnknowns = findAllTypes value forM_ allUnknowns $ \t -> do let unsolvedUnknowns = nub . unknowns $ runSubstitution subst t guardWith ("Escape check fails" ++ show ( runSubstitution subst ty, runSubstitution subst t)) $ null $ unsolvedUnknowns \\ visibleUnknowns -- | -- Find all type annotations occuring inside a value -- findAllTypes :: Value -> [Type] findAllTypes = everything (++) (mkQ [] go) where go (TypedValue _ _ ty) = [ty] go _ = [] -- | -- Ensure skolem variables do not escape their scope -- skolemEscapeCheck :: Type -> Check () skolemEscapeCheck ty = case something (mkQ Nothing findSkolems) ty of Nothing -> return () Just _ -> throwError $ "Skolem variables cannot escape. Consider adding a type signature." where findSkolems (Skolem _) = return () findSkolems _ = mzero -- | -- Ensure a row contains no duplicate labels -- setify :: Type -> Type setify = rowFromList . first (M.toList . M.fromList) . rowToList -- | -- \"Setify\" all rows occuring inside a value -- setifyAll :: (D.Data d) => d -> d setifyAll = everywhere (mkT setify) -- | -- Replace outermost unsolved unification variables with named type variables -- varIfUnknown :: Type -> Type varIfUnknown ty = let unks = nub $ unknowns ty toName = (:) 't' . show ty' = everywhere (mkT typeToVar) $ ty typeToVar :: Type -> Type typeToVar (TUnknown (TypedUnknown (Unknown u))) = TypeVar (toName u) typeToVar t = t in mkForAll (sort . map (toName . runUnknown) $ unks) ty' -- | -- Replace named type variables with types -- replaceAllTypeVars :: (D.Data d) => [(String, Type)] -> d -> d replaceAllTypeVars = foldl' (\f (name, ty) -> replaceTypeVars name ty . f) id -- | -- Replace named type variables with new unification variables -- replaceAllVarsWithUnknowns :: Type -> UnifyT Check Type replaceAllVarsWithUnknowns (ForAll ident ty) = replaceVarWithUnknown ident ty >>= replaceAllVarsWithUnknowns replaceAllVarsWithUnknowns ty = return ty -- | -- Replace a single type variable with a new unification variable -- replaceVarWithUnknown :: String -> Type -> UnifyT Check Type replaceVarWithUnknown ident ty = do tu <- fresh return $ replaceTypeVars ident tu $ ty -- | -- Replace fully applied type synonyms with the @SaturatedTypeSynonym@ data constructor, which helps generate -- better error messages during unification. -- replaceAllTypeSynonyms :: (Functor m, MonadState CheckState m, MonadError String m) => (D.Data d) => d -> m d replaceAllTypeSynonyms d = do env <- getEnv Just moduleName <- checkCurrentModule <$> get let syns = map (\((path, name), (args, _)) -> ((path, name), length args)) . M.toList $ typeSynonyms env either throwError return $ saturateAllTypeSynonyms env moduleName syns d -- | -- \"Desaturate\" @SaturatedTypeSynonym@s -- desaturateAllTypeSynonyms :: (D.Data d) => d -> d desaturateAllTypeSynonyms = everywhere (mkT replaceSaturatedTypeSynonym) where replaceSaturatedTypeSynonym (SaturatedTypeSynonym name args) = foldl TypeApp (TypeConstructor name) args replaceSaturatedTypeSynonym t = t -- | -- Replace a type synonym and its arguments with the aliased type -- expandTypeSynonym :: Qualified ProperName -> [Type] -> UnifyT Check Type expandTypeSynonym name args = do env <- getEnv Just moduleName <- checkCurrentModule <$> get case M.lookup (canonicalizeType moduleName env name) (typeSynonyms env) of Just (synArgs, body) -> return $ replaceAllTypeVars (zip synArgs args) body Nothing -> error "Type synonym was not defined" -- | -- Ensure a set of property names and value does not contain duplicate labels -- ensureNoDuplicateProperties :: (MonadError String m) => [(String, Value)] -> m () ensureNoDuplicateProperties ps = guardWith "Duplicate property names" $ length (nub . map fst $ ps) == length ps -- | -- Infer a type for a value, rethrowing any error to provide a more useful error message -- infer :: Value -> UnifyT Check Value infer val = rethrow (\e -> "Error inferring type of term " ++ prettyPrintValue val ++ ":\n" ++ e) $ infer' val -- | -- Infer a type for a value -- infer' :: Value -> UnifyT Check Value infer' v@(NumericLiteral _) = return $ TypedValue True v tyNumber infer' v@(StringLiteral _) = return $ TypedValue True v tyString infer' v@(BooleanLiteral _) = return $ TypedValue True v tyBoolean infer' (ArrayLiteral vals) = do ts <- mapM infer vals els <- fresh forM_ ts $ \(TypedValue _ _ t) -> els ?= TypeApp tyArray t return $ TypedValue True (ArrayLiteral ts) els infer' (Unary op val) = do v <- infer val inferUnary op v infer' (Binary op left right) = do v1 <- infer left v2 <- infer right inferBinary op v1 v2 infer' (ObjectLiteral ps) = do ensureNoDuplicateProperties ps ts <- mapM (infer . snd) ps let fields = zipWith (\name (TypedValue _ _ t) -> (name, t)) (map fst ps) ts ty = Object $ rowFromList (fields, REmpty) return $ TypedValue True (ObjectLiteral (zip (map fst ps) ts)) ty infer' (ObjectUpdate o ps) = do ensureNoDuplicateProperties ps row <- fresh newVals <- zipWith (\(name, _) t -> (name, t)) ps <$> mapM (infer . snd) ps let newTys = map (\(name, TypedValue _ _ ty) -> (name, ty)) newVals oldTys <- zip (map fst ps) <$> replicateM (length ps) fresh o' <- check o $ Object $ rowFromList (oldTys, row) return $ TypedValue True (ObjectUpdate o' newVals) $ Object $ rowFromList (newTys, row) infer' (Indexer index val) = do el <- fresh index' <- check index tyNumber val' <- check val (TypeApp tyArray el) return $ TypedValue True (Indexer (TypedValue True index' tyNumber) (TypedValue True val' (TypeApp tyArray el))) el infer' (Accessor prop val) = do typed@(TypedValue _ _ objTy) <- infer val propTy <- inferProperty objTy prop case propTy of Nothing -> do field <- fresh rest <- fresh _ <- subsumes Nothing objTy (Object (RCons prop field rest)) return $ TypedValue True (Accessor prop typed) field Just ty -> return $ TypedValue True (Accessor prop typed) ty infer' (Abs arg ret) = do ty <- fresh Just moduleName <- checkCurrentModule <$> get bindLocalVariables moduleName [(arg, ty)] $ do body@(TypedValue _ _ bodyTy) <- infer' ret return $ TypedValue True (Abs arg body) $ function ty bodyTy infer' (App f arg) = do f'@(TypedValue _ _ ft) <- infer f ret <- fresh app <- checkFunctionApplication f' ft arg ret return $ TypedValue True app ret infer' (Var var) = do Just moduleName <- checkCurrentModule <$> get ty <- lookupVariable moduleName var ty' <- replaceAllTypeSynonyms ty case ty' of ConstrainedType constraints _ -> do env <- getEnv dicts <- getTypeClassDictionaries return $ TypedValue True (foldl App (Var var) (map (flip TypeClassDictionary dicts) (qualifyAllUnqualifiedNames moduleName env constraints))) ty' _ -> return $ TypedValue True (Var var) ty' infer' (Block ss) = do ret <- fresh (allCodePathsReturn, _, ss') <- checkBlock M.empty ret ss guardWith "Block is missing a return statement" allCodePathsReturn return $ TypedValue True (Block ss') ret infer' v@(Constructor c) = do env <- getEnv Just moduleName <- checkCurrentModule <$> get case M.lookup (qualify moduleName c) (dataConstructors env) of Nothing -> throwError $ "Constructor " ++ show c ++ " is undefined" Just (ty, _) -> do ty' <- replaceAllTypeSynonyms ty return $ TypedValue True v ty' infer' (Case vals binders) = do ts <- mapM infer vals ret <- fresh binders' <- checkBinders (map (\(TypedValue _ _ t) -> t) ts) ret binders return $ TypedValue True (Case ts binders') ret infer' (IfThenElse cond th el) = do cond' <- check cond tyBoolean v2@(TypedValue _ _ t2) <- infer th v3@(TypedValue _ _ t3) <- infer el t2 ?= t3 return $ TypedValue True (IfThenElse cond' v2 v3) t2 infer' (TypedValue checkType val ty) = do Just moduleName <- checkCurrentModule <$> get kind <- liftCheck $ kindOf moduleName ty guardWith ("Expected type of kind *, was " ++ prettyPrintKind kind) $ kind == Star ty' <- replaceAllTypeSynonyms ty val' <- if checkType then check val ty' else return val return $ TypedValue True val' ty infer' _ = error "Invalid argument to infer" -- | -- Infer the type of a property inside a record with a given type -- inferProperty :: Type -> String -> UnifyT Check (Maybe Type) inferProperty (Object row) prop = do let (props, _) = rowToList row return $ lookup prop props inferProperty (SaturatedTypeSynonym name args) prop = do replaced <- expandTypeSynonym name args inferProperty replaced prop inferProperty (ForAll ident ty) prop = do replaced <- replaceVarWithUnknown ident ty inferProperty replaced prop inferProperty _ _ = return Nothing -- | -- Infer the type of a unary operator application -- inferUnary :: UnaryOperator -> Value -> UnifyT Check Value inferUnary op (TypedValue _ val valTy) = case fromMaybe (error "Invalid operator") $ lookup op unaryOps of (valTy', resTy) -> do valTy' ?= valTy return $ TypedValue True (Unary op val) resTy inferUnary _ _ = error "Invalid arguments to inferUnary" -- | -- Check the type of a unary operator application -- checkUnary :: UnaryOperator -> Value -> Type -> UnifyT Check Value checkUnary op val res = case fromMaybe (error "Invalid operator") $ lookup op unaryOps of (valTy, resTy) -> do res ?= resTy val' <- check val valTy return $ Unary op val' -- | -- Built-in unary operators -- unaryOps :: [(UnaryOperator, (Type, Type))] unaryOps = [ (Negate, (tyNumber, tyNumber)) , (Not, (tyBoolean, tyBoolean)) , (BitwiseNot, (tyNumber, tyNumber)) ] -- | -- Infer the type of a binary operator application -- inferBinary :: BinaryOperator -> Value -> Value -> UnifyT Check Value inferBinary op left@(TypedValue _ _ leftTy) right@(TypedValue _ _ rightTy) | isEqualityTest op = do leftTy ?= rightTy return $ TypedValue True (Binary op left right) tyBoolean inferBinary op left@(TypedValue _ _ leftTy) right@(TypedValue _ _ rightTy) = case fromMaybe (error "Invalid operator") $ lookup op binaryOps of (valTy, resTy) -> do leftTy ?= valTy rightTy ?= valTy return $ TypedValue True (Binary op left right) resTy inferBinary _ _ _ = error "Invalid arguments to inferBinary" -- | -- Check the type of a binary operator application -- checkBinary :: BinaryOperator -> Value -> Value -> Type -> UnifyT Check Value checkBinary op left right res | isEqualityTest op = do res ?= tyBoolean left'@(TypedValue _ _ t1) <- infer left right'@(TypedValue _ _ t2) <- infer right t1 ?= t2 return $ Binary op left' right' checkBinary op left right res = case fromMaybe (error "Invalid operator") $ lookup op binaryOps of (valTy, resTy) -> do res ?= resTy left' <- check left valTy right' <- check right valTy return $ Binary op left' right' -- | -- Check if a @BinaryOperator@ is an equality test -- isEqualityTest :: BinaryOperator -> Bool isEqualityTest EqualTo = True isEqualityTest NotEqualTo = True isEqualityTest _ = False -- | -- Built-in binary operators -- binaryOps :: [(BinaryOperator, (Type, Type))] binaryOps = [ (Add, (tyNumber, tyNumber)) , (Subtract, (tyNumber, tyNumber)) , (Multiply, (tyNumber, tyNumber)) , (Divide, (tyNumber, tyNumber)) , (Modulus, (tyNumber, tyNumber)) , (BitwiseAnd, (tyNumber, tyNumber)) , (BitwiseOr, (tyNumber, tyNumber)) , (BitwiseXor, (tyNumber, tyNumber)) , (ShiftRight, (tyNumber, tyNumber)) , (ZeroFillShiftRight, (tyNumber, tyNumber)) , (And, (tyBoolean, tyBoolean)) , (Or, (tyBoolean, tyBoolean)) , (Concat, (tyString, tyString)) , (Modulus, (tyNumber, tyNumber)) , (LessThan, (tyNumber, tyBoolean)) , (LessThanOrEqualTo, (tyNumber, tyBoolean)) , (GreaterThan, (tyNumber, tyBoolean)) , (GreaterThanOrEqualTo, (tyNumber, tyBoolean)) ] -- | -- Infer the types of variables brought into scope by a binder -- inferBinder :: Type -> Binder -> UnifyT Check (M.Map Ident Type) inferBinder _ NullBinder = return M.empty inferBinder val (StringBinder _) = val ?= tyString >> return M.empty inferBinder val (NumberBinder _) = val ?= tyNumber >> return M.empty inferBinder val (BooleanBinder _) = val ?= tyBoolean >> return M.empty inferBinder val (VarBinder name) = return $ M.singleton name val inferBinder val (NullaryBinder ctor) = do env <- getEnv Just moduleName <- checkCurrentModule <$> get case M.lookup (qualify moduleName ctor) (dataConstructors env) of Just (ty, _) -> do _ <- subsumes Nothing ty val return M.empty _ -> throwError $ "Constructor " ++ show ctor ++ " is not defined" inferBinder val (UnaryBinder ctor binder) = do env <- getEnv Just moduleName <- checkCurrentModule <$> get case M.lookup (qualify moduleName ctor) (dataConstructors env) of Just (ty, _) -> do fn <- replaceAllVarsWithUnknowns ty case fn of TypeApp (TypeApp t obj) ret | t == tyFunction -> do _ <- subsumes Nothing val ret inferBinder obj binder _ -> throwError $ "Constructor " ++ show ctor ++ " is not a unary constructor" _ -> throwError $ "Constructor " ++ show ctor ++ " is not defined" inferBinder val (ObjectBinder props) = do row <- fresh rest <- fresh m1 <- inferRowProperties row rest props val ?= Object row return m1 where inferRowProperties :: Type -> Type -> [(String, Binder)] -> UnifyT Check (M.Map Ident Type) inferRowProperties nrow row [] = nrow ?= row >> return M.empty inferRowProperties nrow row ((name, binder):binders) = do propTy <- fresh m1 <- inferBinder propTy binder m2 <- inferRowProperties nrow (RCons name propTy row) binders return $ m1 `M.union` m2 inferBinder val (ArrayBinder binders) = do el <- fresh m1 <- M.unions <$> mapM (inferBinder el) binders val ?= TypeApp tyArray el return m1 inferBinder val (ConsBinder headBinder tailBinder) = do el <- fresh m1 <- inferBinder el headBinder m2 <- inferBinder val tailBinder val ?= TypeApp tyArray el return $ m1 `M.union` m2 inferBinder val (NamedBinder name binder) = do m <- inferBinder val binder return $ M.insert name val m -- | -- Check the types of the return values in a set of binders in a case statement -- checkBinders :: [Type] -> Type -> [([Binder], Maybe Guard, Value)] -> UnifyT Check [([Binder], Maybe Guard, Value)] checkBinders _ _ [] = return [] checkBinders nvals ret ((binders, grd, val):bs) = do Just moduleName <- checkCurrentModule <$> get m1 <- M.unions <$> zipWithM inferBinder nvals binders r <- bindLocalVariables moduleName (M.toList m1) $ do val' <- check val ret case grd of Nothing -> return (binders, Nothing, val') Just g -> do g' <- check g tyBoolean return (binders, Just g', val') rs <- checkBinders nvals ret bs return $ r : rs -- | -- Check that a local variable name is not already used -- assignVariable :: Ident -> UnifyT Check () assignVariable name = do env <- checkEnv <$> get Just moduleName <- checkCurrentModule <$> get case M.lookup (moduleName, name) (names env) of Just _ -> UnifyT . lift . throwError $ "Variable with name " ++ show name ++ " already exists." _ -> return () -- | -- Check the type of the return values of a statement, returning whether or not the statement returns on -- all code paths -- checkStatement :: M.Map Ident Type -> Type -> Statement -> UnifyT Check (Bool, M.Map Ident Type, Statement) checkStatement mass _ (VariableIntroduction name val) = do assignVariable name val'@(TypedValue _ _ t) <- infer val return (False, M.insert name t mass, VariableIntroduction name val') checkStatement mass _ (Assignment ident val) = do val'@(TypedValue _ _ t) <- infer val case M.lookup ident mass of Nothing -> throwError $ "No local variable with name " ++ show ident Just ty -> do t ?= ty return (False, mass, Assignment ident val') checkStatement mass ret (While val inner) = do val' <- check val tyBoolean (allCodePathsReturn, _, inner') <- checkBlock mass ret inner return (allCodePathsReturn, mass, While val' inner') checkStatement mass ret (If ifst) = do (allCodePathsReturn, ifst') <- checkIfStatement mass ret ifst return (allCodePathsReturn, mass, If ifst') checkStatement mass ret (For ident start end inner) = do Just moduleName <- checkCurrentModule <$> get assignVariable ident start' <- check start tyNumber end' <- check end tyNumber (allCodePathsReturn, _, inner') <- bindLocalVariables moduleName [(ident, tyNumber)] $ checkBlock mass ret inner return (allCodePathsReturn, mass, For ident start' end' inner') checkStatement mass ret (Return val) = do val' <- check val ret return (True, mass, Return val') -- | -- Check the type of an if-then-else statement -- checkIfStatement :: M.Map Ident Type -> Type -> IfStatement -> UnifyT Check (Bool, IfStatement) checkIfStatement mass ret (IfStatement val thens Nothing) = do val' <- check val tyBoolean (_, _, thens') <- checkBlock mass ret thens return (False, IfStatement val' thens' Nothing) checkIfStatement mass ret (IfStatement val thens (Just elses)) = do val' <- check val tyBoolean (allCodePathsReturn1, _, thens') <- checkBlock mass ret thens (allCodePathsReturn2, elses') <- checkElseStatement mass ret elses return (allCodePathsReturn1 && allCodePathsReturn2, IfStatement val' thens' (Just elses')) -- | -- Check the type of an else statement -- checkElseStatement :: M.Map Ident Type -> Type -> ElseStatement -> UnifyT Check (Bool, ElseStatement) checkElseStatement mass ret (Else elses) = do (allCodePathsReturn, _, elses') <- checkBlock mass ret elses return (allCodePathsReturn, Else elses') checkElseStatement mass ret (ElseIf ifst) = (id *** ElseIf) <$> checkIfStatement mass ret ifst -- | -- Check the type of the return value of a block of statements -- checkBlock :: M.Map Ident Type -> Type -> [Statement] -> UnifyT Check (Bool, M.Map Ident Type, [Statement]) checkBlock mass _ [] = return (False, mass, []) checkBlock mass ret (s:ss) = do Just moduleName <- checkCurrentModule <$> get (b1, mass1, s') <- checkStatement mass ret s bindLocalVariables moduleName (M.toList mass1) $ case (b1, ss) of (True, []) -> return (True, mass1, [s']) (True, _) -> throwError "Unreachable code" (False, ss') -> do (b, m, ss'') <- checkBlock mass1 ret ss' return (b, m, s':ss'') -- | -- Skolemize a type variable by replacing its instances with fresh skolem constants -- skolemize :: String -> Type -> UnifyT Check Type skolemize ident ty = do tsk <- Skolem . runUnknown <$> fresh' return $ replaceTypeVars ident tsk ty -- | -- Check the type of a value, rethrowing errors to provide a better error message -- check :: Value -> Type -> UnifyT Check Value check val ty = rethrow errorMessage $ check' val ty where errorMessage msg = "Error checking type of term " ++ prettyPrintValue val ++ " against type " ++ prettyPrintType ty ++ ":\n" ++ msg -- | -- Check the type of a value -- check' :: Value -> Type -> UnifyT Check Value check' val (ForAll idents ty) = do sk <- skolemize idents ty check val sk check' val (ConstrainedType constraints ty) = do env <- getEnv Just moduleName <- checkCurrentModule <$> get dictNames <- flip mapM constraints $ \(Qualified _ (ProperName className), _) -> do n <- liftCheck freshDictionaryName return $ Ident $ "__dict_" ++ className ++ "_" ++ show n val' <- withTypeClassDictionaries (zipWith (\name (className, instanceTy) -> TypeClassDictionaryInScope name className instanceTy Nothing TCDRegular) (map (Qualified Nothing) dictNames) (qualifyAllUnqualifiedNames moduleName env constraints)) $ check val ty return $ foldr Abs val' dictNames check' val u@(TUnknown _) = do val'@(TypedValue _ _ ty) <- infer val -- Don't unify an unknown with an inferred polytype ty' <- replaceAllVarsWithUnknowns ty ty' ?= u return val' check' v@(NumericLiteral _) t | t == tyNumber = return v check' v@(StringLiteral _) t | t == tyString = return v check' v@(BooleanLiteral _) t | t == tyBoolean = return v check' (Unary op val) ty = checkUnary op val ty check' (Binary op left right) ty = checkBinary op left right ty check' (ArrayLiteral vals) (TypeApp a ty) | a == tyArray = ArrayLiteral <$> forM vals (\val -> check val ty) check' (Indexer index vals) ty = do index' <- check index tyNumber vals' <- check vals (TypeApp tyArray ty) return $ Indexer index' vals' check' (Abs arg ret) (TypeApp (TypeApp t argTy) retTy) | t == tyFunction = do Just moduleName <- checkCurrentModule <$> get ret' <- bindLocalVariables moduleName [(arg, argTy)] $ check ret retTy return $ Abs arg ret' check' (App f arg) ret = do f'@(TypedValue _ _ ft) <- infer f app <- checkFunctionApplication f' ft arg ret return $ app check' v@(Var var) ty = do Just moduleName <- checkCurrentModule <$> get ty1 <- lookupVariable moduleName var repl <- replaceAllTypeSynonyms ty1 v' <- subsumes (Just v) repl ty case v' of Nothing -> throwError "Unable to check type subsumption" Just v'' -> return v'' check' (TypedValue checkType val ty1) ty2 = do Just moduleName <- checkCurrentModule <$> get kind <- liftCheck $ kindOf moduleName ty1 guardWith ("Expected type of kind *, was " ++ prettyPrintKind kind) $ kind == Star val' <- subsumes (Just val) ty1 ty2 case val' of Nothing -> throwError "Unable to check type subsumption" Just val'' -> do val''' <- if checkType then check val'' ty1 else return val'' return $ TypedValue True val''' ty1 check' (Case vals binders) ret = do vals' <- mapM infer vals let ts = map (\(TypedValue _ _ t) -> t) vals' binders' <- checkBinders ts ret binders return $ Case vals' binders' check' (IfThenElse cond th el) ty = do cond' <- check cond tyBoolean th' <- check th ty el' <- check el ty return $ IfThenElse cond' th' el' check' (ObjectLiteral ps) (Object row) = do ensureNoDuplicateProperties ps ps' <- checkProperties ps row False return $ ObjectLiteral ps' check' (ObjectUpdate obj ps) (Object row) = do ensureNoDuplicateProperties ps us <- zip (map fst ps) <$> replicateM (length ps) fresh let (propsToCheck, rest) = rowToList row propsToRemove = map fst ps remainingProps = filter (\(p, _) -> p `notElem` propsToRemove) propsToCheck obj' <- check obj (Object (rowFromList (us ++ remainingProps, rest))) ps' <- checkProperties ps row True return $ ObjectUpdate obj' ps' check' (Accessor prop val) ty = do rest <- fresh val' <- check val (Object (RCons prop ty rest)) return $ Accessor prop val' check' (Block ss) ret = do (allCodePathsReturn, _, ss') <- checkBlock M.empty ret ss guardWith "Block is missing a return statement" allCodePathsReturn return $ Block ss' check' (Constructor c) ty = do env <- getEnv Just moduleName <- checkCurrentModule <$> get case M.lookup (qualify moduleName c) (dataConstructors env) of Nothing -> throwError $ "Constructor " ++ show c ++ " is undefined" Just (ty1, _) -> do repl <- replaceAllTypeSynonyms ty1 _ <- subsumes Nothing repl ty return $ Constructor c check' val (SaturatedTypeSynonym name args) = do ty <- expandTypeSynonym name args check val ty check' val ty = throwError $ prettyPrintValue val ++ " does not have type " ++ prettyPrintType ty -- | -- Check the type of a collection of named record fields -- -- The @lax@ parameter controls whether or not every record member has to be provided. For object updates, this is not the case. -- checkProperties :: [(String, Value)] -> Type -> Bool -> UnifyT Check [(String, Value)] checkProperties ps row lax = let (ts, r') = rowToList row in go ps ts r' where go [] [] REmpty = return [] go [] [] u@(TUnknown _) = do u ?= REmpty return [] go [] [] (Skolem _) | lax = return [] go [] ((p, _): _) _ | lax = return [] | otherwise = throwError $ prettyPrintValue (ObjectLiteral ps) ++ " does not have property " ++ p go ((p,_):_) [] REmpty = throwError $ "Property " ++ p ++ " is not present in closed object type " ++ prettyPrintRow row go ((p,v):ps') [] u@(TUnknown _) = do v'@(TypedValue _ _ ty) <- infer v rest <- fresh u ?= RCons p ty rest ps'' <- go ps' [] rest return $ (p, v') : ps'' go ((p,v):ps') ts r = case lookup p ts of Nothing -> do v'@(TypedValue _ _ ty) <- infer v rest <- fresh r ?= RCons p ty rest ps'' <- go ps' ts rest return $ (p, v') : ps'' Just ty -> do v' <- check v ty ps'' <- go ps' (delete (p, ty) ts) r return $ (p, v') : ps'' go _ _ _ = throwError $ prettyPrintValue (ObjectLiteral ps) ++ " does not have type " ++ prettyPrintType (Object row) -- | -- Check the type of a function application, rethrowing errors to provide a better error message -- checkFunctionApplication :: Value -> Type -> Value -> Type -> UnifyT Check Value checkFunctionApplication fn fnTy arg ret = rethrow errorMessage $ checkFunctionApplication' fn fnTy arg ret where errorMessage msg = "Error applying function of type " ++ prettyPrintType fnTy ++ " to argument " ++ prettyPrintValue arg ++ ":\n" ++ msg -- | -- Check the type of a function application -- checkFunctionApplication' :: Value -> Type -> Value -> Type -> UnifyT Check Value checkFunctionApplication' fn (TypeApp (TypeApp tyFunction' argTy) retTy) arg ret = do tyFunction' ?= tyFunction arg' <- check arg argTy _ <- subsumes Nothing retTy ret return $ App fn arg' checkFunctionApplication' fn (ForAll ident ty) arg ret = do replaced <- replaceVarWithUnknown ident ty checkFunctionApplication fn replaced arg ret checkFunctionApplication' fn u@(TUnknown _) arg ret = do arg' <- do TypedValue _ v t <- infer arg TypedValue True v <$> replaceAllVarsWithUnknowns t let ty = (\(TypedValue _ _ t) -> t) arg' u ?= function ty ret return $ App fn arg' checkFunctionApplication' fn (SaturatedTypeSynonym name tyArgs) arg ret = do ty <- expandTypeSynonym name tyArgs checkFunctionApplication fn ty arg ret checkFunctionApplication' fn (ConstrainedType constraints fnTy) arg ret = do env <- getEnv dicts <- getTypeClassDictionaries Just moduleName <- checkCurrentModule <$> get checkFunctionApplication' (foldl App fn (map (flip TypeClassDictionary dicts) (qualifyAllUnqualifiedNames moduleName env constraints))) fnTy arg ret checkFunctionApplication' _ fnTy arg ret = throwError $ "Applying a function of type " ++ prettyPrintType fnTy ++ " to argument(s) " ++ prettyPrintValue arg ++ " does not yield a value of type " ++ prettyPrintType ret ++ "." -- | -- Check whether one type subsumes another, rethrowing errors to provide a better error message -- subsumes :: Maybe Value -> Type -> Type -> UnifyT Check (Maybe Value) subsumes val ty1 ty2 = rethrow errorMessage $ subsumes' val ty1 ty2 where errorMessage msg = "Error checking that type " ++ prettyPrintType ty1 ++ " subsumes type " ++ prettyPrintType ty2 ++ ":\n" ++ msg -- | -- Check whether one type subsumes another -- subsumes' :: Maybe Value -> Type -> Type -> UnifyT Check (Maybe Value) subsumes' val (ForAll ident ty1) ty2 = do replaced <- replaceVarWithUnknown ident ty1 subsumes val replaced ty2 subsumes' val (TypeApp (TypeApp f1 arg1) ret1) (TypeApp (TypeApp f2 arg2) ret2) | f1 == tyFunction && f2 == tyFunction = do subsumes Nothing arg2 arg1 subsumes Nothing ret1 ret2 return val subsumes' (Just val) (ConstrainedType constraints ty1) ty2 = do env <- getEnv dicts <- getTypeClassDictionaries Just moduleName <- checkCurrentModule <$> get _ <- subsumes' Nothing ty1 ty2 return . Just $ foldl App val (map (flip TypeClassDictionary dicts) (qualifyAllUnqualifiedNames moduleName env constraints)) subsumes' val ty1 ty2 = do ty1 ?= ty2 return val