// // Copyright 2010 Ettus Research LLC // // 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 #include #include #include #include "usrp2_impl.hpp" #include "usrp2_dboard_interface.hpp" using namespace uhd; using namespace uhd::usrp; /*********************************************************************** * USRP2 DBoard Wrapper **********************************************************************/ usrp2_dboard::usrp2_dboard(uhd::usrp::dboard::manager::sptr mgr, type_t type){ _mgr = mgr; _type = type; } usrp2_dboard::~usrp2_dboard(void){ /* NOP */ } void usrp2_dboard::get(const wax::obj &key_, wax::obj &val){ wax::obj key; std::string name; boost::tie(key, name) = extract_named_prop(key_); //handle the get request conditioned on the key switch(wax::cast(key)){ case DBOARD_PROP_NAME: val = std::string("usrp2 dboard"); return; case DBOARD_PROP_SUBDEV: switch(_type){ case TYPE_RX: val = _mgr->get_rx_subdev(name); return; case TYPE_TX: val = _mgr->get_tx_subdev(name); return; } case DBOARD_PROP_SUBDEV_NAMES: switch(_type){ case TYPE_RX: val = _mgr->get_rx_subdev_names(); return; case TYPE_TX: val = _mgr->get_tx_subdev_names(); return; } case DBOARD_PROP_CODEC: throw std::runtime_error("unhandled prop in usrp2 dboard"); } } void usrp2_dboard::set(const wax::obj &, const wax::obj &){ throw std::runtime_error("Cannot set on usrp2 dboard"); } /*********************************************************************** * USRP2 Implementation **********************************************************************/ usrp2_impl::usrp2_impl( uhd::transport::udp::sptr ctrl_transport, uhd::transport::udp::sptr data_transport ){ _ctrl_transport = ctrl_transport; _data_transport = data_transport; //grab the dboard ids over the control line usrp2_ctrl_data_t out_data; out_data.id = htonl(USRP2_CTRL_ID_GIVE_ME_YOUR_DBOARD_IDS_BRO); usrp2_ctrl_data_t in_data = ctrl_send_and_recv(out_data); ASSERT_THROW(htonl(in_data.id) == USRP2_CTRL_ID_THESE_ARE_MY_DBOARD_IDS_DUDE); std::cout << boost::format("rx id 0x%.2x, tx id 0x%.2x") % ntohs(in_data.data.dboard_ids.rx_id) % ntohs(in_data.data.dboard_ids.tx_id) << std::endl; //extract the dboard ids an convert them to enums dboard::dboard_id_t rx_dboard_id = static_cast( ntohs(in_data.data.dboard_ids.rx_id) ); dboard::dboard_id_t tx_dboard_id = static_cast( ntohs(in_data.data.dboard_ids.tx_id) ); //create a new dboard interface and manager dboard::interface::sptr dboard_interface( new usrp2_dboard_interface(this) ); dboard::manager::sptr dboard_manager( new dboard::manager(rx_dboard_id, tx_dboard_id, dboard_interface) ); //load dboards _rx_dboards[""] = usrp2_dboard::sptr(new usrp2_dboard(dboard_manager, usrp2_dboard::TYPE_RX)); _tx_dboards[""] = usrp2_dboard::sptr(new usrp2_dboard(dboard_manager, usrp2_dboard::TYPE_TX)); //TOD load dsps } usrp2_impl::~usrp2_impl(void){ /* NOP */ } /*********************************************************************** * Control Send/Recv **********************************************************************/ usrp2_ctrl_data_t usrp2_impl::ctrl_send_and_recv(const usrp2_ctrl_data_t &out_data){ boost::mutex::scoped_lock lock(_ctrl_mutex); //fill in the seq number and send usrp2_ctrl_data_t out_copy = out_data; out_copy.seq = htonl(++_ctrl_seq_num); _ctrl_transport->send(boost::asio::buffer(&out_copy, sizeof(usrp2_ctrl_data_t))); //loop and recieve until the time is up size_t num_timeouts = 0; while(true){ uhd::shared_iovec iov = _ctrl_transport->recv(); if (iov.len < sizeof(usrp2_ctrl_data_t)){ //sleep a little so we dont burn cpu if (num_timeouts++ > 50) break; boost::this_thread::sleep(boost::posix_time::milliseconds(1)); }else{ //handle the received data usrp2_ctrl_data_t in_data = *reinterpret_cast(iov.base); if (ntohl(in_data.seq) == _ctrl_seq_num){ return in_data; } //didnt get seq, continue on... } } throw std::runtime_error("usrp2 no control response"); } /*********************************************************************** * Get Properties **********************************************************************/ void usrp2_impl::get(const wax::obj &key_, wax::obj &val){ wax::obj key; std::string name; boost::tie(key, name) = extract_named_prop(key_); //handle the get request conditioned on the key switch(wax::cast(key)){ case MBOARD_PROP_NAME: val = std::string("usrp2 mboard"); return; case MBOARD_PROP_OTHERS: val = prop_names_t(); //empty other props return; case MBOARD_PROP_RX_DBOARD: case MBOARD_PROP_RX_DBOARD_NAMES: case MBOARD_PROP_TX_DBOARD: case MBOARD_PROP_TX_DBOARD_NAMES: case MBOARD_PROP_MTU: case MBOARD_PROP_CLOCK_RATE: case MBOARD_PROP_RX_DSP: case MBOARD_PROP_RX_DSP_NAMES: case MBOARD_PROP_TX_DSP: case MBOARD_PROP_TX_DSP_NAMES: case MBOARD_PROP_PPS_SOURCE: case MBOARD_PROP_PPS_SOURCE_NAMES: case MBOARD_PROP_PPS_POLARITY: case MBOARD_PROP_REF_SOURCE: case MBOARD_PROP_REF_SOURCE_NAMES: case MBOARD_PROP_TIME_NOW: case MBOARD_PROP_TIME_NEXT_PPS: throw std::runtime_error("unhandled prop in usrp2 mboard"); } } /*********************************************************************** * Set Properties **********************************************************************/ void usrp2_impl::set(const wax::obj &, const wax::obj &){ throw std::runtime_error("Cannot set on usrp2 mboard"); }