module Data.Array.Vector.Algorithms.Radix (sort, Radix(..)) where
import Control.Monad
import Control.Monad.ST
import Data.Array.Vector
import Data.Array.Vector.Algorithms.Common
import Data.Bits
import Data.Int
import Data.Word
import Foreign.Storable
class UA e => Radix e where
passes :: e -> Int
size :: e -> Int
radix :: Int -> e -> Int
instance Radix Int where
passes _ = sizeOf (undefined :: Int)
size _ = 256
radix 0 e = e .&. 255
radix i e
| i == passes e 1 = radix' (e + minBound)
| otherwise = radix' e
where radix' e = (e `shiftR` (i `shiftL` 3)) .&. 255
instance Radix Int8 where
passes _ = 1
size _ = 256
radix _ e = fromIntegral e + 128
instance Radix Int16 where
passes _ = 2
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix 1 e = fromIntegral (((e + minBound) `shiftR` 8) .&. 255)
instance Radix Int32 where
passes _ = 4
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix 1 e = fromIntegral ((e `shiftR` 8) .&. 255)
radix 2 e = fromIntegral ((e `shiftR` 16) .&. 255)
radix 3 e = fromIntegral (((e + minBound) `shiftR` 24) .&. 255)
instance Radix Int64 where
passes _ = 8
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix 1 e = fromIntegral ((e `shiftR` 8) .&. 255)
radix 2 e = fromIntegral ((e `shiftR` 16) .&. 255)
radix 3 e = fromIntegral ((e `shiftR` 24) .&. 255)
radix 4 e = fromIntegral ((e `shiftR` 32) .&. 255)
radix 5 e = fromIntegral ((e `shiftR` 40) .&. 255)
radix 6 e = fromIntegral ((e `shiftR` 48) .&. 255)
radix 7 e = fromIntegral (((e + minBound) `shiftR` 56) .&. 255)
instance Radix Word where
passes _ = sizeOf (undefined :: Word)
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix i e = fromIntegral ((e `shiftR` (i `shiftL` 3)) .&. 255)
instance Radix Word8 where
passes _ = 1
size _ = 256
radix _ = fromIntegral
instance Radix Word16 where
passes _ = 2
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix 1 e = fromIntegral ((e `shiftR` 8) .&. 255)
instance Radix Word32 where
passes _ = 4
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix 1 e = fromIntegral ((e `shiftR` 8) .&. 255)
radix 2 e = fromIntegral ((e `shiftR` 16) .&. 255)
radix 3 e = fromIntegral ((e `shiftR` 24) .&. 255)
instance Radix Word64 where
passes _ = 8
size _ = 256
radix 0 e = fromIntegral (e .&. 255)
radix 1 e = fromIntegral ((e `shiftR` 8) .&. 255)
radix 2 e = fromIntegral ((e `shiftR` 16) .&. 255)
radix 3 e = fromIntegral ((e `shiftR` 24) .&. 255)
radix 4 e = fromIntegral ((e `shiftR` 32) .&. 255)
radix 5 e = fromIntegral ((e `shiftR` 40) .&. 255)
radix 6 e = fromIntegral ((e `shiftR` 48) .&. 255)
radix 7 e = fromIntegral ((e `shiftR` 56) .&. 255)
sort :: forall e s. Radix e => MUArr e s -> ST s ()
sort arr = do
tmp <- newMU len
count <- newMU (size e)
prefix <- newMU (size e)
go False arr tmp count prefix 0
where
len = lengthMU arr
e :: e
e = undefined
go !swap src dst count prefix k
| k < passes e = do zero 0 count
countLoop 0 k src count
writeMU prefix 0 0
prefixLoop 1 0 count prefix
moveLoop 0 k src dst prefix
go (not swap) dst src count prefix (k+1)
| otherwise = when swap (mcopyMU src dst 0 0 len)
zero i a
| i < size e = writeMU a i 0 >> zero (i+1) a
| otherwise = return ()
countLoop i k src count
| i < len = readMU src i >>= inc count . radix k >> countLoop (i+1) k src count
| otherwise = return ()
prefixLoop i pi count prefix
| i < size e = do ci <- readMU count (i1)
let pi' = pi + ci
writeMU prefix i pi'
prefixLoop (i+1) pi' count prefix
| otherwise = return ()
moveLoop i k src dst prefix
| i < len = do srci <- readMU src i
pf <- inc prefix (radix k srci)
writeMU dst pf srci
moveLoop (i+1) k src dst prefix
| otherwise = return ()
inc :: MUArr Int s -> Int -> ST s Int
inc arr i = readMU arr i >>= \e -> writeMU arr i (e+1) >> return e