blob: 6d1750d32b72f9818a020455e08d5f3f05aa63e3 [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) {
Linfeng Zhang0c72b2f2017-12-04 10:59:28 -0800146 int eob_extra, dummy;
147 const int eob_pt = get_eob_pos_token(eob, &eob_extra);
148 const int max_eob_pt = get_eob_pos_token(seg_eob, &dummy);
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -0800149 TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Dake Hea47cd6c2017-10-13 18:09:58 -0700150
151 for (int i = 1; i < max_eob_pt; i++) {
Jingning Han35deaa72017-10-26 15:36:30 -0700152 int eob_pos_ctx = av1_get_eob_pos_ctx(tx_type, i);
153 counts->eob_flag[txs_ctx][plane][eob_pos_ctx][eob_pt == i]++;
Yunqing Wang0e141b52017-11-02 15:08:58 -0700154 if (allow_update_cdf)
155 update_cdf(ec_ctx->eob_flag_cdf[txs_ctx][plane][eob_pos_ctx], eob_pt == i,
156 2);
Dake Hea47cd6c2017-10-13 18:09:58 -0700157 if (eob_pt == i) {
158 break;
159 }
160 }
Jingning Han00803a72017-10-25 16:04:34 -0700161
Angie Chiang7ab884e2017-10-18 15:57:12 -0700162 if (k_eob_offset_bits[eob_pt] > 0) {
163 int eob_shift = k_eob_offset_bits[eob_pt] - 1;
164 int bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
Jingning Han35deaa72017-10-26 15:36:30 -0700165 counts->eob_extra[txs_ctx][plane][eob_pt][bit]++;
Yunqing Wang0e141b52017-11-02 15:08:58 -0700166 if (allow_update_cdf)
167 update_cdf(ec_ctx->eob_extra_cdf[txs_ctx][plane][eob_pt], bit, 2);
Angie Chiang7ab884e2017-10-18 15:57:12 -0700168 }
Dake Hea47cd6c2017-10-13 18:09:58 -0700169}
170
171static int get_eob_cost(int eob, int seg_eob,
Jingning Han35deaa72017-10-26 15:36:30 -0700172 const LV_MAP_COEFF_COST *txb_costs, TX_TYPE tx_type) {
Linfeng Zhang0c72b2f2017-12-04 10:59:28 -0800173 int eob_extra, dummy;
174 const int eob_pt = get_eob_pos_token(eob, &eob_extra);
175 const int max_eob_pt = get_eob_pos_token(seg_eob, &dummy);
Dake Hea47cd6c2017-10-13 18:09:58 -0700176 int eob_cost = 0;
177
178 // printf("Enc: [%d, %d], (%d, %d) ", seg_eob, eob, eob_pt, eob_extra);
179 for (int i = 1; i < max_eob_pt; i++) {
Jingning Han35deaa72017-10-26 15:36:30 -0700180 int eob_pos_ctx = av1_get_eob_pos_ctx(tx_type, i);
Dake Hea47cd6c2017-10-13 18:09:58 -0700181 eob_cost += txb_costs->eob_cost[eob_pos_ctx][eob_pt == i];
182 if (eob_pt == i) {
183 break;
184 }
185 }
186 if (k_eob_offset_bits[eob_pt] > 0) {
Angie Chiang7ab884e2017-10-18 15:57:12 -0700187 int eob_shift = k_eob_offset_bits[eob_pt] - 1;
188 int bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
189 eob_cost += txb_costs->eob_extra_cost[eob_pt][bit];
190 for (int i = 1; i < k_eob_offset_bits[eob_pt]; i++) {
191 eob_shift = k_eob_offset_bits[eob_pt] - 1 - i;
192 bit = (eob_extra & (1 << eob_shift)) ? 1 : 0;
Dake Hea47cd6c2017-10-13 18:09:58 -0700193 eob_cost += av1_cost_bit(128, bit);
194 }
195 }
196 return eob_cost;
197}
198
Linfeng Zhang1015a342017-10-24 16:20:41 -0700199static int get_coeff_cost(const tran_low_t qc, const int scan_idx,
Ola Hugosson13892102017-11-06 08:01:44 +0100200#if CONFIG_LV_MAP_MULTI
201 const int is_eob,
202#endif
Linfeng Zhang1015a342017-10-24 16:20:41 -0700203 const TxbInfo *const txb_info,
204 const LV_MAP_COEFF_COST *const txb_costs);
Dake Hea47cd6c2017-10-13 18:09:58 -0700205
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800206static void get_dist_cost_stats(LevelDownStats *const stats, const int scan_idx,
Ola Hugosson13892102017-11-06 08:01:44 +0100207#if CONFIG_LV_MAP_MULTI
208 const int is_eob,
209#endif
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800210 const LV_MAP_COEFF_COST *const txb_costs,
211 TxbInfo *const txb_info) {
212 const int16_t *const scan = txb_info->scan_order->scan;
Dake Hea47cd6c2017-10-13 18:09:58 -0700213 const int coeff_idx = scan[scan_idx];
214 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
Linfeng Zhang1015a342017-10-24 16:20:41 -0700215 const uint8_t *const levels = txb_info->levels;
Dake Hea47cd6c2017-10-13 18:09:58 -0700216 stats->new_eob = -1;
217 stats->update = 0;
218
219 const tran_low_t tqc = txb_info->tcoeff[coeff_idx];
220 const int dqv = txb_info->dequant[coeff_idx != 0];
221
222 const tran_low_t dqc = qcoeff_to_dqcoeff(qc, dqv, txb_info->shift);
223 const int64_t dqc_dist = get_coeff_dist(tqc, dqc, txb_info->shift);
Ola Hugosson13892102017-11-06 08:01:44 +0100224#if CONFIG_LV_MAP_MULTI
225 const int qc_cost = get_coeff_cost(qc, scan_idx, is_eob, txb_info, txb_costs);
226#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700227 const int qc_cost = get_coeff_cost(qc, scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +0100228#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700229
230 // distortion difference when coefficient is quantized to 0
231 const tran_low_t dqc0 = qcoeff_to_dqcoeff(0, dqv, txb_info->shift);
232 stats->dist0 = get_coeff_dist(tqc, dqc0, txb_info->shift);
233 stats->dist = dqc_dist - stats->dist0;
234 stats->rate = qc_cost;
235
236 if (qc == 0) {
237 return;
238 }
239 stats->rd = RDCOST(txb_info->rdmult, stats->rate, stats->dist);
240
241 stats->low_qc = get_lower_coeff(qc);
242 stats->low_dqc = qcoeff_to_dqcoeff(stats->low_qc, dqv, txb_info->shift);
243 const int64_t low_dqc_dist =
244 get_coeff_dist(tqc, stats->low_dqc, txb_info->shift);
Ola Hugosson13892102017-11-06 08:01:44 +0100245#if CONFIG_LV_MAP_MULTI
246 const int low_qc_cost =
247 get_coeff_cost(stats->low_qc, scan_idx, is_eob, txb_info, txb_costs);
248#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700249 const int low_qc_cost =
250 get_coeff_cost(stats->low_qc, scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +0100251#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700252 stats->dist_low = low_dqc_dist - stats->dist0;
253 stats->rate_low = low_qc_cost;
254 stats->rd_low = RDCOST(txb_info->rdmult, stats->rate_low, stats->dist_low);
255
Ola Hugosson13892102017-11-06 08:01:44 +0100256#if CONFIG_LV_MAP_MULTI
Ola Hugossone5a9b382017-11-16 17:02:28 +0100257 (void)levels;
Ola Hugosson13892102017-11-06 08:01:44 +0100258 if ((stats->rd_low < stats->rd) && (stats->low_qc == 0)) {
Ola Hugossone5a9b382017-11-16 17:02:28 +0100259 stats->nz_rate = low_qc_cost;
Ola Hugosson13892102017-11-06 08:01:44 +0100260 } else {
Ola Hugossone5a9b382017-11-16 17:02:28 +0100261 if (stats->rd_low < stats->rd) {
262 const int low_qc_eob_cost =
263 get_coeff_cost(stats->low_qc, scan_idx, 1, txb_info, txb_costs);
264 stats->nz_rate = low_qc_cost - low_qc_eob_cost;
265 } else {
266 const int qc_eob_cost =
267 get_coeff_cost(qc, scan_idx, 1, txb_info, txb_costs);
268 stats->nz_rate = qc_cost - qc_eob_cost;
269 }
Ola Hugosson13892102017-11-06 08:01:44 +0100270 }
271#else
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800272 const int coeff_ctx = get_nz_map_ctx(levels, coeff_idx, txb_info->bwl,
273 txb_info->height, txb_info->tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -0700274 if ((stats->rd_low < stats->rd) && (stats->low_qc == 0)) {
275 stats->nz_rate = txb_costs->nz_map_cost[coeff_ctx][0];
276 } else {
277 stats->nz_rate = txb_costs->nz_map_cost[coeff_ctx][1];
278 }
Ola Hugosson13892102017-11-06 08:01:44 +0100279#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700280}
281
Linfeng Zhang1015a342017-10-24 16:20:41 -0700282static INLINE void update_qcoeff(const int coeff_idx, const tran_low_t qc,
283 const TxbInfo *const txb_info) {
Dake Hea47cd6c2017-10-13 18:09:58 -0700284 txb_info->qcoeff[coeff_idx] = qc;
Linfeng Zhangd5647372017-12-05 17:06:07 -0800285 txb_info->levels[get_padded_idx(coeff_idx, txb_info->bwl)] =
Jingning Han5cb408e2017-11-17 14:43:39 -0800286 (uint8_t)clamp(abs(qc), 0, INT8_MAX);
Linfeng Zhang1015a342017-10-24 16:20:41 -0700287}
288
289static INLINE void update_coeff(const int coeff_idx, const tran_low_t qc,
290 const TxbInfo *const txb_info) {
291 update_qcoeff(coeff_idx, qc, txb_info);
Dake Hea47cd6c2017-10-13 18:09:58 -0700292 const int dqv = txb_info->dequant[coeff_idx != 0];
293 txb_info->dqcoeff[coeff_idx] = qcoeff_to_dqcoeff(qc, dqv, txb_info->shift);
294}
295
Linfeng Zhang1015a342017-10-24 16:20:41 -0700296static INLINE void av1_txb_init_levels(const tran_low_t *const coeff,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700297 const int width, const int height,
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700298 uint8_t *const levels) {
299 const int stride = width + TX_PAD_HOR;
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700300 uint8_t *ls = levels;
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700301
302 memset(levels - TX_PAD_TOP * stride, 0,
303 sizeof(*levels) * TX_PAD_TOP * stride);
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700304 memset(levels + stride * height, 0,
305 sizeof(*levels) * (TX_PAD_BOTTOM * stride + TX_PAD_END));
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700306
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700307 for (int i = 0; i < height; i++) {
308 for (int j = 0; j < width; j++) {
Jingning Han5cb408e2017-11-17 14:43:39 -0800309 *ls++ = (uint8_t)clamp(abs(coeff[i * width + j]), 0, INT8_MAX);
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700310 }
311 for (int j = 0; j < TX_PAD_HOR; j++) {
312 *ls++ = 0;
313 }
Linfeng Zhang1015a342017-10-24 16:20:41 -0700314 }
315}
316
Angie Chiang80b82262017-02-24 11:39:47 -0800317void av1_write_coeffs_txb(const AV1_COMMON *const cm, MACROBLOCKD *xd,
Jingning Han7eab9ff2017-07-06 10:12:54 -0700318 aom_writer *w, int blk_row, int blk_col, int block,
319 int plane, TX_SIZE tx_size, const tran_low_t *tcoeff,
320 uint16_t eob, TXB_CTX *txb_ctx) {
Angie Chiang80b82262017-02-24 11:39:47 -0800321 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
Angie Chiangcea11f22017-02-24 12:30:40 -0800322 const PLANE_TYPE plane_type = get_plane_type(plane);
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -0800323 const TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Jingning Han19b5c8f2017-07-06 15:10:12 -0700324 const TX_TYPE tx_type =
325 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiangbd99b382017-06-20 15:11:16 -0700326 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800327 const int16_t *const scan = scan_order->scan;
Urvang Joshi80893152017-10-27 11:51:14 -0700328 const int seg_eob = av1_get_max_eob(tx_size);
Angie Chiang80b82262017-02-24 11:39:47 -0800329 int c;
Angie Chianga9ba58e2017-12-01 19:22:43 -0800330 const int bwl = get_txb_bwl(tx_size);
331 const int width = get_txb_wide(tx_size);
332 const int height = get_txb_high(tx_size);
Linfeng Zhang848f7bc2017-10-31 15:26:07 -0700333 int update_eob = -1;
Jingning Han41c7f442017-09-05 14:54:00 -0700334 FRAME_CONTEXT *ec_ctx = xd->tile_ctx;
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700335 uint8_t levels_buf[TX_PAD_2D];
336 uint8_t *const levels = set_levels(levels_buf, width);
Linfeng Zhangae7b2f32017-11-08 15:46:57 -0800337 DECLARE_ALIGNED(16, uint8_t, level_counts[MAX_TX_SQUARE]);
Angie Chiang80b82262017-02-24 11:39:47 -0800338
Jingning Han7eab9ff2017-07-06 10:12:54 -0700339 (void)blk_row;
340 (void)blk_col;
Jingning Han94cea4a2017-09-30 14:13:23 -0700341 aom_write_bin(w, eob == 0,
342 ec_ctx->txb_skip_cdf[txs_ctx][txb_ctx->txb_skip_ctx], 2);
Angie Chianga3f7d2e2017-12-07 19:51:14 -0800343 if (plane == 0 && eob == 0) {
344 assert(tx_type == DCT_DCT);
345 }
Angie Chiang80b82262017-02-24 11:39:47 -0800346 if (eob == 0) return;
Linfeng Zhangce065ca2017-10-17 16:49:30 -0700347
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700348 av1_txb_init_levels(tcoeff, width, height, levels);
Linfeng Zhangce065ca2017-10-17 16:49:30 -0700349
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700350#if CONFIG_TXK_SEL
Jingning Han19b5c8f2017-07-06 15:10:12 -0700351 av1_write_tx_type(cm, xd, blk_row, blk_col, block, plane,
352 get_min_tx_size(tx_size), w);
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700353#endif
Angie Chiang80b82262017-02-24 11:39:47 -0800354
Linfeng Zhang0c72b2f2017-12-04 10:59:28 -0800355 int eob_extra, dummy;
356 const int eob_pt = get_eob_pos_token(eob, &eob_extra);
357 const int max_eob_pt = get_eob_pos_token(seg_eob, &dummy);
Dake Hea47cd6c2017-10-13 18:09:58 -0700358
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;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800389 const int pos = scan[c];
Dake He03a32922017-10-31 08:06:45 -0700390
Ola Hugosson13892102017-11-06 08:01:44 +0100391#if CONFIG_LV_MAP_MULTI
392 coeff_ctx =
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800393 get_nz_map_ctx(levels, pos, bwl, height, c, c == eob - 1, tx_type);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800394 const tran_low_t v = tcoeff[pos];
Dake He3fe369c2017-11-16 17:56:44 -0800395#if USE_BASE_EOB_ALPHABET
396 if (c == eob - 1) {
397 aom_write_symbol(
398 w, AOMMIN(abs(v), 3) - 1,
399 ec_ctx->coeff_base_eob_cdf[txs_ctx][plane_type]
400 [coeff_ctx - SIG_COEF_CONTEXTS +
401 SIG_COEF_CONTEXTS_EOB],
402 3);
403 } else {
404 aom_write_symbol(w, AOMMIN(abs(v), 3),
405 ec_ctx->coeff_base_cdf[txs_ctx][plane_type][coeff_ctx],
406 4);
407 }
408#else
Thomas Davies736ddef2017-11-09 09:46:08 +0000409 aom_write_symbol(w, AOMMIN(abs(v), 3),
410 ec_ctx->coeff_base_cdf[txs_ctx][plane_type][coeff_ctx], 4);
Dake He3fe369c2017-11-16 17:56:44 -0800411#endif
Ola Hugosson13892102017-11-06 08:01:44 +0100412#else
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800413 coeff_ctx = get_nz_map_ctx(levels, pos, bwl, height, tx_type);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800414 const tran_low_t v = tcoeff[pos];
415 const int is_nz = (v != 0);
Dake He03a32922017-10-31 08:06:45 -0700416
417 if (c < eob - 1) {
418 aom_write_bin(w, is_nz,
419 ec_ctx->nz_map_cdf[txs_ctx][plane_type][coeff_ctx], 2);
420 }
421 if (is_nz) {
422 const int level = abs(v);
423 int k;
424 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
425 int is_k = (level > (k + 1));
426 int ctx = coeff_ctx;
427 aom_write_bin(w, is_k,
428 ec_ctx->coeff_base_cdf[txs_ctx][plane_type][k][ctx], 2);
429 if (is_k == 0) break;
430 }
431 }
Ola Hugosson13892102017-11-06 08:01:44 +0100432#endif
Dake He03a32922017-10-31 08:06:45 -0700433 }
434 update_eob = eob - 1;
435#else
Dake Hea47cd6c2017-10-13 18:09:58 -0700436 for (int i = 1; i < eob; ++i) {
437 c = eob - 1 - i;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800438 const int pos = scan[c];
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800439 const int coeff_ctx = get_nz_map_ctx(levels, pos, bwl, height, tx_type);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800440 const tran_low_t v = tcoeff[pos];
441 const int is_nz = (v != 0);
Dake Hea47cd6c2017-10-13 18:09:58 -0700442
Dake Hea47cd6c2017-10-13 18:09:58 -0700443 aom_write_bin(w, is_nz, ec_ctx->nz_map_cdf[txs_ctx][plane_type][coeff_ctx],
444 2);
Dake Hea47cd6c2017-10-13 18:09:58 -0700445 }
446
Dake Hea47cd6c2017-10-13 18:09:58 -0700447 for (int i = 0; i < NUM_BASE_LEVELS; ++i) {
Linfeng Zhangf72e1382017-11-01 13:59:41 -0700448 av1_get_base_level_counts(levels, i, width, height, level_counts);
Angie Chiang80b82262017-02-24 11:39:47 -0800449 for (c = eob - 1; c >= 0; --c) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800450 const int pos = scan[c];
451 const tran_low_t level = abs(tcoeff[pos]);
Angie Chiang80b82262017-02-24 11:39:47 -0800452 int ctx;
453
454 if (level <= i) continue;
455
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800456 ctx = get_base_ctx(levels, pos, bwl, i, level_counts[pos]);
Angie Chiang80b82262017-02-24 11:39:47 -0800457
458 if (level == i + 1) {
Jingning Han94cea4a2017-09-30 14:13:23 -0700459 aom_write_bin(w, 1, ec_ctx->coeff_base_cdf[txs_ctx][plane_type][i][ctx],
460 2);
Angie Chiang80b82262017-02-24 11:39:47 -0800461 continue;
462 }
Jingning Han24b0cf92017-08-18 22:50:18 -0700463
Jingning Han94cea4a2017-09-30 14:13:23 -0700464 aom_write_bin(w, 0, ec_ctx->coeff_base_cdf[txs_ctx][plane_type][i][ctx],
465 2);
Angie Chiang80b82262017-02-24 11:39:47 -0800466 update_eob = AOMMAX(update_eob, c);
467 }
468 }
Dake He03a32922017-10-31 08:06:45 -0700469#endif
Angie Chiang80b82262017-02-24 11:39:47 -0800470
Dake He43edb762017-10-26 10:29:46 -0700471 // Loop to code all signs in the transform block,
472 // starting with the sign of DC (if applicable)
473 for (c = 0; c < eob; ++c) {
Linfeng Zhang1015a342017-10-24 16:20:41 -0700474 const tran_low_t v = tcoeff[scan[c]];
475 const tran_low_t level = abs(v);
476 const int sign = (v < 0) ? 1 : 0;
Dake He43edb762017-10-26 10:29:46 -0700477 if (level == 0) continue;
478
479 if (c == 0) {
480#if LV_MAP_PROB
481 aom_write_bin(w, sign,
482 ec_ctx->dc_sign_cdf[plane_type][txb_ctx->dc_sign_ctx], 2);
483#else
484 aom_write(w, sign, ec_ctx->dc_sign[plane_type][txb_ctx->dc_sign_ctx]);
485#endif
486 } else {
487 aom_write_bit(w, sign);
488 }
489 }
490
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800491 if (update_eob >= 0) {
Dake He7d01ab52017-11-24 17:53:28 -0800492#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800493 av1_get_br_level_counts(levels, width, height, level_counts);
Dake He7d01ab52017-11-24 17:53:28 -0800494#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800495 for (c = update_eob; c >= 0; --c) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800496 const int pos = scan[c];
497 const tran_low_t level = abs(tcoeff[pos]);
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800498 int idx;
499 int ctx;
Angie Chiang80b82262017-02-24 11:39:47 -0800500
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800501 if (level <= NUM_BASE_LEVELS) continue;
Angie Chiang80b82262017-02-24 11:39:47 -0800502
Dake He7d01ab52017-11-24 17:53:28 -0800503// level is above 1.
504#if !CONFIG_LV_MAP_MULTI
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800505 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos]);
Dake He7d01ab52017-11-24 17:53:28 -0800506#endif
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800507 const int base_range = level - 1 - NUM_BASE_LEVELS;
Ola Hugossone72a2092017-11-12 09:11:53 +0100508#if CONFIG_LV_MAP_MULTI
Dake He7d01ab52017-11-24 17:53:28 -0800509#if USE_CAUSAL_BR_CTX
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800510 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos], tx_type);
Dake He7d01ab52017-11-24 17:53:28 -0800511
512#else
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800513 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos]);
Dake He7d01ab52017-11-24 17:53:28 -0800514#endif
Ola Hugossone72a2092017-11-12 09:11:53 +0100515 for (idx = 0; idx < COEFF_BASE_RANGE; idx += BR_CDF_SIZE - 1) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800516 const int k = AOMMIN(base_range - idx, BR_CDF_SIZE - 1);
Dake He7d01ab52017-11-24 17:53:28 -0800517 aom_write_symbol(
518 w, k,
519 ec_ctx->coeff_br_cdf[AOMMIN(txs_ctx, TX_16X16)][plane_type][ctx],
520 BR_CDF_SIZE);
Ola Hugossone72a2092017-11-12 09:11:53 +0100521 if (k < BR_CDF_SIZE - 1) break;
522 }
523 if (base_range < COEFF_BASE_RANGE) continue;
524#else
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800525 int br_set_idx = 0;
526 int br_base = 0;
527 int br_offset = 0;
Jingning Han87b01b52017-08-31 12:07:20 -0700528
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800529 if (base_range >= COEFF_BASE_RANGE)
530 br_set_idx = BASE_RANGE_SETS;
531 else
532 br_set_idx = coeff_to_br_index[base_range];
Jingning Han87b01b52017-08-31 12:07:20 -0700533
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800534 for (idx = 0; idx < BASE_RANGE_SETS; ++idx) {
Ola Hugosson13892102017-11-06 08:01:44 +0100535 // printf("br: %d %d %d %d\n", txs_ctx, plane_type, idx, ctx);
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800536 aom_write_bin(w, idx == br_set_idx,
537 ec_ctx->coeff_br_cdf[txs_ctx][plane_type][idx][ctx], 2);
538 if (idx == br_set_idx) {
539 br_base = br_index_to_coeff[br_set_idx];
540 br_offset = base_range - br_base;
541 int extra_bits = (1 << br_extra_bits[idx]) - 1;
542 for (int tok = 0; tok < extra_bits; ++tok) {
543 if (tok == br_offset) {
544 aom_write_bin(w, 1,
545 ec_ctx->coeff_lps_cdf[txs_ctx][plane_type][ctx], 2);
546 break;
547 }
548 aom_write_bin(w, 0, ec_ctx->coeff_lps_cdf[txs_ctx][plane_type][ctx],
Jingning Han94cea4a2017-09-30 14:13:23 -0700549 2);
Jingning Han87b01b52017-08-31 12:07:20 -0700550 }
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800551 // aom_write_literal(w, br_offset, br_extra_bits[idx]);
552 break;
Jingning Han87b01b52017-08-31 12:07:20 -0700553 }
Jingning Han87b01b52017-08-31 12:07:20 -0700554 }
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800555
556 if (br_set_idx < BASE_RANGE_SETS) continue;
Ola Hugossone72a2092017-11-12 09:11:53 +0100557#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800558 // use 0-th order Golomb code to handle the residual level.
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800559 write_golomb(w,
560 abs(tcoeff[pos]) - COEFF_BASE_RANGE - 1 - NUM_BASE_LEVELS);
Jingning Han87b01b52017-08-31 12:07:20 -0700561 }
Angie Chiang80b82262017-02-24 11:39:47 -0800562 }
Jingning Han0bd3bf62017-11-28 17:11:51 -0800563
564#if CONFIG_ADAPT_SCAN
565 const int mi_row = -xd->mb_to_top_edge >> (3 + MI_SIZE_LOG2);
566 av1_update_scan_count_facade(cm, xd, mi_row, tx_size, tx_type, tcoeff, eob);
567#endif
Angie Chiang80b82262017-02-24 11:39:47 -0800568}
Angie Chiang47c72182017-02-27 14:30:38 -0800569
Angie Chiangc8af6112017-03-16 16:11:22 -0700570void av1_write_coeffs_mb(const AV1_COMMON *const cm, MACROBLOCK *x,
571 aom_writer *w, int plane) {
572 MACROBLOCKD *xd = &x->e_mbd;
573 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
574 BLOCK_SIZE bsize = mbmi->sb_type;
575 struct macroblockd_plane *pd = &xd->plane[plane];
Angie Chiangc8af6112017-03-16 16:11:22 -0700576 const BLOCK_SIZE plane_bsize =
Angie Chiangee4a34c2017-06-07 15:23:17 -0700577 AOMMAX(BLOCK_4X4, get_plane_block_size(bsize, pd));
Angie Chiang8d6bc802017-04-12 15:14:05 -0700578 const int max_blocks_wide = max_block_wide(xd, plane_bsize, plane);
579 const int max_blocks_high = max_block_high(xd, plane_bsize, plane);
hui su0c6244b2017-07-12 17:11:43 -0700580 const TX_SIZE tx_size = av1_get_tx_size(plane, xd);
Angie Chiangc8af6112017-03-16 16:11:22 -0700581 const int bkw = tx_size_wide_unit[tx_size];
582 const int bkh = tx_size_high_unit[tx_size];
583 const int step = tx_size_wide_unit[tx_size] * tx_size_high_unit[tx_size];
584 int row, col;
585 int block = 0;
Angie Chiang8d6bc802017-04-12 15:14:05 -0700586 for (row = 0; row < max_blocks_high; row += bkh) {
587 for (col = 0; col < max_blocks_wide; col += bkw) {
Angie Chiangc8af6112017-03-16 16:11:22 -0700588 tran_low_t *tcoeff = BLOCK_OFFSET(x->mbmi_ext->tcoeff[plane], block);
589 uint16_t eob = x->mbmi_ext->eobs[plane][block];
590 TXB_CTX txb_ctx = { x->mbmi_ext->txb_skip_ctx[plane][block],
591 x->mbmi_ext->dc_sign_ctx[plane][block] };
Jingning Han7eab9ff2017-07-06 10:12:54 -0700592 av1_write_coeffs_txb(cm, xd, w, row, col, block, plane, tx_size, tcoeff,
593 eob, &txb_ctx);
Angie Chiangc8af6112017-03-16 16:11:22 -0700594 block += step;
595 }
596 }
597}
598
Yaowu Xu9b803912017-11-16 08:16:33 -0800599#if !USE_CAUSAL_BASE_CTX
Linfeng Zhang841a84d2017-10-30 14:31:11 -0700600static INLINE void get_base_ctx_set(const uint8_t *const levels,
601 const int c, // raster order
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700602 const int bwl,
Angie Chiang47c72182017-02-27 14:30:38 -0800603 int ctx_set[NUM_BASE_LEVELS]) {
604 const int row = c >> bwl;
605 const int col = c - (row << bwl);
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700606 const int width = 1 << bwl;
607 const int stride = width + TX_PAD_HOR;
Jingning Hane775b8b2017-10-26 16:56:22 -0700608 int mag_count[NUM_BASE_LEVELS] = { 0 };
609 int nb_mag[NUM_BASE_LEVELS][3] = { { 0 } };
Angie Chiang47c72182017-02-27 14:30:38 -0800610 int idx;
Angie Chiang47c72182017-02-27 14:30:38 -0800611 int i;
612
613 for (idx = 0; idx < BASE_CONTEXT_POSITION_NUM; ++idx) {
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700614 const int ref_row = row + base_ref_offset[idx][0];
615 const int ref_col = col + base_ref_offset[idx][1];
616 const int pos = ref_row * stride + ref_col;
Linfeng Zhang841a84d2017-10-30 14:31:11 -0700617 const uint8_t abs_coeff = levels[pos];
Angie Chiang47c72182017-02-27 14:30:38 -0800618
619 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
620 ctx_set[i] += abs_coeff > i;
Jingning Hane775b8b2017-10-26 16:56:22 -0700621 if (base_ref_offset[idx][0] == 0 && base_ref_offset[idx][1] == 1)
622 nb_mag[i][0] = abs_coeff;
623 if (base_ref_offset[idx][0] == 1 && base_ref_offset[idx][1] == 0)
624 nb_mag[i][1] = abs_coeff;
625 if (base_ref_offset[idx][0] == 1 && base_ref_offset[idx][1] == 1)
626 nb_mag[i][2] = abs_coeff;
Angie Chiang47c72182017-02-27 14:30:38 -0800627 }
628 }
629
630 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
Jingning Hane775b8b2017-10-26 16:56:22 -0700631 for (idx = 0; idx < 3; ++idx) mag_count[i] += nb_mag[i][idx] > i + 1;
632 ctx_set[i] = get_base_ctx_from_count_mag(row, col, ctx_set[i],
633 AOMMIN(2, mag_count[i]));
Angie Chiang47c72182017-02-27 14:30:38 -0800634 }
635 return;
636}
Yaowu Xu9b803912017-11-16 08:16:33 -0800637#endif
Angie Chiang47c72182017-02-27 14:30:38 -0800638
Angie Chiang488f9212017-05-30 12:46:26 -0700639static INLINE int get_br_cost(tran_low_t abs_qc, int ctx,
Angie Chiang26d3e452017-09-29 17:40:02 -0700640 const int *coeff_lps) {
Angie Chiang488f9212017-05-30 12:46:26 -0700641 const tran_low_t min_level = 1 + NUM_BASE_LEVELS;
642 const tran_low_t max_level = 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE;
Jingning Handfd72322017-08-09 14:04:12 -0700643 (void)ctx;
Angie Chiang488f9212017-05-30 12:46:26 -0700644 if (abs_qc >= min_level) {
Angie Chiang488f9212017-05-30 12:46:26 -0700645 if (abs_qc >= max_level)
Jingning Han772dee32017-09-15 08:53:18 -0700646 return coeff_lps[COEFF_BASE_RANGE]; // COEFF_BASE_RANGE * cost0;
Angie Chiang488f9212017-05-30 12:46:26 -0700647 else
Jingning Han772dee32017-09-15 08:53:18 -0700648 return coeff_lps[(abs_qc - min_level)]; // * cost0 + cost1;
Angie Chiang488f9212017-05-30 12:46:26 -0700649 } else {
650 return 0;
651 }
652}
653
654static INLINE int get_base_cost(tran_low_t abs_qc, int ctx,
Jingning Handfd72322017-08-09 14:04:12 -0700655 const int coeff_base[2], int base_idx) {
Angie Chiang488f9212017-05-30 12:46:26 -0700656 const int level = base_idx + 1;
Jingning Handfd72322017-08-09 14:04:12 -0700657 (void)ctx;
Angie Chiang488f9212017-05-30 12:46:26 -0700658 if (abs_qc < level)
659 return 0;
660 else
Jingning Handfd72322017-08-09 14:04:12 -0700661 return coeff_base[abs_qc == level];
Angie Chiang488f9212017-05-30 12:46:26 -0700662}
663
Angie Chiang3627de22017-08-18 20:15:59 -0700664int av1_cost_coeffs_txb(const AV1_COMMON *const cm, MACROBLOCK *x, int plane,
Jingning Han7eab9ff2017-07-06 10:12:54 -0700665 int blk_row, int blk_col, int block, TX_SIZE tx_size,
666 TXB_CTX *txb_ctx) {
Angie Chiang47c72182017-02-27 14:30:38 -0800667 MACROBLOCKD *const xd = &x->e_mbd;
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -0800668 TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Angie Chiang47c72182017-02-27 14:30:38 -0800669 const PLANE_TYPE plane_type = get_plane_type(plane);
Jingning Han19b5c8f2017-07-06 15:10:12 -0700670 const TX_TYPE tx_type =
671 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiang47c72182017-02-27 14:30:38 -0800672 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
673 const struct macroblock_plane *p = &x->plane[plane];
674 const int eob = p->eobs[block];
675 const tran_low_t *const qcoeff = BLOCK_OFFSET(p->qcoeff, block);
676 int c, cost;
Angie Chiang47c72182017-02-27 14:30:38 -0800677 int txb_skip_ctx = txb_ctx->txb_skip_ctx;
Angie Chiang47c72182017-02-27 14:30:38 -0800678
Angie Chianga9ba58e2017-12-01 19:22:43 -0800679 const int bwl = get_txb_bwl(tx_size);
680 const int width = get_txb_wide(tx_size);
681 const int height = get_txb_high(tx_size);
Jingning Han341d79e2017-06-13 15:57:59 -0700682
Angie Chiangbd99b382017-06-20 15:11:16 -0700683 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800684 const int16_t *const scan = scan_order->scan;
Linfeng Zhang679d81e2017-10-31 15:27:42 -0700685 uint8_t levels_buf[TX_PAD_2D];
686 uint8_t *const levels = set_levels(levels_buf, width);
Linfeng Zhangae7b2f32017-11-08 15:46:57 -0800687 DECLARE_ALIGNED(16, uint8_t, level_counts[MAX_TX_SQUARE]);
Angie Chiang47c72182017-02-27 14:30:38 -0800688
Jingning Handfd72322017-08-09 14:04:12 -0700689 LV_MAP_COEFF_COST *coeff_costs = &x->coeff_costs[txs_ctx][plane_type];
690
Angie Chiang47c72182017-02-27 14:30:38 -0800691 cost = 0;
692
693 if (eob == 0) {
Jingning Handfd72322017-08-09 14:04:12 -0700694 cost = coeff_costs->txb_skip_cost[txb_skip_ctx][1];
Angie Chiang47c72182017-02-27 14:30:38 -0800695 return cost;
696 }
Jingning Handfd72322017-08-09 14:04:12 -0700697 cost = coeff_costs->txb_skip_cost[txb_skip_ctx][0];
Angie Chiang47c72182017-02-27 14:30:38 -0800698
Linfeng Zhang1122d7d2017-10-31 15:30:28 -0700699 av1_txb_init_levels(qcoeff, width, height, levels);
Linfeng Zhang1015a342017-10-24 16:20:41 -0700700
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700701#if CONFIG_TXK_SEL
Yue Chenb23d00a2017-07-28 17:01:21 -0700702 cost += av1_tx_type_cost(cm, x, xd, mbmi->sb_type, plane, tx_size, tx_type);
Angie Chiangcd9b03f2017-04-16 13:37:13 -0700703#endif
Angie Chiang05917872017-04-15 12:28:56 -0700704
Urvang Joshi80893152017-10-27 11:51:14 -0700705 const int seg_eob = av1_get_max_eob(tx_size);
Jingning Han35deaa72017-10-26 15:36:30 -0700706 int eob_cost = get_eob_cost(eob, seg_eob, coeff_costs, tx_type);
Dake He7d01ab52017-11-24 17:53:28 -0800707#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -0800708 av1_get_br_level_counts(levels, width, height, level_counts);
Dake He7d01ab52017-11-24 17:53:28 -0800709#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700710 cost += eob_cost;
Dake He03a32922017-10-31 08:06:45 -0700711#if USE_CAUSAL_BASE_CTX
712 int coeff_ctx = 0;
713#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700714 for (c = eob - 1; c >= 0; --c) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800715 const int pos = scan[c];
716 const tran_low_t v = qcoeff[pos];
717 const int is_nz = (v != 0);
718 const int level = abs(v);
Ola Hugosson13892102017-11-06 08:01:44 +0100719#if CONFIG_LV_MAP_MULTI
720 coeff_ctx =
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800721 get_nz_map_ctx(levels, pos, bwl, height, c, c == eob - 1, tx_type);
Dake He3fe369c2017-11-16 17:56:44 -0800722#if USE_BASE_EOB_ALPHABET
723 if (c == eob - 1) {
724 cost += coeff_costs
725 ->base_eob_cost[coeff_ctx - SIG_COEF_CONTEXTS +
726 SIG_COEF_CONTEXTS_EOB][AOMMIN(level, 3) - 1];
727 } else {
728 cost += coeff_costs->base_cost[coeff_ctx][AOMMIN(level, 3)];
729 }
730#else
Ola Hugosson13892102017-11-06 08:01:44 +0100731 cost += coeff_costs->base_cost[coeff_ctx][AOMMIN(level, 3)];
Dake He3fe369c2017-11-16 17:56:44 -0800732#endif
Ola Hugosson13892102017-11-06 08:01:44 +0100733#else // CONFIG_LV_MAP_MULTI
Dake He03a32922017-10-31 08:06:45 -0700734#if USE_CAUSAL_BASE_CTX
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800735 coeff_ctx = get_nz_map_ctx(levels, pos, bwl, height, tx_type);
Dake He03a32922017-10-31 08:06:45 -0700736#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700737
738 if (c < eob - 1) {
Dake He03a32922017-10-31 08:06:45 -0700739#if !USE_CAUSAL_BASE_CTX
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800740 const int coeff_ctx = get_nz_map_ctx(levels, pos, bwl, height, tx_type);
Dake He03a32922017-10-31 08:06:45 -0700741#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700742 cost += coeff_costs->nz_map_cost[coeff_ctx][is_nz];
743 }
Ola Hugosson13892102017-11-06 08:01:44 +0100744#endif // CONFIG_LV_MAP_MULTI
Dake Hea47cd6c2017-10-13 18:09:58 -0700745
746 if (is_nz) {
Dake He03a32922017-10-31 08:06:45 -0700747#if !USE_CAUSAL_BASE_CTX
Dake Hea47cd6c2017-10-13 18:09:58 -0700748 int ctx_ls[NUM_BASE_LEVELS] = { 0 };
Dake He03a32922017-10-31 08:06:45 -0700749#endif
Dake Hea47cd6c2017-10-13 18:09:58 -0700750 int sign = (v < 0) ? 1 : 0;
751
752 // sign bit cost
753 if (c == 0) {
754 int dc_sign_ctx = txb_ctx->dc_sign_ctx;
755 cost += coeff_costs->dc_sign_cost[dc_sign_ctx][sign];
756 } else {
757 cost += av1_cost_bit(128, sign);
758 }
Ola Hugosson13892102017-11-06 08:01:44 +0100759#if !CONFIG_LV_MAP_MULTI
Dake He03a32922017-10-31 08:06:45 -0700760#if USE_CAUSAL_BASE_CTX
761 int k;
762 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
763 int is_k = (level > (k + 1));
764 int ctx = coeff_ctx;
765 // get_base_ctx_from_b(c, k, b0, b1, b2);
766 cost += coeff_costs->base_cost[k][ctx][is_k];
767 if (is_k == 0) break;
768 }
769#else
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800770 get_base_ctx_set(levels, pos, bwl, ctx_ls);
Dake Hea47cd6c2017-10-13 18:09:58 -0700771
772 int i;
773 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
774 if (level <= i) continue;
775
776 if (level == i + 1) {
777 cost += coeff_costs->base_cost[i][ctx_ls[i]][1];
778 continue;
779 }
780 cost += coeff_costs->base_cost[i][ctx_ls[i]][0];
781 }
Dake He03a32922017-10-31 08:06:45 -0700782#endif
Ola Hugosson13892102017-11-06 08:01:44 +0100783#endif // CONFIG_LV_MAP_MULTI
Dake Hea47cd6c2017-10-13 18:09:58 -0700784 if (level > NUM_BASE_LEVELS) {
785 int ctx;
Dake He7d01ab52017-11-24 17:53:28 -0800786#if CONFIG_LV_MAP_MULTI && USE_CAUSAL_BR_CTX
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800787 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos], tx_type);
Dake He7d01ab52017-11-24 17:53:28 -0800788#else
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800789 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos]);
Dake He7d01ab52017-11-24 17:53:28 -0800790#endif
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800791 const int base_range = level - 1 - NUM_BASE_LEVELS;
Dake Hea47cd6c2017-10-13 18:09:58 -0700792 if (base_range < COEFF_BASE_RANGE) {
793 cost += coeff_costs->lps_cost[ctx][base_range];
794 } else {
795 cost += coeff_costs->lps_cost[ctx][COEFF_BASE_RANGE];
796 }
797
Dake Hea47cd6c2017-10-13 18:09:58 -0700798 if (level >= 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
799 // residual cost
800 int r = level - COEFF_BASE_RANGE - NUM_BASE_LEVELS;
801 int ri = r;
802 int length = 0;
803
804 while (ri) {
805 ri >>= 1;
806 ++length;
807 }
808
809 for (ri = 0; ri < length - 1; ++ri) cost += av1_cost_bit(128, 0);
810
811 for (ri = length - 1; ri >= 0; --ri)
812 cost += av1_cost_bit(128, (r >> ri) & 0x01);
813 }
814 }
815 }
816 }
Angie Chiang47c72182017-02-27 14:30:38 -0800817 return cost;
818}
Angie Chiang0397eda2017-03-15 16:57:14 -0700819
Angie Chiang2affb3b2017-05-30 15:23:51 -0700820static INLINE int has_base(tran_low_t qc, int base_idx) {
821 const int level = base_idx + 1;
822 return abs(qc) >= level;
823}
824
Angie Chiang2affb3b2017-05-30 15:23:51 -0700825static INLINE int has_br(tran_low_t qc) {
826 return abs(qc) >= 1 + NUM_BASE_LEVELS;
827}
828
Angie Chiang488f9212017-05-30 12:46:26 -0700829static INLINE int get_sign_bit_cost(tran_low_t qc, int coeff_idx,
Jingning Handfd72322017-08-09 14:04:12 -0700830 const int (*dc_sign_cost)[2],
Angie Chiang488f9212017-05-30 12:46:26 -0700831 int dc_sign_ctx) {
832 const int sign = (qc < 0) ? 1 : 0;
833 // sign bit cost
834 if (coeff_idx == 0) {
Jingning Handfd72322017-08-09 14:04:12 -0700835 return dc_sign_cost[dc_sign_ctx][sign];
Angie Chiang488f9212017-05-30 12:46:26 -0700836 } else {
837 return av1_cost_bit(128, sign);
838 }
839}
840static INLINE int get_golomb_cost(int abs_qc) {
841 if (abs_qc >= 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
842 // residual cost
843 int r = abs_qc - COEFF_BASE_RANGE - NUM_BASE_LEVELS;
844 int ri = r;
845 int length = 0;
846
847 while (ri) {
848 ri >>= 1;
849 ++length;
850 }
851
852 return av1_cost_literal(2 * length - 1);
853 } else {
854 return 0;
855 }
856}
857
Angie Chiang2affb3b2017-05-30 15:23:51 -0700858void gen_txb_cache(TxbCache *txb_cache, TxbInfo *txb_info) {
Angie Chiang481c01f2017-08-15 16:24:30 -0700859 // gen_nz_count_arr
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -0800860 const int16_t *const scan = txb_info->scan_order->scan;
Angie Chiang481c01f2017-08-15 16:24:30 -0700861 const int bwl = txb_info->bwl;
862 const int height = txb_info->height;
Linfeng Zhang72e0b3f2017-10-18 13:32:12 -0700863 const tran_low_t *const qcoeff = txb_info->qcoeff;
Linfeng Zhangb6957c22017-10-25 13:17:28 -0700864 const uint8_t *const levels = txb_info->levels;
Angie Chiangdef11252017-08-18 14:37:32 -0700865 const BASE_CTX_TABLE *base_ctx_table =
866 txb_info->coeff_ctx_table->base_ctx_table;
Angie Chiang481c01f2017-08-15 16:24:30 -0700867 for (int c = 0; c < txb_info->eob; ++c) {
868 const int coeff_idx = scan[c]; // raster order
869 const int row = coeff_idx >> bwl;
870 const int col = coeff_idx - (row << bwl);
Jingning Han3422ac12017-10-25 20:37:53 -0700871
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800872 txb_cache->nz_count_arr[coeff_idx] =
Linfeng Zhangd5647372017-12-05 17:06:07 -0800873 get_nz_count(levels + get_padded_idx(coeff_idx, bwl), bwl,
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -0800874 tx_type_to_class[txb_info->tx_type]);
Dake Hea47cd6c2017-10-13 18:09:58 -0700875
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800876 txb_cache->nz_ctx_arr[coeff_idx] = get_nz_map_ctx_from_stats(
Dake He03a32922017-10-31 08:06:45 -0700877#if USE_CAUSAL_BASE_CTX
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800878 0,
879#else
880 txb_cache->nz_count_arr[coeff_idx],
Dake He03a32922017-10-31 08:06:45 -0700881#endif
Linfeng Zhang8ac45572017-11-29 11:39:51 -0800882 coeff_idx, bwl, height, txb_info->tx_type);
Angie Chiang481c01f2017-08-15 16:24:30 -0700883
884 // gen_base_count_mag_arr
885 if (!has_base(qcoeff[coeff_idx], 0)) continue;
886 int *base_mag = txb_cache->base_mag_arr[coeff_idx];
Angie Chiang9cde59f2017-08-16 15:24:55 -0700887 int count[NUM_BASE_LEVELS];
888 get_base_count_mag(base_mag, count, qcoeff, bwl, height, row, col);
Angie Chiang481c01f2017-08-15 16:24:30 -0700889
890 for (int i = 0; i < NUM_BASE_LEVELS; ++i) {
Angie Chiang9cde59f2017-08-16 15:24:55 -0700891 if (!has_base(qcoeff[coeff_idx], i)) break;
892 txb_cache->base_count_arr[i][coeff_idx] = count[i];
Angie Chiang481c01f2017-08-15 16:24:30 -0700893 const int level = i + 1;
Angie Chiang9cde59f2017-08-16 15:24:55 -0700894 txb_cache->base_ctx_arr[i][coeff_idx] =
Angie Chiangdef11252017-08-18 14:37:32 -0700895 base_ctx_table[row != 0][col != 0][base_mag[0] > level][count[i]];
Angie Chiang481c01f2017-08-15 16:24:30 -0700896 }
897
898 // gen_br_count_mag_arr
899 if (!has_br(qcoeff[coeff_idx])) continue;
900 int *br_count = txb_cache->br_count_arr + coeff_idx;
901 int *br_mag = txb_cache->br_mag_arr[coeff_idx];
Angie Chiangea8183b2017-08-17 14:36:35 -0700902 *br_count = get_br_count_mag(br_mag, qcoeff, bwl, height, row, col,
903 NUM_BASE_LEVELS);
Angie Chiang481c01f2017-08-15 16:24:30 -0700904 txb_cache->br_ctx_arr[coeff_idx] =
905 get_br_ctx_from_count_mag(row, col, *br_count, br_mag[0]);
906 }
Angie Chiang2affb3b2017-05-30 15:23:51 -0700907}
908
Jingning Handfd72322017-08-09 14:04:12 -0700909static INLINE const int *get_level_prob(int level, int coeff_idx,
910 const TxbCache *txb_cache,
911 const LV_MAP_COEFF_COST *txb_costs) {
Ola Hugosson13892102017-11-06 08:01:44 +0100912#if CONFIG_LV_MAP_MULTI
913 if (level < 1 + NUM_BASE_LEVELS) {
914 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
915 return &txb_costs->base_cost[ctx][level];
916#else
Angie Chiang7afbba42017-05-30 15:59:15 -0700917 if (level == 0) {
Angie Chiang1ae0ebf2017-08-15 15:16:47 -0700918 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -0700919 return txb_costs->nz_map_cost[ctx];
Angie Chiang7afbba42017-05-30 15:59:15 -0700920 } else if (level >= 1 && level < 1 + NUM_BASE_LEVELS) {
921 const int idx = level - 1;
Angie Chiang1ae0ebf2017-08-15 15:16:47 -0700922 const int ctx = txb_cache->base_ctx_arr[idx][coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -0700923 return txb_costs->base_cost[idx][ctx];
Ola Hugosson13892102017-11-06 08:01:44 +0100924#endif
Angie Chiang7afbba42017-05-30 15:59:15 -0700925 } else if (level >= 1 + NUM_BASE_LEVELS &&
926 level < 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Angie Chiang1ae0ebf2017-08-15 15:16:47 -0700927 const int ctx = txb_cache->br_ctx_arr[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -0700928 return txb_costs->lps_cost[ctx];
Angie Chiang7afbba42017-05-30 15:59:15 -0700929 } else if (level >= 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Ola Hugosson13892102017-11-06 08:01:44 +0100930 // printf("get_level_prob does not support golomb\n");
Angie Chiang7afbba42017-05-30 15:59:15 -0700931 assert(0);
932 return 0;
933 } else {
934 assert(0);
935 return 0;
936 }
937}
938
Angie Chiangd19969e2017-05-30 18:02:33 -0700939static INLINE void update_mag_arr(int *mag_arr, int abs_qc) {
940 if (mag_arr[0] == abs_qc) {
941 mag_arr[1] -= 1;
942 assert(mag_arr[1] >= 0);
943 }
944}
945
Angie Chianga530ef42017-05-30 16:32:36 -0700946static INLINE int get_mag_from_mag_arr(const int *mag_arr) {
947 int mag;
948 if (mag_arr[1] > 0) {
949 mag = mag_arr[0];
950 } else if (mag_arr[0] > 0) {
951 mag = mag_arr[0] - 1;
952 } else {
953 // no neighbor
954 assert(mag_arr[0] == 0 && mag_arr[1] == 0);
955 mag = 0;
956 }
957 return mag;
958}
959
960static int neighbor_level_down_update(int *new_count, int *new_mag, int count,
961 const int *mag, int coeff_idx,
962 tran_low_t abs_nb_coeff, int nb_coeff_idx,
963 int level, const TxbInfo *txb_info) {
964 *new_count = count;
965 *new_mag = get_mag_from_mag_arr(mag);
966
967 int update = 0;
968 // check if br_count changes
969 if (abs_nb_coeff == level) {
970 update = 1;
971 *new_count -= 1;
972 assert(*new_count >= 0);
973 }
974 const int row = coeff_idx >> txb_info->bwl;
975 const int col = coeff_idx - (row << txb_info->bwl);
976 const int nb_row = nb_coeff_idx >> txb_info->bwl;
977 const int nb_col = nb_coeff_idx - (nb_row << txb_info->bwl);
978
979 // check if mag changes
980 if (nb_row >= row && nb_col >= col) {
981 if (abs_nb_coeff == mag[0]) {
982 assert(mag[1] > 0);
983 if (mag[1] == 1) {
984 // the nb is the only qc with max mag
985 *new_mag -= 1;
986 assert(*new_mag >= 0);
987 update = 1;
988 }
989 }
990 }
991 return update;
992}
993
994static int try_neighbor_level_down_br(int coeff_idx, int nb_coeff_idx,
995 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -0700996 const LV_MAP_COEFF_COST *txb_costs,
Angie Chianga530ef42017-05-30 16:32:36 -0700997 const TxbInfo *txb_info) {
998 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
999 const tran_low_t abs_qc = abs(qc);
1000 const int level = NUM_BASE_LEVELS + 1;
1001 if (abs_qc < level) return 0;
1002
1003 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1004 const tran_low_t abs_nb_coeff = abs(nb_coeff);
1005 const int count = txb_cache->br_count_arr[coeff_idx];
1006 const int *mag = txb_cache->br_mag_arr[coeff_idx];
1007 int new_count;
1008 int new_mag;
1009 const int update =
1010 neighbor_level_down_update(&new_count, &new_mag, count, mag, coeff_idx,
1011 abs_nb_coeff, nb_coeff_idx, level, txb_info);
1012 if (update) {
1013 const int row = coeff_idx >> txb_info->bwl;
1014 const int col = coeff_idx - (row << txb_info->bwl);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001015 const int ctx = txb_cache->br_ctx_arr[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001016 const int org_cost = get_br_cost(abs_qc, ctx, txb_costs->lps_cost[ctx]);
Angie Chianga530ef42017-05-30 16:32:36 -07001017
1018 const int new_ctx = get_br_ctx_from_count_mag(row, col, new_count, new_mag);
Jingning Handfd72322017-08-09 14:04:12 -07001019 const int new_cost =
1020 get_br_cost(abs_qc, new_ctx, txb_costs->lps_cost[new_ctx]);
Angie Chianga530ef42017-05-30 16:32:36 -07001021 const int cost_diff = -org_cost + new_cost;
1022 return cost_diff;
1023 } else {
1024 return 0;
1025 }
1026}
1027
1028static int try_neighbor_level_down_base(int coeff_idx, int nb_coeff_idx,
1029 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001030 const LV_MAP_COEFF_COST *txb_costs,
Angie Chianga530ef42017-05-30 16:32:36 -07001031 const TxbInfo *txb_info) {
Ola Hugosson13892102017-11-06 08:01:44 +01001032#if CONFIG_LV_MAP_MULTI
1033 // TODO(olah): not implemented yet
1034 (void)coeff_idx;
1035 (void)nb_coeff_idx;
1036 (void)txb_cache;
1037 (void)txb_costs;
1038 (void)txb_info;
1039 return 0;
1040#else
Angie Chianga530ef42017-05-30 16:32:36 -07001041 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1042 const tran_low_t abs_qc = abs(qc);
Angie Chiangdef11252017-08-18 14:37:32 -07001043 const BASE_CTX_TABLE *base_ctx_table =
1044 txb_info->coeff_ctx_table->base_ctx_table;
Angie Chianga530ef42017-05-30 16:32:36 -07001045
1046 int cost_diff = 0;
1047 for (int base_idx = 0; base_idx < NUM_BASE_LEVELS; ++base_idx) {
1048 const int level = base_idx + 1;
1049 if (abs_qc < level) continue;
1050
1051 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1052 const tran_low_t abs_nb_coeff = abs(nb_coeff);
1053
1054 const int count = txb_cache->base_count_arr[base_idx][coeff_idx];
1055 const int *mag = txb_cache->base_mag_arr[coeff_idx];
1056 int new_count;
1057 int new_mag;
1058 const int update =
1059 neighbor_level_down_update(&new_count, &new_mag, count, mag, coeff_idx,
1060 abs_nb_coeff, nb_coeff_idx, level, txb_info);
1061 if (update) {
1062 const int row = coeff_idx >> txb_info->bwl;
1063 const int col = coeff_idx - (row << txb_info->bwl);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001064 const int ctx = txb_cache->base_ctx_arr[base_idx][coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001065 const int org_cost = get_base_cost(
1066 abs_qc, ctx, txb_costs->base_cost[base_idx][ctx], base_idx);
Angie Chianga530ef42017-05-30 16:32:36 -07001067
1068 const int new_ctx =
Angie Chiangdef11252017-08-18 14:37:32 -07001069 base_ctx_table[row != 0][col != 0][new_mag > level][new_count];
Jingning Handfd72322017-08-09 14:04:12 -07001070 const int new_cost = get_base_cost(
1071 abs_qc, new_ctx, txb_costs->base_cost[base_idx][new_ctx], base_idx);
Angie Chianga530ef42017-05-30 16:32:36 -07001072 cost_diff += -org_cost + new_cost;
1073 }
1074 }
1075 return cost_diff;
Ola Hugosson13892102017-11-06 08:01:44 +01001076#endif
Angie Chianga530ef42017-05-30 16:32:36 -07001077}
1078
1079static int try_neighbor_level_down_nz(int coeff_idx, int nb_coeff_idx,
1080 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001081 const LV_MAP_COEFF_COST *txb_costs,
Angie Chianga530ef42017-05-30 16:32:36 -07001082 TxbInfo *txb_info) {
1083 // assume eob doesn't change
1084 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1085 const tran_low_t abs_qc = abs(qc);
1086 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1087 const tran_low_t abs_nb_coeff = abs(nb_coeff);
1088 if (abs_nb_coeff != 1) return 0;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001089 const int16_t *const iscan = txb_info->scan_order->iscan;
Angie Chianga530ef42017-05-30 16:32:36 -07001090 const int scan_idx = iscan[coeff_idx];
1091 if (scan_idx == txb_info->seg_eob) return 0;
1092 const int nb_scan_idx = iscan[nb_coeff_idx];
1093 if (nb_scan_idx < scan_idx) {
1094 const int count = txb_cache->nz_count_arr[coeff_idx];
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001095 (void)count;
Angie Chianga530ef42017-05-30 16:32:36 -07001096 assert(count > 0);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001097 update_qcoeff(nb_coeff_idx, get_lower_coeff(nb_coeff), txb_info);
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001098 const int new_ctx = get_nz_map_ctx_from_stats(
Dake He03a32922017-10-31 08:06:45 -07001099#if USE_CAUSAL_BASE_CTX
1100 0,
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001101#else
1102 count - 1,
Dake He03a32922017-10-31 08:06:45 -07001103#endif
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001104 coeff_idx, txb_info->bwl, txb_info->height, txb_info->tx_type);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001105 update_qcoeff(nb_coeff_idx, nb_coeff, txb_info);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001106 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
Ola Hugosson13892102017-11-06 08:01:44 +01001107#if CONFIG_LV_MAP_MULTI
1108 const int org_cost = txb_costs->base_cost[ctx][AOMMIN(abs_qc, 3)];
1109 const int new_cost = txb_costs->base_cost[new_ctx][AOMMIN(abs_qc, 3)];
1110#else
Angie Chianga530ef42017-05-30 16:32:36 -07001111 const int is_nz = abs_qc > 0;
Jingning Handfd72322017-08-09 14:04:12 -07001112 const int org_cost = txb_costs->nz_map_cost[ctx][is_nz];
1113 const int new_cost = txb_costs->nz_map_cost[new_ctx][is_nz];
Ola Hugosson13892102017-11-06 08:01:44 +01001114#endif
Angie Chianga530ef42017-05-30 16:32:36 -07001115 const int cost_diff = new_cost - org_cost;
1116 return cost_diff;
1117 } else {
1118 return 0;
1119 }
1120}
1121
1122static int try_self_level_down(tran_low_t *low_coeff, int coeff_idx,
1123 const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001124 const LV_MAP_COEFF_COST *txb_costs,
1125 TxbInfo *txb_info) {
Angie Chiang7afbba42017-05-30 15:59:15 -07001126 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1127 if (qc == 0) {
1128 *low_coeff = 0;
1129 return 0;
1130 }
1131 const tran_low_t abs_qc = abs(qc);
1132 *low_coeff = get_lower_coeff(qc);
1133 int cost_diff;
1134 if (*low_coeff == 0) {
1135 const int scan_idx = txb_info->scan_order->iscan[coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001136 const int *level_cost =
1137 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1138 const int *low_level_cost =
1139 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
Jingning Han3422ac12017-10-25 20:37:53 -07001140
Dake Hea47cd6c2017-10-13 18:09:58 -07001141 if (scan_idx < txb_info->eob - 1) {
Angie Chiang7afbba42017-05-30 15:59:15 -07001142 // When level-0, we code the binary of abs_qc > level
1143 // but when level-k k > 0 we code the binary of abs_qc == level
1144 // That's why wee need this special treatment for level-0 map
1145 // TODO(angiebird): make leve-0 consistent to other levels
Jingning Handfd72322017-08-09 14:04:12 -07001146 cost_diff = -level_cost[1] + low_level_cost[0] - low_level_cost[1];
Angie Chiang7afbba42017-05-30 15:59:15 -07001147 } else {
Jingning Handfd72322017-08-09 14:04:12 -07001148 cost_diff = -level_cost[1];
Angie Chiang7afbba42017-05-30 15:59:15 -07001149 }
1150
Angie Chiang7afbba42017-05-30 15:59:15 -07001151 const int sign_cost = get_sign_bit_cost(
Jingning Handfd72322017-08-09 14:04:12 -07001152 qc, coeff_idx, txb_costs->dc_sign_cost, txb_info->txb_ctx->dc_sign_ctx);
Angie Chiang7afbba42017-05-30 15:59:15 -07001153 cost_diff -= sign_cost;
Jingning Han772dee32017-09-15 08:53:18 -07001154 } else if (abs_qc <= NUM_BASE_LEVELS) {
1155 const int *level_cost =
1156 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1157 const int *low_level_cost =
1158 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
1159 cost_diff = -level_cost[1] + low_level_cost[1] - low_level_cost[0];
1160 } else if (abs_qc == NUM_BASE_LEVELS + 1) {
1161 const int *level_cost =
1162 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1163 const int *low_level_cost =
1164 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
1165 cost_diff = -level_cost[0] + low_level_cost[1] - low_level_cost[0];
Angie Chiang7afbba42017-05-30 15:59:15 -07001166 } else if (abs_qc < 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Jingning Handfd72322017-08-09 14:04:12 -07001167 const int *level_cost =
1168 get_level_prob(abs_qc, coeff_idx, txb_cache, txb_costs);
1169 const int *low_level_cost =
1170 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
1171
Jingning Han772dee32017-09-15 08:53:18 -07001172 cost_diff = -level_cost[abs_qc - 1 - NUM_BASE_LEVELS] +
1173 low_level_cost[abs(*low_coeff) - 1 - NUM_BASE_LEVELS];
Angie Chiang7afbba42017-05-30 15:59:15 -07001174 } else if (abs_qc == 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE) {
Jingning Handfd72322017-08-09 14:04:12 -07001175 const int *low_level_cost =
1176 get_level_prob(abs(*low_coeff), coeff_idx, txb_cache, txb_costs);
Jingning Han772dee32017-09-15 08:53:18 -07001177 cost_diff = -get_golomb_cost(abs_qc) - low_level_cost[COEFF_BASE_RANGE] +
1178 low_level_cost[COEFF_BASE_RANGE - 1];
Angie Chiang7afbba42017-05-30 15:59:15 -07001179 } else {
1180 assert(abs_qc > 1 + NUM_BASE_LEVELS + COEFF_BASE_RANGE);
1181 const tran_low_t abs_low_coeff = abs(*low_coeff);
1182 cost_diff = -get_golomb_cost(abs_qc) + get_golomb_cost(abs_low_coeff);
1183 }
1184 return cost_diff;
1185}
1186
Angie Chianga530ef42017-05-30 16:32:36 -07001187#define COST_MAP_SIZE 5
1188#define COST_MAP_OFFSET 2
1189
1190static INLINE int check_nz_neighbor(tran_low_t qc) { return abs(qc) == 1; }
1191
1192static INLINE int check_base_neighbor(tran_low_t qc) {
1193 return abs(qc) <= 1 + NUM_BASE_LEVELS;
1194}
1195
1196static INLINE int check_br_neighbor(tran_low_t qc) {
1197 return abs(qc) > BR_MAG_OFFSET;
1198}
1199
Angie Chiang87278292017-10-18 09:59:47 -07001200#define FAST_OPTIMIZE_TXB 1
Angie Chiange80957f2017-09-05 10:48:00 -07001201
1202#if FAST_OPTIMIZE_TXB
1203#define ALNB_REF_OFFSET_NUM 2
Linfeng Zhang4afda452017-10-24 10:34:04 -07001204static const int alnb_ref_offset[ALNB_REF_OFFSET_NUM][2] = {
Angie Chiange80957f2017-09-05 10:48:00 -07001205 { -1, 0 }, { 0, -1 },
1206};
1207#define NB_REF_OFFSET_NUM 4
Linfeng Zhang4afda452017-10-24 10:34:04 -07001208static const int nb_ref_offset[NB_REF_OFFSET_NUM][2] = {
Angie Chiange80957f2017-09-05 10:48:00 -07001209 { -1, 0 }, { 0, -1 }, { 1, 0 }, { 0, 1 },
1210};
1211#endif // FAST_OPTIMIZE_TXB
1212
Angie Chianga530ef42017-05-30 16:32:36 -07001213// TODO(angiebird): add static to this function once it's called
1214int try_level_down(int coeff_idx, const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001215 const LV_MAP_COEFF_COST *txb_costs, TxbInfo *txb_info,
Angie Chiang25645b72017-09-24 14:28:49 -07001216 int (*cost_map)[COST_MAP_SIZE], int fast_mode) {
1217#if !FAST_OPTIMIZE_TXB
1218 (void)fast_mode;
1219#endif
Angie Chianga530ef42017-05-30 16:32:36 -07001220 if (cost_map) {
1221 for (int i = 0; i < COST_MAP_SIZE; ++i) av1_zero(cost_map[i]);
1222 }
1223
1224 tran_low_t qc = txb_info->qcoeff[coeff_idx];
1225 tran_low_t low_coeff;
1226 if (qc == 0) return 0;
1227 int accu_cost_diff = 0;
1228
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001229 const int16_t *const iscan = txb_info->scan_order->iscan;
Angie Chianga530ef42017-05-30 16:32:36 -07001230 const int eob = txb_info->eob;
1231 const int scan_idx = iscan[coeff_idx];
1232 if (scan_idx < eob) {
1233 const int cost_diff = try_self_level_down(&low_coeff, coeff_idx, txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001234 txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001235 if (cost_map)
1236 cost_map[0 + COST_MAP_OFFSET][0 + COST_MAP_OFFSET] = cost_diff;
1237 accu_cost_diff += cost_diff;
1238 }
1239
1240 const int row = coeff_idx >> txb_info->bwl;
1241 const int col = coeff_idx - (row << txb_info->bwl);
1242 if (check_nz_neighbor(qc)) {
Angie Chiange80957f2017-09-05 10:48:00 -07001243#if FAST_OPTIMIZE_TXB
Linfeng Zhang4afda452017-10-24 10:34:04 -07001244 const int(*ref_offset)[2];
Angie Chiang25645b72017-09-24 14:28:49 -07001245 int ref_num;
1246 if (fast_mode) {
1247 ref_offset = alnb_ref_offset;
1248 ref_num = ALNB_REF_OFFSET_NUM;
1249 } else {
1250 ref_offset = sig_ref_offset;
1251 ref_num = SIG_REF_OFFSET_NUM;
1252 }
Angie Chiange80957f2017-09-05 10:48:00 -07001253#else
Linfeng Zhang4afda452017-10-24 10:34:04 -07001254 const int(*ref_offset)[2] = sig_ref_offset;
Angie Chiange80957f2017-09-05 10:48:00 -07001255 const int ref_num = SIG_REF_OFFSET_NUM;
1256#endif
1257 for (int i = 0; i < ref_num; ++i) {
1258 const int nb_row = row - ref_offset[i][0];
1259 const int nb_col = col - ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001260 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han4cbb3632017-06-13 12:50:33 -07001261
Angie Chiang0c89dca2017-08-17 16:36:18 -07001262 if (nb_row < 0 || nb_col < 0 || nb_row >= txb_info->height ||
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001263 nb_col >= txb_info->width)
Jingning Han3455e762017-06-13 21:15:44 -07001264 continue;
1265
Angie Chianga530ef42017-05-30 16:32:36 -07001266 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001267 if (nb_scan_idx < eob) {
Angie Chianga530ef42017-05-30 16:32:36 -07001268 const int cost_diff = try_neighbor_level_down_nz(
Jingning Handfd72322017-08-09 14:04:12 -07001269 nb_coeff_idx, coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001270 if (cost_map)
1271 cost_map[nb_row - row + COST_MAP_OFFSET]
1272 [nb_col - col + COST_MAP_OFFSET] += cost_diff;
1273 accu_cost_diff += cost_diff;
1274 }
1275 }
1276 }
1277
1278 if (check_base_neighbor(qc)) {
Angie Chiange80957f2017-09-05 10:48:00 -07001279#if FAST_OPTIMIZE_TXB
Linfeng Zhang4afda452017-10-24 10:34:04 -07001280 const int(*ref_offset)[2];
Angie Chiang25645b72017-09-24 14:28:49 -07001281 int ref_num;
1282 if (fast_mode) {
1283 ref_offset = nb_ref_offset;
1284 ref_num = NB_REF_OFFSET_NUM;
1285 } else {
1286 ref_offset = base_ref_offset;
1287 ref_num = BASE_CONTEXT_POSITION_NUM;
1288 }
Angie Chiange80957f2017-09-05 10:48:00 -07001289#else
Linfeng Zhangce065ca2017-10-17 16:49:30 -07001290 const int(*ref_offset)[2] = base_ref_offset;
Angie Chiang25645b72017-09-24 14:28:49 -07001291 int ref_num = BASE_CONTEXT_POSITION_NUM;
Angie Chiange80957f2017-09-05 10:48:00 -07001292#endif
1293 for (int i = 0; i < ref_num; ++i) {
1294 const int nb_row = row - ref_offset[i][0];
1295 const int nb_col = col - ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001296 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001297
Angie Chiang0c89dca2017-08-17 16:36:18 -07001298 if (nb_row < 0 || nb_col < 0 || nb_row >= txb_info->height ||
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001299 nb_col >= txb_info->width)
Jingning Han3455e762017-06-13 21:15:44 -07001300 continue;
1301
Angie Chianga530ef42017-05-30 16:32:36 -07001302 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001303 if (nb_scan_idx < eob) {
Angie Chianga530ef42017-05-30 16:32:36 -07001304 const int cost_diff = try_neighbor_level_down_base(
Jingning Handfd72322017-08-09 14:04:12 -07001305 nb_coeff_idx, coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001306 if (cost_map)
1307 cost_map[nb_row - row + COST_MAP_OFFSET]
1308 [nb_col - col + COST_MAP_OFFSET] += cost_diff;
1309 accu_cost_diff += cost_diff;
1310 }
1311 }
1312 }
1313
1314 if (check_br_neighbor(qc)) {
Angie Chiange80957f2017-09-05 10:48:00 -07001315#if FAST_OPTIMIZE_TXB
Linfeng Zhang4afda452017-10-24 10:34:04 -07001316 const int(*ref_offset)[2];
Angie Chiang25645b72017-09-24 14:28:49 -07001317 int ref_num;
1318 if (fast_mode) {
1319 ref_offset = nb_ref_offset;
1320 ref_num = NB_REF_OFFSET_NUM;
1321 } else {
1322 ref_offset = br_ref_offset;
1323 ref_num = BR_CONTEXT_POSITION_NUM;
1324 }
Angie Chiange80957f2017-09-05 10:48:00 -07001325#else
Linfeng Zhangce065ca2017-10-17 16:49:30 -07001326 const int(*ref_offset)[2] = br_ref_offset;
Angie Chiange80957f2017-09-05 10:48:00 -07001327 const int ref_num = BR_CONTEXT_POSITION_NUM;
1328#endif
1329 for (int i = 0; i < ref_num; ++i) {
1330 const int nb_row = row - ref_offset[i][0];
1331 const int nb_col = col - ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001332 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001333
Angie Chiang0c89dca2017-08-17 16:36:18 -07001334 if (nb_row < 0 || nb_col < 0 || nb_row >= txb_info->height ||
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001335 nb_col >= txb_info->width)
Jingning Han3455e762017-06-13 21:15:44 -07001336 continue;
1337
Angie Chianga530ef42017-05-30 16:32:36 -07001338 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001339 if (nb_scan_idx < eob) {
Angie Chianga530ef42017-05-30 16:32:36 -07001340 const int cost_diff = try_neighbor_level_down_br(
Jingning Handfd72322017-08-09 14:04:12 -07001341 nb_coeff_idx, coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chianga530ef42017-05-30 16:32:36 -07001342 if (cost_map)
1343 cost_map[nb_row - row + COST_MAP_OFFSET]
1344 [nb_col - col + COST_MAP_OFFSET] += cost_diff;
1345 accu_cost_diff += cost_diff;
1346 }
1347 }
1348 }
1349
1350 return accu_cost_diff;
1351}
1352
Angie Chiangc77799b2017-05-30 17:08:17 -07001353static int get_low_coeff_cost(int coeff_idx, const TxbCache *txb_cache,
Jingning Handfd72322017-08-09 14:04:12 -07001354 const LV_MAP_COEFF_COST *txb_costs,
Angie Chiangc77799b2017-05-30 17:08:17 -07001355 const TxbInfo *txb_info) {
1356 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1357 const int abs_qc = abs(qc);
1358 assert(abs_qc <= 1);
1359 int cost = 0;
Jingning Han3422ac12017-10-25 20:37:53 -07001360
Ola Hugosson13892102017-11-06 08:01:44 +01001361#if CONFIG_LV_MAP_MULTI
1362 const int ctx = txb_cache->nz_ctx_arr[coeff_idx];
1363 cost += txb_costs->base_cost[ctx][AOMMIN(abs_qc, 3)];
1364 if (qc != 0) {
1365 cost += get_sign_bit_cost(qc, coeff_idx, txb_costs->dc_sign_cost,
1366 txb_info->txb_ctx->dc_sign_ctx);
1367 }
1368#else
1369 const int scan_idx = txb_info->scan_order->iscan[coeff_idx];
Dake Hea47cd6c2017-10-13 18:09:58 -07001370 if (scan_idx < txb_info->eob - 1) {
Jingning Handfd72322017-08-09 14:04:12 -07001371 const int *level_cost = get_level_prob(0, coeff_idx, txb_cache, txb_costs);
1372 cost += level_cost[qc != 0];
Angie Chiangc77799b2017-05-30 17:08:17 -07001373 }
1374
1375 if (qc != 0) {
1376 const int base_idx = 0;
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001377 const int ctx = txb_cache->base_ctx_arr[base_idx][coeff_idx];
Jingning Handfd72322017-08-09 14:04:12 -07001378 cost += get_base_cost(abs_qc, ctx, txb_costs->base_cost[base_idx][ctx],
1379 base_idx);
Jingning Handfd72322017-08-09 14:04:12 -07001380 cost += get_sign_bit_cost(qc, coeff_idx, txb_costs->dc_sign_cost,
Angie Chiangc77799b2017-05-30 17:08:17 -07001381 txb_info->txb_ctx->dc_sign_ctx);
1382 }
Ola Hugosson13892102017-11-06 08:01:44 +01001383#endif
Angie Chiangc77799b2017-05-30 17:08:17 -07001384 return cost;
1385}
1386
1387static INLINE void set_eob(TxbInfo *txb_info, int eob) {
1388 txb_info->eob = eob;
Urvang Joshi80893152017-10-27 11:51:14 -07001389 txb_info->seg_eob = av1_get_max_eob(txb_info->tx_size);
Angie Chiangc77799b2017-05-30 17:08:17 -07001390}
1391
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001392static INLINE int get_eob_ctx(const int coeff_idx, // raster order
1393 const TX_SIZE txs_ctx) {
1394 if (txs_ctx == TX_4X4) return av1_coeff_band_4x4[coeff_idx];
1395 if (txs_ctx == TX_8X8) return av1_coeff_band_8x8[coeff_idx];
1396 if (txs_ctx == TX_16X16) return av1_coeff_band_16x16[coeff_idx];
1397 if (txs_ctx == TX_32X32) return av1_coeff_band_32x32[coeff_idx];
1398
1399#if CONFIG_TX64X64
1400 // Since TX64X64 use 32x32 coeff buffer, so it share the same coeff_band with
1401 // TX32X32
1402 if (txs_ctx == TX_64X64) return av1_coeff_band_32x32[coeff_idx];
1403#endif // CONFIG_TX64X64
1404
1405 assert(0 && "Invalid value of txs_ctx");
1406 return 0;
1407}
1408
Angie Chiangc77799b2017-05-30 17:08:17 -07001409// TODO(angiebird): add static to this function once it's called
1410int try_change_eob(int *new_eob, int coeff_idx, const TxbCache *txb_cache,
Angie Chiang25645b72017-09-24 14:28:49 -07001411 const LV_MAP_COEFF_COST *txb_costs, TxbInfo *txb_info,
1412 int fast_mode) {
Angie Chiangc77799b2017-05-30 17:08:17 -07001413 assert(txb_info->eob > 0);
1414 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1415 const int abs_qc = abs(qc);
1416 if (abs_qc != 1) {
1417 *new_eob = -1;
1418 return 0;
1419 }
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001420 const int16_t *const iscan = txb_info->scan_order->iscan;
1421 const int16_t *const scan = txb_info->scan_order->scan;
Angie Chiangc77799b2017-05-30 17:08:17 -07001422 const int scan_idx = iscan[coeff_idx];
1423 *new_eob = 0;
1424 int cost_diff = 0;
Jingning Handfd72322017-08-09 14:04:12 -07001425 cost_diff -= get_low_coeff_cost(coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chiangc77799b2017-05-30 17:08:17 -07001426 // int coeff_cost =
Angie Chiang5e012fe2017-05-30 18:47:39 -07001427 // get_coeff_cost(qc, scan_idx, txb_info, txb_probs);
Angie Chiangc77799b2017-05-30 17:08:17 -07001428 // if (-cost_diff != coeff_cost) {
1429 // printf("-cost_diff %d coeff_cost %d\n", -cost_diff, coeff_cost);
1430 // get_low_coeff_cost(coeff_idx, txb_cache, txb_probs, txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001431 // get_coeff_cost(qc, scan_idx, txb_info, txb_probs);
Angie Chiangc77799b2017-05-30 17:08:17 -07001432 // }
1433 for (int si = scan_idx - 1; si >= 0; --si) {
1434 const int ci = scan[si];
1435 if (txb_info->qcoeff[ci] != 0) {
1436 *new_eob = si + 1;
1437 break;
1438 } else {
Jingning Handfd72322017-08-09 14:04:12 -07001439 cost_diff -= get_low_coeff_cost(ci, txb_cache, txb_costs, txb_info);
Angie Chiangc77799b2017-05-30 17:08:17 -07001440 }
1441 }
1442
1443 const int org_eob = txb_info->eob;
1444 set_eob(txb_info, *new_eob);
Angie Chiang25645b72017-09-24 14:28:49 -07001445 cost_diff += try_level_down(coeff_idx, txb_cache, txb_costs, txb_info, NULL,
1446 fast_mode);
Angie Chiangc77799b2017-05-30 17:08:17 -07001447 set_eob(txb_info, org_eob);
1448
1449 if (*new_eob > 0) {
1450 // Note that get_eob_ctx does NOT actually account for qcoeff, so we don't
1451 // need to lower down the qcoeff here
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001452 const int eob_ctx = get_eob_ctx(scan[*new_eob - 1], txb_info->txs_ctx);
Jingning Handfd72322017-08-09 14:04:12 -07001453 cost_diff -= txb_costs->eob_cost[eob_ctx][0];
1454 cost_diff += txb_costs->eob_cost[eob_ctx][1];
Angie Chiangc77799b2017-05-30 17:08:17 -07001455 } else {
1456 const int txb_skip_ctx = txb_info->txb_ctx->txb_skip_ctx;
Jingning Handfd72322017-08-09 14:04:12 -07001457 cost_diff -= txb_costs->txb_skip_cost[txb_skip_ctx][0];
1458 cost_diff += txb_costs->txb_skip_cost[txb_skip_ctx][1];
Angie Chiangc77799b2017-05-30 17:08:17 -07001459 }
1460 return cost_diff;
1461}
Angie Chiang47e07072017-05-30 17:27:01 -07001462
Angie Chiangd19969e2017-05-30 18:02:33 -07001463// TODO(angiebird): add static to this function it's called
Linfeng Zhang1015a342017-10-24 16:20:41 -07001464void update_level_down(const int coeff_idx, TxbCache *const txb_cache,
1465 TxbInfo *const txb_info) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001466 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1467 const int abs_qc = abs(qc);
1468 if (qc == 0) return;
1469 const tran_low_t low_coeff = get_lower_coeff(qc);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001470 update_coeff(coeff_idx, low_coeff, txb_info);
Angie Chiangd19969e2017-05-30 18:02:33 -07001471
1472 const int row = coeff_idx >> txb_info->bwl;
1473 const int col = coeff_idx - (row << txb_info->bwl);
1474 const int eob = txb_info->eob;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001475 const int16_t *const iscan = txb_info->scan_order->iscan;
Angie Chiangd19969e2017-05-30 18:02:33 -07001476 for (int i = 0; i < SIG_REF_OFFSET_NUM; ++i) {
1477 const int nb_row = row - sig_ref_offset[i][0];
1478 const int nb_col = col - sig_ref_offset[i][1];
Jingning Han3455e762017-06-13 21:15:44 -07001479
1480 if (!(nb_row >= 0 && nb_col >= 0 && nb_row < txb_info->height &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001481 nb_col < txb_info->width))
Jingning Han3455e762017-06-13 21:15:44 -07001482 continue;
1483
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001484 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Angie Chiangd19969e2017-05-30 18:02:33 -07001485 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001486 if (nb_scan_idx < eob) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001487 const int scan_idx = iscan[coeff_idx];
1488 if (scan_idx < nb_scan_idx) {
1489 const int level = 1;
1490 if (abs_qc == level) {
1491 txb_cache->nz_count_arr[nb_coeff_idx] -= 1;
1492 assert(txb_cache->nz_count_arr[nb_coeff_idx] >= 0);
1493 }
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001494 txb_cache->nz_ctx_arr[nb_coeff_idx] = get_nz_map_ctx_from_stats(
Dake He03a32922017-10-31 08:06:45 -07001495#if USE_CAUSAL_BASE_CTX
1496 0,
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001497#else
1498 txb_cache->nz_count_arr[nb_coeff_idx],
Dake He03a32922017-10-31 08:06:45 -07001499#endif
Linfeng Zhang8ac45572017-11-29 11:39:51 -08001500 nb_coeff_idx, txb_info->bwl, txb_info->height, txb_info->tx_type);
Angie Chiangd19969e2017-05-30 18:02:33 -07001501 }
1502 }
1503 }
1504
Angie Chiangdef11252017-08-18 14:37:32 -07001505 const BASE_CTX_TABLE *base_ctx_table =
1506 txb_info->coeff_ctx_table->base_ctx_table;
Angie Chiangd19969e2017-05-30 18:02:33 -07001507 for (int i = 0; i < BASE_CONTEXT_POSITION_NUM; ++i) {
1508 const int nb_row = row - base_ref_offset[i][0];
1509 const int nb_col = col - base_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001510 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001511
1512 if (!(nb_row >= 0 && nb_col >= 0 && nb_row < txb_info->height &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001513 nb_col < txb_info->width))
Jingning Han3455e762017-06-13 21:15:44 -07001514 continue;
1515
Angie Chiangd19969e2017-05-30 18:02:33 -07001516 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1517 if (!has_base(nb_coeff, 0)) continue;
1518 const int nb_scan_idx = iscan[nb_coeff_idx];
Jingning Han3455e762017-06-13 21:15:44 -07001519 if (nb_scan_idx < eob) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001520 if (row >= nb_row && col >= nb_col)
1521 update_mag_arr(txb_cache->base_mag_arr[nb_coeff_idx], abs_qc);
1522 const int mag =
1523 get_mag_from_mag_arr(txb_cache->base_mag_arr[nb_coeff_idx]);
1524 for (int base_idx = 0; base_idx < NUM_BASE_LEVELS; ++base_idx) {
1525 if (!has_base(nb_coeff, base_idx)) continue;
1526 const int level = base_idx + 1;
1527 if (abs_qc == level) {
1528 txb_cache->base_count_arr[base_idx][nb_coeff_idx] -= 1;
1529 assert(txb_cache->base_count_arr[base_idx][nb_coeff_idx] >= 0);
1530 }
1531 const int count = txb_cache->base_count_arr[base_idx][nb_coeff_idx];
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001532 txb_cache->base_ctx_arr[base_idx][nb_coeff_idx] =
Angie Chiangdef11252017-08-18 14:37:32 -07001533 base_ctx_table[nb_row != 0][nb_col != 0][mag > level][count];
Angie Chiangd19969e2017-05-30 18:02:33 -07001534 // int ref_ctx = get_base_ctx(txb_info->qcoeff, nb_coeff_idx,
Linfeng Zhang8c841f72017-10-18 15:32:29 -07001535 // txb_info->bwl, level - 1);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001536 // if (ref_ctx != txb_cache->base_ctx_arr[base_idx][nb_coeff_idx]) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001537 // printf("base ctx %d ref_ctx %d\n",
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001538 // txb_cache->base_ctx_arr[base_idx][nb_coeff_idx], ref_ctx);
Angie Chiangd19969e2017-05-30 18:02:33 -07001539 // }
1540 }
1541 }
1542 }
1543
1544 for (int i = 0; i < BR_CONTEXT_POSITION_NUM; ++i) {
1545 const int nb_row = row - br_ref_offset[i][0];
1546 const int nb_col = col - br_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001547 const int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Jingning Han3455e762017-06-13 21:15:44 -07001548
1549 if (!(nb_row >= 0 && nb_col >= 0 && nb_row < txb_info->height &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001550 nb_col < txb_info->width))
Jingning Han3455e762017-06-13 21:15:44 -07001551 continue;
1552
Angie Chiangd19969e2017-05-30 18:02:33 -07001553 const int nb_scan_idx = iscan[nb_coeff_idx];
1554 const tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
1555 if (!has_br(nb_coeff)) continue;
Jingning Han3455e762017-06-13 21:15:44 -07001556 if (nb_scan_idx < eob) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001557 const int level = 1 + NUM_BASE_LEVELS;
1558 if (abs_qc == level) {
1559 txb_cache->br_count_arr[nb_coeff_idx] -= 1;
1560 assert(txb_cache->br_count_arr[nb_coeff_idx] >= 0);
1561 }
1562 if (row >= nb_row && col >= nb_col)
1563 update_mag_arr(txb_cache->br_mag_arr[nb_coeff_idx], abs_qc);
1564 const int count = txb_cache->br_count_arr[nb_coeff_idx];
1565 const int mag = get_mag_from_mag_arr(txb_cache->br_mag_arr[nb_coeff_idx]);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001566 txb_cache->br_ctx_arr[nb_coeff_idx] =
Angie Chiangd19969e2017-05-30 18:02:33 -07001567 get_br_ctx_from_count_mag(nb_row, nb_col, count, mag);
1568 // int ref_ctx = get_level_ctx(txb_info->qcoeff, nb_coeff_idx,
1569 // txb_info->bwl);
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001570 // if (ref_ctx != txb_cache->br_ctx_arr[nb_coeff_idx]) {
Angie Chiangd19969e2017-05-30 18:02:33 -07001571 // printf("base ctx %d ref_ctx %d\n",
Angie Chiang1ae0ebf2017-08-15 15:16:47 -07001572 // txb_cache->br_ctx_arr[nb_coeff_idx], ref_ctx);
Angie Chiangd19969e2017-05-30 18:02:33 -07001573 // }
1574 }
1575 }
1576}
1577
Linfeng Zhang1015a342017-10-24 16:20:41 -07001578static int get_coeff_cost(const tran_low_t qc, const int scan_idx,
Ola Hugosson13892102017-11-06 08:01:44 +01001579#if CONFIG_LV_MAP_MULTI
1580 const int is_eob,
1581#endif
Linfeng Zhang1015a342017-10-24 16:20:41 -07001582 const TxbInfo *const txb_info,
1583 const LV_MAP_COEFF_COST *const txb_costs) {
Angie Chiang488f9212017-05-30 12:46:26 -07001584 const TXB_CTX *txb_ctx = txb_info->txb_ctx;
1585 const int is_nz = (qc != 0);
1586 const tran_low_t abs_qc = abs(qc);
1587 int cost = 0;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001588 const int16_t *const scan = txb_info->scan_order->scan;
1589 const int pos = scan[scan_idx];
Dake He03a32922017-10-31 08:06:45 -07001590
Ola Hugosson13892102017-11-06 08:01:44 +01001591#if CONFIG_LV_MAP_MULTI
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -08001592 const int coeff_ctx =
1593 get_nz_map_ctx(txb_info->levels, pos, txb_info->bwl, txb_info->height,
1594 scan_idx, is_eob, txb_info->tx_type);
Dake He3fe369c2017-11-16 17:56:44 -08001595#if USE_BASE_EOB_ALPHABET
1596 if (is_eob) {
1597 cost +=
1598 txb_costs->base_eob_cost[coeff_ctx - SIG_COEF_CONTEXTS +
1599 SIG_COEF_CONTEXTS_EOB][AOMMIN(abs_qc, 3) - 1];
1600 } else {
1601 cost += txb_costs->base_cost[coeff_ctx][AOMMIN(abs_qc, 3)];
1602 }
1603
1604#else
Ola Hugosson13892102017-11-06 08:01:44 +01001605 cost += txb_costs->base_cost[coeff_ctx][AOMMIN(abs_qc, 3)];
Dake He3fe369c2017-11-16 17:56:44 -08001606#endif
Ola Hugosson13892102017-11-06 08:01:44 +01001607#else
Dake He03a32922017-10-31 08:06:45 -07001608#if USE_CAUSAL_BASE_CTX
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -08001609 const int coeff_ctx = get_nz_map_ctx(txb_info->levels, pos, txb_info->bwl,
1610 txb_info->height, txb_info->tx_type);
Dake He03a32922017-10-31 08:06:45 -07001611#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07001612 if (scan_idx < txb_info->eob - 1) {
Dake He03a32922017-10-31 08:06:45 -07001613#if !USE_CAUSAL_BASE_CTX
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -08001614 const int coeff_ctx = get_nz_map_ctx(txb_info->levels, pos, txb_info->bwl,
1615 txb_info->height, txb_info->tx_type);
Dake He03a32922017-10-31 08:06:45 -07001616#endif
Jingning Handfd72322017-08-09 14:04:12 -07001617 cost += txb_costs->nz_map_cost[coeff_ctx][is_nz];
Angie Chiang488f9212017-05-30 12:46:26 -07001618 }
Ola Hugosson13892102017-11-06 08:01:44 +01001619#endif
Angie Chiang488f9212017-05-30 12:46:26 -07001620 if (is_nz) {
Jingning Handfd72322017-08-09 14:04:12 -07001621 cost += get_sign_bit_cost(qc, scan_idx, txb_costs->dc_sign_cost,
Angie Chiang488f9212017-05-30 12:46:26 -07001622 txb_ctx->dc_sign_ctx);
1623
Ola Hugosson13892102017-11-06 08:01:44 +01001624#if !CONFIG_LV_MAP_MULTI
Dake He03a32922017-10-31 08:06:45 -07001625#if USE_CAUSAL_BASE_CTX
1626 int k;
1627 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
1628 int ctx = coeff_ctx;
1629 int is_k = (abs_qc > (k + 1));
1630
1631 cost += txb_costs->base_cost[k][ctx][is_k];
1632 if (is_k == 0) break;
1633 }
1634#else
Angie Chiang488f9212017-05-30 12:46:26 -07001635 int ctx_ls[NUM_BASE_LEVELS] = { 0 };
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001636 get_base_ctx_set(txb_info->levels, pos, txb_info->bwl, ctx_ls);
Angie Chiang488f9212017-05-30 12:46:26 -07001637
1638 int i;
1639 for (i = 0; i < NUM_BASE_LEVELS; ++i) {
Jingning Handfd72322017-08-09 14:04:12 -07001640 cost += get_base_cost(abs_qc, ctx_ls[i],
1641 txb_costs->base_cost[i][ctx_ls[i]], i);
Angie Chiang488f9212017-05-30 12:46:26 -07001642 }
Dake He03a32922017-10-31 08:06:45 -07001643#endif
Ola Hugosson13892102017-11-06 08:01:44 +01001644#endif
Angie Chiang488f9212017-05-30 12:46:26 -07001645 if (abs_qc > NUM_BASE_LEVELS) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001646 const int row = pos >> txb_info->bwl;
1647 const int col = pos - (row << txb_info->bwl);
Dake He7d01ab52017-11-24 17:53:28 -08001648
1649#if CONFIG_LV_MAP_MULTI && USE_CAUSAL_BR_CTX
1650 (void)col;
1651 const int count = 0;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001652 const int ctx = get_br_ctx(txb_info->levels, pos, txb_info->bwl, count,
1653 txb_info->tx_type);
Dake He7d01ab52017-11-24 17:53:28 -08001654#else
Linfeng Zhang97fc4742017-11-07 12:57:25 -08001655 const int count = get_level_count(
1656 txb_info->levels, (1 << txb_info->bwl) + TX_PAD_HOR, row, col,
1657 NUM_BASE_LEVELS, br_ref_offset, BR_CONTEXT_POSITION_NUM);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001658 const int ctx = get_br_ctx(txb_info->levels, pos, txb_info->bwl, count);
Dake He7d01ab52017-11-24 17:53:28 -08001659#endif
Jingning Handfd72322017-08-09 14:04:12 -07001660 cost += get_br_cost(abs_qc, ctx, txb_costs->lps_cost[ctx]);
Angie Chiang488f9212017-05-30 12:46:26 -07001661 cost += get_golomb_cost(abs_qc);
1662 }
Angie Chiang488f9212017-05-30 12:46:26 -07001663 }
1664 return cost;
1665}
1666
Angie Chiang47e07072017-05-30 17:27:01 -07001667#if TEST_OPTIMIZE_TXB
Angie Chiang5e012fe2017-05-30 18:47:39 -07001668#define ALL_REF_OFFSET_NUM 17
Linfeng Zhang4afda452017-10-24 10:34:04 -07001669static const int all_ref_offset[ALL_REF_OFFSET_NUM][2] = {
Angie Chiang5e012fe2017-05-30 18:47:39 -07001670 { 0, 0 }, { -2, -1 }, { -2, 0 }, { -2, 1 }, { -1, -2 }, { -1, -1 },
1671 { -1, 0 }, { -1, 1 }, { 0, -2 }, { 0, -1 }, { 1, -2 }, { 1, -1 },
1672 { 1, 0 }, { 2, 0 }, { 0, 1 }, { 0, 2 }, { 1, 1 },
1673};
1674
Angie Chiang0b2795c2017-09-29 16:00:08 -07001675static int try_level_down_ref(int coeff_idx, const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang5e012fe2017-05-30 18:47:39 -07001676 TxbInfo *txb_info,
1677 int (*cost_map)[COST_MAP_SIZE]) {
1678 if (cost_map) {
1679 for (int i = 0; i < COST_MAP_SIZE; ++i) av1_zero(cost_map[i]);
1680 }
1681 tran_low_t qc = txb_info->qcoeff[coeff_idx];
1682 if (qc == 0) return 0;
1683 int row = coeff_idx >> txb_info->bwl;
1684 int col = coeff_idx - (row << txb_info->bwl);
1685 int org_cost = 0;
1686 for (int i = 0; i < ALL_REF_OFFSET_NUM; ++i) {
1687 int nb_row = row - all_ref_offset[i][0];
1688 int nb_col = col - all_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001689 int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Angie Chiang5e012fe2017-05-30 18:47:39 -07001690 int nb_scan_idx = txb_info->scan_order->iscan[nb_coeff_idx];
1691 if (nb_scan_idx < txb_info->eob && nb_row >= 0 && nb_col >= 0 &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001692 nb_row < txb_info->height && nb_col < txb_info->width) {
Angie Chiang5e012fe2017-05-30 18:47:39 -07001693 tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
Ola Hugosson13892102017-11-06 08:01:44 +01001694#if CONFIG_LV_MAP_MULTI
1695 int cost =
1696 get_coeff_cost(nb_coeff, nb_scan_idx, is_eob, txb_info, txb_costs);
1697#else
Angie Chiang0b2795c2017-09-29 16:00:08 -07001698 int cost = get_coeff_cost(nb_coeff, nb_scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +01001699#endif
Angie Chiang5e012fe2017-05-30 18:47:39 -07001700 if (cost_map)
1701 cost_map[nb_row - row + COST_MAP_OFFSET]
1702 [nb_col - col + COST_MAP_OFFSET] -= cost;
1703 org_cost += cost;
1704 }
1705 }
Linfeng Zhang1015a342017-10-24 16:20:41 -07001706 update_qcoeff(coeff_idx, get_lower_coeff(qc), txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001707 int new_cost = 0;
1708 for (int i = 0; i < ALL_REF_OFFSET_NUM; ++i) {
1709 int nb_row = row - all_ref_offset[i][0];
1710 int nb_col = col - all_ref_offset[i][1];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001711 int nb_coeff_idx = nb_row * txb_info->width + nb_col;
Angie Chiang5e012fe2017-05-30 18:47:39 -07001712 int nb_scan_idx = txb_info->scan_order->iscan[nb_coeff_idx];
1713 if (nb_scan_idx < txb_info->eob && nb_row >= 0 && nb_col >= 0 &&
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001714 nb_row < txb_info->height && nb_col < txb_info->width) {
Angie Chiang5e012fe2017-05-30 18:47:39 -07001715 tran_low_t nb_coeff = txb_info->qcoeff[nb_coeff_idx];
Ola Hugosson13892102017-11-06 08:01:44 +01001716#if CONFIG_LV_MAP_MULTI
1717 int cost =
1718 get_coeff_cost(nb_coeff, nb_scan_idx, is_eob, txb_info, txb_costs);
1719#else
Angie Chiang0b2795c2017-09-29 16:00:08 -07001720 int cost = get_coeff_cost(nb_coeff, nb_scan_idx, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +01001721#endif
Angie Chiang5e012fe2017-05-30 18:47:39 -07001722 if (cost_map)
1723 cost_map[nb_row - row + COST_MAP_OFFSET]
1724 [nb_col - col + COST_MAP_OFFSET] += cost;
1725 new_cost += cost;
1726 }
1727 }
Linfeng Zhang1015a342017-10-24 16:20:41 -07001728 update_qcoeff(coeff_idx, qc, txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001729 return new_cost - org_cost;
1730}
1731
1732static void test_level_down(int coeff_idx, const TxbCache *txb_cache,
Angie Chiang0b2795c2017-09-29 16:00:08 -07001733 const LV_MAP_COEFF_COST *txb_costs,
1734 TxbInfo *txb_info) {
Angie Chiang47e07072017-05-30 17:27:01 -07001735 int cost_map[COST_MAP_SIZE][COST_MAP_SIZE];
1736 int ref_cost_map[COST_MAP_SIZE][COST_MAP_SIZE];
1737 const int cost_diff =
Angie Chiang0b2795c2017-09-29 16:00:08 -07001738 try_level_down(coeff_idx, txb_cache, txb_costs, txb_info, cost_map, 0);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001739 const int cost_diff_ref =
Angie Chiang0b2795c2017-09-29 16:00:08 -07001740 try_level_down_ref(coeff_idx, txb_costs, txb_info, ref_cost_map);
Angie Chiang47e07072017-05-30 17:27:01 -07001741 if (cost_diff != cost_diff_ref) {
1742 printf("qc %d cost_diff %d cost_diff_ref %d\n", txb_info->qcoeff[coeff_idx],
1743 cost_diff, cost_diff_ref);
1744 for (int r = 0; r < COST_MAP_SIZE; ++r) {
1745 for (int c = 0; c < COST_MAP_SIZE; ++c) {
1746 printf("%d:%d ", cost_map[r][c], ref_cost_map[r][c]);
1747 }
1748 printf("\n");
1749 }
1750 }
1751}
1752#endif
1753
Angie Chiang488f9212017-05-30 12:46:26 -07001754// TODO(angiebird): make this static once it's called
Jingning Handfd72322017-08-09 14:04:12 -07001755int get_txb_cost(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs) {
Angie Chiang488f9212017-05-30 12:46:26 -07001756 int cost = 0;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001757 const int txb_skip_ctx = txb_info->txb_ctx->txb_skip_ctx;
1758 const int16_t *const scan = txb_info->scan_order->scan;
Angie Chiang488f9212017-05-30 12:46:26 -07001759 if (txb_info->eob == 0) {
Jingning Handfd72322017-08-09 14:04:12 -07001760 cost = txb_costs->txb_skip_cost[txb_skip_ctx][1];
Angie Chiang488f9212017-05-30 12:46:26 -07001761 return cost;
1762 }
Jingning Handfd72322017-08-09 14:04:12 -07001763 cost = txb_costs->txb_skip_cost[txb_skip_ctx][0];
Angie Chiang488f9212017-05-30 12:46:26 -07001764 for (int c = 0; c < txb_info->eob; ++c) {
1765 tran_low_t qc = txb_info->qcoeff[scan[c]];
Ola Hugosson13892102017-11-06 08:01:44 +01001766#if CONFIG_LV_MAP_MULTI
1767 int coeff_cost =
1768 get_coeff_cost(qc, c, c == txb_info->eob - 1, txb_info, txb_costs);
1769#else
Jingning Handfd72322017-08-09 14:04:12 -07001770 int coeff_cost = get_coeff_cost(qc, c, txb_info, txb_costs);
Ola Hugosson13892102017-11-06 08:01:44 +01001771#endif
Angie Chiang488f9212017-05-30 12:46:26 -07001772 cost += coeff_cost;
1773 }
1774 return cost;
1775}
1776
Angie Chiang5e012fe2017-05-30 18:47:39 -07001777#if TEST_OPTIMIZE_TXB
Angie Chiang0b2795c2017-09-29 16:00:08 -07001778void test_try_change_eob(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang5e012fe2017-05-30 18:47:39 -07001779 TxbCache *txb_cache) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001780 const int eob = txb_info->eob;
1781 const int16_t *const scan = txb_info->scan_order->scan;
Angie Chiang5e012fe2017-05-30 18:47:39 -07001782 if (eob > 0) {
1783 int last_si = eob - 1;
1784 int last_ci = scan[last_si];
1785 int last_coeff = txb_info->qcoeff[last_ci];
1786 if (abs(last_coeff) == 1) {
1787 int new_eob;
1788 int cost_diff =
Angie Chiang0b2795c2017-09-29 16:00:08 -07001789 try_change_eob(&new_eob, last_ci, txb_cache, txb_costs, txb_info, 0);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001790 int org_eob = txb_info->eob;
Angie Chiang0b2795c2017-09-29 16:00:08 -07001791 int cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001792
Linfeng Zhang1015a342017-10-24 16:20:41 -07001793 update_qcoeff(last_ci, get_lower_coeff(last_coeff), txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001794 set_eob(txb_info, new_eob);
Angie Chiang0b2795c2017-09-29 16:00:08 -07001795 int new_cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001796 set_eob(txb_info, org_eob);
Linfeng Zhang1015a342017-10-24 16:20:41 -07001797 update_qcoeff(last_ci, last_coeff, txb_info);
Angie Chiang5e012fe2017-05-30 18:47:39 -07001798
1799 int ref_cost_diff = -cost + new_cost;
1800 if (cost_diff != ref_cost_diff)
1801 printf("org_eob %d new_eob %d cost_diff %d ref_cost_diff %d\n", org_eob,
1802 new_eob, cost_diff, ref_cost_diff);
1803 }
1804 }
1805}
1806#endif
1807
Angie Chiang47e07072017-05-30 17:27:01 -07001808void try_level_down_facade(LevelDownStats *stats, int scan_idx,
Jingning Handfd72322017-08-09 14:04:12 -07001809 const TxbCache *txb_cache,
1810 const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang25645b72017-09-24 14:28:49 -07001811 TxbInfo *txb_info, int fast_mode) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001812 const int16_t *const scan = txb_info->scan_order->scan;
Angie Chiang47e07072017-05-30 17:27:01 -07001813 const int coeff_idx = scan[scan_idx];
1814 const tran_low_t qc = txb_info->qcoeff[coeff_idx];
1815 stats->new_eob = -1;
1816 stats->update = 0;
1817 if (qc == 0) {
1818 return;
1819 }
1820
1821 const tran_low_t tqc = txb_info->tcoeff[coeff_idx];
1822 const int dqv = txb_info->dequant[coeff_idx != 0];
1823
1824 const tran_low_t dqc = qcoeff_to_dqcoeff(qc, dqv, txb_info->shift);
Jingning Han641c1e52017-10-13 09:42:49 -07001825
1826 if (scan_idx != txb_info->eob - 1)
1827 if (abs(dqc) < abs(tqc)) return;
1828
Angie Chiang47e07072017-05-30 17:27:01 -07001829 const int64_t dqc_dist = get_coeff_dist(tqc, dqc, txb_info->shift);
1830
1831 stats->low_qc = get_lower_coeff(qc);
1832 stats->low_dqc = qcoeff_to_dqcoeff(stats->low_qc, dqv, txb_info->shift);
1833 const int64_t low_dqc_dist =
1834 get_coeff_dist(tqc, stats->low_dqc, txb_info->shift);
1835
1836 stats->dist_diff = -dqc_dist + low_dqc_dist;
1837 stats->cost_diff = 0;
1838 stats->new_eob = txb_info->eob;
1839 if (scan_idx == txb_info->eob - 1 && abs(qc) == 1) {
1840 stats->cost_diff = try_change_eob(&stats->new_eob, coeff_idx, txb_cache,
Angie Chiang25645b72017-09-24 14:28:49 -07001841 txb_costs, txb_info, fast_mode);
Angie Chiang47e07072017-05-30 17:27:01 -07001842 } else {
Angie Chiang25645b72017-09-24 14:28:49 -07001843 stats->cost_diff = try_level_down(coeff_idx, txb_cache, txb_costs, txb_info,
1844 NULL, fast_mode);
Angie Chiang47e07072017-05-30 17:27:01 -07001845#if TEST_OPTIMIZE_TXB
Angie Chiang0b2795c2017-09-29 16:00:08 -07001846 test_level_down(coeff_idx, txb_cache, txb_costs, txb_info);
Angie Chiang47e07072017-05-30 17:27:01 -07001847#endif
1848 }
Urvang Joshi70006e42017-06-14 16:08:55 -07001849 stats->rd_diff = RDCOST(txb_info->rdmult, stats->cost_diff, stats->dist_diff);
Angie Chiang47e07072017-05-30 17:27:01 -07001850 if (stats->rd_diff < 0) stats->update = 1;
1851 return;
1852}
Angie Chiang07c57f32017-05-30 18:18:33 -07001853
Jingning Han3422ac12017-10-25 20:37:53 -07001854#if 1
Dake Hea47cd6c2017-10-13 18:09:58 -07001855static int optimize_txb(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs,
1856 TxbCache *txb_cache, int dry_run, int fast_mode) {
1857 (void)fast_mode;
1858 (void)txb_cache;
1859 int update = 0;
Dake He59881772017-11-24 07:00:02 -08001860 // return update; // TODO(DKHE): training only.
Dake Hea47cd6c2017-10-13 18:09:58 -07001861 if (txb_info->eob == 0) return update;
Urvang Joshi80893152017-10-27 11:51:14 -07001862 const int max_eob = av1_get_max_eob(txb_info->tx_size);
Dake Hea47cd6c2017-10-13 18:09:58 -07001863
1864#if TEST_OPTIMIZE_TXB
1865 int64_t sse;
1866 int64_t org_dist =
1867 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
1868 (1 << (2 * txb_info->shift));
1869 int org_cost = get_txb_cost(txb_info, txb_probs);
1870#endif
1871
1872 tran_low_t *org_qcoeff = txb_info->qcoeff;
1873 tran_low_t *org_dqcoeff = txb_info->dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07001874 uint8_t *const org_levels = txb_info->levels;
Dake Hea47cd6c2017-10-13 18:09:58 -07001875
1876 tran_low_t tmp_qcoeff[MAX_TX_SQUARE];
1877 tran_low_t tmp_dqcoeff[MAX_TX_SQUARE];
Linfeng Zhang679d81e2017-10-31 15:27:42 -07001878 uint8_t tmp_levels_buf[TX_PAD_2D];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001879 uint8_t *const tmp_levels = set_levels(tmp_levels_buf, txb_info->width);
Dake Hea47cd6c2017-10-13 18:09:58 -07001880 const int org_eob = txb_info->eob;
1881 if (dry_run) {
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001882 const int stride = txb_info->width + TX_PAD_HOR;
1883 const int levels_size =
1884
1885 (stride * (txb_info->height + TX_PAD_VER) + TX_PAD_END);
Dake Hea47cd6c2017-10-13 18:09:58 -07001886 memcpy(tmp_qcoeff, org_qcoeff, sizeof(org_qcoeff[0]) * max_eob);
1887 memcpy(tmp_dqcoeff, org_dqcoeff, sizeof(org_dqcoeff[0]) * max_eob);
Linfeng Zhang679d81e2017-10-31 15:27:42 -07001888 memcpy(tmp_levels, org_levels - TX_PAD_TOP * stride,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07001889 sizeof(org_levels[0]) * levels_size);
Dake Hea47cd6c2017-10-13 18:09:58 -07001890 txb_info->qcoeff = tmp_qcoeff;
1891 txb_info->dqcoeff = tmp_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07001892 txb_info->levels = tmp_levels;
Dake Hea47cd6c2017-10-13 18:09:58 -07001893 }
1894
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001895 const int16_t *const scan = txb_info->scan_order->scan;
Dake Hea47cd6c2017-10-13 18:09:58 -07001896
1897 // forward optimize the nz_map`
1898 const int init_eob = txb_info->eob;
1899 const int seg_eob = txb_info->seg_eob;
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08001900 const int eob_cost =
1901 get_eob_cost(init_eob, seg_eob, txb_costs, txb_info->tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -07001902
1903 // backward optimize the level-k map
1904 int64_t accu_rate = eob_cost;
1905 int64_t accu_dist = 0;
1906 int64_t prev_eob_rd_cost = INT64_MAX;
1907 int64_t cur_eob_rd_cost = 0;
1908
1909 for (int si = init_eob - 1; si >= 0; --si) {
1910 const int coeff_idx = scan[si];
1911 tran_low_t qc = txb_info->qcoeff[coeff_idx];
1912
1913 LevelDownStats stats;
Ola Hugosson13892102017-11-06 08:01:44 +01001914#if CONFIG_LV_MAP_MULTI
1915 get_dist_cost_stats(&stats, si, si == init_eob - 1, txb_costs, txb_info);
1916#else
Dake Hea47cd6c2017-10-13 18:09:58 -07001917 get_dist_cost_stats(&stats, si, txb_costs, txb_info);
Ola Hugosson13892102017-11-06 08:01:44 +01001918#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07001919
1920 if (qc == 0) {
1921 accu_rate += stats.rate;
1922 } else {
1923 // check if it is better to make this the last significant coefficient
Jingning Han35deaa72017-10-26 15:36:30 -07001924 int cur_eob_rate =
1925 get_eob_cost(si + 1, seg_eob, txb_costs, txb_info->tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -07001926 cur_eob_rd_cost = RDCOST(txb_info->rdmult, cur_eob_rate, 0);
1927 prev_eob_rd_cost =
1928 RDCOST(txb_info->rdmult, accu_rate + stats.nz_rate, accu_dist);
1929 if (cur_eob_rd_cost <= prev_eob_rd_cost) {
1930 update = 1;
1931 for (int j = si + 1; j < txb_info->eob; j++) {
1932 const int coeff_pos_j = scan[j];
1933 update_coeff(coeff_pos_j, 0, txb_info);
1934 }
1935 txb_info->eob = si + 1;
1936
Ola Hugosson13892102017-11-06 08:01:44 +01001937 // rerun cost calculation due to change of eob
Dake Hea47cd6c2017-10-13 18:09:58 -07001938 accu_rate = cur_eob_rate;
1939 accu_dist = 0;
Ola Hugosson13892102017-11-06 08:01:44 +01001940#if CONFIG_LV_MAP_MULTI
1941 get_dist_cost_stats(&stats, si, 1, txb_costs, txb_info);
1942#else
Dake Hea47cd6c2017-10-13 18:09:58 -07001943 get_dist_cost_stats(&stats, si, txb_costs, txb_info);
Ola Hugosson13892102017-11-06 08:01:44 +01001944#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07001945 }
1946
1947 int bUpdCoeff = 0;
1948 if (stats.rd_low < stats.rd) {
1949 if ((stats.low_qc != 0) || (si < txb_info->eob - 1)) {
1950 bUpdCoeff = 1;
1951 update = 1;
1952 }
1953 }
1954
1955 if (bUpdCoeff) {
1956 update_coeff(coeff_idx, stats.low_qc, txb_info);
1957 accu_rate += stats.rate_low;
1958 accu_dist += stats.dist_low;
1959 } else {
1960 accu_rate += stats.rate;
1961 accu_dist += stats.dist;
1962 }
1963 }
1964 } // for (si)
1965 int non_zero_blk_rate =
1966 txb_costs->txb_skip_cost[txb_info->txb_ctx->txb_skip_ctx][0];
1967 prev_eob_rd_cost =
1968 RDCOST(txb_info->rdmult, accu_rate + non_zero_blk_rate, accu_dist);
1969
1970 int zero_blk_rate =
1971 txb_costs->txb_skip_cost[txb_info->txb_ctx->txb_skip_ctx][1];
1972 int64_t zero_blk_rd_cost = RDCOST(txb_info->rdmult, zero_blk_rate, 0);
1973 if (zero_blk_rd_cost <= prev_eob_rd_cost) {
1974 update = 1;
1975 for (int j = 0; j < txb_info->eob; j++) {
1976 const int coeff_pos_j = scan[j];
1977 update_coeff(coeff_pos_j, 0, txb_info);
1978 }
1979 txb_info->eob = 0;
1980 }
1981
1982#if TEST_OPTIMIZE_TXB
1983 int cost_diff = 0;
1984 int64_t dist_diff = 0;
1985 int64_t rd_diff = 0;
1986 int64_t new_dist =
1987 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
1988 (1 << (2 * txb_info->shift));
1989 int new_cost = get_txb_cost(txb_info, txb_probs);
1990 int64_t ref_dist_diff = new_dist - org_dist;
1991 int ref_cost_diff = new_cost - org_cost;
1992 if (cost_diff != ref_cost_diff || dist_diff != ref_dist_diff)
1993 printf(
1994 "overall rd_diff %ld\ncost_diff %d ref_cost_diff%d\ndist_diff %ld "
1995 "ref_dist_diff %ld\neob %d new_eob %d\n\n",
1996 rd_diff, cost_diff, ref_cost_diff, dist_diff, ref_dist_diff, org_eob,
1997 txb_info->eob);
1998#endif
1999 if (dry_run) {
2000 txb_info->qcoeff = org_qcoeff;
2001 txb_info->dqcoeff = org_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002002 txb_info->levels = org_levels;
Dake Hea47cd6c2017-10-13 18:09:58 -07002003 set_eob(txb_info, org_eob);
2004 }
2005 return update;
2006}
2007
2008#else
Jingning Handfd72322017-08-09 14:04:12 -07002009static int optimize_txb(TxbInfo *txb_info, const LV_MAP_COEFF_COST *txb_costs,
Angie Chiang25645b72017-09-24 14:28:49 -07002010 TxbCache *txb_cache, int dry_run, int fast_mode) {
Angie Chiang07c57f32017-05-30 18:18:33 -07002011 int update = 0;
2012 if (txb_info->eob == 0) return update;
2013 int cost_diff = 0;
2014 int64_t dist_diff = 0;
2015 int64_t rd_diff = 0;
Urvang Joshi80893152017-10-27 11:51:14 -07002016 const int max_eob = av1_get_max_eob(txb_info->tx_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002017
2018#if TEST_OPTIMIZE_TXB
2019 int64_t sse;
2020 int64_t org_dist =
2021 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
2022 (1 << (2 * txb_info->shift));
Angie Chiang0b2795c2017-09-29 16:00:08 -07002023 int org_cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang07c57f32017-05-30 18:18:33 -07002024#endif
2025
2026 tran_low_t *org_qcoeff = txb_info->qcoeff;
2027 tran_low_t *org_dqcoeff = txb_info->dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002028 uint8_t *const org_levels = txb_info->levels;
Angie Chiang07c57f32017-05-30 18:18:33 -07002029
2030 tran_low_t tmp_qcoeff[MAX_TX_SQUARE];
2031 tran_low_t tmp_dqcoeff[MAX_TX_SQUARE];
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002032 uint8_t tmp_levels_buf[TX_PAD_2D];
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002033 uint8_t *const tmp_levels = set_levels(tmp_levels_buf, txb_info->width);
Angie Chiang07c57f32017-05-30 18:18:33 -07002034 const int org_eob = txb_info->eob;
2035 if (dry_run) {
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002036 const int stride = txb_info->width + TX_PAD_HOR;
2037 const int levels_size =
2038
2039 (stride * (txb_info->height + TX_PAD_VER) + TX_PAD_END);
Angie Chiang07c57f32017-05-30 18:18:33 -07002040 memcpy(tmp_qcoeff, org_qcoeff, sizeof(org_qcoeff[0]) * max_eob);
2041 memcpy(tmp_dqcoeff, org_dqcoeff, sizeof(org_dqcoeff[0]) * max_eob);
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002042 memcpy(tmp_levels, org_levels - TX_PAD_TOP * stride,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002043 sizeof(org_levels[0]) * levels_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002044 txb_info->qcoeff = tmp_qcoeff;
2045 txb_info->dqcoeff = tmp_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002046 txb_info->levels = tmp_levels;
Angie Chiang07c57f32017-05-30 18:18:33 -07002047 }
2048
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002049 const int16_t *const scan = txb_info->scan_order->scan;
Angie Chiang07c57f32017-05-30 18:18:33 -07002050
2051 // forward optimize the nz_map
2052 const int cur_eob = txb_info->eob;
2053 for (int si = 0; si < cur_eob; ++si) {
2054 const int coeff_idx = scan[si];
2055 tran_low_t qc = txb_info->qcoeff[coeff_idx];
2056 if (abs(qc) == 1) {
2057 LevelDownStats stats;
Angie Chiang25645b72017-09-24 14:28:49 -07002058 try_level_down_facade(&stats, si, txb_cache, txb_costs, txb_info,
2059 fast_mode);
Angie Chiang07c57f32017-05-30 18:18:33 -07002060 if (stats.update) {
2061 update = 1;
2062 cost_diff += stats.cost_diff;
2063 dist_diff += stats.dist_diff;
2064 rd_diff += stats.rd_diff;
2065 update_level_down(coeff_idx, txb_cache, txb_info);
2066 set_eob(txb_info, stats.new_eob);
2067 }
2068 }
2069 }
2070
2071 // backward optimize the level-k map
Angie Chiang530b3042017-08-17 15:08:58 -07002072 int eob_fix = 0;
Angie Chiang07c57f32017-05-30 18:18:33 -07002073 for (int si = txb_info->eob - 1; si >= 0; --si) {
Angie Chiang530b3042017-08-17 15:08:58 -07002074 const int coeff_idx = scan[si];
2075 if (eob_fix == 1 && txb_info->qcoeff[coeff_idx] == 1) {
2076 // when eob is fixed, there is not need to optimize again when
2077 // abs(qc) == 1
2078 continue;
2079 }
Angie Chiang07c57f32017-05-30 18:18:33 -07002080 LevelDownStats stats;
Angie Chiang25645b72017-09-24 14:28:49 -07002081 try_level_down_facade(&stats, si, txb_cache, txb_costs, txb_info,
2082 fast_mode);
Angie Chiang07c57f32017-05-30 18:18:33 -07002083 if (stats.update) {
2084#if TEST_OPTIMIZE_TXB
2085// printf("si %d low_qc %d cost_diff %d dist_diff %ld rd_diff %ld eob %d new_eob
2086// %d\n", si, stats.low_qc, stats.cost_diff, stats.dist_diff, stats.rd_diff,
2087// txb_info->eob, stats.new_eob);
2088#endif
2089 update = 1;
2090 cost_diff += stats.cost_diff;
2091 dist_diff += stats.dist_diff;
2092 rd_diff += stats.rd_diff;
2093 update_level_down(coeff_idx, txb_cache, txb_info);
2094 set_eob(txb_info, stats.new_eob);
2095 }
Angie Chiang530b3042017-08-17 15:08:58 -07002096 if (eob_fix == 0 && txb_info->qcoeff[coeff_idx] != 0) eob_fix = 1;
Angie Chiang07c57f32017-05-30 18:18:33 -07002097 if (si > txb_info->eob) si = txb_info->eob;
2098 }
2099#if TEST_OPTIMIZE_TXB
2100 int64_t new_dist =
2101 av1_block_error_c(txb_info->tcoeff, txb_info->dqcoeff, max_eob, &sse) *
2102 (1 << (2 * txb_info->shift));
Angie Chiang0b2795c2017-09-29 16:00:08 -07002103 int new_cost = get_txb_cost(txb_info, txb_costs);
Angie Chiang07c57f32017-05-30 18:18:33 -07002104 int64_t ref_dist_diff = new_dist - org_dist;
2105 int ref_cost_diff = new_cost - org_cost;
2106 if (cost_diff != ref_cost_diff || dist_diff != ref_dist_diff)
2107 printf(
2108 "overall rd_diff %ld\ncost_diff %d ref_cost_diff%d\ndist_diff %ld "
2109 "ref_dist_diff %ld\neob %d new_eob %d\n\n",
2110 rd_diff, cost_diff, ref_cost_diff, dist_diff, ref_dist_diff, org_eob,
2111 txb_info->eob);
2112#endif
2113 if (dry_run) {
2114 txb_info->qcoeff = org_qcoeff;
2115 txb_info->dqcoeff = org_dqcoeff;
Linfeng Zhang1015a342017-10-24 16:20:41 -07002116 txb_info->levels = org_levels;
Angie Chiang07c57f32017-05-30 18:18:33 -07002117 set_eob(txb_info, org_eob);
2118 }
2119 return update;
2120}
Dake Hea47cd6c2017-10-13 18:09:58 -07002121#endif
Angie Chiang07c57f32017-05-30 18:18:33 -07002122
2123// These numbers are empirically obtained.
2124static const int plane_rd_mult[REF_TYPES][PLANE_TYPES] = {
Angie Chiang07c57f32017-05-30 18:18:33 -07002125 { 17, 13 }, { 16, 10 },
Angie Chiang07c57f32017-05-30 18:18:33 -07002126};
2127
Jingning Han7eab9ff2017-07-06 10:12:54 -07002128int av1_optimize_txb(const AV1_COMMON *cm, MACROBLOCK *x, int plane,
2129 int blk_row, int blk_col, int block, TX_SIZE tx_size,
Angie Chiang25645b72017-09-24 14:28:49 -07002130 TXB_CTX *txb_ctx, int fast_mode) {
Angie Chiang07c57f32017-05-30 18:18:33 -07002131 MACROBLOCKD *const xd = &x->e_mbd;
2132 const PLANE_TYPE plane_type = get_plane_type(plane);
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -08002133 const TX_SIZE txs_ctx = get_txsize_entropy_ctx(tx_size);
Jingning Han19b5c8f2017-07-06 15:10:12 -07002134 const TX_TYPE tx_type =
2135 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002136 const MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
2137 const struct macroblock_plane *p = &x->plane[plane];
2138 struct macroblockd_plane *pd = &xd->plane[plane];
2139 const int eob = p->eobs[block];
2140 tran_low_t *qcoeff = BLOCK_OFFSET(p->qcoeff, block);
2141 tran_low_t *dqcoeff = BLOCK_OFFSET(pd->dqcoeff, block);
2142 const tran_low_t *tcoeff = BLOCK_OFFSET(p->coeff, block);
Monty Montgomery125c0fc2017-10-26 00:44:35 -04002143 const int16_t *dequant = p->dequant_QTX;
Urvang Joshi80893152017-10-27 11:51:14 -07002144 const int seg_eob = av1_get_max_eob(tx_size);
Angie Chianga9ba58e2017-12-01 19:22:43 -08002145 const int bwl = get_txb_bwl(tx_size);
2146 const int width = get_txb_wide(tx_size);
2147 const int height = get_txb_high(tx_size);
Angie Chiang07c57f32017-05-30 18:18:33 -07002148 const int is_inter = is_inter_block(mbmi);
Angie Chiangbd99b382017-06-20 15:11:16 -07002149 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Jingning Handfd72322017-08-09 14:04:12 -07002150 const LV_MAP_COEFF_COST txb_costs = x->coeff_costs[txs_ctx][plane_type];
Angie Chiang07c57f32017-05-30 18:18:33 -07002151
2152 const int shift = av1_get_tx_scale(tx_size);
2153 const int64_t rdmult =
Jingning Hanb433f4c2017-11-17 15:43:59 -08002154 ((x->rdmult * plane_rd_mult[is_inter][plane_type] << (2 * (xd->bd - 8))) +
2155 2) >>
2156 2;
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002157 uint8_t levels_buf[TX_PAD_2D];
2158 uint8_t *const levels = set_levels(levels_buf, width);
Angie Chiang07c57f32017-05-30 18:18:33 -07002159
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002160 assert(width == (1 << bwl));
Linfeng Zhang1015a342017-10-24 16:20:41 -07002161 TxbInfo txb_info = {
2162 qcoeff, levels, dqcoeff, tcoeff, dequant, shift,
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002163 tx_size, txs_ctx, tx_type, bwl, width, height,
Linfeng Zhang1015a342017-10-24 16:20:41 -07002164 eob, seg_eob, scan_order, txb_ctx, rdmult, &cm->coeff_ctx_table
2165 };
2166
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002167 av1_txb_init_levels(qcoeff, width, height, levels);
Urvang Joshi70006e42017-06-14 16:08:55 -07002168
Angie Chiang93af45f2017-10-23 18:19:59 -07002169 const int update = optimize_txb(&txb_info, &txb_costs, NULL, 0, fast_mode);
Angie Chiang07c57f32017-05-30 18:18:33 -07002170
Angie Chiang07c57f32017-05-30 18:18:33 -07002171 if (update) p->eobs[block] = txb_info.eob;
2172 return txb_info.eob;
2173}
Angie Chiang74e23072017-03-24 14:54:23 -07002174int av1_get_txb_entropy_context(const tran_low_t *qcoeff,
2175 const SCAN_ORDER *scan_order, int eob) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002176 const int16_t *const scan = scan_order->scan;
Angie Chiang74e23072017-03-24 14:54:23 -07002177 int cul_level = 0;
2178 int c;
Jingning Han339cf932017-09-18 10:17:02 -07002179
2180 if (eob == 0) return 0;
Angie Chiang74e23072017-03-24 14:54:23 -07002181 for (c = 0; c < eob; ++c) {
2182 cul_level += abs(qcoeff[scan[c]]);
2183 }
2184
2185 cul_level = AOMMIN(COEFF_CONTEXT_MASK, cul_level);
2186 set_dc_sign(&cul_level, qcoeff[0]);
2187
2188 return cul_level;
2189}
2190
Jingning Han4fe5f672017-05-19 15:46:07 -07002191void av1_update_txb_context_b(int plane, int block, int blk_row, int blk_col,
2192 BLOCK_SIZE plane_bsize, TX_SIZE tx_size,
2193 void *arg) {
Jingning Han6171ae72017-05-18 20:15:06 -07002194 struct tokenize_b_args *const args = arg;
Angie Chiang36d616b2017-03-22 13:58:36 -07002195 const AV1_COMP *cpi = args->cpi;
2196 const AV1_COMMON *cm = &cpi->common;
Angie Chiang0397eda2017-03-15 16:57:14 -07002197 ThreadData *const td = args->td;
2198 MACROBLOCK *const x = &td->mb;
2199 MACROBLOCKD *const xd = &x->e_mbd;
Angie Chiang36d616b2017-03-22 13:58:36 -07002200 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
Angie Chiang0397eda2017-03-15 16:57:14 -07002201 struct macroblock_plane *p = &x->plane[plane];
2202 struct macroblockd_plane *pd = &xd->plane[plane];
Angie Chiang36d616b2017-03-22 13:58:36 -07002203 const uint16_t eob = p->eobs[block];
2204 const tran_low_t *qcoeff = BLOCK_OFFSET(p->qcoeff, block);
2205 const PLANE_TYPE plane_type = pd->plane_type;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002206 const TX_TYPE tx_type =
2207 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiangbd99b382017-06-20 15:11:16 -07002208 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Angie Chiang0397eda2017-03-15 16:57:14 -07002209 (void)plane_bsize;
Angie Chiang36d616b2017-03-22 13:58:36 -07002210
Angie Chiang74e23072017-03-24 14:54:23 -07002211 int cul_level = av1_get_txb_entropy_context(qcoeff, scan_order, eob);
Angie Chiang36d616b2017-03-22 13:58:36 -07002212 av1_set_contexts(xd, pd, plane, tx_size, cul_level, blk_col, blk_row);
Angie Chiang0397eda2017-03-15 16:57:14 -07002213}
2214
Jingning Han4fe5f672017-05-19 15:46:07 -07002215void av1_update_and_record_txb_context(int plane, int block, int blk_row,
2216 int blk_col, BLOCK_SIZE plane_bsize,
2217 TX_SIZE tx_size, void *arg) {
Jingning Han6171ae72017-05-18 20:15:06 -07002218 struct tokenize_b_args *const args = arg;
Angie Chiang0397eda2017-03-15 16:57:14 -07002219 const AV1_COMP *cpi = args->cpi;
2220 const AV1_COMMON *cm = &cpi->common;
2221 ThreadData *const td = args->td;
2222 MACROBLOCK *const x = &td->mb;
2223 MACROBLOCKD *const xd = &x->e_mbd;
2224 struct macroblock_plane *p = &x->plane[plane];
2225 struct macroblockd_plane *pd = &xd->plane[plane];
2226 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
Linfeng Zhang848f7bc2017-10-31 15:26:07 -07002227 int eob = p->eobs[block], update_eob = -1;
Angie Chiang0397eda2017-03-15 16:57:14 -07002228 const PLANE_TYPE plane_type = pd->plane_type;
2229 const tran_low_t *qcoeff = BLOCK_OFFSET(p->qcoeff, block);
2230 tran_low_t *tcoeff = BLOCK_OFFSET(x->mbmi_ext->tcoeff[plane], block);
2231 const int segment_id = mbmi->segment_id;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002232 const TX_TYPE tx_type =
2233 av1_get_tx_type(plane_type, xd, blk_row, blk_col, block, tx_size);
Angie Chiangbd99b382017-06-20 15:11:16 -07002234 const SCAN_ORDER *const scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002235 const int16_t *const scan = scan_order->scan;
hui suc0cf71d2017-07-20 16:38:50 -07002236 const int seg_eob = av1_get_tx_eob(&cpi->common.seg, segment_id, tx_size);
Linfeng Zhang1015a342017-10-24 16:20:41 -07002237 int c;
Angie Chiang85901562017-03-17 12:03:27 -07002238 TXB_CTX txb_ctx;
2239 get_txb_ctx(plane_bsize, tx_size, plane, pd->above_context + blk_col,
2240 pd->left_context + blk_row, &txb_ctx);
Angie Chianga9ba58e2017-12-01 19:22:43 -08002241 const int bwl = get_txb_bwl(tx_size);
2242 const int width = get_txb_wide(tx_size);
2243 const int height = get_txb_high(tx_size);
Linfeng Zhang679d81e2017-10-31 15:27:42 -07002244 uint8_t levels_buf[TX_PAD_2D];
2245 uint8_t *const levels = set_levels(levels_buf, width);
Linfeng Zhangae7b2f32017-11-08 15:46:57 -08002246 DECLARE_ALIGNED(16, uint8_t, level_counts[MAX_TX_SQUARE]);
Yunqing Wang0e141b52017-11-02 15:08:58 -07002247 const uint8_t allow_update_cdf = args->allow_update_cdf;
Angie Chiang0397eda2017-03-15 16:57:14 -07002248
Debargha Mukherjeeb3eda2f2017-11-28 16:00:20 -08002249 TX_SIZE txsize_ctx = get_txsize_entropy_ctx(tx_size);
Jingning Han8f661602017-08-19 08:16:50 -07002250 FRAME_CONTEXT *ec_ctx = xd->tile_ctx;
Jingning Han48be0e12017-06-13 12:12:01 -07002251
Angie Chiang0397eda2017-03-15 16:57:14 -07002252 memcpy(tcoeff, qcoeff, sizeof(*tcoeff) * seg_eob);
2253
Jingning Han48be0e12017-06-13 12:12:01 -07002254 ++td->counts->txb_skip[txsize_ctx][txb_ctx.txb_skip_ctx][eob == 0];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002255 if (allow_update_cdf)
2256 update_bin(ec_ctx->txb_skip_cdf[txsize_ctx][txb_ctx.txb_skip_ctx], eob == 0,
2257 2);
Angie Chiang85901562017-03-17 12:03:27 -07002258 x->mbmi_ext->txb_skip_ctx[plane][block] = txb_ctx.txb_skip_ctx;
Angie Chiang0397eda2017-03-15 16:57:14 -07002259
2260 x->mbmi_ext->eobs[plane][block] = eob;
2261
2262 if (eob == 0) {
2263 av1_set_contexts(xd, pd, plane, tx_size, 0, blk_col, blk_row);
2264 return;
2265 }
2266
Linfeng Zhang1122d7d2017-10-31 15:30:28 -07002267 av1_txb_init_levels(tcoeff, width, height, levels);
Linfeng Zhangce065ca2017-10-17 16:49:30 -07002268
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002269#if CONFIG_TXK_SEL
Jingning Han7eab9ff2017-07-06 10:12:54 -07002270 av1_update_tx_type_count(cm, xd, blk_row, blk_col, block, plane,
Yunqing Wang0e141b52017-11-02 15:08:58 -07002271 mbmi->sb_type, get_min_tx_size(tx_size), td->counts,
2272 allow_update_cdf);
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002273#endif
Angie Chiang0397eda2017-03-15 16:57:14 -07002274
Dake Hea47cd6c2017-10-13 18:09:58 -07002275 unsigned int(*nz_map_count)[SIG_COEF_CONTEXTS][2] =
2276 &(td->counts->nz_map[txsize_ctx][plane_type]);
Jingning Han35deaa72017-10-26 15:36:30 -07002277 av1_update_eob_context(eob, seg_eob, tx_size, tx_type, plane_type, ec_ctx,
Yunqing Wang0e141b52017-11-02 15:08:58 -07002278 td->counts, allow_update_cdf);
Dake He03a32922017-10-31 08:06:45 -07002279#if USE_CAUSAL_BASE_CTX
2280 int coeff_ctx = 0;
2281 update_eob = eob - 1;
2282#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07002283 for (c = eob - 1; c >= 0; --c) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002284 const int pos = scan[c];
2285 const tran_low_t v = qcoeff[pos];
2286 const int is_nz = (v != 0);
Dake He03a32922017-10-31 08:06:45 -07002287
Ola Hugosson13892102017-11-06 08:01:44 +01002288#if CONFIG_LV_MAP_MULTI
2289 (void)is_nz;
2290 (void)nz_map_count;
2291 coeff_ctx =
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -08002292 get_nz_map_ctx(levels, pos, bwl, height, c, c == eob - 1, tx_type);
Dake He3fe369c2017-11-16 17:56:44 -08002293#if USE_BASE_EOB_ALPHABET
2294 if (c == eob - 1) {
2295 update_cdf(ec_ctx->coeff_base_eob_cdf[txsize_ctx][plane_type]
2296 [coeff_ctx - SIG_COEF_CONTEXTS +
2297 SIG_COEF_CONTEXTS_EOB],
2298 AOMMIN(abs(v), 3) - 1, 3);
2299 } else {
2300 update_cdf(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][coeff_ctx],
2301 AOMMIN(abs(v), 3), 4);
2302 }
Dake He59881772017-11-24 07:00:02 -08002303 {
2304 if (c < eob - 1) {
2305 ++(*nz_map_count)[coeff_ctx][is_nz];
2306 }
2307 if (is_nz) {
2308 for (int k = 0; k < NUM_BASE_LEVELS; ++k) {
2309 int is_k = (abs(v) > (k + 1));
2310 ++td->counts->coeff_base[txsize_ctx][plane_type][k][coeff_ctx][is_k];
2311 if (is_k == 0) break;
2312 }
2313 }
2314 }
2315
Dake He3fe369c2017-11-16 17:56:44 -08002316#else
Ola Hugosson13892102017-11-06 08:01:44 +01002317 update_cdf(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][coeff_ctx],
2318 AOMMIN(abs(v), 3), 4);
Dake He3fe369c2017-11-16 17:56:44 -08002319#endif
Ola Hugosson13892102017-11-06 08:01:44 +01002320#elif USE_CAUSAL_BASE_CTX
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -08002321 coeff_ctx = get_nz_map_ctx(levels, pos, bwl, height, tx_type);
Ola Hugosson13892102017-11-06 08:01:44 +01002322
Dake He03a32922017-10-31 08:06:45 -07002323 if (c < eob - 1) {
2324 ++(*nz_map_count)[coeff_ctx][is_nz];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002325 if (allow_update_cdf)
2326 update_cdf(ec_ctx->nz_map_cdf[txsize_ctx][plane_type][coeff_ctx], is_nz,
2327 2);
Dake He03a32922017-10-31 08:06:45 -07002328 }
2329
2330 if (is_nz) {
2331 int k;
2332 for (k = 0; k < NUM_BASE_LEVELS; ++k) {
2333 int ctx = coeff_ctx;
2334 int is_k = (abs(v) > (k + 1));
2335
2336 ++td->counts->coeff_base[txsize_ctx][plane_type][k][ctx][is_k];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002337 if (allow_update_cdf)
2338 update_bin(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][k][ctx],
2339 is_k, 2);
Dake He03a32922017-10-31 08:06:45 -07002340 if (is_k == 0) break;
2341 }
2342 }
2343#else
Linfeng Zhangf91f3ca2017-12-05 11:22:41 -08002344 const int coeff_ctx = get_nz_map_ctx(levels, pos, bwl, height, tx_type);
Dake Hea47cd6c2017-10-13 18:09:58 -07002345
2346 if (c == eob - 1) continue;
2347
2348 ++(*nz_map_count)[coeff_ctx][is_nz];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002349 if (allow_update_cdf)
2350 update_cdf(ec_ctx->nz_map_cdf[txsize_ctx][plane_type][coeff_ctx], is_nz,
2351 2);
Dake He03a32922017-10-31 08:06:45 -07002352#endif
Dake Hea47cd6c2017-10-13 18:09:58 -07002353 }
Dake Hea47cd6c2017-10-13 18:09:58 -07002354
Dake He03a32922017-10-31 08:06:45 -07002355#if !USE_CAUSAL_BASE_CTX
Angie Chiang0397eda2017-03-15 16:57:14 -07002356 // Reverse process order to handle coefficient level and sign.
Linfeng Zhang1015a342017-10-24 16:20:41 -07002357 for (int i = 0; i < NUM_BASE_LEVELS; ++i) {
Linfeng Zhangf72e1382017-11-01 13:59:41 -07002358 av1_get_base_level_counts(levels, i, width, height, level_counts);
Angie Chiang0397eda2017-03-15 16:57:14 -07002359 for (c = eob - 1; c >= 0; --c) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002360 const int pos = scan[c];
2361 const tran_low_t level = abs(tcoeff[pos]);
Angie Chiang0397eda2017-03-15 16:57:14 -07002362 int ctx;
2363
2364 if (level <= i) continue;
2365
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002366 ctx = get_base_ctx(levels, pos, bwl, i, level_counts[pos]);
Angie Chiang0397eda2017-03-15 16:57:14 -07002367
2368 if (level == i + 1) {
Jingning Han48be0e12017-06-13 12:12:01 -07002369 ++td->counts->coeff_base[txsize_ctx][plane_type][i][ctx][1];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002370 if (allow_update_cdf)
2371 update_bin(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][i][ctx], 1,
2372 2);
Angie Chiang0397eda2017-03-15 16:57:14 -07002373 continue;
2374 }
Jingning Han48be0e12017-06-13 12:12:01 -07002375 ++td->counts->coeff_base[txsize_ctx][plane_type][i][ctx][0];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002376 if (allow_update_cdf)
2377 update_bin(ec_ctx->coeff_base_cdf[txsize_ctx][plane_type][i][ctx], 0,
2378 2);
Angie Chiang0397eda2017-03-15 16:57:14 -07002379 update_eob = AOMMAX(update_eob, c);
2380 }
2381 }
Dake He03a32922017-10-31 08:06:45 -07002382#endif
Angie Chiang0397eda2017-03-15 16:57:14 -07002383
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002384 // Update the context needed to code the DC sign (if applicable)
2385 const int sign = (tcoeff[0] < 0) ? 1 : 0;
2386 if (tcoeff[0] != 0) {
Dake He43edb762017-10-26 10:29:46 -07002387 int dc_sign_ctx = txb_ctx.dc_sign_ctx;
2388
2389 ++td->counts->dc_sign[plane_type][dc_sign_ctx][sign];
2390#if LV_MAP_PROB
Yunqing Wang0e141b52017-11-02 15:08:58 -07002391 if (allow_update_cdf)
2392 update_bin(ec_ctx->dc_sign_cdf[plane_type][dc_sign_ctx], sign, 2);
Dake He43edb762017-10-26 10:29:46 -07002393#endif
2394 x->mbmi_ext->dc_sign_ctx[plane][block] = dc_sign_ctx;
2395 }
2396
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002397 if (update_eob >= 0) {
Dake He7d01ab52017-11-24 17:53:28 -08002398#if !CONFIG_LV_MAP_MULTI
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002399 av1_get_br_level_counts(levels, width, height, level_counts);
Dake He7d01ab52017-11-24 17:53:28 -08002400#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002401 for (c = update_eob; c >= 0; --c) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002402 const int pos = scan[c];
2403 const tran_low_t level = abs(tcoeff[pos]);
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002404 int idx;
2405 int ctx;
Angie Chiang0397eda2017-03-15 16:57:14 -07002406
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002407 if (level <= NUM_BASE_LEVELS) continue;
Angie Chiang0397eda2017-03-15 16:57:14 -07002408
Dake He7d01ab52017-11-24 17:53:28 -08002409// level is above 1.
2410#if !CONFIG_LV_MAP_MULTI
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002411 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos]);
Dake He7d01ab52017-11-24 17:53:28 -08002412#endif
Jingning Han87b01b52017-08-31 12:07:20 -07002413
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002414 const int base_range = level - 1 - NUM_BASE_LEVELS;
Ola Hugossone72a2092017-11-12 09:11:53 +01002415#if CONFIG_LV_MAP_MULTI
Dake He7d01ab52017-11-24 17:53:28 -08002416#if USE_CAUSAL_BR_CTX
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002417 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos], tx_type);
Dake He7d01ab52017-11-24 17:53:28 -08002418#else
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002419 ctx = get_br_ctx(levels, pos, bwl, level_counts[pos]);
Dake He7d01ab52017-11-24 17:53:28 -08002420#endif
Ola Hugossone72a2092017-11-12 09:11:53 +01002421 for (idx = 0; idx < COEFF_BASE_RANGE; idx += BR_CDF_SIZE - 1) {
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002422 const int k = AOMMIN(base_range - idx, BR_CDF_SIZE - 1);
Ola Hugossone72a2092017-11-12 09:11:53 +01002423 // printf("br_update: %d %d %2d : %2d %d\n", txsize_ctx, plane, ctx,
2424 // base_range, k);
Dake He7d01ab52017-11-24 17:53:28 -08002425 update_cdf(
2426 ec_ctx->coeff_br_cdf[AOMMIN(txsize_ctx, TX_16X16)][plane_type][ctx],
2427 k, BR_CDF_SIZE);
2428 for (int lps = 0; lps < BR_CDF_SIZE - 1; lps++) {
2429 ++td->counts->coeff_lps[AOMMIN(txsize_ctx, TX_16X16)][plane_type][lps]
2430 [ctx][lps == k];
Dake He59881772017-11-24 07:00:02 -08002431 if (lps == k) break;
2432 }
2433
Ola Hugossone72a2092017-11-12 09:11:53 +01002434 if (k < BR_CDF_SIZE - 1) break;
2435 }
2436#else
Linfeng Zhangdb41d1e2017-12-05 11:06:20 -08002437 const int br_set_idx = base_range < COEFF_BASE_RANGE
2438 ? coeff_to_br_index[base_range]
2439 : BASE_RANGE_SETS;
Jingning Han87b01b52017-08-31 12:07:20 -07002440
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002441 for (idx = 0; idx < BASE_RANGE_SETS; ++idx) {
2442 if (idx == br_set_idx) {
2443 int br_base = br_index_to_coeff[br_set_idx];
2444 int br_offset = base_range - br_base;
2445 ++td->counts->coeff_br[txsize_ctx][plane_type][idx][ctx][1];
Yunqing Wang0e141b52017-11-02 15:08:58 -07002446 if (allow_update_cdf)
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002447 update_bin(ec_ctx->coeff_br_cdf[txsize_ctx][plane_type][idx][ctx],
2448 1, 2);
2449 int extra_bits = (1 << br_extra_bits[idx]) - 1;
2450 for (int tok = 0; tok < extra_bits; ++tok) {
2451 if (br_offset == tok) {
2452 ++td->counts->coeff_lps[txsize_ctx][plane_type][ctx][1];
2453 if (allow_update_cdf)
2454 update_bin(ec_ctx->coeff_lps_cdf[txsize_ctx][plane_type][ctx],
2455 1, 2);
2456 break;
2457 }
2458 ++td->counts->coeff_lps[txsize_ctx][plane_type][ctx][0];
2459 if (allow_update_cdf)
2460 update_bin(ec_ctx->coeff_lps_cdf[txsize_ctx][plane_type][ctx], 0,
2461 2);
2462 }
2463 break;
Jingning Han87b01b52017-08-31 12:07:20 -07002464 }
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002465 ++td->counts->coeff_br[txsize_ctx][plane_type][idx][ctx][0];
2466 if (allow_update_cdf)
2467 update_bin(ec_ctx->coeff_br_cdf[txsize_ctx][plane_type][idx][ctx], 0,
2468 2);
Jingning Han87b01b52017-08-31 12:07:20 -07002469 }
Ola Hugossone72a2092017-11-12 09:11:53 +01002470#endif
Linfeng Zhang97fc4742017-11-07 12:57:25 -08002471 // use 0-th order Golomb code to handle the residual level.
Jingning Han87b01b52017-08-31 12:07:20 -07002472 }
Angie Chiang0397eda2017-03-15 16:57:14 -07002473 }
Angie Chiang36d616b2017-03-22 13:58:36 -07002474
Angie Chiang63d190a2017-10-23 15:43:05 -07002475 int cul_level = av1_get_txb_entropy_context(tcoeff, scan_order, eob);
Angie Chiang0397eda2017-03-15 16:57:14 -07002476 av1_set_contexts(xd, pd, plane, tx_size, cul_level, blk_col, blk_row);
Angie Chiang0b205e62017-03-20 17:16:47 -07002477
2478#if CONFIG_ADAPT_SCAN
2479 // Since dqcoeff is not available here, we pass qcoeff into
2480 // av1_update_scan_count_facade(). The update behavior should be the same
2481 // because av1_update_scan_count_facade() only cares if coefficients are zero
2482 // or not.
Jingning Han025c6c42017-11-22 12:06:03 -08002483 const int mi_row = -xd->mb_to_top_edge >> (3 + MI_SIZE_LOG2);
Jingning Han0bd3bf62017-11-28 17:11:51 -08002484 av1_update_scan_count_facade((AV1_COMMON *)cm, xd, mi_row, tx_size, tx_type,
2485 qcoeff, eob);
Angie Chiang0b205e62017-03-20 17:16:47 -07002486#endif
Angie Chiang0397eda2017-03-15 16:57:14 -07002487}
2488
2489void av1_update_txb_context(const AV1_COMP *cpi, ThreadData *td,
2490 RUN_TYPE dry_run, BLOCK_SIZE bsize, int *rate,
Yunqing Wang0e141b52017-11-02 15:08:58 -07002491 int mi_row, int mi_col, uint8_t allow_update_cdf) {
Angie Chiang0397eda2017-03-15 16:57:14 -07002492 MACROBLOCK *const x = &td->mb;
2493 MACROBLOCKD *const xd = &x->e_mbd;
2494 MB_MODE_INFO *const mbmi = &xd->mi[0]->mbmi;
Yunqing Wang0e141b52017-11-02 15:08:58 -07002495 struct tokenize_b_args arg = { cpi, td, NULL, 0, allow_update_cdf };
Angie Chiang0397eda2017-03-15 16:57:14 -07002496 (void)rate;
2497 (void)mi_row;
2498 (void)mi_col;
2499 if (mbmi->skip) {
Timothy B. Terriberrya2d5cde2017-05-10 18:33:50 -07002500 av1_reset_skip_context(xd, mi_row, mi_col, bsize);
Angie Chiang0397eda2017-03-15 16:57:14 -07002501 return;
2502 }
2503
2504 if (!dry_run) {
Jingning Han94652b82017-04-04 09:45:02 -07002505 av1_foreach_transformed_block(xd, bsize, mi_row, mi_col,
Jingning Han4fe5f672017-05-19 15:46:07 -07002506 av1_update_and_record_txb_context, &arg);
Angie Chiangc8af6112017-03-16 16:11:22 -07002507 } else if (dry_run == DRY_RUN_NORMAL) {
Jingning Han4fe5f672017-05-19 15:46:07 -07002508 av1_foreach_transformed_block(xd, bsize, mi_row, mi_col,
2509 av1_update_txb_context_b, &arg);
Angie Chiangc8af6112017-03-16 16:11:22 -07002510 } else {
2511 printf("DRY_RUN_COSTCOEFFS is not supported yet\n");
2512 assert(0);
Angie Chiang0397eda2017-03-15 16:57:14 -07002513 }
2514}
Angie Chiang800df032017-03-22 11:14:12 -07002515
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002516#if CONFIG_TXK_SEL
Angie Chiang808d8592017-04-06 18:36:55 -07002517int64_t av1_search_txk_type(const AV1_COMP *cpi, MACROBLOCK *x, int plane,
2518 int block, int blk_row, int blk_col,
2519 BLOCK_SIZE plane_bsize, TX_SIZE tx_size,
Angie Chiang65a39bb2017-04-11 16:50:04 -07002520 const ENTROPY_CONTEXT *a, const ENTROPY_CONTEXT *l,
2521 int use_fast_coef_costing, RD_STATS *rd_stats) {
Angie Chiang808d8592017-04-06 18:36:55 -07002522 const AV1_COMMON *cm = &cpi->common;
2523 MACROBLOCKD *xd = &x->e_mbd;
2524 MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi;
2525 TX_TYPE txk_start = DCT_DCT;
2526 TX_TYPE txk_end = TX_TYPES - 1;
2527 TX_TYPE best_tx_type = txk_start;
2528 int64_t best_rd = INT64_MAX;
Jingning Han47558172017-07-05 16:33:19 -07002529 uint8_t best_eob = 0;
Jingning Hane3b81bc2017-06-23 11:43:52 -07002530 RD_STATS best_rd_stats;
Angie Chiang808d8592017-04-06 18:36:55 -07002531 TX_TYPE tx_type;
Jingning Hane3b81bc2017-06-23 11:43:52 -07002532
2533 av1_invalid_rd_stats(&best_rd_stats);
2534
Angie Chiang808d8592017-04-06 18:36:55 -07002535 for (tx_type = txk_start; tx_type <= txk_end; ++tx_type) {
Angie Chiangbce07f12017-12-01 16:34:31 -08002536 if (plane == 0)
2537 mbmi->txk_type[(blk_row << MAX_MIB_SIZE_LOG2) + blk_col] = tx_type;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002538 TX_TYPE ref_tx_type = av1_get_tx_type(get_plane_type(plane), xd, blk_row,
2539 blk_col, block, tx_size);
Angie Chiang00491e02017-04-11 17:55:10 -07002540 if (tx_type != ref_tx_type) {
hui su45b64752017-07-12 16:54:35 -07002541 // use av1_get_tx_type() to check if the tx_type is valid for the current
2542 // mode if it's not, we skip it here.
Angie Chiang00491e02017-04-11 17:55:10 -07002543 continue;
2544 }
Jingning Hane57d6322017-07-03 18:50:25 -07002545
Hui Suddbcde22017-09-18 17:22:02 -07002546 const int is_inter = is_inter_block(mbmi);
Angie Chiang53bf1e92017-11-29 16:53:07 -08002547 const TxSetType tx_set_type = get_ext_tx_set_type(
2548 tx_size, mbmi->sb_type, is_inter, cm->reduced_tx_set_used);
Hui Suddbcde22017-09-18 17:22:02 -07002549 if (!av1_ext_tx_used[tx_set_type][tx_type]) continue;
Jingning Hane57d6322017-07-03 18:50:25 -07002550
Angie Chiang808d8592017-04-06 18:36:55 -07002551 RD_STATS this_rd_stats;
2552 av1_invalid_rd_stats(&this_rd_stats);
Angie Chiangdaccae32017-12-04 09:34:44 -08002553#if DISABLE_TRELLISQ_SEARCH
2554 av1_xform_quant(cm, x, plane, block, blk_row, blk_col, plane_bsize, tx_size,
Sarah Parkere8d0d4c2017-12-06 15:11:37 -08002555 AV1_XFORM_QUANT_B);
Angie Chiangdaccae32017-12-04 09:34:44 -08002556#else
Angie Chiang808d8592017-04-06 18:36:55 -07002557 av1_xform_quant(cm, x, plane, block, blk_row, blk_col, plane_bsize, tx_size,
Sarah Parkere8d0d4c2017-12-06 15:11:37 -08002558 AV1_XFORM_QUANT_FP);
Jingning Han7eab9ff2017-07-06 10:12:54 -07002559 av1_optimize_b(cm, x, plane, blk_row, blk_col, block, plane_bsize, tx_size,
Angie Chiang25645b72017-09-24 14:28:49 -07002560 a, l, 1);
Angie Chiangdaccae32017-12-04 09:34:44 -08002561#endif
Angie Chiang808d8592017-04-06 18:36:55 -07002562 av1_dist_block(cpi, x, plane, plane_bsize, block, blk_row, blk_col, tx_size,
Angie Chiang2ed03a32017-04-16 18:00:06 -07002563 &this_rd_stats.dist, &this_rd_stats.sse,
2564 OUTPUT_HAS_PREDICTED_PIXELS);
Angie Chiangbd99b382017-06-20 15:11:16 -07002565 const SCAN_ORDER *scan_order = get_scan(cm, tx_size, tx_type, mbmi);
Jingning Han7eab9ff2017-07-06 10:12:54 -07002566 this_rd_stats.rate =
2567 av1_cost_coeffs(cpi, x, plane, blk_row, blk_col, block, tx_size,
2568 scan_order, a, l, use_fast_coef_costing);
Urvang Joshi70006e42017-06-14 16:08:55 -07002569 int rd = RDCOST(x->rdmult, this_rd_stats.rate, this_rd_stats.dist);
Jingning Han19b5c8f2017-07-06 15:10:12 -07002570
Angie Chiang808d8592017-04-06 18:36:55 -07002571 if (rd < best_rd) {
2572 best_rd = rd;
Jingning Hane3b81bc2017-06-23 11:43:52 -07002573 best_rd_stats = this_rd_stats;
Angie Chiang808d8592017-04-06 18:36:55 -07002574 best_tx_type = tx_type;
Jingning Han47558172017-07-05 16:33:19 -07002575 best_eob = x->plane[plane].txb_entropy_ctx[block];
Angie Chiang808d8592017-04-06 18:36:55 -07002576 }
2577 }
Jingning Hane3b81bc2017-06-23 11:43:52 -07002578
2579 av1_merge_rd_stats(rd_stats, &best_rd_stats);
2580
Angie Chianga3f7d2e2017-12-07 19:51:14 -08002581 if (best_eob == 0) best_tx_type = DCT_DCT;
Jingning Han19b5c8f2017-07-06 15:10:12 -07002582
Angie Chiangbce07f12017-12-01 16:34:31 -08002583 if (plane == 0)
2584 mbmi->txk_type[(blk_row << MAX_MIB_SIZE_LOG2) + blk_col] = best_tx_type;
Jingning Han47558172017-07-05 16:33:19 -07002585 x->plane[plane].txb_entropy_ctx[block] = best_eob;
2586
Angie Chiang2ed03a32017-04-16 18:00:06 -07002587 if (!is_inter_block(mbmi)) {
Angie Chiangdaccae32017-12-04 09:34:44 -08002588// intra mode needs decoded result such that the next transform block
2589// can use it for prediction.
2590#if DISABLE_TRELLISQ_SEARCH
2591 av1_xform_quant(cm, x, plane, block, blk_row, blk_col, plane_bsize, tx_size,
Sarah Parkere8d0d4c2017-12-06 15:11:37 -08002592 AV1_XFORM_QUANT_B);
Angie Chiangdaccae32017-12-04 09:34:44 -08002593#else
Jingning Han47558172017-07-05 16:33:19 -07002594 av1_xform_quant(cm, x, plane, block, blk_row, blk_col, plane_bsize, tx_size,
Sarah Parkere8d0d4c2017-12-06 15:11:37 -08002595 AV1_XFORM_QUANT_FP);
Jingning Han7eab9ff2017-07-06 10:12:54 -07002596 av1_optimize_b(cm, x, plane, blk_row, blk_col, block, plane_bsize, tx_size,
Angie Chiang25645b72017-09-24 14:28:49 -07002597 a, l, 1);
Angie Chiangdaccae32017-12-04 09:34:44 -08002598#endif
Jingning Han47558172017-07-05 16:33:19 -07002599
Angie Chiang2ed03a32017-04-16 18:00:06 -07002600 av1_inverse_transform_block_facade(xd, plane, block, blk_row, blk_col,
Frederic Barbier33b39f02017-11-21 11:11:24 +01002601 x->plane[plane].eobs[block],
2602 cm->reduced_tx_set_used);
Angie Chiang2ed03a32017-04-16 18:00:06 -07002603 }
Angie Chiang808d8592017-04-06 18:36:55 -07002604 return best_rd;
2605}
Angie Chiangcd9b03f2017-04-16 13:37:13 -07002606#endif // CONFIG_TXK_SEL