{-# LANGUAGE CApiFFI #-}
{-# LANGUAGE CPP #-}
module Network.DNS.FFI where
import Control.Concurrent.MVar
import Foreign.C
import Foreign.Marshal.Alloc
import Foreign.Ptr
import System.IO.Unsafe (unsafePerformIO)
#if !defined(USE_RES_NQUERY)
# error USE_RES_NQUERY not defined
#endif
{-# INLINE resIsReentrant #-}
resIsReentrant :: Bool
#if USE_RES_NQUERY
resIsReentrant = True
#else
resIsReentrant = False
#endif
#if !defined(SIZEOF_RES_STATE)
# error SIZEOF_RES_STATE not defined
#endif
#if USE_RES_NQUERY && (SIZEOF_RES_STATE <= 0)
# error broken invariant
#endif
{-# INLINE sizeOfResState #-}
sizeOfResState :: CSize
sizeOfResState = SIZEOF_RES_STATE
data CResState
{-# NOINLINE resolvLock #-}
resolvLock :: MVar ()
resolvLock = unsafePerformIO $ newMVar ()
withCResState :: (Ptr CResState -> IO a) -> IO a
withCResState act
| resIsReentrant = allocaBytes (fromIntegral sizeOfResState) $ \ptr -> do
_ <- c_memset ptr 0 sizeOfResState
act ptr
| otherwise = withMVar resolvLock $ \() -> act nullPtr
foreign import capi unsafe "string.h memset" c_memset :: Ptr a -> CInt -> CSize -> IO (Ptr a)
foreign import capi safe "hs_resolv.h hs_res_query" c_res_query :: Ptr CResState -> CString -> CInt -> CInt -> Ptr CChar -> CInt -> IO CInt
foreign import capi safe "hs_resolv.h hs_res_send" c_res_send :: Ptr CResState -> Ptr CChar -> CInt -> Ptr CChar -> CInt -> IO CInt
foreign import capi safe "hs_resolv.h res_opt_set_use_dnssec" c_res_opt_set_use_dnssec :: Ptr CResState -> IO CInt
foreign import capi safe "hs_resolv.h hs_res_mkquery" c_res_mkquery :: Ptr CResState -> CString -> CInt -> CInt -> Ptr CChar -> CInt -> IO CInt