Go to the documentation of this file.
38 #if HAVE_SPIRV_HEADERS_SPIRV_H || HAVE_SPIRV_UNIFIED1_SPIRV_H
41 #if CONFIG_LIBSHADERC || CONFIG_LIBGLSLANG
47 #if ARCH_AARCH64 && HAVE_NEON
49 #elif ARCH_X86_64 && HAVE_X86ASM
53 #if HAVE_SPIRV_HEADERS_SPIRV_H || HAVE_SPIRV_UNIFIED1_SPIRV_H
56 #if CONFIG_LIBSHADERC || CONFIG_LIBGLSLANG
140 for (
int i = 0;
i < 4;
i++) {
151 for (
int i = 0;
i < 4;
i++) {
152 const int src = swiz->
in[
i];
163 for (
int i = 0;
i < 4;
i++) {
173 switch (
op->rw.mode) {
207 for (
int i = 0;
i < 4;
i++)
215 for (
int i = 0;
i < 4;
i++)
223 for (
int i = 0;
i < 4;
i++) {
229 for (
int i = 0;
i < 4;
i++) {
238 for (
int i = 0;
i < 4;
i++) {
240 x[
i] =
op->clear.value[
i];
246 for (
int i = 0;
i < 4;
i++)
252 for (
int i = 0;
i < 4;
i++)
253 x[
i] = x[
i].den ?
Q((x[
i].num / x[
i].den) >>
op->shift.amount) : x[
i];
257 const AVRational64 orig[4] = { x[0], x[1], x[2], x[3] };
258 for (
int i = 0;
i < 4;
i++)
259 x[
i] = orig[
op->swizzle.in[
i]];
265 for (
int i = 0;
i < 4;
i++) {
266 x[
i] = x[
i].
den ?
Q(x[
i].num / x[
i].den) : x[
i];
267 if (
op->convert.expand)
274 for (
int i = 0;
i < 4;
i++) {
275 if (
op->dither.y_offset[
i] >= 0 && x[
i].
den)
280 for (
int i = 0;
i < 4;
i++)
284 for (
int i = 0;
i < 4;
i++)
289 const AVRational64 orig[4] = { x[0], x[1], x[2], x[3] };
290 for (
int i = 0;
i < 4;
i++) {
292 for (
int j = 0; j < 4; j++)
299 for (
int i = 0;
i < 4;
i++)
324 return ((
a &
b) & flags_and) | ((
a |
b) & flags_or);
332 for (
int i = 0;
i < 4;
i++) {
353 for (
int n = 0; n < ops->
num_ops; n++) {
365 memcpy(
op->comps.min, prev.
min,
sizeof(prev.
min));
366 memcpy(
op->comps.max, prev.
max,
sizeof(prev.
max));
376 for (
int i = 0;
i <
op->rw.elems;
i++) {
378 switch (
op->rw.mode) {
400 if (
op->rw.filter.op) {
406 for (
int i = 0;
i < 4;
i++) {
408 op->comps.min[
i] = prev.
min[
i];
409 op->comps.max[
i] = prev.
max[
i];
413 for (
int i = 0;
i <
op->rw.elems;
i++)
415 for (
int i = 0;
i < 4;
i++)
420 for (
int i = 0;
i < 4;
i++)
426 for (
int i = 0;
i < 4;
i++) {
428 if (
op->clamp.limit[
i].den)
436 for (
int i = 0;
i < 4;
i++) {
438 op->comps.min[
i] = prev.
min[
i];
439 op->comps.max[
i] = prev.
max[
i];
440 if (
op->dither.y_offset[
i] < 0)
449 for (
int i = 0;
i < 4;
i++) {
450 const int pattern =
op->pack.pattern[
i];
454 op->comps.min[
i] =
Q(0);
455 op->comps.max[
i] =
Q((1ULL << pattern) - 1);
462 for (
int i = 0;
i < 4;
i++) {
463 if (
op->pack.pattern[
i])
472 for (
int i = 0;
i < 4;
i++) {
475 if (
op->clear.value[
i].num == 0)
477 if (
op->clear.value[
i].den == 1)
485 for (
int i = 0;
i < 4;
i++)
486 op->comps.flags[
i] = prev.
flags[
op->swizzle.in[
i]];
489 for (
int i = 0;
i < 4;
i++) {
498 for (
int i = 0;
i < 4;
i++) {
502 for (
int j = 0; j < 4; j++) {
519 if (
op->lin.m[
i][4].num) {
521 if (
op->lin.m[
i][4].den != 1)
532 for (
int i = 0;
i < 4;
i++) {
534 if (
op->scale.factor.den != 1)
536 if (
op->scale.factor.num < 0)
555 bool need_out[4] = {
false,
false,
false,
false };
556 for (
int n = ops->num_ops - 1; n >= 0; n--) {
558 bool need_in[4] = {
false,
false,
false,
false };
560 for (
int i = 0;
i < 4;
i++) {
568 for (
int i = 0;
i <
op->rw.elems;
i++)
570 for (
int i =
op->rw.elems;
i < 4;
i++)
571 need_in[
i] = need_out[
i];
583 for (
int i = 0;
i < 4;
i++)
584 need_in[
i] = need_out[
i];
587 for (
int i = 0;
i < 4 &&
op->pack.pattern[
i];
i++)
588 need_in[0] |= need_out[
i];
591 for (
int i = 0;
i < 4 &&
op->pack.pattern[
i];
i++)
592 need_in[
i] = need_out[0];
595 for (
int i = 0;
i < 4;
i++) {
597 need_in[
i] = need_out[
i];
601 for (
int i = 0;
i < 4;
i++)
602 need_in[
op->swizzle.in[
i]] |= need_out[
i];
605 for (
int i = 0;
i < 4;
i++) {
606 for (
int j = 0; j < 4; j++) {
607 if (
op->lin.m[
i][j].num)
608 need_in[j] |= need_out[
i];
614 memcpy(need_out, need_in,
sizeof(need_in));
642 for (
int i = 0;
i < 4;
i++)
680 for (
int i = 0;
i <
copy->num_ops;
i++) {
684 if (
op->rw.filter.kernel)
720 const int end = ops->
num_ops - count;
722 for (
int i = 0;
i < count;
i++)
770 for (
int i = 0;
i < num_planes;
i++) {
792 for (
int i = 0;
i < 4;
i++) {
793 for (
int j = 0; j < 5; j++) {
823 }
else if (q.
den == 1) {
825 }
else if (q.
num > 1000 || q.
num < -1000 || q.
den > 1000 || q.
den < -1000) {
835 for (
int i = 0;
i < 4;
i++) {
868 if (!
op->rw.filter.op)
884 op->pack.pattern[0],
op->pack.pattern[1],
885 op->pack.pattern[2],
op->pack.pattern[3]);
893 op->swizzle.x,
op->swizzle.y,
op->swizzle.z,
op->swizzle.w);
899 op->convert.expand ?
" (expand)" :
"");
903 1 <<
op->dither.size_log2, 1 <<
op->dither.size_log2,
904 op->dither.y_offset[0],
op->dither.y_offset[1],
905 op->dither.y_offset[2],
op->dither.y_offset[3]);
918 for (
int i = 0;
i < 4;
i++) {
920 for (
int j = 0; j < 5; j++) {
930 if (
op->scale.factor.den != 1)
949 for (
int i = 0;
i < nb_planes;
i++)
950 inorder &= order[
i] ==
i;
955 for (
int i = 0;
i < nb_planes;
i++)
996 if (range_mask &
mask) {
1008 av_log(
log,
lev,
" ('X' unused, 'z' byteswapped, '=' copied, '$' const, '+' integer, '0' zero)\n");
1011 #define DUMMY_SIZE 16
1025 static const int dst_sizes[][2] = {
1033 dst.width = dst_sizes[
i][0];
1034 dst.height = dst_sizes[
i][1];
void ff_sws_op_list_free(SwsOpList **p_ops)
int ff_sws_rw_op_planes(const SwsOp *op)
Return the number of planes involved in a read/write operation.
AVPixelFormat
Pixel format.
it s the only field you need to keep assuming you have a context There is some magic you don t need to care about around this just let it vf default minimum maximum flags name is the option name
SwsOpList * ff_sws_op_list_alloc(void)
static int av_bprint_is_complete(const AVBPrint *buf)
Test if the print buffer is complete (not truncated).
Filter the word “frame” indicates either a video frame or a group of audio as stored in an AVFrame structure Format for each input and each output the list of supported formats For video that means pixel format For audio that means channel sample they are references to shared objects When the negotiation mechanism computes the intersection of the formats supported at each end of a all references to both lists are replaced with a reference to the intersection And when a single format is eventually chosen for a link amongst the remaining all references to the list are updated That means that if a filter requires that its input and output have the same format amongst a supported all it has to do is use a reference to the same list of formats query_formats can leave some formats unset and return AVERROR(EAGAIN) to cause the negotiation mechanism toagain later. That can be used by filters with complex requirements to use the format negotiated on one link to set the formats supported on another. Frame references ownership and permissions
int filter_size
The number of source texels to convolve over for each row.
SwsOpList * ff_sws_op_list_duplicate(const SwsOpList *ops)
Returns a duplicate of ops, or NULL on OOM.
@ SWS_RW_PLANAR
Note: 1-component reads are either SWS_RW_PLANAR or SWS_RW_PACKED, depending on the underlying interp...
static void apply_filter_weights(SwsComps *comps, const SwsComps *prev, const SwsFilterWeights *weights)
static const char *const rw_mode_names[]
void av_bprint_init(AVBPrint *buf, unsigned size_init, unsigned size_max)
static double cb(void *priv, double x, double y)
SwsComps comps_src
Source component metadata associated with pixel values from each corresponding component (in plane/me...
static SwsCompFlags merge_comp_flags(SwsCompFlags a, SwsCompFlags b)
const SwsOp * ff_sws_op_list_input(const SwsOpList *ops)
Returns the input operation for a given op list, or NULL if there is none (e.g.
const AVPixFmtDescriptor * av_pix_fmt_desc_get(enum AVPixelFormat pix_fmt)
int ff_sws_op_list_max_size(const SwsOpList *ops)
Returns the size of the largest pixel type used in ops.
const SwsOpBackend backend_x86
int ff_sws_op_list_append(SwsOpList *ops, SwsOp *op)
These will take over ownership of op and set it to {0}, even on failure.
SwsCompMask ff_sws_comp_mask_q4(const AVRational64 q[4])
Represents a computed filter kernel.
static char describe_comp_flags(SwsCompFlags flags)
void * av_dynarray2_add(void **tab_ptr, int *nb_ptr, size_t elem_size, const uint8_t *elem_data)
Add an element of size elem_size to a dynamic array.
static void desc_plane_order(AVBPrint *bp, int nb_planes, const uint8_t *order)
const AVPixFmtDescriptor * av_pix_fmt_desc_next(const AVPixFmtDescriptor *prev)
Iterate over all pixel format descriptors known to libavutil.
static int enum_ops_fmt(SwsContext *ctx, void *opaque, enum AVPixelFormat src_fmt, enum AVPixelFormat dst_fmt, int(*cb)(SwsContext *ctx, void *opaque, SwsOpList *ops))
static AVRational64 av_min_q64(AVRational64 a, AVRational64 b)
int ff_sws_pixel_type_size(SwsPixelType type)
SwsCompMask ff_sws_comp_mask_needed(const SwsOp *op)
void * av_memdup(const void *p, size_t size)
Duplicate a buffer with av_malloc().
void ff_sws_op_list_print(void *log, int lev, int lev_extra, const SwsOpList *ops)
Print out the contents of an operation list.
static void print_q4(AVBPrint *bp, const AVRational64 q4[4], SwsCompMask mask)
#define SWS_COMP_TEST(mask, X)
const SwsOpBackend *const ff_sws_op_backends[]
static void print_q(AVBPrint *bp, const AVRational64 q)
#define AV_BPRINT_SIZE_AUTOMATIC
bool ff_sws_pixel_type_is_int(SwsPixelType type)
static double val(void *priv, double ch)
it s the only field you need to keep assuming you have a context There is some magic you don t need to care about around this just let it vf type
static int16_t mult(Float11 *f1, Float11 *f2)
trying all byte sequences megabyte in length and selecting the best looking sequence will yield cases to try But first
#define FF_ARRAY_ELEMS(a)
const SwsOpBackend backend_aarch64
#define SWS_OP_NEEDED(op, idx)
@ SWS_FILTER_SCALE
14-bit coefficients are picked to fit comfortably within int16_t for efficient SIMD processing (e....
static AVRational64 ff_sws_pixel_expand(SwsPixelType from, SwsPixelType to)
static int op(uint8_t **dst, const uint8_t *dst_end, GetByteContext *gb, int pixel, int count, int *x, int width, int linesize)
Perform decode operation.
const SwsOpBackend backend_c
Copyright (C) 2025 Niklas Haas.
#define av_assert0(cond)
assert() equivalent, that is always enabled.
int av_cmp_q64(AVRational64 a, AVRational64 b)
Compare two 64-bit rationals.
static AVFormatContext * ctx
uint8_t SwsCompMask
Bit-mask of components.
const SwsOp * ff_sws_op_list_output(const SwsOpList *ops)
Returns the output operation for a given op list, or NULL if there is none.
AVRational64 av_mul_q64(AVRational64 b, AVRational64 c)
Multiply two 64-bit rationals.
bool ff_sws_op_list_is_noop(const SwsOpList *ops)
Returns whether an op list represents a true no-op operation, i.e.
#define av_unreachable(msg)
Asserts that are used as compiler optimization hints depending upon ASSERT_LEVEL and NBDEBUG.
SwsOpType
Copyright (C) 2025 Niklas Haas.
void ff_sws_op_list_remove_at(SwsOpList *ops, int index, int count)
void ff_sws_apply_op_q(const SwsOp *op, AVRational64 x[4])
Apply an operation to an AVRational64.
int src_size
Copy of the parameters used to generate this filter, for reference.
Undefined Behavior In the C some operations are like signed integer dereferencing freed accessing outside allocated Undefined Behavior must not occur in a C it is not safe even if the output of undefined operations is unused The unsafety may seem nit picking but Optimizing compilers have in fact optimized code on the assumption that no undefined Behavior occurs Optimizing code based on wrong assumptions can and has in some cases lead to effects beyond the output of computations The signed integer overflow problem in speed critical code Code which is highly optimized and works with signed integers sometimes has the problem that often the output of the computation does not c
void ff_sws_comp_mask_swizzle(SwsCompMask *mask, const SwsSwizzleOp *swiz)
static void copy(const float *p1, float *p2, const int length)
static int shift(int a, int b)
uint8_t ptrdiff_t const uint8_t ptrdiff_t int intptr_t intptr_t int int16_t * dst
#define i(width, name, range_min, range_max)
int ff_sws_op_list_insert_at(SwsOpList *ops, int index, SwsOp *op)
uint32_t ff_sws_linear_mask(const SwsLinearOp *c)
void ff_sws_op_list_update_comps(SwsOpList *ops)
Infer + propagate known information about components.
64-bit Rational number (pair of numerator and denominator).
void * av_refstruct_ref(void *obj)
Create a new reference to an object managed via this API, i.e.
The reader does not expect b to be semantically here and if the code is changed by maybe adding a a division or other the signedness will almost certainly be mistaken To avoid this confusion a new type was SUINT is the C unsigned type but it holds a signed int to use the same example SUINT a
static void op_uninit(SwsOp *op)
enum AVPixelFormat av_pix_fmt_desc_get_id(const AVPixFmtDescriptor *desc)
char name[16]
Extra metadata about the filter, used to inform the optimizer / range tracker about the filter's beha...
struct SwsReadWriteOp::@574 filter
Filter kernel to apply to each plane while sampling.
void ff_sws_op_desc(AVBPrint *bp, const SwsOp *op)
Describe an operation in human-readable form.
void av_refstruct_unref(void *objp)
Decrement the reference count of the underlying object and automatically free the object if there are...
int ff_sws_op_list_optimize(SwsOpList *ops)
Fuse compatible and eliminate redundant operations, as well as replacing some operations with more ef...
#define av_assert2(cond)
assert() equivalent, that does lie in speed critical code.
static const struct @596 planes[]
static const int weights[]
#define av_assert1(cond)
assert() equivalent, that does not lie in speed critical code.
int ff_sws_enum_op_lists(SwsContext *ctx, void *opaque, enum AVPixelFormat src_fmt, enum AVPixelFormat dst_fmt, int(*cb)(SwsContext *ctx, void *opaque, SwsOpList *ops))
Helper function to enumerate over all possible (optimized) operation lists, under the current set of ...
static LevelCodes lev[4+3+3]
static double bound(const double threshold, const double val)
const SwsOpBackend backend_murder
#define FFSWAP(type, a, b)
void av_bprintf(AVBPrint *buf, const char *fmt,...)
static double av_q2d_64(AVRational64 a)
Convert an AVRational64 to a double.
static AVRational64 av_make_q64(int64_t num, int64_t den)
Create an AVRational64.
const char * ff_sws_pixel_type_name(SwsPixelType type)
static AVRational64 av_max_q64(AVRational64 a, AVRational64 b)
void av_bprint_clear(AVBPrint *buf)
Reset the string to "" but keep internal allocated data.
Descriptor that unambiguously describes how the bits of a pixel are stored in the up to 4 data planes...
static void scale(int *out, const int *in, const int w, const int h, const int shift)
static void ff_sws_pack_op_decode(const SwsOp *op, uint64_t mask[4], int shift[4])
SwsReadWriteMode mode
Examples: rgba = 4x u8 packed yuv444p = 3x u8 rgb565 = 1x u16 <- use SWS_OP_UNPACK to unpack monow = ...
Helper struct for representing a list of operations.
const char * ff_sws_op_type_name(SwsOpType op)
Main external API structure.
static uint32_t BS_FUNC() read(BSCTX *bc, unsigned int n)
Return n bits from the buffer, n has to be in the 0-32 range.
AVRational64 av_add_q64(AVRational64 b, AVRational64 c)
Add two 64-bit rationals.