2015-10-04 16:07:14 -04:00
|
|
|
#include "SDREnumerator.h"
|
|
|
|
#include "CubicSDRDefs.h"
|
|
|
|
#include <vector>
|
|
|
|
#include "CubicSDR.h"
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<std::string> SDREnumerator::factories;
|
|
|
|
std::vector<std::string> SDREnumerator::modules;
|
2015-10-05 02:21:08 -04:00
|
|
|
std::vector<std::string> SDREnumerator::remotes;
|
|
|
|
std::map< std::string, std::vector<SDRDeviceInfo *> > SDREnumerator::devs;
|
|
|
|
bool SDREnumerator::soapy_initialized = false;
|
|
|
|
bool SDREnumerator::has_remote = false;
|
2015-10-04 16:07:14 -04:00
|
|
|
|
|
|
|
SDREnumerator::SDREnumerator() : IOThread() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
SDREnumerator::~SDREnumerator() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
std::vector<SDRDeviceInfo *> *SDREnumerator::enumerate_devices(std::string remoteAddr, bool noInit) {
|
2015-10-04 16:07:14 -04:00
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
if (SDREnumerator::devs[remoteAddr].size()) {
|
|
|
|
return &SDREnumerator::devs[remoteAddr];
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
if (noInit) {
|
|
|
|
return NULL;
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
2015-10-05 02:21:08 -04:00
|
|
|
|
|
|
|
if (!soapy_initialized) {
|
|
|
|
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 (SDREnumerator::factories.size()) {
|
|
|
|
SDREnumerator::factories.erase(SDREnumerator::factories.begin(), SDREnumerator::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;
|
|
|
|
|
|
|
|
if (it->first == "remote") {
|
|
|
|
has_remote = true;
|
|
|
|
}
|
|
|
|
SDREnumerator::factories.push_back(it->first);
|
|
|
|
}
|
|
|
|
if (factories.empty()) {
|
|
|
|
std::cout << "No factories found!" << std::endl;
|
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
soapy_initialized = true;
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
std::vector<SoapySDR::Kwargs> results;
|
|
|
|
SoapySDR::Kwargs enumArgs;
|
|
|
|
bool isRemote = false;
|
2015-10-04 16:07:14 -04:00
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
if (remoteAddr.length()) {
|
|
|
|
std::cout << "Enumerating remote address: " << remoteAddr << std::endl;
|
|
|
|
enumArgs["driver"] = "remote";
|
|
|
|
enumArgs["remote"] = remoteAddr;
|
|
|
|
isRemote = true;
|
|
|
|
|
|
|
|
results = SoapySDR::Device::enumerate(enumArgs);
|
|
|
|
} else {
|
|
|
|
results = SoapySDR::Device::enumerate();
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
// */
|
2015-10-05 02:21:08 -04:00
|
|
|
if (isRemote) {
|
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, std::string("Opening remote server ") + remoteAddr + "..");
|
|
|
|
}
|
2015-10-04 16:07:14 -04:00
|
|
|
for (size_t i = 0; i < results.size(); i++) {
|
2015-10-05 02:21:08 -04:00
|
|
|
// std::cout << "Found device " << i << std::endl;
|
2015-10-04 16:07:14 -04:00
|
|
|
SDRDeviceInfo *dev = new SDRDeviceInfo();
|
2015-10-05 02:21:08 -04:00
|
|
|
|
|
|
|
SoapySDR::Kwargs deviceArgs = results[i];
|
|
|
|
SoapySDR::Kwargs streamArgs;
|
|
|
|
|
|
|
|
if (isRemote) {
|
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, "Querying remote " + remoteAddr + " device #" + std::to_string(i));
|
2015-10-14 00:54:48 -04:00
|
|
|
// deviceArgs["remote"] = remoteAddr;
|
2015-10-05 02:21:08 -04:00
|
|
|
if (deviceArgs.count("rtl") != 0) {
|
|
|
|
streamArgs["remote:mtu"] = "8192";
|
|
|
|
streamArgs["remote:format"] = "CS8";
|
|
|
|
streamArgs["remote:window"] = "16384000";
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, std::string("Found local device #") + std::to_string(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (deviceArgs.count("rtl") != 0 || (deviceArgs.count("driver") != 0 && (deviceArgs["driver"] == "rtl" || deviceArgs["driver"] == "rtlsdr"))) {
|
|
|
|
streamArgs["buffers"] = "6";
|
|
|
|
streamArgs["buflen"] = "16384";
|
|
|
|
}
|
|
|
|
|
|
|
|
for (SoapySDR::Kwargs::const_iterator it = deviceArgs.begin(); it != deviceArgs.end(); ++it) {
|
2015-10-04 16:07:14 -04:00
|
|
|
std::cout << " " << it->first << " = " << it->second << std::endl;
|
|
|
|
if (it->first == "driver") {
|
|
|
|
dev->setDriver(it->second);
|
2015-10-11 03:46:16 -04:00
|
|
|
} else if (it->first == "label" || it->first == "device") {
|
2015-10-04 16:07:14 -04:00
|
|
|
dev->setName(it->second);
|
2015-10-11 03:46:16 -04:00
|
|
|
}
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
2015-10-05 02:21:08 -04:00
|
|
|
|
|
|
|
dev->setDeviceArgs(deviceArgs);
|
|
|
|
dev->setStreamArgs(deviceArgs);
|
2015-10-04 16:07:14 -04:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-04 19:55:52 -04:00
|
|
|
int numChan = device->getNumChannels(SOAPY_SDR_RX);
|
|
|
|
for (int i = 0; i < numChan; i++) {
|
|
|
|
SDRDeviceChannel *chan = new SDRDeviceChannel();
|
|
|
|
|
|
|
|
SoapySDR::RangeList rfRange = device->getFrequencyRange(SOAPY_SDR_RX, i);
|
|
|
|
double rfMin = rfRange[0].minimum();
|
|
|
|
double rfMax = rfRange[rfRange.size()-1].maximum();
|
|
|
|
chan->setChannel(i);
|
|
|
|
chan->setFullDuplex(device->getFullDuplex(SOAPY_SDR_RX, i));
|
|
|
|
chan->setRx(true);
|
|
|
|
chan->setTx(false);
|
|
|
|
chan->getRFRange().setLow(rfMin);
|
|
|
|
chan->getRFRange().setHigh(rfMax);
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
std::vector<std::string> freqs = device->listFrequencies(SOAPY_SDR_RX,i);
|
|
|
|
if (std::find(freqs.begin(), freqs.end(), "CORR") != freqs.end()) {
|
|
|
|
chan->setCORR(true);
|
|
|
|
} else {
|
|
|
|
chan->setCORR(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (device->hasDCOffsetMode(SOAPY_SDR_RX, i)) {
|
|
|
|
chan->setHardwareDC(true);
|
|
|
|
} else {
|
|
|
|
chan->setHardwareDC(false);
|
|
|
|
}
|
|
|
|
|
2015-10-04 19:55:52 -04:00
|
|
|
std::vector<double> rates = device->listSampleRates(SOAPY_SDR_RX, i);
|
|
|
|
chan->getSampleRates().assign(rates.begin(), rates.end());
|
|
|
|
dev->addChannel(chan);
|
|
|
|
}
|
|
|
|
|
2015-10-04 16:07:14 -04:00
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
|
2015-10-04 16:07:14 -04:00
|
|
|
SoapySDR::Device::unmake(device);
|
|
|
|
|
|
|
|
dev->setAvailable(true);
|
|
|
|
} catch (const std::exception &ex) {
|
|
|
|
std::cerr << "Error making device: " << ex.what() << std::endl;
|
2015-10-05 02:21:08 -04:00
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, std::string("Error making device #") + std::to_string(i));
|
2015-10-04 16:07:14 -04:00
|
|
|
dev->setAvailable(false);
|
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
SDREnumerator::devs[remoteAddr].push_back(dev);
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
2015-10-05 02:21:08 -04:00
|
|
|
if (SDREnumerator::devs[remoteAddr].empty()) {
|
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, std::string("No devices found!"));
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
return &SDREnumerator::devs[remoteAddr];
|
2015-10-04 16:07:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SDREnumerator::run() {
|
|
|
|
|
|
|
|
std::cout << "SDR enumerator starting." << std::endl;
|
|
|
|
terminated.store(false);
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
// if (!remotes.size()) {
|
|
|
|
// remotes.push_back("raspberrypi.local");
|
|
|
|
// }
|
|
|
|
|
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, "Scanning local devices, please wait..");
|
|
|
|
SDREnumerator::enumerate_devices("");
|
2015-10-04 16:07:14 -04:00
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
if (remotes.size()) {
|
|
|
|
std::vector<std::string>::iterator remote_i;
|
|
|
|
for (remote_i = remotes.begin(); remote_i != remotes.end(); remote_i++) {
|
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_MESSAGE, "Scanning devices at " + (*remote_i) + ", please wait..");
|
|
|
|
SDREnumerator::enumerate_devices(*remote_i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-04 16:07:14 -04:00
|
|
|
std::cout << "Reporting enumeration complete." << std::endl;
|
|
|
|
terminated.store(true);
|
2015-10-05 02:21:08 -04:00
|
|
|
wxGetApp().sdrEnumThreadNotify(SDREnumerator::SDR_ENUM_DEVICES_READY, "Finished scanning devices.");
|
2015-10-04 16:07:14 -04:00
|
|
|
std::cout << "SDR enumerator done." << std::endl;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-10-05 02:21:08 -04:00
|
|
|
|
|
|
|
void SDREnumerator::addRemote(std::string remoteAddr) {
|
|
|
|
std::vector<std::string>::iterator remote_i = std::find(remotes.begin(), remotes.end(), remoteAddr);
|
|
|
|
|
|
|
|
if (remote_i != remotes.end()) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
remotes.push_back(remoteAddr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDREnumerator::removeRemote(std::string remoteAddr) {
|
|
|
|
std::vector<std::string>::iterator remote_i = std::find(remotes.begin(), remotes.end(), remoteAddr);
|
|
|
|
|
|
|
|
if (remote_i != remotes.end()) {
|
|
|
|
if (devs.find(*remote_i) != devs.end()) {
|
|
|
|
while (devs[*remote_i].size()) {
|
|
|
|
SDRDeviceInfo *devRemove = devs[*remote_i].back();
|
|
|
|
devs[*remote_i].pop_back();
|
|
|
|
delete devRemove;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
remotes.erase(remote_i);
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> &SDREnumerator::getRemotes() {
|
|
|
|
return remotes;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SDREnumerator::hasRemoteModule() {
|
|
|
|
return SDREnumerator::has_remote;
|
|
|
|
}
|