{-# LANGUAGE MagicHash #-} -- {-# OPTIONS_GHC -O3 #-} module UHC.Light.Compiler.CoreRun.Run.Val ( module System.IO , module Data.IORef , module Data.Int, module Data.Word , module GHC.Ptr , RVal (..) , mkTuple, mkUnit , RValV, RValMV , HSMarshall (..) , HpPtr, nullPtr, isNullPtr, newHeap, Heap (..) , heapGetM, heapGetM', heapAllocM, heapAllocAsPtrM, heapUpdM, heapSetM, heapSetM' , RValCxt (..), emptyRValCxt , rvalTrEnterLam , mustReturn, needNotReturn, rvalRetEvl, rvalPrimargEvl , vecReverseForM_ , mvecAllocInit, mvecAlloc, mvecFillFromV, mvecReverseFillFromV, mvecFillFromMV, mvecAllocFillFromV, mvecAppend, mvecToList , mvecReverseForM_ , ptr2valM, ref2valM , RValEnv (..), newRValEnv , renvTopFrameM , dumpEnvM , updTopFrameM , renvFrStkPush1, renvFrStkReversePushMV , renvFrStkPop1, renvFrStkPopMV, renvFrStkReversePopMV, renvFrStkReversePopInMV , rsemTr', rsemTr , RValT ) where import UHC.Light.Compiler.Base.HsName.Builtin import UHC.Light.Compiler.Base.Common import UHC.Light.Compiler.Error import UHC.Light.Compiler.CoreRun import UHC.Light.Compiler.CoreRun.Run import UHC.Light.Compiler.CoreRun.Pretty import UHC.Util.Pretty import Control.Monad.Primitive import qualified Data.Vector as V import qualified Data.Vector.Mutable as MV import qualified Data.Vector.Generic as GV import Data.Data import Data.Typeable import System.IO import System.IO.Unsafe import Data.IORef import Data.Int import Data.Word import qualified Data.ByteString.Char8 as BSC8 import GHC.Ptr (Ptr (..)) import GHC.Exts (Addr#) {-# LINE 63 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Representation of run value data RVal = -- Value representations for running itself: literals RVal_Lit { rvalSExp :: !SExp -- ^ a simple literal } | RVal_Char {-# UNPACK #-} !Char | RVal_Int {-# UNPACK #-} !Int {-- | RVal_Int8 {-# UNPACK #-} !Int8 | RVal_Int16 {-# UNPACK #-} !Int16 -} | RVal_Int32 {-# UNPACK #-} !Int32 {-- | RVal_Int64 {-# UNPACK #-} !Int64 | RVal_Word {-# UNPACK #-} !Word | RVal_Word8 {-# UNPACK #-} !Word8 | RVal_Word16 {-# UNPACK #-} !Word16 | RVal_Word32 {-# UNPACK #-} !Word32 | RVal_Word64 {-# UNPACK #-} !Word64 -} | RVal_Integer !Integer | RVal_Float {-# UNPACK #-} !Float | RVal_Double {-# UNPACK #-} !Double | RVal_PackedString !BSC8.ByteString -- ^ packed string, equivalent of low level C string (could be replaced by something more efficient) -- Value representations for running itself: function, application, etc | RVal_Lam { rvalMbNm :: !(Maybe HsName) -- ^ possibly bound to name , rvalBody :: !Exp -- ^ a Exp_Lam, which also encodes a thunk , rvalSLRef :: !(IORef HpPtr) -- ^ static link to enclosing stack frame } -- | special case of Lam taking 0 params | RVal_Thunk { rvalMbNm :: !(Maybe HsName) -- ^ possibly bound to name , rvalBody :: !Exp -- ^ Exp taking no arguments (thunk) , rvalSLRef :: !(IORef HpPtr) -- ^ static link to enclosing stack frame } | RVal_Node { rvalTag :: !Int -- {-# UNPACK #-} !RVal -- ^ node tag , rvalNdVals :: !RValMV -- ^ fields } | RVal_App { rvalFun :: !RVal -- ^ a RVal_App or RVal_PApp , rvalArgs :: !RValMV -- ^ already applied args } | RVal_Frame { rvalRef2Nm :: Ref2Nm -- ^ ref to name mapping , rvalSLRef :: !(IORef HpPtr) -- ^ immediately outer lexical level frame -- , rvalLev :: !Int -- ^ the lexical level this frame is on , rvalFrVals :: !RValMV -- ^ actual frame values, either literals or pointers to heap locations (so we can update them, share them) , rvalFrSP :: !(IORef Int) -- ^ top of expr stack embedded in higher end of top frame } | RVal_Ptr { rvalPtrRef :: !(IORef HpPtr) -- ^ ptr/index into heap } | RVal_Fwd { rvalPtr :: !HpPtr -- ^ forwarding ptr, only used during GC } | RVal_BlackHole | RVal_None -- Value representations for library or runtime env (not Core specific) -- | mutable var | RHsV_MutVar !(IORef RVal) -- | IO handle | RHsV_Handle !Handle -- | Addr inside Ptr | RHsV_Addr Addr# instance Show RVal where show _ = "RVal" ppRVal' :: (HpPtr -> IO (Maybe RVal)) -> RVal -> IO PP_Doc ppRVal' lkptr rval = case rval of RVal_Lit e -> dfltpp e RVal_Char v -> dfltpp $ show v RVal_Int v -> dfltpp v {-- RVal_Int8 v -> dfltpp $ show v RVal_Int16 v -> dfltpp $ show v -} RVal_Int32 v -> dfltpp $ show v {-- RVal_Int64 v -> dfltpp $ show v RVal_Word v -> dfltpp $ show v RVal_Word8 v -> dfltpp $ show v RVal_Word16 v -> dfltpp $ show v RVal_Word32 v -> dfltpp $ show v RVal_Word64 v -> dfltpp $ show v -} RVal_Integer v -> dfltpp v RVal_Float v -> dfltpp v RVal_Double v -> dfltpp $ show v RVal_PackedString v -> dfltpp $ show v RVal_Lam mn b slref -> dfltpp b RVal_Thunk mn e slref -> return $ ppBrackets e RVal_Node t vs -> do vl <- mvecToList vs return $ t >|< ppBracketsCommas vl RVal_App f as -> dfltpp f -- return $ ppBrackets $ f >|< "@" >|< (ppParensCommas $ V.toList as) RVal_Ptr pref -> do p <- readIORef pref vpp <- lkptr p >>= maybe (return empty) (\v -> ppRVal' lkptr v >>= \vpp -> return $ " -> " >|< vpp) return $ "*" >|< p >|< vpp RVal_Fwd p -> return $ "f*" >|< p RVal_Frame _ slref {- lv -} vs spref -> do sl <- readIORef slref sp <- readIORef spref vl <- mvecToList vs vlpp <- forM (zip [0..(sp-1)] vl) $ \(i,v) -> ppRVal' lkptr v >>= \vpp -> return $ i >|< ":" >#< vpp return $ (ppBracketsCommas $ ["sl=" >|< sl, {- "lev=" >|< lv, -} "sz=" >|< MV.length vs, "sp=" >|< sp]) >-< vlist vlpp RVal_BlackHole -> dfltpp "Hole" RVal_None -> dfltpp "None" RHsV_MutVar v -> dfltpp "MutVar" RHsV_Handle h -> dfltpp $ show h RHsV_Addr p -> dfltpp $ show (Ptr p) where dfltpp :: PP x => x -> IO PP_Doc dfltpp = return . pp ppRValWithHp :: Heap -> RVal -> IO PP_Doc ppRValWithHp hp = ppRVal' (\p -> heapGetM'' hp p >>= (return . Just)) ppRVal :: RVal -> IO PP_Doc ppRVal = ppRVal' (\_ -> return Nothing) instance PP RVal where pp rval = unsafePerformIO (ppRVal rval) {-# LINE 197 "src/ehc/CoreRun/Run/Val.chs" #-} mkTuple :: (RunSem RValCxt RValEnv RVal m a) => [RVal] -> RValT m a mkTuple vs = liftIO (mvecAllocFillFromV $ crarrayFromList vs) >>= rsemNode 0 >>= rsemPush {-# INLINE mkTuple #-} mkUnit :: (RunSem RValCxt RValEnv RVal m a) => RValT m a mkUnit = mkTuple [] {-# INLINE mkUnit #-} {-# LINE 207 "src/ehc/CoreRun/Run/Val.chs" #-} -- Vector of RVal type RValV = CRArray RVal -- Mutable vector of RVal type RValMV = CRMArray RVal {-# LINE 220 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Marshalling from/to Haskell values class HSMarshall hs where -- | Marshall to Haskell value, also parameterized by evaluator hsMarshall :: (RunSem RValCxt RValEnv RVal m a) => (RVal -> RValT m a) -> RVal -> RValT m hs -- | Unmarshall from Haskell value hsUnmarshall :: (RunSem RValCxt RValEnv RVal m a) => hs -> RValT m a {-# LINE 230 "src/ehc/CoreRun/Run/Val.chs" #-} instance HSMarshall Int where hsMarshall _ (RVal_Int v) = return v hsUnmarshall v = rsemPush $ RVal_Int v {-# INLINE hsUnmarshall #-} instance HSMarshall Integer where hsMarshall _ (RVal_Integer v) = return v hsUnmarshall v = rsemPush $ RVal_Integer v {-# INLINE hsUnmarshall #-} instance HSMarshall Bool where hsMarshall _ (RVal_Node t _) = return $ t == tagBoolTrue hsMarshall _ v = err $ "CoreRun.Run.Val.HSMarshall Bool:" >#< v hsUnmarshall b = liftIO (mvecAllocFillFromV emptyCRArray) >>= rsemNode (if b then tagBoolTrue else tagBoolFalse) >>= rsemPush {-# INLINE hsUnmarshall #-} instance HSMarshall Char where hsMarshall _ (RVal_Char v) = return v hsMarshall _ v = err $ "CoreRun.Run.Val.HSMarshall Char:" >#< v hsUnmarshall v = rsemPush $ RVal_Char v {-# INLINE hsUnmarshall #-} instance HSMarshall (Ptr a) where hsMarshall _ (RHsV_Addr v) = return $ Ptr v hsMarshall _ v = err $ "CoreRun.Run.Val.HSMarshall (Ptr a):" >#< v hsUnmarshall (Ptr v) = rsemPush $ RHsV_Addr v {-# INLINE hsUnmarshall #-} instance HSMarshall [RVal] where hsMarshall evl (RVal_Node t as) | t == tagListCons = liftIO (MV.read as 1) >>= evl >>= rsemPop >>= hsMarshall evl >>= \tl -> liftIO (MV.read as 0) >>= \hd -> return (hd : tl) | otherwise = return [] hsMarshall _ v = err $ "CoreRun.Run.Val.HSMarshall [RVal]:" >#< v hsUnmarshall [] = liftIO (mvecAllocFillFromV emptyCRArray) >>= rsemNode tagListNil >>= rsemPush hsUnmarshall (h:t) = hsUnmarshall t >>= rsemPop >>= \t' -> (liftIO $ mvecAllocFillFromV $ crarrayFromList [h, t']) >>= rsemNode tagListCons >>= rsemPush instance HSMarshall x => HSMarshall [x] where hsMarshall evl x = hsMarshall evl x >>= mapM (\v -> evl v >>= rsemPop >>= hsMarshall evl) {-# INLINE hsMarshall #-} hsUnmarshall v = forM v (\e -> hsUnmarshall e >>= rsemPop) >>= hsUnmarshall {-# INLINE hsUnmarshall #-} {-# LINE 287 "src/ehc/CoreRun/Run/Val.chs" #-} deriving instance Typeable IOMode deriving instance Data IOMode {-# LINE 296 "src/ehc/CoreRun/Run/Val.chs" #-} -- | HpPtr is index into a heap type HpPtr = Int nullPtr :: HpPtr nullPtr = -1 isNullPtr = (== nullPtr) data Heap = Heap { hpVals :: {-# UNPACK #-} !RValMV -- ^ value array , hpFirst :: {-# UNPACK #-} !HpPtr -- ^ the ptr of hpVals ! 0 , hpFree :: {-# UNPACK #-} !(IORef HpPtr) -- ^ first array location free for alloc , hpSemispaceMultiplier :: {-# UNPACK #-} !(IORef Rational) -- ^ multiplier by which to enlarge/shrink subsequent semispace } newHeap :: Int -> IO Heap newHeap sz = do vs <- mvecAllocInit sz fr <- newIORef 0 ml <- newIORef 1.5 return $ Heap vs 0 fr ml {-# LINE 321 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Garbage collect heap (TBD) heapGcM :: (RunSem RValCxt RValEnv RVal m x) => RVal -> RValT m RVal -- heapGcM = err $ "CoreRun.Run.Val.heapGcM: GC not yet implemented" heapGcM curV = do -- rsemSetTrace True -- rsemTr' True $ "GC starts, curV=" >#< curV -- rsemTr $ "GC starts" env@(RValEnv {renvHeap=hp@(Heap {hpVals=vsOld, hpSemispaceMultiplier=mlRef, hpFirst=offOld, hpFree=hpFrRef}), renvGcRootStack=rootStkRef, renvTopFrame=topFrRef, renvStack=stkRef, renvGlobals=globals}) <- get env' <- liftIO $ do ml <- readIORef mlRef let szOld = MV.length vsOld szNew = (round $ fromIntegral szOld * ml) :: Int offNew = offOld + szOld lwbOld = offOld upbOld = offNew vsNew <- mvecAllocInit szNew greyref <- newIORef offNew let -- copy content of old ptr to new loc, leaving a forwarding on to the old location copyp p -- is this indeed an old ptr? | p >= lwbOld && p < upbOld = do let pOld = p iOld = pOld - offOld v <- MV.read vsOld iOld case v of RVal_Fwd pNew -> do -- putStrLn $ "GC copyp Fwd p=" ++ show pOld ++ ", pNew=" ++ show pNew return pNew v -> do pNew <- readIORef greyref let iNew = pNew - offNew MV.write vsNew iNew v MV.write vsOld iOld (RVal_Fwd pNew) writeIORef greyref (pNew+1) -- putStrLn $ "GC copyp Val p=" ++ show pOld ++ ", pNew=" ++ show pNew ++ ", v=" ++ show (pp v) return pNew | otherwise = do -- putStrLn $ "GC copyp None p=" ++ show p return p -- inspect RVal_Ptr only copypv v = do case v of RVal_Ptr pref -> modifyIORefM pref copyp _ -> return () -- inspect, possibly copy internal part of a RVal, stops with a ptr which later is dealt with (to prevent too deep stack growth), exploiting the in between grey/black area as queue copyv v = do -- putStrLn $ "GC copyv 1 v=" ++ show (pp v) case v of RVal_Ptr pref -> modifyIORefM pref copyp RVal_Frame {rvalSLRef=slref, rvalFrVals=vs, rvalFrSP=spref} -> modifyIORefM slref copyp >> readIORef spref >>= \sp -> mvecForM_' 0 sp vs copyv RVal_Node {rvalNdVals=vs} -> mvecForM_ vs copyv RVal_App {rvalFun=f, rvalArgs=as} -> copyv f >> mvecForM_ as copyv RVal_Thunk {rvalSLRef=slref} -> modifyIORefM slref copyp RVal_Lam {rvalSLRef=slref} -> modifyIORefM slref copyp _ -> return () -- putStrLn $ "GC copyv 2 v=" ++ show (pp v) -- inspect, follow, copy content of RVal follow pBlk pGry | pBlk < pGry = MV.read vsNew (pBlk-offNew) >>= copyv >> follow (pBlk+1) pGry | otherwise = do pGry' <- readIORef greyref if pBlk < pGry' then follow pBlk pGry' else return pBlk -- initial copy: top frame modifyIORefM topFrRef copyp --- $ \topFr -> {- if isNullPtr topFr then return topFr else -} copyp topFr -- initial copy: stack modifyIORefM stkRef $ mapM copyp -- initial copy: globals globals' <- V.forM globals copyp -- initial copy: the RVal to be put on the heap copyv curV -- initial copy: additional roots readIORef rootStkRef >>= mapM_ (mapM_ copyv) -- follow with initial values of grey and black (0) readIORef greyref >>= follow offNew >>= \p -> writeIORef hpFrRef (p - offNew) -- final: return $ env {renvGlobals = globals', renvHeap = hp {hpVals=vsNew, hpFirst=offNew}} put env' -- rsemTr $ "GC done" -- rsemTr' True $ "GC done, curV=" >#< curV -- rsemSetTrace False return curV {-# LINE 417 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Allocate on the heap heapAllocM :: (RunSem RValCxt RValEnv RVal m x) => RVal -> RValT m HpPtr heapAllocM v = do hp@(Heap {hpVals=vs, hpFirst=off, hpFree=fr}) <- gets renvHeap p <- liftIO $ readIORef fr if p >= MV.length vs then heapGcM v >>= heapAllocM else liftIO $ do MV.write vs p v writeIORef fr (p + 1) return $ p + off -- | Allocate on the heap, packing as RVal_Ptr heapAllocAsPtrM :: (RunSem RValCxt RValEnv RVal m x) => RVal -> RValT m RVal heapAllocAsPtrM v = do p <- heapAllocM v liftIO (newIORef p) >>= (return . RVal_Ptr) {- -} -- | Get a value from the heap -- heapGetM'' :: PrimMonad m => Heap -> HpPtr -> m RVal heapGetM'' :: Heap -> HpPtr -> IO RVal heapGetM'' hp@(Heap {hpVals=vs, hpFirst=off}) p = MV.read vs (p - off) {-# INLINE heapGetM'' #-} -- | Get a value from the heap heapGetM' :: (RunSem RValCxt RValEnv RVal m x) => Heap -> HpPtr -> RValT m RVal heapGetM' hp p = liftIO $ heapGetM'' hp p {-# INLINE heapGetM' #-} -- | Get a value from the heap heapGetM :: (RunSem RValCxt RValEnv RVal m x) => HpPtr -> RValT m RVal heapGetM p = do hp <- gets renvHeap heapGetM' hp p {-# INLINE heapGetM #-} -- | Set a value in the heap heapSetM' :: (RunSem RValCxt RValEnv RVal m x) => Heap -> HpPtr -> RVal -> RValT m () heapSetM' hp@(Heap {hpVals=vs, hpFirst=off}) p v = liftIO $ MV.write vs (p - off) v {-# INLINE heapSetM' #-} -- | Set a value in the heap heapSetM :: (RunSem RValCxt RValEnv RVal m x) => HpPtr -> RVal -> RValT m () heapSetM p v = do hp <- gets renvHeap heapSetM' hp p v {-# INLINE heapSetM #-} -- | Update a value in the heap heapUpdM' :: (RunSem RValCxt RValEnv RVal m x) => Heap -> HpPtr -> (RVal -> RValT m RVal) -> RValT m () heapUpdM' hp@(Heap {hpVals=vs, hpFirst=off}) p f = do let p' = p - off v <- liftIO $ MV.read vs p' v' <- f v liftIO $ MV.write vs p' v' {-# INLINE heapUpdM' #-} -- | Update a value in the heap heapUpdM :: (RunSem RValCxt RValEnv RVal m x) => HpPtr -> (RVal -> RValT m RVal) -> RValT m () heapUpdM p f = do hp <- gets renvHeap heapUpdM' hp p f {-# INLINE heapUpdM #-} {-# LINE 489 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Environment: context/reader data RValCxt = RValCxt { rcxtInRet :: !Bool -- ^ in returning context, True by default , rcxtCallCxt :: [Maybe HsName] -- ^ calling context stack, for debugging only } emptyRValCxt :: RValCxt emptyRValCxt = RValCxt True [] {-# LINE 501 "src/ehc/CoreRun/Run/Val.chs" #-} rvalTrEnterLam :: RunSem RValCxt RValEnv RVal m x => Maybe HsName -> RValT m a -> RValT m a rvalTrEnterLam s = local (\r -> r {rcxtCallCxt = s : rcxtCallCxt r}) {-# LINE 506 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Set return context to True mustReturn :: RunSem RValCxt RValEnv RVal m x => RValT m a -> RValT m a mustReturn = local (\r -> r {rcxtInRet = True}) {-# INLINE mustReturn #-} -- | Set return context to False needNotReturn :: RunSem RValCxt RValEnv RVal m x => RValT m a -> RValT m a needNotReturn = local (\r -> r {rcxtInRet = False}) {-# INLINE needNotReturn #-} -- | Variation of `rsemEvl` in return context rvalRetEvl :: RunSem RValCxt RValEnv RVal m x => RVal -> RValT m x rvalRetEvl = mustReturn . rsemEvl {-# INLINE rvalRetEvl #-} -- | Variation of `rsemEvl` in primitve argument context rvalPrimargEvl :: RunSem RValCxt RValEnv RVal m x => RVal -> RValT m x rvalPrimargEvl x = rvalRetEvl x >>= rsemPop >>= rsemDeref {-# INLINE rvalPrimargEvl #-} {-# LINE 532 "src/ehc/CoreRun/Run/Val.chs" #-} {- -- | Loop over a vector, starting at lower bound 'l', ending before 'h' vecLoop :: Monad m => Int -> Int -> (V.Vector a -> Int -> m b) -> V.Vector a -> m () vecLoop l h m v = loop l where loop l | l < h = m v l >> loop (l+1) | otherwise = return () {-# INLINE vecLoop #-} -} -- | Loop over a vector from upb to lwb, starting before 'h', ending at lower bound 'l' vecLoopReverse :: Monad m => Int -> Int -> (V.Vector a -> Int -> m b) -> V.Vector a -> m () vecLoopReverse l h m v = loop (h-1) where loop h | l <= h = m v h >> loop (h-1) | otherwise = return () {-# INLINE vecLoopReverse #-} {-# LINE 551 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Right to left forM_ vecReverseForM_ :: Monad m => V.Vector a -> (a -> m x) -> m () vecReverseForM_ v m = vecLoopReverse 0 (V.length v) (\_ i -> m (v V.! i)) v {-# INLINE vecReverseForM_ #-} {-# LINE 562 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Allocate a mutable vector of given size mvecAllocWith :: PrimMonad m => Int -> a -> m (MV.MVector (PrimState m) a) mvecAllocWith = MV.replicate {-# INLINE mvecAllocWith #-} -- | Allocate a mutable vector of given size, init to default value mvecAllocInit :: Int -> IO RValMV mvecAllocInit sz = mvecAllocWith sz RVal_None {-# INLINE mvecAllocInit #-} -- | Allocate a mutable vector of given size mvecAlloc :: PrimMonad m => Int -> m (MV.MVector (PrimState m) a) mvecAlloc = MV.new {-# INLINE mvecAlloc #-} -- | Loop over a mutable vector, updating the vector as a side effect, starting at lower bound 'l', ending before 'h' mvecLoop :: PrimMonad m => Int -> Int -> (MV.MVector (PrimState m) a -> Int -> m b) -> MV.MVector (PrimState m) a -> m (MV.MVector (PrimState m) a) mvecLoop l h m v = loop l where loop l | l < h = m v l >> loop (l+1) | otherwise = return v {-# INLINE mvecLoop #-} -- | Loop over a mutable vector from upb to lwb, updating the vector as a side effect, starting before 'h', ending at lower bound 'l' mvecLoopReverse :: PrimMonad m => Int -> Int -> (MV.MVector (PrimState m) a -> Int -> m b) -> MV.MVector (PrimState m) a -> m (MV.MVector (PrimState m) a) mvecLoopReverse l h m v = loop (h-1) where loop h | l <= h = m v h >> loop (h-1) | otherwise = return v {-# INLINE mvecLoopReverse #-} -- | Loop over a mutable vector from upb to lwb, updating the vector as a side effect, starting before 'h', ending at lower bound 'l' mvecLoopReverseAccum :: PrimMonad m => acc -> Int -> Int -> (acc -> Int -> m acc) -> MV.MVector (PrimState m) a -> m acc mvecLoopReverseAccum a l h m v = loop (h-1) a where loop h a | l <= h = m a h >>= loop (h-1) | otherwise = return a {-# INLINE mvecLoopReverseAccum #-} -- | Convert to a list mvecToList :: PrimMonad m => MV.MVector (PrimState m) a -> m [a] mvecToList v = mvecLoopReverseAccum [] 0 (MV.length v) (\l i -> MV.read v i >>= \a -> return (a:l)) v -- | Fill a mutable vector from a unmutable vector, starting with filling at the given lowerbound mvecFillFromV :: PrimMonad m => Int -> MV.MVector (PrimState m) a -> CRArray a -> m () mvecFillFromV lwb toarr frarr = forM_ (craAssocs' lwb frarr) $ \(i,e) -> MV.write toarr i e {-# INLINE mvecFillFromV #-} -- | Fill a mutable vector from a unmutable vector, starting with filling at the given lowerbound, reversing the given vector mvecReverseFillFromV :: PrimMonad m => Int -> MV.MVector (PrimState m) a -> CRArray a -> m () mvecReverseFillFromV lwb toarr frarr = forM_ (craReverseAssocs' lwb frarr) $ \(i,e) -> MV.write toarr i e {-# INLINE mvecReverseFillFromV #-} -- | Fill a mutable vector from another mutable vector, starting with copying at the given lowerbounds, copying size elements mvecFillFromMV' :: PrimMonad m => Int -> Int -> Int -> MV.MVector (PrimState m) a -> MV.MVector (PrimState m) a -> m () mvecFillFromMV' lwbTo lwbFr sz toarr frarr = mvecLoop lwbFr (lwbFr+sz) (\v i -> MV.read v i >>= MV.write toarr (i+lwbDiff)) frarr >> return () where lwbDiff = lwbTo - lwbFr {-# INLINE mvecFillFromMV' #-} -- | Fill a mutable vector from another mutable vector, starting with copying at the given lowerbounds, copying size elements, but reversing the given vector mvecReverseFillFromMV' :: PrimMonad m => Int -> Int -> Int -> MV.MVector (PrimState m) a -> MV.MVector (PrimState m) a -> m () -- mvecReverseFillFromMV' lwbTo lwbFr sz toarr frarr = mvecLoop lwbFr (lwbFr+sz) (\v i -> MV.read v (upbFr1-i) >>= MV.write toarr (i+lwbDiff)) frarr >> return () mvecReverseFillFromMV' lwbTo lwbFr sz toarr frarr = mvecLoopReverse lwbFr (lwbFr+sz) (\v i -> MV.read v i >>= MV.write toarr (upbFr1 - i + lwbTo)) frarr >> return () where lwbDiff = lwbTo - lwbFr upbFr1 = lwbFr + sz - 1 {-# INLINE mvecReverseFillFromMV' #-} -- | Fill a mutable vector from another mutable vector, starting with filling at the given lowerbound mvecFillFromMV :: PrimMonad m => Int -> MV.MVector (PrimState m) a -> MV.MVector (PrimState m) a -> m () -- mvecFillFromMV lwb toarr frarr = mvecLoop 0 (MV.length frarr) (\v i -> MV.read frarr i >>= MV.write toarr (i+lwb)) frarr >> return () mvecFillFromMV lwb toarr frarr = mvecFillFromMV' lwb 0 (MV.length frarr) toarr frarr {-# INLINE mvecFillFromMV #-} -- {-# SPECIALIZE mvecFillFromMV :: Int -> MV.MVector (PrimState IO) a -> MV.MVector (PrimState IO) a -> IO () #-} -- | Fill a mutable vector from another mutable vector, starting with filling at the given lowerbound, but reversing the given vector mvecReverseFillFromMV :: PrimMonad m => Int -> MV.MVector (PrimState m) a -> MV.MVector (PrimState m) a -> m () mvecReverseFillFromMV lwb toarr frarr = mvecReverseFillFromMV' lwb 0 (MV.length frarr) toarr frarr {- mvecReverseFillFromMV lwb toarr frarr = mvecLoop 0 l (\v i -> MV.read frarr (l1-i) >>= MV.write toarr (i+lwb)) frarr >> return () where l = MV.length frarr l1 = l - 1 -} {-# INLINE mvecReverseFillFromMV #-} -- | Alloc and fill vector of size taken from fixed vec mvecAllocFillFromV :: PrimMonad m => CRArray a -> m (MV.MVector (PrimState m) a) mvecAllocFillFromV frarr = mvecAlloc (V.length frarr) >>= \toarr -> mvecFillFromV 0 toarr frarr >> return toarr {-# INLINE mvecAllocFillFromV #-} -- | Loop over a mutable vector, updating the vector as a side effect, with explicit range mvecForM_' :: PrimMonad m => Int -> Int -> MV.MVector (PrimState m) a -> (a -> m b) -> m () mvecForM_' lwb upb v m = mvecLoop lwb upb (\v i -> MV.read v i >>= m) v >> return () {-# INLINE mvecForM_' #-} -- | Loop over a mutable vector, updating the vector as a side effect mvecForM_ :: PrimMonad m => MV.MVector (PrimState m) a -> (a -> m b) -> m () mvecForM_ v m = mvecForM_' 0 (MV.length v) v m {-# INLINE mvecForM_ #-} -- | Append mutable 2 vectors mvecAppend :: PrimMonad m => MV.MVector (PrimState m) a -> MV.MVector (PrimState m) a -> m (MV.MVector (PrimState m) a) mvecAppend v1 v2 = mvecAlloc l12 >>= mvecLoop 0 l1 (\v i -> MV.read v1 i >>= MV.write v i) >>= mvecLoop l1 l12 (\v i -> MV.read v2 (i-l1) >>= MV.write v i) where l1 = MV.length v1 l2 = MV.length v2 l12 = l1 + l2 {-# LINE 669 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Right to left forM_ mvecReverseForM_ :: PrimMonad m => MV.MVector (PrimState m) a -> (a -> m x) -> m () mvecReverseForM_ v m = mvecLoopReverse 0 (MV.length v) (\v i -> MV.read v i >>= m) v >> return () {-# INLINE mvecReverseForM_ #-} {-# LINE 680 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Dereference a possibly RVal_Ptr ptr2valM :: (RunSem RValCxt RValEnv RVal m x) => RVal -> RValT m RVal ptr2valM v = case v of RVal_Ptr pref -> liftIO (readIORef pref) >>= heapGetM >>= \v' -> case v' of RVal_Thunk {} -> return v _ -> ptr2valM v' _ -> return v {-# INLINE ptr2valM #-} -- | Dereference a RRef ref2valM :: (RunSem RValCxt RValEnv RVal m x) => RRef -> RValT m RVal ref2valM r = do -- rsemTr $ "R: " ++ show (pp r) env <- get case r of RRef_Glb m e -> do modFrame <- heapGetM (renvGlobals env V.! m) liftIO $ MV.read (rvalFrVals modFrame) e {- RRef_Loc l o -> do topfrp <- renvTopFrameM topfr <- heapGetM topfrp access topfr where access (RVal_Frame {rvalLev=frlev, rvalFrVals=vs}) | l == frlev = do -- rsemTr $ "R o=" ++ show o ++ " len(vs)=" ++ show (MV.length vs) liftIO $ MV.read vs o access (RVal_Frame {rvalLev=frlev, rvalSLRef=slref}) = do sl <- liftIO $ readIORef slref -- rsemTr $ "R sl=" ++ show sl ++ " frlev=" ++ show frlev ++ " l=" ++ show l heapGetM sl >>= access access v = err $ "CoreRun.Run.Val.ref2valM.RRef_Loc.access:" >#< r >#< "in" >#< v -} RRef_LDf ld o -> do topfrp <- renvTopFrameM topfr <- heapGetM topfrp access ld topfr where access 0 (RVal_Frame {rvalFrVals=vs}) = liftIO $ MV.read vs o access ld (RVal_Frame {rvalSLRef=slref}) = do sl <- liftIO $ readIORef slref fr <- heapGetM sl access (ld-1) fr access _ v = err $ "CoreRun.Run.Val.ref2valM.RRef_LDf.access:" >#< r >#< "in" >#< v RRef_Fld r e -> do v <- ptr2valM =<< ref2valM r -- >>= rsemDeref case v of RVal_Node _ vs -> liftIO $ MV.read vs e _ -> err $ "CoreRun.Run.Val.ref2valM.RRef_Fld:" >#< e >#< "in" >#< v RRef_Tag r -> do v <- ptr2valM =<< ref2valM r -- >>= rsemDeref case v of RVal_Node t _ -> return $ RVal_Int t _ -> err $ "CoreRun.Run.Val.ref2valM.RRef_Tag:" >#< v _ -> err $ "CoreRun.Run.Val.ref2valM.r:" >#< r {-# INLINE ref2valM #-} {-# LINE 747 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Frame holding locals indexed by RRef_Loc type RValFrame = HpPtr -- points to heap to a RVal_Frame -- | Frame Stack type RValStack = [RValFrame] -- | Environment: state data RValEnv = RValEnv { renvGlobals :: !(CRArray RValFrame) -- ^ per module frame of globals , renvStack :: !(IORef RValStack) -- ^ stack of frames, except for the top , renvTopFrame :: !(IORef RValFrame) -- ^ current frame, the actual top of the stack -- , renvFrSP :: !(IORef Int) -- ^ top of expr stack embedded in higher end of top frame , renvHeap :: !Heap -- ^ heap , renvDoTrace :: !Bool , renvDoTraceExt :: !Bool -- ^ when tracing, do it extensively? , renvGcRootStack :: !(IORef [[RVal]]) -- ^ stack of roots for GC, use is optional } newRValEnv :: Int -> IO RValEnv newRValEnv hpSz = do st <- newIORef [] tp <- newIORef nullPtr hp <- newHeap hpSz -- sp <- newIORef 0 rtst <- newIORef [] return $ RValEnv V.empty st tp hp False False rtst {-# LINE 777 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Get the top most frame from the stack, 'nullPtr' if absent renvTopFrameM :: (RunSem RValCxt RValEnv RVal m x) => RValT m HpPtr renvTopFrameM = do (RValEnv {renvTopFrame=tf}) <- get liftIO $ readIORef tf {-# INLINE renvTopFrameM #-} {-# LINE 786 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Get all non empty stack frames, still split up in (possibly null) top frame and rest of stack renvAllFrameM' :: (RunSem RValCxt RValEnv RVal m x) => RValT m (Maybe HpPtr,[HpPtr]) renvAllFrameM' = do topfrp <- renvTopFrameM env <- get st <- liftIO $ readIORef (renvStack env) return (if isNullPtr topfrp then Nothing else Just topfrp, st) -- | Get all non empty stack frames renvAllFrameM :: (RunSem RValCxt RValEnv RVal m x) => RValT m [HpPtr] renvAllFrameM = do (mbtop,st) <- renvAllFrameM' return $ maybe [] (:[]) mbtop ++ st {-# LINE 802 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Dump environment dumpPpEnvM :: (RunSem RValCxt RValEnv RVal m x) => Bool -> RValT m PP_Doc dumpPpEnvM extensive = do stkfrs <- renvAllFrameM env <- get callcxt <- asks rcxtCallCxt let hp = renvHeap env stkfrspp <- forM stkfrs $ dumpFrameMinimal hp hpfr <- liftIO $ readIORef (hpFree hp) needRet <- asks rcxtInRet let dash = "====================" header1 = dash >-< "rcxtInRet=" >|< needRet header2 = ppCurly $ "Heap =" >|< hpfr >|< "/" >|< MV.length (hpVals hp) >|< ", CallCxt=" >|< ppBracketsCommas callcxt >|< ", Stack=" >|< ppBracketsCommas stkfrspp footer1 = dash hpPP <- dumpHeap hp hpfr glPP <- dumpGlobals hp (renvGlobals env) frPPs <- forM stkfrs $ dumpFrame hp if extensive then return $ header1 >-< header2 >-< hpPP >-< glPP >-< "====== Frames ======" >-< (indent 2 $ vlist frPPs) >-< footer1 else return $ header2 where dumpFrameMinimal hp fp = do fr@(RVal_Frame {rvalFrVals=vs, rvalFrSP=spref}) <- heapGetM fp sp <- liftIO $ readIORef spref return $ fp >|< (ppParens $ sp >|< "/" >|< MV.length vs) dumpFrame hp fp = do fr@(RVal_Frame {rvalFrVals=vs, rvalFrSP=spref}) <- heapGetM fp sp <- liftIO $ readIORef spref -- pps <- ppa 0 hp sp vs (liftIO $ ppRValWithHp hp fr) >>= \frpp -> return $ "Frame ptr=" >|< fp >|< " sp=" >|< sp >-< (indent 2 $ frpp) -- >-< (indent 2 $ vlist pps)) dumpGlobals hp glbls = do pps <- forM [0 .. V.length glbls - 1] $ \i -> do dumpFrame hp (glbls V.! i) return $ "====== Globals ======" >-< indent 2 (vlist pps) dumpHeap hp@(Heap {hpFirst=off}) hpfr = do pps <- ppa off hp hpfr (hpVals hp) return $ "======= Heap =======" >-< indent 2 (vlist pps) ppb hp k v = (liftIO $ ppRValWithHp hp v) >>= \vpp -> return $ k >|< ":" >#< vpp ppa off hp sz vs = forM [0 .. sz - 1] $ \i -> liftIO (MV.read vs i) >>= ppb hp (i + off) {-# LINE 844 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Dump environment dumpEnvM :: (RunSem RValCxt RValEnv RVal m x) => Bool -> RValT m () dumpEnvM extensive = dumpPpEnvM extensive >>= \p -> liftIO $ putPPLn p >> hFlush stdout {-# LINE 854 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Update top frame updTopFrameM :: (RunSem RValCxt RValEnv RVal m x) => (RVal -> RValT m RVal) -> RValT m () updTopFrameM f = renvTopFrameM >>= flip heapUpdM f {-# INLINE updTopFrameM #-} {-# LINE 866 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Push on the stack embedded in the top frame renvFrStkPush' :: RunSem RValCxt RValEnv RVal m x => (Int -> RValMV -> v -> IO Int) -> v -> RValT m () renvFrStkPush' pushvOn v = do (RValEnv {renvTopFrame=frref, renvHeap=hp}) <- get liftIO $ do (RVal_Frame {rvalFrVals=frvs, rvalFrSP=spref}) <- heapGetM'' hp =<< readIORef frref sp <- readIORef spref sp' <- pushvOn sp frvs v writeIORef spref sp' -- | Push on the stack embedded in the top frame renvFrStkPush1 :: RunSem RValCxt RValEnv RVal m x => RVal -> RValT m () renvFrStkPush1 = renvFrStkPush' (\sp frvs v -> MV.write frvs sp v >> return (sp + 1)) {-# INLINE renvFrStkPush1 #-} -- | Push on the stack embedded in the top frame renvFrStkPushMV :: RunSem RValCxt RValEnv RVal m x => RValMV -> RValT m () renvFrStkPushMV = renvFrStkPush' (\sp frvs vs -> mvecFillFromMV sp frvs vs >> return (sp + MV.length vs)) {-# INLINE renvFrStkPushMV #-} -- | Push reversed on the stack embedded in the top frame renvFrStkReversePushMV :: RunSem RValCxt RValEnv RVal m x => RValMV -> RValT m () renvFrStkReversePushMV = renvFrStkPush' (\sp frvs vs -> mvecReverseFillFromMV sp frvs vs >> return (sp + MV.length vs)) {-# INLINE renvFrStkReversePushMV #-} -- | Push on the stack embedded in the top frame renvFrStkPushV :: RunSem RValCxt RValEnv RVal m x => RValV -> RValT m () renvFrStkPushV = renvFrStkPush' (\sp frvs vs -> mvecFillFromV sp frvs vs >> return (sp + V.length vs)) {-# INLINE renvFrStkPushV #-} -- | Push reversed on the stack embedded in the top frame renvFrStkReversePushV :: RunSem RValCxt RValEnv RVal m x => RValV -> RValT m () renvFrStkReversePushV = renvFrStkPush' (\sp frvs vs -> mvecReverseFillFromV sp frvs vs >> return (sp + V.length vs)) {-# INLINE renvFrStkReversePushV #-} {-# LINE 903 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Pop from the stack embedded in the top frame renvFrStkPop' :: RunSem RValCxt RValEnv RVal m x => (RValMV -> Int -> IO v) -> Int -> RValT m v renvFrStkPop' popvFrom sz = do (RValEnv {renvTopFrame=frref, renvHeap=hp}) <- get liftIO $ do (RVal_Frame {rvalFrVals=frvs, rvalFrSP=spref}) <- heapGetM'' hp =<< readIORef frref sp <- readIORef spref let sp' = sp - sz writeIORef spref sp' popvFrom frvs sp' -- | Pop from the stack embedded in the top frame renvFrStkPop :: RunSem RValCxt RValEnv RVal m x => Int -> RValT m () renvFrStkPop = renvFrStkPop' (\_ _ -> return ()) {-# INLINE renvFrStkPop #-} -- | Pop from the stack embedded in the top frame renvFrStkPop1 :: RunSem RValCxt RValEnv RVal m x => RValT m RVal renvFrStkPop1 = renvFrStkPop' MV.read 1 {-# INLINE renvFrStkPop1 #-} -- | Pop from the stack embedded in the top frame renvFrStkPopInMV :: RunSem RValCxt RValEnv RVal m x => Int -> Int -> RValMV -> RValT m () renvFrStkPopInMV lwbTo sz vs = renvFrStkPop' (\frvs sp -> mvecFillFromMV' lwbTo sp sz vs frvs) sz {-# INLINE renvFrStkPopInMV #-} -- | Pop from the stack embedded in the top frame renvFrStkPopMV :: RunSem RValCxt RValEnv RVal m x => Int -> RValT m RValMV -- renvFrStkPopMV sz = renvFrStkPop' (\frvs sp -> mvecAlloc sz >>= \vs -> mvecFillFromMV' 0 sp sz vs frvs >> return vs) sz renvFrStkPopMV sz = (liftIO $ mvecAlloc sz) >>= \vs -> renvFrStkPopInMV 0 sz vs >> return vs {-# INLINE renvFrStkPopMV #-} -- | Pop from the stack embedded in the top frame renvFrStkReversePopInMV :: RunSem RValCxt RValEnv RVal m x => Int -> Int -> RValMV -> RValT m () renvFrStkReversePopInMV lwbTo sz vs = renvFrStkPop' (\frvs sp -> mvecReverseFillFromMV' lwbTo sp sz vs frvs) sz {-# INLINE renvFrStkReversePopInMV #-} -- | Pop from the stack embedded in the top frame renvFrStkReversePopMV :: RunSem RValCxt RValEnv RVal m x => Int -> RValT m RValMV -- renvFrStkReversePopMV sz = renvFrStkPop' (\frvs sp -> mvecAlloc sz >>= \vs -> mvecReverseFillFromMV' 0 sp sz vs frvs >> return vs) sz renvFrStkReversePopMV sz = (liftIO $ mvecAlloc sz) >>= \vs -> renvFrStkReversePopInMV 0 sz vs >> return vs {-# INLINE renvFrStkReversePopMV #-} {-# LINE 952 "src/ehc/CoreRun/Run/Val.chs" #-} -- | Trace rsemTr' :: (PP msg, RunSem RValCxt RValEnv RVal m x) => Bool -> msg -> RValT m () rsemTr' dumpExtensive msg = do env <- get when (renvDoTrace env) $ do liftIO $ putStrLn $ show $ pp msg dumpEnvM (dumpExtensive || renvDoTraceExt env) liftIO $ hFlush stdout {- INLINE rsemTr' #-} -- | Trace rsemTr :: (PP msg, RunSem RValCxt RValEnv RVal m x) => msg -> RValT m () rsemTr = rsemTr' False {- INLINE rsemTr #-} {-# LINE 973 "src/ehc/CoreRun/Run/Val.chs" #-} -- | RunT' variant for Val type RValT m a = RunT' RValCxt RValEnv RVal m a -- type RValT m a = RunT' () () RValEnv m a