Commit 136819a6 authored by Martin Blumenstingl's avatar Martin Blumenstingl Committed by David S. Miller

net: phy: realtek: add utility functions to read/write page addresses

Realtek PHYs implement the concept of so-called "extension pages". The
reason for this is probably because these PHYs expose more registers
than available in the standard address range.
After all read/write operations on such a page are done the driver
should switch back to page 0 where the standard MII registers (such as
MII_BMCR) are available.

When referring to such a register the datasheets of RTL8211E and
RTL8211F always specify:
- the page / "ext. page" which has to be written to RTL821x_PAGE_SELECT
- an address (sometimes also called reg)

These new utility functions make the existing code easier to read since
it removes some duplication (switching back to page 0 is done within the
new helpers for example).

No functional changes are intended.
Signed-off-by: default avatarMartin Blumenstingl <martin.blumenstingl@googlemail.com>
Reviewed-by: default avatarAndrew Lunn <andrew@lunn.ch>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent f609ab0e
...@@ -41,6 +41,39 @@ MODULE_DESCRIPTION("Realtek PHY driver"); ...@@ -41,6 +41,39 @@ MODULE_DESCRIPTION("Realtek PHY driver");
MODULE_AUTHOR("Johnson Leung"); MODULE_AUTHOR("Johnson Leung");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
static int rtl8211x_page_read(struct phy_device *phydev, u16 page, u16 address)
{
int ret;
ret = phy_write(phydev, RTL821x_PAGE_SELECT, page);
if (ret)
return ret;
ret = phy_read(phydev, address);
/* restore to default page 0 */
phy_write(phydev, RTL821x_PAGE_SELECT, 0x0);
return ret;
}
static int rtl8211x_page_write(struct phy_device *phydev, u16 page,
u16 address, u16 val)
{
int ret;
ret = phy_write(phydev, RTL821x_PAGE_SELECT, page);
if (ret)
return ret;
ret = phy_write(phydev, address, val);
/* restore to default page 0 */
phy_write(phydev, RTL821x_PAGE_SELECT, 0x0);
return ret;
}
static int rtl8201_ack_interrupt(struct phy_device *phydev) static int rtl8201_ack_interrupt(struct phy_device *phydev)
{ {
int err; int err;
...@@ -63,31 +96,21 @@ static int rtl8211f_ack_interrupt(struct phy_device *phydev) ...@@ -63,31 +96,21 @@ static int rtl8211f_ack_interrupt(struct phy_device *phydev)
{ {
int err; int err;
phy_write(phydev, RTL821x_PAGE_SELECT, 0xa43); err = rtl8211x_page_read(phydev, 0xa43, RTL8211F_INSR);
err = phy_read(phydev, RTL8211F_INSR);
/* restore to default page 0 */
phy_write(phydev, RTL821x_PAGE_SELECT, 0x0);
return (err < 0) ? err : 0; return (err < 0) ? err : 0;
} }
static int rtl8201_config_intr(struct phy_device *phydev) static int rtl8201_config_intr(struct phy_device *phydev)
{ {
int err; u16 val;
/* switch to page 7 */
phy_write(phydev, RTL821x_PAGE_SELECT, 0x7);
if (phydev->interrupts == PHY_INTERRUPT_ENABLED) if (phydev->interrupts == PHY_INTERRUPT_ENABLED)
err = phy_write(phydev, RTL8201F_IER, val = BIT(13) | BIT(12) | BIT(11);
BIT(13) | BIT(12) | BIT(11));
else else
err = phy_write(phydev, RTL8201F_IER, 0); val = 0;
/* restore to default page 0 */ return rtl8211x_page_write(phydev, 0x7, RTL8201F_IER, val);
phy_write(phydev, RTL821x_PAGE_SELECT, 0x0);
return err;
} }
static int rtl8211b_config_intr(struct phy_device *phydev) static int rtl8211b_config_intr(struct phy_device *phydev)
...@@ -118,41 +141,41 @@ static int rtl8211e_config_intr(struct phy_device *phydev) ...@@ -118,41 +141,41 @@ static int rtl8211e_config_intr(struct phy_device *phydev)
static int rtl8211f_config_intr(struct phy_device *phydev) static int rtl8211f_config_intr(struct phy_device *phydev)
{ {
int err; u16 val;
phy_write(phydev, RTL821x_PAGE_SELECT, 0xa42);
if (phydev->interrupts == PHY_INTERRUPT_ENABLED) if (phydev->interrupts == PHY_INTERRUPT_ENABLED)
err = phy_write(phydev, RTL821x_INER, val = RTL8211F_INER_LINK_STATUS;
RTL8211F_INER_LINK_STATUS);
else else
err = phy_write(phydev, RTL821x_INER, 0); val = 0;
phy_write(phydev, RTL821x_PAGE_SELECT, 0);
return err; return rtl8211x_page_write(phydev, 0xa42, RTL821x_INER, val);
} }
static int rtl8211f_config_init(struct phy_device *phydev) static int rtl8211f_config_init(struct phy_device *phydev)
{ {
int ret; int ret;
u16 reg; u16 val;
ret = genphy_config_init(phydev); ret = genphy_config_init(phydev);
if (ret < 0) if (ret < 0)
return ret; return ret;
phy_write(phydev, RTL821x_PAGE_SELECT, 0xd08); ret = rtl8211x_page_read(phydev, 0xd08, 0x11);
reg = phy_read(phydev, 0x11); if (ret < 0)
return ret;
val = ret & 0xffff;
/* enable TX-delay for rgmii-id and rgmii-txid, otherwise disable it */ /* enable TX-delay for rgmii-id and rgmii-txid, otherwise disable it */
if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID || if (phydev->interface == PHY_INTERFACE_MODE_RGMII_ID ||
phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID) phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID)
reg |= RTL8211F_TX_DELAY; val |= RTL8211F_TX_DELAY;
else else
reg &= ~RTL8211F_TX_DELAY; val &= ~RTL8211F_TX_DELAY;
phy_write(phydev, 0x11, reg); ret = rtl8211x_page_write(phydev, 0xd08, 0x11, val);
/* restore to default page 0 */ if (ret)
phy_write(phydev, RTL821x_PAGE_SELECT, 0x0); return ret;
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment