[Haskell-cafe] Slow mvar when compiled with threaded

Branimir Maksimovic branimir.maksimovic at gmail.com
Tue Jan 7 18:55:54 UTC 2014


I have test network client, something like apache bench tool.
It uses mvars to synchronize and everything is ok when
compiled without -threaded.
real    0m2.995s
user    0m0.601s
sys    0m2.391s

With -threaded compile option result is following:
real    0m18.196s
user    0m2.054s
sys    0m3.313s

Seems that program is sleeping most of the time for some
reason. I can't explain behavior as it seems that
program is ok. It starts `concurrency` threads which
wait on mvar to process next task.

Program follows:

{-# Language OverloadedStrings #-}
import System.CPUTime
import System.IO
--import System.IO.Error
import Network.Socket hiding(recv)
import Network.Socket.ByteString
import System.Environment
import Control.Concurrent
import Control.Exception

main = do
     n <- getArgs
     let (host,port,conc,reqs) = parse n
     putStrLn $ "Connecting to " ++ host ++ " " ++ port
     s <- getAddrInfo Nothing (Just host) (Just port)
     let servAddr = head s
     begin <- getCPUTime
     process servAddr conc reqs
     end <- getCPUTime
     let diff = (fromIntegral (end - begin))/(10^12) :: Double
     putStrLn $ show (round (fromIntegral reqs / diff)) ++ " r/s"

parse [h,p,conc,reqs] = (h,p,read conc::Int,read reqs::Int)
parse _ = error "usage client host port concurrency requests"

process servAddr conc reqs = do
     let niter = if reqs >= conc then conc else reqs
     putStrLn $ "loop " ++ show niter
     mvars <- initThreads niter []
     putStrLn $ "Initialized " ++ show niter
     let loop n (m:mvs) f | n>0  = do
             flag <- isEmptyMVar m
             if f > length mvars then putStrLn "busy" else return ()
             if flag || f > length mvars
                 then do
                     putMVar m ()
                     loop (n-1) mvs 0
                 else loop n mvs (f+1)
             | otherwise = return ()
         loop n [] f = if n>0 then loop n mvars f else return ()
     putStrLn $ "length " ++ show (length mvars)
     loop (reqs-niter) mvars 0
     where
         initThreads niter vars | niter > 0 =  do
             mvar <- newMVar ()
             forkIO $ process mvar
             initThreads (niter-1) (mvar:vars)
             | otherwise = return vars
         process mvar = do
             sock <- socket (addrFamily servAddr) Stream defaultProtocol
             connect sock (addrAddress servAddr)
             sendAll sock "Hello World!\n"
             buf <- recv sock 1024
             close sock
             takeMVar mvar
             process mvar



More information about the Haskell-Cafe mailing list