{-# LANGUAGE RankNTypes, ScopedTypeVariables #-}
{-# LANGUAGE DataKinds, DefaultSignatures, DeriveAnyClass, EmptyCase #-}
{-# LANGUAGE ExplicitNamespaces, FlexibleContexts, FlexibleInstances #-}
{-# LANGUAGE GADTs, KindSignatures, LambdaCase, PolyKinds #-}
{-# LANGUAGE StandaloneDeriving, TupleSections, TypeOperators #-}
module Proof.Propositional.Empty (Empty(..), withEmpty, withEmpty') where
import Data.Void (Void, absurd)
import GHC.Generics
import Unsafe.Coerce (unsafeCoerce)
class Empty a where
eliminate :: a -> x
default eliminate :: (Generic a, GEmpty (Rep a)) => a -> x
eliminate = Rep a Any -> x
forall k (f :: k -> *) (a :: k) x. GEmpty f => f a -> x
geliminate (Rep a Any -> x) -> (a -> Rep a Any) -> a -> x
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Rep a Any
forall a x. Generic a => a -> Rep a x
from
class GEmpty f where
geliminate :: f a -> x
instance GEmpty f => GEmpty (M1 i t f) where
geliminate :: M1 i t f a -> x
geliminate (M1 f a
a) = f a -> x
forall k (f :: k -> *) (a :: k) x. GEmpty f => f a -> x
geliminate f a
a
instance (GEmpty f, GEmpty g) => GEmpty (f :+: g) where
geliminate :: (:+:) f g a -> x
geliminate (L1 f a
a) = f a -> x
forall k (f :: k -> *) (a :: k) x. GEmpty f => f a -> x
geliminate f a
a
geliminate (R1 g a
b) = g a -> x
forall k (f :: k -> *) (a :: k) x. GEmpty f => f a -> x
geliminate g a
b
instance Empty c => GEmpty (K1 i c) where
geliminate :: K1 i c a -> x
geliminate (K1 c
a) = c -> x
forall a x. Empty a => a -> x
eliminate c
a
instance GEmpty V1 where
geliminate :: V1 a -> x
geliminate = \ case {}
deriving instance (Empty a, Empty b) => Empty (Either a b)
deriving instance Empty Void
newtype MagicEmpty e a = MagicEmpty (Empty e => a)
withEmpty :: forall a b. (a -> Void) -> (Empty a => b) -> b
withEmpty :: (a -> Void) -> (Empty a => b) -> b
withEmpty a -> Void
neg Empty a => b
k = MagicEmpty a b -> (a -> Any) -> b
forall a b. a -> b
unsafeCoerce ((Empty a => b) -> MagicEmpty a b
forall e a. (Empty e => a) -> MagicEmpty e a
MagicEmpty Empty a => b
k :: MagicEmpty a b) (Void -> Any
forall x. Void -> x
absurd (Void -> Any) -> (a -> Void) -> a -> Any
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Void
neg)
withEmpty' :: forall a b. (forall c. a -> c) -> (Empty a => b) -> b
withEmpty' :: (forall c. a -> c) -> (Empty a => b) -> b
withEmpty' forall c. a -> c
neg Empty a => b
k = MagicEmpty a b -> (a -> Any) -> b
forall a b. a -> b
unsafeCoerce ((Empty a => b) -> MagicEmpty a b
forall e a. (Empty e => a) -> MagicEmpty e a
MagicEmpty Empty a => b
k :: MagicEmpty a b) a -> Any
forall c. a -> c
neg