|
|
// cb53mmx.c
#include "cst_lbc.h"
#include "mmxutil.h"
#include "opt.h"
#include "exc_lbc.h"
#include "timer.h"
#include <math.h>
#include <stdlib.h>
#include <stdio.h>
#include "util_lbc.h"
#define ASM_CORHPL 1
#define ASM_CORHDL 1
#define TESTME 0
#define CHTEST 0
#if COMPILE_MMX
void CorrelateIntTri(short *taps, short *array, int *corr, int ncor); void CorrelateInt22(short *taps, short *array, int *corr, int ncor); void Cor_h_Xint(short h[],short X[],int D[]); void Cor_hint0(short *H, int *rr); void Cor_hint1(short *H, int *rr); void cor_h_prodloop(int n, int oddn,short *h,short *h2,int *p3,int *p2,int *p1,int *p0); void cor_h_diag(int n, int oddn,short *h,short *h2,int *p3,int *p2,int *p1,int *p0);
//------------------------------------------------------------
int ACELP_LBC_code_int(float X[], float h[], int T0, float code[], int *ind_gain, int *shift, int *sign, float gain_T0, int flags) { int i, index; float gain_q; float Dn[SubFrLen2], tmp_code[SubFrLen2]; float rr[DIM_RR]; DECLARE_INT(rrint, DIM_RR); DECLARE_SHORT(hint, SubFrLen2); DECLARE_INT(Dnint, SubFrLen2); DECLARE_SHORT(Xint, SubFrLen2); int XScale; float hScale; int m; #if 0//TESTME
float htest[SubFrLen], Xtest[SubFrLen];
for (i = 0; i<SubFrLen; i++) { htest[i] = i; //(float)(i<30?i:60-i);
Xtest[i] = (float)(i<30?i:60-i); } h = htest; X = Xtest; #endif //TESTME
// Include fixed-gain pitch contribution into impulse resp. h[]
if (T0 < SubFrLen-2) for (i = T0; i < SubFrLen; i++) h[i] += gain_T0*h[i-T0];
ALIGN_ARRAY(rrint); ALIGN_ARRAY(hint); ALIGN_ARRAY(Dnint); ALIGN_ARRAY(Xint);
//hScale = FloatToShortScaled(h, hint, SubFrLen, 3);
hScale = (float)sqrt(DotProd(h,h,SubFrLen)/(double)SubFrLen); m = (asint(hScale) & 0x7f800000) >> 23; ScaleFloatToShort(h, hint, SubFrLen, m+3);
XScale = FloatToShortScaled(X, Xint, SubFrLen, 3); //would be better to normalize based on engery, not max
#if 0
for (i = 0; i<SubFrLen; i++) { hint[i] = i; } #endif
// Compute correlations of h[] needed for the codebook search
//TIMER_STAMP(a);
Cor_hint1(hint, rrint); IntToFloat(rrint, DIM_RR, rr); //TIMER_STAMP(b);
// Cor_h(h, rr);
////TIMER_STAMP(c);
#if CHTEST
{ DECLARE_INT(rrint2, DIM_RR);
ALIGN_ARRAY(rrint2);//debug
Cor_hint0(hint, rrint2); for(i = 0; i<DIM_RR; i++) //debug
if(rrint[i] != rrint2[i]) printf("%3d: %8d %8d %8d\n",i, rrint[i], rrint2[i], rrint[i] - rrint2[i]); } #endif //CHTEST
// Compute correlation of target vector with impulse response.
//TIMER_STAMP(c);
Cor_h_Xint(hint, Xint, Dnint); //TIMER_STAMP(d);
IntToFloat(Dnint, SubFrLen, Dn);
//TIMER_STAMP(a);
#if TESTME //test
{ int fpDnint[SubFrLen2]; // float scale;
// scale =
Cor_h_X(h,X,Dn); FloatToIntScaled(Dn, fpDnint, SubFrLen, 7); for (i = 0; i<SubFrLen; i++) if(fpDnint[i] != Dnint[i]) printf("%3d: %8x %8x %8x\n", i, Dnint[i] - fpDnint[i],Dnint[i], fpDnint[i]); } #endif //test
// Find codebook index
//TIMER_STAMP(c);
index = D4i64_LBC(Dn, rr, h, tmp_code, rr, shift, sign, flags); //TIMER_STAMP(f);
// Compute innovation vector gain.
// Include fixed-gain pitch contribution into code[].
*ind_gain = G_code(X, rr, &gain_q);
for (i=0; i < SubFrLen; i++) code[i] = tmp_code[i]*gain_q;
if(T0 < SubFrLen-2) for (i=T0; i < SubFrLen; i++) code[i] += code[i-T0]*gain_T0;
return index; }
//---------------------------------------------------------------
//---------------------------------------------------------------
void Cor_hint0(short *H, int *rr) {
// Compute correlations of h[] needed for the codebook search.
// h[] :Impulse response.
// rr[] :Correlations.
int *rri0i0, *rri1i1, *rri2i2, *rri3i3; int *rri0i1, *rri0i2, *rri0i3; int *rri1i2, *rri1i3, *rri2i3;
int *p0, *p1, *p2, *p3; int cor;
int i, k, m, t; DECLARE_SHORT(h,SubFrLen2); DECLARE_SHORT(h2,SubFrLen2);
ALIGN_ARRAY(h); ALIGN_ARRAY(h2);
for(i=0; i<4; i++) h[i] = (short)0;
for(i=0; i<SubFrLen; i++) h2[i+2] = h[i+4] = H[i];
// Init pointers
rri0i0 = rr; rri1i1 = rri0i0 + NB_POS; rri2i2 = rri1i1 + NB_POS; rri3i3 = rri2i2 + NB_POS;
rri0i1 = rri3i3 + NB_POS; rri0i2 = rri0i1 + MSIZE; rri0i3 = rri0i2 + MSIZE; rri1i2 = rri0i3 + MSIZE; rri1i3 = rri1i2 + MSIZE; rri2i3 = rri1i3 + MSIZE;
// Compute rri0i0[], rri1i1[], rri2i2[] and rri3i3[]
cor = 0; m = 0; for(i=NB_POS-1; i>=0; i--) { cor += h[m+0]*h[m+0] + h[m+1]*h[m+1]; rri3i3[i] = cor; cor += h[m+2]*h[m+2] + h[m+3]*h[m+3]; rri2i2[i] = cor; cor += h[m+4]*h[m+4] + h[m+5]*h[m+5]; rri1i1[i] = cor; cor += h[m+6]*h[m+6] + h[m+7]*h[m+7]; rri0i0[i] = cor;
m += 8; }
// Compute elements of: rri0i1[], rri0i3[], rri1i2[] and rri2i3[]
h2 = h+2; p3 = rri2i3 + MSIZE-1; p2 = rri1i2 + MSIZE-1; p1 = rri0i1 + MSIZE-1; p0 = rri0i3 + MSIZE-2;
for (k=0; k<NB_POS; k++) { cor = 0; m = 0; t = 0;
for(i=k+1; i<NB_POS; i++) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor; cor += h[m+6]*h2[m+6] + h[m+7]*h2[m+7]; p0[t] = cor;
t -= (NB_POS+1); m += 8; } cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor;
h2 += STEP; p3 -= NB_POS; p2 -= NB_POS; p1 -= NB_POS; p0 -= 1; }
// Compute elements of: rri0i2[], rri1i3[]
h2 = h+4; p3 = rri1i3 + MSIZE-1; p2 = rri0i2 + MSIZE-1; p1 = rri1i3 + MSIZE-2; p0 = rri0i2 + MSIZE-2;
for (k=0; k<NB_POS; k++) { cor = 0; m = 0; t = 0;
for(i=k+1; i<NB_POS; i++) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor; cor += h[m+6]*h2[m+6] + h[m+7]*h2[m+7]; p0[t] = cor;
t -= (NB_POS+1); m += 8; } cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor;
h2 += STEP; p3 -= NB_POS; p2 -= NB_POS; p1 -= 1; p0 -= 1; }
// Compute elements of: rri0i1[], rri0i3[], rri1i2[] and rri2i3[]
h2 = h+6; p3 = rri0i3 + MSIZE-1; p2 = rri2i3 + MSIZE-2; p1 = rri1i2 + MSIZE-2; p0 = rri0i1 + MSIZE-2;
for (k=0; k<NB_POS; k++) { cor = 0; m = 0; t = 0;
for(i=k+1; i<NB_POS; i++) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor; cor += h[m+6]*h2[m+6] + h[m+7]*h2[m+7]; p0[t] = cor;
t -= (NB_POS+1); m += 8; } cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor;
h2 += STEP; p3 -= NB_POS; p2 -= 1; p1 -= 1; p0 -= 1; }
return; }
//---------------------------------------------------------------
void cor_h_prods(int oddn,short *h,short *h2,int *p3,int *p2,int *p1,int *p0,int dp3,int dp2,int dp1,int dp0){ int k;
for (k=0; k<NB_POS; k++) { cor_h_prodloop(NB_POS-(k+1),oddn,h,h2,p3,p2,p1,p0); h2 += STEP; p3 -= dp3; p2 -= dp2; p1 -= dp1; p0 -= dp0; } return; }
#if _MSC_FULL_VER >= 13008827 && defined(_M_IX86)
#pragma warning(disable:4731) // EBP modified with inline asm
#endif
void cor_h_prodloop(int n, int oddn,short *h,short *h2,int *p3,int *p2,int *p1,int *p0) { #if ASM_CORHPL
n = n * 4 + oddn;
#define in edi
#define inoff edx
#define out esi
#define out3 out+eax
#define out2 out+ebx
#define out1 out+ebp
#define out0 out
#define L(m,n) ASM movq mm##m, QP[in+8*n]
#define M(m,n) ASM pmaddwd mm##m, QP[in+inoff+8*n]
#define S(m) ASM psrlq mm##m, 32
#define AH(m,n) ASM paddd mm##m, mm##n
#define WH(m,o) ASM movd DP[out##o], mm##m
#define AL(m,n) ASM paddd mm##m, mm##n
#define WL(m,o) ASM movd DP[out##o], mm##m
ASM { push ebp; mov ecx, n; mov in, h; mov inoff, h2; sub inoff, in; mov out, p0; mov eax, p3; mov ebx, p2; mov ebp, p1; sub eax, out; sub ebx, out; sub ebp, out; } L(0,0); ASM pxor mm3,mm3; M(0,0); L(1,1); AL(3,0); //really a copy
M(1,1); S(0); ASM sub ecx,8; ASM jl oddends;
inner: L(2,2); AH(0,3); WL(3,3); WH(0,2); AL(0,1); M(2,2); S(1);
L(3,3); AH(1,0); WL(0,1); WH(1,0); AL(1,2); M(3,3); S(2); ASM sub out, 4*(NB_POS+1);
L(0,4); AH(2,1); WL(1,3); WH(2,2); AL(2,3); M(0,4); S(3);
L(1,5); AH(3,2); WL(2,1); WH(3,0); AL(3,0); M(1,5); S(0); ASM sub out, 4*(NB_POS+1); ASM add in, 16*2; ASM sub ecx, 8; ASM jge inner;
oddends: ASM add ecx, 4; ASM jl cleanup;
//four more
L(2,2); AH(0,3); WL(3,3); WH(0,2); AL(0,1); M(2,2); S(1);
L(3,3); AH(1,0); WL(0,1); WH(1,0); AL(1,2); M(3,3); S(2); ASM sub out, 4*(NB_POS+1);
AH(2,1); ASM dec ecx; ASM jl innerdone; WL(1,3); ASM dec ecx; ASM jl innerdone; WH(2,2); AL(2,3); ASM dec ecx; ASM jl innerdone; WL(2,1); ASM jmp innerdone;
cleanup: ASM add ecx, 4; ASM dec ecx; ASM jl innerdone; AH(0,3); WL(3,3); ASM dec ecx; ASM jl innerdone; WH(0,2); AL(0,1); ASM dec ecx; ASM jl innerdone; WL(0,1);
innerdone: ASM emms; ASM pop ebp; #undef in
#undef inoff
#undef out
#undef out3
#undef out2
#undef out1
#undef out0
#undef L
#undef M
#undef S
#undef AH
#undef WH
#undef AL
#undef WL
#else //ASM_CORHPL
int cor; int i,m,t;
cor = 0; m = 0; t = 0;
for(i=n; i; i--) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor; cor += h[m+6]*h2[m+6] + h[m+7]*h2[m+7]; p0[t] = cor;
t -= (NB_POS+1); m += 8; } if(oddn >= 1) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; if(oddn >= 2) { cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; if(oddn >= 3) { cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor; } } } #endif //ASM_CORHPL
return; }
void cor_h_diag(int n, int oddn,short *h,short *h2,int *p3,int *p2,int *p1,int *p0) { #if ASM_CORHDL
n = n * 4 + oddn;
#define in edi
#define inoff edx
#define out esi
#define out3 out+eax
#define out2 out+ebx
#define out1 out+ebp
#define out0 out
#define L(m,n) ASM movq mm##m, QP[in+8*n]
#define M(m,n) ASM pmaddwd mm##m, QP[in+inoff+8*n]
#define R(m) ASM psrad mm##m, 1
#define S(m) ASM psrlq mm##m, 32
#define AH(m,n) ASM paddd mm##m, mm##n
#define WH(m,o) ASM movd DP[out##o], mm##m
#define AL(m,n) ASM paddd mm##m, mm##n
#define WL(m,o) ASM movd DP[out##o], mm##m
ASM { push ebp; mov ecx, n; mov in, h; mov inoff, h2; sub inoff, in; mov out, p0; mov eax, p3; mov ebx, p2; mov ebp, p1; sub eax, out; sub ebx, out; sub ebp, out; } L(0,0); ASM pxor mm3,mm3; M(0,0); L(1,1); AL(3,0); //really a copy
M(1,1); R(0); S(0); ASM sub ecx,8; ASM jl oddends;
inner: L(2,2); AH(0,3); WL(3,3); R(1); WH(0,2); AL(0,1); M(2,2); S(1);
L(3,3); AH(1,0); WL(0,1); R(2); WH(1,0); AL(1,2); M(3,3); S(2); ASM sub out, 4*1;
L(0,4); AH(2,1); WL(1,3); R(3); WH(2,2); AL(2,3); M(0,4); S(3);
L(1,5); AH(3,2); WL(2,1); R(0); WH(3,0); AL(3,0); M(1,5); S(0); ASM sub out, 4*1; ASM add in, 16*2; ASM sub ecx, 8; ASM jge inner;
oddends: ASM add ecx, 4; ASM jl cleanup;
//four more
L(2,2); AH(0,3); WL(3,3); R(1); WH(0,2); AL(0,1); M(2,2); S(1);
L(3,3); AH(1,0); WL(0,1); R(2); WH(1,0); AL(1,2); M(3,3); S(2); ASM sub out, 4*1;
AH(2,1); ASM dec ecx; ASM jl innerdone; WL(1,3); ASM dec ecx; ASM jl innerdone; WH(2,2); AL(2,3); ASM dec ecx; ASM jl innerdone; WL(2,1); ASM jmp innerdone;
cleanup: ASM add ecx, 4; ASM dec ecx; ASM jl innerdone; AH(0,3); WL(3,3); ASM dec ecx; ASM jl innerdone; WH(0,2); AL(0,1); ASM dec ecx; ASM jl innerdone; WL(0,1);
innerdone: ASM emms; ASM pop ebp; #undef in
#undef inoff
#undef out
#undef out3
#undef out2
#undef out1
#undef out0
#undef L
#undef M
#undef R
#undef S
#undef AH
#undef WH
#undef AL
#undef WL
#else //ASM_CORHDL
int cor; int i,m,t;
cor = 0; m = 0; t = 0;
for(i=n; i; i--) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor>>1; cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor>>1; cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor>>1; cor += h[m+6]*h2[m+6] + h[m+7]*h2[m+7]; p0[t] = cor>>1;
t -= 1; m += 8; } if(oddn >= 1) { cor += h[m+0]*h2[m+0] + h[m+1]*h2[m+1]; p3[t] = cor; if(oddn >= 2) { cor += h[m+2]*h2[m+2] + h[m+3]*h2[m+3]; p2[t] = cor; if(oddn >= 3) { cor += h[m+4]*h2[m+4] + h[m+5]*h2[m+5]; p1[t] = cor; } } } #endif //ASM_CORHDL
return; }
void Cor_hint1(short *H, int *rr) {
// Compute correlations of h[] needed for the codebook search.
// h[] :Impulse response.
// rr[] :Correlations.
int *rri0i0, *rri1i1, *rri2i2, *rri3i3; int *rri0i1, *rri0i2, *rri0i3; int *rri1i2, *rri1i3, *rri2i3;
int *p0, *p1, *p2, *p3; short *h2;
int i; DECLARE_SHORT(h,SubFrLen2); DECLARE_SHORT(hp2,SubFrLen2);
ALIGN_ARRAY(h); ALIGN_ARRAY(hp2);
for(i=0; i<4; i++) h[i] = (short)0;
for(i=0; i<SubFrLen; i++) hp2[i+2] = h[i+4] = H[i];
// Init pointers
rri0i0 = rr; rri1i1 = rri0i0 + NB_POS; rri2i2 = rri1i1 + NB_POS; rri3i3 = rri2i2 + NB_POS;
rri0i1 = rri3i3 + NB_POS; rri0i2 = rri0i1 + MSIZE; rri0i3 = rri0i2 + MSIZE; rri1i2 = rri0i3 + MSIZE; rri1i3 = rri1i2 + MSIZE; rri2i3 = rri1i3 + MSIZE; //TIMER_STAMP(a);
// Compute rri0i0[], rri1i1[], rri2i2[] and rri3i3[]
cor_h_diag(NB_POS,0,h,h,&rri3i3[NB_POS-1],&rri2i2[NB_POS-1],&rri1i1[NB_POS-1],&rri0i0[NB_POS-1]); //TIMER_STAMP(b);
// Compute elements of: rri0i1[], rri0i3[], rri1i2[] and rri2i3[]
h2 = hp2; p3 = rri2i3 + MSIZE-1; p2 = rri1i2 + MSIZE-1; p1 = rri0i1 + MSIZE-1; p0 = rri0i3 + MSIZE-2;
cor_h_prods(4-1,h,h2,p3,p2,p1,p0,NB_POS,NB_POS,NB_POS,1);
// Compute elements of: rri0i2[], rri1i3[]
h2 = h+4; p3 = rri1i3 + MSIZE-1; p2 = rri0i2 + MSIZE-1; p1 = rri1i3 + MSIZE-2; p0 = rri0i2 + MSIZE-2;
cor_h_prods(4-2,h,h2,p3,p2,p1,p0,NB_POS,NB_POS,1,1);
// Compute elements of: rri0i1[], rri0i3[], rri1i2[] and rri2i3[]
h2 = hp2+4; p3 = rri0i3 + MSIZE-1; p2 = rri2i3 + MSIZE-2; p1 = rri1i2 + MSIZE-2; p0 = rri0i1 + MSIZE-2;
cor_h_prods(4-3,h,h2,p3,p2,p1,p0,NB_POS,1,1,1); //TIMER_STAMP(c);
return; }
//---------------------------------------------------------------------------
void Cor_h_Xint(short h[],short X[],int D[]) { int i; DECLARE_SHORT(hh, 2*SubFrLen+16); //h[-1,0,0,1,1,2,2,3,3,4,4,5,...57,58,58,59]
DECLARE_SHORT(XX, 2*SubFrLen+16); //X[ 0,1,0,1,2,3,2,3,4,5,4,5,...58,59,58,59]
#if TESTME
short htest[SubFrLen], Xtest[SubFrLen];
for (i = 0; i<SubFrLen; i++) { htest[i] = 1;//(short)(i<30?i:60-i);
Xtest[i] = 1;//(short)(i<30?i:60-i);
} h = htest; X = Xtest; #endif //TESTME
ALIGN_ARRAY(hh); ALIGN_ARRAY(XX); for (i=2*SubFrLen; i < 2*SubFrLen+16; i++) { XX[i] = hh[i] = (short)0; } // hh += 8; XX += 8;
#define ASM_Cor_h_Xint 1
#if ASM_Cor_h_Xint
ab2ababw(X, XX, SubFrLen); ab2abzaw(h, hh, SubFrLen); //TIMER_STAMP(e);
CorrelateIntTri (hh, XX, D, SubFrLen); #if TESTME
{ int D2[SubFrLen]; CorrelateInt22 (hh, XX, D2, SubFrLen); for (i = 0; i<SubFrLen; i++) { // if(D[i] != D2[i])
printf("%3d: %6d %6d %6d ", i,D[i], D2[i], D[i] - D2[i]); if(i&1) printf("\n"); } } #endif TESTME
#else //ASM_Cor_h_Xint
for (i=0; i < SubFrLen; i+=2) { hh[2*i] = (i-1 >= 0) ? h[i-1] : (short)0; hh[2*i+1] = h[i]; hh[2*i+2] = h[i]; hh[2*i+3] = h[i+1]; XX[2*i] = X[i]; XX[2*i+1] = X[i+1]; XX[2*i+2] = X[i]; XX[2*i+3] = X[i+1]; }
for (i=0; i < 2*SubFrLen; i+=4) { int acc0 = 0, acc1 = 0; for (j=0; j < 2*SubFrLen - i; j+=4) { acc0 += (int)hh[j]*XX[i+j] + (int)hh[j+1]*XX[i+j+1]; acc1 += (int)hh[j+2]*XX[i+j+2] + (int)hh[j+3]*XX[i+j+3]; } D[i/2] = acc0 >> 16; D[i/2+1] = acc1 >> 16; } #endif //ASM_Cor_h_Xint
return; } //---------------------------------------------------------------------------
#define ASM_CORR_TRI 1
//#if ASM_CORR_TRI
//------------------------------------------------------
// triangular correlations
// ASSUMES that array has 8 zero values beyond the end
// and can be read 8 more beyond that (without page fault etc)
// data format is
// taps: 0 t0 t0 t1 t1 t2 t2 t3 t3 t4 t4 t5 ... t57 t58 t58 t59
// arr: a0 a1 a0 a1 a2 a3 a2 a3 a4 a5 a4 a5 ... a58 a59 a58 a59
//
void CorrelateIntTri(short *taps, short *array, int *corr, int ncor) { #define rega0 mm0
#define regb0 mm1
#define rega1 mm2
#define regb1 mm3
#define rega2 mm4
#define regb2 mm5
#define acc0 mm6
#define acc1 mm7
#define arr esi
#define tap edi
#define cor eax
#define icnt ecx
// In the following macros, 'n' is the column number and 'i' is the
// iteration number.
// we use "the convolution trick" or using la twice so that one
// of the pmadd's is reg,reg and thus can be in the V-slot.
// NOTE: we have read ahead up to 2 quadwords
// so we need QP[taps+8*ncor] = QP[taps+8*ncor+8] = [0 0 0 0]
// and reading QP[array+8*ncor] or QP[array+8*ncor+8] must be legal
#define la(n,i) ASM movq rega##n,QP[arr+8*i]
#define lb(n,i) ASM movq regb##n,QP[tap+8*i-8]
#define m0(n,i) ASM pmaddwd regb##n,rega##n
#define m1(n,i) ASM pmaddwd rega##n,QP[tap+8*i]
#define a0(n,i) ASM paddd acc0,regb##n
#define a1(n,i) ASM paddd acc1,rega##n
ASM { shr ncor,1; mov cor,corr; mov tap,taps; mov arr,array; mov icnt,ncor; }
ForEachCorrPair:
// prime the pump
la(0,0); ASM pxor regb0,regb0; // to avoid lb(0,0) reading taps[-1]
la(1,1); ASM pxor acc0,acc0; // clear accumulator
m1(0,0); ASM pxor acc1,acc1; // clear accumulator
lb(1,1); ASM sub icnt, 1; // account for pump priming
ASM jle cleanup; // bypass if only one to do
inner: la(2,2); m0(1,1); m1(1,1); a0(0,0); lb(2,2); a1(0,0); la(0,3); m0(2,2); m1(2,2); a0(1,1); lb(0,3); a1(1,1); la(1,4); m0(0,3); m1(0,3); a0(2,2); lb(1,4); a1(2,2);
ASM add arr,24; ASM add tap,24;
ASM sub icnt,3; ASM jg inner;
cleanup: // last two adds
a0(0,0); a1(0,0);
// Done with one correlation pair. Pack and store 2 results in corr array
ASM { add cor,16; mov arr, array mov tap,taps; add arr,16; mov icnt, ncor; mov array, arr; sub icnt,2; //set flags for jump
movq QP[cor-16],acc1; movq QP[cor-8],acc0;
mov ncor, icnt; jg ForEachCorrPair;
emms; }
} #undef rega0
#undef regb0
#undef rega1
#undef regb1
#undef rega2
#undef regb2
#undef acc0
#undef acc1
#undef arr
#undef tap
#undef cor
#undef icnt
#undef la
#undef lb
#undef m0
#undef m1
#undef a0
#undef a1
// 16 bit output
// psrad acc0,16;//this could be less in some cases
// psrad acc1,16;
// packssdw acc1,acc0;
// movq QP[cor-8],acc0;
//#else
//------------------------------------------------------
void CorrelateInt22(short *taps, short *array, int *corr, int ncor) { int i,j;
for (i=0; i < 2*ncor; i+=4) { int acc0 = 0, acc1 = 0; for (j=0; j < 2*ncor - i; j+=4) { acc0 += (int)taps[j]*array[i+j] + (int)taps[j+1]*array[i+j+1]; acc1 += (int)taps[j+2]*array[i+j+2] + (int)taps[j+3]*array[i+j+3]; } corr[i/2] = acc0 ; corr[i/2+1] = acc1 ; }
return; } //#endif
#endif //COMPILE_MMX
|