{-# language BangPatterns #-}
{-# language LambdaCase #-}
{-# language DataKinds #-}
{-# language MagicHash #-}
module Socket.Stream.Uninterruptible.ByteString
( send
, receiveExactly
, receiveOnce
, receiveBetween
) where
import Data.Bytes.Types (UnmanagedBytes(..))
import Data.ByteString.Unsafe (unsafeUseAsCStringLen)
import Data.ByteString.Internal (ByteString(PS))
import Data.Primitive.Addr (Addr(..))
import Data.Primitive (MutableByteArray(..))
import Data.Bytes.Types (MutableBytes(..))
import GHC.Exts (Ptr(Ptr),RealWorld,byteArrayContents#,unsafeCoerce#,proxy#)
import GHC.ForeignPtr (ForeignPtr(ForeignPtr),ForeignPtrContents(PlainPtr))
import Socket.Stream (Connection,ReceiveException,SendException)
import Socket (Interruptibility(Uninterruptible))
import qualified Data.Primitive as PM
import qualified Socket.Stream.Uninterruptible.Addr.Send as Send
import qualified Socket.Stream.Uninterruptible.MutableBytes.Receive as Receive
send ::
Connection
-> ByteString
-> IO (Either (SendException 'Uninterruptible) ())
{-# inline send #-}
send !conn !bs = unsafeUseAsCStringLen bs
(\(Ptr addr#,len) -> Send.send proxy# conn (UnmanagedBytes (Addr addr#) len))
receiveExactly ::
Connection
-> Int
-> IO (Either (ReceiveException 'Uninterruptible) ByteString)
{-# inline receiveExactly #-}
receiveExactly !conn !n = do
!marr <- PM.newPinnedByteArray n
Receive.receiveExactly proxy# conn (MutableBytes marr 0 n) >>= \case
Left err -> pure (Left err)
Right _ -> pure $! Right $! fromManaged marr 0 n
receiveOnce ::
Connection
-> Int
-> IO (Either (ReceiveException 'Uninterruptible) ByteString)
{-# inline receiveOnce #-}
receiveOnce !conn !n = do
!marr0 <- PM.newPinnedByteArray n
Receive.receiveOnce proxy# conn (MutableBytes marr0 0 n) >>= \case
Left err -> pure (Left err)
Right sz -> do
marr1 <- PM.resizeMutableByteArray marr0 sz
pure $! Right $! fromManaged marr1 0 sz
receiveBetween ::
Connection
-> Int
-> Int
-> IO (Either (ReceiveException 'Uninterruptible) ByteString)
{-# inline receiveBetween #-}
receiveBetween !conn !minLen !maxLen = do
!marr0 <- PM.newPinnedByteArray maxLen
Receive.receiveBetween proxy# conn (MutableBytes marr0 0 maxLen) minLen >>= \case
Left err -> pure (Left err)
Right sz -> do
marr1 <- PM.resizeMutableByteArray marr0 sz
pure $! Right $! fromManaged marr1 0 sz
fromManaged :: MutableByteArray RealWorld -> Int -> Int -> ByteString
{-# inline fromManaged #-}
fromManaged (MutableByteArray marr#) off len =
PS (ForeignPtr (byteArrayContents# (unsafeCoerce# marr#)) (PlainPtr marr#)) off len