@*********************************************************** @ Function: WT_Interpolate @ Processor: ARM-E @ Description: the main synthesis function when fetching @ wavetable samples. @ C-callable. @ @ Usage: @ void WT_Interpolate( @ S_WT_VOICE *pWTVoice, @ S_WT_FRAME *pWTFrame); @ @ Copyright Sonic Network Inc. 2004 @**************************************************************** @ Revision Control: @ $Revision: 496 $ @ $Date: 2006-12-11 14:33:26 -0800 (Mon, 11 Dec 2006) $ @**************************************************************** @ @ where: @ S_WT_VOICE *pWTVoice @ PASSED IN: r0 @ @ S_WT_FRAME *pWTFrame; @ PASSED IN: r1 @**************************************************************** #include "ARM_synth_constants_gnu.inc" .arm .text .global WT_Interpolate @ Register usage @ -------------- pWTVoice .req r0 pWTFrame .req r1 numSamples .req r2 phaseIncrement .req r3 pOutputBuffer .req r4 tmp0 .req r1 @reuse register tmp1 .req r5 tmp2 .req r6 pLoopEnd .req r7 pLoopStart .req r8 pPhaseAccum .req r9 phaseFrac .req r10 phaseFracMask .req r11 @SaveRegs RLIST {r4-r11,lr} @RestoreRegs RLIST {r4-r11,pc} .func WT_Interpolate WT_Interpolate: STMFD sp!,{r4-r11,lr} @ @ Fetch parameters from structures @---------------------------------------------------------------- LDR pOutputBuffer, [pWTFrame, #m_pAudioBuffer] LDR numSamples, [pWTFrame, #m_numSamples] LDR phaseIncrement, [pWTFrame, #m_phaseIncrement] LDR pPhaseAccum, [pWTVoice, #m_pPhaseAccum] LDR phaseFrac, [pWTVoice, #m_phaseFrac] LDR phaseFracMask,=PHASE_FRAC_MASK LDR pLoopStart, [pWTVoice, #m_pLoopStart] LDR pLoopEnd, [pWTVoice, #m_pLoopEnd] ADD pLoopEnd, pLoopEnd, #1 @ need loop end to equal last sample + 1 InterpolationLoop: SUBS tmp0, pPhaseAccum, pLoopEnd @ check for loop end ADDGE pPhaseAccum, pLoopStart, tmp0 @ loop back to start .ifdef SAMPLES_8_BIT LDRSB tmp0, [pPhaseAccum] @ tmp0 = x0 LDRSB tmp1, [pPhaseAccum, #1] @ tmp1 = x1 .else LDRSH tmp0, [pPhaseAccum] @ tmp0 = x0 LDRSH tmp1, [pPhaseAccum, #2] @ tmp1 = x1 .endif ADD tmp2, phaseIncrement, phaseFrac @ increment pointer here to avoid pipeline stall SUB tmp1, tmp1, tmp0 @ tmp1 = x1 - x0 SMULBB tmp1, phaseFrac, tmp1 @ tmp1 = phaseFrac * tmp2 @ This section performs a gain adjustment of -12dB for 16-bit samples @ or +36dB for 8-bit samples. For a high quality synthesizer, the output @ can be set to full scale, however if the filter is used, it can overflow @ with certain coefficients and signal sources. In this case, either a @ saturation operation should take in the filter before scaling back to @ 16 bits or the signal path should be increased to 18 bits or more. .ifdef SAMPLES_8_BIT MOV tmp0, tmp0, LSL #6 @ boost 8-bit signal by 36dB .else MOV tmp0, tmp0, ASR #2 @ reduce 16-bit signal by 12dB .endif ADD tmp1, tmp0, tmp1, ASR #(NUM_EG1_FRAC_BITS-6) @ tmp1 = tmp0 + (tmp1 >> (15-6)) @ = x0 + f * (x1 - x0) == interpolated result STRH tmp1, [pOutputBuffer], #NEXT_OUTPUT_PCM @ *pOutputBuffer++ = interpolated result @ carry overflow from fraction to integer portion ADD pPhaseAccum, pPhaseAccum, tmp2, LSR #(NUM_PHASE_FRAC_BITS - NEXT_INPUT_PCM_SHIFT) AND phaseFrac, tmp2, phaseFracMask @ nphaseFrac = frac part SUBS numSamples, numSamples, #1 BGT InterpolationLoop @ update and store phase STR pPhaseAccum, [pWTVoice, #m_pPhaseAccum] STR phaseFrac, [pWTVoice, #m_phaseFrac] LDMFD sp!,{r4-r11,lr} BX lr .endfunc .end