trap two unchecked error conditions
CostModelBuild() and TrackBackwards() returns weren't checked
+ code clean-up
+ de-inline VP8LBackwardRefs non-critical methods
+ shuffle the .h around to group things together
+ extract some constants as #define's
+ fixed the "if (!(cc_init = ...)) {...}" constructs
+ removed some unneeded VP8L prefixes
Change-Id: Ic634cb87bc6b2033242d3e8e8731fab4c134f327
			
			
This commit is contained in:
		| @@ -17,10 +17,30 @@ | |||||||
| #include "./backward_references.h" | #include "./backward_references.h" | ||||||
| #include "./histogram.h" | #include "./histogram.h" | ||||||
| #include "../utils/color_cache.h" | #include "../utils/color_cache.h" | ||||||
| #include "../webp/format_constants.h" |  | ||||||
|  |  | ||||||
| #define VALUES_IN_BYTE 256 | #define VALUES_IN_BYTE 256 | ||||||
|  |  | ||||||
|  | #define HASH_BITS 18 | ||||||
|  | #define HASH_SIZE (1 << HASH_BITS) | ||||||
|  | #define HASH_MULTIPLIER (0xc6a4a7935bd1e995ULL) | ||||||
|  |  | ||||||
|  | // 1M window (4M bytes) minus 120 special codes for short distances. | ||||||
|  | #define WINDOW_SIZE ((1 << 20) - 120) | ||||||
|  |  | ||||||
|  | // Bounds for the match length. | ||||||
|  | #define MIN_LENGTH 2 | ||||||
|  | #define MAX_LENGTH 4096 | ||||||
|  |  | ||||||
|  | typedef struct { | ||||||
|  |   // Stores the most recently added position with the given hash value. | ||||||
|  |   int32_t hash_to_first_index_[HASH_SIZE]; | ||||||
|  |   // chain_[pos] stores the previous position with the same hash value | ||||||
|  |   // for every pixel in the image. | ||||||
|  |   int32_t* chain_; | ||||||
|  | } HashChain; | ||||||
|  |  | ||||||
|  | // ----------------------------------------------------------------------------- | ||||||
|  |  | ||||||
| static const uint8_t plane_to_code_lut[128] = { | static const uint8_t plane_to_code_lut[128] = { | ||||||
|  96,   73,  55,  39,  23,  13,   5,  1,  255, 255, 255, 255, 255, 255, 255, 255, |  96,   73,  55,  39,  23,  13,   5,  1,  255, 255, 255, 255, 255, 255, 255, 255, | ||||||
|  101,  78,  58,  42,  26,  16,   8,  2,    0,   3,  9,   17,  27,  43,  59,  79, |  101,  78,  58,  42,  26,  16,   8,  2,    0,   3,  9,   17,  27,  43,  59,  79, | ||||||
| @@ -29,11 +49,9 @@ static const uint8_t plane_to_code_lut[128] = { | |||||||
|  110,  99,  82,  66,  48,  35,  30,  24,  22,  25,  31,  36,  49,  67,  83, 100, |  110,  99,  82,  66,  48,  35,  30,  24,  22,  25,  31,  36,  49,  67,  83, 100, | ||||||
|  115, 108,  94,  76,  64,  50,  44,  40,  34,  41,  45,  51,  65,  77,  95, 109, |  115, 108,  94,  76,  64,  50,  44,  40,  34,  41,  45,  51,  65,  77,  95, 109, | ||||||
|  118, 113, 103,  92,  80,  68,  60,  56,  54,  57,  61,  69,  81,  93, 104, 114, |  118, 113, 103,  92,  80,  68,  60,  56,  54,  57,  61,  69,  81,  93, 104, 114, | ||||||
|  119, 116, 111, 106,  97,  88,  84,  74,  72,  75,  85,  89,  98, 107, 112, 117, |  119, 116, 111, 106,  97,  88,  84,  74,  72,  75,  85,  89,  98, 107, 112, 117 | ||||||
| }; | }; | ||||||
|  |  | ||||||
| static const int kMinLength = 2; |  | ||||||
|  |  | ||||||
| static int DistanceToPlaneCode(int xsize, int dist) { | static int DistanceToPlaneCode(int xsize, int dist) { | ||||||
|   const int yoffset = dist / xsize; |   const int yoffset = dist / xsize; | ||||||
|   const int xoffset = dist - yoffset * xsize; |   const int xoffset = dist - yoffset * xsize; | ||||||
| @@ -55,30 +73,44 @@ static WEBP_INLINE int FindMatchLength(const uint32_t* const array1, | |||||||
|   return match_len; |   return match_len; | ||||||
| } | } | ||||||
|  |  | ||||||
| #define HASH_BITS 18 | // ----------------------------------------------------------------------------- | ||||||
| #define HASH_SIZE (1 << HASH_BITS) | //  VP8LBackwardRefs | ||||||
| static const uint64_t kHashMultiplier = 0xc6a4a7935bd1e995ULL; |  | ||||||
| static const int kWindowSize = (1 << 20) - 120;  // A window with 1M pixels | void VP8LInitBackwardRefs(VP8LBackwardRefs* const refs) { | ||||||
|                                                  // (4 megabytes) - 120 |   if (refs != NULL) { | ||||||
|                                                  // special codes for short |     refs->refs = NULL; | ||||||
|                                                  // distances. |     refs->size = 0; | ||||||
|  |     refs->max_size = 0; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void VP8LClearBackwardRefs(VP8LBackwardRefs* const refs) { | ||||||
|  |   if (refs != NULL) { | ||||||
|  |     free(refs->refs); | ||||||
|  |     VP8LInitBackwardRefs(refs); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int VP8LBackwardRefsAlloc(VP8LBackwardRefs* const refs, int max_size) { | ||||||
|  |   assert(refs != NULL); | ||||||
|  |   refs->size = 0; | ||||||
|  |   refs->max_size = 0; | ||||||
|  |   refs->refs = (PixOrCopy*)malloc(max_size * sizeof(*refs->refs)); | ||||||
|  |   if (refs->refs == NULL) return 0; | ||||||
|  |   refs->max_size = max_size; | ||||||
|  |   return 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ----------------------------------------------------------------------------- | ||||||
|  | // Hash chains | ||||||
|  |  | ||||||
| static WEBP_INLINE uint64_t GetPixPairHash64(const uint32_t* const argb) { | static WEBP_INLINE uint64_t GetPixPairHash64(const uint32_t* const argb) { | ||||||
|   uint64_t key = ((uint64_t)(argb[1]) << 32) | argb[0]; |   uint64_t key = ((uint64_t)(argb[1]) << 32) | argb[0]; | ||||||
|   key *= kHashMultiplier; |   key = (key * HASH_MULTIPLIER) >> (64 - HASH_BITS); | ||||||
|   key >>= 64 - HASH_BITS; |  | ||||||
|   return key; |   return key; | ||||||
| } | } | ||||||
|  |  | ||||||
| typedef struct { | static int HashChainInit(HashChain* const p, int size) { | ||||||
|   // Stores the most recently added position with the given hash value. |  | ||||||
|   int32_t hash_to_first_index_[HASH_SIZE]; |  | ||||||
|   // chain_[pos] stores the previous position with the same hash value |  | ||||||
|   // for every pixel in the image. |  | ||||||
|   int32_t* chain_; |  | ||||||
| } VP8LHashChain; |  | ||||||
|  |  | ||||||
| static int VP8LHashChainInit(VP8LHashChain* const p, int size) { |  | ||||||
|   int i; |   int i; | ||||||
|   p->chain_ = (int*)malloc(size * sizeof(*p->chain_)); |   p->chain_ = (int*)malloc(size * sizeof(*p->chain_)); | ||||||
|   if (p->chain_ == NULL) { |   if (p->chain_ == NULL) { | ||||||
| @@ -93,23 +125,25 @@ static int VP8LHashChainInit(VP8LHashChain* const p, int size) { | |||||||
|   return 1; |   return 1; | ||||||
| } | } | ||||||
|  |  | ||||||
| static void VP8LHashChainClear(VP8LHashChain* const p) { | static void HashChainDelete(HashChain* const p) { | ||||||
|   if (p != NULL) { |   if (p != NULL) { | ||||||
|     free(p->chain_); |     free(p->chain_); | ||||||
|  |     free(p); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| static void VP8LHashChainInsert(VP8LHashChain* const p, | // Insertion of two pixels at a time. | ||||||
|                                 const uint32_t* const argb, int32_t pos) { | static void HashChainInsert(HashChain* const p, | ||||||
|   // Insertion of two pixels at a time. |                             const uint32_t* const argb, int pos) { | ||||||
|   const uint64_t hash_code = GetPixPairHash64(argb); |   const uint64_t hash_code = GetPixPairHash64(argb); | ||||||
|   p->chain_[pos] = p->hash_to_first_index_[hash_code]; |   p->chain_[pos] = p->hash_to_first_index_[hash_code]; | ||||||
|   p->hash_to_first_index_[hash_code] = pos; |   p->hash_to_first_index_[hash_code] = pos; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int HashChainFindCopy( | static int HashChainFindCopy(const HashChain* const p, | ||||||
|     const VP8LHashChain* const p, int quality, int index, int xsize, |                              int quality, int index, int xsize, | ||||||
|     const uint32_t* const argb, int maxlen, int* const distance_ptr, |                              const uint32_t* const argb, int maxlen, | ||||||
|  |                              int* const distance_ptr, | ||||||
|                              int* const length_ptr) { |                              int* const length_ptr) { | ||||||
|   const uint64_t hash_code = GetPixPairHash64(&argb[index]); |   const uint64_t hash_code = GetPixPairHash64(&argb[index]); | ||||||
|   int prev_length = 0; |   int prev_length = 0; | ||||||
| @@ -117,7 +151,7 @@ static int HashChainFindCopy( | |||||||
|   const int iter_min_mult = (quality < 50) ? 2 : (quality <= 75) ? 4 : 8; |   const int iter_min_mult = (quality < 50) ? 2 : (quality <= 75) ? 4 : 8; | ||||||
|   const int iter_min = -quality * iter_min_mult; |   const int iter_min = -quality * iter_min_mult; | ||||||
|   int iter_cnt = 10 + (quality >> 1); |   int iter_cnt = 10 + (quality >> 1); | ||||||
|   const int min_pos = (index > kWindowSize) ? index - kWindowSize : 0; |   const int min_pos = (index > WINDOW_SIZE) ? index - WINDOW_SIZE : 0; | ||||||
|   int32_t pos; |   int32_t pos; | ||||||
|   int64_t val; |   int64_t val; | ||||||
|   int best_length = 0; |   int best_length = 0; | ||||||
| @@ -162,7 +196,7 @@ static int HashChainFindCopy( | |||||||
|       best_val = val; |       best_val = val; | ||||||
|       best_length = curr_length; |       best_length = curr_length; | ||||||
|       best_distance = index - pos; |       best_distance = index - pos; | ||||||
|       if (curr_length >= kMaxLength) { |       if (curr_length >= MAX_LENGTH) { | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
|       if ((best_distance == 1 || best_distance == xsize) && |       if ((best_distance == 1 || best_distance == xsize) && | ||||||
| @@ -173,28 +207,32 @@ static int HashChainFindCopy( | |||||||
|   } |   } | ||||||
|   *distance_ptr = best_distance; |   *distance_ptr = best_distance; | ||||||
|   *length_ptr = best_length; |   *length_ptr = best_length; | ||||||
|   return best_length >= kMinLength; |   return (best_length >= MIN_LENGTH); | ||||||
| } | } | ||||||
|  |  | ||||||
| static WEBP_INLINE void PushBackCopy(VP8LBackwardRefs* const refs, int length) { | static WEBP_INLINE void PushBackCopy(VP8LBackwardRefs* const refs, int length) { | ||||||
|   while (length >= kMaxLength) { |   int size = refs->size; | ||||||
|     refs->refs[refs->size++] = PixOrCopyCreateCopy(1, kMaxLength); |   while (length >= MAX_LENGTH) { | ||||||
|     length -= kMaxLength; |     refs->refs[size++] = PixOrCopyCreateCopy(1, MAX_LENGTH); | ||||||
|  |     length -= MAX_LENGTH; | ||||||
|   } |   } | ||||||
|   if (length > 0) { |   if (length > 0) { | ||||||
|     refs->refs[refs->size++] = PixOrCopyCreateCopy(1, length); |     refs->refs[size++] = PixOrCopyCreateCopy(1, length); | ||||||
|   } |   } | ||||||
|  |   refs->size = size; | ||||||
| } | } | ||||||
|  |  | ||||||
| static void BackwardReferencesRle( | static void BackwardReferencesRle(int xsize, int ysize, | ||||||
|     int xsize, int ysize, const uint32_t* const argb, |                                   const uint32_t* const argb, | ||||||
|                                   VP8LBackwardRefs* const refs) { |                                   VP8LBackwardRefs* const refs) { | ||||||
|   const int pix_count = xsize * ysize; |   const int pix_count = xsize * ysize; | ||||||
|   int match_len = 0; |   int match_len = 0; | ||||||
|   int i; |   int i; | ||||||
|   refs->size = 0; |   refs->size = 0; | ||||||
|   for (i = 0; i < pix_count; ++i) { |   PushBackCopy(refs, match_len);    // i=0 case | ||||||
|     if (i >= 1 && argb[i] == argb[i - 1]) { |   refs->refs[refs->size++] = PixOrCopyCreateLiteral(argb[0]); | ||||||
|  |   for (i = 1; i < pix_count; ++i) { | ||||||
|  |     if (argb[i] == argb[i - 1]) { | ||||||
|       ++match_len; |       ++match_len; | ||||||
|     } else { |     } else { | ||||||
|       PushBackCopy(refs, match_len); |       PushBackCopy(refs, match_len); | ||||||
| @@ -205,22 +243,23 @@ static void BackwardReferencesRle( | |||||||
|   PushBackCopy(refs, match_len); |   PushBackCopy(refs, match_len); | ||||||
| } | } | ||||||
|  |  | ||||||
| static int BackwardReferencesHashChain( | static int BackwardReferencesHashChain(int xsize, int ysize, | ||||||
|     int xsize, int ysize, const uint32_t* const argb, |                                        const uint32_t* const argb, | ||||||
|     int cache_bits, int quality, VP8LBackwardRefs* const refs) { |                                        int cache_bits, int quality, | ||||||
|  |                                        VP8LBackwardRefs* const refs) { | ||||||
|   int i; |   int i; | ||||||
|   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; | ||||||
|   VP8LHashChain* hash_chain = (VP8LHashChain*)malloc(sizeof(*hash_chain)); |   HashChain* const hash_chain = (HashChain*)malloc(sizeof(*hash_chain)); | ||||||
|   VP8LColorCache hashers; |   VP8LColorCache hashers; | ||||||
|  |  | ||||||
|   if (hash_chain == NULL) return 0; |   if (hash_chain == NULL) return 0; | ||||||
|   if (!(cc_init = VP8LColorCacheInit(&hashers, cache_bits)) || |   cc_init = VP8LColorCacheInit(&hashers, cache_bits); | ||||||
|       !VP8LHashChainInit(hash_chain, pix_count)) { |   if (!cc_init) goto Error; | ||||||
|     goto Error; |  | ||||||
|   } |   if (!HashChainInit(hash_chain, pix_count)) goto Error; | ||||||
|  |  | ||||||
|   refs->size = 0; |   refs->size = 0; | ||||||
|   for (i = 0; i < pix_count; ) { |   for (i = 0; i < pix_count; ) { | ||||||
| @@ -229,23 +268,23 @@ static int BackwardReferencesHashChain( | |||||||
|     int len = 0; |     int len = 0; | ||||||
|     if (i < pix_count - 1) {  // FindCopy(i,..) reads pixels at [i] and [i + 1]. |     if (i < pix_count - 1) {  // FindCopy(i,..) reads pixels at [i] and [i + 1]. | ||||||
|       int maxlen = pix_count - i; |       int maxlen = pix_count - i; | ||||||
|       if (maxlen > kMaxLength) { |       if (maxlen > MAX_LENGTH) { | ||||||
|         maxlen = kMaxLength; |         maxlen = MAX_LENGTH; | ||||||
|       } |       } | ||||||
|       HashChainFindCopy(hash_chain, quality, i, xsize, argb, maxlen, |       HashChainFindCopy(hash_chain, quality, i, xsize, argb, maxlen, | ||||||
|                         &offset, &len); |                         &offset, &len); | ||||||
|     } |     } | ||||||
|     if (len >= kMinLength) { |     if (len >= MIN_LENGTH) { | ||||||
|       // Alternative#2: Insert the pixel at 'i' as literal, and code the |       // Alternative#2: Insert the pixel at 'i' as literal, and code the | ||||||
|       // pixels starting at 'i + 1' using backward reference. |       // pixels starting at 'i + 1' using backward reference. | ||||||
|       int offset2 = 0; |       int offset2 = 0; | ||||||
|       int len2 = 0; |       int len2 = 0; | ||||||
|       int k; |       int k; | ||||||
|       VP8LHashChainInsert(hash_chain, &argb[i], i); |       HashChainInsert(hash_chain, &argb[i], i); | ||||||
|       if (i < pix_count - 2) {  // FindCopy(i+1,..) reads [i + 1] and [i + 2]. |       if (i < pix_count - 2) {  // FindCopy(i+1,..) reads [i + 1] and [i + 2]. | ||||||
|         int maxlen = pix_count - (i + 1); |         int maxlen = pix_count - (i + 1); | ||||||
|         if (maxlen > kMaxLength) { |         if (maxlen > MAX_LENGTH) { | ||||||
|           maxlen = kMaxLength; |           maxlen = MAX_LENGTH; | ||||||
|         } |         } | ||||||
|         HashChainFindCopy(hash_chain, quality, |         HashChainFindCopy(hash_chain, quality, | ||||||
|                           i + 1, xsize, argb, maxlen, &offset2, &len2); |                           i + 1, xsize, argb, maxlen, &offset2, &len2); | ||||||
| @@ -264,15 +303,15 @@ static int BackwardReferencesHashChain( | |||||||
|           offset = offset2; |           offset = offset2; | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       if (len >= kMaxLength) { |       if (len >= MAX_LENGTH) { | ||||||
|         len = kMaxLength - 1; |         len = MAX_LENGTH - 1; | ||||||
|       } |       } | ||||||
|       refs->refs[refs->size++] = PixOrCopyCreateCopy(offset, len); |       refs->refs[refs->size++] = PixOrCopyCreateCopy(offset, len); | ||||||
|       for (k = 0; k < len; ++k) { |       for (k = 0; k < len; ++k) { | ||||||
|         VP8LColorCacheInsert(&hashers, argb[i + k]); |         VP8LColorCacheInsert(&hashers, argb[i + k]); | ||||||
|         if (k != 0 && i + k + 1 < pix_count) { |         if (k != 0 && i + k + 1 < pix_count) { | ||||||
|           // Add to the hash_chain (but cannot add the last pixel). |           // Add to the hash_chain (but cannot add the last pixel). | ||||||
|           VP8LHashChainInsert(hash_chain, &argb[i + k], i + k); |           HashChainInsert(hash_chain, &argb[i + k], i + k); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       i += len; |       i += len; | ||||||
| @@ -287,7 +326,7 @@ static int BackwardReferencesHashChain( | |||||||
|       ++refs->size; |       ++refs->size; | ||||||
|       VP8LColorCacheInsert(&hashers, argb[i]); |       VP8LColorCacheInsert(&hashers, argb[i]); | ||||||
|       if (i + 1 < pix_count) { |       if (i + 1 < pix_count) { | ||||||
|         VP8LHashChainInsert(hash_chain, &argb[i], i); |         HashChainInsert(hash_chain, &argb[i], i); | ||||||
|       } |       } | ||||||
|       ++i; |       ++i; | ||||||
|     } |     } | ||||||
| @@ -295,8 +334,7 @@ static int BackwardReferencesHashChain( | |||||||
|   ok = 1; |   ok = 1; | ||||||
| Error: | Error: | ||||||
|   if (cc_init) VP8LColorCacheClear(&hashers); |   if (cc_init) VP8LColorCacheClear(&hashers); | ||||||
|   VP8LHashChainClear(hash_chain); |   HashChainDelete(hash_chain); | ||||||
|   free(hash_chain); |  | ||||||
|   return ok; |   return ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -339,16 +377,15 @@ static int CostModelBuild(CostModel* const p, int xsize, int ysize, | |||||||
|   } |   } | ||||||
|   VP8LHistogramCreate(&histo, &refs, cache_bits); |   VP8LHistogramCreate(&histo, &refs, cache_bits); | ||||||
|   VP8LConvertPopulationCountTableToBitEstimates( |   VP8LConvertPopulationCountTableToBitEstimates( | ||||||
|       VP8LHistogramNumCodes(&histo), |       VP8LHistogramNumCodes(&histo), histo.literal_, p->literal_); | ||||||
|       &histo.literal_[0], &p->literal_[0]); |  | ||||||
|   VP8LConvertPopulationCountTableToBitEstimates( |   VP8LConvertPopulationCountTableToBitEstimates( | ||||||
|       VALUES_IN_BYTE, &histo.red_[0], &p->red_[0]); |       VALUES_IN_BYTE, histo.red_, p->red_); | ||||||
|   VP8LConvertPopulationCountTableToBitEstimates( |   VP8LConvertPopulationCountTableToBitEstimates( | ||||||
|       VALUES_IN_BYTE, &histo.blue_[0], &p->blue_[0]); |       VALUES_IN_BYTE, histo.blue_, p->blue_); | ||||||
|   VP8LConvertPopulationCountTableToBitEstimates( |   VP8LConvertPopulationCountTableToBitEstimates( | ||||||
|       VALUES_IN_BYTE, &histo.alpha_[0], &p->alpha_[0]); |       VALUES_IN_BYTE, histo.alpha_, p->alpha_); | ||||||
|   VP8LConvertPopulationCountTableToBitEstimates( |   VP8LConvertPopulationCountTableToBitEstimates( | ||||||
|       NUM_DISTANCE_CODES, &histo.distance_[0], &p->distance_[0]); |       NUM_DISTANCE_CODES, histo.distance_, p->distance_); | ||||||
|   ok = 1; |   ok = 1; | ||||||
|  |  | ||||||
|  Error: |  Error: | ||||||
| @@ -391,23 +428,25 @@ static int BackwardReferencesHashChainDistanceOnly( | |||||||
|   const int quality = 100; |   const int quality = 100; | ||||||
|   const int pix_count = xsize * ysize; |   const int pix_count = xsize * ysize; | ||||||
|   const int use_color_cache = (cache_bits > 0); |   const int use_color_cache = (cache_bits > 0); | ||||||
|   double* cost = (double*)malloc(pix_count * sizeof(*cost)); |   double* const cost = (double*)malloc(pix_count * sizeof(*cost)); | ||||||
|   CostModel* cost_model = (CostModel*)malloc(sizeof(*cost_model)); |   CostModel* cost_model = (CostModel*)malloc(sizeof(*cost_model)); | ||||||
|   VP8LHashChain* hash_chain = (VP8LHashChain*)malloc(sizeof(*hash_chain)); |   HashChain* hash_chain = (HashChain*)malloc(sizeof(*hash_chain)); | ||||||
|   VP8LColorCache hashers; |   VP8LColorCache hashers; | ||||||
|  |   const double mul0 = (recursive_cost_model != 0) ? 1.0 : 0.68; | ||||||
|  |   const double mul1 = (recursive_cost_model != 0) ? 1.0 : 0.82; | ||||||
|  |  | ||||||
|   if (cost == NULL || |   if (cost == NULL || cost_model == NULL || hash_chain == NULL) goto Error; | ||||||
|       cost_model == NULL || |  | ||||||
|       hash_chain == NULL || |   cc_init = VP8LColorCacheInit(&hashers, cache_bits); | ||||||
|       !(cc_init = VP8LColorCacheInit(&hashers, cache_bits)) || |   if (!cc_init || !HashChainInit(hash_chain, pix_count)) goto Error; | ||||||
|       !VP8LHashChainInit(hash_chain, pix_count)) { |  | ||||||
|  |   if (!CostModelBuild(cost_model, xsize, ysize, recursive_cost_model, argb, | ||||||
|  |                       cache_bits)) { | ||||||
|     goto Error; |     goto Error; | ||||||
|   } |   } | ||||||
|   CostModelBuild(cost_model, xsize, ysize, recursive_cost_model, argb, |  | ||||||
|                  cache_bits); |   for (i = 0; i < pix_count; ++i) cost[i] = 1e100; | ||||||
|   for (i = 0; i < pix_count; ++i) { |  | ||||||
|     cost[i] = 1e100; |  | ||||||
|   } |  | ||||||
|   // We loop one pixel at a time, but store all currently best points to |   // We loop one pixel at a time, but store all currently best points to | ||||||
|   // non-processed locations from this point. |   // non-processed locations from this point. | ||||||
|   dist_array[0] = 0; |   dist_array[0] = 0; | ||||||
| @@ -421,14 +460,14 @@ static int BackwardReferencesHashChainDistanceOnly( | |||||||
|       int offset = 0; |       int offset = 0; | ||||||
|       int len = 0; |       int len = 0; | ||||||
|       if (i < pix_count - 1) {  // FindCopy reads pixels at [i] and [i + 1]. |       if (i < pix_count - 1) {  // FindCopy reads pixels at [i] and [i + 1]. | ||||||
|         int maxlen = shortmax ? 2 : kMaxLength; |         int maxlen = shortmax ? 2 : MAX_LENGTH; | ||||||
|         if (maxlen > pix_count - i) { |         if (maxlen > pix_count - i) { | ||||||
|           maxlen = pix_count - i; |           maxlen = pix_count - i; | ||||||
|         } |         } | ||||||
|         HashChainFindCopy(hash_chain, quality, i, xsize, argb, maxlen, |         HashChainFindCopy(hash_chain, quality, i, xsize, argb, maxlen, | ||||||
|                           &offset, &len); |                           &offset, &len); | ||||||
|       } |       } | ||||||
|       if (len >= kMinLength) { |       if (len >= MIN_LENGTH) { | ||||||
|         const int code = DistanceToPlaneCode(xsize, offset); |         const int code = DistanceToPlaneCode(xsize, offset); | ||||||
|         const double distance_cost = |         const double distance_cost = | ||||||
|             prev_cost + GetDistanceCost(cost_model, code); |             prev_cost + GetDistanceCost(cost_model, code); | ||||||
| @@ -451,7 +490,7 @@ static int BackwardReferencesHashChainDistanceOnly( | |||||||
|             VP8LColorCacheInsert(&hashers, argb[i + k]); |             VP8LColorCacheInsert(&hashers, argb[i + k]); | ||||||
|             if (i + k + 1 < pix_count) { |             if (i + k + 1 < pix_count) { | ||||||
|               // Add to the hash_chain (but cannot add the last pixel). |               // Add to the hash_chain (but cannot add the last pixel). | ||||||
|               VP8LHashChainInsert(hash_chain, &argb[i + k], i + k); |               HashChainInsert(hash_chain, &argb[i + k], i + k); | ||||||
|             } |             } | ||||||
|           } |           } | ||||||
|           // 2) jump. |           // 2) jump. | ||||||
| @@ -461,19 +500,13 @@ static int BackwardReferencesHashChainDistanceOnly( | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (i < pix_count - 1) { |     if (i < pix_count - 1) { | ||||||
|       VP8LHashChainInsert(hash_chain, &argb[i], i); |       HashChainInsert(hash_chain, &argb[i], i); | ||||||
|     } |     } | ||||||
|     { |     { | ||||||
|       // inserting a literal pixel |       // inserting a literal pixel | ||||||
|       double cost_val = prev_cost; |       double cost_val = prev_cost; | ||||||
|       double mul0 = 1.0; |  | ||||||
|       double mul1 = 1.0; |  | ||||||
|       if (recursive_cost_model == 0) { |  | ||||||
|         mul0 = 0.68; |  | ||||||
|         mul1 = 0.82; |  | ||||||
|       } |  | ||||||
|       if (use_color_cache && VP8LColorCacheContains(&hashers, argb[i])) { |       if (use_color_cache && VP8LColorCacheContains(&hashers, argb[i])) { | ||||||
|         int ix = VP8LColorCacheGetIndex(&hashers, argb[i]); |         const int ix = VP8LColorCacheGetIndex(&hashers, argb[i]); | ||||||
|         cost_val += GetCacheCost(cost_model, ix) * mul0; |         cost_val += GetCacheCost(cost_model, ix) * mul0; | ||||||
|       } else { |       } else { | ||||||
|         cost_val += GetLiteralCost(cost_model, argb[i]) * mul1; |         cost_val += GetLiteralCost(cost_model, argb[i]) * mul1; | ||||||
| @@ -491,16 +524,16 @@ static int BackwardReferencesHashChainDistanceOnly( | |||||||
|   ok = 1; |   ok = 1; | ||||||
| Error: | Error: | ||||||
|   if (cc_init) VP8LColorCacheClear(&hashers); |   if (cc_init) VP8LColorCacheClear(&hashers); | ||||||
|   VP8LHashChainClear(hash_chain); |   HashChainDelete(hash_chain); | ||||||
|   free(hash_chain); |  | ||||||
|   free(cost_model); |   free(cost_model); | ||||||
|   free(cost); |   free(cost); | ||||||
|   return ok; |   return ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| static void TraceBackwards( | static int TraceBackwards(const uint32_t* const dist_array, | ||||||
|     const uint32_t* const dist_array, int dist_array_size, |                           int dist_array_size, | ||||||
|     uint32_t** const chosen_path, int* const chosen_path_size) { |                           uint32_t** const chosen_path, | ||||||
|  |                           int* const chosen_path_size) { | ||||||
|   int i; |   int i; | ||||||
|   // Count how many. |   // Count how many. | ||||||
|   int count = 0; |   int count = 0; | ||||||
| @@ -513,6 +546,8 @@ static void TraceBackwards( | |||||||
|   // Allocate. |   // Allocate. | ||||||
|   *chosen_path_size = count; |   *chosen_path_size = count; | ||||||
|   *chosen_path = (uint32_t*)malloc(count * sizeof(*chosen_path)); |   *chosen_path = (uint32_t*)malloc(count * sizeof(*chosen_path)); | ||||||
|  |   if (*chosen_path == NULL) return 0; | ||||||
|  |  | ||||||
|   // Write in reverse order. |   // Write in reverse order. | ||||||
|   for (i = dist_array_size - 1; i >= 0; ) { |   for (i = dist_array_size - 1; i >= 0; ) { | ||||||
|     int k = dist_array[i]; |     int k = dist_array[i]; | ||||||
| @@ -520,6 +555,7 @@ static void TraceBackwards( | |||||||
|     (*chosen_path)[--count] = k; |     (*chosen_path)[--count] = k; | ||||||
|     i -= k; |     i -= k; | ||||||
|   } |   } | ||||||
|  |   return 1; | ||||||
| } | } | ||||||
|  |  | ||||||
| static int BackwardReferencesHashChainFollowChosenPath( | static int BackwardReferencesHashChainFollowChosenPath( | ||||||
| @@ -535,12 +571,12 @@ static int BackwardReferencesHashChainFollowChosenPath( | |||||||
|   int ix; |   int ix; | ||||||
|   int ok = 0; |   int ok = 0; | ||||||
|   int cc_init = 0; |   int cc_init = 0; | ||||||
|   VP8LHashChain* hash_chain = (VP8LHashChain*)malloc(sizeof(*hash_chain)); |   HashChain* hash_chain = (HashChain*)malloc(sizeof(*hash_chain)); | ||||||
|   VP8LColorCache hashers; |   VP8LColorCache hashers; | ||||||
|  |  | ||||||
|   if (hash_chain == NULL || |   if (hash_chain == NULL || | ||||||
|       !(cc_init = VP8LColorCacheInit(&hashers, cache_bits)) || |       !(cc_init = VP8LColorCacheInit(&hashers, cache_bits)) || | ||||||
|       !VP8LHashChainInit(hash_chain, pix_count)) { |       !HashChainInit(hash_chain, pix_count)) { | ||||||
|     goto Error; |     goto Error; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -558,7 +594,7 @@ static int BackwardReferencesHashChainFollowChosenPath( | |||||||
|         VP8LColorCacheInsert(&hashers, argb[i + k]); |         VP8LColorCacheInsert(&hashers, argb[i + k]); | ||||||
|         if (i + k + 1 < pix_count) { |         if (i + k + 1 < pix_count) { | ||||||
|           // Add to the hash_chain (but cannot add the last pixel). |           // Add to the hash_chain (but cannot add the last pixel). | ||||||
|           VP8LHashChainInsert(hash_chain, &argb[i + k], i + k); |           HashChainInsert(hash_chain, &argb[i + k], i + k); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       i += len; |       i += len; | ||||||
| @@ -572,7 +608,7 @@ static int BackwardReferencesHashChainFollowChosenPath( | |||||||
|       } |       } | ||||||
|       VP8LColorCacheInsert(&hashers, argb[i]); |       VP8LColorCacheInsert(&hashers, argb[i]); | ||||||
|       if (i + 1 < pix_count) { |       if (i + 1 < pix_count) { | ||||||
|         VP8LHashChainInsert(hash_chain, &argb[i], i); |         HashChainInsert(hash_chain, &argb[i], i); | ||||||
|       } |       } | ||||||
|       ++i; |       ++i; | ||||||
|     } |     } | ||||||
| @@ -582,31 +618,35 @@ static int BackwardReferencesHashChainFollowChosenPath( | |||||||
|   ok = 1; |   ok = 1; | ||||||
| Error: | Error: | ||||||
|   if (cc_init) VP8LColorCacheClear(&hashers); |   if (cc_init) VP8LColorCacheClear(&hashers); | ||||||
|   VP8LHashChainClear(hash_chain); |   HashChainDelete(hash_chain); | ||||||
|   free(hash_chain); |  | ||||||
|   return ok; |   return ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| // Returns 1 on success. | // Returns 1 on success. | ||||||
| static int BackwardReferencesTraceBackwards( | static int BackwardReferencesTraceBackwards(int xsize, int ysize, | ||||||
|     int xsize, int ysize, int recursive_cost_model, const uint32_t* const argb, |                                             int recursive_cost_model, | ||||||
|     int cache_bits, VP8LBackwardRefs* const refs) { |                                             const uint32_t* const argb, | ||||||
|  |                                             int cache_bits, | ||||||
|  |                                             VP8LBackwardRefs* const refs) { | ||||||
|   int ok = 0; |   int ok = 0; | ||||||
|   const int dist_array_size = xsize * ysize; |   const int dist_array_size = xsize * ysize; | ||||||
|   uint32_t* chosen_path = NULL; |   uint32_t* chosen_path = NULL; | ||||||
|   int chosen_path_size = 0; |   int chosen_path_size = 0; | ||||||
|   uint32_t* const dist_array = |   uint32_t* dist_array = | ||||||
|       (uint32_t*)malloc(dist_array_size * sizeof(*dist_array)); |       (uint32_t*)malloc(dist_array_size * sizeof(*dist_array)); | ||||||
|   if (dist_array == NULL) { |  | ||||||
|     goto Error; |   if (dist_array == NULL) goto Error; | ||||||
|   } |  | ||||||
|   if (!BackwardReferencesHashChainDistanceOnly( |   if (!BackwardReferencesHashChainDistanceOnly( | ||||||
|       xsize, ysize, recursive_cost_model, argb, cache_bits, dist_array)) { |       xsize, ysize, recursive_cost_model, argb, cache_bits, dist_array)) { | ||||||
|     free(dist_array); |  | ||||||
|     goto Error; |     goto Error; | ||||||
|   } |   } | ||||||
|   TraceBackwards(dist_array, dist_array_size, &chosen_path, &chosen_path_size); |   if (!TraceBackwards(dist_array, dist_array_size, | ||||||
|   free(dist_array); |                       &chosen_path, &chosen_path_size)) { | ||||||
|  |     goto Error; | ||||||
|  |   } | ||||||
|  |   free(dist_array);   // no need to retain this memory any longer | ||||||
|  |   dist_array = NULL; | ||||||
|   if (!BackwardReferencesHashChainFollowChosenPath( |   if (!BackwardReferencesHashChainFollowChosenPath( | ||||||
|       xsize, ysize, argb, cache_bits, chosen_path, chosen_path_size, refs)) { |       xsize, ysize, argb, cache_bits, chosen_path, chosen_path_size, refs)) { | ||||||
|     goto Error; |     goto Error; | ||||||
| @@ -614,6 +654,7 @@ static int BackwardReferencesTraceBackwards( | |||||||
|   ok = 1; |   ok = 1; | ||||||
|  Error: |  Error: | ||||||
|   free(chosen_path); |   free(chosen_path); | ||||||
|  |   free(dist_array); | ||||||
|   return ok; |   return ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -693,9 +734,8 @@ int VP8LGetBackwardReferences(int width, int height, | |||||||
|     *best = refs_rle; |     *best = refs_rle; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (use_2d_locality) {  // Use backward reference with 2D locality. |   if (use_2d_locality) BackwardReferences2DLocality(width, best); | ||||||
|     BackwardReferences2DLocality(width, best); |  | ||||||
|   } |  | ||||||
|   ok = 1; |   ok = 1; | ||||||
|  |  | ||||||
|  End: |  End: | ||||||
| @@ -706,17 +746,18 @@ int VP8LGetBackwardReferences(int width, int height, | |||||||
| } | } | ||||||
|  |  | ||||||
| // Returns 1 on success. | // Returns 1 on success. | ||||||
| static int ComputeCacheHistogram( | static int ComputeCacheHistogram(const uint32_t* const argb, | ||||||
|     const uint32_t* const argb, int xsize, int ysize, |                                  int xsize, int ysize, | ||||||
|     const VP8LBackwardRefs* const refs, int cache_bits, |                                  const VP8LBackwardRefs* const refs, | ||||||
|  |                                  int cache_bits, | ||||||
|                                  VP8LHistogram* const histo) { |                                  VP8LHistogram* const histo) { | ||||||
|   int pixel_index = 0; |   int pixel_index = 0; | ||||||
|   int i; |   int i; | ||||||
|   uint32_t k; |   uint32_t k; | ||||||
|   VP8LColorCache hashers; |   VP8LColorCache hashers; | ||||||
|   if (!VP8LColorCacheInit(&hashers, cache_bits)) { |  | ||||||
|     return 0; |   if (!VP8LColorCacheInit(&hashers, cache_bits)) return 0; | ||||||
|   } |  | ||||||
|   for (i = 0; i < refs->size; ++i) { |   for (i = 0; i < refs->size; ++i) { | ||||||
|     const PixOrCopy* const v = &refs->refs[i]; |     const PixOrCopy* const v = &refs->refs[i]; | ||||||
|     if (PixOrCopyIsLiteral(v)) { |     if (PixOrCopyIsLiteral(v)) { | ||||||
| @@ -745,8 +786,8 @@ static int ComputeCacheHistogram( | |||||||
| } | } | ||||||
|  |  | ||||||
| // Returns how many bits are to be used for a color cache. | // Returns how many bits are to be used for a color cache. | ||||||
| int VP8LCalculateEstimateForCacheSize( | int VP8LCalculateEstimateForCacheSize(const uint32_t* const argb, | ||||||
|     const uint32_t* const argb, int xsize, int ysize, |                                       int xsize, int ysize, | ||||||
|                                       int* const best_cache_bits) { |                                       int* const best_cache_bits) { | ||||||
|   int ok = 0; |   int ok = 0; | ||||||
|   int cache_bits; |   int cache_bits; | ||||||
| @@ -758,7 +799,7 @@ int VP8LCalculateEstimateForCacheSize( | |||||||
|       !BackwardReferencesHashChain(xsize, ysize, argb, 0, quality, &refs)) { |       !BackwardReferencesHashChain(xsize, ysize, argb, 0, quality, &refs)) { | ||||||
|     goto Error; |     goto Error; | ||||||
|   } |   } | ||||||
|   for (cache_bits = 0; cache_bits <= kColorCacheBitsMax; ++cache_bits) { |   for (cache_bits = 0; cache_bits <= MAX_COLOR_CACHE_BITS; ++cache_bits) { | ||||||
|     double cur_entropy; |     double cur_entropy; | ||||||
|     VP8LHistogram histo; |     VP8LHistogram histo; | ||||||
|     VP8LHistogramInit(&histo, cache_bits); |     VP8LHistogramInit(&histo, cache_bits); | ||||||
|   | |||||||
| @@ -17,16 +17,22 @@ | |||||||
| #include <stdint.h> | #include <stdint.h> | ||||||
| #include <stdlib.h> | #include <stdlib.h> | ||||||
| #include "../webp/types.h" | #include "../webp/types.h" | ||||||
|  | #include "../webp/format_constants.h" | ||||||
|  |  | ||||||
| #if defined(__cplusplus) || defined(c_plusplus) | #if defined(__cplusplus) || defined(c_plusplus) | ||||||
| extern "C" { | extern "C" { | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| // The spec allows 11, we use 9 bits to reduce memory consumption in encoding. | // The spec allows 11, we use 9 bits to reduce memory consumption in encoding. | ||||||
| // Having 9 instead of 11 removes about 0.25 % of compression density. | // Having 9 instead of 11 only removes about 0.25 % of compression density. | ||||||
| static const int kColorCacheBitsMax = 9; | #define MAX_COLOR_CACHE_BITS 9 | ||||||
| #define PIX_OR_COPY_CODES_MAX (256 + 24 + (1 << 9)) |  | ||||||
| static const int kMaxLength = 4096; | // Max ever number of codes we'll use: | ||||||
|  | #define PIX_OR_COPY_CODES_MAX \ | ||||||
|  |     (NUM_LITERAL_CODES + NUM_LENGTH_CODES + (1 << MAX_COLOR_CACHE_BITS)) | ||||||
|  |  | ||||||
|  | // ----------------------------------------------------------------------------- | ||||||
|  | // PrefixEncode() | ||||||
|  |  | ||||||
| // use GNU builtins where available. | // use GNU builtins where available. | ||||||
| #if defined(__GNUC__) && \ | #if defined(__GNUC__) && \ | ||||||
| @@ -36,16 +42,14 @@ static WEBP_INLINE int BitsLog2Floor(uint32_t n) { | |||||||
| } | } | ||||||
| #else | #else | ||||||
| static WEBP_INLINE int BitsLog2Floor(uint32_t n) { | static WEBP_INLINE int BitsLog2Floor(uint32_t n) { | ||||||
|   int log; |   int log = 0; | ||||||
|   uint32_t value; |   uint32_t value = n; | ||||||
|   int i; |   int i; | ||||||
|   if (n == 0) |  | ||||||
|     return -1; |   if (value == 0) return -1; | ||||||
|   log = 0; |  | ||||||
|   value = n; |  | ||||||
|   for (i = 4; i >= 0; --i) { |   for (i = 4; i >= 0; --i) { | ||||||
|     int shift = (1 << i); |     const int shift = (1 << i); | ||||||
|     uint32_t x = value >> shift; |     const uint32_t x = value >> shift; | ||||||
|     if (x != 0) { |     if (x != 0) { | ||||||
|       value = x; |       value = x; | ||||||
|       log += shift; |       log += shift; | ||||||
| @@ -56,7 +60,7 @@ static WEBP_INLINE int BitsLog2Floor(uint32_t n) { | |||||||
| #endif | #endif | ||||||
|  |  | ||||||
| static WEBP_INLINE int VP8LBitsLog2Ceiling(uint32_t n) { | static WEBP_INLINE int VP8LBitsLog2Ceiling(uint32_t n) { | ||||||
|   int floor = BitsLog2Floor(n); |   const int floor = BitsLog2Floor(n); | ||||||
|   if (n == (n & ~(n - 1)))  // zero or a power of two. |   if (n == (n & ~(n - 1)))  // zero or a power of two. | ||||||
|     return floor; |     return floor; | ||||||
|   else |   else | ||||||
| @@ -66,28 +70,29 @@ static WEBP_INLINE int VP8LBitsLog2Ceiling(uint32_t n) { | |||||||
| // Splitting of distance and length codes into prefixes and | // Splitting of distance and length codes into prefixes and | ||||||
| // extra bits. The prefixes are encoded with an entropy code | // extra bits. The prefixes are encoded with an entropy code | ||||||
| // while the extra bits are stored just as normal bits. | // while the extra bits are stored just as normal bits. | ||||||
| static WEBP_INLINE void PrefixEncode( | static WEBP_INLINE void PrefixEncode(int distance, int* const code, | ||||||
|     int distance, |                                      int* const extra_bits_count, | ||||||
|     int *code, |                                      int* const extra_bits_value) { | ||||||
|     int *extra_bits_count, |  | ||||||
|     int *extra_bits_value) { |  | ||||||
|   // Collect the two most significant bits where the highest bit is 1. |   // Collect the two most significant bits where the highest bit is 1. | ||||||
|   const int highest_bit = BitsLog2Floor(--distance); |   const int highest_bit = BitsLog2Floor(--distance); | ||||||
|   // & 0x3f is to make behavior well defined when highest_bit |   // & 0x3f is to make behavior well defined when highest_bit | ||||||
|   // does not exist or is the least significant bit. |   // does not exist or is the least significant bit. | ||||||
|   const int second_highest_bit = |   const int second_highest_bit = | ||||||
|       (distance >> ((highest_bit - 1) & 0x3f)) & 1; |       (distance >> ((highest_bit - 1) & 0x3f)) & 1; | ||||||
|   *extra_bits_count = (highest_bit > 0) ? highest_bit - 1 : 0; |   *extra_bits_count = (highest_bit > 0) ? (highest_bit - 1) : 0; | ||||||
|   *extra_bits_value = distance & ((1 << *extra_bits_count) - 1); |   *extra_bits_value = distance & ((1 << *extra_bits_count) - 1); | ||||||
|   *code = (highest_bit > 0) ? 2 * highest_bit + second_highest_bit : |   *code = (highest_bit > 0) ? (2 * highest_bit + second_highest_bit) | ||||||
|       (highest_bit == 0) ? 1 : 0; |                             : (highest_bit == 0) ? 1 : 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // ----------------------------------------------------------------------------- | ||||||
|  | // PixOrCopy | ||||||
|  |  | ||||||
| enum Mode { | enum Mode { | ||||||
|   kLiteral, |   kLiteral, | ||||||
|   kCacheIdx, |   kCacheIdx, | ||||||
|   kCopy, |   kCopy, | ||||||
|   kNone, |   kNone | ||||||
| }; | }; | ||||||
|  |  | ||||||
| typedef struct { | typedef struct { | ||||||
| @@ -97,7 +102,6 @@ typedef struct { | |||||||
|   uint32_t argb_or_distance; |   uint32_t argb_or_distance; | ||||||
| } PixOrCopy; | } PixOrCopy; | ||||||
|  |  | ||||||
|  |  | ||||||
| static WEBP_INLINE PixOrCopy PixOrCopyCreateCopy(uint32_t distance, | static WEBP_INLINE PixOrCopy PixOrCopyCreateCopy(uint32_t distance, | ||||||
|                                                  uint16_t len) { |                                                  uint16_t len) { | ||||||
|   PixOrCopy retval; |   PixOrCopy retval; | ||||||
| @@ -110,7 +114,7 @@ static WEBP_INLINE PixOrCopy PixOrCopyCreateCopy(uint32_t distance, | |||||||
| static WEBP_INLINE PixOrCopy PixOrCopyCreateCacheIdx(int idx) { | static WEBP_INLINE PixOrCopy PixOrCopyCreateCacheIdx(int idx) { | ||||||
|   PixOrCopy retval; |   PixOrCopy retval; | ||||||
|   assert(idx >= 0); |   assert(idx >= 0); | ||||||
|   assert(idx < (1 << kColorCacheBitsMax)); |   assert(idx < (1 << MAX_COLOR_CACHE_BITS)); | ||||||
|   retval.mode = kCacheIdx; |   retval.mode = kCacheIdx; | ||||||
|   retval.argb_or_distance = idx; |   retval.argb_or_distance = idx; | ||||||
|   retval.len = 1; |   retval.len = 1; | ||||||
| @@ -154,7 +158,7 @@ static WEBP_INLINE uint32_t PixOrCopyArgb(const PixOrCopy* const p) { | |||||||
|  |  | ||||||
| static WEBP_INLINE uint32_t PixOrCopyCacheIdx(const PixOrCopy* const p) { | static WEBP_INLINE uint32_t PixOrCopyCacheIdx(const PixOrCopy* const p) { | ||||||
|   assert(p->mode == kCacheIdx); |   assert(p->mode == kCacheIdx); | ||||||
|   assert(p->argb_or_distance < (1U << kColorCacheBitsMax)); |   assert(p->argb_or_distance < (1U << MAX_COLOR_CACHE_BITS)); | ||||||
|   return p->argb_or_distance; |   return p->argb_or_distance; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -172,33 +176,17 @@ typedef struct { | |||||||
|   int max_size;  // maximum capacity |   int max_size;  // maximum capacity | ||||||
| } VP8LBackwardRefs; | } VP8LBackwardRefs; | ||||||
|  |  | ||||||
|  | // Initialize the object. Must be called first. 'refs' can be NULL. | ||||||
|  | void VP8LInitBackwardRefs(VP8LBackwardRefs* const refs); | ||||||
|  |  | ||||||
| static WEBP_INLINE void VP8LInitBackwardRefs(VP8LBackwardRefs* const refs) { | // Release memory and re-initialize the object. 'refs' can be NULL. | ||||||
|   if (refs != NULL) { | void VP8LClearBackwardRefs(VP8LBackwardRefs* const refs); | ||||||
|     refs->refs = NULL; |  | ||||||
|     refs->size = 0; |  | ||||||
|     refs->max_size = 0; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static WEBP_INLINE void VP8LClearBackwardRefs(VP8LBackwardRefs* const refs) { |  | ||||||
|   if (refs != NULL) { |  | ||||||
|     free(refs->refs); |  | ||||||
|     VP8LInitBackwardRefs(refs); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Allocate 'max_size' references. Returns false in case of memory error. | // Allocate 'max_size' references. Returns false in case of memory error. | ||||||
| static WEBP_INLINE int VP8LBackwardRefsAlloc(VP8LBackwardRefs* const refs, | int VP8LBackwardRefsAlloc(VP8LBackwardRefs* const refs, int max_size); | ||||||
|                                              int max_size) { |  | ||||||
|   assert(refs != NULL); | // ----------------------------------------------------------------------------- | ||||||
|   refs->size = 0; | // Main entry points | ||||||
|   refs->max_size = 0; |  | ||||||
|   refs->refs = (PixOrCopy*)malloc(max_size * sizeof(*refs->refs)); |  | ||||||
|   if (refs->refs == NULL) return 0; |  | ||||||
|   refs->max_size = max_size; |  | ||||||
|   return 1; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Evaluates best possible backward references for specified quality. | // Evaluates best possible backward references for specified quality. | ||||||
| // Further optimize for 2D locality if use_2d_locality flag is set. | // Further optimize for 2D locality if use_2d_locality flag is set. | ||||||
| @@ -208,8 +196,8 @@ int VP8LGetBackwardReferences(int width, int height, | |||||||
|                               VP8LBackwardRefs* const best); |                               VP8LBackwardRefs* const best); | ||||||
|  |  | ||||||
| // Produce an estimate for a good color cache size for the image. | // Produce an estimate for a good color cache size for the image. | ||||||
| int VP8LCalculateEstimateForCacheSize( | int VP8LCalculateEstimateForCacheSize(const uint32_t* const argb, | ||||||
|     const uint32_t* const argb, int xsize, int ysize, |                                       int xsize, int ysize, | ||||||
|                                       int* const best_cache_bits); |                                       int* const best_cache_bits); | ||||||
|  |  | ||||||
| #if defined(__cplusplus) || defined(c_plusplus) | #if defined(__cplusplus) || defined(c_plusplus) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Pascal Massimino
					Pascal Massimino