diff options
-rw-r--r-- | src/du.c | 657 |
1 files changed, 322 insertions, 335 deletions
@@ -109,19 +109,6 @@ typedef struct String String; int stat (); int lstat (); -static int hash_insert PARAMS ((ino_t ino, dev_t dev)); -static int hash_insert2 PARAMS ((struct htab *_htab, ino_t ino, dev_t dev)); -static uintmax_t count_entry PARAMS ((const char *ent, int top, dev_t last_dev, - int depth)); -static void du_files PARAMS ((char **files)); -static void hash_init PARAMS ((unsigned int modulus, - unsigned int entry_tab_size)); -static void hash_reset PARAMS ((void)); -static void str_concatc PARAMS ((String *s1, char *cstr)); -static void str_copyc PARAMS ((String *s1, char *cstr)); -static void str_init PARAMS ((String **s1, unsigned int size)); -static void str_trunc PARAMS ((String *s1, unsigned int length)); - /* Name under which this program was invoked. */ char *program_name; @@ -239,230 +226,226 @@ Summarize disk usage of each FILE, recursively for directories.\n\ exit (status); } -int -main (int argc, char **argv) +/* Initialize string S1 to hold SIZE characters. */ + +static void +str_init (String **s1, unsigned int size) { - int c; - char *cwd_only[2]; - int max_depth_specified = 0; + String *s; - /* If nonzero, display only a total for each argument. */ - int opt_summarize_only = 0; + s = (String *) xmalloc (sizeof (struct String)); + s->text = xmalloc (size + 1); - cwd_only[0] = "."; - cwd_only[1] = NULL; + s->alloc = size; + *s1 = s; +} - program_name = argv[0]; - setlocale (LC_ALL, ""); - bindtextdomain (PACKAGE, LOCALEDIR); - textdomain (PACKAGE); +static void +ensure_space (String *s, unsigned int size) +{ + if (s->alloc < size) + { + s->text = xrealloc (s->text, size + 1); + s->alloc = size; + } +} - exclude = new_exclude (); - xstat = lstat; +/* Assign the null-terminated C-string CSTR to S1. */ - human_block_size (getenv ("DU_BLOCK_SIZE"), 0, &output_block_size); +static void +str_copyc (String *s1, const char *cstr) +{ + unsigned l = strlen (cstr); + ensure_space (s1, l); + strcpy (s1->text, cstr); + s1->length = l; +} - while ((c = getopt_long (argc, argv, "abchHklmsxDLSX:", long_options, NULL)) - != -1) - { - long int tmp_long; - switch (c) - { - case 0: /* Long option. */ - break; +static void +str_concatc (String *s1, const char *cstr) +{ + unsigned l1 = s1->length; + unsigned l2 = strlen (cstr); + unsigned l = l1 + l2; - case 'a': - opt_all = 1; - break; + ensure_space (s1, l); + strcpy (s1->text + l1, cstr); + s1->length = l; +} - case 'b': - output_block_size = 1; - break; +/* Truncate the string S1 to have length LENGTH. */ - case 'c': - opt_combined_arguments = 1; - break; +static void +str_trunc (String *s1, unsigned int length) +{ + if (s1->length > length) + { + s1->text[length] = 0; + s1->length = length; + } +} - case 'h': - output_block_size = -1024; - break; +/* Print N_BLOCKS followed by STRING on a line. NBLOCKS is the number of + ST_NBLOCKSIZE-byte blocks; convert it to OUTPUT_BLOCK_SIZE units before + printing. If OUTPUT_BLOCK_SIZE is negative, use a human readable + notation instead. */ - case 'H': - output_block_size = -1000; - break; +static void +print_size (uintmax_t n_blocks, const char *string) +{ + char buf[LONGEST_HUMAN_READABLE + 1]; + printf ("%s\t%s\n", + human_readable (n_blocks, buf, ST_NBLOCKSIZE, output_block_size), + string); + fflush (stdout); +} - case 'k': - output_block_size = 1024; - break; +/* Reset the hash structure in the global variable `htab' to + contain no entries. */ - case CHAR_MAX + 3: /* --max-depth=N */ - if (xstrtol (optarg, NULL, 0, &tmp_long, NULL) != LONGINT_OK - || tmp_long < 0 || tmp_long > INT_MAX) - error (1, 0, _("invalid maximum depth `%s'"), optarg); +static void +hash_reset (void) +{ + int i; + struct entry **p; - max_depth_specified = 1; - max_depth = (int) tmp_long; - break; + htab->first_free_entry = 0; - case 'm': - output_block_size = 1024 * 1024; - break; + p = htab->hash; + for (i = htab->modulus; i > 0; i--) + *p++ = NULL; +} - case 'l': - opt_count_all = 1; - break; +/* Allocate space for the hash structures, and set the global + variable `htab' to point to it. The initial hash module is specified in + MODULUS, and the number of entries are specified in ENTRY_TAB_SIZE. (The + hash structure will be rebuilt when ENTRY_TAB_SIZE entries have been + inserted, and MODULUS and ENTRY_TAB_SIZE in the global `htab' will be + doubled.) */ - case 's': - opt_summarize_only = 1; - break; +static void +hash_init (unsigned int modulus, unsigned int entry_tab_size) +{ + struct htab *htab_r; - case 'x': - opt_one_file_system = 1; - break; + htab_r = (struct htab *) + xmalloc (sizeof (struct htab) + sizeof (struct entry *) * modulus); - case 'D': - opt_dereference_arguments = 1; - break; + htab_r->entry_tab = (struct entry *) + xmalloc (sizeof (struct entry) * entry_tab_size); - case 'L': - xstat = stat; - break; + htab_r->modulus = modulus; + htab_r->entry_tab_size = entry_tab_size; + htab = htab_r; - case 'S': - opt_separate_dirs = 1; - break; + hash_reset (); +} - case 'X': - if (add_exclude_file (exclude, optarg, '\n') != 0) - error (1, errno, "%s", optarg); - break; +/* Insert INO and DEV in the hash structure HTAB, if not + already present. Return zero if inserted and nonzero if it + already existed. */ - case CHAR_MAX + 1: - add_exclude (exclude, optarg); - break; +static int +hash_insert2 (struct htab *ht, ino_t ino, dev_t dev) +{ + struct entry **hp, *ep2, *ep; + hp = &ht->hash[ino % ht->modulus]; + ep2 = *hp; - case CHAR_MAX + 2: - human_block_size (optarg, 1, &output_block_size); - break; + /* Collision? */ - case_GETOPT_HELP_CHAR; + if (ep2 != NULL) + { + ep = ep2; - case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS); + /* Search for an entry with the same data. */ - default: - usage (1); + do + { + if (ep->ino == ino && ep->dev == dev) + return 1; /* Found an entry with the same data. */ + ep = ep->coll_link; } - } + while (ep != NULL); - if (opt_all && opt_summarize_only) - { - error (0, 0, _("cannot both summarize and show all entries")); - usage (1); - } + /* Did not find it. */ - if (opt_summarize_only && max_depth_specified && max_depth == 0) - { - error (0, 0, - _("warning: summarizing is the same as using --max-depth=0")); } - if (opt_summarize_only && max_depth_specified && max_depth != 0) - { - error (0, 0, - _("warning: summarizing conflicts with --max-depth=%d"), - max_depth); - usage (1); - } + ep = *hp = &ht->entry_tab[ht->first_free_entry++]; + ep->ino = ino; + ep->dev = dev; + ep->coll_link = ep2; /* `ep2' is NULL if no collision. */ - if (opt_summarize_only) - max_depth = 0; + return 0; +} - /* Initialize the hash structure for inode numbers. */ - hash_init (INITIAL_HASH_MODULE, INITIAL_ENTRY_TAB_SIZE); +/* Insert an item (inode INO and device DEV) in the hash + structure in the global variable `htab', if an entry with the same data + was not found already. Return zero if the item was inserted and nonzero + if it wasn't. */ - str_init (&path, INITIAL_PATH_SIZE); +static int +hash_insert (ino_t ino, dev_t dev) +{ + struct htab *htab_r = htab; /* Initially a copy of the global `htab'. */ - du_files (optind == argc ? cwd_only : argv + optind); + if (htab_r->first_free_entry >= htab_r->entry_tab_size) + { + int i; + struct entry *ep; + unsigned modulus; + unsigned entry_tab_size; - close_stdout (); - exit (exit_status); -} + /* Increase the number of hash entries, and re-hash the data. + The method of shrimping and increasing is made to compactify + the heap. If twice as much data would be allocated + straightforwardly, we would never re-use a byte of memory. */ -/* Print N_BLOCKS followed by STRING on a line. NBLOCKS is the number of - ST_NBLOCKSIZE-byte blocks; convert it to OUTPUT_BLOCK_SIZE units before - printing. If OUTPUT_BLOCK_SIZE is negative, use a human readable - notation instead. */ + /* Let `htab' shrimp. Keep only the header, not the pointer vector. */ -static void -print_size (uintmax_t n_blocks, const char *string) -{ - char buf[LONGEST_HUMAN_READABLE + 1]; - printf ("%s\t%s\n", - human_readable (n_blocks, buf, ST_NBLOCKSIZE, output_block_size), - string); - fflush (stdout); -} + htab_r = (struct htab *) + xrealloc ((char *) htab_r, sizeof (struct htab)); -/* Recursively print the sizes of the directories (and, if selected, files) - named in FILES, the last entry of which is NULL. */ + modulus = 2 * htab_r->modulus; + entry_tab_size = 2 * htab_r->entry_tab_size; -static void -du_files (char **files) -{ - struct saved_cwd cwd; - ino_t initial_ino; /* Initial directory's inode. */ - dev_t initial_dev; /* Initial directory's device. */ - int i; /* Index in FILES. */ + /* Increase the number of possible entries. */ - if (save_cwd (&cwd)) - exit (1); + htab_r->entry_tab = (struct entry *) + xrealloc ((char *) htab_r->entry_tab, + sizeof (struct entry) * entry_tab_size); - /* Remember the inode and device number of the current directory. */ - if (stat (".", &stat_buf)) - error (1, errno, _("current directory")); - initial_ino = stat_buf.st_ino; - initial_dev = stat_buf.st_dev; + /* Increase the size of htab again. */ - for (i = 0; files[i]; i++) - { - char *arg; - int s; + htab_r = (struct htab *) + xrealloc ((char *) htab_r, + sizeof (struct htab) + sizeof (struct entry *) * modulus); - arg = files[i]; + htab_r->modulus = modulus; + htab_r->entry_tab_size = entry_tab_size; + htab = htab_r; - /* Delete final slash in the argument, unless the slash is alone. */ - s = strlen (arg) - 1; - if (s != 0) - { - if (arg[s] == '/') - arg[s] = 0; + i = htab_r->first_free_entry; - str_copyc (path, arg); - } - else if (arg[0] == '/') - str_trunc (path, 0); /* Null path for root directory. */ - else - str_copyc (path, arg); + /* Make the increased hash table empty. The entries are still + available in htab->entry_tab. */ - if (!opt_combined_arguments) - hash_reset (); + hash_reset (); - count_entry (arg, 1, 0, 0); + /* Go through the entries and install them in the pointer vector + htab->hash. The items are actually inserted in htab->entry_tab at + the position where they already are. The htab->coll_link need + however be updated. Could be made a little more efficient. */ - /* chdir if `count_entry' has changed the working directory. */ - if (stat (".", &stat_buf)) - error (1, errno, "."); - if (stat_buf.st_ino != initial_ino || stat_buf.st_dev != initial_dev) + for (ep = htab_r->entry_tab; i > 0; i--) { - if (restore_cwd (&cwd, _("starting directory"), NULL)) - exit (1); + hash_insert2 (htab_r, ep->ino, ep->dev); + ep++; } } - if (opt_combined_arguments) - print_size (tot_size, _("total")); - - free_cwd (&cwd); + return hash_insert2 (htab_r, ino, dev); } /* Print (if appropriate) the size (in units determined by `output_block_size') @@ -594,209 +577,213 @@ count_entry (const char *ent, int top, dev_t last_dev, int depth) return size; } -/* Allocate space for the hash structures, and set the global - variable `htab' to point to it. The initial hash module is specified in - MODULUS, and the number of entries are specified in ENTRY_TAB_SIZE. (The - hash structure will be rebuilt when ENTRY_TAB_SIZE entries have been - inserted, and MODULUS and ENTRY_TAB_SIZE in the global `htab' will be - doubled.) */ +/* Recursively print the sizes of the directories (and, if selected, files) + named in FILES, the last entry of which is NULL. */ static void -hash_init (unsigned int modulus, unsigned int entry_tab_size) +du_files (char **files) { - struct htab *htab_r; + struct saved_cwd cwd; + ino_t initial_ino; /* Initial directory's inode. */ + dev_t initial_dev; /* Initial directory's device. */ + int i; /* Index in FILES. */ - htab_r = (struct htab *) - xmalloc (sizeof (struct htab) + sizeof (struct entry *) * modulus); + if (save_cwd (&cwd)) + exit (1); - htab_r->entry_tab = (struct entry *) - xmalloc (sizeof (struct entry) * entry_tab_size); + /* Remember the inode and device number of the current directory. */ + if (stat (".", &stat_buf)) + error (1, errno, _("current directory")); + initial_ino = stat_buf.st_ino; + initial_dev = stat_buf.st_dev; - htab_r->modulus = modulus; - htab_r->entry_tab_size = entry_tab_size; - htab = htab_r; + for (i = 0; files[i]; i++) + { + char *arg; + int s; - hash_reset (); -} + arg = files[i]; -/* Reset the hash structure in the global variable `htab' to - contain no entries. */ + /* Delete final slash in the argument, unless the slash is alone. */ + s = strlen (arg) - 1; + if (s != 0) + { + if (arg[s] == '/') + arg[s] = 0; -static void -hash_reset (void) -{ - int i; - struct entry **p; + str_copyc (path, arg); + } + else if (arg[0] == '/') + str_trunc (path, 0); /* Null path for root directory. */ + else + str_copyc (path, arg); - htab->first_free_entry = 0; + if (!opt_combined_arguments) + hash_reset (); - p = htab->hash; - for (i = htab->modulus; i > 0; i--) - *p++ = NULL; -} + count_entry (arg, 1, 0, 0); -/* Insert an item (inode INO and device DEV) in the hash - structure in the global variable `htab', if an entry with the same data - was not found already. Return zero if the item was inserted and nonzero - if it wasn't. */ + /* chdir if `count_entry' has changed the working directory. */ + if (stat (".", &stat_buf)) + error (1, errno, "."); + if (stat_buf.st_ino != initial_ino || stat_buf.st_dev != initial_dev) + { + if (restore_cwd (&cwd, _("starting directory"), NULL)) + exit (1); + } + } -static int -hash_insert (ino_t ino, dev_t dev) -{ - struct htab *htab_r = htab; /* Initially a copy of the global `htab'. */ + if (opt_combined_arguments) + print_size (tot_size, _("total")); - if (htab_r->first_free_entry >= htab_r->entry_tab_size) - { - int i; - struct entry *ep; - unsigned modulus; - unsigned entry_tab_size; + free_cwd (&cwd); +} - /* Increase the number of hash entries, and re-hash the data. - The method of shrimping and increasing is made to compactify - the heap. If twice as much data would be allocated - straightforwardly, we would never re-use a byte of memory. */ +int +main (int argc, char **argv) +{ + int c; + char *cwd_only[2]; + int max_depth_specified = 0; - /* Let `htab' shrimp. Keep only the header, not the pointer vector. */ + /* If nonzero, display only a total for each argument. */ + int opt_summarize_only = 0; - htab_r = (struct htab *) - xrealloc ((char *) htab_r, sizeof (struct htab)); + cwd_only[0] = "."; + cwd_only[1] = NULL; - modulus = 2 * htab_r->modulus; - entry_tab_size = 2 * htab_r->entry_tab_size; + program_name = argv[0]; + setlocale (LC_ALL, ""); + bindtextdomain (PACKAGE, LOCALEDIR); + textdomain (PACKAGE); - /* Increase the number of possible entries. */ + exclude = new_exclude (); + xstat = lstat; - htab_r->entry_tab = (struct entry *) - xrealloc ((char *) htab_r->entry_tab, - sizeof (struct entry) * entry_tab_size); + human_block_size (getenv ("DU_BLOCK_SIZE"), 0, &output_block_size); - /* Increase the size of htab again. */ + while ((c = getopt_long (argc, argv, "abchHklmsxDLSX:", long_options, NULL)) + != -1) + { + long int tmp_long; + switch (c) + { + case 0: /* Long option. */ + break; - htab_r = (struct htab *) - xrealloc ((char *) htab_r, - sizeof (struct htab) + sizeof (struct entry *) * modulus); + case 'a': + opt_all = 1; + break; - htab_r->modulus = modulus; - htab_r->entry_tab_size = entry_tab_size; - htab = htab_r; + case 'b': + output_block_size = 1; + break; - i = htab_r->first_free_entry; + case 'c': + opt_combined_arguments = 1; + break; - /* Make the increased hash table empty. The entries are still - available in htab->entry_tab. */ + case 'h': + output_block_size = -1024; + break; - hash_reset (); + case 'H': + output_block_size = -1000; + break; - /* Go through the entries and install them in the pointer vector - htab->hash. The items are actually inserted in htab->entry_tab at - the position where they already are. The htab->coll_link need - however be updated. Could be made a little more efficient. */ + case 'k': + output_block_size = 1024; + break; - for (ep = htab_r->entry_tab; i > 0; i--) - { - hash_insert2 (htab_r, ep->ino, ep->dev); - ep++; - } - } + case CHAR_MAX + 3: /* --max-depth=N */ + if (xstrtol (optarg, NULL, 0, &tmp_long, NULL) != LONGINT_OK + || tmp_long < 0 || tmp_long > INT_MAX) + error (1, 0, _("invalid maximum depth `%s'"), optarg); - return hash_insert2 (htab_r, ino, dev); -} + max_depth_specified = 1; + max_depth = (int) tmp_long; + break; -/* Insert INO and DEV in the hash structure HTAB, if not - already present. Return zero if inserted and nonzero if it - already existed. */ + case 'm': + output_block_size = 1024 * 1024; + break; -static int -hash_insert2 (struct htab *ht, ino_t ino, dev_t dev) -{ - struct entry **hp, *ep2, *ep; - hp = &ht->hash[ino % ht->modulus]; - ep2 = *hp; + case 'l': + opt_count_all = 1; + break; - /* Collision? */ + case 's': + opt_summarize_only = 1; + break; - if (ep2 != NULL) - { - ep = ep2; + case 'x': + opt_one_file_system = 1; + break; - /* Search for an entry with the same data. */ + case 'D': + opt_dereference_arguments = 1; + break; - do - { - if (ep->ino == ino && ep->dev == dev) - return 1; /* Found an entry with the same data. */ - ep = ep->coll_link; - } - while (ep != NULL); + case 'L': + xstat = stat; + break; - /* Did not find it. */ + case 'S': + opt_separate_dirs = 1; + break; - } + case 'X': + if (add_exclude_file (exclude, optarg, '\n') != 0) + error (1, errno, "%s", optarg); + break; - ep = *hp = &ht->entry_tab[ht->first_free_entry++]; - ep->ino = ino; - ep->dev = dev; - ep->coll_link = ep2; /* `ep2' is NULL if no collision. */ + case CHAR_MAX + 1: + add_exclude (exclude, optarg); + break; - return 0; -} + case CHAR_MAX + 2: + human_block_size (optarg, 1, &output_block_size); + break; -/* Initialize string S1 to hold SIZE characters. */ + case_GETOPT_HELP_CHAR; -static void -str_init (String **s1, unsigned int size) -{ - String *s; + case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS); - s = (String *) xmalloc (sizeof (struct String)); - s->text = xmalloc (size + 1); + default: + usage (1); + } + } - s->alloc = size; - *s1 = s; -} + if (opt_all && opt_summarize_only) + { + error (0, 0, _("cannot both summarize and show all entries")); + usage (1); + } -static void -ensure_space (String *s, unsigned int size) -{ - if (s->alloc < size) + if (opt_summarize_only && max_depth_specified && max_depth == 0) { - s->text = xrealloc (s->text, size + 1); - s->alloc = size; + error (0, 0, + _("warning: summarizing is the same as using --max-depth=0")); } -} -/* Assign the null-terminated C-string CSTR to S1. */ + if (opt_summarize_only && max_depth_specified && max_depth != 0) + { + error (0, 0, + _("warning: summarizing conflicts with --max-depth=%d"), + max_depth); + usage (1); + } -static void -str_copyc (String *s1, char *cstr) -{ - unsigned l = strlen (cstr); - ensure_space (s1, l); - strcpy (s1->text, cstr); - s1->length = l; -} + if (opt_summarize_only) + max_depth = 0; -static void -str_concatc (String *s1, char *cstr) -{ - unsigned l1 = s1->length; - unsigned l2 = strlen (cstr); - unsigned l = l1 + l2; + /* Initialize the hash structure for inode numbers. */ + hash_init (INITIAL_HASH_MODULE, INITIAL_ENTRY_TAB_SIZE); - ensure_space (s1, l); - strcpy (s1->text + l1, cstr); - s1->length = l; -} + str_init (&path, INITIAL_PATH_SIZE); -/* Truncate the string S1 to have length LENGTH. */ + du_files (optind == argc ? cwd_only : argv + optind); -static void -str_trunc (String *s1, unsigned int length) -{ - if (s1->length > length) - { - s1->text[length] = 0; - s1->length = length; - } + close_stdout (); + exit (exit_status); } |