{- Handles for lock pools. - - Copyright 2015-2020 Joey Hess - - License: BSD-2-clause -} {-# LANGUAGE CPP #-} module Utility.LockPool.LockHandle ( LockHandle, FileLockOps(..), dropLock, #ifndef mingw32_HOST_OS checkSaneLock, #endif makeLockHandle, tryMakeLockHandle, ) where import qualified Utility.LockPool.STM as P import Utility.LockPool.STM (LockFile) import Utility.DebugLocks import Control.Concurrent.STM import Control.Monad.Catch import Control.Monad.IO.Class (liftIO, MonadIO) import Control.Applicative import Prelude data LockHandle = LockHandle P.LockHandle FileLockOps data FileLockOps = FileLockOps { fDropLock :: IO () #ifndef mingw32_HOST_OS , fCheckSaneLock :: LockFile -> IO Bool #endif } dropLock :: LockHandle -> IO () dropLock (LockHandle ph _) = P.releaseLock ph #ifndef mingw32_HOST_OS checkSaneLock :: LockFile -> LockHandle -> IO Bool checkSaneLock lockfile (LockHandle _ flo) = fCheckSaneLock flo lockfile #endif -- Take a lock, by first updating the lock pool, and then taking the file -- lock. If taking the file lock fails for any reason, take care to -- release the lock in the lock pool. makeLockHandle :: (MonadIO m, MonadMask m) => P.LockPool -> LockFile -> (P.LockPool -> LockFile -> STM (P.LockHandle, P.FirstLock)) -> (LockFile -> P.FirstLock -> m FileLockOps) -> m LockHandle makeLockHandle pool file pa fa = bracketOnError setup cleanup go where setup = debugLocks $ liftIO $ atomically (pa pool file) cleanup (ph, _) = debugLocks $ liftIO $ P.releaseLock ph go (ph, firstlock) = liftIO . mkLockHandle ph =<< fa file firstlock tryMakeLockHandle :: (MonadIO m, MonadMask m) => P.LockPool -> LockFile -> (P.LockPool -> LockFile -> STM (Maybe (P.LockHandle, P.FirstLock))) -> (LockFile -> P.FirstLock -> m (Maybe FileLockOps)) -> m (Maybe LockHandle) tryMakeLockHandle pool file pa fa = bracketOnError setup cleanup go where setup = liftIO $ atomically (pa pool file) cleanup Nothing = return () cleanup (Just (ph, _)) = liftIO $ P.releaseLock ph go Nothing = return Nothing go (Just (ph, firstlock)) = do mfo <- fa file firstlock case mfo of Nothing -> do liftIO $ cleanup (Just (ph, firstlock)) return Nothing Just fo -> liftIO $ Just <$> mkLockHandle ph fo mkLockHandle :: P.LockHandle -> FileLockOps -> IO LockHandle mkLockHandle ph fo = do atomically $ P.registerCloseLockFile ph (fDropLock fo) return $ LockHandle ph fo