//
// Copyright 2013-2015 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 "x300_impl.hpp"
#include "x300_lvbitx.hpp"
#include "x310_lvbitx.hpp"
#include "apply_corrections.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define NIUSRPRIO_DEFAULT_RPC_PORT "5444"
#define X300_REV(x) ((x) - "A" + 1)
using namespace uhd;
using namespace uhd::usrp;
using namespace uhd::transport;
using namespace uhd::niusrprio;
using namespace uhd::usrp::gpio_atr;
using namespace uhd::usrp::x300;
namespace asio = boost::asio;
static bool has_dram_buff(wb_iface::sptr zpu_ctrl) {
bool dramR0 = dma_fifo_core_3000::check(
zpu_ctrl, SR_ADDR(SET0_BASE, ZPU_SR_DRAM_FIFO0), SR_ADDR(SET0_BASE, ZPU_RB_DRAM_FIFO0));
bool dramR1 = dma_fifo_core_3000::check(
zpu_ctrl, SR_ADDR(SET0_BASE, ZPU_SR_DRAM_FIFO1), SR_ADDR(SET0_BASE, ZPU_RB_DRAM_FIFO1));
return (dramR0 and dramR1);
}
static std::string get_fpga_option(wb_iface::sptr zpu_ctrl) {
//Possible options:
//1G = {0:1G, 1:1G} w/ DRAM, HG = {0:1G, 1:10G} w/ DRAM, XG = {0:10G, 1:10G} w/ DRAM
//1GS = {0:1G, 1:1G} w/ SRAM, HGS = {0:1G, 1:10G} w/ SRAM, XGS = {0:10G, 1:10G} w/ SRAM
std::string option;
bool eth0XG = (zpu_ctrl->peek32(SR_ADDR(SET0_BASE, ZPU_RB_ETH_TYPE0)) == 0x1);
bool eth1XG = (zpu_ctrl->peek32(SR_ADDR(SET0_BASE, ZPU_RB_ETH_TYPE1)) == 0x1);
option = (eth0XG && eth1XG) ? "XG" : (eth1XG ? "HG" : "1G");
if (not has_dram_buff(zpu_ctrl)) {
option += "S";
}
return option;
}
/***********************************************************************
* Discovery over the udp and pcie transport
**********************************************************************/
//@TODO: Refactor the find functions to collapse common code for ethernet and PCIe
static device_addrs_t x300_find_with_addr(const device_addr_t &hint)
{
udp_simple::sptr comm = udp_simple::make_broadcast(
hint["addr"], BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT));
//load request struct
x300_fw_comms_t request = x300_fw_comms_t();
request.flags = uhd::htonx(X300_FW_COMMS_FLAGS_ACK);
request.sequence = uhd::htonx(std::rand());
//send request
comm->send(asio::buffer(&request, sizeof(request)));
//loop for replies until timeout
device_addrs_t addrs;
while (true)
{
char buff[X300_FW_COMMS_MTU] = {};
const size_t nbytes = comm->recv(asio::buffer(buff), 0.050);
if (nbytes == 0) break;
const x300_fw_comms_t *reply = (const x300_fw_comms_t *)buff;
if (request.flags != reply->flags) break;
if (request.sequence != reply->sequence) break;
device_addr_t new_addr;
new_addr["type"] = "x300";
new_addr["addr"] = comm->get_recv_addr();
//Attempt to read the name from the EEPROM and perform filtering.
//This operation can throw due to compatibility mismatch.
try
{
wb_iface::sptr zpu_ctrl = x300_make_ctrl_iface_enet(
udp_simple::make_connected(new_addr["addr"],
BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT)),
false /* Suppress timeout errors */
);
if (x300_impl::is_claimed(zpu_ctrl)) continue; //claimed by another process
new_addr["fpga"] = get_fpga_option(zpu_ctrl);
i2c_core_100_wb32::sptr zpu_i2c = i2c_core_100_wb32::make(zpu_ctrl, I2C1_BASE);
i2c_iface::sptr eeprom16 = zpu_i2c->eeprom16();
const mboard_eeprom_t mb_eeprom(*eeprom16, "X300");
new_addr["name"] = mb_eeprom["name"];
new_addr["serial"] = mb_eeprom["serial"];
switch (x300_impl::get_mb_type_from_eeprom(mb_eeprom)) {
case x300_impl::USRP_X300_MB:
new_addr["product"] = "X300";
break;
case x300_impl::USRP_X310_MB:
new_addr["product"] = "X310";
break;
default:
break;
}
}
catch(const std::exception &)
{
//set these values as empty string so the device may still be found
//and the filter's below can still operate on the discovered device
new_addr["name"] = "";
new_addr["serial"] = "";
}
//filter the discovered device below by matching optional keys
if (
(not hint.has_key("name") or hint["name"] == new_addr["name"]) and
(not hint.has_key("serial") or hint["serial"] == new_addr["serial"]) and
(not hint.has_key("product") or hint["product"] == new_addr["product"])
){
addrs.push_back(new_addr);
}
}
return addrs;
}
//We need a zpu xport registry to ensure synchronization between the static finder method
//and the instances of the x300_impl class.
typedef uhd::dict< std::string, boost::weak_ptr > pcie_zpu_iface_registry_t;
UHD_SINGLETON_FCN(pcie_zpu_iface_registry_t, get_pcie_zpu_iface_registry)
static boost::mutex pcie_zpu_iface_registry_mutex;
static device_addrs_t x300_find_pcie(const device_addr_t &hint, bool explicit_query)
{
std::string rpc_port_name(NIUSRPRIO_DEFAULT_RPC_PORT);
if (hint.has_key("niusrpriorpc_port")) {
rpc_port_name = hint["niusrpriorpc_port"];
}
device_addrs_t addrs;
niusrprio_session::device_info_vtr dev_info_vtr;
nirio_status status = niusrprio_session::enumerate(rpc_port_name, dev_info_vtr);
if (explicit_query) nirio_status_to_exception(status, "x300_find_pcie: Error enumerating NI-RIO devices.");
BOOST_FOREACH(niusrprio_session::device_info &dev_info, dev_info_vtr)
{
device_addr_t new_addr;
new_addr["type"] = "x300";
new_addr["resource"] = dev_info.resource_name;
std::string resource_d(dev_info.resource_name);
boost::to_upper(resource_d);
switch (x300_impl::get_mb_type_from_pcie(resource_d, rpc_port_name)) {
case x300_impl::USRP_X300_MB:
new_addr["product"] = "X300";
break;
case x300_impl::USRP_X310_MB:
new_addr["product"] = "X310";
break;
default:
continue;
}
niriok_proxy::sptr kernel_proxy = niriok_proxy::make_and_open(dev_info.interface_path);
//Attempt to read the name from the EEPROM and perform filtering.
//This operation can throw due to compatibility mismatch.
try
{
//This block could throw an exception if the user is switching to using UHD
//after LabVIEW FPGA. In that case, skip reading the name and serial and pick
//a default FPGA flavor. During make, a new image will be loaded and everything
//will be OK
wb_iface::sptr zpu_ctrl;
//Hold on to the registry mutex as long as zpu_ctrl is alive
//to prevent any use by different threads while enumerating
boost::mutex::scoped_lock(pcie_zpu_iface_registry_mutex);
if (get_pcie_zpu_iface_registry().has_key(resource_d)) {
zpu_ctrl = get_pcie_zpu_iface_registry()[resource_d].lock();
} else {
zpu_ctrl = x300_make_ctrl_iface_pcie(kernel_proxy, false /* suppress timeout errors */);
//We don't put this zpu_ctrl in the registry because we need
//a persistent niriok_proxy associated with the object
}
if (x300_impl::is_claimed(zpu_ctrl)) continue; //claimed by another process
//Attempt to autodetect the FPGA type
if (not hint.has_key("fpga")) {
new_addr["fpga"] = get_fpga_option(zpu_ctrl);
}
i2c_core_100_wb32::sptr zpu_i2c = i2c_core_100_wb32::make(zpu_ctrl, I2C1_BASE);
i2c_iface::sptr eeprom16 = zpu_i2c->eeprom16();
const mboard_eeprom_t mb_eeprom(*eeprom16, "X300");
new_addr["name"] = mb_eeprom["name"];
new_addr["serial"] = mb_eeprom["serial"];
}
catch(const std::exception &)
{
//set these values as empty string so the device may still be found
//and the filter's below can still operate on the discovered device
if (not hint.has_key("fpga")) {
new_addr["fpga"] = "HGS";
}
new_addr["name"] = "";
new_addr["serial"] = "";
}
//filter the discovered device below by matching optional keys
std::string resource_i = hint.has_key("resource") ? hint["resource"] : "";
boost::to_upper(resource_i);
if (
(not hint.has_key("resource") or resource_i == resource_d) and
(not hint.has_key("name") or hint["name"] == new_addr["name"]) and
(not hint.has_key("serial") or hint["serial"] == new_addr["serial"]) and
(not hint.has_key("product") or hint["product"] == new_addr["product"])
){
addrs.push_back(new_addr);
}
}
return addrs;
}
device_addrs_t x300_find(const device_addr_t &hint_)
{
//handle the multi-device discovery
device_addrs_t hints = separate_device_addr(hint_);
if (hints.size() > 1)
{
device_addrs_t found_devices;
std::string error_msg;
BOOST_FOREACH(const device_addr_t &hint_i, hints)
{
device_addrs_t found_devices_i = x300_find(hint_i);
if (found_devices_i.size() != 1) error_msg += str(boost::format(
"Could not resolve device hint \"%s\" to a single device."
) % hint_i.to_string());
else found_devices.push_back(found_devices_i[0]);
}
if (found_devices.empty()) return device_addrs_t();
if (not error_msg.empty()) throw uhd::value_error(error_msg);
return device_addrs_t(1, combine_device_addrs(found_devices));
}
//initialize the hint for a single device case
UHD_ASSERT_THROW(hints.size() <= 1);
hints.resize(1); //in case it was empty
device_addr_t hint = hints[0];
device_addrs_t addrs;
if (hint.has_key("type") and hint["type"] != "x300") return addrs;
//use the address given
if (hint.has_key("addr"))
{
device_addrs_t reply_addrs;
try
{
reply_addrs = x300_find_with_addr(hint);
}
catch(const std::exception &ex)
{
UHD_MSG(error) << "X300 Network discovery error " << ex.what() << std::endl;
}
catch(...)
{
UHD_MSG(error) << "X300 Network discovery unknown error " << std::endl;
}
BOOST_FOREACH(const device_addr_t &reply_addr, reply_addrs)
{
device_addrs_t new_addrs = x300_find_with_addr(reply_addr);
addrs.insert(addrs.begin(), new_addrs.begin(), new_addrs.end());
}
return addrs;
}
if (!hint.has_key("resource"))
{
//otherwise, no address was specified, send a broadcast on each interface
BOOST_FOREACH(const if_addrs_t &if_addrs, get_if_addrs())
{
//avoid the loopback device
if (if_addrs.inet == asio::ip::address_v4::loopback().to_string()) continue;
//create a new hint with this broadcast address
device_addr_t new_hint = hint;
new_hint["addr"] = if_addrs.bcast;
//call discover with the new hint and append results
device_addrs_t new_addrs = x300_find(new_hint);
addrs.insert(addrs.begin(), new_addrs.begin(), new_addrs.end());
}
}
device_addrs_t pcie_addrs = x300_find_pcie(hint, hint.has_key("resource"));
if (not pcie_addrs.empty()) addrs.insert(addrs.end(), pcie_addrs.begin(), pcie_addrs.end());
return addrs;
}
/***********************************************************************
* Make
**********************************************************************/
static device::sptr x300_make(const device_addr_t &device_addr)
{
return device::sptr(new x300_impl(device_addr));
}
UHD_STATIC_BLOCK(register_x300_device)
{
device::register_device(&x300_find, &x300_make, device::USRP);
}
static void x300_load_fw(wb_iface::sptr fw_reg_ctrl, const std::string &file_name)
{
UHD_MSG(status) << "Loading firmware " << file_name << std::flush;
//load file into memory
std::ifstream fw_file(file_name.c_str());
boost::uint32_t fw_file_buff[X300_FW_NUM_BYTES/sizeof(boost::uint32_t)];
fw_file.read((char *)fw_file_buff, sizeof(fw_file_buff));
fw_file.close();
//Poke the fw words into the WB boot loader
fw_reg_ctrl->poke32(SR_ADDR(BOOT_LDR_BASE, BL_ADDRESS), 0);
for (size_t i = 0; i < X300_FW_NUM_BYTES; i+=sizeof(boost::uint32_t))
{
//@TODO: FIXME: Since x300_ctrl_iface acks each write and traps exceptions, the first try for the last word
// written will print an error because it triggers a FW reload and fails to reply.
fw_reg_ctrl->poke32(SR_ADDR(BOOT_LDR_BASE, BL_DATA), uhd::byteswap(fw_file_buff[i/sizeof(boost::uint32_t)]));
if ((i & 0x1fff) == 0) UHD_MSG(status) << "." << std::flush;
}
//Wait for fimrware to reboot. 3s is an upper bound
boost::this_thread::sleep(boost::posix_time::milliseconds(3000));
UHD_MSG(status) << " done!" << std::endl;
}
x300_impl::x300_impl(const uhd::device_addr_t &dev_addr)
{
UHD_MSG(status) << "X300 initialization sequence..." << std::endl;
_type = device::USRP;
_ignore_cal_file = dev_addr.has_key("ignore-cal-file");
_async_md.reset(new async_md_type(1000/*messages deep*/));
_tree = uhd::property_tree::make();
_tree->create("/name").set("X-Series Device");
_sid_framer = 0;
const device_addrs_t device_args = separate_device_addr(dev_addr);
_mb.resize(device_args.size());
for (size_t i = 0; i < device_args.size(); i++)
{
this->setup_mb(i, device_args[i]);
}
}
void x300_impl::mboard_members_t::discover_eth(
const mboard_eeprom_t mb_eeprom,
const std::vector &ip_addrs)
{
// Clear any previous addresses added
eth_conns.clear();
// Index the MB EEPROM addresses
std::vector mb_eeprom_addrs;
const size_t num_mb_eeprom_addrs = 4;
for (size_t i = 0; i < num_mb_eeprom_addrs; i++) {
const std::string key = "ip-addr" + boost::to_string(i);
// Show a warning if there exists duplicate addresses in the mboard eeprom
if (std::find(mb_eeprom_addrs.begin(), mb_eeprom_addrs.end(), mb_eeprom[key]) != mb_eeprom_addrs.end()) {
UHD_MSG(warning) << str(boost::format(
"Duplicate IP address %s found in mboard EEPROM. "
"Device may not function properly.\nView and reprogram the values "
"using the usrp_burn_mb_eeprom utility.\n") % mb_eeprom[key]);
}
mb_eeprom_addrs.push_back(mb_eeprom[key]);
}
BOOST_FOREACH(const std::string& addr, ip_addrs) {
x300_eth_conn_t conn_iface;
conn_iface.addr = addr;
conn_iface.type = X300_IFACE_NONE;
// Decide from the mboard eeprom what IP corresponds
// to an interface
for (size_t i = 0; i < mb_eeprom_addrs.size(); i++) {
if (addr == mb_eeprom_addrs[i]) {
// Choose the interface based on the index parity
if (i % 2 == 0) {
conn_iface.type = X300_IFACE_ETH0;
} else {
conn_iface.type = X300_IFACE_ETH1;
}
break;
}
}
// Check default IP addresses if we couldn't
// determine the IP from the mboard eeprom
if (conn_iface.type == X300_IFACE_NONE) {
UHD_MSG(warning) << str(boost::format(
"Address %s not found in mboard EEPROM. Address may be wrong or "
"the EEPROM may be corrupt.\n Attempting to continue with default "
"IP addresses.\n") % conn_iface.addr
);
if (addr == boost::asio::ip::address_v4(
boost::uint32_t(X300_DEFAULT_IP_ETH0_1G)).to_string()) {
conn_iface.type = X300_IFACE_ETH0;
} else if (addr == boost::asio::ip::address_v4(
boost::uint32_t(X300_DEFAULT_IP_ETH1_1G)).to_string()) {
conn_iface.type = X300_IFACE_ETH1;
} else if (addr == boost::asio::ip::address_v4(
boost::uint32_t(X300_DEFAULT_IP_ETH0_10G)).to_string()) {
conn_iface.type = X300_IFACE_ETH0;
} else if (addr == boost::asio::ip::address_v4(
boost::uint32_t(X300_DEFAULT_IP_ETH1_10G)).to_string()) {
conn_iface.type = X300_IFACE_ETH1;
} else {
throw uhd::assertion_error(str(boost::format(
"X300 Initialization Error: Failed to match address %s with "
"any addresses for the device. Please check the address.")
% conn_iface.addr
));
}
}
// Save to a vector of connections
if (conn_iface.type != X300_IFACE_NONE) {
// Check the address before we add it
try
{
wb_iface::sptr zpu_ctrl = x300_make_ctrl_iface_enet(
udp_simple::make_connected(conn_iface.addr,
BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT)),
false /* Suppress timeout errors */
);
// Peek the ZPU ctrl to make sure this connection works
zpu_ctrl->peek32(0);
}
// If the address does not work, throw an error
catch(std::exception &)
{
throw uhd::io_error(str(boost::format(
"X300 Initialization Error: Invalid address %s")
% conn_iface.addr));
}
eth_conns.push_back(conn_iface);
}
}
if (eth_conns.size() == 0)
throw uhd::assertion_error("X300 Initialization Error: No ethernet interfaces specified.");
}
void x300_impl::setup_mb(const size_t mb_i, const uhd::device_addr_t &dev_addr)
{
const fs_path mb_path = "/mboards/"+boost::lexical_cast(mb_i);
mboard_members_t &mb = _mb[mb_i];
mb.initialization_done = false;
std::vector eth_addrs;
// Not choosing eth0 based on resource might cause user issues
std::string eth0_addr = dev_addr.has_key("resource") ? dev_addr["resource"] : dev_addr["addr"];
eth_addrs.push_back(eth0_addr);
if (dev_addr.has_key("second_addr")) {
std::string eth1_addr = dev_addr["second_addr"];
// Ensure we do not have duplicate addresses
if (eth1_addr != eth0_addr)
eth_addrs.push_back(eth1_addr);
}
// Initially store the first address provided to setup communication
// Once we read the eeprom, we use it to map IP to its interface
x300_eth_conn_t init;
init.addr = eth_addrs[0];
mb.eth_conns.push_back(init);
mb.xport_path = dev_addr.has_key("resource") ? "nirio" : "eth";
mb.if_pkt_is_big_endian = mb.xport_path != "nirio";
if (mb.xport_path == "nirio")
{
nirio_status status = 0;
std::string rpc_port_name(NIUSRPRIO_DEFAULT_RPC_PORT);
if (dev_addr.has_key("niusrpriorpc_port")) {
rpc_port_name = dev_addr["niusrpriorpc_port"];
}
UHD_MSG(status) << boost::format("Connecting to niusrpriorpc at localhost:%s...\n") % rpc_port_name;
//Instantiate the correct lvbitx object
nifpga_lvbitx::sptr lvbitx;
switch (get_mb_type_from_pcie(dev_addr["resource"], rpc_port_name)) {
case USRP_X300_MB:
lvbitx.reset(new x300_lvbitx(dev_addr["fpga"]));
break;
case USRP_X310_MB:
lvbitx.reset(new x310_lvbitx(dev_addr["fpga"]));
break;
default:
nirio_status_to_exception(status, "Motherboard detection error. Please ensure that you \
have a valid USRP X3x0, NI USRP-294xR or NI USRP-295xR device and that all the device \
drivers have loaded successfully.");
}
//Load the lvbitx onto the device
UHD_MSG(status) << boost::format("Using LVBITX bitfile %s...\n") % lvbitx->get_bitfile_path();
mb.rio_fpga_interface.reset(new niusrprio_session(dev_addr["resource"], rpc_port_name));
nirio_status_chain(mb.rio_fpga_interface->open(lvbitx, dev_addr.has_key("download-fpga")), status);
nirio_status_to_exception(status, "x300_impl: Could not initialize RIO session.");
//Tell the quirks object which FIFOs carry TX stream data
const boost::uint32_t tx_data_fifos[2] = {X300_RADIO_DEST_PREFIX_TX, X300_RADIO_DEST_PREFIX_TX + 3};
mb.rio_fpga_interface->get_kernel_proxy()->get_rio_quirks().register_tx_streams(tx_data_fifos, 2);
_tree->create(mb_path / "link_max_rate").set(X300_MAX_RATE_PCIE);
}
BOOST_FOREACH(const std::string &key, dev_addr.keys())
{
if (key.find("recv") != std::string::npos) mb.recv_args[key] = dev_addr[key];
if (key.find("send") != std::string::npos) mb.send_args[key] = dev_addr[key];
}
if (mb.xport_path == "eth" ) {
/* This is an ETH connection. Figure out what the maximum supported frame
* size is for the transport in the up and down directions. The frame size
* depends on the host PIC's NIC's MTU settings. To determine the frame size,
* we test for support up to an expected "ceiling". If the user
* specified a frame size, we use that frame size as the ceiling. If no
* frame size was specified, we use the maximum UHD frame size.
*
* To optimize performance, the frame size should be greater than or equal
* to the frame size that UHD uses so that frames don't get split across
* multiple transmission units - this is why the limits passed into the
* 'determine_max_frame_size' function are actually frame sizes. */
frame_size_t req_max_frame_size;
req_max_frame_size.recv_frame_size = (mb.recv_args.has_key("recv_frame_size")) \
? boost::lexical_cast(mb.recv_args["recv_frame_size"]) \
: X300_10GE_DATA_FRAME_MAX_SIZE;
req_max_frame_size.send_frame_size = (mb.send_args.has_key("send_frame_size")) \
? boost::lexical_cast(mb.send_args["send_frame_size"]) \
: X300_10GE_DATA_FRAME_MAX_SIZE;
#if defined UHD_PLATFORM_LINUX
const std::string mtu_tool("ip link");
#elif defined UHD_PLATFORM_WIN32
const std::string mtu_tool("netsh");
#else
const std::string mtu_tool("ifconfig");
#endif
// Detect the frame size on the path to the USRP
try {
frame_size_t pri_frame_sizes = determine_max_frame_size(
eth_addrs.at(0), req_max_frame_size
);
_max_frame_sizes = pri_frame_sizes;
if (eth_addrs.size() > 1) {
frame_size_t sec_frame_sizes = determine_max_frame_size(
eth_addrs.at(1), req_max_frame_size
);
// Choose the minimum of the max frame sizes
// to ensure we don't exceed any one of the links' MTU
_max_frame_sizes.recv_frame_size = std::min(
pri_frame_sizes.recv_frame_size,
sec_frame_sizes.recv_frame_size
);
_max_frame_sizes.send_frame_size = std::min(
pri_frame_sizes.send_frame_size,
sec_frame_sizes.send_frame_size
);
}
} catch(std::exception &e) {
UHD_MSG(error) << e.what() << std::endl;
}
if ((mb.recv_args.has_key("recv_frame_size"))
&& (req_max_frame_size.recv_frame_size < _max_frame_sizes.recv_frame_size)) {
UHD_MSG(warning)
<< boost::format("You requested a receive frame size of (%lu) but your NIC's max frame size is (%lu).")
% req_max_frame_size.recv_frame_size << _max_frame_sizes.recv_frame_size << std::endl
<< boost::format("Please verify your NIC's MTU setting using '%s' or set the recv_frame_size argument appropriately.")
% mtu_tool << std::endl
<< "UHD will use the auto-detected max frame size for this connection."
<< std::endl;
}
if ((mb.recv_args.has_key("send_frame_size"))
&& (req_max_frame_size.send_frame_size < _max_frame_sizes.send_frame_size)) {
UHD_MSG(warning)
<< boost::format("You requested a send frame size of (%lu) but your NIC's max frame size is (%lu).")
% req_max_frame_size.send_frame_size << _max_frame_sizes.send_frame_size << std::endl
<< boost::format("Please verify your NIC's MTU setting using '%s' or set the send_frame_size argument appropriately.")
% mtu_tool << std::endl
<< "UHD will use the auto-detected max frame size for this connection."
<< std::endl;
}
_tree->create(mb_path / "link_max_rate").set(X300_MAX_RATE_10GIGE);
}
//create basic communication
UHD_MSG(status) << "Setup basic communication..." << std::endl;
if (mb.xport_path == "nirio") {
boost::mutex::scoped_lock(pcie_zpu_iface_registry_mutex);
if (get_pcie_zpu_iface_registry().has_key(mb.get_pri_eth().addr)) {
throw uhd::assertion_error("Someone else has a ZPU transport to the device open. Internal error!");
} else {
mb.zpu_ctrl = x300_make_ctrl_iface_pcie(mb.rio_fpga_interface->get_kernel_proxy());
get_pcie_zpu_iface_registry()[mb.get_pri_eth().addr] = boost::weak_ptr(mb.zpu_ctrl);
}
} else {
mb.zpu_ctrl = x300_make_ctrl_iface_enet(udp_simple::make_connected(
mb.get_pri_eth().addr, BOOST_STRINGIZE(X300_FW_COMMS_UDP_PORT)));
}
mb.claimer_task = uhd::task::make(boost::bind(&x300_impl::claimer_loop, this, mb.zpu_ctrl));
//extract the FW path for the X300
//and live load fw over ethernet link
if (dev_addr.has_key("fw"))
{
const std::string x300_fw_image = find_image_path(
dev_addr.has_key("fw")? dev_addr["fw"] : X300_FW_FILE_NAME
);
x300_load_fw(mb.zpu_ctrl, x300_fw_image);
}
//check compat numbers
//check fpga compat before fw compat because the fw is a subset of the fpga image
this->check_fpga_compat(mb_path, mb);
this->check_fw_compat(mb_path, mb.zpu_ctrl);
mb.fw_regmap = boost::make_shared();
mb.fw_regmap->initialize(*mb.zpu_ctrl.get(), true);
//store which FPGA image is loaded
mb.loaded_fpga_image = get_fpga_option(mb.zpu_ctrl);
//low speed perif access
mb.zpu_spi = spi_core_3000::make(mb.zpu_ctrl, SR_ADDR(SET0_BASE, ZPU_SR_SPI),
SR_ADDR(SET0_BASE, ZPU_RB_SPI));
mb.zpu_i2c = i2c_core_100_wb32::make(mb.zpu_ctrl, I2C1_BASE);
mb.zpu_i2c->set_clock_rate(X300_BUS_CLOCK_RATE);
////////////////////////////////////////////////////////////////////
// print network routes mapping
////////////////////////////////////////////////////////////////////
/*
const uint32_t routes_addr = mb.zpu_ctrl->peek32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_ROUTE_MAP_ADDR));
const uint32_t routes_len = mb.zpu_ctrl->peek32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_ROUTE_MAP_LEN));
UHD_VAR(routes_len);
for (size_t i = 0; i < routes_len; i+=1)
{
const uint32_t node_addr = mb.zpu_ctrl->peek32(SR_ADDR(routes_addr, i*2+0));
const uint32_t nbor_addr = mb.zpu_ctrl->peek32(SR_ADDR(routes_addr, i*2+1));
if (node_addr != 0 and nbor_addr != 0)
{
UHD_MSG(status) << boost::format("%u: %s -> %s")
% i
% asio::ip::address_v4(node_addr).to_string()
% asio::ip::address_v4(nbor_addr).to_string()
<< std::endl;
}
}
*/
////////////////////////////////////////////////////////////////////
// setup the mboard eeprom
////////////////////////////////////////////////////////////////////
UHD_MSG(status) << "Loading values from EEPROM..." << std::endl;
i2c_iface::sptr eeprom16 = mb.zpu_i2c->eeprom16();
if (dev_addr.has_key("blank_eeprom"))
{
UHD_MSG(warning) << "Obliterating the motherboard EEPROM..." << std::endl;
eeprom16->write_eeprom(0x50, 0, byte_vector_t(256, 0xff));
}
const mboard_eeprom_t mb_eeprom(*eeprom16, "X300");
_tree->create(mb_path / "eeprom")
.set(mb_eeprom)
.add_coerced_subscriber(boost::bind(&x300_impl::set_mb_eeprom, this, mb.zpu_i2c, _1));
bool recover_mb_eeprom = dev_addr.has_key("recover_mb_eeprom");
if (recover_mb_eeprom) {
UHD_MSG(warning) << "UHD is operating in EEPROM Recovery Mode which disables hardware version "
"checks.\nOperating in this mode may cause hardware damage and unstable "
"radio performance!"<< std::endl;
}
////////////////////////////////////////////////////////////////////
// parse the product number
////////////////////////////////////////////////////////////////////
std::string product_name = "X300?";
switch (get_mb_type_from_eeprom(mb_eeprom)) {
case USRP_X300_MB:
product_name = "X300";
break;
case USRP_X310_MB:
product_name = "X310";
break;
default:
if (not recover_mb_eeprom)
throw uhd::runtime_error("Unrecognized product type.\n"
"Either the software does not support this device in which case please update your driver software to the latest version and retry OR\n"
"The product code in the EEPROM is corrupt and may require reprogramming.");
}
_tree->create(mb_path / "name").set(product_name);
_tree->create(mb_path / "codename").set("Yetti");
////////////////////////////////////////////////////////////////////
// determine routing based on address match
////////////////////////////////////////////////////////////////////
if (mb.xport_path != "nirio") {
// Discover ethernet interfaces
mb.discover_eth(mb_eeprom, eth_addrs);
}
////////////////////////////////////////////////////////////////////
// read dboard eeproms
////////////////////////////////////////////////////////////////////
for (size_t i = 0; i < 8; i++)
{
if (i == 0 or i == 2) continue; //not used
mb.db_eeproms[i].load(*mb.zpu_i2c, 0x50 | i);
}
////////////////////////////////////////////////////////////////////
// read hardware revision and compatibility number
////////////////////////////////////////////////////////////////////
mb.hw_rev = 0;
if(mb_eeprom.has_key("revision") and not mb_eeprom["revision"].empty()) {
try {
mb.hw_rev = boost::lexical_cast(mb_eeprom["revision"]);
} catch(...) {
if (not recover_mb_eeprom)
throw uhd::runtime_error("Revision in EEPROM is invalid! Please reprogram your EEPROM.");
}
} else {
if (not recover_mb_eeprom)
throw uhd::runtime_error("No revision detected. MB EEPROM must be reprogrammed!");
}
size_t hw_rev_compat = 0;
if (mb.hw_rev >= 7) { //Revision compat was added with revision 7
if (mb_eeprom.has_key("revision_compat") and not mb_eeprom["revision_compat"].empty()) {
try {
hw_rev_compat = boost::lexical_cast(mb_eeprom["revision_compat"]);
} catch(...) {
if (not recover_mb_eeprom)
throw uhd::runtime_error("Revision compat in EEPROM is invalid! Please reprogram your EEPROM.");
}
} else {
if (not recover_mb_eeprom)
throw uhd::runtime_error("No revision compat detected. MB EEPROM must be reprogrammed!");
}
} else {
//For older HW just assume that revision_compat = revision
hw_rev_compat = mb.hw_rev;
}
if (hw_rev_compat > X300_REVISION_COMPAT) {
if (not recover_mb_eeprom)
throw uhd::runtime_error(str(boost::format(
"Hardware is too new for this software. Please upgrade to a driver that supports hardware revision %d.")
% mb.hw_rev));
} else if (mb.hw_rev < X300_REVISION_MIN) { //Compare min against the revision (and not compat) to give us more leeway for partial support for a compat
if (not recover_mb_eeprom)
throw uhd::runtime_error(str(boost::format(
"Software is too new for this hardware. Please downgrade to a driver that supports hardware revision %d.")
% mb.hw_rev));
}
////////////////////////////////////////////////////////////////////
// create clock control objects
////////////////////////////////////////////////////////////////////
UHD_MSG(status) << "Setup RF frontend clocking..." << std::endl;
//Initialize clock control registers. NOTE: This does not configure the LMK yet.
mb.clock = x300_clock_ctrl::make(mb.zpu_spi,
1 /*slaveno*/,
mb.hw_rev,
dev_addr.cast("master_clock_rate", X300_DEFAULT_TICK_RATE),
dev_addr.cast("dboard_clock_rate", X300_DEFAULT_DBOARD_CLK_RATE),
dev_addr.cast("system_ref_rate", X300_DEFAULT_SYSREF_RATE));
//Initialize clock source to use internal reference and generate
//a valid radio clock. This may change after configuration is done.
//This will configure the LMK and wait for lock
update_clock_source(mb, "internal");
////////////////////////////////////////////////////////////////////
// create clock properties
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "tick_rate")
.set_publisher(boost::bind(&x300_clock_ctrl::get_master_clock_rate, mb.clock));
_tree->create(mb_path / "time" / "cmd");
UHD_MSG(status) << "Radio 1x clock:" << (mb.clock->get_master_clock_rate()/1e6)
<< std::endl;
////////////////////////////////////////////////////////////////////
// Create the GPSDO control
////////////////////////////////////////////////////////////////////
static const boost::uint32_t dont_look_for_gpsdo = 0x1234abcdul;
//otherwise if not disabled, look for the internal GPSDO
if (mb.zpu_ctrl->peek32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_GPSDO_STATUS)) != dont_look_for_gpsdo)
{
UHD_MSG(status) << "Detecting internal GPSDO.... " << std::flush;
try
{
mb.gps = gps_ctrl::make(x300_make_uart_iface(mb.zpu_ctrl));
}
catch(std::exception &e)
{
UHD_MSG(error) << "An error occurred making GPSDO control: " << e.what() << std::endl;
}
if (mb.gps and mb.gps->gps_detected())
{
BOOST_FOREACH(const std::string &name, mb.gps->get_sensors())
{
_tree->create(mb_path / "sensors" / name)
.set_publisher(boost::bind(&gps_ctrl::get_sensor, mb.gps, name));
}
}
else
{
mb.zpu_ctrl->poke32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_GPSDO_STATUS), dont_look_for_gpsdo);
}
}
////////////////////////////////////////////////////////////////////
//clear router?
////////////////////////////////////////////////////////////////////
for (size_t i = 0; i < 512; i++) {
mb.zpu_ctrl->poke32(SR_ADDR(SETXB_BASE, i), 0);
}
////////////////////////////////////////////////////////////////////
// DRAM FIFO initialization
////////////////////////////////////////////////////////////////////
mb.has_dram_buff = has_dram_buff(mb.zpu_ctrl);
if (mb.has_dram_buff) {
for (size_t i = 0; i < mboard_members_t::NUM_RADIOS; i++) {
static const size_t NUM_REGS = 8;
mb.dram_buff_ctrl[i] = dma_fifo_core_3000::make(
mb.zpu_ctrl,
SR_ADDR(SET0_BASE, ZPU_SR_DRAM_FIFO0+(i*NUM_REGS)),
SR_ADDR(SET0_BASE, ZPU_RB_DRAM_FIFO0+i));
mb.dram_buff_ctrl[i]->resize(X300_DRAM_FIFO_SIZE * i, X300_DRAM_FIFO_SIZE);
if (mb.dram_buff_ctrl[i]->ext_bist_supported()) {
UHD_MSG(status) << boost::format("Running BIST for DRAM FIFO %d... ") % i;
boost::uint32_t bisterr = mb.dram_buff_ctrl[i]->run_bist();
if (bisterr != 0) {
throw uhd::runtime_error(str(boost::format("DRAM FIFO BIST failed! (code: %d)\n") % bisterr));
} else {
double throughput = mb.dram_buff_ctrl[i]->get_bist_throughput(X300_BUS_CLOCK_RATE);
UHD_MSG(status) << (boost::format("pass (Throughput: %.1fMB/s)") % (throughput/1e6)) << std::endl;
}
} else {
if (mb.dram_buff_ctrl[i]->run_bist() != 0) {
throw uhd::runtime_error(str(boost::format("DRAM FIFO %d BIST failed!\n") % i));
}
}
}
}
////////////////////////////////////////////////////////////////////
// setup radios
////////////////////////////////////////////////////////////////////
this->setup_radio(mb_i, "A", dev_addr);
this->setup_radio(mb_i, "B", dev_addr);
////////////////////////////////////////////////////////////////////
// ADC test and cal
////////////////////////////////////////////////////////////////////
if (dev_addr.has_key("self_cal_adc_delay")) {
self_cal_adc_xfer_delay(mb, true /* Apply ADC delay */);
}
if (dev_addr.has_key("ext_adc_self_test")) {
extended_adc_test(mb, dev_addr.cast("ext_adc_self_test", 30));
} else if ( ! dev_addr.has_key("disable_adc_self_test") ) {
self_test_adcs(mb);
}
////////////////////////////////////////////////////////////////////
// front panel gpio
////////////////////////////////////////////////////////////////////
mb.fp_gpio = gpio_atr_3000::make(mb.radio_perifs[0].ctrl, radio::sr_addr(radio::FP_GPIO), radio::RB32_FP_GPIO);
BOOST_FOREACH(const gpio_attr_map_t::value_type attr, gpio_attr_map)
{
_tree->create(mb_path / "gpio" / "FP0" / attr.second)
.set(0)
.add_coerced_subscriber(boost::bind(&gpio_atr_3000::set_gpio_attr, mb.fp_gpio, attr.first, _1));
}
_tree->create(mb_path / "gpio" / "FP0" / "READBACK")
.set_publisher(boost::bind(&gpio_atr_3000::read_gpio, mb.fp_gpio));
////////////////////////////////////////////////////////////////////
// register the time keepers - only one can be the highlander
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "time" / "now")
.set_publisher(boost::bind(&time_core_3000::get_time_now, mb.radio_perifs[0].time64))
.add_coerced_subscriber(boost::bind(&x300_impl::sync_times, this, mb, _1))
.set(0.0);
_tree->create(mb_path / "time" / "pps")
.set_publisher(boost::bind(&time_core_3000::get_time_last_pps, mb.radio_perifs[0].time64))
.add_coerced_subscriber(boost::bind(&time_core_3000::set_time_next_pps, mb.radio_perifs[0].time64, _1))
.add_coerced_subscriber(boost::bind(&time_core_3000::set_time_next_pps, mb.radio_perifs[1].time64, _1));
////////////////////////////////////////////////////////////////////
// setup time sources and properties
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "time_source" / "value")
.set("internal")
.add_coerced_subscriber(boost::bind(&x300_impl::update_time_source, this, boost::ref(mb), _1));
static const std::vector time_sources = boost::assign::list_of("internal")("external")("gpsdo");
_tree->create >(mb_path / "time_source" / "options").set(time_sources);
//setup the time output, default to ON
_tree->create(mb_path / "time_source" / "output")
.add_coerced_subscriber(boost::bind(&x300_impl::set_time_source_out, this, boost::ref(mb), _1))
.set(true);
////////////////////////////////////////////////////////////////////
// setup clock sources and properties
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "clock_source" / "value")
.set("internal")
.add_coerced_subscriber(boost::bind(&x300_impl::update_clock_source, this, boost::ref(mb), _1));
static const std::vector clock_source_options = boost::assign::list_of("internal")("external")("gpsdo");
_tree->create >(mb_path / "clock_source" / "options").set(clock_source_options);
//setup external reference options. default to 10 MHz input reference
_tree->create(mb_path / "clock_source" / "external");
static const std::vector external_freq_options = boost::assign::list_of(10e6)(30.72e6)(200e6);
_tree->create >(mb_path / "clock_source" / "external" / "freq" / "options")
.set(external_freq_options);
_tree->create(mb_path / "clock_source" / "external" / "value")
.set(mb.clock->get_sysref_clock_rate());
// FIXME the external clock source settings need to be more robust
//setup the clock output, default to ON
_tree->create(mb_path / "clock_source" / "output")
.add_coerced_subscriber(boost::bind(&x300_clock_ctrl::set_ref_out, mb.clock, _1));
//initialize tick rate (must be done before setting time)
_tree->access(mb_path / "tick_rate")
.add_coerced_subscriber(boost::bind(&x300_impl::set_tick_rate, this, boost::ref(mb), _1))
.add_coerced_subscriber(boost::bind(&x300_impl::update_tick_rate, this, boost::ref(mb), _1))
.set(mb.clock->get_master_clock_rate());
////////////////////////////////////////////////////////////////////
// create frontend mapping
////////////////////////////////////////////////////////////////////
std::vector default_map(2, 0); default_map[1] = 1;
_tree->create >(mb_path / "rx_chan_dsp_mapping").set(default_map);
_tree->create >(mb_path / "tx_chan_dsp_mapping").set(default_map);
_tree->create(mb_path / "rx_subdev_spec")
.add_coerced_subscriber(boost::bind(&x300_impl::update_subdev_spec, this, "rx", mb_i, _1));
_tree->create(mb_path / "tx_subdev_spec")
.add_coerced_subscriber(boost::bind(&x300_impl::update_subdev_spec, this, "tx", mb_i, _1));
////////////////////////////////////////////////////////////////////
// and do the misc mboard sensors
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "sensors" / "ref_locked")
.set_publisher(boost::bind(&x300_impl::get_ref_locked, this, mb));
////////////////////////////////////////////////////////////////////
// do some post-init tasks
////////////////////////////////////////////////////////////////////
subdev_spec_t rx_fe_spec, tx_fe_spec;
rx_fe_spec.push_back(subdev_spec_pair_t("A",
_tree->list(mb_path / "dboards" / "A" / "rx_frontends").at(0)));
rx_fe_spec.push_back(subdev_spec_pair_t("B",
_tree->list(mb_path / "dboards" / "B" / "rx_frontends").at(0)));
tx_fe_spec.push_back(subdev_spec_pair_t("A",
_tree->list(mb_path / "dboards" / "A" / "tx_frontends").at(0)));
tx_fe_spec.push_back(subdev_spec_pair_t("B",
_tree->list(mb_path / "dboards" / "B" / "tx_frontends").at(0)));
_tree->access(mb_path / "rx_subdev_spec").set(rx_fe_spec);
_tree->access(mb_path / "tx_subdev_spec").set(tx_fe_spec);
mb.regmap_db = boost::make_shared();
mb.regmap_db->add(*mb.fw_regmap);
mb.regmap_db->add(*mb.radio_perifs[0].regmap);
mb.regmap_db->add(*mb.radio_perifs[1].regmap);
_tree->create(mb_path / "registers")
.set(mb.regmap_db);
mb.initialization_done = true;
}
x300_impl::~x300_impl(void)
{
try
{
BOOST_FOREACH(mboard_members_t &mb, _mb)
{
//Disable/reset ADC/DAC
mb.radio_perifs[0].regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::ADC_RESET, 1);
mb.radio_perifs[0].regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_RESET_N, 0);
mb.radio_perifs[0].regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_ENABLED, 0);
mb.radio_perifs[0].regmap->misc_outs_reg.flush();
mb.radio_perifs[1].regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_ENABLED, 0);
mb.radio_perifs[1].regmap->misc_outs_reg.flush();
//kill the claimer task and unclaim the device
mb.claimer_task.reset();
{ //Critical section
boost::mutex::scoped_lock(pcie_zpu_iface_registry_mutex);
mb.zpu_ctrl->poke32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_TIME), 0);
mb.zpu_ctrl->poke32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_SRC), 0);
//If the process is killed, the entire registry will disappear so we
//don't need to worry about unclean shutdowns here.
get_pcie_zpu_iface_registry().pop(mb.get_pri_eth().addr);
}
}
}
catch(...)
{
UHD_SAFE_CALL(throw;)
}
}
void x300_impl::setup_radio(const size_t mb_i, const std::string &slot_name, const uhd::device_addr_t &dev_addr)
{
const fs_path mb_path = "/mboards/"+boost::lexical_cast(mb_i);
UHD_ASSERT_THROW(mb_i < _mb.size());
mboard_members_t &mb = _mb[mb_i];
const size_t radio_index = mb.get_radio_index(slot_name);
radio_perifs_t &perif = mb.radio_perifs[radio_index];
UHD_MSG(status) << boost::format("Initialize Radio%d control...") % radio_index << std::endl;
////////////////////////////////////////////////////////////////////
// radio control
////////////////////////////////////////////////////////////////////
boost::uint8_t dest = (radio_index == 0)? X300_XB_DST_R0 : X300_XB_DST_R1;
boost::uint32_t ctrl_sid;
both_xports_t xport = this->make_transport(mb_i, dest, X300_RADIO_DEST_PREFIX_CTRL, device_addr_t(), ctrl_sid);
perif.ctrl = radio_ctrl_core_3000::make(mb.if_pkt_is_big_endian, xport.recv, xport.send, ctrl_sid, slot_name);
perif.regmap = boost::make_shared(radio_index);
perif.regmap->initialize(*perif.ctrl, true);
//Only Radio0 has the ADC/DAC reset bits. Those bits are reserved for Radio1
if (radio_index == 0) {
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::ADC_RESET, 1);
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_RESET_N, 0);
perif.regmap->misc_outs_reg.flush();
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::ADC_RESET, 0);
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_RESET_N, 1);
perif.regmap->misc_outs_reg.flush();
}
perif.regmap->misc_outs_reg.write(radio_regmap_t::misc_outs_reg_t::DAC_ENABLED, 1);
this->register_loopback_self_test(perif.ctrl);
////////////////////////////////////////////////////////////////
// Setup peripherals
////////////////////////////////////////////////////////////////
perif.spi = spi_core_3000::make(perif.ctrl, radio::sr_addr(radio::SPI), radio::RB32_SPI);
perif.adc = x300_adc_ctrl::make(perif.spi, DB_ADC_SEN);
perif.dac = x300_dac_ctrl::make(perif.spi, DB_DAC_SEN, mb.clock->get_master_clock_rate());
perif.leds = gpio_atr_3000::make_write_only(perif.ctrl, radio::sr_addr(radio::LEDS));
perif.leds->set_atr_mode(MODE_ATR, 0xFFFFFFFF);
perif.rx_fe = rx_frontend_core_200::make(perif.ctrl, radio::sr_addr(radio::RX_FRONT));
perif.rx_fe->set_dc_offset(rx_frontend_core_200::DEFAULT_DC_OFFSET_VALUE);
perif.rx_fe->set_dc_offset_auto(rx_frontend_core_200::DEFAULT_DC_OFFSET_ENABLE);
perif.tx_fe = tx_frontend_core_200::make(perif.ctrl, radio::sr_addr(radio::TX_FRONT));
perif.tx_fe->set_dc_offset(tx_frontend_core_200::DEFAULT_DC_OFFSET_VALUE);
perif.tx_fe->set_iq_balance(tx_frontend_core_200::DEFAULT_IQ_BALANCE_VALUE);
perif.framer = rx_vita_core_3000::make(perif.ctrl, radio::sr_addr(radio::RX_CTRL));
perif.ddc = rx_dsp_core_3000::make(perif.ctrl, radio::sr_addr(radio::RX_DSP));
perif.ddc->set_link_rate(10e9/8); //whatever
perif.ddc->set_tick_rate(mb.clock->get_master_clock_rate());
//The DRAM FIFO is treated as in internal radio FIFO for flow control purposes
tx_vita_core_3000::fc_monitor_loc fc_loc =
mb.has_dram_buff ? tx_vita_core_3000::FC_PRE_FIFO : tx_vita_core_3000::FC_PRE_RADIO;
perif.deframer = tx_vita_core_3000::make(perif.ctrl, radio::sr_addr(radio::TX_CTRL), fc_loc);
perif.duc = tx_dsp_core_3000::make(perif.ctrl, radio::sr_addr(radio::TX_DSP));
perif.duc->set_link_rate(10e9/8); //whatever
perif.duc->set_tick_rate(mb.clock->get_master_clock_rate());
////////////////////////////////////////////////////////////////////
// create time control objects
////////////////////////////////////////////////////////////////////
time_core_3000::readback_bases_type time64_rb_bases;
time64_rb_bases.rb_now = radio::RB64_TIME_NOW;
time64_rb_bases.rb_pps = radio::RB64_TIME_PPS;
perif.time64 = time_core_3000::make(perif.ctrl, radio::sr_addr(radio::TIME), time64_rb_bases);
//Capture delays are calibrated every time. The status is only printed is the user
//asks to run the xfer self cal using "self_cal_adc_delay"
self_cal_adc_capture_delay(mb, radio_index, dev_addr.has_key("self_cal_adc_delay"));
_tree->access(mb_path / "time" / "cmd")
.add_coerced_subscriber(boost::bind(&radio_ctrl_core_3000::set_time, perif.ctrl, _1));
////////////////////////////////////////////////////////////////
// create codec control objects
////////////////////////////////////////////////////////////////
_tree->create(mb_path / "rx_codecs" / slot_name / "gains"); //phony property so this dir exists
_tree->create(mb_path / "tx_codecs" / slot_name / "gains"); //phony property so this dir exists
_tree->create(mb_path / "rx_codecs" / slot_name / "name").set("ads62p48");
_tree->create(mb_path / "tx_codecs" / slot_name / "name").set("ad9146");
_tree->create(mb_path / "rx_codecs" / slot_name / "gains" / "digital" / "range").set(meta_range_t(0, 6.0, 0.5));
_tree->create(mb_path / "rx_codecs" / slot_name / "gains" / "digital" / "value")
.add_coerced_subscriber(boost::bind(&x300_adc_ctrl::set_gain, perif.adc, _1)).set(0);
////////////////////////////////////////////////////////////////////
// front end corrections
////////////////////////////////////////////////////////////////////
perif.rx_fe->populate_subtree(_tree->subtree(mb_path / "rx_frontends" / slot_name));
perif.tx_fe->populate_subtree(_tree->subtree(mb_path / "tx_frontends" / slot_name));
////////////////////////////////////////////////////////////////////
// connect rx dsp control objects
////////////////////////////////////////////////////////////////////
const fs_path rx_dsp_path = mb_path / "rx_dsps" / str(boost::format("%u") % radio_index);
perif.ddc->populate_subtree(_tree->subtree(rx_dsp_path));
_tree->access(rx_dsp_path / "rate" / "value")
.add_coerced_subscriber(boost::bind(&x300_impl::update_rx_samp_rate, this, boost::ref(mb), radio_index, _1))
;
_tree->create(rx_dsp_path / "stream_cmd")
.add_coerced_subscriber(boost::bind(&rx_vita_core_3000::issue_stream_command, perif.framer, _1));
////////////////////////////////////////////////////////////////////
// connect tx dsp control objects
////////////////////////////////////////////////////////////////////
const fs_path tx_dsp_path = mb_path / "tx_dsps" / str(boost::format("%u") % radio_index);
perif.duc->populate_subtree(_tree->subtree(tx_dsp_path));
_tree->access(tx_dsp_path / "rate" / "value")
.add_coerced_subscriber(boost::bind(&x300_impl::update_tx_samp_rate, this, boost::ref(mb), radio_index, _1))
;
////////////////////////////////////////////////////////////////////
// create RF frontend interfacing
////////////////////////////////////////////////////////////////////
const fs_path db_path = (mb_path / "dboards" / slot_name);
const size_t j = (slot_name == "B")? 0x2 : 0x0;
_tree->create(db_path / "rx_eeprom")
.set(mb.db_eeproms[X300_DB0_RX_EEPROM | j])
.add_coerced_subscriber(boost::bind(&x300_impl::set_db_eeprom, this, mb.zpu_i2c, (0x50 | X300_DB0_RX_EEPROM | j), _1));
_tree->create(db_path / "tx_eeprom")
.set(mb.db_eeproms[X300_DB0_TX_EEPROM | j])
.add_coerced_subscriber(boost::bind(&x300_impl::set_db_eeprom, this, mb.zpu_i2c, (0x50 | X300_DB0_TX_EEPROM | j), _1));
_tree->create(db_path / "gdb_eeprom")
.set(mb.db_eeproms[X300_DB0_GDB_EEPROM | j])
.add_coerced_subscriber(boost::bind(&x300_impl::set_db_eeprom, this, mb.zpu_i2c, (0x50 | X300_DB0_GDB_EEPROM | j), _1));
//create a new dboard interface
x300_dboard_iface_config_t db_config;
db_config.gpio = db_gpio_atr_3000::make(perif.ctrl, radio::sr_addr(radio::GPIO), radio::RB32_GPIO);
db_config.spi = perif.spi;
db_config.rx_spi_slaveno = DB_RX_SEN;
db_config.tx_spi_slaveno = DB_TX_SEN;
db_config.i2c = mb.zpu_i2c;
db_config.clock = mb.clock;
db_config.rx_dsp = mb.radio_perifs[radio_index].ddc;
db_config.which_rx_clk = (slot_name == "A")? X300_CLOCK_WHICH_DB0_RX : X300_CLOCK_WHICH_DB1_RX;
db_config.which_tx_clk = (slot_name == "A")? X300_CLOCK_WHICH_DB0_TX : X300_CLOCK_WHICH_DB1_TX;
db_config.dboard_slot = (slot_name == "A")? 0 : 1;
db_config.cmd_time_ctrl = perif.ctrl;
//create a new dboard manager
_dboard_managers[db_path] = dboard_manager::make(
mb.db_eeproms[X300_DB0_RX_EEPROM | j].id,
mb.db_eeproms[X300_DB0_TX_EEPROM | j].id,
mb.db_eeproms[X300_DB0_GDB_EEPROM | j].id,
x300_make_dboard_iface(db_config),
_tree->subtree(db_path)
);
//now that dboard is created -- register into rx antenna event
const std::string fe_name = _tree->list(db_path / "rx_frontends").front();
_tree->access(db_path / "rx_frontends" / fe_name / "antenna" / "value")
.add_coerced_subscriber(boost::bind(&x300_impl::update_atr_leds, this, mb.radio_perifs[radio_index].leds, _1));
this->update_atr_leds(mb.radio_perifs[radio_index].leds, ""); //init anyway, even if never called
//bind frontend corrections to the dboard freq props
const fs_path db_tx_fe_path = db_path / "tx_frontends";
BOOST_FOREACH(const std::string &name, _tree->list(db_tx_fe_path)) {
_tree->access(db_tx_fe_path / name / "freq" / "value")
.add_coerced_subscriber(boost::bind(&x300_impl::set_tx_fe_corrections, this, mb_path, slot_name, _1));
}
const fs_path db_rx_fe_path = db_path / "rx_frontends";
BOOST_FOREACH(const std::string &name, _tree->list(db_rx_fe_path)) {
_tree->access(db_rx_fe_path / name / "freq" / "value")
.add_coerced_subscriber(boost::bind(&x300_impl::set_rx_fe_corrections, this, mb_path, slot_name, _1));
}
}
void x300_impl::set_rx_fe_corrections(const uhd::fs_path &mb_path, const std::string &fe_name, const double lo_freq)
{
if(not _ignore_cal_file){
apply_rx_fe_corrections(this->get_tree()->subtree(mb_path), fe_name, lo_freq);
}
}
void x300_impl::set_tx_fe_corrections(const uhd::fs_path &mb_path, const std::string &fe_name, const double lo_freq)
{
if(not _ignore_cal_file){
apply_tx_fe_corrections(this->get_tree()->subtree(mb_path), fe_name, lo_freq);
}
}
boost::uint32_t get_pcie_dma_channel(boost::uint8_t destination, boost::uint8_t prefix)
{
static const boost::uint32_t RADIO_GRP_SIZE = 3;
static const boost::uint32_t RADIO0_GRP = 0;
static const boost::uint32_t RADIO1_GRP = 1;
boost::uint32_t radio_grp = (destination == X300_XB_DST_R0) ? RADIO0_GRP : RADIO1_GRP;
return ((radio_grp * RADIO_GRP_SIZE) + prefix);
}
x300_impl::both_xports_t x300_impl::make_transport(
const size_t mb_index,
const boost::uint8_t& destination,
const boost::uint8_t& prefix,
const uhd::device_addr_t& args,
boost::uint32_t& sid)
{
mboard_members_t &mb = _mb[mb_index];
both_xports_t xports;
sid_config_t config;
config.router_addr_there = X300_DEVICE_THERE;
config.dst_prefix = prefix;
config.router_dst_there = destination;
// Choose the endpoint based on the destination
size_t endpoint = 0;
if (destination == X300_XB_DST_R1) {
if (mb.eth_conns.size() > 1)
endpoint = 1;
}
// Decide on the IP/Interface pair based on the endpoint index
std::string interface_addr = mb.eth_conns[endpoint].addr;
config.iface_index = mb.eth_conns[endpoint].type;
sid = this->allocate_sid(mb, config);
static const uhd::device_addr_t DEFAULT_XPORT_ARGS;
const uhd::device_addr_t& xport_args =
(prefix != X300_RADIO_DEST_PREFIX_CTRL) ? args : DEFAULT_XPORT_ARGS;
zero_copy_xport_params default_buff_args;
if (mb.xport_path == "nirio") {
default_buff_args.send_frame_size =
(prefix == X300_RADIO_DEST_PREFIX_TX)
? X300_PCIE_TX_DATA_FRAME_SIZE
: X300_PCIE_MSG_FRAME_SIZE;
default_buff_args.recv_frame_size =
(prefix == X300_RADIO_DEST_PREFIX_RX)
? X300_PCIE_RX_DATA_FRAME_SIZE
: X300_PCIE_MSG_FRAME_SIZE;
default_buff_args.num_send_frames =
(prefix == X300_RADIO_DEST_PREFIX_TX)
? X300_PCIE_DATA_NUM_FRAMES
: X300_PCIE_MSG_NUM_FRAMES;
default_buff_args.num_recv_frames =
(prefix == X300_RADIO_DEST_PREFIX_RX)
? X300_PCIE_DATA_NUM_FRAMES
: X300_PCIE_MSG_NUM_FRAMES;
xports.recv = nirio_zero_copy::make(
mb.rio_fpga_interface,
get_pcie_dma_channel(destination, prefix),
default_buff_args,
xport_args);
xports.send = xports.recv;
//For the nirio transport, buffer size is depends on the frame size and num frames
xports.recv_buff_size = xports.recv->get_num_recv_frames() * xports.recv->get_recv_frame_size();
xports.send_buff_size = xports.send->get_num_send_frames() * xports.send->get_send_frame_size();
} else if (mb.xport_path == "eth") {
/* Determine what the recommended frame size is for this
* connection type.*/
size_t eth_data_rec_frame_size = 0;
fs_path mboard_path = fs_path("/mboards/"+boost::lexical_cast(mb_index) / "link_max_rate");
if (mb.loaded_fpga_image.substr(0,2) == "HG") {
size_t max_link_rate = 0;
if (config.iface_index == X300_IFACE_ETH0) {
eth_data_rec_frame_size = X300_1GE_DATA_FRAME_MAX_SIZE;
max_link_rate += X300_MAX_RATE_1GIGE;
} else if (config.iface_index == X300_IFACE_ETH1) {
eth_data_rec_frame_size = X300_10GE_DATA_FRAME_MAX_SIZE;
max_link_rate += X300_MAX_RATE_10GIGE;
}
_tree->access(mboard_path).set(max_link_rate);
} else if (mb.loaded_fpga_image.substr(0,2) == "XG") {
eth_data_rec_frame_size = X300_10GE_DATA_FRAME_MAX_SIZE;
size_t max_link_rate = X300_MAX_RATE_10GIGE;
max_link_rate *= mb.eth_conns.size();
_tree->access(mboard_path).set(max_link_rate);
}
if (eth_data_rec_frame_size == 0) {
throw uhd::runtime_error("Unable to determine ETH link type.");
}
/* Print a warning if the system's max available frame size is less than the most optimal
* frame size for this type of connection. */
if (_max_frame_sizes.send_frame_size < eth_data_rec_frame_size) {
UHD_MSG(warning)
<< boost::format("For this connection, UHD recommends a send frame size of at least %lu for best\nperformance, but your system's MTU will only allow %lu.")
% eth_data_rec_frame_size
% _max_frame_sizes.send_frame_size
<< std::endl
<< "This will negatively impact your maximum achievable sample rate."
<< std::endl;
}
if (_max_frame_sizes.recv_frame_size < eth_data_rec_frame_size) {
UHD_MSG(warning)
<< boost::format("For this connection, UHD recommends a receive frame size of at least %lu for best\nperformance, but your system's MTU will only allow %lu.")
% eth_data_rec_frame_size
% _max_frame_sizes.recv_frame_size
<< std::endl
<< "This will negatively impact your maximum achievable sample rate."
<< std::endl;
}
size_t system_max_send_frame_size = (size_t) _max_frame_sizes.send_frame_size;
size_t system_max_recv_frame_size = (size_t) _max_frame_sizes.recv_frame_size;
// Make sure frame sizes do not exceed the max available value supported by UHD
default_buff_args.send_frame_size =
(prefix == X300_RADIO_DEST_PREFIX_TX)
? std::min(system_max_send_frame_size, X300_10GE_DATA_FRAME_MAX_SIZE)
: std::min(system_max_send_frame_size, X300_ETH_MSG_FRAME_SIZE);
default_buff_args.recv_frame_size =
(prefix == X300_RADIO_DEST_PREFIX_RX)
? std::min(system_max_recv_frame_size, X300_10GE_DATA_FRAME_MAX_SIZE)
: std::min(system_max_recv_frame_size, X300_ETH_MSG_FRAME_SIZE);
default_buff_args.num_send_frames =
(prefix == X300_RADIO_DEST_PREFIX_TX)
? X300_ETH_DATA_NUM_FRAMES
: X300_ETH_MSG_NUM_FRAMES;
default_buff_args.num_recv_frames =
(prefix == X300_RADIO_DEST_PREFIX_RX)
? X300_ETH_DATA_NUM_FRAMES
: X300_ETH_MSG_NUM_FRAMES;
//make a new transport - fpga has no idea how to talk to us on this yet
udp_zero_copy::buff_params buff_params;
xports.recv = udp_zero_copy::make(
interface_addr,
BOOST_STRINGIZE(X300_VITA_UDP_PORT),
default_buff_args,
buff_params,
xport_args);
// Create a threaded transport for the receive chain only
// Note that this shouldn't affect PCIe
if (prefix == X300_RADIO_DEST_PREFIX_RX) {
xports.recv = zero_copy_recv_offload::make(
xports.recv,
X300_THREAD_BUFFER_TIMEOUT
);
}
xports.send = xports.recv;
//For the UDP transport the buffer size if the size of the socket buffer
//in the kernel
xports.recv_buff_size = buff_params.recv_buff_size;
xports.send_buff_size = buff_params.send_buff_size;
//clear the ethernet dispatcher's udp port
//NOT clearing this, the dispatcher is now intelligent
//_zpu_ctrl->poke32(SR_ADDR(SET0_BASE, (ZPU_SR_ETHINT0+8+3)), 0);
//send a mini packet with SID into the ZPU
//ZPU will reprogram the ethernet framer
UHD_LOG << "programming packet for new xport on "
<< mb.get_pri_eth().addr << std::hex << "sid 0x" << sid << std::dec << std::endl;
//YES, get a __send__ buffer from the __recv__ socket
//-- this is the only way to program the framer for recv:
managed_send_buffer::sptr buff = xports.recv->get_send_buff();
buff->cast()[0] = 0; //eth dispatch looks for != 0
buff->cast()[1] = uhd::htonx(sid);
buff->commit(8);
buff.reset();
//reprogram the ethernet dispatcher's udp port (should be safe to always set)
UHD_LOG << "reprogram the ethernet dispatcher's udp port" << std::endl;
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, (ZPU_SR_ETHINT0+8+3)), X300_VITA_UDP_PORT);
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, (ZPU_SR_ETHINT1+8+3)), X300_VITA_UDP_PORT);
//Do a peek to an arbitrary address to guarantee that the
//ethernet framer has been programmed before we return.
mb.zpu_ctrl->peek32(0);
}
return xports;
}
boost::uint32_t x300_impl::allocate_sid(mboard_members_t &mb, const sid_config_t &config)
{
const std::string &xport_path = mb.xport_path;
const boost::uint32_t stream = (config.dst_prefix | (config.router_dst_there << 2)) & 0xff;
boost::uint8_t sid_ret_addr = X300_SRC_ADDR_ETH0;
boost::uint32_t xb_port = X300_XB_DST_E0;
// Use the interface index to decide on the ethernet port
if (config.iface_index == X300_IFACE_ETH1) {
sid_ret_addr = X300_SRC_ADDR_ETH1;
xb_port = X300_XB_DST_E1;
}
// Ensure we choose the right port in the case of NI-RIO
if (xport_path == "nirio") {
xb_port = X300_XB_DST_PCI;
}
int int_sid_ret_addr = int(sid_ret_addr);
const boost::uint32_t sid = 0
| (sid_ret_addr << 24)
| (_sid_framer << 16)
| (config.router_addr_there << 8)
| (stream << 0)
;
UHD_LOG << std::hex
<< " sid 0x" << sid
<< " return address 0x" << int_sid_ret_addr
<< " framer 0x" << _sid_framer
<< " stream 0x" << stream
<< " router_dst_there 0x" << int(config.router_dst_there)
<< " router_addr_there 0x" << int(config.router_addr_there)
<< std::dec << std::endl;
// Program the X300 to recognise it's own local address.
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, ZPU_SR_XB_LOCAL), config.router_addr_there);
// Program CAM entry for outgoing packets matching a X300 resource (for example a Radio)
// This type of packet does matches the XB_LOCAL address and is looked up in the upper half of the CAM
mb.zpu_ctrl->poke32(SR_ADDR(SETXB_BASE, 256 + (stream)), config.router_dst_there);
// Program CAM entry for returning packets to us (for example GR host via Eth0)
// This type of packet does not match the XB_LOCAL address and is looked up in the lower half of the CAM
mb.zpu_ctrl->poke32(SR_ADDR(SETXB_BASE, 0 + (sid_ret_addr)), xb_port);
if (xport_path == "nirio") {
boost::uint32_t router_config_word = ((_sid_framer & 0xff) << 16) | //Return SID
get_pcie_dma_channel(config.router_dst_there, config.dst_prefix); //Dest
mb.rio_fpga_interface->get_kernel_proxy()->poke(PCIE_ROUTER_REG(0), router_config_word);
}
UHD_LOG << std::hex
<< "done router config for sid 0x" << sid
<< std::dec << std::endl;
//increment for next setup
_sid_framer++;
return sid;
}
void x300_impl::update_atr_leds(gpio_atr_3000::sptr leds, const std::string &rx_ant)
{
const bool is_txrx = (rx_ant == "TX/RX");
const int rx_led = (1 << 2);
const int tx_led = (1 << 1);
const int txrx_led = (1 << 0);
leds->set_atr_reg(ATR_REG_IDLE, 0);
leds->set_atr_reg(ATR_REG_RX_ONLY, is_txrx? txrx_led : rx_led);
leds->set_atr_reg(ATR_REG_TX_ONLY, tx_led);
leds->set_atr_reg(ATR_REG_FULL_DUPLEX, rx_led | tx_led);
}
void x300_impl::set_tick_rate(mboard_members_t &mb, const double rate)
{
BOOST_FOREACH(radio_perifs_t &perif, mb.radio_perifs) {
perif.ctrl->set_tick_rate(rate);
perif.time64->set_tick_rate(rate);
perif.framer->set_tick_rate(rate);
perif.ddc->set_tick_rate(rate);
perif.duc->set_tick_rate(rate);
}
}
void x300_impl::register_loopback_self_test(wb_iface::sptr iface)
{
bool test_fail = false;
UHD_MSG(status) << "Performing register loopback test... " << std::flush;
size_t hash = size_t(time(NULL));
for (size_t i = 0; i < 100; i++)
{
boost::hash_combine(hash, i);
iface->poke32(radio::sr_addr(radio::TEST), boost::uint32_t(hash));
test_fail = iface->peek32(radio::RB32_TEST) != boost::uint32_t(hash);
if (test_fail) break; //exit loop on any failure
}
UHD_MSG(status) << ((test_fail)? " fail" : "pass") << std::endl;
}
void x300_impl::radio_loopback(wb_iface::sptr iface, const bool on)
{
iface->poke32(radio::sr_addr(radio::LOOPBACK), (on ? 0x1 : 0x0));
UHD_MSG(status) << ((on)? "Radio Loopback On" : "Radio Loopback Off") << std::endl;
}
/***********************************************************************
* clock and time control logic
**********************************************************************/
void x300_impl::set_time_source_out(mboard_members_t &mb, const bool enb)
{
mb.fw_regmap->clock_ctrl_reg.write(fw_regmap_t::clk_ctrl_reg_t::PPS_OUT_EN, enb?1:0);
}
void x300_impl::update_clock_source(mboard_members_t &mb, const std::string &source)
{
//Optimize for the case when the current source is internal and we are trying
//to set it to internal. This is the only case where we are guaranteed that
//the clock has not gone away so we can skip setting the MUX and reseting the LMK.
const bool reconfigure_clks = (mb.current_refclk_src != "internal") or (source != "internal");
if (reconfigure_clks) {
//Update the clock MUX on the motherboard to select the requested source
if (source == "internal") {
mb.fw_regmap->clock_ctrl_reg.set(fw_regmap_t::clk_ctrl_reg_t::CLK_SOURCE, fw_regmap_t::clk_ctrl_reg_t::SRC_INTERNAL);
mb.fw_regmap->clock_ctrl_reg.set(fw_regmap_t::clk_ctrl_reg_t::TCXO_EN, 1);
} else if (source == "external") {
mb.fw_regmap->clock_ctrl_reg.set(fw_regmap_t::clk_ctrl_reg_t::CLK_SOURCE, fw_regmap_t::clk_ctrl_reg_t::SRC_EXTERNAL);
mb.fw_regmap->clock_ctrl_reg.set(fw_regmap_t::clk_ctrl_reg_t::TCXO_EN, 0);
} else if (source == "gpsdo") {
mb.fw_regmap->clock_ctrl_reg.set(fw_regmap_t::clk_ctrl_reg_t::CLK_SOURCE, fw_regmap_t::clk_ctrl_reg_t::SRC_GPSDO);
mb.fw_regmap->clock_ctrl_reg.set(fw_regmap_t::clk_ctrl_reg_t::TCXO_EN, 0);
} else {
throw uhd::key_error("update_clock_source: unknown source: " + source);
}
mb.fw_regmap->clock_ctrl_reg.flush();
//Reset the LMK to make sure it re-locks to the new reference
mb.clock->reset_clocks();
}
//Wait for the LMK to lock (always, as a sanity check that the clock is useable)
//* Currently the LMK can take as long as 30 seconds to lock to a reference but we don't
//* want to wait that long during initialization.
//TODO: Need to verify timeout and settings to make sure lock can be achieved in < 1.0 seconds
double timeout = mb.initialization_done ? 30.0 : 1.0;
//The programming code in x300_clock_ctrl is not compatible with revs <= 4 and may
//lead to locking issues. So, disable the ref-locked check for older (unsupported) boards.
if (mb.hw_rev > 4) {
if (not wait_for_clk_locked(mb, fw_regmap_t::clk_status_reg_t::LMK_LOCK, timeout)) {
//failed to lock on reference
if (mb.initialization_done) {
throw uhd::runtime_error((boost::format("Reference Clock PLL failed to lock to %s source.") % source).str());
} else {
//TODO: Re-enable this warning when we figure out a reliable lock time
//UHD_MSG(warning) << "Reference clock failed to lock to " + source + " during device initialization. " <<
// "Check for the lock before operation or ignore this warning if using another clock source." << std::endl;
}
}
}
if (reconfigure_clks) {
//Reset the radio clock PLL in the FPGA
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, ZPU_SR_SW_RST), ZPU_SR_SW_RST_RADIO_CLK_PLL);
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, ZPU_SR_SW_RST), 0);
//Wait for radio clock PLL to lock
if (not wait_for_clk_locked(mb, fw_regmap_t::clk_status_reg_t::RADIO_CLK_LOCK, 0.01)) {
throw uhd::runtime_error((boost::format("Reference Clock PLL in FPGA failed to lock to %s source.") % source).str());
}
//Reset the IDELAYCTRL used to calibrate the data interface delays
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, ZPU_SR_SW_RST), ZPU_SR_SW_RST_ADC_IDELAYCTRL);
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, ZPU_SR_SW_RST), 0);
//Wait for the ADC IDELAYCTRL to be ready
if (not wait_for_clk_locked(mb, fw_regmap_t::clk_status_reg_t::IDELAYCTRL_LOCK, 0.01)) {
throw uhd::runtime_error((boost::format("ADC Calibration Clock in FPGA failed to lock to %s source.") % source).str());
}
// Reset ADCs and DACs
for (size_t r = 0; r < mboard_members_t::NUM_RADIOS; r++) {
radio_perifs_t &perif = mb.radio_perifs[r];
if (perif.regmap && r==0) { //ADC/DAC reset lines only exist in Radio0
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::ADC_RESET, 1);
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_RESET_N, 0);
perif.regmap->misc_outs_reg.flush();
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::ADC_RESET, 0);
perif.regmap->misc_outs_reg.set(radio_regmap_t::misc_outs_reg_t::DAC_RESET_N, 1);
perif.regmap->misc_outs_reg.flush();
}
if (perif.adc) perif.adc->reset();
if (perif.dac) perif.dac->reset();
}
}
//Update cache value
mb.current_refclk_src = source;
}
void x300_impl::update_time_source(mboard_members_t &mb, const std::string &source)
{
if (source == "internal") {
mb.fw_regmap->clock_ctrl_reg.write(fw_regmap_t::clk_ctrl_reg_t::PPS_SELECT, fw_regmap_t::clk_ctrl_reg_t::SRC_INTERNAL);
} else if (source == "external") {
mb.fw_regmap->clock_ctrl_reg.write(fw_regmap_t::clk_ctrl_reg_t::PPS_SELECT, fw_regmap_t::clk_ctrl_reg_t::SRC_EXTERNAL);
} else if (source == "gpsdo") {
mb.fw_regmap->clock_ctrl_reg.write(fw_regmap_t::clk_ctrl_reg_t::PPS_SELECT, fw_regmap_t::clk_ctrl_reg_t::SRC_GPSDO);
} else {
throw uhd::key_error("update_time_source: unknown source: " + source);
}
/* TODO - Implement intelligent PPS detection
//check for valid pps
if (!is_pps_present(mb)) {
throw uhd::runtime_error((boost::format("The %d PPS was not detected. Please check the PPS source and try again.") % source).str());
}
*/
}
void x300_impl::sync_times(mboard_members_t &mb, const uhd::time_spec_t& t)
{
BOOST_FOREACH(radio_perifs_t &perif, mb.radio_perifs)
perif.time64->set_time_sync(t);
mb.fw_regmap->clock_ctrl_reg.write(fw_regmap_t::clk_ctrl_reg_t::TIME_SYNC, 1);
mb.fw_regmap->clock_ctrl_reg.write(fw_regmap_t::clk_ctrl_reg_t::TIME_SYNC, 0);
}
bool x300_impl::wait_for_clk_locked(mboard_members_t& mb, boost::uint32_t which, double timeout)
{
boost::system_time timeout_time = boost::get_system_time() + boost::posix_time::milliseconds(timeout * 1000.0);
do {
if (mb.fw_regmap->clock_status_reg.read(which)==1)
return true;
boost::this_thread::sleep(boost::posix_time::milliseconds(1));
} while (boost::get_system_time() < timeout_time);
//Check one last time
return (mb.fw_regmap->clock_status_reg.read(which)==1);
}
sensor_value_t x300_impl::get_ref_locked(mboard_members_t& mb)
{
mb.fw_regmap->clock_status_reg.refresh();
const bool lock = (mb.fw_regmap->clock_status_reg.get(fw_regmap_t::clk_status_reg_t::LMK_LOCK)==1) &&
(mb.fw_regmap->clock_status_reg.get(fw_regmap_t::clk_status_reg_t::RADIO_CLK_LOCK)==1) &&
(mb.fw_regmap->clock_status_reg.get(fw_regmap_t::clk_status_reg_t::IDELAYCTRL_LOCK)==1);
return sensor_value_t("Ref", lock, "locked", "unlocked");
}
bool x300_impl::is_pps_present(mboard_members_t& mb)
{
// The ZPU_RB_CLK_STATUS_PPS_DETECT bit toggles with each rising edge of the PPS.
// We monitor it for up to 1.5 seconds looking for it to toggle.
boost::uint32_t pps_detect = mb.fw_regmap->clock_status_reg.read(fw_regmap_t::clk_status_reg_t::PPS_DETECT);
for (int i = 0; i < 15; i++)
{
boost::this_thread::sleep(boost::posix_time::milliseconds(100));
if (pps_detect != mb.fw_regmap->clock_status_reg.read(fw_regmap_t::clk_status_reg_t::PPS_DETECT))
return true;
}
return false;
}
/***********************************************************************
* eeprom
**********************************************************************/
void x300_impl::set_db_eeprom(i2c_iface::sptr i2c, const size_t addr, const uhd::usrp::dboard_eeprom_t &db_eeprom)
{
db_eeprom.store(*i2c, addr);
}
void x300_impl::set_mb_eeprom(i2c_iface::sptr i2c, const mboard_eeprom_t &mb_eeprom)
{
i2c_iface::sptr eeprom16 = i2c->eeprom16();
mb_eeprom.commit(*eeprom16, "X300");
}
/***********************************************************************
* claimer logic
**********************************************************************/
void x300_impl::claimer_loop(wb_iface::sptr iface)
{
{ //Critical section
boost::mutex::scoped_lock(claimer_mutex);
iface->poke32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_TIME), uint32_t(time(NULL)));
iface->poke32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_SRC), get_process_hash());
}
boost::this_thread::sleep(boost::posix_time::milliseconds(1000)); //1 second
}
bool x300_impl::is_claimed(wb_iface::sptr iface)
{
boost::mutex::scoped_lock(claimer_mutex);
//If timed out then device is definitely unclaimed
if (iface->peek32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_STATUS)) == 0)
return false;
//otherwise check claim src to determine if another thread with the same src has claimed the device
return iface->peek32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_SRC)) != get_process_hash();
}
/***********************************************************************
* Frame size detection
**********************************************************************/
x300_impl::frame_size_t x300_impl::determine_max_frame_size(const std::string &addr,
const frame_size_t &user_frame_size)
{
udp_simple::sptr udp = udp_simple::make_connected(addr,
BOOST_STRINGIZE(X300_MTU_DETECT_UDP_PORT));
std::vector buffer(std::max(user_frame_size.recv_frame_size, user_frame_size.send_frame_size));
x300_mtu_t *request = reinterpret_cast(&buffer.front());
static const double echo_timeout = 0.020; //20 ms
//test holler - check if its supported in this fw version
request->flags = uhd::htonx(X300_MTU_DETECT_ECHO_REQUEST);
request->size = uhd::htonx(sizeof(x300_mtu_t));
udp->send(boost::asio::buffer(buffer, sizeof(x300_mtu_t)));
udp->recv(boost::asio::buffer(buffer), echo_timeout);
if (!(uhd::ntohx(request->flags) & X300_MTU_DETECT_ECHO_REPLY))
throw uhd::not_implemented_error("Holler protocol not implemented");
size_t min_recv_frame_size = sizeof(x300_mtu_t);
size_t max_recv_frame_size = user_frame_size.recv_frame_size;
size_t min_send_frame_size = sizeof(x300_mtu_t);
size_t max_send_frame_size = user_frame_size.send_frame_size;
UHD_MSG(status) << "Determining maximum frame size... ";
while (min_recv_frame_size < max_recv_frame_size)
{
size_t test_frame_size = (max_recv_frame_size/2 + min_recv_frame_size/2 + 3) & ~3;
request->flags = uhd::htonx(X300_MTU_DETECT_ECHO_REQUEST);
request->size = uhd::htonx(test_frame_size);
udp->send(boost::asio::buffer(buffer, sizeof(x300_mtu_t)));
size_t len = udp->recv(boost::asio::buffer(buffer), echo_timeout);
if (len >= test_frame_size)
min_recv_frame_size = test_frame_size;
else
max_recv_frame_size = test_frame_size - 4;
}
if(min_recv_frame_size < IP_PROTOCOL_MIN_MTU_SIZE-IP_PROTOCOL_UDP_PLUS_IP_HEADER) {
throw uhd::runtime_error("System receive MTU size is less than the minimum required by the IP protocol.");
}
while (min_send_frame_size < max_send_frame_size)
{
size_t test_frame_size = (max_send_frame_size/2 + min_send_frame_size/2 + 3) & ~3;
request->flags = uhd::htonx(X300_MTU_DETECT_ECHO_REQUEST);
request->size = uhd::htonx(sizeof(x300_mtu_t));
udp->send(boost::asio::buffer(buffer, test_frame_size));
size_t len = udp->recv(boost::asio::buffer(buffer), echo_timeout);
if (len >= sizeof(x300_mtu_t))
len = uhd::ntohx(request->size);
if (len >= test_frame_size)
min_send_frame_size = test_frame_size;
else
max_send_frame_size = test_frame_size - 4;
}
if(min_send_frame_size < IP_PROTOCOL_MIN_MTU_SIZE-IP_PROTOCOL_UDP_PLUS_IP_HEADER) {
throw uhd::runtime_error("System send MTU size is less than the minimum required by the IP protocol.");
}
frame_size_t frame_size;
// There are cases when NICs accept oversized packets, in which case we'd falsely
// detect a larger-than-possible frame size. A safe and sensible value is the minimum
// of the recv and send frame sizes.
frame_size.recv_frame_size = std::min(min_recv_frame_size, min_send_frame_size);
frame_size.send_frame_size = std::min(min_recv_frame_size, min_send_frame_size);
UHD_MSG(status) << frame_size.send_frame_size << " bytes." << std::endl;
return frame_size;
}
/***********************************************************************
* compat checks
**********************************************************************/
void x300_impl::check_fw_compat(const fs_path &mb_path, wb_iface::sptr iface)
{
boost::uint32_t compat_num = iface->peek32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_COMPAT_NUM));
boost::uint32_t compat_major = (compat_num >> 16);
boost::uint32_t compat_minor = (compat_num & 0xffff);
if (compat_major != X300_FW_COMPAT_MAJOR)
{
throw uhd::runtime_error(str(boost::format(
"Expected firmware compatibility number %d.%d, but got %d.%d:\n"
"The firmware build is not compatible with the host code build.\n"
"%s"
) % int(X300_FW_COMPAT_MAJOR) % int(X300_FW_COMPAT_MINOR)
% compat_major % compat_minor % print_utility_error("uhd_images_downloader.py")));
}
_tree->create(mb_path / "fw_version").set(str(boost::format("%u.%u")
% compat_major % compat_minor));
}
void x300_impl::check_fpga_compat(const fs_path &mb_path, const mboard_members_t &members)
{
boost::uint32_t compat_num = members.zpu_ctrl->peek32(SR_ADDR(SET0_BASE, ZPU_RB_COMPAT_NUM));
boost::uint32_t compat_major = (compat_num >> 16);
boost::uint32_t compat_minor = (compat_num & 0xffff);
if (compat_major != X300_FPGA_COMPAT_MAJOR)
{
std::string image_loader_path = (fs::path(uhd::get_pkg_path()) / "bin" / "uhd_image_loader").string();
std::string image_loader_cmd = str(boost::format("\"%s\" --args=\"type=x300,%s=%s\"")
% image_loader_path
% (members.xport_path == "eth" ? "addr"
: "resource")
% members.get_pri_eth().addr);
throw uhd::runtime_error(str(boost::format(
"Expected FPGA compatibility number %d, but got %d:\n"
"The FPGA image on your device is not compatible with this host code build.\n"
"Download the appropriate FPGA images for this version of UHD.\n"
"%s\n\n"
"Then burn a new image to the on-board flash storage of your\n"
"USRP X3xx device using the image loader utility. Use this command:\n\n%s\n\n"
"For more information, refer to the UHD manual:\n\n"
" http://files.ettus.com/manual/page_usrp_x3x0.html#x3x0_flash"
) % int(X300_FPGA_COMPAT_MAJOR) % compat_major
% print_utility_error("uhd_images_downloader.py")
% image_loader_cmd));
}
_tree->create(mb_path / "fpga_version").set(str(boost::format("%u.%u")
% compat_major % compat_minor));
}
x300_impl::x300_mboard_t x300_impl::get_mb_type_from_pcie(const std::string& resource, const std::string& rpc_port)
{
x300_mboard_t mb_type = UNKNOWN;
//Detect the PCIe product ID to distinguish between X300 and X310
nirio_status status = NiRio_Status_Success;
boost::uint32_t pid;
niriok_proxy::sptr discovery_proxy =
niusrprio_session::create_kernel_proxy(resource, rpc_port);
if (discovery_proxy) {
nirio_status_chain(discovery_proxy->get_attribute(RIO_PRODUCT_NUMBER, pid), status);
discovery_proxy->close();
if (nirio_status_not_fatal(status)) {
//The PCIe ID -> MB mapping may be different from the EEPROM -> MB mapping
switch (pid) {
case X300_USRP_PCIE_SSID_ADC_33:
case X300_USRP_PCIE_SSID_ADC_18:
mb_type = USRP_X300_MB; break;
case X310_USRP_PCIE_SSID_ADC_33:
case X310_2940R_40MHz_PCIE_SSID_ADC_33:
case X310_2940R_120MHz_PCIE_SSID_ADC_33:
case X310_2942R_40MHz_PCIE_SSID_ADC_33:
case X310_2942R_120MHz_PCIE_SSID_ADC_33:
case X310_2943R_40MHz_PCIE_SSID_ADC_33:
case X310_2943R_120MHz_PCIE_SSID_ADC_33:
case X310_2944R_40MHz_PCIE_SSID_ADC_33:
case X310_2950R_40MHz_PCIE_SSID_ADC_33:
case X310_2950R_120MHz_PCIE_SSID_ADC_33:
case X310_2952R_40MHz_PCIE_SSID_ADC_33:
case X310_2952R_120MHz_PCIE_SSID_ADC_33:
case X310_2953R_40MHz_PCIE_SSID_ADC_33:
case X310_2953R_120MHz_PCIE_SSID_ADC_33:
case X310_2954R_40MHz_PCIE_SSID_ADC_33:
case X310_USRP_PCIE_SSID_ADC_18:
case X310_2940R_40MHz_PCIE_SSID_ADC_18:
case X310_2940R_120MHz_PCIE_SSID_ADC_18:
case X310_2942R_40MHz_PCIE_SSID_ADC_18:
case X310_2942R_120MHz_PCIE_SSID_ADC_18:
case X310_2943R_40MHz_PCIE_SSID_ADC_18:
case X310_2943R_120MHz_PCIE_SSID_ADC_18:
case X310_2944R_40MHz_PCIE_SSID_ADC_18:
case X310_2950R_40MHz_PCIE_SSID_ADC_18:
case X310_2950R_120MHz_PCIE_SSID_ADC_18:
case X310_2952R_40MHz_PCIE_SSID_ADC_18:
case X310_2952R_120MHz_PCIE_SSID_ADC_18:
case X310_2953R_40MHz_PCIE_SSID_ADC_18:
case X310_2953R_120MHz_PCIE_SSID_ADC_18:
case X310_2954R_40MHz_PCIE_SSID_ADC_18:
mb_type = USRP_X310_MB; break;
default:
mb_type = UNKNOWN; break;
}
}
}
return mb_type;
}
x300_impl::x300_mboard_t x300_impl::get_mb_type_from_eeprom(const uhd::usrp::mboard_eeprom_t& mb_eeprom)
{
x300_mboard_t mb_type = UNKNOWN;
if (not mb_eeprom["product"].empty())
{
boost::uint16_t product_num = 0;
try {
product_num = boost::lexical_cast(mb_eeprom["product"]);
} catch (const boost::bad_lexical_cast &) {
product_num = 0;
}
switch (product_num) {
//The PCIe ID -> MB mapping may be different from the EEPROM -> MB mapping
case X300_USRP_PCIE_SSID_ADC_33:
case X300_USRP_PCIE_SSID_ADC_18:
mb_type = USRP_X300_MB; break;
case X310_USRP_PCIE_SSID_ADC_33:
case X310_2940R_40MHz_PCIE_SSID_ADC_33:
case X310_2940R_120MHz_PCIE_SSID_ADC_33:
case X310_2942R_40MHz_PCIE_SSID_ADC_33:
case X310_2942R_120MHz_PCIE_SSID_ADC_33:
case X310_2943R_40MHz_PCIE_SSID_ADC_33:
case X310_2943R_120MHz_PCIE_SSID_ADC_33:
case X310_2944R_40MHz_PCIE_SSID_ADC_33:
case X310_2950R_40MHz_PCIE_SSID_ADC_33:
case X310_2950R_120MHz_PCIE_SSID_ADC_33:
case X310_2952R_40MHz_PCIE_SSID_ADC_33:
case X310_2952R_120MHz_PCIE_SSID_ADC_33:
case X310_2953R_40MHz_PCIE_SSID_ADC_33:
case X310_2953R_120MHz_PCIE_SSID_ADC_33:
case X310_2954R_40MHz_PCIE_SSID_ADC_33:
case X310_USRP_PCIE_SSID_ADC_18:
case X310_2940R_40MHz_PCIE_SSID_ADC_18:
case X310_2940R_120MHz_PCIE_SSID_ADC_18:
case X310_2942R_40MHz_PCIE_SSID_ADC_18:
case X310_2942R_120MHz_PCIE_SSID_ADC_18:
case X310_2943R_40MHz_PCIE_SSID_ADC_18:
case X310_2943R_120MHz_PCIE_SSID_ADC_18:
case X310_2944R_40MHz_PCIE_SSID_ADC_18:
case X310_2950R_40MHz_PCIE_SSID_ADC_18:
case X310_2950R_120MHz_PCIE_SSID_ADC_18:
case X310_2952R_40MHz_PCIE_SSID_ADC_18:
case X310_2952R_120MHz_PCIE_SSID_ADC_18:
case X310_2953R_40MHz_PCIE_SSID_ADC_18:
case X310_2953R_120MHz_PCIE_SSID_ADC_18:
case X310_2954R_40MHz_PCIE_SSID_ADC_18:
mb_type = USRP_X310_MB; break;
default:
UHD_MSG(warning) << "X300 unknown product code in EEPROM: " << product_num << std::endl;
mb_type = UNKNOWN; break;
}
}
return mb_type;
}