mirror of
https://github.com/daveallie/crosspoint-reader.git
synced 2025-12-16 22:27:42 +03:00
1710 lines
75 KiB
C++
1710 lines
75 KiB
C++
#ifndef MINIZ_EXPORT
|
|
#define MINIZ_EXPORT
|
|
#endif
|
|
/* miniz.c 3.1.0 - public domain deflate/inflate, zlib-subset, ZIP
|
|
reading/writing/appending, PNG writing See "unlicense" statement at the end
|
|
of this file. Rich Geldreich <richgel99@gmail.com>, last updated Oct. 13,
|
|
2013 Implements RFC 1950: http://www.ietf.org/rfc/rfc1950.txt and RFC 1951:
|
|
http://www.ietf.org/rfc/rfc1951.txt
|
|
|
|
Most API's defined in miniz.c are optional. For example, to disable the
|
|
archive related functions just define MINIZ_NO_ARCHIVE_APIS, or to get rid of
|
|
all stdio usage define MINIZ_NO_STDIO (see the list below for more macros).
|
|
|
|
* Low-level Deflate/Inflate implementation notes:
|
|
|
|
Compression: Use the "tdefl" API's. The compressor supports raw, static,
|
|
and dynamic blocks, lazy or greedy parsing, match length filtering, RLE-only,
|
|
and Huffman-only streams. It performs and compresses approximately as well as
|
|
zlib.
|
|
|
|
Decompression: Use the "tinfl" API's. The entire decompressor is
|
|
implemented as a single function coroutine: see tinfl_decompress(). It
|
|
supports decompression into a 32KB (or larger power of 2) wrapping buffer, or
|
|
into a memory block large enough to hold the entire file.
|
|
|
|
The low-level tdefl/tinfl API's do not make any use of dynamic memory
|
|
allocation.
|
|
|
|
* zlib-style API notes:
|
|
|
|
miniz.c implements a fairly large subset of zlib. There's enough
|
|
functionality present for it to be a drop-in zlib replacement in many apps:
|
|
The z_stream struct, optional memory allocation callbacks
|
|
deflateInit/deflateInit2/deflate/deflateReset/deflateEnd/deflateBound
|
|
inflateInit/inflateInit2/inflate/inflateReset/inflateEnd
|
|
compress, compress2, compressBound, uncompress
|
|
CRC-32, Adler-32 - Using modern, minimal code size, CPU cache friendly
|
|
routines. Supports raw deflate streams or standard zlib streams with adler-32
|
|
checking.
|
|
|
|
Limitations:
|
|
The callback API's are not implemented yet. No support for gzip headers or
|
|
zlib static dictionaries. I've tried to closely emulate zlib's various
|
|
flavors of stream flushing and return status codes, but there are no
|
|
guarantees that miniz.c pulls this off perfectly.
|
|
|
|
* PNG writing: See the tdefl_write_image_to_png_file_in_memory() function,
|
|
originally written by Alex Evans. Supports 1-4 bytes/pixel images.
|
|
|
|
* ZIP archive API notes:
|
|
|
|
The ZIP archive API's where designed with simplicity and efficiency in
|
|
mind, with just enough abstraction to get the job done with minimal fuss.
|
|
There are simple API's to retrieve file information, read files from existing
|
|
archives, create new archives, append new files to existing archives, or
|
|
clone archive data from one archive to another. It supports archives located
|
|
in memory or the heap, on disk (using stdio.h), or you can specify custom
|
|
file read/write callbacks.
|
|
|
|
- Archive reading: Just call this function to read a single file from a
|
|
disk archive:
|
|
|
|
void *mz_zip_extract_archive_file_to_heap(const char *pZip_filename, const
|
|
char *pArchive_name, size_t *pSize, mz_uint zip_flags);
|
|
|
|
For more complex cases, use the "mz_zip_reader" functions. Upon opening an
|
|
archive, the entire central directory is located and read as-is into memory,
|
|
and subsequent file access only occurs when reading individual files.
|
|
|
|
- Archives file scanning: The simple way is to use this function to scan a
|
|
loaded archive for a specific file:
|
|
|
|
int mz_zip_reader_locate_file(mz_zip_archive *pZip, const char *pName,
|
|
const char *pComment, mz_uint flags);
|
|
|
|
The locate operation can optionally check file comments too, which (as one
|
|
example) can be used to identify multiple versions of the same file in an
|
|
archive. This function uses a simple linear search through the central
|
|
directory, so it's not very fast.
|
|
|
|
Alternately, you can iterate through all the files in an archive (using
|
|
mz_zip_reader_get_num_files()) and retrieve detailed info on each file by
|
|
calling mz_zip_reader_file_stat().
|
|
|
|
- Archive creation: Use the "mz_zip_writer" functions. The ZIP writer
|
|
immediately writes compressed file data to disk and builds an exact image of
|
|
the central directory in memory. The central directory image is written all
|
|
at once at the end of the archive file when the archive is finalized.
|
|
|
|
The archive writer can optionally align each file's local header and file
|
|
data to any power of 2 alignment, which can be useful when the archive will
|
|
be read from optical media. Also, the writer supports placing arbitrary data
|
|
blobs at the very beginning of ZIP archives. Archives written using either
|
|
feature are still readable by any ZIP tool.
|
|
|
|
- Archive appending: The simple way to add a single file to an archive is
|
|
to call this function:
|
|
|
|
mz_bool mz_zip_add_mem_to_archive_file_in_place(const char *pZip_filename,
|
|
const char *pArchive_name, const void *pBuf, size_t buf_size, const void
|
|
*pComment, mz_uint16 comment_size, mz_uint level_and_flags);
|
|
|
|
The archive will be created if it doesn't already exist, otherwise it'll be
|
|
appended to. Note the appending is done in-place and is not an atomic
|
|
operation, so if something goes wrong during the operation it's possible the
|
|
archive could be left without a central directory (although the local file
|
|
headers and file data will be fine, so the archive will be recoverable).
|
|
|
|
For more complex archive modification scenarios:
|
|
1. The safest way is to use a mz_zip_reader to read the existing archive,
|
|
cloning only those bits you want to preserve into a new archive using using
|
|
the mz_zip_writer_add_from_zip_reader() function (which compiles the
|
|
compressed file data as-is). When you're done, delete the old archive and
|
|
rename the newly written archive, and you're done. This is safe but requires
|
|
a bunch of temporary disk space or heap memory.
|
|
|
|
2. Or, you can convert an mz_zip_reader in-place to an mz_zip_writer using
|
|
mz_zip_writer_init_from_reader(), append new files as needed, then finalize
|
|
the archive which will write an updated central directory to the original
|
|
archive. (This is basically what mz_zip_add_mem_to_archive_file_in_place()
|
|
does.) There's a possibility that the archive's central directory could be
|
|
lost with this method if anything goes wrong, though.
|
|
|
|
- ZIP archive support limitations:
|
|
No spanning support. Extraction functions can only handle unencrypted,
|
|
stored or deflated files. Requires streams capable of seeking.
|
|
|
|
* This is a header file library, like stb_image.c. To get only a header file,
|
|
either cut and paste the below header, or create miniz.h, #define
|
|
MINIZ_HEADER_FILE_ONLY, and then include miniz.c from it.
|
|
|
|
* Important: For best perf. be sure to customize the below macros for your
|
|
target platform: #define MINIZ_USE_UNALIGNED_LOADS_AND_STORES 1 #define
|
|
MINIZ_LITTLE_ENDIAN 1 #define MINIZ_HAS_64BIT_REGISTERS 1
|
|
|
|
* On platforms using glibc, Be sure to "#define _LARGEFILE64_SOURCE 1" before
|
|
including miniz.c to ensure miniz uses the 64-bit variants: fopen64(),
|
|
stat64(), etc. Otherwise you won't be able to process large files (i.e.
|
|
32-bit stat() fails for me on files > 0x7FFFFFFF bytes).
|
|
*/
|
|
#pragma once
|
|
|
|
/* Defines to completely disable specific portions of miniz.c:
|
|
If all macros here are defined the only functionality remaining will be
|
|
CRC-32 and adler-32. */
|
|
|
|
/* Define MINIZ_NO_STDIO to disable all usage and any functions which rely on
|
|
* stdio for file I/O. */
|
|
/*#define MINIZ_NO_STDIO */
|
|
|
|
/* If MINIZ_NO_TIME is specified then the ZIP archive functions will not be able
|
|
* to get the current time, or */
|
|
/* get/set file times, and the C run-time funcs that get/set times won't be
|
|
* called. */
|
|
/* The current downside is the times written to your archives will be from 1979.
|
|
*/
|
|
/*#define MINIZ_NO_TIME */
|
|
|
|
/* Define MINIZ_NO_DEFLATE_APIS to disable all compression API's. */
|
|
/*#define MINIZ_NO_DEFLATE_APIS */
|
|
|
|
/* Define MINIZ_NO_INFLATE_APIS to disable all decompression API's. */
|
|
/*#define MINIZ_NO_INFLATE_APIS */
|
|
|
|
/* Define MINIZ_NO_ARCHIVE_APIS to disable all ZIP archive API's. */
|
|
/*#define MINIZ_NO_ARCHIVE_APIS */
|
|
|
|
/* Define MINIZ_NO_ARCHIVE_WRITING_APIS to disable all writing related ZIP
|
|
* archive API's. */
|
|
/*#define MINIZ_NO_ARCHIVE_WRITING_APIS */
|
|
|
|
/* Define MINIZ_NO_ZLIB_APIS to remove all ZLIB-style compression/decompression
|
|
* API's. */
|
|
/*#define MINIZ_NO_ZLIB_APIS */
|
|
|
|
/* Define MINIZ_NO_ZLIB_COMPATIBLE_NAME to disable zlib names, to prevent
|
|
* conflicts against stock zlib. */
|
|
/*#define MINIZ_NO_ZLIB_COMPATIBLE_NAMES */
|
|
|
|
/* Define MINIZ_NO_MALLOC to disable all calls to malloc, free, and realloc.
|
|
Note if MINIZ_NO_MALLOC is defined then the user must always provide custom
|
|
user alloc/free/realloc callbacks to the zlib and archive API's, and a few
|
|
stand-alone helper API's which don't provide custom user functions (such as
|
|
tdefl_compress_mem_to_heap() and tinfl_decompress_mem_to_heap()) won't work.
|
|
*/
|
|
/*#define MINIZ_NO_MALLOC */
|
|
|
|
#ifdef MINIZ_NO_INFLATE_APIS
|
|
#define MINIZ_NO_ARCHIVE_APIS
|
|
#endif
|
|
|
|
#ifdef MINIZ_NO_DEFLATE_APIS
|
|
#define MINIZ_NO_ARCHIVE_WRITING_APIS
|
|
#endif
|
|
|
|
#if defined(__TINYC__) && (defined(__linux) || defined(__linux__))
|
|
/* TODO: Work around "error: include file 'sys\utime.h' when compiling with tcc
|
|
* on Linux */
|
|
#define MINIZ_NO_TIME
|
|
#endif
|
|
|
|
#include <stddef.h>
|
|
|
|
#if !defined(MINIZ_NO_TIME) && !defined(MINIZ_NO_ARCHIVE_APIS)
|
|
#include <time.h>
|
|
#endif
|
|
|
|
#if defined(_M_IX86) || defined(_M_X64) || defined(__i386__) || defined(__i386) || defined(__i486__) || \
|
|
defined(__i486) || defined(i386) || defined(__ia64__) || defined(__x86_64__)
|
|
/* MINIZ_X86_OR_X64_CPU is only used to help set the below macros. */
|
|
#define MINIZ_X86_OR_X64_CPU 1
|
|
#else
|
|
#define MINIZ_X86_OR_X64_CPU 0
|
|
#endif
|
|
|
|
/* Set MINIZ_LITTLE_ENDIAN only if not set */
|
|
#if !defined(MINIZ_LITTLE_ENDIAN)
|
|
#if defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__)
|
|
|
|
#if (__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__)
|
|
/* Set MINIZ_LITTLE_ENDIAN to 1 if the processor is little endian. */
|
|
#define MINIZ_LITTLE_ENDIAN 1
|
|
#else
|
|
#define MINIZ_LITTLE_ENDIAN 0
|
|
#endif
|
|
|
|
#else
|
|
|
|
#if MINIZ_X86_OR_X64_CPU
|
|
#define MINIZ_LITTLE_ENDIAN 1
|
|
#else
|
|
#define MINIZ_LITTLE_ENDIAN 0
|
|
#endif
|
|
|
|
#endif
|
|
#endif
|
|
|
|
/* Using unaligned loads and stores causes errors when using UBSan */
|
|
#if defined(__has_feature)
|
|
#if __has_feature(undefined_behavior_sanitizer)
|
|
#define MINIZ_USE_UNALIGNED_LOADS_AND_STORES 0
|
|
#endif
|
|
#endif
|
|
|
|
/* Set MINIZ_USE_UNALIGNED_LOADS_AND_STORES only if not set */
|
|
#if !defined(MINIZ_USE_UNALIGNED_LOADS_AND_STORES)
|
|
#if MINIZ_X86_OR_X64_CPU
|
|
/* Set MINIZ_USE_UNALIGNED_LOADS_AND_STORES to 1 on CPU's that permit efficient
|
|
* integer loads and stores from unaligned addresses. */
|
|
#define MINIZ_USE_UNALIGNED_LOADS_AND_STORES 0
|
|
#define MINIZ_UNALIGNED_USE_MEMCPY
|
|
#else
|
|
#define MINIZ_USE_UNALIGNED_LOADS_AND_STORES 0
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(_M_X64) || defined(_WIN64) || defined(__MINGW64__) || defined(_LP64) || defined(__LP64__) || \
|
|
defined(__ia64__) || defined(__x86_64__)
|
|
/* Set MINIZ_HAS_64BIT_REGISTERS to 1 if operations on 64-bit integers are
|
|
* reasonably fast (and don't involve compiler generated calls to helper
|
|
* functions). */
|
|
#define MINIZ_HAS_64BIT_REGISTERS 1
|
|
#else
|
|
#define MINIZ_HAS_64BIT_REGISTERS 0
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/* ------------------- zlib-style API Definitions. */
|
|
|
|
/* For more compatibility with zlib, miniz.c uses unsigned long for some
|
|
* parameters/struct members. Beware: mz_ulong can be either 32 or 64-bits! */
|
|
typedef unsigned long mz_ulong;
|
|
|
|
/* mz_free() internally uses the MZ_FREE() macro (which by default calls free()
|
|
* unless you've modified the MZ_MALLOC macro) to release a block allocated from
|
|
* the heap. */
|
|
MINIZ_EXPORT void mz_free(void* p);
|
|
|
|
#define MZ_ADLER32_INIT (1)
|
|
/* mz_adler32() returns the initial adler-32 value to use when called with
|
|
* ptr==NULL. */
|
|
MINIZ_EXPORT mz_ulong mz_adler32(mz_ulong adler, const unsigned char* ptr, size_t buf_len);
|
|
|
|
#define MZ_CRC32_INIT (0)
|
|
/* mz_crc32() returns the initial CRC-32 value to use when called with
|
|
* ptr==NULL. */
|
|
MINIZ_EXPORT mz_ulong mz_crc32(mz_ulong crc, const unsigned char* ptr, size_t buf_len);
|
|
|
|
/* Compression strategies. */
|
|
enum { MZ_DEFAULT_STRATEGY = 0, MZ_FILTERED = 1, MZ_HUFFMAN_ONLY = 2, MZ_RLE = 3, MZ_FIXED = 4 };
|
|
|
|
/* Method */
|
|
#define MZ_DEFLATED 8
|
|
|
|
/* Heap allocation callbacks.
|
|
Note that mz_alloc_func parameter types purposely differ from zlib's: items/size
|
|
is size_t, not unsigned long. */
|
|
typedef void* (*mz_alloc_func)(void* opaque, size_t items, size_t size);
|
|
typedef void (*mz_free_func)(void* opaque, void* address);
|
|
typedef void* (*mz_realloc_func)(void* opaque, void* address, size_t items, size_t size);
|
|
|
|
/* Compression levels: 0-9 are the standard zlib-style levels, 10 is best
|
|
* possible compression (not zlib compatible, and may be very slow),
|
|
* MZ_DEFAULT_COMPRESSION=MZ_DEFAULT_LEVEL. */
|
|
enum {
|
|
MZ_NO_COMPRESSION = 0,
|
|
MZ_BEST_SPEED = 1,
|
|
MZ_BEST_COMPRESSION = 9,
|
|
MZ_UBER_COMPRESSION = 10,
|
|
MZ_DEFAULT_LEVEL = 6,
|
|
MZ_DEFAULT_COMPRESSION = -1
|
|
};
|
|
|
|
#define MZ_VERSION "11.3.0"
|
|
#define MZ_VERNUM 0xB300
|
|
#define MZ_VER_MAJOR 11
|
|
#define MZ_VER_MINOR 3
|
|
#define MZ_VER_REVISION 0
|
|
#define MZ_VER_SUBREVISION 0
|
|
|
|
#ifndef MINIZ_NO_ZLIB_APIS
|
|
|
|
/* Flush values. For typical usage you only need MZ_NO_FLUSH and MZ_FINISH. The
|
|
* other values are for advanced use (refer to the zlib docs). */
|
|
enum { MZ_NO_FLUSH = 0, MZ_PARTIAL_FLUSH = 1, MZ_SYNC_FLUSH = 2, MZ_FULL_FLUSH = 3, MZ_FINISH = 4, MZ_BLOCK = 5 };
|
|
|
|
/* Return status codes. MZ_PARAM_ERROR is non-standard. */
|
|
enum {
|
|
MZ_OK = 0,
|
|
MZ_STREAM_END = 1,
|
|
MZ_NEED_DICT = 2,
|
|
MZ_ERRNO = -1,
|
|
MZ_STREAM_ERROR = -2,
|
|
MZ_DATA_ERROR = -3,
|
|
MZ_MEM_ERROR = -4,
|
|
MZ_BUF_ERROR = -5,
|
|
MZ_VERSION_ERROR = -6,
|
|
MZ_PARAM_ERROR = -10000
|
|
};
|
|
|
|
/* Window bits */
|
|
#define MZ_DEFAULT_WINDOW_BITS 15
|
|
|
|
struct mz_internal_state;
|
|
|
|
/* Compression/decompression stream struct. */
|
|
typedef struct mz_stream_s {
|
|
const unsigned char* next_in; /* pointer to next byte to read */
|
|
unsigned int avail_in; /* number of bytes available at next_in */
|
|
mz_ulong total_in; /* total number of bytes consumed so far */
|
|
|
|
unsigned char* next_out; /* pointer to next byte to write */
|
|
unsigned int avail_out; /* number of bytes that can be written to next_out */
|
|
mz_ulong total_out; /* total number of bytes produced so far */
|
|
|
|
char* msg; /* error msg (unused) */
|
|
struct mz_internal_state* state; /* internal state, allocated by zalloc/zfree */
|
|
|
|
mz_alloc_func zalloc; /* optional heap allocation function (defaults to malloc) */
|
|
mz_free_func zfree; /* optional heap free function (defaults to free) */
|
|
void* opaque; /* heap alloc function user pointer */
|
|
|
|
int data_type; /* data_type (unused) */
|
|
mz_ulong adler; /* adler32 of the source or uncompressed data */
|
|
mz_ulong reserved; /* not used */
|
|
} mz_stream;
|
|
|
|
typedef mz_stream* mz_streamp;
|
|
|
|
/* Returns the version string of miniz.c. */
|
|
MINIZ_EXPORT const char* mz_version(void);
|
|
|
|
#ifndef MINIZ_NO_DEFLATE_APIS
|
|
|
|
/* mz_deflateInit() initializes a compressor with default options: */
|
|
/* Parameters: */
|
|
/* pStream must point to an initialized mz_stream struct. */
|
|
/* level must be between [MZ_NO_COMPRESSION, MZ_BEST_COMPRESSION]. */
|
|
/* level 1 enables a specially optimized compression function that's been
|
|
* optimized purely for performance, not ratio. */
|
|
/* (This special func. is currently only enabled when
|
|
* MINIZ_USE_UNALIGNED_LOADS_AND_STORES and MINIZ_LITTLE_ENDIAN are defined.) */
|
|
/* Return values: */
|
|
/* MZ_OK on success. */
|
|
/* MZ_STREAM_ERROR if the stream is bogus. */
|
|
/* MZ_PARAM_ERROR if the input parameters are bogus. */
|
|
/* MZ_MEM_ERROR on out of memory. */
|
|
MINIZ_EXPORT int mz_deflateInit(mz_streamp pStream, int level);
|
|
|
|
/* mz_deflateInit2() is like mz_deflate(), except with more control: */
|
|
/* Additional parameters: */
|
|
/* method must be MZ_DEFLATED */
|
|
/* window_bits must be MZ_DEFAULT_WINDOW_BITS (to wrap the deflate stream with
|
|
* zlib header/adler-32 footer) or -MZ_DEFAULT_WINDOW_BITS (raw deflate/no
|
|
* header or footer) */
|
|
/* mem_level must be between [1, 9] (it's checked but ignored by miniz.c) */
|
|
MINIZ_EXPORT int mz_deflateInit2(mz_streamp pStream, int level, int method, int window_bits, int mem_level,
|
|
int strategy);
|
|
|
|
/* Quickly resets a compressor without having to reallocate anything. Same as
|
|
* calling mz_deflateEnd() followed by mz_deflateInit()/mz_deflateInit2(). */
|
|
MINIZ_EXPORT int mz_deflateReset(mz_streamp pStream);
|
|
|
|
/* mz_deflate() compresses the input to output, consuming as much of the input
|
|
* and producing as much output as possible. */
|
|
/* Parameters: */
|
|
/* pStream is the stream to read from and write to. You must initialize/update
|
|
* the next_in, avail_in, next_out, and avail_out members. */
|
|
/* flush may be MZ_NO_FLUSH, MZ_PARTIAL_FLUSH/MZ_SYNC_FLUSH, MZ_FULL_FLUSH, or
|
|
* MZ_FINISH. */
|
|
/* Return values: */
|
|
/* MZ_OK on success (when flushing, or if more input is needed but not
|
|
* available, and/or there's more output to be written but the output buffer is
|
|
* full). */
|
|
/* MZ_STREAM_END if all input has been consumed and all output bytes have been
|
|
* written. Don't call mz_deflate() on the stream anymore. */
|
|
/* MZ_STREAM_ERROR if the stream is bogus. */
|
|
/* MZ_PARAM_ERROR if one of the parameters is invalid. */
|
|
/* MZ_BUF_ERROR if no forward progress is possible because the input and/or
|
|
* output buffers are empty. (Fill up the input buffer or free up some output
|
|
* space and try again.) */
|
|
MINIZ_EXPORT int mz_deflate(mz_streamp pStream, int flush);
|
|
|
|
/* mz_deflateEnd() deinitializes a compressor: */
|
|
/* Return values: */
|
|
/* MZ_OK on success. */
|
|
/* MZ_STREAM_ERROR if the stream is bogus. */
|
|
MINIZ_EXPORT int mz_deflateEnd(mz_streamp pStream);
|
|
|
|
/* mz_deflateBound() returns a (very) conservative upper bound on the amount of
|
|
* data that could be generated by deflate(), assuming flush is set to only
|
|
* MZ_NO_FLUSH or MZ_FINISH. */
|
|
MINIZ_EXPORT mz_ulong mz_deflateBound(mz_streamp pStream, mz_ulong source_len);
|
|
|
|
/* Single-call compression functions mz_compress() and mz_compress2(): */
|
|
/* Returns MZ_OK on success, or one of the error codes from mz_deflate() on
|
|
* failure. */
|
|
MINIZ_EXPORT int mz_compress(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource,
|
|
mz_ulong source_len);
|
|
MINIZ_EXPORT int mz_compress2(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource,
|
|
mz_ulong source_len, int level);
|
|
|
|
/* mz_compressBound() returns a (very) conservative upper bound on the amount of
|
|
* data that could be generated by calling mz_compress(). */
|
|
MINIZ_EXPORT mz_ulong mz_compressBound(mz_ulong source_len);
|
|
|
|
#endif /*#ifndef MINIZ_NO_DEFLATE_APIS*/
|
|
|
|
#ifndef MINIZ_NO_INFLATE_APIS
|
|
|
|
/* Initializes a decompressor. */
|
|
MINIZ_EXPORT int mz_inflateInit(mz_streamp pStream);
|
|
|
|
/* mz_inflateInit2() is like mz_inflateInit() with an additional option that
|
|
* controls the window size and whether or not the stream has been wrapped with
|
|
* a zlib header/footer: */
|
|
/* window_bits must be MZ_DEFAULT_WINDOW_BITS (to parse zlib header/footer) or
|
|
* -MZ_DEFAULT_WINDOW_BITS (raw deflate). */
|
|
MINIZ_EXPORT int mz_inflateInit2(mz_streamp pStream, int window_bits);
|
|
|
|
/* Quickly resets a compressor without having to reallocate anything. Same as
|
|
* calling mz_inflateEnd() followed by mz_inflateInit()/mz_inflateInit2(). */
|
|
MINIZ_EXPORT int mz_inflateReset(mz_streamp pStream);
|
|
|
|
/* Decompresses the input stream to the output, consuming only as much of the
|
|
* input as needed, and writing as much to the output as possible. */
|
|
/* Parameters: */
|
|
/* pStream is the stream to read from and write to. You must initialize/update
|
|
* the next_in, avail_in, next_out, and avail_out members. */
|
|
/* flush may be MZ_NO_FLUSH, MZ_SYNC_FLUSH, or MZ_FINISH. */
|
|
/* On the first call, if flush is MZ_FINISH it's assumed the input and output
|
|
* buffers are both sized large enough to decompress the entire stream in a
|
|
* single call (this is slightly faster). */
|
|
/* MZ_FINISH implies that there are no more source bytes available beside
|
|
* what's already in the input buffer, and that the output buffer is large
|
|
* enough to hold the rest of the decompressed data. */
|
|
/* Return values: */
|
|
/* MZ_OK on success. Either more input is needed but not available, and/or
|
|
* there's more output to be written but the output buffer is full. */
|
|
/* MZ_STREAM_END if all needed input has been consumed and all output bytes
|
|
* have been written. For zlib streams, the adler-32 of the decompressed data
|
|
* has also been verified. */
|
|
/* MZ_STREAM_ERROR if the stream is bogus. */
|
|
/* MZ_DATA_ERROR if the deflate stream is invalid. */
|
|
/* MZ_PARAM_ERROR if one of the parameters is invalid. */
|
|
/* MZ_BUF_ERROR if no forward progress is possible because the input buffer is
|
|
* empty but the inflater needs more input to continue, or if the output buffer
|
|
* is not large enough. Call mz_inflate() again */
|
|
/* with more input data, or with more room in the output buffer (except when
|
|
* using single call decompression, described above). */
|
|
MINIZ_EXPORT int mz_inflate(mz_streamp pStream, int flush);
|
|
|
|
/* Deinitializes a decompressor. */
|
|
MINIZ_EXPORT int mz_inflateEnd(mz_streamp pStream);
|
|
|
|
/* Single-call decompression. */
|
|
/* Returns MZ_OK on success, or one of the error codes from mz_inflate() on
|
|
* failure. */
|
|
MINIZ_EXPORT int mz_uncompress(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource,
|
|
mz_ulong source_len);
|
|
MINIZ_EXPORT int mz_uncompress2(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource,
|
|
mz_ulong* pSource_len);
|
|
#endif /*#ifndef MINIZ_NO_INFLATE_APIS*/
|
|
|
|
/* Returns a string description of the specified error code, or NULL if the
|
|
* error code is invalid. */
|
|
MINIZ_EXPORT const char* mz_error(int err);
|
|
|
|
/* Redefine zlib-compatible names to miniz equivalents, so miniz.c can be used
|
|
* as a drop-in replacement for the subset of zlib that miniz.c supports. */
|
|
/* Define MINIZ_NO_ZLIB_COMPATIBLE_NAMES to disable zlib-compatibility if you
|
|
* use zlib in the same project. */
|
|
#ifndef MINIZ_NO_ZLIB_COMPATIBLE_NAMES
|
|
typedef unsigned char Byte;
|
|
typedef unsigned int uInt;
|
|
typedef mz_ulong uLong;
|
|
typedef Byte Bytef;
|
|
typedef uInt uIntf;
|
|
typedef char charf;
|
|
typedef int intf;
|
|
typedef void* voidpf;
|
|
typedef uLong uLongf;
|
|
typedef void* voidp;
|
|
typedef void* const voidpc;
|
|
#define Z_NULL 0
|
|
#define Z_NO_FLUSH MZ_NO_FLUSH
|
|
#define Z_PARTIAL_FLUSH MZ_PARTIAL_FLUSH
|
|
#define Z_SYNC_FLUSH MZ_SYNC_FLUSH
|
|
#define Z_FULL_FLUSH MZ_FULL_FLUSH
|
|
#define Z_FINISH MZ_FINISH
|
|
#define Z_BLOCK MZ_BLOCK
|
|
#define Z_OK MZ_OK
|
|
#define Z_STREAM_END MZ_STREAM_END
|
|
#define Z_NEED_DICT MZ_NEED_DICT
|
|
#define Z_ERRNO MZ_ERRNO
|
|
#define Z_STREAM_ERROR MZ_STREAM_ERROR
|
|
#define Z_DATA_ERROR MZ_DATA_ERROR
|
|
#define Z_MEM_ERROR MZ_MEM_ERROR
|
|
#define Z_BUF_ERROR MZ_BUF_ERROR
|
|
#define Z_VERSION_ERROR MZ_VERSION_ERROR
|
|
#define Z_PARAM_ERROR MZ_PARAM_ERROR
|
|
#define Z_NO_COMPRESSION MZ_NO_COMPRESSION
|
|
#define Z_BEST_SPEED MZ_BEST_SPEED
|
|
#define Z_BEST_COMPRESSION MZ_BEST_COMPRESSION
|
|
#define Z_DEFAULT_COMPRESSION MZ_DEFAULT_COMPRESSION
|
|
#define Z_DEFAULT_STRATEGY MZ_DEFAULT_STRATEGY
|
|
#define Z_FILTERED MZ_FILTERED
|
|
#define Z_HUFFMAN_ONLY MZ_HUFFMAN_ONLY
|
|
#define Z_RLE MZ_RLE
|
|
#define Z_FIXED MZ_FIXED
|
|
#define Z_DEFLATED MZ_DEFLATED
|
|
#define Z_DEFAULT_WINDOW_BITS MZ_DEFAULT_WINDOW_BITS
|
|
/* See mz_alloc_func */
|
|
typedef void* (*alloc_func)(void* opaque, size_t items, size_t size);
|
|
/* See mz_free_func */
|
|
typedef void (*free_func)(void* opaque, void* address);
|
|
|
|
#define internal_state mz_internal_state
|
|
#define z_stream mz_stream
|
|
|
|
#ifndef MINIZ_NO_DEFLATE_APIS
|
|
/* Compatiblity with zlib API. See called functions for documentation */
|
|
static int deflateInit(mz_streamp pStream, int level) { return mz_deflateInit(pStream, level); }
|
|
static int deflateInit2(mz_streamp pStream, int level, int method, int window_bits, int mem_level, int strategy) {
|
|
return mz_deflateInit2(pStream, level, method, window_bits, mem_level, strategy);
|
|
}
|
|
static int deflateReset(mz_streamp pStream) { return mz_deflateReset(pStream); }
|
|
static int deflate(mz_streamp pStream, int flush) { return mz_deflate(pStream, flush); }
|
|
static int deflateEnd(mz_streamp pStream) { return mz_deflateEnd(pStream); }
|
|
static mz_ulong deflateBound(mz_streamp pStream, mz_ulong source_len) { return mz_deflateBound(pStream, source_len); }
|
|
static int compress(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource, mz_ulong source_len) {
|
|
return mz_compress(pDest, pDest_len, pSource, source_len);
|
|
}
|
|
static int compress2(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource, mz_ulong source_len,
|
|
int level) {
|
|
return mz_compress2(pDest, pDest_len, pSource, source_len, level);
|
|
}
|
|
static mz_ulong compressBound(mz_ulong source_len) { return mz_compressBound(source_len); }
|
|
#endif /*#ifndef MINIZ_NO_DEFLATE_APIS*/
|
|
|
|
#ifndef MINIZ_NO_INFLATE_APIS
|
|
/* Compatiblity with zlib API. See called functions for documentation */
|
|
static int inflateInit(mz_streamp pStream) { return mz_inflateInit(pStream); }
|
|
|
|
static int inflateInit2(mz_streamp pStream, int window_bits) { return mz_inflateInit2(pStream, window_bits); }
|
|
|
|
static int inflateReset(mz_streamp pStream) { return mz_inflateReset(pStream); }
|
|
|
|
static int inflate(mz_streamp pStream, int flush) { return mz_inflate(pStream, flush); }
|
|
|
|
static int inflateEnd(mz_streamp pStream) { return mz_inflateEnd(pStream); }
|
|
|
|
static int uncompress(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource, mz_ulong source_len) {
|
|
return mz_uncompress(pDest, pDest_len, pSource, source_len);
|
|
}
|
|
|
|
static int uncompress2(unsigned char* pDest, mz_ulong* pDest_len, const unsigned char* pSource, mz_ulong* pSource_len) {
|
|
return mz_uncompress2(pDest, pDest_len, pSource, pSource_len);
|
|
}
|
|
#endif /*#ifndef MINIZ_NO_INFLATE_APIS*/
|
|
|
|
static mz_ulong crc32(mz_ulong crc, const unsigned char* ptr, size_t buf_len) { return mz_crc32(crc, ptr, buf_len); }
|
|
|
|
static mz_ulong adler32(mz_ulong adler, const unsigned char* ptr, size_t buf_len) {
|
|
return mz_adler32(adler, ptr, buf_len);
|
|
}
|
|
|
|
#define MAX_WBITS 15
|
|
#define MAX_MEM_LEVEL 9
|
|
|
|
static const char* zError(int err) { return mz_error(err); }
|
|
#define ZLIB_VERSION MZ_VERSION
|
|
#define ZLIB_VERNUM MZ_VERNUM
|
|
#define ZLIB_VER_MAJOR MZ_VER_MAJOR
|
|
#define ZLIB_VER_MINOR MZ_VER_MINOR
|
|
#define ZLIB_VER_REVISION MZ_VER_REVISION
|
|
#define ZLIB_VER_SUBREVISION MZ_VER_SUBREVISION
|
|
|
|
#define zlibVersion mz_version
|
|
#define zlib_version mz_version()
|
|
#endif /* #ifndef MINIZ_NO_ZLIB_COMPATIBLE_NAMES */
|
|
|
|
#endif /* MINIZ_NO_ZLIB_APIS */
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#pragma once
|
|
#include <assert.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
/* ------------------- Types and macros */
|
|
typedef unsigned char mz_uint8;
|
|
typedef int16_t mz_int16;
|
|
typedef uint16_t mz_uint16;
|
|
typedef uint32_t mz_uint32;
|
|
typedef uint32_t mz_uint;
|
|
typedef int64_t mz_int64;
|
|
typedef uint64_t mz_uint64;
|
|
typedef int mz_bool;
|
|
|
|
#define MZ_FALSE (0)
|
|
#define MZ_TRUE (1)
|
|
|
|
/* Works around MSVC's spammy "warning C4127: conditional expression is
|
|
* constant" message. */
|
|
#ifdef _MSC_VER
|
|
#define MZ_MACRO_END while (0, 0)
|
|
#else
|
|
#define MZ_MACRO_END while (0)
|
|
#endif
|
|
|
|
#ifdef MINIZ_NO_STDIO
|
|
#define MZ_FILE void*
|
|
#else
|
|
#include <stdio.h>
|
|
#define MZ_FILE FILE
|
|
#endif /* #ifdef MINIZ_NO_STDIO */
|
|
|
|
#ifdef MINIZ_NO_TIME
|
|
typedef struct mz_dummy_time_t_tag {
|
|
mz_uint32 m_dummy1;
|
|
mz_uint32 m_dummy2;
|
|
} mz_dummy_time_t;
|
|
#define MZ_TIME_T mz_dummy_time_t
|
|
#else
|
|
#define MZ_TIME_T time_t
|
|
#endif
|
|
|
|
#define MZ_ASSERT(x) assert(x)
|
|
|
|
#ifdef MINIZ_NO_MALLOC
|
|
#define MZ_MALLOC(x) NULL
|
|
#define MZ_FREE(x) (void)x, ((void)0)
|
|
#define MZ_REALLOC(p, x) NULL
|
|
#else
|
|
#define MZ_MALLOC(x) malloc(x)
|
|
#define MZ_FREE(x) free(x)
|
|
#define MZ_REALLOC(p, x) realloc(p, x)
|
|
#endif
|
|
|
|
#define MZ_MAX(a, b) (((a) > (b)) ? (a) : (b))
|
|
#define MZ_MIN(a, b) (((a) < (b)) ? (a) : (b))
|
|
#define MZ_CLEAR_OBJ(obj) memset(&(obj), 0, sizeof(obj))
|
|
#define MZ_CLEAR_ARR(obj) memset((obj), 0, sizeof(obj))
|
|
#define MZ_CLEAR_PTR(obj) memset((obj), 0, sizeof(*obj))
|
|
|
|
#if MINIZ_USE_UNALIGNED_LOADS_AND_STORES && MINIZ_LITTLE_ENDIAN
|
|
#define MZ_READ_LE16(p) *((const mz_uint16*)(p))
|
|
#define MZ_READ_LE32(p) *((const mz_uint32*)(p))
|
|
#else
|
|
#define MZ_READ_LE16(p) ((mz_uint32)(((const mz_uint8*)(p))[0]) | ((mz_uint32)(((const mz_uint8*)(p))[1]) << 8U))
|
|
#define MZ_READ_LE32(p) \
|
|
((mz_uint32)(((const mz_uint8*)(p))[0]) | ((mz_uint32)(((const mz_uint8*)(p))[1]) << 8U) | \
|
|
((mz_uint32)(((const mz_uint8*)(p))[2]) << 16U) | ((mz_uint32)(((const mz_uint8*)(p))[3]) << 24U))
|
|
#endif
|
|
|
|
#define MZ_READ_LE64(p) \
|
|
(((mz_uint64)MZ_READ_LE32(p)) | (((mz_uint64)MZ_READ_LE32((const mz_uint8*)(p) + sizeof(mz_uint32))) << 32U))
|
|
|
|
#ifdef _MSC_VER
|
|
#define MZ_FORCEINLINE __forceinline
|
|
#elif defined(__GNUC__)
|
|
#define MZ_FORCEINLINE __inline__ __attribute__((__always_inline__))
|
|
#else
|
|
#define MZ_FORCEINLINE inline
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
extern MINIZ_EXPORT void* miniz_def_alloc_func(void* opaque, size_t items, size_t size);
|
|
extern MINIZ_EXPORT void miniz_def_free_func(void* opaque, void* address);
|
|
extern MINIZ_EXPORT void* miniz_def_realloc_func(void* opaque, void* address, size_t items, size_t size);
|
|
|
|
#define MZ_UINT16_MAX (0xFFFFU)
|
|
#define MZ_UINT32_MAX (0xFFFFFFFFU)
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
#pragma once
|
|
|
|
#ifndef MINIZ_NO_DEFLATE_APIS
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
/* ------------------- Low-level Compression API Definitions */
|
|
|
|
/* Set TDEFL_LESS_MEMORY to 1 to use less memory (compression will be slightly
|
|
* slower, and raw/dynamic blocks will be output more frequently). */
|
|
#ifndef TDEFL_LESS_MEMORY
|
|
#define TDEFL_LESS_MEMORY 0
|
|
#endif
|
|
|
|
/* tdefl_init() compression flags logically OR'd together (low 12 bits contain
|
|
* the max. number of probes per dictionary search): */
|
|
/* TDEFL_DEFAULT_MAX_PROBES: The compressor defaults to 128 dictionary probes
|
|
* per dictionary search. 0=Huffman only, 1=Huffman+LZ (fastest/crap
|
|
* compression), 4095=Huffman+LZ (slowest/best compression). */
|
|
enum { TDEFL_HUFFMAN_ONLY = 0, TDEFL_DEFAULT_MAX_PROBES = 128, TDEFL_MAX_PROBES_MASK = 0xFFF };
|
|
|
|
/* TDEFL_WRITE_ZLIB_HEADER: If set, the compressor outputs a zlib header before
|
|
* the deflate data, and the Adler-32 of the source data at the end. Otherwise,
|
|
* you'll get raw deflate data. */
|
|
/* TDEFL_COMPUTE_ADLER32: Always compute the adler-32 of the input data (even
|
|
* when not writing zlib headers). */
|
|
/* TDEFL_GREEDY_PARSING_FLAG: Set to use faster greedy parsing, instead of more
|
|
* efficient lazy parsing. */
|
|
/* TDEFL_NONDETERMINISTIC_PARSING_FLAG: Enable to decrease the compressor's
|
|
* initialization time to the minimum, but the output may vary from run to run
|
|
* given the same input (depending on the contents of memory). */
|
|
/* TDEFL_RLE_MATCHES: Only look for RLE matches (matches with a distance of 1)
|
|
*/
|
|
/* TDEFL_FILTER_MATCHES: Discards matches <= 5 chars if enabled. */
|
|
/* TDEFL_FORCE_ALL_STATIC_BLOCKS: Disable usage of optimized Huffman tables. */
|
|
/* TDEFL_FORCE_ALL_RAW_BLOCKS: Only use raw (uncompressed) deflate blocks. */
|
|
/* The low 12 bits are reserved to control the max # of hash probes per
|
|
* dictionary lookup (see TDEFL_MAX_PROBES_MASK). */
|
|
enum {
|
|
TDEFL_WRITE_ZLIB_HEADER = 0x01000,
|
|
TDEFL_COMPUTE_ADLER32 = 0x02000,
|
|
TDEFL_GREEDY_PARSING_FLAG = 0x04000,
|
|
TDEFL_NONDETERMINISTIC_PARSING_FLAG = 0x08000,
|
|
TDEFL_RLE_MATCHES = 0x10000,
|
|
TDEFL_FILTER_MATCHES = 0x20000,
|
|
TDEFL_FORCE_ALL_STATIC_BLOCKS = 0x40000,
|
|
TDEFL_FORCE_ALL_RAW_BLOCKS = 0x80000
|
|
};
|
|
|
|
/* High level compression functions: */
|
|
/* tdefl_compress_mem_to_heap() compresses a block in memory to a heap block
|
|
* allocated via malloc(). */
|
|
/* On entry: */
|
|
/* pSrc_buf, src_buf_len: Pointer and size of source block to compress. */
|
|
/* flags: The max match finder probes (default is 128) logically OR'd against
|
|
* the above flags. Higher probes are slower but improve compression. */
|
|
/* On return: */
|
|
/* Function returns a pointer to the compressed data, or NULL on failure. */
|
|
/* *pOut_len will be set to the compressed data's size, which could be larger
|
|
* than src_buf_len on uncompressible data. */
|
|
/* The caller must free() the returned block when it's no longer needed. */
|
|
MINIZ_EXPORT void* tdefl_compress_mem_to_heap(const void* pSrc_buf, size_t src_buf_len, size_t* pOut_len, int flags);
|
|
|
|
/* tdefl_compress_mem_to_mem() compresses a block in memory to another block in
|
|
* memory. */
|
|
/* Returns 0 on failure. */
|
|
MINIZ_EXPORT size_t tdefl_compress_mem_to_mem(void* pOut_buf, size_t out_buf_len, const void* pSrc_buf,
|
|
size_t src_buf_len, int flags);
|
|
|
|
/* Compresses an image to a compressed PNG file in memory. */
|
|
/* On entry: */
|
|
/* pImage, w, h, and num_chans describe the image to compress. num_chans may be
|
|
* 1, 2, 3, or 4. */
|
|
/* The image pitch in bytes per scanline will be w*num_chans. The leftmost
|
|
* pixel on the top scanline is stored first in memory. */
|
|
/* level may range from [0,10], use MZ_NO_COMPRESSION, MZ_BEST_SPEED,
|
|
* MZ_BEST_COMPRESSION, etc. or a decent default is MZ_DEFAULT_LEVEL */
|
|
/* If flip is true, the image will be flipped on the Y axis (useful for OpenGL
|
|
* apps). */
|
|
/* On return: */
|
|
/* Function returns a pointer to the compressed data, or NULL on failure. */
|
|
/* *pLen_out will be set to the size of the PNG image file. */
|
|
/* The caller must mz_free() the returned heap block (which will typically be
|
|
* larger than *pLen_out) when it's no longer needed. */
|
|
MINIZ_EXPORT void* tdefl_write_image_to_png_file_in_memory_ex(const void* pImage, int w, int h, int num_chans,
|
|
size_t* pLen_out, mz_uint level, mz_bool flip);
|
|
MINIZ_EXPORT void* tdefl_write_image_to_png_file_in_memory(const void* pImage, int w, int h, int num_chans,
|
|
size_t* pLen_out);
|
|
|
|
/* Output stream interface. The compressor uses this interface to write
|
|
* compressed data. It'll typically be called TDEFL_OUT_BUF_SIZE at a time. */
|
|
typedef mz_bool (*tdefl_put_buf_func_ptr)(const void* pBuf, int len, void* pUser);
|
|
|
|
/* tdefl_compress_mem_to_output() compresses a block to an output stream. The
|
|
* above helpers use this function internally. */
|
|
MINIZ_EXPORT mz_bool tdefl_compress_mem_to_output(const void* pBuf, size_t buf_len,
|
|
tdefl_put_buf_func_ptr pPut_buf_func, void* pPut_buf_user, int flags);
|
|
|
|
enum {
|
|
TDEFL_MAX_HUFF_TABLES = 3,
|
|
TDEFL_MAX_HUFF_SYMBOLS_0 = 288,
|
|
TDEFL_MAX_HUFF_SYMBOLS_1 = 32,
|
|
TDEFL_MAX_HUFF_SYMBOLS_2 = 19,
|
|
TDEFL_LZ_DICT_SIZE = 32768,
|
|
TDEFL_LZ_DICT_SIZE_MASK = TDEFL_LZ_DICT_SIZE - 1,
|
|
TDEFL_MIN_MATCH_LEN = 3,
|
|
TDEFL_MAX_MATCH_LEN = 258
|
|
};
|
|
|
|
/* TDEFL_OUT_BUF_SIZE MUST be large enough to hold a single entire compressed
|
|
* output block (using static/fixed Huffman codes). */
|
|
#if TDEFL_LESS_MEMORY
|
|
enum {
|
|
TDEFL_LZ_CODE_BUF_SIZE = 24 * 1024,
|
|
TDEFL_OUT_BUF_SIZE = (TDEFL_LZ_CODE_BUF_SIZE * 13) / 10,
|
|
TDEFL_MAX_HUFF_SYMBOLS = 288,
|
|
TDEFL_LZ_HASH_BITS = 12,
|
|
TDEFL_LEVEL1_HASH_SIZE_MASK = 4095,
|
|
TDEFL_LZ_HASH_SHIFT = (TDEFL_LZ_HASH_BITS + 2) / 3,
|
|
TDEFL_LZ_HASH_SIZE = 1 << TDEFL_LZ_HASH_BITS
|
|
};
|
|
#else
|
|
enum {
|
|
TDEFL_LZ_CODE_BUF_SIZE = 64 * 1024,
|
|
TDEFL_OUT_BUF_SIZE = (mz_uint)((TDEFL_LZ_CODE_BUF_SIZE * 13) / 10),
|
|
TDEFL_MAX_HUFF_SYMBOLS = 288,
|
|
TDEFL_LZ_HASH_BITS = 15,
|
|
TDEFL_LEVEL1_HASH_SIZE_MASK = 4095,
|
|
TDEFL_LZ_HASH_SHIFT = (TDEFL_LZ_HASH_BITS + 2) / 3,
|
|
TDEFL_LZ_HASH_SIZE = 1 << TDEFL_LZ_HASH_BITS
|
|
};
|
|
#endif
|
|
|
|
/* The low-level tdefl functions below may be used directly if the above helper
|
|
* functions aren't flexible enough. The low-level functions don't make any heap
|
|
* allocations, unlike the above helper functions. */
|
|
typedef enum {
|
|
TDEFL_STATUS_BAD_PARAM = -2,
|
|
TDEFL_STATUS_PUT_BUF_FAILED = -1,
|
|
TDEFL_STATUS_OKAY = 0,
|
|
TDEFL_STATUS_DONE = 1
|
|
} tdefl_status;
|
|
|
|
/* Must map to MZ_NO_FLUSH, MZ_SYNC_FLUSH, etc. enums */
|
|
typedef enum { TDEFL_NO_FLUSH = 0, TDEFL_SYNC_FLUSH = 2, TDEFL_FULL_FLUSH = 3, TDEFL_FINISH = 4 } tdefl_flush;
|
|
|
|
/* tdefl's compression state structure. */
|
|
typedef struct {
|
|
tdefl_put_buf_func_ptr m_pPut_buf_func;
|
|
void* m_pPut_buf_user;
|
|
mz_uint m_flags, m_max_probes[2];
|
|
int m_greedy_parsing;
|
|
mz_uint m_adler32, m_lookahead_pos, m_lookahead_size, m_dict_size;
|
|
mz_uint8 *m_pLZ_code_buf, *m_pLZ_flags, *m_pOutput_buf, *m_pOutput_buf_end;
|
|
mz_uint m_num_flags_left, m_total_lz_bytes, m_lz_code_buf_dict_pos, m_bits_in, m_bit_buffer;
|
|
mz_uint m_saved_match_dist, m_saved_match_len, m_saved_lit, m_output_flush_ofs, m_output_flush_remaining, m_finished,
|
|
m_block_index, m_wants_to_finish;
|
|
tdefl_status m_prev_return_status;
|
|
const void* m_pIn_buf;
|
|
void* m_pOut_buf;
|
|
size_t *m_pIn_buf_size, *m_pOut_buf_size;
|
|
tdefl_flush m_flush;
|
|
const mz_uint8* m_pSrc;
|
|
size_t m_src_buf_left, m_out_buf_ofs;
|
|
mz_uint8 m_dict[TDEFL_LZ_DICT_SIZE + TDEFL_MAX_MATCH_LEN - 1];
|
|
mz_uint16 m_huff_count[TDEFL_MAX_HUFF_TABLES][TDEFL_MAX_HUFF_SYMBOLS];
|
|
mz_uint16 m_huff_codes[TDEFL_MAX_HUFF_TABLES][TDEFL_MAX_HUFF_SYMBOLS];
|
|
mz_uint8 m_huff_code_sizes[TDEFL_MAX_HUFF_TABLES][TDEFL_MAX_HUFF_SYMBOLS];
|
|
mz_uint8 m_lz_code_buf[TDEFL_LZ_CODE_BUF_SIZE];
|
|
mz_uint16 m_next[TDEFL_LZ_DICT_SIZE];
|
|
mz_uint16 m_hash[TDEFL_LZ_HASH_SIZE];
|
|
mz_uint8 m_output_buf[TDEFL_OUT_BUF_SIZE];
|
|
} tdefl_compressor;
|
|
|
|
/* Initializes the compressor. */
|
|
/* There is no corresponding deinit() function because the tdefl API's do not
|
|
* dynamically allocate memory. */
|
|
/* pBut_buf_func: If NULL, output data will be supplied to the specified
|
|
* callback. In this case, the user should call the tdefl_compress_buffer() API
|
|
* for compression. */
|
|
/* If pBut_buf_func is NULL the user should always call the tdefl_compress()
|
|
* API. */
|
|
/* flags: See the above enums (TDEFL_HUFFMAN_ONLY, TDEFL_WRITE_ZLIB_HEADER,
|
|
* etc.) */
|
|
MINIZ_EXPORT tdefl_status tdefl_init(tdefl_compressor* d, tdefl_put_buf_func_ptr pPut_buf_func, void* pPut_buf_user,
|
|
int flags);
|
|
|
|
/* Compresses a block of data, consuming as much of the specified input buffer
|
|
* as possible, and writing as much compressed data to the specified output
|
|
* buffer as possible. */
|
|
MINIZ_EXPORT tdefl_status tdefl_compress(tdefl_compressor* d, const void* pIn_buf, size_t* pIn_buf_size, void* pOut_buf,
|
|
size_t* pOut_buf_size, tdefl_flush flush);
|
|
|
|
/* tdefl_compress_buffer() is only usable when the tdefl_init() is called with a
|
|
* non-NULL tdefl_put_buf_func_ptr. */
|
|
/* tdefl_compress_buffer() always consumes the entire input buffer. */
|
|
MINIZ_EXPORT tdefl_status tdefl_compress_buffer(tdefl_compressor* d, const void* pIn_buf, size_t in_buf_size,
|
|
tdefl_flush flush);
|
|
|
|
MINIZ_EXPORT tdefl_status tdefl_get_prev_return_status(tdefl_compressor* d);
|
|
MINIZ_EXPORT mz_uint32 tdefl_get_adler32(tdefl_compressor* d);
|
|
|
|
/* Create tdefl_compress() flags given zlib-style compression parameters. */
|
|
/* level may range from [0,10] (where 10 is absolute max compression, but may be
|
|
* much slower on some files) */
|
|
/* window_bits may be -15 (raw deflate) or 15 (zlib) */
|
|
/* strategy may be either MZ_DEFAULT_STRATEGY, MZ_FILTERED, MZ_HUFFMAN_ONLY,
|
|
* MZ_RLE, or MZ_FIXED */
|
|
MINIZ_EXPORT mz_uint tdefl_create_comp_flags_from_zip_params(int level, int window_bits, int strategy);
|
|
|
|
#ifndef MINIZ_NO_MALLOC
|
|
/* Allocate the tdefl_compressor structure in C so that */
|
|
/* non-C language bindings to tdefl_ API don't need to worry about */
|
|
/* structure size and allocation mechanism. */
|
|
MINIZ_EXPORT tdefl_compressor* tdefl_compressor_alloc(void);
|
|
MINIZ_EXPORT void tdefl_compressor_free(tdefl_compressor* pComp);
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /*#ifndef MINIZ_NO_DEFLATE_APIS*/
|
|
#pragma once
|
|
|
|
/* ------------------- Low-level Decompression API Definitions */
|
|
|
|
#ifndef MINIZ_NO_INFLATE_APIS
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
/* Decompression flags used by tinfl_decompress(). */
|
|
/* TINFL_FLAG_PARSE_ZLIB_HEADER: If set, the input has a valid zlib header and
|
|
* ends with an adler32 checksum (it's a valid zlib stream). Otherwise, the
|
|
* input is a raw deflate stream. */
|
|
/* TINFL_FLAG_HAS_MORE_INPUT: If set, there are more input bytes available
|
|
* beyond the end of the supplied input buffer. If clear, the input buffer
|
|
* contains all remaining input. */
|
|
/* TINFL_FLAG_USING_NON_WRAPPING_OUTPUT_BUF: If set, the output buffer is large
|
|
* enough to hold the entire decompressed stream. If clear, the output buffer is
|
|
* at least the size of the dictionary (typically 32KB). */
|
|
/* TINFL_FLAG_COMPUTE_ADLER32: Force adler-32 checksum computation of the
|
|
* decompressed bytes. */
|
|
enum {
|
|
TINFL_FLAG_PARSE_ZLIB_HEADER = 1,
|
|
TINFL_FLAG_HAS_MORE_INPUT = 2,
|
|
TINFL_FLAG_USING_NON_WRAPPING_OUTPUT_BUF = 4,
|
|
TINFL_FLAG_COMPUTE_ADLER32 = 8
|
|
};
|
|
|
|
/* High level decompression functions: */
|
|
/* tinfl_decompress_mem_to_heap() decompresses a block in memory to a heap block
|
|
* allocated via malloc(). */
|
|
/* On entry: */
|
|
/* pSrc_buf, src_buf_len: Pointer and size of the Deflate or zlib source data
|
|
* to decompress. */
|
|
/* On return: */
|
|
/* Function returns a pointer to the decompressed data, or NULL on failure. */
|
|
/* *pOut_len will be set to the decompressed data's size, which could be larger
|
|
* than src_buf_len on uncompressible data. */
|
|
/* The caller must call mz_free() on the returned block when it's no longer
|
|
* needed. */
|
|
MINIZ_EXPORT void* tinfl_decompress_mem_to_heap(const void* pSrc_buf, size_t src_buf_len, size_t* pOut_len, int flags);
|
|
|
|
/* tinfl_decompress_mem_to_mem() decompresses a block in memory to another block
|
|
* in memory. */
|
|
/* Returns TINFL_DECOMPRESS_MEM_TO_MEM_FAILED on failure, or the number of bytes
|
|
* written on success. */
|
|
#define TINFL_DECOMPRESS_MEM_TO_MEM_FAILED ((size_t)(-1))
|
|
MINIZ_EXPORT size_t tinfl_decompress_mem_to_mem(void* pOut_buf, size_t out_buf_len, const void* pSrc_buf,
|
|
size_t src_buf_len, int flags);
|
|
|
|
/* tinfl_decompress_mem_to_callback() decompresses a block in memory to an
|
|
* internal 32KB buffer, and a user provided callback function will be called to
|
|
* flush the buffer. */
|
|
/* Returns 1 on success or 0 on failure. */
|
|
typedef int (*tinfl_put_buf_func_ptr)(const void* pBuf, int len, void* pUser);
|
|
MINIZ_EXPORT int tinfl_decompress_mem_to_callback(const void* pIn_buf, size_t* pIn_buf_size,
|
|
tinfl_put_buf_func_ptr pPut_buf_func, void* pPut_buf_user, int flags);
|
|
|
|
struct tinfl_decompressor_tag;
|
|
typedef struct tinfl_decompressor_tag tinfl_decompressor;
|
|
|
|
#ifndef MINIZ_NO_MALLOC
|
|
/* Allocate the tinfl_decompressor structure in C so that */
|
|
/* non-C language bindings to tinfl_ API don't need to worry about */
|
|
/* structure size and allocation mechanism. */
|
|
MINIZ_EXPORT tinfl_decompressor* tinfl_decompressor_alloc(void);
|
|
MINIZ_EXPORT void tinfl_decompressor_free(tinfl_decompressor* pDecomp);
|
|
#endif
|
|
|
|
/* Max size of LZ dictionary. */
|
|
#define TINFL_LZ_DICT_SIZE 32768
|
|
|
|
/* Return status. */
|
|
typedef enum {
|
|
/* This flags indicates the inflator needs 1 or more input bytes to make
|
|
forward progress, but the caller is indicating that no more are available.
|
|
The compressed data */
|
|
/* is probably corrupted. If you call the inflator again with more bytes it'll
|
|
try to continue processing the input but this is a BAD sign (either the
|
|
data is corrupted or you called it incorrectly). */
|
|
/* If you call it again with no input you'll just get
|
|
TINFL_STATUS_FAILED_CANNOT_MAKE_PROGRESS again. */
|
|
TINFL_STATUS_FAILED_CANNOT_MAKE_PROGRESS = -4,
|
|
|
|
/* This flag indicates that one or more of the input parameters was obviously
|
|
bogus. (You can try calling it again, but if you get this error the calling
|
|
code is wrong.) */
|
|
TINFL_STATUS_BAD_PARAM = -3,
|
|
|
|
/* This flags indicate the inflator is finished but the adler32 check of the
|
|
uncompressed data didn't match. If you call it again it'll return
|
|
TINFL_STATUS_DONE. */
|
|
TINFL_STATUS_ADLER32_MISMATCH = -2,
|
|
|
|
/* This flags indicate the inflator has somehow failed (bad code, corrupted
|
|
input, etc.). If you call it again without resetting via tinfl_init() it
|
|
it'll just keep on returning the same status failure code. */
|
|
TINFL_STATUS_FAILED = -1,
|
|
|
|
/* Any status code less than TINFL_STATUS_DONE must indicate a failure. */
|
|
|
|
/* This flag indicates the inflator has returned every byte of uncompressed
|
|
data that it can, has consumed every byte that it needed, has successfully
|
|
reached the end of the deflate stream, and */
|
|
/* if zlib headers and adler32 checking enabled that it has successfully
|
|
checked the uncompressed data's adler32. If you call it again you'll just
|
|
get TINFL_STATUS_DONE over and over again. */
|
|
TINFL_STATUS_DONE = 0,
|
|
|
|
/* This flag indicates the inflator MUST have more input data (even 1 byte)
|
|
before it can make any more forward progress, or you need to clear the
|
|
TINFL_FLAG_HAS_MORE_INPUT */
|
|
/* flag on the next call if you don't have any more source data. If the source
|
|
data was somehow corrupted it's also possible (but unlikely) for the
|
|
inflator to keep on demanding input to */
|
|
/* proceed, so be sure to properly set the TINFL_FLAG_HAS_MORE_INPUT flag. */
|
|
TINFL_STATUS_NEEDS_MORE_INPUT = 1,
|
|
|
|
/* This flag indicates the inflator definitely has 1 or more bytes of
|
|
uncompressed data available, but it cannot write this data into the output
|
|
buffer. */
|
|
/* Note if the source compressed data was corrupted it's possible for the
|
|
inflator to return a lot of uncompressed data to the caller. I've been
|
|
assuming you know how much uncompressed data to expect */
|
|
/* (either exact or worst case) and will stop calling the inflator and fail
|
|
after receiving too much. In pure streaming scenarios where you have no
|
|
idea how many bytes to expect this may not be possible */
|
|
/* so I may need to add some code to address this. */
|
|
TINFL_STATUS_HAS_MORE_OUTPUT = 2
|
|
} tinfl_status;
|
|
|
|
/* Initializes the decompressor to its initial state. */
|
|
#define tinfl_init(r) \
|
|
do { \
|
|
(r)->m_state = 0; \
|
|
} \
|
|
MZ_MACRO_END
|
|
#define tinfl_get_adler32(r) (r)->m_check_adler32
|
|
|
|
/* Main low-level decompressor coroutine function. This is the only function
|
|
* actually needed for decompression. All the other functions are just
|
|
* high-level helpers for improved usability. */
|
|
/* This is a universal API, i.e. it can be used as a building block to build any
|
|
* desired higher level decompression API. In the limit case, it can be called
|
|
* once per every byte input or output. */
|
|
MINIZ_EXPORT tinfl_status tinfl_decompress(tinfl_decompressor* r, const mz_uint8* pIn_buf_next, size_t* pIn_buf_size,
|
|
mz_uint8* pOut_buf_start, mz_uint8* pOut_buf_next, size_t* pOut_buf_size,
|
|
const mz_uint32 decomp_flags);
|
|
|
|
/* Internal/private bits follow. */
|
|
enum {
|
|
TINFL_MAX_HUFF_TABLES = 3,
|
|
TINFL_MAX_HUFF_SYMBOLS_0 = 288,
|
|
TINFL_MAX_HUFF_SYMBOLS_1 = 32,
|
|
TINFL_MAX_HUFF_SYMBOLS_2 = 19,
|
|
TINFL_FAST_LOOKUP_BITS = 10,
|
|
TINFL_FAST_LOOKUP_SIZE = 1 << TINFL_FAST_LOOKUP_BITS
|
|
};
|
|
|
|
#if MINIZ_HAS_64BIT_REGISTERS
|
|
#define TINFL_USE_64BIT_BITBUF 1
|
|
#else
|
|
#define TINFL_USE_64BIT_BITBUF 0
|
|
#endif
|
|
|
|
#if TINFL_USE_64BIT_BITBUF
|
|
typedef mz_uint64 tinfl_bit_buf_t;
|
|
#define TINFL_BITBUF_SIZE (64)
|
|
#else
|
|
typedef mz_uint32 tinfl_bit_buf_t;
|
|
#define TINFL_BITBUF_SIZE (32)
|
|
#endif
|
|
|
|
struct tinfl_decompressor_tag {
|
|
mz_uint32 m_state, m_num_bits, m_zhdr0, m_zhdr1, m_z_adler32, m_final, m_type, m_check_adler32, m_dist, m_counter,
|
|
m_num_extra, m_table_sizes[TINFL_MAX_HUFF_TABLES];
|
|
tinfl_bit_buf_t m_bit_buf;
|
|
size_t m_dist_from_out_buf_start;
|
|
mz_int16 m_look_up[TINFL_MAX_HUFF_TABLES][TINFL_FAST_LOOKUP_SIZE];
|
|
mz_int16 m_tree_0[TINFL_MAX_HUFF_SYMBOLS_0 * 2];
|
|
mz_int16 m_tree_1[TINFL_MAX_HUFF_SYMBOLS_1 * 2];
|
|
mz_int16 m_tree_2[TINFL_MAX_HUFF_SYMBOLS_2 * 2];
|
|
mz_uint8 m_code_size_0[TINFL_MAX_HUFF_SYMBOLS_0];
|
|
mz_uint8 m_code_size_1[TINFL_MAX_HUFF_SYMBOLS_1];
|
|
mz_uint8 m_code_size_2[TINFL_MAX_HUFF_SYMBOLS_2];
|
|
mz_uint8 m_raw_header[4], m_len_codes[TINFL_MAX_HUFF_SYMBOLS_0 + TINFL_MAX_HUFF_SYMBOLS_1 + 137];
|
|
};
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /*#ifndef MINIZ_NO_INFLATE_APIS*/
|
|
|
|
#pragma once
|
|
|
|
/* ------------------- ZIP archive reading/writing */
|
|
|
|
#ifndef MINIZ_NO_ARCHIVE_APIS
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
enum {
|
|
/* Note: These enums can be reduced as needed to save memory or stack space -
|
|
they are pretty conservative. */
|
|
MZ_ZIP_MAX_IO_BUF_SIZE = 64 * 1024,
|
|
MZ_ZIP_MAX_ARCHIVE_FILENAME_SIZE = 512,
|
|
MZ_ZIP_MAX_ARCHIVE_FILE_COMMENT_SIZE = 512
|
|
};
|
|
|
|
typedef struct {
|
|
/* Central directory file index. */
|
|
mz_uint32 m_file_index;
|
|
|
|
/* Byte offset of this entry in the archive's central directory. Note we
|
|
* currently only support up to UINT_MAX or less bytes in the central dir. */
|
|
mz_uint64 m_central_dir_ofs;
|
|
|
|
/* These fields are copied directly from the zip's central dir. */
|
|
mz_uint16 m_version_made_by;
|
|
mz_uint16 m_version_needed;
|
|
mz_uint16 m_bit_flag;
|
|
mz_uint16 m_method;
|
|
|
|
/* CRC-32 of uncompressed data. */
|
|
mz_uint32 m_crc32;
|
|
|
|
/* File's compressed size. */
|
|
mz_uint64 m_comp_size;
|
|
|
|
/* File's uncompressed size. Note, I've seen some old archives where directory
|
|
* entries had 512 bytes for their uncompressed sizes, but when you try to
|
|
* unpack them you actually get 0 bytes. */
|
|
mz_uint64 m_uncomp_size;
|
|
|
|
/* Zip internal and external file attributes. */
|
|
mz_uint16 m_internal_attr;
|
|
mz_uint32 m_external_attr;
|
|
|
|
/* Entry's local header file offset in bytes. */
|
|
mz_uint64 m_local_header_ofs;
|
|
|
|
/* Size of comment in bytes. */
|
|
mz_uint32 m_comment_size;
|
|
|
|
/* MZ_TRUE if the entry appears to be a directory. */
|
|
mz_bool m_is_directory;
|
|
|
|
/* MZ_TRUE if the entry uses encryption/strong encryption (which miniz_zip
|
|
* doesn't support) */
|
|
mz_bool m_is_encrypted;
|
|
|
|
/* MZ_TRUE if the file is not encrypted, a patch file, and if it uses a
|
|
* compression method we support. */
|
|
mz_bool m_is_supported;
|
|
|
|
/* Filename. If string ends in '/' it's a subdirectory entry. */
|
|
/* Guaranteed to be zero terminated, may be truncated to fit. */
|
|
char m_filename[MZ_ZIP_MAX_ARCHIVE_FILENAME_SIZE];
|
|
|
|
/* Comment field. */
|
|
/* Guaranteed to be zero terminated, may be truncated to fit. */
|
|
char m_comment[MZ_ZIP_MAX_ARCHIVE_FILE_COMMENT_SIZE];
|
|
|
|
#ifdef MINIZ_NO_TIME
|
|
MZ_TIME_T m_padding;
|
|
#else
|
|
MZ_TIME_T m_time;
|
|
#endif
|
|
} mz_zip_archive_file_stat;
|
|
|
|
typedef size_t (*mz_file_read_func)(void* pOpaque, mz_uint64 file_ofs, void* pBuf, size_t n);
|
|
typedef size_t (*mz_file_write_func)(void* pOpaque, mz_uint64 file_ofs, const void* pBuf, size_t n);
|
|
typedef mz_bool (*mz_file_needs_keepalive)(void* pOpaque);
|
|
|
|
struct mz_zip_internal_state_tag;
|
|
typedef struct mz_zip_internal_state_tag mz_zip_internal_state;
|
|
|
|
typedef enum {
|
|
MZ_ZIP_MODE_INVALID = 0,
|
|
MZ_ZIP_MODE_READING = 1,
|
|
MZ_ZIP_MODE_WRITING = 2,
|
|
MZ_ZIP_MODE_WRITING_HAS_BEEN_FINALIZED = 3
|
|
} mz_zip_mode;
|
|
|
|
typedef enum {
|
|
MZ_ZIP_FLAG_CASE_SENSITIVE = 0x0100,
|
|
MZ_ZIP_FLAG_IGNORE_PATH = 0x0200,
|
|
MZ_ZIP_FLAG_COMPRESSED_DATA = 0x0400,
|
|
MZ_ZIP_FLAG_DO_NOT_SORT_CENTRAL_DIRECTORY = 0x0800,
|
|
MZ_ZIP_FLAG_VALIDATE_LOCATE_FILE_FLAG = 0x1000, /* if enabled, mz_zip_reader_locate_file() will be called on each
|
|
file as its validated to ensure the func finds the file in the
|
|
central dir (intended for testing) */
|
|
MZ_ZIP_FLAG_VALIDATE_HEADERS_ONLY = 0x2000, /* validate the local headers, but don't decompress the entire
|
|
file and check the crc32 */
|
|
MZ_ZIP_FLAG_WRITE_ZIP64 = 0x4000, /* always use the zip64 file format, instead of the original zip
|
|
file format with automatic switch to zip64. Use as flags
|
|
parameter with mz_zip_writer_init*_v2 */
|
|
MZ_ZIP_FLAG_WRITE_ALLOW_READING = 0x8000,
|
|
MZ_ZIP_FLAG_ASCII_FILENAME = 0x10000,
|
|
/*After adding a compressed file, seek back
|
|
to local file header and set the correct sizes*/
|
|
MZ_ZIP_FLAG_WRITE_HEADER_SET_SIZE = 0x20000,
|
|
MZ_ZIP_FLAG_READ_ALLOW_WRITING = 0x40000
|
|
} mz_zip_flags;
|
|
|
|
typedef enum {
|
|
MZ_ZIP_TYPE_INVALID = 0,
|
|
MZ_ZIP_TYPE_USER,
|
|
MZ_ZIP_TYPE_MEMORY,
|
|
MZ_ZIP_TYPE_HEAP,
|
|
MZ_ZIP_TYPE_FILE,
|
|
MZ_ZIP_TYPE_CFILE,
|
|
MZ_ZIP_TOTAL_TYPES
|
|
} mz_zip_type;
|
|
|
|
/* miniz error codes. Be sure to update mz_zip_get_error_string() if you add or
|
|
* modify this enum. */
|
|
typedef enum {
|
|
MZ_ZIP_NO_ERROR = 0,
|
|
MZ_ZIP_UNDEFINED_ERROR,
|
|
MZ_ZIP_TOO_MANY_FILES,
|
|
MZ_ZIP_FILE_TOO_LARGE,
|
|
MZ_ZIP_UNSUPPORTED_METHOD,
|
|
MZ_ZIP_UNSUPPORTED_ENCRYPTION,
|
|
MZ_ZIP_UNSUPPORTED_FEATURE,
|
|
MZ_ZIP_FAILED_FINDING_CENTRAL_DIR,
|
|
MZ_ZIP_NOT_AN_ARCHIVE,
|
|
MZ_ZIP_INVALID_HEADER_OR_CORRUPTED,
|
|
MZ_ZIP_UNSUPPORTED_MULTIDISK,
|
|
MZ_ZIP_DECOMPRESSION_FAILED,
|
|
MZ_ZIP_COMPRESSION_FAILED,
|
|
MZ_ZIP_UNEXPECTED_DECOMPRESSED_SIZE,
|
|
MZ_ZIP_CRC_CHECK_FAILED,
|
|
MZ_ZIP_UNSUPPORTED_CDIR_SIZE,
|
|
MZ_ZIP_ALLOC_FAILED,
|
|
MZ_ZIP_FILE_OPEN_FAILED,
|
|
MZ_ZIP_FILE_CREATE_FAILED,
|
|
MZ_ZIP_FILE_WRITE_FAILED,
|
|
MZ_ZIP_FILE_READ_FAILED,
|
|
MZ_ZIP_FILE_CLOSE_FAILED,
|
|
MZ_ZIP_FILE_SEEK_FAILED,
|
|
MZ_ZIP_FILE_STAT_FAILED,
|
|
MZ_ZIP_INVALID_PARAMETER,
|
|
MZ_ZIP_INVALID_FILENAME,
|
|
MZ_ZIP_BUF_TOO_SMALL,
|
|
MZ_ZIP_INTERNAL_ERROR,
|
|
MZ_ZIP_FILE_NOT_FOUND,
|
|
MZ_ZIP_ARCHIVE_TOO_LARGE,
|
|
MZ_ZIP_VALIDATION_FAILED,
|
|
MZ_ZIP_WRITE_CALLBACK_FAILED,
|
|
MZ_ZIP_TOTAL_ERRORS
|
|
} mz_zip_error;
|
|
|
|
typedef struct {
|
|
mz_uint64 m_archive_size;
|
|
mz_uint64 m_central_directory_file_ofs;
|
|
|
|
/* We only support up to UINT32_MAX files in zip64 mode. */
|
|
mz_uint32 m_total_files;
|
|
mz_zip_mode m_zip_mode;
|
|
mz_zip_type m_zip_type;
|
|
mz_zip_error m_last_error;
|
|
|
|
mz_uint64 m_file_offset_alignment;
|
|
|
|
mz_alloc_func m_pAlloc;
|
|
mz_free_func m_pFree;
|
|
mz_realloc_func m_pRealloc;
|
|
void* m_pAlloc_opaque;
|
|
|
|
mz_file_read_func m_pRead;
|
|
mz_file_write_func m_pWrite;
|
|
mz_file_needs_keepalive m_pNeeds_keepalive;
|
|
void* m_pIO_opaque;
|
|
|
|
mz_zip_internal_state* m_pState;
|
|
|
|
} mz_zip_archive;
|
|
|
|
typedef struct {
|
|
mz_zip_archive* pZip;
|
|
mz_uint flags;
|
|
|
|
int status;
|
|
|
|
mz_uint64 read_buf_size, read_buf_ofs, read_buf_avail, comp_remaining, out_buf_ofs, cur_file_ofs;
|
|
mz_zip_archive_file_stat file_stat;
|
|
void* pRead_buf;
|
|
void* pWrite_buf;
|
|
|
|
size_t out_blk_remain;
|
|
|
|
tinfl_decompressor inflator;
|
|
|
|
#ifdef MINIZ_DISABLE_ZIP_READER_CRC32_CHECKS
|
|
mz_uint padding;
|
|
#else
|
|
mz_uint file_crc32;
|
|
#endif
|
|
|
|
} mz_zip_reader_extract_iter_state;
|
|
|
|
/* -------- ZIP reading */
|
|
|
|
/* Inits a ZIP archive reader. */
|
|
/* These functions read and validate the archive's central directory. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_init(mz_zip_archive* pZip, mz_uint64 size, mz_uint flags);
|
|
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_init_mem(mz_zip_archive* pZip, const void* pMem, size_t size, mz_uint flags);
|
|
|
|
#ifndef MINIZ_NO_STDIO
|
|
/* Read a archive from a disk file. */
|
|
/* file_start_ofs is the file offset where the archive actually begins, or 0. */
|
|
/* actual_archive_size is the true total size of the archive, which may be
|
|
* smaller than the file's actual size on disk. If zero the entire file is
|
|
* treated as the archive. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_init_file(mz_zip_archive* pZip, const char* pFilename, mz_uint32 flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_init_file_v2(mz_zip_archive* pZip, const char* pFilename, mz_uint flags,
|
|
mz_uint64 file_start_ofs, mz_uint64 archive_size);
|
|
|
|
/* Read an archive from an already opened FILE, beginning at the current file
|
|
* position. */
|
|
/* The archive is assumed to be archive_size bytes long. If archive_size is 0,
|
|
* then the entire rest of the file is assumed to contain the archive. */
|
|
/* The FILE will NOT be closed when mz_zip_reader_end() is called. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_init_cfile(mz_zip_archive* pZip, MZ_FILE* pFile, mz_uint64 archive_size,
|
|
mz_uint flags);
|
|
#endif
|
|
|
|
/* Ends archive reading, freeing all allocations, and closing the input archive
|
|
* file if mz_zip_reader_init_file() was used. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_end(mz_zip_archive* pZip);
|
|
|
|
/* -------- ZIP reading or writing */
|
|
|
|
/* Clears a mz_zip_archive struct to all zeros. */
|
|
/* Important: This must be done before passing the struct to any mz_zip
|
|
* functions. */
|
|
MINIZ_EXPORT void mz_zip_zero_struct(mz_zip_archive* pZip);
|
|
|
|
MINIZ_EXPORT mz_zip_mode mz_zip_get_mode(mz_zip_archive* pZip);
|
|
MINIZ_EXPORT mz_zip_type mz_zip_get_type(mz_zip_archive* pZip);
|
|
|
|
/* Returns the total number of files in the archive. */
|
|
MINIZ_EXPORT mz_uint mz_zip_reader_get_num_files(mz_zip_archive* pZip);
|
|
|
|
MINIZ_EXPORT mz_uint64 mz_zip_get_archive_size(mz_zip_archive* pZip);
|
|
MINIZ_EXPORT mz_uint64 mz_zip_get_archive_file_start_offset(mz_zip_archive* pZip);
|
|
MINIZ_EXPORT MZ_FILE* mz_zip_get_cfile(mz_zip_archive* pZip);
|
|
|
|
/* Reads n bytes of raw archive data, starting at file offset file_ofs, to pBuf.
|
|
*/
|
|
MINIZ_EXPORT size_t mz_zip_read_archive_data(mz_zip_archive* pZip, mz_uint64 file_ofs, void* pBuf, size_t n);
|
|
|
|
/* All mz_zip funcs set the m_last_error field in the mz_zip_archive struct.
|
|
* These functions retrieve/manipulate this field. */
|
|
/* Note that the m_last_error functionality is not thread safe. */
|
|
MINIZ_EXPORT mz_zip_error mz_zip_set_last_error(mz_zip_archive* pZip, mz_zip_error err_num);
|
|
MINIZ_EXPORT mz_zip_error mz_zip_peek_last_error(mz_zip_archive* pZip);
|
|
MINIZ_EXPORT mz_zip_error mz_zip_clear_last_error(mz_zip_archive* pZip);
|
|
MINIZ_EXPORT mz_zip_error mz_zip_get_last_error(mz_zip_archive* pZip);
|
|
MINIZ_EXPORT const char* mz_zip_get_error_string(mz_zip_error mz_err);
|
|
|
|
/* MZ_TRUE if the archive file entry is a directory entry. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_is_file_a_directory(mz_zip_archive* pZip, mz_uint file_index);
|
|
|
|
/* MZ_TRUE if the file is encrypted/strong encrypted. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_is_file_encrypted(mz_zip_archive* pZip, mz_uint file_index);
|
|
|
|
/* MZ_TRUE if the compression method is supported, and the file is not
|
|
* encrypted, and the file is not a compressed patch file. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_is_file_supported(mz_zip_archive* pZip, mz_uint file_index);
|
|
|
|
/* Retrieves the filename of an archive file entry. */
|
|
/* Returns the number of bytes written to pFilename, or if filename_buf_size is
|
|
* 0 this function returns the number of bytes needed to fully store the
|
|
* filename. */
|
|
MINIZ_EXPORT mz_uint mz_zip_reader_get_filename(mz_zip_archive* pZip, mz_uint file_index, char* pFilename,
|
|
mz_uint filename_buf_size);
|
|
|
|
/* Attempts to locates a file in the archive's central directory. */
|
|
/* Valid flags: MZ_ZIP_FLAG_CASE_SENSITIVE, MZ_ZIP_FLAG_IGNORE_PATH */
|
|
/* Returns -1 if the file cannot be found. */
|
|
MINIZ_EXPORT int mz_zip_reader_locate_file(mz_zip_archive* pZip, const char* pName, const char* pComment,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_locate_file_v2(mz_zip_archive* pZip, const char* pName, const char* pComment,
|
|
mz_uint flags, mz_uint32* file_index);
|
|
|
|
/* Returns detailed information about an archive file entry. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_file_stat(mz_zip_archive* pZip, mz_uint file_index, mz_zip_archive_file_stat* pStat);
|
|
|
|
/* MZ_TRUE if the file is in zip64 format. */
|
|
/* A file is considered zip64 if it contained a zip64 end of central directory
|
|
* marker, or if it contained any zip64 extended file information fields in the
|
|
* central directory. */
|
|
MINIZ_EXPORT mz_bool mz_zip_is_zip64(mz_zip_archive* pZip);
|
|
|
|
/* Returns the total central directory size in bytes. */
|
|
/* The current max supported size is <= MZ_UINT32_MAX. */
|
|
MINIZ_EXPORT size_t mz_zip_get_central_dir_size(mz_zip_archive* pZip);
|
|
|
|
/* Extracts a archive file to a memory buffer using no memory allocation. */
|
|
/* There must be at least enough room on the stack to store the inflator's state
|
|
* (~34KB or so). */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_to_mem_no_alloc(mz_zip_archive* pZip, mz_uint file_index, void* pBuf,
|
|
size_t buf_size, mz_uint flags, void* pUser_read_buf,
|
|
size_t user_read_buf_size);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_file_to_mem_no_alloc(mz_zip_archive* pZip, const char* pFilename, void* pBuf,
|
|
size_t buf_size, mz_uint flags, void* pUser_read_buf,
|
|
size_t user_read_buf_size);
|
|
|
|
/* Extracts a archive file to a memory buffer. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_to_mem(mz_zip_archive* pZip, mz_uint file_index, void* pBuf, size_t buf_size,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_file_to_mem(mz_zip_archive* pZip, const char* pFilename, void* pBuf,
|
|
size_t buf_size, mz_uint flags);
|
|
|
|
/* Extracts a archive file to a dynamically allocated heap buffer. */
|
|
/* The memory will be allocated via the mz_zip_archive's alloc/realloc
|
|
* functions. */
|
|
/* Returns NULL and sets the last error on failure. */
|
|
MINIZ_EXPORT void* mz_zip_reader_extract_to_heap(mz_zip_archive* pZip, mz_uint file_index, size_t* pSize,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT void* mz_zip_reader_extract_file_to_heap(mz_zip_archive* pZip, const char* pFilename, size_t* pSize,
|
|
mz_uint flags);
|
|
|
|
/* Extracts a archive file using a callback function to output the file's data.
|
|
*/
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_to_callback(mz_zip_archive* pZip, mz_uint file_index,
|
|
mz_file_write_func pCallback, void* pOpaque, mz_uint flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_file_to_callback(mz_zip_archive* pZip, const char* pFilename,
|
|
mz_file_write_func pCallback, void* pOpaque, mz_uint flags);
|
|
|
|
/* Extract a file iteratively */
|
|
MINIZ_EXPORT mz_zip_reader_extract_iter_state* mz_zip_reader_extract_iter_new(mz_zip_archive* pZip, mz_uint file_index,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT mz_zip_reader_extract_iter_state* mz_zip_reader_extract_file_iter_new(mz_zip_archive* pZip,
|
|
const char* pFilename,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT size_t mz_zip_reader_extract_iter_read(mz_zip_reader_extract_iter_state* pState, void* pvBuf,
|
|
size_t buf_size);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_iter_free(mz_zip_reader_extract_iter_state* pState);
|
|
|
|
#ifndef MINIZ_NO_STDIO
|
|
/* Extracts a archive file to a disk file and sets its last accessed and
|
|
* modified times. */
|
|
/* This function only extracts files, not archive directory records. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_to_file(mz_zip_archive* pZip, mz_uint file_index, const char* pDst_filename,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_file_to_file(mz_zip_archive* pZip, const char* pArchive_filename,
|
|
const char* pDst_filename, mz_uint flags);
|
|
|
|
/* Extracts a archive file starting at the current position in the destination
|
|
* FILE stream. */
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_to_cfile(mz_zip_archive* pZip, mz_uint file_index, MZ_FILE* File,
|
|
mz_uint flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_reader_extract_file_to_cfile(mz_zip_archive* pZip, const char* pArchive_filename,
|
|
MZ_FILE* pFile, mz_uint flags);
|
|
#endif
|
|
|
|
#if 0
|
|
/* TODO */
|
|
typedef void *mz_zip_streaming_extract_state_ptr;
|
|
mz_zip_streaming_extract_state_ptr mz_zip_streaming_extract_begin(mz_zip_archive *pZip, mz_uint file_index, mz_uint flags);
|
|
mz_uint64 mz_zip_streaming_extract_get_size(mz_zip_archive *pZip, mz_zip_streaming_extract_state_ptr pState);
|
|
mz_uint64 mz_zip_streaming_extract_get_cur_ofs(mz_zip_archive *pZip, mz_zip_streaming_extract_state_ptr pState);
|
|
mz_bool mz_zip_streaming_extract_seek(mz_zip_archive *pZip, mz_zip_streaming_extract_state_ptr pState, mz_uint64 new_ofs);
|
|
size_t mz_zip_streaming_extract_read(mz_zip_archive *pZip, mz_zip_streaming_extract_state_ptr pState, void *pBuf, size_t buf_size);
|
|
mz_bool mz_zip_streaming_extract_end(mz_zip_archive *pZip, mz_zip_streaming_extract_state_ptr pState);
|
|
#endif
|
|
|
|
/* This function compares the archive's local headers, the optional local zip64
|
|
* extended information block, and the optional descriptor following the
|
|
* compressed data vs. the data in the central directory. */
|
|
/* It also validates that each file can be successfully uncompressed unless the
|
|
* MZ_ZIP_FLAG_VALIDATE_HEADERS_ONLY is specified. */
|
|
MINIZ_EXPORT mz_bool mz_zip_validate_file(mz_zip_archive* pZip, mz_uint file_index, mz_uint flags);
|
|
|
|
/* Validates an entire archive by calling mz_zip_validate_file() on each file.
|
|
*/
|
|
MINIZ_EXPORT mz_bool mz_zip_validate_archive(mz_zip_archive* pZip, mz_uint flags);
|
|
|
|
/* Misc utils/helpers, valid for ZIP reading or writing */
|
|
MINIZ_EXPORT mz_bool mz_zip_validate_mem_archive(const void* pMem, size_t size, mz_uint flags, mz_zip_error* pErr);
|
|
#ifndef MINIZ_NO_STDIO
|
|
MINIZ_EXPORT mz_bool mz_zip_validate_file_archive(const char* pFilename, mz_uint flags, mz_zip_error* pErr);
|
|
#endif
|
|
|
|
/* Universal end function - calls either mz_zip_reader_end() or
|
|
* mz_zip_writer_end(). */
|
|
MINIZ_EXPORT mz_bool mz_zip_end(mz_zip_archive* pZip);
|
|
|
|
/* -------- ZIP writing */
|
|
|
|
#ifndef MINIZ_NO_ARCHIVE_WRITING_APIS
|
|
|
|
/* Inits a ZIP archive writer. */
|
|
/*Set pZip->m_pWrite (and pZip->m_pIO_opaque) before calling mz_zip_writer_init
|
|
* or mz_zip_writer_init_v2*/
|
|
/*The output is streamable, i.e. file_ofs in mz_file_write_func always increases
|
|
* only by n*/
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init(mz_zip_archive* pZip, mz_uint64 existing_size);
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_v2(mz_zip_archive* pZip, mz_uint64 existing_size, mz_uint flags);
|
|
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_heap(mz_zip_archive* pZip, size_t size_to_reserve_at_beginning,
|
|
size_t initial_allocation_size);
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_heap_v2(mz_zip_archive* pZip, size_t size_to_reserve_at_beginning,
|
|
size_t initial_allocation_size, mz_uint flags);
|
|
|
|
#ifndef MINIZ_NO_STDIO
|
|
MINIZ_EXPORT
|
|
mz_bool mz_zip_writer_init_file(mz_zip_archive* pZip, const char* pFilename, mz_uint64 size_to_reserve_at_beginning);
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_file_v2(mz_zip_archive* pZip, const char* pFilename,
|
|
mz_uint64 size_to_reserve_at_beginning, mz_uint flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_cfile(mz_zip_archive* pZip, MZ_FILE* pFile, mz_uint flags);
|
|
#endif
|
|
|
|
/* Converts a ZIP archive reader object into a writer object, to allow efficient
|
|
* in-place file appends to occur on an existing archive. */
|
|
/* For archives opened using mz_zip_reader_init_file, pFilename must be the
|
|
* archive's filename so it can be reopened for writing. If the file can't be
|
|
* reopened, mz_zip_reader_end() will be called. */
|
|
/* For archives opened using mz_zip_reader_init_mem, the memory block must be
|
|
* growable using the realloc callback (which defaults to realloc unless you've
|
|
* overridden it). */
|
|
/* Finally, for archives opened using mz_zip_reader_init, the mz_zip_archive's
|
|
* user provided m_pWrite function cannot be NULL. */
|
|
/* Note: In-place archive modification is not recommended unless you know what
|
|
* you're doing, because if execution stops or something goes wrong before */
|
|
/* the archive is finalized the file's central directory will be hosed. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_from_reader(mz_zip_archive* pZip, const char* pFilename);
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_init_from_reader_v2(mz_zip_archive* pZip, const char* pFilename, mz_uint flags);
|
|
|
|
/* Adds the contents of a memory buffer to an archive. These functions record
|
|
* the current local time into the archive. */
|
|
/* To add a directory entry, call this method with an archive name ending in a
|
|
* forwardslash with an empty buffer. */
|
|
/* level_and_flags - compression level (0-10, see MZ_BEST_SPEED,
|
|
* MZ_BEST_COMPRESSION, etc.) logically OR'd with zero or more mz_zip_flags, or
|
|
* just set to MZ_DEFAULT_COMPRESSION. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_mem(mz_zip_archive* pZip, const char* pArchive_name, const void* pBuf,
|
|
size_t buf_size, mz_uint level_and_flags);
|
|
|
|
/* Like mz_zip_writer_add_mem(), except you can specify a file comment field,
|
|
* and optionally supply the function with already compressed data. */
|
|
/* uncomp_size/uncomp_crc32 are only used if the MZ_ZIP_FLAG_COMPRESSED_DATA
|
|
* flag is specified. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_mem_ex(mz_zip_archive* pZip, const char* pArchive_name, const void* pBuf,
|
|
size_t buf_size, const void* pComment, mz_uint16 comment_size,
|
|
mz_uint level_and_flags, mz_uint64 uncomp_size, mz_uint32 uncomp_crc32);
|
|
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_mem_ex_v2(mz_zip_archive* pZip, const char* pArchive_name, const void* pBuf,
|
|
size_t buf_size, const void* pComment, mz_uint16 comment_size,
|
|
mz_uint level_and_flags, mz_uint64 uncomp_size, mz_uint32 uncomp_crc32,
|
|
MZ_TIME_T* last_modified, const char* user_extra_data_local,
|
|
mz_uint user_extra_data_local_len, const char* user_extra_data_central,
|
|
mz_uint user_extra_data_central_len);
|
|
|
|
/* Adds the contents of a file to an archive. This function also records the
|
|
* disk file's modified time into the archive. */
|
|
/* File data is supplied via a read callback function. User
|
|
* mz_zip_writer_add_(c)file to add a file directly.*/
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_read_buf_callback(
|
|
mz_zip_archive* pZip, const char* pArchive_name, mz_file_read_func read_callback, void* callback_opaque,
|
|
mz_uint64 max_size, const MZ_TIME_T* pFile_time, const void* pComment, mz_uint16 comment_size,
|
|
mz_uint level_and_flags, const char* user_extra_data_local, mz_uint user_extra_data_local_len,
|
|
const char* user_extra_data_central, mz_uint user_extra_data_central_len);
|
|
|
|
#ifndef MINIZ_NO_STDIO
|
|
/* Adds the contents of a disk file to an archive. This function also records
|
|
* the disk file's modified time into the archive. */
|
|
/* level_and_flags - compression level (0-10, see MZ_BEST_SPEED,
|
|
* MZ_BEST_COMPRESSION, etc.) logically OR'd with zero or more mz_zip_flags, or
|
|
* just set to MZ_DEFAULT_COMPRESSION. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_file(mz_zip_archive* pZip, const char* pArchive_name, const char* pSrc_filename,
|
|
const void* pComment, mz_uint16 comment_size, mz_uint level_and_flags);
|
|
|
|
/* Like mz_zip_writer_add_file(), except the file data is read from the
|
|
* specified FILE stream. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_cfile(mz_zip_archive* pZip, const char* pArchive_name, MZ_FILE* pSrc_file,
|
|
mz_uint64 max_size, const MZ_TIME_T* pFile_time, const void* pComment,
|
|
mz_uint16 comment_size, mz_uint level_and_flags,
|
|
const char* user_extra_data_local, mz_uint user_extra_data_local_len,
|
|
const char* user_extra_data_central, mz_uint user_extra_data_central_len);
|
|
#endif
|
|
|
|
/* Adds a file to an archive by fully cloning the data from another archive. */
|
|
/* This function fully clones the source file's compressed data (no
|
|
* recompression), along with its full filename, extra data (it may add or
|
|
* modify the zip64 local header extra data field), and the optional descriptor
|
|
* following the compressed data. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_add_from_zip_reader(mz_zip_archive* pZip, mz_zip_archive* pSource_zip,
|
|
mz_uint src_file_index);
|
|
|
|
/* Finalizes the archive by writing the central directory records followed by
|
|
* the end of central directory record. */
|
|
/* After an archive is finalized, the only valid call on the mz_zip_archive
|
|
* struct is mz_zip_writer_end(). */
|
|
/* An archive must be manually finalized by calling this function for it to be
|
|
* valid. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_finalize_archive(mz_zip_archive* pZip);
|
|
|
|
/* Finalizes a heap archive, returning a pointer to the heap block and its size.
|
|
*/
|
|
/* The heap block will be allocated using the mz_zip_archive's alloc/realloc
|
|
* callbacks. */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_finalize_heap_archive(mz_zip_archive* pZip, void** ppBuf, size_t* pSize);
|
|
|
|
/* Ends archive writing, freeing all allocations, and closing the output file if
|
|
* mz_zip_writer_init_file() was used. */
|
|
/* Note for the archive to be valid, it *must* have been finalized before ending
|
|
* (this function will not do it for you). */
|
|
MINIZ_EXPORT mz_bool mz_zip_writer_end(mz_zip_archive* pZip);
|
|
|
|
/* -------- Misc. high-level helper functions: */
|
|
|
|
/* mz_zip_add_mem_to_archive_file_in_place() efficiently (but not atomically)
|
|
* appends a memory blob to a ZIP archive. */
|
|
/* Note this is NOT a fully safe operation. If it crashes or dies in some way
|
|
* your archive can be left in a screwed up state (without a central directory).
|
|
*/
|
|
/* level_and_flags - compression level (0-10, see MZ_BEST_SPEED,
|
|
* MZ_BEST_COMPRESSION, etc.) logically OR'd with zero or more mz_zip_flags, or
|
|
* just set to MZ_DEFAULT_COMPRESSION. */
|
|
/* TODO: Perhaps add an option to leave the existing central dir in place in
|
|
* case the add dies? We could then truncate the file (so the old central dir
|
|
* would be at the end) if something goes wrong. */
|
|
MINIZ_EXPORT mz_bool mz_zip_add_mem_to_archive_file_in_place(const char* pZip_filename, const char* pArchive_name,
|
|
const void* pBuf, size_t buf_size, const void* pComment,
|
|
mz_uint16 comment_size, mz_uint level_and_flags);
|
|
MINIZ_EXPORT mz_bool mz_zip_add_mem_to_archive_file_in_place_v2(const char* pZip_filename, const char* pArchive_name,
|
|
const void* pBuf, size_t buf_size, const void* pComment,
|
|
mz_uint16 comment_size, mz_uint level_and_flags,
|
|
mz_zip_error* pErr);
|
|
|
|
#ifndef MINIZ_NO_STDIO
|
|
/* Reads a single file from an archive into a heap block. */
|
|
/* If pComment is not NULL, only the file with the specified comment will be
|
|
* extracted. */
|
|
/* Returns NULL on failure. */
|
|
MINIZ_EXPORT void* mz_zip_extract_archive_file_to_heap(const char* pZip_filename, const char* pArchive_name,
|
|
size_t* pSize, mz_uint flags);
|
|
MINIZ_EXPORT void* mz_zip_extract_archive_file_to_heap_v2(const char* pZip_filename, const char* pArchive_name,
|
|
const char* pComment, size_t* pSize, mz_uint flags,
|
|
mz_zip_error* pErr);
|
|
#endif
|
|
|
|
#endif /* #ifndef MINIZ_NO_ARCHIVE_WRITING_APIS */
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* MINIZ_NO_ARCHIVE_APIS */
|