| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829 | /*   LZ4 HC - High Compression Mode of LZ4   Copyright (C) 2011-2017, Yann Collet.   Adapted to CGo by Vadim Markovtsev, source{d} - 2018.   BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php)   Redistribution and use in source and binary forms, with or without   modification, are permitted provided that the following conditions are   met:       * Redistributions of source code must retain the above copyright   notice, this list of conditions and the following disclaimer.       * Redistributions in binary form must reproduce the above   copyright notice, this list of conditions and the following disclaimer   in the documentation and/or other materials provided with the   distribution.   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR   A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT   OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,   SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,   DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY   THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE   OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.*//*-************************************ *  Block Compression **************************************//*! LZ4_compress_HC() : *  Compress data from `src` into `dst`, using the more powerful but slower "HC" algorithm. * `dst` must be already allocated. *  Compression is guaranteed to succeed if `dstCapacity >= LZ4_compressBound(srcSize)` (see "lz4.h") *  Max supported `srcSize` value is LZ4_MAX_INPUT_SIZE (see "lz4.h") * @return : the number of bytes written into 'dst' *           or 0 if compression fails. */int LZ4_compress_HC(const void* src, void* dst, int srcSize, int dstCapacity, int compressionLevel);/*! LZ4_decompress_fast() : **unsafe!** *  This function used to be a bit faster than LZ4_decompress_safe(), *  though situation has changed in recent versions, *  and now `LZ4_decompress_safe()` can be as fast and sometimes faster than `LZ4_decompress_fast()`. *  Moreover, LZ4_decompress_fast() is not protected vs malformed input, as it doesn't perform full validation of compressed data. *  As a consequence, this function is no longer recommended, and may be deprecated in future versions. *  It's only remaining specificity is that it can decompress data without knowing its compressed size. * *  originalSize : is the uncompressed size to regenerate. *                 `dst` must be already allocated, its size must be >= 'originalSize' bytes. * @return : number of bytes read from source buffer (== compressed size). *           If the source stream is detected malformed, the function stops decoding and returns a negative result. *  note : This function requires uncompressed originalSize to be known in advance. *         The function never writes past the output buffer. *         However, since it doesn't know its 'src' size, it may read past the intended input. *         Also, because match offsets are not validated during decoding, *         reads from 'src' may underflow. *         Use this function in trusted environment **only**. */int LZ4_decompress_fast(const void* source, void* dest, int originalSize);#define LZ4HC_HEAPMODE 1// Modern Intel CPUs have 32KB of L1#define LZ4_MEMORY_USAGE 15#ifdef __GNUC__#define LZ4_FORCE_INLINE static inline __attribute__((always_inline))#else#define LZ4_FORCE_INLINE static inline#endif/*-*************************************  Memory routines**************************************/#include <stddef.h>#include <stdlib.h>   /* malloc, calloc, free */#define ALLOC(s)          malloc(s)#define ALLOC_AND_ZERO(s) calloc(1,s)#define FREEMEM(p)        free(p)#include <string.h>   /* memset, memcpy */#define MEM_INIT(p,v,s)   memset((p),(v),(s))/*-*************************************  Basic Types**************************************/#include <stdint.h>typedef  uint8_t BYTE;typedef uint16_t U16;typedef uint32_t U32;typedef  int32_t S32;typedef uint64_t U64;typedef uintptr_t uptrval;#if defined(__x86_64__)  typedef U64    reg_t;   /* 64-bits in x32 mode */#else  typedef size_t reg_t;   /* 32-bits in x32 mode */#endif/*-*************************************  CPU Feature Detection**************************************/#define expect(expr,value)    (__builtin_expect ((expr),(value)) )#ifndef likely#define likely(expr)     expect((expr) != 0, 1)#endif#ifndef unlikely#define unlikely(expr)   expect((expr) != 0, 0)#endif/* LZ4_FORCE_MEMORY_ACCESS * By default, access to unaligned memory is controlled by `memcpy()`, which is safe and portable. * Unfortunately, on some target/compiler combinations, the generated assembly is sub-optimal. * The below switch allow to select different access method for improved performance. * Method 0 (default) : use `memcpy()`. Safe and portable. * Method 1 : `__packed` statement. It depends on compiler extension (ie, not portable). *            This method is safe if your compiler supports it, and *generally* as fast or faster than `memcpy`. * Method 2 : direct access. This method is portable but violate C standard. *            It can generate buggy code on targets which assembly generation depends on alignment. *            But in some circumstances, it's the only known way to get the most performance (ie GCC + ARMv6) * See https://fastcompression.blogspot.fr/2015/08/accessing-unaligned-memory.html for details. * Prefer these methods in priority order (0 > 1 > 2) */#ifndef LZ4_FORCE_MEMORY_ACCESS   /* can be defined externally */#  if defined(__GNUC__) && \  ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) \  || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) )#    define LZ4_FORCE_MEMORY_ACCESS 2#  elif (defined(__INTEL_COMPILER) && !defined(_WIN32)) || defined(__GNUC__)#    define LZ4_FORCE_MEMORY_ACCESS 1#  endif#endif/*-*************************************  Reading and writing into memory**************************************/static unsigned LZ4_isLittleEndian(void){    const union { U32 u; BYTE c[4]; } one = { 1 };   /* don't use static : performance detrimental */    return one.c[0];}#if defined(LZ4_FORCE_MEMORY_ACCESS) && (LZ4_FORCE_MEMORY_ACCESS==2)/* lie to the compiler about data alignment; use with caution */static U16 LZ4_read16(const void* memPtr) { return *(const U16*) memPtr; }static U32 LZ4_read32(const void* memPtr) { return *(const U32*) memPtr; }static reg_t LZ4_read_ARCH(const void* memPtr) { return *(const reg_t*) memPtr; }static void LZ4_write16(void* memPtr, U16 value) { *(U16*)memPtr = value; }static void LZ4_write32(void* memPtr, U32 value) { *(U32*)memPtr = value; }#elif defined(LZ4_FORCE_MEMORY_ACCESS) && (LZ4_FORCE_MEMORY_ACCESS==1)/* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers *//* currently only defined for gcc and icc */typedef union { U16 u16; U32 u32; reg_t uArch; } __attribute__((packed)) unalign;static U16 LZ4_read16(const void* ptr) { return ((const unalign*)ptr)->u16; }static U32 LZ4_read32(const void* ptr) { return ((const unalign*)ptr)->u32; }static reg_t LZ4_read_ARCH(const void* ptr) { return ((const unalign*)ptr)->uArch; }static void LZ4_write16(void* memPtr, U16 value) { ((unalign*)memPtr)->u16 = value; }static void LZ4_write32(void* memPtr, U32 value) { ((unalign*)memPtr)->u32 = value; }#else  /* safe and portable access through memcpy() */static U16 LZ4_read16(const void* memPtr){    U16 val; memcpy(&val, memPtr, sizeof(val)); return val;}static U32 LZ4_read32(const void* memPtr){    U32 val; memcpy(&val, memPtr, sizeof(val)); return val;}static reg_t LZ4_read_ARCH(const void* memPtr){    reg_t val; memcpy(&val, memPtr, sizeof(val)); return val;}static void LZ4_write16(void* memPtr, U16 value){    memcpy(memPtr, &value, sizeof(value));}static void LZ4_write32(void* memPtr, U32 value){    memcpy(memPtr, &value, sizeof(value));}#endif /* LZ4_FORCE_MEMORY_ACCESS */static U16 LZ4_readLE16(const void* memPtr){    if (LZ4_isLittleEndian()) {        return LZ4_read16(memPtr);    } else {        const BYTE* p = (const BYTE*)memPtr;        return (U16)((U16)p[0] + (p[1]<<8));    }}static void LZ4_writeLE16(void* memPtr, U16 value){    if (LZ4_isLittleEndian()) {        LZ4_write16(memPtr, value);    } else {        BYTE* p = (BYTE*)memPtr;        p[0] = (BYTE) value;        p[1] = (BYTE)(value>>8);    }}/* customized variant of memcpy, which can overwrite up to 8 bytes beyond dstEnd */LZ4_FORCE_INLINEvoid LZ4_wildCopy(void* dstPtr, const void* srcPtr, void* dstEnd){    BYTE* d = (BYTE*)dstPtr;    const BYTE* s = (const BYTE*)srcPtr;    BYTE* const e = (BYTE*)dstEnd;    do { memcpy(d,s,8); d+=8; s+=8; } while (d<e);}#define assert(condition) ((void)0)#define DEBUGLOG(...) ((void)0)#define LZ4_STATIC_ASSERT(c)   { enum { LZ4_static_assert = 1/(int)(!!(c)) }; }   /* use after variable declarations *//*===   Constants   ===*/#define OPTIMAL_ML (int)((ML_MASK-1)+MINMATCH)#define LZ4_OPT_NUM   (1<<12)#define MINMATCH 4#define LZ4_MAX_INPUT_SIZE        0x7E000000   /* 2 113 929 216 bytes */#define LZ4_COMPRESSBOUND(isize)  ((unsigned)(isize) > (unsigned)LZ4_MAX_INPUT_SIZE ? 0 : (isize) + ((isize)/255) + 16)#define WILDCOPYLENGTH 8#define LASTLITERALS   5   /* see ../doc/lz4_Block_format.md#parsing-restrictions */#define MFLIMIT       12   /* see ../doc/lz4_Block_format.md#parsing-restrictions */#define MATCH_SAFEGUARD_DISTANCE  ((2*WILDCOPYLENGTH) - MINMATCH)   /* ensure it's possible to write 2 x wildcopyLength without overflowing output buffer */static const int LZ4_minLength = (MFLIMIT+1);#define KB *(1 <<10)#define MB *(1 <<20)#define GB *(1U<<30)#define MAXD_LOG 16#define MAX_DISTANCE ((1 << MAXD_LOG) - 1)#define ML_BITS  4#define ML_MASK  ((1U<<ML_BITS)-1)#define RUN_BITS (8-ML_BITS)#define RUN_MASK ((1U<<RUN_BITS)-1)int LZ4_compressBound(int isize)  { return LZ4_COMPRESSBOUND(isize); }/*===   Enums   ===*/typedef enum { noDictCtx, usingDictCtx } dictCtx_directive;#define LZ4HC_CLEVEL_MIN         3#define LZ4HC_CLEVEL_DEFAULT     9#define LZ4HC_CLEVEL_OPT_MIN    10#define LZ4HC_CLEVEL_MAX        12#define LZ4HC_DICTIONARY_LOGSIZE 16#define LZ4HC_MAXD (1<<LZ4HC_DICTIONARY_LOGSIZE)#define LZ4HC_MAXD_MASK (LZ4HC_MAXD - 1)#define LZ4HC_HASH_LOG 15#define LZ4HC_HASHTABLESIZE (1 << LZ4HC_HASH_LOG)#define LZ4HC_HASH_MASK (LZ4HC_HASHTABLESIZE - 1)typedef struct LZ4HC_CCtx_internal LZ4HC_CCtx_internal;struct LZ4HC_CCtx_internal{    uint32_t   hashTable[LZ4HC_HASHTABLESIZE];    uint16_t   chainTable[LZ4HC_MAXD];    const uint8_t* end;         /* next block here to continue on current prefix */    const uint8_t* base;        /* All index relative to this position */    const uint8_t* dictBase;    /* alternate base for extDict */    uint32_t   dictLimit;       /* below that point, need extDict */    uint32_t   lowLimit;        /* below that point, no more dict */    uint32_t   nextToUpdate;    /* index from which to continue dictionary update */    short      compressionLevel;    int8_t     favorDecSpeed;   /* favor decompression speed if this flag set,                                   otherwise, favor compression ratio */    int8_t     dirty;           /* stream has to be fully reset if this flag is set */    const LZ4HC_CCtx_internal* dictCtx;};#define LZ4_STREAMHCSIZE       (4*LZ4HC_HASHTABLESIZE + 2*LZ4HC_MAXD + 56 + ((sizeof(void*)==16) ? 56 : 0) /* AS400*/ ) /* 262200 or 262256*/#define LZ4_STREAMHCSIZE_SIZET (LZ4_STREAMHCSIZE / sizeof(size_t))union LZ4_streamHC_u {    size_t table[LZ4_STREAMHCSIZE_SIZET];    LZ4HC_CCtx_internal internal_donotuse;};typedef union LZ4_streamHC_u LZ4_streamHC_t;/*===   Macros   ===*/#define MIN(a,b)   ( (a) < (b) ? (a) : (b) )#define MAX(a,b)   ( (a) > (b) ? (a) : (b) )#define HASH_FUNCTION(i)         (((i) * 2654435761U) >> ((MINMATCH*8)-LZ4HC_HASH_LOG))#define DELTANEXTMAXD(p)         chainTable[(p) & LZ4HC_MAXD_MASK]    /* flexible, LZ4HC_MAXD dependent */#define DELTANEXTU16(table, pos) table[(U16)(pos)]   /* faster */static U32 LZ4HC_hashPtr(const void* ptr) { return HASH_FUNCTION(LZ4_read32(ptr)); }/*-*************************************  Common functions**************************************/static unsigned LZ4_NbCommonBytes (reg_t val){    if (LZ4_isLittleEndian()) {        if (sizeof(val)==8) {#       if defined(_MSC_VER) && defined(_WIN64) && !defined(LZ4_FORCE_SW_BITCOUNT)            unsigned long r = 0;            _BitScanForward64( &r, (U64)val );            return (int)(r>>3);#       elif (defined(__clang__) || (defined(__GNUC__) && (__GNUC__>=3))) && !defined(LZ4_FORCE_SW_BITCOUNT)            return (__builtin_ctzll((U64)val) >> 3);#       else            static const int DeBruijnBytePos[64] = { 0, 0, 0, 0, 0, 1, 1, 2,                                                     0, 3, 1, 3, 1, 4, 2, 7,                                                     0, 2, 3, 6, 1, 5, 3, 5,                                                     1, 3, 4, 4, 2, 5, 6, 7,                                                     7, 0, 1, 2, 3, 3, 4, 6,                                                     2, 6, 5, 5, 3, 4, 5, 6,                                                     7, 1, 2, 4, 6, 4, 4, 5,                                                     7, 2, 6, 5, 7, 6, 7, 7 };            return DeBruijnBytePos[((U64)((val & -(long long)val) * 0x0218A392CDABBD3FULL)) >> 58];#       endif        } else /* 32 bits */ {#       if defined(_MSC_VER) && !defined(LZ4_FORCE_SW_BITCOUNT)            unsigned long r;            _BitScanForward( &r, (U32)val );            return (int)(r>>3);#       elif (defined(__clang__) || (defined(__GNUC__) && (__GNUC__>=3))) && !defined(LZ4_FORCE_SW_BITCOUNT)            return (__builtin_ctz((U32)val) >> 3);#       else            static const int DeBruijnBytePos[32] = { 0, 0, 3, 0, 3, 1, 3, 0,                                                     3, 2, 2, 1, 3, 2, 0, 1,                                                     3, 3, 1, 2, 2, 2, 2, 0,                                                     3, 1, 2, 0, 1, 0, 1, 1 };            return DeBruijnBytePos[((U32)((val & -(S32)val) * 0x077CB531U)) >> 27];#       endif        }    } else   /* Big Endian CPU */ {        if (sizeof(val)==8) {   /* 64-bits */#       if defined(_MSC_VER) && defined(_WIN64) && !defined(LZ4_FORCE_SW_BITCOUNT)            unsigned long r = 0;            _BitScanReverse64( &r, val );            return (unsigned)(r>>3);#       elif (defined(__clang__) || (defined(__GNUC__) && (__GNUC__>=3))) && !defined(LZ4_FORCE_SW_BITCOUNT)            return (__builtin_clzll((U64)val) >> 3);#       else            static const U32 by32 = sizeof(val)*4;  /* 32 on 64 bits (goal), 16 on 32 bits.                Just to avoid some static analyzer complaining about shift by 32 on 32-bits target.                Note that this code path is never triggered in 32-bits mode. */            unsigned r;            if (!(val>>by32)) { r=4; } else { r=0; val>>=by32; }            if (!(val>>16)) { r+=2; val>>=8; } else { val>>=24; }            r += (!val);            return r;#       endif        } else /* 32 bits */ {#       if defined(_MSC_VER) && !defined(LZ4_FORCE_SW_BITCOUNT)            unsigned long r = 0;            _BitScanReverse( &r, (unsigned long)val );            return (unsigned)(r>>3);#       elif (defined(__clang__) || (defined(__GNUC__) && (__GNUC__>=3))) && !defined(LZ4_FORCE_SW_BITCOUNT)            return (__builtin_clz((U32)val) >> 3);#       else            unsigned r;            if (!(val>>16)) { r=2; val>>=8; } else { r=0; val>>=24; }            r += (!val);            return r;#       endif        }    }}#define STEPSIZE sizeof(reg_t)LZ4_FORCE_INLINEunsigned LZ4_count(const BYTE* pIn, const BYTE* pMatch, const BYTE* pInLimit){    const BYTE* const pStart = pIn;    if (likely(pIn < pInLimit-(STEPSIZE-1))) {        reg_t const diff = LZ4_read_ARCH(pMatch) ^ LZ4_read_ARCH(pIn);        if (!diff) {            pIn+=STEPSIZE; pMatch+=STEPSIZE;        } else {            return LZ4_NbCommonBytes(diff);    }   }    while (likely(pIn < pInLimit-(STEPSIZE-1))) {        reg_t const diff = LZ4_read_ARCH(pMatch) ^ LZ4_read_ARCH(pIn);        if (!diff) { pIn+=STEPSIZE; pMatch+=STEPSIZE; continue; }        pIn += LZ4_NbCommonBytes(diff);        return (unsigned)(pIn - pStart);    }    if ((STEPSIZE==8) && (pIn<(pInLimit-3)) && (LZ4_read32(pMatch) == LZ4_read32(pIn))) { pIn+=4; pMatch+=4; }    if ((pIn<(pInLimit-1)) && (LZ4_read16(pMatch) == LZ4_read16(pIn))) { pIn+=2; pMatch+=2; }    if ((pIn<pInLimit) && (*pMatch == *pIn)) pIn++;    return (unsigned)(pIn - pStart);}/***************************************  HC Compression**************************************/static void LZ4HC_clearTables (LZ4HC_CCtx_internal* hc4){    MEM_INIT((void*)hc4->hashTable, 0, sizeof(hc4->hashTable));    MEM_INIT(hc4->chainTable, 0xFF, sizeof(hc4->chainTable));}static void LZ4HC_init (LZ4HC_CCtx_internal* hc4, const BYTE* start){    uptrval startingOffset = hc4->end - hc4->base;    if (startingOffset > 1 GB) {        LZ4HC_clearTables(hc4);        startingOffset = 0;    }    startingOffset += 64 KB;    hc4->nextToUpdate = (U32) startingOffset;    hc4->base = start - startingOffset;    hc4->end = start;    hc4->dictBase = start - startingOffset;    hc4->dictLimit = (U32) startingOffset;    hc4->lowLimit = (U32) startingOffset;}/* Update chains up to ip (excluded) */LZ4_FORCE_INLINE void LZ4HC_Insert (LZ4HC_CCtx_internal* hc4, const BYTE* ip){    U16* const chainTable = hc4->chainTable;    U32* const hashTable  = hc4->hashTable;    const BYTE* const base = hc4->base;    U32 const target = (U32)(ip - base);    U32 idx = hc4->nextToUpdate;    while (idx < target) {        U32 const h = LZ4HC_hashPtr(base+idx);        size_t delta = idx - hashTable[h];        if (delta>MAX_DISTANCE) delta = MAX_DISTANCE;        DELTANEXTU16(chainTable, idx) = (U16)delta;        hashTable[h] = idx;        idx++;    }    hc4->nextToUpdate = target;}static void LZ4HC_setExternalDict(LZ4HC_CCtx_internal* ctxPtr, const BYTE* newBlock){    DEBUGLOG(4, "LZ4HC_setExternalDict(%p, %p)", ctxPtr, newBlock);    if (ctxPtr->end >= ctxPtr->base + ctxPtr->dictLimit + 4)        LZ4HC_Insert (ctxPtr, ctxPtr->end-3);   /* Referencing remaining dictionary content */    /* Only one memory segment for extDict, so any previous extDict is lost at this stage */    ctxPtr->lowLimit  = ctxPtr->dictLimit;    ctxPtr->dictLimit = (U32)(ctxPtr->end - ctxPtr->base);    ctxPtr->dictBase  = ctxPtr->base;    ctxPtr->base = newBlock - ctxPtr->dictLimit;    ctxPtr->end  = newBlock;    ctxPtr->nextToUpdate = ctxPtr->dictLimit;   /* match referencing will resume from there */}/** LZ4HC_countBack() : * @return : negative value, nb of common bytes before ip/match */LZ4_FORCE_INLINEint LZ4HC_countBack(const BYTE* const ip, const BYTE* const match,                    const BYTE* const iMin, const BYTE* const mMin){    int back = 0;    int const min = (int)MAX(iMin - ip, mMin - match);    assert(min <= 0);    assert(ip >= iMin); assert((size_t)(ip-iMin) < (1U<<31));    assert(match >= mMin); assert((size_t)(match - mMin) < (1U<<31));    while ( (back > min)         && (ip[back-1] == match[back-1]) )            back--;    return back;}/* LZ4HC_countPattern() : * pattern32 must be a sample of repetitive pattern of length 1, 2 or 4 (but not 3!) */static unsignedLZ4HC_countPattern(const BYTE* ip, const BYTE* const iEnd, U32 const pattern32){    const BYTE* const iStart = ip;    reg_t const pattern = (sizeof(pattern)==8) ? (reg_t)pattern32 + (((reg_t)pattern32) << 32) : pattern32;    while (likely(ip < iEnd-(sizeof(pattern)-1))) {        reg_t const diff = LZ4_read_ARCH(ip) ^ pattern;        if (!diff) { ip+=sizeof(pattern); continue; }        ip += LZ4_NbCommonBytes(diff);        return (unsigned)(ip - iStart);    }    if (LZ4_isLittleEndian()) {        reg_t patternByte = pattern;        while ((ip<iEnd) && (*ip == (BYTE)patternByte)) {            ip++; patternByte >>= 8;        }    } else {  /* big endian */        U32 bitOffset = (sizeof(pattern)*8) - 8;        while (ip < iEnd) {            BYTE const byte = (BYTE)(pattern >> bitOffset);            if (*ip != byte) break;            ip ++; bitOffset -= 8;        }    }    return (unsigned)(ip - iStart);}/* LZ4HC_reverseCountPattern() : * pattern must be a sample of repetitive pattern of length 1, 2 or 4 (but not 3!) * read using natural platform endianess */static unsignedLZ4HC_reverseCountPattern(const BYTE* ip, const BYTE* const iLow, U32 pattern){    const BYTE* const iStart = ip;    while (likely(ip >= iLow+4)) {        if (LZ4_read32(ip-4) != pattern) break;        ip -= 4;    }    {   const BYTE* bytePtr = (const BYTE*)(&pattern) + 3; /* works for any endianess */        while (likely(ip>iLow)) {            if (ip[-1] != *bytePtr) break;            ip--; bytePtr--;    }   }    return (unsigned)(iStart - ip);}typedef enum { rep_untested, rep_not, rep_confirmed } repeat_state_e;typedef enum { favorCompressionRatio=0, favorDecompressionSpeed } HCfavor_e;LZ4_FORCE_INLINE intLZ4HC_InsertAndGetWiderMatch (    LZ4HC_CCtx_internal* hc4,    const BYTE* const ip,    const BYTE* const iLowLimit,    const BYTE* const iHighLimit,    int longest,    const BYTE** matchpos,    const BYTE** startpos,    const int maxNbAttempts,    const int patternAnalysis,    const int chainSwap,    const dictCtx_directive dict,    const HCfavor_e favorDecSpeed){    U16* const chainTable = hc4->chainTable;    U32* const HashTable = hc4->hashTable;    const LZ4HC_CCtx_internal * const dictCtx = hc4->dictCtx;    const BYTE* const base = hc4->base;    const U32 dictLimit = hc4->dictLimit;    const BYTE* const lowPrefixPtr = base + dictLimit;    const U32 ipIndex = (U32)(ip - base);    const U32 lowestMatchIndex = (hc4->lowLimit + 64 KB > ipIndex) ? hc4->lowLimit : ipIndex - MAX_DISTANCE;    const BYTE* const dictBase = hc4->dictBase;    int const lookBackLength = (int)(ip-iLowLimit);    int nbAttempts = maxNbAttempts;    int matchChainPos = 0;    U32 const pattern = LZ4_read32(ip);    U32 matchIndex;    U32 dictMatchIndex;    repeat_state_e repeat = rep_untested;    size_t srcPatternLength = 0;    DEBUGLOG(7, "LZ4HC_InsertAndGetWiderMatch");    /* First Match */    LZ4HC_Insert(hc4, ip);    matchIndex = HashTable[LZ4HC_hashPtr(ip)];    DEBUGLOG(7, "First match at index %u / %u (lowestMatchIndex)",                matchIndex, lowestMatchIndex);    while ((matchIndex>=lowestMatchIndex) && (nbAttempts)) {        int matchLength=0;        nbAttempts--;        assert(matchIndex < ipIndex);        if (favorDecSpeed && (ipIndex - matchIndex < 8)) {            /* do nothing */        } else if (matchIndex >= dictLimit) {   /* within current Prefix */            const BYTE* const matchPtr = base + matchIndex;            assert(matchPtr >= lowPrefixPtr);            assert(matchPtr < ip);            assert(longest >= 1);            if (LZ4_read16(iLowLimit + longest - 1) == LZ4_read16(matchPtr - lookBackLength + longest - 1)) {                if (LZ4_read32(matchPtr) == pattern) {                    int const back = lookBackLength ? LZ4HC_countBack(ip, matchPtr, iLowLimit, lowPrefixPtr) : 0;                    matchLength = MINMATCH + LZ4_count(ip+MINMATCH, matchPtr+MINMATCH, iHighLimit);                    matchLength -= back;                    if (matchLength > longest) {                        longest = matchLength;                        *matchpos = matchPtr + back;                        *startpos = ip + back;            }   }   }        } else {   /* lowestMatchIndex <= matchIndex < dictLimit */            const BYTE* const matchPtr = dictBase + matchIndex;            if (LZ4_read32(matchPtr) == pattern) {                const BYTE* const dictStart = dictBase + hc4->lowLimit;                int back = 0;                const BYTE* vLimit = ip + (dictLimit - matchIndex);                if (vLimit > iHighLimit) vLimit = iHighLimit;                matchLength = LZ4_count(ip+MINMATCH, matchPtr+MINMATCH, vLimit) + MINMATCH;                if ((ip+matchLength == vLimit) && (vLimit < iHighLimit))                    matchLength += LZ4_count(ip+matchLength, lowPrefixPtr, iHighLimit);                back = lookBackLength ? LZ4HC_countBack(ip, matchPtr, iLowLimit, dictStart) : 0;                matchLength -= back;                if (matchLength > longest) {                    longest = matchLength;                    *matchpos = base + matchIndex + back;   /* virtual pos, relative to ip, to retrieve offset */                    *startpos = ip + back;        }   }   }        if (chainSwap && matchLength==longest) {    /* better match => select a better chain */            assert(lookBackLength==0);   /* search forward only */            if (matchIndex + longest <= ipIndex) {                U32 distanceToNextMatch = 1;                int pos;                for (pos = 0; pos <= longest - MINMATCH; pos++) {                    U32 const candidateDist = DELTANEXTU16(chainTable, matchIndex + pos);                    if (candidateDist > distanceToNextMatch) {                        distanceToNextMatch = candidateDist;                        matchChainPos = pos;                }   }                if (distanceToNextMatch > 1) {                    if (distanceToNextMatch > matchIndex) break;   /* avoid overflow */                    matchIndex -= distanceToNextMatch;                    continue;        }   }   }        {   U32 const distNextMatch = DELTANEXTU16(chainTable, matchIndex);            if (patternAnalysis && distNextMatch==1 && matchChainPos==0) {                U32 const matchCandidateIdx = matchIndex-1;                /* may be a repeated pattern */                if (repeat == rep_untested) {                    if ( ((pattern & 0xFFFF) == (pattern >> 16))                      &  ((pattern & 0xFF)   == (pattern >> 24)) ) {                        repeat = rep_confirmed;                        srcPatternLength = LZ4HC_countPattern(ip+sizeof(pattern), iHighLimit, pattern) + sizeof(pattern);                    } else {                        repeat = rep_not;                }   }                if ( (repeat == rep_confirmed)                  && (matchCandidateIdx >= dictLimit) ) {   /* same segment only */                    const BYTE* const matchPtr = base + matchCandidateIdx;                    if (LZ4_read32(matchPtr) == pattern) {  /* good candidate */                        size_t const forwardPatternLength = LZ4HC_countPattern(matchPtr+sizeof(pattern), iHighLimit, pattern) + sizeof(pattern);                        const BYTE* const lowestMatchPtr = (lowPrefixPtr + MAX_DISTANCE >= ip) ? lowPrefixPtr : ip - MAX_DISTANCE;                        size_t const backLength = LZ4HC_reverseCountPattern(matchPtr, lowestMatchPtr, pattern);                        size_t const currentSegmentLength = backLength + forwardPatternLength;                        if ( (currentSegmentLength >= srcPatternLength)   /* current pattern segment large enough to contain full srcPatternLength */                          && (forwardPatternLength <= srcPatternLength) ) { /* haven't reached this position yet */                            matchIndex = matchCandidateIdx + (U32)forwardPatternLength - (U32)srcPatternLength;  /* best position, full pattern, might be followed by more match */                        } else {                            matchIndex = matchCandidateIdx - (U32)backLength;   /* farthest position in current segment, will find a match of length currentSegmentLength + maybe some back */                            if (lookBackLength==0) {  /* no back possible */                                size_t const maxML = MIN(currentSegmentLength, srcPatternLength);                                if ((size_t)longest < maxML) {                                    assert(base + matchIndex < ip);                                    if (ip - (base+matchIndex) > MAX_DISTANCE) break;                                    assert(maxML < 2 GB);                                    longest = (int)maxML;                                    *matchpos = base + matchIndex;   /* virtual pos, relative to ip, to retrieve offset */                                    *startpos = ip;                                }                                {   U32 const distToNextPattern = DELTANEXTU16(chainTable, matchIndex);                                    if (distToNextPattern > matchIndex) break;  /* avoid overflow */                                    matchIndex -= distToNextPattern;                        }   }   }                        continue;                }   }        }   }   /* PA optimization */        /* follow current chain */        matchIndex -= DELTANEXTU16(chainTable, matchIndex+matchChainPos);    }  /* while ((matchIndex>=lowestMatchIndex) && (nbAttempts)) */    if (dict == usingDictCtx && nbAttempts && ipIndex - lowestMatchIndex < MAX_DISTANCE) {        size_t const dictEndOffset = dictCtx->end - dictCtx->base;        assert(dictEndOffset <= 1 GB);        dictMatchIndex = dictCtx->hashTable[LZ4HC_hashPtr(ip)];        matchIndex = dictMatchIndex + lowestMatchIndex - (U32)dictEndOffset;        while (ipIndex - matchIndex <= MAX_DISTANCE && nbAttempts--) {            const BYTE* const matchPtr = dictCtx->base + dictMatchIndex;            if (LZ4_read32(matchPtr) == pattern) {                int mlt;                int back = 0;                const BYTE* vLimit = ip + (dictEndOffset - dictMatchIndex);                if (vLimit > iHighLimit) vLimit = iHighLimit;                mlt = LZ4_count(ip+MINMATCH, matchPtr+MINMATCH, vLimit) + MINMATCH;                back = lookBackLength ? LZ4HC_countBack(ip, matchPtr, iLowLimit, dictCtx->base + dictCtx->dictLimit) : 0;                mlt -= back;                if (mlt > longest) {                    longest = mlt;                    *matchpos = base + matchIndex + back;                    *startpos = ip + back;                }            }            {   U32 const nextOffset = DELTANEXTU16(dictCtx->chainTable, dictMatchIndex);                dictMatchIndex -= nextOffset;                matchIndex -= nextOffset;            }        }    }    return longest;}LZ4_FORCE_INLINEint LZ4HC_InsertAndFindBestMatch(LZ4HC_CCtx_internal* const hc4,   /* Index table will be updated */                                 const BYTE* const ip, const BYTE* const iLimit,                                 const BYTE** matchpos,                                 const int maxNbAttempts,                                 const int patternAnalysis,                                 const dictCtx_directive dict){    const BYTE* uselessPtr = ip;    /* note : LZ4HC_InsertAndGetWiderMatch() is able to modify the starting position of a match (*startpos),     * but this won't be the case here, as we define iLowLimit==ip,     * so LZ4HC_InsertAndGetWiderMatch() won't be allowed to search past ip */    return LZ4HC_InsertAndGetWiderMatch(hc4, ip, ip, iLimit, MINMATCH-1, matchpos, &uselessPtr, maxNbAttempts, patternAnalysis, 0 /*chainSwap*/, dict, favorCompressionRatio);}typedef enum {    noLimit = 0,    limitedOutput = 1,    limitedDestSize = 2,} limitedOutput_directive;/* LZ4HC_encodeSequence() : * @return : 0 if ok, *           1 if buffer issue detected */LZ4_FORCE_INLINE int LZ4HC_encodeSequence (    const BYTE** ip,    BYTE** op,    const BYTE** anchor,    int matchLength,    const BYTE* const match,    limitedOutput_directive limit,    BYTE* oend){    size_t length;    BYTE* const token = (*op)++;#if defined(LZ4_DEBUG) && (LZ4_DEBUG >= 6)    static const BYTE* start = NULL;    static U32 totalCost = 0;    U32 const pos = (start==NULL) ? 0 : (U32)(*anchor - start);    U32 const ll = (U32)(*ip - *anchor);    U32 const llAdd = (ll>=15) ? ((ll-15) / 255) + 1 : 0;    U32 const mlAdd = (matchLength>=19) ? ((matchLength-19) / 255) + 1 : 0;    U32 const cost = 1 + llAdd + ll + 2 + mlAdd;    if (start==NULL) start = *anchor;  /* only works for single segment */    /* g_debuglog_enable = (pos >= 2228) & (pos <= 2262); */    DEBUGLOG(6, "pos:%7u -- literals:%3u, match:%4i, offset:%5u, cost:%3u + %u",                pos,                (U32)(*ip - *anchor), matchLength, (U32)(*ip-match),                cost, totalCost);    totalCost += cost;#endif    /* Encode Literal length */    length = (size_t)(*ip - *anchor);    if ((limit) && ((*op + (length >> 8) + length + (2 + 1 + LASTLITERALS)) > oend)) return 1;   /* Check output limit */    if (length >= RUN_MASK) {        size_t len = length - RUN_MASK;        *token = (RUN_MASK << ML_BITS);        for(; len >= 255 ; len -= 255) *(*op)++ = 255;        *(*op)++ = (BYTE)len;    } else {        *token = (BYTE)(length << ML_BITS);    }    /* Copy Literals */    LZ4_wildCopy(*op, *anchor, (*op) + length);    *op += length;    /* Encode Offset */    assert( (*ip - match) <= MAX_DISTANCE );   /* note : consider providing offset as a value, rather than as a pointer difference */    LZ4_writeLE16(*op, (U16)(*ip-match)); *op += 2;    /* Encode MatchLength */    assert(matchLength >= MINMATCH);    length = (size_t)(matchLength - MINMATCH);    if ((limit) && (*op + (length >> 8) + (1 + LASTLITERALS) > oend)) return 1;   /* Check output limit */    if (length >= ML_MASK) {        *token += ML_MASK;        length -= ML_MASK;        for(; length >= 510 ; length -= 510) { *(*op)++ = 255; *(*op)++ = 255; }        if (length >= 255) { length -= 255; *(*op)++ = 255; }        *(*op)++ = (BYTE)length;    } else {        *token += (BYTE)(length);    }    /* Prepare next loop */    *ip += matchLength;    *anchor = *ip;    return 0;}LZ4_FORCE_INLINE int LZ4HC_compress_hashChain (    LZ4HC_CCtx_internal* const ctx,    const char* const source,    char* const dest,    int* srcSizePtr,    int const maxOutputSize,    unsigned maxNbAttempts,    const limitedOutput_directive limit,    const dictCtx_directive dict    ){    const int inputSize = *srcSizePtr;    const int patternAnalysis = (maxNbAttempts > 128);   /* levels 9+ */    const BYTE* ip = (const BYTE*) source;    const BYTE* anchor = ip;    const BYTE* const iend = ip + inputSize;    const BYTE* const mflimit = iend - MFLIMIT;    const BYTE* const matchlimit = (iend - LASTLITERALS);    BYTE* optr = (BYTE*) dest;    BYTE* op = (BYTE*) dest;    BYTE* oend = op + maxOutputSize;    int   ml0, ml, ml2, ml3;    const BYTE* start0;    const BYTE* ref0;    const BYTE* ref = NULL;    const BYTE* start2 = NULL;    const BYTE* ref2 = NULL;    const BYTE* start3 = NULL;    const BYTE* ref3 = NULL;    /* init */    *srcSizePtr = 0;    if (limit == limitedDestSize) oend -= LASTLITERALS;                  /* Hack for support LZ4 format restriction */    if (inputSize < LZ4_minLength) goto _last_literals;                  /* Input too small, no compression (all literals) */    /* Main Loop */    while (ip <= mflimit) {        ml = LZ4HC_InsertAndFindBestMatch (ctx, ip, matchlimit, &ref, maxNbAttempts, patternAnalysis, dict);        if (ml<MINMATCH) { ip++; continue; }        /* saved, in case we would skip too much */        start0 = ip; ref0 = ref; ml0 = ml;_Search2:        if (ip+ml <= mflimit) {            ml2 = LZ4HC_InsertAndGetWiderMatch(ctx,                            ip + ml - 2, ip + 0, matchlimit, ml, &ref2, &start2,                            maxNbAttempts, patternAnalysis, 0, dict, favorCompressionRatio);        } else {            ml2 = ml;        }        if (ml2 == ml) { /* No better match => encode ML1 */            optr = op;            if (LZ4HC_encodeSequence(&ip, &op, &anchor, ml, ref, limit, oend)) goto _dest_overflow;            continue;        }        if (start0 < ip) {   /* first match was skipped at least once */            if (start2 < ip + ml0) {  /* squeezing ML1 between ML0(original ML1) and ML2 */                ip = start0; ref = ref0; ml = ml0;  /* restore initial ML1 */        }   }        /* Here, start0==ip */        if ((start2 - ip) < 3) {  /* First Match too small : removed */            ml = ml2;            ip = start2;            ref =ref2;            goto _Search2;        }_Search3:        /* At this stage, we have :        *  ml2 > ml1, and        *  ip1+3 <= ip2 (usually < ip1+ml1) */        if ((start2 - ip) < OPTIMAL_ML) {            int correction;            int new_ml = ml;            if (new_ml > OPTIMAL_ML) new_ml = OPTIMAL_ML;            if (ip+new_ml > start2 + ml2 - MINMATCH) new_ml = (int)(start2 - ip) + ml2 - MINMATCH;            correction = new_ml - (int)(start2 - ip);            if (correction > 0) {                start2 += correction;                ref2 += correction;                ml2 -= correction;            }        }        /* Now, we have start2 = ip+new_ml, with new_ml = min(ml, OPTIMAL_ML=18) */        if (start2 + ml2 <= mflimit) {            ml3 = LZ4HC_InsertAndGetWiderMatch(ctx,                            start2 + ml2 - 3, start2, matchlimit, ml2, &ref3, &start3,                            maxNbAttempts, patternAnalysis, 0, dict, favorCompressionRatio);        } else {            ml3 = ml2;        }        if (ml3 == ml2) {  /* No better match => encode ML1 and ML2 */            /* ip & ref are known; Now for ml */            if (start2 < ip+ml)  ml = (int)(start2 - ip);            /* Now, encode 2 sequences */            optr = op;            if (LZ4HC_encodeSequence(&ip, &op, &anchor, ml, ref, limit, oend)) goto _dest_overflow;            ip = start2;            optr = op;            if (LZ4HC_encodeSequence(&ip, &op, &anchor, ml2, ref2, limit, oend)) goto _dest_overflow;            continue;        }        if (start3 < ip+ml+3) {  /* Not enough space for match 2 : remove it */            if (start3 >= (ip+ml)) {  /* can write Seq1 immediately ==> Seq2 is removed, so Seq3 becomes Seq1 */                if (start2 < ip+ml) {                    int correction = (int)(ip+ml - start2);                    start2 += correction;                    ref2 += correction;                    ml2 -= correction;                    if (ml2 < MINMATCH) {                        start2 = start3;                        ref2 = ref3;                        ml2 = ml3;                    }                }                optr = op;                if (LZ4HC_encodeSequence(&ip, &op, &anchor, ml, ref, limit, oend)) goto _dest_overflow;                ip  = start3;                ref = ref3;                ml  = ml3;                start0 = start2;                ref0 = ref2;                ml0 = ml2;                goto _Search2;            }            start2 = start3;            ref2 = ref3;            ml2 = ml3;            goto _Search3;        }        /*        * OK, now we have 3 ascending matches;        * let's write the first one ML1.        * ip & ref are known; Now decide ml.        */        if (start2 < ip+ml) {            if ((start2 - ip) < OPTIMAL_ML) {                int correction;                if (ml > OPTIMAL_ML) ml = OPTIMAL_ML;                if (ip + ml > start2 + ml2 - MINMATCH) ml = (int)(start2 - ip) + ml2 - MINMATCH;                correction = ml - (int)(start2 - ip);                if (correction > 0) {                    start2 += correction;                    ref2 += correction;                    ml2 -= correction;                }            } else {                ml = (int)(start2 - ip);            }        }        optr = op;        if (LZ4HC_encodeSequence(&ip, &op, &anchor, ml, ref, limit, oend)) goto _dest_overflow;        /* ML2 becomes ML1 */        ip = start2; ref = ref2; ml = ml2;        /* ML3 becomes ML2 */        start2 = start3; ref2 = ref3; ml2 = ml3;        /* let's find a new ML3 */        goto _Search3;    }_last_literals:    /* Encode Last Literals */    {   size_t lastRunSize = (size_t)(iend - anchor);  /* literals */        size_t litLength = (lastRunSize + 255 - RUN_MASK) / 255;        size_t const totalSize = 1 + litLength + lastRunSize;        if (limit == limitedDestSize) oend += LASTLITERALS;  /* restore correct value */        if (limit && (op + totalSize > oend)) {            if (limit == limitedOutput) return 0;  /* Check output limit */            /* adapt lastRunSize to fill 'dest' */            lastRunSize  = (size_t)(oend - op) - 1;            litLength = (lastRunSize + 255 - RUN_MASK) / 255;            lastRunSize -= litLength;        }        ip = anchor + lastRunSize;        if (lastRunSize >= RUN_MASK) {            size_t accumulator = lastRunSize - RUN_MASK;            *op++ = (RUN_MASK << ML_BITS);            for(; accumulator >= 255 ; accumulator -= 255) *op++ = 255;            *op++ = (BYTE) accumulator;        } else {            *op++ = (BYTE)(lastRunSize << ML_BITS);        }        memcpy(op, anchor, lastRunSize);        op += lastRunSize;    }    /* End */    *srcSizePtr = (int) (((const char*)ip) - source);    return (int) (((char*)op)-dest);_dest_overflow:    if (limit == limitedDestSize) {        op = optr;  /* restore correct out pointer */        goto _last_literals;    }    return 0;}static int LZ4HC_compress_optimal( LZ4HC_CCtx_internal* ctx,    const char* const source, char* dst,    int* srcSizePtr, int dstCapacity,    int const nbSearches, size_t sufficient_len,    const limitedOutput_directive limit, int const fullUpdate,    const dictCtx_directive dict,    HCfavor_e favorDecSpeed);LZ4_FORCE_INLINE int LZ4HC_compress_generic_internal (    LZ4HC_CCtx_internal* const ctx,    const char* const src,    char* const dst,    int* const srcSizePtr,    int const dstCapacity,    int cLevel,    const limitedOutput_directive limit,    const dictCtx_directive dict    ){    typedef enum { lz4hc, lz4opt } lz4hc_strat_e;    typedef struct {        lz4hc_strat_e strat;        U32 nbSearches;        U32 targetLength;    } cParams_t;    static const cParams_t clTable[LZ4HC_CLEVEL_MAX+1] = {        { lz4hc,     2, 16 },  /* 0, unused */        { lz4hc,     2, 16 },  /* 1, unused */        { lz4hc,     2, 16 },  /* 2, unused */        { lz4hc,     4, 16 },  /* 3 */        { lz4hc,     8, 16 },  /* 4 */        { lz4hc,    16, 16 },  /* 5 */        { lz4hc,    32, 16 },  /* 6 */        { lz4hc,    64, 16 },  /* 7 */        { lz4hc,   128, 16 },  /* 8 */        { lz4hc,   256, 16 },  /* 9 */        { lz4opt,   96, 64 },  /*10==LZ4HC_CLEVEL_OPT_MIN*/        { lz4opt,  512,128 },  /*11 */        { lz4opt,16384,LZ4_OPT_NUM },  /* 12==LZ4HC_CLEVEL_MAX */    };    DEBUGLOG(4, "LZ4HC_compress_generic(%p, %p, %d)", ctx, src, *srcSizePtr);    if (limit == limitedDestSize && dstCapacity < 1) return 0;         /* Impossible to store anything */    if ((U32)*srcSizePtr > (U32)LZ4_MAX_INPUT_SIZE) return 0;          /* Unsupported input size (too large or negative) */    ctx->end += *srcSizePtr;    if (cLevel < 1) cLevel = LZ4HC_CLEVEL_DEFAULT;   /* note : convention is different from lz4frame, maybe something to review */    cLevel = MIN(LZ4HC_CLEVEL_MAX, cLevel);    {   cParams_t const cParam = clTable[cLevel];        HCfavor_e const favor = ctx->favorDecSpeed ? favorDecompressionSpeed : favorCompressionRatio;        if (cParam.strat == lz4hc)            return LZ4HC_compress_hashChain(ctx,                                src, dst, srcSizePtr, dstCapacity,                                cParam.nbSearches, limit, dict);        assert(cParam.strat == lz4opt);        return LZ4HC_compress_optimal(ctx,                            src, dst, srcSizePtr, dstCapacity,                            cParam.nbSearches, cParam.targetLength, limit,                            cLevel == LZ4HC_CLEVEL_MAX,   /* ultra mode */                            dict, favor);    }}static void LZ4HC_setExternalDict(LZ4HC_CCtx_internal* ctxPtr, const BYTE* newBlock);static int LZ4HC_compress_generic_noDictCtx (    LZ4HC_CCtx_internal* const ctx,    const char* const src,    char* const dst,    int* const srcSizePtr,    int const dstCapacity,    int cLevel,    limitedOutput_directive limit    ){    assert(ctx->dictCtx == NULL);    return LZ4HC_compress_generic_internal(ctx, src, dst, srcSizePtr, dstCapacity, cLevel, limit, noDictCtx);}static int LZ4HC_compress_generic_dictCtx (    LZ4HC_CCtx_internal* const ctx,    const char* const src,    char* const dst,    int* const srcSizePtr,    int const dstCapacity,    int cLevel,    limitedOutput_directive limit    ){    const size_t position = ctx->end - ctx->base - ctx->lowLimit;    assert(ctx->dictCtx != NULL);    if (position >= 64 KB) {        ctx->dictCtx = NULL;        return LZ4HC_compress_generic_noDictCtx(ctx, src, dst, srcSizePtr, dstCapacity, cLevel, limit);    } else if (position == 0 && *srcSizePtr > 4 KB) {        memcpy(ctx, ctx->dictCtx, sizeof(LZ4HC_CCtx_internal));        LZ4HC_setExternalDict(ctx, (const BYTE *)src);        ctx->compressionLevel = (short)cLevel;        return LZ4HC_compress_generic_noDictCtx(ctx, src, dst, srcSizePtr, dstCapacity, cLevel, limit);    } else {        return LZ4HC_compress_generic_internal(ctx, src, dst, srcSizePtr, dstCapacity, cLevel, limit, usingDictCtx);    }}static int LZ4HC_compress_generic (    LZ4HC_CCtx_internal* const ctx,    const char* const src,    char* const dst,    int* const srcSizePtr,    int const dstCapacity,    int cLevel,    limitedOutput_directive limit    ){    if (ctx->dictCtx == NULL) {        return LZ4HC_compress_generic_noDictCtx(ctx, src, dst, srcSizePtr, dstCapacity, cLevel, limit);    } else {        return LZ4HC_compress_generic_dictCtx(ctx, src, dst, srcSizePtr, dstCapacity, cLevel, limit);    }}static void LZ4_setCompressionLevel(LZ4_streamHC_t* LZ4_streamHCPtr, int compressionLevel){    if (compressionLevel < 1) compressionLevel = LZ4HC_CLEVEL_DEFAULT;    if (compressionLevel > LZ4HC_CLEVEL_MAX) compressionLevel = LZ4HC_CLEVEL_MAX;    LZ4_streamHCPtr->internal_donotuse.compressionLevel = (short)compressionLevel;}static void LZ4_resetStreamHC (LZ4_streamHC_t* LZ4_streamHCPtr, int compressionLevel){    LZ4_STATIC_ASSERT(sizeof(LZ4HC_CCtx_internal) <= sizeof(size_t) * LZ4_STREAMHCSIZE_SIZET);   /* if compilation fails here, LZ4_STREAMHCSIZE must be increased */    DEBUGLOG(4, "LZ4_resetStreamHC(%p, %d)", LZ4_streamHCPtr, compressionLevel);    LZ4_streamHCPtr->internal_donotuse.end = (const BYTE *)(ptrdiff_t)-1;    LZ4_streamHCPtr->internal_donotuse.base = NULL;    LZ4_streamHCPtr->internal_donotuse.dictCtx = NULL;    LZ4_streamHCPtr->internal_donotuse.favorDecSpeed = 0;    LZ4_streamHCPtr->internal_donotuse.dirty = 0;    LZ4_setCompressionLevel(LZ4_streamHCPtr, compressionLevel);}static void LZ4_resetStreamHC_fast (LZ4_streamHC_t* LZ4_streamHCPtr, int compressionLevel){    DEBUGLOG(4, "LZ4_resetStreamHC_fast(%p, %d)", LZ4_streamHCPtr, compressionLevel);    if (LZ4_streamHCPtr->internal_donotuse.dirty) {        LZ4_resetStreamHC(LZ4_streamHCPtr, compressionLevel);    } else {        LZ4_streamHCPtr->internal_donotuse.end -= (uptrval)LZ4_streamHCPtr->internal_donotuse.base;        LZ4_streamHCPtr->internal_donotuse.base = NULL;        LZ4_streamHCPtr->internal_donotuse.dictCtx = NULL;        LZ4_setCompressionLevel(LZ4_streamHCPtr, compressionLevel);    }}static int LZ4_compress_HC_extStateHC_fastReset (void* state, const char* src, char* dst, int srcSize, int dstCapacity, int compressionLevel){    LZ4HC_CCtx_internal* const ctx = &((LZ4_streamHC_t*)state)->internal_donotuse;    if (((size_t)(state)&(sizeof(void*)-1)) != 0) return 0;   /* Error : state is not aligned for pointers (32 or 64 bits) */    LZ4_resetStreamHC_fast((LZ4_streamHC_t*)state, compressionLevel);    LZ4HC_init (ctx, (const BYTE*)src);    if (dstCapacity < LZ4_compressBound(srcSize))        return LZ4HC_compress_generic (ctx, src, dst, &srcSize, dstCapacity, compressionLevel, limitedOutput);    else        return LZ4HC_compress_generic (ctx, src, dst, &srcSize, dstCapacity, compressionLevel, noLimit);}static int LZ4_compress_HC_extStateHC (void* state, const char* src, char* dst, int srcSize, int dstCapacity, int compressionLevel){    if (((size_t)(state)&(sizeof(void*)-1)) != 0) return 0;   /* Error : state is not aligned for pointers (32 or 64 bits) */    LZ4_resetStreamHC ((LZ4_streamHC_t*)state, compressionLevel);    return LZ4_compress_HC_extStateHC_fastReset(state, src, dst, srcSize, dstCapacity, compressionLevel);}int LZ4_compress_HC(const void* src, void* dst, int srcSize, int dstCapacity, int compressionLevel){#if defined(LZ4HC_HEAPMODE) && LZ4HC_HEAPMODE==1    LZ4_streamHC_t* const statePtr = (LZ4_streamHC_t*)ALLOC(sizeof(LZ4_streamHC_t));#else    LZ4_streamHC_t state;    LZ4_streamHC_t* const statePtr = &state;#endif    int const cSize = LZ4_compress_HC_extStateHC(statePtr, src, dst, srcSize, dstCapacity, compressionLevel);#if defined(LZ4HC_HEAPMODE) && LZ4HC_HEAPMODE==1    free(statePtr);#endif    return cSize;}/* LZ4_compress_HC_destSize() : * only compatible with regular HC parser */static int LZ4_compress_HC_destSize(void* LZ4HC_Data, const char* source, char* dest, int* sourceSizePtr, int targetDestSize, int cLevel){    LZ4HC_CCtx_internal* const ctx = &((LZ4_streamHC_t*)LZ4HC_Data)->internal_donotuse;    LZ4_resetStreamHC((LZ4_streamHC_t*)LZ4HC_Data, cLevel);    LZ4HC_init(ctx, (const BYTE*) source);    return LZ4HC_compress_generic(ctx, source, dest, sourceSizePtr, targetDestSize, cLevel, limitedDestSize);}/* ================================================ * LZ4 Optimal parser (levels 10-12) * ===============================================*/typedef struct {    int price;    int off;    int mlen;    int litlen;} LZ4HC_optimal_t;/* price in bytes */LZ4_FORCE_INLINE int LZ4HC_literalsPrice(int const litlen){    int price = litlen;    if (litlen >= (int)RUN_MASK)        price += 1 + (litlen-RUN_MASK)/255;    return price;}/* requires mlen >= MINMATCH */LZ4_FORCE_INLINE int LZ4HC_sequencePrice(int litlen, int mlen){    int price = 1 + 2 ; /* token + 16-bit offset */    price += LZ4HC_literalsPrice(litlen);    if (mlen >= (int)(ML_MASK+MINMATCH))        price += 1 + (mlen-(ML_MASK+MINMATCH))/255;    return price;}typedef struct {    int off;    int len;} LZ4HC_match_t;LZ4_FORCE_INLINE LZ4HC_match_tLZ4HC_FindLongerMatch(LZ4HC_CCtx_internal* const ctx,                      const BYTE* ip, const BYTE* const iHighLimit,                      int minLen, int nbSearches,                      const dictCtx_directive dict,                      const HCfavor_e favorDecSpeed){    LZ4HC_match_t match = { 0 , 0 };    const BYTE* matchPtr = NULL;    /* note : LZ4HC_InsertAndGetWiderMatch() is able to modify the starting position of a match (*startpos),     * but this won't be the case here, as we define iLowLimit==ip,     * so LZ4HC_InsertAndGetWiderMatch() won't be allowed to search past ip */    int matchLength = LZ4HC_InsertAndGetWiderMatch(ctx, ip, ip, iHighLimit, minLen, &matchPtr, &ip, nbSearches, 1 /*patternAnalysis*/, 1 /*chainSwap*/, dict, favorDecSpeed);    if (matchLength <= minLen) return match;    if (favorDecSpeed) {        if ((matchLength>18) & (matchLength<=36)) matchLength=18;   /* favor shortcut */    }    match.len = matchLength;    match.off = (int)(ip-matchPtr);    return match;}static int LZ4HC_compress_optimal ( LZ4HC_CCtx_internal* ctx,                                    const char* const source,                                    char* dst,                                    int* srcSizePtr,                                    int dstCapacity,                                    int const nbSearches,                                    size_t sufficient_len,                                    const limitedOutput_directive limit,                                    int const fullUpdate,                                    const dictCtx_directive dict,                                    const HCfavor_e favorDecSpeed){#define TRAILING_LITERALS 3    LZ4HC_optimal_t opt[LZ4_OPT_NUM + TRAILING_LITERALS];   /* ~64 KB, which is a bit large for stack... */    const BYTE* ip = (const BYTE*) source;    const BYTE* anchor = ip;    const BYTE* const iend = ip + *srcSizePtr;    const BYTE* const mflimit = iend - MFLIMIT;    const BYTE* const matchlimit = iend - LASTLITERALS;    BYTE* op = (BYTE*) dst;    BYTE* opSaved = (BYTE*) dst;    BYTE* oend = op + dstCapacity;    /* init */    DEBUGLOG(5, "LZ4HC_compress_optimal");    *srcSizePtr = 0;    if (limit == limitedDestSize) oend -= LASTLITERALS;   /* Hack for support LZ4 format restriction */    if (sufficient_len >= LZ4_OPT_NUM) sufficient_len = LZ4_OPT_NUM-1;    /* Main Loop */    assert(ip - anchor < LZ4_MAX_INPUT_SIZE);    while (ip <= mflimit) {         int const llen = (int)(ip - anchor);         int best_mlen, best_off;         int cur, last_match_pos = 0;         LZ4HC_match_t const firstMatch = LZ4HC_FindLongerMatch(ctx, ip, matchlimit, MINMATCH-1, nbSearches, dict, favorDecSpeed);         if (firstMatch.len==0) { ip++; continue; }         if ((size_t)firstMatch.len > sufficient_len) {             /* good enough solution : immediate encoding */             int const firstML = firstMatch.len;             const BYTE* const matchPos = ip - firstMatch.off;             opSaved = op;             if ( LZ4HC_encodeSequence(&ip, &op, &anchor, firstML, matchPos, limit, oend) )   /* updates ip, op and anchor */                 goto _dest_overflow;             continue;         }         /* set prices for first positions (literals) */         {   int rPos;             for (rPos = 0 ; rPos < MINMATCH ; rPos++) {                 int const cost = LZ4HC_literalsPrice(llen + rPos);                 opt[rPos].mlen = 1;                 opt[rPos].off = 0;                 opt[rPos].litlen = llen + rPos;                 opt[rPos].price = cost;                 DEBUGLOG(7, "rPos:%3i => price:%3i (litlen=%i) -- initial setup",                             rPos, cost, opt[rPos].litlen);         }   }         /* set prices using initial match */         {   int mlen = MINMATCH;             int const matchML = firstMatch.len;   /* necessarily < sufficient_len < LZ4_OPT_NUM */             int const offset = firstMatch.off;             assert(matchML < LZ4_OPT_NUM);             for ( ; mlen <= matchML ; mlen++) {                 int const cost = LZ4HC_sequencePrice(llen, mlen);                 opt[mlen].mlen = mlen;                 opt[mlen].off = offset;                 opt[mlen].litlen = llen;                 opt[mlen].price = cost;                 DEBUGLOG(7, "rPos:%3i => price:%3i (matchlen=%i) -- initial setup",                             mlen, cost, mlen);         }   }         last_match_pos = firstMatch.len;         {   int addLit;             for (addLit = 1; addLit <= TRAILING_LITERALS; addLit ++) {                 opt[last_match_pos+addLit].mlen = 1; /* literal */                 opt[last_match_pos+addLit].off = 0;                 opt[last_match_pos+addLit].litlen = addLit;                 opt[last_match_pos+addLit].price = opt[last_match_pos].price + LZ4HC_literalsPrice(addLit);                 DEBUGLOG(7, "rPos:%3i => price:%3i (litlen=%i) -- initial setup",                             last_match_pos+addLit, opt[last_match_pos+addLit].price, addLit);         }   }         /* check further positions */         for (cur = 1; cur < last_match_pos; cur++) {             const BYTE* const curPtr = ip + cur;             LZ4HC_match_t newMatch;             if (curPtr > mflimit) break;             DEBUGLOG(7, "rPos:%u[%u] vs [%u]%u",                     cur, opt[cur].price, opt[cur+1].price, cur+1);             if (fullUpdate) {                 /* not useful to search here if next position has same (or lower) cost */                 if ( (opt[cur+1].price <= opt[cur].price)                   /* in some cases, next position has same cost, but cost rises sharply after, so a small match would still be beneficial */                   && (opt[cur+MINMATCH].price < opt[cur].price + 3/*min seq price*/) )                     continue;             } else {                 /* not useful to search here if next position has same (or lower) cost */                 if (opt[cur+1].price <= opt[cur].price) continue;             }             DEBUGLOG(7, "search at rPos:%u", cur);             if (fullUpdate)                 newMatch = LZ4HC_FindLongerMatch(ctx, curPtr, matchlimit, MINMATCH-1, nbSearches, dict, favorDecSpeed);             else                 /* only test matches of minimum length; slightly faster, but misses a few bytes */                 newMatch = LZ4HC_FindLongerMatch(ctx, curPtr, matchlimit, last_match_pos - cur, nbSearches, dict, favorDecSpeed);             if (!newMatch.len) continue;             if ( ((size_t)newMatch.len > sufficient_len)               || (newMatch.len + cur >= LZ4_OPT_NUM) ) {                 /* immediate encoding */                 best_mlen = newMatch.len;                 best_off = newMatch.off;                 last_match_pos = cur + 1;                 goto encode;             }             /* before match : set price with literals at beginning */             {   int const baseLitlen = opt[cur].litlen;                 int litlen;                 for (litlen = 1; litlen < MINMATCH; litlen++) {                     int const price = opt[cur].price - LZ4HC_literalsPrice(baseLitlen) + LZ4HC_literalsPrice(baseLitlen+litlen);                     int const pos = cur + litlen;                     if (price < opt[pos].price) {                         opt[pos].mlen = 1; /* literal */                         opt[pos].off = 0;                         opt[pos].litlen = baseLitlen+litlen;                         opt[pos].price = price;                         DEBUGLOG(7, "rPos:%3i => price:%3i (litlen=%i)",                                     pos, price, opt[pos].litlen);             }   }   }             /* set prices using match at position = cur */             {   int const matchML = newMatch.len;                 int ml = MINMATCH;                 assert(cur + newMatch.len < LZ4_OPT_NUM);                 for ( ; ml <= matchML ; ml++) {                     int const pos = cur + ml;                     int const offset = newMatch.off;                     int price;                     int ll;                     DEBUGLOG(7, "testing price rPos %i (last_match_pos=%i)",                                 pos, last_match_pos);                     if (opt[cur].mlen == 1) {                         ll = opt[cur].litlen;                         price = ((cur > ll) ? opt[cur - ll].price : 0)                               + LZ4HC_sequencePrice(ll, ml);                     } else {                         ll = 0;                         price = opt[cur].price + LZ4HC_sequencePrice(0, ml);                     }                    assert((U32)favorDecSpeed <= 1);                     if (pos > last_match_pos+TRAILING_LITERALS                      || price <= opt[pos].price - (int)favorDecSpeed) {                         DEBUGLOG(7, "rPos:%3i => price:%3i (matchlen=%i)",                                     pos, price, ml);                         assert(pos < LZ4_OPT_NUM);                         if ( (ml == matchML)  /* last pos of last match */                           && (last_match_pos < pos) )                             last_match_pos = pos;                         opt[pos].mlen = ml;                         opt[pos].off = offset;                         opt[pos].litlen = ll;                         opt[pos].price = price;             }   }   }             /* complete following positions with literals */             {   int addLit;                 for (addLit = 1; addLit <= TRAILING_LITERALS; addLit ++) {                     opt[last_match_pos+addLit].mlen = 1; /* literal */                     opt[last_match_pos+addLit].off = 0;                     opt[last_match_pos+addLit].litlen = addLit;                     opt[last_match_pos+addLit].price = opt[last_match_pos].price + LZ4HC_literalsPrice(addLit);                     DEBUGLOG(7, "rPos:%3i => price:%3i (litlen=%i)", last_match_pos+addLit, opt[last_match_pos+addLit].price, addLit);             }   }         }  /* for (cur = 1; cur <= last_match_pos; cur++) */         best_mlen = opt[last_match_pos].mlen;         best_off = opt[last_match_pos].off;         cur = last_match_pos - best_mlen; encode: /* cur, last_match_pos, best_mlen, best_off must be set */         assert(cur < LZ4_OPT_NUM);         assert(last_match_pos >= 1);  /* == 1 when only one candidate */         DEBUGLOG(6, "reverse traversal, looking for shortest path (last_match_pos=%i)", last_match_pos);         {   int candidate_pos = cur;             int selected_matchLength = best_mlen;             int selected_offset = best_off;             while (1) {  /* from end to beginning */                 int const next_matchLength = opt[candidate_pos].mlen;  /* can be 1, means literal */                 int const next_offset = opt[candidate_pos].off;                 DEBUGLOG(7, "pos %i: sequence length %i", candidate_pos, selected_matchLength);                 opt[candidate_pos].mlen = selected_matchLength;                 opt[candidate_pos].off = selected_offset;                 selected_matchLength = next_matchLength;                 selected_offset = next_offset;                 if (next_matchLength > candidate_pos) break; /* last match elected, first match to encode */                 assert(next_matchLength > 0);  /* can be 1, means literal */                 candidate_pos -= next_matchLength;         }   }         /* encode all recorded sequences in order */         {   int rPos = 0;  /* relative position (to ip) */             while (rPos < last_match_pos) {                 int const ml = opt[rPos].mlen;                 int const offset = opt[rPos].off;                 if (ml == 1) { ip++; rPos++; continue; }  /* literal; note: can end up with several literals, in which case, skip them */                 rPos += ml;                 assert(ml >= MINMATCH);                 assert((offset >= 1) && (offset <= MAX_DISTANCE));                 opSaved = op;                 if ( LZ4HC_encodeSequence(&ip, &op, &anchor, ml, ip - offset, limit, oend) )   /* updates ip, op and anchor */                     goto _dest_overflow;         }   }     }  /* while (ip <= mflimit) */ _last_literals:     /* Encode Last Literals */     {   size_t lastRunSize = (size_t)(iend - anchor);  /* literals */         size_t litLength = (lastRunSize + 255 - RUN_MASK) / 255;         size_t const totalSize = 1 + litLength + lastRunSize;         if (limit == limitedDestSize) oend += LASTLITERALS;  /* restore correct value */         if (limit && (op + totalSize > oend)) {             if (limit == limitedOutput) return 0;  /* Check output limit */             /* adapt lastRunSize to fill 'dst' */             lastRunSize  = (size_t)(oend - op) - 1;             litLength = (lastRunSize + 255 - RUN_MASK) / 255;             lastRunSize -= litLength;         }         ip = anchor + lastRunSize;         if (lastRunSize >= RUN_MASK) {             size_t accumulator = lastRunSize - RUN_MASK;             *op++ = (RUN_MASK << ML_BITS);             for(; accumulator >= 255 ; accumulator -= 255) *op++ = 255;             *op++ = (BYTE) accumulator;         } else {             *op++ = (BYTE)(lastRunSize << ML_BITS);         }         memcpy(op, anchor, lastRunSize);         op += lastRunSize;     }     /* End */     *srcSizePtr = (int) (((const char*)ip) - source);     return (int) ((char*)op-dst); _dest_overflow:     if (limit == limitedDestSize) {         op = opSaved;  /* restore correct out pointer */         goto _last_literals;     }     return 0;}typedef enum { endOnOutputSize = 0, endOnInputSize = 1 } endCondition_directive;typedef enum { decode_full_block = 0, partial_decode = 1 } earlyEnd_directive;typedef enum { noDict = 0, withPrefix64k, usingExtDict } dict_directive;typedef enum { noDictIssue = 0, dictSmall } dictIssue_directive;/*! LZ4_decompress_generic() : *  This generic decompression function covers all use cases. *  It shall be instantiated several times, using different sets of directives. *  Note that it is important for performance that this function really get inlined, *  in order to remove useless branches during compilation optimization. */LZ4_FORCE_INLINE intLZ4_decompress_generic(                 const char* const src,                 char* const dst,                 int srcSize,                 int outputSize,         /* If endOnInput==endOnInputSize, this value is `dstCapacity` */                 endCondition_directive endOnInput,   /* endOnOutputSize, endOnInputSize */                 earlyEnd_directive partialDecoding,  /* full, partial */                 dict_directive dict,                 /* noDict, withPrefix64k, usingExtDict */                 const BYTE* const lowPrefix,  /* always <= dst, == dst when no prefix */                 const BYTE* const dictStart,  /* only if dict==usingExtDict */                 const size_t dictSize         /* note : = 0 if noDict */                 ){    const BYTE* ip = (const BYTE*) src;    const BYTE* const iend = ip + srcSize;    BYTE* op = (BYTE*) dst;    BYTE* const oend = op + outputSize;    BYTE* cpy;    const BYTE* const dictEnd = (const BYTE*)dictStart + dictSize;    const unsigned inc32table[8] = {0, 1, 2,  1,  0,  4, 4, 4};    const int      dec64table[8] = {0, 0, 0, -1, -4,  1, 2, 3};    const int safeDecode = (endOnInput==endOnInputSize);    const int checkOffset = ((safeDecode) && (dictSize < (int)(64 KB)));    /* Set up the "end" pointers for the shortcut. */    const BYTE* const shortiend = iend - (endOnInput ? 14 : 8) /*maxLL*/ - 2 /*offset*/;    const BYTE* const shortoend = oend - (endOnInput ? 14 : 8) /*maxLL*/ - 18 /*maxML*/;    DEBUGLOG(5, "LZ4_decompress_generic (srcSize:%i, dstSize:%i)", srcSize, outputSize);    /* Special cases */    assert(lowPrefix <= op);    assert(src != NULL);    if ((endOnInput) && (unlikely(outputSize==0))) return ((srcSize==1) && (*ip==0)) ? 0 : -1;  /* Empty output buffer */    if ((!endOnInput) && (unlikely(outputSize==0))) return (*ip==0 ? 1 : -1);    if ((endOnInput) && unlikely(srcSize==0)) return -1;    /* Main Loop : decode sequences */    while (1) {        const BYTE* match;        size_t offset;        unsigned const token = *ip++;        size_t length = token >> ML_BITS;  /* literal length */        assert(!endOnInput || ip <= iend); /* ip < iend before the increment */        /* A two-stage shortcut for the most common case:         * 1) If the literal length is 0..14, and there is enough space,         * enter the shortcut and copy 16 bytes on behalf of the literals         * (in the fast mode, only 8 bytes can be safely copied this way).         * 2) Further if the match length is 4..18, copy 18 bytes in a similar         * manner; but we ensure that there's enough space in the output for         * those 18 bytes earlier, upon entering the shortcut (in other words,         * there is a combined check for both stages).         */        if ( (endOnInput ? length != RUN_MASK : length <= 8)            /* strictly "less than" on input, to re-enter the loop with at least one byte */          && likely((endOnInput ? ip < shortiend : 1) & (op <= shortoend)) ) {            /* Copy the literals */            memcpy(op, ip, endOnInput ? 16 : 8);            op += length; ip += length;            /* The second stage: prepare for match copying, decode full info.             * If it doesn't work out, the info won't be wasted. */            length = token & ML_MASK; /* match length */            offset = LZ4_readLE16(ip); ip += 2;            match = op - offset;            assert(match <= op); /* check overflow */            /* Do not deal with overlapping matches. */            if ( (length != ML_MASK)              && (offset >= 8)              && (dict==withPrefix64k || match >= lowPrefix) ) {                /* Copy the match. */                memcpy(op + 0, match + 0, 8);                memcpy(op + 8, match + 8, 8);                memcpy(op +16, match +16, 2);                op += length + MINMATCH;                /* Both stages worked, load the next token. */                continue;            }            /* The second stage didn't work out, but the info is ready.             * Propel it right to the point of match copying. */            goto _copy_match;        }        /* decode literal length */        if (length == RUN_MASK) {            unsigned s;            if (unlikely(endOnInput ? ip >= iend-RUN_MASK : 0)) goto _output_error;   /* overflow detection */            do {                s = *ip++;                length += s;            } while ( likely(endOnInput ? ip<iend-RUN_MASK : 1) & (s==255) );            if ((safeDecode) && unlikely((uptrval)(op)+length<(uptrval)(op))) goto _output_error;   /* overflow detection */            if ((safeDecode) && unlikely((uptrval)(ip)+length<(uptrval)(ip))) goto _output_error;   /* overflow detection */        }        /* copy literals */        cpy = op+length;        LZ4_STATIC_ASSERT(MFLIMIT >= WILDCOPYLENGTH);        if ( ((endOnInput) && ((cpy>oend-MFLIMIT) || (ip+length>iend-(2+1+LASTLITERALS))) )          || ((!endOnInput) && (cpy>oend-WILDCOPYLENGTH)) )        {            if (partialDecoding) {                if (cpy > oend) { cpy = oend; length = oend-op; }             /* Partial decoding : stop in the middle of literal segment */                if ((endOnInput) && (ip+length > iend)) goto _output_error;   /* Error : read attempt beyond end of input buffer */            } else {                if ((!endOnInput) && (cpy != oend)) goto _output_error;       /* Error : block decoding must stop exactly there */                if ((endOnInput) && ((ip+length != iend) || (cpy > oend))) goto _output_error;   /* Error : input must be consumed */            }            memcpy(op, ip, length);            ip += length;            op += length;            if (!partialDecoding || (cpy == oend)) {                /* Necessarily EOF, due to parsing restrictions */                break;            }        } else {            LZ4_wildCopy(op, ip, cpy);   /* may overwrite up to WILDCOPYLENGTH beyond cpy */            ip += length; op = cpy;        }        /* get offset */        offset = LZ4_readLE16(ip); ip+=2;        match = op - offset;        /* get matchlength */        length = token & ML_MASK;_copy_match:        if ((checkOffset) && (unlikely(match + dictSize < lowPrefix))) goto _output_error;   /* Error : offset outside buffers */        if (!partialDecoding) {            assert(oend > op);            assert(oend - op >= 4);            LZ4_write32(op, 0);   /* silence an msan warning when offset==0; costs <1%; */        }   /* note : when partialDecoding, there is no guarantee that at least 4 bytes remain available in output buffer */        if (length == ML_MASK) {            unsigned s;            do {                s = *ip++;                if ((endOnInput) && (ip > iend-LASTLITERALS)) goto _output_error;                length += s;            } while (s==255);            if ((safeDecode) && unlikely((uptrval)(op)+length<(uptrval)op)) goto _output_error;   /* overflow detection */        }        length += MINMATCH;        /* match starting within external dictionary */        if ((dict==usingExtDict) && (match < lowPrefix)) {            if (unlikely(op+length > oend-LASTLITERALS)) {                if (partialDecoding) length = MIN(length, (size_t)(oend-op));                else goto _output_error;   /* doesn't respect parsing restriction */            }            if (length <= (size_t)(lowPrefix-match)) {                /* match fits entirely within external dictionary : just copy */                memmove(op, dictEnd - (lowPrefix-match), length);                op += length;            } else {                /* match stretches into both external dictionary and current block */                size_t const copySize = (size_t)(lowPrefix - match);                size_t const restSize = length - copySize;                memcpy(op, dictEnd - copySize, copySize);                op += copySize;                if (restSize > (size_t)(op - lowPrefix)) {  /* overlap copy */                    BYTE* const endOfMatch = op + restSize;                    const BYTE* copyFrom = lowPrefix;                    while (op < endOfMatch) *op++ = *copyFrom++;                } else {                    memcpy(op, lowPrefix, restSize);                    op += restSize;            }   }            continue;        }        /* copy match within block */        cpy = op + length;        /* partialDecoding : may not respect endBlock parsing restrictions */        assert(op<=oend);        if (partialDecoding && (cpy > oend-MATCH_SAFEGUARD_DISTANCE)) {            size_t const mlen = MIN(length, (size_t)(oend-op));            const BYTE* const matchEnd = match + mlen;            BYTE* const copyEnd = op + mlen;            if (matchEnd > op) {   /* overlap copy */                while (op < copyEnd) *op++ = *match++;            } else {                memcpy(op, match, mlen);            }            op = copyEnd;            if (op==oend) break;            continue;        }        if (unlikely(offset<8)) {            op[0] = match[0];            op[1] = match[1];            op[2] = match[2];            op[3] = match[3];            match += inc32table[offset];            memcpy(op+4, match, 4);            match -= dec64table[offset];        } else {            memcpy(op, match, 8);            match += 8;        }        op += 8;        if (unlikely(cpy > oend-MATCH_SAFEGUARD_DISTANCE)) {            BYTE* const oCopyLimit = oend - (WILDCOPYLENGTH-1);            if (cpy > oend-LASTLITERALS) goto _output_error;    /* Error : last LASTLITERALS bytes must be literals (uncompressed) */            if (op < oCopyLimit) {                LZ4_wildCopy(op, match, oCopyLimit);                match += oCopyLimit - op;                op = oCopyLimit;            }            while (op < cpy) *op++ = *match++;        } else {            memcpy(op, match, 8);            if (length > 16) LZ4_wildCopy(op+8, match+8, cpy);        }        op = cpy;   /* wildcopy correction */    }    /* end of decoding */    if (endOnInput)       return (int) (((char*)op)-dst);     /* Nb of output bytes decoded */    else       return (int) (((const char*)ip)-src);   /* Nb of input bytes read */    /* Overflow error detected */_output_error:    return (int) (-(((const char*)ip)-src))-1;}int LZ4_decompress_fast(const void* source, void* dest, int originalSize){    return LZ4_decompress_generic(source, dest, 0, originalSize,                                  endOnOutputSize, decode_full_block, withPrefix64k,                                  (BYTE*)dest - 64 KB, NULL, 0);}
 |