Initial implementation of UIO2

This commit is contained in:
Ryan Trinkle 2023-04-10 11:15:20 -04:00
parent 774e5fbb73
commit 5da5ab9045

65
main.hs
View File

@ -2,6 +2,7 @@
{-# LANGUAGE LambdaCase #-} {-# LANGUAGE LambdaCase #-}
{-# LANGUAGE UnboxedTuples #-} {-# LANGUAGE UnboxedTuples #-}
{-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE RecursiveDo #-}
import System.Mem import System.Mem
import System.Mem.Weak import System.Mem.Weak
import Data.IORef import Data.IORef
@ -21,6 +22,7 @@ import Control.Concurrent
import Data.Foldable import Data.Foldable
import Data.These import Data.These
import Unsafe.Coerce import Unsafe.Coerce
import Control.Monad.Fix
main :: IO () main :: IO ()
main = do main = do
@ -30,6 +32,67 @@ main = do
testGraphX testGraphX
testGraphO testGraphO
-- Unordered IO - we want to allocate things, strictly evaluate things, etc., but we don't actually care what order it is done in
newtype UIO2 a = UIO2 { unUIO2 :: State# RealWorld -> ((), a) }
instance Functor UIO2 where
{-# INLINE fmap #-}
fmap f x = x >>= (pure . f)
instance Applicative UIO2 where
{-# INLINE pure #-}
{-# INLINE (*>) #-}
{-# INLINE (<*>) #-}
pure x = UIO2 (\s -> ((), x))
UIO2 m *> UIO2 k = UIO2 (\s ->
let (ms, _) = m s
(ks, b) = k s
in (ms `seq` ks, b))
UIO2 m <*> UIO2 k = UIO2 (\s ->
let (ms, f) = m s
(ks, x) = k s
in (ms `seq` ks, f x))
instance Monad UIO2 where
{-# INLINE (>>) #-}
{-# INLINE (>>=) #-}
(>>) = (*>)
UIO2 m >>= k = UIO2 (\s ->
let (ms, a) = m s
(ks, b) = unUIO2 (k a) s
in (ms `seq` ks, b))
instance MonadFix UIO2 where
mfix k = do
m <- unordered newEmptyMVar
UIO2 (\s ->
let (rs, ans) = unUIO2 (unordered $ readMVar m) s
(ks, result) = unUIO2 (k ans) s
(ps, _) = unUIO2 (unordered $ putMVar m result) s
in (ps `seq` ks `seq` rs, result))
runUIO2 :: UIO2 a -> IO a
runUIO2 (UIO2 m) = do
(done, result) <- IO (\s -> (# s, m s #))
evaluate result
evaluate done
pure result
-- The following is marked NOINLINE because unsafeDupablePerformIO is marked NOINLINE. I don't really understand it.
{-# NOINLINE unordered #-}
unordered :: IO a -> UIO2 a
unordered (IO m) = UIO2 (\s -> case m s of (# _, x #) -> ((), x))
testUIO2 :: IO ()
testUIO2 = do
r <- runUIO2 $ mdo
unordered $ writeIORef r 5
r <- unordered $ newIORef 2
pure r
print =<< readIORef r
type UIO = IO
data WeakBagInput a data WeakBagInput a
data WeakBagOutput a data WeakBagOutput a
data WeakMutVarInput a = WeakMutVarInput (WeakWithLifespan (IORef a)) data WeakMutVarInput a = WeakMutVarInput (WeakWithLifespan (IORef a))
@ -100,8 +163,6 @@ weakEffect f (Effect e) = fmap Effect $ forWeakWithLifespan' e $ \fe a -> do
runCoeffect :: Coeffect a -> IO a runCoeffect :: Coeffect a -> IO a
runCoeffect (Coeffect c) = c runCoeffect (Coeffect c) = c
-- Unordered IO - we want to allocate things, strictly evaluate things, etc., but we don't actually care what order it is done in
type UIO = IO
emptyWeak :: Weak a emptyWeak :: Weak a
emptyWeak = unsafeCoerce emptyWeakUnit emptyWeak = unsafeCoerce emptyWeakUnit