{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE TypeApplications #-} {-# LANGUAGE BlockArguments #-} ----------------------------------------------------------------------------- -- | -- Module : GHC.StgToJS.Rts.Rts -- Copyright : (c) The University of Glasgow 2001 -- License : BSD-style (see the file LICENSE) -- -- Maintainer : Jeffrey Young -- Luite Stegeman -- Sylvain Henry -- Josh Meredith -- Stability : experimental -- -- Top level driver of the JavaScript Backend RTS. This file is an -- implementation of the JS RTS for the JS backend written as an EDSL in -- Haskell. It assumes the existence of pre-generated JS functions, included as -- js-sources in base. These functions are similarly assumed for non-inline -- Primops, See 'GHC.StgToJS.Prim'. Most of the elements in this module are -- constants in Haskell Land which define pieces of the JS RTS. -- ----------------------------------------------------------------------------- module GHC.StgToJS.Rts.Rts ( rts , assignRegs ) where import GHC.Prelude import GHC.JS.JStg.Syntax import GHC.JS.JStg.Monad import GHC.JS.Make hiding (trace) import GHC.JS.Ident import GHC.StgToJS.Apply import GHC.StgToJS.Closure import GHC.StgToJS.Heap import GHC.StgToJS.Profiling import GHC.StgToJS.Regs import GHC.StgToJS.Types import GHC.StgToJS.Stack import GHC.Data.FastString import GHC.Types.Unique.Map import Data.Array import Data.Monoid import Data.Char (toLower, toUpper) import qualified Data.Bits as Bits -- | The garbageCollector resets registers and result variables. garbageCollector :: JSM JStgStat garbageCollector = jBlock [ jFunction' (global "h$resetRegisters") (return $ mconcat $ map resetRegister [minBound..maxBound]) , jFunction' (global "h$resetResultVars") (return $ mconcat $ map resetResultVar [minBound..maxBound]) ] -- | Reset the register 'r' in JS Land. Note that this "resets" by setting the -- register to a dummy variable called "null", /not/ by setting to JS's nil -- value. resetRegister :: StgReg -> JStgStat resetRegister r = toJExpr r |= null_ -- | Reset the return variable 'r' in JS Land. Note that this "resets" by -- setting the register to a dummy variable called "null", /not/ by setting to -- JS's nil value. resetResultVar :: StgRet -> JStgStat resetResultVar r = toJExpr r |= null_ -- | Define closures based on size, these functions are syntactic sugar, e.g., a -- Haskell function which generates some useful JS. Each Closure constructor -- follows the naming convention h$cN, where N is a natural number. For example, -- h$c (with the nat omitted) is a JS Land Constructor for a closure in JS land -- which has a single entry function 'f', and no fields; identical to h$c0. h$c1 -- is a JS Land Constructor for a closure with an entry function 'f', and a -- /single/ field 'x1', 'Just foo' is an example of this kind of closure. h$c2 -- is a JS Land Constructor for a closure with an entry function and two data -- fields: 'x1' and 'x2'. And so on. Note that this has JIT performance -- implications; you should use h$c1, h$c2, h$c3, ... h$c24 instead of making -- objects manually so layouts and fields can be changed more easily and so the -- JIT can optimize better. closureConstructors :: StgToJSConfig -> JSM JStgStat closureConstructors s = do closures <- mapM mkClosureCon (Nothing : map Just [0..jsClosureCount]) fillers <- mapM mkDataFill [1..jsClosureCount] return $ BlockStat $ closures ++ fillers where prof = csProf s (ccArg,ccVal) -- the cc argument happens to be named just like the cc field... | prof = ([Var $ global closureCC_], Just (var closureCC_)) | otherwise = ([], Nothing) addCCArg' as = as ++ ccArg traceAlloc x | csTraceRts s = appS "h$traceAlloc" [x] | otherwise = mempty notifyAlloc x | csDebugAlloc s = appS "h$debugAlloc_notifyAlloc" [x] | otherwise = mempty -- only JSVal can typically contain undefined or null -- although it's possible (and legal) to make other Haskell types -- to contain JS refs directly -- this can cause false positives here checkC :: JSM JStgStat checkC | csAssertRts s = jVar $ \msg -> jwhenS (var "arguments" .! 0 .!==. jString "h$ghczminternalZCGHCziInternalziJSziPrimziJSVal_con_e") <$> (loop 1 (.<. var "arguments" .^ "length") (\i -> return $ mconcat [msg |= jString "warning: undefined or null in argument: " + i + jString " allocating closure: " + (var "arguments" .! 0 .^ "n") , appS "h$log" [msg] , jwhenS (var "console" .&&. (var "console" .^ "trace")) ((var "console" .^ "trace") `ApplStat` [msg]) , postIncrS i ])) | otherwise = pure mempty -- h$d is never used for JSVal (since it's only for constructors with -- at least three fields, so we always warn here checkD | csAssertRts s = loop 0 (.<. var "arguments" .^ "length") (\i -> jwhenS ((var "arguments" .! i .===. null_) .||. (var "arguments" .! i .===. undefined_)) <$> (jVar \msg-> return $ mconcat [ msg |= jString "warning: undefined or null in argument: " + i + jString " allocating fields" , jwhenS (var "console" .&&. (var "console" .^ "trace")) ((var "console" .^ "trace") `ApplStat` [msg]) ])) | otherwise = pure mempty -- special case handler, the key difference is a call to @jFunction@ instead -- of @jFunctionSized@ singleton_closure_con name = jFunction (global $ mkFastString name) $ \(MkSolo f) -> do chk_c <- checkC jVar $ \x -> return $ mconcat $ [ chk_c , x |= newClosure (mkClosure f mempty 0 ccVal) , notifyAlloc x , traceAlloc x , returnS x ] mkClosureCon :: Maybe Int -> JSM JStgStat -- the h$c special case mkClosureCon Nothing = singleton_closure_con "h$c" -- the h$c0 special case mkClosureCon (Just 0) = singleton_closure_con "h$c0" -- the rest h$c1 .. h$c24. Note that h$c1 takes 2 arguments, one for the -- entry function 'f' and another for the data field 'd1'. Thus the 1 in -- h$c1 means 1 data field argument, not just one argument mkClosureCon (Just n) = jFunctionSized funName (n + 1) funBod where funName = global $ clsName n funBod [] = pure mempty -- impossible funBod (f:vars') = do let vars = addCCArg' vars' chk_c <- checkC jVar $ \x -> return $ mconcat $ [ chk_c , x |= newClosure (mkClosure f vars 0 ccVal) , notifyAlloc x , traceAlloc x , returnS x ] mkDataFill :: Int -> JSM JStgStat mkDataFill n = jFunctionSized funName n body where funName = global $ dataName n ds = map dataFieldName [1..n] extra_args as = ValExpr . JHash . listToUniqMap $ zip ds as body :: [JStgExpr] -> JSM JStgStat body ids = do c <- checkD return (c <> returnS (extra_args ids)) -- | JS Payload to perform stack manipulation in the RTS stackManip :: JSM JStgStat stackManip = do pushes <- mapM mkPush [1..32] ppushes <- mapM mkPpush [1..255] return $ mconcat $ pushes ++ ppushes where mkPush :: Int -> JSM JStgStat mkPush n = let funName = global $ mkFastString ("h$p" ++ show n) body as = return $ ((sp |= sp + toJExpr n) <> mconcat (zipWith (\i a -> stack .! (sp - toJExpr (n-i)) |= a) [1..] as)) in jFunctionSized funName n body -- partial pushes, based on bitmap, increases Sp by highest bit mkPpush :: Integer -> JSM JStgStat mkPpush sig | sig Bits..&. (sig+1) == 0 = pure mempty -- already handled by h$p mkPpush sig = let funName = global $ mkFastString ("h$pp" ++ show sig) bits = bitsIdx sig h = last bits body args = return $ mconcat [ sp |= sp + toJExpr (h+1) , mconcat (zipWith (\b a -> stack .! (sp - toJExpr (h-b)) |= a) bits args) ] in jFunctionSized funName (length bits) body bitsIdx :: Integer -> [Int] bitsIdx n | n < 0 = error "bitsIdx: negative" | otherwise = go n 0 where go 0 _ = [] go m b | Bits.testBit m b = b : go (Bits.clearBit m b) (b+1) | otherwise = go (Bits.clearBit m b) (b+1) bhLneStats :: StgToJSConfig -> JStgExpr -> JStgExpr -> JSM JStgStat bhLneStats _s p frameSize = jVar $ \v -> return $ mconcat [ v |= stack .! p , ifS v ((sp |= sp - frameSize) <> ifS (v .===. var "h$blackhole") (returnS $ app "h$throw" [var "h$ghczminternalZCGHCziInternalziControlziExceptionziBasezinonTermination", false_]) (mconcat [r1 |= v , sp |= sp - frameSize , returnStack ])) ((stack .! p |= var "h$blackhole") <> returnS null_) ] -- | JS payload to declare the registers declRegs :: JSM JStgStat declRegs = do getters_setters <- regGettersSetters loaders <- loadRegs return $ mconcat [ global "h$regs" ||= toJExpr (JList []) , mconcat (map declReg (enumFromTo R1 R32)) , getters_setters , loaders ] where declReg r = (decl . global . mkFastString . ("h$"++) . map toLower . show) r <> BlockStat [toJExpr r |= zero_] -- | JS payload to define getters and setters on the registers. regGettersSetters :: JSM JStgStat regGettersSetters = do setters <- jFunction (global "h$getReg") (\(MkSolo n) -> return $ SwitchStat n getRegCases mempty) getters <- jFunction (global "h$setReg") (\(n,v) -> return $ SwitchStat n (setRegCases v) mempty) return $ setters <> getters where getRegCases = map (\r -> (toJExpr (jsRegToInt r) , returnS (toJExpr r))) regsFromR1 setRegCases :: JStgExpr -> [(JStgExpr,JStgStat)] setRegCases v = map (\r -> (toJExpr (jsRegToInt r), (toJExpr r |= toJExpr v) <> returnS undefined_)) regsFromR1 -- | JS payload that defines the functions to load each register loadRegs :: JSM JStgStat loadRegs = mconcat <$> mapM mkLoad [1..32] where mkLoad :: Int -> JSM JStgStat mkLoad n = let body = \args -> return $ mconcat $ zipWith (\a r -> toJExpr r |= a) args (reverse $ take n regsFromR1) fname = global $ mkFastString ("h$l" ++ show n) in jFunctionSized fname n body -- | Assign registers R1 ... Rn in descending order, that is assign Rn first. -- This function uses the 'assignRegs'' array to construct functions which set -- the registers. assignRegs :: StgToJSConfig -> [JStgExpr] -> JStgStat assignRegs _ [] = mempty assignRegs s xs | l <= 32 && not (csInlineLoadRegs s) = ApplStat (ValExpr (JVar $ assignRegs' ! l)) (reverse xs) | otherwise = mconcat . reverse $ zipWith (\r ex -> toJExpr r |= ex) (take l regsFromR1) xs where l = length xs -- | JS payload which defines an array of function symbols that set N registers -- from M parameters. For example, h$l2 compiles to: -- @ -- function h$l4(x1, x2, x3, x4) { -- h$r4 = x1; -- h$r3 = x2; -- h$r2 = x3; -- h$r1 = x4; -- }; -- @ assignRegs' :: Array Int Ident assignRegs' = listArray (1,32) (map (global . mkFastString . ("h$l"++) . show) [(1::Int)..32]) -- | JS payload to declare return variables. declRets :: JStgStat declRets = mconcat $ map (decl . global . mkFastString . ("h$"++) . map toLower . show) (enumFrom Ret1) -- | JS payload defining the types closures. closureTypes :: JSM JStgStat closureTypes = do cls_typ_nm <- closureTypeName return $ mconcat (map mkClosureType (enumFromTo minBound maxBound)) <> cls_typ_nm where mkClosureType :: ClosureType -> JStgStat mkClosureType c = let s = global . mkFastString $ "h$" ++ map toUpper (show c) ++ "_CLOSURE" in s ||= toJExpr c closureTypeName :: JSM JStgStat closureTypeName = jFunction (global "h$closureTypeName") \(MkSolo c) -> return $ mconcat (map (ifCT c) [minBound..maxBound]) <> returnS (jString "InvalidClosureType") ifCT :: JStgExpr -> ClosureType -> JStgStat ifCT arg ct = jwhenS (arg .===. toJExpr ct) (returnS (toJExpr (show ct))) -- | JS payload declaring the RTS functions. rtsDecls :: JSM JStgStat rtsDecls = do decl_stg_regs <- declRegs return $ mconcat [ global "h$currentThread" ||= null_ -- thread state object for current thread , global "h$stack" ||= null_ -- stack for the current thread , global "h$sp" ||= 0 -- stack pointer for the current thread , global "h$initStatic" ||= toJExpr (JList []) -- we need delayed initialization for static objects, push functions here to be initialized just before haskell runs , global "h$staticThunks" ||= toJExpr (jhFromList []) -- funcName -> heapidx map for srefs , global "h$staticThunksArr" ||= toJExpr (JList []) -- indices of updatable thunks in static heap , global "h$CAFs" ||= toJExpr (JList []) , global "h$CAFsReset" ||= toJExpr (JList []) -- stg registers , decl_stg_regs , declRets] -- | Generated RTS code rts :: StgToJSConfig -> JSM JStgStat rts cfg = withTag "h$RTS" $ do rts_ <- rts_gen cfg rts_decls <- rtsDecls return $ rts_decls <> rts_ -- | JS Payload which defines the embedded RTS. rts_gen :: StgToJSConfig -> JSM JStgStat rts_gen s = do let decls = [ global "h$rts_traceForeign" ||= toJExpr (csTraceForeign s) , global "h$rts_profiling" ||= toJExpr (csProf s) , global "h$ct_fun" ||= toJExpr Fun , global "h$ct_con" ||= toJExpr Con , global "h$ct_thunk" ||= toJExpr Thunk , global "h$ct_pap" ||= toJExpr Pap , global "h$ct_blackhole" ||= toJExpr Blackhole , global "h$ct_stackframe" ||= toJExpr StackFrame , global "h$vt_ptr" ||= toJExpr PtrV , global "h$vt_void" ||= toJExpr VoidV , global "h$vt_double" ||= toJExpr IntV , global "h$vt_long" ||= toJExpr LongV , global "h$vt_addr" ||= toJExpr AddrV , global "h$vt_obj" ||= toJExpr ObjV , global "h$vt_arr" ||= toJExpr ArrV ] gc <- garbageCollector closure_cons <- closureConstructors s stk_manip <- stackManip rest <- impure return $ mconcat $ pure gc <> decls <> [closure_cons, stk_manip] <> rest where impure = sequence [ jFunction' (global "h$bh") (return $ bhStats s True) , jFunction (global "h$bh_lne") (\(x, frameSize) -> bhLneStats s x frameSize) , closure (ClosureInfo (global "h$blackhole") (CIRegs 0 []) "blackhole" (CILayoutUnknown 2) CIBlackhole mempty) (return $ appS "throw" [jString "oops: entered black hole"]) , closure (ClosureInfo (global "h$blackholeTrap") (CIRegs 0 []) "blackhole" (CILayoutUnknown 2) CIThunk mempty) (return $ appS "throw" [jString "oops: entered multiple times"]) , closure (ClosureInfo (global "h$done") (CIRegs 0 [PtrV]) "done" (CILayoutUnknown 0) CIStackFrame mempty) (return $ appS "h$finishThread" [var "h$currentThread"] <> returnS (var "h$reschedule")) , closure (ClosureInfo (global "h$doneMain_e") (CIRegs 0 [PtrV]) "doneMain" (CILayoutUnknown 0) CIStackFrame mempty) (return $ returnS (var "h$doneMain")) , conClosure (global "h$false_e") "GHC.Types.False" (CILayoutFixed 0 []) 1 , conClosure (global "h$true_e" ) "GHC.Types.True" (CILayoutFixed 0 []) 2 -- generic data constructor with 1 non-heapobj field , conClosure (global "h$data1_e") "data1" (CILayoutFixed 1 [ObjV]) 1 -- generic data constructor with 2 non-heapobj fields , conClosure (global "h$data2_e") "data2" (CILayoutFixed 2 [ObjV,ObjV]) 1 , closure (ClosureInfo (global "h$noop_e") (CIRegs 1 [PtrV]) "no-op IO ()" (CILayoutFixed 0 []) (CIFun 1 0) mempty) $ return (returnS (stack .! sp)) , pure (global "h$noop" ||= ApplExpr (var "h$c0") (var "h$noop_e" : [jSystemCCS | csProf s])) , closure (ClosureInfo (global "h$catch_e") (CIRegs 0 [PtrV]) "exception handler" (CILayoutFixed 2 [PtrV,IntV]) CIStackFrame mempty) (return $ adjSpN' 3 <> returnS (stack .! sp)) , closure (ClosureInfo (global "h$dataToTag_e") (CIRegs 0 [PtrV]) "data to tag" (CILayoutFixed 0 []) CIStackFrame mempty) $ return $ mconcat [ r1 |= if_ (r1 .===. true_) 1 (if_ (typeof r1 .===. jTyObject) (r1 .^ "f" .^ "a" - 1) 0) , adjSpN' 1 , returnS (stack .! sp) ] -- function application to one argument , closure (ClosureInfo (global "h$ap1_e") (CIRegs 0 [PtrV]) "apply1" (CILayoutFixed 2 [PtrV, PtrV]) CIThunk mempty) (jVars \(d1, d2) -> return $ mconcat [ d1 |= closureField1 r1 , d2 |= closureField2 r1 , appS "h$bh" [] , profStat s enterCostCentreThunk , r1 |= d1 , r2 |= d2 , returnS (app "h$ap_1_1_fast" []) ]) -- function application to two arguments , closure (ClosureInfo (global "h$ap2_e") (CIRegs 0 [PtrV]) "apply2" (CILayoutFixed 3 [PtrV, PtrV, PtrV]) CIThunk mempty) (jVars \(d1, d2, d3) -> return $ mconcat [ d1 |= closureField1 r1 , d2 |= closureField2 r1 .^ "d1" , d3 |= closureField2 r1 .^ "d2" , appS "h$bh" [] , profStat s enterCostCentreThunk , r1 |= d1 , r2 |= d2 , r3 |= d3 , returnS (app "h$ap_2_2_fast" []) ]) -- function application to three arguments , closure (ClosureInfo (global "h$ap3_e") (CIRegs 0 [PtrV]) "apply3" (CILayoutFixed 4 [PtrV, PtrV, PtrV, PtrV]) CIThunk mempty) (jVars \(d1, d2, d3, d4) -> return $ mconcat [ d1 |= closureField1 r1 , d2 |= closureField2 r1 .^ "d1" , d3 |= closureField2 r1 .^ "d2" , d4 |= closureField2 r1 .^ "d3" , appS "h$bh" [] , r1 |= d1 , r2 |= d2 , r3 |= d3 , r4 |= d4 , returnS (app "h$ap_3_3_fast" []) ]) , closure (ClosureInfo (TxtI "h$upd_thunk_e") (CIRegs 0 [PtrV]) "updatable thunk" (CILayoutFixed 1 [PtrV]) CIThunk mempty) (jVar $ \t -> return $ mconcat [t |= closureField1 r1 , adjSp' 2 , stack .! (sp - 1) |= r1 , stack .! sp |= var "h$upd_frame" , closureEntry r1 |= var "h$blackhole" , closureField1 r1 |= var "h$currentThread" , closureField2 r1 |= null_ , r1 |= t , returnS (app "h$ap_0_0_fast" []) ] ) -- select first field , closure (ClosureInfo (global "h$select1_e") (CIRegs 0 [PtrV]) "select1" (CILayoutFixed 1 [PtrV]) CIThunk mempty) (jVar \t -> return $ mconcat [ t |= closureField1 r1 , adjSp' 3 , stack .! (sp - 2) |= r1 , stack .! (sp - 1) |= var "h$upd_frame" , stack .! sp |= var "h$select1_ret" , closureEntry r1 |= var "h$blackhole" , closureField1 r1 |= var "h$currentThread" , closureField2 r1 |= null_ , r1 |= t , returnS (app "h$ap_0_0_fast" []) ]) , closure (ClosureInfo (global "h$select1_ret") (CIRegs 0 [PtrV]) "select1ret" (CILayoutFixed 0 []) CIStackFrame mempty) (return $ (r1 |= closureField1 r1) <> adjSpN' 1 <> returnS (app "h$ap_0_0_fast" []) ) -- select second field of a two-field constructor , closure (ClosureInfo (global "h$select2_e") (CIRegs 0 [PtrV]) "select2" (CILayoutFixed 1 [PtrV]) CIThunk mempty) (jVar \t -> return $ mconcat [t |= closureField1 r1 , adjSp' 3 , stack .! (sp - 2) |= r1 , stack .! (sp - 1) |= var "h$upd_frame" , stack .! sp |= var "h$select2_ret" , closureEntry r1 |= var "h$blackhole" , closureField1 r1 |= var "h$currentThread" , closureField2 r1 |= null_ , r1 |= t , returnS (app "h$ap_0_0_fast" []) ] ) , closure (ClosureInfo (global "h$select2_ret") (CIRegs 0 [PtrV]) "select2ret" (CILayoutFixed 0 []) CIStackFrame mempty) $ return $ mconcat [ r1 |= closureField2 r1 , adjSpN' 1 , returnS (app "h$ap_0_0_fast" []) ] , closure (ClosureInfo (global "h$keepAlive_e") (CIRegs 0 [PtrV]) "keepAlive" (CILayoutFixed 1 [PtrV]) CIStackFrame mempty) (return $ mconcat [ adjSpN' 2 , returnS (stack .! sp) ] ) -- a thunk that just raises a synchronous exception , closure (ClosureInfo (global "h$raise_e") (CIRegs 0 [PtrV]) "h$raise_e" (CILayoutFixed 0 []) CIThunk mempty) (return $ returnS (app "h$throw" [closureField1 r1, false_])) , closure (ClosureInfo (global "h$raiseAsync_e") (CIRegs 0 [PtrV]) "h$raiseAsync_e" (CILayoutFixed 0 []) CIThunk mempty) (return $ returnS (app "h$throw" [closureField1 r1, true_])) , closure (ClosureInfo (global "h$raiseAsync_frame") (CIRegs 0 []) "h$raiseAsync_frame" (CILayoutFixed 1 []) CIStackFrame mempty) (jVar \ex -> return $ mconcat [ ex |= stack .! (sp - 1) , adjSpN' 2 , returnS (app "h$throw" [ex, true_]) ]) {- reduce result if it's a thunk, follow if it's an ind add this to the stack if you want the outermost result to always be reduced to whnf, and not an ind -} , closure (ClosureInfo (global "h$reduce") (CIRegs 0 [PtrV]) "h$reduce" (CILayoutFixed 0 []) CIStackFrame mempty) (return $ ifS (isThunk r1) (returnS (r1 .^ "f")) (adjSpN' 1 <> returnS (stack .! sp)) ) , rtsApply s , closureTypes , closure (ClosureInfo (global "h$runio_e") (CIRegs 0 [PtrV]) "runio" (CILayoutFixed 1 [PtrV]) CIThunk mempty) $ return $ mconcat [ r1 |= closureField1 r1 , stack .! PreInc sp |= var "h$ap_1_0" , returnS (var "h$ap_1_0") ] , closure (ClosureInfo (global "h$flushStdout_e") (CIRegs 0 []) "flushStdout" (CILayoutFixed 0 []) CIThunk mempty) $ return $ mconcat [ r1 |= var "h$ghczminternalZCGHCziInternalziIOziHandlezihFlush" , r2 |= var "h$ghczminternalZCGHCziInternalziIOziHandleziFDzistdout" , returnS (app "h$ap_1_1_fast" []) ] , pure $ global "h$flushStdout" ||= app "h$static_thunk" [var "h$flushStdout_e"] -- the scheduler pushes this frame when suspending a thread that -- has not called h$reschedule explicitly , closure (ClosureInfo (global "h$restoreThread") (CIRegs 0 []) "restoreThread" CILayoutVariable CIStackFrame mempty) (jVars \(f,frameSize,nregs) -> do set_regs <- loop 1 (.<=. nregs) (\i -> return $ appS "h$setReg" [i, stack .! (sp - 2 - i)] <> postIncrS i) return $ mconcat [f |= stack .! (sp - 2) , frameSize |= stack .! (sp - 1) , nregs |= frameSize - 3 , set_regs , sp |= sp - frameSize , returnS f ]) -- return a closure in the stack frame to the next thing on the stack , closure (ClosureInfo (global "h$return") (CIRegs 0 []) "return" (CILayoutFixed 1 [PtrV]) CIStackFrame mempty) (return $ (r1 |= stack .! (sp - 1)) <> adjSpN' 2 <> returnS (stack .! sp)) -- return a function in the stack frame for the next call , closure (ClosureInfo (global "h$returnf") (CIRegs 0 [PtrV]) "returnf" (CILayoutFixed 1 [ObjV]) CIStackFrame mempty) (jVar \r -> return $ mconcat [ r |= stack .! (sp - 1) , adjSpN' 2 , returnS r ]) -- return this function when the scheduler needs to come into action -- (yield, delay etc), returning thread needs to push all relevant -- registers to stack frame, thread will be resumed by calling the stack top , closure (ClosureInfo (global "h$reschedule") (CIRegs 0 []) "reschedule" (CILayoutFixed 0 []) CIThunk mempty) (return $ returnS $ var "h$reschedule") -- debug thing, insert on stack to dump current result, should be boxed , closure (ClosureInfo (global "h$dumpRes") (CIRegs 0 [PtrV]) "dumpRes" (CILayoutFixed 1 [ObjV]) CIThunk mempty) (jVar \re -> return $ mconcat [ appS "h$log" [jString "h$dumpRes result: " + stack .! (sp-1)] , appS "h$log" [r1] , appS "h$log" [app "h$collectProps" [r1]] , jwhenS ((r1 .^ "f") .&&. (r1 .^ "f" .^ "n")) (appS "h$log" [jString "name: " + r1 .^ "f" .^ "n"]) , jwhenS (ApplExpr (r1 .^ "hasOwnProperty") [jString closureField1_]) (appS "h$log" [jString "d1: " + closureField1 r1]) , jwhenS (ApplExpr (r1 .^ "hasOwnProperty") [jString closureField2_]) (appS "h$log" [jString "d2: " + closureField2 r1]) , jwhenS (r1 .^ "f") $ mconcat [ re |= New (app "RegExp" [jString "([^\\n]+)\\n(.|\\n)*"]) , appS "h$log" [jString "function" + ApplExpr (ApplExpr ((jString "" + r1 .^ "f") .^ "substring") [0, 50] .^ "replace") [r1, jString "$1"]] ] , adjSpN' 2 , r1 |= null_ , returnS (stack .! sp) ]) , closure (ClosureInfo (global "h$resume_e") (CIRegs 0 [PtrV]) "resume" (CILayoutFixed 0 []) CIThunk mempty) (jVar \ss -> do update_stk <- loop 0 (.<. ss .^ "length") (\i -> return $ (stack .! (sp+1+i) |= ss .! i) <> postIncrS i) return $ mconcat [ss |= closureField1 r1 , updateThunk' s , update_stk , sp |= sp + ss .^ "length" , r1 |= null_ , returnS (stack .! sp) ]) , closure (ClosureInfo (global "h$unmaskFrame") (CIRegs 0 [PtrV]) "unmask" (CILayoutFixed 0 []) CIStackFrame mempty) (return $ (var "h$currentThread" .^ "mask" |= 0) <> adjSpN' 1 -- back to scheduler to give us async exception if pending <> ifS (var "h$currentThread" .^ "excep" .^ "length" .>. 0) (push' s [r1, var "h$return"] <> returnS (var "h$reschedule")) (returnS (stack .! sp))) , closure (ClosureInfo (global "h$maskFrame") (CIRegs 0 [PtrV]) "mask" (CILayoutFixed 0 []) CIStackFrame mempty) (return $ (var "h$currentThread" .^ "mask" |= 2) <> adjSpN' 1 <> returnS (stack .! sp)) , closure (ClosureInfo (global "h$maskUnintFrame") (CIRegs 0 [PtrV]) "maskUnint" (CILayoutFixed 0 []) CIStackFrame mempty) (return $ (var "h$currentThread" .^ "mask" |= 1) <> adjSpN' 1 <> returnS (stack .! sp)) , closure (ClosureInfo (global "h$unboxFFIResult") (CIRegs 0 [PtrV]) "unboxFFI" (CILayoutFixed 0 []) CIStackFrame mempty) (jVar \d -> do set_regs <- loop 0 (.<. d .^ "length") (\i -> return $ appS "h$setReg" [i + 1, d .! i] <> postIncrS i) return $ mconcat [ d |= closureField1 r1 , set_regs , adjSpN' 1 , returnS (stack .! sp) ]) , closure (ClosureInfo (global "h$unbox_e") (CIRegs 0 [PtrV]) "unboxed value" (CILayoutFixed 1 [DoubleV]) CIThunk mempty) (return $ (r1 |= closureField1 r1) <> returnS (stack .! sp)) , closure (ClosureInfo (global "h$retryInterrupted") (CIRegs 0 [ObjV]) "retry interrupted operation" (CILayoutFixed 1 [ObjV]) CIStackFrame mempty) (jVar \a -> return $ mconcat [ a |= stack .! (sp - 1) , adjSpN' 2 , returnS (ApplExpr (a .! 0 .^ "apply") [var "this", ApplExpr (a .^ "slice") [1]]) ]) , closure (ClosureInfo (global "h$atomically_e") (CIRegs 0 [PtrV]) "atomic operation" (CILayoutFixed 1 [PtrV]) CIStackFrame mempty) (return $ ifS (app "h$stmValidateTransaction" []) (appS "h$stmCommitTransaction" [] <> adjSpN' 2 <> returnS (stack .! sp)) (returnS (app "h$stmStartTransaction" [stack .! (sp - 1)]))) , closure (ClosureInfo (global "h$stmCatchRetry_e") (CIRegs 0 [PtrV]) "catch retry" (CILayoutFixed 1 [PtrV]) CIStackFrame mempty) (return $ adjSpN' 2 <> appS "h$stmCommitTransaction" [] <> returnS (stack .! sp)) , closure (ClosureInfo (global "h$catchStm_e") (CIRegs 0 [PtrV]) "STM catch" (CILayoutFixed 3 [ObjV,PtrV,ObjV]) CIStackFrame mempty) (return $ adjSpN' 4 <> appS "h$stmCommitTransaction" [] <> returnS (stack .! sp)) , closure (ClosureInfo (global "h$stmResumeRetry_e") (CIRegs 0 [PtrV]) "resume retry" (CILayoutFixed 0 []) CIStackFrame mempty) (jVar \blocked -> return $ mconcat [ jwhenS (stack .! (sp - 2) .!==. var "h$atomically_e") (appS "throw" [jString "h$stmResumeRetry_e: unexpected value on stack"]) , blocked |= stack .! (sp - 1) , adjSpN' 2 , appS "h$stmRemoveBlockedThread" [blocked, var "h$currentThread"] , returnS (app "h$stmStartTransaction" [stack .! (sp - 1)]) ]) , closure (ClosureInfo (global "h$lazy_e") (CIRegs 0 [PtrV]) "generic lazy value" (CILayoutFixed 0 []) CIThunk mempty) (jVar \x -> return $ mconcat [x |= ApplExpr (closureField1 r1) [] , appS "h$bh" [] , profStat s enterCostCentreThunk , r1 |= x , returnS (stack .! sp) ]) -- Top-level statements to generate only in profiling mode , fmap (profStat s) $ (closure (ClosureInfo (global "h$setCcs_e") (CIRegs 0 [PtrV]) "set cost centre stack" (CILayoutFixed 1 [ObjV]) CIStackFrame mempty) (return $ appS "h$restoreCCS" [ stack .! (sp - 1)] <> adjSpN' 2 <> returnS (stack .! sp))) ]