{-# LANGUAGE GADTs #-} {-# LANGUAGE Rank2Types #-} -------------------------------------------------------------------------------- -- | -- Module : Data.Comp.TermRewriting -- Copyright : (c) 2010-2011 Patrick Bahr -- License : BSD3 -- Maintainer : Patrick Bahr -- Stability : experimental -- Portability : non-portable (GHC Extensions) -- -- This module defines term rewriting systems (TRSs) using compositional data -- types. -- -------------------------------------------------------------------------------- module Data.Comp.TermRewriting where import Prelude hiding (any) import Data.Comp.Algebra import Data.Comp.Equality import Data.Comp.Matching import Data.Comp.Sum import Data.Comp.Term import Data.Foldable import Data.Map (Map) import qualified Data.Map as Map import Data.Maybe import qualified Data.Set as Set import Control.Monad {-| This type represents /recursive program schemes/. -} type RPS f g = Hom f g -- | This type represents variables. type Var = Int {-| This type represents term rewrite rules from signature @f@ to signature @g@ over variables of type @v@ -} type Rule f g v = (Context f v, Context g v) {-| This type represents term rewriting systems (TRSs) from signature @f@ to signature @g@ over variables of type @v@. -} type TRS f g v = [Rule f g v] -- | This type represents a potential single step reduction from any -- input. type Step t = t -> Maybe t -- | This type represents a potential single step reduction from any -- input. If there is no single step then the return value is the -- input together with @False@. Otherwise, the successor is returned -- together with @True@. type BStep t = t -> (t,Bool) {-| This function tries to match the given rule against the given term (resp. context in general) at the root. If successful, the function returns the right hand side of the rule and the matching substitution. -} matchRule :: (Ord v, EqF f, Eq a, Functor f, Foldable f) => Rule f g v -> Cxt h f a -> Maybe (Context g v, Map v (Cxt h f a)) matchRule (lhs,rhs) t = do subst <- matchCxt lhs t return (rhs,subst) -- | This function tries to match the rules of the given TRS against -- the given term (resp. context in general) at the root. The first -- rule in the TRS that matches is then used and the corresponding -- right-hand side as well the matching substitution is returned. matchRules :: (Ord v, EqF f, Eq a, Functor f, Foldable f) => TRS f g v -> Cxt h f a -> Maybe (Context g v, Map v (Cxt h f a)) matchRules trs t = listToMaybe $ mapMaybe (`matchRule` t) trs {-| This function tries to apply the given rule at the root of the given term (resp. context in general). If successful, the function returns the result term of the rewrite step; otherwise @Nothing@. -} appRule :: (Ord v, EqF f, Eq a, Functor f, Foldable f) => Rule f f v -> Step (Cxt h f a) appRule rule t = do (res, subst) <- matchRule rule t return $ substHoles' res subst {-| This function tries to apply one of the rules in the given TRS at the root of the given term (resp. context in general) by trying each rule one by one using 'appRule' until one rule is applicable. If no rule is applicable @Nothing@ is returned. -} appTRS :: (Ord v, EqF f, Eq a, Functor f, Foldable f) => TRS f f v -> Step (Cxt h f a) appTRS trs t = listToMaybe $ mapMaybe (`appRule` t) trs {-| This is an auxiliary function that turns function @f@ of type @(t -> Maybe t)@ into functions @f'@ of type @t -> (t,Bool)@. @f' x@ evaluates to @(y,True)@ if @f x@ evaluates to @Just y@, and to @(x,False)@ if @f x@ evaluates to @Nothing@. This function is useful to change the output of functions that apply rules such as 'appTRS'. -} bStep :: Step t -> BStep t bStep f t = case f t of Nothing -> (t, False) Just t' -> (t',True) {-| This function performs a parallel reduction step by trying to apply rules of the given system to all outermost redexes. If the given term contains no redexes, @Nothing@ is returned. -} parTopStep :: (Ord v, EqF f, Eq a, Foldable f, Functor f) => TRS f f v -> Step (Cxt h f a) parTopStep _ Hole{} = Nothing parTopStep trs c@(Term t) = tTop `mplus` tBelow' where tTop = appTRS trs c tBelow = fmap (bStep $ parTopStep trs) t tAny = any snd tBelow tBelow' | tAny = Just $ Term $ fmap fst tBelow | otherwise = Nothing {-| This function performs a parallel reduction step by trying to apply rules of the given system to all outermost redexes and then recursively in the variable positions of the redexes. If the given term does not contain any redexes, @Nothing@ is returned. -} parallelStep :: (Ord v, EqF f, Eq a,Foldable f, Functor f) => TRS f f v -> Step (Cxt h f a) parallelStep _ Hole{} = Nothing parallelStep trs c@(Term t) = case matchRules trs c of Nothing | anyBelow -> Just $ Term $ fmap fst below | otherwise -> Nothing where below = fmap (bStep $ parallelStep trs) t anyBelow = any snd below Just (rhs,subst) -> Just $ substHoles' rhs substBelow where rhsVars = Set.fromList $ toList rhs substBelow = Map.mapMaybeWithKey apply subst apply v t | Set.member v rhsVars = Just $ fst $ bStep (parallelStep trs) t | otherwise = Nothing {-| This function applies the given reduction step repeatedly until a normal form is reached. -} reduce :: Step t -> t -> t reduce s t = case s t of Nothing -> t Just t' -> reduce s t'