{-# LINE 1 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
module System.Socket.Address.SockAddrIn6
{-# LINE 2 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
  ( SockAddrIn6 (..)
  ) where

import Data.Word
import qualified Data.ByteString as BS
import qualified Data.ByteString.Unsafe as BS

import Foreign.Ptr
import Foreign.Storable
import Foreign.Marshal.Utils

import System.Socket.Address
import System.Socket.Internal.FFI


{-# LINE 17 "src/System/Socket/Address/SockAddrIn6.hsc" #-}

{-# LINE 18 "src/System/Socket/Address/SockAddrIn6.hsc" #-}

{-# LINE 19 "src/System/Socket/Address/SockAddrIn6.hsc" #-}

{-# LINE 20 "src/System/Socket/Address/SockAddrIn6.hsc" #-}

{-# LINE 21 "src/System/Socket/Address/SockAddrIn6.hsc" #-}

instance Address SockAddrIn6 where
  addressFamilyNumber _ = (10)
{-# LINE 24 "src/System/Socket/Address/SockAddrIn6.hsc" #-}

data SockAddrIn6
   = SockAddrIn6
     { sin6Port      :: Word16
     , sin6Flowinfo  :: Word32
     , sin6Addr      :: BS.ByteString
     , sin6ScopeId   :: Word32
     } deriving (Eq, Ord)

instance Show SockAddrIn6 where
  show (SockAddrIn6 p _ addr _) = '"':'[':(tail $ t $ BS.unpack addr)
    where
      t []       = ']':':':(show p ++ "\"")
      t [x]      = g x 0 (']':':':(show p) ++ "\"")
      t (x:y:xs) = g x y (t xs)
      g x y s    = let (a,b) = quotRem x 16
                       (c,d) = quotRem y 16
                   in  ':':(h a):(h b):(h c):(h d):s
      h :: Word8 -> Char
      h 0  = '0'
      h 1  = '1'
      h 2  = '2'
      h 3  = '3'
      h 4  = '4'
      h 5  = '5'
      h 6  = '6'
      h 7  = '7'
      h 8  = '8'
      h 9  = '9'
      h 10 = 'a'
      h 11 = 'b'
      h 12 = 'c'
      h 13 = 'd'
      h 14 = 'e'
      h 15 = 'f'
      h  _ = '_'

instance Storable SockAddrIn6 where
  sizeOf    _ = ((28))
{-# LINE 63 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
  alignment _ = (4)
{-# LINE 64 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
  peek ptr    = do
    f   <- peek              (sin6_flowinfo ptr) :: IO Word32
    ph  <- peekByteOff       (sin6_port ptr)  0  :: IO Word8
    pl  <- peekByteOff       (sin6_port ptr)  1  :: IO Word8
    a   <- BS.packCStringLen (sin6_addr ptr, 16) :: IO BS.ByteString
    s   <- peek              (sin6_scope_id ptr) :: IO Word32
    return (SockAddrIn6 (fromIntegral ph * 256 + fromIntegral pl) f a s)
    where
      sin6_flowinfo = ((\hsc_ptr -> hsc_ptr `plusPtr` 4))
{-# LINE 73 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_scope_id = ((\hsc_ptr -> hsc_ptr `plusPtr` 24))
{-# LINE 74 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_port     = ((\hsc_ptr -> hsc_ptr `plusPtr` 2))
{-# LINE 75 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_addr     = ((\hsc_ptr -> hsc_ptr `plusPtr` 0)) . ((\hsc_ptr -> hsc_ptr `plusPtr` 8))
{-# LINE 76 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
  poke ptr (SockAddrIn6 p f a s) = do
    c_memset ptr 0 (28)
{-# LINE 78 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
    poke        (sin6_family   ptr) ((10) :: Word16)
{-# LINE 79 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
    poke        (sin6_flowinfo ptr) f
    poke        (sin6_scope_id ptr) s
    pokeByteOff (sin6_port     ptr)  0 (fromIntegral $ rem (quot p 256) 256 :: Word8)
    pokeByteOff (sin6_port     ptr)  1 (fromIntegral $ rem       p      256 :: Word8)
    BS.unsafeUseAsCString a $ \a'-> do
      copyBytes (sin6_addr ptr) a' (min 16 $ BS.length a)-- copyBytes dest from count
    where
      sin6_family   = ((\hsc_ptr -> hsc_ptr `plusPtr` 0))
{-# LINE 87 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_flowinfo = ((\hsc_ptr -> hsc_ptr `plusPtr` 4))
{-# LINE 88 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_scope_id = ((\hsc_ptr -> hsc_ptr `plusPtr` 24))
{-# LINE 89 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_port     = ((\hsc_ptr -> hsc_ptr `plusPtr` 2))
{-# LINE 90 "src/System/Socket/Address/SockAddrIn6.hsc" #-}
      sin6_addr     = ((\hsc_ptr -> hsc_ptr `plusPtr` 0)) . ((\hsc_ptr -> hsc_ptr `plusPtr` 8))