minizip-ng/ioapi.c

283 lines
7.3 KiB
C
Raw Normal View History

2017-06-05 13:36:52 +08:00
/* ioapi.c -- IO base function header for compress/uncompress .zip
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
Modifications for Zip64 support
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
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>
#include <time.h>
2012-10-21 15:20:13 -07:00
#if defined unix || defined __APPLE__
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <fcntl.h>
#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
#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-10-01 21:12:12 -07:00
2017-09-29 21:02:09 -07:00
typedef struct mzstream_posix_s
{
2017-10-01 21:12:12 -07:00
mzstream stream;
2017-09-29 21:02:09 -07:00
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:
fseek_origin = SEEK_CUR;
break;
2017-09-29 21:02:09 -07:00
case MZSTREAM_SEEK_END:
fseek_origin = SEEK_END;
break;
2017-09-29 21:02:09 -07:00
case MZSTREAM_SEEK_SET:
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 21:12:12 -07:00
int32_t posix_rand(uint8_t *buf, uint32_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
}