1/*-
2 * SPDX-License-Identifier: BSD-3-Clause
3 *
4 * This header is BSD licensed so anyone can use the definitions to implement
5 * compatible drivers/servers.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 *    notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 *    notice, this list of conditions and the following disclaimer in the
14 *    documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of IBM nor the names of its contributors
16 *    may be used to endorse or promote products derived from this software
17 *    without specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
20 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
21 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL IBM OR CONTRIBUTORS BE LIABLE
22 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28 * SUCH DAMAGE.
29 */
30
31#ifndef _VIRTIO_NET_H
32#define _VIRTIO_NET_H
33
34/* The feature bitmap for virtio net */
35#define VIRTIO_NET_F_CSUM		 0x000001 /* Host handles pkts w/ partial csum */
36#define VIRTIO_NET_F_GUEST_CSUM		 0x000002 /* Guest handles pkts w/ partial csum*/
37#define VIRTIO_NET_F_CTRL_GUEST_OFFLOADS 0x000004 /* Dynamic offload configuration. */
38#define VIRTIO_NET_F_MTU		 0x000008 /* Initial MTU advice */
39#define VIRTIO_NET_F_MAC		 0x000020 /* Host has given MAC address. */
40#define VIRTIO_NET_F_GSO		 0x000040 /* Host handles pkts w/ any GSO type */
41#define VIRTIO_NET_F_GUEST_TSO4		 0x000080 /* Guest can handle TSOv4 in. */
42#define VIRTIO_NET_F_GUEST_TSO6		 0x000100 /* Guest can handle TSOv6 in. */
43#define VIRTIO_NET_F_GUEST_ECN		 0x000200 /* Guest can handle TSO[6] w/ ECN in. */
44#define VIRTIO_NET_F_GUEST_UFO		 0x000400 /* Guest can handle UFO in. */
45#define VIRTIO_NET_F_HOST_TSO4		 0x000800 /* Host can handle TSOv4 in. */
46#define VIRTIO_NET_F_HOST_TSO6		 0x001000 /* Host can handle TSOv6 in. */
47#define VIRTIO_NET_F_HOST_ECN		 0x002000 /* Host can handle TSO[6] w/ ECN in. */
48#define VIRTIO_NET_F_HOST_UFO		 0x004000 /* Host can handle UFO in. */
49#define VIRTIO_NET_F_MRG_RXBUF		 0x008000 /* Host can merge receive buffers. */
50#define VIRTIO_NET_F_STATUS		 0x010000 /* virtio_net_config.status available*/
51#define VIRTIO_NET_F_CTRL_VQ		 0x020000 /* Control channel available */
52#define VIRTIO_NET_F_CTRL_RX		 0x040000 /* Control channel RX mode support */
53#define VIRTIO_NET_F_CTRL_VLAN		 0x080000 /* Control channel VLAN filtering */
54#define VIRTIO_NET_F_CTRL_RX_EXTRA	 0x100000 /* Extra RX mode control support */
55#define VIRTIO_NET_F_GUEST_ANNOUNCE	 0x200000 /* Announce device on network */
56#define VIRTIO_NET_F_MQ			 0x400000 /* Device supports Receive Flow Steering */
57#define VIRTIO_NET_F_CTRL_MAC_ADDR	 0x800000 /* Set MAC address */
58#define VIRTIO_NET_F_SPEED_DUPLEX	 (1ULL << 63) /* Device set linkspeed and duplex */
59
60#define VIRTIO_NET_S_LINK_UP	1	/* Link is up */
61#define VIRTIO_NET_S_ANNOUNCE	2	/* Announcement is needed */
62
63struct virtio_net_config {
64	/* The config defining mac address (if VIRTIO_NET_F_MAC) */
65	uint8_t		mac[ETHER_ADDR_LEN];
66	/* See VIRTIO_NET_F_STATUS and VIRTIO_NET_S_* above */
67	uint16_t	status;
68	/* Maximum number of each of transmit and receive queues;
69	 * see VIRTIO_NET_F_MQ and VIRTIO_NET_CTRL_MQ.
70	 * Legal values are between 1 and 0x8000.
71	 */
72	uint16_t	max_virtqueue_pairs;
73	/* Default maximum transmit unit advice */
74	uint16_t	mtu;
75	/*
76	 * speed, in units of 1Mb. All values 0 to INT_MAX are legal.
77	 * Any other value stands for unknown.
78	 */
79	uint32_t	speed;
80	/*
81	 * 0x00 - half duplex
82	 * 0x01 - full duplex
83	 * Any other value stands for unknown.
84	 */
85	uint8_t		duplex;
86} __packed;
87
88/*
89 * This header comes first in the scatter-gather list.  If you don't
90 * specify GSO or CSUM features, you can simply ignore the header.
91 *
92 * This is bitwise-equivalent to the legacy struct virtio_net_hdr_mrg_rxbuf,
93 * only flattened.
94 */
95struct virtio_net_hdr_v1 {
96#define VIRTIO_NET_HDR_F_NEEDS_CSUM	1	/* Use csum_start, csum_offset */
97#define VIRTIO_NET_HDR_F_DATA_VALID	2	/* Csum is valid */
98	uint8_t flags;
99#define VIRTIO_NET_HDR_GSO_NONE		0	/* Not a GSO frame */
100#define VIRTIO_NET_HDR_GSO_TCPV4	1	/* GSO frame, IPv4 TCP (TSO) */
101#define VIRTIO_NET_HDR_GSO_UDP		3	/* GSO frame, IPv4 UDP (UFO) */
102#define VIRTIO_NET_HDR_GSO_TCPV6	4	/* GSO frame, IPv6 TCP */
103#define VIRTIO_NET_HDR_GSO_ECN		0x80	/* TCP has ECN set */
104	uint8_t gso_type;
105	uint16_t hdr_len;	/* Ethernet + IP + tcp/udp hdrs */
106	uint16_t gso_size;	/* Bytes to append to hdr_len per frame */
107	uint16_t csum_start;	/* Position to start checksumming from */
108	uint16_t csum_offset;	/* Offset after that to place checksum */
109	uint16_t num_buffers;	/* Number of merged rx buffers */
110};
111
112/*
113 * This header comes first in the scatter-gather list.
114 * For legacy virtio, if VIRTIO_F_ANY_LAYOUT is not negotiated, it must
115 * be the first element of the scatter-gather list.  If you don't
116 * specify GSO or CSUM features, you can simply ignore the header.
117 */
118struct virtio_net_hdr {
119	/* See VIRTIO_NET_HDR_F_* */
120	uint8_t	flags;
121	/* See VIRTIO_NET_HDR_GSO_* */
122	uint8_t gso_type;
123	uint16_t hdr_len;	/* Ethernet + IP + tcp/udp hdrs */
124	uint16_t gso_size;	/* Bytes to append to hdr_len per frame */
125	uint16_t csum_start;	/* Position to start checksumming from */
126	uint16_t csum_offset;	/* Offset after that to place checksum */
127};
128
129/*
130 * This is the version of the header to use when the MRG_RXBUF
131 * feature has been negotiated.
132 */
133struct virtio_net_hdr_mrg_rxbuf {
134	struct virtio_net_hdr hdr;
135	uint16_t num_buffers;	/* Number of merged rx buffers */
136};
137
138/*
139 * Control virtqueue data structures
140 *
141 * The control virtqueue expects a header in the first sg entry
142 * and an ack/status response in the last entry.  Data for the
143 * command goes in between.
144 */
145struct virtio_net_ctrl_hdr {
146	uint8_t class;
147	uint8_t cmd;
148} __packed;
149
150#define VIRTIO_NET_OK	0
151#define VIRTIO_NET_ERR	1
152
153/*
154 * Control the RX mode, ie. promiscuous, allmulti, etc...
155 * All commands require an "out" sg entry containing a 1 byte
156 * state value, zero = disable, non-zero = enable.  Commands
157 * 0 and 1 are supported with the VIRTIO_NET_F_CTRL_RX feature.
158 * Commands 2-5 are added with VIRTIO_NET_F_CTRL_RX_EXTRA.
159 */
160#define VIRTIO_NET_CTRL_RX	0
161#define VIRTIO_NET_CTRL_RX_PROMISC	0
162#define VIRTIO_NET_CTRL_RX_ALLMULTI	1
163#define VIRTIO_NET_CTRL_RX_ALLUNI	2
164#define VIRTIO_NET_CTRL_RX_NOMULTI	3
165#define VIRTIO_NET_CTRL_RX_NOUNI	4
166#define VIRTIO_NET_CTRL_RX_NOBCAST	5
167
168/*
169 * Control the MAC filter table.
170 *
171 * The MAC filter table is managed by the hypervisor, the guest should
172 * assume the size is infinite.  Filtering should be considered
173 * non-perfect, ie. based on hypervisor resources, the guest may
174 * received packets from sources not specified in the filter list.
175 *
176 * In addition to the class/cmd header, the TABLE_SET command requires
177 * two out scatterlists.  Each contains a 4 byte count of entries followed
178 * by a concatenated byte stream of the ETH_ALEN MAC addresses.  The
179 * first sg list contains unicast addresses, the second is for multicast.
180 * This functionality is present if the VIRTIO_NET_F_CTRL_RX feature
181 * is available.
182 *
183 * The ADDR_SET command requests one out scatterlist, it contains a
184 * 6 bytes MAC address. This functionality is present if the
185 * VIRTIO_NET_F_CTRL_MAC_ADDR feature is available.
186 */
187struct virtio_net_ctrl_mac {
188	uint32_t	entries;
189	uint8_t		macs[][ETHER_ADDR_LEN];
190} __packed;
191
192#define VIRTIO_NET_CTRL_MAC	1
193#define VIRTIO_NET_CTRL_MAC_TABLE_SET	0
194#define VIRTIO_NET_CTRL_MAC_ADDR_SET	1
195
196/*
197 * Control VLAN filtering
198 *
199 * The VLAN filter table is controlled via a simple ADD/DEL interface.
200 * VLAN IDs not added may be filtered by the hypervisor.  Del is the
201 * opposite of add.  Both commands expect an out entry containing a 2
202 * byte VLAN ID.  VLAN filtering is available with the
203 * VIRTIO_NET_F_CTRL_VLAN feature bit.
204 */
205#define VIRTIO_NET_CTRL_VLAN	2
206#define VIRTIO_NET_CTRL_VLAN_ADD	0
207#define VIRTIO_NET_CTRL_VLAN_DEL	1
208
209/*
210 * Control link announce acknowledgement
211 *
212 * The command VIRTIO_NET_CTRL_ANNOUNCE_ACK is used to indicate that
213 * driver has received the notification; device would clear the
214 * VIRTIO_NET_S_ANNOUNCE bit in the status field after it receives
215 * this command.
216 */
217#define VIRTIO_NET_CTRL_ANNOUNCE	3
218#define VIRTIO_NET_CTRL_ANNOUNCE_ACK	0
219
220/*
221 * Control Receive Flow Steering
222 *
223 * The command VIRTIO_NET_CTRL_MQ_VQ_PAIRS_SET enables Receive Flow
224 * Steering, specifying the number of the transmit and receive queues
225 * that will be used. After the command is consumed and acked by the
226 * device, the device will not steer new packets on receive virtqueues
227 * other than specified nor read from transmit virtqueues other than
228 * specified. Accordingly, driver should not transmit new packets on
229 * virtqueues other than specified.
230 */
231struct virtio_net_ctrl_mq {
232	uint16_t	virtqueue_pairs;
233} __packed;
234
235#define VIRTIO_NET_CTRL_MQ	4
236#define VIRTIO_NET_CTRL_MQ_VQ_PAIRS_SET		0
237#define VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MIN		1
238#define VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MAX		0x8000
239
240/*
241 * Control network offloads
242 *
243 * Reconfigures the network offloads that Guest can handle.
244 *
245 * Available with the VIRTIO_NET_F_CTRL_GUEST_OFFLOADS feature bit.
246 *
247 * Command data format matches the feature bit mask exactly.
248 *
249 * See VIRTIO_NET_F_GUEST_* for the list of offloads
250 * that can be enabled/disabled.
251 */
252#define VIRTIO_NET_CTRL_GUEST_OFFLOADS		5
253#define VIRTIO_NET_CTRL_GUEST_OFFLOADS_SET	0
254
255/*
256 * Use the checksum offset in the VirtIO header to set the
257 * correct CSUM_* flags.
258 */
259static inline int
260virtio_net_rx_csum_by_offset(struct mbuf *m, uint16_t eth_type, int ip_start,
261			struct virtio_net_hdr *hdr)
262{
263#if defined(INET) || defined(INET6)
264	int offset = hdr->csum_start + hdr->csum_offset;
265#endif
266
267	/* Only do a basic sanity check on the offset. */
268	switch (eth_type) {
269#if defined(INET)
270	case ETHERTYPE_IP:
271		if (__predict_false(offset < ip_start + sizeof(struct ip)))
272			return (1);
273		break;
274#endif
275#if defined(INET6)
276	case ETHERTYPE_IPV6:
277		if (__predict_false(offset < ip_start + sizeof(struct ip6_hdr)))
278			return (1);
279		break;
280#endif
281	default:
282		/* Here we should increment the rx_csum_bad_ethtype counter. */
283		return (1);
284	}
285
286	/*
287	 * Use the offset to determine the appropriate CSUM_* flags. This is
288	 * a bit dirty, but we can get by with it since the checksum offsets
289	 * happen to be different. We assume the host host does not do IPv4
290	 * header checksum offloading.
291	 */
292	switch (hdr->csum_offset) {
293	case offsetof(struct udphdr, uh_sum):
294	case offsetof(struct tcphdr, th_sum):
295		m->m_pkthdr.csum_flags |= CSUM_DATA_VALID | CSUM_PSEUDO_HDR;
296		m->m_pkthdr.csum_data = 0xFFFF;
297		break;
298	default:
299		/* Here we should increment the rx_csum_bad_offset counter. */
300		return (1);
301	}
302
303	return (0);
304}
305
306static inline int
307virtio_net_rx_csum_by_parse(struct mbuf *m, uint16_t eth_type, int ip_start,
308		       struct virtio_net_hdr *hdr)
309{
310	int offset, proto;
311
312	switch (eth_type) {
313#if defined(INET)
314	case ETHERTYPE_IP: {
315		struct ip *ip;
316		if (__predict_false(m->m_len < ip_start + sizeof(struct ip)))
317			return (1);
318		ip = (struct ip *)(m->m_data + ip_start);
319		proto = ip->ip_p;
320		offset = ip_start + (ip->ip_hl << 2);
321		break;
322	}
323#endif
324#if defined(INET6)
325	case ETHERTYPE_IPV6:
326		if (__predict_false(m->m_len < ip_start +
327		    sizeof(struct ip6_hdr)))
328			return (1);
329		offset = ip6_lasthdr(m, ip_start, IPPROTO_IPV6, &proto);
330		if (__predict_false(offset < 0))
331			return (1);
332		break;
333#endif
334	default:
335		/* Here we should increment the rx_csum_bad_ethtype counter. */
336		return (1);
337	}
338
339	switch (proto) {
340	case IPPROTO_TCP:
341		if (__predict_false(m->m_len < offset + sizeof(struct tcphdr)))
342			return (1);
343		m->m_pkthdr.csum_flags |= CSUM_DATA_VALID | CSUM_PSEUDO_HDR;
344		m->m_pkthdr.csum_data = 0xFFFF;
345		break;
346	case IPPROTO_UDP:
347		if (__predict_false(m->m_len < offset + sizeof(struct udphdr)))
348			return (1);
349		m->m_pkthdr.csum_flags |= CSUM_DATA_VALID | CSUM_PSEUDO_HDR;
350		m->m_pkthdr.csum_data = 0xFFFF;
351		break;
352	default:
353		/*
354		 * For the remaining protocols, FreeBSD does not support
355		 * checksum offloading, so the checksum will be recomputed.
356		 */
357#if 0
358		if_printf(ifp, "cksum offload of unsupported "
359		    "protocol eth_type=%#x proto=%d csum_start=%d "
360		    "csum_offset=%d\n", __func__, eth_type, proto,
361		    hdr->csum_start, hdr->csum_offset);
362#endif
363		break;
364	}
365
366	return (0);
367}
368
369/*
370 * Set the appropriate CSUM_* flags. Unfortunately, the information
371 * provided is not directly useful to us. The VirtIO header gives the
372 * offset of the checksum, which is all Linux needs, but this is not
373 * how FreeBSD does things. We are forced to peek inside the packet
374 * a bit.
375 *
376 * It would be nice if VirtIO gave us the L4 protocol or if FreeBSD
377 * could accept the offsets and let the stack figure it out.
378 */
379static inline int
380virtio_net_rx_csum(struct mbuf *m, struct virtio_net_hdr *hdr)
381{
382	struct ether_header *eh;
383	struct ether_vlan_header *evh;
384	uint16_t eth_type;
385	int offset, error;
386
387	if ((hdr->flags & (VIRTIO_NET_HDR_F_NEEDS_CSUM |
388	    VIRTIO_NET_HDR_F_DATA_VALID)) == 0) {
389		return (0);
390	}
391
392	eh = mtod(m, struct ether_header *);
393	eth_type = ntohs(eh->ether_type);
394	if (eth_type == ETHERTYPE_VLAN) {
395		/* BMV: We should handle nested VLAN tags too. */
396		evh = mtod(m, struct ether_vlan_header *);
397		eth_type = ntohs(evh->evl_proto);
398		offset = sizeof(struct ether_vlan_header);
399	} else
400		offset = sizeof(struct ether_header);
401
402	if (hdr->flags & VIRTIO_NET_HDR_F_NEEDS_CSUM)
403		error = virtio_net_rx_csum_by_offset(m, eth_type, offset, hdr);
404	else
405		error = virtio_net_rx_csum_by_parse(m, eth_type, offset, hdr);
406
407	return (error);
408}
409
410static inline int
411virtio_net_tx_offload_ctx(struct mbuf *m, int *etype, int *proto, int *start)
412{
413	struct ether_vlan_header *evh;
414#if defined(INET) || defined(INET6)
415	int offset;
416#endif
417
418	evh = mtod(m, struct ether_vlan_header *);
419	if (evh->evl_encap_proto == htons(ETHERTYPE_VLAN)) {
420		/* BMV: We should handle nested VLAN tags too. */
421		*etype = ntohs(evh->evl_proto);
422#if defined(INET) || defined(INET6)
423		offset = sizeof(struct ether_vlan_header);
424#endif
425	} else {
426		*etype = ntohs(evh->evl_encap_proto);
427#if defined(INET) || defined(INET6)
428		offset = sizeof(struct ether_header);
429#endif
430	}
431
432	switch (*etype) {
433#if defined(INET)
434	case ETHERTYPE_IP: {
435		struct ip *ip, iphdr;
436		if (__predict_false(m->m_len < offset + sizeof(struct ip))) {
437			m_copydata(m, offset, sizeof(struct ip),
438			    (caddr_t) &iphdr);
439			ip = &iphdr;
440		} else
441			ip = (struct ip *)(m->m_data + offset);
442		*proto = ip->ip_p;
443		*start = offset + (ip->ip_hl << 2);
444		break;
445	}
446#endif
447#if defined(INET6)
448	case ETHERTYPE_IPV6:
449		*proto = -1;
450		*start = ip6_lasthdr(m, offset, IPPROTO_IPV6, proto);
451		/* Assert the network stack sent us a valid packet. */
452		KASSERT(*start > offset,
453		    ("%s: mbuf %p start %d offset %d proto %d", __func__, m,
454		    *start, offset, *proto));
455		break;
456#endif
457	default:
458		/* Here we should increment the tx_csum_bad_ethtype counter. */
459		return (EINVAL);
460	}
461
462	return (0);
463}
464
465static inline int
466virtio_net_tx_offload_tso(if_t ifp, struct mbuf *m, int eth_type,
467		     int offset, bool allow_ecn, struct virtio_net_hdr *hdr)
468{
469	static struct timeval lastecn;
470	static int curecn;
471	struct tcphdr *tcp, tcphdr;
472
473	if (__predict_false(m->m_len < offset + sizeof(struct tcphdr))) {
474		m_copydata(m, offset, sizeof(struct tcphdr), (caddr_t) &tcphdr);
475		tcp = &tcphdr;
476	} else
477		tcp = (struct tcphdr *)(m->m_data + offset);
478
479	hdr->hdr_len = offset + (tcp->th_off << 2);
480	hdr->gso_size = m->m_pkthdr.tso_segsz;
481	hdr->gso_type = eth_type == ETHERTYPE_IP ? VIRTIO_NET_HDR_GSO_TCPV4 :
482	    VIRTIO_NET_HDR_GSO_TCPV6;
483
484	if (tcp->th_flags & TH_CWR) {
485		/*
486		 * Drop if VIRTIO_NET_F_HOST_ECN was not negotiated. In FreeBSD,
487		 * ECN support is not on a per-interface basis, but globally via
488		 * the net.inet.tcp.ecn.enable sysctl knob. The default is off.
489		 */
490		if (!allow_ecn) {
491			if (ppsratecheck(&lastecn, &curecn, 1))
492				if_printf(ifp,
493				    "TSO with ECN not negotiated with host\n");
494			return (ENOTSUP);
495		}
496		hdr->gso_type |= VIRTIO_NET_HDR_GSO_ECN;
497	}
498
499	/* Here we should increment tx_tso counter. */
500
501	return (0);
502}
503
504static inline struct mbuf *
505virtio_net_tx_offload(if_t ifp, struct mbuf *m, bool allow_ecn,
506		 struct virtio_net_hdr *hdr)
507{
508	int flags, etype, csum_start, proto, error;
509
510	flags = m->m_pkthdr.csum_flags;
511
512	error = virtio_net_tx_offload_ctx(m, &etype, &proto, &csum_start);
513	if (error)
514		goto drop;
515
516	if ((etype == ETHERTYPE_IP && (flags & (CSUM_TCP | CSUM_UDP))) ||
517	    (etype == ETHERTYPE_IPV6 &&
518	        (flags & (CSUM_TCP_IPV6 | CSUM_UDP_IPV6)))) {
519		/*
520		 * We could compare the IP protocol vs the CSUM_ flag too,
521		 * but that really should not be necessary.
522		 */
523		hdr->flags |= VIRTIO_NET_HDR_F_NEEDS_CSUM;
524		hdr->csum_start = csum_start;
525		hdr->csum_offset = m->m_pkthdr.csum_data;
526		/* Here we should increment the tx_csum counter. */
527	}
528
529	if (flags & CSUM_TSO) {
530		if (__predict_false(proto != IPPROTO_TCP)) {
531			/* Likely failed to correctly parse the mbuf.
532			 * Here we should increment the tx_tso_not_tcp
533			 * counter. */
534			goto drop;
535		}
536
537		KASSERT(hdr->flags & VIRTIO_NET_HDR_F_NEEDS_CSUM,
538		    ("%s: mbuf %p TSO without checksum offload %#x",
539		    __func__, m, flags));
540
541		error = virtio_net_tx_offload_tso(ifp, m, etype, csum_start,
542					     allow_ecn, hdr);
543		if (error)
544			goto drop;
545	}
546
547	return (m);
548
549drop:
550	m_freem(m);
551	return (NULL);
552}
553
554#endif /* _VIRTIO_NET_H */
555