aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDenys Vlasenko <vda.linux@googlemail.com>2010-04-09 10:52:52 +0200
committerDenys Vlasenko <vda.linux@googlemail.com>2010-04-09 10:52:52 +0200
commit02365a6ef73defb8689d3ed5228125d72993dec9 (patch)
tree26358e718c5232ee44ad750f1b386f0f4138512a
parent823b636cd14d337ebb8766c5c181737fb3860b42 (diff)
downloadbusybox-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.c32
-rw-r--r--archival/libunarchive/get_header_tar.c5
-rwxr-xr-xtestsuite/tar.tests52
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
7mkdir tempdir && cd tempdir || exit 1 7rm -rf tar.tempdir 2>/dev/null
8mkdir 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
11testing "tar hardlinks and repeated files" "\ 12testing "tar hardlinks and repeated files" '\
12rm -rf input_* test.tar 2>/dev/null 13rm -rf input_* test.tar 2>/dev/null
13>input_hard1 14>input_hard1
14ln input_hard1 input_hard2 15ln input_hard1 input_hard2
15mkdir input_dir 16mkdir input_dir
16>input_dir/file 17>input_dir/file
18chmod -R 644 *
19chmod 755 input_dir
17tar cf test.tar input input_dir/ input_hard1 input_hard2 input_hard1 input_dir/ input 20tar cf test.tar input input_dir/ input_hard1 input_hard2 input_hard1 input_dir/ input
18tar tvf test.tar | sed 's/.*[0-9] input/input/' 21tar tvf test.tar | sed "s/.*[0-9] input/input/"
19tar xf test.tar 2>&1 && echo Ok 22tar xf test.tar 2>&1
20" "\ 23echo Ok: $?
24ls -l . input_dir/* | grep input_ | sed "s/\\(^[^ ]*\\) .* input/\\1 input/"
25' "\
21input 26input
22input_dir/ 27input_dir/
23input_dir/file 28input_dir/file
@@ -27,7 +32,40 @@ input_hard1 -> input_hard1
27input_dir/ 32input_dir/
28input_dir/file 33input_dir/file
29input 34input
30Ok 35Ok: 0
36-rw-r--r-- input_dir/file
37drwxr-xr-x input_dir
38-rw-r--r-- input_hard1
39-rw-r--r-- input_hard2
40" \
41"" ""
42
43testing "tar hardlinks mode" '\
44rm -rf input_* test.tar 2>/dev/null
45>input_hard1
46chmod 741 input_hard1
47ln input_hard1 input_hard2
48mkdir input_dir
49chmod 550 input_dir
50ln input_hard1 input_dir
51ln input_hard2 input_dir
52tar cf test.tar input_*
53tar tvf test.tar | sed "s/.*[0-9] input/input/"
54tar xf test.tar 2>&1
55echo Ok: $?
56ls -l . input_dir/* | grep input_ | sed "s/\\(^[^ ]*\\) .* input/\\1 input/"
57' "\
58input_dir/
59input_dir/input_hard1
60input_dir/input_hard2 -> input_dir/input_hard1
61input_hard1 -> input_dir/input_hard1
62input_hard2 -> input_dir/input_hard1
63Ok: 0
64-rwxr----x input_dir/input_hard1
65-rwxr----x input_dir/input_hard2
66dr-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" ""
47SKIP= 85SKIP=
48 86
49cd .. && rm -rf tempdir || exit 1 87cd .. && rm -rf tar.tempdir || exit 1
50 88
51exit $FAILCOUNT 89exit $FAILCOUNT