{-# LANGUAGE ForeignFunctionInterface, NoMonomorphismRestriction, MultiParamTypeClasses, CPP #-} module Context(module Context.Types ,module Lang ,withDefaultContext ,languageState ,doTransform ,getAddressComp ,execCode) where import Bindings.Posix.Sys.Mman import Bindings.Posix.Unistd import Context.Language as Lang import Context.Types import Control.Category ((>>>)) import Data.ByteString.Internal import Data.ByteString.Unsafe import Data.Functor.Identity import Data.IORef import Data.List import Data.Maybe import Foreign hiding (unsafePerformIO,unsafeForeignPtrToPtr,void) import Foreign.C import Foreign.ForeignPtr.Unsafe import Format import ID import My.Control.Monad import My.Control.Monad.State import My.Prelude import Options import PCode import Specialize import Specialize.Architecture import Syntax import System.IO.Unsafe (unsafePerformIO) import qualified Data.Bimap as BM import qualified Data.ByteString as B import qualified Data.Map as M foreign import ccall "mprotect" mprotect :: Ptr () -> CSize -> CInt -> IO CInt foreign import ccall "dynamic" mkProc :: FunPtr (Ptr() -> IO ()) -> Ptr() -> IO () foreign import ccall "dynamic" mkFunSize :: FunPtr (Ptr() -> IO Int) -> Ptr() -> IO Int foreign import ccall "dynamic" mkFunInit :: FunPtr (Ptr () -> Ptr() -> IO ()) -> Ptr() -> Ptr () -> IO () foreign import ccall "dynamic" mkFunTransform :: FunPtr (Ptr () -> Ptr() -> IO (Ptr ())) -> Ptr() -> Ptr () -> IO (Ptr ()) funPtrToInteger f = fromIntegral $ ptrToIntPtr $ castFunPtrToPtr f exportAlpha stub ptr = unsafePerformIO $ do unsafeUseAsCStringLen (stub $ funPtrToInteger ptr) $ \(src,size) -> do ptr <- mallocForeignPtrBytes size withForeignPtr ptr $ \dst -> copyBytes (castPtr dst) src size return ptr initialBindings = [(n,Left $ Builtin b) | (b,n) <- bNames] ++ [ ("alter" ,Left $ Axiom XAlter), ("bind" ,Left $ Axiom XBind), ("choose" ,Left $ Axiom XChoose), ("<-" ,Left $ Axiom XRestart), ("->" ,Left $ Axiom XReturn), ("do" ,Left $ Axiom XDo), ("lang" ,Left $ Axiom XLang), ("verb" ,Left $ Axiom XVerb), ("noun" ,Left $ Axiom XNoun), ("id" ,Left $ Axiom XID), ("@" ,Left $ Axiom XAddr), ("#" ,Left $ Axiom XSize)] ++ [ ("alpha/c@" , Right $ exportAlpha callStub1 alpha_compAddr), ("alpha/create-symbol" , Right $ exportAlpha callStub0 alpha_createSym), ("alpha/number-symbol" , Right $ exportAlpha callStub1 alpha_numSym), ("alpha/symbol-name" , Right $ exportAlpha callStub1 alpha_symName), ("alpha/name-symbol" , Right $ exportAlpha callStub1 alpha_nameSym), ("alpha/set-transform" , Right $ exportAlpha callStub1 alpha_setTransform), ("alpha/allocate" , Right $ exportAlpha callStub1 alpha_allocate), ("alpha/free" , Right $ exportAlpha callStub1 alpha_free), ("alpha/list" , Right $ exportAlpha callStub0 alpha_printList), ("alpha/lang" , Right $ exportAlpha callStub0 alpha_printLang), ("alpha/help" , Right $ exportAlpha callStub0 alpha_printHelp), ("alpha/print-OK" , Right $ exportAlpha callStub0 alpha_printOK), ("alpha/print-num" , Right $ exportAlpha callStub1 alpha_printNum) ] #define str(x) #x #define ALPHA_EXPORT(fun,t) foreign export ccall str(__alpha_##fun) __alpha_##fun :: t ; foreign import ccall str(&__alpha_##fun) alpha_##fun :: FunPtr (t) ; __alpha_##fun ALPHA_EXPORT(setTransform,Ptr () -> IO ()) fun = modify $ \c -> c { transform = Just fun } ALPHA_EXPORT(compAddr,ID -> IO Int) id = readIORef compAddrRef >>= ($id) ALPHA_EXPORT(symName,ID -> IO (Ptr Word8)) sym = do n <- gets (lookupSymName sym . language) ret <- newArray0 0 (map c2w $ fromMaybe "" n) return ret ALPHA_EXPORT(nameSym,Ptr Word8 -> IO ID) p = do l <- peekArray0 0 p viewState language_ (internSym $ map w2c l) ALPHA_EXPORT(createSym,IO ID) = viewState language_ createSym ALPHA_EXPORT(numSym,Int -> IO ID) = viewState language_ . internSym . show ALPHA_EXPORT(allocate,Int -> IO (Ptr())) = mallocBytes ALPHA_EXPORT(free,Ptr() -> IO ()) = free ALPHA_EXPORT(printHelp,IO()) = Prelude.putStrLn helpMsg ALPHA_EXPORT(printOK,IO()) = Prelude.putStrLn "OK" ALPHA_EXPORT(printNum,Int -> IO()) n = print (intPtrToPtr $ fromIntegral n) ALPHA_EXPORT(printList,IO()) = do syms <- getting (language_ >>> syms_) putStrLn $ intercalate " " (map fst $ BM.toList syms) ALPHA_EXPORT(printLang,IO()) = getting language_ >>= print compAddrRef = unsafePerformIO $ newIORef (undefined :: ID -> IO Int) contextRef = unsafePerformIO $ newIORef (error "Undefined context" :: Context) instance MonadState Context IO where get = readIORef contextRef put = writeIORef contextRef initialContext entry = C lang jitA M.empty (fromIntegral entry) Nothing where (lang,jitA) = execState (mapM_ st initialBindings) (Lang.empty,M.empty) where st (s,v) = do i <- viewState fst_ (internSym s) case v of Left v -> modifying fst_ (setSymVal i v) Right p -> modifying snd_ (M.insert i p) withDefaultContext = withState . initialContext contextState sta = (runState sta $< readIORef contextRef) >>= \(a,s') -> writeIORef contextRef s' >> return a languageState = contextState . viewing language_ pageSize = fromIntegral $ unsafePerformIO $ c'sysconf c'_SC_PAGESIZE enableExec p size = do let p' = alignPtr (p`plusPtr`(1-pageSize)) pageSize mprotect (castPtr p') (fromIntegral $ size+ p`minusPtr`p') (c'PROT_READ .|. c'PROT_WRITE .|. c'PROT_EXEC) evalCode :: (FunPtr (Ptr() -> a) -> Ptr() -> a) -> ByteString -> Code -> (a -> IO b) -> IO b evalCode wrap stub code f = do id <- languageState $ state createSym >>= \i -> modify (setSymVal i (Verb code)) >> return i p <- getAddressJIT id unsafeUseAsCString stub $ \stub -> f $ wrap (castPtrToFunPtr stub) (intPtrToPtr $ fromIntegral p) execCode c = evalCode mkProc execStub c id withRef ref val x = readIORef ref >>= \v -> writeIORef ref val >> x >>= \x -> writeIORef ref v >> return x getAddress arch lookup register = withRef compAddrRef getAddr . getAddr where getAddr sym = lookup sym >>= \val -> case val of Just a -> return a Nothing -> gets language >>= \lang -> (>> getAddr sym) $ case lookupSymVal sym lang of Verb c -> void $ do let (size,codem) = specialize arch (sym,getAddr) c ptr <- mallocForeignPtrBytes size register sym ptr size code <- codem withForeignPtr ptr $ \p -> do unsafeUseAsCStringLen code $ \(p',n) -> copyBytes p (castPtr p') n enableExec p size Noun size init -> do size <- evalCode mkFunSize execStub size id ptr <- mallocForeignPtrBytes size register sym ptr size withForeignPtr ptr $ \p -> evalCode mkFunInit initStub init ($castPtr p) _ -> fail $ "Couldn't find definition of symbol "++fromMaybe (show sym) (lookupSymName sym lang) getAddressJIT = getAddress arch_host lookup register where lookup id = do val <- M.lookup id $< gets jitAddresses return $ (fromIntegral . ptrToIntPtr . unsafeForeignPtrToPtr) $< val register id ptr size = modifying jitAddresses_ (M.insert id ptr) getAddressComp arch = getAddress arch lookup register where lookup id = (fst$<) $< M.lookup id $< gets compAddresses register id ptr size = do n <- getting compTop_ modifying compAddresses_ (M.insert id (n,ptr)) modifying compTop_ (+size) doTransform syn = gets transform >>= ($syn) . maybe return tr where tr fun tree = do root <- allocTree tree new <- unsafeUseAsCString initStub $ \stub -> mkFunTransform (castPtrToFunPtr stub) fun root readTree new intS = sizeOf (undefined::Int) ; ptrS = sizeOf (undefined::Ptr()) pok e p = poke (castPtr p) e >> return (p`plusPtr`sizeOf e) pik p = peek (castPtr p) >>= \e -> return (e,p`plusPtr`sizeOf e) allocTree (Group g) = do p <- mallocBytes (intS+intS+(length g*ptrS)) p' <- pok (0::Int) p p'' <- pok (length g) p' mapM allocTree g >>= \l -> pokeArray (castPtr p'') l return p allocTree (Symbol (ID s)) = do p <- mallocBytes (intS+intS) p' <- pok (1::Int) p pok s p' return p readTree p = do (t,p') <- pik p case t :: Int of 0 -> do (s,p'') <- pik p' l <- peekArray s p'' liftM Group $ mapM readTree l 1 -> do liftM (Symbol . ID) $ peek p'