module Ideas.Common.Rewriting.Confluence
( isConfluent, checkConfluence, checkConfluenceWith
, somewhereM
, Config, defaultConfig, showTerm, complexity, termEquality
) where
import Data.Maybe
import Ideas.Common.Id
import Ideas.Common.Rewriting.RewriteRule
import Ideas.Common.Rewriting.Substitution
import Ideas.Common.Rewriting.Term
import Ideas.Common.Rewriting.Unification
import Ideas.Common.Traversal.Navigator
import Ideas.Common.Traversal.Utils
import Ideas.Utils.Uniplate hiding (rewriteM)
normalForm :: [RewriteRule a] -> Term -> Term
normalForm rs = run []
where
run hist a =
case [ b | r <- rs, b <- somewhereM (rewriteTerm r) a ] of
[] -> a
hd:_ -> if hd `elem` hist
then error "cyclic"
else run (a:hist) hd
rewriteTerm :: RewriteRule a -> Term -> [Term]
rewriteTerm r t = do
let lhs :~> rhs = ruleSpecTerm $
renumberRewriteRule (nextMetaVar t) r
sub <- match lhs t
return (sub |-> rhs)
somewhereM :: Uniplate a => (a -> [a]) -> a -> [a]
somewhereM f = map unfocus . rec . uniplateNav
where
rec ca = changeG f ca ++ concatMap rec (downs ca)
uniplateNav :: Uniplate a => a -> UniplateNavigator a
uniplateNav = focus
type Pair a = (RewriteRule a, Term)
type Triple a = (RewriteRule a, Term, Term)
superImpose :: RewriteRule a -> RewriteRule a -> [UniplateNavigator Term]
superImpose r1 r2 = rec (uniplateNav lhs1)
where
lhs1 :~> _ = ruleSpecTerm r1
lhs2 :~> _ = ruleSpecTerm (renumber r1 r2)
rec ca = case current ca of
TMeta _ -> []
a -> maybe [] (return . (`subTop` ca)) (unify a lhs2) ++
concatMap rec (downs ca)
subTop :: Substitution -> UniplateNavigator Term -> UniplateNavigator Term
subTop s ca = fromMaybe ca $
navigateTo (location ca) (change (s |->) (top ca))
renumber r = case metaInRewriteRule r of
[] -> id
xs -> renumberRewriteRule (maximum xs + 1)
criticalPairs :: [RewriteRule a] -> [(Term, Pair a, Pair a)]
criticalPairs rs =
[ (a, (r1, b1), (r2, b2))
| r1 <- rs
, r2 <- rs
, na <- superImpose r1 r2
, compareId r1 r2 == LT || not (null (fromLocation (location na)))
, let a = unfocus na
, b1 <- rewriteTerm r1 a
, b2 <- map unfocus (changeG (rewriteTerm r2) na)
]
noDiamondPairs :: Config -> [RewriteRule a] -> [(Term, Triple a, Triple a)]
noDiamondPairs cfg rs = noDiamondPairsWith (normalForm rs) cfg rs
noDiamondPairsWith :: (Term -> Term) -> Config -> [RewriteRule a] -> [(Term, Triple a, Triple a)]
noDiamondPairsWith f cfg rs =
[ (a, (r1, e1, nf1), (r2, e2, nf2))
| (a, (r1, e1), (r2, e2)) <- criticalPairs rs
, let (nf1, nf2) = (f e1, f e2)
, not (termEquality cfg nf1 nf2)
]
reportPairs :: Config -> [(Term, Triple a, Triple a)] -> IO ()
reportPairs cfg = putStrLn . unlines . zipWith report [1::Int ..]
where
f = showTerm cfg
report i (a, (r1, e1, nf1), (r2, e2, nf2)) = unlines
[ show i ++ ") " ++ f a
, " " ++ showId r1
, " " ++ f e1 ++ if e1==nf1 then "" else " --> " ++ f nf1
, " " ++ showId r2
, " " ++ f e2 ++ if e2==nf2 then "" else " --> " ++ f nf2
]
isConfluent :: [RewriteRule a] -> Bool
isConfluent = null . noDiamondPairs defaultConfig
checkConfluence :: [RewriteRule a] -> IO ()
checkConfluence = checkConfluenceWith defaultConfig
checkConfluenceWith :: Config -> [RewriteRule a] -> IO ()
checkConfluenceWith cfg = reportPairs cfg . noDiamondPairs cfg
data Config = Config
{ showTerm :: Term -> String
, complexity :: Term -> Int
, termEquality :: Term -> Term -> Bool
}
defaultConfig :: Config
defaultConfig = Config show (const 0) (==)