{-# LANGUAGE LambdaCase #-} {-# LANGUAGE OverloadedStrings #-} module Codegen.Codegen (generateCode) where import Auxiliary (snoc) import Codegen.LlvmIr as LIR import Control.Applicative ((<|>)) import Control.Monad.State (StateT, execStateT, foldM_, gets, modify) import qualified Data.Bifunctor as BI import Data.Coerce (coerce) import Data.Map (Map) import qualified Data.Map as Map import Data.Maybe (fromJust, fromMaybe) import Data.Tuple.Extra (dupe, first, second) import qualified Grammar.Abs as GA import Grammar.ErrM (Err) import Monomorphizer.MonomorphizerIr as MIR -- | The record used as the code generator state data CodeGenerator = CodeGenerator { instructions :: [LLVMIr] , functions :: Map MIR.Id FunctionInfo , constructors :: Map MIR.Id ConstructorInfo , variableCount :: Integer , labelCount :: Integer } -- | A state type synonym type CompilerState a = StateT CodeGenerator Err a data FunctionInfo = FunctionInfo { numArgs :: Int , arguments :: [Id] } deriving (Show) data ConstructorInfo = ConstructorInfo { numArgsCI :: Int , argumentsCI :: [Id] , numCI :: Integer } deriving (Show) -- | Adds a instruction to the CodeGenerator state emit :: LLVMIr -> CompilerState () emit l = modify $ \t -> t{instructions = Auxiliary.snoc l $ instructions t} -- | Increases the variable counter in the CodeGenerator state increaseVarCount :: CompilerState () increaseVarCount = modify $ \t -> t{variableCount = variableCount t + 1} -- | Returns the variable count from the CodeGenerator state getVarCount :: CompilerState Integer getVarCount = gets variableCount -- | Increases the variable count and returns it from the CodeGenerator state getNewVar :: CompilerState GA.Ident getNewVar = (GA.Ident . show) <$> (increaseVarCount >> getVarCount) -- | Increses the label count and returns a label from the CodeGenerator state getNewLabel :: CompilerState Integer getNewLabel = do modify (\t -> t{labelCount = labelCount t + 1}) gets labelCount {- | Produces a map of functions infos from a list of binds, which contains useful data for code generation. -} getFunctions :: [MIR.Def] -> Map Id FunctionInfo getFunctions bs = Map.fromList $ go bs where go [] = [] go (MIR.DBind (MIR.Bind id args _) : xs) = (id, FunctionInfo{numArgs = length args, arguments = args}) : go xs go (MIR.DData (MIR.Data n cons) : xs) = do map ( \(Constructor id xs) -> ( (coerce id, MIR.TLit (extractTypeName n)) , FunctionInfo { numArgs = length xs , arguments = createArgs (snd <$> xs) } ) ) cons <> go xs createArgs :: [MIR.Type] -> [Id] createArgs xs = fst $ foldl (\(acc, l) t -> (acc ++ [(GA.Ident ("arg_" <> show l), t)], l + 1)) ([], 0) xs {- | Produces a map of functions infos from a list of binds, which contains useful data for code generation. -} getConstructors :: [MIR.Def] -> Map MIR.Id ConstructorInfo getConstructors bs = Map.fromList $ go bs where go [] = [] go (MIR.DData (MIR.Data t cons) : xs) = do let (GA.Ident n) = extractTypeName t fst ( foldl ( \(acc, i) (Constructor (GA.UIdent id) xs) -> ( ( (GA.Ident (n <> "_" <> id), MIR.TLit (coerce n)) , ConstructorInfo { numArgsCI = length xs , argumentsCI = createArgs (snd <$> xs) , numCI = i } ) : acc , i + 1 ) ) ([], 0) cons ) <> go xs go (_ : xs) = go xs initCodeGenerator :: [MIR.Def] -> CodeGenerator initCodeGenerator scs = CodeGenerator { instructions = defaultStart , functions = getFunctions scs , constructors = getConstructors scs , variableCount = 0 , labelCount = 0 } {- run :: Err String -> IO () run s = do let s' = case s of Right s -> s Left _ -> error "yo" writeFile "output/llvm.ll" s' putStrLn . trim =<< readCreateProcess (shell "lli") s' test :: Integer -> Program test v = Program [ DataType (GA.Ident "Craig") [ Constructor (GA.Ident "Bob") [MIR.Type (GA.Ident "_Int")] , Constructor (GA.Ident "Betty") [MIR.Type (GA.Ident "_Int")] ] , DataType (GA.Ident "Alice") [ Constructor (GA.Ident "Eve") [MIR.Type (GA.Ident "_Int")] -- , -- (GA.Ident "Alice", [TInt, TInt]) ] , Bind (GA.Ident "fibonacci", MIR.Type (GA.Ident "_Int")) [(GA.Ident "x", MIR.Type (GA.Ident "_Int"))] (EId ("x", MIR.Type (GA.Ident "Craig")), MIR.Type (GA.Ident "Craig")) , Bind (GA.Ident "main", MIR.Type (GA.Ident "_Int")) [] -- (EApp (MIR.Type (GA.Ident "Craig")) (EId (GA.Ident "Craig_Bob", MIR.Type (GA.Ident "Craig")), MIR.Type (GA.Ident "Craig")) (ELit (LInt v), MIR.Type (GA.Ident "_Int")), MIR.Type (GA.Ident "Craig"))-- (EInt 92) $ eCaseInt (EApp (MIR.TLit (GA.Ident "Craig")) (EId (GA.Ident "Craig_Bob", MIR.TLit (GA.Ident "Craig")), MIR.TLit (GA.Ident "Craig")) (ELit (LInt v), MIR.Type (GA.Ident "_Int")), MIR.Type (GA.Ident "Craig")) [ injectionCons "Craig_Bob" "Craig" [CIdent (GA.Ident "x")] (EId (GA.Ident "x", MIR.Type (GA.Ident "_Int")), MIR.Type (GA.Ident "_Int")) , injectionCons "Craig_Betty" "Craig" [CLit (LInt 5)] (int 2) , Injection (CIdent (GA.Ident "z")) (int 3) , -- , injectionInt 5 (int 6) injectionCatchAll (int 10) ] ] where injectionCons x y xs = Injection (CCons (GA.Ident x, MIR.Type (GA.Ident y)) xs) injectionInt x = Injection (CLit (LInt x)) injectionCatchAll = Injection CatchAll eCaseInt x xs = (ECase (MIR.TLit (MIR.Ident "_Int")) x xs, MIR.TLit (MIR.Ident "_Int")) int x = (ELit (LInt x), MIR.TLit (MIR.Ident "_Int")) -} {- | Compiles an AST and produces a LLVM Ir string. An easy way to actually "compile" this output is to Simply pipe it to LLI -} generateCode :: MIR.Program -> Err String generateCode (MIR.Program scs) = do let codegen = initCodeGenerator scs llvmIrToString . instructions <$> execStateT (compileScs scs) codegen compileScs :: [MIR.Def] -> CompilerState () compileScs [] = do -- as a last step create all the constructors -- //TODO maybe merge this with the data type match? c <- gets (Map.toList . constructors) mapM_ ( \((id, t), ci) -> do let t' = type2LlvmType t let x = BI.second type2LlvmType <$> argumentsCI ci emit $ Define FastCC t' id x top <- getNewVar ptr <- getNewVar -- allocated the primary type emit $ SetVariable top (Alloca t') -- set the first byte to the index of the constructor emit $ SetVariable ptr $ GetElementPtr t' (Ref t') (VIdent top I8) I64 (VInteger 0) I32 (VInteger 0) emit $ Store I8 (VInteger $ numCI ci) (Ref I8) ptr -- get a pointer of the correct type ptr' <- getNewVar emit $ SetVariable ptr' (Bitcast (Ref t') (VIdent top Ptr) (Ref $ CustomType id)) -- emit $ UnsafeRaw "\n" enumerateOneM_ ( \i (GA.Ident arg_n, arg_t) -> do let arg_t' = type2LlvmType arg_t emit $ Comment (toIr arg_t' <> " " <> arg_n <> " " <> show i) elemPtr <- getNewVar emit $ SetVariable elemPtr ( GetElementPtr (CustomType id) (Ref (CustomType id)) (VIdent ptr' Ptr) I64 (VInteger 0) I32 (VInteger i) ) emit $ Store arg_t' (VIdent (GA.Ident arg_n) arg_t') Ptr elemPtr ) (argumentsCI ci) -- emit $ UnsafeRaw "\n" -- load and return the constructed value emit $ Comment "Return the newly constructed value" load <- getNewVar emit $ SetVariable load (Load t' Ptr top) emit $ Ret t' (VIdent load t') emit DefineEnd modify $ \s -> s{variableCount = 0} ) c compileScs (MIR.DBind (MIR.Bind (name, _t) args exp) : xs) = do emit $ UnsafeRaw "\n" emit . Comment $ show name <> ": " <> show exp let args' = map (second type2LlvmType) args emit $ Define FastCC I64 {-(type2LlvmType t_return)-} name args' functionBody <- exprToValue exp if name == "main" then mapM_ emit $ mainContent functionBody else emit $ Ret I64 functionBody emit DefineEnd modify $ \s -> s{variableCount = 0} compileScs xs compileScs (MIR.DData (MIR.Data typ ts) : xs) = do let (Ident outer_id) = extractTypeName typ let biggestVariant = maximum $ sum <$> (\(Constructor _ t) -> typeByteSize . type2LlvmType . snd <$> t) <$> ts emit $ LIR.Type (coerce outer_id) [I8, Array biggestVariant I8] mapM_ ( \(Constructor (GA.UIdent inner_id) fi) -> do emit $ LIR.Type (GA.Ident $ outer_id <> "_" <> inner_id) (I8 : map type2LlvmType (snd <$> fi)) ) ts compileScs xs mainContent :: LLVMValue -> [LLVMIr] mainContent var = [ UnsafeRaw $ -- "%2 = alloca %Craig\n" <> -- " store %Craig %1, ptr %2\n" <> -- " %3 = bitcast %Craig* %2 to i72*\n" <> -- " %4 = load i72, ptr %3\n" <> -- " call i32 (ptr, ...) @printf(ptr noundef @.str, i72 noundef %4)\n" "call i32 (ptr, ...) @printf(ptr noundef @.str, i64 noundef " <> toIr var <> ")\n" , -- , SetVariable (GA.Ident "p") (Icmp LLEq I64 (VInteger 2) (VInteger 2)) -- , BrCond (VIdent (GA.Ident "p")) (GA.Ident "b_1") (GA.Ident "b_2") -- , Label (GA.Ident "b_1") -- , UnsafeRaw -- "call i32 (ptr, ...) @printf(ptr noundef @.str, i64 noundef 1)\n" -- , Br (GA.Ident "end") -- , Label (GA.Ident "b_2") -- , UnsafeRaw -- "call i32 (ptr, ...) @printf(ptr noundef @.str, i64 noundef 2)\n" -- , Br (GA.Ident "end") -- , Label (GA.Ident "end") Ret I64 (VInteger 0) ] defaultStart :: [LLVMIr] defaultStart = [ UnsafeRaw "target triple = \"x86_64-pc-linux-gnu\"\n" , UnsafeRaw "target datalayout = \"e-m:o-i64:64-f80:128-n8:16:32:64-S128\"\n" , UnsafeRaw "@.str = private unnamed_addr constant [3 x i8] c\"%x\n\", align 1\n" , UnsafeRaw "declare i32 @printf(ptr noalias nocapture, ...)\n" ] compileExp :: ExpT -> CompilerState () compileExp (MIR.ELit lit,t) = emitLit lit compileExp (MIR.EAdd e1 e2,t) = emitAdd t e1 e2 -- compileExp (ESub t e1 e2) = emitSub t e1 e2 compileExp (MIR.EId name,t) = emitIdent name compileExp (MIR.EApp e1 e2,t) = emitApp t e1 e2 -- compileExp (EAbs t ti e) = emitAbs t ti e compileExp (MIR.ELet binds e,t) = undefined -- emitLet binds (fst e) compileExp (MIR.ECase e cs,t) = emitECased t e (map (t,) cs) -- go (EMul e1 e2) = emitMul e1 e2 -- go (EDiv e1 e2) = emitDiv e1 e2 -- go (EMod e1 e2) = emitMod e1 e2 --- aux functions --- emitECased :: MIR.Type -> ExpT -> [(MIR.Type, Injection)] -> CompilerState () emitECased t e cases = do let cs = snd <$> cases let ty = type2LlvmType t let rt = type2LlvmType (snd e) vs <- exprToValue e lbl <- getNewLabel let label = GA.Ident $ "escape_" <> show lbl stackPtr <- getNewVar emit $ SetVariable stackPtr (Alloca ty) mapM_ (emitCases rt ty label stackPtr vs) cs emit $ Label label res <- getNewVar emit $ SetVariable res (Load ty Ptr stackPtr) where emitCases :: LLVMType -> LLVMType -> GA.Ident -> GA.Ident -> LLVMValue -> Injection -> CompilerState () emitCases rt ty label stackPtr vs (Injection (MIR.InitConstructor consId cs, t) exp) = do cons <- gets constructors let r = fromJust $ Map.lookup (coerce consId, t) cons lbl_failPos <- (\x -> GA.Ident $ "failed_" <> show x) <$> getNewLabel lbl_succPos <- (\x -> GA.Ident $ "success_" <> show x) <$> getNewLabel consVal <- getNewVar emit $ SetVariable consVal (ExtractValue rt vs 0) consCheck <- getNewVar emit $ SetVariable consCheck (Icmp LLEq I8 (VIdent consVal I8) (VInteger $ numCI r)) emit $ BrCond (VIdent consCheck ty) lbl_succPos lbl_failPos emit $ Label lbl_succPos castPtr <- getNewVar castedPtr <- getNewVar casted <- getNewVar emit $ SetVariable castPtr (Alloca rt) emit $ Store rt vs Ptr castPtr emit $ SetVariable castedPtr (Bitcast Ptr (VIdent castPtr Ptr) Ptr) emit $ SetVariable casted (Load (CustomType (coerce consId)) Ptr castedPtr) val <- exprToValue exp -- enumerateOneM_ -- (\i c -> do -- case c of -- CIdent x -> do -- emit . Comment $ "ident " <> show x -- emit $ SetVariable x (ExtractValue (CustomType (fst consId)) (VIdent casted Ptr) i) -- emit $ Store ty val Ptr stackPtr -- CCons x cs -> error "nested constructor" -- CLit l -> do -- testVar <- getNewVar -- emit $ SetVariable testVar (ExtractValue (CustomType (fst consId)) (VIdent casted Ptr) i) -- case l of -- LInt l -> emit $ Icmp LLEq I64 (VIdent testVar Ptr) (VInteger l) -- LChar c -> emit $ Icmp LLEq I8 (VIdent testVar Ptr) (VChar c) -- CCatch -> emit . Comment $ "Catch all" -- emit . Comment $ "return this " <> toIr val -- emit . Comment . show $ c -- emit . Comment . show $ i -- ) -- cs -- emit $ Store ty val Ptr stackPtr emit $ Br label emit $ Label lbl_failPos emitCases rt ty label stackPtr vs (Injection (MIR.InitLit i, _) exp) = do let i' = case i of GA.LInt i -> VInteger i GA.LChar i -> VChar i ns <- getNewVar lbl_failPos <- (\x -> GA.Ident $ "failed_" <> show x) <$> getNewLabel lbl_succPos <- (\x -> GA.Ident $ "success_" <> show x) <$> getNewLabel emit $ SetVariable ns (Icmp LLEq ty vs i') emit $ BrCond (VIdent ns ty) lbl_succPos lbl_failPos emit $ Label lbl_succPos val <- exprToValue exp emit $ Store ty val Ptr stackPtr emit $ Br label emit $ Label lbl_failPos -- emitCases rt ty label stackPtr vs (Injection (MIR.CIdent id) exp) = do -- -- //TODO this is pretty disgusting and would heavily benefit from a rewrite -- valPtr <- getNewVar -- emit $ SetVariable valPtr (Alloca rt) -- emit $ Store rt vs Ptr valPtr -- emit $ SetVariable id (Load rt Ptr valPtr) -- increaseVarCount -- val <- exprToValue (fst exp) -- emit $ Store ty val Ptr stackPtr -- emit $ Br label emitCases _ ty label stackPtr _ (Injection (MIR.InitCatch, _) exp) = do val <- exprToValue exp emit $ Store ty val Ptr stackPtr emit $ Br label --emitLet :: Bind -> Exp -> CompilerState () emitLet xs e = do emit $ Comment $ concat [ "ELet (" , show xs , " = " , show e , ") is not implemented!" ] emitApp :: MIR.Type -> ExpT -> ExpT -> CompilerState () emitApp t e1 e2 = appEmitter e1 e2 [] where appEmitter :: ExpT -> ExpT -> [ExpT] -> CompilerState () appEmitter e1 e2 stack = do let newStack = e2 : stack case e1 of (MIR.EApp e1' e2', t) -> appEmitter e1' e2' newStack (MIR.EId name, t) -> do args <- traverse exprToValue newStack vs <- getNewVar funcs <- gets functions consts <- gets constructors let visibility = fromMaybe Local $ Global <$ Map.lookup (name, t) consts <|> Global <$ Map.lookup (name,t) funcs -- this piece of code could probably be improved, i.e remove the double `const Global` args' = map (first valueGetType . dupe) args call = Call FastCC (type2LlvmType t) visibility name args' emit $ SetVariable vs call x -> error $ "The unspeakable happened: " <> show x emitIdent :: GA.Ident -> CompilerState () emitIdent id = do -- !!this should never happen!! emit $ Comment "This should not have happened!" emit $ Variable id emit $ UnsafeRaw "\n" emitLit :: MIR.Lit -> CompilerState () emitLit i = do -- !!this should never happen!! let (i', t) = case i of (MIR.LInt i'') -> (VInteger i'', I64) (MIR.LChar i'') -> (VChar i'', I8) varCount <- getNewVar emit $ Comment "This should not have happened!" emit $ SetVariable (GA.Ident (show varCount)) (Add t i' (VInteger 0)) emitAdd :: MIR.Type -> ExpT -> ExpT -> CompilerState () emitAdd t e1 e2 = do v1 <- exprToValue e1 v2 <- exprToValue e2 v <- getNewVar emit $ SetVariable (GA.Ident $ show v) (Add (type2LlvmType t) v1 v2) emitSub :: MIR.Type -> ExpT -> ExpT -> CompilerState () emitSub t e1 e2 = do v1 <- exprToValue e1 v2 <- exprToValue e2 v <- getNewVar emit $ SetVariable v (Sub (type2LlvmType t) v1 v2) exprToValue :: ExpT -> CompilerState LLVMValue exprToValue = \case (MIR.ELit i, t) -> pure $ case i of (MIR.LInt i) -> VInteger i (MIR.LChar i) -> VChar i (MIR.EId name, t) -> do funcs <- gets functions case Map.lookup (name, t) funcs of Just fi -> do if numArgs fi == 0 then do vc <- getNewVar emit $ SetVariable vc (Call FastCC (type2LlvmType t) Global name []) pure $ VIdent vc (type2LlvmType t) else pure $ VFunction name Global (type2LlvmType t) Nothing -> pure $ VIdent name (type2LlvmType t) e -> do compileExp e v <- getVarCount pure $ VIdent (GA.Ident $ show v) (getType e) type2LlvmType :: MIR.Type -> LLVMType type2LlvmType (MIR.TLit id@(Ident name)) = case name of "Int" -> I64 _ -> CustomType id type2LlvmType (MIR.TFun t xs) = do let (t', xs') = function2LLVMType xs [type2LlvmType t] Function t' xs' where function2LLVMType :: Type -> [LLVMType] -> (LLVMType, [LLVMType]) function2LLVMType (TFun t xs) s = function2LLVMType xs (type2LlvmType t : s) function2LLVMType x s = (type2LlvmType x, s) getType :: ExpT -> LLVMType getType (_, t) = type2LlvmType t extractTypeName :: MIR.Type -> Ident extractTypeName (MIR.TLit id) = id extractTypeName (MIR.TFun t xs) = let (Ident i) = extractTypeName t (Ident is) = extractTypeName xs in Ident $ i <> "_$_" <> is valueGetType :: LLVMValue -> LLVMType valueGetType (VInteger _) = I64 valueGetType (VChar _) = I8 valueGetType (VIdent _ t) = t valueGetType (VConstant s) = Array (fromIntegral $ length s) I8 valueGetType (VFunction _ _ t) = t typeByteSize :: LLVMType -> Integer typeByteSize I1 = 1 typeByteSize I8 = 1 typeByteSize I32 = 4 typeByteSize I64 = 8 typeByteSize Ptr = 8 typeByteSize (Ref _) = 8 typeByteSize (Function _ _) = 8 typeByteSize (Array n t) = n * typeByteSize t typeByteSize (CustomType _) = 8 enumerateOneM_ :: Monad m => (Integer -> a -> m b) -> [a] -> m () enumerateOneM_ f = foldM_ (\i a -> f i a >> pure (i + 1)) 1