diff options
Diffstat (limited to 'host/lib')
| -rw-r--r-- | host/lib/usrp/mpmd/mpmd_impl.cpp | 154 | 
1 files changed, 103 insertions, 51 deletions
| diff --git a/host/lib/usrp/mpmd/mpmd_impl.cpp b/host/lib/usrp/mpmd/mpmd_impl.cpp index 8deac2743..d8d21a464 100644 --- a/host/lib/usrp/mpmd/mpmd_impl.cpp +++ b/host/lib/usrp/mpmd/mpmd_impl.cpp @@ -50,6 +50,8 @@ namespace {      //! Most pessimistic time for a CHDR query to go to device and back      const double MPMD_CHDR_MAX_RTT = 0.02; +    const std::string MPMD_MGMT_ADDR_KEY = "mgmt_addr"; +      /*************************************************************************       * Helper functions       ************************************************************************/ @@ -343,15 +345,21 @@ mpmd_mboard_impl::uptr mpmd_impl::setup_mb(      const size_t mb_index,      const uhd::device_addr_t& device_args  ) { +    const std::string rpc_addr = +        device_args.get(MPMD_MGMT_ADDR_KEY, device_args.get("addr", ""));      UHD_LOGGER_DEBUG("MPMD")          << "Initializing mboard " << mb_index -        << ". Device args: " << device_args.to_string() +        << ". Device args: `" << device_args.to_string() +        << "'. RPC address: " << rpc_addr      ; -    auto mb = mpmd_mboard_impl::make( -        device_args, -        device_args["addr"] -    ); +    if (rpc_addr.empty()) { +        UHD_LOG_ERROR("MPMD", +            "Could not determine RPC address from device args: " +            << device_args.to_string()); +        throw uhd::runtime_error("Could not determine device RPC address."); +    } +    auto mb = mpmd_mboard_impl::make(device_args, rpc_addr);      if (device_args.has_key("skip_init")) {          return mb; @@ -460,11 +468,20 @@ size_t mpmd_impl::allocate_xbar_local_addr()   ****************************************************************************/  device_addrs_t mpmd_find_with_addr(const device_addr_t& hint_)  { +    device_addrs_t addrs; +    const std::string query_addr = +        hint_.get(MPMD_MGMT_ADDR_KEY, hint_.get("addr", "")); +    if (query_addr.empty()) { +        UHD_LOG_WARNING("MPMD FIND", +            "Was supposed to find with addr, but no addr was given (hint was: " +            << hint_.to_string() << ")"); +        return addrs; +    } +      transport::udp_simple::sptr comm = transport::udp_simple::make_broadcast( -        hint_["addr"], std::to_string(MPM_DISCOVERY_PORT)); +        query_addr, std::to_string(MPM_DISCOVERY_PORT));      comm->send(          boost::asio::buffer(&MPM_DISCOVERY_CMD, sizeof(MPM_DISCOVERY_CMD))); -    device_addrs_t addrs;      while (true) {          char buff[4096] = {};          const size_t nbytes = comm->recv( // TODO make sure we don't buf overflow @@ -483,7 +500,8 @@ device_addrs_t mpmd_find_with_addr(const device_addr_t& hint_)          if (result.empty()) {              continue;          } -        // who else is reposending to our request !? +        // Verify we didn't receive something other than an MPM discovery +        // response          if (result[0] != "USRP-MPM") {              continue;          } @@ -502,7 +520,10 @@ device_addrs_t mpmd_find_with_addr(const device_addr_t& hint_)              continue;          }          device_addr_t new_addr; -        new_addr["addr"] = recv_addr; +        new_addr[MPMD_MGMT_ADDR_KEY] = recv_addr; +        if (not new_addr.has_key("addr")) { +            new_addr["addr"] = recv_addr; +        }          new_addr["type"] = "mpmd"; // hwd will overwrite this          // remove ident string and put other informations into device_args dict          result.erase(result.begin()); @@ -522,6 +543,8 @@ device_addrs_t mpmd_find_with_addr(const device_addr_t& hint_)              and (not hint_.has_key("type")    or hint_["type"]    == new_addr["type"])              and (not hint_.has_key("product") or hint_["product"] == new_addr["product"])          ){ +            UHD_LOG_TRACE("MPMD FIND", +                "Found device that matches hints: " << new_addr.to_string());              addrs.push_back(new_addr);          } else {              UHD_LOG_DEBUG("MPMD FIND", @@ -532,55 +555,48 @@ device_addrs_t mpmd_find_with_addr(const device_addr_t& hint_)      return addrs;  }; -device_addrs_t mpmd_find(const device_addr_t& hint_) -{ -    // handle cases: -    // -    //  - empty hint -    //  - multiple addrs -    //  - single addr -    device_addrs_t hints = separate_device_addr(hint_); -    // either hints has: -    // multiple entries -    //   -> search for multiple devices and join them back into one -    //   device_addr_t -    // one entry with addr: -    //   -> search for one device with this addr -    // one -    // multiple addrs -    if (hints.size() > 1) { -        device_addrs_t found_devices; -        found_devices.reserve(hints.size()); -        for (const auto& hint : hints) { -            if (not hint.has_key("addr")) { // maybe allow other attributes as well -                return device_addrs_t(); -            } -            device_addrs_t reply_addrs = mpmd_find_with_addr(hint); -            if (reply_addrs.size() > 1) { -                throw uhd::value_error( -                    str(boost::format("Could not resolve device hint \"%s\" to " -                                      "a single device.") % -                        hint.to_string())); -            } else if (reply_addrs.empty()) { -                return device_addrs_t(); -            } -            found_devices.push_back(reply_addrs[0]); + +// Implements scenario 1) (see below) +device_addrs_t mpmd_find_with_addrs(const device_addrs_t& hints) +{ +    UHD_LOG_TRACE("MPMD FIND", "Finding with addrs."); +    device_addrs_t found_devices; +    found_devices.reserve(hints.size()); +    for (const auto& hint : hints) { +        if (not (hint.has_key("addr") or hint.has_key(MPMD_MGMT_ADDR_KEY))) { +            UHD_LOG_DEBUG("MPMD FIND", +                "No address given in hint " << hint.to_string()); +            continue;          } -        return device_addrs_t(1, combine_device_addrs(found_devices)); +        device_addrs_t reply_addrs = mpmd_find_with_addr(hint); +        if (reply_addrs.size() > 1) { +            UHD_LOG_ERROR("MPMD", +                "Could not resolve device hint \"" << hint.to_string() +                << "\" to a unique device."); +            continue; +        } else if (reply_addrs.empty()) { +            continue; +        } +        UHD_LOG_TRACE("MPMD FIND", +            "Device responded: " << reply_addrs[0].to_string()); +        found_devices.push_back(reply_addrs[0]);      } -    hints.resize(1); -    device_addr_t hint = hints[0]; -    device_addrs_t addrs; - -    if (hint.has_key("addr")) { -        // is this safe? -        return mpmd_find_with_addr(hint); +    if (found_devices.size() == 1) { +        return found_devices; +    } else { +        return device_addrs_t(1, combine_device_addrs(found_devices));      } +} +device_addrs_t mpmd_find_with_bcast(const device_addr_t& hint) +{ +    device_addrs_t addrs; +    UHD_LOG_TRACE("MPMD FIND", +            "Broadcasting on all available interfaces to find MPM devices.");      for (const transport::if_addrs_t& if_addr : transport::get_if_addrs()) {          device_addr_t new_hint = hint; -        new_hint["addr"] = if_addr.bcast; +        new_hint[MPMD_MGMT_ADDR_KEY] = if_addr.bcast;          device_addrs_t reply_addrs = mpmd_find_with_addr(new_hint);          addrs.insert(addrs.begin(), reply_addrs.begin(), reply_addrs.end()); @@ -588,6 +604,42 @@ device_addrs_t mpmd_find(const device_addr_t& hint_)      return addrs;  } +/*! Find MPM devices based on a hint + * + * There are two scenarios: + * + * 1) an addr or mgmt_addr was defined + * + * In this case, we will go through all the addrs. If they point to a device, + * we will then compare the other attributes (serial, etc.). If they match, + * the device goes into a list. + * + * 2) No addrs were defined + * + * In this case, we do a broadcast ping to see if any devices respond. After + * that, we do the same matching. + * + */ +device_addrs_t mpmd_find(const device_addr_t& hint_) +{ +    device_addrs_t hints = separate_device_addr(hint_); +    UHD_LOG_TRACE("MPMD FIND", +        "Finding with " << hints.size() << " different hint(s)."); + +    // Scenario 1): User gave us at least one address +    if (not hints.empty() and +            (hints[0].has_key("addr") or +             hints[0].has_key(MPMD_MGMT_ADDR_KEY))) { +        return mpmd_find_with_addrs(hints); +    } + +    // Scenario 2): User gave us no address, and we need to broadcast +    if (hints.empty()) { +        hints.resize(1); +    } +    return mpmd_find_with_bcast(hints[0]); +} +  static device::sptr mpmd_make(const device_addr_t& device_args)  {      return device::sptr(boost::make_shared<mpmd_impl>(device_args)); | 
