module System.LibFuse3.Utils
(
testBitSet
,
unErrno, ioErrorToErrno, throwErrnoOf, tryErrno, tryErrno_, tryErrno', tryErrno_'
,
pread, pwrite, c_pread, c_pwrite
,
pokeCStringLen0
,
timeSpecToPOSIXTime
)
where
import Control.Exception (SomeException, try, tryJust)
import Data.Bits ((.&.), Bits)
import Data.ByteString (ByteString)
import Data.Ratio ((%))
import Data.Time.Clock.POSIX (POSIXTime)
import Foreign (Ptr, allocaBytes, copyArray, pokeElemOff)
import Foreign.C (CInt(CInt), CSize(CSize), CStringLen, Errno(Errno), eIO, eOK, errnoToIOError, getErrno, throwErrno, throwErrnoIfMinus1, withCStringLen)
import GHC.IO.Exception (IOException(IOError, ioe_errno))
import System.Clock (TimeSpec)
import System.Posix.Types (ByteCount, COff(COff), CSsize(CSsize), Fd(Fd), FileOffset)
import qualified Data.ByteString as B
import qualified Data.ByteString.Unsafe as BU
import qualified System.Clock as TimeSpec
try_ :: IO a -> IO (Either SomeException a)
try_ :: forall a. IO a -> IO (Either SomeException a)
try_ = forall e a. Exception e => IO a -> IO (Either e a)
try
unErrno :: Errno -> CInt
unErrno :: Errno -> CInt
unErrno (Errno CInt
errno) = CInt
errno
ioErrorToErrno :: IOError -> Maybe Errno
ioErrorToErrno :: IOError -> Maybe Errno
ioErrorToErrno IOError{ioe_errno :: IOError -> Maybe CInt
ioe_errno=Just CInt
e} = forall a. a -> Maybe a
Just forall a b. (a -> b) -> a -> b
$ CInt -> Errno
Errno CInt
e
ioErrorToErrno IOError
_ = forall a. Maybe a
Nothing
throwErrnoOf
:: String
-> Errno
-> IO a
throwErrnoOf :: forall a. String -> Errno -> IO a
throwErrnoOf String
loc Errno
errno = forall a. IOError -> IO a
ioError (String -> Errno -> Maybe Handle -> Maybe String -> IOError
errnoToIOError String
loc Errno
errno forall a. Maybe a
Nothing forall a. Maybe a
Nothing)
where
_dummyToSuppressWarnings :: t
_dummyToSuppressWarnings = forall a. HasCallStack => String -> a
error String
"dummy" IO Errno
getErrno forall a. String -> IO a
throwErrno
tryErrno :: IO a -> IO (Either Errno a)
tryErrno :: forall a. IO a -> IO (Either Errno a)
tryErrno = forall e b a.
Exception e =>
(e -> Maybe b) -> IO a -> IO (Either b a)
tryJust IOError -> Maybe Errno
ioErrorToErrno
tryErrno_ :: IO a -> IO Errno
tryErrno_ :: forall a. IO a -> IO Errno
tryErrno_ = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a. a -> a
id (forall a b. a -> b -> a
const Errno
eOK)) forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. IO a -> IO (Either Errno a)
tryErrno
tryErrno' :: IO a -> IO (Either Errno a)
tryErrno' :: forall a. IO a -> IO (Either Errno a)
tryErrno' = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either (forall a b. a -> b -> a
const forall a b. (a -> b) -> a -> b
$ forall a b. a -> Either a b
Left Errno
eIO) forall a. a -> a
id) forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. IO a -> IO (Either SomeException a)
try_ forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. IO a -> IO (Either Errno a)
tryErrno
tryErrno_' :: IO a -> IO Errno
tryErrno_' :: forall a. IO a -> IO Errno
tryErrno_' = forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either (forall a b. a -> b -> a
const Errno
eIO) forall a. a -> a
id) forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. IO a -> IO (Either SomeException a)
try_ forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. IO a -> IO Errno
tryErrno_
timeSpecToPOSIXTime :: TimeSpec -> POSIXTime
timeSpecToPOSIXTime :: TimeSpec -> POSIXTime
timeSpecToPOSIXTime TimeSpec
ts = forall a. Fractional a => Rational -> a
fromRational forall a b. (a -> b) -> a -> b
$ TimeSpec -> Integer
TimeSpec.toNanoSecs TimeSpec
ts forall a. Integral a => a -> a -> Ratio a
% Integer
10forall a b. (Num a, Integral b) => a -> b -> a
^(Int
9::Int)
pokeCStringLen0 :: CStringLen -> String -> IO ()
pokeCStringLen0 :: CStringLen -> String -> IO ()
pokeCStringLen0 (Ptr CChar
pBuf, Int
bufSize) String
src =
forall a. String -> (CStringLen -> IO a) -> IO a
withCStringLen String
src forall a b. (a -> b) -> a -> b
$ \(Ptr CChar
pSrc, Int
srcSize) -> do
let bufSize0 :: Int
bufSize0 = Int
bufSize forall a. Num a => a -> a -> a
- Int
1
forall a. Storable a => Ptr a -> Ptr a -> Int -> IO ()
copyArray Ptr CChar
pBuf Ptr CChar
pSrc (forall a. Ord a => a -> a -> a
min Int
bufSize0 Int
srcSize)
forall a. Storable a => Ptr a -> Int -> a -> IO ()
pokeElemOff Ptr CChar
pBuf (forall a. Ord a => a -> a -> a
min Int
bufSize0 Int
srcSize) CChar
0
testBitSet :: Bits a => a -> a -> Bool
testBitSet :: forall a. Bits a => a -> a -> Bool
testBitSet a
bits a
mask = a
bits forall a. Bits a => a -> a -> a
.&. a
mask forall a. Eq a => a -> a -> Bool
== a
mask
pread :: Fd -> ByteCount -> FileOffset -> IO ByteString
pread :: Fd -> ByteCount -> FileOffset -> IO ByteString
pread (Fd CInt
fd) ByteCount
size FileOffset
off =
forall a b. Int -> (Ptr a -> IO b) -> IO b
allocaBytes (forall a b. (Integral a, Num b) => a -> b
fromIntegral ByteCount
size) forall a b. (a -> b) -> a -> b
$ \Ptr CChar
buf -> do
CSsize
readBytes <- forall a. (Eq a, Num a) => String -> IO a -> IO a
throwErrnoIfMinus1 String
"pread" forall a b. (a -> b) -> a -> b
$ forall a. CInt -> Ptr a -> ByteCount -> FileOffset -> IO CSsize
c_pread CInt
fd Ptr CChar
buf ByteCount
size FileOffset
off
CStringLen -> IO ByteString
B.packCStringLen (Ptr CChar
buf, forall a b. (Integral a, Num b) => a -> b
fromIntegral CSsize
readBytes)
pwrite :: Fd -> ByteString -> FileOffset -> IO CSsize
pwrite :: Fd -> ByteString -> FileOffset -> IO CSsize
pwrite (Fd CInt
fd) ByteString
bs FileOffset
off =
forall a. ByteString -> (CStringLen -> IO a) -> IO a
BU.unsafeUseAsCStringLen ByteString
bs forall a b. (a -> b) -> a -> b
$ \(Ptr CChar
buf, Int
size) ->
forall a. (Eq a, Num a) => String -> IO a -> IO a
throwErrnoIfMinus1 String
"pwrite" forall a b. (a -> b) -> a -> b
$ forall a. CInt -> Ptr a -> ByteCount -> FileOffset -> IO CSsize
c_pwrite CInt
fd Ptr CChar
buf (forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
size) FileOffset
off
foreign import ccall "pread"
c_pread :: CInt -> Ptr a -> CSize -> COff -> IO CSsize
foreign import ccall "pwrite"
c_pwrite :: CInt -> Ptr a -> CSize -> COff -> IO CSsize