xref: /aosp_15_r20/external/webp/src/enc/cost_enc.c (revision b2055c353e87c8814eb2b6b1b11112a1562253bd)
1*b2055c35SXin Li // Copyright 2011 Google Inc. All Rights Reserved.
2*b2055c35SXin Li //
3*b2055c35SXin Li // Use of this source code is governed by a BSD-style license
4*b2055c35SXin Li // that can be found in the COPYING file in the root of the source
5*b2055c35SXin Li // tree. An additional intellectual property rights grant can be found
6*b2055c35SXin Li // in the file PATENTS. All contributing project authors may
7*b2055c35SXin Li // be found in the AUTHORS file in the root of the source tree.
8*b2055c35SXin Li // -----------------------------------------------------------------------------
9*b2055c35SXin Li //
10*b2055c35SXin Li // Cost tables for level and modes
11*b2055c35SXin Li //
12*b2055c35SXin Li // Author: Skal ([email protected])
13*b2055c35SXin Li 
14*b2055c35SXin Li #include "src/enc/cost_enc.h"
15*b2055c35SXin Li 
16*b2055c35SXin Li //------------------------------------------------------------------------------
17*b2055c35SXin Li // Level cost tables
18*b2055c35SXin Li 
19*b2055c35SXin Li // For each given level, the following table gives the pattern of contexts to
20*b2055c35SXin Li // use for coding it (in [][0]) as well as the bit value to use for each
21*b2055c35SXin Li // context (in [][1]).
22*b2055c35SXin Li const uint16_t VP8LevelCodes[MAX_VARIABLE_LEVEL][2] = {
23*b2055c35SXin Li                   {0x001, 0x000}, {0x007, 0x001}, {0x00f, 0x005},
24*b2055c35SXin Li   {0x00f, 0x00d}, {0x033, 0x003}, {0x033, 0x003}, {0x033, 0x023},
25*b2055c35SXin Li   {0x033, 0x023}, {0x033, 0x023}, {0x033, 0x023}, {0x0d3, 0x013},
26*b2055c35SXin Li   {0x0d3, 0x013}, {0x0d3, 0x013}, {0x0d3, 0x013}, {0x0d3, 0x013},
27*b2055c35SXin Li   {0x0d3, 0x013}, {0x0d3, 0x013}, {0x0d3, 0x013}, {0x0d3, 0x093},
28*b2055c35SXin Li   {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093},
29*b2055c35SXin Li   {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093},
30*b2055c35SXin Li   {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093},
31*b2055c35SXin Li   {0x0d3, 0x093}, {0x0d3, 0x093}, {0x0d3, 0x093}, {0x153, 0x053},
32*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
33*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
34*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
35*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
36*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
37*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
38*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053},
39*b2055c35SXin Li   {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x053}, {0x153, 0x153}
40*b2055c35SXin Li };
41*b2055c35SXin Li 
VariableLevelCost(int level,const uint8_t probas[NUM_PROBAS])42*b2055c35SXin Li static int VariableLevelCost(int level, const uint8_t probas[NUM_PROBAS]) {
43*b2055c35SXin Li   int pattern = VP8LevelCodes[level - 1][0];
44*b2055c35SXin Li   int bits = VP8LevelCodes[level - 1][1];
45*b2055c35SXin Li   int cost = 0;
46*b2055c35SXin Li   int i;
47*b2055c35SXin Li   for (i = 2; pattern; ++i) {
48*b2055c35SXin Li     if (pattern & 1) {
49*b2055c35SXin Li       cost += VP8BitCost(bits & 1, probas[i]);
50*b2055c35SXin Li     }
51*b2055c35SXin Li     bits >>= 1;
52*b2055c35SXin Li     pattern >>= 1;
53*b2055c35SXin Li   }
54*b2055c35SXin Li   return cost;
55*b2055c35SXin Li }
56*b2055c35SXin Li 
57*b2055c35SXin Li //------------------------------------------------------------------------------
58*b2055c35SXin Li // Pre-calc level costs once for all
59*b2055c35SXin Li 
VP8CalculateLevelCosts(VP8EncProba * const proba)60*b2055c35SXin Li void VP8CalculateLevelCosts(VP8EncProba* const proba) {
61*b2055c35SXin Li   int ctype, band, ctx;
62*b2055c35SXin Li 
63*b2055c35SXin Li   if (!proba->dirty_) return;  // nothing to do.
64*b2055c35SXin Li 
65*b2055c35SXin Li   for (ctype = 0; ctype < NUM_TYPES; ++ctype) {
66*b2055c35SXin Li     int n;
67*b2055c35SXin Li     for (band = 0; band < NUM_BANDS; ++band) {
68*b2055c35SXin Li       for (ctx = 0; ctx < NUM_CTX; ++ctx) {
69*b2055c35SXin Li         const uint8_t* const p = proba->coeffs_[ctype][band][ctx];
70*b2055c35SXin Li         uint16_t* const table = proba->level_cost_[ctype][band][ctx];
71*b2055c35SXin Li         const int cost0 = (ctx > 0) ? VP8BitCost(1, p[0]) : 0;
72*b2055c35SXin Li         const int cost_base = VP8BitCost(1, p[1]) + cost0;
73*b2055c35SXin Li         int v;
74*b2055c35SXin Li         table[0] = VP8BitCost(0, p[1]) + cost0;
75*b2055c35SXin Li         for (v = 1; v <= MAX_VARIABLE_LEVEL; ++v) {
76*b2055c35SXin Li           table[v] = cost_base + VariableLevelCost(v, p);
77*b2055c35SXin Li         }
78*b2055c35SXin Li         // Starting at level 67 and up, the variable part of the cost is
79*b2055c35SXin Li         // actually constant.
80*b2055c35SXin Li       }
81*b2055c35SXin Li     }
82*b2055c35SXin Li     for (n = 0; n < 16; ++n) {    // replicate bands. We don't need to sentinel.
83*b2055c35SXin Li       for (ctx = 0; ctx < NUM_CTX; ++ctx) {
84*b2055c35SXin Li         proba->remapped_costs_[ctype][n][ctx] =
85*b2055c35SXin Li             proba->level_cost_[ctype][VP8EncBands[n]][ctx];
86*b2055c35SXin Li       }
87*b2055c35SXin Li     }
88*b2055c35SXin Li   }
89*b2055c35SXin Li   proba->dirty_ = 0;
90*b2055c35SXin Li }
91*b2055c35SXin Li 
92*b2055c35SXin Li //------------------------------------------------------------------------------
93*b2055c35SXin Li // Mode cost tables.
94*b2055c35SXin Li 
95*b2055c35SXin Li // These are the fixed probabilities (in the coding trees) turned into bit-cost
96*b2055c35SXin Li // by calling VP8BitCost().
97*b2055c35SXin Li const uint16_t VP8FixedCostsUV[4] = { 302, 984, 439, 642 };
98*b2055c35SXin Li // note: these values include the fixed VP8BitCost(1, 145) mode selection cost.
99*b2055c35SXin Li const uint16_t VP8FixedCostsI16[4] = { 663, 919, 872, 919 };
100*b2055c35SXin Li const uint16_t VP8FixedCostsI4[NUM_BMODES][NUM_BMODES][NUM_BMODES] = {
101*b2055c35SXin Li   { {   40, 1151, 1723, 1874, 2103, 2019, 1628, 1777, 2226, 2137 },
102*b2055c35SXin Li     {  192,  469, 1296, 1308, 1849, 1794, 1781, 1703, 1713, 1522 },
103*b2055c35SXin Li     {  142,  910,  762, 1684, 1849, 1576, 1460, 1305, 1801, 1657 },
104*b2055c35SXin Li     {  559,  641, 1370,  421, 1182, 1569, 1612, 1725,  863, 1007 },
105*b2055c35SXin Li     {  299, 1059, 1256, 1108,  636, 1068, 1581, 1883,  869, 1142 },
106*b2055c35SXin Li     {  277, 1111,  707, 1362, 1089,  672, 1603, 1541, 1545, 1291 },
107*b2055c35SXin Li     {  214,  781, 1609, 1303, 1632, 2229,  726, 1560, 1713,  918 },
108*b2055c35SXin Li     {  152, 1037, 1046, 1759, 1983, 2174, 1358,  742, 1740, 1390 },
109*b2055c35SXin Li     {  512, 1046, 1420,  753,  752, 1297, 1486, 1613,  460, 1207 },
110*b2055c35SXin Li     {  424,  827, 1362,  719, 1462, 1202, 1199, 1476, 1199,  538 } },
111*b2055c35SXin Li   { {  240,  402, 1134, 1491, 1659, 1505, 1517, 1555, 1979, 2099 },
112*b2055c35SXin Li     {  467,  242,  960, 1232, 1714, 1620, 1834, 1570, 1676, 1391 },
113*b2055c35SXin Li     {  500,  455,  463, 1507, 1699, 1282, 1564,  982, 2114, 2114 },
114*b2055c35SXin Li     {  672,  643, 1372,  331, 1589, 1667, 1453, 1938,  996,  876 },
115*b2055c35SXin Li     {  458,  783, 1037,  911,  738,  968, 1165, 1518,  859, 1033 },
116*b2055c35SXin Li     {  504,  815,  504, 1139, 1219,  719, 1506, 1085, 1268, 1268 },
117*b2055c35SXin Li     {  333,  630, 1445, 1239, 1883, 3672,  799, 1548, 1865,  598 },
118*b2055c35SXin Li     {  399,  644,  746, 1342, 1856, 1350, 1493,  613, 1855, 1015 },
119*b2055c35SXin Li     {  622,  749, 1205,  608, 1066, 1408, 1290, 1406,  546,  971 },
120*b2055c35SXin Li     {  500,  753, 1041,  668, 1230, 1617, 1297, 1425, 1383,  523 } },
121*b2055c35SXin Li   { {  394,  553,  523, 1502, 1536,  981, 1608, 1142, 1666, 2181 },
122*b2055c35SXin Li     {  655,  430,  375, 1411, 1861, 1220, 1677, 1135, 1978, 1553 },
123*b2055c35SXin Li     {  690,  640,  245, 1954, 2070, 1194, 1528,  982, 1972, 2232 },
124*b2055c35SXin Li     {  559,  834,  741,  867, 1131,  980, 1225,  852, 1092,  784 },
125*b2055c35SXin Li     {  690,  875,  516,  959,  673,  894, 1056, 1190, 1528, 1126 },
126*b2055c35SXin Li     {  740,  951,  384, 1277, 1177,  492, 1579, 1155, 1846, 1513 },
127*b2055c35SXin Li     {  323,  775, 1062, 1776, 3062, 1274,  813, 1188, 1372,  655 },
128*b2055c35SXin Li     {  488,  971,  484, 1767, 1515, 1775, 1115,  503, 1539, 1461 },
129*b2055c35SXin Li     {  740, 1006,  998,  709,  851, 1230, 1337,  788,  741,  721 },
130*b2055c35SXin Li     {  522, 1073,  573, 1045, 1346,  887, 1046, 1146, 1203,  697 } },
131*b2055c35SXin Li   { {  105,  864, 1442, 1009, 1934, 1840, 1519, 1920, 1673, 1579 },
132*b2055c35SXin Li     {  534,  305, 1193,  683, 1388, 2164, 1802, 1894, 1264, 1170 },
133*b2055c35SXin Li     {  305,  518,  877, 1108, 1426, 3215, 1425, 1064, 1320, 1242 },
134*b2055c35SXin Li     {  683,  732, 1927,  257, 1493, 2048, 1858, 1552, 1055,  947 },
135*b2055c35SXin Li     {  394,  814, 1024,  660,  959, 1556, 1282, 1289,  893, 1047 },
136*b2055c35SXin Li     {  528,  615,  996,  940, 1201,  635, 1094, 2515,  803, 1358 },
137*b2055c35SXin Li     {  347,  614, 1609, 1187, 3133, 1345, 1007, 1339, 1017,  667 },
138*b2055c35SXin Li     {  218,  740,  878, 1605, 3650, 3650, 1345,  758, 1357, 1617 },
139*b2055c35SXin Li     {  672,  750, 1541,  558, 1257, 1599, 1870, 2135,  402, 1087 },
140*b2055c35SXin Li     {  592,  684, 1161,  430, 1092, 1497, 1475, 1489, 1095,  822 } },
141*b2055c35SXin Li   { {  228, 1056, 1059, 1368,  752,  982, 1512, 1518,  987, 1782 },
142*b2055c35SXin Li     {  494,  514,  818,  942,  965,  892, 1610, 1356, 1048, 1363 },
143*b2055c35SXin Li     {  512,  648,  591, 1042,  761,  991, 1196, 1454, 1309, 1463 },
144*b2055c35SXin Li     {  683,  749, 1043,  676,  841, 1396, 1133, 1138,  654,  939 },
145*b2055c35SXin Li     {  622, 1101, 1126,  994,  361, 1077, 1203, 1318,  877, 1219 },
146*b2055c35SXin Li     {  631, 1068,  857, 1650,  651,  477, 1650, 1419,  828, 1170 },
147*b2055c35SXin Li     {  555,  727, 1068, 1335, 3127, 1339,  820, 1331, 1077,  429 },
148*b2055c35SXin Li     {  504,  879,  624, 1398,  889,  889, 1392,  808,  891, 1406 },
149*b2055c35SXin Li     {  683, 1602, 1289,  977,  578,  983, 1280, 1708,  406, 1122 },
150*b2055c35SXin Li     {  399,  865, 1433, 1070, 1072,  764,  968, 1477, 1223,  678 } },
151*b2055c35SXin Li   { {  333,  760,  935, 1638, 1010,  529, 1646, 1410, 1472, 2219 },
152*b2055c35SXin Li     {  512,  494,  750, 1160, 1215,  610, 1870, 1868, 1628, 1169 },
153*b2055c35SXin Li     {  572,  646,  492, 1934, 1208,  603, 1580, 1099, 1398, 1995 },
154*b2055c35SXin Li     {  786,  789,  942,  581, 1018,  951, 1599, 1207,  731,  768 },
155*b2055c35SXin Li     {  690, 1015,  672, 1078,  582,  504, 1693, 1438, 1108, 2897 },
156*b2055c35SXin Li     {  768, 1267,  571, 2005, 1243,  244, 2881, 1380, 1786, 1453 },
157*b2055c35SXin Li     {  452,  899, 1293,  903, 1311, 3100,  465, 1311, 1319,  813 },
158*b2055c35SXin Li     {  394,  927,  942, 1103, 1358, 1104,  946,  593, 1363, 1109 },
159*b2055c35SXin Li     {  559, 1005, 1007, 1016,  658, 1173, 1021, 1164,  623, 1028 },
160*b2055c35SXin Li     {  564,  796,  632, 1005, 1014,  863, 2316, 1268,  938,  764 } },
161*b2055c35SXin Li   { {  266,  606, 1098, 1228, 1497, 1243,  948, 1030, 1734, 1461 },
162*b2055c35SXin Li     {  366,  585,  901, 1060, 1407, 1247,  876, 1134, 1620, 1054 },
163*b2055c35SXin Li     {  452,  565,  542, 1729, 1479, 1479, 1016,  886, 2938, 1150 },
164*b2055c35SXin Li     {  555, 1088, 1533,  950, 1354,  895,  834, 1019, 1021,  496 },
165*b2055c35SXin Li     {  704,  815, 1193,  971,  973,  640, 1217, 2214,  832,  578 },
166*b2055c35SXin Li     {  672, 1245,  579,  871,  875,  774,  872, 1273, 1027,  949 },
167*b2055c35SXin Li     {  296, 1134, 2050, 1784, 1636, 3425,  442, 1550, 2076,  722 },
168*b2055c35SXin Li     {  342,  982, 1259, 1846, 1848, 1848,  622,  568, 1847, 1052 },
169*b2055c35SXin Li     {  555, 1064, 1304,  828,  746, 1343, 1075, 1329, 1078,  494 },
170*b2055c35SXin Li     {  288, 1167, 1285, 1174, 1639, 1639,  833, 2254, 1304,  509 } },
171*b2055c35SXin Li   { {  342,  719,  767, 1866, 1757, 1270, 1246,  550, 1746, 2151 },
172*b2055c35SXin Li     {  483,  653,  694, 1509, 1459, 1410, 1218,  507, 1914, 1266 },
173*b2055c35SXin Li     {  488,  757,  447, 2979, 1813, 1268, 1654,  539, 1849, 2109 },
174*b2055c35SXin Li     {  522, 1097, 1085,  851, 1365, 1111,  851,  901,  961,  605 },
175*b2055c35SXin Li     {  709,  716,  841,  728,  736,  945,  941,  862, 2845, 1057 },
176*b2055c35SXin Li     {  512, 1323,  500, 1336, 1083,  681, 1342,  717, 1604, 1350 },
177*b2055c35SXin Li     {  452, 1155, 1372, 1900, 1501, 3290,  311,  944, 1919,  922 },
178*b2055c35SXin Li     {  403, 1520,  977, 2132, 1733, 3522, 1076,  276, 3335, 1547 },
179*b2055c35SXin Li     {  559, 1374, 1101,  615,  673, 2462,  974,  795,  984,  984 },
180*b2055c35SXin Li     {  547, 1122, 1062,  812, 1410,  951, 1140,  622, 1268,  651 } },
181*b2055c35SXin Li   { {  165,  982, 1235,  938, 1334, 1366, 1659, 1578,  964, 1612 },
182*b2055c35SXin Li     {  592,  422,  925,  847, 1139, 1112, 1387, 2036,  861, 1041 },
183*b2055c35SXin Li     {  403,  837,  732,  770,  941, 1658, 1250,  809, 1407, 1407 },
184*b2055c35SXin Li     {  896,  874, 1071,  381, 1568, 1722, 1437, 2192,  480, 1035 },
185*b2055c35SXin Li     {  640, 1098, 1012, 1032,  684, 1382, 1581, 2106,  416,  865 },
186*b2055c35SXin Li     {  559, 1005,  819,  914,  710,  770, 1418,  920,  838, 1435 },
187*b2055c35SXin Li     {  415, 1258, 1245,  870, 1278, 3067,  770, 1021, 1287,  522 },
188*b2055c35SXin Li     {  406,  990,  601, 1009, 1265, 1265, 1267,  759, 1017, 1277 },
189*b2055c35SXin Li     {  968, 1182, 1329,  788, 1032, 1292, 1705, 1714,  203, 1403 },
190*b2055c35SXin Li     {  732,  877, 1279,  471,  901, 1161, 1545, 1294,  755,  755 } },
191*b2055c35SXin Li   { {  111,  931, 1378, 1185, 1933, 1648, 1148, 1714, 1873, 1307 },
192*b2055c35SXin Li     {  406,  414, 1030, 1023, 1910, 1404, 1313, 1647, 1509,  793 },
193*b2055c35SXin Li     {  342,  640,  575, 1088, 1241, 1349, 1161, 1350, 1756, 1502 },
194*b2055c35SXin Li     {  559,  766, 1185,  357, 1682, 1428, 1329, 1897, 1219,  802 },
195*b2055c35SXin Li     {  473,  909, 1164,  771,  719, 2508, 1427, 1432,  722,  782 },
196*b2055c35SXin Li     {  342,  892,  785, 1145, 1150,  794, 1296, 1550,  973, 1057 },
197*b2055c35SXin Li     {  208, 1036, 1326, 1343, 1606, 3395,  815, 1455, 1618,  712 },
198*b2055c35SXin Li     {  228,  928,  890, 1046, 3499, 1711,  994,  829, 1720, 1318 },
199*b2055c35SXin Li     {  768,  724, 1058,  636,  991, 1075, 1319, 1324,  616,  825 },
200*b2055c35SXin Li     {  305, 1167, 1358,  899, 1587, 1587,  987, 1988, 1332,  501 } }
201*b2055c35SXin Li };
202*b2055c35SXin Li 
203*b2055c35SXin Li //------------------------------------------------------------------------------
204*b2055c35SXin Li // helper functions for residuals struct VP8Residual.
205*b2055c35SXin Li 
VP8InitResidual(int first,int coeff_type,VP8Encoder * const enc,VP8Residual * const res)206*b2055c35SXin Li void VP8InitResidual(int first, int coeff_type,
207*b2055c35SXin Li                      VP8Encoder* const enc, VP8Residual* const res) {
208*b2055c35SXin Li   res->coeff_type = coeff_type;
209*b2055c35SXin Li   res->prob  = enc->proba_.coeffs_[coeff_type];
210*b2055c35SXin Li   res->stats = enc->proba_.stats_[coeff_type];
211*b2055c35SXin Li   res->costs = enc->proba_.remapped_costs_[coeff_type];
212*b2055c35SXin Li   res->first = first;
213*b2055c35SXin Li }
214*b2055c35SXin Li 
215*b2055c35SXin Li //------------------------------------------------------------------------------
216*b2055c35SXin Li // Mode costs
217*b2055c35SXin Li 
VP8GetCostLuma4(VP8EncIterator * const it,const int16_t levels[16])218*b2055c35SXin Li int VP8GetCostLuma4(VP8EncIterator* const it, const int16_t levels[16]) {
219*b2055c35SXin Li   const int x = (it->i4_ & 3), y = (it->i4_ >> 2);
220*b2055c35SXin Li   VP8Residual res;
221*b2055c35SXin Li   VP8Encoder* const enc = it->enc_;
222*b2055c35SXin Li   int R = 0;
223*b2055c35SXin Li   int ctx;
224*b2055c35SXin Li 
225*b2055c35SXin Li   VP8InitResidual(0, 3, enc, &res);
226*b2055c35SXin Li   ctx = it->top_nz_[x] + it->left_nz_[y];
227*b2055c35SXin Li   VP8SetResidualCoeffs(levels, &res);
228*b2055c35SXin Li   R += VP8GetResidualCost(ctx, &res);
229*b2055c35SXin Li   return R;
230*b2055c35SXin Li }
231*b2055c35SXin Li 
VP8GetCostLuma16(VP8EncIterator * const it,const VP8ModeScore * const rd)232*b2055c35SXin Li int VP8GetCostLuma16(VP8EncIterator* const it, const VP8ModeScore* const rd) {
233*b2055c35SXin Li   VP8Residual res;
234*b2055c35SXin Li   VP8Encoder* const enc = it->enc_;
235*b2055c35SXin Li   int x, y;
236*b2055c35SXin Li   int R = 0;
237*b2055c35SXin Li 
238*b2055c35SXin Li   VP8IteratorNzToBytes(it);   // re-import the non-zero context
239*b2055c35SXin Li 
240*b2055c35SXin Li   // DC
241*b2055c35SXin Li   VP8InitResidual(0, 1, enc, &res);
242*b2055c35SXin Li   VP8SetResidualCoeffs(rd->y_dc_levels, &res);
243*b2055c35SXin Li   R += VP8GetResidualCost(it->top_nz_[8] + it->left_nz_[8], &res);
244*b2055c35SXin Li 
245*b2055c35SXin Li   // AC
246*b2055c35SXin Li   VP8InitResidual(1, 0, enc, &res);
247*b2055c35SXin Li   for (y = 0; y < 4; ++y) {
248*b2055c35SXin Li     for (x = 0; x < 4; ++x) {
249*b2055c35SXin Li       const int ctx = it->top_nz_[x] + it->left_nz_[y];
250*b2055c35SXin Li       VP8SetResidualCoeffs(rd->y_ac_levels[x + y * 4], &res);
251*b2055c35SXin Li       R += VP8GetResidualCost(ctx, &res);
252*b2055c35SXin Li       it->top_nz_[x] = it->left_nz_[y] = (res.last >= 0);
253*b2055c35SXin Li     }
254*b2055c35SXin Li   }
255*b2055c35SXin Li   return R;
256*b2055c35SXin Li }
257*b2055c35SXin Li 
VP8GetCostUV(VP8EncIterator * const it,const VP8ModeScore * const rd)258*b2055c35SXin Li int VP8GetCostUV(VP8EncIterator* const it, const VP8ModeScore* const rd) {
259*b2055c35SXin Li   VP8Residual res;
260*b2055c35SXin Li   VP8Encoder* const enc = it->enc_;
261*b2055c35SXin Li   int ch, x, y;
262*b2055c35SXin Li   int R = 0;
263*b2055c35SXin Li 
264*b2055c35SXin Li   VP8IteratorNzToBytes(it);  // re-import the non-zero context
265*b2055c35SXin Li 
266*b2055c35SXin Li   VP8InitResidual(0, 2, enc, &res);
267*b2055c35SXin Li   for (ch = 0; ch <= 2; ch += 2) {
268*b2055c35SXin Li     for (y = 0; y < 2; ++y) {
269*b2055c35SXin Li       for (x = 0; x < 2; ++x) {
270*b2055c35SXin Li         const int ctx = it->top_nz_[4 + ch + x] + it->left_nz_[4 + ch + y];
271*b2055c35SXin Li         VP8SetResidualCoeffs(rd->uv_levels[ch * 2 + x + y * 2], &res);
272*b2055c35SXin Li         R += VP8GetResidualCost(ctx, &res);
273*b2055c35SXin Li         it->top_nz_[4 + ch + x] = it->left_nz_[4 + ch + y] = (res.last >= 0);
274*b2055c35SXin Li       }
275*b2055c35SXin Li     }
276*b2055c35SXin Li   }
277*b2055c35SXin Li   return R;
278*b2055c35SXin Li }
279*b2055c35SXin Li 
280*b2055c35SXin Li 
281*b2055c35SXin Li //------------------------------------------------------------------------------
282*b2055c35SXin Li // Recording of token probabilities.
283*b2055c35SXin Li 
284*b2055c35SXin Li // We keep the table-free variant around for reference, in case.
285*b2055c35SXin Li #define USE_LEVEL_CODE_TABLE
286*b2055c35SXin Li 
287*b2055c35SXin Li // Simulate block coding, but only record statistics.
288*b2055c35SXin Li // Note: no need to record the fixed probas.
VP8RecordCoeffs(int ctx,const VP8Residual * const res)289*b2055c35SXin Li int VP8RecordCoeffs(int ctx, const VP8Residual* const res) {
290*b2055c35SXin Li   int n = res->first;
291*b2055c35SXin Li   // should be stats[VP8EncBands[n]], but it's equivalent for n=0 or 1
292*b2055c35SXin Li   proba_t* s = res->stats[n][ctx];
293*b2055c35SXin Li   if (res->last  < 0) {
294*b2055c35SXin Li     VP8RecordStats(0, s + 0);
295*b2055c35SXin Li     return 0;
296*b2055c35SXin Li   }
297*b2055c35SXin Li   while (n <= res->last) {
298*b2055c35SXin Li     int v;
299*b2055c35SXin Li     VP8RecordStats(1, s + 0);  // order of record doesn't matter
300*b2055c35SXin Li     while ((v = res->coeffs[n++]) == 0) {
301*b2055c35SXin Li       VP8RecordStats(0, s + 1);
302*b2055c35SXin Li       s = res->stats[VP8EncBands[n]][0];
303*b2055c35SXin Li     }
304*b2055c35SXin Li     VP8RecordStats(1, s + 1);
305*b2055c35SXin Li     if (!VP8RecordStats(2u < (unsigned int)(v + 1), s + 2)) {  // v = -1 or 1
306*b2055c35SXin Li       s = res->stats[VP8EncBands[n]][1];
307*b2055c35SXin Li     } else {
308*b2055c35SXin Li       v = abs(v);
309*b2055c35SXin Li #if !defined(USE_LEVEL_CODE_TABLE)
310*b2055c35SXin Li       if (!VP8RecordStats(v > 4, s + 3)) {
311*b2055c35SXin Li         if (VP8RecordStats(v != 2, s + 4))
312*b2055c35SXin Li           VP8RecordStats(v == 4, s + 5);
313*b2055c35SXin Li       } else if (!VP8RecordStats(v > 10, s + 6)) {
314*b2055c35SXin Li         VP8RecordStats(v > 6, s + 7);
315*b2055c35SXin Li       } else if (!VP8RecordStats((v >= 3 + (8 << 2)), s + 8)) {
316*b2055c35SXin Li         VP8RecordStats((v >= 3 + (8 << 1)), s + 9);
317*b2055c35SXin Li       } else {
318*b2055c35SXin Li         VP8RecordStats((v >= 3 + (8 << 3)), s + 10);
319*b2055c35SXin Li       }
320*b2055c35SXin Li #else
321*b2055c35SXin Li       if (v > MAX_VARIABLE_LEVEL) {
322*b2055c35SXin Li         v = MAX_VARIABLE_LEVEL;
323*b2055c35SXin Li       }
324*b2055c35SXin Li 
325*b2055c35SXin Li       {
326*b2055c35SXin Li         const int bits = VP8LevelCodes[v - 1][1];
327*b2055c35SXin Li         int pattern = VP8LevelCodes[v - 1][0];
328*b2055c35SXin Li         int i;
329*b2055c35SXin Li         for (i = 0; (pattern >>= 1) != 0; ++i) {
330*b2055c35SXin Li           const int mask = 2 << i;
331*b2055c35SXin Li           if (pattern & 1) VP8RecordStats(!!(bits & mask), s + 3 + i);
332*b2055c35SXin Li         }
333*b2055c35SXin Li       }
334*b2055c35SXin Li #endif
335*b2055c35SXin Li       s = res->stats[VP8EncBands[n]][2];
336*b2055c35SXin Li     }
337*b2055c35SXin Li   }
338*b2055c35SXin Li   if (n < 16) VP8RecordStats(0, s + 0);
339*b2055c35SXin Li   return 1;
340*b2055c35SXin Li }
341*b2055c35SXin Li 
342*b2055c35SXin Li //------------------------------------------------------------------------------
343