/* * Copyright (c) 2010 The WebM project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include <math.h> #include <stdlib.h> #include "vpx_scale/yv12config.h" #include "pragmas.h" #define VP8_FILTER_WEIGHT 128 #define VP8_FILTER_SHIFT 7 /* static constants */ __declspec(align(16)) const static short Blur[48] = { 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 64, 64, 64, 64, 64, 64, 64, 64, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 0, 0, 0, 0, 0, 0, 0, 0, }; #define RD __declspec(align(16)) __int64 rd = 0x0040004000400040; #define R4D2 __declspec(align(16)) __int64 rd42[2] = {0x0004000400040004,0x0004000400040004}; #ifndef RELOCATEABLE const static RD; const static R4D2; #endif /* external references */ extern double vp8_gaussian(double sigma, double mu, double x); extern short vp8_rv[]; extern int vp8_q2mbl(int x) ; void vp8_post_proc_down_and_across_mmx ( unsigned char *src_ptr, unsigned char *dst_ptr, int src_pixels_per_line, int dst_pixels_per_line, int rows, int cols, int flimit ) { #ifdef RELOCATEABLE RD R4D2 #endif __asm { push ebx lea ebx, Blur movd mm2, flimit punpcklwd mm2, mm2 punpckldq mm2, mm2 mov esi, src_ptr mov edi, dst_ptr mov ecx, DWORD PTR rows mov eax, src_pixels_per_line ; destination pitch? pxor mm0, mm0 ; mm0 = 00000000 nextrow: xor edx, edx ; clear out edx for use as loop counter nextcol: pxor mm7, mm7 ; mm7 = 00000000 movq mm6, [ebx + 32 ] ; mm6 = kernel 2 taps movq mm3, [esi] ; mm4 = r0 p0..p7 punpcklbw mm3, mm0 ; mm3 = p0..p3 movq mm1, mm3 ; mm1 = p0..p3 pmullw mm3, mm6 ; mm3 *= kernel 2 modifiers movq mm6, [ebx + 48] ; mm6 = kernel 3 taps movq mm5, [esi + eax] ; mm4 = r1 p0..p7 punpcklbw mm5, mm0 ; mm5 = r1 p0..p3 pmullw mm6, mm5 ; mm6 *= p0..p3 * kernel 3 modifiers paddusw mm3, mm6 ; mm3 += mm6 ; thresholding movq mm7, mm1 ; mm7 = r0 p0..p3 psubusw mm7, mm5 ; mm7 = r0 p0..p3 - r1 p0..p3 psubusw mm5, mm1 ; mm5 = r1 p0..p3 - r0 p0..p3 paddusw mm7, mm5 ; mm7 = abs(r0 p0..p3 - r1 p0..p3) pcmpgtw mm7, mm2 movq mm6, [ebx + 64 ] ; mm6 = kernel 4 modifiers movq mm5, [esi + 2*eax] ; mm4 = r2 p0..p7 punpcklbw mm5, mm0 ; mm5 = r2 p0..p3 pmullw mm6, mm5 ; mm5 *= kernel 4 modifiers paddusw mm3, mm6 ; mm3 += mm5 ; thresholding movq mm6, mm1 ; mm6 = r0 p0..p3 psubusw mm6, mm5 ; mm6 = r0 p0..p3 - r2 p0..p3 psubusw mm5, mm1 ; mm5 = r2 p0..p3 - r2 p0..p3 paddusw mm6, mm5 ; mm6 = abs(r0 p0..p3 - r2 p0..p3) pcmpgtw mm6, mm2 por mm7, mm6 ; accumulate thresholds neg eax movq mm6, [ebx ] ; kernel 0 taps movq mm5, [esi+2*eax] ; mm4 = r-2 p0..p7 punpcklbw mm5, mm0 ; mm5 = r-2 p0..p3 pmullw mm6, mm5 ; mm5 *= kernel 0 modifiers paddusw mm3, mm6 ; mm3 += mm5 ; thresholding movq mm6, mm1 ; mm6 = r0 p0..p3 psubusw mm6, mm5 ; mm6 = p0..p3 - r-2 p0..p3 psubusw mm5, mm1 ; mm5 = r-2 p0..p3 - p0..p3 paddusw mm6, mm5 ; mm6 = abs(r0 p0..p3 - r-2 p0..p3) pcmpgtw mm6, mm2 por mm7, mm6 ; accumulate thresholds movq mm6, [ebx + 16] ; kernel 1 taps movq mm4, [esi+eax] ; mm4 = r-1 p0..p7 punpcklbw mm4, mm0 ; mm4 = r-1 p0..p3 pmullw mm6, mm4 ; mm4 *= kernel 1 modifiers. paddusw mm3, mm6 ; mm3 += mm5 ; thresholding movq mm6, mm1 ; mm6 = r0 p0..p3 psubusw mm6, mm4 ; mm6 = p0..p3 - r-2 p0..p3 psubusw mm4, mm1 ; mm5 = r-1 p0..p3 - p0..p3 paddusw mm6, mm4 ; mm6 = abs(r0 p0..p3 - r-1 p0..p3) pcmpgtw mm6, mm2 por mm7, mm6 ; accumulate thresholds paddusw mm3, rd ; mm3 += round value psraw mm3, VP8_FILTER_SHIFT ; mm3 /= 128 pand mm1, mm7 ; mm1 select vals > thresh from source pandn mm7, mm3 ; mm7 select vals < thresh from blurred result paddusw mm1, mm7 ; combination packuswb mm1, mm0 ; pack to bytes movd [edi], mm1 ; neg eax ; pitch is positive add esi, 4 add edi, 4 add edx, 4 cmp edx, cols jl nextcol // done with the all cols, start the across filtering in place sub esi, edx sub edi, edx push eax xor edx, edx mov eax, [edi-4]; acrossnextcol: pxor mm7, mm7 ; mm7 = 00000000 movq mm6, [ebx + 32 ] ; movq mm4, [edi+edx] ; mm4 = p0..p7 movq mm3, mm4 ; mm3 = p0..p7 punpcklbw mm3, mm0 ; mm3 = p0..p3 movq mm1, mm3 ; mm1 = p0..p3 pmullw mm3, mm6 ; mm3 *= kernel 2 modifiers movq mm6, [ebx + 48] psrlq mm4, 8 ; mm4 = p1..p7 movq mm5, mm4 ; mm5 = p1..p7 punpcklbw mm5, mm0 ; mm5 = p1..p4 pmullw mm6, mm5 ; mm6 *= p1..p4 * kernel 3 modifiers paddusw mm3, mm6 ; mm3 += mm6 ; thresholding movq mm7, mm1 ; mm7 = p0..p3 psubusw mm7, mm5 ; mm7 = p0..p3 - p1..p4 psubusw mm5, mm1 ; mm5 = p1..p4 - p0..p3 paddusw mm7, mm5 ; mm7 = abs(p0..p3 - p1..p4) pcmpgtw mm7, mm2 movq mm6, [ebx + 64 ] psrlq mm4, 8 ; mm4 = p2..p7 movq mm5, mm4 ; mm5 = p2..p7 punpcklbw mm5, mm0 ; mm5 = p2..p5 pmullw mm6, mm5 ; mm5 *= kernel 4 modifiers paddusw mm3, mm6 ; mm3 += mm5 ; thresholding movq mm6, mm1 ; mm6 = p0..p3 psubusw mm6, mm5 ; mm6 = p0..p3 - p1..p4 psubusw mm5, mm1 ; mm5 = p1..p4 - p0..p3 paddusw mm6, mm5 ; mm6 = abs(p0..p3 - p1..p4) pcmpgtw mm6, mm2 por mm7, mm6 ; accumulate thresholds movq mm6, [ebx ] movq mm4, [edi+edx-2] ; mm4 = p-2..p5 movq mm5, mm4 ; mm5 = p-2..p5 punpcklbw mm5, mm0 ; mm5 = p-2..p1 pmullw mm6, mm5 ; mm5 *= kernel 0 modifiers paddusw mm3, mm6 ; mm3 += mm5 ; thresholding movq mm6, mm1 ; mm6 = p0..p3 psubusw mm6, mm5 ; mm6 = p0..p3 - p1..p4 psubusw mm5, mm1 ; mm5 = p1..p4 - p0..p3 paddusw mm6, mm5 ; mm6 = abs(p0..p3 - p1..p4) pcmpgtw mm6, mm2 por mm7, mm6 ; accumulate thresholds movq mm6, [ebx + 16] psrlq mm4, 8 ; mm4 = p-1..p5 punpcklbw mm4, mm0 ; mm4 = p-1..p2 pmullw mm6, mm4 ; mm4 *= kernel 1 modifiers. paddusw mm3, mm6 ; mm3 += mm5 ; thresholding movq mm6, mm1 ; mm6 = p0..p3 psubusw mm6, mm4 ; mm6 = p0..p3 - p1..p4 psubusw mm4, mm1 ; mm5 = p1..p4 - p0..p3 paddusw mm6, mm4 ; mm6 = abs(p0..p3 - p1..p4) pcmpgtw mm6, mm2 por mm7, mm6 ; accumulate thresholds paddusw mm3, rd ; mm3 += round value psraw mm3, VP8_FILTER_SHIFT ; mm3 /= 128 pand mm1, mm7 ; mm1 select vals > thresh from source pandn mm7, mm3 ; mm7 select vals < thresh from blurred result paddusw mm1, mm7 ; combination packuswb mm1, mm0 ; pack to bytes mov DWORD PTR [edi+edx-4], eax ; store previous four bytes movd eax, mm1 add edx, 4 cmp edx, cols jl acrossnextcol; mov DWORD PTR [edi+edx-4], eax pop eax // done with this rwo add esi, eax ; next line mov eax, dst_pixels_per_line ; destination pitch? add edi, eax ; next destination mov eax, src_pixels_per_line ; destination pitch? dec ecx ; decrement count jnz nextrow ; next row pop ebx } } void vp8_post_proc_down_and_across_xmm ( unsigned char *src_ptr, unsigned char *dst_ptr, int src_pixels_per_line, int dst_pixels_per_line, int rows, int cols, int flimit ) { #ifdef RELOCATEABLE R4D2 #endif __asm { movd xmm2, flimit punpcklwd xmm2, xmm2 punpckldq xmm2, xmm2 punpcklqdq xmm2, xmm2 mov esi, src_ptr mov edi, dst_ptr mov ecx, DWORD PTR rows mov eax, src_pixels_per_line ; destination pitch? pxor xmm0, xmm0 ; mm0 = 00000000 nextrow: xor edx, edx ; clear out edx for use as loop counter nextcol: movq xmm3, QWORD PTR [esi] ; mm4 = r0 p0..p7 punpcklbw xmm3, xmm0 ; mm3 = p0..p3 movdqa xmm1, xmm3 ; mm1 = p0..p3 psllw xmm3, 2 ; movq xmm5, QWORD PTR [esi + eax] ; mm4 = r1 p0..p7 punpcklbw xmm5, xmm0 ; mm5 = r1 p0..p3 paddusw xmm3, xmm5 ; mm3 += mm6 ; thresholding movdqa xmm7, xmm1 ; mm7 = r0 p0..p3 psubusw xmm7, xmm5 ; mm7 = r0 p0..p3 - r1 p0..p3 psubusw xmm5, xmm1 ; mm5 = r1 p0..p3 - r0 p0..p3 paddusw xmm7, xmm5 ; mm7 = abs(r0 p0..p3 - r1 p0..p3) pcmpgtw xmm7, xmm2 movq xmm5, QWORD PTR [esi + 2*eax] ; mm4 = r2 p0..p7 punpcklbw xmm5, xmm0 ; mm5 = r2 p0..p3 paddusw xmm3, xmm5 ; mm3 += mm5 ; thresholding movdqa xmm6, xmm1 ; mm6 = r0 p0..p3 psubusw xmm6, xmm5 ; mm6 = r0 p0..p3 - r2 p0..p3 psubusw xmm5, xmm1 ; mm5 = r2 p0..p3 - r2 p0..p3 paddusw xmm6, xmm5 ; mm6 = abs(r0 p0..p3 - r2 p0..p3) pcmpgtw xmm6, xmm2 por xmm7, xmm6 ; accumulate thresholds neg eax movq xmm5, QWORD PTR [esi+2*eax] ; mm4 = r-2 p0..p7 punpcklbw xmm5, xmm0 ; mm5 = r-2 p0..p3 paddusw xmm3, xmm5 ; mm3 += mm5 ; thresholding movdqa xmm6, xmm1 ; mm6 = r0 p0..p3 psubusw xmm6, xmm5 ; mm6 = p0..p3 - r-2 p0..p3 psubusw xmm5, xmm1 ; mm5 = r-2 p0..p3 - p0..p3 paddusw xmm6, xmm5 ; mm6 = abs(r0 p0..p3 - r-2 p0..p3) pcmpgtw xmm6, xmm2 por xmm7, xmm6 ; accumulate thresholds movq xmm4, QWORD PTR [esi+eax] ; mm4 = r-1 p0..p7 punpcklbw xmm4, xmm0 ; mm4 = r-1 p0..p3 paddusw xmm3, xmm4 ; mm3 += mm5 ; thresholding movdqa xmm6, xmm1 ; mm6 = r0 p0..p3 psubusw xmm6, xmm4 ; mm6 = p0..p3 - r-2 p0..p3 psubusw xmm4, xmm1 ; mm5 = r-1 p0..p3 - p0..p3 paddusw xmm6, xmm4 ; mm6 = abs(r0 p0..p3 - r-1 p0..p3) pcmpgtw xmm6, xmm2 por xmm7, xmm6 ; accumulate thresholds paddusw xmm3, rd42 ; mm3 += round value psraw xmm3, 3 ; mm3 /= 8 pand xmm1, xmm7 ; mm1 select vals > thresh from source pandn xmm7, xmm3 ; mm7 select vals < thresh from blurred result paddusw xmm1, xmm7 ; combination packuswb xmm1, xmm0 ; pack to bytes movq QWORD PTR [edi], xmm1 ; neg eax ; pitch is positive add esi, 8 add edi, 8 add edx, 8 cmp edx, cols jl nextcol // done with the all cols, start the across filtering in place sub esi, edx sub edi, edx xor edx, edx movq mm0, QWORD PTR [edi-8]; acrossnextcol: movq xmm7, QWORD PTR [edi +edx -2] movd xmm4, DWORD PTR [edi +edx +6] pslldq xmm4, 8 por xmm4, xmm7 movdqa xmm3, xmm4 psrldq xmm3, 2 punpcklbw xmm3, xmm0 ; mm3 = p0..p3 movdqa xmm1, xmm3 ; mm1 = p0..p3 psllw xmm3, 2 movdqa xmm5, xmm4 psrldq xmm5, 3 punpcklbw xmm5, xmm0 ; mm5 = p1..p4 paddusw xmm3, xmm5 ; mm3 += mm6 ; thresholding movdqa xmm7, xmm1 ; mm7 = p0..p3 psubusw xmm7, xmm5 ; mm7 = p0..p3 - p1..p4 psubusw xmm5, xmm1 ; mm5 = p1..p4 - p0..p3 paddusw xmm7, xmm5 ; mm7 = abs(p0..p3 - p1..p4) pcmpgtw xmm7, xmm2 movdqa xmm5, xmm4 psrldq xmm5, 4 punpcklbw xmm5, xmm0 ; mm5 = p2..p5 paddusw xmm3, xmm5 ; mm3 += mm5 ; thresholding movdqa xmm6, xmm1 ; mm6 = p0..p3 psubusw xmm6, xmm5 ; mm6 = p0..p3 - p1..p4 psubusw xmm5, xmm1 ; mm5 = p1..p4 - p0..p3 paddusw xmm6, xmm5 ; mm6 = abs(p0..p3 - p1..p4) pcmpgtw xmm6, xmm2 por xmm7, xmm6 ; accumulate thresholds movdqa xmm5, xmm4 ; mm5 = p-2..p5 punpcklbw xmm5, xmm0 ; mm5 = p-2..p1 paddusw xmm3, xmm5 ; mm3 += mm5 ; thresholding movdqa xmm6, xmm1 ; mm6 = p0..p3 psubusw xmm6, xmm5 ; mm6 = p0..p3 - p1..p4 psubusw xmm5, xmm1 ; mm5 = p1..p4 - p0..p3 paddusw xmm6, xmm5 ; mm6 = abs(p0..p3 - p1..p4) pcmpgtw xmm6, xmm2 por xmm7, xmm6 ; accumulate thresholds psrldq xmm4, 1 ; mm4 = p-1..p5 punpcklbw xmm4, xmm0 ; mm4 = p-1..p2 paddusw xmm3, xmm4 ; mm3 += mm5 ; thresholding movdqa xmm6, xmm1 ; mm6 = p0..p3 psubusw xmm6, xmm4 ; mm6 = p0..p3 - p1..p4 psubusw xmm4, xmm1 ; mm5 = p1..p4 - p0..p3 paddusw xmm6, xmm4 ; mm6 = abs(p0..p3 - p1..p4) pcmpgtw xmm6, xmm2 por xmm7, xmm6 ; accumulate thresholds paddusw xmm3, rd42 ; mm3 += round value psraw xmm3, 3 ; mm3 /= 8 pand xmm1, xmm7 ; mm1 select vals > thresh from source pandn xmm7, xmm3 ; mm7 select vals < thresh from blurred result paddusw xmm1, xmm7 ; combination packuswb xmm1, xmm0 ; pack to bytes movq QWORD PTR [edi+edx-8], mm0 ; store previous four bytes movdq2q mm0, xmm1 add edx, 8 cmp edx, cols jl acrossnextcol; // last 8 pixels movq QWORD PTR [edi+edx-8], mm0 // done with this rwo add esi, eax ; next line mov eax, dst_pixels_per_line ; destination pitch? add edi, eax ; next destination mov eax, src_pixels_per_line ; destination pitch? dec ecx ; decrement count jnz nextrow ; next row } } void vp8_mbpost_proc_down_mmx(unsigned char *dst, int pitch, int rows, int cols, int flimit) { int c, i; __declspec(align(16)) int flimit2[2]; __declspec(align(16)) unsigned char d[16][8]; flimit = vp8_q2mbl(flimit); for (i = 0; i < 2; i++) flimit2[i] = flimit; rows += 8; for (c = 0; c < cols; c += 4) { unsigned char *s = &dst[c]; __asm { mov esi, s ; pxor mm0, mm0 ; mov eax, pitch ; neg eax // eax = -pitch lea esi, [esi + eax*8]; // edi = s[-pitch*8] neg eax pxor mm5, mm5 pxor mm6, mm6 ; pxor mm7, mm7 ; mov edi, esi mov ecx, 15 ; loop_initvar: movd mm1, DWORD PTR [edi]; punpcklbw mm1, mm0 ; paddw mm5, mm1 ; pmullw mm1, mm1 ; movq mm2, mm1 ; punpcklwd mm1, mm0 ; punpckhwd mm2, mm0 ; paddd mm6, mm1 ; paddd mm7, mm2 ; lea edi, [edi+eax] ; dec ecx jne loop_initvar //save the var and sum xor edx, edx loop_row: movd mm1, DWORD PTR [esi] // [s-pitch*8] movd mm2, DWORD PTR [edi] // [s+pitch*7] punpcklbw mm1, mm0 punpcklbw mm2, mm0 paddw mm5, mm2 psubw mm5, mm1 pmullw mm2, mm2 movq mm4, mm2 punpcklwd mm2, mm0 punpckhwd mm4, mm0 paddd mm6, mm2 paddd mm7, mm4 pmullw mm1, mm1 movq mm2, mm1 punpcklwd mm1, mm0 psubd mm6, mm1 punpckhwd mm2, mm0 psubd mm7, mm2 movq mm3, mm6 pslld mm3, 4 psubd mm3, mm6 movq mm1, mm5 movq mm4, mm5 pmullw mm1, mm1 pmulhw mm4, mm4 movq mm2, mm1 punpcklwd mm1, mm4 punpckhwd mm2, mm4 movq mm4, mm7 pslld mm4, 4 psubd mm4, mm7 psubd mm3, mm1 psubd mm4, mm2 psubd mm3, flimit2 psubd mm4, flimit2 psrad mm3, 31 psrad mm4, 31 packssdw mm3, mm4 packsswb mm3, mm0 movd mm1, DWORD PTR [esi+eax*8] movq mm2, mm1 punpcklbw mm1, mm0 paddw mm1, mm5 mov ecx, edx and ecx, 127 movq mm4, vp8_rv[ecx*2] paddw mm1, mm4 //paddw xmm1, eight8s psraw mm1, 4 packuswb mm1, mm0 pand mm1, mm3 pandn mm3, mm2 por mm1, mm3 and ecx, 15 movd DWORD PTR d[ecx*4], mm1 mov ecx, edx sub ecx, 8 and ecx, 15 movd mm1, DWORD PTR d[ecx*4] movd [esi], mm1 lea esi, [esi+eax] lea edi, [edi+eax] add edx, 1 cmp edx, rows jl loop_row } } } void vp8_mbpost_proc_down_xmm(unsigned char *dst, int pitch, int rows, int cols, int flimit) { int c, i; __declspec(align(16)) int flimit4[4]; __declspec(align(16)) unsigned char d[16][8]; flimit = vp8_q2mbl(flimit); for (i = 0; i < 4; i++) flimit4[i] = flimit; rows += 8; for (c = 0; c < cols; c += 8) { unsigned char *s = &dst[c]; __asm { mov esi, s ; pxor xmm0, xmm0 ; mov eax, pitch ; neg eax // eax = -pitch lea esi, [esi + eax*8]; // edi = s[-pitch*8] neg eax pxor xmm5, xmm5 pxor xmm6, xmm6 ; pxor xmm7, xmm7 ; mov edi, esi mov ecx, 15 ; loop_initvar: movq xmm1, QWORD PTR [edi]; punpcklbw xmm1, xmm0 ; paddw xmm5, xmm1 ; pmullw xmm1, xmm1 ; movdqa xmm2, xmm1 ; punpcklwd xmm1, xmm0 ; punpckhwd xmm2, xmm0 ; paddd xmm6, xmm1 ; paddd xmm7, xmm2 ; lea edi, [edi+eax] ; dec ecx jne loop_initvar //save the var and sum xor edx, edx loop_row: movq xmm1, QWORD PTR [esi] // [s-pitch*8] movq xmm2, QWORD PTR [edi] // [s+pitch*7] punpcklbw xmm1, xmm0 punpcklbw xmm2, xmm0 paddw xmm5, xmm2 psubw xmm5, xmm1 pmullw xmm2, xmm2 movdqa xmm4, xmm2 punpcklwd xmm2, xmm0 punpckhwd xmm4, xmm0 paddd xmm6, xmm2 paddd xmm7, xmm4 pmullw xmm1, xmm1 movdqa xmm2, xmm1 punpcklwd xmm1, xmm0 psubd xmm6, xmm1 punpckhwd xmm2, xmm0 psubd xmm7, xmm2 movdqa xmm3, xmm6 pslld xmm3, 4 psubd xmm3, xmm6 movdqa xmm1, xmm5 movdqa xmm4, xmm5 pmullw xmm1, xmm1 pmulhw xmm4, xmm4 movdqa xmm2, xmm1 punpcklwd xmm1, xmm4 punpckhwd xmm2, xmm4 movdqa xmm4, xmm7 pslld xmm4, 4 psubd xmm4, xmm7 psubd xmm3, xmm1 psubd xmm4, xmm2 psubd xmm3, flimit4 psubd xmm4, flimit4 psrad xmm3, 31 psrad xmm4, 31 packssdw xmm3, xmm4 packsswb xmm3, xmm0 movq xmm1, QWORD PTR [esi+eax*8] movq xmm2, xmm1 punpcklbw xmm1, xmm0 paddw xmm1, xmm5 mov ecx, edx and ecx, 127 movdqu xmm4, vp8_rv[ecx*2] paddw xmm1, xmm4 //paddw xmm1, eight8s psraw xmm1, 4 packuswb xmm1, xmm0 pand xmm1, xmm3 pandn xmm3, xmm2 por xmm1, xmm3 and ecx, 15 movq QWORD PTR d[ecx*8], xmm1 mov ecx, edx sub ecx, 8 and ecx, 15 movq mm0, d[ecx*8] movq [esi], mm0 lea esi, [esi+eax] lea edi, [edi+eax] add edx, 1 cmp edx, rows jl loop_row } } } #if 0 /**************************************************************************** * * ROUTINE : plane_add_noise_wmt * * INPUTS : unsigned char *Start starting address of buffer to add gaussian * noise to * unsigned int Width width of plane * unsigned int Height height of plane * int Pitch distance between subsequent lines of frame * int q quantizer used to determine amount of noise * to add * * OUTPUTS : None. * * RETURNS : void. * * FUNCTION : adds gaussian noise to a plane of pixels * * SPECIAL NOTES : None. * ****************************************************************************/ void vp8_plane_add_noise_wmt(unsigned char *Start, unsigned int Width, unsigned int Height, int Pitch, int q, int a) { unsigned int i; __declspec(align(16)) unsigned char blackclamp[16]; __declspec(align(16)) unsigned char whiteclamp[16]; __declspec(align(16)) unsigned char bothclamp[16]; char char_dist[300]; char Rand[2048]; double sigma; // return; __asm emms sigma = a + .5 + .6 * (63 - q) / 63.0; // set up a lookup table of 256 entries that matches // a gaussian distribution with sigma determined by q. // { double i; int next, j; next = 0; for (i = -32; i < 32; i++) { double g = 256 * vp8_gaussian(sigma, 0, 1.0 * i); int a = (int)(g + .5); if (a) { for (j = 0; j < a; j++) { char_dist[next+j] = (char) i; } next = next + j; } } for (next = next; next < 256; next++) char_dist[next] = 0; } for (i = 0; i < 2048; i++) { Rand[i] = char_dist[rand() & 0xff]; } for (i = 0; i < 16; i++) { blackclamp[i] = -char_dist[0]; whiteclamp[i] = -char_dist[0]; bothclamp[i] = -2 * char_dist[0]; } for (i = 0; i < Height; i++) { unsigned char *Pos = Start + i * Pitch; char *Ref = Rand + (rand() & 0xff); __asm { mov ecx, [Width] mov esi, Pos mov edi, Ref xor eax, eax nextset: movdqu xmm1, [esi+eax] // get the source psubusb xmm1, blackclamp // clamp both sides so we don't outrange adding noise paddusb xmm1, bothclamp psubusb xmm1, whiteclamp movdqu xmm2, [edi+eax] // get the noise for this line paddb xmm1, xmm2 // add it in movdqu [esi+eax], xmm1 // store the result add eax, 16 // move to the next line cmp eax, ecx jl nextset } } } #endif __declspec(align(16)) static const int four8s[4] = { 8, 8, 8, 8}; void vp8_mbpost_proc_across_ip_xmm(unsigned char *src, int pitch, int rows, int cols, int flimit) { int r, i; __declspec(align(16)) int flimit4[4]; unsigned char *s = src; int sumsq; int sum; flimit = vp8_q2mbl(flimit); flimit4[0] = flimit4[1] = flimit4[2] = flimit4[3] = flimit; for (r = 0; r < rows; r++) { sumsq = 0; sum = 0; for (i = -8; i <= 6; i++) { sumsq += s[i] * s[i]; sum += s[i]; } __asm { mov eax, sumsq movd xmm7, eax mov eax, sum movd xmm6, eax mov esi, s xor ecx, ecx mov edx, cols add edx, 8 pxor mm0, mm0 pxor mm1, mm1 pxor xmm0, xmm0 nextcol4: movd xmm1, DWORD PTR [esi+ecx-8] // -8 -7 -6 -5 movd xmm2, DWORD PTR [esi+ecx+7] // +7 +8 +9 +10 punpcklbw xmm1, xmm0 // expanding punpcklbw xmm2, xmm0 // expanding punpcklwd xmm1, xmm0 // expanding to dwords punpcklwd xmm2, xmm0 // expanding to dwords psubd xmm2, xmm1 // 7--8 8--7 9--6 10--5 paddd xmm1, xmm1 // -8*2 -7*2 -6*2 -5*2 paddd xmm1, xmm2 // 7+-8 8+-7 9+-6 10+-5 pmaddwd xmm1, xmm2 // squared of 7+-8 8+-7 9+-6 10+-5 paddd xmm6, xmm2 paddd xmm7, xmm1 pshufd xmm6, xmm6, 0 // duplicate the last ones pshufd xmm7, xmm7, 0 // duplicate the last ones psrldq xmm1, 4 // 8--7 9--6 10--5 0000 psrldq xmm2, 4 // 8--7 9--6 10--5 0000 pshufd xmm3, xmm1, 3 // 0000 8--7 8--7 8--7 squared pshufd xmm4, xmm2, 3 // 0000 8--7 8--7 8--7 squared paddd xmm6, xmm4 paddd xmm7, xmm3 pshufd xmm3, xmm1, 01011111b // 0000 0000 9--6 9--6 squared pshufd xmm4, xmm2, 01011111b // 0000 0000 9--6 9--6 squared paddd xmm7, xmm3 paddd xmm6, xmm4 pshufd xmm3, xmm1, 10111111b // 0000 0000 8--7 8--7 squared pshufd xmm4, xmm2, 10111111b // 0000 0000 8--7 8--7 squared paddd xmm7, xmm3 paddd xmm6, xmm4 movdqa xmm3, xmm6 pmaddwd xmm3, xmm3 movdqa xmm5, xmm7 pslld xmm5, 4 psubd xmm5, xmm7 psubd xmm5, xmm3 psubd xmm5, flimit4 psrad xmm5, 31 packssdw xmm5, xmm0 packsswb xmm5, xmm0 movd xmm1, DWORD PTR [esi+ecx] movq xmm2, xmm1 punpcklbw xmm1, xmm0 punpcklwd xmm1, xmm0 paddd xmm1, xmm6 paddd xmm1, four8s psrad xmm1, 4 packssdw xmm1, xmm0 packuswb xmm1, xmm0 pand xmm1, xmm5 pandn xmm5, xmm2 por xmm5, xmm1 movd [esi+ecx-8], mm0 movq mm0, mm1 movdq2q mm1, xmm5 psrldq xmm7, 12 psrldq xmm6, 12 add ecx, 4 cmp ecx, edx jl nextcol4 } s += pitch; } } #if 0 /**************************************************************************** * * ROUTINE : plane_add_noise_mmx * * INPUTS : unsigned char *Start starting address of buffer to add gaussian * noise to * unsigned int Width width of plane * unsigned int Height height of plane * int Pitch distance between subsequent lines of frame * int q quantizer used to determine amount of noise * to add * * OUTPUTS : None. * * RETURNS : void. * * FUNCTION : adds gaussian noise to a plane of pixels * * SPECIAL NOTES : None. * ****************************************************************************/ void vp8_plane_add_noise_mmx(unsigned char *Start, unsigned int Width, unsigned int Height, int Pitch, int q, int a) { unsigned int i; int Pitch4 = Pitch * 4; const int noise_amount = 2; const int noise_adder = 2 * noise_amount + 1; __declspec(align(16)) unsigned char blackclamp[16]; __declspec(align(16)) unsigned char whiteclamp[16]; __declspec(align(16)) unsigned char bothclamp[16]; char char_dist[300]; char Rand[2048]; double sigma; __asm emms sigma = a + .5 + .6 * (63 - q) / 63.0; // set up a lookup table of 256 entries that matches // a gaussian distribution with sigma determined by q. // { double i, sum = 0; int next, j; next = 0; for (i = -32; i < 32; i++) { int a = (int)(.5 + 256 * vp8_gaussian(sigma, 0, i)); if (a) { for (j = 0; j < a; j++) { char_dist[next+j] = (char) i; } next = next + j; } } for (next = next; next < 256; next++) char_dist[next] = 0; } for (i = 0; i < 2048; i++) { Rand[i] = char_dist[rand() & 0xff]; } for (i = 0; i < 16; i++) { blackclamp[i] = -char_dist[0]; whiteclamp[i] = -char_dist[0]; bothclamp[i] = -2 * char_dist[0]; } for (i = 0; i < Height; i++) { unsigned char *Pos = Start + i * Pitch; char *Ref = Rand + (rand() & 0xff); __asm { mov ecx, [Width] mov esi, Pos mov edi, Ref xor eax, eax nextset: movq mm1, [esi+eax] // get the source psubusb mm1, blackclamp // clamp both sides so we don't outrange adding noise paddusb mm1, bothclamp psubusb mm1, whiteclamp movq mm2, [edi+eax] // get the noise for this line paddb mm1, mm2 // add it in movq [esi+eax], mm1 // store the result add eax, 8 // move to the next line cmp eax, ecx jl nextset } } } #else extern char an[8][64][3072]; extern int cd[8][64]; void vp8_plane_add_noise_mmx(unsigned char *Start, unsigned int Width, unsigned int Height, int Pitch, int q, int a) { unsigned int i; __declspec(align(16)) unsigned char blackclamp[16]; __declspec(align(16)) unsigned char whiteclamp[16]; __declspec(align(16)) unsigned char bothclamp[16]; __asm emms for (i = 0; i < 16; i++) { blackclamp[i] = -cd[a][q]; whiteclamp[i] = -cd[a][q]; bothclamp[i] = -2 * cd[a][q]; } for (i = 0; i < Height; i++) { unsigned char *Pos = Start + i * Pitch; char *Ref = an[a][q] + (rand() & 0xff); __asm { mov ecx, [Width] mov esi, Pos mov edi, Ref xor eax, eax nextset: movq mm1, [esi+eax] // get the source psubusb mm1, blackclamp // clamp both sides so we don't outrange adding noise paddusb mm1, bothclamp psubusb mm1, whiteclamp movq mm2, [edi+eax] // get the noise for this line paddb mm1, mm2 // add it in movq [esi+eax], mm1 // store the result add eax, 8 // move to the next line cmp eax, ecx jl nextset } } } void vp8_plane_add_noise_wmt(unsigned char *Start, unsigned int Width, unsigned int Height, int Pitch, int q, int a) { unsigned int i; __declspec(align(16)) unsigned char blackclamp[16]; __declspec(align(16)) unsigned char whiteclamp[16]; __declspec(align(16)) unsigned char bothclamp[16]; __asm emms for (i = 0; i < 16; i++) { blackclamp[i] = -cd[a][q]; whiteclamp[i] = -cd[a][q]; bothclamp[i] = -2 * cd[a][q]; } for (i = 0; i < Height; i++) { unsigned char *Pos = Start + i * Pitch; char *Ref = an[a][q] + (rand() & 0xff); __asm { mov ecx, [Width] mov esi, Pos mov edi, Ref xor eax, eax nextset: movdqu xmm1, [esi+eax] // get the source psubusb xmm1, blackclamp // clamp both sides so we don't outrange adding noise paddusb xmm1, bothclamp psubusb xmm1, whiteclamp movdqu xmm2, [edi+eax] // get the noise for this line paddb xmm1, xmm2 // add it in movdqu [esi+eax], xmm1 // store the result add eax, 16 // move to the next line cmp eax, ecx jl nextset } } } #endif