diff options
| author | Martin Braun <martin.braun@ettus.com> | 2019-08-19 17:14:11 -0700 | 
|---|---|---|
| committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 11:49:40 -0800 | 
| commit | 4c6ee083ef13184ba828d73b9dd2a95d7fa5453f (patch) | |
| tree | 3364e7e147ed229732fca66282323c65d5de4102 /host/lib/include | |
| parent | a4274c19ebb42db112aa68a26fa499a52b9dd103 (diff) | |
| download | uhd-4c6ee083ef13184ba828d73b9dd2a95d7fa5453f.tar.gz uhd-4c6ee083ef13184ba828d73b9dd2a95d7fa5453f.tar.bz2 uhd-4c6ee083ef13184ba828d73b9dd2a95d7fa5453f.zip | |
rfnoc: Add nirio_link link object
Diffstat (limited to 'host/lib/include')
| -rw-r--r-- | host/lib/include/uhdlib/transport/nirio_link.hpp | 210 | 
1 files changed, 210 insertions, 0 deletions
| diff --git a/host/lib/include/uhdlib/transport/nirio_link.hpp b/host/lib/include/uhdlib/transport/nirio_link.hpp new file mode 100644 index 000000000..055874146 --- /dev/null +++ b/host/lib/include/uhdlib/transport/nirio_link.hpp @@ -0,0 +1,210 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_UHD_TRANSPORT_NIRIO_LINK_HPP +#define INCLUDED_UHD_TRANSPORT_NIRIO_LINK_HPP + +#include <uhd/config.hpp> +#include <uhd/transport/buffer_pool.hpp> +#include <uhd/transport/nirio/nirio_fifo.h> +#include <uhd/transport/nirio/niriok_proxy.h> +#include <uhd/transport/nirio/niusrprio_session.h> +#include <uhd/types/device_addr.hpp> +#include <uhdlib/transport/adapter_info.hpp> +#include <uhdlib/transport/link_base.hpp> +#include <uhdlib/transport/links.hpp> +#include <memory> +#include <vector> + +namespace uhd { namespace transport { + +using fifo_data_t = uint64_t; + +/*! NI-RIO frame_buff + * + * NI-RIO internally manages the memory, so these frame_buff objects just need + * a way of internally manipulating the _data pointer + */ +class nirio_frame_buff : public frame_buff +{ +public: +    nirio_frame_buff() +    { +        _data = nullptr; +    } + +    fifo_data_t** get_fifo_ptr_ref() +    { +        return reinterpret_cast<fifo_data_t**>(&_data); +    }; +}; + +class nirio_adapter_info : public adapter_info +{ +public: +    nirio_adapter_info(const std::string& resource) : _resource(resource) {} + +    ~nirio_adapter_info() {} + +    std::string to_string() +    { +        return std::string("NIRIO:") + _resource; +    } + +    bool operator==(const nirio_adapter_info& rhs) const +    { +        return (_resource == rhs._resource); +    } + +private: +    const std::string _resource; +}; + +/*! Link object to talking to NI-RIO USRPs (X310 variants using PCIe) + * + * \b Note: This link cannot release frame buffers out of order, which means it + *          can't be used with an IO service that does that. + */ +class nirio_link : public recv_link_base<nirio_link>, public send_link_base<nirio_link> +{ +public: +    using sptr = std::shared_ptr<nirio_link>; + +    ~nirio_link(); + +    /*! Make a new NI-RIO link. +     * +     * \param addr a string representing the destination address +     * \param port a string representing the destination port +     * \param params Values for frame sizes, num frames, and buffer sizes +     * \param[out] recv_socket_buff_size Returns the recv socket buffer size +     * \param[out] send_socket_buff_size Returns the send socket buffer size +     */ +    static sptr make(uhd::niusrprio::niusrprio_session::sptr fpga_session, +        const uint32_t instance, +        const link_params_t& params, +        const uhd::device_addr_t& hints); + +    /*! +     * Get the physical adapter ID used for this link +     */ +    adapter_id_t get_send_adapter_id() const +    { +        return _adapter_id; +    } + +    /*! +     * Get the physical adapter ID used for this link +     */ +    adapter_id_t get_recv_adapter_id() const +    { +        return _adapter_id; +    } + +private: +    using recv_link_base_t = recv_link_base<nirio_link>; +    using send_link_base_t = send_link_base<nirio_link>; + +    // Friend declarations to allow base classes to call private methods +    friend recv_link_base_t; +    friend send_link_base_t; + +    nirio_link(uhd::niusrprio::niusrprio_session::sptr fpga_session, +        uint32_t instance, +        const link_params_t& params); + +    /************************************************************************** +     * NI-RIO specific helpers +     *************************************************************************/ +    void _flush_rx_buff(); + +    void _wait_until_stream_ready(); + +    /************************************************************************** +     * recv_link/send_link API +     *************************************************************************/ +    // Methods called by recv_link_base +    UHD_FORCE_INLINE size_t get_recv_buff_derived(frame_buff& buff, int32_t timeout_ms) +    { +        using namespace uhd::niusrprio; +        nirio_status status    = 0; +        size_t elems_acquired  = 0; +        size_t elems_remaining = 0; +        // This will modify the data pointer in buff if successful: +        fifo_data_t** data_ptr = static_cast<nirio_frame_buff&>(buff).get_fifo_ptr_ref(); +        nirio_status_chain(_recv_fifo->acquire(*data_ptr, +                               _link_params.recv_frame_size / sizeof(fifo_data_t), +                               static_cast<uint32_t>(timeout_ms * 1000), +                               elems_acquired, +                               elems_remaining), +            status); +        const size_t length = elems_acquired * sizeof(fifo_data_t); + +        if (nirio_status_not_fatal(status)) { +            return length; +        } else if (status == NiRio_Status_CommunicationTimeout) { +            nirio_status_to_exception(status, "NI-RIO PCIe data transfer failed."); +        } +        return 0; // zero for timeout or error. +    } + +    UHD_FORCE_INLINE void release_recv_buff_derived(frame_buff& /*buff*/) +    { +        _recv_fifo->release(_link_params.recv_frame_size / sizeof(fifo_data_t)); +    } + +    // Methods called by send_link_base +    UHD_FORCE_INLINE bool get_send_buff_derived(frame_buff& buff, int32_t timeout_ms) +    { +        using namespace uhd::niusrprio; +        nirio_status status    = 0; +        size_t elems_acquired  = 0; +        size_t elems_remaining = 0; +        // This will modify the data pointer in buff if successful: +        fifo_data_t** data_ptr = static_cast<nirio_frame_buff&>(buff).get_fifo_ptr_ref(); +        nirio_status_chain(_send_fifo->acquire(*data_ptr, +                               _link_params.send_frame_size / sizeof(fifo_data_t), +                               static_cast<uint32_t>(timeout_ms * 1000), +                               elems_acquired, +                               elems_remaining), +            status); +        // const size_t length = elems_acquired * sizeof(fifo_data_t); + +        if (nirio_status_not_fatal(status)) { +            return true; +        } else if (status == NiRio_Status_CommunicationTimeout) { +            nirio_status_to_exception(status, "NI-RIO PCIe data transfer failed."); +        } +        return false; +    } + +    UHD_FORCE_INLINE void release_send_buff_derived(frame_buff& /*buff*/) +    { +        _send_fifo->release(_link_params.send_frame_size / sizeof(fifo_data_t)); +    } + +    /************************************************************************** +     * Private attributes +     *************************************************************************/ +    //! Reference to the NI-RIO session +    niusrprio::niusrprio_session::sptr _fpga_session; +    //! DMA channel index +    const uint32_t _fifo_instance; +    //! Recv and send FIFO objects +    uhd::niusrprio::nirio_fifo<fifo_data_t>::sptr _recv_fifo, _send_fifo; + +    const link_params_t _link_params; + +    std::vector<nirio_frame_buff> _recv_buffs; +    std::vector<nirio_frame_buff> _send_buffs; + +    adapter_id_t _adapter_id; +}; + +}} // namespace uhd::transport + + +#endif /* INCLUDED_UHD_TRANSPORT_NIRIO_LINK_HPP */ | 
