{-# LANGUAGE TemplateHaskell, ScopedTypeVariables, MultiParamTypeClasses #-} -- | This Haskell module is for/of functions for handling LLVM modules. module LLVM.General.Internal.Module where import Control.Monad.Trans import Control.Monad.State import Control.Monad.Error import Control.Monad.AnyCont import Control.Applicative import Control.Exception import Foreign.Ptr import Foreign.C import Data.IORef import qualified Data.ByteString as BS import qualified Data.Map as Map import qualified LLVM.General.Internal.FFI.Assembly as FFI import qualified LLVM.General.Internal.FFI.Builder as FFI import qualified LLVM.General.Internal.FFI.Bitcode as FFI import qualified LLVM.General.Internal.FFI.Function as FFI import qualified LLVM.General.Internal.FFI.GlobalAlias as FFI import qualified LLVM.General.Internal.FFI.GlobalValue as FFI import qualified LLVM.General.Internal.FFI.GlobalVariable as FFI import qualified LLVM.General.Internal.FFI.Iterate as FFI import qualified LLVM.General.Internal.FFI.LLVMCTypes as FFI import qualified LLVM.General.Internal.FFI.MemoryBuffer as FFI import qualified LLVM.General.Internal.FFI.Metadata as FFI import qualified LLVM.General.Internal.FFI.Module as FFI import qualified LLVM.General.Internal.FFI.PtrHierarchy as FFI import qualified LLVM.General.Internal.FFI.RawOStream as FFI import qualified LLVM.General.Internal.FFI.Target as FFI import qualified LLVM.General.Internal.FFI.Value as FFI import LLVM.General.Internal.BasicBlock import LLVM.General.Internal.Coding import LLVM.General.Internal.Context import LLVM.General.Internal.DecodeAST import LLVM.General.Internal.Diagnostic import LLVM.General.Internal.EncodeAST import LLVM.General.Internal.Function import LLVM.General.Internal.Global import LLVM.General.Internal.Instruction () import qualified LLVM.General.Internal.MemoryBuffer as MB import LLVM.General.Internal.Metadata import LLVM.General.Internal.Operand import LLVM.General.Internal.RawOStream import LLVM.General.Internal.String import LLVM.General.Internal.Target import LLVM.General.Internal.Type import LLVM.General.Internal.Value import LLVM.General.DataLayout import LLVM.General.Diagnostic import qualified LLVM.General.AST as A import qualified LLVM.General.AST.DataLayout as A import qualified LLVM.General.AST.AddrSpace as A import qualified LLVM.General.AST.Global as A.G -- | newtype Module = Module (Ptr FFI.Module) -- | A newtype to distinguish strings used for paths from other strings newtype File = File FilePath deriving (Eq, Ord, Read, Show) instance Error (Either String Diagnostic) where strMsg = Left genCodingInstance [t| Bool |] ''FFI.LinkerMode [ (FFI.linkerModeDestroySource, False), (FFI.linkerModePreserveSource, True) ] -- | link LLVM modules - move or copy parts of a source module into a destination module. -- Note that this operation is not commutative - not only concretely (e.g. the destination module -- is modified, becoming the result) but abstractly (e.g. unused private globals in the source -- module do not appear in the result, but similar globals in the destination remain). linkModules :: Bool -- ^ True to leave the right module unmodified, False to cannibalize it (for efficiency's sake). -> Module -- ^ The module into which to link -> Module -- ^ The module to link into the other (and cannibalize or not) -> ErrorT String IO () linkModules preserveRight (Module m) (Module m') = flip runAnyContT return $ do preserveRight <- encodeM preserveRight msgPtr <- alloca result <- decodeM =<< (liftIO $ FFI.linkModules m m' preserveRight msgPtr) when result $ fail =<< decodeM msgPtr class LLVMAssemblyInput s where llvmAssemblyMemoryBuffer :: (MonadIO e, MonadAnyCont IO e) => s -> e (FFI.OwnerTransfered (Ptr FFI.MemoryBuffer)) instance LLVMAssemblyInput (String, String) where llvmAssemblyMemoryBuffer (id, s) = do UTF8ByteString bs <- encodeM s encodeM (MB.Bytes id bs) instance LLVMAssemblyInput String where llvmAssemblyMemoryBuffer s = llvmAssemblyMemoryBuffer ("", s) instance LLVMAssemblyInput File where llvmAssemblyMemoryBuffer (File p) = encodeM (MB.File p) -- | parse 'Module' from LLVM assembly withModuleFromLLVMAssembly :: LLVMAssemblyInput s => Context -> s -> (Module -> IO a) -> ErrorT (Either String Diagnostic) IO a withModuleFromLLVMAssembly (Context c) s f = flip runAnyContT return $ do mb <- llvmAssemblyMemoryBuffer s smDiag <- anyContToM withSMDiagnostic m <- anyContToM $ bracket (FFI.parseLLVMAssembly c mb smDiag) FFI.disposeModule when (m == nullPtr) $ throwError . Right =<< liftIO (getDiagnostic smDiag) liftIO $ f (Module m) -- | generate LLVM assembly from a 'Module' moduleLLVMAssembly :: Module -> IO String moduleLLVMAssembly (Module m) = do resultRef <- newIORef Nothing let saveBuffer :: Ptr CChar -> CSize -> IO () saveBuffer start size = do r <- decodeM (start, fromIntegral size) writeIORef resultRef (Just r) FFI.withBufferRawOStream saveBuffer $ FFI.writeLLVMAssembly m Just s <- readIORef resultRef return s -- | write LLVM assembly for a 'Module' to a file writeLLVMAssemblyToFile :: File -> Module -> ErrorT String IO () writeLLVMAssemblyToFile (File path) (Module m) = flip runAnyContT return $ do withFileRawOStream path False False $ liftIO . FFI.writeLLVMAssembly m class BitcodeInput b where bitcodeMemoryBuffer :: (MonadIO e, MonadAnyCont IO e) => b -> e (Ptr FFI.MemoryBuffer) instance BitcodeInput (String, BS.ByteString) where bitcodeMemoryBuffer (s, bs) = encodeM (MB.Bytes s bs) instance BitcodeInput File where bitcodeMemoryBuffer (File p) = encodeM (MB.File p) -- | parse 'Module' from LLVM bitcode withModuleFromBitcode :: BitcodeInput b => Context -> b -> (Module -> IO a) -> ErrorT String IO a withModuleFromBitcode (Context c) b f = flip runAnyContT return $ do mb <- bitcodeMemoryBuffer b msgPtr <- alloca m <- anyContToM $ bracket (FFI.parseBitcode c mb msgPtr) FFI.disposeModule when (m == nullPtr) $ fail =<< decodeM msgPtr liftIO $ f (Module m) -- | generate LLVM bitcode from a 'Module' moduleBitcode :: Module -> IO BS.ByteString moduleBitcode (Module m) = withBufferRawOStream (liftIO . FFI.writeBitcode m) -- | write LLVM bitcode from a 'Module' into a file writeBitcodeToFile :: File -> Module -> ErrorT String IO () writeBitcodeToFile (File path) (Module m) = flip runAnyContT return $ do withFileRawOStream path False True $ liftIO . FFI.writeBitcode m targetMachineEmit :: FFI.CodeGenFileType -> TargetMachine -> Module -> Ptr FFI.RawOStream -> ErrorT String IO () targetMachineEmit fileType (TargetMachine tm) (Module m) os = flip runAnyContT return $ do msgPtr <- alloca r <- decodeM =<< (liftIO $ FFI.targetMachineEmit tm m fileType msgPtr os) when r $ fail =<< decodeM msgPtr emitToFile :: FFI.CodeGenFileType -> TargetMachine -> File -> Module -> ErrorT String IO () emitToFile fileType tm (File path) m = flip runAnyContT return $ do withFileRawOStream path False True $ targetMachineEmit fileType tm m emitToByteString :: FFI.CodeGenFileType -> TargetMachine -> Module -> ErrorT String IO BS.ByteString emitToByteString fileType tm m = flip runAnyContT return $ do withBufferRawOStream $ targetMachineEmit fileType tm m -- | write target-specific assembly directly into a file writeTargetAssemblyToFile :: TargetMachine -> File -> Module -> ErrorT String IO () writeTargetAssemblyToFile = emitToFile FFI.codeGenFileTypeAssembly -- | produce target-specific assembly as a 'String' moduleTargetAssembly :: TargetMachine -> Module -> ErrorT String IO String moduleTargetAssembly tm m = decodeM . UTF8ByteString =<< emitToByteString FFI.codeGenFileTypeAssembly tm m -- | produce target-specific object code as a 'ByteString' moduleObject :: TargetMachine -> Module -> ErrorT String IO BS.ByteString moduleObject = emitToByteString FFI.codeGenFileTypeObject -- | write target-specific object code directly into a file writeObjectToFile :: TargetMachine -> File -> Module -> ErrorT String IO () writeObjectToFile = emitToFile FFI.codeGenFileTypeObject setTargetTriple :: Ptr FFI.Module -> String -> EncodeAST () setTargetTriple m t = do t <- encodeM t liftIO $ FFI.setTargetTriple m t getTargetTriple :: Ptr FFI.Module -> IO (Maybe String) getTargetTriple m = do s <- decodeM =<< liftIO (FFI.getTargetTriple m) return $ if s == "" then Nothing else Just s setDataLayout :: Ptr FFI.Module -> A.DataLayout -> EncodeAST () setDataLayout m dl = do s <- encodeM (dataLayoutToString dl) liftIO $ FFI.setDataLayout m s getDataLayout :: Ptr FFI.Module -> IO (Maybe A.DataLayout) getDataLayout m = parseDataLayout <$> (decodeM =<< FFI.getDataLayout m) type P a = a -> a -- | Build an LLVM.General.'Module' from a LLVM.General.AST.'LLVM.General.AST.Module' - i.e. -- lower an AST from Haskell into C++ objects. withModuleFromAST :: Context -> A.Module -> (Module -> IO a) -> ErrorT String IO a withModuleFromAST context@(Context c) (A.Module moduleId dataLayout triple definitions) f = runEncodeAST context $ do moduleId <- encodeM moduleId m <- anyContToM $ bracket (FFI.moduleCreateWithNameInContext moduleId c) FFI.disposeModule maybe (return ()) (setDataLayout m) dataLayout maybe (return ()) (setTargetTriple m) triple let sequencePhases :: EncodeAST [EncodeAST (EncodeAST (EncodeAST (EncodeAST ())))] -> EncodeAST () sequencePhases l = (l >>= (sequence >=> sequence >=> sequence >=> sequence)) >> (return ()) sequencePhases $ forM definitions $ \d -> case d of A.TypeDefinition n t -> do t' <- createNamedType n defineType n t' return $ do maybe (return ()) (setNamedType t') t return . return . return $ return () A.MetadataNodeDefinition i os -> return . return $ do t <- liftIO $ FFI.createTemporaryMDNodeInContext c defineMDNode i t return $ do n <- encodeM (A.MetadataNode os) liftIO $ FFI.replaceAllUsesWith (FFI.upCast t) (FFI.upCast n) defineMDNode i n liftIO $ FFI.destroyTemporaryMDNode t return $ return () A.NamedMetadataDefinition n ids -> return . return . return . return $ do n <- encodeM n ids <- encodeM (map A.MetadataNodeReference ids) nm <- liftIO $ FFI.getOrAddNamedMetadata m n liftIO $ FFI.namedMetadataAddOperands nm ids return () A.ModuleInlineAssembly s -> do s <- encodeM s liftIO $ FFI.moduleAppendInlineAsm m (FFI.ModuleAsm s) return . return . return . return $ return () A.GlobalDefinition g -> return . phase $ do eg' :: EncodeAST (Ptr FFI.GlobalValue) <- case g of g@(A.GlobalVariable { A.G.name = n }) -> do typ <- encodeM (A.G.type' g) g' <- liftIO $ withName n $ \gName -> FFI.addGlobalInAddressSpace m typ gName (fromIntegral ((\(A.AddrSpace a) -> a) $ A.G.addrSpace g)) defineGlobal n g' liftIO $ do tl <- encodeM (A.G.isThreadLocal g) FFI.setThreadLocal g' tl hua <- encodeM (A.G.hasUnnamedAddr g) FFI.setUnnamedAddr (FFI.upCast g') hua ic <- encodeM (A.G.isConstant g) FFI.setGlobalConstant g' ic return $ do maybe (return ()) ((liftIO . FFI.setInitializer g') <=< encodeM) (A.G.initializer g) setSection g' (A.G.section g) setAlignment g' (A.G.alignment g) return (FFI.upCast g') (a@A.G.GlobalAlias { A.G.name = n }) -> do typ <- encodeM (A.G.type' a) a' <- liftIO $ withName n $ \name -> FFI.justAddAlias m typ name defineGlobal n a' return $ do (liftIO . FFI.setAliasee a') =<< encodeM (A.G.aliasee a) return (FFI.upCast a') (A.Function _ _ cc rAttrs resultType fName (args,isVarArgs) attrs _ _ gc blocks) -> do typ <- encodeM $ A.FunctionType resultType (map (\(A.Parameter t _ _) -> t) args) isVarArgs f <- liftIO . withName fName $ \fName -> FFI.addFunction m fName typ defineGlobal fName f cc <- encodeM cc liftIO $ FFI.setFunctionCallConv f cc rAttrs <- encodeM rAttrs liftIO $ FFI.addFunctionRetAttr f rAttrs liftIO $ setFunctionAttrs f attrs setSection f (A.G.section g) setAlignment f (A.G.alignment g) setGC f gc forM blocks $ \(A.BasicBlock bName _ _) -> do b <- liftIO $ withName bName $ \bName -> FFI.appendBasicBlockInContext c f bName defineBasicBlock fName bName b phase $ do let nParams = length args ps <- allocaArray nParams liftIO $ FFI.getParams f ps params <- peekArray nParams ps forM (zip args params) $ \(A.Parameter _ n attrs, p) -> do defineLocal n p n <- encodeM n liftIO $ FFI.setValueName (FFI.upCast p) n unless (null attrs) $ do attrs <- encodeM attrs liftIO $ FFI.addAttribute p attrs return () return () finishInstrs <- forM blocks $ \(A.BasicBlock bName namedInstrs term) -> do b <- encodeM bName (do builder <- gets encodeStateBuilder liftIO $ FFI.positionBuilderAtEnd builder b) finishes <- mapM encodeM namedInstrs :: EncodeAST [EncodeAST ()] (encodeM term :: EncodeAST (Ptr FFI.Instruction)) return (sequence_ finishes) sequence_ finishInstrs locals <- gets $ Map.toList . encodeStateLocals forM [ n | (n, ForwardValue _) <- locals ] $ \n -> failAsUndefined "local" n return (FFI.upCast f) return $ do g' <- eg' setLinkage g' (A.G.linkage g) setVisibility g' (A.G.visibility g) return $ return () liftIO $ f (Module m) -- | Get an LLVM.General.AST.'LLVM.General.AST.Module' from a LLVM.General.'Module' - i.e. -- raise C++ objects into an Haskell AST. moduleAST :: Module -> IO A.Module moduleAST (Module mod) = runDecodeAST $ do c <- return Context `ap` liftIO (FFI.getModuleContext mod) getMetadataKindNames c return A.Module `ap` (liftIO $ decodeM =<< FFI.getModuleIdentifier mod) `ap` (liftIO $ getDataLayout mod) `ap` (liftIO $ do s <- decodeM <=< FFI.getTargetTriple $ mod return $ if s == "" then Nothing else Just s) `ap` ( do gs <- map A.GlobalDefinition . concat <$> (join . liftM sequence . sequence) [ do ffiGlobals <- liftIO $ FFI.getXs (FFI.getFirstGlobal mod) FFI.getNextGlobal liftM sequence . forM ffiGlobals $ \g -> do A.PointerType t as <- typeOf g n <- getGlobalName g return $ return A.GlobalVariable `ap` return n `ap` getLinkage g `ap` getVisibility g `ap` (liftIO $ decodeM =<< FFI.isThreadLocal g) `ap` return as `ap` (liftIO $ decodeM =<< FFI.hasUnnamedAddr (FFI.upCast g)) `ap` (liftIO $ decodeM =<< FFI.isGlobalConstant g) `ap` return t `ap` (do i <- liftIO $ FFI.getInitializer g if i == nullPtr then return Nothing else Just <$> decodeM i) `ap` getSection g `ap` getAlignment g, do ffiAliases <- liftIO $ FFI.getXs (FFI.getFirstAlias mod) FFI.getNextAlias liftM sequence . forM ffiAliases $ \a -> do n <- getGlobalName a return $ return A.G.GlobalAlias `ap` return n `ap` getLinkage a `ap` getVisibility a `ap` typeOf a `ap` (decodeM =<< (liftIO $ FFI.getAliasee a)), do ffiFunctions <- liftIO $ FFI.getXs (FFI.getFirstFunction mod) FFI.getNextFunction liftM sequence . forM ffiFunctions $ \f -> localScope $ do A.PointerType (A.FunctionType returnType _ isVarArg) _ <- typeOf f n <- getGlobalName f parameters <- getParameters f decodeBlocks <- do ffiBasicBlocks <- liftIO $ FFI.getXs (FFI.getFirstBasicBlock f) FFI.getNextBasicBlock liftM sequence . forM ffiBasicBlocks $ \b -> do n <- getLocalName b decodeInstructions <- getNamedInstructions b decodeTerminator <- getBasicBlockTerminator b return $ return A.BasicBlock `ap` return n `ap` decodeInstructions `ap` decodeTerminator return $ return A.Function `ap` getLinkage f `ap` getVisibility f `ap` (liftIO $ decodeM =<< FFI.getFunctionCallConv f) `ap` (liftIO $ decodeM =<< FFI.getFunctionRetAttr f) `ap` return returnType `ap` return n `ap` return (parameters, isVarArg) `ap` (liftIO $ getFunctionAttrs f) `ap` getSection f `ap` getAlignment f `ap` getGC f `ap` decodeBlocks ] tds <- getStructDefinitions ias <- decodeM =<< liftIO (FFI.moduleGetInlineAsm mod) nmds <- do ffiNamedMetadataNodes <- liftIO $ FFI.getXs (FFI.getFirstNamedMetadata mod) FFI.getNextNamedMetadata forM ffiNamedMetadataNodes $ \nm -> scopeAnyCont $ do n <- liftIO $ FFI.getNamedMetadataNumOperands nm os <- allocaArray n liftIO $ FFI.getNamedMetadataOperands nm os return A.NamedMetadataDefinition `ap` (decodeM $ FFI.getNamedMetadataName nm) `ap` liftM (map (\(A.MetadataNodeReference mid) -> mid)) (decodeM (n, os)) mds <- getMetadataDefinitions return $ tds ++ ias ++ gs ++ nmds ++ mds )