Lines Matching +full:rx +full:- +full:common +full:- +full:refclk +full:- +full:mode
1 // SPDX-License-Identifier: GPL-2.0
11 * Structure inspired from phy-mvebu-cp110-comphy.c written by Antoine Tenart.
40 * When accessing common PHY lane registers directly, we need to shift by 1,
41 * since the registers are 16-bit.
184 #define COMPHY_PHY_REG(lane, reg) (((1 - (lane)) * 0x28) + ((reg) & 0x3f))
228 enum phy_mode mode; member
235 .mode = _mode, \
274 enum phy_mode mode; member
285 /* Changes to 40M1G25 mode data required for running 40M3G125 init mode */
298 /* 40M1G25 mode init data */
301 /*-----------------------------------------------------------*/
392 priv->lane2_phy_indirect + COMPHY_LANE2_INDIR_ADDR); in comphy_set_indirect()
393 comphy_reg_set(priv->lane2_phy_indirect + COMPHY_LANE2_INDIR_DATA, in comphy_set_indirect()
400 if (lane->id == 2) { in comphy_lane_reg_set()
402 comphy_set_indirect(lane->priv, in comphy_lane_reg_set()
406 void __iomem *base = lane->id == 1 ? in comphy_lane_reg_set()
407 lane->priv->lane1_phy_regs : in comphy_lane_reg_set()
408 lane->priv->lane0_phy_regs; in comphy_lane_reg_set()
421 if (lane->id == 2) { in comphy_lane_reg_poll()
426 lane->priv->lane2_phy_indirect + in comphy_lane_reg_poll()
429 ret = readl_poll_timeout(lane->priv->lane2_phy_indirect + in comphy_lane_reg_poll()
434 void __iomem *base = lane->id == 1 ? in comphy_lane_reg_poll()
435 lane->priv->lane1_phy_regs : in comphy_lane_reg_poll()
436 lane->priv->lane0_phy_regs; in comphy_lane_reg_poll()
450 comphy_reg_set(lane->priv->comphy_regs + COMPHY_PHY_REG(lane->id, reg), in comphy_periph_reg_set()
460 return readl_poll_timeout(lane->priv->comphy_regs + in comphy_periph_reg_poll()
461 COMPHY_PHY_REG(lane->id, reg), in comphy_periph_reg_poll()
473 switch (lane->mode) { in mvebu_a3700_comphy_set_phy_selector()
476 if (lane->id == 2) in mvebu_a3700_comphy_set_phy_selector()
483 if (lane->id == 0) in mvebu_a3700_comphy_set_phy_selector()
485 else if (lane->id == 1) in mvebu_a3700_comphy_set_phy_selector()
492 if (lane->id == 2) in mvebu_a3700_comphy_set_phy_selector()
494 else if (lane->id == 0) in mvebu_a3700_comphy_set_phy_selector()
502 if (lane->id == 1) in mvebu_a3700_comphy_set_phy_selector()
512 spin_lock_irqsave(&lane->priv->lock, flags); in mvebu_a3700_comphy_set_phy_selector()
514 old = readl(lane->priv->comphy_regs + COMPHY_SELECTOR_PHY_REG); in mvebu_a3700_comphy_set_phy_selector()
516 writel(new, lane->priv->comphy_regs + COMPHY_SELECTOR_PHY_REG); in mvebu_a3700_comphy_set_phy_selector()
518 spin_unlock_irqrestore(&lane->priv->lock, flags); in mvebu_a3700_comphy_set_phy_selector()
520 dev_dbg(lane->dev, in mvebu_a3700_comphy_set_phy_selector()
521 "COMPHY[%d] mode[%d] changed PHY selector 0x%08x -> 0x%08x\n", in mvebu_a3700_comphy_set_phy_selector()
522 lane->id, lane->mode, old, new); in mvebu_a3700_comphy_set_phy_selector()
526 dev_err(lane->dev, "COMPHY[%d] mode[%d] is invalid\n", lane->id, in mvebu_a3700_comphy_set_phy_selector()
527 lane->mode); in mvebu_a3700_comphy_set_phy_selector()
528 return -EINVAL; in mvebu_a3700_comphy_set_phy_selector()
542 /* Clear phy isolation mode to make it work in normal mode */ in mvebu_a3700_comphy_sata_power_on()
548 if (lane->invert_tx) in mvebu_a3700_comphy_sata_power_on()
550 if (lane->invert_rx) in mvebu_a3700_comphy_sata_power_on()
555 /* 1. Select 40-bit data width */ in mvebu_a3700_comphy_sata_power_on()
559 /* 2. Select reference clock(25M) and PHY mode (SATA) */ in mvebu_a3700_comphy_sata_power_on()
560 if (lane->priv->xtal_is_40m) in mvebu_a3700_comphy_sata_power_on()
574 comphy_set_indirect(lane->priv, COMPHY_RESERVED_REG, in mvebu_a3700_comphy_sata_power_on()
577 /* 5. Set vendor-specific configuration (It is done in sata driver) */ in mvebu_a3700_comphy_sata_power_on()
578 /* XXX: in U-Boot below sequence was executed in this place, in Linux in mvebu_a3700_comphy_sata_power_on()
579 * not. Now it is done only in U-Boot before this comphy in mvebu_a3700_comphy_sata_power_on()
580 * initialization - tests shows that it works ok, but in case of any in mvebu_a3700_comphy_sata_power_on()
594 dev_err(lane->dev, "Failed to lock SATA PLL\n"); in mvebu_a3700_comphy_sata_power_on()
659 switch (lane->submode) { in mvebu_a3700_comphy_ethernet_power_on()
666 /* 2500Base-X, SerDes speed 3.125G */ in mvebu_a3700_comphy_ethernet_power_on()
671 dev_err(lane->dev, in mvebu_a3700_comphy_ethernet_power_on()
673 lane->submode, lane->id); in mvebu_a3700_comphy_ethernet_power_on()
674 return -EINVAL; in mvebu_a3700_comphy_ethernet_power_on()
692 * 8. Set COMPHY register REFCLK_SEL to select the correct REFCLK in mvebu_a3700_comphy_ethernet_power_on()
703 if (lane->priv->xtal_is_40m) in mvebu_a3700_comphy_ethernet_power_on()
728 * 12. As long as DFE function needs to be enabled in any mode, in mvebu_a3700_comphy_ethernet_power_on()
739 * the related GEN table during real chip bring-up. We only required to in mvebu_a3700_comphy_ethernet_power_on()
744 dev_dbg(lane->dev, "Running C-DPI phy init %s mode\n", in mvebu_a3700_comphy_ethernet_power_on()
745 lane->submode == PHY_INTERFACE_MODE_2500BASEX ? "2G5" : "1G"); in mvebu_a3700_comphy_ethernet_power_on()
746 if (lane->priv->xtal_is_40m) in mvebu_a3700_comphy_ethernet_power_on()
748 lane->submode != PHY_INTERFACE_MODE_2500BASEX); in mvebu_a3700_comphy_ethernet_power_on()
754 if (lane->invert_tx) in mvebu_a3700_comphy_ethernet_power_on()
756 if (lane->invert_rx) in mvebu_a3700_comphy_ethernet_power_on()
780 dev_err(lane->dev, "Failed to lock PLL for SERDES PHY %d\n", in mvebu_a3700_comphy_ethernet_power_on()
781 lane->id); in mvebu_a3700_comphy_ethernet_power_on()
792 * start RX initialization. PIN_RX_INIT_DONE will be cleared to 0 by the in mvebu_a3700_comphy_ethernet_power_on()
793 * PHY After RX initialization is done, PIN_RX_INIT_DONE will be set to in mvebu_a3700_comphy_ethernet_power_on()
795 * refer to RX initialization part for details. in mvebu_a3700_comphy_ethernet_power_on()
805 dev_err(lane->dev, "Failed to lock PLL for SERDES PHY %d\n", in mvebu_a3700_comphy_ethernet_power_on()
806 lane->id); in mvebu_a3700_comphy_ethernet_power_on()
814 dev_err(lane->dev, "Failed to init RX of SERDES PHY %d\n", in mvebu_a3700_comphy_ethernet_power_on()
815 lane->id); in mvebu_a3700_comphy_ethernet_power_on()
840 * 1. Set PRD_TXDEEMPH (3.5db de-emph) in mvebu_a3700_comphy_usb3_power_on()
848 * 2. Set BIT0: enable transmitter in high impedance mode in mvebu_a3700_comphy_usb3_power_on()
850 * Set BIT6: Tx detect Rx at HiZ mode in mvebu_a3700_comphy_usb3_power_on()
851 * Unset BIT15: set to 0 to set USB3 De-emphasize level to -3.5db in mvebu_a3700_comphy_usb3_power_on()
873 * 5. Set Lane-to-Lane Bundle Clock Sampling Period = per PCLK cycles in mvebu_a3700_comphy_usb3_power_on()
874 * set Mode Clock Source = PCLK is generated from REFCLK in mvebu_a3700_comphy_usb3_power_on()
889 * set G3 TX and RX Register Master Current Select in mvebu_a3700_comphy_usb3_power_on()
898 * accordingly Change RX wait in mvebu_a3700_comphy_usb3_power_on()
900 if (lane->priv->xtal_is_40m) { in mvebu_a3700_comphy_usb3_power_on()
932 * 11. Set 20-bit data width in mvebu_a3700_comphy_usb3_power_on()
948 if (lane->invert_tx) in mvebu_a3700_comphy_usb3_power_on()
950 if (lane->invert_rx) in mvebu_a3700_comphy_usb3_power_on()
980 dev_err(lane->dev, "Failed to lock USB3 PLL\n"); in mvebu_a3700_comphy_usb3_power_on()
1004 /* 3. Force to use reg setting for PCIe mode */ in mvebu_a3700_comphy_pcie_power_on()
1008 /* 4. Change RX wait */ in mvebu_a3700_comphy_pcie_power_on()
1025 * PCI-E driver in mvebu_a3700_comphy_pcie_power_on()
1033 if (lane->priv->xtal_is_40m) in mvebu_a3700_comphy_pcie_power_on()
1050 if (lane->invert_tx) in mvebu_a3700_comphy_pcie_power_on()
1052 if (lane->invert_rx) in mvebu_a3700_comphy_pcie_power_on()
1068 dev_err(lane->dev, "Failed to lock PCIE PLL\n"); in mvebu_a3700_comphy_pcie_power_on()
1076 /* Set phy isolation mode */ in mvebu_a3700_comphy_sata_power_off()
1080 /* Power off PLL, Tx, Rx */ in mvebu_a3700_comphy_sata_power_off()
1099 /* Power off PLL, Tx, Rx */ in mvebu_a3700_comphy_pcie_power_off()
1113 enum phy_mode mode, in mvebu_a3700_comphy_check_mode() argument
1119 if (mode == PHY_MODE_INVALID) in mvebu_a3700_comphy_check_mode()
1124 mvebu_a3700_comphy_modes[i].mode == mode && in mvebu_a3700_comphy_check_mode()
1135 static int mvebu_a3700_comphy_set_mode(struct phy *phy, enum phy_mode mode, in mvebu_a3700_comphy_set_mode() argument
1140 if (!mvebu_a3700_comphy_check_mode(lane->id, mode, submode)) { in mvebu_a3700_comphy_set_mode()
1141 dev_err(lane->dev, "invalid COMPHY mode\n"); in mvebu_a3700_comphy_set_mode()
1142 return -EINVAL; in mvebu_a3700_comphy_set_mode()
1145 /* Mode cannot be changed while the PHY is powered on */ in mvebu_a3700_comphy_set_mode()
1146 if (phy->power_count && in mvebu_a3700_comphy_set_mode()
1147 (lane->mode != mode || lane->submode != submode)) in mvebu_a3700_comphy_set_mode()
1148 return -EBUSY; in mvebu_a3700_comphy_set_mode()
1150 /* Just remember the mode, ->power_on() will do the real setup */ in mvebu_a3700_comphy_set_mode()
1151 lane->mode = mode; in mvebu_a3700_comphy_set_mode()
1152 lane->submode = submode; in mvebu_a3700_comphy_set_mode()
1161 if (!mvebu_a3700_comphy_check_mode(lane->id, lane->mode, in mvebu_a3700_comphy_power_on()
1162 lane->submode)) { in mvebu_a3700_comphy_power_on()
1163 dev_err(lane->dev, "invalid COMPHY mode\n"); in mvebu_a3700_comphy_power_on()
1164 return -EINVAL; in mvebu_a3700_comphy_power_on()
1167 switch (lane->mode) { in mvebu_a3700_comphy_power_on()
1169 dev_dbg(lane->dev, "set lane %d to USB3 host mode\n", lane->id); in mvebu_a3700_comphy_power_on()
1172 dev_dbg(lane->dev, "set lane %d to SATA mode\n", lane->id); in mvebu_a3700_comphy_power_on()
1175 dev_dbg(lane->dev, "set lane %d to Ethernet mode\n", lane->id); in mvebu_a3700_comphy_power_on()
1178 dev_dbg(lane->dev, "set lane %d to PCIe mode\n", lane->id); in mvebu_a3700_comphy_power_on()
1181 dev_err(lane->dev, "unsupported PHY mode (%d)\n", lane->mode); in mvebu_a3700_comphy_power_on()
1182 return -EOPNOTSUPP; in mvebu_a3700_comphy_power_on()
1190 switch (lane->id) { in mvebu_a3700_comphy_power_off()
1204 dev_err(lane->dev, "invalid COMPHY mode\n"); in mvebu_a3700_comphy_power_off()
1205 return -EINVAL; in mvebu_a3700_comphy_power_off()
1229 port = args->args[0]; in mvebu_a3700_comphy_xlate()
1230 if (port != 0 && (port != 1 || lane->id != 0)) { in mvebu_a3700_comphy_xlate()
1231 dev_err(lane->dev, "invalid port number %u\n", port); in mvebu_a3700_comphy_xlate()
1232 return ERR_PTR(-EINVAL); in mvebu_a3700_comphy_xlate()
1235 lane->invert_tx = args->args[1] & BIT(0); in mvebu_a3700_comphy_xlate()
1236 lane->invert_rx = args->args[1] & BIT(1); in mvebu_a3700_comphy_xlate()
1250 priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL); in mvebu_a3700_comphy_probe()
1252 return -ENOMEM; in mvebu_a3700_comphy_probe()
1254 spin_lock_init(&priv->lock); in mvebu_a3700_comphy_probe()
1257 priv->comphy_regs = devm_ioremap_resource(&pdev->dev, res); in mvebu_a3700_comphy_probe()
1258 if (IS_ERR(priv->comphy_regs)) in mvebu_a3700_comphy_probe()
1259 return PTR_ERR(priv->comphy_regs); in mvebu_a3700_comphy_probe()
1263 priv->lane1_phy_regs = devm_ioremap_resource(&pdev->dev, res); in mvebu_a3700_comphy_probe()
1264 if (IS_ERR(priv->lane1_phy_regs)) in mvebu_a3700_comphy_probe()
1265 return PTR_ERR(priv->lane1_phy_regs); in mvebu_a3700_comphy_probe()
1269 priv->lane0_phy_regs = devm_ioremap_resource(&pdev->dev, res); in mvebu_a3700_comphy_probe()
1270 if (IS_ERR(priv->lane0_phy_regs)) in mvebu_a3700_comphy_probe()
1271 return PTR_ERR(priv->lane0_phy_regs); in mvebu_a3700_comphy_probe()
1275 priv->lane2_phy_indirect = devm_ioremap_resource(&pdev->dev, res); in mvebu_a3700_comphy_probe()
1276 if (IS_ERR(priv->lane2_phy_indirect)) in mvebu_a3700_comphy_probe()
1277 return PTR_ERR(priv->lane2_phy_indirect); in mvebu_a3700_comphy_probe()
1284 clk = clk_get(&pdev->dev, "xtal"); in mvebu_a3700_comphy_probe()
1286 if (PTR_ERR(clk) == -EPROBE_DEFER) in mvebu_a3700_comphy_probe()
1287 return -EPROBE_DEFER; in mvebu_a3700_comphy_probe()
1288 dev_warn(&pdev->dev, "missing 'xtal' clk (%ld)\n", in mvebu_a3700_comphy_probe()
1293 dev_warn(&pdev->dev, "enabling xtal clk failed (%d)\n", in mvebu_a3700_comphy_probe()
1297 priv->xtal_is_40m = true; in mvebu_a3700_comphy_probe()
1303 dev_set_drvdata(&pdev->dev, priv); in mvebu_a3700_comphy_probe()
1305 for_each_available_child_of_node(pdev->dev.of_node, child) { in mvebu_a3700_comphy_probe()
1313 dev_err(&pdev->dev, "missing 'reg' property (%d)\n", in mvebu_a3700_comphy_probe()
1319 dev_err(&pdev->dev, "invalid 'reg' property\n"); in mvebu_a3700_comphy_probe()
1323 lane = devm_kzalloc(&pdev->dev, sizeof(*lane), GFP_KERNEL); in mvebu_a3700_comphy_probe()
1326 return -ENOMEM; in mvebu_a3700_comphy_probe()
1329 phy = devm_phy_create(&pdev->dev, child, in mvebu_a3700_comphy_probe()
1336 lane->priv = priv; in mvebu_a3700_comphy_probe()
1337 lane->dev = &pdev->dev; in mvebu_a3700_comphy_probe()
1338 lane->mode = PHY_MODE_INVALID; in mvebu_a3700_comphy_probe()
1339 lane->submode = PHY_INTERFACE_MODE_NA; in mvebu_a3700_comphy_probe()
1340 lane->id = lane_id; in mvebu_a3700_comphy_probe()
1341 lane->invert_tx = false; in mvebu_a3700_comphy_probe()
1342 lane->invert_rx = false; in mvebu_a3700_comphy_probe()
1352 provider = devm_of_phy_provider_register(&pdev->dev, in mvebu_a3700_comphy_probe()
1359 { .compatible = "marvell,comphy-a3700" },
1367 .name = "mvebu-a3700-comphy",
1376 MODULE_DESCRIPTION("Common PHY driver for A3700");