#include "SoapySDRThread.h" #include "CubicSDRDefs.h" #include #include "CubicSDR.h" #include std::vector SDRThread::factories; std::vector SDRThread::modules; std::vector SDRThread::devs; SDRThread::SDRThread() : IOThread() { device = NULL; deviceConfig.store(NULL); deviceInfo.store(NULL); sampleRate.store(DEFAULT_SAMPLE_RATE); frequency.store(0); offset.store(0); ppm.store(0); direct_sampling_mode.store(0); numElems.store(0); rate_changed.store(false); freq_changed.store(false); offset_changed.store(false); ppm_changed .store(false); direct_sampling_changed.store(false); device_changed.store(false); hasPPM.store(false); hasHardwareDC.store(false); } SDRThread::~SDRThread() { } std::vector *SDRThread::enumerate_devices() { if (SDRThread::devs.size()) { return &SDRThread::devs; } std::cout << "SoapySDR init.." << std::endl; std::cout << "\tAPI Version: v" << SoapySDR::getAPIVersion() << std::endl; std::cout << "\tABI Version: v" << SoapySDR::getABIVersion() << std::endl; std::cout << "\tInstall root: " << SoapySDR::getRootPath() << std::endl; modules = SoapySDR::listModules(); for (size_t i = 0; i < modules.size(); i++) { std::cout << "\tModule found: " << modules[i] << std::endl; } if (modules.empty()) { std::cout << "No modules found!" << std::endl; } std::cout << "\tLoading modules... " << std::flush; SoapySDR::loadModules(); std::cout << "done" << std::endl; if (SDRThread::factories.size()) { SDRThread::factories.erase(SDRThread::factories.begin(), SDRThread::factories.end()); } std::cout << "\tAvailable factories..."; SoapySDR::FindFunctions factories = SoapySDR::Registry::listFindFunctions(); for (SoapySDR::FindFunctions::const_iterator it = factories.begin(); it != factories.end(); ++it) { if (it != factories.begin()) { std::cout << ", "; } std::cout << it->first; SDRThread::factories.push_back(it->first); } if (factories.empty()) { std::cout << "No factories found!" << std::endl; } std::cout << std::endl; std::vector results = SoapySDR::Device::enumerate(); // Remote driver test.. /* * / SDRDeviceInfo *remoteDev = new SDRDeviceInfo(); remoteDev->setDriver("remote"); remoteDev->setName("SoapySDR Remote Test"); SoapySDR::Kwargs remoteArgs; remoteArgs["driver"] = "remote"; // remoteArgs["remote"] = "127.0.0.1"; remoteArgs["remote"] = "192.168.1.103"; remoteArgs["remote:driver"] = "rtlsdr"; remoteArgs["buffers"] = "6"; remoteArgs["buflen"] = "16384"; remoteDev->setDeviceArgs(remoteArgs); SoapySDR::Kwargs streamArgs; streamArgs["remote:mtu"] = "8192"; streamArgs["remote:format"] = "CS8"; streamArgs["remote:window"] = "16384000"; remoteDev->setStreamArgs(streamArgs); SDRThread::devs.push_back(remoteDev); // */ for (size_t i = 0; i < results.size(); i++) { std::cout << "Found device " << i << std::endl; SDRDeviceInfo *dev = new SDRDeviceInfo(); for (SoapySDR::Kwargs::const_iterator it = results[i].begin(); it != results[i].end(); ++it) { std::cout << " " << it->first << " = " << it->second << std::endl; if (it->first == "driver") { dev->setDriver(it->second); } else if (it->first == "label") { dev->setName(it->second); } } dev->setDeviceArgs(results[i]); std::cout << "Make device " << i << std::endl; try { SoapySDR::Device *device = SoapySDR::Device::make(dev->getDeviceArgs()); SoapySDR::Kwargs info = device->getHardwareInfo(); for (SoapySDR::Kwargs::const_iterator it = info.begin(); it != info.end(); ++it) { std::cout << " " << it->first << "=" << it->second << std::endl; if (it->first == "hardware") { dev->setHardware(it->second); } } if (device->hasDCOffsetMode(SOAPY_SDR_RX, 0)) { device->setDCOffsetMode(SOAPY_SDR_RX, 0, true); std::cout << "Hardware DC offset support detected; internal DC offset correction will be disabled." << std::endl; dev->setHardwareDC(true); } else { dev->setHardwareDC(false); } SoapySDR::Device::unmake(device); dev->setAvailable(true); } catch (const std::exception &ex) { std::cerr << "Error making device: " << ex.what() << std::endl; dev->setAvailable(false); } std::cout << std::endl; SDRThread::devs.push_back(dev); } if (results.empty()) { std::cout << "No devices found!" << std::endl; } std::cout << std::endl; return &SDRThread::devs; } void SDRThread::init() { SDRDeviceInfo *devInfo = deviceInfo.load(); deviceConfig.store(wxGetApp().getConfig()->getDevice(devInfo->getDeviceId())); DeviceConfig *devConfig = deviceConfig.load(); frequency = wxGetApp().getConfig()->getCenterFreq(); ppm.store(devConfig->getPPM()); direct_sampling_mode.store(devConfig->getDirectSampling()); std::string driverName = devInfo->getDriver(); offset = devConfig->getOffset(); wxGetApp().setSwapIQ(devConfig->getIQSwap()); SoapySDR::Kwargs args = devInfo->getDeviceArgs(); args["direct_samp"] = std::to_string(devConfig->getDirectSampling()); if (driverName == "rtl" || driverName == "rtlsdr") { args["buffers"] = "6"; args["buflen"] = "16384"; hasPPM = true; } else { hasPPM = false; } device = SoapySDR::Device::make(args); stream = device->setupStream(SOAPY_SDR_RX,"CF32", std::vector(), devInfo->getStreamArgs()); device->activateStream(stream); device->setSampleRate(SOAPY_SDR_RX,0,sampleRate.load()); device->setFrequency(SOAPY_SDR_RX,0,"RF",frequency - offset.load()); if (hasPPM) { device->setFrequency(SOAPY_SDR_RX,0,"CORR",ppm); } device->setGainMode(SOAPY_SDR_RX,0,true); hasHardwareDC = devInfo->hasHardwareDC(); numElems = getOptimalElementCount(sampleRate.load(), 60); buffs[0] = malloc(numElems * 2 * sizeof(float)); } void SDRThread::deinit() { device->deactivateStream(stream); device->closeStream(stream); SoapySDR::Device::unmake(device); free(buffs[0]); } void SDRThread::readStream(SDRThreadIQDataQueue* iqDataOutQueue) { int flags; long long timeNs; SDRThreadIQData *dataOut = buffers.getBuffer(); if (dataOut->data.size() != numElems * 2) { dataOut->data.resize(numElems * 2); } int n_read = 0; while (n_read != numElems) { int n_stream_read = device->readStream(stream, buffs, numElems-n_read, flags, timeNs); if (n_stream_read > 0) { memcpy(&dataOut->data[n_read * 2], buffs[0], n_stream_read * sizeof(float) * 2); n_read += n_stream_read; } else { dataOut->data.resize(n_read); break; } } // std::cout << n_read << std::endl; if (n_read > 0) { dataOut->setRefCount(1); dataOut->frequency = frequency; dataOut->sampleRate = sampleRate.load(); dataOut->dcCorrected = hasHardwareDC; iqDataOutQueue->push(dataOut); } } void SDRThread::readLoop() { SDRThreadIQDataQueue* iqDataOutQueue = (SDRThreadIQDataQueue*) getOutputQueue("IQDataOutput"); if (iqDataOutQueue == NULL) { return; } while (!terminated.load()) { if (offset_changed.load()) { if (!freq_changed.load()) { frequency.store(frequency.load()); freq_changed.store(true); } offset_changed.store(false); } if (rate_changed.load()) { device->setSampleRate(SOAPY_SDR_RX,0,sampleRate.load()); sampleRate.store(device->getSampleRate(SOAPY_SDR_RX,0)); numElems.store(getOptimalElementCount(sampleRate.load(), 60)); free(buffs[0]); buffs[0] = malloc(numElems.load() * 2 * sizeof(float)); rate_changed.store(false); } if (ppm_changed.load() && hasPPM.load()) { device->setFrequency(SOAPY_SDR_RX,0,"CORR",ppm.load()); direct_sampling_changed.store(false); } if (freq_changed.load()) { device->setFrequency(SOAPY_SDR_RX,0,"RF",frequency.load() - offset.load()); freq_changed.store(false); } if (direct_sampling_changed.load()) { // rtlsdr_set_direct_sampling(dev, direct_sampling_mode); } readStream(iqDataOutQueue); } buffers.purge(); } void SDRThread::run() { //#ifdef __APPLE__ // pthread_t tID = pthread_self(); // ID of this thread // int priority = sched_get_priority_max( SCHED_FIFO) - 1; // sched_param prio = { priority }; // scheduling priority of thread // pthread_setschedparam(tID, SCHED_FIFO, &prio); //#endif std::cout << "SDR thread starting." << std::endl; terminated.store(false); if (deviceInfo.load() != NULL) { std::cout << "device init()" << std::endl; init(); std::cout << "starting readLoop()" << std::endl; readLoop(); std::cout << "readLoop() ended." << std::endl; deinit(); std::cout << "device deinit()" << std::endl; } else { std::cout << "Device setting not found, enumerating." << std::endl; SDRThread::enumerate_devices(); std::cout << "Reporting enumeration complete." << std::endl; wxGetApp().sdrThreadNotify(SDRThread::SDR_THREAD_DEVICES_READY, "Devices Ready."); terminated.store(true); return; } std::cout << "SDR thread done." << std::endl; if (!terminated.load()) { terminated.store(true); wxGetApp().sdrThreadNotify(SDRThread::SDR_THREAD_TERMINATED, "Done."); } } SDRDeviceInfo *SDRThread::getDevice() { return deviceInfo.load(); } void SDRThread::setDevice(SDRDeviceInfo *dev) { deviceInfo.store(dev); deviceConfig.store(wxGetApp().getConfig()->getDevice(dev->getDeviceId())); } int SDRThread::getOptimalElementCount(long long sampleRate, int fps) { int elemCount = (int)floor((double)sampleRate/(double)fps); elemCount = int(ceil((double)elemCount/512.0)*512.0); std::cout << "Calculated optimal element count of " << elemCount << std::endl; return elemCount; } void SDRThread::setFrequency(long long freq) { if (freq < sampleRate.load() / 2) { freq = sampleRate.load() / 2; } frequency.store(freq); freq_changed.store(true); } long long SDRThread::getFrequency() { return frequency.load(); } void SDRThread::setOffset(long long ofs) { offset.store(ofs); offset_changed.store(true); std::cout << "Set offset: " << offset.load() << std::endl; } long long SDRThread::getOffset() { return offset.load(); } void SDRThread::setSampleRate(int rate) { sampleRate.store(rate); rate_changed = true; std::cout << "Set sample rate: " << sampleRate.load() << std::endl; } int SDRThread::getSampleRate() { return sampleRate.load(); } void SDRThread::setPPM(int ppm) { this->ppm.store(ppm); ppm_changed.store(true); std::cout << "Set PPM: " << this->ppm.load() << std::endl; } int SDRThread::getPPM() { return ppm.load(); } void SDRThread::setDirectSampling(int dsMode) { direct_sampling_mode.store(dsMode); direct_sampling_changed.store(true); std::cout << "Set direct sampling mode: " << this->direct_sampling_mode.load() << std::endl; } int SDRThread::getDirectSampling() { return direct_sampling_mode.load(); }