18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+ 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * drivers/net/phy/davicom.c 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Driver for Davicom PHYs 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Author: Andy Fleming 88c2ecf20Sopenharmony_ci * 98c2ecf20Sopenharmony_ci * Copyright (c) 2004 Freescale Semiconductor, Inc. 108c2ecf20Sopenharmony_ci */ 118c2ecf20Sopenharmony_ci#include <linux/kernel.h> 128c2ecf20Sopenharmony_ci#include <linux/string.h> 138c2ecf20Sopenharmony_ci#include <linux/errno.h> 148c2ecf20Sopenharmony_ci#include <linux/unistd.h> 158c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 168c2ecf20Sopenharmony_ci#include <linux/init.h> 178c2ecf20Sopenharmony_ci#include <linux/delay.h> 188c2ecf20Sopenharmony_ci#include <linux/netdevice.h> 198c2ecf20Sopenharmony_ci#include <linux/etherdevice.h> 208c2ecf20Sopenharmony_ci#include <linux/skbuff.h> 218c2ecf20Sopenharmony_ci#include <linux/spinlock.h> 228c2ecf20Sopenharmony_ci#include <linux/mm.h> 238c2ecf20Sopenharmony_ci#include <linux/module.h> 248c2ecf20Sopenharmony_ci#include <linux/mii.h> 258c2ecf20Sopenharmony_ci#include <linux/ethtool.h> 268c2ecf20Sopenharmony_ci#include <linux/phy.h> 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci#include <asm/io.h> 298c2ecf20Sopenharmony_ci#include <asm/irq.h> 308c2ecf20Sopenharmony_ci#include <linux/uaccess.h> 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_ci#define MII_DM9161_SCR 0x10 338c2ecf20Sopenharmony_ci#define MII_DM9161_SCR_INIT 0x0610 348c2ecf20Sopenharmony_ci#define MII_DM9161_SCR_RMII 0x0100 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_ci/* DM9161 Interrupt Register */ 378c2ecf20Sopenharmony_ci#define MII_DM9161_INTR 0x15 388c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_PEND 0x8000 398c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_DPLX_MASK 0x0800 408c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_SPD_MASK 0x0400 418c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_LINK_MASK 0x0200 428c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_MASK 0x0100 438c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_DPLX_CHANGE 0x0010 448c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_SPD_CHANGE 0x0008 458c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_LINK_CHANGE 0x0004 468c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_INIT 0x0000 478c2ecf20Sopenharmony_ci#define MII_DM9161_INTR_STOP \ 488c2ecf20Sopenharmony_ci(MII_DM9161_INTR_DPLX_MASK | MII_DM9161_INTR_SPD_MASK \ 498c2ecf20Sopenharmony_ci | MII_DM9161_INTR_LINK_MASK | MII_DM9161_INTR_MASK) 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci/* DM9161 10BT Configuration/Status */ 528c2ecf20Sopenharmony_ci#define MII_DM9161_10BTCSR 0x12 538c2ecf20Sopenharmony_ci#define MII_DM9161_10BTCSR_INIT 0x7800 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Davicom PHY driver"); 568c2ecf20Sopenharmony_ciMODULE_AUTHOR("Andy Fleming"); 578c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci#define DM9161_DELAY 1 618c2ecf20Sopenharmony_cistatic int dm9161_config_intr(struct phy_device *phydev) 628c2ecf20Sopenharmony_ci{ 638c2ecf20Sopenharmony_ci int temp; 648c2ecf20Sopenharmony_ci 658c2ecf20Sopenharmony_ci temp = phy_read(phydev, MII_DM9161_INTR); 668c2ecf20Sopenharmony_ci 678c2ecf20Sopenharmony_ci if (temp < 0) 688c2ecf20Sopenharmony_ci return temp; 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ci if (PHY_INTERRUPT_ENABLED == phydev->interrupts) 718c2ecf20Sopenharmony_ci temp &= ~(MII_DM9161_INTR_STOP); 728c2ecf20Sopenharmony_ci else 738c2ecf20Sopenharmony_ci temp |= MII_DM9161_INTR_STOP; 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci temp = phy_write(phydev, MII_DM9161_INTR, temp); 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci return temp; 788c2ecf20Sopenharmony_ci} 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_cistatic int dm9161_config_aneg(struct phy_device *phydev) 818c2ecf20Sopenharmony_ci{ 828c2ecf20Sopenharmony_ci int err; 838c2ecf20Sopenharmony_ci 848c2ecf20Sopenharmony_ci /* Isolate the PHY */ 858c2ecf20Sopenharmony_ci err = phy_write(phydev, MII_BMCR, BMCR_ISOLATE); 868c2ecf20Sopenharmony_ci 878c2ecf20Sopenharmony_ci if (err < 0) 888c2ecf20Sopenharmony_ci return err; 898c2ecf20Sopenharmony_ci 908c2ecf20Sopenharmony_ci /* Configure the new settings */ 918c2ecf20Sopenharmony_ci err = genphy_config_aneg(phydev); 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_ci if (err < 0) 948c2ecf20Sopenharmony_ci return err; 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ci return 0; 978c2ecf20Sopenharmony_ci} 988c2ecf20Sopenharmony_ci 998c2ecf20Sopenharmony_cistatic int dm9161_config_init(struct phy_device *phydev) 1008c2ecf20Sopenharmony_ci{ 1018c2ecf20Sopenharmony_ci int err, temp; 1028c2ecf20Sopenharmony_ci 1038c2ecf20Sopenharmony_ci /* Isolate the PHY */ 1048c2ecf20Sopenharmony_ci err = phy_write(phydev, MII_BMCR, BMCR_ISOLATE); 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci if (err < 0) 1078c2ecf20Sopenharmony_ci return err; 1088c2ecf20Sopenharmony_ci 1098c2ecf20Sopenharmony_ci switch (phydev->interface) { 1108c2ecf20Sopenharmony_ci case PHY_INTERFACE_MODE_MII: 1118c2ecf20Sopenharmony_ci temp = MII_DM9161_SCR_INIT; 1128c2ecf20Sopenharmony_ci break; 1138c2ecf20Sopenharmony_ci case PHY_INTERFACE_MODE_RMII: 1148c2ecf20Sopenharmony_ci temp = MII_DM9161_SCR_INIT | MII_DM9161_SCR_RMII; 1158c2ecf20Sopenharmony_ci break; 1168c2ecf20Sopenharmony_ci default: 1178c2ecf20Sopenharmony_ci return -EINVAL; 1188c2ecf20Sopenharmony_ci } 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci /* Do not bypass the scrambler/descrambler */ 1218c2ecf20Sopenharmony_ci err = phy_write(phydev, MII_DM9161_SCR, temp); 1228c2ecf20Sopenharmony_ci if (err < 0) 1238c2ecf20Sopenharmony_ci return err; 1248c2ecf20Sopenharmony_ci 1258c2ecf20Sopenharmony_ci /* Clear 10BTCSR to default */ 1268c2ecf20Sopenharmony_ci err = phy_write(phydev, MII_DM9161_10BTCSR, MII_DM9161_10BTCSR_INIT); 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_ci if (err < 0) 1298c2ecf20Sopenharmony_ci return err; 1308c2ecf20Sopenharmony_ci 1318c2ecf20Sopenharmony_ci /* Reconnect the PHY, and enable Autonegotiation */ 1328c2ecf20Sopenharmony_ci return phy_write(phydev, MII_BMCR, BMCR_ANENABLE); 1338c2ecf20Sopenharmony_ci} 1348c2ecf20Sopenharmony_ci 1358c2ecf20Sopenharmony_cistatic int dm9161_ack_interrupt(struct phy_device *phydev) 1368c2ecf20Sopenharmony_ci{ 1378c2ecf20Sopenharmony_ci int err = phy_read(phydev, MII_DM9161_INTR); 1388c2ecf20Sopenharmony_ci 1398c2ecf20Sopenharmony_ci return (err < 0) ? err : 0; 1408c2ecf20Sopenharmony_ci} 1418c2ecf20Sopenharmony_ci 1428c2ecf20Sopenharmony_cistatic struct phy_driver dm91xx_driver[] = { 1438c2ecf20Sopenharmony_ci{ 1448c2ecf20Sopenharmony_ci .phy_id = 0x0181b880, 1458c2ecf20Sopenharmony_ci .name = "Davicom DM9161E", 1468c2ecf20Sopenharmony_ci .phy_id_mask = 0x0ffffff0, 1478c2ecf20Sopenharmony_ci /* PHY_BASIC_FEATURES */ 1488c2ecf20Sopenharmony_ci .config_init = dm9161_config_init, 1498c2ecf20Sopenharmony_ci .config_aneg = dm9161_config_aneg, 1508c2ecf20Sopenharmony_ci .ack_interrupt = dm9161_ack_interrupt, 1518c2ecf20Sopenharmony_ci .config_intr = dm9161_config_intr, 1528c2ecf20Sopenharmony_ci}, { 1538c2ecf20Sopenharmony_ci .phy_id = 0x0181b8b0, 1548c2ecf20Sopenharmony_ci .name = "Davicom DM9161B/C", 1558c2ecf20Sopenharmony_ci .phy_id_mask = 0x0ffffff0, 1568c2ecf20Sopenharmony_ci /* PHY_BASIC_FEATURES */ 1578c2ecf20Sopenharmony_ci .config_init = dm9161_config_init, 1588c2ecf20Sopenharmony_ci .config_aneg = dm9161_config_aneg, 1598c2ecf20Sopenharmony_ci .ack_interrupt = dm9161_ack_interrupt, 1608c2ecf20Sopenharmony_ci .config_intr = dm9161_config_intr, 1618c2ecf20Sopenharmony_ci}, { 1628c2ecf20Sopenharmony_ci .phy_id = 0x0181b8a0, 1638c2ecf20Sopenharmony_ci .name = "Davicom DM9161A", 1648c2ecf20Sopenharmony_ci .phy_id_mask = 0x0ffffff0, 1658c2ecf20Sopenharmony_ci /* PHY_BASIC_FEATURES */ 1668c2ecf20Sopenharmony_ci .config_init = dm9161_config_init, 1678c2ecf20Sopenharmony_ci .config_aneg = dm9161_config_aneg, 1688c2ecf20Sopenharmony_ci .ack_interrupt = dm9161_ack_interrupt, 1698c2ecf20Sopenharmony_ci .config_intr = dm9161_config_intr, 1708c2ecf20Sopenharmony_ci}, { 1718c2ecf20Sopenharmony_ci .phy_id = 0x00181b80, 1728c2ecf20Sopenharmony_ci .name = "Davicom DM9131", 1738c2ecf20Sopenharmony_ci .phy_id_mask = 0x0ffffff0, 1748c2ecf20Sopenharmony_ci /* PHY_BASIC_FEATURES */ 1758c2ecf20Sopenharmony_ci .ack_interrupt = dm9161_ack_interrupt, 1768c2ecf20Sopenharmony_ci .config_intr = dm9161_config_intr, 1778c2ecf20Sopenharmony_ci} }; 1788c2ecf20Sopenharmony_ci 1798c2ecf20Sopenharmony_cimodule_phy_driver(dm91xx_driver); 1808c2ecf20Sopenharmony_ci 1818c2ecf20Sopenharmony_cistatic struct mdio_device_id __maybe_unused davicom_tbl[] = { 1828c2ecf20Sopenharmony_ci { 0x0181b880, 0x0ffffff0 }, 1838c2ecf20Sopenharmony_ci { 0x0181b8b0, 0x0ffffff0 }, 1848c2ecf20Sopenharmony_ci { 0x0181b8a0, 0x0ffffff0 }, 1858c2ecf20Sopenharmony_ci { 0x00181b80, 0x0ffffff0 }, 1868c2ecf20Sopenharmony_ci { } 1878c2ecf20Sopenharmony_ci}; 1888c2ecf20Sopenharmony_ci 1898c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(mdio, davicom_tbl); 190