minizip-ng/minizip.c

462 lines
12 KiB
C
Raw Normal View History

2012-01-21 14:53:44 -07:00
/*
minizip.c
Version 1.1, February 14h, 2010
sample part of the MiniZip project - ( http://www.winimage.com/zLibDll/minizip.html )
2012-10-25 00:21:50 -07:00
Copyright (C) 1998-2010 Gilles Vollant (minizip) ( http://www.winimage.com/zLibDll/minizip.html )
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
Modifications of Unzip for Zip64
Copyright (C) 2007-2008 Even Rouault
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
Modifications for Zip64 support on both zip and unzip
Copyright (C) 2009-2010 Mathias Svensson ( http://result42.com )
2012-01-21 14:53:44 -07:00
*/
2012-01-21 15:12:36 -07:00
#if (!defined(_WIN32)) && (!defined(WIN32)) && (!defined(__APPLE__))
2012-10-25 00:21:50 -07:00
#ifndef __USE_FILE_OFFSET64
#define __USE_FILE_OFFSET64
#endif
#ifndef __USE_LARGEFILE64
#define __USE_LARGEFILE64
#endif
#ifndef _LARGEFILE64_SOURCE
#define _LARGEFILE64_SOURCE
#endif
#ifndef _FILE_OFFSET_BIT
#define _FILE_OFFSET_BIT 64
#endif
2012-01-21 14:53:44 -07:00
#endif
2012-07-14 16:45:49 -07:00
#if (defined(_WIN32))
#ifndef _CRT_SECURE_NO_WARNINGS
#define _CRT_SECURE_NO_WARNINGS
#endif
#endif
2012-01-21 15:12:36 -07:00
#ifdef __APPLE__
// In darwin and perhaps other BSD variants off_t is a 64 bit value, hence no need for specific 64 bit functions
#define FOPEN_FUNC(filename, mode) fopen(filename, mode)
#define FTELLO_FUNC(stream) ftello(stream)
#define FSEEKO_FUNC(stream, offset, origin) fseeko(stream, offset, origin)
#else
#define FOPEN_FUNC(filename, mode) fopen64(filename, mode)
#define FTELLO_FUNC(stream) ftello64(stream)
#define FSEEKO_FUNC(stream, offset, origin) fseeko64(stream, offset, origin)
#endif
2012-01-21 14:53:44 -07:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <errno.h>
#include <fcntl.h>
2012-01-21 15:12:36 -07:00
#ifdef _WIN32
# include <direct.h>
# include <io.h>
#else
2012-01-21 14:53:44 -07:00
# include <unistd.h>
# include <utime.h>
# include <sys/types.h>
# include <sys/stat.h>
#endif
#include "zip.h"
#ifdef _WIN32
2012-10-25 00:21:50 -07:00
#define USEWIN32IOAPI
#include "iowin32.h"
2012-01-21 14:53:44 -07:00
#endif
#define WRITEBUFFERSIZE (16384)
2012-10-25 00:21:50 -07:00
#define MAXFILENAME (256)
2012-01-21 14:53:44 -07:00
#ifdef _WIN32
2012-10-25 00:21:50 -07:00
uLong filetime(const char *filename, tm_zip *tmzip, uLong *dostime)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
int ret = 0;
FILETIME ftLocal;
HANDLE hFind;
WIN32_FIND_DATAA ff32;
hFind = FindFirstFileA(filename, &ff32);
if (hFind != INVALID_HANDLE_VALUE)
{
FileTimeToLocalFileTime(&(ff32.ftLastWriteTime), &ftLocal);
FileTimeToDosDateTime(&ftLocal,((LPWORD)dostime)+1,((LPWORD)dostime)+0);
2012-01-21 14:53:44 -07:00
FindClose(hFind);
ret = 1;
2012-10-25 00:21:50 -07:00
}
return ret;
2012-01-21 14:53:44 -07:00
}
#else
2012-01-21 15:12:36 -07:00
#ifdef unix || __APPLE__
2012-10-25 00:21:50 -07:00
uLong filetime(char *filename, tm_zip *tmzip, uLong *dostime)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
int ret = 0;
struct stat s; /* results of stat() */
struct tm* filedate;
time_t tm_t = 0;
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
if (strcmp(filename,"-")!=0)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
char name[MAXFILENAME+1];
int len = strlen(filename);
if (len > MAXFILENAME)
len = MAXFILENAME;
strncpy(name, filename, MAXFILENAME - 1);
name[MAXFILENAME] = 0;
if (name[len - 1] == '/')
name[len - 1] = 0;
/* not all systems allow stat'ing a file with / appended */
if (stat(name,&s) == 0)
{
tm_t = s.st_mtime;
ret = 1;
}
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -07:00
filedate = localtime(&tm_t);
tmzip->tm_sec = filedate->tm_sec;
tmzip->tm_min = filedate->tm_min;
tmzip->tm_hour = filedate->tm_hour;
tmzip->tm_mday = filedate->tm_mday;
tmzip->tm_mon = filedate->tm_mon ;
tmzip->tm_year = filedate->tm_year;
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
return ret;
2012-01-21 14:53:44 -07:00
}
#else
2012-10-25 00:21:50 -07:00
uLong filetime(char *filename, tm_zip *tmzip, uLong *dostime)
2012-01-21 14:53:44 -07:00
{
return 0;
}
#endif
#endif
2012-10-25 00:21:50 -07:00
int check_exist_file(const char* filename)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
FILE* ftestexist = FOPEN_FUNC(filename,"rb");
if (ftestexist == NULL)
return 0;
fclose(ftestexist);
return 1;
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -07:00
int isLargeFile(const char* filename)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
ZPOS64_T pos = 0;
FILE* pFile = FOPEN_FUNC(filename, "rb");
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
if (pFile == NULL)
return 0;
FSEEKO_FUNC(pFile, 0, SEEK_END);
pos = FTELLO_FUNC(pFile);
fclose(pFile);
printf("File : %s is %lld bytes\n", filename, pos);
return (pos >= 0xffffffff);
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -07:00
/* Calculate the CRC32 of a file, because to encrypt a file, we need known the CRC32 of the file before */
int getFileCrc(const char* filenameinzip, void *buf, unsigned long size_buf, unsigned long* result_crc)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
unsigned long calculate_crc = 0;
int err = ZIP_OK;
FILE *fin = NULL;
unsigned long size_read = 0;
unsigned long total_read = 0;
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
fin = FOPEN_FUNC(filenameinzip,"rb");
if (fin == NULL)
err = ZIP_ERRNO;
2012-10-25 00:29:35 -07:00
else
2012-10-25 00:21:50 -07:00
{
2012-01-21 14:53:44 -07:00
do
{
size_read = (int)fread(buf,1,size_buf,fin);
2012-10-25 00:21:50 -07:00
if ((size_read < size_buf) && (feof(fin) == 0))
2012-01-21 14:53:44 -07:00
{
printf("error in reading %s\n",filenameinzip);
err = ZIP_ERRNO;
}
2012-10-25 00:21:50 -07:00
if (size_read > 0)
2012-01-21 14:53:44 -07:00
calculate_crc = crc32(calculate_crc,buf,size_read);
2012-10-25 00:21:50 -07:00
2012-01-21 14:53:44 -07:00
total_read += size_read;
2012-10-25 00:21:50 -07:00
}
while ((err == ZIP_OK) && (size_read > 0));
}
2012-01-21 14:53:44 -07:00
if (fin)
fclose(fin);
printf("file %s crc %lx\n", filenameinzip, calculate_crc);
2012-10-25 00:29:35 -07:00
*result_crc = calculate_crc;
2012-01-21 14:53:44 -07:00
return err;
}
2012-10-25 00:21:50 -07:00
void do_banner()
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
printf("MiniZip 1.1, demo of zLib + MiniZip64 package, written by Gilles Vollant\n");
printf("more info on MiniZip at http://www.winimage.com/zLibDll/minizip.html\n\n");
}
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
void do_help()
{
printf("Usage : minizip [-o] [-a] [-0 to -9] [-p password] [-j] file.zip [files_to_add]\n\n" \
" -o Overwrite existing file.zip\n" \
" -a Append to existing file.zip\n" \
" -0 Store only\n" \
" -1 Compress faster\n" \
" -9 Compress better\n\n" \
" -j exclude path. store only the file name.\n\n");
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:29:35 -07:00
int main(int argc, char *argv[])
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
zipFile zf = NULL;
zlib_filefunc64_def ffunc = {0};
2012-10-25 00:29:35 -07:00
char *zipfilename = NULL;
const char* password = NULL;
2012-10-25 00:21:50 -07:00
void* buf = NULL;
int size_buf = WRITEBUFFERSIZE;
int zipfilenamearg = 0;
int errclose = 0;
int err = 0;
int len = 0;
int i = 0;
int opt_overwrite = 0;
int opt_compress_level = Z_DEFAULT_COMPRESSION;
int opt_exclude_path = 0;
2012-01-21 14:53:44 -07:00
do_banner();
2012-10-25 00:21:50 -07:00
if (argc == 1)
2012-01-21 14:53:44 -07:00
{
do_help();
return 0;
}
2012-10-25 00:21:50 -07:00
2012-10-25 00:29:35 -07:00
/* Parse command line options */
2012-10-25 00:21:50 -07:00
for (i = 1; i < argc; i++)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
if ((*argv[i]) == '-')
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
const char *p = argv[i]+1;
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
while ((*p) != '\0')
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
char c = *(p++);;
if ((c == 'o') || (c == 'O'))
opt_overwrite = 1;
if ((c == 'a') || (c == 'A'))
opt_overwrite = 2;
if ((c >= '0') && (c <= '9'))
opt_compress_level = (c - '0');
if ((c == 'j') || (c == 'J'))
opt_exclude_path = 1;
if (((c == 'p') || (c == 'P')) && (i+1 < argc))
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
password=argv[i+1];
i++;
2012-01-21 14:53:44 -07:00
}
}
}
2012-10-25 00:21:50 -07:00
else
{
if (zipfilenamearg == 0)
zipfilenamearg = i;
}
}
if (zipfilenamearg == 0)
{
do_help();
return 0;
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:29:35 -07:00
zipfilename = argv[zipfilenamearg];
2012-01-21 14:53:44 -07:00
buf = (void*)malloc(size_buf);
2012-10-25 00:21:50 -07:00
if (buf == NULL)
2012-01-21 14:53:44 -07:00
{
printf("Error allocating memory\n");
return ZIP_INTERNALERROR;
}
2012-10-25 00:21:50 -07:00
if (opt_overwrite == 2)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
/* If the file don't exist, we not append file */
2012-10-25 00:29:35 -07:00
if (check_exist_file(zipfilename) == 0)
2012-10-25 00:21:50 -07:00
opt_overwrite = 1;
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -07:00
else if (opt_overwrite == 0)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
/* If ask the user what to do because append and overwrite args not set */
2012-10-25 00:29:35 -07:00
if (check_exist_file(zipfilename) != 0)
2012-10-25 00:21:50 -07:00
{
char rep = 0;
do
{
char answer[128];
2012-10-25 00:29:35 -07:00
printf("The file %s exists. Overwrite ? [y]es, [n]o, [a]ppend : ", zipfilename);
if (scanf("%1s", answer) != 1)
exit(EXIT_FAILURE);
2012-10-25 00:21:50 -07:00
rep = answer[0];
2012-01-21 14:53:44 -07:00
2012-10-25 00:29:35 -07:00
if ((rep >= 'a') && (rep <= 'z'))
2012-10-25 00:21:50 -07:00
rep -= 0x20;
}
2012-10-25 00:29:35 -07:00
while ((rep != 'Y') && (rep != 'N') && (rep != 'A'));
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
if (rep == 'A')
opt_overwrite = 2;
2012-10-25 00:29:35 -07:00
else if (rep == 'N')
{
do_help();
free(buf);
return 0;
}
2012-01-21 14:53:44 -07:00
}
}
2012-10-25 00:21:50 -07:00
#ifdef USEWIN32IOAPI
fill_win32_filefunc64A(&ffunc);
2012-10-25 00:29:35 -07:00
zf = zipOpen2_64(zipfilename, (opt_overwrite == 2) ? APPEND_STATUS_ADDINZIP : 0, NULL, &ffunc);
2012-10-25 00:21:50 -07:00
#else
2012-10-25 00:29:35 -07:00
zf = zipOpen64(zipfilename, (opt_overwrite == 2) ? APPEND_STATUS_ADDINZIP : 0);
2012-10-25 00:21:50 -07:00
#endif
if (zf == NULL)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:29:35 -07:00
printf("error opening %s\n", zipfilename);
2012-10-25 00:21:50 -07:00
err = ZIP_ERRNO;
}
else
2012-10-25 00:29:35 -07:00
printf("creating %s\n", zipfilename);
2012-10-25 00:21:50 -07:00
/* Go through command line args looking for files to add to zip */
for (i = zipfilenamearg+1; (i < argc) && (err == ZIP_OK); i++)
2012-10-25 00:21:50 -07:00
{
FILE *fin = NULL;
int size_read = 0;
2012-10-25 00:21:50 -07:00
const char* filenameinzip = argv[i];
const char *savefilenameinzip;
zip_fileinfo zi = {0};
unsigned long crcFile = 0;
int zip64 = 0;
/* Skip command line options */
if ((((*(argv[i])) == '-') || ((*(argv[i])) == '/')) && (strlen(argv[i]) == 2) &&
((argv[i][1] == 'o') || (argv[i][1] == 'O') || (argv[i][1] == 'a') || (argv[i][1] == 'A') ||
(argv[i][1] == 'p') || (argv[i][1] == 'P') || ((argv[i][1] >= '0') && (argv[i][1] <= '9'))))
2012-10-25 00:21:50 -07:00
continue;
/* Get information about the file on disk so we can store it in zip */
filetime(filenameinzip, &zi.tmz_date, &zi.dosDate);
if ((password != NULL) && (err == ZIP_OK))
2012-10-25 00:29:35 -07:00
err = getFileCrc(filenameinzip, buf, size_buf, &crcFile);
2012-10-25 00:21:50 -07:00
zip64 = isLargeFile(filenameinzip);
/* Construct the filename that our file will be stored in the zip as.
The path name saved, should not include a leading slash.
If it did, windows/xp and dynazip couldn't read the zip file. */
savefilenameinzip = filenameinzip;
while (savefilenameinzip[0] == '\\' || savefilenameinzip[0] == '/')
savefilenameinzip++;
/* Should the file be stored with any path info at all? */
if (opt_exclude_path)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
const char *tmpptr = NULL;
const char *lastslash = 0;
for (tmpptr = savefilenameinzip; *tmpptr; tmpptr++)
{
if (*tmpptr == '\\' || *tmpptr == '/')
lastslash = tmpptr;
}
if (lastslash != NULL)
savefilenameinzip = lastslash + 1; // base filename follows last slash.
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -07:00
/* Add to zip file */
err = zipOpenNewFileInZip3_64(zf, savefilenameinzip, &zi,
NULL, 0, NULL, 0, NULL /* comment*/,
(opt_compress_level != 0) ? Z_DEFLATED : 0,
opt_compress_level,0,
/* -MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY, */
-MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY,
password, crcFile, zip64);
if (err != ZIP_OK)
printf("error in opening %s in zipfile (%d)\n", filenameinzip, err);
2012-01-21 14:53:44 -07:00
else
2012-10-25 00:21:50 -07:00
{
fin = FOPEN_FUNC(filenameinzip, "rb");
if (fin == NULL)
{
err = ZIP_ERRNO;
printf("error in opening %s for reading\n", filenameinzip);
}
}
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
if (err == ZIP_OK)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
/* Read contents of file and write it to zip */
do
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
size_read = (int)fread(buf, 1, size_buf, fin);
if ((size_read < size_buf) && (feof(fin)==0))
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
printf("error in reading %s\n",filenameinzip);
err = ZIP_ERRNO;
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -07:00
if (size_read > 0)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
err = zipWriteInFileInZip(zf, buf, size_read);
if (err < 0)
printf("error in writing %s in the zipfile (%d)\n", filenameinzip, err);
2012-01-21 14:53:44 -07:00
}
}
2012-10-25 00:21:50 -07:00
while ((err == ZIP_OK) && (size_read > 0));
}
if (fin)
fclose(fin);
if (err < 0)
err = ZIP_ERRNO;
else
{
err = zipCloseFileInZip(zf);
if (err != ZIP_OK)
printf("error in closing %s in the zipfile (%d)\n", filenameinzip, err);
2012-01-21 14:53:44 -07:00
}
}
2012-10-25 00:21:50 -07:00
errclose = zipClose(zf, NULL);
if (errclose != ZIP_OK)
printf("error in closing %s (%d)\n", zipfilename, errclose);
2012-10-25 00:21:50 -07:00
2012-01-21 14:53:44 -07:00
free(buf);
return err;
2012-01-21 14:53:44 -07:00
}