|
@@ -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",
|
|
netdev_dbg(dev->net, "ax88172_link_reset() speed: %u duplex: %d setting mode to 0x%04x\n",
|
|
ethtool_cmd_speed(&ecmd), ecmd.duplex, mode);
|
|
ethtool_cmd_speed(&ecmd), ecmd.duplex, mode);
|
|
|
|
|
|
- asix_write_medium_mode(dev, mode);
|
|
|
|
|
|
+ asix_write_medium_mode(dev, mode, 0);
|
|
|
|
|
|
return 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 */
|
|
/* Toggle the GPIOs in a manufacturer/model specific way */
|
|
for (i = 2; i >= 0; i--) {
|
|
for (i = 2; i >= 0; i--) {
|
|
ret = asix_write_cmd(dev, AX_CMD_WRITE_GPIOS,
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
goto out;
|
|
msleep(5);
|
|
msleep(5);
|
|
}
|
|
}
|
|
|
|
|
|
- ret = asix_write_rx_ctl(dev, 0x80);
|
|
|
|
|
|
+ ret = asix_write_rx_ctl(dev, 0x80, 0);
|
|
if (ret < 0)
|
|
if (ret < 0)
|
|
goto out;
|
|
goto out;
|
|
|
|
|
|
/* Get the MAC address */
|
|
/* 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) {
|
|
if (ret < 0) {
|
|
netdev_dbg(dev->net, "read AX_CMD_READ_NODE_ID failed: %d\n",
|
|
netdev_dbg(dev->net, "read AX_CMD_READ_NODE_ID failed: %d\n",
|
|
ret);
|
|
ret);
|
|
@@ -290,86 +291,200 @@ 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",
|
|
netdev_dbg(dev->net, "ax88772_link_reset() speed: %u duplex: %d setting mode to 0x%04x\n",
|
|
ethtool_cmd_speed(&ecmd), ecmd.duplex, mode);
|
|
ethtool_cmd_speed(&ecmd), ecmd.duplex, mode);
|
|
|
|
|
|
- asix_write_medium_mode(dev, mode);
|
|
|
|
|
|
+ asix_write_medium_mode(dev, mode, 0);
|
|
|
|
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
static int ax88772_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;
|
|
struct asix_data *data = (struct asix_data *)&dev->data;
|
|
int ret, embd_phy;
|
|
int ret, embd_phy;
|
|
u16 rx_ctl;
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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) {
|
|
if (ret < 0) {
|
|
netdev_dbg(dev->net, "Select PHY #1 failed: %d\n", ret);
|
|
netdev_dbg(dev->net, "Select PHY #1 failed: %d\n", ret);
|
|
goto out;
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
goto out;
|
|
} else {
|
|
} 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)
|
|
if (ret < 0)
|
|
goto out;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
|
|
msleep(150);
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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,
|
|
ret = asix_write_cmd(dev, AX_CMD_WRITE_IPG0,
|
|
AX88772_IPG0_DEFAULT | AX88772_IPG1_DEFAULT,
|
|
AX88772_IPG0_DEFAULT | AX88772_IPG1_DEFAULT,
|
|
- AX88772_IPG2_DEFAULT, 0, NULL);
|
|
|
|
|
|
+ AX88772_IPG2_DEFAULT, 0, NULL, in_pm);
|
|
if (ret < 0) {
|
|
if (ret < 0) {
|
|
netdev_dbg(dev->net, "Write IPG,IPG1,IPG2 failed: %d\n", ret);
|
|
netdev_dbg(dev->net, "Write IPG,IPG1,IPG2 failed: %d\n", ret);
|
|
goto out;
|
|
goto out;
|
|
@@ -378,20 +493,29 @@ static int ax88772_reset(struct usbnet *dev)
|
|
/* Rewrite MAC address */
|
|
/* Rewrite MAC address */
|
|
memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
|
|
memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
|
|
ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0, 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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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 */
|
|
/* 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)
|
|
if (ret < 0)
|
|
goto out;
|
|
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",
|
|
netdev_dbg(dev->net, "RX_CTL is 0x%04x after all initializations\n",
|
|
rx_ctl);
|
|
rx_ctl);
|
|
|
|
|
|
- rx_ctl = asix_read_medium_status(dev);
|
|
|
|
|
|
+ rx_ctl = asix_read_medium_status(dev, in_pm);
|
|
netdev_dbg(dev->net,
|
|
netdev_dbg(dev->net,
|
|
"Medium Status is 0x%04x after all initializations\n",
|
|
"Medium Status is 0x%04x after all initializations\n",
|
|
rx_ctl);
|
|
rx_ctl);
|
|
@@ -400,7 +524,6 @@ static int ax88772_reset(struct usbnet *dev)
|
|
|
|
|
|
out:
|
|
out:
|
|
return ret;
|
|
return ret;
|
|
-
|
|
|
|
}
|
|
}
|
|
|
|
|
|
static const struct net_device_ops ax88772_netdev_ops = {
|
|
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,
|
|
.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)
|
|
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;
|
|
u32 phyid;
|
|
|
|
+ struct asix_common_private *priv;
|
|
|
|
|
|
usbnet_get_endpoints(dev,intf);
|
|
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) {
|
|
if (dev->driver_info->data & FLAG_EEPROM_MAC) {
|
|
for (i = 0; i < (ETH_ALEN >> 1); i++) {
|
|
for (i = 0; i < (ETH_ALEN >> 1); i++) {
|
|
ret = asix_read_cmd(dev, AX_CMD_READ_EEPROM, 0x04 + 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)
|
|
if (ret < 0)
|
|
break;
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
} else {
|
|
ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID,
|
|
ret = asix_read_cmd(dev, AX_CMD_READ_NODE_ID,
|
|
- 0, 0, ETH_ALEN, buf);
|
|
|
|
|
|
+ 0, 0, ETH_ALEN, buf, 0);
|
|
}
|
|
}
|
|
|
|
|
|
if (ret < 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_headroom = 4; /* cf asix_tx_fixup() */
|
|
dev->net->needed_tailroom = 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 */
|
|
/* Read PHYID register *AFTER* the PHY was reset properly */
|
|
phyid = asix_get_phyid(dev);
|
|
phyid = asix_get_phyid(dev);
|
|
@@ -482,6 +676,18 @@ static int ax88772_bind(struct usbnet *dev, struct usb_interface *intf)
|
|
if (!dev->driver_priv)
|
|
if (!dev->driver_priv)
|
|
return -ENOMEM;
|
|
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;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -593,12 +799,12 @@ static int ax88178_reset(struct usbnet *dev)
|
|
int gpio0 = 0;
|
|
int gpio0 = 0;
|
|
u32 phyid;
|
|
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);
|
|
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);
|
|
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);
|
|
netdev_dbg(dev->net, "GPIO0: %d, PhyMode: %d\n", gpio0, data->phymode);
|
|
|
|
|
|
/* Power up external GigaPHY through AX88178 GPIO pin */
|
|
/* 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) {
|
|
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 {
|
|
} else {
|
|
netdev_dbg(dev->net, "gpio phymode == 1 path\n");
|
|
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 */
|
|
/* 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);
|
|
netdev_dbg(dev->net, "PHYID=0x%08x\n", phyid);
|
|
|
|
|
|
/* Set AX88178 to enable MII/GMII/RGMII interface for external PHY */
|
|
/* 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);
|
|
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);
|
|
msleep(150);
|
|
|
|
|
|
- asix_write_rx_ctl(dev, 0);
|
|
|
|
|
|
+ asix_write_rx_ctl(dev, 0, 0);
|
|
|
|
|
|
if (data->phymode == PHY_MODE_MARVELL) {
|
|
if (data->phymode == PHY_MODE_MARVELL) {
|
|
marvell_phy_init(dev);
|
|
marvell_phy_init(dev);
|
|
@@ -655,18 +862,18 @@ static int ax88178_reset(struct usbnet *dev)
|
|
|
|
|
|
mii_nway_restart(&dev->mii);
|
|
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)
|
|
if (ret < 0)
|
|
return ret;
|
|
return ret;
|
|
|
|
|
|
/* Rewrite MAC address */
|
|
/* Rewrite MAC address */
|
|
memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
|
|
memcpy(data->mac_addr, dev->net->dev_addr, ETH_ALEN);
|
|
ret = asix_write_cmd(dev, AX_CMD_WRITE_NODE_ID, 0, 0, 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)
|
|
if (ret < 0)
|
|
return ret;
|
|
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)
|
|
if (ret < 0)
|
|
return ret;
|
|
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",
|
|
netdev_dbg(dev->net, "ax88178_link_reset() speed: %u duplex: %d setting mode to 0x%04x\n",
|
|
speed, ecmd.duplex, mode);
|
|
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)
|
|
if (data->phymode == PHY_MODE_MARVELL && data->ledmode)
|
|
marvell_led_status(dev, speed);
|
|
marvell_led_status(dev, speed);
|
|
@@ -733,15 +940,15 @@ static void ax88178_set_mfb(struct usbnet *dev)
|
|
mfb = AX_RX_CTL_MFB_16384;
|
|
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)
|
|
if (dev->net->mtu > 1500)
|
|
medium |= AX_MEDIUM_JFE;
|
|
medium |= AX_MEDIUM_JFE;
|
|
else
|
|
else
|
|
medium &= ~AX_MEDIUM_JFE;
|
|
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)
|
|
if (dev->rx_urb_size > old_rx_urb_size)
|
|
usbnet_unlink_rx_urbs(dev);
|
|
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);
|
|
usbnet_get_endpoints(dev,intf);
|
|
|
|
|
|
/* Get the MAC address */
|
|
/* 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) {
|
|
if (ret < 0) {
|
|
netdev_dbg(dev->net, "Failed to read MAC address: %d\n", ret);
|
|
netdev_dbg(dev->net, "Failed to read MAC address: %d\n", ret);
|
|
return 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;
|
|
dev->net->ethtool_ops = &ax88178_ethtool_ops;
|
|
|
|
|
|
/* Blink LEDS so users know driver saw dongle */
|
|
/* Blink LEDS so users know driver saw dongle */
|
|
- asix_sw_reset(dev, 0);
|
|
|
|
|
|
+ asix_sw_reset(dev, 0, 0);
|
|
msleep(150);
|
|
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);
|
|
msleep(150);
|
|
|
|
|
|
/* Asix framing packs multiple eth frames into a 2K usb bulk transfer */
|
|
/* 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,
|
|
.unbind = ax88772_unbind,
|
|
.status = asix_status,
|
|
.status = asix_status,
|
|
.link_reset = ax88772_link_reset,
|
|
.link_reset = ax88772_link_reset,
|
|
- .reset = ax88772_link_reset,
|
|
|
|
|
|
+ .reset = ax88772_reset,
|
|
.flags = FLAG_ETHER | FLAG_FRAMING_AX | FLAG_LINK_INTR | FLAG_MULTI_PACKET,
|
|
.flags = FLAG_ETHER | FLAG_FRAMING_AX | FLAG_LINK_INTR | FLAG_MULTI_PACKET,
|
|
.rx_fixup = asix_rx_fixup_common,
|
|
.rx_fixup = asix_rx_fixup_common,
|
|
.tx_fixup = asix_tx_fixup,
|
|
.tx_fixup = asix_tx_fixup,
|
|
@@ -1005,7 +1212,7 @@ static const struct usb_device_id products [] = {
|
|
}, {
|
|
}, {
|
|
// Lenovo U2L100P 10/100
|
|
// Lenovo U2L100P 10/100
|
|
USB_DEVICE (0x17ef, 0x7203),
|
|
USB_DEVICE (0x17ef, 0x7203),
|
|
- .driver_info = (unsigned long) &ax88772_info,
|
|
|
|
|
|
+ .driver_info = (unsigned long)&ax88772b_info,
|
|
}, {
|
|
}, {
|
|
// ASIX AX88772B 10/100
|
|
// ASIX AX88772B 10/100
|
|
USB_DEVICE (0x0b95, 0x772b),
|
|
USB_DEVICE (0x0b95, 0x772b),
|
|
@@ -1073,7 +1280,7 @@ static const struct usb_device_id products [] = {
|
|
}, {
|
|
}, {
|
|
// Asus USB Ethernet Adapter
|
|
// Asus USB Ethernet Adapter
|
|
USB_DEVICE (0x0b95, 0x7e2b),
|
|
USB_DEVICE (0x0b95, 0x7e2b),
|
|
- .driver_info = (unsigned long) &ax88772_info,
|
|
|
|
|
|
+ .driver_info = (unsigned long)&ax88772b_info,
|
|
}, {
|
|
}, {
|
|
/* ASIX 88172a demo board */
|
|
/* ASIX 88172a demo board */
|
|
USB_DEVICE(0x0b95, 0x172a),
|
|
USB_DEVICE(0x0b95, 0x172a),
|
|
@@ -1095,8 +1302,8 @@ static struct usb_driver asix_driver = {
|
|
.name = DRIVER_NAME,
|
|
.name = DRIVER_NAME,
|
|
.id_table = products,
|
|
.id_table = products,
|
|
.probe = usbnet_probe,
|
|
.probe = usbnet_probe,
|
|
- .suspend = usbnet_suspend,
|
|
|
|
- .resume = usbnet_resume,
|
|
|
|
|
|
+ .suspend = asix_suspend,
|
|
|
|
+ .resume = asix_resume,
|
|
.disconnect = usbnet_disconnect,
|
|
.disconnect = usbnet_disconnect,
|
|
.supports_autosuspend = 1,
|
|
.supports_autosuspend = 1,
|
|
.disable_hub_initiated_lpm = 1,
|
|
.disable_hub_initiated_lpm = 1,
|