Revamp tar_getcwd/normalize_filename stuff.
The changes are based on the discussion with Nathan. * src/common.h (normalize_filename): Take two arguments. All callers updated. (tar_getcwd): Replaced with .. (tar_getcdpath): New proto. * src/misc.c (normalize_filename): Take two arguments. (chdir_arg): Populate cwd along with creating the structure. (tar_getcwd): Removed. (tar_getcdpath): New function. * tests/incr09.at: New test case. * tests/Makefile.am: Add new tests. * tests/testsuite.at: Likewise.
This commit is contained in:
@@ -596,7 +596,7 @@ void skip_member (void);
|
|||||||
void assign_string (char **dest, const char *src);
|
void assign_string (char **dest, const char *src);
|
||||||
int unquote_string (char *str);
|
int unquote_string (char *str);
|
||||||
char *zap_slashes (char *name);
|
char *zap_slashes (char *name);
|
||||||
char *normalize_filename (const char *name);
|
char *normalize_filename (int cdidx, const char *name);
|
||||||
void normalize_filename_x (char *name);
|
void normalize_filename_x (char *name);
|
||||||
void replace_prefix (char **pname, const char *samp, size_t slen,
|
void replace_prefix (char **pname, const char *samp, size_t slen,
|
||||||
const char *repl, size_t rlen);
|
const char *repl, size_t rlen);
|
||||||
@@ -609,7 +609,7 @@ char *namebuf_name (namebuf_t buf, const char *name);
|
|||||||
void namebuf_add_dir (namebuf_t buf, const char *name);
|
void namebuf_add_dir (namebuf_t buf, const char *name);
|
||||||
char *namebuf_finish (namebuf_t buf);
|
char *namebuf_finish (namebuf_t buf);
|
||||||
|
|
||||||
const char *tar_getcwd (void);
|
const char *tar_getcdpath (int);
|
||||||
const char *tar_dirname (void);
|
const char *tar_dirname (void);
|
||||||
|
|
||||||
/* Represent N using a signed integer I such that (uintmax_t) I == N.
|
/* Represent N using a signed integer I such that (uintmax_t) I == N.
|
||||||
|
|||||||
@@ -280,7 +280,7 @@ free_directory (struct directory *dir)
|
|||||||
static struct directory *
|
static struct directory *
|
||||||
attach_directory (const char *name)
|
attach_directory (const char *name)
|
||||||
{
|
{
|
||||||
char *cname = normalize_filename (name);
|
char *cname = normalize_filename (chdir_current, name);
|
||||||
struct directory *dir = make_directory (name, cname);
|
struct directory *dir = make_directory (name, cname);
|
||||||
if (dirtail)
|
if (dirtail)
|
||||||
dirtail->next = dir;
|
dirtail->next = dir;
|
||||||
@@ -370,7 +370,7 @@ find_directory (const char *name)
|
|||||||
return 0;
|
return 0;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
char *caname = normalize_filename (name);
|
char *caname = normalize_filename (chdir_current, name);
|
||||||
struct directory *dir = make_directory (name, caname);
|
struct directory *dir = make_directory (name, caname);
|
||||||
struct directory *ret = hash_lookup (directory_table, dir);
|
struct directory *ret = hash_lookup (directory_table, dir);
|
||||||
free_directory (dir);
|
free_directory (dir);
|
||||||
|
|||||||
48
src/misc.c
48
src/misc.c
@@ -273,7 +273,7 @@ normalize_filename_x (char *file_name)
|
|||||||
Return a normalized newly-allocated copy. */
|
Return a normalized newly-allocated copy. */
|
||||||
|
|
||||||
char *
|
char *
|
||||||
normalize_filename (const char *name)
|
normalize_filename (int cdidx, const char *name)
|
||||||
{
|
{
|
||||||
char *copy = NULL;
|
char *copy = NULL;
|
||||||
|
|
||||||
@@ -285,7 +285,7 @@ normalize_filename (const char *name)
|
|||||||
getcwd is slow, it might fail, and it does not necessarily
|
getcwd is slow, it might fail, and it does not necessarily
|
||||||
return a canonical name even when it succeeds. Perhaps we
|
return a canonical name even when it succeeds. Perhaps we
|
||||||
can use dev+ino pairs instead of names? */
|
can use dev+ino pairs instead of names? */
|
||||||
const char *cwd = tar_getcwd ();
|
const char *cwd = tar_getcdpath (cdidx);
|
||||||
size_t copylen;
|
size_t copylen;
|
||||||
bool need_separator;
|
bool need_separator;
|
||||||
|
|
||||||
@@ -888,7 +888,7 @@ chdir_arg (char const *dir)
|
|||||||
if (! wd_count)
|
if (! wd_count)
|
||||||
{
|
{
|
||||||
wd[wd_count].name = ".";
|
wd[wd_count].name = ".";
|
||||||
wd[wd_count].cwd = NULL;
|
wd[wd_count].cwd = xgetcwd ();
|
||||||
wd[wd_count].fd = AT_FDCWD;
|
wd[wd_count].fd = AT_FDCWD;
|
||||||
wd_count++;
|
wd_count++;
|
||||||
}
|
}
|
||||||
@@ -906,7 +906,14 @@ chdir_arg (char const *dir)
|
|||||||
}
|
}
|
||||||
|
|
||||||
wd[wd_count].name = dir;
|
wd[wd_count].name = dir;
|
||||||
wd[wd_count].cwd = NULL;
|
if (IS_ABSOLUTE_FILE_NAME (wd[wd_count].name))
|
||||||
|
wd[wd_count].cwd = xstrdup (wd[wd_count].name);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
namebuf_t nbuf = namebuf_create (wd[wd_count - 1].cwd);
|
||||||
|
namebuf_add_dir (nbuf, wd[wd_count].name);
|
||||||
|
wd[wd_count].cwd = namebuf_finish (nbuf);
|
||||||
|
}
|
||||||
wd[wd_count].fd = 0;
|
wd[wd_count].fd = 0;
|
||||||
return wd_count++;
|
return wd_count++;
|
||||||
}
|
}
|
||||||
@@ -987,37 +994,16 @@ tar_dirname (void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
const char *
|
const char *
|
||||||
tar_getcwd (void)
|
tar_getcdpath (int idx)
|
||||||
{
|
{
|
||||||
static char *cwd;
|
|
||||||
namebuf_t nbuf;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
if (!cwd)
|
|
||||||
cwd = xgetcwd ();
|
|
||||||
if (!wd)
|
if (!wd)
|
||||||
return cwd;
|
|
||||||
|
|
||||||
if (0 == chdir_current || !wd[chdir_current].cwd)
|
|
||||||
{
|
{
|
||||||
if (IS_ABSOLUTE_FILE_NAME (wd[chdir_current].name))
|
static char *cwd;
|
||||||
{
|
if (!cwd)
|
||||||
wd[chdir_current].cwd = xstrdup (wd[chdir_current].name);
|
cwd = xgetcwd ();
|
||||||
return wd[chdir_current].cwd;
|
return cwd;
|
||||||
}
|
|
||||||
if (!wd[0].cwd)
|
|
||||||
wd[0].cwd = cwd;
|
|
||||||
|
|
||||||
for (i = chdir_current - 1; i > 0; i--)
|
|
||||||
if (wd[i].cwd)
|
|
||||||
break;
|
|
||||||
|
|
||||||
nbuf = namebuf_create (wd[i].cwd);
|
|
||||||
for (i++; i <= chdir_current; i++)
|
|
||||||
namebuf_add_dir (nbuf, wd[i].name);
|
|
||||||
wd[chdir_current].cwd = namebuf_finish (nbuf);
|
|
||||||
}
|
}
|
||||||
return wd[chdir_current].cwd;
|
return wd[idx].cwd;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|||||||
@@ -1238,13 +1238,11 @@ collect_and_sort_names (void)
|
|||||||
namelist = merge_sort (namelist, num_names, compare_names);
|
namelist = merge_sort (namelist, num_names, compare_names);
|
||||||
|
|
||||||
num_names = 0;
|
num_names = 0;
|
||||||
nametab = hash_initialize (0, 0,
|
nametab = hash_initialize (0, 0, name_hash, name_compare, NULL);
|
||||||
name_hash,
|
|
||||||
name_compare, NULL);
|
|
||||||
for (name = namelist; name; name = next_name)
|
for (name = namelist; name; name = next_name)
|
||||||
{
|
{
|
||||||
next_name = name->next;
|
next_name = name->next;
|
||||||
name->caname = normalize_filename (name->name);
|
name->caname = normalize_filename (name->change_dir, name->name);
|
||||||
if (prev_name)
|
if (prev_name)
|
||||||
{
|
{
|
||||||
struct name *p = hash_lookup (nametab, name);
|
struct name *p = hash_lookup (nametab, name);
|
||||||
|
|||||||
@@ -113,6 +113,7 @@ TESTSUITE_AT = \
|
|||||||
incr06.at\
|
incr06.at\
|
||||||
incr07.at\
|
incr07.at\
|
||||||
incr08.at\
|
incr08.at\
|
||||||
|
incr09.at\
|
||||||
indexfile.at\
|
indexfile.at\
|
||||||
ignfail.at\
|
ignfail.at\
|
||||||
label01.at\
|
label01.at\
|
||||||
|
|||||||
59
tests/incr09.at
Normal file
59
tests/incr09.at
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
# Process this file with autom4te to create testsuite. -*- Autotest -*-
|
||||||
|
# Test suite for GNU tar.
|
||||||
|
# Copyright 2013 Free Software Foundation, Inc.
|
||||||
|
#
|
||||||
|
# GNU tar is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# GNU tar is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
AT_SETUP([incremental with alternating -C])
|
||||||
|
AT_KEYWORDS([incremental create incr09])
|
||||||
|
|
||||||
|
AT_TAR_CHECK([
|
||||||
|
AT_SORT_PREREQ
|
||||||
|
mkdir foo bar middle
|
||||||
|
echo foo/foo_file > foo/foo_file
|
||||||
|
echo bar/bar_file > bar/bar_file
|
||||||
|
echo middle/file > middle/middle_file
|
||||||
|
decho A
|
||||||
|
tar -cvf foo.tar --incremental -C foo . -C `pwd` middle -C bar .
|
||||||
|
|
||||||
|
rm foo.tar
|
||||||
|
>toplevel_file
|
||||||
|
decho B
|
||||||
|
tar -cvf foo.tar --incremental -C foo . -C `pwd` toplevel_file -C bar .
|
||||||
|
],
|
||||||
|
[0],
|
||||||
|
[A
|
||||||
|
./
|
||||||
|
./
|
||||||
|
middle/
|
||||||
|
./bar_file
|
||||||
|
./foo_file
|
||||||
|
middle/middle_file
|
||||||
|
B
|
||||||
|
./
|
||||||
|
./
|
||||||
|
toplevel_file
|
||||||
|
./bar_file
|
||||||
|
./foo_file
|
||||||
|
],
|
||||||
|
[A
|
||||||
|
tar: .: Directory is new
|
||||||
|
tar: middle: Directory is new
|
||||||
|
tar: .: Directory is new
|
||||||
|
B
|
||||||
|
tar: .: Directory is new
|
||||||
|
tar: .: Directory is new
|
||||||
|
],[],[],[gnu])
|
||||||
|
|
||||||
|
AT_CLEANUP
|
||||||
@@ -298,6 +298,7 @@ m4_include([incr05.at])
|
|||||||
m4_include([incr06.at])
|
m4_include([incr06.at])
|
||||||
m4_include([incr07.at])
|
m4_include([incr07.at])
|
||||||
m4_include([incr08.at])
|
m4_include([incr08.at])
|
||||||
|
m4_include([incr09.at])
|
||||||
|
|
||||||
AT_BANNER([Files removed while archiving])
|
AT_BANNER([Files removed while archiving])
|
||||||
m4_include([filerem01.at])
|
m4_include([filerem01.at])
|
||||||
|
|||||||
Reference in New Issue
Block a user