commit 2cf82f4c16c0dde6643e8482ed5d5409845016a7
parent b6b8fe9591b4e16e0c262965cc48ea6fe1f9ebfc
Author: Hiltjo Posthuma <hiltjo@codemadness.org>
Date: Sat, 18 Oct 2014 21:26:35 +0000
du: add -d flag to specify the max depth to show files or directories
we don't allow to use it with -s (like GNU du). busybox allows it.
Diffstat:
M | du.1 | | | 8 | ++++++-- |
M | du.c | | | 24 | ++++++++++++++++++------ |
2 files changed, 24 insertions(+), 8 deletions(-)
diff --git a/du.1 b/du.1
@@ -5,8 +5,9 @@ du \- display disk usage statistics
.B du
.RB [ \-a
.RB |
-.B \-s
-.RB ]
+.BR \-s ]
+.RB [ \-d
+.IR depth ]
.RB [ \-k ]
.RB [ \-h ]
.RI [ file ...]
@@ -25,6 +26,9 @@ Display an entry for each file in the file hierarchy.
.BI \-s
Display only the grand total for the specified files.
.TP
+.BI "\-d depth"
+Maximum directory depth to print files and directories.
+.TP
.BI \-k
By default all sizes are reported in 512-byte block counts.
The -k option causes the numbers to be reported in kilobyte counts.
diff --git a/du.c b/du.c
@@ -12,8 +12,11 @@
static long blksize = 512;
static char file[PATH_MAX];
+static long depth = -1;
+static long curdepth = 0;
static bool aflag = false;
+static bool dflag = false;
static bool sflag = false;
static bool kflag = false;
static bool hflag = false;
@@ -24,7 +27,7 @@ static void print(long n, char *path);
static void
usage(void)
{
- eprintf("usage: %s [-a | -s] [-k] [file...]\n", argv0);
+ eprintf("usage: %s [-a | -s] [-d depth] [-h] [-k] [file...]\n", argv0);
}
static char *
@@ -48,6 +51,10 @@ main(int argc, char *argv[])
case 'a':
aflag = true;
break;
+ case 'd':
+ dflag = true;
+ depth = estrtol(EARGF(usage()), 0);
+ break;
case 's':
sflag = true;
break;
@@ -61,7 +68,7 @@ main(int argc, char *argv[])
usage();
} ARGEND;
- if (aflag && sflag)
+ if ((aflag && sflag) || (dflag && sflag))
usage();
bsize = getenv("BLOCKSIZE");
@@ -77,6 +84,7 @@ main(int argc, char *argv[])
print(n, xrealpath(".", file));
} else {
for (; argc > 0; argc--, argv++) {
+ curdepth = 0;
n = du(argv[0]);
if (sflag)
print(n, xrealpath(argv[0], file));
@@ -126,7 +134,7 @@ du(const char *path)
char *cwd;
struct dirent *dent;
struct stat st;
- long n = 0, m;
+ long n = 0, m, t;
int r;
if (lstat(path, &st) < 0)
@@ -150,7 +158,10 @@ du(const char *path)
if (lstat(dent->d_name, &st) < 0)
eprintf("stat: %s:", dent->d_name);
if (S_ISDIR(st.st_mode)) {
+ t = curdepth;
+ curdepth++;
n += du(dent->d_name);
+ curdepth = t;
continue;
}
m = nblks(&st);
@@ -159,19 +170,20 @@ du(const char *path)
if (S_ISLNK(st.st_mode)) {
r = snprintf(file, sizeof(file), "%s/%s",
cwd, dent->d_name);
- if(r >= sizeof(file) || r < 0)
+ if (r >= sizeof(file) || r < 0)
eprintf("path too long\n");
} else {
xrealpath(dent->d_name, file);
}
- print(m, file);
+ if (!dflag || (depth != -1 && curdepth < depth))
+ print(m, file);
}
}
pop(cwd);
closedir(dp);
done:
- if (!sflag)
+ if (!sflag && (!dflag || (depth != -1 && curdepth <= depth)))
print(n, xrealpath(path, file));
return n;
}