diff options
Diffstat (limited to 'drivers/net/phy/mdio-bitbang.c')
-rw-r--r-- | drivers/net/phy/mdio-bitbang.c | 60 |
1 files changed, 51 insertions, 9 deletions
diff --git a/drivers/net/phy/mdio-bitbang.c b/drivers/net/phy/mdio-bitbang.c index 2576055b350b..0ff06617a4ab 100644 --- a/drivers/net/phy/mdio-bitbang.c +++ b/drivers/net/phy/mdio-bitbang.c | |||
@@ -23,8 +23,13 @@ | |||
23 | #include <linux/types.h> | 23 | #include <linux/types.h> |
24 | #include <linux/delay.h> | 24 | #include <linux/delay.h> |
25 | 25 | ||
26 | #define MDIO_READ 1 | 26 | #define MDIO_READ 2 |
27 | #define MDIO_WRITE 0 | 27 | #define MDIO_WRITE 1 |
28 | |||
29 | #define MDIO_C45 (1<<15) | ||
30 | #define MDIO_C45_ADDR (MDIO_C45 | 0) | ||
31 | #define MDIO_C45_READ (MDIO_C45 | 3) | ||
32 | #define MDIO_C45_WRITE (MDIO_C45 | 1) | ||
28 | 33 | ||
29 | #define MDIO_SETUP_TIME 10 | 34 | #define MDIO_SETUP_TIME 10 |
30 | #define MDIO_HOLD_TIME 10 | 35 | #define MDIO_HOLD_TIME 10 |
@@ -90,7 +95,7 @@ static u16 mdiobb_get_num(struct mdiobb_ctrl *ctrl, int bits) | |||
90 | /* Utility to send the preamble, address, and | 95 | /* Utility to send the preamble, address, and |
91 | * register (common to read and write). | 96 | * register (common to read and write). |
92 | */ | 97 | */ |
93 | static void mdiobb_cmd(struct mdiobb_ctrl *ctrl, int read, u8 phy, u8 reg) | 98 | static void mdiobb_cmd(struct mdiobb_ctrl *ctrl, int op, u8 phy, u8 reg) |
94 | { | 99 | { |
95 | const struct mdiobb_ops *ops = ctrl->ops; | 100 | const struct mdiobb_ops *ops = ctrl->ops; |
96 | int i; | 101 | int i; |
@@ -109,23 +114,56 @@ static void mdiobb_cmd(struct mdiobb_ctrl *ctrl, int read, u8 phy, u8 reg) | |||
109 | for (i = 0; i < 32; i++) | 114 | for (i = 0; i < 32; i++) |
110 | mdiobb_send_bit(ctrl, 1); | 115 | mdiobb_send_bit(ctrl, 1); |
111 | 116 | ||
112 | /* send the start bit (01) and the read opcode (10) or write (10) */ | 117 | /* send the start bit (01) and the read opcode (10) or write (10). |
118 | Clause 45 operation uses 00 for the start and 11, 10 for | ||
119 | read/write */ | ||
113 | mdiobb_send_bit(ctrl, 0); | 120 | mdiobb_send_bit(ctrl, 0); |
114 | mdiobb_send_bit(ctrl, 1); | 121 | if (op & MDIO_C45) |
115 | mdiobb_send_bit(ctrl, read); | 122 | mdiobb_send_bit(ctrl, 0); |
116 | mdiobb_send_bit(ctrl, !read); | 123 | else |
124 | mdiobb_send_bit(ctrl, 1); | ||
125 | mdiobb_send_bit(ctrl, (op >> 1) & 1); | ||
126 | mdiobb_send_bit(ctrl, (op >> 0) & 1); | ||
117 | 127 | ||
118 | mdiobb_send_num(ctrl, phy, 5); | 128 | mdiobb_send_num(ctrl, phy, 5); |
119 | mdiobb_send_num(ctrl, reg, 5); | 129 | mdiobb_send_num(ctrl, reg, 5); |
120 | } | 130 | } |
121 | 131 | ||
132 | /* In clause 45 mode all commands are prefixed by MDIO_ADDR to specify the | ||
133 | lower 16 bits of the 21 bit address. This transfer is done identically to a | ||
134 | MDIO_WRITE except for a different code. To enable clause 45 mode or | ||
135 | MII_ADDR_C45 into the address. Theoretically clause 45 and normal devices | ||
136 | can exist on the same bus. Normal devices should ignore the MDIO_ADDR | ||
137 | phase. */ | ||
138 | static int mdiobb_cmd_addr(struct mdiobb_ctrl *ctrl, int phy, u32 addr) | ||
139 | { | ||
140 | unsigned int dev_addr = (addr >> 16) & 0x1F; | ||
141 | unsigned int reg = addr & 0xFFFF; | ||
142 | mdiobb_cmd(ctrl, MDIO_C45_ADDR, phy, dev_addr); | ||
143 | |||
144 | /* send the turnaround (10) */ | ||
145 | mdiobb_send_bit(ctrl, 1); | ||
146 | mdiobb_send_bit(ctrl, 0); | ||
147 | |||
148 | mdiobb_send_num(ctrl, reg, 16); | ||
149 | |||
150 | ctrl->ops->set_mdio_dir(ctrl, 0); | ||
151 | mdiobb_get_bit(ctrl); | ||
152 | |||
153 | return dev_addr; | ||
154 | } | ||
122 | 155 | ||
123 | static int mdiobb_read(struct mii_bus *bus, int phy, int reg) | 156 | static int mdiobb_read(struct mii_bus *bus, int phy, int reg) |
124 | { | 157 | { |
125 | struct mdiobb_ctrl *ctrl = bus->priv; | 158 | struct mdiobb_ctrl *ctrl = bus->priv; |
126 | int ret, i; | 159 | int ret, i; |
127 | 160 | ||
128 | mdiobb_cmd(ctrl, MDIO_READ, phy, reg); | 161 | if (reg & MII_ADDR_C45) { |
162 | reg = mdiobb_cmd_addr(ctrl, phy, reg); | ||
163 | mdiobb_cmd(ctrl, MDIO_C45_READ, phy, reg); | ||
164 | } else | ||
165 | mdiobb_cmd(ctrl, MDIO_READ, phy, reg); | ||
166 | |||
129 | ctrl->ops->set_mdio_dir(ctrl, 0); | 167 | ctrl->ops->set_mdio_dir(ctrl, 0); |
130 | 168 | ||
131 | /* check the turnaround bit: the PHY should be driving it to zero */ | 169 | /* check the turnaround bit: the PHY should be driving it to zero */ |
@@ -148,7 +186,11 @@ static int mdiobb_write(struct mii_bus *bus, int phy, int reg, u16 val) | |||
148 | { | 186 | { |
149 | struct mdiobb_ctrl *ctrl = bus->priv; | 187 | struct mdiobb_ctrl *ctrl = bus->priv; |
150 | 188 | ||
151 | mdiobb_cmd(ctrl, MDIO_WRITE, phy, reg); | 189 | if (reg & MII_ADDR_C45) { |
190 | reg = mdiobb_cmd_addr(ctrl, phy, reg); | ||
191 | mdiobb_cmd(ctrl, MDIO_C45_WRITE, phy, reg); | ||
192 | } else | ||
193 | mdiobb_cmd(ctrl, MDIO_WRITE, phy, reg); | ||
152 | 194 | ||
153 | /* send the turnaround (10) */ | 195 | /* send the turnaround (10) */ |
154 | mdiobb_send_bit(ctrl, 1); | 196 | mdiobb_send_bit(ctrl, 1); |