1
1
mirror of https://github.com/github/semantic.git synced 2024-11-25 11:04:00 +03:00

Redefine Element using a type family computing which side to recur on.

This avoids having to reassociate the tree, which is considerably more expensive for balanced trees.
This commit is contained in:
Rob Rix 2019-10-08 15:30:37 -04:00
parent 75ed50c574
commit da890fa747
No known key found for this signature in database
GPG Key ID: F188A01508EA1CF7

View File

@ -19,41 +19,39 @@ instance (Element' elem sub sup, elem ~ Elem sub sup) => Element sub sup where
pattern Prj :: Element sub sup => sub a -> sup a pattern Prj :: Element sub sup => sub a -> sup a
pattern Prj sub <- (prj -> Just sub) pattern Prj sub <- (prj -> Just sub)
data Side = None | Here | L | R
type family Elem sub sup where type family Elem sub sup :: Side where
Elem t t = 'True Elem t t = 'Here
Elem t (l :+: r) = Elem t l || Elem t r Elem t (l :+: r) = Elem' 'L t l <> Elem' 'R t r
Elem _ _ = 'False Elem _ _ = 'None
type family a || b where type family Elem' (side :: Side) sub sup :: Side where
'True || _ = 'True Elem' s t t = s
_ || 'True = 'True Elem' s t (l :+: r) = Elem' s t l <> Elem' s t r
_ || _ = 'False Elem' _ _ _ = 'None
class Element' (elem :: Bool) sub sup where type family (a :: Side) <> (b :: Side) :: Side where
'None <> b = b
a <> _ = a
class Element' (elem :: Side) sub sup where
prj' :: sup a -> Maybe (sub a) prj' :: sup a -> Maybe (sub a)
instance {-# OVERLAPPABLE #-} instance {-# OVERLAPPABLE #-}
Element' 'True t t where Element' 'Here t t where
prj' = Just prj' = Just
instance {-# OVERLAPPABLE #-} instance {-# OVERLAPPABLE #-}
Element' 'True t (l1 :+: l2 :+: r) Element t l
=> Element' 'True t ((l1 :+: l2) :+: r) where => Element' 'L t (l :+: r) where
prj' = prj' @'True . reassoc where prj' (L1 l) = prj l
reassoc (L1 (L1 l)) = L1 l
reassoc (L1 (R1 l)) = R1 (L1 l)
reassoc (R1 r) = R1 (R1 r)
instance {-# OVERLAPPABLE #-}
Element' 'True t (t :+: r) where
prj' (L1 l) = Just l
prj' _ = Nothing prj' _ = Nothing
instance {-# OVERLAPPABLE #-} instance {-# OVERLAPPABLE #-}
Element' 'True t r Element t r
=> Element' 'True t (l :+: r) where => Element' 'R t (l :+: r) where
prj' (R1 r) = prj' @'True r prj' (R1 r) = prj r
prj' _ = Nothing prj' _ = Nothing
@ -68,5 +66,5 @@ type family ShowSum' p t where
instance TypeError instance TypeError
( 'ShowType t ':<>: 'Text " is not in" ( 'ShowType t ':<>: 'Text " is not in"
':$$: ShowSum u) ':$$: ShowSum u)
=> Element' 'False t u where => Element' 'None t u where
prj' _ = Nothing prj' _ = Nothing