core_ARMv5E.c
上传用户:dangjiwu
上传日期:2013-07-19
资源大小:42019k
文件大小:12k
源码类别:

Symbian

开发平台:

Visual C++

  1. /* ***** BEGIN LICENSE BLOCK ***** 
  2.  * Version: RCSL 1.0/RPSL 1.0 
  3.  *  
  4.  * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved. 
  5.  *      
  6.  * The contents of this file, and the files included with this file, are 
  7.  * subject to the current version of the RealNetworks Public Source License 
  8.  * Version 1.0 (the "RPSL") available at 
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed 
  10.  * the file under the RealNetworks Community Source License Version 1.0 
  11.  * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl, 
  12.  * in which case the RCSL will apply. You may also obtain the license terms 
  13.  * directly from RealNetworks.  You may not use this file except in 
  14.  * compliance with the RPSL or, if you have a valid RCSL with RealNetworks 
  15.  * applicable to this file, the RCSL.  Please see the applicable RPSL or 
  16.  * RCSL for the rights, obligations and limitations governing use of the 
  17.  * contents of the file.  
  18.  *  
  19.  * This file is part of the Helix DNA Technology. RealNetworks is the 
  20.  * developer of the Original Code and owns the copyrights in the portions 
  21.  * it created. 
  22.  *  
  23.  * This file, and the files included with this file, is distributed and made 
  24.  * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  25.  * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  26.  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS 
  27.  * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  28.  * 
  29.  * Technology Compatibility Kit Test Suite(s) Location: 
  30.  *    http://www.helixcommunity.org/content/tck 
  31.  * 
  32.  * Contributor(s): 
  33.  *  
  34.  * ***** END LICENSE BLOCK ***** */ 
  35. /*
  36.  * Fixed-point sampling rate conversion library
  37.  * Developed by Ken Cooke (kenc@real.com)
  38.  * May 2003
  39.  *
  40.  * Core filtering functions, ARMv5E version.
  41.  * This version requires an XScale, ARM9E, ARM10E, etc.
  42.  * NOTE: intended to be compiled with ARM ADS1.2 or equivalent.
  43.  */
  44. #include "core.h"
  45. #define ASSERT(x) { while (!(x)) ; }
  46. __inline int MULHI(int a, int b) {
  47. int low;
  48. __asm { SMULL low, a, b, a }
  49. return a;
  50. }
  51. short *
  52. RATCoreMono(short *pcmptr, short *pcmend, short *outptr, state_t *s)
  53. {
  54. short *rwgptr, *lwgptr, *revptr;
  55. int acc;
  56. int i;
  57. int *tab;
  58. rwgptr = s->rwing;
  59. lwgptr = s->lwing;
  60. pcmptr += s->offset;
  61. /* these must be word-aligned */
  62. ASSERT(((int)rwgptr & 0x3) == 0);
  63. ASSERT(((int)lwgptr & 0x3) == 0);
  64. while (pcmptr < pcmend) {
  65. revptr = pcmptr - 1;
  66. acc = 1 << 14;
  67. /* FIR filter */
  68. for (i = s->nwing >> 1; i != 0; i--) {
  69. register int pcm0, pcm1, rev0, rev1, lwg, rwg;
  70. __asm {
  71. LDRH pcm0, [pcmptr],#2
  72. LDR lwg,  [lwgptr],#4
  73. LDRH pcm1, [pcmptr],#2
  74. LDRH rev1, [revptr],#-2
  75. LDR rwg,  [rwgptr],#4
  76. LDRH rev0, [revptr],#-2
  77. SMLABB acc, pcm0, lwg, acc
  78. SMLABT acc, pcm1, lwg, acc
  79. SMLABB acc, rev1, rwg, acc
  80. SMLABT acc, rev0, rwg, acc
  81. }
  82. }
  83. if (s->nwing & 0x1) {
  84. register int pcm, rev, lwg, rwg;
  85. __asm {
  86. LDRH pcm, [pcmptr],#2
  87. LDRH lwg, [lwgptr],#2
  88. LDRH rev, [revptr],#-2
  89. LDRH rwg, [rwgptr],#2
  90. SMLABB acc, pcm, lwg, acc
  91. SMLABB acc, rev, rwg, acc
  92. }
  93. }
  94. /* saturate */
  95. __asm { QADD acc, acc, acc }
  96. *outptr++ = (short) (acc >> 16);
  97. /* step phase by N */
  98. tab = (rwgptr > s->stepNptr ? s->stepNbak : s->stepNfwd);
  99. rwgptr += tab[0];
  100. lwgptr += tab[1];
  101. pcmptr += tab[2];
  102. }
  103. s->offset = pcmptr - pcmend;
  104. s->rwing = rwgptr;
  105. s->lwing = lwgptr;
  106. return outptr;
  107. }
  108. short *
  109. RATCoreStereo(short *pcmptr, short *pcmend, short *outptr, state_t *s)
  110. {
  111. short *rwgptr, *lwgptr, *revptr;
  112. int acc0, acc1;
  113. int i;
  114. int *tab;
  115. rwgptr = s->rwing;
  116. lwgptr = s->lwing;
  117. pcmptr += s->offset;
  118. /* these must be word-aligned */
  119. ASSERT(((int)rwgptr & 0x3) == 0);
  120. ASSERT(((int)lwgptr & 0x3) == 0);
  121. ASSERT(((int)pcmptr & 0x3) == 0);
  122. while (pcmptr+1 < pcmend) {
  123. revptr = pcmptr - 2;
  124. acc0 = acc1 = 1 << 14;
  125. /* FIR filter */
  126. for (i = s->nwing >> 1; i != 0; i--) {
  127. register int pcm0, pcm1, rev0, rev1, lwg, rwg;
  128. __asm {
  129. LDR pcm0, [pcmptr],#4
  130. LDR lwg,  [lwgptr],#4
  131. LDR pcm1, [pcmptr],#4
  132. LDR rev1, [revptr],#-4
  133. LDR rwg,  [rwgptr],#4
  134. LDR rev0, [revptr],#-4
  135. SMLABB acc0, pcm0, lwg, acc0
  136. SMLATB acc1, pcm0, lwg, acc1
  137. SMLABT acc0, pcm1, lwg, acc0
  138. SMLATT acc1, pcm1, lwg, acc1
  139. SMLATB acc1, rev1, rwg, acc1
  140. SMLABB acc0, rev1, rwg, acc0
  141. SMLATT acc1, rev0, rwg, acc1
  142. SMLABT acc0, rev0, rwg, acc0
  143. }
  144. }
  145. if (s->nwing & 0x1) {
  146. register int pcm, rev, lwg, rwg;
  147. __asm {
  148. LDR pcm, [pcmptr],#4
  149. LDRH lwg, [lwgptr],#2
  150. LDR rev, [revptr],#-4
  151. LDRH rwg, [rwgptr],#2
  152. SMLABB acc0, pcm, lwg, acc0
  153. SMLATB acc1, pcm, lwg, acc1
  154. SMLATB acc1, rev, rwg, acc1
  155. SMLABB acc0, rev, rwg, acc0
  156. }
  157. }
  158. /* saturate */
  159. __asm { QADD acc0, acc0, acc0 }
  160. __asm { QADD acc1, acc1, acc1 }
  161. *outptr++ = (short) (acc0 >> 16);
  162. *outptr++ = (short) (acc1 >> 16);
  163. /* step phase by N */
  164. tab = (rwgptr > s->stepNptr ? s->stepNbak : s->stepNfwd);
  165. rwgptr += tab[0];
  166. lwgptr += tab[1];
  167. pcmptr += tab[2];
  168. }
  169. s->offset = pcmptr - pcmend;
  170. s->rwing = rwgptr;
  171. s->lwing = lwgptr;
  172. return outptr;
  173. }
  174. short *
  175. ARBCoreMono(short *pcmptr, short *pcmend, short *outptr, state_t *s)
  176. {
  177. register short *rwgptr, *lwgptr, *revptr;
  178. register short *rwgptr1, *lwgptr1;
  179. register int acc0, acc1;
  180. int pcmstep, i;
  181. int *tab;
  182. uint phasef;
  183. rwgptr = s->rwing;
  184. lwgptr = s->lwing;
  185. phasef = s->phasef;
  186. pcmptr += s->offset;
  187. /* phase+1 */
  188. tab = (rwgptr >= s->step1ptr ? s->step1bak : s->step1fwd);
  189. rwgptr1 = rwgptr + tab[0];
  190. lwgptr1 = lwgptr + tab[1];
  191. pcmstep = tab[2];
  192. /* these must be word-aligned */
  193. ASSERT(((int)rwgptr & 0x3) == 0);
  194. ASSERT(((int)lwgptr & 0x3) == 0);
  195. ASSERT(((int)rwgptr1 & 0x3) == 0);
  196. ASSERT(((int)lwgptr1 & 0x3) == 0);
  197. while (pcmptr+pcmstep < pcmend) {
  198. revptr = pcmptr - 1;
  199. acc0 = acc1 = 1 << 14;
  200. if (!pcmstep) {
  201. for (i = s->nwing >> 1; i != 0; i--) {
  202. register int pcm0, pcm1, wng0, wng1;
  203. __asm {
  204. LDRH pcm0, [pcmptr],#2
  205. LDRH pcm1, [pcmptr],#2
  206. LDR wng0, [lwgptr],#4
  207. LDR wng1, [lwgptr1],#4
  208. SMLABB acc0, pcm0, wng0, acc0
  209. SMLABB acc1, pcm0, wng1, acc1
  210. SMLABT acc0, pcm1, wng0, acc0
  211. SMLABT acc1, pcm1, wng1, acc1
  212. LDRH pcm1, [revptr],#-2
  213. LDRH pcm0, [revptr],#-2
  214. LDR wng0, [rwgptr],#4
  215. LDR wng1, [rwgptr1],#4
  216. SMLABB acc0, pcm1, wng0, acc0
  217. SMLABB acc1, pcm1, wng1, acc1
  218. SMLABT acc0, pcm0, wng0, acc0
  219. SMLABT acc1, pcm0, wng1, acc1
  220. }
  221. }
  222. if (s->nwing & 0x1) {
  223. register int pcm, wng0, wng1;
  224. __asm {
  225. LDRH pcm,  [pcmptr],#2
  226. LDRH wng0, [lwgptr],#2
  227. LDRH wng1, [lwgptr1],#2
  228. SMLABB acc0, pcm, wng0, acc0
  229. SMLABB acc1, pcm, wng1, acc1
  230. LDRH pcm,  [revptr],#-2
  231. LDRH wng0, [rwgptr],#2
  232. LDRH wng1, [rwgptr1],#2
  233. SMLABB acc0, pcm, wng0, acc0
  234. SMLABB acc1, pcm, wng1, acc1
  235. }
  236. }
  237. } else {
  238. for (i = s->nwing >> 1; i != 0; i--) {
  239. register int pcm0, pcm1, pcm2, wng0, wng1;
  240. __asm {
  241. LDRH pcm0, [pcmptr],#2
  242. LDRH pcm1, [pcmptr],#2
  243. LDRH pcm2, [pcmptr,#0]
  244. LDR wng0, [lwgptr],#4
  245. LDR wng1, [lwgptr1],#4
  246. SMLABB acc0, pcm0, wng0, acc0
  247. SMLABB acc1, pcm1, wng1, acc1
  248. SMLABT acc0, pcm1, wng0, acc0
  249. SMLABT acc1, pcm2, wng1, acc1
  250. LDRH pcm2, [revptr,#2]
  251. LDRH pcm1, [revptr],#-2
  252. LDRH pcm0, [revptr],#-2
  253. LDR wng0, [rwgptr],#4
  254. LDR wng1, [rwgptr1],#4
  255. SMLABB acc0, pcm1, wng0, acc0
  256. SMLABB acc1, pcm2, wng1, acc1
  257. SMLABT acc0, pcm0, wng0, acc0
  258. SMLABT acc1, pcm1, wng1, acc1
  259. }
  260. }
  261. if (s->nwing & 0x1) {
  262. register int pcm0, pcm1, wng0, wng1;
  263. __asm {
  264. LDRH pcm0, [pcmptr],#2
  265. LDRH pcm1, [pcmptr,#0]
  266. LDRH wng0, [lwgptr],#2
  267. LDRH wng1, [lwgptr1],#2
  268. SMLABB acc0, pcm0, wng0, acc0
  269. SMLABB acc1, pcm1, wng1, acc1
  270. LDRH pcm1, [revptr,#2]
  271. LDRH pcm0, [revptr],#-2
  272. LDRH wng0, [rwgptr],#2
  273. LDRH wng1, [rwgptr1],#2
  274. SMLABB acc0, pcm0, wng0, acc0
  275. SMLABB acc1, pcm1, wng1, acc1
  276. }
  277. }
  278. }
  279. /* interpolate and saturate */
  280. acc1 = MULHI(acc1 - acc0, phasef >> 1);
  281. __asm { QDADD acc0, acc0, acc1 }
  282. __asm { QADD  acc0, acc0, acc0 }
  283. *outptr++ = (short) (acc0 >> 16);
  284. /* step phase fraction */
  285. phasef += s->stepf;
  286. if (phasef < s->stepf) {
  287. rwgptr = rwgptr1;
  288. lwgptr = lwgptr1;
  289. pcmptr += pcmstep;
  290. }
  291. /* step phase by N */
  292. tab = (rwgptr > s->stepNptr ? s->stepNbak : s->stepNfwd);
  293. rwgptr += tab[0];
  294. lwgptr += tab[1];
  295. pcmptr += tab[2];
  296. /* phase+1 */
  297. tab = (rwgptr >= s->step1ptr ? s->step1bak : s->step1fwd);
  298. rwgptr1 = rwgptr + tab[0];
  299. lwgptr1 = lwgptr + tab[1];
  300. pcmstep = tab[2];
  301. }
  302. s->offset = pcmptr - pcmend;
  303. s->rwing = rwgptr;
  304. s->lwing = lwgptr;
  305. s->phasef = phasef;
  306. return outptr;
  307. }
  308. short *
  309. ARBCoreStereo(short *pcmptr, short *pcmend, short *outptr, state_t *s)
  310. {
  311. register short *rwgptr, *lwgptr, *revptr;
  312. register short *rwgptr1, *lwgptr1;
  313. register int acc0, acc1, acc2, acc3;
  314. int pcmstep, i;
  315. int *tab;
  316. uint phasef;
  317. rwgptr = s->rwing;
  318. lwgptr = s->lwing;
  319. phasef = s->phasef;
  320. pcmptr += s->offset;
  321. /* phase+1 */
  322. tab = (rwgptr >= s->step1ptr ? s->step1bak : s->step1fwd);
  323. rwgptr1 = rwgptr + tab[0];
  324. lwgptr1 = lwgptr + tab[1];
  325. pcmstep = tab[2];
  326. /* these must be word-aligned */
  327. ASSERT(((int)rwgptr & 0x3) == 0);
  328. ASSERT(((int)lwgptr & 0x3) == 0);
  329. ASSERT(((int)rwgptr1 & 0x3) == 0);
  330. ASSERT(((int)lwgptr1 & 0x3) == 0);
  331. ASSERT(((int)pcmptr & 0x3) == 0);
  332. while (pcmptr+pcmstep+1 < pcmend) {
  333. revptr = pcmptr - 2;
  334. acc0 = acc1 = acc2 = acc3 = 1 << 14;
  335. if (!pcmstep) {
  336. for (i = s->nwing; i != 0; i--) {
  337. register int pcm, wng, tmp;
  338. __asm {
  339. LDRH wng, [lwgptr],#2
  340. LDRH tmp, [lwgptr1],#2
  341. LDR pcm, [pcmptr],#4
  342. ORR wng, wng, tmp,LSL#16
  343. SMLABB acc0, pcm, wng, acc0
  344. SMLATB acc1, pcm, wng, acc1
  345. SMLABT acc2, pcm, wng, acc2
  346. SMLATT acc3, pcm, wng, acc3
  347. LDRH wng, [rwgptr],#2
  348. LDRH tmp, [rwgptr1],#2
  349. LDR pcm, [revptr],#-4
  350. ORR wng, wng, tmp,LSL#16
  351. SMLABB acc0, pcm, wng, acc0
  352. SMLATB acc1, pcm, wng, acc1
  353. SMLABT acc2, pcm, wng, acc2
  354. SMLATT acc3, pcm, wng, acc3
  355. }
  356. }
  357. } else {
  358. for (i = s->nwing; i != 0; i--) {
  359. register int pcm0, pcm1, wng, tmp;
  360. __asm {
  361. LDRH wng,  [lwgptr],#2
  362. LDRH tmp,  [lwgptr1],#2
  363. LDR pcm0, [pcmptr],#4
  364. LDR pcm1, [pcmptr,#0]
  365. ORR wng, wng, tmp,LSL#16
  366. SMLABB acc0, pcm0, wng, acc0
  367. SMLATB acc1, pcm0, wng, acc1
  368. SMLABT acc2, pcm1, wng, acc2
  369. SMLATT acc3, pcm1, wng, acc3
  370. LDRH wng,  [rwgptr],#2
  371. LDRH tmp,  [rwgptr1],#2
  372. LDR pcm1, [revptr,#4]
  373. LDR pcm0, [revptr],#-4
  374. ORR wng, wng, tmp,LSL#16
  375. SMLABB acc0, pcm0, wng, acc0
  376. SMLATB acc1, pcm0, wng, acc1
  377. SMLABT acc2, pcm1, wng, acc2
  378. SMLATT acc3, pcm1, wng, acc3
  379. }
  380. }
  381. }
  382. /* interpolate and saturate */
  383. acc2 = MULHI(acc2 - acc0, phasef >> 1);
  384. __asm { QDADD acc0, acc0, acc2 }
  385. __asm { QADD  acc0, acc0, acc0 }
  386. acc3 = MULHI(acc3 - acc1, phasef >> 1);
  387. __asm { QDADD acc1, acc1, acc3 }
  388. __asm { QADD  acc1, acc1, acc1 }
  389. *outptr++ = (short) (acc0 >> 16);
  390. *outptr++ = (short) (acc1 >> 16);
  391. /* step phase fraction */
  392. phasef += s->stepf;
  393. if (phasef < s->stepf) {
  394. rwgptr = rwgptr1;
  395. lwgptr = lwgptr1;
  396. pcmptr += pcmstep;
  397. }
  398. /* step phase by N */
  399. tab = (rwgptr > s->stepNptr ? s->stepNbak : s->stepNfwd);
  400. rwgptr += tab[0];
  401. lwgptr += tab[1];
  402. pcmptr += tab[2];
  403. /* phase+1 */
  404. tab = (rwgptr >= s->step1ptr ? s->step1bak : s->step1fwd);
  405. rwgptr1 = rwgptr + tab[0];
  406. lwgptr1 = lwgptr + tab[1];
  407. pcmstep = tab[2];
  408. }
  409. s->offset = pcmptr - pcmend;
  410. s->rwing = rwgptr;
  411. s->lwing = lwgptr;
  412. s->phasef = phasef;
  413. return outptr;
  414. }