{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE MultiWayIf #-}
{-# LANGUAGE TypeApplications #-}
module Data.ByteString.Base64.Internal.W64.Loop
( innerLoop
, decodeLoop
, lenientLoop
) where
import Data.Bits
import Data.ByteString.Internal
import Data.ByteString.Base64.Internal.Utils
import qualified Data.ByteString.Base64.Internal.W16.Loop as W16
import Data.Text (Text)
import Foreign.ForeignPtr
import Foreign.Ptr
import Foreign.Storable
import GHC.Word
innerLoop
:: Ptr Word16
-> Ptr Word64
-> Ptr Word64
-> Ptr Word64
-> (Ptr Word8 -> Ptr Word8 -> IO ByteString)
-> IO ByteString
innerLoop !etable !sptr !dptr !end finish = go sptr dptr
where
go !src !dst
| plusPtr src 7 >= end =
W16.innerLoop etable (castPtr src) (castPtr dst) (castPtr end) finish
| otherwise = do
!t <- peekWord64BE src
let !a = (unsafeShiftR t 52) .&. 0xfff
!b = (unsafeShiftR t 40) .&. 0xfff
!c = (unsafeShiftR t 28) .&. 0xfff
!d = (unsafeShiftR t 16) .&. 0xfff
!w <- w64_16 <$> peekElemOff etable (fromIntegral a)
!x <- w64_16 <$> peekElemOff etable (fromIntegral b)
!y <- w64_16 <$> peekElemOff etable (fromIntegral c)
!z <- w64_16 <$> peekElemOff etable (fromIntegral d)
let !xx = w
.|. (unsafeShiftL x 16)
.|. (unsafeShiftL y 32)
.|. (unsafeShiftL z 48)
poke dst (fromIntegral xx)
go (plusPtr src 6) (plusPtr dst 8)
{-# inline innerLoop #-}
decodeLoop
:: Ptr Word8
-> Ptr Word8
-> Ptr Word8
-> Ptr Word8
-> (Ptr Word8 -> Ptr Word8 -> IO (Either Text ByteString))
-> IO (Either Text ByteString)
decodeLoop = W16.decodeLoop
{-# inline decodeLoop #-}
lenientLoop
:: Ptr Word8
-> Ptr Word8
-> Ptr Word8
-> Ptr Word8
-> ForeignPtr Word8
-> IO ByteString
lenientLoop = W16.lenientLoop