{-# LANGUAGE TemplateHaskell, ForeignFunctionInterface, InterruptibleFFI #-}
{-# LANGUAGE ViewPatterns, PatternSynonyms, TupleSections #-}
module NgxExport (
ContentHandlerResult
,UnsafeContentHandlerResult
,ngxExportSS
,ngxExportSSS
,ngxExportSLS
,ngxExportBS
,ngxExportBSS
,ngxExportBLS
,ngxExportYY
,ngxExportBY
,ngxExportIOYY
,ngxExportAsyncIOYY
,ngxExportAsyncOnReqBody
,ngxExportServiceIOYY
,ngxExportHandler
,ngxExportDefHandler
,ngxExportUnsafeHandler
,ngxExportAsyncHandler
,ngxExportAsyncHandlerOnReqBody
,ngxExportServiceHook
,ngxCyclePtr
,ngxUpstreamMainConfPtr
,ngxCachedTimePtr
,Foreign.C.CInt (..)
,Foreign.C.CUInt (..)
) where
import Language.Haskell.TH
import Foreign.C
import Foreign.Ptr
import Foreign.StablePtr
import Foreign.Storable
import Foreign.Marshal.Alloc
import Foreign.Marshal.Utils
import Data.IORef
import System.IO.Unsafe
import System.IO.Error
import System.Posix.IO
import System.Posix.Types
import System.Posix.Signals hiding (Handler)
import System.Posix.Internals
import Control.Monad
import Control.Monad.Loops
import Control.DeepSeq
import qualified Control.Exception as E
import Control.Exception hiding (Handler)
import GHC.IO.Exception (ioe_errno)
import GHC.IO.Device (SeekMode (..))
import Control.Concurrent.Async
import qualified Data.ByteString as B
import qualified Data.ByteString.Unsafe as B
import qualified Data.ByteString.Lazy as L
import qualified Data.ByteString.Lazy.Char8 as C8L
import Data.Binary.Put
import Paths_ngx_export (version)
import Data.Version
#if MIN_VERSION_template_haskell(2,11,0)
#define EXTRA_WILDCARD_BEFORE_CON _
#else
#define EXTRA_WILDCARD_BEFORE_CON
#endif
#if MIN_TOOL_VERSION_ghc(8,0,1)
pattern I :: (Num i, Integral a) => i -> a
#endif
pattern I i <- (fromIntegral -> i)
#if MIN_TOOL_VERSION_ghc(8,2,1)
{-# COMPLETE I :: CInt #-}
#endif
#if MIN_TOOL_VERSION_ghc(8,0,1)
pattern PtrLen :: Num l => Ptr s -> l -> (Ptr s, Int)
#endif
pattern PtrLen s l <- (s, I l)
#if MIN_TOOL_VERSION_ghc(8,0,1)
pattern ToBool :: (Num i, Eq i) => Bool -> i
#endif
pattern ToBool i <- (toBool -> i)
#if MIN_TOOL_VERSION_ghc(8,2,1)
{-# COMPLETE ToBool :: CUInt #-}
#endif
type ContentHandlerResult = (L.ByteString, B.ByteString, Int)
type UnsafeContentHandlerResult = (B.ByteString, B.ByteString, Int)
data NgxExport = SS (String -> String)
| SSS (String -> String -> String)
| SLS ([String] -> String)
| BS (String -> Bool)
| BSS (String -> String -> Bool)
| BLS ([String] -> Bool)
| YY (B.ByteString -> L.ByteString)
| BY (B.ByteString -> Bool)
| IOYY (B.ByteString -> Bool -> IO L.ByteString)
| IOYYY (L.ByteString -> B.ByteString ->
IO L.ByteString)
| Handler (B.ByteString -> ContentHandlerResult)
| UnsafeHandler (B.ByteString -> UnsafeContentHandlerResult)
| AsyncHandler (B.ByteString -> IO ContentHandlerResult)
| AsyncHandlerRB (L.ByteString -> B.ByteString ->
IO ContentHandlerResult)
data NgxExportDisambiguation = Unambiguous
| YYSync
| YYDefHandler
| IOYYSync
| IOYYAsync
do
TyConI (DataD _ _ _ EXTRA_WILDCARD_BEFORE_CON tCs _) <-
reify ''NgxExport
TyConI (DataD _ _ _ EXTRA_WILDCARD_BEFORE_CON aCs _) <-
reify ''NgxExportDisambiguation
let tName = mkName "exportType"
aName = mkName "exportTypeAmbiguity"
tCons = map (\(NormalC con [(_, typ)]) -> (con, typ)) tCs
aCons = map (\(NormalC con []) -> con) aCs
sequence $
[sigD tName [t|NgxExport -> IO CInt|]
,funD tName $
map (\(fst -> c, i) ->
clause [conP c [wildP]] (normalB [|return i|]) []
) (zip tCons [1 ..] :: [((Name, Type), Int)])
,sigD aName [t|NgxExportDisambiguation -> IO CInt|]
,funD aName $
map (\(c, i) ->
clause [conP c []] (normalB [|return i|]) []
) (zip aCons [0 ..] :: [(Name, Int)])
]
++
map (\(c, t) -> tySynD (mkName $ nameBase c) [] $ return t) tCons
ngxExport' :: (Name -> Q Exp) ->
Name -> Name -> Name -> Q Type -> Name -> Q [Dec]
ngxExport' m e a h t f = sequence
[sigD nameFt typeFt
,funD nameFt $ body [|exportType $cefVar|]
,ForeignD . ExportF CCall ftName nameFt <$> typeFt
,sigD nameFta typeFta
,funD nameFta $ body [|exportTypeAmbiguity $(conE a)|]
,ForeignD . ExportF CCall ftaName nameFta <$> typeFta
,sigD nameF t
,funD nameF $ body [|$(varE h) $efVar|]
,ForeignD . ExportF CCall fName nameF <$> t
]
where efVar = m f
cefVar = conE e `appE` efVar
fName = "ngx_hs_" ++ nameBase f
nameF = mkName fName
ftName = "type_" ++ fName
nameFt = mkName ftName
typeFt = [t|IO CInt|]
ftaName = "ambiguity_" ++ fName
nameFta = mkName ftaName
typeFta = [t|IO CInt|]
body b = [clause [] (normalB b) []]
ngxExport :: Name -> Name -> Name -> Q Type -> Name -> Q [Dec]
ngxExport = ngxExport' varE
ngxExportC :: Name -> Name -> Name -> Q Type -> Name -> Q [Dec]
ngxExportC = ngxExport' $ infixE (Just $ varE 'const) (varE '(.)) . Just . varE
ngxExportSS :: Name -> Q [Dec]
ngxExportSS =
ngxExport 'SS 'Unambiguous 'sS
[t|CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportSSS :: Name -> Q [Dec]
ngxExportSSS =
ngxExport 'SSS 'Unambiguous 'sSS
[t|CString -> CInt -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportSLS :: Name -> Q [Dec]
ngxExportSLS =
ngxExport 'SLS 'Unambiguous 'sLS
[t|Ptr NgxStrType -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportBS :: Name -> Q [Dec]
ngxExportBS =
ngxExport 'BS 'Unambiguous 'bS
[t|CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportBSS :: Name -> Q [Dec]
ngxExportBSS =
ngxExport 'BSS 'Unambiguous 'bSS
[t|CString -> CInt -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportBLS :: Name -> Q [Dec]
ngxExportBLS =
ngxExport 'BLS 'Unambiguous 'bLS
[t|Ptr NgxStrType -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportYY :: Name -> Q [Dec]
ngxExportYY =
ngxExport 'YY 'YYSync 'yY
[t|CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt|]
ngxExportBY :: Name -> Q [Dec]
ngxExportBY =
ngxExport 'BY 'Unambiguous 'bY
[t|CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt|]
ngxExportIOYY :: Name -> Q [Dec]
ngxExportIOYY =
ngxExportC 'IOYY 'IOYYSync 'ioyY
[t|CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt|]
ngxExportAsyncIOYY :: Name -> Q [Dec]
ngxExportAsyncIOYY =
ngxExportC 'IOYY 'IOYYAsync 'asyncIOYY
[t|CString -> CInt -> CInt -> CInt -> Ptr CUInt -> CUInt -> CUInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))|]
ngxExportAsyncOnReqBody :: Name -> Q [Dec]
ngxExportAsyncOnReqBody =
ngxExport 'IOYYY 'Unambiguous 'asyncIOYYY
[t|Ptr NgxStrType -> Ptr NgxStrType -> CInt ->
CString -> CInt -> CInt -> CUInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))|]
ngxExportServiceIOYY :: Name -> Q [Dec]
ngxExportServiceIOYY =
ngxExport 'IOYY 'IOYYAsync 'asyncIOYY
[t|CString -> CInt -> CInt -> CInt -> Ptr CUInt -> CUInt -> CUInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))|]
ngxExportHandler :: Name -> Q [Dec]
ngxExportHandler =
ngxExport 'Handler 'Unambiguous 'handler
[t|CString -> CInt -> Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CString -> Ptr CSize -> Ptr (StablePtr B.ByteString) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt|]
ngxExportDefHandler :: Name -> Q [Dec]
ngxExportDefHandler =
ngxExport 'YY 'YYDefHandler 'defHandler
[t|CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt -> Ptr CString ->
Ptr (StablePtr L.ByteString) -> IO CUInt|]
ngxExportUnsafeHandler :: Name -> Q [Dec]
ngxExportUnsafeHandler =
ngxExport 'UnsafeHandler 'Unambiguous 'unsafeHandler
[t|CString -> CInt -> Ptr CString -> Ptr CSize ->
Ptr CString -> Ptr CSize -> Ptr CInt -> IO CUInt|]
ngxExportAsyncHandler :: Name -> Q [Dec]
ngxExportAsyncHandler =
ngxExport 'AsyncHandler 'Unambiguous 'asyncHandler
[t|CString -> CInt -> CInt -> CUInt ->
Ptr CString -> Ptr CSize -> Ptr (StablePtr B.ByteString) -> Ptr CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))|]
ngxExportAsyncHandlerOnReqBody :: Name -> Q [Dec]
ngxExportAsyncHandlerOnReqBody =
ngxExport 'AsyncHandlerRB 'Unambiguous 'asyncHandlerRB
[t|Ptr NgxStrType -> Ptr NgxStrType -> CInt ->
CString -> CInt -> CInt -> CUInt ->
Ptr CString -> Ptr CSize -> Ptr (StablePtr B.ByteString) -> Ptr CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))|]
ngxExportServiceHook :: Name -> Q [Dec]
ngxExportServiceHook =
ngxExportC 'IOYY 'IOYYSync 'ioyYWithFree
[t|CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt|]
data NgxStrType = NgxStrType CSize CString
instance Storable NgxStrType where
alignment = const $ max (alignment (undefined :: CSize))
(alignment (undefined :: CString))
sizeOf = (2 *) . alignment
peek p = do
n <- peekByteOff p 0
s <- peekByteOff p $ alignment (undefined :: NgxStrType)
return $ NgxStrType n s
poke p x@(NgxStrType n s) = do
poke (castPtr p) n
poke (plusPtr p $ alignment x) s
data ServiceHookInterrupt = ServiceHookInterrupt
instance Exception ServiceHookInterrupt
instance Show ServiceHookInterrupt where
show = const "Service was interrupted by a service hook"
safeMallocBytes :: Int -> IO (Ptr a)
safeMallocBytes =
flip catchIOError (const $ return nullPtr) . mallocBytes
{-# INLINE safeMallocBytes #-}
safeNewCStringLen :: String -> IO CStringLen
safeNewCStringLen =
flip catchIOError (const $ return (nullPtr, -1)) . newCStringLen
{-# INLINE safeNewCStringLen #-}
peekNgxStringArrayLen :: (CStringLen -> IO a) -> Ptr NgxStrType -> Int -> IO [a]
peekNgxStringArrayLen f x = sequence .
foldr (\k ->
((peekElemOff x k >>= (\(NgxStrType (I m) y) -> f (y, m))) :)
) [] . flip take [0 ..]
{-# SPECIALIZE INLINE peekNgxStringArrayLen ::
(CStringLen -> IO String) -> Ptr NgxStrType -> Int ->
IO [String] #-}
{-# SPECIALIZE INLINE peekNgxStringArrayLen ::
(CStringLen -> IO B.ByteString) -> Ptr NgxStrType -> Int ->
IO [B.ByteString] #-}
peekNgxStringArrayLenLS :: Ptr NgxStrType -> Int -> IO [String]
peekNgxStringArrayLenLS =
peekNgxStringArrayLen peekCStringLen
peekNgxStringArrayLenY :: Ptr NgxStrType -> Int -> IO L.ByteString
peekNgxStringArrayLenY =
(fmap L.fromChunks .) . peekNgxStringArrayLen B.unsafePackCStringLen
pokeCStringLen :: Storable a => CString -> a -> Ptr CString -> Ptr a -> IO ()
pokeCStringLen x n p s = poke p x >> poke s n
{-# SPECIALIZE INLINE pokeCStringLen ::
CString -> CInt -> Ptr CString -> Ptr CInt -> IO () #-}
{-# SPECIALIZE INLINE pokeCStringLen ::
CString -> CSize -> Ptr CString -> Ptr CSize -> IO () #-}
toBuffers :: L.ByteString -> Ptr NgxStrType -> IO (Ptr NgxStrType, Int)
toBuffers (L.null -> True) _ =
return (nullPtr, 0)
toBuffers s p = do
let n = L.foldlChunks (const . succ) 0 s
if n == 1
then do
B.unsafeUseAsCStringLen (head $ L.toChunks s) $
\(x, I l) -> poke p $ NgxStrType l x
return (p, 1)
else do
t <- safeMallocBytes $ n * sizeOf (undefined :: NgxStrType)
if t == nullPtr
then return (nullPtr, -1)
else (t, ) <$>
L.foldlChunks
(\a c -> do
off <- a
B.unsafeUseAsCStringLen c $
\(x, I l) ->
pokeElemOff t off $ NgxStrType l x
return $ off + 1
) (return 0) s
pokeLazyByteString :: L.ByteString ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO ()
pokeLazyByteString s p pl spd = do
PtrLen t l <- peek p >>= toBuffers s
when (l /= 1) (poke p t) >> poke pl l
when (t /= nullPtr) $ newStablePtr s >>= poke spd
pokeContentTypeAndStatus :: B.ByteString ->
Ptr CString -> Ptr CSize -> Ptr CInt -> CInt -> IO ()
pokeContentTypeAndStatus ct pct plct pst st = do
PtrLen sct lct <- B.unsafeUseAsCStringLen ct return
pokeCStringLen sct lct pct plct >> poke pst st
peekRequestBodyChunks :: Ptr NgxStrType -> Ptr NgxStrType -> Int ->
IO L.ByteString
peekRequestBodyChunks tmpf b m =
if tmpf /= nullPtr
then do
c <- peek tmpf >>=
(\(NgxStrType (I l) s) -> peekCStringLen (s, l)) >>=
L.readFile
L.length c `seq` return c
else peekNgxStringArrayLenY b m
pokeAsyncHandlerData :: B.ByteString -> Ptr CString -> Ptr CSize ->
Ptr (StablePtr B.ByteString) -> Ptr CInt -> CInt -> IO ()
pokeAsyncHandlerData ct pct plct spct pst st = do
pokeContentTypeAndStatus ct pct plct pst st
newStablePtr ct >>= poke spct
safeHandler :: Ptr CString -> Ptr CInt -> IO CUInt -> IO CUInt
safeHandler p pl = handle $ \e -> do
PtrLen x l <- safeNewCStringLen $ show (e :: SomeException)
pokeCStringLen x l p pl
return 1
safeYYHandler :: IO (L.ByteString, (CUInt, Bool)) ->
IO (L.ByteString, (CUInt, Bool))
safeYYHandler = handle $ \e ->
return (C8L.pack $ show e,
(case fromException e of
Just ServiceHookInterrupt -> 2
_ -> 1
,case asyncExceptionFromException e of
Just ThreadKilled -> True
_ -> False
)
)
{-# INLINE safeYYHandler #-}
isEINTR :: IOError -> Bool
isEINTR = (Just ((\(Errno i) -> i) eINTR) ==) . ioe_errno
{-# INLINE isEINTR #-}
sS :: SS -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
sS f x (I n) p pl =
safeHandler p pl $ do
PtrLen s l <- f <$> peekCStringLen (x, n)
>>= newCStringLen
pokeCStringLen s l p pl
return 0
sSS :: SSS -> CString -> CInt -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
sSS f x (I n) y (I m) p pl =
safeHandler p pl $ do
PtrLen s l <- f <$> peekCStringLen (x, n)
<*> peekCStringLen (y, m)
>>= newCStringLen
pokeCStringLen s l p pl
return 0
sLS :: SLS -> Ptr NgxStrType -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
sLS f x (I n) p pl =
safeHandler p pl $ do
PtrLen s l <- f <$> peekNgxStringArrayLenLS x n
>>= newCStringLen
pokeCStringLen s l p pl
return 0
yY :: YY -> CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt
yY f x (I n) p pl spd = do
(s, (r, _)) <- safeYYHandler $ do
s <- f <$> B.unsafePackCStringLen (x, n)
fmap (, (0, False)) $ return $!! s
pokeLazyByteString s p pl spd
return r
ioyYCommon :: (CStringLen -> IO B.ByteString) ->
IOYY -> CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt
ioyYCommon pack f x (I n) p pl spd = do
(s, (r, _)) <- safeYYHandler $ do
s <- pack (x, n) >>= flip f False
fmap (, (0, False)) $ return $!! s
pokeLazyByteString s p pl spd
return r
ioyY :: IOYY -> CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt
ioyY = ioyYCommon B.unsafePackCStringLen
ioyYWithFree :: IOYY -> CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt
ioyYWithFree = ioyYCommon B.unsafePackMallocCStringLen
asyncIOFlag1b :: B.ByteString
asyncIOFlag1b = L.toStrict $ runPut $ putWord8 1
asyncIOFlag8b :: B.ByteString
asyncIOFlag8b = L.toStrict $ runPut $ putWord64host 1
asyncIOCommon :: IO (L.ByteString, Bool) ->
CInt -> Bool -> Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))
asyncIOCommon a (I fd) efd p pl pr spd =
async
(do
(s, (r, exiting)) <- safeYYHandler $ do
(s, exiting) <- a
fmap (, (0, exiting)) $ return $!! s
pokeLazyByteString s p pl spd
poke pr r
if exiting
then unless efd closeChannel
else uninterruptibleMask_ $
if efd
then writeFlag8b
else writeFlag1b >> closeChannel
) >>= newStablePtr
where writeBufN n s =
iterateUntilM (>= n)
(\w -> (w +) <$>
fdWriteBuf fd (plusPtr s $ fromIntegral w) (n - w)
`catchIOError`
(\e -> return $ if isEINTR e
then 0
else n + 1
)
) 0 >>= \w -> when (w > n) cleanupOnWriteError
writeFlag1b = B.unsafeUseAsCString asyncIOFlag1b $ writeBufN 1
writeFlag8b = B.unsafeUseAsCString asyncIOFlag8b $ writeBufN 8
closeChannel = closeFd fd `catchIOError` const (return ())
cleanupOnWriteError = return ()
asyncIOYY :: IOYY -> CString -> CInt ->
CInt -> CInt -> Ptr CUInt -> CUInt -> CUInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))
asyncIOYY f x (I n) fd (I fdlk) active (ToBool efd) (ToBool fstRun) =
asyncIOCommon
(do
exiting <- if fstRun && fdlk /= -1
then snd <$>
iterateUntil ((True ==) . fst)
(safeWaitToSetLock fdlk
(WriteLock, AbsoluteSeek, 0, 0) >>
return (True, False)
)
`catches`
[E.Handler $ return . (, False) . not . isEINTR
,E.Handler $ return . (True, ) . (== ThreadKilled)
]
else return False
if exiting
then return (L.empty, True)
else do
when fstRun $ poke active 1
x' <- B.unsafePackCStringLen (x, n)
(, False) <$> f x' fstRun
) fd efd
asyncIOYYY :: IOYYY -> Ptr NgxStrType -> Ptr NgxStrType -> CInt ->
CString -> CInt -> CInt -> CUInt -> Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))
asyncIOYYY f tmpf b (I m) x (I n) fd (ToBool efd) =
asyncIOCommon
(do
b' <- peekRequestBodyChunks tmpf b m
x' <- B.unsafePackCStringLen (x, n)
(, False) <$> f b' x'
) fd efd
asyncHandler :: AsyncHandler -> CString -> CInt ->
CInt -> CUInt ->
Ptr CString -> Ptr CSize -> Ptr (StablePtr B.ByteString) -> Ptr CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))
asyncHandler f x (I n) fd (ToBool efd) pct plct spct pst =
asyncIOCommon
(do
x' <- B.unsafePackCStringLen (x, n)
(s, ct, I st) <- f x'
(return $!! s) >> pokeAsyncHandlerData ct pct plct spct pst st
return (s, False)
) fd efd
asyncHandlerRB :: AsyncHandlerRB -> Ptr NgxStrType -> Ptr NgxStrType -> CInt ->
CString -> CInt -> CInt -> CUInt ->
Ptr CString -> Ptr CSize -> Ptr (StablePtr B.ByteString) -> Ptr CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CUInt -> Ptr (StablePtr L.ByteString) -> IO (StablePtr (Async ()))
asyncHandlerRB f tmpf b (I m) x (I n) fd (ToBool efd) pct plct spct pst =
asyncIOCommon
(do
b' <- peekRequestBodyChunks tmpf b m
x' <- B.unsafePackCStringLen (x, n)
(s, ct, I st) <- f b' x'
(return $!! s) >> pokeAsyncHandlerData ct pct plct spct pst st
return (s, False)
) fd efd
bS :: BS -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
bS f x (I n) p pl =
safeHandler p pl $ do
r <- fromBool . f <$> peekCStringLen (x, n)
pokeCStringLen nullPtr 0 p pl
return r
bSS :: BSS -> CString -> CInt -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
bSS f x (I n) y (I m) p pl =
safeHandler p pl $ do
r <- (fromBool .) . f <$> peekCStringLen (x, n)
<*> peekCStringLen (y, m)
pokeCStringLen nullPtr 0 p pl
return r
bLS :: BLS -> Ptr NgxStrType -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
bLS f x (I n) p pl =
safeHandler p pl $ do
r <- fromBool . f <$> peekNgxStringArrayLenLS x n
pokeCStringLen nullPtr 0 p pl
return r
bY :: BY -> CString -> CInt ->
Ptr CString -> Ptr CInt -> IO CUInt
bY f x (I n) p pl =
safeHandler p pl $ do
r <- fromBool . f <$> B.unsafePackCStringLen (x, n)
pokeCStringLen nullPtr 0 p pl
return r
handler :: Handler -> CString -> CInt -> Ptr (Ptr NgxStrType) -> Ptr CInt ->
Ptr CString -> Ptr CSize -> Ptr (StablePtr B.ByteString) -> Ptr CInt ->
Ptr (StablePtr L.ByteString) -> IO CUInt
handler f x (I n) p pl pct plct spct pst spd =
safeHandler pct pst $ do
(s, ct, I st) <- f <$> B.unsafePackCStringLen (x, n)
(return $!! s) >> pokeContentTypeAndStatus ct pct plct pst st
pokeLazyByteString s p pl spd
newStablePtr ct >>= poke spct
return 0
defHandler :: YY -> CString -> CInt ->
Ptr (Ptr NgxStrType) -> Ptr CInt -> Ptr CString ->
Ptr (StablePtr L.ByteString) -> IO CUInt
defHandler f x (I n) p pl pe spd =
safeHandler pe pl $ do
s <- f <$> B.unsafePackCStringLen (x, n)
pokeLazyByteString s p pl spd
return 0
unsafeHandler :: UnsafeHandler -> CString -> CInt -> Ptr CString -> Ptr CSize ->
Ptr CString -> Ptr CSize -> Ptr CInt -> IO CUInt
unsafeHandler f x (I n) p pl pct plct pst =
safeHandler pct pst $ do
(s, ct, I st) <- f <$> B.unsafePackCStringLen (x, n)
(return $!! s) >> pokeContentTypeAndStatus ct pct plct pst st
PtrLen t l <- B.unsafeUseAsCStringLen s return
pokeCStringLen t l p pl
return 0
foreign import ccall interruptible "HsBase.h fcntl"
safe_c_fcntl_lock :: CInt -> CInt -> Ptr CFLock -> IO CInt
mode2Int :: SeekMode -> CShort
mode2Int AbsoluteSeek = 0
mode2Int RelativeSeek = 1
mode2Int SeekFromEnd = 2
lockReq2Int :: LockRequest -> CShort
lockReq2Int ReadLock = 0
lockReq2Int WriteLock = 1
lockReq2Int Unlock = 2
allocaLock :: FileLock -> (Ptr CFLock -> IO a) -> IO a
allocaLock (lockreq, mode, start, len) io =
allocaBytes 32 $ \p -> do
(`pokeByteOff` 0) p (lockReq2Int lockreq)
(`pokeByteOff` 2) p (mode2Int mode)
(`pokeByteOff` 8) p start
(`pokeByteOff` 16) p len
io p
safeWaitToSetLock :: Fd -> FileLock -> IO ()
safeWaitToSetLock (Fd fd) lock = allocaLock lock $
\p_flock -> throwErrnoIfMinus1_ "safeWaitToSetLock" $
safe_c_fcntl_lock fd 7 p_flock
foreign export ccall ngxExportInstallSignalHandler :: IO ()
ngxExportInstallSignalHandler :: IO ()
ngxExportInstallSignalHandler = void $
installHandler keyboardSignal Ignore Nothing
foreign export ccall ngxExportTerminateTask ::
StablePtr (Async ()) -> IO ()
ngxExportTerminateTask ::
StablePtr (Async ()) -> IO ()
ngxExportTerminateTask = deRefStablePtr >=>
flip cancelWith ThreadKilled
foreign export ccall ngxExportServiceHookInterrupt ::
StablePtr (Async ()) -> IO ()
ngxExportServiceHookInterrupt ::
StablePtr (Async ()) -> IO ()
ngxExportServiceHookInterrupt = deRefStablePtr >=>
flip throwTo ServiceHookInterrupt . asyncThreadId
foreign export ccall ngxExportVersion ::
Ptr CInt -> CInt -> IO CInt
ngxExportVersion ::
Ptr CInt -> CInt -> IO CInt
ngxExportVersion x (I n) = fromIntegral <$>
foldM (\k (I v) -> pokeElemOff x k v >> return (k + 1)) 0
(take n $ versionBranch version)
ngxCyclePtr :: IO (Ptr ())
ngxCyclePtr = readIORef ngxCyclePtrStore
ngxUpstreamMainConfPtr :: IO (Ptr ())
ngxUpstreamMainConfPtr = readIORef ngxUpstreamMainConfPtrStore
ngxCachedTimePtr :: IO (Ptr (Ptr ()))
ngxCachedTimePtr = readIORef ngxCachedTimePtrStore
foreign export ccall ngxExportSetCyclePtr :: Ptr () -> IO ()
ngxExportSetCyclePtr :: Ptr () -> IO ()
ngxExportSetCyclePtr = writeIORef ngxCyclePtrStore
foreign export ccall ngxExportSetUpstreamMainConfPtr :: Ptr () -> IO ()
ngxExportSetUpstreamMainConfPtr :: Ptr () -> IO ()
ngxExportSetUpstreamMainConfPtr = writeIORef ngxUpstreamMainConfPtrStore
foreign export ccall ngxExportSetCachedTimePtr :: Ptr (Ptr ()) -> IO ()
ngxExportSetCachedTimePtr :: Ptr (Ptr ()) -> IO ()
ngxExportSetCachedTimePtr = writeIORef ngxCachedTimePtrStore
ngxCyclePtrStore :: IORef (Ptr ())
ngxCyclePtrStore = unsafePerformIO $ newIORef nullPtr
{-# NOINLINE ngxCyclePtrStore #-}
ngxUpstreamMainConfPtrStore :: IORef (Ptr ())
ngxUpstreamMainConfPtrStore = unsafePerformIO $ newIORef nullPtr
{-# NOINLINE ngxUpstreamMainConfPtrStore #-}
ngxCachedTimePtrStore :: IORef (Ptr (Ptr ()))
ngxCachedTimePtrStore = unsafePerformIO $ newIORef nullPtr
{-# NOINLINE ngxCachedTimePtrStore #-}