]> Dogcows Code - chaz/tar/blobdiff - src/tar.c
(decode_options): Don't pass names longer than UNAME_FIELD_SIZE to
[chaz/tar] / src / tar.c
index 24b0f90bf72097c17f9dd5da1ca734397968d49c..10ac93c699a894620baf8c372d88724f1eb8a900 100644 (file)
--- a/src/tar.c
+++ b/src/tar.c
@@ -1,5 +1,5 @@
 /* A tar (tape archiver) program.
-   Copyright (C) 1988, 92, 93, 94, 95, 96, 97 Free Software Foundation, Inc.
+   Copyright (C) 1988, 92,93,94,95,96,97, 1999 Free Software Foundation, Inc.
    Written by John Gilmore, starting 1985-08-25.
 
    This program is free software; you can redistribute it and/or modify it
@@ -31,8 +31,8 @@
 enum backup_type get_version ();
 
 /* FIXME: We should use a conversion routine that does reasonable error
-   checking -- atoi doesn't.  For now, punt.  */
-#define intconv        atoi
+   checking -- atol doesn't.  For now, punt.  */
+#define intconv        atol
 
 time_t get_date ();
 
@@ -51,16 +51,17 @@ static void usage PARAMS ((int));
 /* Miscellaneous.  */
 
 /*------------------------------------------------------------------------.
-| Check if STRING is the decimal representation of number, and return its |
-| value.  If not a decimal number, return -1.                            |
+| Check if STRING0 is the decimal representation of number, and store its |
+| value.  If not a decimal number, return 0.                             |
 `------------------------------------------------------------------------*/
 
 static int
-check_decimal (const char *string)
+check_decimal (const char *string0, uintmax_t *result)
 {
-  int value = -1;
+  const char *string = string0;
+  uintmax_t value = 0;
 
-  while (*string)
+  do
     switch (*string)
       {
       case '0':
@@ -73,14 +74,22 @@ check_decimal (const char *string)
       case '7':
       case '8':
       case '9':
-       value = value < 0 ? *string - '0' : 10 * value + *string - '0';
-       string++;
+       {
+         uintmax_t v10 = value * 10;
+         uintmax_t v10d = v10 + (*string - '0');
+         if (v10 / 10 != value || v10d < v10)
+           return 0;
+         value = v10d;
+       }
        break;
 
       default:
-       return -1;
+       return 0;
       }
-    return value;
+  while (*++string);
+
+  *result = value;
+  return 1;
 }
 
 /*----------------------------------------------.
@@ -208,6 +217,7 @@ struct option long_options[] =
   {"dereference", no_argument, NULL, 'h'},
   {"diff", no_argument, NULL, 'd'},
   {"directory", required_argument, NULL, 'C'},
+  {"ending-file", required_argument, NULL, 'E'},
   {"exclude", required_argument, NULL, EXCLUDE_OPTION},
   {"exclude-from", required_argument, NULL, 'X'},
   {"extract", no_argument, NULL, 'x'},
@@ -388,6 +398,7 @@ Local file selection:\n\
   -h, --dereference            dump instead the files symlinks point to\n\
       --no-recursion           avoid descending automatically in directories\n\
   -l, --one-file-system        stay in local file system when creating archive\n\
+  -E, --ending-file=NAME       end reading the archive before file NAME\n\
   -K, --starting-file=NAME     begin at file NAME in the archive\n"),
             stdout);
 #if !MSDOS
@@ -442,13 +453,13 @@ Report bugs to <tar-bugs@gnu.ai.mit.edu>.\n"),
 | Parse the options for tar.  |
 `----------------------------*/
 
-/* Available option letters are DEHIJQY and aejnqy.  Some are reserved:
+/* Available option letters are DHIJQY and aejnqy.  Some are reserved:
 
    y  per-file gzip compression
    Y  per-block gzip compression */
 
 #define OPTION_STRING \
-  "-01234567ABC:F:GK:L:MN:OPRST:UV:WX:Zb:cdf:g:hiklmoprstuvwxz"
+  "-01234567ABC:E:F:GK:L:MN:OPRST:UV:WX:Zb:cdf:g:hiklmoprstuvwxz"
 
 static void
 set_subcommand_option (enum subcommand subcommand)
@@ -526,11 +537,13 @@ decode_options (int argc, char *const *argv)
          *out++ = xstrdup (buffer);
          cursor = strchr (OPTION_STRING, *letter);
          if (cursor && cursor[1] == ':')
-           if (in < argv + argc)
-             *out++ = *in++;
-           else
-             USAGE_ERROR ((0, 0, _("Old option `%c' requires an argument."),
-                           *letter));
+           {
+             if (in < argv + argc)
+               *out++ = *in++;
+             else
+               USAGE_ERROR ((0, 0, _("Old option `%c' requires an argument."),
+                             *letter));
+           }
        }
 
       /* Copy all remaining options.  */
@@ -580,7 +593,7 @@ decode_options (int argc, char *const *argv)
 
       case 'b':
        blocking_factor = intconv (optarg);
-       record_size = blocking_factor * BLOCKSIZE;
+       record_size = blocking_factor * (size_t) BLOCKSIZE;
        break;
 
       case OBSOLETE_READ_FULL_RECORDS:
@@ -612,6 +625,10 @@ decode_options (int argc, char *const *argv)
        set_subcommand_option (DIFF_SUBCOMMAND);
        break;
 
+      case 'E':
+       ending_file_option = optarg;
+       break;
+
       case 'f':
        if (archive_names == allocated_archive_names)
          {
@@ -835,11 +852,15 @@ decode_options (int argc, char *const *argv)
        break;
 
       case GROUP_OPTION:
-       if (!gname_to_gid (optarg, &group_option))
-         if (!check_decimal (optarg) >= 0)
-           ERROR ((TAREXIT_FAILURE, 0, _("Invalid group given on option")));
-         else
-           group_option = check_decimal (optarg);
+       if (! (strlen (optarg) < GNAME_FIELD_SIZE
+              && gname_to_gid (optarg, &group_option)))
+         {
+           uintmax_t g;
+           if (!check_decimal (optarg, &g) || g != (gid_t) g)
+             ERROR ((TAREXIT_FAILURE, 0, _("Invalid group given on option")));
+           else
+             group_option = g;
+         }
        break;
 
       case MODE_OPTION:
@@ -861,11 +882,15 @@ decode_options (int argc, char *const *argv)
        break;
 
       case OWNER_OPTION:
-       if (!uname_to_uid (optarg, &owner_option))
-         if (!check_decimal (optarg) >= 0)
-           ERROR ((TAREXIT_FAILURE, 0, _("Invalid owner given on option")));
-         else
-           owner_option = check_decimal (optarg);
+       if (! (strlen (optarg) < UNAME_FIELD_SIZE
+              && uname_to_uid (optarg, &owner_option)))
+         {
+           uintmax_t u;
+           if (!check_decimal (optarg, &u) || u != (uid_t) u)
+             ERROR ((TAREXIT_FAILURE, 0, _("Invalid owner given on option")));
+           else
+             owner_option = u;
+         }
        break;
 
       case POSIX_OPTION:
This page took 0.023642 seconds and 4 git commands to generate.