ref: b9be7a464f065c3700f937eea01d9298002b37eb
dir: /vp8/common/arm/neon/loopfilterverticaledge_y_neon.asm/
; ; 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. ; EXPORT |vp8_loop_filter_vertical_edge_y_neon| ARM REQUIRE8 PRESERVE8 AREA ||.text||, CODE, READONLY, ALIGN=2 ;Note: flimit, limit, and thresh shpuld be positive numbers. All 16 elements in flimit ;are equal. So, in the code, only one load is needed ;for flimit. Same way applies to limit and thresh. ; r0 unsigned char *s, ; r1 int p, //pitch ; r2 const signed char *flimit, ; r3 const signed char *limit, ; stack(r4) const signed char *thresh, ; //stack(r5) int count --unused |vp8_loop_filter_vertical_edge_y_neon| PROC sub r0, r0, #4 ; move src pointer down by 4 columns ldr r12, [sp, #0] ; load thresh pointer vld1.u8 {d6}, [r0], r1 ; load first 8-line src data vld1.s8 {d0[], d1[]}, [r2] ; flimit vld1.u8 {d8}, [r0], r1 vld1.s8 {d2[], d3[]}, [r3] ; limit vld1.u8 {d10}, [r0], r1 vld1.s8 {d4[], d5[]}, [r12] ; thresh vld1.u8 {d12}, [r0], r1 ldr r12, _vlfy_coeff_ vld1.u8 {d14}, [r0], r1 vld1.u8 {d16}, [r0], r1 vld1.u8 {d18}, [r0], r1 vld1.u8 {d20}, [r0], r1 vld1.u8 {d7}, [r0], r1 ; load second 8-line src data vld1.u8 {d9}, [r0], r1 vld1.u8 {d11}, [r0], r1 vld1.u8 {d13}, [r0], r1 vld1.u8 {d15}, [r0], r1 vld1.u8 {d17}, [r0], r1 vld1.u8 {d19}, [r0], r1 vld1.u8 {d21}, [r0], r1 ;transpose to 8x16 matrix vtrn.32 q3, q7 vtrn.32 q4, q8 vtrn.32 q5, q9 vtrn.32 q6, q10 vtrn.16 q3, q5 vtrn.16 q4, q6 vtrn.16 q7, q9 vtrn.16 q8, q10 vtrn.8 q3, q4 vtrn.8 q5, q6 vtrn.8 q7, q8 vtrn.8 q9, q10 ; vp8_filter_mask vabd.u8 q11, q3, q4 ; abs(p3 - p2) vabd.u8 q12, q4, q5 ; abs(p2 - p1) vabd.u8 q13, q5, q6 ; abs(p1 - p0) vabd.u8 q14, q8, q7 ; abs(q1 - q0) vabd.u8 q3, q9, q8 ; abs(q2 - q1) vabd.u8 q4, q10, q9 ; abs(q3 - q2) vabd.u8 q9, q6, q7 ; abs(p0 - q0) vmax.u8 q11, q11, q12 vmax.u8 q12, q13, q14 vmax.u8 q3, q3, q4 vmax.u8 q15, q11, q12 ; vp8_hevmask vcgt.u8 q13, q13, q2 ; (abs(p1 - p0) > thresh)*-1 vcgt.u8 q14, q14, q2 ; (abs(q1 - q0) > thresh)*-1 vmax.u8 q15, q15, q3 vadd.u8 q0, q0, q0 ; flimit * 2 vadd.u8 q0, q0, q1 ; flimit * 2 + limit vcge.u8 q15, q1, q15 vabd.u8 q2, q5, q8 ; abs(p1 - q1) vqadd.u8 q9, q9, q9 ; abs(p0 - q0) * 2 vshr.u8 q2, q2, #1 ; abs(p1 - q1) / 2 vqadd.u8 q9, q9, q2 ; abs(p0 - q0) * 2 + abs(p1 - q1) / 2 vcge.u8 q9, q0, q9 ; (abs(p0 - q0)*2 + abs(p1-q1)/2 > flimit*2 + limit)*-1 vld1.u8 {q0}, [r12]! ;vp8_filter() function veor q7, q7, q0 ; qs0: q0 offset to convert to a signed value veor q6, q6, q0 ; ps0: p0 offset to convert to a signed value veor q5, q5, q0 ; ps1: p1 offset to convert to a signed value veor q8, q8, q0 ; qs1: q1 offset to convert to a signed value ;;;;;;;;;;;;;; vld1.u8 {q10}, [r12]! vsubl.s8 q2, d14, d12 ; ( qs0 - ps0) vsubl.s8 q11, d15, d13 vmovl.u8 q4, d20 vqsub.s8 q1, q5, q8 ; vp8_filter = vp8_signed_char_clamp(ps1-qs1) vorr q14, q13, q14 ; q14: vp8_hevmask vmul.i16 q2, q2, q4 ; 3 * ( qs0 - ps0) vmul.i16 q11, q11, q4 vand q1, q1, q14 ; vp8_filter &= hev vand q15, q15, q9 ; vp8_filter_mask vaddw.s8 q2, q2, d2 vaddw.s8 q11, q11, d3 vld1.u8 {q9}, [r12]! ; vqmovn.s16 d2, q2 ; vp8_filter = vp8_signed_char_clamp(vp8_filter + 3 * ( qs0 - ps0)) vqmovn.s16 d3, q11 ;; vand q1, q1, q15 ; vp8_filter &= mask vqadd.s8 q2, q1, q10 ; Filter2 = vp8_signed_char_clamp(vp8_filter+3) vqadd.s8 q1, q1, q9 ; Filter1 = vp8_signed_char_clamp(vp8_filter+4) vshr.s8 q2, q2, #3 ; Filter2 >>= 3 vshr.s8 q1, q1, #3 ; Filter1 >>= 3 ;calculate output vqadd.s8 q11, q6, q2 ; u = vp8_signed_char_clamp(ps0 + Filter2) vqsub.s8 q10, q7, q1 ; u = vp8_signed_char_clamp(qs0 - Filter1) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; vrshr.s8 q1, q1, #1 ;round/shift: vp8_filter += 1; vp8_filter >>= 1 sub r0, r0, r1, lsl #4 add r0, r0, #2 ; vbic q1, q1, q14 ; vp8_filter &= ~hev add r2, r0, r1 ; vqadd.s8 q13, q5, q1 ; u = vp8_signed_char_clamp(ps1 + vp8_filter) vqsub.s8 q12, q8, q1 ; u = vp8_signed_char_clamp(qs1 - vp8_filter) veor q7, q10, q0 ; *oq0 = u^0x80 veor q5, q13, q0 ; *op1 = u^0x80 veor q6, q11, q0 ; *op0 = u^0x80 veor q8, q12, q0 ; *oq1 = u^0x80 add r3, r2, r1 ; vswp d12, d11 vswp d16, d13 add r12, r3, r1 vswp d14, d12 vswp d16, d15 ;store op1, op0, oq0, oq1 vst4.8 {d10[0], d11[0], d12[0], d13[0]}, [r0] vst4.8 {d10[1], d11[1], d12[1], d13[1]}, [r2] vst4.8 {d10[2], d11[2], d12[2], d13[2]}, [r3] vst4.8 {d10[3], d11[3], d12[3], d13[3]}, [r12], r1 add r0, r12, r1 vst4.8 {d10[4], d11[4], d12[4], d13[4]}, [r12] vst4.8 {d10[5], d11[5], d12[5], d13[5]}, [r0], r1 add r2, r0, r1 vst4.8 {d10[6], d11[6], d12[6], d13[6]}, [r0] vst4.8 {d10[7], d11[7], d12[7], d13[7]}, [r2], r1 add r3, r2, r1 vst4.8 {d14[0], d15[0], d16[0], d17[0]}, [r2] vst4.8 {d14[1], d15[1], d16[1], d17[1]}, [r3], r1 add r12, r3, r1 vst4.8 {d14[2], d15[2], d16[2], d17[2]}, [r3] vst4.8 {d14[3], d15[3], d16[3], d17[3]}, [r12], r1 add r0, r12, r1 vst4.8 {d14[4], d15[4], d16[4], d17[4]}, [r12] vst4.8 {d14[5], d15[5], d16[5], d17[5]}, [r0], r1 add r2, r0, r1 vst4.8 {d14[6], d15[6], d16[6], d17[6]}, [r0] vst4.8 {d14[7], d15[7], d16[7], d17[7]}, [r2] bx lr ENDP ; |vp8_loop_filter_vertical_edge_y_neon| ;----------------- AREA vloopfiltery_dat, DATA, READWRITE ;read/write by default ;Data section with name data_area is specified. DCD reserves space in memory for 16 data. ;One word each is reserved. Label filter_coeff can be used to access the data. ;Data address: filter_coeff, filter_coeff+4, filter_coeff+8 ... _vlfy_coeff_ DCD vlfy_coeff vlfy_coeff DCD 0x80808080, 0x80808080, 0x80808080, 0x80808080 DCD 0x03030303, 0x03030303, 0x03030303, 0x03030303 DCD 0x04040404, 0x04040404, 0x04040404, 0x04040404 DCD 0x01010101, 0x01010101, 0x01010101, 0x01010101 END