162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Driver for Teranetics PHY 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Author: Shaohui Xie <Shaohui.Xie@freescale.com> 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Copyright 2015 Freescale Semiconductor, Inc. 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include <linux/kernel.h> 1162306a36Sopenharmony_ci#include <linux/module.h> 1262306a36Sopenharmony_ci#include <linux/mii.h> 1362306a36Sopenharmony_ci#include <linux/ethtool.h> 1462306a36Sopenharmony_ci#include <linux/mdio.h> 1562306a36Sopenharmony_ci#include <linux/phy.h> 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ciMODULE_DESCRIPTION("Teranetics PHY driver"); 1862306a36Sopenharmony_ciMODULE_AUTHOR("Shaohui Xie <Shaohui.Xie@freescale.com>"); 1962306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#define PHY_ID_TN2020 0x00a19410 2262306a36Sopenharmony_ci#define MDIO_PHYXS_LNSTAT_SYNC0 0x0001 2362306a36Sopenharmony_ci#define MDIO_PHYXS_LNSTAT_SYNC1 0x0002 2462306a36Sopenharmony_ci#define MDIO_PHYXS_LNSTAT_SYNC2 0x0004 2562306a36Sopenharmony_ci#define MDIO_PHYXS_LNSTAT_SYNC3 0x0008 2662306a36Sopenharmony_ci#define MDIO_PHYXS_LNSTAT_ALIGN 0x1000 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci#define MDIO_PHYXS_LANE_READY (MDIO_PHYXS_LNSTAT_SYNC0 | \ 2962306a36Sopenharmony_ci MDIO_PHYXS_LNSTAT_SYNC1 | \ 3062306a36Sopenharmony_ci MDIO_PHYXS_LNSTAT_SYNC2 | \ 3162306a36Sopenharmony_ci MDIO_PHYXS_LNSTAT_SYNC3 | \ 3262306a36Sopenharmony_ci MDIO_PHYXS_LNSTAT_ALIGN) 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cistatic int teranetics_aneg_done(struct phy_device *phydev) 3562306a36Sopenharmony_ci{ 3662306a36Sopenharmony_ci /* auto negotiation state can only be checked when using copper 3762306a36Sopenharmony_ci * port, if using fiber port, just lie it's done. 3862306a36Sopenharmony_ci */ 3962306a36Sopenharmony_ci if (!phy_read_mmd(phydev, MDIO_MMD_VEND1, 93)) 4062306a36Sopenharmony_ci return genphy_c45_aneg_done(phydev); 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_ci return 1; 4362306a36Sopenharmony_ci} 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_cistatic int teranetics_read_status(struct phy_device *phydev) 4662306a36Sopenharmony_ci{ 4762306a36Sopenharmony_ci int reg; 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci phydev->link = 1; 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci phydev->speed = SPEED_10000; 5262306a36Sopenharmony_ci phydev->duplex = DUPLEX_FULL; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci if (!phy_read_mmd(phydev, MDIO_MMD_VEND1, 93)) { 5562306a36Sopenharmony_ci reg = phy_read_mmd(phydev, MDIO_MMD_PHYXS, MDIO_PHYXS_LNSTAT); 5662306a36Sopenharmony_ci if (reg < 0 || 5762306a36Sopenharmony_ci !((reg & MDIO_PHYXS_LANE_READY) == MDIO_PHYXS_LANE_READY)) { 5862306a36Sopenharmony_ci phydev->link = 0; 5962306a36Sopenharmony_ci return 0; 6062306a36Sopenharmony_ci } 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_ci reg = phy_read_mmd(phydev, MDIO_MMD_AN, MDIO_STAT1); 6362306a36Sopenharmony_ci if (reg < 0 || !(reg & MDIO_STAT1_LSTATUS)) 6462306a36Sopenharmony_ci phydev->link = 0; 6562306a36Sopenharmony_ci } 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci return 0; 6862306a36Sopenharmony_ci} 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_cistatic int teranetics_match_phy_device(struct phy_device *phydev) 7162306a36Sopenharmony_ci{ 7262306a36Sopenharmony_ci return phydev->c45_ids.device_ids[3] == PHY_ID_TN2020; 7362306a36Sopenharmony_ci} 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_cistatic struct phy_driver teranetics_driver[] = { 7662306a36Sopenharmony_ci{ 7762306a36Sopenharmony_ci .phy_id = PHY_ID_TN2020, 7862306a36Sopenharmony_ci .phy_id_mask = 0xffffffff, 7962306a36Sopenharmony_ci .name = "Teranetics TN2020", 8062306a36Sopenharmony_ci .features = PHY_10GBIT_FEATURES, 8162306a36Sopenharmony_ci .aneg_done = teranetics_aneg_done, 8262306a36Sopenharmony_ci .config_aneg = gen10g_config_aneg, 8362306a36Sopenharmony_ci .read_status = teranetics_read_status, 8462306a36Sopenharmony_ci .match_phy_device = teranetics_match_phy_device, 8562306a36Sopenharmony_ci}, 8662306a36Sopenharmony_ci}; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cimodule_phy_driver(teranetics_driver); 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_cistatic struct mdio_device_id __maybe_unused teranetics_tbl[] = { 9162306a36Sopenharmony_ci { PHY_ID_TN2020, 0xffffffff }, 9262306a36Sopenharmony_ci { } 9362306a36Sopenharmony_ci}; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(mdio, teranetics_tbl); 96