{-# LANGUAGE CPP #-} {-# LANGUAGE PatternSynonyms #-} #if __GLASGOW_HASKELL__ >= 800 {-# LANGUAGE GeneralizedNewtypeDeriving #-} #else {-# LANGUAGE GADTs #-} {-# LANGUAGE StandaloneDeriving #-} #endif -- | -- Module : Data.Massiv.Core.Index.Stride -- Copyright : (c) Alexey Kuleshevich 2018 -- License : BSD3 -- Maintainer : Alexey Kuleshevich <lehins@yandex.ru> -- Stability : experimental -- Portability : non-portable -- module Data.Massiv.Core.Index.Stride ( Stride(SafeStride) , pattern Stride , unStride , oneStride , toLinearIndexStride , strideStart , strideSize ) where import Control.DeepSeq import Data.Massiv.Core.Index.Class -- | Stride provides a way to ignore elements of an array if an index is divisible by a -- corresponding value in a stride. So, for a @Stride (i :. j)@ only elements with indices will be -- kept around: -- -- @ -- ( 0 :. 0) ( 0 :. j) ( 0 :. 2j) ( 0 :. 3j) ... -- ( i :. 0) ( i :. j) ( i :. 2j) ( i :. 3j) ... -- (2i :. 0) (2i :. j) (2i :. 2j) (2i :. 3j) ... -- ... -- @ -- -- Only positive strides make sense, so `Stride` pattern synonym constructor will prevent a user -- from creating a stride with negative or zero values, thus promoting safety of the library. -- -- ====__Examples:__ -- -- * Default and minimal stride of @`Stride` (`pureIndex` 1)@ will have no affect and all elements -- will kept. -- -- * If stride is @`Stride` 2@, then every 2nd element (i.e. with index 1, 3, 5, ..) will be skipped -- and only elemnts with indices divisible by 2 will be kept around. -- -- * In case of two dimensions, if what you want is to keep all rows divisible by 5, but keep every -- column intact then you'd use @Stride (5 :. 1)@. -- #if __GLASGOW_HASKELL__ >= 800 newtype Stride ix = SafeStride ix deriving (Eq, Ord, NFData) #else -- There is an issue in GHC 7.10 which prevents from placing `Index` constraint on a pattern. data Stride ix where SafeStride :: Index ix => ix -> Stride ix deriving instance Eq ix => Eq (Stride ix) deriving instance Ord ix => Ord (Stride ix) instance NFData ix => NFData (Stride ix) where rnf (SafeStride ix) = rnf ix #endif instance Index ix => Show (Stride ix) where show (SafeStride ix) = "Stride (" ++ show ix ++ ")" -- | A safe bidirectional pattern synonym for `Stride` construction that will make sure stride -- elements are always positive. pattern Stride :: Index ix => ix -> Stride ix pattern Stride ix <- SafeStride ix where Stride ix = SafeStride (liftIndex (max 1) ix) -- | Just a helper function for unwrapping `Stride`. unStride :: Stride ix -> ix unStride (SafeStride ix) = ix {-# INLINE unStride #-} -- | Adjust strating index according to the stride strideStart :: Index ix => Stride ix -> ix -> ix strideStart (SafeStride stride) ix = liftIndex2 (+) ix (liftIndex2 mod (liftIndex2 subtract (liftIndex2 mod ix stride) stride) stride) {-# INLINE strideStart #-} -- | Adjust size according to the stride. strideSize :: Index ix => Stride ix -> ix -> ix strideSize (SafeStride stride) sz = liftIndex (+ 1) $ liftIndex2 div (liftIndex (subtract 1) sz) stride {-# INLINE strideSize #-} -- | Compute an index with stride using the original size and index toLinearIndexStride :: Index ix => Stride ix -- ^ Stride -> ix -- ^ Size -> ix -- ^ Index -> Int toLinearIndexStride (SafeStride stride) sz ix = toLinearIndex sz (liftIndex2 div ix stride) {-# INLINE toLinearIndexStride #-} -- | A default stride of @1@, where all elements are kept oneStride :: Index ix => Stride ix oneStride = SafeStride (pureIndex 1) {-# INLINE oneStride #-}