Coverage Report

Created: 2026-06-10 07:00

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/work/dav1d/src/ipred_prepare_tmpl.c
Line
Count
Source
1
/*
2
 * Copyright © 2018, VideoLAN and dav1d authors
3
 * Copyright © 2018, Two Orioles, LLC
4
 * All rights reserved.
5
 *
6
 * Redistribution and use in source and binary forms, with or without
7
 * modification, are permitted provided that the following conditions are met:
8
 *
9
 * 1. Redistributions of source code must retain the above copyright notice, this
10
 *    list of conditions and the following disclaimer.
11
 *
12
 * 2. Redistributions in binary form must reproduce the above copyright notice,
13
 *    this list of conditions and the following disclaimer in the documentation
14
 *    and/or other materials provided with the distribution.
15
 *
16
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
17
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
20
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
 */
27
28
#include "config.h"
29
30
#include <stdint.h>
31
#include <string.h>
32
33
#include "common/intops.h"
34
35
#include "src/ipred_prepare.h"
36
37
static const uint8_t av1_mode_conv[N_INTRA_PRED_MODES]
38
                                  [2 /* have_left */][2 /* have_top */] =
39
{
40
    [DC_PRED]    = { { DC_128_PRED,  TOP_DC_PRED },
41
                     { LEFT_DC_PRED, DC_PRED     } },
42
    [PAETH_PRED] = { { DC_128_PRED,  VERT_PRED   },
43
                     { HOR_PRED,     PAETH_PRED  } },
44
};
45
46
static const uint8_t av1_mode_to_angle_map[8] = {
47
    90, 180, 45, 135, 113, 157, 203, 67
48
};
49
50
static const struct {
51
    uint8_t needs_left:1;
52
    uint8_t needs_top:1;
53
    uint8_t needs_topleft:1;
54
    uint8_t needs_topright:1;
55
    uint8_t needs_bottomleft:1;
56
} av1_intra_prediction_edges[N_IMPL_INTRA_PRED_MODES] = {
57
    [DC_PRED]       = { .needs_top  = 1, .needs_left = 1 },
58
    [VERT_PRED]     = { .needs_top  = 1 },
59
    [HOR_PRED]      = { .needs_left = 1 },
60
    [LEFT_DC_PRED]  = { .needs_left = 1 },
61
    [TOP_DC_PRED]   = { .needs_top  = 1 },
62
    [DC_128_PRED]   = { 0 },
63
    [Z1_PRED]       = { .needs_top = 1, .needs_topright = 1,
64
                        .needs_topleft = 1 },
65
    [Z2_PRED]       = { .needs_left = 1, .needs_top = 1, .needs_topleft = 1 },
66
    [Z3_PRED]       = { .needs_left = 1, .needs_bottomleft = 1,
67
                        .needs_topleft = 1 },
68
    [SMOOTH_PRED]   = { .needs_left = 1, .needs_top = 1 },
69
    [SMOOTH_V_PRED] = { .needs_left = 1, .needs_top = 1 },
70
    [SMOOTH_H_PRED] = { .needs_left = 1, .needs_top = 1 },
71
    [PAETH_PRED]    = { .needs_left = 1, .needs_top = 1, .needs_topleft = 1 },
72
    [FILTER_PRED]   = { .needs_left = 1, .needs_top = 1, .needs_topleft = 1 },
73
};
74
75
enum IntraPredMode
76
bytefn(dav1d_prepare_intra_edges)(const int x, const int have_left,
77
                                  const int y, const int have_top,
78
                                  const int w, const int h,
79
                                  const enum EdgeFlags edge_flags,
80
                                  const pixel *const dst,
81
                                  const ptrdiff_t stride,
82
                                  const pixel *prefilter_toplevel_sb_edge,
83
                                  enum IntraPredMode mode, int *const angle,
84
                                  const int tw, const int th, const int filter_edge,
85
                                  pixel *const topleft_out HIGHBD_DECL_SUFFIX)
86
16.7M
{
87
16.7M
    const int bitdepth = bitdepth_from_max(bitdepth_max);
88
16.7M
    assert(y < h && x < w);
89
90
16.7M
    switch (mode) {
91
409k
    case VERT_PRED:
92
1.09M
    case HOR_PRED:
93
1.24M
    case DIAG_DOWN_LEFT_PRED:
94
1.37M
    case DIAG_DOWN_RIGHT_PRED:
95
1.50M
    case VERT_RIGHT_PRED:
96
1.66M
    case HOR_DOWN_PRED:
97
2.04M
    case HOR_UP_PRED:
98
2.23M
    case VERT_LEFT_PRED: {
99
2.23M
        *angle = av1_mode_to_angle_map[mode - VERT_PRED] + 3 * *angle;
100
101
2.23M
        if (*angle <= 90)
102
640k
            mode = *angle < 90 && have_top ? Z1_PRED : VERT_PRED;
103
1.59M
        else if (*angle < 180)
104
702k
            mode = Z2_PRED;
105
895k
        else
106
895k
            mode = *angle > 180 && have_left ? Z3_PRED : HOR_PRED;
107
2.23M
        break;
108
2.04M
    }
109
11.9M
    case DC_PRED:
110
12.7M
    case PAETH_PRED:
111
12.7M
        mode = av1_mode_conv[mode][have_left][have_top];
112
12.7M
        break;
113
1.68M
    default:
114
1.68M
        break;
115
16.7M
    }
116
117
16.5M
    const pixel *dst_top;
118
16.5M
    if (have_top &&
119
15.4M
        (av1_intra_prediction_edges[mode].needs_top ||
120
797k
         av1_intra_prediction_edges[mode].needs_topleft ||
121
352k
         (av1_intra_prediction_edges[mode].needs_left && !have_left)))
122
15.1M
    {
123
15.1M
        if (prefilter_toplevel_sb_edge) {
124
1.45M
            dst_top = &prefilter_toplevel_sb_edge[x * 4];
125
13.7M
        } else {
126
13.7M
            dst_top = &dst[-PXSTRIDE(stride)];
127
13.7M
        }
128
15.1M
    }
129
130
16.5M
    if (av1_intra_prediction_edges[mode].needs_left) {
131
14.6M
        const int sz = th << 2;
132
14.6M
        pixel *const left = &topleft_out[-sz];
133
134
14.6M
        if (have_left) {
135
14.2M
            const int px_have = imin(sz, (h - y) << 2);
136
137
129M
            for (int i = 0; i < px_have; i++)
138
115M
                left[sz - 1 - i] = dst[PXSTRIDE(stride) * i - 1];
139
14.2M
            if (px_have < sz)
140
254k
                pixel_set(left, left[sz - px_have], sz - px_have);
141
14.2M
        } else {
142
416k
            pixel_set(left, have_top ? *dst_top : ((1 << bitdepth) >> 1) + 1, sz);
143
416k
        }
144
145
14.6M
        if (av1_intra_prediction_edges[mode].needs_bottomleft) {
146
477k
            const int have_bottomleft = (!have_left || y + th >= h) ? 0 :
147
477k
                                        (edge_flags & EDGE_I444_LEFT_HAS_BOTTOM);
148
149
477k
            if (have_bottomleft) {
150
114k
                const int px_have = imin(sz, (h - y - th) << 2);
151
152
1.39M
                for (int i = 0; i < px_have; i++)
153
1.28M
                    left[-(i + 1)] = dst[(sz + i) * PXSTRIDE(stride) - 1];
154
114k
                if (px_have < sz)
155
1.59k
                    pixel_set(left - sz, left[-px_have], sz - px_have);
156
363k
            } else {
157
363k
                pixel_set(left - sz, left[0], sz);
158
363k
            }
159
477k
        }
160
14.6M
    }
161
162
16.5M
    if (av1_intra_prediction_edges[mode].needs_top) {
163
14.8M
        const int sz = tw << 2;
164
14.8M
        pixel *const top = &topleft_out[1];
165
166
14.8M
        if (have_top) {
167
14.5M
            const int px_have = imin(sz, (w - x) << 2);
168
14.5M
            pixel_copy(top, dst_top, px_have);
169
14.5M
            if (px_have < sz)
170
583k
                pixel_set(top + px_have, top[px_have - 1], sz - px_have);
171
14.5M
        } else {
172
323k
            pixel_set(top, have_left ? dst[-1] : ((1 << bitdepth) >> 1) - 1, sz);
173
323k
        }
174
175
14.8M
        if (av1_intra_prediction_edges[mode].needs_topright) {
176
380k
            const int have_topright = (!have_top || x + tw >= w) ? 0 :
177
380k
                                      (edge_flags & EDGE_I444_TOP_HAS_RIGHT);
178
179
380k
            if (have_topright) {
180
226k
                const int px_have = imin(sz, (w - x - tw) << 2);
181
182
226k
                pixel_copy(top + sz, &dst_top[sz], px_have);
183
226k
                if (px_have < sz)
184
3.89k
                    pixel_set(top + sz + px_have, top[sz + px_have - 1],
185
1.89k
                              sz - px_have);
186
226k
            } else {
187
153k
                pixel_set(top + sz, top[sz - 1], sz);
188
153k
            }
189
380k
        }
190
14.8M
    }
191
192
16.5M
    if (av1_intra_prediction_edges[mode].needs_topleft) {
193
2.48M
        if (have_left)
194
2.28M
            *topleft_out = have_top ? dst_top[-1] : dst[-1];
195
193k
        else
196
193k
            *topleft_out = have_top ? *dst_top : (1 << bitdepth) >> 1;
197
198
2.48M
        if (mode == Z2_PRED && tw + th >= 6 && filter_edge)
199
212k
            *topleft_out = ((topleft_out[-1] + topleft_out[1]) * 5 +
200
212k
                            topleft_out[0] * 6 + 8) >> 4;
201
2.48M
    }
202
203
16.5M
    return mode;
204
16.7M
}
dav1d_prepare_intra_edges_8bpc
Line
Count
Source
86
7.01M
{
87
7.01M
    const int bitdepth = bitdepth_from_max(bitdepth_max);
88
7.01M
    assert(y < h && x < w);
89
90
7.01M
    switch (mode) {
91
153k
    case VERT_PRED:
92
427k
    case HOR_PRED:
93
483k
    case DIAG_DOWN_LEFT_PRED:
94
544k
    case DIAG_DOWN_RIGHT_PRED:
95
602k
    case VERT_RIGHT_PRED:
96
669k
    case HOR_DOWN_PRED:
97
819k
    case HOR_UP_PRED:
98
893k
    case VERT_LEFT_PRED: {
99
893k
        *angle = av1_mode_to_angle_map[mode - VERT_PRED] + 3 * *angle;
100
101
893k
        if (*angle <= 90)
102
245k
            mode = *angle < 90 && have_top ? Z1_PRED : VERT_PRED;
103
648k
        else if (*angle < 180)
104
298k
            mode = Z2_PRED;
105
350k
        else
106
350k
            mode = *angle > 180 && have_left ? Z3_PRED : HOR_PRED;
107
893k
        break;
108
819k
    }
109
5.01M
    case DC_PRED:
110
5.45M
    case PAETH_PRED:
111
5.45M
        mode = av1_mode_conv[mode][have_left][have_top];
112
5.45M
        break;
113
687k
    default:
114
687k
        break;
115
7.01M
    }
116
117
6.98M
    const pixel *dst_top;
118
6.98M
    if (have_top &&
119
6.47M
        (av1_intra_prediction_edges[mode].needs_top ||
120
309k
         av1_intra_prediction_edges[mode].needs_topleft ||
121
142k
         (av1_intra_prediction_edges[mode].needs_left && !have_left)))
122
6.38M
    {
123
6.38M
        if (prefilter_toplevel_sb_edge) {
124
673k
            dst_top = &prefilter_toplevel_sb_edge[x * 4];
125
5.70M
        } else {
126
5.70M
            dst_top = &dst[-PXSTRIDE(stride)];
127
5.70M
        }
128
6.38M
    }
129
130
6.98M
    if (av1_intra_prediction_edges[mode].needs_left) {
131
6.12M
        const int sz = th << 2;
132
6.12M
        pixel *const left = &topleft_out[-sz];
133
134
6.12M
        if (have_left) {
135
5.90M
            const int px_have = imin(sz, (h - y) << 2);
136
137
55.6M
            for (int i = 0; i < px_have; i++)
138
49.7M
                left[sz - 1 - i] = dst[PXSTRIDE(stride) * i - 1];
139
5.90M
            if (px_have < sz)
140
124k
                pixel_set(left, left[sz - px_have], sz - px_have);
141
5.90M
        } else {
142
219k
            pixel_set(left, have_top ? *dst_top : ((1 << bitdepth) >> 1) + 1, sz);
143
219k
        }
144
145
6.12M
        if (av1_intra_prediction_edges[mode].needs_bottomleft) {
146
187k
            const int have_bottomleft = (!have_left || y + th >= h) ? 0 :
147
187k
                                        (edge_flags & EDGE_I444_LEFT_HAS_BOTTOM);
148
149
187k
            if (have_bottomleft) {
150
42.6k
                const int px_have = imin(sz, (h - y - th) << 2);
151
152
485k
                for (int i = 0; i < px_have; i++)
153
442k
                    left[-(i + 1)] = dst[(sz + i) * PXSTRIDE(stride) - 1];
154
42.6k
                if (px_have < sz)
155
688
                    pixel_set(left - sz, left[-px_have], sz - px_have);
156
145k
            } else {
157
145k
                pixel_set(left - sz, left[0], sz);
158
145k
            }
159
187k
        }
160
6.12M
    }
161
162
6.98M
    if (av1_intra_prediction_edges[mode].needs_top) {
163
6.31M
        const int sz = tw << 2;
164
6.31M
        pixel *const top = &topleft_out[1];
165
166
6.31M
        if (have_top) {
167
6.16M
            const int px_have = imin(sz, (w - x) << 2);
168
6.16M
            pixel_copy(top, dst_top, px_have);
169
6.16M
            if (px_have < sz)
170
287k
                pixel_set(top + px_have, top[px_have - 1], sz - px_have);
171
6.16M
        } else {
172
145k
            pixel_set(top, have_left ? dst[-1] : ((1 << bitdepth) >> 1) - 1, sz);
173
145k
        }
174
175
6.31M
        if (av1_intra_prediction_edges[mode].needs_topright) {
176
143k
            const int have_topright = (!have_top || x + tw >= w) ? 0 :
177
143k
                                      (edge_flags & EDGE_I444_TOP_HAS_RIGHT);
178
179
143k
            if (have_topright) {
180
80.0k
                const int px_have = imin(sz, (w - x - tw) << 2);
181
182
80.0k
                pixel_copy(top + sz, &dst_top[sz], px_have);
183
80.0k
                if (px_have < sz)
184
1.89k
                    pixel_set(top + sz + px_have, top[sz + px_have - 1],
185
1.89k
                              sz - px_have);
186
80.0k
            } else {
187
63.8k
                pixel_set(top + sz, top[sz - 1], sz);
188
63.8k
            }
189
143k
        }
190
6.31M
    }
191
192
6.98M
    if (av1_intra_prediction_edges[mode].needs_topleft) {
193
1.09M
        if (have_left)
194
987k
            *topleft_out = have_top ? dst_top[-1] : dst[-1];
195
102k
        else
196
102k
            *topleft_out = have_top ? *dst_top : (1 << bitdepth) >> 1;
197
198
1.09M
        if (mode == Z2_PRED && tw + th >= 6 && filter_edge)
199
83.2k
            *topleft_out = ((topleft_out[-1] + topleft_out[1]) * 5 +
200
83.2k
                            topleft_out[0] * 6 + 8) >> 4;
201
1.09M
    }
202
203
6.98M
    return mode;
204
7.01M
}
dav1d_prepare_intra_edges_16bpc
Line
Count
Source
86
9.70M
{
87
9.70M
    const int bitdepth = bitdepth_from_max(bitdepth_max);
88
9.70M
    assert(y < h && x < w);
89
90
9.70M
    switch (mode) {
91
255k
    case VERT_PRED:
92
672k
    case HOR_PRED:
93
759k
    case DIAG_DOWN_LEFT_PRED:
94
832k
    case DIAG_DOWN_RIGHT_PRED:
95
903k
    case VERT_RIGHT_PRED:
96
991k
    case HOR_DOWN_PRED:
97
1.22M
    case HOR_UP_PRED:
98
1.34M
    case VERT_LEFT_PRED: {
99
1.34M
        *angle = av1_mode_to_angle_map[mode - VERT_PRED] + 3 * *angle;
100
101
1.34M
        if (*angle <= 90)
102
395k
            mode = *angle < 90 && have_top ? Z1_PRED : VERT_PRED;
103
950k
        else if (*angle < 180)
104
404k
            mode = Z2_PRED;
105
545k
        else
106
545k
            mode = *angle > 180 && have_left ? Z3_PRED : HOR_PRED;
107
1.34M
        break;
108
1.22M
    }
109
6.93M
    case DC_PRED:
110
7.31M
    case PAETH_PRED:
111
7.31M
        mode = av1_mode_conv[mode][have_left][have_top];
112
7.31M
        break;
113
999k
    default:
114
999k
        break;
115
9.70M
    }
116
117
9.54M
    const pixel *dst_top;
118
9.54M
    if (have_top &&
119
8.95M
        (av1_intra_prediction_edges[mode].needs_top ||
120
487k
         av1_intra_prediction_edges[mode].needs_topleft ||
121
209k
         (av1_intra_prediction_edges[mode].needs_left && !have_left)))
122
8.79M
    {
123
8.79M
        if (prefilter_toplevel_sb_edge) {
124
777k
            dst_top = &prefilter_toplevel_sb_edge[x * 4];
125
8.01M
        } else {
126
8.01M
            dst_top = &dst[-PXSTRIDE(stride)];
127
8.01M
        }
128
8.79M
    }
129
130
9.54M
    if (av1_intra_prediction_edges[mode].needs_left) {
131
8.52M
        const int sz = th << 2;
132
8.52M
        pixel *const left = &topleft_out[-sz];
133
134
8.52M
        if (have_left) {
135
8.32M
            const int px_have = imin(sz, (h - y) << 2);
136
137
74.1M
            for (int i = 0; i < px_have; i++)
138
65.8M
                left[sz - 1 - i] = dst[PXSTRIDE(stride) * i - 1];
139
8.32M
            if (px_have < sz)
140
130k
                pixel_set(left, left[sz - px_have], sz - px_have);
141
8.32M
        } else {
142
196k
            pixel_set(left, have_top ? *dst_top : ((1 << bitdepth) >> 1) + 1, sz);
143
196k
        }
144
145
8.52M
        if (av1_intra_prediction_edges[mode].needs_bottomleft) {
146
290k
            const int have_bottomleft = (!have_left || y + th >= h) ? 0 :
147
290k
                                        (edge_flags & EDGE_I444_LEFT_HAS_BOTTOM);
148
149
290k
            if (have_bottomleft) {
150
71.9k
                const int px_have = imin(sz, (h - y - th) << 2);
151
152
914k
                for (int i = 0; i < px_have; i++)
153
842k
                    left[-(i + 1)] = dst[(sz + i) * PXSTRIDE(stride) - 1];
154
71.9k
                if (px_have < sz)
155
908
                    pixel_set(left - sz, left[-px_have], sz - px_have);
156
218k
            } else {
157
218k
                pixel_set(left - sz, left[0], sz);
158
218k
            }
159
290k
        }
160
8.52M
    }
161
162
9.54M
    if (av1_intra_prediction_edges[mode].needs_top) {
163
8.53M
        const int sz = tw << 2;
164
8.53M
        pixel *const top = &topleft_out[1];
165
166
8.53M
        if (have_top) {
167
8.35M
            const int px_have = imin(sz, (w - x) << 2);
168
8.35M
            pixel_copy(top, dst_top, px_have);
169
8.35M
            if (px_have < sz)
170
296k
                pixel_set(top + px_have, top[px_have - 1], sz - px_have);
171
8.35M
        } else {
172
178k
            pixel_set(top, have_left ? dst[-1] : ((1 << bitdepth) >> 1) - 1, sz);
173
178k
        }
174
175
8.53M
        if (av1_intra_prediction_edges[mode].needs_topright) {
176
236k
            const int have_topright = (!have_top || x + tw >= w) ? 0 :
177
236k
                                      (edge_flags & EDGE_I444_TOP_HAS_RIGHT);
178
179
236k
            if (have_topright) {
180
146k
                const int px_have = imin(sz, (w - x - tw) << 2);
181
182
146k
                pixel_copy(top + sz, &dst_top[sz], px_have);
183
146k
                if (px_have < sz)
184
2.00k
                    pixel_set(top + sz + px_have, top[sz + px_have - 1],
185
2.00k
                              sz - px_have);
186
146k
            } else {
187
89.6k
                pixel_set(top + sz, top[sz - 1], sz);
188
89.6k
            }
189
236k
        }
190
8.53M
    }
191
192
9.54M
    if (av1_intra_prediction_edges[mode].needs_topleft) {
193
1.39M
        if (have_left)
194
1.30M
            *topleft_out = have_top ? dst_top[-1] : dst[-1];
195
90.6k
        else
196
90.6k
            *topleft_out = have_top ? *dst_top : (1 << bitdepth) >> 1;
197
198
1.39M
        if (mode == Z2_PRED && tw + th >= 6 && filter_edge)
199
129k
            *topleft_out = ((topleft_out[-1] + topleft_out[1]) * 5 +
200
129k
                            topleft_out[0] * 6 + 8) >> 4;
201
1.39M
    }
202
203
9.54M
    return mode;
204
9.70M
}