| 1 | #include <ftw.h>
|
|---|
| 2 | #include <getopt.h>
|
|---|
| 3 | #include <mcheck.h>
|
|---|
| 4 | #include <stdio.h>
|
|---|
| 5 | #include <stdlib.h>
|
|---|
| 6 | #include <string.h>
|
|---|
| 7 | #include <unistd.h>
|
|---|
| 8 | #include <sys/stat.h>
|
|---|
| 9 | #ifdef FTW_CONTINUE
|
|---|
| 10 | #define FTW_CONTINUE 0
|
|---|
| 11 | #endif
|
|---|
| 12 | #ifdef FTW_STOP
|
|---|
| 13 | #define FTW_STOP 1
|
|---|
| 14 | #endif
|
|---|
| 15 |
|
|---|
| 16 |
|
|---|
| 17 |
|
|---|
| 18 | int do_depth;
|
|---|
| 19 | int do_chdir;
|
|---|
| 20 | int do_phys;
|
|---|
| 21 | int do_exit;
|
|---|
| 22 | char *skip_subtree;
|
|---|
| 23 | char *skip_siblings;
|
|---|
| 24 |
|
|---|
| 25 | struct option options[] =
|
|---|
| 26 | {
|
|---|
| 27 | { "depth", no_argument, &do_depth, 1 },
|
|---|
| 28 | { "chdir", no_argument, &do_chdir, 1 },
|
|---|
| 29 | { "phys", no_argument, &do_phys, 1 },
|
|---|
| 30 | { "skip-subtree", required_argument, NULL, 't' },
|
|---|
| 31 | { "skip-siblings", required_argument, NULL, 's' },
|
|---|
| 32 | { "early-exit", no_argument, &do_exit, 1 },
|
|---|
| 33 | { NULL, 0, NULL, 0 }
|
|---|
| 34 | };
|
|---|
| 35 |
|
|---|
| 36 | const char *flag2name[] =
|
|---|
| 37 | {
|
|---|
| 38 | [FTW_F] = "FTW_F",
|
|---|
| 39 | [FTW_D] = "FTW_D",
|
|---|
| 40 | [FTW_DNR] = "FTW_DNR",
|
|---|
| 41 | [FTW_NS] = "FTW_NS",
|
|---|
| 42 | [FTW_SL] = "FTW_SL",
|
|---|
| 43 | [FTW_DP] = "FTW_DP",
|
|---|
| 44 | [FTW_SLN] = "FTW_SLN"
|
|---|
| 45 | };
|
|---|
| 46 |
|
|---|
| 47 |
|
|---|
| 48 | static int
|
|---|
| 49 | cb (const char *name, const struct stat *st, int flag, struct FTW *f)
|
|---|
| 50 | {
|
|---|
| 51 | if (do_exit && strcmp (name + f->base, "file@2"))
|
|---|
| 52 | return FTW_CONTINUE;
|
|---|
| 53 |
|
|---|
| 54 | printf ("base = \"%.*s\", file = \"%s\", flag = %s",
|
|---|
| 55 | f->base, name, name + f->base, flag2name[flag]);
|
|---|
| 56 | if (do_chdir)
|
|---|
| 57 | {
|
|---|
| 58 | char *cwd = getcwd (NULL, 0);
|
|---|
| 59 | printf (", cwd = %s", cwd);
|
|---|
| 60 | free (cwd);
|
|---|
| 61 | }
|
|---|
| 62 | printf (", level = %d\n", f->level);
|
|---|
| 63 |
|
|---|
| 64 | #ifdef FTW_ACTIONRETVAL
|
|---|
| 65 | if (skip_siblings && strcmp (name + f->base, skip_siblings) == 0)
|
|---|
| 66 | return FTW_SKIP_SIBLINGS;
|
|---|
| 67 |
|
|---|
| 68 | if (skip_subtree && strcmp (name + f->base, skip_subtree) == 0)
|
|---|
| 69 | return FTW_SKIP_SUBTREE;
|
|---|
| 70 | #endif
|
|---|
| 71 |
|
|---|
| 72 | return do_exit ? 26 : FTW_CONTINUE;
|
|---|
| 73 | }
|
|---|
| 74 |
|
|---|
| 75 | int
|
|---|
| 76 | main (int argc, char *argv[])
|
|---|
| 77 | {
|
|---|
| 78 | int opt;
|
|---|
| 79 | int r;
|
|---|
| 80 | int flag = 0;
|
|---|
| 81 | mtrace ();
|
|---|
| 82 |
|
|---|
| 83 | while ((opt = getopt_long_only (argc, argv, "", options, NULL)) != -1)
|
|---|
| 84 | {
|
|---|
| 85 | if (opt == 't')
|
|---|
| 86 | skip_subtree = optarg;
|
|---|
| 87 | else if (opt == 's')
|
|---|
| 88 | skip_siblings = optarg;
|
|---|
| 89 | }
|
|---|
| 90 |
|
|---|
| 91 | if (do_chdir)
|
|---|
| 92 | flag |= FTW_CHDIR;
|
|---|
| 93 | if (do_depth)
|
|---|
| 94 | flag |= FTW_DEPTH;
|
|---|
| 95 | if (do_phys)
|
|---|
| 96 | flag |= FTW_PHYS;
|
|---|
| 97 | if (skip_subtree || skip_siblings)
|
|---|
| 98 | {
|
|---|
| 99 | #ifdef FTW_ACTIONRETVAL
|
|---|
| 100 | flag |= FTW_ACTIONRETVAL;
|
|---|
| 101 | #endif
|
|---|
| 102 | if (do_exit)
|
|---|
| 103 | {
|
|---|
| 104 | printf ("--early-exit cannot be used together with --skip-{siblings,subtree}");
|
|---|
| 105 | exit (1);
|
|---|
| 106 | }
|
|---|
| 107 | }
|
|---|
| 108 |
|
|---|
| 109 | char *cw1 = getcwd (NULL, 0);
|
|---|
| 110 |
|
|---|
| 111 | r = nftw (optind < argc ? argv[optind] : ".", cb, do_exit ? 1 : 3, flag);
|
|---|
| 112 | if (r < 0)
|
|---|
| 113 | perror ("nftw");
|
|---|
| 114 |
|
|---|
| 115 | char *cw2 = getcwd (NULL, 0);
|
|---|
| 116 |
|
|---|
| 117 | if (strcmp (cw1, cw2) != 0)
|
|---|
| 118 | {
|
|---|
| 119 | printf ("current working directory before and after nftw call differ:\n"
|
|---|
| 120 | "before: %s\n"
|
|---|
| 121 | "after: %s\n", cw1, cw2);
|
|---|
| 122 | exit (1);
|
|---|
| 123 | }
|
|---|
| 124 |
|
|---|
| 125 | if (do_exit)
|
|---|
| 126 | {
|
|---|
| 127 | puts (r == 26 ? "succeeded" : "failed");
|
|---|
| 128 | return r == 26 ? 0 : 1;
|
|---|
| 129 | }
|
|---|
| 130 | return r;
|
|---|
| 131 | }
|
|---|