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 | ||
