mirror of
https://github.com/haskell/ghcide.git
synced 2025-01-05 17:33:05 +03:00
Force values in setValues and getValues (#2494)
Otherwise, we can end up retaining references to the old map which prevent it from being garbage collected. On a simple testcase that repeatedly opens and closes a module, this seems to make memory usage constant whereas it was increasing each time before.
This commit is contained in:
parent
79c04cd202
commit
7e0f263720
@ -248,16 +248,31 @@ setValues :: IdeRule k v
|
|||||||
-> NormalizedFilePath
|
-> NormalizedFilePath
|
||||||
-> Value v
|
-> Value v
|
||||||
-> IO ()
|
-> IO ()
|
||||||
setValues state key file val = modifyVar_ state $
|
setValues state key file val = modifyVar_ state $ \vals -> do
|
||||||
pure . HMap.insert (file, Key key) (fmap toDyn val)
|
-- Force to make sure the old HashMap is not retained
|
||||||
|
evaluate $ HMap.insert (file, Key key) (fmap toDyn val) vals
|
||||||
|
|
||||||
-- | We return Nothing if the rule has not run and Just Failed if it has failed to produce a value.
|
-- | We return Nothing if the rule has not run and Just Failed if it has failed to produce a value.
|
||||||
getValues :: forall k v. IdeRule k v => Var Values -> k -> NormalizedFilePath -> IO (Maybe (Value v))
|
getValues :: forall k v. IdeRule k v => Var Values -> k -> NormalizedFilePath -> IO (Maybe (Value v))
|
||||||
getValues state key file = do
|
getValues state key file = do
|
||||||
vs <- readVar state
|
vs <- readVar state
|
||||||
return $ do
|
case HMap.lookup (file, Key key) vs of
|
||||||
v <- HMap.lookup (file, Key key) vs
|
Nothing -> pure Nothing
|
||||||
pure $ fmap (fromJust . fromDynamic @v) v
|
Just v -> do
|
||||||
|
let r = fmap (fromJust . fromDynamic @v) v
|
||||||
|
-- Force to make sure we do not retain a reference to the HashMap
|
||||||
|
-- and we blow up immediately if the fromJust should fail
|
||||||
|
-- (which would be an internal error).
|
||||||
|
evaluate (r `seqValue` Just r)
|
||||||
|
|
||||||
|
-- | Seq the result stored in the Shake value. This only
|
||||||
|
-- evaluates the value to WHNF not NF. We take care of the latter
|
||||||
|
-- elsewhere and doing it twice is expensive.
|
||||||
|
seqValue :: Value v -> b -> b
|
||||||
|
seqValue v b = case v of
|
||||||
|
Succeeded ver v -> rnf ver `seq` v `seq` b
|
||||||
|
Stale ver v -> rnf ver `seq` v `seq` b
|
||||||
|
Failed -> b
|
||||||
|
|
||||||
-- | Open a 'IdeState', should be shut using 'shakeShut'.
|
-- | Open a 'IdeState', should be shut using 'shakeShut'.
|
||||||
shakeOpen :: (LSP.FromServerMessage -> IO ()) -- ^ diagnostic handler
|
shakeOpen :: (LSP.FromServerMessage -> IO ()) -- ^ diagnostic handler
|
||||||
|
Loading…
Reference in New Issue
Block a user