giuliomoro@181: /* giuliomoro@181: * render.cpp giuliomoro@181: * giuliomoro@181: * Created on: Oct 24, 2014 giuliomoro@181: * Author: parallels giuliomoro@181: */ giuliomoro@181: giuliomoro@181: giuliomoro@181: #include giuliomoro@181: #include giuliomoro@181: #include giuliomoro@181: #include giuliomoro@181: #include giuliomoro@181: #include giuliomoro@181: giuliomoro@181: // setup() is called once before the audio rendering starts. giuliomoro@181: // Use it to perform any initialisation and allocation which is dependent giuliomoro@181: // on the period size or sample rate. giuliomoro@181: // giuliomoro@181: // userData holds an opaque pointer to a data structure that was passed giuliomoro@181: // in from the call to initAudio(). giuliomoro@181: // giuliomoro@181: // Return true on success; returning false halts the program. giuliomoro@181: Midi midi; giuliomoro@181: bool setup(BeagleRTContext *context, void *userData) giuliomoro@181: { giuliomoro@181: midi.readFrom(0); giuliomoro@181: if(context->analogFrames == 0) { giuliomoro@181: rt_printf("Error: this example needs the matrix enabled\n"); giuliomoro@181: return false; giuliomoro@181: } giuliomoro@181: return true; giuliomoro@181: } giuliomoro@181: giuliomoro@181: // render() is called regularly at the highest priority by the audio engine. giuliomoro@181: // Input and output are given from the audio hardware and the other giuliomoro@181: // ADCs and DACs (if available). If only audio is available, numMatrixFrames giuliomoro@181: // will be 0. giuliomoro@181: giuliomoro@181: static midi_byte_t noteOnStatus =0x90; //on channel 1 giuliomoro@181: giuliomoro@181: enum {kVelocity, kNoteOn, kNoteNumber}; giuliomoro@181: void render(BeagleRTContext *context, void *userData) giuliomoro@181: { giuliomoro@181: static float f0; giuliomoro@181: static float phaseIncrement = 0; giuliomoro@181: int message; giuliomoro@181: static bool noteOn = 0; giuliomoro@181: static int velocity = 0; giuliomoro@181: static int noteNumber = 0; giuliomoro@181: static int waitingFor = kNoteOn; giuliomoro@181: while ((message = midi.getInput()) >= 0){ giuliomoro@181: switch(waitingFor){ giuliomoro@181: case kNoteOn: giuliomoro@181: if(message == noteOnStatus){ giuliomoro@181: noteOn = true; giuliomoro@181: waitingFor = kNoteNumber; giuliomoro@181: } giuliomoro@181: break; giuliomoro@181: case kNoteNumber: giuliomoro@181: if((message & (1<<8)) == 0){ giuliomoro@181: noteNumber = message; giuliomoro@181: waitingFor = kVelocity; giuliomoro@181: f0 = powf(2, (noteNumber-69)/12.0f) * 440; giuliomoro@181: phaseIncrement = 2 * M_PI * f0 / context->audioSampleRate; giuliomoro@181: } giuliomoro@181: break; giuliomoro@181: case kVelocity: giuliomoro@181: if((message & (1<<8)) == 0){ giuliomoro@181: velocity = message; giuliomoro@181: waitingFor = kNoteOn; giuliomoro@181: if(velocity == 0) giuliomoro@181: noteOn = false; giuliomoro@181: rt_printf("NoteOn: %d, NoteNumber: %d, velocity: %d\n", noteOn, noteNumber, velocity); giuliomoro@181: } giuliomoro@181: break; giuliomoro@181: } giuliomoro@181: } giuliomoro@181: for(unsigned int n = 0; n < context->audioFrames; n++){ giuliomoro@181: if(noteOn == 1){ giuliomoro@181: static float phase = 0; giuliomoro@181: phase += phaseIncrement; giuliomoro@181: if(phase > 2 * M_PI) giuliomoro@181: phase -= 2 * M_PI; giuliomoro@181: float value = sinf(phase) * velocity/128.0f; giuliomoro@181: audioWriteFrame(context, n, 0, value); giuliomoro@181: audioWriteFrame(context, n, 1, value); giuliomoro@181: } else { giuliomoro@181: audioWriteFrame(context, n, 0, 0); giuliomoro@181: audioWriteFrame(context, n, 1, 0); giuliomoro@181: } giuliomoro@181: } giuliomoro@181: } giuliomoro@181: giuliomoro@181: // cleanup() is called once at the end, after the audio has stopped. giuliomoro@181: // Release any resources that were allocated in setup(). giuliomoro@181: giuliomoro@181: void cleanup(BeagleRTContext *context, void *userData) giuliomoro@181: { giuliomoro@181: giuliomoro@181: }