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 | * |
| 9 | * This program is free software; you can redistribute it and/or modify |
| 10 | * it under the terms of the GNU General Public License as published by |
| 11 | * the Free Software Foundation; either version 2 of the License, or |
| 12 | * (at your option) any later version. |
| 13 | * |
| 14 | * This program is distributed in the hope that it will be useful, |
| 15 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 16 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 17 | * General Public License for more details. |
| 18 | * |
| 19 | * You should have received a copy of the GNU General Public License |
| 20 | * along with this program; if not, write to the Free Software |
| 21 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| 22 | * |
| 23 | */ |
| 24 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 25 | /* BB_AUDIT SUSv3 compliant (unless default blocksize set to 1k) */ |
| 26 | /* http://www.opengroup.org/onlinepubs/007904975/utilities/du.html */ |
| 27 | |
| 28 | /* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org) |
| 29 | * |
| 30 | * Mostly rewritten for SUSv3 compliance and to fix bugs/defects. |
| 31 | * 1) Added support for SUSv3 -a, -H, -L, gnu -c, and (busybox) -d options. |
| 32 | * The -d option allows setting of max depth (similar to gnu --max-depth). |
| 33 | * 2) Fixed incorrect size calculations for links and directories, especially |
| 34 | * when errors occurred. Calculates sizes should now match gnu du output. |
| 35 | * 3) Added error checking of output. |
| 36 | * 4) Fixed busybox bug #1284 involving long overflow with human_readable. |
| 37 | */ |
| 38 | |
Eric Andersen | ed3ef50 | 2001-01-27 08:24:39 +0000 | [diff] [blame] | 39 | #include <stdlib.h> |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 40 | #include <limits.h> |
| 41 | #include <unistd.h> |
| 42 | #include <dirent.h> |
| 43 | #include <sys/stat.h> |
Eric Andersen | cbe31da | 2001-02-20 06:14:08 +0000 | [diff] [blame] | 44 | #include "busybox.h" |
Eric Andersen | cbe31da | 2001-02-20 06:14:08 +0000 | [diff] [blame] | 45 | |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 46 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 47 | # ifdef CONFIG_FEATURE_DU_DEFALT_BLOCKSIZE_1K |
Eric Andersen | ec9fad9 | 2001-03-07 06:04:08 +0000 | [diff] [blame] | 48 | static unsigned long disp_hr = KILOBYTE; |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 49 | # else |
| 50 | static unsigned long disp_hr = 512; |
| 51 | # endif |
| 52 | #elif defined CONFIG_FEATURE_DU_DEFALT_BLOCKSIZE_1K |
| 53 | static unsigned int disp_k = 1; |
| 54 | #else |
| 55 | static unsigned int disp_k; /* bss inits to 0 */ |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 56 | #endif |
| 57 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 58 | static int max_print_depth = INT_MAX; |
| 59 | static int count_hardlinks = INT_MAX; |
| 60 | |
| 61 | static int status |
| 62 | #if EXIT_SUCCESS == 0 |
| 63 | = EXIT_SUCCESS |
| 64 | #endif |
| 65 | ; |
| 66 | static int print_files; |
| 67 | static int slink_depth; |
| 68 | static int du_depth; |
| 69 | static int one_file_system; |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 70 | static dev_t dir_dev; |
John Beppu | e1618e4 | 1999-12-15 18:52:17 +0000 | [diff] [blame] | 71 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 72 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 73 | static void print(long size, char *filename) |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 74 | { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 75 | /* TODO - May not want to defer error checking here. */ |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 76 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 77 | 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] | 78 | filename); |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 79 | #else |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 80 | bb_printf("%ld\t%s\n", size >> disp_k, filename); |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 81 | #endif |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 82 | } |
| 83 | |
| 84 | /* tiny recursive du */ |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 85 | static long du(char *filename) |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 86 | { |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 87 | struct stat statbuf; |
Erik Andersen | fac10d7 | 2000-02-07 05:29:42 +0000 | [diff] [blame] | 88 | long sum; |
John Beppu | 14c82b6 | 1999-12-10 06:15:27 +0000 | [diff] [blame] | 89 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 90 | if ((lstat(filename, &statbuf)) != 0) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 91 | bb_perror_msg("%s", filename); |
| 92 | status = EXIT_FAILURE; |
Eric Andersen | e5dfced | 2001-04-09 22:48:12 +0000 | [diff] [blame] | 93 | return 0; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 94 | } |
| 95 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 96 | if (one_file_system) { |
| 97 | if (du_depth == 0) { |
| 98 | dir_dev = statbuf.st_dev; |
| 99 | } else if (dir_dev != statbuf.st_dev) { |
| 100 | return 0; |
Eric Andersen | 8fa1bf7 | 2001-06-30 17:54:20 +0000 | [diff] [blame] | 101 | } |
Erik Andersen | 9ffdaa6 | 2000-02-11 21:55:04 +0000 | [diff] [blame] | 102 | } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 103 | |
| 104 | sum = statbuf.st_blocks; |
| 105 | |
| 106 | if (S_ISLNK(statbuf.st_mode)) { |
| 107 | if (slink_depth > du_depth) { /* -H or -L */ |
| 108 | if ((stat(filename, &statbuf)) != 0) { |
| 109 | bb_perror_msg("%s", filename); |
| 110 | status = EXIT_FAILURE; |
| 111 | return 0; |
| 112 | } |
| 113 | sum = statbuf.st_blocks; |
| 114 | if (slink_depth == 1) { |
| 115 | slink_depth = INT_MAX; /* Convert -H to -L. */ |
| 116 | } |
| 117 | } |
| 118 | } |
| 119 | |
| 120 | if (statbuf.st_nlink > count_hardlinks) { |
| 121 | /* Add files/directories with links only once */ |
| 122 | if (is_in_ino_dev_hashtable(&statbuf, NULL)) { |
| 123 | return 0; |
| 124 | } |
| 125 | add_to_ino_dev_hashtable(&statbuf, NULL); |
| 126 | } |
| 127 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 128 | if (S_ISDIR(statbuf.st_mode)) { |
| 129 | DIR *dir; |
| 130 | struct dirent *entry; |
Eric Andersen | 04b0354 | 2001-05-07 22:49:43 +0000 | [diff] [blame] | 131 | char *newfile; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 132 | |
| 133 | dir = opendir(filename); |
| 134 | if (!dir) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 135 | bb_perror_msg("%s", filename); |
| 136 | status = EXIT_FAILURE; |
| 137 | return sum; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 138 | } |
Erik Andersen | 42387e4 | 2000-02-21 17:27:17 +0000 | [diff] [blame] | 139 | |
Eric Andersen | 04b0354 | 2001-05-07 22:49:43 +0000 | [diff] [blame] | 140 | newfile = last_char_is(filename, '/'); |
| 141 | if (newfile) |
| 142 | *newfile = '\0'; |
Erik Andersen | 42387e4 | 2000-02-21 17:27:17 +0000 | [diff] [blame] | 143 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 144 | while ((entry = readdir(dir))) { |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 145 | char *name = entry->d_name; |
| 146 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 147 | if ((name[0] == '.') && (!name[1] || (name[1] == '.' && !name[2]))) { |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 148 | continue; |
| 149 | } |
Eric Andersen | e5dfced | 2001-04-09 22:48:12 +0000 | [diff] [blame] | 150 | newfile = concat_path_file(filename, name); |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 151 | ++du_depth; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 152 | sum += du(newfile); |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 153 | --du_depth; |
Eric Andersen | e5dfced | 2001-04-09 22:48:12 +0000 | [diff] [blame] | 154 | free(newfile); |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 155 | } |
| 156 | closedir(dir); |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 157 | } else if (du_depth > print_files) { |
| 158 | return sum; |
Erik Andersen | 27fdd08 | 2000-02-19 18:16:49 +0000 | [diff] [blame] | 159 | } |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 160 | if (du_depth <= max_print_depth) { |
| 161 | print(sum, filename); |
| 162 | } |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 163 | return sum; |
John Beppu | 0f5e1ab | 1999-12-09 18:23:54 +0000 | [diff] [blame] | 164 | } |
| 165 | |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 166 | int du_main(int argc, char **argv) |
| 167 | { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 168 | long total; |
| 169 | int slink_depth_save; |
| 170 | int print_final_total = 0; |
Pavel Roskin | 47d4926 | 2000-07-17 16:17:19 +0000 | [diff] [blame] | 171 | int c; |
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 | #ifdef CONFIG_FEATURE_DU_DEFALT_BLOCKSIZE_1K |
| 174 | if (getenv("POSIXLY_CORRECT")) { /* TODO - a new libbb function? */ |
| 175 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
| 176 | disp_hr = 512; |
| 177 | #else |
| 178 | disp_k = 0; |
| 179 | #endif |
| 180 | } |
| 181 | #endif |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 182 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 183 | /* Note: SUSv3 specifies that -a and -s options can not be used together |
| 184 | * in strictly conforming applications. However, it also says that some |
| 185 | * du implementations may produce output when -a and -s are used together. |
| 186 | * gnu du exits with an error code in this case. We choose to simply |
| 187 | * ignore -a. This is consistent with -s being equivalent to -d 0. |
| 188 | */ |
| 189 | |
| 190 | while ((c = getopt(argc, argv, "aHkLsx" "d:" "lc" |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 191 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 192 | "hm" |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 193 | #endif |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 194 | )) > 0) { |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 195 | switch (c) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 196 | case 'a': |
| 197 | print_files = INT_MAX; |
| 198 | break; |
| 199 | case 'H': |
| 200 | slink_depth = 1; |
| 201 | break; |
| 202 | case 'k': |
| 203 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
| 204 | disp_hr = KILOBYTE; |
| 205 | #elif !defined CONFIG_FEATURE_DU_DEFALT_BLOCKSIZE_1K |
| 206 | disp_k = 1; |
| 207 | #endif |
| 208 | break; |
| 209 | case 'L': |
| 210 | slink_depth = INT_MAX; |
| 211 | break; |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 212 | case 's': |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 213 | max_print_depth = 0; |
| 214 | break; |
| 215 | case 'x': |
| 216 | one_file_system = 1; |
| 217 | break; |
| 218 | |
| 219 | case 'd': |
| 220 | max_print_depth = bb_xgetularg10_bnd(optarg, 0, INT_MAX); |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 221 | break; |
| 222 | case 'l': |
| 223 | count_hardlinks = 1; |
| 224 | break; |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 225 | case 'c': |
| 226 | print_final_total = 1; |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 227 | break; |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 228 | #ifdef CONFIG_FEATURE_HUMAN_READABLE |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 229 | case 'h': |
| 230 | disp_hr = 0; |
| 231 | break; |
| 232 | case 'm': |
| 233 | disp_hr = MEGABYTE; |
| 234 | break; |
Richard June | 6d0921c | 2001-01-22 22:35:38 +0000 | [diff] [blame] | 235 | #endif |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 236 | default: |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 237 | bb_show_usage(); |
Glenn L McGrath | 14dad70 | 2002-08-23 03:25:22 +0000 | [diff] [blame] | 238 | } |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 239 | } |
| 240 | |
| 241 | /* go through remaining args (if any) */ |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 242 | argv += optind; |
Eric Andersen | 17ad45a | 2000-07-14 18:38:26 +0000 | [diff] [blame] | 243 | if (optind >= argc) { |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 244 | *--argv = "."; |
| 245 | if (slink_depth == 1) { |
| 246 | slink_depth = 0; |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 247 | } |
| 248 | } |
| 249 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 250 | slink_depth_save = slink_depth; |
| 251 | total = 0; |
| 252 | do { |
| 253 | total += du(*argv); |
| 254 | slink_depth = slink_depth_save; |
| 255 | } while (*++argv); |
| 256 | reset_ino_dev_hashtable(); |
Erik Andersen | e49d5ec | 2000-02-08 19:58:47 +0000 | [diff] [blame] | 257 | |
Manuel Novoa III | cad5364 | 2003-03-19 09:13:01 +0000 | [diff] [blame^] | 258 | if (print_final_total) { |
| 259 | print(total, "total"); |
| 260 | } |
| 261 | |
| 262 | bb_fflush_stdout_and_exit(status); |
| 263 | } |