2019-06-07 21:15:40 +00:00
|
|
|
/*
|
|
|
|
* expand_inmem.c - in-memory decompression implementation
|
|
|
|
*
|
|
|
|
* Copyright (C) 2019 Emmanuel Marty
|
|
|
|
*
|
|
|
|
* This software is provided 'as-is', without any express or implied
|
|
|
|
* warranty. In no event will the authors be held liable for any damages
|
|
|
|
* arising from the use of this software.
|
|
|
|
*
|
|
|
|
* Permission is granted to anyone to use this software for any purpose,
|
|
|
|
* including commercial applications, and to alter it and redistribute it
|
|
|
|
* freely, subject to the following restrictions:
|
|
|
|
*
|
|
|
|
* 1. The origin of this software must not be misrepresented; you must not
|
|
|
|
* claim that you wrote the original software. If you use this software
|
|
|
|
* in a product, an acknowledgment in the product documentation would be
|
|
|
|
* appreciated but is not required.
|
|
|
|
* 2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
* misrepresented as being the original software.
|
|
|
|
* 3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Uses the libdivsufsort library Copyright (c) 2003-2008 Yuta Mori
|
|
|
|
*
|
|
|
|
* Inspired by LZ4 by Yann Collet. https://github.com/lz4/lz4
|
|
|
|
* With help, ideas, optimizations and speed measurements by spke <zxintrospec@gmail.com>
|
|
|
|
* With ideas from Lizard by Przemyslaw Skibinski and Yann Collet. https://github.com/inikep/lizard
|
|
|
|
* Also with ideas from smallz4 by Stephan Brumme. https://create.stephan-brumme.com/smallz4/
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "expand_inmem.h"
|
|
|
|
#include "lib.h"
|
|
|
|
#include "frame.h"
|
|
|
|
|
|
|
|
#define BLOCK_SIZE 65536
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get maximum decompressed size of compressed data
|
|
|
|
*
|
|
|
|
* @param pFileData compressed data
|
|
|
|
* @param nFileSize compressed size in bytes
|
|
|
|
*
|
|
|
|
* @return maximum decompressed size
|
|
|
|
*/
|
|
|
|
size_t lzsa_get_max_decompressed_size_inmem(const unsigned char *pFileData, size_t nFileSize) {
|
|
|
|
const unsigned char *pCurFileData = pFileData;
|
|
|
|
const unsigned char *pEndFileData = pCurFileData + nFileSize;
|
|
|
|
int nFormatVersion = 0;
|
|
|
|
size_t nMaxDecompressedSize = 0;
|
|
|
|
const int nHeaderSize = lzsa_get_header_size();
|
|
|
|
|
|
|
|
/* Check header */
|
|
|
|
if ((pCurFileData + nHeaderSize) > pEndFileData ||
|
|
|
|
lzsa_decode_header(pCurFileData, nHeaderSize, &nFormatVersion) != 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
pCurFileData += nHeaderSize;
|
|
|
|
|
|
|
|
while (pCurFileData < pEndFileData) {
|
|
|
|
unsigned int nBlockDataSize = 0;
|
|
|
|
int nIsUncompressed = 0;
|
|
|
|
const int nFrameSize = lzsa_get_frame_size();
|
|
|
|
|
|
|
|
/* Decode frame header */
|
|
|
|
if ((pCurFileData + nFrameSize) > pEndFileData ||
|
|
|
|
lzsa_decode_frame(pCurFileData, nFrameSize, &nBlockDataSize, &nIsUncompressed) != 0)
|
|
|
|
return -1;
|
|
|
|
pCurFileData += nFrameSize;
|
|
|
|
|
|
|
|
if (!nBlockDataSize)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Add one potentially full block to the decompressed size */
|
|
|
|
nMaxDecompressedSize += BLOCK_SIZE;
|
|
|
|
|
|
|
|
if ((pCurFileData + nBlockDataSize) > pEndFileData)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
pCurFileData += nBlockDataSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nMaxDecompressedSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decompress data in memory
|
|
|
|
*
|
|
|
|
* @param pFileData compressed data
|
|
|
|
* @param pOutBuffer buffer for decompressed data
|
|
|
|
* @param nFileSize compressed size in bytes
|
|
|
|
* @param nMaxOutBufferSize maximum capacity of decompression buffer
|
2019-06-08 16:05:00 +00:00
|
|
|
* @param nFlags compression flags (LZSA_FLAG_xxx)
|
2019-06-07 21:15:40 +00:00
|
|
|
* @param pFormatVersion pointer to format version, updated if this function is successful
|
|
|
|
*
|
|
|
|
* @return actual decompressed size, or -1 for error
|
|
|
|
*/
|
2019-07-24 18:08:23 +00:00
|
|
|
size_t lzsa_decompress_inmem(unsigned char *pFileData, unsigned char *pOutBuffer, size_t nFileSize, size_t nMaxOutBufferSize, const unsigned int nFlags, int *pFormatVersion) {
|
|
|
|
unsigned char *pCurFileData = pFileData;
|
2019-06-07 21:15:40 +00:00
|
|
|
const unsigned char *pEndFileData = pCurFileData + nFileSize;
|
|
|
|
unsigned char *pCurOutBuffer = pOutBuffer;
|
|
|
|
const unsigned char *pEndOutBuffer = pCurOutBuffer + nMaxOutBufferSize;
|
|
|
|
int nPreviousBlockSize;
|
|
|
|
const int nHeaderSize = lzsa_get_header_size();
|
|
|
|
|
2019-06-08 16:05:00 +00:00
|
|
|
if (nFlags & LZSA_FLAG_RAW_BLOCK) {
|
2019-07-24 18:08:23 +00:00
|
|
|
return (size_t)lzsa_decompressor_expand_block(pFileData, (int)nFileSize, pOutBuffer, 0, (int)nMaxOutBufferSize, *pFormatVersion, nFlags);
|
2019-06-08 16:05:00 +00:00
|
|
|
}
|
|
|
|
|
2019-06-07 21:15:40 +00:00
|
|
|
/* Check header */
|
|
|
|
if ((pCurFileData + nHeaderSize) > pEndFileData ||
|
2019-06-08 16:05:00 +00:00
|
|
|
lzsa_decode_header(pCurFileData, nHeaderSize, pFormatVersion) != 0)
|
2019-06-07 21:15:40 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
pCurFileData += nHeaderSize;
|
|
|
|
nPreviousBlockSize = 0;
|
|
|
|
|
|
|
|
while (pCurFileData < pEndFileData) {
|
|
|
|
unsigned int nBlockDataSize = 0;
|
|
|
|
int nIsUncompressed = 0;
|
|
|
|
const int nFrameSize = lzsa_get_frame_size();
|
|
|
|
|
|
|
|
/* Decode frame header */
|
|
|
|
if ((pCurFileData + nFrameSize) > pEndFileData ||
|
|
|
|
lzsa_decode_frame(pCurFileData, nFrameSize, &nBlockDataSize, &nIsUncompressed) != 0)
|
|
|
|
return -1;
|
|
|
|
pCurFileData += nFrameSize;
|
|
|
|
|
|
|
|
if (!nBlockDataSize)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (!nIsUncompressed) {
|
|
|
|
int nDecompressedSize;
|
|
|
|
|
|
|
|
/* Decompress block */
|
|
|
|
if ((pCurFileData + nBlockDataSize) > pEndFileData)
|
|
|
|
return -1;
|
|
|
|
|
2019-07-24 18:08:23 +00:00
|
|
|
nDecompressedSize = lzsa_decompressor_expand_block(pCurFileData, nBlockDataSize, pCurOutBuffer - nPreviousBlockSize, nPreviousBlockSize, (int)(pEndOutBuffer - pCurOutBuffer + nPreviousBlockSize), *pFormatVersion, nFlags);
|
2019-06-07 21:15:40 +00:00
|
|
|
if (nDecompressedSize < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
pCurOutBuffer += nDecompressedSize;
|
|
|
|
nPreviousBlockSize = nDecompressedSize;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Copy uncompressed block */
|
|
|
|
if ((pCurFileData + nBlockDataSize) > pEndFileData)
|
|
|
|
return -1;
|
|
|
|
if ((pCurOutBuffer + nBlockDataSize) > pEndOutBuffer)
|
|
|
|
return -1;
|
|
|
|
memcpy(pCurOutBuffer, pCurFileData, nBlockDataSize);
|
|
|
|
pCurOutBuffer += nBlockDataSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
pCurFileData += nBlockDataSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (int)(pCurOutBuffer - pOutBuffer);
|
|
|
|
}
|