From 77adcffe0689f120fee71854ec0f627a32991f79 Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Tue, 5 Dec 2017 11:17:03 -0800 Subject: multi_usrp: Expose APIs for TX LO controls Reviewed-By: Ashish Chaudhari Reviewed-By: Derek Kozel --- host/lib/usrp/multi_usrp.cpp | 220 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 220 insertions(+) (limited to 'host/lib') diff --git a/host/lib/usrp/multi_usrp.cpp b/host/lib/usrp/multi_usrp.cpp index 3dcfe3bd9..d7d925f3c 100644 --- a/host/lib/usrp/multi_usrp.cpp +++ b/host/lib/usrp/multi_usrp.cpp @@ -870,6 +870,9 @@ public: return _tree->access(rx_rf_fe_root(chan) / "freq" / "range").get(); } + /************************************************************************** + * LO controls + *************************************************************************/ std::vector get_rx_lo_names(size_t chan = 0){ std::vector lo_names; if (_tree->exists(rx_rf_fe_root(chan) / "los")) { @@ -1035,6 +1038,223 @@ public: } } + std::vector get_tx_lo_names(const size_t chan = 0){ + std::vector lo_names; + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + for (const std::string &name : _tree->list(tx_rf_fe_root(chan) / "los")) { + lo_names.push_back(name); + } + } + return lo_names; + } + + void set_tx_lo_source( + const std::string &src, + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (name == ALL_LOS) { + if (_tree->exists(tx_rf_fe_root(chan) / "los" / ALL_LOS)) { + // Special value ALL_LOS support atomically sets the source + // for all LOs + _tree->access( + tx_rf_fe_root(chan) / "los" / ALL_LOS / + "source" / "value" + ).set(src); + } else { + for (const auto &n : _tree->list(tx_rf_fe_root(chan) / "los")) { + this->set_tx_lo_source(src, n, chan); + } + } + } else { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + _tree->access( + tx_rf_fe_root(chan) / "los" / name / "source" / + "value" + ).set(src); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } + } else { + throw uhd::runtime_error("This device does not support manual " + "configuration of LOs"); + } + } + + const std::string get_tx_lo_source( + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + return _tree->access( + tx_rf_fe_root(chan) / "los" / name / "source" / "value" + ).get(); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } else { + // If the daughterboard doesn't expose its LO(s) then it can only + // be internal + return "internal"; + } + } + + std::vector get_tx_lo_sources( + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (name == ALL_LOS) { + if (_tree->exists(tx_rf_fe_root(chan) / "los" / ALL_LOS)) { + // Special value ALL_LOS support atomically sets the source + // for all LOs + return _tree->access>( + tx_rf_fe_root(chan) / "los" / ALL_LOS / + "source" / "options" + ).get(); + } else { + return std::vector(); + } + } else { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + return _tree->access< std::vector >(tx_rf_fe_root(chan) / "los" / name / "source" / "options").get(); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } + } else { + // If the daughterboard doesn't expose its LO(s) then it can only + // be internal + return std::vector(1, "internal"); + } + } + + void set_tx_lo_export_enabled( + const bool enabled, + const std::string &name = ALL_LOS, + const size_t chan=0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (name == ALL_LOS) { + if (_tree->exists(tx_rf_fe_root(chan) / "los" / ALL_LOS)) { + //Special value ALL_LOS support atomically sets the source for all LOs + _tree->access(tx_rf_fe_root(chan) / "los" / ALL_LOS / "export").set(enabled); + } else { + for(const std::string &n: _tree->list(tx_rf_fe_root(chan) / "los")) { + this->set_tx_lo_export_enabled(enabled, n, chan); + } + } + } else { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + _tree->access(tx_rf_fe_root(chan) / "los" / name / "export").set(enabled); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } + } else { + throw uhd::runtime_error("This device does not support manual configuration of LOs"); + } + } + + bool get_tx_lo_export_enabled( + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + return _tree->access( + tx_rf_fe_root(chan) / "los" / name / "export" + ).get(); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } else { + // If the daughterboard doesn't expose its LO(s), assume it cannot + // export + return false; + } + } + + double set_tx_lo_freq( + const double freq, + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (name == ALL_LOS) { + throw uhd::runtime_error("LO frequency must be set for each " + "stage individually"); + } else { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + return _tree->access( + tx_rf_fe_root(chan) / "los" / name / "freq" / "value" + ).set(freq).get(); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } + } else { + throw uhd::runtime_error("This device does not support manual " + "configuration of LOs"); + } + } + + double get_tx_lo_freq( + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (name == ALL_LOS) { + throw uhd::runtime_error("LO frequency must be retrieved for " + "each stage individually"); + } else { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + return _tree->access(tx_rf_fe_root(chan) / "los" / name / "freq" / "value").get(); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } + } else { + // Return actual RF frequency if the daughterboard doesn't expose + // its LO(s) + return _tree->access( + tx_rf_fe_root(chan) / "freq" /" value" + ).get(); + } + } + + freq_range_t get_tx_lo_freq_range( + const std::string &name = ALL_LOS, + const size_t chan = 0 + ) { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + if (name == ALL_LOS) { + throw uhd::runtime_error("LO frequency range must be retrieved " + "for each stage individually"); + } else { + if (_tree->exists(tx_rf_fe_root(chan) / "los")) { + return _tree->access( + tx_rf_fe_root(chan) / "los" / name / "freq" / "range" + ).get(); + } else { + throw uhd::runtime_error("Could not find LO stage " + name); + } + } + } else { + // Return the actual RF range if the daughterboard doesn't expose + // its LO(s) + return _tree->access( + tx_rf_fe_root(chan) / "freq" / "range" + ).get(); + } + } + + /************************************************************************** + * Gain control + *************************************************************************/ void set_rx_gain(double gain, const std::string &name, size_t chan){ /* Check if any AGC mode is enable and if so warn the user */ if (chan != ALL_CHANS) { -- cgit v1.2.3