-- Deactivate warning because it is painful to refactor functions with two -- rebinded-do with different bind functions. Such as in the 'run' -- function. Which is a good argument for having support for F#-style builders. {-# LANGUAGE DerivingVia #-} {-# LANGUAGE GADTs #-} {-# LANGUAGE LinearTypes #-} {-# LANGUAGE QualifiedDo #-} {-# LANGUAGE RankNTypes #-} {-# OPTIONS_GHC -fno-warn-name-shadowing #-} {-# OPTIONS_HADDOCK hide #-} module System.IO.Resource.Linear.Internal where import Control.Exception (finally, mask, onException) import qualified Control.Functor.Linear as Control import qualified Control.Monad as Ur (fmap) import Data.Coerce import qualified Data.Functor.Linear as Data import Data.IORef (IORef) import qualified Data.IORef as System import Data.IntMap.Strict (IntMap) import qualified Data.IntMap.Strict as IntMap import Data.Monoid (Ap (..)) import Data.Text (Text) import qualified Data.Text.IO as Text import Prelude.Linear hiding (IO) import qualified System.IO as System import qualified System.IO.Linear as Linear import qualified Prelude -- XXX: This would be better as a multiplicity-parametric relative monad, but -- until we have multiplicity polymorphism, we use a linear monad. newtype ReleaseMap = ReleaseMap (IntMap (Linear.IO ())) -- | The resource-aware I/O monad. This monad guarantees that acquired resources -- are always released. newtype RIO a = RIO (IORef ReleaseMap -> Linear.IO a) deriving (Data.Functor, Data.Applicative) via (Control.Data RIO) deriving (Semigroup, Monoid) via (Ap RIO a) unRIO :: RIO a %1 -> IORef ReleaseMap -> Linear.IO a unRIO (RIO action) = action -- | Take a @RIO@ computation with a value @a@ that is not linearly bound and -- make it a "System.IO" computation. run :: RIO (Ur a) -> System.IO a run (RIO action) = do rrm <- System.newIORef (ReleaseMap IntMap.empty) mask ( \restore -> onException (restore (Linear.withLinearIO (action rrm))) ( do -- release stray resources ReleaseMap releaseMap <- System.readIORef rrm safeRelease $ Ur.fmap snd $ IntMap.toList releaseMap ) ) where -- Remarks: resources are guaranteed to be released on non-exceptional -- return. So, contrary to a standard bracket/ResourceT implementation, we -- only release exceptions in the release map upon exception. safeRelease :: [Linear.IO ()] -> System.IO () safeRelease [] = Prelude.return () safeRelease (finalizer : fs) = Linear.withLinearIO (moveLinearIO finalizer) `finally` safeRelease fs -- Should be just an application of a linear `(<$>)`. moveLinearIO :: Movable a => Linear.IO a %1 -> Linear.IO (Ur a) moveLinearIO action' = Control.do result <- action' Control.return $ move result -- | Should not be applied to a function that acquires or releases resources. unsafeFromSystemIO :: System.IO a %1 -> RIO a unsafeFromSystemIO action = RIO (\_ -> Linear.fromSystemIO action) -- monad instance Control.Functor RIO where fmap f (RIO action) = RIO $ \releaseMap -> Control.fmap f (action releaseMap) instance Control.Applicative RIO where pure a = RIO $ \_releaseMap -> Control.pure a (<*>) = Control.ap instance Control.Monad RIO where x >>= f = RIO $ \releaseMap -> Control.do a <- unRIO x releaseMap unRIO (f a) releaseMap x >> y = RIO $ \releaseMap -> Control.do unRIO x releaseMap unRIO y releaseMap -- files -- Remark: Handle needs to be private otherwise `Data.Coerce.coerce` could wreak -- Havoc on the abstraction. But we could provide a smart constructor/view to -- unsafely convert to file handles in order for the Handle API to be -- extensible. newtype Handle = Handle (UnsafeResource System.Handle) -- | See 'System.IO.openFile' openFile :: FilePath -> System.IOMode -> RIO Handle openFile path mode = Control.do h <- unsafeAcquire (Linear.fromSystemIOU $ System.openFile path mode) (\h -> Linear.fromSystemIO $ System.hClose h) Control.return $ Handle h hClose :: Handle %1 -> RIO () hClose (Handle h) = unsafeRelease h hIsEOF :: Handle %1 -> RIO (Ur Bool, Handle) hIsEOF = coerce (unsafeFromSystemIOResource System.hIsEOF) hGetChar :: Handle %1 -> RIO (Ur Char, Handle) hGetChar = coerce (unsafeFromSystemIOResource System.hGetChar) hPutChar :: Handle %1 -> Char -> RIO Handle hPutChar h c = flipHPutChar c h -- needs a multiplicity polymorphic flip where flipHPutChar :: Char -> Handle %1 -> RIO Handle flipHPutChar c = coerce (unsafeFromSystemIOResource_ (\h' -> System.hPutChar h' c)) hGetLine :: Handle %1 -> RIO (Ur Text, Handle) hGetLine = coerce (unsafeFromSystemIOResource Text.hGetLine) hPutStr :: Handle %1 -> Text -> RIO Handle hPutStr h s = flipHPutStr s h -- needs a multiplicity polymorphic flip where flipHPutStr :: Text -> Handle %1 -> RIO Handle flipHPutStr s = coerce (unsafeFromSystemIOResource_ (\h' -> Text.hPutStr h' s)) hPutStrLn :: Handle %1 -> Text -> RIO Handle hPutStrLn h s = flipHPutStrLn s h -- needs a multiplicity polymorphic flip where flipHPutStrLn :: Text -> Handle %1 -> RIO Handle flipHPutStrLn s = coerce (unsafeFromSystemIOResource_ (\h' -> Text.hPutStrLn h' s)) -- new-resources -- | The type of system resources. To create and use resources, you need to -- use the API since the constructor is not released. data UnsafeResource a where UnsafeResource :: Int -> a -> UnsafeResource a -- Note that both components are unrestricted. -- | Given an unsafe resource, release it with the linear IO action provided -- when the resrouce was acquired. unsafeRelease :: UnsafeResource a %1 -> RIO () unsafeRelease (UnsafeResource key _) = RIO (\st -> Linear.mask_ (releaseWith key st)) where releaseWith key rrm = Control.do Ur (ReleaseMap releaseMap) <- Linear.readIORef rrm () <- releaseMap IntMap.! key Linear.writeIORef rrm (ReleaseMap (IntMap.delete key releaseMap)) -- | Given a resource in the "System.IO.Linear.IO" monad, and -- given a function to release that resource, provides that resource in -- the @RIO@ monad. For example, releasing a @Handle@ from "System.IO" -- would be done with @fromSystemIO hClose@. Because this release function -- is an input, and could be wrong, this function is unsafe. unsafeAcquire :: Linear.IO (Ur a) -> (a -> Linear.IO ()) -> RIO (UnsafeResource a) unsafeAcquire acquire release = RIO $ \rrm -> Linear.mask_ ( Control.do Ur resource <- acquire Ur (ReleaseMap releaseMap) <- Linear.readIORef rrm () <- Linear.writeIORef rrm ( ReleaseMap (IntMap.insert (releaseKey releaseMap) (release resource) releaseMap) ) Control.return $ UnsafeResource (releaseKey releaseMap) resource ) where releaseKey releaseMap = case IntMap.null releaseMap of True -> 0 False -> fst (IntMap.findMax releaseMap) + 1 -- | Given a "System.IO" computation on an unsafe resource, -- lift it to @RIO@ computaton on the acquired resource. -- That is function of type @a -> IO b@ turns into a function of type -- @UnsafeResource a %1-> RIO (Ur b)@ -- along with threading the @UnsafeResource a@. -- -- Note that the result @b@ can be used non-linearly. unsafeFromSystemIOResource :: (a -> System.IO b) -> (UnsafeResource a %1 -> RIO (Ur b, UnsafeResource a)) unsafeFromSystemIOResource action (UnsafeResource key resource) = unsafeFromSystemIO ( do c <- action resource Prelude.return (Ur c, UnsafeResource key resource) ) unsafeFromSystemIOResource_ :: (a -> System.IO ()) -> (UnsafeResource a %1 -> RIO (UnsafeResource a)) unsafeFromSystemIOResource_ action resource = Control.do (Ur _, resource) <- unsafeFromSystemIOResource action resource Control.return resource