Commit e2f11afe authored by lorenm's avatar lorenm

sse2 & ssse3 versions of dct_quantize.

core2: mmx2=154 sse2=73 ssse3=66 (cycles)
k8: mmx2=179 sse2=149
p4: mmx2=284 sse2=194


git-svn-id: file:///var/local/repositories/ffmpeg/trunk@9003 9553f0bf-9b14-0410-a0b8-cfaf0461ba5b
parent 60a070a3
...@@ -673,6 +673,12 @@ static void denoise_dct_sse2(MpegEncContext *s, DCTELEM *block){ ...@@ -673,6 +673,12 @@ static void denoise_dct_sse2(MpegEncContext *s, DCTELEM *block){
); );
} }
#ifdef HAVE_SSSE3
#define HAVE_SSSE3_BAK
#endif
#undef HAVE_SSSE3
#undef HAVE_SSE2
#undef HAVE_MMX2 #undef HAVE_MMX2
#define RENAME(a) a ## _MMX #define RENAME(a) a ## _MMX
#define RENAMEl(a) a ## _mmx #define RENAMEl(a) a ## _mmx
...@@ -685,12 +691,22 @@ static void denoise_dct_sse2(MpegEncContext *s, DCTELEM *block){ ...@@ -685,12 +691,22 @@ static void denoise_dct_sse2(MpegEncContext *s, DCTELEM *block){
#define RENAMEl(a) a ## _mmx2 #define RENAMEl(a) a ## _mmx2
#include "mpegvideo_mmx_template.c" #include "mpegvideo_mmx_template.c"
#define HAVE_SSE2
#undef RENAME #undef RENAME
#undef RENAMEl #undef RENAMEl
#define RENAME(a) a ## _SSE2 #define RENAME(a) a ## _SSE2
#define RENAMEl(a) a ## _sse2 #define RENAMEl(a) a ## _sse2
#include "mpegvideo_mmx_template.c" #include "mpegvideo_mmx_template.c"
#ifdef HAVE_SSSE3_BAK
#define HAVE_SSSE3
#undef RENAME
#undef RENAMEl
#define RENAME(a) a ## _SSSE3
#define RENAMEl(a) a ## _sse2
#include "mpegvideo_mmx_template.c"
#endif
void MPV_common_init_mmx(MpegEncContext *s) void MPV_common_init_mmx(MpegEncContext *s)
{ {
if (mm_flags & MM_MMX) { if (mm_flags & MM_MMX) {
...@@ -713,6 +729,11 @@ void MPV_common_init_mmx(MpegEncContext *s) ...@@ -713,6 +729,11 @@ void MPV_common_init_mmx(MpegEncContext *s)
} }
if(dct_algo==FF_DCT_AUTO || dct_algo==FF_DCT_MMX){ if(dct_algo==FF_DCT_AUTO || dct_algo==FF_DCT_MMX){
#ifdef HAVE_SSSE3
if(mm_flags & MM_SSSE3){
s->dct_quantize= dct_quantize_SSSE3;
} else
#endif
if(mm_flags & MM_SSE2){ if(mm_flags & MM_SSE2){
s->dct_quantize= dct_quantize_SSE2; s->dct_quantize= dct_quantize_SSE2;
} else if(mm_flags & MM_MMXEXT){ } else if(mm_flags & MM_MMXEXT){
......
...@@ -19,33 +19,77 @@ ...@@ -19,33 +19,77 @@
* License along with FFmpeg; if not, write to the Free Software * License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/ */
#undef MMREG_WIDTH
#undef MM
#undef MOVQ
#undef SPREADW #undef SPREADW
#undef PMAXW #undef PMAXW
#undef PMAX #undef PMAX
#ifdef HAVE_MMX2 #undef SAVE_SIGN
#define SPREADW(a) "pshufw $0, " #a ", " #a " \n\t" #undef RESTORE_SIGN
#define PMAXW(a,b) "pmaxsw " #a ", " #b " \n\t"
#if defined(HAVE_SSE2)
#define MMREG_WIDTH "16"
#define MM "%%xmm"
#define MOVQ "movdqa"
#define SPREADW(a) \
"pshuflw $0, "a", "a" \n\t"\
"punpcklwd "a", "a" \n\t"
#define PMAXW(a,b) "pmaxsw "a", "b" \n\t"
#define PMAX(a,b) \ #define PMAX(a,b) \
"pshufw $0x0E," #a ", " #b " \n\t"\ "movhlps "a", "b" \n\t"\
PMAXW(b, a)\ PMAXW(b, a)\
"pshufw $0x01," #a ", " #b " \n\t"\ "pshuflw $0x0E, "a", "b" \n\t"\
PMAXW(b, a)\
"pshuflw $0x01, "a", "b" \n\t"\
PMAXW(b, a)
#else
#define MMREG_WIDTH "8"
#define MM "%%mm"
#define MOVQ "movq"
#if defined(HAVE_MMX2)
#define SPREADW(a) "pshufw $0, "a", "a" \n\t"
#define PMAXW(a,b) "pmaxsw "a", "b" \n\t"
#define PMAX(a,b) \
"pshufw $0x0E, "a", "b" \n\t"\
PMAXW(b, a)\
"pshufw $0x01, "a", "b" \n\t"\
PMAXW(b, a) PMAXW(b, a)
#else #else
#define SPREADW(a) \ #define SPREADW(a) \
"punpcklwd " #a ", " #a " \n\t"\ "punpcklwd "a", "a" \n\t"\
"punpcklwd " #a ", " #a " \n\t" "punpcklwd "a", "a" \n\t"
#define PMAXW(a,b) \ #define PMAXW(a,b) \
"psubusw " #a ", " #b " \n\t"\ "psubusw "a", "b" \n\t"\
"paddw " #a ", " #b " \n\t" "paddw "a", "b" \n\t"
#define PMAX(a,b) \ #define PMAX(a,b) \
"movq " #a ", " #b " \n\t"\ "movq "a", "b" \n\t"\
"psrlq $32, " #a " \n\t"\ "psrlq $32, "a" \n\t"\
PMAXW(b, a)\ PMAXW(b, a)\
"movq " #a ", " #b " \n\t"\ "movq "a", "b" \n\t"\
"psrlq $16, " #a " \n\t"\ "psrlq $16, "a" \n\t"\
PMAXW(b, a) PMAXW(b, a)
#endif #endif
#endif
#ifdef HAVE_SSSE3
#define SAVE_SIGN(a,b) \
"movdqa "b", "a" \n\t"\
"pabsw "b", "b" \n\t"
#define RESTORE_SIGN(a,b) \
"psignw "a", "b" \n\t"
#else
#define SAVE_SIGN(a,b) \
"pxor "a", "a" \n\t"\
"pcmpgtw "b", "a" \n\t" /* block[i] <= 0 ? 0xFF : 0x00 */\
"pxor "a", "b" \n\t"\
"psubw "a", "b" \n\t" /* ABS(block[i]) */
#define RESTORE_SIGN(a,b) \
"pxor "a", "b" \n\t"\
"psubw "a", "b" \n\t" // out=((ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16)*sign(block[i])
#endif
static int RENAME(dct_quantize)(MpegEncContext *s, static int RENAME(dct_quantize)(MpegEncContext *s,
DCTELEM *block, int n, DCTELEM *block, int n,
...@@ -54,7 +98,7 @@ static int RENAME(dct_quantize)(MpegEncContext *s, ...@@ -54,7 +98,7 @@ static int RENAME(dct_quantize)(MpegEncContext *s,
long last_non_zero_p1; long last_non_zero_p1;
int level=0, q; //=0 is cuz gcc says uninitalized ... int level=0, q; //=0 is cuz gcc says uninitalized ...
const uint16_t *qmat, *bias; const uint16_t *qmat, *bias;
DECLARE_ALIGNED_8(int16_t, temp_block[64]); DECLARE_ALIGNED_16(int16_t, temp_block[64]);
assert((7&(int)(&temp_block[0])) == 0); //did gcc align it correctly? assert((7&(int)(&temp_block[0])) == 0); //did gcc align it correctly?
...@@ -106,98 +150,82 @@ static int RENAME(dct_quantize)(MpegEncContext *s, ...@@ -106,98 +150,82 @@ static int RENAME(dct_quantize)(MpegEncContext *s,
if((s->out_format == FMT_H263 || s->out_format == FMT_H261) && s->mpeg_quant==0){ if((s->out_format == FMT_H263 || s->out_format == FMT_H261) && s->mpeg_quant==0){
asm volatile( asm volatile(
"movd %%"REG_a", %%mm3 \n\t" // last_non_zero_p1 "movd %%"REG_a", "MM"3 \n\t" // last_non_zero_p1
SPREADW(%%mm3) SPREADW(MM"3")
"pxor %%mm7, %%mm7 \n\t" // 0 "pxor "MM"7, "MM"7 \n\t" // 0
"pxor %%mm4, %%mm4 \n\t" // 0 "pxor "MM"4, "MM"4 \n\t" // 0
"movq (%2), %%mm5 \n\t" // qmat[0] MOVQ" (%2), "MM"5 \n\t" // qmat[0]
"pxor %%mm6, %%mm6 \n\t" "pxor "MM"6, "MM"6 \n\t"
"psubw (%3), %%mm6 \n\t" // -bias[0] "psubw (%3), "MM"6 \n\t" // -bias[0]
"mov $-128, %%"REG_a" \n\t" "mov $-128, %%"REG_a" \n\t"
ASMALIGN(4) ASMALIGN(4)
"1: \n\t" "1: \n\t"
"pxor %%mm1, %%mm1 \n\t" // 0 MOVQ" (%1, %%"REG_a"), "MM"0 \n\t" // block[i]
"movq (%1, %%"REG_a"), %%mm0 \n\t" // block[i] SAVE_SIGN(MM"1", MM"0") // ABS(block[i])
"pcmpgtw %%mm0, %%mm1 \n\t" // block[i] <= 0 ? 0xFF : 0x00 "psubusw "MM"6, "MM"0 \n\t" // ABS(block[i]) + bias[0]
"pxor %%mm1, %%mm0 \n\t" "pmulhw "MM"5, "MM"0 \n\t" // (ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16
"psubw %%mm1, %%mm0 \n\t" // ABS(block[i]) "por "MM"0, "MM"4 \n\t"
"psubusw %%mm6, %%mm0 \n\t" // ABS(block[i]) + bias[0] RESTORE_SIGN(MM"1", MM"0") // out=((ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16)*sign(block[i])
"pmulhw %%mm5, %%mm0 \n\t" // (ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16 MOVQ" "MM"0, (%5, %%"REG_a") \n\t"
"por %%mm0, %%mm4 \n\t" "pcmpeqw "MM"7, "MM"0 \n\t" // out==0 ? 0xFF : 0x00
"pxor %%mm1, %%mm0 \n\t" MOVQ" (%4, %%"REG_a"), "MM"1 \n\t"
"psubw %%mm1, %%mm0 \n\t" // out=((ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16)*sign(block[i]) MOVQ" "MM"7, (%1, %%"REG_a") \n\t" // 0
"movq %%mm0, (%5, %%"REG_a") \n\t" "pandn "MM"1, "MM"0 \n\t"
"pcmpeqw %%mm7, %%mm0 \n\t" // out==0 ? 0xFF : 0x00 PMAXW(MM"0", MM"3")
"movq (%4, %%"REG_a"), %%mm1 \n\t" "add $"MMREG_WIDTH", %%"REG_a" \n\t"
"movq %%mm7, (%1, %%"REG_a") \n\t" // 0
"pandn %%mm1, %%mm0 \n\t"
PMAXW(%%mm0, %%mm3)
"add $8, %%"REG_a" \n\t"
" js 1b \n\t" " js 1b \n\t"
PMAX(%%mm3, %%mm0) PMAX(MM"3", MM"0")
"movd %%mm3, %%"REG_a" \n\t" "movd "MM"3, %%"REG_a" \n\t"
"movzb %%al, %%"REG_a" \n\t" // last_non_zero_p1 "movzb %%al, %%"REG_a" \n\t" // last_non_zero_p1
: "+a" (last_non_zero_p1) : "+a" (last_non_zero_p1)
: "r" (block+64), "r" (qmat), "r" (bias), : "r" (block+64), "r" (qmat), "r" (bias),
"r" (inv_zigzag_direct16+64), "r" (temp_block+64) "r" (inv_zigzag_direct16+64), "r" (temp_block+64)
); );
// note the asm is split cuz gcc doesnt like that many operands ...
asm volatile(
"movd %1, %%mm1 \n\t" // max_qcoeff
SPREADW(%%mm1)
"psubusw %%mm1, %%mm4 \n\t"
"packuswb %%mm4, %%mm4 \n\t"
"movd %%mm4, %0 \n\t" // *overflow
: "=g" (*overflow)
: "g" (s->max_qcoeff)
);
}else{ // FMT_H263 }else{ // FMT_H263
asm volatile( asm volatile(
"movd %%"REG_a", %%mm3 \n\t" // last_non_zero_p1 "movd %%"REG_a", "MM"3 \n\t" // last_non_zero_p1
SPREADW(%%mm3) SPREADW(MM"3")
"pxor %%mm7, %%mm7 \n\t" // 0 "pxor "MM"7, "MM"7 \n\t" // 0
"pxor %%mm4, %%mm4 \n\t" // 0 "pxor "MM"4, "MM"4 \n\t" // 0
"mov $-128, %%"REG_a" \n\t" "mov $-128, %%"REG_a" \n\t"
ASMALIGN(4) ASMALIGN(4)
"1: \n\t" "1: \n\t"
"pxor %%mm1, %%mm1 \n\t" // 0 MOVQ" (%1, %%"REG_a"), "MM"0 \n\t" // block[i]
"movq (%1, %%"REG_a"), %%mm0 \n\t" // block[i] SAVE_SIGN(MM"1", MM"0") // ABS(block[i])
"pcmpgtw %%mm0, %%mm1 \n\t" // block[i] <= 0 ? 0xFF : 0x00 MOVQ" (%3, %%"REG_a"), "MM"6 \n\t" // bias[0]
"pxor %%mm1, %%mm0 \n\t" "paddusw "MM"6, "MM"0 \n\t" // ABS(block[i]) + bias[0]
"psubw %%mm1, %%mm0 \n\t" // ABS(block[i]) MOVQ" (%2, %%"REG_a"), "MM"5 \n\t" // qmat[i]
"movq (%3, %%"REG_a"), %%mm6 \n\t" // bias[0] "pmulhw "MM"5, "MM"0 \n\t" // (ABS(block[i])*qmat[0] + bias[0]*qmat[0])>>16
"paddusw %%mm6, %%mm0 \n\t" // ABS(block[i]) + bias[0] "por "MM"0, "MM"4 \n\t"
"movq (%2, %%"REG_a"), %%mm5 \n\t" // qmat[i] RESTORE_SIGN(MM"1", MM"0") // out=((ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16)*sign(block[i])
"pmulhw %%mm5, %%mm0 \n\t" // (ABS(block[i])*qmat[0] + bias[0]*qmat[0])>>16 MOVQ" "MM"0, (%5, %%"REG_a") \n\t"
"por %%mm0, %%mm4 \n\t" "pcmpeqw "MM"7, "MM"0 \n\t" // out==0 ? 0xFF : 0x00
"pxor %%mm1, %%mm0 \n\t" MOVQ" (%4, %%"REG_a"), "MM"1 \n\t"
"psubw %%mm1, %%mm0 \n\t" // out=((ABS(block[i])*qmat[0] - bias[0]*qmat[0])>>16)*sign(block[i]) MOVQ" "MM"7, (%1, %%"REG_a") \n\t" // 0
"movq %%mm0, (%5, %%"REG_a") \n\t" "pandn "MM"1, "MM"0 \n\t"
"pcmpeqw %%mm7, %%mm0 \n\t" // out==0 ? 0xFF : 0x00 PMAXW(MM"0", MM"3")
"movq (%4, %%"REG_a"), %%mm1 \n\t" "add $"MMREG_WIDTH", %%"REG_a" \n\t"
"movq %%mm7, (%1, %%"REG_a") \n\t" // 0
"pandn %%mm1, %%mm0 \n\t"
PMAXW(%%mm0, %%mm3)
"add $8, %%"REG_a" \n\t"
" js 1b \n\t" " js 1b \n\t"
PMAX(%%mm3, %%mm0) PMAX(MM"3", MM"0")
"movd %%mm3, %%"REG_a" \n\t" "movd "MM"3, %%"REG_a" \n\t"
"movzb %%al, %%"REG_a" \n\t" // last_non_zero_p1 "movzb %%al, %%"REG_a" \n\t" // last_non_zero_p1
: "+a" (last_non_zero_p1) : "+a" (last_non_zero_p1)
: "r" (block+64), "r" (qmat+64), "r" (bias+64), : "r" (block+64), "r" (qmat+64), "r" (bias+64),
"r" (inv_zigzag_direct16+64), "r" (temp_block+64) "r" (inv_zigzag_direct16+64), "r" (temp_block+64)
); );
// note the asm is split cuz gcc doesnt like that many operands ... }
asm volatile( asm volatile(
"movd %1, %%mm1 \n\t" // max_qcoeff "movd %1, "MM"1 \n\t" // max_qcoeff
SPREADW(%%mm1) SPREADW(MM"1")
"psubusw %%mm1, %%mm4 \n\t" "psubusw "MM"1, "MM"4 \n\t"
"packuswb %%mm4, %%mm4 \n\t" "packuswb "MM"4, "MM"4 \n\t"
"movd %%mm4, %0 \n\t" // *overflow #ifdef HAVE_SSE2
"packuswb "MM"4, "MM"4 \n\t"
#endif
"movd "MM"4, %0 \n\t" // *overflow
: "=g" (*overflow) : "=g" (*overflow)
: "g" (s->max_qcoeff) : "g" (s->max_qcoeff)
); );
}
if(s->mb_intra) block[0]= level; if(s->mb_intra) block[0]= level;
else block[0]= temp_block[0]; else block[0]= temp_block[0];
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment