[hackers] [sbase] Add primitive du(1) || sin

From: <git_AT_suckless.org>
Date: Wed, 16 Oct 2013 20:22:42 +0200

commit 09fcbfc338f7401019565ef286d3cc508a59f4e9
Author: sin <sin_AT_2f30.org>
Date: Wed Oct 16 16:58:52 2013 +0100

    Add primitive du(1)
    
    Defaults to a 512-byte blocksize.

diff --git a/Makefile b/Makefile
index 1cd6fbb..326cae8 100644
--- a/Makefile
+++ b/Makefile
_AT_@ -39,6 +39,7 @@ SRC = \
         cut.c \
         date.c \
         dirname.c \
+ du.c \
         echo.c \
         env.c \
         expand.c \
diff --git a/du.1 b/du.1
new file mode 100644
index 0000000..a5046e1
--- /dev/null
+++ b/du.1
_AT_@ -0,0 +1,18 @@
+.TH DU 1 sbase\-VERSION
+.SH NAME
+du \- display disk usage statistics
+.SH SYNOPSIS
+.B du
+.RB [ \-a ]
+.RI [ file ...]
+.SH DESCRIPTION
+.B du
+displays the file system block usage for each
+.I file
+argument and for each directory in the file hierarchy rooted in directory argument.
+If no file is specified, the block usage of the hierarchy rooted in the current
+directory is displayed.
+.SH OPTIONS
+.TP
+.BI \-a
+Display an entry for each file in the file hierarchy.
diff --git a/du.c b/du.c
new file mode 100644
index 0000000..bfb40e6
--- /dev/null
+++ b/du.c
_AT_@ -0,0 +1,120 @@
+/* See LICENSE file for copyright and license details. */
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <dirent.h>
+#include <unistd.h>
+#include <errno.h>
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <stdbool.h>
+#include <limits.h>
+#include "util.h"
+
+static long blksize = 512;
+
+static bool aflag = false;
+
+static long du(const char *);
+
+void
+usage(void)
+{
+ eprintf("usage: %s [-a] [file...]
", argv0);
+}
+
+int
+main(int argc, char *argv[])
+{
+ char *bsize;
+
+ ARGBEGIN {
+ case 'a':
+ aflag = true;
+ break;
+ default:
+ usage();
+ } ARGEND;
+
+ bsize = getenv("BLOCKSIZE");
+ if (bsize)
+ blksize = estrtol(bsize, 0);
+
+ if (argc < 1) {
+ du(".");
+ } else {
+ for (; argc > 0; argc--, argv++)
+ du(argv[0]);
+ }
+ return EXIT_SUCCESS;
+}
+
+static void
+print(long n, char *path)
+{
+ printf("%lu %s
", n, path);
+ free(path);
+}
+
+static char *
+push(const char *path)
+{
+ char *cwd;
+
+ cwd = agetcwd();
+ if (chdir(path) < 0)
+ eprintf("chdir: %s:", path);
+ return cwd;
+}
+
+static void
+pop(char *path)
+{
+ if (chdir(path) < 0)
+ eprintf("chdir: %s:", path);
+ free(path);
+}
+
+static long
+du(const char *path)
+{
+ DIR *dp;
+ char *cwd;
+ struct dirent *dent;
+ struct stat st;
+ long n = 0, m;
+
+ if (lstat(path, &st) < 0)
+ eprintf("stat: %s:", path);
+ n = 512 * st.st_blocks / blksize;
+
+ if (S_ISDIR(st.st_mode)) {
+ dp = opendir(path);
+ if (!dp) {
+ fprintf(stderr, "opendir: %s: %s
", path,
+ strerror(errno));
+ } else {
+ cwd = push(path);
+ while ((dent = readdir(dp))) {
+ if (strcmp(dent->d_name, ".") == 0 ||
+ strcmp(dent->d_name, "..") == 0)
+ continue;
+ if (lstat(dent->d_name, &st) < 0)
+ eprintf("stat: %s:", dent->d_name);
+ if (S_ISDIR(st.st_mode)) {
+ n += du(dent->d_name);
+ } else {
+ m = 512 * st.st_blocks / blksize;
+ n += m;
+ if (aflag)
+ print(m, realpath(dent->d_name, NULL));
+ }
+ }
+ pop(cwd);
+ closedir(dp);
+ }
+ }
+
+ print(n, realpath(path, NULL));
+ return n;
+}
Received on Wed Oct 16 2013 - 20:22:42 CEST

This archive was generated by hypermail 2.3.0 : Wed Oct 16 2013 - 20:24:15 CEST