// // 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 "impl_base.hpp" #include "dboard_interface.hpp" using namespace uhd; using namespace uhd::usrp; /*********************************************************************** * Structors **********************************************************************/ impl_base::impl_base( 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 dboard_interface(this) ); dboard::manager::sptr dboard_manager( new dboard::manager(rx_dboard_id, tx_dboard_id, _dboard_interface) ); //load dboards _rx_dboards[""] = dboard_impl::sptr(new dboard_impl(dboard_manager, dboard_impl::TYPE_RX)); _tx_dboards[""] = dboard_impl::sptr(new dboard_impl(dboard_manager, dboard_impl::TYPE_TX)); //TOD load dsps //init the pps source clock config _pps_source_dict["sma"] = USRP2_PPS_SOURCE_SMA; _pps_source_dict["mimo"] = USRP2_PPS_SOURCE_MIMO; _pps_source = "sma"; //init the pps polarity clock config _pps_polarity_dict["pos"] = USRP2_PPS_POLARITY_POS; _pps_polarity_dict["neg"] = USRP2_PPS_POLARITY_NEG; _pps_polarity = "neg"; //init the ref source clock config _ref_source_dict["int"] = USRP2_REF_SOURCE_INT; _ref_source_dict["sma"] = USRP2_REF_SOURCE_SMA; _ref_source_dict["mimo"] = USRP2_REF_SOURCE_MIMO; _ref_source = "int"; //update the clock config (sends a control packet) update_clock_config(); } impl_base::~impl_base(void){ /* NOP */ } /*********************************************************************** * Misc Access Methods **********************************************************************/ double impl_base::get_master_clock_freq(void){ return 100e6; } void impl_base::update_clock_config(void){ //setup the out data usrp2_ctrl_data_t out_data; out_data.id = htonl(USRP2_CTRL_ID_HERES_A_NEW_CLOCK_CONFIG_BRO); out_data.data.clock_config.pps_source = _pps_source_dict [_pps_source]; out_data.data.clock_config.pps_polarity = _pps_polarity_dict[_pps_polarity]; out_data.data.clock_config.ref_source = _ref_source_dict [_ref_source]; //send and recv usrp2_ctrl_data_t in_data = ctrl_send_and_recv(out_data); ASSERT_THROW(htonl(in_data.id) == USRP2_CTRL_ID_GOT_THE_NEW_CLOCK_CONFIG_DUDE); } /*********************************************************************** * Control Send/Recv **********************************************************************/ usrp2_ctrl_data_t impl_base::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 impl_base::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: val = _rx_dboards[name]->get_link(); return; case MBOARD_PROP_RX_DBOARD_NAMES: val = prop_names_t(_rx_dboards.get_keys()); return; case MBOARD_PROP_TX_DBOARD: val = _tx_dboards[name]->get_link(); return; case MBOARD_PROP_TX_DBOARD_NAMES: val = prop_names_t(_tx_dboards.get_keys()); return; case MBOARD_PROP_MTU: // FIXME we dont know the real MTU... // give them something to fragment about val = size_t(1500); return; case MBOARD_PROP_CLOCK_RATE: val = freq_t(get_master_clock_freq()); return; case MBOARD_PROP_RX_DSP: throw std::runtime_error("unhandled prop in usrp2 mboard"); case MBOARD_PROP_RX_DSP_NAMES: throw std::runtime_error("unhandled prop in usrp2 mboard"); case MBOARD_PROP_TX_DSP: throw std::runtime_error("unhandled prop in usrp2 mboard"); case MBOARD_PROP_TX_DSP_NAMES: throw std::runtime_error("unhandled prop in usrp2 mboard"); case MBOARD_PROP_PPS_SOURCE: val = _pps_source; return; case MBOARD_PROP_PPS_SOURCE_NAMES: val = prop_names_t(_pps_source_dict.get_keys()); return; case MBOARD_PROP_PPS_POLARITY: val = _pps_polarity; return; case MBOARD_PROP_REF_SOURCE: val = _ref_source; return; case MBOARD_PROP_REF_SOURCE_NAMES: val = prop_names_t(_ref_source_dict.get_keys()); return; case MBOARD_PROP_TIME_NOW: case MBOARD_PROP_TIME_NEXT_PPS: throw std::runtime_error("Error: trying to get write-only property on usrp2 mboard"); } } /*********************************************************************** * Set Properties **********************************************************************/ void impl_base::set(const wax::obj &key, const wax::obj &val){ //handle the get request conditioned on the key switch(wax::cast(key)){ case MBOARD_PROP_PPS_SOURCE:{ std::string name = wax::cast(val); ASSERT_THROW(_pps_source_dict.has_key(name)); _pps_source = name; //shadow update_clock_config(); } return; case MBOARD_PROP_PPS_POLARITY:{ std::string name = wax::cast(val); ASSERT_THROW(_pps_polarity_dict.has_key(name)); _pps_polarity = name; //shadow update_clock_config(); } return; case MBOARD_PROP_REF_SOURCE:{ std::string name = wax::cast(val); ASSERT_THROW(_ref_source_dict.has_key(name)); _ref_source = name; //shadow update_clock_config(); } return; case MBOARD_PROP_NAME: case MBOARD_PROP_OTHERS: 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_RX_DBOARD: case MBOARD_PROP_RX_DBOARD_NAMES: case MBOARD_PROP_TX_DBOARD: case MBOARD_PROP_TX_DBOARD_NAMES: case MBOARD_PROP_PPS_SOURCE_NAMES: case MBOARD_PROP_REF_SOURCE_NAMES: case MBOARD_PROP_TIME_NOW: case MBOARD_PROP_TIME_NEXT_PPS: throw std::runtime_error("Error: trying to set read-only property on usrp2 mboard"); } }