Blame sysdeps/posix/readdir.c

Packit 6c4009
/* Copyright (C) 1991-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <limits.h>
Packit 6c4009
#include <stddef.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <dirent.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <sys/types.h>
Packit 6c4009
#include <assert.h>
Packit 6c4009
Packit 6c4009
#include <dirstream.h>
Packit 6c4009
Packit 6c4009
#ifndef __READDIR
Packit 6c4009
# define __READDIR __readdir
Packit 6c4009
# define __GETDENTS __getdents
Packit 6c4009
# define DIRENT_TYPE struct dirent
Packit 6c4009
# define __READDIR_ALIAS
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
/* Read a directory entry from DIRP.  */
Packit 6c4009
DIRENT_TYPE *
Packit 6c4009
__READDIR (DIR *dirp)
Packit 6c4009
{
Packit 6c4009
  DIRENT_TYPE *dp;
Packit 6c4009
  int saved_errno = errno;
Packit 6c4009
Packit 6c4009
#if IS_IN (libc)
Packit 6c4009
  __libc_lock_lock (dirp->lock);
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
  do
Packit 6c4009
    {
Packit 6c4009
      size_t reclen;
Packit 6c4009
Packit 6c4009
      if (dirp->offset >= dirp->size)
Packit 6c4009
	{
Packit 6c4009
	  /* We've emptied out our buffer.  Refill it.  */
Packit 6c4009
Packit 6c4009
	  size_t maxread;
Packit 6c4009
	  ssize_t bytes;
Packit 6c4009
Packit 6c4009
#ifndef _DIRENT_HAVE_D_RECLEN
Packit 6c4009
	  /* Fixed-size struct; must read one at a time (see below).  */
Packit 6c4009
	  maxread = sizeof *dp;
Packit 6c4009
#else
Packit 6c4009
	  maxread = dirp->allocation;
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
	  bytes = __GETDENTS (dirp->fd, dirp->data, maxread);
Packit 6c4009
	  if (bytes <= 0)
Packit 6c4009
	    {
Packit 6c4009
	      /* On some systems getdents fails with ENOENT when the
Packit 6c4009
		 open directory has been rmdir'd already.  POSIX.1
Packit 6c4009
		 requires that we treat this condition like normal EOF.  */
Packit 6c4009
	      if (bytes < 0 && errno == ENOENT)
Packit 6c4009
		bytes = 0;
Packit 6c4009
Packit 6c4009
	      /* Don't modifiy errno when reaching EOF.  */
Packit 6c4009
	      if (bytes == 0)
Packit 6c4009
		__set_errno (saved_errno);
Packit 6c4009
	      dp = NULL;
Packit 6c4009
	      break;
Packit 6c4009
	    }
Packit 6c4009
	  dirp->size = (size_t) bytes;
Packit 6c4009
Packit 6c4009
	  /* Reset the offset into the buffer.  */
Packit 6c4009
	  dirp->offset = 0;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      dp = (DIRENT_TYPE *) &dirp->data[dirp->offset];
Packit 6c4009
Packit 6c4009
#ifdef _DIRENT_HAVE_D_RECLEN
Packit 6c4009
      reclen = dp->d_reclen;
Packit 6c4009
#else
Packit 6c4009
      /* The only version of `struct dirent*' that lacks `d_reclen'
Packit 6c4009
	 is fixed-size.  */
Packit 6c4009
      assert (sizeof dp->d_name > 1);
Packit 6c4009
      reclen = sizeof *dp;
Packit 6c4009
      /* The name is not terminated if it is the largest possible size.
Packit 6c4009
	 Clobber the following byte to ensure proper null termination.  We
Packit 6c4009
	 read jst one entry at a time above so we know that byte will not
Packit 6c4009
	 be used later.  */
Packit 6c4009
      dp->d_name[sizeof dp->d_name] = '\0';
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
      dirp->offset += reclen;
Packit 6c4009
Packit 6c4009
#ifdef _DIRENT_HAVE_D_OFF
Packit 6c4009
      dirp->filepos = dp->d_off;
Packit 6c4009
#else
Packit 6c4009
      dirp->filepos += reclen;
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
      /* Skip deleted files.  */
Packit 6c4009
    } while (dp->d_ino == 0);
Packit 6c4009
Packit 6c4009
#if IS_IN (libc)
Packit 6c4009
  __libc_lock_unlock (dirp->lock);
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
  return dp;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#ifdef __READDIR_ALIAS
Packit 6c4009
weak_alias (__readdir, readdir)
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#undef __READDIR
Packit 6c4009
#undef __GETDENTS
Packit 6c4009
#undef DIRENT_TYPE
Packit 6c4009
#undef __READDIR_ALIAS