Lines Matching +full:per +full:- +full:port +full:- +full:set
1 // SPDX-License-Identifier: (GPL-2.0 OR MIT)
2 /* Copyright 2020-2021 NXP
8 * Resource 0: Memory tracked per source port
9 * Resource 1: Frame references tracked per source port
10 * Resource 2: Memory tracked per destination port
11 * Resource 3: Frame references tracked per destination port
22 * Q_RSRV: reservation per QoS class per port
23 * PRIO_SHR: sharing watermark per QoS class across all ports
24 * P_RSRV: reservation per port
25 * COL_SHR: sharing watermark per color (drop precedence) across all ports
33 * ----------------------
35 * For setting up the reserved areas, egress watermarks exist per port and per
40 * | per QoS class
42 * | | | per egress port
47 #define BUF_Q_RSRV_E(port, prio) \ argument
48 (BUF_xxxx_E + xxx_Q_RSRV_x + OCELOT_NUM_TC * (port) + (prio))
51 * | for all port's traffic classes
53 * | | | per egress port
58 #define BUF_P_RSRV_E(port) \ argument
59 (BUF_xxxx_E + xxx_P_RSRV_x + (port))
62 * | per QoS class
64 * | | | per ingress port
69 #define BUF_Q_RSRV_I(port, prio) \ argument
70 (BUF_xxxx_I + xxx_Q_RSRV_x + OCELOT_NUM_TC * (port) + (prio))
73 * | for all port's traffic classes
75 * | | | per ingress port
80 #define BUF_P_RSRV_I(port) \ argument
81 (BUF_xxxx_I + xxx_P_RSRV_x + (port))
84 * | per QoS class
86 * | | | per egress port
91 #define REF_Q_RSRV_E(port, prio) \ argument
92 (REF_xxxx_E + xxx_Q_RSRV_x + OCELOT_NUM_TC * (port) + (prio))
95 * | for all port's traffic classes
97 * | | | per egress port
102 #define REF_P_RSRV_E(port) \ argument
103 (REF_xxxx_E + xxx_P_RSRV_x + (port))
106 * | per QoS class
108 * | | | per ingress port
113 #define REF_Q_RSRV_I(port, prio) \ argument
114 (REF_xxxx_I + xxx_Q_RSRV_x + OCELOT_NUM_TC * (port) + (prio))
117 * | for all port's traffic classes
119 * | | | per ingress port
124 #define REF_P_RSRV_I(port) \ argument
125 (REF_xxxx_I + xxx_P_RSRV_x + (port))
128 * ------------------
134 * | per QoS class
145 * | per color (drop precedence level)
153 (BUF_xxxx_E + xxx_COL_SHR_x + (1 - (dp)))
156 * | per QoS class
167 * | per color (drop precedence level)
175 (BUF_xxxx_I + xxx_COL_SHR_x + (1 - (dp)))
178 * | per QoS class
189 * | per color (drop precedence level)
197 (REF_xxxx_E + xxx_COL_SHR_x + (1 - (dp)))
200 * | per QoS class
211 * | per color (drop precedence level)
219 (REF_xxxx_I + xxx_COL_SHR_x + (1 - (dp)))
225 return ocelot->ops->wm_dec(wm); in ocelot_wm_read()
230 u32 wm = ocelot->ops->wm_enc(val); in ocelot_wm_write()
240 return ocelot->ops->wm_stat(res_stat, inuse, maxuse); in ocelot_wm_status()
249 * precedence 0. The user can still explicitly request reservations per port
250 * and per port-tc through devlink-sb.
253 int port) in ocelot_disable_reservation_watermarks() argument
258 ocelot_wm_write(ocelot, BUF_Q_RSRV_I(port, prio), 0); in ocelot_disable_reservation_watermarks()
259 ocelot_wm_write(ocelot, BUF_Q_RSRV_E(port, prio), 0); in ocelot_disable_reservation_watermarks()
260 ocelot_wm_write(ocelot, REF_Q_RSRV_I(port, prio), 0); in ocelot_disable_reservation_watermarks()
261 ocelot_wm_write(ocelot, REF_Q_RSRV_E(port, prio), 0); in ocelot_disable_reservation_watermarks()
264 ocelot_wm_write(ocelot, BUF_P_RSRV_I(port), 0); in ocelot_disable_reservation_watermarks()
265 ocelot_wm_write(ocelot, BUF_P_RSRV_E(port), 0); in ocelot_disable_reservation_watermarks()
266 ocelot_wm_write(ocelot, REF_P_RSRV_I(port), 0); in ocelot_disable_reservation_watermarks()
267 ocelot_wm_write(ocelot, REF_P_RSRV_E(port), 0); in ocelot_disable_reservation_watermarks()
274 * The switch has 10 sharing watermarks per lookup: 8 per traffic class and 2
275 * per color (drop precedence).
278 * (a) all 8 per-TC sharing watermarks to the max
279 * (b) all 2 per-color sharing watermarks to the max
281 * (a) all 8 per-TC sharing watermarks to "max / 8"
282 * (b) all 2 per-color sharing watermarks to "max / 2"
283 * So for Linux, let's just disable the sharing watermarks per traffic class
285 * sharing watermark for drop priority 0. So frames with drop priority set to 1
287 * the port and port-TC reservations are not exceeded.
304 int port, prio; in ocelot_get_buf_rsrv() local
309 for (port = 0; port <= ocelot->num_phys_ports; port++) { in ocelot_get_buf_rsrv()
312 BUF_Q_RSRV_I(port, prio)); in ocelot_get_buf_rsrv()
314 BUF_Q_RSRV_E(port, prio)); in ocelot_get_buf_rsrv()
317 *buf_rsrv_i += ocelot_wm_read(ocelot, BUF_P_RSRV_I(port)); in ocelot_get_buf_rsrv()
318 *buf_rsrv_e += ocelot_wm_read(ocelot, BUF_P_RSRV_E(port)); in ocelot_get_buf_rsrv()
328 int port, prio; in ocelot_get_ref_rsrv() local
333 for (port = 0; port <= ocelot->num_phys_ports; port++) { in ocelot_get_ref_rsrv()
336 REF_Q_RSRV_I(port, prio)); in ocelot_get_ref_rsrv()
338 REF_Q_RSRV_E(port, prio)); in ocelot_get_ref_rsrv()
341 *ref_rsrv_i += ocelot_wm_read(ocelot, REF_P_RSRV_I(port)); in ocelot_get_ref_rsrv()
342 *ref_rsrv_e += ocelot_wm_read(ocelot, REF_P_RSRV_E(port)); in ocelot_get_ref_rsrv()
346 /* Calculate all reservations, then set up the sharing watermark for DP=0 to
359 buf_shr_i = ocelot->pool_size[OCELOT_SB_BUF][OCELOT_SB_POOL_ING] - in ocelot_setup_sharing_watermarks()
361 buf_shr_e = ocelot->pool_size[OCELOT_SB_BUF][OCELOT_SB_POOL_EGR] - in ocelot_setup_sharing_watermarks()
363 ref_shr_i = ocelot->pool_size[OCELOT_SB_REF][OCELOT_SB_POOL_ING] - in ocelot_setup_sharing_watermarks()
365 ref_shr_e = ocelot->pool_size[OCELOT_SB_REF][OCELOT_SB_POOL_EGR] - in ocelot_setup_sharing_watermarks()
391 if (buf_rsrv_i > ocelot->pool_size[OCELOT_SB_BUF][OCELOT_SB_POOL_ING]) { in ocelot_watermark_validate()
394 return -ERANGE; in ocelot_watermark_validate()
396 if (buf_rsrv_e > ocelot->pool_size[OCELOT_SB_BUF][OCELOT_SB_POOL_EGR]) { in ocelot_watermark_validate()
399 return -ERANGE; in ocelot_watermark_validate()
401 if (ref_rsrv_i > ocelot->pool_size[OCELOT_SB_REF][OCELOT_SB_POOL_ING]) { in ocelot_watermark_validate()
404 return -ERANGE; in ocelot_watermark_validate()
406 if (ref_rsrv_e > ocelot->pool_size[OCELOT_SB_REF][OCELOT_SB_POOL_EGR]) { in ocelot_watermark_validate()
409 return -ERANGE; in ocelot_watermark_validate()
420 * +--------------------+--------------------+--------------------+
428 *(src port, prio) -+ (dst port, prio) -+ (src port, prio) -+ (dst port, prio) -+
433 * (src port) ----+ (dst port) ----+ (src port) ----+ (dst port) -----+
438 * (prio) ------+ (prio) ------+ (prio) ------+ (prio) -------+
443 * (dp) -------+ (dp) -------+ (dp) -------+ (dp) --------+
450 * +-----+----+ +-----+----+ +-----+----+ +-----+-----+
452 * +-------> OR <-------+ +-------> OR <-------+
455 * +----------------> AND <-----------------+
460 * We are modeling each of the 4 parallel lookups as a devlink-sb pool.
464 * The following watermarks are controlled explicitly through devlink-sb:
467 * The following watermarks are controlled implicitly through devlink-sb:
478 int all_tcs = GENMASK(OCELOT_NUM_TC - 1, 0); in ocelot_watermark_init()
479 int port; in ocelot_watermark_init() local
483 for (port = 0; port <= ocelot->num_phys_ports; port++) in ocelot_watermark_init()
484 ocelot_disable_reservation_watermarks(ocelot, port); in ocelot_watermark_init()
510 return -ENODEV; in ocelot_sb_pool_get()
512 return -ENODEV; in ocelot_sb_pool_get()
515 pool_info->size = ocelot->pool_size[sb_index][pool_index]; in ocelot_sb_pool_get()
517 pool_info->pool_type = DEVLINK_SB_POOL_TYPE_INGRESS; in ocelot_sb_pool_get()
519 pool_info->pool_type = DEVLINK_SB_POOL_TYPE_EGRESS; in ocelot_sb_pool_get()
527 * the values for the port and port-tc reservations, is written into the
541 return -ENODEV; in ocelot_sb_pool_set()
546 return -ENODEV; in ocelot_sb_pool_set()
551 return -EOPNOTSUPP; in ocelot_sb_pool_set()
554 old_pool_size = ocelot->pool_size[sb_index][pool_index]; in ocelot_sb_pool_set()
555 ocelot->pool_size[sb_index][pool_index] = size; in ocelot_sb_pool_set()
559 ocelot->pool_size[sb_index][pool_index] = old_pool_size; in ocelot_sb_pool_set()
570 int ocelot_sb_port_pool_get(struct ocelot *ocelot, int port, in ocelot_sb_port_pool_get() argument
579 wm_index = BUF_P_RSRV_I(port); in ocelot_sb_port_pool_get()
581 wm_index = BUF_P_RSRV_E(port); in ocelot_sb_port_pool_get()
585 wm_index = REF_P_RSRV_I(port); in ocelot_sb_port_pool_get()
587 wm_index = REF_P_RSRV_E(port); in ocelot_sb_port_pool_get()
590 return -ENODEV; in ocelot_sb_port_pool_get()
600 /* This configures the P_RSRV per-port reserved resource watermark */
601 int ocelot_sb_port_pool_set(struct ocelot *ocelot, int port, in ocelot_sb_port_pool_set() argument
611 wm_index = BUF_P_RSRV_I(port); in ocelot_sb_port_pool_set()
613 wm_index = BUF_P_RSRV_E(port); in ocelot_sb_port_pool_set()
617 wm_index = REF_P_RSRV_I(port); in ocelot_sb_port_pool_set()
619 wm_index = REF_P_RSRV_E(port); in ocelot_sb_port_pool_set()
623 return -ENODEV; in ocelot_sb_port_pool_set()
644 int ocelot_sb_tc_pool_bind_get(struct ocelot *ocelot, int port, in ocelot_sb_tc_pool_bind_get() argument
654 wm_index = BUF_Q_RSRV_I(port, tc_index); in ocelot_sb_tc_pool_bind_get()
656 wm_index = BUF_Q_RSRV_E(port, tc_index); in ocelot_sb_tc_pool_bind_get()
660 wm_index = REF_Q_RSRV_I(port, tc_index); in ocelot_sb_tc_pool_bind_get()
662 wm_index = REF_Q_RSRV_E(port, tc_index); in ocelot_sb_tc_pool_bind_get()
665 return -ENODEV; in ocelot_sb_tc_pool_bind_get()
680 /* This configures the Q_RSRV per-port-tc reserved resource watermark */
681 int ocelot_sb_tc_pool_bind_set(struct ocelot *ocelot, int port, in ocelot_sb_tc_pool_bind_set() argument
693 return -EINVAL; in ocelot_sb_tc_pool_bind_set()
696 return -EINVAL; in ocelot_sb_tc_pool_bind_set()
701 wm_index = BUF_Q_RSRV_I(port, tc_index); in ocelot_sb_tc_pool_bind_set()
703 wm_index = BUF_Q_RSRV_E(port, tc_index); in ocelot_sb_tc_pool_bind_set()
707 wm_index = REF_Q_RSRV_I(port, tc_index); in ocelot_sb_tc_pool_bind_set()
709 wm_index = REF_Q_RSRV_E(port, tc_index); in ocelot_sb_tc_pool_bind_set()
713 return -ENODEV; in ocelot_sb_tc_pool_bind_set()
747 int port, prio; in ocelot_sb_occ_max_clear() local
751 for (port = 0; port <= ocelot->num_phys_ports; port++) { in ocelot_sb_occ_max_clear()
753 ocelot_wm_status(ocelot, BUF_Q_RSRV_I(port, prio), in ocelot_sb_occ_max_clear()
755 ocelot_wm_status(ocelot, BUF_Q_RSRV_E(port, prio), in ocelot_sb_occ_max_clear()
758 ocelot_wm_status(ocelot, BUF_P_RSRV_I(port), in ocelot_sb_occ_max_clear()
760 ocelot_wm_status(ocelot, BUF_P_RSRV_E(port), in ocelot_sb_occ_max_clear()
765 for (port = 0; port <= ocelot->num_phys_ports; port++) { in ocelot_sb_occ_max_clear()
767 ocelot_wm_status(ocelot, REF_Q_RSRV_I(port, prio), in ocelot_sb_occ_max_clear()
769 ocelot_wm_status(ocelot, REF_Q_RSRV_E(port, prio), in ocelot_sb_occ_max_clear()
772 ocelot_wm_status(ocelot, REF_P_RSRV_I(port), in ocelot_sb_occ_max_clear()
774 ocelot_wm_status(ocelot, REF_P_RSRV_E(port), in ocelot_sb_occ_max_clear()
779 return -ENODEV; in ocelot_sb_occ_max_clear()
786 /* This retrieves the watermark occupancy for per-port P_RSRV watermarks */
787 int ocelot_sb_occ_port_pool_get(struct ocelot *ocelot, int port, in ocelot_sb_occ_port_pool_get() argument
796 wm_index = BUF_P_RSRV_I(port); in ocelot_sb_occ_port_pool_get()
798 wm_index = BUF_P_RSRV_E(port); in ocelot_sb_occ_port_pool_get()
802 wm_index = REF_P_RSRV_I(port); in ocelot_sb_occ_port_pool_get()
804 wm_index = REF_P_RSRV_E(port); in ocelot_sb_occ_port_pool_get()
807 return -ENODEV; in ocelot_sb_occ_port_pool_get()
818 /* This retrieves the watermark occupancy for per-port-tc Q_RSRV watermarks */
819 int ocelot_sb_occ_tc_port_bind_get(struct ocelot *ocelot, int port, in ocelot_sb_occ_tc_port_bind_get() argument
829 wm_index = BUF_Q_RSRV_I(port, tc_index); in ocelot_sb_occ_tc_port_bind_get()
831 wm_index = BUF_Q_RSRV_E(port, tc_index); in ocelot_sb_occ_tc_port_bind_get()
835 wm_index = REF_Q_RSRV_I(port, tc_index); in ocelot_sb_occ_tc_port_bind_get()
837 wm_index = REF_Q_RSRV_E(port, tc_index); in ocelot_sb_occ_tc_port_bind_get()
840 return -ENODEV; in ocelot_sb_occ_tc_port_bind_get()
855 err = devlink_sb_register(ocelot->devlink, OCELOT_SB_BUF, in ocelot_devlink_sb_register()
856 ocelot->packet_buffer_size, 1, 1, in ocelot_devlink_sb_register()
861 err = devlink_sb_register(ocelot->devlink, OCELOT_SB_REF, in ocelot_devlink_sb_register()
862 ocelot->num_frame_refs, 1, 1, in ocelot_devlink_sb_register()
865 devlink_sb_unregister(ocelot->devlink, OCELOT_SB_BUF); in ocelot_devlink_sb_register()
869 ocelot->pool_size[OCELOT_SB_BUF][OCELOT_SB_POOL_ING] = ocelot->packet_buffer_size; in ocelot_devlink_sb_register()
870 ocelot->pool_size[OCELOT_SB_BUF][OCELOT_SB_POOL_EGR] = ocelot->packet_buffer_size; in ocelot_devlink_sb_register()
871 ocelot->pool_size[OCELOT_SB_REF][OCELOT_SB_POOL_ING] = ocelot->num_frame_refs; in ocelot_devlink_sb_register()
872 ocelot->pool_size[OCELOT_SB_REF][OCELOT_SB_POOL_EGR] = ocelot->num_frame_refs; in ocelot_devlink_sb_register()
882 devlink_sb_unregister(ocelot->devlink, OCELOT_SB_BUF); in ocelot_devlink_sb_unregister()
883 devlink_sb_unregister(ocelot->devlink, OCELOT_SB_REF); in ocelot_devlink_sb_unregister()