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,
287 EFX_MAE_FIELD_CAP_NIDS
288 } efx_mae_field_cap_id_t;
290 typedef enum efx_mae_field_endianness_e {
291 EFX_MAE_FIELD_LE = 0,
294 EFX_MAE_FIELD_ENDIANNESS_NTYPES
295 } efx_mae_field_endianness_t;
298 * The following structure is a means to describe an MAE field.
299 * The information in it is meant to be used internally by
300 * APIs for addressing a given field in a mask-value pairs
301 * structure and for validation purposes.
303 typedef struct efx_mae_mv_desc_s {
304 efx_mae_field_cap_id_t emmd_field_cap_id;
306 size_t emmd_value_size;
307 size_t emmd_value_offset;
308 size_t emmd_mask_size;
309 size_t emmd_mask_offset;
311 efx_mae_field_endianness_t emmd_endianness;
314 /* Indices to this array are provided by efx_mae_field_id_t */
315 static const efx_mae_mv_desc_t __efx_mae_action_rule_mv_desc_set[] = {
316 #define EFX_MAE_MV_DESC(_name, _endianness) \
317 [EFX_MAE_FIELD_##_name] = \
319 EFX_MAE_FIELD_ID_##_name, \
320 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_LEN, \
321 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_OFST, \
322 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_MASK_LEN, \
323 MAE_FIELD_MASK_VALUE_PAIRS_##_name##_MASK_OFST, \
327 EFX_MAE_MV_DESC(INGRESS_MPORT_SELECTOR, EFX_MAE_FIELD_LE),
329 #undef EFX_MAE_MV_DESC
332 __checkReturn efx_rc_t
333 efx_mae_mport_by_phy_port(
334 __in uint32_t phy_port,
335 __out efx_mport_sel_t *mportp)
340 if (phy_port > EFX_MASK32(MAE_MPORT_SELECTOR_PPORT_ID)) {
345 EFX_POPULATE_DWORD_2(dword,
346 MAE_MPORT_SELECTOR_TYPE, MAE_MPORT_SELECTOR_TYPE_PPORT,
347 MAE_MPORT_SELECTOR_PPORT_ID, phy_port);
349 memset(mportp, 0, sizeof (*mportp));
350 mportp->sel = dword.ed_u32[0];
355 EFSYS_PROBE1(fail1, efx_rc_t, rc);
359 __checkReturn efx_rc_t
360 efx_mae_match_spec_field_set(
361 __in efx_mae_match_spec_t *spec,
362 __in efx_mae_field_id_t field_id,
363 __in size_t value_size,
364 __in_bcount(value_size) const uint8_t *value,
365 __in size_t mask_size,
366 __in_bcount(mask_size) const uint8_t *mask)
368 const efx_mae_mv_desc_t *descp;
372 if (field_id >= EFX_MAE_FIELD_NIDS) {
377 switch (spec->emms_type) {
378 case EFX_MAE_RULE_ACTION:
379 descp = &__efx_mae_action_rule_mv_desc_set[field_id];
380 mvp = spec->emms_mask_value_pairs.action;
387 if (value_size != descp->emmd_value_size) {
392 if (mask_size != descp->emmd_mask_size) {
397 if (descp->emmd_endianness == EFX_MAE_FIELD_BE) {
399 * The mask/value are in network (big endian) order.
400 * The MCDI request field is also big endian.
402 memcpy(mvp + descp->emmd_value_offset, value, value_size);
403 memcpy(mvp + descp->emmd_mask_offset, mask, mask_size);
408 * The mask/value are in host byte order.
409 * The MCDI request field is little endian.
411 switch (value_size) {
413 EFX_POPULATE_DWORD_1(dword,
414 EFX_DWORD_0, *(const uint32_t *)value);
416 memcpy(mvp + descp->emmd_value_offset,
417 &dword, sizeof (dword));
420 EFSYS_ASSERT(B_FALSE);
425 EFX_POPULATE_DWORD_1(dword,
426 EFX_DWORD_0, *(const uint32_t *)mask);
428 memcpy(mvp + descp->emmd_mask_offset,
429 &dword, sizeof (dword));
432 EFSYS_ASSERT(B_FALSE);
445 EFSYS_PROBE1(fail1, efx_rc_t, rc);
449 __checkReturn efx_rc_t
450 efx_mae_match_spec_mport_set(
451 __in efx_mae_match_spec_t *spec,
452 __in const efx_mport_sel_t *valuep,
453 __in_opt const efx_mport_sel_t *maskp)
455 uint32_t full_mask = UINT32_MAX;
460 if (valuep == NULL) {
465 vp = (const uint8_t *)&valuep->sel;
467 mp = (const uint8_t *)&maskp->sel;
469 mp = (const uint8_t *)&full_mask;
471 rc = efx_mae_match_spec_field_set(spec,
472 EFX_MAE_FIELD_INGRESS_MPORT_SELECTOR,
473 sizeof (valuep->sel), vp, sizeof (maskp->sel), mp);
482 EFSYS_PROBE1(fail1, efx_rc_t, rc);
486 #define EFX_MASK_BIT_IS_SET(_mask, _mask_page_nbits, _bit) \
487 ((_mask)[(_bit) / (_mask_page_nbits)] & \
488 (1ULL << ((_bit) & ((_mask_page_nbits) - 1))))
490 static inline boolean_t
492 __in size_t mask_nbytes,
493 __in_bcount(mask_nbytes) const uint8_t *maskp)
495 boolean_t prev_bit_is_set = B_TRUE;
498 for (i = 0; i < 8 * mask_nbytes; ++i) {
499 boolean_t bit_is_set = EFX_MASK_BIT_IS_SET(maskp, 8, i);
501 if (!prev_bit_is_set && bit_is_set)
504 prev_bit_is_set = bit_is_set;
510 static inline boolean_t
511 efx_mask_is_all_ones(
512 __in size_t mask_nbytes,
513 __in_bcount(mask_nbytes) const uint8_t *maskp)
518 for (i = 0; i < mask_nbytes; ++i)
521 return (t == (uint8_t)(~0));
524 static inline boolean_t
525 efx_mask_is_all_zeros(
526 __in size_t mask_nbytes,
527 __in_bcount(mask_nbytes) const uint8_t *maskp)
532 for (i = 0; i < mask_nbytes; ++i)
538 __checkReturn boolean_t
539 efx_mae_match_spec_is_valid(
541 __in const efx_mae_match_spec_t *spec)
543 efx_mae_t *maep = enp->en_maep;
544 unsigned int field_ncaps = maep->em_max_nfields;
545 const efx_mae_field_cap_t *field_caps;
546 const efx_mae_mv_desc_t *desc_setp;
547 unsigned int desc_set_nentries;
548 boolean_t is_valid = B_TRUE;
549 efx_mae_field_id_t field_id;
552 switch (spec->emms_type) {
553 case EFX_MAE_RULE_ACTION:
554 field_caps = maep->em_action_rule_field_caps;
555 desc_setp = __efx_mae_action_rule_mv_desc_set;
557 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
558 mvp = spec->emms_mask_value_pairs.action;
564 if (field_caps == NULL)
567 for (field_id = 0; field_id < desc_set_nentries; ++field_id) {
568 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
569 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
570 const uint8_t *m_buf = mvp + descp->emmd_mask_offset;
571 size_t m_size = descp->emmd_mask_size;
574 continue; /* Skip array gap */
576 if (field_cap_id >= field_ncaps)
579 switch (field_caps[field_cap_id].emfc_support) {
580 case MAE_FIELD_SUPPORTED_MATCH_MASK:
583 case MAE_FIELD_SUPPORTED_MATCH_PREFIX:
584 is_valid = efx_mask_is_prefix(m_size, m_buf);
586 case MAE_FIELD_SUPPORTED_MATCH_OPTIONAL:
587 is_valid = (efx_mask_is_all_ones(m_size, m_buf) ||
588 efx_mask_is_all_zeros(m_size, m_buf));
590 case MAE_FIELD_SUPPORTED_MATCH_ALWAYS:
591 is_valid = efx_mask_is_all_ones(m_size, m_buf);
593 case MAE_FIELD_SUPPORTED_MATCH_NEVER:
594 case MAE_FIELD_UNSUPPORTED:
596 is_valid = efx_mask_is_all_zeros(m_size, m_buf);
600 if (is_valid == B_FALSE)
607 __checkReturn efx_rc_t
608 efx_mae_action_set_spec_init(
610 __out efx_mae_actions_t **specp)
612 efx_mae_actions_t *spec;
615 EFSYS_KMEM_ALLOC(enp->en_esip, sizeof (*spec), spec);
626 EFSYS_PROBE1(fail1, efx_rc_t, rc);
631 efx_mae_action_set_spec_fini(
633 __in efx_mae_actions_t *spec)
635 EFSYS_KMEM_FREE(enp->en_esip, sizeof (*spec), spec);
638 __checkReturn boolean_t
639 efx_mae_action_set_specs_equal(
640 __in const efx_mae_actions_t *left,
641 __in const efx_mae_actions_t *right)
643 return ((memcmp(left, right, sizeof (*left)) == 0) ? B_TRUE : B_FALSE);
646 __checkReturn efx_rc_t
647 efx_mae_match_specs_class_cmp(
649 __in const efx_mae_match_spec_t *left,
650 __in const efx_mae_match_spec_t *right,
651 __out boolean_t *have_same_classp)
653 efx_mae_t *maep = enp->en_maep;
654 unsigned int field_ncaps = maep->em_max_nfields;
655 const efx_mae_field_cap_t *field_caps;
656 const efx_mae_mv_desc_t *desc_setp;
657 unsigned int desc_set_nentries;
658 boolean_t have_same_class = B_TRUE;
659 efx_mae_field_id_t field_id;
664 switch (left->emms_type) {
665 case EFX_MAE_RULE_ACTION:
666 field_caps = maep->em_action_rule_field_caps;
667 desc_setp = __efx_mae_action_rule_mv_desc_set;
669 EFX_ARRAY_SIZE(__efx_mae_action_rule_mv_desc_set);
670 mvpl = left->emms_mask_value_pairs.action;
671 mvpr = right->emms_mask_value_pairs.action;
678 if (field_caps == NULL) {
683 if (left->emms_type != right->emms_type ||
684 left->emms_prio != right->emms_prio) {
686 * Rules of different types can never map to the same class.
688 * The FW can support some set of match criteria for one
689 * priority and not support the very same set for
690 * another priority. Thus, two rules which have
691 * different priorities can never map to
694 *have_same_classp = B_FALSE;
698 for (field_id = 0; field_id < desc_set_nentries; ++field_id) {
699 const efx_mae_mv_desc_t *descp = &desc_setp[field_id];
700 efx_mae_field_cap_id_t field_cap_id = descp->emmd_field_cap_id;
702 if (descp->emmd_mask_size == 0)
703 continue; /* Skip array gap */
705 if (field_cap_id >= field_ncaps)
708 if (field_caps[field_cap_id].emfc_mask_affects_class) {
709 const uint8_t *lmaskp = mvpl + descp->emmd_mask_offset;
710 const uint8_t *rmaskp = mvpr + descp->emmd_mask_offset;
711 size_t mask_size = descp->emmd_mask_size;
713 if (memcmp(lmaskp, rmaskp, mask_size) != 0) {
714 have_same_class = B_FALSE;
719 if (field_caps[field_cap_id].emfc_match_affects_class) {
720 const uint8_t *lvalp = mvpl + descp->emmd_value_offset;
721 const uint8_t *rvalp = mvpr + descp->emmd_value_offset;
722 size_t value_size = descp->emmd_value_size;
724 if (memcmp(lvalp, rvalp, value_size) != 0) {
725 have_same_class = B_FALSE;
731 *have_same_classp = have_same_class;
738 EFSYS_PROBE1(fail1, efx_rc_t, rc);
742 #endif /* EFSYS_OPT_MAE */