Coverage Report

Created: 2025-07-18 06:57

/src/libavif/ext/aom/av1/common/reconinter_template.inc
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 2022, 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
12
#ifndef IS_DEC
13
#error "IS_DEC must be defined for reconinter_template.inc."
14
#endif
15
16
#if IS_DEC
17
static inline void build_one_inter_predictor(uint8_t *dst, int dst_stride,
18
                                             const MV *src_mv,
19
                                             InterPredParams *inter_pred_params,
20
                                             MACROBLOCKD *xd, int mi_x,
21
                                             int mi_y, int ref,
22
505k
                                             uint8_t **mc_buf) {
23
#else
24
static inline void build_one_inter_predictor(
25
    uint8_t *dst, int dst_stride, const MV *src_mv,
26
15.0M
    InterPredParams *inter_pred_params) {
27
15.0M
#endif  // IS_DEC
28
15.0M
  SubpelParams subpel_params;
29
15.0M
  uint8_t *src;
30
15.0M
  int src_stride;
31
#if IS_DEC
32
  dec_calc_subpel_params_and_extend(src_mv, inter_pred_params, xd, mi_x, mi_y,
33
                                    ref, mc_buf, &src, &subpel_params,
34
                                    &src_stride);
35
#else
36
  enc_calc_subpel_params(src_mv, inter_pred_params, &src, &subpel_params,
37
                         &src_stride);
38
#endif  // IS_DEC
39
15.5M
  if (inter_pred_params->comp_mode == UNIFORM_SINGLE ||
40
15.5M
      inter_pred_params->comp_mode == UNIFORM_COMP) {
41
15.5M
    av1_make_inter_predictor(src, src_stride, dst, dst_stride,
42
15.5M
                             inter_pred_params, &subpel_params);
43
15.5M
  } else {
44
23.6k
    av1_make_masked_inter_predictor(src, src_stride, dst, dst_stride,
45
23.6k
                                    inter_pred_params, &subpel_params);
46
23.6k
  }
47
15.0M
}
48
49
// True if the following hold:
50
//  1. Not intrabc and not build_for_obmc
51
//  2. At least one dimension is size 4 with subsampling
52
//  3. If sub-sampled, none of the previous blocks around the sub-sample
53
//     are intrabc or inter-blocks
54
static bool is_sub8x8_inter(const MACROBLOCKD *xd, int plane, BLOCK_SIZE bsize,
55
15.1M
                            int is_intrabc, int build_for_obmc) {
56
15.1M
  if (is_intrabc || build_for_obmc) {
57
62.9k
    return false;
58
62.9k
  }
59
60
15.0M
  const struct macroblockd_plane *const pd = &xd->plane[plane];
61
15.0M
  const int ss_x = pd->subsampling_x;
62
15.0M
  const int ss_y = pd->subsampling_y;
63
15.0M
  const int is_sub4_x = (block_size_wide[bsize] == 4) && ss_x;
64
15.0M
  const int is_sub4_y = (block_size_high[bsize] == 4) && ss_y;
65
15.0M
  if (!is_sub4_x && !is_sub4_y) {
66
15.0M
    return false;
67
15.0M
  }
68
69
  // For sub8x8 chroma blocks, we may be covering more than one luma block's
70
  // worth of pixels. Thus (mi_x, mi_y) may not be the correct coordinates for
71
  // the top-left corner of the prediction source - the correct top-left corner
72
  // is at (pre_x, pre_y).
73
26.3k
  const int row_start = is_sub4_y ? -1 : 0;
74
26.3k
  const int col_start = is_sub4_x ? -1 : 0;
75
76
48.9k
  for (int row = row_start; row <= 0; ++row) {
77
53.4k
    for (int col = col_start; col <= 0; ++col) {
78
30.8k
      const MB_MODE_INFO *this_mbmi = xd->mi[row * xd->mi_stride + col];
79
30.8k
      if (!is_inter_block(this_mbmi)) return false;
80
30.5k
      if (is_intrabc_block(this_mbmi)) return false;
81
30.5k
    }
82
22.8k
  }
83
26.0k
  return true;
84
26.3k
}
decodeframe.c:is_sub8x8_inter
Line
Count
Source
55
460k
                            int is_intrabc, int build_for_obmc) {
56
460k
  if (is_intrabc || build_for_obmc) {
57
62.9k
    return false;
58
62.9k
  }
59
60
397k
  const struct macroblockd_plane *const pd = &xd->plane[plane];
61
397k
  const int ss_x = pd->subsampling_x;
62
397k
  const int ss_y = pd->subsampling_y;
63
397k
  const int is_sub4_x = (block_size_wide[bsize] == 4) && ss_x;
64
397k
  const int is_sub4_y = (block_size_high[bsize] == 4) && ss_y;
65
397k
  if (!is_sub4_x && !is_sub4_y) {
66
383k
    return false;
67
383k
  }
68
69
  // For sub8x8 chroma blocks, we may be covering more than one luma block's
70
  // worth of pixels. Thus (mi_x, mi_y) may not be the correct coordinates for
71
  // the top-left corner of the prediction source - the correct top-left corner
72
  // is at (pre_x, pre_y).
73
13.9k
  const int row_start = is_sub4_y ? -1 : 0;
74
13.9k
  const int col_start = is_sub4_x ? -1 : 0;
75
76
36.5k
  for (int row = row_start; row <= 0; ++row) {
77
53.4k
    for (int col = col_start; col <= 0; ++col) {
78
30.8k
      const MB_MODE_INFO *this_mbmi = xd->mi[row * xd->mi_stride + col];
79
30.8k
      if (!is_inter_block(this_mbmi)) return false;
80
30.5k
      if (is_intrabc_block(this_mbmi)) return false;
81
30.5k
    }
82
22.8k
  }
83
13.6k
  return true;
84
13.9k
}
reconinter_enc.c:is_sub8x8_inter
Line
Count
Source
55
14.6M
                            int is_intrabc, int build_for_obmc) {
56
14.6M
  if (is_intrabc || build_for_obmc) {
57
0
    return false;
58
0
  }
59
60
14.6M
  const struct macroblockd_plane *const pd = &xd->plane[plane];
61
14.6M
  const int ss_x = pd->subsampling_x;
62
14.6M
  const int ss_y = pd->subsampling_y;
63
14.6M
  const int is_sub4_x = (block_size_wide[bsize] == 4) && ss_x;
64
14.6M
  const int is_sub4_y = (block_size_high[bsize] == 4) && ss_y;
65
14.6M
  if (!is_sub4_x && !is_sub4_y) {
66
14.6M
    return false;
67
14.6M
  }
68
69
  // For sub8x8 chroma blocks, we may be covering more than one luma block's
70
  // worth of pixels. Thus (mi_x, mi_y) may not be the correct coordinates for
71
  // the top-left corner of the prediction source - the correct top-left corner
72
  // is at (pre_x, pre_y).
73
12.4k
  const int row_start = is_sub4_y ? -1 : 0;
74
12.4k
  const int col_start = is_sub4_x ? -1 : 0;
75
76
12.4k
  for (int row = row_start; row <= 0; ++row) {
77
0
    for (int col = col_start; col <= 0; ++col) {
78
0
      const MB_MODE_INFO *this_mbmi = xd->mi[row * xd->mi_stride + col];
79
0
      if (!is_inter_block(this_mbmi)) return false;
80
0
      if (is_intrabc_block(this_mbmi)) return false;
81
0
    }
82
0
  }
83
12.4k
  return true;
84
12.4k
}
85
86
#if IS_DEC
87
static inline void build_inter_predictors_sub8x8(const AV1_COMMON *cm,
88
                                                 MACROBLOCKD *xd, int plane,
89
                                                 const MB_MODE_INFO *mi,
90
                                                 int mi_x, int mi_y,
91
13.7k
                                                 uint8_t **mc_buf) {
92
#else
93
static inline void build_inter_predictors_sub8x8(const AV1_COMMON *cm,
94
                                                 MACROBLOCKD *xd, int plane,
95
                                                 const MB_MODE_INFO *mi,
96
0
                                                 int mi_x, int mi_y) {
97
0
#endif  // IS_DEC
98
0
  const BLOCK_SIZE bsize = mi->bsize;
99
0
  struct macroblockd_plane *const pd = &xd->plane[plane];
100
0
  const bool ss_x = pd->subsampling_x;
101
0
  const bool ss_y = pd->subsampling_y;
102
0
  const int b4_w = block_size_wide[bsize] >> ss_x;
103
0
  const int b4_h = block_size_high[bsize] >> ss_y;
104
0
  const BLOCK_SIZE plane_bsize = get_plane_block_size(bsize, ss_x, ss_y);
105
0
  const int b8_w = block_size_wide[plane_bsize];
106
0
  const int b8_h = block_size_high[plane_bsize];
107
0
  const int is_compound = has_second_ref(mi);
108
0
  assert(!is_compound);
109
0
  assert(!is_intrabc_block(mi));
110
111
  // For sub8x8 chroma blocks, we may be covering more than one luma block's
112
  // worth of pixels. Thus (mi_x, mi_y) may not be the correct coordinates for
113
  // the top-left corner of the prediction source - the correct top-left corner
114
  // is at (pre_x, pre_y).
115
13.7k
  const int row_start = (block_size_high[bsize] == 4) && ss_y ? -1 : 0;
116
13.7k
  const int col_start = (block_size_wide[bsize] == 4) && ss_x ? -1 : 0;
117
13.7k
  const int pre_x = (mi_x + MI_SIZE * col_start) >> ss_x;
118
13.7k
  const int pre_y = (mi_y + MI_SIZE * row_start) >> ss_y;
119
120
0
  int row = row_start;
121
36.3k
  for (int y = 0; y < b8_h; y += b4_h) {
122
22.5k
    int col = col_start;
123
53.0k
    for (int x = 0; x < b8_w; x += b4_w) {
124
30.4k
      MB_MODE_INFO *this_mbmi = xd->mi[row * xd->mi_stride + col];
125
30.4k
      struct buf_2d *const dst_buf = &pd->dst;
126
30.4k
      uint8_t *dst = dst_buf->buf + dst_buf->stride * y + x;
127
30.4k
      int ref = 0;
128
30.4k
      const RefCntBuffer *ref_buf =
129
30.4k
          get_ref_frame_buf(cm, this_mbmi->ref_frame[ref]);
130
30.4k
      const struct scale_factors *ref_scale_factors =
131
30.4k
          get_ref_scale_factors_const(cm, this_mbmi->ref_frame[ref]);
132
30.4k
      const struct scale_factors *const sf = ref_scale_factors;
133
30.4k
      const struct buf_2d pre_buf = {
134
30.4k
        NULL,
135
30.4k
        (plane == 1) ? ref_buf->buf.u_buffer : ref_buf->buf.v_buffer,
136
30.4k
        ref_buf->buf.uv_crop_width,
137
30.4k
        ref_buf->buf.uv_crop_height,
138
30.4k
        ref_buf->buf.uv_stride,
139
30.4k
      };
140
141
30.4k
      const MV mv = this_mbmi->mv[ref].as_mv;
142
143
30.4k
      InterPredParams inter_pred_params;
144
30.4k
      av1_init_inter_params(&inter_pred_params, b4_w, b4_h, pre_y + y,
145
30.4k
                            pre_x + x, pd->subsampling_x, pd->subsampling_y,
146
30.4k
                            xd->bd, is_cur_buf_hbd(xd), mi->use_intrabc, sf,
147
30.4k
                            &pre_buf, this_mbmi->interp_filters);
148
30.4k
      inter_pred_params.conv_params =
149
30.4k
          get_conv_params_no_round(ref, plane, NULL, 0, is_compound, xd->bd);
150
151
#if IS_DEC
152
      build_one_inter_predictor(dst, dst_buf->stride, &mv, &inter_pred_params,
153
                                xd, mi_x + x, mi_y + y, ref, mc_buf);
154
#else
155
      build_one_inter_predictor(dst, dst_buf->stride, &mv, &inter_pred_params);
156
#endif  // IS_DEC
157
158
30.4k
      ++col;
159
30.4k
    }
160
22.5k
    ++row;
161
22.5k
  }
162
0
}
163
164
#if IS_DEC
165
static inline void build_inter_predictors_8x8_and_bigger(
166
    const AV1_COMMON *cm, MACROBLOCKD *xd, int plane, const MB_MODE_INFO *mi,
167
446k
    int build_for_obmc, int bw, int bh, int mi_x, int mi_y, uint8_t **mc_buf) {
168
#else
169
static inline void build_inter_predictors_8x8_and_bigger(
170
    const AV1_COMMON *cm, MACROBLOCKD *xd, int plane, const MB_MODE_INFO *mi,
171
14.6M
    int build_for_obmc, int bw, int bh, int mi_x, int mi_y) {
172
14.6M
#endif  // IS_DEC
173
14.6M
  const int is_compound = has_second_ref(mi);
174
14.6M
  const int is_intrabc = is_intrabc_block(mi);
175
14.6M
  assert(IMPLIES(is_intrabc, !is_compound));
176
14.6M
  struct macroblockd_plane *const pd = &xd->plane[plane];
177
14.6M
  struct buf_2d *const dst_buf = &pd->dst;
178
14.6M
  uint8_t *const dst = dst_buf->buf;
179
180
14.6M
  int is_global[2] = { 0, 0 };
181
30.2M
  for (int ref = 0; ref < 1 + is_compound; ++ref) {
182
15.1M
    const WarpedMotionParams *const wm = &xd->global_motion[mi->ref_frame[ref]];
183
15.1M
    is_global[ref] = is_global_mv_block(mi, wm->wmtype);
184
15.1M
  }
185
186
14.6M
  const BLOCK_SIZE bsize = mi->bsize;
187
14.6M
  const int ss_x = pd->subsampling_x;
188
14.6M
  const int ss_y = pd->subsampling_y;
189
14.6M
  const int row_start =
190
15.1M
      (block_size_high[bsize] == 4) && ss_y && !build_for_obmc ? -1 : 0;
191
14.6M
  const int col_start =
192
15.1M
      (block_size_wide[bsize] == 4) && ss_x && !build_for_obmc ? -1 : 0;
193
15.1M
  const int pre_x = (mi_x + MI_SIZE * col_start) >> ss_x;
194
15.1M
  const int pre_y = (mi_y + MI_SIZE * row_start) >> ss_y;
195
196
30.2M
  for (int ref = 0; ref < 1 + is_compound; ++ref) {
197
15.1M
    const struct scale_factors *const sf =
198
15.1M
        is_intrabc ? &cm->sf_identity : xd->block_ref_scale_factors[ref];
199
15.1M
    struct buf_2d *const pre_buf = is_intrabc ? dst_buf : &pd->pre[ref];
200
15.1M
    const MV mv = mi->mv[ref].as_mv;
201
15.1M
    const WarpTypesAllowed warp_types = { is_global[ref],
202
15.1M
                                          mi->motion_mode == WARPED_CAUSAL };
203
204
15.1M
    InterPredParams inter_pred_params;
205
15.1M
    av1_init_inter_params(&inter_pred_params, bw, bh, pre_y, pre_x,
206
15.1M
                          pd->subsampling_x, pd->subsampling_y, xd->bd,
207
15.1M
                          is_cur_buf_hbd(xd), mi->use_intrabc, sf, pre_buf,
208
15.1M
                          mi->interp_filters);
209
15.1M
    if (is_compound) av1_init_comp_mode(&inter_pred_params);
210
15.1M
    inter_pred_params.conv_params = get_conv_params_no_round(
211
15.1M
        ref, plane, xd->tmp_conv_dst, MAX_SB_SIZE, is_compound, xd->bd);
212
213
15.1M
    av1_dist_wtd_comp_weight_assign(
214
15.1M
        cm, mi, &inter_pred_params.conv_params.fwd_offset,
215
15.1M
        &inter_pred_params.conv_params.bck_offset,
216
15.1M
        &inter_pred_params.conv_params.use_dist_wtd_comp_avg, is_compound);
217
218
15.1M
    if (!build_for_obmc)
219
15.1M
      av1_init_warp_params(&inter_pred_params, &warp_types, ref, xd, mi);
220
221
15.1M
    if (is_masked_compound_type(mi->interinter_comp.type)) {
222
12.5k
      inter_pred_params.sb_type = mi->bsize;
223
12.5k
      inter_pred_params.mask_comp = mi->interinter_comp;
224
12.5k
      if (ref == 1) {
225
6.27k
        inter_pred_params.conv_params.do_average = 0;
226
6.27k
        inter_pred_params.comp_mode = MASK_COMP;
227
6.27k
      }
228
      // Assign physical buffer.
229
12.5k
      inter_pred_params.mask_comp.seg_mask = xd->seg_mask;
230
12.5k
    }
231
232
#if IS_DEC
233
    build_one_inter_predictor(dst, dst_buf->stride, &mv, &inter_pred_params, xd,
234
                              mi_x, mi_y, ref, mc_buf);
235
#else
236
    build_one_inter_predictor(dst, dst_buf->stride, &mv, &inter_pred_params);
237
#endif  // IS_DEC
238
15.1M
  }
239
14.6M
}
240
241
#if IS_DEC
242
static inline void build_inter_predictors(const AV1_COMMON *cm, MACROBLOCKD *xd,
243
                                          int plane, const MB_MODE_INFO *mi,
244
                                          int build_for_obmc, int bw, int bh,
245
                                          int mi_x, int mi_y,
246
460k
                                          uint8_t **mc_buf) {
247
460k
  if (is_sub8x8_inter(xd, plane, mi->bsize, is_intrabc_block(mi),
248
460k
                      build_for_obmc)) {
249
13.7k
    assert(bw < 8 || bh < 8);
250
13.7k
    build_inter_predictors_sub8x8(cm, xd, plane, mi, mi_x, mi_y, mc_buf);
251
446k
  } else {
252
446k
    build_inter_predictors_8x8_and_bigger(cm, xd, plane, mi, build_for_obmc, bw,
253
446k
                                          bh, mi_x, mi_y, mc_buf);
254
446k
  }
255
460k
}
256
#else
257
static inline void build_inter_predictors(const AV1_COMMON *cm, MACROBLOCKD *xd,
258
                                          int plane, const MB_MODE_INFO *mi,
259
                                          int build_for_obmc, int bw, int bh,
260
14.6M
                                          int mi_x, int mi_y) {
261
14.6M
  if (is_sub8x8_inter(xd, plane, mi->bsize, is_intrabc_block(mi),
262
14.6M
                      build_for_obmc)) {
263
0
    assert(bw < 8 || bh < 8);
264
0
    build_inter_predictors_sub8x8(cm, xd, plane, mi, mi_x, mi_y);
265
14.6M
  } else {
266
14.6M
    build_inter_predictors_8x8_and_bigger(cm, xd, plane, mi, build_for_obmc, bw,
267
14.6M
                                          bh, mi_x, mi_y);
268
14.6M
  }
269
14.6M
}
270
#endif  // IS_DEC