165 lines
5.0 KiB
Plaintext
165 lines
5.0 KiB
Plaintext
|
{-# LANGUAGE CApiFFI #-}
|
||
|
{-# LANGUAGE NondecreasingIndentation #-}
|
||
|
#if __GLASGOW_HASKELL__ >= 709
|
||
|
{-# LANGUAGE Safe #-}
|
||
|
#else
|
||
|
{-# LANGUAGE Trustworthy #-}
|
||
|
#endif
|
||
|
|
||
|
-----------------------------------------------------------------------------
|
||
|
-- |
|
||
|
-- Module : System.Posix.Directory
|
||
|
-- Copyright : (c) The University of Glasgow 2002
|
||
|
-- License : BSD-style (see the file libraries/base/LICENSE)
|
||
|
--
|
||
|
-- Maintainer : libraries@haskell.org
|
||
|
-- Stability : provisional
|
||
|
-- Portability : non-portable (requires POSIX)
|
||
|
--
|
||
|
-- String-based POSIX directory support
|
||
|
--
|
||
|
-----------------------------------------------------------------------------
|
||
|
|
||
|
#include "HsUnix.h"
|
||
|
|
||
|
-- hack copied from System.Posix.Files
|
||
|
#if !defined(PATH_MAX)
|
||
|
# define PATH_MAX 4096
|
||
|
#endif
|
||
|
|
||
|
module System.Posix.Directory (
|
||
|
-- * Creating and removing directories
|
||
|
createDirectory, removeDirectory,
|
||
|
|
||
|
-- * Reading directories
|
||
|
DirStream,
|
||
|
openDirStream,
|
||
|
readDirStream,
|
||
|
rewindDirStream,
|
||
|
closeDirStream,
|
||
|
DirStreamOffset,
|
||
|
#ifdef HAVE_TELLDIR
|
||
|
tellDirStream,
|
||
|
#endif
|
||
|
#ifdef HAVE_SEEKDIR
|
||
|
seekDirStream,
|
||
|
#endif
|
||
|
|
||
|
-- * The working dirctory
|
||
|
getWorkingDirectory,
|
||
|
changeWorkingDirectory,
|
||
|
changeWorkingDirectoryFd,
|
||
|
) where
|
||
|
|
||
|
import System.IO.Error
|
||
|
import System.Posix.Error
|
||
|
import System.Posix.Types
|
||
|
import Foreign
|
||
|
import Foreign.C
|
||
|
|
||
|
import System.Posix.Directory.Common
|
||
|
import System.Posix.Internals (withFilePath, peekFilePath)
|
||
|
|
||
|
-- | @createDirectory dir mode@ calls @mkdir@ to
|
||
|
-- create a new directory, @dir@, with permissions based on
|
||
|
-- @mode@.
|
||
|
createDirectory :: FilePath -> FileMode -> IO ()
|
||
|
createDirectory name mode =
|
||
|
withFilePath name $ \s ->
|
||
|
throwErrnoPathIfMinus1Retry_ "createDirectory" name (c_mkdir s mode)
|
||
|
-- POSIX doesn't allow mkdir() to return EINTR, but it does on
|
||
|
-- OS X (#5184), so we need the Retry variant here.
|
||
|
|
||
|
foreign import ccall unsafe "mkdir"
|
||
|
c_mkdir :: CString -> CMode -> IO CInt
|
||
|
|
||
|
-- | @openDirStream dir@ calls @opendir@ to obtain a
|
||
|
-- directory stream for @dir@.
|
||
|
openDirStream :: FilePath -> IO DirStream
|
||
|
openDirStream name =
|
||
|
withFilePath name $ \s -> do
|
||
|
dirp <- throwErrnoPathIfNullRetry "openDirStream" name $ c_opendir s
|
||
|
return (DirStream dirp)
|
||
|
|
||
|
foreign import capi unsafe "HsUnix.h opendir"
|
||
|
c_opendir :: CString -> IO (Ptr CDir)
|
||
|
|
||
|
-- | @readDirStream dp@ calls @readdir@ to obtain the
|
||
|
-- next directory entry (@struct dirent@) for the open directory
|
||
|
-- stream @dp@, and returns the @d_name@ member of that
|
||
|
-- structure.
|
||
|
readDirStream :: DirStream -> IO FilePath
|
||
|
readDirStream (DirStream dirp) =
|
||
|
alloca $ \ptr_dEnt -> loop ptr_dEnt
|
||
|
where
|
||
|
loop ptr_dEnt = do
|
||
|
resetErrno
|
||
|
r <- c_readdir dirp ptr_dEnt
|
||
|
if (r == 0)
|
||
|
then do dEnt <- peek ptr_dEnt
|
||
|
if (dEnt == nullPtr)
|
||
|
then return []
|
||
|
else do
|
||
|
entry <- (d_name dEnt >>= peekFilePath)
|
||
|
c_freeDirEnt dEnt
|
||
|
return entry
|
||
|
else do errno <- getErrno
|
||
|
if (errno == eINTR) then loop ptr_dEnt else do
|
||
|
let (Errno eo) = errno
|
||
|
if (eo == 0)
|
||
|
then return []
|
||
|
else throwErrno "readDirStream"
|
||
|
|
||
|
-- traversing directories
|
||
|
foreign import ccall unsafe "__hscore_readdir"
|
||
|
c_readdir :: Ptr CDir -> Ptr (Ptr CDirent) -> IO CInt
|
||
|
|
||
|
foreign import ccall unsafe "__hscore_free_dirent"
|
||
|
c_freeDirEnt :: Ptr CDirent -> IO ()
|
||
|
|
||
|
foreign import ccall unsafe "__hscore_d_name"
|
||
|
d_name :: Ptr CDirent -> IO CString
|
||
|
|
||
|
|
||
|
-- | @getWorkingDirectory@ calls @getcwd@ to obtain the name
|
||
|
-- of the current working directory.
|
||
|
getWorkingDirectory :: IO FilePath
|
||
|
getWorkingDirectory = go (#const PATH_MAX)
|
||
|
where
|
||
|
go bytes = do
|
||
|
r <- allocaBytes bytes $ \buf -> do
|
||
|
buf' <- c_getcwd buf (fromIntegral bytes)
|
||
|
if buf' /= nullPtr
|
||
|
then do s <- peekFilePath buf
|
||
|
return (Just s)
|
||
|
else do errno <- getErrno
|
||
|
if errno == eRANGE
|
||
|
-- we use Nothing to indicate that we should
|
||
|
-- try again with a bigger buffer
|
||
|
then return Nothing
|
||
|
else throwErrno "getWorkingDirectory"
|
||
|
maybe (go (2 * bytes)) return r
|
||
|
|
||
|
foreign import ccall unsafe "getcwd"
|
||
|
c_getcwd :: Ptr CChar -> CSize -> IO (Ptr CChar)
|
||
|
|
||
|
-- | @changeWorkingDirectory dir@ calls @chdir@ to change
|
||
|
-- the current working directory to @dir@.
|
||
|
changeWorkingDirectory :: FilePath -> IO ()
|
||
|
changeWorkingDirectory path =
|
||
|
modifyIOError (`ioeSetFileName` path) $
|
||
|
withFilePath path $ \s ->
|
||
|
throwErrnoIfMinus1Retry_ "changeWorkingDirectory" (c_chdir s)
|
||
|
|
||
|
foreign import ccall unsafe "chdir"
|
||
|
c_chdir :: CString -> IO CInt
|
||
|
|
||
|
removeDirectory :: FilePath -> IO ()
|
||
|
removeDirectory path =
|
||
|
modifyIOError (`ioeSetFileName` path) $
|
||
|
withFilePath path $ \s ->
|
||
|
throwErrnoIfMinus1Retry_ "removeDirectory" (c_rmdir s)
|
||
|
|
||
|
foreign import ccall unsafe "rmdir"
|
||
|
c_rmdir :: CString -> IO CInt
|