Mercurial > hg > beaglert
diff examples/basic_libpd/render.cpp @ 341:7af9c5be3434 prerelease
libpd: using smaller blocksizes for processing, so that events can be scheduled more accurately
author | Giulio Moro <giuliomoro@yahoo.it> |
---|---|
date | Mon, 06 Jun 2016 03:31:22 +0100 |
parents | 69d86429a1cf |
children | 860c42b3830e |
line wrap: on
line diff
--- a/examples/basic_libpd/render.cpp Mon Jun 06 02:37:30 2016 +0100 +++ b/examples/basic_libpd/render.cpp Mon Jun 06 03:31:22 2016 +0100 @@ -118,11 +118,13 @@ void render(BelaContext *context, void *userData) { int num; + // the safest thread-safe option to handle MIDI input is to process the MIDI buffer + // from the audio thread. #ifdef PARSE_MIDI while((num = midi.getParser()->numAvailableMessages()) > 0){ static MidiChannelMessage message; message = midi.getParser()->getNextChannelMessage(); -// message.prettyPrint(); // use this to print beautified message (channel, data bytes) + //message.prettyPrint(); // use this to print beautified message (channel, data bytes) switch(message.getType()){ case kmmNoteOn: { @@ -193,8 +195,8 @@ } #endif /* PARSE_MIDI */ - static unsigned int inW = 0; - static unsigned int outR = 0; + unsigned int inW = 0; + unsigned int outR = 0; /* * NOTE: if you are only using audio (or only analogs) and you are using interleaved buffers * and the blocksize of Bela is the same as gLibPdBlockSize, then you probably @@ -202,72 +204,64 @@ * memory operations. */ static unsigned int analogChannelsInUse = min(context->analogChannels, gChannelsInUse - context->audioChannels); + static unsigned int numberOfPdBlocksToProcess = gBufLength / gLibpdBlockSize; // rt_printf("channelsInUse: %d, analogChannels in Use: %d\n", gChannelsInUse, analogChannelsInUse); - for(unsigned int n = 0; n < context->audioFrames; ++n){ //pd buffers are interleaved - for(unsigned int ch = 0; ch < context->audioChannels; ++ch){ //first two channels are audio - gInBuf[inW++] = audioRead(context, n, ch); - } - // then analogs - // this loop resamples by ZOH, as needed, using m - if(context->analogChannels == 8 ){ //hold the value for two frames - for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ - gInBuf[inW++] = analogRead(context, n/2, analogCh); // n/2 wil be the same for n and n+1 when n is even + for(unsigned int j = 0; j < numberOfPdBlocksToProcess; ++j){ + int oldInW = inW; + for(unsigned int n = 0; n < gLibpdBlockSize; ++n){ //pd buffers are interleaved + for(unsigned int ch = 0; ch < context->audioChannels; ++ch){ //first two channels are audio + gInBuf[inW++] = audioRead(context, n, ch); } - } else if(context->analogChannels == 4){ //write every frame - for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ - gInBuf[inW++] = analogRead(context, n, analogCh); - } - } else if(context->analogChannels == 2){ //drop every other frame - for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ - gInBuf[inW++] = analogRead(context, n*2, analogCh); + // then analogs + // this loop resamples by ZOH, as needed, using m + if(context->analogChannels == 8 ){ //hold the value for two frames + for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ + gInBuf[inW++] = analogRead(context, n/2, analogCh); // n/2 wil be the same for n and n+1 when n is even + } + } else if(context->analogChannels == 4){ //write every frame + for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ + gInBuf[inW++] = analogRead(context, n, analogCh); + } + } else if(context->analogChannels == 2){ //drop every other frame + for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ + gInBuf[inW++] = analogRead(context, n*2, analogCh); + } } } - if(inW == gBufLength * gChannelsInUse){ - inW = 0; - } - } - // rt_printf("inW %d\n", inW); - if(inW == 0){ //if the buffer is full, process it - static int numberOfPdBlocksToProcess = gBufLength/gLibpdBlockSize; // TODO: see if we can rewrite libpd_process_float so that it takes a buffer // of interleaved channels of arbitrary length rather than a series of // stacked buffers of size gLibPdBlockSize as it currently does. - libpd_process_float(numberOfPdBlocksToProcess, gInBuf, gOutBuf); - outR = 0; // reset the read pointer. NOTE: hopefully this is needed only the first time - } - - for(unsigned int n = 0; n < context->audioFrames; n++){ //pd buffers are interleaved - for(unsigned int ch = 0; ch < context->audioChannels; ++ch){ - audioWrite(context, n, ch, gOutBuf[outR++]); - } - //and analogs - if(context->analogChannels == 8){ - for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ - float analogOut = gOutBuf[outR++]; - if((n&1) == 0){//write every two frames - analogWrite(context, n/2, analogCh, analogOut); - } else { - // discard this sample + // This would allow to use the buffers as they are rather than having to copy them + libpd_process_float(1, &gInBuf[oldInW], &gOutBuf[oldInW]); + for(unsigned int n = 0; n < gLibpdBlockSize; ++n){ //pd buffers are interleaved + unsigned int outAudioFrame = n + j * gLibpdBlockSize; + for(unsigned int ch = 0; ch < context->audioChannels; ++ch){ + audioWrite(context, outAudioFrame, ch, gOutBuf[outR++]); + } + //and analogs + if(context->analogChannels == 8){ + for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ + float analogOut = gOutBuf[outR++]; + if((n&1) == 0){//write every two frames + analogWrite(context, outAudioFrame/2, analogCh, analogOut); + } else { + // discard this sample + } + } + } else if(context->analogChannels == 4){ //write every frame + for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ + float analogOut = gOutBuf[outR++]; + analogWrite(context, outAudioFrame, analogCh, analogOut); + } + } else if(context->analogChannels == 2){ //write twice every frame + for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ + float analogOut = gOutBuf[outR++]; + analogWrite(context, 2*outAudioFrame, analogCh, analogOut); + analogWrite(context, 2*outAudioFrame + 1, analogCh, analogOut); } } - } else if(context->analogChannels == 4){ //write every frame - for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ - float analogOut = gOutBuf[outR++]; - analogWrite(context, n, analogCh, analogOut); - } - } else if(context->analogChannels == 2){ //write twice every frame - for(unsigned int analogCh = 0; analogCh < analogChannelsInUse; ++analogCh){ - float analogOut = gOutBuf[outR++]; - analogWrite(context, 2*n, analogCh, analogOut); - analogWrite(context, 2*n + 1, analogCh, analogOut); - } - } - if(outR == gBufLength * gChannelsInUse){ - outR = 0; } } - // rt_printf("outR %d, analogChannelsInUse %d, channelsInUse %d\n", - // outR , analogChannelsInUse, gChannelsInUse); Bela_scheduleAuxiliaryTask(libpdProcessMidiQueueTask); Bela_scheduleAuxiliaryTask(libpdProcessMessageQueueTask); }