module Sound.Iteratee.Codecs.Common (
stringRead4
,joinMaybe
,convFunc
)
where
import Sound.Iteratee.Base
import Data.Iteratee as I
import qualified Data.Vector.Storable as V
import Foreign
import Control.Monad (replicateM, liftM)
import Control.Monad.CatchIO
import Data.Char (chr)
import Data.Int.Int24
import Data.Word.Word24
import Data.ListLike.Vector.Storable ()
be :: IO Bool
be = fmap (==1) $ with (1 :: Word16) (\p -> peekByteOff p 1 :: IO Word8)
stringRead4 :: MonadCatchIO m => Iteratee (V.Vector Word8) m String
stringRead4 = (liftM . map) (chr . fromIntegral) $ replicateM 4 I.head
unroll8 :: (MonadCatchIO m) => Iteratee (V.Vector Word8) m (Maybe (V.Vector Word8))
unroll8 = liftI step
where
step (I.Chunk buf)
| V.null buf = liftI step
| otherwise = idone (Just buf) (I.Chunk V.empty)
step stream = idone Nothing stream
unroller :: (Storable a, MonadCatchIO m) =>
Int
-> Iteratee (V.Vector Word8) m (Maybe (V.Vector a))
unroller wSize = liftI step
where
step (I.Chunk buf)
| V.null buf = liftI step
| otherwise = do
let len = V.length buf
if len < wSize
then liftI $ step' buf
else if len `rem` wSize == 0
then do
let buf' = convert_vec buf
idone (Just buf') (I.Chunk V.empty)
else let newLen = (len `div` wSize) * wSize
h = convert_vec $ V.take newLen buf
t = V.drop newLen buf
in do
idone (Just h) (I.Chunk t)
step stream = idone Nothing stream
step' i (I.Chunk buf)
| V.null buf = liftI (step' i)
| otherwise = do
let l = V.length buf
iLen = V.length i
newbuf = i V.++ buf
if l+iLen < wSize then liftI (step' newbuf)
else do
let newLen = V.length newbuf
newLen' = (newLen `div` wSize) * wSize
h = convert_vec $ V.take newLen' newbuf
t = V.drop newLen' newbuf
idone (Just h) (I.Chunk t)
step' _i stream = idone Nothing stream
convert_vec = hostToLE
hostToLE :: forall a. Storable a => V.Vector Word8 -> V.Vector a
hostToLE vec = let be' = unsafePerformIO be in if be'
then error "wrong endian-ness. Ask the maintainer to implement hostToLE"
else let (ptr, offset,len) = V.unsafeToForeignPtr vec
in V.unsafeFromForeignPtr (castForeignPtr ptr)
offset
(len `quot` sizeOf (undefined :: a))
w16 :: Word16
w16 = 0
w24 :: Word24
w24 = 0
w32 :: Word32
w32 = 0
convFunc :: (MonadCatchIO m) =>
AudioFormat
-> Iteratee (V.Vector Word8) m (V.Vector Double)
convFunc (AudioFormat _nc _sr 8) = do
mbuf <- unroll8
return $ maybe (error "error in convFunc") (V.map
(normalize 8 . (fromIntegral :: Word8 -> Int8))) mbuf
convFunc (AudioFormat _nc _sr 16) = do
mbuf <- unroller (sizeOf w16)
return $ maybe (error "error in convFunc") (V.map
(normalize 16 . (fromIntegral :: Word16 -> Int16))) mbuf
convFunc (AudioFormat _nc _sr 24) = do
mbuf <- unroller (sizeOf w24)
return $ maybe (error "error in convFunc") (V.map
(normalize 24 . (fromIntegral :: Word24 -> Int24))) mbuf
convFunc (AudioFormat _nc _sr 32) = do
mbuf <- unroller (sizeOf w32)
return $ maybe (error "error in convFunc") (V.map
(normalize 32 . (fromIntegral :: Word32 -> Int32))) mbuf
convFunc _ = I.throwErr (I.iterStrExc "Invalid wave bit depth")
joinMaybe :: Maybe [a] -> [a]
joinMaybe Nothing = []
joinMaybe (Just a) = a
normalize :: Integral a => BitDepth -> a -> Double
normalize 8 = \a -> let m = 1 / 128 in m * (fromIntegral a 128)
normalize bd = \a -> if a > 0
then fromIntegral a * mPos
else fromIntegral a * mNeg
where
mPos = 1/ (fromIntegral (1 `shiftL` fromIntegral (bd 1) :: Integer) 1)
mNeg = 1/ fromIntegral (1 `shiftL` fromIntegral (bd 1) :: Integer)