Merge "Improve speed and compression in backward reference for lossless."

This commit is contained in:
Pascal Massimino 2016-06-08 19:54:51 +00:00 committed by Gerrit Code Review
commit 043c33f1ce
3 changed files with 162 additions and 140 deletions

View File

@ -27,11 +27,19 @@
#define MAX_ENTROPY (1e30f) #define MAX_ENTROPY (1e30f)
// 1M window (4M bytes) minus 120 special codes for short distances. // 1M window (4M bytes) minus 120 special codes for short distances.
#define WINDOW_SIZE ((1 << 20) - 120) #define WINDOW_SIZE_BITS 20
#define WINDOW_SIZE ((1 << WINDOW_SIZE_BITS) - 120)
// Bounds for the match length. // Bounds for the match length.
#define MIN_LENGTH 2 #define MIN_LENGTH 2
#define MAX_LENGTH 4096 // If you change this, you need MAX_LENGTH_BITS + WINDOW_SIZE_BITS <= 32 as it
// is used in VP8LHashChain.
#define MAX_LENGTH_BITS 12
// We want the max value to be attainable and stored in MAX_LENGTH_BITS bits.
#define MAX_LENGTH ((1 << MAX_LENGTH_BITS) - 1)
#if MAX_LENGTH_BITS + WINDOW_SIZE_BITS > 32
#error "MAX_LENGTH_BITS + WINDOW_SIZE_BITS > 32"
#endif
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
@ -183,10 +191,11 @@ int VP8LBackwardRefsCopy(const VP8LBackwardRefs* const src,
int VP8LHashChainInit(VP8LHashChain* const p, int size) { int VP8LHashChainInit(VP8LHashChain* const p, int size) {
assert(p->size_ == 0); assert(p->size_ == 0);
assert(p->chain_ == NULL); assert(p->offset_length_ == NULL);
assert(size > 0); assert(size > 0);
p->chain_ = (int32_t*)WebPSafeMalloc(size, sizeof(*p->chain_)); p->offset_length_ =
if (p->chain_ == NULL) return 0; (uint32_t*)WebPSafeMalloc(size, sizeof(*p->offset_length_));
if (p->offset_length_ == NULL) return 0;
p->size_ = size; p->size_ = size;
return 1; return 1;
@ -194,9 +203,10 @@ int VP8LHashChainInit(VP8LHashChain* const p, int size) {
void VP8LHashChainClear(VP8LHashChain* const p) { void VP8LHashChainClear(VP8LHashChain* const p) {
assert(p != NULL); assert(p != NULL);
WebPSafeFree(p->chain_); WebPSafeFree(p->offset_length_);
p->size_ = 0; p->size_ = 0;
p->chain_ = NULL; p->offset_length_ = NULL;
} }
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
@ -213,9 +223,9 @@ static WEBP_INLINE uint32_t GetPixPairHash64(const uint32_t* const argb) {
} }
// Returns the maximum number of hash chain lookups to do for a // Returns the maximum number of hash chain lookups to do for a
// given compression quality. Return value in range [6, 86]. // given compression quality. Return value in range [8, 86].
static int GetMaxItersForQuality(int quality, int low_effort) { static int GetMaxItersForQuality(int quality) {
return (low_effort ? 6 : 8) + (quality * quality) / 128; return 8 + (quality * quality) / 128;
} }
static int GetWindowSizeForHashChain(int quality, int xsize) { static int GetWindowSizeForHashChain(int quality, int xsize) {
@ -231,13 +241,18 @@ static WEBP_INLINE int MaxFindCopyLength(int len) {
return (len < MAX_LENGTH) ? len : MAX_LENGTH; return (len < MAX_LENGTH) ? len : MAX_LENGTH;
} }
int VP8LHashChainFill(VP8LHashChain* const p, int VP8LHashChainFill(VP8LHashChain* const p, int quality,
const uint32_t* const argb, int xsize, int ysize) { const uint32_t* const argb, int xsize, int ysize) {
const int size = xsize * ysize; const int size = xsize * ysize;
const int iter_max = GetMaxItersForQuality(quality);
const uint32_t window_size = GetWindowSizeForHashChain(quality, xsize);
int pos; int pos;
uint32_t base_position;
int32_t* hash_to_first_index; int32_t* hash_to_first_index;
// Temporarily use the p->offset_length_ as a hash chain.
int32_t* chain = (int32_t*)p->offset_length_;
assert(p->size_ != 0); assert(p->size_ != 0);
assert(p->chain_ != NULL); assert(p->offset_length_ != NULL);
hash_to_first_index = hash_to_first_index =
(int32_t*)WebPSafeMalloc(HASH_SIZE, sizeof(*hash_to_first_index)); (int32_t*)WebPSafeMalloc(HASH_SIZE, sizeof(*hash_to_first_index));
@ -248,71 +263,93 @@ int VP8LHashChainFill(VP8LHashChain* const p,
// Fill the chain linking pixels with the same hash. // Fill the chain linking pixels with the same hash.
for (pos = 0; pos < size - 1; ++pos) { for (pos = 0; pos < size - 1; ++pos) {
const uint32_t hash_code = GetPixPairHash64(argb + pos); const uint32_t hash_code = GetPixPairHash64(argb + pos);
p->chain_[pos] = hash_to_first_index[hash_code]; chain[pos] = hash_to_first_index[hash_code];
hash_to_first_index[hash_code] = pos; hash_to_first_index[hash_code] = pos;
} }
WebPSafeFree(hash_to_first_index); WebPSafeFree(hash_to_first_index);
return 1; // Find the best match interval at each pixel, defined by an offset to the
} // pixel and a length. The right-most pixel cannot match anything to the right
// (hence a best length of 0) and the left-most pixel nothing to the left
// (hence an offset of 0).
p->offset_length_[0] = p->offset_length_[size - 1] = 0;
for (base_position = size - 2 < 0 ? 0 : size - 2; base_position > 0;) {
const int max_len = MaxFindCopyLength(size - 1 - base_position);
const uint32_t* const argb_start = argb + base_position;
int iter = iter_max;
int best_length = 0;
uint32_t best_distance = 0;
const int min_pos =
(base_position > window_size) ? base_position - window_size : 0;
const int length_max = (max_len < 256) ? max_len : 256;
uint32_t max_base_position;
static void HashChainFindOffset(const VP8LHashChain* const p, int base_position, for (pos = chain[base_position]; pos >= min_pos; pos = chain[pos]) {
const uint32_t* const argb, int len, int curr_length;
int window_size, int* const distance_ptr) { if (--iter < 0) {
const uint32_t* const argb_start = argb + base_position;
const int min_pos =
(base_position > window_size) ? base_position - window_size : 0;
int pos;
assert(len <= MAX_LENGTH);
for (pos = p->chain_[base_position];
pos >= min_pos;
pos = p->chain_[pos]) {
const int curr_length =
FindMatchLength(argb + pos, argb_start, len - 1, len);
if (curr_length == len) break;
}
*distance_ptr = base_position - pos;
}
static int HashChainFindCopy(const VP8LHashChain* const p,
int base_position,
const uint32_t* const argb, int max_len,
int window_size, int iter_max,
int* const distance_ptr,
int* const length_ptr) {
const uint32_t* const argb_start = argb + base_position;
int iter = iter_max;
int best_length = 0;
int best_distance = 0;
const int min_pos =
(base_position > window_size) ? base_position - window_size : 0;
int pos;
int length_max = 256;
if (max_len < length_max) {
length_max = max_len;
}
for (pos = p->chain_[base_position];
pos >= min_pos;
pos = p->chain_[pos]) {
int curr_length;
int distance;
if (--iter < 0) {
break;
}
curr_length = FindMatchLength(argb + pos, argb_start, best_length, max_len);
if (best_length < curr_length) {
distance = base_position - pos;
best_length = curr_length;
best_distance = distance;
if (curr_length >= length_max) {
break; break;
} }
assert(base_position > (uint32_t)pos);
curr_length =
FindMatchLength(argb + pos, argb_start, best_length, max_len);
if (best_length < curr_length) {
best_length = curr_length;
best_distance = base_position - pos;
if (curr_length >= length_max) {
break;
}
}
}
// We have the best match but in case the two intervals continue matching
// to the left, we have the best matches for the left-extended pixels.
max_base_position = base_position;
while (1) {
assert(best_length <= MAX_LENGTH);
assert(best_distance <= WINDOW_SIZE);
p->offset_length_[base_position] =
(best_distance << MAX_LENGTH_BITS) | (uint32_t)best_length;
--base_position;
// Stop if we don't have a match or if we are out of bounds.
if (best_distance == 0 || base_position == 0) break;
// Stop if we cannot extend the matching intervals to the left.
if (base_position < best_distance ||
argb[base_position - best_distance] != argb[base_position]) {
break;
}
// Stop if we are matching at its limit because there could be a closer
// matching interval with the same maximum length. Then again, if the
// matching interval is as close as possible (best_distance == 1), we will
// never find anything better so let's continue.
if (best_length == MAX_LENGTH && best_distance != 1 &&
base_position + MAX_LENGTH < max_base_position) {
break;
}
if (best_length < MAX_LENGTH) {
++best_length;
max_base_position = base_position;
}
} }
} }
*distance_ptr = best_distance; return 1;
*length_ptr = best_length; }
return (best_length >= MIN_LENGTH);
static WEBP_INLINE int HashChainFindOffset(const VP8LHashChain* const p,
const int base_position) {
return p->offset_length_[base_position] >> MAX_LENGTH_BITS;
}
static WEBP_INLINE int HashChainFindLength(const VP8LHashChain* const p,
const int base_position) {
return p->offset_length_[base_position] & ((1U << MAX_LENGTH_BITS) - 1);
}
static WEBP_INLINE void HashChainFindCopy(const VP8LHashChain* const p,
int base_position,
int* const offset_ptr,
int* const length_ptr) {
*offset_ptr = HashChainFindOffset(p, base_position);
*length_ptr = HashChainFindLength(p, base_position);
} }
static WEBP_INLINE void AddSingleLiteral(uint32_t pixel, int use_color_cache, static WEBP_INLINE void AddSingleLiteral(uint32_t pixel, int use_color_cache,
@ -379,70 +416,63 @@ static int BackwardReferencesRle(int xsize, int ysize,
static int BackwardReferencesLz77(int xsize, int ysize, static int BackwardReferencesLz77(int xsize, int ysize,
const uint32_t* const argb, int cache_bits, const uint32_t* const argb, int cache_bits,
int quality, int low_effort,
const VP8LHashChain* const hash_chain, const VP8LHashChain* const hash_chain,
VP8LBackwardRefs* const refs) { VP8LBackwardRefs* const refs) {
int i; int i;
int i_last_check = -1;
int ok = 0; int ok = 0;
int cc_init = 0; int cc_init = 0;
const int use_color_cache = (cache_bits > 0); const int use_color_cache = (cache_bits > 0);
const int pix_count = xsize * ysize; const int pix_count = xsize * ysize;
VP8LColorCache hashers; VP8LColorCache hashers;
int iter_max = GetMaxItersForQuality(quality, low_effort);
const int window_size = GetWindowSizeForHashChain(quality, xsize);
int min_matches = 32;
if (use_color_cache) { if (use_color_cache) {
cc_init = VP8LColorCacheInit(&hashers, cache_bits); cc_init = VP8LColorCacheInit(&hashers, cache_bits);
if (!cc_init) goto Error; if (!cc_init) goto Error;
} }
ClearBackwardRefs(refs); ClearBackwardRefs(refs);
for (i = 0; i < pix_count - 2; ) { for (i = 0; i < pix_count;) {
// Alternative#1: Code the pixels starting at 'i' using backward reference. // Alternative#1: Code the pixels starting at 'i' using backward reference.
int offset = 0; int offset = 0;
int len = 0; int len = 0;
const int max_len = MaxFindCopyLength(pix_count - i); int j;
HashChainFindCopy(hash_chain, i, argb, max_len, window_size, HashChainFindCopy(hash_chain, i, &offset, &len);
iter_max, &offset, &len); // MIN_LENGTH+1 is empirically better than MIN_LENGTH.
if (len > MIN_LENGTH || (len == MIN_LENGTH && offset <= 512)) { if (len > MIN_LENGTH + 1) {
int offset2 = 0; const int len_ini = len;
int len2 = 0; int max_reach = 0;
int k; assert(i + len < pix_count);
min_matches = 8; // Only start from what we have not checked already.
if ((len < (max_len >> 2)) && !low_effort) { i_last_check = (i > i_last_check) ? i : i_last_check;
// Evaluate Alternative#2: Insert the pixel at 'i' as literal, and code // We know the best match for the current pixel but we try to find the
// the pixels starting at 'i + 1' using backward reference. // best matches for the current pixel AND the next one combined.
HashChainFindCopy(hash_chain, i + 1, argb, max_len - 1, // The naive method would use the intervals:
window_size, iter_max, &offset2, // [i,i+len) + [i+len, length of best match at i+len)
&len2); // while we check if we can use:
if (len2 > len + 1) { // [i,j) (where j<=i+len) + [j, length of best match at j)
AddSingleLiteral(argb[i], use_color_cache, &hashers, refs); for (j = i_last_check + 1; j <= i + len_ini; ++j) {
i++; // Backward reference to be done for next pixel. const int len_j = HashChainFindLength(hash_chain, j);
len = len2; const int reach =
offset = offset2; j + (len_j > MIN_LENGTH + 1 ? len_j : 1); // 1 for single literal.
if (reach > max_reach) {
len = j - i;
max_reach = reach;
} }
} }
} else {
len = 1;
}
// Go with literal or backward reference.
assert(len > 0);
if (len == 1) {
AddSingleLiteral(argb[i], use_color_cache, &hashers, refs);
} else {
BackwardRefsCursorAdd(refs, PixOrCopyCreateCopy(offset, len)); BackwardRefsCursorAdd(refs, PixOrCopyCreateCopy(offset, len));
if (use_color_cache) { if (use_color_cache) {
for (k = 0; k < len; ++k) { for (j = i; j < i + len; ++j) VP8LColorCacheInsert(&hashers, argb[j]);
VP8LColorCacheInsert(&hashers, argb[i + k]);
}
}
i += len;
} else {
AddSingleLiteral(argb[i], use_color_cache, &hashers, refs);
++i;
--min_matches;
if (min_matches <= 0) {
AddSingleLiteral(argb[i], use_color_cache, &hashers, refs);
++i;
} }
} }
} i += len;
while (i < pix_count) {
// Handle the last pixel(s).
AddSingleLiteral(argb[i], use_color_cache, &hashers, refs);
++i;
} }
ok = !refs->error_; ok = !refs->error_;
@ -1045,8 +1075,6 @@ static int BackwardReferencesHashChainDistanceOnly(
VP8LColorCache hashers; VP8LColorCache hashers;
const int skip_length = 32 + quality; const int skip_length = 32 + quality;
const int skip_min_distance_code = 2; const int skip_min_distance_code = 2;
int iter_max = GetMaxItersForQuality(quality, 0);
const int window_size = GetWindowSizeForHashChain(quality, xsize);
CostManager* cost_manager = CostManager* cost_manager =
(CostManager*)WebPSafeMalloc(1ULL, sizeof(*cost_manager)); (CostManager*)WebPSafeMalloc(1ULL, sizeof(*cost_manager));
@ -1076,9 +1104,7 @@ static int BackwardReferencesHashChainDistanceOnly(
int offset = 0; int offset = 0;
int len = 0; int len = 0;
double prev_cost = cost_manager->costs_[i - 1]; double prev_cost = cost_manager->costs_[i - 1];
const int max_len = MaxFindCopyLength(pix_count - i); HashChainFindCopy(hash_chain, i, &offset, &len);
HashChainFindCopy(hash_chain, i, argb, max_len, window_size,
iter_max, &offset, &len);
if (len >= MIN_LENGTH) { if (len >= MIN_LENGTH) {
const int code = DistanceToPlaneCode(xsize, offset); const int code = DistanceToPlaneCode(xsize, offset);
const double distance_cost = const double distance_cost =
@ -1110,8 +1136,7 @@ static int BackwardReferencesHashChainDistanceOnly(
if (len != MIN_LENGTH) { if (len != MIN_LENGTH) {
int code_min_length; int code_min_length;
double cost_total; double cost_total;
HashChainFindOffset(hash_chain, i, argb, MIN_LENGTH, window_size, offset = HashChainFindOffset(hash_chain, i);
&offset);
code_min_length = DistanceToPlaneCode(xsize, offset); code_min_length = DistanceToPlaneCode(xsize, offset);
cost_total = prev_cost + cost_total = prev_cost +
GetDistanceCost(cost_model, code_min_length) + GetDistanceCost(cost_model, code_min_length) +
@ -1167,17 +1192,14 @@ static void TraceBackwards(uint16_t* const dist_array,
} }
static int BackwardReferencesHashChainFollowChosenPath( static int BackwardReferencesHashChainFollowChosenPath(
int xsize, const uint32_t* const argb, const uint32_t* const argb, int cache_bits,
int quality, int cache_bits,
const uint16_t* const chosen_path, int chosen_path_size, const uint16_t* const chosen_path, int chosen_path_size,
const VP8LHashChain* const hash_chain, const VP8LHashChain* const hash_chain, VP8LBackwardRefs* const refs) {
VP8LBackwardRefs* const refs) {
const int use_color_cache = (cache_bits > 0); const int use_color_cache = (cache_bits > 0);
int ix; int ix;
int i = 0; int i = 0;
int ok = 0; int ok = 0;
int cc_init = 0; int cc_init = 0;
const int window_size = GetWindowSizeForHashChain(quality, xsize);
VP8LColorCache hashers; VP8LColorCache hashers;
if (use_color_cache) { if (use_color_cache) {
@ -1187,11 +1209,10 @@ static int BackwardReferencesHashChainFollowChosenPath(
ClearBackwardRefs(refs); ClearBackwardRefs(refs);
for (ix = 0; ix < chosen_path_size; ++ix) { for (ix = 0; ix < chosen_path_size; ++ix) {
int offset = 0;
const int len = chosen_path[ix]; const int len = chosen_path[ix];
if (len != 1) { if (len != 1) {
int k; int k;
HashChainFindOffset(hash_chain, i, argb, len, window_size, &offset); const int offset = HashChainFindOffset(hash_chain, i);
BackwardRefsCursorAdd(refs, PixOrCopyCreateCopy(offset, len)); BackwardRefsCursorAdd(refs, PixOrCopyCreateCopy(offset, len));
if (use_color_cache) { if (use_color_cache) {
for (k = 0; k < len; ++k) { for (k = 0; k < len; ++k) {
@ -1240,8 +1261,7 @@ static int BackwardReferencesTraceBackwards(
} }
TraceBackwards(dist_array, dist_array_size, &chosen_path, &chosen_path_size); TraceBackwards(dist_array, dist_array_size, &chosen_path, &chosen_path_size);
if (!BackwardReferencesHashChainFollowChosenPath( if (!BackwardReferencesHashChainFollowChosenPath(
xsize, argb, quality, cache_bits, chosen_path, chosen_path_size, argb, cache_bits, chosen_path, chosen_path_size, hash_chain, refs)) {
hash_chain, refs)) {
goto Error; goto Error;
} }
ok = 1; ok = 1;
@ -1349,8 +1369,8 @@ static int CalculateBestCacheSize(const uint32_t* const argb,
// Local color cache is disabled. // Local color cache is disabled.
return 1; return 1;
} }
if (!BackwardReferencesLz77(xsize, ysize, argb, cache_bits_low, quality, 0, if (!BackwardReferencesLz77(xsize, ysize, argb, cache_bits_low, hash_chain,
hash_chain, refs)) { refs)) {
return 0; return 0;
} }
// Do a binary search to find the optimal entropy for cache_bits. // Do a binary search to find the optimal entropy for cache_bits.
@ -1415,13 +1435,12 @@ static int BackwardRefsWithLocalCache(const uint32_t* const argb,
} }
static VP8LBackwardRefs* GetBackwardReferencesLowEffort( static VP8LBackwardRefs* GetBackwardReferencesLowEffort(
int width, int height, const uint32_t* const argb, int quality, int width, int height, const uint32_t* const argb,
int* const cache_bits, const VP8LHashChain* const hash_chain, int* const cache_bits, const VP8LHashChain* const hash_chain,
VP8LBackwardRefs refs_array[2]) { VP8LBackwardRefs refs_array[2]) {
VP8LBackwardRefs* refs_lz77 = &refs_array[0]; VP8LBackwardRefs* refs_lz77 = &refs_array[0];
*cache_bits = 0; *cache_bits = 0;
if (!BackwardReferencesLz77(width, height, argb, 0, quality, if (!BackwardReferencesLz77(width, height, argb, 0, hash_chain, refs_lz77)) {
1 /* Low effort. */, hash_chain, refs_lz77)) {
return NULL; return NULL;
} }
BackwardReferences2DLocality(width, refs_lz77); BackwardReferences2DLocality(width, refs_lz77);
@ -1453,8 +1472,8 @@ static VP8LBackwardRefs* GetBackwardReferences(
} }
} }
} else { } else {
if (!BackwardReferencesLz77(width, height, argb, *cache_bits, quality, if (!BackwardReferencesLz77(width, height, argb, *cache_bits, hash_chain,
0 /* Low effort. */, hash_chain, refs_lz77)) { refs_lz77)) {
goto Error; goto Error;
} }
} }
@ -1516,8 +1535,8 @@ VP8LBackwardRefs* VP8LGetBackwardReferences(
int low_effort, int* const cache_bits, int low_effort, int* const cache_bits,
const VP8LHashChain* const hash_chain, VP8LBackwardRefs refs_array[2]) { const VP8LHashChain* const hash_chain, VP8LBackwardRefs refs_array[2]) {
if (low_effort) { if (low_effort) {
return GetBackwardReferencesLowEffort(width, height, argb, quality, return GetBackwardReferencesLowEffort(width, height, argb, cache_bits,
cache_bits, hash_chain, refs_array); hash_chain, refs_array);
} else { } else {
return GetBackwardReferences(width, height, argb, quality, cache_bits, return GetBackwardReferences(width, height, argb, quality, cache_bits,
hash_chain, refs_array); hash_chain, refs_array);

View File

@ -115,9 +115,12 @@ static WEBP_INLINE uint32_t PixOrCopyDistance(const PixOrCopy* const p) {
typedef struct VP8LHashChain VP8LHashChain; typedef struct VP8LHashChain VP8LHashChain;
struct VP8LHashChain { struct VP8LHashChain {
// chain_[pos] stores the previous position with the same hash value // The 20 most significant bits contain the offset at which the best match
// for every pixel in the image. // is found. These 20 bits are the limit defined by GetWindowSizeForHashChain
int32_t* chain_; // (through WINDOW_SIZE = 1<<20).
// The lower 12 bits contain the length of the match. The 12 bit limit is
// defined in MaxFindCopyLength with MAX_LENGTH=4096.
uint32_t* offset_length_;
// This is the maximum size of the hash_chain that can be constructed. // This is the maximum size of the hash_chain that can be constructed.
// Typically this is the pixel count (width x height) for a given image. // Typically this is the pixel count (width x height) for a given image.
int size_; int size_;
@ -126,7 +129,7 @@ struct VP8LHashChain {
// Must be called first, to set size. // Must be called first, to set size.
int VP8LHashChainInit(VP8LHashChain* const p, int size); int VP8LHashChainInit(VP8LHashChain* const p, int size);
// Pre-compute the best matches for argb. // Pre-compute the best matches for argb.
int VP8LHashChainFill(VP8LHashChain* const p, int VP8LHashChainFill(VP8LHashChain* const p, int quality,
const uint32_t* const argb, int xsize, int ysize); const uint32_t* const argb, int xsize, int ysize);
void VP8LHashChainClear(VP8LHashChain* const p); // release memory void VP8LHashChainClear(VP8LHashChain* const p); // release memory

View File

@ -761,7 +761,7 @@ static WebPEncodingError EncodeImageNoHuffman(VP8LBitWriter* const bw,
} }
// Calculate backward references from ARGB image. // Calculate backward references from ARGB image.
if (VP8LHashChainFill(hash_chain, argb, width, height) == 0) { if (VP8LHashChainFill(hash_chain, quality, argb, width, height) == 0) {
err = VP8_ENC_ERROR_OUT_OF_MEMORY; err = VP8_ENC_ERROR_OUT_OF_MEMORY;
goto Error; goto Error;
} }
@ -865,7 +865,7 @@ static WebPEncodingError EncodeImageInternal(VP8LBitWriter* const bw,
// 'best_refs' is the reference to the best backward refs and points to one // 'best_refs' is the reference to the best backward refs and points to one
// of refs_array[0] or refs_array[1]. // of refs_array[0] or refs_array[1].
// Calculate backward references from ARGB image. // Calculate backward references from ARGB image.
if (VP8LHashChainFill(hash_chain, argb, width, height) == 0) { if (VP8LHashChainFill(hash_chain, quality, argb, width, height) == 0) {
err = VP8_ENC_ERROR_OUT_OF_MEMORY; err = VP8_ENC_ERROR_OUT_OF_MEMORY;
goto Error; goto Error;
} }