{-# OPTIONS_GHC -W #-} module AST.Type where import Control.Applicative ((<$>), (<*>)) import Data.Binary import qualified Data.Map as Map import qualified AST.Variable as Var import AST.PrettyPrint import qualified AST.Helpers as Help import Text.PrettyPrint as P data Type var = Lambda (Type var) (Type var) | Var String | Type var | App (Type var) [Type var] | Record [(String, Type var)] (Maybe (Type var)) | Aliased Var.Canonical (Type var) deriving (Eq,Show) type RawType = Type Var.Raw type CanonicalType = Type Var.Canonical fieldMap :: [(String,a)] -> Map.Map String [a] fieldMap fields = foldl (\r (x,t) -> Map.insertWith (++) x [t] r) Map.empty fields recordOf :: [(String, Type var)] -> Type var recordOf fields = Record fields Nothing listOf :: RawType -> RawType listOf t = App (Type (Var.Raw "_List")) [t] tupleOf :: [RawType] -> RawType tupleOf ts = App (Type t) ts where t = Var.Raw ("_Tuple" ++ show (length ts)) instance (Var.ToString var, Pretty var) => Pretty (Type var) where pretty tipe = case tipe of Lambda _ _ -> P.sep [ t, P.sep (map (P.text "->" <+>) ts) ] where t:ts = map prettyLambda (collectLambdas tipe) prettyLambda t = case t of Lambda _ _ -> P.parens (pretty t) _ -> pretty t Var x -> P.text x Type var -> let v = Var.toString var in P.text (if v == "_Tuple0" then "()" else v) App f args -> case (f,args) of (Type name, [t]) | Var.toString name == "_List" -> P.brackets (pretty t) (Type name, _) | Help.isTuple (Var.toString name) -> P.parens . P.sep . P.punctuate P.comma $ map pretty args _ -> P.hang (pretty f) 2 (P.sep $ map prettyParens args) Record _ _ -> P.braces $ case flattenRecord tipe of (fields, Nothing) -> prettyFields fields (fields, Just x) -> P.hang (P.text x <+> P.text "|") 4 (prettyFields fields) where prettyField (f,t) = P.text f <+> P.text ":" <+> pretty t prettyFields fields = commaSep (map prettyField fields) Aliased name t -> let t' = pretty t in if show t' `elem` ["Int", "Float", "String", "Char", "Bool"] then t' else pretty name collectLambdas :: Type var -> [Type var] collectLambdas tipe = case tipe of Lambda arg body -> arg : collectLambdas body _ -> [tipe] prettyParens :: (Var.ToString var, Pretty var) => Type var -> Doc prettyParens tipe = parensIf (needed tipe) (pretty tipe) where needed t = case t of Aliased _ t' -> needed t' Lambda _ _ -> True App (Type name) [_] | Var.toString name == "_List" -> False App (Type name) _ | Help.isTuple (Var.toString name) -> False App t' [] -> needed t' App _ _ -> True _ -> False flattenRecord :: Type var -> ( [(String, Type var)], Maybe String ) flattenRecord tipe = case tipe of Var x -> ([], Just x) Record fields Nothing -> (fields, Nothing) Record fields (Just ext) -> let (fields',ext') = flattenRecord ext in (fields' ++ fields, ext') Aliased _ tipe' -> flattenRecord tipe' _ -> error "Trying to flatten ill-formed record." instance Binary var => Binary (Type var) where put tipe = case tipe of Lambda t1 t2 -> putWord8 0 >> put t1 >> put t2 Var x -> putWord8 1 >> put x Type name -> putWord8 2 >> put name App t1 t2 -> putWord8 3 >> put t1 >> put t2 Record fs ext -> putWord8 4 >> put fs >> put ext Aliased var t -> putWord8 5 >> put var >> put t get = do n <- getWord8 case n of 0 -> Lambda <$> get <*> get 1 -> Var <$> get 2 -> Type <$> get 3 -> App <$> get <*> get 4 -> Record <$> get <*> get 5 -> Aliased <$> get <*> get _ -> error "Error reading a valid type from serialized string"