463 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			463 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* 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
 | |
| 
 | |
| 
 | |
| // 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
 | |
| 
 | |
| 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;
 | |
| }
 | 
