module System.Socket.Internal.Socket (
Socket (..)
, GetSockOpt (..)
, getSockOptBool
, SetSockOpt (..)
, setSockOptBool
, SO_ERROR (..)
, SO_REUSEADDR (..)
) where
import Control.Concurrent.MVar
import Control.Exception
import Control.Monad
import Control.Applicative
import Foreign.Ptr
import Foreign.Storable
import Foreign.C.Types
import Foreign.Marshal.Alloc
import System.Posix.Types
import System.Socket.Internal.Platform
import System.Socket.Internal.Exception
newtype Socket f t p
= Socket (MVar Fd)
class GetSockOpt o where
getSockOpt :: Socket f t p -> IO o
class SetSockOpt o where
setSockOpt :: Socket f t p -> o -> IO ()
data SO_ERROR
= SO_ERROR SocketException
deriving (Eq, Ord, Show)
instance GetSockOpt SO_ERROR where
getSockOpt s =
SO_ERROR . SocketException <$> getSockOptCInt s (1) (4)
data SO_REUSEADDR
= SO_REUSEADDR Bool
deriving (Eq, Ord, Show)
instance GetSockOpt SO_REUSEADDR where
getSockOpt s =
SO_REUSEADDR <$> getSockOptBool s (1) (2)
instance SetSockOpt SO_REUSEADDR where
setSockOpt s (SO_REUSEADDR o) =
setSockOptBool s (1) (2) o
setSockOptBool :: Socket f t p -> CInt -> CInt -> Bool -> IO ()
setSockOptBool (Socket mfd) level name value = do
withMVar mfd $ \fd->
alloca $ \vPtr-> do
if value
then poke vPtr 1
else poke vPtr 0
i <- c_setsockopt fd level name
(vPtr :: Ptr CInt)
(fromIntegral $ sizeOf (undefined :: CInt))
when (i < 0) $ do
c_get_last_socket_error >>= throwIO
getSockOptBool :: Socket f t p -> CInt -> CInt -> IO Bool
getSockOptBool (Socket mfd) level name = do
withMVar mfd $ \fd->
alloca $ \vPtr-> do
alloca $ \lPtr-> do
i <- c_getsockopt fd level name
(vPtr :: Ptr CInt)
(lPtr :: Ptr CInt)
if i < 0 then do
c_get_last_socket_error >>= throwIO
else do
v <- peek vPtr
return (v == 1)
setSockOptInt :: Socket f t p -> CInt -> CInt -> Int -> IO ()
setSockOptInt (Socket mfd) level name value = do
withMVar mfd $ \fd->
alloca $ \vPtr-> do
poke vPtr (fromIntegral value :: CInt)
i <- c_setsockopt fd level name
(vPtr :: Ptr CInt)
(fromIntegral $ sizeOf (undefined :: CInt))
when (i < 0) $ do
c_get_last_socket_error >>= throwIO
getSockOptInt :: Socket f t p -> CInt -> CInt -> IO Int
getSockOptInt (Socket mfd) level name = do
withMVar mfd $ \fd->
alloca $ \vPtr-> do
alloca $ \lPtr-> do
i <- c_getsockopt fd level name
(vPtr :: Ptr CInt)
(lPtr :: Ptr CInt)
if i < 0 then do
c_get_last_socket_error >>= throwIO
else do
v <- peek vPtr
return (fromIntegral v)
setSockOptCInt :: Socket f t p -> CInt -> CInt -> CInt -> IO ()
setSockOptCInt (Socket mfd) level name value = do
withMVar mfd $ \fd->
alloca $ \vPtr-> do
poke vPtr value
i <- c_setsockopt fd level name
(vPtr :: Ptr CInt)
(fromIntegral $ sizeOf (undefined :: CInt))
when (i < 0) $ do
c_get_last_socket_error >>= throwIO
getSockOptCInt :: Socket f t p -> CInt -> CInt -> IO CInt
getSockOptCInt (Socket mfd) level name = do
withMVar mfd $ \fd->
alloca $ \vPtr-> do
alloca $ \lPtr-> do
i <- c_getsockopt fd level name
(vPtr :: Ptr CInt)
(lPtr :: Ptr CInt)
if i < 0 then do
c_get_last_socket_error >>= throwIO
else do
peek vPtr