1d28d6d2eSHoratiu Vultur // SPDX-License-Identifier: GPL-2.0+
2d28d6d2eSHoratiu Vultur 
3d28d6d2eSHoratiu Vultur #include <linux/netdevice.h>
4d28d6d2eSHoratiu Vultur #include <linux/phy/phy.h>
5d28d6d2eSHoratiu Vultur 
6d28d6d2eSHoratiu Vultur #include "lan966x_main.h"
7d28d6d2eSHoratiu Vultur 
8d28d6d2eSHoratiu Vultur /* Watermark encode */
9d28d6d2eSHoratiu Vultur #define MULTIPLIER_BIT BIT(8)
lan966x_wm_enc(u32 value)10d28d6d2eSHoratiu Vultur static u32 lan966x_wm_enc(u32 value)
11d28d6d2eSHoratiu Vultur {
12d28d6d2eSHoratiu Vultur 	value /= LAN966X_BUFFER_CELL_SZ;
13d28d6d2eSHoratiu Vultur 
14d28d6d2eSHoratiu Vultur 	if (value >= MULTIPLIER_BIT) {
15d28d6d2eSHoratiu Vultur 		value /= 16;
16d28d6d2eSHoratiu Vultur 		if (value >= MULTIPLIER_BIT)
17d28d6d2eSHoratiu Vultur 			value = (MULTIPLIER_BIT - 1);
18d28d6d2eSHoratiu Vultur 
19d28d6d2eSHoratiu Vultur 		value |= MULTIPLIER_BIT;
20d28d6d2eSHoratiu Vultur 	}
21d28d6d2eSHoratiu Vultur 
22d28d6d2eSHoratiu Vultur 	return value;
23d28d6d2eSHoratiu Vultur }
24d28d6d2eSHoratiu Vultur 
lan966x_port_link_down(struct lan966x_port * port)25d28d6d2eSHoratiu Vultur static void lan966x_port_link_down(struct lan966x_port *port)
26d28d6d2eSHoratiu Vultur {
27d28d6d2eSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
28d28d6d2eSHoratiu Vultur 	u32 val, delay = 0;
29d28d6d2eSHoratiu Vultur 
30d28d6d2eSHoratiu Vultur 	/* 0.5: Disable any AFI */
31d28d6d2eSHoratiu Vultur 	lan_rmw(AFI_PORT_CFG_FC_SKIP_TTI_INJ_SET(1) |
32d28d6d2eSHoratiu Vultur 		AFI_PORT_CFG_FRM_OUT_MAX_SET(0),
33d28d6d2eSHoratiu Vultur 		AFI_PORT_CFG_FC_SKIP_TTI_INJ |
34d28d6d2eSHoratiu Vultur 		AFI_PORT_CFG_FRM_OUT_MAX,
35d28d6d2eSHoratiu Vultur 		lan966x, AFI_PORT_CFG(port->chip_port));
36d28d6d2eSHoratiu Vultur 
37d28d6d2eSHoratiu Vultur 	/* wait for reg afi_port_frm_out to become 0 for the port */
38d28d6d2eSHoratiu Vultur 	while (true) {
39d28d6d2eSHoratiu Vultur 		val = lan_rd(lan966x, AFI_PORT_FRM_OUT(port->chip_port));
40d28d6d2eSHoratiu Vultur 		if (!AFI_PORT_FRM_OUT_FRM_OUT_CNT_GET(val))
41d28d6d2eSHoratiu Vultur 			break;
42d28d6d2eSHoratiu Vultur 
43d28d6d2eSHoratiu Vultur 		usleep_range(USEC_PER_MSEC, 2 * USEC_PER_MSEC);
44d28d6d2eSHoratiu Vultur 		delay++;
45d28d6d2eSHoratiu Vultur 		if (delay == 2000) {
46d28d6d2eSHoratiu Vultur 			pr_err("AFI timeout chip port %u", port->chip_port);
47d28d6d2eSHoratiu Vultur 			break;
48d28d6d2eSHoratiu Vultur 		}
49d28d6d2eSHoratiu Vultur 	}
50d28d6d2eSHoratiu Vultur 
51d28d6d2eSHoratiu Vultur 	delay = 0;
52d28d6d2eSHoratiu Vultur 
53d28d6d2eSHoratiu Vultur 	/* 1: Reset the PCS Rx clock domain  */
54d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_CLOCK_CFG_PCS_RX_RST_SET(1),
55d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_RX_RST,
56d28d6d2eSHoratiu Vultur 		lan966x, DEV_CLOCK_CFG(port->chip_port));
57d28d6d2eSHoratiu Vultur 
58d28d6d2eSHoratiu Vultur 	/* 2: Disable MAC frame reception */
59d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_MAC_ENA_CFG_RX_ENA_SET(0),
60d28d6d2eSHoratiu Vultur 		DEV_MAC_ENA_CFG_RX_ENA,
61d28d6d2eSHoratiu Vultur 		lan966x, DEV_MAC_ENA_CFG(port->chip_port));
62d28d6d2eSHoratiu Vultur 
63d28d6d2eSHoratiu Vultur 	/* 3: Disable traffic being sent to or from switch port */
64d28d6d2eSHoratiu Vultur 	lan_rmw(QSYS_SW_PORT_MODE_PORT_ENA_SET(0),
65d28d6d2eSHoratiu Vultur 		QSYS_SW_PORT_MODE_PORT_ENA,
66d28d6d2eSHoratiu Vultur 		lan966x, QSYS_SW_PORT_MODE(port->chip_port));
67d28d6d2eSHoratiu Vultur 
68d28d6d2eSHoratiu Vultur 	/* 4: Disable dequeuing from the egress queues  */
69d28d6d2eSHoratiu Vultur 	lan_rmw(QSYS_PORT_MODE_DEQUEUE_DIS_SET(1),
70d28d6d2eSHoratiu Vultur 		QSYS_PORT_MODE_DEQUEUE_DIS,
71d28d6d2eSHoratiu Vultur 		lan966x, QSYS_PORT_MODE(port->chip_port));
72d28d6d2eSHoratiu Vultur 
73d28d6d2eSHoratiu Vultur 	/* 5: Disable Flowcontrol */
74d28d6d2eSHoratiu Vultur 	lan_rmw(SYS_PAUSE_CFG_PAUSE_ENA_SET(0),
75d28d6d2eSHoratiu Vultur 		SYS_PAUSE_CFG_PAUSE_ENA,
76d28d6d2eSHoratiu Vultur 		lan966x, SYS_PAUSE_CFG(port->chip_port));
77d28d6d2eSHoratiu Vultur 
78d28d6d2eSHoratiu Vultur 	/* 5.1: Disable PFC */
79d28d6d2eSHoratiu Vultur 	lan_rmw(QSYS_SW_PORT_MODE_TX_PFC_ENA_SET(0),
80d28d6d2eSHoratiu Vultur 		QSYS_SW_PORT_MODE_TX_PFC_ENA,
81d28d6d2eSHoratiu Vultur 		lan966x, QSYS_SW_PORT_MODE(port->chip_port));
82d28d6d2eSHoratiu Vultur 
83d28d6d2eSHoratiu Vultur 	/* 6: Wait a worst case time 8ms (jumbo/10Mbit) */
84d28d6d2eSHoratiu Vultur 	usleep_range(8 * USEC_PER_MSEC, 9 * USEC_PER_MSEC);
85d28d6d2eSHoratiu Vultur 
86d28d6d2eSHoratiu Vultur 	/* 7: Disable HDX backpressure */
87d28d6d2eSHoratiu Vultur 	lan_rmw(SYS_FRONT_PORT_MODE_HDX_MODE_SET(0),
88d28d6d2eSHoratiu Vultur 		SYS_FRONT_PORT_MODE_HDX_MODE,
89d28d6d2eSHoratiu Vultur 		lan966x, SYS_FRONT_PORT_MODE(port->chip_port));
90d28d6d2eSHoratiu Vultur 
91d28d6d2eSHoratiu Vultur 	/* 8: Flush the queues accociated with the port */
92d28d6d2eSHoratiu Vultur 	lan_rmw(QSYS_SW_PORT_MODE_AGING_MODE_SET(3),
93d28d6d2eSHoratiu Vultur 		QSYS_SW_PORT_MODE_AGING_MODE,
94d28d6d2eSHoratiu Vultur 		lan966x, QSYS_SW_PORT_MODE(port->chip_port));
95d28d6d2eSHoratiu Vultur 
96d28d6d2eSHoratiu Vultur 	/* 9: Enable dequeuing from the egress queues */
97d28d6d2eSHoratiu Vultur 	lan_rmw(QSYS_PORT_MODE_DEQUEUE_DIS_SET(0),
98d28d6d2eSHoratiu Vultur 		QSYS_PORT_MODE_DEQUEUE_DIS,
99d28d6d2eSHoratiu Vultur 		lan966x, QSYS_PORT_MODE(port->chip_port));
100d28d6d2eSHoratiu Vultur 
101d28d6d2eSHoratiu Vultur 	/* 10: Wait until flushing is complete */
102d28d6d2eSHoratiu Vultur 	while (true) {
103d28d6d2eSHoratiu Vultur 		val = lan_rd(lan966x, QSYS_SW_STATUS(port->chip_port));
104d28d6d2eSHoratiu Vultur 		if (!QSYS_SW_STATUS_EQ_AVAIL_GET(val))
105d28d6d2eSHoratiu Vultur 			break;
106d28d6d2eSHoratiu Vultur 
107d28d6d2eSHoratiu Vultur 		usleep_range(USEC_PER_MSEC, 2 * USEC_PER_MSEC);
108d28d6d2eSHoratiu Vultur 		delay++;
109d28d6d2eSHoratiu Vultur 		if (delay == 2000) {
110d28d6d2eSHoratiu Vultur 			pr_err("Flush timeout chip port %u", port->chip_port);
111d28d6d2eSHoratiu Vultur 			break;
112d28d6d2eSHoratiu Vultur 		}
113d28d6d2eSHoratiu Vultur 	}
114d28d6d2eSHoratiu Vultur 
115d28d6d2eSHoratiu Vultur 	/* 11: Reset the Port and MAC clock domains */
116d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_MAC_ENA_CFG_TX_ENA_SET(0),
117d28d6d2eSHoratiu Vultur 		DEV_MAC_ENA_CFG_TX_ENA,
118d28d6d2eSHoratiu Vultur 		lan966x, DEV_MAC_ENA_CFG(port->chip_port));
119d28d6d2eSHoratiu Vultur 
120d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_CLOCK_CFG_PORT_RST_SET(1),
121d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PORT_RST,
122d28d6d2eSHoratiu Vultur 		lan966x, DEV_CLOCK_CFG(port->chip_port));
123d28d6d2eSHoratiu Vultur 
124d28d6d2eSHoratiu Vultur 	usleep_range(USEC_PER_MSEC, 2 * USEC_PER_MSEC);
125d28d6d2eSHoratiu Vultur 
126d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_CLOCK_CFG_MAC_TX_RST_SET(1) |
127d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_MAC_RX_RST_SET(1) |
128d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PORT_RST_SET(1),
129d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_MAC_TX_RST |
130d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_MAC_RX_RST |
131d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PORT_RST,
132d28d6d2eSHoratiu Vultur 		lan966x, DEV_CLOCK_CFG(port->chip_port));
133d28d6d2eSHoratiu Vultur 
134d28d6d2eSHoratiu Vultur 	/* 12: Clear flushing */
135d28d6d2eSHoratiu Vultur 	lan_rmw(QSYS_SW_PORT_MODE_AGING_MODE_SET(2),
136d28d6d2eSHoratiu Vultur 		QSYS_SW_PORT_MODE_AGING_MODE,
137d28d6d2eSHoratiu Vultur 		lan966x, QSYS_SW_PORT_MODE(port->chip_port));
138d28d6d2eSHoratiu Vultur 
139d28d6d2eSHoratiu Vultur 	/* The port is disabled and flushed, now set up the port in the
140d28d6d2eSHoratiu Vultur 	 * new operating mode
141d28d6d2eSHoratiu Vultur 	 */
142d28d6d2eSHoratiu Vultur }
143d28d6d2eSHoratiu Vultur 
lan966x_port_link_up(struct lan966x_port * port)144d28d6d2eSHoratiu Vultur static void lan966x_port_link_up(struct lan966x_port *port)
145d28d6d2eSHoratiu Vultur {
146d28d6d2eSHoratiu Vultur 	struct lan966x_port_config *config = &port->config;
147d28d6d2eSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
148d28d6d2eSHoratiu Vultur 	int speed = 0, mode = 0;
149d28d6d2eSHoratiu Vultur 	int atop_wm = 0;
150d28d6d2eSHoratiu Vultur 
151d28d6d2eSHoratiu Vultur 	switch (config->speed) {
152d28d6d2eSHoratiu Vultur 	case SPEED_10:
153d28d6d2eSHoratiu Vultur 		speed = LAN966X_SPEED_10;
154d28d6d2eSHoratiu Vultur 		break;
155d28d6d2eSHoratiu Vultur 	case SPEED_100:
156d28d6d2eSHoratiu Vultur 		speed = LAN966X_SPEED_100;
157d28d6d2eSHoratiu Vultur 		break;
158d28d6d2eSHoratiu Vultur 	case SPEED_1000:
159d28d6d2eSHoratiu Vultur 		speed = LAN966X_SPEED_1000;
160d28d6d2eSHoratiu Vultur 		mode = DEV_MAC_MODE_CFG_GIGA_MODE_ENA_SET(1);
161d28d6d2eSHoratiu Vultur 		break;
162d28d6d2eSHoratiu Vultur 	case SPEED_2500:
163d28d6d2eSHoratiu Vultur 		speed = LAN966X_SPEED_2500;
164d28d6d2eSHoratiu Vultur 		mode = DEV_MAC_MODE_CFG_GIGA_MODE_ENA_SET(1);
165d28d6d2eSHoratiu Vultur 		break;
166d28d6d2eSHoratiu Vultur 	}
167d28d6d2eSHoratiu Vultur 
168e462b271SHoratiu Vultur 	lan966x_taprio_speed_set(port, config->speed);
169e462b271SHoratiu Vultur 
170d28d6d2eSHoratiu Vultur 	/* Also the GIGA_MODE_ENA(1) needs to be set regardless of the
171*f0025c92SHoratiu Vultur 	 * port speed for QSGMII or SGMII ports.
172d28d6d2eSHoratiu Vultur 	 */
173*f0025c92SHoratiu Vultur 	if (phy_interface_num_ports(config->portmode) == 4 ||
174*f0025c92SHoratiu Vultur 	    config->portmode == PHY_INTERFACE_MODE_SGMII)
175d28d6d2eSHoratiu Vultur 		mode = DEV_MAC_MODE_CFG_GIGA_MODE_ENA_SET(1);
176d28d6d2eSHoratiu Vultur 
177d28d6d2eSHoratiu Vultur 	lan_wr(config->duplex | mode,
178d28d6d2eSHoratiu Vultur 	       lan966x, DEV_MAC_MODE_CFG(port->chip_port));
179d28d6d2eSHoratiu Vultur 
180d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_MAC_IFG_CFG_TX_IFG_SET(config->duplex ? 6 : 5) |
181d28d6d2eSHoratiu Vultur 		DEV_MAC_IFG_CFG_RX_IFG1_SET(config->speed == SPEED_10 ? 2 : 1) |
182d28d6d2eSHoratiu Vultur 		DEV_MAC_IFG_CFG_RX_IFG2_SET(2),
183d28d6d2eSHoratiu Vultur 		DEV_MAC_IFG_CFG_TX_IFG |
184d28d6d2eSHoratiu Vultur 		DEV_MAC_IFG_CFG_RX_IFG1 |
185d28d6d2eSHoratiu Vultur 		DEV_MAC_IFG_CFG_RX_IFG2,
186d28d6d2eSHoratiu Vultur 		lan966x, DEV_MAC_IFG_CFG(port->chip_port));
187d28d6d2eSHoratiu Vultur 
188d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_MAC_HDX_CFG_SEED_SET(4) |
189d28d6d2eSHoratiu Vultur 		DEV_MAC_HDX_CFG_SEED_LOAD_SET(1),
190d28d6d2eSHoratiu Vultur 		DEV_MAC_HDX_CFG_SEED |
191d28d6d2eSHoratiu Vultur 		DEV_MAC_HDX_CFG_SEED_LOAD,
192d28d6d2eSHoratiu Vultur 		lan966x, DEV_MAC_HDX_CFG(port->chip_port));
193d28d6d2eSHoratiu Vultur 
194d28d6d2eSHoratiu Vultur 	if (config->portmode == PHY_INTERFACE_MODE_GMII) {
195d28d6d2eSHoratiu Vultur 		if (config->speed == SPEED_1000)
196d28d6d2eSHoratiu Vultur 			lan_rmw(CHIP_TOP_CUPHY_PORT_CFG_GTX_CLK_ENA_SET(1),
197d28d6d2eSHoratiu Vultur 				CHIP_TOP_CUPHY_PORT_CFG_GTX_CLK_ENA,
198d28d6d2eSHoratiu Vultur 				lan966x,
199d28d6d2eSHoratiu Vultur 				CHIP_TOP_CUPHY_PORT_CFG(port->chip_port));
200d28d6d2eSHoratiu Vultur 		else
201d28d6d2eSHoratiu Vultur 			lan_rmw(CHIP_TOP_CUPHY_PORT_CFG_GTX_CLK_ENA_SET(0),
202d28d6d2eSHoratiu Vultur 				CHIP_TOP_CUPHY_PORT_CFG_GTX_CLK_ENA,
203d28d6d2eSHoratiu Vultur 				lan966x,
204d28d6d2eSHoratiu Vultur 				CHIP_TOP_CUPHY_PORT_CFG(port->chip_port));
205d28d6d2eSHoratiu Vultur 	}
206d28d6d2eSHoratiu Vultur 
207d28d6d2eSHoratiu Vultur 	/* No PFC */
208d28d6d2eSHoratiu Vultur 	lan_wr(ANA_PFC_CFG_FC_LINK_SPEED_SET(speed),
209d28d6d2eSHoratiu Vultur 	       lan966x, ANA_PFC_CFG(port->chip_port));
210d28d6d2eSHoratiu Vultur 
211d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_PCS1G_CFG_PCS_ENA_SET(1),
212d28d6d2eSHoratiu Vultur 		DEV_PCS1G_CFG_PCS_ENA,
213d28d6d2eSHoratiu Vultur 		lan966x, DEV_PCS1G_CFG(port->chip_port));
214d28d6d2eSHoratiu Vultur 
215d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_PCS1G_SD_CFG_SD_ENA_SET(0),
216d28d6d2eSHoratiu Vultur 		DEV_PCS1G_SD_CFG_SD_ENA,
217d28d6d2eSHoratiu Vultur 		lan966x, DEV_PCS1G_SD_CFG(port->chip_port));
218d28d6d2eSHoratiu Vultur 
219d28d6d2eSHoratiu Vultur 	/* Set Pause WM hysteresis, start/stop are in 1518 byte units */
220d28d6d2eSHoratiu Vultur 	lan_wr(SYS_PAUSE_CFG_PAUSE_ENA_SET(1) |
221d28d6d2eSHoratiu Vultur 	       SYS_PAUSE_CFG_PAUSE_STOP_SET(lan966x_wm_enc(4 * 1518)) |
222d28d6d2eSHoratiu Vultur 	       SYS_PAUSE_CFG_PAUSE_START_SET(lan966x_wm_enc(6 * 1518)),
223d28d6d2eSHoratiu Vultur 	       lan966x, SYS_PAUSE_CFG(port->chip_port));
224d28d6d2eSHoratiu Vultur 
225d28d6d2eSHoratiu Vultur 	/* Set SMAC of Pause frame (00:00:00:00:00:00) */
226d28d6d2eSHoratiu Vultur 	lan_wr(0, lan966x, DEV_FC_MAC_LOW_CFG(port->chip_port));
227d28d6d2eSHoratiu Vultur 	lan_wr(0, lan966x, DEV_FC_MAC_HIGH_CFG(port->chip_port));
228d28d6d2eSHoratiu Vultur 
229d28d6d2eSHoratiu Vultur 	/* Flow control */
230d28d6d2eSHoratiu Vultur 	lan_rmw(SYS_MAC_FC_CFG_FC_LINK_SPEED_SET(speed) |
231d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_FC_LATENCY_CFG_SET(7) |
232d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_ZERO_PAUSE_ENA_SET(1) |
233d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_PAUSE_VAL_CFG_SET(0xffff) |
234d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_RX_FC_ENA_SET(config->pause & MLO_PAUSE_RX ? 1 : 0) |
235d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_TX_FC_ENA_SET(config->pause & MLO_PAUSE_TX ? 1 : 0),
236d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_FC_LINK_SPEED |
237d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_FC_LATENCY_CFG |
238d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_ZERO_PAUSE_ENA |
239d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_PAUSE_VAL_CFG |
240d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_RX_FC_ENA |
241d28d6d2eSHoratiu Vultur 		SYS_MAC_FC_CFG_TX_FC_ENA,
242d28d6d2eSHoratiu Vultur 		lan966x, SYS_MAC_FC_CFG(port->chip_port));
243d28d6d2eSHoratiu Vultur 
244d28d6d2eSHoratiu Vultur 	/* Tail dropping watermark */
245d28d6d2eSHoratiu Vultur 	atop_wm = lan966x->shared_queue_sz;
246d28d6d2eSHoratiu Vultur 
247d28d6d2eSHoratiu Vultur 	/* The total memory size is diveded by number of front ports plus CPU
248d28d6d2eSHoratiu Vultur 	 * port
249d28d6d2eSHoratiu Vultur 	 */
250d28d6d2eSHoratiu Vultur 	lan_wr(lan966x_wm_enc(atop_wm / lan966x->num_phys_ports + 1), lan966x,
251d28d6d2eSHoratiu Vultur 	       SYS_ATOP(port->chip_port));
252d28d6d2eSHoratiu Vultur 	lan_wr(lan966x_wm_enc(atop_wm), lan966x, SYS_ATOP_TOT_CFG);
253d28d6d2eSHoratiu Vultur 
254d28d6d2eSHoratiu Vultur 	/* This needs to be at the end */
255d28d6d2eSHoratiu Vultur 	/* Enable MAC module */
256d28d6d2eSHoratiu Vultur 	lan_wr(DEV_MAC_ENA_CFG_RX_ENA_SET(1) |
257d28d6d2eSHoratiu Vultur 	       DEV_MAC_ENA_CFG_TX_ENA_SET(1),
258d28d6d2eSHoratiu Vultur 	       lan966x, DEV_MAC_ENA_CFG(port->chip_port));
259d28d6d2eSHoratiu Vultur 
260d28d6d2eSHoratiu Vultur 	/* Take out the clock from reset */
261d28d6d2eSHoratiu Vultur 	lan_wr(DEV_CLOCK_CFG_LINK_SPEED_SET(speed),
262d28d6d2eSHoratiu Vultur 	       lan966x, DEV_CLOCK_CFG(port->chip_port));
263d28d6d2eSHoratiu Vultur 
264d28d6d2eSHoratiu Vultur 	/* Core: Enable port for frame transfer */
265d28d6d2eSHoratiu Vultur 	lan_wr(QSYS_SW_PORT_MODE_PORT_ENA_SET(1) |
266d28d6d2eSHoratiu Vultur 	       QSYS_SW_PORT_MODE_SCH_NEXT_CFG_SET(1) |
267d28d6d2eSHoratiu Vultur 	       QSYS_SW_PORT_MODE_INGRESS_DROP_MODE_SET(1),
268d28d6d2eSHoratiu Vultur 	       lan966x, QSYS_SW_PORT_MODE(port->chip_port));
269d28d6d2eSHoratiu Vultur 
270d28d6d2eSHoratiu Vultur 	lan_rmw(AFI_PORT_CFG_FC_SKIP_TTI_INJ_SET(0) |
271d28d6d2eSHoratiu Vultur 		AFI_PORT_CFG_FRM_OUT_MAX_SET(16),
272d28d6d2eSHoratiu Vultur 		AFI_PORT_CFG_FC_SKIP_TTI_INJ |
273d28d6d2eSHoratiu Vultur 		AFI_PORT_CFG_FRM_OUT_MAX,
274d28d6d2eSHoratiu Vultur 		lan966x, AFI_PORT_CFG(port->chip_port));
275d28d6d2eSHoratiu Vultur }
276d28d6d2eSHoratiu Vultur 
lan966x_port_config_down(struct lan966x_port * port)277d28d6d2eSHoratiu Vultur void lan966x_port_config_down(struct lan966x_port *port)
278d28d6d2eSHoratiu Vultur {
279d28d6d2eSHoratiu Vultur 	lan966x_port_link_down(port);
280d28d6d2eSHoratiu Vultur }
281d28d6d2eSHoratiu Vultur 
lan966x_port_config_up(struct lan966x_port * port)282d28d6d2eSHoratiu Vultur void lan966x_port_config_up(struct lan966x_port *port)
283d28d6d2eSHoratiu Vultur {
284d28d6d2eSHoratiu Vultur 	lan966x_port_link_up(port);
285d28d6d2eSHoratiu Vultur }
286d28d6d2eSHoratiu Vultur 
lan966x_port_status_get(struct lan966x_port * port,struct phylink_link_state * state)287d28d6d2eSHoratiu Vultur void lan966x_port_status_get(struct lan966x_port *port,
288d28d6d2eSHoratiu Vultur 			     struct phylink_link_state *state)
289d28d6d2eSHoratiu Vultur {
290d28d6d2eSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
291d28d6d2eSHoratiu Vultur 	bool link_down;
292d28d6d2eSHoratiu Vultur 	u16 bmsr = 0;
293d28d6d2eSHoratiu Vultur 	u16 lp_adv;
294d28d6d2eSHoratiu Vultur 	u32 val;
295d28d6d2eSHoratiu Vultur 
296d28d6d2eSHoratiu Vultur 	val = lan_rd(lan966x, DEV_PCS1G_STICKY(port->chip_port));
297d28d6d2eSHoratiu Vultur 	link_down = DEV_PCS1G_STICKY_LINK_DOWN_STICKY_GET(val);
298d28d6d2eSHoratiu Vultur 	if (link_down)
299d28d6d2eSHoratiu Vultur 		lan_wr(val, lan966x, DEV_PCS1G_STICKY(port->chip_port));
300d28d6d2eSHoratiu Vultur 
301d28d6d2eSHoratiu Vultur 	/* Get both current Link and Sync status */
302d28d6d2eSHoratiu Vultur 	val = lan_rd(lan966x, DEV_PCS1G_LINK_STATUS(port->chip_port));
303d28d6d2eSHoratiu Vultur 	state->link = DEV_PCS1G_LINK_STATUS_LINK_STATUS_GET(val) &&
304d28d6d2eSHoratiu Vultur 		      DEV_PCS1G_LINK_STATUS_SYNC_STATUS_GET(val);
305d28d6d2eSHoratiu Vultur 	state->link &= !link_down;
306d28d6d2eSHoratiu Vultur 
307d28d6d2eSHoratiu Vultur 	/* Get PCS ANEG status register */
308d28d6d2eSHoratiu Vultur 	val = lan_rd(lan966x, DEV_PCS1G_ANEG_STATUS(port->chip_port));
309d28d6d2eSHoratiu Vultur 	/* Aneg complete provides more information  */
310d28d6d2eSHoratiu Vultur 	if (DEV_PCS1G_ANEG_STATUS_ANEG_COMPLETE_GET(val)) {
311d28d6d2eSHoratiu Vultur 		state->an_complete = true;
312d28d6d2eSHoratiu Vultur 
313d28d6d2eSHoratiu Vultur 		bmsr |= state->link ? BMSR_LSTATUS : 0;
314d28d6d2eSHoratiu Vultur 		bmsr |= BMSR_ANEGCOMPLETE;
315d28d6d2eSHoratiu Vultur 
316d28d6d2eSHoratiu Vultur 		lp_adv = DEV_PCS1G_ANEG_STATUS_LP_ADV_GET(val);
317d28d6d2eSHoratiu Vultur 		phylink_mii_c22_pcs_decode_state(state, bmsr, lp_adv);
318d28d6d2eSHoratiu Vultur 	} else {
319d28d6d2eSHoratiu Vultur 		if (!state->link)
320d28d6d2eSHoratiu Vultur 			return;
321d28d6d2eSHoratiu Vultur 
322d28d6d2eSHoratiu Vultur 		if (state->interface == PHY_INTERFACE_MODE_1000BASEX)
323d28d6d2eSHoratiu Vultur 			state->speed = SPEED_1000;
324d28d6d2eSHoratiu Vultur 		else if (state->interface == PHY_INTERFACE_MODE_2500BASEX)
325d28d6d2eSHoratiu Vultur 			state->speed = SPEED_2500;
326d28d6d2eSHoratiu Vultur 
327d28d6d2eSHoratiu Vultur 		state->duplex = DUPLEX_FULL;
328d28d6d2eSHoratiu Vultur 	}
329d28d6d2eSHoratiu Vultur }
330d28d6d2eSHoratiu Vultur 
lan966x_port_pcs_set(struct lan966x_port * port,struct lan966x_port_config * config)331d28d6d2eSHoratiu Vultur int lan966x_port_pcs_set(struct lan966x_port *port,
332d28d6d2eSHoratiu Vultur 			 struct lan966x_port_config *config)
333d28d6d2eSHoratiu Vultur {
334d28d6d2eSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
335d28d6d2eSHoratiu Vultur 	bool inband_aneg = false;
336d28d6d2eSHoratiu Vultur 	bool outband;
337ac0167fbSMaxime Chevallier 	bool full_preamble = false;
338ac0167fbSMaxime Chevallier 
339ac0167fbSMaxime Chevallier 	if (config->portmode == PHY_INTERFACE_MODE_QUSGMII)
340ac0167fbSMaxime Chevallier 		full_preamble = true;
341d28d6d2eSHoratiu Vultur 
342d28d6d2eSHoratiu Vultur 	if (config->inband) {
343d28d6d2eSHoratiu Vultur 		if (config->portmode == PHY_INTERFACE_MODE_SGMII ||
344ac0167fbSMaxime Chevallier 		    phy_interface_num_ports(config->portmode) == 4)
345d28d6d2eSHoratiu Vultur 			inband_aneg = true; /* Cisco-SGMII in-band-aneg */
346d28d6d2eSHoratiu Vultur 		else if (config->portmode == PHY_INTERFACE_MODE_1000BASEX &&
347d28d6d2eSHoratiu Vultur 			 config->autoneg)
348d28d6d2eSHoratiu Vultur 			inband_aneg = true; /* Clause-37 in-band-aneg */
349d28d6d2eSHoratiu Vultur 
350d28d6d2eSHoratiu Vultur 		outband = false;
351d28d6d2eSHoratiu Vultur 	} else {
352d28d6d2eSHoratiu Vultur 		outband = true;
353d28d6d2eSHoratiu Vultur 	}
354d28d6d2eSHoratiu Vultur 
355ac0167fbSMaxime Chevallier 	/* Disable or enable inband.
356ac0167fbSMaxime Chevallier 	 * For QUSGMII, we rely on the preamble to transmit data such as
357ac0167fbSMaxime Chevallier 	 * timestamps, therefore force full preamble transmission, and prevent
358ac0167fbSMaxime Chevallier 	 * premable shortening
359ac0167fbSMaxime Chevallier 	 */
360ac0167fbSMaxime Chevallier 	lan_rmw(DEV_PCS1G_MODE_CFG_SGMII_MODE_ENA_SET(outband) |
361ac0167fbSMaxime Chevallier 		DEV_PCS1G_MODE_CFG_SAVE_PREAMBLE_ENA_SET(full_preamble),
362ac0167fbSMaxime Chevallier 		DEV_PCS1G_MODE_CFG_SGMII_MODE_ENA |
363ac0167fbSMaxime Chevallier 		DEV_PCS1G_MODE_CFG_SAVE_PREAMBLE_ENA,
364d28d6d2eSHoratiu Vultur 		lan966x, DEV_PCS1G_MODE_CFG(port->chip_port));
365d28d6d2eSHoratiu Vultur 
366d28d6d2eSHoratiu Vultur 	/* Enable PCS */
367d28d6d2eSHoratiu Vultur 	lan_wr(DEV_PCS1G_CFG_PCS_ENA_SET(1),
368d28d6d2eSHoratiu Vultur 	       lan966x, DEV_PCS1G_CFG(port->chip_port));
369d28d6d2eSHoratiu Vultur 
370d28d6d2eSHoratiu Vultur 	if (inband_aneg) {
371d28d6d2eSHoratiu Vultur 		int adv = phylink_mii_c22_pcs_encode_advertisement(config->portmode,
372d28d6d2eSHoratiu Vultur 								   config->advertising);
373d28d6d2eSHoratiu Vultur 		if (adv >= 0)
374d28d6d2eSHoratiu Vultur 			/* Enable in-band aneg */
375d28d6d2eSHoratiu Vultur 			lan_wr(DEV_PCS1G_ANEG_CFG_ADV_ABILITY_SET(adv) |
376d28d6d2eSHoratiu Vultur 			       DEV_PCS1G_ANEG_CFG_SW_RESOLVE_ENA_SET(1) |
377d28d6d2eSHoratiu Vultur 			       DEV_PCS1G_ANEG_CFG_ENA_SET(1) |
378d28d6d2eSHoratiu Vultur 			       DEV_PCS1G_ANEG_CFG_RESTART_ONE_SHOT_SET(1),
379d28d6d2eSHoratiu Vultur 			       lan966x, DEV_PCS1G_ANEG_CFG(port->chip_port));
380d28d6d2eSHoratiu Vultur 	} else {
381d28d6d2eSHoratiu Vultur 		lan_wr(0, lan966x, DEV_PCS1G_ANEG_CFG(port->chip_port));
382d28d6d2eSHoratiu Vultur 	}
383d28d6d2eSHoratiu Vultur 
384d28d6d2eSHoratiu Vultur 	/* Take PCS out of reset */
385d717f947SHoratiu Vultur 	lan_rmw(DEV_CLOCK_CFG_LINK_SPEED_SET(LAN966X_SPEED_1000) |
386d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_RX_RST_SET(0) |
387d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_TX_RST_SET(0),
388d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_LINK_SPEED |
389d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_RX_RST |
390d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_TX_RST,
391d28d6d2eSHoratiu Vultur 		lan966x, DEV_CLOCK_CFG(port->chip_port));
392d28d6d2eSHoratiu Vultur 
393d28d6d2eSHoratiu Vultur 	port->config = *config;
394d28d6d2eSHoratiu Vultur 
395d28d6d2eSHoratiu Vultur 	return 0;
396d28d6d2eSHoratiu Vultur }
397d28d6d2eSHoratiu Vultur 
lan966x_port_qos_pcp_set(struct lan966x_port * port,struct lan966x_port_qos_pcp * qos)398a83e4630SHoratiu Vultur static void lan966x_port_qos_pcp_set(struct lan966x_port *port,
399a83e4630SHoratiu Vultur 				     struct lan966x_port_qos_pcp *qos)
400a83e4630SHoratiu Vultur {
401a83e4630SHoratiu Vultur 	u8 *pcp_itr = qos->map;
402a83e4630SHoratiu Vultur 	u8 pcp, dp;
403a83e4630SHoratiu Vultur 
404a83e4630SHoratiu Vultur 	lan_rmw(ANA_QOS_CFG_QOS_PCP_ENA_SET(qos->enable),
405a83e4630SHoratiu Vultur 		ANA_QOS_CFG_QOS_PCP_ENA,
406a83e4630SHoratiu Vultur 		port->lan966x, ANA_QOS_CFG(port->chip_port));
407a83e4630SHoratiu Vultur 
408a83e4630SHoratiu Vultur 	/* Map PCP and DEI to priority */
409a83e4630SHoratiu Vultur 	for (int i = 0; i < ARRAY_SIZE(qos->map); i++) {
410a83e4630SHoratiu Vultur 		pcp = *(pcp_itr + i);
411a83e4630SHoratiu Vultur 		dp = (i < LAN966X_PORT_QOS_PCP_COUNT) ? 0 : 1;
412a83e4630SHoratiu Vultur 
413a83e4630SHoratiu Vultur 		lan_rmw(ANA_PCP_DEI_CFG_QOS_PCP_DEI_VAL_SET(pcp) |
414a83e4630SHoratiu Vultur 			ANA_PCP_DEI_CFG_DP_PCP_DEI_VAL_SET(dp),
415a83e4630SHoratiu Vultur 			ANA_PCP_DEI_CFG_QOS_PCP_DEI_VAL |
416a83e4630SHoratiu Vultur 			ANA_PCP_DEI_CFG_DP_PCP_DEI_VAL,
417a83e4630SHoratiu Vultur 			port->lan966x,
418a83e4630SHoratiu Vultur 			ANA_PCP_DEI_CFG(port->chip_port, i));
419a83e4630SHoratiu Vultur 	}
420a83e4630SHoratiu Vultur }
421a83e4630SHoratiu Vultur 
lan966x_port_qos_dscp_set(struct lan966x_port * port,struct lan966x_port_qos_dscp * qos)4220c88d981SHoratiu Vultur static void lan966x_port_qos_dscp_set(struct lan966x_port *port,
4230c88d981SHoratiu Vultur 				      struct lan966x_port_qos_dscp *qos)
4240c88d981SHoratiu Vultur {
4250c88d981SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
4260c88d981SHoratiu Vultur 
4270c88d981SHoratiu Vultur 	/* Enable/disable dscp for qos classification. */
4280c88d981SHoratiu Vultur 	lan_rmw(ANA_QOS_CFG_QOS_DSCP_ENA_SET(qos->enable),
4290c88d981SHoratiu Vultur 		ANA_QOS_CFG_QOS_DSCP_ENA,
4300c88d981SHoratiu Vultur 		lan966x, ANA_QOS_CFG(port->chip_port));
4310c88d981SHoratiu Vultur 
4320c88d981SHoratiu Vultur 	/* Map each dscp value to priority and dp */
4330c88d981SHoratiu Vultur 	for (int i = 0; i < ARRAY_SIZE(qos->map); i++)
4340c88d981SHoratiu Vultur 		lan_rmw(ANA_DSCP_CFG_DP_DSCP_VAL_SET(0) |
4350c88d981SHoratiu Vultur 			ANA_DSCP_CFG_QOS_DSCP_VAL_SET(*(qos->map + i)),
4360c88d981SHoratiu Vultur 			ANA_DSCP_CFG_DP_DSCP_VAL |
4370c88d981SHoratiu Vultur 			ANA_DSCP_CFG_QOS_DSCP_VAL,
4380c88d981SHoratiu Vultur 			lan966x, ANA_DSCP_CFG(i));
4390c88d981SHoratiu Vultur 
4400c88d981SHoratiu Vultur 	/* Set per-dscp trust */
4410c88d981SHoratiu Vultur 	for (int i = 0; i <  ARRAY_SIZE(qos->map); i++)
4420c88d981SHoratiu Vultur 		lan_rmw(ANA_DSCP_CFG_DSCP_TRUST_ENA_SET(qos->enable),
4430c88d981SHoratiu Vultur 			ANA_DSCP_CFG_DSCP_TRUST_ENA,
4440c88d981SHoratiu Vultur 			lan966x, ANA_DSCP_CFG(i));
4450c88d981SHoratiu Vultur }
4460c88d981SHoratiu Vultur 
lan966x_port_qos_default_set(struct lan966x_port * port,struct lan966x_port_qos * qos)447f8ba50eaSHoratiu Vultur static int lan966x_port_qos_default_set(struct lan966x_port *port,
448f8ba50eaSHoratiu Vultur 					struct lan966x_port_qos *qos)
449f8ba50eaSHoratiu Vultur {
450f8ba50eaSHoratiu Vultur 	/* Set default prio and dp level */
451f8ba50eaSHoratiu Vultur 	lan_rmw(ANA_QOS_CFG_DP_DEFAULT_VAL_SET(0) |
452f8ba50eaSHoratiu Vultur 		ANA_QOS_CFG_QOS_DEFAULT_VAL_SET(qos->default_prio),
453f8ba50eaSHoratiu Vultur 		ANA_QOS_CFG_DP_DEFAULT_VAL |
454f8ba50eaSHoratiu Vultur 		ANA_QOS_CFG_QOS_DEFAULT_VAL,
455f8ba50eaSHoratiu Vultur 		port->lan966x, ANA_QOS_CFG(port->chip_port));
456f8ba50eaSHoratiu Vultur 
457f8ba50eaSHoratiu Vultur 	/* Set default pcp and dei for untagged frames */
458f8ba50eaSHoratiu Vultur 	lan_rmw(ANA_VLAN_CFG_VLAN_DEI_SET(0) |
459f8ba50eaSHoratiu Vultur 		ANA_VLAN_CFG_VLAN_PCP_SET(0),
460f8ba50eaSHoratiu Vultur 		ANA_VLAN_CFG_VLAN_DEI |
461f8ba50eaSHoratiu Vultur 		ANA_VLAN_CFG_VLAN_PCP,
462f8ba50eaSHoratiu Vultur 		port->lan966x, ANA_VLAN_CFG(port->chip_port));
463f8ba50eaSHoratiu Vultur 
464f8ba50eaSHoratiu Vultur 	return 0;
465f8ba50eaSHoratiu Vultur }
466f8ba50eaSHoratiu Vultur 
lan966x_port_qos_pcp_rewr_set(struct lan966x_port * port,struct lan966x_port_qos_pcp_rewr * qos)467363f98b9SHoratiu Vultur static void lan966x_port_qos_pcp_rewr_set(struct lan966x_port *port,
468363f98b9SHoratiu Vultur 					  struct lan966x_port_qos_pcp_rewr *qos)
469363f98b9SHoratiu Vultur {
470363f98b9SHoratiu Vultur 	u8 mode = LAN966X_PORT_REW_TAG_CTRL_CLASSIFIED;
471363f98b9SHoratiu Vultur 	u8 pcp, dei;
472363f98b9SHoratiu Vultur 
473363f98b9SHoratiu Vultur 	if (qos->enable)
474363f98b9SHoratiu Vultur 		mode = LAN966X_PORT_REW_TAG_CTRL_MAPPED;
475363f98b9SHoratiu Vultur 
476363f98b9SHoratiu Vultur 	/* Map the values only if it is enabled otherwise will be the classified
477363f98b9SHoratiu Vultur 	 * value
478363f98b9SHoratiu Vultur 	 */
479363f98b9SHoratiu Vultur 	lan_rmw(REW_TAG_CFG_TAG_PCP_CFG_SET(mode) |
480363f98b9SHoratiu Vultur 		REW_TAG_CFG_TAG_DEI_CFG_SET(mode),
481363f98b9SHoratiu Vultur 		REW_TAG_CFG_TAG_PCP_CFG |
482363f98b9SHoratiu Vultur 		REW_TAG_CFG_TAG_DEI_CFG,
483363f98b9SHoratiu Vultur 		port->lan966x, REW_TAG_CFG(port->chip_port));
484363f98b9SHoratiu Vultur 
485363f98b9SHoratiu Vultur 	/* Map each value to pcp and dei */
486363f98b9SHoratiu Vultur 	for (int i = 0; i < ARRAY_SIZE(qos->map); i++) {
487363f98b9SHoratiu Vultur 		pcp = qos->map[i];
488363f98b9SHoratiu Vultur 		if (pcp > LAN966X_PORT_QOS_PCP_COUNT)
489363f98b9SHoratiu Vultur 			dei = 1;
490363f98b9SHoratiu Vultur 		else
491363f98b9SHoratiu Vultur 			dei = 0;
492363f98b9SHoratiu Vultur 
493363f98b9SHoratiu Vultur 		lan_rmw(REW_PCP_DEI_CFG_DEI_QOS_VAL_SET(dei) |
494363f98b9SHoratiu Vultur 			REW_PCP_DEI_CFG_PCP_QOS_VAL_SET(pcp),
495363f98b9SHoratiu Vultur 			REW_PCP_DEI_CFG_DEI_QOS_VAL |
496363f98b9SHoratiu Vultur 			REW_PCP_DEI_CFG_PCP_QOS_VAL,
497363f98b9SHoratiu Vultur 			port->lan966x,
498363f98b9SHoratiu Vultur 			REW_PCP_DEI_CFG(port->chip_port,
499363f98b9SHoratiu Vultur 					i + dei * LAN966X_PORT_QOS_PCP_COUNT));
500363f98b9SHoratiu Vultur 	}
501363f98b9SHoratiu Vultur }
502363f98b9SHoratiu Vultur 
lan966x_port_qos_dscp_rewr_set(struct lan966x_port * port,struct lan966x_port_qos_dscp_rewr * qos)503d38ddd56SHoratiu Vultur static void lan966x_port_qos_dscp_rewr_set(struct lan966x_port *port,
504d38ddd56SHoratiu Vultur 					   struct lan966x_port_qos_dscp_rewr *qos)
505d38ddd56SHoratiu Vultur {
506d38ddd56SHoratiu Vultur 	u16 dscp;
507d38ddd56SHoratiu Vultur 	u8 mode;
508d38ddd56SHoratiu Vultur 
509d38ddd56SHoratiu Vultur 	if (qos->enable)
510d38ddd56SHoratiu Vultur 		mode = LAN966X_PORT_REW_DSCP_ANALIZER;
511d38ddd56SHoratiu Vultur 	else
512d38ddd56SHoratiu Vultur 		mode = LAN966X_PORT_REW_DSCP_FRAME;
513d38ddd56SHoratiu Vultur 
514d38ddd56SHoratiu Vultur 	/* Enable the rewrite otherwise will use the values from the frame */
515d38ddd56SHoratiu Vultur 	lan_rmw(REW_DSCP_CFG_DSCP_REWR_CFG_SET(mode),
516d38ddd56SHoratiu Vultur 		REW_DSCP_CFG_DSCP_REWR_CFG,
517d38ddd56SHoratiu Vultur 		port->lan966x, REW_DSCP_CFG(port->chip_port));
518d38ddd56SHoratiu Vultur 
519d38ddd56SHoratiu Vultur 	/* Map each classified Qos class and DP to classified DSCP value */
520d38ddd56SHoratiu Vultur 	for (int i = 0; i < ARRAY_SIZE(qos->map); i++) {
521d38ddd56SHoratiu Vultur 		dscp = qos->map[i];
522d38ddd56SHoratiu Vultur 
523d38ddd56SHoratiu Vultur 		lan_rmw(ANA_DSCP_REWR_CFG_DSCP_QOS_REWR_VAL_SET(dscp),
524d38ddd56SHoratiu Vultur 			ANA_DSCP_REWR_CFG_DSCP_QOS_REWR_VAL,
525d38ddd56SHoratiu Vultur 			port->lan966x, ANA_DSCP_REWR_CFG(i));
526d38ddd56SHoratiu Vultur 	}
527d38ddd56SHoratiu Vultur }
528d38ddd56SHoratiu Vultur 
lan966x_port_qos_dscp_rewr_mode_set(struct lan966x_port * port,int mode)529d38ddd56SHoratiu Vultur void lan966x_port_qos_dscp_rewr_mode_set(struct lan966x_port *port,
530d38ddd56SHoratiu Vultur 					 int mode)
531d38ddd56SHoratiu Vultur {
532d38ddd56SHoratiu Vultur 	lan_rmw(ANA_QOS_CFG_DSCP_REWR_CFG_SET(mode),
533d38ddd56SHoratiu Vultur 		ANA_QOS_CFG_DSCP_REWR_CFG,
534d38ddd56SHoratiu Vultur 		port->lan966x, ANA_QOS_CFG(port->chip_port));
535d38ddd56SHoratiu Vultur }
536d38ddd56SHoratiu Vultur 
lan966x_port_qos_set(struct lan966x_port * port,struct lan966x_port_qos * qos)537a83e4630SHoratiu Vultur void lan966x_port_qos_set(struct lan966x_port *port,
538a83e4630SHoratiu Vultur 			  struct lan966x_port_qos *qos)
539a83e4630SHoratiu Vultur {
540a83e4630SHoratiu Vultur 	lan966x_port_qos_pcp_set(port, &qos->pcp);
5410c88d981SHoratiu Vultur 	lan966x_port_qos_dscp_set(port, &qos->dscp);
542f8ba50eaSHoratiu Vultur 	lan966x_port_qos_default_set(port, qos);
543363f98b9SHoratiu Vultur 	lan966x_port_qos_pcp_rewr_set(port, &qos->pcp_rewr);
544d38ddd56SHoratiu Vultur 	lan966x_port_qos_dscp_rewr_set(port, &qos->dscp_rewr);
545a83e4630SHoratiu Vultur }
546a83e4630SHoratiu Vultur 
lan966x_port_init(struct lan966x_port * port)547d28d6d2eSHoratiu Vultur void lan966x_port_init(struct lan966x_port *port)
548d28d6d2eSHoratiu Vultur {
549d28d6d2eSHoratiu Vultur 	struct lan966x_port_config *config = &port->config;
550d28d6d2eSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
551d28d6d2eSHoratiu Vultur 
552d28d6d2eSHoratiu Vultur 	lan_rmw(ANA_PORT_CFG_LEARN_ENA_SET(0),
553d28d6d2eSHoratiu Vultur 		ANA_PORT_CFG_LEARN_ENA,
554d28d6d2eSHoratiu Vultur 		lan966x, ANA_PORT_CFG(port->chip_port));
555d28d6d2eSHoratiu Vultur 
556d28d6d2eSHoratiu Vultur 	lan966x_port_config_down(port);
557d28d6d2eSHoratiu Vultur 
558c8349639SHoratiu Vultur 	if (lan966x->fdma)
559c8349639SHoratiu Vultur 		lan966x_fdma_netdev_init(lan966x, port->dev);
560c8349639SHoratiu Vultur 
561ac0167fbSMaxime Chevallier 	if (phy_interface_num_ports(config->portmode) != 4)
562d28d6d2eSHoratiu Vultur 		return;
563d28d6d2eSHoratiu Vultur 
564d28d6d2eSHoratiu Vultur 	lan_rmw(DEV_CLOCK_CFG_PCS_RX_RST_SET(0) |
565d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_TX_RST_SET(0) |
566d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_LINK_SPEED_SET(LAN966X_SPEED_1000),
567d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_RX_RST |
568d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_PCS_TX_RST |
569d28d6d2eSHoratiu Vultur 		DEV_CLOCK_CFG_LINK_SPEED,
570d28d6d2eSHoratiu Vultur 		lan966x, DEV_CLOCK_CFG(port->chip_port));
571d28d6d2eSHoratiu Vultur }
572