xref: /f-stack/dpdk/drivers/net/octeontx2/otx2_rx.h (revision 2d9fd380)
1 /* SPDX-License-Identifier: BSD-3-Clause
2  * Copyright(C) 2019 Marvell International Ltd.
3  */
4 
5 #ifndef __OTX2_RX_H__
6 #define __OTX2_RX_H__
7 
8 #include <rte_ether.h>
9 
10 #include "otx2_common.h"
11 #include "otx2_ethdev_sec.h"
12 #include "otx2_ipsec_anti_replay.h"
13 #include "otx2_ipsec_fp.h"
14 
15 /* Default mark value used when none is provided. */
16 #define OTX2_FLOW_ACTION_FLAG_DEFAULT	0xffff
17 
18 #define PTYPE_NON_TUNNEL_WIDTH		16
19 #define PTYPE_TUNNEL_WIDTH		12
20 #define PTYPE_NON_TUNNEL_ARRAY_SZ	BIT(PTYPE_NON_TUNNEL_WIDTH)
21 #define PTYPE_TUNNEL_ARRAY_SZ		BIT(PTYPE_TUNNEL_WIDTH)
22 #define PTYPE_ARRAY_SZ			((PTYPE_NON_TUNNEL_ARRAY_SZ +\
23 					 PTYPE_TUNNEL_ARRAY_SZ) *\
24 					 sizeof(uint16_t))
25 
26 #define NIX_RX_OFFLOAD_NONE            (0)
27 #define NIX_RX_OFFLOAD_RSS_F           BIT(0)
28 #define NIX_RX_OFFLOAD_PTYPE_F         BIT(1)
29 #define NIX_RX_OFFLOAD_CHECKSUM_F      BIT(2)
30 #define NIX_RX_OFFLOAD_VLAN_STRIP_F    BIT(3)
31 #define NIX_RX_OFFLOAD_MARK_UPDATE_F   BIT(4)
32 #define NIX_RX_OFFLOAD_TSTAMP_F        BIT(5)
33 #define NIX_RX_OFFLOAD_SECURITY_F      BIT(6)
34 
35 /* Flags to control cqe_to_mbuf conversion function.
36  * Defining it from backwards to denote its been
37  * not used as offload flags to pick function
38  */
39 #define NIX_RX_MULTI_SEG_F            BIT(15)
40 #define NIX_TIMESYNC_RX_OFFSET		8
41 
42 /* Inline IPsec offsets */
43 
44 #define INLINE_INB_RPTR_HDR		16
45 /* nix_cqe_hdr_s + nix_rx_parse_s + nix_rx_sg_s + nix_iova_s */
46 #define INLINE_CPT_RESULT_OFFSET	80
47 
48 struct otx2_timesync_info {
49 	uint64_t	rx_tstamp;
50 	rte_iova_t	tx_tstamp_iova;
51 	uint64_t	*tx_tstamp;
52 	uint64_t	rx_tstamp_dynflag;
53 	int		tstamp_dynfield_offset;
54 	uint8_t		tx_ready;
55 	uint8_t		rx_ready;
56 } __rte_cache_aligned;
57 
58 union mbuf_initializer {
59 	struct {
60 		uint16_t data_off;
61 		uint16_t refcnt;
62 		uint16_t nb_segs;
63 		uint16_t port;
64 	} fields;
65 	uint64_t value;
66 };
67 
68 static inline rte_mbuf_timestamp_t *
otx2_timestamp_dynfield(struct rte_mbuf * mbuf,struct otx2_timesync_info * info)69 otx2_timestamp_dynfield(struct rte_mbuf *mbuf,
70 		struct otx2_timesync_info *info)
71 {
72 	return RTE_MBUF_DYNFIELD(mbuf,
73 		info->tstamp_dynfield_offset, rte_mbuf_timestamp_t *);
74 }
75 
76 static __rte_always_inline void
otx2_nix_mbuf_to_tstamp(struct rte_mbuf * mbuf,struct otx2_timesync_info * tstamp,const uint16_t flag,uint64_t * tstamp_ptr)77 otx2_nix_mbuf_to_tstamp(struct rte_mbuf *mbuf,
78 			struct otx2_timesync_info *tstamp, const uint16_t flag,
79 			uint64_t *tstamp_ptr)
80 {
81 	if ((flag & NIX_RX_OFFLOAD_TSTAMP_F) &&
82 	    (mbuf->data_off == RTE_PKTMBUF_HEADROOM +
83 	     NIX_TIMESYNC_RX_OFFSET)) {
84 
85 		mbuf->pkt_len -= NIX_TIMESYNC_RX_OFFSET;
86 
87 		/* Reading the rx timestamp inserted by CGX, viz at
88 		 * starting of the packet data.
89 		 */
90 		*otx2_timestamp_dynfield(mbuf, tstamp) =
91 				rte_be_to_cpu_64(*tstamp_ptr);
92 		/* PKT_RX_IEEE1588_TMST flag needs to be set only in case
93 		 * PTP packets are received.
94 		 */
95 		if (mbuf->packet_type == RTE_PTYPE_L2_ETHER_TIMESYNC) {
96 			tstamp->rx_tstamp =
97 					*otx2_timestamp_dynfield(mbuf, tstamp);
98 			tstamp->rx_ready = 1;
99 			mbuf->ol_flags |= PKT_RX_IEEE1588_PTP |
100 				PKT_RX_IEEE1588_TMST |
101 				tstamp->rx_tstamp_dynflag;
102 		}
103 	}
104 }
105 
106 static __rte_always_inline uint64_t
nix_clear_data_off(uint64_t oldval)107 nix_clear_data_off(uint64_t oldval)
108 {
109 	union mbuf_initializer mbuf_init = { .value = oldval };
110 
111 	mbuf_init.fields.data_off = 0;
112 	return mbuf_init.value;
113 }
114 
115 static __rte_always_inline struct rte_mbuf *
nix_get_mbuf_from_cqe(void * cq,const uint64_t data_off)116 nix_get_mbuf_from_cqe(void *cq, const uint64_t data_off)
117 {
118 	rte_iova_t buff;
119 
120 	/* Skip CQE, NIX_RX_PARSE_S and SG HDR(9 DWORDs) and peek buff addr */
121 	buff = *((rte_iova_t *)((uint64_t *)cq + 9));
122 	return (struct rte_mbuf *)(buff - data_off);
123 }
124 
125 
126 static __rte_always_inline uint32_t
nix_ptype_get(const void * const lookup_mem,const uint64_t in)127 nix_ptype_get(const void * const lookup_mem, const uint64_t in)
128 {
129 	const uint16_t * const ptype = lookup_mem;
130 	const uint16_t lh_lg_lf = (in & 0xFFF0000000000000) >> 52;
131 	const uint16_t tu_l2 = ptype[(in & 0x000FFFF000000000) >> 36];
132 	const uint16_t il4_tu = ptype[PTYPE_NON_TUNNEL_ARRAY_SZ + lh_lg_lf];
133 
134 	return (il4_tu << PTYPE_NON_TUNNEL_WIDTH) | tu_l2;
135 }
136 
137 static __rte_always_inline uint32_t
nix_rx_olflags_get(const void * const lookup_mem,const uint64_t in)138 nix_rx_olflags_get(const void * const lookup_mem, const uint64_t in)
139 {
140 	const uint32_t * const ol_flags = (const uint32_t *)
141 			((const uint8_t *)lookup_mem + PTYPE_ARRAY_SZ);
142 
143 	return ol_flags[(in & 0xfff00000) >> 20];
144 }
145 
146 static inline uint64_t
nix_update_match_id(const uint16_t match_id,uint64_t ol_flags,struct rte_mbuf * mbuf)147 nix_update_match_id(const uint16_t match_id, uint64_t ol_flags,
148 		    struct rte_mbuf *mbuf)
149 {
150 	/* There is no separate bit to check match_id
151 	 * is valid or not? and no flag to identify it is an
152 	 * RTE_FLOW_ACTION_TYPE_FLAG vs RTE_FLOW_ACTION_TYPE_MARK
153 	 * action. The former case addressed through 0 being invalid
154 	 * value and inc/dec match_id pair when MARK is activated.
155 	 * The later case addressed through defining
156 	 * OTX2_FLOW_MARK_DEFAULT as value for
157 	 * RTE_FLOW_ACTION_TYPE_MARK.
158 	 * This would translate to not use
159 	 * OTX2_FLOW_ACTION_FLAG_DEFAULT - 1 and
160 	 * OTX2_FLOW_ACTION_FLAG_DEFAULT for match_id.
161 	 * i.e valid mark_id's are from
162 	 * 0 to OTX2_FLOW_ACTION_FLAG_DEFAULT - 2
163 	 */
164 	if (likely(match_id)) {
165 		ol_flags |= PKT_RX_FDIR;
166 		if (match_id != OTX2_FLOW_ACTION_FLAG_DEFAULT) {
167 			ol_flags |= PKT_RX_FDIR_ID;
168 			mbuf->hash.fdir.hi = match_id - 1;
169 		}
170 	}
171 
172 	return ol_flags;
173 }
174 
175 static __rte_always_inline void
nix_cqe_xtract_mseg(const struct nix_rx_parse_s * rx,struct rte_mbuf * mbuf,uint64_t rearm)176 nix_cqe_xtract_mseg(const struct nix_rx_parse_s *rx,
177 		    struct rte_mbuf *mbuf, uint64_t rearm)
178 {
179 	const rte_iova_t *iova_list;
180 	struct rte_mbuf *head;
181 	const rte_iova_t *eol;
182 	uint8_t nb_segs;
183 	uint64_t sg;
184 
185 	sg = *(const uint64_t *)(rx + 1);
186 	nb_segs = (sg >> 48) & 0x3;
187 	mbuf->nb_segs = nb_segs;
188 	mbuf->data_len = sg & 0xFFFF;
189 	sg = sg >> 16;
190 
191 	eol = ((const rte_iova_t *)(rx + 1) + ((rx->desc_sizem1 + 1) << 1));
192 	/* Skip SG_S and first IOVA*/
193 	iova_list = ((const rte_iova_t *)(rx + 1)) + 2;
194 	nb_segs--;
195 
196 	rearm = rearm & ~0xFFFF;
197 
198 	head = mbuf;
199 	while (nb_segs) {
200 		mbuf->next = ((struct rte_mbuf *)*iova_list) - 1;
201 		mbuf = mbuf->next;
202 
203 		__mempool_check_cookies(mbuf->pool, (void **)&mbuf, 1, 1);
204 
205 		mbuf->data_len = sg & 0xFFFF;
206 		sg = sg >> 16;
207 		*(uint64_t *)(&mbuf->rearm_data) = rearm;
208 		nb_segs--;
209 		iova_list++;
210 
211 		if (!nb_segs && (iova_list + 1 < eol)) {
212 			sg = *(const uint64_t *)(iova_list);
213 			nb_segs = (sg >> 48) & 0x3;
214 			head->nb_segs += nb_segs;
215 			iova_list = (const rte_iova_t *)(iova_list + 1);
216 		}
217 	}
218 }
219 
220 static __rte_always_inline uint16_t
nix_rx_sec_cptres_get(const void * cq)221 nix_rx_sec_cptres_get(const void *cq)
222 {
223 	volatile const struct otx2_cpt_res *res;
224 
225 	res = (volatile const struct otx2_cpt_res *)((const char *)cq +
226 			INLINE_CPT_RESULT_OFFSET);
227 
228 	return res->u16[0];
229 }
230 
231 static __rte_always_inline void *
nix_rx_sec_sa_get(const void * const lookup_mem,int spi,uint16_t port)232 nix_rx_sec_sa_get(const void * const lookup_mem, int spi, uint16_t port)
233 {
234 	const uint64_t *const *sa_tbl = (const uint64_t * const *)
235 			((const uint8_t *)lookup_mem + OTX2_NIX_SA_TBL_START);
236 
237 	return (void *)sa_tbl[port][spi];
238 }
239 
240 static __rte_always_inline uint64_t
nix_rx_sec_mbuf_update(const struct nix_cqe_hdr_s * cq,struct rte_mbuf * m,const void * const lookup_mem)241 nix_rx_sec_mbuf_update(const struct nix_cqe_hdr_s *cq, struct rte_mbuf *m,
242 		       const void * const lookup_mem)
243 {
244 	struct otx2_ipsec_fp_in_sa *sa;
245 	struct rte_ipv4_hdr *ipv4;
246 	uint16_t m_len;
247 	uint32_t spi;
248 	char *data;
249 
250 	if (unlikely(nix_rx_sec_cptres_get(cq) != OTX2_SEC_COMP_GOOD))
251 		return PKT_RX_SEC_OFFLOAD | PKT_RX_SEC_OFFLOAD_FAILED;
252 
253 	/* 20 bits of tag would have the SPI */
254 	spi = cq->tag & 0xFFFFF;
255 
256 	sa = nix_rx_sec_sa_get(lookup_mem, spi, m->port);
257 	*rte_security_dynfield(m) = sa->udata64;
258 
259 	data = rte_pktmbuf_mtod(m, char *);
260 
261 	if (sa->replay_win_sz) {
262 		if (cpt_ipsec_antireplay_check(sa, data) < 0)
263 			return PKT_RX_SEC_OFFLOAD | PKT_RX_SEC_OFFLOAD_FAILED;
264 	}
265 
266 	memcpy(data + INLINE_INB_RPTR_HDR, data, RTE_ETHER_HDR_LEN);
267 
268 	m->data_off += INLINE_INB_RPTR_HDR;
269 
270 	ipv4 = (struct rte_ipv4_hdr *)(data + INLINE_INB_RPTR_HDR +
271 				       RTE_ETHER_HDR_LEN);
272 
273 	m_len = rte_be_to_cpu_16(ipv4->total_length) + RTE_ETHER_HDR_LEN;
274 
275 	m->data_len = m_len;
276 	m->pkt_len = m_len;
277 	return PKT_RX_SEC_OFFLOAD;
278 }
279 
280 static __rte_always_inline void
otx2_nix_cqe_to_mbuf(const struct nix_cqe_hdr_s * cq,const uint32_t tag,struct rte_mbuf * mbuf,const void * lookup_mem,const uint64_t val,const uint16_t flag)281 otx2_nix_cqe_to_mbuf(const struct nix_cqe_hdr_s *cq, const uint32_t tag,
282 		     struct rte_mbuf *mbuf, const void *lookup_mem,
283 		     const uint64_t val, const uint16_t flag)
284 {
285 	const struct nix_rx_parse_s *rx =
286 		 (const struct nix_rx_parse_s *)((const uint64_t *)cq + 1);
287 	const uint64_t w1 = *(const uint64_t *)rx;
288 	const uint16_t len = rx->pkt_lenm1 + 1;
289 	uint64_t ol_flags = 0;
290 
291 	/* Mark mempool obj as "get" as it is alloc'ed by NIX */
292 	__mempool_check_cookies(mbuf->pool, (void **)&mbuf, 1, 1);
293 
294 	if (flag & NIX_RX_OFFLOAD_PTYPE_F)
295 		mbuf->packet_type = nix_ptype_get(lookup_mem, w1);
296 	else
297 		mbuf->packet_type = 0;
298 
299 	if (flag & NIX_RX_OFFLOAD_RSS_F) {
300 		mbuf->hash.rss = tag;
301 		ol_flags |= PKT_RX_RSS_HASH;
302 	}
303 
304 	if (flag & NIX_RX_OFFLOAD_CHECKSUM_F)
305 		ol_flags |= nix_rx_olflags_get(lookup_mem, w1);
306 
307 	if (flag & NIX_RX_OFFLOAD_VLAN_STRIP_F) {
308 		if (rx->vtag0_gone) {
309 			ol_flags |= PKT_RX_VLAN | PKT_RX_VLAN_STRIPPED;
310 			mbuf->vlan_tci = rx->vtag0_tci;
311 		}
312 		if (rx->vtag1_gone) {
313 			ol_flags |= PKT_RX_QINQ | PKT_RX_QINQ_STRIPPED;
314 			mbuf->vlan_tci_outer = rx->vtag1_tci;
315 		}
316 	}
317 
318 	if (flag & NIX_RX_OFFLOAD_MARK_UPDATE_F)
319 		ol_flags = nix_update_match_id(rx->match_id, ol_flags, mbuf);
320 
321 	if ((flag & NIX_RX_OFFLOAD_SECURITY_F) &&
322 	    cq->cqe_type == NIX_XQE_TYPE_RX_IPSECH) {
323 		*(uint64_t *)(&mbuf->rearm_data) = val;
324 		ol_flags |= nix_rx_sec_mbuf_update(cq, mbuf, lookup_mem);
325 		mbuf->ol_flags = ol_flags;
326 		return;
327 	}
328 
329 	mbuf->ol_flags = ol_flags;
330 	*(uint64_t *)(&mbuf->rearm_data) = val;
331 	mbuf->pkt_len = len;
332 
333 	if (flag & NIX_RX_MULTI_SEG_F)
334 		nix_cqe_xtract_mseg(rx, mbuf, val);
335 	else
336 		mbuf->data_len = len;
337 }
338 
339 #define CKSUM_F NIX_RX_OFFLOAD_CHECKSUM_F
340 #define PTYPE_F NIX_RX_OFFLOAD_PTYPE_F
341 #define RSS_F	NIX_RX_OFFLOAD_RSS_F
342 #define RX_VLAN_F  NIX_RX_OFFLOAD_VLAN_STRIP_F
343 #define MARK_F  NIX_RX_OFFLOAD_MARK_UPDATE_F
344 #define TS_F	NIX_RX_OFFLOAD_TSTAMP_F
345 #define RX_SEC_F   NIX_RX_OFFLOAD_SECURITY_F
346 
347 /* [SEC] [TSMP] [MARK] [VLAN] [CKSUM] [PTYPE] [RSS] */
348 #define NIX_RX_FASTPATH_MODES						       \
349 R(no_offload,			0, 0, 0, 0, 0, 0, 0, NIX_RX_OFFLOAD_NONE)      \
350 R(rss,				0, 0, 0, 0, 0, 0, 1, RSS_F)		       \
351 R(ptype,			0, 0, 0, 0, 0, 1, 0, PTYPE_F)		       \
352 R(ptype_rss,			0, 0, 0, 0, 0, 1, 1, PTYPE_F | RSS_F)	       \
353 R(cksum,			0, 0, 0, 0, 1, 0, 0, CKSUM_F)		       \
354 R(cksum_rss,			0, 0, 0, 0, 1, 0, 1, CKSUM_F | RSS_F)	       \
355 R(cksum_ptype,			0, 0, 0, 0, 1, 1, 0, CKSUM_F | PTYPE_F)	       \
356 R(cksum_ptype_rss,		0, 0, 0, 0, 1, 1, 1, CKSUM_F | PTYPE_F | RSS_F)\
357 R(vlan,				0, 0, 0, 1, 0, 0, 0, RX_VLAN_F)		       \
358 R(vlan_rss,			0, 0, 0, 1, 0, 0, 1, RX_VLAN_F | RSS_F)	       \
359 R(vlan_ptype,			0, 0, 0, 1, 0, 1, 0, RX_VLAN_F | PTYPE_F)      \
360 R(vlan_ptype_rss,		0, 0, 0, 1, 0, 1, 1,			       \
361 			RX_VLAN_F | PTYPE_F | RSS_F)			       \
362 R(vlan_cksum,			0, 0, 0, 1, 1, 0, 0, RX_VLAN_F | CKSUM_F)      \
363 R(vlan_cksum_rss,		0, 0, 0, 1, 1, 0, 1,			       \
364 			RX_VLAN_F | CKSUM_F | RSS_F)			       \
365 R(vlan_cksum_ptype,		0, 0, 0, 1, 1, 1, 0,			       \
366 			RX_VLAN_F | CKSUM_F | PTYPE_F)			       \
367 R(vlan_cksum_ptype_rss,		0, 0, 0, 1, 1, 1, 1,			       \
368 			RX_VLAN_F | CKSUM_F | PTYPE_F | RSS_F)		       \
369 R(mark,				0, 0, 1, 0, 0, 0, 0, MARK_F)		       \
370 R(mark_rss,			0, 0, 1, 0, 0, 0, 1, MARK_F | RSS_F)	       \
371 R(mark_ptype,			0, 0, 1, 0, 0, 1, 0, MARK_F | PTYPE_F)	       \
372 R(mark_ptype_rss,		0, 0, 1, 0, 0, 1, 1, MARK_F | PTYPE_F | RSS_F) \
373 R(mark_cksum,			0, 0, 1, 0, 1, 0, 0, MARK_F | CKSUM_F)	       \
374 R(mark_cksum_rss,		0, 0, 1, 0, 1, 0, 1, MARK_F | CKSUM_F | RSS_F) \
375 R(mark_cksum_ptype,		0, 0, 1, 0, 1, 1, 0,			       \
376 			MARK_F | CKSUM_F | PTYPE_F)			       \
377 R(mark_cksum_ptype_rss,		0, 0, 1, 0, 1, 1, 1,			       \
378 			MARK_F | CKSUM_F | PTYPE_F | RSS_F)		       \
379 R(mark_vlan,			0, 0, 1, 1, 0, 0, 0, MARK_F | RX_VLAN_F)       \
380 R(mark_vlan_rss,		0, 0, 1, 1, 0, 0, 1,			       \
381 			MARK_F | RX_VLAN_F | RSS_F)			       \
382 R(mark_vlan_ptype,		0, 0, 1, 1, 0, 1, 0,			       \
383 			MARK_F | RX_VLAN_F | PTYPE_F)			       \
384 R(mark_vlan_ptype_rss,		0, 0, 1, 1, 0, 1, 1,			       \
385 			MARK_F | RX_VLAN_F | PTYPE_F | RSS_F)		       \
386 R(mark_vlan_cksum,		0, 0, 1, 1, 1, 0, 0,			       \
387 			MARK_F | RX_VLAN_F | CKSUM_F)			       \
388 R(mark_vlan_cksum_rss,		0, 0, 1, 1, 1, 0, 1,			       \
389 			MARK_F | RX_VLAN_F | CKSUM_F | RSS_F)		       \
390 R(mark_vlan_cksum_ptype,	0, 0, 1, 1, 1, 1, 0,			       \
391 			MARK_F | RX_VLAN_F | CKSUM_F | PTYPE_F)		       \
392 R(mark_vlan_cksum_ptype_rss,	0, 0, 1, 1, 1, 1, 1,			       \
393 			MARK_F | RX_VLAN_F | CKSUM_F | PTYPE_F | RSS_F)	       \
394 R(ts,				0, 1, 0, 0, 0, 0, 0, TS_F)		       \
395 R(ts_rss,			0, 1, 0, 0, 0, 0, 1, TS_F | RSS_F)	       \
396 R(ts_ptype,			0, 1, 0, 0, 0, 1, 0, TS_F | PTYPE_F)	       \
397 R(ts_ptype_rss,			0, 1, 0, 0, 0, 1, 1, TS_F | PTYPE_F | RSS_F)   \
398 R(ts_cksum,			0, 1, 0, 0, 1, 0, 0, TS_F | CKSUM_F)	       \
399 R(ts_cksum_rss,			0, 1, 0, 0, 1, 0, 1, TS_F | CKSUM_F | RSS_F)   \
400 R(ts_cksum_ptype,		0, 1, 0, 0, 1, 1, 0, TS_F | CKSUM_F | PTYPE_F) \
401 R(ts_cksum_ptype_rss,		0, 1, 0, 0, 1, 1, 1,			       \
402 			TS_F | CKSUM_F | PTYPE_F | RSS_F)		       \
403 R(ts_vlan,			0, 1, 0, 1, 0, 0, 0, TS_F | RX_VLAN_F)	       \
404 R(ts_vlan_rss,			0, 1, 0, 1, 0, 0, 1, TS_F | RX_VLAN_F | RSS_F) \
405 R(ts_vlan_ptype,		0, 1, 0, 1, 0, 1, 0,			       \
406 			TS_F | RX_VLAN_F | PTYPE_F)			       \
407 R(ts_vlan_ptype_rss,		0, 1, 0, 1, 0, 1, 1,			       \
408 			TS_F | RX_VLAN_F | PTYPE_F | RSS_F)		       \
409 R(ts_vlan_cksum,		0, 1, 0, 1, 1, 0, 0,			       \
410 			TS_F | RX_VLAN_F | CKSUM_F)			       \
411 R(ts_vlan_cksum_rss,		0, 1, 0, 1, 1, 0, 1,			       \
412 			MARK_F | RX_VLAN_F | CKSUM_F | RSS_F)		       \
413 R(ts_vlan_cksum_ptype,		0, 1, 0, 1, 1, 1, 0,			       \
414 			TS_F | RX_VLAN_F | CKSUM_F | PTYPE_F)		       \
415 R(ts_vlan_cksum_ptype_rss,	0, 1, 0, 1, 1, 1, 1,			       \
416 			TS_F | RX_VLAN_F | CKSUM_F | PTYPE_F | RSS_F)	       \
417 R(ts_mark,			0, 1, 1, 0, 0, 0, 0, TS_F | MARK_F)	       \
418 R(ts_mark_rss,			0, 1, 1, 0, 0, 0, 1, TS_F | MARK_F | RSS_F)    \
419 R(ts_mark_ptype,		0, 1, 1, 0, 0, 1, 0, TS_F | MARK_F | PTYPE_F)  \
420 R(ts_mark_ptype_rss,		0, 1, 1, 0, 0, 1, 1,			       \
421 			TS_F | MARK_F | PTYPE_F | RSS_F)		       \
422 R(ts_mark_cksum,		0, 1, 1, 0, 1, 0, 0, TS_F | MARK_F | CKSUM_F)  \
423 R(ts_mark_cksum_rss,		0, 1, 1, 0, 1, 0, 1,			       \
424 			TS_F | MARK_F | CKSUM_F | RSS_F)		       \
425 R(ts_mark_cksum_ptype,		0, 1, 1, 0, 1, 1, 0,			       \
426 			TS_F | MARK_F | CKSUM_F | PTYPE_F)		       \
427 R(ts_mark_cksum_ptype_rss,	0, 1, 1, 0, 1, 1, 1,			       \
428 			TS_F | MARK_F | CKSUM_F | PTYPE_F | RSS_F)	       \
429 R(ts_mark_vlan,			0, 1, 1, 1, 0, 0, 0, TS_F | MARK_F | RX_VLAN_F)\
430 R(ts_mark_vlan_rss,		0, 1, 1, 1, 0, 0, 1,			       \
431 			TS_F | MARK_F | RX_VLAN_F | RSS_F)		       \
432 R(ts_mark_vlan_ptype,		0, 1, 1, 1, 0, 1, 0,			       \
433 			TS_F | MARK_F | RX_VLAN_F | PTYPE_F)		       \
434 R(ts_mark_vlan_ptype_rss,	0, 1, 1, 1, 0, 1, 1,			       \
435 			TS_F | MARK_F | RX_VLAN_F | PTYPE_F | RSS_F)	       \
436 R(ts_mark_vlan_cksum_ptype,	0, 1, 1, 1, 1, 1, 0,			       \
437 			TS_F | MARK_F | RX_VLAN_F | CKSUM_F | PTYPE_F)	       \
438 R(ts_mark_vlan_cksum_ptype_rss,	0, 1, 1, 1, 1, 1, 1,			       \
439 			TS_F | MARK_F | RX_VLAN_F | CKSUM_F | PTYPE_F | RSS_F) \
440 R(sec,				1, 0, 0, 0, 0, 0, 0, RX_SEC_F)		       \
441 R(sec_rss,			1, 0, 0, 0, 0, 0, 1, RX_SEC_F | RSS_F)	       \
442 R(sec_ptype,			1, 0, 0, 0, 0, 1, 0, RX_SEC_F | PTYPE_F)       \
443 R(sec_ptype_rss,		1, 0, 0, 0, 0, 1, 1,			       \
444 			RX_SEC_F | PTYPE_F | RSS_F)			       \
445 R(sec_cksum,			1, 0, 0, 0, 1, 0, 0, RX_SEC_F | CKSUM_F)       \
446 R(sec_cksum_rss,		1, 0, 0, 0, 1, 0, 1,			       \
447 			RX_SEC_F | CKSUM_F | RSS_F)			       \
448 R(sec_cksum_ptype,		1, 0, 0, 0, 1, 1, 0,			       \
449 			RX_SEC_F | CKSUM_F | PTYPE_F)			       \
450 R(sec_cksum_ptype_rss,		1, 0, 0, 0, 1, 1, 1,			       \
451 			RX_SEC_F | CKSUM_F | PTYPE_F | RSS_F)		       \
452 R(sec_vlan,			1, 0, 0, 1, 0, 0, 0, RX_SEC_F | RX_VLAN_F)     \
453 R(sec_vlan_rss,			1, 0, 0, 1, 0, 0, 1,			       \
454 			RX_SEC_F | RX_VLAN_F | RSS_F)			       \
455 R(sec_vlan_ptype,		1, 0, 0, 1, 0, 1, 0,			       \
456 			RX_SEC_F | RX_VLAN_F | PTYPE_F)			       \
457 R(sec_vlan_ptype_rss,		1, 0, 0, 1, 0, 1, 1,			       \
458 			RX_SEC_F | RX_VLAN_F | PTYPE_F | RSS_F)		       \
459 R(sec_vlan_cksum,		1, 0, 0, 1, 1, 0, 0,			       \
460 			RX_SEC_F | RX_VLAN_F | CKSUM_F)			       \
461 R(sec_vlan_cksum_rss,		1, 0, 0, 1, 1, 0, 1,			       \
462 			RX_SEC_F | RX_VLAN_F | CKSUM_F | RSS_F)		       \
463 R(sec_vlan_cksum_ptype,		1, 0, 0, 1, 1, 1, 0,			       \
464 			RX_SEC_F | RX_VLAN_F | CKSUM_F | PTYPE_F)	       \
465 R(sec_vlan_cksum_ptype_rss,	1, 0, 0, 1, 1, 1, 1,			       \
466 			RX_SEC_F | RX_VLAN_F | CKSUM_F | PTYPE_F | RSS_F)      \
467 R(sec_mark,			1, 0, 1, 0, 0, 0, 0, RX_SEC_F | MARK_F)	       \
468 R(sec_mark_rss,			1, 0, 1, 0, 0, 0, 1, RX_SEC_F | MARK_F | RSS_F)\
469 R(sec_mark_ptype,		1, 0, 1, 0, 0, 1, 0,			       \
470 			RX_SEC_F | MARK_F | PTYPE_F)			       \
471 R(sec_mark_ptype_rss,		1, 0, 1, 0, 0, 1, 1,			       \
472 			RX_SEC_F | MARK_F | PTYPE_F | RSS_F)		       \
473 R(sec_mark_cksum,		1, 0, 1, 0, 1, 0, 0,			       \
474 			RX_SEC_F | MARK_F | CKSUM_F)			       \
475 R(sec_mark_cksum_rss,		1, 0, 1, 0, 1, 0, 1,			       \
476 			RX_SEC_F | MARK_F | CKSUM_F | RSS_F)		       \
477 R(sec_mark_cksum_ptype,		1, 0, 1, 0, 1, 1, 0,			       \
478 			RX_SEC_F | MARK_F | CKSUM_F | PTYPE_F)		       \
479 R(sec_mark_cksum_ptype_rss,	1, 0, 1, 0, 1, 1, 1,			       \
480 			RX_SEC_F | MARK_F | CKSUM_F | PTYPE_F | RSS_F)	       \
481 R(sec_mark_vlan,		1, 0, 1, 1, 0, 0, 0, RX_SEC_F | RX_VLAN_F)     \
482 R(sec_mark_vlan_rss,		1, 0, 1, 1, 0, 0, 1,			       \
483 			RX_SEC_F | MARK_F | RX_VLAN_F | RSS_F)		       \
484 R(sec_mark_vlan_ptype,		1, 0, 1, 1, 0, 1, 0,			       \
485 			RX_SEC_F | MARK_F | RX_VLAN_F | PTYPE_F)	       \
486 R(sec_mark_vlan_ptype_rss,	1, 0, 1, 1, 0, 1, 1,			       \
487 			RX_SEC_F | MARK_F | RX_VLAN_F | PTYPE_F | RSS_F)       \
488 R(sec_mark_vlan_cksum,		1, 0, 1, 1, 1, 0, 0,			       \
489 			RX_SEC_F | MARK_F | RX_VLAN_F | CKSUM_F)	       \
490 R(sec_mark_vlan_cksum_rss,	1, 0, 1, 1, 1, 0, 1,			       \
491 			RX_SEC_F | MARK_F | RX_VLAN_F | CKSUM_F | RSS_F)       \
492 R(sec_mark_vlan_cksum_ptype,	1, 0, 1, 1, 1, 1, 0,			       \
493 			RX_SEC_F | MARK_F | RX_VLAN_F | CKSUM_F | PTYPE_F)     \
494 R(sec_mark_vlan_cksum_ptype_rss,					       \
495 				1, 0, 1, 1, 1, 1, 1,			       \
496 			RX_SEC_F | MARK_F | RX_VLAN_F | CKSUM_F | PTYPE_F |    \
497 			RSS_F)						       \
498 R(sec_ts,			1, 1, 0, 0, 0, 0, 0, RX_SEC_F | TS_F)	       \
499 R(sec_ts_rss,			1, 1, 0, 0, 0, 0, 1, RX_SEC_F | TS_F | RSS_F)  \
500 R(sec_ts_ptype,			1, 1, 0, 0, 0, 1, 0, RX_SEC_F | TS_F | PTYPE_F)\
501 R(sec_ts_ptype_rss,		1, 1, 0, 0, 0, 1, 1,			       \
502 			RX_SEC_F | TS_F | PTYPE_F | RSS_F)		       \
503 R(sec_ts_cksum,			1, 1, 0, 0, 1, 0, 0, RX_SEC_F | TS_F | CKSUM_F)\
504 R(sec_ts_cksum_rss,		1, 1, 0, 0, 1, 0, 1,			       \
505 			RX_SEC_F | TS_F | CKSUM_F | RSS_F)		       \
506 R(sec_ts_cksum_ptype,		1, 1, 0, 0, 1, 1, 0,			       \
507 			RX_SEC_F | CKSUM_F | PTYPE_F)			       \
508 R(sec_ts_cksum_ptype_rss,	1, 1, 0, 0, 1, 1, 1,			       \
509 			RX_SEC_F | TS_F | CKSUM_F | PTYPE_F | RSS_F)	       \
510 R(sec_ts_vlan,			1, 1, 0, 1, 0, 0, 0,			       \
511 			RX_SEC_F | TS_F | RX_VLAN_F)			       \
512 R(sec_ts_vlan_rss,		1, 1, 0, 1, 0, 0, 1,			       \
513 			RX_SEC_F | TS_F | RX_VLAN_F | RSS_F)		       \
514 R(sec_ts_vlan_ptype,		1, 1, 0, 1, 0, 1, 0,			       \
515 			RX_SEC_F | TS_F | RX_VLAN_F | PTYPE_F)		       \
516 R(sec_ts_vlan_ptype_rss,	1, 1, 0, 1, 0, 1, 1,			       \
517 			RX_SEC_F | TS_F | RX_VLAN_F | PTYPE_F | RSS_F)	       \
518 R(sec_ts_vlan_cksum,		1, 1, 0, 1, 1, 0, 0,			       \
519 			RX_SEC_F | TS_F | RX_VLAN_F | CKSUM_F)		       \
520 R(sec_ts_vlan_cksum_rss,	1, 1, 0, 1, 1, 0, 1,			       \
521 			RX_SEC_F | TS_F | RX_VLAN_F | CKSUM_F | RSS_F)	       \
522 R(sec_ts_vlan_cksum_ptype,	1, 1, 0, 1, 1, 1, 0,			       \
523 			RX_SEC_F | TS_F | RX_VLAN_F | CKSUM_F | PTYPE_F)       \
524 R(sec_ts_vlan_cksum_ptype_rss,	1, 1, 0, 1, 1, 1, 1,			       \
525 			RX_SEC_F | TS_F | RX_VLAN_F | CKSUM_F | PTYPE_F |      \
526 			RSS_F)						       \
527 R(sec_ts_mark,			1, 1, 1, 0, 0, 0, 0, RX_SEC_F | TS_F | MARK_F) \
528 R(sec_ts_mark_rss,		1, 1, 1, 0, 0, 0, 1,			       \
529 			RX_SEC_F | TS_F | MARK_F | RSS_F)		       \
530 R(sec_ts_mark_ptype,		1, 1, 1, 0, 0, 1, 0,			       \
531 			RX_SEC_F | TS_F | MARK_F | PTYPE_F)		       \
532 R(sec_ts_mark_ptype_rss,	1, 1, 1, 0, 0, 1, 1,			       \
533 			RX_SEC_F | TS_F | MARK_F | PTYPE_F | RSS_F)	       \
534 R(sec_ts_mark_cksum,		1, 1, 1, 0, 1, 0, 0,			       \
535 			RX_SEC_F | TS_F | MARK_F | CKSUM_F)		       \
536 R(sec_ts_mark_cksum_rss,	1, 1, 1, 0, 1, 0, 1,			       \
537 			RX_SEC_F | TS_F | MARK_F | CKSUM_F | RSS_F)	       \
538 R(sec_ts_mark_cksum_ptype,	1, 1, 1, 0, 1, 1, 0,			       \
539 			RX_SEC_F | TS_F | MARK_F | CKSUM_F | PTYPE_F)	       \
540 R(sec_ts_mark_cksum_ptype_rss,	1, 1, 1, 0, 1, 1, 1,			       \
541 			RX_SEC_F | TS_F | MARK_F | CKSUM_F | PTYPE_F | RSS_F)  \
542 R(sec_ts_mark_vlan,		1, 1, 1, 1, 0, 0, 0,			       \
543 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F)		       \
544 R(sec_ts_mark_vlan_rss,		1, 1, 1, 1, 0, 0, 1,			       \
545 			RX_SEC_F | RX_VLAN_F | RSS_F)			       \
546 R(sec_ts_mark_vlan_ptype,	1, 1, 1, 1, 0, 1, 0,			       \
547 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F | PTYPE_F)	       \
548 R(sec_ts_mark_vlan_ptype_rss,	1, 1, 1, 1, 0, 1, 1,			       \
549 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F | PTYPE_F | RSS_F)\
550 R(sec_ts_mark_vlan_cksum,	1, 1, 1, 1, 1, 0, 0,			       \
551 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F | CKSUM_F)	       \
552 R(sec_ts_mark_vlan_cksum_rss,	1, 1, 1, 1, 1, 0, 1,			       \
553 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F | CKSUM_F | RSS_F)\
554 R(sec_ts_mark_vlan_cksum_ptype,	1, 1, 1, 1, 1, 1, 0,			       \
555 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F | CKSUM_F |       \
556 			PTYPE_F)					       \
557 R(sec_ts_mark_vlan_cksum_ptype_rss,					       \
558 				1, 1, 1, 1, 1, 1, 1,			       \
559 			RX_SEC_F | TS_F | MARK_F | RX_VLAN_F | CKSUM_F |       \
560 			PTYPE_F | RSS_F)
561 #endif /* __OTX2_RX_H__ */
562