341 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			341 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* dir.cc: Posix directory-related routines
 | 
						|
 | 
						|
   Copyright 1996, 1997, 1998, 1999, 2000 Cygnus Solutions.
 | 
						|
 | 
						|
This file is part of Cygwin.
 | 
						|
 | 
						|
This software is a copyrighted work licensed under the terms of the
 | 
						|
Cygwin license.  Please consult the file "CYGWIN_LICENSE" for
 | 
						|
details. */
 | 
						|
 | 
						|
#include <unistd.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <errno.h>
 | 
						|
#include "winsup.h"
 | 
						|
 | 
						|
#define _COMPILING_NEWLIB
 | 
						|
#include "dirent.h"
 | 
						|
 | 
						|
/* Cygwin internal */
 | 
						|
/* Return whether the directory of a file is writable.  Return 1 if it
 | 
						|
   is.  Otherwise, return 0, and set errno appropriately.  */
 | 
						|
int __stdcall
 | 
						|
writable_directory (const char *file)
 | 
						|
{
 | 
						|
  char dir[strlen (file) + 1];
 | 
						|
 | 
						|
  strcpy (dir, file);
 | 
						|
 | 
						|
  const char *usedir;
 | 
						|
  char *slash = strrchr (dir, '\\');
 | 
						|
  if (slash == NULL)
 | 
						|
    usedir = ".";
 | 
						|
  else
 | 
						|
    {
 | 
						|
      *slash = '\0';
 | 
						|
      usedir = dir;
 | 
						|
    }
 | 
						|
 | 
						|
  int acc = access (usedir, W_OK);
 | 
						|
 | 
						|
  return acc == 0;
 | 
						|
}
 | 
						|
 | 
						|
/* opendir: POSIX 5.1.2.1 */
 | 
						|
extern "C" DIR *
 | 
						|
opendir (const char *dirname)
 | 
						|
{
 | 
						|
  int len;
 | 
						|
  DIR *dir;
 | 
						|
  DIR *res = 0;
 | 
						|
  struct stat statbuf;
 | 
						|
 | 
						|
  path_conv real_dirname (dirname, SYMLINK_FOLLOW, 1);
 | 
						|
 | 
						|
  if (real_dirname.error)
 | 
						|
    {
 | 
						|
      set_errno (real_dirname.error);
 | 
						|
      goto failed;
 | 
						|
    }
 | 
						|
 | 
						|
  if (stat (real_dirname.get_win32 (), &statbuf) == -1)
 | 
						|
    goto failed;
 | 
						|
 | 
						|
  if (!(statbuf.st_mode & S_IFDIR))
 | 
						|
    {
 | 
						|
      set_errno (ENOTDIR);
 | 
						|
      goto failed;
 | 
						|
    }
 | 
						|
 | 
						|
  len = strlen (real_dirname.get_win32 ());
 | 
						|
  if (len > MAX_PATH - 3)
 | 
						|
    {
 | 
						|
      set_errno (ENAMETOOLONG);
 | 
						|
      goto failed;
 | 
						|
    }
 | 
						|
 | 
						|
  if ((dir = (DIR *) malloc (sizeof (DIR))) == NULL)
 | 
						|
    {
 | 
						|
      set_errno (ENOMEM);
 | 
						|
      goto failed;
 | 
						|
    }
 | 
						|
  if ((dir->__d_dirname = (char *) malloc (len + 3)) == NULL)
 | 
						|
    {
 | 
						|
      free (dir);
 | 
						|
      set_errno (ENOMEM);
 | 
						|
      goto failed;
 | 
						|
    }
 | 
						|
  if ((dir->__d_dirent =
 | 
						|
	    (struct dirent *) malloc (sizeof (struct dirent))) == NULL)
 | 
						|
    {
 | 
						|
      free (dir->__d_dirname);
 | 
						|
      free (dir);
 | 
						|
      set_errno (ENOMEM);
 | 
						|
      goto failed;
 | 
						|
    }
 | 
						|
  strcpy (dir->__d_dirname, real_dirname.get_win32 ());
 | 
						|
  /* FindFirstFile doesn't seem to like duplicate /'s. */
 | 
						|
  len = strlen (dir->__d_dirname);
 | 
						|
  if (len == 0 || SLASH_P (dir->__d_dirname[len - 1]))
 | 
						|
    strcat (dir->__d_dirname, "*");
 | 
						|
  else
 | 
						|
    strcat (dir->__d_dirname, "\\*");  /**/
 | 
						|
  dir->__d_cookie = __DIRENT_COOKIE;
 | 
						|
  dir->__d_u.__d_data.__handle = INVALID_HANDLE_VALUE;
 | 
						|
  dir->__d_position = 0;
 | 
						|
  dir->__d_dirhash = statbuf.st_ino;
 | 
						|
 | 
						|
  res = dir;
 | 
						|
 | 
						|
failed:
 | 
						|
  syscall_printf ("%p = opendir (%s)", res, dirname);
 | 
						|
  return res;
 | 
						|
}
 | 
						|
 | 
						|
/* readdir: POSIX 5.1.2.1 */
 | 
						|
extern "C" struct dirent *
 | 
						|
readdir (DIR * dir)
 | 
						|
{
 | 
						|
  WIN32_FIND_DATA buf;
 | 
						|
  HANDLE handle;
 | 
						|
  struct dirent *res = 0;
 | 
						|
  int prior_errno;
 | 
						|
 | 
						|
  if (dir->__d_cookie != __DIRENT_COOKIE)
 | 
						|
    {
 | 
						|
      set_errno (EBADF);
 | 
						|
      syscall_printf ("%p = readdir (%p)", res, dir);
 | 
						|
      return res;
 | 
						|
    }
 | 
						|
 | 
						|
  if (dir->__d_u.__d_data.__handle != INVALID_HANDLE_VALUE)
 | 
						|
    {
 | 
						|
      if (FindNextFileA (dir->__d_u.__d_data.__handle, &buf) == 0)
 | 
						|
	{
 | 
						|
	  prior_errno = get_errno();
 | 
						|
	  (void) FindClose (dir->__d_u.__d_data.__handle);
 | 
						|
	  dir->__d_u.__d_data.__handle = INVALID_HANDLE_VALUE;
 | 
						|
	  __seterrno ();
 | 
						|
	  /* POSIX says you shouldn't set errno when readdir can't
 | 
						|
	     find any more files; if another error we leave it set. */
 | 
						|
	  if (get_errno () == ENMFILE)
 | 
						|
	      set_errno (prior_errno);
 | 
						|
	  syscall_printf ("%p = readdir (%p)", res, dir);
 | 
						|
	  return res;
 | 
						|
	}
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      handle = FindFirstFileA (dir->__d_dirname, &buf);
 | 
						|
 | 
						|
      if (handle == INVALID_HANDLE_VALUE)
 | 
						|
	{
 | 
						|
	  /* It's possible that someone else deleted or emptied the directory
 | 
						|
	     or some such between the opendir () call and here.  */
 | 
						|
	  prior_errno = get_errno ();
 | 
						|
	  __seterrno ();
 | 
						|
	  /* POSIX says you shouldn't set errno when readdir can't
 | 
						|
	     find any more files; if another error we leave it set. */
 | 
						|
	  if (get_errno () == ENMFILE)
 | 
						|
	      set_errno (prior_errno);
 | 
						|
	  syscall_printf ("%p = readdir (%p)", res, dir);
 | 
						|
	  return res;
 | 
						|
	}
 | 
						|
      dir->__d_u.__d_data.__handle = handle;
 | 
						|
    }
 | 
						|
 | 
						|
  /* We get here if `buf' contains valid data.  */
 | 
						|
  strcpy (dir->__d_dirent->d_name, buf.cFileName);
 | 
						|
 | 
						|
  /* Compute d_ino by combining filename hash with the directory hash
 | 
						|
     (which was stored in dir->__d_dirhash when opendir was called). */
 | 
						|
  if (buf.cFileName[0] == '.')
 | 
						|
    {
 | 
						|
      if (buf.cFileName[1] == '\0')
 | 
						|
	dir->__d_dirent->d_ino = dir->__d_dirhash;
 | 
						|
      else if (buf.cFileName[1] != '.' || buf.cFileName[2] != '\0')
 | 
						|
	goto hashit;
 | 
						|
      else
 | 
						|
	{
 | 
						|
	  char *p, up[strlen (dir->__d_dirname) + 1];
 | 
						|
	  strcpy (up, dir->__d_dirname);
 | 
						|
	  if (!(p = strrchr (up, '\\')))
 | 
						|
	    goto hashit;
 | 
						|
	  *p = '\0';
 | 
						|
	  if (!(p = strrchr (up, '\\')))
 | 
						|
	    dir->__d_dirent->d_ino = hash_path_name (0, ".");
 | 
						|
	  else
 | 
						|
	    {
 | 
						|
	      *p = '\0';
 | 
						|
	      dir->__d_dirent->d_ino = hash_path_name (0, up);
 | 
						|
	    }
 | 
						|
	}
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
  hashit:
 | 
						|
      ino_t dino = hash_path_name (dir->__d_dirhash, "\\");
 | 
						|
      dir->__d_dirent->d_ino = hash_path_name (dino, buf.cFileName);
 | 
						|
    }
 | 
						|
 | 
						|
  ++dir->__d_position;
 | 
						|
  res = dir->__d_dirent;
 | 
						|
  syscall_printf ("%p = readdir (%p) (%s)",
 | 
						|
		  &dir->__d_dirent, dir, buf.cFileName);
 | 
						|
  return res;
 | 
						|
}
 | 
						|
 | 
						|
/* telldir */
 | 
						|
extern "C" off_t
 | 
						|
telldir (DIR * dir)
 | 
						|
{
 | 
						|
  if (dir->__d_cookie != __DIRENT_COOKIE)
 | 
						|
    return 0;
 | 
						|
  return dir->__d_position;
 | 
						|
}
 | 
						|
 | 
						|
/* seekdir */
 | 
						|
extern "C" void
 | 
						|
seekdir (DIR * dir, off_t loc)
 | 
						|
{
 | 
						|
  if (dir->__d_cookie != __DIRENT_COOKIE)
 | 
						|
    return;
 | 
						|
  rewinddir (dir);
 | 
						|
  while (loc > dir->__d_position)
 | 
						|
    if (! readdir (dir))
 | 
						|
      break;
 | 
						|
}
 | 
						|
 | 
						|
/* rewinddir: POSIX 5.1.2.1 */
 | 
						|
extern "C" void
 | 
						|
rewinddir (DIR * dir)
 | 
						|
{
 | 
						|
  syscall_printf ("rewinddir (%p)", dir);
 | 
						|
 | 
						|
  if (dir->__d_cookie != __DIRENT_COOKIE)
 | 
						|
    return;
 | 
						|
  if (dir->__d_u.__d_data.__handle != INVALID_HANDLE_VALUE)
 | 
						|
    {
 | 
						|
      (void) FindClose (dir->__d_u.__d_data.__handle);
 | 
						|
      dir->__d_u.__d_data.__handle = INVALID_HANDLE_VALUE;
 | 
						|
      dir->__d_position = 0;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/* closedir: POSIX 5.1.2.1 */
 | 
						|
extern "C" int
 | 
						|
closedir (DIR * dir)
 | 
						|
{
 | 
						|
  if (dir->__d_cookie != __DIRENT_COOKIE)
 | 
						|
    {
 | 
						|
      set_errno (EBADF);
 | 
						|
      syscall_printf ("-1 = closedir (%p)", dir);
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  if (dir->__d_u.__d_data.__handle != INVALID_HANDLE_VALUE &&
 | 
						|
      FindClose (dir->__d_u.__d_data.__handle) == 0)
 | 
						|
    {
 | 
						|
      __seterrno ();
 | 
						|
      syscall_printf ("-1 = closedir (%p)", dir);
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
 | 
						|
  /* Reset the marker in case the caller tries to use `dir' again.  */
 | 
						|
  dir->__d_cookie = 0;
 | 
						|
 | 
						|
  free (dir->__d_dirname);
 | 
						|
  free (dir->__d_dirent);
 | 
						|
  free (dir);
 | 
						|
  syscall_printf ("0 = closedir (%p)", dir);
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* mkdir: POSIX 5.4.1.1 */
 | 
						|
extern "C" int
 | 
						|
mkdir (const char *dir, mode_t mode)
 | 
						|
{
 | 
						|
  int res = -1;
 | 
						|
 | 
						|
  path_conv real_dir (dir, SYMLINK_NOFOLLOW);
 | 
						|
 | 
						|
  if (real_dir.error)
 | 
						|
    {
 | 
						|
      set_errno (real_dir.error);
 | 
						|
      goto done;
 | 
						|
    }
 | 
						|
 | 
						|
  nofinalslash(real_dir.get_win32 (), real_dir.get_win32 ());
 | 
						|
  if (! writable_directory (real_dir.get_win32 ()))
 | 
						|
    goto done;
 | 
						|
 | 
						|
  if (CreateDirectoryA (real_dir.get_win32 (), 0))
 | 
						|
    {
 | 
						|
      set_file_attribute (real_dir.has_acls (), real_dir.get_win32 (),
 | 
						|
                          (mode & 0777) & ~myself->umask);
 | 
						|
      res = 0;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    __seterrno ();
 | 
						|
 | 
						|
done:
 | 
						|
  syscall_printf ("%d = mkdir (%s, %d)", res, dir, mode);
 | 
						|
  return res;
 | 
						|
}
 | 
						|
 | 
						|
/* rmdir: POSIX 5.5.2.1 */
 | 
						|
extern "C" int
 | 
						|
rmdir (const char *dir)
 | 
						|
{
 | 
						|
  int res = -1;
 | 
						|
 | 
						|
  path_conv real_dir (dir, SYMLINK_NOFOLLOW);
 | 
						|
 | 
						|
  if (real_dir.error)
 | 
						|
    {
 | 
						|
      set_errno (real_dir.error);
 | 
						|
      goto done;
 | 
						|
    }
 | 
						|
 | 
						|
  if (RemoveDirectoryA (real_dir.get_win32 ()))
 | 
						|
    res = 0;
 | 
						|
  else if (os_being_run != winNT && GetLastError() == ERROR_ACCESS_DENIED)
 | 
						|
    {
 | 
						|
      /* Under Windows 95 & 98, ERROR_ACCESS_DENIED is returned
 | 
						|
	 if you try to remove a file or a non-empty directory. */
 | 
						|
     if (GetFileAttributes (real_dir.get_win32()) != FILE_ATTRIBUTE_DIRECTORY)
 | 
						|
       set_errno (ENOTDIR);
 | 
						|
     else
 | 
						|
       set_errno (ENOTEMPTY);
 | 
						|
    }
 | 
						|
  else if (GetLastError () == ERROR_DIRECTORY)
 | 
						|
    set_errno (ENOTDIR);
 | 
						|
  else
 | 
						|
    __seterrno ();
 | 
						|
 | 
						|
done:
 | 
						|
  syscall_printf ("%d = rmdir (%s)", res, dir);
 | 
						|
  return res;
 | 
						|
}
 |