1*638691a0SAndroid Build Coastguard Worker/* Intel SIMD (SSE) implementation of Viterbi ACS butterflies 2*638691a0SAndroid Build Coastguard Worker for 64-state (k=7) convolutional code 3*638691a0SAndroid Build Coastguard Worker Copyright 2001 Phil Karn, KA9Q 4*638691a0SAndroid Build Coastguard Worker This code may be used under the terms of the GNU Lesser General Public License (LGPL) 5*638691a0SAndroid Build Coastguard Worker 6*638691a0SAndroid Build Coastguard Worker int update_viterbi27_blk_sse(struct v27 *vp,unsigned char syms[],int nbits) ; 7*638691a0SAndroid Build Coastguard Worker*/ 8*638691a0SAndroid Build Coastguard Worker 9*638691a0SAndroid Build Coastguard Worker # SSE (64-bit integer SIMD) version 10*638691a0SAndroid Build Coastguard Worker # Requires Pentium III or better 11*638691a0SAndroid Build Coastguard Worker 12*638691a0SAndroid Build Coastguard Worker # These are offsets into struct v27, defined in viterbi27.h 13*638691a0SAndroid Build Coastguard Worker .set DP,128 14*638691a0SAndroid Build Coastguard Worker .set OLDMETRICS,132 15*638691a0SAndroid Build Coastguard Worker .set NEWMETRICS,136 16*638691a0SAndroid Build Coastguard Worker.text 17*638691a0SAndroid Build Coastguard Worker.global update_viterbi27_blk_sse,Branchtab27_sse 18*638691a0SAndroid Build Coastguard Worker .type update_viterbi27_blk_sse,@function 19*638691a0SAndroid Build Coastguard Worker .align 16 20*638691a0SAndroid Build Coastguard Worker 21*638691a0SAndroid Build Coastguard Workerupdate_viterbi27_blk_sse: 22*638691a0SAndroid Build Coastguard Worker pushl %ebp 23*638691a0SAndroid Build Coastguard Worker movl %esp,%ebp 24*638691a0SAndroid Build Coastguard Worker pushl %esi 25*638691a0SAndroid Build Coastguard Worker pushl %edi 26*638691a0SAndroid Build Coastguard Worker pushl %edx 27*638691a0SAndroid Build Coastguard Worker pushl %ebx 28*638691a0SAndroid Build Coastguard Worker 29*638691a0SAndroid Build Coastguard Worker movl 8(%ebp),%edx # edx = vp 30*638691a0SAndroid Build Coastguard Worker testl %edx,%edx 31*638691a0SAndroid Build Coastguard Worker jnz 0f 32*638691a0SAndroid Build Coastguard Worker movl -1,%eax 33*638691a0SAndroid Build Coastguard Worker jmp err 34*638691a0SAndroid Build Coastguard Worker0: movl OLDMETRICS(%edx),%esi # esi -> old metrics 35*638691a0SAndroid Build Coastguard Worker movl NEWMETRICS(%edx),%edi # edi -> new metrics 36*638691a0SAndroid Build Coastguard Worker movl DP(%edx),%edx # edx -> decisions 37*638691a0SAndroid Build Coastguard Worker 38*638691a0SAndroid Build Coastguard Worker1: movl 16(%ebp),%eax # eax = nbits 39*638691a0SAndroid Build Coastguard Worker decl %eax 40*638691a0SAndroid Build Coastguard Worker jl 2f # passed zero, we're done 41*638691a0SAndroid Build Coastguard Worker movl %eax,16(%ebp) 42*638691a0SAndroid Build Coastguard Worker 43*638691a0SAndroid Build Coastguard Worker xorl %eax,%eax 44*638691a0SAndroid Build Coastguard Worker movl 12(%ebp),%ebx # %ebx = syms 45*638691a0SAndroid Build Coastguard Worker movb (%ebx),%al 46*638691a0SAndroid Build Coastguard Worker movd %eax,%mm6 # mm6[0] = first symbol 47*638691a0SAndroid Build Coastguard Worker movb 1(%ebx),%al 48*638691a0SAndroid Build Coastguard Worker movd %eax,%mm5 # mm5[0] = second symbol 49*638691a0SAndroid Build Coastguard Worker addl $2,%ebx 50*638691a0SAndroid Build Coastguard Worker movl %ebx,12(%ebp) 51*638691a0SAndroid Build Coastguard Worker 52*638691a0SAndroid Build Coastguard Worker punpcklbw %mm6,%mm6 # mm6[1] = mm6[0] 53*638691a0SAndroid Build Coastguard Worker punpcklbw %mm5,%mm5 54*638691a0SAndroid Build Coastguard Worker movq thirtyones,%mm7 55*638691a0SAndroid Build Coastguard Worker 56*638691a0SAndroid Build Coastguard Worker pshufw $0,%mm6,%mm6 # copy low word to upper 3 57*638691a0SAndroid Build Coastguard Worker pshufw $0,%mm5,%mm5 58*638691a0SAndroid Build Coastguard Worker # mm6 now contains first symbol in each byte, mm5 the second 59*638691a0SAndroid Build Coastguard Worker 60*638691a0SAndroid Build Coastguard Worker # each invocation of this macro does 8 butterflies in parallel 61*638691a0SAndroid Build Coastguard Worker .MACRO butterfly GROUP 62*638691a0SAndroid Build Coastguard Worker # compute branch metrics 63*638691a0SAndroid Build Coastguard Worker movq Branchtab27_sse+(8*\GROUP),%mm4 64*638691a0SAndroid Build Coastguard Worker movq Branchtab27_sse+32+(8*\GROUP),%mm3 65*638691a0SAndroid Build Coastguard Worker pxor %mm6,%mm4 66*638691a0SAndroid Build Coastguard Worker pxor %mm5,%mm3 67*638691a0SAndroid Build Coastguard Worker pavgb %mm3,%mm4 # mm4 contains branch metrics 68*638691a0SAndroid Build Coastguard Worker psrlw $3,%mm4 69*638691a0SAndroid Build Coastguard Worker pand %mm7,%mm4 70*638691a0SAndroid Build Coastguard Worker 71*638691a0SAndroid Build Coastguard Worker movq (8*\GROUP)(%esi),%mm0 # Incoming path metric, high bit = 0 72*638691a0SAndroid Build Coastguard Worker movq ((8*\GROUP)+32)(%esi),%mm3 # Incoming path metric, high bit = 1 73*638691a0SAndroid Build Coastguard Worker movq %mm0,%mm2 74*638691a0SAndroid Build Coastguard Worker movq %mm3,%mm1 75*638691a0SAndroid Build Coastguard Worker paddusb %mm4,%mm0 76*638691a0SAndroid Build Coastguard Worker paddusb %mm4,%mm3 77*638691a0SAndroid Build Coastguard Worker 78*638691a0SAndroid Build Coastguard Worker # invert branch metrics. This works only because they're 5 bits 79*638691a0SAndroid Build Coastguard Worker pxor %mm7,%mm4 80*638691a0SAndroid Build Coastguard Worker 81*638691a0SAndroid Build Coastguard Worker paddusb %mm4,%mm1 82*638691a0SAndroid Build Coastguard Worker paddusb %mm4,%mm2 83*638691a0SAndroid Build Coastguard Worker 84*638691a0SAndroid Build Coastguard Worker # Find survivors, leave in mm0,2 85*638691a0SAndroid Build Coastguard Worker pminub %mm1,%mm0 86*638691a0SAndroid Build Coastguard Worker pminub %mm3,%mm2 87*638691a0SAndroid Build Coastguard Worker # get decisions, leave in mm1,3 88*638691a0SAndroid Build Coastguard Worker pcmpeqb %mm0,%mm1 89*638691a0SAndroid Build Coastguard Worker pcmpeqb %mm2,%mm3 90*638691a0SAndroid Build Coastguard Worker 91*638691a0SAndroid Build Coastguard Worker # interleave and store new branch metrics in mm0,2 92*638691a0SAndroid Build Coastguard Worker movq %mm0,%mm4 93*638691a0SAndroid Build Coastguard Worker punpckhbw %mm2,%mm0 # interleave second 8 new metrics 94*638691a0SAndroid Build Coastguard Worker punpcklbw %mm2,%mm4 # interleave first 8 new metrics 95*638691a0SAndroid Build Coastguard Worker movq %mm0,(16*\GROUP+8)(%edi) 96*638691a0SAndroid Build Coastguard Worker movq %mm4,(16*\GROUP)(%edi) 97*638691a0SAndroid Build Coastguard Worker 98*638691a0SAndroid Build Coastguard Worker # interleave decisions, accumulate into %ebx 99*638691a0SAndroid Build Coastguard Worker movq %mm1,%mm4 100*638691a0SAndroid Build Coastguard Worker punpckhbw %mm3,%mm1 101*638691a0SAndroid Build Coastguard Worker punpcklbw %mm3,%mm4 102*638691a0SAndroid Build Coastguard Worker # Due to an error in the Intel instruction set ref (the register 103*638691a0SAndroid Build Coastguard Worker # fields are swapped), gas assembles pmovmskb incorrectly 104*638691a0SAndroid Build Coastguard Worker # See http://mail.gnu.org/pipermail/bug-gnu-utils/2000-August/002341.html 105*638691a0SAndroid Build Coastguard Worker .byte 0x0f,0xd7,0xc1 # pmovmskb %mm1,%eax 106*638691a0SAndroid Build Coastguard Worker shll $((16*\GROUP+8)&31),%eax 107*638691a0SAndroid Build Coastguard Worker orl %eax,%ebx 108*638691a0SAndroid Build Coastguard Worker .byte 0x0f,0xd7,0xc4 # pmovmskb %mm4,%eax 109*638691a0SAndroid Build Coastguard Worker shll $((16*\GROUP)&31),%eax 110*638691a0SAndroid Build Coastguard Worker orl %eax,%ebx 111*638691a0SAndroid Build Coastguard Worker .endm 112*638691a0SAndroid Build Coastguard Worker 113*638691a0SAndroid Build Coastguard Worker # invoke macro 4 times for a total of 32 butterflies 114*638691a0SAndroid Build Coastguard Worker xorl %ebx,%ebx # clear decisions 115*638691a0SAndroid Build Coastguard Worker butterfly GROUP=0 116*638691a0SAndroid Build Coastguard Worker butterfly GROUP=1 117*638691a0SAndroid Build Coastguard Worker movl %ebx,(%edx) # stash first 32 decisions 118*638691a0SAndroid Build Coastguard Worker xorl %ebx,%ebx 119*638691a0SAndroid Build Coastguard Worker butterfly GROUP=2 120*638691a0SAndroid Build Coastguard Worker butterfly GROUP=3 121*638691a0SAndroid Build Coastguard Worker movl %ebx,4(%edx) # stash second 32 decisions 122*638691a0SAndroid Build Coastguard Worker 123*638691a0SAndroid Build Coastguard Worker addl $8,%edx # bump decision pointer 124*638691a0SAndroid Build Coastguard Worker 125*638691a0SAndroid Build Coastguard Worker # see if we have to normalize 126*638691a0SAndroid Build Coastguard Worker movl (%edi),%eax # extract first output metric 127*638691a0SAndroid Build Coastguard Worker andl $255,%eax 128*638691a0SAndroid Build Coastguard Worker cmpl $150,%eax # is it greater than 150? 129*638691a0SAndroid Build Coastguard Worker movl $0,%eax 130*638691a0SAndroid Build Coastguard Worker jle done # No, no need to normalize 131*638691a0SAndroid Build Coastguard Worker 132*638691a0SAndroid Build Coastguard Worker # Normalize by finding smallest metric and subtracting it 133*638691a0SAndroid Build Coastguard Worker # from all metrics 134*638691a0SAndroid Build Coastguard Worker movq (%edi),%mm0 135*638691a0SAndroid Build Coastguard Worker pminub 8(%edi),%mm0 136*638691a0SAndroid Build Coastguard Worker pminub 16(%edi),%mm0 137*638691a0SAndroid Build Coastguard Worker pminub 24(%edi),%mm0 138*638691a0SAndroid Build Coastguard Worker pminub 32(%edi),%mm0 139*638691a0SAndroid Build Coastguard Worker pminub 40(%edi),%mm0 140*638691a0SAndroid Build Coastguard Worker pminub 48(%edi),%mm0 141*638691a0SAndroid Build Coastguard Worker pminub 56(%edi),%mm0 142*638691a0SAndroid Build Coastguard Worker # mm0 contains 8 smallest metrics 143*638691a0SAndroid Build Coastguard Worker # crunch down to single lowest metric 144*638691a0SAndroid Build Coastguard Worker movq %mm0,%mm1 145*638691a0SAndroid Build Coastguard Worker psrlq $32,%mm0 146*638691a0SAndroid Build Coastguard Worker pminub %mm1,%mm0 147*638691a0SAndroid Build Coastguard Worker movq %mm0,%mm1 148*638691a0SAndroid Build Coastguard Worker psrlq $16,%mm0 149*638691a0SAndroid Build Coastguard Worker pminub %mm1,%mm0 150*638691a0SAndroid Build Coastguard Worker movq %mm0,%mm1 151*638691a0SAndroid Build Coastguard Worker psrlq $8,%mm0 152*638691a0SAndroid Build Coastguard Worker pminub %mm1,%mm0 153*638691a0SAndroid Build Coastguard Worker punpcklbw %mm0,%mm0 # expand to all 8 bytes 154*638691a0SAndroid Build Coastguard Worker pshufw $0,%mm0,%mm0 155*638691a0SAndroid Build Coastguard Worker 156*638691a0SAndroid Build Coastguard Worker # mm0 now contains lowest metric in all 8 bytes 157*638691a0SAndroid Build Coastguard Worker # subtract it from every output metric 158*638691a0SAndroid Build Coastguard Worker # Trashes %mm7 159*638691a0SAndroid Build Coastguard Worker .macro PSUBUSBM REG,MEM 160*638691a0SAndroid Build Coastguard Worker movq \MEM,%mm7 161*638691a0SAndroid Build Coastguard Worker psubusb \REG,%mm7 162*638691a0SAndroid Build Coastguard Worker movq %mm7,\MEM 163*638691a0SAndroid Build Coastguard Worker .endm 164*638691a0SAndroid Build Coastguard Worker 165*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,(%edi) 166*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,8(%edi) 167*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,16(%edi) 168*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,24(%edi) 169*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,32(%edi) 170*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,40(%edi) 171*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,48(%edi) 172*638691a0SAndroid Build Coastguard Worker PSUBUSBM %mm0,56(%edi) 173*638691a0SAndroid Build Coastguard Worker 174*638691a0SAndroid Build Coastguard Worker movd %mm0,%eax 175*638691a0SAndroid Build Coastguard Worker and $0xff,%eax 176*638691a0SAndroid Build Coastguard Worker 177*638691a0SAndroid Build Coastguard Workerdone: # swap metrics 178*638691a0SAndroid Build Coastguard Worker movl %esi,%eax 179*638691a0SAndroid Build Coastguard Worker movl %edi,%esi 180*638691a0SAndroid Build Coastguard Worker movl %eax,%edi 181*638691a0SAndroid Build Coastguard Worker jmp 1b 182*638691a0SAndroid Build Coastguard Worker 183*638691a0SAndroid Build Coastguard Worker2: emms 184*638691a0SAndroid Build Coastguard Worker movl 8(%ebp),%ebx # ebx = vp 185*638691a0SAndroid Build Coastguard Worker # stash metric pointers 186*638691a0SAndroid Build Coastguard Worker movl %esi,OLDMETRICS(%ebx) 187*638691a0SAndroid Build Coastguard Worker movl %edi,NEWMETRICS(%ebx) 188*638691a0SAndroid Build Coastguard Worker movl %edx,DP(%ebx) # stash incremented value of vp->dp 189*638691a0SAndroid Build Coastguard Worker xorl %eax,%eax 190*638691a0SAndroid Build Coastguard Workererr: popl %ebx 191*638691a0SAndroid Build Coastguard Worker popl %edx 192*638691a0SAndroid Build Coastguard Worker popl %edi 193*638691a0SAndroid Build Coastguard Worker popl %esi 194*638691a0SAndroid Build Coastguard Worker popl %ebp 195*638691a0SAndroid Build Coastguard Worker 196*638691a0SAndroid Build Coastguard Worker ret 197*638691a0SAndroid Build Coastguard Worker 198*638691a0SAndroid Build Coastguard Worker .data 199*638691a0SAndroid Build Coastguard Worker 200*638691a0SAndroid Build Coastguard Worker .align 16 201*638691a0SAndroid Build Coastguard Workerthirtyones: 202*638691a0SAndroid Build Coastguard Worker .byte 31,31,31,31,31,31,31,31 203*638691a0SAndroid Build Coastguard Worker 204*638691a0SAndroid Build Coastguard Worker 205*638691a0SAndroid Build Coastguard Worker 206