forked from jkff/minxmod
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBuchi.hs
56 lines (47 loc) · 1.88 KB
/
Buchi.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
module Buchi where
import Data.Graph.Inductive
import Data.Graph.Inductive.Query.DFS
import Data.Maybe
import Data.Ord
-- Non-deterministic Buchi automaton
data NBA e = NBA
{
nba_graph :: Gr (Bool, Bool) e -- (accept, init)
}
data SCC a = AcyclicSCC a | CyclicSCC [a] deriving (Show)
exampleNBA = NBA {
nba_graph = mkGraph
(zip [0..6] $ zip
[False,False,False,False,False,False,True]
[False,True,False,False,False,False,False])
(concatMap (\(i,os) -> [(i, o, ()) | o <- os])
[(0, [5]),
(1, [0,2,5]),
(2, [6]),
(3, [1,6]),
(4, [5]),
(5, [2,4]),
(6, [4])])
}
-- | Returns: (non-cyclic prefix (stem), cycle body (loop))
findAcceptingCycle :: NBA e -> Maybe ([Int], [Int])
findAcceptingCycle (NBA {nba_graph = g}) = case stem of [] -> Nothing; _ -> Just (stem, cycle)
where
initNodes = [j | (j, (_,True)) <- labNodes g]
acceptsInCyclicSCCs = [j | CyclicSCC cs <- findSCC g, j <- cs, isAcceptNode j]
isAcceptNode = fst . fromJust . lab g
stem = shortestPathBetween initNodes acceptsInCyclicSCCs g
cycle = shortestCycleThrough (last stem) g
shortestCycleThrough v g = head [p | p <- bft v g, v `elem` suc g (head p)]
shortestPathBetween froms tos g = init . tail $ esp fromAux toAux g'
where
[fromAux, toAux] = newNodes 2 g
g' = insEdges [(fromAux, i, undefined) | i <- froms] $
insEdges [(i, toAux, undefined) | i <- tos ] $
insNodes [(fromAux, undefined), (toAux, undefined)] $
g
findSCC :: Graph gr => gr a b -> [SCC Node]
findSCC g = map toSCC . scc $ g
where toSCC [n] | n `elem` suc g n = CyclicSCC [n]
| otherwise = AcyclicSCC n
toSCC ns = CyclicSCC ns