diff options
Diffstat (limited to 'host/lib/transport')
-rw-r--r-- | host/lib/transport/CMakeLists.txt | 3 | ||||
-rw-r--r-- | host/lib/transport/usb_zero_copy_wrapper.cpp | 239 |
2 files changed, 1 insertions, 241 deletions
diff --git a/host/lib/transport/CMakeLists.txt b/host/lib/transport/CMakeLists.txt index 4eb7c8c92..963edcf85 100644 --- a/host/lib/transport/CMakeLists.txt +++ b/host/lib/transport/CMakeLists.txt @@ -1,5 +1,5 @@ # -# Copyright 2010-2011 Ettus Research LLC +# Copyright 2010-2013 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 @@ -122,5 +122,4 @@ LIBUHD_APPEND_SOURCES( ${CMAKE_CURRENT_SOURCE_DIR}/buffer_pool.cpp ${CMAKE_CURRENT_SOURCE_DIR}/if_addrs.cpp ${CMAKE_CURRENT_SOURCE_DIR}/udp_simple.cpp - ${CMAKE_CURRENT_SOURCE_DIR}/usb_zero_copy_wrapper.cpp ) diff --git a/host/lib/transport/usb_zero_copy_wrapper.cpp b/host/lib/transport/usb_zero_copy_wrapper.cpp deleted file mode 100644 index ec5667020..000000000 --- a/host/lib/transport/usb_zero_copy_wrapper.cpp +++ /dev/null @@ -1,239 +0,0 @@ -// -// Copyright 2011-2012 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 <http://www.gnu.org/licenses/>. -// - -#include <uhd/transport/usb_zero_copy.hpp> -#include <uhd/transport/buffer_pool.hpp> -#include <uhd/utils/byteswap.hpp> -#include <uhd/utils/msg.hpp> -#include <uhd/utils/tasks.hpp> -#include <uhd/utils/atomic.hpp> -#include <boost/foreach.hpp> -#include <boost/make_shared.hpp> -#include <boost/thread/mutex.hpp> -#include <boost/thread/condition_variable.hpp> -#include <boost/bind.hpp> -#include <vector> -#include <iostream> - -using namespace uhd; -using namespace uhd::transport; - -static const boost::posix_time::time_duration AUTOFLUSH_TIMEOUT(boost::posix_time::milliseconds(1)); - -/*********************************************************************** - * USB zero copy wrapper - managed receive buffer - **********************************************************************/ -class usb_zero_copy_wrapper_mrb : public managed_recv_buffer{ -public: - usb_zero_copy_wrapper_mrb(void){/*NOP*/} - - void release(void){ - _mrb.reset(); //decrement ref count, other MRB's may hold a ref - _claimer.release(); - } - - UHD_INLINE sptr get_new( - managed_recv_buffer::sptr &mrb, size_t &offset_bytes, - const double timeout, size_t &index - ){ - if (not mrb or not _claimer.claim_with_wait(timeout)) return sptr(); - - index++; //advances the caller's buffer - - //hold a copy of the buffer shared pointer - _mrb = mrb; - - //extract this packet's memory address and length in bytes - char *mem = mrb->cast<char *>() + offset_bytes; - const boost::uint32_t *mem32 = reinterpret_cast<const boost::uint32_t *>(mem); - const size_t words32 = (uhd::wtohx(mem32[0]) & 0xffff); //length in words32 (from VRT header) - const size_t len = words32*sizeof(boost::uint32_t); //length in bytes - - //check if this receive buffer has been exhausted - offset_bytes += len; - if (offset_bytes >= mrb->size()) mrb.reset(); //drop caller's ref - else if (uhd::wtohx(mem32[words32]) == 0) mrb.reset(); - - return make(this, mem, len); - } - -private: - managed_recv_buffer::sptr _mrb; - simple_claimer _claimer; -}; - -/*********************************************************************** - * USB zero copy wrapper - managed send buffer - **********************************************************************/ -class usb_zero_copy_wrapper_msb : public managed_send_buffer{ -public: - usb_zero_copy_wrapper_msb(const usb_zero_copy::sptr internal, const size_t fragmentation_size): - _internal(internal), _fragmentation_size(fragmentation_size) - { - _ok_to_auto_flush = false; - _task = uhd::task::make(boost::bind(&usb_zero_copy_wrapper_msb::auto_flush, this)); - } - - ~usb_zero_copy_wrapper_msb(void) - { - //ensure the task has exited before anything auto deconstructs - _task.reset(); - } - - void release(void){ - boost::mutex::scoped_lock lock(_mutex); - _ok_to_auto_flush = true; - - //get a reference to the VITA header before incrementing - const boost::uint32_t vita_header = reinterpret_cast<const boost::uint32_t *>(_mem_buffer_tip)[0]; - - _bytes_in_buffer += size(); - _mem_buffer_tip += size(); - - //extract VITA end of packet flag, we must force flush under eof conditions - const bool eop = (uhd::wtohx(vita_header) & (0x1 << 24)) != 0; - const bool full = _bytes_in_buffer >= (_last_send_buff->size() - _fragmentation_size); - if (eop or full){ - _last_send_buff->commit(_bytes_in_buffer); - _last_send_buff.reset(); - - //notify the auto-flusher to restart its timed_wait - lock.unlock(); _cond.notify_one(); - } - } - - UHD_INLINE sptr get_new(const double timeout){ - boost::mutex::scoped_lock lock(_mutex); - _ok_to_auto_flush = false; - - if (not _last_send_buff){ - _last_send_buff = _internal->get_send_buff(timeout); - if (not _last_send_buff) return sptr(); - _mem_buffer_tip = _last_send_buff->cast<char *>(); - _bytes_in_buffer = 0; - } - - return make(this, _mem_buffer_tip, _fragmentation_size); - } - -private: - usb_zero_copy::sptr _internal; - const size_t _fragmentation_size; - managed_send_buffer::sptr _last_send_buff; - size_t _bytes_in_buffer; - char *_mem_buffer_tip; - - //private variables for auto flusher - boost::mutex _mutex; - boost::condition_variable _cond; - uhd::task::sptr _task; - bool _ok_to_auto_flush; - - /*! - * The auto flusher ensures that buffers are force committed when - * the user has not called get_new() within a certain time window. - */ - void auto_flush(void) - { - boost::mutex::scoped_lock lock(_mutex); - const bool timeout = not _cond.timed_wait(lock, AUTOFLUSH_TIMEOUT); - if (timeout and _ok_to_auto_flush and _last_send_buff and _bytes_in_buffer != 0) - { - _last_send_buff->commit(_bytes_in_buffer); - _last_send_buff.reset(); - } - } -}; - -/*********************************************************************** - * USB zero copy wrapper implementation - **********************************************************************/ -class usb_zero_copy_wrapper : public usb_zero_copy{ -public: - usb_zero_copy_wrapper(sptr usb_zc, const size_t frame_boundary): - _internal_zc(usb_zc), - _frame_boundary(frame_boundary), - _next_recv_buff_index(0) - { - for (size_t i = 0; i < this->get_num_recv_frames(); i++){ - _mrb_pool.push_back(boost::make_shared<usb_zero_copy_wrapper_mrb>()); - } - _the_only_msb = boost::make_shared<usb_zero_copy_wrapper_msb>(usb_zc, frame_boundary); - } - - managed_recv_buffer::sptr get_recv_buff(double timeout){ - //lazy flush mechanism - negative timeout - if (timeout < 0.0) - { - _last_recv_buff.reset(); - while (_internal_zc->get_recv_buff()){} - return managed_recv_buffer::sptr(); - } - - //attempt to get a managed recv buffer - if (not _last_recv_buff){ - _last_recv_buff = _internal_zc->get_recv_buff(timeout); - _last_recv_offset = 0; //reset offset into buffer - } - - //get the buffer to be returned to the user - if (_next_recv_buff_index == _mrb_pool.size()) _next_recv_buff_index = 0; - return _mrb_pool[_next_recv_buff_index]->get_new( - _last_recv_buff, _last_recv_offset, timeout, _next_recv_buff_index - ); - } - - size_t get_num_recv_frames(void) const{ - return _internal_zc->get_num_recv_frames(); - } - - size_t get_recv_frame_size(void) const{ - return std::min(_frame_boundary, _internal_zc->get_recv_frame_size()); - } - - managed_send_buffer::sptr get_send_buff(double timeout){ - return _the_only_msb->get_new(timeout); - } - - size_t get_num_send_frames(void) const{ - return _internal_zc->get_num_send_frames(); - } - - size_t get_send_frame_size(void) const{ - return std::min(_frame_boundary, _internal_zc->get_send_frame_size()); - } - -private: - sptr _internal_zc; - size_t _frame_boundary; - std::vector<boost::shared_ptr<usb_zero_copy_wrapper_mrb> > _mrb_pool; - boost::shared_ptr<usb_zero_copy_wrapper_msb> _the_only_msb; - - //state for last recv buffer to create multiple managed buffers - managed_recv_buffer::sptr _last_recv_buff; - size_t _last_recv_offset; - size_t _next_recv_buff_index; -}; - -/*********************************************************************** - * USB zero copy wrapper factory function - **********************************************************************/ -usb_zero_copy::sptr usb_zero_copy::make_wrapper( - sptr usb_zc, size_t usb_frame_boundary -){ - return sptr(new usb_zero_copy_wrapper(usb_zc, usb_frame_boundary)); -} |