-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
5 changed files
with
191 additions
and
31 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,190 @@ | ||
{-# LANGUAGE UndecidableInstances #-} | ||
|
||
{- | Primitive predicate simplifier. | ||
This is related to an NP-complete problem (see Boolean satisfiability problem). | ||
We focus on /immediate, operation-reducing simplifications/, and hope that the | ||
input is formed in such a way that our rules match. | ||
The simplifier may not be expectected to consistently implement any | ||
transformations whatsoever. The only guarantees are | ||
* the output has the same or fewer operations | ||
* the output meaning is identical to the input | ||
Implementation pitfalls: | ||
* not extensible: only works for built-in logical & relational predicates | ||
* no protection against non-termination e.g. if a pair of transformations loop | ||
* very tedious to write. that's life | ||
-} | ||
|
||
module Rerefined.Predicate.Simplify | ||
( Simplify | ||
, Simplify' | ||
) where | ||
|
||
import Rerefined.Predicate.Succeed | ||
import Rerefined.Predicate.Fail | ||
|
||
import Rerefined.Predicate.Logical | ||
|
||
import Rerefined.Predicate.Relational | ||
import GHC.TypeNats ( Natural, CmpNat ) | ||
import Data.Type.Ord ( OrdCond ) | ||
|
||
-- | Simplify the given predicate. | ||
-- | ||
-- Returns the input predicate if we were unable to simplify. | ||
type Simplify p = FromMaybe p (Simplify' p) | ||
|
||
-- | Promoted 'Data.Maybe.fromMaybe'. | ||
type family FromMaybe a ma where | ||
FromMaybe a Nothing = a | ||
FromMaybe a (Just a') = a' | ||
|
||
-- | Try to simplify the given predicate. | ||
-- | ||
-- Returns 'Nothing' if we were unable to simplify. | ||
type family Simplify' p where | ||
Simplify' (And l r) = SimplifyAnd l r | ||
Simplify' (Or l r) = SimplifyOr l r | ||
Simplify' (Nand l r) = SimplifyNand l r | ||
Simplify' (Not p) = SimplifyNot p | ||
|
||
Simplify' (CompareLength op n) = SimplifyCompareLength op n | ||
|
||
Simplify' p = Nothing | ||
|
||
type family SimplifyAnd l r where | ||
-- identity laws | ||
SimplifyAnd p Fail = Just Fail | ||
SimplifyAnd Fail p = Just Fail | ||
SimplifyAnd p Succeed = Just p | ||
SimplifyAnd Succeed p = Just p | ||
|
||
SimplifyAnd p p = Just p | ||
|
||
-- distributivity | ||
SimplifyAnd (Or x y) (Or x z) = Just (Or x (And y z)) | ||
|
||
-- special | ||
SimplifyAnd (CompareLength lop ln) (CompareLength rop rn) = | ||
SimplifyCompareLengthAnd lop ln rop rn | ||
|
||
-- recurse | ||
SimplifyAnd l r = | ||
(OrElseAndL r (Simplify' l) | ||
(OrElseAndR l (Simplify' r) | ||
Nothing)) | ||
|
||
type family OrElseAndL r mp cont where | ||
OrElseAndL r Nothing cont = cont | ||
OrElseAndL r (Just l') cont = Just (And l' r) | ||
|
||
type family OrElseAndR l mp cont where | ||
OrElseAndR l Nothing cont = cont | ||
OrElseAndR l (Just r') cont = Just (And l r') | ||
|
||
type family SimplifyOr l r where | ||
-- identity laws | ||
SimplifyOr Succeed p = Just Succeed | ||
SimplifyOr p Succeed = Just Succeed | ||
SimplifyOr Fail p = Just p | ||
SimplifyOr p Fail = Just p | ||
|
||
SimplifyOr p p = Just p | ||
|
||
-- distributivity | ||
SimplifyOr (And x y) (And x z) = Just (And x (Or y z)) | ||
|
||
-- special | ||
SimplifyOr (CompareLength lop ln) (CompareLength rop rn) = | ||
SimplifyCompareLengthOr lop ln rop rn | ||
|
||
-- recurse | ||
SimplifyOr l r = | ||
(OrElseOrL r (Simplify' l) | ||
(OrElseOrR l (Simplify' r) | ||
Nothing)) | ||
|
||
type family OrElseOrL r mp cont where | ||
OrElseOrL r Nothing cont = cont | ||
OrElseOrL r (Just l') cont = Just (Or l' r) | ||
|
||
type family OrElseOrR l mp cont where | ||
OrElseOrR l Nothing cont = cont | ||
OrElseOrR l (Just r') cont = Just (Or l r') | ||
|
||
type family SimplifyCompareLength (op :: RelOp) (n :: Natural) where | ||
SimplifyCompareLength RelOpLT 0 = Just Fail | ||
SimplifyCompareLength RelOpLTE 0 = Just (CompareLength RelOpEQ 0) | ||
|
||
-- TODO I think that's it for single relational predicates. | ||
SimplifyCompareLength op n = Nothing | ||
|
||
type family SimplifyCompareLengthAnd | ||
(lop :: RelOp) (ln :: Natural) (rop :: RelOp) (rn :: Natural) where | ||
-- @<n AND >m@: @n<=m@ -> False | ||
SimplifyCompareLengthAnd RelOpLT n RelOpGT m = | ||
OrdCond (CmpNat n m) | ||
(Just Fail) | ||
(Just Fail) | ||
Nothing | ||
SimplifyCompareLengthAnd RelOpGT m RelOpLT n = | ||
OrdCond (CmpNat n m) | ||
(Just Fail) | ||
(Just Fail) | ||
Nothing | ||
|
||
SimplifyCompareLengthAnd lop ln rop rn = Nothing | ||
|
||
type family SimplifyCompareLengthOr | ||
(lop :: RelOp) (ln :: Natural) (rop :: RelOp) (rn :: Natural) where | ||
-- @<n OR >m@: @n==m@ -> NEQ; @n>m@ -> True (?) | ||
SimplifyCompareLengthOr RelOpLT n RelOpGT m = | ||
OrdCond (CmpNat n m) | ||
Nothing | ||
(Just (CompareLength RelOpNEQ n)) | ||
(Just Succeed) | ||
SimplifyCompareLengthOr RelOpGT m RelOpLT n = | ||
OrdCond (CmpNat n m) | ||
Nothing | ||
(Just (CompareLength RelOpNEQ n)) | ||
(Just Succeed) | ||
|
||
SimplifyCompareLengthOr lop ln rop rn = Nothing | ||
|
||
type family SimplifyNand l r where | ||
-- TODO fill in rest | ||
SimplifyNand p p = Just (Not p) | ||
|
||
-- recurse | ||
SimplifyNand l r = | ||
(OrElseNandL r (Simplify' l) | ||
(OrElseNandR l (Simplify' r) | ||
Nothing)) | ||
|
||
type family OrElseNandL r mp cont where | ||
OrElseNandL r Nothing cont = cont | ||
OrElseNandL r (Just l') cont = Just (Nand l' r) | ||
|
||
type family OrElseNandR l mp cont where | ||
OrElseNandR l Nothing cont = cont | ||
OrElseNandR l (Just r') cont = Just (Nand l r') | ||
|
||
type family SimplifyNot p where | ||
-- double negation | ||
SimplifyNot (Not p) = Just p | ||
|
||
SimplifyNot Succeed = Just Fail | ||
SimplifyNot Fail = Just Succeed | ||
|
||
-- TODO handle relational predicates! | ||
|
||
-- recurse | ||
SimplifyNot p = OrElseNot (Simplify' p) Nothing | ||
|
||
type family OrElseNot mp cont where | ||
OrElseNot Nothing cont = cont | ||
OrElseNot (Just p') cont = Just (Not p') |