andrewm@0: /* andrewm@0: * render.cpp andrewm@0: * andrewm@0: * Created on: Oct 24, 2014 andrewm@0: * Author: parallels andrewm@0: */ andrewm@0: andrewm@0: andrewm@0: #include "../../include/render.h" andrewm@0: #include "../../include/Utilities.h" andrewm@0: #include andrewm@0: #include andrewm@0: andrewm@0: // Set range for analog outputs designed for driving LEDs andrewm@0: const float kMinimumAmplitude = (1.5 / 5.0) * MATRIX_MAX; andrewm@0: const float kAmplitudeRange = MATRIX_MAX - kMinimumAmplitude; andrewm@0: andrewm@0: float gFrequency; andrewm@0: float gPhase; andrewm@0: float gInverseSampleRate; andrewm@0: andrewm@0: // initialise_render() is called once before the audio rendering starts. andrewm@0: // Use it to perform any initialisation and allocation which is dependent andrewm@0: // on the period size or sample rate. andrewm@0: // andrewm@0: // userData holds an opaque pointer to a data structure that was passed andrewm@0: // in from the call to initAudio(). andrewm@0: // andrewm@0: // Return true on success; returning false halts the program. andrewm@0: andrewm@12: bool initialise_render(int numMatrixChannels, int numAudioChannels, andrewm@12: int numMatrixFramesPerPeriod, andrewm@12: int numAudioFramesPerPeriod, andrewm@12: float matrixSampleRate, float audioSampleRate, andrewm@12: void *userData) andrewm@0: { andrewm@0: // Retrieve a parameter passed in from the initAudio() call andrewm@0: gFrequency = *(float *)userData; andrewm@0: andrewm@12: if(numMatrixFramesPerPeriod == 0) { andrewm@12: rt_printf("Error: this example needs the matrix enabled\n"); andrewm@0: return false; andrewm@0: } andrewm@0: andrewm@12: gInverseSampleRate = 1.0 / matrixSampleRate; andrewm@0: gPhase = 0.0; andrewm@0: andrewm@0: return true; andrewm@0: } andrewm@0: andrewm@0: // render() is called regularly at the highest priority by the audio engine. andrewm@0: // Input and output are given from the audio hardware and the other andrewm@0: // ADCs and DACs (if available). If only audio is available, numMatrixFrames andrewm@0: // will be 0. andrewm@0: andrewm@0: void render(int numMatrixFrames, int numAudioFrames, float *audioIn, float *audioOut, andrewm@0: uint16_t *matrixIn, uint16_t *matrixOut) andrewm@0: { andrewm@0: for(int n = 0; n < numMatrixFrames; n++) { andrewm@0: // Set LED to different phase for each matrix channel andrewm@0: float relativePhase = 0.0; andrewm@13: for(int channel = 0; channel < gNumMatrixChannels; channel++) { andrewm@0: float out = kMinimumAmplitude + kAmplitudeRange * 0.5f * (1.0f + sinf(gPhase + relativePhase)); andrewm@0: if(out > MATRIX_MAX) andrewm@0: out = MATRIX_MAX; andrewm@0: andrewm@13: analogWrite(channel, n, out); andrewm@0: andrewm@0: // Advance by pi/4 (1/8 of a full rotation) for each channel andrewm@0: relativePhase += M_PI * 0.25; andrewm@0: } andrewm@0: andrewm@0: gPhase += 2.0 * M_PI * gFrequency * gInverseSampleRate; andrewm@0: if(gPhase > 2.0 * M_PI) andrewm@0: gPhase -= 2.0 * M_PI; andrewm@0: } andrewm@0: } andrewm@0: andrewm@0: // cleanup_render() is called once at the end, after the audio has stopped. andrewm@0: // Release any resources that were allocated in initialise_render(). andrewm@0: andrewm@0: void cleanup_render() andrewm@0: { andrewm@0: andrewm@0: }