1
0
mirror of https://github.com/soarqin/DSP_Mods_TO.git synced 2025-12-14 18:43:29 +08:00
Files
DSP_Mods_TO/CompressSave/LZ4WrapC/LZ4Wrap.c

166 lines
5.2 KiB
C

// MathLibrary.cpp : Defines the exported functions for the DLL.
#include "LZ4Wrap.h"
#include <windows.h>
#include <limits.h>
#include <assert.h>
#include <stdio.h>
#define Check(assert, errorcode) \
if (!(assert)) \
{ \
printf("%s\n", LZ4F_getErrorName(errorcode)); \
return errorcode; \
}
static CContext *CreateCompressContext()
{
return (CContext *)malloc(sizeof(CContext));
}
void __stdcall CompressContextFree(CContext *ctx)
{
if (ctx != NULL)
{
LZ4F_freeCompressionContext(ctx->cctx);
LZ4F_freeCDict(ctx->dict);
free(ctx);
}
}
static LZ4F_preferences_t kPrefs = {
{LZ4F_max4MB, LZ4F_blockLinked, LZ4F_contentChecksumEnabled, LZ4F_frame,
0 /* unknown content size */, 0 /* no dictID */, LZ4F_blockChecksumEnabled},
0, /* compression level; 0 == default */
0, /* autoflush */
0, /* favor decompression speed */
{0, 0, 0}, /* reserved, must be set to 0 */
};
size_t __stdcall CompressBufferBound(size_t inBufferSize)
{
return LZ4F_compressBound(inBufferSize, &kPrefs) + LZ4F_HEADER_SIZE_MAX;
}
CContext *CreateCompressContextFromBuffer(void *dict, size_t dictSize)
{
CContext *ctx = CreateCompressContext();
if (dict)
ctx->dict = LZ4F_createCDict(dict, dictSize);
else
ctx->dict = NULL;
if (ctx == NULL) return NULL;
LZ4F_compressionContext_t innerctx;
LZ4F_errorCode_t ctxCreation = LZ4F_createCompressionContext(&innerctx, LZ4F_VERSION);
if (LZ4F_isError(ctxCreation))
{
LZ4F_freeCompressionContext(innerctx);
CompressContextFree(ctx);
return NULL;
}
ctx->cctx = innerctx;
return ctx;
}
size_t __stdcall CompressBegin(CContext **pctx, int compressionLevel, void *outBuff, size_t outCapacity, void *dict, size_t dictSize)
{
CContext *ctx = CreateCompressContextFromBuffer(dict, dictSize);
if (ctx == NULL)
return -1;
if (outCapacity < LZ4F_HEADER_SIZE_MAX || outCapacity < LZ4F_compressBound(0, &kPrefs))
return LZ4F_ERROR_dstMaxSize_tooSmall;
kPrefs.compressionLevel = compressionLevel;
/* write frame header */
size_t const headerSize = ctx->dict == NULL
? LZ4F_compressBegin(ctx->cctx, outBuff, outCapacity, &kPrefs)
: LZ4F_compressBegin_usingCDict(ctx->cctx, outBuff, outCapacity, ctx->dict, &kPrefs);
if (LZ4F_isError(headerSize))
{
return headerSize;
}
*pctx = ctx;
return headerSize;
}
size_t __stdcall CompressUpdate(CContext *ctx, void *dstBuffer, size_t dstCapacity, const void *srcBuffer, size_t srcSize)
{
size_t result = ctx->dict == NULL
? LZ4F_compressUpdate(ctx->cctx, dstBuffer, dstCapacity, srcBuffer, srcSize, NULL)
: LZ4F_compressFrame_usingCDict(ctx->cctx, dstBuffer, dstCapacity, srcBuffer, srcSize, ctx->dict, NULL);
if (LZ4F_isError(result))
{
const char *str = LZ4F_getErrorName(result);
fprintf(stderr, "%s\n", str);
}
return result;
}
size_t __stdcall CompressEnd(CContext *ctx, void *dstBuffer, size_t dstCapacity)
{
size_t writeSize = LZ4F_compressEnd(ctx->cctx, dstBuffer, dstCapacity, NULL);
return writeSize;
}
static size_t get_block_size(const LZ4F_frameInfo_t *info)
{
switch (info->blockSizeID)
{
case LZ4F_default:
case LZ4F_max64KB:
return 1 << 16;
case LZ4F_max256KB:
return 1 << 18;
case LZ4F_max1MB:
return 1 << 20;
case LZ4F_max4MB:
return 1 << 22;
default:
return -1;
}
}
// return: input bytes expects for next call
size_t __stdcall DecompressBegin(DContext **pdctx, void *inBuffer, size_t *inBufferSize, size_t *blockSize, void *dict, size_t dictSize)
{
DContext *dctx = (DContext *)malloc(sizeof(DContext));
size_t const dctxStatus = LZ4F_createDecompressionContext(&dctx->dctx, LZ4F_VERSION);
Check(!LZ4F_isError(dctxStatus), dctxStatus);
Check(*inBufferSize >= LZ4F_HEADER_SIZE_MAX, LZ4F_ERROR_dstMaxSize_tooSmall);
LZ4F_frameInfo_t info;
size_t const fires = LZ4F_getFrameInfo(dctx->dctx, &info, inBuffer, inBufferSize);
Check(!LZ4F_isError(fires), fires);
*blockSize = get_block_size(&info);
dctx->dict = dict;
dctx->dictSize = dictSize;
*pdctx = dctx;
return fires;
}
void __stdcall DecompressContextReset(DContext *dctx)
{
LZ4F_resetDecompressionContext(dctx->dctx);
}
size_t __stdcall DecompressUpdate(DContext *dctx, void *outBuffer, size_t *outBufferSize, void *inBuffer, size_t *inBufferSize)
{
size_t ret = dctx->dict == NULL
? LZ4F_decompress(dctx->dctx, outBuffer, outBufferSize, inBuffer, inBufferSize, NULL)
: LZ4F_decompress_usingDict(dctx->dctx, outBuffer, outBufferSize, inBuffer, inBufferSize, dctx->dict, dctx->dictSize, NULL);
Check(!LZ4F_isError(ret), ret);
return ret;
}
size_t __stdcall DecompressEnd(DContext *dctx)
{
if (!dctx) return 0;
size_t r = LZ4F_freeDecompressionContext(dctx->dctx);
free(dctx);
return r;
}