aboutsummaryrefslogtreecommitdiffstats
path: root/libswscale
diff options
context:
space:
mode:
authorKieran Kunhya <kieran@kunhya.com>2011-10-09 16:20:48 +0100
committerRonald S. Bultje <rsbultje@gmail.com>2011-10-22 10:35:14 -0700
commit7fbbf9529397756a31850fe37036f026f34f80fc (patch)
tree152cae6ebdf3271cd8f971acc9bc21010259c0e5 /libswscale
parent109f62e8f89d25ec16026e4076b5407fc07c00e7 (diff)
downloadffmpeg-7fbbf9529397756a31850fe37036f026f34f80fc.tar.gz
yuv2planeX10 SIMD
Signed-off-by: Ronald S. Bultje <rsbultje@gmail.com>
Diffstat (limited to 'libswscale')
-rw-r--r--libswscale/x86/scale.asm77
-rw-r--r--libswscale/x86/swscale_mmx.c15
2 files changed, 91 insertions, 1 deletions
diff --git a/libswscale/x86/scale.asm b/libswscale/x86/scale.asm
index d35589419c..05e2d96fe6 100644
--- a/libswscale/x86/scale.asm
+++ b/libswscale/x86/scale.asm
@@ -1,6 +1,7 @@
;******************************************************************************
-;* x86-optimized horizontal line scaling functions
+;* x86-optimized horizontal/vertical line scaling functions
;* Copyright (c) 2011 Ronald S. Bultje <rsbultje@gmail.com>
+;* Kieran Kunhya <kieran@kunhya.com>
;*
;* This file is part of Libav.
;*
@@ -28,6 +29,8 @@ max_19bit_int: times 4 dd 0x7ffff
max_19bit_flt: times 4 dd 524287.0
minshort: times 8 dw 0x8000
unicoeff: times 4 dd 0x20000000
+yuv2yuvX_10_start: times 4 dd 0x10000
+yuv2yuvX_10_upper: times 8 dw 0x3ff
SECTION .text
@@ -427,3 +430,75 @@ INIT_XMM
SCALE_FUNCS2 sse2, 6, 7, 8
SCALE_FUNCS2 ssse3, 6, 6, 8
SCALE_FUNCS2 sse4, 6, 6, 8
+
+;-----------------------------------------------------------------------------
+; vertical line scaling
+;
+; void yuv2plane1_<output_size>_<opt>(const int16_t *src, uint8_t *dst, int dstW,
+; const uint8_t *dither, int offset)
+; and
+; void yuv2planeX_<output_size>_<opt>(const int16_t *filter, int filterSize,
+; const int16_t **src, uint8_t *dst, int dstW,
+; const uint8_t *dither, int offset)
+;
+; Scale one or $filterSize lines of source data to generate one line of output
+; data. The input is 15-bit in int16_t if $output_size is [8,10] and 19-bit in
+; int32_t if $output_size is 16. $filter is 12-bits. $filterSize is a multiple
+; of 2. $offset is either 0 or 3. $dither holds 8 values.
+;-----------------------------------------------------------------------------
+
+%macro yuv2planeX10 1
+
+%ifdef ARCH_X86_32
+%define cntr_reg r1
+%else
+%define cntr_reg r11
+%endif
+
+cglobal yuv2planeX10_%1, 7, 7
+ xor r5, r5
+.pixelloop
+ mova m1, [yuv2yuvX_10_start]
+ mova m2, m1
+ movsxdifnidn cntr_reg, r1d
+.filterloop
+ pxor m0, m0
+
+ mov r6, [r2+gprsize*cntr_reg-2*gprsize]
+ mova m3, [r6+r5]
+
+ mov r6, [r2+gprsize*cntr_reg-gprsize]
+ mova m4, [r6+r5]
+
+ punpcklwd m5, m3, m4
+ punpckhwd m3, m4
+
+ movd m0, [r0+2*cntr_reg-4]
+ SPLATD m0, m0
+
+ pmaddwd m5, m0
+ pmaddwd m3, m0
+
+ paddd m2, m5
+ paddd m1, m3
+
+ sub cntr_reg, 2
+ jg .filterloop
+
+ psrad m2, 17
+ psrad m1, 17
+
+ packusdw m2, m1
+ pminsw m2, [yuv2yuvX_10_upper]
+ mova [r3+r5], m2
+
+ add r5, mmsize
+ sub r4d, mmsize/2
+ jg .pixelloop
+ REP_RET
+%endmacro
+
+INIT_XMM
+yuv2planeX10 sse4
+INIT_AVX
+yuv2planeX10 avx
diff --git a/libswscale/x86/swscale_mmx.c b/libswscale/x86/swscale_mmx.c
index dd7aea1492..3c0632dca5 100644
--- a/libswscale/x86/swscale_mmx.c
+++ b/libswscale/x86/swscale_mmx.c
@@ -211,6 +211,14 @@ SCALE_FUNCS_SSE(sse2);
SCALE_FUNCS_SSE(ssse3);
SCALE_FUNCS_SSE(sse4);
+extern void ff_yuv2planeX10_sse4(const int16_t *filter, int filterSize,
+ const int16_t **src, uint8_t *dest, int dstW,
+ const uint8_t *dither, int offset);
+
+extern void ff_yuv2planeX10_avx(const int16_t *filter, int filterSize,
+ const int16_t **src, uint8_t *dest, int dstW,
+ const uint8_t *dither, int offset);
+
void ff_sws_init_swScale_mmx(SwsContext *c)
{
int cpu_flags = av_get_cpu_flags();
@@ -270,6 +278,13 @@ void ff_sws_init_swScale_mmx(SwsContext *c)
/* Xto15 don't need special sse4 functions */
ASSIGN_SSE_SCALE_FUNC(c->hyScale, c->hLumFilterSize, sse4, ssse3);
ASSIGN_SSE_SCALE_FUNC(c->hcScale, c->hChrFilterSize, sse4, ssse3);
+ if (c->dstBpc == 10 && !isBE(c->dstFormat))
+ c->yuv2planeX = ff_yuv2planeX10_sse4;
+ }
+
+ if (cpu_flags & AV_CPU_FLAG_AVX) {
+ if (c->dstBpc == 10 && !isBE(c->dstFormat))
+ c->yuv2planeX = ff_yuv2planeX10_avx;
}
#endif
}