Coverage Report

Created: 2026-02-26 06:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libvpx/vp9/common/vp9_pred_common.c
Line
Count
Source
1
2
/*
3
 *  Copyright (c) 2012 The WebM project authors. All Rights Reserved.
4
 *
5
 *  Use of this source code is governed by a BSD-style license
6
 *  that can be found in the LICENSE file in the root of the source
7
 *  tree. An additional intellectual property rights grant can be found
8
 *  in the file PATENTS.  All contributing project authors may
9
 *  be found in the AUTHORS file in the root of the source tree.
10
 */
11
12
#include "vp9/common/vp9_common.h"
13
#include "vp9/common/vp9_pred_common.h"
14
#include "vp9/common/vp9_seg_common.h"
15
16
558k
int vp9_compound_reference_allowed(const VP9_COMMON *cm) {
17
558k
  int i;
18
1.48M
  for (i = 1; i < REFS_PER_FRAME; ++i)
19
1.06M
    if (cm->ref_frame_sign_bias[i + 1] != cm->ref_frame_sign_bias[1]) return 1;
20
21
424k
  return 0;
22
558k
}
23
24
44.0k
void vp9_setup_compound_reference_mode(VP9_COMMON *cm) {
25
44.0k
  if (cm->ref_frame_sign_bias[LAST_FRAME] ==
26
44.0k
      cm->ref_frame_sign_bias[GOLDEN_FRAME]) {
27
24.0k
    cm->comp_fixed_ref = ALTREF_FRAME;
28
24.0k
    cm->comp_var_ref[0] = LAST_FRAME;
29
24.0k
    cm->comp_var_ref[1] = GOLDEN_FRAME;
30
24.0k
  } else if (cm->ref_frame_sign_bias[LAST_FRAME] ==
31
20.0k
             cm->ref_frame_sign_bias[ALTREF_FRAME]) {
32
12.0k
    cm->comp_fixed_ref = GOLDEN_FRAME;
33
12.0k
    cm->comp_var_ref[0] = LAST_FRAME;
34
12.0k
    cm->comp_var_ref[1] = ALTREF_FRAME;
35
12.0k
  } else {
36
8.04k
    cm->comp_fixed_ref = LAST_FRAME;
37
8.04k
    cm->comp_var_ref[0] = GOLDEN_FRAME;
38
8.04k
    cm->comp_var_ref[1] = ALTREF_FRAME;
39
8.04k
  }
40
44.0k
}
41
42
int vp9_get_reference_mode_context(const VP9_COMMON *cm,
43
13.1M
                                   const MACROBLOCKD *xd) {
44
13.1M
  int ctx;
45
13.1M
  const MODE_INFO *const above_mi = xd->above_mi;
46
13.1M
  const MODE_INFO *const left_mi = xd->left_mi;
47
13.1M
  const int has_above = !!above_mi;
48
13.1M
  const int has_left = !!left_mi;
49
  // Note:
50
  // The mode info data structure has a one element border above and to the
51
  // left of the entries corresponding to real macroblocks.
52
  // The prediction flags in these dummy entries are initialized to 0.
53
13.1M
  if (has_above && has_left) {  // both edges available
54
12.7M
    if (!has_second_ref(above_mi) && !has_second_ref(left_mi))
55
      // neither edge uses comp pred (0/1)
56
5.37M
      ctx = (above_mi->ref_frame[0] == cm->comp_fixed_ref) ^
57
5.37M
            (left_mi->ref_frame[0] == cm->comp_fixed_ref);
58
7.33M
    else if (!has_second_ref(above_mi))
59
      // one of two edges uses comp pred (2/3)
60
2.63M
      ctx = 2 + (above_mi->ref_frame[0] == cm->comp_fixed_ref ||
61
1.96M
                 !is_inter_block(above_mi));
62
4.70M
    else if (!has_second_ref(left_mi))
63
      // one of two edges uses comp pred (2/3)
64
2.56M
      ctx = 2 + (left_mi->ref_frame[0] == cm->comp_fixed_ref ||
65
1.90M
                 !is_inter_block(left_mi));
66
2.13M
    else  // both edges use comp pred (4)
67
2.13M
      ctx = 4;
68
12.7M
  } else if (has_above || has_left) {  // one edge available
69
381k
    const MODE_INFO *edge_mi = has_above ? above_mi : left_mi;
70
71
381k
    if (!has_second_ref(edge_mi))
72
      // edge does not use comp pred (0/1)
73
284k
      ctx = edge_mi->ref_frame[0] == cm->comp_fixed_ref;
74
97.1k
    else
75
      // edge uses comp pred (3)
76
97.1k
      ctx = 3;
77
381k
  } else {  // no edges available (1)
78
15.0k
    ctx = 1;
79
15.0k
  }
80
13.1M
  assert(ctx >= 0 && ctx < COMP_INTER_CONTEXTS);
81
13.1M
  return ctx;
82
13.1M
}
83
84
// Returns a context number for the given MB prediction signal
85
int vp9_get_pred_context_comp_ref_p(const VP9_COMMON *cm,
86
9.92M
                                    const MACROBLOCKD *xd) {
87
9.92M
  int pred_context;
88
9.92M
  const MODE_INFO *const above_mi = xd->above_mi;
89
9.92M
  const MODE_INFO *const left_mi = xd->left_mi;
90
9.92M
  const int above_in_image = !!above_mi;
91
9.92M
  const int left_in_image = !!left_mi;
92
93
  // Note:
94
  // The mode info data structure has a one element border above and to the
95
  // left of the entries corresponding to real macroblocks.
96
  // The prediction flags in these dummy entries are initialized to 0.
97
9.92M
  const int fix_ref_idx = cm->ref_frame_sign_bias[cm->comp_fixed_ref];
98
9.92M
  const int var_ref_idx = !fix_ref_idx;
99
100
9.92M
  if (above_in_image && left_in_image) {  // both edges available
101
9.39M
    const int above_intra = !is_inter_block(above_mi);
102
9.39M
    const int left_intra = !is_inter_block(left_mi);
103
104
9.39M
    if (above_intra && left_intra) {  // intra/intra (2)
105
302k
      pred_context = 2;
106
9.09M
    } else if (above_intra || left_intra) {  // intra/inter
107
1.34M
      const MODE_INFO *edge_mi = above_intra ? left_mi : above_mi;
108
109
1.34M
      if (!has_second_ref(edge_mi))  // single pred (1/3)
110
570k
        pred_context = 1 + 2 * (edge_mi->ref_frame[0] != cm->comp_var_ref[1]);
111
773k
      else  // comp pred (1/3)
112
773k
        pred_context =
113
773k
            1 + 2 * (edge_mi->ref_frame[var_ref_idx] != cm->comp_var_ref[1]);
114
7.75M
    } else {  // inter/inter
115
7.75M
      const int l_sg = !has_second_ref(left_mi);
116
7.75M
      const int a_sg = !has_second_ref(above_mi);
117
7.75M
      const MV_REFERENCE_FRAME vrfa =
118
7.75M
          a_sg ? above_mi->ref_frame[0] : above_mi->ref_frame[var_ref_idx];
119
7.75M
      const MV_REFERENCE_FRAME vrfl =
120
7.75M
          l_sg ? left_mi->ref_frame[0] : left_mi->ref_frame[var_ref_idx];
121
122
7.75M
      if (vrfa == vrfl && cm->comp_var_ref[1] == vrfa) {
123
1.23M
        pred_context = 0;
124
6.51M
      } else if (l_sg && a_sg) {  // single/single
125
2.55M
        if ((vrfa == cm->comp_fixed_ref && vrfl == cm->comp_var_ref[0]) ||
126
2.23M
            (vrfl == cm->comp_fixed_ref && vrfa == cm->comp_var_ref[0]))
127
632k
          pred_context = 4;
128
1.91M
        else if (vrfa == vrfl)
129
1.11M
          pred_context = 3;
130
808k
        else
131
808k
          pred_context = 1;
132
3.96M
      } else if (l_sg || a_sg) {  // single/comp
133
648k
        const MV_REFERENCE_FRAME vrfc = l_sg ? vrfa : vrfl;
134
648k
        const MV_REFERENCE_FRAME rfs = a_sg ? vrfa : vrfl;
135
648k
        if (vrfc == cm->comp_var_ref[1] && rfs != cm->comp_var_ref[1])
136
183k
          pred_context = 1;
137
464k
        else if (rfs == cm->comp_var_ref[1] && vrfc != cm->comp_var_ref[1])
138
70.3k
          pred_context = 2;
139
394k
        else
140
394k
          pred_context = 4;
141
3.31M
      } else if (vrfa == vrfl) {  // comp/comp
142
1.76M
        pred_context = 4;
143
1.76M
      } else {
144
1.55M
        pred_context = 2;
145
1.55M
      }
146
7.75M
    }
147
9.39M
  } else if (above_in_image || left_in_image) {  // one edge available
148
501k
    const MODE_INFO *edge_mi = above_in_image ? above_mi : left_mi;
149
150
501k
    if (!is_inter_block(edge_mi)) {
151
27.6k
      pred_context = 2;
152
473k
    } else {
153
473k
      if (has_second_ref(edge_mi))
154
406k
        pred_context =
155
406k
            4 * (edge_mi->ref_frame[var_ref_idx] != cm->comp_var_ref[1]);
156
67.6k
      else
157
67.6k
        pred_context = 3 * (edge_mi->ref_frame[0] != cm->comp_var_ref[1]);
158
473k
    }
159
501k
  } else {  // no edges available (2)
160
21.0k
    pred_context = 2;
161
21.0k
  }
162
9.92M
  assert(pred_context >= 0 && pred_context < REF_CONTEXTS);
163
164
9.92M
  return pred_context;
165
9.92M
}
166
167
24.8M
int vp9_get_pred_context_single_ref_p1(const MACROBLOCKD *xd) {
168
24.8M
  int pred_context;
169
24.8M
  const MODE_INFO *const above_mi = xd->above_mi;
170
24.8M
  const MODE_INFO *const left_mi = xd->left_mi;
171
24.8M
  const int has_above = !!above_mi;
172
24.8M
  const int has_left = !!left_mi;
173
  // Note:
174
  // The mode info data structure has a one element border above and to the
175
  // left of the entries corresponding to real macroblocks.
176
  // The prediction flags in these dummy entries are initialized to 0.
177
24.8M
  if (has_above && has_left) {  // both edges available
178
15.5M
    const int above_intra = !is_inter_block(above_mi);
179
15.5M
    const int left_intra = !is_inter_block(left_mi);
180
181
15.5M
    if (above_intra && left_intra) {  // intra/intra
182
2.21M
      pred_context = 2;
183
13.3M
    } else if (above_intra || left_intra) {  // intra/inter or inter/intra
184
2.28M
      const MODE_INFO *edge_mi = above_intra ? left_mi : above_mi;
185
2.28M
      if (!has_second_ref(edge_mi))
186
1.72M
        pred_context = 4 * (edge_mi->ref_frame[0] == LAST_FRAME);
187
560k
      else
188
560k
        pred_context = 1 + (edge_mi->ref_frame[0] == LAST_FRAME ||
189
349k
                            edge_mi->ref_frame[1] == LAST_FRAME);
190
11.0M
    } else {  // inter/inter
191
11.0M
      const int above_has_second = has_second_ref(above_mi);
192
11.0M
      const int left_has_second = has_second_ref(left_mi);
193
11.0M
      const MV_REFERENCE_FRAME above0 = above_mi->ref_frame[0];
194
11.0M
      const MV_REFERENCE_FRAME above1 = above_mi->ref_frame[1];
195
11.0M
      const MV_REFERENCE_FRAME left0 = left_mi->ref_frame[0];
196
11.0M
      const MV_REFERENCE_FRAME left1 = left_mi->ref_frame[1];
197
198
11.0M
      if (above_has_second && left_has_second) {
199
1.25M
        pred_context = 1 + (above0 == LAST_FRAME || above1 == LAST_FRAME ||
200
649k
                            left0 == LAST_FRAME || left1 == LAST_FRAME);
201
9.82M
      } else if (above_has_second || left_has_second) {
202
3.78M
        const MV_REFERENCE_FRAME rfs = !above_has_second ? above0 : left0;
203
3.78M
        const MV_REFERENCE_FRAME crf1 = above_has_second ? above0 : left0;
204
3.78M
        const MV_REFERENCE_FRAME crf2 = above_has_second ? above1 : left1;
205
206
3.78M
        if (rfs == LAST_FRAME)
207
1.83M
          pred_context = 3 + (crf1 == LAST_FRAME || crf2 == LAST_FRAME);
208
1.94M
        else
209
1.94M
          pred_context = (crf1 == LAST_FRAME || crf2 == LAST_FRAME);
210
6.03M
      } else {
211
6.03M
        pred_context = 2 * (above0 == LAST_FRAME) + 2 * (left0 == LAST_FRAME);
212
6.03M
      }
213
11.0M
    }
214
15.5M
  } else if (has_above || has_left) {  // one edge available
215
7.21M
    const MODE_INFO *edge_mi = has_above ? above_mi : left_mi;
216
7.21M
    if (!is_inter_block(edge_mi)) {  // intra
217
2.19M
      pred_context = 2;
218
5.02M
    } else {  // inter
219
5.02M
      if (!has_second_ref(edge_mi))
220
4.95M
        pred_context = 4 * (edge_mi->ref_frame[0] == LAST_FRAME);
221
64.8k
      else
222
64.8k
        pred_context = 1 + (edge_mi->ref_frame[0] == LAST_FRAME ||
223
44.8k
                            edge_mi->ref_frame[1] == LAST_FRAME);
224
5.02M
    }
225
7.21M
  } else {  // no edges available
226
2.00M
    pred_context = 2;
227
2.00M
  }
228
229
24.8M
  assert(pred_context >= 0 && pred_context < REF_CONTEXTS);
230
24.8M
  return pred_context;
231
24.8M
}
232
233
16.2M
int vp9_get_pred_context_single_ref_p2(const MACROBLOCKD *xd) {
234
16.2M
  int pred_context;
235
16.2M
  const MODE_INFO *const above_mi = xd->above_mi;
236
16.2M
  const MODE_INFO *const left_mi = xd->left_mi;
237
16.2M
  const int has_above = !!above_mi;
238
16.2M
  const int has_left = !!left_mi;
239
240
  // Note:
241
  // The mode info data structure has a one element border above and to the
242
  // left of the entries corresponding to real macroblocks.
243
  // The prediction flags in these dummy entries are initialized to 0.
244
16.2M
  if (has_above && has_left) {  // both edges available
245
9.17M
    const int above_intra = !is_inter_block(above_mi);
246
9.17M
    const int left_intra = !is_inter_block(left_mi);
247
248
9.17M
    if (above_intra && left_intra) {  // intra/intra
249
2.10M
      pred_context = 2;
250
7.07M
    } else if (above_intra || left_intra) {  // intra/inter or inter/intra
251
1.45M
      const MODE_INFO *edge_mi = above_intra ? left_mi : above_mi;
252
1.45M
      if (!has_second_ref(edge_mi)) {
253
1.23M
        if (edge_mi->ref_frame[0] == LAST_FRAME)
254
600k
          pred_context = 3;
255
633k
        else
256
633k
          pred_context = 4 * (edge_mi->ref_frame[0] == GOLDEN_FRAME);
257
1.23M
      } else {
258
221k
        pred_context = 1 + 2 * (edge_mi->ref_frame[0] == GOLDEN_FRAME ||
259
163k
                                edge_mi->ref_frame[1] == GOLDEN_FRAME);
260
221k
      }
261
5.62M
    } else {  // inter/inter
262
5.62M
      const int above_has_second = has_second_ref(above_mi);
263
5.62M
      const int left_has_second = has_second_ref(left_mi);
264
5.62M
      const MV_REFERENCE_FRAME above0 = above_mi->ref_frame[0];
265
5.62M
      const MV_REFERENCE_FRAME above1 = above_mi->ref_frame[1];
266
5.62M
      const MV_REFERENCE_FRAME left0 = left_mi->ref_frame[0];
267
5.62M
      const MV_REFERENCE_FRAME left1 = left_mi->ref_frame[1];
268
269
5.62M
      if (above_has_second && left_has_second) {
270
399k
        if (above0 == left0 && above1 == left1)
271
270k
          pred_context =
272
270k
              3 * (above0 == GOLDEN_FRAME || above1 == GOLDEN_FRAME ||
273
196k
                   left0 == GOLDEN_FRAME || left1 == GOLDEN_FRAME);
274
128k
        else
275
128k
          pred_context = 2;
276
5.22M
      } else if (above_has_second || left_has_second) {
277
2.23M
        const MV_REFERENCE_FRAME rfs = !above_has_second ? above0 : left0;
278
2.23M
        const MV_REFERENCE_FRAME crf1 = above_has_second ? above0 : left0;
279
2.23M
        const MV_REFERENCE_FRAME crf2 = above_has_second ? above1 : left1;
280
281
2.23M
        if (rfs == GOLDEN_FRAME)
282
303k
          pred_context = 3 + (crf1 == GOLDEN_FRAME || crf2 == GOLDEN_FRAME);
283
1.93M
        else if (rfs == ALTREF_FRAME)
284
703k
          pred_context = crf1 == GOLDEN_FRAME || crf2 == GOLDEN_FRAME;
285
1.23M
        else
286
1.23M
          pred_context = 1 + 2 * (crf1 == GOLDEN_FRAME || crf2 == GOLDEN_FRAME);
287
2.98M
      } else {
288
2.98M
        if (above0 == LAST_FRAME && left0 == LAST_FRAME) {
289
982k
          pred_context = 3;
290
2.00M
        } else if (above0 == LAST_FRAME || left0 == LAST_FRAME) {
291
775k
          const MV_REFERENCE_FRAME edge0 =
292
775k
              (above0 == LAST_FRAME) ? left0 : above0;
293
775k
          pred_context = 4 * (edge0 == GOLDEN_FRAME);
294
1.22M
        } else {
295
1.22M
          pred_context =
296
1.22M
              2 * (above0 == GOLDEN_FRAME) + 2 * (left0 == GOLDEN_FRAME);
297
1.22M
        }
298
2.98M
      }
299
5.62M
    }
300
9.17M
  } else if (has_above || has_left) {  // one edge available
301
5.51M
    const MODE_INFO *edge_mi = has_above ? above_mi : left_mi;
302
303
5.51M
    if (!is_inter_block(edge_mi) ||
304
3.48M
        (edge_mi->ref_frame[0] == LAST_FRAME && !has_second_ref(edge_mi)))
305
3.82M
      pred_context = 2;
306
1.69M
    else if (!has_second_ref(edge_mi))
307
1.65M
      pred_context = 4 * (edge_mi->ref_frame[0] == GOLDEN_FRAME);
308
34.9k
    else
309
34.9k
      pred_context = 3 * (edge_mi->ref_frame[0] == GOLDEN_FRAME ||
310
29.2k
                          edge_mi->ref_frame[1] == GOLDEN_FRAME);
311
5.51M
  } else {  // no edges available (2)
312
1.57M
    pred_context = 2;
313
1.57M
  }
314
16.2M
  assert(pred_context >= 0 && pred_context < REF_CONTEXTS);
315
16.2M
  return pred_context;
316
16.2M
}