diff options
author | Martin Braun <martin.braun@ettus.com> | 2019-11-04 17:16:22 -0800 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 12:21:33 -0800 |
commit | b7488af14e6bf23eb52b815d30c3f37836834387 (patch) | |
tree | 3eb7ef57d00066ce5d4d6db41518a94dbcd826ca /host/lib | |
parent | 77cb9836ab8018a8807e30d54ea4b91d63ff6e3d (diff) | |
download | uhd-b7488af14e6bf23eb52b815d30c3f37836834387.tar.gz uhd-b7488af14e6bf23eb52b815d30c3f37836834387.tar.bz2 uhd-b7488af14e6bf23eb52b815d30c3f37836834387.zip |
multi_usrp: Fix GPIO bank selection
When calling a multi_usrp object like this:
usrp->set_gpio_attr("TXB", "CTRL", 0xFFFF);
Previously, it would only be able to address daughterboard A. Now, there
is a full backward-compatible solution (compatible with 3.15), that will
address either daughterboard's GPIOs.
Diffstat (limited to 'host/lib')
-rw-r--r-- | host/lib/usrp/multi_usrp_rfnoc.cpp | 79 |
1 files changed, 68 insertions, 11 deletions
diff --git a/host/lib/usrp/multi_usrp_rfnoc.cpp b/host/lib/usrp/multi_usrp_rfnoc.cpp index d6f16824d..5242da965 100644 --- a/host/lib/usrp/multi_usrp_rfnoc.cpp +++ b/host/lib/usrp/multi_usrp_rfnoc.cpp @@ -1900,19 +1900,73 @@ public: /******************************************************************* * GPIO methods ******************************************************************/ - /* Helper function to get the radio block controller which controls the GPIOs for a - * given motherboard + /*! Helper function to identify the radio and the bank on that radio. + * + * Background: Historically, multi_usrp has made up its own GPIO bank names, + * unrelated to the radios. Now, we need to be a bit more clever to get that + * legacy behaviour to work. + * + * Here's the algorithm: + * - If the bank ends with 'A' or 'B' we'll use that to identify the radio + * - Otherwise, we'll pick the first radio + * - If the bank ends with 'A' or 'B' we strip that suffix + * + * The returned radio will now have a GPIO bank with the returned name. */ - uhd::rfnoc::radio_control::sptr _get_gpio_radio(const size_t mboard) - { - // We assume that the first radio block on each board controls the GPIO banks - return _graph->get_block<uhd::rfnoc::radio_control>( - block_id_t(mboard, "Radio", 0)); + std::pair<uhd::rfnoc::radio_control::sptr, std::string> _get_gpio_radio_bank( + const std::string& bank, const size_t mboard) + { + UHD_ASSERT_THROW(!bank.empty()); + char suffix = bank[bank.size() - 1]; + std::string slot_name; + if (suffix == 'A' || suffix == 'a') { + slot_name = "A"; + } + if (suffix == 'B' || suffix == 'b') { + slot_name = "B"; + } + + uhd::rfnoc::radio_control::sptr radio = [bank, mboard, slot_name, this]() { + auto radio_blocks = _graph->find_blocks<uhd::rfnoc::radio_control>( + std::to_string(mboard) + "/Radio"); + for (auto radio_id : radio_blocks) { + auto radio = _graph->get_block<uhd::rfnoc::radio_control>(radio_id); + if (slot_name.empty() || radio->get_slot_name() == slot_name) { + return radio; + } + } + throw uhd::runtime_error(std::string("Could not match GPIO bank ") + bank + + " to a radio block controller."); + }(); + + const std::string normalized_bank = [radio, bank]() { + auto radio_banks = radio->get_gpio_banks(); + for (auto& radio_bank : radio_banks) { + if (bank.find(radio_bank) == 0) { + return radio_bank; + } + } + throw uhd::runtime_error(std::string("Could not match GPIO bank ") + bank + + " to radio " + radio->get_unique_id()); + }(); + + return {radio, normalized_bank}; } std::vector<std::string> get_gpio_banks(const size_t mboard) { - return _get_gpio_radio(mboard)->get_gpio_banks(); + auto radio_blocks = _graph->find_blocks<uhd::rfnoc::radio_control>( + std::to_string(mboard) + "/Radio"); + std::vector<std::string> gpio_banks; + for (auto radio_id : radio_blocks) { + auto radio = _graph->get_block<uhd::rfnoc::radio_control>(radio_id); + auto radio_banks = radio->get_gpio_banks(); + for (const auto& bank : radio_banks) { + gpio_banks.push_back(bank + radio->get_slot_name()); + } + } + + return gpio_banks; } void set_gpio_attr(const std::string& bank, @@ -1921,15 +1975,18 @@ public: const uint32_t mask = 0xffffffff, const size_t mboard = 0) { - const uint32_t current = get_gpio_attr(bank, attr, mboard); + auto radio_bank_pair = _get_gpio_radio_bank(bank, mboard); + const uint32_t current = + radio_bank_pair.first->get_gpio_attr(radio_bank_pair.second, attr); const uint32_t new_value = (current & ~mask) | (value & mask); - return _get_gpio_radio(mboard)->set_gpio_attr(bank, attr, new_value); + radio_bank_pair.first->set_gpio_attr(radio_bank_pair.second, attr, new_value); } uint32_t get_gpio_attr( const std::string& bank, const std::string& attr, const size_t mboard) { - return _get_gpio_radio(mboard)->get_gpio_attr(bank, attr); + auto radio_bank_pair = _get_gpio_radio_bank(bank, mboard); + return radio_bank_pair.first->get_gpio_attr(radio_bank_pair.second, attr); } std::vector<std::string> get_gpio_srcs(const std::string& bank, const size_t mboard) |