1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-26 03:52:01 +02:00
vcmi/lib/filesystem/MinizipExtensions.cpp

313 lines
7.6 KiB
C++
Raw Normal View History

/*
* MinizipExtensions.cpp, part of VCMI engine
*
* Authors: listed in file AUTHORS in main folder
*
* License: GNU General Public License v2.0 or later
* Full text of license available in license.txt file, in main folder
*
*/
#include "StdInc.h"
#include "MinizipExtensions.h"
2015-08-11 21:20:13 +03:00
#include "CMemoryBuffer.h"
#include <mutex>
VCMI_LIB_NAMESPACE_BEGIN
2023-02-15 01:09:07 +03:00
template<class Stream>
inline uLong streamRead(voidpf opaque, voidpf stream, void * buf, uLong size)
{
assert(opaque != nullptr);
assert(stream != nullptr);
2016-02-05 09:28:36 +03:00
2023-02-15 01:09:07 +03:00
auto * actualStream = static_cast<Stream *>(stream);
2016-02-05 09:28:36 +03:00
2023-02-15 01:09:07 +03:00
return static_cast<uLong>(actualStream->read(static_cast<ui8 *>(buf), size));
}
2023-02-15 01:09:07 +03:00
template<class Stream>
inline ZPOS64_T streamTell(voidpf opaque, voidpf stream)
{
assert(opaque != nullptr);
assert(stream != nullptr);
2016-02-05 09:28:36 +03:00
2023-02-15 01:09:07 +03:00
auto * actualStream = static_cast<Stream *>(stream);
return actualStream->tell();
}
2023-02-15 01:09:07 +03:00
template<class Stream>
inline long streamSeek(voidpf opaque, voidpf stream, ZPOS64_T offset, int origin)
{
assert(opaque != nullptr);
assert(stream != nullptr);
2016-02-05 09:28:36 +03:00
2023-02-15 01:09:07 +03:00
auto * actualStream = static_cast<Stream *>(stream);
2023-02-15 01:09:07 +03:00
long ret = 0;
switch(origin)
{
case ZLIB_FILEFUNC_SEEK_CUR:
if(actualStream->skip(offset) != offset)
ret = -1;
break;
case ZLIB_FILEFUNC_SEEK_END:
{
const si64 pos = actualStream->getSize() - offset;
if(actualStream->seek(pos) != pos)
ret = -1;
}
break;
case ZLIB_FILEFUNC_SEEK_SET:
if(actualStream->seek(offset) != offset)
ret = -1;
2023-02-15 01:09:07 +03:00
break;
default:
ret = -1;
2023-02-15 01:09:07 +03:00
}
if(ret == -1)
2016-02-10 07:28:00 +03:00
logGlobal->error("Stream seek failed");
2023-04-20 19:21:37 +04:00
return 0;
}
2023-02-15 01:09:07 +03:00
template<class Stream>
inline int streamProxyClose(voidpf opaque, voidpf stream)
{
assert(opaque != nullptr);
assert(stream != nullptr);
2016-02-05 09:28:36 +03:00
2023-02-15 01:09:07 +03:00
auto * actualStream = static_cast<Stream *>(stream);
2016-02-10 07:28:00 +03:00
2017-08-10 19:39:27 +03:00
logGlobal->trace("Proxy stream closed");
2016-02-05 09:28:36 +03:00
2016-02-10 07:28:00 +03:00
actualStream->seek(0);
2016-02-05 09:28:36 +03:00
return 0;
}
2016-02-10 07:28:00 +03:00
///CDefaultIOApi
#define GETFILE static_cast<std::FILE*>(filePtr)
#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
static inline FILE* do_open(const CharType* name, const CharType* mode)
{
#ifdef VCMI_WINDOWS
return _wfopen(name, mode);
#else
return std::fopen(name, mode);
#endif
}
static 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;
}
2016-02-10 07:28:00 +03:00
zlib_filefunc64_def CDefaultIOApi::getApiStructure()
{
static zlib_filefunc64_def MinizipFilefunc;
static std::once_flag flag;
std::call_once(flag, []
{
fill_fopen64_filefunc(&MinizipFilefunc);
MinizipFilefunc.zopen64_file = &MinizipOpenFunc;
});
return MinizipFilefunc;
}
#if MINIZIP_NEEDS_32BIT_FUNCS
zlib_filefunc_def CDefaultIOApi::getApiStructure32()
{
static zlib_filefunc_def MinizipFilefunc;
static std::once_flag flag;
std::call_once(flag, []
{
fill_fopen_filefunc(&MinizipFilefunc);
MinizipFilefunc.zopen_file = reinterpret_cast<void*(*)(void*, const char*, int)>(&MinizipOpenFunc);
});
return MinizipFilefunc;
2016-02-10 07:28:00 +03:00
}
#endif
2016-02-10 07:28:00 +03:00
///CProxyIOApi
CProxyIOApi::CProxyIOApi(CInputOutputStream * buffer):
data(buffer)
{
}
2023-02-15 01:09:07 +03:00
//must be instantiated in .cpp file for access to complete types of all member fields
CProxyIOApi::~CProxyIOApi() = default;
2016-02-10 07:28:00 +03:00
zlib_filefunc64_def CProxyIOApi::getApiStructure()
{
zlib_filefunc64_def api;
2016-02-10 07:28:00 +03:00
api.opaque = this;
api.zopen64_file = &openFileProxy;
api.zread_file = &readFileProxy;
api.zwrite_file = &writeFileProxy;
api.ztell64_file = &tellFileProxy;
api.zseek64_file = &seekFileProxy;
api.zclose_file = &closeFileProxy;
2016-02-05 09:28:36 +03:00
api.zerror_file = &errorFileProxy;
return api;
}
2016-02-10 07:28:00 +03:00
voidpf ZCALLBACK CProxyIOApi::openFileProxy(voidpf opaque, const void * filename, int mode)
2015-08-11 21:20:13 +03:00
{
2016-02-10 07:28:00 +03:00
assert(opaque != nullptr);
boost::filesystem::path path;
if(filename != nullptr)
path = static_cast<const boost::filesystem::path::value_type *>(filename);
2023-02-15 01:09:07 +03:00
return (static_cast<CProxyIOApi *>(opaque))->openFile(path, mode);
2015-08-11 21:20:13 +03:00
}
2016-02-10 07:28:00 +03:00
uLong ZCALLBACK CProxyIOApi::readFileProxy(voidpf opaque, voidpf stream, void * buf, uLong size)
{
2016-02-10 07:28:00 +03:00
return streamRead<CInputOutputStream>(opaque, stream, buf, size);
}
2016-02-05 09:28:36 +03:00
2016-02-10 07:28:00 +03:00
uLong ZCALLBACK CProxyIOApi::writeFileProxy(voidpf opaque, voidpf stream, const void * buf, uLong size)
{
assert(opaque != nullptr);
assert(stream != nullptr);
2023-02-15 01:09:07 +03:00
auto * actualStream = static_cast<CInputOutputStream *>(stream);
return static_cast<uLong>(actualStream->write(static_cast<const ui8 *>(buf), size));
}
2016-02-10 07:28:00 +03:00
ZPOS64_T ZCALLBACK CProxyIOApi::tellFileProxy(voidpf opaque, voidpf stream)
{
2016-02-10 07:28:00 +03:00
return streamTell<CInputOutputStream>(opaque, stream);
}
2016-02-05 09:28:36 +03:00
2016-02-10 07:28:00 +03:00
long ZCALLBACK CProxyIOApi::seekFileProxy(voidpf opaque, voidpf stream, ZPOS64_T offset, int origin)
{
return streamSeek<CInputOutputStream>(opaque, stream, offset, origin);
}
2016-02-10 07:28:00 +03:00
int ZCALLBACK CProxyIOApi::closeFileProxy(voidpf opaque, voidpf stream)
{
2016-02-10 07:28:00 +03:00
return streamProxyClose<CInputOutputStream>(opaque, stream);
}
2016-02-10 07:28:00 +03:00
int ZCALLBACK CProxyIOApi::errorFileProxy(voidpf opaque, voidpf stream)
{
2016-02-10 07:28:00 +03:00
return 0;
}
2016-02-10 07:28:00 +03:00
CInputOutputStream * CProxyIOApi::openFile(const boost::filesystem::path & filename, int mode)
{
logGlobal->trace("CProxyIOApi: stream opened for %s with mode %d", filename.string(), mode);
2016-02-10 07:28:00 +03:00
data->seek(0);
return data;
}
///CProxyROIOApi
CProxyROIOApi::CProxyROIOApi(CInputStream * buffer):
2015-08-11 21:20:13 +03:00
data(buffer)
{
2016-02-05 09:28:36 +03:00
}
2023-02-15 01:09:07 +03:00
//must be instantiated in .cpp file for access to complete types of all member fields
CProxyROIOApi::~CProxyROIOApi() = default;
2016-02-10 07:28:00 +03:00
zlib_filefunc64_def CProxyROIOApi::getApiStructure()
{
2016-02-10 07:28:00 +03:00
zlib_filefunc64_def api;
api.opaque = this;
api.zopen64_file = &openFileProxy;
api.zread_file = &readFileProxy;
api.zwrite_file = &writeFileProxy;
api.ztell64_file = &tellFileProxy;
api.zseek64_file = &seekFileProxy;
api.zclose_file = &closeFileProxy;
api.zerror_file = &errorFileProxy;
2016-02-05 09:28:36 +03:00
2016-02-10 07:28:00 +03:00
return api;
}
2015-08-11 21:20:13 +03:00
2016-02-10 07:28:00 +03:00
CInputStream * CProxyROIOApi::openFile(const boost::filesystem::path& filename, int mode)
2015-08-11 21:20:13 +03:00
{
logGlobal->trace("CProxyROIOApi: stream opened for %s with mode %d", filename.string(), mode);
2016-02-05 09:28:36 +03:00
2015-08-11 21:20:13 +03:00
data->seek(0);
return data;
2015-08-11 21:20:13 +03:00
}
2016-02-10 07:28:00 +03:00
voidpf ZCALLBACK CProxyROIOApi::openFileProxy(voidpf opaque, const void* filename, int mode)
{
assert(opaque != nullptr);
boost::filesystem::path path;
if(filename != nullptr)
path = static_cast<const boost::filesystem::path::value_type *>(filename);
2023-02-15 01:09:07 +03:00
return (static_cast<CProxyROIOApi *>(opaque))->openFile(path, mode);
2016-02-10 07:28:00 +03:00
}
uLong ZCALLBACK CProxyROIOApi::readFileProxy(voidpf opaque, voidpf stream, void * buf, uLong size)
{
return streamRead<CInputStream>(opaque, stream, buf, size);
}
uLong ZCALLBACK CProxyROIOApi::writeFileProxy(voidpf opaque, voidpf stream, const void* buf, uLong size)
{
2017-08-10 19:39:27 +03:00
logGlobal->error("Attempt to write to read-only stream");
2016-02-10 07:28:00 +03:00
return 0;
}
ZPOS64_T ZCALLBACK CProxyROIOApi::tellFileProxy(voidpf opaque, voidpf stream)
{
return streamTell<CInputStream>(opaque, stream);
}
long ZCALLBACK CProxyROIOApi::seekFileProxy(voidpf opaque, voidpf stream, ZPOS64_T offset, int origin)
{
return streamSeek<CInputStream>(opaque, stream, offset, origin);
}
int ZCALLBACK CProxyROIOApi::closeFileProxy(voidpf opaque, voidpf stream)
{
return streamProxyClose<CInputStream>(opaque, stream);
}
int ZCALLBACK CProxyROIOApi::errorFileProxy(voidpf opaque, voidpf stream)
2015-08-11 21:20:13 +03:00
{
2016-02-10 07:28:00 +03:00
return 0;
2015-08-11 21:20:13 +03:00
}
VCMI_LIB_NAMESPACE_END