1 // SPDX-License-Identifier: GPL-2.0
2 /* Marvell CN10K RPM driver
4 * Copyright (C) 2020 Marvell.
9 #include "lmac_common.h"
11 static struct mac_ops rpm_mac_ops = {
15 .int_register = RPMX_CMRX_SW_INT,
16 .int_set_reg = RPMX_CMRX_SW_INT_ENA_W1S,
18 .int_ena_bit = BIT_ULL(0),
19 .lmac_fwi = RPM_LMAC_FWI,
20 .non_contiguous_serdes_lane = true,
23 .get_nr_lmacs = rpm_get_nr_lmacs,
24 .get_lmac_type = rpm_get_lmac_type,
25 .mac_lmac_intl_lbk = rpm_lmac_internal_loopback,
26 .mac_get_rx_stats = rpm_get_rx_stats,
27 .mac_get_tx_stats = rpm_get_tx_stats,
28 .mac_enadis_rx_pause_fwding = rpm_lmac_enadis_rx_pause_fwding,
29 .mac_get_pause_frm_status = rpm_lmac_get_pause_frm_status,
30 .mac_enadis_pause_frm = rpm_lmac_enadis_pause_frm,
31 .mac_pause_frm_config = rpm_lmac_pause_frm_config,
32 .mac_enadis_ptp_config = rpm_lmac_ptp_config,
33 .mac_rx_tx_enable = rpm_lmac_rx_tx_enable,
34 .mac_tx_enable = rpm_lmac_tx_enable,
35 .pfc_config = rpm_lmac_pfc_config,
36 .mac_get_pfc_frm_cfg = rpm_lmac_get_pfc_frm_cfg,
39 struct mac_ops *rpm_get_mac_ops(void)
44 static void rpm_write(rpm_t *rpm, u64 lmac, u64 offset, u64 val)
46 cgx_write(rpm, lmac, offset, val);
49 static u64 rpm_read(rpm_t *rpm, u64 lmac, u64 offset)
51 return cgx_read(rpm, lmac, offset);
54 int rpm_get_nr_lmacs(void *rpmd)
58 return hweight8(rpm_read(rpm, 0, CGXX_CMRX_RX_LMACS) & 0xFULL);
61 int rpm_lmac_tx_enable(void *rpmd, int lmac_id, bool enable)
66 if (!is_lmac_valid(rpm, lmac_id))
69 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
77 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
78 return !!(last & RPM_TX_EN);
81 int rpm_lmac_rx_tx_enable(void *rpmd, int lmac_id, bool enable)
86 if (!is_lmac_valid(rpm, lmac_id))
89 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
91 cfg |= RPM_RX_EN | RPM_TX_EN;
93 cfg &= ~(RPM_RX_EN | RPM_TX_EN);
94 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
98 void rpm_lmac_enadis_rx_pause_fwding(void *rpmd, int lmac_id, bool enable)
107 lmac = lmac_pdata(lmac_id, rpm);
111 /* Pause frames are not enabled just return */
112 if (!bitmap_weight(lmac->rx_fc_pfvf_bmap.bmap, lmac->rx_fc_pfvf_bmap.max))
116 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
117 cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE;
118 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
120 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
121 cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE;
122 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
126 int rpm_lmac_get_pause_frm_status(void *rpmd, int lmac_id,
127 u8 *tx_pause, u8 *rx_pause)
132 if (!is_lmac_valid(rpm, lmac_id))
135 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
136 if (!(cfg & RPMX_MTI_MAC100X_COMMAND_CONFIG_PFC_MODE)) {
137 *rx_pause = !(cfg & RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE);
138 *tx_pause = !(cfg & RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE);
144 static void rpm_cfg_pfc_quanta_thresh(rpm_t *rpm, int lmac_id,
145 unsigned long pfc_en,
148 u64 quanta_offset = 0, quanta_thresh = 0, cfg;
151 /* Set pause time and interval */
152 for_each_set_bit(i, &pfc_en, 16) {
156 quanta_offset = RPMX_MTI_MAC100X_CL01_PAUSE_QUANTA;
157 quanta_thresh = RPMX_MTI_MAC100X_CL01_QUANTA_THRESH;
161 quanta_offset = RPMX_MTI_MAC100X_CL23_PAUSE_QUANTA;
162 quanta_thresh = RPMX_MTI_MAC100X_CL23_QUANTA_THRESH;
166 quanta_offset = RPMX_MTI_MAC100X_CL45_PAUSE_QUANTA;
167 quanta_thresh = RPMX_MTI_MAC100X_CL45_QUANTA_THRESH;
171 quanta_offset = RPMX_MTI_MAC100X_CL67_PAUSE_QUANTA;
172 quanta_thresh = RPMX_MTI_MAC100X_CL67_QUANTA_THRESH;
176 quanta_offset = RPMX_MTI_MAC100X_CL89_PAUSE_QUANTA;
177 quanta_thresh = RPMX_MTI_MAC100X_CL89_QUANTA_THRESH;
181 quanta_offset = RPMX_MTI_MAC100X_CL1011_PAUSE_QUANTA;
182 quanta_thresh = RPMX_MTI_MAC100X_CL1011_QUANTA_THRESH;
186 quanta_offset = RPMX_MTI_MAC100X_CL1213_PAUSE_QUANTA;
187 quanta_thresh = RPMX_MTI_MAC100X_CL1213_QUANTA_THRESH;
191 quanta_offset = RPMX_MTI_MAC100X_CL1415_PAUSE_QUANTA;
192 quanta_thresh = RPMX_MTI_MAC100X_CL1415_QUANTA_THRESH;
196 if (!quanta_offset || !quanta_thresh)
199 shift = (i % 2) ? 1 : 0;
200 cfg = rpm_read(rpm, lmac_id, quanta_offset);
202 cfg |= ((u64)RPM_DEFAULT_PAUSE_TIME << shift * 16);
205 cfg &= ~GENMASK_ULL(15, 0);
207 cfg &= ~GENMASK_ULL(31, 16);
209 rpm_write(rpm, lmac_id, quanta_offset, cfg);
211 cfg = rpm_read(rpm, lmac_id, quanta_thresh);
213 cfg |= ((u64)(RPM_DEFAULT_PAUSE_TIME / 2) << shift * 16);
216 cfg &= ~GENMASK_ULL(15, 0);
218 cfg &= ~GENMASK_ULL(31, 16);
220 rpm_write(rpm, lmac_id, quanta_thresh, cfg);
224 int rpm_lmac_enadis_pause_frm(void *rpmd, int lmac_id, u8 tx_pause,
230 if (!is_lmac_valid(rpm, lmac_id))
233 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
234 cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE;
235 cfg |= rx_pause ? 0x0 : RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE;
236 cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE;
237 cfg |= rx_pause ? 0x0 : RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE;
238 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
240 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
241 cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE;
242 cfg |= tx_pause ? 0x0 : RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE;
243 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
245 cfg = rpm_read(rpm, 0, RPMX_CMR_RX_OVR_BP);
247 /* Configure CL0 Pause Quanta & threshold for 802.3X frames */
248 rpm_cfg_pfc_quanta_thresh(rpm, lmac_id, 1, true);
249 cfg &= ~RPMX_CMR_RX_OVR_BP_EN(lmac_id);
251 /* Disable all Pause Quanta & threshold values */
252 rpm_cfg_pfc_quanta_thresh(rpm, lmac_id, 0xffff, false);
253 cfg |= RPMX_CMR_RX_OVR_BP_EN(lmac_id);
254 cfg &= ~RPMX_CMR_RX_OVR_BP_BP(lmac_id);
256 rpm_write(rpm, 0, RPMX_CMR_RX_OVR_BP, cfg);
260 void rpm_lmac_pause_frm_config(void *rpmd, int lmac_id, bool enable)
265 /* ALL pause frames received are completely ignored */
266 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
267 cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE;
268 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
270 /* Disable forward pause to TX block */
271 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
272 cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE;
273 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
275 /* Disable pause frames transmission */
276 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
277 cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE;
278 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
281 int rpm_get_rx_stats(void *rpmd, int lmac_id, int idx, u64 *rx_stat)
286 if (!rpm || lmac_id >= rpm->lmac_count)
289 mutex_lock(&rpm->lock);
291 /* Update idx to point per lmac Rx statistics page */
292 idx += lmac_id * rpm->mac_ops->rx_stats_cnt;
294 /* Read lower 32 bits of counter */
295 val_lo = rpm_read(rpm, 0, RPMX_MTI_STAT_RX_STAT_PAGES_COUNTERX +
298 /* upon read of lower 32 bits, higher 32 bits are written
299 * to RPMX_MTI_STAT_DATA_HI_CDC
301 val_hi = rpm_read(rpm, 0, RPMX_MTI_STAT_DATA_HI_CDC);
303 *rx_stat = (val_hi << 32 | val_lo);
305 mutex_unlock(&rpm->lock);
309 int rpm_get_tx_stats(void *rpmd, int lmac_id, int idx, u64 *tx_stat)
314 if (!rpm || lmac_id >= rpm->lmac_count)
317 mutex_lock(&rpm->lock);
319 /* Update idx to point per lmac Tx statistics page */
320 idx += lmac_id * rpm->mac_ops->tx_stats_cnt;
322 val_lo = rpm_read(rpm, 0, RPMX_MTI_STAT_TX_STAT_PAGES_COUNTERX +
324 val_hi = rpm_read(rpm, 0, RPMX_MTI_STAT_DATA_HI_CDC);
326 *tx_stat = (val_hi << 32 | val_lo);
328 mutex_unlock(&rpm->lock);
332 u8 rpm_get_lmac_type(void *rpmd, int lmac_id)
338 req = FIELD_SET(CMDREG_ID, CGX_CMD_GET_LINK_STS, req);
339 err = cgx_fwi_cmd_generic(req, &resp, rpm, 0);
341 return FIELD_GET(RESP_LINKSTAT_LMAC_TYPE, resp);
345 int rpm_lmac_internal_loopback(void *rpmd, int lmac_id, bool enable)
351 if (!rpm || lmac_id >= rpm->lmac_count)
353 lmac_type = rpm->mac_ops->get_lmac_type(rpm, lmac_id);
355 if (lmac_type == LMAC_MODE_QSGMII || lmac_type == LMAC_MODE_SGMII) {
356 dev_err(&rpm->pdev->dev, "loopback not supported for LPC mode\n");
360 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_PCS100X_CONTROL1);
363 cfg |= RPMX_MTI_PCS_LBK;
365 cfg &= ~RPMX_MTI_PCS_LBK;
366 rpm_write(rpm, lmac_id, RPMX_MTI_PCS100X_CONTROL1, cfg);
371 void rpm_lmac_ptp_config(void *rpmd, int lmac_id, bool enable)
376 if (!is_lmac_valid(rpm, lmac_id))
379 cfg = rpm_read(rpm, lmac_id, RPMX_CMRX_CFG);
381 cfg |= RPMX_RX_TS_PREPEND;
383 cfg &= ~RPMX_RX_TS_PREPEND;
384 rpm_write(rpm, lmac_id, RPMX_CMRX_CFG, cfg);
387 int rpm_lmac_pfc_config(void *rpmd, int lmac_id, u8 tx_pause, u8 rx_pause, u16 pfc_en)
392 if (!is_lmac_valid(rpm, lmac_id))
395 /* reset PFC class quanta and threshold */
396 rpm_cfg_pfc_quanta_thresh(rpm, lmac_id, 0xffff, false);
398 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
401 cfg &= ~(RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE |
402 RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE |
403 RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_FWD);
405 cfg |= (RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE |
406 RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_IGNORE |
407 RPMX_MTI_MAC100X_COMMAND_CONFIG_PAUSE_FWD);
411 rpm_cfg_pfc_quanta_thresh(rpm, lmac_id, pfc_en, true);
412 cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE;
414 rpm_cfg_pfc_quanta_thresh(rpm, lmac_id, 0xfff, false);
415 cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE;
418 if (!rx_pause && !tx_pause)
419 cfg &= ~RPMX_MTI_MAC100X_COMMAND_CONFIG_PFC_MODE;
421 cfg |= RPMX_MTI_MAC100X_COMMAND_CONFIG_PFC_MODE;
423 rpm_write(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG, cfg);
425 cfg = rpm_read(rpm, lmac_id, RPMX_CMRX_PRT_CBFC_CTL);
426 cfg = FIELD_SET(RPM_PFC_CLASS_MASK, pfc_en, cfg);
427 rpm_write(rpm, lmac_id, RPMX_CMRX_PRT_CBFC_CTL, cfg);
432 int rpm_lmac_get_pfc_frm_cfg(void *rpmd, int lmac_id, u8 *tx_pause, u8 *rx_pause)
437 if (!is_lmac_valid(rpm, lmac_id))
440 cfg = rpm_read(rpm, lmac_id, RPMX_MTI_MAC100X_COMMAND_CONFIG);
441 if (cfg & RPMX_MTI_MAC100X_COMMAND_CONFIG_PFC_MODE) {
442 *rx_pause = !(cfg & RPMX_MTI_MAC100X_COMMAND_CONFIG_RX_P_DISABLE);
443 *tx_pause = !(cfg & RPMX_MTI_MAC100X_COMMAND_CONFIG_TX_P_DISABLE);