| /* iowin32.c -- IO base function header for compress/uncompress .zip |
| Version 1.1, February 14h, 2010 |
| part of the MiniZip project - ( http://www.winimage.com/zLibDll/minizip.html ) |
| |
| Copyright (C) 1998-2010 Gilles Vollant (minizip) ( http://www.winimage.com/zLibDll/minizip.html ) |
| |
| Modifications for Zip64 support |
| Copyright (C) 2009-2010 Mathias Svensson ( http://result42.com ) |
| |
| For more info read MiniZip_info.txt |
| |
| */ |
| |
| #include <stdlib.h> |
| |
| #include "zlib.h" |
| #include "ioapi.h" |
| #include "iowin32.h" |
| |
| #ifndef INVALID_HANDLE_VALUE |
| #define INVALID_HANDLE_VALUE (0xFFFFFFFF) |
| #endif |
| |
| #ifndef INVALID_SET_FILE_POINTER |
| #define INVALID_SET_FILE_POINTER ((DWORD)-1) |
| #endif |
| |
| |
| #ifdef _WIN32_WINNT |
| #undef _WIN32_WINNT |
| #define _WIN32_WINNT 0x601 |
| #endif |
| |
| #if _WIN32_WINNT >= _WIN32_WINNT_WIN8 |
| // see Include/shared/winapifamily.h in the Windows Kit |
| #if defined(WINAPI_FAMILY_PARTITION) && (!(defined(IOWIN32_USING_WINRT_API))) |
| #if WINAPI_FAMILY_ONE_PARTITION(WINAPI_FAMILY, WINAPI_PARTITION_APP) |
| #define IOWIN32_USING_WINRT_API 1 |
| #endif |
| #endif |
| #endif |
| |
| voidpf ZCALLBACK win32_open_file_func OF((voidpf opaque, const char* filename, int mode)); |
| uLong ZCALLBACK win32_read_file_func OF((voidpf opaque, voidpf stream, void* buf, uLong size)); |
| uLong ZCALLBACK win32_write_file_func OF((voidpf opaque, voidpf stream, const void* buf, uLong size)); |
| ZPOS64_T ZCALLBACK win32_tell64_file_func OF((voidpf opaque, voidpf stream)); |
| long ZCALLBACK win32_seek64_file_func OF((voidpf opaque, voidpf stream, ZPOS64_T offset, int origin)); |
| int ZCALLBACK win32_close_file_func OF((voidpf opaque, voidpf stream)); |
| int ZCALLBACK win32_error_file_func OF((voidpf opaque, voidpf stream)); |
| |
| typedef struct |
| { |
| HANDLE hf; |
| int error; |
| } WIN32FILE_IOWIN; |
| |
| |
| static void win32_translate_open_mode(int mode, |
| DWORD* lpdwDesiredAccess, |
| DWORD* lpdwCreationDisposition, |
| DWORD* lpdwShareMode, |
| DWORD* lpdwFlagsAndAttributes) |
| { |
| *lpdwDesiredAccess = *lpdwShareMode = *lpdwFlagsAndAttributes = *lpdwCreationDisposition = 0; |
| |
| if ((mode & ZLIB_FILEFUNC_MODE_READWRITEFILTER)==ZLIB_FILEFUNC_MODE_READ) |
| { |
| *lpdwDesiredAccess = GENERIC_READ; |
| *lpdwCreationDisposition = OPEN_EXISTING; |
| *lpdwShareMode = FILE_SHARE_READ; |
| } |
| else if (mode & ZLIB_FILEFUNC_MODE_EXISTING) |
| { |
| *lpdwDesiredAccess = GENERIC_WRITE | GENERIC_READ; |
| *lpdwCreationDisposition = OPEN_EXISTING; |
| } |
| else if (mode & ZLIB_FILEFUNC_MODE_CREATE) |
| { |
| *lpdwDesiredAccess = GENERIC_WRITE | GENERIC_READ; |
| *lpdwCreationDisposition = CREATE_ALWAYS; |
| } |
| } |
| |
| static voidpf win32_build_iowin(HANDLE hFile) |
| { |
| voidpf ret=NULL; |
| |
| if ((hFile != NULL) && (hFile != INVALID_HANDLE_VALUE)) |
| { |
| WIN32FILE_IOWIN w32fiow; |
| w32fiow.hf = hFile; |
| w32fiow.error = 0; |
| ret = malloc(sizeof(WIN32FILE_IOWIN)); |
| |
| if (ret==NULL) |
| CloseHandle(hFile); |
| else |
| *((WIN32FILE_IOWIN*)ret) = w32fiow; |
| } |
| return ret; |
| } |
| |
| voidpf ZCALLBACK win32_open64_file_func (voidpf opaque,const void* filename,int mode) |
| { |
| const char* mode_fopen = NULL; |
| DWORD dwDesiredAccess,dwCreationDisposition,dwShareMode,dwFlagsAndAttributes ; |
| HANDLE hFile = NULL; |
| |
| win32_translate_open_mode(mode,&dwDesiredAccess,&dwCreationDisposition,&dwShareMode,&dwFlagsAndAttributes); |
| |
| #ifdef IOWIN32_USING_WINRT_API |
| #ifdef UNICODE |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFile2((LPCTSTR)filename, dwDesiredAccess, dwShareMode, dwCreationDisposition, NULL); |
| #else |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| { |
| WCHAR filenameW[FILENAME_MAX + 0x200 + 1]; |
| MultiByteToWideChar(CP_ACP,0,(const char*)filename,-1,filenameW,FILENAME_MAX + 0x200); |
| hFile = CreateFile2(filenameW, dwDesiredAccess, dwShareMode, dwCreationDisposition, NULL); |
| } |
| #endif |
| #else |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFile((LPCTSTR)filename, dwDesiredAccess, dwShareMode, NULL, dwCreationDisposition, dwFlagsAndAttributes, NULL); |
| #endif |
| |
| return win32_build_iowin(hFile); |
| } |
| |
| |
| voidpf ZCALLBACK win32_open64_file_funcA (voidpf opaque,const void* filename,int mode) |
| { |
| const char* mode_fopen = NULL; |
| DWORD dwDesiredAccess,dwCreationDisposition,dwShareMode,dwFlagsAndAttributes ; |
| HANDLE hFile = NULL; |
| |
| win32_translate_open_mode(mode,&dwDesiredAccess,&dwCreationDisposition,&dwShareMode,&dwFlagsAndAttributes); |
| |
| #ifdef IOWIN32_USING_WINRT_API |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| { |
| WCHAR filenameW[FILENAME_MAX + 0x200 + 1]; |
| MultiByteToWideChar(CP_ACP,0,(const char*)filename,-1,filenameW,FILENAME_MAX + 0x200); |
| hFile = CreateFile2(filenameW, dwDesiredAccess, dwShareMode, dwCreationDisposition, NULL); |
| } |
| #else |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFileA((LPCSTR)filename, dwDesiredAccess, dwShareMode, NULL, dwCreationDisposition, dwFlagsAndAttributes, NULL); |
| #endif |
| |
| return win32_build_iowin(hFile); |
| } |
| |
| |
| voidpf ZCALLBACK win32_open64_file_funcW (voidpf opaque,const void* filename,int mode) |
| { |
| const char* mode_fopen = NULL; |
| DWORD dwDesiredAccess,dwCreationDisposition,dwShareMode,dwFlagsAndAttributes ; |
| HANDLE hFile = NULL; |
| |
| win32_translate_open_mode(mode,&dwDesiredAccess,&dwCreationDisposition,&dwShareMode,&dwFlagsAndAttributes); |
| |
| #ifdef IOWIN32_USING_WINRT_API |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFile2((LPCWSTR)filename, dwDesiredAccess, dwShareMode, dwCreationDisposition,NULL); |
| #else |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFileW((LPCWSTR)filename, dwDesiredAccess, dwShareMode, NULL, dwCreationDisposition, dwFlagsAndAttributes, NULL); |
| #endif |
| |
| return win32_build_iowin(hFile); |
| } |
| |
| |
| voidpf ZCALLBACK win32_open_file_func (voidpf opaque,const char* filename,int mode) |
| { |
| const char* mode_fopen = NULL; |
| DWORD dwDesiredAccess,dwCreationDisposition,dwShareMode,dwFlagsAndAttributes ; |
| HANDLE hFile = NULL; |
| |
| win32_translate_open_mode(mode,&dwDesiredAccess,&dwCreationDisposition,&dwShareMode,&dwFlagsAndAttributes); |
| |
| #ifdef IOWIN32_USING_WINRT_API |
| #ifdef UNICODE |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFile2((LPCTSTR)filename, dwDesiredAccess, dwShareMode, dwCreationDisposition, NULL); |
| #else |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| { |
| WCHAR filenameW[FILENAME_MAX + 0x200 + 1]; |
| MultiByteToWideChar(CP_ACP,0,(const char*)filename,-1,filenameW,FILENAME_MAX + 0x200); |
| hFile = CreateFile2(filenameW, dwDesiredAccess, dwShareMode, dwCreationDisposition, NULL); |
| } |
| #endif |
| #else |
| if ((filename!=NULL) && (dwDesiredAccess != 0)) |
| hFile = CreateFile((LPCTSTR)filename, dwDesiredAccess, dwShareMode, NULL, dwCreationDisposition, dwFlagsAndAttributes, NULL); |
| #endif |
| |
| return win32_build_iowin(hFile); |
| } |
| |
| |
| uLong ZCALLBACK win32_read_file_func (voidpf opaque, voidpf stream, void* buf,uLong size) |
| { |
| uLong ret=0; |
| HANDLE hFile = NULL; |
| if (stream!=NULL) |
| hFile = ((WIN32FILE_IOWIN*)stream) -> hf; |
| |
| if (hFile != NULL) |
| { |
| if (!ReadFile(hFile, buf, size, &ret, NULL)) |
| { |
| DWORD dwErr = GetLastError(); |
| if (dwErr == ERROR_HANDLE_EOF) |
| dwErr = 0; |
| ((WIN32FILE_IOWIN*)stream) -> error=(int)dwErr; |
| } |
| } |
| |
| return ret; |
| } |
| |
| |
| uLong ZCALLBACK win32_write_file_func (voidpf opaque,voidpf stream,const void* buf,uLong size) |
| { |
| uLong ret=0; |
| HANDLE hFile = NULL; |
| if (stream!=NULL) |
| hFile = ((WIN32FILE_IOWIN*)stream) -> hf; |
| |
| if (hFile != NULL) |
| { |
| if (!WriteFile(hFile, buf, size, &ret, NULL)) |
| { |
| DWORD dwErr = GetLastError(); |
| if (dwErr == ERROR_HANDLE_EOF) |
| dwErr = 0; |
| ((WIN32FILE_IOWIN*)stream) -> error=(int)dwErr; |
| } |
| } |
| |
| return ret; |
| } |
| |
| static BOOL MySetFilePointerEx(HANDLE hFile, LARGE_INTEGER pos, LARGE_INTEGER *newPos, DWORD dwMoveMethod) |
| { |
| #ifdef IOWIN32_USING_WINRT_API |
| return SetFilePointerEx(hFile, pos, newPos, dwMoveMethod); |
| #else |
| LONG lHigh = pos.HighPart; |
| DWORD dwNewPos = SetFilePointer(hFile, pos.LowPart, &lHigh, dwMoveMethod); |
| BOOL fOk = TRUE; |
| if (dwNewPos == 0xFFFFFFFF) |
| if (GetLastError() != NO_ERROR) |
| fOk = FALSE; |
| if ((newPos != NULL) && (fOk)) |
| { |
| newPos->LowPart = dwNewPos; |
| newPos->HighPart = lHigh; |
| } |
| return fOk; |
| #endif |
| } |
| |
| long ZCALLBACK win32_tell_file_func (voidpf opaque,voidpf stream) |
| { |
| long ret=-1; |
| HANDLE hFile = NULL; |
| if (stream!=NULL) |
| hFile = ((WIN32FILE_IOWIN*)stream) -> hf; |
| if (hFile != NULL) |
| { |
| LARGE_INTEGER pos; |
| pos.QuadPart = 0; |
| |
| if (!MySetFilePointerEx(hFile, pos, &pos, FILE_CURRENT)) |
| { |
| DWORD dwErr = GetLastError(); |
| ((WIN32FILE_IOWIN*)stream) -> error=(int)dwErr; |
| ret = -1; |
| } |
| else |
| ret=(long)pos.LowPart; |
| } |
| return ret; |
| } |
| |
| ZPOS64_T ZCALLBACK win32_tell64_file_func (voidpf opaque, voidpf stream) |
| { |
| ZPOS64_T ret= (ZPOS64_T)-1; |
| HANDLE hFile = NULL; |
| if (stream!=NULL) |
| hFile = ((WIN32FILE_IOWIN*)stream)->hf; |
| |
| if (hFile) |
| { |
| LARGE_INTEGER pos; |
| pos.QuadPart = 0; |
| |
| if (!MySetFilePointerEx(hFile, pos, &pos, FILE_CURRENT)) |
| { |
| DWORD dwErr = GetLastError(); |
| ((WIN32FILE_IOWIN*)stream) -> error=(int)dwErr; |
| ret = (ZPOS64_T)-1; |
| } |
| else |
| ret=pos.QuadPart; |
| } |
| return ret; |
| } |
| |
| |
| long ZCALLBACK win32_seek_file_func (voidpf opaque,voidpf stream,uLong offset,int origin) |
| { |
| DWORD dwMoveMethod=0xFFFFFFFF; |
| HANDLE hFile = NULL; |
| |
| long ret=-1; |
| if (stream!=NULL) |
| hFile = ((WIN32FILE_IOWIN*)stream) -> hf; |
| switch (origin) |
| { |
| case ZLIB_FILEFUNC_SEEK_CUR : |
| dwMoveMethod = FILE_CURRENT; |
| break; |
| case ZLIB_FILEFUNC_SEEK_END : |
| dwMoveMethod = FILE_END; |
| break; |
| case ZLIB_FILEFUNC_SEEK_SET : |
| dwMoveMethod = FILE_BEGIN; |
| break; |
| default: return -1; |
| } |
| |
| if (hFile != NULL) |
| { |
| LARGE_INTEGER pos; |
| pos.QuadPart = offset; |
| if (!MySetFilePointerEx(hFile, pos, NULL, dwMoveMethod)) |
| { |
| DWORD dwErr = GetLastError(); |
| ((WIN32FILE_IOWIN*)stream) -> error=(int)dwErr; |
| ret = -1; |
| } |
| else |
| ret=0; |
| } |
| return ret; |
| } |
| |
| long ZCALLBACK win32_seek64_file_func (voidpf opaque, voidpf stream,ZPOS64_T offset,int origin) |
| { |
| DWORD dwMoveMethod=0xFFFFFFFF; |
| HANDLE hFile = NULL; |
| long ret=-1; |
| |
| if (stream!=NULL) |
| hFile = ((WIN32FILE_IOWIN*)stream)->hf; |
| |
| switch (origin) |
| { |
| case ZLIB_FILEFUNC_SEEK_CUR : |
| dwMoveMethod = FILE_CURRENT; |
| break; |
| case ZLIB_FILEFUNC_SEEK_END : |
| dwMoveMethod = FILE_END; |
| break; |
| case ZLIB_FILEFUNC_SEEK_SET : |
| dwMoveMethod = FILE_BEGIN; |
| break; |
| default: return -1; |
| } |
| |
| if (hFile) |
| { |
| LARGE_INTEGER pos; |
| pos.QuadPart = offset; |
| if (!MySetFilePointerEx(hFile, pos, NULL, dwMoveMethod)) |
| { |
| DWORD dwErr = GetLastError(); |
| ((WIN32FILE_IOWIN*)stream) -> error=(int)dwErr; |
| ret = -1; |
| } |
| else |
| ret=0; |
| } |
| return ret; |
| } |
| |
| int ZCALLBACK win32_close_file_func (voidpf opaque, voidpf stream) |
| { |
| int ret=-1; |
| |
| if (stream!=NULL) |
| { |
| HANDLE hFile; |
| hFile = ((WIN32FILE_IOWIN*)stream) -> hf; |
| if (hFile != NULL) |
| { |
| CloseHandle(hFile); |
| ret=0; |
| } |
| free(stream); |
| } |
| return ret; |
| } |
| |
| int ZCALLBACK win32_error_file_func (voidpf opaque,voidpf stream) |
| { |
| int ret=-1; |
| if (stream!=NULL) |
| { |
| ret = ((WIN32FILE_IOWIN*)stream) -> error; |
| } |
| return ret; |
| } |
| |
| void fill_win32_filefunc (zlib_filefunc_def* pzlib_filefunc_def) |
| { |
| pzlib_filefunc_def->zopen_file = win32_open_file_func; |
| pzlib_filefunc_def->zread_file = win32_read_file_func; |
| pzlib_filefunc_def->zwrite_file = win32_write_file_func; |
| pzlib_filefunc_def->ztell_file = win32_tell_file_func; |
| pzlib_filefunc_def->zseek_file = win32_seek_file_func; |
| pzlib_filefunc_def->zclose_file = win32_close_file_func; |
| pzlib_filefunc_def->zerror_file = win32_error_file_func; |
| pzlib_filefunc_def->opaque = NULL; |
| } |
| |
| void fill_win32_filefunc64(zlib_filefunc64_def* pzlib_filefunc_def) |
| { |
| pzlib_filefunc_def->zopen64_file = win32_open64_file_func; |
| pzlib_filefunc_def->zread_file = win32_read_file_func; |
| pzlib_filefunc_def->zwrite_file = win32_write_file_func; |
| pzlib_filefunc_def->ztell64_file = win32_tell64_file_func; |
| pzlib_filefunc_def->zseek64_file = win32_seek64_file_func; |
| pzlib_filefunc_def->zclose_file = win32_close_file_func; |
| pzlib_filefunc_def->zerror_file = win32_error_file_func; |
| pzlib_filefunc_def->opaque = NULL; |
| } |
| |
| |
| void fill_win32_filefunc64A(zlib_filefunc64_def* pzlib_filefunc_def) |
| { |
| pzlib_filefunc_def->zopen64_file = win32_open64_file_funcA; |
| pzlib_filefunc_def->zread_file = win32_read_file_func; |
| pzlib_filefunc_def->zwrite_file = win32_write_file_func; |
| pzlib_filefunc_def->ztell64_file = win32_tell64_file_func; |
| pzlib_filefunc_def->zseek64_file = win32_seek64_file_func; |
| pzlib_filefunc_def->zclose_file = win32_close_file_func; |
| pzlib_filefunc_def->zerror_file = win32_error_file_func; |
| pzlib_filefunc_def->opaque = NULL; |
| } |
| |
| |
| void fill_win32_filefunc64W(zlib_filefunc64_def* pzlib_filefunc_def) |
| { |
| pzlib_filefunc_def->zopen64_file = win32_open64_file_funcW; |
| pzlib_filefunc_def->zread_file = win32_read_file_func; |
| pzlib_filefunc_def->zwrite_file = win32_write_file_func; |
| pzlib_filefunc_def->ztell64_file = win32_tell64_file_func; |
| pzlib_filefunc_def->zseek64_file = win32_seek64_file_func; |
| pzlib_filefunc_def->zclose_file = win32_close_file_func; |
| pzlib_filefunc_def->zerror_file = win32_error_file_func; |
| pzlib_filefunc_def->opaque = NULL; |
| } |