Weak graph is working
This commit is contained in:
parent
a671900c9b
commit
e49fb1ef18
165
main.hs
165
main.hs
@ -23,6 +23,10 @@ import Data.Foldable
|
|||||||
main :: IO ()
|
main :: IO ()
|
||||||
main = do
|
main = do
|
||||||
testDualWeak
|
testDualWeak
|
||||||
|
testWeakChain
|
||||||
|
testGraphChain
|
||||||
|
testGraphX
|
||||||
|
testGraphO
|
||||||
|
|
||||||
-- Demonstrate that a Weak's key keeps its value alive, even if the Weak is dead
|
-- Demonstrate that a Weak's key keeps its value alive, even if the Weak is dead
|
||||||
testWeakChain :: IO ()
|
testWeakChain :: IO ()
|
||||||
@ -30,9 +34,9 @@ testWeakChain = do
|
|||||||
a <- newIORef ()
|
a <- newIORef ()
|
||||||
b <- newIORef ()
|
b <- newIORef ()
|
||||||
_ <- mkWeakWithIORefKey a b
|
_ <- mkWeakWithIORefKey a b
|
||||||
w <- mkWeakWithIORefKey b "Still Alive"
|
w <- mkWeakWithIORefKey b "b is still alive"
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
print =<< deRefWeak w
|
Just "b is still alive" <- deRefWeak w
|
||||||
touch a
|
touch a
|
||||||
|
|
||||||
testGraphChain :: IO ()
|
testGraphChain :: IO ()
|
||||||
@ -40,18 +44,18 @@ testGraphChain = do
|
|||||||
do
|
do
|
||||||
(nai, neo) <- buildGraphChain
|
(nai, neo) <- buildGraphChain
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
print =<< findForward nai
|
["a", "b", "c", "d", "e"] <- toList <$> findForwardVals nai
|
||||||
print =<< findBackward neo
|
["a", "b", "c", "d", "e"] <- toList <$> findBackwardVals neo
|
||||||
touch (nai, neo)
|
touch (nai, neo)
|
||||||
do
|
do
|
||||||
(nai, _) <- buildGraphChain
|
(nai, _) <- buildGraphChain
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
print =<< findForward nai
|
[] <- toList <$> findForwardVals nai
|
||||||
touch nai
|
touch nai
|
||||||
do
|
do
|
||||||
(_, neo) <- buildGraphChain
|
(_, neo) <- buildGraphChain
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
print =<< findBackward neo
|
[] <- toList <$> findBackwardVals neo
|
||||||
touch neo
|
touch neo
|
||||||
|
|
||||||
buildGraphChain :: IO (NodeInput String, NodeOutput String)
|
buildGraphChain :: IO (NodeInput String, NodeOutput String)
|
||||||
@ -67,8 +71,8 @@ buildGraphChain = do
|
|||||||
link ndo nei
|
link ndo nei
|
||||||
pure (nai, neo)
|
pure (nai, neo)
|
||||||
|
|
||||||
testGraphX :: IO ()
|
buildGraphX :: IO (NodeInput String, NodeInput String, NodeOutput String, NodeOutput String)
|
||||||
testGraphX = do
|
buildGraphX = do
|
||||||
(nai, nao) <- newNode "a"
|
(nai, nao) <- newNode "a"
|
||||||
(nbi, nbo) <- newNode "b"
|
(nbi, nbo) <- newNode "b"
|
||||||
(nci, nco) <- newNode "c"
|
(nci, nco) <- newNode "c"
|
||||||
@ -78,50 +82,72 @@ testGraphX = do
|
|||||||
link nbo nci
|
link nbo nci
|
||||||
link nco ndi
|
link nco ndi
|
||||||
link nco nei
|
link nco nei
|
||||||
performGCUntilFinalizersQuiesce
|
pure (nai, nbi, ndo, neo)
|
||||||
print =<< findForward nai
|
|
||||||
print =<< findForward nbi
|
testGraphX :: IO ()
|
||||||
print =<< findBackward ndo
|
testGraphX = do
|
||||||
print =<< findBackward neo
|
do
|
||||||
touch (nai, nbi, ndo, neo)
|
(nai, nbi, ndo, neo) <- buildGraphX
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
print =<< findForward nai
|
["a", "c", "d", "e"] <- toList <$> findForwardVals nai
|
||||||
print =<< findForward nbi
|
["b", "c", "d", "e"] <- toList <$> findForwardVals nbi
|
||||||
touch ndo
|
["a", "b", "c", "d"] <- toList <$> findBackwardVals ndo
|
||||||
performGCUntilFinalizersQuiesce
|
["a", "b", "c", "e"] <- toList <$> findBackwardVals neo
|
||||||
print =<< findForward nai
|
touch (nai, nbi, ndo, neo)
|
||||||
print =<< findForward nbi
|
do
|
||||||
|
(nai, nbi, ndo, neo) <- buildGraphX
|
||||||
|
performGCUntilFinalizersQuiesce
|
||||||
|
["a", "c", "d"] <- toList <$> findForwardVals nai
|
||||||
|
["b", "c", "d"] <- toList <$> findForwardVals nbi
|
||||||
|
touch (nai, nbi, ndo)
|
||||||
|
do
|
||||||
|
(nai, nbi, ndo, neo) <- buildGraphX
|
||||||
|
performGCUntilFinalizersQuiesce
|
||||||
|
[] <- toList <$> findForwardVals nai
|
||||||
|
[] <- toList <$> findForwardVals nbi
|
||||||
|
touch (nai, nbi)
|
||||||
|
|
||||||
testGraphO :: IO ()
|
testGraphO :: IO ()
|
||||||
testGraphO = do
|
testGraphO = do
|
||||||
|
do
|
||||||
|
(nai, nao) <- buildGraphO
|
||||||
|
performGCUntilFinalizersQuiesce
|
||||||
|
["a", "b", "c"] <- toList <$> findBackward nao
|
||||||
|
["a", "b", "c"] <- toList <$> findForward nai
|
||||||
|
touch (nai, nao)
|
||||||
|
do
|
||||||
|
(_, nao) <- buildGraphO
|
||||||
|
performGCUntilFinalizersQuiesce
|
||||||
|
[] <- toList <$> findBackward nao
|
||||||
|
touch nao
|
||||||
|
do
|
||||||
|
(nai, _) <- buildGraphO
|
||||||
|
performGCUntilFinalizersQuiesce
|
||||||
|
[] <- toList <$> findForward nai
|
||||||
|
touch nai
|
||||||
|
|
||||||
|
buildGraphO :: IO (NodeInput String, NodeOutput String)
|
||||||
|
buildGraphO = do
|
||||||
(nai, nao) <- newNode "a"
|
(nai, nao) <- newNode "a"
|
||||||
(nbi, nbo) <- newNode "b"
|
(nbi, nbo) <- newNode "b"
|
||||||
(nci, nco) <- newNode "c"
|
(nci, nco) <- newNode "c"
|
||||||
link nao nbi
|
link nao nbi
|
||||||
link nbo nci
|
link nbo nci
|
||||||
link nco nai
|
link nco nai
|
||||||
performGCUntilFinalizersQuiesce
|
pure (nai, nao)
|
||||||
[] <- toList <$> findForward nai
|
|
||||||
pure ()
|
|
||||||
|
|
||||||
data NodeContents a = NodeContents
|
|
||||||
{ _nodeContents_key :: !Int
|
|
||||||
, _nodeContents_value :: a
|
|
||||||
, _nodeContents_forwardLinks :: !(IORef (IntMap (NodeInput a)))
|
|
||||||
, _nodeContents_backLinks :: !(IORef (IntMap (NodeOutput a)))
|
|
||||||
}
|
|
||||||
|
|
||||||
data NodeInput a = NodeInput
|
data NodeInput a = NodeInput
|
||||||
{ _nodeInput_key :: !Int
|
{ _nodeInput_key :: !Int
|
||||||
, _nodeInput_contents :: !(DualWeak (NodeContents a))
|
, _nodeInput_contents :: !(Weak (IORef (Maybe a)))
|
||||||
, _nodeInput_ticket :: !Ticket
|
, _nodeInput_forwardLinks :: !(IORef (IntMap (NodeInput a)))
|
||||||
|
, _nodeInput_backLinks :: !(Weak (IORef (IntMap (NodeOutput a))))
|
||||||
}
|
}
|
||||||
|
|
||||||
data NodeOutput a = NodeOutput
|
data NodeOutput a = NodeOutput
|
||||||
{ _nodeOutput_key :: !Int
|
{ _nodeOutput_key :: !Int
|
||||||
, _nodeOutput_contents :: !(DualWeak (NodeContents a))
|
, _nodeOutput_contents :: !(Weak (IORef (Maybe a)))
|
||||||
-- , _nodeOutput_backLinks :: !(IORef (IntMap (NodeOutput a)))
|
, _nodeOutput_forwardLinks :: !(Weak (IORef (IntMap (NodeInput a))))
|
||||||
, _nodeOutput_ticket :: !Ticket
|
, _nodeOutput_backLinks :: !(IORef (IntMap (NodeOutput a)))
|
||||||
}
|
}
|
||||||
|
|
||||||
{-# NOINLINE globalNodeIdRef #-}
|
{-# NOINLINE globalNodeIdRef #-}
|
||||||
@ -136,35 +162,50 @@ newNode v = do
|
|||||||
nodeId <- newNodeId
|
nodeId <- newNodeId
|
||||||
backLinks <- newIORef mempty
|
backLinks <- newIORef mempty
|
||||||
forwardLinks <- newIORef mempty
|
forwardLinks <- newIORef mempty
|
||||||
(w, tBackward, tForward) <- newDualWeak $ NodeContents
|
contentsRef <- newIORef $ Just v
|
||||||
{ _nodeContents_key = nodeId
|
w <- mkWeakWithIORefKey backLinks contentsRef
|
||||||
, _nodeContents_value = v
|
wBack <- mkWeakWithIORefKey backLinks backLinks
|
||||||
, _nodeContents_backLinks = backLinks
|
wForward <- mkWeakWithIORefKeyWithFinalizer forwardLinks forwardLinks $ do
|
||||||
, _nodeContents_forwardLinks = forwardLinks
|
writeIORef finalizerDidRunRef True
|
||||||
}
|
deRefWeak w >>= \case
|
||||||
|
Nothing -> pure ()
|
||||||
|
Just contentsRef' -> do
|
||||||
|
writeIORef contentsRef' Nothing
|
||||||
|
deRefWeak wBack >>= \case
|
||||||
|
Nothing -> pure ()
|
||||||
|
Just backLinks' -> do
|
||||||
|
writeIORef backLinks' mempty
|
||||||
|
--TODO: Clear out all the nodes forward of us
|
||||||
pure
|
pure
|
||||||
( NodeInput
|
( NodeInput
|
||||||
{ _nodeInput_key = nodeId
|
{ _nodeInput_key = nodeId
|
||||||
, _nodeInput_contents = w
|
, _nodeInput_contents = w
|
||||||
, _nodeInput_ticket = tForward
|
, _nodeInput_backLinks = wBack
|
||||||
|
, _nodeInput_forwardLinks = forwardLinks
|
||||||
}
|
}
|
||||||
, NodeOutput
|
, NodeOutput
|
||||||
{ _nodeOutput_key = nodeId
|
{ _nodeOutput_key = nodeId
|
||||||
, _nodeOutput_contents = w
|
, _nodeOutput_contents = w
|
||||||
-- , _nodeOutput_backLinks = backLinks
|
, _nodeOutput_backLinks = backLinks
|
||||||
, _nodeOutput_ticket = tBackward
|
, _nodeOutput_forwardLinks = wForward
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
getNodeContents :: Weak (IORef (Maybe a)) -> IO (Maybe a)
|
||||||
|
getNodeContents w = do
|
||||||
|
deRefWeak w >>= \case
|
||||||
|
Nothing -> pure Nothing
|
||||||
|
Just r -> readIORef r
|
||||||
|
|
||||||
link :: NodeOutput a -> NodeInput a -> IO ()
|
link :: NodeOutput a -> NodeInput a -> IO ()
|
||||||
link aOut bIn = do
|
link aOut bIn = do
|
||||||
getDualWeak (_nodeOutput_contents aOut) >>= \case
|
deRefWeak (_nodeInput_backLinks bIn) >>= \case
|
||||||
Nothing -> pure () -- Linking to a dead node does nothing
|
Nothing -> pure ()
|
||||||
Just a -> getDualWeak (_nodeInput_contents bIn) >>= \case
|
Just backLinks -> deRefWeak (_nodeOutput_forwardLinks aOut) >>= \case
|
||||||
Nothing -> pure () -- Linking to a dead node does nothing
|
Nothing -> pure ()
|
||||||
Just b -> do
|
Just forwardLinks -> do
|
||||||
atomicModifyIORef' (_nodeContents_forwardLinks a) $ \m -> (IntMap.insert (_nodeContents_key b) bIn m, ())
|
atomicModifyIORef' forwardLinks $ \m -> (IntMap.insert (_nodeInput_key bIn) bIn m, ())
|
||||||
atomicModifyIORef' (_nodeContents_backLinks b) $ \m -> (IntMap.insert (_nodeContents_key a) aOut m, ())
|
atomicModifyIORef' backLinks $ \m -> (IntMap.insert (_nodeOutput_key aOut) aOut m, ())
|
||||||
|
|
||||||
findForwardVals :: Ord a => NodeInput a -> IO (Set a)
|
findForwardVals :: Ord a => NodeInput a -> IO (Set a)
|
||||||
findForwardVals i = Set.fromList . IntMap.elems <$> findForward i
|
findForwardVals i = Set.fromList . IntMap.elems <$> findForward i
|
||||||
@ -176,11 +217,14 @@ findForward i = go mempty (IntMap.singleton (_nodeInput_key i) i)
|
|||||||
go found toSearch = case IntMap.minViewWithKey toSearch of
|
go found toSearch = case IntMap.minViewWithKey toSearch of
|
||||||
Nothing -> pure found
|
Nothing -> pure found
|
||||||
Just ((k, thisIn), restToSearch) -> do
|
Just ((k, thisIn), restToSearch) -> do
|
||||||
getDualWeak (_nodeInput_contents thisIn :: DualWeak (NodeContents a)) >>= \case
|
getNodeContents (_nodeInput_contents thisIn) >>= \case
|
||||||
Nothing -> go found restToSearch
|
Nothing -> go found restToSearch
|
||||||
Just this -> do
|
Just this -> do
|
||||||
newLinks <- readIORef (_nodeContents_forwardLinks this)
|
newLinks <- readIORef (_nodeInput_forwardLinks thisIn)
|
||||||
go (IntMap.insert k (_nodeContents_value this) found) (IntMap.union restToSearch (newLinks `IntMap.difference` found))
|
go (IntMap.insert k this found) (IntMap.union restToSearch (newLinks `IntMap.difference` found))
|
||||||
|
|
||||||
|
findBackwardVals :: Ord a => NodeOutput a -> IO (Set a)
|
||||||
|
findBackwardVals i = Set.fromList . IntMap.elems <$> findBackward i
|
||||||
|
|
||||||
findBackward :: forall a. NodeOutput a -> IO (IntMap a)
|
findBackward :: forall a. NodeOutput a -> IO (IntMap a)
|
||||||
findBackward o = go mempty (IntMap.singleton (_nodeOutput_key o) o)
|
findBackward o = go mempty (IntMap.singleton (_nodeOutput_key o) o)
|
||||||
@ -188,17 +232,16 @@ findBackward o = go mempty (IntMap.singleton (_nodeOutput_key o) o)
|
|||||||
go :: IntMap a -> IntMap (NodeOutput a) -> IO (IntMap a)
|
go :: IntMap a -> IntMap (NodeOutput a) -> IO (IntMap a)
|
||||||
go found toSearch = case IntMap.minViewWithKey toSearch of
|
go found toSearch = case IntMap.minViewWithKey toSearch of
|
||||||
Nothing -> pure found
|
Nothing -> pure found
|
||||||
Just ((k, thisIn), restToSearch) -> do
|
Just ((k, thisOut), restToSearch) -> do
|
||||||
getDualWeak (_nodeOutput_contents thisIn :: DualWeak (NodeContents a)) >>= \case
|
getNodeContents (_nodeOutput_contents thisOut) >>= \case
|
||||||
Nothing -> go found restToSearch
|
Nothing -> go found restToSearch
|
||||||
Just this -> do
|
Just this -> do
|
||||||
newLinks <- readIORef (_nodeContents_backLinks this)
|
newLinks <- readIORef (_nodeOutput_backLinks thisOut)
|
||||||
go (IntMap.insert k (_nodeContents_value this) found) (IntMap.union restToSearch (newLinks `IntMap.difference` found))
|
go (IntMap.insert k this found) (IntMap.union restToSearch (newLinks `IntMap.difference` found))
|
||||||
|
|
||||||
testDualWeak :: IO ()
|
testDualWeak :: IO ()
|
||||||
testDualWeak = do
|
testDualWeak = do
|
||||||
do
|
do
|
||||||
putStrLn "A"
|
|
||||||
target <- newIORef ()
|
target <- newIORef ()
|
||||||
r <- mkWeakWithIORefKey target ()
|
r <- mkWeakWithIORefKey target ()
|
||||||
(w, a, b) <- newDualWeak target
|
(w, a, b) <- newDualWeak target
|
||||||
@ -209,7 +252,6 @@ testDualWeak = do
|
|||||||
touch (a, b)
|
touch (a, b)
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
do
|
do
|
||||||
putStrLn "B"
|
|
||||||
target <- newIORef ()
|
target <- newIORef ()
|
||||||
r <- mkWeakWithIORefKey target ()
|
r <- mkWeakWithIORefKey target ()
|
||||||
(w, a, _) <- newDualWeak target
|
(w, a, _) <- newDualWeak target
|
||||||
@ -220,7 +262,6 @@ testDualWeak = do
|
|||||||
touch a
|
touch a
|
||||||
performGCUntilFinalizersQuiesce
|
performGCUntilFinalizersQuiesce
|
||||||
do
|
do
|
||||||
putStrLn "C"
|
|
||||||
target <- newIORef ()
|
target <- newIORef ()
|
||||||
r <- mkWeakWithIORefKey target ()
|
r <- mkWeakWithIORefKey target ()
|
||||||
(w, _, b) <- newDualWeak target
|
(w, _, b) <- newDualWeak target
|
||||||
|
Loading…
Reference in New Issue
Block a user