|  | /* | 
|  | * drivers/net/phy/fixed.c | 
|  | * | 
|  | * Driver for fixed PHYs, when transceiver is able to operate in one fixed mode. | 
|  | * | 
|  | * Author: Vitaly Bordug | 
|  | * | 
|  | * Copyright (c) 2006 MontaVista Software, Inc. | 
|  | * | 
|  | * This program is free software; you can redistribute  it and/or modify it | 
|  | * under  the terms of  the GNU General  Public License as published by the | 
|  | * Free Software Foundation;  either version 2 of the  License, or (at your | 
|  | * option) any later version. | 
|  | * | 
|  | */ | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/string.h> | 
|  | #include <linux/errno.h> | 
|  | #include <linux/unistd.h> | 
|  | #include <linux/slab.h> | 
|  | #include <linux/interrupt.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/delay.h> | 
|  | #include <linux/netdevice.h> | 
|  | #include <linux/etherdevice.h> | 
|  | #include <linux/skbuff.h> | 
|  | #include <linux/spinlock.h> | 
|  | #include <linux/mm.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/mii.h> | 
|  | #include <linux/ethtool.h> | 
|  | #include <linux/phy.h> | 
|  | #include <linux/phy_fixed.h> | 
|  |  | 
|  | #include <asm/io.h> | 
|  | #include <asm/irq.h> | 
|  | #include <asm/uaccess.h> | 
|  |  | 
|  | /* we need to track the allocated pointers in order to free them on exit */ | 
|  | static struct fixed_info *fixed_phy_ptrs[CONFIG_FIXED_MII_AMNT*MAX_PHY_AMNT]; | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | *  If something weird is required to be done with link/speed, | 
|  | * network driver is able to assign a function to implement this. | 
|  | * May be useful for PHY's that need to be software-driven. | 
|  | *-----------------------------------------------------------------------------*/ | 
|  | int fixed_mdio_set_link_update(struct phy_device *phydev, | 
|  | int (*link_update) (struct net_device *, | 
|  | struct fixed_phy_status *)) | 
|  | { | 
|  | struct fixed_info *fixed; | 
|  |  | 
|  | if (link_update == NULL) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (phydev) { | 
|  | if (phydev->bus) { | 
|  | fixed = phydev->bus->priv; | 
|  | fixed->link_update = link_update; | 
|  | return 0; | 
|  | } | 
|  | } | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | EXPORT_SYMBOL(fixed_mdio_set_link_update); | 
|  |  | 
|  | struct fixed_info *fixed_mdio_get_phydev (int phydev_ind) | 
|  | { | 
|  | if (phydev_ind >= MAX_PHY_AMNT) | 
|  | return NULL; | 
|  | return fixed_phy_ptrs[phydev_ind]; | 
|  | } | 
|  |  | 
|  | EXPORT_SYMBOL(fixed_mdio_get_phydev); | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | *  This is used for updating internal mii regs from the status | 
|  | *-----------------------------------------------------------------------------*/ | 
|  | #if defined(CONFIG_FIXED_MII_100_FDX) || defined(CONFIG_FIXED_MII_10_FDX) || defined(CONFIG_FIXED_MII_1000_FDX) | 
|  | static int fixed_mdio_update_regs(struct fixed_info *fixed) | 
|  | { | 
|  | u16 *regs = fixed->regs; | 
|  | u16 bmsr = 0; | 
|  | u16 bmcr = 0; | 
|  |  | 
|  | if (!regs) { | 
|  | printk(KERN_ERR "%s: regs not set up", __FUNCTION__); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if (fixed->phy_status.link) | 
|  | bmsr |= BMSR_LSTATUS; | 
|  |  | 
|  | if (fixed->phy_status.duplex) { | 
|  | bmcr |= BMCR_FULLDPLX; | 
|  |  | 
|  | switch (fixed->phy_status.speed) { | 
|  | case 100: | 
|  | bmsr |= BMSR_100FULL; | 
|  | bmcr |= BMCR_SPEED100; | 
|  | break; | 
|  |  | 
|  | case 10: | 
|  | bmsr |= BMSR_10FULL; | 
|  | break; | 
|  | } | 
|  | } else { | 
|  | switch (fixed->phy_status.speed) { | 
|  | case 100: | 
|  | bmsr |= BMSR_100HALF; | 
|  | bmcr |= BMCR_SPEED100; | 
|  | break; | 
|  |  | 
|  | case 10: | 
|  | bmsr |= BMSR_100HALF; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | regs[MII_BMCR] = bmcr; | 
|  | regs[MII_BMSR] = bmsr | 0x800;	/*we are always capable of 10 hdx */ | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int fixed_mii_read(struct mii_bus *bus, int phy_id, int location) | 
|  | { | 
|  | struct fixed_info *fixed = bus->priv; | 
|  |  | 
|  | /* if user has registered link update callback, use it */ | 
|  | if (fixed->phydev) | 
|  | if (fixed->phydev->attached_dev) { | 
|  | if (fixed->link_update) { | 
|  | fixed->link_update(fixed->phydev->attached_dev, | 
|  | &fixed->phy_status); | 
|  | fixed_mdio_update_regs(fixed); | 
|  | } | 
|  | } | 
|  |  | 
|  | if ((unsigned int)location >= fixed->regs_num) | 
|  | return -1; | 
|  | return fixed->regs[location]; | 
|  | } | 
|  |  | 
|  | static int fixed_mii_write(struct mii_bus *bus, int phy_id, int location, | 
|  | u16 val) | 
|  | { | 
|  | /* do nothing for now */ | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int fixed_mii_reset(struct mii_bus *bus) | 
|  | { | 
|  | /*nothing here - no way/need to reset it */ | 
|  | return 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | static int fixed_config_aneg(struct phy_device *phydev) | 
|  | { | 
|  | /* :TODO:03/13/2006 09:45:37 PM:: | 
|  | The full autoneg funcionality can be emulated, | 
|  | but no need to have anything here for now | 
|  | */ | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | * the manual bind will do the magic - with phy_id_mask == 0 | 
|  | * match will never return true... | 
|  | *-----------------------------------------------------------------------------*/ | 
|  | static struct phy_driver fixed_mdio_driver = { | 
|  | .name = "Fixed PHY", | 
|  | #ifdef CONFIG_FIXED_MII_1000_FDX | 
|  | .features = PHY_GBIT_FEATURES, | 
|  | #else | 
|  | .features = PHY_BASIC_FEATURES, | 
|  | #endif | 
|  | .config_aneg = fixed_config_aneg, | 
|  | .read_status = genphy_read_status, | 
|  | .driver = { .owner = THIS_MODULE, }, | 
|  | }; | 
|  |  | 
|  | static void fixed_mdio_release(struct device *dev) | 
|  | { | 
|  | struct phy_device *phydev = container_of(dev, struct phy_device, dev); | 
|  | struct mii_bus *bus = phydev->bus; | 
|  | struct fixed_info *fixed = bus->priv; | 
|  |  | 
|  | kfree(phydev); | 
|  | kfree(bus->dev); | 
|  | kfree(bus); | 
|  | kfree(fixed->regs); | 
|  | kfree(fixed); | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | *  This func is used to create all the necessary stuff, bind | 
|  | * the fixed phy driver and register all it on the mdio_bus_type. | 
|  | * speed is either 10 or 100 or 1000, duplex is boolean. | 
|  | * number is used to create multiple fixed PHYs, so that several devices can | 
|  | * utilize them simultaneously. | 
|  | * | 
|  | * The device on mdio bus will look like [bus_id]:[phy_id], | 
|  | * bus_id = number | 
|  | * phy_id = speed+duplex. | 
|  | *-----------------------------------------------------------------------------*/ | 
|  | #if defined(CONFIG_FIXED_MII_100_FDX) || defined(CONFIG_FIXED_MII_10_FDX) || defined(CONFIG_FIXED_MII_1000_FDX) | 
|  | struct fixed_info *fixed_mdio_register_device( | 
|  | int bus_id, int speed, int duplex, u8 phy_id) | 
|  | { | 
|  | struct mii_bus *new_bus; | 
|  | struct fixed_info *fixed; | 
|  | struct phy_device *phydev; | 
|  | int err; | 
|  |  | 
|  | struct device *dev = kzalloc(sizeof(struct device), GFP_KERNEL); | 
|  |  | 
|  | if (dev == NULL) | 
|  | goto err_dev_alloc; | 
|  |  | 
|  | new_bus = kzalloc(sizeof(struct mii_bus), GFP_KERNEL); | 
|  |  | 
|  | if (new_bus == NULL) | 
|  | goto err_bus_alloc; | 
|  |  | 
|  | fixed = kzalloc(sizeof(struct fixed_info), GFP_KERNEL); | 
|  |  | 
|  | if (fixed == NULL) | 
|  | goto err_fixed_alloc; | 
|  |  | 
|  | fixed->regs = kzalloc(MII_REGS_NUM * sizeof(int), GFP_KERNEL); | 
|  | if (NULL == fixed->regs) | 
|  | goto err_fixed_regs_alloc; | 
|  |  | 
|  | fixed->regs_num = MII_REGS_NUM; | 
|  | fixed->phy_status.speed = speed; | 
|  | fixed->phy_status.duplex = duplex; | 
|  | fixed->phy_status.link = 1; | 
|  |  | 
|  | new_bus->name = "Fixed MII Bus"; | 
|  | new_bus->read = &fixed_mii_read; | 
|  | new_bus->write = &fixed_mii_write; | 
|  | new_bus->reset = &fixed_mii_reset; | 
|  | /*set up workspace */ | 
|  | fixed_mdio_update_regs(fixed); | 
|  | new_bus->priv = fixed; | 
|  |  | 
|  | new_bus->dev = dev; | 
|  | dev_set_drvdata(dev, new_bus); | 
|  |  | 
|  | /* create phy_device and register it on the mdio bus */ | 
|  | phydev = phy_device_create(new_bus, 0, 0); | 
|  | if (phydev == NULL) | 
|  | goto err_phy_dev_create; | 
|  |  | 
|  | /* | 
|  | * Put the phydev pointer into the fixed pack so that bus read/write | 
|  | * code could be able to access for instance attached netdev. Well it | 
|  | * doesn't have to do so, only in case of utilizing user-specified | 
|  | * link-update... | 
|  | */ | 
|  |  | 
|  | fixed->phydev = phydev; | 
|  | phydev->speed = speed; | 
|  | phydev->duplex = duplex; | 
|  |  | 
|  | phydev->irq = PHY_IGNORE_INTERRUPT; | 
|  | phydev->dev.bus = &mdio_bus_type; | 
|  |  | 
|  | snprintf(phydev->dev.bus_id, BUS_ID_SIZE, | 
|  | PHY_ID_FMT, bus_id, phy_id); | 
|  |  | 
|  | phydev->bus = new_bus; | 
|  |  | 
|  | phydev->dev.driver = &fixed_mdio_driver.driver; | 
|  | phydev->dev.release = fixed_mdio_release; | 
|  | err = phydev->dev.driver->probe(&phydev->dev); | 
|  | if (err < 0) { | 
|  | printk(KERN_ERR "Phy %s: problems with fixed driver\n", | 
|  | phydev->dev.bus_id); | 
|  | goto err_out; | 
|  | } | 
|  | err = device_register(&phydev->dev); | 
|  | if (err) { | 
|  | printk(KERN_ERR "Phy %s failed to register\n", | 
|  | phydev->dev.bus_id); | 
|  | goto err_out; | 
|  | } | 
|  | //phydev->state = PHY_RUNNING; /* make phy go up quick, but in 10Mbit/HDX | 
|  | return fixed; | 
|  |  | 
|  | err_out: | 
|  | kfree(phydev); | 
|  | err_phy_dev_create: | 
|  | kfree(fixed->regs); | 
|  | err_fixed_regs_alloc: | 
|  | kfree(fixed); | 
|  | err_fixed_alloc: | 
|  | kfree(new_bus); | 
|  | err_bus_alloc: | 
|  | kfree(dev); | 
|  | err_dev_alloc: | 
|  |  | 
|  | return NULL; | 
|  |  | 
|  | } | 
|  | #endif | 
|  |  | 
|  | MODULE_DESCRIPTION("Fixed PHY device & driver for PAL"); | 
|  | MODULE_AUTHOR("Vitaly Bordug"); | 
|  | MODULE_LICENSE("GPL"); | 
|  |  | 
|  | static int __init fixed_init(void) | 
|  | { | 
|  | int cnt = 0; | 
|  | int i; | 
|  | /* register on the bus... Not expected to be matched | 
|  | * with anything there... | 
|  | * | 
|  | */ | 
|  | phy_driver_register(&fixed_mdio_driver); | 
|  |  | 
|  | /* We will create several mdio devices here, and will bound the upper | 
|  | * driver to them. | 
|  | * | 
|  | * Then the external software can lookup the phy bus by searching | 
|  | * for 0:101, to be connected to the virtual 100M Fdx phy. | 
|  | * | 
|  | * In case several virtual PHYs required, the bus_id will be in form | 
|  | * [num]:[duplex]+[speed], which make it able even to define | 
|  | * driver-specific link control callback, if for instance PHY is | 
|  | * completely SW-driven. | 
|  | */ | 
|  | for (i=1; i <= CONFIG_FIXED_MII_AMNT; i++) { | 
|  | #ifdef CONFIG_FIXED_MII_1000_FDX | 
|  | fixed_phy_ptrs[cnt++] = fixed_mdio_register_device(0, 1000, 1, i); | 
|  | #endif | 
|  | #ifdef CONFIG_FIXED_MII_100_FDX | 
|  | fixed_phy_ptrs[cnt++] = fixed_mdio_register_device(1, 100, 1, i); | 
|  | #endif | 
|  | #ifdef CONFIG_FIXED_MII_10_FDX | 
|  | fixed_phy_ptrs[cnt++] = fixed_mdio_register_device(2, 10, 1, i); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void __exit fixed_exit(void) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | phy_driver_unregister(&fixed_mdio_driver); | 
|  | for (i=0; i < MAX_PHY_AMNT; i++) | 
|  | if ( fixed_phy_ptrs[i] ) | 
|  | device_unregister(&fixed_phy_ptrs[i]->phydev->dev); | 
|  | } | 
|  |  | 
|  | module_init(fixed_init); | 
|  | module_exit(fixed_exit); |