Merge pull request #1154 from facebook/altOpt
minor optimal parser optimization
This commit is contained in:
commit
ac4f7ead3b
@ -49,7 +49,7 @@ const char* ZSTD_getErrorString(ZSTD_ErrorCode code) { return ERR_getErrorString
|
|||||||
/*! g_debuglog_enable :
|
/*! g_debuglog_enable :
|
||||||
* turn on/off debug traces (global switch) */
|
* turn on/off debug traces (global switch) */
|
||||||
#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG >= 2)
|
#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG >= 2)
|
||||||
int g_debuglog_enable = 1;
|
int g_debuglevel = ZSTD_DEBUG;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
@ -56,7 +56,7 @@ extern "C" {
|
|||||||
#undef DEBUGLOG
|
#undef DEBUGLOG
|
||||||
#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG>=2)
|
#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG>=2)
|
||||||
# include <stdio.h>
|
# include <stdio.h>
|
||||||
extern int g_debuglog_enable;
|
extern int g_debuglevel;
|
||||||
/* recommended values for ZSTD_DEBUG display levels :
|
/* recommended values for ZSTD_DEBUG display levels :
|
||||||
* 1 : no display, enables assert() only
|
* 1 : no display, enables assert() only
|
||||||
* 2 : reserved for currently active debug path
|
* 2 : reserved for currently active debug path
|
||||||
@ -65,11 +65,11 @@ extern int g_debuglog_enable;
|
|||||||
* 5 : events once per block
|
* 5 : events once per block
|
||||||
* 6 : events once per sequence (*very* verbose) */
|
* 6 : events once per sequence (*very* verbose) */
|
||||||
# define RAWLOG(l, ...) { \
|
# define RAWLOG(l, ...) { \
|
||||||
if ((g_debuglog_enable) & (l<=ZSTD_DEBUG)) { \
|
if (l<=g_debuglevel) { \
|
||||||
fprintf(stderr, __VA_ARGS__); \
|
fprintf(stderr, __VA_ARGS__); \
|
||||||
} }
|
} }
|
||||||
# define DEBUGLOG(l, ...) { \
|
# define DEBUGLOG(l, ...) { \
|
||||||
if ((g_debuglog_enable) & (l<=ZSTD_DEBUG)) { \
|
if (l<=g_debuglevel) { \
|
||||||
fprintf(stderr, __FILE__ ": " __VA_ARGS__); \
|
fprintf(stderr, __FILE__ ": " __VA_ARGS__); \
|
||||||
fprintf(stderr, " \n"); \
|
fprintf(stderr, " \n"); \
|
||||||
} }
|
} }
|
||||||
|
@ -299,7 +299,7 @@ MEM_STATIC void ZSTD_storeSeq(seqStore_t* seqStorePtr, size_t litLength, const v
|
|||||||
static const BYTE* g_start = NULL;
|
static const BYTE* g_start = NULL;
|
||||||
if (g_start==NULL) g_start = (const BYTE*)literals; /* note : index only works for compression within a single segment */
|
if (g_start==NULL) g_start = (const BYTE*)literals; /* note : index only works for compression within a single segment */
|
||||||
{ U32 const pos = (U32)((const BYTE*)literals - g_start);
|
{ U32 const pos = (U32)((const BYTE*)literals - g_start);
|
||||||
DEBUGLOG(6, "Cpos%7u :%3u literals, match%3u bytes at dist.code%7u",
|
DEBUGLOG(6, "Cpos%7u :%3u literals, match%4u bytes at offCode%7u",
|
||||||
pos, (U32)litLength, (U32)mlBase+MINMATCH, (U32)offsetCode);
|
pos, (U32)litLength, (U32)mlBase+MINMATCH, (U32)offsetCode);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -191,8 +191,10 @@ static U32 ZSTD_rawLiteralsCost(const BYTE* const literals, U32 const litLength,
|
|||||||
/* dynamic statistics */
|
/* dynamic statistics */
|
||||||
{ U32 price = litLength * optPtr->litSumBasePrice;
|
{ U32 price = litLength * optPtr->litSumBasePrice;
|
||||||
U32 u;
|
U32 u;
|
||||||
for (u=0; u < litLength; u++)
|
for (u=0; u < litLength; u++) {
|
||||||
|
assert(WEIGHT(optPtr->litFreq[literals[u]], optLevel) <= optPtr->litSumBasePrice); /* literal cost should never be negative */
|
||||||
price -= WEIGHT(optPtr->litFreq[literals[u]], optLevel);
|
price -= WEIGHT(optPtr->litFreq[literals[u]], optLevel);
|
||||||
|
}
|
||||||
return price;
|
return price;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -209,17 +211,6 @@ static U32 ZSTD_litLengthPrice(U32 const litLength, const optState_t* const optP
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ZSTD_litLengthPrice() :
|
|
||||||
* cost of the literal part of a sequence,
|
|
||||||
* including literals themselves, and literalLength symbol */
|
|
||||||
static U32 ZSTD_fullLiteralsCost(const BYTE* const literals, U32 const litLength,
|
|
||||||
const optState_t* const optPtr,
|
|
||||||
int optLevel)
|
|
||||||
{
|
|
||||||
return ZSTD_rawLiteralsCost(literals, litLength, optPtr, optLevel)
|
|
||||||
+ ZSTD_litLengthPrice(litLength, optPtr, optLevel);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ZSTD_litLengthContribution() :
|
/* ZSTD_litLengthContribution() :
|
||||||
* @return ( cost(litlength) - cost(0) )
|
* @return ( cost(litlength) - cost(0) )
|
||||||
* this value can then be added to rawLiteralsCost()
|
* this value can then be added to rawLiteralsCost()
|
||||||
@ -288,6 +279,8 @@ ZSTD_getMatchPrice(U32 const offset,
|
|||||||
return price;
|
return price;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* ZSTD_updateStats() :
|
||||||
|
* assumption : literals + litLengtn <= iend */
|
||||||
static void ZSTD_updateStats(optState_t* const optPtr,
|
static void ZSTD_updateStats(optState_t* const optPtr,
|
||||||
U32 litLength, const BYTE* literals,
|
U32 litLength, const BYTE* literals,
|
||||||
U32 offsetCode, U32 matchLength)
|
U32 offsetCode, U32 matchLength)
|
||||||
@ -556,8 +549,8 @@ U32 ZSTD_insertBtAndGetAllMatches (
|
|||||||
} }
|
} }
|
||||||
/* save longer solution */
|
/* save longer solution */
|
||||||
if (repLen > bestLength) {
|
if (repLen > bestLength) {
|
||||||
DEBUGLOG(8, "found rep-match %u of length %u",
|
DEBUGLOG(8, "found repCode %u (ll0:%u, offset:%u) of length %u",
|
||||||
repCode - ll0, (U32)repLen);
|
repCode, ll0, repOffset, repLen);
|
||||||
bestLength = repLen;
|
bestLength = repLen;
|
||||||
matches[mnum].off = repCode - ll0;
|
matches[mnum].off = repCode - ll0;
|
||||||
matches[mnum].len = (U32)repLen;
|
matches[mnum].len = (U32)repLen;
|
||||||
@ -617,8 +610,8 @@ U32 ZSTD_insertBtAndGetAllMatches (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (matchLength > bestLength) {
|
if (matchLength > bestLength) {
|
||||||
DEBUGLOG(8, "found match of length %u at distance %u",
|
DEBUGLOG(8, "found match of length %u at distance %u (offCode=%u)",
|
||||||
(U32)matchLength, current - matchIndex);
|
(U32)matchLength, current - matchIndex, current - matchIndex + ZSTD_REP_MOVE);
|
||||||
assert(matchEndIdx > matchIndex);
|
assert(matchEndIdx > matchIndex);
|
||||||
if (matchLength > matchEndIdx - matchIndex)
|
if (matchLength > matchEndIdx - matchIndex)
|
||||||
matchEndIdx = matchIndex + (U32)matchLength;
|
matchEndIdx = matchIndex + (U32)matchLength;
|
||||||
@ -706,60 +699,9 @@ repcodes_t ZSTD_updateRep(U32 const rep[3], U32 const offset, U32 const ll0)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
typedef struct {
|
static U32 ZSTD_totalLen(ZSTD_optimal_t sol)
|
||||||
const BYTE* anchor;
|
|
||||||
U32 litlen;
|
|
||||||
U32 rawLitCost;
|
|
||||||
} cachedLiteralPrice_t;
|
|
||||||
|
|
||||||
static U32 ZSTD_rawLiteralsCost_cached(
|
|
||||||
cachedLiteralPrice_t* const cachedLitPrice,
|
|
||||||
const BYTE* const anchor, U32 const litlen,
|
|
||||||
const optState_t* const optStatePtr,
|
|
||||||
int optLevel)
|
|
||||||
{
|
{
|
||||||
U32 startCost;
|
return sol.litlen + sol.mlen;
|
||||||
U32 remainingLength;
|
|
||||||
const BYTE* startPosition;
|
|
||||||
|
|
||||||
if (anchor == cachedLitPrice->anchor) {
|
|
||||||
startCost = cachedLitPrice->rawLitCost;
|
|
||||||
startPosition = anchor + cachedLitPrice->litlen;
|
|
||||||
assert(litlen >= cachedLitPrice->litlen);
|
|
||||||
remainingLength = litlen - cachedLitPrice->litlen;
|
|
||||||
} else {
|
|
||||||
startCost = 0;
|
|
||||||
startPosition = anchor;
|
|
||||||
remainingLength = litlen;
|
|
||||||
}
|
|
||||||
|
|
||||||
{ U32 const rawLitCost = startCost + ZSTD_rawLiteralsCost(startPosition, remainingLength, optStatePtr, optLevel);
|
|
||||||
cachedLitPrice->anchor = anchor;
|
|
||||||
cachedLitPrice->litlen = litlen;
|
|
||||||
cachedLitPrice->rawLitCost = rawLitCost;
|
|
||||||
return rawLitCost;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static U32 ZSTD_fullLiteralsCost_cached(
|
|
||||||
cachedLiteralPrice_t* const cachedLitPrice,
|
|
||||||
const BYTE* const anchor, U32 const litlen,
|
|
||||||
const optState_t* const optStatePtr,
|
|
||||||
int optLevel)
|
|
||||||
{
|
|
||||||
return ZSTD_rawLiteralsCost_cached(cachedLitPrice, anchor, litlen, optStatePtr, optLevel)
|
|
||||||
+ ZSTD_litLengthPrice(litlen, optStatePtr, optLevel);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int ZSTD_literalsContribution_cached(
|
|
||||||
cachedLiteralPrice_t* const cachedLitPrice,
|
|
||||||
const BYTE* const anchor, U32 const litlen,
|
|
||||||
const optState_t* const optStatePtr,
|
|
||||||
int optLevel)
|
|
||||||
{
|
|
||||||
int const contribution = ZSTD_rawLiteralsCost_cached(cachedLitPrice, anchor, litlen, optStatePtr, optLevel)
|
|
||||||
+ ZSTD_litLengthContribution(litlen, optStatePtr, optLevel);
|
|
||||||
return contribution;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FORCE_INLINE_TEMPLATE size_t
|
FORCE_INLINE_TEMPLATE size_t
|
||||||
@ -784,7 +726,7 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
|
|
||||||
ZSTD_optimal_t* const opt = optStatePtr->priceTable;
|
ZSTD_optimal_t* const opt = optStatePtr->priceTable;
|
||||||
ZSTD_match_t* const matches = optStatePtr->matchTable;
|
ZSTD_match_t* const matches = optStatePtr->matchTable;
|
||||||
cachedLiteralPrice_t cachedLitPrice;
|
ZSTD_optimal_t lastSequence;
|
||||||
|
|
||||||
/* init */
|
/* init */
|
||||||
DEBUGLOG(5, "ZSTD_compressBlock_opt_generic");
|
DEBUGLOG(5, "ZSTD_compressBlock_opt_generic");
|
||||||
@ -792,12 +734,10 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
ms->nextToUpdate3 = ms->nextToUpdate;
|
ms->nextToUpdate3 = ms->nextToUpdate;
|
||||||
ZSTD_rescaleFreqs(optStatePtr, (const BYTE*)src, srcSize, optLevel);
|
ZSTD_rescaleFreqs(optStatePtr, (const BYTE*)src, srcSize, optLevel);
|
||||||
ip += (ip==prefixStart);
|
ip += (ip==prefixStart);
|
||||||
memset(&cachedLitPrice, 0, sizeof(cachedLitPrice));
|
|
||||||
|
|
||||||
/* Match Loop */
|
/* Match Loop */
|
||||||
while (ip < ilimit) {
|
while (ip < ilimit) {
|
||||||
U32 cur, last_pos = 0;
|
U32 cur, last_pos = 0;
|
||||||
U32 best_mlen, best_off;
|
|
||||||
|
|
||||||
/* find first match */
|
/* find first match */
|
||||||
{ U32 const litlen = (U32)(ip - anchor);
|
{ U32 const litlen = (U32)(ip - anchor);
|
||||||
@ -807,27 +747,29 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
|
|
||||||
/* initialize opt[0] */
|
/* initialize opt[0] */
|
||||||
{ U32 i ; for (i=0; i<ZSTD_REP_NUM; i++) opt[0].rep[i] = rep[i]; }
|
{ U32 i ; for (i=0; i<ZSTD_REP_NUM; i++) opt[0].rep[i] = rep[i]; }
|
||||||
opt[0].mlen = 1; /* means is_a_literal */
|
opt[0].mlen = 0; /* means is_a_literal */
|
||||||
opt[0].litlen = litlen;
|
opt[0].litlen = litlen;
|
||||||
|
opt[0].price = ZSTD_literalsContribution(anchor, litlen, optStatePtr, optLevel);
|
||||||
|
|
||||||
/* large match -> immediate encoding */
|
/* large match -> immediate encoding */
|
||||||
{ U32 const maxML = matches[nbMatches-1].len;
|
{ U32 const maxML = matches[nbMatches-1].len;
|
||||||
U32 const maxOffset = matches[nbMatches-1].off;
|
U32 const maxOffset = matches[nbMatches-1].off;
|
||||||
DEBUGLOG(7, "found %u matches of maxLength=%u and maxOffset=%u at cPos=%u => start new serie",
|
DEBUGLOG(6, "found %u matches of maxLength=%u and maxOffCode=%u at cPos=%u => start new serie",
|
||||||
nbMatches, maxML, maxOffset, (U32)(ip-prefixStart));
|
nbMatches, maxML, maxOffset, (U32)(ip-prefixStart));
|
||||||
|
|
||||||
if (maxML > sufficient_len) {
|
if (maxML > sufficient_len) {
|
||||||
best_mlen = maxML;
|
lastSequence.litlen = litlen;
|
||||||
best_off = maxOffset;
|
lastSequence.mlen = maxML;
|
||||||
DEBUGLOG(7, "large match (%u>%u), immediate encoding",
|
lastSequence.off = maxOffset;
|
||||||
best_mlen, sufficient_len);
|
DEBUGLOG(6, "large match (%u>%u), immediate encoding",
|
||||||
|
maxML, sufficient_len);
|
||||||
cur = 0;
|
cur = 0;
|
||||||
last_pos = 1;
|
last_pos = ZSTD_totalLen(lastSequence);
|
||||||
goto _shortestPath;
|
goto _shortestPath;
|
||||||
} }
|
} }
|
||||||
|
|
||||||
/* set prices for first matches starting position == 0 */
|
/* set prices for first matches starting position == 0 */
|
||||||
{ U32 const literalsPrice = ZSTD_fullLiteralsCost_cached(&cachedLitPrice, anchor, litlen, optStatePtr, optLevel);
|
{ U32 const literalsPrice = opt[0].price + ZSTD_litLengthPrice(0, optStatePtr, optLevel);
|
||||||
U32 pos;
|
U32 pos;
|
||||||
U32 matchNb;
|
U32 matchNb;
|
||||||
for (pos = 1; pos < minMatch; pos++) {
|
for (pos = 1; pos < minMatch; pos++) {
|
||||||
@ -857,27 +799,28 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
for (cur = 1; cur <= last_pos; cur++) {
|
for (cur = 1; cur <= last_pos; cur++) {
|
||||||
const BYTE* const inr = ip + cur;
|
const BYTE* const inr = ip + cur;
|
||||||
assert(cur < ZSTD_OPT_NUM);
|
assert(cur < ZSTD_OPT_NUM);
|
||||||
|
DEBUGLOG(7, "cPos:%zi==rPos:%u", inr-istart, cur)
|
||||||
|
|
||||||
/* Fix current position with one literal if cheaper */
|
/* Fix current position with one literal if cheaper */
|
||||||
{ U32 const litlen = (opt[cur-1].mlen == 1) ? opt[cur-1].litlen + 1 : 1;
|
{ U32 const litlen = (opt[cur-1].mlen == 0) ? opt[cur-1].litlen + 1 : 1;
|
||||||
int price; /* note : contribution can be negative */
|
int const price = opt[cur-1].price
|
||||||
if (cur > litlen) {
|
+ ZSTD_rawLiteralsCost(ip+cur-1, 1, optStatePtr, optLevel)
|
||||||
price = opt[cur - litlen].price + ZSTD_literalsContribution(inr-litlen, litlen, optStatePtr, optLevel);
|
+ ZSTD_litLengthPrice(litlen, optStatePtr, optLevel)
|
||||||
} else {
|
- ZSTD_litLengthPrice(litlen-1, optStatePtr, optLevel);
|
||||||
price = ZSTD_literalsContribution_cached(&cachedLitPrice, anchor, litlen, optStatePtr, optLevel);
|
|
||||||
}
|
|
||||||
assert(price < 1000000000); /* overflow check */
|
assert(price < 1000000000); /* overflow check */
|
||||||
if (price <= opt[cur].price) {
|
if (price <= opt[cur].price) {
|
||||||
DEBUGLOG(7, "rPos:%u : better price (%.2f<=%.2f) using literal",
|
DEBUGLOG(7, "cPos:%zi==rPos:%u : better price (%.2f<=%.2f) using literal (ll==%u) (hist:%u,%u,%u)",
|
||||||
cur, ZSTD_fCost(price), ZSTD_fCost(opt[cur].price));
|
inr-istart, cur, ZSTD_fCost(price), ZSTD_fCost(opt[cur].price), litlen,
|
||||||
opt[cur].mlen = 1;
|
opt[cur-1].rep[0], opt[cur-1].rep[1], opt[cur-1].rep[2]);
|
||||||
|
opt[cur].mlen = 0;
|
||||||
opt[cur].off = 0;
|
opt[cur].off = 0;
|
||||||
opt[cur].litlen = litlen;
|
opt[cur].litlen = litlen;
|
||||||
opt[cur].price = price;
|
opt[cur].price = price;
|
||||||
memcpy(opt[cur].rep, opt[cur-1].rep, sizeof(opt[cur].rep));
|
memcpy(opt[cur].rep, opt[cur-1].rep, sizeof(opt[cur].rep));
|
||||||
} else {
|
} else {
|
||||||
DEBUGLOG(7, "rPos:%u : literal would cost more (%.2f>%.2f)",
|
DEBUGLOG(7, "cPos:%zi==rPos:%u : literal would cost more (%.2f>%.2f) (hist:%u,%u,%u)",
|
||||||
cur, ZSTD_fCost(price), ZSTD_fCost(opt[cur].price));
|
inr-istart, cur, ZSTD_fCost(price), ZSTD_fCost(opt[cur].price),
|
||||||
|
opt[cur].rep[0], opt[cur].rep[1], opt[cur].rep[2]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -892,10 +835,10 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
continue; /* skip unpromising positions; about ~+6% speed, -0.01 ratio */
|
continue; /* skip unpromising positions; about ~+6% speed, -0.01 ratio */
|
||||||
}
|
}
|
||||||
|
|
||||||
{ U32 const ll0 = (opt[cur].mlen != 1);
|
{ U32 const ll0 = (opt[cur].mlen != 0);
|
||||||
U32 const litlen = (opt[cur].mlen == 1) ? opt[cur].litlen : 0;
|
U32 const litlen = (opt[cur].mlen == 0) ? opt[cur].litlen : 0;
|
||||||
U32 const previousPrice = (cur > litlen) ? opt[cur-litlen].price : 0;
|
U32 const previousPrice = opt[cur].price;
|
||||||
U32 const basePrice = previousPrice + ZSTD_fullLiteralsCost(inr-litlen, litlen, optStatePtr, optLevel);
|
U32 const basePrice = previousPrice + ZSTD_litLengthPrice(0, optStatePtr, optLevel);
|
||||||
U32 const nbMatches = ZSTD_BtGetAllMatches(ms, cParams, inr, iend, extDict, opt[cur].rep, ll0, matches, minMatch);
|
U32 const nbMatches = ZSTD_BtGetAllMatches(ms, cParams, inr, iend, extDict, opt[cur].rep, ll0, matches, minMatch);
|
||||||
U32 matchNb;
|
U32 matchNb;
|
||||||
if (!nbMatches) {
|
if (!nbMatches) {
|
||||||
@ -904,14 +847,17 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
}
|
}
|
||||||
|
|
||||||
{ U32 const maxML = matches[nbMatches-1].len;
|
{ U32 const maxML = matches[nbMatches-1].len;
|
||||||
DEBUGLOG(7, "rPos:%u, found %u matches, of maxLength=%u",
|
DEBUGLOG(7, "cPos:%zi==rPos:%u, found %u matches, of maxLength=%u",
|
||||||
cur, nbMatches, maxML);
|
inr-istart, cur, nbMatches, maxML);
|
||||||
|
|
||||||
if ( (maxML > sufficient_len)
|
if ( (maxML > sufficient_len)
|
||||||
|| (cur + maxML >= ZSTD_OPT_NUM) ) {
|
|| (cur + maxML >= ZSTD_OPT_NUM) ) {
|
||||||
best_mlen = maxML;
|
lastSequence.mlen = maxML;
|
||||||
best_off = matches[nbMatches-1].off;
|
lastSequence.off = matches[nbMatches-1].off;
|
||||||
last_pos = cur + 1;
|
lastSequence.litlen = litlen;
|
||||||
|
cur -= (opt[cur].mlen==0) ? opt[cur].litlen : 0; /* last sequence is actually only literals, fix cur to last match - note : may underflow, in which case, it's first sequence, and it's okay */
|
||||||
|
last_pos = cur + ZSTD_totalLen(lastSequence);
|
||||||
|
if (cur > ZSTD_OPT_NUM) cur = 0; /* underflow => first match */
|
||||||
goto _shortestPath;
|
goto _shortestPath;
|
||||||
} }
|
} }
|
||||||
|
|
||||||
@ -923,7 +869,7 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
U32 const startML = (matchNb>0) ? matches[matchNb-1].len+1 : minMatch;
|
U32 const startML = (matchNb>0) ? matches[matchNb-1].len+1 : minMatch;
|
||||||
U32 mlen;
|
U32 mlen;
|
||||||
|
|
||||||
DEBUGLOG(7, "testing match %u => offCode=%u, mlen=%u, llen=%u",
|
DEBUGLOG(7, "testing match %u => offCode=%4u, mlen=%2u, llen=%2u",
|
||||||
matchNb, matches[matchNb].off, lastML, litlen);
|
matchNb, matches[matchNb].off, lastML, litlen);
|
||||||
|
|
||||||
for (mlen = lastML; mlen >= startML; mlen--) { /* scan downward */
|
for (mlen = lastML; mlen >= startML; mlen--) { /* scan downward */
|
||||||
@ -931,8 +877,8 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
int const price = basePrice + ZSTD_getMatchPrice(offset, mlen, optStatePtr, optLevel);
|
int const price = basePrice + ZSTD_getMatchPrice(offset, mlen, optStatePtr, optLevel);
|
||||||
|
|
||||||
if ((pos > last_pos) || (price < opt[pos].price)) {
|
if ((pos > last_pos) || (price < opt[pos].price)) {
|
||||||
DEBUGLOG(7, "rPos:%u => new better price (%.2f<%.2f)",
|
DEBUGLOG(7, "rPos:%u (ml=%2u) => new better price (%.2f<%.2f)",
|
||||||
pos, ZSTD_fCost(price), ZSTD_fCost(opt[pos].price));
|
pos, mlen, ZSTD_fCost(price), ZSTD_fCost(opt[pos].price));
|
||||||
while (last_pos < pos) { opt[last_pos+1].price = ZSTD_MAX_PRICE; last_pos++; } /* fill empty positions */
|
while (last_pos < pos) { opt[last_pos+1].price = ZSTD_MAX_PRICE; last_pos++; } /* fill empty positions */
|
||||||
opt[pos].mlen = mlen;
|
opt[pos].mlen = mlen;
|
||||||
opt[pos].off = offset;
|
opt[pos].off = offset;
|
||||||
@ -941,51 +887,63 @@ ZSTD_compressBlock_opt_generic(ZSTD_matchState_t* ms,
|
|||||||
ZSTD_STATIC_ASSERT(sizeof(opt[pos].rep) == sizeof(repHistory));
|
ZSTD_STATIC_ASSERT(sizeof(opt[pos].rep) == sizeof(repHistory));
|
||||||
memcpy(opt[pos].rep, &repHistory, sizeof(repHistory));
|
memcpy(opt[pos].rep, &repHistory, sizeof(repHistory));
|
||||||
} else {
|
} else {
|
||||||
|
DEBUGLOG(7, "rPos:%u (ml=%2u) => new price is worse (%.2f>=%.2f)",
|
||||||
|
pos, mlen, ZSTD_fCost(price), ZSTD_fCost(opt[pos].price));
|
||||||
if (optLevel==0) break; /* early update abort; gets ~+10% speed for about -0.01 ratio loss */
|
if (optLevel==0) break; /* early update abort; gets ~+10% speed for about -0.01 ratio loss */
|
||||||
}
|
}
|
||||||
} } }
|
} } }
|
||||||
} /* for (cur = 1; cur <= last_pos; cur++) */
|
} /* for (cur = 1; cur <= last_pos; cur++) */
|
||||||
|
|
||||||
best_mlen = opt[last_pos].mlen;
|
lastSequence = opt[last_pos];
|
||||||
best_off = opt[last_pos].off;
|
cur = last_pos > ZSTD_totalLen(lastSequence) ? last_pos - ZSTD_totalLen(lastSequence) : 0; /* single sequence, and it starts before `ip` */
|
||||||
cur = last_pos - best_mlen;
|
assert(cur < ZSTD_OPT_NUM); /* control overflow*/
|
||||||
|
|
||||||
_shortestPath: /* cur, last_pos, best_mlen, best_off have to be set */
|
_shortestPath: /* cur, last_pos, best_mlen, best_off have to be set */
|
||||||
assert(opt[0].mlen == 1);
|
assert(opt[0].mlen == 0);
|
||||||
|
|
||||||
/* reverse traversal */
|
{ U32 const storeEnd = cur + 1;
|
||||||
DEBUGLOG(7, "start reverse traversal (last_pos:%u, cur:%u)",
|
U32 storeStart = storeEnd;
|
||||||
|
U32 seqPos = cur;
|
||||||
|
|
||||||
|
DEBUGLOG(6, "start reverse traversal (last_pos:%u, cur:%u)",
|
||||||
last_pos, cur);
|
last_pos, cur);
|
||||||
{ U32 selectedMatchLength = best_mlen;
|
assert(storeEnd < ZSTD_OPT_NUM);
|
||||||
U32 selectedOffset = best_off;
|
DEBUGLOG(6, "last sequence copied into pos=%u (llen=%u,mlen=%u,ofc=%u)",
|
||||||
U32 pos = cur;
|
storeEnd, lastSequence.litlen, lastSequence.mlen, lastSequence.off);
|
||||||
while (1) {
|
opt[storeEnd] = lastSequence;
|
||||||
U32 const mlen = opt[pos].mlen;
|
while (seqPos > 0) {
|
||||||
U32 const off = opt[pos].off;
|
U32 const backDist = ZSTD_totalLen(opt[seqPos]);
|
||||||
opt[pos].mlen = selectedMatchLength;
|
storeStart--;
|
||||||
opt[pos].off = selectedOffset;
|
DEBUGLOG(6, "sequence from rPos=%u copied into pos=%u (llen=%u,mlen=%u,ofc=%u)",
|
||||||
selectedMatchLength = mlen;
|
seqPos, storeStart, opt[seqPos].litlen, opt[seqPos].mlen, opt[seqPos].off);
|
||||||
selectedOffset = off;
|
opt[storeStart] = opt[seqPos];
|
||||||
if (mlen > pos) break;
|
seqPos = (seqPos > backDist) ? seqPos - backDist : 0;
|
||||||
pos -= mlen;
|
}
|
||||||
} }
|
|
||||||
|
|
||||||
/* save sequences */
|
/* save sequences */
|
||||||
{ U32 pos;
|
DEBUGLOG(6, "sending selected sequences into seqStore")
|
||||||
for (pos=0; pos < last_pos; ) {
|
{ U32 storePos;
|
||||||
U32 const llen = (U32)(ip - anchor);
|
for (storePos=storeStart; storePos <= storeEnd; storePos++) {
|
||||||
U32 const mlen = opt[pos].mlen;
|
U32 const llen = opt[storePos].litlen;
|
||||||
U32 const offset = opt[pos].off;
|
U32 const mlen = opt[storePos].mlen;
|
||||||
if (mlen == 1) { ip++; pos++; continue; } /* literal position => move on */
|
U32 const offCode = opt[storePos].off;
|
||||||
pos += mlen; ip += mlen;
|
U32 const advance = llen + mlen;
|
||||||
|
DEBUGLOG(6, "considering seq starting at %zi, llen=%u, mlen=%u",
|
||||||
|
anchor - istart, llen, mlen);
|
||||||
|
|
||||||
|
if (mlen==0) { /* only literals => must be last "sequence", actually starting a new stream of sequences */
|
||||||
|
assert(storePos == storeEnd); /* must be last sequence */
|
||||||
|
ip = anchor + llen; /* last "sequence" is a bunch of literals => don't progress anchor */
|
||||||
|
continue; /* will finish */
|
||||||
|
}
|
||||||
|
|
||||||
/* repcodes update : like ZSTD_updateRep(), but update in place */
|
/* repcodes update : like ZSTD_updateRep(), but update in place */
|
||||||
if (offset >= ZSTD_REP_NUM) { /* full offset */
|
if (offCode >= ZSTD_REP_NUM) { /* full offset */
|
||||||
rep[2] = rep[1];
|
rep[2] = rep[1];
|
||||||
rep[1] = rep[0];
|
rep[1] = rep[0];
|
||||||
rep[0] = offset - ZSTD_REP_MOVE;
|
rep[0] = offCode - ZSTD_REP_MOVE;
|
||||||
} else { /* repcode */
|
} else { /* repcode */
|
||||||
U32 const repCode = offset + (llen==0);
|
U32 const repCode = offCode + (llen==0);
|
||||||
if (repCode) { /* note : if repCode==0, no change */
|
if (repCode) { /* note : if repCode==0, no change */
|
||||||
U32 const currentOffset = (repCode==ZSTD_REP_NUM) ? (rep[0] - 1) : rep[repCode];
|
U32 const currentOffset = (repCode==ZSTD_REP_NUM) ? (rep[0] - 1) : rep[repCode];
|
||||||
if (repCode >= 2) rep[2] = rep[1];
|
if (repCode >= 2) rep[2] = rep[1];
|
||||||
@ -993,11 +951,15 @@ _shortestPath: /* cur, last_pos, best_mlen, best_off have to be set */
|
|||||||
rep[0] = currentOffset;
|
rep[0] = currentOffset;
|
||||||
} }
|
} }
|
||||||
|
|
||||||
ZSTD_updateStats(optStatePtr, llen, anchor, offset, mlen);
|
assert(anchor + llen <= iend);
|
||||||
ZSTD_storeSeq(seqStore, llen, anchor, offset, mlen-MINMATCH);
|
ZSTD_updateStats(optStatePtr, llen, anchor, offCode, mlen);
|
||||||
anchor = ip;
|
ZSTD_storeSeq(seqStore, llen, anchor, offCode, mlen-MINMATCH);
|
||||||
|
anchor += advance;
|
||||||
|
ip = anchor;
|
||||||
} }
|
} }
|
||||||
ZSTD_setBasePrices(optStatePtr, optLevel);
|
ZSTD_setBasePrices(optStatePtr, optLevel);
|
||||||
|
}
|
||||||
|
|
||||||
} /* while (ip < ilimit) */
|
} /* while (ip < ilimit) */
|
||||||
|
|
||||||
/* Return the last literals size */
|
/* Return the last literals size */
|
||||||
|
@ -9,13 +9,18 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
/*- Dependencies -*/
|
/******************************************
|
||||||
|
* Includes
|
||||||
|
******************************************/
|
||||||
|
#include <stddef.h> /* size_t, ptrdiff_t */
|
||||||
|
#include <string.h> /* memcpy */
|
||||||
|
|
||||||
#include "zstd_v04.h"
|
#include "zstd_v04.h"
|
||||||
#include "error_private.h"
|
#include "error_private.h"
|
||||||
|
|
||||||
|
|
||||||
/* ******************************************************************
|
/* ******************************************************************
|
||||||
mem.h
|
* mem.h
|
||||||
*******************************************************************/
|
*******************************************************************/
|
||||||
#ifndef MEM_H_MODULE
|
#ifndef MEM_H_MODULE
|
||||||
#define MEM_H_MODULE
|
#define MEM_H_MODULE
|
||||||
@ -24,12 +29,6 @@
|
|||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/******************************************
|
|
||||||
* Includes
|
|
||||||
******************************************/
|
|
||||||
#include <stddef.h> /* size_t, ptrdiff_t */
|
|
||||||
#include <string.h> /* memcpy */
|
|
||||||
|
|
||||||
|
|
||||||
/******************************************
|
/******************************************
|
||||||
* Compiler-specific
|
* Compiler-specific
|
||||||
@ -87,7 +86,7 @@ extern "C" {
|
|||||||
|
|
||||||
#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG>=2)
|
#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG>=2)
|
||||||
# include <stdio.h>
|
# include <stdio.h>
|
||||||
extern int g_debuglog_enable;
|
extern int g_debuglevel;
|
||||||
/* recommended values for ZSTD_DEBUG display levels :
|
/* recommended values for ZSTD_DEBUG display levels :
|
||||||
* 1 : no display, enables assert() only
|
* 1 : no display, enables assert() only
|
||||||
* 2 : reserved for currently active debug path
|
* 2 : reserved for currently active debug path
|
||||||
@ -96,11 +95,11 @@ extern int g_debuglog_enable;
|
|||||||
* 5 : events once per block
|
* 5 : events once per block
|
||||||
* 6 : events once per sequence (*very* verbose) */
|
* 6 : events once per sequence (*very* verbose) */
|
||||||
# define RAWLOG(l, ...) { \
|
# define RAWLOG(l, ...) { \
|
||||||
if ((g_debuglog_enable) & (l<=ZSTD_DEBUG)) { \
|
if (l<=g_debuglevel) { \
|
||||||
fprintf(stderr, __VA_ARGS__); \
|
fprintf(stderr, __VA_ARGS__); \
|
||||||
} }
|
} }
|
||||||
# define DEBUGLOG(l, ...) { \
|
# define DEBUGLOG(l, ...) { \
|
||||||
if ((g_debuglog_enable) & (l<=ZSTD_DEBUG)) { \
|
if (l<=g_debuglevel) { \
|
||||||
fprintf(stderr, __FILE__ ": " __VA_ARGS__); \
|
fprintf(stderr, __FILE__ ": " __VA_ARGS__); \
|
||||||
fprintf(stderr, " \n"); \
|
fprintf(stderr, " \n"); \
|
||||||
} }
|
} }
|
||||||
@ -266,14 +265,6 @@ MEM_STATIC size_t MEM_readLEST(const void* memPtr)
|
|||||||
#ifndef ZSTD_STATIC_H
|
#ifndef ZSTD_STATIC_H
|
||||||
#define ZSTD_STATIC_H
|
#define ZSTD_STATIC_H
|
||||||
|
|
||||||
/* The objects defined into this file shall be considered experimental.
|
|
||||||
* They are not considered stable, as their prototype may change in the future.
|
|
||||||
* You can use them for tests, provide feedback, or if you can endure risks of future changes.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#if defined (__cplusplus)
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* *************************************
|
/* *************************************
|
||||||
* Types
|
* Types
|
||||||
@ -360,9 +351,6 @@ static size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t maxDstS
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#if defined (__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
#endif /* ZSTD_STATIC_H */
|
#endif /* ZSTD_STATIC_H */
|
||||||
@ -375,10 +363,6 @@ static size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t maxDstS
|
|||||||
#ifndef ZSTD_CCOMMON_H_MODULE
|
#ifndef ZSTD_CCOMMON_H_MODULE
|
||||||
#define ZSTD_CCOMMON_H_MODULE
|
#define ZSTD_CCOMMON_H_MODULE
|
||||||
|
|
||||||
#if defined (__cplusplus)
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* *************************************
|
/* *************************************
|
||||||
* Common macros
|
* Common macros
|
||||||
***************************************/
|
***************************************/
|
||||||
@ -450,10 +434,6 @@ static void ZSTD_wildcopy(void* dst, const void* src, ptrdiff_t length)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#if defined (__cplusplus)
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
/* ******************************************************************
|
/* ******************************************************************
|
||||||
FSE : Finite State Entropy coder
|
FSE : Finite State Entropy coder
|
||||||
@ -2991,7 +2971,7 @@ static size_t ZSTD_execSequence(BYTE* op,
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */
|
ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8, but must be signed */
|
||||||
}
|
}
|
||||||
return sequenceLength;
|
return sequenceLength;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user