diff options
author | Denis Vlasenko <vda.linux@googlemail.com> | 2007-03-03 20:06:59 +0000 |
---|---|---|
committer | Denis Vlasenko <vda.linux@googlemail.com> | 2007-03-03 20:06:59 +0000 |
commit | dcbd51dd2874164b693fc7133a07714c8ea14234 (patch) | |
tree | 74675981e9ef16a40b0f258c0aa173b164393682 | |
parent | 19c238bc909d1e06a6bc70fe5f74124b64a4fa9d (diff) | |
download | busybox-w32-dcbd51dd2874164b693fc7133a07714c8ea14234.tar.gz busybox-w32-dcbd51dd2874164b693fc7133a07714c8ea14234.tar.bz2 busybox-w32-dcbd51dd2874164b693fc7133a07714c8ea14234.zip |
tar: handle tarfiles with (broken) checksums a-la Sun.
-rw-r--r-- | archival/libunarchive/get_header_tar.c | 19 | ||||
-rw-r--r-- | archival/tar.c | 3 |
2 files changed, 16 insertions, 6 deletions
diff --git a/archival/libunarchive/get_header_tar.c b/archival/libunarchive/get_header_tar.c index 622f09fa5..a9659181e 100644 --- a/archival/libunarchive/get_header_tar.c +++ b/archival/libunarchive/get_header_tar.c | |||
@@ -69,7 +69,7 @@ char get_header_tar(archive_handle_t *archive_handle) | |||
69 | char padding[12]; /* 500-512 */ | 69 | char padding[12]; /* 500-512 */ |
70 | } tar; | 70 | } tar; |
71 | char *cp; | 71 | char *cp; |
72 | int sum, i; | 72 | int i, sum_u, sum_s, sum; |
73 | int parse_names; | 73 | int parse_names; |
74 | 74 | ||
75 | if (sizeof(tar) != 512) | 75 | if (sizeof(tar) != 512) |
@@ -110,16 +110,23 @@ char get_header_tar(archive_handle_t *archive_handle) | |||
110 | #endif | 110 | #endif |
111 | bb_error_msg_and_die("invalid tar magic"); | 111 | bb_error_msg_and_die("invalid tar magic"); |
112 | } | 112 | } |
113 | /* Do checksum on headers */ | 113 | |
114 | sum = ' ' * sizeof(tar.chksum); | 114 | /* Do checksum on headers. |
115 | * POSIX says that checksum is done on unsigned bytes, but | ||
116 | * Sun and HP-UX fucked it up... more details in | ||
117 | * GNU tar source. */ | ||
118 | sum_s = sum_u = ' ' * sizeof(tar.chksum); | ||
115 | for (i = 0; i < 148 ; i++) { | 119 | for (i = 0; i < 148 ; i++) { |
116 | sum += ((char*)&tar)[i]; | 120 | sum_u += ((unsigned char*)&tar)[i]; |
121 | sum_s += ((signed char*)&tar)[i]; | ||
117 | } | 122 | } |
118 | for (i = 156; i < 512 ; i++) { | 123 | for (i = 156; i < 512 ; i++) { |
119 | sum += ((char*)&tar)[i]; | 124 | sum_u += ((unsigned char*)&tar)[i]; |
125 | sum_s += ((signed char*)&tar)[i]; | ||
120 | } | 126 | } |
121 | /* This field does not need special treatment (getOctal) */ | 127 | /* This field does not need special treatment (getOctal) */ |
122 | if (sum != xstrtoul(tar.chksum, 8)) { | 128 | sum = xstrtoul(tar.chksum, 8); |
129 | if (sum_u != sum && sum_s != sum) { | ||
123 | bb_error_msg_and_die("invalid tar header checksum"); | 130 | bb_error_msg_and_die("invalid tar header checksum"); |
124 | } | 131 | } |
125 | 132 | ||
diff --git a/archival/tar.c b/archival/tar.c index 1cdae296d..992128ef4 100644 --- a/archival/tar.c +++ b/archival/tar.c | |||
@@ -173,6 +173,9 @@ static void putOctal(char *cp, int len, off_t value) | |||
173 | 173 | ||
174 | static void chksum_and_xwrite(int fd, struct TarHeader* hp) | 174 | static void chksum_and_xwrite(int fd, struct TarHeader* hp) |
175 | { | 175 | { |
176 | /* POSIX says that checksum is done on unsigned bytes | ||
177 | * (Sun and HP-UX fucked it up... more details in | ||
178 | * GNU tar source) */ | ||
176 | const unsigned char *cp; | 179 | const unsigned char *cp; |
177 | int chksum, size; | 180 | int chksum, size; |
178 | 181 | ||