1 /* SPDX-License-Identifier: BSD-3-Clause
3 * Copyright(c) 2019 Xilinx, Inc. All rights reserved.
13 static __checkReturn efx_rc_t
14 efx_mae_get_capabilities(
18 EFX_MCDI_DECLARE_BUF(payload,
19 MC_CMD_MAE_GET_CAPS_IN_LEN,
20 MC_CMD_MAE_GET_CAPS_OUT_LEN);
21 struct efx_mae_s *maep = enp->en_maep;
24 req.emr_cmd = MC_CMD_MAE_GET_CAPS;
25 req.emr_in_buf = payload;
26 req.emr_in_length = MC_CMD_MAE_GET_CAPS_IN_LEN;
27 req.emr_out_buf = payload;
28 req.emr_out_length = MC_CMD_MAE_GET_CAPS_OUT_LEN;
30 efx_mcdi_execute(enp, &req);
32 if (req.emr_rc != 0) {
37 if (req.emr_out_length_used < MC_CMD_MAE_GET_CAPS_OUT_LEN) {
42 maep->em_max_n_outer_prios =
43 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_OUTER_PRIOS);
45 maep->em_max_n_action_prios =
46 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_ACTION_PRIOS);
48 maep->em_encap_types_supported = 0;
50 if (MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_ENCAP_TYPE_VXLAN) == 1) {
51 maep->em_encap_types_supported |=
52 (1U << EFX_TUNNEL_PROTOCOL_VXLAN);
55 if (MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_ENCAP_TYPE_GENEVE) == 1) {
56 maep->em_encap_types_supported |=
57 (1U << EFX_TUNNEL_PROTOCOL_GENEVE);
60 if (MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_ENCAP_TYPE_NVGRE) == 1) {
61 maep->em_encap_types_supported |=
62 (1U << EFX_TUNNEL_PROTOCOL_NVGRE);
65 maep->em_max_nfields =
66 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_MATCH_FIELD_COUNT);
73 EFSYS_PROBE1(fail1, efx_rc_t, rc);
77 static __checkReturn efx_rc_t
78 efx_mae_get_outer_rule_caps(
80 __in unsigned int field_ncaps,
81 __out_ecount(field_ncaps) efx_mae_field_cap_t *field_caps)
84 EFX_MCDI_DECLARE_BUF(payload,
85 MC_CMD_MAE_GET_OR_CAPS_IN_LEN,
86 MC_CMD_MAE_GET_OR_CAPS_OUT_LENMAX_MCDI2);
87 unsigned int mcdi_field_ncaps;
91 if (MC_CMD_MAE_GET_OR_CAPS_OUT_LEN(field_ncaps) >
92 MC_CMD_MAE_GET_OR_CAPS_OUT_LENMAX_MCDI2) {
97 req.emr_cmd = MC_CMD_MAE_GET_OR_CAPS;
98 req.emr_in_buf = payload;
99 req.emr_in_length = MC_CMD_MAE_GET_OR_CAPS_IN_LEN;
100 req.emr_out_buf = payload;
101 req.emr_out_length = MC_CMD_MAE_GET_OR_CAPS_OUT_LEN(field_ncaps);
103 efx_mcdi_execute(enp, &req);
105 if (req.emr_rc != 0) {
110 mcdi_field_ncaps = MCDI_OUT_DWORD(req, MAE_GET_OR_CAPS_OUT_COUNT);
112 if (req.emr_out_length_used <
113 MC_CMD_MAE_GET_OR_CAPS_OUT_LEN(mcdi_field_ncaps)) {
118 if (mcdi_field_ncaps > field_ncaps) {
123 for (i = 0; i < mcdi_field_ncaps; ++i) {
127 field_caps[i].emfc_support = MCDI_OUT_INDEXED_DWORD_FIELD(req,
128 MAE_GET_OR_CAPS_OUT_FIELD_FLAGS, i,
129 MAE_FIELD_FLAGS_SUPPORT_STATUS);
131 match_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
132 MAE_GET_OR_CAPS_OUT_FIELD_FLAGS, i,
133 MAE_FIELD_FLAGS_MATCH_AFFECTS_CLASS);
135 field_caps[i].emfc_match_affects_class =
136 (match_flag != 0) ? B_TRUE : B_FALSE;
138 mask_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
139 MAE_GET_OR_CAPS_OUT_FIELD_FLAGS, i,
140 MAE_FIELD_FLAGS_MASK_AFFECTS_CLASS);
142 field_caps[i].emfc_mask_affects_class =
143 (mask_flag != 0) ? B_TRUE : B_FALSE;
155 EFSYS_PROBE1(fail1, efx_rc_t, rc);
159 static __checkReturn efx_rc_t
160 efx_mae_get_action_rule_caps(
162 __in unsigned int field_ncaps,
163 __out_ecount(field_ncaps) efx_mae_field_cap_t *field_caps)
166 EFX_MCDI_DECLARE_BUF(payload,
167 MC_CMD_MAE_GET_AR_CAPS_IN_LEN,
168 MC_CMD_MAE_GET_AR_CAPS_OUT_LENMAX_MCDI2);
169 unsigned int mcdi_field_ncaps;
173 if (MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(field_ncaps) >
174 MC_CMD_MAE_GET_AR_CAPS_OUT_LENMAX_MCDI2) {
179 req.emr_cmd = MC_CMD_MAE_GET_AR_CAPS;
180 req.emr_in_buf = payload;
181 req.emr_in_length = MC_CMD_MAE_GET_AR_CAPS_IN_LEN;
182 req.emr_out_buf = payload;
183 req.emr_out_length = MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(field_ncaps);
185 efx_mcdi_execute(enp, &req);
187 if (req.emr_rc != 0) {
192 mcdi_field_ncaps = MCDI_OUT_DWORD(req, MAE_GET_OR_CAPS_OUT_COUNT);
194 if (req.emr_out_length_used <
195 MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(mcdi_field_ncaps)) {
200 if (mcdi_field_ncaps > field_ncaps) {
205 for (i = 0; i < mcdi_field_ncaps; ++i) {
209 field_caps[i].emfc_support = MCDI_OUT_INDEXED_DWORD_FIELD(req,
210 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
211 MAE_FIELD_FLAGS_SUPPORT_STATUS);
213 match_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
214 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
215 MAE_FIELD_FLAGS_MATCH_AFFECTS_CLASS);
217 field_caps[i].emfc_match_affects_class =
218 (match_flag != 0) ? B_TRUE : B_FALSE;
220 mask_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
221 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
222 MAE_FIELD_FLAGS_MASK_AFFECTS_CLASS);
224 field_caps[i].emfc_mask_affects_class =
225 (mask_flag != 0) ? B_TRUE : B_FALSE;
237 EFSYS_PROBE1(fail1, efx_rc_t, rc);
241 __checkReturn efx_rc_t
245 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
246 efx_mae_field_cap_t *or_fcaps;
247 size_t or_fcaps_size;
248 efx_mae_field_cap_t *ar_fcaps;
249 size_t ar_fcaps_size;
253 if (encp->enc_mae_supported == B_FALSE) {
258 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*maep), maep);
266 rc = efx_mae_get_capabilities(enp);
270 or_fcaps_size = maep->em_max_nfields * sizeof (*or_fcaps);
271 EFSYS_KMEM_ALLOC(enp->en_esip, or_fcaps_size, or_fcaps);
272 if (or_fcaps == NULL) {
277 maep->em_outer_rule_field_caps_size = or_fcaps_size;
278 maep->em_outer_rule_field_caps = or_fcaps;
280 rc = efx_mae_get_outer_rule_caps(enp, maep->em_max_nfields, or_fcaps);
284 ar_fcaps_size = maep->em_max_nfields * sizeof (*ar_fcaps);
285 EFSYS_KMEM_ALLOC(enp->en_esip, ar_fcaps_size, ar_fcaps);
286 if (ar_fcaps == NULL) {
291 maep->em_action_rule_field_caps_size = ar_fcaps_size;
292 maep->em_action_rule_field_caps = ar_fcaps;
294 rc = efx_mae_get_action_rule_caps(enp, maep->em_max_nfields, ar_fcaps);
302 EFSYS_KMEM_FREE(enp->en_esip, ar_fcaps_size, ar_fcaps);
307 EFSYS_KMEM_FREE(enp->en_esip, or_fcaps_size, or_fcaps);
312 EFSYS_KMEM_FREE(enp->en_esip, sizeof (struct efx_mae_s), enp->en_maep);
317 EFSYS_PROBE1(fail1, efx_rc_t, rc);
325 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
326 efx_mae_t *maep = enp->en_maep;
328 if (encp->enc_mae_supported == B_FALSE)
331 EFSYS_KMEM_FREE(enp->en_esip, maep->em_action_rule_field_caps_size,
332 maep->em_action_rule_field_caps);
333 EFSYS_KMEM_FREE(enp->en_esip, maep->em_outer_rule_field_caps_size,
334 maep->em_outer_rule_field_caps);
335 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*maep), maep);
339 __checkReturn efx_rc_t
342 __out efx_mae_limits_t *emlp)
344 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
345 struct efx_mae_s *maep = enp->en_maep;
348 if (encp->enc_mae_supported == B_FALSE) {
353 emlp->eml_max_n_outer_prios = maep->em_max_n_outer_prios;
354 emlp->eml_max_n_action_prios = maep->em_max_n_action_prios;
355 emlp->eml_encap_types_supported = maep->em_encap_types_supported;
360 EFSYS_PROBE1(fail1, efx_rc_t, rc);
364 __checkReturn efx_rc_t
365 efx_mae_match_spec_init(
367 __in efx_mae_rule_type_t type,
369 __out efx_mae_match_spec_t **specp)
371 efx_mae_match_spec_t *spec;
375 case EFX_MAE_RULE_OUTER:
377 case EFX_MAE_RULE_ACTION:
384 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
390 spec->emms_type = type;
391 spec->emms_prio = prio;
400 EFSYS_PROBE1(fail1, efx_rc_t, rc);
405 efx_mae_match_spec_fini(
407 __in efx_mae_match_spec_t *spec)
409 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
412 /* Named identifiers which are valid indices to efx_mae_field_cap_t */
413 typedef enum efx_mae_field_cap_id_e {
414 EFX_MAE_FIELD_ID_INGRESS_MPORT_SELECTOR = MAE_FIELD_INGRESS_PORT,
415 EFX_MAE_FIELD_ID_ETHER_TYPE_BE = MAE_FIELD_ETHER_TYPE,
416 EFX_MAE_FIELD_ID_ETH_SADDR_BE = MAE_FIELD_ETH_SADDR,
417 EFX_MAE_FIELD_ID_ETH_DADDR_BE = MAE_FIELD_ETH_DADDR,
418 EFX_MAE_FIELD_ID_VLAN0_TCI_BE = MAE_FIELD_VLAN0_TCI,
419 EFX_MAE_FIELD_ID_VLAN0_PROTO_BE = MAE_FIELD_VLAN0_PROTO,
420 EFX_MAE_FIELD_ID_VLAN1_TCI_BE = MAE_FIELD_VLAN1_TCI,
421 EFX_MAE_FIELD_ID_VLAN1_PROTO_BE = MAE_FIELD_VLAN1_PROTO,
422 EFX_MAE_FIELD_ID_SRC_IP4_BE = MAE_FIELD_SRC_IP4,
423 EFX_MAE_FIELD_ID_DST_IP4_BE = MAE_FIELD_DST_IP4,
424 EFX_MAE_FIELD_ID_IP_PROTO = MAE_FIELD_IP_PROTO,
425 EFX_MAE_FIELD_ID_IP_TOS = MAE_FIELD_IP_TOS,
426 EFX_MAE_FIELD_ID_IP_TTL = MAE_FIELD_IP_TTL,
427 EFX_MAE_FIELD_ID_SRC_IP6_BE = MAE_FIELD_SRC_IP6,
428 EFX_MAE_FIELD_ID_DST_IP6_BE = MAE_FIELD_DST_IP6,
429 EFX_MAE_FIELD_ID_L4_SPORT_BE = MAE_FIELD_L4_SPORT,
430 EFX_MAE_FIELD_ID_L4_DPORT_BE = MAE_FIELD_L4_DPORT,
431 EFX_MAE_FIELD_ID_TCP_FLAGS_BE = MAE_FIELD_TCP_FLAGS,
432 EFX_MAE_FIELD_ID_ENC_ETHER_TYPE_BE = MAE_FIELD_ENC_ETHER_TYPE,
433 EFX_MAE_FIELD_ID_ENC_ETH_SADDR_BE = MAE_FIELD_ENC_ETH_SADDR,
434 EFX_MAE_FIELD_ID_ENC_ETH_DADDR_BE = MAE_FIELD_ENC_ETH_DADDR,
435 EFX_MAE_FIELD_ID_ENC_VLAN0_TCI_BE = MAE_FIELD_ENC_VLAN0_TCI,
436 EFX_MAE_FIELD_ID_ENC_VLAN0_PROTO_BE = MAE_FIELD_ENC_VLAN0_PROTO,
437 EFX_MAE_FIELD_ID_ENC_VLAN1_TCI_BE = MAE_FIELD_ENC_VLAN1_TCI,
438 EFX_MAE_FIELD_ID_ENC_VLAN1_PROTO_BE = MAE_FIELD_ENC_VLAN1_PROTO,
439 EFX_MAE_FIELD_ID_ENC_SRC_IP4_BE = MAE_FIELD_ENC_SRC_IP4,
440 EFX_MAE_FIELD_ID_ENC_DST_IP4_BE = MAE_FIELD_ENC_DST_IP4,
441 EFX_MAE_FIELD_ID_ENC_IP_PROTO = MAE_FIELD_ENC_IP_PROTO,
442 EFX_MAE_FIELD_ID_ENC_IP_TOS = MAE_FIELD_ENC_IP_TOS,
443 EFX_MAE_FIELD_ID_ENC_IP_TTL = MAE_FIELD_ENC_IP_TTL,
444 EFX_MAE_FIELD_ID_ENC_SRC_IP6_BE = MAE_FIELD_ENC_SRC_IP6,
445 EFX_MAE_FIELD_ID_ENC_DST_IP6_BE = MAE_FIELD_ENC_DST_IP6,
446 EFX_MAE_FIELD_ID_ENC_L4_SPORT_BE = MAE_FIELD_ENC_L4_SPORT,
447 EFX_MAE_FIELD_ID_ENC_L4_DPORT_BE = MAE_FIELD_ENC_L4_DPORT,
448 EFX_MAE_FIELD_ID_ENC_VNET_ID_BE = MAE_FIELD_ENC_VNET_ID,
449 EFX_MAE_FIELD_ID_OUTER_RULE_ID = MAE_FIELD_OUTER_RULE_ID,
451 EFX_MAE_FIELD_CAP_NIDS
452 } efx_mae_field_cap_id_t;
454 typedef enum efx_mae_field_endianness_e {
455 EFX_MAE_FIELD_LE = 0,
458 EFX_MAE_FIELD_ENDIANNESS_NTYPES
459 } efx_mae_field_endianness_t;
462 * The following structure is a means to describe an MAE field.
463 * The information in it is meant to be used internally by
464 * APIs for addressing a given field in a mask-value pairs
465 * structure and for validation purposes.
467 * A field may have an alternative one. This structure
468 * has additional members to reference the alternative
469 * field's mask. See efx_mae_match_spec_is_valid().
471 typedef struct efx_mae_mv_desc_s {
472 efx_mae_field_cap_id_t emmd_field_cap_id;
474 size_t emmd_value_size;
475 size_t emmd_value_offset;
476 size_t emmd_mask_size;
477 size_t emmd_mask_offset;
480 * Having the alternative field's mask size set to 0
481 * means that there's no alternative field specified.
483 size_t emmd_alt_mask_size;
484 size_t emmd_alt_mask_offset;
486 /* Primary field and the alternative one are of the same endianness. */
487 efx_mae_field_endianness_t emmd_endianness;
490 /* Indices to this array are provided by efx_mae_field_id_t */
491 static const efx_mae_mv_desc_t __efx_mae_action_rule_mv_desc_set[] = {
492 #define EFX_MAE_MV_DESC(_name, _endianness) \
493 [EFX_MAE_FIELD_##_name] = \
495 EFX_MAE_FIELD_ID_##_name, \
496 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_LEN, \
497 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_OFST, \
498 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_MASK_LEN, \
499 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_MASK_OFST, \
500 0, 0 /* no alternative field */, \
504 EFX_MAE_MV_DESC(INGRESS_MPORT_SELECTOR, EFX_MAE_FIELD_LE),
505 EFX_MAE_MV_DESC(ETHER_TYPE_BE, EFX_MAE_FIELD_BE),
506 EFX_MAE_MV_DESC(ETH_SADDR_BE, EFX_MAE_FIELD_BE),
507 EFX_MAE_MV_DESC(ETH_DADDR_BE, EFX_MAE_FIELD_BE),
508 EFX_MAE_MV_DESC(VLAN0_TCI_BE, EFX_MAE_FIELD_BE),
509 EFX_MAE_MV_DESC(VLAN0_PROTO_BE, EFX_MAE_FIELD_BE),
510 EFX_MAE_MV_DESC(VLAN1_TCI_BE, EFX_MAE_FIELD_BE),
511 EFX_MAE_MV_DESC(VLAN1_PROTO_BE, EFX_MAE_FIELD_BE),
512 EFX_MAE_MV_DESC(SRC_IP4_BE, EFX_MAE_FIELD_BE),
513 EFX_MAE_MV_DESC(DST_IP4_BE, EFX_MAE_FIELD_BE),
514 EFX_MAE_MV_DESC(IP_PROTO, EFX_MAE_FIELD_BE),
515 EFX_MAE_MV_DESC(IP_TOS, EFX_MAE_FIELD_BE),
516 EFX_MAE_MV_DESC(IP_TTL, EFX_MAE_FIELD_BE),
517 EFX_MAE_MV_DESC(SRC_IP6_BE, EFX_MAE_FIELD_BE),
518 EFX_MAE_MV_DESC(DST_IP6_BE, EFX_MAE_FIELD_BE),
519 EFX_MAE_MV_DESC(L4_SPORT_BE, EFX_MAE_FIELD_BE),
520 EFX_MAE_MV_DESC(L4_DPORT_BE, EFX_MAE_FIELD_BE),
521 EFX_MAE_MV_DESC(TCP_FLAGS_BE, EFX_MAE_FIELD_BE),
522 EFX_MAE_MV_DESC(ENC_VNET_ID_BE, EFX_MAE_FIELD_BE),
523 EFX_MAE_MV_DESC(OUTER_RULE_ID, EFX_MAE_FIELD_LE),
525 #undef EFX_MAE_MV_DESC
528 /* Indices to this array are provided by efx_mae_field_id_t */
529 static const efx_mae_mv_desc_t __efx_mae_outer_rule_mv_desc_set[] = {
530 #define EFX_MAE_MV_DESC(_name, _endianness) \
531 [EFX_MAE_FIELD_##_name] = \
533 EFX_MAE_FIELD_ID_##_name, \
534 MAE_ENC_FIELD_PAIRS_##_name##_LEN, \
535 MAE_ENC_FIELD_PAIRS_##_name##_OFST, \
536 MAE_ENC_FIELD_PAIRS_##_name##_MASK_LEN, \
537 MAE_ENC_FIELD_PAIRS_##_name##_MASK_OFST, \
538 0, 0 /* no alternative field */, \
542 /* Same as EFX_MAE_MV_DESC(), but also indicates an alternative field. */
543 #define EFX_MAE_MV_DESC_ALT(_name, _alt_name, _endianness) \
544 [EFX_MAE_FIELD_##_name] = \
546 EFX_MAE_FIELD_ID_##_name, \
547 MAE_ENC_FIELD_PAIRS_##_name##_LEN, \
548 MAE_ENC_FIELD_PAIRS_##_name##_OFST, \
549 MAE_ENC_FIELD_PAIRS_##_name##_MASK_LEN, \
550 MAE_ENC_FIELD_PAIRS_##_name##_MASK_OFST, \
551 MAE_ENC_FIELD_PAIRS_##_alt_name##_MASK_LEN, \
552 MAE_ENC_FIELD_PAIRS_##_alt_name##_MASK_OFST, \
556 EFX_MAE_MV_DESC(INGRESS_MPORT_SELECTOR, EFX_MAE_FIELD_LE),
557 EFX_MAE_MV_DESC(ENC_ETHER_TYPE_BE, EFX_MAE_FIELD_BE),
558 EFX_MAE_MV_DESC(ENC_ETH_SADDR_BE, EFX_MAE_FIELD_BE),
559 EFX_MAE_MV_DESC(ENC_ETH_DADDR_BE, EFX_MAE_FIELD_BE),
560 EFX_MAE_MV_DESC(ENC_VLAN0_TCI_BE, EFX_MAE_FIELD_BE),
561 EFX_MAE_MV_DESC(ENC_VLAN0_PROTO_BE, EFX_MAE_FIELD_BE),
562 EFX_MAE_MV_DESC(ENC_VLAN1_TCI_BE, EFX_MAE_FIELD_BE),
563 EFX_MAE_MV_DESC(ENC_VLAN1_PROTO_BE, EFX_MAE_FIELD_BE),
564 EFX_MAE_MV_DESC_ALT(ENC_SRC_IP4_BE, ENC_SRC_IP6_BE, EFX_MAE_FIELD_BE),
565 EFX_MAE_MV_DESC_ALT(ENC_DST_IP4_BE, ENC_DST_IP6_BE, EFX_MAE_FIELD_BE),
566 EFX_MAE_MV_DESC(ENC_IP_PROTO, EFX_MAE_FIELD_BE),
567 EFX_MAE_MV_DESC(ENC_IP_TOS, EFX_MAE_FIELD_BE),
568 EFX_MAE_MV_DESC(ENC_IP_TTL, EFX_MAE_FIELD_BE),
569 EFX_MAE_MV_DESC_ALT(ENC_SRC_IP6_BE, ENC_SRC_IP4_BE, EFX_MAE_FIELD_BE),
570 EFX_MAE_MV_DESC_ALT(ENC_DST_IP6_BE, ENC_DST_IP4_BE, EFX_MAE_FIELD_BE),
571 EFX_MAE_MV_DESC(ENC_L4_SPORT_BE, EFX_MAE_FIELD_BE),
572 EFX_MAE_MV_DESC(ENC_L4_DPORT_BE, EFX_MAE_FIELD_BE),
574 #undef EFX_MAE_MV_DESC_ALT
575 #undef EFX_MAE_MV_DESC
578 __checkReturn efx_rc_t
579 efx_mae_mport_by_phy_port(
580 __in uint32_t phy_port,
581 __out efx_mport_sel_t *mportp)
586 if (phy_port > EFX_MASK32(MAE_MPORT_SELECTOR_PPORT_ID)) {
591 EFX_POPULATE_DWORD_2(dword,
592 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_PPORT,
593 MAE_MPORT_SELECTOR_PPORT_ID, phy_port);
595 memset(mportp, 0, sizeof (*mportp));
597 * The constructed DWORD is little-endian,
598 * but the resulting value is meant to be
599 * passed to MCDIs, where it will undergo
600 * host-order to little endian conversion.
602 mportp->sel = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
607 EFSYS_PROBE1(fail1, efx_rc_t, rc);
611 __checkReturn efx_rc_t
612 efx_mae_mport_by_pcie_function(
615 __out efx_mport_sel_t *mportp)
620 EFX_STATIC_ASSERT(EFX_PCI_VF_INVALID ==
621 MAE_MPORT_SELECTOR_FUNC_VF_ID_NULL);
623 if (pf > EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_PF_ID)) {
628 if (vf > EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_VF_ID)) {
633 EFX_POPULATE_DWORD_3(dword,
634 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_FUNC,
635 MAE_MPORT_SELECTOR_FUNC_PF_ID, pf,
636 MAE_MPORT_SELECTOR_FUNC_VF_ID, vf);
638 memset(mportp, 0, sizeof (*mportp));
640 * The constructed DWORD is little-endian,
641 * but the resulting value is meant to be
642 * passed to MCDIs, where it will undergo
643 * host-order to little endian conversion.
645 mportp->sel = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
652 EFSYS_PROBE1(fail1, efx_rc_t, rc);
656 __checkReturn efx_rc_t
657 efx_mae_match_spec_field_set(
658 __in efx_mae_match_spec_t *spec,
659 __in efx_mae_field_id_t field_id,
660 __in size_t value_size,
661 __in_bcount(value_size) const uint8_t *value,
662 __in size_t mask_size,
663 __in_bcount(mask_size) const uint8_t *mask)
665 const efx_mae_mv_desc_t *descp;
666 unsigned int desc_set_nentries;
670 switch (spec->emms_type) {
671 case EFX_MAE_RULE_OUTER:
673 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_desc_set);
674 descp = &__efx_mae_outer_rule_mv_desc_set[field_id];
675 mvp = spec->emms_mask_value_pairs.outer;
677 case EFX_MAE_RULE_ACTION:
679 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
680 descp = &__efx_mae_action_rule_mv_desc_set[field_id];
681 mvp = spec->emms_mask_value_pairs.action;
688 if ((unsigned int)field_id >= desc_set_nentries) {
693 if (descp->emmd_mask_size == 0) {
694 /* The ID points to a gap in the array of field descriptors. */
699 if (value_size != descp->emmd_value_size) {
704 if (mask_size != descp->emmd_mask_size) {
709 if (descp->emmd_endianness == EFX_MAE_FIELD_BE) {
713 * The mask/value are in network (big endian) order.
714 * The MCDI request field is also big endian.
717 EFSYS_ASSERT3U(value_size, ==, mask_size);
719 for (i = 0; i < value_size; ++i) {
720 uint8_t *v_bytep = mvp + descp->emmd_value_offset + i;
721 uint8_t *m_bytep = mvp + descp->emmd_mask_offset + i;
724 * Apply the mask (which may be all-zeros) to the value.
726 * If this API is provided with some value to set for a
727 * given field in one specification and with some other
728 * value to set for this field in another specification,
729 * then, if the two masks are all-zeros, the field will
730 * avoid being counted as a mismatch when comparing the
731 * specifications using efx_mae_match_specs_equal() API.
733 *v_bytep = value[i] & mask[i];
740 * The mask/value are in host byte order.
741 * The MCDI request field is little endian.
743 switch (value_size) {
745 EFX_POPULATE_DWORD_1(dword,
746 EFX_DWORD_0, *(const uint32_t *)value);
748 memcpy(mvp + descp->emmd_value_offset,
749 &dword, sizeof (dword));
752 EFSYS_ASSERT(B_FALSE);
757 EFX_POPULATE_DWORD_1(dword,
758 EFX_DWORD_0, *(const uint32_t *)mask);
760 memcpy(mvp + descp->emmd_mask_offset,
761 &dword, sizeof (dword));
764 EFSYS_ASSERT(B_FALSE);
779 EFSYS_PROBE1(fail1, efx_rc_t, rc);
783 __checkReturn efx_rc_t
784 efx_mae_match_spec_mport_set(
785 __in efx_mae_match_spec_t *spec,
786 __in const efx_mport_sel_t *valuep,
787 __in_opt const efx_mport_sel_t *maskp)
789 uint32_t full_mask = UINT32_MAX;
794 if (valuep == NULL) {
799 vp = (const uint8_t *)&valuep->sel;
801 mp = (const uint8_t *)&maskp->sel;
803 mp = (const uint8_t *)&full_mask;
805 rc = efx_mae_match_spec_field_set(spec,
806 EFX_MAE_FIELD_INGRESS_MPORT_SELECTOR,
807 sizeof (valuep->sel), vp, sizeof (maskp->sel), mp);
816 EFSYS_PROBE1(fail1, efx_rc_t, rc);
820 __checkReturn boolean_t
821 efx_mae_match_specs_equal(
822 __in const efx_mae_match_spec_t *left,
823 __in const efx_mae_match_spec_t *right)
825 return ((memcmp(left, right, sizeof (*left)) == 0) ? B_TRUE : B_FALSE);
828 #define EFX_MASK_BIT_IS_SET(_mask, _mask_page_nbits, _bit) \
829 ((_mask)[(_bit) / (_mask_page_nbits)] & \
830 (1ULL << ((_bit) & ((_mask_page_nbits) - 1))))
834 __in size_t mask_nbytes,
835 __in_bcount(mask_nbytes) const uint8_t *maskp)
837 boolean_t prev_bit_is_set = B_TRUE;
840 for (i = 0; i < 8 * mask_nbytes; ++i) {
841 boolean_t bit_is_set = EFX_MASK_BIT_IS_SET(maskp, 8, i);
843 if (!prev_bit_is_set && bit_is_set)
846 prev_bit_is_set = bit_is_set;
853 efx_mask_is_all_ones(
854 __in size_t mask_nbytes,
855 __in_bcount(mask_nbytes) const uint8_t *maskp)
860 for (i = 0; i < mask_nbytes; ++i)
863 return (t == (uint8_t)(~0));
867 efx_mask_is_all_zeros(
868 __in size_t mask_nbytes,
869 __in_bcount(mask_nbytes) const uint8_t *maskp)
874 for (i = 0; i < mask_nbytes; ++i)
880 __checkReturn boolean_t
881 efx_mae_match_spec_is_valid(
883 __in const efx_mae_match_spec_t *spec)
885 efx_mae_t *maep = enp->en_maep;
886 unsigned int field_ncaps = maep->em_max_nfields;
887 const efx_mae_field_cap_t *field_caps;
888 const efx_mae_mv_desc_t *desc_setp;
889 unsigned int desc_set_nentries;
890 boolean_t is_valid = B_TRUE;
891 efx_mae_field_id_t field_id;
894 switch (spec->emms_type) {
895 case EFX_MAE_RULE_OUTER:
896 field_caps = maep->em_outer_rule_field_caps;
897 desc_setp = __efx_mae_outer_rule_mv_desc_set;
899 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_desc_set);
900 mvp = spec->emms_mask_value_pairs.outer;
902 case EFX_MAE_RULE_ACTION:
903 field_caps = maep->em_action_rule_field_caps;
904 desc_setp = __efx_mae_action_rule_mv_desc_set;
906 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
907 mvp = spec->emms_mask_value_pairs.action;
913 if (field_caps == NULL)
916 for (field_id = 0; (unsigned int)field_id < desc_set_nentries;
918 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
919 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
920 const uint8_t *alt_m_buf = mvp + descp->emmd_alt_mask_offset;
921 const uint8_t *m_buf = mvp + descp->emmd_mask_offset;
922 size_t alt_m_size = descp->emmd_alt_mask_size;
923 size_t m_size = descp->emmd_mask_size;
926 continue; /* Skip array gap */
928 if ((unsigned int)field_cap_id >= field_ncaps) {
930 * The FW has not reported capability status for
931 * this field. Make sure that its mask is zeroed.
933 is_valid = efx_mask_is_all_zeros(m_size, m_buf);
934 if (is_valid != B_FALSE)
940 switch (field_caps[field_cap_id].emfc_support) {
941 case MAE_FIELD_SUPPORTED_MATCH_MASK:
944 case MAE_FIELD_SUPPORTED_MATCH_PREFIX:
945 is_valid = efx_mask_is_prefix(m_size, m_buf);
947 case MAE_FIELD_SUPPORTED_MATCH_OPTIONAL:
948 is_valid = (efx_mask_is_all_ones(m_size, m_buf) ||
949 efx_mask_is_all_zeros(m_size, m_buf));
951 case MAE_FIELD_SUPPORTED_MATCH_ALWAYS:
952 is_valid = efx_mask_is_all_ones(m_size, m_buf);
954 if ((is_valid == B_FALSE) && (alt_m_size != 0)) {
956 * This field has an alternative one. The FW
957 * reports ALWAYS for both implying that one
958 * of them is required to have all-ones mask.
960 * The primary field's mask is incorrect; go
961 * on to check that of the alternative field.
963 is_valid = efx_mask_is_all_ones(alt_m_size,
967 case MAE_FIELD_SUPPORTED_MATCH_NEVER:
968 case MAE_FIELD_UNSUPPORTED:
970 is_valid = efx_mask_is_all_zeros(m_size, m_buf);
974 if (is_valid == B_FALSE)
981 __checkReturn efx_rc_t
982 efx_mae_action_set_spec_init(
984 __out efx_mae_actions_t **specp)
986 efx_mae_actions_t *spec;
989 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
1000 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1005 efx_mae_action_set_spec_fini(
1006 __in efx_nic_t *enp,
1007 __in efx_mae_actions_t *spec)
1009 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
1012 static __checkReturn efx_rc_t
1013 efx_mae_action_set_add_vlan_pop(
1014 __in efx_mae_actions_t *spec,
1015 __in size_t arg_size,
1016 __in_bcount(arg_size) const uint8_t *arg)
1020 if (arg_size != 0) {
1030 if (spec->ema_n_vlan_tags_to_pop == EFX_MAE_VLAN_POP_MAX_NTAGS) {
1035 ++spec->ema_n_vlan_tags_to_pop;
1044 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1048 static __checkReturn efx_rc_t
1049 efx_mae_action_set_add_vlan_push(
1050 __in efx_mae_actions_t *spec,
1051 __in size_t arg_size,
1052 __in_bcount(arg_size) const uint8_t *arg)
1054 unsigned int n_tags = spec->ema_n_vlan_tags_to_push;
1057 if (arg_size != sizeof (*spec->ema_vlan_push_descs)) {
1067 if (n_tags == EFX_MAE_VLAN_PUSH_MAX_NTAGS) {
1072 memcpy(&spec->ema_vlan_push_descs[n_tags], arg, arg_size);
1073 ++(spec->ema_n_vlan_tags_to_push);
1082 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1086 static __checkReturn efx_rc_t
1087 efx_mae_action_set_add_flag(
1088 __in efx_mae_actions_t *spec,
1089 __in size_t arg_size,
1090 __in_bcount(arg_size) const uint8_t *arg)
1094 _NOTE(ARGUNUSED(spec))
1096 if (arg_size != 0) {
1106 /* This action does not have any arguments, so do nothing here. */
1113 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1117 static __checkReturn efx_rc_t
1118 efx_mae_action_set_add_mark(
1119 __in efx_mae_actions_t *spec,
1120 __in size_t arg_size,
1121 __in_bcount(arg_size) const uint8_t *arg)
1125 if (arg_size != sizeof (spec->ema_mark_value)) {
1135 memcpy(&spec->ema_mark_value, arg, arg_size);
1142 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1146 static __checkReturn efx_rc_t
1147 efx_mae_action_set_add_deliver(
1148 __in efx_mae_actions_t *spec,
1149 __in size_t arg_size,
1150 __in_bcount(arg_size) const uint8_t *arg)
1154 if (arg_size != sizeof (spec->ema_deliver_mport)) {
1164 memcpy(&spec->ema_deliver_mport, arg, arg_size);
1171 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1175 typedef struct efx_mae_action_desc_s {
1176 /* Action specific handler */
1177 efx_rc_t (*emad_add)(efx_mae_actions_t *,
1178 size_t, const uint8_t *);
1179 } efx_mae_action_desc_t;
1181 static const efx_mae_action_desc_t efx_mae_actions[EFX_MAE_NACTIONS] = {
1182 [EFX_MAE_ACTION_VLAN_POP] = {
1183 .emad_add = efx_mae_action_set_add_vlan_pop
1185 [EFX_MAE_ACTION_VLAN_PUSH] = {
1186 .emad_add = efx_mae_action_set_add_vlan_push
1188 [EFX_MAE_ACTION_FLAG] = {
1189 .emad_add = efx_mae_action_set_add_flag
1191 [EFX_MAE_ACTION_MARK] = {
1192 .emad_add = efx_mae_action_set_add_mark
1194 [EFX_MAE_ACTION_DELIVER] = {
1195 .emad_add = efx_mae_action_set_add_deliver
1199 static const uint32_t efx_mae_action_ordered_map =
1200 (1U << EFX_MAE_ACTION_VLAN_POP) |
1201 (1U << EFX_MAE_ACTION_VLAN_PUSH) |
1202 (1U << EFX_MAE_ACTION_FLAG) |
1203 (1U << EFX_MAE_ACTION_MARK) |
1204 (1U << EFX_MAE_ACTION_DELIVER);
1207 * These actions must not be added after DELIVER, but
1208 * they can have any place among the rest of
1209 * strictly ordered actions.
1211 static const uint32_t efx_mae_action_nonstrict_map =
1212 (1U << EFX_MAE_ACTION_FLAG) |
1213 (1U << EFX_MAE_ACTION_MARK);
1215 static const uint32_t efx_mae_action_repeat_map =
1216 (1U << EFX_MAE_ACTION_VLAN_POP) |
1217 (1U << EFX_MAE_ACTION_VLAN_PUSH);
1220 * Add an action to an action set.
1222 * This has to be invoked in the desired action order.
1223 * An out-of-order action request will be turned down.
1225 static __checkReturn efx_rc_t
1226 efx_mae_action_set_spec_populate(
1227 __in efx_mae_actions_t *spec,
1228 __in efx_mae_action_t type,
1229 __in size_t arg_size,
1230 __in_bcount(arg_size) const uint8_t *arg)
1232 uint32_t action_mask;
1235 EFX_STATIC_ASSERT(EFX_MAE_NACTIONS <=
1236 (sizeof (efx_mae_action_ordered_map) * 8));
1237 EFX_STATIC_ASSERT(EFX_MAE_NACTIONS <=
1238 (sizeof (efx_mae_action_repeat_map) * 8));
1240 EFX_STATIC_ASSERT(EFX_MAE_ACTION_DELIVER + 1 == EFX_MAE_NACTIONS);
1241 EFX_STATIC_ASSERT(EFX_MAE_ACTION_FLAG + 1 == EFX_MAE_ACTION_MARK);
1242 EFX_STATIC_ASSERT(EFX_MAE_ACTION_MARK + 1 == EFX_MAE_ACTION_DELIVER);
1244 if (type >= EFX_ARRAY_SIZE(efx_mae_actions)) {
1249 action_mask = (1U << type);
1251 if ((spec->ema_actions & action_mask) != 0) {
1252 /* The action set already contains this action. */
1253 if ((efx_mae_action_repeat_map & action_mask) == 0) {
1254 /* Cannot add another non-repeatable action. */
1260 if ((efx_mae_action_ordered_map & action_mask) != 0) {
1261 uint32_t strict_ordered_map =
1262 efx_mae_action_ordered_map & ~efx_mae_action_nonstrict_map;
1263 uint32_t later_actions_mask =
1264 strict_ordered_map & ~(action_mask | (action_mask - 1));
1266 if ((spec->ema_actions & later_actions_mask) != 0) {
1267 /* Cannot add an action after later ordered actions. */
1273 if (efx_mae_actions[type].emad_add != NULL) {
1274 rc = efx_mae_actions[type].emad_add(spec, arg_size, arg);
1279 spec->ema_actions |= action_mask;
1290 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1294 __checkReturn efx_rc_t
1295 efx_mae_action_set_populate_vlan_pop(
1296 __in efx_mae_actions_t *spec)
1298 return (efx_mae_action_set_spec_populate(spec,
1299 EFX_MAE_ACTION_VLAN_POP, 0, NULL));
1302 __checkReturn efx_rc_t
1303 efx_mae_action_set_populate_vlan_push(
1304 __in efx_mae_actions_t *spec,
1305 __in uint16_t tpid_be,
1306 __in uint16_t tci_be)
1308 efx_mae_action_vlan_push_t action;
1309 const uint8_t *arg = (const uint8_t *)&action;
1311 action.emavp_tpid_be = tpid_be;
1312 action.emavp_tci_be = tci_be;
1314 return (efx_mae_action_set_spec_populate(spec,
1315 EFX_MAE_ACTION_VLAN_PUSH, sizeof (action), arg));
1318 __checkReturn efx_rc_t
1319 efx_mae_action_set_populate_flag(
1320 __in efx_mae_actions_t *spec)
1322 return (efx_mae_action_set_spec_populate(spec,
1323 EFX_MAE_ACTION_FLAG, 0, NULL));
1326 __checkReturn efx_rc_t
1327 efx_mae_action_set_populate_mark(
1328 __in efx_mae_actions_t *spec,
1329 __in uint32_t mark_value)
1331 const uint8_t *arg = (const uint8_t *)&mark_value;
1333 return (efx_mae_action_set_spec_populate(spec,
1334 EFX_MAE_ACTION_MARK, sizeof (mark_value), arg));
1337 __checkReturn efx_rc_t
1338 efx_mae_action_set_populate_deliver(
1339 __in efx_mae_actions_t *spec,
1340 __in const efx_mport_sel_t *mportp)
1345 if (mportp == NULL) {
1350 arg = (const uint8_t *)&mportp->sel;
1352 return (efx_mae_action_set_spec_populate(spec,
1353 EFX_MAE_ACTION_DELIVER, sizeof (mportp->sel), arg));
1356 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1360 __checkReturn efx_rc_t
1361 efx_mae_action_set_populate_drop(
1362 __in efx_mae_actions_t *spec)
1364 efx_mport_sel_t mport;
1368 EFX_POPULATE_DWORD_1(dword,
1369 MAE_MPORT_SELECTOR_FLAT, MAE_MPORT_SELECTOR_NULL);
1372 * The constructed DWORD is little-endian,
1373 * but the resulting value is meant to be
1374 * passed to MCDIs, where it will undergo
1375 * host-order to little endian conversion.
1377 mport.sel = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
1379 arg = (const uint8_t *)&mport.sel;
1381 return (efx_mae_action_set_spec_populate(spec,
1382 EFX_MAE_ACTION_DELIVER, sizeof (mport.sel), arg));
1385 __checkReturn boolean_t
1386 efx_mae_action_set_specs_equal(
1387 __in const efx_mae_actions_t *left,
1388 __in const efx_mae_actions_t *right)
1390 return ((memcmp(left, right, sizeof (*left)) == 0) ? B_TRUE : B_FALSE);
1393 __checkReturn efx_rc_t
1394 efx_mae_match_specs_class_cmp(
1395 __in efx_nic_t *enp,
1396 __in const efx_mae_match_spec_t *left,
1397 __in const efx_mae_match_spec_t *right,
1398 __out boolean_t *have_same_classp)
1400 efx_mae_t *maep = enp->en_maep;
1401 unsigned int field_ncaps = maep->em_max_nfields;
1402 const efx_mae_field_cap_t *field_caps;
1403 const efx_mae_mv_desc_t *desc_setp;
1404 unsigned int desc_set_nentries;
1405 boolean_t have_same_class = B_TRUE;
1406 efx_mae_field_id_t field_id;
1407 const uint8_t *mvpl;
1408 const uint8_t *mvpr;
1411 switch (left->emms_type) {
1412 case EFX_MAE_RULE_OUTER:
1413 field_caps = maep->em_outer_rule_field_caps;
1414 desc_setp = __efx_mae_outer_rule_mv_desc_set;
1416 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_desc_set);
1417 mvpl = left->emms_mask_value_pairs.outer;
1418 mvpr = right->emms_mask_value_pairs.outer;
1420 case EFX_MAE_RULE_ACTION:
1421 field_caps = maep->em_action_rule_field_caps;
1422 desc_setp = __efx_mae_action_rule_mv_desc_set;
1424 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
1425 mvpl = left->emms_mask_value_pairs.action;
1426 mvpr = right->emms_mask_value_pairs.action;
1433 if (field_caps == NULL) {
1438 if (left->emms_type != right->emms_type ||
1439 left->emms_prio != right->emms_prio) {
1441 * Rules of different types can never map to the same class.
1443 * The FW can support some set of match criteria for one
1444 * priority and not support the very same set for
1445 * another priority. Thus, two rules which have
1446 * different priorities can never map to
1449 *have_same_classp = B_FALSE;
1453 for (field_id = 0; (unsigned int)field_id < desc_set_nentries;
1455 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
1456 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
1457 const uint8_t *lmaskp = mvpl + descp->emmd_mask_offset;
1458 const uint8_t *rmaskp = mvpr + descp->emmd_mask_offset;
1459 size_t mask_size = descp->emmd_mask_size;
1460 const uint8_t *lvalp = mvpl + descp->emmd_value_offset;
1461 const uint8_t *rvalp = mvpr + descp->emmd_value_offset;
1462 size_t value_size = descp->emmd_value_size;
1465 continue; /* Skip array gap */
1467 if ((unsigned int)field_cap_id >= field_ncaps) {
1469 * The FW has not reported capability status for this
1470 * field. It's unknown whether any difference between
1471 * the two masks / values affects the class. The only
1472 * case when the class must be the same is when these
1473 * mask-value pairs match. Otherwise, report mismatch.
1475 if ((memcmp(lmaskp, rmaskp, mask_size) == 0) &&
1476 (memcmp(lvalp, rvalp, value_size) == 0))
1482 if (field_caps[field_cap_id].emfc_mask_affects_class) {
1483 if (memcmp(lmaskp, rmaskp, mask_size) != 0) {
1484 have_same_class = B_FALSE;
1489 if (field_caps[field_cap_id].emfc_match_affects_class) {
1490 if (memcmp(lvalp, rvalp, value_size) != 0) {
1491 have_same_class = B_FALSE;
1497 *have_same_classp = have_same_class;
1504 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1508 __checkReturn efx_rc_t
1509 efx_mae_outer_rule_insert(
1510 __in efx_nic_t *enp,
1511 __in const efx_mae_match_spec_t *spec,
1512 __in efx_tunnel_protocol_t encap_type,
1513 __out efx_mae_rule_id_t *or_idp)
1515 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1517 EFX_MCDI_DECLARE_BUF(payload,
1518 MC_CMD_MAE_OUTER_RULE_INSERT_IN_LENMAX_MCDI2,
1519 MC_CMD_MAE_OUTER_RULE_INSERT_OUT_LEN);
1520 uint32_t encap_type_mcdi;
1521 efx_mae_rule_id_t or_id;
1525 EFX_STATIC_ASSERT(sizeof (or_idp->id) ==
1526 MC_CMD_MAE_OUTER_RULE_INSERT_OUT_OR_ID_LEN);
1528 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
1529 MC_CMD_MAE_OUTER_RULE_INSERT_OUT_OUTER_RULE_ID_NULL);
1531 if (encp->enc_mae_supported == B_FALSE) {
1536 if (spec->emms_type != EFX_MAE_RULE_OUTER) {
1541 switch (encap_type) {
1542 case EFX_TUNNEL_PROTOCOL_NONE:
1543 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_NONE;
1545 case EFX_TUNNEL_PROTOCOL_VXLAN:
1546 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_VXLAN;
1548 case EFX_TUNNEL_PROTOCOL_GENEVE:
1549 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_GENEVE;
1551 case EFX_TUNNEL_PROTOCOL_NVGRE:
1552 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_NVGRE;
1559 req.emr_cmd = MC_CMD_MAE_OUTER_RULE_INSERT;
1560 req.emr_in_buf = payload;
1561 req.emr_in_length = MC_CMD_MAE_OUTER_RULE_INSERT_IN_LENMAX_MCDI2;
1562 req.emr_out_buf = payload;
1563 req.emr_out_length = MC_CMD_MAE_OUTER_RULE_INSERT_OUT_LEN;
1565 MCDI_IN_SET_DWORD(req,
1566 MAE_OUTER_RULE_INSERT_IN_ENCAP_TYPE, encap_type_mcdi);
1568 MCDI_IN_SET_DWORD(req, MAE_OUTER_RULE_INSERT_IN_PRIO, spec->emms_prio);
1571 * Mask-value pairs have been stored in the byte order needed for the
1572 * MCDI request and are thus safe to be copied directly to the buffer.
1573 * The library cares about byte order in efx_mae_match_spec_field_set().
1575 EFX_STATIC_ASSERT(sizeof (spec->emms_mask_value_pairs.outer) >=
1576 MAE_ENC_FIELD_PAIRS_LEN);
1577 offset = MC_CMD_MAE_OUTER_RULE_INSERT_IN_FIELD_MATCH_CRITERIA_OFST;
1578 memcpy(payload + offset, spec->emms_mask_value_pairs.outer,
1579 MAE_ENC_FIELD_PAIRS_LEN);
1581 efx_mcdi_execute(enp, &req);
1583 if (req.emr_rc != 0) {
1588 if (req.emr_out_length_used < MC_CMD_MAE_OUTER_RULE_INSERT_OUT_LEN) {
1593 or_id.id = MCDI_OUT_DWORD(req, MAE_OUTER_RULE_INSERT_OUT_OR_ID);
1594 if (or_id.id == EFX_MAE_RSRC_ID_INVALID) {
1599 or_idp->id = or_id.id;
1614 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1618 __checkReturn efx_rc_t
1619 efx_mae_outer_rule_remove(
1620 __in efx_nic_t *enp,
1621 __in const efx_mae_rule_id_t *or_idp)
1623 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1625 EFX_MCDI_DECLARE_BUF(payload,
1626 MC_CMD_MAE_OUTER_RULE_REMOVE_IN_LEN(1),
1627 MC_CMD_MAE_OUTER_RULE_REMOVE_OUT_LEN(1));
1630 if (encp->enc_mae_supported == B_FALSE) {
1635 req.emr_cmd = MC_CMD_MAE_OUTER_RULE_REMOVE;
1636 req.emr_in_buf = payload;
1637 req.emr_in_length = MC_CMD_MAE_OUTER_RULE_REMOVE_IN_LEN(1);
1638 req.emr_out_buf = payload;
1639 req.emr_out_length = MC_CMD_MAE_OUTER_RULE_REMOVE_OUT_LEN(1);
1641 MCDI_IN_SET_DWORD(req, MAE_OUTER_RULE_REMOVE_IN_OR_ID, or_idp->id);
1643 efx_mcdi_execute(enp, &req);
1645 if (req.emr_rc != 0) {
1650 if (MCDI_OUT_DWORD(req, MAE_OUTER_RULE_REMOVE_OUT_REMOVED_OR_ID) !=
1652 /* Firmware failed to remove the outer rule. */
1664 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1668 __checkReturn efx_rc_t
1669 efx_mae_match_spec_outer_rule_id_set(
1670 __in efx_mae_match_spec_t *spec,
1671 __in const efx_mae_rule_id_t *or_idp)
1673 uint32_t full_mask = UINT32_MAX;
1676 if (spec->emms_type != EFX_MAE_RULE_ACTION) {
1681 if (or_idp == NULL) {
1686 rc = efx_mae_match_spec_field_set(spec, EFX_MAE_FIELD_OUTER_RULE_ID,
1687 sizeof (or_idp->id), (const uint8_t *)&or_idp->id,
1688 sizeof (full_mask), (const uint8_t *)&full_mask);
1699 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1703 __checkReturn efx_rc_t
1704 efx_mae_action_set_alloc(
1705 __in efx_nic_t *enp,
1706 __in const efx_mae_actions_t *spec,
1707 __out efx_mae_aset_id_t *aset_idp)
1709 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1711 EFX_MCDI_DECLARE_BUF(payload,
1712 MC_CMD_MAE_ACTION_SET_ALLOC_IN_LEN,
1713 MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN);
1714 efx_mae_aset_id_t aset_id;
1717 if (encp->enc_mae_supported == B_FALSE) {
1722 req.emr_cmd = MC_CMD_MAE_ACTION_SET_ALLOC;
1723 req.emr_in_buf = payload;
1724 req.emr_in_length = MC_CMD_MAE_ACTION_SET_ALLOC_IN_LEN;
1725 req.emr_out_buf = payload;
1726 req.emr_out_length = MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN;
1729 * TODO: Remove these EFX_MAE_RSRC_ID_INVALID assignments once the
1730 * corresponding resource types are supported by the implementation.
1731 * Use proper resource ID assignments instead.
1733 MCDI_IN_SET_DWORD(req,
1734 MAE_ACTION_SET_ALLOC_IN_COUNTER_LIST_ID, EFX_MAE_RSRC_ID_INVALID);
1735 MCDI_IN_SET_DWORD(req,
1736 MAE_ACTION_SET_ALLOC_IN_COUNTER_ID, EFX_MAE_RSRC_ID_INVALID);
1737 MCDI_IN_SET_DWORD(req,
1738 MAE_ACTION_SET_ALLOC_IN_ENCAP_HEADER_ID, EFX_MAE_RSRC_ID_INVALID);
1740 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
1741 MAE_ACTION_SET_ALLOC_IN_VLAN_POP, spec->ema_n_vlan_tags_to_pop);
1743 if (spec->ema_n_vlan_tags_to_push > 0) {
1744 unsigned int outer_tag_idx;
1746 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
1747 MAE_ACTION_SET_ALLOC_IN_VLAN_PUSH,
1748 spec->ema_n_vlan_tags_to_push);
1750 if (spec->ema_n_vlan_tags_to_push ==
1751 EFX_MAE_VLAN_PUSH_MAX_NTAGS) {
1752 MCDI_IN_SET_WORD(req,
1753 MAE_ACTION_SET_ALLOC_IN_VLAN1_PROTO_BE,
1754 spec->ema_vlan_push_descs[0].emavp_tpid_be);
1755 MCDI_IN_SET_WORD(req,
1756 MAE_ACTION_SET_ALLOC_IN_VLAN1_TCI_BE,
1757 spec->ema_vlan_push_descs[0].emavp_tci_be);
1760 outer_tag_idx = spec->ema_n_vlan_tags_to_push - 1;
1762 MCDI_IN_SET_WORD(req, MAE_ACTION_SET_ALLOC_IN_VLAN0_PROTO_BE,
1763 spec->ema_vlan_push_descs[outer_tag_idx].emavp_tpid_be);
1764 MCDI_IN_SET_WORD(req, MAE_ACTION_SET_ALLOC_IN_VLAN0_TCI_BE,
1765 spec->ema_vlan_push_descs[outer_tag_idx].emavp_tci_be);
1768 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_FLAG)) != 0) {
1769 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
1770 MAE_ACTION_SET_ALLOC_IN_FLAG, 1);
1773 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_MARK)) != 0) {
1774 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
1775 MAE_ACTION_SET_ALLOC_IN_MARK, 1);
1777 MCDI_IN_SET_DWORD(req,
1778 MAE_ACTION_SET_ALLOC_IN_MARK_VALUE, spec->ema_mark_value);
1781 MCDI_IN_SET_DWORD(req,
1782 MAE_ACTION_SET_ALLOC_IN_DELIVER, spec->ema_deliver_mport.sel);
1784 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_SRC_MAC_ID,
1785 MC_CMD_MAE_MAC_ADDR_ALLOC_OUT_MAC_ID_NULL);
1786 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_DST_MAC_ID,
1787 MC_CMD_MAE_MAC_ADDR_ALLOC_OUT_MAC_ID_NULL);
1789 efx_mcdi_execute(enp, &req);
1791 if (req.emr_rc != 0) {
1796 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN) {
1801 aset_id.id = MCDI_OUT_DWORD(req, MAE_ACTION_SET_ALLOC_OUT_AS_ID);
1802 if (aset_id.id == EFX_MAE_RSRC_ID_INVALID) {
1807 aset_idp->id = aset_id.id;
1818 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1822 __checkReturn efx_rc_t
1823 efx_mae_action_set_free(
1824 __in efx_nic_t *enp,
1825 __in const efx_mae_aset_id_t *aset_idp)
1827 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1829 EFX_MCDI_DECLARE_BUF(payload,
1830 MC_CMD_MAE_ACTION_SET_FREE_IN_LEN(1),
1831 MC_CMD_MAE_ACTION_SET_FREE_OUT_LEN(1));
1834 if (encp->enc_mae_supported == B_FALSE) {
1839 req.emr_cmd = MC_CMD_MAE_ACTION_SET_FREE;
1840 req.emr_in_buf = payload;
1841 req.emr_in_length = MC_CMD_MAE_ACTION_SET_FREE_IN_LEN(1);
1842 req.emr_out_buf = payload;
1843 req.emr_out_length = MC_CMD_MAE_ACTION_SET_FREE_OUT_LEN(1);
1845 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_FREE_IN_AS_ID, aset_idp->id);
1847 efx_mcdi_execute(enp, &req);
1849 if (req.emr_rc != 0) {
1854 if (MCDI_OUT_DWORD(req, MAE_ACTION_SET_FREE_OUT_FREED_AS_ID) !=
1856 /* Firmware failed to free the action set. */
1868 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1872 __checkReturn efx_rc_t
1873 efx_mae_action_rule_insert(
1874 __in efx_nic_t *enp,
1875 __in const efx_mae_match_spec_t *spec,
1876 __in const efx_mae_aset_list_id_t *asl_idp,
1877 __in const efx_mae_aset_id_t *as_idp,
1878 __out efx_mae_rule_id_t *ar_idp)
1880 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1882 EFX_MCDI_DECLARE_BUF(payload,
1883 MC_CMD_MAE_ACTION_RULE_INSERT_IN_LENMAX_MCDI2,
1884 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN);
1885 efx_oword_t *rule_response;
1886 efx_mae_rule_id_t ar_id;
1890 EFX_STATIC_ASSERT(sizeof (ar_idp->id) ==
1891 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_AR_ID_LEN);
1893 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
1894 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_ACTION_RULE_ID_NULL);
1896 if (encp->enc_mae_supported == B_FALSE) {
1901 if (spec->emms_type != EFX_MAE_RULE_ACTION ||
1902 (asl_idp != NULL && as_idp != NULL) ||
1903 (asl_idp == NULL && as_idp == NULL)) {
1908 req.emr_cmd = MC_CMD_MAE_ACTION_RULE_INSERT;
1909 req.emr_in_buf = payload;
1910 req.emr_in_length = MC_CMD_MAE_ACTION_RULE_INSERT_IN_LENMAX_MCDI2;
1911 req.emr_out_buf = payload;
1912 req.emr_out_length = MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN;
1914 EFX_STATIC_ASSERT(sizeof (*rule_response) <=
1915 MC_CMD_MAE_ACTION_RULE_INSERT_IN_RESPONSE_LEN);
1916 offset = MC_CMD_MAE_ACTION_RULE_INSERT_IN_RESPONSE_OFST;
1917 rule_response = (efx_oword_t *)(payload + offset);
1918 EFX_POPULATE_OWORD_3(*rule_response,
1919 MAE_ACTION_RULE_RESPONSE_ASL_ID,
1920 (asl_idp != NULL) ? asl_idp->id : EFX_MAE_RSRC_ID_INVALID,
1921 MAE_ACTION_RULE_RESPONSE_AS_ID,
1922 (as_idp != NULL) ? as_idp->id : EFX_MAE_RSRC_ID_INVALID,
1923 MAE_ACTION_RULE_RESPONSE_COUNTER_ID, EFX_MAE_RSRC_ID_INVALID);
1925 MCDI_IN_SET_DWORD(req, MAE_ACTION_RULE_INSERT_IN_PRIO, spec->emms_prio);
1928 * Mask-value pairs have been stored in the byte order needed for the
1929 * MCDI request and are thus safe to be copied directly to the buffer.
1931 EFX_STATIC_ASSERT(sizeof (spec->emms_mask_value_pairs.action) >=
1932 MAE_FIELD_MASK_VALUE_PAIRS_LEN);
1933 offset = MC_CMD_MAE_ACTION_RULE_INSERT_IN_MATCH_CRITERIA_OFST;
1934 memcpy(payload + offset, spec->emms_mask_value_pairs.action,
1935 MAE_FIELD_MASK_VALUE_PAIRS_LEN);
1937 efx_mcdi_execute(enp, &req);
1939 if (req.emr_rc != 0) {
1944 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN) {
1949 ar_id.id = MCDI_OUT_DWORD(req, MAE_ACTION_RULE_INSERT_OUT_AR_ID);
1950 if (ar_id.id == EFX_MAE_RSRC_ID_INVALID) {
1955 ar_idp->id = ar_id.id;
1968 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1972 __checkReturn efx_rc_t
1973 efx_mae_action_rule_remove(
1974 __in efx_nic_t *enp,
1975 __in const efx_mae_rule_id_t *ar_idp)
1977 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1979 EFX_MCDI_DECLARE_BUF(payload,
1980 MC_CMD_MAE_ACTION_RULE_DELETE_IN_LEN(1),
1981 MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LEN(1));
1984 if (encp->enc_mae_supported == B_FALSE) {
1989 req.emr_cmd = MC_CMD_MAE_ACTION_RULE_DELETE;
1990 req.emr_in_buf = payload;
1991 req.emr_in_length = MC_CMD_MAE_ACTION_RULE_DELETE_IN_LEN(1);
1992 req.emr_out_buf = payload;
1993 req.emr_out_length = MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LEN(1);
1995 MCDI_IN_SET_DWORD(req, MAE_ACTION_RULE_DELETE_IN_AR_ID, ar_idp->id);
1997 efx_mcdi_execute(enp, &req);
1999 if (req.emr_rc != 0) {
2004 if (MCDI_OUT_DWORD(req, MAE_ACTION_RULE_DELETE_OUT_DELETED_AR_ID) !=
2006 /* Firmware failed to delete the action rule. */
2018 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2022 #endif /* EFSYS_OPT_MAE */