1 /* SPDX-License-Identifier: BSD-3-Clause
3 * Copyright(c) 2019-2021 Xilinx, Inc.
12 static __checkReturn efx_rc_t
13 efx_mae_get_capabilities(
17 EFX_MCDI_DECLARE_BUF(payload,
18 MC_CMD_MAE_GET_CAPS_IN_LEN,
19 MC_CMD_MAE_GET_CAPS_OUT_LEN);
20 struct efx_mae_s *maep = enp->en_maep;
23 req.emr_cmd = MC_CMD_MAE_GET_CAPS;
24 req.emr_in_buf = payload;
25 req.emr_in_length = MC_CMD_MAE_GET_CAPS_IN_LEN;
26 req.emr_out_buf = payload;
27 req.emr_out_length = MC_CMD_MAE_GET_CAPS_OUT_LEN;
29 efx_mcdi_execute(enp, &req);
31 if (req.emr_rc != 0) {
36 if (req.emr_out_length_used < MC_CMD_MAE_GET_CAPS_OUT_LEN) {
41 maep->em_max_n_outer_prios =
42 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_OUTER_PRIOS);
44 maep->em_max_n_action_prios =
45 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_ACTION_PRIOS);
47 maep->em_encap_types_supported = 0;
49 if (MCDI_OUT_DWORD_FIELD(req, MAE_GET_CAPS_OUT_ENCAP_TYPES_SUPPORTED,
50 MAE_GET_CAPS_OUT_ENCAP_TYPE_VXLAN) != 0) {
51 maep->em_encap_types_supported |=
52 (1U << EFX_TUNNEL_PROTOCOL_VXLAN);
55 if (MCDI_OUT_DWORD_FIELD(req, MAE_GET_CAPS_OUT_ENCAP_TYPES_SUPPORTED,
56 MAE_GET_CAPS_OUT_ENCAP_TYPE_GENEVE) != 0) {
57 maep->em_encap_types_supported |=
58 (1U << EFX_TUNNEL_PROTOCOL_GENEVE);
61 if (MCDI_OUT_DWORD_FIELD(req, MAE_GET_CAPS_OUT_ENCAP_TYPES_SUPPORTED,
62 MAE_GET_CAPS_OUT_ENCAP_TYPE_NVGRE) != 0) {
63 maep->em_encap_types_supported |=
64 (1U << EFX_TUNNEL_PROTOCOL_NVGRE);
67 maep->em_max_nfields =
68 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_MATCH_FIELD_COUNT);
70 maep->em_max_ncounters =
71 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_COUNTERS);
78 EFSYS_PROBE1(fail1, efx_rc_t, rc);
82 static __checkReturn efx_rc_t
83 efx_mae_get_outer_rule_caps(
85 __in unsigned int field_ncaps,
86 __out_ecount(field_ncaps) efx_mae_field_cap_t *field_caps)
89 EFX_MCDI_DECLARE_BUF(payload,
90 MC_CMD_MAE_GET_OR_CAPS_IN_LEN,
91 MC_CMD_MAE_GET_OR_CAPS_OUT_LENMAX_MCDI2);
92 unsigned int mcdi_field_ncaps;
96 if (MC_CMD_MAE_GET_OR_CAPS_OUT_LEN(field_ncaps) >
97 MC_CMD_MAE_GET_OR_CAPS_OUT_LENMAX_MCDI2) {
102 req.emr_cmd = MC_CMD_MAE_GET_OR_CAPS;
103 req.emr_in_buf = payload;
104 req.emr_in_length = MC_CMD_MAE_GET_OR_CAPS_IN_LEN;
105 req.emr_out_buf = payload;
106 req.emr_out_length = MC_CMD_MAE_GET_OR_CAPS_OUT_LEN(field_ncaps);
108 efx_mcdi_execute(enp, &req);
110 if (req.emr_rc != 0) {
115 if (req.emr_out_length_used < MC_CMD_MAE_GET_OR_CAPS_OUT_LENMIN) {
120 mcdi_field_ncaps = MCDI_OUT_DWORD(req, MAE_GET_OR_CAPS_OUT_COUNT);
122 if (req.emr_out_length_used <
123 MC_CMD_MAE_GET_OR_CAPS_OUT_LEN(mcdi_field_ncaps)) {
128 if (mcdi_field_ncaps > field_ncaps) {
133 for (i = 0; i < mcdi_field_ncaps; ++i) {
137 field_caps[i].emfc_support = MCDI_OUT_INDEXED_DWORD_FIELD(req,
138 MAE_GET_OR_CAPS_OUT_FIELD_FLAGS, i,
139 MAE_FIELD_FLAGS_SUPPORT_STATUS);
141 match_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
142 MAE_GET_OR_CAPS_OUT_FIELD_FLAGS, i,
143 MAE_FIELD_FLAGS_MATCH_AFFECTS_CLASS);
145 field_caps[i].emfc_match_affects_class =
146 (match_flag != 0) ? B_TRUE : B_FALSE;
148 mask_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
149 MAE_GET_OR_CAPS_OUT_FIELD_FLAGS, i,
150 MAE_FIELD_FLAGS_MASK_AFFECTS_CLASS);
152 field_caps[i].emfc_mask_affects_class =
153 (mask_flag != 0) ? B_TRUE : B_FALSE;
167 EFSYS_PROBE1(fail1, efx_rc_t, rc);
171 static __checkReturn efx_rc_t
172 efx_mae_get_action_rule_caps(
174 __in unsigned int field_ncaps,
175 __out_ecount(field_ncaps) efx_mae_field_cap_t *field_caps)
178 EFX_MCDI_DECLARE_BUF(payload,
179 MC_CMD_MAE_GET_AR_CAPS_IN_LEN,
180 MC_CMD_MAE_GET_AR_CAPS_OUT_LENMAX_MCDI2);
181 unsigned int mcdi_field_ncaps;
185 if (MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(field_ncaps) >
186 MC_CMD_MAE_GET_AR_CAPS_OUT_LENMAX_MCDI2) {
191 req.emr_cmd = MC_CMD_MAE_GET_AR_CAPS;
192 req.emr_in_buf = payload;
193 req.emr_in_length = MC_CMD_MAE_GET_AR_CAPS_IN_LEN;
194 req.emr_out_buf = payload;
195 req.emr_out_length = MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(field_ncaps);
197 efx_mcdi_execute(enp, &req);
199 if (req.emr_rc != 0) {
204 if (req.emr_out_length_used < MC_CMD_MAE_GET_AR_CAPS_OUT_LENMIN) {
209 mcdi_field_ncaps = MCDI_OUT_DWORD(req, MAE_GET_AR_CAPS_OUT_COUNT);
211 if (req.emr_out_length_used <
212 MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(mcdi_field_ncaps)) {
217 if (mcdi_field_ncaps > field_ncaps) {
222 for (i = 0; i < mcdi_field_ncaps; ++i) {
226 field_caps[i].emfc_support = MCDI_OUT_INDEXED_DWORD_FIELD(req,
227 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
228 MAE_FIELD_FLAGS_SUPPORT_STATUS);
230 match_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
231 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
232 MAE_FIELD_FLAGS_MATCH_AFFECTS_CLASS);
234 field_caps[i].emfc_match_affects_class =
235 (match_flag != 0) ? B_TRUE : B_FALSE;
237 mask_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
238 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
239 MAE_FIELD_FLAGS_MASK_AFFECTS_CLASS);
241 field_caps[i].emfc_mask_affects_class =
242 (mask_flag != 0) ? B_TRUE : B_FALSE;
256 EFSYS_PROBE1(fail1, efx_rc_t, rc);
260 __checkReturn efx_rc_t
264 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
265 efx_mae_field_cap_t *or_fcaps;
266 size_t or_fcaps_size;
267 efx_mae_field_cap_t *ar_fcaps;
268 size_t ar_fcaps_size;
272 if (encp->enc_mae_supported == B_FALSE) {
277 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*maep), maep);
285 rc = efx_mae_get_capabilities(enp);
289 or_fcaps_size = maep->em_max_nfields * sizeof (*or_fcaps);
290 EFSYS_KMEM_ALLOC(enp->en_esip, or_fcaps_size, or_fcaps);
291 if (or_fcaps == NULL) {
296 maep->em_outer_rule_field_caps_size = or_fcaps_size;
297 maep->em_outer_rule_field_caps = or_fcaps;
299 rc = efx_mae_get_outer_rule_caps(enp, maep->em_max_nfields, or_fcaps);
303 ar_fcaps_size = maep->em_max_nfields * sizeof (*ar_fcaps);
304 EFSYS_KMEM_ALLOC(enp->en_esip, ar_fcaps_size, ar_fcaps);
305 if (ar_fcaps == NULL) {
310 maep->em_action_rule_field_caps_size = ar_fcaps_size;
311 maep->em_action_rule_field_caps = ar_fcaps;
313 rc = efx_mae_get_action_rule_caps(enp, maep->em_max_nfields, ar_fcaps);
321 EFSYS_KMEM_FREE(enp->en_esip, ar_fcaps_size, ar_fcaps);
326 EFSYS_KMEM_FREE(enp->en_esip, or_fcaps_size, or_fcaps);
331 EFSYS_KMEM_FREE(enp->en_esip, sizeof (struct efx_mae_s), enp->en_maep);
336 EFSYS_PROBE1(fail1, efx_rc_t, rc);
344 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
345 efx_mae_t *maep = enp->en_maep;
347 if (encp->enc_mae_supported == B_FALSE)
350 EFSYS_KMEM_FREE(enp->en_esip, maep->em_action_rule_field_caps_size,
351 maep->em_action_rule_field_caps);
352 EFSYS_KMEM_FREE(enp->en_esip, maep->em_outer_rule_field_caps_size,
353 maep->em_outer_rule_field_caps);
354 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*maep), maep);
358 __checkReturn efx_rc_t
361 __out efx_mae_limits_t *emlp)
363 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
364 struct efx_mae_s *maep = enp->en_maep;
367 if (encp->enc_mae_supported == B_FALSE) {
372 emlp->eml_max_n_outer_prios = maep->em_max_n_outer_prios;
373 emlp->eml_max_n_action_prios = maep->em_max_n_action_prios;
374 emlp->eml_encap_types_supported = maep->em_encap_types_supported;
375 emlp->eml_encap_header_size_limit =
376 MC_CMD_MAE_ENCAP_HEADER_ALLOC_IN_HDR_DATA_MAXNUM_MCDI2;
377 emlp->eml_max_n_counters = maep->em_max_ncounters;
382 EFSYS_PROBE1(fail1, efx_rc_t, rc);
386 __checkReturn efx_rc_t
387 efx_mae_match_spec_init(
389 __in efx_mae_rule_type_t type,
391 __out efx_mae_match_spec_t **specp)
393 efx_mae_match_spec_t *spec;
397 case EFX_MAE_RULE_OUTER:
399 case EFX_MAE_RULE_ACTION:
406 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
412 spec->emms_type = type;
413 spec->emms_prio = prio;
422 EFSYS_PROBE1(fail1, efx_rc_t, rc);
427 efx_mae_match_spec_fini(
429 __in efx_mae_match_spec_t *spec)
431 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
434 /* Named identifiers which are valid indices to efx_mae_field_cap_t */
435 typedef enum efx_mae_field_cap_id_e {
436 EFX_MAE_FIELD_ID_INGRESS_MPORT_SELECTOR = MAE_FIELD_INGRESS_PORT,
437 EFX_MAE_FIELD_ID_ETHER_TYPE_BE = MAE_FIELD_ETHER_TYPE,
438 EFX_MAE_FIELD_ID_ETH_SADDR_BE = MAE_FIELD_ETH_SADDR,
439 EFX_MAE_FIELD_ID_ETH_DADDR_BE = MAE_FIELD_ETH_DADDR,
440 EFX_MAE_FIELD_ID_VLAN0_TCI_BE = MAE_FIELD_VLAN0_TCI,
441 EFX_MAE_FIELD_ID_VLAN0_PROTO_BE = MAE_FIELD_VLAN0_PROTO,
442 EFX_MAE_FIELD_ID_VLAN1_TCI_BE = MAE_FIELD_VLAN1_TCI,
443 EFX_MAE_FIELD_ID_VLAN1_PROTO_BE = MAE_FIELD_VLAN1_PROTO,
444 EFX_MAE_FIELD_ID_SRC_IP4_BE = MAE_FIELD_SRC_IP4,
445 EFX_MAE_FIELD_ID_DST_IP4_BE = MAE_FIELD_DST_IP4,
446 EFX_MAE_FIELD_ID_IP_PROTO = MAE_FIELD_IP_PROTO,
447 EFX_MAE_FIELD_ID_IP_TOS = MAE_FIELD_IP_TOS,
448 EFX_MAE_FIELD_ID_IP_TTL = MAE_FIELD_IP_TTL,
449 EFX_MAE_FIELD_ID_SRC_IP6_BE = MAE_FIELD_SRC_IP6,
450 EFX_MAE_FIELD_ID_DST_IP6_BE = MAE_FIELD_DST_IP6,
451 EFX_MAE_FIELD_ID_L4_SPORT_BE = MAE_FIELD_L4_SPORT,
452 EFX_MAE_FIELD_ID_L4_DPORT_BE = MAE_FIELD_L4_DPORT,
453 EFX_MAE_FIELD_ID_TCP_FLAGS_BE = MAE_FIELD_TCP_FLAGS,
454 EFX_MAE_FIELD_ID_ENC_ETHER_TYPE_BE = MAE_FIELD_ENC_ETHER_TYPE,
455 EFX_MAE_FIELD_ID_ENC_ETH_SADDR_BE = MAE_FIELD_ENC_ETH_SADDR,
456 EFX_MAE_FIELD_ID_ENC_ETH_DADDR_BE = MAE_FIELD_ENC_ETH_DADDR,
457 EFX_MAE_FIELD_ID_ENC_VLAN0_TCI_BE = MAE_FIELD_ENC_VLAN0_TCI,
458 EFX_MAE_FIELD_ID_ENC_VLAN0_PROTO_BE = MAE_FIELD_ENC_VLAN0_PROTO,
459 EFX_MAE_FIELD_ID_ENC_VLAN1_TCI_BE = MAE_FIELD_ENC_VLAN1_TCI,
460 EFX_MAE_FIELD_ID_ENC_VLAN1_PROTO_BE = MAE_FIELD_ENC_VLAN1_PROTO,
461 EFX_MAE_FIELD_ID_ENC_SRC_IP4_BE = MAE_FIELD_ENC_SRC_IP4,
462 EFX_MAE_FIELD_ID_ENC_DST_IP4_BE = MAE_FIELD_ENC_DST_IP4,
463 EFX_MAE_FIELD_ID_ENC_IP_PROTO = MAE_FIELD_ENC_IP_PROTO,
464 EFX_MAE_FIELD_ID_ENC_IP_TOS = MAE_FIELD_ENC_IP_TOS,
465 EFX_MAE_FIELD_ID_ENC_IP_TTL = MAE_FIELD_ENC_IP_TTL,
466 EFX_MAE_FIELD_ID_ENC_SRC_IP6_BE = MAE_FIELD_ENC_SRC_IP6,
467 EFX_MAE_FIELD_ID_ENC_DST_IP6_BE = MAE_FIELD_ENC_DST_IP6,
468 EFX_MAE_FIELD_ID_ENC_L4_SPORT_BE = MAE_FIELD_ENC_L4_SPORT,
469 EFX_MAE_FIELD_ID_ENC_L4_DPORT_BE = MAE_FIELD_ENC_L4_DPORT,
470 EFX_MAE_FIELD_ID_ENC_VNET_ID_BE = MAE_FIELD_ENC_VNET_ID,
471 EFX_MAE_FIELD_ID_OUTER_RULE_ID = MAE_FIELD_OUTER_RULE_ID,
472 EFX_MAE_FIELD_ID_HAS_OVLAN = MAE_FIELD_HAS_OVLAN,
473 EFX_MAE_FIELD_ID_HAS_IVLAN = MAE_FIELD_HAS_IVLAN,
474 EFX_MAE_FIELD_ID_ENC_HAS_OVLAN = MAE_FIELD_ENC_HAS_OVLAN,
475 EFX_MAE_FIELD_ID_ENC_HAS_IVLAN = MAE_FIELD_ENC_HAS_IVLAN,
476 EFX_MAE_FIELD_ID_RECIRC_ID = MAE_FIELD_RECIRC_ID,
478 EFX_MAE_FIELD_CAP_NIDS
479 } efx_mae_field_cap_id_t;
481 typedef enum efx_mae_field_endianness_e {
482 EFX_MAE_FIELD_LE = 0,
485 EFX_MAE_FIELD_ENDIANNESS_NTYPES
486 } efx_mae_field_endianness_t;
489 * The following structure is a means to describe an MAE field.
490 * The information in it is meant to be used internally by
491 * APIs for addressing a given field in a mask-value pairs
492 * structure and for validation purposes.
494 * A field may have an alternative one. This structure
495 * has additional members to reference the alternative
496 * field's mask. See efx_mae_match_spec_is_valid().
498 typedef struct efx_mae_mv_desc_s {
499 efx_mae_field_cap_id_t emmd_field_cap_id;
501 size_t emmd_value_size;
502 size_t emmd_value_offset;
503 size_t emmd_mask_size;
504 size_t emmd_mask_offset;
507 * Having the alternative field's mask size set to 0
508 * means that there's no alternative field specified.
510 size_t emmd_alt_mask_size;
511 size_t emmd_alt_mask_offset;
513 /* Primary field and the alternative one are of the same endianness. */
514 efx_mae_field_endianness_t emmd_endianness;
517 /* Indices to this array are provided by efx_mae_field_id_t */
518 static const efx_mae_mv_desc_t __efx_mae_action_rule_mv_desc_set[] = {
519 #define EFX_MAE_MV_DESC(_name, _endianness) \
520 [EFX_MAE_FIELD_##_name] = \
522 EFX_MAE_FIELD_ID_##_name, \
523 MAE_FIELD_MASK_VALUE_PAIRS_V2_##_name##_LEN, \
524 MAE_FIELD_MASK_VALUE_PAIRS_V2_##_name##_OFST, \
525 MAE_FIELD_MASK_VALUE_PAIRS_V2_##_name##_MASK_LEN, \
526 MAE_FIELD_MASK_VALUE_PAIRS_V2_##_name##_MASK_OFST, \
527 0, 0 /* no alternative field */, \
531 EFX_MAE_MV_DESC(INGRESS_MPORT_SELECTOR, EFX_MAE_FIELD_LE),
532 EFX_MAE_MV_DESC(ETHER_TYPE_BE, EFX_MAE_FIELD_BE),
533 EFX_MAE_MV_DESC(ETH_SADDR_BE, EFX_MAE_FIELD_BE),
534 EFX_MAE_MV_DESC(ETH_DADDR_BE, EFX_MAE_FIELD_BE),
535 EFX_MAE_MV_DESC(VLAN0_TCI_BE, EFX_MAE_FIELD_BE),
536 EFX_MAE_MV_DESC(VLAN0_PROTO_BE, EFX_MAE_FIELD_BE),
537 EFX_MAE_MV_DESC(VLAN1_TCI_BE, EFX_MAE_FIELD_BE),
538 EFX_MAE_MV_DESC(VLAN1_PROTO_BE, EFX_MAE_FIELD_BE),
539 EFX_MAE_MV_DESC(SRC_IP4_BE, EFX_MAE_FIELD_BE),
540 EFX_MAE_MV_DESC(DST_IP4_BE, EFX_MAE_FIELD_BE),
541 EFX_MAE_MV_DESC(IP_PROTO, EFX_MAE_FIELD_BE),
542 EFX_MAE_MV_DESC(IP_TOS, EFX_MAE_FIELD_BE),
543 EFX_MAE_MV_DESC(IP_TTL, EFX_MAE_FIELD_BE),
544 EFX_MAE_MV_DESC(SRC_IP6_BE, EFX_MAE_FIELD_BE),
545 EFX_MAE_MV_DESC(DST_IP6_BE, EFX_MAE_FIELD_BE),
546 EFX_MAE_MV_DESC(L4_SPORT_BE, EFX_MAE_FIELD_BE),
547 EFX_MAE_MV_DESC(L4_DPORT_BE, EFX_MAE_FIELD_BE),
548 EFX_MAE_MV_DESC(TCP_FLAGS_BE, EFX_MAE_FIELD_BE),
549 EFX_MAE_MV_DESC(ENC_VNET_ID_BE, EFX_MAE_FIELD_BE),
550 EFX_MAE_MV_DESC(OUTER_RULE_ID, EFX_MAE_FIELD_LE),
551 EFX_MAE_MV_DESC(RECIRC_ID, EFX_MAE_FIELD_LE),
553 #undef EFX_MAE_MV_DESC
556 /* Indices to this array are provided by efx_mae_field_id_t */
557 static const efx_mae_mv_desc_t __efx_mae_outer_rule_mv_desc_set[] = {
558 #define EFX_MAE_MV_DESC(_name, _endianness) \
559 [EFX_MAE_FIELD_##_name] = \
561 EFX_MAE_FIELD_ID_##_name, \
562 MAE_ENC_FIELD_PAIRS_##_name##_LEN, \
563 MAE_ENC_FIELD_PAIRS_##_name##_OFST, \
564 MAE_ENC_FIELD_PAIRS_##_name##_MASK_LEN, \
565 MAE_ENC_FIELD_PAIRS_##_name##_MASK_OFST, \
566 0, 0 /* no alternative field */, \
570 /* Same as EFX_MAE_MV_DESC(), but also indicates an alternative field. */
571 #define EFX_MAE_MV_DESC_ALT(_name, _alt_name, _endianness) \
572 [EFX_MAE_FIELD_##_name] = \
574 EFX_MAE_FIELD_ID_##_name, \
575 MAE_ENC_FIELD_PAIRS_##_name##_LEN, \
576 MAE_ENC_FIELD_PAIRS_##_name##_OFST, \
577 MAE_ENC_FIELD_PAIRS_##_name##_MASK_LEN, \
578 MAE_ENC_FIELD_PAIRS_##_name##_MASK_OFST, \
579 MAE_ENC_FIELD_PAIRS_##_alt_name##_MASK_LEN, \
580 MAE_ENC_FIELD_PAIRS_##_alt_name##_MASK_OFST, \
584 EFX_MAE_MV_DESC(INGRESS_MPORT_SELECTOR, EFX_MAE_FIELD_LE),
585 EFX_MAE_MV_DESC(ENC_ETHER_TYPE_BE, EFX_MAE_FIELD_BE),
586 EFX_MAE_MV_DESC(ENC_ETH_SADDR_BE, EFX_MAE_FIELD_BE),
587 EFX_MAE_MV_DESC(ENC_ETH_DADDR_BE, EFX_MAE_FIELD_BE),
588 EFX_MAE_MV_DESC(ENC_VLAN0_TCI_BE, EFX_MAE_FIELD_BE),
589 EFX_MAE_MV_DESC(ENC_VLAN0_PROTO_BE, EFX_MAE_FIELD_BE),
590 EFX_MAE_MV_DESC(ENC_VLAN1_TCI_BE, EFX_MAE_FIELD_BE),
591 EFX_MAE_MV_DESC(ENC_VLAN1_PROTO_BE, EFX_MAE_FIELD_BE),
592 EFX_MAE_MV_DESC_ALT(ENC_SRC_IP4_BE, ENC_SRC_IP6_BE, EFX_MAE_FIELD_BE),
593 EFX_MAE_MV_DESC_ALT(ENC_DST_IP4_BE, ENC_DST_IP6_BE, EFX_MAE_FIELD_BE),
594 EFX_MAE_MV_DESC(ENC_IP_PROTO, EFX_MAE_FIELD_BE),
595 EFX_MAE_MV_DESC(ENC_IP_TOS, EFX_MAE_FIELD_BE),
596 EFX_MAE_MV_DESC(ENC_IP_TTL, EFX_MAE_FIELD_BE),
597 EFX_MAE_MV_DESC_ALT(ENC_SRC_IP6_BE, ENC_SRC_IP4_BE, EFX_MAE_FIELD_BE),
598 EFX_MAE_MV_DESC_ALT(ENC_DST_IP6_BE, ENC_DST_IP4_BE, EFX_MAE_FIELD_BE),
599 EFX_MAE_MV_DESC(ENC_L4_SPORT_BE, EFX_MAE_FIELD_BE),
600 EFX_MAE_MV_DESC(ENC_L4_DPORT_BE, EFX_MAE_FIELD_BE),
602 #undef EFX_MAE_MV_DESC_ALT
603 #undef EFX_MAE_MV_DESC
607 * The following structure is a means to describe an MAE bit.
608 * The information in it is meant to be used internally by
609 * APIs for addressing a given flag in a mask-value pairs
610 * structure and for validation purposes.
612 typedef struct efx_mae_mv_bit_desc_s {
614 * Arrays using this struct are indexed by field IDs.
615 * Fields which aren't meant to be referenced by these
616 * arrays comprise gaps (invalid entries). Below field
617 * helps to identify such entries.
619 boolean_t emmbd_entry_is_valid;
620 efx_mae_field_cap_id_t emmbd_bit_cap_id;
621 size_t emmbd_value_ofst;
622 unsigned int emmbd_value_lbn;
623 size_t emmbd_mask_ofst;
624 unsigned int emmbd_mask_lbn;
625 } efx_mae_mv_bit_desc_t;
627 static const efx_mae_mv_bit_desc_t __efx_mae_outer_rule_mv_bit_desc_set[] = {
628 #define EFX_MAE_MV_BIT_DESC(_name) \
629 [EFX_MAE_FIELD_##_name] = \
632 EFX_MAE_FIELD_ID_##_name, \
633 MAE_ENC_FIELD_PAIRS_##_name##_OFST, \
634 MAE_ENC_FIELD_PAIRS_##_name##_LBN, \
635 MAE_ENC_FIELD_PAIRS_##_name##_MASK_OFST, \
636 MAE_ENC_FIELD_PAIRS_##_name##_MASK_LBN, \
639 EFX_MAE_MV_BIT_DESC(ENC_HAS_OVLAN),
640 EFX_MAE_MV_BIT_DESC(ENC_HAS_IVLAN),
642 #undef EFX_MAE_MV_BIT_DESC
645 static const efx_mae_mv_bit_desc_t __efx_mae_action_rule_mv_bit_desc_set[] = {
646 #define EFX_MAE_MV_BIT_DESC(_name) \
647 [EFX_MAE_FIELD_##_name] = \
650 EFX_MAE_FIELD_ID_##_name, \
651 MAE_FIELD_MASK_VALUE_PAIRS_V2_FLAGS_OFST, \
652 MAE_FIELD_MASK_VALUE_PAIRS_V2_##_name##_LBN, \
653 MAE_FIELD_MASK_VALUE_PAIRS_V2_FLAGS_MASK_OFST, \
654 MAE_FIELD_MASK_VALUE_PAIRS_V2_##_name##_LBN, \
657 EFX_MAE_MV_BIT_DESC(HAS_OVLAN),
658 EFX_MAE_MV_BIT_DESC(HAS_IVLAN),
659 EFX_MAE_MV_BIT_DESC(ENC_HAS_OVLAN),
660 EFX_MAE_MV_BIT_DESC(ENC_HAS_IVLAN),
662 #undef EFX_MAE_MV_BIT_DESC
665 __checkReturn efx_rc_t
666 efx_mae_mport_invalid(
667 __out efx_mport_sel_t *mportp)
672 if (mportp == NULL) {
677 EFX_POPULATE_DWORD_1(dword,
678 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_INVALID);
680 memset(mportp, 0, sizeof (*mportp));
681 mportp->sel = dword.ed_u32[0];
686 EFSYS_PROBE1(fail1, efx_rc_t, rc);
690 __checkReturn efx_rc_t
691 efx_mae_mport_by_phy_port(
692 __in uint32_t phy_port,
693 __out efx_mport_sel_t *mportp)
698 if (phy_port > EFX_MASK32(MAE_MPORT_SELECTOR_PPORT_ID)) {
703 EFX_POPULATE_DWORD_2(dword,
704 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_PPORT,
705 MAE_MPORT_SELECTOR_PPORT_ID, phy_port);
707 memset(mportp, 0, sizeof (*mportp));
709 * The constructed DWORD is little-endian,
710 * but the resulting value is meant to be
711 * passed to MCDIs, where it will undergo
712 * host-order to little endian conversion.
714 mportp->sel = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
719 EFSYS_PROBE1(fail1, efx_rc_t, rc);
723 __checkReturn efx_rc_t
724 efx_mae_mport_by_pcie_function(
727 __out efx_mport_sel_t *mportp)
732 rc = efx_mae_mport_by_pcie_mh_function(EFX_PCIE_INTERFACE_CALLER,
740 EFSYS_PROBE1(fail1, efx_rc_t, rc);
744 static __checkReturn efx_rc_t
745 efx_mae_intf_to_selector(
746 __in efx_pcie_interface_t intf,
747 __out uint32_t *selector_intfp)
752 case EFX_PCIE_INTERFACE_HOST_PRIMARY:
753 EFX_STATIC_ASSERT(MAE_MPORT_SELECTOR_HOST_PRIMARY <=
754 EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_INTF_ID));
755 *selector_intfp = MAE_MPORT_SELECTOR_HOST_PRIMARY;
757 case EFX_PCIE_INTERFACE_NIC_EMBEDDED:
758 EFX_STATIC_ASSERT(MAE_MPORT_SELECTOR_NIC_EMBEDDED <=
759 EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_INTF_ID));
760 *selector_intfp = MAE_MPORT_SELECTOR_NIC_EMBEDDED;
762 case EFX_PCIE_INTERFACE_CALLER:
763 EFX_STATIC_ASSERT(MAE_MPORT_SELECTOR_CALLER_INTF <=
764 EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_INTF_ID));
765 *selector_intfp = MAE_MPORT_SELECTOR_CALLER_INTF;
775 EFSYS_PROBE1(fail1, efx_rc_t, rc);
779 __checkReturn efx_rc_t
780 efx_mae_mport_by_pcie_mh_function(
781 __in efx_pcie_interface_t intf,
784 __out efx_mport_sel_t *mportp)
786 uint32_t selector_intf;
790 EFX_STATIC_ASSERT(EFX_PCI_VF_INVALID ==
791 MAE_MPORT_SELECTOR_FUNC_VF_ID_NULL);
793 rc = efx_mae_intf_to_selector(intf, &selector_intf);
797 if (pf > EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_MH_PF_ID)) {
802 if (vf > EFX_MASK32(MAE_MPORT_SELECTOR_FUNC_VF_ID)) {
808 EFX_POPULATE_DWORD_4(dword,
809 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_MH_FUNC,
810 MAE_MPORT_SELECTOR_FUNC_INTF_ID, selector_intf,
811 MAE_MPORT_SELECTOR_FUNC_MH_PF_ID, pf,
812 MAE_MPORT_SELECTOR_FUNC_VF_ID, vf);
814 memset(mportp, 0, sizeof (*mportp));
815 mportp->sel = dword.ed_u32[0];
824 EFSYS_PROBE1(fail1, efx_rc_t, rc);
828 static __checkReturn efx_rc_t
829 efx_mcdi_mae_mport_lookup(
831 __in const efx_mport_sel_t *mport_selectorp,
832 __out efx_mport_id_t *mport_idp)
835 EFX_MCDI_DECLARE_BUF(payload,
836 MC_CMD_MAE_MPORT_LOOKUP_IN_LEN,
837 MC_CMD_MAE_MPORT_LOOKUP_OUT_LEN);
840 req.emr_cmd = MC_CMD_MAE_MPORT_LOOKUP;
841 req.emr_in_buf = payload;
842 req.emr_in_length = MC_CMD_MAE_MPORT_LOOKUP_IN_LEN;
843 req.emr_out_buf = payload;
844 req.emr_out_length = MC_CMD_MAE_MPORT_LOOKUP_OUT_LEN;
846 MCDI_IN_SET_DWORD(req, MAE_MPORT_LOOKUP_IN_MPORT_SELECTOR,
847 mport_selectorp->sel);
849 efx_mcdi_execute(enp, &req);
851 if (req.emr_rc != 0) {
856 mport_idp->id = MCDI_OUT_DWORD(req, MAE_MPORT_LOOKUP_OUT_MPORT_ID);
861 EFSYS_PROBE1(fail1, efx_rc_t, rc);
865 __checkReturn efx_rc_t
866 efx_mae_mport_id_by_selector(
868 __in const efx_mport_sel_t *mport_selectorp,
869 __out efx_mport_id_t *mport_idp)
871 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
874 if (encp->enc_mae_supported == B_FALSE) {
879 rc = efx_mcdi_mae_mport_lookup(enp, mport_selectorp, mport_idp);
888 EFSYS_PROBE1(fail1, efx_rc_t, rc);
892 __checkReturn efx_rc_t
893 efx_mae_match_spec_recirc_id_set(
894 __in efx_mae_match_spec_t *spec,
895 __in uint8_t recirc_id)
897 uint8_t full_mask = UINT8_MAX;
902 vp = (const uint8_t *)&recirc_id;
903 mp = (const uint8_t *)&full_mask;
905 rc = efx_mae_match_spec_field_set(spec, EFX_MAE_FIELD_RECIRC_ID,
906 sizeof (recirc_id), vp,
907 sizeof (full_mask), mp);
914 EFSYS_PROBE1(fail1, efx_rc_t, rc);
918 __checkReturn efx_rc_t
920 __in const efx_mport_id_t *mport_idp,
921 __out efx_mport_sel_t *mportp)
925 EFX_POPULATE_DWORD_2(dword,
926 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_MPORT_ID,
927 MAE_MPORT_SELECTOR_MPORT_ID, mport_idp->id);
929 memset(mportp, 0, sizeof (*mportp));
930 mportp->sel = __LE_TO_CPU_32(dword.ed_u32[0]);
935 __checkReturn efx_rc_t
936 efx_mae_match_spec_field_set(
937 __in efx_mae_match_spec_t *spec,
938 __in efx_mae_field_id_t field_id,
939 __in size_t value_size,
940 __in_bcount(value_size) const uint8_t *value,
941 __in size_t mask_size,
942 __in_bcount(mask_size) const uint8_t *mask)
944 const efx_mae_mv_desc_t *descp;
945 unsigned int desc_set_nentries;
949 switch (spec->emms_type) {
950 case EFX_MAE_RULE_OUTER:
952 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_desc_set);
953 descp = &__efx_mae_outer_rule_mv_desc_set[field_id];
954 mvp = spec->emms_mask_value_pairs.outer;
956 case EFX_MAE_RULE_ACTION:
958 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
959 descp = &__efx_mae_action_rule_mv_desc_set[field_id];
960 mvp = spec->emms_mask_value_pairs.action;
967 if ((unsigned int)field_id >= desc_set_nentries) {
972 if (descp->emmd_mask_size == 0) {
973 /* The ID points to a gap in the array of field descriptors. */
978 if (value_size != descp->emmd_value_size) {
983 if (mask_size != descp->emmd_mask_size) {
988 if (descp->emmd_endianness == EFX_MAE_FIELD_BE) {
992 * The mask/value are in network (big endian) order.
993 * The MCDI request field is also big endian.
996 EFSYS_ASSERT3U(value_size, ==, mask_size);
998 for (i = 0; i < value_size; ++i) {
999 uint8_t *v_bytep = mvp + descp->emmd_value_offset + i;
1000 uint8_t *m_bytep = mvp + descp->emmd_mask_offset + i;
1003 * Apply the mask (which may be all-zeros) to the value.
1005 * If this API is provided with some value to set for a
1006 * given field in one specification and with some other
1007 * value to set for this field in another specification,
1008 * then, if the two masks are all-zeros, the field will
1009 * avoid being counted as a mismatch when comparing the
1010 * specifications using efx_mae_match_specs_equal() API.
1012 *v_bytep = value[i] & mask[i];
1019 * The mask/value are in host byte order.
1020 * The MCDI request field is little endian.
1022 switch (value_size) {
1024 EFX_POPULATE_DWORD_1(dword,
1025 EFX_DWORD_0, *(const uint32_t *)value);
1027 memcpy(mvp + descp->emmd_value_offset,
1028 &dword, sizeof (dword));
1031 EFSYS_ASSERT(B_FALSE);
1034 switch (mask_size) {
1036 EFX_POPULATE_DWORD_1(dword,
1037 EFX_DWORD_0, *(const uint32_t *)mask);
1039 memcpy(mvp + descp->emmd_mask_offset,
1040 &dword, sizeof (dword));
1043 EFSYS_ASSERT(B_FALSE);
1058 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1062 __checkReturn efx_rc_t
1063 efx_mae_match_spec_bit_set(
1064 __in efx_mae_match_spec_t *spec,
1065 __in efx_mae_field_id_t field_id,
1066 __in boolean_t value)
1068 const efx_mae_mv_bit_desc_t *bit_descp;
1069 unsigned int bit_desc_set_nentries;
1070 unsigned int byte_idx;
1071 unsigned int bit_idx;
1075 switch (spec->emms_type) {
1076 case EFX_MAE_RULE_OUTER:
1077 bit_desc_set_nentries =
1078 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_bit_desc_set);
1079 bit_descp = &__efx_mae_outer_rule_mv_bit_desc_set[field_id];
1080 mvp = spec->emms_mask_value_pairs.outer;
1082 case EFX_MAE_RULE_ACTION:
1083 bit_desc_set_nentries =
1084 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_bit_desc_set);
1085 bit_descp = &__efx_mae_action_rule_mv_bit_desc_set[field_id];
1086 mvp = spec->emms_mask_value_pairs.action;
1093 if ((unsigned int)field_id >= bit_desc_set_nentries) {
1098 if (bit_descp->emmbd_entry_is_valid == B_FALSE) {
1103 byte_idx = bit_descp->emmbd_value_ofst + bit_descp->emmbd_value_lbn / 8;
1104 bit_idx = bit_descp->emmbd_value_lbn % 8;
1106 if (value != B_FALSE)
1107 mvp[byte_idx] |= (1U << bit_idx);
1109 mvp[byte_idx] &= ~(1U << bit_idx);
1111 byte_idx = bit_descp->emmbd_mask_ofst + bit_descp->emmbd_mask_lbn / 8;
1112 bit_idx = bit_descp->emmbd_mask_lbn % 8;
1113 mvp[byte_idx] |= (1U << bit_idx);
1122 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1126 __checkReturn efx_rc_t
1127 efx_mae_match_spec_mport_set(
1128 __in efx_mae_match_spec_t *spec,
1129 __in const efx_mport_sel_t *valuep,
1130 __in_opt const efx_mport_sel_t *maskp)
1132 uint32_t full_mask = UINT32_MAX;
1137 if (valuep == NULL) {
1142 vp = (const uint8_t *)&valuep->sel;
1144 mp = (const uint8_t *)&maskp->sel;
1146 mp = (const uint8_t *)&full_mask;
1148 rc = efx_mae_match_spec_field_set(spec,
1149 EFX_MAE_FIELD_INGRESS_MPORT_SELECTOR,
1150 sizeof (valuep->sel), vp, sizeof (maskp->sel), mp);
1159 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1163 __checkReturn boolean_t
1164 efx_mae_match_specs_equal(
1165 __in const efx_mae_match_spec_t *left,
1166 __in const efx_mae_match_spec_t *right)
1168 return ((memcmp(left, right, sizeof (*left)) == 0) ? B_TRUE : B_FALSE);
1171 #define EFX_MASK_BIT_IS_SET(_mask, _mask_page_nbits, _bit) \
1172 ((_mask)[(_bit) / (_mask_page_nbits)] & \
1173 (1ULL << ((_bit) & ((_mask_page_nbits) - 1))))
1177 __in size_t mask_nbytes,
1178 __in_bcount(mask_nbytes) const uint8_t *maskp)
1180 boolean_t prev_bit_is_set = B_TRUE;
1183 for (i = 0; i < 8 * mask_nbytes; ++i) {
1184 boolean_t bit_is_set = EFX_MASK_BIT_IS_SET(maskp, 8, i);
1186 if (!prev_bit_is_set && bit_is_set)
1189 prev_bit_is_set = bit_is_set;
1196 efx_mask_is_all_ones(
1197 __in size_t mask_nbytes,
1198 __in_bcount(mask_nbytes) const uint8_t *maskp)
1203 for (i = 0; i < mask_nbytes; ++i)
1206 return (t == (uint8_t)(~0));
1210 efx_mask_is_all_zeros(
1211 __in size_t mask_nbytes,
1212 __in_bcount(mask_nbytes) const uint8_t *maskp)
1217 for (i = 0; i < mask_nbytes; ++i)
1223 __checkReturn boolean_t
1224 efx_mae_match_spec_is_valid(
1225 __in efx_nic_t *enp,
1226 __in const efx_mae_match_spec_t *spec)
1228 efx_mae_t *maep = enp->en_maep;
1229 unsigned int field_ncaps = maep->em_max_nfields;
1230 const efx_mae_field_cap_t *field_caps;
1231 const efx_mae_mv_desc_t *desc_setp;
1232 unsigned int desc_set_nentries;
1233 const efx_mae_mv_bit_desc_t *bit_desc_setp;
1234 unsigned int bit_desc_set_nentries;
1235 boolean_t is_valid = B_TRUE;
1236 efx_mae_field_id_t field_id;
1239 switch (spec->emms_type) {
1240 case EFX_MAE_RULE_OUTER:
1241 field_caps = maep->em_outer_rule_field_caps;
1242 desc_setp = __efx_mae_outer_rule_mv_desc_set;
1244 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_desc_set);
1245 bit_desc_setp = __efx_mae_outer_rule_mv_bit_desc_set;
1246 bit_desc_set_nentries =
1247 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_bit_desc_set);
1248 mvp = spec->emms_mask_value_pairs.outer;
1250 case EFX_MAE_RULE_ACTION:
1251 field_caps = maep->em_action_rule_field_caps;
1252 desc_setp = __efx_mae_action_rule_mv_desc_set;
1254 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
1255 bit_desc_setp = __efx_mae_action_rule_mv_bit_desc_set;
1256 bit_desc_set_nentries =
1257 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_bit_desc_set);
1258 mvp = spec->emms_mask_value_pairs.action;
1264 if (field_caps == NULL)
1267 for (field_id = 0; (unsigned int)field_id < desc_set_nentries;
1269 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
1270 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
1271 const uint8_t *alt_m_buf = mvp + descp->emmd_alt_mask_offset;
1272 const uint8_t *m_buf = mvp + descp->emmd_mask_offset;
1273 size_t alt_m_size = descp->emmd_alt_mask_size;
1274 size_t m_size = descp->emmd_mask_size;
1277 continue; /* Skip array gap */
1279 if ((unsigned int)field_cap_id >= field_ncaps) {
1281 * The FW has not reported capability status for
1282 * this field. Make sure that its mask is zeroed.
1284 is_valid = efx_mask_is_all_zeros(m_size, m_buf);
1285 if (is_valid != B_FALSE)
1291 switch (field_caps[field_cap_id].emfc_support) {
1292 case MAE_FIELD_SUPPORTED_MATCH_MASK:
1295 case MAE_FIELD_SUPPORTED_MATCH_PREFIX:
1296 is_valid = efx_mask_is_prefix(m_size, m_buf);
1298 case MAE_FIELD_SUPPORTED_MATCH_OPTIONAL:
1299 is_valid = (efx_mask_is_all_ones(m_size, m_buf) ||
1300 efx_mask_is_all_zeros(m_size, m_buf));
1302 case MAE_FIELD_SUPPORTED_MATCH_ALWAYS:
1303 is_valid = efx_mask_is_all_ones(m_size, m_buf);
1305 if ((is_valid == B_FALSE) && (alt_m_size != 0)) {
1307 * This field has an alternative one. The FW
1308 * reports ALWAYS for both implying that one
1309 * of them is required to have all-ones mask.
1311 * The primary field's mask is incorrect; go
1312 * on to check that of the alternative field.
1314 is_valid = efx_mask_is_all_ones(alt_m_size,
1318 case MAE_FIELD_SUPPORTED_MATCH_NEVER:
1319 case MAE_FIELD_UNSUPPORTED:
1321 is_valid = efx_mask_is_all_zeros(m_size, m_buf);
1325 if (is_valid == B_FALSE)
1329 for (field_id = 0; (unsigned int)field_id < bit_desc_set_nentries;
1331 const efx_mae_mv_bit_desc_t *bit_descp =
1332 &bit_desc_setp[field_id];
1333 unsigned int byte_idx =
1334 bit_descp->emmbd_mask_ofst +
1335 bit_descp->emmbd_mask_lbn / 8;
1336 unsigned int bit_idx =
1337 bit_descp->emmbd_mask_lbn % 8;
1338 efx_mae_field_cap_id_t bit_cap_id =
1339 bit_descp->emmbd_bit_cap_id;
1341 if (bit_descp->emmbd_entry_is_valid == B_FALSE)
1342 continue; /* Skip array gap */
1344 if ((unsigned int)bit_cap_id >= field_ncaps) {
1345 /* No capability for this bit = unsupported. */
1346 is_valid = ((mvp[byte_idx] & (1U << bit_idx)) == 0);
1347 if (is_valid == B_FALSE)
1353 switch (field_caps[bit_cap_id].emfc_support) {
1354 case MAE_FIELD_SUPPORTED_MATCH_OPTIONAL:
1357 case MAE_FIELD_SUPPORTED_MATCH_ALWAYS:
1358 is_valid = ((mvp[byte_idx] & (1U << bit_idx)) != 0);
1360 case MAE_FIELD_SUPPORTED_MATCH_NEVER:
1361 case MAE_FIELD_UNSUPPORTED:
1363 is_valid = ((mvp[byte_idx] & (1U << bit_idx)) == 0);
1367 if (is_valid == B_FALSE)
1374 __checkReturn efx_rc_t
1375 efx_mae_action_set_spec_init(
1376 __in efx_nic_t *enp,
1377 __out efx_mae_actions_t **specp)
1379 efx_mae_actions_t *spec;
1382 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
1388 spec->ema_rsrc.emar_eh_id.id = EFX_MAE_RSRC_ID_INVALID;
1389 spec->ema_rsrc.emar_counter_id.id = EFX_MAE_RSRC_ID_INVALID;
1396 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1401 efx_mae_action_set_spec_fini(
1402 __in efx_nic_t *enp,
1403 __in efx_mae_actions_t *spec)
1405 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
1408 static __checkReturn efx_rc_t
1409 efx_mae_action_set_add_decap(
1410 __in efx_mae_actions_t *spec,
1411 __in size_t arg_size,
1412 __in_bcount(arg_size) const uint8_t *arg)
1416 _NOTE(ARGUNUSED(spec))
1418 if (arg_size != 0) {
1428 /* This action does not have any arguments, so do nothing here. */
1435 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1439 static __checkReturn efx_rc_t
1440 efx_mae_action_set_add_vlan_pop(
1441 __in efx_mae_actions_t *spec,
1442 __in size_t arg_size,
1443 __in_bcount(arg_size) const uint8_t *arg)
1447 if (arg_size != 0) {
1457 if (spec->ema_n_vlan_tags_to_pop == EFX_MAE_VLAN_POP_MAX_NTAGS) {
1462 ++spec->ema_n_vlan_tags_to_pop;
1471 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1475 static __checkReturn efx_rc_t
1476 efx_mae_action_set_add_vlan_push(
1477 __in efx_mae_actions_t *spec,
1478 __in size_t arg_size,
1479 __in_bcount(arg_size) const uint8_t *arg)
1481 unsigned int n_tags = spec->ema_n_vlan_tags_to_push;
1484 if (arg_size != sizeof (*spec->ema_vlan_push_descs)) {
1494 if (n_tags == EFX_MAE_VLAN_PUSH_MAX_NTAGS) {
1499 memcpy(&spec->ema_vlan_push_descs[n_tags], arg, arg_size);
1500 ++(spec->ema_n_vlan_tags_to_push);
1509 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1513 static __checkReturn efx_rc_t
1514 efx_mae_action_set_add_encap(
1515 __in efx_mae_actions_t *spec,
1516 __in size_t arg_size,
1517 __in_bcount(arg_size) const uint8_t *arg)
1522 * Adding this specific action to an action set spec and setting encap.
1523 * header ID in the spec are two individual steps. This design allows
1524 * the client driver to avoid encap. header allocation when it simply
1525 * needs to check the order of actions submitted by user ("validate"),
1526 * without actually allocating an action set and inserting a rule.
1528 * In order to fill in the encap. header ID, the caller is supposed to
1529 * invoke efx_mae_action_set_fill_in_eh_id(). If they do not do that,
1530 * efx_mae_action_set_alloc() invocation will throw an error.
1532 * For now, no more work is supposed to be done.
1535 if (arg_size != 0) {
1550 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1554 static __checkReturn efx_rc_t
1555 efx_mae_action_set_add_count(
1556 __in efx_mae_actions_t *spec,
1557 __in size_t arg_size,
1558 __in_bcount(arg_size) const uint8_t *arg)
1562 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
1563 MC_CMD_MAE_COUNTER_ALLOC_OUT_COUNTER_ID_NULL);
1566 * Preparing an action set spec to update a counter requires
1567 * two steps: first add this action to the action spec, and then
1568 * add the counter ID to the spec. This allows validity checking
1569 * and resource allocation to be done separately.
1571 * In order to fill in the counter ID, the caller is supposed to invoke
1572 * efx_mae_action_set_fill_in_counter_id(). If they do not do that,
1573 * efx_mae_action_set_alloc() invocation will throw an error.
1575 * For now, no arguments are supposed to be handled.
1578 if (arg_size != 0) {
1588 ++(spec->ema_n_count_actions);
1595 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1599 static __checkReturn efx_rc_t
1600 efx_mae_action_set_add_flag(
1601 __in efx_mae_actions_t *spec,
1602 __in size_t arg_size,
1603 __in_bcount(arg_size) const uint8_t *arg)
1607 _NOTE(ARGUNUSED(spec))
1609 if (arg_size != 0) {
1619 /* This action does not have any arguments, so do nothing here. */
1626 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1630 static __checkReturn efx_rc_t
1631 efx_mae_action_set_add_mark(
1632 __in efx_mae_actions_t *spec,
1633 __in size_t arg_size,
1634 __in_bcount(arg_size) const uint8_t *arg)
1638 if (arg_size != sizeof (spec->ema_mark_value)) {
1648 memcpy(&spec->ema_mark_value, arg, arg_size);
1655 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1659 static __checkReturn efx_rc_t
1660 efx_mae_action_set_add_deliver(
1661 __in efx_mae_actions_t *spec,
1662 __in size_t arg_size,
1663 __in_bcount(arg_size) const uint8_t *arg)
1667 if (arg_size != sizeof (spec->ema_deliver_mport)) {
1677 memcpy(&spec->ema_deliver_mport, arg, arg_size);
1684 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1688 typedef struct efx_mae_action_desc_s {
1689 /* Action specific handler */
1690 efx_rc_t (*emad_add)(efx_mae_actions_t *,
1691 size_t, const uint8_t *);
1692 } efx_mae_action_desc_t;
1694 static const efx_mae_action_desc_t efx_mae_actions[EFX_MAE_NACTIONS] = {
1695 [EFX_MAE_ACTION_DECAP] = {
1696 .emad_add = efx_mae_action_set_add_decap
1698 [EFX_MAE_ACTION_VLAN_POP] = {
1699 .emad_add = efx_mae_action_set_add_vlan_pop
1701 [EFX_MAE_ACTION_VLAN_PUSH] = {
1702 .emad_add = efx_mae_action_set_add_vlan_push
1704 [EFX_MAE_ACTION_ENCAP] = {
1705 .emad_add = efx_mae_action_set_add_encap
1707 [EFX_MAE_ACTION_COUNT] = {
1708 .emad_add = efx_mae_action_set_add_count
1710 [EFX_MAE_ACTION_FLAG] = {
1711 .emad_add = efx_mae_action_set_add_flag
1713 [EFX_MAE_ACTION_MARK] = {
1714 .emad_add = efx_mae_action_set_add_mark
1716 [EFX_MAE_ACTION_DELIVER] = {
1717 .emad_add = efx_mae_action_set_add_deliver
1721 static const uint32_t efx_mae_action_ordered_map =
1722 (1U << EFX_MAE_ACTION_DECAP) |
1723 (1U << EFX_MAE_ACTION_VLAN_POP) |
1724 (1U << EFX_MAE_ACTION_VLAN_PUSH) |
1726 * HW will conduct action COUNT after
1727 * the matching packet has been modified by
1728 * length-affecting actions except for ENCAP.
1730 (1U << EFX_MAE_ACTION_COUNT) |
1731 (1U << EFX_MAE_ACTION_ENCAP) |
1732 (1U << EFX_MAE_ACTION_FLAG) |
1733 (1U << EFX_MAE_ACTION_MARK) |
1734 (1U << EFX_MAE_ACTION_DELIVER);
1737 * These actions must not be added after DELIVER, but
1738 * they can have any place among the rest of
1739 * strictly ordered actions.
1741 static const uint32_t efx_mae_action_nonstrict_map =
1742 (1U << EFX_MAE_ACTION_COUNT) |
1743 (1U << EFX_MAE_ACTION_FLAG) |
1744 (1U << EFX_MAE_ACTION_MARK);
1746 static const uint32_t efx_mae_action_repeat_map =
1747 (1U << EFX_MAE_ACTION_VLAN_POP) |
1748 (1U << EFX_MAE_ACTION_VLAN_PUSH) |
1749 (1U << EFX_MAE_ACTION_COUNT);
1752 * Add an action to an action set.
1754 * This has to be invoked in the desired action order.
1755 * An out-of-order action request will be turned down.
1757 static __checkReturn efx_rc_t
1758 efx_mae_action_set_spec_populate(
1759 __in efx_mae_actions_t *spec,
1760 __in efx_mae_action_t type,
1761 __in size_t arg_size,
1762 __in_bcount(arg_size) const uint8_t *arg)
1764 uint32_t action_mask;
1767 EFX_STATIC_ASSERT(EFX_MAE_NACTIONS <=
1768 (sizeof (efx_mae_action_ordered_map) * 8));
1769 EFX_STATIC_ASSERT(EFX_MAE_NACTIONS <=
1770 (sizeof (efx_mae_action_repeat_map) * 8));
1772 EFX_STATIC_ASSERT(EFX_MAE_ACTION_DELIVER + 1 == EFX_MAE_NACTIONS);
1773 EFX_STATIC_ASSERT(EFX_MAE_ACTION_FLAG + 1 == EFX_MAE_ACTION_MARK);
1774 EFX_STATIC_ASSERT(EFX_MAE_ACTION_MARK + 1 == EFX_MAE_ACTION_DELIVER);
1776 if (type >= EFX_ARRAY_SIZE(efx_mae_actions)) {
1781 action_mask = (1U << type);
1783 if ((spec->ema_actions & action_mask) != 0) {
1784 /* The action set already contains this action. */
1785 if ((efx_mae_action_repeat_map & action_mask) == 0) {
1786 /* Cannot add another non-repeatable action. */
1792 if ((efx_mae_action_ordered_map & action_mask) != 0) {
1793 uint32_t strict_ordered_map =
1794 efx_mae_action_ordered_map & ~efx_mae_action_nonstrict_map;
1795 uint32_t later_actions_mask =
1796 strict_ordered_map & ~(action_mask | (action_mask - 1));
1798 if ((spec->ema_actions & later_actions_mask) != 0) {
1799 /* Cannot add an action after later ordered actions. */
1805 if (efx_mae_actions[type].emad_add != NULL) {
1806 rc = efx_mae_actions[type].emad_add(spec, arg_size, arg);
1811 spec->ema_actions |= action_mask;
1822 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1826 __checkReturn efx_rc_t
1827 efx_mae_action_set_populate_decap(
1828 __in efx_mae_actions_t *spec)
1830 return (efx_mae_action_set_spec_populate(spec,
1831 EFX_MAE_ACTION_DECAP, 0, NULL));
1834 __checkReturn efx_rc_t
1835 efx_mae_action_set_populate_vlan_pop(
1836 __in efx_mae_actions_t *spec)
1838 return (efx_mae_action_set_spec_populate(spec,
1839 EFX_MAE_ACTION_VLAN_POP, 0, NULL));
1842 __checkReturn efx_rc_t
1843 efx_mae_action_set_populate_vlan_push(
1844 __in efx_mae_actions_t *spec,
1845 __in uint16_t tpid_be,
1846 __in uint16_t tci_be)
1848 efx_mae_action_vlan_push_t action;
1849 const uint8_t *arg = (const uint8_t *)&action;
1851 action.emavp_tpid_be = tpid_be;
1852 action.emavp_tci_be = tci_be;
1854 return (efx_mae_action_set_spec_populate(spec,
1855 EFX_MAE_ACTION_VLAN_PUSH, sizeof (action), arg));
1858 __checkReturn efx_rc_t
1859 efx_mae_action_set_populate_encap(
1860 __in efx_mae_actions_t *spec)
1863 * There is no argument to pass encap. header ID, thus, one does not
1864 * need to allocate an encap. header while parsing application input.
1865 * This is useful since building an action set may be done simply to
1866 * validate a rule, whilst resource allocation usually consumes time.
1868 return (efx_mae_action_set_spec_populate(spec,
1869 EFX_MAE_ACTION_ENCAP, 0, NULL));
1872 __checkReturn efx_rc_t
1873 efx_mae_action_set_populate_count(
1874 __in efx_mae_actions_t *spec)
1877 * There is no argument to pass counter ID, thus, one does not
1878 * need to allocate a counter while parsing application input.
1879 * This is useful since building an action set may be done simply to
1880 * validate a rule, whilst resource allocation usually consumes time.
1882 return (efx_mae_action_set_spec_populate(spec,
1883 EFX_MAE_ACTION_COUNT, 0, NULL));
1886 __checkReturn efx_rc_t
1887 efx_mae_action_set_populate_flag(
1888 __in efx_mae_actions_t *spec)
1890 return (efx_mae_action_set_spec_populate(spec,
1891 EFX_MAE_ACTION_FLAG, 0, NULL));
1894 __checkReturn efx_rc_t
1895 efx_mae_action_set_populate_mark(
1896 __in efx_mae_actions_t *spec,
1897 __in uint32_t mark_value)
1899 const uint8_t *arg = (const uint8_t *)&mark_value;
1901 return (efx_mae_action_set_spec_populate(spec,
1902 EFX_MAE_ACTION_MARK, sizeof (mark_value), arg));
1905 __checkReturn efx_rc_t
1906 efx_mae_action_set_populate_deliver(
1907 __in efx_mae_actions_t *spec,
1908 __in const efx_mport_sel_t *mportp)
1913 if (mportp == NULL) {
1918 arg = (const uint8_t *)&mportp->sel;
1920 return (efx_mae_action_set_spec_populate(spec,
1921 EFX_MAE_ACTION_DELIVER, sizeof (mportp->sel), arg));
1924 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1928 __checkReturn efx_rc_t
1929 efx_mae_action_set_populate_drop(
1930 __in efx_mae_actions_t *spec)
1932 efx_mport_sel_t mport;
1936 EFX_POPULATE_DWORD_1(dword,
1937 MAE_MPORT_SELECTOR_FLAT, MAE_MPORT_SELECTOR_NULL);
1940 * The constructed DWORD is little-endian,
1941 * but the resulting value is meant to be
1942 * passed to MCDIs, where it will undergo
1943 * host-order to little endian conversion.
1945 mport.sel = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
1947 arg = (const uint8_t *)&mport.sel;
1949 return (efx_mae_action_set_spec_populate(spec,
1950 EFX_MAE_ACTION_DELIVER, sizeof (mport.sel), arg));
1953 __checkReturn boolean_t
1954 efx_mae_action_set_specs_equal(
1955 __in const efx_mae_actions_t *left,
1956 __in const efx_mae_actions_t *right)
1958 size_t cmp_size = EFX_FIELD_OFFSET(efx_mae_actions_t, ema_rsrc);
1961 * An action set specification consists of two parts. The first part
1962 * indicates what actions are included in the action set, as well as
1963 * extra quantitative values (in example, the number of VLAN tags to
1964 * push). The second part comprises resource IDs used by the actions.
1966 * A resource, in example, a counter, is allocated from the hardware
1967 * by the client, and it's the client who is responsible for keeping
1968 * track of allocated resources and comparing resource IDs if needed.
1970 * In this API, don't compare resource IDs in the two specifications.
1973 return ((memcmp(left, right, cmp_size) == 0) ? B_TRUE : B_FALSE);
1976 __checkReturn efx_rc_t
1977 efx_mae_match_specs_class_cmp(
1978 __in efx_nic_t *enp,
1979 __in const efx_mae_match_spec_t *left,
1980 __in const efx_mae_match_spec_t *right,
1981 __out boolean_t *have_same_classp)
1983 efx_mae_t *maep = enp->en_maep;
1984 unsigned int field_ncaps = maep->em_max_nfields;
1985 const efx_mae_field_cap_t *field_caps;
1986 const efx_mae_mv_desc_t *desc_setp;
1987 unsigned int desc_set_nentries;
1988 const efx_mae_mv_bit_desc_t *bit_desc_setp;
1989 unsigned int bit_desc_set_nentries;
1990 boolean_t have_same_class = B_TRUE;
1991 efx_mae_field_id_t field_id;
1992 const uint8_t *mvpl;
1993 const uint8_t *mvpr;
1996 switch (left->emms_type) {
1997 case EFX_MAE_RULE_OUTER:
1998 field_caps = maep->em_outer_rule_field_caps;
1999 desc_setp = __efx_mae_outer_rule_mv_desc_set;
2001 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_desc_set);
2002 bit_desc_setp = __efx_mae_outer_rule_mv_bit_desc_set;
2003 bit_desc_set_nentries =
2004 EFX_ARRAY_SIZE(__efx_mae_outer_rule_mv_bit_desc_set);
2005 mvpl = left->emms_mask_value_pairs.outer;
2006 mvpr = right->emms_mask_value_pairs.outer;
2008 case EFX_MAE_RULE_ACTION:
2009 field_caps = maep->em_action_rule_field_caps;
2010 desc_setp = __efx_mae_action_rule_mv_desc_set;
2012 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
2013 bit_desc_setp = __efx_mae_action_rule_mv_bit_desc_set;
2014 bit_desc_set_nentries =
2015 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_bit_desc_set);
2016 mvpl = left->emms_mask_value_pairs.action;
2017 mvpr = right->emms_mask_value_pairs.action;
2024 if (field_caps == NULL) {
2029 if (left->emms_type != right->emms_type ||
2030 left->emms_prio != right->emms_prio) {
2032 * Rules of different types can never map to the same class.
2034 * The FW can support some set of match criteria for one
2035 * priority and not support the very same set for
2036 * another priority. Thus, two rules which have
2037 * different priorities can never map to
2040 *have_same_classp = B_FALSE;
2044 for (field_id = 0; (unsigned int)field_id < desc_set_nentries;
2046 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
2047 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
2048 const uint8_t *lmaskp = mvpl + descp->emmd_mask_offset;
2049 const uint8_t *rmaskp = mvpr + descp->emmd_mask_offset;
2050 size_t mask_size = descp->emmd_mask_size;
2051 const uint8_t *lvalp = mvpl + descp->emmd_value_offset;
2052 const uint8_t *rvalp = mvpr + descp->emmd_value_offset;
2053 size_t value_size = descp->emmd_value_size;
2056 continue; /* Skip array gap */
2058 if ((unsigned int)field_cap_id >= field_ncaps) {
2060 * The FW has not reported capability status for this
2061 * field. It's unknown whether any difference between
2062 * the two masks / values affects the class. The only
2063 * case when the class must be the same is when these
2064 * mask-value pairs match. Otherwise, report mismatch.
2066 if ((memcmp(lmaskp, rmaskp, mask_size) == 0) &&
2067 (memcmp(lvalp, rvalp, value_size) == 0))
2073 if (field_caps[field_cap_id].emfc_mask_affects_class) {
2074 if (memcmp(lmaskp, rmaskp, mask_size) != 0) {
2075 have_same_class = B_FALSE;
2080 if (field_caps[field_cap_id].emfc_match_affects_class) {
2081 if (memcmp(lvalp, rvalp, value_size) != 0) {
2082 have_same_class = B_FALSE;
2088 if (have_same_class == B_FALSE)
2091 for (field_id = 0; (unsigned int)field_id < bit_desc_set_nentries;
2093 const efx_mae_mv_bit_desc_t *bit_descp =
2094 &bit_desc_setp[field_id];
2095 efx_mae_field_cap_id_t bit_cap_id =
2096 bit_descp->emmbd_bit_cap_id;
2097 unsigned int byte_idx;
2098 unsigned int bit_idx;
2100 if (bit_descp->emmbd_entry_is_valid == B_FALSE)
2101 continue; /* Skip array gap */
2103 if ((unsigned int)bit_cap_id >= field_ncaps)
2107 bit_descp->emmbd_mask_ofst +
2108 bit_descp->emmbd_mask_lbn / 8;
2110 bit_descp->emmbd_mask_lbn % 8;
2112 if (field_caps[bit_cap_id].emfc_mask_affects_class &&
2113 (mvpl[byte_idx] & (1U << bit_idx)) !=
2114 (mvpr[byte_idx] & (1U << bit_idx))) {
2115 have_same_class = B_FALSE;
2120 bit_descp->emmbd_value_ofst +
2121 bit_descp->emmbd_value_lbn / 8;
2123 bit_descp->emmbd_value_lbn % 8;
2125 if (field_caps[bit_cap_id].emfc_match_affects_class &&
2126 (mvpl[byte_idx] & (1U << bit_idx)) !=
2127 (mvpr[byte_idx] & (1U << bit_idx))) {
2128 have_same_class = B_FALSE;
2134 *have_same_classp = have_same_class;
2141 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2145 __checkReturn efx_rc_t
2146 efx_mae_outer_rule_recirc_id_set(
2147 __in efx_mae_match_spec_t *spec,
2148 __in uint8_t recirc_id)
2152 if (spec->emms_type != EFX_MAE_RULE_OUTER) {
2157 spec->emms_outer_rule_recirc_id = recirc_id;
2162 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2166 __checkReturn efx_rc_t
2167 efx_mae_outer_rule_insert(
2168 __in efx_nic_t *enp,
2169 __in const efx_mae_match_spec_t *spec,
2170 __in efx_tunnel_protocol_t encap_type,
2171 __out efx_mae_rule_id_t *or_idp)
2173 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
2175 EFX_MCDI_DECLARE_BUF(payload,
2176 MC_CMD_MAE_OUTER_RULE_INSERT_IN_LENMAX_MCDI2,
2177 MC_CMD_MAE_OUTER_RULE_INSERT_OUT_LEN);
2178 uint32_t encap_type_mcdi;
2179 efx_mae_rule_id_t or_id;
2183 EFX_STATIC_ASSERT(sizeof (or_idp->id) ==
2184 MC_CMD_MAE_OUTER_RULE_INSERT_OUT_OR_ID_LEN);
2186 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
2187 MC_CMD_MAE_OUTER_RULE_INSERT_OUT_OUTER_RULE_ID_NULL);
2189 if (encp->enc_mae_supported == B_FALSE) {
2194 if (spec->emms_type != EFX_MAE_RULE_OUTER) {
2199 switch (encap_type) {
2200 case EFX_TUNNEL_PROTOCOL_NONE:
2201 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_NONE;
2203 case EFX_TUNNEL_PROTOCOL_VXLAN:
2204 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_VXLAN;
2206 case EFX_TUNNEL_PROTOCOL_GENEVE:
2207 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_GENEVE;
2209 case EFX_TUNNEL_PROTOCOL_NVGRE:
2210 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_NVGRE;
2217 req.emr_cmd = MC_CMD_MAE_OUTER_RULE_INSERT;
2218 req.emr_in_buf = payload;
2219 req.emr_in_length = MC_CMD_MAE_OUTER_RULE_INSERT_IN_LENMAX_MCDI2;
2220 req.emr_out_buf = payload;
2221 req.emr_out_length = MC_CMD_MAE_OUTER_RULE_INSERT_OUT_LEN;
2223 MCDI_IN_SET_DWORD(req,
2224 MAE_OUTER_RULE_INSERT_IN_ENCAP_TYPE, encap_type_mcdi);
2226 MCDI_IN_SET_DWORD(req, MAE_OUTER_RULE_INSERT_IN_PRIO, spec->emms_prio);
2229 * Mask-value pairs have been stored in the byte order needed for the
2230 * MCDI request and are thus safe to be copied directly to the buffer.
2231 * The library cares about byte order in efx_mae_match_spec_field_set().
2233 EFX_STATIC_ASSERT(sizeof (spec->emms_mask_value_pairs.outer) >=
2234 MAE_ENC_FIELD_PAIRS_LEN);
2235 offset = MC_CMD_MAE_OUTER_RULE_INSERT_IN_FIELD_MATCH_CRITERIA_OFST;
2236 memcpy(payload + offset, spec->emms_mask_value_pairs.outer,
2237 MAE_ENC_FIELD_PAIRS_LEN);
2239 MCDI_IN_SET_BYTE(req, MAE_OUTER_RULE_INSERT_IN_RECIRC_ID,
2240 spec->emms_outer_rule_recirc_id);
2242 efx_mcdi_execute(enp, &req);
2244 if (req.emr_rc != 0) {
2249 if (req.emr_out_length_used < MC_CMD_MAE_OUTER_RULE_INSERT_OUT_LEN) {
2254 or_id.id = MCDI_OUT_DWORD(req, MAE_OUTER_RULE_INSERT_OUT_OR_ID);
2255 if (or_id.id == EFX_MAE_RSRC_ID_INVALID) {
2260 or_idp->id = or_id.id;
2275 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2279 __checkReturn efx_rc_t
2280 efx_mae_outer_rule_remove(
2281 __in efx_nic_t *enp,
2282 __in const efx_mae_rule_id_t *or_idp)
2284 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
2286 EFX_MCDI_DECLARE_BUF(payload,
2287 MC_CMD_MAE_OUTER_RULE_REMOVE_IN_LEN(1),
2288 MC_CMD_MAE_OUTER_RULE_REMOVE_OUT_LEN(1));
2291 if (encp->enc_mae_supported == B_FALSE) {
2296 req.emr_cmd = MC_CMD_MAE_OUTER_RULE_REMOVE;
2297 req.emr_in_buf = payload;
2298 req.emr_in_length = MC_CMD_MAE_OUTER_RULE_REMOVE_IN_LEN(1);
2299 req.emr_out_buf = payload;
2300 req.emr_out_length = MC_CMD_MAE_OUTER_RULE_REMOVE_OUT_LEN(1);
2302 MCDI_IN_SET_DWORD(req, MAE_OUTER_RULE_REMOVE_IN_OR_ID, or_idp->id);
2304 efx_mcdi_execute(enp, &req);
2306 if (req.emr_rc != 0) {
2311 if (req.emr_out_length_used < MC_CMD_MAE_OUTER_RULE_REMOVE_OUT_LENMIN) {
2316 if (MCDI_OUT_DWORD(req, MAE_OUTER_RULE_REMOVE_OUT_REMOVED_OR_ID) !=
2318 /* Firmware failed to remove the outer rule. */
2332 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2336 __checkReturn efx_rc_t
2337 efx_mae_match_spec_outer_rule_id_set(
2338 __in efx_mae_match_spec_t *spec,
2339 __in const efx_mae_rule_id_t *or_idp)
2341 uint32_t full_mask = UINT32_MAX;
2344 if (spec->emms_type != EFX_MAE_RULE_ACTION) {
2349 if (or_idp == NULL) {
2354 rc = efx_mae_match_spec_field_set(spec, EFX_MAE_FIELD_OUTER_RULE_ID,
2355 sizeof (or_idp->id), (const uint8_t *)&or_idp->id,
2356 sizeof (full_mask), (const uint8_t *)&full_mask);
2367 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2371 __checkReturn efx_rc_t
2372 efx_mae_encap_header_alloc(
2373 __in efx_nic_t *enp,
2374 __in efx_tunnel_protocol_t encap_type,
2375 __in_bcount(header_size) uint8_t *header_data,
2376 __in size_t header_size,
2377 __out efx_mae_eh_id_t *eh_idp)
2379 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
2381 EFX_MCDI_DECLARE_BUF(payload,
2382 MC_CMD_MAE_ENCAP_HEADER_ALLOC_IN_LENMAX_MCDI2,
2383 MC_CMD_MAE_ENCAP_HEADER_ALLOC_OUT_LEN);
2384 uint32_t encap_type_mcdi;
2385 efx_mae_eh_id_t eh_id;
2388 EFX_STATIC_ASSERT(sizeof (eh_idp->id) ==
2389 MC_CMD_MAE_ENCAP_HEADER_ALLOC_OUT_ENCAP_HEADER_ID_LEN);
2391 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
2392 MC_CMD_MAE_ENCAP_HEADER_ALLOC_OUT_ENCAP_HEADER_ID_NULL);
2394 if (encp->enc_mae_supported == B_FALSE) {
2399 switch (encap_type) {
2400 case EFX_TUNNEL_PROTOCOL_NONE:
2401 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_NONE;
2403 case EFX_TUNNEL_PROTOCOL_VXLAN:
2404 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_VXLAN;
2406 case EFX_TUNNEL_PROTOCOL_GENEVE:
2407 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_GENEVE;
2409 case EFX_TUNNEL_PROTOCOL_NVGRE:
2410 encap_type_mcdi = MAE_MCDI_ENCAP_TYPE_NVGRE;
2418 MC_CMD_MAE_ENCAP_HEADER_ALLOC_IN_HDR_DATA_MAXNUM_MCDI2) {
2423 req.emr_cmd = MC_CMD_MAE_ENCAP_HEADER_ALLOC;
2424 req.emr_in_buf = payload;
2425 req.emr_in_length = MC_CMD_MAE_ENCAP_HEADER_ALLOC_IN_LEN(header_size);
2426 req.emr_out_buf = payload;
2427 req.emr_out_length = MC_CMD_MAE_ENCAP_HEADER_ALLOC_OUT_LEN;
2429 MCDI_IN_SET_DWORD(req,
2430 MAE_ENCAP_HEADER_ALLOC_IN_ENCAP_TYPE, encap_type_mcdi);
2432 memcpy(payload + MC_CMD_MAE_ENCAP_HEADER_ALLOC_IN_HDR_DATA_OFST,
2433 header_data, header_size);
2435 efx_mcdi_execute(enp, &req);
2437 if (req.emr_rc != 0) {
2442 if (req.emr_out_length_used < MC_CMD_MAE_ENCAP_HEADER_ALLOC_OUT_LEN) {
2447 eh_id.id = MCDI_OUT_DWORD(req,
2448 MAE_ENCAP_HEADER_ALLOC_OUT_ENCAP_HEADER_ID);
2450 if (eh_id.id == EFX_MAE_RSRC_ID_INVALID) {
2455 eh_idp->id = eh_id.id;
2470 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2474 __checkReturn efx_rc_t
2475 efx_mae_encap_header_free(
2476 __in efx_nic_t *enp,
2477 __in const efx_mae_eh_id_t *eh_idp)
2479 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
2481 EFX_MCDI_DECLARE_BUF(payload,
2482 MC_CMD_MAE_ENCAP_HEADER_FREE_IN_LEN(1),
2483 MC_CMD_MAE_ENCAP_HEADER_FREE_OUT_LEN(1));
2486 if (encp->enc_mae_supported == B_FALSE) {
2491 req.emr_cmd = MC_CMD_MAE_ENCAP_HEADER_FREE;
2492 req.emr_in_buf = payload;
2493 req.emr_in_length = MC_CMD_MAE_ENCAP_HEADER_FREE_IN_LEN(1);
2494 req.emr_out_buf = payload;
2495 req.emr_out_length = MC_CMD_MAE_ENCAP_HEADER_FREE_OUT_LEN(1);
2497 MCDI_IN_SET_DWORD(req, MAE_ENCAP_HEADER_FREE_IN_EH_ID, eh_idp->id);
2499 efx_mcdi_execute(enp, &req);
2501 if (req.emr_rc != 0) {
2506 if (MCDI_OUT_DWORD(req, MAE_ENCAP_HEADER_FREE_OUT_FREED_EH_ID) !=
2508 /* Firmware failed to remove the encap. header. */
2520 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2524 __checkReturn efx_rc_t
2525 efx_mae_action_set_fill_in_eh_id(
2526 __in efx_mae_actions_t *spec,
2527 __in const efx_mae_eh_id_t *eh_idp)
2531 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_ENCAP)) == 0) {
2533 * The caller has not intended to have action ENCAP originally,
2534 * hence, this attempt to indicate encap. header ID is invalid.
2540 if (spec->ema_rsrc.emar_eh_id.id != EFX_MAE_RSRC_ID_INVALID) {
2541 /* The caller attempts to indicate encap. header ID twice. */
2546 if (eh_idp->id == EFX_MAE_RSRC_ID_INVALID) {
2551 spec->ema_rsrc.emar_eh_id.id = eh_idp->id;
2560 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2564 __checkReturn efx_rc_t
2565 efx_mae_action_set_alloc(
2566 __in efx_nic_t *enp,
2567 __in const efx_mae_actions_t *spec,
2568 __out efx_mae_aset_id_t *aset_idp)
2570 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
2572 EFX_MCDI_DECLARE_BUF(payload,
2573 MC_CMD_MAE_ACTION_SET_ALLOC_IN_LEN,
2574 MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN);
2575 efx_mae_aset_id_t aset_id;
2578 if (encp->enc_mae_supported == B_FALSE) {
2583 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_ENCAP)) != 0 &&
2584 spec->ema_rsrc.emar_eh_id.id == EFX_MAE_RSRC_ID_INVALID) {
2589 if (spec->ema_n_count_actions == 1 &&
2590 spec->ema_rsrc.emar_counter_id.id == EFX_MAE_RSRC_ID_INVALID) {
2595 req.emr_cmd = MC_CMD_MAE_ACTION_SET_ALLOC;
2596 req.emr_in_buf = payload;
2597 req.emr_in_length = MC_CMD_MAE_ACTION_SET_ALLOC_IN_LEN;
2598 req.emr_out_buf = payload;
2599 req.emr_out_length = MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN;
2602 * TODO: Remove these EFX_MAE_RSRC_ID_INVALID assignments once the
2603 * corresponding resource types are supported by the implementation.
2604 * Use proper resource ID assignments instead.
2606 MCDI_IN_SET_DWORD(req,
2607 MAE_ACTION_SET_ALLOC_IN_COUNTER_LIST_ID, EFX_MAE_RSRC_ID_INVALID);
2609 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_DECAP)) != 0) {
2610 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
2611 MAE_ACTION_SET_ALLOC_IN_DECAP, 1);
2614 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
2615 MAE_ACTION_SET_ALLOC_IN_VLAN_POP, spec->ema_n_vlan_tags_to_pop);
2617 if (spec->ema_n_vlan_tags_to_push > 0) {
2618 unsigned int outer_tag_idx;
2620 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
2621 MAE_ACTION_SET_ALLOC_IN_VLAN_PUSH,
2622 spec->ema_n_vlan_tags_to_push);
2624 if (spec->ema_n_vlan_tags_to_push ==
2625 EFX_MAE_VLAN_PUSH_MAX_NTAGS) {
2626 MCDI_IN_SET_WORD(req,
2627 MAE_ACTION_SET_ALLOC_IN_VLAN1_PROTO_BE,
2628 spec->ema_vlan_push_descs[0].emavp_tpid_be);
2629 MCDI_IN_SET_WORD(req,
2630 MAE_ACTION_SET_ALLOC_IN_VLAN1_TCI_BE,
2631 spec->ema_vlan_push_descs[0].emavp_tci_be);
2634 outer_tag_idx = spec->ema_n_vlan_tags_to_push - 1;
2636 MCDI_IN_SET_WORD(req, MAE_ACTION_SET_ALLOC_IN_VLAN0_PROTO_BE,
2637 spec->ema_vlan_push_descs[outer_tag_idx].emavp_tpid_be);
2638 MCDI_IN_SET_WORD(req, MAE_ACTION_SET_ALLOC_IN_VLAN0_TCI_BE,
2639 spec->ema_vlan_push_descs[outer_tag_idx].emavp_tci_be);
2642 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_ENCAP_HEADER_ID,
2643 spec->ema_rsrc.emar_eh_id.id);
2644 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_COUNTER_ID,
2645 spec->ema_rsrc.emar_counter_id.id);
2647 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_FLAG)) != 0) {
2648 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
2649 MAE_ACTION_SET_ALLOC_IN_FLAG, 1);
2652 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_MARK)) != 0) {
2653 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
2654 MAE_ACTION_SET_ALLOC_IN_MARK, 1);
2656 MCDI_IN_SET_DWORD(req,
2657 MAE_ACTION_SET_ALLOC_IN_MARK_VALUE, spec->ema_mark_value);
2660 MCDI_IN_SET_DWORD(req,
2661 MAE_ACTION_SET_ALLOC_IN_DELIVER, spec->ema_deliver_mport.sel);
2663 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_SRC_MAC_ID,
2664 MC_CMD_MAE_MAC_ADDR_ALLOC_OUT_MAC_ID_NULL);
2665 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_DST_MAC_ID,
2666 MC_CMD_MAE_MAC_ADDR_ALLOC_OUT_MAC_ID_NULL);
2668 efx_mcdi_execute(enp, &req);
2670 if (req.emr_rc != 0) {
2675 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN) {
2680 aset_id.id = MCDI_OUT_DWORD(req, MAE_ACTION_SET_ALLOC_OUT_AS_ID);
2681 if (aset_id.id == EFX_MAE_RSRC_ID_INVALID) {
2686 aset_idp->id = aset_id.id;
2701 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2705 __checkReturn unsigned int
2706 efx_mae_action_set_get_nb_count(
2707 __in const efx_mae_actions_t *spec)
2709 return (spec->ema_n_count_actions);
2712 __checkReturn efx_rc_t
2713 efx_mae_action_set_fill_in_counter_id(
2714 __in efx_mae_actions_t *spec,
2715 __in const efx_counter_t *counter_idp)
2719 if ((spec->ema_actions & (1U << EFX_MAE_ACTION_COUNT)) == 0) {
2721 * Invalid to add counter ID if spec does not have COUNT action.
2727 if (spec->ema_n_count_actions != 1) {
2729 * Having multiple COUNT actions in the spec requires a counter
2730 * list to be used. This API must only be used for a single
2731 * counter per spec. Turn down the request as inappropriate.
2737 if (spec->ema_rsrc.emar_counter_id.id != EFX_MAE_RSRC_ID_INVALID) {
2738 /* The caller attempts to indicate counter ID twice. */
2743 if (counter_idp->id == EFX_MAE_RSRC_ID_INVALID) {
2748 spec->ema_rsrc.emar_counter_id.id = counter_idp->id;
2759 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2763 __checkReturn efx_rc_t
2764 efx_mae_counters_alloc(
2765 __in efx_nic_t *enp,
2766 __in uint32_t n_counters,
2767 __out uint32_t *n_allocatedp,
2768 __out_ecount(n_counters) efx_counter_t *countersp,
2769 __out_opt uint32_t *gen_countp)
2771 EFX_MCDI_DECLARE_BUF(payload,
2772 MC_CMD_MAE_COUNTER_ALLOC_IN_LEN,
2773 MC_CMD_MAE_COUNTER_ALLOC_OUT_LENMAX_MCDI2);
2774 efx_mae_t *maep = enp->en_maep;
2775 uint32_t n_allocated;
2780 if (n_counters > maep->em_max_ncounters ||
2781 n_counters < MC_CMD_MAE_COUNTER_ALLOC_OUT_COUNTER_ID_MINNUM ||
2782 n_counters > MC_CMD_MAE_COUNTER_ALLOC_OUT_COUNTER_ID_MAXNUM_MCDI2) {
2787 req.emr_cmd = MC_CMD_MAE_COUNTER_ALLOC;
2788 req.emr_in_buf = payload;
2789 req.emr_in_length = MC_CMD_MAE_COUNTER_ALLOC_IN_LEN;
2790 req.emr_out_buf = payload;
2791 req.emr_out_length = MC_CMD_MAE_COUNTER_ALLOC_OUT_LEN(n_counters);
2793 MCDI_IN_SET_DWORD(req, MAE_COUNTER_ALLOC_IN_REQUESTED_COUNT,
2796 efx_mcdi_execute(enp, &req);
2798 if (req.emr_rc != 0) {
2803 if (req.emr_out_length_used < MC_CMD_MAE_COUNTER_ALLOC_OUT_LENMIN) {
2808 n_allocated = MCDI_OUT_DWORD(req,
2809 MAE_COUNTER_ALLOC_OUT_COUNTER_ID_COUNT);
2810 if (n_allocated < MC_CMD_MAE_COUNTER_ALLOC_OUT_COUNTER_ID_MINNUM) {
2815 for (i = 0; i < n_allocated; i++) {
2816 countersp[i].id = MCDI_OUT_INDEXED_DWORD(req,
2817 MAE_COUNTER_ALLOC_OUT_COUNTER_ID, i);
2820 if (gen_countp != NULL) {
2821 *gen_countp = MCDI_OUT_DWORD(req,
2822 MAE_COUNTER_ALLOC_OUT_GENERATION_COUNT);
2825 *n_allocatedp = n_allocated;
2836 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2841 __checkReturn efx_rc_t
2842 efx_mae_counters_free(
2843 __in efx_nic_t *enp,
2844 __in uint32_t n_counters,
2845 __out uint32_t *n_freedp,
2846 __in_ecount(n_counters) const efx_counter_t *countersp,
2847 __out_opt uint32_t *gen_countp)
2849 EFX_MCDI_DECLARE_BUF(payload,
2850 MC_CMD_MAE_COUNTER_FREE_IN_LENMAX_MCDI2,
2851 MC_CMD_MAE_COUNTER_FREE_OUT_LENMAX_MCDI2);
2852 efx_mae_t *maep = enp->en_maep;
2858 if (n_counters > maep->em_max_ncounters ||
2859 n_counters < MC_CMD_MAE_COUNTER_FREE_IN_FREE_COUNTER_ID_MINNUM ||
2861 MC_CMD_MAE_COUNTER_FREE_IN_FREE_COUNTER_ID_MAXNUM_MCDI2) {
2866 req.emr_cmd = MC_CMD_MAE_COUNTER_FREE;
2867 req.emr_in_buf = payload;
2868 req.emr_in_length = MC_CMD_MAE_COUNTER_FREE_IN_LEN(n_counters);
2869 req.emr_out_buf = payload;
2870 req.emr_out_length = MC_CMD_MAE_COUNTER_FREE_OUT_LEN(n_counters);
2872 for (i = 0; i < n_counters; i++) {
2873 MCDI_IN_SET_INDEXED_DWORD(req,
2874 MAE_COUNTER_FREE_IN_FREE_COUNTER_ID, i, countersp[i].id);
2876 MCDI_IN_SET_DWORD(req, MAE_COUNTER_FREE_IN_COUNTER_ID_COUNT,
2879 efx_mcdi_execute(enp, &req);
2881 if (req.emr_rc != 0) {
2886 if (req.emr_out_length_used < MC_CMD_MAE_COUNTER_FREE_OUT_LENMIN) {
2891 n_freed = MCDI_OUT_DWORD(req, MAE_COUNTER_FREE_OUT_COUNTER_ID_COUNT);
2893 if (n_freed < MC_CMD_MAE_COUNTER_FREE_OUT_FREED_COUNTER_ID_MINNUM) {
2898 if (gen_countp != NULL) {
2899 *gen_countp = MCDI_OUT_DWORD(req,
2900 MAE_COUNTER_FREE_OUT_GENERATION_COUNT);
2903 *n_freedp = n_freed;
2914 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2919 __checkReturn efx_rc_t
2920 efx_mae_counters_stream_start(
2921 __in efx_nic_t *enp,
2922 __in uint16_t rxq_id,
2923 __in uint16_t packet_size,
2924 __in uint32_t flags_in,
2925 __out uint32_t *flags_out)
2928 EFX_MCDI_DECLARE_BUF(payload, MC_CMD_MAE_COUNTERS_STREAM_START_IN_LEN,
2929 MC_CMD_MAE_COUNTERS_STREAM_START_OUT_LEN);
2932 EFX_STATIC_ASSERT(EFX_MAE_COUNTERS_STREAM_IN_ZERO_SQUASH_DISABLE ==
2933 1U << MC_CMD_MAE_COUNTERS_STREAM_START_IN_ZERO_SQUASH_DISABLE_LBN);
2935 EFX_STATIC_ASSERT(EFX_MAE_COUNTERS_STREAM_OUT_USES_CREDITS ==
2936 1U << MC_CMD_MAE_COUNTERS_STREAM_START_OUT_USES_CREDITS_LBN);
2938 req.emr_cmd = MC_CMD_MAE_COUNTERS_STREAM_START;
2939 req.emr_in_buf = payload;
2940 req.emr_in_length = MC_CMD_MAE_COUNTERS_STREAM_START_IN_LEN;
2941 req.emr_out_buf = payload;
2942 req.emr_out_length = MC_CMD_MAE_COUNTERS_STREAM_START_OUT_LEN;
2944 MCDI_IN_SET_WORD(req, MAE_COUNTERS_STREAM_START_IN_QID, rxq_id);
2945 MCDI_IN_SET_WORD(req, MAE_COUNTERS_STREAM_START_IN_PACKET_SIZE,
2947 MCDI_IN_SET_DWORD(req, MAE_COUNTERS_STREAM_START_IN_FLAGS, flags_in);
2949 efx_mcdi_execute(enp, &req);
2951 if (req.emr_rc != 0) {
2956 if (req.emr_out_length_used <
2957 MC_CMD_MAE_COUNTERS_STREAM_START_OUT_LEN) {
2962 *flags_out = MCDI_OUT_DWORD(req, MAE_COUNTERS_STREAM_START_OUT_FLAGS);
2969 EFSYS_PROBE1(fail1, efx_rc_t, rc);
2974 __checkReturn efx_rc_t
2975 efx_mae_counters_stream_stop(
2976 __in efx_nic_t *enp,
2977 __in uint16_t rxq_id,
2978 __out_opt uint32_t *gen_countp)
2981 EFX_MCDI_DECLARE_BUF(payload, MC_CMD_MAE_COUNTERS_STREAM_STOP_IN_LEN,
2982 MC_CMD_MAE_COUNTERS_STREAM_STOP_OUT_LEN);
2985 req.emr_cmd = MC_CMD_MAE_COUNTERS_STREAM_STOP;
2986 req.emr_in_buf = payload;
2987 req.emr_in_length = MC_CMD_MAE_COUNTERS_STREAM_STOP_IN_LEN;
2988 req.emr_out_buf = payload;
2989 req.emr_out_length = MC_CMD_MAE_COUNTERS_STREAM_STOP_OUT_LEN;
2991 MCDI_IN_SET_WORD(req, MAE_COUNTERS_STREAM_STOP_IN_QID, rxq_id);
2993 efx_mcdi_execute(enp, &req);
2995 if (req.emr_rc != 0) {
3000 if (req.emr_out_length_used <
3001 MC_CMD_MAE_COUNTERS_STREAM_STOP_OUT_LEN) {
3006 if (gen_countp != NULL) {
3007 *gen_countp = MCDI_OUT_DWORD(req,
3008 MAE_COUNTERS_STREAM_STOP_OUT_GENERATION_COUNT);
3016 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3021 __checkReturn efx_rc_t
3022 efx_mae_counters_stream_give_credits(
3023 __in efx_nic_t *enp,
3024 __in uint32_t n_credits)
3027 EFX_MCDI_DECLARE_BUF(payload,
3028 MC_CMD_MAE_COUNTERS_STREAM_GIVE_CREDITS_IN_LEN,
3029 MC_CMD_MAE_COUNTERS_STREAM_GIVE_CREDITS_OUT_LEN);
3032 req.emr_cmd = MC_CMD_MAE_COUNTERS_STREAM_GIVE_CREDITS;
3033 req.emr_in_buf = payload;
3034 req.emr_in_length = MC_CMD_MAE_COUNTERS_STREAM_GIVE_CREDITS_IN_LEN;
3035 req.emr_out_buf = payload;
3036 req.emr_out_length = MC_CMD_MAE_COUNTERS_STREAM_GIVE_CREDITS_OUT_LEN;
3038 MCDI_IN_SET_DWORD(req, MAE_COUNTERS_STREAM_GIVE_CREDITS_IN_NUM_CREDITS,
3041 efx_mcdi_execute(enp, &req);
3043 if (req.emr_rc != 0) {
3051 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3056 __checkReturn efx_rc_t
3057 efx_mae_action_set_free(
3058 __in efx_nic_t *enp,
3059 __in const efx_mae_aset_id_t *aset_idp)
3061 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
3063 EFX_MCDI_DECLARE_BUF(payload,
3064 MC_CMD_MAE_ACTION_SET_FREE_IN_LEN(1),
3065 MC_CMD_MAE_ACTION_SET_FREE_OUT_LEN(1));
3068 if (encp->enc_mae_supported == B_FALSE) {
3073 req.emr_cmd = MC_CMD_MAE_ACTION_SET_FREE;
3074 req.emr_in_buf = payload;
3075 req.emr_in_length = MC_CMD_MAE_ACTION_SET_FREE_IN_LEN(1);
3076 req.emr_out_buf = payload;
3077 req.emr_out_length = MC_CMD_MAE_ACTION_SET_FREE_OUT_LEN(1);
3079 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_FREE_IN_AS_ID, aset_idp->id);
3081 efx_mcdi_execute(enp, &req);
3083 if (req.emr_rc != 0) {
3088 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_SET_FREE_OUT_LENMIN) {
3093 if (MCDI_OUT_DWORD(req, MAE_ACTION_SET_FREE_OUT_FREED_AS_ID) !=
3095 /* Firmware failed to free the action set. */
3109 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3113 __checkReturn efx_rc_t
3114 efx_mae_action_rule_insert(
3115 __in efx_nic_t *enp,
3116 __in const efx_mae_match_spec_t *spec,
3117 __in const efx_mae_aset_list_id_t *asl_idp,
3118 __in const efx_mae_aset_id_t *as_idp,
3119 __out efx_mae_rule_id_t *ar_idp)
3121 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
3123 EFX_MCDI_DECLARE_BUF(payload,
3124 MC_CMD_MAE_ACTION_RULE_INSERT_IN_LENMAX_MCDI2,
3125 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN);
3126 efx_oword_t *rule_response;
3127 efx_mae_rule_id_t ar_id;
3131 EFX_STATIC_ASSERT(sizeof (ar_idp->id) ==
3132 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_AR_ID_LEN);
3134 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
3135 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_ACTION_RULE_ID_NULL);
3137 if (encp->enc_mae_supported == B_FALSE) {
3142 if (spec->emms_type != EFX_MAE_RULE_ACTION ||
3143 (asl_idp != NULL && as_idp != NULL) ||
3144 (asl_idp == NULL && as_idp == NULL)) {
3149 req.emr_cmd = MC_CMD_MAE_ACTION_RULE_INSERT;
3150 req.emr_in_buf = payload;
3151 req.emr_in_length = MC_CMD_MAE_ACTION_RULE_INSERT_IN_LENMAX_MCDI2;
3152 req.emr_out_buf = payload;
3153 req.emr_out_length = MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN;
3155 EFX_STATIC_ASSERT(sizeof (*rule_response) <=
3156 MC_CMD_MAE_ACTION_RULE_INSERT_IN_RESPONSE_LEN);
3157 offset = MC_CMD_MAE_ACTION_RULE_INSERT_IN_RESPONSE_OFST;
3158 rule_response = (efx_oword_t *)(payload + offset);
3159 EFX_POPULATE_OWORD_3(*rule_response,
3160 MAE_ACTION_RULE_RESPONSE_ASL_ID,
3161 (asl_idp != NULL) ? asl_idp->id : EFX_MAE_RSRC_ID_INVALID,
3162 MAE_ACTION_RULE_RESPONSE_AS_ID,
3163 (as_idp != NULL) ? as_idp->id : EFX_MAE_RSRC_ID_INVALID,
3164 MAE_ACTION_RULE_RESPONSE_COUNTER_ID, EFX_MAE_RSRC_ID_INVALID);
3166 MCDI_IN_SET_DWORD(req, MAE_ACTION_RULE_INSERT_IN_PRIO, spec->emms_prio);
3169 * Mask-value pairs have been stored in the byte order needed for the
3170 * MCDI request and are thus safe to be copied directly to the buffer.
3172 EFX_STATIC_ASSERT(sizeof (spec->emms_mask_value_pairs.action) >=
3173 MAE_FIELD_MASK_VALUE_PAIRS_V2_LEN);
3174 offset = MC_CMD_MAE_ACTION_RULE_INSERT_IN_MATCH_CRITERIA_OFST;
3175 memcpy(payload + offset, spec->emms_mask_value_pairs.action,
3176 MAE_FIELD_MASK_VALUE_PAIRS_V2_LEN);
3178 efx_mcdi_execute(enp, &req);
3180 if (req.emr_rc != 0) {
3185 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN) {
3190 ar_id.id = MCDI_OUT_DWORD(req, MAE_ACTION_RULE_INSERT_OUT_AR_ID);
3191 if (ar_id.id == EFX_MAE_RSRC_ID_INVALID) {
3196 ar_idp->id = ar_id.id;
3209 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3213 __checkReturn efx_rc_t
3214 efx_mae_action_rule_remove(
3215 __in efx_nic_t *enp,
3216 __in const efx_mae_rule_id_t *ar_idp)
3218 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
3220 EFX_MCDI_DECLARE_BUF(payload,
3221 MC_CMD_MAE_ACTION_RULE_DELETE_IN_LEN(1),
3222 MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LEN(1));
3225 if (encp->enc_mae_supported == B_FALSE) {
3230 req.emr_cmd = MC_CMD_MAE_ACTION_RULE_DELETE;
3231 req.emr_in_buf = payload;
3232 req.emr_in_length = MC_CMD_MAE_ACTION_RULE_DELETE_IN_LEN(1);
3233 req.emr_out_buf = payload;
3234 req.emr_out_length = MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LEN(1);
3236 MCDI_IN_SET_DWORD(req, MAE_ACTION_RULE_DELETE_IN_AR_ID, ar_idp->id);
3238 efx_mcdi_execute(enp, &req);
3240 if (req.emr_rc != 0) {
3245 if (req.emr_out_length_used <
3246 MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LENMIN) {
3251 if (MCDI_OUT_DWORD(req, MAE_ACTION_RULE_DELETE_OUT_DELETED_AR_ID) !=
3253 /* Firmware failed to delete the action rule. */
3267 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3271 __checkReturn efx_rc_t
3272 efx_mcdi_mport_alloc_alias(
3273 __in efx_nic_t *enp,
3274 __out efx_mport_id_t *mportp,
3275 __out_opt uint32_t *labelp)
3277 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
3279 EFX_MCDI_DECLARE_BUF(payload,
3280 MC_CMD_MAE_MPORT_ALLOC_ALIAS_IN_LEN,
3281 MC_CMD_MAE_MPORT_ALLOC_ALIAS_OUT_LEN);
3284 if (encp->enc_mae_supported == B_FALSE) {
3289 req.emr_cmd = MC_CMD_MAE_MPORT_ALLOC;
3290 req.emr_in_buf = payload;
3291 req.emr_in_length = MC_CMD_MAE_MPORT_ALLOC_ALIAS_IN_LEN;
3292 req.emr_out_buf = payload;
3293 req.emr_out_length = MC_CMD_MAE_MPORT_ALLOC_ALIAS_OUT_LEN;
3295 MCDI_IN_SET_DWORD(req, MAE_MPORT_ALLOC_IN_TYPE,
3296 MC_CMD_MAE_MPORT_ALLOC_IN_MPORT_TYPE_ALIAS);
3297 MCDI_IN_SET_DWORD(req, MAE_MPORT_ALLOC_ALIAS_IN_DELIVER_MPORT,
3298 MAE_MPORT_SELECTOR_ASSIGNED);
3300 efx_mcdi_execute(enp, &req);
3302 if (req.emr_rc != 0) {
3307 mportp->id = MCDI_OUT_DWORD(req, MAE_MPORT_ALLOC_OUT_MPORT_ID);
3309 *labelp = MCDI_OUT_DWORD(req, MAE_MPORT_ALLOC_ALIAS_OUT_LABEL);
3316 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3320 __checkReturn efx_rc_t
3322 __in efx_nic_t *enp,
3323 __in const efx_mport_id_t *mportp)
3325 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
3327 EFX_MCDI_DECLARE_BUF(payload,
3328 MC_CMD_MAE_MPORT_FREE_IN_LEN,
3329 MC_CMD_MAE_MPORT_FREE_OUT_LEN);
3332 if (encp->enc_mae_supported == B_FALSE) {
3337 req.emr_cmd = MC_CMD_MAE_MPORT_FREE;
3338 req.emr_in_buf = payload;
3339 req.emr_in_length = MC_CMD_MAE_MPORT_FREE_IN_LEN;
3340 req.emr_out_buf = payload;
3341 req.emr_out_length = MC_CMD_MAE_MPORT_FREE_OUT_LEN;
3343 MCDI_IN_SET_DWORD(req, MAE_MPORT_FREE_IN_MPORT_ID, mportp->id);
3345 efx_mcdi_execute(enp, &req);
3347 if (req.emr_rc != 0) {
3357 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3361 static __checkReturn efx_rc_t
3362 efx_mae_read_mport_journal_single(
3363 __in uint8_t *entry_buf,
3364 __out efx_mport_desc_t *desc)
3369 memset(desc, 0, sizeof (*desc));
3371 desc->emd_id.id = MCDI_STRUCT_DWORD(entry_buf,
3372 MAE_MPORT_DESC_V2_MPORT_ID);
3374 desc->emd_can_receive_on = MCDI_STRUCT_DWORD_FIELD(entry_buf,
3375 MAE_MPORT_DESC_V2_FLAGS,
3376 MAE_MPORT_DESC_V2_CAN_RECEIVE_ON);
3378 desc->emd_can_deliver_to = MCDI_STRUCT_DWORD_FIELD(entry_buf,
3379 MAE_MPORT_DESC_V2_FLAGS,
3380 MAE_MPORT_DESC_V2_CAN_DELIVER_TO);
3382 desc->emd_can_delete = MCDI_STRUCT_DWORD_FIELD(entry_buf,
3383 MAE_MPORT_DESC_V2_FLAGS,
3384 MAE_MPORT_DESC_V2_CAN_DELETE);
3386 desc->emd_zombie = MCDI_STRUCT_DWORD_FIELD(entry_buf,
3387 MAE_MPORT_DESC_V2_FLAGS,
3388 MAE_MPORT_DESC_V2_IS_ZOMBIE);
3390 desc->emd_type = MCDI_STRUCT_DWORD(entry_buf,
3391 MAE_MPORT_DESC_V2_MPORT_TYPE);
3394 * We can't check everything here. If some additional checks are
3395 * required, they should be performed by the callback function.
3397 switch (desc->emd_type) {
3398 case EFX_MPORT_TYPE_NET_PORT:
3399 desc->emd_net_port.ep_index =
3400 MCDI_STRUCT_DWORD(entry_buf,
3401 MAE_MPORT_DESC_V2_NET_PORT_IDX);
3403 case EFX_MPORT_TYPE_ALIAS:
3404 desc->emd_alias.ea_target_mport_id.id =
3405 MCDI_STRUCT_DWORD(entry_buf,
3406 MAE_MPORT_DESC_V2_ALIAS_DELIVER_MPORT_ID);
3408 case EFX_MPORT_TYPE_VNIC:
3409 desc->emd_vnic.ev_client_type =
3410 MCDI_STRUCT_DWORD(entry_buf,
3411 MAE_MPORT_DESC_V2_VNIC_CLIENT_TYPE);
3412 if (desc->emd_vnic.ev_client_type !=
3413 EFX_MPORT_VNIC_CLIENT_FUNCTION)
3416 pcie_intf = MCDI_STRUCT_DWORD(entry_buf,
3417 MAE_MPORT_DESC_V2_VNIC_FUNCTION_INTERFACE);
3418 rc = efx_mcdi_intf_from_pcie(pcie_intf,
3419 &desc->emd_vnic.ev_intf);
3423 desc->emd_vnic.ev_pf = MCDI_STRUCT_WORD(entry_buf,
3424 MAE_MPORT_DESC_V2_VNIC_FUNCTION_PF_IDX);
3425 desc->emd_vnic.ev_vf = MCDI_STRUCT_WORD(entry_buf,
3426 MAE_MPORT_DESC_V2_VNIC_FUNCTION_VF_IDX);
3427 desc->emd_vnic.ev_handle = MCDI_STRUCT_DWORD(entry_buf,
3428 MAE_MPORT_DESC_V2_VNIC_CLIENT_HANDLE);
3440 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3444 static __checkReturn efx_rc_t
3445 efx_mae_read_mport_journal_batch(
3446 __in efx_nic_t *enp,
3447 __in efx_mae_read_mport_journal_cb *cbp,
3448 __in void *cb_datap,
3449 __out uint32_t *morep)
3452 EFX_MCDI_DECLARE_BUF(payload,
3453 MC_CMD_MAE_MPORT_READ_JOURNAL_IN_LEN,
3454 MC_CMD_MAE_MPORT_READ_JOURNAL_OUT_LENMAX_MCDI2);
3461 EFX_STATIC_ASSERT(EFX_MPORT_TYPE_NET_PORT ==
3462 MAE_MPORT_DESC_V2_MPORT_TYPE_NET_PORT);
3463 EFX_STATIC_ASSERT(EFX_MPORT_TYPE_ALIAS ==
3464 MAE_MPORT_DESC_V2_MPORT_TYPE_ALIAS);
3465 EFX_STATIC_ASSERT(EFX_MPORT_TYPE_VNIC ==
3466 MAE_MPORT_DESC_V2_MPORT_TYPE_VNIC);
3468 EFX_STATIC_ASSERT(EFX_MPORT_VNIC_CLIENT_FUNCTION ==
3469 MAE_MPORT_DESC_V2_VNIC_CLIENT_TYPE_FUNCTION);
3470 EFX_STATIC_ASSERT(EFX_MPORT_VNIC_CLIENT_PLUGIN ==
3471 MAE_MPORT_DESC_V2_VNIC_CLIENT_TYPE_PLUGIN);
3478 req.emr_cmd = MC_CMD_MAE_MPORT_READ_JOURNAL;
3479 req.emr_in_buf = payload;
3480 req.emr_in_length = MC_CMD_MAE_MPORT_READ_JOURNAL_IN_LEN;
3481 req.emr_out_buf = payload;
3482 req.emr_out_length = MC_CMD_MAE_MPORT_READ_JOURNAL_OUT_LENMAX_MCDI2;
3484 MCDI_IN_SET_DWORD(req, MAE_MPORT_READ_JOURNAL_IN_FLAGS, 0);
3486 efx_mcdi_execute(enp, &req);
3488 if (req.emr_rc != 0) {
3493 if (req.emr_out_length_used <
3494 MC_CMD_MAE_MPORT_READ_JOURNAL_OUT_LENMIN) {
3499 if (morep != NULL) {
3500 *morep = MCDI_OUT_DWORD_FIELD(req,
3501 MAE_MPORT_READ_JOURNAL_OUT_FLAGS,
3502 MAE_MPORT_READ_JOURNAL_OUT_MORE);
3504 n_entries = MCDI_OUT_DWORD(req,
3505 MAE_MPORT_READ_JOURNAL_OUT_MPORT_DESC_COUNT);
3506 entry_sz = MCDI_OUT_DWORD(req,
3507 MAE_MPORT_READ_JOURNAL_OUT_SIZEOF_MPORT_DESC);
3508 entry_buf = MCDI_OUT2(req, uint8_t,
3509 MAE_MPORT_READ_JOURNAL_OUT_MPORT_DESC_DATA);
3511 if (entry_sz < MAE_MPORT_DESC_V2_VNIC_CLIENT_HANDLE_OFST +
3512 MAE_MPORT_DESC_V2_VNIC_CLIENT_HANDLE_LEN) {
3516 if (n_entries * entry_sz / entry_sz != n_entries) {
3520 if (req.emr_out_length_used !=
3521 MC_CMD_MAE_MPORT_READ_JOURNAL_OUT_LENMIN + n_entries * entry_sz) {
3526 for (i = 0; i < n_entries; i++) {
3527 efx_mport_desc_t desc;
3529 rc = efx_mae_read_mport_journal_single(entry_buf, &desc);
3533 (*cbp)(cb_datap, &desc, sizeof (desc));
3534 entry_buf += entry_sz;
3550 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3554 __checkReturn efx_rc_t
3555 efx_mae_read_mport_journal(
3556 __in efx_nic_t *enp,
3557 __in efx_mae_read_mport_journal_cb *cbp,
3558 __in void *cb_datap)
3560 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
3564 if (encp->enc_mae_supported == B_FALSE) {
3570 rc = efx_mae_read_mport_journal_batch(enp, cbp, cb_datap,
3574 } while (more != 0);
3581 EFSYS_PROBE1(fail1, efx_rc_t, rc);
3585 #endif /* EFSYS_OPT_MAE */