Skip to content

Commit

Permalink
Pull out terminateProcess wrapper function
Browse files Browse the repository at this point in the history
  • Loading branch information
tomjaguarpaw committed May 23, 2023
1 parent e6667db commit bbf07a9
Showing 1 changed file with 21 additions and 18 deletions.
39 changes: 21 additions & 18 deletions src/System/Process/Typed.hs
Original file line number Diff line number Diff line change
Expand Up @@ -256,24 +256,7 @@ startProcess pConfig'@ProcessConfig {..} = liftIO $ do
-- Process didn't exit yet, let's terminate it and
-- then call waitForProcess ourselves
Left _ -> do
eres <- try $ P.terminateProcess pHandle
case eres of
Left e
-- On Windows, with the single-threaded runtime, it
-- seems that if a process has already exited, the
-- call to terminateProcess will fail with a
-- permission denied error. To work around this, we
-- catch this exception and then immediately
-- waitForProcess. There's a chance that there may be
-- other reasons for this permission error to appear,
-- in which case this code may allow us to wait too
-- long for a child process instead of erroring out.
-- Recommendation: always use the multi-threaded
-- runtime!
| isPermissionError e && not multiThreadedRuntime && isWindows ->
pure ()
| otherwise -> throwIO e
Right () -> pure ()
terminateProcess pHandle
ec <- P.waitForProcess pHandle
success <- atomically $ tryPutTMVar pExitCode ec
evaluate $ assert success ()
Expand All @@ -282,6 +265,26 @@ startProcess pConfig'@ProcessConfig {..} = liftIO $ do
where
pConfig = clearStreams pConfig'

terminateProcess pHandle = do
eres <- try $ P.terminateProcess pHandle
case eres of
Left e
-- On Windows, with the single-threaded runtime, it
-- seems that if a process has already exited, the
-- call to terminateProcess will fail with a
-- permission denied error. To work around this, we
-- catch this exception and then immediately
-- waitForProcess. There's a chance that there may be
-- other reasons for this permission error to appear,
-- in which case this code may allow us to wait too
-- long for a child process instead of erroring out.
-- Recommendation: always use the multi-threaded
-- runtime!
| isPermissionError e && not multiThreadedRuntime && isWindows ->
pure ()
| otherwise -> throwIO e
Right () -> pure ()

foreign import ccall unsafe "rtsSupportsBoundThreads"
multiThreadedRuntime :: Bool

Expand Down

0 comments on commit bbf07a9

Please sign in to comment.