summaryrefslogtreecommitdiff
path: root/Utility/Kqueue.hs
diff options
context:
space:
mode:
authorGravatar Joey Hess <joeyh@fischer.debian.org>2012-06-19 04:52:55 +0000
committerGravatar Joey Hess <joeyh@fischer.debian.org>2012-06-19 04:52:55 +0000
commit03b9341356c8d4eabfec5864957a4e49e7fcac67 (patch)
tree080f68b49b38e24f79eea14fcc2971559e68b374 /Utility/Kqueue.hs
parenta5cceb7d4ff83b11da95cac204e99d1bfdbaecc9 (diff)
fix scheduling
Handle kevent interruptions in the haskell code, so it can yield to other threads
Diffstat (limited to 'Utility/Kqueue.hs')
-rw-r--r--Utility/Kqueue.hs17
1 files changed, 12 insertions, 5 deletions
diff --git a/Utility/Kqueue.hs b/Utility/Kqueue.hs
index da43a2d86..1f65b2dba 100644
--- a/Utility/Kqueue.hs
+++ b/Utility/Kqueue.hs
@@ -24,11 +24,13 @@ import Utility.Types.DirWatcher
import System.Posix.Types
import Foreign.C.Types
+import Foreign.C.Error
import Foreign.Ptr
import Foreign.Marshal
import qualified Data.Map as M
import qualified Data.Set as S
import qualified System.Posix.Files as Files
+import Control.Concurrent
data Change
= Deleted FilePath
@@ -146,9 +148,14 @@ stopKqueue (Kqueue h _ _) = closeFd h
waitChange :: Kqueue -> IO (Kqueue, [Change])
waitChange kq@(Kqueue h dirmap _) = do
changedfd <- c_waitchange_kqueue h
- case M.lookup changedfd dirmap of
- Nothing -> return (kq, [])
- Just info -> handleChange kq changedfd info
+ if changedfd == -1
+ then ifM ((==) eINTR <$> getErrno)
+ (yield >> waitChange kq, nochange)
+ else case M.lookup changedfd dirmap of
+ Nothing -> nochange
+ Just info -> handleChange kq changedfd info
+ where
+ nochange = return (kq, [])
{- The kqueue interface does not tell what type of change took place in
- the directory; it could be an added file, a deleted file, a renamed
@@ -212,9 +219,9 @@ runHooks kq hooks = do
| Files.isSymbolicLink s = callhook addSymlinkHook (Just s) change
| Files.isDirectory s = print $ "TODO: recursive directory add: " ++ show change
| Files.isRegularFile s = callhook addHook (Just s) change
- | otherwise = noop
+ | otherwise = print "not a file??"
callhook h s change = case h hooks of
- Nothing -> noop
+ Nothing -> print "missing hook??"
Just a -> a (changedFile change) s
withstatus change a = maybe noop (a change) =<<
(catchMaybeIO (getSymbolicLinkStatus (changedFile change)))