18127224cSIvan Khoronzhuk // SPDX-License-Identifier: GPL-2.0
28127224cSIvan Khoronzhuk /* Texas Instruments K3 AM65 Ethernet QoS submodule
38127224cSIvan Khoronzhuk * Copyright (C) 2020 Texas Instruments Incorporated - http://www.ti.com/
48127224cSIvan Khoronzhuk *
58127224cSIvan Khoronzhuk * quality of service module includes:
68127224cSIvan Khoronzhuk * Enhanced Scheduler Traffic (EST - P802.1Qbv/D2.2)
78127224cSIvan Khoronzhuk */
88127224cSIvan Khoronzhuk
98127224cSIvan Khoronzhuk #include <linux/pm_runtime.h>
108127224cSIvan Khoronzhuk #include <linux/time.h>
115ec836beSGrygorii Strashko #include <net/pkt_cls.h>
128127224cSIvan Khoronzhuk
138127224cSIvan Khoronzhuk #include "am65-cpsw-nuss.h"
148127224cSIvan Khoronzhuk #include "am65-cpsw-qos.h"
158127224cSIvan Khoronzhuk #include "am65-cpts.h"
165ec836beSGrygorii Strashko #include "cpsw_ale.h"
178127224cSIvan Khoronzhuk
188127224cSIvan Khoronzhuk #define AM65_CPSW_REG_CTL 0x004
198127224cSIvan Khoronzhuk #define AM65_CPSW_PN_REG_CTL 0x004
208127224cSIvan Khoronzhuk #define AM65_CPSW_PN_REG_FIFO_STATUS 0x050
218127224cSIvan Khoronzhuk #define AM65_CPSW_PN_REG_EST_CTL 0x060
225c8560c4SGrygorii Strashko #define AM65_CPSW_PN_REG_PRI_CIR(pri) (0x140 + 4 * (pri))
238127224cSIvan Khoronzhuk
248127224cSIvan Khoronzhuk /* AM65_CPSW_REG_CTL register fields */
258127224cSIvan Khoronzhuk #define AM65_CPSW_CTL_EST_EN BIT(18)
268127224cSIvan Khoronzhuk
278127224cSIvan Khoronzhuk /* AM65_CPSW_PN_REG_CTL register fields */
288127224cSIvan Khoronzhuk #define AM65_CPSW_PN_CTL_EST_PORT_EN BIT(17)
298127224cSIvan Khoronzhuk
308127224cSIvan Khoronzhuk /* AM65_CPSW_PN_REG_EST_CTL register fields */
318127224cSIvan Khoronzhuk #define AM65_CPSW_PN_EST_ONEBUF BIT(0)
328127224cSIvan Khoronzhuk #define AM65_CPSW_PN_EST_BUFSEL BIT(1)
338127224cSIvan Khoronzhuk #define AM65_CPSW_PN_EST_TS_EN BIT(2)
348127224cSIvan Khoronzhuk #define AM65_CPSW_PN_EST_TS_FIRST BIT(3)
358127224cSIvan Khoronzhuk #define AM65_CPSW_PN_EST_ONEPRI BIT(4)
368127224cSIvan Khoronzhuk #define AM65_CPSW_PN_EST_TS_PRI_MSK GENMASK(7, 5)
378127224cSIvan Khoronzhuk
388127224cSIvan Khoronzhuk /* AM65_CPSW_PN_REG_FIFO_STATUS register fields */
398127224cSIvan Khoronzhuk #define AM65_CPSW_PN_FST_TX_PRI_ACTIVE_MSK GENMASK(7, 0)
408127224cSIvan Khoronzhuk #define AM65_CPSW_PN_FST_TX_E_MAC_ALLOW_MSK GENMASK(15, 8)
418127224cSIvan Khoronzhuk #define AM65_CPSW_PN_FST_EST_CNT_ERR BIT(16)
428127224cSIvan Khoronzhuk #define AM65_CPSW_PN_FST_EST_ADD_ERR BIT(17)
438127224cSIvan Khoronzhuk #define AM65_CPSW_PN_FST_EST_BUFACT BIT(18)
448127224cSIvan Khoronzhuk
458127224cSIvan Khoronzhuk /* EST FETCH COMMAND RAM */
468127224cSIvan Khoronzhuk #define AM65_CPSW_FETCH_RAM_CMD_NUM 0x80
478127224cSIvan Khoronzhuk #define AM65_CPSW_FETCH_CNT_MSK GENMASK(21, 8)
488127224cSIvan Khoronzhuk #define AM65_CPSW_FETCH_CNT_MAX (AM65_CPSW_FETCH_CNT_MSK >> 8)
498127224cSIvan Khoronzhuk #define AM65_CPSW_FETCH_CNT_OFFSET 8
508127224cSIvan Khoronzhuk #define AM65_CPSW_FETCH_ALLOW_MSK GENMASK(7, 0)
518127224cSIvan Khoronzhuk #define AM65_CPSW_FETCH_ALLOW_MAX AM65_CPSW_FETCH_ALLOW_MSK
528127224cSIvan Khoronzhuk
538127224cSIvan Khoronzhuk enum timer_act {
548127224cSIvan Khoronzhuk TACT_PROG, /* need program timer */
558127224cSIvan Khoronzhuk TACT_NEED_STOP, /* need stop first */
568127224cSIvan Khoronzhuk TACT_SKIP_PROG, /* just buffer can be updated */
578127224cSIvan Khoronzhuk };
588127224cSIvan Khoronzhuk
am65_cpsw_port_est_enabled(struct am65_cpsw_port * port)598127224cSIvan Khoronzhuk static int am65_cpsw_port_est_enabled(struct am65_cpsw_port *port)
608127224cSIvan Khoronzhuk {
618127224cSIvan Khoronzhuk return port->qos.est_oper || port->qos.est_admin;
628127224cSIvan Khoronzhuk }
638127224cSIvan Khoronzhuk
am65_cpsw_est_enable(struct am65_cpsw_common * common,int enable)648127224cSIvan Khoronzhuk static void am65_cpsw_est_enable(struct am65_cpsw_common *common, int enable)
658127224cSIvan Khoronzhuk {
668127224cSIvan Khoronzhuk u32 val;
678127224cSIvan Khoronzhuk
688127224cSIvan Khoronzhuk val = readl(common->cpsw_base + AM65_CPSW_REG_CTL);
698127224cSIvan Khoronzhuk
708127224cSIvan Khoronzhuk if (enable)
718127224cSIvan Khoronzhuk val |= AM65_CPSW_CTL_EST_EN;
728127224cSIvan Khoronzhuk else
738127224cSIvan Khoronzhuk val &= ~AM65_CPSW_CTL_EST_EN;
748127224cSIvan Khoronzhuk
758127224cSIvan Khoronzhuk writel(val, common->cpsw_base + AM65_CPSW_REG_CTL);
768127224cSIvan Khoronzhuk common->est_enabled = enable;
778127224cSIvan Khoronzhuk }
788127224cSIvan Khoronzhuk
am65_cpsw_port_est_enable(struct am65_cpsw_port * port,int enable)798127224cSIvan Khoronzhuk static void am65_cpsw_port_est_enable(struct am65_cpsw_port *port, int enable)
808127224cSIvan Khoronzhuk {
818127224cSIvan Khoronzhuk u32 val;
828127224cSIvan Khoronzhuk
838127224cSIvan Khoronzhuk val = readl(port->port_base + AM65_CPSW_PN_REG_CTL);
848127224cSIvan Khoronzhuk if (enable)
858127224cSIvan Khoronzhuk val |= AM65_CPSW_PN_CTL_EST_PORT_EN;
868127224cSIvan Khoronzhuk else
878127224cSIvan Khoronzhuk val &= ~AM65_CPSW_PN_CTL_EST_PORT_EN;
888127224cSIvan Khoronzhuk
898127224cSIvan Khoronzhuk writel(val, port->port_base + AM65_CPSW_PN_REG_CTL);
908127224cSIvan Khoronzhuk }
918127224cSIvan Khoronzhuk
928127224cSIvan Khoronzhuk /* target new EST RAM buffer, actual toggle happens after cycle completion */
am65_cpsw_port_est_assign_buf_num(struct net_device * ndev,int buf_num)938127224cSIvan Khoronzhuk static void am65_cpsw_port_est_assign_buf_num(struct net_device *ndev,
948127224cSIvan Khoronzhuk int buf_num)
958127224cSIvan Khoronzhuk {
968127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
978127224cSIvan Khoronzhuk u32 val;
988127224cSIvan Khoronzhuk
998127224cSIvan Khoronzhuk val = readl(port->port_base + AM65_CPSW_PN_REG_EST_CTL);
1008127224cSIvan Khoronzhuk if (buf_num)
1018127224cSIvan Khoronzhuk val |= AM65_CPSW_PN_EST_BUFSEL;
1028127224cSIvan Khoronzhuk else
1038127224cSIvan Khoronzhuk val &= ~AM65_CPSW_PN_EST_BUFSEL;
1048127224cSIvan Khoronzhuk
1058127224cSIvan Khoronzhuk writel(val, port->port_base + AM65_CPSW_PN_REG_EST_CTL);
1068127224cSIvan Khoronzhuk }
1078127224cSIvan Khoronzhuk
1088127224cSIvan Khoronzhuk /* am65_cpsw_port_est_is_swapped() - Indicate if h/w is transitioned
1098127224cSIvan Khoronzhuk * admin -> oper or not
1108127224cSIvan Khoronzhuk *
1118127224cSIvan Khoronzhuk * Return true if already transitioned. i.e oper is equal to admin and buf
1128127224cSIvan Khoronzhuk * numbers match (est_oper->buf match with est_admin->buf).
1138127224cSIvan Khoronzhuk * false if before transition. i.e oper is not equal to admin, (i.e a
1148127224cSIvan Khoronzhuk * previous admin command is waiting to be transitioned to oper state
1158127224cSIvan Khoronzhuk * and est_oper->buf not match with est_oper->buf).
1168127224cSIvan Khoronzhuk */
am65_cpsw_port_est_is_swapped(struct net_device * ndev,int * oper,int * admin)1178127224cSIvan Khoronzhuk static int am65_cpsw_port_est_is_swapped(struct net_device *ndev, int *oper,
1188127224cSIvan Khoronzhuk int *admin)
1198127224cSIvan Khoronzhuk {
1208127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
1218127224cSIvan Khoronzhuk u32 val;
1228127224cSIvan Khoronzhuk
1238127224cSIvan Khoronzhuk val = readl(port->port_base + AM65_CPSW_PN_REG_FIFO_STATUS);
1248127224cSIvan Khoronzhuk *oper = !!(val & AM65_CPSW_PN_FST_EST_BUFACT);
1258127224cSIvan Khoronzhuk
1268127224cSIvan Khoronzhuk val = readl(port->port_base + AM65_CPSW_PN_REG_EST_CTL);
1278127224cSIvan Khoronzhuk *admin = !!(val & AM65_CPSW_PN_EST_BUFSEL);
1288127224cSIvan Khoronzhuk
1298127224cSIvan Khoronzhuk return *admin == *oper;
1308127224cSIvan Khoronzhuk }
1318127224cSIvan Khoronzhuk
1328127224cSIvan Khoronzhuk /* am65_cpsw_port_est_get_free_buf_num() - Get free buffer number for
1338127224cSIvan Khoronzhuk * Admin to program the new schedule.
1348127224cSIvan Khoronzhuk *
1358127224cSIvan Khoronzhuk * Logic as follows:-
1368127224cSIvan Khoronzhuk * If oper is same as admin, return the other buffer (!oper) as the admin
1378127224cSIvan Khoronzhuk * buffer. If oper is not the same, driver let the current oper to continue
1388127224cSIvan Khoronzhuk * as it is in the process of transitioning from admin -> oper. So keep the
1398127224cSIvan Khoronzhuk * oper by selecting the same oper buffer by writing to EST_BUFSEL bit in
1408127224cSIvan Khoronzhuk * EST CTL register. In the second iteration they will match and code returns.
1418127224cSIvan Khoronzhuk * The actual buffer to write command is selected later before it is ready
1428127224cSIvan Khoronzhuk * to update the schedule.
1438127224cSIvan Khoronzhuk */
am65_cpsw_port_est_get_free_buf_num(struct net_device * ndev)1448127224cSIvan Khoronzhuk static int am65_cpsw_port_est_get_free_buf_num(struct net_device *ndev)
1458127224cSIvan Khoronzhuk {
1468127224cSIvan Khoronzhuk int oper, admin;
1478127224cSIvan Khoronzhuk int roll = 2;
1488127224cSIvan Khoronzhuk
1498127224cSIvan Khoronzhuk while (roll--) {
1508127224cSIvan Khoronzhuk if (am65_cpsw_port_est_is_swapped(ndev, &oper, &admin))
1518127224cSIvan Khoronzhuk return !oper;
1528127224cSIvan Khoronzhuk
1538127224cSIvan Khoronzhuk /* admin is not set, so hinder transition as it's not allowed
1548127224cSIvan Khoronzhuk * to touch memory in-flight, by targeting same oper buf.
1558127224cSIvan Khoronzhuk */
1568127224cSIvan Khoronzhuk am65_cpsw_port_est_assign_buf_num(ndev, oper);
1578127224cSIvan Khoronzhuk
1588127224cSIvan Khoronzhuk dev_info(&ndev->dev,
1598127224cSIvan Khoronzhuk "Prev. EST admin cycle is in transit %d -> %d\n",
1608127224cSIvan Khoronzhuk oper, admin);
1618127224cSIvan Khoronzhuk }
1628127224cSIvan Khoronzhuk
1638127224cSIvan Khoronzhuk return admin;
1648127224cSIvan Khoronzhuk }
1658127224cSIvan Khoronzhuk
am65_cpsw_admin_to_oper(struct net_device * ndev)1668127224cSIvan Khoronzhuk static void am65_cpsw_admin_to_oper(struct net_device *ndev)
1678127224cSIvan Khoronzhuk {
1688127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
1698127224cSIvan Khoronzhuk
1708127224cSIvan Khoronzhuk devm_kfree(&ndev->dev, port->qos.est_oper);
1718127224cSIvan Khoronzhuk
1728127224cSIvan Khoronzhuk port->qos.est_oper = port->qos.est_admin;
1738127224cSIvan Khoronzhuk port->qos.est_admin = NULL;
1748127224cSIvan Khoronzhuk }
1758127224cSIvan Khoronzhuk
am65_cpsw_port_est_get_buf_num(struct net_device * ndev,struct am65_cpsw_est * est_new)1768127224cSIvan Khoronzhuk static void am65_cpsw_port_est_get_buf_num(struct net_device *ndev,
1778127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new)
1788127224cSIvan Khoronzhuk {
1798127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
1808127224cSIvan Khoronzhuk u32 val;
1818127224cSIvan Khoronzhuk
1828127224cSIvan Khoronzhuk val = readl(port->port_base + AM65_CPSW_PN_REG_EST_CTL);
1838127224cSIvan Khoronzhuk val &= ~AM65_CPSW_PN_EST_ONEBUF;
1848127224cSIvan Khoronzhuk writel(val, port->port_base + AM65_CPSW_PN_REG_EST_CTL);
1858127224cSIvan Khoronzhuk
1868127224cSIvan Khoronzhuk est_new->buf = am65_cpsw_port_est_get_free_buf_num(ndev);
1878127224cSIvan Khoronzhuk
1888127224cSIvan Khoronzhuk /* rolled buf num means changed buf while configuring */
1898127224cSIvan Khoronzhuk if (port->qos.est_oper && port->qos.est_admin &&
1908127224cSIvan Khoronzhuk est_new->buf == port->qos.est_oper->buf)
1918127224cSIvan Khoronzhuk am65_cpsw_admin_to_oper(ndev);
1928127224cSIvan Khoronzhuk }
1938127224cSIvan Khoronzhuk
am65_cpsw_est_set(struct net_device * ndev,int enable)1948127224cSIvan Khoronzhuk static void am65_cpsw_est_set(struct net_device *ndev, int enable)
1958127224cSIvan Khoronzhuk {
1968127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
1978127224cSIvan Khoronzhuk struct am65_cpsw_common *common = port->common;
1988127224cSIvan Khoronzhuk int common_enable = 0;
1998127224cSIvan Khoronzhuk int i;
2008127224cSIvan Khoronzhuk
2018127224cSIvan Khoronzhuk am65_cpsw_port_est_enable(port, enable);
2028127224cSIvan Khoronzhuk
2038127224cSIvan Khoronzhuk for (i = 0; i < common->port_num; i++)
2048127224cSIvan Khoronzhuk common_enable |= am65_cpsw_port_est_enabled(&common->ports[i]);
2058127224cSIvan Khoronzhuk
2068127224cSIvan Khoronzhuk common_enable |= enable;
2078127224cSIvan Khoronzhuk am65_cpsw_est_enable(common, common_enable);
2088127224cSIvan Khoronzhuk }
2098127224cSIvan Khoronzhuk
2108127224cSIvan Khoronzhuk /* This update is supposed to be used in any routine before getting real state
2118127224cSIvan Khoronzhuk * of admin -> oper transition, particularly it's supposed to be used in some
2128127224cSIvan Khoronzhuk * generic routine for providing real state to Taprio Qdisc.
2138127224cSIvan Khoronzhuk */
am65_cpsw_est_update_state(struct net_device * ndev)2148127224cSIvan Khoronzhuk static void am65_cpsw_est_update_state(struct net_device *ndev)
2158127224cSIvan Khoronzhuk {
2168127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
2178127224cSIvan Khoronzhuk int oper, admin;
2188127224cSIvan Khoronzhuk
2198127224cSIvan Khoronzhuk if (!port->qos.est_admin)
2208127224cSIvan Khoronzhuk return;
2218127224cSIvan Khoronzhuk
2228127224cSIvan Khoronzhuk if (!am65_cpsw_port_est_is_swapped(ndev, &oper, &admin))
2238127224cSIvan Khoronzhuk return;
2248127224cSIvan Khoronzhuk
2258127224cSIvan Khoronzhuk am65_cpsw_admin_to_oper(ndev);
2268127224cSIvan Khoronzhuk }
2278127224cSIvan Khoronzhuk
2288127224cSIvan Khoronzhuk /* Fetch command count it's number of bytes in Gigabit mode or nibbles in
2298127224cSIvan Khoronzhuk * 10/100Mb mode. So, having speed and time in ns, recalculate ns to number of
2308127224cSIvan Khoronzhuk * bytes/nibbles that can be sent while transmission on given speed.
2318127224cSIvan Khoronzhuk */
am65_est_cmd_ns_to_cnt(u64 ns,int link_speed)2328127224cSIvan Khoronzhuk static int am65_est_cmd_ns_to_cnt(u64 ns, int link_speed)
2338127224cSIvan Khoronzhuk {
2348127224cSIvan Khoronzhuk u64 temp;
2358127224cSIvan Khoronzhuk
2368127224cSIvan Khoronzhuk temp = ns * link_speed;
2378127224cSIvan Khoronzhuk if (link_speed < SPEED_1000)
2388127224cSIvan Khoronzhuk temp <<= 1;
2398127224cSIvan Khoronzhuk
2408127224cSIvan Khoronzhuk return DIV_ROUND_UP(temp, 8 * 1000);
2418127224cSIvan Khoronzhuk }
2428127224cSIvan Khoronzhuk
am65_cpsw_est_set_sched_cmds(void __iomem * addr,int fetch_cnt,int fetch_allow)2438127224cSIvan Khoronzhuk static void __iomem *am65_cpsw_est_set_sched_cmds(void __iomem *addr,
2448127224cSIvan Khoronzhuk int fetch_cnt,
2458127224cSIvan Khoronzhuk int fetch_allow)
2468127224cSIvan Khoronzhuk {
2478127224cSIvan Khoronzhuk u32 prio_mask, cmd_fetch_cnt, cmd;
2488127224cSIvan Khoronzhuk
2498127224cSIvan Khoronzhuk do {
2508127224cSIvan Khoronzhuk if (fetch_cnt > AM65_CPSW_FETCH_CNT_MAX) {
2518127224cSIvan Khoronzhuk fetch_cnt -= AM65_CPSW_FETCH_CNT_MAX;
2528127224cSIvan Khoronzhuk cmd_fetch_cnt = AM65_CPSW_FETCH_CNT_MAX;
2538127224cSIvan Khoronzhuk } else {
2548127224cSIvan Khoronzhuk cmd_fetch_cnt = fetch_cnt;
2558127224cSIvan Khoronzhuk /* fetch count can't be less than 16? */
2568127224cSIvan Khoronzhuk if (cmd_fetch_cnt && cmd_fetch_cnt < 16)
2578127224cSIvan Khoronzhuk cmd_fetch_cnt = 16;
2588127224cSIvan Khoronzhuk
2598127224cSIvan Khoronzhuk fetch_cnt = 0;
2608127224cSIvan Khoronzhuk }
2618127224cSIvan Khoronzhuk
2628127224cSIvan Khoronzhuk prio_mask = fetch_allow & AM65_CPSW_FETCH_ALLOW_MSK;
2638127224cSIvan Khoronzhuk cmd = (cmd_fetch_cnt << AM65_CPSW_FETCH_CNT_OFFSET) | prio_mask;
2648127224cSIvan Khoronzhuk
2658127224cSIvan Khoronzhuk writel(cmd, addr);
2668127224cSIvan Khoronzhuk addr += 4;
2678127224cSIvan Khoronzhuk } while (fetch_cnt);
2688127224cSIvan Khoronzhuk
2698127224cSIvan Khoronzhuk return addr;
2708127224cSIvan Khoronzhuk }
2718127224cSIvan Khoronzhuk
am65_cpsw_est_calc_cmd_num(struct net_device * ndev,struct tc_taprio_qopt_offload * taprio,int link_speed)2728127224cSIvan Khoronzhuk static int am65_cpsw_est_calc_cmd_num(struct net_device *ndev,
2738127224cSIvan Khoronzhuk struct tc_taprio_qopt_offload *taprio,
2748127224cSIvan Khoronzhuk int link_speed)
2758127224cSIvan Khoronzhuk {
2768127224cSIvan Khoronzhuk int i, cmd_cnt, cmd_sum = 0;
2778127224cSIvan Khoronzhuk u32 fetch_cnt;
2788127224cSIvan Khoronzhuk
2798127224cSIvan Khoronzhuk for (i = 0; i < taprio->num_entries; i++) {
2808127224cSIvan Khoronzhuk if (taprio->entries[i].command != TC_TAPRIO_CMD_SET_GATES) {
2818127224cSIvan Khoronzhuk dev_err(&ndev->dev, "Only SET command is supported");
2828127224cSIvan Khoronzhuk return -EINVAL;
2838127224cSIvan Khoronzhuk }
2848127224cSIvan Khoronzhuk
2858127224cSIvan Khoronzhuk fetch_cnt = am65_est_cmd_ns_to_cnt(taprio->entries[i].interval,
2868127224cSIvan Khoronzhuk link_speed);
2878127224cSIvan Khoronzhuk
2888127224cSIvan Khoronzhuk cmd_cnt = DIV_ROUND_UP(fetch_cnt, AM65_CPSW_FETCH_CNT_MAX);
2898127224cSIvan Khoronzhuk if (!cmd_cnt)
2908127224cSIvan Khoronzhuk cmd_cnt++;
2918127224cSIvan Khoronzhuk
2928127224cSIvan Khoronzhuk cmd_sum += cmd_cnt;
2938127224cSIvan Khoronzhuk
2948127224cSIvan Khoronzhuk if (!fetch_cnt)
2958127224cSIvan Khoronzhuk break;
2968127224cSIvan Khoronzhuk }
2978127224cSIvan Khoronzhuk
2988127224cSIvan Khoronzhuk return cmd_sum;
2998127224cSIvan Khoronzhuk }
3008127224cSIvan Khoronzhuk
am65_cpsw_est_check_scheds(struct net_device * ndev,struct am65_cpsw_est * est_new)3018127224cSIvan Khoronzhuk static int am65_cpsw_est_check_scheds(struct net_device *ndev,
3028127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new)
3038127224cSIvan Khoronzhuk {
3048127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
3058127224cSIvan Khoronzhuk int cmd_num;
3068127224cSIvan Khoronzhuk
3078127224cSIvan Khoronzhuk cmd_num = am65_cpsw_est_calc_cmd_num(ndev, &est_new->taprio,
3088127224cSIvan Khoronzhuk port->qos.link_speed);
3098127224cSIvan Khoronzhuk if (cmd_num < 0)
3108127224cSIvan Khoronzhuk return cmd_num;
3118127224cSIvan Khoronzhuk
3128127224cSIvan Khoronzhuk if (cmd_num > AM65_CPSW_FETCH_RAM_CMD_NUM / 2) {
3138127224cSIvan Khoronzhuk dev_err(&ndev->dev, "No fetch RAM");
3148127224cSIvan Khoronzhuk return -ENOMEM;
3158127224cSIvan Khoronzhuk }
3168127224cSIvan Khoronzhuk
3178127224cSIvan Khoronzhuk return 0;
3188127224cSIvan Khoronzhuk }
3198127224cSIvan Khoronzhuk
am65_cpsw_est_set_sched_list(struct net_device * ndev,struct am65_cpsw_est * est_new)3208127224cSIvan Khoronzhuk static void am65_cpsw_est_set_sched_list(struct net_device *ndev,
3218127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new)
3228127224cSIvan Khoronzhuk {
3238127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
3248127224cSIvan Khoronzhuk u32 fetch_cnt, fetch_allow, all_fetch_allow = 0;
3258127224cSIvan Khoronzhuk void __iomem *ram_addr, *max_ram_addr;
3268127224cSIvan Khoronzhuk struct tc_taprio_sched_entry *entry;
3278127224cSIvan Khoronzhuk int i, ram_size;
3288127224cSIvan Khoronzhuk
3298127224cSIvan Khoronzhuk ram_addr = port->fetch_ram_base;
3308127224cSIvan Khoronzhuk ram_size = AM65_CPSW_FETCH_RAM_CMD_NUM * 2;
3318127224cSIvan Khoronzhuk ram_addr += est_new->buf * ram_size;
3328127224cSIvan Khoronzhuk
3338127224cSIvan Khoronzhuk max_ram_addr = ram_size + ram_addr;
3348127224cSIvan Khoronzhuk for (i = 0; i < est_new->taprio.num_entries; i++) {
3358127224cSIvan Khoronzhuk entry = &est_new->taprio.entries[i];
3368127224cSIvan Khoronzhuk
3378127224cSIvan Khoronzhuk fetch_cnt = am65_est_cmd_ns_to_cnt(entry->interval,
3388127224cSIvan Khoronzhuk port->qos.link_speed);
3398127224cSIvan Khoronzhuk fetch_allow = entry->gate_mask;
3408127224cSIvan Khoronzhuk if (fetch_allow > AM65_CPSW_FETCH_ALLOW_MAX)
3418127224cSIvan Khoronzhuk dev_dbg(&ndev->dev, "fetch_allow > 8 bits: %d\n",
3428127224cSIvan Khoronzhuk fetch_allow);
3438127224cSIvan Khoronzhuk
3448127224cSIvan Khoronzhuk ram_addr = am65_cpsw_est_set_sched_cmds(ram_addr, fetch_cnt,
3458127224cSIvan Khoronzhuk fetch_allow);
3468127224cSIvan Khoronzhuk
3478127224cSIvan Khoronzhuk if (!fetch_cnt && i < est_new->taprio.num_entries - 1) {
3488127224cSIvan Khoronzhuk dev_info(&ndev->dev,
3498127224cSIvan Khoronzhuk "next scheds after %d have no impact", i + 1);
3508127224cSIvan Khoronzhuk break;
3518127224cSIvan Khoronzhuk }
3528127224cSIvan Khoronzhuk
3538127224cSIvan Khoronzhuk all_fetch_allow |= fetch_allow;
3548127224cSIvan Khoronzhuk }
3558127224cSIvan Khoronzhuk
3568127224cSIvan Khoronzhuk /* end cmd, enabling non-timed queues for potential over cycle time */
3578127224cSIvan Khoronzhuk if (ram_addr < max_ram_addr)
3588127224cSIvan Khoronzhuk writel(~all_fetch_allow & AM65_CPSW_FETCH_ALLOW_MSK, ram_addr);
3598127224cSIvan Khoronzhuk }
3608127224cSIvan Khoronzhuk
361935888cdSLee Jones /*
3628127224cSIvan Khoronzhuk * Enable ESTf periodic output, set cycle start time and interval.
3638127224cSIvan Khoronzhuk */
am65_cpsw_timer_set(struct net_device * ndev,struct am65_cpsw_est * est_new)3648127224cSIvan Khoronzhuk static int am65_cpsw_timer_set(struct net_device *ndev,
3658127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new)
3668127224cSIvan Khoronzhuk {
3678127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
3688127224cSIvan Khoronzhuk struct am65_cpsw_common *common = port->common;
3698127224cSIvan Khoronzhuk struct am65_cpts *cpts = common->cpts;
3708127224cSIvan Khoronzhuk struct am65_cpts_estf_cfg cfg;
3718127224cSIvan Khoronzhuk
3728127224cSIvan Khoronzhuk cfg.ns_period = est_new->taprio.cycle_time;
3738127224cSIvan Khoronzhuk cfg.ns_start = est_new->taprio.base_time;
3748127224cSIvan Khoronzhuk
3758127224cSIvan Khoronzhuk return am65_cpts_estf_enable(cpts, port->port_id - 1, &cfg);
3768127224cSIvan Khoronzhuk }
3778127224cSIvan Khoronzhuk
am65_cpsw_timer_stop(struct net_device * ndev)3788127224cSIvan Khoronzhuk static void am65_cpsw_timer_stop(struct net_device *ndev)
3798127224cSIvan Khoronzhuk {
3808127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
3818127224cSIvan Khoronzhuk struct am65_cpts *cpts = port->common->cpts;
3828127224cSIvan Khoronzhuk
3838127224cSIvan Khoronzhuk am65_cpts_estf_disable(cpts, port->port_id - 1);
3848127224cSIvan Khoronzhuk }
3858127224cSIvan Khoronzhuk
am65_cpsw_timer_act(struct net_device * ndev,struct am65_cpsw_est * est_new)3868127224cSIvan Khoronzhuk static enum timer_act am65_cpsw_timer_act(struct net_device *ndev,
3878127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new)
3888127224cSIvan Khoronzhuk {
3898127224cSIvan Khoronzhuk struct tc_taprio_qopt_offload *taprio_oper, *taprio_new;
3908127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
3918127224cSIvan Khoronzhuk struct am65_cpts *cpts = port->common->cpts;
3928127224cSIvan Khoronzhuk u64 cur_time;
3938127224cSIvan Khoronzhuk s64 diff;
3948127224cSIvan Khoronzhuk
3958127224cSIvan Khoronzhuk if (!port->qos.est_oper)
3968127224cSIvan Khoronzhuk return TACT_PROG;
3978127224cSIvan Khoronzhuk
3988127224cSIvan Khoronzhuk taprio_new = &est_new->taprio;
3998127224cSIvan Khoronzhuk taprio_oper = &port->qos.est_oper->taprio;
4008127224cSIvan Khoronzhuk
4018127224cSIvan Khoronzhuk if (taprio_new->cycle_time != taprio_oper->cycle_time)
4028127224cSIvan Khoronzhuk return TACT_NEED_STOP;
4038127224cSIvan Khoronzhuk
4048127224cSIvan Khoronzhuk /* in order to avoid timer reset get base_time form oper taprio */
4058127224cSIvan Khoronzhuk if (!taprio_new->base_time && taprio_oper)
4068127224cSIvan Khoronzhuk taprio_new->base_time = taprio_oper->base_time;
4078127224cSIvan Khoronzhuk
4088127224cSIvan Khoronzhuk if (taprio_new->base_time == taprio_oper->base_time)
4098127224cSIvan Khoronzhuk return TACT_SKIP_PROG;
4108127224cSIvan Khoronzhuk
4118127224cSIvan Khoronzhuk /* base times are cycle synchronized */
4128127224cSIvan Khoronzhuk diff = taprio_new->base_time - taprio_oper->base_time;
4138127224cSIvan Khoronzhuk diff = diff < 0 ? -diff : diff;
4148127224cSIvan Khoronzhuk if (diff % taprio_new->cycle_time)
4158127224cSIvan Khoronzhuk return TACT_NEED_STOP;
4168127224cSIvan Khoronzhuk
4178127224cSIvan Khoronzhuk cur_time = am65_cpts_ns_gettime(cpts);
4188127224cSIvan Khoronzhuk if (taprio_new->base_time <= cur_time + taprio_new->cycle_time)
4198127224cSIvan Khoronzhuk return TACT_SKIP_PROG;
4208127224cSIvan Khoronzhuk
4218127224cSIvan Khoronzhuk /* TODO: Admin schedule at future time is not currently supported */
4228127224cSIvan Khoronzhuk return TACT_NEED_STOP;
4238127224cSIvan Khoronzhuk }
4248127224cSIvan Khoronzhuk
am65_cpsw_stop_est(struct net_device * ndev)4258127224cSIvan Khoronzhuk static void am65_cpsw_stop_est(struct net_device *ndev)
4268127224cSIvan Khoronzhuk {
4278127224cSIvan Khoronzhuk am65_cpsw_est_set(ndev, 0);
4288127224cSIvan Khoronzhuk am65_cpsw_timer_stop(ndev);
4298127224cSIvan Khoronzhuk }
4308127224cSIvan Khoronzhuk
am65_cpsw_purge_est(struct net_device * ndev)4318127224cSIvan Khoronzhuk static void am65_cpsw_purge_est(struct net_device *ndev)
4328127224cSIvan Khoronzhuk {
4338127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
4348127224cSIvan Khoronzhuk
4358127224cSIvan Khoronzhuk am65_cpsw_stop_est(ndev);
4368127224cSIvan Khoronzhuk
4378127224cSIvan Khoronzhuk devm_kfree(&ndev->dev, port->qos.est_admin);
4388127224cSIvan Khoronzhuk devm_kfree(&ndev->dev, port->qos.est_oper);
4398127224cSIvan Khoronzhuk
4408127224cSIvan Khoronzhuk port->qos.est_oper = NULL;
4418127224cSIvan Khoronzhuk port->qos.est_admin = NULL;
4428127224cSIvan Khoronzhuk }
4438127224cSIvan Khoronzhuk
am65_cpsw_configure_taprio(struct net_device * ndev,struct am65_cpsw_est * est_new)4448127224cSIvan Khoronzhuk static int am65_cpsw_configure_taprio(struct net_device *ndev,
4458127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new)
4468127224cSIvan Khoronzhuk {
4478127224cSIvan Khoronzhuk struct am65_cpsw_common *common = am65_ndev_to_common(ndev);
4488127224cSIvan Khoronzhuk struct am65_cpts *cpts = common->cpts;
4498127224cSIvan Khoronzhuk int ret = 0, tact = TACT_PROG;
4508127224cSIvan Khoronzhuk
4518127224cSIvan Khoronzhuk am65_cpsw_est_update_state(ndev);
4528127224cSIvan Khoronzhuk
4532d800bc5SVladimir Oltean if (est_new->taprio.cmd == TAPRIO_CMD_DESTROY) {
4548127224cSIvan Khoronzhuk am65_cpsw_stop_est(ndev);
4558127224cSIvan Khoronzhuk return ret;
4568127224cSIvan Khoronzhuk }
4578127224cSIvan Khoronzhuk
4588127224cSIvan Khoronzhuk ret = am65_cpsw_est_check_scheds(ndev, est_new);
4598127224cSIvan Khoronzhuk if (ret < 0)
4608127224cSIvan Khoronzhuk return ret;
4618127224cSIvan Khoronzhuk
4628127224cSIvan Khoronzhuk tact = am65_cpsw_timer_act(ndev, est_new);
4638127224cSIvan Khoronzhuk if (tact == TACT_NEED_STOP) {
4648127224cSIvan Khoronzhuk dev_err(&ndev->dev,
4658127224cSIvan Khoronzhuk "Can't toggle estf timer, stop taprio first");
4668127224cSIvan Khoronzhuk return -EINVAL;
4678127224cSIvan Khoronzhuk }
4688127224cSIvan Khoronzhuk
4698127224cSIvan Khoronzhuk if (tact == TACT_PROG)
4708127224cSIvan Khoronzhuk am65_cpsw_timer_stop(ndev);
4718127224cSIvan Khoronzhuk
4728127224cSIvan Khoronzhuk if (!est_new->taprio.base_time)
4738127224cSIvan Khoronzhuk est_new->taprio.base_time = am65_cpts_ns_gettime(cpts);
4748127224cSIvan Khoronzhuk
4758127224cSIvan Khoronzhuk am65_cpsw_port_est_get_buf_num(ndev, est_new);
4768127224cSIvan Khoronzhuk am65_cpsw_est_set_sched_list(ndev, est_new);
4778127224cSIvan Khoronzhuk am65_cpsw_port_est_assign_buf_num(ndev, est_new->buf);
4788127224cSIvan Khoronzhuk
4792d800bc5SVladimir Oltean am65_cpsw_est_set(ndev, est_new->taprio.cmd == TAPRIO_CMD_REPLACE);
4808127224cSIvan Khoronzhuk
4818127224cSIvan Khoronzhuk if (tact == TACT_PROG) {
4828127224cSIvan Khoronzhuk ret = am65_cpsw_timer_set(ndev, est_new);
4838127224cSIvan Khoronzhuk if (ret) {
4848127224cSIvan Khoronzhuk dev_err(&ndev->dev, "Failed to set cycle time");
4858127224cSIvan Khoronzhuk return ret;
4868127224cSIvan Khoronzhuk }
4878127224cSIvan Khoronzhuk }
4888127224cSIvan Khoronzhuk
4898127224cSIvan Khoronzhuk return ret;
4908127224cSIvan Khoronzhuk }
4918127224cSIvan Khoronzhuk
am65_cpsw_cp_taprio(struct tc_taprio_qopt_offload * from,struct tc_taprio_qopt_offload * to)4928127224cSIvan Khoronzhuk static void am65_cpsw_cp_taprio(struct tc_taprio_qopt_offload *from,
4938127224cSIvan Khoronzhuk struct tc_taprio_qopt_offload *to)
4948127224cSIvan Khoronzhuk {
4958127224cSIvan Khoronzhuk int i;
4968127224cSIvan Khoronzhuk
4978127224cSIvan Khoronzhuk *to = *from;
4988127224cSIvan Khoronzhuk for (i = 0; i < from->num_entries; i++)
4998127224cSIvan Khoronzhuk to->entries[i] = from->entries[i];
5008127224cSIvan Khoronzhuk }
5018127224cSIvan Khoronzhuk
am65_cpsw_set_taprio(struct net_device * ndev,void * type_data)5028127224cSIvan Khoronzhuk static int am65_cpsw_set_taprio(struct net_device *ndev, void *type_data)
5038127224cSIvan Khoronzhuk {
5048127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
5058127224cSIvan Khoronzhuk struct tc_taprio_qopt_offload *taprio = type_data;
5068127224cSIvan Khoronzhuk struct am65_cpsw_est *est_new;
5078127224cSIvan Khoronzhuk int ret = 0;
5088127224cSIvan Khoronzhuk
5098127224cSIvan Khoronzhuk if (taprio->cycle_time_extension) {
5108127224cSIvan Khoronzhuk dev_err(&ndev->dev, "Failed to set cycle time extension");
5118127224cSIvan Khoronzhuk return -EOPNOTSUPP;
5128127224cSIvan Khoronzhuk }
5138127224cSIvan Khoronzhuk
514f362b70bSGustavo A. R. Silva est_new = devm_kzalloc(&ndev->dev,
515f362b70bSGustavo A. R. Silva struct_size(est_new, taprio.entries, taprio->num_entries),
516f362b70bSGustavo A. R. Silva GFP_KERNEL);
5178127224cSIvan Khoronzhuk if (!est_new)
5188127224cSIvan Khoronzhuk return -ENOMEM;
5198127224cSIvan Khoronzhuk
5208127224cSIvan Khoronzhuk am65_cpsw_cp_taprio(taprio, &est_new->taprio);
5218127224cSIvan Khoronzhuk ret = am65_cpsw_configure_taprio(ndev, est_new);
5228127224cSIvan Khoronzhuk if (!ret) {
5232d800bc5SVladimir Oltean if (taprio->cmd == TAPRIO_CMD_REPLACE) {
5248127224cSIvan Khoronzhuk devm_kfree(&ndev->dev, port->qos.est_admin);
5258127224cSIvan Khoronzhuk
5268127224cSIvan Khoronzhuk port->qos.est_admin = est_new;
5278127224cSIvan Khoronzhuk } else {
5288127224cSIvan Khoronzhuk devm_kfree(&ndev->dev, est_new);
5298127224cSIvan Khoronzhuk am65_cpsw_purge_est(ndev);
5308127224cSIvan Khoronzhuk }
5318127224cSIvan Khoronzhuk } else {
5328127224cSIvan Khoronzhuk devm_kfree(&ndev->dev, est_new);
5338127224cSIvan Khoronzhuk }
5348127224cSIvan Khoronzhuk
5358127224cSIvan Khoronzhuk return ret;
5368127224cSIvan Khoronzhuk }
5378127224cSIvan Khoronzhuk
am65_cpsw_est_link_up(struct net_device * ndev,int link_speed)5388127224cSIvan Khoronzhuk static void am65_cpsw_est_link_up(struct net_device *ndev, int link_speed)
5398127224cSIvan Khoronzhuk {
5408127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
5418127224cSIvan Khoronzhuk ktime_t cur_time;
5428127224cSIvan Khoronzhuk s64 delta;
5438127224cSIvan Khoronzhuk
544*42b118c9SJakub Kicinski port->qos.link_speed = link_speed;
5458127224cSIvan Khoronzhuk if (!am65_cpsw_port_est_enabled(port))
5468127224cSIvan Khoronzhuk return;
5478127224cSIvan Khoronzhuk
5488127224cSIvan Khoronzhuk if (port->qos.link_down_time) {
5498127224cSIvan Khoronzhuk cur_time = ktime_get();
5508127224cSIvan Khoronzhuk delta = ktime_us_delta(cur_time, port->qos.link_down_time);
5518127224cSIvan Khoronzhuk if (delta > USEC_PER_SEC) {
5528127224cSIvan Khoronzhuk dev_err(&ndev->dev,
5538127224cSIvan Khoronzhuk "Link has been lost too long, stopping TAS");
5548127224cSIvan Khoronzhuk goto purge_est;
5558127224cSIvan Khoronzhuk }
5568127224cSIvan Khoronzhuk }
5578127224cSIvan Khoronzhuk
5588127224cSIvan Khoronzhuk return;
5598127224cSIvan Khoronzhuk
5608127224cSIvan Khoronzhuk purge_est:
5618127224cSIvan Khoronzhuk am65_cpsw_purge_est(ndev);
5628127224cSIvan Khoronzhuk }
5638127224cSIvan Khoronzhuk
am65_cpsw_setup_taprio(struct net_device * ndev,void * type_data)5648127224cSIvan Khoronzhuk static int am65_cpsw_setup_taprio(struct net_device *ndev, void *type_data)
5658127224cSIvan Khoronzhuk {
5668127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
5672d800bc5SVladimir Oltean struct tc_taprio_qopt_offload *taprio = type_data;
5688127224cSIvan Khoronzhuk struct am65_cpsw_common *common = port->common;
5698127224cSIvan Khoronzhuk
5702d800bc5SVladimir Oltean if (taprio->cmd != TAPRIO_CMD_REPLACE &&
5712d800bc5SVladimir Oltean taprio->cmd != TAPRIO_CMD_DESTROY)
5722d800bc5SVladimir Oltean return -EOPNOTSUPP;
5732d800bc5SVladimir Oltean
5748127224cSIvan Khoronzhuk if (!IS_ENABLED(CONFIG_TI_AM65_CPSW_TAS))
5758127224cSIvan Khoronzhuk return -ENODEV;
5768127224cSIvan Khoronzhuk
5778127224cSIvan Khoronzhuk if (!netif_running(ndev)) {
5788127224cSIvan Khoronzhuk dev_err(&ndev->dev, "interface is down, link speed unknown\n");
5798127224cSIvan Khoronzhuk return -ENETDOWN;
5808127224cSIvan Khoronzhuk }
5818127224cSIvan Khoronzhuk
5828127224cSIvan Khoronzhuk if (common->pf_p0_rx_ptype_rrobin) {
5838127224cSIvan Khoronzhuk dev_err(&ndev->dev,
5848127224cSIvan Khoronzhuk "p0-rx-ptype-rrobin flag conflicts with taprio qdisc\n");
5858127224cSIvan Khoronzhuk return -EINVAL;
5868127224cSIvan Khoronzhuk }
5878127224cSIvan Khoronzhuk
5888127224cSIvan Khoronzhuk if (port->qos.link_speed == SPEED_UNKNOWN)
5898127224cSIvan Khoronzhuk return -ENOLINK;
5908127224cSIvan Khoronzhuk
5918127224cSIvan Khoronzhuk return am65_cpsw_set_taprio(ndev, type_data);
5928127224cSIvan Khoronzhuk }
5938127224cSIvan Khoronzhuk
am65_cpsw_tc_query_caps(struct net_device * ndev,void * type_data)594522d15eaSVladimir Oltean static int am65_cpsw_tc_query_caps(struct net_device *ndev, void *type_data)
595522d15eaSVladimir Oltean {
596522d15eaSVladimir Oltean struct tc_query_caps_base *base = type_data;
597522d15eaSVladimir Oltean
598522d15eaSVladimir Oltean switch (base->type) {
599522d15eaSVladimir Oltean case TC_SETUP_QDISC_TAPRIO: {
600522d15eaSVladimir Oltean struct tc_taprio_caps *caps = base->caps;
601522d15eaSVladimir Oltean
602522d15eaSVladimir Oltean if (!IS_ENABLED(CONFIG_TI_AM65_CPSW_TAS))
603522d15eaSVladimir Oltean return -EOPNOTSUPP;
604522d15eaSVladimir Oltean
605522d15eaSVladimir Oltean caps->gate_mask_per_txq = true;
606522d15eaSVladimir Oltean
607522d15eaSVladimir Oltean return 0;
608522d15eaSVladimir Oltean }
609522d15eaSVladimir Oltean default:
610522d15eaSVladimir Oltean return -EOPNOTSUPP;
611522d15eaSVladimir Oltean }
612522d15eaSVladimir Oltean }
613522d15eaSVladimir Oltean
am65_cpsw_qos_clsflower_add_policer(struct am65_cpsw_port * port,struct netlink_ext_ack * extack,struct flow_cls_offload * cls,u64 rate_pkt_ps)6145ec836beSGrygorii Strashko static int am65_cpsw_qos_clsflower_add_policer(struct am65_cpsw_port *port,
6155ec836beSGrygorii Strashko struct netlink_ext_ack *extack,
6165ec836beSGrygorii Strashko struct flow_cls_offload *cls,
6175ec836beSGrygorii Strashko u64 rate_pkt_ps)
6185ec836beSGrygorii Strashko {
6195ec836beSGrygorii Strashko struct flow_rule *rule = flow_cls_offload_flow_rule(cls);
6205ec836beSGrygorii Strashko struct flow_dissector *dissector = rule->match.dissector;
6215ec836beSGrygorii Strashko static const u8 mc_mac[] = {0x01, 0x00, 0x00, 0x00, 0x00, 0x00};
6225ec836beSGrygorii Strashko struct am65_cpsw_qos *qos = &port->qos;
6235ec836beSGrygorii Strashko struct flow_match_eth_addrs match;
6245ec836beSGrygorii Strashko int ret;
6255ec836beSGrygorii Strashko
6265ec836beSGrygorii Strashko if (dissector->used_keys &
6272b3082c6SRatheesh Kannoth ~(BIT_ULL(FLOW_DISSECTOR_KEY_BASIC) |
6282b3082c6SRatheesh Kannoth BIT_ULL(FLOW_DISSECTOR_KEY_CONTROL) |
6292b3082c6SRatheesh Kannoth BIT_ULL(FLOW_DISSECTOR_KEY_ETH_ADDRS))) {
6305ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
6315ec836beSGrygorii Strashko "Unsupported keys used");
6325ec836beSGrygorii Strashko return -EOPNOTSUPP;
6335ec836beSGrygorii Strashko }
6345ec836beSGrygorii Strashko
6355ec836beSGrygorii Strashko if (!flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
6365ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack, "Not matching on eth address");
6375ec836beSGrygorii Strashko return -EOPNOTSUPP;
6385ec836beSGrygorii Strashko }
6395ec836beSGrygorii Strashko
6405ec836beSGrygorii Strashko flow_rule_match_eth_addrs(rule, &match);
6415ec836beSGrygorii Strashko
6425ec836beSGrygorii Strashko if (!is_zero_ether_addr(match.mask->src)) {
6435ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
6445ec836beSGrygorii Strashko "Matching on source MAC not supported");
6455ec836beSGrygorii Strashko return -EOPNOTSUPP;
6465ec836beSGrygorii Strashko }
6475ec836beSGrygorii Strashko
6485ec836beSGrygorii Strashko if (is_broadcast_ether_addr(match.key->dst) &&
6495ec836beSGrygorii Strashko is_broadcast_ether_addr(match.mask->dst)) {
6505ec836beSGrygorii Strashko ret = cpsw_ale_rx_ratelimit_bc(port->common->ale, port->port_id, rate_pkt_ps);
6515ec836beSGrygorii Strashko if (ret)
6525ec836beSGrygorii Strashko return ret;
6535ec836beSGrygorii Strashko
6545ec836beSGrygorii Strashko qos->ale_bc_ratelimit.cookie = cls->cookie;
6555ec836beSGrygorii Strashko qos->ale_bc_ratelimit.rate_packet_ps = rate_pkt_ps;
6565ec836beSGrygorii Strashko } else if (ether_addr_equal_unaligned(match.key->dst, mc_mac) &&
6575ec836beSGrygorii Strashko ether_addr_equal_unaligned(match.mask->dst, mc_mac)) {
6585ec836beSGrygorii Strashko ret = cpsw_ale_rx_ratelimit_mc(port->common->ale, port->port_id, rate_pkt_ps);
6595ec836beSGrygorii Strashko if (ret)
6605ec836beSGrygorii Strashko return ret;
6615ec836beSGrygorii Strashko
6625ec836beSGrygorii Strashko qos->ale_mc_ratelimit.cookie = cls->cookie;
6635ec836beSGrygorii Strashko qos->ale_mc_ratelimit.rate_packet_ps = rate_pkt_ps;
6645ec836beSGrygorii Strashko } else {
6655ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack, "Not supported matching key");
6665ec836beSGrygorii Strashko return -EOPNOTSUPP;
6675ec836beSGrygorii Strashko }
6685ec836beSGrygorii Strashko
6695ec836beSGrygorii Strashko return 0;
6705ec836beSGrygorii Strashko }
6715ec836beSGrygorii Strashko
am65_cpsw_qos_clsflower_policer_validate(const struct flow_action * action,const struct flow_action_entry * act,struct netlink_ext_ack * extack)6725ec836beSGrygorii Strashko static int am65_cpsw_qos_clsflower_policer_validate(const struct flow_action *action,
6735ec836beSGrygorii Strashko const struct flow_action_entry *act,
6745ec836beSGrygorii Strashko struct netlink_ext_ack *extack)
6755ec836beSGrygorii Strashko {
6765ec836beSGrygorii Strashko if (act->police.exceed.act_id != FLOW_ACTION_DROP) {
6775ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
6785ec836beSGrygorii Strashko "Offload not supported when exceed action is not drop");
6795ec836beSGrygorii Strashko return -EOPNOTSUPP;
6805ec836beSGrygorii Strashko }
6815ec836beSGrygorii Strashko
6825ec836beSGrygorii Strashko if (act->police.notexceed.act_id != FLOW_ACTION_PIPE &&
6835ec836beSGrygorii Strashko act->police.notexceed.act_id != FLOW_ACTION_ACCEPT) {
6845ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
6855ec836beSGrygorii Strashko "Offload not supported when conform action is not pipe or ok");
6865ec836beSGrygorii Strashko return -EOPNOTSUPP;
6875ec836beSGrygorii Strashko }
6885ec836beSGrygorii Strashko
6895ec836beSGrygorii Strashko if (act->police.notexceed.act_id == FLOW_ACTION_ACCEPT &&
6905ec836beSGrygorii Strashko !flow_action_is_last_entry(action, act)) {
6915ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
6925ec836beSGrygorii Strashko "Offload not supported when conform action is ok, but action is not last");
6935ec836beSGrygorii Strashko return -EOPNOTSUPP;
6945ec836beSGrygorii Strashko }
6955ec836beSGrygorii Strashko
6965ec836beSGrygorii Strashko if (act->police.rate_bytes_ps || act->police.peakrate_bytes_ps ||
6975ec836beSGrygorii Strashko act->police.avrate || act->police.overhead) {
6985ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
6995ec836beSGrygorii Strashko "Offload not supported when bytes per second/peakrate/avrate/overhead is configured");
7005ec836beSGrygorii Strashko return -EOPNOTSUPP;
7015ec836beSGrygorii Strashko }
7025ec836beSGrygorii Strashko
7035ec836beSGrygorii Strashko return 0;
7045ec836beSGrygorii Strashko }
7055ec836beSGrygorii Strashko
am65_cpsw_qos_configure_clsflower(struct am65_cpsw_port * port,struct flow_cls_offload * cls)7065ec836beSGrygorii Strashko static int am65_cpsw_qos_configure_clsflower(struct am65_cpsw_port *port,
7075ec836beSGrygorii Strashko struct flow_cls_offload *cls)
7085ec836beSGrygorii Strashko {
7095ec836beSGrygorii Strashko struct flow_rule *rule = flow_cls_offload_flow_rule(cls);
7105ec836beSGrygorii Strashko struct netlink_ext_ack *extack = cls->common.extack;
7115ec836beSGrygorii Strashko const struct flow_action_entry *act;
7125ec836beSGrygorii Strashko int i, ret;
7135ec836beSGrygorii Strashko
7145ec836beSGrygorii Strashko flow_action_for_each(i, act, &rule->action) {
7155ec836beSGrygorii Strashko switch (act->id) {
7165ec836beSGrygorii Strashko case FLOW_ACTION_POLICE:
7175ec836beSGrygorii Strashko ret = am65_cpsw_qos_clsflower_policer_validate(&rule->action, act, extack);
7185ec836beSGrygorii Strashko if (ret)
7195ec836beSGrygorii Strashko return ret;
7205ec836beSGrygorii Strashko
7215ec836beSGrygorii Strashko return am65_cpsw_qos_clsflower_add_policer(port, extack, cls,
7225ec836beSGrygorii Strashko act->police.rate_pkt_ps);
7235ec836beSGrygorii Strashko default:
7245ec836beSGrygorii Strashko NL_SET_ERR_MSG_MOD(extack,
7255ec836beSGrygorii Strashko "Action not supported");
7265ec836beSGrygorii Strashko return -EOPNOTSUPP;
7275ec836beSGrygorii Strashko }
7285ec836beSGrygorii Strashko }
7295ec836beSGrygorii Strashko return -EOPNOTSUPP;
7305ec836beSGrygorii Strashko }
7315ec836beSGrygorii Strashko
am65_cpsw_qos_delete_clsflower(struct am65_cpsw_port * port,struct flow_cls_offload * cls)7325ec836beSGrygorii Strashko static int am65_cpsw_qos_delete_clsflower(struct am65_cpsw_port *port, struct flow_cls_offload *cls)
7335ec836beSGrygorii Strashko {
7345ec836beSGrygorii Strashko struct am65_cpsw_qos *qos = &port->qos;
7355ec836beSGrygorii Strashko
7365ec836beSGrygorii Strashko if (cls->cookie == qos->ale_bc_ratelimit.cookie) {
7375ec836beSGrygorii Strashko qos->ale_bc_ratelimit.cookie = 0;
7385ec836beSGrygorii Strashko qos->ale_bc_ratelimit.rate_packet_ps = 0;
7395ec836beSGrygorii Strashko cpsw_ale_rx_ratelimit_bc(port->common->ale, port->port_id, 0);
7405ec836beSGrygorii Strashko }
7415ec836beSGrygorii Strashko
7425ec836beSGrygorii Strashko if (cls->cookie == qos->ale_mc_ratelimit.cookie) {
7435ec836beSGrygorii Strashko qos->ale_mc_ratelimit.cookie = 0;
7445ec836beSGrygorii Strashko qos->ale_mc_ratelimit.rate_packet_ps = 0;
7455ec836beSGrygorii Strashko cpsw_ale_rx_ratelimit_mc(port->common->ale, port->port_id, 0);
7465ec836beSGrygorii Strashko }
7475ec836beSGrygorii Strashko
7485ec836beSGrygorii Strashko return 0;
7495ec836beSGrygorii Strashko }
7505ec836beSGrygorii Strashko
am65_cpsw_qos_setup_tc_clsflower(struct am65_cpsw_port * port,struct flow_cls_offload * cls_flower)7515ec836beSGrygorii Strashko static int am65_cpsw_qos_setup_tc_clsflower(struct am65_cpsw_port *port,
7525ec836beSGrygorii Strashko struct flow_cls_offload *cls_flower)
7535ec836beSGrygorii Strashko {
7545ec836beSGrygorii Strashko switch (cls_flower->command) {
7555ec836beSGrygorii Strashko case FLOW_CLS_REPLACE:
7565ec836beSGrygorii Strashko return am65_cpsw_qos_configure_clsflower(port, cls_flower);
7575ec836beSGrygorii Strashko case FLOW_CLS_DESTROY:
7585ec836beSGrygorii Strashko return am65_cpsw_qos_delete_clsflower(port, cls_flower);
7595ec836beSGrygorii Strashko default:
7605ec836beSGrygorii Strashko return -EOPNOTSUPP;
7615ec836beSGrygorii Strashko }
7625ec836beSGrygorii Strashko }
7635ec836beSGrygorii Strashko
am65_cpsw_qos_setup_tc_block_cb(enum tc_setup_type type,void * type_data,void * cb_priv)7645ec836beSGrygorii Strashko static int am65_cpsw_qos_setup_tc_block_cb(enum tc_setup_type type, void *type_data, void *cb_priv)
7655ec836beSGrygorii Strashko {
7665ec836beSGrygorii Strashko struct am65_cpsw_port *port = cb_priv;
7675ec836beSGrygorii Strashko
7685ec836beSGrygorii Strashko if (!tc_cls_can_offload_and_chain0(port->ndev, type_data))
7695ec836beSGrygorii Strashko return -EOPNOTSUPP;
7705ec836beSGrygorii Strashko
7715ec836beSGrygorii Strashko switch (type) {
7725ec836beSGrygorii Strashko case TC_SETUP_CLSFLOWER:
7735ec836beSGrygorii Strashko return am65_cpsw_qos_setup_tc_clsflower(port, type_data);
7745ec836beSGrygorii Strashko default:
7755ec836beSGrygorii Strashko return -EOPNOTSUPP;
7765ec836beSGrygorii Strashko }
7775ec836beSGrygorii Strashko }
7785ec836beSGrygorii Strashko
7795ec836beSGrygorii Strashko static LIST_HEAD(am65_cpsw_qos_block_cb_list);
7805ec836beSGrygorii Strashko
am65_cpsw_qos_setup_tc_block(struct net_device * ndev,struct flow_block_offload * f)7815ec836beSGrygorii Strashko static int am65_cpsw_qos_setup_tc_block(struct net_device *ndev, struct flow_block_offload *f)
7825ec836beSGrygorii Strashko {
7835ec836beSGrygorii Strashko struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
7845ec836beSGrygorii Strashko
7855ec836beSGrygorii Strashko return flow_block_cb_setup_simple(f, &am65_cpsw_qos_block_cb_list,
7865ec836beSGrygorii Strashko am65_cpsw_qos_setup_tc_block_cb,
7875ec836beSGrygorii Strashko port, port, true);
7885ec836beSGrygorii Strashko }
7895ec836beSGrygorii Strashko
am65_cpsw_qos_ndo_setup_tc(struct net_device * ndev,enum tc_setup_type type,void * type_data)7908127224cSIvan Khoronzhuk int am65_cpsw_qos_ndo_setup_tc(struct net_device *ndev, enum tc_setup_type type,
7918127224cSIvan Khoronzhuk void *type_data)
7928127224cSIvan Khoronzhuk {
7938127224cSIvan Khoronzhuk switch (type) {
794522d15eaSVladimir Oltean case TC_QUERY_CAPS:
795522d15eaSVladimir Oltean return am65_cpsw_tc_query_caps(ndev, type_data);
7968127224cSIvan Khoronzhuk case TC_SETUP_QDISC_TAPRIO:
7978127224cSIvan Khoronzhuk return am65_cpsw_setup_taprio(ndev, type_data);
7985ec836beSGrygorii Strashko case TC_SETUP_BLOCK:
7995ec836beSGrygorii Strashko return am65_cpsw_qos_setup_tc_block(ndev, type_data);
8008127224cSIvan Khoronzhuk default:
8018127224cSIvan Khoronzhuk return -EOPNOTSUPP;
8028127224cSIvan Khoronzhuk }
8038127224cSIvan Khoronzhuk }
8048127224cSIvan Khoronzhuk
am65_cpsw_qos_link_up(struct net_device * ndev,int link_speed)8058127224cSIvan Khoronzhuk void am65_cpsw_qos_link_up(struct net_device *ndev, int link_speed)
8068127224cSIvan Khoronzhuk {
8078127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
8088127224cSIvan Khoronzhuk
8098127224cSIvan Khoronzhuk if (!IS_ENABLED(CONFIG_TI_AM65_CPSW_TAS))
8108127224cSIvan Khoronzhuk return;
8118127224cSIvan Khoronzhuk
8128127224cSIvan Khoronzhuk am65_cpsw_est_link_up(ndev, link_speed);
8138127224cSIvan Khoronzhuk port->qos.link_down_time = 0;
8148127224cSIvan Khoronzhuk }
8158127224cSIvan Khoronzhuk
am65_cpsw_qos_link_down(struct net_device * ndev)8168127224cSIvan Khoronzhuk void am65_cpsw_qos_link_down(struct net_device *ndev)
8178127224cSIvan Khoronzhuk {
8188127224cSIvan Khoronzhuk struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
8198127224cSIvan Khoronzhuk
8208127224cSIvan Khoronzhuk if (!IS_ENABLED(CONFIG_TI_AM65_CPSW_TAS))
8218127224cSIvan Khoronzhuk return;
8228127224cSIvan Khoronzhuk
8238127224cSIvan Khoronzhuk if (!port->qos.link_down_time)
8248127224cSIvan Khoronzhuk port->qos.link_down_time = ktime_get();
8258127224cSIvan Khoronzhuk
8268127224cSIvan Khoronzhuk port->qos.link_speed = SPEED_UNKNOWN;
8278127224cSIvan Khoronzhuk }
8285c8560c4SGrygorii Strashko
8295c8560c4SGrygorii Strashko static u32
am65_cpsw_qos_tx_rate_calc(u32 rate_mbps,unsigned long bus_freq)8305c8560c4SGrygorii Strashko am65_cpsw_qos_tx_rate_calc(u32 rate_mbps, unsigned long bus_freq)
8315c8560c4SGrygorii Strashko {
8325c8560c4SGrygorii Strashko u32 ir;
8335c8560c4SGrygorii Strashko
8345c8560c4SGrygorii Strashko bus_freq /= 1000000;
8355c8560c4SGrygorii Strashko ir = DIV_ROUND_UP(((u64)rate_mbps * 32768), bus_freq);
8365c8560c4SGrygorii Strashko return ir;
8375c8560c4SGrygorii Strashko }
8385c8560c4SGrygorii Strashko
8395c8560c4SGrygorii Strashko static void
am65_cpsw_qos_tx_p0_rate_apply(struct am65_cpsw_common * common,int tx_ch,u32 rate_mbps)8405c8560c4SGrygorii Strashko am65_cpsw_qos_tx_p0_rate_apply(struct am65_cpsw_common *common,
8415c8560c4SGrygorii Strashko int tx_ch, u32 rate_mbps)
8425c8560c4SGrygorii Strashko {
8435c8560c4SGrygorii Strashko struct am65_cpsw_host *host = am65_common_get_host(common);
8445c8560c4SGrygorii Strashko u32 ch_cir;
8455c8560c4SGrygorii Strashko int i;
8465c8560c4SGrygorii Strashko
8475c8560c4SGrygorii Strashko ch_cir = am65_cpsw_qos_tx_rate_calc(rate_mbps, common->bus_freq);
8485c8560c4SGrygorii Strashko writel(ch_cir, host->port_base + AM65_CPSW_PN_REG_PRI_CIR(tx_ch));
8495c8560c4SGrygorii Strashko
8505c8560c4SGrygorii Strashko /* update rates for every port tx queues */
8515c8560c4SGrygorii Strashko for (i = 0; i < common->port_num; i++) {
8525c8560c4SGrygorii Strashko struct net_device *ndev = common->ports[i].ndev;
8535c8560c4SGrygorii Strashko
8545c8560c4SGrygorii Strashko if (!ndev)
8555c8560c4SGrygorii Strashko continue;
8565c8560c4SGrygorii Strashko netdev_get_tx_queue(ndev, tx_ch)->tx_maxrate = rate_mbps;
8575c8560c4SGrygorii Strashko }
8585c8560c4SGrygorii Strashko }
8595c8560c4SGrygorii Strashko
am65_cpsw_qos_ndo_tx_p0_set_maxrate(struct net_device * ndev,int queue,u32 rate_mbps)8605c8560c4SGrygorii Strashko int am65_cpsw_qos_ndo_tx_p0_set_maxrate(struct net_device *ndev,
8615c8560c4SGrygorii Strashko int queue, u32 rate_mbps)
8625c8560c4SGrygorii Strashko {
8635c8560c4SGrygorii Strashko struct am65_cpsw_port *port = am65_ndev_to_port(ndev);
8645c8560c4SGrygorii Strashko struct am65_cpsw_common *common = port->common;
8655c8560c4SGrygorii Strashko struct am65_cpsw_tx_chn *tx_chn;
8665c8560c4SGrygorii Strashko u32 ch_rate, tx_ch_rate_msk_new;
8675c8560c4SGrygorii Strashko u32 ch_msk = 0;
8685c8560c4SGrygorii Strashko int ret;
8695c8560c4SGrygorii Strashko
8705c8560c4SGrygorii Strashko dev_dbg(common->dev, "apply TX%d rate limiting %uMbps tx_rate_msk%x\n",
8715c8560c4SGrygorii Strashko queue, rate_mbps, common->tx_ch_rate_msk);
8725c8560c4SGrygorii Strashko
8735c8560c4SGrygorii Strashko if (common->pf_p0_rx_ptype_rrobin) {
8745c8560c4SGrygorii Strashko dev_err(common->dev, "TX Rate Limiting failed - rrobin mode\n");
8755c8560c4SGrygorii Strashko return -EINVAL;
8765c8560c4SGrygorii Strashko }
8775c8560c4SGrygorii Strashko
8785c8560c4SGrygorii Strashko ch_rate = netdev_get_tx_queue(ndev, queue)->tx_maxrate;
8795c8560c4SGrygorii Strashko if (ch_rate == rate_mbps)
8805c8560c4SGrygorii Strashko return 0;
8815c8560c4SGrygorii Strashko
8825c8560c4SGrygorii Strashko ret = pm_runtime_get_sync(common->dev);
8835c8560c4SGrygorii Strashko if (ret < 0) {
8845c8560c4SGrygorii Strashko pm_runtime_put_noidle(common->dev);
8855c8560c4SGrygorii Strashko return ret;
8865c8560c4SGrygorii Strashko }
8875c8560c4SGrygorii Strashko ret = 0;
8885c8560c4SGrygorii Strashko
8895c8560c4SGrygorii Strashko tx_ch_rate_msk_new = common->tx_ch_rate_msk;
8905c8560c4SGrygorii Strashko if (rate_mbps && !(tx_ch_rate_msk_new & BIT(queue))) {
8915c8560c4SGrygorii Strashko tx_ch_rate_msk_new |= BIT(queue);
8925c8560c4SGrygorii Strashko ch_msk = GENMASK(common->tx_ch_num - 1, queue);
8935c8560c4SGrygorii Strashko ch_msk = tx_ch_rate_msk_new ^ ch_msk;
8945c8560c4SGrygorii Strashko } else if (!rate_mbps) {
8955c8560c4SGrygorii Strashko tx_ch_rate_msk_new &= ~BIT(queue);
8965c8560c4SGrygorii Strashko ch_msk = queue ? GENMASK(queue - 1, 0) : 0;
8975c8560c4SGrygorii Strashko ch_msk = tx_ch_rate_msk_new & ch_msk;
8985c8560c4SGrygorii Strashko }
8995c8560c4SGrygorii Strashko
9005c8560c4SGrygorii Strashko if (ch_msk) {
9015c8560c4SGrygorii Strashko dev_err(common->dev, "TX rate limiting has to be enabled sequentially hi->lo tx_rate_msk:%x tx_rate_msk_new:%x\n",
9025c8560c4SGrygorii Strashko common->tx_ch_rate_msk, tx_ch_rate_msk_new);
9035c8560c4SGrygorii Strashko ret = -EINVAL;
9045c8560c4SGrygorii Strashko goto exit_put;
9055c8560c4SGrygorii Strashko }
9065c8560c4SGrygorii Strashko
9075c8560c4SGrygorii Strashko tx_chn = &common->tx_chns[queue];
9085c8560c4SGrygorii Strashko tx_chn->rate_mbps = rate_mbps;
9095c8560c4SGrygorii Strashko common->tx_ch_rate_msk = tx_ch_rate_msk_new;
9105c8560c4SGrygorii Strashko
9115c8560c4SGrygorii Strashko if (!common->usage_count)
9125c8560c4SGrygorii Strashko /* will be applied on next netif up */
9135c8560c4SGrygorii Strashko goto exit_put;
9145c8560c4SGrygorii Strashko
9155c8560c4SGrygorii Strashko am65_cpsw_qos_tx_p0_rate_apply(common, queue, rate_mbps);
9165c8560c4SGrygorii Strashko
9175c8560c4SGrygorii Strashko exit_put:
9185c8560c4SGrygorii Strashko pm_runtime_put(common->dev);
9195c8560c4SGrygorii Strashko return ret;
9205c8560c4SGrygorii Strashko }
9215c8560c4SGrygorii Strashko
am65_cpsw_qos_tx_p0_rate_init(struct am65_cpsw_common * common)9225c8560c4SGrygorii Strashko void am65_cpsw_qos_tx_p0_rate_init(struct am65_cpsw_common *common)
9235c8560c4SGrygorii Strashko {
9245c8560c4SGrygorii Strashko struct am65_cpsw_host *host = am65_common_get_host(common);
9255c8560c4SGrygorii Strashko int tx_ch;
9265c8560c4SGrygorii Strashko
9275c8560c4SGrygorii Strashko for (tx_ch = 0; tx_ch < common->tx_ch_num; tx_ch++) {
9285c8560c4SGrygorii Strashko struct am65_cpsw_tx_chn *tx_chn = &common->tx_chns[tx_ch];
9295c8560c4SGrygorii Strashko u32 ch_cir;
9305c8560c4SGrygorii Strashko
9315c8560c4SGrygorii Strashko if (!tx_chn->rate_mbps)
9325c8560c4SGrygorii Strashko continue;
9335c8560c4SGrygorii Strashko
9345c8560c4SGrygorii Strashko ch_cir = am65_cpsw_qos_tx_rate_calc(tx_chn->rate_mbps,
9355c8560c4SGrygorii Strashko common->bus_freq);
9365c8560c4SGrygorii Strashko writel(ch_cir,
9375c8560c4SGrygorii Strashko host->port_base + AM65_CPSW_PN_REG_PRI_CIR(tx_ch));
9385c8560c4SGrygorii Strashko }
9395c8560c4SGrygorii Strashko }
940