|
@@ -106,6 +106,7 @@ static void ZSTD_initDCtx_internal(ZSTD_DCtx* dctx)
|
|
|
dctx->ddictLocal = NULL;
|
|
|
dctx->dictEnd = NULL;
|
|
|
dctx->ddictIsCold = 0;
|
|
|
+ dctx->dictUses = ZSTD_dont_use;
|
|
|
dctx->inBuff = NULL;
|
|
|
dctx->inBuffSize = 0;
|
|
|
dctx->outBuffSize = 0;
|
|
@@ -147,13 +148,20 @@ ZSTD_DCtx* ZSTD_createDCtx(void)
|
|
|
return ZSTD_createDCtx_advanced(ZSTD_defaultCMem);
|
|
|
}
|
|
|
|
|
|
+static void ZSTD_clearDict(ZSTD_DCtx* dctx)
|
|
|
+{
|
|
|
+ ZSTD_freeDDict(dctx->ddictLocal);
|
|
|
+ dctx->ddictLocal = NULL;
|
|
|
+ dctx->ddict = NULL;
|
|
|
+ dctx->dictUses = ZSTD_dont_use;
|
|
|
+}
|
|
|
+
|
|
|
size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx)
|
|
|
{
|
|
|
if (dctx==NULL) return 0; /* support free on NULL */
|
|
|
- if (dctx->staticSize) return ERROR(memory_allocation); /* not compatible with static DCtx */
|
|
|
+ RETURN_ERROR_IF(dctx->staticSize, memory_allocation, "not compatible with static DCtx");
|
|
|
{ ZSTD_customMem const cMem = dctx->customMem;
|
|
|
- ZSTD_freeDDict(dctx->ddictLocal);
|
|
|
- dctx->ddictLocal = NULL;
|
|
|
+ ZSTD_clearDict(dctx);
|
|
|
ZSTD_free(dctx->inBuff, cMem);
|
|
|
dctx->inBuff = NULL;
|
|
|
#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1)
|
|
@@ -203,7 +211,7 @@ unsigned ZSTD_isFrame(const void* buffer, size_t size)
|
|
|
static size_t ZSTD_frameHeaderSize_internal(const void* src, size_t srcSize, ZSTD_format_e format)
|
|
|
{
|
|
|
size_t const minInputSize = ZSTD_startingInputLength(format);
|
|
|
- if (srcSize < minInputSize) return ERROR(srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(srcSize < minInputSize, srcSize_wrong);
|
|
|
|
|
|
{ BYTE const fhd = ((const BYTE*)src)[minInputSize-1];
|
|
|
U32 const dictID= fhd & 3;
|
|
@@ -238,7 +246,7 @@ size_t ZSTD_getFrameHeader_advanced(ZSTD_frameHeader* zfhPtr, const void* src, s
|
|
|
|
|
|
memset(zfhPtr, 0, sizeof(*zfhPtr)); /* not strictly necessary, but static analyzer do not understand that zfhPtr is only going to be read only if return value is zero, since they are 2 different signals */
|
|
|
if (srcSize < minInputSize) return minInputSize;
|
|
|
- if (src==NULL) return ERROR(GENERIC); /* invalid parameter */
|
|
|
+ RETURN_ERROR_IF(src==NULL, GENERIC, "invalid parameter");
|
|
|
|
|
|
if ( (format != ZSTD_f_zstd1_magicless)
|
|
|
&& (MEM_readLE32(src) != ZSTD_MAGICNUMBER) ) {
|
|
@@ -251,7 +259,7 @@ size_t ZSTD_getFrameHeader_advanced(ZSTD_frameHeader* zfhPtr, const void* src, s
|
|
|
zfhPtr->frameType = ZSTD_skippableFrame;
|
|
|
return 0;
|
|
|
}
|
|
|
- return ERROR(prefix_unknown);
|
|
|
+ RETURN_ERROR(prefix_unknown);
|
|
|
}
|
|
|
|
|
|
/* ensure there is enough `srcSize` to fully read/decode frame header */
|
|
@@ -269,14 +277,13 @@ size_t ZSTD_getFrameHeader_advanced(ZSTD_frameHeader* zfhPtr, const void* src, s
|
|
|
U64 windowSize = 0;
|
|
|
U32 dictID = 0;
|
|
|
U64 frameContentSize = ZSTD_CONTENTSIZE_UNKNOWN;
|
|
|
- if ((fhdByte & 0x08) != 0)
|
|
|
- return ERROR(frameParameter_unsupported); /* reserved bits, must be zero */
|
|
|
+ RETURN_ERROR_IF((fhdByte & 0x08) != 0, frameParameter_unsupported,
|
|
|
+ "reserved bits, must be zero");
|
|
|
|
|
|
if (!singleSegment) {
|
|
|
BYTE const wlByte = ip[pos++];
|
|
|
U32 const windowLog = (wlByte >> 3) + ZSTD_WINDOWLOG_ABSOLUTEMIN;
|
|
|
- if (windowLog > ZSTD_WINDOWLOG_MAX)
|
|
|
- return ERROR(frameParameter_windowTooLarge);
|
|
|
+ RETURN_ERROR_IF(windowLog > ZSTD_WINDOWLOG_MAX, frameParameter_windowTooLarge);
|
|
|
windowSize = (1ULL << windowLog);
|
|
|
windowSize += (windowSize >> 3) * (wlByte&7);
|
|
|
}
|
|
@@ -348,12 +355,11 @@ static size_t readSkippableFrameSize(void const* src, size_t srcSize)
|
|
|
size_t const skippableHeaderSize = ZSTD_SKIPPABLEHEADERSIZE;
|
|
|
U32 sizeU32;
|
|
|
|
|
|
- if (srcSize < ZSTD_SKIPPABLEHEADERSIZE)
|
|
|
- return ERROR(srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(srcSize < ZSTD_SKIPPABLEHEADERSIZE, srcSize_wrong);
|
|
|
|
|
|
sizeU32 = MEM_readLE32((BYTE const*)src + ZSTD_FRAMEIDSIZE);
|
|
|
- if ((U32)(sizeU32 + ZSTD_SKIPPABLEHEADERSIZE) < sizeU32)
|
|
|
- return ERROR(frameParameter_unsupported);
|
|
|
+ RETURN_ERROR_IF((U32)(sizeU32 + ZSTD_SKIPPABLEHEADERSIZE) < sizeU32,
|
|
|
+ frameParameter_unsupported);
|
|
|
|
|
|
return skippableHeaderSize + sizeU32;
|
|
|
}
|
|
@@ -428,67 +434,124 @@ static size_t ZSTD_decodeFrameHeader(ZSTD_DCtx* dctx, const void* src, size_t he
|
|
|
{
|
|
|
size_t const result = ZSTD_getFrameHeader_advanced(&(dctx->fParams), src, headerSize, dctx->format);
|
|
|
if (ZSTD_isError(result)) return result; /* invalid header */
|
|
|
- if (result>0) return ERROR(srcSize_wrong); /* headerSize too small */
|
|
|
- if (dctx->fParams.dictID && (dctx->dictID != dctx->fParams.dictID))
|
|
|
- return ERROR(dictionary_wrong);
|
|
|
+ RETURN_ERROR_IF(result>0, srcSize_wrong, "headerSize too small");
|
|
|
+#ifndef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
|
|
|
+ /* Skip the dictID check in fuzzing mode, because it makes the search
|
|
|
+ * harder.
|
|
|
+ */
|
|
|
+ RETURN_ERROR_IF(dctx->fParams.dictID && (dctx->dictID != dctx->fParams.dictID),
|
|
|
+ dictionary_wrong);
|
|
|
+#endif
|
|
|
if (dctx->fParams.checksumFlag) XXH64_reset(&dctx->xxhState, 0);
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+static ZSTD_frameSizeInfo ZSTD_errorFrameSizeInfo(size_t ret)
|
|
|
+{
|
|
|
+ ZSTD_frameSizeInfo frameSizeInfo;
|
|
|
+ frameSizeInfo.compressedSize = ret;
|
|
|
+ frameSizeInfo.decompressedBound = ZSTD_CONTENTSIZE_ERROR;
|
|
|
+ return frameSizeInfo;
|
|
|
+}
|
|
|
|
|
|
-/** ZSTD_findFrameCompressedSize() :
|
|
|
- * compatible with legacy mode
|
|
|
- * `src` must point to the start of a ZSTD frame, ZSTD legacy frame, or skippable frame
|
|
|
- * `srcSize` must be at least as large as the frame contained
|
|
|
- * @return : the compressed size of the frame starting at `src` */
|
|
|
-size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize)
|
|
|
+static ZSTD_frameSizeInfo ZSTD_findFrameSizeInfo(const void* src, size_t srcSize)
|
|
|
{
|
|
|
+ ZSTD_frameSizeInfo frameSizeInfo;
|
|
|
+ memset(&frameSizeInfo, 0, sizeof(ZSTD_frameSizeInfo));
|
|
|
+
|
|
|
#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1)
|
|
|
if (ZSTD_isLegacy(src, srcSize))
|
|
|
- return ZSTD_findFrameCompressedSizeLegacy(src, srcSize);
|
|
|
+ return ZSTD_findFrameSizeInfoLegacy(src, srcSize);
|
|
|
#endif
|
|
|
- if ( (srcSize >= ZSTD_SKIPPABLEHEADERSIZE)
|
|
|
- && (MEM_readLE32(src) & ZSTD_MAGIC_SKIPPABLE_MASK) == ZSTD_MAGIC_SKIPPABLE_START ) {
|
|
|
- return readSkippableFrameSize(src, srcSize);
|
|
|
+
|
|
|
+ if ((srcSize >= ZSTD_SKIPPABLEHEADERSIZE)
|
|
|
+ && (MEM_readLE32(src) & ZSTD_MAGIC_SKIPPABLE_MASK) == ZSTD_MAGIC_SKIPPABLE_START) {
|
|
|
+ frameSizeInfo.compressedSize = readSkippableFrameSize(src, srcSize);
|
|
|
+ return frameSizeInfo;
|
|
|
} else {
|
|
|
const BYTE* ip = (const BYTE*)src;
|
|
|
const BYTE* const ipstart = ip;
|
|
|
size_t remainingSize = srcSize;
|
|
|
+ size_t nbBlocks = 0;
|
|
|
ZSTD_frameHeader zfh;
|
|
|
|
|
|
/* Extract Frame Header */
|
|
|
{ size_t const ret = ZSTD_getFrameHeader(&zfh, src, srcSize);
|
|
|
- if (ZSTD_isError(ret)) return ret;
|
|
|
- if (ret > 0) return ERROR(srcSize_wrong);
|
|
|
+ if (ZSTD_isError(ret))
|
|
|
+ return ZSTD_errorFrameSizeInfo(ret);
|
|
|
+ if (ret > 0)
|
|
|
+ return ZSTD_errorFrameSizeInfo(ERROR(srcSize_wrong));
|
|
|
}
|
|
|
|
|
|
ip += zfh.headerSize;
|
|
|
remainingSize -= zfh.headerSize;
|
|
|
|
|
|
- /* Loop on each block */
|
|
|
+ /* Iterate over each block */
|
|
|
while (1) {
|
|
|
blockProperties_t blockProperties;
|
|
|
size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties);
|
|
|
- if (ZSTD_isError(cBlockSize)) return cBlockSize;
|
|
|
+ if (ZSTD_isError(cBlockSize))
|
|
|
+ return ZSTD_errorFrameSizeInfo(cBlockSize);
|
|
|
|
|
|
if (ZSTD_blockHeaderSize + cBlockSize > remainingSize)
|
|
|
- return ERROR(srcSize_wrong);
|
|
|
+ return ZSTD_errorFrameSizeInfo(ERROR(srcSize_wrong));
|
|
|
|
|
|
ip += ZSTD_blockHeaderSize + cBlockSize;
|
|
|
remainingSize -= ZSTD_blockHeaderSize + cBlockSize;
|
|
|
+ nbBlocks++;
|
|
|
|
|
|
if (blockProperties.lastBlock) break;
|
|
|
}
|
|
|
|
|
|
- if (zfh.checksumFlag) { /* Final frame content checksum */
|
|
|
- if (remainingSize < 4) return ERROR(srcSize_wrong);
|
|
|
+ /* Final frame content checksum */
|
|
|
+ if (zfh.checksumFlag) {
|
|
|
+ if (remainingSize < 4)
|
|
|
+ return ZSTD_errorFrameSizeInfo(ERROR(srcSize_wrong));
|
|
|
ip += 4;
|
|
|
}
|
|
|
|
|
|
- return ip - ipstart;
|
|
|
+ frameSizeInfo.compressedSize = ip - ipstart;
|
|
|
+ frameSizeInfo.decompressedBound = (zfh.frameContentSize != ZSTD_CONTENTSIZE_UNKNOWN)
|
|
|
+ ? zfh.frameContentSize
|
|
|
+ : nbBlocks * zfh.blockSizeMax;
|
|
|
+ return frameSizeInfo;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+/** ZSTD_findFrameCompressedSize() :
|
|
|
+ * compatible with legacy mode
|
|
|
+ * `src` must point to the start of a ZSTD frame, ZSTD legacy frame, or skippable frame
|
|
|
+ * `srcSize` must be at least as large as the frame contained
|
|
|
+ * @return : the compressed size of the frame starting at `src` */
|
|
|
+size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize)
|
|
|
+{
|
|
|
+ ZSTD_frameSizeInfo const frameSizeInfo = ZSTD_findFrameSizeInfo(src, srcSize);
|
|
|
+ return frameSizeInfo.compressedSize;
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+/** ZSTD_decompressBound() :
|
|
|
+ * compatible with legacy mode
|
|
|
+ * `src` must point to the start of a ZSTD frame or a skippeable frame
|
|
|
+ * `srcSize` must be at least as large as the frame contained
|
|
|
+ * @return : the maximum decompressed size of the compressed source
|
|
|
+ */
|
|
|
+unsigned long long ZSTD_decompressBound(const void* src, size_t srcSize)
|
|
|
+{
|
|
|
+ unsigned long long bound = 0;
|
|
|
+ /* Iterate over each frame */
|
|
|
+ while (srcSize > 0) {
|
|
|
+ ZSTD_frameSizeInfo const frameSizeInfo = ZSTD_findFrameSizeInfo(src, srcSize);
|
|
|
+ size_t const compressedSize = frameSizeInfo.compressedSize;
|
|
|
+ unsigned long long const decompressedBound = frameSizeInfo.decompressedBound;
|
|
|
+ if (ZSTD_isError(compressedSize) || decompressedBound == ZSTD_CONTENTSIZE_ERROR)
|
|
|
+ return ZSTD_CONTENTSIZE_ERROR;
|
|
|
+ src = (const BYTE*)src + compressedSize;
|
|
|
+ srcSize -= compressedSize;
|
|
|
+ bound += decompressedBound;
|
|
|
+ }
|
|
|
+ return bound;
|
|
|
+}
|
|
|
|
|
|
|
|
|
/*-*************************************************************
|
|
@@ -522,9 +585,9 @@ static size_t ZSTD_copyRawBlock(void* dst, size_t dstCapacity,
|
|
|
DEBUGLOG(5, "ZSTD_copyRawBlock");
|
|
|
if (dst == NULL) {
|
|
|
if (srcSize == 0) return 0;
|
|
|
- return ERROR(dstBuffer_null);
|
|
|
+ RETURN_ERROR(dstBuffer_null);
|
|
|
}
|
|
|
- if (srcSize > dstCapacity) return ERROR(dstSize_tooSmall);
|
|
|
+ RETURN_ERROR_IF(srcSize > dstCapacity, dstSize_tooSmall);
|
|
|
memcpy(dst, src, srcSize);
|
|
|
return srcSize;
|
|
|
}
|
|
@@ -535,9 +598,9 @@ static size_t ZSTD_setRleBlock(void* dst, size_t dstCapacity,
|
|
|
{
|
|
|
if (dst == NULL) {
|
|
|
if (regenSize == 0) return 0;
|
|
|
- return ERROR(dstBuffer_null);
|
|
|
+ RETURN_ERROR(dstBuffer_null);
|
|
|
}
|
|
|
- if (regenSize > dstCapacity) return ERROR(dstSize_tooSmall);
|
|
|
+ RETURN_ERROR_IF(regenSize > dstCapacity, dstSize_tooSmall);
|
|
|
memset(dst, b, regenSize);
|
|
|
return regenSize;
|
|
|
}
|
|
@@ -560,15 +623,16 @@ static size_t ZSTD_decompressFrame(ZSTD_DCtx* dctx,
|
|
|
DEBUGLOG(4, "ZSTD_decompressFrame (srcSize:%i)", (int)*srcSizePtr);
|
|
|
|
|
|
/* check */
|
|
|
- if (remainingSrcSize < ZSTD_FRAMEHEADERSIZE_MIN+ZSTD_blockHeaderSize)
|
|
|
- return ERROR(srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ remainingSrcSize < ZSTD_FRAMEHEADERSIZE_MIN+ZSTD_blockHeaderSize,
|
|
|
+ srcSize_wrong);
|
|
|
|
|
|
/* Frame Header */
|
|
|
{ size_t const frameHeaderSize = ZSTD_frameHeaderSize(ip, ZSTD_FRAMEHEADERSIZE_PREFIX);
|
|
|
if (ZSTD_isError(frameHeaderSize)) return frameHeaderSize;
|
|
|
- if (remainingSrcSize < frameHeaderSize+ZSTD_blockHeaderSize)
|
|
|
- return ERROR(srcSize_wrong);
|
|
|
- CHECK_F( ZSTD_decodeFrameHeader(dctx, ip, frameHeaderSize) );
|
|
|
+ RETURN_ERROR_IF(remainingSrcSize < frameHeaderSize+ZSTD_blockHeaderSize,
|
|
|
+ srcSize_wrong);
|
|
|
+ FORWARD_IF_ERROR( ZSTD_decodeFrameHeader(dctx, ip, frameHeaderSize) );
|
|
|
ip += frameHeaderSize; remainingSrcSize -= frameHeaderSize;
|
|
|
}
|
|
|
|
|
@@ -581,7 +645,7 @@ static size_t ZSTD_decompressFrame(ZSTD_DCtx* dctx,
|
|
|
|
|
|
ip += ZSTD_blockHeaderSize;
|
|
|
remainingSrcSize -= ZSTD_blockHeaderSize;
|
|
|
- if (cBlockSize > remainingSrcSize) return ERROR(srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(cBlockSize > remainingSrcSize, srcSize_wrong);
|
|
|
|
|
|
switch(blockProperties.blockType)
|
|
|
{
|
|
@@ -596,7 +660,7 @@ static size_t ZSTD_decompressFrame(ZSTD_DCtx* dctx,
|
|
|
break;
|
|
|
case bt_reserved :
|
|
|
default:
|
|
|
- return ERROR(corruption_detected);
|
|
|
+ RETURN_ERROR(corruption_detected);
|
|
|
}
|
|
|
|
|
|
if (ZSTD_isError(decodedSize)) return decodedSize;
|
|
@@ -609,15 +673,15 @@ static size_t ZSTD_decompressFrame(ZSTD_DCtx* dctx,
|
|
|
}
|
|
|
|
|
|
if (dctx->fParams.frameContentSize != ZSTD_CONTENTSIZE_UNKNOWN) {
|
|
|
- if ((U64)(op-ostart) != dctx->fParams.frameContentSize) {
|
|
|
- return ERROR(corruption_detected);
|
|
|
- } }
|
|
|
+ RETURN_ERROR_IF((U64)(op-ostart) != dctx->fParams.frameContentSize,
|
|
|
+ corruption_detected);
|
|
|
+ }
|
|
|
if (dctx->fParams.checksumFlag) { /* Frame content checksum verification */
|
|
|
U32 const checkCalc = (U32)XXH64_digest(&dctx->xxhState);
|
|
|
U32 checkRead;
|
|
|
- if (remainingSrcSize<4) return ERROR(checksum_wrong);
|
|
|
+ RETURN_ERROR_IF(remainingSrcSize<4, checksum_wrong);
|
|
|
checkRead = MEM_readLE32(ip);
|
|
|
- if (checkRead != checkCalc) return ERROR(checksum_wrong);
|
|
|
+ RETURN_ERROR_IF(checkRead != checkCalc, checksum_wrong);
|
|
|
ip += 4;
|
|
|
remainingSrcSize -= 4;
|
|
|
}
|
|
@@ -652,8 +716,8 @@ static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx,
|
|
|
size_t decodedSize;
|
|
|
size_t const frameSize = ZSTD_findFrameCompressedSizeLegacy(src, srcSize);
|
|
|
if (ZSTD_isError(frameSize)) return frameSize;
|
|
|
- /* legacy support is not compatible with static dctx */
|
|
|
- if (dctx->staticSize) return ERROR(memory_allocation);
|
|
|
+ RETURN_ERROR_IF(dctx->staticSize, memory_allocation,
|
|
|
+ "legacy support is not compatible with static dctx");
|
|
|
|
|
|
decodedSize = ZSTD_decompressLegacy(dst, dstCapacity, src, frameSize, dict, dictSize);
|
|
|
if (ZSTD_isError(decodedSize)) return decodedSize;
|
|
@@ -676,7 +740,7 @@ static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx,
|
|
|
size_t const skippableSize = readSkippableFrameSize(src, srcSize);
|
|
|
if (ZSTD_isError(skippableSize))
|
|
|
return skippableSize;
|
|
|
- if (srcSize < skippableSize) return ERROR(srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(srcSize < skippableSize, srcSize_wrong);
|
|
|
|
|
|
src = (const BYTE *)src + skippableSize;
|
|
|
srcSize -= skippableSize;
|
|
@@ -685,29 +749,29 @@ static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx,
|
|
|
|
|
|
if (ddict) {
|
|
|
/* we were called from ZSTD_decompress_usingDDict */
|
|
|
- CHECK_F(ZSTD_decompressBegin_usingDDict(dctx, ddict));
|
|
|
+ FORWARD_IF_ERROR(ZSTD_decompressBegin_usingDDict(dctx, ddict));
|
|
|
} else {
|
|
|
/* this will initialize correctly with no dict if dict == NULL, so
|
|
|
* use this in all cases but ddict */
|
|
|
- CHECK_F(ZSTD_decompressBegin_usingDict(dctx, dict, dictSize));
|
|
|
+ FORWARD_IF_ERROR(ZSTD_decompressBegin_usingDict(dctx, dict, dictSize));
|
|
|
}
|
|
|
ZSTD_checkContinuity(dctx, dst);
|
|
|
|
|
|
{ const size_t res = ZSTD_decompressFrame(dctx, dst, dstCapacity,
|
|
|
&src, &srcSize);
|
|
|
- if ( (ZSTD_getErrorCode(res) == ZSTD_error_prefix_unknown)
|
|
|
- && (moreThan1Frame==1) ) {
|
|
|
- /* at least one frame successfully completed,
|
|
|
- * but following bytes are garbage :
|
|
|
- * it's more likely to be a srcSize error,
|
|
|
- * specifying more bytes than compressed size of frame(s).
|
|
|
- * This error message replaces ERROR(prefix_unknown),
|
|
|
- * which would be confusing, as the first header is actually correct.
|
|
|
- * Note that one could be unlucky, it might be a corruption error instead,
|
|
|
- * happening right at the place where we expect zstd magic bytes.
|
|
|
- * But this is _much_ less likely than a srcSize field error. */
|
|
|
- return ERROR(srcSize_wrong);
|
|
|
- }
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ (ZSTD_getErrorCode(res) == ZSTD_error_prefix_unknown)
|
|
|
+ && (moreThan1Frame==1),
|
|
|
+ srcSize_wrong,
|
|
|
+ "at least one frame successfully completed, but following "
|
|
|
+ "bytes are garbage: it's more likely to be a srcSize error, "
|
|
|
+ "specifying more bytes than compressed size of frame(s). This "
|
|
|
+ "error message replaces ERROR(prefix_unknown), which would be "
|
|
|
+ "confusing, as the first header is actually correct. Note that "
|
|
|
+ "one could be unlucky, it might be a corruption error instead, "
|
|
|
+ "happening right at the place where we expect zstd magic "
|
|
|
+ "bytes. But this is _much_ less likely than a srcSize field "
|
|
|
+ "error.");
|
|
|
if (ZSTD_isError(res)) return res;
|
|
|
assert(res <= dstCapacity);
|
|
|
dst = (BYTE*)dst + res;
|
|
@@ -716,7 +780,7 @@ static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx,
|
|
|
moreThan1Frame = 1;
|
|
|
} /* while (srcSize >= ZSTD_frameHeaderSize_prefix) */
|
|
|
|
|
|
- if (srcSize) return ERROR(srcSize_wrong); /* input not entirely consumed */
|
|
|
+ RETURN_ERROR_IF(srcSize, srcSize_wrong, "input not entirely consumed");
|
|
|
|
|
|
return (BYTE*)dst - (BYTE*)dststart;
|
|
|
}
|
|
@@ -730,9 +794,26 @@ size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx,
|
|
|
}
|
|
|
|
|
|
|
|
|
+static ZSTD_DDict const* ZSTD_getDDict(ZSTD_DCtx* dctx)
|
|
|
+{
|
|
|
+ switch (dctx->dictUses) {
|
|
|
+ default:
|
|
|
+ assert(0 /* Impossible */);
|
|
|
+ /* fall-through */
|
|
|
+ case ZSTD_dont_use:
|
|
|
+ ZSTD_clearDict(dctx);
|
|
|
+ return NULL;
|
|
|
+ case ZSTD_use_indefinitely:
|
|
|
+ return dctx->ddict;
|
|
|
+ case ZSTD_use_once:
|
|
|
+ dctx->dictUses = ZSTD_dont_use;
|
|
|
+ return dctx->ddict;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
size_t ZSTD_decompressDCtx(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize)
|
|
|
{
|
|
|
- return ZSTD_decompress_usingDict(dctx, dst, dstCapacity, src, srcSize, NULL, 0);
|
|
|
+ return ZSTD_decompress_usingDDict(dctx, dst, dstCapacity, src, srcSize, ZSTD_getDDict(dctx));
|
|
|
}
|
|
|
|
|
|
|
|
@@ -741,7 +822,7 @@ size_t ZSTD_decompress(void* dst, size_t dstCapacity, const void* src, size_t sr
|
|
|
#if defined(ZSTD_HEAPMODE) && (ZSTD_HEAPMODE>=1)
|
|
|
size_t regenSize;
|
|
|
ZSTD_DCtx* const dctx = ZSTD_createDCtx();
|
|
|
- if (dctx==NULL) return ERROR(memory_allocation);
|
|
|
+ RETURN_ERROR_IF(dctx==NULL, memory_allocation);
|
|
|
regenSize = ZSTD_decompressDCtx(dctx, dst, dstCapacity, src, srcSize);
|
|
|
ZSTD_freeDCtx(dctx);
|
|
|
return regenSize;
|
|
@@ -791,8 +872,7 @@ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, c
|
|
|
{
|
|
|
DEBUGLOG(5, "ZSTD_decompressContinue (srcSize:%u)", (unsigned)srcSize);
|
|
|
/* Sanity check */
|
|
|
- if (srcSize != dctx->expected)
|
|
|
- return ERROR(srcSize_wrong); /* not allowed */
|
|
|
+ RETURN_ERROR_IF(srcSize != dctx->expected, srcSize_wrong, "not allowed");
|
|
|
if (dstCapacity) ZSTD_checkContinuity(dctx, dst);
|
|
|
|
|
|
switch (dctx->stage)
|
|
@@ -817,7 +897,7 @@ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, c
|
|
|
case ZSTDds_decodeFrameHeader:
|
|
|
assert(src != NULL);
|
|
|
memcpy(dctx->headerBuffer + (dctx->headerSize - srcSize), src, srcSize);
|
|
|
- CHECK_F(ZSTD_decodeFrameHeader(dctx, dctx->headerBuffer, dctx->headerSize));
|
|
|
+ FORWARD_IF_ERROR(ZSTD_decodeFrameHeader(dctx, dctx->headerBuffer, dctx->headerSize));
|
|
|
dctx->expected = ZSTD_blockHeaderSize;
|
|
|
dctx->stage = ZSTDds_decodeBlockHeader;
|
|
|
return 0;
|
|
@@ -867,7 +947,7 @@ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, c
|
|
|
break;
|
|
|
case bt_reserved : /* should never happen */
|
|
|
default:
|
|
|
- return ERROR(corruption_detected);
|
|
|
+ RETURN_ERROR(corruption_detected);
|
|
|
}
|
|
|
if (ZSTD_isError(rSize)) return rSize;
|
|
|
DEBUGLOG(5, "ZSTD_decompressContinue: decoded size from block : %u", (unsigned)rSize);
|
|
@@ -876,10 +956,10 @@ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, c
|
|
|
|
|
|
if (dctx->stage == ZSTDds_decompressLastBlock) { /* end of frame */
|
|
|
DEBUGLOG(4, "ZSTD_decompressContinue: decoded size from frame : %u", (unsigned)dctx->decodedSize);
|
|
|
- if (dctx->fParams.frameContentSize != ZSTD_CONTENTSIZE_UNKNOWN) {
|
|
|
- if (dctx->decodedSize != dctx->fParams.frameContentSize) {
|
|
|
- return ERROR(corruption_detected);
|
|
|
- } }
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ dctx->fParams.frameContentSize != ZSTD_CONTENTSIZE_UNKNOWN
|
|
|
+ && dctx->decodedSize != dctx->fParams.frameContentSize,
|
|
|
+ corruption_detected);
|
|
|
if (dctx->fParams.checksumFlag) { /* another round for frame checksum */
|
|
|
dctx->expected = 4;
|
|
|
dctx->stage = ZSTDds_checkChecksum;
|
|
@@ -900,7 +980,7 @@ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, c
|
|
|
{ U32 const h32 = (U32)XXH64_digest(&dctx->xxhState);
|
|
|
U32 const check32 = MEM_readLE32(src);
|
|
|
DEBUGLOG(4, "ZSTD_decompressContinue: checksum : calculated %08X :: %08X read", (unsigned)h32, (unsigned)check32);
|
|
|
- if (check32 != h32) return ERROR(checksum_wrong);
|
|
|
+ RETURN_ERROR_IF(check32 != h32, checksum_wrong);
|
|
|
dctx->expected = 0;
|
|
|
dctx->stage = ZSTDds_getFrameHeaderSize;
|
|
|
return 0;
|
|
@@ -921,7 +1001,7 @@ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, c
|
|
|
|
|
|
default:
|
|
|
assert(0); /* impossible */
|
|
|
- return ERROR(GENERIC); /* some compiler require default to do something */
|
|
|
+ RETURN_ERROR(GENERIC); /* some compiler require default to do something */
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -945,7 +1025,7 @@ ZSTD_loadDEntropy(ZSTD_entropyDTables_t* entropy,
|
|
|
const BYTE* dictPtr = (const BYTE*)dict;
|
|
|
const BYTE* const dictEnd = dictPtr + dictSize;
|
|
|
|
|
|
- if (dictSize <= 8) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(dictSize <= 8, dictionary_corrupted);
|
|
|
assert(MEM_readLE32(dict) == ZSTD_MAGIC_DICTIONARY); /* dict must be valid */
|
|
|
dictPtr += 8; /* skip header = magic + dictID */
|
|
|
|
|
@@ -964,16 +1044,16 @@ ZSTD_loadDEntropy(ZSTD_entropyDTables_t* entropy,
|
|
|
dictPtr, dictEnd - dictPtr,
|
|
|
workspace, workspaceSize);
|
|
|
#endif
|
|
|
- if (HUF_isError(hSize)) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(HUF_isError(hSize), dictionary_corrupted);
|
|
|
dictPtr += hSize;
|
|
|
}
|
|
|
|
|
|
{ short offcodeNCount[MaxOff+1];
|
|
|
unsigned offcodeMaxValue = MaxOff, offcodeLog;
|
|
|
size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr);
|
|
|
- if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted);
|
|
|
- if (offcodeMaxValue > MaxOff) return ERROR(dictionary_corrupted);
|
|
|
- if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(FSE_isError(offcodeHeaderSize), dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(offcodeMaxValue > MaxOff, dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(offcodeLog > OffFSELog, dictionary_corrupted);
|
|
|
ZSTD_buildFSETable( entropy->OFTable,
|
|
|
offcodeNCount, offcodeMaxValue,
|
|
|
OF_base, OF_bits,
|
|
@@ -984,9 +1064,9 @@ ZSTD_loadDEntropy(ZSTD_entropyDTables_t* entropy,
|
|
|
{ short matchlengthNCount[MaxML+1];
|
|
|
unsigned matchlengthMaxValue = MaxML, matchlengthLog;
|
|
|
size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr);
|
|
|
- if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted);
|
|
|
- if (matchlengthMaxValue > MaxML) return ERROR(dictionary_corrupted);
|
|
|
- if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(FSE_isError(matchlengthHeaderSize), dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(matchlengthMaxValue > MaxML, dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(matchlengthLog > MLFSELog, dictionary_corrupted);
|
|
|
ZSTD_buildFSETable( entropy->MLTable,
|
|
|
matchlengthNCount, matchlengthMaxValue,
|
|
|
ML_base, ML_bits,
|
|
@@ -997,9 +1077,9 @@ ZSTD_loadDEntropy(ZSTD_entropyDTables_t* entropy,
|
|
|
{ short litlengthNCount[MaxLL+1];
|
|
|
unsigned litlengthMaxValue = MaxLL, litlengthLog;
|
|
|
size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr);
|
|
|
- if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted);
|
|
|
- if (litlengthMaxValue > MaxLL) return ERROR(dictionary_corrupted);
|
|
|
- if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(FSE_isError(litlengthHeaderSize), dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(litlengthMaxValue > MaxLL, dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(litlengthLog > LLFSELog, dictionary_corrupted);
|
|
|
ZSTD_buildFSETable( entropy->LLTable,
|
|
|
litlengthNCount, litlengthMaxValue,
|
|
|
LL_base, LL_bits,
|
|
@@ -1007,12 +1087,13 @@ ZSTD_loadDEntropy(ZSTD_entropyDTables_t* entropy,
|
|
|
dictPtr += litlengthHeaderSize;
|
|
|
}
|
|
|
|
|
|
- if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(dictPtr+12 > dictEnd, dictionary_corrupted);
|
|
|
{ int i;
|
|
|
size_t const dictContentSize = (size_t)(dictEnd - (dictPtr+12));
|
|
|
for (i=0; i<3; i++) {
|
|
|
U32 const rep = MEM_readLE32(dictPtr); dictPtr += 4;
|
|
|
- if (rep==0 || rep >= dictContentSize) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(rep==0 || rep >= dictContentSize,
|
|
|
+ dictionary_corrupted);
|
|
|
entropy->rep[i] = rep;
|
|
|
} }
|
|
|
|
|
@@ -1030,7 +1111,7 @@ static size_t ZSTD_decompress_insertDictionary(ZSTD_DCtx* dctx, const void* dict
|
|
|
|
|
|
/* load entropy tables */
|
|
|
{ size_t const eSize = ZSTD_loadDEntropy(&dctx->entropy, dict, dictSize);
|
|
|
- if (ZSTD_isError(eSize)) return ERROR(dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(ZSTD_isError(eSize), dictionary_corrupted);
|
|
|
dict = (const char*)dict + eSize;
|
|
|
dictSize -= eSize;
|
|
|
}
|
|
@@ -1064,9 +1145,11 @@ size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx)
|
|
|
|
|
|
size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize)
|
|
|
{
|
|
|
- CHECK_F( ZSTD_decompressBegin(dctx) );
|
|
|
+ FORWARD_IF_ERROR( ZSTD_decompressBegin(dctx) );
|
|
|
if (dict && dictSize)
|
|
|
- CHECK_E(ZSTD_decompress_insertDictionary(dctx, dict, dictSize), dictionary_corrupted);
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ ZSTD_isError(ZSTD_decompress_insertDictionary(dctx, dict, dictSize)),
|
|
|
+ dictionary_corrupted);
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -1085,7 +1168,7 @@ size_t ZSTD_decompressBegin_usingDDict(ZSTD_DCtx* dctx, const ZSTD_DDict* ddict)
|
|
|
DEBUGLOG(4, "DDict is %s",
|
|
|
dctx->ddictIsCold ? "~cold~" : "hot!");
|
|
|
}
|
|
|
- CHECK_F( ZSTD_decompressBegin(dctx) );
|
|
|
+ FORWARD_IF_ERROR( ZSTD_decompressBegin(dctx) );
|
|
|
if (ddict) { /* NULL ddict is equivalent to no dictionary */
|
|
|
ZSTD_copyDDictParameters(dctx, ddict);
|
|
|
}
|
|
@@ -1104,7 +1187,7 @@ unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize)
|
|
|
}
|
|
|
|
|
|
/*! ZSTD_getDictID_fromFrame() :
|
|
|
- * Provides the dictID required to decompresse frame stored within `src`.
|
|
|
+ * Provides the dictID required to decompress frame stored within `src`.
|
|
|
* If @return == 0, the dictID could not be decoded.
|
|
|
* This could for one of the following reasons :
|
|
|
* - The frame does not require a dictionary (most common case).
|
|
@@ -1176,15 +1259,14 @@ size_t ZSTD_DCtx_loadDictionary_advanced(ZSTD_DCtx* dctx,
|
|
|
ZSTD_dictLoadMethod_e dictLoadMethod,
|
|
|
ZSTD_dictContentType_e dictContentType)
|
|
|
{
|
|
|
- if (dctx->streamStage != zdss_init) return ERROR(stage_wrong);
|
|
|
- ZSTD_freeDDict(dctx->ddictLocal);
|
|
|
+ RETURN_ERROR_IF(dctx->streamStage != zdss_init, stage_wrong);
|
|
|
+ ZSTD_clearDict(dctx);
|
|
|
if (dict && dictSize >= 8) {
|
|
|
dctx->ddictLocal = ZSTD_createDDict_advanced(dict, dictSize, dictLoadMethod, dictContentType, dctx->customMem);
|
|
|
- if (dctx->ddictLocal == NULL) return ERROR(memory_allocation);
|
|
|
- } else {
|
|
|
- dctx->ddictLocal = NULL;
|
|
|
+ RETURN_ERROR_IF(dctx->ddictLocal == NULL, memory_allocation);
|
|
|
+ dctx->ddict = dctx->ddictLocal;
|
|
|
+ dctx->dictUses = ZSTD_use_indefinitely;
|
|
|
}
|
|
|
- dctx->ddict = dctx->ddictLocal;
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -1200,7 +1282,9 @@ size_t ZSTD_DCtx_loadDictionary(ZSTD_DCtx* dctx, const void* dict, size_t dictSi
|
|
|
|
|
|
size_t ZSTD_DCtx_refPrefix_advanced(ZSTD_DCtx* dctx, const void* prefix, size_t prefixSize, ZSTD_dictContentType_e dictContentType)
|
|
|
{
|
|
|
- return ZSTD_DCtx_loadDictionary_advanced(dctx, prefix, prefixSize, ZSTD_dlm_byRef, dictContentType);
|
|
|
+ FORWARD_IF_ERROR(ZSTD_DCtx_loadDictionary_advanced(dctx, prefix, prefixSize, ZSTD_dlm_byRef, dictContentType));
|
|
|
+ dctx->dictUses = ZSTD_use_once;
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
size_t ZSTD_DCtx_refPrefix(ZSTD_DCtx* dctx, const void* prefix, size_t prefixSize)
|
|
@@ -1215,9 +1299,8 @@ size_t ZSTD_DCtx_refPrefix(ZSTD_DCtx* dctx, const void* prefix, size_t prefixSiz
|
|
|
size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize)
|
|
|
{
|
|
|
DEBUGLOG(4, "ZSTD_initDStream_usingDict");
|
|
|
- zds->streamStage = zdss_init;
|
|
|
- zds->noForwardProgress = 0;
|
|
|
- CHECK_F( ZSTD_DCtx_loadDictionary(zds, dict, dictSize) );
|
|
|
+ FORWARD_IF_ERROR( ZSTD_DCtx_reset(zds, ZSTD_reset_session_only) );
|
|
|
+ FORWARD_IF_ERROR( ZSTD_DCtx_loadDictionary(zds, dict, dictSize) );
|
|
|
return ZSTD_FRAMEHEADERSIZE_PREFIX;
|
|
|
}
|
|
|
|
|
@@ -1225,7 +1308,7 @@ size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t di
|
|
|
size_t ZSTD_initDStream(ZSTD_DStream* zds)
|
|
|
{
|
|
|
DEBUGLOG(4, "ZSTD_initDStream");
|
|
|
- return ZSTD_initDStream_usingDict(zds, NULL, 0);
|
|
|
+ return ZSTD_initDStream_usingDDict(zds, NULL);
|
|
|
}
|
|
|
|
|
|
/* ZSTD_initDStream_usingDDict() :
|
|
@@ -1233,9 +1316,9 @@ size_t ZSTD_initDStream(ZSTD_DStream* zds)
|
|
|
* this function cannot fail */
|
|
|
size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* dctx, const ZSTD_DDict* ddict)
|
|
|
{
|
|
|
- size_t const initResult = ZSTD_initDStream(dctx);
|
|
|
- dctx->ddict = ddict;
|
|
|
- return initResult;
|
|
|
+ FORWARD_IF_ERROR( ZSTD_DCtx_reset(dctx, ZSTD_reset_session_only) );
|
|
|
+ FORWARD_IF_ERROR( ZSTD_DCtx_refDDict(dctx, ddict) );
|
|
|
+ return ZSTD_FRAMEHEADERSIZE_PREFIX;
|
|
|
}
|
|
|
|
|
|
/* ZSTD_resetDStream() :
|
|
@@ -1243,19 +1326,19 @@ size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* dctx, const ZSTD_DDict* ddict)
|
|
|
* this function cannot fail */
|
|
|
size_t ZSTD_resetDStream(ZSTD_DStream* dctx)
|
|
|
{
|
|
|
- DEBUGLOG(4, "ZSTD_resetDStream");
|
|
|
- dctx->streamStage = zdss_loadHeader;
|
|
|
- dctx->lhSize = dctx->inPos = dctx->outStart = dctx->outEnd = 0;
|
|
|
- dctx->legacyVersion = 0;
|
|
|
- dctx->hostageByte = 0;
|
|
|
+ FORWARD_IF_ERROR(ZSTD_DCtx_reset(dctx, ZSTD_reset_session_only));
|
|
|
return ZSTD_FRAMEHEADERSIZE_PREFIX;
|
|
|
}
|
|
|
|
|
|
|
|
|
size_t ZSTD_DCtx_refDDict(ZSTD_DCtx* dctx, const ZSTD_DDict* ddict)
|
|
|
{
|
|
|
- if (dctx->streamStage != zdss_init) return ERROR(stage_wrong);
|
|
|
- dctx->ddict = ddict;
|
|
|
+ RETURN_ERROR_IF(dctx->streamStage != zdss_init, stage_wrong);
|
|
|
+ ZSTD_clearDict(dctx);
|
|
|
+ if (ddict) {
|
|
|
+ dctx->ddict = ddict;
|
|
|
+ dctx->dictUses = ZSTD_use_indefinitely;
|
|
|
+ }
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
@@ -1267,9 +1350,9 @@ size_t ZSTD_DCtx_setMaxWindowSize(ZSTD_DCtx* dctx, size_t maxWindowSize)
|
|
|
ZSTD_bounds const bounds = ZSTD_dParam_getBounds(ZSTD_d_windowLogMax);
|
|
|
size_t const min = (size_t)1 << bounds.lowerBound;
|
|
|
size_t const max = (size_t)1 << bounds.upperBound;
|
|
|
- if (dctx->streamStage != zdss_init) return ERROR(stage_wrong);
|
|
|
- if (maxWindowSize < min) return ERROR(parameter_outOfBound);
|
|
|
- if (maxWindowSize > max) return ERROR(parameter_outOfBound);
|
|
|
+ RETURN_ERROR_IF(dctx->streamStage != zdss_init, stage_wrong);
|
|
|
+ RETURN_ERROR_IF(maxWindowSize < min, parameter_outOfBound);
|
|
|
+ RETURN_ERROR_IF(maxWindowSize > max, parameter_outOfBound);
|
|
|
dctx->maxWindowSize = maxWindowSize;
|
|
|
return 0;
|
|
|
}
|
|
@@ -1311,15 +1394,15 @@ static int ZSTD_dParam_withinBounds(ZSTD_dParameter dParam, int value)
|
|
|
}
|
|
|
|
|
|
#define CHECK_DBOUNDS(p,v) { \
|
|
|
- if (!ZSTD_dParam_withinBounds(p, v)) \
|
|
|
- return ERROR(parameter_outOfBound); \
|
|
|
+ RETURN_ERROR_IF(!ZSTD_dParam_withinBounds(p, v), parameter_outOfBound); \
|
|
|
}
|
|
|
|
|
|
size_t ZSTD_DCtx_setParameter(ZSTD_DCtx* dctx, ZSTD_dParameter dParam, int value)
|
|
|
{
|
|
|
- if (dctx->streamStage != zdss_init) return ERROR(stage_wrong);
|
|
|
+ RETURN_ERROR_IF(dctx->streamStage != zdss_init, stage_wrong);
|
|
|
switch(dParam) {
|
|
|
case ZSTD_d_windowLogMax:
|
|
|
+ if (value == 0) value = ZSTD_WINDOWLOG_LIMIT_DEFAULT;
|
|
|
CHECK_DBOUNDS(ZSTD_d_windowLogMax, value);
|
|
|
dctx->maxWindowSize = ((size_t)1) << value;
|
|
|
return 0;
|
|
@@ -1329,19 +1412,20 @@ size_t ZSTD_DCtx_setParameter(ZSTD_DCtx* dctx, ZSTD_dParameter dParam, int value
|
|
|
return 0;
|
|
|
default:;
|
|
|
}
|
|
|
- return ERROR(parameter_unsupported);
|
|
|
+ RETURN_ERROR(parameter_unsupported);
|
|
|
}
|
|
|
|
|
|
size_t ZSTD_DCtx_reset(ZSTD_DCtx* dctx, ZSTD_ResetDirective reset)
|
|
|
{
|
|
|
if ( (reset == ZSTD_reset_session_only)
|
|
|
|| (reset == ZSTD_reset_session_and_parameters) ) {
|
|
|
- (void)ZSTD_initDStream(dctx);
|
|
|
+ dctx->streamStage = zdss_init;
|
|
|
+ dctx->noForwardProgress = 0;
|
|
|
}
|
|
|
if ( (reset == ZSTD_reset_parameters)
|
|
|
|| (reset == ZSTD_reset_session_and_parameters) ) {
|
|
|
- if (dctx->streamStage != zdss_init)
|
|
|
- return ERROR(stage_wrong);
|
|
|
+ RETURN_ERROR_IF(dctx->streamStage != zdss_init, stage_wrong);
|
|
|
+ ZSTD_clearDict(dctx);
|
|
|
dctx->format = ZSTD_f_zstd1;
|
|
|
dctx->maxWindowSize = ZSTD_MAXWINDOWSIZE_DEFAULT;
|
|
|
}
|
|
@@ -1360,7 +1444,8 @@ size_t ZSTD_decodingBufferSize_min(unsigned long long windowSize, unsigned long
|
|
|
unsigned long long const neededRBSize = windowSize + blockSize + (WILDCOPY_OVERLENGTH * 2);
|
|
|
unsigned long long const neededSize = MIN(frameContentSize, neededRBSize);
|
|
|
size_t const minRBSize = (size_t) neededSize;
|
|
|
- if ((unsigned long long)minRBSize != neededSize) return ERROR(frameParameter_windowTooLarge);
|
|
|
+ RETURN_ERROR_IF((unsigned long long)minRBSize != neededSize,
|
|
|
+ frameParameter_windowTooLarge);
|
|
|
return minRBSize;
|
|
|
}
|
|
|
|
|
@@ -1378,9 +1463,9 @@ size_t ZSTD_estimateDStreamSize_fromFrame(const void* src, size_t srcSize)
|
|
|
ZSTD_frameHeader zfh;
|
|
|
size_t const err = ZSTD_getFrameHeader(&zfh, src, srcSize);
|
|
|
if (ZSTD_isError(err)) return err;
|
|
|
- if (err>0) return ERROR(srcSize_wrong);
|
|
|
- if (zfh.windowSize > windowSizeMax)
|
|
|
- return ERROR(frameParameter_windowTooLarge);
|
|
|
+ RETURN_ERROR_IF(err>0, srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(zfh.windowSize > windowSizeMax,
|
|
|
+ frameParameter_windowTooLarge);
|
|
|
return ZSTD_estimateDStreamSize((size_t)zfh.windowSize);
|
|
|
}
|
|
|
|
|
@@ -1406,16 +1491,16 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
U32 someMoreWork = 1;
|
|
|
|
|
|
DEBUGLOG(5, "ZSTD_decompressStream");
|
|
|
- if (input->pos > input->size) { /* forbidden */
|
|
|
- DEBUGLOG(5, "in: pos: %u vs size: %u",
|
|
|
- (U32)input->pos, (U32)input->size);
|
|
|
- return ERROR(srcSize_wrong);
|
|
|
- }
|
|
|
- if (output->pos > output->size) { /* forbidden */
|
|
|
- DEBUGLOG(5, "out: pos: %u vs size: %u",
|
|
|
- (U32)output->pos, (U32)output->size);
|
|
|
- return ERROR(dstSize_tooSmall);
|
|
|
- }
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ input->pos > input->size,
|
|
|
+ srcSize_wrong,
|
|
|
+ "forbidden. in: pos: %u vs size: %u",
|
|
|
+ (U32)input->pos, (U32)input->size);
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ output->pos > output->size,
|
|
|
+ dstSize_tooSmall,
|
|
|
+ "forbidden. out: pos: %u vs size: %u",
|
|
|
+ (U32)output->pos, (U32)output->size);
|
|
|
DEBUGLOG(5, "input size : %u", (U32)(input->size - input->pos));
|
|
|
|
|
|
while (someMoreWork) {
|
|
@@ -1423,15 +1508,18 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
{
|
|
|
case zdss_init :
|
|
|
DEBUGLOG(5, "stage zdss_init => transparent reset ");
|
|
|
- ZSTD_resetDStream(zds); /* transparent reset on starting decoding a new frame */
|
|
|
+ zds->streamStage = zdss_loadHeader;
|
|
|
+ zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0;
|
|
|
+ zds->legacyVersion = 0;
|
|
|
+ zds->hostageByte = 0;
|
|
|
/* fall-through */
|
|
|
|
|
|
case zdss_loadHeader :
|
|
|
DEBUGLOG(5, "stage zdss_loadHeader (srcSize : %u)", (U32)(iend - ip));
|
|
|
#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1)
|
|
|
if (zds->legacyVersion) {
|
|
|
- /* legacy support is incompatible with static dctx */
|
|
|
- if (zds->staticSize) return ERROR(memory_allocation);
|
|
|
+ RETURN_ERROR_IF(zds->staticSize, memory_allocation,
|
|
|
+ "legacy support is incompatible with static dctx");
|
|
|
{ size_t const hint = ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input);
|
|
|
if (hint==0) zds->streamStage = zdss_init;
|
|
|
return hint;
|
|
@@ -1443,12 +1531,13 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1)
|
|
|
U32 const legacyVersion = ZSTD_isLegacy(istart, iend-istart);
|
|
|
if (legacyVersion) {
|
|
|
- const void* const dict = zds->ddict ? ZSTD_DDict_dictContent(zds->ddict) : NULL;
|
|
|
- size_t const dictSize = zds->ddict ? ZSTD_DDict_dictSize(zds->ddict) : 0;
|
|
|
+ ZSTD_DDict const* const ddict = ZSTD_getDDict(zds);
|
|
|
+ const void* const dict = ddict ? ZSTD_DDict_dictContent(ddict) : NULL;
|
|
|
+ size_t const dictSize = ddict ? ZSTD_DDict_dictSize(ddict) : 0;
|
|
|
DEBUGLOG(5, "ZSTD_decompressStream: detected legacy version v0.%u", legacyVersion);
|
|
|
- /* legacy support is incompatible with static dctx */
|
|
|
- if (zds->staticSize) return ERROR(memory_allocation);
|
|
|
- CHECK_F(ZSTD_initLegacyStream(&zds->legacyContext,
|
|
|
+ RETURN_ERROR_IF(zds->staticSize, memory_allocation,
|
|
|
+ "legacy support is incompatible with static dctx");
|
|
|
+ FORWARD_IF_ERROR(ZSTD_initLegacyStream(&zds->legacyContext,
|
|
|
zds->previousLegacyVersion, legacyVersion,
|
|
|
dict, dictSize));
|
|
|
zds->legacyVersion = zds->previousLegacyVersion = legacyVersion;
|
|
@@ -1482,7 +1571,7 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
size_t const cSize = ZSTD_findFrameCompressedSize(istart, iend-istart);
|
|
|
if (cSize <= (size_t)(iend-istart)) {
|
|
|
/* shortcut : using single-pass mode */
|
|
|
- size_t const decompressedSize = ZSTD_decompress_usingDDict(zds, op, oend-op, istart, cSize, zds->ddict);
|
|
|
+ size_t const decompressedSize = ZSTD_decompress_usingDDict(zds, op, oend-op, istart, cSize, ZSTD_getDDict(zds));
|
|
|
if (ZSTD_isError(decompressedSize)) return decompressedSize;
|
|
|
DEBUGLOG(4, "shortcut to single-pass ZSTD_decompress_usingDDict()")
|
|
|
ip = istart + cSize;
|
|
@@ -1495,13 +1584,13 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
|
|
|
/* Consume header (see ZSTDds_decodeFrameHeader) */
|
|
|
DEBUGLOG(4, "Consume header");
|
|
|
- CHECK_F(ZSTD_decompressBegin_usingDDict(zds, zds->ddict));
|
|
|
+ FORWARD_IF_ERROR(ZSTD_decompressBegin_usingDDict(zds, ZSTD_getDDict(zds)));
|
|
|
|
|
|
if ((MEM_readLE32(zds->headerBuffer) & ZSTD_MAGIC_SKIPPABLE_MASK) == ZSTD_MAGIC_SKIPPABLE_START) { /* skippable frame */
|
|
|
zds->expected = MEM_readLE32(zds->headerBuffer + ZSTD_FRAMEIDSIZE);
|
|
|
zds->stage = ZSTDds_skipFrame;
|
|
|
} else {
|
|
|
- CHECK_F(ZSTD_decodeFrameHeader(zds, zds->headerBuffer, zds->lhSize));
|
|
|
+ FORWARD_IF_ERROR(ZSTD_decodeFrameHeader(zds, zds->headerBuffer, zds->lhSize));
|
|
|
zds->expected = ZSTD_blockHeaderSize;
|
|
|
zds->stage = ZSTDds_decodeBlockHeader;
|
|
|
}
|
|
@@ -1511,7 +1600,8 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
(U32)(zds->fParams.windowSize >>10),
|
|
|
(U32)(zds->maxWindowSize >> 10) );
|
|
|
zds->fParams.windowSize = MAX(zds->fParams.windowSize, 1U << ZSTD_WINDOWLOG_ABSOLUTEMIN);
|
|
|
- if (zds->fParams.windowSize > zds->maxWindowSize) return ERROR(frameParameter_windowTooLarge);
|
|
|
+ RETURN_ERROR_IF(zds->fParams.windowSize > zds->maxWindowSize,
|
|
|
+ frameParameter_windowTooLarge);
|
|
|
|
|
|
/* Adapt buffer sizes to frame header instructions */
|
|
|
{ size_t const neededInBuffSize = MAX(zds->fParams.blockSizeMax, 4 /* frame checksum */);
|
|
@@ -1525,14 +1615,15 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
if (zds->staticSize) { /* static DCtx */
|
|
|
DEBUGLOG(4, "staticSize : %u", (U32)zds->staticSize);
|
|
|
assert(zds->staticSize >= sizeof(ZSTD_DCtx)); /* controlled at init */
|
|
|
- if (bufferSize > zds->staticSize - sizeof(ZSTD_DCtx))
|
|
|
- return ERROR(memory_allocation);
|
|
|
+ RETURN_ERROR_IF(
|
|
|
+ bufferSize > zds->staticSize - sizeof(ZSTD_DCtx),
|
|
|
+ memory_allocation);
|
|
|
} else {
|
|
|
ZSTD_free(zds->inBuff, zds->customMem);
|
|
|
zds->inBuffSize = 0;
|
|
|
zds->outBuffSize = 0;
|
|
|
zds->inBuff = (char*)ZSTD_malloc(bufferSize, zds->customMem);
|
|
|
- if (zds->inBuff == NULL) return ERROR(memory_allocation);
|
|
|
+ RETURN_ERROR_IF(zds->inBuff == NULL, memory_allocation);
|
|
|
}
|
|
|
zds->inBuffSize = neededInBuffSize;
|
|
|
zds->outBuff = zds->inBuff + zds->inBuffSize;
|
|
@@ -1574,7 +1665,9 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
if (isSkipFrame) {
|
|
|
loadedSize = MIN(toLoad, (size_t)(iend-ip));
|
|
|
} else {
|
|
|
- if (toLoad > zds->inBuffSize - zds->inPos) return ERROR(corruption_detected); /* should never happen */
|
|
|
+ RETURN_ERROR_IF(toLoad > zds->inBuffSize - zds->inPos,
|
|
|
+ corruption_detected,
|
|
|
+ "should never happen");
|
|
|
loadedSize = ZSTD_limitCopy(zds->inBuff + zds->inPos, toLoad, ip, iend-ip);
|
|
|
}
|
|
|
ip += loadedSize;
|
|
@@ -1615,7 +1708,7 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
|
|
|
default:
|
|
|
assert(0); /* impossible */
|
|
|
- return ERROR(GENERIC); /* some compiler require default to do something */
|
|
|
+ RETURN_ERROR(GENERIC); /* some compiler require default to do something */
|
|
|
} }
|
|
|
|
|
|
/* result */
|
|
@@ -1624,8 +1717,8 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
|
|
|
if ((ip==istart) && (op==ostart)) { /* no forward progress */
|
|
|
zds->noForwardProgress ++;
|
|
|
if (zds->noForwardProgress >= ZSTD_NO_FORWARD_PROGRESS_MAX) {
|
|
|
- if (op==oend) return ERROR(dstSize_tooSmall);
|
|
|
- if (ip==iend) return ERROR(srcSize_wrong);
|
|
|
+ RETURN_ERROR_IF(op==oend, dstSize_tooSmall);
|
|
|
+ RETURN_ERROR_IF(ip==iend, srcSize_wrong);
|
|
|
assert(0);
|
|
|
}
|
|
|
} else {
|