1 /* ioapi.c -- IO base function header for compress/uncompress .zip
2 part of the MiniZip project
4 Copyright (C) 2012-2017 Nathan Moinvaziri
5 https://github.com/nmoinvaz/minizip
6 Modifications for Zip64 support
7 Copyright (C) 2009-2010 Mathias Svensson
9 Copyright (C) 1998-2010 Gilles Vollant
10 http://www.winimage.com/zLibDll/minizip.html
12 This program is distributed under the terms of the same license as zlib.
13 See the accompanying LICENSE file for the full text of the license.
19 #if defined unix || defined __APPLE__
20 #include <sys/types.h>
27 # define snprintf _snprintf
30 voidpf call_zopen64(const zlib_filefunc64_32_def *pfilefunc, const void *filename, int mode)
32 if (pfilefunc->zfile_func64.zopen64_file != NULL)
33 return (*(pfilefunc->zfile_func64.zopen64_file)) (pfilefunc->zfile_func64.opaque, filename, mode);
34 return (*(pfilefunc->zopen32_file))(pfilefunc->zfile_func64.opaque, (const char*)filename, mode);
37 voidpf call_zopendisk64(const zlib_filefunc64_32_def *pfilefunc, voidpf filestream, uint32_t number_disk, int mode)
39 if (pfilefunc->zfile_func64.zopendisk64_file != NULL)
40 return (*(pfilefunc->zfile_func64.zopendisk64_file)) (pfilefunc->zfile_func64.opaque, filestream, number_disk, mode);
41 return (*(pfilefunc->zopendisk32_file))(pfilefunc->zfile_func64.opaque, filestream, number_disk, mode);
44 long call_zseek64(const zlib_filefunc64_32_def *pfilefunc, voidpf filestream, uint64_t offset, int origin)
46 uint32_t offset_truncated = 0;
47 if (pfilefunc->zfile_func64.zseek64_file != NULL)
48 return (*(pfilefunc->zfile_func64.zseek64_file)) (pfilefunc->zfile_func64.opaque,filestream,offset,origin);
49 offset_truncated = (uint32_t)offset;
50 if (offset_truncated != offset)
52 return (*(pfilefunc->zseek32_file))(pfilefunc->zfile_func64.opaque,filestream, offset_truncated, origin);
55 uint64_t call_ztell64(const zlib_filefunc64_32_def *pfilefunc, voidpf filestream)
58 if (pfilefunc->zfile_func64.zseek64_file != NULL)
59 return (*(pfilefunc->zfile_func64.ztell64_file)) (pfilefunc->zfile_func64.opaque, filestream);
60 position = (*(pfilefunc->ztell32_file))(pfilefunc->zfile_func64.opaque, filestream);
61 if ((position) == UINT32_MAX)
66 void fill_zlib_filefunc64_32_def_from_filefunc32(zlib_filefunc64_32_def *p_filefunc64_32, const zlib_filefunc_def *p_filefunc32)
68 p_filefunc64_32->zfile_func64.zopen64_file = NULL;
69 p_filefunc64_32->zfile_func64.zopendisk64_file = NULL;
70 p_filefunc64_32->zopen32_file = p_filefunc32->zopen_file;
71 p_filefunc64_32->zopendisk32_file = p_filefunc32->zopendisk_file;
72 p_filefunc64_32->zfile_func64.zerror_file = p_filefunc32->zerror_file;
73 p_filefunc64_32->zfile_func64.zread_file = p_filefunc32->zread_file;
74 p_filefunc64_32->zfile_func64.zwrite_file = p_filefunc32->zwrite_file;
75 p_filefunc64_32->zfile_func64.ztell64_file = NULL;
76 p_filefunc64_32->zfile_func64.zseek64_file = NULL;
77 p_filefunc64_32->zfile_func64.zclose_file = p_filefunc32->zclose_file;
78 p_filefunc64_32->zfile_func64.zerror_file = p_filefunc32->zerror_file;
79 p_filefunc64_32->zfile_func64.opaque = p_filefunc32->opaque;
80 p_filefunc64_32->zseek32_file = p_filefunc32->zseek_file;
81 p_filefunc64_32->ztell32_file = p_filefunc32->ztell_file;
84 static voidpf ZCALLBACK fopen_file_func(ZIP_UNUSED voidpf opaque, const char *filename, int mode);
85 static uint32_t ZCALLBACK fread_file_func(voidpf opaque, voidpf stream, void* buf, uint32_t size);
86 static uint32_t ZCALLBACK fwrite_file_func(voidpf opaque, voidpf stream, const void *buf, uint32_t size);
87 static uint64_t ZCALLBACK ftell64_file_func(voidpf opaque, voidpf stream);
88 static long ZCALLBACK fseek64_file_func(voidpf opaque, voidpf stream, uint64_t offset, int origin);
89 static int ZCALLBACK fclose_file_func(voidpf opaque, voidpf stream);
90 static int ZCALLBACK ferror_file_func(voidpf opaque, voidpf stream);
99 static voidpf file_build_ioposix(FILE *file, const char *filename)
101 FILE_IOPOSIX *ioposix = NULL;
104 ioposix = (FILE_IOPOSIX*)malloc(sizeof(FILE_IOPOSIX));
105 ioposix->file = file;
106 ioposix->filenameLength = (int)strlen(filename) + 1;
107 ioposix->filename = (char*)malloc(ioposix->filenameLength * sizeof(char));
108 strncpy((char*)ioposix->filename, filename, ioposix->filenameLength);
109 return (voidpf)ioposix;
112 static voidpf ZCALLBACK fopen_file_func(ZIP_UNUSED voidpf opaque, const char *filename, int mode)
115 const char *mode_fopen = NULL;
116 if ((mode & ZLIB_FILEFUNC_MODE_READWRITEFILTER) == ZLIB_FILEFUNC_MODE_READ)
118 else if (mode & ZLIB_FILEFUNC_MODE_EXISTING)
120 else if (mode & ZLIB_FILEFUNC_MODE_CREATE)
123 if ((filename != NULL) && (mode_fopen != NULL))
125 file = fopen(filename, mode_fopen);
126 return file_build_ioposix(file, filename);
131 static voidpf ZCALLBACK fopen64_file_func(ZIP_UNUSED voidpf opaque, const void *filename, int mode)
134 const char *mode_fopen = NULL;
135 if ((mode & ZLIB_FILEFUNC_MODE_READWRITEFILTER) == ZLIB_FILEFUNC_MODE_READ)
137 else if (mode & ZLIB_FILEFUNC_MODE_EXISTING)
139 else if (mode & ZLIB_FILEFUNC_MODE_CREATE)
142 if ((filename != NULL) && (mode_fopen != NULL))
144 file = fopen64((const char*)filename, mode_fopen);
145 return file_build_ioposix(file, (const char*)filename);
150 static voidpf ZCALLBACK fopendisk64_file_func(voidpf opaque, voidpf stream, uint32_t number_disk, int mode)
152 FILE_IOPOSIX *ioposix = NULL;
153 char *diskFilename = NULL;
159 ioposix = (FILE_IOPOSIX*)stream;
160 diskFilename = (char*)malloc(ioposix->filenameLength * sizeof(char));
161 strncpy(diskFilename, (const char*)ioposix->filename, ioposix->filenameLength);
162 for (i = ioposix->filenameLength - 1; i >= 0; i -= 1)
164 if (diskFilename[i] != '.')
166 snprintf(&diskFilename[i], ioposix->filenameLength - i, ".z%02u", number_disk + 1);
170 ret = fopen64_file_func(opaque, diskFilename, mode);
175 static voidpf ZCALLBACK fopendisk_file_func(voidpf opaque, voidpf stream, uint32_t number_disk, int mode)
177 FILE_IOPOSIX *ioposix = NULL;
178 char *diskFilename = NULL;
184 ioposix = (FILE_IOPOSIX*)stream;
185 diskFilename = (char*)malloc(ioposix->filenameLength * sizeof(char));
186 strncpy(diskFilename, (const char*)ioposix->filename, ioposix->filenameLength);
187 for (i = ioposix->filenameLength - 1; i >= 0; i -= 1)
189 if (diskFilename[i] != '.')
191 snprintf(&diskFilename[i], ioposix->filenameLength - i, ".z%02u", number_disk + 1);
195 ret = fopen_file_func(opaque, diskFilename, mode);
200 static uint32_t ZCALLBACK fread_file_func(ZIP_UNUSED voidpf opaque, voidpf stream, void* buf, uint32_t size)
202 FILE_IOPOSIX *ioposix = NULL;
203 uint32_t read = (uint32_t)-1;
206 ioposix = (FILE_IOPOSIX*)stream;
207 read = (uint32_t)fread(buf, 1, (size_t)size, ioposix->file);
211 static uint32_t ZCALLBACK fwrite_file_func(ZIP_UNUSED voidpf opaque, voidpf stream, const void *buf, uint32_t size)
213 FILE_IOPOSIX *ioposix = NULL;
214 uint32_t written = (uint32_t)-1;
217 ioposix = (FILE_IOPOSIX*)stream;
218 written = (uint32_t)fwrite(buf, 1, (size_t)size, ioposix->file);
222 static long ZCALLBACK ftell_file_func(ZIP_UNUSED voidpf opaque, voidpf stream)
224 FILE_IOPOSIX *ioposix = NULL;
228 ioposix = (FILE_IOPOSIX*)stream;
229 ret = ftell(ioposix->file);
233 static uint64_t ZCALLBACK ftell64_file_func(ZIP_UNUSED voidpf opaque, voidpf stream)
235 FILE_IOPOSIX *ioposix = NULL;
236 uint64_t ret = (uint64_t)-1;
239 ioposix = (FILE_IOPOSIX*)stream;
240 ret = ftello64(ioposix->file);
244 static long ZCALLBACK fseek_file_func(ZIP_UNUSED voidpf opaque, voidpf stream, uint32_t offset, int origin)
246 FILE_IOPOSIX *ioposix = NULL;
247 int fseek_origin = 0;
252 ioposix = (FILE_IOPOSIX*)stream;
256 case ZLIB_FILEFUNC_SEEK_CUR:
257 fseek_origin = SEEK_CUR;
259 case ZLIB_FILEFUNC_SEEK_END:
260 fseek_origin = SEEK_END;
262 case ZLIB_FILEFUNC_SEEK_SET:
263 fseek_origin = SEEK_SET;
268 if (fseek(ioposix->file, offset, fseek_origin) != 0)
273 static long ZCALLBACK fseek64_file_func(ZIP_UNUSED voidpf opaque, voidpf stream, uint64_t offset, int origin)
275 FILE_IOPOSIX *ioposix = NULL;
276 int fseek_origin = 0;
281 ioposix = (FILE_IOPOSIX*)stream;
285 case ZLIB_FILEFUNC_SEEK_CUR:
286 fseek_origin = SEEK_CUR;
288 case ZLIB_FILEFUNC_SEEK_END:
289 fseek_origin = SEEK_END;
291 case ZLIB_FILEFUNC_SEEK_SET:
292 fseek_origin = SEEK_SET;
298 if (fseeko64(ioposix->file, offset, fseek_origin) != 0)
304 static int ZCALLBACK fclose_file_func(ZIP_UNUSED voidpf opaque, voidpf stream)
306 FILE_IOPOSIX *ioposix = NULL;
310 ioposix = (FILE_IOPOSIX*)stream;
311 if (ioposix->filename != NULL)
312 free(ioposix->filename);
313 ret = fclose(ioposix->file);
318 static int ZCALLBACK ferror_file_func(ZIP_UNUSED voidpf opaque, voidpf stream)
320 FILE_IOPOSIX *ioposix = NULL;
324 ioposix = (FILE_IOPOSIX*)stream;
325 ret = ferror(ioposix->file);
329 void fill_fopen_filefunc(zlib_filefunc_def *pzlib_filefunc_def)
331 pzlib_filefunc_def->zopen_file = fopen_file_func;
332 pzlib_filefunc_def->zopendisk_file = fopendisk_file_func;
333 pzlib_filefunc_def->zread_file = fread_file_func;
334 pzlib_filefunc_def->zwrite_file = fwrite_file_func;
335 pzlib_filefunc_def->ztell_file = ftell_file_func;
336 pzlib_filefunc_def->zseek_file = fseek_file_func;
337 pzlib_filefunc_def->zclose_file = fclose_file_func;
338 pzlib_filefunc_def->zerror_file = ferror_file_func;
339 pzlib_filefunc_def->opaque = NULL;
342 void fill_fopen64_filefunc(zlib_filefunc64_def *pzlib_filefunc_def)
344 pzlib_filefunc_def->zopen64_file = fopen64_file_func;
345 pzlib_filefunc_def->zopendisk64_file = fopendisk64_file_func;
346 pzlib_filefunc_def->zread_file = fread_file_func;
347 pzlib_filefunc_def->zwrite_file = fwrite_file_func;
348 pzlib_filefunc_def->ztell64_file = ftell64_file_func;
349 pzlib_filefunc_def->zseek64_file = fseek64_file_func;
350 pzlib_filefunc_def->zclose_file = fclose_file_func;
351 pzlib_filefunc_def->zerror_file = ferror_file_func;
352 pzlib_filefunc_def->opaque = NULL;