# HG changeset patch # User Giulio Moro # Date 1440166894 -3600 # Node ID c692827083e13150776693701916b13e4ad0cd77 # Parent 26ad97b8aa9eb85b0aa86a0c183830bb606e4328 Enabled multi channel audio receive diff -r 26ad97b8aa9e -r c692827083e1 .cproject --- a/.cproject Fri Aug 21 14:37:19 2015 +0100 +++ b/.cproject Fri Aug 21 15:21:34 2015 +0100 @@ -18,7 +18,7 @@ - + diff -r 26ad97b8aa9e -r c692827083e1 core/NetworkSend.cpp --- a/core/NetworkSend.cpp Fri Aug 21 14:37:19 2015 +0100 +++ b/core/NetworkSend.cpp Fri Aug 21 15:21:34 2015 +0100 @@ -133,6 +133,15 @@ } } +void Scope::setPort(int port){ + for(int n=0; n ReceiveAudioThread::objAddrs(0); +bool ReceiveAudioThread::threadRunning; +bool ReceiveAudioThread::threadIsExiting; +int ReceiveAudioThread::sleepTime; + +void receiveData(){ + ReceiveAudioThread::run(); +} +void ReceiveAudioThread::staticConstructor(){ + if(staticConstructed==true) + return; + staticConstructed=true; + threadIsExiting=false; + receiveDataTask=BeagleRT_createAuxiliaryTask(receiveData, 90, "receiveDataTask"); //TODO: allow different priorities +} +#endif void ReceiveAudioThread::dealloc(){ free(buffer); @@ -45,8 +65,6 @@ //read header+payload //JUCE int numBytes=socket.read(buffer+writePointer, bytesToRead,1); int numBytes=socket.read(buffer+writePointer, bytesToRead); - //TODO: do something with the data in the header (e.g.: check that timestamp is sequential, - // check the channel number) //TODO: (if using variable-length payload) validate the actual numBytes read against the size declared in the header if(numBytes<0){ printf("error numBytes1\n"); @@ -56,13 +74,16 @@ // printf("received 0 bytes\n"); return 0; } - if(numBytes != bytesToRead){ //this is equivalent to (numByteslastValidPointer){ @@ -86,33 +107,33 @@ socket(0), listening(false), bufferReady(false), - threadIsExiting(false), buffer(NULL), stackBuffer(NULL), bufferLength(0), lastValidPointer(0), - sleepTime(2000), waitForSocketTime(100), threadPriority(95) {}; ReceiveAudioThread::~ReceiveAudioThread(){ //JUCE stopThread(1000); while(threadRunning){ - sleep(sleepTime*2); //wait for thread to stop + usleep(sleepTime*2); //wait for thread to stop std::cout<< "Waiting for receiveAudioTask to stop" << std::endl; } //TODO: check if thread stopped, otherwise kill it before dealloc dealloc(); } -ReceiveAudioThread *gRAT; -void receiveData(){ - gRAT->run(); -} -void ReceiveAudioThread::init(int aSamplesPerBlock){ +void ReceiveAudioThread::init(int aPort, int aSamplesPerBlock, int aChannel){ dealloc(); +#ifdef JUCE +#else + staticConstructor(); + objAddrs.push_back(this);//TODO: this line should be in the constructor +#endif + bindToPort(aPort); + channel=aChannel; // fd=fopen("output.m","w"); //DEBUG // fprintf(fd,"var=["); //DEBUG - gRAT=this; headerLength=2; payloadLength=300; //TODO: make sure that payloadLength and headerLength are the same as the client is sending. bufferLength=std::max(headerLength+(payloadLength*4), headerLength+(aSamplesPerBlock*4)); //there are many considerations that can be done here ... @@ -128,12 +149,16 @@ writePointer=-1; readPointer=0; //TODO: this *4 is sortof a security margin sleepTime=payloadLength/(float)44100 /4.0; //set sleepTime so that you do not check too often or too infrequently - receiveDataTask=BeagleRT_createAuxiliaryTask( receiveData, threadPriority, "receiveDataTask"); //JUCE startThread(threadPriority); } void ReceiveAudioThread::bindToPort(int aPort){ listening=socket.bindToPort(aPort); +#ifdef JUCE +#else + if(listening==false) + printf("Could not bind to port %d\n",aPort); +#endif } bool ReceiveAudioThread::isListening(){ return listening; @@ -196,8 +221,15 @@ // fprintf(fd2, "buf=["); //DEBUG threadRunning=true; while(!threadShouldExit()){ //TODO: check that the socket buffer is empty before starting +#ifdef JUCE readUdpToBuffer(); // read into the oldBuffer usleep(sleepTime); +#else + for(unsigned int n=0; nreadUdpToBuffer(); + } + usleep(sleepTime); //TODO: use rt_task_sleep instead +#endif } threadRunning=false; // fprintf(fd,"];readPointer,writePointer,lastValidPointer,destination]=deal(var(:,1), var(:,2), var(:,3), var(:,4));"); //DEBUG diff -r 26ad97b8aa9e -r c692827083e1 include/NetworkSend.h --- a/include/NetworkSend.h Fri Aug 21 14:37:19 2015 +0100 +++ b/include/NetworkSend.h Fri Aug 21 15:21:34 2015 +0100 @@ -60,7 +60,8 @@ void setup(); void setup(float sampleRate, int aPort, const char* aServer); void sendData(); - void setPort(); + void setPort(int port); + void setPort(int channel, int port); int getNumChannels(); }; #endif /* SCOPE_H */ diff -r 26ad97b8aa9e -r c692827083e1 include/ReceiveAudioThread.h --- a/include/ReceiveAudioThread.h Fri Aug 21 14:37:19 2015 +0100 +++ b/include/ReceiveAudioThread.h Fri Aug 21 15:21:34 2015 +0100 @@ -14,42 +14,61 @@ //JUCE DatagramSocket socket; bool listening; bool bufferReady; - bool threadIsExiting; +#ifdef JUCE bool threadRunning; +#else + static bool threadRunning; + static bool threadIsExiting; +#endif float *buffer; float *stackBuffer; int bufferLength; float readPointer; int writePointer; int lastValidPointer; +#ifdef JUCE int sleepTime; +#else + static int sleepTime; +#endif int waitForSocketTime; int payloadLength; //size of the payload of each datagram int headerLength; //size of the header of each datagram int bytesToRead; int threadPriority; + int channel; void dealloc(); void wrapWritePointer(); void pushPayload(int startIndex); void popPayload(int startIndex); int readUdpToBuffer(); - bool threadShouldExit(); +#ifdef JUCE +#else + static bool threadShouldExit(); + static bool staticConstructed; + static void staticConstructor(); static AuxiliaryTask receiveDataTask; //TODO: allow different AuxiliaryTasks for different priorities (e.g.: audio vs scope) -// static std::vector objAddrs; + static std::vector objAddrs; +#endif public: ReceiveAudioThread(); ~ReceiveAudioThread(); - void init(int aSamplesPerBlock); + void init(int port, int aSamplesPerBlock, int channel); void setup(); void bindToPort(int aPort); bool isListening(); float* getCurrentBuffer(int length); int getSamplesSrc(float *destination, int length, float samplingRateRatio); bool isBufferReady(); +#ifdef JUCE // if we are in Juce, then we run a separate thread for each receiver + // (as each of them are typically receiving on a mono or stereo track) void run(); - void startThread(); - void stopThread(); -// static int getNumInstances(); -// static void receiveAllData(); +#else + void static run(); //while in BeagleRT we have a single thread that receives for all the instances. + //TODO: make run() private in BeagleRT + static void startThread(); + static void stopThread(); + static int getNumInstances(); +#endif // JUCE }; #endif // RECEIVEAUDIOTHREAD_H_INCLUDED diff -r 26ad97b8aa9e -r c692827083e1 projects/scope/render.cpp --- a/projects/scope/render.cpp Fri Aug 21 14:37:19 2015 +0100 +++ b/projects/scope/render.cpp Fri Aug 21 15:21:34 2015 +0100 @@ -7,7 +7,7 @@ float gFrequency1, gFrequency2; float gInverseSampleRate; -Scope scope(6); //create a scope object with 6 channels +Scope scope(2); //create a scope object with 2 channels NetworkSend networkSend; // initialise_render() is called once before the audio rendering starts. @@ -18,12 +18,16 @@ // in from the call to initAudio(). // // Return true on success; returning false halts the program. -ReceiveAudioThread receiveAudio; +ReceiveAudioThread receiveAudio0; +ReceiveAudioThread receiveAudio1; bool setup(BeagleRTContext *context, void *userData) { - receiveAudio.bindToPort(9999); - receiveAudio.init(context->audioFrames); + receiveAudio0.init(9999, context->audioFrames, 0); + receiveAudio1.init(10000, context->audioFrames, 1); + scope.setup(); //call this once in setup to initialise the scope + scope.setPort(0, 9999); + scope.setPort(1, 10000); // networkSend.setup(context->audioSampleRate, 0, 9999, "192.168.7.1"); gInverseSampleRate = 1.0/context->audioSampleRate; @@ -46,12 +50,12 @@ { static int count=0; if(count==0) - receiveAudio.startThread(); + ReceiveAudioThread::startThread(); for(unsigned int n = 0; n < context->audioFrames; n++) { float chn0 = sinf(gPhase1); - // float chn1 = sinf(gPhase2); + float chn1 = sinf(gPhase2); // float chn2 = context->audioIn[n*2 + 0]; // float chn3 = context->audioIn[n*2 + 1]; @@ -59,7 +63,7 @@ // float chn4 = context->analogIn[(int)n/2*8 + 0]; // float chn5 = context->analogIn[(int)n/2*8 + 1]; scope.log(0, chn0); - // scope.log(1, chn1); + scope.log(1, chn1); // scope.log(2, chn2); // scope.log(3, chn3); // scope.log(4, chn4); @@ -80,18 +84,18 @@ gPhase2 -= 2.0 * M_PI; } - static float buffer[32]; //this should be context->audioFrames + static float buffer[2][32]; //this should be context->audioFrames if(count==0){ - for(int n=0; n<32; n++) - buffer[n]=0; + memset(buffer,2*32*sizeof(float),0); } if(count>0){ - int readPointer=receiveAudio.getSamplesSrc(buffer, context->audioFrames, 1); - for(int n=0; naudioFrames; n++){ - context->audioOut[n*2]=buffer[n]; - context->audioOut[n*2+1]=buffer[n]; - } - } + int readPointer0=receiveAudio0.getSamplesSrc(buffer[0], context->audioFrames, 1); + int readPointer1=receiveAudio1.getSamplesSrc(buffer[1], context->audioFrames, 1); + for(unsigned int n=0; naudioFrames; n++){ + context->audioOut[n*2]=buffer[0][n]; + context->audioOut[n*2+1]=buffer[1][n]; + } + } count++; }