blob: 3dc8e58a64de9d68676258ea3bb899a8960b20dc [file] [log] [blame]
/*
zipcmp.c -- compare zip files
Copyright (C) 2003-2022 Dieter Baron and Thomas Klausner
This file is part of libzip, a library to manipulate ZIP archives.
The authors can be contacted at <libzip@nih.at>
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in
the documentation and/or other materials provided with the
distribution.
3. The names of the authors may not be used to endorse or promote
products derived from this software without specific prior
written permission.
THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS
OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#ifdef HAVE_STRINGS_H
#include <strings.h>
#endif
#ifdef HAVE_FTS_H
#include <fts.h>
#endif
#include <zlib.h>
#ifndef HAVE_GETOPT
#include "getopt.h"
#endif
#include "zip.h"
#include "compat.h"
#include "diff_output.h"
struct archive {
const char *name;
zip_t *za;
zip_uint64_t nentry;
struct entry *entry;
const char *comment;
size_t comment_length;
};
struct ef {
const char *name;
zip_uint16_t flags;
zip_uint16_t id;
zip_uint16_t size;
const zip_uint8_t *data;
};
struct entry {
char *name;
zip_uint64_t size;
zip_uint32_t crc;
zip_uint32_t comp_method;
struct ef *extra_fields;
zip_uint16_t n_extra_fields;
const char *comment;
zip_uint32_t comment_length;
};
typedef struct {
uint32_t value;
const char * const name;
} enum_map_t;
const enum_map_t comp_methods[] = {
{ 0, "Stored (no compression)" },
{ 1, "Shrunk" },
{ 2, "Reduced with compression factor 1" },
{ 3, "Reduced with compression factor 2" },
{ 4, "Reduced with compression factor 3" },
{ 5, "Reduced with compression factor 4" },
{ 6, "Imploded" },
{ 7, "Reserved for Tokenizing compression algorithm" },
{ 8, "Deflated" },
{ 9, "Enhanced Deflating using Deflate64(tm)" },
{ 10, "PKWARE Data Compression Library Imploding (old IBM TERSE)" },
{ 11, "11 (Reserved by PKWARE)" },
{ 12, "BZIP2" },
{ 13, "13 (Reserved by PKWARE)" },
{ 14, "LZMA (EFS)" },
{ 15, "15 (Reserved by PKWARE)" },
{ 16, "16 (Reserved by PKWARE)" },
{ 17, "17 (Reserved by PKWARE)" },
{ 18, "IBM TERSE (new)" },
{ 19, "IBM LZ77 z Architecture (PFS)" },
{ 20, "Zstandard compressed data (obsolete)" },
{ 93, "Zstandard compressed data" },
{ 95, "XZ compressed data" },
{ 97, "WavPack compressed data" },
{ 98, "PPMd version I, Rev 1" },
{ 99, "WinZIP AES Encryption" },
{ UINT32_MAX, NULL }
};
const enum_map_t extra_fields[] = {
/* PKWARE defined */
{ 0x0001, "Zip64 extended information" },
{ 0x0007, "AV Info" },
{ 0x0008, "Reserved for extended language encoding data (PFS)" },
{ 0x0009, "OS/2" },
{ 0x000a, "NTFS" },
{ 0x000c, "OpenVMS" },
{ 0x000d, "UNIX" },
{ 0x000e, "Reserved for file stream and fork descriptors" },
{ 0x000f, "Patch Descriptor" },
{ 0x0014, "PKCS#7 Store for X.509 Certificates" },
{ 0x0015, "X.509 Certificate ID and Signature for individual file" },
{ 0x0016, "X.509 Certificate ID for Central Directory" },
{ 0x0017, "Strong Encryption Header" },
{ 0x0018, "Record Management Controls" },
{ 0x0019, "PKCS#7 Encryption Recipient Certificate List" },
{ 0x0065, "IBM S/390 (Z390), AS/400 (I400) attributes - uncompressed" },
{ 0x0066, "Reserved for IBM S/390 (Z390), AS/400 (I400) attributes - compressed" },
{ 0x4690, "POSZIP 4690 (reserved)" },
/* Third-Party defined; see InfoZIP unzip sources proginfo/extrafld.txt */
{ 0x07c8, "Info-ZIP Macintosh (old)" },
{ 0x2605, "ZipIt Macintosh (first version)" },
{ 0x2705, "ZipIt Macintosh 1.3.5+ (w/o full filename)" },
{ 0x2805, "ZipIt Macintosh 1.3.5+" },
{ 0x334d, "Info-ZIP Macintosh (new)" },
{ 0x4154, "Tandem NSK" },
{ 0x4341, "Acorn/SparkFS" },
{ 0x4453, "Windows NT security descriptor" },
{ 0x4704, "VM/CMS" },
{ 0x470f, "MVS" },
{ 0x4854, "Theos, old unofficial port" },
{ 0x4b46, "FWKCS MD5" },
{ 0x4c41, "OS/2 access control list (text ACL)" },
{ 0x4d49, "Info-ZIP OpenVMS (obsolete)" },
{ 0x4d63, "Macintosh SmartZIP" },
{ 0x4f4c, "Xceed original location extra field" },
{ 0x5356, "AOS/VS (ACL)" },
{ 0x5455, "extended timestamp" },
{ 0x554e, "Xceed unicode extra field" },
{ 0x5855, "Info-ZIP UNIX (original)" },
{ 0x6375, "Info-ZIP UTF-8 comment field" },
{ 0x6542, "BeOS (BeBox, PowerMac, etc.)" },
{ 0x6854, "Theos" },
{ 0x7075, "Info-ZIP UTF-8 name field" },
{ 0x7441, "AtheOS (AtheOS/Syllable attributes)" },
{ 0x756e, "ASi UNIX" },
{ 0x7855, "Info-ZIP UNIX" },
{ 0x7875, "Info-ZIP UNIX 3rd generation" },
{ 0x9901, "WinZIP AES encryption" },
{ 0xa220, "Microsoft Open Packaging Growth Hint" },
{ 0xcafe, "executable Java JAR file" },
{ 0xfb4a, "SMS/QDOS" }, /* per InfoZIP extrafld.txt */
{ 0xfd4a, "SMS/QDOS" }, /* per appnote.txt */
{ UINT32_MAX, NULL }
};
const char *progname;
#define PROGRAM "zipcmp"
#define USAGE "usage: %s [-hipqtVv] archive1 archive2\n"
char help_head[] = PROGRAM " (" PACKAGE ") by Dieter Baron and Thomas Klausner\n\n";
char help[] = "\n\
-h display this help message\n\
-C check archive consistencies\n\
-i compare names ignoring case distinctions\n\
-p compare as many details as possible\n\
-q be quiet\n\
-s print a summary\n\
-t test zip files (compare file contents to checksum)\n\
-V display version number\n\
-v be verbose (print differences, default)\n\
\n\
Report bugs to <libzip@nih.at>.\n";
char version_string[] = PROGRAM " (" PACKAGE " " VERSION ")\n\
Copyright (C) 2003-2022 Dieter Baron and Thomas Klausner\n\
" PACKAGE " comes with ABSOLUTELY NO WARRANTY, to the extent permitted by law.\n";
#define OPTIONS "hVCipqstv"
#define BOTH_ARE_ZIPS(a) (a[0].za && a[1].za)
static int comment_compare(const char *c1, size_t l1, const char *c2, size_t l2);
static int compare_list(char *const name[2], const void *list[2], const zip_uint64_t list_length[2], int element_size, int (*cmp)(const void *a, const void *b), int (*ignore)(const void *list, int last, const void *other), int (*check)(char *const name[2], const void *a, const void *b), void (*print)(char side, const void *element), void (*start_file)(const void *element));
static int compare_zip(char *const zn[]);
static int ef_compare(char *const name[2], const struct entry *e1, const struct entry *e2);
static int ef_order(const void *a, const void *b);
static void ef_print(char side, const void *p);
static int ef_read(zip_t *za, zip_uint64_t idx, struct entry *e);
static int entry_cmp(const void *p1, const void *p2);
static int entry_ignore(const void *p1, int last, const void *o);
static int entry_paranoia_checks(char *const name[2], const void *p1, const void *p2);
static void entry_print(char side, const void *p);
static void entry_start_file(const void *p);
static const char *map_enum(const enum_map_t *map, uint32_t value);
static int is_directory(const char *name);
#ifdef HAVE_FTS_H
static int list_directory(const char *name, struct archive *a);
#endif
static int list_zip(const char *name, struct archive *a);
static int test_file(zip_t *za, zip_uint64_t idx, const char *zipname, const char *filename, zip_uint64_t size, zip_uint32_t crc);
int ignore_case, test_files, paranoid, verbose, have_directory, check_consistency, summary;
int plus_count = 0, minus_count = 0;
diff_output_t output;
int
main(int argc, char *const argv[]) {
int c;
progname = argv[0];
ignore_case = 0;
test_files = 0;
check_consistency = 0;
paranoid = 0;
have_directory = 0;
verbose = 1;
summary = 0;
while ((c = getopt(argc, argv, OPTIONS)) != -1) {
switch (c) {
case 'C':
check_consistency = 1;
break;
case 'i':
ignore_case = 1;
break;
case 'p':
paranoid = 1;
break;
case 'q':
verbose = 0;
break;
case 's':
summary = 1;
break;
case 't':
test_files = 1;
break;
case 'v':
verbose = 1;
break;
case 'h':
fputs(help_head, stdout);
printf(USAGE, progname);
fputs(help, stdout);
exit(0);
case 'V':
fputs(version_string, stdout);
exit(0);
default:
fprintf(stderr, USAGE, progname);
exit(2);
}
}
if (argc != optind + 2) {
fprintf(stderr, USAGE, progname);
exit(2);
}
exit((compare_zip(argv + optind) == 0) ? 0 : 1);
}
static int
compare_zip(char *const zn[]) {
struct archive a[2];
struct entry *e[2];
zip_uint64_t n[2];
int i;
int res;
for (i = 0; i < 2; i++) {
a[i].name = zn[i];
a[i].entry = NULL;
a[i].nentry = 0;
a[i].za = NULL;
a[i].comment = NULL;
a[i].comment_length = 0;
if (is_directory(zn[i])) {
#ifndef HAVE_FTS_H
fprintf(stderr, "%s: reading directories not supported\n", progname);
exit(2);
#else
if (list_directory(zn[i], a + i) < 0)
exit(2);
have_directory = 1;
paranoid = 0; /* paranoid checks make no sense for directories, since they compare zip metadata */
#endif
}
else {
if (list_zip(zn[i], a + i) < 0)
exit(2);
}
if (a[i].nentry > 0)
qsort(a[i].entry, a[i].nentry, sizeof(a[i].entry[0]), entry_cmp);
}
diff_output_init(&output, verbose, zn);
e[0] = a[0].entry;
e[1] = a[1].entry;
n[0] = a[0].nentry;
n[1] = a[1].nentry;
res = compare_list(zn, (const void **)e, n, sizeof(e[i][0]), entry_cmp, have_directory ? entry_ignore : NULL, paranoid ? entry_paranoia_checks : NULL, entry_print, entry_start_file);
if (paranoid) {
if (comment_compare(a[0].comment, a[0].comment_length, a[1].comment, a[1].comment_length) != 0) {
if (a[0].comment_length > 0) {
diff_output_data(&output, '-', (const zip_uint8_t *)a[0].comment, a[0].comment_length, "archive comment");
minus_count++;
}
if (a[1].comment_length > 0) {
diff_output_data(&output, '+', (const zip_uint8_t *)a[1].comment, a[1].comment_length, "archive comment");
plus_count++;
}
res = 1;
}
}
for (i = 0; i < 2; i++) {
zip_uint64_t j;
if (a[i].za) {
zip_close(a[i].za);
}
for (j = 0; j < a[i].nentry; j++) {
free(a[i].entry[j].name);
}
free(a[i].entry);
}
if (summary) {
printf("%d files removed, %d files added\n", minus_count, plus_count);
}
switch (res) {
case 0:
exit(0);
case 1:
exit(1);
default:
exit(2);
}
}
#ifdef HAVE_FTS_H
static zip_int64_t
compute_crc(const char *fname) {
FILE *f;
uLong crc = crc32(0L, Z_NULL, 0);
size_t n;
Bytef buffer[8192];
if ((f = fopen(fname, "rb")) == NULL) {
fprintf(stderr, "%s: can't open %s: %s\n", progname, fname, strerror(errno));
return -1;
}
while ((n = fread(buffer, 1, sizeof(buffer), f)) > 0) {
crc = crc32(crc, buffer, (unsigned int)n);
}
if (ferror(f)) {
fprintf(stderr, "%s: read error on %s: %s\n", progname, fname, strerror(errno));
fclose(f);
return -1;
}
fclose(f);
return (zip_int64_t)crc;
}
#endif
static int
is_directory(const char *name) {
struct stat st;
if (stat(name, &st) < 0)
return 0;
return S_ISDIR(st.st_mode);
}
#ifdef HAVE_FTS_H
static int
list_directory(const char *name, struct archive *a) {
FTS *fts;
FTSENT *ent;
zip_uint64_t nalloc;
size_t prefix_length;
char *const names[2] = {(char *)name, NULL};
if ((fts = fts_open(names, FTS_NOCHDIR | FTS_LOGICAL, NULL)) == NULL) {
fprintf(stderr, "%s: can't open directory '%s': %s\n", progname, name, strerror(errno));
return -1;
}
prefix_length = strlen(name) + 1;
nalloc = 0;
while ((ent = fts_read(fts))) {
zip_int64_t crc;
switch (ent->fts_info) {
case FTS_DOT:
case FTS_DP:
case FTS_DEFAULT:
case FTS_SL:
case FTS_NSOK:
break;
case FTS_DC:
case FTS_DNR:
case FTS_ERR:
case FTS_NS:
case FTS_SLNONE:
/* TODO: error */
fts_close(fts);
return -1;
case FTS_D:
case FTS_F:
if (a->nentry >= nalloc) {
nalloc += 16;
if (nalloc > SIZE_MAX / sizeof(a->entry[0])) {
fprintf(stderr, "%s: malloc failure\n", progname);
exit(1);
}
a->entry = realloc(a->entry, sizeof(a->entry[0]) * nalloc);
if (a->entry == NULL) {
fprintf(stderr, "%s: malloc failure\n", progname);
exit(1);
}
}
if (ent->fts_info == FTS_D) {
char *dir_name;
if (ent->fts_path[prefix_length - 1] == '\0') {
break;
}
dir_name = malloc(strlen(ent->fts_path + prefix_length) + 2);
if (dir_name == NULL) {
fprintf(stderr, "%s: malloc failure\n", progname);
exit(1);
}
sprintf(dir_name, "%s/", ent->fts_path + prefix_length);
a->entry[a->nentry].name = dir_name;
a->entry[a->nentry].size = 0;
a->entry[a->nentry].crc = 0;
}
else {
a->entry[a->nentry].name = strdup(ent->fts_path + prefix_length);
a->entry[a->nentry].size = (zip_uint64_t)ent->fts_statp->st_size;
if ((crc = compute_crc(ent->fts_accpath)) < 0) {
fts_close(fts);
return -1;
}
a->entry[a->nentry].crc = (zip_uint32_t)crc;
}
a->nentry++;
break;
}
}
if (fts_close(fts)) {
fprintf(stderr, "%s: error closing directory '%s': %s\n", progname, a->name, strerror(errno));
return -1;
}
return 0;
}
#endif
static int
list_zip(const char *name, struct archive *a) {
zip_t *za;
int err;
struct zip_stat st;
unsigned int i;
if ((za = zip_open(name, check_consistency ? ZIP_CHECKCONS : 0, &err)) == NULL) {
zip_error_t error;
zip_error_init_with_code(&error, err);
fprintf(stderr, "%s: cannot open zip archive '%s': %s\n", progname, name, zip_error_strerror(&error));
zip_error_fini(&error);
return -1;
}
a->za = za;
a->nentry = (zip_uint64_t)zip_get_num_entries(za, 0);
if (a->nentry == 0)
a->entry = NULL;
else {
if ((a->nentry > SIZE_MAX / sizeof(a->entry[0])) || (a->entry = (struct entry *)malloc(sizeof(a->entry[0]) * a->nentry)) == NULL) {
fprintf(stderr, "%s: malloc failure\n", progname);
exit(1);
}
for (i = 0; i < a->nentry; i++) {
zip_stat_index(za, i, 0, &st);
a->entry[i].name = strdup(st.name);
a->entry[i].size = st.size;
a->entry[i].crc = st.crc;
if (test_files)
test_file(za, i, name, st.name, st.size, st.crc);
if (paranoid) {
a->entry[i].comp_method = st.comp_method;
ef_read(za, i, a->entry + i);
a->entry[i].comment = zip_file_get_comment(za, i, &a->entry[i].comment_length, 0);
}
else {
a->entry[i].comp_method = 0;
a->entry[i].n_extra_fields = 0;
}
}
if (paranoid) {
int length;
a->comment = zip_get_archive_comment(za, &length, 0);
a->comment_length = (size_t)length;
}
else {
a->comment = NULL;
a->comment_length = 0;
}
}
return 0;
}
static int
comment_compare(const char *c1, size_t l1, const char *c2, size_t l2) {
if (l1 != l2)
return 1;
if (l1 == 0)
return 0;
if (c1 == NULL || c2 == NULL)
return c1 == c2;
return memcmp(c1, c2, (size_t)l2);
}
static int compare_list(char *const name[2], const void *list[2], const zip_uint64_t list_length[2], int element_size, int (*cmp)(const void *a, const void *b), int (*ignore)(const void *list, int last, const void *other), int (*check)(char *const name[2], const void *a, const void *b), void (*print)(char side, const void *element), void (*start_file)(const void *element)) {
unsigned int i[2];
int j;
int diff;
#define INC(k) (i[k]++, list[k] = ((const char *)list[k]) + element_size)
#define PRINT(k) \
do { \
if (ignore && ignore(list[k], i[k] >= list_length[k] - 1, i[1-k] < list_length[1-k] ? list[1-k] : NULL)) { \
break; \
} \
print((k) ? '+' : '-', list[k]); \
(k) ? plus_count++ : minus_count++; \
diff = 1; \
} while (0)
i[0] = i[1] = 0;
diff = 0;
while (i[0] < list_length[0] && i[1] < list_length[1]) {
int c = cmp(list[0], list[1]);
if (c == 0) {
if (check) {
if (start_file) {
start_file(list[0]);
}
diff |= check(name, list[0], list[1]);
if (start_file) {
diff_output_end_file(&output);
}
}
INC(0);
INC(1);
}
else if (c < 0) {
PRINT(0);
INC(0);
}
else {
PRINT(1);
INC(1);
}
}
for (j = 0; j < 2; j++) {
while (i[j] < list_length[j]) {
PRINT(j);
INC(j);
}
}
return diff;
}
static int
ef_read(zip_t *za, zip_uint64_t idx, struct entry *e) {
zip_int16_t n_local, n_central;
zip_uint16_t i;
if ((n_local = zip_file_extra_fields_count(za, idx, ZIP_FL_LOCAL)) < 0 || (n_central = zip_file_extra_fields_count(za, idx, ZIP_FL_CENTRAL)) < 0) {
return -1;
}
e->n_extra_fields = (zip_uint16_t)(n_local + n_central);
if ((e->extra_fields = (struct ef *)malloc(sizeof(e->extra_fields[0]) * e->n_extra_fields)) == NULL)
return -1;
for (i = 0; i < n_local; i++) {
e->extra_fields[i].name = e->name;
e->extra_fields[i].data = zip_file_extra_field_get(za, idx, i, &e->extra_fields[i].id, &e->extra_fields[i].size, ZIP_FL_LOCAL);
if (e->extra_fields[i].data == NULL)
return -1;
e->extra_fields[i].flags = ZIP_FL_LOCAL;
}
for (; i < e->n_extra_fields; i++) {
e->extra_fields[i].name = e->name;
e->extra_fields[i].data = zip_file_extra_field_get(za, idx, (zip_uint16_t)(i - n_local), &e->extra_fields[i].id, &e->extra_fields[i].size, ZIP_FL_CENTRAL);
if (e->extra_fields[i].data == NULL)
return -1;
e->extra_fields[i].flags = ZIP_FL_CENTRAL;
}
qsort(e->extra_fields, e->n_extra_fields, sizeof(e->extra_fields[0]), ef_order);
return 0;
}
static int
ef_compare(char *const name[2], const struct entry *e1, const struct entry *e2) {
struct ef *ef[2];
zip_uint64_t n[2];
ef[0] = e1->extra_fields;
ef[1] = e2->extra_fields;
n[0] = e1->n_extra_fields;
n[1] = e2->n_extra_fields;
return compare_list(name, (const void **)ef, n, sizeof(struct ef), ef_order, NULL, NULL, ef_print, NULL);
}
static int
ef_order(const void *ap, const void *bp) {
const struct ef *a, *b;
a = (struct ef *)ap;
b = (struct ef *)bp;
if (a->flags != b->flags)
return a->flags - b->flags;
if (a->id != b->id)
return a->id - b->id;
if (a->size != b->size)
return a->size - b->size;
return memcmp(a->data, b->data, a->size);
}
static void
ef_print(char side, const void *p) {
const struct ef *ef = (struct ef *)p;
diff_output_data(&output, side, ef->data, ef->size, " %s extra field %s", ef->flags == ZIP_FL_LOCAL ? "local" : "central", map_enum(extra_fields, ef->id));
}
static int
entry_cmp(const void *p1, const void *p2) {
const struct entry *e1, *e2;
int c;
e1 = (struct entry *)p1;
e2 = (struct entry *)p2;
if ((c = (ignore_case ? strcasecmp : strcmp)(e1->name, e2->name)) != 0)
return c;
if (e1->size != e2->size) {
if (e1->size > e2->size)
return 1;
else
return -1;
}
if (e1->crc != e2->crc)
return (int)e1->crc - (int)e2->crc;
return 0;
}
static int
entry_ignore(const void *p, int last, const void *o) {
const struct entry *e = (const struct entry *)p;
const struct entry *other = (const struct entry *)o;
size_t length = strlen(e[0].name);
if (length == 0 || e[0].name[length - 1] != '/') {
/* not a directory */
return 0;
}
if (other != NULL && strlen(other->name) > length && strncmp(other->name, e[0].name, length) == 0) {
/* not empty in other archive */
return 1;
}
if (last || (strlen(e[1].name) < length || strncmp(e[0].name, e[1].name, length) != 0)) {
/* empty in this archive */
return 0;
}
/* not empty in this archive */
return 1;
}
static int
entry_paranoia_checks(char *const name[2], const void *p1, const void *p2) {
const struct entry *e1, *e2;
int ret;
e1 = (struct entry *)p1;
e2 = (struct entry *)p2;
ret = 0;
if (e1->comp_method != e2->comp_method) {
diff_output(&output, '-', " compression method %s", map_enum(comp_methods, e1->comp_method));
diff_output(&output, '+', " compression method %s", map_enum(comp_methods, e2->comp_method));
ret = 1;
}
if (ef_compare(name, e1, e2) != 0) {
ret = 1;
}
if (comment_compare(e1->comment, e1->comment_length, e2->comment, e2->comment_length) != 0) {
diff_output_data(&output, '-', (const zip_uint8_t *)e1->comment, e1->comment_length, " comment");
diff_output_data(&output, '+', (const zip_uint8_t *)e2->comment, e2->comment_length, " comment");
ret = 1;
}
return ret;
}
static void entry_print(char side, const void *p) {
const struct entry *e = (struct entry *)p;
diff_output_file(&output, side, e->name, e->size, e->crc);
}
static void entry_start_file(const void *p) {
const struct entry *e = (struct entry *)p;
diff_output_start_file(&output, e->name, e->size, e->crc);
}
static int
test_file(zip_t *za, zip_uint64_t idx, const char *zipname, const char *filename, zip_uint64_t size, zip_uint32_t crc) {
zip_file_t *zf;
char buf[8192];
zip_uint64_t nsize;
zip_int64_t n;
zip_uint32_t ncrc;
if ((zf = zip_fopen_index(za, idx, 0)) == NULL) {
fprintf(stderr, "%s: %s: cannot open file %s (index %" PRIu64 "): %s\n", progname, zipname, filename, idx, zip_strerror(za));
return -1;
}
ncrc = (zip_uint32_t)crc32(0, NULL, 0);
nsize = 0;
while ((n = zip_fread(zf, buf, sizeof(buf))) > 0) {
nsize += (zip_uint64_t)n;
ncrc = (zip_uint32_t)crc32(ncrc, (const Bytef *)buf, (unsigned int)n);
}
if (n < 0) {
fprintf(stderr, "%s: %s: error reading file %s (index %" PRIu64 "): %s\n", progname, zipname, filename, idx, zip_file_strerror(zf));
zip_fclose(zf);
return -1;
}
zip_fclose(zf);
if (nsize != size) {
fprintf(stderr, "%s: %s: file %s (index %" PRIu64 "): unexpected length %" PRId64 " (should be %" PRId64 ")\n", progname, zipname, filename, idx, nsize, size);
return -2;
}
if (ncrc != crc) {
fprintf(stderr, "%s: %s: file %s (index %" PRIu64 "): unexpected length %x (should be %x)\n", progname, zipname, filename, idx, ncrc, crc);
return -2;
}
return 0;
}
static const char *map_enum(const enum_map_t *map, uint32_t value) {
static char unknown[16];
size_t i = 0;
while (map[i].value < UINT32_MAX) {
if (map[i].value == value) {
return map[i].name;
}
i++;
}
snprintf(unknown, sizeof(unknown), "unknown (%u)", value);
unknown[sizeof(unknown) - 1] = '\0';
return unknown;
}