----------------------------------------------------------------------------- -- Copyright 2016, Ideas project team. This file is distributed under the -- terms of the Apache License 2.0. For more information, see the files -- "LICENSE.txt" and "NOTICE.txt", which are included in the distribution. ----------------------------------------------------------------------------- -- | -- Maintainer : bastiaan.heeren@ou.nl -- Stability : provisional -- Portability : portable (depends on ghc) -- ----------------------------------------------------------------------------- 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) -- uniplate-like helper-functions 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) (==) ---------------------------------------------------- -- Example {- r1, r2, r3, r4, r5 :: RewriteRule SLogic r1 = rewriteRule "R1" $ \p q r -> p :||: (q :||: r) :~> (p :||: q) :||: r r2 = rewriteRule "R2" $ \p q -> p :||: q :~> q :||: p r3 = rewriteRule "R3" $ \p -> p :||: p :~> p r4 = rewriteRule "R4" $ \p -> p :||: T :~> T r5 = rewriteRule "R5" $ \p -> p :||: F :~> p this = [r1, r2, r3, r4, r5, r6] go = reportPairs $ noDiamondPairs this r6 :: RewriteRule SLogic r6 = rewriteRule "R6" $ \p -> p :||: T :~> F r1, r2, r3 :: RewriteRule Expr r1 = rewriteRule "a1" $ \a -> 0+a :~> a r2 = rewriteRule "a3" $ \a b c -> a+(b+c) :~> (a+b)+c r3 = rewriteRule "a2" $ \a -> a+0 :~> a go = do -- putStrLn $ unlines $ map show $ criticalPairs [r1,r2] checkConfluence [r1,r2,r3] -}