// // 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 "usrp_e100_impl.hpp" #include "usrp_e100_regs.hpp" #include #include #include #include #include #include #include #include #include #include #include using namespace uhd; using namespace uhd::usrp; namespace fs = boost::filesystem; /*********************************************************************** * Discovery **********************************************************************/ static device_addrs_t usrp_e100_find(const device_addr_t &hint){ device_addrs_t usrp_e100_addrs; //return an empty list of addresses when type is set to non-usrp-e if (hint.has_key("type") and hint["type"] != "usrp-e") return usrp_e100_addrs; //device node not provided, assume its 0 if (not hint.has_key("node")){ device_addr_t new_addr = hint; new_addr["node"] = "/dev/usrp_e0"; return usrp_e100_find(new_addr); } //use the given device node name if (fs::exists(hint["node"])){ device_addr_t new_addr; new_addr["type"] = "usrp-e"; new_addr["node"] = fs::system_complete(fs::path(hint["node"])).file_string(); try{ usrp_e100_iface::sptr iface = usrp_e100_iface::make(new_addr["node"]); new_addr["name"] = iface->mb_eeprom["name"]; new_addr["serial"] = iface->mb_eeprom["serial"]; if ( (not hint.has_key("name") or hint["name"] == new_addr["name"]) and (not hint.has_key("serial") or hint["serial"] == new_addr["serial"]) ){ usrp_e100_addrs.push_back(new_addr); } } catch(const std::exception &e){ uhd::warning::post( std::string("Ignoring discovered device\n") + e.what() ); } } return usrp_e100_addrs; } /*********************************************************************** * Make **********************************************************************/ static device::sptr usrp_e100_make(const device_addr_t &device_addr){ //setup the main interface into fpga std::string node = device_addr["node"]; std::cout << boost::format("Opening USRP-E on %s") % node << std::endl; usrp_e100_iface::sptr iface = usrp_e100_iface::make(node); //------------------------------------------------------------------ //-- Handle the FPGA loading... //-- The image can be confimed as already loaded when: //-- 1) The compatibility number matches. //-- 2) The hash in the hash-file matches. //------------------------------------------------------------------ static const char *hash_file_path = "/tmp/usrp_e100_hash"; //extract the fpga path for usrp-e std::string usrp_e100_fpga_image = find_image_path( device_addr.has_key("fpga")? device_addr["fpga"] : "usrp_e100_fpga.bin" ); //calculate a hash of the fpga file size_t fpga_hash = 0; { std::ifstream file(usrp_e100_fpga_image.c_str()); if (not file.good()) throw std::runtime_error( "cannot open fpga file for read: " + usrp_e100_fpga_image ); do{ boost::hash_combine(fpga_hash, file.get()); } while (file.good()); file.close(); } //read the compatibility number boost::uint16_t fpga_compat_num = iface->peek16(UE_REG_MISC_COMPAT); //read the hash in the hash-file size_t loaded_hash = 0; try{std::ifstream(hash_file_path) >> loaded_hash;}catch(...){} //if not loaded: load the fpga image and write the hash-file if (fpga_compat_num != USRP_E_COMPAT_NUM or loaded_hash != fpga_hash){ iface.reset(); usrp_e100_load_fpga(usrp_e100_fpga_image); sleep(1); ///\todo do this better one day. std::cout << boost::format("re-Opening USRP-E on %s") % node << std::endl; iface = usrp_e100_iface::make(node); try{std::ofstream(hash_file_path) << fpga_hash;}catch(...){} } //check that the compatibility is correct fpga_compat_num = iface->peek16(UE_REG_MISC_COMPAT); if (fpga_compat_num != USRP_E_COMPAT_NUM){ throw std::runtime_error(str(boost::format( "Expected fpga compatibility number 0x%x, but got 0x%x:\n" "The fpga build is not compatible with the host code build." ) % USRP_E_COMPAT_NUM % fpga_compat_num)); } return device::sptr(new usrp_e100_impl(iface)); } UHD_STATIC_BLOCK(register_usrp_e100_device){ device::register_device(&usrp_e100_find, &usrp_e100_make); } /*********************************************************************** * Structors **********************************************************************/ usrp_e100_impl::usrp_e100_impl(usrp_e100_iface::sptr iface): _iface(iface){ //setup interfaces into hardware _clock_ctrl = usrp_e100_clock_ctrl::make(_iface); _codec_ctrl = usrp_e100_codec_ctrl::make(_iface); //initialize the mboard mboard_init(); //initialize the dboards dboard_init(); //initialize the dsps rx_ddc_init(); tx_duc_init(); //init the codec properties codec_init(); //init the io send/recv io_init(); //set default subdev specs this->mboard_set(MBOARD_PROP_RX_SUBDEV_SPEC, subdev_spec_t()); this->mboard_set(MBOARD_PROP_TX_SUBDEV_SPEC, subdev_spec_t()); } usrp_e100_impl::~usrp_e100_impl(void){ /* NOP */ } /*********************************************************************** * Device Get **********************************************************************/ void usrp_e100_impl::get(const wax::obj &key_, wax::obj &val){ named_prop_t key = named_prop_t::extract(key_); //handle the get request conditioned on the key switch(key.as()){ case DEVICE_PROP_NAME: val = std::string("usrp-e device"); return; case DEVICE_PROP_MBOARD: UHD_ASSERT_THROW(key.name == ""); val = _mboard_proxy->get_link(); return; case DEVICE_PROP_MBOARD_NAMES: val = prop_names_t(1, ""); //vector of size 1 with empty string return; default: UHD_THROW_PROP_GET_ERROR(); } } /*********************************************************************** * Device Set **********************************************************************/ void usrp_e100_impl::set(const wax::obj &, const wax::obj &){ UHD_THROW_PROP_SET_ERROR(); }