annotate src/libvorbis-1.3.3/lib/mdct.c @ 147:45360b968bf4

Cap'n Proto v0.6 + build for OSX
author Chris Cannam <cannam@all-day-breakfast.com>
date Mon, 22 May 2017 10:01:37 +0100
parents 98c1576536ae
children
rev   line source
cannam@86 1 /********************************************************************
cannam@86 2 * *
cannam@86 3 * THIS FILE IS PART OF THE OggVorbis SOFTWARE CODEC SOURCE CODE. *
cannam@86 4 * USE, DISTRIBUTION AND REPRODUCTION OF THIS LIBRARY SOURCE IS *
cannam@86 5 * GOVERNED BY A BSD-STYLE SOURCE LICENSE INCLUDED WITH THIS SOURCE *
cannam@86 6 * IN 'COPYING'. PLEASE READ THESE TERMS BEFORE DISTRIBUTING. *
cannam@86 7 * *
cannam@86 8 * THE OggVorbis SOURCE CODE IS (C) COPYRIGHT 1994-2009 *
cannam@86 9 * by the Xiph.Org Foundation http://www.xiph.org/ *
cannam@86 10 * *
cannam@86 11 ********************************************************************
cannam@86 12
cannam@86 13 function: normalized modified discrete cosine transform
cannam@86 14 power of two length transform only [64 <= n ]
cannam@86 15 last mod: $Id: mdct.c 16227 2009-07-08 06:58:46Z xiphmont $
cannam@86 16
cannam@86 17 Original algorithm adapted long ago from _The use of multirate filter
cannam@86 18 banks for coding of high quality digital audio_, by T. Sporer,
cannam@86 19 K. Brandenburg and B. Edler, collection of the European Signal
cannam@86 20 Processing Conference (EUSIPCO), Amsterdam, June 1992, Vol.1, pp
cannam@86 21 211-214
cannam@86 22
cannam@86 23 The below code implements an algorithm that no longer looks much like
cannam@86 24 that presented in the paper, but the basic structure remains if you
cannam@86 25 dig deep enough to see it.
cannam@86 26
cannam@86 27 This module DOES NOT INCLUDE code to generate/apply the window
cannam@86 28 function. Everybody has their own weird favorite including me... I
cannam@86 29 happen to like the properties of y=sin(.5PI*sin^2(x)), but others may
cannam@86 30 vehemently disagree.
cannam@86 31
cannam@86 32 ********************************************************************/
cannam@86 33
cannam@86 34 /* this can also be run as an integer transform by uncommenting a
cannam@86 35 define in mdct.h; the integerization is a first pass and although
cannam@86 36 it's likely stable for Vorbis, the dynamic range is constrained and
cannam@86 37 roundoff isn't done (so it's noisy). Consider it functional, but
cannam@86 38 only a starting point. There's no point on a machine with an FPU */
cannam@86 39
cannam@86 40 #include <stdio.h>
cannam@86 41 #include <stdlib.h>
cannam@86 42 #include <string.h>
cannam@86 43 #include <math.h>
cannam@86 44 #include "vorbis/codec.h"
cannam@86 45 #include "mdct.h"
cannam@86 46 #include "os.h"
cannam@86 47 #include "misc.h"
cannam@86 48
cannam@86 49 /* build lookups for trig functions; also pre-figure scaling and
cannam@86 50 some window function algebra. */
cannam@86 51
cannam@86 52 void mdct_init(mdct_lookup *lookup,int n){
cannam@86 53 int *bitrev=_ogg_malloc(sizeof(*bitrev)*(n/4));
cannam@86 54 DATA_TYPE *T=_ogg_malloc(sizeof(*T)*(n+n/4));
cannam@86 55
cannam@86 56 int i;
cannam@86 57 int n2=n>>1;
cannam@86 58 int log2n=lookup->log2n=rint(log((float)n)/log(2.f));
cannam@86 59 lookup->n=n;
cannam@86 60 lookup->trig=T;
cannam@86 61 lookup->bitrev=bitrev;
cannam@86 62
cannam@86 63 /* trig lookups... */
cannam@86 64
cannam@86 65 for(i=0;i<n/4;i++){
cannam@86 66 T[i*2]=FLOAT_CONV(cos((M_PI/n)*(4*i)));
cannam@86 67 T[i*2+1]=FLOAT_CONV(-sin((M_PI/n)*(4*i)));
cannam@86 68 T[n2+i*2]=FLOAT_CONV(cos((M_PI/(2*n))*(2*i+1)));
cannam@86 69 T[n2+i*2+1]=FLOAT_CONV(sin((M_PI/(2*n))*(2*i+1)));
cannam@86 70 }
cannam@86 71 for(i=0;i<n/8;i++){
cannam@86 72 T[n+i*2]=FLOAT_CONV(cos((M_PI/n)*(4*i+2))*.5);
cannam@86 73 T[n+i*2+1]=FLOAT_CONV(-sin((M_PI/n)*(4*i+2))*.5);
cannam@86 74 }
cannam@86 75
cannam@86 76 /* bitreverse lookup... */
cannam@86 77
cannam@86 78 {
cannam@86 79 int mask=(1<<(log2n-1))-1,i,j;
cannam@86 80 int msb=1<<(log2n-2);
cannam@86 81 for(i=0;i<n/8;i++){
cannam@86 82 int acc=0;
cannam@86 83 for(j=0;msb>>j;j++)
cannam@86 84 if((msb>>j)&i)acc|=1<<j;
cannam@86 85 bitrev[i*2]=((~acc)&mask)-1;
cannam@86 86 bitrev[i*2+1]=acc;
cannam@86 87
cannam@86 88 }
cannam@86 89 }
cannam@86 90 lookup->scale=FLOAT_CONV(4.f/n);
cannam@86 91 }
cannam@86 92
cannam@86 93 /* 8 point butterfly (in place, 4 register) */
cannam@86 94 STIN void mdct_butterfly_8(DATA_TYPE *x){
cannam@86 95 REG_TYPE r0 = x[6] + x[2];
cannam@86 96 REG_TYPE r1 = x[6] - x[2];
cannam@86 97 REG_TYPE r2 = x[4] + x[0];
cannam@86 98 REG_TYPE r3 = x[4] - x[0];
cannam@86 99
cannam@86 100 x[6] = r0 + r2;
cannam@86 101 x[4] = r0 - r2;
cannam@86 102
cannam@86 103 r0 = x[5] - x[1];
cannam@86 104 r2 = x[7] - x[3];
cannam@86 105 x[0] = r1 + r0;
cannam@86 106 x[2] = r1 - r0;
cannam@86 107
cannam@86 108 r0 = x[5] + x[1];
cannam@86 109 r1 = x[7] + x[3];
cannam@86 110 x[3] = r2 + r3;
cannam@86 111 x[1] = r2 - r3;
cannam@86 112 x[7] = r1 + r0;
cannam@86 113 x[5] = r1 - r0;
cannam@86 114
cannam@86 115 }
cannam@86 116
cannam@86 117 /* 16 point butterfly (in place, 4 register) */
cannam@86 118 STIN void mdct_butterfly_16(DATA_TYPE *x){
cannam@86 119 REG_TYPE r0 = x[1] - x[9];
cannam@86 120 REG_TYPE r1 = x[0] - x[8];
cannam@86 121
cannam@86 122 x[8] += x[0];
cannam@86 123 x[9] += x[1];
cannam@86 124 x[0] = MULT_NORM((r0 + r1) * cPI2_8);
cannam@86 125 x[1] = MULT_NORM((r0 - r1) * cPI2_8);
cannam@86 126
cannam@86 127 r0 = x[3] - x[11];
cannam@86 128 r1 = x[10] - x[2];
cannam@86 129 x[10] += x[2];
cannam@86 130 x[11] += x[3];
cannam@86 131 x[2] = r0;
cannam@86 132 x[3] = r1;
cannam@86 133
cannam@86 134 r0 = x[12] - x[4];
cannam@86 135 r1 = x[13] - x[5];
cannam@86 136 x[12] += x[4];
cannam@86 137 x[13] += x[5];
cannam@86 138 x[4] = MULT_NORM((r0 - r1) * cPI2_8);
cannam@86 139 x[5] = MULT_NORM((r0 + r1) * cPI2_8);
cannam@86 140
cannam@86 141 r0 = x[14] - x[6];
cannam@86 142 r1 = x[15] - x[7];
cannam@86 143 x[14] += x[6];
cannam@86 144 x[15] += x[7];
cannam@86 145 x[6] = r0;
cannam@86 146 x[7] = r1;
cannam@86 147
cannam@86 148 mdct_butterfly_8(x);
cannam@86 149 mdct_butterfly_8(x+8);
cannam@86 150 }
cannam@86 151
cannam@86 152 /* 32 point butterfly (in place, 4 register) */
cannam@86 153 STIN void mdct_butterfly_32(DATA_TYPE *x){
cannam@86 154 REG_TYPE r0 = x[30] - x[14];
cannam@86 155 REG_TYPE r1 = x[31] - x[15];
cannam@86 156
cannam@86 157 x[30] += x[14];
cannam@86 158 x[31] += x[15];
cannam@86 159 x[14] = r0;
cannam@86 160 x[15] = r1;
cannam@86 161
cannam@86 162 r0 = x[28] - x[12];
cannam@86 163 r1 = x[29] - x[13];
cannam@86 164 x[28] += x[12];
cannam@86 165 x[29] += x[13];
cannam@86 166 x[12] = MULT_NORM( r0 * cPI1_8 - r1 * cPI3_8 );
cannam@86 167 x[13] = MULT_NORM( r0 * cPI3_8 + r1 * cPI1_8 );
cannam@86 168
cannam@86 169 r0 = x[26] - x[10];
cannam@86 170 r1 = x[27] - x[11];
cannam@86 171 x[26] += x[10];
cannam@86 172 x[27] += x[11];
cannam@86 173 x[10] = MULT_NORM(( r0 - r1 ) * cPI2_8);
cannam@86 174 x[11] = MULT_NORM(( r0 + r1 ) * cPI2_8);
cannam@86 175
cannam@86 176 r0 = x[24] - x[8];
cannam@86 177 r1 = x[25] - x[9];
cannam@86 178 x[24] += x[8];
cannam@86 179 x[25] += x[9];
cannam@86 180 x[8] = MULT_NORM( r0 * cPI3_8 - r1 * cPI1_8 );
cannam@86 181 x[9] = MULT_NORM( r1 * cPI3_8 + r0 * cPI1_8 );
cannam@86 182
cannam@86 183 r0 = x[22] - x[6];
cannam@86 184 r1 = x[7] - x[23];
cannam@86 185 x[22] += x[6];
cannam@86 186 x[23] += x[7];
cannam@86 187 x[6] = r1;
cannam@86 188 x[7] = r0;
cannam@86 189
cannam@86 190 r0 = x[4] - x[20];
cannam@86 191 r1 = x[5] - x[21];
cannam@86 192 x[20] += x[4];
cannam@86 193 x[21] += x[5];
cannam@86 194 x[4] = MULT_NORM( r1 * cPI1_8 + r0 * cPI3_8 );
cannam@86 195 x[5] = MULT_NORM( r1 * cPI3_8 - r0 * cPI1_8 );
cannam@86 196
cannam@86 197 r0 = x[2] - x[18];
cannam@86 198 r1 = x[3] - x[19];
cannam@86 199 x[18] += x[2];
cannam@86 200 x[19] += x[3];
cannam@86 201 x[2] = MULT_NORM(( r1 + r0 ) * cPI2_8);
cannam@86 202 x[3] = MULT_NORM(( r1 - r0 ) * cPI2_8);
cannam@86 203
cannam@86 204 r0 = x[0] - x[16];
cannam@86 205 r1 = x[1] - x[17];
cannam@86 206 x[16] += x[0];
cannam@86 207 x[17] += x[1];
cannam@86 208 x[0] = MULT_NORM( r1 * cPI3_8 + r0 * cPI1_8 );
cannam@86 209 x[1] = MULT_NORM( r1 * cPI1_8 - r0 * cPI3_8 );
cannam@86 210
cannam@86 211 mdct_butterfly_16(x);
cannam@86 212 mdct_butterfly_16(x+16);
cannam@86 213
cannam@86 214 }
cannam@86 215
cannam@86 216 /* N point first stage butterfly (in place, 2 register) */
cannam@86 217 STIN void mdct_butterfly_first(DATA_TYPE *T,
cannam@86 218 DATA_TYPE *x,
cannam@86 219 int points){
cannam@86 220
cannam@86 221 DATA_TYPE *x1 = x + points - 8;
cannam@86 222 DATA_TYPE *x2 = x + (points>>1) - 8;
cannam@86 223 REG_TYPE r0;
cannam@86 224 REG_TYPE r1;
cannam@86 225
cannam@86 226 do{
cannam@86 227
cannam@86 228 r0 = x1[6] - x2[6];
cannam@86 229 r1 = x1[7] - x2[7];
cannam@86 230 x1[6] += x2[6];
cannam@86 231 x1[7] += x2[7];
cannam@86 232 x2[6] = MULT_NORM(r1 * T[1] + r0 * T[0]);
cannam@86 233 x2[7] = MULT_NORM(r1 * T[0] - r0 * T[1]);
cannam@86 234
cannam@86 235 r0 = x1[4] - x2[4];
cannam@86 236 r1 = x1[5] - x2[5];
cannam@86 237 x1[4] += x2[4];
cannam@86 238 x1[5] += x2[5];
cannam@86 239 x2[4] = MULT_NORM(r1 * T[5] + r0 * T[4]);
cannam@86 240 x2[5] = MULT_NORM(r1 * T[4] - r0 * T[5]);
cannam@86 241
cannam@86 242 r0 = x1[2] - x2[2];
cannam@86 243 r1 = x1[3] - x2[3];
cannam@86 244 x1[2] += x2[2];
cannam@86 245 x1[3] += x2[3];
cannam@86 246 x2[2] = MULT_NORM(r1 * T[9] + r0 * T[8]);
cannam@86 247 x2[3] = MULT_NORM(r1 * T[8] - r0 * T[9]);
cannam@86 248
cannam@86 249 r0 = x1[0] - x2[0];
cannam@86 250 r1 = x1[1] - x2[1];
cannam@86 251 x1[0] += x2[0];
cannam@86 252 x1[1] += x2[1];
cannam@86 253 x2[0] = MULT_NORM(r1 * T[13] + r0 * T[12]);
cannam@86 254 x2[1] = MULT_NORM(r1 * T[12] - r0 * T[13]);
cannam@86 255
cannam@86 256 x1-=8;
cannam@86 257 x2-=8;
cannam@86 258 T+=16;
cannam@86 259
cannam@86 260 }while(x2>=x);
cannam@86 261 }
cannam@86 262
cannam@86 263 /* N/stage point generic N stage butterfly (in place, 2 register) */
cannam@86 264 STIN void mdct_butterfly_generic(DATA_TYPE *T,
cannam@86 265 DATA_TYPE *x,
cannam@86 266 int points,
cannam@86 267 int trigint){
cannam@86 268
cannam@86 269 DATA_TYPE *x1 = x + points - 8;
cannam@86 270 DATA_TYPE *x2 = x + (points>>1) - 8;
cannam@86 271 REG_TYPE r0;
cannam@86 272 REG_TYPE r1;
cannam@86 273
cannam@86 274 do{
cannam@86 275
cannam@86 276 r0 = x1[6] - x2[6];
cannam@86 277 r1 = x1[7] - x2[7];
cannam@86 278 x1[6] += x2[6];
cannam@86 279 x1[7] += x2[7];
cannam@86 280 x2[6] = MULT_NORM(r1 * T[1] + r0 * T[0]);
cannam@86 281 x2[7] = MULT_NORM(r1 * T[0] - r0 * T[1]);
cannam@86 282
cannam@86 283 T+=trigint;
cannam@86 284
cannam@86 285 r0 = x1[4] - x2[4];
cannam@86 286 r1 = x1[5] - x2[5];
cannam@86 287 x1[4] += x2[4];
cannam@86 288 x1[5] += x2[5];
cannam@86 289 x2[4] = MULT_NORM(r1 * T[1] + r0 * T[0]);
cannam@86 290 x2[5] = MULT_NORM(r1 * T[0] - r0 * T[1]);
cannam@86 291
cannam@86 292 T+=trigint;
cannam@86 293
cannam@86 294 r0 = x1[2] - x2[2];
cannam@86 295 r1 = x1[3] - x2[3];
cannam@86 296 x1[2] += x2[2];
cannam@86 297 x1[3] += x2[3];
cannam@86 298 x2[2] = MULT_NORM(r1 * T[1] + r0 * T[0]);
cannam@86 299 x2[3] = MULT_NORM(r1 * T[0] - r0 * T[1]);
cannam@86 300
cannam@86 301 T+=trigint;
cannam@86 302
cannam@86 303 r0 = x1[0] - x2[0];
cannam@86 304 r1 = x1[1] - x2[1];
cannam@86 305 x1[0] += x2[0];
cannam@86 306 x1[1] += x2[1];
cannam@86 307 x2[0] = MULT_NORM(r1 * T[1] + r0 * T[0]);
cannam@86 308 x2[1] = MULT_NORM(r1 * T[0] - r0 * T[1]);
cannam@86 309
cannam@86 310 T+=trigint;
cannam@86 311 x1-=8;
cannam@86 312 x2-=8;
cannam@86 313
cannam@86 314 }while(x2>=x);
cannam@86 315 }
cannam@86 316
cannam@86 317 STIN void mdct_butterflies(mdct_lookup *init,
cannam@86 318 DATA_TYPE *x,
cannam@86 319 int points){
cannam@86 320
cannam@86 321 DATA_TYPE *T=init->trig;
cannam@86 322 int stages=init->log2n-5;
cannam@86 323 int i,j;
cannam@86 324
cannam@86 325 if(--stages>0){
cannam@86 326 mdct_butterfly_first(T,x,points);
cannam@86 327 }
cannam@86 328
cannam@86 329 for(i=1;--stages>0;i++){
cannam@86 330 for(j=0;j<(1<<i);j++)
cannam@86 331 mdct_butterfly_generic(T,x+(points>>i)*j,points>>i,4<<i);
cannam@86 332 }
cannam@86 333
cannam@86 334 for(j=0;j<points;j+=32)
cannam@86 335 mdct_butterfly_32(x+j);
cannam@86 336
cannam@86 337 }
cannam@86 338
cannam@86 339 void mdct_clear(mdct_lookup *l){
cannam@86 340 if(l){
cannam@86 341 if(l->trig)_ogg_free(l->trig);
cannam@86 342 if(l->bitrev)_ogg_free(l->bitrev);
cannam@86 343 memset(l,0,sizeof(*l));
cannam@86 344 }
cannam@86 345 }
cannam@86 346
cannam@86 347 STIN void mdct_bitreverse(mdct_lookup *init,
cannam@86 348 DATA_TYPE *x){
cannam@86 349 int n = init->n;
cannam@86 350 int *bit = init->bitrev;
cannam@86 351 DATA_TYPE *w0 = x;
cannam@86 352 DATA_TYPE *w1 = x = w0+(n>>1);
cannam@86 353 DATA_TYPE *T = init->trig+n;
cannam@86 354
cannam@86 355 do{
cannam@86 356 DATA_TYPE *x0 = x+bit[0];
cannam@86 357 DATA_TYPE *x1 = x+bit[1];
cannam@86 358
cannam@86 359 REG_TYPE r0 = x0[1] - x1[1];
cannam@86 360 REG_TYPE r1 = x0[0] + x1[0];
cannam@86 361 REG_TYPE r2 = MULT_NORM(r1 * T[0] + r0 * T[1]);
cannam@86 362 REG_TYPE r3 = MULT_NORM(r1 * T[1] - r0 * T[0]);
cannam@86 363
cannam@86 364 w1 -= 4;
cannam@86 365
cannam@86 366 r0 = HALVE(x0[1] + x1[1]);
cannam@86 367 r1 = HALVE(x0[0] - x1[0]);
cannam@86 368
cannam@86 369 w0[0] = r0 + r2;
cannam@86 370 w1[2] = r0 - r2;
cannam@86 371 w0[1] = r1 + r3;
cannam@86 372 w1[3] = r3 - r1;
cannam@86 373
cannam@86 374 x0 = x+bit[2];
cannam@86 375 x1 = x+bit[3];
cannam@86 376
cannam@86 377 r0 = x0[1] - x1[1];
cannam@86 378 r1 = x0[0] + x1[0];
cannam@86 379 r2 = MULT_NORM(r1 * T[2] + r0 * T[3]);
cannam@86 380 r3 = MULT_NORM(r1 * T[3] - r0 * T[2]);
cannam@86 381
cannam@86 382 r0 = HALVE(x0[1] + x1[1]);
cannam@86 383 r1 = HALVE(x0[0] - x1[0]);
cannam@86 384
cannam@86 385 w0[2] = r0 + r2;
cannam@86 386 w1[0] = r0 - r2;
cannam@86 387 w0[3] = r1 + r3;
cannam@86 388 w1[1] = r3 - r1;
cannam@86 389
cannam@86 390 T += 4;
cannam@86 391 bit += 4;
cannam@86 392 w0 += 4;
cannam@86 393
cannam@86 394 }while(w0<w1);
cannam@86 395 }
cannam@86 396
cannam@86 397 void mdct_backward(mdct_lookup *init, DATA_TYPE *in, DATA_TYPE *out){
cannam@86 398 int n=init->n;
cannam@86 399 int n2=n>>1;
cannam@86 400 int n4=n>>2;
cannam@86 401
cannam@86 402 /* rotate */
cannam@86 403
cannam@86 404 DATA_TYPE *iX = in+n2-7;
cannam@86 405 DATA_TYPE *oX = out+n2+n4;
cannam@86 406 DATA_TYPE *T = init->trig+n4;
cannam@86 407
cannam@86 408 do{
cannam@86 409 oX -= 4;
cannam@86 410 oX[0] = MULT_NORM(-iX[2] * T[3] - iX[0] * T[2]);
cannam@86 411 oX[1] = MULT_NORM (iX[0] * T[3] - iX[2] * T[2]);
cannam@86 412 oX[2] = MULT_NORM(-iX[6] * T[1] - iX[4] * T[0]);
cannam@86 413 oX[3] = MULT_NORM (iX[4] * T[1] - iX[6] * T[0]);
cannam@86 414 iX -= 8;
cannam@86 415 T += 4;
cannam@86 416 }while(iX>=in);
cannam@86 417
cannam@86 418 iX = in+n2-8;
cannam@86 419 oX = out+n2+n4;
cannam@86 420 T = init->trig+n4;
cannam@86 421
cannam@86 422 do{
cannam@86 423 T -= 4;
cannam@86 424 oX[0] = MULT_NORM (iX[4] * T[3] + iX[6] * T[2]);
cannam@86 425 oX[1] = MULT_NORM (iX[4] * T[2] - iX[6] * T[3]);
cannam@86 426 oX[2] = MULT_NORM (iX[0] * T[1] + iX[2] * T[0]);
cannam@86 427 oX[3] = MULT_NORM (iX[0] * T[0] - iX[2] * T[1]);
cannam@86 428 iX -= 8;
cannam@86 429 oX += 4;
cannam@86 430 }while(iX>=in);
cannam@86 431
cannam@86 432 mdct_butterflies(init,out+n2,n2);
cannam@86 433 mdct_bitreverse(init,out);
cannam@86 434
cannam@86 435 /* roatate + window */
cannam@86 436
cannam@86 437 {
cannam@86 438 DATA_TYPE *oX1=out+n2+n4;
cannam@86 439 DATA_TYPE *oX2=out+n2+n4;
cannam@86 440 DATA_TYPE *iX =out;
cannam@86 441 T =init->trig+n2;
cannam@86 442
cannam@86 443 do{
cannam@86 444 oX1-=4;
cannam@86 445
cannam@86 446 oX1[3] = MULT_NORM (iX[0] * T[1] - iX[1] * T[0]);
cannam@86 447 oX2[0] = -MULT_NORM (iX[0] * T[0] + iX[1] * T[1]);
cannam@86 448
cannam@86 449 oX1[2] = MULT_NORM (iX[2] * T[3] - iX[3] * T[2]);
cannam@86 450 oX2[1] = -MULT_NORM (iX[2] * T[2] + iX[3] * T[3]);
cannam@86 451
cannam@86 452 oX1[1] = MULT_NORM (iX[4] * T[5] - iX[5] * T[4]);
cannam@86 453 oX2[2] = -MULT_NORM (iX[4] * T[4] + iX[5] * T[5]);
cannam@86 454
cannam@86 455 oX1[0] = MULT_NORM (iX[6] * T[7] - iX[7] * T[6]);
cannam@86 456 oX2[3] = -MULT_NORM (iX[6] * T[6] + iX[7] * T[7]);
cannam@86 457
cannam@86 458 oX2+=4;
cannam@86 459 iX += 8;
cannam@86 460 T += 8;
cannam@86 461 }while(iX<oX1);
cannam@86 462
cannam@86 463 iX=out+n2+n4;
cannam@86 464 oX1=out+n4;
cannam@86 465 oX2=oX1;
cannam@86 466
cannam@86 467 do{
cannam@86 468 oX1-=4;
cannam@86 469 iX-=4;
cannam@86 470
cannam@86 471 oX2[0] = -(oX1[3] = iX[3]);
cannam@86 472 oX2[1] = -(oX1[2] = iX[2]);
cannam@86 473 oX2[2] = -(oX1[1] = iX[1]);
cannam@86 474 oX2[3] = -(oX1[0] = iX[0]);
cannam@86 475
cannam@86 476 oX2+=4;
cannam@86 477 }while(oX2<iX);
cannam@86 478
cannam@86 479 iX=out+n2+n4;
cannam@86 480 oX1=out+n2+n4;
cannam@86 481 oX2=out+n2;
cannam@86 482 do{
cannam@86 483 oX1-=4;
cannam@86 484 oX1[0]= iX[3];
cannam@86 485 oX1[1]= iX[2];
cannam@86 486 oX1[2]= iX[1];
cannam@86 487 oX1[3]= iX[0];
cannam@86 488 iX+=4;
cannam@86 489 }while(oX1>oX2);
cannam@86 490 }
cannam@86 491 }
cannam@86 492
cannam@86 493 void mdct_forward(mdct_lookup *init, DATA_TYPE *in, DATA_TYPE *out){
cannam@86 494 int n=init->n;
cannam@86 495 int n2=n>>1;
cannam@86 496 int n4=n>>2;
cannam@86 497 int n8=n>>3;
cannam@86 498 DATA_TYPE *w=alloca(n*sizeof(*w)); /* forward needs working space */
cannam@86 499 DATA_TYPE *w2=w+n2;
cannam@86 500
cannam@86 501 /* rotate */
cannam@86 502
cannam@86 503 /* window + rotate + step 1 */
cannam@86 504
cannam@86 505 REG_TYPE r0;
cannam@86 506 REG_TYPE r1;
cannam@86 507 DATA_TYPE *x0=in+n2+n4;
cannam@86 508 DATA_TYPE *x1=x0+1;
cannam@86 509 DATA_TYPE *T=init->trig+n2;
cannam@86 510
cannam@86 511 int i=0;
cannam@86 512
cannam@86 513 for(i=0;i<n8;i+=2){
cannam@86 514 x0 -=4;
cannam@86 515 T-=2;
cannam@86 516 r0= x0[2] + x1[0];
cannam@86 517 r1= x0[0] + x1[2];
cannam@86 518 w2[i]= MULT_NORM(r1*T[1] + r0*T[0]);
cannam@86 519 w2[i+1]= MULT_NORM(r1*T[0] - r0*T[1]);
cannam@86 520 x1 +=4;
cannam@86 521 }
cannam@86 522
cannam@86 523 x1=in+1;
cannam@86 524
cannam@86 525 for(;i<n2-n8;i+=2){
cannam@86 526 T-=2;
cannam@86 527 x0 -=4;
cannam@86 528 r0= x0[2] - x1[0];
cannam@86 529 r1= x0[0] - x1[2];
cannam@86 530 w2[i]= MULT_NORM(r1*T[1] + r0*T[0]);
cannam@86 531 w2[i+1]= MULT_NORM(r1*T[0] - r0*T[1]);
cannam@86 532 x1 +=4;
cannam@86 533 }
cannam@86 534
cannam@86 535 x0=in+n;
cannam@86 536
cannam@86 537 for(;i<n2;i+=2){
cannam@86 538 T-=2;
cannam@86 539 x0 -=4;
cannam@86 540 r0= -x0[2] - x1[0];
cannam@86 541 r1= -x0[0] - x1[2];
cannam@86 542 w2[i]= MULT_NORM(r1*T[1] + r0*T[0]);
cannam@86 543 w2[i+1]= MULT_NORM(r1*T[0] - r0*T[1]);
cannam@86 544 x1 +=4;
cannam@86 545 }
cannam@86 546
cannam@86 547
cannam@86 548 mdct_butterflies(init,w+n2,n2);
cannam@86 549 mdct_bitreverse(init,w);
cannam@86 550
cannam@86 551 /* roatate + window */
cannam@86 552
cannam@86 553 T=init->trig+n2;
cannam@86 554 x0=out+n2;
cannam@86 555
cannam@86 556 for(i=0;i<n4;i++){
cannam@86 557 x0--;
cannam@86 558 out[i] =MULT_NORM((w[0]*T[0]+w[1]*T[1])*init->scale);
cannam@86 559 x0[0] =MULT_NORM((w[0]*T[1]-w[1]*T[0])*init->scale);
cannam@86 560 w+=2;
cannam@86 561 T+=2;
cannam@86 562 }
cannam@86 563 }