2019-12-28 00:26:45 +01:00
|
|
|
from abc import ABCMeta, abstractmethod
|
|
|
|
from owrx.command import Option
|
2021-02-19 15:29:17 +01:00
|
|
|
from owrx.source.connector import ConnectorSource, ConnectorDeviceDescription
|
|
|
|
from typing import List
|
2021-04-29 15:17:21 +02:00
|
|
|
from owrx.form.input import Input, TextInput
|
|
|
|
from owrx.form.input.device import GainInput
|
2021-02-20 00:16:32 +01:00
|
|
|
from owrx.soapy import SoapySettings
|
2019-12-21 20:58:28 +01:00
|
|
|
|
|
|
|
|
2019-12-28 00:26:45 +01:00
|
|
|
class SoapyConnectorSource(ConnectorSource, metaclass=ABCMeta):
|
2019-12-31 19:14:05 +01:00
|
|
|
def getCommandMapper(self):
|
2021-01-20 17:01:46 +01:00
|
|
|
return (
|
|
|
|
super()
|
|
|
|
.getCommandMapper()
|
|
|
|
.setBase("soapy_connector")
|
|
|
|
.setMappings(
|
|
|
|
{
|
|
|
|
"antenna": Option("-a"),
|
|
|
|
"soapy_settings": Option("-t"),
|
|
|
|
}
|
|
|
|
)
|
2020-03-14 01:04:52 +01:00
|
|
|
)
|
2019-12-28 00:26:45 +01:00
|
|
|
|
2019-12-21 20:58:28 +01:00
|
|
|
"""
|
|
|
|
must be implemented by child classes to be able to build a driver-based device selector by default.
|
|
|
|
return value must be the corresponding soapy driver identifier.
|
|
|
|
"""
|
2019-12-28 01:24:07 +01:00
|
|
|
|
2019-12-28 00:26:45 +01:00
|
|
|
@abstractmethod
|
2019-12-21 20:58:28 +01:00
|
|
|
def getDriver(self):
|
|
|
|
pass
|
|
|
|
|
2019-12-28 00:26:45 +01:00
|
|
|
def getEventNames(self):
|
2020-04-02 00:10:28 +02:00
|
|
|
return super().getEventNames() + list(self.getSoapySettingsMappings().keys())
|
2019-12-28 00:26:45 +01:00
|
|
|
|
2020-02-09 13:59:37 +01:00
|
|
|
def buildSoapyDeviceParameters(self, parsed, values):
|
|
|
|
"""
|
|
|
|
this method always attempts to inject a driver= part into the soapysdr query, depending on what connector was used.
|
|
|
|
this prevents the soapy_connector from using the wrong device in scenarios where there's no same-type sdrs.
|
|
|
|
"""
|
|
|
|
parsed = [v for v in parsed if "driver" not in v]
|
|
|
|
parsed += [{"driver": self.getDriver()}]
|
|
|
|
return parsed
|
2019-12-28 01:24:07 +01:00
|
|
|
|
2020-03-14 01:04:52 +01:00
|
|
|
def getSoapySettingsMappings(self):
|
|
|
|
return {}
|
|
|
|
|
|
|
|
def buildSoapySettings(self, values):
|
|
|
|
settings = {}
|
|
|
|
for k, v in self.getSoapySettingsMappings().items():
|
2020-03-14 01:21:30 +01:00
|
|
|
if k in values and values[k] is not None:
|
2020-03-14 23:07:23 +01:00
|
|
|
settings[v] = self.convertSoapySettingsValue(values[k])
|
2020-03-14 01:04:52 +01:00
|
|
|
return settings
|
|
|
|
|
2020-03-14 23:07:23 +01:00
|
|
|
def convertSoapySettingsValue(self, value):
|
|
|
|
if isinstance(value, bool):
|
|
|
|
return "true" if value else "false"
|
|
|
|
return value
|
|
|
|
|
2019-12-21 20:58:28 +01:00
|
|
|
def getCommandValues(self):
|
|
|
|
values = super().getCommandValues()
|
|
|
|
if "device" in values and values["device"] is not None:
|
2021-02-20 00:16:32 +01:00
|
|
|
parsed = SoapySettings.parse(values["device"])
|
2019-12-21 20:58:28 +01:00
|
|
|
else:
|
2020-02-09 13:59:37 +01:00
|
|
|
parsed = []
|
|
|
|
modified = self.buildSoapyDeviceParameters(parsed, values)
|
2021-02-20 00:16:32 +01:00
|
|
|
values["device"] = SoapySettings.encode(modified)
|
2020-03-14 01:04:52 +01:00
|
|
|
settings = ",".join(["{0}={1}".format(k, v) for k, v in self.buildSoapySettings(values).items()])
|
|
|
|
if len(settings):
|
|
|
|
values["soapy_settings"] = settings
|
2019-12-21 20:58:28 +01:00
|
|
|
return values
|
2020-03-14 01:04:52 +01:00
|
|
|
|
2020-12-30 17:18:46 +01:00
|
|
|
def onPropertyChange(self, changes):
|
2020-03-14 01:04:52 +01:00
|
|
|
mappings = self.getSoapySettingsMappings()
|
2020-12-30 17:18:46 +01:00
|
|
|
settings = {}
|
|
|
|
for prop, value in changes.items():
|
|
|
|
if prop in mappings.keys():
|
|
|
|
settings[mappings[prop]] = self.convertSoapySettingsValue(value)
|
|
|
|
if settings:
|
|
|
|
changes["settings"] = ",".join("{0}={1}".format(k, v) for k, v in settings.items())
|
|
|
|
super().onPropertyChange(changes)
|
2021-02-19 15:29:17 +01:00
|
|
|
|
|
|
|
|
|
|
|
class SoapyConnectorDeviceDescription(ConnectorDeviceDescription):
|
|
|
|
def getInputs(self) -> List[Input]:
|
2021-02-22 00:35:47 +01:00
|
|
|
return super().getInputs() + [
|
|
|
|
TextInput(
|
|
|
|
"device",
|
2021-04-02 21:43:46 +02:00
|
|
|
"Device identifier",
|
2021-02-22 00:35:47 +01:00
|
|
|
infotext='SoapySDR device identifier string (example: "serial=123456789")',
|
|
|
|
),
|
|
|
|
GainInput(
|
|
|
|
"rf_gain",
|
|
|
|
"Device Gain",
|
|
|
|
gain_stages=self.getGainStages(),
|
2021-02-23 23:24:30 +01:00
|
|
|
has_agc=self.hasAgc(),
|
2021-02-22 00:35:47 +01:00
|
|
|
),
|
2021-02-23 00:27:29 +01:00
|
|
|
TextInput("antenna", "Antenna"),
|
2021-02-22 00:35:47 +01:00
|
|
|
]
|
|
|
|
|
2021-03-24 23:17:50 +01:00
|
|
|
def getDeviceOptionalKeys(self):
|
|
|
|
return super().getDeviceOptionalKeys() + ["device", "rf_gain", "antenna"]
|
2021-02-19 18:18:25 +01:00
|
|
|
|
2021-02-23 18:32:23 +01:00
|
|
|
def getProfileOptionalKeys(self):
|
|
|
|
return super().getProfileOptionalKeys() + ["antenna"]
|
|
|
|
|
2021-02-19 18:18:25 +01:00
|
|
|
def getGainStages(self):
|
2021-02-20 00:16:32 +01:00
|
|
|
return None
|