]>
git.saurik.com Git - apple/libc.git/blob - gen/nftw.c
1 /* $OpenBSD: nftw.c,v 1.2 2003/07/21 21:15:32 millert Exp $ */
4 * Copyright (c) 2003 Todd C. Miller <Todd.Miller@courtesan.com>
6 * Permission to use, copy, modify, and distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the above
8 * copyright notice and this permission notice appear in all copies.
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18 * Sponsored in part by the Defense Advanced Research Projects
19 * Agency (DARPA) and Air Force Research Laboratory, Air Force
20 * Materiel Command, USAF, under agreement number F39502-99-1-0512.
23 #if defined(LIBC_SCCS) && !defined(lint)
24 static const char rcsid
[] = "$OpenBSD: nftw.c,v 1.2 2003/07/21 21:15:32 millert Exp $";
25 #endif /* LIBC_SCCS and not lint */
28 #include <sys/cdefs.h>
29 #include <sys/types.h>
41 both_ftw(const char *path
,
42 int (*ofn
)(const char *, const struct stat
*, int),
43 int (*nfn
)(const char *, const struct stat
*, int, struct FTW
*),
44 int nfds
, int ftwflags
)
50 int ftsflags
, fnflag
, error
, postorder
, sverrno
;
51 int cwd_fd
= -1; /* cwd_fd != -1 means call chdir a lot */
54 /* Macro to skip the mount point itself in UNiX03 mode, in legcy
55 mode the mount point is returned, but we don't decend into it */
56 #define SKIP_MOUNT if ((ftwflags & FTW_MOUNT) \
57 && cur->fts_statp->st_dev != path_stat.st_dev) { \
64 /* XXX - nfds is currently unused */
65 if (nfds
< 1 || nfds
> OPEN_MAX
) {
71 ftsflags
= FTS_COMFOLLOW
;
72 if (!(ftwflags
& FTW_CHDIR
))
73 ftsflags
|= FTS_NOCHDIR
;
74 if (ftwflags
& FTW_MOUNT
)
76 if (ftwflags
& FTW_PHYS
) {
77 ftsflags
|= FTS_PHYSICAL
;
79 ftsflags
|= FTS_LOGICAL
;
81 postorder
= (ftwflags
& FTW_DEPTH
) != 0;
83 /* We have been requested to change directories, and fts doesn't
84 always do it (never for FTS_LOGICAL, and sometimes not for
86 if (ftwflags
& FTW_CHDIR
) {
87 cwd_fd
= open(".", O_RDONLY
, 0);
92 /* Prevent problems if fts ever starts using chdir when passed
94 ftsflags
|= FTS_NOCHDIR
;
98 struct stat path_stat
;
100 /* UNIX03 requires us to return -1/errno=ELOOP if path
101 is a looping symlink; fts_open is succesful and fts_read
102 gives us FTS_NS which isn't very useful, in fact we get
103 pretty much the same behaviour for ENAMETOOLONG, ENOENT,
104 ENOTDIR, and EACCES */
106 int rc
= stat(path
, &path_stat
);
108 && (errno
== ELOOP
|| errno
== ENAMETOOLONG
|| errno
== ENOENT
109 || errno
== ENOTDIR
|| errno
== EACCES
)) {
113 if (rc
>= 0 && nfn
) {
114 if (!S_ISDIR(path_stat
.st_mode
)) {
124 ftsp
= fts_open((char * const *)paths
, ftsflags
, NULL
);
130 while ((cur
= fts_read(ftsp
)) != NULL
) {
131 switch (cur
->fts_info
) {
136 /* we will get FTS_DNR next (this is not an issue for
137 FTS_DP, only FTS_D) */
138 if (access(cur
->fts_path
, R_OK
) != 0)
163 fnflag
= nfn
? FTW_SLN
: FTW_SL
;
166 /* The legacy behaviour did not signal an error
167 on symlink loops unless they ended up causing
168 a directory cycle, but the Unix2003 standard
169 requires ELOOP to end ftw and nftw walks with
172 int rc
= stat(cur
->fts_path
, &sb
);
173 if (rc
< 0 && errno
== ELOOP
) {
182 /* Unix03 says nftw should break cycles and not return
183 errors in non-physical mode (which is definitly what it
184 says ftw can't do) */
185 if (nfn
&& !(ftwflags
& FTW_PHYS
)) {
186 /* 4489297 - when FTW_DEPTH is set, skip
202 char *dir
, *free_me
= NULL
;
203 if (fnflag
== FTW_D
) {
206 /* we could alloc just enough for the directory,
207 and use memmove -- but that is a little more
208 error prone, and not noticable in with all the
210 dir
= free_me
= strdup(cur
->fts_path
);
211 dir
[cur
->fts_pathlen
- cur
->fts_namelen
] = '\0';
218 if(cur
->fts_pathlen
== cur
->fts_namelen
&&
220 /* If cwd_fd is our last FD, fts_read will give us FTS_DNR
221 * and fts_path == fts_name == "."
222 * This check results in the correct errno being returned.
231 ftw
.base
= cur
->fts_pathlen
- cur
->fts_namelen
;
232 ftw
.level
= cur
->fts_level
;
233 error
= nfn(cur
->fts_path
, cur
->fts_statp
, fnflag
, &ftw
);
235 error
= ofn(cur
->fts_path
, cur
->fts_statp
, fnflag
);
238 if (fchdir(cwd_fd
) < 0) {
250 (void) fts_close(ftsp
);
252 (void) close(cwd_fd
);
258 ftw(const char *path
, int (*fn
)(const char *, const struct stat
*, int),
261 /* The legacy implmentation didn't follow symlinks, but Unix03
262 does - this was likely a bug in the legacy implemtation; JKH
263 thinks we ought change the legacy behaviour, and I agree; anyone
264 who doesn't should replace FTW_PHYS with
265 __DARWIN_UNIX03 ? 0 : FTW_PHYS */
266 return both_ftw(path
, fn
, NULL
, nfds
, FTW_PHYS
);
270 nftw(const char *path
, int (*fn
)(const char *, const struct stat
*, int,
271 struct FTW
*), int nfds
, int ftwflags
)
273 return both_ftw(path
, NULL
, fn
, nfds
, ftwflags
);