aboutsummaryrefslogtreecommitdiffstats
path: root/libswscale/output.c
diff options
context:
space:
mode:
authorJames Almer <jamrial@gmail.com>2024-10-06 14:47:52 -0300
committerJames Almer <jamrial@gmail.com>2024-10-12 10:06:09 -0300
commit6cd52c108048b1c71607dec144f547890fcc904f (patch)
treebfdb283f06833613abb8852d971a78898431a5c6 /libswscale/output.c
parent5f1bf3cd655aa04eec69e21ea25cb67fd6f8cb68 (diff)
downloadffmpeg-6cd52c108048b1c71607dec144f547890fcc904f.tar.gz
swscale/output: add AYUV output support
Signed-off-by: James Almer <jamrial@gmail.com>
Diffstat (limited to 'libswscale/output.c')
-rw-r--r--libswscale/output.c109
1 files changed, 77 insertions, 32 deletions
diff --git a/libswscale/output.c b/libswscale/output.c
index c9dfd6f60a..1ccc158d78 100644
--- a/libswscale/output.c
+++ b/libswscale/output.c
@@ -2668,11 +2668,24 @@ yuv2xv36le_X_c(SwsContext *c, const int16_t *lumFilter,
}
}
-static void
-yuv2vuyX_1_c(SwsContext *c, const int16_t *buf0,
- const int16_t *ubuf[2], const int16_t *vbuf[2],
- const int16_t *abuf0, uint8_t *dest, int dstW,
- int uvalpha, int y)
+#define output_pixels(pos, A, Y, U, V) \
+ if (target == AV_PIX_FMT_AYUV) { \
+ dest[pos + 0] = A; \
+ dest[pos + 1] = Y; \
+ dest[pos + 2] = U; \
+ dest[pos + 3] = V; \
+ } else { /* AV_PIX_FMT_VUYA || AV_PIX_FMT_VUYX */ \
+ dest[pos + 0] = V; \
+ dest[pos + 1] = U; \
+ dest[pos + 2] = Y; \
+ dest[pos + 3] = A; \
+ }
+
+static av_always_inline void
+yuv2ayuv_1_c_template(SwsContext *c, const int16_t *buf0,
+ const int16_t *ubuf[2], const int16_t *vbuf[2],
+ const int16_t *abuf0, uint8_t *dest, int dstW,
+ int uvalpha, int y, enum AVPixelFormat target)
{
int hasAlpha = !!abuf0;
int i;
@@ -2697,10 +2710,7 @@ yuv2vuyX_1_c(SwsContext *c, const int16_t *buf0,
A = av_clip_uint8(A);
}
- dest[4 * i ] = V;
- dest[4 * i + 1] = U;
- dest[4 * i + 2] = Y;
- dest[4 * i + 3] = A;
+ output_pixels(i * 4, A, Y, U, V)
}
} else {
for (i = 0; i < dstW; i++) {
@@ -2722,19 +2732,17 @@ yuv2vuyX_1_c(SwsContext *c, const int16_t *buf0,
A = av_clip_uint8(A);
}
- dest[4 * i ] = V;
- dest[4 * i + 1] = U;
- dest[4 * i + 2] = Y;
- dest[4 * i + 3] = A;
+ output_pixels(i * 4, A, Y, U, V)
}
}
}
-static void
-yuv2vuyX_2_c(SwsContext *c, const int16_t *buf[2],
- const int16_t *ubuf[2], const int16_t *vbuf[2],
- const int16_t *abuf[2], uint8_t *dest, int dstW,
- int yalpha, int uvalpha, int y)
+static av_always_inline void
+yuv2ayuv_2_c_template(SwsContext *c, const int16_t *buf[2],
+ const int16_t *ubuf[2], const int16_t *vbuf[2],
+ const int16_t *abuf[2], uint8_t *dest, int dstW,
+ int yalpha, int uvalpha, int y,
+ enum AVPixelFormat target)
{
int hasAlpha = abuf && abuf[0] && abuf[1];
const int16_t *buf0 = buf[0], *buf1 = buf[1],
@@ -2767,19 +2775,17 @@ yuv2vuyX_2_c(SwsContext *c, const int16_t *buf[2],
A = av_clip_uint8(A);
}
- dest[4 * i ] = V;
- dest[4 * i + 1] = U;
- dest[4 * i + 2] = Y;
- dest[4 * i + 3] = A;
+ output_pixels(i * 4, A, Y, U, V)
}
}
-static void
-yuv2vuyX_X_c(SwsContext *c, const int16_t *lumFilter,
- const int16_t **lumSrc, int lumFilterSize,
- const int16_t *chrFilter, const int16_t **chrUSrc,
- const int16_t **chrVSrc, int chrFilterSize,
- const int16_t **alpSrc, uint8_t *dest, int dstW, int y)
+static av_always_inline void
+yuv2ayuv_X_c_template(SwsContext *c, const int16_t *lumFilter,
+ const int16_t **lumSrc, int lumFilterSize,
+ const int16_t *chrFilter, const int16_t **chrUSrc,
+ const int16_t **chrVSrc, int chrFilterSize,
+ const int16_t **alpSrc, uint8_t *dest, int dstW,
+ int y, enum AVPixelFormat target)
{
int i;
@@ -2820,13 +2826,47 @@ yuv2vuyX_X_c(SwsContext *c, const int16_t *lumFilter,
A = av_clip_uint8(A);
}
- dest[4 * i ] = V;
- dest[4 * i + 1] = U;
- dest[4 * i + 2] = Y;
- dest[4 * i + 3] = A;
+ output_pixels(i * 4, A, Y, U, V)
}
}
+#undef output_pixels
+
+#define AYUVPACKEDWRAPPER(name, fmt) \
+static void yuv2 ## name ## _X_c(SwsContext *c, const int16_t *lumFilter, \
+ const int16_t **lumSrc, int lumFilterSize, \
+ const int16_t *chrFilter, const int16_t **chrUSrc, \
+ const int16_t **chrVSrc, int chrFilterSize, \
+ const int16_t **alpSrc, uint8_t *dest, int dstW, \
+ int y) \
+{ \
+ yuv2ayuv_X_c_template(c, lumFilter, lumSrc, lumFilterSize, \
+ chrFilter, chrUSrc, chrVSrc, chrFilterSize, \
+ alpSrc, dest, dstW, y, fmt); \
+} \
+ \
+static void yuv2 ## name ## _2_c(SwsContext *c, const int16_t *buf[2], \
+ const int16_t *ubuf[2], const int16_t *vbuf[2], \
+ const int16_t *abuf[2], uint8_t *dest, int dstW, \
+ int yalpha, int uvalpha, int y) \
+{ \
+ yuv2ayuv_2_c_template(c, buf, ubuf, vbuf, abuf, \
+ dest, dstW, yalpha, uvalpha, y, fmt); \
+} \
+ \
+static void yuv2 ## name ## _1_c(SwsContext *c, const int16_t *buf0, \
+ const int16_t *ubuf[2], const int16_t *vbuf[2], \
+ const int16_t *abuf0, uint8_t *dest, int dstW, \
+ int uvalpha, int y) \
+{ \
+ yuv2ayuv_1_c_template(c, buf0, ubuf, vbuf, \
+ abuf0, dest, dstW, uvalpha, \
+ y, fmt); \
+}
+
+AYUVPACKEDWRAPPER(vuyX, AV_PIX_FMT_VUYX)
+AYUVPACKEDWRAPPER(ayuv, AV_PIX_FMT_AYUV)
+
#define output_pixel(pos, val, bits) \
AV_WL16(pos, av_clip_uintp2(val >> shift, bits) << output_shift);
@@ -3379,6 +3419,11 @@ av_cold void ff_sws_init_output_funcs(SwsContext *c,
case AV_PIX_FMT_AYUV64LE:
*yuv2packedX = yuv2ayuv64le_X_c;
break;
+ case AV_PIX_FMT_AYUV:
+ *yuv2packed1 = yuv2ayuv_1_c;
+ *yuv2packed2 = yuv2ayuv_2_c;
+ *yuv2packedX = yuv2ayuv_X_c;
+ break;
case AV_PIX_FMT_VUYA:
case AV_PIX_FMT_VUYX:
*yuv2packed1 = yuv2vuyX_1_c;