Skip to content

Commit

Permalink
Merge pull request haskell#580 from kazu-yamamoto/gracefully-graceful…
Browse files Browse the repository at this point in the history
…-shutdown

making gracefulClose more graceful
  • Loading branch information
kazu-yamamoto authored Jun 12, 2024
2 parents e8310db + 107486a commit a2e111e
Showing 1 changed file with 10 additions and 10 deletions.
20 changes: 10 additions & 10 deletions Network/Socket/Shutdown.hs
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ module Network.Socket.Shutdown (
import qualified Control.Exception as E
import Foreign.Marshal.Alloc (mallocBytes, free)

import Control.Concurrent (threadDelay)
import Control.Concurrent (threadDelay, yield)

import Network.Socket.Buffer
import Network.Socket.Imports
Expand Down Expand Up @@ -47,31 +47,31 @@ foreign import CALLCONV unsafe "shutdown"
--
-- Since: 3.1.1.0
gracefulClose :: Socket -> Int -> IO ()
gracefulClose s tmout = sendRecvFIN `E.finally` close s
gracefulClose s tmout0 = sendRecvFIN `E.finally` close s
where
sendRecvFIN = do
-- Sending TCP FIN.
ex <- E.try $ shutdown s ShutdownSend
case ex of
Left (E.SomeException _) -> return ()
Right () -> do
-- Giving CPU time to other threads hoping that
-- FIN arrives meanwhile.
yield
-- Waiting TCP FIN.
E.bracket (mallocBytes bufSize) free recvEOFloop
-- milliseconds. Taken from BSD fast clock value.
clock = 200
recvEOFloop buf = loop 0
recvEOFloop buf = loop 1 0
where
loop delay = do
loop delay tmout = do
-- We don't check the (positive) length.
-- In normal case, it's 0. That is, only FIN is received.
-- In error cases, data is available. But there is no
-- application which can read it. So, let's stop receiving
-- to prevent attacks.
r <- recvBufNoWait s buf bufSize
let delay' = delay + clock
when (r == -1 && delay' < tmout) $ do
threadDelay (clock * 1000)
loop delay'
when (r == -1 && tmout < tmout0) $ do
threadDelay (delay * 1000)
loop (delay * 2) (tmout + delay)
-- Don't use 4092 here. The GHC runtime takes the global lock
-- if the length is over 3276 bytes in 32bit or 3272 bytes in 64bit.
bufSize = 1024

0 comments on commit a2e111e

Please sign in to comment.