FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavfilter/vf_signalstats.c
Date: 2025-01-20 09:27:23
Exec Total Coverage
Lines: 282 486 58.0%
Functions: 8 17 47.1%
Branches: 119 294 40.5%

Line Branch Exec Source
1 /*
2 * Copyright (c) 2010 Mark Heath mjpeg0 @ silicontrip dot org
3 * Copyright (c) 2014 Clément Bœsch
4 * Copyright (c) 2014 Dave Rice @dericed
5 *
6 * This file is part of FFmpeg.
7 *
8 * FFmpeg is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
12 *
13 * FFmpeg is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with FFmpeg; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 */
22
23 #include "libavutil/intreadwrite.h"
24 #include "libavutil/mem.h"
25 #include "libavutil/opt.h"
26 #include "libavutil/pixdesc.h"
27 #include "filters.h"
28
29 enum FilterMode {
30 FILTER_NONE = -1,
31 FILTER_TOUT,
32 FILTER_VREP,
33 FILTER_BRNG,
34 FILT_NUMB
35 };
36
37 typedef struct SignalstatsContext {
38 const AVClass *class;
39 int chromah; // height of chroma plane
40 int chromaw; // width of chroma plane
41 int hsub; // horizontal subsampling
42 int vsub; // vertical subsampling
43 int depth; // pixel depth
44 int fs; // pixel count per frame
45 int cfs; // pixel count per frame of chroma planes
46 int outfilter; // FilterMode
47 int filters;
48 AVFrame *frame_prev;
49 uint8_t rgba_color[4];
50 int yuv_color[3];
51 int nb_jobs;
52 int *jobs_rets;
53
54 int maxsize; // history stats array size
55 int *histy, *histu, *histv, *histsat;
56
57 AVFrame *frame_sat;
58 AVFrame *frame_hue;
59 } SignalstatsContext;
60
61 typedef struct ThreadData {
62 const AVFrame *in;
63 AVFrame *out;
64 } ThreadData;
65
66 typedef struct ThreadDataHueSatMetrics {
67 const AVFrame *src;
68 AVFrame *dst_sat, *dst_hue;
69 } ThreadDataHueSatMetrics;
70
71 #define OFFSET(x) offsetof(SignalstatsContext, x)
72 #define FLAGS AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_VIDEO_PARAM
73
74 static const AVOption signalstats_options[] = {
75 {"stat", "set statistics filters", OFFSET(filters), AV_OPT_TYPE_FLAGS, {.i64=0}, 0, INT_MAX, FLAGS, .unit = "filters"},
76 {"tout", "analyze pixels for temporal outliers", 0, AV_OPT_TYPE_CONST, {.i64=1<<FILTER_TOUT}, 0, 0, FLAGS, .unit = "filters"},
77 {"vrep", "analyze video lines for vertical line repetition", 0, AV_OPT_TYPE_CONST, {.i64=1<<FILTER_VREP}, 0, 0, FLAGS, .unit = "filters"},
78 {"brng", "analyze for pixels outside of broadcast range", 0, AV_OPT_TYPE_CONST, {.i64=1<<FILTER_BRNG}, 0, 0, FLAGS, .unit = "filters"},
79 {"out", "set video filter", OFFSET(outfilter), AV_OPT_TYPE_INT, {.i64=FILTER_NONE}, -1, FILT_NUMB-1, FLAGS, .unit = "out"},
80 {"tout", "highlight pixels that depict temporal outliers", 0, AV_OPT_TYPE_CONST, {.i64=FILTER_TOUT}, 0, 0, FLAGS, .unit = "out"},
81 {"vrep", "highlight video lines that depict vertical line repetition", 0, AV_OPT_TYPE_CONST, {.i64=FILTER_VREP}, 0, 0, FLAGS, .unit = "out"},
82 {"brng", "highlight pixels that are outside of broadcast range", 0, AV_OPT_TYPE_CONST, {.i64=FILTER_BRNG}, 0, 0, FLAGS, .unit = "out"},
83 {"c", "set highlight color", OFFSET(rgba_color), AV_OPT_TYPE_COLOR, {.str="yellow"}, .flags=FLAGS},
84 {"color", "set highlight color", OFFSET(rgba_color), AV_OPT_TYPE_COLOR, {.str="yellow"}, .flags=FLAGS},
85 {NULL}
86 };
87
88 AVFILTER_DEFINE_CLASS(signalstats);
89
90 2 static av_cold int init(AVFilterContext *ctx)
91 {
92 uint8_t r, g, b;
93 2 SignalstatsContext *s = ctx->priv;
94
95
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
2 if (s->outfilter != FILTER_NONE)
96 s->filters |= 1 << s->outfilter;
97
98 2 r = s->rgba_color[0];
99 2 g = s->rgba_color[1];
100 2 b = s->rgba_color[2];
101 2 s->yuv_color[0] = (( 66*r + 129*g + 25*b + (1<<7)) >> 8) + 16;
102 2 s->yuv_color[1] = ((-38*r + -74*g + 112*b + (1<<7)) >> 8) + 128;
103 2 s->yuv_color[2] = ((112*r + -94*g + -18*b + (1<<7)) >> 8) + 128;
104 2 return 0;
105 }
106
107 2 static av_cold void uninit(AVFilterContext *ctx)
108 {
109 2 SignalstatsContext *s = ctx->priv;
110 2 av_frame_free(&s->frame_prev);
111 2 av_frame_free(&s->frame_sat);
112 2 av_frame_free(&s->frame_hue);
113 2 av_freep(&s->jobs_rets);
114 2 av_freep(&s->histy);
115 2 av_freep(&s->histu);
116 2 av_freep(&s->histv);
117 2 av_freep(&s->histsat);
118 2 }
119
120 // TODO: add more
121 static const enum AVPixelFormat pix_fmts[] = {
122 AV_PIX_FMT_YUV444P, AV_PIX_FMT_YUV422P, AV_PIX_FMT_YUV420P, AV_PIX_FMT_YUV411P,
123 AV_PIX_FMT_YUV440P,
124 AV_PIX_FMT_YUVJ422P, AV_PIX_FMT_YUVJ444P, AV_PIX_FMT_YUVJ420P, AV_PIX_FMT_YUVJ411P,
125 AV_PIX_FMT_YUVJ440P,
126 AV_PIX_FMT_YUV444P9, AV_PIX_FMT_YUV422P9, AV_PIX_FMT_YUV420P9,
127 AV_PIX_FMT_YUV444P10, AV_PIX_FMT_YUV422P10, AV_PIX_FMT_YUV420P10,
128 AV_PIX_FMT_YUV440P10,
129 AV_PIX_FMT_YUV444P12, AV_PIX_FMT_YUV422P12, AV_PIX_FMT_YUV420P12,
130 AV_PIX_FMT_YUV440P12,
131 AV_PIX_FMT_YUV444P14, AV_PIX_FMT_YUV422P14, AV_PIX_FMT_YUV420P14,
132 AV_PIX_FMT_YUV444P16, AV_PIX_FMT_YUV422P16, AV_PIX_FMT_YUV420P16,
133 AV_PIX_FMT_NONE
134 };
135
136 4 static AVFrame *alloc_frame(enum AVPixelFormat pixfmt, int w, int h)
137 {
138 4 AVFrame *frame = av_frame_alloc();
139
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
4 if (!frame)
140 return NULL;
141
142 4 frame->format = pixfmt;
143 4 frame->width = w;
144 4 frame->height = h;
145
146
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 4 times.
4 if (av_frame_get_buffer(frame, 0) < 0) {
147 av_frame_free(&frame);
148 return NULL;
149 }
150
151 4 return frame;
152 }
153
154 2 static int config_output(AVFilterLink *outlink)
155 {
156 2 AVFilterContext *ctx = outlink->src;
157 2 SignalstatsContext *s = ctx->priv;
158 2 AVFilterLink *inlink = outlink->src->inputs[0];
159 2 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(outlink->format);
160 2 s->hsub = desc->log2_chroma_w;
161 2 s->vsub = desc->log2_chroma_h;
162 2 s->depth = desc->comp[0].depth;
163 2 s->maxsize = 1 << s->depth;
164 2 s->histy = av_malloc_array(s->maxsize, sizeof(*s->histy));
165 2 s->histu = av_malloc_array(s->maxsize, sizeof(*s->histu));
166 2 s->histv = av_malloc_array(s->maxsize, sizeof(*s->histv));
167 2 s->histsat = av_malloc_array(s->maxsize, sizeof(*s->histsat));
168
169
4/8
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 2 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 2 times.
2 if (!s->histy || !s->histu || !s->histv || !s->histsat)
170 return AVERROR(ENOMEM);
171
172 2 outlink->w = inlink->w;
173 2 outlink->h = inlink->h;
174
175 2 s->chromaw = AV_CEIL_RSHIFT(inlink->w, s->hsub);
176 2 s->chromah = AV_CEIL_RSHIFT(inlink->h, s->vsub);
177
178 2 s->fs = inlink->w * inlink->h;
179 2 s->cfs = s->chromaw * s->chromah;
180
181
3/6
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
✓ Branch 4 taken 2 times.
✗ Branch 5 not taken.
2 s->nb_jobs = FFMAX(1, FFMIN(inlink->h, ff_filter_get_nb_threads(ctx)));
182 2 s->jobs_rets = av_malloc_array(s->nb_jobs, sizeof(*s->jobs_rets));
183
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
2 if (!s->jobs_rets)
184 return AVERROR(ENOMEM);
185
186
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
2 s->frame_sat = alloc_frame(s->depth > 8 ? AV_PIX_FMT_GRAY16 : AV_PIX_FMT_GRAY8, inlink->w, inlink->h);
187 2 s->frame_hue = alloc_frame(AV_PIX_FMT_GRAY16, inlink->w, inlink->h);
188
2/4
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
2 if (!s->frame_sat || !s->frame_hue)
189 return AVERROR(ENOMEM);
190
191 2 return 0;
192 }
193
194 static void burn_frame8(const SignalstatsContext *s, AVFrame *f, int x, int y)
195 {
196 const int chromax = x >> s->hsub;
197 const int chromay = y >> s->vsub;
198 f->data[0][y * f->linesize[0] + x] = s->yuv_color[0];
199 f->data[1][chromay * f->linesize[1] + chromax] = s->yuv_color[1];
200 f->data[2][chromay * f->linesize[2] + chromax] = s->yuv_color[2];
201 }
202
203 static void burn_frame16(const SignalstatsContext *s, AVFrame *f, int x, int y)
204 {
205 const int chromax = x >> s->hsub;
206 const int chromay = y >> s->vsub;
207 const int mult = 1 << (s->depth - 8);
208 AV_WN16(f->data[0] + y * f->linesize[0] + x * 2, s->yuv_color[0] * mult);
209 AV_WN16(f->data[1] + chromay * f->linesize[1] + chromax * 2, s->yuv_color[1] * mult);
210 AV_WN16(f->data[2] + chromay * f->linesize[2] + chromax * 2, s->yuv_color[2] * mult);
211 }
212
213 static int filter8_brng(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
214 {
215 ThreadData *td = arg;
216 const SignalstatsContext *s = ctx->priv;
217 const AVFrame *in = td->in;
218 AVFrame *out = td->out;
219 const int w = in->width;
220 const int h = in->height;
221 const int slice_start = (h * jobnr ) / nb_jobs;
222 const int slice_end = (h * (jobnr+1)) / nb_jobs;
223 int x, y, score = 0;
224
225 for (y = slice_start; y < slice_end; y++) {
226 const int yc = y >> s->vsub;
227 const uint8_t *pluma = &in->data[0][y * in->linesize[0]];
228 const uint8_t *pchromau = &in->data[1][yc * in->linesize[1]];
229 const uint8_t *pchromav = &in->data[2][yc * in->linesize[2]];
230
231 for (x = 0; x < w; x++) {
232 const int xc = x >> s->hsub;
233 const int luma = pluma[x];
234 const int chromau = pchromau[xc];
235 const int chromav = pchromav[xc];
236 const int filt = luma < 16 || luma > 235 ||
237 chromau < 16 || chromau > 240 ||
238 chromav < 16 || chromav > 240;
239 score += filt;
240 if (out && filt)
241 burn_frame8(s, out, x, y);
242 }
243 }
244 return score;
245 }
246
247 static int filter16_brng(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
248 {
249 ThreadData *td = arg;
250 const SignalstatsContext *s = ctx->priv;
251 const AVFrame *in = td->in;
252 AVFrame *out = td->out;
253 const int mult = 1 << (s->depth - 8);
254 const int w = in->width;
255 const int h = in->height;
256 const int slice_start = (h * jobnr ) / nb_jobs;
257 const int slice_end = (h * (jobnr+1)) / nb_jobs;
258 int x, y, score = 0;
259
260 for (y = slice_start; y < slice_end; y++) {
261 const int yc = y >> s->vsub;
262 const uint16_t *pluma = (uint16_t *)&in->data[0][y * in->linesize[0]];
263 const uint16_t *pchromau = (uint16_t *)&in->data[1][yc * in->linesize[1]];
264 const uint16_t *pchromav = (uint16_t *)&in->data[2][yc * in->linesize[2]];
265
266 for (x = 0; x < w; x++) {
267 const int xc = x >> s->hsub;
268 const int luma = pluma[x];
269 const int chromau = pchromau[xc];
270 const int chromav = pchromav[xc];
271 const int filt = luma < 16 * mult || luma > 235 * mult ||
272 chromau < 16 * mult || chromau > 240 * mult ||
273 chromav < 16 * mult || chromav > 240 * mult;
274 score += filt;
275 if (out && filt)
276 burn_frame16(s, out, x, y);
277 }
278 }
279 return score;
280 }
281
282 static int filter_tout_outlier(uint8_t x, uint8_t y, uint8_t z)
283 {
284 return ((abs(x - y) + abs (z - y)) / 2) - abs(z - x) > 4; // make 4 configurable?
285 }
286
287 static int filter8_tout(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
288 {
289 ThreadData *td = arg;
290 const SignalstatsContext *s = ctx->priv;
291 const AVFrame *in = td->in;
292 AVFrame *out = td->out;
293 const int w = in->width;
294 const int h = in->height;
295 const int slice_start = (h * jobnr ) / nb_jobs;
296 const int slice_end = (h * (jobnr+1)) / nb_jobs;
297 const uint8_t *p = in->data[0];
298 int lw = in->linesize[0];
299 int x, y, score = 0, filt;
300
301 for (y = slice_start; y < slice_end; y++) {
302
303 if (y - 1 < 0 || y + 1 >= h)
304 continue;
305
306 // detect two pixels above and below (to eliminate interlace artefacts)
307 // should check that video format is infact interlaced.
308
309 #define FILTER(i, j) \
310 filter_tout_outlier(p[(y-j) * lw + x + i], \
311 p[ y * lw + x + i], \
312 p[(y+j) * lw + x + i])
313
314 #define FILTER3(j) (FILTER(-1, j) && FILTER(0, j) && FILTER(1, j))
315
316 if (y - 2 >= 0 && y + 2 < h) {
317 for (x = 1; x < w - 1; x++) {
318 filt = FILTER3(2) && FILTER3(1);
319 score += filt;
320 if (filt && out)
321 burn_frame8(s, out, x, y);
322 }
323 } else {
324 for (x = 1; x < w - 1; x++) {
325 filt = FILTER3(1);
326 score += filt;
327 if (filt && out)
328 burn_frame8(s, out, x, y);
329 }
330 }
331 }
332 return score;
333 }
334
335 static int filter16_tout(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
336 {
337 ThreadData *td = arg;
338 const SignalstatsContext *s = ctx->priv;
339 const AVFrame *in = td->in;
340 AVFrame *out = td->out;
341 const int w = in->width;
342 const int h = in->height;
343 const int slice_start = (h * jobnr ) / nb_jobs;
344 const int slice_end = (h * (jobnr+1)) / nb_jobs;
345 const uint16_t *p = (uint16_t *)in->data[0];
346 int lw = in->linesize[0] / 2;
347 int x, y, score = 0, filt;
348
349 for (y = slice_start; y < slice_end; y++) {
350
351 if (y - 1 < 0 || y + 1 >= h)
352 continue;
353
354 // detect two pixels above and below (to eliminate interlace artefacts)
355 // should check that video format is infact interlaced.
356
357 if (y - 2 >= 0 && y + 2 < h) {
358 for (x = 1; x < w - 1; x++) {
359 filt = FILTER3(2) && FILTER3(1);
360 score += filt;
361 if (filt && out)
362 burn_frame16(s, out, x, y);
363 }
364 } else {
365 for (x = 1; x < w - 1; x++) {
366 filt = FILTER3(1);
367 score += filt;
368 if (filt && out)
369 burn_frame16(s, out, x, y);
370 }
371 }
372 }
373 return score;
374 }
375
376 #define VREP_START 4
377
378 static int filter8_vrep(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
379 {
380 ThreadData *td = arg;
381 const SignalstatsContext *s = ctx->priv;
382 const AVFrame *in = td->in;
383 AVFrame *out = td->out;
384 const int w = in->width;
385 const int h = in->height;
386 const int slice_start = (h * jobnr ) / nb_jobs;
387 const int slice_end = (h * (jobnr+1)) / nb_jobs;
388 const uint8_t *p = in->data[0];
389 const int lw = in->linesize[0];
390 int x, y, score = 0;
391
392 for (y = slice_start; y < slice_end; y++) {
393 const int y2lw = (y - VREP_START) * lw;
394 const int ylw = y * lw;
395 int filt, totdiff = 0;
396
397 if (y < VREP_START)
398 continue;
399
400 for (x = 0; x < w; x++)
401 totdiff += abs(p[y2lw + x] - p[ylw + x]);
402 filt = totdiff < w;
403
404 score += filt;
405 if (filt && out)
406 for (x = 0; x < w; x++)
407 burn_frame8(s, out, x, y);
408 }
409 return score * w;
410 }
411
412 static int filter16_vrep(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
413 {
414 ThreadData *td = arg;
415 const SignalstatsContext *s = ctx->priv;
416 const AVFrame *in = td->in;
417 AVFrame *out = td->out;
418 const int w = in->width;
419 const int h = in->height;
420 const int slice_start = (h * jobnr ) / nb_jobs;
421 const int slice_end = (h * (jobnr+1)) / nb_jobs;
422 const uint16_t *p = (uint16_t *)in->data[0];
423 const int lw = in->linesize[0] / 2;
424 int x, y, score = 0;
425
426 for (y = slice_start; y < slice_end; y++) {
427 const int y2lw = (y - VREP_START) * lw;
428 const int ylw = y * lw;
429 int64_t totdiff = 0;
430 int filt;
431
432 if (y < VREP_START)
433 continue;
434
435 for (x = 0; x < w; x++)
436 totdiff += abs(p[y2lw + x] - p[ylw + x]);
437 filt = totdiff < w;
438
439 score += filt;
440 if (filt && out)
441 for (x = 0; x < w; x++)
442 burn_frame16(s, out, x, y);
443 }
444 return score * w;
445 }
446
447 static const struct {
448 const char *name;
449 int (*process8)(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs);
450 int (*process16)(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs);
451 } filters_def[] = {
452 {"TOUT", filter8_tout, filter16_tout},
453 {"VREP", filter8_vrep, filter16_vrep},
454 {"BRNG", filter8_brng, filter16_brng},
455 {NULL}
456 };
457
458 9 static int compute_sat_hue_metrics8(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
459 {
460 int i, j;
461 9 ThreadDataHueSatMetrics *td = arg;
462 9 const SignalstatsContext *s = ctx->priv;
463 9 const AVFrame *src = td->src;
464 9 AVFrame *dst_sat = td->dst_sat;
465 9 AVFrame *dst_hue = td->dst_hue;
466
467 9 const int slice_start = (s->chromah * jobnr ) / nb_jobs;
468 9 const int slice_end = (s->chromah * (jobnr+1)) / nb_jobs;
469
470 9 const int lsz_u = src->linesize[1];
471 9 const int lsz_v = src->linesize[2];
472 9 const uint8_t *p_u = src->data[1] + slice_start * lsz_u;
473 9 const uint8_t *p_v = src->data[2] + slice_start * lsz_v;
474
475 9 const int lsz_sat = dst_sat->linesize[0];
476 9 const int lsz_hue = dst_hue->linesize[0];
477 9 uint8_t *p_sat = dst_sat->data[0] + slice_start * lsz_sat;
478 9 uint8_t *p_hue = dst_hue->data[0] + slice_start * lsz_hue;
479
480
2/2
✓ Branch 0 taken 120 times.
✓ Branch 1 taken 9 times.
129 for (j = slice_start; j < slice_end; j++) {
481
2/2
✓ Branch 0 taken 19200 times.
✓ Branch 1 taken 120 times.
19320 for (i = 0; i < s->chromaw; i++) {
482 19200 const int yuvu = p_u[i];
483 19200 const int yuvv = p_v[i];
484 19200 p_sat[i] = hypotf(yuvu - 128, yuvv - 128); // int or round?
485 19200 ((int16_t*)p_hue)[i] = fmodf(floorf((180.f / M_PI) * atan2f(yuvu-128, yuvv-128) + 180.f), 360.f);
486 }
487 120 p_u += lsz_u;
488 120 p_v += lsz_v;
489 120 p_sat += lsz_sat;
490 120 p_hue += lsz_hue;
491 }
492
493 9 return 0;
494 }
495
496 9 static int compute_sat_hue_metrics16(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
497 {
498 int i, j;
499 9 ThreadDataHueSatMetrics *td = arg;
500 9 const SignalstatsContext *s = ctx->priv;
501 9 const AVFrame *src = td->src;
502 9 AVFrame *dst_sat = td->dst_sat;
503 9 AVFrame *dst_hue = td->dst_hue;
504 9 const int mid = 1 << (s->depth - 1);
505
506 9 const int slice_start = (s->chromah * jobnr ) / nb_jobs;
507 9 const int slice_end = (s->chromah * (jobnr+1)) / nb_jobs;
508
509 9 const int lsz_u = src->linesize[1] / 2;
510 9 const int lsz_v = src->linesize[2] / 2;
511 9 const uint16_t *p_u = (uint16_t*)src->data[1] + slice_start * lsz_u;
512 9 const uint16_t *p_v = (uint16_t*)src->data[2] + slice_start * lsz_v;
513
514 9 const int lsz_sat = dst_sat->linesize[0] / 2;
515 9 const int lsz_hue = dst_hue->linesize[0] / 2;
516 9 uint16_t *p_sat = (uint16_t*)dst_sat->data[0] + slice_start * lsz_sat;
517 9 uint16_t *p_hue = (uint16_t*)dst_hue->data[0] + slice_start * lsz_hue;
518
519
2/2
✓ Branch 0 taken 120 times.
✓ Branch 1 taken 9 times.
129 for (j = slice_start; j < slice_end; j++) {
520
2/2
✓ Branch 0 taken 19200 times.
✓ Branch 1 taken 120 times.
19320 for (i = 0; i < s->chromaw; i++) {
521 19200 const int yuvu = p_u[i];
522 19200 const int yuvv = p_v[i];
523 19200 p_sat[i] = hypotf(yuvu - mid, yuvv - mid); // int or round?
524 19200 ((int16_t*)p_hue)[i] = fmodf(floorf((180.f / M_PI) * atan2f(yuvu-mid, yuvv-mid) + 180.f), 360.f);
525 }
526 120 p_u += lsz_u;
527 120 p_v += lsz_v;
528 120 p_sat += lsz_sat;
529 120 p_hue += lsz_hue;
530 }
531
532 9 return 0;
533 }
534
535 6 static unsigned compute_bit_depth(uint16_t mask)
536 {
537 6 return av_popcount(mask);
538 }
539
540 2 static int filter_frame(AVFilterLink *link, AVFrame *in)
541 {
542 2 AVFilterContext *ctx = link->dst;
543 2 SignalstatsContext *s = ctx->priv;
544 2 AVFilterLink *outlink = ctx->outputs[0];
545 2 AVFrame *out = in;
546 2 int w = 0, cw = 0, // in
547 2 pw = 0, cpw = 0; // prev
548 int fil;
549 char metabuf[128];
550 2 unsigned int *histy = s->histy,
551 2 *histu = s->histu,
552 2 *histv = s->histv,
553 2 histhue[360] = {0},
554 2 *histsat = s->histsat;
555 2 int miny = -1, minu = -1, minv = -1;
556 2 int maxy = -1, maxu = -1, maxv = -1;
557 2 int lowy = -1, lowu = -1, lowv = -1;
558 2 int highy = -1, highu = -1, highv = -1;
559 2 int minsat = -1, maxsat = -1, lowsat = -1, highsat = -1;
560 int lowp, highp, clowp, chighp;
561 int accy, accu, accv;
562 2 int accsat, acchue = 0;
563 int medhue, maxhue;
564 2 int64_t toty = 0, totu = 0, totv = 0, totsat=0;
565 2 int64_t tothue = 0;
566 2 int64_t dify = 0, difu = 0, difv = 0;
567 2 uint16_t masky = 0, masku = 0, maskv = 0;
568
569 2 int filtot[FILT_NUMB] = {0};
570 AVFrame *prev;
571 int ret;
572 2 AVFrame *sat = s->frame_sat;
573 2 AVFrame *hue = s->frame_hue;
574 2 const int hbd = s->depth > 8;
575 2 ThreadDataHueSatMetrics td_huesat = {
576 .src = in,
577 .dst_sat = sat,
578 .dst_hue = hue,
579 };
580
581
1/2
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
2 if (!s->frame_prev)
582 2 s->frame_prev = av_frame_clone(in);
583
584 2 prev = s->frame_prev;
585
586
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
2 if (s->outfilter != FILTER_NONE) {
587 out = av_frame_clone(in);
588 if (!out) {
589 av_frame_free(&in);
590 return AVERROR(ENOMEM);
591 }
592 ret = ff_inlink_make_frame_writable(link, &out);
593 if (ret < 0) {
594 av_frame_free(&out);
595 av_frame_free(&in);
596 return ret;
597 }
598 }
599
600
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
2 ff_filter_execute(ctx, hbd ? compute_sat_hue_metrics16
601 : compute_sat_hue_metrics8, &td_huesat,
602
1/2
✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
2 NULL, FFMIN(s->chromah, ff_filter_get_nb_threads(ctx)));
603
604 2 memset(s->histy, 0, s->maxsize * sizeof(*s->histy));
605 2 memset(s->histu, 0, s->maxsize * sizeof(*s->histu));
606 2 memset(s->histv, 0, s->maxsize * sizeof(*s->histv));
607 2 memset(s->histsat, 0, s->maxsize * sizeof(*s->histsat));
608
609
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
2 if (hbd) {
610 1 const uint16_t *p_sat = (uint16_t *)sat->data[0];
611 1 const uint16_t *p_hue = (uint16_t *)hue->data[0];
612 1 const int lsz_sat = sat->linesize[0] / 2;
613 1 const int lsz_hue = hue->linesize[0] / 2;
614 // Calculate luma histogram and difference with previous frame or field.
615
2/2
✓ Branch 0 taken 240 times.
✓ Branch 1 taken 1 times.
241 for (int j = 0; j < link->h; j++) {
616
2/2
✓ Branch 0 taken 76800 times.
✓ Branch 1 taken 240 times.
77040 for (int i = 0; i < link->w; i++) {
617 76800 const int yuv = AV_RN16(in->data[0] + w + i * 2);
618
619 76800 masky |= yuv;
620 76800 histy[yuv]++;
621 76800 dify += abs(yuv - (int)AV_RN16(prev->data[0] + pw + i * 2));
622 }
623 240 w += in->linesize[0];
624 240 pw += prev->linesize[0];
625 }
626
627 // Calculate chroma histogram and difference with previous frame or field.
628
2/2
✓ Branch 0 taken 120 times.
✓ Branch 1 taken 1 times.
121 for (int j = 0; j < s->chromah; j++) {
629
2/2
✓ Branch 0 taken 19200 times.
✓ Branch 1 taken 120 times.
19320 for (int i = 0; i < s->chromaw; i++) {
630 19200 const int yuvu = AV_RN16(in->data[1] + cw + i * 2);
631 19200 const int yuvv = AV_RN16(in->data[2] + cw + i * 2);
632
633 19200 masku |= yuvu;
634 19200 maskv |= yuvv;
635 19200 histu[yuvu]++;
636 19200 difu += abs(yuvu - (int)AV_RN16(prev->data[1] + cpw + i * 2));
637 19200 histv[yuvv]++;
638 19200 difv += abs(yuvv - (int)AV_RN16(prev->data[2] + cpw + i * 2));
639
640 19200 histsat[p_sat[i]]++;
641 19200 histhue[((int16_t*)p_hue)[i]]++;
642 }
643 120 cw += in->linesize[1];
644 120 cpw += prev->linesize[1];
645 120 p_sat += lsz_sat;
646 120 p_hue += lsz_hue;
647 }
648 } else {
649 1 const uint8_t *p_sat = sat->data[0];
650 1 const uint8_t *p_hue = hue->data[0];
651 1 const int lsz_sat = sat->linesize[0];
652 1 const int lsz_hue = hue->linesize[0];
653 // Calculate luma histogram and difference with previous frame or field.
654
2/2
✓ Branch 0 taken 240 times.
✓ Branch 1 taken 1 times.
241 for (int j = 0; j < link->h; j++) {
655
2/2
✓ Branch 0 taken 76800 times.
✓ Branch 1 taken 240 times.
77040 for (int i = 0; i < link->w; i++) {
656 76800 const int yuv = in->data[0][w + i];
657
658 76800 masky |= yuv;
659 76800 histy[yuv]++;
660 76800 dify += abs(yuv - prev->data[0][pw + i]);
661 }
662 240 w += in->linesize[0];
663 240 pw += prev->linesize[0];
664 }
665
666 // Calculate chroma histogram and difference with previous frame or field.
667
2/2
✓ Branch 0 taken 120 times.
✓ Branch 1 taken 1 times.
121 for (int j = 0; j < s->chromah; j++) {
668
2/2
✓ Branch 0 taken 19200 times.
✓ Branch 1 taken 120 times.
19320 for (int i = 0; i < s->chromaw; i++) {
669 19200 const int yuvu = in->data[1][cw+i];
670 19200 const int yuvv = in->data[2][cw+i];
671
672 19200 masku |= yuvu;
673 19200 maskv |= yuvv;
674 19200 histu[yuvu]++;
675 19200 difu += abs(yuvu - prev->data[1][cpw+i]);
676 19200 histv[yuvv]++;
677 19200 difv += abs(yuvv - prev->data[2][cpw+i]);
678
679 19200 histsat[p_sat[i]]++;
680 19200 histhue[((int16_t*)p_hue)[i]]++;
681 }
682 120 cw += in->linesize[1];
683 120 cpw += prev->linesize[1];
684 120 p_sat += lsz_sat;
685 120 p_hue += lsz_hue;
686 }
687 }
688
689
2/2
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 2 times.
8 for (fil = 0; fil < FILT_NUMB; fil ++) {
690
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (s->filters & 1<<fil) {
691 ThreadData td = {
692 .in = in,
693 .out = out != in && s->outfilter == fil ? out : NULL,
694 };
695 memset(s->jobs_rets, 0, s->nb_jobs * sizeof(*s->jobs_rets));
696 ff_filter_execute(ctx, hbd ? filters_def[fil].process16 : filters_def[fil].process8,
697 &td, s->jobs_rets, s->nb_jobs);
698 for (int i = 0; i < s->nb_jobs; i++)
699 filtot[fil] += s->jobs_rets[i];
700 }
701 }
702
703 // find low / high based on histogram percentile
704 // these only need to be calculated once.
705
706 2 lowp = lrint(s->fs * 10 / 100.);
707 2 highp = lrint(s->fs * 90 / 100.);
708 2 clowp = lrint(s->cfs * 10 / 100.);
709 2 chighp = lrint(s->cfs * 90 / 100.);
710
711 2 accy = accu = accv = accsat = 0;
712
2/2
✓ Branch 0 taken 1280 times.
✓ Branch 1 taken 2 times.
1282 for (fil = 0; fil < s->maxsize; fil++) {
713
4/4
✓ Branch 0 taken 1180 times.
✓ Branch 1 taken 100 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 1178 times.
1280 if (miny < 0 && histy[fil]) miny = fil;
714
4/4
✓ Branch 0 taken 644 times.
✓ Branch 1 taken 636 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 642 times.
1280 if (minu < 0 && histu[fil]) minu = fil;
715
4/4
✓ Branch 0 taken 644 times.
✓ Branch 1 taken 636 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 642 times.
1280 if (minv < 0 && histv[fil]) minv = fil;
716
4/4
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 1276 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 2 times.
1280 if (minsat < 0 && histsat[fil]) minsat = fil;
717
718
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 1278 times.
1280 if (histy[fil]) maxy = fil;
719
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 1278 times.
1280 if (histu[fil]) maxu = fil;
720
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 1278 times.
1280 if (histv[fil]) maxv = fil;
721
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 1278 times.
1280 if (histsat[fil]) maxsat = fil;
722
723 1280 toty += (uint64_t)histy[fil] * fil;
724 1280 totu += (uint64_t)histu[fil] * fil;
725 1280 totv += (uint64_t)histv[fil] * fil;
726 1280 totsat += (uint64_t)histsat[fil] * fil;
727
728 1280 accy += histy[fil];
729 1280 accu += histu[fil];
730 1280 accv += histv[fil];
731 1280 accsat += histsat[fil];
732
733
4/4
✓ Branch 0 taken 1180 times.
✓ Branch 1 taken 100 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 1178 times.
1280 if (lowy == -1 && accy >= lowp) lowy = fil;
734
4/4
✓ Branch 0 taken 644 times.
✓ Branch 1 taken 636 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 642 times.
1280 if (lowu == -1 && accu >= clowp) lowu = fil;
735
4/4
✓ Branch 0 taken 644 times.
✓ Branch 1 taken 636 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 642 times.
1280 if (lowv == -1 && accv >= clowp) lowv = fil;
736
4/4
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 1276 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 2 times.
1280 if (lowsat == -1 && accsat >= clowp) lowsat = fil;
737
738
4/4
✓ Branch 0 taken 1180 times.
✓ Branch 1 taken 100 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 1178 times.
1280 if (highy == -1 && accy >= highp) highy = fil;
739
4/4
✓ Branch 0 taken 644 times.
✓ Branch 1 taken 636 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 642 times.
1280 if (highu == -1 && accu >= chighp) highu = fil;
740
4/4
✓ Branch 0 taken 644 times.
✓ Branch 1 taken 636 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 642 times.
1280 if (highv == -1 && accv >= chighp) highv = fil;
741
4/4
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 1276 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 2 times.
1280 if (highsat == -1 && accsat >= chighp) highsat = fil;
742 }
743
744 2 maxhue = histhue[0];
745 2 medhue = -1;
746
2/2
✓ Branch 0 taken 720 times.
✓ Branch 1 taken 2 times.
722 for (fil = 0; fil < 360; fil++) {
747 720 tothue += (uint64_t)histhue[fil] * fil;
748 720 acchue += histhue[fil];
749
750
4/4
✓ Branch 0 taken 407 times.
✓ Branch 1 taken 313 times.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 405 times.
720 if (medhue == -1 && acchue > s->cfs / 2)
751 2 medhue = fil;
752
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 718 times.
720 if (histhue[fil] > maxhue) {
753 2 maxhue = histhue[fil];
754 }
755 }
756
757 2 av_frame_free(&s->frame_prev);
758 2 s->frame_prev = av_frame_clone(in);
759
760 #define SET_META(key, fmt, val) do { \
761 snprintf(metabuf, sizeof(metabuf), fmt, val); \
762 av_dict_set(&out->metadata, "lavfi.signalstats." key, metabuf, 0); \
763 } while (0)
764
765 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.YMIN", miny, 0);
766 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.YLOW", lowy, 0);
767 2 SET_META("YAVG", "%g", 1.0 * toty / s->fs);
768 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.YHIGH", highy, 0);
769 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.YMAX", maxy, 0);
770
771 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.UMIN", minu, 0);
772 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.ULOW", lowu, 0);
773 2 SET_META("UAVG", "%g", 1.0 * totu / s->cfs);
774 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.UHIGH", highu, 0);
775 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.UMAX", maxu, 0);
776
777 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.VMIN", minv, 0);
778 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.VLOW", lowv, 0);
779 2 SET_META("VAVG", "%g", 1.0 * totv / s->cfs);
780 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.VHIGH", highv, 0);
781 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.VMAX", maxv, 0);
782
783 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.SATMIN", minsat, 0);
784 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.SATLOW", lowsat, 0);
785 2 SET_META("SATAVG", "%g", 1.0 * totsat / s->cfs);
786 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.SATHIGH", highsat, 0);
787 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.SATMAX", maxsat, 0);
788
789 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.HUEMED", medhue, 0);
790 2 SET_META("HUEAVG", "%g", 1.0 * tothue / s->cfs);
791
792 2 SET_META("YDIF", "%g", 1.0 * dify / s->fs);
793 2 SET_META("UDIF", "%g", 1.0 * difu / s->cfs);
794 2 SET_META("VDIF", "%g", 1.0 * difv / s->cfs);
795
796 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.YBITDEPTH", compute_bit_depth(masky), 0);
797 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.UBITDEPTH", compute_bit_depth(masku), 0);
798 2 av_dict_set_int(&out->metadata, "lavfi.signalstats.VBITDEPTH", compute_bit_depth(maskv), 0);
799
800
2/2
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 2 times.
8 for (fil = 0; fil < FILT_NUMB; fil ++) {
801
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (s->filters & 1<<fil) {
802 char metaname[128];
803 snprintf(metabuf, sizeof(metabuf), "%g", 1.0 * filtot[fil] / s->fs);
804 snprintf(metaname, sizeof(metaname), "lavfi.signalstats.%s", filters_def[fil].name);
805 av_dict_set(&out->metadata, metaname, metabuf, 0);
806 }
807 }
808
809
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
2 if (in != out)
810 av_frame_free(&in);
811 2 return ff_filter_frame(outlink, out);
812 }
813
814 static const AVFilterPad signalstats_inputs[] = {
815 {
816 .name = "default",
817 .type = AVMEDIA_TYPE_VIDEO,
818 .filter_frame = filter_frame,
819 },
820 };
821
822 static const AVFilterPad signalstats_outputs[] = {
823 {
824 .name = "default",
825 .config_props = config_output,
826 .type = AVMEDIA_TYPE_VIDEO,
827 },
828 };
829
830 const FFFilter ff_vf_signalstats = {
831 .p.name = "signalstats",
832 .p.description = "Generate statistics from video analysis.",
833 .p.priv_class = &signalstats_class,
834 .p.flags = AVFILTER_FLAG_SLICE_THREADS,
835 .init = init,
836 .uninit = uninit,
837 .priv_size = sizeof(SignalstatsContext),
838 FILTER_INPUTS(signalstats_inputs),
839 FILTER_OUTPUTS(signalstats_outputs),
840 FILTER_PIXFMTS_ARRAY(pix_fmts),
841 };
842