d921b2ecc0
things like xasprintf() into xfuncs.c, remove xprint_file_by_name() (it only had one user), clean up lots of #includes... General cleanup pass. What I've been doing for the last couple days. And it conflicts! I've removed httpd.c from this checkin due to somebody else touching that file. It builds for me. I have to catch a bus. (Now you know why I'm looking forward to Mercurial.)
114 lines
2.6 KiB
C
114 lines
2.6 KiB
C
/* vi: set sw=4 ts=4: */
|
|
/*
|
|
* Utility routines.
|
|
*
|
|
* Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
|
|
*
|
|
* Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
|
|
*/
|
|
|
|
#include "libbb.h"
|
|
|
|
#undef DEBUG_RECURS_ACTION
|
|
|
|
|
|
/*
|
|
* Walk down all the directories under the specified
|
|
* location, and do something (something specified
|
|
* by the fileAction and dirAction function pointers).
|
|
*
|
|
* Unfortunately, while nftw(3) could replace this and reduce
|
|
* code size a bit, nftw() wasn't supported before GNU libc 2.1,
|
|
* and so isn't sufficiently portable to take over since glibc2.1
|
|
* is so stinking huge.
|
|
*/
|
|
int recursive_action(const char *fileName,
|
|
int recurse, int followLinks, int depthFirst,
|
|
int (*fileAction) (const char *fileName,
|
|
struct stat * statbuf,
|
|
void* userData),
|
|
int (*dirAction) (const char *fileName,
|
|
struct stat * statbuf,
|
|
void* userData),
|
|
void* userData)
|
|
{
|
|
int status;
|
|
struct stat statbuf;
|
|
struct dirent *next;
|
|
|
|
if (followLinks)
|
|
status = stat(fileName, &statbuf);
|
|
else
|
|
status = lstat(fileName, &statbuf);
|
|
|
|
if (status < 0) {
|
|
#ifdef DEBUG_RECURS_ACTION
|
|
bb_error_msg("status=%d followLinks=%d TRUE=%d",
|
|
status, followLinks, TRUE);
|
|
#endif
|
|
bb_perror_msg("%s", fileName);
|
|
return FALSE;
|
|
}
|
|
|
|
if (! followLinks && (S_ISLNK(statbuf.st_mode))) {
|
|
if (fileAction == NULL)
|
|
return TRUE;
|
|
else
|
|
return fileAction(fileName, &statbuf, userData);
|
|
}
|
|
|
|
if (! recurse) {
|
|
if (S_ISDIR(statbuf.st_mode)) {
|
|
if (dirAction != NULL)
|
|
return (dirAction(fileName, &statbuf, userData));
|
|
else
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
if (S_ISDIR(statbuf.st_mode)) {
|
|
DIR *dir;
|
|
|
|
if (dirAction != NULL && ! depthFirst) {
|
|
status = dirAction(fileName, &statbuf, userData);
|
|
if (! status) {
|
|
bb_perror_msg("%s", fileName);
|
|
return FALSE;
|
|
} else if (status == SKIP)
|
|
return TRUE;
|
|
}
|
|
dir = opendir(fileName);
|
|
if (!dir) {
|
|
return FALSE;
|
|
}
|
|
status = TRUE;
|
|
while ((next = readdir(dir)) != NULL) {
|
|
char *nextFile;
|
|
|
|
nextFile = concat_subpath_file(fileName, next->d_name);
|
|
if(nextFile == NULL)
|
|
continue;
|
|
if (! recursive_action(nextFile, TRUE, followLinks, depthFirst,
|
|
fileAction, dirAction, userData)) {
|
|
status = FALSE;
|
|
}
|
|
free(nextFile);
|
|
}
|
|
closedir(dir);
|
|
if (dirAction != NULL && depthFirst) {
|
|
if (! dirAction(fileName, &statbuf, userData)) {
|
|
bb_perror_msg("%s", fileName);
|
|
return FALSE;
|
|
}
|
|
}
|
|
if (! status)
|
|
return FALSE;
|
|
} else {
|
|
if (fileAction == NULL)
|
|
return TRUE;
|
|
else
|
|
return fileAction(fileName, &statbuf, userData);
|
|
}
|
|
return TRUE;
|
|
}
|