123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364 |
- #include "StdInc.h"
- #include "FileStream.h"
- #ifdef USE_SYSTEM_MINIZIP
- #include <minizip/unzip.h>
- #else
- #include "../minizip/unzip.h"
- #endif
- #include <cstdio>
- ///copied from ioapi.c due to linker issues on MSVS
- #include "../minizip/ioapi.h"
- #if defined(__APPLE__) || defined(IOAPI_NO_64)
- // In darwin and perhaps other BSD variants off_t is a 64 bit value, hence no need for specific 64 bit functions
- #define FOPEN_FUNC(filename, mode) fopen(filename, mode)
- #define FTELLO_FUNC(stream) ftello(stream)
- #define FSEEKO_FUNC(stream, offset, origin) fseeko(stream, offset, origin)
- #else
- #define FOPEN_FUNC(filename, mode) fopen64(filename, mode)
- #define FTELLO_FUNC(stream) ftello64(stream)
- #define FSEEKO_FUNC(stream, offset, origin) fseeko64(stream, offset, origin)
- #endif
- voidpf call_zopen64 (const zlib_filefunc64_32_def* pfilefunc,const void*filename,int mode)
- {
- if (pfilefunc->zfile_func64.zopen64_file != NULL)
- return (*(pfilefunc->zfile_func64.zopen64_file)) (pfilefunc->zfile_func64.opaque,filename,mode);
- else
- {
- return (*(pfilefunc->zopen32_file))(pfilefunc->zfile_func64.opaque,(const char*)filename,mode);
- }
- }
- long call_zseek64(const zlib_filefunc64_32_def* pfilefunc, voidpf filestream, ZPOS64_T offset, int origin)
- {
- if (pfilefunc->zfile_func64.zseek64_file != NULL)
- return (*(pfilefunc->zfile_func64.zseek64_file)) (pfilefunc->zfile_func64.opaque, filestream, offset, origin);
- else
- {
- uLong offsetTruncated = (uLong)offset;
- if (offsetTruncated != offset)
- return -1;
- else
- return (*(pfilefunc->zseek32_file))(pfilefunc->zfile_func64.opaque, filestream, offsetTruncated, origin);
- }
- }
- ZPOS64_T call_ztell64(const zlib_filefunc64_32_def* pfilefunc, voidpf filestream)
- {
- if (pfilefunc->zfile_func64.zseek64_file != NULL)
- return (*(pfilefunc->zfile_func64.ztell64_file)) (pfilefunc->zfile_func64.opaque, filestream);
- else
- {
- uLong tell_uLong = (*(pfilefunc->ztell32_file))(pfilefunc->zfile_func64.opaque, filestream);
- if ((tell_uLong) == MAXU32)
- return (ZPOS64_T)-1;
- else
- return tell_uLong;
- }
- }
- static voidpf ZCALLBACK fopen_file_func OF((voidpf opaque, const char* filename, int mode));
- static uLong ZCALLBACK fread_file_func OF((voidpf opaque, voidpf stream, void* buf, uLong size));
- static uLong ZCALLBACK fwrite_file_func OF((voidpf opaque, voidpf stream, const void* buf,uLong size));
- static ZPOS64_T ZCALLBACK ftell64_file_func OF((voidpf opaque, voidpf stream));
- static long ZCALLBACK fseek64_file_func OF((voidpf opaque, voidpf stream, ZPOS64_T offset, int origin));
- static int ZCALLBACK fclose_file_func OF((voidpf opaque, voidpf stream));
- static int ZCALLBACK ferror_file_func OF((voidpf opaque, voidpf stream));
- static voidpf ZCALLBACK fopen_file_func (voidpf opaque, const char* filename, int mode)
- {
- FILE* file = NULL;
- const char* mode_fopen = NULL;
- if ((mode & ZLIB_FILEFUNC_MODE_READWRITEFILTER) == ZLIB_FILEFUNC_MODE_READ)
- mode_fopen = "rb";
- else
- if (mode & ZLIB_FILEFUNC_MODE_EXISTING)
- mode_fopen = "r+b";
- else
- if (mode & ZLIB_FILEFUNC_MODE_CREATE)
- mode_fopen = "wb";
- if ((filename != NULL) && (mode_fopen != NULL))
- file = fopen(filename, mode_fopen);
- return file;
- }
- static voidpf ZCALLBACK fopen64_file_func(voidpf opaque, const void* filename, int mode)
- {
- FILE* file = NULL;
- const char* mode_fopen = NULL;
- if ((mode & ZLIB_FILEFUNC_MODE_READWRITEFILTER) == ZLIB_FILEFUNC_MODE_READ)
- mode_fopen = "rb";
- else
- if (mode & ZLIB_FILEFUNC_MODE_EXISTING)
- mode_fopen = "r+b";
- else
- if (mode & ZLIB_FILEFUNC_MODE_CREATE)
- mode_fopen = "wb";
- if ((filename != NULL) && (mode_fopen != NULL))
- file = FOPEN_FUNC((const char*)filename, mode_fopen);
- return file;
- }
- static uLong ZCALLBACK fread_file_func(voidpf opaque, voidpf stream, void* buf, uLong size)
- {
- uLong ret;
- ret = (uLong)fread(buf, 1, (size_t)size, (FILE *)stream);
- return ret;
- }
- static uLong ZCALLBACK fwrite_file_func(voidpf opaque, voidpf stream, const void* buf, uLong size)
- {
- uLong ret;
- ret = (uLong)fwrite(buf, 1, (size_t)size, (FILE *)stream);
- return ret;
- }
- static long ZCALLBACK ftell_file_func(voidpf opaque, voidpf stream)
- {
- long ret;
- ret = ftell((FILE *)stream);
- return ret;
- }
- static ZPOS64_T ZCALLBACK ftell64_file_func(voidpf opaque, voidpf stream)
- {
- ZPOS64_T ret;
- ret = FTELLO_FUNC((FILE *)stream);
- return ret;
- }
- static long ZCALLBACK fseek_file_func(voidpf opaque, voidpf stream, uLong offset, int origin)
- {
- int fseek_origin = 0;
- long ret;
- switch (origin)
- {
- case ZLIB_FILEFUNC_SEEK_CUR:
- fseek_origin = SEEK_CUR;
- break;
- case ZLIB_FILEFUNC_SEEK_END:
- fseek_origin = SEEK_END;
- break;
- case ZLIB_FILEFUNC_SEEK_SET:
- fseek_origin = SEEK_SET;
- break;
- default: return -1;
- }
- ret = 0;
- if (fseek((FILE *)stream, offset, fseek_origin) != 0)
- ret = -1;
- return ret;
- }
- static long ZCALLBACK fseek64_file_func(voidpf opaque, voidpf stream, ZPOS64_T offset, int origin)
- {
- int fseek_origin = 0;
- long ret;
- switch (origin)
- {
- case ZLIB_FILEFUNC_SEEK_CUR:
- fseek_origin = SEEK_CUR;
- break;
- case ZLIB_FILEFUNC_SEEK_END:
- fseek_origin = SEEK_END;
- break;
- case ZLIB_FILEFUNC_SEEK_SET:
- fseek_origin = SEEK_SET;
- break;
- default: return -1;
- }
- ret = 0;
- if (FSEEKO_FUNC((FILE *)stream, offset, fseek_origin) != 0)
- ret = -1;
- return ret;
- }
- static int ZCALLBACK fclose_file_func(voidpf opaque, voidpf stream)
- {
- int ret;
- ret = fclose((FILE *)stream);
- return ret;
- }
- static int ZCALLBACK ferror_file_func(voidpf opaque, voidpf stream)
- {
- int ret;
- ret = ferror((FILE *)stream);
- return ret;
- }
- ///end of ioapi.c
- //extern MINIZIP_EXPORT void fill_fopen64_filefunc(zlib_filefunc64_def* pzlib_filefunc_def);
- #ifdef VCMI_WINDOWS
- #ifndef _CRT_SECURE_NO_WARNINGS
- #define _CRT_SECURE_NO_WARNINGS
- #endif
- #include <cwchar>
- #define CHAR_LITERAL(s) L##s
- using CharType = wchar_t;
- #else
- #define CHAR_LITERAL(s) s
- using CharType = char;
- #endif
- inline FILE* do_open(const CharType* name, const CharType* mode)
- {
- #ifdef VCMI_WINDOWS
- return _wfopen(name, mode);
- #else
- return std::fopen(name, mode);
- #endif
- }
- #define GETFILE static_cast<std::FILE*>(filePtr)
- voidpf ZCALLBACK MinizipOpenFunc(voidpf opaque, const void* filename, int mode)
- {
- const CharType* mode_fopen = [mode]() -> const CharType*
- {
- if ((mode & ZLIB_FILEFUNC_MODE_READWRITEFILTER) == ZLIB_FILEFUNC_MODE_READ)
- return CHAR_LITERAL("rb");
- else if (mode & ZLIB_FILEFUNC_MODE_EXISTING)
- return CHAR_LITERAL("r+b");
- else if (mode & ZLIB_FILEFUNC_MODE_CREATE)
- return CHAR_LITERAL("wb");
- return nullptr;
- }();
- if (filename != nullptr && mode_fopen != nullptr)
- return do_open(static_cast<const CharType*>(filename), mode_fopen);
- else
- return nullptr;
- }
- void fill_fopen64_filefunc(zlib_filefunc64_def* pzlib_filefunc_def)
- {
- pzlib_filefunc_def->zopen64_file = fopen64_file_func;
- pzlib_filefunc_def->zread_file = fread_file_func;
- pzlib_filefunc_def->zwrite_file = fwrite_file_func;
- pzlib_filefunc_def->ztell64_file = ftell64_file_func;
- pzlib_filefunc_def->zseek64_file = fseek64_file_func;
- pzlib_filefunc_def->zclose_file = fclose_file_func;
- pzlib_filefunc_def->zerror_file = ferror_file_func;
- pzlib_filefunc_def->opaque = NULL;
- }
- zlib_filefunc64_def* FileStream::GetMinizipFilefunc()
- {
- static zlib_filefunc64_def MinizipFilefunc;
- static bool initialized = false;
- if (!initialized)
- {
- fill_fopen64_filefunc(&MinizipFilefunc);
- MinizipFilefunc.zopen64_file = &MinizipOpenFunc;
- initialized = true;
- }
- return &MinizipFilefunc;
- }
- template class boost::iostreams::stream<FileBuf>;
- /*static*/
- bool FileStream::CreateFile(const boost::filesystem::path& filename)
- {
- FILE* f = do_open(filename.c_str(), CHAR_LITERAL("wb"));
- bool result = (f != nullptr);
- fclose(f);
- return result;
- }
- FileBuf::FileBuf(const boost::filesystem::path& filename, std::ios_base::openmode mode)
- {
- auto openmode = [mode]() -> std::basic_string<CharType>
- {
- using namespace std;
- switch (mode & (~ios_base::ate & ~ios_base::binary))
- {
- case (ios_base::in):
- return CHAR_LITERAL("r");
- case (ios_base::out):
- case (ios_base::out | ios_base::trunc):
- return CHAR_LITERAL("w");
- case (ios_base::app):
- case (ios_base::out | ios_base::app):
- return CHAR_LITERAL("a");
- case (ios_base::out | ios_base::in):
- return CHAR_LITERAL("r+");
- case (ios_base::out | ios_base::in | ios_base::trunc):
- return CHAR_LITERAL("w+");
- case (ios_base::out | ios_base::in | ios_base::app):
- case (ios_base::in | ios_base::app):
- return CHAR_LITERAL("a+");
- default:
- throw std::ios_base::failure("invalid open mode");
- }
- }();
- if (mode & std::ios_base::binary)
- openmode += CHAR_LITERAL('b');
- filePtr = do_open(filename.c_str(), openmode.c_str());
- if (filePtr == nullptr)
- throw std::ios_base::failure("could not open file");
- if (mode & std::ios_base::ate) {
- if (std::fseek(GETFILE, 0, SEEK_END)) {
- fclose(GETFILE);
- throw std::ios_base::failure("could not open file");
- }
- }
- }
- void FileBuf::close()
- {
- std::fclose(GETFILE);
- }
- std::streamsize FileBuf::read(char* s, std::streamsize n)
- {
- return static_cast<std::streamsize>(std::fread(s, 1, n, GETFILE));
- }
- std::streamsize FileBuf::write(const char* s, std::streamsize n)
- {
- return static_cast<std::streamsize>(std::fwrite(s, 1, n, GETFILE));
- }
- std::streamoff FileBuf::seek(std::streamoff off, std::ios_base::seekdir way)
- {
- const auto src = [way]() -> int
- {
- switch(way)
- {
- case std::ios_base::beg:
- return SEEK_SET;
- case std::ios_base::cur:
- return SEEK_CUR;
- case std::ios_base::end:
- return SEEK_END;
- default:
- throw std::ios_base::failure("bad seek direction");
- }
- }();
- if(std::fseek(GETFILE, off, src))
- throw std::ios_base::failure("bad seek offset");
- return static_cast<std::streamsize>(std::ftell(GETFILE));
- }
|