{-# LINE 1 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LANGUAGE Trustworthy #-}
module System.Posix.Files.Common (
unionFileModes, intersectFileModes,
nullFileMode,
ownerReadMode, ownerWriteMode, ownerExecuteMode, ownerModes,
groupReadMode, groupWriteMode, groupExecuteMode, groupModes,
otherReadMode, otherWriteMode, otherExecuteMode, otherModes,
setUserIDMode, setGroupIDMode,
stdFileMode, accessModes,
fileTypeModes,
blockSpecialMode, characterSpecialMode, namedPipeMode, regularFileMode,
directoryMode, symbolicLinkMode, socketMode,
setFdMode, setFileCreationMask,
FileStatus(..),
getFdStatus,
deviceID, fileID, fileMode, linkCount, fileOwner, fileGroup,
specialDeviceID, fileSize, accessTime, modificationTime,
statusChangeTime,
accessTimeHiRes, modificationTimeHiRes, statusChangeTimeHiRes,
setFdTimesHiRes, touchFd,
isBlockDevice, isCharacterDevice, isNamedPipe, isRegularFile,
isDirectory, isSymbolicLink, isSocket,
setFdSize,
setFdOwnerAndGroup,
PathVar(..), getFdPathVar, pathVarConst,
{-# LINE 74 "libraries/unix/System/Posix/Files/Common.hsc" #-}
CTimeVal(..),
toCTimeVal,
c_utimes,
{-# LINE 78 "libraries/unix/System/Posix/Files/Common.hsc" #-}
c_lutimes,
{-# LINE 80 "libraries/unix/System/Posix/Files/Common.hsc" #-}
) where
import System.Posix.Types
import System.IO.Unsafe
import Data.Bits
import Data.Int
import Data.Ratio
import Data.Time.Clock.POSIX (POSIXTime)
import System.Posix.Internals
import Foreign.C
import Foreign.ForeignPtr
{-# LINE 92 "libraries/unix/System/Posix/Files/Common.hsc" #-}
import Foreign.Marshal (withArray)
{-# LINE 94 "libraries/unix/System/Posix/Files/Common.hsc" #-}
import Foreign.Ptr
import Foreign.Storable
nullFileMode :: FileMode
nullFileMode = 0
ownerReadMode :: FileMode
ownerReadMode = (256)
{-# LINE 110 "libraries/unix/System/Posix/Files/Common.hsc" #-}
ownerWriteMode :: FileMode
ownerWriteMode = (128)
{-# LINE 114 "libraries/unix/System/Posix/Files/Common.hsc" #-}
ownerExecuteMode :: FileMode
ownerExecuteMode = (64)
{-# LINE 118 "libraries/unix/System/Posix/Files/Common.hsc" #-}
groupReadMode :: FileMode
groupReadMode = (32)
{-# LINE 122 "libraries/unix/System/Posix/Files/Common.hsc" #-}
groupWriteMode :: FileMode
groupWriteMode = (16)
{-# LINE 126 "libraries/unix/System/Posix/Files/Common.hsc" #-}
groupExecuteMode :: FileMode
groupExecuteMode = (8)
{-# LINE 130 "libraries/unix/System/Posix/Files/Common.hsc" #-}
otherReadMode :: FileMode
otherReadMode = (4)
{-# LINE 134 "libraries/unix/System/Posix/Files/Common.hsc" #-}
otherWriteMode :: FileMode
otherWriteMode = (2)
{-# LINE 138 "libraries/unix/System/Posix/Files/Common.hsc" #-}
otherExecuteMode :: FileMode
otherExecuteMode = (1)
{-# LINE 142 "libraries/unix/System/Posix/Files/Common.hsc" #-}
setUserIDMode :: FileMode
setUserIDMode = (2048)
{-# LINE 146 "libraries/unix/System/Posix/Files/Common.hsc" #-}
setGroupIDMode :: FileMode
setGroupIDMode = (1024)
{-# LINE 150 "libraries/unix/System/Posix/Files/Common.hsc" #-}
stdFileMode :: FileMode
stdFileMode = ownerReadMode .|. ownerWriteMode .|.
groupReadMode .|. groupWriteMode .|.
otherReadMode .|. otherWriteMode
ownerModes :: FileMode
ownerModes = (448)
{-# LINE 160 "libraries/unix/System/Posix/Files/Common.hsc" #-}
groupModes :: FileMode
groupModes = (56)
{-# LINE 164 "libraries/unix/System/Posix/Files/Common.hsc" #-}
otherModes :: FileMode
otherModes = (7)
{-# LINE 168 "libraries/unix/System/Posix/Files/Common.hsc" #-}
accessModes :: FileMode
accessModes = ownerModes .|. groupModes .|. otherModes
unionFileModes :: FileMode -> FileMode -> FileMode
unionFileModes m1 m2 = m1 .|. m2
intersectFileModes :: FileMode -> FileMode -> FileMode
intersectFileModes m1 m2 = m1 .&. m2
fileTypeModes :: FileMode
fileTypeModes = (61440)
{-# LINE 185 "libraries/unix/System/Posix/Files/Common.hsc" #-}
blockSpecialMode :: FileMode
blockSpecialMode = (24576)
{-# LINE 188 "libraries/unix/System/Posix/Files/Common.hsc" #-}
characterSpecialMode :: FileMode
characterSpecialMode = (8192)
{-# LINE 191 "libraries/unix/System/Posix/Files/Common.hsc" #-}
namedPipeMode :: FileMode
namedPipeMode = (4096)
{-# LINE 194 "libraries/unix/System/Posix/Files/Common.hsc" #-}
regularFileMode :: FileMode
regularFileMode = (32768)
{-# LINE 197 "libraries/unix/System/Posix/Files/Common.hsc" #-}
directoryMode :: FileMode
directoryMode = (16384)
{-# LINE 200 "libraries/unix/System/Posix/Files/Common.hsc" #-}
symbolicLinkMode :: FileMode
symbolicLinkMode = (40960)
{-# LINE 203 "libraries/unix/System/Posix/Files/Common.hsc" #-}
socketMode :: FileMode
socketMode = (49152)
{-# LINE 206 "libraries/unix/System/Posix/Files/Common.hsc" #-}
setFdMode :: Fd -> FileMode -> IO ()
setFdMode (Fd fd) m =
throwErrnoIfMinus1_ "setFdMode" (c_fchmod fd m)
foreign import ccall unsafe "fchmod"
c_fchmod :: CInt -> CMode -> IO CInt
setFileCreationMask :: FileMode -> IO FileMode
setFileCreationMask mask = c_umask mask
newtype FileStatus = FileStatus (ForeignPtr CStat)
deviceID :: FileStatus -> DeviceID
fileID :: FileStatus -> FileID
fileMode :: FileStatus -> FileMode
linkCount :: FileStatus -> LinkCount
fileOwner :: FileStatus -> UserID
fileGroup :: FileStatus -> GroupID
specialDeviceID :: FileStatus -> DeviceID
fileSize :: FileStatus -> FileOffset
accessTime :: FileStatus -> EpochTime
accessTimeHiRes :: FileStatus -> POSIXTime
modificationTime :: FileStatus -> EpochTime
modificationTimeHiRes :: FileStatus -> POSIXTime
statusChangeTime :: FileStatus -> EpochTime
statusChangeTimeHiRes :: FileStatus -> POSIXTime
deviceID (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 0))
{-# LINE 268 "libraries/unix/System/Posix/Files/Common.hsc" #-}
fileID (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 8))
{-# LINE 270 "libraries/unix/System/Posix/Files/Common.hsc" #-}
fileMode (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 4))
{-# LINE 272 "libraries/unix/System/Posix/Files/Common.hsc" #-}
linkCount (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 6))
{-# LINE 274 "libraries/unix/System/Posix/Files/Common.hsc" #-}
fileOwner (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 16))
{-# LINE 276 "libraries/unix/System/Posix/Files/Common.hsc" #-}
fileGroup (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 20))
{-# LINE 278 "libraries/unix/System/Posix/Files/Common.hsc" #-}
specialDeviceID (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 24))
{-# LINE 280 "libraries/unix/System/Posix/Files/Common.hsc" #-}
fileSize (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 96))
{-# LINE 282 "libraries/unix/System/Posix/Files/Common.hsc" #-}
accessTime (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 32))
{-# LINE 284 "libraries/unix/System/Posix/Files/Common.hsc" #-}
modificationTime (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 48))
{-# LINE 286 "libraries/unix/System/Posix/Files/Common.hsc" #-}
statusChangeTime (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ ((\hsc_ptr -> peekByteOff hsc_ptr 64))
{-# LINE 288 "libraries/unix/System/Posix/Files/Common.hsc" #-}
accessTimeHiRes (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ \stat_ptr -> do
sec <- ((\hsc_ptr -> peekByteOff hsc_ptr 32)) stat_ptr :: IO EpochTime
{-# LINE 292 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 296 "libraries/unix/System/Posix/Files/Common.hsc" #-}
nsec <- ((\hsc_ptr -> peekByteOff hsc_ptr 40)) stat_ptr :: IO (Int64)
{-# LINE 297 "libraries/unix/System/Posix/Files/Common.hsc" #-}
let frac = toInteger nsec % 10^(9::Int)
{-# LINE 310 "libraries/unix/System/Posix/Files/Common.hsc" #-}
return $ fromRational $ toRational sec + frac
modificationTimeHiRes (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ \stat_ptr -> do
sec <- ((\hsc_ptr -> peekByteOff hsc_ptr 48)) stat_ptr :: IO EpochTime
{-# LINE 315 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 319 "libraries/unix/System/Posix/Files/Common.hsc" #-}
nsec <- ((\hsc_ptr -> peekByteOff hsc_ptr 56)) stat_ptr :: IO (Int64)
{-# LINE 320 "libraries/unix/System/Posix/Files/Common.hsc" #-}
let frac = toInteger nsec % 10^(9::Int)
{-# LINE 333 "libraries/unix/System/Posix/Files/Common.hsc" #-}
return $ fromRational $ toRational sec + frac
statusChangeTimeHiRes (FileStatus stat) =
unsafePerformIO $ withForeignPtr stat $ \stat_ptr -> do
sec <- ((\hsc_ptr -> peekByteOff hsc_ptr 64)) stat_ptr :: IO EpochTime
{-# LINE 338 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 342 "libraries/unix/System/Posix/Files/Common.hsc" #-}
nsec <- ((\hsc_ptr -> peekByteOff hsc_ptr 72)) stat_ptr :: IO (Int64)
{-# LINE 343 "libraries/unix/System/Posix/Files/Common.hsc" #-}
let frac = toInteger nsec % 10^(9::Int)
{-# LINE 356 "libraries/unix/System/Posix/Files/Common.hsc" #-}
return $ fromRational $ toRational sec + frac
isBlockDevice :: FileStatus -> Bool
isCharacterDevice :: FileStatus -> Bool
isNamedPipe :: FileStatus -> Bool
isRegularFile :: FileStatus -> Bool
isDirectory :: FileStatus -> Bool
isSymbolicLink :: FileStatus -> Bool
isSocket :: FileStatus -> Bool
isBlockDevice stat =
(fileMode stat `intersectFileModes` fileTypeModes) == blockSpecialMode
isCharacterDevice stat =
(fileMode stat `intersectFileModes` fileTypeModes) == characterSpecialMode
isNamedPipe stat =
(fileMode stat `intersectFileModes` fileTypeModes) == namedPipeMode
isRegularFile stat =
(fileMode stat `intersectFileModes` fileTypeModes) == regularFileMode
isDirectory stat =
(fileMode stat `intersectFileModes` fileTypeModes) == directoryMode
isSymbolicLink stat =
(fileMode stat `intersectFileModes` fileTypeModes) == symbolicLinkMode
isSocket stat =
(fileMode stat `intersectFileModes` fileTypeModes) == socketMode
getFdStatus :: Fd -> IO FileStatus
getFdStatus (Fd fd) = do
fp <- mallocForeignPtrBytes (144)
{-# LINE 394 "libraries/unix/System/Posix/Files/Common.hsc" #-}
withForeignPtr fp $ \p ->
throwErrnoIfMinus1_ "getFdStatus" (c_fstat fd p)
return (FileStatus fp)
{-# LINE 421 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 426 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 431 "libraries/unix/System/Posix/Files/Common.hsc" #-}
data CTimeVal = CTimeVal CLong CLong
instance Storable CTimeVal where
sizeOf _ = (16)
{-# LINE 436 "libraries/unix/System/Posix/Files/Common.hsc" #-}
alignment _ = alignment (undefined :: CInt)
poke p (CTimeVal sec usec) = do
((\hsc_ptr -> pokeByteOff hsc_ptr 0)) p sec
{-# LINE 439 "libraries/unix/System/Posix/Files/Common.hsc" #-}
((\hsc_ptr -> pokeByteOff hsc_ptr 8)) p usec
{-# LINE 440 "libraries/unix/System/Posix/Files/Common.hsc" #-}
peek p = do
sec <- (\hsc_ptr -> peekByteOff hsc_ptr 0) p
{-# LINE 442 "libraries/unix/System/Posix/Files/Common.hsc" #-}
usec <- (\hsc_ptr -> peekByteOff hsc_ptr 8) p
{-# LINE 443 "libraries/unix/System/Posix/Files/Common.hsc" #-}
return $ CTimeVal sec usec
toCTimeVal :: POSIXTime -> CTimeVal
toCTimeVal t = CTimeVal sec (truncate $ 10^(6::Int) * frac)
where
(sec, frac) = if (frac' < 0) then (sec' - 1, frac' + 1) else (sec', frac')
(sec', frac') = properFraction $ toRational t
foreign import ccall unsafe "utimes"
c_utimes :: CString -> Ptr CTimeVal -> IO CInt
{-# LINE 455 "libraries/unix/System/Posix/Files/Common.hsc" #-}
foreign import ccall unsafe "lutimes"
c_lutimes :: CString -> Ptr CTimeVal -> IO CInt
{-# LINE 458 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 460 "libraries/unix/System/Posix/Files/Common.hsc" #-}
foreign import ccall unsafe "futimes"
c_futimes :: CInt -> Ptr CTimeVal -> IO CInt
{-# LINE 463 "libraries/unix/System/Posix/Files/Common.hsc" #-}
setFdTimesHiRes :: Fd -> POSIXTime -> POSIXTime -> IO ()
{-# LINE 477 "libraries/unix/System/Posix/Files/Common.hsc" #-}
setFdTimesHiRes (Fd fd) atime mtime =
withArray [toCTimeVal atime, toCTimeVal mtime] $ \times ->
throwErrnoIfMinus1_ "setFdTimesHiRes" (c_futimes fd times)
{-# LINE 484 "libraries/unix/System/Posix/Files/Common.hsc" #-}
touchFd :: Fd -> IO ()
{-# LINE 494 "libraries/unix/System/Posix/Files/Common.hsc" #-}
touchFd (Fd fd) =
throwErrnoIfMinus1_ "touchFd" (c_futimes fd nullPtr)
{-# LINE 500 "libraries/unix/System/Posix/Files/Common.hsc" #-}
setFdOwnerAndGroup :: Fd -> UserID -> GroupID -> IO ()
setFdOwnerAndGroup (Fd fd) uid gid =
throwErrnoIfMinus1_ "setFdOwnerAndGroup" (c_fchown fd uid gid)
foreign import ccall unsafe "fchown"
c_fchown :: CInt -> CUid -> CGid -> IO CInt
setFdSize :: Fd -> FileOffset -> IO ()
setFdSize (Fd fd) off =
throwErrnoIfMinus1_ "setFdSize" (c_ftruncate fd off)
data PathVar
= FileSizeBits
| LinkLimit
| InputLineLimit
| InputQueueLimit
| FileNameLimit
| PathNameLimit
| PipeBufferLimit
| SymbolicLinkLimit
| SetOwnerAndGroupIsRestricted
| FileNamesAreNotTruncated
| VDisableChar
| AsyncIOAvailable
| PrioIOAvailable
| SyncIOAvailable
pathVarConst :: PathVar -> CInt
pathVarConst v = case v of
LinkLimit -> (1)
{-# LINE 553 "libraries/unix/System/Posix/Files/Common.hsc" #-}
InputLineLimit -> (2)
{-# LINE 554 "libraries/unix/System/Posix/Files/Common.hsc" #-}
InputQueueLimit -> (3)
{-# LINE 555 "libraries/unix/System/Posix/Files/Common.hsc" #-}
FileNameLimit -> (4)
{-# LINE 556 "libraries/unix/System/Posix/Files/Common.hsc" #-}
PathNameLimit -> (5)
{-# LINE 557 "libraries/unix/System/Posix/Files/Common.hsc" #-}
PipeBufferLimit -> (6)
{-# LINE 558 "libraries/unix/System/Posix/Files/Common.hsc" #-}
SetOwnerAndGroupIsRestricted -> (7)
{-# LINE 559 "libraries/unix/System/Posix/Files/Common.hsc" #-}
FileNamesAreNotTruncated -> (8)
{-# LINE 560 "libraries/unix/System/Posix/Files/Common.hsc" #-}
VDisableChar -> (9)
{-# LINE 561 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 563 "libraries/unix/System/Posix/Files/Common.hsc" #-}
SyncIOAvailable -> (25)
{-# LINE 564 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 567 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 569 "libraries/unix/System/Posix/Files/Common.hsc" #-}
AsyncIOAvailable -> (17)
{-# LINE 570 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 573 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 575 "libraries/unix/System/Posix/Files/Common.hsc" #-}
PrioIOAvailable -> (19)
{-# LINE 576 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 579 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 581 "libraries/unix/System/Posix/Files/Common.hsc" #-}
FileSizeBits -> (18)
{-# LINE 582 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 585 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 587 "libraries/unix/System/Posix/Files/Common.hsc" #-}
SymbolicLinkLimit -> (24)
{-# LINE 588 "libraries/unix/System/Posix/Files/Common.hsc" #-}
{-# LINE 591 "libraries/unix/System/Posix/Files/Common.hsc" #-}
getFdPathVar :: Fd -> PathVar -> IO Limit
getFdPathVar (Fd fd) v =
throwErrnoIfMinus1 "getFdPathVar" $
c_fpathconf fd (pathVarConst v)
foreign import ccall unsafe "fpathconf"
c_fpathconf :: CInt -> CInt -> IO CLong