aboutsummaryrefslogtreecommitdiff
path: root/archival/libunarchive/data_extract_all.c
diff options
context:
space:
mode:
Diffstat (limited to 'archival/libunarchive/data_extract_all.c')
-rw-r--r--archival/libunarchive/data_extract_all.c32
1 files changed, 25 insertions, 7 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