More UIO2 work

This commit is contained in:
Ryan Trinkle 2023-04-10 11:30:30 -04:00
parent 5da5ab9045
commit b6c2099b02

37
main.hs
View File

@ -23,6 +23,8 @@ import Data.Foldable
import Data.These import Data.These
import Unsafe.Coerce import Unsafe.Coerce
import Control.Monad.Fix import Control.Monad.Fix
import Data.List.NonEmpty (NonEmpty (..))
import Data.Semigroup
main :: IO () main :: IO ()
main = do main = do
@ -32,8 +34,23 @@ main = do
testGraphX testGraphX
testGraphO testGraphO
data Done = Done
instance Semigroup Done where
{-# INLINE (<>) #-}
(<>) = seq
sconcat = mconcat . toList
stimes _ d = d
instance Monoid Done where
{-# INLINE mempty #-}
mempty = Done
mconcat = \case
[] -> Done
x : xs -> x `seq` mconcat xs
-- Unordered IO - we want to allocate things, strictly evaluate things, etc., but we don't actually care what order it is done in -- 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) } newtype UIO2 a = UIO2 { unUIO2 :: State# RealWorld -> (Done, a) }
instance Functor UIO2 where instance Functor UIO2 where
{-# INLINE fmap #-} {-# INLINE fmap #-}
@ -43,7 +60,7 @@ instance Applicative UIO2 where
{-# INLINE pure #-} {-# INLINE pure #-}
{-# INLINE (*>) #-} {-# INLINE (*>) #-}
{-# INLINE (<*>) #-} {-# INLINE (<*>) #-}
pure x = UIO2 (\s -> ((), x)) pure x = UIO2 (\s -> (Done, x))
UIO2 m *> UIO2 k = UIO2 (\s -> UIO2 m *> UIO2 k = UIO2 (\s ->
let (ms, _) = m s let (ms, _) = m s
(ks, b) = k s (ks, b) = k s
@ -63,31 +80,27 @@ instance Monad UIO2 where
in (ms `seq` ks, b)) in (ms `seq` ks, b))
instance MonadFix UIO2 where instance MonadFix UIO2 where
mfix k = do mfix k = UIO2 (\s ->
m <- unordered newEmptyMVar let (ks, result) = unUIO2 (k result) s
UIO2 (\s -> in (ks, result))
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 a -> IO a
runUIO2 (UIO2 m) = do runUIO2 (UIO2 m) = do
(done, result) <- IO (\s -> (# s, m s #)) (done, result) <- IO (\s -> (# s, m s #))
evaluate result
evaluate done evaluate done
pure result pure result
-- The following is marked NOINLINE because unsafeDupablePerformIO is marked NOINLINE. I don't really understand it. -- The following is marked NOINLINE because unsafeDupablePerformIO is marked NOINLINE. I don't really understand it.
{-# NOINLINE unordered #-} {-# NOINLINE unordered #-}
unordered :: IO a -> UIO2 a unordered :: IO a -> UIO2 a
unordered (IO m) = UIO2 (\s -> case m s of (# _, x #) -> ((), x)) unordered (IO m) = UIO2 (\s -> case m s of (# _, x #) -> (Done, x))
testUIO2 :: IO () testUIO2 :: IO ()
testUIO2 = do testUIO2 = do
r <- runUIO2 $ mdo r <- runUIO2 $ mdo
unordered $ writeIORef r 5 unordered $ atomicModifyIORef' r $ \v -> (v + 5, ())
r <- unordered $ newIORef 2 r <- unordered $ newIORef 2
unordered $ atomicModifyIORef' r $ \v -> (v + 10, ())
pure r pure r
print =<< readIORef r print =<< readIORef r