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_action_prios =
43 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_ACTION_PRIOS);
45 maep->em_max_nfields =
46 MCDI_OUT_DWORD(req, MAE_GET_CAPS_OUT_MATCH_FIELD_COUNT);
53 EFSYS_PROBE1(fail1, efx_rc_t, rc);
57 static __checkReturn efx_rc_t
58 efx_mae_get_action_rule_caps(
60 __in unsigned int field_ncaps,
61 __out_ecount(field_ncaps) efx_mae_field_cap_t *field_caps)
64 EFX_MCDI_DECLARE_BUF(payload,
65 MC_CMD_MAE_GET_AR_CAPS_IN_LEN,
66 MC_CMD_MAE_GET_AR_CAPS_OUT_LENMAX_MCDI2);
67 unsigned int mcdi_field_ncaps;
71 if (MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(field_ncaps) >
72 MC_CMD_MAE_GET_AR_CAPS_OUT_LENMAX_MCDI2) {
77 req.emr_cmd = MC_CMD_MAE_GET_AR_CAPS;
78 req.emr_in_buf = payload;
79 req.emr_in_length = MC_CMD_MAE_GET_AR_CAPS_IN_LEN;
80 req.emr_out_buf = payload;
81 req.emr_out_length = MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(field_ncaps);
83 efx_mcdi_execute(enp, &req);
85 if (req.emr_rc != 0) {
90 mcdi_field_ncaps = MCDI_OUT_DWORD(req, MAE_GET_OR_CAPS_OUT_COUNT);
92 if (req.emr_out_length_used <
93 MC_CMD_MAE_GET_AR_CAPS_OUT_LEN(mcdi_field_ncaps)) {
98 if (mcdi_field_ncaps > field_ncaps) {
103 for (i = 0; i < mcdi_field_ncaps; ++i) {
107 field_caps[i].emfc_support = MCDI_OUT_INDEXED_DWORD_FIELD(req,
108 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
109 MAE_FIELD_FLAGS_SUPPORT_STATUS);
111 match_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
112 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
113 MAE_FIELD_FLAGS_MATCH_AFFECTS_CLASS);
115 field_caps[i].emfc_match_affects_class =
116 (match_flag != 0) ? B_TRUE : B_FALSE;
118 mask_flag = MCDI_OUT_INDEXED_DWORD_FIELD(req,
119 MAE_GET_AR_CAPS_OUT_FIELD_FLAGS, i,
120 MAE_FIELD_FLAGS_MASK_AFFECTS_CLASS);
122 field_caps[i].emfc_mask_affects_class =
123 (mask_flag != 0) ? B_TRUE : B_FALSE;
135 EFSYS_PROBE1(fail1, efx_rc_t, rc);
139 __checkReturn efx_rc_t
143 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
144 efx_mae_field_cap_t *ar_fcaps;
145 size_t ar_fcaps_size;
149 if (encp->enc_mae_supported == B_FALSE) {
154 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*maep), maep);
162 rc = efx_mae_get_capabilities(enp);
166 ar_fcaps_size = maep->em_max_nfields * sizeof (*ar_fcaps);
167 EFSYS_KMEM_ALLOC(enp->en_esip, ar_fcaps_size, ar_fcaps);
168 if (ar_fcaps == NULL) {
173 maep->em_action_rule_field_caps_size = ar_fcaps_size;
174 maep->em_action_rule_field_caps = ar_fcaps;
176 rc = efx_mae_get_action_rule_caps(enp, maep->em_max_nfields, ar_fcaps);
184 EFSYS_KMEM_FREE(enp->en_esip, ar_fcaps_size, ar_fcaps);
189 EFSYS_KMEM_FREE(enp->en_esip, sizeof (struct efx_mae_s), enp->en_maep);
194 EFSYS_PROBE1(fail1, efx_rc_t, rc);
202 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
203 efx_mae_t *maep = enp->en_maep;
205 if (encp->enc_mae_supported == B_FALSE)
208 EFSYS_KMEM_FREE(enp->en_esip, maep->em_action_rule_field_caps_size,
209 maep->em_action_rule_field_caps);
210 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*maep), maep);
214 __checkReturn efx_rc_t
217 __out efx_mae_limits_t *emlp)
219 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
220 struct efx_mae_s *maep = enp->en_maep;
223 if (encp->enc_mae_supported == B_FALSE) {
228 emlp->eml_max_n_action_prios = maep->em_max_n_action_prios;
233 EFSYS_PROBE1(fail1, efx_rc_t, rc);
237 __checkReturn efx_rc_t
238 efx_mae_match_spec_init(
240 __in efx_mae_rule_type_t type,
242 __out efx_mae_match_spec_t **specp)
244 efx_mae_match_spec_t *spec;
248 case EFX_MAE_RULE_ACTION:
255 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
261 spec->emms_type = type;
262 spec->emms_prio = prio;
271 EFSYS_PROBE1(fail1, efx_rc_t, rc);
276 efx_mae_match_spec_fini(
278 __in efx_mae_match_spec_t *spec)
280 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
283 /* Named identifiers which are valid indices to efx_mae_field_cap_t */
284 typedef enum efx_mae_field_cap_id_e {
285 EFX_MAE_FIELD_ID_INGRESS_MPORT_SELECTOR = MAE_FIELD_INGRESS_PORT,
286 EFX_MAE_FIELD_ID_ETHER_TYPE_BE = MAE_FIELD_ETHER_TYPE,
287 EFX_MAE_FIELD_ID_ETH_SADDR_BE = MAE_FIELD_ETH_SADDR,
288 EFX_MAE_FIELD_ID_ETH_DADDR_BE = MAE_FIELD_ETH_DADDR,
290 EFX_MAE_FIELD_CAP_NIDS
291 } efx_mae_field_cap_id_t;
293 typedef enum efx_mae_field_endianness_e {
294 EFX_MAE_FIELD_LE = 0,
297 EFX_MAE_FIELD_ENDIANNESS_NTYPES
298 } efx_mae_field_endianness_t;
301 * The following structure is a means to describe an MAE field.
302 * The information in it is meant to be used internally by
303 * APIs for addressing a given field in a mask-value pairs
304 * structure and for validation purposes.
306 typedef struct efx_mae_mv_desc_s {
307 efx_mae_field_cap_id_t emmd_field_cap_id;
309 size_t emmd_value_size;
310 size_t emmd_value_offset;
311 size_t emmd_mask_size;
312 size_t emmd_mask_offset;
314 efx_mae_field_endianness_t emmd_endianness;
317 /* Indices to this array are provided by efx_mae_field_id_t */
318 static const efx_mae_mv_desc_t __efx_mae_action_rule_mv_desc_set[] = {
319 #define EFX_MAE_MV_DESC(_name, _endianness) \
320 [EFX_MAE_FIELD_##_name] = \
322 EFX_MAE_FIELD_ID_##_name, \
323 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_LEN, \
324 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_OFST, \
325 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_MASK_LEN, \
326 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_MASK_OFST, \
330 EFX_MAE_MV_DESC(INGRESS_MPORT_SELECTOR, EFX_MAE_FIELD_LE),
331 EFX_MAE_MV_DESC(ETHER_TYPE_BE, EFX_MAE_FIELD_BE),
332 EFX_MAE_MV_DESC(ETH_SADDR_BE, EFX_MAE_FIELD_BE),
333 EFX_MAE_MV_DESC(ETH_DADDR_BE, EFX_MAE_FIELD_BE),
335 #undef EFX_MAE_MV_DESC
338 __checkReturn efx_rc_t
339 efx_mae_mport_by_phy_port(
340 __in uint32_t phy_port,
341 __out efx_mport_sel_t *mportp)
346 if (phy_port > EFX_MASK32(MAE_MPORT_SELECTOR_PPORT_ID)) {
351 EFX_POPULATE_DWORD_2(dword,
352 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_PPORT,
353 MAE_MPORT_SELECTOR_PPORT_ID, phy_port);
355 memset(mportp, 0, sizeof (*mportp));
356 mportp->sel = dword.ed_u32[0];
361 EFSYS_PROBE1(fail1, efx_rc_t, rc);
365 __checkReturn efx_rc_t
366 efx_mae_match_spec_field_set(
367 __in efx_mae_match_spec_t *spec,
368 __in efx_mae_field_id_t field_id,
369 __in size_t value_size,
370 __in_bcount(value_size) const uint8_t *value,
371 __in size_t mask_size,
372 __in_bcount(mask_size) const uint8_t *mask)
374 const efx_mae_mv_desc_t *descp;
378 if (field_id >= EFX_MAE_FIELD_NIDS) {
383 switch (spec->emms_type) {
384 case EFX_MAE_RULE_ACTION:
385 descp = &__efx_mae_action_rule_mv_desc_set[field_id];
386 mvp = spec->emms_mask_value_pairs.action;
393 if (value_size != descp->emmd_value_size) {
398 if (mask_size != descp->emmd_mask_size) {
403 if (descp->emmd_endianness == EFX_MAE_FIELD_BE) {
405 * The mask/value are in network (big endian) order.
406 * The MCDI request field is also big endian.
408 memcpy(mvp + descp->emmd_value_offset, value, value_size);
409 memcpy(mvp + descp->emmd_mask_offset, mask, mask_size);
414 * The mask/value are in host byte order.
415 * The MCDI request field is little endian.
417 switch (value_size) {
419 EFX_POPULATE_DWORD_1(dword,
420 EFX_DWORD_0, *(const uint32_t *)value);
422 memcpy(mvp + descp->emmd_value_offset,
423 &dword, sizeof (dword));
426 EFSYS_ASSERT(B_FALSE);
431 EFX_POPULATE_DWORD_1(dword,
432 EFX_DWORD_0, *(const uint32_t *)mask);
434 memcpy(mvp + descp->emmd_mask_offset,
435 &dword, sizeof (dword));
438 EFSYS_ASSERT(B_FALSE);
451 EFSYS_PROBE1(fail1, efx_rc_t, rc);
455 __checkReturn efx_rc_t
456 efx_mae_match_spec_mport_set(
457 __in efx_mae_match_spec_t *spec,
458 __in const efx_mport_sel_t *valuep,
459 __in_opt const efx_mport_sel_t *maskp)
461 uint32_t full_mask = UINT32_MAX;
466 if (valuep == NULL) {
471 vp = (const uint8_t *)&valuep->sel;
473 mp = (const uint8_t *)&maskp->sel;
475 mp = (const uint8_t *)&full_mask;
477 rc = efx_mae_match_spec_field_set(spec,
478 EFX_MAE_FIELD_INGRESS_MPORT_SELECTOR,
479 sizeof (valuep->sel), vp, sizeof (maskp->sel), mp);
488 EFSYS_PROBE1(fail1, efx_rc_t, rc);
492 #define EFX_MASK_BIT_IS_SET(_mask, _mask_page_nbits, _bit) \
493 ((_mask)[(_bit) / (_mask_page_nbits)] & \
494 (1ULL << ((_bit) & ((_mask_page_nbits) - 1))))
496 static inline boolean_t
498 __in size_t mask_nbytes,
499 __in_bcount(mask_nbytes) const uint8_t *maskp)
501 boolean_t prev_bit_is_set = B_TRUE;
504 for (i = 0; i < 8 * mask_nbytes; ++i) {
505 boolean_t bit_is_set = EFX_MASK_BIT_IS_SET(maskp, 8, i);
507 if (!prev_bit_is_set && bit_is_set)
510 prev_bit_is_set = bit_is_set;
516 static inline boolean_t
517 efx_mask_is_all_ones(
518 __in size_t mask_nbytes,
519 __in_bcount(mask_nbytes) const uint8_t *maskp)
524 for (i = 0; i < mask_nbytes; ++i)
527 return (t == (uint8_t)(~0));
530 static inline boolean_t
531 efx_mask_is_all_zeros(
532 __in size_t mask_nbytes,
533 __in_bcount(mask_nbytes) const uint8_t *maskp)
538 for (i = 0; i < mask_nbytes; ++i)
544 __checkReturn boolean_t
545 efx_mae_match_spec_is_valid(
547 __in const efx_mae_match_spec_t *spec)
549 efx_mae_t *maep = enp->en_maep;
550 unsigned int field_ncaps = maep->em_max_nfields;
551 const efx_mae_field_cap_t *field_caps;
552 const efx_mae_mv_desc_t *desc_setp;
553 unsigned int desc_set_nentries;
554 boolean_t is_valid = B_TRUE;
555 efx_mae_field_id_t field_id;
558 switch (spec->emms_type) {
559 case EFX_MAE_RULE_ACTION:
560 field_caps = maep->em_action_rule_field_caps;
561 desc_setp = __efx_mae_action_rule_mv_desc_set;
563 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
564 mvp = spec->emms_mask_value_pairs.action;
570 if (field_caps == NULL)
573 for (field_id = 0; field_id < desc_set_nentries; ++field_id) {
574 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
575 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
576 const uint8_t *m_buf = mvp + descp->emmd_mask_offset;
577 size_t m_size = descp->emmd_mask_size;
580 continue; /* Skip array gap */
582 if (field_cap_id >= field_ncaps)
585 switch (field_caps[field_cap_id].emfc_support) {
586 case MAE_FIELD_SUPPORTED_MATCH_MASK:
589 case MAE_FIELD_SUPPORTED_MATCH_PREFIX:
590 is_valid = efx_mask_is_prefix(m_size, m_buf);
592 case MAE_FIELD_SUPPORTED_MATCH_OPTIONAL:
593 is_valid = (efx_mask_is_all_ones(m_size, m_buf) ||
594 efx_mask_is_all_zeros(m_size, m_buf));
596 case MAE_FIELD_SUPPORTED_MATCH_ALWAYS:
597 is_valid = efx_mask_is_all_ones(m_size, m_buf);
599 case MAE_FIELD_SUPPORTED_MATCH_NEVER:
600 case MAE_FIELD_UNSUPPORTED:
602 is_valid = efx_mask_is_all_zeros(m_size, m_buf);
606 if (is_valid == B_FALSE)
613 __checkReturn efx_rc_t
614 efx_mae_action_set_spec_init(
616 __out efx_mae_actions_t **specp)
618 efx_mae_actions_t *spec;
621 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
632 EFSYS_PROBE1(fail1, efx_rc_t, rc);
637 efx_mae_action_set_spec_fini(
639 __in efx_mae_actions_t *spec)
641 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
644 static __checkReturn efx_rc_t
645 efx_mae_action_set_add_vlan_pop(
646 __in efx_mae_actions_t *spec,
647 __in size_t arg_size,
648 __in_bcount(arg_size) const uint8_t *arg)
662 if (spec->ema_n_vlan_tags_to_pop == EFX_MAE_VLAN_POP_MAX_NTAGS) {
667 ++spec->ema_n_vlan_tags_to_pop;
676 EFSYS_PROBE1(fail1, efx_rc_t, rc);
680 static __checkReturn efx_rc_t
681 efx_mae_action_set_add_deliver(
682 __in efx_mae_actions_t *spec,
683 __in size_t arg_size,
684 __in_bcount(arg_size) const uint8_t *arg)
688 if (arg_size != sizeof (spec->ema_deliver_mport)) {
698 memcpy(&spec->ema_deliver_mport, arg, arg_size);
705 EFSYS_PROBE1(fail1, efx_rc_t, rc);
709 typedef struct efx_mae_action_desc_s {
710 /* Action specific handler */
711 efx_rc_t (*emad_add)(efx_mae_actions_t *,
712 size_t, const uint8_t *);
713 } efx_mae_action_desc_t;
715 static const efx_mae_action_desc_t efx_mae_actions[EFX_MAE_NACTIONS] = {
716 [EFX_MAE_ACTION_VLAN_POP] = {
717 .emad_add = efx_mae_action_set_add_vlan_pop
719 [EFX_MAE_ACTION_DELIVER] = {
720 .emad_add = efx_mae_action_set_add_deliver
724 static const uint32_t efx_mae_action_ordered_map =
725 (1U << EFX_MAE_ACTION_VLAN_POP) |
726 (1U << EFX_MAE_ACTION_DELIVER);
728 static const uint32_t efx_mae_action_repeat_map =
729 (1U << EFX_MAE_ACTION_VLAN_POP);
732 * Add an action to an action set.
734 * This has to be invoked in the desired action order.
735 * An out-of-order action request will be turned down.
737 static __checkReturn efx_rc_t
738 efx_mae_action_set_spec_populate(
739 __in efx_mae_actions_t *spec,
740 __in efx_mae_action_t type,
741 __in size_t arg_size,
742 __in_bcount(arg_size) const uint8_t *arg)
744 uint32_t action_mask;
747 EFX_STATIC_ASSERT(EFX_MAE_NACTIONS <=
748 (sizeof (efx_mae_action_ordered_map) * 8));
749 EFX_STATIC_ASSERT(EFX_MAE_NACTIONS <=
750 (sizeof (efx_mae_action_repeat_map) * 8));
752 EFX_STATIC_ASSERT(EFX_MAE_ACTION_DELIVER + 1 == EFX_MAE_NACTIONS);
754 if (type >= EFX_ARRAY_SIZE(efx_mae_actions)) {
759 action_mask = (1U << type);
761 if ((spec->ema_actions & action_mask) != 0) {
762 /* The action set already contains this action. */
763 if ((efx_mae_action_repeat_map & action_mask) == 0) {
764 /* Cannot add another non-repeatable action. */
770 if ((efx_mae_action_ordered_map & action_mask) != 0) {
771 uint32_t later_actions_mask =
772 efx_mae_action_ordered_map &
773 ~(action_mask | (action_mask - 1));
775 if ((spec->ema_actions & later_actions_mask) != 0) {
776 /* Cannot add an action after later ordered actions. */
782 if (efx_mae_actions[type].emad_add != NULL) {
783 rc = efx_mae_actions[type].emad_add(spec, arg_size, arg);
788 spec->ema_actions |= action_mask;
799 EFSYS_PROBE1(fail1, efx_rc_t, rc);
803 __checkReturn efx_rc_t
804 efx_mae_action_set_populate_vlan_pop(
805 __in efx_mae_actions_t *spec)
807 return (efx_mae_action_set_spec_populate(spec,
808 EFX_MAE_ACTION_VLAN_POP, 0, NULL));
811 __checkReturn efx_rc_t
812 efx_mae_action_set_populate_deliver(
813 __in efx_mae_actions_t *spec,
814 __in const efx_mport_sel_t *mportp)
819 if (mportp == NULL) {
824 arg = (const uint8_t *)&mportp->sel;
826 return (efx_mae_action_set_spec_populate(spec,
827 EFX_MAE_ACTION_DELIVER, sizeof (mportp->sel), arg));
830 EFSYS_PROBE1(fail1, efx_rc_t, rc);
834 __checkReturn boolean_t
835 efx_mae_action_set_specs_equal(
836 __in const efx_mae_actions_t *left,
837 __in const efx_mae_actions_t *right)
839 return ((memcmp(left, right, sizeof (*left)) == 0) ? B_TRUE : B_FALSE);
842 __checkReturn efx_rc_t
843 efx_mae_match_specs_class_cmp(
845 __in const efx_mae_match_spec_t *left,
846 __in const efx_mae_match_spec_t *right,
847 __out boolean_t *have_same_classp)
849 efx_mae_t *maep = enp->en_maep;
850 unsigned int field_ncaps = maep->em_max_nfields;
851 const efx_mae_field_cap_t *field_caps;
852 const efx_mae_mv_desc_t *desc_setp;
853 unsigned int desc_set_nentries;
854 boolean_t have_same_class = B_TRUE;
855 efx_mae_field_id_t field_id;
860 switch (left->emms_type) {
861 case EFX_MAE_RULE_ACTION:
862 field_caps = maep->em_action_rule_field_caps;
863 desc_setp = __efx_mae_action_rule_mv_desc_set;
865 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
866 mvpl = left->emms_mask_value_pairs.action;
867 mvpr = right->emms_mask_value_pairs.action;
874 if (field_caps == NULL) {
879 if (left->emms_type != right->emms_type ||
880 left->emms_prio != right->emms_prio) {
882 * Rules of different types can never map to the same class.
884 * The FW can support some set of match criteria for one
885 * priority and not support the very same set for
886 * another priority. Thus, two rules which have
887 * different priorities can never map to
890 *have_same_classp = B_FALSE;
894 for (field_id = 0; field_id < desc_set_nentries; ++field_id) {
895 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
896 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
898 if (descp->emmd_mask_size == 0)
899 continue; /* Skip array gap */
901 if (field_cap_id >= field_ncaps)
904 if (field_caps[field_cap_id].emfc_mask_affects_class) {
905 const uint8_t *lmaskp = mvpl + descp->emmd_mask_offset;
906 const uint8_t *rmaskp = mvpr + descp->emmd_mask_offset;
907 size_t mask_size = descp->emmd_mask_size;
909 if (memcmp(lmaskp, rmaskp, mask_size) != 0) {
910 have_same_class = B_FALSE;
915 if (field_caps[field_cap_id].emfc_match_affects_class) {
916 const uint8_t *lvalp = mvpl + descp->emmd_value_offset;
917 const uint8_t *rvalp = mvpr + descp->emmd_value_offset;
918 size_t value_size = descp->emmd_value_size;
920 if (memcmp(lvalp, rvalp, value_size) != 0) {
921 have_same_class = B_FALSE;
927 *have_same_classp = have_same_class;
934 EFSYS_PROBE1(fail1, efx_rc_t, rc);
938 __checkReturn efx_rc_t
939 efx_mae_action_set_alloc(
941 __in const efx_mae_actions_t *spec,
942 __out efx_mae_aset_id_t *aset_idp)
944 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
946 EFX_MCDI_DECLARE_BUF(payload,
947 MC_CMD_MAE_ACTION_SET_ALLOC_IN_LEN,
948 MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN);
949 efx_mae_aset_id_t aset_id;
952 if (encp->enc_mae_supported == B_FALSE) {
957 req.emr_cmd = MC_CMD_MAE_ACTION_SET_ALLOC;
958 req.emr_in_buf = payload;
959 req.emr_in_length = MC_CMD_MAE_ACTION_SET_ALLOC_IN_LEN;
960 req.emr_out_buf = payload;
961 req.emr_out_length = MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN;
964 * TODO: Remove these EFX_MAE_RSRC_ID_INVALID assignments once the
965 * corresponding resource types are supported by the implementation.
966 * Use proper resource ID assignments instead.
968 MCDI_IN_SET_DWORD(req,
969 MAE_ACTION_SET_ALLOC_IN_COUNTER_LIST_ID, EFX_MAE_RSRC_ID_INVALID);
970 MCDI_IN_SET_DWORD(req,
971 MAE_ACTION_SET_ALLOC_IN_COUNTER_ID, EFX_MAE_RSRC_ID_INVALID);
972 MCDI_IN_SET_DWORD(req,
973 MAE_ACTION_SET_ALLOC_IN_ENCAP_HEADER_ID, EFX_MAE_RSRC_ID_INVALID);
975 MCDI_IN_SET_DWORD_FIELD(req, MAE_ACTION_SET_ALLOC_IN_FLAGS,
976 MAE_ACTION_SET_ALLOC_IN_VLAN_POP, spec->ema_n_vlan_tags_to_pop);
978 MCDI_IN_SET_DWORD(req,
979 MAE_ACTION_SET_ALLOC_IN_DELIVER, spec->ema_deliver_mport.sel);
981 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_SRC_MAC_ID,
982 MC_CMD_MAE_MAC_ADDR_ALLOC_OUT_MAC_ID_NULL);
983 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_ALLOC_IN_DST_MAC_ID,
984 MC_CMD_MAE_MAC_ADDR_ALLOC_OUT_MAC_ID_NULL);
986 efx_mcdi_execute(enp, &req);
988 if (req.emr_rc != 0) {
993 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_SET_ALLOC_OUT_LEN) {
998 aset_id.id = MCDI_OUT_DWORD(req, MAE_ACTION_SET_ALLOC_OUT_AS_ID);
999 if (aset_id.id == EFX_MAE_RSRC_ID_INVALID) {
1004 aset_idp->id = aset_id.id;
1015 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1019 __checkReturn efx_rc_t
1020 efx_mae_action_set_free(
1021 __in efx_nic_t *enp,
1022 __in const efx_mae_aset_id_t *aset_idp)
1024 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1026 EFX_MCDI_DECLARE_BUF(payload,
1027 MC_CMD_MAE_ACTION_SET_FREE_IN_LEN(1),
1028 MC_CMD_MAE_ACTION_SET_FREE_OUT_LEN(1));
1031 if (encp->enc_mae_supported == B_FALSE) {
1036 req.emr_cmd = MC_CMD_MAE_ACTION_SET_FREE;
1037 req.emr_in_buf = payload;
1038 req.emr_in_length = MC_CMD_MAE_ACTION_SET_FREE_IN_LEN(1);
1039 req.emr_out_buf = payload;
1040 req.emr_out_length = MC_CMD_MAE_ACTION_SET_FREE_OUT_LEN(1);
1042 MCDI_IN_SET_DWORD(req, MAE_ACTION_SET_FREE_IN_AS_ID, aset_idp->id);
1044 efx_mcdi_execute(enp, &req);
1046 if (req.emr_rc != 0) {
1051 if (MCDI_OUT_DWORD(req, MAE_ACTION_SET_FREE_OUT_FREED_AS_ID) !=
1053 /* Firmware failed to free the action set. */
1065 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1069 __checkReturn efx_rc_t
1070 efx_mae_action_rule_insert(
1071 __in efx_nic_t *enp,
1072 __in const efx_mae_match_spec_t *spec,
1073 __in const efx_mae_aset_list_id_t *asl_idp,
1074 __in const efx_mae_aset_id_t *as_idp,
1075 __out efx_mae_rule_id_t *ar_idp)
1077 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1079 EFX_MCDI_DECLARE_BUF(payload,
1080 MC_CMD_MAE_ACTION_RULE_INSERT_IN_LENMAX_MCDI2,
1081 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN);
1082 efx_oword_t *rule_response;
1083 efx_mae_rule_id_t ar_id;
1087 EFX_STATIC_ASSERT(sizeof (ar_idp->id) ==
1088 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_AR_ID_LEN);
1090 EFX_STATIC_ASSERT(EFX_MAE_RSRC_ID_INVALID ==
1091 MC_CMD_MAE_ACTION_RULE_INSERT_OUT_ACTION_RULE_ID_NULL);
1093 if (encp->enc_mae_supported == B_FALSE) {
1098 if (spec->emms_type != EFX_MAE_RULE_ACTION ||
1099 (asl_idp != NULL && as_idp != NULL) ||
1100 (asl_idp == NULL && as_idp == NULL)) {
1105 req.emr_cmd = MC_CMD_MAE_ACTION_RULE_INSERT;
1106 req.emr_in_buf = payload;
1107 req.emr_in_length = MC_CMD_MAE_ACTION_RULE_INSERT_IN_LENMAX_MCDI2;
1108 req.emr_out_buf = payload;
1109 req.emr_out_length = MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN;
1111 EFX_STATIC_ASSERT(sizeof (*rule_response) <=
1112 MC_CMD_MAE_ACTION_RULE_INSERT_IN_RESPONSE_LEN);
1113 offset = MC_CMD_MAE_ACTION_RULE_INSERT_IN_RESPONSE_OFST;
1114 rule_response = (efx_oword_t *)(payload + offset);
1115 EFX_POPULATE_OWORD_3(*rule_response,
1116 MAE_ACTION_RULE_RESPONSE_ASL_ID,
1117 (asl_idp != NULL) ? asl_idp->id : EFX_MAE_RSRC_ID_INVALID,
1118 MAE_ACTION_RULE_RESPONSE_AS_ID,
1119 (as_idp != NULL) ? as_idp->id : EFX_MAE_RSRC_ID_INVALID,
1120 MAE_ACTION_RULE_RESPONSE_COUNTER_ID, EFX_MAE_RSRC_ID_INVALID);
1122 MCDI_IN_SET_DWORD(req, MAE_ACTION_RULE_INSERT_IN_PRIO, spec->emms_prio);
1125 * Mask-value pairs have been stored in the byte order needed for the
1126 * MCDI request and are thus safe to be copied directly to the buffer.
1128 EFX_STATIC_ASSERT(sizeof (spec->emms_mask_value_pairs.action) >=
1129 MAE_FIELD_MASK_VALUE_PAIRS_LEN);
1130 offset = MC_CMD_MAE_ACTION_RULE_INSERT_IN_MATCH_CRITERIA_OFST;
1131 memcpy(payload + offset, spec->emms_mask_value_pairs.action,
1132 MAE_FIELD_MASK_VALUE_PAIRS_LEN);
1134 efx_mcdi_execute(enp, &req);
1136 if (req.emr_rc != 0) {
1141 if (req.emr_out_length_used < MC_CMD_MAE_ACTION_RULE_INSERT_OUT_LEN) {
1146 ar_id.id = MCDI_OUT_DWORD(req, MAE_ACTION_RULE_INSERT_OUT_AR_ID);
1147 if (ar_id.id == EFX_MAE_RSRC_ID_INVALID) {
1152 ar_idp->id = ar_id.id;
1165 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1169 __checkReturn efx_rc_t
1170 efx_mae_action_rule_remove(
1171 __in efx_nic_t *enp,
1172 __in const efx_mae_rule_id_t *ar_idp)
1174 const efx_nic_cfg_t *encp = efx_nic_cfg_get(enp);
1176 EFX_MCDI_DECLARE_BUF(payload,
1177 MC_CMD_MAE_ACTION_RULE_DELETE_IN_LEN(1),
1178 MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LEN(1));
1181 if (encp->enc_mae_supported == B_FALSE) {
1186 req.emr_cmd = MC_CMD_MAE_ACTION_RULE_DELETE;
1187 req.emr_in_buf = payload;
1188 req.emr_in_length = MC_CMD_MAE_ACTION_RULE_DELETE_IN_LEN(1);
1189 req.emr_out_buf = payload;
1190 req.emr_out_length = MC_CMD_MAE_ACTION_RULE_DELETE_OUT_LEN(1);
1192 MCDI_IN_SET_DWORD(req, MAE_ACTION_RULE_DELETE_IN_AR_ID, ar_idp->id);
1194 efx_mcdi_execute(enp, &req);
1196 if (req.emr_rc != 0) {
1201 if (MCDI_OUT_DWORD(req, MAE_ACTION_RULE_DELETE_OUT_DELETED_AR_ID) !=
1203 /* Firmware failed to delete the action rule. */
1215 EFSYS_PROBE1(fail1, efx_rc_t, rc);
1219 #endif /* EFSYS_OPT_MAE */