mirror of
git://sourceware.org/git/glibc.git
synced 2024-11-27 03:41:23 +08:00
Update.
* io/Makefile (test-srcs): Add ftwtest. (distribute): Add ftwtest-sh. (tests): Call ftwtest-sh for this goal. * io/ftwtest-sh: New file. Sets up test environment, calls test program and compares the result. * io/ftwtest.c: Test program for ftw. * misc/search.h: Add comments. Declare tdestroy. * misc/tsearch.c (tdestroy): New function.
This commit is contained in:
parent
76b87c039b
commit
d951286f64
@ -14,6 +14,7 @@ gpl2lgpl.sed
|
|||||||
distinfo
|
distinfo
|
||||||
distinfo
|
distinfo
|
||||||
|
|
||||||
|
test-include
|
||||||
analysis
|
analysis
|
||||||
docs
|
docs
|
||||||
|
|
||||||
|
@ -2,6 +2,15 @@
|
|||||||
|
|
||||||
* io/ftw.c: Complete rewrite. Add implementation of `nftw'.
|
* io/ftw.c: Complete rewrite. Add implementation of `nftw'.
|
||||||
* io/ftw.h: Update for new implementation and XPG4.2.
|
* io/ftw.h: Update for new implementation and XPG4.2.
|
||||||
|
* io/Makefile (test-srcs): Add ftwtest.
|
||||||
|
(distribute): Add ftwtest-sh.
|
||||||
|
(tests): Call ftwtest-sh for this goal.
|
||||||
|
* io/ftwtest-sh: New file. Sets up test environment, calls test
|
||||||
|
program and compares the result.
|
||||||
|
* io/ftwtest.c: Test program for ftw.
|
||||||
|
|
||||||
|
* misc/search.h: Add comments. Declare tdestroy.
|
||||||
|
* misc/tsearch.c (tdestroy): New function.
|
||||||
|
|
||||||
* login/Makefile: Update for UTMP daemon implementation.
|
* login/Makefile: Update for UTMP daemon implementation.
|
||||||
|
|
||||||
|
@ -51,9 +51,15 @@ routines := \
|
|||||||
static-only-routines = stat fstat lstat mknod
|
static-only-routines = stat fstat lstat mknod
|
||||||
|
|
||||||
others := pwd
|
others := pwd
|
||||||
|
test-srcs := ftwtest
|
||||||
tests := test-utime
|
tests := test-utime
|
||||||
|
|
||||||
|
distribute := ftwtest-sh
|
||||||
|
|
||||||
include ../Rules
|
include ../Rules
|
||||||
|
|
||||||
CFLAGS-fts.c = -Wno-uninitialized
|
CFLAGS-fts.c = -Wno-uninitialized
|
||||||
CFLAGS-ftw.c = -Wno-uninitialized
|
CFLAGS-ftw.c = -Wno-uninitialized
|
||||||
|
|
||||||
|
tests: $(objpfx)ftwtest
|
||||||
|
$(SHELL) -e ftwtest-sh $(common-objpfx) $<
|
||||||
|
191
io/ftw.c
191
io/ftw.c
@ -38,24 +38,44 @@ struct dir_data
|
|||||||
char *content;
|
char *content;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct known_object
|
||||||
|
{
|
||||||
|
dev_t dev;
|
||||||
|
ino_t ino;
|
||||||
|
};
|
||||||
|
|
||||||
struct ftw_data
|
struct ftw_data
|
||||||
{
|
{
|
||||||
|
/* Array with pointers to open directory streams. */
|
||||||
struct dir_data **dirstreams;
|
struct dir_data **dirstreams;
|
||||||
size_t actdir;
|
size_t actdir;
|
||||||
size_t maxdir;
|
size_t maxdir;
|
||||||
|
|
||||||
|
/* Buffer containing name of currently processed object. */
|
||||||
char *dirbuf;
|
char *dirbuf;
|
||||||
size_t dirbufsize;
|
size_t dirbufsize;
|
||||||
|
|
||||||
|
/* Passed as fourth argument to `nftw' callback. The `base' member
|
||||||
|
tracks the content of the `dirbuf'. */
|
||||||
struct FTW ftw;
|
struct FTW ftw;
|
||||||
|
|
||||||
|
/* Flags passed to `nftw' function. 0 for `ftw'. */
|
||||||
int flags;
|
int flags;
|
||||||
|
|
||||||
|
/* Conversion array for flag values. It is the identity mapping for
|
||||||
|
`nftw' calls, otherwise it maps the values to those know by
|
||||||
|
`ftw'. */
|
||||||
int *cvt_arr;
|
int *cvt_arr;
|
||||||
|
|
||||||
|
/* Callback function. We always use the `nftw' form. */
|
||||||
__nftw_func_t func;
|
__nftw_func_t func;
|
||||||
|
|
||||||
struct stat st;
|
/* Device of starting point. Needed for FTW_MOUNT. */
|
||||||
|
|
||||||
dev_t dev;
|
dev_t dev;
|
||||||
|
|
||||||
|
/* Data structure for keeping fingerprints of already processed
|
||||||
|
object. This is needed when not using FTW_PHYS. */
|
||||||
|
void *known_objects;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -74,7 +94,36 @@ static int ftw_arr[] =
|
|||||||
|
|
||||||
|
|
||||||
/* Forward declarations of local functions. */
|
/* Forward declarations of local functions. */
|
||||||
static int ftw_dir (struct ftw_data *data);
|
static int ftw_dir (struct ftw_data *data, struct stat *st);
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
object_compare (const void *p1, const void *p2)
|
||||||
|
{
|
||||||
|
/* We don't need a sophisticated and useful comparison. We are only
|
||||||
|
interested in equality. */
|
||||||
|
return memcmp (p1, p2, sizeof (struct known_object));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
add_object (struct ftw_data *data, struct stat *st)
|
||||||
|
{
|
||||||
|
struct known_object *newp = malloc (sizeof (struct known_object));
|
||||||
|
if (newp == NULL)
|
||||||
|
return -1;
|
||||||
|
newp->dev = st->st_dev;
|
||||||
|
newp->ino = st->st_ino;
|
||||||
|
return __tsearch (newp, &data->known_objects, object_compare) ? 0 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
find_object (struct ftw_data *data, struct stat *st)
|
||||||
|
{
|
||||||
|
struct known_object obj = { dev: st->st_dev, ino: st->st_ino };
|
||||||
|
return __tfind (&obj, &data->known_objects, object_compare) != NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static inline int
|
static inline int
|
||||||
@ -170,6 +219,7 @@ static inline int
|
|||||||
process_entry (struct ftw_data *data, struct dir_data *dir, const char *name,
|
process_entry (struct ftw_data *data, struct dir_data *dir, const char *name,
|
||||||
size_t namlen)
|
size_t namlen)
|
||||||
{
|
{
|
||||||
|
struct stat st;
|
||||||
int result = 0;
|
int result = 0;
|
||||||
int flag;
|
int flag;
|
||||||
|
|
||||||
@ -193,65 +243,72 @@ process_entry (struct ftw_data *data, struct dir_data *dir, const char *name,
|
|||||||
memcpy (data->dirbuf + data->ftw.base, name, namlen);
|
memcpy (data->dirbuf + data->ftw.base, name, namlen);
|
||||||
data->dirbuf[data->ftw.base + namlen] = '\0';
|
data->dirbuf[data->ftw.base + namlen] = '\0';
|
||||||
|
|
||||||
if (((data->flags & FTW_PHYS) ? lstat : stat) (data->dirbuf, &data->st) < 0)
|
if (((data->flags & FTW_PHYS) ? lstat : stat) (data->dirbuf, &st) < 0)
|
||||||
{
|
{
|
||||||
if (errno != EACCES && errno != ENOENT)
|
if (errno != EACCES && errno != ENOENT)
|
||||||
result = -1;
|
result = -1;
|
||||||
else if (!(data->flags & FTW_PHYS)
|
else if (!(data->flags & FTW_PHYS)
|
||||||
&& lstat (data->dirbuf, &data->st) == 0
|
&& lstat (data->dirbuf, &st) == 0
|
||||||
&& S_ISLNK (data->st.st_mode))
|
&& S_ISLNK (st.st_mode))
|
||||||
flag = FTW_SLN;
|
flag = FTW_SLN;
|
||||||
else
|
else
|
||||||
flag = FTW_NS;
|
flag = FTW_NS;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (S_ISDIR (data->st.st_mode))
|
if (S_ISDIR (st.st_mode))
|
||||||
flag = FTW_D;
|
flag = FTW_D;
|
||||||
else if (S_ISLNK (data->st.st_mode))
|
else if (S_ISLNK (st.st_mode))
|
||||||
flag = FTW_SL;
|
flag = FTW_SL;
|
||||||
else
|
else
|
||||||
flag = FTW_F;
|
flag = FTW_F;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (result == 0
|
if (result == 0
|
||||||
&& (!(data->flags & FTW_MOUNT) || data->st.st_dev == data->dev))
|
&& (!(data->flags & FTW_MOUNT) || flag == FTW_NS
|
||||||
|
|| st.st_dev == data->dev))
|
||||||
{
|
{
|
||||||
if (flag == FTW_D)
|
if ((data->flags & FTW_PHYS) || flag == FTW_NS
|
||||||
|
|| (!find_object (data, &st)
|
||||||
|
/* Remember the object. */
|
||||||
|
&& (result = add_object (data, &st)) == 0))
|
||||||
{
|
{
|
||||||
result = ftw_dir (data);
|
if (flag == FTW_D)
|
||||||
|
|
||||||
if (result == 0 && (data->flags & FTW_CHDIR))
|
|
||||||
{
|
{
|
||||||
/* Change back to current directory. */
|
result = ftw_dir (data, &st);
|
||||||
int done = 0;
|
|
||||||
if (dir->stream != NULL)
|
|
||||||
if (fchdir (dirfd (dir->stream)) == 0)
|
|
||||||
done = 1;
|
|
||||||
|
|
||||||
if (!done)
|
if (result == 0 && (data->flags & FTW_CHDIR))
|
||||||
{
|
{
|
||||||
if (data->ftw.base == 1)
|
/* Change back to current directory. */
|
||||||
{
|
int done = 0;
|
||||||
if (chdir ("/") < 0)
|
if (dir->stream != NULL)
|
||||||
result = -1;
|
if (fchdir (dirfd (dir->stream)) == 0)
|
||||||
}
|
done = 1;
|
||||||
else
|
|
||||||
{
|
|
||||||
/* Please note that we overwrite a slash. */
|
|
||||||
data->dirbuf[data->ftw.base - 1] = '\0';
|
|
||||||
|
|
||||||
if (chdir (data->dirbuf) < 0)
|
if (!done)
|
||||||
result = -1;
|
{
|
||||||
|
if (data->ftw.base == 1)
|
||||||
|
{
|
||||||
|
if (chdir ("/") < 0)
|
||||||
|
result = -1;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
data->dirbuf[data->ftw.base - 1] = '/';
|
{
|
||||||
|
/* Please note that we overwrite a slash. */
|
||||||
|
data->dirbuf[data->ftw.base - 1] = '\0';
|
||||||
|
|
||||||
|
if (chdir (data->dirbuf) < 0)
|
||||||
|
result = -1;
|
||||||
|
|
||||||
|
data->dirbuf[data->ftw.base - 1] = '/';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
result = (*data->func) (data->dirbuf, &st, data->cvt_arr[flag],
|
||||||
|
&data->ftw);
|
||||||
}
|
}
|
||||||
else
|
|
||||||
result = (*data->func) (data->dirbuf, &data->st, data->cvt_arr[flag],
|
|
||||||
&data->ftw);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
@ -259,27 +316,33 @@ process_entry (struct ftw_data *data, struct dir_data *dir, const char *name,
|
|||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
ftw_dir (struct ftw_data *data)
|
ftw_dir (struct ftw_data *data, struct stat *st)
|
||||||
{
|
{
|
||||||
struct dir_data dir;
|
struct dir_data dir;
|
||||||
struct dirent *d;
|
struct dirent *d;
|
||||||
int previous_base = data->ftw.base;
|
int previous_base = data->ftw.base;
|
||||||
int result = 0;
|
int result;
|
||||||
char *startp;
|
char *startp;
|
||||||
|
|
||||||
/* First, report the directory (if not depth-first). */
|
|
||||||
if (!(data->flags & FTW_DEPTH))
|
|
||||||
{
|
|
||||||
result = (*data->func) (data->dirbuf, &data->st, FTW_D, &data->ftw);
|
|
||||||
if (result != 0)
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Open the stream for this directory. This might require that
|
/* Open the stream for this directory. This might require that
|
||||||
another stream has to be closed. */
|
another stream has to be closed. */
|
||||||
result = open_dir_stream (data, &dir);
|
result = open_dir_stream (data, &dir);
|
||||||
if (result != 0)
|
if (result != 0)
|
||||||
return result;
|
{
|
||||||
|
if (errno == EACCES)
|
||||||
|
/* We cannot read the directory. Signal this with a special flag. */
|
||||||
|
result = (*data->func) (data->dirbuf, st, FTW_DNR, &data->ftw);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* First, report the directory (if not depth-first). */
|
||||||
|
if (!(data->flags & FTW_DEPTH))
|
||||||
|
{
|
||||||
|
result = (*data->func) (data->dirbuf, st, FTW_D, &data->ftw);
|
||||||
|
if (result != 0)
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
/* If necessary, change to this directory. */
|
/* If necessary, change to this directory. */
|
||||||
if (data->flags & FTW_CHDIR)
|
if (data->flags & FTW_CHDIR)
|
||||||
@ -361,12 +424,13 @@ ftw_dir (struct ftw_data *data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Prepare the return, revert the `struct FTW' information. */
|
/* Prepare the return, revert the `struct FTW' information. */
|
||||||
|
data->dirbuf[data->ftw.base - 1] = '\0';
|
||||||
--data->ftw.level;
|
--data->ftw.level;
|
||||||
data->ftw.base = previous_base;
|
data->ftw.base = previous_base;
|
||||||
|
|
||||||
/* Finally, if we process depth-first report the directory. */
|
/* Finally, if we process depth-first report the directory. */
|
||||||
if (result == 0 && (data->flags & FTW_DEPTH))
|
if (result == 0 && (data->flags & FTW_DEPTH))
|
||||||
result = (*data->func) (data->dirbuf, &data->st, FTW_DP, &data->ftw);
|
result = (*data->func) (data->dirbuf, st, FTW_DP, &data->ftw);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -377,9 +441,10 @@ ftw_startup (const char *dir, int is_nftw, void *func, int descriptors,
|
|||||||
int flags)
|
int flags)
|
||||||
{
|
{
|
||||||
struct ftw_data data;
|
struct ftw_data data;
|
||||||
|
struct stat st;
|
||||||
int result = 0;
|
int result = 0;
|
||||||
int save_err;
|
int save_err;
|
||||||
char *cwd;
|
char *cwd = NULL;
|
||||||
char *cp;
|
char *cp;
|
||||||
|
|
||||||
/* First make sure the parameters are reasonable. */
|
/* First make sure the parameters are reasonable. */
|
||||||
@ -399,7 +464,7 @@ ftw_startup (const char *dir, int is_nftw, void *func, int descriptors,
|
|||||||
data.dirbuf = (char *) malloc (data.dirbufsize);
|
data.dirbuf = (char *) malloc (data.dirbufsize);
|
||||||
if (data.dirbuf == NULL)
|
if (data.dirbuf == NULL)
|
||||||
return -1;
|
return -1;
|
||||||
cp = stpcpy (data.dirbuf, dir);
|
cp = __stpcpy (data.dirbuf, dir);
|
||||||
/* Strip trailing slashes. */
|
/* Strip trailing slashes. */
|
||||||
while (cp > data.dirbuf + 1 && cp[-1] == '/')
|
while (cp > data.dirbuf + 1 && cp[-1] == '/')
|
||||||
--cp;
|
--cp;
|
||||||
@ -426,6 +491,9 @@ ftw_startup (const char *dir, int is_nftw, void *func, int descriptors,
|
|||||||
to reduce the value range before calling a `ftw' callback. */
|
to reduce the value range before calling a `ftw' callback. */
|
||||||
data.cvt_arr = is_nftw ? nftw_arr : ftw_arr;
|
data.cvt_arr = is_nftw ? nftw_arr : ftw_arr;
|
||||||
|
|
||||||
|
/* No object known so far. */
|
||||||
|
data.known_objects = NULL;
|
||||||
|
|
||||||
/* Now go to the directory containing the initial file/directory. */
|
/* Now go to the directory containing the initial file/directory. */
|
||||||
if ((flags & FTW_CHDIR) && data.ftw.base > 0)
|
if ((flags & FTW_CHDIR) && data.ftw.base > 0)
|
||||||
{
|
{
|
||||||
@ -453,14 +521,14 @@ ftw_startup (const char *dir, int is_nftw, void *func, int descriptors,
|
|||||||
|
|
||||||
/* Get stat info for start directory. */
|
/* Get stat info for start directory. */
|
||||||
if (result == 0)
|
if (result == 0)
|
||||||
if (((flags & FTW_PHYS) ? lstat : stat) (data.dirbuf, &data.st) < 0)
|
if (((flags & FTW_PHYS) ? lstat : stat) (data.dirbuf, &st) < 0)
|
||||||
{
|
{
|
||||||
if (errno == EACCES)
|
if (errno == EACCES)
|
||||||
result = (*data.func) (data.dirbuf, &data.st, FTW_NS, &data.ftw);
|
result = (*data.func) (data.dirbuf, &st, FTW_NS, &data.ftw);
|
||||||
else if (!(flags & FTW_PHYS)
|
else if (!(flags & FTW_PHYS)
|
||||||
&& errno == ENOENT
|
&& errno == ENOENT
|
||||||
&& lstat (dir, &data.st) == 0 && S_ISLNK (data.st.st_mode))
|
&& lstat (dir, &st) == 0 && S_ISLNK (st.st_mode))
|
||||||
result = (*data.func) (data.dirbuf, &data.st, data.cvt_arr[FTW_SLN],
|
result = (*data.func) (data.dirbuf, &st, data.cvt_arr[FTW_SLN],
|
||||||
&data.ftw);
|
&data.ftw);
|
||||||
else
|
else
|
||||||
/* No need to call the callback since we cannot say anything
|
/* No need to call the callback since we cannot say anything
|
||||||
@ -469,16 +537,24 @@ ftw_startup (const char *dir, int is_nftw, void *func, int descriptors,
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (S_ISDIR (data.st.st_mode))
|
if (S_ISDIR (st.st_mode))
|
||||||
{
|
{
|
||||||
data.dev = data.st.st_dev;
|
/* Remember the device of the initial directory in case
|
||||||
result = ftw_dir (&data);
|
FTW_MOUNT is given. */
|
||||||
|
data.dev = st.st_dev;
|
||||||
|
|
||||||
|
/* We know this directory now. */
|
||||||
|
if (!(flags & FTW_PHYS))
|
||||||
|
result = add_object (&data, &st);
|
||||||
|
|
||||||
|
if (result == 0)
|
||||||
|
result = ftw_dir (&data, &st);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
int flag = S_ISLNK (data.st.st_mode) ? FTW_SL : FTW_F;
|
int flag = S_ISLNK (st.st_mode) ? FTW_SL : FTW_F;
|
||||||
|
|
||||||
result = (*data.func) (data.dirbuf, &data.st, data.cvt_arr[flag],
|
result = (*data.func) (data.dirbuf, &st, data.cvt_arr[flag],
|
||||||
&data.ftw);
|
&data.ftw);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -494,6 +570,7 @@ ftw_startup (const char *dir, int is_nftw, void *func, int descriptors,
|
|||||||
|
|
||||||
/* Free all memory. */
|
/* Free all memory. */
|
||||||
save_err = errno;
|
save_err = errno;
|
||||||
|
__tdestroy (data.known_objects, free);
|
||||||
free (data.dirbuf);
|
free (data.dirbuf);
|
||||||
__set_errno (save_err);
|
__set_errno (save_err);
|
||||||
|
|
||||||
|
4
io/ftw.h
4
io/ftw.h
@ -44,11 +44,13 @@ enum
|
|||||||
FTW_NS, /* Unstatable file. */
|
FTW_NS, /* Unstatable file. */
|
||||||
#define FTW_NS FTW_NS
|
#define FTW_NS FTW_NS
|
||||||
|
|
||||||
#ifdef __USE_XOPEN_EXTENDED
|
#if defined __USE_BSD || defined __USE_XOPEN_EXTENDED
|
||||||
|
|
||||||
FTW_SL, /* Symbolic link. */
|
FTW_SL, /* Symbolic link. */
|
||||||
# define FTW_SL FTW_SL
|
# define FTW_SL FTW_SL
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __USE_XOPEN_EXTENDED
|
||||||
/* These flags are only passed from the `nftw' function. */
|
/* These flags are only passed from the `nftw' function. */
|
||||||
FTW_DP, /* Directory, all subdirs have been visited. */
|
FTW_DP, /* Directory, all subdirs have been visited. */
|
||||||
# define FTW_DP FTW_DP
|
# define FTW_DP FTW_DP
|
||||||
|
84
io/ftwtest-sh
Normal file
84
io/ftwtest-sh
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
#! /bin/sh
|
||||||
|
|
||||||
|
# The common objpfx, used to find libraries and the dynamic loader.
|
||||||
|
objpfx=$1
|
||||||
|
|
||||||
|
# We expect one parameter which is the test program. This must understand
|
||||||
|
# a number options:
|
||||||
|
# --phys use the FTW_PHYS flag
|
||||||
|
# --chdir use the FTW_CHDIR and print the current directory in the
|
||||||
|
# callback
|
||||||
|
# --depth use the FTW_DEPTH flag
|
||||||
|
testprogram=$2
|
||||||
|
|
||||||
|
|
||||||
|
# First create our scenario:
|
||||||
|
tmp=${TMPDIR:-/tmp}
|
||||||
|
tmpdir=$tmp/ftwtest.d
|
||||||
|
|
||||||
|
trap 'rm -fr $tmpdir $testout' 1 2 3 15
|
||||||
|
|
||||||
|
if test -d $tmpdir; then
|
||||||
|
chmod -R a+x $tmpdir
|
||||||
|
rm -fr $tmpdir
|
||||||
|
fi
|
||||||
|
mkdir $tmpdir
|
||||||
|
mkdir $tmpdir/foo
|
||||||
|
mkdir $tmpdir/bar
|
||||||
|
echo > $tmpdir/baz
|
||||||
|
mkdir $tmpdir/foo/lvl1
|
||||||
|
echo > $tmpdir/foo/lvl1/file@1
|
||||||
|
mkdir $tmpdir/foo/lvl1/lvl2
|
||||||
|
echo > $tmpdir/foo/lvl1/lvl2/file@2
|
||||||
|
mkdir $tmpdir/foo/lvl1/lvl2/lvl3
|
||||||
|
echo > $tmpdir/foo/lvl1/lvl2/lvl3/file@3
|
||||||
|
ln -s $tmpdir $tmpdir/foo/lvl1/lvl2/lvl3/link@3
|
||||||
|
ln -s $tmpdir/foo/lvl1/lvl2 $tmpdir/foo/lvl1/lvl2/link@2
|
||||||
|
ln -s $tmpdir/foo/lvl1/lvl2/lvl3/lvl4 $tmpdir/foo/lvl1/link@1
|
||||||
|
echo > $tmpdir/bar/xo
|
||||||
|
chmod a-x,a+r $tmpdir/bar
|
||||||
|
|
||||||
|
testout=${TMPDIR:-/tmp}/ftwtest.out
|
||||||
|
LD_LIBRARY_PATH=$objpfx $objpfx/elf/ld.so $testprogram $tmpdir |
|
||||||
|
sort > $testout
|
||||||
|
|
||||||
|
cat <<EOF | cmp $testout - || exit 1
|
||||||
|
base = "$tmp/", file = "ftwtest.d", flag = FTW_D
|
||||||
|
base = "$tmp/ftwtest.d/", file = "bar", flag = FTW_D
|
||||||
|
base = "$tmp/ftwtest.d/", file = "baz", flag = FTW_F
|
||||||
|
base = "$tmp/ftwtest.d/", file = "foo", flag = FTW_D
|
||||||
|
base = "$tmp/ftwtest.d/bar/", file = "xo", flag = FTW_NS
|
||||||
|
base = "$tmp/ftwtest.d/foo/", file = "lvl1", flag = FTW_D
|
||||||
|
base = "$tmp/ftwtest.d/foo/lvl1/", file = "file@1", flag = FTW_F
|
||||||
|
base = "$tmp/ftwtest.d/foo/lvl1/", file = "link@1", flag = FTW_SLN
|
||||||
|
base = "$tmp/ftwtest.d/foo/lvl1/", file = "lvl2", flag = FTW_D
|
||||||
|
base = "$tmp/ftwtest.d/foo/lvl1/lvl2/", file = "file@2", flag = FTW_F
|
||||||
|
base = "$tmp/ftwtest.d/foo/lvl1/lvl2/", file = "lvl3", flag = FTW_D
|
||||||
|
base = "$tmp/ftwtest.d/foo/lvl1/lvl2/lvl3/", file = "file@3", flag = FTW_F
|
||||||
|
EOF
|
||||||
|
rm $testout
|
||||||
|
|
||||||
|
testout=${TMPDIR:-/tmp}/ftwtest.out
|
||||||
|
LD_LIBRARY_PATH=$objpfx $objpfx/elf/ld.so $testprogram --depth $tmpdir |
|
||||||
|
sort > $testout
|
||||||
|
|
||||||
|
cat <<EOF | cmp $testout - || exit 1
|
||||||
|
base = "/tmp/", file = "ftwtest.d", flag = FTW_DP
|
||||||
|
base = "/tmp/ftwtest.d/", file = "bar", flag = FTW_DP
|
||||||
|
base = "/tmp/ftwtest.d/", file = "baz", flag = FTW_F
|
||||||
|
base = "/tmp/ftwtest.d/", file = "foo", flag = FTW_DP
|
||||||
|
base = "/tmp/ftwtest.d/bar/", file = "xo", flag = FTW_NS
|
||||||
|
base = "/tmp/ftwtest.d/foo/", file = "lvl1", flag = FTW_DP
|
||||||
|
base = "/tmp/ftwtest.d/foo/lvl1/", file = "file@1", flag = FTW_F
|
||||||
|
base = "/tmp/ftwtest.d/foo/lvl1/", file = "link@1", flag = FTW_SLN
|
||||||
|
base = "/tmp/ftwtest.d/foo/lvl1/", file = "lvl2", flag = FTW_DP
|
||||||
|
base = "/tmp/ftwtest.d/foo/lvl1/lvl2/", file = "file@2", flag = FTW_F
|
||||||
|
base = "/tmp/ftwtest.d/foo/lvl1/lvl2/", file = "lvl3", flag = FTW_DP
|
||||||
|
base = "/tmp/ftwtest.d/foo/lvl1/lvl2/lvl3/", file = "file@3", flag = FTW_F
|
||||||
|
EOF
|
||||||
|
rm $testout
|
||||||
|
|
||||||
|
chmod -R a+x $tmpdir
|
||||||
|
rm -fr $tmpdir
|
||||||
|
|
||||||
|
exit 0
|
71
io/ftwtest.c
Normal file
71
io/ftwtest.c
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
#include <ftw.h>
|
||||||
|
#include <getopt.h>
|
||||||
|
#include <mcheck.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
|
||||||
|
int do_depth;
|
||||||
|
int do_chdir;
|
||||||
|
int do_phys;
|
||||||
|
|
||||||
|
struct option options[] =
|
||||||
|
{
|
||||||
|
{ "depth", no_argument, &do_depth, 1 },
|
||||||
|
{ "chdir", no_argument, &do_chdir, 1 },
|
||||||
|
{ "phys", no_argument, &do_phys, 1 },
|
||||||
|
{ NULL, 0, NULL, 0 }
|
||||||
|
};
|
||||||
|
|
||||||
|
const char *flag2name[] =
|
||||||
|
{
|
||||||
|
[FTW_F] = "FTW_F",
|
||||||
|
[FTW_D] = "FTW_D",
|
||||||
|
[FTW_DNR] = "FTW_DNR",
|
||||||
|
[FTW_NS] = "FTW_NS",
|
||||||
|
[FTW_SL] = "FTW_SL",
|
||||||
|
[FTW_DP] = "FTW_DP",
|
||||||
|
[FTW_SLN] = "FTW_SLN"
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
int
|
||||||
|
cb (const char *name, const struct stat *st, int flag, struct FTW *f)
|
||||||
|
{
|
||||||
|
printf ("base = \"%.*s\", file = \"%s\", flag = %s",
|
||||||
|
f->base, name, name + f->base, flag2name[flag]);
|
||||||
|
if (do_chdir)
|
||||||
|
{
|
||||||
|
char *cwd = getcwd (NULL, 0);
|
||||||
|
printf (", cwd = %s", cwd);
|
||||||
|
free (cwd);
|
||||||
|
}
|
||||||
|
puts ("");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
main (int argc, char *argv[])
|
||||||
|
{
|
||||||
|
int opt;
|
||||||
|
int r;
|
||||||
|
int flag = 0;
|
||||||
|
mtrace ();
|
||||||
|
|
||||||
|
while ((opt = getopt_long_only (argc, argv, "", options, NULL)) != -1)
|
||||||
|
;
|
||||||
|
|
||||||
|
if (do_chdir)
|
||||||
|
flag |= FTW_CHDIR;
|
||||||
|
if (do_depth)
|
||||||
|
flag |= FTW_DEPTH;
|
||||||
|
if (do_phys)
|
||||||
|
flag |= FTW_PHYS;
|
||||||
|
|
||||||
|
r = nftw (optind < argc ? argv[optind] : ".", cb, 3, flag);
|
||||||
|
if (r)
|
||||||
|
perror ("nftw");
|
||||||
|
return r;
|
||||||
|
}
|
@ -106,16 +106,21 @@ typedef enum
|
|||||||
}
|
}
|
||||||
VISIT;
|
VISIT;
|
||||||
|
|
||||||
|
/* Search for an entry matching the given KEY in the tree pointed to
|
||||||
|
by *ROOTP and insert a new element if not found. */
|
||||||
extern void *tsearch __P ((__const void * __key, void **__rootp,
|
extern void *tsearch __P ((__const void * __key, void **__rootp,
|
||||||
__compar_fn_t compar));
|
__compar_fn_t compar));
|
||||||
extern void *__tsearch __P ((__const void * __key, void **__rootp,
|
extern void *__tsearch __P ((__const void * __key, void **__rootp,
|
||||||
__compar_fn_t compar));
|
__compar_fn_t compar));
|
||||||
|
|
||||||
extern void *tfind __P ((__const void * __key, __const void ** __rootp,
|
/* Search for an entry matching the given KEY in the tree pointed to
|
||||||
|
by *ROOTP. If no matching entry is available return NULL. */
|
||||||
|
extern void *tfind __P ((__const void * __key, void *__const * __rootp,
|
||||||
__compar_fn_t compar));
|
__compar_fn_t compar));
|
||||||
extern void *__tfind __P ((__const void * __key, __const void ** __rootp,
|
extern void *__tfind __P ((__const void * __key, void *__const * __rootp,
|
||||||
__compar_fn_t compar));
|
__compar_fn_t compar));
|
||||||
|
|
||||||
|
/* Remove the element matching KEY from the tree pointed to by *ROOTP. */
|
||||||
extern void *tdelete __P ((__const void * __key, void ** __rootp,
|
extern void *tdelete __P ((__const void * __key, void ** __rootp,
|
||||||
__compar_fn_t compar));
|
__compar_fn_t compar));
|
||||||
extern void *__tdelete __P ((__const void * __key, void ** __rootp,
|
extern void *__tdelete __P ((__const void * __key, void ** __rootp,
|
||||||
@ -128,10 +133,22 @@ typedef void (*__action_fn_t) __P ((__const void *__nodep,
|
|||||||
int __level));
|
int __level));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Walk through the whole tree and call the ACTION callback for every node
|
||||||
|
or leaf. */
|
||||||
extern void twalk __P ((__const void * __root, __action_fn_t action));
|
extern void twalk __P ((__const void * __root, __action_fn_t action));
|
||||||
|
|
||||||
extern void __twalk __P ((__const void * __root, __action_fn_t action));
|
extern void __twalk __P ((__const void * __root, __action_fn_t action));
|
||||||
|
|
||||||
|
#ifdef __USE_GNU
|
||||||
|
/* Callback type for function to free a tree node. If the keys are atomic
|
||||||
|
data this function should do nothing. */
|
||||||
|
typedef void (*__free_fn_t) __P ((void *__nodep));
|
||||||
|
|
||||||
|
/* Destroy the whole tree, call FREEFCT for each node or leaf. */
|
||||||
|
extern void __tdestroy __P ((void *__root, __free_fn_t freefct));
|
||||||
|
extern void tdestroy __P ((void *__root, __free_fn_t freefct));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
/* Perform linear search for KEY by comparing by COMPAR in an array
|
/* Perform linear search for KEY by comparing by COMPAR in an array
|
||||||
[BASE,BASE+NMEMB*SIZE). */
|
[BASE,BASE+NMEMB*SIZE). */
|
||||||
|
@ -300,7 +300,7 @@ weak_alias (__tsearch, tsearch)
|
|||||||
void *
|
void *
|
||||||
__tfind (key, vrootp, compar)
|
__tfind (key, vrootp, compar)
|
||||||
const void *key;
|
const void *key;
|
||||||
const void **vrootp;
|
void *const *vrootp;
|
||||||
__compar_fn_t compar;
|
__compar_fn_t compar;
|
||||||
{
|
{
|
||||||
node *rootp = (node *) vrootp;
|
node *rootp = (node *) vrootp;
|
||||||
@ -625,3 +625,36 @@ __twalk (const void *vroot, __action_fn_t action)
|
|||||||
trecurse (root, action, 0);
|
trecurse (root, action, 0);
|
||||||
}
|
}
|
||||||
weak_alias (__twalk, twalk)
|
weak_alias (__twalk, twalk)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* The standardized functions miss an important functionality: the
|
||||||
|
tree cannot be removed easily. We provide a function to do this. */
|
||||||
|
static void
|
||||||
|
tdestroy_recurse (node root, __free_fn_t freefct)
|
||||||
|
{
|
||||||
|
if (root->left == NULL && root->right == NULL)
|
||||||
|
(*freefct) (root->key);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (root->left != NULL)
|
||||||
|
tdestroy_recurse (root->left, freefct);
|
||||||
|
if (root->right != NULL)
|
||||||
|
tdestroy_recurse (root->right, freefct);
|
||||||
|
(*freefct) (root->key);
|
||||||
|
}
|
||||||
|
/* Free the node itself. */
|
||||||
|
free (root);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
__tdestroy (void *vroot, __free_fn_t freefct)
|
||||||
|
{
|
||||||
|
node root = (node) vroot;
|
||||||
|
|
||||||
|
CHECK_TREE (root);
|
||||||
|
|
||||||
|
if (root != NULL)
|
||||||
|
tdestroy_recurse (root, freefct);
|
||||||
|
}
|
||||||
|
weak_alias (__tdestroy, tdestroy)
|
||||||
|
Loading…
Reference in New Issue
Block a user