\begin{code}
module SequentialComputation (computeSequential,Vertex,Edge,Table) where
import SequentialTypes
import InterfacesRules
import CodeSyntax
import GrammarInfo
import Control.Monad(when,unless)
import Control.Monad.ST(ST, runST)
import Data.Array(Array,(!),bounds)
import Data.Array.ST(STArray, newArray, readArray, writeArray, freeze)
import Data.Maybe(isJust,fromJust)
import Data.List(partition,(\\))
import qualified Data.Set as Set
import qualified Data.Map as Map
\end{code}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Collecting information}
In the Data.Graph library,
a graph is represented as |Array Vertex [Vertex]|,
mapping each vertex to a list of adjacent vertices.
A |Vertex| is simply encoded by an |Int|.
So to test whether an edge |(x,y)| belongs to |g|
we can evaluate |y `elem` g!x|
For more efficiency, we use Maps instead of lists.
Sets would also have done, but we also want to each edge to have a path as a witness.
Moreover, as we will mostly be adding edges to the graph,
we use a mutable array.
If we want to use any of the library
functions, we can convert our representation by |fmap Map.keys . freeze|.
\begin{code}
type Graph = Array Vertex [Vertex]
type MGraph = Array Vertex (Map.Map Vertex Path)
type MMGraph s = STArray s Vertex (Map.Map Vertex Path)
singleStep :: (Vertex->Vertex->PathStep) -> Edge -> EdgePath
singleStep f e@(s,t) = (e, [f s t])
\end{code}
We can add an edge to a graph, or remove it. These functions return
whether they did something (resp. addition or removal) or not. hasEdge
only checks whether a graph contains an edge or not.
\begin{code}
addEdge :: MMGraph s -> EdgePath -> ST s Bool
addEdge graph ((s,t),p)
= do m <- readArray graph s
let b = not (Map.member t m)
when b (writeArray graph s (Map.insert t p m))
return b
hasEdge :: MMGraph s -> EdgePath -> ST s Bool
hasEdge graph ((s,t),_)
= do m <- readArray graph s
return (Map.member t m)
\end{code}
The first step is to assign a number to all attributes, and a
different one to all attribute occurrences. We create an array mapping
the numbers to the information about the attribute occurrences
(|ruleTable|), so we can look up this information in $O(1)$ time. We
also build mappings from attributes to their occurrences (|tdsToTdp|)
and vice versa (|tdpToTds|). |LMH| indicates the division of the
attributes - an element |(l,m,h) `elem` LMH| means that vertices |i, l
<= i <= h| are attributes of the same nonterminal, with vertices |j, l
<= j < m| being inherited and |k, m <= k <= h| being synthesized
attributes.
See the |SequentialTypes.Info| and |SequentialTypes.LMH|
Then we collect the direct dependencies, using the integer
representations. This list of tuples (edges in the dependency graph)
all information that is collected is passed to a function that will
compute the interfaces and visit sub-sequences. We cannot do this
computation in AG, because mutable arrays require the ST monad, which
cannot be used inside AG.
Now we can build a graph for attributes, and a graph for ao's, and add
the direct dependencies to the ao graph. Like Pennings we will call
the attribte graph Tds (transitive dependencies of symbols), and the
ao-graph Tdp (transitive dependencies of productions). Unlike him, we
will have only one Tds and one Tdp graph. In |STGraph|, we can lookup
outgoing edges in |O(1)| time, but looking up incoming edges will take
|O(e)| time, where |e| is the number of edges in the graph. As we will
be doing this quite often it is worthwhile to keep both Tdp and its
transposed version. The computation will involve both Tds and Tdp. It
treats specially. TODO elaborate on that.
\begin{code}
type Tdp s = (MMGraph s, MMGraph s)
type Tds s = MMGraph s
type Comp s = (Tds s, Tdp s)
\end{code}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Generating IDS}
As we insert edges into Tdp we keep it transitively closed, so every
time we add the edge $(s,t)$ to V, we also add the edges
$\{ (r,t) || (r,s) \in V \}$ and
$\{ (s,u) || (t,u) \in V \}$.
\begin{code}
insertTdp :: Info -> Comp s -> EdgePath -> ST s ()
insertTdp info comp@(_,(tdpN,tdpT)) e@((s,t),ee)
= do b <- hasEdge tdpN e
unless b
(do rs <- readArray tdpT s
us <- readArray tdpN t
let edges = e :[ ((r,t),er++ee ) | (r,er) <- Map.toList rs ]
++ [ ((s,u), ee++eu) | (u,eu) <- Map.toList us ]
++ [ ((r,u),er++ee++eu) | (r,er) <- Map.toList rs, (u,eu) <- Map.toList us ]
mapM_ (addTdpEdge info comp) edges
)
\end{code}
Edges in |Tdp| can induce edges in |Tds|, so whenever we add
an edge, we also add the induced edge if necessary
\begin{code}
addTdpEdge :: Info -> Comp s -> EdgePath -> ST s ()
addTdpEdge info comp@(_,(tdpN,tdpT)) e@((s,t),ee)
= do b <- addEdge tdpN e
when b
(do addEdge tdpT ((t,s),ee)
let u = tdpToTds info ! s
v = tdpToTds info ! t
nonlocal = u /= 1 && v /= 1
equalfield = isEqualField (ruleTable info ! s) (ruleTable info ! t)
when (nonlocal && equalfield)
(insertTds info comp ((u,v),ee))
)
\end{code}
Inserting edges into |Tds| will insert edges between the occurrences
of the attributes into |Tdp|.
\begin{code}
insertTds :: Info -> Comp s -> EdgePath -> ST s ()
insertTds info comp@(tds,_) e@((u,v),ee)
= do b <- addEdge tds e
when b
(mapM_ (insertTdp info comp) [ ( (s,t), [AttrStep u v] )
| s <- tdsToTdp info ! u
, not (getIsIn (ruleTable info ! s))
, t <- tdsToTdp info ! v
, getIsIn (ruleTable info ! t)
, isEqualField (ruleTable info ! s) (ruleTable info ! t)
]
)
\end{code}
If we add the direct dependencies to the Tdp graph in the way above, the
Tds graph is filled with IDS.
Below is a way to only build up the Tdp graph, without reflect the changes in the Tds graph.
\begin{code}
simpleInsert :: Tdp s -> EdgePath -> ST s ()
simpleInsert tdp@(tdpN,tdpT) e@((s,t),ee)
= do b <- hasEdge tdpT ((t,s),undefined)
unless b (do rs <- readArray tdpT s
us <- readArray tdpN t
let edges = e :[ ((r,t),er++ee ) | (r,er) <- Map.toList rs ]
++ [ ((s,u), ee++eu) | (u,eu) <- Map.toList us ]
++ [ ((r,u),er++ee++eu) | (r,er) <- Map.toList rs, (u,eu) <- Map.toList us ]
mapM_ (addSimpleEdge tdp) edges
)
addSimpleEdge :: Tdp s -> EdgePath -> ST s ()
addSimpleEdge (tdpN,tdpT) e@((s,t),ee)
= do b <- addEdge tdpN e
when b (do addEdge tdpT ((t,s),ee)
return ()
)
\end{code}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Interfaces}
In absence of cycles we can find the interfaces. We only take
attributes that are used.
When an attribute has no incoming edges it can be computed. As the
emphasis is on incoming edges, we will work with the transposed Tds
graph. The funtion |used| indicates which vertices are included in the
interfaces.
See modules Interfaces and InterfacesRules for more information.
%format sem_IRoot_IRoot = "sem_{IRoot}"
%format sem_Interface_Interface = "sem_{Interface}"
%format sem_Interfaces_Cons = ":_{Interfaces}"
%format sem_Interfaces_Nil = "[]_{Interfaces}"
%format sem_Segments_Cons = ":_{Segments}"
%format sem_Segments_Nil = "[]_{Segments}"
\begin{code}
makeInterfaces :: Info -> Graph -> T_IRoot
makeInterfaces info tds
= let interslist = reverse . makeInterface tds []
mkSegments = foldr (sem_Segments_Cons . uncurry sem_Segment_Segment) sem_Segments_Nil . interslist
mkInter ((nt,cons),lmh) = sem_Interface_Interface nt cons (mkSegments lmh)
inters = foldr (sem_Interfaces_Cons . mkInter) sem_Interfaces_Nil (zip (nonts info) (lmh info))
in sem_IRoot_IRoot inters
\end{code}
The sinks of a graph are those vertices that have no outgoing
edges. We define a function that determines whether a vertex is a sink
if a set |del| of vertices had been removed from the graph. This means
that the attribute can be computed if all attributes in |del| have
been computed.
\begin{code}
isSink :: Graph -> [Vertex] -> Vertex -> Bool
isSink graph del v = null (graph ! v \\ del)
\end{code}
Now we can make interfaces by taking inherited sinks and synthesized
sinks alternatively. If there are no synthesized attributes at all,
generate an interface with one visit computing nothing.
\begin{code}
makeInterface :: Graph -> [Vertex] -> LMH -> [([Vertex],[Vertex])]
makeInterface tds del (l,m,h)
| m > h = [([],[])]
| otherwise = let syn = filter (isSink tds del) ([m..h] \\ del)
del' = del ++ syn
inh = filter (isSink tds del') ([l..(m1)] \\ del')
del'' = del' ++ inh
rest = makeInterface tds del'' (l,m,h)
in if null inh && null syn
then []
else (inh,syn) : rest
\end{code}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Detecting of cycles}
We only want to return s2i edges.
\begin{code}
findCycles :: Info -> MGraph -> [EdgePaths]
findCycles info tds
= [ ((u,v),p1,p2)
| (l,m,h) <- lmh info
, v <- [m..h]
, (u,p1) <- Map.toList (tds ! v)
, l <= u, u < m
, let mbp2 = Map.lookup v (tds ! u)
, isJust mbp2
, let p2 = fromJust mbp2
]
findLocCycles :: MGraph -> [EdgePath]
findLocCycles tdp
= let (low, high) = bounds tdp
in [ ((u,u),p)
| u <- [low..high]
, (v,p) <- Map.toList (tdp ! u)
, v==u
]
findInstCycles :: [Edge] -> MGraph -> [EdgePath]
findInstCycles instToSynEdges tdp
= [ ((i,s), fromJust mbp)
| (i, s) <- instToSynEdges
, let mbp = Map.lookup i (tdp ! s)
, isJust mbp
]
\end{code}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Tying it together}
\begin{code}
generateVisits :: Info -> MGraph -> MGraph -> [Edge] -> (CInterfaceMap, CVisitsMap, [Edge])
generateVisits info tds tdp dpr
= let inters = makeInterfaces info (fmap Map.keys tds)
inhs = Inh_IRoot{ info_Inh_IRoot = info
, tdp_Inh_IRoot = fmap Map.keys tdp
, dpr_Inh_IRoot = dpr
}
iroot = wrap_IRoot inters inhs
in (inters_Syn_IRoot iroot, visits_Syn_IRoot iroot, edp_Syn_IRoot iroot)
reportLocalCycle :: MGraph -> [EdgePath] -> [[Vertex]]
reportLocalCycle tds cyc
= fst (foldr f ([],Set.empty) (map (edgePathToEdgeRoute tds) cyc))
where f ((x,_),p) res@(paths,syms) | Set.member x syms = res
| otherwise = (p:paths, Set.union syms (Set.fromList p))
reportCycle :: Info -> MGraph -> [EdgePaths] -> [EdgeRoutes]
reportCycle info tds cyc
= fst (foldr f ([],Set.empty) (map (edgePathsToEdgeRoutes tds) cyc))
where f epp@((x,y),p1,p2) res@(paths,syms) | Set.member x syms &&
Set.member y syms = res
| otherwise = (epp:paths, Set.union syms (Set.fromList (map tdp2tds (p1++p2))))
tdp2tds (2) = 2
tdp2tds v = tdpToTds info ! v
edgePathsToEdgeRoutes :: MGraph -> EdgePaths -> EdgeRoutes
edgePathsToEdgeRoutes tds (e,p1,p2) = ( e, pathToRoute tds p1, pathToRoute tds p2 )
edgePathToEdgeRoute :: MGraph -> EdgePath -> EdgeRoute
edgePathToEdgeRoute tds (e,p) = ( e, pathToRoute tds p )
pathToRoute :: MGraph -> Path -> Route
pathToRoute tds p = convertPath (expandAll p)
where expandAll :: Path -> Path
expandAll p | hasAttrStep p = expandAll (expandOne p)
| otherwise = p
expandOne :: Path -> Path
expandOne p = shortcut (concatMap expandStep p)
expandStep :: PathStep -> Path
expandStep (AttrStep u v) = fromJust (Map.lookup v (tds!u))
expandStep x = [x]
convertPath :: Path -> Route
convertPath p = concatMap convertStep p
convertStep :: PathStep -> Route
convertStep (AtOcStep s t) = [s,t]
convertStep (AttrIndu s t) = [2,2]
hasAttrStep :: Path -> Bool
hasAttrStep [] = False
hasAttrStep (AttrStep _ _ : _ ) = True
hasAttrStep (_ : xs) = hasAttrStep xs
shortcut :: Eq a => [a] -> [a]
shortcut [] = []
shortcut (x:xs) = x : shortcut (removeBefore x xs)
removeBefore :: Eq a => a -> [a] -> [a]
removeBefore x ys = reverse (takeWhile (/=x) (reverse ys))
isLocLoc :: Table CRule -> EdgePath -> Bool
isLocLoc rt ((s,t),_) = isLocal (rt ! s) && isLocal (rt ! t)
computeSequential :: Info -> [Edge] -> [Edge] -> CycleStatus
computeSequential info dpr instToSynEdges
= runST
(do let bigBounds = bounds (tdpToTds info)
smallBounds = bounds (tdsToTdp info)
(ll,es) = partition (isLocLoc (ruleTable info)) (map (singleStep AtOcStep) (dpr ++ instToSynEdges))
tds <- newArray smallBounds Map.empty
tdpN <- newArray bigBounds Map.empty
tdpT <- newArray bigBounds Map.empty
let tdp = (tdpN,tdpT)
comp = (tds,tdp)
mapM_ (simpleInsert tdp) ll
tdp1 <- freeze tdpN
let cyc1 = findLocCycles tdp1
if not (null cyc1)
then do return (LocalCycle (reportLocalCycle undefined cyc1))
else do mapM_ (insertTdp info comp) es
tds2 <- freeze tds
let cyc2 = findCycles info tds2
if not (null cyc2)
then do return (DirectCycle (reportCycle info tds2 cyc2))
else do tdp2 <- freeze tdpN
let cyc4 = findInstCycles instToSynEdges tdp2
if not (null cyc4)
then do return (InstCycle (reportLocalCycle tds2 cyc4))
else do let (cim,cvm,edp) = generateVisits info tds2 tdp2 dpr
mapM_ (insertTds info comp) (map (singleStep AttrIndu) edp)
tds3 <- freeze tds
let cyc3 = findCycles info tds3
if not (null cyc3)
then return (InducedCycle cim (reportCycle info tds3 cyc3))
else do tdp3 <- freeze tdpN
let cyc5 = findInstCycles instToSynEdges tdp3
if not (null cyc5)
then do return (InstCycle (reportLocalCycle tds3 cyc5))
else do return (CycleFree cim cvm)
)
\end{code}
\end{document}