aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDenis Vlasenko <vda.linux@googlemail.com>2007-03-03 20:06:59 +0000
committerDenis Vlasenko <vda.linux@googlemail.com>2007-03-03 20:06:59 +0000
commitdcbd51dd2874164b693fc7133a07714c8ea14234 (patch)
tree74675981e9ef16a40b0f258c0aa173b164393682
parent19c238bc909d1e06a6bc70fe5f74124b64a4fa9d (diff)
downloadbusybox-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.c19
-rw-r--r--archival/tar.c3
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
174static void chksum_and_xwrite(int fd, struct TarHeader* hp) 174static 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