2017-10-04 22:06:33 -07:00
|
|
|
/* mzstrm_posix.c -- Stream for filesystem access for posix/linux
|
|
|
|
part of the MiniZip project
|
|
|
|
|
|
|
|
Copyright (C) 2012-2017 Nathan Moinvaziri
|
|
|
|
https://github.com/nmoinvaz/minizip
|
|
|
|
Modifications for Zip64 support
|
|
|
|
Copyright (C) 2009-2010 Mathias Svensson
|
|
|
|
http://result42.com
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#if defined unix || defined __APPLE__
|
|
|
|
# include <unistd.h>
|
|
|
|
# include <utime.h>
|
|
|
|
# include <fcntl.h>
|
|
|
|
#elif defined _WIN32
|
2017-10-05 00:02:07 -07:00
|
|
|
# include <sys/utime.h>
|
2017-10-04 22:06:33 -07:00
|
|
|
# include <direct.h>
|
|
|
|
# include <io.h>
|
|
|
|
# define chdir(x) _chdir(x)
|
|
|
|
# define mkdir(x,y) _mkdir(x)
|
|
|
|
#endif
|
|
|
|
|
2017-10-04 22:10:11 -07:00
|
|
|
#include "mz_strm.h"
|
|
|
|
#include "mz_strm_posix.h"
|
2017-10-04 22:06:33 -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
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef ZCR_SEED2
|
|
|
|
# define ZCR_SEED2 3141592654UL // use PI as default pattern
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
typedef struct mz_stream_posix_s
|
|
|
|
{
|
|
|
|
mz_stream stream;
|
|
|
|
FILE *handle;
|
|
|
|
void *path;
|
|
|
|
uint16_t path_size;
|
|
|
|
} mz_stream_posix;
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_open(void *stream, const char *path, int mode)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix *)stream;
|
|
|
|
const char *mode_fopen = NULL;
|
|
|
|
|
|
|
|
if (path == NULL)
|
|
|
|
return MZ_STREAM_ERROR;
|
|
|
|
|
|
|
|
if ((mode & MZ_STREAM_MODE_READWRITE) == MZ_STREAM_MODE_READ)
|
|
|
|
mode_fopen = "rb";
|
|
|
|
else if (mode & MZ_STREAM_MODE_APPEND)
|
2017-10-05 00:25:34 -07:00
|
|
|
mode_fopen = "ab";
|
2017-10-04 22:06:33 -07:00
|
|
|
else if (mode & MZ_STREAM_MODE_CREATE)
|
|
|
|
mode_fopen = "wb";
|
|
|
|
else
|
|
|
|
return MZ_STREAM_ERROR;
|
|
|
|
|
|
|
|
posix->handle = fopen64(path, mode_fopen);
|
|
|
|
strncpy((char *)posix->path, path, posix->path_size);
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_is_open(void *stream)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
if (posix->handle == NULL)
|
|
|
|
return MZ_STREAM_ERROR;
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_read(void *stream, void* buf, uint32_t size)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
return (uint32_t)fread(buf, 1, (size_t)size, posix->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_write(void *stream, const void *buf, uint32_t size)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
return (uint32_t)fwrite(buf, 1, (size_t)size, posix->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t mz_stream_posix_tell(void *stream)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
return ftello64(posix->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_seek(void *stream, uint64_t offset, int origin)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
int fseek_origin = 0;
|
|
|
|
|
|
|
|
switch (origin)
|
|
|
|
{
|
|
|
|
case MZ_STREAM_SEEK_CUR:
|
|
|
|
fseek_origin = SEEK_CUR;
|
|
|
|
break;
|
|
|
|
case MZ_STREAM_SEEK_END:
|
|
|
|
fseek_origin = SEEK_END;
|
|
|
|
break;
|
|
|
|
case MZ_STREAM_SEEK_SET:
|
|
|
|
fseek_origin = SEEK_SET;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return MZ_STREAM_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fseeko64(posix->handle, offset, fseek_origin) != 0)
|
|
|
|
return MZ_STREAM_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_close(void *stream)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
int closed = fclose(posix->handle);
|
|
|
|
if (posix->path != NULL)
|
|
|
|
free(posix->path);
|
|
|
|
posix->handle = NULL;
|
|
|
|
if (closed != 0)
|
|
|
|
return MZ_STREAM_ERROR;
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t mz_stream_posix_error(void *stream)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = (mz_stream_posix*)stream;
|
|
|
|
return ferror(posix->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *mz_stream_posix_create(void **stream)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = NULL;
|
|
|
|
|
|
|
|
posix = (mz_stream_posix *)malloc(sizeof(mz_stream_posix));
|
|
|
|
if (posix != NULL)
|
|
|
|
{
|
|
|
|
posix->stream.open = mz_stream_posix_open;
|
|
|
|
posix->stream.is_open = mz_stream_posix_is_open;
|
|
|
|
posix->stream.read = mz_stream_posix_read;
|
|
|
|
posix->stream.write = mz_stream_posix_write;
|
|
|
|
posix->stream.tell = mz_stream_posix_tell;
|
|
|
|
posix->stream.seek = mz_stream_posix_seek;
|
|
|
|
posix->stream.close = mz_stream_posix_close;
|
|
|
|
posix->stream.error = mz_stream_posix_error;
|
|
|
|
posix->stream.create = mz_stream_posix_create;
|
|
|
|
posix->stream.delete = mz_stream_posix_delete;
|
|
|
|
}
|
|
|
|
if (stream != NULL)
|
|
|
|
*stream = posix;
|
|
|
|
|
|
|
|
return posix;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mz_stream_posix_delete(void **stream)
|
|
|
|
{
|
|
|
|
mz_stream_posix *posix = NULL;
|
|
|
|
if (stream == NULL)
|
|
|
|
return;
|
|
|
|
posix = (mz_stream_posix *)*stream;
|
|
|
|
if (posix != NULL)
|
|
|
|
free(posix);
|
|
|
|
}
|
|
|
|
|
2017-10-05 00:21:27 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2017-10-04 22:06:33 -07:00
|
|
|
int32_t mz_posix_rand(uint8_t *buf, uint32_t size)
|
|
|
|
{
|
|
|
|
static unsigned calls = 0;
|
|
|
|
void *rand_stream = NULL;
|
|
|
|
int32_t len = 0;
|
|
|
|
|
|
|
|
|
|
|
|
if (mz_stream_posix_create(&rand_stream) == NULL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (mz_stream_posix_open(rand_stream, "/dev/urandom", MZ_STREAM_MODE_READ) == MZ_OK)
|
|
|
|
{
|
|
|
|
len = mz_stream_posix_read(rand_stream, buf, size);
|
|
|
|
mz_stream_posix_close(rand_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
mz_stream_posix_delete(&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;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t mz_posix_get_file_date(const char *path, uint32_t *dos_date)
|
|
|
|
{
|
|
|
|
struct stat stat_info;
|
|
|
|
struct tm *filedate = NULL;
|
|
|
|
time_t tm_t = 0;
|
|
|
|
int16_t err = MZ_INTERNAL_ERROR;
|
|
|
|
|
|
|
|
memset(&stat_info, 0, sizeof(stat_info));
|
|
|
|
|
|
|
|
if (strcmp(path, "-") != 0)
|
|
|
|
{
|
|
|
|
size_t len = strlen(path);
|
|
|
|
char *name = (char *)malloc(len + 1);
|
|
|
|
strncpy(name, path, len + 1);
|
|
|
|
name[len] = 0;
|
|
|
|
if (name[len - 1] == '/')
|
|
|
|
name[len - 1] = 0;
|
|
|
|
|
|
|
|
/* Not all systems allow stat'ing a file with / appended */
|
|
|
|
if (stat(name, &stat_info) == 0)
|
|
|
|
{
|
|
|
|
tm_t = stat_info.st_mtime;
|
|
|
|
err = MZ_OK;
|
|
|
|
}
|
|
|
|
free(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
filedate = localtime(&tm_t);
|
|
|
|
*dos_date = mz_tm_to_dosdate(filedate);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t mz_posix_set_file_date(const char *path, uint32_t dos_date)
|
|
|
|
{
|
|
|
|
struct utimbuf ut;
|
|
|
|
|
|
|
|
ut.actime = mz_dosdate_to_time_t(dos_date);
|
|
|
|
ut.modtime = mz_dosdate_to_time_t(dos_date);
|
|
|
|
|
|
|
|
if (utime(path, &ut) != 0)
|
|
|
|
return MZ_INTERNAL_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t mz_posix_change_dir(const char *path)
|
|
|
|
{
|
|
|
|
if (chdir(path) != 0)
|
|
|
|
return MZ_INTERNAL_ERROR;
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t mz_posix_make_dir(const char *path)
|
|
|
|
{
|
|
|
|
if (mkdir(path, 0755) != 0)
|
|
|
|
return MZ_INTERNAL_ERROR;
|
|
|
|
return MZ_OK;
|
2017-10-05 00:02:07 -07:00
|
|
|
}
|