/* * Texas Instruments Keystone SerDes driver * Authors: WingMan Kwok * * This is the SerDes Phy driver for Keystone devices. This is * required to support PCIe RC functionality based on designware * PCIe hardware, gbe and 10gbe found on these devices. * * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the * distribution. * * Neither the name of Texas Instruments Incorporated nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include #include #include #include #define KSERDES_SS_OFFSET 0x1fc0 #define MOD_VER_REG (KSERDES_SS_OFFSET + 0x00) #define MEM_ADR_REG (KSERDES_SS_OFFSET + 0x04) #define MEM_DAT_REG (KSERDES_SS_OFFSET + 0x08) #define MEM_DATINC_REG (KSERDES_SS_OFFSET + 0x0c) #define CPU_CTRL_REG (KSERDES_SS_OFFSET + 0x10) #define LANE_CTRL_STS_REG(x) (KSERDES_SS_OFFSET + 0x20 + (x * 0x04)) #define LINK_LOSS_WAIT_REG (KSERDES_SS_OFFSET + 0x30) #define PLL_CTRL_REG (KSERDES_SS_OFFSET + 0x34) #define CMU0_SS_OFFSET 0x0000 #define CMU0_REG(x) (CMU0_SS_OFFSET + x) #define LANE0_SS_OFFSET 0x0200 #define LANEX_SS_OFFSET(x) (LANE0_SS_OFFSET * (x + 1)) #define LANEX_REG(x, y) (LANEX_SS_OFFSET(x) + y) #define CML_SS_OFFSET 0x0a00 #define CML_REG(x) (CML_SS_OFFSET + x) #define CMU1_SS_OFFSET 0x0c00 #define CMU1_REG(x) (CMU1_SS_OFFSET + x) #define PCSR_OFFSET(x) (x * 0x80) #define PCSR_TX_CTL(x) (PCSR_OFFSET(x) + 0x00) #define PCSR_TX_STATUS(x) (PCSR_OFFSET(x) + 0x04) #define PCSR_RX_CTL(x) (PCSR_OFFSET(x) + 0x08) #define PCSR_RX_STATUS(x) (PCSR_OFFSET(x) + 0x0C) #define XGE_CTRL_OFFSET 0x0c #define PCIE_PL_GEN2_OFFSET 0x180c #define reg_rmw(addr, value, mask) \ writel(((readl(addr) & (~(mask))) | (value & (mask))), (addr)) #define FINSR(base, offset, msb, lsb, val) \ reg_rmw((base) + (offset), ((val) << (lsb)), GENMASK((msb), (lsb))) #define FEXTR(val, msb, lsb) \ (((val) >> (lsb)) & ((1 << ((msb) - (lsb) + 1)) - 1)) #define MOD_VER(serdes) \ ((kserdes_readl(serdes, MOD_VER_REG) >> 16) & 0xffff) #define PHY_A(serdes) (MOD_VER(serdes) != 0x4eba) #define FOUR_LANE(serdes) \ ((MOD_VER(serdes) == 0x4eb9) || (MOD_VER(serdes) == 0x4ebd)) #define MAX_COMPARATORS 5 #define OFFSET_SAMPLES 100 #define for_each_comparator(i) \ for (i = 1; i < MAX_COMPARATORS; i++) #define CPU_EN BIT(31) #define CPU_GO BIT(30) #define POR_EN BIT(29) #define CPUREG_EN BIT(28) #define AUTONEG_CTL BIT(27) #define DATASPLIT BIT(26) #define LNKTRN_SIG_DET BIT(8) #define PLL_ENABLE_1P25G 0xe0000000 #define LANE_CTRL_1P25G 0xf800f8c0 #define XFM_FLUSH_CMD 0x00009c9c #define ANEG_LINK_CTL_10GKR_MASK GENMASK(21, 20) #define ANEG_LINK_CTL_1GKX_MASK GENMASK(17, 16) #define ANEG_LINK_CTL_1G10G_MASK \ (ANEG_LINK_CTL_10GKR_MASK | ANEG_LINK_CTL_1GKX_MASK) #define ANEG_1G_10G_OPT_MASK GENMASK(7, 5) #define SERDES_REG_INDEX 0 #define KSERDES_XFW_MEM_SIZE SZ_64K #define KSERDES_XFW_CONFIG_MEM_SIZE SZ_64 #define KSERDES_XFW_NUM_PARAMS 5 #define KSERDES_XFW_CONFIG_START_ADDR \ (KSERDES_XFW_MEM_SIZE - KSERDES_XFW_CONFIG_MEM_SIZE) #define KSERDES_XFW_PARAM_START_ADDR \ (KSERDES_XFW_MEM_SIZE - (KSERDES_XFW_NUM_PARAMS * 4)) /* * All firmware file names end up here. List the firmware file names below. * Newest first. Search starts from the 0-th array entry until a firmware * file is found. */ static const char * const ks2_gbe_serdes_firmwares[] = {"ks2_gbe_serdes.bin"}; static const char * const ks2_xgbe_serdes_firmwares[] = {"ks2_xgbe_serdes.bin"}; static const char * const ks2_pcie_serdes_firmwares[] = {"ks2_pcie_serdes.bin"}; enum kserdes_link_rate { KSERDES_LINK_RATE_1P25G = 1250000, KSERDES_LINK_RATE_3P125G = 3125000, KSERDES_LINK_RATE_4P9152G = 4915200, KSERDES_LINK_RATE_5G = 5000000, KSERDES_LINK_RATE_6P144G = 6144000, KSERDES_LINK_RATE_6P25G = 6250000, KSERDES_LINK_RATE_7P3728G = 7372800, KSERDES_LINK_RATE_9P8304G = 9830400, KSERDES_LINK_RATE_10G = 10000000, KSERDES_LINK_RATE_10P3125G = 10312500, KSERDES_LINK_RATE_12P5G = 12500000, }; enum kserdes_lane_ctrl_rate { KSERDES_FULL_RATE, KSERDES_HALF_RATE, KSERDES_QUARTER_RATE, }; enum kserdes_phy_type { KSERDES_PHY_SGMII, KSERDES_PHY_XGE, KSERDES_PHY_PCIE, KSERDES_PHY_HYPERLINK, }; struct kserdes_tx_coeff { u32 c1; u32 c2; u32 cm; u32 att; u32 vreg; }; struct kserdes_equalizer { u32 att; u32 boost; }; struct kserdes_lane_config { bool enable; u32 ctrl_rate; struct kserdes_tx_coeff tx_coeff; struct kserdes_equalizer rx_start; struct kserdes_equalizer rx_force; bool loopback; }; #define KSERDES_MAX_LANES 4 struct kserdes_fw_config { bool on; u32 rate; u32 link_loss_wait; u32 lane_seeds; u32 fast_train; u32 active_lane; u32 c1, c2, cm, attn, boost, dlpf, rxcal; u32 lane_config[KSERDES_MAX_LANES]; }; struct kserdes_config { struct device *dev; enum kserdes_phy_type phy_type; u32 lanes; void __iomem *regs; struct regmap *peripheral_regmap; struct regmap *pcsr_regmap; const char *init_fw; struct serdes_cfg *init_cfg; int init_cfg_len; enum kserdes_link_rate link_rate; bool rx_force_enable; struct kserdes_lane_config lane[KSERDES_MAX_LANES]; bool firmware; struct kserdes_fw_config fw; }; struct kserdes_dev { struct device *dev; struct phy *phy; struct kserdes_config sc; }; struct kserdes_cmp_tap_ofs { u32 cmp; u32 tap1; u32 tap2; u32 tap3; u32 tap4; u32 tap5; }; struct kserdes_lane_offsets { struct kserdes_cmp_tap_ofs ct_ofs[MAX_COMPARATORS]; }; struct kserdes_offsets { struct kserdes_lane_offsets lane_ofs[KSERDES_MAX_LANES]; }; struct serdes_cfg { u32 ofs; u32 msb; u32 lsb; u32 val; }; static struct platform_device_id kserdes_devtype[] = { { .name = "kserdes-gbe", .driver_data = KSERDES_PHY_SGMII, }, { .name = "kserdes-xgbe", .driver_data = KSERDES_PHY_XGE, }, { .name = "kserdes-pcie", .driver_data = KSERDES_PHY_PCIE, } }; static inline int next_enable_lane(struct kserdes_config *sc, int i) { int j = i; while (++j < sc->lanes) { if (sc->lane[j].enable) return j; } return j; } #define for_each_lane(sc, i) \ for (i = 0; i < (sc)->lanes; i++) #define for_each_enable_lane(sc, i) \ for (i = -1; i = next_enable_lane(sc, i), i < sc->lanes; ) static inline u32 kserdes_readl(void __iomem *base, u32 offset) { return readl(base + offset); } static inline void kserdes_writel(void __iomem *base, u32 offset, u32 value) { writel(value, base + offset); } static void kserdes_do_config(void __iomem *base, struct serdes_cfg *cfg, u32 size) { u32 i; for (i = 0; i < size; i++) FINSR(base, cfg[i].ofs, cfg[i].msb, cfg[i].lsb, cfg[i].val); } static int kserdes_load_init_fw(struct kserdes_config *sc, const char * const *a_firmwares, int n_firmwares) { const struct firmware *fw; bool found = false; int ret, i; for (i = 0; i < n_firmwares; i++) { if (a_firmwares[i]) { ret = request_firmware(&fw, a_firmwares[i], sc->dev); if (!ret) { found = true; break; } } } if (!found) { dev_err(sc->dev, "can't get any serdes init fw"); return -ENODEV; } sc->init_fw = a_firmwares[i]; sc->init_cfg = devm_kzalloc(sc->dev, fw->size, GFP_KERNEL); memcpy((void *)sc->init_cfg, fw->data, fw->size); sc->init_cfg_len = fw->size; release_firmware(fw); kserdes_do_config(sc->regs, sc->init_cfg, sc->init_cfg_len / sizeof(struct serdes_cfg)); return 0; } static inline u32 _kserdes_read_tbus_val(void __iomem *sregs) { u32 tmp; if (PHY_A(sregs)) { tmp = ((kserdes_readl(sregs, CMU0_REG(0xec))) >> 24) & 0x0ff; tmp |= ((kserdes_readl(sregs, CMU0_REG(0xfc))) >> 16) & 0xf00; } else { tmp = ((kserdes_readl(sregs, CMU0_REG(0xf8))) >> 16) & 0xfff; } return tmp; } static void _kserdes_write_tbus_addr(void __iomem *sregs, int select, int ofs) { if (select && !FOUR_LANE(sregs)) ++select; if (PHY_A(sregs)) FINSR(sregs, CMU0_REG(0x8), 31, 24, ((select << 5) + ofs)); else FINSR(sregs, CMU0_REG(0xfc), 26, 16, ((select << 8) + ofs)); } static u32 _kserdes_read_select_tbus(void __iomem *sregs, int select, int ofs) { _kserdes_write_tbus_addr(sregs, select, ofs); return _kserdes_read_tbus_val(sregs); } static inline void kserdes_highspeed_cfg1(struct kserdes_config *sc) { FINSR(sc->regs, CML_REG(0xbc), 28, 24, 0x1e); } static inline void kserdes_set_tx_idle(struct kserdes_config *sc, u32 lane) { if (sc->phy_type != KSERDES_PHY_XGE) FINSR(sc->regs, LANEX_REG(lane, 0xb8), 17, 16, 3); FINSR(sc->regs, LANE_CTRL_STS_REG(lane), 25, 24, 3); FINSR(sc->regs, LANEX_REG(lane, 0x28), 21, 20, 0); } static inline void kserdes_clr_tx_idle(struct kserdes_config *sc, u32 lane) { if (sc->phy_type != KSERDES_PHY_XGE) FINSR(sc->regs, LANEX_REG(lane, 0xb8), 17, 16, 0); FINSR(sc->regs, LANE_CTRL_STS_REG(lane), 25, 24, 0); FINSR(sc->regs, LANEX_REG(lane, 0x28), 21, 20, 0); } static inline void kserdes_cdfe_enable(struct kserdes_config *sc, u32 lane) { FINSR(sc->regs, LANEX_REG(lane, 0x94), 24, 24, 0x1); } static inline void kserdes_cdfe_force_calibration_enable( struct kserdes_config *sc, u32 lane) { FINSR(sc->regs, LANEX_REG(lane, 0x98), 0, 0, 0x1); } static void kserdes_phyb_cfg(struct kserdes_config *sc) { int lane; for_each_enable_lane(sc, lane) kserdes_cdfe_enable(sc, lane); FINSR(sc->regs, CML_REG(0x108), 23, 16, 0x04); FINSR(sc->regs, CML_REG(0xbc), 28, 24, 0x0); for_each_lane(sc, lane) kserdes_cdfe_force_calibration_enable(sc, lane); } static inline void kserdes_set_lane_starts(struct kserdes_config *sc, u32 lane) { FINSR(sc->regs, LANEX_REG(lane, 0x8c), 11, 8, sc->lane[lane].rx_start.att); FINSR(sc->regs, LANEX_REG(lane, 0x8c), 15, 12, sc->lane[lane].rx_start.boost); } static void kserdes_highspeed_cfg(struct kserdes_config *sc) { int lane; if (sc->phy_type == KSERDES_PHY_XGE) { kserdes_phyb_cfg(sc); } for_each_enable_lane(sc, lane) kserdes_set_lane_starts(sc, lane); } static void kserdes_set_lane_overrides(struct kserdes_config *sc, u32 lane) { u32 val_0, val_1, val; val_0 = _kserdes_read_select_tbus(sc->regs, lane + 1, 0); val_1 = _kserdes_read_select_tbus(sc->regs, lane + 1, 1); val = 0; val |= ((val_1 >> 9) & 0x3) << 1; val |= (val_0 & 0x3) << 3; val |= ((val_0 >> 2) & 0x1ff) << 5; val |= (1 << 14); val &= ~0x60; FINSR(sc->regs, LANEX_REG(lane, 0x028), 29, 15, val); } static inline void kserdes_assert_reset(struct kserdes_config *sc) { int lane; for_each_enable_lane(sc, lane) kserdes_set_lane_overrides(sc, lane); } static inline void kserdes_config_c1_c2_cm(struct kserdes_config *sc, u32 lane) { u32 c1, c2, cm; c1 = sc->lane[lane].tx_coeff.c1; c2 = sc->lane[lane].tx_coeff.c2; cm = sc->lane[lane].tx_coeff.cm; if (sc->phy_type == KSERDES_PHY_XGE) { FINSR(sc->regs, LANEX_REG(lane, 0x8), 11, 8, (cm & 0xf)); FINSR(sc->regs, LANEX_REG(lane, 0x8), 4, 0, (c1 & 0x1f)); FINSR(sc->regs, LANEX_REG(lane, 0x8), 7, 5, (c2 & 0x7)); FINSR(sc->regs, LANEX_REG(lane, 0x4), 18, 18, ((c2 >> 3) & 0x1)); } else { FINSR(sc->regs, LANEX_REG(lane, 0x8), 15, 12, (cm & 0xf)); FINSR(sc->regs, LANEX_REG(lane, 0x8), 4, 0, (c1 & 0x1f)); FINSR(sc->regs, LANEX_REG(lane, 0x8), 11, 8, (c2 & 0xf)); } } static inline void kserdes_config_att_boost(struct kserdes_config *sc, u32 lane) { u32 att, boost; att = sc->lane[lane].rx_force.att; boost = sc->lane[lane].rx_force.boost; if (sc->phy_type == KSERDES_PHY_XGE) { FINSR(sc->regs, LANEX_REG(lane, 0x98), 13, 13, 0); FINSR(sc->regs, LANEX_REG(lane, 0x8c), 15, 12, boost); FINSR(sc->regs, LANEX_REG(lane, 0x8c), 11, 8, att); } else { if (att != 1) { FINSR(sc->regs, CML_REG(0x84), 0, 0, 0); FINSR(sc->regs, CML_REG(0x8c), 24, 24, 0); FINSR(sc->regs, LANEX_REG(lane, 0x8c), 11, 8, att); } if (boost != 1) { FINSR(sc->regs, CML_REG(0x84), 1, 1, 0); FINSR(sc->regs, CML_REG(0x8c), 25, 25, 0); FINSR(sc->regs, LANEX_REG(lane, 0x8c), 15, 12, boost); } } } static void kserdes_set_tx_rx_fir_coeff(struct kserdes_config *sc, u32 lane) { struct kserdes_tx_coeff *tc = &sc->lane[lane].tx_coeff; if (sc->phy_type == KSERDES_PHY_XGE) { FINSR(sc->regs, LANEX_REG(lane, 0x004), 29, 26, tc->att); FINSR(sc->regs, LANEX_REG(lane, 0x0a4), 2, 0, tc->vreg); } else { FINSR(sc->regs, LANEX_REG(lane, 0x004), 28, 25, tc->att); FINSR(sc->regs, LANEX_REG(lane, 0x084), 7, 5, tc->vreg); } kserdes_config_c1_c2_cm(sc, lane); if (sc->rx_force_enable) kserdes_config_att_boost(sc, lane); } static inline void _kserdes_force_signal_detect_low( void __iomem *sregs, u32 lane) { FINSR(sregs, LANEX_REG(lane, 0x004), 2, 1, 0x2); } static inline void kserdes_force_signal_detect_low( struct kserdes_config *sc, u32 lane) { _kserdes_force_signal_detect_low(sc->regs, lane); } static inline void _kserdes_force_signal_detect_high( void __iomem *sregs, u32 lane) { FINSR(sregs, LANEX_REG(lane, 0x004), 2, 1, 0x0); } static inline void kserdes_force_signal_detect_high( struct kserdes_config *sc, u32 lane) { _kserdes_force_signal_detect_high(sc->regs, lane); } static int kserdes_deassert_reset_poll_others(struct kserdes_config *sc) { unsigned long timeout = jiffies + msecs_to_jiffies(500); unsigned long time_check; u32 lanes_not_ok = 0; u32 ofs = 28; u32 ret, i; for_each_enable_lane(sc, i) lanes_not_ok |= (1 << i); if (!FOUR_LANE(sc->regs)) ofs = 29; do { time_check = jiffies; for_each_enable_lane(sc, i) { if (!(lanes_not_ok & (1 << i))) continue; ret = kserdes_readl(sc->regs, CML_REG(0x1f8)); if (ret & BIT(ofs + i)) lanes_not_ok &= ~(1 << i); } if (!lanes_not_ok) return 0; if (time_after(time_check, timeout)) return -ETIMEDOUT; cpu_relax(); } while (true); } static int kserdes_deassert_reset_poll_pcie(struct kserdes_config *sc) { unsigned long timeout = jiffies + msecs_to_jiffies(500); unsigned long time_check; u32 lanes_not_ok = 0; u32 ret, i; for_each_enable_lane(sc, i) lanes_not_ok |= (1 << i); do { time_check = jiffies; for_each_enable_lane(sc, i) { if (!(lanes_not_ok & (1 << i))) continue; ret = _kserdes_read_select_tbus(sc->regs, i + 1, 0x02); if (!(ret & BIT(4))) lanes_not_ok &= ~(1 << i); } if (!lanes_not_ok) return 0; if (time_after(time_check, timeout)) return -ETIMEDOUT; cpu_relax(); } while (true); } static inline void _kserdes_lane_reset(void __iomem *serdes, u32 lane, u32 reset) { FINSR(serdes, LANEX_REG(lane, 0x28), 29, 29, !!reset); } static inline void kserdes_release_reset(struct kserdes_config *sc, u32 lane) { if (sc->phy_type == KSERDES_PHY_XGE) { FINSR(sc->regs, LANEX_REG(lane, 0x60), 0, 0, 0x1); } _kserdes_lane_reset(sc->regs, lane, 0); } static int kserdes_deassert_reset(struct kserdes_config *sc, u32 poll) { int ret = 0, lane; for_each_enable_lane(sc, lane) kserdes_release_reset(sc, lane); if (!poll) goto done; if (sc->phy_type == KSERDES_PHY_PCIE) ret = kserdes_deassert_reset_poll_pcie(sc); else ret = kserdes_deassert_reset_poll_others(sc); done: return ret; } static inline void kserdes_lane_disable(struct kserdes_config *sc, u32 lane) { FINSR(sc->regs, LANE_CTRL_STS_REG(lane), 31, 29, 0x4); FINSR(sc->regs, LANE_CTRL_STS_REG(lane), 15, 13, 0x4); } static inline void _kserdes_lane_enable(void __iomem *sregs, u32 lane) { FINSR(sregs, LANE_CTRL_STS_REG(lane), 31, 29, 0x7); FINSR(sregs, LANE_CTRL_STS_REG(lane), 15, 13, 0x7); } static inline int _kserdes_set_lane_ctrl_rate(void __iomem *sregs, u32 lane, enum kserdes_lane_ctrl_rate rate) { u32 rate_mode; switch (rate) { case KSERDES_FULL_RATE: rate_mode = 0x4; break; case KSERDES_QUARTER_RATE: rate_mode = 0x6; break; case KSERDES_HALF_RATE: rate_mode = 0x5; break; default: return -EINVAL; } FINSR(sregs, LANE_CTRL_STS_REG(lane), 28, 26, rate_mode); FINSR(sregs, LANE_CTRL_STS_REG(lane), 12, 10, rate_mode); return 0; } static inline void _kserdes_set_lane_loopback(void __iomem *sregs, u32 lane, enum kserdes_link_rate link_rate) { if (link_rate == KSERDES_LINK_RATE_10P3125G) { FINSR(sregs, LANEX_REG(lane, 0x0), 7, 0, 0x4); FINSR(sregs, LANEX_REG(lane, 0x4), 2, 1, 0x3); } else { FINSR(sregs, LANEX_REG(lane, 0x0), 31, 24, 0x40); } } static void kserdes_set_lane_rate(struct kserdes_config *sc, u32 lane) { int ret; ret = _kserdes_set_lane_ctrl_rate(sc->regs, lane, sc->lane[lane].ctrl_rate); if (ret) { dev_err(sc->dev, "set_lane_rate FAILED: lane = %d err = %d\n", lane, ret); return; } FINSR(sc->regs, LANEX_REG(lane, 0x30), 11, 11, 0x1); FINSR(sc->regs, LANEX_REG(lane, 0x30), 13, 12, 0x0); if (sc->lane[lane].loopback) _kserdes_set_lane_loopback(sc->regs, lane, sc->link_rate); _kserdes_lane_enable(sc->regs, lane); } static inline void _kserdes_set_wait_after(void __iomem *sregs) { FINSR(sregs, PLL_CTRL_REG, 17, 16, 0x3); } static inline void _kserdes_clear_wait_after(void __iomem *sregs) { FINSR(sregs, PLL_CTRL_REG, 17, 16, 0); } static inline void _kserdes_pll_enable(void __iomem *sregs) { FINSR(sregs, PLL_CTRL_REG, 31, 29, 0x7); } static inline void _kserdes_pll2_enable(void __iomem *sregs) { FINSR(sregs, PLL_CTRL_REG, 27, 25, 0x7); } static inline void _kserdes_pll_disable(void __iomem *sregs) { FINSR(sregs, PLL_CTRL_REG, 31, 29, 0x4); } static inline void _kserdes_pll2_disable(void __iomem *sregs) { FINSR(sregs, PLL_CTRL_REG, 27, 25, 0x4); } static inline u32 _kserdes_get_pll_status(void __iomem *sregs) { return FEXTR(kserdes_readl(sregs, PLL_CTRL_REG), 28, 28); } static inline u32 _kserdes_get_pll2_status(void __iomem *sregs) { return FEXTR(kserdes_readl(sregs, PLL_CTRL_REG), 24, 24); } static inline void kserdes_lane_enable_loopback(void __iomem *serdes, u32 lane) { FINSR(serdes, LANEX_REG(lane, 0), 31, 24, 0x40); } static inline u32 _kserdes_get_lane_status(void __iomem *sregs, u32 lane, enum kserdes_phy_type phy_type) { switch (phy_type) { case KSERDES_PHY_PCIE: return FEXTR(kserdes_readl(sregs, PLL_CTRL_REG), lane, lane); case KSERDES_PHY_XGE: return FEXTR(kserdes_readl(sregs, CML_REG(0x1f8)), (29 + lane), (29 + lane)); default: return FEXTR(kserdes_readl(sregs, PLL_CTRL_REG), (8 + lane), (8 + lane)); } } static u32 kserdes_get_pll_lanes_status(struct kserdes_config *sc) { u32 val, i; val = _kserdes_get_pll_status(sc->regs); if (!val) { goto done; } if (sc->phy_type == KSERDES_PHY_XGE) { val = _kserdes_get_pll2_status(sc->regs); if (!val) goto done; } for_each_enable_lane(sc, i) val &= _kserdes_get_lane_status(sc->regs, i, sc->phy_type); done: return val; } static int kserdes_get_status(struct kserdes_config *sc) { unsigned long timeout = jiffies + msecs_to_jiffies(500); unsigned long time_check; do { time_check = jiffies; if (kserdes_get_pll_lanes_status(sc)) break; if (time_after(time_check, timeout)) return -ETIMEDOUT; cpu_relax(); } while (true); return 0; } static inline u32 _kserdes_get_tx_termination(void __iomem *sregs, u32 lane, enum kserdes_phy_type phy_type) { return (_kserdes_read_select_tbus(sregs, lane + 1, ((phy_type == KSERDES_PHY_XGE) ? 0x1a : 0x1b)) & 0xff); } static void kserdes_set_tx_terminations(struct kserdes_config *sc, u32 term) { u32 i; for_each_lane(sc, i) { FINSR(sc->regs, LANEX_REG(i, 0x7c), 31, 24, term); FINSR(sc->regs, LANEX_REG(i, 0x7c), 20, 20, 0x1); } } static void _kserdes_write_offsets_xge(void __iomem *sregs, u32 lane, u32 cmp, struct kserdes_cmp_tap_ofs *ofs) { FINSR(sregs, CML_REG(0x8c), 23, 21, cmp); FINSR(sregs, CMU0_REG(0xfc), 26, 16, ((lane + 2) << 8) + 0x11); ofs->cmp = (_kserdes_read_tbus_val(sregs) & 0x0ff0) >> 4; FINSR(sregs, CMU0_REG(0xfc), 26, 16, ((lane + 2) << 8) + 0x11); ofs->tap1 = (_kserdes_read_tbus_val(sregs) & 0x000f) << 3; FINSR(sregs, CMU0_REG(0xfc), 26, 16, ((lane + 2) << 8) + 0x12); ofs->tap1 |= (_kserdes_read_tbus_val(sregs) & 0x0e00) >> 9; ofs->tap2 = (_kserdes_read_tbus_val(sregs) & 0x01f8) >> 3; ofs->tap3 = (_kserdes_read_tbus_val(sregs) & 0x0007) << 3; FINSR(sregs, CMU0_REG(0xfc), 26, 16, ((lane + 2) << 8) + 0x13); ofs->tap3 |= (_kserdes_read_tbus_val(sregs) & 0x0e00) >> 9; ofs->tap4 = (_kserdes_read_tbus_val(sregs) & 0x01f8) >> 3; ofs->tap5 = (_kserdes_read_tbus_val(sregs) & 0x0007) << 3; FINSR(sregs, CMU0_REG(0xfc), 26, 16, ((lane + 2) << 8) + 0x14); ofs->tap5 |= (_kserdes_read_tbus_val(sregs) & 0x0e00) >> 9; } static void kserdes_add_offsets_xge(struct kserdes_config *sc, struct kserdes_offsets *sofs) { struct kserdes_cmp_tap_ofs *ctofs; struct kserdes_cmp_tap_ofs sample; struct kserdes_lane_offsets *lofs; u32 lane, cmp; for_each_lane(sc, lane) { lofs = &sofs->lane_ofs[lane]; for_each_comparator(cmp) { ctofs = &lofs->ct_ofs[cmp]; _kserdes_write_offsets_xge(sc->regs, lane, cmp, &sample); ctofs->cmp += sample.cmp; ctofs->tap1 += sample.tap1; ctofs->tap2 += sample.tap2; ctofs->tap3 += sample.tap3; ctofs->tap4 += sample.tap4; ctofs->tap5 += sample.tap5; } } } static void kserdes_get_cmp_tap_offsets_non_xge(void __iomem *sregs, u32 lane, u32 cmp, struct kserdes_cmp_tap_ofs *ofs) { FINSR(sregs, CML_REG(0x8c), 23, 21, cmp); FINSR(sregs, CMU0_REG(0x8), 31, 24, ((lane + 1) << 5) + 0x12); ofs->cmp = (_kserdes_read_tbus_val(sregs) & 0x0ff0) >> 4; } static void kserdes_add_offsets_non_xge(struct kserdes_config *sc, struct kserdes_offsets *sofs) { struct kserdes_cmp_tap_ofs *ctofs; struct kserdes_cmp_tap_ofs sample; struct kserdes_lane_offsets *lofs; u32 lane, cmp; for_each_lane(sc, lane) { lofs = &sofs->lane_ofs[lane]; for_each_comparator(cmp) { ctofs = &lofs->ct_ofs[cmp]; kserdes_get_cmp_tap_offsets_non_xge(sc->regs, lane, cmp, &sample); ctofs->cmp += sample.cmp; } } } static void kserdes_get_average_offsets(struct kserdes_config *sc, u32 samples, struct kserdes_offsets *sofs) { struct kserdes_cmp_tap_ofs *ctofs; struct kserdes_lane_offsets *lofs; u32 i, lane, cmp; int ret; memset(sofs, 0, sizeof(*sofs)); for (i = 0; i < samples; i++) { kserdes_assert_reset(sc); ret = kserdes_deassert_reset(sc, 1); if (ret) { dev_err(sc->dev, "kserdes_get_average_offsets: reset failed %d\n", ret); return; } if (sc->phy_type == KSERDES_PHY_XGE) kserdes_add_offsets_xge(sc, sofs); else kserdes_add_offsets_non_xge(sc, sofs); } for_each_lane(sc, lane) { lofs = &sofs->lane_ofs[lane]; for_each_comparator(cmp) { ctofs = &lofs->ct_ofs[cmp]; if (sc->phy_type == KSERDES_PHY_XGE) { ctofs->cmp /= samples; ctofs->tap1 /= samples; ctofs->tap2 /= samples; ctofs->tap3 /= samples; ctofs->tap4 /= samples; ctofs->tap5 /= samples; } else { ctofs->cmp /= samples; } } } } static void _kserdes_override_cmp_tap_offsets(void __iomem *sregs, u32 lane, u32 cmp, struct kserdes_cmp_tap_ofs *ofs) { FINSR(sregs, CML_REG(0xf0), 27, 26, (lane + 1)); FINSR(sregs, CML_REG(0x98), 24, 24, 0x1); FINSR(sregs, LANEX_REG(lane, 0x2c), 2, 2, 0x1); FINSR(sregs, LANEX_REG(lane, 0x30), 7, 5, cmp); FINSR(sregs, LANEX_REG(lane, 0x5c), 31, 31, 0x1); FINSR(sregs, CML_REG(0x9c), 7, 0, ofs->cmp); FINSR(sregs, LANEX_REG(lane, 0x58), 30, 24, ofs->tap1); FINSR(sregs, LANEX_REG(lane, 0x5c), 5, 0, ofs->tap2); FINSR(sregs, LANEX_REG(lane, 0x5c), 13, 8, ofs->tap3); FINSR(sregs, LANEX_REG(lane, 0x5c), 21, 16, ofs->tap4); FINSR(sregs, LANEX_REG(lane, 0x5c), 29, 24, ofs->tap5); FINSR(sregs, LANEX_REG(lane, 0x2c), 10, 10, 0x1); FINSR(sregs, LANEX_REG(lane, 0x2c), 10, 10, 0x0); FINSR(sregs, CML_REG(0x98), 24, 24, 0x0); FINSR(sregs, LANEX_REG(lane, 0x2c), 2, 2, 0x0); FINSR(sregs, LANEX_REG(lane, 0x5c), 31, 31, 0x0); } static inline void _kserdes_override_cmp_offset_cdfe(void __iomem *sregs, u32 lane, u32 cmp, u32 cmp_offset) { FINSR(sregs, LANEX_REG(lane, 0x58), 18, 18, 0x1); FINSR(sregs, LANEX_REG(lane, 0x4c), 5, 2, (0x1 << (cmp - 1))); FINSR(sregs, LANEX_REG(lane, 0x48), 24, 17, cmp_offset); FINSR(sregs, LANEX_REG(lane, 0x48), 29, 29, 0x1); FINSR(sregs, LANEX_REG(lane, 0x48), 29, 29, 0x0); FINSR(sregs, LANEX_REG(lane, 0x58), 18, 18, 0x0); } static inline void _kserdes_override_tap_offset_cdfe(void __iomem *sregs, u32 lane, u32 tap, u32 width, u32 tap_offset) { FINSR(sregs, LANEX_REG(lane, 0x58), 23, 19, BIT(tap - 1)); FINSR(sregs, LANEX_REG(lane, 0x48), 17 + (width - 1), 17, tap_offset); FINSR(sregs, LANEX_REG(lane, 0x48), 29, 29, 0x1); FINSR(sregs, LANEX_REG(lane, 0x48), 29, 29, 0x0); } static void _kserdes_override_cmp_tap_offsets_cdfe(void __iomem *sregs, u32 lane, u32 cmp, struct kserdes_cmp_tap_ofs *ofs) { FINSR(sregs, LANEX_REG(lane, 0x58), 16, 16, 0x1); FINSR(sregs, LANEX_REG(lane, 0x48), 16, 16, 0x1); _kserdes_override_cmp_offset_cdfe(sregs, lane, cmp, ofs->cmp); FINSR(sregs, LANEX_REG(lane, 0x58), 17, 17, 0x1); _kserdes_override_tap_offset_cdfe(sregs, lane, 1, 7, ofs->tap1); _kserdes_override_tap_offset_cdfe(sregs, lane, 2, 6, ofs->tap2); _kserdes_override_tap_offset_cdfe(sregs, lane, 3, 6, ofs->tap3); _kserdes_override_tap_offset_cdfe(sregs, lane, 4, 6, ofs->tap4); _kserdes_override_tap_offset_cdfe(sregs, lane, 5, 6, ofs->tap5); FINSR(sregs, LANEX_REG(lane, 0x58), 16, 16, 0x0); FINSR(sregs, LANEX_REG(lane, 0x48), 16, 16, 0x0); FINSR(sregs, LANEX_REG(lane, 0x58), 18, 18, 0x0); FINSR(sregs, LANEX_REG(lane, 0x58), 17, 17, 0x0); } static void kserdes_set_offsets_xge(struct kserdes_config *sc, struct kserdes_offsets *sofs) { struct kserdes_cmp_tap_ofs *ctofs; struct kserdes_lane_offsets *lofs; u32 lane, cmp; for_each_lane(sc, lane) { lofs = &sofs->lane_ofs[lane]; for_each_comparator(cmp) { ctofs = &lofs->ct_ofs[cmp]; _kserdes_override_cmp_tap_offsets(sc->regs, lane, cmp, ctofs); _kserdes_override_cmp_tap_offsets_cdfe(sc->regs, lane, cmp, ctofs); } } } static void kserdes_set_offsets_non_xge(struct kserdes_config *sc, struct kserdes_offsets *sofs) { struct kserdes_cmp_tap_ofs *ctofs; struct kserdes_lane_offsets *lofs; u32 lane, cmp; for_each_lane(sc, lane) { lofs = &sofs->lane_ofs[lane]; for_each_comparator(cmp) { ctofs = &lofs->ct_ofs[cmp]; _kserdes_override_cmp_tap_offsets(sc->regs, lane, cmp, ctofs); } } } static void kserdes_set_offsets(struct kserdes_config *sc, struct kserdes_offsets *sofs) { if (sc->phy_type == KSERDES_PHY_XGE) kserdes_set_offsets_xge(sc, sofs); else kserdes_set_offsets_non_xge(sc, sofs); } static void kserdes_phyb_init_cfg(struct kserdes_config *sc, struct kserdes_offsets *sofs) { int lane; for_each_enable_lane(sc, lane) kserdes_force_signal_detect_low(sc, lane); /* amount of time to sleep is by experiment */ usleep_range(10, 20); kserdes_get_average_offsets(sc, OFFSET_SAMPLES, sofs); kserdes_set_offsets(sc, sofs); /* amount of time to sleep is by experiment */ usleep_range(10, 20); for_each_lane(sc, lane) kserdes_force_signal_detect_high(sc, lane); /* amount of time to sleep is by experiment */ usleep_range(10, 20); } static int kserdes_wait_lane_rx_valid(struct kserdes_config *sc, u32 lane) { unsigned long timeout = jiffies + msecs_to_jiffies(500); unsigned long time_check; u32 status; do { time_check = jiffies; status = _kserdes_read_select_tbus(sc->regs, lane + 1, 0x02); if (status & 0x20) return 0; if (time_after(time_check, timeout)) return -ETIMEDOUT; cpu_relax(); } while (true); } static int kserdes_att_phya_cfg(struct kserdes_config *sc) { u32 i, att_read[KSERDES_MAX_LANES], att_start[KSERDES_MAX_LANES]; int ret; for_each_lane(sc, i) { att_start[i] = kserdes_readl(sc->regs, LANEX_REG(i, 0x8c)); att_start[i] = (att_start[i] >> 8) & 0xf; } for_each_lane(sc, i) { att_read[i] = _kserdes_read_select_tbus( sc->regs, i + 1, (sc->phy_type == KSERDES_PHY_XGE) ? 0x10 : 0x11); att_read[i] = (att_read[i] >> 4) & 0xf; } for_each_lane(sc, i) { FINSR(sc->regs, LANEX_REG(i, 0x8c), 11, 8, att_read[i]); } FINSR(sc->regs, CML_REG(0x84), 0, 0, 0x0); FINSR(sc->regs, CML_REG(0x8c), 24, 24, 0x0); FINSR(sc->regs, CML_REG(0x98), 7, 7, 0x1); FINSR(sc->regs, CML_REG(0x98), 7, 7, 0x0); usleep_range(300, 400); for_each_enable_lane(sc, i) { ret = kserdes_wait_lane_rx_valid(sc, i); if (ret) { dev_err(sc->dev, "lane %d wait rx valid failed: %d\n", i, ret); return ret; } } for_each_lane(sc, i) FINSR(sc->regs, LANEX_REG(i, 0x8c), 11, 8, att_start[i]); FINSR(sc->regs, CML_REG(0x84), 0, 0, 0x1); FINSR(sc->regs, CML_REG(0x8c), 24, 24, 0x1); return 0; } static int kserdes_boost_phya_cfg(struct kserdes_config *sc, u32 lane) { u32 boost_read; int ret; ret = kserdes_wait_lane_rx_valid(sc, lane); if (ret) { dev_err(sc->dev, "lane %d wait rx valid failed: %d\n", lane, ret); return ret; } boost_read = _kserdes_read_select_tbus( sc->regs, lane + 1, (sc->phy_type == KSERDES_PHY_XGE) ? 0x10 : 0x11); boost_read = (boost_read >> 8) & 0xf; if (!boost_read) { FINSR(sc->regs, LANEX_REG(lane, 0x2c), 2, 2, 0x1); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 18, 12, 0x2); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 9, 3, 0x1); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 10, 10, 0x1); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 10, 10, 0x0); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 2, 2, 0x0); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 18, 12, 0x0); FINSR(sc->regs, LANEX_REG(lane, 0x2c), 9, 3, 0x0); } return 0; } static inline void kserdes_rx_att_boost_phya_cfg(struct kserdes_config *sc) { int lane; kserdes_att_phya_cfg(sc); for_each_enable_lane(sc, lane) kserdes_boost_phya_cfg(sc, lane); } static void kserdes_rx_att_boost_phyb_cfg(struct kserdes_config *sc, u32 lane) { u32 tbus_ofs, rxeq_init_reg_ofs, rxeq_ln_reg_ofs, rxeq_ln_force_bit; void __iomem *sregs = sc->regs; u32 att_start, att_read, boost_read; int ret; if (sc->phy_type == KSERDES_PHY_XGE) { tbus_ofs = 0x10; rxeq_init_reg_ofs = 0x9c; rxeq_ln_reg_ofs = 0x98; rxeq_ln_force_bit = 14; } else { tbus_ofs = 0x11; rxeq_init_reg_ofs = 0x84; rxeq_ln_reg_ofs = 0xac; rxeq_ln_force_bit = 11; } att_start = kserdes_readl(sregs, LANEX_REG(lane, 0x8c)); att_start = (att_start >> 8) & 0xf; att_read = _kserdes_read_select_tbus(sregs, lane + 1, tbus_ofs); att_read = (att_read >> 4) & 0xf; FINSR(sregs, LANEX_REG(lane, 0x8c), 11, 8, att_read); FINSR(sregs, LANEX_REG(lane, rxeq_init_reg_ofs), 0, 0, 0x0); FINSR(sregs, CML_REG(0x8c), 24, 24, 0x0); FINSR(sregs, LANEX_REG(lane, rxeq_ln_reg_ofs), rxeq_ln_force_bit, rxeq_ln_force_bit, 0x1); FINSR(sregs, LANEX_REG(lane, rxeq_ln_reg_ofs), rxeq_ln_force_bit, rxeq_ln_force_bit, 0x0); ret = kserdes_wait_lane_rx_valid(sc, lane); if (ret) { dev_err(sc->dev, "lane %d wait rx valid failed: %d\n", lane, ret); } /* amount of time to sleep is by experiment */ usleep_range(300, 400); boost_read = _kserdes_read_select_tbus(sregs, lane + 1, tbus_ofs); boost_read = (boost_read >> 8) & 0xf; if (!boost_read) { FINSR(sregs, LANEX_REG(lane, 0x2c), 2, 2, 0x1); FINSR(sregs, LANEX_REG(lane, 0x2c), 18, 12, 0x2); FINSR(sregs, LANEX_REG(lane, 0x2c), 9, 3, 0x1); FINSR(sregs, LANEX_REG(lane, 0x2c), 10, 10, 0x1); FINSR(sregs, LANEX_REG(lane, 0x2c), 10, 10, 0x0); FINSR(sregs, LANEX_REG(lane, 0x2c), 2, 2, 0x0); FINSR(sregs, LANEX_REG(lane, 0x2c), 18, 12, 0x0); FINSR(sregs, LANEX_REG(lane, 0x2c), 9, 3, 0x0); } FINSR(sregs, LANEX_REG(lane, 0x8c), 11, 8, att_start); FINSR(sregs, LANEX_REG(lane, rxeq_init_reg_ofs), 0, 0, 0x1); FINSR(sregs, CML_REG(0x8c), 24, 24, 0x1); } static void kserdes_rx_att_boost_cfg(struct kserdes_config *sc) { int lane; if (sc->phy_type != KSERDES_PHY_XGE) { kserdes_rx_att_boost_phya_cfg(sc); } else { for_each_lane(sc, lane) kserdes_rx_att_boost_phyb_cfg(sc, lane); } } static int kserdes_sgmii_lanes_enable(struct kserdes_config *sc) { int ret, i; u32 val, lanes_enable = 0; for_each_enable_lane(sc, i) lanes_enable |= (1 << i); for_each_lane(sc, i) kserdes_set_tx_idle(sc, i); kserdes_highspeed_cfg(sc); kserdes_assert_reset(sc); for_each_enable_lane(sc, i) kserdes_set_tx_rx_fir_coeff(sc, i); for_each_enable_lane(sc, i) kserdes_force_signal_detect_low(sc, i); ret = kserdes_deassert_reset(sc, 0); if (ret) { dev_err(sc->dev, "kserdes_deassert_reset FAILED %d\n", ret); return ret; } for_each_enable_lane(sc, i) kserdes_set_lane_rate(sc, i); _kserdes_pll_enable(sc->regs); ret = kserdes_get_status(sc); if (ret) { dev_err(sc->dev, "kserdes_get_status FAILED %d\n", ret); return ret; } val = _kserdes_get_tx_termination(sc->regs, 0, sc->phy_type); kserdes_set_tx_terminations(sc, val); for_each_enable_lane(sc, i) kserdes_clr_tx_idle(sc, i); for_each_enable_lane(sc, i) kserdes_force_signal_detect_high(sc, i); for_each_enable_lane(sc, i) { ret = kserdes_wait_lane_rx_valid(sc, i); if (ret) { dev_err(sc->dev, "lane %d wait rx valid failed: %d\n", i, ret); return ret; } } if (!sc->rx_force_enable) kserdes_rx_att_boost_cfg(sc); _kserdes_clear_wait_after(sc->regs); return lanes_enable; } static int kserdes_sgmii_init(struct kserdes_config *sc) { return kserdes_load_init_fw(sc, ks2_gbe_serdes_firmwares, ARRAY_SIZE(ks2_gbe_serdes_firmwares)); } static inline void _kserdes_set_link_loss_wait(void __iomem *sregs, u32 link_loss_wait) { kserdes_writel(sregs, LINK_LOSS_WAIT_REG, link_loss_wait); } static inline void _kserdes_reset(void __iomem *sregs) { FINSR(sregs, CPU_CTRL_REG, 29, 29, 0x1); /* amount of time to sleep is by experiment */ usleep_range(10, 20); FINSR(sregs, CPU_CTRL_REG, 29, 29, 0x0); /* amount of time to sleep is by experiment */ usleep_range(10, 20); } static inline void kserdes_xge_pll_enable(struct kserdes_config *sc) { if (!sc->firmware) FINSR(sc->regs, CML_REG(0), 7, 0, 0x1f); if (sc->link_rate == KSERDES_LINK_RATE_10P3125G) { _kserdes_pll_enable(sc->regs); _kserdes_pll2_enable(sc->regs); } else if (sc->link_rate == KSERDES_LINK_RATE_1P25G) { kserdes_writel(sc->regs, PLL_CTRL_REG, PLL_ENABLE_1P25G); } } static inline void _kserdes_xge_enable_pcs(void __iomem *sregs, u32 lane) { FINSR(sregs, LANE_CTRL_STS_REG(lane), 23, 21, 0x7); FINSR(sregs, LANE_CTRL_STS_REG(lane), 5, 3, 0x7); FINSR(sregs, LANE_CTRL_STS_REG(lane), 16, 16, 0x1); FINSR(sregs, LANE_CTRL_STS_REG(lane), 19, 19, 0x1); } static inline void kserdes_xge_lane_enable(struct kserdes_config *sc, u32 lane) { u32 lane_ctrl_rate = sc->lane[lane].ctrl_rate; if (sc->link_rate == KSERDES_LINK_RATE_10P3125G) _kserdes_set_lane_ctrl_rate(sc->regs, lane, lane_ctrl_rate); else if (sc->link_rate == KSERDES_LINK_RATE_1P25G) kserdes_writel(sc->regs, LANE_CTRL_STS_REG(lane), LANE_CTRL_1P25G); _kserdes_xge_enable_pcs(sc->regs, lane); _kserdes_lane_enable(sc->regs, lane); if (sc->lane[lane].loopback) _kserdes_set_lane_loopback(sc->regs, lane, sc->link_rate); } static inline void _kserdes_enable_xgmii_port(struct regmap *peripheral_regmap, u32 port) { regmap_update_bits(peripheral_regmap, XGE_CTRL_OFFSET, GENMASK(port, port), BIT(port)); } static inline void _kserdes_reset_rx(void __iomem *sregs, int lane) { _kserdes_force_signal_detect_low(sregs, lane); /* amount of time to sleep is by experiment */ usleep_range(1000, 2000); _kserdes_force_signal_detect_high(sregs, lane); } /* Call every 10 ms */ static int kserdes_check_link_status(struct kserdes_config *sc, u32 *current_state, u32 *lanes_up_map) { u32 pcsr_rx_stat, blk_lock, blk_errs; int loss, i, status = 1; int ret; for_each_enable_lane(sc, i) { loss = (kserdes_readl(sc->regs, LANE_CTRL_STS_REG(i))) & 0x01; ret = regmap_read(sc->pcsr_regmap, PCSR_RX_STATUS(i), &pcsr_rx_stat); if (ret) return ret; blk_lock = (pcsr_rx_stat >> 30) & 0x1; blk_errs = (pcsr_rx_stat >> 16) & 0x0ff; if (blk_errs) blk_lock = 0; switch (current_state[i]) { case 0: if (!loss && blk_lock) { dev_dbg(sc->dev, "XGE PCSR Linked Lane: %d\n", i); FINSR(sc->regs, LANEX_REG(i, 0x04), 2, 1, 0x3); current_state[i] = 1; } else { if (!blk_lock) { dev_dbg(sc->dev, "XGE PCSR Recover Lane: %d\n", i); _kserdes_reset_rx(sc->regs, i); } } break; case 1: if (!blk_lock) { current_state[i] = 2; } break; case 2: if (blk_lock) { current_state[i] = 1; } else { _kserdes_reset_rx(sc->regs, i); current_state[i] = 0; } break; default: dev_info(sc->dev, "XGE: unknown current_state[%d] %d\n", i, current_state[i]); break; } if (blk_errs) { regmap_update_bits(sc->pcsr_regmap, PCSR_RX_CTL(i), GENMASK(7, 0), 0x19); regmap_update_bits(sc->pcsr_regmap, PCSR_RX_CTL(i), GENMASK(7, 0), 0x00); } if (current_state[i] == 1) { *lanes_up_map |= BIT(i); } else { *lanes_up_map &= ~BIT(i); status = 0; } } return status; } static int kserdes_wait_link_up(struct kserdes_config *sc, u32 *lanes_up_map) { u32 current_state[KSERDES_MAX_LANES]; unsigned long timeout, time_check; int i, link_up, ret = 0; memset(current_state, 0, sizeof(current_state)); timeout = jiffies + 2 * HZ; do { time_check = jiffies; /* amount of time to sleep is by experiment */ usleep_range(10000, 20000); link_up = kserdes_check_link_status(sc, current_state, lanes_up_map); /* * if we did not get link up then wait 100ms * before calling it again */ if (link_up) break; for_each_enable_lane(sc, i) { if (!(*lanes_up_map & BIT(i))) { dev_dbg(sc->dev, "XGE: detected lane %d down\n", i); } } if (time_after(time_check, timeout)) { ret = -ETIMEDOUT; break; } } while (1); return ret; } static int kserdes_xge_lanes_enable(struct kserdes_config *sc) { struct kserdes_offsets sofs; int ret, i; int lanes_up_map = 0; if (sc->firmware) { return 0; } kserdes_highspeed_cfg(sc); kserdes_xge_pll_enable(sc); for_each_lane(sc, i) kserdes_xge_lane_enable(sc, i); ret = kserdes_get_status(sc); if (ret) { dev_err(sc->dev, "kserdes_xge_lanes_enable get status FAILED %d\n", ret); return ret; } kserdes_phyb_init_cfg(sc, &sofs); for_each_lane(sc, i) _kserdes_enable_xgmii_port(sc->peripheral_regmap, i); kserdes_wait_link_up(sc, &lanes_up_map); return lanes_up_map; } static inline void kserdes_xfw_get_lane_params(struct kserdes_config *sc, int lane) { struct kserdes_fw_config *fw = &sc->fw; u32 tx_ctrl, val_0, val_1; u32 phy_a = PHY_A(sc->regs); val_0 = kserdes_readl(sc->regs, LANEX_REG(lane, 0x04)); val_1 = kserdes_readl(sc->regs, LANEX_REG(lane, 0x08)); tx_ctrl = ((((val_0 >> 18) & 0x1) << 24) | (((val_1 >> 0) & 0xffff) << 8) | (((val_0 >> 24) & 0xff) << 0)); if (phy_a) { fw->cm = (val_1 >> 12) & 0xf; fw->c1 = (val_1 >> 0) & 0x1f; fw->c2 = (val_1 >> 8) & 0xf; } else { fw->cm = (tx_ctrl >> 16) & 0xf; fw->c1 = (tx_ctrl >> 8) & 0x1f; fw->c2 = (tx_ctrl >> 13) & 0x7; fw->c2 = fw->c2 | (((tx_ctrl >> 24) & 0x1) << 3); } val_0 = _kserdes_read_select_tbus(sc->regs, lane + 1, (phy_a ? 0x11 : 0x10)); fw->attn = (val_0 >> 4) & 0xf; fw->boost = (val_0 >> 8) & 0xf; val_0 = _kserdes_read_select_tbus(sc->regs, lane + 1, 0x5); fw->dlpf = (val_0 >> 2) & 0x3ff; val_0 = _kserdes_read_select_tbus(sc->regs, lane + 1, 0x6); fw->rxcal = (val_0 >> 3) & 0xff; } static inline void kserdes_xfw_mem_init(struct kserdes_config *sc) { struct kserdes_fw_config *fw = &sc->fw; u32 i, lane_config = 0; for_each_lane(sc, i) lane_config = (lane_config << 8) | (fw->lane_config[i] & 0xff); lane_config <<= 8; kserdes_writel(sc->regs, MEM_ADR_REG, KSERDES_XFW_CONFIG_START_ADDR); for (i = KSERDES_XFW_CONFIG_START_ADDR; i < KSERDES_XFW_PARAM_START_ADDR; i += 4) kserdes_writel(sc->regs, MEM_DATINC_REG, 0x00000000); kserdes_writel(sc->regs, MEM_DATINC_REG, XFM_FLUSH_CMD); kserdes_writel(sc->regs, MEM_DATINC_REG, fw->fast_train); kserdes_writel(sc->regs, MEM_DATINC_REG, 0x00000000); kserdes_writel(sc->regs, MEM_DATINC_REG, fw->lane_seeds); kserdes_writel(sc->regs, MEM_DATINC_REG, lane_config); } static int kserdes_xge_init(struct kserdes_config *sc) { return kserdes_load_init_fw(sc, ks2_xgbe_serdes_firmwares, ARRAY_SIZE(ks2_xgbe_serdes_firmwares)); } static int kserdes_pcie_lanes_enable(struct kserdes_config *sc) { int ret, i; u32 lanes_enable = 0; for_each_enable_lane(sc, i) lanes_enable |= (1 << i); for_each_lane(sc, i) { kserdes_release_reset(sc, i); if (sc->lane[i].loopback) _kserdes_set_lane_loopback(sc->regs, i, sc->link_rate); } ret = kserdes_get_status(sc); if (ret) return ret; return lanes_enable; } static int kserdes_pcie_init(struct kserdes_config *sc) { return kserdes_load_init_fw(sc, ks2_pcie_serdes_firmwares, ARRAY_SIZE(ks2_pcie_serdes_firmwares)); } static int kserdes_lanes_enable(struct kserdes_config *sc) { switch (sc->phy_type) { case KSERDES_PHY_SGMII: return kserdes_sgmii_lanes_enable(sc); case KSERDES_PHY_XGE: return kserdes_xge_lanes_enable(sc); case KSERDES_PHY_PCIE: return kserdes_pcie_lanes_enable(sc); default: return -EINVAL; } } static int kserdes_init(struct phy *phy) { struct kserdes_dev *sd = phy_get_drvdata(phy); struct kserdes_config *sc = &sd->sc; int ret; switch (sc->phy_type) { case KSERDES_PHY_SGMII: ret = kserdes_sgmii_init(sc); break; case KSERDES_PHY_XGE: ret = kserdes_xge_init(sc); break; case KSERDES_PHY_PCIE: ret = kserdes_pcie_init(sc); break; default: ret = -EINVAL; } if (ret < 0) { dev_err(sd->dev, "serdes initialization failed %d\n", ret); goto done; } ret = kserdes_lanes_enable(sc); if (ret < 0) { dev_err(sd->dev, "serdes lanes enable failed: %d\n", ret); goto done; } dev_dbg(sd->dev, "serdes config done lanes(mask) 0x%x\n", ret); done: return ret; } static int kserdes_of_parse_lane(struct device *dev, struct device_node *np, struct kserdes_config *sc) { struct kserdes_lane_config *lc; struct kserdes_equalizer *eq; struct kserdes_tx_coeff *tc; int lane_num, ret; ret = of_property_read_u32(np, "reg", &lane_num); if (ret) { dev_err(dev, "Failed to parse reg\n"); return -EINVAL; } if (lane_num >= sc->lanes) { dev_err(dev, "Invalid lane number %u\n", lane_num); return -EINVAL; } lc = &sc->lane[lane_num]; lc->enable = true; dev_dbg(dev, "lane %d enabled\n", lane_num); if (of_property_read_u32(np, "control-rate", &lc->ctrl_rate)) { dev_info(dev, "use default lane control-rate: %u\n", lc->ctrl_rate); } dev_dbg(dev, "lane control-rate: %d\n", lc->ctrl_rate); if (of_find_property(np, "loopback", NULL)) lc->loopback = true; else lc->loopback = false; dev_dbg(dev, "lane loopback: %d\n", lc->loopback); eq = &lc->rx_start; if (of_property_read_u32_array(np, "rx-start", &eq->att, 2)) { dev_info(dev, "use default lane rx-start 0 0\n"); eq->att = 0; eq->boost = 0; } dev_dbg(dev, "lane rx-start: %d %d\n", eq->att, eq->boost); eq = &lc->rx_force; if (of_property_read_u32_array(np, "rx-force", &eq->att, 2)) { dev_info(dev, "use default lane rx-force 0 0\n"); eq->att = 0; eq->boost = 0; } dev_dbg(dev, "lane rx-force: %d %d\n", eq->att, eq->boost); tc = &lc->tx_coeff; if (of_property_read_u32_array(np, "tx-coeff", &tc->c1, 5)) { dev_info(dev, "use default tx-coeff 0\n"); tc->c1 = 0; } dev_dbg(dev, "tx-coeff: %d %d %d %d %d\n", tc->c1, tc->c2, tc->cm, tc->att, tc->vreg); return 0; } static void kserdes_set_sgmii_defaults(struct kserdes_config *sc) { int i; sc->link_rate = KSERDES_LINK_RATE_1P25G; sc->lanes = 4; sc->rx_force_enable = false; for_each_lane(sc, i) { memset(&sc->lane[i], 0, sizeof(sc->lane[i])); sc->lane[i].ctrl_rate = KSERDES_QUARTER_RATE; } } static void kserdes_set_xge_defaults(struct kserdes_config *sc) { int i; sc->link_rate = KSERDES_LINK_RATE_10P3125G; sc->lanes = 2; sc->rx_force_enable = false; for_each_lane(sc, i) { memset(&sc->lane[i], 0, sizeof(sc->lane[i])); sc->lane[i].ctrl_rate = KSERDES_FULL_RATE; } } static void kserdes_set_pcie_defaults(struct kserdes_config *sc) { int i; sc->link_rate = KSERDES_LINK_RATE_5G; sc->lanes = 2; sc->rx_force_enable = false; for_each_lane(sc, i) memset(&sc->lane[i], 0, sizeof(sc->lane[i])); } static void kserdes_set_defaults(struct kserdes_config *sc, enum kserdes_phy_type phy_type) { switch (phy_type) { case KSERDES_PHY_SGMII: kserdes_set_sgmii_defaults(sc); break; case KSERDES_PHY_XGE: kserdes_set_xge_defaults(sc); break; case KSERDES_PHY_PCIE: kserdes_set_pcie_defaults(sc); break; default: break; } } static const struct of_device_id kserdes_of_match[] = { { .compatible = "ti,keystone-serdes-gbe", .data = &kserdes_devtype[KSERDES_PHY_SGMII], }, { .compatible = "ti,keystone-serdes-xgbe", .data = &kserdes_devtype[KSERDES_PHY_XGE], }, { .compatible = "ti,keystone-serdes-pcie", .data = &kserdes_devtype[KSERDES_PHY_PCIE], }, { }, }; MODULE_DEVICE_TABLE(of, kserdes_of_match); static int kserdes_of_parse(struct platform_device *pdev, struct kserdes_dev *sd, struct device_node *np) { const struct of_device_id *of_id; struct kserdes_config *sc = &sd->sc; struct device_node *lanes_np, *child; struct device *dev = sd->dev; struct resource res; void __iomem *regs; int ret; ret = of_address_to_resource(np, SERDES_REG_INDEX, &res); if (ret) { dev_err(dev, "Can't xlate serdes reg addr of node(%s)\n", np->name); return ret; } regs = devm_ioremap_resource(dev, &res); if (IS_ERR(regs)) { dev_err(dev, "Failed to map serdes register base\n"); return PTR_ERR(regs); } sc->regs = regs; of_id = of_match_device(kserdes_of_match, dev); if (!of_id) { dev_err(dev, "unknown phy type\n"); return -EINVAL; } pdev->id_entry = of_id->data; sc->phy_type = pdev->id_entry->driver_data; sc->dev = dev; kserdes_set_defaults(sc, sc->phy_type); if (sc->phy_type == KSERDES_PHY_XGE) { sc->peripheral_regmap = syscon_regmap_lookup_by_phandle(np, "syscon-peripheral"); if (IS_ERR(sc->peripheral_regmap)) { dev_err(sc->dev, "peripheral regmap lookup failed: %ld\n", PTR_ERR(sc->peripheral_regmap)); return PTR_ERR(sc->peripheral_regmap); } sc->pcsr_regmap = syscon_regmap_lookup_by_phandle(np, "syscon-link"); if (IS_ERR(sc->pcsr_regmap)) { dev_err(sc->dev, "link regmap lookup failed: %ld\n", PTR_ERR(sc->pcsr_regmap)); return PTR_ERR(sc->pcsr_regmap); } } if (of_property_read_u32(np, "link-rate-kbps", &sc->link_rate)) { dev_info(dev, "use default link-rate-kbps: %u\n", sc->link_rate); } if (of_property_read_u32(np, "num-lanes", &sc->lanes)) dev_info(dev, "use default num-lanes %d\n", sc->lanes); if (sc->lanes > KSERDES_MAX_LANES) { sc->lanes = KSERDES_MAX_LANES; dev_info(dev, "use max allowed lanes %d\n", sc->lanes); } if (of_property_read_bool(np, "rx-force-enable")) sc->rx_force_enable = true; else sc->rx_force_enable = false; lanes_np = of_get_child_by_name(np, "lanes"); if (lanes_np) { for_each_available_child_of_node(lanes_np, child) { ret = kserdes_of_parse_lane(dev, child, sc); if (ret) { of_node_put(child); of_node_put(lanes_np); return ret; } of_node_put(child); } of_node_put(lanes_np); } return 0; } static struct phy_ops kserdes_ops = { .init = kserdes_init, .owner = THIS_MODULE, }; static int kserdes_probe(struct platform_device *pdev) { struct phy_provider *phy_provider; struct kserdes_dev *sd; struct device_node *np = pdev->dev.of_node; struct device *dev = &pdev->dev; int ret; sd = devm_kzalloc(dev, sizeof(*sd), GFP_KERNEL); if (!sd) return -ENOMEM; sd->dev = dev; ret = kserdes_of_parse(pdev, sd, np); if (ret) return ret; dev_set_drvdata(dev, sd); sd->phy = devm_phy_create(dev, NULL, &kserdes_ops); if (IS_ERR(sd->phy)) return PTR_ERR(sd->phy); phy_set_drvdata(sd->phy, sd); phy_provider = devm_of_phy_provider_register(sd->dev, of_phy_simple_xlate); if (IS_ERR(phy_provider)) return PTR_ERR_OR_ZERO(phy_provider); dev_vdbg(&pdev->dev, "probed"); return 0; } static struct platform_driver kserdes_driver = { .probe = kserdes_probe, .driver = { .of_match_table = kserdes_of_match, .name = "ti,keystone-serdes", } }; module_platform_driver(kserdes_driver); MODULE_AUTHOR("WingMan Kwok "); MODULE_DESCRIPTION("TI Keystone SerDes driver"); MODULE_LICENSE("GPL v2");