Skip to content

Commit

Permalink
Consider array reallocations when doing monadic state updates
Browse files Browse the repository at this point in the history
Fixes #237.
  • Loading branch information
arybczak committed Sep 4, 2024
1 parent e51ab36 commit 2e8c933
Show file tree
Hide file tree
Showing 6 changed files with 29 additions and 11 deletions.
2 changes: 2 additions & 0 deletions effectful-core/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,8 @@
`throwError_` for them.
- `ThrowError` operation from the dynamic version of the `Error` effect was
replaced with `ThrowErrorWith`.
- `stateEnv` and `modifyEnv` now take pure modification functions. Use a
combination of `getEnv` and `putEnv` for forward compatibility.

# effectful-core-2.3.1.0 (2024-06-07)
* Drop support for GHC 8.8.
Expand Down
8 changes: 4 additions & 4 deletions effectful-core/src/Effectful/Internal/Env.hs
Original file line number Diff line number Diff line change
Expand Up @@ -321,23 +321,23 @@ putEnv env e = do
stateEnv
:: forall e es a. e :> es
=> Env es -- ^ The environment.
-> (EffectRep (DispatchOf e) e -> IO (a, EffectRep (DispatchOf e) e))
-> (EffectRep (DispatchOf e) e -> (a, EffectRep (DispatchOf e) e))
-> IO a
stateEnv env f = do
(i, es) <- getLocation @e env
(a, e) <- f . fromAny =<< readSmallArray es i
(a, e) <- f . fromAny <$> readSmallArray es i
writeSmallArray' es i (toAny e)
pure a

-- | Modify the data type in the environment (in place).
modifyEnv
:: forall e es. e :> es
=> Env es -- ^ The environment.
-> (EffectRep (DispatchOf e) e -> IO (EffectRep (DispatchOf e) e))
-> (EffectRep (DispatchOf e) e -> (EffectRep (DispatchOf e) e))
-> IO ()
modifyEnv env f = do
(i, es) <- getLocation @e env
e <- f . fromAny =<< readSmallArray es i
e <- f . fromAny <$> readSmallArray es i
writeSmallArray' es i (toAny e)

-- | Determine location of the effect in the environment.
Expand Down
8 changes: 5 additions & 3 deletions effectful-core/src/Effectful/Internal/Monad.hs
Original file line number Diff line number Diff line change
Expand Up @@ -604,7 +604,7 @@ stateStaticRep
=> (StaticRep e -> (a, StaticRep e))
-- ^ The function to modify the representation.
-> Eff es a
stateStaticRep f = unsafeEff $ \es -> stateEnv es (pure . f)
stateStaticRep f = unsafeEff $ \es -> stateEnv es f

-- | Apply the monadic function to the current representation of the effect and
-- return a value.
Expand All @@ -614,7 +614,9 @@ stateStaticRepM
-- ^ The function to modify the representation.
-> Eff es a
stateStaticRepM f = unsafeEff $ \es -> E.mask $ \unmask -> do
stateEnv es $ unmask . (`unEff` es) . f
(a, e) <- unmask . (`unEff` es) . f =<< getEnv es
putEnv es e
pure a

-- | Execute a computation with a temporarily modified representation of the
-- effect.
Expand All @@ -626,6 +628,6 @@ localStaticRep
-> Eff es a
localStaticRep f m = unsafeEff $ \es -> do
inlineBracket
(stateEnv es $ \s -> pure (s, f s))
(stateEnv es $ \s -> (s, f s))
(\s -> putEnv es s)
(\_ -> unEff m es)
4 changes: 2 additions & 2 deletions effectful-core/src/Effectful/Writer/Static/Local.hs
Original file line number Diff line number Diff line change
Expand Up @@ -78,13 +78,13 @@ tell w = stateStaticRep $ \(Writer w0) -> ((), Writer (w0 <> w))
-- "Hi there!"
listen :: (Writer w :> es, Monoid w) => Eff es a -> Eff es (a, w)
listen m = unsafeEff $ \es -> mask $ \unmask -> do
w0 <- stateEnv es $ \(Writer w) -> pure (w, Writer mempty)
w0 <- stateEnv es $ \(Writer w) -> (w, Writer mempty)
a <- unmask (unEff m es) `onException` merge es w0
(a, ) <$> merge es w0
where
merge es w0 =
-- If an exception is thrown, restore w0 and keep parts of w1.
stateEnv es $ \(Writer w1) -> pure (w1, Writer (w0 <> w1))
stateEnv es $ \(Writer w1) -> (w1, Writer (w0 <> w1))

-- | Execute an action and append its output to the overall output of the
-- 'Writer', then return the final value along with a function of the recorded
Expand Down
2 changes: 1 addition & 1 deletion effectful-core/src/Effectful/Writer/Static/Shared.hs
Original file line number Diff line number Diff line change
Expand Up @@ -88,7 +88,7 @@ listen m = unsafeEff $ \es -> do
uninterruptibleMask $ \unmask -> do
v1 <- newMVar' mempty
-- Replace thread local MVar with a fresh one for isolated listening.
v0 <- stateEnv es $ \(Writer v) -> pure (v, Writer v1)
v0 <- stateEnv es $ \(Writer v) -> (v, Writer v1)
a <- unmask (unEff m es) `onException` merge es v0 v1
(a, ) <$> merge es v0 v1
where
Expand Down
16 changes: 15 additions & 1 deletion effectful/tests/StateTests.hs
Original file line number Diff line number Diff line change
@@ -1,13 +1,17 @@
module StateTests (stateTests) where

import Control.Exception.Lifted qualified as LE
import Control.Monad
import Control.Monad.Catch qualified as E
import Test.Tasty
import Test.Tasty.HUnit
import UnliftIO.Exception qualified as UE

import Effectful
import Effectful.Dispatch.Dynamic
import Effectful.Dispatch.Static
import Effectful.Internal.Env
import Effectful.Internal.Utils
import Effectful.State.Static.Local
import Utils qualified as U

Expand All @@ -34,9 +38,19 @@ test_evalState = runEff $ do

test_stateM :: Assertion
test_stateM = runEff $ do
(a, b) <- runState "hi" . stateM $ \s -> pure (s, s ++ "!!!")
(a, b) <- runState "hi" $ do
stateM $ \s -> do
effs0 <- getEffectReps
-- Trigger reallocation of the internal array in Storage.
_ <- evalState () $ pure ()
effs1 <- getEffectReps
when (effs0 == effs1) $ do
U.assertFailure "Internal array was not reallocated"
pure (s, s ++ "!!!")
U.assertEqual "correct a" "hi" a
U.assertEqual "correct b" "hi!!!" b
where
getEffectReps = unsafeEff $ \es -> stEffects <$> readIORef' (envStorage es)

test_deepStack :: Assertion
test_deepStack = runEff $ do
Expand Down

0 comments on commit 2e8c933

Please sign in to comment.