{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE RecordWildCards #-}
{-# OPTIONS_GHC -fno-warn-orphans #-}
{-# OPTIONS_HADDOCK hide #-}
module Reanimate.Math.SSSP
(
SSSP
, sssp
, ssspFinger
, dual
, Dual(..)
, DualTree(..)
, dualToTriangulation
, visibilityArray
, naive
, naive2
, drawDual
) where
import Control.Monad
import Control.Monad.ST
import qualified Data.FingerTree as F
import Data.Foldable
import Data.List
import qualified Data.Map as Map
import Data.Maybe
import Data.STRef
import Data.Tree
import qualified Data.Vector as V
import qualified Data.Vector.Mutable as MV
import Reanimate.Math.Common
import Reanimate.Math.Triangulate
type SSSP = V.Vector Int
visibilityArray :: Ring Rational -> V.Vector [Int]
visibilityArray p = arr
where
n = ringSize p
arr = V.fromList
[ visibility y
| y <- [0..n-1]
]
visibility y =
[ i
| i <- [0..y-1]
, y `elem` arr V.! i ] ++
[ i
| i <- [y+1 .. n-1]
, let pI = ringAccess p i
isOpen = isRightTurn pYp pY pYn
, ringClamp p (y+1) == i || ringClamp p (y-1) == i || if isOpen
then isLeftTurnOrLinear pY pYn pI ||
isLeftTurnOrLinear pYp pY pI
else not $ isRightTurn pY pYn pI ||
isRightTurn pYp pY pI
, let myEdges = [(e1,e2) | (e1,e2) <- edges, e1/=y, e1/=i, e2/=y,e2/=i]
, all (isNothing . lineIntersect (pY,pI))
[ (ringAccess p e1, ringAccess p e2) | (e1,e2) <- myEdges ]]
where
pY = ringAccess p y
pYn = ringAccess p $ y+1
pYp = ringAccess p $ y-1
edges = zip [0..n-1] (tail [0..n-1] ++ [0])
naive :: Ring Rational -> SSSP
naive p =
V.fromList $ Map.elems $
Map.map snd $
worker initial
where
initial = Map.singleton 0 (0,0)
visibility = visibilityArray p
worker :: Map.Map Int (Rational, Int) -> Map.Map Int (Rational, Int)
worker m
| m==newM = newM
| otherwise = worker newM
where
ms' = [ Map.fromList
[ case Map.lookup v m of
Nothing -> (v, (distThroughI, i))
Just (otherDist,parent)
| otherDist > distThroughI -> (v, (distThroughI, i))
| otherwise -> (v, (otherDist, parent))
| v <- visibility V.! i
, let distThroughI = dist + approxDist (ringAccess p i) (ringAccess p v) ]
| (i,(dist,_)) <- Map.toList m
]
newM = Map.unionsWith g (m:ms') :: Map.Map Int (Rational,Int)
g a b = if fst a < fst b then a else b
naive2 :: Ring Rational -> SSSP
naive2 p = runST $ do
parents <- MV.replicate (ringSize p) (-1)
costs <- MV.replicate (ringSize p) (-1)
MV.write parents 0 0
MV.write costs 0 0
changedRef <- newSTRef False
let loop i
| i == ringSize p = do
changed <- readSTRef changedRef
when changed $ do
writeSTRef changedRef False
loop 0
| otherwise = do
myCost <- MV.read costs i
unless (myCost < 0) $
forM_ (visibility V.! i) $ \n -> do
theirCost <- MV.read costs n
let throughCost = myCost + approxDist (ringAccess p i) (ringAccess p n)
when (throughCost < theirCost || theirCost < 0) $ do
MV.write parents n i
MV.write costs n throughCost
writeSTRef changedRef True
loop (i+1)
loop 0
V.unsafeFreeze parents
where
visibility = visibilityArray p
data Dual = Dual (Int,Int,Int)
DualTree
DualTree
deriving (Show)
data DualTree
= EmptyDual
| NodeDual Int
DualTree
DualTree
deriving (Show)
drawDual :: Dual -> String
drawDual d = drawTree $
case d of
Dual (a,b,c) l r -> Node (show (a,b,c)) [worker c a l, worker b c r]
where
worker _a _b EmptyDual = Node "Leaf" []
worker a b (NodeDual x l r) =
Node (show (b,a,x)) [worker x b l, worker a x r]
dualToTriangulation :: Ring Rational -> Dual -> Triangulation
dualToTriangulation p d = edgesToTriangulation (ringSize p) $ filter goodEdge $
case d of
Dual (a,b,c) l r ->
(a,b):(a,c):(b,c):worker c a l ++ worker b c r
where
goodEdge (a,b)
= a /= ringClamp p (b+1) && a /= ringClamp p (b-1)
worker _a _b EmptyDual = []
worker a b (NodeDual x l r) =
(a,x) : (x, b) : worker x b l ++ worker a x r
dual :: Int -> Triangulation -> Dual
dual root t =
case hasTriangle of
[] -> error "weird triangulation"
(x:_) -> Dual (root,rootNext,x) (dualTree t (x,root) rootNext) (dualTree t (rootNext,x) root)
where
rootNext = idx (root+1)
rootPrev = idx (root-1)
rootNNext = idx (root+2)
idx i = i `mod` n
hasTriangle = (rootPrev : t V.! root) `intersect` (rootNNext : t V.! rootNext)
n = V.length t
dualTree :: Triangulation -> (Int,Int) -> Int -> DualTree
dualTree t (a,b) e =
case hasTriangle of
[] -> EmptyDual
[ab] ->
NodeDual ab
(dualTree t (ab,b) a)
(dualTree t (a,ab) b)
_ -> error $ "Invalid triangulation: " ++ show (a,b,e,hasTriangle)
where
hasTriangle = (prev a : next a : t V.! a) `intersect` (prev b : next b : t V.! b)
\\ [e]
n = V.length t
next x = (x+1) `mod` n
prev x = (x-1) `mod` n
sssp :: (Fractional a, Ord a, Epsilon a) => Ring a -> Dual -> SSSP
sssp p d = toSSSP $
case d of
Dual (a,b,c) l r ->
(a, a) :
(b, a) :
(c, a) :
worker [c] [b] a r ++
loopLeft a c l
where
toSSSP =
V.fromList . map snd . sortOn fst
loopLeft a outer l =
case l of
EmptyDual -> []
NodeDual x l' r' ->
(x,a) :
worker [x] [outer] a r' ++
loopLeft a x l'
searchFn _checkStep _cusp _x [] = Nothing
searchFn checkStep cusp x (y:ys)
| not (checkStep (ringAccess p cusp) (ringAccess p y) (ringAccess p x))
= Just $ helper [] y ys
| otherwise = Nothing
where
helper acc v [] = (v, [], reverse acc)
helper acc v1 (v2:vs)
| checkStep (ringAccess p v1) (ringAccess p v2) (ringAccess p x) =
(v1, v2:vs, reverse acc)
| otherwise = helper (v1:acc) v2 vs
searchRight = searchFn isLeftTurn
searchLeft = searchFn isRightTurn
worker _ _ _ EmptyDual = []
worker f1 f2 cusp (NodeDual x l r) =
case searchLeft cusp x (toList f1) of
Just (v, f1Hi, f1Lo) ->
(x, v::Int) :
worker f1Hi [x] v l ++
worker (f1Lo ++ [v, x]) f2 cusp r
Nothing ->
case searchRight cusp x (toList f2) of
Just (v, f2Hi, f2Lo) ->
(x, v::Int) :
worker f1 (f2Lo ++ [v, x]) cusp l ++
worker [x] f2Hi v r
Nothing ->
(x, cusp::Int) :
worker f1 [x] cusp l ++
worker [x] f2 cusp r
data MinMax = MinMax Int Int | MinMaxEmpty deriving (Show)
instance Semigroup MinMax where
MinMaxEmpty <> b = b
a <> MinMaxEmpty = a
MinMax a _b <> MinMax _c d
= MinMax a d
instance Monoid MinMax where
mempty = MinMaxEmpty
type Chain = F.FingerTree MinMax Int
data Funnel = Funnel
{ funnelLeft :: Chain
, funnelCusp :: Int
, funnelRight :: Chain
}
instance F.Measured MinMax Int where
measure i = MinMax i i
splitFunnel :: (Epsilon a, Fractional a, Ord a) => Ring a -> Int -> Funnel -> (Int, Funnel, Funnel)
splitFunnel p x Funnel{..}
| isOnLeftChain =
case doSearch isRightTurn funnelLeft of
(lower, t, upper) ->
( t
, Funnel upper t (F.singleton x)
, Funnel (lower F.|> t F.|> x) funnelCusp funnelRight)
| isOnRightChain =
case doSearch isLeftTurn funnelRight of
(lower, t, upper) ->
( t
, Funnel funnelLeft funnelCusp (lower F.|> t F.|> x)
, Funnel (F.singleton x) t upper)
| otherwise =
( funnelCusp
, Funnel funnelLeft funnelCusp (F.singleton x)
, Funnel (F.singleton x) funnelCusp funnelRight)
where
isOnLeftChain = fromMaybe False $
isLeftTurnOrLinear cuspElt <$> leftElt <*> pure targetElt
isOnRightChain = fromMaybe False $
isRightTurnOrLinear cuspElt <$> rightElt <*> pure targetElt
doSearch fn chain =
case F.search (searchChain fn) (chain::Chain) of
F.Position lower t upper -> (lower, t, upper)
F.OnLeft -> error "cannot happen"
F.OnRight -> error "cannot happen"
F.Nowhere -> error "cannot happen"
searchChain _ MinMaxEmpty _ = False
searchChain _ _ MinMaxEmpty = True
searchChain check (MinMax _ l) (MinMax r _) =
check (ringAccess p l) (ringAccess p r) targetElt
cuspElt = ringAccess p funnelCusp
targetElt = ringAccess p x
leftElt = ringAccess p <$> chainLeft funnelLeft
rightElt = ringAccess p <$> chainLeft funnelRight
chainLeft chain =
case F.viewl chain of
F.EmptyL -> Nothing
elt F.:< _ -> Just elt
ssspFinger :: (Epsilon a, Fractional a, Ord a) => Ring a -> Dual -> SSSP
ssspFinger p d = toSSSP $
case d of
Dual (a,b,c) l r ->
(a, a) :
(b, a) :
(c, a) :
worker (Funnel (F.singleton c) a (F.singleton b)) r ++
loopLeft a c l
where
toSSSP =
V.fromList . map snd . sortOn fst
loopLeft a outer l =
case l of
EmptyDual -> []
NodeDual x l' r' ->
(x,a) :
worker (Funnel (F.singleton x) a (F.singleton outer)) r' ++
loopLeft a x l'
worker _ EmptyDual = []
worker f (NodeDual x l r) =
case splitFunnel p x f of
(v, fL, fR) ->
(x, v) :
worker fL l ++
worker fR r