lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <CANEJEGsU97GQ9SzmXGJjdSd_fbTptttHJZb1g_GvNjwEgEzqHg@mail.gmail.com>
Date:	Mon, 25 Jul 2016 14:26:43 -0700
From:	Grant Grundler <grundler@...omium.org>
To:	robert.foss@...labora.com
Cc:	David Miller <davem@...emloft.net>,
	Grant Grundler <grundler@...omium.org>, Mark_Craske@...tor.com,
	Dean_Jenkins@...tor.com, Vincent Palatin <vpalatin@...omium.org>,
	ivecera@...hat.com, linux@...idrobins.net,
	WK TSAI <wk.tsai@...dia.com>, changchias@...il.com,
	"linux-usb@...r.kernel.org" <linux-usb@...r.kernel.org>,
	netdev <netdev@...r.kernel.org>,
	LKML <linux-kernel@...r.kernel.org>,
	enric.balletbo@...labora.com, tomeu.vizoso@...labora.com
Subject: Re: [PATCH 1/3] net: asix: Add in_pm parameter

[as plain text this time...]

Robert,

On Mon, Jul 25, 2016 at 10:40 AM,  <robert.foss@...labora.com> wrote:
> From: Grant Grundler <grundler@...omium.org>

For the record, I believe I am not the author of these patches.

I believe the original author is
    Signed-off-by: Freddy Xin <freddy@...x.com.tw>

as recorded in the following code reviews (and testing) that I was
responsible for:
    https://chromium-review.googlesource.com/#/q/owner:%22Grant+Grundler%22+status:merged+asix+in_pm

And I would certainly be happy to see this code go upstream and
expected ASIX would submit this change upstream.

> In order to R/W registers in suspend/resume functions, in_pm flags are
> added to some functions to determine whether the nopm version of usb
> functions is called.

FTR, current  drivers/net/usb/ax88179_178a.c uses "in_pm" as well.

> Save BMCR and ANAR PHY registers in suspend function and restore them
> in resume function.
>
> Reset HW in resume function to ensure the PHY works correctly.
>
> Signed-off-by: Grant Grundler <grundler@...omium.org>

BTW, I have two additional changes for AX88772x support sitting in my
"needs more work" queue (for quite a while already):
   https://chromium-review.googlesource.com/#/c/229620/
   "asix: autoneg will set WRITE_MEDIUM reg"

   https://chromium-review.googlesource.com/#/c/231162/
   "net: asix: see 802.3 spec for phy reset"

I would certainly approve if _anyone_ picked these up, tested them,
and then submitted them to netdev.

cheers,
grant

> ---
>  drivers/net/usb/asix.h         |  40 +++--
>  drivers/net/usb/asix_common.c  | 180 +++++++++++++++-----
>  drivers/net/usb/asix_devices.c | 373 ++++++++++++++++++++++++++++++++---------
>  drivers/net/usb/ax88172a.c     |  29 ++--
>  4 files changed, 472 insertions(+), 150 deletions(-)
>
> diff --git a/drivers/net/usb/asix.h b/drivers/net/usb/asix.h
> index a2d3ea6..d109242 100644
> --- a/drivers/net/usb/asix.h
> +++ b/drivers/net/usb/asix.h
> @@ -46,6 +46,7 @@
>  #define AX_CMD_SET_SW_MII              0x06
>  #define AX_CMD_READ_MII_REG            0x07
>  #define AX_CMD_WRITE_MII_REG           0x08
> +#define AX_CMD_STATMNGSTS_REG          0x09
>  #define AX_CMD_SET_HW_MII              0x0a
>  #define AX_CMD_READ_EEPROM             0x0b
>  #define AX_CMD_WRITE_EEPROM            0x0c
> @@ -71,6 +72,17 @@
>  #define AX_CMD_SW_RESET                        0x20
>  #define AX_CMD_SW_PHY_STATUS           0x21
>  #define AX_CMD_SW_PHY_SELECT           0x22
> +#define AX_QCTCTRL                     0x2A
> +
> +#define AX_CHIPCODE_MASK               0x70
> +#define AX_AX88772_CHIPCODE            0x00
> +#define AX_AX88772A_CHIPCODE           0x10
> +#define AX_AX88772B_CHIPCODE           0x20
> +#define AX_HOST_EN                     0x01
> +
> +#define AX_PHYSEL_PSEL                 0x01
> +#define AX_PHYSEL_SSMII                        0
> +#define AX_PHYSEL_SSEN                 0x10
>
>  #define AX_PHY_SELECT_MASK             (BIT(3) | BIT(2))
>  #define AX_PHY_SELECT_INTERNAL         0
> @@ -173,6 +185,10 @@ struct asix_rx_fixup_info {
>  };
>
>  struct asix_common_private {
> +       void (*resume)(struct usbnet *dev);
> +       void (*suspend)(struct usbnet *dev);
> +       u16 presvd_phy_advertise;
> +       u16 presvd_phy_bmcr;
>         struct asix_rx_fixup_info rx_fixup_info;
>  };
>
> @@ -182,10 +198,10 @@ extern const struct driver_info ax88172a_info;
>  #define FLAG_EEPROM_MAC                (1UL << 0)  /* init device MAC from eeprom */
>
>  int asix_read_cmd(struct usbnet *dev, u8 cmd, u16 value, u16 index,
> -                 u16 size, void *data);
> +                 u16 size, void *data, int in_pm);
>
>  int asix_write_cmd(struct usbnet *dev, u8 cmd, u16 value, u16 index,
> -                  u16 size, void *data);
> +                  u16 size, void *data, int in_pm);
>
>  void asix_write_cmd_async(struct usbnet *dev, u8 cmd, u16 value,
>                           u16 index, u16 size, void *data);
> @@ -197,27 +213,31 @@ int asix_rx_fixup_common(struct usbnet *dev, struct sk_buff *skb);
>  struct sk_buff *asix_tx_fixup(struct usbnet *dev, struct sk_buff *skb,
>                               gfp_t flags);
>
> -int asix_set_sw_mii(struct usbnet *dev);
> -int asix_set_hw_mii(struct usbnet *dev);
> +int asix_set_sw_mii(struct usbnet *dev, int in_pm);
> +int asix_set_hw_mii(struct usbnet *dev, int in_pm);
>
>  int asix_read_phy_addr(struct usbnet *dev, int internal);
>  int asix_get_phy_addr(struct usbnet *dev);
>
> -int asix_sw_reset(struct usbnet *dev, u8 flags);
> +int asix_sw_reset(struct usbnet *dev, u8 flags, int in_pm);
>
> -u16 asix_read_rx_ctl(struct usbnet *dev);
> -int asix_write_rx_ctl(struct usbnet *dev, u16 mode);
> +u16 asix_read_rx_ctl(struct usbnet *dev, int in_pm);
> +int asix_write_rx_ctl(struct usbnet *dev, u16 mode, int in_pm);
>
> -u16 asix_read_medium_status(struct usbnet *dev);
> -int asix_write_medium_mode(struct usbnet *dev, u16 mode);
> +u16 asix_read_medium_status(struct usbnet *dev, int in_pm);
> +int asix_write_medium_mode(struct usbnet *dev, u16 mode, int in_pm);
>
> -int asix_write_gpio(struct usbnet *dev, u16 value, int sleep);
> +int asix_write_gpio(struct usbnet *dev, u16 value, int sleep, int in_pm);
>
>  void asix_set_multicast(struct net_device *net);
>
>  int asix_mdio_read(struct net_device *netdev, int phy_id, int loc);
>  void asix_mdio_write(struct net_device *netdev, int phy_id, int loc, int val);
>
> +int asix_mdio_read_nopm(struct net_device *netdev, int phy_id, int loc);
> +void asix_mdio_write_nopm(struct net_device *netdev, int phy_id, int loc,
> +                         int val);
> +
>  void asix_get_wol(struct net_device *net, struct ethtool_wolinfo *wolinfo);
>  int asix_set_wol(struct net_device *net, struct ethtool_wolinfo *wolinfo);
>
> diff --git a/drivers/net/usb/asix_common.c b/drivers/net/usb/asix_common.c
> index bd9acff..f0ccf76 100644
> --- a/drivers/net/usb/asix_common.c
> +++ b/drivers/net/usb/asix_common.c
> @@ -22,24 +22,49 @@
>  #include "asix.h"
>
>  int asix_read_cmd(struct usbnet *dev, u8 cmd, u16 value, u16 index,
> -                 u16 size, void *data)
> +                 u16 size, void *data, int in_pm)
>  {
>         int ret;
> -       ret = usbnet_read_cmd(dev, cmd,
> -                              USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
> -                              value, index, data, size);
> +       int (*fn)(struct usbnet *, u8, u8, u16, u16, void *, u16);
> +
> +       BUG_ON(!dev);
> +
> +       if (!in_pm)
> +               fn = usbnet_read_cmd;
> +       else
> +               fn = usbnet_read_cmd_nopm;
> +
> +       ret = fn(dev, cmd, USB_DIR_IN | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
> +                value, index, data, size);
> +
> +       if (unlikely(ret < 0))
> +               netdev_warn(dev->net, "Failed to read reg index 0x%04x: %d\n",
> +                           index, ret);
>
> -       if (ret != size && ret >= 0)
> -               return -EINVAL;
>         return ret;
>  }
>
>  int asix_write_cmd(struct usbnet *dev, u8 cmd, u16 value, u16 index,
> -                  u16 size, void *data)
> +                  u16 size, void *data, int in_pm)
>  {
> -       return usbnet_write_cmd(dev, cmd,
> -                               USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
> -                               value, index, data, size);
> +       int ret;
> +       int (*fn)(struct usbnet *, u8, u8, u16, u16, const void *, u16);
> +
> +       BUG_ON(!dev);
> +
> +       if (!in_pm)
> +               fn = usbnet_write_cmd;
> +       else
> +               fn = usbnet_write_cmd_nopm;
> +
> +       ret = fn(dev, cmd, USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE,
> +                value, index, data, size);
> +
> +       if (unlikely(ret < 0))
> +               netdev_warn(dev->net, "Failed to write reg index 0x%04x: %d\n",
> +                           index, ret);
> +
> +       return ret;
>  }
>
>  void asix_write_cmd_async(struct usbnet *dev, u8 cmd, u16 value, u16 index,
> @@ -225,19 +250,20 @@ struct sk_buff *asix_tx_fixup(struct usbnet *dev, struct sk_buff *skb,
>         return skb;
>  }
>
> -int asix_set_sw_mii(struct usbnet *dev)
> +int asix_set_sw_mii(struct usbnet *dev, int in_pm)
>  {
>         int ret;
> -       ret = asix_write_cmd(dev, AX_CMD_SET_SW_MII, 0x0000, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_SET_SW_MII, 0x0000, 0, 0, NULL, in_pm);
> +
>         if (ret < 0)
>                 netdev_err(dev->net, "Failed to enable software MII access\n");
>         return ret;
>  }
>
> -int asix_set_hw_mii(struct usbnet *dev)
> +int asix_set_hw_mii(struct usbnet *dev, int in_pm)
>  {
>         int ret;
> -       ret = asix_write_cmd(dev, AX_CMD_SET_HW_MII, 0x0000, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_SET_HW_MII, 0x0000, 0, 0, NULL, in_pm);
>         if (ret < 0)
>                 netdev_err(dev->net, "Failed to enable hardware MII access\n");
>         return ret;
> @@ -247,7 +273,7 @@ int asix_read_phy_addr(struct usbnet *dev, int internal)
>  {
>         int offset = (internal ? 1 : 0);
>         u8 buf[2];
> -       int ret = asix_read_cmd(dev, AX_CMD_READ_PHY_ID, 0, 0, 2, buf);
> +       int ret = asix_read_cmd(dev, AX_CMD_READ_PHY_ID, 0, 0, 2, buf, 0);
>
>         netdev_dbg(dev->net, "asix_get_phy_addr()\n");
>
> @@ -270,21 +296,21 @@ int asix_get_phy_addr(struct usbnet *dev)
>  }
>
>
> -int asix_sw_reset(struct usbnet *dev, u8 flags)
> +int asix_sw_reset(struct usbnet *dev, u8 flags, int in_pm)
>  {
>         int ret;
>
> -        ret = asix_write_cmd(dev, AX_CMD_SW_RESET, flags, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_SW_RESET, flags, 0, 0, NULL, in_pm);
>         if (ret < 0)
>                 netdev_err(dev->net, "Failed to send software reset: %02x\n", ret);
>
>         return ret;
>  }
>
> -u16 asix_read_rx_ctl(struct usbnet *dev)
> +u16 asix_read_rx_ctl(struct usbnet *dev, int in_pm)
>  {
>         __le16 v;
> -       int ret = asix_read_cmd(dev, AX_CMD_READ_RX_CTL, 0, 0, 2, &v);
> +       int ret = asix_read_cmd(dev, AX_CMD_READ_RX_CTL, 0, 0, 2, &v, in_pm);
>
>         if (ret < 0) {
>                 netdev_err(dev->net, "Error reading RX_CTL register: %02x\n", ret);
> @@ -295,12 +321,12 @@ out:
>         return ret;
>  }
>
> -int asix_write_rx_ctl(struct usbnet *dev, u16 mode)
> +int asix_write_rx_ctl(struct usbnet *dev, u16 mode, int in_pm)
>  {
>         int ret;
>
>         netdev_dbg(dev->net, "asix_write_rx_ctl() - mode = 0x%04x\n", mode);
> -       ret = asix_write_cmd(dev, AX_CMD_WRITE_RX_CTL, mode, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_RX_CTL, mode, 0, 0, NULL, in_pm);
>         if (ret < 0)
>                 netdev_err(dev->net, "Failed to write RX_CTL mode to 0x%04x: %02x\n",
>                            mode, ret);
> @@ -308,10 +334,11 @@ int asix_write_rx_ctl(struct usbnet *dev, u16 mode)
>         return ret;
>  }
>
> -u16 asix_read_medium_status(struct usbnet *dev)
> +u16 asix_read_medium_status(struct usbnet *dev, int in_pm)
>  {
>         __le16 v;
> -       int ret = asix_read_cmd(dev, AX_CMD_READ_MEDIUM_STATUS, 0, 0, 2, &v);
> +       int ret = asix_read_cmd(dev, AX_CMD_READ_MEDIUM_STATUS,
> +                               0, 0, 2, &v, in_pm);
>
>         if (ret < 0) {
>                 netdev_err(dev->net, "Error reading Medium Status register: %02x\n",
> @@ -323,12 +350,13 @@ u16 asix_read_medium_status(struct usbnet *dev)
>
>  }
>
> -int asix_write_medium_mode(struct usbnet *dev, u16 mode)
> +int asix_write_medium_mode(struct usbnet *dev, u16 mode, int in_pm)
>  {
>         int ret;
>
>         netdev_dbg(dev->net, "asix_write_medium_mode() - mode = 0x%04x\n", mode);
> -       ret = asix_write_cmd(dev, AX_CMD_WRITE_MEDIUM_MODE, mode, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_MEDIUM_MODE,
> +                            mode, 0, 0, NULL, in_pm);
>         if (ret < 0)
>                 netdev_err(dev->net, "Failed to write Medium Mode mode to 0x%04x: %02x\n",
>                            mode, ret);
> @@ -336,12 +364,12 @@ int asix_write_medium_mode(struct usbnet *dev, u16 mode)
>         return ret;
>  }
>
> -int asix_write_gpio(struct usbnet *dev, u16 value, int sleep)
> +int asix_write_gpio(struct usbnet *dev, u16 value, int sleep, int in_pm)
>  {
>         int ret;
>
>         netdev_dbg(dev->net, "asix_write_gpio() - value = 0x%04x\n", value);
> -       ret = asix_write_cmd(dev, AX_CMD_WRITE_GPIOS, value, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_GPIOS, value, 0, 0, NULL, in_pm);
>         if (ret < 0)
>                 netdev_err(dev->net, "Failed to write GPIO value 0x%04x: %02x\n",
>                            value, ret);
> @@ -398,16 +426,23 @@ int asix_mdio_read(struct net_device *netdev, int phy_id, int loc)
>  {
>         struct usbnet *dev = netdev_priv(netdev);
>         __le16 res;
> +       u8 smsr;
> +       int i = 0;
>
>         mutex_lock(&dev->phy_mutex);
> -       asix_set_sw_mii(dev);
> +       do {
> +               asix_set_sw_mii(dev, 0);
> +               usleep_range(1000, 1100);
> +               asix_read_cmd(dev, AX_CMD_STATMNGSTS_REG, 0, 0, 1, &smsr, 0);
> +       } while (!(smsr & AX_HOST_EN) && (i++ < 30));
> +
>         asix_read_cmd(dev, AX_CMD_READ_MII_REG, phy_id,
> -                               (__u16)loc, 2, &res);
> -       asix_set_hw_mii(dev);
> +                               (__u16)loc, 2, &res, 0);
> +       asix_set_hw_mii(dev, 0);
>         mutex_unlock(&dev->phy_mutex);
>
>         netdev_dbg(dev->net, "asix_mdio_read() phy_id=0x%02x, loc=0x%02x, returns=0x%04x\n",
> -                  phy_id, loc, le16_to_cpu(res));
> +                       phy_id, loc, le16_to_cpu(res));
>
>         return le16_to_cpu(res);
>  }
> @@ -416,13 +451,71 @@ void asix_mdio_write(struct net_device *netdev, int phy_id, int loc, int val)
>  {
>         struct usbnet *dev = netdev_priv(netdev);
>         __le16 res = cpu_to_le16(val);
> +       u8 smsr;
> +       int i = 0;
>
>         netdev_dbg(dev->net, "asix_mdio_write() phy_id=0x%02x, loc=0x%02x, val=0x%04x\n",
> -                  phy_id, loc, val);
> +                       phy_id, loc, val);
> +
> +       mutex_lock(&dev->phy_mutex);
> +       do {
> +               asix_set_sw_mii(dev, 0);
> +               usleep_range(1000, 1100);
> +               asix_read_cmd(dev, AX_CMD_STATMNGSTS_REG, 0, 0, 1, &smsr, 0);
> +       } while (!(smsr & AX_HOST_EN) && (i++ < 30));
> +
> +       asix_write_cmd(dev, AX_CMD_WRITE_MII_REG, phy_id,
> +                      (__u16)loc, 2, &res, 0);
> +       asix_set_hw_mii(dev, 0);
> +       mutex_unlock(&dev->phy_mutex);
> +}
> +
> +int asix_mdio_read_nopm(struct net_device *netdev, int phy_id, int loc)
> +{
> +       struct usbnet *dev = netdev_priv(netdev);
> +       __le16 res;
> +       u8 smsr;
> +       int i = 0;
> +
> +       mutex_lock(&dev->phy_mutex);
> +       do {
> +               asix_set_sw_mii(dev, 1);
> +               usleep_range(1000, 1100);
> +               asix_read_cmd(dev, AX_CMD_STATMNGSTS_REG, 0, 0, 1, &smsr, 1);
> +       } while (!(smsr & AX_HOST_EN) && (i++ < 30));
> +
> +       asix_read_cmd(dev, AX_CMD_READ_MII_REG, phy_id,
> +                     (__u16)loc, 2, &res, 1);
> +       asix_set_hw_mii(dev, 1);
> +       mutex_unlock(&dev->phy_mutex);
> +
> +       netdev_dbg(dev->net, "asix_mdio_read_nopm() phy_id=0x%02x, loc=0x%02x, returns=0x%04x\n",
> +                       phy_id, loc, le16_to_cpu(res));
> +
> +       return le16_to_cpu(res);
> +}
> +
> +void
> +asix_mdio_write_nopm(struct net_device *netdev, int phy_id, int loc, int val)
> +{
> +       struct usbnet *dev = netdev_priv(netdev);
> +       __le16 res = cpu_to_le16(val);
> +       u8 smsr;
> +       int i = 0;
> +
> +       netdev_dbg(dev->net, "asix_mdio_write() phy_id=0x%02x, loc=0x%02x, val=0x%04x\n",
> +                       phy_id, loc, val);
> +
>         mutex_lock(&dev->phy_mutex);
> -       asix_set_sw_mii(dev);
> -       asix_write_cmd(dev, AX_CMD_WRITE_MII_REG, phy_id, (__u16)loc, 2, &res);
> -       asix_set_hw_mii(dev);
> +       do {
> +               asix_set_sw_mii(dev, 1);
> +               usleep_range(1000, 1100);
> +               asix_read_cmd(dev, AX_CMD_STATMNGSTS_REG, 0, 0, 1, &smsr, 1);
> +       } while (!(smsr & AX_HOST_EN) && (i++ < 30));
> +
> +       asix_write_cmd(dev, AX_CMD_WRITE_MII_REG, phy_id,
> +                      (__u16)loc, 2, &res, 1);
> +       asix_set_hw_mii(dev, 1);
>         mutex_unlock(&dev->phy_mutex);
>  }
>
> @@ -431,7 +524,8 @@ void asix_get_wol(struct net_device *net, struct ethtool_wolinfo *wolinfo)
>         struct usbnet *dev = netdev_priv(net);
>         u8 opt;
>
> -       if (asix_read_cmd(dev, AX_CMD_READ_MONITOR_MODE, 0, 0, 1, &opt) < 0) {
> +       if (asix_read_cmd(dev, AX_CMD_READ_MONITOR_MODE,
> +                         0, 0, 1, &opt, 0) < 0) {
>                 wolinfo->supported = 0;
>                 wolinfo->wolopts = 0;
>                 return;
> @@ -455,7 +549,7 @@ int asix_set_wol(struct net_device *net, struct ethtool_wolinfo *wolinfo)
>                 opt |= AX_MONITOR_MAGIC;
>
>         if (asix_write_cmd(dev, AX_CMD_WRITE_MONITOR_MODE,
> -                             opt, 0, 0, NULL) < 0)
> +                             opt, 0, 0, NULL, 0) < 0)
>                 return -EINVAL;
>
>         return 0;
> @@ -490,7 +584,7 @@ int asix_get_eeprom(struct net_device *net, struct ethtool_eeprom *eeprom,
>         /* ax8817x returns 2 bytes from eeprom on read */
>         for (i = first_word; i <= last_word; i++) {
>                 if (asix_read_cmd(dev, AX_CMD_READ_EEPROM, i, 0, 2,
> -                                 &(eeprom_buff[i - first_word])) < 0) {
> +                                 &eeprom_buff[i - first_word], 0) < 0) {
>                         kfree(eeprom_buff);
>                         return -EIO;
>                 }
> @@ -531,7 +625,7 @@ int asix_set_eeprom(struct net_device *net, struct ethtool_eeprom *eeprom,
>            the EEPROM */
>         if (eeprom->offset & 1) {
>                 ret = asix_read_cmd(dev, AX_CMD_READ_EEPROM, first_word, 0, 2,
> -                                   &(eeprom_buff[0]));
> +                                   &eeprom_buff[0], 0);
>                 if (ret < 0) {
>                         netdev_err(net, "Failed to read EEPROM at offset 0x%02x.\n", first_word);
>                         goto free;
> @@ -540,7 +634,7 @@ int asix_set_eeprom(struct net_device *net, struct ethtool_eeprom *eeprom,
>
>         if ((eeprom->offset + eeprom->len) & 1) {
>                 ret = asix_read_cmd(dev, AX_CMD_READ_EEPROM, last_word, 0, 2,
> -                                   &(eeprom_buff[last_word - first_word]));
> +                                   &eeprom_buff[last_word - first_word], 0);
>                 if (ret < 0) {
>                         netdev_err(net, "Failed to read EEPROM at offset 0x%02x.\n", last_word);
>                         goto free;
> @@ -550,7 +644,7 @@ int asix_set_eeprom(struct net_device *net, struct ethtool_eeprom *eeprom,
>         memcpy((u8 *)eeprom_buff + (eeprom->offset & 1), data, eeprom->len);
>
>         /* write data to EEPROM */
> -       ret = asix_write_cmd(dev, AX_CMD_WRITE_ENABLE, 0x0000, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_ENABLE, 0x0000, 0, 0, NULL, 0);
>         if (ret < 0) {
>                 netdev_err(net, "Failed to enable EEPROM write\n");
>                 goto free;
> @@ -561,7 +655,7 @@ int asix_set_eeprom(struct net_device *net, struct ethtool_eeprom *eeprom,
>                 netdev_dbg(net, "write to EEPROM at offset 0x%02x, data 0x%04x\n",
>                            i, eeprom_buff[i - first_word]);
>                 ret = asix_write_cmd(dev, AX_CMD_WRITE_EEPROM, i,
> -                                    eeprom_buff[i - first_word], 0, NULL);
> +                                    eeprom_buff[i - first_word], 0, NULL, 0);
>                 if (ret < 0) {
>                         netdev_err(net, "Failed to write EEPROM at offset 0x%02x.\n",
>                                    i);
> @@ -570,7 +664,7 @@ int asix_set_eeprom(struct net_device *net, struct ethtool_eeprom *eeprom,
>                 msleep(20);
>         }
>
> -       ret = asix_write_cmd(dev, AX_CMD_WRITE_DISABLE, 0x0000, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_DISABLE, 0x0000, 0, 0, NULL, 0);
>         if (ret < 0) {
>                 netdev_err(net, "Failed to disable EEPROM write\n");
>                 goto free;
> diff --git a/drivers/net/usb/asix_devices.c b/drivers/net/usb/asix_devices.c
> index 5cabefc..aaa4290 100644
> --- a/drivers/net/usb/asix_devices.c
> +++ b/drivers/net/usb/asix_devices.c
> @@ -184,7 +184,7 @@ static int ax88172_link_reset(struct usbnet *dev)
>         netdev_dbg(dev->net, "ax88172_link_reset() speed: %u duplex: %d setting mode to 0x%04x\n",
>                    ethtool_cmd_speed(&ecmd), ecmd.duplex, mode);
>
> -       asix_write_medium_mode(dev, mode);
> +       asix_write_medium_mode(dev, mode, 0);
>
>         return 0;
>  }
> @@ -213,18 +213,19 @@ static int ax88172_bind(struct usbnet *dev, struct usb_interface *intf)
>         /* Toggle the GPIOs in a manufacturer/model specific way */
>         for (i = 2; i >= 0; i--) {
>                 ret = asix_write_cmd(dev, AX_CMD_WRITE_GPIOS,
> -                               (gpio_bits >> (i * 8)) & 0xff, 0, 0, NULL);
> +                               (gpio_bits >> (i * 8)) & 0xff, 0, 0, NULL, 0);
>                 if (ret < 0)
>                         goto out;
>                 msleep(5);
>         }
>
> -       ret = asix_write_rx_ctl(dev, 0x80);
> +       ret = asix_write_rx_ctl(dev, 0x80, 0);
>         if (ret < 0)
>                 goto out;
>
>         /* Get the MAC address */
> -       ret = asix_read_cmd(dev, AX88172_CMD_READ_NODE_ID, 0, 0, ETH_ALEN, buf);
> +       ret = asix_read_cmd(dev, AX88172_CMD_READ_NODE_ID,
> +                           0, 0, ETH_ALEN, buf, 0);
>         if (ret < 0) {
>                 netdev_dbg(dev->net, "read AX_CMD_READ_NODE_ID failed: %d\n",
>                            ret);
> @@ -290,7 +291,7 @@ static int ax88772_link_reset(struct usbnet *dev)
>         netdev_dbg(dev->net, "ax88772_link_reset() speed: %u duplex: %d setting mode to 0x%04x\n",
>                    ethtool_cmd_speed(&ecmd), ecmd.duplex, mode);
>
> -       asix_write_medium_mode(dev, mode);
> +       asix_write_medium_mode(dev, mode, 0);
>
>         return 0;
>  }
> @@ -298,78 +299,192 @@ static int ax88772_link_reset(struct usbnet *dev)
>  static int ax88772_reset(struct usbnet *dev)
>  {
>         struct asix_data *data = (struct asix_data *)&dev->data;
> +       int ret;
> +
> +       /* Rewrite MAC address */
> +       ether_addr_copy(data->mac_addr, dev->net->dev_addr);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0,
> +                            ETH_ALEN, data->mac_addr, 0);
> +       if (ret < 0)
> +               goto out;
> +
> +       /* Set RX_CTL to default values with 2k buffer, and enable cactus */
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, 0);
> +       if (ret < 0)
> +               goto out;
> +
> +       asix_write_medium_mode(dev, AX88772_MEDIUM_DEFAULT, 0);
> +       if (ret < 0)
> +               goto out;
> +
> +       return 0;
> +
> +out:
> +       return ret;
> +}
> +
> +static int ax88772_hw_reset(struct usbnet *dev, int in_pm)
> +{
> +       struct asix_data *data = (struct asix_data *)&dev->data;
>         int ret, embd_phy;
>         u16 rx_ctl;
>
> -       ret = asix_write_gpio(dev,
> -                       AX_GPIO_RSE | AX_GPIO_GPO_2 | AX_GPIO_GPO2EN, 5);
> +       ret = asix_write_gpio(dev, AX_GPIO_RSE | AX_GPIO_GPO_2 |
> +                             AX_GPIO_GPO2EN, 5, in_pm);
>         if (ret < 0)
>                 goto out;
>
> -       embd_phy = ((asix_get_phy_addr(dev) & 0x1f) == 0x10 ? 1 : 0);
> +       embd_phy = ((dev->mii.phy_id & 0x1f) == 0x10 ? 1 : 0);
>
> -       ret = asix_write_cmd(dev, AX_CMD_SW_PHY_SELECT, embd_phy, 0, 0, NULL);
> +       ret = asix_write_cmd(dev, AX_CMD_SW_PHY_SELECT, embd_phy,
> +                            0, 0, NULL, in_pm);
>         if (ret < 0) {
>                 netdev_dbg(dev->net, "Select PHY #1 failed: %d\n", ret);
>                 goto out;
>         }
>
> -       ret = asix_sw_reset(dev, AX_SWRESET_IPPD | AX_SWRESET_PRL);
> -       if (ret < 0)
> -               goto out;
> +       if (embd_phy) {
> +               ret = asix_sw_reset(dev, AX_SWRESET_IPPD, in_pm);
> +               if (ret < 0)
> +                       goto out;
>
> -       msleep(150);
> +               usleep_range(10000, 11000);
>
> -       ret = asix_sw_reset(dev, AX_SWRESET_CLEAR);
> -       if (ret < 0)
> -               goto out;
> +               ret = asix_sw_reset(dev, AX_SWRESET_CLEAR, in_pm);
> +               if (ret < 0)
> +                       goto out;
>
> -       msleep(150);
> +               msleep(60);
>
> -       if (embd_phy) {
> -               ret = asix_sw_reset(dev, AX_SWRESET_IPRL);
> +               ret = asix_sw_reset(dev, AX_SWRESET_IPRL | AX_SWRESET_PRL,
> +                                   in_pm);
>                 if (ret < 0)
>                         goto out;
>         } else {
> -               ret = asix_sw_reset(dev, AX_SWRESET_PRTE);
> +               ret = asix_sw_reset(dev, AX_SWRESET_IPPD | AX_SWRESET_PRL,
> +                                   in_pm);
>                 if (ret < 0)
>                         goto out;
>         }
>
>         msleep(150);
> -       rx_ctl = asix_read_rx_ctl(dev);
> -       netdev_dbg(dev->net, "RX_CTL is 0x%04x after software reset\n", rx_ctl);
> -       ret = asix_write_rx_ctl(dev, 0x0000);
> +
> +       if (in_pm && (!asix_mdio_read_nopm(dev->net, dev->mii.phy_id,
> +                                          MII_PHYSID1))){
> +               ret = -EIO;
> +               goto out;
> +       }
> +
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       ret = asix_write_medium_mode(dev, AX88772_MEDIUM_DEFAULT, in_pm);
>         if (ret < 0)
>                 goto out;
>
> -       rx_ctl = asix_read_rx_ctl(dev);
> -       netdev_dbg(dev->net, "RX_CTL is 0x%04x setting to 0x0000\n", rx_ctl);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_IPG0,
> +                            AX88772_IPG0_DEFAULT | AX88772_IPG1_DEFAULT,
> +                            AX88772_IPG2_DEFAULT, 0, NULL, in_pm);
> +       if (ret < 0) {
> +               netdev_dbg(dev->net, "Write IPG,IPG1,IPG2 failed: %d\n", ret);
> +               goto out;
> +       }
>
> -       ret = asix_sw_reset(dev, AX_SWRESET_PRL);
> +       /* Rewrite MAC address */
> +       ether_addr_copy(data->mac_addr, dev->net->dev_addr);
> +       ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0,
> +                            ETH_ALEN, data->mac_addr, in_pm);
>         if (ret < 0)
>                 goto out;
>
> -       msleep(150);
> +       /* Set RX_CTL to default values with 2k buffer, and enable cactus */
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       rx_ctl = asix_read_rx_ctl(dev, in_pm);
> +       netdev_dbg(dev->net, "RX_CTL is 0x%04x after all initializations\n",
> +                  rx_ctl);
> +
> +       rx_ctl = asix_read_medium_status(dev, in_pm);
> +       netdev_dbg(dev->net,
> +                  "Medium Status is 0x%04x after all initializations\n",
> +                  rx_ctl);
> +
> +       return 0;
> +
> +out:
> +       return ret;
> +}
> +
> +static int ax88772a_hw_reset(struct usbnet *dev, int in_pm)
> +{
> +       struct asix_data *data = (struct asix_data *)&dev->data;
> +       int ret, embd_phy;
> +       u16 rx_ctl;
> +       u8 chipcode = 0;
>
> -       ret = asix_sw_reset(dev, AX_SWRESET_IPRL | AX_SWRESET_PRL);
> +       ret = asix_write_gpio(dev, AX_GPIO_RSE, 5, in_pm);
>         if (ret < 0)
>                 goto out;
>
> -       msleep(150);
> +       embd_phy = ((dev->mii.phy_id & 0x1f) == 0x10 ? 1 : 0);
>
> -       asix_mdio_write(dev->net, dev->mii.phy_id, MII_BMCR, BMCR_RESET);
> -       asix_mdio_write(dev->net, dev->mii.phy_id, MII_ADVERTISE,
> -                       ADVERTISE_ALL | ADVERTISE_CSMA);
> -       mii_nway_restart(&dev->mii);
> +       ret = asix_write_cmd(dev, AX_CMD_SW_PHY_SELECT, embd_phy |
> +                            AX_PHYSEL_SSEN, 0, 0, NULL, in_pm);
> +       if (ret < 0) {
> +               netdev_dbg(dev->net, "Select PHY #1 failed: %d\n", ret);
> +               goto out;
> +       }
> +       usleep_range(10000, 11000);
>
> -       ret = asix_write_medium_mode(dev, AX88772_MEDIUM_DEFAULT);
> +       ret = asix_sw_reset(dev, AX_SWRESET_IPPD | AX_SWRESET_IPRL, in_pm);
>         if (ret < 0)
>                 goto out;
>
> +       usleep_range(10000, 11000);
> +
> +       ret = asix_sw_reset(dev, AX_SWRESET_IPRL, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       msleep(160);
> +
> +       ret = asix_sw_reset(dev, AX_SWRESET_CLEAR, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       ret = asix_sw_reset(dev, AX_SWRESET_IPRL, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       msleep(200);
> +
> +       if (in_pm && (!asix_mdio_read_nopm(dev->net, dev->mii.phy_id,
> +                                          MII_PHYSID1))) {
> +               ret = -1;
> +               goto out;
> +       }
> +
> +       ret = asix_read_cmd(dev, AX_CMD_STATMNGSTS_REG, 0,
> +                           0, 1, &chipcode, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       if ((chipcode & AX_CHIPCODE_MASK) == AX_AX88772B_CHIPCODE) {
> +               ret = asix_write_cmd(dev, AX_QCTCTRL, 0x8000, 0x8001,
> +                                    0, NULL, in_pm);
> +               if (ret < 0) {
> +                       netdev_dbg(dev->net, "Write BQ setting failed: %d\n",
> +                                  ret);
> +                       goto out;
> +               }
> +       }
> +
>         ret = asix_write_cmd(dev, AX_CMD_WRITE_IPG0,
>                                 AX88772_IPG0_DEFAULT | AX88772_IPG1_DEFAULT,
> -                               AX88772_IPG2_DEFAULT, 0, NULL);
> +                               AX88772_IPG2_DEFAULT, 0, NULL, in_pm);
>         if (ret < 0) {
>                 netdev_dbg(dev->net, "Write IPG,IPG1,IPG2 failed: %d\n", ret);
>                 goto out;
> @@ -378,20 +493,29 @@ static int ax88772_reset(struct usbnet *dev)
>         /* Rewrite MAC address */
>         memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
>         ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0, ETH_ALEN,
> -                                                       data->mac_addr);
> +                                                       data->mac_addr, in_pm);
> +       if (ret < 0)
> +               goto out;
> +
> +       /* Set RX_CTL to default values with 2k buffer, and enable cactus */
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, in_pm);
>         if (ret < 0)
>                 goto out;
>
> +       ret = asix_write_medium_mode(dev, AX88772_MEDIUM_DEFAULT, in_pm);
> +       if (ret < 0)
> +               return ret;
> +
>         /* Set RX_CTL to default values with 2k buffer, and enable cactus */
> -       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL);
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, in_pm);
>         if (ret < 0)
>                 goto out;
>
> -       rx_ctl = asix_read_rx_ctl(dev);
> +       rx_ctl = asix_read_rx_ctl(dev, in_pm);
>         netdev_dbg(dev->net, "RX_CTL is 0x%04x after all initializations\n",
>                    rx_ctl);
>
> -       rx_ctl = asix_read_medium_status(dev);
> +       rx_ctl = asix_read_medium_status(dev, in_pm);
>         netdev_dbg(dev->net,
>                    "Medium Status is 0x%04x after all initializations\n",
>                    rx_ctl);
> @@ -400,7 +524,6 @@ static int ax88772_reset(struct usbnet *dev)
>
>  out:
>         return ret;
> -
>  }
>
>  static const struct net_device_ops ax88772_netdev_ops = {
> @@ -415,11 +538,87 @@ static const struct net_device_ops ax88772_netdev_ops = {
>         .ndo_set_rx_mode        = asix_set_multicast,
>  };
>
> +static void ax88772_suspend(struct usbnet *dev)
> +{
> +       struct asix_common_private *priv = dev->driver_priv;
> +
> +       /* Preserve BMCR for restoring */
> +       priv->presvd_phy_bmcr =
> +               asix_mdio_read_nopm(dev->net, dev->mii.phy_id, MII_BMCR);
> +
> +       /* Preserve ANAR for restoring */
> +       priv->presvd_phy_advertise =
> +               asix_mdio_read_nopm(dev->net, dev->mii.phy_id, MII_ADVERTISE);
> +}
> +
> +static int asix_suspend(struct usb_interface *intf, pm_message_t message)
> +{
> +       struct usbnet *dev = usb_get_intfdata(intf);
> +       struct asix_common_private *priv = dev->driver_priv;
> +
> +       if (priv->suspend)
> +               priv->suspend(dev);
> +
> +       return usbnet_suspend(intf, message);
> +}
> +
> +static void ax88772_restore_phy(struct usbnet *dev)
> +{
> +       struct asix_common_private *priv = dev->driver_priv;
> +
> +       if (priv->presvd_phy_advertise) {
> +               /* Restore Advertisement control reg */
> +               asix_mdio_write_nopm(dev->net, dev->mii.phy_id, MII_ADVERTISE,
> +                                    priv->presvd_phy_advertise);
> +
> +               /* Restore BMCR */
> +               asix_mdio_write_nopm(dev->net, dev->mii.phy_id, MII_BMCR,
> +                                    priv->presvd_phy_bmcr);
> +
> +               priv->presvd_phy_advertise = 0;
> +               priv->presvd_phy_bmcr = 0;
> +       }
> +}
> +
> +static void ax88772_resume(struct usbnet *dev)
> +{
> +       int i;
> +
> +       for (i = 0; i < 3; i++)
> +               if (!ax88772_hw_reset(dev, 1))
> +                       break;
> +       ax88772_restore_phy(dev);
> +}
> +
> +static void ax88772a_resume(struct usbnet *dev)
> +{
> +       int i;
> +
> +       for (i = 0; i < 3; i++) {
> +               if (!ax88772a_hw_reset(dev, 1))
> +                       break;
> +       }
> +
> +       ax88772_restore_phy(dev);
> +}
> +
> +static int asix_resume(struct usb_interface *intf)
> +{
> +       struct usbnet *dev = usb_get_intfdata(intf);
> +       struct asix_common_private *priv = dev->driver_priv;
> +
> +       if (priv->resume)
> +               priv->resume(dev);
> +
> +       return usbnet_resume(intf);
> +}
> +
>  static int ax88772_bind(struct usbnet *dev, struct usb_interface *intf)
>  {
> -       int ret, embd_phy, i;
> -       u8 buf[ETH_ALEN];
> +       int ret, i;
> +       u8 buf[ETH_ALEN], chipcode = 0;
>         u32 phyid;
> +       struct asix_common_private *priv;
>
>         usbnet_get_endpoints(dev,intf);
>
> @@ -427,13 +626,13 @@ static int ax88772_bind(struct usbnet *dev, struct usb_interface *intf)
>         if (dev->driver_info->data & FLAG_EEPROM_MAC) {
>                 for (i = 0; i < (ETH_ALEN >> 1); i++) {
>                         ret = asix_read_cmd(dev, AX_CMD_READ_EEPROM, 0x04 + i,
> -                                       0, 2, buf + i * 2);
> +                                           0, 2, buf + i * 2, 0);
>                         if (ret < 0)
>                                 break;
>                 }
>         } else {
>                 ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID,
> -                               0, 0, ETH_ALEN, buf);
> +                               0, 0, ETH_ALEN, buf, 0);
>         }
>
>         if (ret < 0) {
> @@ -456,16 +655,11 @@ static int ax88772_bind(struct usbnet *dev, struct usb_interface *intf)
>         dev->net->needed_headroom = 4; /* cf asix_tx_fixup() */
>         dev->net->needed_tailroom = 4; /* cf asix_tx_fixup() */
>
> -       embd_phy = ((dev->mii.phy_id & 0x1f) == 0x10 ? 1 : 0);
> +       asix_read_cmd(dev, AX_CMD_STATMNGSTS_REG, 0, 0, 1, &chipcode, 0);
> +       chipcode &= AX_CHIPCODE_MASK;
>
> -       /* Reset the PHY to normal operation mode */
> -       ret = asix_write_cmd(dev, AX_CMD_SW_PHY_SELECT, embd_phy, 0, 0, NULL);
> -       if (ret < 0) {
> -               netdev_dbg(dev->net, "Select PHY #1 failed: %d\n", ret);
> -               return ret;
> -       }
> -
> -       ax88772_reset(dev);
> +       (chipcode == AX_AX88772_CHIPCODE) ? ax88772_hw_reset(dev, 0) :
> +                                           ax88772a_hw_reset(dev, 0);
>
>         /* Read PHYID register *AFTER* the PHY was reset properly */
>         phyid = asix_get_phyid(dev);
> @@ -482,6 +676,18 @@ static int ax88772_bind(struct usbnet *dev, struct usb_interface *intf)
>         if (!dev->driver_priv)
>                 return -ENOMEM;
>
> +       priv = dev->driver_priv;
> +
> +       priv->presvd_phy_bmcr = 0;
> +       priv->presvd_phy_advertise = 0;
> +       if (chipcode == AX_AX88772_CHIPCODE) {
> +               priv->resume = ax88772_resume;
> +               priv->suspend = ax88772_suspend;
> +       } else {
> +               priv->resume = ax88772a_resume;
> +               priv->suspend = ax88772_suspend;
> +       }
> +
>         return 0;
>  }
>
> @@ -593,12 +799,12 @@ static int ax88178_reset(struct usbnet *dev)
>         int gpio0 = 0;
>         u32 phyid;
>
> -       asix_read_cmd(dev, AX_CMD_READ_GPIOS, 0, 0, 1, &status);
> +       asix_read_cmd(dev, AX_CMD_READ_GPIOS, 0, 0, 1, &status, 0);
>         netdev_dbg(dev->net, "GPIO Status: 0x%04x\n", status);
>
> -       asix_write_cmd(dev, AX_CMD_WRITE_ENABLE, 0, 0, 0, NULL);
> -       asix_read_cmd(dev, AX_CMD_READ_EEPROM, 0x0017, 0, 2, &eeprom);
> -       asix_write_cmd(dev, AX_CMD_WRITE_DISABLE, 0, 0, 0, NULL);
> +       asix_write_cmd(dev, AX_CMD_WRITE_ENABLE, 0, 0, 0, NULL, 0);
> +       asix_read_cmd(dev, AX_CMD_READ_EEPROM, 0x0017, 0, 2, &eeprom, 0);
> +       asix_write_cmd(dev, AX_CMD_WRITE_DISABLE, 0, 0, 0, NULL, 0);
>
>         netdev_dbg(dev->net, "EEPROM index 0x17 is 0x%04x\n", eeprom);
>
> @@ -614,15 +820,16 @@ static int ax88178_reset(struct usbnet *dev)
>         netdev_dbg(dev->net, "GPIO0: %d, PhyMode: %d\n", gpio0, data->phymode);
>
>         /* Power up external GigaPHY through AX88178 GPIO pin */
> -       asix_write_gpio(dev, AX_GPIO_RSE | AX_GPIO_GPO_1 | AX_GPIO_GPO1EN, 40);
> +       asix_write_gpio(dev, AX_GPIO_RSE | AX_GPIO_GPO_1 |
> +                       AX_GPIO_GPO1EN, 40, 0);
>         if ((le16_to_cpu(eeprom) >> 8) != 1) {
> -               asix_write_gpio(dev, 0x003c, 30);
> -               asix_write_gpio(dev, 0x001c, 300);
> -               asix_write_gpio(dev, 0x003c, 30);
> +               asix_write_gpio(dev, 0x003c, 30, 0);
> +               asix_write_gpio(dev, 0x001c, 300, 0);
> +               asix_write_gpio(dev, 0x003c, 30, 0);
>         } else {
>                 netdev_dbg(dev->net, "gpio phymode == 1 path\n");
> -               asix_write_gpio(dev, AX_GPIO_GPO1EN, 30);
> -               asix_write_gpio(dev, AX_GPIO_GPO1EN | AX_GPIO_GPO_1, 30);
> +               asix_write_gpio(dev, AX_GPIO_GPO1EN, 30, 0);
> +               asix_write_gpio(dev, AX_GPIO_GPO1EN | AX_GPIO_GPO_1, 30, 0);
>         }
>
>         /* Read PHYID register *AFTER* powering up PHY */
> @@ -630,15 +837,15 @@ static int ax88178_reset(struct usbnet *dev)
>         netdev_dbg(dev->net, "PHYID=0x%08x\n", phyid);
>
>         /* Set AX88178 to enable MII/GMII/RGMII interface for external PHY */
> -       asix_write_cmd(dev, AX_CMD_SW_PHY_SELECT, 0, 0, 0, NULL);
> +       asix_write_cmd(dev, AX_CMD_SW_PHY_SELECT, 0, 0, 0, NULL, 0);
>
> -       asix_sw_reset(dev, 0);
> +       asix_sw_reset(dev, 0, 0);
>         msleep(150);
>
> -       asix_sw_reset(dev, AX_SWRESET_PRL | AX_SWRESET_IPPD);
> +       asix_sw_reset(dev, AX_SWRESET_PRL | AX_SWRESET_IPPD, 0);
>         msleep(150);
>
> -       asix_write_rx_ctl(dev, 0);
> +       asix_write_rx_ctl(dev, 0, 0);
>
>         if (data->phymode == PHY_MODE_MARVELL) {
>                 marvell_phy_init(dev);
> @@ -655,18 +862,18 @@ static int ax88178_reset(struct usbnet *dev)
>
>         mii_nway_restart(&dev->mii);
>
> -       ret = asix_write_medium_mode(dev, AX88178_MEDIUM_DEFAULT);
> +       ret = asix_write_medium_mode(dev, AX88178_MEDIUM_DEFAULT, 0);
>         if (ret < 0)
>                 return ret;
>
>         /* Rewrite MAC address */
>         memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
>         ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0, ETH_ALEN,
> -                                                       data->mac_addr);
> +                                                       data->mac_addr, 0);
>         if (ret < 0)
>                 return ret;
>
> -       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL);
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, 0);
>         if (ret < 0)
>                 return ret;
>
> @@ -704,7 +911,7 @@ static int ax88178_link_reset(struct usbnet *dev)
>         netdev_dbg(dev->net, "ax88178_link_reset() speed: %u duplex: %d setting mode to 0x%04x\n",
>                    speed, ecmd.duplex, mode);
>
> -       asix_write_medium_mode(dev, mode);
> +       asix_write_medium_mode(dev, mode, 0);
>
>         if (data->phymode == PHY_MODE_MARVELL && data->ledmode)
>                 marvell_led_status(dev, speed);
> @@ -733,15 +940,15 @@ static void ax88178_set_mfb(struct usbnet *dev)
>                 mfb = AX_RX_CTL_MFB_16384;
>         }
>
> -       rxctl = asix_read_rx_ctl(dev);
> -       asix_write_rx_ctl(dev, (rxctl & ~AX_RX_CTL_MFB_16384) | mfb);
> +       rxctl = asix_read_rx_ctl(dev, 0);
> +       asix_write_rx_ctl(dev, (rxctl & ~AX_RX_CTL_MFB_16384) | mfb, 0);
>
> -       medium = asix_read_medium_status(dev);
> +       medium = asix_read_medium_status(dev, 0);
>         if (dev->net->mtu > 1500)
>                 medium |= AX_MEDIUM_JFE;
>         else
>                 medium &= ~AX_MEDIUM_JFE;
> -       asix_write_medium_mode(dev, medium);
> +       asix_write_medium_mode(dev, medium, 0);
>
>         if (dev->rx_urb_size > old_rx_urb_size)
>                 usbnet_unlink_rx_urbs(dev);
> @@ -790,7 +997,7 @@ static int ax88178_bind(struct usbnet *dev, struct usb_interface *intf)
>         usbnet_get_endpoints(dev,intf);
>
>         /* Get the MAC address */
> -       ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID, 0, 0, ETH_ALEN, buf);
> +       ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID, 0, 0, ETH_ALEN, buf, 0);
>         if (ret < 0) {
>                 netdev_dbg(dev->net, "Failed to read MAC address: %d\n", ret);
>                 return ret;
> @@ -811,10 +1018,10 @@ static int ax88178_bind(struct usbnet *dev, struct usb_interface *intf)
>         dev->net->ethtool_ops = &ax88178_ethtool_ops;
>
>         /* Blink LEDS so users know driver saw dongle */
> -       asix_sw_reset(dev, 0);
> +       asix_sw_reset(dev, 0, 0);
>         msleep(150);
>
> -       asix_sw_reset(dev, AX_SWRESET_PRL | AX_SWRESET_IPPD);
> +       asix_sw_reset(dev, AX_SWRESET_PRL | AX_SWRESET_IPPD, 0);
>         msleep(150);
>
>         /* Asix framing packs multiple eth frames into a 2K usb bulk transfer */
> @@ -877,7 +1084,7 @@ static const struct driver_info ax88772_info = {
>         .unbind = ax88772_unbind,
>         .status = asix_status,
>         .link_reset = ax88772_link_reset,
> -       .reset = ax88772_link_reset,
> +       .reset = ax88772_reset,
>         .flags = FLAG_ETHER | FLAG_FRAMING_AX | FLAG_LINK_INTR | FLAG_MULTI_PACKET,
>         .rx_fixup = asix_rx_fixup_common,
>         .tx_fixup = asix_tx_fixup,
> @@ -1005,7 +1212,7 @@ static const struct usb_device_id products [] = {
>  }, {
>         // Lenovo U2L100P 10/100
>         USB_DEVICE (0x17ef, 0x7203),
> -       .driver_info = (unsigned long) &ax88772_info,
> +       .driver_info = (unsigned long)&ax88772b_info,
>  }, {
>         // ASIX AX88772B 10/100
>         USB_DEVICE (0x0b95, 0x772b),
> @@ -1073,7 +1280,7 @@ static const struct usb_device_id products [] = {
>  }, {
>         // Asus USB Ethernet Adapter
>         USB_DEVICE (0x0b95, 0x7e2b),
> -       .driver_info = (unsigned long) &ax88772_info,
> +       .driver_info = (unsigned long)&ax88772b_info,
>  }, {
>         /* ASIX 88172a demo board */
>         USB_DEVICE(0x0b95, 0x172a),
> @@ -1095,8 +1302,8 @@ static struct usb_driver asix_driver = {
>         .name =         DRIVER_NAME,
>         .id_table =     products,
>         .probe =        usbnet_probe,
> -       .suspend =      usbnet_suspend,
> -       .resume =       usbnet_resume,
> +       .suspend =      asix_suspend,
> +       .resume =       asix_resume,
>         .disconnect =   usbnet_disconnect,
>         .supports_autosuspend = 1,
>         .disable_hub_initiated_lpm = 1,
> diff --git a/drivers/net/usb/ax88172a.c b/drivers/net/usb/ax88172a.c
> index 5f18fcb..c2394f1 100644
> --- a/drivers/net/usb/ax88172a.c
> +++ b/drivers/net/usb/ax88172a.c
> @@ -81,7 +81,7 @@ static void ax88172a_adjust_link(struct net_device *netdev)
>         }
>
>         if (mode != priv->oldmode) {
> -               asix_write_medium_mode(dev, mode);
> +               asix_write_medium_mode(dev, mode, 0);
>                 priv->oldmode = mode;
>                 netdev_dbg(netdev, "speed %u duplex %d, setting mode to 0x%04x\n",
>                            phydev->speed, phydev->duplex, mode);
> @@ -205,18 +205,19 @@ static int ax88172a_reset_phy(struct usbnet *dev, int embd_phy)
>  {
>         int ret;
>
> -       ret = asix_sw_reset(dev, AX_SWRESET_IPPD);
> +       ret = asix_sw_reset(dev, AX_SWRESET_IPPD, 0);
>         if (ret < 0)
>                 goto err;
>
>         msleep(150);
> -       ret = asix_sw_reset(dev, AX_SWRESET_CLEAR);
> +       ret = asix_sw_reset(dev, AX_SWRESET_CLEAR, 0);
>         if (ret < 0)
>                 goto err;
>
>         msleep(150);
>
> -       ret = asix_sw_reset(dev, embd_phy ? AX_SWRESET_IPRL : AX_SWRESET_IPPD);
> +       ret = asix_sw_reset(dev, embd_phy ? AX_SWRESET_IPRL : AX_SWRESET_IPPD,
> +                           0);
>         if (ret < 0)
>                 goto err;
>
> @@ -242,7 +243,7 @@ static int ax88172a_bind(struct usbnet *dev, struct usb_interface *intf)
>         dev->driver_priv = priv;
>
>         /* Get the MAC address */
> -       ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID, 0, 0, ETH_ALEN, buf);
> +       ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID, 0, 0, ETH_ALEN, buf, 0);
>         if (ret < 0) {
>                 netdev_err(dev->net, "Failed to read MAC address: %d\n", ret);
>                 goto free;
> @@ -253,7 +254,7 @@ static int ax88172a_bind(struct usbnet *dev, struct usb_interface *intf)
>         dev->net->ethtool_ops = &ax88172a_ethtool_ops;
>
>         /* are we using the internal or the external phy? */
> -       ret = asix_read_cmd(dev, AX_CMD_SW_PHY_STATUS, 0, 0, 1, buf);
> +       ret = asix_read_cmd(dev, AX_CMD_SW_PHY_STATUS, 0, 0, 1, buf, 0);
>         if (ret < 0) {
>                 netdev_err(dev->net, "Failed to read software interface selection register: %d\n",
>                            ret);
> @@ -332,20 +333,20 @@ static int ax88172a_reset(struct usbnet *dev)
>         ax88172a_reset_phy(dev, priv->use_embdphy);
>
>         msleep(150);
> -       rx_ctl = asix_read_rx_ctl(dev);
> +       rx_ctl = asix_read_rx_ctl(dev, 0);
>         netdev_dbg(dev->net, "RX_CTL is 0x%04x after software reset\n", rx_ctl);
> -       ret = asix_write_rx_ctl(dev, 0x0000);
> +       ret = asix_write_rx_ctl(dev, 0x0000, 0);
>         if (ret < 0)
>                 goto out;
>
> -       rx_ctl = asix_read_rx_ctl(dev);
> +       rx_ctl = asix_read_rx_ctl(dev, 0);
>         netdev_dbg(dev->net, "RX_CTL is 0x%04x setting to 0x0000\n", rx_ctl);
>
>         msleep(150);
>
>         ret = asix_write_cmd(dev, AX_CMD_WRITE_IPG0,
>                              AX88772_IPG0_DEFAULT | AX88772_IPG1_DEFAULT,
> -                            AX88772_IPG2_DEFAULT, 0, NULL);
> +                            AX88772_IPG2_DEFAULT, 0, NULL, 0);
>         if (ret < 0) {
>                 netdev_err(dev->net, "Write IPG,IPG1,IPG2 failed: %d\n", ret);
>                 goto out;
> @@ -354,20 +355,20 @@ static int ax88172a_reset(struct usbnet *dev)
>         /* Rewrite MAC address */
>         memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
>         ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0, ETH_ALEN,
> -                            data->mac_addr);
> +                            data->mac_addr, 0);
>         if (ret < 0)
>                 goto out;
>
>         /* Set RX_CTL to default values with 2k buffer, and enable cactus */
> -       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL);
> +       ret = asix_write_rx_ctl(dev, AX_DEFAULT_RX_CTL, 0);
>         if (ret < 0)
>                 goto out;
>
> -       rx_ctl = asix_read_rx_ctl(dev);
> +       rx_ctl = asix_read_rx_ctl(dev, 0);
>         netdev_dbg(dev->net, "RX_CTL is 0x%04x after all initializations\n",
>                    rx_ctl);
>
> -       rx_ctl = asix_read_medium_status(dev);
> +       rx_ctl = asix_read_medium_status(dev, 0);
>         netdev_dbg(dev->net, "Medium Status is 0x%04x after all initializations\n",
>                    rx_ctl);
>
> --
> 2.7.4
>

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ