FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavfilter/vf_scale.c
Date: 2026-06-16 12:54:33
Exec Total Coverage
Lines: 377 597 63.1%
Functions: 12 21 57.1%
Branches: 249 507 49.1%

Line Branch Exec Source
1 /*
2 * Copyright (c) 2007 Bobby Bingham
3 *
4 * This file is part of FFmpeg.
5 *
6 * FFmpeg is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * FFmpeg is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with FFmpeg; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21 /**
22 * @file
23 * scale video filter
24 */
25
26 #include <float.h>
27 #include <stdio.h>
28 #include <string.h>
29
30 #include "avfilter.h"
31 #include "filters.h"
32 #include "formats.h"
33 #include "framesync.h"
34 #include "libavutil/pixfmt.h"
35 #include "scale_eval.h"
36 #include "video.h"
37 #include "libavutil/eval.h"
38 #include "libavutil/imgutils_internal.h"
39 #include "libavutil/internal.h"
40 #include "libavutil/mem.h"
41 #include "libavutil/opt.h"
42 #include "libavutil/parseutils.h"
43 #include "libavutil/pixdesc.h"
44 #include "libswscale/swscale.h"
45
46 static const char *const var_names[] = {
47 "in_w", "iw",
48 "in_h", "ih",
49 "out_w", "ow",
50 "out_h", "oh",
51 "a",
52 "sar",
53 "dar",
54 "hsub",
55 "vsub",
56 "ohsub",
57 "ovsub",
58 "n",
59 "t",
60 "ref_w", "rw",
61 "ref_h", "rh",
62 "ref_a",
63 "ref_sar",
64 "ref_dar", "rdar",
65 "ref_hsub",
66 "ref_vsub",
67 "ref_n",
68 "ref_t",
69 "ref_pos",
70 /* Legacy variables for scale2ref */
71 "main_w",
72 "main_h",
73 "main_a",
74 "main_sar",
75 "main_dar", "mdar",
76 "main_hsub",
77 "main_vsub",
78 "main_n",
79 "main_t",
80 "main_pos",
81 NULL
82 };
83
84 enum var_name {
85 VAR_IN_W, VAR_IW,
86 VAR_IN_H, VAR_IH,
87 VAR_OUT_W, VAR_OW,
88 VAR_OUT_H, VAR_OH,
89 VAR_A,
90 VAR_SAR,
91 VAR_DAR,
92 VAR_HSUB,
93 VAR_VSUB,
94 VAR_OHSUB,
95 VAR_OVSUB,
96 VAR_N,
97 VAR_T,
98 VAR_REF_W, VAR_RW,
99 VAR_REF_H, VAR_RH,
100 VAR_REF_A,
101 VAR_REF_SAR,
102 VAR_REF_DAR, VAR_RDAR,
103 VAR_REF_HSUB,
104 VAR_REF_VSUB,
105 VAR_REF_N,
106 VAR_REF_T,
107 VAR_REF_POS,
108 VAR_S2R_MAIN_W,
109 VAR_S2R_MAIN_H,
110 VAR_S2R_MAIN_A,
111 VAR_S2R_MAIN_SAR,
112 VAR_S2R_MAIN_DAR, VAR_S2R_MDAR,
113 VAR_S2R_MAIN_HSUB,
114 VAR_S2R_MAIN_VSUB,
115 VAR_S2R_MAIN_N,
116 VAR_S2R_MAIN_T,
117 VAR_S2R_MAIN_POS,
118 VARS_NB
119 };
120
121 enum EvalMode {
122 EVAL_MODE_INIT,
123 EVAL_MODE_FRAME,
124 EVAL_MODE_NB
125 };
126
127 typedef struct ScaleContext {
128 const AVClass *class;
129 SwsContext *sws;
130 FFFrameSync fs;
131
132 /**
133 * New dimensions. Special values are:
134 * 0 = original width/height
135 * -1 = keep original aspect
136 * -N = try to keep aspect but make sure it is divisible by N
137 */
138 int w, h;
139 char *size_str;
140 double param[2]; // sws params
141
142 int hsub, vsub; ///< chroma subsampling
143 int slice_y; ///< top of current output slice
144 int interlaced;
145 int uses_ref;
146
147 char *w_expr; ///< width expression string
148 char *h_expr; ///< height expression string
149 AVExpr *w_pexpr;
150 AVExpr *h_pexpr;
151 double var_values[VARS_NB];
152
153 char *flags_str;
154
155 int in_color_matrix;
156 int out_color_matrix;
157 int in_primaries;
158 int out_primaries;
159 int in_transfer;
160 int out_transfer;
161 int in_range;
162 int out_range;
163
164 int in_chroma_loc;
165 int out_chroma_loc;
166 int out_h_chr_pos;
167 int out_v_chr_pos;
168 int in_h_chr_pos;
169 int in_v_chr_pos;
170
171 int force_original_aspect_ratio;
172 int force_divisible_by;
173 int reset_sar;
174
175 int eval_mode; ///< expression evaluation mode
176
177 } ScaleContext;
178
179 const FFFilter ff_vf_scale2ref;
180 #define IS_SCALE2REF(ctx) ((ctx)->filter == &ff_vf_scale2ref.p)
181
182 static int config_props(AVFilterLink *outlink);
183
184 22508 static int check_exprs(AVFilterContext *ctx)
185 {
186 22508 ScaleContext *scale = ctx->priv;
187 22508 unsigned vars_w[VARS_NB] = { 0 }, vars_h[VARS_NB] = { 0 };
188
189
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 22508 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
22508 if (!scale->w_pexpr && !scale->h_pexpr)
190 return AVERROR(EINVAL);
191
192
1/2
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
22508 if (scale->w_pexpr)
193 22508 av_expr_count_vars(scale->w_pexpr, vars_w, VARS_NB);
194
2/2
✓ Branch 0 taken 11273 times.
✓ Branch 1 taken 11235 times.
22508 if (scale->h_pexpr)
195 11273 av_expr_count_vars(scale->h_pexpr, vars_h, VARS_NB);
196
197
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 22508 times.
22508 if (vars_w[VAR_OUT_W] || vars_w[VAR_OW]) {
198 av_log(ctx, AV_LOG_ERROR, "Width expression cannot be self-referencing: '%s'.\n", scale->w_expr);
199 return AVERROR(EINVAL);
200 }
201
202
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 22508 times.
22508 if (vars_h[VAR_OUT_H] || vars_h[VAR_OH]) {
203 av_log(ctx, AV_LOG_ERROR, "Height expression cannot be self-referencing: '%s'.\n", scale->h_expr);
204 return AVERROR(EINVAL);
205 }
206
207
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 22508 times.
22508 if ((vars_w[VAR_OUT_H] || vars_w[VAR_OH]) &&
208 (vars_h[VAR_OUT_W] || vars_h[VAR_OW])) {
209 av_log(ctx, AV_LOG_WARNING, "Circular references detected for width '%s' and height '%s' - possibly invalid.\n", scale->w_expr, scale->h_expr);
210 }
211
212
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 if (vars_w[VAR_REF_W] || vars_h[VAR_REF_W] ||
213
3/4
✓ Branch 0 taken 22504 times.
✓ Branch 1 taken 4 times.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_RW] || vars_h[VAR_RW] ||
214
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_H] || vars_h[VAR_REF_H] ||
215
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_RH] || vars_h[VAR_RH] ||
216
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_A] || vars_h[VAR_REF_A] ||
217
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_SAR] || vars_h[VAR_REF_SAR] ||
218
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_DAR] || vars_h[VAR_REF_DAR] ||
219
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_RDAR] || vars_h[VAR_RDAR] ||
220
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_HSUB] || vars_h[VAR_REF_HSUB] ||
221
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_VSUB] || vars_h[VAR_REF_VSUB] ||
222
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_N] || vars_h[VAR_REF_N] ||
223
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22504 times.
✗ Branch 3 not taken.
22504 vars_w[VAR_REF_T] || vars_h[VAR_REF_T] ||
224
2/4
✓ Branch 0 taken 22504 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 22504 times.
22504 vars_w[VAR_REF_POS] || vars_h[VAR_REF_POS]) {
225 4 scale->uses_ref = 1;
226 }
227
228
1/2
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
22508 if (!IS_SCALE2REF(ctx) &&
229
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 (vars_w[VAR_S2R_MAIN_W] || vars_h[VAR_S2R_MAIN_W] ||
230
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_H] || vars_h[VAR_S2R_MAIN_H] ||
231
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_A] || vars_h[VAR_S2R_MAIN_A] ||
232
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_SAR] || vars_h[VAR_S2R_MAIN_SAR] ||
233
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_DAR] || vars_h[VAR_S2R_MAIN_DAR] ||
234
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MDAR] || vars_h[VAR_S2R_MDAR] ||
235
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_HSUB] || vars_h[VAR_S2R_MAIN_HSUB] ||
236
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_VSUB] || vars_h[VAR_S2R_MAIN_VSUB] ||
237
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_N] || vars_h[VAR_S2R_MAIN_N] ||
238
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_T] || vars_h[VAR_S2R_MAIN_T] ||
239
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 22508 times.
22508 vars_w[VAR_S2R_MAIN_POS] || vars_h[VAR_S2R_MAIN_POS]) ) {
240 av_log(ctx, AV_LOG_ERROR, "Expressions with scale2ref variables are not valid in scale filter.\n");
241 return AVERROR(EINVAL);
242 }
243
244
1/2
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
22508 if (scale->eval_mode == EVAL_MODE_INIT &&
245
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 (vars_w[VAR_N] || vars_h[VAR_N] ||
246
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_T] || vars_h[VAR_T] ||
247
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_N] || vars_h[VAR_S2R_MAIN_N] ||
248
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 22508 times.
✗ Branch 3 not taken.
22508 vars_w[VAR_S2R_MAIN_T] || vars_h[VAR_S2R_MAIN_T] ||
249
2/4
✓ Branch 0 taken 22508 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 22508 times.
22508 vars_w[VAR_S2R_MAIN_POS] || vars_h[VAR_S2R_MAIN_POS]) ) {
250 av_log(ctx, AV_LOG_ERROR, "Expressions with frame variables 'n', 't', 'pos' are not valid in init eval_mode.\n");
251 return AVERROR(EINVAL);
252 }
253
254 22508 return 0;
255 }
256
257 22508 static int scale_parse_expr(AVFilterContext *ctx, char *str_expr, AVExpr **pexpr_ptr, const char *var, const char *args)
258 {
259 22508 ScaleContext *scale = ctx->priv;
260 22508 int ret, is_inited = 0;
261 22508 char *old_str_expr = NULL;
262 22508 AVExpr *old_pexpr = NULL;
263
264
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 22508 times.
22508 if (str_expr) {
265 old_str_expr = av_strdup(str_expr);
266 if (!old_str_expr)
267 return AVERROR(ENOMEM);
268 av_opt_set(scale, var, args, 0);
269 }
270
271
2/2
✓ Branch 0 taken 38 times.
✓ Branch 1 taken 22470 times.
22508 if (*pexpr_ptr) {
272 38 old_pexpr = *pexpr_ptr;
273 38 *pexpr_ptr = NULL;
274 38 is_inited = 1;
275 }
276
277 22508 ret = av_expr_parse(pexpr_ptr, args, var_names,
278 NULL, NULL, NULL, NULL, 0, ctx);
279
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 22508 times.
22508 if (ret < 0) {
280 av_log(ctx, AV_LOG_ERROR, "Cannot parse expression for %s: '%s'\n", var, args);
281 goto revert;
282 }
283
284 22508 ret = check_exprs(ctx);
285
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 22508 times.
22508 if (ret < 0)
286 goto revert;
287
288
3/4
✓ Branch 0 taken 38 times.
✓ Branch 1 taken 22470 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 38 times.
22508 if (is_inited && (ret = config_props(ctx->outputs[0])) < 0)
289 goto revert;
290
291 22508 av_expr_free(old_pexpr);
292 22508 old_pexpr = NULL;
293 22508 av_freep(&old_str_expr);
294
295 22508 return 0;
296
297 revert:
298 av_expr_free(*pexpr_ptr);
299 *pexpr_ptr = NULL;
300 if (old_str_expr) {
301 av_opt_set(scale, var, old_str_expr, 0);
302 av_free(old_str_expr);
303 }
304 if (old_pexpr)
305 *pexpr_ptr = old_pexpr;
306
307 return ret;
308 }
309
310 11235 static av_cold int preinit(AVFilterContext *ctx)
311 {
312 11235 ScaleContext *scale = ctx->priv;
313
314 11235 scale->sws = sws_alloc_context();
315
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
11235 if (!scale->sws)
316 return AVERROR(ENOMEM);
317
318 // set threads=0, so we can later check whether the user modified it
319 11235 scale->sws->threads = 0;
320
321 11235 ff_framesync_preinit(&scale->fs);
322
323 11235 return 0;
324 }
325
326 static int do_scale(FFFrameSync *fs);
327
328 11235 static av_cold int init(AVFilterContext *ctx)
329 {
330 11235 ScaleContext *scale = ctx->priv;
331 int ret;
332
333
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
11235 if (IS_SCALE2REF(ctx))
334 av_log(ctx, AV_LOG_WARNING, "scale2ref is deprecated, use scale=rw:rh instead\n");
335
336
1/6
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
11235 if (scale->size_str && (scale->w_expr || scale->h_expr)) {
337 av_log(ctx, AV_LOG_ERROR,
338 "Size and width/height expressions cannot be set at the same time.\n");
339 return AVERROR(EINVAL);
340 }
341
342
3/4
✓ Branch 0 taken 1874 times.
✓ Branch 1 taken 9361 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 1874 times.
11235 if (scale->w_expr && !scale->h_expr)
343 FFSWAP(char *, scale->w_expr, scale->size_str);
344
345
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
11235 if (scale->size_str) {
346 char buf[32];
347 if ((ret = av_parse_video_size(&scale->w, &scale->h, scale->size_str)) < 0) {
348 av_log(ctx, AV_LOG_ERROR,
349 "Invalid size '%s'\n", scale->size_str);
350 return ret;
351 }
352 snprintf(buf, sizeof(buf)-1, "%d", scale->w);
353 av_opt_set(scale, "w", buf, 0);
354 snprintf(buf, sizeof(buf)-1, "%d", scale->h);
355 av_opt_set(scale, "h", buf, 0);
356 }
357
2/2
✓ Branch 0 taken 9361 times.
✓ Branch 1 taken 1874 times.
11235 if (!scale->w_expr)
358 9361 av_opt_set(scale, "w", "iw", 0);
359
2/2
✓ Branch 0 taken 9361 times.
✓ Branch 1 taken 1874 times.
11235 if (!scale->h_expr)
360 9361 av_opt_set(scale, "h", "ih", 0);
361
362 11235 ret = scale_parse_expr(ctx, NULL, &scale->w_pexpr, "width", scale->w_expr);
363
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
11235 if (ret < 0)
364 return ret;
365
366 11235 ret = scale_parse_expr(ctx, NULL, &scale->h_pexpr, "height", scale->h_expr);
367
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
11235 if (ret < 0)
368 return ret;
369
370
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
11235 if (scale->in_primaries != -1 && !sws_test_primaries(scale->in_primaries, 0)) {
371 av_log(ctx, AV_LOG_ERROR, "Unsupported input primaries '%s'\n",
372 av_color_primaries_name(scale->in_primaries));
373 return AVERROR(EINVAL);
374 }
375
376
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
11235 if (scale->out_primaries != -1 && !sws_test_primaries(scale->out_primaries, 1)) {
377 av_log(ctx, AV_LOG_ERROR, "Unsupported output primaries '%s'\n",
378 av_color_primaries_name(scale->out_primaries));
379 return AVERROR(EINVAL);
380 }
381
382
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
11235 if (scale->in_transfer != -1 && !sws_test_transfer(scale->in_transfer, 0)) {
383 av_log(ctx, AV_LOG_ERROR, "Unsupported input transfer '%s'\n",
384 av_color_transfer_name(scale->in_transfer));
385 return AVERROR(EINVAL);
386 }
387
388
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 11235 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
11235 if (scale->out_transfer != -1 && !sws_test_transfer(scale->out_transfer, 1)) {
389 av_log(ctx, AV_LOG_ERROR, "Unsupported output transfer '%s'\n",
390 av_color_transfer_name(scale->out_transfer));
391 return AVERROR(EINVAL);
392 }
393
394
3/4
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 11231 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 4 times.
11235 if (scale->in_color_matrix != -1 && !sws_test_colorspace(scale->in_color_matrix, 0)) {
395 av_log(ctx, AV_LOG_ERROR, "Unsupported input color matrix '%s'\n",
396 av_color_space_name(scale->in_color_matrix));
397 return AVERROR(EINVAL);
398 }
399
400
2/4
✓ Branch 0 taken 11235 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 11235 times.
11235 if (scale->out_color_matrix != -1 && !sws_test_colorspace(scale->out_color_matrix, 1)) {
401 av_log(ctx, AV_LOG_ERROR, "Unsupported output color matrix '%s'\n",
402 av_color_space_name(scale->out_color_matrix));
403 return AVERROR(EINVAL);
404 }
405
406 11235 av_log(ctx, AV_LOG_VERBOSE, "w:%s h:%s flags:'%s' interl:%d\n",
407 11235 scale->w_expr, scale->h_expr, (char *)av_x_if_null(scale->flags_str, ""), scale->interlaced);
408
409
3/4
✓ Branch 0 taken 11235 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 2272 times.
✓ Branch 3 taken 8963 times.
11235 if (scale->flags_str && *scale->flags_str) {
410 2272 ret = av_opt_set(scale->sws, "sws_flags", scale->flags_str, 0);
411
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2272 times.
2272 if (ret < 0)
412 return ret;
413 }
414
415
2/2
✓ Branch 0 taken 22470 times.
✓ Branch 1 taken 11235 times.
33705 for (int i = 0; i < FF_ARRAY_ELEMS(scale->param); i++)
416
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 22470 times.
22470 if (scale->param[i] != DBL_MAX)
417 scale->sws->scaler_params[i] = scale->param[i];
418
419 11235 scale->sws->src_h_chr_pos = scale->in_h_chr_pos;
420 11235 scale->sws->src_v_chr_pos = scale->in_v_chr_pos;
421 11235 scale->sws->dst_h_chr_pos = scale->out_h_chr_pos;
422 11235 scale->sws->dst_v_chr_pos = scale->out_v_chr_pos;
423
424 // use generic thread-count if the user did not set it explicitly
425
1/2
✓ Branch 0 taken 11235 times.
✗ Branch 1 not taken.
11235 if (!scale->sws->threads)
426 11235 scale->sws->threads = ff_filter_get_nb_threads(ctx);
427
428
3/4
✓ Branch 0 taken 11235 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 11233 times.
11235 if (!IS_SCALE2REF(ctx) && scale->uses_ref) {
429 2 AVFilterPad pad = {
430 .name = "ref",
431 .type = AVMEDIA_TYPE_VIDEO,
432 };
433 2 ret = ff_append_inpad(ctx, &pad);
434
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 2 times.
2 if (ret < 0)
435 return ret;
436 }
437
438 11235 return 0;
439 }
440
441 11235 static av_cold void uninit(AVFilterContext *ctx)
442 {
443 11235 ScaleContext *scale = ctx->priv;
444 11235 av_expr_free(scale->w_pexpr);
445 11235 av_expr_free(scale->h_pexpr);
446 11235 scale->w_pexpr = scale->h_pexpr = NULL;
447 11235 ff_framesync_uninit(&scale->fs);
448 11235 sws_free_context(&scale->sws);
449 11235 }
450
451 6667 static int query_formats(const AVFilterContext *ctx,
452 AVFilterFormatsConfig **cfg_in,
453 AVFilterFormatsConfig **cfg_out)
454 {
455 6667 const ScaleContext *scale = ctx->priv;
456 AVFilterFormats *formats;
457 const AVPixFmtDescriptor *desc;
458 enum AVPixelFormat pix_fmt;
459 int ret;
460
461 6667 desc = NULL;
462 6667 formats = NULL;
463
2/2
✓ Branch 1 taken 1780089 times.
✓ Branch 2 taken 6667 times.
1786756 while ((desc = av_pix_fmt_desc_next(desc))) {
464 1780089 pix_fmt = av_pix_fmt_desc_get_id(desc);
465
2/2
✓ Branch 1 taken 1546744 times.
✓ Branch 2 taken 233345 times.
1780089 if (sws_test_format(pix_fmt, 0)) {
466
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1546744 times.
1546744 if ((ret = ff_add_format(&formats, pix_fmt)) < 0)
467 return ret;
468 }
469 }
470
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6667 times.
6667 if ((ret = ff_formats_ref(formats, &cfg_in[0]->formats)) < 0)
471 return ret;
472
473 6667 desc = NULL;
474 6667 formats = NULL;
475
2/2
✓ Branch 1 taken 1780089 times.
✓ Branch 2 taken 6667 times.
1786756 while ((desc = av_pix_fmt_desc_next(desc))) {
476 1780089 pix_fmt = av_pix_fmt_desc_get_id(desc);
477
4/4
✓ Branch 1 taken 420021 times.
✓ Branch 2 taken 1360068 times.
✓ Branch 3 taken 6667 times.
✓ Branch 4 taken 413354 times.
1780089 if (sws_test_format(pix_fmt, 1) || pix_fmt == AV_PIX_FMT_PAL8) {
478
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 1366735 times.
1366735 if ((ret = ff_add_format(&formats, pix_fmt)) < 0)
479 return ret;
480 }
481 }
482
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6667 times.
6667 if ((ret = ff_formats_ref(formats, &cfg_out[0]->formats)) < 0)
483 return ret;
484
485 /* accept all supported inputs, even if user overrides their properties */
486 6667 formats = ff_all_color_spaces();
487
2/2
✓ Branch 0 taken 113339 times.
✓ Branch 1 taken 6667 times.
120006 for (int i = 0; i < formats->nb_formats; i++) {
488
2/2
✓ Branch 1 taken 60003 times.
✓ Branch 2 taken 53336 times.
113339 if (!sws_test_colorspace(formats->formats[i], 0)) {
489
2/2
✓ Branch 0 taken 246679 times.
✓ Branch 1 taken 60003 times.
306682 for (int j = i--; j + 1 < formats->nb_formats; j++)
490 246679 formats->formats[j] = formats->formats[j + 1];
491 60003 formats->nb_formats--;
492 }
493 }
494
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6667 times.
6667 if ((ret = ff_formats_ref(formats, &cfg_in[0]->color_spaces)) < 0)
495 return ret;
496
497
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6667 times.
6667 if ((ret = ff_formats_ref(ff_all_color_ranges(),
498 6667 &cfg_in[0]->color_ranges)) < 0)
499 return ret;
500
501 /* propagate output properties if overridden */
502
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 6665 times.
6667 if (scale->out_color_matrix != AVCOL_SPC_UNSPECIFIED) {
503 2 formats = ff_make_formats_list_singleton(scale->out_color_matrix);
504 } else {
505 6665 formats = ff_all_color_spaces();
506
2/2
✓ Branch 0 taken 113305 times.
✓ Branch 1 taken 6665 times.
119970 for (int i = 0; i < formats->nb_formats; i++) {
507
2/2
✓ Branch 1 taken 59985 times.
✓ Branch 2 taken 53320 times.
113305 if (!sws_test_colorspace(formats->formats[i], 1)) {
508
2/2
✓ Branch 0 taken 246605 times.
✓ Branch 1 taken 59985 times.
306590 for (int j = i--; j + 1 < formats->nb_formats; j++)
509 246605 formats->formats[j] = formats->formats[j + 1];
510 59985 formats->nb_formats--;
511 }
512 }
513 }
514
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6667 times.
6667 if ((ret = ff_formats_ref(formats, &cfg_out[0]->color_spaces)) < 0)
515 return ret;
516
517 13334 formats = scale->out_range != AVCOL_RANGE_UNSPECIFIED
518 2 ? ff_make_formats_list_singleton(scale->out_range)
519
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 6665 times.
6667 : ff_all_color_ranges();
520
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6667 times.
6667 if ((ret = ff_formats_ref(formats, &cfg_out[0]->color_ranges)) < 0)
521 return ret;
522
523
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6667 times.
6667 if (scale->sws->alpha_blend) {
524 if ((ret = ff_formats_ref(ff_make_formats_list_singleton(AVALPHA_MODE_STRAIGHT),
525 &cfg_in[0]->alpha_modes)) < 0)
526 return ret;
527 }
528
529 6667 return 0;
530 }
531
532 6678 static int scale_eval_dimensions(AVFilterContext *ctx)
533 {
534 6678 ScaleContext *scale = ctx->priv;
535 6678 const char scale2ref = IS_SCALE2REF(ctx);
536
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 const AVFilterLink *inlink = scale2ref ? ctx->inputs[1] : ctx->inputs[0];
537 6678 const AVFilterLink *outlink = ctx->outputs[0];
538 6678 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(inlink->format);
539 6678 const AVPixFmtDescriptor *out_desc = av_pix_fmt_desc_get(outlink->format);
540 char *expr;
541 int eval_w, eval_h;
542 int ret;
543 double res;
544 const AVPixFmtDescriptor *main_desc;
545 const AVFilterLink *main_link;
546
547
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 if (scale2ref) {
548 main_link = ctx->inputs[0];
549 main_desc = av_pix_fmt_desc_get(main_link->format);
550 }
551
552 6678 scale->var_values[VAR_IN_W] = scale->var_values[VAR_IW] = inlink->w;
553 6678 scale->var_values[VAR_IN_H] = scale->var_values[VAR_IH] = inlink->h;
554 6678 scale->var_values[VAR_OUT_W] = scale->var_values[VAR_OW] = NAN;
555 6678 scale->var_values[VAR_OUT_H] = scale->var_values[VAR_OH] = NAN;
556 6678 scale->var_values[VAR_A] = (double) inlink->w / inlink->h;
557 13356 scale->var_values[VAR_SAR] = inlink->sample_aspect_ratio.num ?
558
2/2
✓ Branch 0 taken 2540 times.
✓ Branch 1 taken 4138 times.
6678 (double) inlink->sample_aspect_ratio.num / inlink->sample_aspect_ratio.den : 1;
559 6678 scale->var_values[VAR_DAR] = scale->var_values[VAR_A] * scale->var_values[VAR_SAR];
560 6678 scale->var_values[VAR_HSUB] = 1 << desc->log2_chroma_w;
561 6678 scale->var_values[VAR_VSUB] = 1 << desc->log2_chroma_h;
562 6678 scale->var_values[VAR_OHSUB] = 1 << out_desc->log2_chroma_w;
563 6678 scale->var_values[VAR_OVSUB] = 1 << out_desc->log2_chroma_h;
564
565
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 if (scale2ref) {
566 scale->var_values[VAR_S2R_MAIN_W] = main_link->w;
567 scale->var_values[VAR_S2R_MAIN_H] = main_link->h;
568 scale->var_values[VAR_S2R_MAIN_A] = (double) main_link->w / main_link->h;
569 scale->var_values[VAR_S2R_MAIN_SAR] = main_link->sample_aspect_ratio.num ?
570 (double) main_link->sample_aspect_ratio.num / main_link->sample_aspect_ratio.den : 1;
571 scale->var_values[VAR_S2R_MAIN_DAR] = scale->var_values[VAR_S2R_MDAR] =
572 scale->var_values[VAR_S2R_MAIN_A] * scale->var_values[VAR_S2R_MAIN_SAR];
573 scale->var_values[VAR_S2R_MAIN_HSUB] = 1 << main_desc->log2_chroma_w;
574 scale->var_values[VAR_S2R_MAIN_VSUB] = 1 << main_desc->log2_chroma_h;
575 }
576
577
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 6677 times.
6678 if (scale->uses_ref) {
578 1 const AVFilterLink *reflink = ctx->inputs[1];
579 1 const AVPixFmtDescriptor *ref_desc = av_pix_fmt_desc_get(reflink->format);
580 1 scale->var_values[VAR_REF_W] = scale->var_values[VAR_RW] = reflink->w;
581 1 scale->var_values[VAR_REF_H] = scale->var_values[VAR_RH] = reflink->h;
582 1 scale->var_values[VAR_REF_A] = (double) reflink->w / reflink->h;
583 2 scale->var_values[VAR_REF_SAR] = reflink->sample_aspect_ratio.num ?
584
1/2
✓ Branch 0 taken 1 times.
✗ Branch 1 not taken.
1 (double) reflink->sample_aspect_ratio.num / reflink->sample_aspect_ratio.den : 1;
585 1 scale->var_values[VAR_REF_DAR] = scale->var_values[VAR_RDAR] =
586 1 scale->var_values[VAR_REF_A] * scale->var_values[VAR_REF_SAR];
587 1 scale->var_values[VAR_REF_HSUB] = 1 << ref_desc->log2_chroma_w;
588 1 scale->var_values[VAR_REF_VSUB] = 1 << ref_desc->log2_chroma_h;
589 }
590
591 6678 res = av_expr_eval(scale->w_pexpr, scale->var_values, NULL);
592
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 eval_w = scale->var_values[VAR_OUT_W] = scale->var_values[VAR_OW] = (int) res == 0 ? inlink->w : (int) res;
593
594 6678 res = av_expr_eval(scale->h_pexpr, scale->var_values, NULL);
595
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 if (isnan(res)) {
596 expr = scale->h_expr;
597 ret = AVERROR(EINVAL);
598 goto fail;
599 }
600
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 eval_h = scale->var_values[VAR_OUT_H] = scale->var_values[VAR_OH] = (int) res == 0 ? inlink->h : (int) res;
601
602 6678 res = av_expr_eval(scale->w_pexpr, scale->var_values, NULL);
603
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 if (isnan(res)) {
604 expr = scale->w_expr;
605 ret = AVERROR(EINVAL);
606 goto fail;
607 }
608
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 eval_w = scale->var_values[VAR_OUT_W] = scale->var_values[VAR_OW] = (int) res == 0 ? inlink->w : (int) res;
609
610 6678 scale->w = eval_w;
611 6678 scale->h = eval_h;
612
613 6678 return 0;
614
615 fail:
616 av_log(ctx, AV_LOG_ERROR,
617 "Error when evaluating the expression '%s'.\n", expr);
618 return ret;
619 }
620
621 6678 static int config_props(AVFilterLink *outlink)
622 {
623 6678 AVFilterContext *ctx = outlink->src;
624 6678 AVFilterLink *inlink0 = outlink->src->inputs[0];
625 13356 AVFilterLink *inlink = IS_SCALE2REF(ctx) ?
626
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 outlink->src->inputs[1] :
627 6678 outlink->src->inputs[0];
628 6678 ScaleContext *scale = ctx->priv;
629 6678 uint8_t *flags_val = NULL;
630 6678 double w_adj = 1.0;
631 int ret;
632
633
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 6678 times.
6678 if ((ret = scale_eval_dimensions(ctx)) < 0)
634 goto fail;
635
636 6678 outlink->w = scale->w;
637 6678 outlink->h = scale->h;
638
639
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6678 times.
6678 if (scale->reset_sar)
640 w_adj = IS_SCALE2REF(ctx) ? scale->var_values[VAR_S2R_MAIN_SAR] :
641 scale->var_values[VAR_SAR];
642
643 6678 ret = ff_scale_adjust_dimensions(inlink, &outlink->w, &outlink->h,
644 scale->force_original_aspect_ratio,
645 scale->force_divisible_by, w_adj);
646
647
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 6677 times.
6678 if (ret < 0)
648 1 goto fail;
649
650 6677 if (outlink->w > INT_MAX ||
651 outlink->h > INT_MAX ||
652
1/2
✓ Branch 0 taken 6677 times.
✗ Branch 1 not taken.
6677 (outlink->h * (uint64_t)inlink->w) > INT_MAX ||
653
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6677 times.
6677 (outlink->w * (uint64_t)inlink->h) > INT_MAX)
654 av_log(ctx, AV_LOG_ERROR, "Rescaled value for width or height is too big.\n");
655
656 /* TODO: make algorithm configurable */
657
658
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6677 times.
6677 if (scale->reset_sar)
659 outlink->sample_aspect_ratio = (AVRational){1, 1};
660
2/2
✓ Branch 0 taken 2539 times.
✓ Branch 1 taken 4138 times.
6677 else if (inlink0->sample_aspect_ratio.num){
661 2539 AVRational q = av_div_q((AVRational){inlink0->w, inlink0->h}, (AVRational){outlink->w, outlink->h});
662 2539 outlink->sample_aspect_ratio = av_mul_q(q, inlink0->sample_aspect_ratio);
663 } else
664 4138 outlink->sample_aspect_ratio = inlink0->sample_aspect_ratio;
665
666 6677 av_opt_get(scale->sws, "sws_flags", 0, &flags_val);
667 20031 av_log(ctx, AV_LOG_VERBOSE, "w:%d h:%d fmt:%s csp:%s range:%s sar:%d/%d -> w:%d h:%d fmt:%s csp:%s range:%s sar:%d/%d flags:%s\n",
668 6677 inlink ->w, inlink ->h, av_get_pix_fmt_name( inlink->format),
669 av_color_space_name(inlink->colorspace), av_color_range_name(inlink->color_range),
670 inlink->sample_aspect_ratio.num, inlink->sample_aspect_ratio.den,
671 6677 outlink->w, outlink->h, av_get_pix_fmt_name(outlink->format),
672 av_color_space_name(outlink->colorspace), av_color_range_name(outlink->color_range),
673 outlink->sample_aspect_ratio.num, outlink->sample_aspect_ratio.den,
674 flags_val);
675 6677 av_freep(&flags_val);
676
677
4/4
✓ Branch 0 taken 6221 times.
✓ Branch 1 taken 456 times.
✓ Branch 2 taken 10 times.
✓ Branch 3 taken 6211 times.
6677 if (inlink->w != outlink->w || inlink->h != outlink->h) {
678 466 av_frame_side_data_remove_by_props(&outlink->side_data, &outlink->nb_side_data,
679 AV_SIDE_DATA_PROP_SIZE_DEPENDENT);
680 }
681
682
2/4
✓ Branch 0 taken 6677 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 6677 times.
6677 if (scale->in_primaries != scale->out_primaries || scale->in_transfer != scale->out_transfer) {
683 av_frame_side_data_remove_by_props(&outlink->side_data, &outlink->nb_side_data,
684 AV_SIDE_DATA_PROP_COLOR_DEPENDENT);
685 }
686
687
1/2
✓ Branch 0 taken 6677 times.
✗ Branch 1 not taken.
6677 if (!IS_SCALE2REF(ctx)) {
688 6677 ff_framesync_uninit(&scale->fs);
689 6677 ret = ff_framesync_init(&scale->fs, ctx, ctx->nb_inputs);
690
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6677 times.
6677 if (ret < 0)
691 return ret;
692 6677 scale->fs.on_event = do_scale;
693 6677 scale->fs.in[0].time_base = ctx->inputs[0]->time_base;
694 6677 scale->fs.in[0].sync = 1;
695 6677 scale->fs.in[0].before = EXT_STOP;
696 6677 scale->fs.in[0].after = EXT_STOP;
697
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 6676 times.
6677 if (scale->uses_ref) {
698
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
1 av_assert0(ctx->nb_inputs == 2);
699 1 scale->fs.in[1].time_base = ctx->inputs[1]->time_base;
700 1 scale->fs.in[1].sync = 0;
701 1 scale->fs.in[1].before = EXT_NULL;
702 1 scale->fs.in[1].after = EXT_INFINITY;
703 }
704
705 6677 ret = ff_framesync_configure(&scale->fs);
706
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6677 times.
6677 if (ret < 0)
707 return ret;
708 }
709
710 6677 return 0;
711
712 1 fail:
713 1 return ret;
714 }
715
716 static int config_props_ref(AVFilterLink *outlink)
717 {
718 AVFilterLink *inlink = outlink->src->inputs[1];
719 FilterLink *il = ff_filter_link(inlink);
720 FilterLink *ol = ff_filter_link(outlink);
721
722 outlink->w = inlink->w;
723 outlink->h = inlink->h;
724 outlink->sample_aspect_ratio = inlink->sample_aspect_ratio;
725 outlink->time_base = inlink->time_base;
726 ol->frame_rate = il->frame_rate;
727 outlink->colorspace = inlink->colorspace;
728 outlink->color_range = inlink->color_range;
729
730 return 0;
731 }
732
733 static int request_frame(AVFilterLink *outlink)
734 {
735 return ff_request_frame(outlink->src->inputs[0]);
736 }
737
738 static int request_frame_ref(AVFilterLink *outlink)
739 {
740 return ff_request_frame(outlink->src->inputs[1]);
741 }
742
743 /* Takes over ownership of *frame_in, passes ownership of *frame_out to caller */
744 110433 static int scale_frame(AVFilterLink *link, AVFrame **frame_in,
745 AVFrame **frame_out)
746 {
747 110433 FilterLink *inl = ff_filter_link(link);
748 110433 AVFilterContext *ctx = link->dst;
749 110433 ScaleContext *scale = ctx->priv;
750 110433 AVFilterLink *outlink = ctx->outputs[0];
751 110433 AVFrame *out, *in = *frame_in;
752 110433 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(link->format);
753 char buf[32];
754 int ret, flags_orig, frame_changed;
755
756 110433 *frame_in = NULL;
757
758 331298 frame_changed = in->width != link->w ||
759
1/2
✓ Branch 0 taken 110432 times.
✗ Branch 1 not taken.
110432 in->height != link->h ||
760
1/2
✓ Branch 0 taken 110432 times.
✗ Branch 1 not taken.
110432 in->format != link->format ||
761
1/2
✓ Branch 0 taken 110432 times.
✗ Branch 1 not taken.
110432 in->sample_aspect_ratio.den != link->sample_aspect_ratio.den ||
762
1/2
✓ Branch 0 taken 110432 times.
✗ Branch 1 not taken.
110432 in->sample_aspect_ratio.num != link->sample_aspect_ratio.num ||
763
4/4
✓ Branch 0 taken 110432 times.
✓ Branch 1 taken 1 times.
✓ Branch 2 taken 110416 times.
✓ Branch 3 taken 16 times.
331281 in->colorspace != link->colorspace ||
764
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 110414 times.
110416 in->color_range != link->color_range;
765
766
3/4
✓ Branch 0 taken 110433 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 19 times.
✓ Branch 3 taken 110414 times.
110433 if (scale->eval_mode == EVAL_MODE_FRAME || frame_changed) {
767 19 unsigned vars_w[VARS_NB] = { 0 }, vars_h[VARS_NB] = { 0 };
768
769 19 av_expr_count_vars(scale->w_pexpr, vars_w, VARS_NB);
770 19 av_expr_count_vars(scale->h_pexpr, vars_h, VARS_NB);
771
772
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 19 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
19 if (scale->eval_mode == EVAL_MODE_FRAME &&
773 !frame_changed &&
774 !IS_SCALE2REF(ctx) &&
775 !(vars_w[VAR_N] || vars_w[VAR_T]) &&
776 !(vars_h[VAR_N] || vars_h[VAR_T]) &&
777 scale->w && scale->h)
778 goto scale;
779
780
1/2
✓ Branch 0 taken 19 times.
✗ Branch 1 not taken.
19 if (scale->eval_mode == EVAL_MODE_INIT) {
781 19 snprintf(buf, sizeof(buf) - 1, "%d", scale->w);
782 19 av_opt_set(scale, "w", buf, 0);
783 19 snprintf(buf, sizeof(buf) - 1, "%d", scale->h);
784 19 av_opt_set(scale, "h", buf, 0);
785
786 19 ret = scale_parse_expr(ctx, NULL, &scale->w_pexpr, "width", scale->w_expr);
787
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 19 times.
19 if (ret < 0)
788 goto err;
789
790 19 ret = scale_parse_expr(ctx, NULL, &scale->h_pexpr, "height", scale->h_expr);
791
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 19 times.
19 if (ret < 0)
792 goto err;
793 }
794
795
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 19 times.
19 if (IS_SCALE2REF(ctx)) {
796 scale->var_values[VAR_S2R_MAIN_N] = inl->frame_count_out;
797 scale->var_values[VAR_S2R_MAIN_T] = TS2T(in->pts, link->time_base);
798 } else {
799 19 scale->var_values[VAR_N] = inl->frame_count_out;
800
1/2
✓ Branch 0 taken 19 times.
✗ Branch 1 not taken.
19 scale->var_values[VAR_T] = TS2T(in->pts, link->time_base);
801 }
802
803 19 link->format = in->format;
804 19 link->w = in->width;
805 19 link->h = in->height;
806 19 link->colorspace = in->colorspace;
807 19 link->color_range = in->color_range;
808 19 link->sample_aspect_ratio = in->sample_aspect_ratio;
809
810
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 19 times.
19 if ((ret = config_props(outlink)) < 0)
811 goto err;
812 }
813
814 110414 scale:
815 110433 scale->hsub = desc->log2_chroma_w;
816 110433 scale->vsub = desc->log2_chroma_h;
817
818 110433 out = ff_get_video_buffer(outlink, outlink->w, outlink->h);
819
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (!out) {
820 ret = AVERROR(ENOMEM);
821 goto err;
822 }
823
824
2/2
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 110429 times.
110433 if (scale->in_color_matrix != -1)
825 4 in->colorspace = scale->in_color_matrix;
826
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (scale->in_primaries != -1)
827 in->color_primaries = scale->in_primaries;
828
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (scale->in_transfer != -1)
829 in->color_trc = scale->in_transfer;
830
2/2
✓ Branch 0 taken 4 times.
✓ Branch 1 taken 110429 times.
110433 if (scale->in_range != AVCOL_RANGE_UNSPECIFIED)
831 4 in->color_range = scale->in_range;
832 110433 in->chroma_location = scale->in_chroma_loc;
833
834 110433 flags_orig = in->flags;
835
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (scale->interlaced > 0)
836 in->flags |= AV_FRAME_FLAG_INTERLACED;
837
1/2
✓ Branch 0 taken 110433 times.
✗ Branch 1 not taken.
110433 else if (!scale->interlaced)
838 110433 in->flags &= ~AV_FRAME_FLAG_INTERLACED;
839
840 110433 av_frame_copy_props(out, in);
841 110433 out->width = outlink->w;
842 110433 out->height = outlink->h;
843 110433 out->color_range = outlink->color_range;
844 110433 out->colorspace = outlink->colorspace;
845 110433 out->alpha_mode = outlink->alpha_mode;
846
2/2
✓ Branch 0 taken 25 times.
✓ Branch 1 taken 110408 times.
110433 if (scale->out_chroma_loc != AVCHROMA_LOC_UNSPECIFIED)
847 25 out->chroma_location = scale->out_chroma_loc;
848
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (scale->out_primaries != -1)
849 out->color_primaries = scale->out_primaries;
850
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (scale->out_transfer != -1)
851 out->color_trc = scale->out_transfer;
852
853
4/4
✓ Branch 0 taken 104248 times.
✓ Branch 1 taken 6185 times.
✓ Branch 2 taken 13 times.
✓ Branch 3 taken 104235 times.
110433 if (out->width != in->width || out->height != in->height) {
854 6198 av_frame_side_data_remove_by_props(&out->side_data, &out->nb_side_data,
855 AV_SIDE_DATA_PROP_SIZE_DEPENDENT);
856 }
857
858
2/4
✓ Branch 0 taken 110433 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 110433 times.
110433 if (in->color_primaries != out->color_primaries || in->color_trc != out->color_trc) {
859 av_frame_side_data_remove_by_props(&out->side_data, &out->nb_side_data,
860 AV_SIDE_DATA_PROP_COLOR_DEPENDENT);
861 }
862
863
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (scale->reset_sar) {
864 out->sample_aspect_ratio = outlink->sample_aspect_ratio;
865 } else {
866 110433 av_reduce(&out->sample_aspect_ratio.num, &out->sample_aspect_ratio.den,
867 110433 (int64_t)in->sample_aspect_ratio.num * outlink->h * link->w,
868 110433 (int64_t)in->sample_aspect_ratio.den * outlink->w * link->h,
869 INT_MAX);
870 }
871
872
2/2
✓ Branch 1 taken 4918 times.
✓ Branch 2 taken 105515 times.
110433 if (sws_is_noop(out, in)) {
873 4918 av_frame_free(&out);
874 4918 in->flags = flags_orig;
875 4918 *frame_out = in;
876 4918 return 0;
877 }
878
879
2/2
✓ Branch 0 taken 589 times.
✓ Branch 1 taken 104926 times.
105515 if (out->format == AV_PIX_FMT_PAL8) {
880 589 out->format = AV_PIX_FMT_BGR8;
881 589 avpriv_set_systematic_pal2((uint32_t*) out->data[1], out->format);
882 }
883
884 105515 ret = sws_scale_frame(scale->sws, out, in);
885 105515 av_frame_free(&in);
886 105515 out->flags = flags_orig;
887 105515 out->format = outlink->format; /* undo PAL8 handling */
888
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 105515 times.
105515 if (ret < 0)
889 av_frame_free(&out);
890 105515 *frame_out = out;
891 105515 return ret;
892
893 err:
894 av_frame_free(&in);
895 return ret;
896 }
897
898 110433 static int do_scale(FFFrameSync *fs)
899 {
900 110433 AVFilterContext *ctx = fs->parent;
901 110433 ScaleContext *scale = ctx->priv;
902 110433 AVFilterLink *outlink = ctx->outputs[0];
903 110433 AVFrame *out, *in = NULL, *ref = NULL;
904 110433 int ret = 0, frame_changed;
905
906 110433 ret = ff_framesync_get_frame(fs, 0, &in, 1);
907
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (ret < 0)
908 goto err;
909
910
2/2
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 110427 times.
110433 if (scale->uses_ref) {
911 6 ret = ff_framesync_get_frame(fs, 1, &ref, 0);
912
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (ret < 0)
913 goto err;
914 }
915
916
2/2
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 110427 times.
110433 if (ref) {
917 6 AVFilterLink *reflink = ctx->inputs[1];
918 6 FilterLink *rl = ff_filter_link(reflink);
919
920 18 frame_changed = ref->width != reflink->w ||
921
1/2
✓ Branch 0 taken 6 times.
✗ Branch 1 not taken.
6 ref->height != reflink->h ||
922
1/2
✓ Branch 0 taken 6 times.
✗ Branch 1 not taken.
6 ref->format != reflink->format ||
923
1/2
✓ Branch 0 taken 6 times.
✗ Branch 1 not taken.
6 ref->sample_aspect_ratio.den != reflink->sample_aspect_ratio.den ||
924
1/2
✓ Branch 0 taken 6 times.
✗ Branch 1 not taken.
6 ref->sample_aspect_ratio.num != reflink->sample_aspect_ratio.num ||
925
2/4
✓ Branch 0 taken 6 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 6 times.
✗ Branch 3 not taken.
18 ref->colorspace != reflink->colorspace ||
926
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 ref->color_range != reflink->color_range;
927
928
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (frame_changed) {
929 reflink->format = ref->format;
930 reflink->w = ref->width;
931 reflink->h = ref->height;
932 reflink->sample_aspect_ratio.num = ref->sample_aspect_ratio.num;
933 reflink->sample_aspect_ratio.den = ref->sample_aspect_ratio.den;
934 reflink->colorspace = ref->colorspace;
935 reflink->color_range = ref->color_range;
936
937 ret = config_props(outlink);
938 if (ret < 0)
939 goto err;
940 }
941
942
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
6 if (scale->eval_mode == EVAL_MODE_FRAME) {
943 scale->var_values[VAR_REF_N] = rl->frame_count_out;
944 scale->var_values[VAR_REF_T] = TS2T(ref->pts, reflink->time_base);
945 }
946 }
947
948 110433 ret = scale_frame(ctx->inputs[0], &in, &out);
949
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 if (ret < 0)
950 goto err;
951
952
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 110433 times.
110433 av_assert0(out);
953 110433 out->pts = av_rescale_q_rnd(fs->pts, fs->time_base, outlink->time_base, AV_ROUND_NEAR_INF | AV_ROUND_PASS_MINMAX);
954 110433 return ff_filter_frame(outlink, out);
955
956 err:
957 av_frame_free(&in);
958 return ret;
959 }
960
961 static int filter_frame(AVFilterLink *link, AVFrame *in)
962 {
963 AVFilterContext *ctx = link->dst;
964 AVFilterLink *outlink = ctx->outputs[0];
965 AVFrame *out;
966 int ret;
967
968 ret = scale_frame(link, &in, &out);
969 if (out)
970 return ff_filter_frame(outlink, out);
971
972 return ret;
973 }
974
975 static int filter_frame_ref(AVFilterLink *link, AVFrame *in)
976 {
977 FilterLink *l = ff_filter_link(link);
978 ScaleContext *scale = link->dst->priv;
979 AVFilterLink *outlink = link->dst->outputs[1];
980 int frame_changed;
981
982 frame_changed = in->width != link->w ||
983 in->height != link->h ||
984 in->format != link->format ||
985 in->sample_aspect_ratio.den != link->sample_aspect_ratio.den ||
986 in->sample_aspect_ratio.num != link->sample_aspect_ratio.num ||
987 in->colorspace != link->colorspace ||
988 in->color_range != link->color_range;
989
990 if (frame_changed) {
991 link->format = in->format;
992 link->w = in->width;
993 link->h = in->height;
994 link->sample_aspect_ratio.num = in->sample_aspect_ratio.num;
995 link->sample_aspect_ratio.den = in->sample_aspect_ratio.den;
996 link->colorspace = in->colorspace;
997 link->color_range = in->color_range;
998
999 config_props_ref(outlink);
1000 }
1001
1002 if (scale->eval_mode == EVAL_MODE_FRAME) {
1003 scale->var_values[VAR_N] = l->frame_count_out;
1004 scale->var_values[VAR_T] = TS2T(in->pts, link->time_base);
1005 }
1006
1007 return ff_filter_frame(outlink, in);
1008 }
1009
1010 static int process_command(AVFilterContext *ctx, const char *cmd, const char *args,
1011 char *res, int res_len, int flags)
1012 {
1013 ScaleContext *scale = ctx->priv;
1014 char *str_expr;
1015 AVExpr **pexpr_ptr;
1016 int ret, w, h;
1017
1018 w = !strcmp(cmd, "width") || !strcmp(cmd, "w");
1019 h = !strcmp(cmd, "height") || !strcmp(cmd, "h");
1020
1021 if (w || h) {
1022 str_expr = w ? scale->w_expr : scale->h_expr;
1023 pexpr_ptr = w ? &scale->w_pexpr : &scale->h_pexpr;
1024
1025 ret = scale_parse_expr(ctx, str_expr, pexpr_ptr, cmd, args);
1026 } else
1027 ret = AVERROR(ENOSYS);
1028
1029 if (ret < 0)
1030 av_log(ctx, AV_LOG_ERROR, "Failed to process command. Continuing with existing parameters.\n");
1031
1032 return ret;
1033 }
1034
1035 221179 static int activate(AVFilterContext *ctx)
1036 {
1037 221179 ScaleContext *scale = ctx->priv;
1038 221179 return ff_framesync_activate(&scale->fs);
1039 }
1040
1041 static const AVClass *child_class_iterate(void **iter)
1042 {
1043 switch ((uintptr_t) *iter) {
1044 case 0:
1045 *iter = (void*)(uintptr_t) 1;
1046 return sws_get_class();
1047 case 1:
1048 *iter = (void*)(uintptr_t) 2;
1049 return &ff_framesync_class;
1050 }
1051
1052 return NULL;
1053 }
1054
1055 25364 static void *child_next(void *obj, void *prev)
1056 {
1057 25364 ScaleContext *s = obj;
1058
2/2
✓ Branch 0 taken 13286 times.
✓ Branch 1 taken 12078 times.
25364 if (!prev)
1059 13286 return s->sws;
1060
2/2
✓ Branch 0 taken 6039 times.
✓ Branch 1 taken 6039 times.
12078 if (prev == s->sws)
1061 6039 return &s->fs;
1062 6039 return NULL;
1063 }
1064
1065 #define OFFSET(x) offsetof(ScaleContext, x)
1066 #define FLAGS AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_FILTERING_PARAM
1067 #define TFLAGS AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_RUNTIME_PARAM
1068
1069 static const AVOption scale_options[] = {
1070 { "w", "Output video width", OFFSET(w_expr), AV_OPT_TYPE_STRING, .flags = TFLAGS },
1071 { "width", "Output video width", OFFSET(w_expr), AV_OPT_TYPE_STRING, .flags = TFLAGS },
1072 { "h", "Output video height", OFFSET(h_expr), AV_OPT_TYPE_STRING, .flags = TFLAGS },
1073 { "height","Output video height", OFFSET(h_expr), AV_OPT_TYPE_STRING, .flags = TFLAGS },
1074 { "flags", "Flags to pass to libswscale", OFFSET(flags_str), AV_OPT_TYPE_STRING, { .str = "" }, .flags = FLAGS },
1075 { "interl", "set interlacing", OFFSET(interlaced), AV_OPT_TYPE_BOOL, {.i64 = 0 }, -1, 1, FLAGS },
1076 { "size", "set video size", OFFSET(size_str), AV_OPT_TYPE_STRING, {.str = NULL}, 0, .flags = FLAGS },
1077 { "s", "set video size", OFFSET(size_str), AV_OPT_TYPE_STRING, {.str = NULL}, 0, .flags = FLAGS },
1078 { "in_color_matrix", "set input YCbCr type", OFFSET(in_color_matrix), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, AVCOL_SPC_NB-1, .flags = FLAGS, .unit = "color" },
1079 { "out_color_matrix", "set output YCbCr type", OFFSET(out_color_matrix), AV_OPT_TYPE_INT, { .i64 = AVCOL_SPC_UNSPECIFIED }, 0, AVCOL_SPC_NB-1, .flags = FLAGS, .unit = "color"},
1080 { "auto", NULL, 0, AV_OPT_TYPE_CONST, {.i64=-1}, 0, 0, FLAGS, .unit = "color" },
1081 { "bt601", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT470BG}, 0, 0, FLAGS, .unit = "color" },
1082 { "bt470", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT470BG}, 0, 0, FLAGS, .unit = "color" },
1083 { "smpte170m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT470BG}, 0, 0, FLAGS, .unit = "color" },
1084 { "bt470bg", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT470BG}, 0, 0, FLAGS, .unit = "color" },
1085 { "bt709", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT709}, 0, 0, FLAGS, .unit = "color" },
1086 { "fcc", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_FCC}, 0, 0, FLAGS, .unit = "color" },
1087 { "smpte240m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_SMPTE240M}, 0, 0, FLAGS, .unit = "color" },
1088 { "bt2020", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT2020_NCL}, 0, 0, FLAGS, .unit = "color" },
1089 { "bt2020nc", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_SPC_BT2020_NCL}, 0, 0, FLAGS, .unit = "color" },
1090 { "in_range", "set input color range", OFFSET( in_range), AV_OPT_TYPE_INT, {.i64 = AVCOL_RANGE_UNSPECIFIED }, 0, 2, FLAGS, .unit = "range" },
1091 { "out_range", "set output color range", OFFSET(out_range), AV_OPT_TYPE_INT, {.i64 = AVCOL_RANGE_UNSPECIFIED }, 0, 2, FLAGS, .unit = "range" },
1092 { "auto", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_UNSPECIFIED }, 0, 0, FLAGS, .unit = "range" },
1093 { "unknown", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_UNSPECIFIED }, 0, 0, FLAGS, .unit = "range" },
1094 { "full", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_JPEG}, 0, 0, FLAGS, .unit = "range" },
1095 { "limited", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_MPEG}, 0, 0, FLAGS, .unit = "range" },
1096 { "jpeg", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_JPEG}, 0, 0, FLAGS, .unit = "range" },
1097 { "mpeg", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_MPEG}, 0, 0, FLAGS, .unit = "range" },
1098 { "tv", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_MPEG}, 0, 0, FLAGS, .unit = "range" },
1099 { "pc", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_RANGE_JPEG}, 0, 0, FLAGS, .unit = "range" },
1100 { "in_chroma_loc", "set input chroma sample location", OFFSET(in_chroma_loc), AV_OPT_TYPE_INT, { .i64 = AVCHROMA_LOC_UNSPECIFIED }, 0, AVCHROMA_LOC_NB-1, .flags = FLAGS, .unit = "chroma_loc" },
1101 { "out_chroma_loc", "set output chroma sample location", OFFSET(out_chroma_loc), AV_OPT_TYPE_INT, { .i64 = AVCHROMA_LOC_UNSPECIFIED }, 0, AVCHROMA_LOC_NB-1, .flags = FLAGS, .unit = "chroma_loc" },
1102 {"auto", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_UNSPECIFIED}, 0, 0, FLAGS, .unit = "chroma_loc"},
1103 {"unknown", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_UNSPECIFIED}, 0, 0, FLAGS, .unit = "chroma_loc"},
1104 {"left", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_LEFT}, 0, 0, FLAGS, .unit = "chroma_loc"},
1105 {"center", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_CENTER}, 0, 0, FLAGS, .unit = "chroma_loc"},
1106 {"topleft", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_TOPLEFT}, 0, 0, FLAGS, .unit = "chroma_loc"},
1107 {"top", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_TOP}, 0, 0, FLAGS, .unit = "chroma_loc"},
1108 {"bottomleft", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_BOTTOMLEFT}, 0, 0, FLAGS, .unit = "chroma_loc"},
1109 {"bottom", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCHROMA_LOC_BOTTOM}, 0, 0, FLAGS, .unit = "chroma_loc"},
1110 { "in_primaries", "set input primaries", OFFSET(in_primaries), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, AVCOL_PRI_NB-1, .flags = FLAGS, .unit = "primaries" },
1111 { "out_primaries", "set output primaries", OFFSET(out_primaries), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, AVCOL_PRI_NB-1, .flags = FLAGS, .unit = "primaries"},
1112 {"auto", NULL, 0, AV_OPT_TYPE_CONST, {.i64=-1}, 0, 0, FLAGS, .unit = "primaries"},
1113 {"bt709", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_BT709}, 0, 0, FLAGS, .unit = "primaries"},
1114 {"bt470m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_BT470M}, 0, 0, FLAGS, .unit = "primaries"},
1115 {"bt470bg", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_BT470BG}, 0, 0, FLAGS, .unit = "primaries"},
1116 {"smpte170m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_SMPTE170M}, 0, 0, FLAGS, .unit = "primaries"},
1117 {"smpte240m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_SMPTE240M}, 0, 0, FLAGS, .unit = "primaries"},
1118 {"film", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_FILM}, 0, 0, FLAGS, .unit = "primaries"},
1119 {"bt2020", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_BT2020}, 0, 0, FLAGS, .unit = "primaries"},
1120 {"smpte428", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_SMPTE428}, 0, 0, FLAGS, .unit = "primaries"},
1121 {"smpte431", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_SMPTE431}, 0, 0, FLAGS, .unit = "primaries"},
1122 {"smpte432", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_SMPTE432}, 0, 0, FLAGS, .unit = "primaries"},
1123 {"jedec-p22", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_JEDEC_P22}, 0, 0, FLAGS, .unit = "primaries"},
1124 {"ebu3213", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_PRI_EBU3213}, 0, 0, FLAGS, .unit = "primaries"},
1125 { "in_transfer", "set output color transfer", OFFSET(in_transfer), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, AVCOL_TRC_NB-1, .flags = FLAGS, .unit = "transfer"},
1126 {"out_transfer", "set output color transfer", OFFSET(out_transfer), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, AVCOL_TRC_NB-1, .flags = FLAGS, .unit = "transfer"},
1127 {"auto", NULL, 0, AV_OPT_TYPE_CONST, {.i64=-1}, 0, 0, FLAGS, .unit = "transfer"},
1128 {"bt709", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_BT709}, 0, 0, FLAGS, .unit = "transfer"},
1129 {"bt470m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_GAMMA22}, 0, 0, FLAGS, .unit = "transfer"},
1130 {"gamma22", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_GAMMA22}, 0, 0, FLAGS, .unit = "transfer"},
1131 {"bt470bg", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_GAMMA28}, 0, 0, FLAGS, .unit = "transfer"},
1132 {"gamma28", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_GAMMA28}, 0, 0, FLAGS, .unit = "transfer"},
1133 {"smpte170m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_SMPTE170M}, 0, 0, FLAGS, .unit = "transfer"},
1134 {"smpte240m", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_SMPTE240M}, 0, 0, FLAGS, .unit = "transfer"},
1135 {"linear", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_LINEAR}, 0, 0, FLAGS, .unit = "transfer"},
1136 {"iec61966-2-1", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_IEC61966_2_1}, 0, 0, FLAGS, .unit = "transfer"},
1137 {"srgb", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_IEC61966_2_1}, 0, 0, FLAGS, .unit = "transfer"},
1138 {"iec61966-2-4", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_IEC61966_2_4}, 0, 0, FLAGS, .unit = "transfer"},
1139 {"xvycc", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_IEC61966_2_4}, 0, 0, FLAGS, .unit = "transfer"},
1140 {"bt1361e", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_BT1361_ECG}, 0, 0, FLAGS, .unit = "transfer"},
1141 {"bt2020-10", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_BT2020_10}, 0, 0, FLAGS, .unit = "transfer"},
1142 {"bt2020-12", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_BT2020_12}, 0, 0, FLAGS, .unit = "transfer"},
1143 {"smpte2084", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_SMPTE2084}, 0, 0, FLAGS, .unit = "transfer"},
1144 {"smpte428", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_SMPTE428}, 0, 0, FLAGS, .unit = "transfer"},
1145 {"arib-std-b67", NULL, 0, AV_OPT_TYPE_CONST, {.i64=AVCOL_TRC_ARIB_STD_B67}, 0, 0, FLAGS, .unit = "transfer"},
1146 { "in_v_chr_pos", "input vertical chroma position in luma grid/256" , OFFSET(in_v_chr_pos), AV_OPT_TYPE_INT, { .i64 = -513}, -513, 512, FLAGS },
1147 { "in_h_chr_pos", "input horizontal chroma position in luma grid/256", OFFSET(in_h_chr_pos), AV_OPT_TYPE_INT, { .i64 = -513}, -513, 512, FLAGS },
1148 { "out_v_chr_pos", "output vertical chroma position in luma grid/256" , OFFSET(out_v_chr_pos), AV_OPT_TYPE_INT, { .i64 = -513}, -513, 512, FLAGS },
1149 { "out_h_chr_pos", "output horizontal chroma position in luma grid/256", OFFSET(out_h_chr_pos), AV_OPT_TYPE_INT, { .i64 = -513}, -513, 512, FLAGS },
1150 { "force_original_aspect_ratio", "decrease or increase w/h if necessary to keep the original AR", OFFSET(force_original_aspect_ratio), AV_OPT_TYPE_INT, { .i64 = 0}, 0, SCALE_FORCE_OAR_NB-1, FLAGS, .unit = "force_oar" },
1151 { "disable", NULL, 0, AV_OPT_TYPE_CONST, {.i64 = SCALE_FORCE_OAR_DISABLE }, 0, 0, FLAGS, .unit = "force_oar" },
1152 { "decrease", NULL, 0, AV_OPT_TYPE_CONST, {.i64 = SCALE_FORCE_OAR_DECREASE }, 0, 0, FLAGS, .unit = "force_oar" },
1153 { "increase", NULL, 0, AV_OPT_TYPE_CONST, {.i64 = SCALE_FORCE_OAR_INCREASE }, 0, 0, FLAGS, .unit = "force_oar" },
1154 { "force_divisible_by", "enforce that the output resolution is divisible by a defined integer when force_original_aspect_ratio is used", OFFSET(force_divisible_by), AV_OPT_TYPE_INT, { .i64 = 1}, 1, 256, FLAGS },
1155 { "reset_sar", "reset SAR to 1 and scale to square pixels if scaling proportionally", OFFSET(reset_sar), AV_OPT_TYPE_BOOL, { .i64 = 0}, 0, 1, FLAGS },
1156 { "param0", "Scaler param 0", OFFSET(param[0]), AV_OPT_TYPE_DOUBLE, { .dbl = DBL_MAX }, -DBL_MAX, DBL_MAX, FLAGS },
1157 { "param1", "Scaler param 1", OFFSET(param[1]), AV_OPT_TYPE_DOUBLE, { .dbl = DBL_MAX }, -DBL_MAX, DBL_MAX, FLAGS },
1158 { "eval", "specify when to evaluate expressions", OFFSET(eval_mode), AV_OPT_TYPE_INT, {.i64 = EVAL_MODE_INIT}, 0, EVAL_MODE_NB-1, FLAGS, .unit = "eval" },
1159 { "init", "eval expressions once during initialization", 0, AV_OPT_TYPE_CONST, {.i64=EVAL_MODE_INIT}, .flags = FLAGS, .unit = "eval" },
1160 { "frame", "eval expressions during initialization and per-frame", 0, AV_OPT_TYPE_CONST, {.i64=EVAL_MODE_FRAME}, .flags = FLAGS, .unit = "eval" },
1161 { NULL }
1162 };
1163
1164 static const AVClass scale_class = {
1165 .class_name = "scale",
1166 .item_name = av_default_item_name,
1167 .option = scale_options,
1168 .version = LIBAVUTIL_VERSION_INT,
1169 .category = AV_CLASS_CATEGORY_FILTER,
1170 .child_class_iterate = child_class_iterate,
1171 .child_next = child_next,
1172 };
1173
1174 static const AVFilterPad avfilter_vf_scale_inputs[] = {
1175 {
1176 .name = "default",
1177 .type = AVMEDIA_TYPE_VIDEO,
1178 },
1179 };
1180
1181 static const AVFilterPad avfilter_vf_scale_outputs[] = {
1182 {
1183 .name = "default",
1184 .type = AVMEDIA_TYPE_VIDEO,
1185 .config_props = config_props,
1186 },
1187 };
1188
1189 const FFFilter ff_vf_scale = {
1190 .p.name = "scale",
1191 .p.description = NULL_IF_CONFIG_SMALL("Scale the input video size and/or convert the image format."),
1192 .p.priv_class = &scale_class,
1193 .p.flags = AVFILTER_FLAG_DYNAMIC_INPUTS,
1194 .preinit = preinit,
1195 .init = init,
1196 .uninit = uninit,
1197 .priv_size = sizeof(ScaleContext),
1198 FILTER_INPUTS(avfilter_vf_scale_inputs),
1199 FILTER_OUTPUTS(avfilter_vf_scale_outputs),
1200 FILTER_QUERY_FUNC2(query_formats),
1201 .activate = activate,
1202 .process_command = process_command,
1203 };
1204
1205 static const AVClass *scale2ref_child_class_iterate(void **iter)
1206 {
1207 const AVClass *c = *iter ? NULL : sws_get_class();
1208 *iter = (void*)(uintptr_t)c;
1209 return c;
1210 }
1211
1212 static void *scale2ref_child_next(void *obj, void *prev)
1213 {
1214 ScaleContext *s = obj;
1215 if (!prev)
1216 return s->sws;
1217 return NULL;
1218 }
1219
1220 static const AVClass scale2ref_class = {
1221 .class_name = "scale(2ref)",
1222 .item_name = av_default_item_name,
1223 .option = scale_options,
1224 .version = LIBAVUTIL_VERSION_INT,
1225 .category = AV_CLASS_CATEGORY_FILTER,
1226 .child_class_iterate = scale2ref_child_class_iterate,
1227 .child_next = scale2ref_child_next,
1228 };
1229
1230 static const AVFilterPad avfilter_vf_scale2ref_inputs[] = {
1231 {
1232 .name = "default",
1233 .type = AVMEDIA_TYPE_VIDEO,
1234 .filter_frame = filter_frame,
1235 },
1236 {
1237 .name = "ref",
1238 .type = AVMEDIA_TYPE_VIDEO,
1239 .filter_frame = filter_frame_ref,
1240 },
1241 };
1242
1243 static const AVFilterPad avfilter_vf_scale2ref_outputs[] = {
1244 {
1245 .name = "default",
1246 .type = AVMEDIA_TYPE_VIDEO,
1247 .config_props = config_props,
1248 .request_frame= request_frame,
1249 },
1250 {
1251 .name = "ref",
1252 .type = AVMEDIA_TYPE_VIDEO,
1253 .config_props = config_props_ref,
1254 .request_frame= request_frame_ref,
1255 },
1256 };
1257
1258 const FFFilter ff_vf_scale2ref = {
1259 .p.name = "scale2ref",
1260 .p.description = NULL_IF_CONFIG_SMALL("Scale the input video size and/or convert the image format to the given reference."),
1261 .p.priv_class = &scale2ref_class,
1262 .preinit = preinit,
1263 .init = init,
1264 .uninit = uninit,
1265 .priv_size = sizeof(ScaleContext),
1266 FILTER_INPUTS(avfilter_vf_scale2ref_inputs),
1267 FILTER_OUTPUTS(avfilter_vf_scale2ref_outputs),
1268 FILTER_QUERY_FUNC2(query_formats),
1269 .process_command = process_command,
1270 };
1271