mirror of
				https://github.com/cjcliffe/CubicSDR.git
				synced 2025-10-26 10:30:24 -04:00 
			
		
		
		
	SDR->Demod->Audio chain all using std::thread
2% CPU usage drop on 3.2G i7 desktop
This commit is contained in:
		
							parent
							
								
									26168a2713
								
							
						
					
					
						commit
						58708a720e
					
				| @ -91,8 +91,6 @@ SET (cubicsdr_sources | |||||||
| 	src/AppFrame.cpp | 	src/AppFrame.cpp | ||||||
| 	src/sdr/SDRThread.cpp | 	src/sdr/SDRThread.cpp | ||||||
| 	src/demod/DemodulatorThread.cpp | 	src/demod/DemodulatorThread.cpp | ||||||
| 	src/demod/DemodulatorThreadQueue.cpp |  | ||||||
| 	src/demod/DemodulatorThreadTask.cpp |  | ||||||
| 	src/demod/DemodulatorMgr.cpp | 	src/demod/DemodulatorMgr.cpp | ||||||
| 	src/audio/AudioThread.cpp | 	src/audio/AudioThread.cpp | ||||||
| 	src/util/Gradient.cpp | 	src/util/Gradient.cpp | ||||||
| @ -112,8 +110,6 @@ SET (cubicsdr_headers | |||||||
| 	src/AppFrame.h | 	src/AppFrame.h | ||||||
| 	src/sdr/SDRThread.h | 	src/sdr/SDRThread.h | ||||||
| 	src/demod/DemodulatorThread.h | 	src/demod/DemodulatorThread.h | ||||||
| 	src/demod/DemodulatorThreadQueue.h |  | ||||||
| 	src/demod/DemodulatorThreadTask.h |  | ||||||
| 	src/demod/DemodulatorMgr.h | 	src/demod/DemodulatorMgr.h | ||||||
| 	src/audio/AudioThread.h | 	src/audio/AudioThread.h | ||||||
| 	src/util/Gradient.h | 	src/util/Gradient.h | ||||||
|  | |||||||
| @ -65,11 +65,14 @@ AppFrame::AppFrame() : | |||||||
| 
 | 
 | ||||||
|     demodulatorTest = demodMgr.newThread(this); |     demodulatorTest = demodMgr.newThread(this); | ||||||
|     demodulatorTest->params.audioInputQueue = audioInputQueue; |     demodulatorTest->params.audioInputQueue = audioInputQueue; | ||||||
|     demodulatorTest->run(); |     demodulatorTest->init(); | ||||||
|  | 
 | ||||||
|  |     audioVisualQueue = new DemodulatorThreadOutputQueue(); | ||||||
|  |     demodulatorTest->setVisualOutputQueue(audioVisualQueue); | ||||||
| 
 | 
 | ||||||
|     threadCmdQueueSDR = new SDRThreadCommandQueue; |     threadCmdQueueSDR = new SDRThreadCommandQueue; | ||||||
|     sdrThread = new SDRThread(threadCmdQueueSDR); |     sdrThread = new SDRThread(threadCmdQueueSDR); | ||||||
|     sdrThread->bindDemodulator(*demodulatorTest); |     sdrThread->bindDemodulator(demodulatorTest); | ||||||
| 
 | 
 | ||||||
|     iqVisualQueue = new SDRThreadIQDataQueue; |     iqVisualQueue = new SDRThreadIQDataQueue; | ||||||
|     sdrThread->setIQVisualQueue(iqVisualQueue); |     sdrThread->setIQVisualQueue(iqVisualQueue); | ||||||
| @ -101,39 +104,14 @@ void AppFrame::OnNewWindow(wxCommandEvent& WXUNUSED(event)) { | |||||||
| 
 | 
 | ||||||
| void AppFrame::OnThread(wxCommandEvent& event) { | void AppFrame::OnThread(wxCommandEvent& event) { | ||||||
|     SDRThreadIQData *iqData; |     SDRThreadIQData *iqData; | ||||||
|     DemodulatorThreadAudioData *demodAudioData; |  | ||||||
| 
 | 
 | ||||||
|     std::vector<signed char> *new_uc_buffer; |     std::vector<signed char> *new_uc_buffer; | ||||||
|     std::vector<float> *new_float_buffer; |     std::vector<float> *new_float_buffer; | ||||||
|     std::string asdf("beep"); |     std::string asdf("beep"); | ||||||
| 
 | 
 | ||||||
|     switch (event.GetId()) { |  | ||||||
| 
 |  | ||||||
|     // SDR IQ -> Demodulator
 |     // SDR IQ -> Demodulator
 | ||||||
| 
 | 
 | ||||||
|     case DemodulatorThreadTask::DEMOD_THREAD_AUDIO_DATA: |     event.Skip(); | ||||||
|         demodAudioData = (DemodulatorThreadAudioData *) event.GetClientData(); |  | ||||||
|         new_float_buffer = &(demodAudioData->data); |  | ||||||
|         if (new_float_buffer != NULL && new_float_buffer->size()) { |  | ||||||
| 
 |  | ||||||
|             if (scopeCanvas->waveform_points.size() != new_float_buffer->size() * 2) { |  | ||||||
|                 scopeCanvas->waveform_points.resize(new_float_buffer->size() * 2); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             for (int i = 0, iMax = new_float_buffer->size(); i < iMax; i++) { |  | ||||||
|                 scopeCanvas->waveform_points[i * 2 + 1] = (*new_float_buffer)[i] * 0.5f; |  | ||||||
|                 scopeCanvas->waveform_points[i * 2] = ((double) i / (double) iMax); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         } else { |  | ||||||
|             std::cout << "Incoming Demodulator data empty?" << std::endl; |  | ||||||
|         } |  | ||||||
|         delete demodAudioData; |  | ||||||
| 
 |  | ||||||
|         break; |  | ||||||
|     default: |  | ||||||
|         event.Skip(); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AppFrame::OnIdle(wxIdleEvent& event) { | void AppFrame::OnIdle(wxIdleEvent& event) { | ||||||
| @ -152,6 +130,26 @@ void AppFrame::OnIdle(wxIdleEvent& event) { | |||||||
|         work_done = true; |         work_done = true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     if (!audioVisualQueue->empty()) { | ||||||
|  |         AudioThreadInput demodAudioData; | ||||||
|  |         audioVisualQueue->pop(demodAudioData); | ||||||
|  |         if (demodAudioData.data.size()) { | ||||||
|  | 
 | ||||||
|  |             if (scopeCanvas->waveform_points.size() != demodAudioData.data.size() * 2) { | ||||||
|  |                 scopeCanvas->waveform_points.resize(demodAudioData.data.size() * 2); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             for (int i = 0, iMax = demodAudioData.data.size(); i < iMax; i++) { | ||||||
|  |                 scopeCanvas->waveform_points[i * 2 + 1] = demodAudioData.data[i] * 0.5f; | ||||||
|  |                 scopeCanvas->waveform_points[i * 2] = ((double) i / (double) iMax); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  |             std::cout << "Incoming Demodulator data empty?" << std::endl; | ||||||
|  |         } | ||||||
|  |         work_done = true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     if (!work_done) { |     if (!work_done) { | ||||||
|         event.Skip(); |         event.Skip(); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -45,6 +45,7 @@ private: | |||||||
|     SDRThread *sdrThread; |     SDRThread *sdrThread; | ||||||
|     SDRThreadCommandQueue* threadCmdQueueSDR; |     SDRThreadCommandQueue* threadCmdQueueSDR; | ||||||
|     SDRThreadIQDataQueue* iqVisualQueue; |     SDRThreadIQDataQueue* iqVisualQueue; | ||||||
|  |     DemodulatorThreadOutputQueue* audioVisualQueue; | ||||||
| 
 | 
 | ||||||
|     std::thread *t1; |     std::thread *t1; | ||||||
|     std::thread *t_SDR; |     std::thread *t_SDR; | ||||||
|  | |||||||
| @ -4,44 +4,36 @@ | |||||||
| #include <map> | #include <map> | ||||||
| 
 | 
 | ||||||
| #include "DemodulatorThread.h" | #include "DemodulatorThread.h" | ||||||
| #include "DemodulatorThreadQueue.h" |  | ||||||
| #include "DemodulatorThreadTask.h" |  | ||||||
| 
 | 
 | ||||||
| class DemodulatorInstance { | class DemodulatorInstance { | ||||||
| public: | public: | ||||||
|     DemodulatorThread *t_Demod; |     DemodulatorThread *demodulatorThread; | ||||||
|     DemodulatorThreadQueue* threadQueueDemod; |     std::thread *t_Demod; | ||||||
|     DemodulatorThreadParameters params; |  | ||||||
|     wxEvtHandler* parent; |  | ||||||
| 
 | 
 | ||||||
|     DemodulatorInstance(wxEvtHandler* pParent) : |     DemodulatorThreadInputQueue* threadQueueDemod; | ||||||
|             t_Demod(NULL), threadQueueDemod(NULL), parent(pParent) { |     DemodulatorThreadParameters params; | ||||||
|  | 
 | ||||||
|  |     DemodulatorInstance() : | ||||||
|  |             t_Demod(NULL), threadQueueDemod(NULL), demodulatorThread(NULL) { | ||||||
|  |     } | ||||||
|  |     void setVisualOutputQueue(DemodulatorThreadOutputQueue *tQueue) { | ||||||
|  |         demodulatorThread->setVisualOutputQueue(tQueue); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void init() { |     void init() { | ||||||
|         threadQueueDemod = new DemodulatorThreadQueue(parent); |         if (threadQueueDemod) { | ||||||
|         t_Demod = new DemodulatorThread(threadQueueDemod, ¶ms); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void addTask(const DemodulatorThreadTask& task, const DemodulatorThreadQueue::DEMOD_PRIORITY& priority) { |  | ||||||
|         threadQueueDemod->addTask(task, priority); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     bool run() { |  | ||||||
|         init(); |  | ||||||
|         if (t_Demod->Run() != wxTHREAD_NO_ERROR) { |  | ||||||
|             wxLogError |  | ||||||
|             ("Can't create the Demodulator thread!"); |  | ||||||
|             delete t_Demod; |  | ||||||
|             delete threadQueueDemod; |             delete threadQueueDemod; | ||||||
|             t_Demod = NULL; |  | ||||||
|             threadQueueDemod = NULL; |  | ||||||
|             return false; |  | ||||||
|         } |         } | ||||||
|         t_Demod->SetPriority(80); |         if (demodulatorThread) { | ||||||
|  |             delete demodulatorThread; | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         return true; |         threadQueueDemod = new DemodulatorThreadInputQueue; | ||||||
|  |         demodulatorThread = new DemodulatorThread(threadQueueDemod, ¶ms); | ||||||
|  | 
 | ||||||
|  |         t_Demod = new std::thread(&DemodulatorThread::threadMain, demodulatorThread); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class DemodulatorMgr { | class DemodulatorMgr { | ||||||
| @ -59,7 +51,7 @@ public: | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     DemodulatorInstance *newThread(wxEvtHandler* pParent) { |     DemodulatorInstance *newThread(wxEvtHandler* pParent) { | ||||||
|         DemodulatorInstance *newDemod = new DemodulatorInstance(pParent); |         DemodulatorInstance *newDemod = new DemodulatorInstance; | ||||||
|         demods.push_back(newDemod); |         demods.push_back(newDemod); | ||||||
|         return newDemod; |         return newDemod; | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -2,9 +2,8 @@ | |||||||
| #include "CubicSDRDefs.h" | #include "CubicSDRDefs.h" | ||||||
| #include <vector> | #include <vector> | ||||||
| 
 | 
 | ||||||
| 
 | DemodulatorThread::DemodulatorThread(DemodulatorThreadInputQueue* pQueue, DemodulatorThreadParameters *params_in) : | ||||||
| DemodulatorThread::DemodulatorThread(DemodulatorThreadQueue* pQueue, DemodulatorThreadParameters *params_in, int id) : |         m_pQueue(pQueue), visOutQueue(NULL) { | ||||||
|         wxThread(wxTHREAD_DETACHED), m_pQueue(pQueue), m_ID(id) { |  | ||||||
| 
 | 
 | ||||||
|     DemodulatorThreadParameters defaultParams; |     DemodulatorThreadParameters defaultParams; | ||||||
|     if (!params_in) { |     if (!params_in) { | ||||||
| @ -30,7 +29,7 @@ DemodulatorThread::DemodulatorThread(DemodulatorThreadQueue* pQueue, Demodulator | |||||||
|     fir_filter = firfilt_crcf_create(h, h_len); |     fir_filter = firfilt_crcf_create(h, h_len); | ||||||
| 
 | 
 | ||||||
|     h_len = estimate_req_filter_len(ft, As); |     h_len = estimate_req_filter_len(ft, As); | ||||||
|     liquid_firdes_kaiser(h_len, (float)params.filterFrequency / (float) params.demodResampleRate, As, mu, h); |     liquid_firdes_kaiser(h_len, (float) params.filterFrequency / (float) params.demodResampleRate, As, mu, h); | ||||||
| 
 | 
 | ||||||
|     fir_audio_filter = firfilt_crcf_create(h, h_len); |     fir_audio_filter = firfilt_crcf_create(h, h_len); | ||||||
| 
 | 
 | ||||||
| @ -51,117 +50,112 @@ DemodulatorThread::DemodulatorThread(DemodulatorThreadQueue* pQueue, Demodulator | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| DemodulatorThread::~DemodulatorThread() { | DemodulatorThread::~DemodulatorThread() { | ||||||
| 
 |     std::cout << std::endl << "Demodulator Thread Done." << std::endl << std::endl; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| wxThread::ExitCode DemodulatorThread::Entry() { | void DemodulatorThread::threadMain() { | ||||||
| 
 | 
 | ||||||
|     while (!TestDestroy()) { |     while (1) { | ||||||
|  |         DemodulatorThreadIQData inp; | ||||||
|  |         m_pQueue->pop(inp); | ||||||
| 
 | 
 | ||||||
|         if (m_pQueue->stackSize()) { |         std::vector<signed char> *data = &inp.data; | ||||||
|  |         if (data->size()) { | ||||||
|  |             liquid_float_complex filtered_input[BUF_SIZE / 2]; | ||||||
| 
 | 
 | ||||||
|             while (m_pQueue->stackSize()) { |             for (int i = 0; i < BUF_SIZE / 2; i++) { | ||||||
|                 DemodulatorThreadTask task = m_pQueue->pop(); // pop a task from the queue. this will block the worker thread if queue is empty
 |  | ||||||
|                 switch (task.m_cmd) { |  | ||||||
|                 case DemodulatorThreadTask::DEMOD_THREAD_DATA: |  | ||||||
|                     std::vector<signed char> *data = &task.data->data; |  | ||||||
|                     if (data->size()) { |  | ||||||
|                         liquid_float_complex filtered_input[BUF_SIZE / 2]; |  | ||||||
| 
 | 
 | ||||||
|                         for (int i = 0; i < BUF_SIZE / 2; i++) { |                 liquid_float_complex x; | ||||||
|  |                 liquid_float_complex y; | ||||||
| 
 | 
 | ||||||
|                             liquid_float_complex x; |                 x.real = (float) (*data)[i * 2] / 127.0f; | ||||||
|                             liquid_float_complex y; |                 x.imag = (float) (*data)[i * 2 + 1] / 127.0f; | ||||||
| 
 | 
 | ||||||
|                             x.real = (float) (*data)[i * 2] / 127.0f; |                 firfilt_crcf_push(fir_filter, x);      // push input sample
 | ||||||
|                             x.imag = (float) (*data)[i * 2 + 1] / 127.0f; |                 firfilt_crcf_execute(fir_filter, &y);   // compute output
 | ||||||
| 
 | 
 | ||||||
|                             firfilt_crcf_push(fir_filter, x);      // push input sample
 |                 filtered_input[i] = y; | ||||||
|                             firfilt_crcf_execute(fir_filter, &y);   // compute output
 |             } | ||||||
| 
 | 
 | ||||||
|                             filtered_input[i] = y; |             int out_size = ceil((float) (BUF_SIZE / 2) * resample_ratio); | ||||||
|                         } |  | ||||||
| 
 | 
 | ||||||
|                         int out_size = ceil((float) (BUF_SIZE / 2) * resample_ratio); |             liquid_float_complex resampled_output[out_size]; | ||||||
| 
 | 
 | ||||||
|                         liquid_float_complex resampled_output[out_size]; |             unsigned int num_written;       // number of values written to buffer
 | ||||||
|  |             msresamp_crcf_execute(resampler, filtered_input, (BUF_SIZE / 2), resampled_output, &num_written); | ||||||
| 
 | 
 | ||||||
|                         unsigned int num_written;       // number of values written to buffer
 |             float waveform_ceil = 0, waveform_floor = 0; | ||||||
|                         msresamp_crcf_execute(resampler, filtered_input, (BUF_SIZE / 2), resampled_output, &num_written); |  | ||||||
| 
 | 
 | ||||||
|                         float waveform_ceil = 0, waveform_floor = 0; |             float pcm = 0; | ||||||
| 
 | 
 | ||||||
|                         float pcm = 0; |             for (int i = 0; i < num_written; i++) { | ||||||
|  |                 freqdem_demodulate(fdem, resampled_output[i], &pcm); | ||||||
| 
 | 
 | ||||||
|                         for (int i = 0; i < num_written; i++) { |                 resampled_output[i].real = (float) pcm; | ||||||
|                             freqdem_demodulate(fdem, resampled_output[i], &pcm); |                 resampled_output[i].imag = 0; | ||||||
| 
 | 
 | ||||||
|                             resampled_output[i].real = (float) pcm; |                 if (waveform_ceil < resampled_output[i].real) { | ||||||
|                             resampled_output[i].imag = 0; |                     waveform_ceil = resampled_output[i].real; | ||||||
|  |                 } | ||||||
| 
 | 
 | ||||||
|                             if (waveform_ceil < resampled_output[i].real) { |                 if (waveform_floor > resampled_output[i].real) { | ||||||
|                                 waveform_ceil = resampled_output[i].real; |                     waveform_floor = resampled_output[i].real; | ||||||
|                             } |  | ||||||
| 
 |  | ||||||
|                             if (waveform_floor > resampled_output[i].real) { |  | ||||||
|                                 waveform_floor = resampled_output[i].real; |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         int wbfm_out_size = ceil((float) (num_written) * second_resampler_ratio); |  | ||||||
|                         liquid_float_complex resampled_wbfm_output[wbfm_out_size]; |  | ||||||
| 
 |  | ||||||
|                         unsigned int num_wbfm_written; |  | ||||||
|                         msresamp_crcf_execute(second_resampler, resampled_output, num_written, resampled_wbfm_output, &num_wbfm_written); |  | ||||||
| 
 |  | ||||||
|                         for (int i = 0; i < num_wbfm_written; i++) { |  | ||||||
|                             firfilt_crcf_push(fir_audio_filter, resampled_wbfm_output[i]); |  | ||||||
|                             firfilt_crcf_execute(fir_audio_filter, &resampled_wbfm_output[i]); |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         int audio_out_size = ceil((float) (num_wbfm_written) * audio_resample_ratio); |  | ||||||
|                         liquid_float_complex resampled_audio_output[audio_out_size]; |  | ||||||
| 
 |  | ||||||
|                         unsigned int num_audio_written; |  | ||||||
|                         msresamp_crcf_execute(audio_resampler, resampled_wbfm_output, num_wbfm_written, resampled_audio_output, &num_audio_written); |  | ||||||
| 
 |  | ||||||
|                         std::vector<float> newBuffer; |  | ||||||
|                          newBuffer.resize(num_audio_written * 2); |  | ||||||
|                          for (int i = 0; i < num_audio_written; i++) { |  | ||||||
|                              liquid_float_complex y = resampled_audio_output[i]; |  | ||||||
| 
 |  | ||||||
|                              newBuffer[i * 2] = y.real; |  | ||||||
|                              newBuffer[i * 2 + 1] = y.real; |  | ||||||
|                          } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|                         if (!TestDestroy()) { |  | ||||||
|                             DemodulatorThreadAudioData *audioOut = new DemodulatorThreadAudioData(task.data->frequency,params.audioSampleRate,newBuffer); |  | ||||||
| 
 |  | ||||||
|                             m_pQueue->sendAudioData(DemodulatorThreadTask::DEMOD_THREAD_AUDIO_DATA,audioOut); |  | ||||||
| 
 |  | ||||||
|                             if (params.audioInputQueue != NULL) { |  | ||||||
|                                 AudioThreadInput ati; |  | ||||||
|                                 ati.data = newBuffer; |  | ||||||
|                                 params.audioInputQueue->push(ati); |  | ||||||
| //                                AudioThreadTask audio_task = AudioThreadTask(AudioThreadTask::AUDIO_THREAD_DATA);
 |  | ||||||
| //                                audio_task.data = new AudioThreadData(task.data->frequency, params.audioSampleRate, newBuffer);
 |  | ||||||
| //                                params.audioQueue->addTask(audio_task, AudioThreadQueue::AUDIO_PRIORITY_HIGHEST);
 |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         delete task.data; |  | ||||||
|                     } |  | ||||||
|                     break; |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } else { | 
 | ||||||
|             this->Yield(); |             int wbfm_out_size = ceil((float) (num_written) * second_resampler_ratio); | ||||||
|             this->Sleep(1); |             liquid_float_complex resampled_wbfm_output[wbfm_out_size]; | ||||||
|  | 
 | ||||||
|  |             unsigned int num_wbfm_written; | ||||||
|  |             msresamp_crcf_execute(second_resampler, resampled_output, num_written, resampled_wbfm_output, &num_wbfm_written); | ||||||
|  | 
 | ||||||
|  |             for (int i = 0; i < num_wbfm_written; i++) { | ||||||
|  |                 firfilt_crcf_push(fir_audio_filter, resampled_wbfm_output[i]); | ||||||
|  |                 firfilt_crcf_execute(fir_audio_filter, &resampled_wbfm_output[i]); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             int audio_out_size = ceil((float) (num_wbfm_written) * audio_resample_ratio); | ||||||
|  |             liquid_float_complex resampled_audio_output[audio_out_size]; | ||||||
|  | 
 | ||||||
|  |             unsigned int num_audio_written; | ||||||
|  |             msresamp_crcf_execute(audio_resampler, resampled_wbfm_output, num_wbfm_written, resampled_audio_output, &num_audio_written); | ||||||
|  | 
 | ||||||
|  |             std::vector<float> newBuffer; | ||||||
|  |             newBuffer.resize(num_audio_written * 2); | ||||||
|  |             for (int i = 0; i < num_audio_written; i++) { | ||||||
|  |                 liquid_float_complex y = resampled_audio_output[i]; | ||||||
|  | 
 | ||||||
|  |                 newBuffer[i * 2] = y.real; | ||||||
|  |                 newBuffer[i * 2 + 1] = y.real; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             AudioThreadInput ati; | ||||||
|  |             ati.data = newBuffer; | ||||||
|  | 
 | ||||||
|  |             if (params.audioInputQueue != NULL) { | ||||||
|  |                 params.audioInputQueue->push(ati); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (visOutQueue != NULL) { | ||||||
|  |                 visOutQueue->push(ati); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             /*if (!TestDestroy()) {
 | ||||||
|  |              DemodulatorThreadAudioData *audioOut = new DemodulatorThreadAudioData(task.data->frequency, params.audioSampleRate, newBuffer); | ||||||
|  | 
 | ||||||
|  |              m_pQueue->sendAudioData(DemodulatorThreadTask::DEMOD_THREAD_AUDIO_DATA, audioOut); | ||||||
|  | 
 | ||||||
|  |              if (params.audioInputQueue != NULL) { | ||||||
|  |              AudioThreadInput ati; | ||||||
|  |              ati.data = newBuffer; | ||||||
|  |              params.audioInputQueue->push(ati); | ||||||
|  |              //                                AudioThreadTask audio_task = AudioThreadTask(AudioThreadTask::AUDIO_THREAD_DATA);
 | ||||||
|  |              //                                audio_task.data = new AudioThreadData(task.data->frequency, params.audioSampleRate, newBuffer);
 | ||||||
|  |              //                                params.audioQueue->addTask(audio_task, AudioThreadQueue::AUDIO_PRIORITY_HIGHEST);
 | ||||||
|  |              } | ||||||
|  |              }*/ | ||||||
|  | 
 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     std::cout << std::endl << "Demodulator Thread Done." << std::endl << std::endl; |  | ||||||
| 
 |  | ||||||
|     return (wxThread::ExitCode) 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -10,20 +10,100 @@ | |||||||
| 
 | 
 | ||||||
| #include "wx/thread.h" | #include "wx/thread.h" | ||||||
| 
 | 
 | ||||||
| #include "DemodulatorThreadQueue.h" |  | ||||||
| #include "liquid/liquid.h" | #include "liquid/liquid.h" | ||||||
| #include "AudioThread.h" | #include "AudioThread.h" | ||||||
|  | #include "ThreadQueue.h" | ||||||
|  | #include "CubicSDRDefs.h" | ||||||
| 
 | 
 | ||||||
| class DemodulatorThread: public wxThread { | enum DemodulatorType { | ||||||
|  |     DEMOD_TYPE_NULL, DEMOD_TYPE_AM, DEMOD_TYPE_FM, DEMOD_TYPE_LSB, DEMOD_TYPE_USB, DEMOD_TYPE_WFM | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class DemodulatorThreadIQData { | ||||||
|  | public: | ||||||
|  |     unsigned int frequency; | ||||||
|  |     unsigned int bandwidth; | ||||||
|  |     std::vector<signed char> data; | ||||||
|  | 
 | ||||||
|  |     DemodulatorThreadIQData() : | ||||||
|  |             frequency(0), bandwidth(0) { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     DemodulatorThreadIQData(unsigned int bandwidth, unsigned int frequency, std::vector<signed char> data) : | ||||||
|  |             data(data), frequency(frequency), bandwidth(bandwidth) { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ~DemodulatorThreadIQData() { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class DemodulatorThreadAudioData { | ||||||
|  | public: | ||||||
|  |     unsigned int frequency; | ||||||
|  |     unsigned int sampleRate; | ||||||
|  |     unsigned char channels; | ||||||
|  | 
 | ||||||
|  |     std::vector<float> data; | ||||||
|  | 
 | ||||||
|  |     DemodulatorThreadAudioData() : | ||||||
|  |             sampleRate(0), frequency(0), channels(0) { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     DemodulatorThreadAudioData(unsigned int frequency, unsigned int sampleRate, std::vector<float> data) : | ||||||
|  |             data(data), sampleRate(sampleRate), frequency(frequency), channels(1) { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ~DemodulatorThreadAudioData() { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class DemodulatorThreadParameters { | ||||||
|  | public: | ||||||
|  |     unsigned int inputRate; | ||||||
|  |     unsigned int inputResampleRate; // set equal to disable second stage re-sampling?
 | ||||||
|  |     unsigned int demodResampleRate; | ||||||
|  |     unsigned int filterFrequency; | ||||||
|  |     unsigned int audioSampleRate; | ||||||
|  |     AudioThreadInputQueue *audioInputQueue; | ||||||
|  | 
 | ||||||
|  |     DemodulatorType demodType; | ||||||
|  | 
 | ||||||
|  |     DemodulatorThreadParameters() : | ||||||
|  |             audioInputQueue(NULL), inputRate(SRATE), inputResampleRate(200000), demodResampleRate(100000), audioSampleRate(48000), filterFrequency( | ||||||
|  |                     32000), demodType(DEMOD_TYPE_WFM) { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ~DemodulatorThreadParameters() { | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | typedef ThreadQueue<DemodulatorThreadIQData> DemodulatorThreadInputQueue; | ||||||
|  | typedef ThreadQueue<AudioThreadInput> DemodulatorThreadOutputQueue; | ||||||
|  | 
 | ||||||
|  | class DemodulatorThread { | ||||||
| public: | public: | ||||||
| 
 | 
 | ||||||
|     DemodulatorThread(DemodulatorThreadQueue* pQueue, DemodulatorThreadParameters *params, int id = 0); |     DemodulatorThread(DemodulatorThreadInputQueue* pQueue, DemodulatorThreadParameters *params); | ||||||
|     ~DemodulatorThread(); |     ~DemodulatorThread(); | ||||||
| 
 | 
 | ||||||
|  |     void threadMain(); | ||||||
|  | 
 | ||||||
|  |     void setVisualOutputQueue(DemodulatorThreadOutputQueue *tQueue) { | ||||||
|  |         visOutQueue = tQueue; | ||||||
|  |         visOutQueue->set_max_num_items(1); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
| protected: | protected: | ||||||
|     virtual ExitCode Entry(); |     DemodulatorThreadInputQueue* m_pQueue; | ||||||
|     DemodulatorThreadQueue* m_pQueue; |     DemodulatorThreadOutputQueue* visOutQueue; | ||||||
|     int m_ID; |  | ||||||
| 
 | 
 | ||||||
|     firfilt_crcf fir_filter; |     firfilt_crcf fir_filter; | ||||||
|     firfilt_crcf fir_audio_filter; |     firfilt_crcf fir_audio_filter; | ||||||
|  | |||||||
| @ -1,44 +0,0 @@ | |||||||
| #include "DemodulatorThreadQueue.h" |  | ||||||
| 
 |  | ||||||
| #include "wx/wxprec.h" |  | ||||||
| 
 |  | ||||||
| #ifndef WX_PRECOMP |  | ||||||
| #include "wx/wx.h" |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| DemodulatorThreadQueue::DemodulatorThreadQueue(wxEvtHandler* pParent) : |  | ||||||
|         m_pParent(pParent) { |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void DemodulatorThreadQueue::addTask(const DemodulatorThreadTask& task, const DEMOD_PRIORITY& priority) { |  | ||||||
|     wxMutexLocker lock(m_MutexQueue); |  | ||||||
|     m_Tasks.insert(std::make_pair(priority, task)); |  | ||||||
|     m_QueueCount.Post(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| DemodulatorThreadTask DemodulatorThreadQueue::pop() { |  | ||||||
|     DemodulatorThreadTask element; |  | ||||||
|     m_QueueCount.Wait(); |  | ||||||
|     m_MutexQueue.Lock(); |  | ||||||
|     element = (m_Tasks.begin())->second; |  | ||||||
|     m_Tasks.erase(m_Tasks.begin()); |  | ||||||
|     m_MutexQueue.Unlock(); |  | ||||||
|     return element; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void DemodulatorThreadQueue::sendAudioData(const DemodulatorThreadTask::DEMOD_THREAD_COMMAND& cmd, DemodulatorThreadAudioData *data) { |  | ||||||
|     wxCommandEvent evt(wxEVT_THREAD, cmd); |  | ||||||
| 
 |  | ||||||
|     evt.SetClientData(data); |  | ||||||
| 
 |  | ||||||
|     m_pParent->AddPendingEvent(evt); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| size_t DemodulatorThreadQueue::stackSize() { |  | ||||||
|     wxMutexLocker lock(m_MutexQueue); |  | ||||||
|     return m_Tasks.size(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| wxEvtHandler* DemodulatorThreadQueue::getHandler() { |  | ||||||
|     return m_pParent; |  | ||||||
| } |  | ||||||
| @ -1,28 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| 
 |  | ||||||
| #include <map> |  | ||||||
| #include "DemodulatorThreadTask.h" |  | ||||||
| 
 |  | ||||||
| #include "wx/event.h" |  | ||||||
| 
 |  | ||||||
| class DemodulatorThreadQueue { |  | ||||||
| public: |  | ||||||
|     enum DEMOD_PRIORITY { |  | ||||||
|         DEMOD_PRIORITY_HIGHEST, DEMOD_PRIORITY_HIGHER, DEMOD_PRIORITY_NORMAL, DEMOD_PRIORITY_BELOW_NORMAL, DEMOD_PRIORITY_LOW, DEMOD_PRIORITY_IDLE |  | ||||||
|     }; |  | ||||||
|     DemodulatorThreadQueue(wxEvtHandler* pParent); |  | ||||||
| 
 |  | ||||||
|     void addTask(const DemodulatorThreadTask& task, const DEMOD_PRIORITY& priority = DEMOD_PRIORITY_NORMAL); |  | ||||||
|     void sendAudioData(const DemodulatorThreadTask::DEMOD_THREAD_COMMAND& cmd, DemodulatorThreadAudioData *data); |  | ||||||
| 
 |  | ||||||
|     DemodulatorThreadTask pop(); |  | ||||||
|     size_t stackSize(); |  | ||||||
| 
 |  | ||||||
|     wxEvtHandler* getHandler(); |  | ||||||
| 
 |  | ||||||
| private: |  | ||||||
|     wxEvtHandler* m_pParent; |  | ||||||
|     std::multimap<DEMOD_PRIORITY, DemodulatorThreadTask> m_Tasks; |  | ||||||
|     wxMutex m_MutexQueue; |  | ||||||
|     wxSemaphore m_QueueCount; |  | ||||||
| }; |  | ||||||
| @ -1,2 +0,0 @@ | |||||||
| #include "DemodulatorThreadTask.h" |  | ||||||
| 
 |  | ||||||
| @ -1,91 +0,0 @@ | |||||||
| #pragma once |  | ||||||
| 
 |  | ||||||
| #include <vector> |  | ||||||
| #include "wx/defs.h" |  | ||||||
| #include "wx/string.h" |  | ||||||
| #include "wx/object.h" |  | ||||||
| #include "CubicSDRDefs.h" |  | ||||||
| #include "AudioThread.h" |  | ||||||
| 
 |  | ||||||
| enum DemodulatorType { |  | ||||||
|     DEMOD_TYPE_NULL, DEMOD_TYPE_AM, DEMOD_TYPE_FM, DEMOD_TYPE_LSB, DEMOD_TYPE_USB, DEMOD_TYPE_WFM |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| class DemodulatorThreadIQData: public wxObject { |  | ||||||
| public: |  | ||||||
|     unsigned int frequency; |  | ||||||
|     unsigned int bandwidth; |  | ||||||
|     std::vector<signed char> data; |  | ||||||
| 
 |  | ||||||
|     DemodulatorThreadIQData(unsigned int bandwidth, unsigned int frequency, std::vector<signed char> data) : |  | ||||||
|             data(data), frequency(frequency), bandwidth(bandwidth) { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~DemodulatorThreadIQData() { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| class DemodulatorThreadAudioData: public wxObject { |  | ||||||
| public: |  | ||||||
|     unsigned int frequency; |  | ||||||
|     unsigned int sampleRate; |  | ||||||
|     unsigned char channels; |  | ||||||
| 
 |  | ||||||
|     std::vector<float> data; |  | ||||||
| 
 |  | ||||||
|     DemodulatorThreadAudioData(unsigned int frequency, unsigned int sampleRate, std::vector<float> data) : |  | ||||||
|             data(data), sampleRate(sampleRate), frequency(frequency), channels(1) { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~DemodulatorThreadAudioData() { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| class DemodulatorThreadParameters: public wxObject { |  | ||||||
| public: |  | ||||||
|     unsigned int inputRate; |  | ||||||
|     unsigned int inputResampleRate; // set equal to disable second stage re-sampling?
 |  | ||||||
|     unsigned int demodResampleRate; |  | ||||||
|     unsigned int filterFrequency; |  | ||||||
|     unsigned int audioSampleRate; |  | ||||||
|     AudioThreadInputQueue *audioInputQueue; |  | ||||||
| 
 |  | ||||||
|     DemodulatorType demodType; |  | ||||||
| 
 |  | ||||||
|     DemodulatorThreadParameters() : |  | ||||||
|         audioInputQueue(NULL), inputRate(SRATE), inputResampleRate(200000), demodResampleRate(100000), audioSampleRate(48000), filterFrequency(32000), demodType(DEMOD_TYPE_WFM) { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ~DemodulatorThreadParameters() { |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| class DemodulatorThreadTask { |  | ||||||
| public: |  | ||||||
|     enum DEMOD_THREAD_COMMAND { |  | ||||||
|         DEMOD_THREAD_EXIT = wxID_EXIT, |  | ||||||
|         DEMOD_THREAD_NULL = wxID_HIGHEST + 100, |  | ||||||
|         DEMOD_THREAD_STARTED, |  | ||||||
|         DEMOD_THREAD_PROCESS, |  | ||||||
|         DEMOD_THREAD_ERROR, |  | ||||||
|         DEMOD_THREAD_DATA, |  | ||||||
|         DEMOD_THREAD_AUDIO_DATA |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     DemodulatorThreadTask() : |  | ||||||
|             m_cmd(DEMOD_THREAD_NULL), data(NULL) { |  | ||||||
|     } |  | ||||||
|     DemodulatorThreadTask(DEMOD_THREAD_COMMAND cmd) : |  | ||||||
|             m_cmd(cmd), data(NULL) { |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     DEMOD_THREAD_COMMAND m_cmd; |  | ||||||
| 
 |  | ||||||
|     DemodulatorThreadIQData *data; |  | ||||||
| }; |  | ||||||
| @ -170,11 +170,15 @@ void SDRThread::threadMain() { | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (demodulators.size()) { |         if (demodulators.size()) { | ||||||
|  |             DemodulatorThreadIQData demodDataOut; | ||||||
|  |             demodDataOut.frequency = frequency; | ||||||
|  |             demodDataOut.bandwidth = bandwidth; | ||||||
|  |             demodDataOut.data = new_buffer; | ||||||
|  | 
 | ||||||
|             for (int i = 0, iMax = demodulators.size(); i < iMax; i++) { |             for (int i = 0, iMax = demodulators.size(); i < iMax; i++) { | ||||||
|                 DemodulatorThreadQueue *demodQueue = demodulators[i]; |                 DemodulatorInstance *demod = demodulators[i]; | ||||||
|                 DemodulatorThreadTask demod_task = DemodulatorThreadTask(DemodulatorThreadTask::DEMOD_THREAD_DATA); |                 DemodulatorThreadInputQueue *demodQueue = demod->threadQueueDemod; | ||||||
|                 demod_task.data = new DemodulatorThreadIQData(bandwidth, frequency, new_buffer); |                 demodQueue->push(demodDataOut); | ||||||
|                 demodQueue->addTask(demod_task, DemodulatorThreadQueue::DEMOD_PRIORITY_HIGHEST); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -9,9 +9,8 @@ | |||||||
| 
 | 
 | ||||||
| #include "wx/thread.h" | #include "wx/thread.h" | ||||||
| 
 | 
 | ||||||
| #include "DemodulatorThreadQueue.h" |  | ||||||
| #include "DemodulatorMgr.h" |  | ||||||
| #include "ThreadQueue.h" | #include "ThreadQueue.h" | ||||||
|  | #include "DemodulatorMgr.h" | ||||||
| 
 | 
 | ||||||
| class SDRThreadCommand { | class SDRThreadCommand { | ||||||
| public: | public: | ||||||
| @ -65,8 +64,8 @@ public: | |||||||
| 
 | 
 | ||||||
|     int enumerate_rtl(); |     int enumerate_rtl(); | ||||||
| 
 | 
 | ||||||
|     void bindDemodulator(DemodulatorInstance &demod) { |     void bindDemodulator(DemodulatorInstance *demod) { | ||||||
|         demodulators.push_back(demod.threadQueueDemod); |         demodulators.push_back(demod); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void threadMain(); |     void threadMain(); | ||||||
| @ -85,5 +84,5 @@ protected: | |||||||
|     SDRThreadIQDataQueue* iqDataOutQueue; |     SDRThreadIQDataQueue* iqDataOutQueue; | ||||||
|     SDRThreadIQDataQueue* iqVisualQueue; |     SDRThreadIQDataQueue* iqVisualQueue; | ||||||
| 
 | 
 | ||||||
|     std::vector<DemodulatorThreadQueue *> demodulators; |     std::vector<DemodulatorInstance *> demodulators; | ||||||
| }; | }; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user