// // Copyright 2017 Ettus Research (National Instruments) // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // #include "mpmd_impl.hpp" #include #include #include #include namespace { /************************************************************************* * Local constants ************************************************************************/ //! Time between reclaims (ms) const size_t MPMD_RECLAIM_INTERVAL_MS = 1000; //! Default timeout value for the init() RPC call (ms) const size_t MPMD_DEFAULT_INIT_TIMEOUT = 120000; //! Default timeout value for RPC calls (ms) const size_t MPMD_DEFAULT_RPC_TIMEOUT = 2000; //! Default session ID (MPM will recognize a session by this name) const std::string MPMD_DEFAULT_SESSION_ID = "UHD"; /************************************************************************* * Helper functions ************************************************************************/ void init_device( uhd::rpc_client::sptr rpc, const uhd::device_addr_t mb_args ) { // TODO maybe put this somewhere else? const std::set key_blacklist{ "serial", "claimed", "type", "rev", "addr" }; std::map mpm_device_args; for (const auto &key : mb_args.keys()) { if (not key_blacklist.count(key)) { mpm_device_args[key] = mb_args[key]; } } rpc->set_timeout(mb_args.cast( "init_timeout", MPMD_DEFAULT_INIT_TIMEOUT )); if (not rpc->request_with_token("init", mpm_device_args)) { throw uhd::runtime_error("Failed to initialize device."); } rpc->set_timeout(mb_args.cast( "rpc_timeout", MPMD_DEFAULT_RPC_TIMEOUT )); } } using namespace uhd; /***************************************************************************** * Structors ****************************************************************************/ mpmd_mboard_impl::mpmd_mboard_impl( const device_addr_t &mb_args_, const std::string& rpc_server_addr ) : mb_args(mb_args_) , rpc(uhd::rpc_client::make( rpc_server_addr, MPM_RPC_PORT, MPM_RPC_GET_LAST_ERROR_CMD)) { UHD_LOGGER_TRACE("MPMD") << "Initializing mboard, connecting to RPC server address: " << rpc_server_addr << " mboard args: " << mb_args.to_string() ; _claimer_task = claim_device_and_make_task(rpc, mb_args); // No one else can now claim the device. if (mb_args_.has_key("skip_init")) { UHD_LOG_DEBUG("MPMD", "Claimed device, but skipped init."); return; } init_device(rpc, mb_args); // RFNoC block clocks are now on. Noc-IDs can be read back. auto device_info_dict = rpc->request("get_device_info"); for (const auto &info_pair : device_info_dict) { device_info[info_pair.first] = info_pair.second; } UHD_LOGGER_TRACE("MPMD") << "MPM reports device info: " << device_info.to_string(); auto dboards_info = rpc->request>("get_dboard_info"); UHD_ASSERT_THROW(this->dboard_info.size() == 0); for (const auto &dboard_info_dict : dboards_info) { uhd::device_addr_t this_db_info; for (const auto &info_pair : dboard_info_dict) { this_db_info[info_pair.first] = info_pair.second; } UHD_LOGGER_TRACE("MPMD") << "MPM reports dboard info for slot " << this->dboard_info.size() << ": " << this_db_info.to_string(); this->dboard_info.push_back(this_db_info); } // Initialize properties this->num_xbars = rpc->request("get_num_xbars"); // xbar_local_addrs is not yet valid after this! this->xbar_local_addrs.resize(this->num_xbars, 0xFF); // std::vector data_ifaces = // rpc.call>("get_interfaces", rpc_token); // discover path to device and tell MPM our MAC address seen at the data // interfaces // move this into make_transport //for (const auto& iface : data_ifaces) { //std::vector addrs = rpc.call>( //"get_interface_addrs", _rpc_token, iface); //for (const auto& iface_addr : addrs) { //if (rpc_client(iface_addr, MPM_RPC_PORT) //.call("probe_interface", _rpc_token)) { //data_interfaces.emplace(iface, iface_addr); //break; //} //} //} } mpmd_mboard_impl::~mpmd_mboard_impl() { UHD_SAFE_CALL( if (not rpc->request_with_token("unclaim")) { UHD_LOG_WARNING("MPMD", "Failure to ack unclaim!"); } ); } /***************************************************************************** * API ****************************************************************************/ uhd::sid_t mpmd_mboard_impl::allocate_sid( const uint16_t port, const uhd::sid_t address, const uint32_t xbar_src_addr, const uint32_t xbar_src_port, const uint32_t dst_addr ) { const auto sid = rpc->request_with_token( "allocate_sid", port, address.get(), xbar_src_addr, xbar_src_port, dst_addr ); return uhd::sid_t(sid); } void mpmd_mboard_impl::set_xbar_local_addr( const size_t xbar_index, const size_t local_addr ) { UHD_ASSERT_THROW(rpc->request_with_token("set_xbar_local_addr", xbar_index, local_addr)); UHD_ASSERT_THROW(xbar_index < xbar_local_addrs.size()); xbar_local_addrs.at(xbar_index) = local_addr; } /***************************************************************************** * Private methods ****************************************************************************/ bool mpmd_mboard_impl::claim() { try { return rpc->request_with_token("reclaim"); } catch (...) { UHD_LOG_WARNING("MPMD", "Reclaim failed. Exiting claimer loop."); return false; } } uhd::task::sptr mpmd_mboard_impl::claim_device_and_make_task( uhd::rpc_client::sptr rpc, const uhd::device_addr_t mb_args ) { auto rpc_token = rpc->request("claim", mb_args.get("session_id", MPMD_DEFAULT_SESSION_ID) ); if (rpc_token.empty()) { throw uhd::value_error("mpmd device claiming failed!"); } UHD_LOG_TRACE("MPMD", "Received claim token " << rpc_token); rpc->set_token(rpc_token); return uhd::task::make([this] { if (not this->claim()) { throw uhd::value_error("mpmd device reclaiming loop failed!"); }; std::this_thread::sleep_for( std::chrono::milliseconds(MPMD_RECLAIM_INTERVAL_MS) ); }); } /***************************************************************************** * Factory ****************************************************************************/ mpmd_mboard_impl::uptr mpmd_mboard_impl::make( const uhd::device_addr_t &mb_args, const std::string& addr ) { mpmd_mboard_impl::uptr mb = mpmd_mboard_impl::uptr(new mpmd_mboard_impl(mb_args, addr)); // implicit move return mb; }