Add primitive du(1)
Defaults to a 512-byte blocksize.
This commit is contained in:
		
							
								
								
									
										1
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								Makefile
									
									
									
									
									
								
							@@ -39,6 +39,7 @@ SRC = \
 | 
			
		||||
	cut.c      \
 | 
			
		||||
	date.c     \
 | 
			
		||||
	dirname.c  \
 | 
			
		||||
	du.c       \
 | 
			
		||||
	echo.c     \
 | 
			
		||||
	env.c      \
 | 
			
		||||
	expand.c   \
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										18
									
								
								du.1
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								du.1
									
									
									
									
									
										Normal file
									
								
							@@ -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.
 | 
			
		||||
							
								
								
									
										120
									
								
								du.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								du.c
									
									
									
									
									
										Normal file
									
								
							@@ -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...]\n", 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\t%s\n", 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\n", 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;
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user