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
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
|
//
// Copyright 2019 Ettus Research, a National Instruments Brand
//
// SPDX-License-Identifier: GPL-3.0-or-later
//
#pragma once
#include <uhd/config.hpp>
#include <uhd/transport/frame_buff.hpp>
#include <uhd/types/device_addr.hpp>
#include <uhdlib/transport/adapter_info.hpp>
#include <rte_ethdev.h>
#include <rte_ether.h>
#include <rte_flow.h>
#include <rte_mbuf.h>
#include <rte_mempool.h>
#include <rte_spinlock.h>
#include <rte_version.h>
#include <unordered_map>
#include <array>
#include <atomic>
#include <mutex>
#include <set>
#include <string>
/* NOTE: There are changes to all the network standard fields in 19.x */
namespace uhd { namespace transport {
class dpdk_io_service;
namespace dpdk {
struct arp_entry;
using queue_id_t = uint16_t;
using port_id_t = uint16_t;
using ipv4_addr = uint32_t;
class dpdk_adapter_info : public adapter_info
{
public:
dpdk_adapter_info(port_id_t port) : _port(port) {}
~dpdk_adapter_info() {}
std::string to_string()
{
return std::string("DPDK:") + std::to_string(_port);
}
bool operator==(const dpdk_adapter_info& rhs) const
{
return (_port == rhs._port);
}
private:
// Port ID
port_id_t _port;
};
/*!
* Packet/Frame buffer class for DPDK
*
* This class is intended to be placed in the private area of the rte_mbuf, and
* its memory is part of the rte_mbuf, so its life is tied to the underlying
* buffer (or more precisely, the encapsulating one).
*/
class dpdk_frame_buff : public frame_buff
{
public:
dpdk_frame_buff(struct rte_mbuf* mbuf) : _mbuf(mbuf)
{
_data = rte_pktmbuf_mtod(mbuf, void*);
_packet_size = 0;
}
~dpdk_frame_buff() = default;
/*!
* Simple getter for the underlying rte_mbuf.
* The rte_mbuf may need further modification before sending packets,
* like adjusting the IP and UDP lengths.
*/
inline struct rte_mbuf* get_pktmbuf()
{
return _mbuf;
}
/*!
* Move the data pointer by the indicated size, to some desired
* encapsulated frame.
*
* \param hdr_size Size (in bytes) of the headers to skip. Can be negative
* to pull the header back.
*/
inline void header_jump(ssize_t hdr_size)
{
_data = (void*)((uint8_t*)_data + hdr_size);
}
//! Embedded list node's next ptr
dpdk_frame_buff* next = nullptr;
//! Embedded list node's prev ptr
dpdk_frame_buff* prev = nullptr;
private:
struct rte_mbuf* _mbuf;
};
/*!
* The size (in bytes) of the private area reserved within the rte_mbuf.
* This portion of the rte_mbuf is used for the embedded dpdk_frame_buff data
* structure.
*/
constexpr size_t DPDK_MBUF_PRIV_SIZE =
RTE_ALIGN(sizeof(struct dpdk_frame_buff), RTE_MBUF_PRIV_ALIGN);
/*!
* Class representing a DPDK NIC port
*
* The dpdk_port object possesses all the data needed to send and receive
* packets between this port and a remote host.
*
* The logical link can then get the packet buffer pools associated with this
* NIC port and use them to send and receive packets.
*
* dpdk_port objects are _only_ created by the dpdk_ctx.
*/
class dpdk_port
{
public:
using uptr = std::unique_ptr<dpdk_port>;
/*! Construct a DPDK NIC port and bring the link up
*
* \param port The port ID
* \param mtu The intended MTU for the port
* \param num_queues Number of DMA queues to reserve for this port
* \param num_desc The number of descriptors per DMA queue
* \param rx_pktbuf_pool A pointer to the port's RX packet buffer pool
* \param tx_pktbuf_pool A pointer to the port's TX packet buffer pool
* \param ipv4_address The IPv4 network address (w/ netmask)
* \return A unique_ptr to a dpdk_port object
*/
static dpdk_port::uptr make(port_id_t port,
size_t mtu,
uint16_t num_queues,
uint16_t num_desc,
struct rte_mempool* rx_pktbuf_pool,
struct rte_mempool* tx_pktbuf_pool,
std::string ipv4_address);
dpdk_port(port_id_t port,
size_t mtu,
uint16_t num_queues,
uint16_t num_desc,
struct rte_mempool* rx_pktbuf_pool,
struct rte_mempool* tx_pktbuf_pool,
std::string ipv4_address);
~dpdk_port();
/*! Getter for this port's ID
* \return this port's ID
*/
inline port_id_t get_port_id() const
{
return _port;
}
inline dpdk_adapter_info get_adapter_info() const
{
return dpdk_adapter_info(_port);
}
/*! Getter for this port's MAC address
* \return this port's MAC address
*/
inline ether_addr get_mac_addr() const
{
return _mac_addr;
}
/*! Getter for this port's MTU
* \return this port's MTU
*/
inline size_t get_mtu() const
{
return _mtu;
}
/*! Getter for this port's IPv4 address
* \return this port's IPv4 address (in network order)
*/
inline ipv4_addr get_ipv4() const
{
return _ipv4;
}
/*! Getter for this port's subnet mask
* \return this port's subnet mask (in network order)
*/
inline ipv4_addr get_netmask() const
{
return _netmask;
}
/*! Getter for this port's total DMA queue count, including initialized,
* but unallocated queues
*
* \return The number of queues initialized on this port
*/
inline size_t get_queue_count() const
{
return _num_queues;
}
/*! Getter for this port's RX packet buffer memory pool
*
* \return The RX packet buffer pool
*/
inline struct rte_mempool* get_rx_pktbuf_pool() const
{
return _rx_pktbuf_pool;
}
/*! Getter for this port's TX packet buffer memory pool
*
* \return The TX packet buffer pool
*/
inline struct rte_mempool* get_tx_pktbuf_pool() const
{
return _tx_pktbuf_pool;
}
/*! Determine if the destination address is a broadcast address for this port
* \param dst_ipv4_addr The destination IPv4 address (in network order)
* \return whether the destination address matches this port's broadcast address
*/
inline bool dst_is_broadcast(const ipv4_addr dst_ipv4_addr) const
{
uint32_t network = _netmask | ((~_netmask) & dst_ipv4_addr);
return (network == 0xffffffff);
}
/*!
* Allocate a UDP port and return it in network order
*
* \param udp_port UDP port to attempt to allocate. Use 0 for no preference.
* \return 0 for failure, else the allocated UDP port in network order.
*/
uint16_t alloc_udp_port(uint16_t udp_port);
private:
friend uhd::transport::dpdk_io_service;
/*!
* Construct and transmit an ARP reply (for the given ARP request)
*/
int _arp_reply(queue_id_t queue_id, struct arp_hdr* arp_req);
port_id_t _port;
size_t _mtu;
size_t _num_queues;
struct rte_mempool* _rx_pktbuf_pool;
struct rte_mempool* _tx_pktbuf_pool;
struct ether_addr _mac_addr;
ipv4_addr _ipv4;
ipv4_addr _netmask;
// Structures protected by mutex
std::mutex _mutex;
std::set<uint16_t> _udp_ports;
uint16_t _next_udp_port = 0xffff;
// Structures protected by spin lock
rte_spinlock_t _spinlock = RTE_SPINLOCK_INITIALIZER;
std::unordered_map<ipv4_addr, struct arp_entry*> _arp_table;
};
/*!
* Handles initialization of DPDK, configuration of ports, setting up DMA
* engines/queues, etc.
*/
class dpdk_ctx : uhd::noncopyable, public std::enable_shared_from_this<dpdk_ctx>
{
public:
using sptr = std::shared_ptr<dpdk_ctx>;
/*! Factory to generate the single dpdk_ctx instance, but with the ability
* to release the resources
*
* FIXME: Without C++17, this mechanism has a race condition between
* creating and destroying the single instance. Don't do those at the
* same time. (shared_from_this() doesn't check the weak_ptr in C++14)
*/
static dpdk_ctx::sptr get();
dpdk_ctx(void);
~dpdk_ctx(void);
/*!
* Init DPDK environment, including DPDK's EAL.
* This will make available information about the DPDK-assigned NIC devices.
*
* \param user_args User args passed in to override config files
*/
void init(const device_addr_t& user_args);
/*!
* Get port from provided MAC address
* \param mac_addr MAC address
* \return pointer to port if match found, else nullptr
*/
dpdk_port* get_port(struct ether_addr mac_addr) const;
/*!
* Get port structure from provided port ID
* \param port Port ID
* \return pointer to port if match found, else nullptr
*/
dpdk_port* get_port(port_id_t port) const;
/*!
* \return Returns number of ports registered to DPDK.
*/
int get_port_count(void);
/*!
* \param port_id NIC port ID
* \return Returns number of DMA queues available for a given port
*/
int get_port_queue_count(port_id_t portid);
/*!
* \param portid NIC port ID
* \return Returns 0 if link is down, 1 if link is up
*/
int get_port_link_status(port_id_t portid) const;
/*!
* Get port for routing packet destined for given address
* \param addr Destination address
* \return pointer to the port from routing table
*/
dpdk_port* get_route(const std::string& addr) const;
/*!
* \return whether init() has been called
*/
bool is_init_done(void) const;
/*! Return a reference to an IO service given a port ID
*/
std::shared_ptr<uhd::transport::dpdk_io_service> get_io_service(const size_t port_id);
private:
/*! Convert the args to DPDK's EAL args and Initialize the EAL
*
* \param eal_args The global DPDK configuration
*/
void _eal_init(const device_addr_t& eal_args);
/*! Either allocate or return a pointer to the RX packet buffer pool for the
* given CPU socket
*
* \param cpu_socket The CPU socket ID
* \param num_bufs Number of buffers to allocate to the pool
* \return A pointer to the memory pool
*/
struct rte_mempool* _get_rx_pktbuf_pool(unsigned int cpu_socket, size_t num_bufs);
/*! Either allocate or return a pointer to the TX packet buffer pool for the
* given CPU socket
*
* \param cpu_socket The CPU socket ID
* \param num_bufs Number of buffers to allocate to the pool
* \return A pointer to the memory pool
*/
struct rte_mempool* _get_tx_pktbuf_pool(unsigned int cpu_socket, size_t num_bufs);
size_t _mtu;
int _num_mbufs;
int _mbuf_cache_size;
int _link_init_timeout;
std::mutex _init_mutex;
std::atomic<bool> _init_done;
uhd::dict<uint32_t, port_id_t> _routes;
std::unordered_map<port_id_t, dpdk_port::uptr> _ports;
std::vector<struct rte_mempool*> _rx_pktbuf_pools;
std::vector<struct rte_mempool*> _tx_pktbuf_pools;
// Store all the I/O services, and also store the corresponding port ID
std::map<std::shared_ptr<uhd::transport::dpdk_io_service>, std::vector<size_t>>
_io_srv_portid_map;
};
} // namespace dpdk
}} // namespace uhd::transport
|