diff options
author | Denys Vlasenko <vda.linux@googlemail.com> | 2010-04-09 10:52:52 +0200 |
---|---|---|
committer | Denys Vlasenko <vda.linux@googlemail.com> | 2010-04-09 10:52:52 +0200 |
commit | 02365a6ef73defb8689d3ed5228125d72993dec9 (patch) | |
tree | 26358e718c5232ee44ad750f1b386f0f4138512a | |
parent | 823b636cd14d337ebb8766c5c181737fb3860b42 (diff) | |
download | busybox-w32-02365a6ef73defb8689d3ed5228125d72993dec9.tar.gz busybox-w32-02365a6ef73defb8689d3ed5228125d72993dec9.tar.bz2 busybox-w32-02365a6ef73defb8689d3ed5228125d72993dec9.zip |
tar: fix mishandling of repeated hardlink in tarball; expand tests
function old new delta
data_extract_all 727 767 +40
get_header_tar 1576 1572 -4
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
-rw-r--r-- | archival/libunarchive/data_extract_all.c | 32 | ||||
-rw-r--r-- | archival/libunarchive/get_header_tar.c | 5 | ||||
-rwxr-xr-x | testsuite/tar.tests | 52 |
3 files changed, 73 insertions, 16 deletions
diff --git a/archival/libunarchive/data_extract_all.c b/archival/libunarchive/data_extract_all.c index cc4894229..de2367ac2 100644 --- a/archival/libunarchive/data_extract_all.c +++ b/archival/libunarchive/data_extract_all.c | |||
@@ -34,12 +34,30 @@ void FAST_FUNC data_extract_all(archive_handle_t *archive_handle) | |||
34 | 34 | ||
35 | if (archive_handle->ah_flags & ARCHIVE_UNLINK_OLD) { | 35 | if (archive_handle->ah_flags & ARCHIVE_UNLINK_OLD) { |
36 | /* Remove the entry if it exists */ | 36 | /* Remove the entry if it exists */ |
37 | if ((!S_ISDIR(file_header->mode)) | 37 | if (!S_ISDIR(file_header->mode)) { |
38 | && (unlink(file_header->name) == -1) | 38 | /* Is it hardlink? |
39 | && (errno != ENOENT) | 39 | * We encode hard links as regular files of size 0 with a symlink */ |
40 | ) { | 40 | if (S_ISREG(file_header->mode) |
41 | bb_perror_msg_and_die("can't remove old file %s", | 41 | && file_header->link_target |
42 | file_header->name); | 42 | && file_header->size == 0 |
43 | ) { | ||
44 | /* Ugly special case: | ||
45 | * tar cf t.tar hardlink1 hardlink2 hardlink1 | ||
46 | * results in this tarball structure: | ||
47 | * hardlink1 | ||
48 | * hardlink2 -> hardlink1 | ||
49 | * hardlink1 -> hardlink1 <== !!! | ||
50 | */ | ||
51 | if (strcmp(file_header->link_target, file_header->name) == 0) | ||
52 | goto ret; | ||
53 | } | ||
54 | /* Proceed with deleting */ | ||
55 | if (unlink(file_header->name) == -1 | ||
56 | && errno != ENOENT | ||
57 | ) { | ||
58 | bb_perror_msg_and_die("can't remove old file %s", | ||
59 | file_header->name); | ||
60 | } | ||
43 | } | 61 | } |
44 | } | 62 | } |
45 | else if (archive_handle->ah_flags & ARCHIVE_EXTRACT_NEWER) { | 63 | else if (archive_handle->ah_flags & ARCHIVE_EXTRACT_NEWER) { |
@@ -65,7 +83,7 @@ void FAST_FUNC data_extract_all(archive_handle_t *archive_handle) | |||
65 | } | 83 | } |
66 | 84 | ||
67 | /* Handle hard links separately | 85 | /* Handle hard links separately |
68 | * We identified hard links as regular files of size 0 with a symlink */ | 86 | * We encode hard links as regular files of size 0 with a symlink */ |
69 | if (S_ISREG(file_header->mode) | 87 | if (S_ISREG(file_header->mode) |
70 | && file_header->link_target | 88 | && file_header->link_target |
71 | && file_header->size == 0 | 89 | && file_header->size == 0 |
diff --git a/archival/libunarchive/get_header_tar.c b/archival/libunarchive/get_header_tar.c index cf0b9ab02..adb4c157b 100644 --- a/archival/libunarchive/get_header_tar.c +++ b/archival/libunarchive/get_header_tar.c | |||
@@ -507,8 +507,9 @@ char FAST_FUNC get_header_tar(archive_handle_t *archive_handle) | |||
507 | archive_handle->action_header(/*archive_handle->*/ file_header); | 507 | archive_handle->action_header(/*archive_handle->*/ file_header); |
508 | /* Note that we kill the '/' only after action_header() */ | 508 | /* Note that we kill the '/' only after action_header() */ |
509 | /* (like GNU tar 1.15.1: verbose mode outputs "dir/dir/") */ | 509 | /* (like GNU tar 1.15.1: verbose mode outputs "dir/dir/") */ |
510 | if (cp) *cp = '\0'; | 510 | if (cp) |
511 | archive_handle->ah_flags |= ARCHIVE_EXTRACT_QUIET; | 511 | *cp = '\0'; |
512 | //archive_handle->ah_flags |= ARCHIVE_EXTRACT_QUIET; // why?? | ||
512 | archive_handle->action_data(archive_handle); | 513 | archive_handle->action_data(archive_handle); |
513 | llist_add_to(&(archive_handle->passed), file_header->name); | 514 | llist_add_to(&(archive_handle->passed), file_header->name); |
514 | } else { | 515 | } else { |
diff --git a/testsuite/tar.tests b/testsuite/tar.tests index 71095cb20..dd8f11062 100755 --- a/testsuite/tar.tests +++ b/testsuite/tar.tests | |||
@@ -4,20 +4,25 @@ | |||
4 | 4 | ||
5 | . ./testing.sh | 5 | . ./testing.sh |
6 | 6 | ||
7 | mkdir tempdir && cd tempdir || exit 1 | 7 | rm -rf tar.tempdir 2>/dev/null |
8 | mkdir tar.tempdir && cd tar.tempdir || exit 1 | ||
8 | 9 | ||
9 | # testing "test name" "script" "expected result" "file input" "stdin" | 10 | # testing "test name" "script" "expected result" "file input" "stdin" |
10 | 11 | ||
11 | testing "tar hardlinks and repeated files" "\ | 12 | testing "tar hardlinks and repeated files" '\ |
12 | rm -rf input_* test.tar 2>/dev/null | 13 | rm -rf input_* test.tar 2>/dev/null |
13 | >input_hard1 | 14 | >input_hard1 |
14 | ln input_hard1 input_hard2 | 15 | ln input_hard1 input_hard2 |
15 | mkdir input_dir | 16 | mkdir input_dir |
16 | >input_dir/file | 17 | >input_dir/file |
18 | chmod -R 644 * | ||
19 | chmod 755 input_dir | ||
17 | tar cf test.tar input input_dir/ input_hard1 input_hard2 input_hard1 input_dir/ input | 20 | tar cf test.tar input input_dir/ input_hard1 input_hard2 input_hard1 input_dir/ input |
18 | tar tvf test.tar | sed 's/.*[0-9] input/input/' | 21 | tar tvf test.tar | sed "s/.*[0-9] input/input/" |
19 | tar xf test.tar 2>&1 && echo Ok | 22 | tar xf test.tar 2>&1 |
20 | " "\ | 23 | echo Ok: $? |
24 | ls -l . input_dir/* | grep input_ | sed "s/\\(^[^ ]*\\) .* input/\\1 input/" | ||
25 | ' "\ | ||
21 | input | 26 | input |
22 | input_dir/ | 27 | input_dir/ |
23 | input_dir/file | 28 | input_dir/file |
@@ -27,7 +32,40 @@ input_hard1 -> input_hard1 | |||
27 | input_dir/ | 32 | input_dir/ |
28 | input_dir/file | 33 | input_dir/file |
29 | input | 34 | input |
30 | Ok | 35 | Ok: 0 |
36 | -rw-r--r-- input_dir/file | ||
37 | drwxr-xr-x input_dir | ||
38 | -rw-r--r-- input_hard1 | ||
39 | -rw-r--r-- input_hard2 | ||
40 | " \ | ||
41 | "" "" | ||
42 | |||
43 | testing "tar hardlinks mode" '\ | ||
44 | rm -rf input_* test.tar 2>/dev/null | ||
45 | >input_hard1 | ||
46 | chmod 741 input_hard1 | ||
47 | ln input_hard1 input_hard2 | ||
48 | mkdir input_dir | ||
49 | chmod 550 input_dir | ||
50 | ln input_hard1 input_dir | ||
51 | ln input_hard2 input_dir | ||
52 | tar cf test.tar input_* | ||
53 | tar tvf test.tar | sed "s/.*[0-9] input/input/" | ||
54 | tar xf test.tar 2>&1 | ||
55 | echo Ok: $? | ||
56 | ls -l . input_dir/* | grep input_ | sed "s/\\(^[^ ]*\\) .* input/\\1 input/" | ||
57 | ' "\ | ||
58 | input_dir/ | ||
59 | input_dir/input_hard1 | ||
60 | input_dir/input_hard2 -> input_dir/input_hard1 | ||
61 | input_hard1 -> input_dir/input_hard1 | ||
62 | input_hard2 -> input_dir/input_hard1 | ||
63 | Ok: 0 | ||
64 | -rwxr----x input_dir/input_hard1 | ||
65 | -rwxr----x input_dir/input_hard2 | ||
66 | dr-xr-x--- input_dir | ||
67 | -rwxr----x input_hard1 | ||
68 | -rwxr----x input_hard2 | ||
31 | " \ | 69 | " \ |
32 | "" "" | 70 | "" "" |
33 | 71 | ||
@@ -46,6 +84,6 @@ Ok | |||
46 | "Ok\n" "" | 84 | "Ok\n" "" |
47 | SKIP= | 85 | SKIP= |
48 | 86 | ||
49 | cd .. && rm -rf tempdir || exit 1 | 87 | cd .. && rm -rf tar.tempdir || exit 1 |
50 | 88 | ||
51 | exit $FAILCOUNT | 89 | exit $FAILCOUNT |