| /* |
| * ibm_ocp_phy.c |
| * |
| * PHY drivers for the ibm ocp ethernet driver. Borrowed |
| * from sungem_phy.c, though I only kept the generic MII |
| * driver for now. |
| * |
| * This file should be shared with other drivers or eventually |
| * merged as the "low level" part of miilib |
| * |
| * (c) 2003, Benjamin Herrenscmidt (benh@kernel.crashing.org) |
| * |
| */ |
| |
| #include <linux/config.h> |
| |
| #include <linux/module.h> |
| |
| #include <linux/kernel.h> |
| #include <linux/sched.h> |
| #include <linux/types.h> |
| #include <linux/netdevice.h> |
| #include <linux/etherdevice.h> |
| #include <linux/mii.h> |
| #include <linux/ethtool.h> |
| #include <linux/delay.h> |
| |
| #include "ibm_emac_phy.h" |
| |
| static int reset_one_mii_phy(struct mii_phy *phy, int phy_id) |
| { |
| u16 val; |
| int limit = 10000; |
| |
| val = __phy_read(phy, phy_id, MII_BMCR); |
| val &= ~BMCR_ISOLATE; |
| val |= BMCR_RESET; |
| __phy_write(phy, phy_id, MII_BMCR, val); |
| |
| udelay(100); |
| |
| while (limit--) { |
| val = __phy_read(phy, phy_id, MII_BMCR); |
| if ((val & BMCR_RESET) == 0) |
| break; |
| udelay(10); |
| } |
| if ((val & BMCR_ISOLATE) && limit > 0) |
| __phy_write(phy, phy_id, MII_BMCR, val & ~BMCR_ISOLATE); |
| |
| return (limit <= 0); |
| } |
| |
| static int cis8201_init(struct mii_phy *phy) |
| { |
| u16 epcr; |
| |
| epcr = phy_read(phy, MII_CIS8201_EPCR); |
| epcr &= ~EPCR_MODE_MASK; |
| |
| switch (phy->mode) { |
| case PHY_MODE_TBI: |
| epcr |= EPCR_TBI_MODE; |
| break; |
| case PHY_MODE_RTBI: |
| epcr |= EPCR_RTBI_MODE; |
| break; |
| case PHY_MODE_GMII: |
| epcr |= EPCR_GMII_MODE; |
| break; |
| case PHY_MODE_RGMII: |
| default: |
| epcr |= EPCR_RGMII_MODE; |
| } |
| |
| phy_write(phy, MII_CIS8201_EPCR, epcr); |
| |
| return 0; |
| } |
| |
| static int genmii_setup_aneg(struct mii_phy *phy, u32 advertise) |
| { |
| u16 ctl, adv; |
| |
| phy->autoneg = 1; |
| phy->speed = SPEED_10; |
| phy->duplex = DUPLEX_HALF; |
| phy->pause = 0; |
| phy->advertising = advertise; |
| |
| /* Setup standard advertise */ |
| adv = phy_read(phy, MII_ADVERTISE); |
| adv &= ~(ADVERTISE_ALL | ADVERTISE_100BASE4); |
| if (advertise & ADVERTISED_10baseT_Half) |
| adv |= ADVERTISE_10HALF; |
| if (advertise & ADVERTISED_10baseT_Full) |
| adv |= ADVERTISE_10FULL; |
| if (advertise & ADVERTISED_100baseT_Half) |
| adv |= ADVERTISE_100HALF; |
| if (advertise & ADVERTISED_100baseT_Full) |
| adv |= ADVERTISE_100FULL; |
| phy_write(phy, MII_ADVERTISE, adv); |
| |
| /* Start/Restart aneg */ |
| ctl = phy_read(phy, MII_BMCR); |
| ctl |= (BMCR_ANENABLE | BMCR_ANRESTART); |
| phy_write(phy, MII_BMCR, ctl); |
| |
| return 0; |
| } |
| |
| static int genmii_setup_forced(struct mii_phy *phy, int speed, int fd) |
| { |
| u16 ctl; |
| |
| phy->autoneg = 0; |
| phy->speed = speed; |
| phy->duplex = fd; |
| phy->pause = 0; |
| |
| ctl = phy_read(phy, MII_BMCR); |
| ctl &= ~(BMCR_FULLDPLX | BMCR_SPEED100 | BMCR_ANENABLE); |
| |
| /* First reset the PHY */ |
| phy_write(phy, MII_BMCR, ctl | BMCR_RESET); |
| |
| /* Select speed & duplex */ |
| switch (speed) { |
| case SPEED_10: |
| break; |
| case SPEED_100: |
| ctl |= BMCR_SPEED100; |
| break; |
| case SPEED_1000: |
| default: |
| return -EINVAL; |
| } |
| if (fd == DUPLEX_FULL) |
| ctl |= BMCR_FULLDPLX; |
| phy_write(phy, MII_BMCR, ctl); |
| |
| return 0; |
| } |
| |
| static int genmii_poll_link(struct mii_phy *phy) |
| { |
| u16 status; |
| |
| (void)phy_read(phy, MII_BMSR); |
| status = phy_read(phy, MII_BMSR); |
| if ((status & BMSR_LSTATUS) == 0) |
| return 0; |
| if (phy->autoneg && !(status & BMSR_ANEGCOMPLETE)) |
| return 0; |
| return 1; |
| } |
| |
| #define MII_CIS8201_ACSR 0x1c |
| #define ACSR_DUPLEX_STATUS 0x0020 |
| #define ACSR_SPEED_1000BASET 0x0010 |
| #define ACSR_SPEED_100BASET 0x0008 |
| |
| static int cis8201_read_link(struct mii_phy *phy) |
| { |
| u16 acsr; |
| |
| if (phy->autoneg) { |
| acsr = phy_read(phy, MII_CIS8201_ACSR); |
| |
| if (acsr & ACSR_DUPLEX_STATUS) |
| phy->duplex = DUPLEX_FULL; |
| else |
| phy->duplex = DUPLEX_HALF; |
| if (acsr & ACSR_SPEED_1000BASET) { |
| phy->speed = SPEED_1000; |
| } else if (acsr & ACSR_SPEED_100BASET) |
| phy->speed = SPEED_100; |
| else |
| phy->speed = SPEED_10; |
| phy->pause = 0; |
| } |
| /* On non-aneg, we assume what we put in BMCR is the speed, |
| * though magic-aneg shouldn't prevent this case from occurring |
| */ |
| |
| return 0; |
| } |
| |
| static int genmii_read_link(struct mii_phy *phy) |
| { |
| u16 lpa; |
| |
| if (phy->autoneg) { |
| lpa = phy_read(phy, MII_LPA) & phy_read(phy, MII_ADVERTISE); |
| |
| phy->speed = SPEED_10; |
| phy->duplex = DUPLEX_HALF; |
| phy->pause = 0; |
| |
| if (lpa & (LPA_100FULL | LPA_100HALF)) { |
| phy->speed = SPEED_100; |
| if (lpa & LPA_100FULL) |
| phy->duplex = DUPLEX_FULL; |
| } else if (lpa & LPA_10FULL) |
| phy->duplex = DUPLEX_FULL; |
| } |
| /* On non-aneg, we assume what we put in BMCR is the speed, |
| * though magic-aneg shouldn't prevent this case from occurring |
| */ |
| |
| return 0; |
| } |
| |
| #define MII_BASIC_FEATURES (SUPPORTED_10baseT_Half | SUPPORTED_10baseT_Full | \ |
| SUPPORTED_100baseT_Half | SUPPORTED_100baseT_Full | \ |
| SUPPORTED_Autoneg | SUPPORTED_TP | SUPPORTED_MII) |
| #define MII_GBIT_FEATURES (MII_BASIC_FEATURES | \ |
| SUPPORTED_1000baseT_Half | SUPPORTED_1000baseT_Full) |
| |
| /* CIS8201 phy ops */ |
| static struct mii_phy_ops cis8201_phy_ops = { |
| init:cis8201_init, |
| setup_aneg:genmii_setup_aneg, |
| setup_forced:genmii_setup_forced, |
| poll_link:genmii_poll_link, |
| read_link:cis8201_read_link |
| }; |
| |
| /* Generic implementation for most 10/100 PHYs */ |
| static struct mii_phy_ops generic_phy_ops = { |
| setup_aneg:genmii_setup_aneg, |
| setup_forced:genmii_setup_forced, |
| poll_link:genmii_poll_link, |
| read_link:genmii_read_link |
| }; |
| |
| static struct mii_phy_def cis8201_phy_def = { |
| phy_id:0x000fc410, |
| phy_id_mask:0x000ffff0, |
| name:"CIS8201 Gigabit Ethernet", |
| features:MII_GBIT_FEATURES, |
| magic_aneg:0, |
| ops:&cis8201_phy_ops |
| }; |
| |
| static struct mii_phy_def genmii_phy_def = { |
| phy_id:0x00000000, |
| phy_id_mask:0x00000000, |
| name:"Generic MII", |
| features:MII_BASIC_FEATURES, |
| magic_aneg:0, |
| ops:&generic_phy_ops |
| }; |
| |
| static struct mii_phy_def *mii_phy_table[] = { |
| &cis8201_phy_def, |
| &genmii_phy_def, |
| NULL |
| }; |
| |
| int mii_phy_probe(struct mii_phy *phy, int mii_id) |
| { |
| int rc; |
| u32 id; |
| struct mii_phy_def *def; |
| int i; |
| |
| phy->autoneg = 0; |
| phy->advertising = 0; |
| phy->mii_id = mii_id; |
| phy->speed = 0; |
| phy->duplex = 0; |
| phy->pause = 0; |
| |
| /* Take PHY out of isloate mode and reset it. */ |
| rc = reset_one_mii_phy(phy, mii_id); |
| if (rc) |
| return -ENODEV; |
| |
| /* Read ID and find matching entry */ |
| id = (phy_read(phy, MII_PHYSID1) << 16 | phy_read(phy, MII_PHYSID2)) |
| & 0xfffffff0; |
| for (i = 0; (def = mii_phy_table[i]) != NULL; i++) |
| if ((id & def->phy_id_mask) == def->phy_id) |
| break; |
| /* Should never be NULL (we have a generic entry), but... */ |
| if (def == NULL) |
| return -ENODEV; |
| |
| phy->def = def; |
| |
| /* Setup default advertising */ |
| phy->advertising = def->features; |
| |
| return 0; |
| } |
| |
| MODULE_LICENSE("GPL"); |