The release type can be specified with the -DBUILD_TYPE
cmake option. It takes the following values:
RELEASE
(default): produces production release code i.e.optimized and no debug symbolsRELEASEWITHDBGINFO
: optimized with debug infoDEBUG
: unoptimized with debug info
You can specify whether or not you want to see debug messages printed out to the console with the -DDEBUG_OUTPUT
cmake option:
OFF
(default): no debug outputON
: debug output
You can add -Wno-dev
on the cmake
command line to avoid warnings.
The existing receiving flow is represented with green boxes. The future Tx flow with red boxes. Some classes and file paths in the Rx part were renamed to avoid collision with future Tx names in this case the old name appears below the present name in italics.
- The I/Q samples are collected from a physical device or UDP flow with a device plugin that derives from the
DeviceSampleSource
class. - These I/Q samples are downsampled by a factor set in the device plugin GUI and fed into a
BasebandSampleSink
- The
DownChannelizer
class downsamples further down (or not) the baseband I/Q samples depending on the requirements of the Rx plugin. It cascades the downsampling from the center, left or right half of the baseband in order to fit the Rx plugin bandwidth. It contains the NCO to adjust to the Rx plugin center frequency. - The special
FileSource
plugin reads a file that was recorded using theFileRecord
class directly into the baseband as there is no downsampling from the sample rate at which the file was recorded. - The baseband I/Q samples can be recorded to a file using the
FileRecord
class
At present the following plugins are available:
AirspyXxx
classes inplugins/samplesource/airspy
: Interface with Airspy devicesBladeRFXxx
classes inplugins/samplesource/bladerf
: Interface with BladeRF devicesBladeRFXxx
classes inplugins/samplesource/bladerf
: Interface with BladeRF devicesFCDProXxx
classes inplugins/samplesource/fcdpro
: Interface with Funcube Pro devicesFCDProPlusXxx
classes inplugins/samplesource/fcdproplus
: Interface with Funcube Pro+ devicesHackRFXxx
classes inplugins/samplesource/hackrf
: Interface with HackRF devicesRTLSDRXxx
classes inplugins/samplesource/rtlsdr
: Interface with RTL-SDR devicesRemoteInput
class inplugins/samplesource/remoteinput
: Special interface collecting I/Q samples from an UDP flow sent by a remote instance of SDRangel using Remote sink channel.FileSource
classes inplugins/samplesource/filesource
: Special interface reading I/Q samples from a file directly into the baseband skipping the downsampling block
At present the following plugins are available:
FileSink
classes inplugins/samplesink/filesink
: Special interface writing baseband I/Q samples to a file skipping the final upsampling block
At present the following plugins are available:
ChannelAnalyzerXxx
classes inplugins/channelrx/chanalyzer
: Signal analysis tool pretty much like a DSA/DSO signal analyzer like the venerable HP 4406A (although still far from it!)AMDemodXxx
classes inplugins/channelrx/demodam
: AM demodulator with audio outputBFMDemodXxx
classes inplugins/channelrx/demodbfm
: Broadcast FM demodulator with audio mono/stereo output and RDSDSDDemodXxx
classes inplugins/channelrx/demoddsd
: Digital Speech demodulator/decoder built on top of the DSDcc library. Produces audio output and some communication data from various digital voice standards: DMR, dPMR, D-Star, Yaesu System Fusion (YSF).LoraDemodXxx
classes inplugins/channelrx/demodlora
: Decodes LoRa transmissions. This is legacy code that is not very well maintained so it may or may not work.NFMDemodXxx
classes inplugins/channelrx/demodnfm
: Narrowband FM demodulator with audio output.SSBDemodXxx
classes inplugins/channelrx/demodssb
: SSB/DSB/CW demodulator with audio output.WFMDemodXxx
classes inplugins/channelrx/demodwfm
: Wideband FM demodulator with audio output. This is a basic demodulator.UDPSinkXxx
classes inplugins/channelrx/udpsink
: Sends channel I/Q or FM demodulated samples via UDP
At present the following plugins are available:
AMMmodXxx
classes inplugins/channeltx/modam
: AM modulator with simple sine
At the first subdirectory level indclude
and sdrbase
contain the common core components include and source files respectively. They are further broken down in subdirectories corresponding to a specific area:
audio
contains the interface with the audio device(s)dsp
contains the common blocks for Digital Signal Processing like filters, scope and spectrum analyzer internalsgui
contains the common Graphical User Interface components like the scope and spectrum analyzer controls and displayplugin
contains the common blocks for managing pluginssettings
contains components to manage presets and preferencesutil
contains common utilities such as the message queue
The plugins
subdirectory contains the associated plugins used to manage devices and channel components. Naming convention of various items depend on the usage and Rx (reception side) or Tx (transmission side) affinity. Transmission side is a work in progress.
-
Receiver functions (Rx):
samplesource
: Device managers:xxx
: Device manager (e.g. xxx = airspy)xxxinput.h/cpp
: Device interfacexxxgui.h/cpp
: GUIxxxplugin.h/cpp
: Plugin interfacexxxsettings.h/cpp
: Configuration managerxxxthread.h/cpp
: Reading samplesxxx.pro
: Qt .pro file for Windows/Android build
channelrx
: Channel handlers:demodxxx
: Demodulator internal handler (e.g xxx = demodam)xxxdemod.h/cpp
: Demodulator corexxxdemodgui.h/cpp
: Demodulator GUIxxxdemodplugin.h/cpp
: Plugin interfacedemodxxx.pro
: Qt .pro file for Windows/Android build
xxxanalyzer
: Analyzer internal handler (e.g xxx = channel)xxxanalyzer.h/cpp
: Analyzer corexxxanalyzergui.h/cpp
: Analyzer GUIxxxanalyzerplugin.h/cpp
: Analyzer plugin managerxxxanalyzer.pro
: Qt .pro file for Windows/Android build
xxxsink
: Interface to the outside (e.g xxx = udp):xxxsink.h/cpp
: Interface corexxxsinkgui.h/cpp
: Interface GUIxxxsinkplugin/h/cpp
: Interface plugin managerxxxsink.pro
: Qt .pro file for Windows/Android build
-
Transmitter functions (Tx):
samplesink
: Device managers:xxx
: Device manager (e.g. xxx = bladerf)xxxsinkoutput.h/cpp
: Device interfacexxxsinkgui.h/cpp
: GUIxxxsinkplugin.h/cpp
: Plugin interfacexxxsinksettings.h/cpp
: Configuration managerxxxsinkthread.h/cpp
: Writing samplesxxxsink.pro
: Qt .pro file for Windows/Android build
channeltx
: Channel handlers:modxxx
: Modulator internal handler (e.g xxx = modam)xxxmod.h/cpp
: Modulator corexxxmodgui.h/cpp
: Modulator GUIxxxmodplugin.h/cpp
: Plugin interfacemodxxx.pro
: Qt .pro file for Windows/Android build
xxxgenerator
: Generator internal handler (e.g xxx = channel)xxxgenerator.h/cpp
: Generator corexxxgeneratorgui.h/cpp
: Generator GUIxxxgeneratorplugin.h/cpp
: Generator plugin managerxxxgenerator.pro
: Qt .pro file for Windows/Android build
xxxsource
: Interface to the outside (e.g xxx = udp):xxxsource.h/cpp
: Interface corexxxsourcegui.h/cpp
: Interface GUIxxxsourceplugin/h/cpp
: Interface plugin managerxxxsource.pro
: Qt .pro file for Windows/Android build
Since version 3.4.0 the opening and closing of the physical device has been split off the start and stop methods of the device interface. The opening and closing of the physical device is handled in private methods called from the constructor and destructor of the device interface respectively.
The device interface is itself created in the constructor of the GUI and is deleted at the destruction of the GUI.
The lifecycle of the GUI is controlled from the "Sampling Device Control" device selector in the main window using the plugin manager. When there is a change in the hardware device selection (you can also re-cycle the same device by clicking again on this button) validated by the confirmation button (check sign icon). the following steps are executed in sequence:
- Stop streaming
- Delete the current GUI this will in turn delete the device interface and always close the physical device unless the physical device has a SISO or MIMO architecture (more on that later)
- Remove the current device API from the relevant buddies lists. Buddies list are effective only for physical devices with SISO or MIMO architecture (more on that later)
- Create the new device API
- Add the new device API to the relevant devices APIs buddies list
- Creates the new GUI and hence new device interface. This will always open the physical device unless the physical device has a SISO or MIMO architecture
Here is the relevant part of the code (source side) in the MainWindow::on_sampleSource_confirmClicked
method:
deviceUI->m_deviceSourceAPI->stopAcquisition();
deviceUI->m_deviceSourceAPI->setSampleSourcePluginInstanceUI(0); // deletes old UI and input object
deviceUI->m_deviceSourceAPI->clearBuddiesLists(); // clear old API buddies lists
m_pluginManager->selectSampleSourceByDevice(devicePtr, deviceUI->m_deviceSourceAPI); // sets the new API
// add to buddies list
... here it looks for open tabs with the same physical device and adds the deviceUI->m_deviceSourceAPI to the corresponding m_deviceSourceAPI buddies list ...
// constructs new GUI and input object
QWidget *gui;
PluginManager::SamplingDevice *sampleSourceDevice = (PluginManager::SamplingDevice *) devicePtr; // lightweight representation of the device
PluginInstanceUI *pluginUI = sampleSourceDevice->m_plugin->createSampleSourcePluginInstanceUI(sampleSourceDevice->m_deviceId, &gui, deviceUI->m_deviceSourceAPI);
deviceUI->m_deviceSourceAPI->setSampleSourcePluginInstanceUI(pluginUI);
deviceUI->m_deviceSourceAPI->setInputGUI(gui, sampleSourceDevice->m_displayName);
Some SDR hardware have Rx/Tx capabilities in a SISO or MIMO configuration. That is a physical device can handle more than one I/Q sample flow in either direction. Such devices supported by SDRangel are the following:
- SISO full duplex: BladeRF
- SISO half duplex: HackRF
- MIMO: LimeSDR
Note that the following would also work for multiple sample channels Rx or Tx only devices but none exists or is supported at this moment.
In SDRangel there is a complete receiver or transmitter per I/Q sample flow. These transmitters and receivers are visually represented by the Rn and Tn tabs in the main window. They are created and disposed in the way explained in the previous paragraph using the source or sink selection confirmation button. In fact it acts as if each receiver or transmitter was controlled independently. In single input or single output (none at the moment) devices this is a true independence but with SISO or MIMO devices this cannot be the case and although each receiver or transmitter looks like it was handled independently there are things in common that have to be taken into account. For example in all cases the device handle should be unique and opening and closing the device has to be done only once per physical device usage cycle.
This is where the "buddies list" come into play. Devices exhibit a generic interface in the form of the DeviceAPI class. Through this API some information and control can flow between receivers and transmitters. The point is that all receivers and/or transmitters pertaining to the same physical device must "know" each other in order to be able to exchange information or control each other. For this purpose the DeviceAPI maintain a list of DeviceAPI siblings called "buddies". Thus any multi flow Rx/Tx configuration can be handled.
The exact behaviour of these coupled receivers and/or transmitters is dependent on the hardware therefore a generic pointer attached to the API can be used to convey any kind of class or structure tailored for the exact hardware use case. Through this structure the state of the receiver or transmitter can be exposed therefore there is one structure per receiver and transmitter in the device interface. This structure may contain pointers to common areas (dynamically allocated) related to the physical device. One such "area" is the device handle which is present in all cases.
Normally the first buddy would create the common areas (through new) and the last would delete them (through delete) and the individual structure (superstructure) would be on the stack of each buddy. Thus by copying this superstructure a buddy would gain access to common areas from another (already present) buddy along with static information from the other buddy (such as which hardware Rx or Tx channel it uses in a MIMO architecture). Exchange of dynamic information between buddies is done using message passing.
The degree of entanglement between the different coupled flows in a single hardware can be very different:
-
BladeRF: a single Rx and Tx in SISO configuration loosely coupled:
- independent Rx and Tx sample rates
- independent Rx and Tx center frequencies
- independent Gain, bandwidth, ...
- only the device handle and indication of the presence of the XB200 accessory board is common
-
HackRF: this is a half duplex device. Rx and Tx might appear as tightly coupled but since you can use only one or the other then in fact you can control them differently as this is done in sequence. In fact only the common device handle has to be taken care of
-
LimeSDR: dual Rx and dual Tx in MIMO configuration. This implies tightly coupling between receivers on one part and transmitter on the other. But in the case of the LimeSDR (LMS7002M chip) there is even a tight coupling between Rx and Tx parts since the ADC/DAC clock is in common which means both have the same base sample rate that can only be varied by the mean of the hardware decimators and interpolators. This means:
- Rx share the same sample rate, hardware decimation factor and center frequency
- Tx share the same sample rate, hardware interpolation factor and center frequency
- Rx and Tx share the same base sample rate (decimation/interpolation apart)
- Independent Rx and Tx center frequencies
- Independent gains, bandwidths, etc... per Rx or Tx channel
The openDevice and closeDevice methods of the device interface are designed specifically for each physical device type in order to manage the common resources appropriately at receiver or transmitter construction and destruction. For example opening and closing the device and the related device handle is always done this way:
-
openDevice:
- check if there is any "buddy" present in one of the lists
- if there is a buddy then grab the device handle and use it for next operations
- if there is no buddy open the device and store the handle in the common area so it will be visible for future buddies
-
closeDevice:
- if there are no buddies then effectively close the device else just zero out the own copy of the device handle
Exchange of dynamic information when necessary such as sample rate or center frequency is done by message passing between buddies.