2013-12-30 17:16:52 +00:00
/*
2014-12-10 08:05:44 +00:00
fuzzer . c - Fuzzer test tool for LZ4
2017-03-16 22:10:38 +00:00
Copyright ( C ) Yann Collet 2012 - 2017
2014-12-10 08:05:44 +00:00
GPL v2 License
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License along
with this program ; if not , write to the Free Software Foundation , Inc . ,
51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA .
You can contact the author at :
2016-06-29 19:48:27 +00:00
- LZ4 homepage : http : //www.lz4.org
2016-11-03 14:12:57 +00:00
- LZ4 source repo : https : //github.com/lz4/lz4
2013-12-30 17:16:52 +00:00
*/
2016-06-29 19:48:27 +00:00
/*-************************************
2015-03-07 18:30:45 +00:00
* Compiler options
2014-04-28 20:45:35 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2014-07-02 21:03:58 +00:00
# ifdef _MSC_VER /* Visual Studio */
2016-12-21 15:10:09 +00:00
# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */
2014-12-10 08:05:44 +00:00
# pragma warning(disable : 4146) /* disable: C4146: minus unsigned expression */
# pragma warning(disable : 4310) /* disable: C4310: constant char value > 127 */
2014-07-02 21:03:58 +00:00
# endif
2013-12-30 17:16:52 +00:00
2017-03-08 08:13:28 +00:00
# define LZ4_DISABLE_DEPRECATE_WARNINGS
2013-12-30 17:16:52 +00:00
2016-06-29 19:48:27 +00:00
/*-************************************
2016-11-13 01:14:57 +00:00
* Dependencies
2014-04-28 20:45:35 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-12-21 15:10:09 +00:00
# include "platform.h" /* _CRT_SECURE_NO_WARNINGS */
2016-12-21 08:44:59 +00:00
# include "util.h" /* U32 */
2013-12-30 17:16:52 +00:00
# include <stdlib.h>
2014-12-10 08:05:44 +00:00
# include <stdio.h> /* fgets, sscanf */
# include <string.h> /* strcmp */
2016-09-03 02:48:24 +00:00
# include <time.h> /* clock_t, clock, CLOCKS_PER_SEC */
2017-03-16 09:16:24 +00:00
# define LZ4_HC_STATIC_LINKING_ONLY
2013-12-30 17:16:52 +00:00
# include "lz4hc.h"
2016-06-29 19:48:27 +00:00
# define XXH_STATIC_LINKING_ONLY
2014-04-28 20:45:35 +00:00
# include "xxhash.h"
2016-12-21 09:39:27 +00:00
/*-************************************
* Basic Types
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if !defined(__cplusplus) && !(defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */ )
typedef size_t uintptr_t ; /* true on most systems, except OpenVMS-64 (which doesn't need address overflow test) */
# endif
2016-08-10 07:14:11 +00:00
/*-************************************
2015-03-07 18:30:45 +00:00
* Constants
2014-04-28 20:45:35 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2014-04-22 23:54:32 +00:00
# define NB_ATTEMPTS (1<<16)
2014-04-28 20:45:35 +00:00
# define COMPRESSIBLE_NOISE_LENGTH (1 << 21)
# define FUZ_MAX_BLOCK_SIZE (1 << 17)
# define FUZ_MAX_DICT_SIZE (1 << 15)
2014-12-10 08:05:44 +00:00
# define FUZ_COMPRESSIBILITY_DEFAULT 60
2013-12-30 17:16:52 +00:00
# define PRIME1 2654435761U
# define PRIME2 2246822519U
# define PRIME3 3266489917U
2014-07-02 08:38:34 +00:00
# define KB *(1U<<10)
# define MB *(1U<<20)
# define GB *(1U<<30)
2016-08-10 07:14:11 +00:00
/*-***************************************
2015-03-07 18:30:45 +00:00
* Macros
2014-08-30 11:32:09 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-11-13 01:14:57 +00:00
# define DISPLAY(...) fprintf(stdout, __VA_ARGS__)
2014-10-18 12:27:32 +00:00
# define DISPLAYLEVEL(l, ...) if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); }
static int g_displayLevel = 2 ;
2016-11-13 01:14:57 +00:00
# define MIN(a,b) ( (a) < (b) ? (a) : (b) )
2014-04-29 20:39:08 +00:00
2016-08-10 07:14:11 +00:00
/*-*******************************************************
2015-03-07 18:30:45 +00:00
* Fuzzer functions
2014-04-28 20:45:35 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-09-03 02:48:24 +00:00
static clock_t FUZ_GetClockSpan ( clock_t clockStart )
2013-12-30 17:16:52 +00:00
{
2016-09-03 02:48:24 +00:00
return clock ( ) - clockStart ; /* works even if overflow; max span ~ 30mn */
2013-12-30 17:16:52 +00:00
}
2016-11-13 01:14:57 +00:00
static void FUZ_displayUpdate ( unsigned testNb )
{
static clock_t g_time = 0 ;
static const clock_t g_refreshRate = CLOCKS_PER_SEC / 5 ;
if ( ( FUZ_GetClockSpan ( g_time ) > g_refreshRate ) | | ( g_displayLevel > = 4 ) ) {
g_time = clock ( ) ;
DISPLAY ( " \r %5u " , testNb ) ;
fflush ( stdout ) ;
}
}
2014-10-19 12:47:22 +00:00
static U32 FUZ_rotl32 ( U32 u32 , U32 nbBits )
{
return ( ( u32 < < nbBits ) | ( u32 > > ( 32 - nbBits ) ) ) ;
}
static U32 FUZ_rand ( U32 * src )
2013-12-30 17:16:52 +00:00
{
2014-04-28 22:49:31 +00:00
U32 rand32 = * src ;
rand32 * = PRIME1 ;
2015-04-24 09:15:12 +00:00
rand32 ^ = PRIME2 ;
2014-04-28 22:49:31 +00:00
rand32 = FUZ_rotl32 ( rand32 , 13 ) ;
* src = rand32 ;
2015-04-24 09:15:12 +00:00
return rand32 ;
2013-12-30 17:16:52 +00:00
}
2014-04-28 20:45:35 +00:00
# define FUZ_RAND15BITS ((FUZ_rand(seed) >> 3) & 32767)
2014-08-30 11:32:09 +00:00
# define FUZ_RANDLENGTH ( ((FUZ_rand(seed) >> 7) & 3) ? (FUZ_rand(seed) % 15) : (FUZ_rand(seed) % 510) + 15)
2014-12-10 08:05:44 +00:00
static void FUZ_fillCompressibleNoiseBuffer ( void * buffer , size_t bufferSize , double proba , U32 * seed )
2013-12-30 17:16:52 +00:00
{
2016-11-13 01:14:57 +00:00
BYTE * const BBuffer = ( BYTE * ) buffer ;
2014-12-10 08:05:44 +00:00
size_t pos = 0 ;
2016-11-13 01:14:57 +00:00
U32 const P32 = ( U32 ) ( 32768 * proba ) ;
2014-04-28 20:45:35 +00:00
2015-04-24 09:15:12 +00:00
/* First Bytes */
while ( pos < 20 )
BBuffer [ pos + + ] = ( BYTE ) ( FUZ_rand ( seed ) ) ;
2014-04-28 20:45:35 +00:00
2016-11-13 01:14:57 +00:00
while ( pos < bufferSize ) {
2015-04-24 09:15:12 +00:00
/* Select : Literal (noise) or copy (within 64K) */
2016-11-13 01:14:57 +00:00
if ( FUZ_RAND15BITS < P32 ) {
2015-04-24 09:15:12 +00:00
/* Copy (within 64K) */
2016-11-13 01:14:57 +00:00
size_t const length = FUZ_RANDLENGTH + 4 ;
size_t const d = MIN ( pos + length , bufferSize ) ;
size_t match ;
2014-12-10 08:05:44 +00:00
size_t offset = FUZ_RAND15BITS + 1 ;
2015-04-24 09:15:12 +00:00
while ( offset > pos ) offset > > = 1 ;
2014-12-10 08:05:44 +00:00
match = pos - offset ;
while ( pos < d ) BBuffer [ pos + + ] = BBuffer [ match + + ] ;
2016-11-13 01:14:57 +00:00
} else {
2015-04-24 09:15:12 +00:00
/* Literal (noise) */
2016-11-13 01:14:57 +00:00
size_t const length = FUZ_RANDLENGTH ;
size_t const d = MIN ( pos + length , bufferSize ) ;
2014-04-28 20:45:35 +00:00
while ( pos < d ) BBuffer [ pos + + ] = ( BYTE ) ( FUZ_rand ( seed ) > > 5 ) ;
}
}
2013-12-30 17:16:52 +00:00
}
2014-07-02 17:02:29 +00:00
# define MAX_NB_BUFF_I134 150
2014-07-05 11:50:05 +00:00
# define BLOCKSIZE_I134 (32 MB)
2016-08-11 10:18:45 +00:00
/*! FUZ_AddressOverflow() :
* Aggressively pushes memory allocation limits ,
* and generates patterns which create address space overflow .
* only possible in 32 - bits mode */
2014-10-10 19:58:42 +00:00
static int FUZ_AddressOverflow ( void )
2014-07-02 08:38:34 +00:00
{
2015-03-15 00:42:27 +00:00
char * buffers [ MAX_NB_BUFF_I134 + 1 ] ;
2016-11-17 21:02:06 +00:00
int nbBuff = 0 ;
2014-10-25 19:52:10 +00:00
int highAddress = 0 ;
2014-07-02 08:38:34 +00:00
2015-06-29 08:31:41 +00:00
DISPLAY ( " Overflow tests : " ) ;
2014-07-02 08:38:34 +00:00
2015-04-10 22:42:17 +00:00
/* Only possible in 32-bits */
2016-11-13 01:14:57 +00:00
if ( sizeof ( void * ) = = 8 ) {
2015-06-29 08:31:41 +00:00
DISPLAY ( " 64 bits mode : no overflow \n " ) ;
2014-07-05 11:50:05 +00:00
fflush ( stdout ) ;
2014-10-25 19:52:10 +00:00
return 0 ;
2014-07-02 08:38:34 +00:00
}
2014-07-05 11:50:05 +00:00
2014-10-25 19:52:10 +00:00
buffers [ 0 ] = ( char * ) malloc ( BLOCKSIZE_I134 ) ;
buffers [ 1 ] = ( char * ) malloc ( BLOCKSIZE_I134 ) ;
2016-08-11 10:18:45 +00:00
if ( ( ! buffers [ 0 ] ) | | ( ! buffers [ 1 ] ) ) {
2016-11-17 21:02:06 +00:00
free ( buffers [ 0 ] ) ; free ( buffers [ 1 ] ) ;
2015-06-29 08:31:41 +00:00
DISPLAY ( " not enough memory for tests \n " ) ;
2014-10-25 19:52:10 +00:00
return 0 ;
}
2015-06-29 08:31:41 +00:00
2016-08-11 10:18:45 +00:00
for ( nbBuff = 2 ; nbBuff < MAX_NB_BUFF_I134 ; nbBuff + + ) {
2016-11-17 21:02:06 +00:00
DISPLAY ( " %3i \b \b \b \b " , nbBuff ) ; fflush ( stdout ) ;
2014-10-25 19:52:10 +00:00
buffers [ nbBuff ] = ( char * ) malloc ( BLOCKSIZE_I134 ) ;
2015-06-29 08:31:41 +00:00
if ( buffers [ nbBuff ] = = NULL ) goto _endOfTests ;
2014-10-25 19:52:10 +00:00
2016-11-14 04:50:58 +00:00
if ( ( ( uintptr_t ) buffers [ nbBuff ] > ( uintptr_t ) 0x80000000 ) & & ( ! highAddress ) ) {
2015-06-29 08:31:41 +00:00
DISPLAY ( " high address detected : " ) ;
2014-10-25 19:52:10 +00:00
fflush ( stdout ) ;
highAddress = 1 ;
}
2016-11-14 04:50:58 +00:00
{ size_t const sizeToGenerateOverflow = ( size_t ) ( - ( ( uintptr_t ) buffers [ nbBuff - 1 ] ) + 512 ) ;
2016-11-17 21:02:06 +00:00
unsigned const nbOf255 = ( unsigned ) ( ( sizeToGenerateOverflow / 255 ) + 1 ) ;
2016-08-11 10:18:45 +00:00
char * const input = buffers [ nbBuff - 1 ] ;
2014-10-25 19:52:10 +00:00
char * output = buffers [ nbBuff ] ;
int r ;
2016-08-11 10:18:45 +00:00
input [ 0 ] = ( char ) 0xF0 ; /* Literal length overflow */
2014-10-25 19:52:10 +00:00
input [ 1 ] = ( char ) 0xFF ;
input [ 2 ] = ( char ) 0xFF ;
input [ 3 ] = ( char ) 0xFF ;
2016-11-17 21:02:06 +00:00
{ unsigned u ; for ( u = 4 ; u < = nbOf255 + 4 ; u + + ) input [ u ] = ( char ) 0xff ; }
2014-10-25 19:52:10 +00:00
r = LZ4_decompress_safe ( input , output , nbOf255 + 64 , BLOCKSIZE_I134 ) ;
2016-11-17 21:02:06 +00:00
if ( r > 0 ) { DISPLAY ( " LZ4_decompress_safe = %i \n " , r ) ; goto _overflowError ; }
2016-08-11 10:18:45 +00:00
input [ 0 ] = ( char ) 0x1F ; /* Match length overflow */
2014-10-25 19:52:10 +00:00
input [ 1 ] = ( char ) 0x01 ;
input [ 2 ] = ( char ) 0x01 ;
input [ 3 ] = ( char ) 0x00 ;
r = LZ4_decompress_safe ( input , output , nbOf255 + 64 , BLOCKSIZE_I134 ) ;
2016-11-17 21:02:06 +00:00
if ( r > 0 ) { DISPLAY ( " LZ4_decompress_safe = %i \n " , r ) ; goto _overflowError ; }
2014-10-25 19:52:10 +00:00
2016-08-11 10:18:45 +00:00
output = buffers [ nbBuff - 2 ] ; /* Reverse in/out pointer order */
input [ 0 ] = ( char ) 0xF0 ; /* Literal length overflow */
2014-10-25 19:52:10 +00:00
input [ 1 ] = ( char ) 0xFF ;
input [ 2 ] = ( char ) 0xFF ;
input [ 3 ] = ( char ) 0xFF ;
r = LZ4_decompress_safe ( input , output , nbOf255 + 64 , BLOCKSIZE_I134 ) ;
if ( r > 0 ) goto _overflowError ;
2016-08-11 10:18:45 +00:00
input [ 0 ] = ( char ) 0x1F ; /* Match length overflow */
2014-10-25 19:52:10 +00:00
input [ 1 ] = ( char ) 0x01 ;
input [ 2 ] = ( char ) 0x01 ;
input [ 3 ] = ( char ) 0x00 ;
r = LZ4_decompress_safe ( input , output , nbOf255 + 64 , BLOCKSIZE_I134 ) ;
if ( r > 0 ) goto _overflowError ;
}
2014-07-02 08:38:34 +00:00
}
2014-10-25 19:52:10 +00:00
nbBuff + + ;
2014-07-05 11:50:05 +00:00
_endOfTests :
2016-11-17 21:02:06 +00:00
{ int i ; for ( i = 0 ; i < nbBuff ; i + + ) free ( buffers [ i ] ) ; }
2015-06-29 08:31:41 +00:00
if ( ! highAddress ) DISPLAY ( " high address not possible \n " ) ;
else DISPLAY ( " all overflows correctly detected \n " ) ;
2014-10-25 19:52:10 +00:00
return 0 ;
2014-07-05 11:50:05 +00:00
_overflowError :
2015-06-29 08:31:41 +00:00
DISPLAY ( " Address space overflow error !! \n " ) ;
2014-10-25 19:52:10 +00:00
exit ( 1 ) ;
2014-07-02 08:38:34 +00:00
}
2016-08-11 10:18:45 +00:00
/*! FUZ_findDiff() :
* find the first different byte between buff1 and buff2 .
* presumes buff1 ! = buff2 .
* presumes a difference exists before end of either buffer .
* Typically invoked after a checksum mismatch .
*/
2015-04-24 09:15:12 +00:00
static void FUZ_findDiff ( const void * buff1 , const void * buff2 )
{
2016-08-11 10:18:45 +00:00
const BYTE * const b1 = ( const BYTE * ) buff1 ;
const BYTE * const b2 = ( const BYTE * ) buff2 ;
2016-11-13 01:14:57 +00:00
size_t u = 0 ;
while ( b1 [ u ] = = b2 [ u ] ) u + + ;
DISPLAY ( " Wrong Byte at position %u \n " , ( unsigned ) u ) ;
2015-04-24 09:15:12 +00:00
}
2016-09-03 02:48:24 +00:00
static int FUZ_test ( U32 seed , U32 nbCycles , const U32 startCycle , const double compressibility , U32 duration_s )
2014-10-10 19:58:42 +00:00
{
unsigned long long bytes = 0 ;
unsigned long long cbytes = 0 ;
unsigned long long hcbytes = 0 ;
unsigned long long ccbytes = 0 ;
2016-11-13 01:14:57 +00:00
void * const CNBuffer = malloc ( COMPRESSIBLE_NOISE_LENGTH ) ;
2016-11-12 16:48:42 +00:00
size_t const compressedBufferSize = LZ4_compressBound ( FUZ_MAX_BLOCK_SIZE ) ;
2016-11-13 01:14:57 +00:00
char * const compressedBuffer = ( char * ) malloc ( compressedBufferSize ) ;
char * const decodedBuffer = ( char * ) malloc ( FUZ_MAX_DICT_SIZE + FUZ_MAX_BLOCK_SIZE ) ;
void * const stateLZ4 = malloc ( LZ4_sizeofState ( ) ) ;
void * const stateLZ4HC = malloc ( LZ4_sizeofStateHC ( ) ) ;
2014-10-10 19:58:42 +00:00
LZ4_stream_t LZ4dict ;
2014-10-25 19:52:10 +00:00
LZ4_streamHC_t LZ4dictHC ;
2014-10-19 15:06:33 +00:00
U32 coreRandState = seed ;
2016-09-03 02:48:24 +00:00
clock_t const clockStart = clock ( ) ;
clock_t const clockDuration = ( clock_t ) duration_s * CLOCKS_PER_SEC ;
2016-11-13 01:14:57 +00:00
int result = 0 ;
unsigned cycleNb ;
# define FUZ_CHECKTEST(cond, ...) if (cond) { printf("Test %u : ", testNb); printf(__VA_ARGS__); \
printf ( " (seed %u, cycle %u) \n " , seed , cycleNb ) ; goto _output_error ; }
# define FUZ_DISPLAYTEST { testNb++; g_displayLevel>=4 ? printf("%2u\b\b", testNb), fflush(stdout) : 0; }
2014-10-10 19:58:42 +00:00
2015-04-10 22:42:17 +00:00
/* init */
2016-11-13 01:14:57 +00:00
if ( ! CNBuffer | | ! compressedBuffer | | ! decodedBuffer ) {
DISPLAY ( " Not enough memory to start fuzzer tests " ) ;
goto _output_error ;
}
2014-10-10 19:58:42 +00:00
memset ( & LZ4dict , 0 , sizeof ( LZ4dict ) ) ;
2016-11-13 01:14:57 +00:00
{ U32 randState = coreRandState ^ PRIME3 ;
FUZ_fillCompressibleNoiseBuffer ( CNBuffer , COMPRESSIBLE_NOISE_LENGTH , compressibility , & randState ) ;
}
2014-10-10 19:58:42 +00:00
2015-04-10 22:42:17 +00:00
/* move to startCycle */
2016-11-14 18:16:11 +00:00
for ( cycleNb = 0 ; cycleNb < startCycle ; cycleNb + + )
( void ) FUZ_rand ( & coreRandState ) ; /* sync coreRandState */
2014-10-10 19:58:42 +00:00
2015-04-10 22:42:17 +00:00
/* Main test loop */
2016-11-13 01:14:57 +00:00
for ( cycleNb = startCycle ;
( cycleNb < nbCycles ) | | ( FUZ_GetClockSpan ( clockStart ) < clockDuration ) ;
cycleNb + + ) {
2014-10-20 00:08:21 +00:00
U32 testNb = 0 ;
2016-11-13 01:14:57 +00:00
U32 randState = FUZ_rand ( & coreRandState ) ^ PRIME3 ;
int const blockSize = ( FUZ_rand ( & randState ) % ( FUZ_MAX_BLOCK_SIZE - 1 ) ) + 1 ;
int const blockStart = FUZ_rand ( & randState ) % ( COMPRESSIBLE_NOISE_LENGTH - blockSize ) ;
int const dictSizeRand = FUZ_rand ( & randState ) % FUZ_MAX_DICT_SIZE ;
int const dictSize = MIN ( dictSizeRand , blockStart ) ;
2016-12-28 16:47:10 +00:00
int const compressionLevel = FUZ_rand ( & randState ) % ( LZ4HC_CLEVEL_MAX + 1 ) ;
2016-11-13 01:14:57 +00:00
char * const block = ( ( char * ) CNBuffer ) + blockStart ;
const char * dict = block - dictSize ;
int compressedSize , HCcompressedSize ;
2014-10-10 19:58:42 +00:00
int blockContinueCompressedSize ;
2016-11-13 01:14:57 +00:00
U32 const crcOrig = XXH32 ( block , blockSize , 0 ) ;
U32 crcCheck ;
int ret ;
2014-10-10 19:58:42 +00:00
FUZ_displayUpdate ( cycleNb ) ;
/* Compression tests */
2015-05-06 01:29:04 +00:00
/* Test compression destSize */
FUZ_DISPLAYTEST ;
2016-08-11 10:18:45 +00:00
{ int srcSize = blockSize ;
2016-11-13 01:14:57 +00:00
int const targetSize = srcSize * ( ( FUZ_rand ( & randState ) & 127 ) + 1 ) > > 7 ;
2015-05-06 01:29:04 +00:00
char endCheck = FUZ_rand ( & randState ) & 255 ;
compressedBuffer [ targetSize ] = endCheck ;
ret = LZ4_compress_destSize ( block , compressedBuffer , & srcSize , targetSize ) ;
FUZ_CHECKTEST ( ret > targetSize , " LZ4_compress_destSize() result larger than dst buffer ! " ) ;
FUZ_CHECKTEST ( compressedBuffer [ targetSize ] ! = endCheck , " LZ4_compress_destSize() overwrite dst buffer ! " ) ;
FUZ_CHECKTEST ( srcSize > blockSize , " LZ4_compress_destSize() fed more than src buffer ! " ) ;
2015-06-27 21:44:13 +00:00
DISPLAYLEVEL ( 5 , " destSize : %7i/%7i; content%7i/%7i " , ret , targetSize , srcSize , blockSize ) ;
2016-08-11 10:18:45 +00:00
if ( targetSize > 0 ) {
2015-05-06 01:29:04 +00:00
/* check correctness */
2016-11-13 01:14:57 +00:00
U32 const crcBase = XXH32 ( block , srcSize , 0 ) ;
char const canary = FUZ_rand ( & randState ) & 255 ;
FUZ_CHECKTEST ( ( ret = = 0 ) , " LZ4_compress_destSize() compression failed " ) ;
2015-05-06 01:29:04 +00:00
FUZ_DISPLAYTEST ;
compressedSize = ret ;
2016-11-13 01:14:57 +00:00
decodedBuffer [ srcSize ] = canary ;
2015-05-06 01:29:04 +00:00
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize , srcSize ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_safe() failed on data compressed by LZ4_compress_destSize " ) ;
FUZ_CHECKTEST ( ret ! = srcSize , " LZ4_decompress_safe() failed : did not fully decompressed data " ) ;
2016-11-13 01:14:57 +00:00
FUZ_CHECKTEST ( decodedBuffer [ srcSize ] ! = canary , " LZ4_decompress_safe() overwrite dst buffer ! " ) ;
{ U32 const crcDec = XXH32 ( decodedBuffer , srcSize , 0 ) ;
FUZ_CHECKTEST ( crcDec ! = crcBase , " LZ4_decompress_safe() corrupted decoded data " ) ; }
2015-05-06 01:29:04 +00:00
2015-06-27 21:44:13 +00:00
DISPLAYLEVEL ( 5 , " OK \n " ) ;
2017-12-22 11:47:59 +00:00
} else {
2015-06-27 21:44:13 +00:00
DISPLAYLEVEL ( 5 , " \n " ) ;
2017-12-22 11:47:59 +00:00
} }
2015-05-06 01:29:04 +00:00
2017-03-08 08:13:28 +00:00
/* Test compression HC destSize */
FUZ_DISPLAYTEST ;
{ int srcSize = blockSize ;
int const targetSize = srcSize * ( ( FUZ_rand ( & randState ) & 127 ) + 1 ) > > 7 ;
2017-03-16 09:16:24 +00:00
char const endCheck = FUZ_rand ( & randState ) & 255 ;
void * ctx = LZ4_createHC ( block ) ;
2017-03-08 08:13:28 +00:00
FUZ_CHECKTEST ( ctx = = NULL , " LZ4_createHC() allocation failed " ) ;
compressedBuffer [ targetSize ] = endCheck ;
2017-03-16 09:16:24 +00:00
ret = LZ4_compress_HC_destSize ( ctx , block , compressedBuffer , & srcSize , targetSize , compressionLevel ) ;
2017-12-22 11:47:59 +00:00
DISPLAYLEVEL ( 5 , " LZ4_compress_HC_destSize(%i): destSize : %7i/%7i; content%7i/%7i " ,
compressionLevel , ret , targetSize , srcSize , blockSize ) ;
2017-03-08 08:13:28 +00:00
LZ4_freeHC ( ctx ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret > targetSize , " LZ4_compress_HC_destSize() result larger than dst buffer ! " ) ;
FUZ_CHECKTEST ( compressedBuffer [ targetSize ] ! = endCheck , " LZ4_compress_HC_destSize() overwrite dst buffer ! " ) ;
FUZ_CHECKTEST ( srcSize > blockSize , " LZ4_compress_HC_destSize() fed more than src buffer ! " ) ;
2017-12-22 11:47:59 +00:00
DISPLAYLEVEL ( 5 , " LZ4_compress_HC_destSize(%i): destSize : %7i/%7i; content%7i/%7i " ,
compressionLevel , ret , targetSize , srcSize , blockSize ) ;
2017-03-08 08:13:28 +00:00
if ( targetSize > 0 ) {
/* check correctness */
U32 const crcBase = XXH32 ( block , srcSize , 0 ) ;
char const canary = FUZ_rand ( & randState ) & 255 ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ( ret = = 0 ) , " LZ4_compress_HC_destSize() compression failed " ) ;
2017-03-08 08:13:28 +00:00
FUZ_DISPLAYTEST ;
compressedSize = ret ;
decodedBuffer [ srcSize ] = canary ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize , srcSize ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_safe() failed on data compressed by LZ4_compressHC_destSize " ) ;
FUZ_CHECKTEST ( ret ! = srcSize , " LZ4_decompress_safe() failed : did not fully decompressed data " ) ;
FUZ_CHECKTEST ( decodedBuffer [ srcSize ] ! = canary , " LZ4_decompress_safe() overwrite dst buffer ! " ) ;
2017-11-03 18:28:28 +00:00
{ U32 const crcDec = XXH32 ( decodedBuffer , srcSize , 0 ) ;
FUZ_CHECKTEST ( crcDec ! = crcBase , " LZ4_decompress_safe() corrupted decoded data " ) ;
}
2017-03-08 08:13:28 +00:00
DISPLAYLEVEL ( 5 , " OK \n " ) ;
2017-12-22 11:47:59 +00:00
} else {
2017-03-08 08:13:28 +00:00
DISPLAYLEVEL ( 5 , " \n " ) ;
2017-12-22 11:47:59 +00:00
} }
2017-03-08 08:13:28 +00:00
2015-04-10 22:42:17 +00:00
/* Test compression HC */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-12-28 16:47:10 +00:00
ret = LZ4_compress_HC ( block , compressedBuffer , blockSize , ( int ) compressedBufferSize , compressionLevel ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_HC() failed " ) ;
2014-10-10 19:58:42 +00:00
HCcompressedSize = ret ;
2015-04-24 09:15:12 +00:00
/* Test compression HC using external state */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-12-28 16:47:10 +00:00
ret = LZ4_compress_HC_extStateHC ( stateLZ4HC , block , compressedBuffer , blockSize , ( int ) compressedBufferSize , compressionLevel ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_HC_extStateHC() failed " ) ;
2014-10-10 19:58:42 +00:00
2015-04-24 09:15:12 +00:00
/* Test compression using external state */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-12-28 12:08:38 +00:00
ret = LZ4_compress_fast_extState ( stateLZ4 , block , compressedBuffer , blockSize , ( int ) compressedBufferSize , 8 ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_fast_extState() failed " ) ;
2014-10-10 19:58:42 +00:00
2015-04-24 09:15:12 +00:00
/* Test compression */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-11-14 15:10:31 +00:00
ret = LZ4_compress_default ( block , compressedBuffer , blockSize , ( int ) compressedBufferSize ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_default() failed " ) ;
2014-10-10 19:58:42 +00:00
compressedSize = ret ;
/* Decompression tests */
2015-04-24 09:15:12 +00:00
/* Test decoding with output size being exactly what's necessary => must work */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
ret = LZ4_decompress_fast ( compressedBuffer , decodedBuffer , blockSize ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_fast failed despite correct space " ) ;
FUZ_CHECKTEST ( ret ! = compressedSize , " LZ4_decompress_fast failed : did not fully read compressed data " ) ;
crcCheck = XXH32 ( decodedBuffer , blockSize , 0 ) ;
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_fast corrupted decoded data " ) ;
2015-04-24 09:15:12 +00:00
/* Test decoding with one byte missing => must fail */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize - 1 ] = 0 ;
ret = LZ4_decompress_fast ( compressedBuffer , decodedBuffer , blockSize - 1 ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_fast should have failed, due to Output Size being too small " ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize - 1 ] , " LZ4_decompress_fast overrun specified output buffer " ) ;
2015-04-24 09:15:12 +00:00
/* Test decoding with one byte too much => must fail */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
ret = LZ4_decompress_fast ( compressedBuffer , decodedBuffer , blockSize + 1 ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_fast should have failed, due to Output Size being too large " ) ;
2015-04-24 09:15:12 +00:00
/* Test decoding with output size exactly what's necessary => must work */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize ] = 0 ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize , blockSize ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_safe failed despite sufficient space " ) ;
FUZ_CHECKTEST ( ret ! = blockSize , " LZ4_decompress_safe did not regenerate original data " ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize ] , " LZ4_decompress_safe overrun specified output buffer size " ) ;
crcCheck = XXH32 ( decodedBuffer , blockSize , 0 ) ;
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_safe corrupted decoded data " ) ;
// Test decoding with more than enough output size => must work
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize ] = 0 ;
decodedBuffer [ blockSize + 1 ] = 0 ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize , blockSize + 1 ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_safe failed despite amply sufficient space " ) ;
FUZ_CHECKTEST ( ret ! = blockSize , " LZ4_decompress_safe did not regenerate original data " ) ;
//FUZ_CHECKTEST(decodedBuffer[blockSize], "LZ4_decompress_safe wrote more than (unknown) target size"); // well, is that an issue ?
FUZ_CHECKTEST ( decodedBuffer [ blockSize + 1 ] , " LZ4_decompress_safe overrun specified output buffer size " ) ;
crcCheck = XXH32 ( decodedBuffer , blockSize , 0 ) ;
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_safe corrupted decoded data " ) ;
// Test decoding with output size being one byte too short => must fail
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize - 1 ] = 0 ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize , blockSize - 1 ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_safe should have failed, due to Output Size being one byte too short " ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize - 1 ] , " LZ4_decompress_safe overrun specified output buffer size " ) ;
// Test decoding with output size being 10 bytes too short => must fail
FUZ_DISPLAYTEST ;
2017-11-03 18:28:28 +00:00
if ( blockSize > 10 ) {
2014-10-10 19:58:42 +00:00
decodedBuffer [ blockSize - 10 ] = 0 ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize , blockSize - 10 ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_safe should have failed, due to Output Size being 10 bytes too short " ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize - 10 ] , " LZ4_decompress_safe overrun specified output buffer size " ) ;
2014-05-19 23:40:29 +00:00
}
2014-10-10 19:58:42 +00:00
// Test decoding with input size being one byte too short => must fail
FUZ_DISPLAYTEST ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize - 1 , blockSize ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_safe should have failed, due to input size being one byte too short (blockSize=%i, ret=%i, compressedSize=%i) " , blockSize , ret , compressedSize ) ;
// Test decoding with input size being one byte too large => must fail
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize ] = 0 ;
ret = LZ4_decompress_safe ( compressedBuffer , decodedBuffer , compressedSize + 1 , blockSize ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_safe should have failed, due to input size being too large " ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize ] , " LZ4_decompress_safe overrun specified output buffer size " ) ;
// Test partial decoding with target output size being max/2 => must work
FUZ_DISPLAYTEST ;
ret = LZ4_decompress_safe_partial ( compressedBuffer , decodedBuffer , compressedSize , blockSize / 2 , blockSize ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_safe_partial failed despite sufficient space " ) ;
// Test partial decoding with target output size being just below max => must work
FUZ_DISPLAYTEST ;
ret = LZ4_decompress_safe_partial ( compressedBuffer , decodedBuffer , compressedSize , blockSize - 3 , blockSize ) ;
FUZ_CHECKTEST ( ret < 0 , " LZ4_decompress_safe_partial failed despite sufficient space " ) ;
/* Test Compression with limited output size */
2014-12-10 08:05:44 +00:00
/* Test compression with output size being exactly what's necessary (should work) */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_default ( block , compressedBuffer , blockSize , compressedSize ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_default() failed despite sufficient space " ) ;
2014-10-10 19:58:42 +00:00
2014-12-10 08:05:44 +00:00
/* Test compression with output size being exactly what's necessary and external state (should work) */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_fast_extState ( stateLZ4 , block , compressedBuffer , blockSize , compressedSize , 1 ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_fast_extState() failed despite sufficient space " ) ;
2014-10-10 19:58:42 +00:00
2014-12-10 08:05:44 +00:00
/* Test HC compression with output size being exactly what's necessary (should work) */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-12-28 16:47:10 +00:00
ret = LZ4_compress_HC ( block , compressedBuffer , blockSize , HCcompressedSize , compressionLevel ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_HC() failed despite sufficient space " ) ;
2014-10-10 19:58:42 +00:00
2014-12-10 08:05:44 +00:00
/* Test HC compression with output size being exactly what's necessary (should work) */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2016-12-28 16:47:10 +00:00
ret = LZ4_compress_HC_extStateHC ( stateLZ4HC , block , compressedBuffer , blockSize , HCcompressedSize , compressionLevel ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret = = 0 , " LZ4_compress_HC_extStateHC() failed despite sufficient space " ) ;
2014-10-10 19:58:42 +00:00
2014-12-10 16:58:15 +00:00
/* Test compression with missing bytes into output buffer => must fail */
2014-12-10 08:05:44 +00:00
FUZ_DISPLAYTEST ;
2016-08-11 10:18:45 +00:00
{ int missingBytes = ( FUZ_rand ( & randState ) % 0x3F ) + 1 ;
2014-12-10 08:05:44 +00:00
if ( missingBytes > = compressedSize ) missingBytes = compressedSize - 1 ;
2014-12-10 16:58:15 +00:00
missingBytes + = ! missingBytes ; /* avoid special case missingBytes==0 */
2014-12-10 08:05:44 +00:00
compressedBuffer [ compressedSize - missingBytes ] = 0 ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_default ( block , compressedBuffer , blockSize , compressedSize - missingBytes ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret , " LZ4_compress_default should have failed (output buffer too small by %i byte) " , missingBytes ) ;
FUZ_CHECKTEST ( compressedBuffer [ compressedSize - missingBytes ] , " LZ4_compress_default overran output buffer ! (%i missingBytes) " , missingBytes )
2014-12-10 08:05:44 +00:00
}
2014-12-10 16:58:15 +00:00
/* Test HC compression with missing bytes into output buffer => must fail */
FUZ_DISPLAYTEST ;
2016-08-11 10:18:45 +00:00
{ int missingBytes = ( FUZ_rand ( & randState ) % 0x3F ) + 1 ;
2014-12-10 16:58:15 +00:00
if ( missingBytes > = HCcompressedSize ) missingBytes = HCcompressedSize - 1 ;
missingBytes + = ! missingBytes ; /* avoid special case missingBytes==0 */
compressedBuffer [ HCcompressedSize - missingBytes ] = 0 ;
2016-12-28 16:47:10 +00:00
ret = LZ4_compress_HC ( block , compressedBuffer , blockSize , HCcompressedSize - missingBytes , compressionLevel ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret , " LZ4_compress_HC should have failed (output buffer too small by %i byte) " , missingBytes ) ;
FUZ_CHECKTEST ( compressedBuffer [ HCcompressedSize - missingBytes ] , " LZ4_compress_HC overran output buffer ! (%i missingBytes) " , missingBytes )
2014-12-10 16:58:15 +00:00
}
2014-10-10 19:58:42 +00:00
2016-08-11 10:18:45 +00:00
/*-******************/
2014-12-10 08:05:44 +00:00
/* Dictionary tests */
2016-08-11 10:18:45 +00:00
/*-******************/
2014-12-10 08:05:44 +00:00
/* Compress using dictionary */
FUZ_DISPLAYTEST ;
2016-08-11 10:18:45 +00:00
{ LZ4_stream_t LZ4_stream ;
2015-04-11 17:59:22 +00:00
LZ4_resetStream ( & LZ4_stream ) ;
2016-11-14 15:10:31 +00:00
LZ4_compress_fast_continue ( & LZ4_stream , dict , compressedBuffer , dictSize , ( int ) compressedBufferSize , 1 ) ; /* Just to fill hash tables */
blockContinueCompressedSize = LZ4_compress_fast_continue ( & LZ4_stream , block , compressedBuffer , blockSize , ( int ) compressedBufferSize , 1 ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( blockContinueCompressedSize = = 0 , " LZ4_compress_fast_continue failed " ) ;
2015-04-11 17:59:22 +00:00
}
2014-10-10 19:58:42 +00:00
2014-12-10 08:05:44 +00:00
/* Decompress with dictionary as prefix */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
memcpy ( decodedBuffer , dict , dictSize ) ;
2015-04-11 17:59:22 +00:00
ret = LZ4_decompress_fast_usingDict ( compressedBuffer , decodedBuffer + dictSize , blockSize , decodedBuffer , dictSize ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret ! = blockContinueCompressedSize , " LZ4_decompress_fast_usingDict did not read all compressed block input " ) ;
2014-10-10 19:58:42 +00:00
crcCheck = XXH32 ( decodedBuffer + dictSize , blockSize , 0 ) ;
2016-08-11 10:18:45 +00:00
if ( crcCheck ! = crcOrig ) {
2014-10-10 19:58:42 +00:00
int i = 0 ;
while ( block [ i ] = = decodedBuffer [ i ] ) i + + ;
printf ( " Wrong Byte at position %i/%i \n " , i , blockSize ) ;
2014-06-02 06:07:19 +00:00
2014-10-10 19:58:42 +00:00
}
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_fast_usingDict corrupted decoded data (dict %i) " , dictSize ) ;
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2015-04-11 17:59:22 +00:00
ret = LZ4_decompress_safe_usingDict ( compressedBuffer , decodedBuffer + dictSize , blockContinueCompressedSize , blockSize , decodedBuffer , dictSize ) ;
2015-04-24 09:15:12 +00:00
FUZ_CHECKTEST ( ret ! = blockSize , " LZ4_decompress_safe_usingDict did not regenerate original data " ) ;
2014-10-10 19:58:42 +00:00
crcCheck = XXH32 ( decodedBuffer + dictSize , blockSize , 0 ) ;
2015-04-24 09:15:12 +00:00
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_safe_usingDict corrupted decoded data " ) ;
2014-10-10 19:58:42 +00:00
2015-04-12 07:21:35 +00:00
/* Compress using External dictionary */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
2015-04-12 07:21:35 +00:00
dict - = ( FUZ_rand ( & randState ) & 0xF ) + 1 ; /* Separation, so it is an ExtDict */
2014-10-10 19:58:42 +00:00
if ( dict < ( char * ) CNBuffer ) dict = ( char * ) CNBuffer ;
LZ4_loadDict ( & LZ4dict , dict , dictSize ) ;
2016-11-14 15:10:31 +00:00
blockContinueCompressedSize = LZ4_compress_fast_continue ( & LZ4dict , block , compressedBuffer , blockSize , ( int ) compressedBufferSize , 1 ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( blockContinueCompressedSize = = 0 , " LZ4_compress_fast_continue failed " ) ;
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
LZ4_loadDict ( & LZ4dict , dict , dictSize ) ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_fast_continue ( & LZ4dict , block , compressedBuffer , blockSize , blockContinueCompressedSize - 1 , 1 ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret > 0 , " LZ4_compress_fast_continue using ExtDict should fail : one missing byte for output buffer : %i written, %i buffer " , ret , blockContinueCompressedSize ) ;
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
LZ4_loadDict ( & LZ4dict , dict , dictSize ) ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_fast_continue ( & LZ4dict , block , compressedBuffer , blockSize , blockContinueCompressedSize , 1 ) ;
2014-10-10 19:58:42 +00:00
FUZ_CHECKTEST ( ret ! = blockContinueCompressedSize , " LZ4_compress_limitedOutput_compressed size is different (%i != %i) " , ret , blockContinueCompressedSize ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret < = 0 , " LZ4_compress_fast_continue should work : enough size available within output buffer " ) ;
2014-10-10 19:58:42 +00:00
2015-04-24 09:15:12 +00:00
/* Decompress with dictionary as external */
2014-10-10 19:58:42 +00:00
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize ] = 0 ;
ret = LZ4_decompress_fast_usingDict ( compressedBuffer , decodedBuffer , blockSize , dict , dictSize ) ;
FUZ_CHECKTEST ( ret ! = blockContinueCompressedSize , " LZ4_decompress_fast_usingDict did not read all compressed block input " ) ;
2016-11-13 01:14:57 +00:00
FUZ_CHECKTEST ( decodedBuffer [ blockSize ] , " LZ4_decompress_fast_usingDict overrun specified output buffer size " ) ;
crcCheck = XXH32 ( decodedBuffer , blockSize , 0 ) ;
if ( crcCheck ! = crcOrig ) FUZ_findDiff ( block , decodedBuffer ) ;
2014-10-10 19:58:42 +00:00
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_fast_usingDict corrupted decoded data (dict %i) " , dictSize ) ;
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize ] = 0 ;
ret = LZ4_decompress_safe_usingDict ( compressedBuffer , decodedBuffer , blockContinueCompressedSize , blockSize , dict , dictSize ) ;
FUZ_CHECKTEST ( ret ! = blockSize , " LZ4_decompress_safe_usingDict did not regenerate original data " ) ;
2016-11-13 01:14:57 +00:00
FUZ_CHECKTEST ( decodedBuffer [ blockSize ] , " LZ4_decompress_safe_usingDict overrun specified output buffer size " ) ;
crcCheck = XXH32 ( decodedBuffer , blockSize , 0 ) ;
2014-10-10 19:58:42 +00:00
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_safe_usingDict corrupted decoded data " ) ;
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize - 1 ] = 0 ;
ret = LZ4_decompress_fast_usingDict ( compressedBuffer , decodedBuffer , blockSize - 1 , dict , dictSize ) ;
2015-04-24 09:15:12 +00:00
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_fast_usingDict should have failed : wrong original size (-1 byte) " ) ;
2014-10-10 19:58:42 +00:00
FUZ_CHECKTEST ( decodedBuffer [ blockSize - 1 ] , " LZ4_decompress_fast_usingDict overrun specified output buffer size " ) ;
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize - 1 ] = 0 ;
ret = LZ4_decompress_safe_usingDict ( compressedBuffer , decodedBuffer , blockContinueCompressedSize , blockSize - 1 , dict , dictSize ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_safe_usingDict should have failed : not enough output size (-1 byte) " ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize - 1 ] , " LZ4_decompress_safe_usingDict overrun specified output buffer size " ) ;
FUZ_DISPLAYTEST ;
2016-08-11 10:18:45 +00:00
{ U32 const missingBytes = ( FUZ_rand ( & randState ) & 0xF ) + 2 ;
if ( ( U32 ) blockSize > missingBytes ) {
2014-10-20 00:08:21 +00:00
decodedBuffer [ blockSize - missingBytes ] = 0 ;
ret = LZ4_decompress_safe_usingDict ( compressedBuffer , decodedBuffer , blockContinueCompressedSize , blockSize - missingBytes , dict , dictSize ) ;
FUZ_CHECKTEST ( ret > = 0 , " LZ4_decompress_safe_usingDict should have failed : output buffer too small (-%u byte) " , missingBytes ) ;
FUZ_CHECKTEST ( decodedBuffer [ blockSize - missingBytes ] , " LZ4_decompress_safe_usingDict overrun specified output buffer size (-%u byte) (blockSize=%i) " , missingBytes , blockSize ) ;
2016-08-11 10:18:45 +00:00
} }
2014-10-20 00:08:21 +00:00
2015-04-24 09:15:12 +00:00
/* Compress HC using External dictionary */
2014-10-25 19:52:10 +00:00
FUZ_DISPLAYTEST ;
2015-04-24 09:15:12 +00:00
dict - = ( FUZ_rand ( & randState ) & 7 ) ; /* even bigger separation */
2014-10-25 19:52:10 +00:00
if ( dict < ( char * ) CNBuffer ) dict = ( char * ) CNBuffer ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & LZ4dictHC , compressionLevel ) ;
2014-10-25 19:52:10 +00:00
LZ4_loadDictHC ( & LZ4dictHC , dict , dictSize ) ;
2017-11-03 18:28:28 +00:00
LZ4_setCompressionLevel ( & LZ4dictHC , compressionLevel - 1 ) ;
2016-11-14 15:10:31 +00:00
blockContinueCompressedSize = LZ4_compress_HC_continue ( & LZ4dictHC , block , compressedBuffer , blockSize , ( int ) compressedBufferSize ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( blockContinueCompressedSize = = 0 , " LZ4_compress_HC_continue failed " ) ;
2014-10-20 00:08:21 +00:00
2014-10-25 19:52:10 +00:00
FUZ_DISPLAYTEST ;
LZ4_loadDictHC ( & LZ4dictHC , dict , dictSize ) ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_HC_continue ( & LZ4dictHC , block , compressedBuffer , blockSize , blockContinueCompressedSize - 1 ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret > 0 , " LZ4_compress_HC_continue using ExtDict should fail : one missing byte for output buffer (%i != %i) " , ret , blockContinueCompressedSize ) ;
2014-10-20 00:08:21 +00:00
2014-10-25 19:52:10 +00:00
FUZ_DISPLAYTEST ;
LZ4_loadDictHC ( & LZ4dictHC , dict , dictSize ) ;
2016-11-12 16:48:42 +00:00
ret = LZ4_compress_HC_continue ( & LZ4dictHC , block , compressedBuffer , blockSize , blockContinueCompressedSize ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( ret ! = blockContinueCompressedSize , " LZ4_compress_HC_continue size is different (%i != %i) " , ret , blockContinueCompressedSize ) ;
FUZ_CHECKTEST ( ret < = 0 , " LZ4_compress_HC_continue should work : enough size available within output buffer " ) ;
2014-10-20 00:08:21 +00:00
FUZ_DISPLAYTEST ;
decodedBuffer [ blockSize ] = 0 ;
ret = LZ4_decompress_safe_usingDict ( compressedBuffer , decodedBuffer , blockContinueCompressedSize , blockSize , dict , dictSize ) ;
FUZ_CHECKTEST ( ret ! = blockSize , " LZ4_decompress_safe_usingDict did not regenerate original data " ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( decodedBuffer [ blockSize ] , " LZ4_decompress_safe_usingDict overrun specified output buffer size " ) ;
crcCheck = XXH32 ( decodedBuffer , blockSize , 0 ) ;
2014-10-20 00:08:21 +00:00
if ( crcCheck ! = crcOrig )
2015-04-24 09:15:12 +00:00
FUZ_findDiff ( block , decodedBuffer ) ;
2014-10-20 00:08:21 +00:00
FUZ_CHECKTEST ( crcCheck ! = crcOrig , " LZ4_decompress_safe_usingDict corrupted decoded data " ) ;
2014-04-28 20:45:35 +00:00
2017-03-16 22:10:38 +00:00
/* Compress HC continue destSize */
2017-11-03 18:28:28 +00:00
FUZ_DISPLAYTEST ;
2017-03-16 22:10:38 +00:00
{ int const availableSpace = ( FUZ_rand ( & randState ) % blockSize ) + 5 ;
int consumedSize = blockSize ;
FUZ_DISPLAYTEST ;
LZ4_resetStreamHC ( & LZ4dictHC , compressionLevel ) ;
LZ4_loadDictHC ( & LZ4dictHC , dict , dictSize ) ;
blockContinueCompressedSize = LZ4_compress_HC_continue_destSize ( & LZ4dictHC , block , compressedBuffer , & consumedSize , availableSpace ) ;
2017-03-17 22:11:09 +00:00
DISPLAYLEVEL ( 5 , " LZ4_compress_HC_continue_destSize : compressed %6i/%6i into %6i/%6i at cLevel=%i \n " , consumedSize , blockSize , blockContinueCompressedSize , availableSpace , compressionLevel ) ;
2017-03-16 22:10:38 +00:00
FUZ_CHECKTEST ( blockContinueCompressedSize = = 0 , " LZ4_compress_HC_continue_destSize failed " ) ;
FUZ_CHECKTEST ( blockContinueCompressedSize > availableSpace , " LZ4_compress_HC_continue_destSize write overflow " ) ;
FUZ_CHECKTEST ( consumedSize > blockSize , " LZ4_compress_HC_continue_destSize read overflow " ) ;
FUZ_DISPLAYTEST ;
decodedBuffer [ consumedSize ] = 0 ;
ret = LZ4_decompress_safe_usingDict ( compressedBuffer , decodedBuffer , blockContinueCompressedSize , consumedSize , dict , dictSize ) ;
FUZ_CHECKTEST ( ret ! = consumedSize , " LZ4_decompress_safe_usingDict did not regenerate original data " ) ;
FUZ_CHECKTEST ( decodedBuffer [ consumedSize ] , " LZ4_decompress_safe_usingDict overrun specified output buffer size " )
{ U32 const crcSrc = XXH32 ( block , consumedSize , 0 ) ;
U32 const crcDst = XXH32 ( decodedBuffer , consumedSize , 0 ) ;
if ( crcSrc ! = crcDst )
FUZ_findDiff ( block , decodedBuffer ) ;
FUZ_CHECKTEST ( crcSrc ! = crcDst , " LZ4_decompress_safe_usingDict corrupted decoded data " ) ;
}
}
2015-04-10 22:42:17 +00:00
/* ***** End of tests *** */
/* Fill stats */
2014-10-10 19:58:42 +00:00
bytes + = blockSize ;
cbytes + = compressedSize ;
hcbytes + = HCcompressedSize ;
ccbytes + = blockContinueCompressedSize ;
}
2013-12-30 17:16:52 +00:00
2015-04-12 07:21:35 +00:00
if ( nbCycles < = 1 ) nbCycles = cycleNb ; /* end by time */
bytes + = ! bytes ; /* avoid division by 0 */
2014-10-20 00:08:21 +00:00
printf ( " \r %7u /%7u - " , cycleNb , nbCycles ) ;
2014-10-10 19:58:42 +00:00
printf ( " all tests completed successfully \n " ) ;
printf ( " compression ratio: %0.3f%% \n " , ( double ) cbytes / bytes * 100 ) ;
printf ( " HC compression ratio: %0.3f%% \n " , ( double ) hcbytes / bytes * 100 ) ;
printf ( " ratio with dict: %0.3f%% \n " , ( double ) ccbytes / bytes * 100 ) ;
2014-04-28 22:17:49 +00:00
2015-04-10 22:42:17 +00:00
/* release memory */
2014-10-10 19:58:42 +00:00
{
_exit :
2014-04-28 22:17:49 +00:00
free ( CNBuffer ) ;
free ( compressedBuffer ) ;
free ( decodedBuffer ) ;
2014-04-29 20:39:08 +00:00
free ( stateLZ4 ) ;
free ( stateLZ4HC ) ;
2014-10-10 19:58:42 +00:00
return result ;
2013-12-30 17:16:52 +00:00
_output_error :
2014-10-10 19:58:42 +00:00
result = 1 ;
goto _exit ;
}
}
2014-10-25 19:52:10 +00:00
2014-11-04 09:04:37 +00:00
# define testInputSize (192 KB)
# define testCompressedSize (128 KB)
2014-10-25 19:52:10 +00:00
# define ringBufferSize (8 KB)
2014-10-10 19:58:42 +00:00
2016-12-28 16:47:10 +00:00
static void FUZ_unitTests ( int compressionLevel )
2014-10-10 19:58:42 +00:00
{
2014-10-18 10:18:14 +00:00
const unsigned testNb = 0 ;
const unsigned seed = 0 ;
const unsigned cycleNb = 0 ;
char testInput [ testInputSize ] ;
char testCompressed [ testCompressedSize ] ;
2014-10-20 23:12:55 +00:00
char testVerify [ testInputSize ] ;
2014-10-25 19:52:10 +00:00
char ringBuffer [ ringBufferSize ] ;
U32 randState = 1 ;
2014-10-18 10:18:14 +00:00
2015-04-24 09:15:12 +00:00
/* Init */
2014-10-18 10:18:14 +00:00
FUZ_fillCompressibleNoiseBuffer ( testInput , testInputSize , 0.50 , & randState ) ;
2015-04-24 09:15:12 +00:00
/* 32-bits address space overflow test */
2014-10-10 19:58:42 +00:00
FUZ_AddressOverflow ( ) ;
2014-10-18 10:18:14 +00:00
2015-04-24 09:15:12 +00:00
/* LZ4 streaming tests */
2016-08-11 10:18:45 +00:00
{ LZ4_stream_t * statePtr ;
2014-10-25 19:52:10 +00:00
LZ4_stream_t streamingState ;
U64 crcOrig ;
int result ;
2015-04-24 09:15:12 +00:00
/* Allocation test */
2014-10-25 19:52:10 +00:00
statePtr = LZ4_createStream ( ) ;
FUZ_CHECKTEST ( statePtr = = NULL , " LZ4_createStream() allocation failed " ) ;
LZ4_freeStream ( statePtr ) ;
2015-04-24 09:15:12 +00:00
/* simple compression test */
2014-10-25 19:52:10 +00:00
crcOrig = XXH64 ( testInput , testCompressedSize , 0 ) ;
LZ4_resetStream ( & streamingState ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_fast_continue ( & streamingState , testInput , testCompressed , testCompressedSize , testCompressedSize - 1 , 1 ) ;
2017-03-08 08:13:28 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compress_fast_continue() compression failed! " ) ;
2014-10-25 19:52:10 +00:00
result = LZ4_decompress_safe ( testCompressed , testVerify , result , testCompressedSize ) ;
FUZ_CHECKTEST ( result ! = ( int ) testCompressedSize , " LZ4_decompress_safe() decompression failed " ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64 ( testVerify , testCompressedSize , 0 ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() decompression corruption " ) ; }
2014-10-25 19:52:10 +00:00
2015-04-24 09:15:12 +00:00
/* ring buffer test */
2016-08-11 10:18:45 +00:00
{ XXH64_state_t xxhOrig ;
2014-10-25 19:52:10 +00:00
XXH64_state_t xxhNew ;
LZ4_streamDecode_t decodeState ;
const U32 maxMessageSizeLog = 10 ;
const U32 maxMessageSizeMask = ( 1 < < maxMessageSizeLog ) - 1 ;
U32 messageSize = ( FUZ_rand ( & randState ) & maxMessageSizeMask ) + 1 ;
U32 iNext = 0 ;
U32 rNext = 0 ;
U32 dNext = 0 ;
const U32 dBufferSize = ringBufferSize + maxMessageSizeMask ;
XXH64_reset ( & xxhOrig , 0 ) ;
XXH64_reset ( & xxhNew , 0 ) ;
LZ4_resetStream ( & streamingState ) ;
LZ4_setStreamDecode ( & decodeState , NULL , 0 ) ;
2016-08-11 10:18:45 +00:00
while ( iNext + messageSize < testCompressedSize ) {
2014-10-25 19:52:10 +00:00
XXH64_update ( & xxhOrig , testInput + iNext , messageSize ) ;
crcOrig = XXH64_digest ( & xxhOrig ) ;
memcpy ( ringBuffer + rNext , testInput + iNext , messageSize ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_fast_continue ( & streamingState , ringBuffer + rNext , testCompressed , messageSize , testCompressedSize - ringBufferSize , 1 ) ;
2017-03-08 08:13:28 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compress_fast_continue() compression failed " ) ;
2014-10-25 19:52:10 +00:00
result = LZ4_decompress_safe_continue ( & decodeState , testCompressed , testVerify + dNext , result , messageSize ) ;
FUZ_CHECKTEST ( result ! = ( int ) messageSize , " ringBuffer : LZ4_decompress_safe() test failed " ) ;
XXH64_update ( & xxhNew , testVerify + dNext , messageSize ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64_digest ( & xxhNew ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() decompression corruption " ) ; }
2014-10-25 19:52:10 +00:00
2015-04-24 09:15:12 +00:00
/* prepare next message */
2014-10-25 19:52:10 +00:00
iNext + = messageSize ;
rNext + = messageSize ;
dNext + = messageSize ;
messageSize = ( FUZ_rand ( & randState ) & maxMessageSizeMask ) + 1 ;
if ( rNext + messageSize > ringBufferSize ) rNext = 0 ;
if ( dNext + messageSize > dBufferSize ) dNext = 0 ;
}
}
}
2015-04-24 09:15:12 +00:00
/* LZ4 HC streaming tests */
2016-08-11 10:18:45 +00:00
{ LZ4_streamHC_t * sp ;
2014-10-18 10:18:14 +00:00
LZ4_streamHC_t sHC ;
U64 crcOrig ;
int result ;
2015-04-24 09:15:12 +00:00
/* Allocation test */
2014-10-18 10:18:14 +00:00
sp = LZ4_createStreamHC ( ) ;
FUZ_CHECKTEST ( sp = = NULL , " LZ4_createStreamHC() allocation failed " ) ;
LZ4_freeStreamHC ( sp ) ;
2015-04-24 09:15:12 +00:00
/* simple HC compression test */
2014-10-18 10:18:14 +00:00
crcOrig = XXH64 ( testInput , testCompressedSize , 0 ) ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_HC_continue ( & sHC , testInput , testCompressed , testCompressedSize , testCompressedSize - 1 ) ;
2014-10-18 10:18:14 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() compression failed " ) ;
result = LZ4_decompress_safe ( testCompressed , testVerify , result , testCompressedSize ) ;
FUZ_CHECKTEST ( result ! = ( int ) testCompressedSize , " LZ4_decompress_safe() decompression failed " ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64 ( testVerify , testCompressedSize , 0 ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() decompression corruption " ) ; }
2014-10-18 10:18:14 +00:00
2015-04-24 09:15:12 +00:00
/* simple dictionary HC compression test */
2014-10-18 10:18:14 +00:00
crcOrig = XXH64 ( testInput + 64 KB , testCompressedSize , 0 ) ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2014-10-18 10:18:14 +00:00
LZ4_loadDictHC ( & sHC , testInput , 64 KB ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_HC_continue ( & sHC , testInput + 64 KB , testCompressed , testCompressedSize , testCompressedSize - 1 ) ;
2014-10-18 10:18:14 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() dictionary compression failed : result = %i " , result ) ;
result = LZ4_decompress_safe_usingDict ( testCompressed , testVerify , result , testCompressedSize , testInput , 64 KB ) ;
2014-11-04 09:04:37 +00:00
FUZ_CHECKTEST ( result ! = ( int ) testCompressedSize , " LZ4_decompress_safe() simple dictionary decompression test failed " ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64 ( testVerify , testCompressedSize , 0 ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() simple dictionary decompression test : corruption " ) ; }
2014-10-18 10:18:14 +00:00
2015-04-10 22:42:17 +00:00
/* multiple HC compression test with dictionary */
2016-08-11 10:18:45 +00:00
{ int result1 , result2 ;
2014-10-18 10:18:14 +00:00
int segSize = testCompressedSize / 2 ;
2014-11-04 09:04:37 +00:00
crcOrig = XXH64 ( testInput + segSize , testCompressedSize , 0 ) ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2014-10-18 10:18:14 +00:00
LZ4_loadDictHC ( & sHC , testInput , segSize ) ;
2016-11-12 16:48:42 +00:00
result1 = LZ4_compress_HC_continue ( & sHC , testInput + segSize , testCompressed , segSize , segSize - 1 ) ;
2014-10-20 23:12:55 +00:00
FUZ_CHECKTEST ( result1 = = 0 , " LZ4_compressHC_limitedOutput_continue() dictionary compression failed : result = %i " , result1 ) ;
2016-11-12 16:48:42 +00:00
result2 = LZ4_compress_HC_continue ( & sHC , testInput + 2 * segSize , testCompressed + result1 , segSize , segSize - 1 ) ;
2014-10-20 23:12:55 +00:00
FUZ_CHECKTEST ( result2 = = 0 , " LZ4_compressHC_limitedOutput_continue() dictionary compression failed : result = %i " , result2 ) ;
2014-10-18 10:18:14 +00:00
result = LZ4_decompress_safe_usingDict ( testCompressed , testVerify , result1 , segSize , testInput , segSize ) ;
FUZ_CHECKTEST ( result ! = segSize , " LZ4_decompress_safe() dictionary decompression part 1 failed " ) ;
result = LZ4_decompress_safe_usingDict ( testCompressed + result1 , testVerify + segSize , result2 , segSize , testInput , 2 * segSize ) ;
FUZ_CHECKTEST ( result ! = segSize , " LZ4_decompress_safe() dictionary decompression part 2 failed " ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64 ( testVerify , testCompressedSize , 0 ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() dictionary decompression corruption " ) ; }
2014-10-18 10:18:14 +00:00
}
2015-04-24 09:15:12 +00:00
/* remote dictionary HC compression test */
2014-10-18 10:18:14 +00:00
crcOrig = XXH64 ( testInput + 64 KB , testCompressedSize , 0 ) ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2014-10-18 10:18:14 +00:00
LZ4_loadDictHC ( & sHC , testInput , 32 KB ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_HC_continue ( & sHC , testInput + 64 KB , testCompressed , testCompressedSize , testCompressedSize - 1 ) ;
2014-10-20 00:08:21 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() remote dictionary failed : result = %i " , result ) ;
2014-10-18 10:18:14 +00:00
result = LZ4_decompress_safe_usingDict ( testCompressed , testVerify , result , testCompressedSize , testInput , 32 KB ) ;
2014-10-20 00:08:21 +00:00
FUZ_CHECKTEST ( result ! = ( int ) testCompressedSize , " LZ4_decompress_safe_usingDict() decompression failed following remote dictionary HC compression test " ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64 ( testVerify , testCompressedSize , 0 ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe_usingDict() decompression corruption " ) ; }
2014-10-20 23:12:55 +00:00
2015-04-10 22:42:17 +00:00
/* multiple HC compression with ext. dictionary */
2016-08-11 10:18:45 +00:00
{ XXH64_state_t crcOrigState ;
2014-10-20 23:12:55 +00:00
XXH64_state_t crcNewState ;
const char * dict = testInput + 3 ;
int dictSize = ( FUZ_rand ( & randState ) & 8191 ) ;
char * dst = testVerify ;
size_t segStart = dictSize + 7 ;
int segSize = ( FUZ_rand ( & randState ) & 8191 ) ;
int segNb = 1 ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2014-10-20 23:12:55 +00:00
LZ4_loadDictHC ( & sHC , dict , dictSize ) ;
XXH64_reset ( & crcOrigState , 0 ) ;
XXH64_reset ( & crcNewState , 0 ) ;
2016-08-11 10:18:45 +00:00
while ( segStart + segSize < testInputSize ) {
2014-10-20 23:12:55 +00:00
XXH64_update ( & crcOrigState , testInput + segStart , segSize ) ;
crcOrig = XXH64_digest ( & crcOrigState ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_HC_continue ( & sHC , testInput + segStart , testCompressed , segSize , LZ4_compressBound ( segSize ) ) ;
2014-10-20 23:12:55 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() dictionary compression failed : result = %i " , result ) ;
result = LZ4_decompress_safe_usingDict ( testCompressed , dst , result , segSize , dict , dictSize ) ;
FUZ_CHECKTEST ( result ! = segSize , " LZ4_decompress_safe_usingDict() dictionary decompression part %i failed " , segNb ) ;
XXH64_update ( & crcNewState , dst , segSize ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64_digest ( & crcNewState ) ;
if ( crcOrig ! = crcNew ) FUZ_findDiff ( dst , testInput + segStart ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe_usingDict() part %i corruption " , segNb ) ;
2014-10-25 19:52:10 +00:00
}
2014-10-20 23:12:55 +00:00
dict = dst ;
dictSize = segSize ;
dst + = segSize + 1 ;
segNb + + ;
segStart + = segSize + ( FUZ_rand ( & randState ) & 0xF ) + 1 ;
segSize = ( FUZ_rand ( & randState ) & 8191 ) ;
}
}
2014-10-25 19:52:10 +00:00
2015-04-10 22:42:17 +00:00
/* ring buffer test */
2016-08-11 10:18:45 +00:00
{ XXH64_state_t xxhOrig ;
2014-10-25 19:52:10 +00:00
XXH64_state_t xxhNew ;
LZ4_streamDecode_t decodeState ;
const U32 maxMessageSizeLog = 10 ;
const U32 maxMessageSizeMask = ( 1 < < maxMessageSizeLog ) - 1 ;
U32 messageSize = ( FUZ_rand ( & randState ) & maxMessageSizeMask ) + 1 ;
U32 iNext = 0 ;
U32 rNext = 0 ;
U32 dNext = 0 ;
const U32 dBufferSize = ringBufferSize + maxMessageSizeMask ;
XXH64_reset ( & xxhOrig , 0 ) ;
XXH64_reset ( & xxhNew , 0 ) ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2014-10-25 19:52:10 +00:00
LZ4_setStreamDecode ( & decodeState , NULL , 0 ) ;
2016-08-11 10:18:45 +00:00
while ( iNext + messageSize < testCompressedSize ) {
2014-10-25 19:52:10 +00:00
XXH64_update ( & xxhOrig , testInput + iNext , messageSize ) ;
crcOrig = XXH64_digest ( & xxhOrig ) ;
memcpy ( ringBuffer + rNext , testInput + iNext , messageSize ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_HC_continue ( & sHC , ringBuffer + rNext , testCompressed , messageSize , testCompressedSize - ringBufferSize ) ;
2014-10-25 19:52:10 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() compression failed " ) ;
result = LZ4_decompress_safe_continue ( & decodeState , testCompressed , testVerify + dNext , result , messageSize ) ;
FUZ_CHECKTEST ( result ! = ( int ) messageSize , " ringBuffer : LZ4_decompress_safe() test failed " ) ;
XXH64_update ( & xxhNew , testVerify + dNext , messageSize ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64_digest ( & xxhNew ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() decompression corruption " ) ; }
2014-10-25 19:52:10 +00:00
2016-08-11 10:18:45 +00:00
/* prepare next message */
2014-10-25 19:52:10 +00:00
iNext + = messageSize ;
rNext + = messageSize ;
dNext + = messageSize ;
messageSize = ( FUZ_rand ( & randState ) & maxMessageSizeMask ) + 1 ;
if ( rNext + messageSize > ringBufferSize ) rNext = 0 ;
if ( dNext + messageSize > dBufferSize ) dNext = 0 ;
}
}
2015-04-10 22:42:17 +00:00
/* small decoder-side ring buffer test */
2016-08-11 10:18:45 +00:00
{ XXH64_state_t xxhOrig ;
2014-11-04 09:04:37 +00:00
XXH64_state_t xxhNew ;
LZ4_streamDecode_t decodeState ;
2015-04-24 09:15:12 +00:00
const U32 maxMessageSizeLog = 12 ;
2014-11-04 09:04:37 +00:00
const U32 maxMessageSizeMask = ( 1 < < maxMessageSizeLog ) - 1 ;
2015-04-24 10:04:21 +00:00
U32 messageSize ;
2014-11-04 09:04:37 +00:00
U32 totalMessageSize = 0 ;
U32 iNext = 0 ;
U32 dNext = 0 ;
2015-04-24 09:15:12 +00:00
const U32 dBufferSize = 64 KB ;
2014-11-04 09:04:37 +00:00
XXH64_reset ( & xxhOrig , 0 ) ;
XXH64_reset ( & xxhNew , 0 ) ;
2016-12-28 16:47:10 +00:00
LZ4_resetStreamHC ( & sHC , compressionLevel ) ;
2014-11-04 09:04:37 +00:00
LZ4_setStreamDecode ( & decodeState , NULL , 0 ) ;
2015-04-24 09:15:12 +00:00
# define BSIZE1 65537
# define BSIZE2 16435
/* first block */
2017-12-22 10:54:43 +00:00
messageSize = BSIZE1 ;
XXH64_update ( & xxhOrig , testInput + iNext , messageSize ) ;
crcOrig = XXH64_digest ( & xxhOrig ) ;
result = LZ4_compress_HC_continue ( & sHC , testInput + iNext , testCompressed , messageSize , testCompressedSize - ringBufferSize ) ;
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() compression failed " ) ;
result = LZ4_decompress_safe_continue ( & decodeState , testCompressed , testVerify + dNext , result , messageSize ) ;
FUZ_CHECKTEST ( result ! = ( int ) messageSize , " 64K D.ringBuffer : LZ4_decompress_safe() test failed " ) ;
XXH64_update ( & xxhNew , testVerify + dNext , messageSize ) ;
{ U64 const crcNew = XXH64_digest ( & xxhNew ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() decompression corruption " ) ; }
/* prepare next message */
dNext + = messageSize ;
totalMessageSize + = messageSize ;
messageSize = BSIZE2 ;
iNext = 132000 ;
memcpy ( testInput + iNext , testInput + 8 , messageSize ) ;
if ( dNext > dBufferSize ) dNext = 0 ;
2015-04-24 09:15:12 +00:00
2016-08-11 10:18:45 +00:00
while ( totalMessageSize < 9 MB ) {
2014-11-04 09:04:37 +00:00
XXH64_update ( & xxhOrig , testInput + iNext , messageSize ) ;
crcOrig = XXH64_digest ( & xxhOrig ) ;
2016-11-12 16:48:42 +00:00
result = LZ4_compress_HC_continue ( & sHC , testInput + iNext , testCompressed , messageSize , testCompressedSize - ringBufferSize ) ;
2014-11-04 09:04:37 +00:00
FUZ_CHECKTEST ( result = = 0 , " LZ4_compressHC_limitedOutput_continue() compression failed " ) ;
result = LZ4_decompress_safe_continue ( & decodeState , testCompressed , testVerify + dNext , result , messageSize ) ;
2015-04-24 09:15:12 +00:00
FUZ_CHECKTEST ( result ! = ( int ) messageSize , " 64K D.ringBuffer : LZ4_decompress_safe() test failed " ) ;
2014-11-04 09:04:37 +00:00
XXH64_update ( & xxhNew , testVerify + dNext , messageSize ) ;
2016-11-14 04:50:58 +00:00
{ U64 const crcNew = XXH64_digest ( & xxhNew ) ;
if ( crcOrig ! = crcNew ) FUZ_findDiff ( testInput + iNext , testVerify + dNext ) ;
FUZ_CHECKTEST ( crcOrig ! = crcNew , " LZ4_decompress_safe() decompression corruption during small decoder-side ring buffer test " ) ;
}
2015-03-22 12:42:00 +00:00
/* prepare next message */
2014-11-04 09:04:37 +00:00
dNext + = messageSize ;
totalMessageSize + = messageSize ;
messageSize = ( FUZ_rand ( & randState ) & maxMessageSizeMask ) + 1 ;
iNext = ( FUZ_rand ( & randState ) & 65535 ) ;
2015-04-24 09:15:12 +00:00
if ( dNext > dBufferSize ) dNext = 0 ;
2014-11-04 09:04:37 +00:00
}
}
2014-10-18 10:18:14 +00:00
}
2016-12-28 16:47:10 +00:00
printf ( " All unit tests completed successfully compressionLevel=%d \n " , compressionLevel ) ;
2014-10-18 10:18:14 +00:00
return ;
_output_error :
exit ( 1 ) ;
2013-12-30 17:16:52 +00:00
}
2014-04-29 20:39:08 +00:00
2016-11-13 01:14:57 +00:00
static int FUZ_usage ( const char * programName )
2014-04-29 20:39:08 +00:00
{
DISPLAY ( " Usage : \n " ) ;
DISPLAY ( " %s [args] \n " , programName ) ;
DISPLAY ( " \n " ) ;
DISPLAY ( " Arguments : \n " ) ;
DISPLAY ( " -i# : Nb of tests (default:%i) \n " , NB_ATTEMPTS ) ;
2015-04-12 15:40:58 +00:00
DISPLAY ( " -T# : Duration of tests, in seconds (default: use Nb of tests) \n " ) ;
2014-04-29 20:39:08 +00:00
DISPLAY ( " -s# : Select seed (default:prompt user) \n " ) ;
2014-05-19 23:40:29 +00:00
DISPLAY ( " -t# : Select starting test number (default:0) \n " ) ;
2014-10-10 19:58:42 +00:00
DISPLAY ( " -P# : Select compressibility in %% (default:%i%%) \n " , FUZ_COMPRESSIBILITY_DEFAULT ) ;
2014-06-09 01:42:39 +00:00
DISPLAY ( " -v : verbose \n " ) ;
2014-10-10 19:58:42 +00:00
DISPLAY ( " -p : pause at the end \n " ) ;
2014-04-29 20:39:08 +00:00
DISPLAY ( " -h : display help and exit \n " ) ;
return 0 ;
}
2016-11-13 01:14:57 +00:00
int main ( int argc , const char * * argv )
2014-10-09 21:15:57 +00:00
{
2016-11-13 01:14:57 +00:00
U32 seed = 0 ;
int seedset = 0 ;
2014-04-29 20:39:08 +00:00
int argNb ;
int nbTests = NB_ATTEMPTS ;
2014-05-19 23:40:29 +00:00
int testNb = 0 ;
2014-04-29 22:44:49 +00:00
int proba = FUZ_COMPRESSIBILITY_DEFAULT ;
2016-12-21 09:39:27 +00:00
int use_pause = 0 ;
2016-11-13 01:14:57 +00:00
const char * programName = argv [ 0 ] ;
2015-04-10 22:42:17 +00:00
U32 duration = 0 ;
2014-04-29 20:39:08 +00:00
2015-04-10 22:42:17 +00:00
/* Check command line */
2016-08-11 10:18:45 +00:00
for ( argNb = 1 ; argNb < argc ; argNb + + ) {
2016-11-13 01:14:57 +00:00
const char * argument = argv [ argNb ] ;
2014-04-29 20:39:08 +00:00
if ( ! argument ) continue ; // Protection if argument empty
// Decode command (note : aggregated commands are allowed)
2016-08-11 10:18:45 +00:00
if ( argument [ 0 ] = = ' - ' ) {
2016-12-21 09:39:27 +00:00
if ( ! strcmp ( argument , " --no-prompt " ) ) { use_pause = 0 ; seedset = 1 ; g_displayLevel = 1 ; continue ; }
2014-10-18 12:27:32 +00:00
argument + + ;
2014-04-29 20:39:08 +00:00
2016-08-11 10:18:45 +00:00
while ( * argument ! = 0 ) {
2014-04-29 20:39:08 +00:00
switch ( * argument )
{
2014-10-10 19:58:42 +00:00
case ' h ' : /* display help */
2014-10-19 15:06:33 +00:00
return FUZ_usage ( programName ) ;
2014-10-10 19:58:42 +00:00
case ' v ' : /* verbose mode */
2014-06-09 01:42:39 +00:00
argument + + ;
2016-11-13 01:14:57 +00:00
g_displayLevel + + ;
2014-06-09 01:42:39 +00:00
break ;
2014-10-10 19:58:42 +00:00
case ' p ' : /* pause at the end */
argument + + ;
2016-12-21 09:39:27 +00:00
use_pause = 1 ;
2014-10-10 19:58:42 +00:00
break ;
2014-04-29 20:39:08 +00:00
case ' i ' :
argument + + ;
2015-04-10 22:42:17 +00:00
nbTests = 0 ; duration = 0 ;
2016-08-11 10:18:45 +00:00
while ( ( * argument > = ' 0 ' ) & & ( * argument < = ' 9 ' ) ) {
2014-04-29 20:39:08 +00:00
nbTests * = 10 ;
nbTests + = * argument - ' 0 ' ;
argument + + ;
}
break ;
2014-10-10 19:58:42 +00:00
2015-04-10 22:42:17 +00:00
case ' T ' :
argument + + ;
nbTests = 0 ; duration = 0 ;
2016-08-11 10:18:45 +00:00
for ( ; ; ) {
2015-04-12 15:40:58 +00:00
switch ( * argument )
2015-04-10 22:42:17 +00:00
{
2015-04-12 15:40:58 +00:00
case ' m ' : duration * = 60 ; argument + + ; continue ;
case ' s ' :
case ' n ' : argument + + ; continue ;
case ' 0 ' :
case ' 1 ' :
case ' 2 ' :
case ' 3 ' :
case ' 4 ' :
case ' 5 ' :
case ' 6 ' :
case ' 7 ' :
case ' 8 ' :
case ' 9 ' : duration * = 10 ; duration + = * argument + + - ' 0 ' ; continue ;
2015-04-10 22:42:17 +00:00
}
break ;
}
break ;
2014-04-29 20:39:08 +00:00
case ' s ' :
argument + + ;
seed = 0 ; seedset = 1 ;
2016-08-11 10:18:45 +00:00
while ( ( * argument > = ' 0 ' ) & & ( * argument < = ' 9 ' ) ) {
2014-04-29 20:39:08 +00:00
seed * = 10 ;
seed + = * argument - ' 0 ' ;
argument + + ;
}
break ;
2014-10-10 19:58:42 +00:00
case ' t ' : /* select starting test nb */
2014-05-19 23:40:29 +00:00
argument + + ;
2014-12-10 08:05:44 +00:00
testNb = 0 ;
2016-08-11 10:18:45 +00:00
while ( ( * argument > = ' 0 ' ) & & ( * argument < = ' 9 ' ) ) {
2014-05-19 23:40:29 +00:00
testNb * = 10 ;
testNb + = * argument - ' 0 ' ;
argument + + ;
}
break ;
2014-10-10 19:58:42 +00:00
case ' P ' : /* change probability */
2014-04-29 22:44:49 +00:00
argument + + ;
proba = 0 ;
2016-08-11 10:18:45 +00:00
while ( ( * argument > = ' 0 ' ) & & ( * argument < = ' 9 ' ) ) {
2014-04-29 22:44:49 +00:00
proba * = 10 ;
proba + = * argument - ' 0 ' ;
argument + + ;
}
if ( proba < 0 ) proba = 0 ;
if ( proba > 100 ) proba = 100 ;
break ;
2014-04-29 20:39:08 +00:00
default : ;
}
}
}
}
2016-12-04 19:05:36 +00:00
printf ( " Starting LZ4 fuzzer (%i-bits, v%s) \n " , ( int ) ( sizeof ( size_t ) * 8 ) , LZ4_versionString ( ) ) ;
2014-04-29 20:39:08 +00:00
2016-09-03 02:48:24 +00:00
if ( ! seedset ) {
time_t const t = time ( NULL ) ;
U32 const h = XXH32 ( & t , sizeof ( t ) , 1 ) ;
seed = h % 10000 ;
}
2014-04-29 20:39:08 +00:00
printf ( " Seed = %u \n " , seed ) ;
2016-09-03 02:48:24 +00:00
2014-04-29 22:44:49 +00:00
if ( proba ! = FUZ_COMPRESSIBILITY_DEFAULT ) printf ( " Compressibility : %i%% \n " , proba ) ;
2014-04-29 20:39:08 +00:00
2016-12-28 16:47:10 +00:00
if ( ( seedset = = 0 ) & & ( testNb = = 0 ) ) { FUZ_unitTests ( LZ4HC_CLEVEL_DEFAULT ) ; FUZ_unitTests ( LZ4HC_CLEVEL_OPT_MIN ) ; }
2014-04-29 20:39:08 +00:00
if ( nbTests < = 0 ) nbTests = 1 ;
2016-08-11 10:18:45 +00:00
{ int const result = FUZ_test ( seed , nbTests , testNb , ( ( double ) proba ) / 100 , duration ) ;
2016-12-21 09:39:27 +00:00
if ( use_pause ) {
2014-10-10 19:58:42 +00:00
DISPLAY ( " press enter ... \n " ) ;
2015-04-12 07:21:35 +00:00
( void ) getchar ( ) ;
2014-10-10 19:58:42 +00:00
}
return result ;
}
2014-04-29 20:39:08 +00:00
}