module LLVM.Internal.OrcJIT.LinkingLayer where
import LLVM.Prelude
import Control.Exception
import Control.Monad.AnyCont
import Control.Monad.IO.Class
import Data.IORef
import Foreign.Ptr
import LLVM.Internal.OrcJIT
import LLVM.Internal.Coding
import LLVM.Internal.ObjectFile
import qualified LLVM.Internal.FFI.PtrHierarchy as FFI
import qualified LLVM.Internal.FFI.OrcJIT.LinkingLayer as FFI
class LinkingLayer l where
getLinkingLayer :: l -> Ptr FFI.LinkingLayer
getCleanups :: l -> IORef [IO ()]
disposeLinkingLayer :: LinkingLayer l => l -> IO ()
disposeLinkingLayer l = do
FFI.disposeLinkingLayer (getLinkingLayer l)
sequence_ =<< readIORef (getCleanups l)
addObjectFile :: LinkingLayer l => l -> ObjectFile -> SymbolResolver
-> IO FFI.ObjectHandle
addObjectFile linkingLayer (ObjectFile obj) resolver = flip runAnyContT return $ do
resolverAct <- encodeM resolver
resolver' <- liftIO $ resolverAct (getCleanups linkingLayer)
errMsg <- alloca
liftIO $
FFI.addObjectFile
(getLinkingLayer linkingLayer)
obj
resolver'
errMsg
data ObjectLinkingLayer = ObjectLinkingLayer {
linkingLayer :: !(Ptr FFI.ObjectLinkingLayer),
cleanupActions :: !(IORef [IO ()])
}
instance LinkingLayer ObjectLinkingLayer where
getLinkingLayer (ObjectLinkingLayer ptr _) = FFI.upCast ptr
getCleanups = cleanupActions
newObjectLinkingLayer :: IO ObjectLinkingLayer
newObjectLinkingLayer = do
linkingLayer <- FFI.createObjectLinkingLayer
cleanups <- liftIO (newIORef [])
return $ ObjectLinkingLayer linkingLayer cleanups
withObjectLinkingLayer :: (ObjectLinkingLayer -> IO a) -> IO a
withObjectLinkingLayer = bracket newObjectLinkingLayer disposeLinkingLayer