FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavfilter/drawutils.c
Date: 2026-04-19 15:33:43
Exec Total Coverage
Lines: 344 402 85.6%
Functions: 22 22 100.0%
Branches: 215 270 79.6%

Line Branch Exec Source
1 /*
2 * Copyright 2011 Stefano Sabatini <stefano.sabatini-lala poste it>
3 * Copyright 2012 Nicolas George <nicolas.george normalesup org>
4 *
5 * This file is part of FFmpeg.
6 *
7 * FFmpeg is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
11 *
12 * FFmpeg is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with FFmpeg; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21
22 #include <string.h>
23
24 #include "libavutil/avassert.h"
25 #include "libavutil/avutil.h"
26 #include "libavutil/csp.h"
27 #include "libavutil/intreadwrite.h"
28 #include "libavutil/pixdesc.h"
29 #include "colorspace.h"
30 #include "drawutils.h"
31 #include "formats.h"
32
33 enum { RED = 0, GREEN, BLUE, ALPHA };
34
35 1188 static int fill_map(const AVPixFmtDescriptor *desc, uint8_t *map)
36 {
37
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1188 times.
1188 if (desc->flags & (AV_PIX_FMT_FLAG_BITSTREAM | AV_PIX_FMT_FLAG_HWACCEL |
38 AV_PIX_FMT_FLAG_BAYER | AV_PIX_FMT_FLAG_XYZ | AV_PIX_FMT_FLAG_PAL))
39 return AVERROR(EINVAL);
40
3/4
✓ Branch 0 taken 34 times.
✓ Branch 1 taken 1154 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 1188 times.
1188 av_assert0(desc->nb_components == 3 + !!(desc->flags & AV_PIX_FMT_FLAG_ALPHA));
41
2/2
✓ Branch 0 taken 949 times.
✓ Branch 1 taken 239 times.
1188 if (desc->flags & AV_PIX_FMT_FLAG_PLANAR) {
42
2/2
✓ Branch 2 taken 307 times.
✓ Branch 3 taken 642 times.
949 if (desc->nb_components != av_pix_fmt_count_planes(av_pix_fmt_desc_get_id(desc)))
43 307 return AVERROR(EINVAL);
44 642 map[RED] = desc->comp[0].plane;
45 642 map[GREEN] = desc->comp[1].plane;
46 642 map[BLUE] = desc->comp[2].plane;
47
2/2
✓ Branch 0 taken 16 times.
✓ Branch 1 taken 626 times.
642 map[ALPHA] = (desc->flags & AV_PIX_FMT_FLAG_ALPHA) ? desc->comp[3].plane : 3;
48 } else {
49 239 int had0 = 0;
50 239 unsigned depthb = 0;
51
2/2
✓ Branch 0 taken 730 times.
✓ Branch 1 taken 236 times.
966 for (unsigned i = 0; i < desc->nb_components; i++) {
52 /* all components must have same depth in bytes */
53 730 unsigned db = (desc->comp[i].depth + 7) / 8;
54 730 unsigned pos = desc->comp[i].offset / db;
55
3/4
✓ Branch 0 taken 491 times.
✓ Branch 1 taken 239 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 491 times.
730 if (depthb && (depthb != db))
56 return AVERROR(ENOSYS);
57
58
2/2
✓ Branch 0 taken 3 times.
✓ Branch 1 taken 727 times.
730 if (desc->comp[i].offset % db)
59 3 return AVERROR(ENOSYS);
60
61 727 had0 |= pos == 0;
62 727 map[i] = pos;
63 727 depthb = db;
64 }
65
66
2/2
✓ Branch 0 taken 218 times.
✓ Branch 1 taken 18 times.
236 if (desc->nb_components == 3)
67
1/2
✓ Branch 0 taken 218 times.
✗ Branch 1 not taken.
218 map[ALPHA] = had0 ? 3 : 0;
68 }
69
70
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 878 times.
878 av_assert0(map[RED] != map[GREEN]);
71
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 878 times.
878 av_assert0(map[GREEN] != map[BLUE]);
72
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 878 times.
878 av_assert0(map[BLUE] != map[RED]);
73
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 878 times.
878 av_assert0(map[RED] != map[ALPHA]);
74
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 878 times.
878 av_assert0(map[GREEN] != map[ALPHA]);
75
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 878 times.
878 av_assert0(map[BLUE] != map[ALPHA]);
76
77 878 return 0;
78 }
79
80 612 int ff_fill_rgba_map(uint8_t *rgba_map, enum AVPixelFormat pix_fmt)
81 {
82 612 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(pix_fmt);
83
2/2
✓ Branch 0 taken 48 times.
✓ Branch 1 taken 564 times.
612 if (!(desc->flags & AV_PIX_FMT_FLAG_RGB))
84 48 return AVERROR(EINVAL);
85 564 return fill_map(desc, rgba_map);
86 }
87
88 624 int ff_fill_ayuv_map(uint8_t *ayuv_map, enum AVPixelFormat pix_fmt)
89 {
90 624 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(pix_fmt);
91
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 624 times.
624 if (desc->flags & AV_PIX_FMT_FLAG_RGB)
92 return AVERROR(EINVAL);
93 624 return fill_map(desc, ayuv_map);
94 }
95
96 49638 int ff_draw_init2(FFDrawContext *draw, enum AVPixelFormat format, enum AVColorSpace csp,
97 enum AVColorRange range, enum AVAlphaMode alpha, unsigned flags)
98 {
99 49638 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(format);
100 49638 const AVLumaCoefficients *luma = NULL;
101 const AVComponentDescriptor *c;
102 49638 unsigned nb_planes = 0;
103 49638 int pixelstep[MAX_PLANES] = { 0 };
104 49638 int depthb = 0;
105
106
2/4
✓ Branch 0 taken 49638 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 49638 times.
49638 if (!desc || !desc->name)
107 return AVERROR(EINVAL);
108
2/2
✓ Branch 0 taken 18130 times.
✓ Branch 1 taken 31508 times.
49638 if (desc->flags & AV_PIX_FMT_FLAG_BE)
109 18130 return AVERROR(ENOSYS);
110
2/2
✓ Branch 0 taken 7770 times.
✓ Branch 1 taken 23738 times.
31508 if (desc->flags & ~(AV_PIX_FMT_FLAG_PLANAR | AV_PIX_FMT_FLAG_RGB | AV_PIX_FMT_FLAG_ALPHA))
111 7770 return AVERROR(ENOSYS);
112
2/2
✓ Branch 0 taken 23688 times.
✓ Branch 1 taken 50 times.
23738 if (csp == AVCOL_SPC_UNSPECIFIED)
113
2/2
✓ Branch 0 taken 7770 times.
✓ Branch 1 taken 15918 times.
23688 csp = (desc->flags & AV_PIX_FMT_FLAG_RGB) ? AVCOL_SPC_RGB : AVCOL_SPC_SMPTE170M;
114
3/4
✓ Branch 0 taken 15918 times.
✓ Branch 1 taken 7820 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 15918 times.
23738 if (!(desc->flags & AV_PIX_FMT_FLAG_RGB) && !(luma = av_csp_luma_coeffs_from_avcsp(csp)))
115 return AVERROR(EINVAL);
116
2/2
✓ Branch 0 taken 23554 times.
✓ Branch 1 taken 184 times.
23738 if (range == AVCOL_RANGE_UNSPECIFIED)
117
4/4
✓ Branch 0 taken 23184 times.
✓ Branch 1 taken 185 times.
✓ Branch 2 taken 22999 times.
✓ Branch 3 taken 185 times.
46923 range = (format == AV_PIX_FMT_YUVJ420P || format == AV_PIX_FMT_YUVJ422P ||
118
4/4
✓ Branch 0 taken 22814 times.
✓ Branch 1 taken 185 times.
✓ Branch 2 taken 22629 times.
✓ Branch 3 taken 185 times.
22999 format == AV_PIX_FMT_YUVJ444P || format == AV_PIX_FMT_YUVJ411P ||
119
2/2
✓ Branch 0 taken 7770 times.
✓ Branch 1 taken 14859 times.
22629 format == AV_PIX_FMT_YUVJ440P || csp == AVCOL_SPC_RGB)
120
2/2
✓ Branch 0 taken 23369 times.
✓ Branch 1 taken 185 times.
46923 ? AVCOL_RANGE_JPEG : AVCOL_RANGE_MPEG;
121
3/4
✓ Branch 0 taken 14972 times.
✓ Branch 1 taken 8766 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 14972 times.
23738 if (range != AVCOL_RANGE_JPEG && range != AVCOL_RANGE_MPEG)
122 return AVERROR(EINVAL);
123
2/2
✓ Branch 0 taken 63854 times.
✓ Branch 1 taken 18188 times.
82042 for (unsigned i = 0; i < desc->nb_components; i++) {
124 int db;
125 63854 c = &desc->comp[i];
126 /* for now, only 8-16 bits formats */
127
4/4
✓ Branch 0 taken 62004 times.
✓ Branch 1 taken 1850 times.
✓ Branch 2 taken 740 times.
✓ Branch 3 taken 61264 times.
63854 if (c->depth < 8 || c->depth > 16)
128 2590 return AVERROR(ENOSYS);
129
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 61264 times.
61264 if (c->plane >= MAX_PLANES)
130 return AVERROR(ENOSYS);
131 /* data must either be in the high or low bits, never middle */
132
4/4
✓ Branch 0 taken 7245 times.
✓ Branch 1 taken 54019 times.
✓ Branch 2 taken 740 times.
✓ Branch 3 taken 6505 times.
61264 if (c->shift && ((c->shift + c->depth) & 0x7))
133 740 return AVERROR(ENOSYS);
134 /* mixed >8 and <=8 depth */
135 60524 db = (c->depth + 7) / 8;
136
3/4
✓ Branch 0 taken 39931 times.
✓ Branch 1 taken 20593 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 39931 times.
60524 if (depthb && (depthb != db))
137 return AVERROR(ENOSYS);
138 60524 depthb = db;
139
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 60524 times.
60524 if (db * (c->offset + 1) > 16)
140 return AVERROR(ENOSYS);
141
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 60524 times.
60524 if (c->offset % db)
142 return AVERROR(ENOSYS);
143 /* strange interleaving */
144
2/2
✓ Branch 0 taken 12133 times.
✓ Branch 1 taken 48391 times.
60524 if (pixelstep[c->plane] != 0 &&
145
2/2
✓ Branch 0 taken 1295 times.
✓ Branch 1 taken 10838 times.
12133 pixelstep[c->plane] != c->step)
146 1295 return AVERROR(ENOSYS);
147 59229 pixelstep[c->plane] = c->step;
148
2/2
✓ Branch 0 taken 925 times.
✓ Branch 1 taken 58304 times.
59229 if (pixelstep[c->plane] >= 8)
149 925 return AVERROR(ENOSYS);
150 58304 nb_planes = FFMAX(nb_planes, c->plane + 1);
151 }
152 18188 memset(draw, 0, sizeof(*draw));
153 18188 draw->desc = desc;
154 18188 draw->format = format;
155 18188 draw->nb_planes = nb_planes;
156 18188 draw->range = range;
157 18188 draw->csp = csp;
158 18188 draw->alpha = alpha;
159 18188 draw->flags = flags;
160
2/2
✓ Branch 0 taken 13513 times.
✓ Branch 1 taken 4675 times.
18188 if (luma)
161 13513 ff_fill_rgb2yuv_table(luma, draw->rgb2yuv);
162 18188 memcpy(draw->pixelstep, pixelstep, sizeof(draw->pixelstep));
163 18188 draw->hsub[1] = draw->hsub[2] = draw->hsub_max = desc->log2_chroma_w;
164 18188 draw->vsub[1] = draw->vsub[2] = draw->vsub_max = desc->log2_chroma_h;
165 18188 return 0;
166 }
167
168 243 int ff_draw_init_from_link(FFDrawContext *draw, const AVFilterLink *link,
169 unsigned flags)
170 {
171 243 return ff_draw_init2(draw, link->format, link->colorspace, link->color_range, link->alpha_mode, flags);
172 }
173
174 49395 int ff_draw_init(FFDrawContext *draw, enum AVPixelFormat format, unsigned flags)
175 {
176 49395 return ff_draw_init2(draw, format, AVCOL_SPC_UNSPECIFIED, AVCOL_RANGE_UNSPECIFIED, AVALPHA_MODE_UNSPECIFIED, flags);
177 }
178
179 220923 void ff_draw_color(FFDrawContext *draw, FFDrawColor *color, const uint8_t rgba[4])
180 {
181 double yuvad[4];
182 double rgbad[4];
183 220923 const AVPixFmtDescriptor *desc = draw->desc;
184
185
1/2
✓ Branch 0 taken 220923 times.
✗ Branch 1 not taken.
220923 if (rgba != color->rgba)
186 220923 memcpy(color->rgba, rgba, sizeof(color->rgba));
187
188 220923 memset(color->comp, 0, sizeof(color->comp));
189
190
2/2
✓ Branch 0 taken 883692 times.
✓ Branch 1 taken 220923 times.
1104615 for (int i = 0; i < 4; i++)
191 883692 rgbad[i] = color->rgba[i] / 255.;
192
193
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 220923 times.
220923 if (draw->alpha == AVALPHA_MODE_PREMULTIPLIED) {
194 for (int i = 0; i < 3; i++)
195 rgbad[i] *= rgbad[3];
196 }
197
198
2/2
✓ Branch 0 taken 54602 times.
✓ Branch 1 taken 166321 times.
220923 if (draw->desc->flags & AV_PIX_FMT_FLAG_RGB)
199 54602 memcpy(yuvad, rgbad, sizeof(double) * 3);
200 else
201 166321 ff_matrix_mul_3x3_vec(yuvad, rgbad, draw->rgb2yuv);
202
203 220923 yuvad[3] = rgbad[3];
204
205
2/2
✓ Branch 0 taken 662769 times.
✓ Branch 1 taken 220923 times.
883692 for (int i = 0; i < 3; i++) {
206
4/4
✓ Branch 0 taken 498963 times.
✓ Branch 1 taken 163806 times.
✓ Branch 2 taken 332642 times.
✓ Branch 3 taken 166321 times.
662769 int chroma = (!(draw->desc->flags & AV_PIX_FMT_FLAG_RGB) && i > 0);
207
2/2
✓ Branch 0 taken 498900 times.
✓ Branch 1 taken 163869 times.
662769 if (draw->range == AVCOL_RANGE_MPEG) {
208
2/2
✓ Branch 0 taken 332600 times.
✓ Branch 1 taken 166300 times.
498900 yuvad[i] *= (chroma ? 224. : 219.) / 255.;
209
2/2
✓ Branch 0 taken 332600 times.
✓ Branch 1 taken 166300 times.
498900 yuvad[i] += (chroma ? 128. : 16.) / 255.;
210
2/2
✓ Branch 0 taken 42 times.
✓ Branch 1 taken 163827 times.
163869 } else if (chroma) {
211 42 yuvad[i] += 0.5;
212 }
213 }
214
215 // Ensure we place the alpha appropriately for gray formats
216
2/2
✓ Branch 0 taken 10 times.
✓ Branch 1 taken 220913 times.
220923 if (desc->nb_components <= 2)
217 10 yuvad[1] = yuvad[3];
218
219
2/2
✓ Branch 0 taken 688780 times.
✓ Branch 1 taken 220923 times.
909703 for (unsigned i = 0; i < desc->nb_components; i++) {
220 688780 unsigned val = yuvad[i] * ((1 << (draw->desc->comp[i].depth + draw->desc->comp[i].shift)) - 1) + 0.5;
221
2/2
✓ Branch 0 taken 59677 times.
✓ Branch 1 taken 629103 times.
688780 if (desc->comp[i].depth > 8)
222 59677 color->comp[desc->comp[i].plane].u16[desc->comp[i].offset / 2] = val;
223 else
224 629103 color->comp[desc->comp[i].plane].u8[desc->comp[i].offset] = val;
225 }
226 220923 }
227
228 649702 static uint8_t *pointer_at(FFDrawContext *draw, uint8_t *data[], int linesize[],
229 int plane, int x, int y)
230 {
231 649702 return data[plane] +
232 1299404 (y >> draw->vsub[plane]) * linesize[plane] +
233 649702 (x >> draw->hsub[plane]) * draw->pixelstep[plane];
234 }
235
236 40 void ff_copy_rectangle2(FFDrawContext *draw,
237 uint8_t *dst[], int dst_linesize[],
238 uint8_t *src[], int src_linesize[],
239 int dst_x, int dst_y, int src_x, int src_y,
240 int w, int h)
241 {
242 int wp, hp;
243 uint8_t *p, *q;
244
245
2/2
✓ Branch 0 taken 120 times.
✓ Branch 1 taken 40 times.
160 for (int plane = 0; plane < draw->nb_planes; plane++) {
246 120 p = pointer_at(draw, src, src_linesize, plane, src_x, src_y);
247 120 q = pointer_at(draw, dst, dst_linesize, plane, dst_x, dst_y);
248 120 wp = AV_CEIL_RSHIFT(w, draw->hsub[plane]) * draw->pixelstep[plane];
249 120 hp = AV_CEIL_RSHIFT(h, draw->vsub[plane]);
250
2/2
✓ Branch 0 taken 23040 times.
✓ Branch 1 taken 120 times.
23160 for (int y = 0; y < hp; y++) {
251 23040 memcpy(q, p, wp);
252 23040 p += src_linesize[plane];
253 23040 q += dst_linesize[plane];
254 }
255 }
256 40 }
257
258 223154 void ff_fill_rectangle(FFDrawContext *draw, FFDrawColor *color,
259 uint8_t *dst[], int dst_linesize[],
260 int dst_x, int dst_y, int w, int h)
261 {
262 int wp, hp;
263 uint8_t *p0, *p;
264 223154 FFDrawColor color_tmp = *color;
265
266
2/2
✓ Branch 0 taken 560858 times.
✓ Branch 1 taken 223154 times.
784012 for (int plane = 0; plane < draw->nb_planes; plane++) {
267 560858 p0 = pointer_at(draw, dst, dst_linesize, plane, dst_x, dst_y);
268 560858 wp = AV_CEIL_RSHIFT(w, draw->hsub[plane]);
269 560858 hp = AV_CEIL_RSHIFT(h, draw->vsub[plane]);
270
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 560858 times.
560858 if (!hp)
271 return;
272 560858 p = p0;
273
274 if (HAVE_BIGENDIAN && draw->desc->comp[0].depth > 8) {
275 for (int x = 0; 2*x < draw->pixelstep[plane]; x++)
276 color_tmp.comp[plane].u16[x] = av_bswap16(color_tmp.comp[plane].u16[x]);
277 }
278
279 /* copy first line from color */
280
2/2
✓ Branch 0 taken 2423261 times.
✓ Branch 1 taken 560858 times.
2984119 for (int x = 0; x < wp; x++) {
281 2423261 memcpy(p, color_tmp.comp[plane].u8, draw->pixelstep[plane]);
282 2423261 p += draw->pixelstep[plane];
283 }
284 560858 wp *= draw->pixelstep[plane];
285 /* copy next lines from first line */
286 560858 p = p0 + dst_linesize[plane];
287
2/2
✓ Branch 0 taken 9684415 times.
✓ Branch 1 taken 560858 times.
10245273 for (int y = 1; y < hp; y++) {
288 9684415 memcpy(p, p0, wp);
289 9684415 p += dst_linesize[plane];
290 }
291 }
292 }
293
294 /**
295 * Clip interval [x; x+w[ within [0; wmax[.
296 * The resulting w may be negative if the final interval is empty.
297 * dx, if not null, return the difference between in and out value of x.
298 */
299 66016 static void clip_interval(int wmax, int *x, int *w, int *dx)
300 {
301
2/2
✓ Branch 0 taken 63958 times.
✓ Branch 1 taken 2058 times.
66016 if (dx)
302 63958 *dx = 0;
303
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 66016 times.
66016 if (*x < 0) {
304 if (dx)
305 *dx = -*x;
306 *w += *x;
307 *x = 0;
308 }
309
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 66016 times.
66016 if (*x + *w > wmax)
310 *w = wmax - *x;
311 66016 }
312
313 /**
314 * Decompose w pixels starting at x
315 * into start + (w starting at x) + end
316 * with x and w aligned on multiples of 1<<sub.
317 */
318 177208 static void subsampling_bounds(int sub, int *x, int *w, int *start, int *end)
319 {
320 177208 int mask = (1 << sub) - 1;
321
322 177208 *start = (-*x) & mask;
323 177208 *x += *start;
324 177208 *start = FFMIN(*start, *w);
325 177208 *w -= *start;
326 177208 *end = *w & mask;
327 177208 *w >>= sub;
328 177208 }
329
330 /* If alpha is in the [ 0 ; 0x1010101 ] range,
331 then alpha * value is in the [ 0 ; 0xFFFFFFFF ] range,
332 and >> 24 gives a correct rounding. */
333 74916 static void blend_line(uint8_t *dst, unsigned src, unsigned alpha,
334 int dx, int w, unsigned hsub, int left, int right)
335 {
336 74916 unsigned asrc = alpha * src;
337 74916 unsigned tau = 0x1010101 - alpha;
338
339
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 74916 times.
74916 if (left) {
340 unsigned suba = (left * alpha) >> hsub;
341 *dst = (*dst * (0x1010101 - suba) + src * suba) >> 24;
342 dst += dx;
343 }
344
2/2
✓ Branch 0 taken 6213600 times.
✓ Branch 1 taken 74916 times.
6288516 for (int x = 0; x < w; x++) {
345 6213600 *dst = (*dst * tau + asrc) >> 24;
346 6213600 dst += dx;
347 }
348
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 74916 times.
74916 if (right) {
349 unsigned suba = (right * alpha) >> hsub;
350 *dst = (*dst * (0x1010101 - suba) + src * suba) >> 24;
351 }
352 74916 }
353
354 11880 static void blend_line16(uint8_t *dst, unsigned src, unsigned alpha,
355 int dx, int w, unsigned hsub, int left, int right)
356 {
357 11880 unsigned asrc = alpha * src;
358 11880 unsigned tau = 0x10001 - alpha;
359
360
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11880 times.
11880 if (left) {
361 unsigned suba = (left * alpha) >> hsub;
362 uint16_t value = AV_RL16(dst);
363 AV_WL16(dst, (value * (0x10001 - suba) + src * suba) >> 16);
364 dst += dx;
365 }
366
2/2
✓ Branch 0 taken 792000 times.
✓ Branch 1 taken 11880 times.
803880 for (int x = 0; x < w; x++) {
367 792000 uint16_t value = AV_RL16(dst);
368 792000 AV_WL16(dst, (value * tau + asrc) >> 16);
369 792000 dst += dx;
370 }
371
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11880 times.
11880 if (right) {
372 unsigned suba = (right * alpha) >> hsub;
373 uint16_t value = AV_RL16(dst);
374 AV_WL16(dst, (value * (0x10001 - suba) + src * suba) >> 16);
375 }
376 11880 }
377
378 1029 void ff_blend_rectangle(FFDrawContext *draw, FFDrawColor *color,
379 uint8_t *dst[], int dst_linesize[],
380 int dst_w, int dst_h,
381 int x0, int y0, int w, int h)
382 {
383 unsigned alpha, nb_planes, nb_comp;
384 int w_sub, h_sub, x_sub, y_sub, left, right, top, bottom;
385 uint8_t *p0, *p;
386
387 2058 nb_comp = draw->desc->nb_components -
388
3/4
✓ Branch 0 taken 71 times.
✓ Branch 1 taken 958 times.
✓ Branch 2 taken 71 times.
✗ Branch 3 not taken.
1029 !!(draw->desc->flags & AV_PIX_FMT_FLAG_ALPHA && !(draw->flags & FF_DRAW_PROCESS_ALPHA));
389
390 /* TODO optimize if alpha = 0xFF */
391 1029 clip_interval(dst_w, &x0, &w, NULL);
392 1029 clip_interval(dst_h, &y0, &h, NULL);
393
3/6
✓ Branch 0 taken 1029 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 1029 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 1029 times.
1029 if (w <= 0 || h <= 0 || !color->rgba[3])
394 return;
395
2/2
✓ Branch 0 taken 919 times.
✓ Branch 1 taken 110 times.
1029 if (draw->desc->comp[0].depth <= 8) {
396 /* 0x10203 * alpha + 2 is in the [ 2 ; 0x1010101 - 2 ] range */
397 919 alpha = 0x10203 * color->rgba[3] + 0x2;
398 } else {
399 /* 0x101 * alpha is in the [ 2 ; 0x1001] range */
400 110 alpha = 0x101 * color->rgba[3] + 0x2;
401 }
402
3/4
✓ Branch 0 taken 71 times.
✓ Branch 1 taken 958 times.
✓ Branch 2 taken 71 times.
✗ Branch 3 not taken.
1029 nb_planes = draw->nb_planes - !!(draw->desc->flags & AV_PIX_FMT_FLAG_ALPHA && !(draw->flags & FF_DRAW_PROCESS_ALPHA));
403 1029 nb_planes += !nb_planes;
404
2/2
✓ Branch 0 taken 2775 times.
✓ Branch 1 taken 1029 times.
3804 for (unsigned plane = 0; plane < nb_planes; plane++) {
405 2775 p0 = pointer_at(draw, dst, dst_linesize, plane, x0, y0);
406 2775 w_sub = w;
407 2775 h_sub = h;
408 2775 x_sub = x0;
409 2775 y_sub = y0;
410 2775 subsampling_bounds(draw->hsub[plane], &x_sub, &w_sub, &left, &right);
411 2775 subsampling_bounds(draw->vsub[plane], &y_sub, &h_sub, &top, &bottom);
412
2/2
✓ Branch 0 taken 8325 times.
✓ Branch 1 taken 2775 times.
11100 for (unsigned comp = 0; comp < nb_comp; comp++) {
413 8325 const int depth = draw->desc->comp[comp].depth;
414 8325 const int offset = draw->desc->comp[comp].offset;
415 8325 const int index = offset / ((depth + 7) / 8);
416
417
2/2
✓ Branch 0 taken 5238 times.
✓ Branch 1 taken 3087 times.
8325 if (draw->desc->comp[comp].plane != plane)
418 5238 continue;
419 3087 p = p0 + offset;
420
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 3087 times.
3087 if (top) {
421 if (depth <= 8) {
422 blend_line(p, color->comp[plane].u8[index], alpha >> 1,
423 draw->pixelstep[plane], w_sub,
424 draw->hsub[plane], left, right);
425 } else {
426 blend_line16(p, color->comp[plane].u16[index], alpha >> 1,
427 draw->pixelstep[plane], w_sub,
428 draw->hsub[plane], left, right);
429 }
430 p += dst_linesize[plane];
431 }
432
2/2
✓ Branch 0 taken 2757 times.
✓ Branch 1 taken 330 times.
3087 if (depth <= 8) {
433
2/2
✓ Branch 0 taken 74916 times.
✓ Branch 1 taken 2757 times.
77673 for (int y = 0; y < h_sub; y++) {
434 74916 blend_line(p, color->comp[plane].u8[index], alpha,
435 draw->pixelstep[plane], w_sub,
436 74916 draw->hsub[plane], left, right);
437 74916 p += dst_linesize[plane];
438 }
439 } else {
440
2/2
✓ Branch 0 taken 11880 times.
✓ Branch 1 taken 330 times.
12210 for (int y = 0; y < h_sub; y++) {
441 11880 blend_line16(p, color->comp[plane].u16[index], alpha,
442 draw->pixelstep[plane], w_sub,
443 11880 draw->hsub[plane], left, right);
444 11880 p += dst_linesize[plane];
445 }
446 }
447
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 3087 times.
3087 if (bottom) {
448 if (depth <= 8) {
449 blend_line(p, color->comp[plane].u8[index], alpha >> 1,
450 draw->pixelstep[plane], w_sub,
451 draw->hsub[plane], left, right);
452 } else {
453 blend_line16(p, color->comp[plane].u16[index], alpha >> 1,
454 draw->pixelstep[plane], w_sub,
455 draw->hsub[plane], left, right);
456 }
457 }
458 }
459 }
460 }
461
462 1070080 static void blend_pixel16(uint8_t *dst, unsigned src, unsigned alpha,
463 const uint8_t *mask, int mask_linesize, int l2depth,
464 unsigned w, unsigned h, unsigned shift, unsigned xm0)
465 {
466 1070080 unsigned t = 0;
467 1070080 unsigned xmshf = 3 - l2depth;
468 1070080 unsigned xmmod = 7 >> l2depth;
469 1070080 unsigned mbits = (1 << (1 << l2depth)) - 1;
470 1070080 unsigned mmult = 255 / mbits;
471 1070080 uint16_t value = AV_RL16(dst);
472
473
2/2
✓ Branch 0 taken 1070080 times.
✓ Branch 1 taken 1070080 times.
2140160 for (unsigned y = 0; y < h; y++) {
474 1070080 unsigned xm = xm0;
475
2/2
✓ Branch 0 taken 1605120 times.
✓ Branch 1 taken 1070080 times.
2675200 for (unsigned x = 0; x < w; x++) {
476 1605120 t += ((mask[xm >> xmshf] >> ((~xm & xmmod) << l2depth)) & mbits)
477 1605120 * mmult;
478 1605120 xm++;
479 }
480 1070080 mask += mask_linesize;
481 }
482 1070080 alpha = (t >> shift) * alpha;
483 1070080 AV_WL16(dst, ((0x10001 - alpha) * value + alpha * src) >> 16);
484 1070080 }
485
486 8509504 static void blend_pixel(uint8_t *dst, unsigned src, unsigned alpha,
487 const uint8_t *mask, int mask_linesize, int l2depth,
488 unsigned w, unsigned h, unsigned shift, unsigned xm0)
489 {
490 8509504 unsigned t = 0;
491 8509504 unsigned xmshf = 3 - l2depth;
492 8509504 unsigned xmmod = 7 >> l2depth;
493 8509504 unsigned mbits = (1 << (1 << l2depth)) - 1;
494 8509504 unsigned mmult = 255 / mbits;
495
496
2/2
✓ Branch 0 taken 10136064 times.
✓ Branch 1 taken 8509504 times.
18645568 for (unsigned y = 0; y < h; y++) {
497 10136064 unsigned xm = xm0;
498
2/2
✓ Branch 0 taken 13471488 times.
✓ Branch 1 taken 10136064 times.
23607552 for (unsigned x = 0; x < w; x++) {
499 13471488 t += ((mask[xm >> xmshf] >> ((~xm & xmmod) << l2depth)) & mbits)
500 13471488 * mmult;
501 13471488 xm++;
502 }
503 10136064 mask += mask_linesize;
504 }
505 8509504 alpha = (t >> shift) * alpha;
506 8509504 *dst = ((0x1010101 - alpha) * *dst + alpha * src) >> 24;
507 8509504 }
508
509 163680 static void blend_line_hv16(uint8_t *dst, int dst_delta,
510 unsigned src, unsigned alpha,
511 const uint8_t *mask, int mask_linesize, int l2depth, int w,
512 unsigned hsub, unsigned vsub,
513 int xm, int left, int right, int hband)
514 {
515
516
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 163680 times.
163680 if (left) {
517 blend_pixel16(dst, src, alpha, mask, mask_linesize, l2depth,
518 left, hband, hsub + vsub, xm);
519 dst += dst_delta;
520 xm += left;
521 }
522
2/2
✓ Branch 0 taken 1070080 times.
✓ Branch 1 taken 163680 times.
1233760 for (int x = 0; x < w; x++) {
523 1070080 blend_pixel16(dst, src, alpha, mask, mask_linesize, l2depth,
524 1070080 1 << hsub, hband, hsub + vsub, xm);
525 1070080 dst += dst_delta;
526 1070080 xm += 1 << hsub;
527 }
528
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 163680 times.
163680 if (right)
529 blend_pixel16(dst, src, alpha, mask, mask_linesize, l2depth,
530 right, hband, hsub + vsub, xm);
531 163680 }
532
533 1038200 static void blend_line_hv(uint8_t *dst, int dst_delta,
534 unsigned src, unsigned alpha,
535 const uint8_t *mask, int mask_linesize, int l2depth, int w,
536 unsigned hsub, unsigned vsub,
537 int xm, int left, int right, int hband)
538 {
539
540
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1038200 times.
1038200 if (left) {
541 blend_pixel(dst, src, alpha, mask, mask_linesize, l2depth,
542 left, hband, hsub + vsub, xm);
543 dst += dst_delta;
544 xm += left;
545 }
546
2/2
✓ Branch 0 taken 8509504 times.
✓ Branch 1 taken 1038200 times.
9547704 for (int x = 0; x < w; x++) {
547 8509504 blend_pixel(dst, src, alpha, mask, mask_linesize, l2depth,
548 8509504 1 << hsub, hband, hsub + vsub, xm);
549 8509504 dst += dst_delta;
550 8509504 xm += 1 << hsub;
551 }
552
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1038200 times.
1038200 if (right)
553 blend_pixel(dst, src, alpha, mask, mask_linesize, l2depth,
554 right, hband, hsub + vsub, xm);
555 1038200 }
556
557 31979 void ff_blend_mask(FFDrawContext *draw, FFDrawColor *color,
558 uint8_t *dst[], int dst_linesize[], int dst_w, int dst_h,
559 const uint8_t *mask, int mask_linesize, int mask_w, int mask_h,
560 int l2depth, unsigned endianness, int x0, int y0)
561 {
562 unsigned alpha, nb_planes, nb_comp;
563 int xm0, ym0, w_sub, h_sub, x_sub, y_sub, left, right, top, bottom;
564 uint8_t *p;
565 const uint8_t *m;
566
567 63958 nb_comp = draw->desc->nb_components -
568
3/4
✓ Branch 0 taken 2434 times.
✓ Branch 1 taken 29545 times.
✓ Branch 2 taken 2434 times.
✗ Branch 3 not taken.
31979 !!(draw->desc->flags & AV_PIX_FMT_FLAG_ALPHA && !(draw->flags & FF_DRAW_PROCESS_ALPHA));
569
570 31979 clip_interval(dst_w, &x0, &mask_w, &xm0);
571 31979 clip_interval(dst_h, &y0, &mask_h, &ym0);
572 31979 mask += ym0 * mask_linesize;
573
3/6
✓ Branch 0 taken 31979 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 31979 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 31979 times.
31979 if (mask_w <= 0 || mask_h <= 0 || !color->rgba[3])
574 return;
575
2/2
✓ Branch 0 taken 28569 times.
✓ Branch 1 taken 3410 times.
31979 if (draw->desc->comp[0].depth <= 8) {
576 /* alpha is in the [ 0 ; 0x10203 ] range,
577 alpha * mask is in the [ 0 ; 0x1010101 - 4 ] range */
578 28569 alpha = (0x10307 * color->rgba[3] + 0x3) >> 8;
579 } else {
580 3410 alpha = (0x101 * color->rgba[3] + 0x2) >> 8;
581 }
582
3/4
✓ Branch 0 taken 2434 times.
✓ Branch 1 taken 29545 times.
✓ Branch 2 taken 2434 times.
✗ Branch 3 not taken.
31979 nb_planes = draw->nb_planes - !!(draw->desc->flags & AV_PIX_FMT_FLAG_ALPHA && !(draw->flags & FF_DRAW_PROCESS_ALPHA));
583 31979 nb_planes += !nb_planes;
584
2/2
✓ Branch 0 taken 85829 times.
✓ Branch 1 taken 31979 times.
117808 for (unsigned plane = 0; plane < nb_planes; plane++) {
585 85829 uint8_t *p0 = pointer_at(draw, dst, dst_linesize, plane, x0, y0);
586 85829 w_sub = mask_w;
587 85829 h_sub = mask_h;
588 85829 x_sub = x0;
589 85829 y_sub = y0;
590 85829 subsampling_bounds(draw->hsub[plane], &x_sub, &w_sub, &left, &right);
591 85829 subsampling_bounds(draw->vsub[plane], &y_sub, &h_sub, &top, &bottom);
592
2/2
✓ Branch 0 taken 257487 times.
✓ Branch 1 taken 85829 times.
343316 for (unsigned comp = 0; comp < nb_comp; comp++) {
593 257487 const int depth = draw->desc->comp[comp].depth;
594 257487 const int offset = draw->desc->comp[comp].offset;
595 257487 const int index = offset / ((depth + 7) / 8);
596
597
2/2
✓ Branch 0 taken 161550 times.
✓ Branch 1 taken 95937 times.
257487 if (draw->desc->comp[comp].plane != plane)
598 161550 continue;
599 95937 p = p0 + offset;
600 95937 m = mask;
601
2/2
✓ Branch 0 taken 216 times.
✓ Branch 1 taken 95721 times.
95937 if (top) {
602
1/2
✓ Branch 0 taken 216 times.
✗ Branch 1 not taken.
216 if (depth <= 8) {
603 216 blend_line_hv(p, draw->pixelstep[plane],
604 216 color->comp[plane].u8[index], alpha,
605 m, mask_linesize, l2depth, w_sub,
606 216 draw->hsub[plane], draw->vsub[plane],
607 xm0, left, right, top);
608 } else {
609 blend_line_hv16(p, draw->pixelstep[plane],
610 color->comp[plane].u16[index], alpha,
611 m, mask_linesize, l2depth, w_sub,
612 draw->hsub[plane], draw->vsub[plane],
613 xm0, left, right, top);
614 }
615 216 p += dst_linesize[plane];
616 216 m += top * mask_linesize;
617 }
618
2/2
✓ Branch 0 taken 85707 times.
✓ Branch 1 taken 10230 times.
95937 if (depth <= 8) {
619
2/2
✓ Branch 0 taken 1037768 times.
✓ Branch 1 taken 85707 times.
1123475 for (int y = 0; y < h_sub; y++) {
620 1037768 blend_line_hv(p, draw->pixelstep[plane],
621 1037768 color->comp[plane].u8[index], alpha,
622 m, mask_linesize, l2depth, w_sub,
623 1037768 draw->hsub[plane], draw->vsub[plane],
624 1037768 xm0, left, right, 1 << draw->vsub[plane]);
625 1037768 p += dst_linesize[plane];
626 1037768 m += mask_linesize << draw->vsub[plane];
627 }
628 } else {
629
2/2
✓ Branch 0 taken 163680 times.
✓ Branch 1 taken 10230 times.
173910 for (int y = 0; y < h_sub; y++) {
630 163680 blend_line_hv16(p, draw->pixelstep[plane],
631 163680 color->comp[plane].u16[index], alpha,
632 m, mask_linesize, l2depth, w_sub,
633 163680 draw->hsub[plane], draw->vsub[plane],
634 163680 xm0, left, right, 1 << draw->vsub[plane]);
635 163680 p += dst_linesize[plane];
636 163680 m += mask_linesize << draw->vsub[plane];
637 }
638 }
639
2/2
✓ Branch 0 taken 216 times.
✓ Branch 1 taken 95721 times.
95937 if (bottom) {
640
1/2
✓ Branch 0 taken 216 times.
✗ Branch 1 not taken.
216 if (depth <= 8) {
641 216 blend_line_hv(p, draw->pixelstep[plane],
642 216 color->comp[plane].u8[index], alpha,
643 m, mask_linesize, l2depth, w_sub,
644 216 draw->hsub[plane], draw->vsub[plane],
645 xm0, left, right, bottom);
646 } else {
647 blend_line_hv16(p, draw->pixelstep[plane],
648 color->comp[plane].u16[index], alpha,
649 m, mask_linesize, l2depth, w_sub,
650 draw->hsub[plane], draw->vsub[plane],
651 xm0, left, right, bottom);
652 }
653 }
654 }
655 }
656 }
657
658 243091 int ff_draw_round_to_sub(FFDrawContext *draw, int sub_dir, int round_dir,
659 int value)
660 {
661
2/2
✓ Branch 0 taken 217980 times.
✓ Branch 1 taken 25111 times.
243091 unsigned shift = sub_dir ? draw->vsub_max : draw->hsub_max;
662
663
2/2
✓ Branch 0 taken 105742 times.
✓ Branch 1 taken 137349 times.
243091 if (!shift)
664 105742 return value;
665
2/2
✓ Branch 0 taken 114861 times.
✓ Branch 1 taken 22488 times.
137349 if (round_dir >= 0)
666
2/2
✓ Branch 0 taken 3761 times.
✓ Branch 1 taken 111100 times.
114861 value += round_dir ? (1 << shift) - 1 : 1 << (shift - 1);
667 137349 return (value >> shift) << shift;
668 }
669
670 185 AVFilterFormats *ff_draw_supported_pixel_formats(unsigned flags)
671 {
672 FFDrawContext draw;
673 185 AVFilterFormats *fmts = NULL;
674 int ret;
675
676
2/2
✓ Branch 1 taken 49395 times.
✓ Branch 2 taken 185 times.
49580 for (enum AVPixelFormat i = 0; av_pix_fmt_desc_get(i); i++)
677
3/4
✓ Branch 1 taken 17945 times.
✓ Branch 2 taken 31450 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 17945 times.
67340 if (ff_draw_init(&draw, i, flags) >= 0 &&
678 17945 (ret = ff_add_format(&fmts, i)) < 0)
679 return NULL;
680 185 return fmts;
681 }
682