ls: reorder and rename functions. No code changes

function                                             old     new   delta
display_single                                         -     931    +931
scan_and_display_dirs_recur                            -     497    +497
display_files                                          -     422    +422
showfiles                                            422       -    -422
showdirs                                             497       -    -497
list_single                                          931       -    -931
------------------------------------------------------------------------------
(add/remove: 3/3 grow/shrink: 0/0 up/down: 1850/-1850)          Total: 0 bytes

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
Denys Vlasenko 2011-05-13 17:28:46 +02:00
parent 2a81639534
commit 4029e21b37

View File

@ -326,7 +326,7 @@ struct dnode {
// (such as nanosecond-resolution timespamps) // (such as nanosecond-resolution timespamps)
// and padding, which we also don't want to store. // and padding, which we also don't want to store.
// We also can pre-parse dev_t dn_rdev (in glibc, it's huge). // We also can pre-parse dev_t dn_rdev (in glibc, it's huge).
// On 32-bit uclibc: dnode size went from 112 to 84 bytes // On 32-bit uclibc: dnode size went from 112 to 84 bytes.
// //
/* Same names as in struct stat, but with dn_ instead of st_ pfx: */ /* Same names as in struct stat, but with dn_ instead of st_ pfx: */
mode_t dn_mode; /* obtained with lstat OR stat, depending on -L etc */ mode_t dn_mode; /* obtained with lstat OR stat, depending on -L etc */
@ -376,61 +376,8 @@ struct globals {
} while (0) } while (0)
static struct dnode *my_stat(const char *fullname, const char *name, int force_follow) /*** Output code ***/
{
struct stat statbuf;
struct dnode *cur;
cur = xzalloc(sizeof(*cur));
cur->fullname = fullname;
cur->name = name;
if ((option_mask32 & OPT_L) || force_follow) {
#if ENABLE_SELINUX
if (is_selinux_enabled()) {
getfilecon(fullname, &cur->sid);
}
#endif
if (stat(fullname, &statbuf)) {
bb_simple_perror_msg(fullname);
G.exit_code = EXIT_FAILURE;
free(cur);
return NULL;
}
cur->dn_mode_stat = statbuf.st_mode;
} else {
#if ENABLE_SELINUX
if (is_selinux_enabled()) {
lgetfilecon(fullname, &cur->sid);
}
#endif
if (lstat(fullname, &statbuf)) {
bb_simple_perror_msg(fullname);
G.exit_code = EXIT_FAILURE;
free(cur);
return NULL;
}
cur->dn_mode_lstat = statbuf.st_mode;
}
/* cur->dstat = statbuf: */
cur->dn_mode = statbuf.st_mode ;
cur->dn_size = statbuf.st_size ;
#if ENABLE_FEATURE_LS_TIMESTAMPS || ENABLE_FEATURE_LS_SORTFILES
cur->dn_atime = statbuf.st_atime ;
cur->dn_mtime = statbuf.st_mtime ;
cur->dn_ctime = statbuf.st_ctime ;
#endif
cur->dn_ino = statbuf.st_ino ;
cur->dn_blocks = statbuf.st_blocks;
cur->dn_nlink = statbuf.st_nlink ;
cur->dn_uid = statbuf.st_uid ;
cur->dn_gid = statbuf.st_gid ;
cur->dn_rdev_maj = major(statbuf.st_rdev);
cur->dn_rdev_min = minor(statbuf.st_rdev);
return cur;
}
/* FYI type values: 1:fifo 2:char 4:dir 6:blk 8:file 10:link 12:socket /* FYI type values: 1:fifo 2:char 4:dir 6:blk 8:file 10:link 12:socket
* (various wacky OSes: 13:Sun door 14:BSD whiteout 5:XENIX named file * (various wacky OSes: 13:Sun door 14:BSD whiteout 5:XENIX named file
@ -493,155 +440,6 @@ static char append_char(mode_t mode)
} }
#endif #endif
static unsigned count_dirs(struct dnode **dn, int which)
{
unsigned dirs, all;
if (!dn)
return 0;
dirs = all = 0;
for (; *dn; dn++) {
const char *name;
all++;
if (!S_ISDIR((*dn)->dn_mode))
continue;
name = (*dn)->name;
if (which != SPLIT_SUBDIR /* if not requested to skip . / .. */
/* or if it's not . or .. */
|| name[0] != '.'
|| (name[1] && (name[1] != '.' || name[2]))
) {
dirs++;
}
}
return which != SPLIT_FILE ? dirs : all - dirs;
}
/* get memory to hold an array of pointers */
static struct dnode **dnalloc(unsigned num)
{
if (num < 1)
return NULL;
num++; /* so that we have terminating NULL */
return xzalloc(num * sizeof(struct dnode *));
}
#if ENABLE_FEATURE_LS_RECURSIVE
static void dfree(struct dnode **dnp)
{
unsigned i;
if (dnp == NULL)
return;
for (i = 0; dnp[i]; i++) {
struct dnode *cur = dnp[i];
if (cur->fname_allocated)
free((char*)cur->fullname);
free(cur);
}
free(dnp);
}
#else
#define dfree(...) ((void)0)
#endif
/* Returns NULL-terminated malloced vector of pointers (or NULL) */
static struct dnode **splitdnarray(struct dnode **dn, int which)
{
unsigned dncnt, d;
struct dnode **dnp;
if (dn == NULL)
return NULL;
/* count how many dirs or files there are */
dncnt = count_dirs(dn, which);
/* allocate a file array and a dir array */
dnp = dnalloc(dncnt);
/* copy the entrys into the file or dir array */
for (d = 0; *dn; dn++) {
if (S_ISDIR((*dn)->dn_mode)) {
const char *name;
if (which == SPLIT_FILE)
continue;
name = (*dn)->name;
if ((which & SPLIT_DIR) /* any dir... */
/* ... or not . or .. */
|| name[0] != '.'
|| (name[1] && (name[1] != '.' || name[2]))
) {
dnp[d++] = *dn;
}
} else
if (which == SPLIT_FILE) {
dnp[d++] = *dn;
}
}
return dnp;
}
#if ENABLE_FEATURE_LS_SORTFILES
static int sortcmp(const void *a, const void *b)
{
struct dnode *d1 = *(struct dnode **)a;
struct dnode *d2 = *(struct dnode **)b;
unsigned sort_opts = G.all_fmt & SORT_MASK;
off_t dif;
dif = 0; /* assume SORT_NAME */
// TODO: use pre-initialized function pointer
// instead of branch forest
if (sort_opts == SORT_SIZE) {
dif = (d2->dn_size - d1->dn_size);
} else if (sort_opts == SORT_ATIME) {
dif = (d2->dn_atime - d1->dn_atime);
} else if (sort_opts == SORT_CTIME) {
dif = (d2->dn_ctime - d1->dn_ctime);
} else if (sort_opts == SORT_MTIME) {
dif = (d2->dn_mtime - d1->dn_mtime);
} else if (sort_opts == SORT_DIR) {
dif = S_ISDIR(d2->dn_mode) - S_ISDIR(d1->dn_mode);
/* } else if (sort_opts == SORT_VERSION) { */
/* } else if (sort_opts == SORT_EXT) { */
}
if (dif == 0) {
/* sort by name, or tie_breaker for other sorts */
if (ENABLE_LOCALE_SUPPORT)
dif = strcoll(d1->name, d2->name);
else
dif = strcmp(d1->name, d2->name);
}
/* Make dif fit into an int */
if (sizeof(dif) > sizeof(int)) {
enum { BITS_TO_SHIFT = 8 * (sizeof(dif) - sizeof(int)) };
/* shift leaving only "int" worth of bits */
if (dif != 0) {
dif = 1 | (int)((uoff_t)dif >> BITS_TO_SHIFT);
}
}
return (G.all_fmt & SORT_REVERSE) ? -(int)dif : (int)dif;
}
static void dnsort(struct dnode **dn, int size)
{
qsort(dn, size, sizeof(*dn), sortcmp);
}
#else
#define dnsort(dn, size) ((void)0)
#endif
static unsigned calc_name_len(const char *name) static unsigned calc_name_len(const char *name)
{ {
unsigned len; unsigned len;
@ -664,7 +462,6 @@ static unsigned calc_name_len(const char *name)
return len; return len;
} }
/* Return the number of used columns. /* Return the number of used columns.
* Note that only STYLE_COLUMNAR uses return value. * Note that only STYLE_COLUMNAR uses return value.
* STYLE_SINGLE and STYLE_LONG don't care. * STYLE_SINGLE and STYLE_LONG don't care.
@ -703,7 +500,7 @@ static unsigned print_name(const char *name)
* Note that only STYLE_COLUMNAR uses return value, * Note that only STYLE_COLUMNAR uses return value,
* STYLE_SINGLE and STYLE_LONG don't care. * STYLE_SINGLE and STYLE_LONG don't care.
*/ */
static NOINLINE unsigned list_single(const struct dnode *dn) static NOINLINE unsigned display_single(const struct dnode *dn)
{ {
unsigned column = 0; unsigned column = 0;
char *lpath; char *lpath;
@ -854,7 +651,7 @@ static NOINLINE unsigned list_single(const struct dnode *dn)
return column; return column;
} }
static void showfiles(struct dnode **dn, unsigned nfiles) static void display_files(struct dnode **dn, unsigned nfiles)
{ {
unsigned i, ncols, nrows, row, nc; unsigned i, ncols, nrows, row, nc;
unsigned column; unsigned column;
@ -902,7 +699,7 @@ static void showfiles(struct dnode **dn, unsigned nfiles)
column += nexttab + 1; column += nexttab + 1;
} }
nexttab = column + column_width; nexttab = column + column_width;
column += list_single(dn[i]); column += display_single(dn[i]);
} }
} }
putchar('\n'); putchar('\n');
@ -911,38 +708,214 @@ static void showfiles(struct dnode **dn, unsigned nfiles)
} }
#if ENABLE_DESKTOP /*** Dir scanning code ***/
/* http://www.opengroup.org/onlinepubs/9699919799/utilities/ls.html
* If any of the -l, -n, -s options is specified, each list static struct dnode *my_stat(const char *fullname, const char *name, int force_follow)
* of files within the directory shall be preceded by a
* status line indicating the number of file system blocks
* occupied by files in the directory in 512-byte units if
* the -k option is not specified, or 1024-byte units if the
* -k option is specified, rounded up to the next integral
* number of units.
*/
/* by Jorgen Overgaard (jorgen AT antistaten.se) */
static off_t calculate_blocks(struct dnode **dn)
{ {
uoff_t blocks = 1; struct stat statbuf;
if (dn) { struct dnode *cur;
while (*dn) {
/* st_blocks is in 512 byte blocks */ cur = xzalloc(sizeof(*cur));
blocks += (*dn)->dn_blocks; cur->fullname = fullname;
dn++; cur->name = name;
if ((option_mask32 & OPT_L) || force_follow) {
#if ENABLE_SELINUX
if (is_selinux_enabled()) {
getfilecon(fullname, &cur->sid);
} }
#endif
if (stat(fullname, &statbuf)) {
bb_simple_perror_msg(fullname);
G.exit_code = EXIT_FAILURE;
free(cur);
return NULL;
}
cur->dn_mode_stat = statbuf.st_mode;
} else {
#if ENABLE_SELINUX
if (is_selinux_enabled()) {
lgetfilecon(fullname, &cur->sid);
}
#endif
if (lstat(fullname, &statbuf)) {
bb_simple_perror_msg(fullname);
G.exit_code = EXIT_FAILURE;
free(cur);
return NULL;
}
cur->dn_mode_lstat = statbuf.st_mode;
} }
/* Even though standard says use 512 byte blocks, coreutils use 1k */ /* cur->dstat = statbuf: */
/* Actually, we round up by calculating (blocks + 1) / 2, cur->dn_mode = statbuf.st_mode ;
* "+ 1" was done when we initialized blocks to 1 */ cur->dn_size = statbuf.st_size ;
return blocks >> 1; #if ENABLE_FEATURE_LS_TIMESTAMPS || ENABLE_FEATURE_LS_SORTFILES
cur->dn_atime = statbuf.st_atime ;
cur->dn_mtime = statbuf.st_mtime ;
cur->dn_ctime = statbuf.st_ctime ;
#endif
cur->dn_ino = statbuf.st_ino ;
cur->dn_blocks = statbuf.st_blocks;
cur->dn_nlink = statbuf.st_nlink ;
cur->dn_uid = statbuf.st_uid ;
cur->dn_gid = statbuf.st_gid ;
cur->dn_rdev_maj = major(statbuf.st_rdev);
cur->dn_rdev_min = minor(statbuf.st_rdev);
return cur;
} }
static unsigned count_dirs(struct dnode **dn, int which)
{
unsigned dirs, all;
if (!dn)
return 0;
dirs = all = 0;
for (; *dn; dn++) {
const char *name;
all++;
if (!S_ISDIR((*dn)->dn_mode))
continue;
name = (*dn)->name;
if (which != SPLIT_SUBDIR /* if not requested to skip . / .. */
/* or if it's not . or .. */
|| name[0] != '.'
|| (name[1] && (name[1] != '.' || name[2]))
) {
dirs++;
}
}
return which != SPLIT_FILE ? dirs : all - dirs;
}
/* get memory to hold an array of pointers */
static struct dnode **dnalloc(unsigned num)
{
if (num < 1)
return NULL;
num++; /* so that we have terminating NULL */
return xzalloc(num * sizeof(struct dnode *));
}
#if ENABLE_FEATURE_LS_RECURSIVE
static void dfree(struct dnode **dnp)
{
unsigned i;
if (dnp == NULL)
return;
for (i = 0; dnp[i]; i++) {
struct dnode *cur = dnp[i];
if (cur->fname_allocated)
free((char*)cur->fullname);
free(cur);
}
free(dnp);
}
#else
#define dfree(...) ((void)0)
#endif #endif
/* Returns NULL-terminated malloced vector of pointers (or NULL) */
static struct dnode **splitdnarray(struct dnode **dn, int which)
{
unsigned dncnt, d;
struct dnode **dnp;
if (dn == NULL)
return NULL;
/* count how many dirs or files there are */
dncnt = count_dirs(dn, which);
/* allocate a file array and a dir array */
dnp = dnalloc(dncnt);
/* copy the entrys into the file or dir array */
for (d = 0; *dn; dn++) {
if (S_ISDIR((*dn)->dn_mode)) {
const char *name;
if (which == SPLIT_FILE)
continue;
name = (*dn)->name;
if ((which & SPLIT_DIR) /* any dir... */
/* ... or not . or .. */
|| name[0] != '.'
|| (name[1] && (name[1] != '.' || name[2]))
) {
dnp[d++] = *dn;
}
} else
if (which == SPLIT_FILE) {
dnp[d++] = *dn;
}
}
return dnp;
}
#if ENABLE_FEATURE_LS_SORTFILES
static int sortcmp(const void *a, const void *b)
{
struct dnode *d1 = *(struct dnode **)a;
struct dnode *d2 = *(struct dnode **)b;
unsigned sort_opts = G.all_fmt & SORT_MASK;
off_t dif;
dif = 0; /* assume SORT_NAME */
// TODO: use pre-initialized function pointer
// instead of branch forest
if (sort_opts == SORT_SIZE) {
dif = (d2->dn_size - d1->dn_size);
} else if (sort_opts == SORT_ATIME) {
dif = (d2->dn_atime - d1->dn_atime);
} else if (sort_opts == SORT_CTIME) {
dif = (d2->dn_ctime - d1->dn_ctime);
} else if (sort_opts == SORT_MTIME) {
dif = (d2->dn_mtime - d1->dn_mtime);
} else if (sort_opts == SORT_DIR) {
dif = S_ISDIR(d2->dn_mode) - S_ISDIR(d1->dn_mode);
/* } else if (sort_opts == SORT_VERSION) { */
/* } else if (sort_opts == SORT_EXT) { */
}
if (dif == 0) {
/* sort by name, or tie_breaker for other sorts */
if (ENABLE_LOCALE_SUPPORT)
dif = strcoll(d1->name, d2->name);
else
dif = strcmp(d1->name, d2->name);
}
/* Make dif fit into an int */
if (sizeof(dif) > sizeof(int)) {
enum { BITS_TO_SHIFT = 8 * (sizeof(dif) - sizeof(int)) };
/* shift leaving only "int" worth of bits */
if (dif != 0) {
dif = 1 | (int)((uoff_t)dif >> BITS_TO_SHIFT);
}
}
return (G.all_fmt & SORT_REVERSE) ? -(int)dif : (int)dif;
}
static void dnsort(struct dnode **dn, int size)
{
qsort(dn, size, sizeof(*dn), sortcmp);
}
#else
#define dnsort(dn, size) ((void)0)
#endif
/* Returns NULL-terminated malloced vector of pointers (or NULL) */ /* Returns NULL-terminated malloced vector of pointers (or NULL) */
static struct dnode **list_dir(const char *path, unsigned *nfiles_p) static struct dnode **scan_one_dir(const char *path, unsigned *nfiles_p)
{ {
struct dnode *dn, *cur, **dnp; struct dnode *dn, *cur, **dnp;
struct dirent *entry; struct dirent *entry;
@ -1001,8 +974,36 @@ static struct dnode **list_dir(const char *path, unsigned *nfiles_p)
return dnp; return dnp;
} }
#if ENABLE_DESKTOP
/* http://www.opengroup.org/onlinepubs/9699919799/utilities/ls.html
* If any of the -l, -n, -s options is specified, each list
* of files within the directory shall be preceded by a
* status line indicating the number of file system blocks
* occupied by files in the directory in 512-byte units if
* the -k option is not specified, or 1024-byte units if the
* -k option is specified, rounded up to the next integral
* number of units.
*/
/* by Jorgen Overgaard (jorgen AT antistaten.se) */
static off_t calculate_blocks(struct dnode **dn)
{
uoff_t blocks = 1;
if (dn) {
while (*dn) {
/* st_blocks is in 512 byte blocks */
blocks += (*dn)->dn_blocks;
dn++;
}
}
static void showdirs(struct dnode **dn, int first) /* Even though standard says use 512 byte blocks, coreutils use 1k */
/* Actually, we round up by calculating (blocks + 1) / 2,
* "+ 1" was done when we initialized blocks to 1 */
return blocks >> 1;
}
#endif
static void scan_and_display_dirs_recur(struct dnode **dn, int first)
{ {
unsigned nfiles; unsigned nfiles;
struct dnode **subdnp; struct dnode **subdnp;
@ -1014,7 +1015,7 @@ static void showdirs(struct dnode **dn, int first)
first = 0; first = 0;
printf("%s:\n", (*dn)->fullname); printf("%s:\n", (*dn)->fullname);
} }
subdnp = list_dir((*dn)->fullname, &nfiles); subdnp = scan_one_dir((*dn)->fullname, &nfiles);
#if ENABLE_DESKTOP #if ENABLE_DESKTOP
if ((G.all_fmt & STYLE_MASK) == STYLE_LONG) if ((G.all_fmt & STYLE_MASK) == STYLE_LONG)
printf("total %"OFF_FMT"u\n", calculate_blocks(subdnp)); printf("total %"OFF_FMT"u\n", calculate_blocks(subdnp));
@ -1022,7 +1023,8 @@ static void showdirs(struct dnode **dn, int first)
if (nfiles > 0) { if (nfiles > 0) {
/* list all files at this level */ /* list all files at this level */
dnsort(subdnp, nfiles); dnsort(subdnp, nfiles);
showfiles(subdnp, nfiles); display_files(subdnp, nfiles);
if (ENABLE_FEATURE_LS_RECURSIVE if (ENABLE_FEATURE_LS_RECURSIVE
&& (G.all_fmt & DISP_RECURSIVE) && (G.all_fmt & DISP_RECURSIVE)
) { ) {
@ -1033,7 +1035,7 @@ static void showdirs(struct dnode **dn, int first)
dndirs = count_dirs(subdnp, SPLIT_SUBDIR); dndirs = count_dirs(subdnp, SPLIT_SUBDIR);
if (dndirs > 0) { if (dndirs > 0) {
dnsort(dnd, dndirs); dnsort(dnd, dndirs);
showdirs(dnd, 0); scan_and_display_dirs_recur(dnd, 0);
/* free the array of dnode pointers to the dirs */ /* free the array of dnode pointers to the dirs */
free(dnd); free(dnd);
} }
@ -1215,7 +1217,7 @@ int ls_main(int argc UNUSED_PARAM, char **argv)
if (G.all_fmt & DISP_NOLIST) { if (G.all_fmt & DISP_NOLIST) {
dnsort(dnp, nfiles); dnsort(dnp, nfiles);
showfiles(dnp, nfiles); display_files(dnp, nfiles);
} else { } else {
dnd = splitdnarray(dnp, SPLIT_DIR); dnd = splitdnarray(dnp, SPLIT_DIR);
dnf = splitdnarray(dnp, SPLIT_FILE); dnf = splitdnarray(dnp, SPLIT_FILE);
@ -1223,13 +1225,13 @@ int ls_main(int argc UNUSED_PARAM, char **argv)
dnfiles = nfiles - dndirs; dnfiles = nfiles - dndirs;
if (dnfiles > 0) { if (dnfiles > 0) {
dnsort(dnf, dnfiles); dnsort(dnf, dnfiles);
showfiles(dnf, dnfiles); display_files(dnf, dnfiles);
if (ENABLE_FEATURE_CLEAN_UP) if (ENABLE_FEATURE_CLEAN_UP)
free(dnf); free(dnf);
} }
if (dndirs > 0) { if (dndirs > 0) {
dnsort(dnd, dndirs); dnsort(dnd, dndirs);
showdirs(dnd, dnfiles == 0); scan_and_display_dirs_recur(dnd, dnfiles == 0);
if (ENABLE_FEATURE_CLEAN_UP) if (ENABLE_FEATURE_CLEAN_UP)
free(dnd); free(dnd);
} }