-
Notifications
You must be signed in to change notification settings - Fork 0
/
du.c
109 lines (100 loc) · 2.9 KB
/
du.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
// Copyright (C) 2021 Benjamin Stürz
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
#define PROG_NAME "du"
#include <sys/types.h>
#include <sys/stat.h>
#include <stdbool.h>
#include <string.h>
#include <unistd.h>
#include <dirent.h>
#include <stdio.h>
#include "errprintf.h"
static bool use_kilo = false;
static bool summarize = false;
static bool show_all = false;
static char HL = ' ';
static size_t traverse(const char* path, bool top) {
struct stat st;
int ec;
if (HL == 'L' || (HL == 'H' && top))
ec = stat(path, &st);
else ec = lstat(path, &st);
if (ec) {
errprintf("failed to stat '%s'", path);
return 0;
}
const bool is_dir = st.st_mode & S_IFDIR;
size_t size = st.st_blocks;
if (is_dir) {
const size_t len_path = strlen(path);
DIR* dir = opendir(path);
struct dirent* ent;
if (!dir) {
errprintf("failed to open directory '%s'", path);
return 0;
}
char buffer[len_path + 262];
memcpy(buffer, path, len_path);
buffer[len_path] = '/';
while ((ent = readdir(dir)) != NULL) {
if (!strcmp(".", ent->d_name) || !strcmp("..", ent->d_name))
continue;
strncpy(buffer + len_path + 1, ent->d_name, 260);
size += traverse(buffer, false);
}
}
if ((summarize && top) || (!summarize && (is_dir || show_all || top))) {
printf("%-7zu %s\n", size / (use_kilo ? 2 : 1), path);
}
return size;
}
int main(int argc, char* argv[]) {
int option;
while ((option = getopt(argc, argv, "askxHL")) != -1) {
switch (option) {
case 'a':
show_all = true;
break;
case 'k':
use_kilo = true;
break;
case 's':
summarize = true;
break;
case 'H':
case 'L':
HL = option;
break;
case 'x':
fputs("du: option '-x' is not implemented!\n", stderr);
return 1;
default:
puts("Usage: du [-a|-s] [-k] [-H|-L] [file...]");
return 0;
}
}
if (show_all && summarize) {
fputs("du: cannot both summarize and show all entries\n", stderr);
return 1;
}
if (optind < argc) {
for (; optind < argc; ++optind) {
traverse(argv[optind], true);
}
} else {
traverse(".", true);
}
return 0;
}