Mercurial > hg > beaglert
view examples/04-Audio/FFT-phase-vocoder/render.cpp @ 539:b486344aa796 prerelease
merge
author | Giulio Moro <giuliomoro@yahoo.it> |
---|---|
date | Fri, 24 Jun 2016 01:43:53 +0100 |
parents | 1cec96845a23 |
children | cdabbaf3a252 |
line wrap: on
line source
/* ____ _____ _ _ | __ )| ____| | / \ | _ \| _| | | / _ \ | |_) | |___| |___ / ___ \ |____/|_____|_____/_/ \_\ The platform for ultra-low latency audio and sensor processing http://bela.io A project of the Augmented Instruments Laboratory within the Centre for Digital Music at Queen Mary University of London. http://www.eecs.qmul.ac.uk/~andrewm (c) 2016 Augmented Instruments Laboratory: Andrew McPherson, Astrid Bin, Liam Donovan, Christian Heinrichs, Robert Jack, Giulio Moro, Laurel Pardue, Victor Zappi. All rights reserved. The Bela software is distributed under the GNU Lesser General Public License (LGPL 3.0), available here: https://www.gnu.org/licenses/lgpl-3.0.txt */ #include <Bela.h> #include <rtdk.h> #include <ne10/NE10.h> // NEON FFT library #include <cmath> #include "SampleData.h" #include <Midi.h> #define BUFFER_SIZE 16384 // TODO: your buffer and counter go here! float gInputBuffer[BUFFER_SIZE]; int gInputBufferPointer = 0; float gOutputBuffer[BUFFER_SIZE]; int gOutputBufferWritePointer = 0; int gOutputBufferReadPointer = 0; int gSampleCount = 0; float *gWindowBuffer; // ----------------------------------------------- // These variables used internally in the example: int gFFTSize = 2048; int gHopSize = 512; int gPeriod = 512; float gFFTScaleFactor = 0; // FFT vars ne10_fft_cpx_float32_t* timeDomainIn; ne10_fft_cpx_float32_t* timeDomainOut; ne10_fft_cpx_float32_t* frequencyDomain; ne10_fft_cfg_float32_t cfg; // Sample info SampleData gSampleData; // User defined structure to get complex data from main int gReadPtr = 0; // Position of last read sample from file // Auxiliary task for calculating FFT AuxiliaryTask gFFTTask; int gFFTInputBufferPointer = 0; int gFFTOutputBufferPointer = 0; void process_fft_background(); int gEffect = 0; // change this here or with midi CC enum{ kBypass, kRobot, kWhisper, }; float gDryWet = 1; // mix between the unprocessed and processed sound float gPlaybackLive = 0.5f; // mix between the file playback and the live audio input float gGain = 1; // overall gain float *gInputAudio = NULL; Midi midi; void midiCallback(MidiChannelMessage message, void* arg){ if(message.getType() == kmmNoteOn){ if(message.getDataByte(1) > 0){ int note = message.getDataByte(0); float frequency = powf(2, (note-69)/12.f)*440; gPeriod = (int)(44100 / frequency + 0.5); printf("\nnote: %d, frequency: %f, hop: %d\n", note, frequency, gPeriod); } } bool shouldPrint = false; if(message.getType() == kmmControlChange){ float data = message.getDataByte(1) / 127.0f; switch (message.getDataByte(0)){ case 2 : gEffect = (int)(data * 2 + 0.5); // CC2 selects an effect between 0,1,2 break; case 3 : gPlaybackLive = data; break; case 4 : gDryWet = data; break; case 5: gGain = data*10; break; default: shouldPrint = true; } } if(shouldPrint){ message.prettyPrint(); } } // userData holds an opaque pointer to a data structure that was passed // in from the call to initAudio(). // // Return true on success; returning false halts the program. bool setup(BelaContext* context, void* userData) { midi.readFrom(0); midi.setParserCallback(midiCallback); // Retrieve a parameter passed in from the initAudio() call gSampleData = *(SampleData *)userData; gFFTScaleFactor = 1.0f / (float)gFFTSize; gOutputBufferWritePointer += gHopSize; timeDomainIn = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t)); timeDomainOut = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t)); frequencyDomain = (ne10_fft_cpx_float32_t*) NE10_MALLOC (gFFTSize * sizeof (ne10_fft_cpx_float32_t)); cfg = ne10_fft_alloc_c2c_float32_neon (gFFTSize); memset(timeDomainOut, 0, gFFTSize * sizeof (ne10_fft_cpx_float32_t)); memset(gOutputBuffer, 0, BUFFER_SIZE * sizeof(float)); // Allocate buffer to mirror and modify the input gInputAudio = (float *)malloc(context->audioFrames * context->audioChannels * sizeof(float)); if(gInputAudio == 0) return false; // Allocate the window buffer based on the FFT size gWindowBuffer = (float *)malloc(gFFTSize * sizeof(float)); if(gWindowBuffer == 0) return false; // Calculate a Hann window for(int n = 0; n < gFFTSize; n++) { gWindowBuffer[n] = 0.5f * (1.0f - cosf(2.0 * M_PI * n / (float)(gFFTSize - 1))); } // Initialise auxiliary tasks if((gFFTTask = Bela_createAuxiliaryTask(&process_fft_background, 90, "fft-calculation")) == 0) return false; rt_printf("You are listening to an FFT phase-vocoder with overlap-and-add.\n" "Use Midi Control Change to control:\n" "CC 2: effect type (bypass/robotization/whisperization)\n" "CC 3: mix between recorded sample and live audio input\n" "CC 4: mix between the unprocessed and processed sound\n" "CC 5: gain\n" ); return true; } // This function handles the FFT processing in this example once the buffer has // been assembled. void process_fft(float *inBuffer, int inWritePointer, float *outBuffer, int outWritePointer) { // Copy buffer into FFT input int pointer = (inWritePointer - gFFTSize + BUFFER_SIZE) % BUFFER_SIZE; for(int n = 0; n < gFFTSize; n++) { timeDomainIn[n].r = (ne10_float32_t) inBuffer[pointer] * gWindowBuffer[n]; timeDomainIn[n].i = 0; pointer++; if(pointer >= BUFFER_SIZE) pointer = 0; } // Run the FFT ne10_fft_c2c_1d_float32_neon (frequencyDomain, timeDomainIn, cfg, 0); switch (gEffect){ case kRobot : // Robotise the output for(int n = 0; n < gFFTSize; n++) { float amplitude = sqrtf(frequencyDomain[n].r * frequencyDomain[n].r + frequencyDomain[n].i * frequencyDomain[n].i); frequencyDomain[n].r = amplitude; frequencyDomain[n].i = 0; } break; case kWhisper : for(int n = 0; n < gFFTSize; n++) { float amplitude = sqrtf(frequencyDomain[n].r * frequencyDomain[n].r + frequencyDomain[n].i * frequencyDomain[n].i); float phase = rand()/(float)RAND_MAX * 2 * M_PI; frequencyDomain[n].r = cosf(phase) * amplitude; frequencyDomain[n].i = sinf(phase) * amplitude; } break; case kBypass: //bypass break; } // Run the inverse FFT ne10_fft_c2c_1d_float32_neon (timeDomainOut, frequencyDomain, cfg, 1); // Overlap-and-add timeDomainOut into the output buffer pointer = outWritePointer; for(int n = 0; n < gFFTSize; n++) { outBuffer[pointer] += (timeDomainOut[n].r) * gFFTScaleFactor; if(isnan(outBuffer[pointer])) rt_printf("outBuffer OLA\n"); pointer++; if(pointer >= BUFFER_SIZE) pointer = 0; } } // Function to process the FFT in a thread at lower priority void process_fft_background() { process_fft(gInputBuffer, gFFTInputBufferPointer, gOutputBuffer, gFFTOutputBufferPointer); } // render() is called regularly at the highest priority by the audio engine. // Input and output are given from the audio hardware and the other // ADCs and DACs (if available). If only audio is available, numMatrixFrames // will be 0. void render(BelaContext* context, void* userData) { float* audioOut = context->audioOut; int numAudioFrames = context->audioFrames; int numAudioChannels = context->audioChannels; // ------ this code internal to the demo; leave as is ---------------- // Prep the "input" to be the sound file played in a loop for(int n = 0; n < numAudioFrames; n++) { if(gReadPtr < gSampleData.sampleLen) gInputAudio[2*n] = gInputAudio[2*n+1] = gSampleData.samples[gReadPtr]*(1-gPlaybackLive) + gPlaybackLive*0.5f*(audioRead(context,n,0)+audioRead(context,n,1)); else gInputAudio[2*n] = gInputAudio[2*n+1] = 0; if(++gReadPtr >= gSampleData.sampleLen) gReadPtr = 0; } // ------------------------------------------------------------------- for(int n = 0; n < numAudioFrames; n++) { gInputBuffer[gInputBufferPointer] = ((gInputAudio[n*numAudioChannels] + gInputAudio[n*numAudioChannels+1]) * 0.5); // Copy output buffer to output for(int channel = 0; channel < numAudioChannels; channel++){ audioOut[n * numAudioChannels + channel] = gOutputBuffer[gOutputBufferReadPointer] * gGain * gDryWet + (1 - gDryWet) * gInputAudio[n * numAudioChannels + channel]; } // Clear the output sample in the buffer so it is ready for the next overlap-add gOutputBuffer[gOutputBufferReadPointer] = 0; gOutputBufferReadPointer++; if(gOutputBufferReadPointer >= BUFFER_SIZE) gOutputBufferReadPointer = 0; gOutputBufferWritePointer++; if(gOutputBufferWritePointer >= BUFFER_SIZE) gOutputBufferWritePointer = 0; gInputBufferPointer++; if(gInputBufferPointer >= BUFFER_SIZE) gInputBufferPointer = 0; gSampleCount++; if(gSampleCount >= gHopSize) { //process_fft(gInputBuffer, gInputBufferPointer, gOutputBuffer, gOutputBufferPointer); gFFTInputBufferPointer = gInputBufferPointer; gFFTOutputBufferPointer = gOutputBufferWritePointer; Bela_scheduleAuxiliaryTask(gFFTTask); gSampleCount = 0; } } gHopSize = gPeriod; } // cleanup_render() is called once at the end, after the audio has stopped. // Release any resources that were allocated in initialise_render(). void cleanup(BelaContext* context, void* userData) { NE10_FREE(timeDomainIn); NE10_FREE(timeDomainOut); NE10_FREE(frequencyDomain); NE10_FREE(cfg); free(gInputAudio); free(gWindowBuffer); } /** \example FFT-phase-vocoder/render.cpp Phase Vocoder ---------------------- This sketch shows an implementation of a phase vocoder and builds on the previous FFT example. Again it uses the NE10 library, included at the top of the file. Read the documentation on the NE10 library [here](http://projectne10.github.io/Ne10/doc/annotated.html). */