1*0cefeebaSMichael Barkowski /* 2*0cefeebaSMichael Barkowski * Driver for ICPlus PHYs 3*0cefeebaSMichael Barkowski * 4*0cefeebaSMichael Barkowski * Copyright (c) 2007 Freescale Semiconductor, Inc. 5*0cefeebaSMichael Barkowski * 6*0cefeebaSMichael Barkowski * This program is free software; you can redistribute it and/or modify it 7*0cefeebaSMichael Barkowski * under the terms of the GNU General Public License as published by the 8*0cefeebaSMichael Barkowski * Free Software Foundation; either version 2 of the License, or (at your 9*0cefeebaSMichael Barkowski * option) any later version. 10*0cefeebaSMichael Barkowski * 11*0cefeebaSMichael Barkowski */ 12*0cefeebaSMichael Barkowski #include <linux/kernel.h> 13*0cefeebaSMichael Barkowski #include <linux/string.h> 14*0cefeebaSMichael Barkowski #include <linux/errno.h> 15*0cefeebaSMichael Barkowski #include <linux/unistd.h> 16*0cefeebaSMichael Barkowski #include <linux/slab.h> 17*0cefeebaSMichael Barkowski #include <linux/interrupt.h> 18*0cefeebaSMichael Barkowski #include <linux/init.h> 19*0cefeebaSMichael Barkowski #include <linux/delay.h> 20*0cefeebaSMichael Barkowski #include <linux/netdevice.h> 21*0cefeebaSMichael Barkowski #include <linux/etherdevice.h> 22*0cefeebaSMichael Barkowski #include <linux/skbuff.h> 23*0cefeebaSMichael Barkowski #include <linux/spinlock.h> 24*0cefeebaSMichael Barkowski #include <linux/mm.h> 25*0cefeebaSMichael Barkowski #include <linux/module.h> 26*0cefeebaSMichael Barkowski #include <linux/mii.h> 27*0cefeebaSMichael Barkowski #include <linux/ethtool.h> 28*0cefeebaSMichael Barkowski #include <linux/phy.h> 29*0cefeebaSMichael Barkowski 30*0cefeebaSMichael Barkowski #include <asm/io.h> 31*0cefeebaSMichael Barkowski #include <asm/irq.h> 32*0cefeebaSMichael Barkowski #include <asm/uaccess.h> 33*0cefeebaSMichael Barkowski 34*0cefeebaSMichael Barkowski MODULE_DESCRIPTION("ICPlus IP175C PHY driver"); 35*0cefeebaSMichael Barkowski MODULE_AUTHOR("Michael Barkowski"); 36*0cefeebaSMichael Barkowski MODULE_LICENSE("GPL"); 37*0cefeebaSMichael Barkowski 38*0cefeebaSMichael Barkowski static int ip175c_config_init(struct phy_device *phydev) 39*0cefeebaSMichael Barkowski { 40*0cefeebaSMichael Barkowski int err, i; 41*0cefeebaSMichael Barkowski static int full_reset_performed = 0; 42*0cefeebaSMichael Barkowski 43*0cefeebaSMichael Barkowski if (full_reset_performed == 0) { 44*0cefeebaSMichael Barkowski 45*0cefeebaSMichael Barkowski /* master reset */ 46*0cefeebaSMichael Barkowski err = phydev->bus->write(phydev->bus, 30, 0, 0x175c); 47*0cefeebaSMichael Barkowski if (err < 0) 48*0cefeebaSMichael Barkowski return err; 49*0cefeebaSMichael Barkowski 50*0cefeebaSMichael Barkowski /* ensure no bus delays overlap reset period */ 51*0cefeebaSMichael Barkowski err = phydev->bus->read(phydev->bus, 30, 0); 52*0cefeebaSMichael Barkowski 53*0cefeebaSMichael Barkowski /* data sheet specifies reset period is 2 msec */ 54*0cefeebaSMichael Barkowski mdelay(2); 55*0cefeebaSMichael Barkowski 56*0cefeebaSMichael Barkowski /* enable IP175C mode */ 57*0cefeebaSMichael Barkowski err = phydev->bus->write(phydev->bus, 29, 31, 0x175c); 58*0cefeebaSMichael Barkowski if (err < 0) 59*0cefeebaSMichael Barkowski return err; 60*0cefeebaSMichael Barkowski 61*0cefeebaSMichael Barkowski /* Set MII0 speed and duplex (in PHY mode) */ 62*0cefeebaSMichael Barkowski err = phydev->bus->write(phydev->bus, 29, 22, 0x420); 63*0cefeebaSMichael Barkowski if (err < 0) 64*0cefeebaSMichael Barkowski return err; 65*0cefeebaSMichael Barkowski 66*0cefeebaSMichael Barkowski /* reset switch ports */ 67*0cefeebaSMichael Barkowski for (i = 0; i < 5; i++) { 68*0cefeebaSMichael Barkowski err = phydev->bus->write(phydev->bus, i, 69*0cefeebaSMichael Barkowski MII_BMCR, BMCR_RESET); 70*0cefeebaSMichael Barkowski if (err < 0) 71*0cefeebaSMichael Barkowski return err; 72*0cefeebaSMichael Barkowski } 73*0cefeebaSMichael Barkowski 74*0cefeebaSMichael Barkowski for (i = 0; i < 5; i++) 75*0cefeebaSMichael Barkowski err = phydev->bus->read(phydev->bus, i, MII_BMCR); 76*0cefeebaSMichael Barkowski 77*0cefeebaSMichael Barkowski mdelay(2); 78*0cefeebaSMichael Barkowski 79*0cefeebaSMichael Barkowski full_reset_performed = 1; 80*0cefeebaSMichael Barkowski } 81*0cefeebaSMichael Barkowski 82*0cefeebaSMichael Barkowski if (phydev->addr != 4) { 83*0cefeebaSMichael Barkowski phydev->state = PHY_RUNNING; 84*0cefeebaSMichael Barkowski phydev->speed = SPEED_100; 85*0cefeebaSMichael Barkowski phydev->duplex = DUPLEX_FULL; 86*0cefeebaSMichael Barkowski phydev->link = 1; 87*0cefeebaSMichael Barkowski netif_carrier_on(phydev->attached_dev); 88*0cefeebaSMichael Barkowski } 89*0cefeebaSMichael Barkowski 90*0cefeebaSMichael Barkowski return 0; 91*0cefeebaSMichael Barkowski } 92*0cefeebaSMichael Barkowski 93*0cefeebaSMichael Barkowski static int ip175c_read_status(struct phy_device *phydev) 94*0cefeebaSMichael Barkowski { 95*0cefeebaSMichael Barkowski if (phydev->addr == 4) /* WAN port */ 96*0cefeebaSMichael Barkowski genphy_read_status(phydev); 97*0cefeebaSMichael Barkowski else 98*0cefeebaSMichael Barkowski /* Don't need to read status for switch ports */ 99*0cefeebaSMichael Barkowski phydev->irq = PHY_IGNORE_INTERRUPT; 100*0cefeebaSMichael Barkowski 101*0cefeebaSMichael Barkowski return 0; 102*0cefeebaSMichael Barkowski } 103*0cefeebaSMichael Barkowski 104*0cefeebaSMichael Barkowski static int ip175c_config_aneg(struct phy_device *phydev) 105*0cefeebaSMichael Barkowski { 106*0cefeebaSMichael Barkowski if (phydev->addr == 4) /* WAN port */ 107*0cefeebaSMichael Barkowski genphy_config_aneg(phydev); 108*0cefeebaSMichael Barkowski 109*0cefeebaSMichael Barkowski return 0; 110*0cefeebaSMichael Barkowski } 111*0cefeebaSMichael Barkowski 112*0cefeebaSMichael Barkowski static struct phy_driver ip175c_driver = { 113*0cefeebaSMichael Barkowski .phy_id = 0x02430d80, 114*0cefeebaSMichael Barkowski .name = "ICPlus IP175C", 115*0cefeebaSMichael Barkowski .phy_id_mask = 0x0ffffff0, 116*0cefeebaSMichael Barkowski .features = PHY_BASIC_FEATURES, 117*0cefeebaSMichael Barkowski .config_init = &ip175c_config_init, 118*0cefeebaSMichael Barkowski .config_aneg = &ip175c_config_aneg, 119*0cefeebaSMichael Barkowski .read_status = &ip175c_read_status, 120*0cefeebaSMichael Barkowski .driver = { .owner = THIS_MODULE,}, 121*0cefeebaSMichael Barkowski }; 122*0cefeebaSMichael Barkowski 123*0cefeebaSMichael Barkowski static int __init ip175c_init(void) 124*0cefeebaSMichael Barkowski { 125*0cefeebaSMichael Barkowski return phy_driver_register(&ip175c_driver); 126*0cefeebaSMichael Barkowski } 127*0cefeebaSMichael Barkowski 128*0cefeebaSMichael Barkowski static void __exit ip175c_exit(void) 129*0cefeebaSMichael Barkowski { 130*0cefeebaSMichael Barkowski phy_driver_unregister(&ip175c_driver); 131*0cefeebaSMichael Barkowski } 132*0cefeebaSMichael Barkowski 133*0cefeebaSMichael Barkowski module_init(ip175c_init); 134*0cefeebaSMichael Barkowski module_exit(ip175c_exit); 135