//
// Copyright 2010 Ettus Research LLC
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see .
//
#include "../../transport/vrt_packet_handler.hpp"
#include "usrp_commands.h"
#include "usrp1_impl.hpp"
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace uhd;
using namespace uhd::usrp;
using namespace uhd::transport;
namespace asio = boost::asio;
static const float poll_interval = 0.1; //100ms
/*
* The FX2 firmware bursts data to the FPGA in 512 byte chunks so
* maintain send state to make sure that happens.
*/
struct usrp1_send_state {
uhd::transport::managed_send_buffer::sptr send_buff;
size_t bytes_used;
size_t bytes_free;
};
/***********************************************************************
* IO Implementation Details
**********************************************************************/
struct usrp1_impl::io_impl {
io_impl(zero_copy_if::sptr zc_if);
~io_impl(void);
bool get_recv_buff(managed_recv_buffer::sptr buff);
//state management for the vrt packet handler code
vrt_packet_handler::recv_state packet_handler_recv_state;
usrp1_send_state send_state;
zero_copy_if::sptr data_transport;
//overun-underrun values
unsigned int tx_underrun_count;
unsigned int rx_overrun_count;
};
usrp1_impl::io_impl::io_impl(zero_copy_if::sptr zc_if)
: packet_handler_recv_state(1), data_transport(zc_if),
tx_underrun_count(0), rx_overrun_count(0)
{
/* NOP */
}
usrp1_impl::io_impl::~io_impl(void)
{
/* NOP */
}
void usrp1_impl::io_init(void)
{
_rx_otw_type.width = 16;
_rx_otw_type.shift = 0;
_rx_otw_type.byteorder = otw_type_t::BO_LITTLE_ENDIAN;
_tx_otw_type.width = 16;
_tx_otw_type.shift = 0;
_tx_otw_type.byteorder = otw_type_t::BO_LITTLE_ENDIAN;
_io_impl = UHD_PIMPL_MAKE(io_impl, (_data_transport));
}
/***********************************************************************
* Data Send
**********************************************************************/
size_t usrp1_impl::send(const std::vector &buffs,
size_t num_samps,
const tx_metadata_t &,
const io_type_t &io_type,
send_mode_t)
{
UHD_ASSERT_THROW(buffs.size() == 1);
size_t total_samps_sent = 0;
while (total_samps_sent < num_samps) {
if (_io_impl->send_state.send_buff == NULL) {
_io_impl->send_state.send_buff = _data_transport->get_send_buff();
if (_io_impl->send_state.send_buff == NULL) {
return 0;
}
_io_impl->send_state.bytes_used = 0;
_io_impl->send_state.bytes_free = _io_impl->send_state.send_buff->size();
}
size_t copy_samps =
std::min(num_samps - total_samps_sent, _io_impl->send_state.bytes_free / _tx_otw_type.get_sample_size());
const boost::uint8_t *io_mem =
reinterpret_cast(buffs[0]);
boost::uint8_t *otw_mem = _io_impl->send_state.send_buff->cast();
// Type conversion and copy
convert_io_type_to_otw_type(
io_mem + total_samps_sent * io_type.size,
io_type,
otw_mem + _io_impl->send_state.bytes_used,
_tx_otw_type,
copy_samps);
_io_impl->send_state.bytes_used += copy_samps * _tx_otw_type.get_sample_size();
_io_impl->send_state.bytes_free -= copy_samps * _tx_otw_type.get_sample_size();
if (_io_impl->send_state.bytes_free == 0) {
_io_impl->send_state.send_buff->commit(_io_impl->send_state.bytes_used);
_io_impl->send_state.send_buff = uhd::transport::managed_send_buffer::sptr();
}
total_samps_sent += copy_samps;
_io_impl->tx_underrun_count += copy_samps * _tx_otw_type.get_sample_size();
//check for underruns
if (!(_io_impl->tx_underrun_count > _tx_dsp_freq * poll_interval * _tx_otw_type.get_sample_size())) {
unsigned char underrun;
int ret = _ctrl_transport->usrp_control_read(VRQ_GET_STATUS,
0,
GS_TX_UNDERRUN,
&underrun, sizeof(char));
if (ret < 0)
std::cerr << "error: underrun check failed" << std::endl;
if (underrun)
std::cerr << "U" << std::endl;
_io_impl->tx_underrun_count = 0;
}
}
return total_samps_sent;
}
/***********************************************************************
* Data Recv
**********************************************************************/
void _recv_helper(vrt_packet_handler::recv_state &state)
{
size_t num_packet_words32 =
state.managed_buffs[0]->size() / sizeof(boost::uint32_t);
const boost::uint32_t *data =
state.managed_buffs[0]->cast();
state.copy_buffs[0] = reinterpret_cast(data);
size_t num_payload_bytes = num_packet_words32 * sizeof(boost::uint32_t);
state.size_of_copy_buffs = num_payload_bytes;
}
size_t usrp1_impl::recv(const std::vector &buffs,
size_t num_samps,
rx_metadata_t &,
const io_type_t &io_type,
recv_mode_t,
size_t)
{
UHD_ASSERT_THROW(_io_impl->packet_handler_recv_state.width == 1);
UHD_ASSERT_THROW(buffs.size() == 1);
size_t sent_samps = 0;
size_t nsamps_to_copy = 0;;
while (sent_samps < num_samps) {
if (_io_impl->packet_handler_recv_state.size_of_copy_buffs == 0) {
_io_impl->packet_handler_recv_state.fragment_offset_in_samps = 0;
_io_impl->packet_handler_recv_state.managed_buffs[0] =
_io_impl->data_transport->get_recv_buff();
//timeout or something bad returns zero
if (!_io_impl->packet_handler_recv_state.managed_buffs[0].get())
return 0;
_recv_helper(_io_impl->packet_handler_recv_state);
}
size_t bytes_per_item = _rx_otw_type.get_sample_size();
size_t nsamps_available =
_io_impl->packet_handler_recv_state.size_of_copy_buffs / bytes_per_item;
nsamps_to_copy = std::min(num_samps, nsamps_available);
size_t bytes_to_copy = nsamps_to_copy * bytes_per_item;
convert_otw_type_to_io_type(
_io_impl->packet_handler_recv_state.copy_buffs[0],
_rx_otw_type,
reinterpret_cast(buffs[0]) + sent_samps * io_type.size,
io_type,
nsamps_to_copy);
_io_impl->packet_handler_recv_state.copy_buffs[0] += bytes_to_copy;
_io_impl->packet_handler_recv_state.size_of_copy_buffs -= bytes_to_copy;
_io_impl->rx_overrun_count += nsamps_to_copy;
sent_samps += nsamps_to_copy;
//check for overruns
if (_io_impl->rx_overrun_count > (8e6 * poll_interval)) {
unsigned char overrun;
int ret = _ctrl_transport->usrp_control_read(VRQ_GET_STATUS,
0,
GS_RX_OVERRUN,
&overrun, sizeof(char));
if (ret < 0)
std::cerr << "error: overrun check failed" << std::endl;
if (overrun)
std::cerr << "O" << std::endl;
_io_impl->rx_overrun_count = 0;
}
}
return sent_samps;
}