www.pudn.com > helix.src.0812.rar > xfade.c


/* ***** BEGIN LICENSE BLOCK *****  
 * Version: RCSL 1.0/RPSL 1.0  
 *   
 * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved.  
 *       
 * The contents of this file, and the files included with this file, are  
 * subject to the current version of the RealNetworks Public Source License  
 * Version 1.0 (the "RPSL") available at  
 * http://www.helixcommunity.org/content/rpsl unless you have licensed  
 * the file under the RealNetworks Community Source License Version 1.0  
 * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl,  
 * in which case the RCSL will apply. You may also obtain the license terms  
 * directly from RealNetworks.  You may not use this file except in  
 * compliance with the RPSL or, if you have a valid RCSL with RealNetworks  
 * applicable to this file, the RCSL.  Please see the applicable RPSL or  
 * RCSL for the rights, obligations and limitations governing use of the  
 * contents of the file.   
 *   
 * This file is part of the Helix DNA Technology. RealNetworks is the  
 * developer of the Original Code and owns the copyrights in the portions  
 * it created.  
 *   
 * This file, and the files included with this file, is distributed and made  
 * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER  
 * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES,  
 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS  
 * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.  
 *  
 * Technology Compatibility Kit Test Suite(s) Location:  
 *    http://www.helixcommunity.org/content/tck  
 *  
 * Contributor(s):  
 *   
 * ***** END LICENSE BLOCK ***** */  
 
#include  
 
#include "xfade.h" 
#include "math64.h" 
 
#ifndef MIN 
#define MIN(a,b) ((a)<=(b)?(a):(b)) 
#endif 
 
#define HEADROOM 0 
 
#define NALPHA 256 
#define FRACBITS 16 
#define EXTRABITS 8 
#define FRACMASK ((1<nChannels = nChannels ; 
        instance->coeff = coeff ; 
        instance->tabacc = NALPHA << FRACBITS ; // end of fade 
 
        // make sure batchsize is divisible by nChannels 
        instance->batchSize = BATCHSIZE - BATCHSIZE % nChannels ; 
    } 
 
    return instance ; 
} 
 
void XFader_free(XFADER_STATE* instance) 
{ 
    if (instance) free(instance) ; 
} 
 
void XFader_start(int nSamples, XFADER_STATE* instance) 
{ 
    nSamples /= instance->nChannels ; 
    instance->tabacc = 0 ; 
 
    if (nSamples < 1) nSamples = 1 ; 
 
    // (tabstep * nSamples) >> FRACBITS = NALPHA 
 
//    instance->tabstep = ((NALPHA << FRACBITS) + nSamples - 1) / nSamples ; // round up 
    instance->tabstep = ((NALPHA << FRACBITS) + nSamples/2) / nSamples ; // round nearest 
} 
 
int XFader_active(XFADER_STATE* instance) 
{ 
    unsigned int tabacc = instance->tabacc ; 
    unsigned int tabint = tabacc >> FRACBITS ; 
    return (tabint < NALPHA) ; 
} 
 
/* In-place crossfading functionality currently untested */ 
#if 0 
static int XFader_feed_mono(const INT32* in, INT32* inout, int nSamples, XFADER_STATE* instance) 
{ 
    int i ; 
    unsigned int tabacc = instance->tabacc ; 
    unsigned int tabstep = instance->tabstep ; 
    unsigned int tabint = tabacc >> FRACBITS ; 
 
    for (i = 0 ; i < nSamples && tabint < NALPHA ; i++) 
    { 
        INT32 gain1, gain2 ; 
 
        /* interpolate new alpha */ 
        gain1 = instance->coeff[tabint].gain ; 
        gain1 += (instance->coeff[tabint].delta * (signed)(tabacc & FRACMASK)) >> (FRACBITS - EXTRABITS) ; 
 
        gain2 = instance->coeff[tabint+NALPHA+1].gain ; 
        gain2 += (instance->coeff[tabint+NALPHA+1].delta * (signed)(tabacc & FRACMASK)) >> (FRACBITS - EXTRABITS) ; 
 
        /* next table step */ 
        tabacc += tabstep; 
        tabint = tabacc >> FRACBITS; 
 
        // creates one guard bit 
        *inout = MulShift31(*in++, gain1) + MulShift31(*inout, gain2) ; 
        inout++ ; 
    } 
 
    instance->tabacc = tabacc ; 
 
    return nSamples ; 
} 
 
static int XFader_feed_stereo(const INT32* in, INT32* inout, int nSamples, XFADER_STATE* instance) 
{ 
    int i ; 
    unsigned int tabacc = instance->tabacc ; 
    unsigned int tabstep = instance->tabstep ; 
    unsigned int tabint = tabacc >> FRACBITS ; 
 
    for (i = 0 ; i < nSamples && tabint < NALPHA ; i+=2) 
    { 
        INT32 gain1, gain2 ; 
 
        /* interpolate new alpha */ 
        gain1 = instance->coeff[tabint].gain ; 
        gain1 += (instance->coeff[tabint].delta * (signed)(tabacc & FRACMASK)) >> (FRACBITS - EXTRABITS) ; 
 
        gain2 = instance->coeff[tabint+NALPHA+1].gain ; 
        gain2 += (instance->coeff[tabint+NALPHA+1].delta * (signed)(tabacc & FRACMASK)) >> (FRACBITS - EXTRABITS) ; 
 
        /* next table step */ 
        tabacc += tabstep; 
        tabint = tabacc >> FRACBITS; 
 
        // creates one guard bit 
        *inout = MulShift31(*in++, gain1) + MulShift31(*inout, gain2) ; 
        inout++ ; 
        *inout = MulShift31(*in++, gain1) + MulShift31(*inout, gain2) ; 
        inout++ ; 
    } 
 
    instance->tabacc = tabacc ; 
 
    return nSamples ; 
} 
 
static const INT32 silence[BATCHSIZE] ; // a bunch of zeros 
 
int XFader_feed(const INT32* in, INT32* inout, int nSamples, XFADER_STATE* instance) 
{ 
    int nRead = 0 ; 
 
    while (nRead < nSamples) 
    { 
        int nBatch = MIN(nSamples - nRead, instance->batchSize) ; 
        const INT32 *in1 = in ? (in + nRead) : silence ; 
        int i ; 
 
        switch (instance->nChannels) 
        { 
        case 1: 
            i = XFader_feed_mono  (in1, inout + nRead, nBatch, instance) ; 
            break ; 
        case 2: 
            i = XFader_feed_stereo(in1, inout + nRead, nBatch, instance) ; 
            break ; 
        } 
 
#if HEADROOM > 0 
        for (; i < nBatch ; i++) 
        { 
            inout[nRead + i] >>= HEADROOM ; 
        } 
#endif 
 
        nRead += nBatch ; 
    } 
 
    return nSamples ; 
} 
#endif 
 
int Fader_feed_mono(INT32 *inout, int nSamples, int fadeout, XFADER_STATE* instance) 
{ 
    int i ; 
    unsigned int tabacc = instance->tabacc ; 
    unsigned int tabstep = instance->tabstep ; 
    unsigned int tabint = tabacc >> FRACBITS ; 
    const struct COEFF *coeff = instance->coeff + (fadeout ? 0 : (NALPHA+1)) ; 
 
    for (i = 0 ; i < nSamples && tabint < NALPHA ; i++) 
    { 
        /* interpolate new alpha */ 
        INT32 gain = coeff[tabint].gain ; 
        gain += (coeff[tabint].delta * (signed)(tabacc & FRACMASK)) >> (FRACBITS - EXTRABITS) ; 
 
        /* next table step */ 
        tabacc += tabstep; 
        tabint = tabacc >> FRACBITS; 
 
        inout[0] = MulShift30(inout[0], gain) ; 
        inout++ ; 
    } 
 
    instance->tabacc = tabacc ; 
 
    return i ; 
} 
 
int Fader_feed_stereo(INT32 *inout, int nSamples, int fadeout, XFADER_STATE* instance) 
{ 
    int i ; 
    unsigned int tabacc = instance->tabacc ; 
    unsigned int tabstep = instance->tabstep ; 
    unsigned int tabint = tabacc >> FRACBITS ; 
    const struct COEFF *coeff = instance->coeff + (fadeout ? 0 : (NALPHA+1)) ; 
 
    for (i = 0 ; i < nSamples && tabint < NALPHA ; i+=2) 
    { 
        /* interpolate new alpha */ 
        INT32 gain = coeff[tabint].gain ; 
        gain += (coeff[tabint].delta * (signed)(tabacc & FRACMASK)) >> (FRACBITS - EXTRABITS) ; 
 
        /* next table step */ 
        tabacc += tabstep; 
        tabint = tabacc >> FRACBITS; 
 
        inout[0] = MulShift30(inout[0], gain) ; 
        inout[1] = MulShift30(inout[1], gain) ; 
        inout+=2 ; 
    } 
 
    instance->tabacc = tabacc ; 
 
    return i ; 
} 
 
int Fader_feed(INT32 *inout, int nSamples, int fadeout, XFADER_STATE* instance) 
{ 
    int i ; 
    ASSERT(fadeout ==1 || fadeout==0); 
 
    switch (instance->nChannels) 
    { 
    case 1: 
        i = Fader_feed_mono  (inout, nSamples, fadeout, instance) ; 
        break ; 
    case 2: 
        i = Fader_feed_stereo(inout, nSamples, fadeout, instance) ; 
        break ; 
    } 
 
    if (fadeout) 
        for (; i < nSamples ; i++) inout[i] = 0 ; 
#if HEADROOM > 0 
    else 
        for (; i < nSamples ; i++) inout[i] >>= HEADROOM ; 
#endif 
 
    return nSamples ; 
} 
 
#if 0 // use just for table initialization 
static void setupDelta(struct COEFF* c) 
{ 
    int i ; 
 
    for (i = 0 ; i < NALPHA ; i++) 
    { 
        c[i].delta = (c[i+1].gain - c[i].gain + (1<<(EXTRABITS-1))) >> EXTRABITS ; 
        assert(c[i].delta < (1<<(31-FRACBITS))); 
        c[i+NALPHA+1].delta = (c[i+NALPHA+2].gain - c[i+NALPHA+1].gain + (1<<(EXTRABITS-1))) >> EXTRABITS ; 
        assert(c[i+NALPHA+1].delta < (1<<(31-FRACBITS))); 
    } 
    c[i].delta = c[i+NALPHA+1].delta = 0 ; 
} 
 
#include  
#include  
#include  
 
#ifndef M_PI 
#define M_PI 3.14159265358979323846 
#endif 
 
struct COEFF* sintab(void) 
{ 
    int i ; 
    struct COEFF* c = (struct COEFF*) calloc(NALPHA+1, 2*sizeof(struct COEFF)) ; 
 
    for (i = 0 ; i <= NALPHA ; i++) 
    { 
        double x = i * 0.5 * M_PI / NALPHA ; 
        double x1 = cos(x) ; 
        double x2 = sin(x) ; 
 
        x1 *= x1 ; 
        x2 *= x2 ; 
 
        c[i].gain          = (int)floor(0.5 + x1*(1L<<(30-HEADROOM))) ; 
        c[i+NALPHA+1].gain = (int)floor(0.5 + x2*(1L<<(30-HEADROOM))) ; 
    } 
    setupDelta(c) ; 
 
 
    { 
      FILE *f = fopen("c:\\temp\\table.c","w") ; 
 
      fprintf(f,"const struct COEFF XFader_sin2tab[2*%d] = {\n /* fade out table */\n", 
        NALPHA+1) ; 
      for (i=0; i