module Ideas.Common.Rewriting.RewriteRule
(
Different(..)
, RewriteRule, ruleSpecTerm, RuleSpec(..)
, makeRewriteRule, termRewriteRule, RuleBuilder(..)
, showRewriteRule
, metaInRewriteRule, renumberRewriteRule
, symbolMatcher, symbolBuilder
) where
import Data.Maybe
import Data.Monoid
import Ideas.Common.Classes
import Ideas.Common.Environment
import Ideas.Common.Id
import Ideas.Common.Rewriting.Substitution
import Ideas.Common.Rewriting.Term
import Ideas.Common.Rewriting.Unification
import Ideas.Common.Utils.Uniplate (descend)
import Ideas.Common.View hiding (match)
import qualified Data.IntSet as IS
import qualified Data.Map as M
infixl 1 :~>
data RuleSpec a = a :~> a deriving Show
instance Functor RuleSpec where
fmap f (a :~> b) = f a :~> f b
data RewriteRule a = R
{ ruleId :: Id
, ruleSpecTerm :: RuleSpec Term
, ruleShow :: a -> String
, ruleTermView :: View Term a
, ruleMatchers :: M.Map Symbol SymbolMatch
, ruleBuilders :: M.Map Symbol ([Term] -> Term)
}
instance Show (RewriteRule a) where
show = showId
instance HasId (RewriteRule a) where
getId = ruleId
changeId f r = r {ruleId = f (ruleId r)}
class Different a where
different :: (a, a)
instance Different a => Different [a] where
different = ([], [fst different])
instance Different Char where
different = ('a', 'b')
class (IsTerm a, Show a) => RuleBuilder t a | t -> a where
buildRuleSpec :: Int -> t -> RuleSpec Term
instance (IsTerm a, Show a) => RuleBuilder (RuleSpec a) a where
buildRuleSpec = const $ fmap toTerm
instance (Different a, RuleBuilder t b) => RuleBuilder (a -> t) b where
buildRuleSpec i f = buildFunction i (buildRuleSpec (i+1) . f)
buildFunction :: Different a => Int -> (a -> RuleSpec Term) -> RuleSpec Term
buildFunction n f = fzip (fill n) ((f *** f) different)
where
fzip g (a :~> b, c :~> d) = g a c :~> g b d
fill :: Int -> Term -> Term -> Term
fill i = rec
where
rec (TCon s xs) (TCon t ys) | s == t && length xs == length ys =
TCon s (zipWith rec xs ys)
rec (TList xs) (TList ys) | length xs == length ys =
TList (zipWith rec xs ys)
rec a b
| a == b = a
| otherwise = TMeta i
buildSpec :: M.Map Symbol SymbolMatch
-> M.Map Symbol ([Term] -> Term)
-> RuleSpec Term -> Term -> [(Term, [Term])]
buildSpec sm sb (lhs :~> rhs) a = do
(sub, ml, mr) <- matchExtended sm lhs a
let sym = maybe (error "buildSpec") fst (getFunction lhs)
extLeft = maybe id (binary sym) ml
extRight = maybe id (flip (binary sym)) mr
new = useBuilders $ extLeft $ extRight $ sub |-> rhs
args = mapMaybe (`lookupVar` sub) $ IS.toList $ dom sub
return (new, args)
where
useBuilders
| M.null sb = id
| otherwise = rec
where
rec (TCon s xs) =
fromMaybe (TCon s) (M.lookup s sb) (map rec xs)
rec term = term
makeRewriteRule :: (IsId n, RuleBuilder f a) => n -> f -> RewriteRule a
makeRewriteRule s f =
R (newId s) (buildRuleSpec 0 f) show termView M.empty M.empty
termRewriteRule :: (IsId n, IsTerm a, Show a) => n -> RuleSpec Term -> RewriteRule a
termRewriteRule s spec =
R (newId s) spec show termView M.empty M.empty
symbolMatcher :: Symbol -> SymbolMatch -> RewriteRule a -> RewriteRule a
symbolMatcher s f r = r {ruleMatchers = M.insert s f (ruleMatchers r)}
symbolBuilder :: Symbol -> ([Term] -> Term) -> RewriteRule a -> RewriteRule a
symbolBuilder s f r = r {ruleBuilders = M.insert s f (ruleBuilders r)}
instance Apply RewriteRule where
applyAll r = map fst . applyRewriteRule r
applyRewriteRule :: RewriteRule a -> a -> [(a, Environment)]
applyRewriteRule r a = do
let builder = buildSpec (ruleMatchers r) (ruleBuilders r) (ruleSpecTerm r)
term = toTermRR r a
(out, xs) <- builder term
let env = mconcat (zipWith make xs [1::Int ..])
make t = flip singleBinding t . makeRef . show
b <- fromTermRR r out
return (b, env)
showRewriteRule :: Bool -> RewriteRule a -> Maybe String
showRewriteRule sound r = do
x <- fromTermRR r (sub |-> a)
y <- fromTermRR r (sub |-> b)
let op = if sound then "~>" else "/~>"
return (ruleShow r x ++ " " ++ op ++ " " ++ ruleShow r y)
where
a :~> b = ruleSpecTerm r
vs = IS.toList (metaVarSet a `IS.union` metaVarSet b)
sub = listToSubst $ zip vs [ TVar [c] | c <- ['a' ..] ]
metaInRewriteRule :: RewriteRule a -> [Int]
metaInRewriteRule r = metaVars a ++ metaVars b
where a :~> b = ruleSpecTerm r
renumberRewriteRule :: Int -> RewriteRule a -> RewriteRule a
renumberRewriteRule n r = r {ruleSpecTerm = fmap f (ruleSpecTerm r)}
where
f (TMeta i) = TMeta (i+n)
f term = descend f term
toTermRR :: RewriteRule a -> a -> Term
toTermRR = build . ruleTermView
fromTermRR :: Monad m => RewriteRule a -> Term -> m a
fromTermRR = matchM . ruleTermView