{-# OPTIONS_GHC -O2 #-}
{-# OPTIONS_HADDOCK hide #-}
{-# LANGUAGE CPP #-}
#if !defined(__GLASGOW_HASKELL__)
#error "Your compiler is not GHC. Let us know if dlist can be made to work on it."
#endif
{-# LANGUAGE TypeFamilies #-}
#if __GLASGOW_HASKELL__ >= 708
{-# LANGUAGE PatternSynonyms #-}
{-# LANGUAGE Unsafe #-}
{-# LANGUAGE ViewPatterns #-}
#endif
module Data.DList.Internal where
import qualified Control.Applicative as Applicative
import Control.DeepSeq (NFData (..))
import qualified Control.Monad as Monad
#if MIN_VERSION_base(4,9,0) && !MIN_VERSION_base(4,13,0)
import qualified Control.Monad.Fail as Monad
#endif
import qualified Data.Foldable as Foldable
import Data.Function (on)
import qualified Data.List as List
import qualified Data.Monoid as Monoid
#if MIN_VERSION_base(4,9,0)
import qualified Data.Semigroup as Semigroup
#endif
import Data.String (IsString (..))
import qualified Data.Traversable as Traversable
#if __GLASGOW_HASKELL__ >= 708
import qualified GHC.Exts as Exts
#endif
import qualified Text.Read as Read
import Prelude hiding (concat, foldr, head, map, replicate, tail)
newtype DList a = UnsafeDList {forall a. DList a -> [a] -> [a]
unsafeApplyDList :: [a] -> [a]}
{-# INLINE fromList #-}
fromList :: [a] -> DList a
fromList :: forall a. [a] -> DList a
fromList = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. [a] -> [a] -> [a]
(++)
{-# INLINE toList #-}
toList :: DList a -> [a]
toList :: forall a. DList a -> [a]
toList = (forall a b. (a -> b) -> a -> b
$ []) forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a] -> [a]
unsafeApplyDList
#if __GLASGOW_HASKELL__ >= 708
#if __GLASGOW_HASKELL__ >= 710
pattern Nil :: DList a
#endif
pattern $mNil :: forall {r} {a}. DList a -> ((# #) -> r) -> ((# #) -> r) -> r
Nil <- (toList -> [])
#if __GLASGOW_HASKELL__ >= 710
pattern Cons :: a -> [a] -> DList a
#endif
pattern $mCons :: forall {r} {a}. DList a -> (a -> [a] -> r) -> ((# #) -> r) -> r
Cons x xs <- (toList -> x : xs)
#endif
{-# INLINE apply #-}
apply :: DList a -> [a] -> [a]
apply :: forall a. DList a -> [a] -> [a]
apply = forall a. DList a -> [a] -> [a]
unsafeApplyDList
{-# INLINE empty #-}
empty :: DList a
empty :: forall a. DList a
empty = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall a. a -> a
id
{-# INLINE singleton #-}
singleton :: a -> DList a
singleton :: forall a. a -> DList a
singleton = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall b c a. (b -> c) -> (a -> b) -> a -> c
. (:)
infixr 9 `cons`
{-# INLINE cons #-}
cons :: a -> DList a -> DList a
cons :: forall a. a -> DList a -> DList a
cons a
x DList a
xs = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall a b. (a -> b) -> a -> b
$ (a
x forall a. a -> [a] -> [a]
:) forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a] -> [a]
unsafeApplyDList DList a
xs
infixl 9 `snoc`
{-# INLINE snoc #-}
snoc :: DList a -> a -> DList a
snoc :: forall a. DList a -> a -> DList a
snoc DList a
xs a
x = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall a b. (a -> b) -> a -> b
$ forall a. DList a -> [a] -> [a]
unsafeApplyDList DList a
xs forall b c a. (b -> c) -> (a -> b) -> a -> c
. (a
x forall a. a -> [a] -> [a]
:)
{-# INLINE append #-}
append :: DList a -> DList a -> DList a
append :: forall a. DList a -> DList a -> DList a
append DList a
xs DList a
ys = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall a b. (a -> b) -> a -> b
$ forall a. DList a -> [a] -> [a]
unsafeApplyDList DList a
xs forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a] -> [a]
unsafeApplyDList DList a
ys
{-# INLINE concat #-}
concat :: [DList a] -> DList a
concat :: forall a. [DList a] -> DList a
concat = forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
List.foldr forall a. DList a -> DList a -> DList a
append forall a. DList a
empty
{-# INLINE replicate #-}
replicate :: Int -> a -> DList a
replicate :: forall a. Int -> a -> DList a
replicate Int
n a
x = forall a. ([a] -> [a]) -> DList a
UnsafeDList forall a b. (a -> b) -> a -> b
$ \[a]
xs ->
let go :: Int -> [a]
go Int
m
| Int
m forall a. Ord a => a -> a -> Bool
<= Int
0 = [a]
xs
| Bool
otherwise = a
x forall a. a -> [a] -> [a]
: Int -> [a]
go (Int
m forall a. Num a => a -> a -> a
-Int
1)
in Int -> [a]
go Int
n
{-# INLINE head #-}
head :: DList a -> a
head :: forall a. DList a -> a
head DList a
xs = case forall a. DList a -> [a]
toList DList a
xs of
a
x : [a]
_ -> a
x
[] -> forall a. HasCallStack => [Char] -> a
error [Char]
"Data.DList.head: empty DList"
{-# INLINE tail #-}
tail :: DList a -> [a]
tail :: forall a. DList a -> [a]
tail DList a
xs = case forall a. DList a -> [a]
toList DList a
xs of
a
_ : [a]
ys -> [a]
ys
[] -> forall a. HasCallStack => [Char] -> a
error [Char]
"Data.DList.tail: empty DList"
unfoldr :: (b -> Maybe (a, b)) -> b -> DList a
unfoldr :: forall b a. (b -> Maybe (a, b)) -> b -> DList a
unfoldr b -> Maybe (a, b)
f b
z =
case b -> Maybe (a, b)
f b
z of
Maybe (a, b)
Nothing -> forall a. DList a
empty
Just (a
x, b
z') -> forall a. a -> DList a -> DList a
cons a
x forall a b. (a -> b) -> a -> b
$ forall b a. (b -> Maybe (a, b)) -> b -> DList a
unfoldr b -> Maybe (a, b)
f b
z'
{-# INLINE foldr #-}
foldr :: (a -> b -> b) -> b -> DList a -> b
foldr :: forall a b. (a -> b -> b) -> b -> DList a -> b
foldr a -> b -> b
f b
z = forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
List.foldr a -> b -> b
f b
z forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE map #-}
map :: (a -> b) -> DList a -> DList b
map :: forall a b. (a -> b) -> DList a -> DList b
map a -> b
f = forall a b. (a -> b -> b) -> b -> DList a -> b
foldr (forall a. a -> DList a -> DList a
cons forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> b
f) forall a. DList a
empty
{-# INLINE intercalate #-}
intercalate :: DList a -> [DList a] -> DList a
intercalate :: forall a. DList a -> [DList a] -> DList a
intercalate DList a
sep = forall a. [DList a] -> DList a
concat forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. a -> [a] -> [a]
List.intersperse DList a
sep
instance Eq a => Eq (DList a) where
== :: DList a -> DList a -> Bool
(==) = forall a. Eq a => a -> a -> Bool
(==) forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` forall a. DList a -> [a]
toList
instance Ord a => Ord (DList a) where
compare :: DList a -> DList a -> Ordering
compare = forall a. Ord a => a -> a -> Ordering
compare forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` forall a. DList a -> [a]
toList
instance Read a => Read (DList a) where
readPrec :: ReadPrec (DList a)
readPrec = forall a. ReadPrec a -> ReadPrec a
Read.parens forall a b. (a -> b) -> a -> b
$
forall a. Int -> ReadPrec a -> ReadPrec a
Read.prec Int
10 forall a b. (a -> b) -> a -> b
$ do
Read.Ident [Char]
"fromList" <- ReadPrec Lexeme
Read.lexP
[a]
dl <- forall a. Read a => ReadPrec a
Read.readPrec
forall (m :: * -> *) a. Monad m => a -> m a
return (forall a. [a] -> DList a
fromList [a]
dl)
readListPrec :: ReadPrec [DList a]
readListPrec = forall a. Read a => ReadPrec [a]
Read.readListPrecDefault
instance Show a => Show (DList a) where
showsPrec :: Int -> DList a -> ShowS
showsPrec Int
p DList a
dl =
Bool -> ShowS -> ShowS
showParen (Int
p forall a. Ord a => a -> a -> Bool
> Int
10) forall a b. (a -> b) -> a -> b
$
[Char] -> ShowS
showString [Char]
"fromList " forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. Show a => a -> ShowS
shows (forall a. DList a -> [a]
toList DList a
dl)
instance Monoid.Monoid (DList a) where
{-# INLINE mempty #-}
mempty :: DList a
mempty = forall a. DList a
empty
#if MIN_VERSION_base(4,11,0)
#else
{-# INLINE mappend #-}
#if MIN_VERSION_base(4,9,0)
mappend = (Semigroup.<>)
#else
mappend = append
#endif
#endif
instance Functor DList where
{-# INLINE fmap #-}
fmap :: forall a b. (a -> b) -> DList a -> DList b
fmap = forall a b. (a -> b) -> DList a -> DList b
map
instance Applicative.Applicative DList where
{-# INLINE pure #-}
pure :: forall a. a -> DList a
pure = forall a. a -> DList a
singleton
{-# INLINE (<*>) #-}
<*> :: forall a b. DList (a -> b) -> DList a -> DList b
(<*>) = forall (m :: * -> *) a b. Monad m => m (a -> b) -> m a -> m b
Monad.ap
instance Applicative.Alternative DList where
{-# INLINE empty #-}
empty :: forall a. DList a
empty = forall a. DList a
empty
{-# INLINE (<|>) #-}
<|> :: forall a. DList a -> DList a -> DList a
(<|>) = forall a. DList a -> DList a -> DList a
append
instance Monad DList where
{-# INLINE (>>=) #-}
DList a
m >>= :: forall a b. DList a -> (a -> DList b) -> DList b
>>= a -> DList b
k =
forall a b. (a -> b -> b) -> b -> DList a -> b
foldr (forall a. DList a -> DList a -> DList a
append forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> DList b
k) forall a. DList a
empty DList a
m
{-# INLINE return #-}
return :: forall a. a -> DList a
return = forall (f :: * -> *) a. Applicative f => a -> f a
Applicative.pure
#if !MIN_VERSION_base(4,13,0)
{-# INLINE fail #-}
fail _ = empty
#endif
#if MIN_VERSION_base(4,9,0)
instance Monad.MonadFail DList where
{-# INLINE fail #-}
fail :: forall a. [Char] -> DList a
fail [Char]
_ = forall a. DList a
empty
#endif
instance Monad.MonadPlus DList where
{-# INLINE mzero #-}
mzero :: forall a. DList a
mzero = forall a. DList a
empty
{-# INLINE mplus #-}
mplus :: forall a. DList a -> DList a -> DList a
mplus = forall a. DList a -> DList a -> DList a
append
instance Foldable.Foldable DList where
{-# INLINE fold #-}
fold :: forall m. Monoid m => DList m -> m
fold = forall a. Monoid a => [a] -> a
Monoid.mconcat forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE foldMap #-}
foldMap :: forall m a. Monoid m => (a -> m) -> DList a -> m
foldMap a -> m
f = forall (t :: * -> *) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
Foldable.foldMap a -> m
f forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE foldr #-}
foldr :: forall a b. (a -> b -> b) -> b -> DList a -> b
foldr a -> b -> b
f b
x = forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
List.foldr a -> b -> b
f b
x forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE foldl #-}
foldl :: forall b a. (b -> a -> b) -> b -> DList a -> b
foldl b -> a -> b
f b
x = forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
List.foldl b -> a -> b
f b
x forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE foldr1 #-}
foldr1 :: forall a. (a -> a -> a) -> DList a -> a
foldr1 a -> a -> a
f = forall (t :: * -> *) a. Foldable t => (a -> a -> a) -> t a -> a
List.foldr1 a -> a -> a
f forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE foldl1 #-}
foldl1 :: forall a. (a -> a -> a) -> DList a -> a
foldl1 a -> a -> a
f = forall (t :: * -> *) a. Foldable t => (a -> a -> a) -> t a -> a
List.foldl1 a -> a -> a
f forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
#if __GLASGOW_HASKELL__ >= 706
{-# INLINE foldl' #-}
foldl' :: forall b a. (b -> a -> b) -> b -> DList a -> b
foldl' b -> a -> b
f b
x = forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
List.foldl' b -> a -> b
f b
x forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
{-# INLINE foldr' #-}
foldr' :: forall a b. (a -> b -> b) -> b -> DList a -> b
foldr' a -> b -> b
f b
x = forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
Foldable.foldr' a -> b -> b
f b
x forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
#endif
#if MIN_VERSION_base(4,8,0)
{-# INLINE toList #-}
toList :: forall a. DList a -> [a]
toList = forall a. DList a -> [a]
Data.DList.Internal.toList
#endif
instance Traversable.Traversable DList where
{-# INLINE traverse #-}
traverse :: forall (f :: * -> *) a b.
Applicative f =>
(a -> f b) -> DList a -> f (DList b)
traverse a -> f b
f = forall a b. (a -> b -> b) -> b -> DList a -> b
foldr a -> f (DList b) -> f (DList b)
cons_f (forall (f :: * -> *) a. Applicative f => a -> f a
Applicative.pure forall a. DList a
empty)
where
cons_f :: a -> f (DList b) -> f (DList b)
cons_f a
x = forall (f :: * -> *) a b c.
Applicative f =>
(a -> b -> c) -> f a -> f b -> f c
Applicative.liftA2 forall a. a -> DList a -> DList a
cons (a -> f b
f a
x)
instance NFData a => NFData (DList a) where
{-# INLINE rnf #-}
rnf :: DList a -> ()
rnf = forall a. NFData a => a -> ()
rnf forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. DList a -> [a]
toList
instance a ~ Char => IsString (DList a) where
{-# INLINE fromString #-}
fromString :: [Char] -> DList a
fromString = forall a. [a] -> DList a
fromList
#if __GLASGOW_HASKELL__ >= 708
instance Exts.IsList (DList a) where
type Item (DList a) = a
{-# INLINE fromList #-}
fromList :: [Item (DList a)] -> DList a
fromList = forall a. [a] -> DList a
fromList
{-# INLINE toList #-}
toList :: DList a -> [Item (DList a)]
toList = forall a. DList a -> [a]
toList
#endif
#if MIN_VERSION_base(4,9,0)
instance Semigroup.Semigroup (DList a) where
{-# INLINE (<>) #-}
<> :: DList a -> DList a -> DList a
(<>) = forall a. DList a -> DList a -> DList a
append
stimes :: forall b. Integral b => b -> DList a -> DList a
stimes b
n = case forall a. Ord a => a -> a -> Ordering
compare b
n b
0 of
Ordering
LT -> forall a. HasCallStack => [Char] -> a
error [Char]
"Data.DList.stimes: negative multiplier"
Ordering
_ -> forall b a. (Integral b, Monoid a) => b -> a -> a
Semigroup.stimesMonoid b
n
#endif