summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--lib/Makefile.in2
-rw-r--r--old/fileutils/ChangeLog30
-rw-r--r--old/fileutils/NEWS5
-rw-r--r--src/cp.c29
-rw-r--r--src/install.c3
-rw-r--r--src/ln.c58
-rw-r--r--src/ls.c14
7 files changed, 107 insertions, 34 deletions
diff --git a/lib/Makefile.in b/lib/Makefile.in
index 3a71cac2b..ae30a2df5 100644
--- a/lib/Makefile.in
+++ b/lib/Makefile.in
@@ -24,7 +24,7 @@ CC = @CC@
AR = ar
RANLIB = @RANLIB@
DEFS = @DEFS@
-CFLAGS = -g
+CFLAGS = @CFLAGS@
SOURCES = getdate.y posixtm.y \
argmatch.c backupfile.c basename.c dirname.c eaccess.c \
diff --git a/old/fileutils/ChangeLog b/old/fileutils/ChangeLog
index b74560f97..4ededb20e 100644
--- a/old/fileutils/ChangeLog
+++ b/old/fileutils/ChangeLog
@@ -1,4 +1,21 @@
-Wed Aug 25 21:40:00 EDT 1993
+Thu Aug 26 22:26:09 1993 Jim Meyering (meyering@comco.com)
+
+ * ls.c (print_long_format, print_file_name_and_frills): Cast inode
+ number to unsigned long and print it with %lu to avoid warnings from
+ gcc -Wformat because the size and type of ino_t are system dependent.
+
+ * cp.c (do_copy): Plug a memory leak with --parents.
+
+ * ln.c (main): Like mv and cp, convert `ln x y/' to ln x y/x
+ when a is not a directory.
+ [PATH_BASENAME_CONCAT]: New macro.
+ (do_link): Use it here, too.
+
+ * ls.c (sort_files): Add `default: abort();' clause to switch stmts.
+
+ * cp.c (do_copy): Don't remove trailing slashes from source.
+
+Wed Aug 25 21:40:00 1993 Jim Meyering (meyering@comco.com)
* cp.c: Add --parents as synonym for --path. Change --path to
--parents in comments. --path is deprecated.
@@ -16,13 +33,12 @@ Fri Aug 13 17:19:52 1993 Jim Meyering (meyering@comco.com)
(is_real_dir): New function.
(movefile): In addition to when dest is a directory, if dest has
a trailing `/' and source is not a directory, presume the target
- is dest/`basename source`. This converts `mv a b/' to `mv a b/a'
- when a is not a directory -- so the command will fail when a is a
- non-directory and (b doesn't exist or b isn't a directory or a
- symlink to a directory).
+ is dest/`basename source`. This converts `mv x y/' to `mv x y/x'
+ when x is not a directory. This change means that the command
+ `mv any file/' will now fail rather than performing the move.
- * ls.c (do_copy): Similarly, convert `cp a b/' to cp a b/a when
- a is not a directory.
+ * cp.c (do_copy): Similarly, convert `cp x y/' to cp x y/x when
+ x is not a directory.
Wed Aug 4 17:43:18 1993 Jim Meyering (meyering@comco.com)
diff --git a/old/fileutils/NEWS b/old/fileutils/NEWS
index 16246d08f..e4121a4d9 100644
--- a/old/fileutils/NEWS
+++ b/old/fileutils/NEWS
@@ -2,6 +2,11 @@ Major changes in release 3.9:
* ls and cp can handle mount points on more systems
* cp, mkdir, and rmdir long option --path renamed to --parents; --path
will still work for a while
+* cp, ln, and mv convert `cp A B/' to cp A B/A when A is not a directory.
+ This change affects only the two-argument form of the commands. It makes
+ such commands fail when the target has a trailing slash but is not a
+ directory or symlink to a directory and the source is not a directory.
+ They used to succeed, ignoring the implicitly contradictory trailing slash.
Major changes in release 3.8:
* install isn't as likely to produce spurious errors
diff --git a/src/cp.c b/src/cp.c
index 7ba654679..f6dd934e2 100644
--- a/src/cp.c
+++ b/src/cp.c
@@ -41,6 +41,7 @@ struct dir_attr
};
char *dirname ();
+char *xstrdup ();
enum backup_type get_version ();
int eaccess_stat ();
@@ -375,7 +376,7 @@ do_copy (argc, argv)
ap = basename (arg);
/* For `cp -R source/.. dest', don't copy into `dest/..'. */
if (!strcmp (ap, ".."))
- dst_path = dest;
+ dst_path = xstrdup (dest);
else
{
dst_path = xmalloc (strlen (dest) + strlen (ap) + 2);
@@ -400,6 +401,7 @@ do_copy (argc, argv)
}
}
+ free (dst_path);
++optind;
if (optind == argc - 1)
break;
@@ -408,7 +410,7 @@ do_copy (argc, argv)
}
else if (argc - optind == 2)
{
- char *dst_path;
+ char *new_dest;
char *source;
struct stat source_stats;
@@ -417,23 +419,32 @@ do_copy (argc, argv)
source = argv[optind];
+ /* When the destination is specified with a trailing slash and the
+ source exists but is not a directory, convert the user's command
+ `cp source dest/' to `cp source dest/basename(source)'. */
+
if (dest[strlen (dest) - 1] == '/'
&& lstat (source, &source_stats) == 0
&& !S_ISDIR (source_stats.st_mode))
{
- char *base;
+ char *source_base;
+ char *tmp_source;
+
+ tmp_source = (char *) alloca (strlen (source) + 1);
+ strcpy (tmp_source, source);
+ strip_trailing_slashes (tmp_source);
+ source_base = basename (tmp_source);
- strip_trailing_slashes (source);
- base = basename (source);
- dst_path = (char *) alloca (strlen (dest) + 1 + strlen (base) + 1);
- stpcpy (stpcpy (stpcpy (dst_path, dest), "/"), base);
+ new_dest = (char *) alloca (strlen (dest) + 1 +
+ strlen (source_base) + 1);
+ stpcpy (stpcpy (stpcpy (new_dest, dest), "/"), source_base);
}
else
{
- dst_path = dest;
+ new_dest = dest;
}
- return copy (argv[optind], dst_path, new_dst, 0, (struct dir_list *) 0);
+ return copy (source, new_dest, new_dst, 0, (struct dir_list *) 0);
}
else
usage ("when copying multiple files, last argument must be a directory");
diff --git a/src/install.c b/src/install.c
index 9d579c22d..e6fc16276 100644
--- a/src/install.c
+++ b/src/install.c
@@ -91,6 +91,7 @@ int wait ();
#define READ_SIZE (32 * 1024)
char *basename ();
+char *stpcpy ();
char *xmalloc ();
void error ();
int make_path ();
@@ -291,7 +292,7 @@ install_file_in_dir (from, to_dir)
from_base = basename (from);
to = xmalloc ((unsigned) (strlen (to_dir) + strlen (from_base) + 2));
- sprintf (to, "%s/%s", to_dir, from_base);
+ stpcpy (stpcpy (stpcpy (to, to_dir), "/"), from_base);
ret = install_file_in_file (from, to);
free (to);
return ret;
diff --git a/src/ln.c b/src/ln.c
index 95938f533..e2ddda982 100644
--- a/src/ln.c
+++ b/src/ln.c
@@ -33,12 +33,33 @@ int link (); /* Some systems don't declare this anywhere. */
int symlink ();
#endif
+/* Construct a string NEW_DEST by concatenating DEST, a slash, and
+ basename(SOURCE) in alloca'd memory. Don't modify DEST or SOURCE. */
+
+#define PATH_BASENAME_CONCAT(new_dest, dest, source) \
+ do \
+ { \
+ char *source_base; \
+ char *tmp_source; \
+ \
+ tmp_source = (char *) alloca (strlen ((source)) + 1); \
+ strcpy (tmp_source, (source)); \
+ strip_trailing_slashes (tmp_source); \
+ source_base = basename (tmp_source); \
+ \
+ (new_dest) = (char *) alloca (strlen ((dest)) + 1 \
+ + strlen (source_base) + 1); \
+ stpcpy (stpcpy (stpcpy ((new_dest), (dest)), "/"), source_base);\
+ } \
+ while (0)
+
char *basename ();
enum backup_type get_version ();
int isdir ();
int yesno ();
void error ();
void strip_trailing_slashes ();
+char *stpcpy ();
static void usage ();
static int do_link ();
@@ -174,7 +195,30 @@ main (argc, argv)
errors = do_link (argv[optind], ".");
else if (optind == argc - 2)
{
- errors = do_link (argv[optind], argv[optind + 1]);
+ struct stat source_stats;
+ char *source;
+ char *dest;
+ char *new_dest;
+
+ source = argv[optind];
+ dest = argv[optind + 1];
+
+ /* When the destination is specified with a trailing slash and the
+ source exists but is not a directory, convert the user's command
+ `ln source dest/' to `ln source dest/basename(source)'. */
+
+ if (dest[strlen (dest) - 1] == '/'
+ && lstat (source, &source_stats) == 0
+ && !S_ISDIR (source_stats.st_mode))
+ {
+ PATH_BASENAME_CONCAT (new_dest, dest, source);
+ }
+ else
+ {
+ new_dest = dest;
+ }
+
+ errors = do_link (source, new_dest);
}
else
{
@@ -215,17 +259,7 @@ do_link (source, dest)
{
/* Target is a directory; build the full filename. */
char *new_dest;
- char *source_base;
- char *tmp_source;
-
- tmp_source = (char *) alloca (strlen (source) + 1);
- strcpy (tmp_source, source);
- strip_trailing_slashes (tmp_source);
-
- source_base = basename (tmp_source);
- new_dest = (char *)
- alloca (strlen (source_base) + 1 + strlen (dest) + 1);
- sprintf (new_dest, "%s/%s", dest, source_base);
+ PATH_BASENAME_CONCAT (new_dest, dest, source);
dest = new_dest;
}
diff --git a/src/ls.c b/src/ls.c
index 1b562be8c..b20b21177 100644
--- a/src/ls.c
+++ b/src/ls.c
@@ -1218,6 +1218,8 @@ sort_files ()
case time_atime:
func = sort_reverse ? rev_cmp_atime : compare_atime;
break;
+ default:
+ abort ();
}
break;
case sort_name:
@@ -1229,6 +1231,8 @@ sort_files ()
case sort_size:
func = sort_reverse ? rev_cmp_size : compare_size;
break;
+ default:
+ abort ();
}
qsort (files, files_index, sizeof (struct file), func);
@@ -1435,11 +1439,12 @@ print_long_format (f)
}
if (print_inode)
- printf ("%6u ", f->stat.st_ino);
+ printf ("%6lu ", (unsigned long) f->stat.st_ino);
if (print_block_size)
printf ("%*u ", block_size_size,
- convert_blocks (ST_NBLOCKS (f->stat), kilobyte_blocks));
+ (unsigned) convert_blocks (ST_NBLOCKS (f->stat),
+ kilobyte_blocks));
/* The space between the mode and the number of links is the POSIX
"optional alternate access method flag". */
@@ -1560,11 +1565,12 @@ print_file_name_and_frills (f)
struct file *f;
{
if (print_inode)
- printf ("%6u ", f->stat.st_ino);
+ printf ("%6lu ", (unsigned long) f->stat.st_ino);
if (print_block_size)
printf ("%*u ", block_size_size,
- convert_blocks (ST_NBLOCKS (f->stat), kilobyte_blocks));
+ (unsigned) convert_blocks (ST_NBLOCKS (f->stat),
+ kilobyte_blocks));
print_name_with_quoting (f->name);