From 831532b14142c0087fe98cd0cc20cfa046217249 Mon Sep 17 00:00:00 2001 From: Ben Gamari Date: Fri, 5 Oct 2012 15:03:20 -0400 Subject: [PATCH 1/5] Admit base-4.6 in Build-Depends --- git-annex.cabal | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/git-annex.cabal b/git-annex.cabal index 048cad3d3c..518c63d8d8 100644 --- a/git-annex.cabal +++ b/git-annex.cabal @@ -48,7 +48,7 @@ Executable git-annex Build-Depends: MissingH, hslogger, directory, filepath, unix, containers, utf8-string, network, mtl, bytestring, old-locale, time, pcre-light, extensible-exceptions, dataenc, SHA, process, json, HTTP, - base == 4.5.*, monad-control, transformers-base, lifted-base, + base >= 4.5 && < 4.7, monad-control, transformers-base, lifted-base, IfElse, text, QuickCheck >= 2.1, bloomfilter, edit-distance, process -- Need to list these because they're generated from .hsc files. Other-Modules: Utility.Touch Utility.Mounts @@ -95,7 +95,7 @@ Test-Suite test Build-Depends: testpack, HUnit, MissingH, hslogger, directory, filepath, unix, containers, utf8-string, network, mtl, bytestring, old-locale, time, pcre-light, extensible-exceptions, dataenc, SHA, process, json, HTTP, - base == 4.5.*, monad-control, transformers-base, lifted-base, + base >= 4.5 && < 4.7, monad-control, transformers-base, lifted-base, IfElse, text, QuickCheck >= 2.1, bloomfilter, edit-distance, process Other-Modules: Utility.Touch Include-Dirs: Utility From cff451b37c45fdafcfa6976a59842ec6938ac2c5 Mon Sep 17 00:00:00 2001 From: Ben Gamari Date: Fri, 5 Oct 2012 17:02:51 -0400 Subject: [PATCH 2/5] TransferSlots: Use SafeSemaphore's MSemN instead of QSemN from base As described in the documentation, QSemN is unsafe for a variety of reasons. --- Assistant/TransferSlots.hs | 13 +++++++------ git-annex.cabal | 5 +++-- 2 files changed, 10 insertions(+), 8 deletions(-) diff --git a/Assistant/TransferSlots.hs b/Assistant/TransferSlots.hs index 9e9156ad9f..c41b1d28c1 100644 --- a/Assistant/TransferSlots.hs +++ b/Assistant/TransferSlots.hs @@ -16,9 +16,10 @@ import Logs.Transfer import qualified Control.Exception as E import Control.Concurrent +import qualified Control.Concurrent.MSemN as MSemN import Data.Typeable -type TransferSlots = QSemN +type TransferSlots = MSemN.MSemN Int {- A special exception that can be thrown to pause or resume a transfer, while - keeping its slot in use. -} @@ -39,21 +40,21 @@ numSlots :: Int numSlots = 1 newTransferSlots :: IO TransferSlots -newTransferSlots = newQSemN numSlots +newTransferSlots = MSemN.new numSlots {- Waits until a transfer slot becomes available, then runs a - TransferGenerator, and then runs the transfer action in its own thread. -} inTransferSlot :: TransferSlotRunner inTransferSlot dstatus s gen = do - waitQSemN s 1 + MSemN.wait s 1 runTransferThread dstatus s =<< gen {- Runs a TransferGenerator, and its transfer action, - without waiting for a slot to become available. -} inImmediateTransferSlot :: TransferSlotRunner inImmediateTransferSlot dstatus s gen = do - signalQSemN s (-1) + MSemN.signal s (-1) runTransferThread dstatus s =<< gen {- Runs a transfer action, in an already allocated transfer slot. @@ -67,7 +68,7 @@ inImmediateTransferSlot dstatus s gen = do - then rerunning the action. -} runTransferThread :: DaemonStatusHandle -> TransferSlots -> Maybe (Transfer, TransferInfo, IO ()) -> IO () -runTransferThread _ s Nothing = signalQSemN s 1 +runTransferThread _ s Nothing = MSemN.signal s 1 runTransferThread dstatus s (Just (t, info, a)) = do tid <- forkIO go updateTransferInfo dstatus t $ info { transferTid = Just tid } @@ -86,4 +87,4 @@ runTransferThread dstatus s (Just (t, info, a)) = do Just ResumeTransfer -> go _ -> done _ -> done - done = signalQSemN s 1 + done = MSemN.signal s 1 diff --git a/git-annex.cabal b/git-annex.cabal index 518c63d8d8..d760d4adb6 100644 --- a/git-annex.cabal +++ b/git-annex.cabal @@ -1,5 +1,5 @@ Name: git-annex -Version: 3.20121001 +Version: 3.20121002 Cabal-Version: >= 1.8 License: GPL Maintainer: Joey Hess @@ -49,7 +49,8 @@ Executable git-annex unix, containers, utf8-string, network, mtl, bytestring, old-locale, time, pcre-light, extensible-exceptions, dataenc, SHA, process, json, HTTP, base >= 4.5 && < 4.7, monad-control, transformers-base, lifted-base, - IfElse, text, QuickCheck >= 2.1, bloomfilter, edit-distance, process + IfElse, text, QuickCheck >= 2.1, bloomfilter, edit-distance, process, + SafeSemaphore -- Need to list these because they're generated from .hsc files. Other-Modules: Utility.Touch Utility.Mounts Include-Dirs: Utility From 179aeeaacc2748ef869864415ad3f172129cd369 Mon Sep 17 00:00:00 2001 From: Ben Gamari Date: Fri, 5 Oct 2012 17:03:58 -0400 Subject: [PATCH 3/5] Remote/Git: Use SampleVar from SafeSemaphore instead of base SampleVars from base are unsafe --- Remote/Git.hs | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/Remote/Git.hs b/Remote/Git.hs index df97db7a6d..6d95d5de0f 100644 --- a/Remote/Git.hs +++ b/Remote/Git.hs @@ -38,6 +38,7 @@ import Types.Key import qualified Fields import Control.Concurrent +import Control.Concurrent.MSampleVar import System.Process (std_in, std_err) remote :: RemoteType @@ -273,9 +274,9 @@ copyFromRemote r key file dest : maybe [] (\f -> [(Fields.associatedFile, f)]) file Just (cmd, params) <- git_annex_shell r "transferinfo" [Param $ key2file key] fields - v <- liftIO $ newEmptySampleVar + v <- liftIO $ newEmptySV tid <- liftIO $ forkIO $ void $ tryIO $ do - bytes <- readSampleVar v + bytes <- readSV v p <- createProcess $ (proc cmd (toCommand params)) { std_in = CreatePipe @@ -288,8 +289,8 @@ copyFromRemote r key file dest hFlush h send bytes forever $ - send =<< readSampleVar v - let feeder = writeSampleVar v + send =<< readSV v + let feeder = writeSV v bracketIO noop (const $ tryIO $ killThread tid) (a feeder) copyFromRemoteCheap :: Git.Repo -> Key -> FilePath -> Annex Bool From 7fc4ee0dee5a72a2760980172a057a9a20132d3b Mon Sep 17 00:00:00 2001 From: Ben Gamari Date: Fri, 5 Oct 2012 17:04:46 -0400 Subject: [PATCH 4/5] NotificationBroadcaster: Use SampleVars from SafeSemaphores instead of base SampleVars from base are unsafe --- Utility/NotificationBroadcaster.hs | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Utility/NotificationBroadcaster.hs b/Utility/NotificationBroadcaster.hs index accc35fe18..4bbbc544aa 100644 --- a/Utility/NotificationBroadcaster.hs +++ b/Utility/NotificationBroadcaster.hs @@ -26,10 +26,10 @@ module Utility.NotificationBroadcaster ( import Common import Control.Concurrent.STM -import Control.Concurrent.SampleVar +import Control.Concurrent.MSampleVar -{- One SampleVar per client. The TMVar is never empty, so never blocks. -} -type NotificationBroadcaster = TMVar [SampleVar ()] +{- One MSampleVar per client. The TMVar is never empty, so never blocks. -} +type NotificationBroadcaster = TMVar [MSampleVar ()] newtype NotificationId = NotificationId Int deriving (Read, Show, Eq, Ord) @@ -47,7 +47,7 @@ newNotificationHandle b = NotificationHandle <*> addclient where addclient = do - s <- newEmptySampleVar + s <- newEmptySV atomically $ do l <- takeTMVar b putTMVar b $ l ++ [s] @@ -67,11 +67,11 @@ sendNotification b = do l <- atomically $ readTMVar b mapM_ notify l where - notify s = writeSampleVar s () + notify s = writeSV s () {- Used by a client to block until a new notification is available since - the last time it tried. -} waitNotification :: NotificationHandle -> IO () waitNotification (NotificationHandle b (NotificationId i)) = do l <- atomically $ readTMVar b - readSampleVar (l !! i) + readSV (l !! i) From 33a2af36f221e76c245648f9e06bd79269bd35f8 Mon Sep 17 00:00:00 2001 From: Joey Hess Date: Fri, 5 Oct 2012 17:50:17 -0400 Subject: [PATCH 5/5] Depend on and use the Haskell SafeSemaphore library, which provides exception-safe versions of SampleVar and QSemN. Thanks, Ben Gamari for an excellent patch set. --- debian/changelog | 3 +++ debian/control | 1 + doc/install/fromscratch.mdwn | 1 + 3 files changed, 5 insertions(+) diff --git a/debian/changelog b/debian/changelog index 560447e10b..77fcf7bd5d 100644 --- a/debian/changelog +++ b/debian/changelog @@ -10,6 +10,9 @@ git-annex (3.20121002) UNRELEASED; urgency=low of the repository configuration settings stored in the git-annex branch. * Only build-depend on libghc-clientsession-dev on arches that will have the webapp. + * Depend on and use the Haskell SafeSemaphore library, which provides + exception-safe versions of SampleVar and QSemN. + Thanks, Ben Gamari for an excellent patch set. -- Joey Hess Mon, 01 Oct 2012 15:09:49 -0400 diff --git a/debian/control b/debian/control index 505ea6ebb2..24464f980f 100644 --- a/debian/control +++ b/debian/control @@ -39,6 +39,7 @@ Build-Depends: libghc-crypto-api-dev, libghc-network-multicast-dev, libghc-network-info-dev, + libghc-safesemaphore-dev, ikiwiki, perlmagick, git, diff --git a/doc/install/fromscratch.mdwn b/doc/install/fromscratch.mdwn index 795443b71d..4410a59b9f 100644 --- a/doc/install/fromscratch.mdwn +++ b/doc/install/fromscratch.mdwn @@ -18,6 +18,7 @@ quite a lot. * [bloomfilter](http://hackage.haskell.org/package/bloomfilter) * [edit-distance](http://hackage.haskell.org/package/edit-distance) * [hS3](http://hackage.haskell.org/package/hS3) (optional) + * [SafeSemaphore](http://hackage.haskell.org/package/SafeSemaphore) * Optional haskell stuff, used by the [[assistant]] and its webapp (edit Makefile to disable) * [stm](http://hackage.haskell.org/package/stm) (version 2.3 or newer)