aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp')
-rw-r--r--host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp200
1 files changed, 178 insertions, 22 deletions
diff --git a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
index b659902ae..cc51562aa 100644
--- a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
+++ b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
@@ -23,90 +23,246 @@
#include <uhd/utils/math.hpp>
#include <boost/algorithm/string.hpp>
#include <boost/make_shared.hpp>
-#include <boost/date_time/posix_time/posix_time_io.hpp>
+#include <boost/format.hpp>
+#include <sstream>
using namespace uhd;
using namespace uhd::usrp;
using namespace uhd::rfnoc;
static const size_t IO_MASTER_RADIO = 0;
+static const double MAGNESIUM_TICK_RATE = 125e6; // Hz
+static const double MAGNESIUM_RADIO_RATE = 125e6; // Hz
+static const double MAGNESIUM_CENTER_FREQ = 2.5e9; // Hz
+static const double MAGNESIUM_DEFAULT_GAIN = 0.0; // dB
+static const double MAGNESIUM_DEFAULT_BANDWIDTH = 40e6; // Hz TODO: fix
+static const size_t MAGNESIUM_NUM_TX_CHANS = 2;
+static const size_t MAGNESIUM_NUM_RX_CHANS = 2;
+
+std::string _get_which(direction_t dir, size_t chan)
+{
+ std::stringstream ss;
+ switch (dir)
+ {
+ case RX_DIRECTION:
+ ss << "RX";
+ break;
+ case TX_DIRECTION:
+ ss << "TX";
+ break;
+ default:
+ UHD_THROW_INVALID_CODE_PATH();
+ }
+
+ switch (chan)
+ {
+ case 0:
+ ss << "1";
+ break;
+ case 1:
+ ss << "2";
+ break;
+ default:
+ throw uhd::runtime_error("invalid channel number");
+ }
+
+ return ss.str();
+}
UHD_RFNOC_RADIO_BLOCK_CONSTRUCTOR(magnesium_radio_ctrl)
{
- std::cout << "magnesium_radio_ctrl_impl::ctor() " << std::endl;
+ UHD_LOG_TRACE("MAGNESIUM", "magnesium_radio_ctrl_impl::ctor() ");
+ _radio_slot = (get_block_id().get_block_count() == IO_MASTER_RADIO) ? "A" : "B";
+ _slot_prefix = (get_block_id().get_block_count() == IO_MASTER_RADIO) ? "db_0_" : "db_1_";
+ UHD_LOG_TRACE("MAGNESIUM", "Radio slot: " << _radio_slot);
+
+
+ const size_t num_rx_chans = get_output_ports().size();
+ UHD_ASSERT_THROW(num_rx_chans == MAGNESIUM_NUM_RX_CHANS);
+ const size_t num_tx_chans = get_input_ports().size();
+ UHD_ASSERT_THROW(num_tx_chans == MAGNESIUM_NUM_TX_CHANS);
+
+ UHD_LOG_TRACE("MAGNESIUM", "Setting tick rate to " << MAGNESIUM_TICK_RATE / 1e6 << " MHz");
+ radio_ctrl_impl::set_rate(MAGNESIUM_TICK_RATE);
+
+ for (size_t chan = 0; chan < num_rx_chans; chan++) {
+ radio_ctrl_impl::set_rx_frequency(MAGNESIUM_CENTER_FREQ, chan);
+ radio_ctrl_impl::set_rx_gain(MAGNESIUM_DEFAULT_GAIN, chan);
+ // TODO: fix antenna name
+ radio_ctrl_impl::set_rx_antenna(str(boost::format("RX%d") % (chan+1)), chan);
+ radio_ctrl_impl::set_rx_bandwidth(MAGNESIUM_DEFAULT_BANDWIDTH, chan);
+ }
+
+ for (size_t chan = 0; chan < num_tx_chans; chan++) {
+ radio_ctrl_impl::set_tx_frequency(MAGNESIUM_CENTER_FREQ, chan);
+ radio_ctrl_impl::set_tx_gain(MAGNESIUM_DEFAULT_GAIN, chan);
+ // TODO: fix antenna name
+ radio_ctrl_impl::set_tx_antenna(str(boost::format("TX%d") % (chan + 1)), chan);
+ }
+
+ /**** Set up legacy compatible properties ******************************/
+ // For use with multi_usrp APIs etc.
+ // For legacy prop tree init:
+ // TODO: determine DB number
+ fs_path fe_base = fs_path("dboards") / _radio_slot;
+ std::vector<std::string> fe({ "rx_frontends" , "tx_frontends" });
+ std::vector<std::string> ant({ "RX" , "TX" });
+
+ UHD_ASSERT_THROW(MAGNESIUM_NUM_RX_CHANS == MAGNESIUM_NUM_TX_CHANS);
+ for (size_t fe_idx = 0; fe_idx < fe.size(); ++fe_idx)
+ {
+ fs_path fe_direction_path = fe_base / fe[fe_idx];
+ for (size_t chan = 0; chan < MAGNESIUM_NUM_RX_CHANS; ++chan) {
+ fs_path fe_path = fe_direction_path / chan;
+ UHD_LOG_TRACE("MAGNESIUM", "Adding FE at " << fe_path);
+ _tree->create<std::string>(fe_path / "name")
+ .set(str(boost::format("Magnesium %s %d") % ant[fe_idx] % chan))
+ ;
+ _tree->create<std::string>(fe_path / "connection")
+ .set("IQ")
+ ;
+ // TODO: fix antenna name
+ _tree->create<std::string>(fe_path / "antenna" / "value")
+ .set(str(boost::format("%s%d") % ant[fe_idx] % (chan+1)))
+ .add_coerced_subscriber(boost::bind(&magnesium_radio_ctrl_impl::set_rx_antenna, this, _1, chan))
+ .set_publisher(boost::bind(&radio_ctrl_impl::get_rx_antenna, this, chan))
+ ;
+ _tree->create<std::vector<std::string>>(fe_path / "antenna" / "options")
+ .set(std::vector<std::string>(1, str(boost::format("%s%d") % ant[fe_idx] % (chan + 1))))
+ ;
+ _tree->create<double>(fe_path / "freq" / "value")
+ .set(MAGNESIUM_CENTER_FREQ)
+ .set_coercer(boost::bind(&magnesium_radio_ctrl_impl::set_rx_frequency, this, _1, chan))
+ .set_publisher(boost::bind(&radio_ctrl_impl::get_rx_frequency, this, chan))
+ ;
+ _tree->create<meta_range_t>(fe_path / "freq" / "range")
+ .set(meta_range_t(MAGNESIUM_CENTER_FREQ, MAGNESIUM_CENTER_FREQ))
+ ;
+ _tree->create<double>(fe_path / "gains" / "null" / "value")
+ .set(MAGNESIUM_DEFAULT_GAIN)
+ .set_coercer(boost::bind(&magnesium_radio_ctrl_impl::set_rx_gain, this, _1, chan))
+ .set_publisher(boost::bind(&radio_ctrl_impl::get_rx_gain, this, chan))
+ ;
+ _tree->create<meta_range_t>(fe_path / "gains" / "null" / "range")
+ .set(meta_range_t(MAGNESIUM_DEFAULT_GAIN, MAGNESIUM_DEFAULT_GAIN))
+ ;
+ _tree->create<double>(fe_path / "bandwidth" / "value")
+ .set(MAGNESIUM_DEFAULT_BANDWIDTH)
+ .set_coercer(boost::bind(&magnesium_radio_ctrl_impl::set_rx_bandwidth, this, _1, chan))
+ .set_publisher(boost::bind(&radio_ctrl_impl::get_rx_bandwidth, this, chan))
+ ;
+ _tree->create<meta_range_t>(fe_path / "bandwidth" / "range")
+ .set(meta_range_t(MAGNESIUM_DEFAULT_BANDWIDTH, MAGNESIUM_DEFAULT_BANDWIDTH))
+ ;
+ }
+ }
+
+ // TODO change codec names
+ _tree->create<int>("rx_codecs" / _radio_slot / "gains");
+ _tree->create<int>("tx_codecs" / _radio_slot / "gains");
+ _tree->create<std::string>("rx_codecs" / _radio_slot / "name").set("AD9361 Dual ADC");
+ _tree->create<std::string>("tx_codecs" / _radio_slot / "name").set("AD9361 Dual DAC");
+
+ // TODO remove this dirty hack
+ if (not _tree->exists("tick_rate"))
+ {
+ _tree->create<double>("tick_rate").set(MAGNESIUM_TICK_RATE);
+ }
}
magnesium_radio_ctrl_impl::~magnesium_radio_ctrl_impl()
{
+ UHD_LOG_TRACE("MAGNESIUM", "magnesium_radio_ctrl_impl::dtor() ");
}
double magnesium_radio_ctrl_impl::set_rate(double rate)
{
- return 0;
+ // TODO: implement
+ if (rate != get_rate()) {
+ UHD_LOG_WARNING("MAGNESIUM", "Attempting to set sampling rate to invalid value " << rate);
+ }
+ return get_rate();
}
void magnesium_radio_ctrl_impl::set_tx_antenna(const std::string &ant, const size_t chan)
{
+ // TODO: implement
+ UHD_LOG_WARNING("MAGNESIUM", "Ignoring attempt to set Tx antenna");
+ // CPLD control?
}
void magnesium_radio_ctrl_impl::set_rx_antenna(const std::string &ant, const size_t chan)
{
+ // TODO: implement
+ UHD_LOG_WARNING("MAGNESIUM", "Ignoring attempt to set Rx antenna");
+ // CPLD control?
}
double magnesium_radio_ctrl_impl::set_tx_frequency(const double freq, const size_t chan)
{
- return 0;
+ return _set_freq(freq, chan, TX_DIRECTION);
}
double magnesium_radio_ctrl_impl::set_rx_frequency(const double freq, const size_t chan)
{
- return 0;
+ return _set_freq(freq, chan, RX_DIRECTION);
}
double magnesium_radio_ctrl_impl::set_rx_bandwidth(const double bandwidth, const size_t chan)
{
- return 0;
+ // TODO: implement
+ return get_rx_bandwidth(chan);
}
-double magnesium_radio_ctrl_impl::get_tx_frequency(const size_t chan)
+double magnesium_radio_ctrl_impl::set_tx_gain(const double gain, const size_t chan)
{
- return 0;
+ return _set_gain(gain, chan, TX_DIRECTION);
}
-double magnesium_radio_ctrl_impl::get_rx_frequency(const size_t chan)
+double magnesium_radio_ctrl_impl::set_rx_gain(const double gain, const size_t chan)
{
- return 0;
+ return _set_gain(gain, chan, RX_DIRECTION);
}
-double magnesium_radio_ctrl_impl::get_rx_bandwidth(const size_t chan)
+size_t magnesium_radio_ctrl_impl::get_chan_from_dboard_fe(const std::string &fe, const direction_t dir)
{
- return 0;
+ UHD_LOG_TRACE("MAGNESIUM", "get_chan_from_dboard_fe " << fe << " returns " << boost::lexical_cast<size_t>(fe));
+ return boost::lexical_cast<size_t>(fe);
}
-double magnesium_radio_ctrl_impl::set_tx_gain(const double gain, const size_t chan)
+std::string magnesium_radio_ctrl_impl::get_dboard_fe_from_chan(const size_t chan, const direction_t dir)
{
- return 0;
+ UHD_LOG_TRACE("MAGNESIUM", "get_dboard_fe_from_chan " << chan << " returns " << std::to_string(chan));
+ return std::to_string(chan);
}
-double magnesium_radio_ctrl_impl::set_rx_gain(const double gain, const size_t chan)
+double magnesium_radio_ctrl_impl::get_output_samp_rate(size_t port)
{
- return 0;
+ return 125e6;
}
-size_t magnesium_radio_ctrl_impl::get_chan_from_dboard_fe(const std::string &fe, const direction_t dir)
+void magnesium_radio_ctrl_impl::set_rpc_client(
+ uhd::rpc_client::sptr rpcc,
+ const uhd::device_addr_t &block_args
+)
{
- return 0;
+ _rpcc = rpcc;
+ _block_args = block_args;
}
-std::string magnesium_radio_ctrl_impl::get_dboard_fe_from_chan(const size_t chan, const direction_t dir)
+double magnesium_radio_ctrl_impl::_set_freq(const double freq, const size_t chan, const direction_t dir)
{
- return "";
+ auto which = _get_which(dir, chan);
+ UHD_LOG_TRACE("MAGNESIUM", "calling " << _slot_prefix << "set_freq on " << which << " with " << freq);
+ return _rpcc->request_with_token<double>(_slot_prefix + "set_freq", which, freq, false);
}
-double magnesium_radio_ctrl_impl::get_output_samp_rate(size_t port)
+double magnesium_radio_ctrl_impl::_set_gain(const double gain, const size_t chan, const direction_t dir)
{
- return 0;
+ auto which = _get_which(dir, chan);
+ UHD_LOG_TRACE("MAGNESIUM", "calling " << _slot_prefix << "set_gain on " << which << " with " << gain);
+ return _rpcc->request_with_token<double>(_slot_prefix + "set_gain", which, gain);
}
UHD_RFNOC_BLOCK_REGISTER(magnesium_radio_ctrl, "MagnesiumRadio");