Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 2 | /* |
| 3 | * Mini du implementation for busybox |
| 4 | * |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 5 | * Copyright (C) 1999,2000,2001 by Lineo, inc. and John Beppu |
| 6 | * Copyright (C) 1999,2000,2001 by John Beppu <beppu@codepoet.org> |
Eric Andersen | 265d229 | 2002-04-06 23:16:44 +0000 | [diff] [blame] | 7 | * Copyright (C) 2002 Edward Betts <edward@debian.org> |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 8 | * |
Bernhard Reutner-Fischer | cb44816 | 2006-04-12 07:35:12 +0000 | [diff] [blame] | 9 | * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 10 | */ |
| 11 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 12 | /* BB_AUDIT SUSv3 compliant (unless default blocksize set to 1k) */ |
| 13 | /* http://www.opengroup.org/onlinepubs/007904975/utilities/du.html */ |
| 14 | |
| 15 | /* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org) |
| 16 | * |
| 17 | * Mostly rewritten for SUSv3 compliance and to fix bugs/defects. |
| 18 | * 1) Added support for SUSv3 -a, -H, -L, gnu -c, and (busybox) -d options. |
| 19 | * The -d option allows setting of max depth (similar to gnu --max-depth). |
| 20 | * 2) Fixed incorrect size calculations for links and directories, especially |
| 21 | * when errors occurred. Calculates sizes should now match gnu du output. |
| 22 | * 3) Added error checking of output. |
| 23 | * 4) Fixed busybox bug #1284 involving long overflow with human_readable. |
| 24 | */ |
| 25 | |
Eric Andersen | ed3ef50 | 2001-01-27 08:24:39 +0000 | [diff] [blame] | 26 | #include <stdlib.h> |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 27 | #include <limits.h> |
| 28 | #include <unistd.h> |
| 29 | #include <dirent.h> |
| 30 | #include <sys/stat.h> |
Eric Andersen | cbe31da | 2001-02-20 06:14:08 +0000 | [diff] [blame] | 31 | #include "busybox.h" |
Eric Andersen | cbe31da | 2001-02-20 06:14:08 +0000 | [diff] [blame] | 32 | |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 33 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
Rob Landley | 5fe8d5b | 2006-04-25 22:40:53 +0000 | [diff] [blame] | 34 | # ifdef CONFIG_FEATURE_DU_DEFAULT_BLOCKSIZE_1K |
Eric Andersen | ec9fad9 | 2001-03-07 06:04:08 +0000 | [diff] [blame] | 35 | static unsigned long disp_hr = KILOBYTE; |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 36 | # else |
| 37 | static unsigned long disp_hr = 512; |
| 38 | # endif |
Rob Landley | 5fe8d5b | 2006-04-25 22:40:53 +0000 | [diff] [blame] | 39 | #elif defined CONFIG_FEATURE_DU_DEFAULT_BLOCKSIZE_1K |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 40 | static unsigned int disp_k = 1; |
| 41 | #else |
| 42 | static unsigned int disp_k; /* bss inits to 0 */ |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 43 | #endif |
| 44 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 45 | static int max_print_depth = INT_MAX; |
"Vladimir N. Oleynik" | 57545c8 | 2006-01-31 12:06:57 +0000 | [diff] [blame] | 46 | static nlink_t count_hardlinks = 1; |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 47 | |
| 48 | static int status |
| 49 | #if EXIT_SUCCESS == 0 |
| 50 | = EXIT_SUCCESS |
| 51 | #endif |
| 52 | ; |
| 53 | static int print_files; |
| 54 | static int slink_depth; |
| 55 | static int du_depth; |
| 56 | static int one_file_system; |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 57 | static dev_t dir_dev; |
John Beppu | e1618e4 | 1999-12-15 18:52:17 +0000 | [diff] [blame] | 58 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 59 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 60 | static void print(long size, char *filename) |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 61 | { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 62 | /* TODO - May not want to defer error checking here. */ |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 63 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 64 | bb_printf("%s\t%s\n", make_human_readable_str(size, 512, disp_hr), |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 65 | filename); |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 66 | #else |
Glenn L McGrath | c66ebe4 | 2004-03-10 09:58:51 +0000 | [diff] [blame] | 67 | if (disp_k) { |
| 68 | size++; |
| 69 | size >>= 1; |
| 70 | } |
| 71 | bb_printf("%ld\t%s\n", size, filename); |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 72 | #endif |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 73 | } |
| 74 | |
| 75 | /* tiny recursive du */ |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 76 | static long du(char *filename) |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 77 | { |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 78 | struct stat statbuf; |
Erik Andersen | fac10d7 | 2000-02-07 05:29:42 +0000 | [diff] [blame] | 79 | long sum; |
John Beppu | 14c82b6 | 1999-12-10 06:15:27 +0000 | [diff] [blame] | 80 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 81 | if ((lstat(filename, &statbuf)) != 0) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 82 | bb_perror_msg("%s", filename); |
| 83 | status = EXIT_FAILURE; |
Eric Andersen | e5dfced | 2001-04-09 22:48:12 +0000 | [diff] [blame] | 84 | return 0; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 85 | } |
| 86 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 87 | if (one_file_system) { |
| 88 | if (du_depth == 0) { |
| 89 | dir_dev = statbuf.st_dev; |
| 90 | } else if (dir_dev != statbuf.st_dev) { |
| 91 | return 0; |
Eric Andersen | 8fa1bf7 | 2001-06-30 17:54:20 +0000 | [diff] [blame] | 92 | } |
Erik Andersen | 9ffdaa6 | 2000-02-11 21:55:04 +0000 | [diff] [blame] | 93 | } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 94 | |
| 95 | sum = statbuf.st_blocks; |
| 96 | |
| 97 | if (S_ISLNK(statbuf.st_mode)) { |
| 98 | if (slink_depth > du_depth) { /* -H or -L */ |
| 99 | if ((stat(filename, &statbuf)) != 0) { |
| 100 | bb_perror_msg("%s", filename); |
| 101 | status = EXIT_FAILURE; |
| 102 | return 0; |
| 103 | } |
| 104 | sum = statbuf.st_blocks; |
| 105 | if (slink_depth == 1) { |
| 106 | slink_depth = INT_MAX; /* Convert -H to -L. */ |
| 107 | } |
| 108 | } |
| 109 | } |
| 110 | |
| 111 | if (statbuf.st_nlink > count_hardlinks) { |
| 112 | /* Add files/directories with links only once */ |
| 113 | if (is_in_ino_dev_hashtable(&statbuf, NULL)) { |
| 114 | return 0; |
| 115 | } |
| 116 | add_to_ino_dev_hashtable(&statbuf, NULL); |
| 117 | } |
| 118 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 119 | if (S_ISDIR(statbuf.st_mode)) { |
| 120 | DIR *dir; |
| 121 | struct dirent *entry; |
Eric Andersen | 04b0354 | 2001-05-07 22:49:43 +0000 | [diff] [blame] | 122 | char *newfile; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 123 | |
Rob Landley | 86b4d64 | 2006-08-03 17:58:17 +0000 | [diff] [blame^] | 124 | dir = warn_opendir(filename); |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 125 | if (!dir) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 126 | status = EXIT_FAILURE; |
| 127 | return sum; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 128 | } |
Erik Andersen | 42387e4 | 2000-02-21 17:27:17 +0000 | [diff] [blame] | 129 | |
Eric Andersen | 04b0354 | 2001-05-07 22:49:43 +0000 | [diff] [blame] | 130 | newfile = last_char_is(filename, '/'); |
| 131 | if (newfile) |
| 132 | *newfile = '\0'; |
Erik Andersen | 42387e4 | 2000-02-21 17:27:17 +0000 | [diff] [blame] | 133 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 134 | while ((entry = readdir(dir))) { |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 135 | char *name = entry->d_name; |
| 136 | |
Glenn L McGrath | 393183d | 2003-05-26 14:07:50 +0000 | [diff] [blame] | 137 | newfile = concat_subpath_file(filename, name); |
| 138 | if(newfile == NULL) |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 139 | continue; |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 140 | ++du_depth; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 141 | sum += du(newfile); |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 142 | --du_depth; |
Eric Andersen | e5dfced | 2001-04-09 22:48:12 +0000 | [diff] [blame] | 143 | free(newfile); |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 144 | } |
| 145 | closedir(dir); |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 146 | } else if (du_depth > print_files) { |
| 147 | return sum; |
Erik Andersen | 27fdd08 | 2000-02-19 18:16:49 +0000 | [diff] [blame] | 148 | } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 149 | if (du_depth <= max_print_depth) { |
| 150 | print(sum, filename); |
| 151 | } |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 152 | return sum; |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 153 | } |
| 154 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 155 | int du_main(int argc, char **argv) |
| 156 | { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 157 | long total; |
| 158 | int slink_depth_save; |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 159 | int print_final_total; |
| 160 | char *smax_print_depth; |
| 161 | unsigned long opt; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 162 | |
Rob Landley | 5fe8d5b | 2006-04-25 22:40:53 +0000 | [diff] [blame] | 163 | #ifdef CONFIG_FEATURE_DU_DEFUALT_BLOCKSIZE_1K |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 164 | if (getenv("POSIXLY_CORRECT")) { /* TODO - a new libbb function? */ |
| 165 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
| 166 | disp_hr = 512; |
| 167 | #else |
| 168 | disp_k = 0; |
| 169 | #endif |
Eric Andersen | c7bda1c | 2004-03-15 08:29:22 +0000 | [diff] [blame] | 170 | } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 171 | #endif |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 172 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 173 | /* Note: SUSv3 specifies that -a and -s options can not be used together |
| 174 | * in strictly conforming applications. However, it also says that some |
| 175 | * du implementations may produce output when -a and -s are used together. |
| 176 | * gnu du exits with an error code in this case. We choose to simply |
| 177 | * ignore -a. This is consistent with -s being equivalent to -d 0. |
| 178 | */ |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 179 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
"Vladimir N. Oleynik" | 27421a1 | 2005-09-05 14:46:07 +0000 | [diff] [blame] | 180 | bb_opt_complementally = "h-km:k-hm:m-hk:H-L:L-H:s-d:d-s"; |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 181 | opt = bb_getopt_ulflags(argc, argv, "aHkLsx" "d:" "lc" "hm", &smax_print_depth); |
| 182 | if((opt & (1 << 9))) { |
| 183 | /* -h opt */ |
| 184 | disp_hr = 0; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 185 | } |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 186 | if((opt & (1 << 10))) { |
| 187 | /* -m opt */ |
| 188 | disp_hr = MEGABYTE; |
| 189 | } |
| 190 | if((opt & (1 << 2))) { |
| 191 | /* -k opt */ |
| 192 | disp_hr = KILOBYTE; |
| 193 | } |
| 194 | #else |
"Vladimir N. Oleynik" | 27421a1 | 2005-09-05 14:46:07 +0000 | [diff] [blame] | 195 | bb_opt_complementally = "H-L:L-H:s-d:d-s"; |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 196 | opt = bb_getopt_ulflags(argc, argv, "aHkLsx" "d:" "lc", &smax_print_depth); |
Rob Landley | 5fe8d5b | 2006-04-25 22:40:53 +0000 | [diff] [blame] | 197 | #if !defined CONFIG_FEATURE_DU_DEFAULT_BLOCKSIZE_1K |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 198 | if((opt & (1 << 2))) { |
| 199 | /* -k opt */ |
| 200 | disp_k = 1; |
| 201 | } |
| 202 | #endif |
| 203 | #endif |
| 204 | if((opt & (1 << 0))) { |
| 205 | /* -a opt */ |
| 206 | print_files = INT_MAX; |
| 207 | } |
| 208 | if((opt & (1 << 1))) { |
| 209 | /* -H opt */ |
| 210 | slink_depth = 1; |
| 211 | } |
| 212 | if((opt & (1 << 3))) { |
| 213 | /* -L opt */ |
| 214 | slink_depth = INT_MAX; |
| 215 | } |
| 216 | if((opt & (1 << 4))) { |
| 217 | /* -s opt */ |
| 218 | max_print_depth = 0; |
| 219 | } |
| 220 | one_file_system = opt & (1 << 5); /* -x opt */ |
| 221 | if((opt & (1 << 6))) { |
| 222 | /* -d opt */ |
| 223 | max_print_depth = bb_xgetularg10_bnd(smax_print_depth, 0, INT_MAX); |
| 224 | } |
| 225 | if((opt & (1 << 7))) { |
| 226 | /* -l opt */ |
Glenn L McGrath | 44c0e17 | 2004-03-10 09:10:53 +0000 | [diff] [blame] | 227 | count_hardlinks = INT_MAX; |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 228 | } |
| 229 | print_final_total = opt & (1 << 8); /* -c opt */ |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 230 | |
| 231 | /* go through remaining args (if any) */ |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 232 | argv += optind; |
Eric Andersen | 17ad45a | 2000-07-14 18:38:26 +0000 | [diff] [blame] | 233 | if (optind >= argc) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 234 | *--argv = "."; |
| 235 | if (slink_depth == 1) { |
| 236 | slink_depth = 0; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 237 | } |
| 238 | } |
| 239 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 240 | slink_depth_save = slink_depth; |
| 241 | total = 0; |
| 242 | do { |
| 243 | total += du(*argv); |
| 244 | slink_depth = slink_depth_save; |
| 245 | } while (*++argv); |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 246 | #ifdef CONFIG_FEATURE_CLEAN_UP |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 247 | reset_ino_dev_hashtable(); |
Eric Andersen | 8876fb2 | 2003-06-20 09:01:58 +0000 | [diff] [blame] | 248 | #endif |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 249 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame] | 250 | if (print_final_total) { |
| 251 | print(total, "total"); |
| 252 | } |
| 253 | |
| 254 | bb_fflush_stdout_and_exit(status); |
| 255 | } |