/*----------------------------------------------------------------------------
 *
 * File:
 * eas_reverb.c
 *
 * Contents and purpose:
 * Contains the implementation of the Reverb effect.
 *
 *
 * Copyright Sonic Network Inc. 2006

 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 *----------------------------------------------------------------------------
 * Revision Control:
 *   $Revision: 510 $
 *   $Date: 2006-12-19 01:47:33 -0800 (Tue, 19 Dec 2006) $
 *----------------------------------------------------------------------------
*/

/*------------------------------------
 * includes
 *------------------------------------
*/

#include "eas_data.h"
#include "eas_effects.h"
#include "eas_math.h"
#include "eas_reverbdata.h"
#include "eas_reverb.h"
#include "eas_config.h"
#include "eas_host.h"
#include "eas_report.h"

/* prototypes for effects interface */
static EAS_RESULT ReverbInit (EAS_DATA_HANDLE pEASData, EAS_VOID_PTR *pInstData);
static void ReverbProcess (EAS_VOID_PTR pInstData, EAS_PCM *pSrc, EAS_PCM *pDst, EAS_I32 numSamples);
static EAS_RESULT ReverbShutdown (EAS_DATA_HANDLE pEASData, EAS_VOID_PTR pInstData);
static EAS_RESULT ReverbGetParam (EAS_VOID_PTR pInstData, EAS_I32 param, EAS_I32 *pValue);
static EAS_RESULT ReverbSetParam (EAS_VOID_PTR pInstData, EAS_I32 param, EAS_I32 value);

/* common effects interface for configuration module */
const S_EFFECTS_INTERFACE EAS_Reverb =
{
    ReverbInit,
    ReverbProcess,
    ReverbShutdown,
    ReverbGetParam,
    ReverbSetParam
};



/*----------------------------------------------------------------------------
 * InitializeReverb()
 *----------------------------------------------------------------------------
 * Purpose:
 *
 * Inputs:
 *
 * Outputs:
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbInit(EAS_DATA_HANDLE pEASData, EAS_VOID_PTR *pInstData)
{
    EAS_I32 i;
    EAS_U16 nOffset;
    EAS_INT temp;

    S_REVERB_OBJECT *pReverbData;
    S_REVERB_PRESET *pPreset;

    /* check Configuration Module for data allocation */
    if (pEASData->staticMemoryModel)
        pReverbData = EAS_CMEnumFXData(EAS_MODULE_REVERB);

    /* allocate dynamic memory */
    else
        pReverbData = EAS_HWMalloc(pEASData->hwInstData, sizeof(S_REVERB_OBJECT));

    if (pReverbData == NULL)
    {
        { /* dpp: EAS_ReportEx(_EAS_SEVERITY_FATAL, "Failed to allocate Reverb memory\n"); */ }
        return EAS_ERROR_MALLOC_FAILED;
    }

    /* clear the structure */
    EAS_HWMemSet(pReverbData, 0, sizeof(S_REVERB_OBJECT));

    ReverbReadInPresets(pReverbData);

    pReverbData->m_nMinSamplesToAdd = REVERB_UPDATE_PERIOD_IN_SAMPLES;

    pReverbData->m_nRevOutFbkR = 0;
    pReverbData->m_nRevOutFbkL = 0;

    pReverbData->m_sAp0.m_zApIn  = AP0_IN;
    pReverbData->m_sAp0.m_zApOut = AP0_IN + DEFAULT_AP0_LENGTH;
    pReverbData->m_sAp0.m_nApGain = DEFAULT_AP0_GAIN;

    pReverbData->m_zD0In = DELAY0_IN;

    pReverbData->m_sAp1.m_zApIn  = AP1_IN;
    pReverbData->m_sAp1.m_zApOut = AP1_IN + DEFAULT_AP1_LENGTH;
    pReverbData->m_sAp1.m_nApGain = DEFAULT_AP1_GAIN;

    pReverbData->m_zD1In = DELAY1_IN;

    pReverbData->m_zLpf0    = 0;
    pReverbData->m_zLpf1    = 0;
    pReverbData->m_nLpfFwd  = 8837;
    pReverbData->m_nLpfFbk  = 6494;

    pReverbData->m_nSin     = 0;
    pReverbData->m_nCos     = 0;
    pReverbData->m_nSinIncrement    = 0;
    pReverbData->m_nCosIncrement    = 0;

    // set xfade parameters
    pReverbData->m_nXfadeInterval = (EAS_U16)REVERB_XFADE_PERIOD_IN_SAMPLES;
    pReverbData->m_nXfadeCounter = pReverbData->m_nXfadeInterval + 1;   // force update on first iteration
    pReverbData->m_nPhase = -32768;
    pReverbData->m_nPhaseIncrement = REVERB_XFADE_PHASE_INCREMENT;

    pReverbData->m_nNoise = (EAS_I16)0xABCD;

    pReverbData->m_nMaxExcursion = 0x007F;

    // set delay tap lengths
    nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion,
                                    &pReverbData->m_nNoise );

    pReverbData->m_zD1Cross =
        DELAY1_OUT - pReverbData->m_nMaxExcursion + nOffset;

    nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion,
                                    &pReverbData->m_nNoise );

    pReverbData->m_zD0Cross =
        DELAY1_OUT - pReverbData->m_nMaxExcursion - nOffset;

    nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion,
                                    &pReverbData->m_nNoise );

    pReverbData->m_zD0Self  =
        DELAY0_OUT - pReverbData->m_nMaxExcursion - nOffset;

    nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion,
                                    &pReverbData->m_nNoise );

    pReverbData->m_zD1Self  =
        DELAY1_OUT - pReverbData->m_nMaxExcursion + nOffset;

    // for debugging purposes, allow noise generator
    pReverbData->m_bUseNoise = EAS_FALSE;

    // for debugging purposes, allow bypass
    pReverbData->m_bBypass = EAS_TRUE;  //EAS_FALSE;

    pReverbData->m_nNextRoom = 1;

    pReverbData->m_nCurrentRoom = pReverbData->m_nNextRoom + 1; // force update on first iteration

    pReverbData->m_nWet = REVERB_DEFAULT_WET;

    pReverbData->m_nDry = REVERB_DEFAULT_DRY;

    // set base index into circular buffer
    pReverbData->m_nBaseIndex = 0;

    // set the early reflections, L
    pReverbData->m_sEarlyL.m_nLpfFbk = 4915;
    pReverbData->m_sEarlyL.m_nLpfFwd = 27852;
    pReverbData->m_sEarlyL.m_zLpf = 0;

    for (i=0; i < REVERB_MAX_NUM_REFLECTIONS; i++)
    {
        pReverbData->m_sEarlyL.m_nGain[i] = 0;
        pReverbData->m_sEarlyL.m_zDelay[i] = 0;
    }

    // set the early reflections, R
    pReverbData->m_sEarlyR.m_nLpfFbk = 4915;
    pReverbData->m_sEarlyR.m_nLpfFwd = 27852;
    pReverbData->m_sEarlyR.m_zLpf = 0;

    for (i=0; i < REVERB_MAX_NUM_REFLECTIONS; i++)
    {
        pReverbData->m_sEarlyR.m_nGain[i] = 0;
        pReverbData->m_sEarlyR.m_zDelay[i] = 0;
    }

    // clear the reverb delay line
    for (i=0; i < REVERB_BUFFER_SIZE_IN_SAMPLES; i++)
    {
        pReverbData->m_nDelayLine[i] = 0;
    }

    ////////////////////////////////
    ///code from the EAS DEMO Reverb
    //now copy from the new preset into the reverb
    pPreset = &pReverbData->m_sPreset.m_sPreset[pReverbData->m_nNextRoom];

    pReverbData->m_nLpfFbk = pPreset->m_nLpfFbk;
    pReverbData->m_nLpfFwd = pPreset->m_nLpfFwd;

    pReverbData->m_nEarly = pPreset->m_nEarly;
    pReverbData->m_nWet = pPreset->m_nWet;
    pReverbData->m_nDry = pPreset->m_nDry;

    pReverbData->m_nMaxExcursion = pPreset->m_nMaxExcursion;
    //stored as time based, convert to sample based
    temp = pPreset->m_nXfadeInterval;
    /*lint -e{702} shift for performance */
    temp = (temp * _OUTPUT_SAMPLE_RATE) >> 16;
    pReverbData->m_nXfadeInterval = (EAS_U16) temp;
    //gsReverbObject.m_nXfadeInterval = pPreset->m_nXfadeInterval;

    pReverbData->m_sAp0.m_nApGain = pPreset->m_nAp0_ApGain;
    //stored as time based, convert to absolute sample value
    temp = pPreset->m_nAp0_ApOut;
    /*lint -e{702} shift for performance */
    temp = (temp * _OUTPUT_SAMPLE_RATE) >> 16;
    pReverbData->m_sAp0.m_zApOut = (EAS_U16) (pReverbData->m_sAp0.m_zApIn + temp);
    //gsReverbObject.m_sAp0.m_zApOut = pPreset->m_nAp0_ApOut;

    pReverbData->m_sAp1.m_nApGain = pPreset->m_nAp1_ApGain;
    //stored as time based, convert to absolute sample value
    temp = pPreset->m_nAp1_ApOut;
    /*lint -e{702} shift for performance */
    temp = (temp * _OUTPUT_SAMPLE_RATE) >> 16;
    pReverbData->m_sAp1.m_zApOut = (EAS_U16) (pReverbData->m_sAp1.m_zApIn + temp);
    //gsReverbObject.m_sAp1.m_zApOut = pPreset->m_nAp1_ApOut;
    ///code from the EAS DEMO Reverb
    ////////////////////////////////

    *pInstData = pReverbData;

    return EAS_SUCCESS;

}   /* end InitializeReverb */



/*----------------------------------------------------------------------------
 * ReverbProcess()
 *----------------------------------------------------------------------------
 * Purpose:
 * Reverberate the requested number of samples (block based processing)
 *
 * Inputs:
 * pInputBuffer - src buffer
 * pOutputBuffer - dst buffer
 * nNumSamplesToAdd - number of samples to write to buffer
 *
 * Outputs:
 * number of samples actually written to buffer
 *
 * Side Effects:
 * - samples are added to the presently free buffer
 *
 *----------------------------------------------------------------------------
*/
static void ReverbProcess(EAS_VOID_PTR pInstData, EAS_PCM *pSrc, EAS_PCM *pDst, EAS_I32 numSamples)
{
    S_REVERB_OBJECT *pReverbData;

    pReverbData = (S_REVERB_OBJECT*) pInstData;

    //if bypassed or the preset forces the signal to be completely dry
    if (pReverbData->m_bBypass ||
        (pReverbData->m_nWet == 0 && pReverbData->m_nDry == 32767))
    {
        if (pSrc != pDst)
            EAS_HWMemCpy(pSrc, pDst, numSamples * NUM_OUTPUT_CHANNELS * (EAS_I32) sizeof(EAS_PCM));
        return;
    }

    if (pReverbData->m_nNextRoom != pReverbData->m_nCurrentRoom)
    {
        ReverbUpdateRoom(pReverbData);
    }

    ReverbUpdateXfade(pReverbData, numSamples);

    Reverb(pReverbData, numSamples, pDst, pSrc);

    /* check if update counter needs to be reset */
    if (pReverbData->m_nUpdateCounter >= REVERB_MODULO_UPDATE_PERIOD_IN_SAMPLES)
    {
        /* update interval has elapsed, so reset counter */
        pReverbData->m_nUpdateCounter = 0;
    }   /* end if m_nUpdateCounter >= update interval */

    /* increment update counter */
    pReverbData->m_nUpdateCounter += (EAS_I16)numSamples;

}   /* end ComputeReverb */

/*----------------------------------------------------------------------------
 * ReverbUpdateXfade
 *----------------------------------------------------------------------------
 * Purpose:
 * Update the xfade parameters as required
 *
 * Inputs:
 * nNumSamplesToAdd - number of samples to write to buffer
 *
 * Outputs:
 *
 *
 * Side Effects:
 * - xfade parameters will be changed
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbUpdateXfade(S_REVERB_OBJECT *pReverbData, EAS_INT nNumSamplesToAdd)
{
    EAS_U16 nOffset;
    EAS_I16 tempCos;
    EAS_I16 tempSin;

    if (pReverbData->m_nXfadeCounter >= pReverbData->m_nXfadeInterval)
    {
        /* update interval has elapsed, so reset counter */
        pReverbData->m_nXfadeCounter = 0;

        // Pin the sin,cos values to min / max values to ensure that the
        // modulated taps' coefs are zero (thus no clicks)
        if (pReverbData->m_nPhaseIncrement > 0)
        {
            // if phase increment > 0, then sin -> 1, cos -> 0
            pReverbData->m_nSin = 32767;
            pReverbData->m_nCos = 0;

            // reset the phase to match the sin, cos values
            pReverbData->m_nPhase = 32767;

            // modulate the cross taps because their tap coefs are zero
            nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion, &pReverbData->m_nNoise );

            pReverbData->m_zD1Cross =
                DELAY1_OUT - pReverbData->m_nMaxExcursion + nOffset;

            nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion, &pReverbData->m_nNoise );

            pReverbData->m_zD0Cross =
                DELAY0_OUT - pReverbData->m_nMaxExcursion - nOffset;
        }
        else
        {
            // if phase increment < 0, then sin -> 0, cos -> 1
            pReverbData->m_nSin = 0;
            pReverbData->m_nCos = 32767;

            // reset the phase to match the sin, cos values
            pReverbData->m_nPhase = -32768;

            // modulate the self taps because their tap coefs are zero
            nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion, &pReverbData->m_nNoise );

            pReverbData->m_zD0Self  =
                DELAY0_OUT - pReverbData->m_nMaxExcursion - nOffset;

            nOffset = ReverbCalculateNoise( pReverbData->m_nMaxExcursion, &pReverbData->m_nNoise );

            pReverbData->m_zD1Self  =
                DELAY1_OUT - pReverbData->m_nMaxExcursion + nOffset;

        }   // end if-else (pReverbData->m_nPhaseIncrement > 0)

        // Reverse the direction of the sin,cos so that the
        // tap whose coef was previously increasing now decreases
        // and vice versa
        pReverbData->m_nPhaseIncrement = -pReverbData->m_nPhaseIncrement;

    }   // end if counter >= update interval

    //compute what phase will be next time
    pReverbData->m_nPhase += pReverbData->m_nPhaseIncrement;

    //calculate what the new sin and cos need to reach by the next update
    ReverbCalculateSinCos(pReverbData->m_nPhase, &tempSin, &tempCos);

    //calculate the per-sample increment required to get there by the next update
    /*lint -e{702} shift for performance */
    pReverbData->m_nSinIncrement =
            (tempSin - pReverbData->m_nSin) >> REVERB_UPDATE_PERIOD_IN_BITS;

    /*lint -e{702} shift for performance */
    pReverbData->m_nCosIncrement =
            (tempCos - pReverbData->m_nCos) >> REVERB_UPDATE_PERIOD_IN_BITS;


    /* increment update counter */
    pReverbData->m_nXfadeCounter += (EAS_U16) nNumSamplesToAdd;

    return EAS_SUCCESS;

}   /* end ReverbUpdateXfade */


/*----------------------------------------------------------------------------
 * ReverbCalculateNoise
 *----------------------------------------------------------------------------
 * Purpose:
 * Calculate a noise sample and limit its value
 *
 * Inputs:
 * nMaxExcursion - noise value is limited to this value
 * pnNoise - return new noise sample in this (not limited)
 *
 * Outputs:
 * new limited noise value
 *
 * Side Effects:
 * - *pnNoise noise value is updated
 *
 *----------------------------------------------------------------------------
*/
static EAS_U16 ReverbCalculateNoise(EAS_U16 nMaxExcursion, EAS_I16 *pnNoise)
{
    // calculate new noise value
    *pnNoise = (EAS_I16) (*pnNoise * 5 + 1);

#if 0   // 1xxx, test
    *pnNoise = 0;
#endif  // 1xxx, test

    // return the limited noise value
    return (nMaxExcursion & (*pnNoise));

}   /* end ReverbCalculateNoise */

/*----------------------------------------------------------------------------
 * ReverbCalculateSinCos
 *----------------------------------------------------------------------------
 * Purpose:
 * Calculate a new sin and cosine value based on the given phase
 *
 * Inputs:
 * nPhase   - phase angle
 * pnSin    - input old value, output new value
 * pnCos    - input old value, output new value
 *
 * Outputs:
 *
 * Side Effects:
 * - *pnSin, *pnCos are updated
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbCalculateSinCos(EAS_I16 nPhase, EAS_I16 *pnSin, EAS_I16 *pnCos)
{
    EAS_I32 nTemp;
    EAS_I32 nNetAngle;

    //  -1 <=  nPhase  < 1
    // However, for the calculation, we need a value
    // that ranges from -1/2 to +1/2, so divide the phase by 2
    /*lint -e{702} shift for performance */
    nNetAngle = nPhase >> 1;

    /*
    Implement the following
    sin(x) = (2-4*c)*x^2 + c + x
    cos(x) = (2-4*c)*x^2 + c - x

      where  c = 1/sqrt(2)
    using the a0 + x*(a1 + x*a2) approach
    */

    /* limit the input "angle" to be between -0.5 and +0.5 */
    if (nNetAngle > EG1_HALF)
    {
        nNetAngle = EG1_HALF;
    }
    else if (nNetAngle < EG1_MINUS_HALF)
    {
        nNetAngle = EG1_MINUS_HALF;
    }

    /* calculate sin */
    nTemp = EG1_ONE + MULT_EG1_EG1(REVERB_PAN_G2, nNetAngle);
    nTemp = REVERB_PAN_G0 + MULT_EG1_EG1(nTemp, nNetAngle);
    *pnSin = (EAS_I16) SATURATE_EG1(nTemp);

    /* calculate cos */
    nTemp = -EG1_ONE + MULT_EG1_EG1(REVERB_PAN_G2, nNetAngle);
    nTemp = REVERB_PAN_G0 + MULT_EG1_EG1(nTemp, nNetAngle);
    *pnCos = (EAS_I16) SATURATE_EG1(nTemp);

    return EAS_SUCCESS;
}   /* end ReverbCalculateSinCos */

/*----------------------------------------------------------------------------
 * Reverb
 *----------------------------------------------------------------------------
 * Purpose:
 * apply reverb to the given signal
 *
 * Inputs:
 * nNu
 * pnSin    - input old value, output new value
 * pnCos    - input old value, output new value
 *
 * Outputs:
 * number of samples actually reverberated
 *
 * Side Effects:
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT Reverb(S_REVERB_OBJECT *pReverbData, EAS_INT nNumSamplesToAdd, EAS_PCM *pOutputBuffer, EAS_PCM *pInputBuffer)
{
    EAS_I32 i;
    EAS_I32 nDelayOut;
    EAS_U16 nBase;

    EAS_U32 nAddr;
    EAS_I32 nTemp1;
    EAS_I32 nTemp2;
    EAS_I32 nApIn;
    EAS_I32 nApOut;

    EAS_I32 j;
    EAS_I32 nEarlyOut;

    EAS_I32 tempValue;


    // get the base address
    nBase = pReverbData->m_nBaseIndex;

    for (i=0; i < nNumSamplesToAdd; i++)
    {
        // ********** Left Allpass - start
        // left input = (left dry/4) + right feedback from previous period
        /*lint -e{702} use shift for performance */
        nApIn = ((*pInputBuffer++)>>2) + pReverbData->m_nRevOutFbkR;
//      nApIn = *pInputBuffer++;    // 1xxx test and debug ap

        // fetch allpass delay line out
        //nAddr = CIRCULAR(nBase, psAp0->m_zApOut, REVERB_BUFFER_MASK);
        nAddr = CIRCULAR(nBase, pReverbData->m_sAp0.m_zApOut, REVERB_BUFFER_MASK);
        nDelayOut = pReverbData->m_nDelayLine[nAddr];

        // calculate allpass feedforward; subtract the feedforward result
        nTemp1 = MULT_EG1_EG1(nApIn, pReverbData->m_sAp0.m_nApGain);
        nApOut = SATURATE(nDelayOut - nTemp1);          // allpass output

        // calculate allpass feedback; add the feedback result
        nTemp1 = MULT_EG1_EG1(nApOut, pReverbData->m_sAp0.m_nApGain);
        nTemp1 = SATURATE(nApIn + nTemp1);

        // inject into allpass delay
        nAddr = CIRCULAR(nBase, pReverbData->m_sAp0.m_zApIn, REVERB_BUFFER_MASK);
        pReverbData->m_nDelayLine[nAddr] = (EAS_PCM) nTemp1;

        // inject allpass output into delay line
        nAddr = CIRCULAR(nBase, pReverbData->m_zD0In, REVERB_BUFFER_MASK);
        pReverbData->m_nDelayLine[nAddr] = (EAS_PCM) nApOut;

        // ********** Left Allpass - end

        // ********** Right Allpass - start
        // right input = (right dry/4) + left feedback from previous period
        /*lint -e{702} use shift for performance */
        nApIn = ((*pInputBuffer++)>>2) + pReverbData->m_nRevOutFbkL;
//      nApIn = *pInputBuffer++;    // 1xxx test and debug ap

        // fetch allpass delay line out
        nAddr = CIRCULAR(nBase, pReverbData->m_sAp1.m_zApOut, REVERB_BUFFER_MASK);
        nDelayOut = pReverbData->m_nDelayLine[nAddr];

        // calculate allpass feedforward; subtract the feedforward result
        nTemp1 = MULT_EG1_EG1(nApIn, pReverbData->m_sAp1.m_nApGain);
        nApOut = SATURATE(nDelayOut - nTemp1);          // allpass output

        // calculate allpass feedback; add the feedback result
        nTemp1 = MULT_EG1_EG1(nApOut, pReverbData->m_sAp1.m_nApGain);
        nTemp1 = SATURATE(nApIn + nTemp1);

        // inject into allpass delay
        nAddr = CIRCULAR(nBase, pReverbData->m_sAp1.m_zApIn, REVERB_BUFFER_MASK);
        pReverbData->m_nDelayLine[nAddr] = (EAS_PCM) nTemp1;

        // inject allpass output into delay line
        nAddr = CIRCULAR(nBase, pReverbData->m_zD1In, REVERB_BUFFER_MASK);
        pReverbData->m_nDelayLine[nAddr] = (EAS_PCM) nApOut;

        // ********** Right Allpass - end

        // ********** D0 output - start
        // fetch delay line self out
        nAddr = CIRCULAR(nBase, pReverbData->m_zD0Self, REVERB_BUFFER_MASK);
        nDelayOut = pReverbData->m_nDelayLine[nAddr];

        // calculate delay line self out
        nTemp1 = MULT_EG1_EG1(nDelayOut, pReverbData->m_nSin);

        // fetch delay line cross out
        nAddr = CIRCULAR(nBase, pReverbData->m_zD1Cross, REVERB_BUFFER_MASK);
        nDelayOut = pReverbData->m_nDelayLine[nAddr];

        // calculate delay line self out
        nTemp2 = MULT_EG1_EG1(nDelayOut, pReverbData->m_nCos);

        // calculate unfiltered delay out
        nDelayOut = SATURATE(nTemp1 + nTemp2);

        // calculate lowpass filter (mixer scale factor included in LPF feedforward)
        nTemp1 = MULT_EG1_EG1(nDelayOut, pReverbData->m_nLpfFwd);

        nTemp2 = MULT_EG1_EG1(pReverbData->m_zLpf0, pReverbData->m_nLpfFbk);

        // calculate filtered delay out and simultaneously update LPF state variable
        // filtered delay output is stored in m_zLpf0
        pReverbData->m_zLpf0 = (EAS_PCM) SATURATE(nTemp1 + nTemp2);

        // ********** D0 output - end

        // ********** D1 output - start
        // fetch delay line self out
        nAddr = CIRCULAR(nBase, pReverbData->m_zD1Self, REVERB_BUFFER_MASK);
        nDelayOut = pReverbData->m_nDelayLine[nAddr];

        // calculate delay line self out
        nTemp1 = MULT_EG1_EG1(nDelayOut, pReverbData->m_nSin);

        // fetch delay line cross out
        nAddr = CIRCULAR(nBase, pReverbData->m_zD0Cross, REVERB_BUFFER_MASK);
        nDelayOut = pReverbData->m_nDelayLine[nAddr];

        // calculate delay line self out
        nTemp2 = MULT_EG1_EG1(nDelayOut, pReverbData->m_nCos);

        // calculate unfiltered delay out
        nDelayOut = SATURATE(nTemp1 + nTemp2);

        // calculate lowpass filter (mixer scale factor included in LPF feedforward)
        nTemp1 = MULT_EG1_EG1(nDelayOut, pReverbData->m_nLpfFwd);

        nTemp2 = MULT_EG1_EG1(pReverbData->m_zLpf1, pReverbData->m_nLpfFbk);

        // calculate filtered delay out and simultaneously update LPF state variable
        // filtered delay output is stored in m_zLpf1
        pReverbData->m_zLpf1 = (EAS_PCM)SATURATE(nTemp1 + nTemp2);

        // ********** D1 output - end

        // ********** mixer and feedback - start
        // sum is fedback to right input (R + L)
        pReverbData->m_nRevOutFbkL =
            (EAS_PCM)SATURATE((EAS_I32)pReverbData->m_zLpf1 + (EAS_I32)pReverbData->m_zLpf0);

        // difference is feedback to left input (R - L)
        /*lint -e{685} lint complains that it can't saturate negative */
        pReverbData->m_nRevOutFbkR =
            (EAS_PCM)SATURATE((EAS_I32)pReverbData->m_zLpf1 - (EAS_I32)pReverbData->m_zLpf0);

        // ********** mixer and feedback - end

        // ********** start early reflection generator, left
        //psEarly = &(pReverbData->m_sEarlyL);

        nEarlyOut = 0;

        for (j=0; j < REVERB_MAX_NUM_REFLECTIONS; j++)
        {
            // fetch delay line out
            //nAddr = CIRCULAR(nBase, psEarly->m_zDelay[j], REVERB_BUFFER_MASK);
            nAddr = CIRCULAR(nBase, pReverbData->m_sEarlyL.m_zDelay[j], REVERB_BUFFER_MASK);

            nDelayOut = pReverbData->m_nDelayLine[nAddr];

            // calculate reflection
            //nTemp1 = MULT_EG1_EG1(nDelayOut, psEarly->m_nGain[j]);
            nTemp1 = MULT_EG1_EG1(nDelayOut, pReverbData->m_sEarlyL.m_nGain[j]);

            nEarlyOut = SATURATE(nEarlyOut + nTemp1);

        }   // end for (j=0; j < REVERB_MAX_NUM_REFLECTIONS; j++)

        // apply lowpass to early reflections
        //nTemp1 = MULT_EG1_EG1(nEarlyOut, psEarly->m_nLpfFwd);
        nTemp1 = MULT_EG1_EG1(nEarlyOut, pReverbData->m_sEarlyL.m_nLpfFwd);

        //nTemp2 = MULT_EG1_EG1(psEarly->m_zLpf, psEarly->m_nLpfFbk);
        nTemp2 = MULT_EG1_EG1(pReverbData->m_sEarlyL.m_zLpf, pReverbData->m_sEarlyL.m_nLpfFbk);


        // calculate filtered out and simultaneously update LPF state variable
        // filtered output is stored in m_zLpf1
        //psEarly->m_zLpf = SATURATE(nTemp1 + nTemp2);
        pReverbData->m_sEarlyL.m_zLpf = (EAS_PCM) SATURATE(nTemp1 + nTemp2);

        // combine filtered early and late reflections for output
        //*pOutputBuffer++ = inL;
        //tempValue = SATURATE(psEarly->m_zLpf + pReverbData->m_nRevOutFbkL);
        tempValue = SATURATE((EAS_I32)pReverbData->m_sEarlyL.m_zLpf + (EAS_I32)pReverbData->m_nRevOutFbkL);
        //scale reverb output by wet level
        /*lint -e{701} use shift for performance */
        tempValue = MULT_EG1_EG1(tempValue, (pReverbData->m_nWet<<1));
        //sum with output buffer
        tempValue += *pOutputBuffer;
        *pOutputBuffer++ = (EAS_PCM)SATURATE(tempValue);

        // ********** end early reflection generator, left

        // ********** start early reflection generator, right
        //psEarly = &(pReverbData->m_sEarlyR);

        nEarlyOut = 0;

        for (j=0; j < REVERB_MAX_NUM_REFLECTIONS; j++)
        {
            // fetch delay line out
            nAddr = CIRCULAR(nBase, pReverbData->m_sEarlyR.m_zDelay[j], REVERB_BUFFER_MASK);
            nDelayOut = pReverbData->m_nDelayLine[nAddr];

            // calculate reflection
            nTemp1 = MULT_EG1_EG1(nDelayOut, pReverbData->m_sEarlyR.m_nGain[j]);

            nEarlyOut = SATURATE(nEarlyOut + nTemp1);

        }   // end for (j=0; j < REVERB_MAX_NUM_REFLECTIONS; j++)

        // apply lowpass to early reflections
        nTemp1 = MULT_EG1_EG1(nEarlyOut, pReverbData->m_sEarlyR.m_nLpfFwd);

        nTemp2 = MULT_EG1_EG1(pReverbData->m_sEarlyR.m_zLpf, pReverbData->m_sEarlyR.m_nLpfFbk);

        // calculate filtered out and simultaneously update LPF state variable
        // filtered output is stored in m_zLpf1
        pReverbData->m_sEarlyR.m_zLpf = (EAS_PCM)SATURATE(nTemp1 + nTemp2);

        // combine filtered early and late reflections for output
        //*pOutputBuffer++ = inR;
        tempValue = SATURATE((EAS_I32)pReverbData->m_sEarlyR.m_zLpf + (EAS_I32)pReverbData->m_nRevOutFbkR);
        //scale reverb output by wet level
        /*lint -e{701} use shift for performance */
        tempValue = MULT_EG1_EG1(tempValue, (pReverbData->m_nWet << 1));
        //sum with output buffer
        tempValue = tempValue + *pOutputBuffer;
        *pOutputBuffer++ = (EAS_PCM)SATURATE(tempValue);

        // ********** end early reflection generator, right

        // decrement base addr for next sample period
        nBase--;

        pReverbData->m_nSin += pReverbData->m_nSinIncrement;
        pReverbData->m_nCos += pReverbData->m_nCosIncrement;

    }   // end for (i=0; i < nNumSamplesToAdd; i++)

    // store the most up to date version
    pReverbData->m_nBaseIndex = nBase;

    return EAS_SUCCESS;
}   /* end Reverb */



/*----------------------------------------------------------------------------
 * ReverbShutdown()
 *----------------------------------------------------------------------------
 * Purpose:
 * Initializes the Reverb effect.
 *
 * Inputs:
 * pInstData        - handle to instance data
 *
 * Outputs:
 *
 *
 * Side Effects:
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbShutdown (EAS_DATA_HANDLE pEASData, EAS_VOID_PTR pInstData)
{
    /* check Configuration Module for static memory allocation */
    if (!pEASData->staticMemoryModel)
        EAS_HWFree(pEASData->hwInstData, pInstData);
    return EAS_SUCCESS;
} /* end ReverbShutdown */

/*----------------------------------------------------------------------------
 * ReverbGetParam()
 *----------------------------------------------------------------------------
 * Purpose:
 * Get a Reverb parameter
 *
 * Inputs:
 * pInstData        - handle to instance data
 * param            - parameter index
 * *pValue          - pointer to variable to hold retrieved value
 *
 * Outputs:
 *
 *
 * Side Effects:
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbGetParam (EAS_VOID_PTR pInstData, EAS_I32 param, EAS_I32 *pValue)
{
    S_REVERB_OBJECT *p;

    p = (S_REVERB_OBJECT*) pInstData;

    switch (param)
    {
        case EAS_PARAM_REVERB_BYPASS:
            *pValue = (EAS_I32) p->m_bBypass;
            break;
        case EAS_PARAM_REVERB_PRESET:
            *pValue = (EAS_I8) p->m_nCurrentRoom;
            break;
        case EAS_PARAM_REVERB_WET:
            *pValue = p->m_nWet;
            break;
        case EAS_PARAM_REVERB_DRY:
            *pValue = p->m_nDry;
            break;
        default:
            return EAS_ERROR_INVALID_PARAMETER;
    }
    return EAS_SUCCESS;
} /* end ReverbGetParam */


/*----------------------------------------------------------------------------
 * ReverbSetParam()
 *----------------------------------------------------------------------------
 * Purpose:
 * Set a Reverb parameter
 *
 * Inputs:
 * pInstData        - handle to instance data
 * param            - parameter index
 * *pValue          - new paramter value
 *
 * Outputs:
 *
 *
 * Side Effects:
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbSetParam (EAS_VOID_PTR pInstData, EAS_I32 param, EAS_I32 value)
{
    S_REVERB_OBJECT *p;

    p = (S_REVERB_OBJECT*) pInstData;

    switch (param)
    {
        case EAS_PARAM_REVERB_BYPASS:
            p->m_bBypass = (EAS_BOOL) value;
            break;
        case EAS_PARAM_REVERB_PRESET:
            if(value!=EAS_PARAM_REVERB_LARGE_HALL && value!=EAS_PARAM_REVERB_HALL &&
                value!=EAS_PARAM_REVERB_CHAMBER && value!=EAS_PARAM_REVERB_ROOM)
                return EAS_ERROR_INVALID_PARAMETER;
            p->m_nNextRoom = (EAS_I16)value;
            break;
        case EAS_PARAM_REVERB_WET:
            if(value>EAS_REVERB_WET_MAX || value<EAS_REVERB_WET_MIN)
                return EAS_ERROR_INVALID_PARAMETER;
            p->m_nWet = (EAS_I16)value;
            break;
        case EAS_PARAM_REVERB_DRY:
            if(value>EAS_REVERB_DRY_MAX || value<EAS_REVERB_DRY_MIN)
                return EAS_ERROR_INVALID_PARAMETER;
            p->m_nDry = (EAS_I16)value;
            break;
        default:
            return EAS_ERROR_INVALID_PARAMETER;
    }
    return EAS_SUCCESS;
} /* end ReverbSetParam */


/*----------------------------------------------------------------------------
 * ReverbUpdateRoom
 *----------------------------------------------------------------------------
 * Purpose:
 * Update the room's preset parameters as required
 *
 * Inputs:
 *
 * Outputs:
 *
 *
 * Side Effects:
 * - reverb paramters (fbk, fwd, etc) will be changed
 * - m_nCurrentRoom := m_nNextRoom
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbUpdateRoom(S_REVERB_OBJECT *pReverbData)
{
    EAS_INT temp;

    S_REVERB_PRESET *pPreset = &pReverbData->m_sPreset.m_sPreset[pReverbData->m_nNextRoom];

    pReverbData->m_nLpfFwd = pPreset->m_nLpfFwd;
    pReverbData->m_nLpfFbk = pPreset->m_nLpfFbk;

    pReverbData->m_nEarly = pPreset->m_nEarly;
    pReverbData->m_nWet = pPreset->m_nWet;
    pReverbData->m_nDry = pPreset->m_nDry;


    pReverbData->m_nMaxExcursion = pPreset->m_nMaxExcursion;
    //stored as time based, convert to sample based
    temp = pPreset->m_nXfadeInterval;
    /*lint -e{702} shift for performance */
    temp = (temp * _OUTPUT_SAMPLE_RATE) >> 16;
    pReverbData->m_nXfadeInterval = (EAS_U16) temp;
    //gpsReverbObject->m_nXfadeInterval = pPreset->m_nXfadeInterval;
    pReverbData->m_sAp0.m_nApGain = pPreset->m_nAp0_ApGain;
    //stored as time based, convert to absolute sample value
    temp = pPreset->m_nAp0_ApOut;
    /*lint -e{702} shift for performance */
    temp = (temp * _OUTPUT_SAMPLE_RATE) >> 16;
    pReverbData->m_sAp0.m_zApOut = (EAS_U16) (pReverbData->m_sAp0.m_zApIn + temp);
    //gpsReverbObject->m_sAp0.m_zApOut = pPreset->m_nAp0_ApOut;
    pReverbData->m_sAp1.m_nApGain = pPreset->m_nAp1_ApGain;
    //stored as time based, convert to absolute sample value
    temp = pPreset->m_nAp1_ApOut;
    /*lint -e{702} shift for performance */
    temp = (temp * _OUTPUT_SAMPLE_RATE) >> 16;
    pReverbData->m_sAp1.m_zApOut = (EAS_U16) (pReverbData->m_sAp1.m_zApIn + temp);
    //gpsReverbObject->m_sAp1.m_zApOut = pPreset->m_nAp1_ApOut;

    pReverbData->m_nCurrentRoom = pReverbData->m_nNextRoom;

    return EAS_SUCCESS;

}   /* end ReverbUpdateRoom */


/*----------------------------------------------------------------------------
 * ReverbReadInPresets()
 *----------------------------------------------------------------------------
 * Purpose: sets global reverb preset bank to defaults
 *
 * Inputs:
 *
 * Outputs:
 *
 *----------------------------------------------------------------------------
*/
static EAS_RESULT ReverbReadInPresets(S_REVERB_OBJECT *pReverbData)
{

    int preset = 0;
    int defaultPreset = 0;

    //now init any remaining presets to defaults
    for (defaultPreset = preset; defaultPreset < REVERB_MAX_ROOM_TYPE; defaultPreset++)
    {
        S_REVERB_PRESET *pPreset = &pReverbData->m_sPreset.m_sPreset[defaultPreset];
        if (defaultPreset == 0 || defaultPreset > REVERB_MAX_ROOM_TYPE-1)
        {
            pPreset->m_nLpfFbk = 8307;
            pPreset->m_nLpfFwd = 14768;
            pPreset->m_nEarly = 0;
            pPreset->m_nWet = 27690;
            pPreset->m_nDry = 32767;
            pPreset->m_nEarlyL_LpfFbk = 3692;
            pPreset->m_nEarlyL_LpfFwd = 29075;
            pPreset->m_nEarlyL_Delay0 = 922;
            pPreset->m_nEarlyL_Gain0 = 22152;
            pPreset->m_nEarlyL_Delay1 = 1462;
            pPreset->m_nEarlyL_Gain1 = 17537;
            pPreset->m_nEarlyL_Delay2 = 0;
            pPreset->m_nEarlyL_Gain2 = 14768;
            pPreset->m_nEarlyL_Delay3 = 1221;
            pPreset->m_nEarlyL_Gain3 = 14307;
            pPreset->m_nEarlyL_Delay4 = 0;
            pPreset->m_nEarlyL_Gain4 = 13384;
            pPreset->m_nEarlyR_Delay0 = 502;
            pPreset->m_nEarlyR_Gain0 = 20306;
            pPreset->m_nEarlyR_Delay1 = 1762;
            pPreset->m_nEarlyR_Gain1 = 17537;
            pPreset->m_nEarlyR_Delay2 = 0;
            pPreset->m_nEarlyR_Gain2 = 14768;
            pPreset->m_nEarlyR_Delay3 = 0;
            pPreset->m_nEarlyR_Gain3 = 16153;
            pPreset->m_nEarlyR_Delay4 = 0;
            pPreset->m_nEarlyR_Gain4 = 13384;
            pPreset->m_nMaxExcursion = 127;
            pPreset->m_nXfadeInterval = 6388;
            pPreset->m_nAp0_ApGain = 15691;
            pPreset->m_nAp0_ApOut = 711;
            pPreset->m_nAp1_ApGain = 17999;
            pPreset->m_nAp1_ApOut = 1113;
            pPreset->m_rfu4 = 0;
            pPreset->m_rfu5 = 0;
            pPreset->m_rfu6 = 0;
            pPreset->m_rfu7 = 0;
            pPreset->m_rfu8 = 0;
            pPreset->m_rfu9 = 0;
            pPreset->m_rfu10 = 0;
        }
        else if (defaultPreset == 1)
        {
            pPreset->m_nLpfFbk = 6461;
            pPreset->m_nLpfFwd = 14307;
            pPreset->m_nEarly = 0;
            pPreset->m_nWet = 27690;
            pPreset->m_nDry = 32767;
            pPreset->m_nEarlyL_LpfFbk = 3692;
            pPreset->m_nEarlyL_LpfFwd = 29075;
            pPreset->m_nEarlyL_Delay0 = 922;
            pPreset->m_nEarlyL_Gain0 = 22152;
            pPreset->m_nEarlyL_Delay1 = 1462;
            pPreset->m_nEarlyL_Gain1 = 17537;
            pPreset->m_nEarlyL_Delay2 = 0;
            pPreset->m_nEarlyL_Gain2 = 14768;
            pPreset->m_nEarlyL_Delay3 = 1221;
            pPreset->m_nEarlyL_Gain3 = 14307;
            pPreset->m_nEarlyL_Delay4 = 0;
            pPreset->m_nEarlyL_Gain4 = 13384;
            pPreset->m_nEarlyR_Delay0 = 502;
            pPreset->m_nEarlyR_Gain0 = 20306;
            pPreset->m_nEarlyR_Delay1 = 1762;
            pPreset->m_nEarlyR_Gain1 = 17537;
            pPreset->m_nEarlyR_Delay2 = 0;
            pPreset->m_nEarlyR_Gain2 = 14768;
            pPreset->m_nEarlyR_Delay3 = 0;
            pPreset->m_nEarlyR_Gain3 = 16153;
            pPreset->m_nEarlyR_Delay4 = 0;
            pPreset->m_nEarlyR_Gain4 = 13384;
            pPreset->m_nMaxExcursion = 127;
            pPreset->m_nXfadeInterval = 6391;
            pPreset->m_nAp0_ApGain = 15230;
            pPreset->m_nAp0_ApOut = 708;
            pPreset->m_nAp1_ApGain = 9692;
            pPreset->m_nAp1_ApOut = 1113;
            pPreset->m_rfu4 = 0;
            pPreset->m_rfu5 = 0;
            pPreset->m_rfu6 = 0;
            pPreset->m_rfu7 = 0;
            pPreset->m_rfu8 = 0;
            pPreset->m_rfu9 = 0;
            pPreset->m_rfu10 = 0;
        }
        else if (defaultPreset == 2)
        {
            pPreset->m_nLpfFbk = 5077;
            pPreset->m_nLpfFwd = 12922;
            pPreset->m_nEarly = 0;
            pPreset->m_nWet = 24460;
            pPreset->m_nDry = 32767;
            pPreset->m_nEarlyL_LpfFbk = 3692;
            pPreset->m_nEarlyL_LpfFwd = 29075;
            pPreset->m_nEarlyL_Delay0 = 922;
            pPreset->m_nEarlyL_Gain0 = 22152;
            pPreset->m_nEarlyL_Delay1 = 1462;
            pPreset->m_nEarlyL_Gain1 = 17537;
            pPreset->m_nEarlyL_Delay2 = 0;
            pPreset->m_nEarlyL_Gain2 = 14768;
            pPreset->m_nEarlyL_Delay3 = 1221;
            pPreset->m_nEarlyL_Gain3 = 14307;
            pPreset->m_nEarlyL_Delay4 = 0;
            pPreset->m_nEarlyL_Gain4 = 13384;
            pPreset->m_nEarlyR_Delay0 = 502;
            pPreset->m_nEarlyR_Gain0 = 20306;
            pPreset->m_nEarlyR_Delay1 = 1762;
            pPreset->m_nEarlyR_Gain1 = 17537;
            pPreset->m_nEarlyR_Delay2 = 0;
            pPreset->m_nEarlyR_Gain2 = 14768;
            pPreset->m_nEarlyR_Delay3 = 0;
            pPreset->m_nEarlyR_Gain3 = 16153;
            pPreset->m_nEarlyR_Delay4 = 0;
            pPreset->m_nEarlyR_Gain4 = 13384;
            pPreset->m_nMaxExcursion = 127;
            pPreset->m_nXfadeInterval = 6449;
            pPreset->m_nAp0_ApGain = 15691;
            pPreset->m_nAp0_ApOut = 774;
            pPreset->m_nAp1_ApGain = 15691;
            pPreset->m_nAp1_ApOut = 1113;
            pPreset->m_rfu4 = 0;
            pPreset->m_rfu5 = 0;
            pPreset->m_rfu6 = 0;
            pPreset->m_rfu7 = 0;
            pPreset->m_rfu8 = 0;
            pPreset->m_rfu9 = 0;
            pPreset->m_rfu10 = 0;
        }
        else if (defaultPreset == 3)
        {
            pPreset->m_nLpfFbk = 5077;
            pPreset->m_nLpfFwd = 11076;
            pPreset->m_nEarly = 0;
            pPreset->m_nWet = 23075;
            pPreset->m_nDry = 32767;
            pPreset->m_nEarlyL_LpfFbk = 3692;
            pPreset->m_nEarlyL_LpfFwd = 29075;
            pPreset->m_nEarlyL_Delay0 = 922;
            pPreset->m_nEarlyL_Gain0 = 22152;
            pPreset->m_nEarlyL_Delay1 = 1462;
            pPreset->m_nEarlyL_Gain1 = 17537;
            pPreset->m_nEarlyL_Delay2 = 0;
            pPreset->m_nEarlyL_Gain2 = 14768;
            pPreset->m_nEarlyL_Delay3 = 1221;
            pPreset->m_nEarlyL_Gain3 = 14307;
            pPreset->m_nEarlyL_Delay4 = 0;
            pPreset->m_nEarlyL_Gain4 = 13384;
            pPreset->m_nEarlyR_Delay0 = 502;
            pPreset->m_nEarlyR_Gain0 = 20306;
            pPreset->m_nEarlyR_Delay1 = 1762;
            pPreset->m_nEarlyR_Gain1 = 17537;
            pPreset->m_nEarlyR_Delay2 = 0;
            pPreset->m_nEarlyR_Gain2 = 14768;
            pPreset->m_nEarlyR_Delay3 = 0;
            pPreset->m_nEarlyR_Gain3 = 16153;
            pPreset->m_nEarlyR_Delay4 = 0;
            pPreset->m_nEarlyR_Gain4 = 13384;
            pPreset->m_nMaxExcursion = 127;
            pPreset->m_nXfadeInterval = 6470;   //6483;
            pPreset->m_nAp0_ApGain = 14768;
            pPreset->m_nAp0_ApOut = 792;
            pPreset->m_nAp1_ApGain = 15783;
            pPreset->m_nAp1_ApOut = 1113;
            pPreset->m_rfu4 = 0;
            pPreset->m_rfu5 = 0;
            pPreset->m_rfu6 = 0;
            pPreset->m_rfu7 = 0;
            pPreset->m_rfu8 = 0;
            pPreset->m_rfu9 = 0;
            pPreset->m_rfu10 = 0;

        }
    }

    return EAS_SUCCESS;
}