aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/include/uhdlib/transport/link_base.hpp
blob: 078b82fc646d65c12f85ee8c70cbf0e3fe539686 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
//
// Copyright 2019 Ettus Research, a National Instruments Brand
//
// SPDX-License-Identifier: GPL-3.0-or-later
//

#pragma once

#include <uhdlib/transport/link_if.hpp>
#include <cassert>
#include <vector>

namespace uhd { namespace transport {
namespace detail {

/*!
 * Container for free buffers used by link base classes.
 */
class free_buff_pool
{
public:
    free_buff_pool(const size_t capacity)
    {
        _buffs.reserve(capacity);
    }

    frame_buff* pop()
    {
        // Free buffer pool should not be empty unless user has requested more
        // buffers than the number of frames in the link.
        assert(!_buffs.empty());

        frame_buff* buff = _buffs.back();
        _buffs.pop_back();
        return buff;
    }

    void push(frame_buff* buff)
    {
        _buffs.push_back(buff);
    }

private:
    std::vector<frame_buff*> _buffs;
};

} // namespace detail

/*!
 * Reusable base class implementation of send_link_iface. Link implementations
 * should derive from this template and pass their own type as the derived_t
 * template parameter.
 *
 * The base class template manages a pool of frame_buff pointers and implements
 * the link interface methods.
 *
 * This template requires the following methods in the derived class:
 *   bool get_send_buff_derived(frame_buff& buf, int32_t timeout_ms);
 *   void release_send_buf_derived(frame_buff& buf);
 *
 * Additionally, the subclass must call preload_free_buf for each frame_buff
 * object it owns during initialization to add it to the free buffer pool.
 *
 * \param derived_t type of the derived class
 */
template <typename derived_t>
class send_link_base : public virtual send_link_if
{
public:
    send_link_base(const size_t num_frames, const size_t frame_size)
        : _send_frame_size(frame_size)
        , _num_send_frames(num_frames)
        , _free_send_buffs(num_frames)
    {
    }

    virtual size_t get_num_send_frames() const
    {
        return _num_send_frames;
    }

    virtual size_t get_send_frame_size() const
    {
        return _send_frame_size;
    }

    virtual frame_buff::uptr get_send_buff(int32_t timeout_ms)
    {
        frame_buff* buff = _free_send_buffs.pop();

        // Call the derived class for link-specific implementation
        auto* derived = static_cast<derived_t*>(this);

        if (!derived->get_send_buff_derived(*buff, timeout_ms)) {
            _free_send_buffs.push(buff);
            return frame_buff::uptr();
        }

        return frame_buff::uptr(buff);
    }

    virtual void release_send_buff(frame_buff::uptr buff)
    {
        frame_buff* buff_ptr = buff.release();
        assert(buff_ptr);

        if (buff_ptr->packet_size() != 0) {
            // Call the derived class for link-specific implementation
            auto* derived = static_cast<derived_t*>(this);
            derived->release_send_buff_derived(*buff_ptr);
        }

        // Reset buff and re-add to free pool
        buff_ptr->set_packet_size(0);
        _free_send_buffs.push(buff_ptr);
    }

protected:
    /*!
     * Add buffer pointer to free buffer pool.
     *
     * Derived classes should call this method during initialization for each
     * frame buffer it owns.
     *
     * \param buffer pointer to the buffer to add to the free buffer pool.
     */
    void preload_free_buff(frame_buff* buff)
    {
        _free_send_buffs.push(buff);
    }

private:
    size_t _send_frame_size;
    size_t _num_send_frames;
    detail::free_buff_pool _free_send_buffs;
};

/*!
 * Reusable base class implementation of recv_link_if. Link implementations
 * should derive from this template and pass their own type as the derived_t
 * template parameter.
 *
 * The base class template manages a pool of free_buff pointers and implements
 * the link interface methods.
 *
 * This template requires the following methods in the derived class:
 *   size_t get_recv_buff_derived(frame_buff& buff, int32_t timeout_ms);
 *   void release_recv_buff_derived(frame_buff& buff);
 *
 * Additionally, the subclass must call preload_free_buff for each
 * frame_buff object it owns during initialization to add it to the free
 * buff pool.
 *
 * \param derived_t type of the derived class
 */
template <typename derived_t>
class recv_link_base : public virtual recv_link_if
{
public:
    recv_link_base(const size_t num_frames, const size_t frame_size)
        : _recv_frame_size(frame_size)
        , _num_recv_frames(num_frames)
        , _free_recv_buffs(num_frames)
    {
    }

    virtual size_t get_num_recv_frames() const
    {
        return _num_recv_frames;
    }

    virtual size_t get_recv_frame_size() const
    {
        return _recv_frame_size;
    }

    virtual frame_buff::uptr get_recv_buff(int32_t timeout_ms)
    {
        frame_buff* buff = _free_recv_buffs.pop();

        // Call the derived class for link specific implementation
        auto* derived = static_cast<derived_t*>(this);

        size_t len = derived->get_recv_buff_derived(*buff, timeout_ms);

        if (len == 0) {
            _free_recv_buffs.push(buff);
            return frame_buff::uptr();
        } else {
            buff->set_packet_size(len);
            return frame_buff::uptr(buff);
        }
    }

    virtual void release_recv_buff(frame_buff::uptr buff)
    {
        frame_buff* buff_ptr = buff.release();
        assert(buff_ptr);

        // Call the derived class for link specific implementation
        auto* derived = static_cast<derived_t*>(this);

        derived->release_recv_buff_derived(*buff_ptr);

        // Reset buffer and re-add to free pool
        buff_ptr->set_packet_size(0);
        _free_recv_buffs.push(buff_ptr);
    }

protected:
    /*!
     * Add buffer pointer to free buffer pool.
     *
     * Derived classes should call this method during initialization for each
     * frame buffer it owns.
     *
     * \param buffer pointer to the buffer to add to the free buffer pool.
     */
    void preload_free_buff(frame_buff* buff)
    {
        _free_recv_buffs.push(buff);
    }

private:
    size_t _recv_frame_size;
    size_t _num_recv_frames;
    detail::free_buff_pool _free_recv_buffs;
};

}} // namespace uhd::transport