blob: a2b3c6dabd10619d61413fe2d0c7bc13f9bf5dad [file] [log] [blame]
Angie Chiang80b82262017-02-24 11:39:47 -08001/*
2 * Copyright (c) 2017, Alliance for Open Media. All rights reserved
3 *
4 * This source code is subject to the terms of the BSD 2 Clause License and
5 * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6 * was not distributed with this source code in the LICENSE file, you can
7 * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8 * Media Patent License 1.0 was not distributed with this source code in the
9 * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10 */
11
Linfeng Zhangae7b2f32017-11-08 15:46:57 -080012#include "aom_ports/mem.h"
Angie Chiang80b82262017-02-24 11:39:47 -080013#include "av1/common/scan.h"
Angie Chiang0397eda2017-03-15 16:57:14 -070014#include "av1/common/blockd.h"
Angie Chiange50f3ec2017-04-10 15:50:33 -070015#include "av1/common/idct.h"
Angie Chiang0397eda2017-03-15 16:57:14 -070016#include "av1/common/pred_common.h"
Angie Chiang1628fcc2017-04-13 16:30:30 -070017#include "av1/encoder/bitstream.h"
18#include "av1/encoder/encodeframe.h"
Angie Chiang47c72182017-02-27 14:30:38 -080019#include "av1/encoder/cost.h"
Angie Chiang80b82262017-02-24 11:39:47 -080020#include "av1/encoder/encodetxb.h"
Angie Chiang808d8592017-04-06 18:36:55 -070021#include "av1/encoder/rdopt.h"
Angie Chiang800df032017-03-22 11:14:12 -070022#include "av1/encoder/subexp.h"
Angie Chiang0397eda2017-03-15 16:57:14 -070023#include "av1/encoder/tokenize.h"
Angie Chiang80b82262017-02-24 11:39:47 -080024
Angie Chiang47e07072017-05-30 17:27:01 -070025#define TEST_OPTIMIZE_TXB 0
26
Dake Hea47cd6c2017-10-13 18:09:58 -070027typedef struct LevelDownStats {
28 int update;
29 tran_low_t low_qc;
30 tran_low_t low_dqc;
Dake Hea47cd6c2017-10-13 18:09:58 -070031 int64_t dist0;
32 int rate;
33 int rate_low;
34 int64_t dist;
35 int64_t dist_low;
36 int64_t rd;
37 int64_t rd_low;
38 int nz_rate; // for eob
Dake Hea47cd6c2017-10-13 18:09:58 -070039 int64_t rd_diff;
40 int cost_diff;
41 int64_t dist_diff;
42 int new_eob;
43} LevelDownStats;
44
Angie Chiangf0fbf9d2017-03-15 15:01:22 -070045void av1_alloc_txb_buf(AV1_COMP *cpi) {
Angie Chiangc484abe2017-03-20 15:43:11 -070046#if 0
Angie Chiangf0fbf9d2017-03-15 15:01:22 -070047 AV1_COMMON *cm = &cpi->common;
48 int mi_block_size = 1 << MI_SIZE_LOG2;
49 // TODO(angiebird): Make sure cm->subsampling_x/y is set correctly, and then
50 // use precise buffer size according to cm->subsampling_x/y
51 int pixel_stride = mi_block_size * cm->mi_cols;
52 int pixel_height = mi_block_size * cm->mi_rows;
53 int i;
54 for (i = 0; i < MAX_MB_PLANE; ++i) {
55 CHECK_MEM_ERROR(
56 cm, cpi->tcoeff_buf[i],
57 aom_malloc(sizeof(*cpi->tcoeff_buf[i]) * pixel_stride * pixel_height));
58 }
Angie Chiangc484abe2017-03-20 15:43:11 -070059#else
Jingning Hanf5a4d3b2017-08-27 23:01:19 -070060 AV1_COMMON *cm = &cpi->common;
Dominic Symes917d6c02017-10-11 18:00:52 +020061 int size = ((cm->mi_rows >> cm->mib_size_log2) + 1) *
62 ((cm->mi_cols >> cm->mib_size_log2) + 1);
Jingning Hanf5a4d3b2017-08-27 23:01:19 -070063
Angie Chiang9367e3e2017-10-02 16:28:11 -070064 av1_free_txb_buf(cpi);
Jingning Hanf5a4d3b2017-08-27 23:01:19 -070065 // TODO(jingning): This should be further reduced.
66 CHECK_MEM_ERROR(cm, cpi->coeff_buffer_base,
67 aom_malloc(sizeof(*cpi->coeff_buffer_base) * size));
Angie Chiangc484abe2017-03-20 15:43:11 -070068#endif
Angie Chiangf0fbf9d2017-03-15 15:01:22 -070069}
70
71void av1_free_txb_buf(AV1_COMP *cpi) {
Angie Chiangc484abe2017-03-20 15:43:11 -070072#if 0
Angie Chiangf0fbf9d2017-03-15 15:01:22 -070073 int i;
74 for (i = 0; i < MAX_MB_PLANE; ++i) {
75 aom_free(cpi->tcoeff_buf[i]);
76 }
Angie Chiangc484abe2017-03-20 15:43:11 -070077#else
Jingning Hanf5a4d3b2017-08-27 23:01:19 -070078 aom_free(cpi->coeff_buffer_base);
Angie Chiangc484abe2017-03-20 15:43:11 -070079#endif
Angie Chiangf0fbf9d2017-03-15 15:01:22 -070080}
81
Jingning Hanf5a4d3b2017-08-27 23:01:19 -070082void av1_set_coeff_buffer(const AV1_COMP *const cpi, MACROBLOCK *const x,
83 int mi_row, int mi_col) {
Dominic Symes917d6c02017-10-11 18:00:52 +020084 int mib_size_log2 = cpi->common.mib_size_log2;
85 int stride = (cpi->common.mi_cols >> mib_size_log2) + 1;
86 int offset = (mi_row >> mib_size_log2) * stride + (mi_col >> mib_size_log2);
Jingning Hanf5a4d3b2017-08-27 23:01:19 -070087 CB_COEFF_BUFFER *coeff_buf = &cpi->coeff_buffer_base[offset];
88 const int txb_offset = x->cb_offset / (TX_SIZE_W_MIN * TX_SIZE_H_MIN);
89 for (int plane = 0; plane < MAX_MB_PLANE; ++plane) {
90 x->mbmi_ext->tcoeff[plane] = coeff_buf->tcoeff[plane] + x->cb_offset;
91 x->mbmi_ext->eobs[plane] = coeff_buf->eobs[plane] + txb_offset;
92 x->mbmi_ext->txb_skip_ctx[plane] =
93 coeff_buf->txb_skip_ctx[plane] + txb_offset;
94 x->mbmi_ext->dc_sign_ctx[plane] =
95 coeff_buf->dc_sign_ctx[plane] + txb_offset;
96 }
97}
98
Angie Chiang80b82262017-02-24 11:39:47 -080099static void write_golomb(aom_writer *w, int level) {
100 int x = level + 1;
101 int i = x;
102 int length = 0;
103
104 while (i) {
105 i >>= 1;
106 ++length;
107 }
108 assert(length > 0);
109
110 for (i = 0; i < length - 1; ++i) aom_write_bit(w, 0);
111
112 for (i = length - 1; i >= 0; --i) aom_write_bit(w, (x >> i) & 0x01);
113}
114
Dake Hea47cd6c2017-10-13 18:09:58 -0700115static INLINE tran_low_t get_lower_coeff(tran_low_t qc) {
116 if (qc == 0) {
117 return 0;
118 }
119 return qc > 0 ? qc - 1 : qc + 1;
120}
121
122static INLINE tran_low_t qcoeff_to_dqcoeff(tran_low_t qc, int dqv, int shift) {
123 int sgn = qc < 0 ? -1 : 1;
124 return sgn * ((abs(qc) * dqv) >> shift);
125}
126
127static INLINE int64_t get_coeff_dist(tran_low_t tcoeff, tran_low_t dqcoeff,
128 int shift) {
Monty Montgomery4a05a582017-11-01 21:21:07 -0400129#if CONFIG_DAALA_TX
130 int depth_shift = (TX_COEFF_DEPTH - 11) * 2;
131 int depth_round = depth_shift > 1 ? (1 << (depth_shift - 1)) : 0;
132 const int64_t diff = tcoeff - dqcoeff;
133 const int64_t error = diff * diff + depth_round >> depth_shift;
134 (void)shift;
135#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700136 const int64_t diff = (tcoeff - dqcoeff) * (1 << shift);
137 const int64_t error = diff * diff;
Monty Montgomery4a05a582017-11-01 21:21:07 -0400138#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700139 return error;
140}
141
Jingning Han35deaa72017-10-26 15:36:30 -0700142void av1_update_eob_context(int eob, int seg_eob, TX_SIZE tx_size,
143 TX_TYPE tx_type, PLANE_TYPE plane,
Yunqing Wang0e141b52017-11-02 15:08:58 -0700144 FRAME_CONTEXT *ec_ctx, FRAME_COUNTS *counts,
145 uint8_t allow_update_cdf) {
Dake Hea47cd6c2017-10-13 18:09:58 -0700146 int16_t eob_extra;
147 int16_t eob_pt = get_eob_pos_token(eob, &eob_extra);
148 int16_t dummy;
149 int16_t max_eob_pt = get_eob_pos_token(seg_eob, &dummy);
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -0800150 TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Dake Hea47cd6c2017-10-13 18:09:58 -0700151
152 for (int i = 1; i < max_eob_pt; i++) {
Jingning Han35deaa72017-10-26 15:36:30 -0700153 int eob_pos_ctx = av1_get_eob_pos_ctx(tx_type, i);
154 counts->eob_flag[txs_ctx][plane][eob_pos_ctx][eob_pt == i]++;
Yunqing Wang0e141b52017-11-02 15:08:58 -0700155 if (allow_update_cdf)
156 update_cdf(ec_ctx->eob_flag_cdf[txs_ctx][plane][eob_pos_ctx], eob_pt == i,
157 2);
Dake Hea47cd6c2017-10-13 18:09:58 -0700158 if (eob_pt == i) {
159 break;
160 }
161 }
Jingning Han00803a72017-10-25 16:04:34 -0700162
Angie Chiang7ab884e2017-10-18 15:57:12 -0700163 if (k_eob_offset_bits[eob_pt] > 0) {
164 int eob_shift = k_eob_offset_bits[eob_pt] - 1;
165 int bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
Jingning Han35deaa72017-10-26 15:36:30 -0700166 counts->eob_extra[txs_ctx][plane][eob_pt][bit]++;
Yunqing Wang0e141b52017-11-02 15:08:58 -0700167 if (allow_update_cdf)
168 update_cdf(ec_ctx->eob_extra_cdf[txs_ctx][plane][eob_pt], bit, 2);
Angie Chiang7ab884e2017-10-18 15:57:12 -0700169 }
Dake Hea47cd6c2017-10-13 18:09:58 -0700170}
171
172static int get_eob_cost(int eob, int seg_eob,
Jingning Han35deaa72017-10-26 15:36:30 -0700173 const LV_MAP_COEFF_COST *txb_costs, TX_TYPE tx_type) {
Dake Hea47cd6c2017-10-13 18:09:58 -0700174 int16_t eob_extra;
175 int16_t eob_pt = get_eob_pos_token(eob, &eob_extra);
176 int16_t dummy;
177 int16_t max_eob_pt = get_eob_pos_token(seg_eob, &dummy);
178 int eob_cost = 0;
179
180 // printf("Enc: [%d, %d], (%d, %d) ", seg_eob, eob, eob_pt, eob_extra);
181 for (int i = 1; i < max_eob_pt; i++) {
Jingning Han35deaa72017-10-26 15:36:30 -0700182 int eob_pos_ctx = av1_get_eob_pos_ctx(tx_type, i);
Dake Hea47cd6c2017-10-13 18:09:58 -0700183 eob_cost += txb_costs->eob_cost[eob_pos_ctx][eob_pt == i];
184 if (eob_pt == i) {
185 break;
186 }
187 }
188 if (k_eob_offset_bits[eob_pt] > 0) {
Angie Chiang7ab884e2017-10-18 15:57:12 -0700189 int eob_shift = k_eob_offset_bits[eob_pt] - 1;
190 int bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
191 eob_cost += txb_costs->eob_extra_cost[eob_pt][bit];
192 for (int i = 1; i < k_eob_offset_bits[eob_pt]; i++) {
193 eob_shift = k_eob_offset_bits[eob_pt] - 1 - i;
194 bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
Dake Hea47cd6c2017-10-13 18:09:58 -0700195 eob_cost += av1_cost_bit(128, bit);
196 }
197 }
198 return eob_cost;
199}
200
Linfeng Zhang1015a342017-10-24 16:20:41 -0700201static int get_coeff_cost(const tran_low_t qc, const int scan_idx,
Ola Hugosson13892102017-11-06 08:01:44 +0100202#if CONFIG_LV_MAP_MULTI
203 const int is_eob,
204#endif
Linfeng Zhang1015a342017-10-24 16:20:41 -0700205 const TxbInfo *const txb_info,
206 const LV_MAP_COEFF_COST *const txb_costs);
Dake Hea47cd6c2017-10-13 18:09:58 -0700207
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800208static void get_dist_cost_stats(LevelDownStats *stats, int scan_idx,
Ola Hugosson13892102017-11-06 08:01:44 +0100209#if CONFIG_LV_MAP_MULTI
210 const int is_eob,
211#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800212 const LV_MAP_COEFF_COST *txb_costs,
213 TxbInfo *txb_info) {
Dake Hea47cd6c2017-10-13 18:09:58 -0700214 const int16_t *scan = txb_info->scan_order->scan;
215 const int coeff_idx = scan[scan_idx];
216 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
Linfeng Zhang1015a342017-10-24 16:20:41 -0700217 const uint8_t *const levels = txb_info->levels;
Dake Hea47cd6c2017-10-13 18:09:58 -0700218 stats->new_eob = -1;
219 stats->update = 0;
220
221 const tran_low_t tqc = txb_info->tcoeff[coeff_idx];
222 const int dqv = txb_info->dequant[coeff_idx != 0];
223
224 const tran_low_t dqc = qcoeff_to_dqcoeff(qc, dqv, txb_info->shift);
225 const int64_t dqc_dist = get_coeff_dist(tqc, dqc, txb_info->shift);
Ola Hugosson13892102017-11-06 08:01:44 +0100226#if CONFIG_LV_MAP_MULTI
227 const int qc_cost = get_coeff_cost(qc, scan_idx, is_eob, txb_info, txb_costs);
228#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700229 const int qc_cost = get_coeff_cost(qc, scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +0100230#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700231
232 // distortion difference when coefficient is quantized to 0
233 const tran_low_t dqc0 = qcoeff_to_dqcoeff(0, dqv, txb_info->shift);
234 stats->dist0 = get_coeff_dist(tqc, dqc0, txb_info->shift);
235 stats->dist = dqc_dist - stats->dist0;
236 stats->rate = qc_cost;
237
238 if (qc == 0) {
239 return;
240 }
241 stats->rd = RDCOST(txb_info->rdmult, stats->rate, stats->dist);
242
243 stats->low_qc = get_lower_coeff(qc);
244 stats->low_dqc = qcoeff_to_dqcoeff(stats->low_qc, dqv, txb_info->shift);
245 const int64_t low_dqc_dist =
246 get_coeff_dist(tqc, stats->low_dqc, txb_info->shift);
Ola Hugosson13892102017-11-06 08:01:44 +0100247#if CONFIG_LV_MAP_MULTI
248 const int low_qc_cost =
249 get_coeff_cost(stats->low_qc, scan_idx, is_eob, txb_info, txb_costs);
250#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700251 const int low_qc_cost =
252 get_coeff_cost(stats->low_qc, scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +0100253#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700254 stats->dist_low = low_dqc_dist - stats->dist0;
255 stats->rate_low = low_qc_cost;
256 stats->rd_low = RDCOST(txb_info->rdmult, stats->rate_low, stats->dist_low);
257
Ola Hugosson13892102017-11-06 08:01:44 +0100258#if CONFIG_LV_MAP_MULTI
Ola Hugossone5a9b382017-11-16 17:02:28 +0100259 (void)levels;
Ola Hugosson13892102017-11-06 08:01:44 +0100260 if ((stats->rd_low < stats->rd) && (stats->low_qc == 0)) {
Ola Hugossone5a9b382017-11-16 17:02:28 +0100261 stats->nz_rate = low_qc_cost;
Ola Hugosson13892102017-11-06 08:01:44 +0100262 } else {
Ola Hugossone5a9b382017-11-16 17:02:28 +0100263 if (stats->rd_low < stats->rd) {
264 const int low_qc_eob_cost =
265 get_coeff_cost(stats->low_qc, scan_idx, 1, txb_info, txb_costs);
266 stats->nz_rate = low_qc_cost - low_qc_eob_cost;
267 } else {
268 const int qc_eob_cost =
269 get_coeff_cost(qc, scan_idx, 1, txb_info, txb_costs);
270 stats->nz_rate = qc_cost - qc_eob_cost;
271 }
Ola Hugosson13892102017-11-06 08:01:44 +0100272 }
273#else
Linfeng Zhang1015a342017-10-24 16:20:41 -0700274 int coeff_ctx = get_nz_map_ctx(levels, scan_idx, scan, txb_info->bwl,
Linfeng Zhangb6957c22017-10-25 13:17:28 -0700275 txb_info->height, txb_info->tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -0700276 if ((stats->rd_low < stats->rd) && (stats->low_qc == 0)) {
277 stats->nz_rate = txb_costs->nz_map_cost[coeff_ctx][0];
278 } else {
279 stats->nz_rate = txb_costs->nz_map_cost[coeff_ctx][1];
280 }
Ola Hugosson13892102017-11-06 08:01:44 +0100281#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700282}
283
Linfeng Zhang1015a342017-10-24 16:20:41 -0700284static INLINE void update_qcoeff(const int coeff_idx, const tran_low_t qc,
285 const TxbInfo *const txb_info) {
Dake Hea47cd6c2017-10-13 18:09:58 -0700286 txb_info->qcoeff[coeff_idx] = qc;
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700287 txb_info->levels[get_paded_idx(coeff_idx, txb_info->bwl)] =
Jingning Han5cb408e2017-11-17 14:43:39 -0800288 (uint8_t)clamp(abs(qc), 0, INT8_MAX);
Linfeng Zhang1015a342017-10-24 16:20:41 -0700289}
290
291static INLINE void update_coeff(const int coeff_idx, const tran_low_t qc,
292 const TxbInfo *const txb_info) {
293 update_qcoeff(coeff_idx, qc, txb_info);
Dake Hea47cd6c2017-10-13 18:09:58 -0700294 const int dqv = txb_info->dequant[coeff_idx != 0];
295 txb_info->dqcoeff[coeff_idx] = qcoeff_to_dqcoeff(qc, dqv, txb_info->shift);
296}
297
Linfeng Zhang1015a342017-10-24 16:20:41 -0700298static INLINE void av1_txb_init_levels(const tran_low_t *const coeff,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700299 const int width, const int height,
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700300 uint8_t *const levels) {
301 const int stride = width + TX_PAD_HOR;
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700302 uint8_t *ls = levels;
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700303
304 memset(levels - TX_PAD_TOP * stride, 0,
305 sizeof(*levels) * TX_PAD_TOP * stride);
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700306 memset(levels + stride * height, 0,
307 sizeof(*levels) * (TX_PAD_BOTTOM * stride + TX_PAD_END));
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700308
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700309 for (int i = 0; i < height; i++) {
310 for (int j = 0; j < width; j++) {
Jingning Han5cb408e2017-11-17 14:43:39 -0800311 *ls++ = (uint8_t)clamp(abs(coeff[i * width + j]), 0, INT8_MAX);
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700312 }
313 for (int j = 0; j < TX_PAD_HOR; j++) {
314 *ls++ = 0;
315 }
Linfeng Zhang1015a342017-10-24 16:20:41 -0700316 }
317}
318
Angie Chiang80b82262017-02-24 11:39:47 -0800319void av1_write_coeffs_txb(const AV1_COMMON *const cm, MACROBLOCKD *xd,
Jingning Han7eab9ff2017-07-06 10:12:54 -0700320 aom_writer *w, int blk_row, int blk_col, int block,
321 int plane, TX_SIZE tx_size, const tran_low_t *tcoeff,
322 uint16_t eob, TXB_CTX *txb_ctx) {
Angie Chiang80b82262017-02-24 11:39:47 -0800323 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
Angie Chiangcea11f22017-02-24 12:30:40 -0800324 const PLANE_TYPE plane_type = get_plane_type(plane);
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -0800325 const TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Jingning Han19b5c8f2017-07-06 15:10:12 -0700326 const TX_TYPE tx_type =
327 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiangbd99b382017-06-20 15:11:16 -0700328 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Angie Chiang80b82262017-02-24 11:39:47 -0800329 const int16_t *scan = scan_order->scan;
Urvang Joshi80893152017-10-27 11:51:14 -0700330 const int seg_eob = av1_get_max_eob(tx_size);
Angie Chiang80b82262017-02-24 11:39:47 -0800331 int c;
Angie Chianga9ba58e2017-12-01 19:22:43 -0800332 const int bwl = get_txb_bwl(tx_size);
333 const int width = get_txb_wide(tx_size);
334 const int height = get_txb_high(tx_size);
Linfeng Zhang848f7bc2017-10-31 15:26:07 -0700335 int update_eob = -1;
Jingning Han41c7f442017-09-05 14:54:00 -0700336 FRAME_CONTEXT *ec_ctx = xd->tile_ctx;
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700337 uint8_t levels_buf[TX_PAD_2D];
338 uint8_t *const levels = set_levels(levels_buf, width);
Linfeng Zhangae7b2f32017-11-08 15:46:57 -0800339 DECLARE_ALIGNED(16, uint8_t, level_counts[MAX_TX_SQUARE]);
Angie Chiang80b82262017-02-24 11:39:47 -0800340
Jingning Han7eab9ff2017-07-06 10:12:54 -0700341 (void)blk_row;
342 (void)blk_col;
Jingning Han94cea4a2017-09-30 14:13:23 -0700343 aom_write_bin(w, eob == 0,
344 ec_ctx->txb_skip_cdf[txs_ctx][txb_ctx->txb_skip_ctx], 2);
Angie Chiang80b82262017-02-24 11:39:47 -0800345 if (eob == 0) return;
Linfeng Zhangce065ca2017-10-17 16:49:30 -0700346
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700347 av1_txb_init_levels(tcoeff, width, height, levels);
Linfeng Zhangce065ca2017-10-17 16:49:30 -0700348
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700349#if CONFIG_TXK_SEL
Jingning Han19b5c8f2017-07-06 15:10:12 -0700350 av1_write_tx_type(cm, xd, blk_row, blk_col, block, plane,
351 get_min_tx_size(tx_size), w);
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700352#endif
Angie Chiang80b82262017-02-24 11:39:47 -0800353
Dake Hea47cd6c2017-10-13 18:09:58 -0700354 int16_t eob_extra;
355 int16_t eob_pt = get_eob_pos_token(eob, &eob_extra);
356 int16_t dummy;
357 int16_t max_eob_pt = get_eob_pos_token(seg_eob, &dummy);
358
359 // printf("Enc: [%d, %d], (%d, %d) ", seg_eob, eob, eob_pt, eob_extra);
360 for (int i = 1; i < max_eob_pt; i++) {
Jingning Han35deaa72017-10-26 15:36:30 -0700361 int eob_pos_ctx = av1_get_eob_pos_ctx(tx_type, i);
Dake Hea47cd6c2017-10-13 18:09:58 -0700362
363 aom_write_bin(w, eob_pt == i,
364 ec_ctx->eob_flag_cdf[txs_ctx][plane_type][eob_pos_ctx], 2);
365 // aom_write_symbol(w, eob_pt == i,
366 // ec_ctx->eob_flag_cdf[AOMMIN(txs_ctx,3)][plane_type][eob_pos_ctx], 2);
367 if (eob_pt == i) {
368 break;
369 }
370 }
371
372 if (k_eob_offset_bits[eob_pt] > 0) {
Angie Chiang7ab884e2017-10-18 15:57:12 -0700373 int eob_shift = k_eob_offset_bits[eob_pt] - 1;
374 int bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
375 aom_write_bin(w, bit, ec_ctx->eob_extra_cdf[txs_ctx][plane_type][eob_pt],
376 2);
377 for (int i = 1; i < k_eob_offset_bits[eob_pt]; i++) {
378 eob_shift = k_eob_offset_bits[eob_pt] - 1 - i;
379 bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
Dake Hea47cd6c2017-10-13 18:09:58 -0700380 aom_write_bit(w, bit);
381 // printf("%d ", bit);
382 }
383 }
Dake He03a32922017-10-31 08:06:45 -0700384
385#if USE_CAUSAL_BASE_CTX
386 int coeff_ctx = 0;
387 for (int i = 0; i < eob; ++i) {
388 c = eob - 1 - i;
Dake He03a32922017-10-31 08:06:45 -0700389
Ola Hugosson13892102017-11-06 08:01:44 +0100390#if CONFIG_LV_MAP_MULTI
391 coeff_ctx =
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800392 get_nz_map_ctx(levels, c, scan, bwl, height, c == eob - 1, tx_type);
Ola Hugosson13892102017-11-06 08:01:44 +0100393 tran_low_t v = tcoeff[scan[c]];
Dake He3fe369c2017-11-16 17:56:44 -0800394#if USE_BASE_EOB_ALPHABET
395 if (c == eob - 1) {
396 aom_write_symbol(
397 w, AOMMIN(abs(v), 3) - 1,
398 ec_ctx->coeff_base_eob_cdf[txs_ctx][plane_type]
399 [coeff_ctx - SIG_COEF_CONTEXTS +
400 SIG_COEF_CONTEXTS_EOB],
401 3);
402 } else {
403 aom_write_symbol(w, AOMMIN(abs(v), 3),
404 ec_ctx->coeff_base_cdf[txs_ctx][plane_type][coeff_ctx],
405 4);
406 }
407#else
Thomas Davies736ddef2017-11-09 09:46:08 +0000408 aom_write_symbol(w, AOMMIN(abs(v), 3),
409 ec_ctx->coeff_base_cdf[txs_ctx][plane_type][coeff_ctx], 4);
Dake He3fe369c2017-11-16 17:56:44 -0800410#endif
Ola Hugosson13892102017-11-06 08:01:44 +0100411#else
412 coeff_ctx = get_nz_map_ctx(levels, c, scan, bwl, height, tx_type);
Dake He03a32922017-10-31 08:06:45 -0700413 tran_low_t v = tcoeff[scan[c]];
414 int is_nz = (v != 0);
415
416 if (c < eob - 1) {
417 aom_write_bin(w, is_nz,
418 ec_ctx->nz_map_cdf[txs_ctx][plane_type][coeff_ctx], 2);
419 }
420 if (is_nz) {
421 const int level = abs(v);
422 int k;
423 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
424 int is_k = (level > (k + 1));
425 int ctx = coeff_ctx;
426 aom_write_bin(w, is_k,
427 ec_ctx->coeff_base_cdf[txs_ctx][plane_type][k][ctx], 2);
428 if (is_k == 0) break;
429 }
430 }
Ola Hugosson13892102017-11-06 08:01:44 +0100431#endif
Dake He03a32922017-10-31 08:06:45 -0700432 }
433 update_eob = eob - 1;
434#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700435 for (int i = 1; i < eob; ++i) {
436 c = eob - 1 - i;
Linfeng Zhangb6957c22017-10-25 13:17:28 -0700437 int coeff_ctx = get_nz_map_ctx(levels, c, scan, bwl, height, tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -0700438
439 tran_low_t v = tcoeff[scan[c]];
440 int is_nz = (v != 0);
441
Dake Hea47cd6c2017-10-13 18:09:58 -0700442 aom_write_bin(w, is_nz, ec_ctx->nz_map_cdf[txs_ctx][plane_type][coeff_ctx],
443 2);
Dake Hea47cd6c2017-10-13 18:09:58 -0700444 }
445
Dake Hea47cd6c2017-10-13 18:09:58 -0700446 for (int i = 0; i < NUM_BASE_LEVELS; ++i) {
Linfeng Zhangf72e1382017-11-01 13:59:41 -0700447 av1_get_base_level_counts(levels, i, width, height, level_counts);
Angie Chiang80b82262017-02-24 11:39:47 -0800448 for (c = eob - 1; c >= 0; --c) {
Linfeng Zhang1015a342017-10-24 16:20:41 -0700449 const tran_low_t level = abs(tcoeff[scan[c]]);
Angie Chiang80b82262017-02-24 11:39:47 -0800450 int ctx;
451
452 if (level <= i) continue;
453
Linfeng Zhangf72e1382017-11-01 13:59:41 -0700454 ctx = get_base_ctx(levels, scan[c], bwl, i, level_counts[scan[c]]);
Angie Chiang80b82262017-02-24 11:39:47 -0800455
456 if (level == i + 1) {
Jingning Han94cea4a2017-09-30 14:13:23 -0700457 aom_write_bin(w, 1, ec_ctx->coeff_base_cdf[txs_ctx][plane_type][i][ctx],
458 2);
Angie Chiang80b82262017-02-24 11:39:47 -0800459 continue;
460 }
Jingning Han24b0cf92017-08-18 22:50:18 -0700461
Jingning Han94cea4a2017-09-30 14:13:23 -0700462 aom_write_bin(w, 0, ec_ctx->coeff_base_cdf[txs_ctx][plane_type][i][ctx],
463 2);
Angie Chiang80b82262017-02-24 11:39:47 -0800464 update_eob = AOMMAX(update_eob, c);
465 }
466 }
Dake He03a32922017-10-31 08:06:45 -0700467#endif
Angie Chiang80b82262017-02-24 11:39:47 -0800468
Dake He43edb762017-10-26 10:29:46 -0700469 // Loop to code all signs in the transform block,
470 // starting with the sign of DC (if applicable)
471 for (c = 0; c < eob; ++c) {
Linfeng Zhang1015a342017-10-24 16:20:41 -0700472 const tran_low_t v = tcoeff[scan[c]];
473 const tran_low_t level = abs(v);
474 const int sign = (v < 0) ? 1 : 0;
Dake He43edb762017-10-26 10:29:46 -0700475 if (level == 0) continue;
476
477 if (c == 0) {
478#if LV_MAP_PROB
479 aom_write_bin(w, sign,
480 ec_ctx->dc_sign_cdf[plane_type][txb_ctx->dc_sign_ctx], 2);
481#else
482 aom_write(w, sign, ec_ctx->dc_sign[plane_type][txb_ctx->dc_sign_ctx]);
483#endif
484 } else {
485 aom_write_bit(w, sign);
486 }
487 }
488
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800489 if (update_eob >= 0) {
Dake He7d01ab52017-11-24 17:53:28 -0800490#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800491 av1_get_br_level_counts(levels, width, height, level_counts);
Dake He7d01ab52017-11-24 17:53:28 -0800492#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800493 for (c = update_eob; c >= 0; --c) {
494 const tran_low_t level = abs(tcoeff[scan[c]]);
495 int idx;
496 int ctx;
Angie Chiang80b82262017-02-24 11:39:47 -0800497
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800498 if (level <= NUM_BASE_LEVELS) continue;
Angie Chiang80b82262017-02-24 11:39:47 -0800499
Dake He7d01ab52017-11-24 17:53:28 -0800500// level is above 1.
501#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800502 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]]);
Dake He7d01ab52017-11-24 17:53:28 -0800503#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800504 int base_range = level - 1 - NUM_BASE_LEVELS;
Ola Hugossone72a2092017-11-12 09:11:53 +0100505#if CONFIG_LV_MAP_MULTI
Dake He7d01ab52017-11-24 17:53:28 -0800506#if USE_CAUSAL_BR_CTX
507 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]], tx_type);
508
509#else
510 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]]);
511#endif
Ola Hugossone72a2092017-11-12 09:11:53 +0100512 for (idx = 0; idx < COEFF_BASE_RANGE; idx += BR_CDF_SIZE - 1) {
513 int k = AOMMIN(base_range - idx, BR_CDF_SIZE - 1);
Dake He7d01ab52017-11-24 17:53:28 -0800514 aom_write_symbol(
515 w, k,
516 ec_ctx->coeff_br_cdf[AOMMIN(txs_ctx, TX_16X16)][plane_type][ctx],
517 BR_CDF_SIZE);
Ola Hugossone72a2092017-11-12 09:11:53 +0100518 if (k < BR_CDF_SIZE - 1) break;
519 }
520 if (base_range < COEFF_BASE_RANGE) continue;
521#else
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800522 int br_set_idx = 0;
523 int br_base = 0;
524 int br_offset = 0;
Jingning Han87b01b52017-08-31 12:07:20 -0700525
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800526 if (base_range >= COEFF_BASE_RANGE)
527 br_set_idx = BASE_RANGE_SETS;
528 else
529 br_set_idx = coeff_to_br_index[base_range];
Jingning Han87b01b52017-08-31 12:07:20 -0700530
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800531 for (idx = 0; idx < BASE_RANGE_SETS; ++idx) {
Ola Hugosson13892102017-11-06 08:01:44 +0100532 // printf("br: %d %d %d %d\n", txs_ctx, plane_type, idx, ctx);
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800533 aom_write_bin(w, idx == br_set_idx,
534 ec_ctx->coeff_br_cdf[txs_ctx][plane_type][idx][ctx], 2);
535 if (idx == br_set_idx) {
536 br_base = br_index_to_coeff[br_set_idx];
537 br_offset = base_range - br_base;
538 int extra_bits = (1 << br_extra_bits[idx]) - 1;
539 for (int tok = 0; tok < extra_bits; ++tok) {
540 if (tok == br_offset) {
541 aom_write_bin(w, 1,
542 ec_ctx->coeff_lps_cdf[txs_ctx][plane_type][ctx], 2);
543 break;
544 }
545 aom_write_bin(w, 0, ec_ctx->coeff_lps_cdf[txs_ctx][plane_type][ctx],
Jingning Han94cea4a2017-09-30 14:13:23 -0700546 2);
Jingning Han87b01b52017-08-31 12:07:20 -0700547 }
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800548 // aom_write_literal(w, br_offset, br_extra_bits[idx]);
549 break;
Jingning Han87b01b52017-08-31 12:07:20 -0700550 }
Jingning Han87b01b52017-08-31 12:07:20 -0700551 }
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800552
553 if (br_set_idx < BASE_RANGE_SETS) continue;
Ola Hugossone72a2092017-11-12 09:11:53 +0100554#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800555 // use 0-th order Golomb code to handle the residual level.
556 write_golomb(
557 w, abs(tcoeff[scan[c]]) - COEFF_BASE_RANGE - 1 - NUM_BASE_LEVELS);
Jingning Han87b01b52017-08-31 12:07:20 -0700558 }
Angie Chiang80b82262017-02-24 11:39:47 -0800559 }
Jingning Han0bd3bf62017-11-28 17:11:51 -0800560
561#if CONFIG_ADAPT_SCAN
562 const int mi_row = -xd->mb_to_top_edge >> (3 + MI_SIZE_LOG2);
563 av1_update_scan_count_facade(cm, xd, mi_row, tx_size, tx_type, tcoeff, eob);
564#endif
Angie Chiang80b82262017-02-24 11:39:47 -0800565}
Angie Chiang47c72182017-02-27 14:30:38 -0800566
Angie Chiangc8af6112017-03-16 16:11:22 -0700567void av1_write_coeffs_mb(const AV1_COMMON *const cm, MACROBLOCK *x,
568 aom_writer *w, int plane) {
569 MACROBLOCKD *xd = &x->e_mbd;
570 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
571 BLOCK_SIZE bsize = mbmi->sb_type;
572 struct macroblockd_plane *pd = &xd->plane[plane];
Angie Chiangc8af6112017-03-16 16:11:22 -0700573 const BLOCK_SIZE plane_bsize =
Angie Chiangee4a34c2017-06-07 15:23:17 -0700574 AOMMAX(BLOCK_4X4, get_plane_block_size(bsize, pd));
Angie Chiang8d6bc802017-04-12 15:14:05 -0700575 const int max_blocks_wide = max_block_wide(xd, plane_bsize, plane);
576 const int max_blocks_high = max_block_high(xd, plane_bsize, plane);
hui su0c6244b2017-07-12 17:11:43 -0700577 const TX_SIZE tx_size = av1_get_tx_size(plane, xd);
Angie Chiangc8af6112017-03-16 16:11:22 -0700578 const int bkw = tx_size_wide_unit[tx_size];
579 const int bkh = tx_size_high_unit[tx_size];
580 const int step = tx_size_wide_unit[tx_size] * tx_size_high_unit[tx_size];
581 int row, col;
582 int block = 0;
Angie Chiang8d6bc802017-04-12 15:14:05 -0700583 for (row = 0; row < max_blocks_high; row += bkh) {
584 for (col = 0; col < max_blocks_wide; col += bkw) {
Angie Chiangc8af6112017-03-16 16:11:22 -0700585 tran_low_t *tcoeff = BLOCK_OFFSET(x->mbmi_ext->tcoeff[plane], block);
586 uint16_t eob = x->mbmi_ext->eobs[plane][block];
587 TXB_CTX txb_ctx = { x->mbmi_ext->txb_skip_ctx[plane][block],
588 x->mbmi_ext->dc_sign_ctx[plane][block] };
Jingning Han7eab9ff2017-07-06 10:12:54 -0700589 av1_write_coeffs_txb(cm, xd, w, row, col, block, plane, tx_size, tcoeff,
590 eob, &txb_ctx);
Angie Chiangc8af6112017-03-16 16:11:22 -0700591 block += step;
592 }
593 }
594}
595
Yaowu Xu9b803912017-11-16 08:16:33 -0800596#if !USE_CAUSAL_BASE_CTX
Linfeng Zhang841a84d2017-10-30 14:31:11 -0700597static INLINE void get_base_ctx_set(const uint8_t *const levels,
598 const int c, // raster order
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700599 const int bwl,
Angie Chiang47c72182017-02-27 14:30:38 -0800600 int ctx_set[NUM_BASE_LEVELS]) {
601 const int row = c >> bwl;
602 const int col = c - (row << bwl);
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700603 const int width = 1 << bwl;
604 const int stride = width + TX_PAD_HOR;
Jingning Hane775b8b2017-10-26 16:56:22 -0700605 int mag_count[NUM_BASE_LEVELS] = { 0 };
606 int nb_mag[NUM_BASE_LEVELS][3] = { { 0 } };
Angie Chiang47c72182017-02-27 14:30:38 -0800607 int idx;
Angie Chiang47c72182017-02-27 14:30:38 -0800608 int i;
609
610 for (idx = 0; idx < BASE_CONTEXT_POSITION_NUM; ++idx) {
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700611 const int ref_row = row + base_ref_offset[idx][0];
612 const int ref_col = col + base_ref_offset[idx][1];
613 const int pos = ref_row * stride + ref_col;
Linfeng Zhang841a84d2017-10-30 14:31:11 -0700614 const uint8_t abs_coeff = levels[pos];
Angie Chiang47c72182017-02-27 14:30:38 -0800615
616 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
617 ctx_set[i] += abs_coeff > i;
Jingning Hane775b8b2017-10-26 16:56:22 -0700618 if (base_ref_offset[idx][0] == 0 && base_ref_offset[idx][1] == 1)
619 nb_mag[i][0] = abs_coeff;
620 if (base_ref_offset[idx][0] == 1 && base_ref_offset[idx][1] == 0)
621 nb_mag[i][1] = abs_coeff;
622 if (base_ref_offset[idx][0] == 1 && base_ref_offset[idx][1] == 1)
623 nb_mag[i][2] = abs_coeff;
Angie Chiang47c72182017-02-27 14:30:38 -0800624 }
625 }
626
627 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
Jingning Hane775b8b2017-10-26 16:56:22 -0700628 for (idx = 0; idx < 3; ++idx) mag_count[i] += nb_mag[i][idx] > i + 1;
629 ctx_set[i] = get_base_ctx_from_count_mag(row, col, ctx_set[i],
630 AOMMIN(2, mag_count[i]));
Angie Chiang47c72182017-02-27 14:30:38 -0800631 }
632 return;
633}
Yaowu Xu9b803912017-11-16 08:16:33 -0800634#endif
Angie Chiang47c72182017-02-27 14:30:38 -0800635
Angie Chiang488f9212017-05-30 12:46:26 -0700636static INLINE int get_br_cost(tran_low_t abs_qc, int ctx,
Angie Chiang26d3e452017-09-29 17:40:02 -0700637 const int *coeff_lps) {
Angie Chiang488f9212017-05-30 12:46:26 -0700638 const tran_low_t min_level = 1 + NUM_BASE_LEVELS;
639 const tran_low_t max_level = 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE;
Jingning Handfd72322017-08-09 14:04:12 -0700640 (void)ctx;
Angie Chiang488f9212017-05-30 12:46:26 -0700641 if (abs_qc >= min_level) {
Angie Chiang488f9212017-05-30 12:46:26 -0700642 if (abs_qc >= max_level)
Jingning Han772dee32017-09-15 08:53:18 -0700643 return coeff_lps[COEFF_BASE_RANGE]; // COEFF_BASE_RANGE * cost0;
Angie Chiang488f9212017-05-30 12:46:26 -0700644 else
Jingning Han772dee32017-09-15 08:53:18 -0700645 return coeff_lps[(abs_qc - min_level)]; // * cost0 + cost1;
Angie Chiang488f9212017-05-30 12:46:26 -0700646 } else {
647 return 0;
648 }
649}
650
651static INLINE int get_base_cost(tran_low_t abs_qc, int ctx,
Jingning Handfd72322017-08-09 14:04:12 -0700652 const int coeff_base[2], int base_idx) {
Angie Chiang488f9212017-05-30 12:46:26 -0700653 const int level = base_idx + 1;
Jingning Handfd72322017-08-09 14:04:12 -0700654 (void)ctx;
Angie Chiang488f9212017-05-30 12:46:26 -0700655 if (abs_qc < level)
656 return 0;
657 else
Jingning Handfd72322017-08-09 14:04:12 -0700658 return coeff_base[abs_qc == level];
Angie Chiang488f9212017-05-30 12:46:26 -0700659}
660
Angie Chiang3627de22017-08-18 20:15:59 -0700661int av1_cost_coeffs_txb(const AV1_COMMON *const cm, MACROBLOCK *x, int plane,
Jingning Han7eab9ff2017-07-06 10:12:54 -0700662 int blk_row, int blk_col, int block, TX_SIZE tx_size,
663 TXB_CTX *txb_ctx) {
Angie Chiang47c72182017-02-27 14:30:38 -0800664 MACROBLOCKD *const xd = &x->e_mbd;
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -0800665 TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Angie Chiang47c72182017-02-27 14:30:38 -0800666 const PLANE_TYPE plane_type = get_plane_type(plane);
Jingning Han19b5c8f2017-07-06 15:10:12 -0700667 const TX_TYPE tx_type =
668 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiang47c72182017-02-27 14:30:38 -0800669 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
670 const struct macroblock_plane *p = &x->plane[plane];
671 const int eob = p->eobs[block];
672 const tran_low_t *const qcoeff = BLOCK_OFFSET(p->qcoeff, block);
673 int c, cost;
Angie Chiang47c72182017-02-27 14:30:38 -0800674 int txb_skip_ctx = txb_ctx->txb_skip_ctx;
Angie Chiang47c72182017-02-27 14:30:38 -0800675
Angie Chianga9ba58e2017-12-01 19:22:43 -0800676 const int bwl = get_txb_bwl(tx_size);
677 const int width = get_txb_wide(tx_size);
678 const int height = get_txb_high(tx_size);
Jingning Han341d79e2017-06-13 15:57:59 -0700679
Angie Chiangbd99b382017-06-20 15:11:16 -0700680 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Angie Chiang47c72182017-02-27 14:30:38 -0800681 const int16_t *scan = scan_order->scan;
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700682 uint8_t levels_buf[TX_PAD_2D];
683 uint8_t *const levels = set_levels(levels_buf, width);
Linfeng Zhangae7b2f32017-11-08 15:46:57 -0800684 DECLARE_ALIGNED(16, uint8_t, level_counts[MAX_TX_SQUARE]);
Angie Chiang47c72182017-02-27 14:30:38 -0800685
Jingning Handfd72322017-08-09 14:04:12 -0700686 LV_MAP_COEFF_COST *coeff_costs = &x->coeff_costs[txs_ctx][plane_type];
687
Angie Chiang47c72182017-02-27 14:30:38 -0800688 cost = 0;
689
690 if (eob == 0) {
Jingning Handfd72322017-08-09 14:04:12 -0700691 cost = coeff_costs->txb_skip_cost[txb_skip_ctx][1];
Angie Chiang47c72182017-02-27 14:30:38 -0800692 return cost;
693 }
Jingning Handfd72322017-08-09 14:04:12 -0700694 cost = coeff_costs->txb_skip_cost[txb_skip_ctx][0];
Angie Chiang47c72182017-02-27 14:30:38 -0800695
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700696 av1_txb_init_levels(qcoeff, width, height, levels);
Linfeng Zhang1015a342017-10-24 16:20:41 -0700697
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700698#if CONFIG_TXK_SEL
Yue Chenb23d00a2017-07-28 17:01:21 -0700699 cost += av1_tx_type_cost(cm, x, xd, mbmi->sb_type, plane, tx_size, tx_type);
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700700#endif
Angie Chiang05917872017-04-15 12:28:56 -0700701
Urvang Joshi80893152017-10-27 11:51:14 -0700702 const int seg_eob = av1_get_max_eob(tx_size);
Jingning Han35deaa72017-10-26 15:36:30 -0700703 int eob_cost = get_eob_cost(eob, seg_eob, coeff_costs, tx_type);
Dake He7d01ab52017-11-24 17:53:28 -0800704#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800705 av1_get_br_level_counts(levels, width, height, level_counts);
Dake He7d01ab52017-11-24 17:53:28 -0800706#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700707 cost += eob_cost;
Dake He03a32922017-10-31 08:06:45 -0700708#if USE_CAUSAL_BASE_CTX
709 int coeff_ctx = 0;
710#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700711 for (c = eob - 1; c >= 0; --c) {
712 tran_low_t v = qcoeff[scan[c]];
713 int is_nz = (v != 0);
714 int level = abs(v);
Ola Hugosson13892102017-11-06 08:01:44 +0100715#if CONFIG_LV_MAP_MULTI
716 coeff_ctx =
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800717 get_nz_map_ctx(levels, c, scan, bwl, height, c == eob - 1, tx_type);
Dake He3fe369c2017-11-16 17:56:44 -0800718#if USE_BASE_EOB_ALPHABET
719 if (c == eob - 1) {
720 cost += coeff_costs
721 ->base_eob_cost[coeff_ctx - SIG_COEF_CONTEXTS +
722 SIG_COEF_CONTEXTS_EOB][AOMMIN(level, 3) - 1];
723 } else {
724 cost += coeff_costs->base_cost[coeff_ctx][AOMMIN(level, 3)];
725 }
726#else
Ola Hugosson13892102017-11-06 08:01:44 +0100727 cost += coeff_costs->base_cost[coeff_ctx][AOMMIN(level, 3)];
Dake He3fe369c2017-11-16 17:56:44 -0800728#endif
Ola Hugosson13892102017-11-06 08:01:44 +0100729#else // CONFIG_LV_MAP_MULTI
Dake He03a32922017-10-31 08:06:45 -0700730#if USE_CAUSAL_BASE_CTX
Linfeng Zhangb6957c22017-10-25 13:17:28 -0700731 coeff_ctx = get_nz_map_ctx(levels, c, scan, bwl, height, tx_type);
Dake He03a32922017-10-31 08:06:45 -0700732#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700733
734 if (c < eob - 1) {
Dake He03a32922017-10-31 08:06:45 -0700735#if !USE_CAUSAL_BASE_CTX
Linfeng Zhangb6957c22017-10-25 13:17:28 -0700736 int coeff_ctx = get_nz_map_ctx(levels, c, scan, bwl, height, tx_type);
Dake He03a32922017-10-31 08:06:45 -0700737#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700738 cost += coeff_costs->nz_map_cost[coeff_ctx][is_nz];
739 }
Ola Hugosson13892102017-11-06 08:01:44 +0100740#endif // CONFIG_LV_MAP_MULTI
Dake Hea47cd6c2017-10-13 18:09:58 -0700741
742 if (is_nz) {
Dake He03a32922017-10-31 08:06:45 -0700743#if !USE_CAUSAL_BASE_CTX
Dake Hea47cd6c2017-10-13 18:09:58 -0700744 int ctx_ls[NUM_BASE_LEVELS] = { 0 };
Dake He03a32922017-10-31 08:06:45 -0700745#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700746 int sign = (v < 0) ? 1 : 0;
747
748 // sign bit cost
749 if (c == 0) {
750 int dc_sign_ctx = txb_ctx->dc_sign_ctx;
751 cost += coeff_costs->dc_sign_cost[dc_sign_ctx][sign];
752 } else {
753 cost += av1_cost_bit(128, sign);
754 }
Ola Hugosson13892102017-11-06 08:01:44 +0100755#if !CONFIG_LV_MAP_MULTI
Dake He03a32922017-10-31 08:06:45 -0700756#if USE_CAUSAL_BASE_CTX
757 int k;
758 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
759 int is_k = (level > (k + 1));
760 int ctx = coeff_ctx;
761 // get_base_ctx_from_b(c, k, b0, b1, b2);
762 cost += coeff_costs->base_cost[k][ctx][is_k];
763 if (is_k == 0) break;
764 }
765#else
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700766 get_base_ctx_set(levels, scan[c], bwl, ctx_ls);
Dake Hea47cd6c2017-10-13 18:09:58 -0700767
768 int i;
769 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
770 if (level <= i) continue;
771
772 if (level == i + 1) {
773 cost += coeff_costs->base_cost[i][ctx_ls[i]][1];
774 continue;
775 }
776 cost += coeff_costs->base_cost[i][ctx_ls[i]][0];
777 }
Dake He03a32922017-10-31 08:06:45 -0700778#endif
Ola Hugosson13892102017-11-06 08:01:44 +0100779#endif // CONFIG_LV_MAP_MULTI
Dake Hea47cd6c2017-10-13 18:09:58 -0700780 if (level > NUM_BASE_LEVELS) {
781 int ctx;
Dake He7d01ab52017-11-24 17:53:28 -0800782#if CONFIG_LV_MAP_MULTI && USE_CAUSAL_BR_CTX
783 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]], tx_type);
784#else
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800785 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]]);
Dake He7d01ab52017-11-24 17:53:28 -0800786#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700787 int base_range = level - 1 - NUM_BASE_LEVELS;
788 if (base_range < COEFF_BASE_RANGE) {
789 cost += coeff_costs->lps_cost[ctx][base_range];
790 } else {
791 cost += coeff_costs->lps_cost[ctx][COEFF_BASE_RANGE];
792 }
793
Dake Hea47cd6c2017-10-13 18:09:58 -0700794 if (level >= 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
795 // residual cost
796 int r = level - COEFF_BASE_RANGE - NUM_BASE_LEVELS;
797 int ri = r;
798 int length = 0;
799
800 while (ri) {
801 ri >>= 1;
802 ++length;
803 }
804
805 for (ri = 0; ri < length - 1; ++ri) cost += av1_cost_bit(128, 0);
806
807 for (ri = length - 1; ri >= 0; --ri)
808 cost += av1_cost_bit(128, (r >> ri) & 0x01);
809 }
810 }
811 }
812 }
Angie Chiang47c72182017-02-27 14:30:38 -0800813 return cost;
814}
Angie Chiang0397eda2017-03-15 16:57:14 -0700815
Angie Chiang2affb3b2017-05-30 15:23:51 -0700816static INLINE int has_base(tran_low_t qc, int base_idx) {
817 const int level = base_idx + 1;
818 return abs(qc) >= level;
819}
820
Angie Chiang2affb3b2017-05-30 15:23:51 -0700821static INLINE int has_br(tran_low_t qc) {
822 return abs(qc) >= 1 + NUM_BASE_LEVELS;
823}
824
Angie Chiang488f9212017-05-30 12:46:26 -0700825static INLINE int get_sign_bit_cost(tran_low_t qc, int coeff_idx,
Jingning Handfd72322017-08-09 14:04:12 -0700826 const int (*dc_sign_cost)[2],
Angie Chiang488f9212017-05-30 12:46:26 -0700827 int dc_sign_ctx) {
828 const int sign = (qc < 0) ? 1 : 0;
829 // sign bit cost
830 if (coeff_idx == 0) {
Jingning Handfd72322017-08-09 14:04:12 -0700831 return dc_sign_cost[dc_sign_ctx][sign];
Angie Chiang488f9212017-05-30 12:46:26 -0700832 } else {
833 return av1_cost_bit(128, sign);
834 }
835}
836static INLINE int get_golomb_cost(int abs_qc) {
837 if (abs_qc >= 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
838 // residual cost
839 int r = abs_qc - COEFF_BASE_RANGE - NUM_BASE_LEVELS;
840 int ri = r;
841 int length = 0;
842
843 while (ri) {
844 ri >>= 1;
845 ++length;
846 }
847
848 return av1_cost_literal(2 * length - 1);
849 } else {
850 return 0;
851 }
852}
853
Angie Chiang2affb3b2017-05-30 15:23:51 -0700854void gen_txb_cache(TxbCache *txb_cache, TxbInfo *txb_info) {
Angie Chiang481c01f2017-08-15 16:24:30 -0700855 // gen_nz_count_arr
Angie Chiang2affb3b2017-05-30 15:23:51 -0700856 const int16_t *scan = txb_info->scan_order->scan;
Angie Chiang481c01f2017-08-15 16:24:30 -0700857 const int bwl = txb_info->bwl;
858 const int height = txb_info->height;
Linfeng Zhang72e0b3f2017-10-18 13:32:12 -0700859 const tran_low_t *const qcoeff = txb_info->qcoeff;
Linfeng Zhangb6957c22017-10-25 13:17:28 -0700860 const uint8_t *const levels = txb_info->levels;
Angie Chiangdef11252017-08-18 14:37:32 -0700861 const BASE_CTX_TABLE *base_ctx_table =
862 txb_info->coeff_ctx_table->base_ctx_table;
Angie Chiang481c01f2017-08-15 16:24:30 -0700863 for (int c = 0; c < txb_info->eob; ++c) {
864 const int coeff_idx = scan[c]; // raster order
865 const int row = coeff_idx >> bwl;
866 const int col = coeff_idx - (row << bwl);
Jingning Han3422ac12017-10-25 20:37:53 -0700867
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700868 txb_cache->nz_count_arr[coeff_idx] =
869 get_nz_count(levels, bwl, row, col, get_tx_class(txb_info->tx_type));
Dake Hea47cd6c2017-10-13 18:09:58 -0700870
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800871 txb_cache->nz_ctx_arr[coeff_idx] = get_nz_map_ctx_from_stats(
Dake He03a32922017-10-31 08:06:45 -0700872#if USE_CAUSAL_BASE_CTX
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800873 0,
874#else
875 txb_cache->nz_count_arr[coeff_idx],
Dake He03a32922017-10-31 08:06:45 -0700876#endif
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800877 coeff_idx, bwl, height, txb_info->tx_type);
Angie Chiang481c01f2017-08-15 16:24:30 -0700878
879 // gen_base_count_mag_arr
880 if (!has_base(qcoeff[coeff_idx], 0)) continue;
881 int *base_mag = txb_cache->base_mag_arr[coeff_idx];
Angie Chiang9cde59f2017-08-16 15:24:55 -0700882 int count[NUM_BASE_LEVELS];
883 get_base_count_mag(base_mag, count, qcoeff, bwl, height, row, col);
Angie Chiang481c01f2017-08-15 16:24:30 -0700884
885 for (int i = 0; i < NUM_BASE_LEVELS; ++i) {
Angie Chiang9cde59f2017-08-16 15:24:55 -0700886 if (!has_base(qcoeff[coeff_idx], i)) break;
887 txb_cache->base_count_arr[i][coeff_idx] = count[i];
Angie Chiang481c01f2017-08-15 16:24:30 -0700888 const int level = i + 1;
Angie Chiang9cde59f2017-08-16 15:24:55 -0700889 txb_cache->base_ctx_arr[i][coeff_idx] =
Angie Chiangdef11252017-08-18 14:37:32 -0700890 base_ctx_table[row != 0][col != 0][base_mag[0] > level][count[i]];
Angie Chiang481c01f2017-08-15 16:24:30 -0700891 }
892
893 // gen_br_count_mag_arr
894 if (!has_br(qcoeff[coeff_idx])) continue;
895 int *br_count = txb_cache->br_count_arr + coeff_idx;
896 int *br_mag = txb_cache->br_mag_arr[coeff_idx];
Angie Chiangea8183b2017-08-17 14:36:35 -0700897 *br_count = get_br_count_mag(br_mag, qcoeff, bwl, height, row, col,
898 NUM_BASE_LEVELS);
Angie Chiang481c01f2017-08-15 16:24:30 -0700899 txb_cache->br_ctx_arr[coeff_idx] =
900 get_br_ctx_from_count_mag(row, col, *br_count, br_mag[0]);
901 }
Angie Chiang2affb3b2017-05-30 15:23:51 -0700902}
903
Jingning Handfd72322017-08-09 14:04:12 -0700904static INLINE const int *get_level_prob(int level, int coeff_idx,
905 const TxbCache *txb_cache,
906 const LV_MAP_COEFF_COST *txb_costs) {
Ola Hugosson13892102017-11-06 08:01:44 +0100907#if CONFIG_LV_MAP_MULTI
908 if (level < 1 + NUM_BASE_LEVELS) {
909 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
910 return &txb_costs->base_cost[ctx][level];
911#else
Angie Chiang7afbba42017-05-30 15:59:15 -0700912 if (level == 0) {
Angie Chiang1ae0ebf2017-08-15 15:16:47 -0700913 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -0700914 return txb_costs->nz_map_cost[ctx];
Angie Chiang7afbba42017-05-30 15:59:15 -0700915 } else if (level >= 1 && level < 1 + NUM_BASE_LEVELS) {
916 const int idx = level - 1;
Angie Chiang1ae0ebf2017-08-15 15:16:47 -0700917 const int ctx = txb_cache->base_ctx_arr[idx][coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -0700918 return txb_costs->base_cost[idx][ctx];
Ola Hugosson13892102017-11-06 08:01:44 +0100919#endif
Angie Chiang7afbba42017-05-30 15:59:15 -0700920 } else if (level >= 1 + NUM_BASE_LEVELS &&
921 level < 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Angie Chiang1ae0ebf2017-08-15 15:16:47 -0700922 const int ctx = txb_cache->br_ctx_arr[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -0700923 return txb_costs->lps_cost[ctx];
Angie Chiang7afbba42017-05-30 15:59:15 -0700924 } else if (level >= 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Ola Hugosson13892102017-11-06 08:01:44 +0100925 // printf("get_level_prob does not support golomb\n");
Angie Chiang7afbba42017-05-30 15:59:15 -0700926 assert(0);
927 return 0;
928 } else {
929 assert(0);
930 return 0;
931 }
932}
933
Angie Chiangd19969e2017-05-30 18:02:33 -0700934static INLINE void update_mag_arr(int *mag_arr, int abs_qc) {
935 if (mag_arr[0] == abs_qc) {
936 mag_arr[1] -= 1;
937 assert(mag_arr[1] >= 0);
938 }
939}
940
Angie Chianga530ef42017-05-30 16:32:36 -0700941static INLINE int get_mag_from_mag_arr(const int *mag_arr) {
942 int mag;
943 if (mag_arr[1] > 0) {
944 mag = mag_arr[0];
945 } else if (mag_arr[0] > 0) {
946 mag = mag_arr[0] - 1;
947 } else {
948 // no neighbor
949 assert(mag_arr[0] == 0 && mag_arr[1] == 0);
950 mag = 0;
951 }
952 return mag;
953}
954
955static int neighbor_level_down_update(int *new_count, int *new_mag, int count,
956 const int *mag, int coeff_idx,
957 tran_low_t abs_nb_coeff, int nb_coeff_idx,
958 int level, const TxbInfo *txb_info) {
959 *new_count = count;
960 *new_mag = get_mag_from_mag_arr(mag);
961
962 int update = 0;
963 // check if br_count changes
964 if (abs_nb_coeff == level) {
965 update = 1;
966 *new_count -= 1;
967 assert(*new_count >= 0);
968 }
969 const int row = coeff_idx >> txb_info->bwl;
970 const int col = coeff_idx - (row << txb_info->bwl);
971 const int nb_row = nb_coeff_idx >> txb_info->bwl;
972 const int nb_col = nb_coeff_idx - (nb_row << txb_info->bwl);
973
974 // check if mag changes
975 if (nb_row >= row && nb_col >= col) {
976 if (abs_nb_coeff == mag[0]) {
977 assert(mag[1] > 0);
978 if (mag[1] == 1) {
979 // the nb is the only qc with max mag
980 *new_mag -= 1;
981 assert(*new_mag >= 0);
982 update = 1;
983 }
984 }
985 }
986 return update;
987}
988
989static int try_neighbor_level_down_br(int coeff_idx, int nb_coeff_idx,
990 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -0700991 const LV_MAP_COEFF_COST *txb_costs,
Angie Chianga530ef42017-05-30 16:32:36 -0700992 const TxbInfo *txb_info) {
993 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
994 const tran_low_t abs_qc = abs(qc);
995 const int level = NUM_BASE_LEVELS + 1;
996 if (abs_qc < level) return 0;
997
998 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
999 const tran_low_t abs_nb_coeff = abs(nb_coeff);
1000 const int count = txb_cache->br_count_arr[coeff_idx];
1001 const int *mag = txb_cache->br_mag_arr[coeff_idx];
1002 int new_count;
1003 int new_mag;
1004 const int update =
1005 neighbor_level_down_update(&new_count, &new_mag, count, mag, coeff_idx,
1006 abs_nb_coeff, nb_coeff_idx, level, txb_info);
1007 if (update) {
1008 const int row = coeff_idx >> txb_info->bwl;
1009 const int col = coeff_idx - (row << txb_info->bwl);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001010 const int ctx = txb_cache->br_ctx_arr[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001011 const int org_cost = get_br_cost(abs_qc, ctx, txb_costs->lps_cost[ctx]);
Angie Chianga530ef42017-05-30 16:32:36 -07001012
1013 const int new_ctx = get_br_ctx_from_count_mag(row, col, new_count, new_mag);
Jingning Handfd72322017-08-09 14:04:12 -07001014 const int new_cost =
1015 get_br_cost(abs_qc, new_ctx, txb_costs->lps_cost[new_ctx]);
Angie Chianga530ef42017-05-30 16:32:36 -07001016 const int cost_diff = -org_cost + new_cost;
1017 return cost_diff;
1018 } else {
1019 return 0;
1020 }
1021}
1022
1023static int try_neighbor_level_down_base(int coeff_idx, int nb_coeff_idx,
1024 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001025 const LV_MAP_COEFF_COST *txb_costs,
Angie Chianga530ef42017-05-30 16:32:36 -07001026 const TxbInfo *txb_info) {
Ola Hugosson13892102017-11-06 08:01:44 +01001027#if CONFIG_LV_MAP_MULTI
1028 // TODO(olah): not implemented yet
1029 (void)coeff_idx;
1030 (void)nb_coeff_idx;
1031 (void)txb_cache;
1032 (void)txb_costs;
1033 (void)txb_info;
1034 return 0;
1035#else
Angie Chianga530ef42017-05-30 16:32:36 -07001036 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1037 const tran_low_t abs_qc = abs(qc);
Angie Chiangdef11252017-08-18 14:37:32 -07001038 const BASE_CTX_TABLE *base_ctx_table =
1039 txb_info->coeff_ctx_table->base_ctx_table;
Angie Chianga530ef42017-05-30 16:32:36 -07001040
1041 int cost_diff = 0;
1042 for (int base_idx = 0; base_idx < NUM_BASE_LEVELS; ++base_idx) {
1043 const int level = base_idx + 1;
1044 if (abs_qc < level) continue;
1045
1046 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1047 const tran_low_t abs_nb_coeff = abs(nb_coeff);
1048
1049 const int count = txb_cache->base_count_arr[base_idx][coeff_idx];
1050 const int *mag = txb_cache->base_mag_arr[coeff_idx];
1051 int new_count;
1052 int new_mag;
1053 const int update =
1054 neighbor_level_down_update(&new_count, &new_mag, count, mag, coeff_idx,
1055 abs_nb_coeff, nb_coeff_idx, level, txb_info);
1056 if (update) {
1057 const int row = coeff_idx >> txb_info->bwl;
1058 const int col = coeff_idx - (row << txb_info->bwl);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001059 const int ctx = txb_cache->base_ctx_arr[base_idx][coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001060 const int org_cost = get_base_cost(
1061 abs_qc, ctx, txb_costs->base_cost[base_idx][ctx], base_idx);
Angie Chianga530ef42017-05-30 16:32:36 -07001062
1063 const int new_ctx =
Angie Chiangdef11252017-08-18 14:37:32 -07001064 base_ctx_table[row != 0][col != 0][new_mag > level][new_count];
Jingning Handfd72322017-08-09 14:04:12 -07001065 const int new_cost = get_base_cost(
1066 abs_qc, new_ctx, txb_costs->base_cost[base_idx][new_ctx], base_idx);
Angie Chianga530ef42017-05-30 16:32:36 -07001067 cost_diff += -org_cost + new_cost;
1068 }
1069 }
1070 return cost_diff;
Ola Hugosson13892102017-11-06 08:01:44 +01001071#endif
Angie Chianga530ef42017-05-30 16:32:36 -07001072}
1073
1074static int try_neighbor_level_down_nz(int coeff_idx, int nb_coeff_idx,
1075 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001076 const LV_MAP_COEFF_COST *txb_costs,
Angie Chianga530ef42017-05-30 16:32:36 -07001077 TxbInfo *txb_info) {
1078 // assume eob doesn't change
1079 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1080 const tran_low_t abs_qc = abs(qc);
1081 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1082 const tran_low_t abs_nb_coeff = abs(nb_coeff);
1083 if (abs_nb_coeff != 1) return 0;
1084 const int16_t *iscan = txb_info->scan_order->iscan;
1085 const int scan_idx = iscan[coeff_idx];
1086 if (scan_idx == txb_info->seg_eob) return 0;
1087 const int nb_scan_idx = iscan[nb_coeff_idx];
1088 if (nb_scan_idx < scan_idx) {
1089 const int count = txb_cache->nz_count_arr[coeff_idx];
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001090 (void)count;
Angie Chianga530ef42017-05-30 16:32:36 -07001091 assert(count > 0);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001092 update_qcoeff(nb_coeff_idx, get_lower_coeff(nb_coeff), txb_info);
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001093 const int new_ctx = get_nz_map_ctx_from_stats(
Dake He03a32922017-10-31 08:06:45 -07001094#if USE_CAUSAL_BASE_CTX
1095 0,
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001096#else
1097 count - 1,
Dake He03a32922017-10-31 08:06:45 -07001098#endif
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001099 coeff_idx, txb_info->bwl, txb_info->height, txb_info->tx_type);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001100 update_qcoeff(nb_coeff_idx, nb_coeff, txb_info);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001101 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
Ola Hugosson13892102017-11-06 08:01:44 +01001102#if CONFIG_LV_MAP_MULTI
1103 const int org_cost = txb_costs->base_cost[ctx][AOMMIN(abs_qc, 3)];
1104 const int new_cost = txb_costs->base_cost[new_ctx][AOMMIN(abs_qc, 3)];
1105#else
Angie Chianga530ef42017-05-30 16:32:36 -07001106 const int is_nz = abs_qc > 0;
Jingning Handfd72322017-08-09 14:04:12 -07001107 const int org_cost = txb_costs->nz_map_cost[ctx][is_nz];
1108 const int new_cost = txb_costs->nz_map_cost[new_ctx][is_nz];
Ola Hugosson13892102017-11-06 08:01:44 +01001109#endif
Angie Chianga530ef42017-05-30 16:32:36 -07001110 const int cost_diff = new_cost - org_cost;
1111 return cost_diff;
1112 } else {
1113 return 0;
1114 }
1115}
1116
1117static int try_self_level_down(tran_low_t *low_coeff, int coeff_idx,
1118 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001119 const LV_MAP_COEFF_COST *txb_costs,
1120 TxbInfo *txb_info) {
Angie Chiang7afbba42017-05-30 15:59:15 -07001121 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1122 if (qc == 0) {
1123 *low_coeff = 0;
1124 return 0;
1125 }
1126 const tran_low_t abs_qc = abs(qc);
1127 *low_coeff = get_lower_coeff(qc);
1128 int cost_diff;
1129 if (*low_coeff == 0) {
1130 const int scan_idx = txb_info->scan_order->iscan[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001131 const int *level_cost =
1132 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1133 const int *low_level_cost =
1134 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
Jingning Han3422ac12017-10-25 20:37:53 -07001135
Dake Hea47cd6c2017-10-13 18:09:58 -07001136 if (scan_idx < txb_info->eob - 1) {
Angie Chiang7afbba42017-05-30 15:59:15 -07001137 // When level-0, we code the binary of abs_qc > level
1138 // but when level-k k > 0 we code the binary of abs_qc == level
1139 // That's why wee need this special treatment for level-0 map
1140 // TODO(angiebird): make leve-0 consistent to other levels
Jingning Handfd72322017-08-09 14:04:12 -07001141 cost_diff = -level_cost[1] + low_level_cost[0] - low_level_cost[1];
Angie Chiang7afbba42017-05-30 15:59:15 -07001142 } else {
Jingning Handfd72322017-08-09 14:04:12 -07001143 cost_diff = -level_cost[1];
Angie Chiang7afbba42017-05-30 15:59:15 -07001144 }
1145
Angie Chiang7afbba42017-05-30 15:59:15 -07001146 const int sign_cost = get_sign_bit_cost(
Jingning Handfd72322017-08-09 14:04:12 -07001147 qc, coeff_idx, txb_costs->dc_sign_cost, txb_info->txb_ctx->dc_sign_ctx);
Angie Chiang7afbba42017-05-30 15:59:15 -07001148 cost_diff -= sign_cost;
Jingning Han772dee32017-09-15 08:53:18 -07001149 } else if (abs_qc <= NUM_BASE_LEVELS) {
1150 const int *level_cost =
1151 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1152 const int *low_level_cost =
1153 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
1154 cost_diff = -level_cost[1] + low_level_cost[1] - low_level_cost[0];
1155 } else if (abs_qc == NUM_BASE_LEVELS + 1) {
1156 const int *level_cost =
1157 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1158 const int *low_level_cost =
1159 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
1160 cost_diff = -level_cost[0] + low_level_cost[1] - low_level_cost[0];
Angie Chiang7afbba42017-05-30 15:59:15 -07001161 } else if (abs_qc < 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Jingning Handfd72322017-08-09 14:04:12 -07001162 const int *level_cost =
1163 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1164 const int *low_level_cost =
1165 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
1166
Jingning Han772dee32017-09-15 08:53:18 -07001167 cost_diff = -level_cost[abs_qc - 1 - NUM_BASE_LEVELS] +
1168 low_level_cost[abs(*low_coeff) - 1 - NUM_BASE_LEVELS];
Angie Chiang7afbba42017-05-30 15:59:15 -07001169 } else if (abs_qc == 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Jingning Handfd72322017-08-09 14:04:12 -07001170 const int *low_level_cost =
1171 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
Jingning Han772dee32017-09-15 08:53:18 -07001172 cost_diff = -get_golomb_cost(abs_qc) - low_level_cost[COEFF_BASE_RANGE] +
1173 low_level_cost[COEFF_BASE_RANGE - 1];
Angie Chiang7afbba42017-05-30 15:59:15 -07001174 } else {
1175 assert(abs_qc > 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE);
1176 const tran_low_t abs_low_coeff = abs(*low_coeff);
1177 cost_diff = -get_golomb_cost(abs_qc) + get_golomb_cost(abs_low_coeff);
1178 }
1179 return cost_diff;
1180}
1181
Angie Chianga530ef42017-05-30 16:32:36 -07001182#define COST_MAP_SIZE 5
1183#define COST_MAP_OFFSET 2
1184
1185static INLINE int check_nz_neighbor(tran_low_t qc) { return abs(qc) == 1; }
1186
1187static INLINE int check_base_neighbor(tran_low_t qc) {
1188 return abs(qc) <= 1 + NUM_BASE_LEVELS;
1189}
1190
1191static INLINE int check_br_neighbor(tran_low_t qc) {
1192 return abs(qc) > BR_MAG_OFFSET;
1193}
1194
Angie Chiang87278292017-10-18 09:59:47 -07001195#define FAST_OPTIMIZE_TXB 1
Angie Chiange80957f2017-09-05 10:48:00 -07001196
1197#if FAST_OPTIMIZE_TXB
1198#define ALNB_REF_OFFSET_NUM 2
Linfeng Zhang4afda452017-10-24 10:34:04 -07001199static const int alnb_ref_offset[ALNB_REF_OFFSET_NUM][2] = {
Angie Chiange80957f2017-09-05 10:48:00 -07001200 { -1, 0 }, { 0, -1 },
1201};
1202#define NB_REF_OFFSET_NUM 4
Linfeng Zhang4afda452017-10-24 10:34:04 -07001203static const int nb_ref_offset[NB_REF_OFFSET_NUM][2] = {
Angie Chiange80957f2017-09-05 10:48:00 -07001204 { -1, 0 }, { 0, -1 }, { 1, 0 }, { 0, 1 },
1205};
1206#endif // FAST_OPTIMIZE_TXB
1207
Angie Chianga530ef42017-05-30 16:32:36 -07001208// TODO(angiebird): add static to this function once it's called
1209int try_level_down(int coeff_idx, const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001210 const LV_MAP_COEFF_COST *txb_costs, TxbInfo *txb_info,
Angie Chiang25645b72017-09-24 14:28:49 -07001211 int (*cost_map)[COST_MAP_SIZE], int fast_mode) {
1212#if !FAST_OPTIMIZE_TXB
1213 (void)fast_mode;
1214#endif
Angie Chianga530ef42017-05-30 16:32:36 -07001215 if (cost_map) {
1216 for (int i = 0; i < COST_MAP_SIZE; ++i) av1_zero(cost_map[i]);
1217 }
1218
1219 tran_low_t qc = txb_info->qcoeff[coeff_idx];
1220 tran_low_t low_coeff;
1221 if (qc == 0) return 0;
1222 int accu_cost_diff = 0;
1223
1224 const int16_t *iscan = txb_info->scan_order->iscan;
1225 const int eob = txb_info->eob;
1226 const int scan_idx = iscan[coeff_idx];
1227 if (scan_idx < eob) {
1228 const int cost_diff = try_self_level_down(&low_coeff, coeff_idx, txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001229 txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001230 if (cost_map)
1231 cost_map[0 + COST_MAP_OFFSET][0 + COST_MAP_OFFSET] = cost_diff;
1232 accu_cost_diff += cost_diff;
1233 }
1234
1235 const int row = coeff_idx >> txb_info->bwl;
1236 const int col = coeff_idx - (row << txb_info->bwl);
1237 if (check_nz_neighbor(qc)) {
Angie Chiange80957f2017-09-05 10:48:00 -07001238#if FAST_OPTIMIZE_TXB
Linfeng Zhang4afda452017-10-24 10:34:04 -07001239 const int(*ref_offset)[2];
Angie Chiang25645b72017-09-24 14:28:49 -07001240 int ref_num;
1241 if (fast_mode) {
1242 ref_offset = alnb_ref_offset;
1243 ref_num = ALNB_REF_OFFSET_NUM;
1244 } else {
1245 ref_offset = sig_ref_offset;
1246 ref_num = SIG_REF_OFFSET_NUM;
1247 }
Angie Chiange80957f2017-09-05 10:48:00 -07001248#else
Linfeng Zhang4afda452017-10-24 10:34:04 -07001249 const int(*ref_offset)[2] = sig_ref_offset;
Angie Chiange80957f2017-09-05 10:48:00 -07001250 const int ref_num = SIG_REF_OFFSET_NUM;
1251#endif
1252 for (int i = 0; i < ref_num; ++i) {
1253 const int nb_row = row - ref_offset[i][0];
1254 const int nb_col = col - ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001255 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han4cbb3632017-06-13 12:50:33 -07001256
Angie Chiang0c89dca2017-08-17 16:36:18 -07001257 if (nb_row < 0 || nb_col < 0 || nb_row >= txb_info->height ||
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001258 nb_col >= txb_info->width)
Jingning Han3455e762017-06-13 21:15:44 -07001259 continue;
1260
Angie Chianga530ef42017-05-30 16:32:36 -07001261 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001262 if (nb_scan_idx < eob) {
Angie Chianga530ef42017-05-30 16:32:36 -07001263 const int cost_diff = try_neighbor_level_down_nz(
Jingning Handfd72322017-08-09 14:04:12 -07001264 nb_coeff_idx, coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001265 if (cost_map)
1266 cost_map[nb_row - row + COST_MAP_OFFSET]
1267 [nb_col - col + COST_MAP_OFFSET] += cost_diff;
1268 accu_cost_diff += cost_diff;
1269 }
1270 }
1271 }
1272
1273 if (check_base_neighbor(qc)) {
Angie Chiange80957f2017-09-05 10:48:00 -07001274#if FAST_OPTIMIZE_TXB
Linfeng Zhang4afda452017-10-24 10:34:04 -07001275 const int(*ref_offset)[2];
Angie Chiang25645b72017-09-24 14:28:49 -07001276 int ref_num;
1277 if (fast_mode) {
1278 ref_offset = nb_ref_offset;
1279 ref_num = NB_REF_OFFSET_NUM;
1280 } else {
1281 ref_offset = base_ref_offset;
1282 ref_num = BASE_CONTEXT_POSITION_NUM;
1283 }
Angie Chiange80957f2017-09-05 10:48:00 -07001284#else
Linfeng Zhangce065ca2017-10-17 16:49:30 -07001285 const int(*ref_offset)[2] = base_ref_offset;
Angie Chiang25645b72017-09-24 14:28:49 -07001286 int ref_num = BASE_CONTEXT_POSITION_NUM;
Angie Chiange80957f2017-09-05 10:48:00 -07001287#endif
1288 for (int i = 0; i < ref_num; ++i) {
1289 const int nb_row = row - ref_offset[i][0];
1290 const int nb_col = col - ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001291 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001292
Angie Chiang0c89dca2017-08-17 16:36:18 -07001293 if (nb_row < 0 || nb_col < 0 || nb_row >= txb_info->height ||
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001294 nb_col >= txb_info->width)
Jingning Han3455e762017-06-13 21:15:44 -07001295 continue;
1296
Angie Chianga530ef42017-05-30 16:32:36 -07001297 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001298 if (nb_scan_idx < eob) {
Angie Chianga530ef42017-05-30 16:32:36 -07001299 const int cost_diff = try_neighbor_level_down_base(
Jingning Handfd72322017-08-09 14:04:12 -07001300 nb_coeff_idx, coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001301 if (cost_map)
1302 cost_map[nb_row - row + COST_MAP_OFFSET]
1303 [nb_col - col + COST_MAP_OFFSET] += cost_diff;
1304 accu_cost_diff += cost_diff;
1305 }
1306 }
1307 }
1308
1309 if (check_br_neighbor(qc)) {
Angie Chiange80957f2017-09-05 10:48:00 -07001310#if FAST_OPTIMIZE_TXB
Linfeng Zhang4afda452017-10-24 10:34:04 -07001311 const int(*ref_offset)[2];
Angie Chiang25645b72017-09-24 14:28:49 -07001312 int ref_num;
1313 if (fast_mode) {
1314 ref_offset = nb_ref_offset;
1315 ref_num = NB_REF_OFFSET_NUM;
1316 } else {
1317 ref_offset = br_ref_offset;
1318 ref_num = BR_CONTEXT_POSITION_NUM;
1319 }
Angie Chiange80957f2017-09-05 10:48:00 -07001320#else
Linfeng Zhangce065ca2017-10-17 16:49:30 -07001321 const int(*ref_offset)[2] = br_ref_offset;
Angie Chiange80957f2017-09-05 10:48:00 -07001322 const int ref_num = BR_CONTEXT_POSITION_NUM;
1323#endif
1324 for (int i = 0; i < ref_num; ++i) {
1325 const int nb_row = row - ref_offset[i][0];
1326 const int nb_col = col - ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001327 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001328
Angie Chiang0c89dca2017-08-17 16:36:18 -07001329 if (nb_row < 0 || nb_col < 0 || nb_row >= txb_info->height ||
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001330 nb_col >= txb_info->width)
Jingning Han3455e762017-06-13 21:15:44 -07001331 continue;
1332
Angie Chianga530ef42017-05-30 16:32:36 -07001333 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001334 if (nb_scan_idx < eob) {
Angie Chianga530ef42017-05-30 16:32:36 -07001335 const int cost_diff = try_neighbor_level_down_br(
Jingning Handfd72322017-08-09 14:04:12 -07001336 nb_coeff_idx, coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001337 if (cost_map)
1338 cost_map[nb_row - row + COST_MAP_OFFSET]
1339 [nb_col - col + COST_MAP_OFFSET] += cost_diff;
1340 accu_cost_diff += cost_diff;
1341 }
1342 }
1343 }
1344
1345 return accu_cost_diff;
1346}
1347
Angie Chiangc77799b2017-05-30 17:08:17 -07001348static int get_low_coeff_cost(int coeff_idx, const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001349 const LV_MAP_COEFF_COST *txb_costs,
Angie Chiangc77799b2017-05-30 17:08:17 -07001350 const TxbInfo *txb_info) {
1351 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1352 const int abs_qc = abs(qc);
1353 assert(abs_qc <= 1);
1354 int cost = 0;
Jingning Han3422ac12017-10-25 20:37:53 -07001355
Ola Hugosson13892102017-11-06 08:01:44 +01001356#if CONFIG_LV_MAP_MULTI
1357 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
1358 cost += txb_costs->base_cost[ctx][AOMMIN(abs_qc, 3)];
1359 if (qc != 0) {
1360 cost += get_sign_bit_cost(qc, coeff_idx, txb_costs->dc_sign_cost,
1361 txb_info->txb_ctx->dc_sign_ctx);
1362 }
1363#else
1364 const int scan_idx = txb_info->scan_order->iscan[coeff_idx];
Dake Hea47cd6c2017-10-13 18:09:58 -07001365 if (scan_idx < txb_info->eob - 1) {
Jingning Handfd72322017-08-09 14:04:12 -07001366 const int *level_cost = get_level_prob(0, coeff_idx, txb_cache, txb_costs);
1367 cost += level_cost[qc != 0];
Angie Chiangc77799b2017-05-30 17:08:17 -07001368 }
1369
1370 if (qc != 0) {
1371 const int base_idx = 0;
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001372 const int ctx = txb_cache->base_ctx_arr[base_idx][coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001373 cost += get_base_cost(abs_qc, ctx, txb_costs->base_cost[base_idx][ctx],
1374 base_idx);
Jingning Handfd72322017-08-09 14:04:12 -07001375 cost += get_sign_bit_cost(qc, coeff_idx, txb_costs->dc_sign_cost,
Angie Chiangc77799b2017-05-30 17:08:17 -07001376 txb_info->txb_ctx->dc_sign_ctx);
1377 }
Ola Hugosson13892102017-11-06 08:01:44 +01001378#endif
Angie Chiangc77799b2017-05-30 17:08:17 -07001379 return cost;
1380}
1381
1382static INLINE void set_eob(TxbInfo *txb_info, int eob) {
1383 txb_info->eob = eob;
Urvang Joshi80893152017-10-27 11:51:14 -07001384 txb_info->seg_eob = av1_get_max_eob(txb_info->tx_size);
Angie Chiangc77799b2017-05-30 17:08:17 -07001385}
1386
1387// TODO(angiebird): add static to this function once it's called
1388int try_change_eob(int *new_eob, int coeff_idx, const TxbCache *txb_cache,
Angie Chiang25645b72017-09-24 14:28:49 -07001389 const LV_MAP_COEFF_COST *txb_costs, TxbInfo *txb_info,
1390 int fast_mode) {
Angie Chiangc77799b2017-05-30 17:08:17 -07001391 assert(txb_info->eob > 0);
1392 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1393 const int abs_qc = abs(qc);
1394 if (abs_qc != 1) {
1395 *new_eob = -1;
1396 return 0;
1397 }
1398 const int16_t *iscan = txb_info->scan_order->iscan;
1399 const int16_t *scan = txb_info->scan_order->scan;
1400 const int scan_idx = iscan[coeff_idx];
1401 *new_eob = 0;
1402 int cost_diff = 0;
Jingning Handfd72322017-08-09 14:04:12 -07001403 cost_diff -= get_low_coeff_cost(coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chiangc77799b2017-05-30 17:08:17 -07001404 // int coeff_cost =
Angie Chiang5e012fe2017-05-30 18:47:39 -07001405 // get_coeff_cost(qc, scan_idx, txb_info, txb_probs);
Angie Chiangc77799b2017-05-30 17:08:17 -07001406 // if (-cost_diff != coeff_cost) {
1407 // printf("-cost_diff %d coeff_cost %d\n", -cost_diff, coeff_cost);
1408 // get_low_coeff_cost(coeff_idx, txb_cache, txb_probs, txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001409 // get_coeff_cost(qc, scan_idx, txb_info, txb_probs);
Angie Chiangc77799b2017-05-30 17:08:17 -07001410 // }
1411 for (int si = scan_idx - 1; si >= 0; --si) {
1412 const int ci = scan[si];
1413 if (txb_info->qcoeff[ci] != 0) {
1414 *new_eob = si + 1;
1415 break;
1416 } else {
Jingning Handfd72322017-08-09 14:04:12 -07001417 cost_diff -= get_low_coeff_cost(ci, txb_cache, txb_costs, txb_info);
Angie Chiangc77799b2017-05-30 17:08:17 -07001418 }
1419 }
1420
1421 const int org_eob = txb_info->eob;
1422 set_eob(txb_info, *new_eob);
Angie Chiang25645b72017-09-24 14:28:49 -07001423 cost_diff += try_level_down(coeff_idx, txb_cache, txb_costs, txb_info, NULL,
1424 fast_mode);
Angie Chiangc77799b2017-05-30 17:08:17 -07001425 set_eob(txb_info, org_eob);
1426
1427 if (*new_eob > 0) {
1428 // Note that get_eob_ctx does NOT actually account for qcoeff, so we don't
1429 // need to lower down the qcoeff here
Linfeng Zhang7f7e89f2017-10-17 17:07:41 -07001430 const int eob_ctx =
1431 get_eob_ctx(scan[*new_eob - 1], txb_info->txs_ctx, txb_info->tx_type);
Jingning Handfd72322017-08-09 14:04:12 -07001432 cost_diff -= txb_costs->eob_cost[eob_ctx][0];
1433 cost_diff += txb_costs->eob_cost[eob_ctx][1];
Angie Chiangc77799b2017-05-30 17:08:17 -07001434 } else {
1435 const int txb_skip_ctx = txb_info->txb_ctx->txb_skip_ctx;
Jingning Handfd72322017-08-09 14:04:12 -07001436 cost_diff -= txb_costs->txb_skip_cost[txb_skip_ctx][0];
1437 cost_diff += txb_costs->txb_skip_cost[txb_skip_ctx][1];
Angie Chiangc77799b2017-05-30 17:08:17 -07001438 }
1439 return cost_diff;
1440}
Angie Chiang47e07072017-05-30 17:27:01 -07001441
Angie Chiangd19969e2017-05-30 18:02:33 -07001442// TODO(angiebird): add static to this function it's called
Linfeng Zhang1015a342017-10-24 16:20:41 -07001443void update_level_down(const int coeff_idx, TxbCache *const txb_cache,
1444 TxbInfo *const txb_info) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001445 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1446 const int abs_qc = abs(qc);
1447 if (qc == 0) return;
1448 const tran_low_t low_coeff = get_lower_coeff(qc);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001449 update_coeff(coeff_idx, low_coeff, txb_info);
Angie Chiangd19969e2017-05-30 18:02:33 -07001450
1451 const int row = coeff_idx >> txb_info->bwl;
1452 const int col = coeff_idx - (row << txb_info->bwl);
1453 const int eob = txb_info->eob;
1454 const int16_t *iscan = txb_info->scan_order->iscan;
1455 for (int i = 0; i < SIG_REF_OFFSET_NUM; ++i) {
1456 const int nb_row = row - sig_ref_offset[i][0];
1457 const int nb_col = col - sig_ref_offset[i][1];
Jingning Han3455e762017-06-13 21:15:44 -07001458
1459 if (!(nb_row >= 0 && nb_col >= 0 && nb_row < txb_info->height &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001460 nb_col < txb_info->width))
Jingning Han3455e762017-06-13 21:15:44 -07001461 continue;
1462
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001463 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Angie Chiangd19969e2017-05-30 18:02:33 -07001464 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001465 if (nb_scan_idx < eob) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001466 const int scan_idx = iscan[coeff_idx];
1467 if (scan_idx < nb_scan_idx) {
1468 const int level = 1;
1469 if (abs_qc == level) {
1470 txb_cache->nz_count_arr[nb_coeff_idx] -= 1;
1471 assert(txb_cache->nz_count_arr[nb_coeff_idx] >= 0);
1472 }
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001473 txb_cache->nz_ctx_arr[nb_coeff_idx] = get_nz_map_ctx_from_stats(
Dake He03a32922017-10-31 08:06:45 -07001474#if USE_CAUSAL_BASE_CTX
1475 0,
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001476#else
1477 txb_cache->nz_count_arr[nb_coeff_idx],
Dake He03a32922017-10-31 08:06:45 -07001478#endif
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001479 nb_coeff_idx, txb_info->bwl, txb_info->height, txb_info->tx_type);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001480 // int ref_ctx = get_nz_map_ctx(txb_info->levels, nb_coeff_idx,
Linfeng Zhangb6957c22017-10-25 13:17:28 -07001481 // txb_info->bwl, tx_type);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001482 // if (ref_ctx != txb_cache->nz_ctx_arr[nb_coeff_idx])
Angie Chiangd19969e2017-05-30 18:02:33 -07001483 // printf("nz ctx %d ref_ctx %d\n",
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001484 // txb_cache->nz_ctx_arr[nb_coeff_idx], ref_ctx);
Angie Chiangd19969e2017-05-30 18:02:33 -07001485 }
1486 }
1487 }
1488
Angie Chiangdef11252017-08-18 14:37:32 -07001489 const BASE_CTX_TABLE *base_ctx_table =
1490 txb_info->coeff_ctx_table->base_ctx_table;
Angie Chiangd19969e2017-05-30 18:02:33 -07001491 for (int i = 0; i < BASE_CONTEXT_POSITION_NUM; ++i) {
1492 const int nb_row = row - base_ref_offset[i][0];
1493 const int nb_col = col - base_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001494 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001495
1496 if (!(nb_row >= 0 && nb_col >= 0 && nb_row < txb_info->height &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001497 nb_col < txb_info->width))
Jingning Han3455e762017-06-13 21:15:44 -07001498 continue;
1499
Angie Chiangd19969e2017-05-30 18:02:33 -07001500 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1501 if (!has_base(nb_coeff, 0)) continue;
1502 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001503 if (nb_scan_idx < eob) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001504 if (row >= nb_row && col >= nb_col)
1505 update_mag_arr(txb_cache->base_mag_arr[nb_coeff_idx], abs_qc);
1506 const int mag =
1507 get_mag_from_mag_arr(txb_cache->base_mag_arr[nb_coeff_idx]);
1508 for (int base_idx = 0; base_idx < NUM_BASE_LEVELS; ++base_idx) {
1509 if (!has_base(nb_coeff, base_idx)) continue;
1510 const int level = base_idx + 1;
1511 if (abs_qc == level) {
1512 txb_cache->base_count_arr[base_idx][nb_coeff_idx] -= 1;
1513 assert(txb_cache->base_count_arr[base_idx][nb_coeff_idx] >= 0);
1514 }
1515 const int count = txb_cache->base_count_arr[base_idx][nb_coeff_idx];
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001516 txb_cache->base_ctx_arr[base_idx][nb_coeff_idx] =
Angie Chiangdef11252017-08-18 14:37:32 -07001517 base_ctx_table[nb_row != 0][nb_col != 0][mag > level][count];
Angie Chiangd19969e2017-05-30 18:02:33 -07001518 // int ref_ctx = get_base_ctx(txb_info->qcoeff, nb_coeff_idx,
Linfeng Zhang8c841f72017-10-18 15:32:29 -07001519 // txb_info->bwl, level - 1);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001520 // if (ref_ctx != txb_cache->base_ctx_arr[base_idx][nb_coeff_idx]) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001521 // printf("base ctx %d ref_ctx %d\n",
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001522 // txb_cache->base_ctx_arr[base_idx][nb_coeff_idx], ref_ctx);
Angie Chiangd19969e2017-05-30 18:02:33 -07001523 // }
1524 }
1525 }
1526 }
1527
1528 for (int i = 0; i < BR_CONTEXT_POSITION_NUM; ++i) {
1529 const int nb_row = row - br_ref_offset[i][0];
1530 const int nb_col = col - br_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001531 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001532
1533 if (!(nb_row >= 0 && nb_col >= 0 && nb_row < txb_info->height &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001534 nb_col < txb_info->width))
Jingning Han3455e762017-06-13 21:15:44 -07001535 continue;
1536
Angie Chiangd19969e2017-05-30 18:02:33 -07001537 const int nb_scan_idx = iscan[nb_coeff_idx];
1538 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1539 if (!has_br(nb_coeff)) continue;
Jingning Han3455e762017-06-13 21:15:44 -07001540 if (nb_scan_idx < eob) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001541 const int level = 1 + NUM_BASE_LEVELS;
1542 if (abs_qc == level) {
1543 txb_cache->br_count_arr[nb_coeff_idx] -= 1;
1544 assert(txb_cache->br_count_arr[nb_coeff_idx] >= 0);
1545 }
1546 if (row >= nb_row && col >= nb_col)
1547 update_mag_arr(txb_cache->br_mag_arr[nb_coeff_idx], abs_qc);
1548 const int count = txb_cache->br_count_arr[nb_coeff_idx];
1549 const int mag = get_mag_from_mag_arr(txb_cache->br_mag_arr[nb_coeff_idx]);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001550 txb_cache->br_ctx_arr[nb_coeff_idx] =
Angie Chiangd19969e2017-05-30 18:02:33 -07001551 get_br_ctx_from_count_mag(nb_row, nb_col, count, mag);
1552 // int ref_ctx = get_level_ctx(txb_info->qcoeff, nb_coeff_idx,
1553 // txb_info->bwl);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001554 // if (ref_ctx != txb_cache->br_ctx_arr[nb_coeff_idx]) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001555 // printf("base ctx %d ref_ctx %d\n",
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001556 // txb_cache->br_ctx_arr[nb_coeff_idx], ref_ctx);
Angie Chiangd19969e2017-05-30 18:02:33 -07001557 // }
1558 }
1559 }
1560}
1561
Linfeng Zhang1015a342017-10-24 16:20:41 -07001562static int get_coeff_cost(const tran_low_t qc, const int scan_idx,
Ola Hugosson13892102017-11-06 08:01:44 +01001563#if CONFIG_LV_MAP_MULTI
1564 const int is_eob,
1565#endif
Linfeng Zhang1015a342017-10-24 16:20:41 -07001566 const TxbInfo *const txb_info,
1567 const LV_MAP_COEFF_COST *const txb_costs) {
Angie Chiang488f9212017-05-30 12:46:26 -07001568 const TXB_CTX *txb_ctx = txb_info->txb_ctx;
1569 const int is_nz = (qc != 0);
1570 const tran_low_t abs_qc = abs(qc);
1571 int cost = 0;
1572 const int16_t *scan = txb_info->scan_order->scan;
Dake He03a32922017-10-31 08:06:45 -07001573
Ola Hugosson13892102017-11-06 08:01:44 +01001574#if CONFIG_LV_MAP_MULTI
1575 int coeff_ctx =
1576 get_nz_map_ctx(txb_info->levels, scan_idx, scan, txb_info->bwl,
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001577 txb_info->height, is_eob, txb_info->tx_type);
Dake He3fe369c2017-11-16 17:56:44 -08001578#if USE_BASE_EOB_ALPHABET
1579 if (is_eob) {
1580 cost +=
1581 txb_costs->base_eob_cost[coeff_ctx - SIG_COEF_CONTEXTS +
1582 SIG_COEF_CONTEXTS_EOB][AOMMIN(abs_qc, 3) - 1];
1583 } else {
1584 cost += txb_costs->base_cost[coeff_ctx][AOMMIN(abs_qc, 3)];
1585 }
1586
1587#else
Ola Hugosson13892102017-11-06 08:01:44 +01001588 cost += txb_costs->base_cost[coeff_ctx][AOMMIN(abs_qc, 3)];
Dake He3fe369c2017-11-16 17:56:44 -08001589#endif
Ola Hugosson13892102017-11-06 08:01:44 +01001590#else
Dake He03a32922017-10-31 08:06:45 -07001591#if USE_CAUSAL_BASE_CTX
1592 int coeff_ctx =
Linfeng Zhangb6957c22017-10-25 13:17:28 -07001593 get_nz_map_ctx(txb_info->levels, scan_idx, scan, txb_info->bwl,
1594 txb_info->height, txb_info->tx_type);
Dake He03a32922017-10-31 08:06:45 -07001595#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07001596 if (scan_idx < txb_info->eob - 1) {
Dake He03a32922017-10-31 08:06:45 -07001597#if !USE_CAUSAL_BASE_CTX
Angie Chiange9125ba2017-08-20 12:03:41 -07001598 int coeff_ctx =
Linfeng Zhang1015a342017-10-24 16:20:41 -07001599 get_nz_map_ctx(txb_info->levels, scan_idx, scan, txb_info->bwl,
Linfeng Zhangb6957c22017-10-25 13:17:28 -07001600 txb_info->height, txb_info->tx_type);
Dake He03a32922017-10-31 08:06:45 -07001601#endif
Jingning Handfd72322017-08-09 14:04:12 -07001602 cost += txb_costs->nz_map_cost[coeff_ctx][is_nz];
Angie Chiang488f9212017-05-30 12:46:26 -07001603 }
Ola Hugosson13892102017-11-06 08:01:44 +01001604#endif
Angie Chiang488f9212017-05-30 12:46:26 -07001605 if (is_nz) {
Jingning Handfd72322017-08-09 14:04:12 -07001606 cost += get_sign_bit_cost(qc, scan_idx, txb_costs->dc_sign_cost,
Angie Chiang488f9212017-05-30 12:46:26 -07001607 txb_ctx->dc_sign_ctx);
1608
Ola Hugosson13892102017-11-06 08:01:44 +01001609#if !CONFIG_LV_MAP_MULTI
Dake He03a32922017-10-31 08:06:45 -07001610#if USE_CAUSAL_BASE_CTX
1611 int k;
1612 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
1613 int ctx = coeff_ctx;
1614 int is_k = (abs_qc > (k + 1));
1615
1616 cost += txb_costs->base_cost[k][ctx][is_k];
1617 if (is_k == 0) break;
1618 }
1619#else
Angie Chiang488f9212017-05-30 12:46:26 -07001620 int ctx_ls[NUM_BASE_LEVELS] = { 0 };
Linfeng Zhang679d81e2017-10-31 15:27:42 -07001621 get_base_ctx_set(txb_info->levels, scan[scan_idx], txb_info->bwl, ctx_ls);
Angie Chiang488f9212017-05-30 12:46:26 -07001622
1623 int i;
1624 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
Jingning Handfd72322017-08-09 14:04:12 -07001625 cost += get_base_cost(abs_qc, ctx_ls[i],
1626 txb_costs->base_cost[i][ctx_ls[i]], i);
Angie Chiang488f9212017-05-30 12:46:26 -07001627 }
Dake He03a32922017-10-31 08:06:45 -07001628#endif
Ola Hugosson13892102017-11-06 08:01:44 +01001629#endif
Angie Chiang488f9212017-05-30 12:46:26 -07001630 if (abs_qc > NUM_BASE_LEVELS) {
Linfeng Zhang97fc4742017-11-07 12:57:25 -08001631 const int row = scan[scan_idx] >> txb_info->bwl;
1632 const int col = scan[scan_idx] - (row << txb_info->bwl);
Dake He7d01ab52017-11-24 17:53:28 -08001633
1634#if CONFIG_LV_MAP_MULTI && USE_CAUSAL_BR_CTX
1635 (void)col;
1636 const int count = 0;
1637 const int ctx = get_br_ctx(txb_info->levels, scan[scan_idx],
1638 txb_info->bwl, count, txb_info->tx_type);
1639#else
Linfeng Zhang97fc4742017-11-07 12:57:25 -08001640 const int count = get_level_count(
1641 txb_info->levels, (1 << txb_info->bwl) + TX_PAD_HOR, row, col,
1642 NUM_BASE_LEVELS, br_ref_offset, BR_CONTEXT_POSITION_NUM);
1643 const int ctx =
1644 get_br_ctx(txb_info->levels, scan[scan_idx], txb_info->bwl, count);
Dake He7d01ab52017-11-24 17:53:28 -08001645#endif
Jingning Handfd72322017-08-09 14:04:12 -07001646 cost += get_br_cost(abs_qc, ctx, txb_costs->lps_cost[ctx]);
Angie Chiang488f9212017-05-30 12:46:26 -07001647 cost += get_golomb_cost(abs_qc);
1648 }
Angie Chiang488f9212017-05-30 12:46:26 -07001649 }
1650 return cost;
1651}
1652
Angie Chiang47e07072017-05-30 17:27:01 -07001653#if TEST_OPTIMIZE_TXB
Angie Chiang5e012fe2017-05-30 18:47:39 -07001654#define ALL_REF_OFFSET_NUM 17
Linfeng Zhang4afda452017-10-24 10:34:04 -07001655static const int all_ref_offset[ALL_REF_OFFSET_NUM][2] = {
Angie Chiang5e012fe2017-05-30 18:47:39 -07001656 { 0, 0 }, { -2, -1 }, { -2, 0 }, { -2, 1 }, { -1, -2 }, { -1, -1 },
1657 { -1, 0 }, { -1, 1 }, { 0, -2 }, { 0, -1 }, { 1, -2 }, { 1, -1 },
1658 { 1, 0 }, { 2, 0 }, { 0, 1 }, { 0, 2 }, { 1, 1 },
1659};
1660
Angie Chiang0b2795c2017-09-29 16:00:08 -07001661static int try_level_down_ref(int coeff_idx, const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang5e012fe2017-05-30 18:47:39 -07001662 TxbInfo *txb_info,
1663 int (*cost_map)[COST_MAP_SIZE]) {
1664 if (cost_map) {
1665 for (int i = 0; i < COST_MAP_SIZE; ++i) av1_zero(cost_map[i]);
1666 }
1667 tran_low_t qc = txb_info->qcoeff[coeff_idx];
1668 if (qc == 0) return 0;
1669 int row = coeff_idx >> txb_info->bwl;
1670 int col = coeff_idx - (row << txb_info->bwl);
1671 int org_cost = 0;
1672 for (int i = 0; i < ALL_REF_OFFSET_NUM; ++i) {
1673 int nb_row = row - all_ref_offset[i][0];
1674 int nb_col = col - all_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001675 int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Angie Chiang5e012fe2017-05-30 18:47:39 -07001676 int nb_scan_idx = txb_info->scan_order->iscan[nb_coeff_idx];
1677 if (nb_scan_idx < txb_info->eob && nb_row >= 0 && nb_col >= 0 &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001678 nb_row < txb_info->height && nb_col < txb_info->width) {
Angie Chiang5e012fe2017-05-30 18:47:39 -07001679 tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
Ola Hugosson13892102017-11-06 08:01:44 +01001680#if CONFIG_LV_MAP_MULTI
1681 int cost =
1682 get_coeff_cost(nb_coeff, nb_scan_idx, is_eob, txb_info, txb_costs);
1683#else
Angie Chiang0b2795c2017-09-29 16:00:08 -07001684 int cost = get_coeff_cost(nb_coeff, nb_scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +01001685#endif
Angie Chiang5e012fe2017-05-30 18:47:39 -07001686 if (cost_map)
1687 cost_map[nb_row - row + COST_MAP_OFFSET]
1688 [nb_col - col + COST_MAP_OFFSET] -= cost;
1689 org_cost += cost;
1690 }
1691 }
Linfeng Zhang1015a342017-10-24 16:20:41 -07001692 update_qcoeff(coeff_idx, get_lower_coeff(qc), txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001693 int new_cost = 0;
1694 for (int i = 0; i < ALL_REF_OFFSET_NUM; ++i) {
1695 int nb_row = row - all_ref_offset[i][0];
1696 int nb_col = col - all_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001697 int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Angie Chiang5e012fe2017-05-30 18:47:39 -07001698 int nb_scan_idx = txb_info->scan_order->iscan[nb_coeff_idx];
1699 if (nb_scan_idx < txb_info->eob && nb_row >= 0 && nb_col >= 0 &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001700 nb_row < txb_info->height && nb_col < txb_info->width) {
Angie Chiang5e012fe2017-05-30 18:47:39 -07001701 tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
Ola Hugosson13892102017-11-06 08:01:44 +01001702#if CONFIG_LV_MAP_MULTI
1703 int cost =
1704 get_coeff_cost(nb_coeff, nb_scan_idx, is_eob, txb_info, txb_costs);
1705#else
Angie Chiang0b2795c2017-09-29 16:00:08 -07001706 int cost = get_coeff_cost(nb_coeff, nb_scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +01001707#endif
Angie Chiang5e012fe2017-05-30 18:47:39 -07001708 if (cost_map)
1709 cost_map[nb_row - row + COST_MAP_OFFSET]
1710 [nb_col - col + COST_MAP_OFFSET] += cost;
1711 new_cost += cost;
1712 }
1713 }
Linfeng Zhang1015a342017-10-24 16:20:41 -07001714 update_qcoeff(coeff_idx, qc, txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001715 return new_cost - org_cost;
1716}
1717
1718static void test_level_down(int coeff_idx, const TxbCache *txb_cache,
Angie Chiang0b2795c2017-09-29 16:00:08 -07001719 const LV_MAP_COEFF_COST *txb_costs,
1720 TxbInfo *txb_info) {
Angie Chiang47e07072017-05-30 17:27:01 -07001721 int cost_map[COST_MAP_SIZE][COST_MAP_SIZE];
1722 int ref_cost_map[COST_MAP_SIZE][COST_MAP_SIZE];
1723 const int cost_diff =
Angie Chiang0b2795c2017-09-29 16:00:08 -07001724 try_level_down(coeff_idx, txb_cache, txb_costs, txb_info, cost_map, 0);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001725 const int cost_diff_ref =
Angie Chiang0b2795c2017-09-29 16:00:08 -07001726 try_level_down_ref(coeff_idx, txb_costs, txb_info, ref_cost_map);
Angie Chiang47e07072017-05-30 17:27:01 -07001727 if (cost_diff != cost_diff_ref) {
1728 printf("qc %d cost_diff %d cost_diff_ref %d\n", txb_info->qcoeff[coeff_idx],
1729 cost_diff, cost_diff_ref);
1730 for (int r = 0; r < COST_MAP_SIZE; ++r) {
1731 for (int c = 0; c < COST_MAP_SIZE; ++c) {
1732 printf("%d:%d ", cost_map[r][c], ref_cost_map[r][c]);
1733 }
1734 printf("\n");
1735 }
1736 }
1737}
1738#endif
1739
Angie Chiang488f9212017-05-30 12:46:26 -07001740// TODO(angiebird): make this static once it's called
Jingning Handfd72322017-08-09 14:04:12 -07001741int get_txb_cost(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs) {
Angie Chiang488f9212017-05-30 12:46:26 -07001742 int cost = 0;
1743 int txb_skip_ctx = txb_info->txb_ctx->txb_skip_ctx;
1744 const int16_t *scan = txb_info->scan_order->scan;
1745 if (txb_info->eob == 0) {
Jingning Handfd72322017-08-09 14:04:12 -07001746 cost = txb_costs->txb_skip_cost[txb_skip_ctx][1];
Angie Chiang488f9212017-05-30 12:46:26 -07001747 return cost;
1748 }
Jingning Handfd72322017-08-09 14:04:12 -07001749 cost = txb_costs->txb_skip_cost[txb_skip_ctx][0];
Angie Chiang488f9212017-05-30 12:46:26 -07001750 for (int c = 0; c < txb_info->eob; ++c) {
1751 tran_low_t qc = txb_info->qcoeff[scan[c]];
Ola Hugosson13892102017-11-06 08:01:44 +01001752#if CONFIG_LV_MAP_MULTI
1753 int coeff_cost =
1754 get_coeff_cost(qc, c, c == txb_info->eob - 1, txb_info, txb_costs);
1755#else
Jingning Handfd72322017-08-09 14:04:12 -07001756 int coeff_cost = get_coeff_cost(qc, c, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +01001757#endif
Angie Chiang488f9212017-05-30 12:46:26 -07001758 cost += coeff_cost;
1759 }
1760 return cost;
1761}
1762
Angie Chiang5e012fe2017-05-30 18:47:39 -07001763#if TEST_OPTIMIZE_TXB
Angie Chiang0b2795c2017-09-29 16:00:08 -07001764void test_try_change_eob(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang5e012fe2017-05-30 18:47:39 -07001765 TxbCache *txb_cache) {
1766 int eob = txb_info->eob;
1767 const int16_t *scan = txb_info->scan_order->scan;
1768 if (eob > 0) {
1769 int last_si = eob - 1;
1770 int last_ci = scan[last_si];
1771 int last_coeff = txb_info->qcoeff[last_ci];
1772 if (abs(last_coeff) == 1) {
1773 int new_eob;
1774 int cost_diff =
Angie Chiang0b2795c2017-09-29 16:00:08 -07001775 try_change_eob(&new_eob, last_ci, txb_cache, txb_costs, txb_info, 0);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001776 int org_eob = txb_info->eob;
Angie Chiang0b2795c2017-09-29 16:00:08 -07001777 int cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001778
Linfeng Zhang1015a342017-10-24 16:20:41 -07001779 update_qcoeff(last_ci, get_lower_coeff(last_coeff), txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001780 set_eob(txb_info, new_eob);
Angie Chiang0b2795c2017-09-29 16:00:08 -07001781 int new_cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001782 set_eob(txb_info, org_eob);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001783 update_qcoeff(last_ci, last_coeff, txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001784
1785 int ref_cost_diff = -cost + new_cost;
1786 if (cost_diff != ref_cost_diff)
1787 printf("org_eob %d new_eob %d cost_diff %d ref_cost_diff %d\n", org_eob,
1788 new_eob, cost_diff, ref_cost_diff);
1789 }
1790 }
1791}
1792#endif
1793
Angie Chiang47e07072017-05-30 17:27:01 -07001794void try_level_down_facade(LevelDownStats *stats, int scan_idx,
Jingning Handfd72322017-08-09 14:04:12 -07001795 const TxbCache *txb_cache,
1796 const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang25645b72017-09-24 14:28:49 -07001797 TxbInfo *txb_info, int fast_mode) {
Angie Chiang47e07072017-05-30 17:27:01 -07001798 const int16_t *scan = txb_info->scan_order->scan;
1799 const int coeff_idx = scan[scan_idx];
1800 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1801 stats->new_eob = -1;
1802 stats->update = 0;
1803 if (qc == 0) {
1804 return;
1805 }
1806
1807 const tran_low_t tqc = txb_info->tcoeff[coeff_idx];
1808 const int dqv = txb_info->dequant[coeff_idx != 0];
1809
1810 const tran_low_t dqc = qcoeff_to_dqcoeff(qc, dqv, txb_info->shift);
Jingning Han641c1e52017-10-13 09:42:49 -07001811
1812 if (scan_idx != txb_info->eob - 1)
1813 if (abs(dqc) < abs(tqc)) return;
1814
Angie Chiang47e07072017-05-30 17:27:01 -07001815 const int64_t dqc_dist = get_coeff_dist(tqc, dqc, txb_info->shift);
1816
1817 stats->low_qc = get_lower_coeff(qc);
1818 stats->low_dqc = qcoeff_to_dqcoeff(stats->low_qc, dqv, txb_info->shift);
1819 const int64_t low_dqc_dist =
1820 get_coeff_dist(tqc, stats->low_dqc, txb_info->shift);
1821
1822 stats->dist_diff = -dqc_dist + low_dqc_dist;
1823 stats->cost_diff = 0;
1824 stats->new_eob = txb_info->eob;
1825 if (scan_idx == txb_info->eob - 1 && abs(qc) == 1) {
1826 stats->cost_diff = try_change_eob(&stats->new_eob, coeff_idx, txb_cache,
Angie Chiang25645b72017-09-24 14:28:49 -07001827 txb_costs, txb_info, fast_mode);
Angie Chiang47e07072017-05-30 17:27:01 -07001828 } else {
Angie Chiang25645b72017-09-24 14:28:49 -07001829 stats->cost_diff = try_level_down(coeff_idx, txb_cache, txb_costs, txb_info,
1830 NULL, fast_mode);
Angie Chiang47e07072017-05-30 17:27:01 -07001831#if TEST_OPTIMIZE_TXB
Angie Chiang0b2795c2017-09-29 16:00:08 -07001832 test_level_down(coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chiang47e07072017-05-30 17:27:01 -07001833#endif
1834 }
Urvang Joshi70006e42017-06-14 16:08:55 -07001835 stats->rd_diff = RDCOST(txb_info->rdmult, stats->cost_diff, stats->dist_diff);
Angie Chiang47e07072017-05-30 17:27:01 -07001836 if (stats->rd_diff < 0) stats->update = 1;
1837 return;
1838}
Angie Chiang07c57f32017-05-30 18:18:33 -07001839
Jingning Han3422ac12017-10-25 20:37:53 -07001840#if 1
Dake Hea47cd6c2017-10-13 18:09:58 -07001841static int optimize_txb(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs,
1842 TxbCache *txb_cache, int dry_run, int fast_mode) {
1843 (void)fast_mode;
1844 (void)txb_cache;
1845 int update = 0;
Dake He59881772017-11-24 07:00:02 -08001846 // return update; // TODO(DKHE): training only.
Dake Hea47cd6c2017-10-13 18:09:58 -07001847 if (txb_info->eob == 0) return update;
Urvang Joshi80893152017-10-27 11:51:14 -07001848 const int max_eob = av1_get_max_eob(txb_info->tx_size);
Dake Hea47cd6c2017-10-13 18:09:58 -07001849
1850#if TEST_OPTIMIZE_TXB
1851 int64_t sse;
1852 int64_t org_dist =
1853 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
1854 (1 << (2 * txb_info->shift));
1855 int org_cost = get_txb_cost(txb_info, txb_probs);
1856#endif
1857
1858 tran_low_t *org_qcoeff = txb_info->qcoeff;
1859 tran_low_t *org_dqcoeff = txb_info->dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07001860 uint8_t *const org_levels = txb_info->levels;
Dake Hea47cd6c2017-10-13 18:09:58 -07001861
1862 tran_low_t tmp_qcoeff[MAX_TX_SQUARE];
1863 tran_low_t tmp_dqcoeff[MAX_TX_SQUARE];
Linfeng Zhang679d81e2017-10-31 15:27:42 -07001864 uint8_t tmp_levels_buf[TX_PAD_2D];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001865 uint8_t *const tmp_levels = set_levels(tmp_levels_buf, txb_info->width);
Dake Hea47cd6c2017-10-13 18:09:58 -07001866 const int org_eob = txb_info->eob;
1867 if (dry_run) {
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001868 const int stride = txb_info->width + TX_PAD_HOR;
1869 const int levels_size =
1870
1871 (stride * (txb_info->height + TX_PAD_VER) + TX_PAD_END);
Dake Hea47cd6c2017-10-13 18:09:58 -07001872 memcpy(tmp_qcoeff, org_qcoeff, sizeof(org_qcoeff[0]) * max_eob);
1873 memcpy(tmp_dqcoeff, org_dqcoeff, sizeof(org_dqcoeff[0]) * max_eob);
Linfeng Zhang679d81e2017-10-31 15:27:42 -07001874 memcpy(tmp_levels, org_levels - TX_PAD_TOP * stride,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001875 sizeof(org_levels[0]) * levels_size);
Dake Hea47cd6c2017-10-13 18:09:58 -07001876 txb_info->qcoeff = tmp_qcoeff;
1877 txb_info->dqcoeff = tmp_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07001878 txb_info->levels = tmp_levels;
Dake Hea47cd6c2017-10-13 18:09:58 -07001879 }
1880
1881 const int16_t *scan = txb_info->scan_order->scan;
1882
1883 // forward optimize the nz_map`
1884 const int init_eob = txb_info->eob;
1885 const int seg_eob = txb_info->seg_eob;
Jingning Han35deaa72017-10-26 15:36:30 -07001886 int eob_cost = get_eob_cost(init_eob, seg_eob, txb_costs, txb_info->tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -07001887
1888 // backward optimize the level-k map
1889 int64_t accu_rate = eob_cost;
1890 int64_t accu_dist = 0;
1891 int64_t prev_eob_rd_cost = INT64_MAX;
1892 int64_t cur_eob_rd_cost = 0;
1893
1894 for (int si = init_eob - 1; si >= 0; --si) {
1895 const int coeff_idx = scan[si];
1896 tran_low_t qc = txb_info->qcoeff[coeff_idx];
1897
1898 LevelDownStats stats;
Ola Hugosson13892102017-11-06 08:01:44 +01001899#if CONFIG_LV_MAP_MULTI
1900 get_dist_cost_stats(&stats, si, si == init_eob - 1, txb_costs, txb_info);
1901#else
Dake Hea47cd6c2017-10-13 18:09:58 -07001902 get_dist_cost_stats(&stats, si, txb_costs, txb_info);
Ola Hugosson13892102017-11-06 08:01:44 +01001903#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07001904
1905 if (qc == 0) {
1906 accu_rate += stats.rate;
1907 } else {
1908 // check if it is better to make this the last significant coefficient
Jingning Han35deaa72017-10-26 15:36:30 -07001909 int cur_eob_rate =
1910 get_eob_cost(si + 1, seg_eob, txb_costs, txb_info->tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -07001911 cur_eob_rd_cost = RDCOST(txb_info->rdmult, cur_eob_rate, 0);
1912 prev_eob_rd_cost =
1913 RDCOST(txb_info->rdmult, accu_rate + stats.nz_rate, accu_dist);
1914 if (cur_eob_rd_cost <= prev_eob_rd_cost) {
1915 update = 1;
1916 for (int j = si + 1; j < txb_info->eob; j++) {
1917 const int coeff_pos_j = scan[j];
1918 update_coeff(coeff_pos_j, 0, txb_info);
1919 }
1920 txb_info->eob = si + 1;
1921
Ola Hugosson13892102017-11-06 08:01:44 +01001922 // rerun cost calculation due to change of eob
Dake Hea47cd6c2017-10-13 18:09:58 -07001923 accu_rate = cur_eob_rate;
1924 accu_dist = 0;
Ola Hugosson13892102017-11-06 08:01:44 +01001925#if CONFIG_LV_MAP_MULTI
1926 get_dist_cost_stats(&stats, si, 1, txb_costs, txb_info);
1927#else
Dake Hea47cd6c2017-10-13 18:09:58 -07001928 get_dist_cost_stats(&stats, si, txb_costs, txb_info);
Ola Hugosson13892102017-11-06 08:01:44 +01001929#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07001930 }
1931
1932 int bUpdCoeff = 0;
1933 if (stats.rd_low < stats.rd) {
1934 if ((stats.low_qc != 0) || (si < txb_info->eob - 1)) {
1935 bUpdCoeff = 1;
1936 update = 1;
1937 }
1938 }
1939
1940 if (bUpdCoeff) {
1941 update_coeff(coeff_idx, stats.low_qc, txb_info);
1942 accu_rate += stats.rate_low;
1943 accu_dist += stats.dist_low;
1944 } else {
1945 accu_rate += stats.rate;
1946 accu_dist += stats.dist;
1947 }
1948 }
1949 } // for (si)
1950 int non_zero_blk_rate =
1951 txb_costs->txb_skip_cost[txb_info->txb_ctx->txb_skip_ctx][0];
1952 prev_eob_rd_cost =
1953 RDCOST(txb_info->rdmult, accu_rate + non_zero_blk_rate, accu_dist);
1954
1955 int zero_blk_rate =
1956 txb_costs->txb_skip_cost[txb_info->txb_ctx->txb_skip_ctx][1];
1957 int64_t zero_blk_rd_cost = RDCOST(txb_info->rdmult, zero_blk_rate, 0);
1958 if (zero_blk_rd_cost <= prev_eob_rd_cost) {
1959 update = 1;
1960 for (int j = 0; j < txb_info->eob; j++) {
1961 const int coeff_pos_j = scan[j];
1962 update_coeff(coeff_pos_j, 0, txb_info);
1963 }
1964 txb_info->eob = 0;
1965 }
1966
1967#if TEST_OPTIMIZE_TXB
1968 int cost_diff = 0;
1969 int64_t dist_diff = 0;
1970 int64_t rd_diff = 0;
1971 int64_t new_dist =
1972 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
1973 (1 << (2 * txb_info->shift));
1974 int new_cost = get_txb_cost(txb_info, txb_probs);
1975 int64_t ref_dist_diff = new_dist - org_dist;
1976 int ref_cost_diff = new_cost - org_cost;
1977 if (cost_diff != ref_cost_diff || dist_diff != ref_dist_diff)
1978 printf(
1979 "overall rd_diff %ld\ncost_diff %d ref_cost_diff%d\ndist_diff %ld "
1980 "ref_dist_diff %ld\neob %d new_eob %d\n\n",
1981 rd_diff, cost_diff, ref_cost_diff, dist_diff, ref_dist_diff, org_eob,
1982 txb_info->eob);
1983#endif
1984 if (dry_run) {
1985 txb_info->qcoeff = org_qcoeff;
1986 txb_info->dqcoeff = org_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07001987 txb_info->levels = org_levels;
Dake Hea47cd6c2017-10-13 18:09:58 -07001988 set_eob(txb_info, org_eob);
1989 }
1990 return update;
1991}
1992
1993#else
Jingning Handfd72322017-08-09 14:04:12 -07001994static int optimize_txb(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang25645b72017-09-24 14:28:49 -07001995 TxbCache *txb_cache, int dry_run, int fast_mode) {
Angie Chiang07c57f32017-05-30 18:18:33 -07001996 int update = 0;
1997 if (txb_info->eob == 0) return update;
1998 int cost_diff = 0;
1999 int64_t dist_diff = 0;
2000 int64_t rd_diff = 0;
Urvang Joshi80893152017-10-27 11:51:14 -07002001 const int max_eob = av1_get_max_eob(txb_info->tx_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002002
2003#if TEST_OPTIMIZE_TXB
2004 int64_t sse;
2005 int64_t org_dist =
2006 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
2007 (1 << (2 * txb_info->shift));
Angie Chiang0b2795c2017-09-29 16:00:08 -07002008 int org_cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang07c57f32017-05-30 18:18:33 -07002009#endif
2010
2011 tran_low_t *org_qcoeff = txb_info->qcoeff;
2012 tran_low_t *org_dqcoeff = txb_info->dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002013 uint8_t *const org_levels = txb_info->levels;
Angie Chiang07c57f32017-05-30 18:18:33 -07002014
2015 tran_low_t tmp_qcoeff[MAX_TX_SQUARE];
2016 tran_low_t tmp_dqcoeff[MAX_TX_SQUARE];
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002017 uint8_t tmp_levels_buf[TX_PAD_2D];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002018 uint8_t *const tmp_levels = set_levels(tmp_levels_buf, txb_info->width);
Angie Chiang07c57f32017-05-30 18:18:33 -07002019 const int org_eob = txb_info->eob;
2020 if (dry_run) {
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002021 const int stride = txb_info->width + TX_PAD_HOR;
2022 const int levels_size =
2023
2024 (stride * (txb_info->height + TX_PAD_VER) + TX_PAD_END);
Angie Chiang07c57f32017-05-30 18:18:33 -07002025 memcpy(tmp_qcoeff, org_qcoeff, sizeof(org_qcoeff[0]) * max_eob);
2026 memcpy(tmp_dqcoeff, org_dqcoeff, sizeof(org_dqcoeff[0]) * max_eob);
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002027 memcpy(tmp_levels, org_levels - TX_PAD_TOP * stride,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002028 sizeof(org_levels[0]) * levels_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002029 txb_info->qcoeff = tmp_qcoeff;
2030 txb_info->dqcoeff = tmp_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002031 txb_info->levels = tmp_levels;
Angie Chiang07c57f32017-05-30 18:18:33 -07002032 }
2033
2034 const int16_t *scan = txb_info->scan_order->scan;
2035
2036 // forward optimize the nz_map
2037 const int cur_eob = txb_info->eob;
2038 for (int si = 0; si < cur_eob; ++si) {
2039 const int coeff_idx = scan[si];
2040 tran_low_t qc = txb_info->qcoeff[coeff_idx];
2041 if (abs(qc) == 1) {
2042 LevelDownStats stats;
Angie Chiang25645b72017-09-24 14:28:49 -07002043 try_level_down_facade(&stats, si, txb_cache, txb_costs, txb_info,
2044 fast_mode);
Angie Chiang07c57f32017-05-30 18:18:33 -07002045 if (stats.update) {
2046 update = 1;
2047 cost_diff += stats.cost_diff;
2048 dist_diff += stats.dist_diff;
2049 rd_diff += stats.rd_diff;
2050 update_level_down(coeff_idx, txb_cache, txb_info);
2051 set_eob(txb_info, stats.new_eob);
2052 }
2053 }
2054 }
2055
2056 // backward optimize the level-k map
Angie Chiang530b3042017-08-17 15:08:58 -07002057 int eob_fix = 0;
Angie Chiang07c57f32017-05-30 18:18:33 -07002058 for (int si = txb_info->eob - 1; si >= 0; --si) {
Angie Chiang530b3042017-08-17 15:08:58 -07002059 const int coeff_idx = scan[si];
2060 if (eob_fix == 1 && txb_info->qcoeff[coeff_idx] == 1) {
2061 // when eob is fixed, there is not need to optimize again when
2062 // abs(qc) == 1
2063 continue;
2064 }
Angie Chiang07c57f32017-05-30 18:18:33 -07002065 LevelDownStats stats;
Angie Chiang25645b72017-09-24 14:28:49 -07002066 try_level_down_facade(&stats, si, txb_cache, txb_costs, txb_info,
2067 fast_mode);
Angie Chiang07c57f32017-05-30 18:18:33 -07002068 if (stats.update) {
2069#if TEST_OPTIMIZE_TXB
2070// printf("si %d low_qc %d cost_diff %d dist_diff %ld rd_diff %ld eob %d new_eob
2071// %d\n", si, stats.low_qc, stats.cost_diff, stats.dist_diff, stats.rd_diff,
2072// txb_info->eob, stats.new_eob);
2073#endif
2074 update = 1;
2075 cost_diff += stats.cost_diff;
2076 dist_diff += stats.dist_diff;
2077 rd_diff += stats.rd_diff;
2078 update_level_down(coeff_idx, txb_cache, txb_info);
2079 set_eob(txb_info, stats.new_eob);
2080 }
Angie Chiang530b3042017-08-17 15:08:58 -07002081 if (eob_fix == 0 && txb_info->qcoeff[coeff_idx] != 0) eob_fix = 1;
Angie Chiang07c57f32017-05-30 18:18:33 -07002082 if (si > txb_info->eob) si = txb_info->eob;
2083 }
2084#if TEST_OPTIMIZE_TXB
2085 int64_t new_dist =
2086 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
2087 (1 << (2 * txb_info->shift));
Angie Chiang0b2795c2017-09-29 16:00:08 -07002088 int new_cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang07c57f32017-05-30 18:18:33 -07002089 int64_t ref_dist_diff = new_dist - org_dist;
2090 int ref_cost_diff = new_cost - org_cost;
2091 if (cost_diff != ref_cost_diff || dist_diff != ref_dist_diff)
2092 printf(
2093 "overall rd_diff %ld\ncost_diff %d ref_cost_diff%d\ndist_diff %ld "
2094 "ref_dist_diff %ld\neob %d new_eob %d\n\n",
2095 rd_diff, cost_diff, ref_cost_diff, dist_diff, ref_dist_diff, org_eob,
2096 txb_info->eob);
2097#endif
2098 if (dry_run) {
2099 txb_info->qcoeff = org_qcoeff;
2100 txb_info->dqcoeff = org_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002101 txb_info->levels = org_levels;
Angie Chiang07c57f32017-05-30 18:18:33 -07002102 set_eob(txb_info, org_eob);
2103 }
2104 return update;
2105}
Dake Hea47cd6c2017-10-13 18:09:58 -07002106#endif
Angie Chiang07c57f32017-05-30 18:18:33 -07002107
2108// These numbers are empirically obtained.
2109static const int plane_rd_mult[REF_TYPES][PLANE_TYPES] = {
Angie Chiang07c57f32017-05-30 18:18:33 -07002110 { 17, 13 }, { 16, 10 },
Angie Chiang07c57f32017-05-30 18:18:33 -07002111};
2112
Jingning Han7eab9ff2017-07-06 10:12:54 -07002113int av1_optimize_txb(const AV1_COMMON *cm, MACROBLOCK *x, int plane,
2114 int blk_row, int blk_col, int block, TX_SIZE tx_size,
Angie Chiang25645b72017-09-24 14:28:49 -07002115 TXB_CTX *txb_ctx, int fast_mode) {
Angie Chiang07c57f32017-05-30 18:18:33 -07002116 MACROBLOCKD *const xd = &x->e_mbd;
2117 const PLANE_TYPE plane_type = get_plane_type(plane);
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -08002118 const TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Jingning Han19b5c8f2017-07-06 15:10:12 -07002119 const TX_TYPE tx_type =
2120 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002121 const MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
2122 const struct macroblock_plane *p = &x->plane[plane];
2123 struct macroblockd_plane *pd = &xd->plane[plane];
2124 const int eob = p->eobs[block];
2125 tran_low_t *qcoeff = BLOCK_OFFSET(p->qcoeff, block);
2126 tran_low_t *dqcoeff = BLOCK_OFFSET(pd->dqcoeff, block);
2127 const tran_low_t *tcoeff = BLOCK_OFFSET(p->coeff, block);
Monty Montgomery125c0fc2017-10-26 00:44:35 -04002128 const int16_t *dequant = p->dequant_QTX;
Urvang Joshi80893152017-10-27 11:51:14 -07002129 const int seg_eob = av1_get_max_eob(tx_size);
Angie Chianga9ba58e2017-12-01 19:22:43 -08002130 const int bwl = get_txb_bwl(tx_size);
2131 const int width = get_txb_wide(tx_size);
2132 const int height = get_txb_high(tx_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002133 const int is_inter = is_inter_block(mbmi);
Angie Chiangbd99b382017-06-20 15:11:16 -07002134 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Jingning Handfd72322017-08-09 14:04:12 -07002135 const LV_MAP_COEFF_COST txb_costs = x->coeff_costs[txs_ctx][plane_type];
Angie Chiang07c57f32017-05-30 18:18:33 -07002136
2137 const int shift = av1_get_tx_scale(tx_size);
2138 const int64_t rdmult =
Jingning Hanb433f4c2017-11-17 15:43:59 -08002139 ((x->rdmult * plane_rd_mult[is_inter][plane_type] << (2 * (xd->bd - 8))) +
2140 2) >>
2141 2;
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002142 uint8_t levels_buf[TX_PAD_2D];
2143 uint8_t *const levels = set_levels(levels_buf, width);
Angie Chiang07c57f32017-05-30 18:18:33 -07002144
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002145 assert(width == (1 << bwl));
Linfeng Zhang1015a342017-10-24 16:20:41 -07002146 TxbInfo txb_info = {
2147 qcoeff, levels, dqcoeff, tcoeff, dequant, shift,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002148 tx_size, txs_ctx, tx_type, bwl, width, height,
Linfeng Zhang1015a342017-10-24 16:20:41 -07002149 eob, seg_eob, scan_order, txb_ctx, rdmult, &cm->coeff_ctx_table
2150 };
2151
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002152 av1_txb_init_levels(qcoeff, width, height, levels);
Urvang Joshi70006e42017-06-14 16:08:55 -07002153
Angie Chiang93af45f2017-10-23 18:19:59 -07002154 const int update = optimize_txb(&txb_info, &txb_costs, NULL, 0, fast_mode);
Angie Chiang07c57f32017-05-30 18:18:33 -07002155
Angie Chiang07c57f32017-05-30 18:18:33 -07002156 if (update) p->eobs[block] = txb_info.eob;
2157 return txb_info.eob;
2158}
Angie Chiang74e23072017-03-24 14:54:23 -07002159int av1_get_txb_entropy_context(const tran_low_t *qcoeff,
2160 const SCAN_ORDER *scan_order, int eob) {
2161 const int16_t *scan = scan_order->scan;
2162 int cul_level = 0;
2163 int c;
Jingning Han339cf932017-09-18 10:17:02 -07002164
2165 if (eob == 0) return 0;
Angie Chiang74e23072017-03-24 14:54:23 -07002166 for (c = 0; c < eob; ++c) {
2167 cul_level += abs(qcoeff[scan[c]]);
2168 }
2169
2170 cul_level = AOMMIN(COEFF_CONTEXT_MASK, cul_level);
2171 set_dc_sign(&cul_level, qcoeff[0]);
2172
2173 return cul_level;
2174}
2175
Jingning Han4fe5f672017-05-19 15:46:07 -07002176void av1_update_txb_context_b(int plane, int block, int blk_row, int blk_col,
2177 BLOCK_SIZE plane_bsize, TX_SIZE tx_size,
2178 void *arg) {
Jingning Han6171ae72017-05-18 20:15:06 -07002179 struct tokenize_b_args *const args = arg;
Angie Chiang36d616b2017-03-22 13:58:36 -07002180 const AV1_COMP *cpi = args->cpi;
2181 const AV1_COMMON *cm = &cpi->common;
Angie Chiang0397eda2017-03-15 16:57:14 -07002182 ThreadData *const td = args->td;
2183 MACROBLOCK *const x = &td->mb;
2184 MACROBLOCKD *const xd = &x->e_mbd;
Angie Chiang36d616b2017-03-22 13:58:36 -07002185 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
Angie Chiang0397eda2017-03-15 16:57:14 -07002186 struct macroblock_plane *p = &x->plane[plane];
2187 struct macroblockd_plane *pd = &xd->plane[plane];
Angie Chiang36d616b2017-03-22 13:58:36 -07002188 const uint16_t eob = p->eobs[block];
2189 const tran_low_t *qcoeff = BLOCK_OFFSET(p->qcoeff, block);
2190 const PLANE_TYPE plane_type = pd->plane_type;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002191 const TX_TYPE tx_type =
2192 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiangbd99b382017-06-20 15:11:16 -07002193 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Angie Chiang0397eda2017-03-15 16:57:14 -07002194 (void)plane_bsize;
Angie Chiang36d616b2017-03-22 13:58:36 -07002195
Angie Chiang74e23072017-03-24 14:54:23 -07002196 int cul_level = av1_get_txb_entropy_context(qcoeff, scan_order, eob);
Angie Chiang36d616b2017-03-22 13:58:36 -07002197 av1_set_contexts(xd, pd, plane, tx_size, cul_level, blk_col, blk_row);
Angie Chiang0397eda2017-03-15 16:57:14 -07002198}
2199
Jingning Han4fe5f672017-05-19 15:46:07 -07002200void av1_update_and_record_txb_context(int plane, int block, int blk_row,
2201 int blk_col, BLOCK_SIZE plane_bsize,
2202 TX_SIZE tx_size, void *arg) {
Jingning Han6171ae72017-05-18 20:15:06 -07002203 struct tokenize_b_args *const args = arg;
Angie Chiang0397eda2017-03-15 16:57:14 -07002204 const AV1_COMP *cpi = args->cpi;
2205 const AV1_COMMON *cm = &cpi->common;
2206 ThreadData *const td = args->td;
2207 MACROBLOCK *const x = &td->mb;
2208 MACROBLOCKD *const xd = &x->e_mbd;
2209 struct macroblock_plane *p = &x->plane[plane];
2210 struct macroblockd_plane *pd = &xd->plane[plane];
2211 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
Linfeng Zhang848f7bc2017-10-31 15:26:07 -07002212 int eob = p->eobs[block], update_eob = -1;
Angie Chiang0397eda2017-03-15 16:57:14 -07002213 const PLANE_TYPE plane_type = pd->plane_type;
2214 const tran_low_t *qcoeff = BLOCK_OFFSET(p->qcoeff, block);
2215 tran_low_t *tcoeff = BLOCK_OFFSET(x->mbmi_ext->tcoeff[plane], block);
2216 const int segment_id = mbmi->segment_id;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002217 const TX_TYPE tx_type =
2218 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiangbd99b382017-06-20 15:11:16 -07002219 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Angie Chianga2d5c942017-03-22 13:37:51 -07002220 const int16_t *scan = scan_order->scan;
hui suc0cf71d2017-07-20 16:38:50 -07002221 const int seg_eob = av1_get_tx_eob(&cpi->common.seg, segment_id, tx_size);
Linfeng Zhang1015a342017-10-24 16:20:41 -07002222 int c;
Angie Chiang85901562017-03-17 12:03:27 -07002223 TXB_CTX txb_ctx;
2224 get_txb_ctx(plane_bsize, tx_size, plane, pd->above_context + blk_col,
2225 pd->left_context + blk_row, &txb_ctx);
Angie Chianga9ba58e2017-12-01 19:22:43 -08002226 const int bwl = get_txb_bwl(tx_size);
2227 const int width = get_txb_wide(tx_size);
2228 const int height = get_txb_high(tx_size);
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002229 uint8_t levels_buf[TX_PAD_2D];
2230 uint8_t *const levels = set_levels(levels_buf, width);
Linfeng Zhangae7b2f32017-11-08 15:46:57 -08002231 DECLARE_ALIGNED(16, uint8_t, level_counts[MAX_TX_SQUARE]);
Yunqing Wang0e141b52017-11-02 15:08:58 -07002232 const uint8_t allow_update_cdf = args->allow_update_cdf;
Angie Chiang0397eda2017-03-15 16:57:14 -07002233
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -08002234 TX_SIZE txsize_ctx = get_txsize_entropy_ctx(tx_size);
Jingning Han8f661602017-08-19 08:16:50 -07002235 FRAME_CONTEXT *ec_ctx = xd->tile_ctx;
Jingning Han48be0e12017-06-13 12:12:01 -07002236
Angie Chiang0397eda2017-03-15 16:57:14 -07002237 memcpy(tcoeff, qcoeff, sizeof(*tcoeff) * seg_eob);
2238
Jingning Han48be0e12017-06-13 12:12:01 -07002239 ++td->counts->txb_skip[txsize_ctx][txb_ctx.txb_skip_ctx][eob == 0];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002240 if (allow_update_cdf)
2241 update_bin(ec_ctx->txb_skip_cdf[txsize_ctx][txb_ctx.txb_skip_ctx], eob == 0,
2242 2);
Angie Chiang85901562017-03-17 12:03:27 -07002243 x->mbmi_ext->txb_skip_ctx[plane][block] = txb_ctx.txb_skip_ctx;
Angie Chiang0397eda2017-03-15 16:57:14 -07002244
2245 x->mbmi_ext->eobs[plane][block] = eob;
2246
2247 if (eob == 0) {
2248 av1_set_contexts(xd, pd, plane, tx_size, 0, blk_col, blk_row);
2249 return;
2250 }
2251
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002252 av1_txb_init_levels(tcoeff, width, height, levels);
Linfeng Zhangce065ca2017-10-17 16:49:30 -07002253
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002254#if CONFIG_TXK_SEL
Jingning Han7eab9ff2017-07-06 10:12:54 -07002255 av1_update_tx_type_count(cm, xd, blk_row, blk_col, block, plane,
Yunqing Wang0e141b52017-11-02 15:08:58 -07002256 mbmi->sb_type, get_min_tx_size(tx_size), td->counts,
2257 allow_update_cdf);
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002258#endif
Angie Chiang0397eda2017-03-15 16:57:14 -07002259
Dake Hea47cd6c2017-10-13 18:09:58 -07002260 unsigned int(*nz_map_count)[SIG_COEF_CONTEXTS][2] =
2261 &(td->counts->nz_map[txsize_ctx][plane_type]);
Jingning Han35deaa72017-10-26 15:36:30 -07002262 av1_update_eob_context(eob, seg_eob, tx_size, tx_type, plane_type, ec_ctx,
Yunqing Wang0e141b52017-11-02 15:08:58 -07002263 td->counts, allow_update_cdf);
Dake He03a32922017-10-31 08:06:45 -07002264#if USE_CAUSAL_BASE_CTX
2265 int coeff_ctx = 0;
2266 update_eob = eob - 1;
2267#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07002268 for (c = eob - 1; c >= 0; --c) {
2269 tran_low_t v = qcoeff[scan[c]];
2270 int is_nz = (v != 0);
Dake He03a32922017-10-31 08:06:45 -07002271
Ola Hugosson13892102017-11-06 08:01:44 +01002272#if CONFIG_LV_MAP_MULTI
2273 (void)is_nz;
2274 (void)nz_map_count;
2275 coeff_ctx =
Linfeng Zhang8ac45572017-11-29 11:39:51 -08002276 get_nz_map_ctx(levels, c, scan, bwl, height, c == eob - 1, tx_type);
Dake He3fe369c2017-11-16 17:56:44 -08002277#if USE_BASE_EOB_ALPHABET
2278 if (c == eob - 1) {
2279 update_cdf(ec_ctx->coeff_base_eob_cdf[txsize_ctx][plane_type]
2280 [coeff_ctx - SIG_COEF_CONTEXTS +
2281 SIG_COEF_CONTEXTS_EOB],
2282 AOMMIN(abs(v), 3) - 1, 3);
2283 } else {
2284 update_cdf(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][coeff_ctx],
2285 AOMMIN(abs(v), 3), 4);
2286 }
Dake He59881772017-11-24 07:00:02 -08002287 {
2288 if (c < eob - 1) {
2289 ++(*nz_map_count)[coeff_ctx][is_nz];
2290 }
2291 if (is_nz) {
2292 for (int k = 0; k < NUM_BASE_LEVELS; ++k) {
2293 int is_k = (abs(v) > (k + 1));
2294 ++td->counts->coeff_base[txsize_ctx][plane_type][k][coeff_ctx][is_k];
2295 if (is_k == 0) break;
2296 }
2297 }
2298 }
2299
Dake He3fe369c2017-11-16 17:56:44 -08002300#else
Ola Hugosson13892102017-11-06 08:01:44 +01002301 update_cdf(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][coeff_ctx],
2302 AOMMIN(abs(v), 3), 4);
Dake He3fe369c2017-11-16 17:56:44 -08002303#endif
Ola Hugosson13892102017-11-06 08:01:44 +01002304#elif USE_CAUSAL_BASE_CTX
Linfeng Zhangb6957c22017-10-25 13:17:28 -07002305 coeff_ctx = get_nz_map_ctx(levels, c, scan, bwl, height, tx_type);
Ola Hugosson13892102017-11-06 08:01:44 +01002306
Dake He03a32922017-10-31 08:06:45 -07002307 if (c < eob - 1) {
2308 ++(*nz_map_count)[coeff_ctx][is_nz];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002309 if (allow_update_cdf)
2310 update_cdf(ec_ctx->nz_map_cdf[txsize_ctx][plane_type][coeff_ctx], is_nz,
2311 2);
Dake He03a32922017-10-31 08:06:45 -07002312 }
2313
2314 if (is_nz) {
2315 int k;
2316 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
2317 int ctx = coeff_ctx;
2318 int is_k = (abs(v) > (k + 1));
2319
2320 ++td->counts->coeff_base[txsize_ctx][plane_type][k][ctx][is_k];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002321 if (allow_update_cdf)
2322 update_bin(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][k][ctx],
2323 is_k, 2);
Dake He03a32922017-10-31 08:06:45 -07002324 if (is_k == 0) break;
2325 }
2326 }
2327#else
Linfeng Zhangb6957c22017-10-25 13:17:28 -07002328 int coeff_ctx = get_nz_map_ctx(levels, c, scan, bwl, height, tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -07002329
2330 if (c == eob - 1) continue;
2331
2332 ++(*nz_map_count)[coeff_ctx][is_nz];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002333 if (allow_update_cdf)
2334 update_cdf(ec_ctx->nz_map_cdf[txsize_ctx][plane_type][coeff_ctx], is_nz,
2335 2);
Dake He03a32922017-10-31 08:06:45 -07002336#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07002337 }
Dake Hea47cd6c2017-10-13 18:09:58 -07002338
Dake He03a32922017-10-31 08:06:45 -07002339#if !USE_CAUSAL_BASE_CTX
Angie Chiang0397eda2017-03-15 16:57:14 -07002340 // Reverse process order to handle coefficient level and sign.
Linfeng Zhang1015a342017-10-24 16:20:41 -07002341 for (int i = 0; i < NUM_BASE_LEVELS; ++i) {
Linfeng Zhangf72e1382017-11-01 13:59:41 -07002342 av1_get_base_level_counts(levels, i, width, height, level_counts);
Angie Chiang0397eda2017-03-15 16:57:14 -07002343 for (c = eob - 1; c >= 0; --c) {
Linfeng Zhang1015a342017-10-24 16:20:41 -07002344 const tran_low_t level = abs(tcoeff[scan[c]]);
Angie Chiang0397eda2017-03-15 16:57:14 -07002345 int ctx;
2346
2347 if (level <= i) continue;
2348
Linfeng Zhangf72e1382017-11-01 13:59:41 -07002349 ctx = get_base_ctx(levels, scan[c], bwl, i, level_counts[scan[c]]);
Angie Chiang0397eda2017-03-15 16:57:14 -07002350
2351 if (level == i + 1) {
Jingning Han48be0e12017-06-13 12:12:01 -07002352 ++td->counts->coeff_base[txsize_ctx][plane_type][i][ctx][1];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002353 if (allow_update_cdf)
2354 update_bin(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][i][ctx], 1,
2355 2);
Angie Chiang0397eda2017-03-15 16:57:14 -07002356 continue;
2357 }
Jingning Han48be0e12017-06-13 12:12:01 -07002358 ++td->counts->coeff_base[txsize_ctx][plane_type][i][ctx][0];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002359 if (allow_update_cdf)
2360 update_bin(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][i][ctx], 0,
2361 2);
Angie Chiang0397eda2017-03-15 16:57:14 -07002362 update_eob = AOMMAX(update_eob, c);
2363 }
2364 }
Dake He03a32922017-10-31 08:06:45 -07002365#endif
Angie Chiang0397eda2017-03-15 16:57:14 -07002366
Dake He43edb762017-10-26 10:29:46 -07002367 c = 0; // Update the context needed to code the DC sign (if applicable)
Linfeng Zhang1015a342017-10-24 16:20:41 -07002368 const int sign = (tcoeff[scan[c]] < 0) ? 1 : 0;
2369 if (tcoeff[scan[c]] != 0) {
Dake He43edb762017-10-26 10:29:46 -07002370 int dc_sign_ctx = txb_ctx.dc_sign_ctx;
2371
2372 ++td->counts->dc_sign[plane_type][dc_sign_ctx][sign];
2373#if LV_MAP_PROB
Yunqing Wang0e141b52017-11-02 15:08:58 -07002374 if (allow_update_cdf)
2375 update_bin(ec_ctx->dc_sign_cdf[plane_type][dc_sign_ctx], sign, 2);
Dake He43edb762017-10-26 10:29:46 -07002376#endif
2377 x->mbmi_ext->dc_sign_ctx[plane][block] = dc_sign_ctx;
2378 }
2379
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002380 if (update_eob >= 0) {
Dake He7d01ab52017-11-24 17:53:28 -08002381#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002382 av1_get_br_level_counts(levels, width, height, level_counts);
Dake He7d01ab52017-11-24 17:53:28 -08002383#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002384 for (c = update_eob; c >= 0; --c) {
2385 const tran_low_t level = abs(tcoeff[scan[c]]);
2386 int idx;
2387 int ctx;
Angie Chiang0397eda2017-03-15 16:57:14 -07002388
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002389 if (level <= NUM_BASE_LEVELS) continue;
Angie Chiang0397eda2017-03-15 16:57:14 -07002390
Dake He7d01ab52017-11-24 17:53:28 -08002391// level is above 1.
2392#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002393 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]]);
Dake He7d01ab52017-11-24 17:53:28 -08002394#endif
Jingning Han87b01b52017-08-31 12:07:20 -07002395
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002396 int base_range = level - 1 - NUM_BASE_LEVELS;
Ola Hugossone72a2092017-11-12 09:11:53 +01002397#if CONFIG_LV_MAP_MULTI
Dake He7d01ab52017-11-24 17:53:28 -08002398#if USE_CAUSAL_BR_CTX
2399 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]], tx_type);
2400#else
2401 ctx = get_br_ctx(levels, scan[c], bwl, level_counts[scan[c]]);
2402#endif
Ola Hugossone72a2092017-11-12 09:11:53 +01002403 for (idx = 0; idx < COEFF_BASE_RANGE; idx += BR_CDF_SIZE - 1) {
2404 int k = AOMMIN(base_range - idx, BR_CDF_SIZE - 1);
2405 // printf("br_update: %d %d %2d : %2d %d\n", txsize_ctx, plane, ctx,
2406 // base_range, k);
Dake He7d01ab52017-11-24 17:53:28 -08002407 update_cdf(
2408 ec_ctx->coeff_br_cdf[AOMMIN(txsize_ctx, TX_16X16)][plane_type][ctx],
2409 k, BR_CDF_SIZE);
2410 for (int lps = 0; lps < BR_CDF_SIZE - 1; lps++) {
2411 ++td->counts->coeff_lps[AOMMIN(txsize_ctx, TX_16X16)][plane_type][lps]
2412 [ctx][lps == k];
Dake He59881772017-11-24 07:00:02 -08002413 if (lps == k) break;
2414 }
2415
Ola Hugossone72a2092017-11-12 09:11:53 +01002416 if (k < BR_CDF_SIZE - 1) break;
2417 }
2418#else
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002419 int br_set_idx = base_range < COEFF_BASE_RANGE
2420 ? coeff_to_br_index[base_range]
2421 : BASE_RANGE_SETS;
Jingning Han87b01b52017-08-31 12:07:20 -07002422
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002423 for (idx = 0; idx < BASE_RANGE_SETS; ++idx) {
2424 if (idx == br_set_idx) {
2425 int br_base = br_index_to_coeff[br_set_idx];
2426 int br_offset = base_range - br_base;
2427 ++td->counts->coeff_br[txsize_ctx][plane_type][idx][ctx][1];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002428 if (allow_update_cdf)
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002429 update_bin(ec_ctx->coeff_br_cdf[txsize_ctx][plane_type][idx][ctx],
2430 1, 2);
2431 int extra_bits = (1 << br_extra_bits[idx]) - 1;
2432 for (int tok = 0; tok < extra_bits; ++tok) {
2433 if (br_offset == tok) {
2434 ++td->counts->coeff_lps[txsize_ctx][plane_type][ctx][1];
2435 if (allow_update_cdf)
2436 update_bin(ec_ctx->coeff_lps_cdf[txsize_ctx][plane_type][ctx],
2437 1, 2);
2438 break;
2439 }
2440 ++td->counts->coeff_lps[txsize_ctx][plane_type][ctx][0];
2441 if (allow_update_cdf)
2442 update_bin(ec_ctx->coeff_lps_cdf[txsize_ctx][plane_type][ctx], 0,
2443 2);
2444 }
2445 break;
Jingning Han87b01b52017-08-31 12:07:20 -07002446 }
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002447 ++td->counts->coeff_br[txsize_ctx][plane_type][idx][ctx][0];
2448 if (allow_update_cdf)
2449 update_bin(ec_ctx->coeff_br_cdf[txsize_ctx][plane_type][idx][ctx], 0,
2450 2);
Jingning Han87b01b52017-08-31 12:07:20 -07002451 }
Ola Hugossone72a2092017-11-12 09:11:53 +01002452#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002453 // use 0-th order Golomb code to handle the residual level.
Jingning Han87b01b52017-08-31 12:07:20 -07002454 }
Angie Chiang0397eda2017-03-15 16:57:14 -07002455 }
Angie Chiang36d616b2017-03-22 13:58:36 -07002456
Angie Chiang63d190a2017-10-23 15:43:05 -07002457 int cul_level = av1_get_txb_entropy_context(tcoeff, scan_order, eob);
Angie Chiang0397eda2017-03-15 16:57:14 -07002458 av1_set_contexts(xd, pd, plane, tx_size, cul_level, blk_col, blk_row);
Angie Chiang0b205e62017-03-20 17:16:47 -07002459
2460#if CONFIG_ADAPT_SCAN
2461 // Since dqcoeff is not available here, we pass qcoeff into
2462 // av1_update_scan_count_facade(). The update behavior should be the same
2463 // because av1_update_scan_count_facade() only cares if coefficients are zero
2464 // or not.
Jingning Han025c6c42017-11-22 12:06:03 -08002465 const int mi_row = -xd->mb_to_top_edge >> (3 + MI_SIZE_LOG2);
Jingning Han0bd3bf62017-11-28 17:11:51 -08002466 av1_update_scan_count_facade((AV1_COMMON *)cm, xd, mi_row, tx_size, tx_type,
2467 qcoeff, eob);
Angie Chiang0b205e62017-03-20 17:16:47 -07002468#endif
Angie Chiang0397eda2017-03-15 16:57:14 -07002469}
2470
2471void av1_update_txb_context(const AV1_COMP *cpi, ThreadData *td,
2472 RUN_TYPE dry_run, BLOCK_SIZE bsize, int *rate,
Yunqing Wang0e141b52017-11-02 15:08:58 -07002473 int mi_row, int mi_col, uint8_t allow_update_cdf) {
Angie Chiang0397eda2017-03-15 16:57:14 -07002474 MACROBLOCK *const x = &td->mb;
2475 MACROBLOCKD *const xd = &x->e_mbd;
2476 MB_MODE_INFO *const mbmi = &xd->mi[0]->mbmi;
Yunqing Wang0e141b52017-11-02 15:08:58 -07002477 struct tokenize_b_args arg = { cpi, td, NULL, 0, allow_update_cdf };
Angie Chiang0397eda2017-03-15 16:57:14 -07002478 (void)rate;
2479 (void)mi_row;
2480 (void)mi_col;
2481 if (mbmi->skip) {
Timothy B. Terriberrya2d5cde2017-05-10 18:33:50 -07002482 av1_reset_skip_context(xd, mi_row, mi_col, bsize);
Angie Chiang0397eda2017-03-15 16:57:14 -07002483 return;
2484 }
2485
2486 if (!dry_run) {
Jingning Han94652b82017-04-04 09:45:02 -07002487 av1_foreach_transformed_block(xd, bsize, mi_row, mi_col,
Jingning Han4fe5f672017-05-19 15:46:07 -07002488 av1_update_and_record_txb_context, &arg);
Angie Chiangc8af6112017-03-16 16:11:22 -07002489 } else if (dry_run == DRY_RUN_NORMAL) {
Jingning Han4fe5f672017-05-19 15:46:07 -07002490 av1_foreach_transformed_block(xd, bsize, mi_row, mi_col,
2491 av1_update_txb_context_b, &arg);
Angie Chiangc8af6112017-03-16 16:11:22 -07002492 } else {
2493 printf("DRY_RUN_COSTCOEFFS is not supported yet\n");
2494 assert(0);
Angie Chiang0397eda2017-03-15 16:57:14 -07002495 }
2496}
Angie Chiang800df032017-03-22 11:14:12 -07002497
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002498#if CONFIG_TXK_SEL
Angie Chiang808d8592017-04-06 18:36:55 -07002499int64_t av1_search_txk_type(const AV1_COMP *cpi, MACROBLOCK *x, int plane,
2500 int block, int blk_row, int blk_col,
2501 BLOCK_SIZE plane_bsize, TX_SIZE tx_size,
Angie Chiang65a39bb2017-04-11 16:50:04 -07002502 const ENTROPY_CONTEXT *a, const ENTROPY_CONTEXT *l,
2503 int use_fast_coef_costing, RD_STATS *rd_stats) {
Angie Chiang808d8592017-04-06 18:36:55 -07002504 const AV1_COMMON *cm = &cpi->common;
2505 MACROBLOCKD *xd = &x->e_mbd;
2506 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
2507 TX_TYPE txk_start = DCT_DCT;
2508 TX_TYPE txk_end = TX_TYPES - 1;
2509 TX_TYPE best_tx_type = txk_start;
2510 int64_t best_rd = INT64_MAX;
Jingning Han47558172017-07-05 16:33:19 -07002511 uint8_t best_eob = 0;
Angie Chiang65a39bb2017-04-11 16:50:04 -07002512 const int coeff_ctx = combine_entropy_contexts(*a, *l);
Jingning Hane3b81bc2017-06-23 11:43:52 -07002513 RD_STATS best_rd_stats;
Angie Chiang808d8592017-04-06 18:36:55 -07002514 TX_TYPE tx_type;
Jingning Hane3b81bc2017-06-23 11:43:52 -07002515
2516 av1_invalid_rd_stats(&best_rd_stats);
2517
Angie Chiang808d8592017-04-06 18:36:55 -07002518 for (tx_type = txk_start; tx_type <= txk_end; ++tx_type) {
Angie Chiangbce07f12017-12-01 16:34:31 -08002519 if (plane == 0)
2520 mbmi->txk_type[(blk_row << MAX_MIB_SIZE_LOG2) + blk_col] = tx_type;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002521 TX_TYPE ref_tx_type = av1_get_tx_type(get_plane_type(plane), xd, blk_row,
2522 blk_col, block, tx_size);
Angie Chiang00491e02017-04-11 17:55:10 -07002523 if (tx_type != ref_tx_type) {
hui su45b64752017-07-12 16:54:35 -07002524 // use av1_get_tx_type() to check if the tx_type is valid for the current
2525 // mode if it's not, we skip it here.
Angie Chiang00491e02017-04-11 17:55:10 -07002526 continue;
2527 }
Jingning Hane57d6322017-07-03 18:50:25 -07002528
Hui Suddbcde22017-09-18 17:22:02 -07002529 const int is_inter = is_inter_block(mbmi);
Angie Chiang53bf1e92017-11-29 16:53:07 -08002530 const TxSetType tx_set_type = get_ext_tx_set_type(
2531 tx_size, mbmi->sb_type, is_inter, cm->reduced_tx_set_used);
Hui Suddbcde22017-09-18 17:22:02 -07002532 if (!av1_ext_tx_used[tx_set_type][tx_type]) continue;
Jingning Hane57d6322017-07-03 18:50:25 -07002533
Angie Chiang808d8592017-04-06 18:36:55 -07002534 RD_STATS this_rd_stats;
2535 av1_invalid_rd_stats(&this_rd_stats);
Angie Chiang808d8592017-04-06 18:36:55 -07002536 av1_xform_quant(cm, x, plane, block, blk_row, blk_col, plane_bsize, tx_size,
2537 coeff_ctx, AV1_XFORM_QUANT_FP);
Jingning Han7eab9ff2017-07-06 10:12:54 -07002538 av1_optimize_b(cm, x, plane, blk_row, blk_col, block, plane_bsize, tx_size,
Angie Chiang25645b72017-09-24 14:28:49 -07002539 a, l, 1);
Angie Chiang808d8592017-04-06 18:36:55 -07002540 av1_dist_block(cpi, x, plane, plane_bsize, block, blk_row, blk_col, tx_size,
Angie Chiang2ed03a32017-04-16 18:00:06 -07002541 &this_rd_stats.dist, &this_rd_stats.sse,
2542 OUTPUT_HAS_PREDICTED_PIXELS);
Angie Chiangbd99b382017-06-20 15:11:16 -07002543 const SCAN_ORDER *scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Jingning Han7eab9ff2017-07-06 10:12:54 -07002544 this_rd_stats.rate =
2545 av1_cost_coeffs(cpi, x, plane, blk_row, blk_col, block, tx_size,
2546 scan_order, a, l, use_fast_coef_costing);
Urvang Joshi70006e42017-06-14 16:08:55 -07002547 int rd = RDCOST(x->rdmult, this_rd_stats.rate, this_rd_stats.dist);
Jingning Han19b5c8f2017-07-06 15:10:12 -07002548
Angie Chiang808d8592017-04-06 18:36:55 -07002549 if (rd < best_rd) {
2550 best_rd = rd;
Jingning Hane3b81bc2017-06-23 11:43:52 -07002551 best_rd_stats = this_rd_stats;
Angie Chiang808d8592017-04-06 18:36:55 -07002552 best_tx_type = tx_type;
Jingning Han47558172017-07-05 16:33:19 -07002553 best_eob = x->plane[plane].txb_entropy_ctx[block];
Angie Chiang808d8592017-04-06 18:36:55 -07002554 }
2555 }
Jingning Hane3b81bc2017-06-23 11:43:52 -07002556
2557 av1_merge_rd_stats(rd_stats, &best_rd_stats);
2558
Jingning Han19b5c8f2017-07-06 15:10:12 -07002559 if (best_eob == 0 && is_inter_block(mbmi)) best_tx_type = DCT_DCT;
2560
Angie Chiangbce07f12017-12-01 16:34:31 -08002561 if (plane == 0)
2562 mbmi->txk_type[(blk_row << MAX_MIB_SIZE_LOG2) + blk_col] = best_tx_type;
Jingning Han47558172017-07-05 16:33:19 -07002563 x->plane[plane].txb_entropy_ctx[block] = best_eob;
2564
Angie Chiang2ed03a32017-04-16 18:00:06 -07002565 if (!is_inter_block(mbmi)) {
2566 // intra mode needs decoded result such that the next transform block
2567 // can use it for prediction.
Jingning Han47558172017-07-05 16:33:19 -07002568 av1_xform_quant(cm, x, plane, block, blk_row, blk_col, plane_bsize, tx_size,
2569 coeff_ctx, AV1_XFORM_QUANT_FP);
Jingning Han7eab9ff2017-07-06 10:12:54 -07002570 av1_optimize_b(cm, x, plane, blk_row, blk_col, block, plane_bsize, tx_size,
Angie Chiang25645b72017-09-24 14:28:49 -07002571 a, l, 1);
Jingning Han47558172017-07-05 16:33:19 -07002572
Angie Chiang2ed03a32017-04-16 18:00:06 -07002573 av1_inverse_transform_block_facade(xd, plane, block, blk_row, blk_col,
Frederic Barbier33b39f02017-11-21 11:11:24 +01002574 x->plane[plane].eobs[block],
2575 cm->reduced_tx_set_used);
Angie Chiang2ed03a32017-04-16 18:00:06 -07002576 }
Angie Chiang808d8592017-04-06 18:36:55 -07002577 return best_rd;
2578}
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002579#endif // CONFIG_TXK_SEL