//
// Copyright 2013-2014 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_regs.hpp"
#include "x300_lvbitx.hpp"
#include "x310_lvbitx.hpp"
#include
#include
#include "apply_corrections.hpp"
#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;
namespace asio = boost::asio;
/***********************************************************************
* Discovery over the udp and pcie transport
**********************************************************************/
static std::string get_fpga_option(wb_iface::sptr zpu_ctrl) {
//1G = {0:1G, 1:1G} w/ DRAM, HG = {0:1G, 1:10G} w/ DRAM, XG = {0:10G, 1:10G} w/ DRAM
//HGS = {0:1G, 1:10G} w/ SRAM, XGS = {0:10G, 1:10G} w/ SRAM
//In the default configuration, UHD does not support the HG and XG images so
//they are never autodetected.
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);
return (eth0XG && eth1XG) ? "XGS" : (eth1XG ? "HGS" : "1G");
}
//@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)));
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;
}
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 kernel_proxy;
kernel_proxy.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 call 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 = x300_make_ctrl_iface_pcie(kernel_proxy);
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"] = "";
}
kernel_proxy.close();
//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;
}
static 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);
}
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;
}
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;
_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::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.addr = dev_addr.has_key("resource") ? dev_addr["resource"] : dev_addr["addr"];
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 \
driver have been loaded.");
}
//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 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);
}
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 {
_max_frame_sizes = determine_max_frame_size(mb.addr, req_max_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;
}
}
const std::vector DB_NAMES = boost::assign::list_of("A")("B");
//create basic communication
UHD_MSG(status) << "Setup basic communication..." << std::endl;
if (mb.xport_path == "nirio") {
mb.zpu_ctrl = x300_make_ctrl_iface_pcie(mb.rio_fpga_interface->get_kernel_proxy());
} else {
mb.zpu_ctrl = x300_make_ctrl_iface_enet(udp_simple::make_connected(mb.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 -- good place to do after conditional loading
this->check_fw_compat(mb_path, mb.zpu_ctrl);
this->check_fpga_compat(mb_path, mb.zpu_ctrl);
//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)
.subscribe(boost::bind(&x300_impl::set_mb_eeprom, this, mb.zpu_i2c, _1));
////////////////////////////////////////////////////////////////////
// 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:
break;
}
_tree->create(mb_path / "name").set(product_name);
_tree->create(mb_path / "codename").set("Yetti");
////////////////////////////////////////////////////////////////////
// determine routing based on address match
////////////////////////////////////////////////////////////////////
mb.router_dst_here = X300_XB_DST_E0; //some default if eeprom not match
if (mb.xport_path == "nirio") {
mb.router_dst_here = X300_XB_DST_PCI;
} else {
if (mb.addr == mb_eeprom["ip-addr0"]) mb.router_dst_here = X300_XB_DST_E0;
else if (mb.addr == mb_eeprom["ip-addr1"]) mb.router_dst_here = X300_XB_DST_E1;
else if (mb.addr == mb_eeprom["ip-addr2"]) mb.router_dst_here = X300_XB_DST_E0;
else if (mb.addr == mb_eeprom["ip-addr3"]) mb.router_dst_here = X300_XB_DST_E1;
else if (mb.addr == boost::asio::ip::address_v4(boost::uint32_t(X300_DEFAULT_IP_ETH0_1G)).to_string()) mb.router_dst_here = X300_XB_DST_E0;
else if (mb.addr == boost::asio::ip::address_v4(boost::uint32_t(X300_DEFAULT_IP_ETH1_1G)).to_string()) mb.router_dst_here = X300_XB_DST_E1;
else if (mb.addr == boost::asio::ip::address_v4(boost::uint32_t(X300_DEFAULT_IP_ETH0_10G)).to_string()) mb.router_dst_here = X300_XB_DST_E0;
else if (mb.addr == boost::asio::ip::address_v4(boost::uint32_t(X300_DEFAULT_IP_ETH1_10G)).to_string()) mb.router_dst_here = X300_XB_DST_E1;
}
////////////////////////////////////////////////////////////////////
// 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);
}
////////////////////////////////////////////////////////////////////
// create clock control objects
////////////////////////////////////////////////////////////////////
UHD_MSG(status) << "Setup RF frontend clocking..." << std::endl;
// Init shadow and clock source; the device comes up with it's internal
// clock source before locking to something else (if requested).
mb.clock_control_regs__clock_source = 0;
mb.clock_control_regs__pps_select = 0;
mb.clock_control_regs__pps_out_enb = 0;
mb.clock_control_regs__tcxo_enb = 1;
mb.clock_control_regs__gpsdo_pwr = 1;
this->update_clock_source(mb, "internal");
this->update_clock_control(mb);
size_t hw_rev = 0;
if(mb_eeprom.has_key("revision") and not mb_eeprom["revision"].empty()) {
try {
hw_rev = boost::lexical_cast(mb_eeprom["revision"]);
} catch(...) {
UHD_MSG(warning) << "Revision in EEPROM is invalid! Please reprogram your EEPROM." << std::endl;
}
} else {
UHD_MSG(warning) << "No revision detected MB EEPROM must be reprogrammed!" << std::endl;
}
if(hw_rev == 0) {
UHD_MSG(warning) << "Defaulting to X300 RevD Clock Settings. This will result in non-optimal lock times." << std::endl;
hw_rev = X300_REV("D");
}
mb.clock = x300_clock_ctrl::make(mb.zpu_spi,
1 /*slaveno*/,
hw_rev,
dev_addr.cast("master_clock_rate", X300_DEFAULT_TICK_RATE),
dev_addr.cast("system_ref_rate", X300_DEFAULT_SYSREF_RATE));
////////////////////////////////////////////////////////////////////
// create clock properties
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "tick_rate")
.publish(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)
.publish(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);
}
////////////////////////////////////////////////////////////////////
// setup radios
////////////////////////////////////////////////////////////////////
UHD_MSG(status) << "Initialize Radio control..." << std::endl;
this->setup_radio(mb_i, 0, DB_NAMES[0]);
this->setup_radio(mb_i, 1, DB_NAMES[1]);
////////////////////////////////////////////////////////////////////
// front panel gpio
////////////////////////////////////////////////////////////////////
mb.fp_gpio = gpio_core_200::make(mb.radio_perifs[0].ctrl, TOREG(SR_FP_GPIO), RB32_FP_GPIO);
const std::vector GPIO_ATTRS = boost::assign::list_of("CTRL")("DDR")("OUT")("ATR_0X")("ATR_RX")("ATR_TX")("ATR_XX");
BOOST_FOREACH(const std::string &attr, GPIO_ATTRS)
{
_tree->create(mb_path / "gpio" / "FP0" / attr)
.set(0)
.subscribe(boost::bind(&x300_impl::set_fp_gpio, this, mb.fp_gpio, attr, _1));
}
_tree->create(mb_path / "gpio" / "FP0" / "READBACK")
.publish(boost::bind(&x300_impl::get_fp_gpio, this, mb.fp_gpio, "READBACK"));
////////////////////////////////////////////////////////////////////
// register the time keepers - only one can be the highlander
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "time" / "now")
.publish(boost::bind(&time_core_3000::get_time_now, mb.radio_perifs[0].time64))
.subscribe(boost::bind(&time_core_3000::set_time_now, mb.radio_perifs[0].time64, _1))
.subscribe(boost::bind(&time_core_3000::set_time_now, mb.radio_perifs[1].time64, _1));
_tree->create(mb_path / "time" / "pps")
.publish(boost::bind(&time_core_3000::get_time_last_pps, mb.radio_perifs[0].time64))
.subscribe(boost::bind(&time_core_3000::set_time_next_pps, mb.radio_perifs[0].time64, _1))
.subscribe(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")
.subscribe(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")
.subscribe(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")
.subscribe(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")
.subscribe(boost::bind(&x300_clock_ctrl::set_ref_out, mb.clock, _1));
////////////////////////////////////////////////////////////////////
// create frontend mapping
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "rx_subdev_spec")
.subscribe(boost::bind(&x300_impl::update_rx_subdev_spec, this, mb_i, _1));
_tree->create(mb_path / "tx_subdev_spec")
.subscribe(boost::bind(&x300_impl::update_tx_subdev_spec, this, mb_i, _1));
////////////////////////////////////////////////////////////////////
// and do the misc mboard sensors
////////////////////////////////////////////////////////////////////
_tree->create(mb_path / "sensors" / "ref_locked")
.publish(boost::bind(&x300_impl::get_ref_locked, this, mb.zpu_ctrl));
////////////////////////////////////////////////////////////////////
// create clock properties
////////////////////////////////////////////////////////////////////
_tree->access(mb_path / "tick_rate")
.subscribe(boost::bind(&x300_impl::set_tick_rate, this, boost::ref(mb), _1))
.subscribe(boost::bind(&x300_impl::update_tick_rate, this, boost::ref(mb), _1))
.set(mb.clock->get_master_clock_rate());
////////////////////////////////////////////////////////////////////
// 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);
//GPS installed: use external ref, time, and init time spec
if (mb.gps and mb.gps->gps_detected())
{
UHD_MSG(status) << "Setting references to the internal GPSDO" << std::endl;
_tree->access(mb_path / "time_source" / "value").set("gpsdo");
_tree->access(mb_path / "clock_source" / "value").set("gpsdo");
UHD_MSG(status) << "Initializing time to the internal GPSDO" << std::endl;
const time_t tp = time_t(mb.gps->get_sensor("gps_time").to_int()+1);
_tree->access(mb_path / "time" / "pps").set(time_spec_t(tp));
}
else
{
_tree->access(mb_path / "time_source" / "value").set("external");
_tree->access(mb_path / "clock_source" / "value").set("external");
boost::this_thread::sleep(boost::posix_time::milliseconds(100));
if (this->get_ref_locked(mb.zpu_ctrl).to_bool())
{
UHD_MSG(status) << "Setting references to external sources" << std::endl;
}
else
{
UHD_MSG(status) << "Setting references to internal sources" << std::endl;
_tree->access(mb_path / "time_source" / "value").set("internal");
_tree->access(mb_path / "clock_source" / "value").set("internal");
}
}
}
x300_impl::~x300_impl(void)
{
try
{
BOOST_FOREACH(mboard_members_t &mb, _mb)
{
mb.radio_perifs[0].ctrl->poke32(TOREG(SR_MISC_OUTS), (1 << 2)); //disable/reset ADC/DAC
mb.radio_perifs[1].ctrl->poke32(TOREG(SR_MISC_OUTS), (1 << 2)); //disable/reset ADC/DAC
//kill the claimer task and unclaim the device
mb.claimer_task.reset();
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);
}
}
catch(...)
{
UHD_SAFE_CALL(throw;)
}
}
static void check_adc(wb_iface::sptr iface, const boost::uint32_t val)
{
boost::uint32_t adc_rb = iface->peek32(RB32_RX);
adc_rb ^= 0xfffc0000; //adapt for I inversion in FPGA
//UHD_MSG(status) << "adc_rb " << std::hex << adc_rb << " val " << std::hex << val << std::endl;
UHD_ASSERT_THROW(adc_rb == val);
}
void x300_impl::setup_radio(const size_t mb_i, const size_t i, const std::string &db_name)
{
const fs_path mb_path = "/mboards/"+boost::lexical_cast(mb_i);
mboard_members_t &mb = _mb[mb_i];
radio_perifs_t &perif = mb.radio_perifs[i];
const size_t dspno = i;
////////////////////////////////////////////////////////////////////
// radio control
////////////////////////////////////////////////////////////////////
uint8_t dest = (i == 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, db_name);
perif.ctrl->poke32(TOREG(SR_MISC_OUTS), (1 << 2)); //reset adc + dac
perif.ctrl->poke32(TOREG(SR_MISC_OUTS), (1 << 1) | (1 << 0)); //out of reset + dac enable
this->register_loopback_self_test(perif.ctrl);
perif.spi = spi_core_3000::make(perif.ctrl, TOREG(SR_SPI), 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_core_200_32wo::make(perif.ctrl, TOREG(SR_LEDS));
_tree->access(mb_path / "time" / "cmd")
.subscribe(boost::bind(&radio_ctrl_core_3000::set_time, perif.ctrl, _1));
_tree->access(mb_path / "tick_rate")
.subscribe(boost::bind(&radio_ctrl_core_3000::set_tick_rate, perif.ctrl, _1));
////////////////////////////////////////////////////////////////
// ADC self test
////////////////////////////////////////////////////////////////
perif.adc->set_test_word("ones", "ones"); check_adc(perif.ctrl, 0xfffcfffc);
perif.adc->set_test_word("zeros", "zeros"); check_adc(perif.ctrl, 0x00000000);
perif.adc->set_test_word("ones", "zeros"); check_adc(perif.ctrl, 0xfffc0000);
perif.adc->set_test_word("zeros", "ones"); check_adc(perif.ctrl, 0x0000fffc);
for (size_t k = 0; k < 14; k++)
{
perif.adc->set_test_word("zeros", "custom", 1 << k);
check_adc(perif.ctrl, 1 << (k+2));
}
for (size_t k = 0; k < 14; k++)
{
perif.adc->set_test_word("custom", "zeros", 1 << k);
check_adc(perif.ctrl, 1 << (k+18));
}
perif.adc->set_test_word("normal", "normal");
////////////////////////////////////////////////////////////////
// Sync DAC's for MIMO
////////////////////////////////////////////////////////////////
UHD_MSG(status) << "Sync DAC's." << std::endl;
perif.dac->arm_dac_sync(); // Put DAC into data Sync mode
perif.ctrl->poke32(TOREG(SR_DACSYNC), 0x1); // Arm FRAMEP/N sync pulse
////////////////////////////////////////////////////////////////
// create codec control objects
////////////////////////////////////////////////////////////////
_tree->create(mb_path / "rx_codecs" / db_name / "gains"); //phony property so this dir exists
_tree->create(mb_path / "tx_codecs" / db_name / "gains"); //phony property so this dir exists
_tree->create(mb_path / "rx_codecs" / db_name / "name").set("ads62p48");
_tree->create(mb_path / "tx_codecs" / db_name / "name").set("ad9146");
_tree->create(mb_path / "rx_codecs" / db_name / "gains" / "digital" / "range").set(meta_range_t(0, 6.0, 0.5));
_tree->create(mb_path / "rx_codecs" / db_name / "gains" / "digital" / "value")
.subscribe(boost::bind(&x300_adc_ctrl::set_gain, perif.adc, _1)).set(0);
////////////////////////////////////////////////////////////////////
// front end corrections
////////////////////////////////////////////////////////////////////
perif.rx_fe = rx_frontend_core_200::make(perif.ctrl, TOREG(SR_RX_FRONT));
const fs_path rx_fe_path = mb_path / "rx_frontends" / db_name;
_tree->create >(rx_fe_path / "dc_offset" / "value")
.coerce(boost::bind(&rx_frontend_core_200::set_dc_offset, perif.rx_fe, _1))
.set(std::complex(0.0, 0.0));
_tree->create(rx_fe_path / "dc_offset" / "enable")
.subscribe(boost::bind(&rx_frontend_core_200::set_dc_offset_auto, perif.rx_fe, _1))
.set(true);
_tree->create >(rx_fe_path / "iq_balance" / "value")
.subscribe(boost::bind(&rx_frontend_core_200::set_iq_balance, perif.rx_fe, _1))
.set(std::complex(0.0, 0.0));
perif.tx_fe = tx_frontend_core_200::make(perif.ctrl, TOREG(SR_TX_FRONT));
const fs_path tx_fe_path = mb_path / "tx_frontends" / db_name;
_tree->create >(tx_fe_path / "dc_offset" / "value")
.coerce(boost::bind(&tx_frontend_core_200::set_dc_offset, perif.tx_fe, _1))
.set(std::complex(0.0, 0.0));
_tree->create >(tx_fe_path / "iq_balance" / "value")
.subscribe(boost::bind(&tx_frontend_core_200::set_iq_balance, perif.tx_fe, _1))
.set(std::complex(0.0, 0.0));
////////////////////////////////////////////////////////////////////
// create rx dsp control objects
////////////////////////////////////////////////////////////////////
perif.framer = rx_vita_core_3000::make(perif.ctrl, TOREG(SR_RX_CTRL));
perif.ddc = rx_dsp_core_3000::make(perif.ctrl, TOREG(SR_RX_DSP));
perif.ddc->set_link_rate(10e9/8); //whatever
_tree->access(mb_path / "tick_rate")
.subscribe(boost::bind(&rx_vita_core_3000::set_tick_rate, perif.framer, _1))
.subscribe(boost::bind(&rx_dsp_core_3000::set_tick_rate, perif.ddc, _1));
const fs_path rx_dsp_path = mb_path / "rx_dsps" / str(boost::format("%u") % dspno);
_tree->create(rx_dsp_path / "rate" / "range")
.publish(boost::bind(&rx_dsp_core_3000::get_host_rates, perif.ddc));
_tree->create(rx_dsp_path / "rate" / "value")
.coerce(boost::bind(&rx_dsp_core_3000::set_host_rate, perif.ddc, _1))
.subscribe(boost::bind(&x300_impl::update_rx_samp_rate, this, boost::ref(mb), dspno, _1))
.set(1e6);
_tree->create(rx_dsp_path / "freq" / "value")
.coerce(boost::bind(&rx_dsp_core_3000::set_freq, perif.ddc, _1))
.set(0.0);
_tree->create(rx_dsp_path / "freq" / "range")
.publish(boost::bind(&rx_dsp_core_3000::get_freq_range, perif.ddc));
_tree->create(rx_dsp_path / "stream_cmd")
.subscribe(boost::bind(&rx_vita_core_3000::issue_stream_command, perif.framer, _1));
////////////////////////////////////////////////////////////////////
// create tx dsp control objects
////////////////////////////////////////////////////////////////////
perif.deframer = tx_vita_core_3000::make(perif.ctrl, TOREG(SR_TX_CTRL));
perif.duc = tx_dsp_core_3000::make(perif.ctrl, TOREG(SR_TX_DSP));
perif.duc->set_link_rate(10e9/8); //whatever
_tree->access(mb_path / "tick_rate")
.subscribe(boost::bind(&tx_vita_core_3000::set_tick_rate, perif.deframer, _1))
.subscribe(boost::bind(&tx_dsp_core_3000::set_tick_rate, perif.duc, _1));
const fs_path tx_dsp_path = mb_path / "tx_dsps" / str(boost::format("%u") % dspno);
_tree->create(tx_dsp_path / "rate" / "range")
.publish(boost::bind(&tx_dsp_core_3000::get_host_rates, perif.duc));
_tree->create(tx_dsp_path / "rate" / "value")
.coerce(boost::bind(&tx_dsp_core_3000::set_host_rate, perif.duc, _1))
.subscribe(boost::bind(&x300_impl::update_tx_samp_rate, this, boost::ref(mb), dspno, _1))
.set(1e6);
_tree->create(tx_dsp_path / "freq" / "value")
.coerce(boost::bind(&tx_dsp_core_3000::set_freq, perif.duc, _1))
.set(0.0);
_tree->create(tx_dsp_path / "freq" / "range")
.publish(boost::bind(&tx_dsp_core_3000::get_freq_range, perif.duc));
////////////////////////////////////////////////////////////////////
// create time control objects
////////////////////////////////////////////////////////////////////
time_core_3000::readback_bases_type time64_rb_bases;
time64_rb_bases.rb_now = RB64_TIME_NOW;
time64_rb_bases.rb_pps = RB64_TIME_PPS;
perif.time64 = time_core_3000::make(perif.ctrl, TOREG(SR_TIME), time64_rb_bases);
////////////////////////////////////////////////////////////////////
// create RF frontend interfacing
////////////////////////////////////////////////////////////////////
const size_t j = (db_name == "B")? 0x2 : 0x0;
_tree->create(mb_path / "dboards" / db_name / "rx_eeprom")
.set(mb.db_eeproms[X300_DB0_RX_EEPROM | j])
.subscribe(boost::bind(&x300_impl::set_db_eeprom, this, mb.zpu_i2c, (0x50 | X300_DB0_RX_EEPROM | j), _1));
_tree->create(mb_path / "dboards" / db_name / "tx_eeprom")
.set(mb.db_eeproms[X300_DB0_TX_EEPROM | j])
.subscribe(boost::bind(&x300_impl::set_db_eeprom, this, mb.zpu_i2c, (0x50 | X300_DB0_TX_EEPROM | j), _1));
_tree->create(mb_path / "dboards" / db_name / "gdb_eeprom")
.set(mb.db_eeproms[X300_DB0_GDB_EEPROM | j])
.subscribe(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 = gpio_core_200::make(perif.ctrl, TOREG(SR_GPIO), 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.which_rx_clk = (db_name == "A")? X300_CLOCK_WHICH_DB0_RX : X300_CLOCK_WHICH_DB1_RX;
db_config.which_tx_clk = (db_name == "A")? X300_CLOCK_WHICH_DB0_TX : X300_CLOCK_WHICH_DB1_TX;
db_config.dboard_slot = (db_name == "A")? 0 : 1;
_dboard_ifaces[db_name] = x300_make_dboard_iface(db_config);
//create a new dboard manager
_tree->create(mb_path / "dboards" / db_name / "iface").set(_dboard_ifaces[db_name]);
_dboard_managers[db_name] = 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,
_dboard_ifaces[db_name],
_tree->subtree(mb_path / "dboards" / db_name)
);
//now that dboard is created -- register into rx antenna event
const std::string fe_name = _tree->list(mb_path / "dboards" / db_name / "rx_frontends").front();
_tree->access(mb_path / "dboards" / db_name / "rx_frontends" / fe_name / "antenna" / "value")
.subscribe(boost::bind(&x300_impl::update_atr_leds, this, mb.radio_perifs[i].leds, _1));
this->update_atr_leds(mb.radio_perifs[i].leds, ""); //init anyway, even if never called
//bind frontend corrections to the dboard freq props
const fs_path db_rx_fe_path = mb_path / "dboards" / db_name / "rx_frontends";
BOOST_FOREACH(const std::string &name, _tree->list(db_rx_fe_path))
{
_tree->access(db_rx_fe_path / name / "freq" / "value")
.subscribe(boost::bind(&x300_impl::set_rx_fe_corrections, this, mb_path, db_name, _1));
}
}
void x300_impl::set_rx_fe_corrections(const uhd::fs_path &mb_path, const std::string &fe_name, const double lo_freq)
{
apply_rx_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 uint8_t& destination,
const 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;
config.router_dst_here = mb.router_dst_here;
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_DATA_FRAME_SIZE
: X300_PCIE_MSG_FRAME_SIZE;
default_buff_args.recv_frame_size =
(prefix == X300_RADIO_DEST_PREFIX_RX)
? X300_PCIE_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;
if (mb.loaded_fpga_image == "HGS") {
if (mb.router_dst_here == X300_XB_DST_E0) {
eth_data_rec_frame_size = X300_1GE_DATA_FRAME_MAX_SIZE;
} else if (mb.router_dst_here == X300_XB_DST_E1) {
eth_data_rec_frame_size = X300_10GE_DATA_FRAME_MAX_SIZE;
}
} else if (mb.loaded_fpga_image == "XGS") {
eth_data_rec_frame_size = X300_10GE_DATA_FRAME_MAX_SIZE;
}
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(mb.addr,
BOOST_STRINGIZE(X300_VITA_UDP_PORT),
default_buff_args,
buff_params,
xport_args);
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.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;
const boost::uint32_t sid = 0
| (X300_DEVICE_HERE << 24)
| (_sid_framer << 16)
| (config.router_addr_there << 8)
| (stream << 0)
;
UHD_LOG << std::hex
<< " sid 0x" << sid
<< " 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 + (X300_DEVICE_HERE)), config.router_dst_here);
if (xport_path == "nirio") {
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_core_200_32wo::sptr leds, const std::string &rx_ant)
{
const bool is_txrx = (rx_ant == "TX/RX");
const int rx_led = (1 << 2);
const int txrx_led = (1 << 1);
const int tx_led = (1 << 0);
leds->set_atr_reg(dboard_iface::ATR_REG_IDLE, 0);
leds->set_atr_reg(dboard_iface::ATR_REG_RX_ONLY, is_txrx? txrx_led : rx_led);
leds->set_atr_reg(dboard_iface::ATR_REG_TX_ONLY, tx_led);
leds->set_atr_reg(dboard_iface::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.time64->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 = time(NULL);
for (size_t i = 0; i < 100; i++)
{
boost::hash_combine(hash, i);
iface->poke32(TOREG(SR_TEST), boost::uint32_t(hash));
test_fail = iface->peek32(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::set_time_source_out(mboard_members_t &mb, const bool enb)
{
mb.clock_control_regs__pps_out_enb = enb? 1 : 0;
this->update_clock_control(mb);
}
void x300_impl::update_clock_control(mboard_members_t &mb)
{
const size_t reg = mb.clock_control_regs__clock_source
| (mb.clock_control_regs__pps_select << 2)
| (mb.clock_control_regs__pps_out_enb << 3)
| (mb.clock_control_regs__tcxo_enb << 4)
| (mb.clock_control_regs__gpsdo_pwr << 5)
;
mb.zpu_ctrl->poke32(SR_ADDR(SET0_BASE, ZPU_SR_CLOCK_CTRL), reg);
}
void x300_impl::update_clock_source(mboard_members_t &mb, const std::string &source)
{
mb.clock_control_regs__clock_source = 0;
if (source == "internal") {
mb.clock_control_regs__clock_source = 0x2;
mb.clock_control_regs__tcxo_enb = (source == "internal")? 1 : 0;
} else if (source == "external") {
mb.clock_control_regs__clock_source = 0x0;
} else if (source == "gpsdo") {
mb.clock_control_regs__clock_source = 0x3;
} else {
throw uhd::key_error("update_clock_source: unknown source: " + source);
}
this->update_clock_control(mb);
}
void x300_impl::update_time_source(mboard_members_t &mb, const std::string &source)
{
if (source == "internal") {
// no action needed
} else if (source == "external") {
mb.clock_control_regs__pps_select = (source == "external")? 1 : 0;
} else if (source == "gpsdo") {
// no action needed
} else {
throw uhd::key_error("update_time_source: unknown source: " + source);
}
this->update_clock_control(mb);
}
sensor_value_t x300_impl::get_ref_locked(wb_iface::sptr ctrl)
{
const bool lock = (ctrl->peek32(SR_ADDR(SET0_BASE, ZPU_RB_CLK_STATUS)) & (1 << 2)) != 0;
return sensor_value_t("Ref", lock, "locked", "unlocked");
}
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");
}
boost::uint32_t x300_impl::get_fp_gpio(gpio_core_200::sptr gpio, const std::string &)
{
return boost::uint32_t(gpio->read_gpio(dboard_iface::UNIT_RX));
}
void x300_impl::set_fp_gpio(gpio_core_200::sptr gpio, const std::string &attr, const boost::uint32_t value)
{
if (attr == "CTRL") return gpio->set_pin_ctrl(dboard_iface::UNIT_RX, value);
if (attr == "DDR") return gpio->set_gpio_ddr(dboard_iface::UNIT_RX, value);
if (attr == "OUT") return gpio->set_gpio_out(dboard_iface::UNIT_RX, value);
if (attr == "ATR_0X") return gpio->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, value);
if (attr == "ATR_RX") return gpio->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, value);
if (attr == "ATR_TX") return gpio->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, value);
if (attr == "ATR_XX") return gpio->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, value);
}
/***********************************************************************
* claimer logic
**********************************************************************/
void x300_impl::claimer_loop(wb_iface::sptr iface)
{
iface->poke32(SR_ADDR(X300_FW_SHMEM_BASE, X300_FW_SHMEM_CLAIM_TIME), 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(1500)); //1.5 seconds
}
bool x300_impl::is_claimed(wb_iface::sptr iface)
{
//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 0x%x, but got 0x%x.%x:\n"
"The firmware build is not compatible with the host code build.\n"
"%s"
) % int(X300_FW_COMPAT_MAJOR) % compat_major % compat_minor
% print_images_error()));
}
_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, wb_iface::sptr iface)
{
boost::uint32_t compat_num = iface->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)
{
throw uhd::runtime_error(str(boost::format(
"Expected FPGA compatibility number 0x%x, but got 0x%x.%x:\n"
"The FPGA build is not compatible with the host code build.\n"
"%s"
) % int(X300_FPGA_COMPAT_MAJOR) % compat_major % compat_minor
% print_images_error()));
}
_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(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:
mb_type = USRP_X300_MB; break;
case X310_USRP_PCIE_SSID:
case X310_2940R_PCIE_SSID:
case X310_2942R_PCIE_SSID:
case X310_2943R_PCIE_SSID:
case X310_2944R_PCIE_SSID:
case X310_2950R_PCIE_SSID:
case X310_2952R_PCIE_SSID:
case X310_2953R_PCIE_SSID:
case X310_2954R_PCIE_SSID:
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:
mb_type = USRP_X300_MB; break;
case X310_USRP_PCIE_SSID:
case X310_2940R_PCIE_SSID:
case X310_2942R_PCIE_SSID:
case X310_2943R_PCIE_SSID:
case X310_2944R_PCIE_SSID:
case X310_2950R_PCIE_SSID:
case X310_2952R_PCIE_SSID:
case X310_2953R_PCIE_SSID:
case X310_2954R_PCIE_SSID:
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;
}