2 ; Copyright (c) 2010 The WebM project authors. All Rights Reserved.
4 ; Use of this source code is governed by a BSD-style license
5 ; that can be found in the LICENSE file in the root of the source
6 ; tree. An additional intellectual property rights grant can be found
7 ; in the file PATENTS. All contributing project authors may
8 ; be found in the AUTHORS file in the root of the source tree.
12 EXPORT |vp8_mbloop_filter_horizontal_edge_y_neon|
17 AREA ||.text||, CODE, READONLY, ALIGN=2
18 ;Note: flimit, limit, and thresh shpuld be positive numbers. All 16 elements in flimit
19 ;are equal. So, in the code, only one load is needed
20 ;for flimit. Same way applies to limit and thresh.
21 ; r0 unsigned char *s,
23 ; r2 const signed char *flimit,
24 ; r3 const signed char *limit,
25 ; stack(r4) const signed char *thresh,
26 ; //stack(r5) int count --unused
27 |vp8_mbloop_filter_horizontal_edge_y_neon| PROC
28 sub r0, r0, r1, lsl #2 ; move src pointer down by 4 lines
29 ldr r12, [sp, #0] ; load thresh pointer
31 vld1.u8 {q3}, [r0], r1 ; p3
32 vld1.s8 {d2[], d3[]}, [r3] ; limit
33 vld1.u8 {q4}, [r0], r1 ; p2
34 vld1.s8 {d4[], d5[]}, [r12] ; thresh
35 vld1.u8 {q5}, [r0], r1 ; p1
36 ldr r12, _mbhlfy_coeff_
37 vld1.u8 {q6}, [r0], r1 ; p0
40 vabd.u8 q11, q3, q4 ; abs(p3 - p2)
41 vld1.u8 {q7}, [r0], r1 ; q0
42 vabd.u8 q12, q4, q5 ; abs(p2 - p1)
43 vld1.u8 {q8}, [r0], r1 ; q1
44 vabd.u8 q13, q5, q6 ; abs(p1 - p0)
45 vld1.u8 {q9}, [r0], r1 ; q2
46 vabd.u8 q14, q8, q7 ; abs(q1 - q0)
47 vld1.u8 {q10}, [r0], r1 ; q3
48 vabd.u8 q3, q9, q8 ; abs(q2 - q1)
49 vabd.u8 q0, q10, q9 ; abs(q3 - q2)
56 vabd.u8 q12, q6, q7 ; abs(p0 - q0)
59 vcgt.u8 q13, q13, q2 ; (abs(p1 - p0) > thresh)*-1
60 vcgt.u8 q14, q14, q2 ; (abs(q1 - q0) > thresh)*-1
63 vld1.s8 {d4[], d5[]}, [r2] ; flimit
67 vadd.u8 q2, q2, q2 ; flimit * 2
68 vadd.u8 q2, q2, q1 ; flimit * 2 + limit
71 vabd.u8 q1, q5, q8 ; abs(p1 - q1)
72 vqadd.u8 q12, q12, q12 ; abs(p0 - q0) * 2
73 vshr.u8 q1, q1, #1 ; abs(p1 - q1) / 2
74 vqadd.u8 q12, q12, q1 ; abs(p0 - q0) * 2 + abs(p1 - q1) / 2
75 vcge.u8 q12, q2, q12 ; (abs(p0 - q0)*2 + abs(p1 - q1)/2 > flimit*2 + limit)*-1
77 ;vp8_filter() function
78 veor q7, q7, q0 ; qs0: q0 offset to convert to a signed value
79 veor q6, q6, q0 ; ps0: p0 offset to convert to a signed value
80 veor q5, q5, q0 ; ps1: p1 offset to convert to a signed value
81 veor q8, q8, q0 ; qs1: q1 offset to convert to a signed value
82 veor q4, q4, q0 ; ps2: p2 offset to convert to a signed value
83 veor q9, q9, q0 ; qs2: q2 offset to convert to a signed value
85 vorr q14, q13, q14 ; q14: vp8_hevmask
87 vsubl.s8 q2, d14, d12 ; ( qs0 - ps0)
88 vsubl.s8 q13, d15, d13
90 vqsub.s8 q1, q5, q8 ; vp8_filter = vp8_signed_char_clamp(ps1-qs1)
92 vadd.s16 q10, q2, q2 ; 3 * ( qs0 - ps0)
93 vadd.s16 q11, q13, q13
94 vand q15, q15, q12 ; vp8_filter_mask
97 vadd.s16 q13, q13, q11
99 vld1.u8 {q12}, [r12]! ;#3
101 vaddw.s8 q2, q2, d2 ; vp8_filter + 3 * ( qs0 - ps0)
102 vaddw.s8 q13, q13, d3
104 vld1.u8 {q11}, [r12]! ;#4
106 vqmovn.s16 d2, q2 ; vp8_filter = vp8_signed_char_clamp(vp8_filter + 3 * ( qs0 - ps0))
110 vand q1, q1, q15 ; vp8_filter &= mask
112 vld1.u8 {q15}, [r12]! ;#63
114 vand q13, q1, q14 ; Filter2: q13; Filter2 &= hev
116 vld1.u8 {d7}, [r12]! ;#9
117 sub r0, r0, r1, lsl #3
119 vqadd.s8 q2, q13, q11 ; Filter1 = vp8_signed_char_clamp(Filter2+4)
120 vqadd.s8 q13, q13, q12 ; Filter2 = vp8_signed_char_clamp(Filter2+3)
122 vld1.u8 {d6}, [r12]! ;#18
126 vshr.s8 q2, q2, #3 ; Filter1 >>= 3
127 vshr.s8 q13, q13, #3 ; Filter2 >>= 3
132 vqsub.s8 q7, q7, q2 ; qs0 = vp8_signed_char_clamp(qs0 - Filter1)
134 vld1.u8 {d5}, [r12]! ;#27
137 vqadd.s8 q6, q6, q13 ; ps0 = vp8_signed_char_clamp(ps0 + Filter2)
138 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
140 vbic q1, q1, q14 ; Filter2: q1; vp8_filter &= ~hev; Filter2 = vp8_filter
142 ; roughly 1/7th difference across boundary
143 ; roughly 2/7th difference across boundary
144 ; roughly 3/7th difference across boundary
149 vmlal.s8 q10, d2, d7 ; Filter2 * 9
151 vmlal.s8 q12, d2, d6 ; Filter2 * 18
153 vmlal.s8 q14, d2, d5 ; Filter2 * 27
155 vqshrn.s16 d20, q10, #7 ; u = vp8_signed_char_clamp((63 + Filter2 * 9)>>7)
156 vqshrn.s16 d21, q11, #7
157 vqshrn.s16 d24, q12, #7 ; u = vp8_signed_char_clamp((63 + Filter2 * 18)>>7)
158 vqshrn.s16 d25, q13, #7
159 vqshrn.s16 d28, q14, #7 ; u = vp8_signed_char_clamp((63 + Filter2 * 27)>>7)
160 vqshrn.s16 d29, q15, #7
162 vqsub.s8 q11, q9, q10 ; s = vp8_signed_char_clamp(qs2 - u)
163 vqadd.s8 q10, q4, q10 ; s = vp8_signed_char_clamp(ps2 + u)
164 vqsub.s8 q13, q8, q12 ; s = vp8_signed_char_clamp(qs1 - u)
165 vqadd.s8 q12, q5, q12 ; s = vp8_signed_char_clamp(ps1 + u)
166 vqsub.s8 q15, q7, q14 ; s = vp8_signed_char_clamp(qs0 - u)
167 vqadd.s8 q14, q6, q14 ; s = vp8_signed_char_clamp(ps0 + u)
168 veor q9, q11, q0 ; *oq2 = s^0x80
169 veor q4, q10, q0 ; *op2 = s^0x80
170 veor q5, q12, q0 ; *op2 = s^0x80
171 veor q6, q14, q0 ; *op0 = s^0x80
172 veor q8, q13, q0 ; *oq1 = s^0x80
173 veor q7, q15, q0 ; *oq0 = s^0x80
175 vst1.u8 {q4}, [r0] ; store op2
176 vst1.u8 {q5}, [r2] ; store op1
177 vst1.u8 {q6}, [r3], r1 ; store op0
179 vst1.u8 {q7}, [r3] ; store oq0
180 vst1.u8 {q8}, [r12], r1 ; store oq1
181 vst1.u8 {q9}, [r12] ; store oq2
184 ENDP ; |vp8_mbloop_filter_horizontal_edge_y_neon|
187 AREA mbhloopfiltery_dat, DATA, READWRITE ;read/write by default
188 ;Data section with name data_area is specified. DCD reserves space in memory for 16 data.
189 ;One word each is reserved. Label filter_coeff can be used to access the data.
190 ;Data address: filter_coeff, filter_coeff+4, filter_coeff+8 ...
194 DCD 0x80808080, 0x80808080, 0x80808080, 0x80808080
195 DCD 0x03030303, 0x03030303, 0x03030303, 0x03030303
196 DCD 0x04040404, 0x04040404, 0x04040404, 0x04040404
197 DCD 0x003f003f, 0x003f003f, 0x003f003f, 0x003f003f
198 DCD 0x09090909, 0x09090909, 0x12121212, 0x12121212
199 DCD 0x1b1b1b1b, 0x1b1b1b1b