1 /****************************************************************************
 
   2  * Driver for Solarflare Solarstorm network controllers and boards
 
   3  * Copyright 2006-2008 Solarflare Communications Inc.
 
   5  * This program is free software; you can redistribute it and/or modify it
 
   6  * under the terms of the GNU General Public License version 2 as published
 
   7  * by the Free Software Foundation, incorporated herein by reference.
 
  10  * Useful functions for working with MDIO clause 45 PHYs
 
  12 #include <linux/types.h>
 
  13 #include <linux/ethtool.h>
 
  14 #include <linux/delay.h>
 
  15 #include "net_driver.h"
 
  19 int mdio_clause45_reset_mmd(struct efx_nic *port, int mmd,
 
  20                             int spins, int spintime)
 
  23         int phy_id = port->mii.phy_id;
 
  25         /* Catch callers passing values in the wrong units (or just silly) */
 
  26         EFX_BUG_ON_PARANOID(spins * spintime >= 5000);
 
  28         mdio_clause45_write(port, phy_id, mmd, MDIO_MMDREG_CTRL1,
 
  29                             (1 << MDIO_MMDREG_CTRL1_RESET_LBN));
 
  30         /* Wait for the reset bit to clear. */
 
  33                 ctrl = mdio_clause45_read(port, phy_id, mmd, MDIO_MMDREG_CTRL1);
 
  36         } while (spins && (ctrl & (1 << MDIO_MMDREG_CTRL1_RESET_LBN)));
 
  38         return spins ? spins : -ETIMEDOUT;
 
  41 static int mdio_clause45_check_mmd(struct efx_nic *efx, int mmd,
 
  45         int phy_id = efx->mii.phy_id;
 
  47         if (LOOPBACK_INTERNAL(efx))
 
  50         /* Read MMD STATUS2 to check it is responding. */
 
  51         status = mdio_clause45_read(efx, phy_id, mmd, MDIO_MMDREG_STAT2);
 
  52         if (((status >> MDIO_MMDREG_STAT2_PRESENT_LBN) &
 
  53              ((1 << MDIO_MMDREG_STAT2_PRESENT_WIDTH) - 1)) !=
 
  54             MDIO_MMDREG_STAT2_PRESENT_VAL) {
 
  55                 EFX_ERR(efx, "PHY MMD %d not responding.\n", mmd);
 
  59         /* Read MMD STATUS 1 to check for fault. */
 
  60         status = mdio_clause45_read(efx, phy_id, mmd, MDIO_MMDREG_STAT1);
 
  61         if ((status & (1 << MDIO_MMDREG_STAT1_FAULT_LBN)) != 0) {
 
  63                         EFX_ERR(efx, "PHY MMD %d reporting fatal"
 
  64                                 " fault: status %x\n", mmd, status);
 
  67                         EFX_LOG(efx, "PHY MMD %d reporting status"
 
  68                                 " %x (expected)\n", mmd, status);
 
  74 /* This ought to be ridiculous overkill. We expect it to fail rarely */
 
  75 #define MDIO45_RESET_TIME       1000 /* ms */
 
  76 #define MDIO45_RESET_ITERS      100
 
  78 int mdio_clause45_wait_reset_mmds(struct efx_nic *efx,
 
  79                                   unsigned int mmd_mask)
 
  81         const int spintime = MDIO45_RESET_TIME / MDIO45_RESET_ITERS;
 
  82         int tries = MDIO45_RESET_ITERS;
 
  93                                 stat = mdio_clause45_read(efx,
 
  98                                         EFX_ERR(efx, "failed to read status of"
 
 102                                 if (stat & (1 << MDIO_MMDREG_CTRL1_RESET_LBN))
 
 103                                         in_reset |= (1 << mmd);
 
 114                 EFX_ERR(efx, "not all MMDs came out of reset in time."
 
 115                         " MMDs still in reset: %x\n", in_reset);
 
 121 int mdio_clause45_check_mmds(struct efx_nic *efx,
 
 122                              unsigned int mmd_mask, unsigned int fatal_mask)
 
 124         int devices, mmd = 0;
 
 127         /* Historically we have probed the PHYXS to find out what devices are
 
 128          * present,but that doesn't work so well if the PHYXS isn't expected
 
 129          * to exist, if so just find the first item in the list supplied. */
 
 130         probe_mmd = (mmd_mask & MDIO_MMDREG_DEVS0_PHYXS) ? MDIO_MMD_PHYXS :
 
 132         devices = mdio_clause45_read(efx, efx->mii.phy_id,
 
 133                                      probe_mmd, MDIO_MMDREG_DEVS0);
 
 135         /* Check all the expected MMDs are present */
 
 137                 EFX_ERR(efx, "failed to read devices present\n");
 
 140         if ((devices & mmd_mask) != mmd_mask) {
 
 141                 EFX_ERR(efx, "required MMDs not present: got %x, "
 
 142                         "wanted %x\n", devices, mmd_mask);
 
 145         EFX_TRACE(efx, "Devices present: %x\n", devices);
 
 147         /* Check all required MMDs are responding and happy. */
 
 150                         int fault_fatal = fatal_mask & 1;
 
 151                         if (mdio_clause45_check_mmd(efx, mmd, fault_fatal))
 
 154                 mmd_mask = mmd_mask >> 1;
 
 155                 fatal_mask = fatal_mask >> 1;
 
 162 bool mdio_clause45_links_ok(struct efx_nic *efx, unsigned int mmd_mask)
 
 164         int phy_id = efx->mii.phy_id;
 
 169         /* If the port is in loopback, then we should only consider a subset
 
 171         if (LOOPBACK_INTERNAL(efx))
 
 173         else if (efx->loopback_mode == LOOPBACK_NETWORK)
 
 175         else if (efx_phy_mode_disabled(efx->phy_mode))
 
 177         else if (efx->loopback_mode == LOOPBACK_PHYXS)
 
 178                 mmd_mask &= ~(MDIO_MMDREG_DEVS0_PHYXS |
 
 179                               MDIO_MMDREG_DEVS0_PCS |
 
 180                               MDIO_MMDREG_DEVS0_PMAPMD);
 
 181         else if (efx->loopback_mode == LOOPBACK_PCS)
 
 182                 mmd_mask &= ~(MDIO_MMDREG_DEVS0_PCS |
 
 183                               MDIO_MMDREG_DEVS0_PMAPMD);
 
 184         else if (efx->loopback_mode == LOOPBACK_PMAPMD)
 
 185                 mmd_mask &= ~MDIO_MMDREG_DEVS0_PMAPMD;
 
 189                         /* Double reads because link state is latched, and a
 
 190                          * read moves the current state into the register */
 
 191                         status = mdio_clause45_read(efx, phy_id,
 
 192                                                     mmd, MDIO_MMDREG_STAT1);
 
 193                         status = mdio_clause45_read(efx, phy_id,
 
 194                                                     mmd, MDIO_MMDREG_STAT1);
 
 196                         ok = ok && (status & (1 << MDIO_MMDREG_STAT1_LINK_LBN));
 
 198                 mmd_mask = (mmd_mask >> 1);
 
 204 void mdio_clause45_transmit_disable(struct efx_nic *efx)
 
 206         int phy_id = efx->mii.phy_id;
 
 209         ctrl1 = ctrl2 = mdio_clause45_read(efx, phy_id, MDIO_MMD_PMAPMD,
 
 211         if (efx->phy_mode & PHY_MODE_TX_DISABLED)
 
 212                 ctrl2 |= (1 << MDIO_MMDREG_TXDIS_GLOBAL_LBN);
 
 214                 ctrl1 &= ~(1 << MDIO_MMDREG_TXDIS_GLOBAL_LBN);
 
 216                 mdio_clause45_write(efx, phy_id, MDIO_MMD_PMAPMD,
 
 217                                     MDIO_MMDREG_TXDIS, ctrl2);
 
 220 void mdio_clause45_phy_reconfigure(struct efx_nic *efx)
 
 222         int phy_id = efx->mii.phy_id;
 
 225         /* Handle (with debouncing) PMA/PMD loopback */
 
 226         ctrl1 = ctrl2 = mdio_clause45_read(efx, phy_id, MDIO_MMD_PMAPMD,
 
 229         if (efx->loopback_mode == LOOPBACK_PMAPMD)
 
 230                 ctrl2 |= (1 << MDIO_PMAPMD_CTRL1_LBACK_LBN);
 
 232                 ctrl2 &= ~(1 << MDIO_PMAPMD_CTRL1_LBACK_LBN);
 
 235                 mdio_clause45_write(efx, phy_id, MDIO_MMD_PMAPMD,
 
 236                                     MDIO_MMDREG_CTRL1, ctrl2);
 
 238         /* Handle (with debouncing) PCS loopback */
 
 239         ctrl1 = ctrl2 = mdio_clause45_read(efx, phy_id, MDIO_MMD_PCS,
 
 241         if (efx->loopback_mode == LOOPBACK_PCS)
 
 242                 ctrl2 |= (1 << MDIO_MMDREG_CTRL1_LBACK_LBN);
 
 244                 ctrl2 &= ~(1 << MDIO_MMDREG_CTRL1_LBACK_LBN);
 
 247                 mdio_clause45_write(efx, phy_id, MDIO_MMD_PCS,
 
 248                                     MDIO_MMDREG_CTRL1, ctrl2);
 
 250         /* Handle (with debouncing) PHYXS network loopback */
 
 251         ctrl1 = ctrl2 = mdio_clause45_read(efx, phy_id, MDIO_MMD_PHYXS,
 
 253         if (efx->loopback_mode == LOOPBACK_NETWORK)
 
 254                 ctrl2 |= (1 << MDIO_MMDREG_CTRL1_LBACK_LBN);
 
 256                 ctrl2 &= ~(1 << MDIO_MMDREG_CTRL1_LBACK_LBN);
 
 259                 mdio_clause45_write(efx, phy_id, MDIO_MMD_PHYXS,
 
 260                                     MDIO_MMDREG_CTRL1, ctrl2);
 
 263 static void mdio_clause45_set_mmd_lpower(struct efx_nic *efx,
 
 266         int phy = efx->mii.phy_id;
 
 267         int stat = mdio_clause45_read(efx, phy, mmd, MDIO_MMDREG_STAT1);
 
 270         EFX_TRACE(efx, "Setting low power mode for MMD %d to %d\n",
 
 273         if (stat & (1 << MDIO_MMDREG_STAT1_LPABLE_LBN)) {
 
 274                 ctrl1 = ctrl2 = mdio_clause45_read(efx, phy,
 
 275                                                    mmd, MDIO_MMDREG_CTRL1);
 
 277                         ctrl2 |= (1 << MDIO_MMDREG_CTRL1_LPOWER_LBN);
 
 279                         ctrl2 &= ~(1 << MDIO_MMDREG_CTRL1_LPOWER_LBN);
 
 281                         mdio_clause45_write(efx, phy, mmd,
 
 282                                             MDIO_MMDREG_CTRL1, ctrl2);
 
 286 void mdio_clause45_set_mmds_lpower(struct efx_nic *efx,
 
 287                                    int low_power, unsigned int mmd_mask)
 
 292                         mdio_clause45_set_mmd_lpower(efx, low_power, mmd);
 
 293                 mmd_mask = (mmd_mask >> 1);
 
 299  * mdio_clause45_get_settings - Read (some of) the PHY settings over MDIO.
 
 301  * @ecmd:               Buffer for settings
 
 303  * On return the 'port', 'speed', 'supported' and 'advertising' fields of
 
 304  * ecmd have been filled out based on the PMA type.
 
 306 void mdio_clause45_get_settings(struct efx_nic *efx,
 
 307                                 struct ethtool_cmd *ecmd)
 
 311         /* If no PMA is present we are presumably talking something XAUI-ish
 
 312          * like CX4. Which we report as FIBRE (see below) */
 
 313         if ((efx->phy_op->mmds & DEV_PRESENT_BIT(MDIO_MMD_PMAPMD)) == 0) {
 
 314                 ecmd->speed = SPEED_10000;
 
 315                 ecmd->port = PORT_FIBRE;
 
 316                 ecmd->supported = SUPPORTED_FIBRE;
 
 317                 ecmd->advertising = ADVERTISED_FIBRE;
 
 321         pma_type = mdio_clause45_read(efx, efx->mii.phy_id,
 
 322                                       MDIO_MMD_PMAPMD, MDIO_MMDREG_CTRL2);
 
 323         pma_type &= MDIO_PMAPMD_CTRL2_TYPE_MASK;
 
 326                 /* We represent CX4 as fibre in the absence of anything
 
 328         case MDIO_PMAPMD_CTRL2_10G_CX4:
 
 329                 ecmd->speed = SPEED_10000;
 
 330                 ecmd->port = PORT_FIBRE;
 
 331                 ecmd->supported = SUPPORTED_FIBRE;
 
 332                 ecmd->advertising = ADVERTISED_FIBRE;
 
 335         case MDIO_PMAPMD_CTRL2_10G_BT:
 
 336                 ecmd->speed = SPEED_10000;
 
 337                 ecmd->port = PORT_TP;
 
 338                 ecmd->supported = SUPPORTED_TP | SUPPORTED_10000baseT_Full;
 
 339                 ecmd->advertising = (ADVERTISED_FIBRE
 
 340                                      | ADVERTISED_10000baseT_Full);
 
 342         case MDIO_PMAPMD_CTRL2_1G_BT:
 
 343                 ecmd->speed = SPEED_1000;
 
 344                 ecmd->port = PORT_TP;
 
 345                 ecmd->supported = SUPPORTED_TP | SUPPORTED_1000baseT_Full;
 
 346                 ecmd->advertising = (ADVERTISED_FIBRE
 
 347                                      | ADVERTISED_1000baseT_Full);
 
 349         case MDIO_PMAPMD_CTRL2_100_BT:
 
 350                 ecmd->speed = SPEED_100;
 
 351                 ecmd->port = PORT_TP;
 
 352                 ecmd->supported = SUPPORTED_TP | SUPPORTED_100baseT_Full;
 
 353                 ecmd->advertising = (ADVERTISED_FIBRE
 
 354                                      | ADVERTISED_100baseT_Full);
 
 356         case MDIO_PMAPMD_CTRL2_10_BT:
 
 357                 ecmd->speed = SPEED_10;
 
 358                 ecmd->port = PORT_TP;
 
 359                 ecmd->supported = SUPPORTED_TP | SUPPORTED_10baseT_Full;
 
 360                 ecmd->advertising = ADVERTISED_FIBRE | ADVERTISED_10baseT_Full;
 
 362         /* All the other defined modes are flavours of
 
 365                 ecmd->speed = SPEED_10000;
 
 366                 ecmd->port = PORT_FIBRE;
 
 367                 ecmd->supported = SUPPORTED_FIBRE;
 
 368                 ecmd->advertising = ADVERTISED_FIBRE;
 
 374  * mdio_clause45_set_settings - Set (some of) the PHY settings over MDIO.
 
 376  * @ecmd:               New settings
 
 378  * Currently this just enforces that we are _not_ changing the
 
 379  * 'port', 'speed', 'supported' or 'advertising' settings as these
 
 380  * cannot be changed on any currently supported PHY.
 
 382 int mdio_clause45_set_settings(struct efx_nic *efx,
 
 383                                struct ethtool_cmd *ecmd)
 
 385         struct ethtool_cmd tmpcmd;
 
 386         mdio_clause45_get_settings(efx, &tmpcmd);
 
 387         /* None of the current PHYs support more than one mode
 
 388          * of operation (and only 10GBT ever will), so keep things
 
 390         if ((ecmd->speed == tmpcmd.speed) && (ecmd->port == tmpcmd.port) &&
 
 391             (ecmd->supported == tmpcmd.supported) &&
 
 392             (ecmd->advertising == tmpcmd.advertising))