1 /* $OpenBSD: nftw.c,v 1.7 2006/03/31 19:41:44 millert Exp $ */ 2 3 /* 4 * Copyright (c) 2003, 2004 Todd C. Miller <Todd.Miller@courtesan.com> 5 * 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. 9 * 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. 17 * 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. 21 */ 22 23 #include <sys/cdefs.h> 24 #include <sys/types.h> 25 #include <sys/stat.h> 26 #include <errno.h> 27 #include <fts.h> 28 #include <ftw.h> 29 30 int 31 nftw(const char *path, int (*fn)(const char *, const struct stat *, int, 32 struct FTW *), int nfds, int ftwflags) 33 { 34 char * const paths[2] = { (char *)path, NULL }; 35 struct FTW ftw; 36 FTSENT *cur; 37 FTS *ftsp; 38 int error = 0, ftsflags, fnflag, postorder, sverrno; 39 40 /* XXX - nfds is currently unused */ 41 if (nfds < 1) { 42 errno = EINVAL; 43 return (-1); 44 } 45 46 ftsflags = FTS_COMFOLLOW; 47 if (!(ftwflags & FTW_CHDIR)) 48 ftsflags |= FTS_NOCHDIR; 49 if (ftwflags & FTW_MOUNT) 50 ftsflags |= FTS_XDEV; 51 if (ftwflags & FTW_PHYS) 52 ftsflags |= FTS_PHYSICAL; 53 else 54 ftsflags |= FTS_LOGICAL; 55 postorder = (ftwflags & FTW_DEPTH) != 0; 56 ftsp = fts_open(paths, ftsflags, NULL); 57 if (ftsp == NULL) 58 return (-1); 59 while ((cur = fts_read(ftsp)) != NULL) { 60 switch (cur->fts_info) { 61 case FTS_D: 62 if (postorder) 63 continue; 64 fnflag = FTW_D; 65 break; 66 case FTS_DC: 67 continue; 68 case FTS_DNR: 69 fnflag = FTW_DNR; 70 break; 71 case FTS_DP: 72 if (!postorder) 73 continue; 74 fnflag = FTW_DP; 75 break; 76 case FTS_F: 77 case FTS_DEFAULT: 78 fnflag = FTW_F; 79 break; 80 case FTS_NS: 81 case FTS_NSOK: 82 fnflag = FTW_NS; 83 break; 84 case FTS_SL: 85 fnflag = FTW_SL; 86 break; 87 case FTS_SLNONE: 88 fnflag = FTW_SLN; 89 break; 90 default: 91 error = -1; 92 goto done; 93 } 94 ftw.base = cur->fts_pathlen - cur->fts_namelen; 95 ftw.level = cur->fts_level; 96 error = fn(cur->fts_path, cur->fts_statp, fnflag, &ftw); 97 if (error != 0) 98 break; 99 } 100 done: 101 sverrno = errno; 102 if (fts_close(ftsp) != 0 && error == 0) 103 error = -1; 104 else 105 errno = sverrno; 106 return (error); 107 } 108