opendir.c 1.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. #include <errno.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <fcntl.h>
  5. #include <unistd.h>
  6. #include <sys/dir.h>
  7. #include <sys/stat.h>
  8. #include "dirstream.h"
  9. /* opendir just makes an open() call - it return NULL if it fails
  10. * (open sets errno), otherwise it returns a DIR * pointer.
  11. */
  12. DIR attribute_hidden *__opendir(const char *name)
  13. {
  14. int fd;
  15. struct stat statbuf;
  16. char *buf;
  17. DIR *ptr;
  18. if (stat(name, &statbuf))
  19. return NULL;
  20. if (!S_ISDIR(statbuf.st_mode)) {
  21. __set_errno(ENOTDIR);
  22. return NULL;
  23. }
  24. if ((fd = __open(name, O_RDONLY)) < 0)
  25. return NULL;
  26. /* According to POSIX, directory streams should be closed when
  27. * exec. From "Anna Pluzhnikov" <besp@midway.uchicago.edu>.
  28. */
  29. if (fcntl(fd, F_SETFD, FD_CLOEXEC) < 0)
  30. return NULL;
  31. if (!(ptr = malloc(sizeof(*ptr)))) {
  32. __close(fd);
  33. __set_errno(ENOMEM);
  34. return NULL;
  35. }
  36. ptr->dd_fd = fd;
  37. ptr->dd_nextloc = ptr->dd_size = ptr->dd_nextoff = 0;
  38. ptr->dd_max = statbuf.st_blksize;
  39. if (ptr->dd_max < 512)
  40. ptr->dd_max = 512;
  41. if (!(buf = calloc(1, ptr->dd_max))) {
  42. __close(fd);
  43. free(ptr);
  44. __set_errno(ENOMEM);
  45. return NULL;
  46. }
  47. ptr->dd_buf = buf;
  48. __pthread_mutex_init(&(ptr->dd_lock), NULL);
  49. return ptr;
  50. }
  51. strong_alias(__opendir,opendir)