* lisp/textmodes/picture.el (picture-mode-exit): Doc fix. (Bug#29949)
[emacs.git] / lib / dirfd.c
blob19c80468bcfbcb7fe299216961fb9138c5fb6324
1 /* dirfd.c -- return the file descriptor associated with an open DIR*
3 Copyright (C) 2001, 2006, 2008-2018 Free Software Foundation, Inc.
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <https://www.gnu.org/licenses/>. */
18 /* Written by Jim Meyering. */
20 #include <config.h>
22 #include <dirent.h>
23 #include <errno.h>
25 #ifdef __KLIBC__
26 # include <stdlib.h>
27 # include <io.h>
29 static struct dirp_fd_list
31 DIR *dirp;
32 int fd;
33 struct dirp_fd_list *next;
34 } *dirp_fd_start = NULL;
36 /* Register fd associated with dirp to dirp_fd_list. */
37 int
38 _gl_register_dirp_fd (int fd, DIR *dirp)
40 struct dirp_fd_list *new_dirp_fd = malloc (sizeof *new_dirp_fd);
41 if (!new_dirp_fd)
42 return -1;
44 new_dirp_fd->dirp = dirp;
45 new_dirp_fd->fd = fd;
46 new_dirp_fd->next = dirp_fd_start;
48 dirp_fd_start = new_dirp_fd;
50 return 0;
53 /* Unregister fd from dirp_fd_list with closing it */
54 void
55 _gl_unregister_dirp_fd (int fd)
57 struct dirp_fd_list *dirp_fd;
58 struct dirp_fd_list *dirp_fd_prev;
60 for (dirp_fd_prev = NULL, dirp_fd = dirp_fd_start; dirp_fd;
61 dirp_fd_prev = dirp_fd, dirp_fd = dirp_fd->next)
63 if (dirp_fd->fd == fd)
65 if (dirp_fd_prev)
66 dirp_fd_prev->next = dirp_fd->next;
67 else /* dirp_fd == dirp_fd_start */
68 dirp_fd_start = dirp_fd_start->next;
70 close (fd);
71 free (dirp_fd);
72 break;
76 #endif
78 int
79 dirfd (DIR *dir_p)
81 int fd = DIR_TO_FD (dir_p);
82 if (fd == -1)
83 #ifndef __KLIBC__
84 errno = ENOTSUP;
85 #else
87 struct dirp_fd_list *dirp_fd;
89 for (dirp_fd = dirp_fd_start; dirp_fd; dirp_fd = dirp_fd->next)
90 if (dirp_fd->dirp == dir_p)
91 return dirp_fd->fd;
93 errno = EINVAL;
95 #endif
97 return fd;