Mercurial > hg > beaglert
diff core/PRU.cpp @ 81:92145ba7aabf
Updated PRU code to allow pinModeFrame() and [untested] digitalWriteFrame() and analogWriteFrame() to be called from setup(). Updated basic_blink example accordingly.
author | andrewm |
---|---|
date | Fri, 17 Jul 2015 22:16:53 +0100 |
parents | 3c3a1357657d |
children | 3068421c0737 c0bf6157f67e |
line wrap: on
line diff
--- a/core/PRU.cpp Fri Jul 17 21:39:51 2015 +0100 +++ b/core/PRU.cpp Fri Jul 17 22:16:53 2015 +0100 @@ -387,6 +387,46 @@ } } + // Allocate audio buffers + context->audioIn = (float *)malloc(2 * context->audioFrames * sizeof(float)); + context->audioOut = (float *)malloc(2 * context->audioFrames * sizeof(float)); + if(context->audioIn == 0 || context->audioOut == 0) { + rt_printf("Error: couldn't allocate audio buffers\n"); + return 1; + } + + // Allocate analog buffers + if(analog_enabled) { + context->analogIn = (float *)malloc(context->analogChannels * context->analogFrames * sizeof(float)); + context->analogOut = (float *)malloc(context->analogChannels * context->analogFrames * sizeof(float)); + last_analog_out_frame = (float *)malloc(context->analogChannels * sizeof(float)); + + if(context->analogIn == 0 || context->analogOut == 0 || last_analog_out_frame == 0) { + rt_printf("Error: couldn't allocate analog buffers\n"); + return 1; + } + + memset(last_analog_out_frame, 0, context->analogChannels * sizeof(float)); + } + + // Allocate digital buffers + digital_buffer0 = pru_buffer_digital; + digital_buffer1 = pru_buffer_digital + MEM_DIGITAL_BUFFER1_OFFSET / sizeof(uint32_t); + if(digital_enabled) { + last_digital_buffer = (uint32_t *)malloc(context->digitalFrames * sizeof(uint32_t)); //temp buffer to hold previous states + if(last_digital_buffer == 0) { + rt_printf("Error: couldn't allocate digital buffers\n"); + return 1; + } + + for(unsigned int n = 0; n < context->digitalFrames; n++){ + // Initialize lastDigitalFrames to all inputs + last_digital_buffer[n] = 0x0000ffff; + } + } + + context->digital = digital_buffer0; + return 0; } @@ -427,45 +467,24 @@ RTIME sleepTime = PRU_SAMPLE_INTERVAL_NS * (context->analogChannels / 2) * context->analogFrames / 4; #endif - float *lastAnalogOutFrame; - uint32_t *digitalBuffer0, *digitalBuffer1, *lastDigitalBuffer; uint32_t pru_audio_offset, pru_spi_offset; - // Allocate audio buffers - context->audioIn = (float *)malloc(2 * context->audioFrames * sizeof(float)); - context->audioOut = (float *)malloc(2 * context->audioFrames * sizeof(float)); - if(context->audioIn == 0 || context->audioOut == 0) { - rt_printf("Error: couldn't allocate audio buffers\n"); - return; + // Before starting, look at the last state of the analog and digital outputs which might + // have been changed by the user during the setup() function. This lets us start with pin + // directions and output values at something other than defaults. + + if(analog_enabled) { + if(context->flags & BEAGLERT_FLAG_ANALOG_OUTPUTS_PERSIST) { + // Remember the content of the last_analog_out_frame + for(unsigned int ch = 0; ch < context->analogChannels; ch++){ + last_analog_out_frame[ch] = context->analogOut[context->analogChannels * (context->analogFrames - 1) + ch]; + } + } } - // Allocate analog buffers - if(analog_enabled) { - context->analogIn = (float *)malloc(context->analogChannels * context->analogFrames * sizeof(float)); - context->analogOut = (float *)malloc(context->analogChannels * context->analogFrames * sizeof(float)); - lastAnalogOutFrame = (float *)malloc(context->analogChannels * sizeof(float)); - - if(context->analogIn == 0 || context->analogOut == 0 || lastAnalogOutFrame == 0) { - rt_printf("Error: couldn't allocate analog buffers\n"); - return; - } - - memset(lastAnalogOutFrame, 0, context->analogChannels * sizeof(float)); - } - - // Allocate digital buffers - digitalBuffer0 = pru_buffer_digital; - digitalBuffer1 = pru_buffer_digital + MEM_DIGITAL_BUFFER1_OFFSET / sizeof(uint32_t); if(digital_enabled) { - lastDigitalBuffer = (uint32_t *)malloc(context->digitalFrames * sizeof(uint32_t)); //temp buffer to hold previous states - if(lastDigitalBuffer == 0) { - rt_printf("Error: couldn't allocate digital buffers\n"); - return; - } - for(unsigned int n = 0; n < context->digitalFrames; n++){ - // Initialize lastDigitalFrames to all inputs - lastDigitalBuffer[n] = 0x0000ffff; + last_digital_buffer[n] = context->digital[n]; } } @@ -518,14 +537,14 @@ pru_audio_offset = 0; pru_spi_offset = 0; if(digital_enabled) - context->digital = digitalBuffer0; + context->digital = digital_buffer0; } else { // PRU is on buffer 0. We read and write to buffer 1 pru_audio_offset = context->audioFrames * 2; pru_spi_offset = context->analogFrames * context->analogChannels; if(digital_enabled) - context->digital = digitalBuffer1; + context->digital = digital_buffer1; } // FIXME: some sort of margin is needed here to prevent the audio @@ -553,7 +572,7 @@ // Initialize the output buffer with the values that were in the last frame of the previous output for(unsigned int ch = 0; ch < context->analogChannels; ch++){ for(unsigned int n = 0; n < context->analogFrames; n++){ - context->analogOut[n * context->analogChannels + ch] = lastAnalogOutFrame[ch]; + context->analogOut[n * context->analogChannels + ch] = last_analog_out_frame[ch]; } } } @@ -570,12 +589,12 @@ // - pins previously set as inputs will carry the newly read input value for(unsigned int n = 0; n < context->digitalFrames; n++){ - uint16_t inputs = lastDigitalBuffer[n] & 0xffff; // half-word, has 1 for inputs and 0 for outputs + uint16_t inputs = last_digital_buffer[n] & 0xffff; // half-word, has 1 for inputs and 0 for outputs uint16_t outputs = ~inputs; // half-word has 1 for outputs and 0 for inputs; - context->digital[n] = (lastDigitalBuffer[context->digitalFrames - 1] & (outputs << 16)) | // keep output values set in the last frame of the previous buffer + context->digital[n] = (last_digital_buffer[context->digitalFrames - 1] & (outputs << 16)) | // keep output values set in the last frame of the previous buffer (context->digital[n] & (inputs << 16)) | // inputs from current context->digital[n]; - (lastDigitalBuffer[n] & (inputs)); // keep pin configuration from previous context->digital[n] + (last_digital_buffer[n] & (inputs)); // keep pin configuration from previous context->digital[n] // context->digital[n]=digitalBufferTemp[n]; //ignores inputs } } @@ -587,9 +606,9 @@ if(analog_enabled) { if(context->flags & BEAGLERT_FLAG_ANALOG_OUTPUTS_PERSIST) { - // Remember the content of the lastAnalogOutFrame + // Remember the content of the last_analog_out_frame for(unsigned int ch = 0; ch < context->analogChannels; ch++){ - lastAnalogOutFrame[ch] = context->analogOut[context->analogChannels * (context->analogFrames - 1) + ch]; + last_analog_out_frame[ch] = context->analogOut[context->analogChannels * (context->analogFrames - 1) + ch]; } } @@ -604,7 +623,7 @@ if(digital_enabled) { // keep track of past digital values for(unsigned int n = 0; n < context->digitalFrames; n++){ - lastDigitalBuffer[n] = context->digital[n]; + last_digital_buffer[n] = context->digital[n]; } } @@ -653,11 +672,11 @@ if(analog_enabled) { free(context->analogIn); free(context->analogOut); - free(lastAnalogOutFrame); + free(last_analog_out_frame); } if(digital_enabled) { - free(lastDigitalBuffer); + free(last_digital_buffer); } context->audioIn = context->audioOut = 0;