2017-06-05 13:36:52 +08:00
|
|
|
/* ioapi.c -- IO base function header for compress/uncompress .zip
|
2014-01-12 14:04:54 -07:00
|
|
|
part of the MiniZip project
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-16 13:25:02 +08:00
|
|
|
Copyright (C) 2012-2017 Nathan Moinvaziri
|
|
|
|
https://github.com/nmoinvaz/minizip
|
2012-07-14 16:31:22 -07:00
|
|
|
Modifications for Zip64 support
|
2014-01-12 14:04:54 -07:00
|
|
|
Copyright (C) 2009-2010 Mathias Svensson
|
|
|
|
http://result42.com
|
2017-09-16 13:25:02 +08:00
|
|
|
Copyright (C) 1998-2010 Gilles Vollant
|
|
|
|
http://www.winimage.com/zLibDll/minizip.html
|
2014-01-12 14:04:54 -07:00
|
|
|
|
|
|
|
This program is distributed under the terms of the same license as zlib.
|
|
|
|
See the accompanying LICENSE file for the full text of the license.
|
2012-02-25 23:28:28 -07:00
|
|
|
*/
|
|
|
|
|
2012-10-21 15:20:13 -07:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2017-10-01 20:33:01 -07:00
|
|
|
#include <time.h>
|
2012-10-21 15:20:13 -07:00
|
|
|
|
2017-03-04 10:01:10 -08:00
|
|
|
#if defined unix || defined __APPLE__
|
|
|
|
#include <sys/types.h>
|
2017-10-01 20:33:01 -07:00
|
|
|
#include <sys/stat.h>
|
2017-03-04 10:01:10 -08:00
|
|
|
#include <unistd.h>
|
2017-10-01 20:33:01 -07:00
|
|
|
#include <fcntl.h>
|
2017-03-04 10:01:10 -08:00
|
|
|
#endif
|
|
|
|
|
2012-10-21 15:20:13 -07:00
|
|
|
#include "ioapi.h"
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
#if defined(USE_FILE32API)
|
|
|
|
# define fopen64 fopen
|
|
|
|
# define ftello64 ftell
|
|
|
|
# define fseeko64 fseek
|
|
|
|
#else
|
|
|
|
# if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__DragonFly__) || defined(__OpenBSD__) || defined(__APPLE__) || defined(__ANDROID__)
|
|
|
|
# define fopen64 fopen
|
|
|
|
# define ftello64 ftello
|
|
|
|
# define fseeko64 fseeko
|
|
|
|
# endif
|
|
|
|
# ifdef _MSC_VER
|
|
|
|
# define fopen64 fopen
|
|
|
|
# if (_MSC_VER >= 1400) && (!(defined(NO_MSCVER_FILE64_FUNC)))
|
|
|
|
# define ftello64 _ftelli64
|
|
|
|
# define fseeko64 _fseeki64
|
|
|
|
# else /* old MSC */
|
|
|
|
# define ftello64 ftell
|
|
|
|
# define fseeko64 fseek
|
|
|
|
# endif
|
|
|
|
# endif
|
2012-02-25 23:28:28 -07:00
|
|
|
#endif
|
2012-07-14 16:31:22 -07:00
|
|
|
|
2017-10-01 20:33:01 -07:00
|
|
|
#ifndef ZCR_SEED2
|
|
|
|
# define ZCR_SEED2 3141592654UL // use PI as default pattern
|
|
|
|
#endif
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_open(voidpf stream, const char *filename, int mode)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->open == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->open(strm, filename, mode);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_is_open(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->is_open == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->is_open(strm);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_read(voidpf stream, void* buf, uint32_t size)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->read == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->read(strm, buf, size);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_write(voidpf stream, const void *buf, uint32_t size)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->write == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->write(strm, buf, size);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int64_t mzstream_tell(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->tell == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->tell(strm);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_seek(voidpf stream, uint64_t offset, int origin)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->seek == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->seek(strm, offset, origin);
|
|
|
|
}
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_close(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->close == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->close(strm);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_error(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
if (strm == NULL || strm->error == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return strm->error(strm);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t mzstream_set_base(voidpf stream, voidpf base)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream *strm = (mzstream *)stream;
|
|
|
|
strm->base = (mzstream *)base;
|
|
|
|
return MZSTREAM_OK;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
2017-09-29 21:02:09 -07:00
|
|
|
typedef struct mzstream_posix_s
|
|
|
|
{
|
|
|
|
mzstream stream;
|
|
|
|
FILE *handle;
|
|
|
|
void *filename;
|
|
|
|
uint16_t filename_size;
|
|
|
|
} mzstream_posix;
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_open(voidpf stream, const char *filename, int mode)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix *)stream;
|
|
|
|
const char *mode_fopen = NULL;
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
if (filename == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
if ((mode & MZSTREAM_MODE_READWRITEFILTER) == MZSTREAM_MODE_READ)
|
|
|
|
mode_fopen = "rb";
|
|
|
|
else if (mode & MZSTREAM_MODE_EXISTING)
|
|
|
|
mode_fopen = "r+b";
|
|
|
|
else if (mode & MZSTREAM_MODE_CREATE)
|
|
|
|
mode_fopen = "wb";
|
|
|
|
else
|
|
|
|
return MZSTREAM_ERR;
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
posix->handle = fopen64((const char*)filename, mode_fopen);
|
|
|
|
strncpy((char *)posix->filename, filename, posix->filename_size);
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
return MZSTREAM_OK;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_is_open(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
|
|
|
if (posix->handle == NULL)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return MZSTREAM_OK;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_read(voidpf stream, void* buf, uint32_t size)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
|
|
|
return (uint32_t)fread(buf, 1, (size_t)size, posix->handle);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_write(voidpf stream, const void *buf, uint32_t size)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
|
|
|
return (uint32_t)fwrite(buf, 1, (size_t)size, posix->handle);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int64_t ZCALLBACK mzstream_posix_tell(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
|
|
|
return ftello64(posix->handle);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_seek(voidpf stream, uint64_t offset, int origin)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
2012-10-25 02:07:21 -07:00
|
|
|
int fseek_origin = 0;
|
2012-02-25 23:28:28 -07:00
|
|
|
|
|
|
|
switch (origin)
|
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
case MZSTREAM_SEEK_CUR:
|
2014-01-07 18:28:33 -07:00
|
|
|
fseek_origin = SEEK_CUR;
|
|
|
|
break;
|
2017-09-29 21:02:09 -07:00
|
|
|
case MZSTREAM_SEEK_END:
|
2014-01-07 18:28:33 -07:00
|
|
|
fseek_origin = SEEK_END;
|
|
|
|
break;
|
2017-09-29 21:02:09 -07:00
|
|
|
case MZSTREAM_SEEK_SET:
|
2014-01-07 18:28:33 -07:00
|
|
|
fseek_origin = SEEK_SET;
|
|
|
|
break;
|
|
|
|
default:
|
2017-09-29 21:02:09 -07:00
|
|
|
return MZSTREAM_ERR;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
if (fseeko64(posix->handle, offset, fseek_origin) != 0)
|
|
|
|
return MZSTREAM_ERR;
|
2012-02-25 23:28:28 -07:00
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
return MZSTREAM_OK;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_close(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
|
|
|
int closed = fclose(posix->handle);
|
|
|
|
if (posix->filename != NULL)
|
|
|
|
free(posix->filename);
|
|
|
|
posix->handle = NULL;
|
|
|
|
if (closed != 0)
|
|
|
|
return MZSTREAM_ERR;
|
|
|
|
return MZSTREAM_OK;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
int32_t ZCALLBACK mzstream_posix_error(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix*)stream;
|
|
|
|
return ferror(posix->handle);
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
voidpf mzstream_posix_alloc(void)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = NULL;
|
|
|
|
|
|
|
|
posix = (mzstream_posix *)malloc(sizeof(mzstream_posix));
|
|
|
|
if (posix == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
posix->stream.open = mzstream_posix_open;
|
|
|
|
posix->stream.is_open = mzstream_posix_is_open;
|
|
|
|
posix->stream.read = mzstream_posix_read;
|
|
|
|
posix->stream.write = mzstream_posix_write;
|
|
|
|
posix->stream.tell = mzstream_posix_tell;
|
|
|
|
posix->stream.seek = mzstream_posix_seek;
|
|
|
|
posix->stream.close = mzstream_posix_close;
|
|
|
|
posix->stream.error = mzstream_posix_error;
|
|
|
|
posix->stream.alloc = mzstream_posix_alloc;
|
|
|
|
posix->stream.free = mzstream_posix_free;
|
|
|
|
|
|
|
|
return (voidpf)posix;
|
2012-02-25 23:28:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:02:09 -07:00
|
|
|
void mzstream_posix_free(voidpf stream)
|
2012-02-25 23:28:28 -07:00
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
mzstream_posix *posix = (mzstream_posix *)stream;
|
|
|
|
if (posix != NULL)
|
|
|
|
free(posix);
|
2017-10-01 20:33:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mzstream_posix_rand(uint8_t *buf, uint16_t size)
|
|
|
|
{
|
|
|
|
static unsigned calls = 0;
|
|
|
|
voidpf rand_stream = NULL;
|
|
|
|
int32_t len = 0;
|
|
|
|
|
|
|
|
|
|
|
|
rand_stream = mzstream_posix_alloc();
|
|
|
|
|
|
|
|
if (mzstream_posix_open(rand_stream, "/dev/urandom", MZSTREAM_MODE_READ) == MZSTREAM_OK)
|
|
|
|
{
|
|
|
|
len = mzstream_posix_read(rand_stream, buf, size);
|
|
|
|
mzstream_posix_close(rand_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
mzstream_posix_free(rand_stream);
|
|
|
|
|
|
|
|
if (len < (int)size)
|
|
|
|
{
|
|
|
|
// Ensure different random header each time
|
|
|
|
if (++calls == 1)
|
|
|
|
srand((unsigned)(time(NULL) ^ ZCR_SEED2));
|
|
|
|
|
|
|
|
while (len < (int)size)
|
|
|
|
buf[len++] = (rand() >> 7) & 0xff;
|
|
|
|
}
|
|
|
|
return len;
|
2017-09-29 21:02:09 -07:00
|
|
|
}
|