{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE CPP #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE ExplicitNamespaces #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE PatternSynonyms #-}
module Data.Massiv.Core.Index (
Ix0 (..),
type Ix1,
pattern Ix1,
type Ix2 (Ix2, (:.)),
IxN ((:>), Ix3, Ix4, Ix5),
HighIxN,
type Ix3,
type Ix4,
type Ix5,
Ix,
type Sz1,
type Sz2,
type Sz3,
type Sz4,
type Sz5,
Sz (Sz, Sz1, Sz2, Sz3, Sz4, Sz5),
unSz,
zeroSz,
oneSz,
liftSz,
liftSz2,
consSz,
unconsSz,
snocSz,
unsnocSz,
setSzM,
insertSzM,
pullOutSzM,
toLinearSz,
mkSzM,
Dim (..),
Dimension (Dim1, Dim2, Dim3, Dim4, Dim5, DimN),
IsIndexDimension,
IsDimValid,
ReportInvalidDim,
Stride (Stride),
unStride,
toLinearIndexStride,
strideStart,
strideSize,
oneStride,
Border (..),
handleBorderIndex,
Lower,
Index (..),
zeroIndex,
oneIndex,
isZeroSz,
isNotZeroSz,
headDim,
tailDim,
lastDim,
initDim,
getDim',
setDim',
modifyDim',
dropDimM,
dropDim',
pullOutDim',
insertDim',
fromDimension,
getDimension,
setDimension,
modifyDimension,
dropDimension,
pullOutDimension,
insertDimension,
iter,
iterA_,
iterM_,
iterLinearM,
iterLinearM_,
module Data.Massiv.Core.Loop,
module Data.Massiv.Core.Index.Iterator,
module Data.Massiv.Core.Index.Tuple,
IndexException (..),
SizeException (..),
ShapeException (..),
guardNumberOfElements,
indexAssert,
indexWith,
) where
import Control.DeepSeq
import Control.Monad.Catch (MonadThrow (..))
import Data.Coerce
import Data.Functor.Identity (runIdentity)
import Data.Massiv.Core.Exception
import Data.Massiv.Core.Index.Internal
import Data.Massiv.Core.Index.Iterator
import Data.Massiv.Core.Index.Ix
import Data.Massiv.Core.Index.Stride
import Data.Massiv.Core.Index.Tuple
import Data.Massiv.Core.Loop
import GHC.TypeLits
#include "massiv.h"
type Sz1 = Sz Ix1
type Sz2 = Sz Ix2
type Sz3 = Sz Ix3
type Sz4 = Sz Ix4
type Sz5 = Sz Ix5
data Border e
=
Fill e
|
Wrap
|
Edge
|
Reflect
|
Continue
deriving (Border e -> Border e -> Bool
forall e. Eq e => Border e -> Border e -> Bool
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: Border e -> Border e -> Bool
$c/= :: forall e. Eq e => Border e -> Border e -> Bool
== :: Border e -> Border e -> Bool
$c== :: forall e. Eq e => Border e -> Border e -> Bool
Eq, Int -> Border e -> ShowS
forall e. Show e => Int -> Border e -> ShowS
forall e. Show e => [Border e] -> ShowS
forall e. Show e => Border e -> String
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Border e] -> ShowS
$cshowList :: forall e. Show e => [Border e] -> ShowS
show :: Border e -> String
$cshow :: forall e. Show e => Border e -> String
showsPrec :: Int -> Border e -> ShowS
$cshowsPrec :: forall e. Show e => Int -> Border e -> ShowS
Show)
instance NFData e => NFData (Border e) where
rnf :: Border e -> ()
rnf Border e
b = case Border e
b of
Fill e
e -> forall a. NFData a => a -> ()
rnf e
e
Border e
Wrap -> ()
Border e
Edge -> ()
Border e
Reflect -> ()
Border e
Continue -> ()
handleBorderIndex
:: Index ix
=> Border e
-> Sz ix
-> (ix -> e)
-> ix
-> e
handleBorderIndex :: forall ix e. Index ix => Border e -> Sz ix -> (ix -> e) -> ix -> e
handleBorderIndex Border e
border !Sz ix
sz ix -> e
getVal !ix
ix =
case Border e
border of
Fill e
val -> if forall ix. Index ix => Sz ix -> ix -> Bool
isSafeIndex Sz ix
sz ix
ix then ix -> e
getVal ix
ix else e
val
Border e
Wrap -> ix -> e
getVal (forall ix.
Index ix =>
Sz ix
-> ix -> (Sz Int -> Int -> Int) -> (Sz Int -> Int -> Int) -> ix
repairIndex Sz ix
sz ix
ix forall {a}. Integral a => Sz a -> a -> a
wrap forall {a}. Integral a => Sz a -> a -> a
wrap)
Border e
Edge -> ix -> e
getVal (forall ix.
Index ix =>
Sz ix
-> ix -> (Sz Int -> Int -> Int) -> (Sz Int -> Int -> Int) -> ix
repairIndex Sz ix
sz ix
ix (forall a b. a -> b -> a
const (forall a b. a -> b -> a
const Int
0)) (\(SafeSz Int
k) Int
_ -> Int
k forall a. Num a => a -> a -> a
- Int
1))
Border e
Reflect ->
ix -> e
getVal
( forall ix.
Index ix =>
Sz ix
-> ix -> (Sz Int -> Int -> Int) -> (Sz Int -> Int -> Int) -> ix
repairIndex
Sz ix
sz
ix
ix
(\(SafeSz Int
k) !Int
i -> (forall a. Num a => a -> a
abs Int
i forall a. Num a => a -> a -> a
- Int
1) forall a. Integral a => a -> a -> a
`mod` Int
k)
(\(SafeSz Int
k) !Int
i -> (-Int
i forall a. Num a => a -> a -> a
- Int
1) forall a. Integral a => a -> a -> a
`mod` Int
k)
)
Border e
Continue ->
ix -> e
getVal
( forall ix.
Index ix =>
Sz ix
-> ix -> (Sz Int -> Int -> Int) -> (Sz Int -> Int -> Int) -> ix
repairIndex
Sz ix
sz
ix
ix
(\(SafeSz Int
k) !Int
i -> forall a. Num a => a -> a
abs Int
i forall a. Integral a => a -> a -> a
`mod` Int
k)
(\(SafeSz Int
k) !Int
i -> (-Int
i forall a. Num a => a -> a -> a
- Int
2) forall a. Integral a => a -> a -> a
`mod` Int
k)
)
where
wrap :: Sz a -> a -> a
wrap (SafeSz a
k) a
i = a
i forall a. Integral a => a -> a -> a
`mod` a
k
{-# INLINE [1] wrap #-}
{-# INLINE [1] handleBorderIndex #-}
zeroIndex :: Index ix => ix
zeroIndex :: forall ix. Index ix => ix
zeroIndex = forall ix. Index ix => Int -> ix
pureIndex Int
0
{-# INLINE [1] zeroIndex #-}
oneIndex :: Index ix => ix
oneIndex :: forall ix. Index ix => ix
oneIndex = forall ix. Index ix => Int -> ix
pureIndex Int
1
{-# INLINE [1] oneIndex #-}
isNotZeroSz :: Index ix => Sz ix -> Bool
isNotZeroSz :: forall ix. Index ix => Sz ix -> Bool
isNotZeroSz !Sz ix
sz = forall ix. Index ix => Sz ix -> ix -> Bool
isSafeIndex Sz ix
sz forall ix. Index ix => ix
zeroIndex
{-# INLINE [1] isNotZeroSz #-}
isZeroSz :: Index ix => Sz ix -> Bool
isZeroSz :: forall ix. Index ix => Sz ix -> Bool
isZeroSz = Bool -> Bool
not forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix. Index ix => Sz ix -> Bool
isNotZeroSz
{-# INLINE [1] isZeroSz #-}
toLinearSz :: Index ix => Sz ix -> Sz1
toLinearSz :: forall ix. Index ix => Sz ix -> Sz Int
toLinearSz = coerce :: forall a b. Coercible a b => a -> b
coerce forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix. Index ix => Sz ix -> Int
totalElem
{-# INLINE [1] toLinearSz #-}
headDim :: Index ix => ix -> Int
headDim :: forall ix. Index ix => ix -> Int
headDim = forall a b. (a, b) -> a
fst forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix. Index ix => ix -> (Int, Lower ix)
unconsDim
{-# INLINE [1] headDim #-}
tailDim :: Index ix => ix -> Lower ix
tailDim :: forall ix. Index ix => ix -> Lower ix
tailDim = forall a b. (a, b) -> b
snd forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix. Index ix => ix -> (Int, Lower ix)
unconsDim
{-# INLINE [1] tailDim #-}
lastDim :: Index ix => ix -> Int
lastDim :: forall ix. Index ix => ix -> Int
lastDim = forall a b. (a, b) -> b
snd forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix. Index ix => ix -> (Lower ix, Int)
unsnocDim
{-# INLINE [1] lastDim #-}
initDim :: Index ix => ix -> Lower ix
initDim :: forall ix. Index ix => ix -> Lower ix
initDim = forall a b. (a, b) -> a
fst forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix. Index ix => ix -> (Lower ix, Int)
unsnocDim
{-# INLINE [1] initDim #-}
setDim' :: (HasCallStack, Index ix) => ix -> Dim -> Int -> ix
setDim' :: forall ix. (HasCallStack, Index ix) => ix -> Dim -> Int -> ix
setDim' ix
ix Dim
dim = forall a. HasCallStack => Either SomeException a -> a
throwEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix (m :: * -> *).
(Index ix, MonadThrow m) =>
ix -> Dim -> Int -> m ix
setDimM ix
ix Dim
dim
{-# INLINE [1] setDim' #-}
getDim' :: (HasCallStack, Index ix) => ix -> Dim -> Int
getDim' :: forall ix. (HasCallStack, Index ix) => ix -> Dim -> Int
getDim' ix
ix = forall a. HasCallStack => Either SomeException a -> a
throwEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix (m :: * -> *).
(Index ix, MonadThrow m) =>
ix -> Dim -> m Int
getDimM ix
ix
{-# INLINE [1] getDim' #-}
modifyDim' :: (HasCallStack, Index ix) => ix -> Dim -> (Int -> Int) -> (Int, ix)
modifyDim' :: forall ix.
(HasCallStack, Index ix) =>
ix -> Dim -> (Int -> Int) -> (Int, ix)
modifyDim' ix
ix Dim
dim = forall a. HasCallStack => Either SomeException a -> a
throwEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix (m :: * -> *).
(Index ix, MonadThrow m) =>
ix -> Dim -> (Int -> Int) -> m (Int, ix)
modifyDimM ix
ix Dim
dim
{-# INLINE [1] modifyDim' #-}
dropDimM :: (MonadThrow m, Index ix) => ix -> Dim -> m (Lower ix)
dropDimM :: forall (m :: * -> *) ix.
(MonadThrow m, Index ix) =>
ix -> Dim -> m (Lower ix)
dropDimM ix
ix = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall a b. (a, b) -> b
snd forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix (m :: * -> *).
(Index ix, MonadThrow m) =>
ix -> Dim -> m (Int, Lower ix)
pullOutDimM ix
ix
{-# INLINE [1] dropDimM #-}
dropDim' :: (HasCallStack, Index ix) => ix -> Dim -> Lower ix
dropDim' :: forall ix. (HasCallStack, Index ix) => ix -> Dim -> Lower ix
dropDim' ix
ix = forall a. HasCallStack => Either SomeException a -> a
throwEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (m :: * -> *) ix.
(MonadThrow m, Index ix) =>
ix -> Dim -> m (Lower ix)
dropDimM ix
ix
{-# INLINE [1] dropDim' #-}
pullOutDim' :: (HasCallStack, Index ix) => ix -> Dim -> (Int, Lower ix)
pullOutDim' :: forall ix. (HasCallStack, Index ix) => ix -> Dim -> (Int, Lower ix)
pullOutDim' ix
ix = forall a. HasCallStack => Either SomeException a -> a
throwEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix (m :: * -> *).
(Index ix, MonadThrow m) =>
ix -> Dim -> m (Int, Lower ix)
pullOutDimM ix
ix
{-# INLINE [1] pullOutDim' #-}
insertDim' :: (HasCallStack, Index ix) => Lower ix -> Dim -> Int -> ix
insertDim' :: forall ix. (HasCallStack, Index ix) => Lower ix -> Dim -> Int -> ix
insertDim' Lower ix
ix Dim
dim = forall a. HasCallStack => Either SomeException a -> a
throwEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall ix (m :: * -> *).
(Index ix, MonadThrow m) =>
Lower ix -> Dim -> Int -> m ix
insertDimM Lower ix
ix Dim
dim
{-# INLINE [1] insertDim' #-}
fromDimension :: KnownNat n => Dimension n -> Dim
fromDimension :: forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension = forall a b. (Integral a, Num b) => a -> b
fromIntegral forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat) (proxy :: Nat -> *).
KnownNat n =>
proxy n -> Integer
natVal
{-# INLINE [1] fromDimension #-}
setDimension :: IsIndexDimension ix n => ix -> Dimension n -> Int -> ix
setDimension :: forall ix (n :: Nat).
IsIndexDimension ix n =>
ix -> Dimension n -> Int -> ix
setDimension ix
ix = forall ix. (HasCallStack, Index ix) => ix -> Dim -> Int -> ix
setDim' ix
ix forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension
{-# INLINE [1] setDimension #-}
modifyDimension :: IsIndexDimension ix n => ix -> Dimension n -> (Int -> Int) -> (Int, ix)
modifyDimension :: forall ix (n :: Nat).
IsIndexDimension ix n =>
ix -> Dimension n -> (Int -> Int) -> (Int, ix)
modifyDimension ix
ix = forall ix.
(HasCallStack, Index ix) =>
ix -> Dim -> (Int -> Int) -> (Int, ix)
modifyDim' ix
ix forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension
{-# INLINE [1] modifyDimension #-}
getDimension :: IsIndexDimension ix n => ix -> Dimension n -> Int
getDimension :: forall ix (n :: Nat).
IsIndexDimension ix n =>
ix -> Dimension n -> Int
getDimension ix
ix = forall ix. (HasCallStack, Index ix) => ix -> Dim -> Int
getDim' ix
ix forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension
{-# INLINE [1] getDimension #-}
dropDimension :: IsIndexDimension ix n => ix -> Dimension n -> Lower ix
dropDimension :: forall ix (n :: Nat).
IsIndexDimension ix n =>
ix -> Dimension n -> Lower ix
dropDimension ix
ix = forall ix. (HasCallStack, Index ix) => ix -> Dim -> Lower ix
dropDim' ix
ix forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension
{-# INLINE [1] dropDimension #-}
pullOutDimension :: IsIndexDimension ix n => ix -> Dimension n -> (Int, Lower ix)
pullOutDimension :: forall ix (n :: Nat).
IsIndexDimension ix n =>
ix -> Dimension n -> (Int, Lower ix)
pullOutDimension ix
ix = forall ix. (HasCallStack, Index ix) => ix -> Dim -> (Int, Lower ix)
pullOutDim' ix
ix forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension
{-# INLINE [1] pullOutDimension #-}
insertDimension :: IsIndexDimension ix n => Lower ix -> Dimension n -> Int -> ix
insertDimension :: forall ix (n :: Nat).
IsIndexDimension ix n =>
Lower ix -> Dimension n -> Int -> ix
insertDimension Lower ix
ix = forall ix. (HasCallStack, Index ix) => Lower ix -> Dim -> Int -> ix
insertDim' Lower ix
ix forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (n :: Nat). KnownNat n => Dimension n -> Dim
fromDimension
{-# INLINE [1] insertDimension #-}
iter
:: Index ix
=> ix
-> ix
-> ix
-> (Int -> Int -> Bool)
-> a
-> (ix -> a -> a)
-> a
iter :: forall ix a.
Index ix =>
ix -> ix -> ix -> (Int -> Int -> Bool) -> a -> (ix -> a -> a) -> a
iter ix
sIx ix
eIx ix
incIx Int -> Int -> Bool
cond a
acc ix -> a -> a
f =
forall a. Identity a -> a
runIdentity forall a b. (a -> b) -> a -> b
$ forall ix (m :: * -> *) a.
(Index ix, Monad m) =>
ix
-> ix -> ix -> (Int -> Int -> Bool) -> a -> (ix -> a -> m a) -> m a
iterM ix
sIx ix
eIx ix
incIx Int -> Int -> Bool
cond a
acc (\ix
ix -> forall (m :: * -> *) a. Monad m => a -> m a
return forall b c a. (b -> c) -> (a -> b) -> a -> c
. ix -> a -> a
f ix
ix)
{-# INLINE iter #-}
iterLinearM
:: (Index ix, Monad m)
=> Sz ix
-> Int
-> Int
-> Int
-> (Int -> Int -> Bool)
-> a
-> (Int -> ix -> a -> m a)
-> m a
iterLinearM :: forall ix (m :: * -> *) a.
(Index ix, Monad m) =>
Sz ix
-> Int
-> Int
-> Int
-> (Int -> Int -> Bool)
-> a
-> (Int -> ix -> a -> m a)
-> m a
iterLinearM !Sz ix
sz !Int
k0 !Int
k1 !Int
inc Int -> Int -> Bool
cond !a
acc Int -> ix -> a -> m a
f =
forall (m :: * -> *) a.
Monad m =>
Int
-> (Int -> Bool) -> (Int -> Int) -> a -> (Int -> a -> m a) -> m a
loopM Int
k0 (Int -> Int -> Bool
`cond` Int
k1) (forall a. Num a => a -> a -> a
+ Int
inc) a
acc forall a b. (a -> b) -> a -> b
$ \ !Int
i !a
acc0 -> Int -> ix -> a -> m a
f Int
i (forall ix. Index ix => Sz ix -> Int -> ix
fromLinearIndex Sz ix
sz Int
i) a
acc0
{-# INLINE iterLinearM #-}
iterLinearM_
:: (Index ix, Monad m)
=> Sz ix
-> Int
-> Int
-> Int
-> (Int -> Int -> Bool)
-> (Int -> ix -> m ())
-> m ()
iterLinearM_ :: forall ix (m :: * -> *).
(Index ix, Monad m) =>
Sz ix
-> Int
-> Int
-> Int
-> (Int -> Int -> Bool)
-> (Int -> ix -> m ())
-> m ()
iterLinearM_ Sz ix
sz !Int
k0 !Int
k1 !Int
inc Int -> Int -> Bool
cond Int -> ix -> m ()
f =
forall (f :: * -> *) a.
Applicative f =>
Int -> (Int -> Bool) -> (Int -> Int) -> (Int -> f a) -> f ()
loopA_ Int
k0 (Int -> Int -> Bool
`cond` Int
k1) (forall a. Num a => a -> a -> a
+ Int
inc) forall a b. (a -> b) -> a -> b
$ \ !Int
i -> Int -> ix -> m ()
f Int
i (forall ix. Index ix => Sz ix -> Int -> ix
fromLinearIndex Sz ix
sz Int
i)
{-# INLINE iterLinearM_ #-}
#ifdef MASSIV_UNSAFE_CHECKS
indexAssert :: (HasCallStack, Index ix) => String -> (a -> Sz ix) -> (a -> ix -> e) -> a -> ix -> e
indexAssert funName getSize f arr ix
| isSafeIndex sz ix = f arr ix
| otherwise = _errorIx ("<" ++ funName ++ ">") sz ix
where
sz = getSize arr
#else
indexAssert :: String -> (a -> Sz ix) -> (a -> ix -> e) -> a -> ix -> e
indexAssert :: forall a ix e.
String -> (a -> Sz ix) -> (a -> ix -> e) -> a -> ix -> e
indexAssert String
_funName a -> Sz ix
_getSize a -> ix -> e
f a
arr ix
ix = a -> ix -> e
f a
arr ix
ix
#endif
{-# INLINE indexAssert #-}
indexWith
:: Index ix
=> String
-> Int
-> String
-> (arr -> Sz ix)
-> (arr -> ix -> e)
-> arr
-> ix
-> e
indexWith :: forall ix arr e.
Index ix =>
String
-> Int
-> String
-> (arr -> Sz ix)
-> (arr -> ix -> e)
-> arr
-> ix
-> e
indexWith String
fileName Int
lineNo String
funName arr -> Sz ix
getSize arr -> ix -> e
f arr
arr ix
ix
| forall ix. Index ix => Sz ix -> ix -> Bool
isSafeIndex Sz ix
sz ix
ix = arr -> ix -> e
f arr
arr ix
ix
| Bool
otherwise = forall ix ix' a.
(HasCallStack, Show ix, Show ix') =>
String -> ix -> ix' -> a
_errorIx (String
"<" forall a. [a] -> [a] -> [a]
++ String
fileName forall a. [a] -> [a] -> [a]
++ String
":" forall a. [a] -> [a] -> [a]
++ forall a. Show a => a -> String
show Int
lineNo forall a. [a] -> [a] -> [a]
++ String
"> " forall a. [a] -> [a] -> [a]
++ String
funName) Sz ix
sz ix
ix
where
sz :: Sz ix
sz = arr -> Sz ix
getSize arr
arr
{-# DEPRECATED indexWith "In favor of `indexAssert` that uses HasCallStack" #-}
_errorIx :: (HasCallStack, Show ix, Show ix') => String -> ix -> ix' -> a
_errorIx :: forall ix ix' a.
(HasCallStack, Show ix, Show ix') =>
String -> ix -> ix' -> a
_errorIx String
fName ix
sz ix'
ix =
forall a. HasCallStack => String -> a
error forall a b. (a -> b) -> a -> b
$
String
fName
forall a. [a] -> [a] -> [a]
++ String
": Index out of bounds: ("
forall a. [a] -> [a] -> [a]
++ forall a. Show a => a -> String
show ix'
ix
forall a. [a] -> [a] -> [a]
++ String
") for Array of size: ("
forall a. [a] -> [a] -> [a]
++ forall a. Show a => a -> String
show ix
sz
forall a. [a] -> [a] -> [a]
++ String
")"
{-# NOINLINE _errorIx #-}