Files
FFmpeg/libswscale/tests/sws_ops.c
Niklas Haas cf2d40f65d swscale/ops: add explicit clear mask to SwsClearOp
Instead of implicitly testing for NaN values. This is mostly a straightforward
translation, but we need some slight extra boilerplate to ensure the mask
is correctly updated when e.g. commuting past a swizzle.

Signed-off-by: Niklas Haas <git@haasn.dev>
2026-04-16 23:23:36 +02:00

210 lines
6.0 KiB
C

/*
* Copyright (C) 2025 Niklas Haas
*
* This file is part of FFmpeg.
*
* FFmpeg is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* FFmpeg is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "libavutil/avassert.h"
#include "libavutil/avstring.h"
#include "libavutil/mem.h"
#include "libavutil/pixdesc.h"
#include "libavutil/tree.h"
#include "libswscale/ops.h"
#include "libswscale/format.h"
#ifdef _WIN32
#include <io.h>
#include <fcntl.h>
#endif
static int print_ops(SwsContext *const ctx, void *opaque, SwsOpList *ops)
{
av_log(opaque, AV_LOG_INFO, "%s -> %s:\n",
av_get_pix_fmt_name(ops->src.format),
av_get_pix_fmt_name(ops->dst.format));
if (ff_sws_op_list_is_noop(ops))
av_log(opaque, AV_LOG_INFO, " (no-op)\n");
else
ff_sws_op_list_print(opaque, AV_LOG_INFO, AV_LOG_INFO, ops);
return 0;
}
static int cmp_str(const void *a, const void *b)
{
return strcmp(a, b);
}
static int register_op(SwsContext *ctx, void *opaque, SwsOp *op)
{
struct AVTreeNode **root = opaque;
AVBPrint bp;
char *desc;
/* Strip irrelevant constant data from some operations */
switch (op->op) {
case SWS_OP_LINEAR:
for (int i = 0; i < 4; i++) {
for (int j = 0; j < 5; j++)
op->lin.m[i][j] = (AVRational) { 0, 1 };
}
break;
case SWS_OP_SCALE:
op->scale.factor = (AVRational) { 0, 1 };
break;
case SWS_OP_MIN:
case SWS_OP_MAX:
for (int i = 0; i < 4; i++)
op->clamp.limit[i] = (AVRational) { 0, 1 };
break;
case SWS_OP_CLEAR:
for (int i = 0; i < 4; i++)
op->clear.value[i] = (AVRational) { 0, SWS_COMP_TEST(op->clear.mask, i) };
break;
case SWS_OP_DITHER:
/* Strip arbitrary offset */
for (int i = 0; i < 4; i++)
op->dither.y_offset[i] = op->dither.y_offset[i] >= 0 ? 0 : -1;
break;
}
av_bprint_init(&bp, 0, AV_BPRINT_SIZE_AUTOMATIC);
ff_sws_op_desc(&bp, op);
int ret = av_bprint_finalize(&bp, &desc);
if (ret < 0)
return ret;
struct AVTreeNode *node = av_tree_node_alloc();
if (!node) {
av_free(desc);
return AVERROR(ENOMEM);
}
av_tree_insert(root, desc, cmp_str, &node);
if (node) {
av_free(node);
av_free(desc);
}
return ret;
}
static int print_and_free_summary(void *opaque, void *key)
{
char *desc = key;
av_log(opaque, AV_LOG_INFO, "%s\n", desc);
av_free(desc);
return 0;
}
static void log_stdout(void *avcl, int level, const char *fmt, va_list vl)
{
if (level != AV_LOG_INFO) {
av_log_default_callback(avcl, level, fmt, vl);
} else if (av_log_get_level() >= AV_LOG_INFO) {
vfprintf(stdout, fmt, vl);
}
}
int main(int argc, char **argv)
{
enum AVPixelFormat src_fmt = AV_PIX_FMT_NONE;
enum AVPixelFormat dst_fmt = AV_PIX_FMT_NONE;
bool summarize = false;
int ret = 1;
#ifdef _WIN32
_setmode(_fileno(stdout), _O_BINARY);
#endif
for (int i = 1; i < argc; i++) {
if (!strcmp(argv[i], "-help") || !strcmp(argv[i], "--help")) {
fprintf(stderr,
"sws_ops [options...]\n"
" -help\n"
" This text\n"
" -dst <pixfmt>\n"
" Only test the specified destination pixel format\n"
" -src <pixfmt>\n"
" Only test the specified source pixel format\n"
" -v <level>\n"
" Enable log verbosity at given level\n"
" -summarize\n"
" Summarize operation types, instead of printing op lists\n"
);
return 0;
}
if (!strcmp(argv[i], "-src")) {
if (i + 1 >= argc)
goto bad_option;
src_fmt = av_get_pix_fmt(argv[i + 1]);
if (src_fmt == AV_PIX_FMT_NONE) {
fprintf(stderr, "invalid pixel format %s\n", argv[i + 1]);
goto error;
}
i++;
} else if (!strcmp(argv[i], "-dst")) {
if (i + 1 >= argc)
goto bad_option;
dst_fmt = av_get_pix_fmt(argv[i + 1]);
if (dst_fmt == AV_PIX_FMT_NONE) {
fprintf(stderr, "invalid pixel format %s\n", argv[i + 1]);
goto error;
}
i++;
} else if (!strcmp(argv[i], "-v")) {
if (i + 1 >= argc)
goto bad_option;
av_log_set_level(atoi(argv[i + 1]));
i++;
} else if (!strcmp(argv[i], "-summarize")) {
summarize = true;
} else {
bad_option:
fprintf(stderr, "bad option or argument missing (%s) see -help\n", argv[i]);
goto error;
}
}
SwsContext *ctx = sws_alloc_context();
if (!ctx)
goto fail;
av_log_set_callback(log_stdout);
if (summarize) {
struct AVTreeNode *root = NULL;
ret = ff_sws_enum_ops(ctx, &root, src_fmt, dst_fmt, register_op);
if (ret < 0)
goto fail;
av_tree_enumerate(root, NULL, NULL, print_and_free_summary);
av_tree_destroy(root);
} else {
ret = ff_sws_enum_op_lists(ctx, NULL, src_fmt, dst_fmt, print_ops);
if (ret < 0)
goto fail;
}
ret = 0;
fail:
sws_free_context(&ctx);
return ret;
error:
return AVERROR(EINVAL);
}