diff options
Diffstat (limited to 'drivers/net/phy/broadcom.c')
-rw-r--r-- | drivers/net/phy/broadcom.c | 90 |
1 files changed, 42 insertions, 48 deletions
diff --git a/drivers/net/phy/broadcom.c b/drivers/net/phy/broadcom.c index f81e53222230..9e574ceacf43 100644 --- a/drivers/net/phy/broadcom.c +++ b/drivers/net/phy/broadcom.c | |||
@@ -237,53 +237,78 @@ static int bcm54xx_auxctl_write(struct phy_device *phydev, u16 regnum, u16 val) | |||
237 | return phy_write(phydev, MII_BCM54XX_AUX_CTL, regnum | val); | 237 | return phy_write(phydev, MII_BCM54XX_AUX_CTL, regnum | val); |
238 | } | 238 | } |
239 | 239 | ||
240 | /* Needs SMDSP clock enabled via bcm54xx_phydsp_config() */ | ||
240 | static int bcm50610_a0_workaround(struct phy_device *phydev) | 241 | static int bcm50610_a0_workaround(struct phy_device *phydev) |
241 | { | 242 | { |
242 | int err; | 243 | int err; |
243 | 244 | ||
244 | err = bcm54xx_auxctl_write(phydev, | ||
245 | MII_BCM54XX_AUXCTL_SHDWSEL_AUXCTL, | ||
246 | MII_BCM54XX_AUXCTL_ACTL_SMDSP_ENA | | ||
247 | MII_BCM54XX_AUXCTL_ACTL_TX_6DB); | ||
248 | if (err < 0) | ||
249 | return err; | ||
250 | |||
251 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP08, | 245 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP08, |
252 | MII_BCM54XX_EXP_EXP08_RJCT_2MHZ | | 246 | MII_BCM54XX_EXP_EXP08_RJCT_2MHZ | |
253 | MII_BCM54XX_EXP_EXP08_EARLY_DAC_WAKE); | 247 | MII_BCM54XX_EXP_EXP08_EARLY_DAC_WAKE); |
254 | if (err < 0) | 248 | if (err < 0) |
255 | goto error; | 249 | return err; |
256 | 250 | ||
257 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_AADJ1CH0, | 251 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_AADJ1CH0, |
258 | MII_BCM54XX_EXP_AADJ1CH0_SWP_ABCD_OEN | | 252 | MII_BCM54XX_EXP_AADJ1CH0_SWP_ABCD_OEN | |
259 | MII_BCM54XX_EXP_AADJ1CH0_SWSEL_THPF); | 253 | MII_BCM54XX_EXP_AADJ1CH0_SWSEL_THPF); |
260 | if (err < 0) | 254 | if (err < 0) |
261 | goto error; | 255 | return err; |
262 | 256 | ||
263 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_AADJ1CH3, | 257 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_AADJ1CH3, |
264 | MII_BCM54XX_EXP_AADJ1CH3_ADCCKADJ); | 258 | MII_BCM54XX_EXP_AADJ1CH3_ADCCKADJ); |
265 | if (err < 0) | 259 | if (err < 0) |
266 | goto error; | 260 | return err; |
267 | 261 | ||
268 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP75, | 262 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP75, |
269 | MII_BCM54XX_EXP_EXP75_VDACCTRL); | 263 | MII_BCM54XX_EXP_EXP75_VDACCTRL); |
270 | if (err < 0) | 264 | if (err < 0) |
271 | goto error; | 265 | return err; |
272 | 266 | ||
273 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP96, | 267 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP96, |
274 | MII_BCM54XX_EXP_EXP96_MYST); | 268 | MII_BCM54XX_EXP_EXP96_MYST); |
275 | if (err < 0) | 269 | if (err < 0) |
276 | goto error; | 270 | return err; |
277 | 271 | ||
278 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP97, | 272 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP97, |
279 | MII_BCM54XX_EXP_EXP97_MYST); | 273 | MII_BCM54XX_EXP_EXP97_MYST); |
280 | 274 | ||
275 | return err; | ||
276 | } | ||
277 | |||
278 | static int bcm54xx_phydsp_config(struct phy_device *phydev) | ||
279 | { | ||
280 | int err, err2; | ||
281 | |||
282 | /* Enable the SMDSP clock */ | ||
283 | err = bcm54xx_auxctl_write(phydev, | ||
284 | MII_BCM54XX_AUXCTL_SHDWSEL_AUXCTL, | ||
285 | MII_BCM54XX_AUXCTL_ACTL_SMDSP_ENA | | ||
286 | MII_BCM54XX_AUXCTL_ACTL_TX_6DB); | ||
287 | if (err < 0) | ||
288 | return err; | ||
289 | |||
290 | if (phydev->drv->phy_id == PHY_ID_BCM50610) | ||
291 | err = bcm50610_a0_workaround(phydev); | ||
292 | |||
293 | if (BRCM_PHY_MODEL(phydev) == PHY_ID_BCM57780) { | ||
294 | int val; | ||
295 | |||
296 | val = bcm54xx_exp_read(phydev, MII_BCM54XX_EXP_EXP75); | ||
297 | if (val < 0) | ||
298 | goto error; | ||
299 | |||
300 | val |= MII_BCM54XX_EXP_EXP75_CM_OSC; | ||
301 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP75, val); | ||
302 | } | ||
303 | |||
281 | error: | 304 | error: |
282 | bcm54xx_auxctl_write(phydev, | 305 | /* Disable the SMDSP clock */ |
283 | MII_BCM54XX_AUXCTL_SHDWSEL_AUXCTL, | 306 | err2 = bcm54xx_auxctl_write(phydev, |
284 | MII_BCM54XX_AUXCTL_ACTL_TX_6DB); | 307 | MII_BCM54XX_AUXCTL_SHDWSEL_AUXCTL, |
308 | MII_BCM54XX_AUXCTL_ACTL_TX_6DB); | ||
285 | 309 | ||
286 | return err; | 310 | /* Return the first error reported. */ |
311 | return err ? err : err2; | ||
287 | } | 312 | } |
288 | 313 | ||
289 | static int bcm54xx_config_init(struct phy_device *phydev) | 314 | static int bcm54xx_config_init(struct phy_device *phydev) |
@@ -308,38 +333,7 @@ static int bcm54xx_config_init(struct phy_device *phydev) | |||
308 | if (err < 0) | 333 | if (err < 0) |
309 | return err; | 334 | return err; |
310 | 335 | ||
311 | if (phydev->drv->phy_id == PHY_ID_BCM50610) { | 336 | bcm54xx_phydsp_config(phydev); |
312 | err = bcm50610_a0_workaround(phydev); | ||
313 | if (err < 0) | ||
314 | return err; | ||
315 | } | ||
316 | |||
317 | if (BRCM_PHY_MODEL(phydev) == PHY_ID_BCM57780) { | ||
318 | int err2; | ||
319 | |||
320 | err = bcm54xx_auxctl_write(phydev, | ||
321 | MII_BCM54XX_AUXCTL_SHDWSEL_AUXCTL, | ||
322 | MII_BCM54XX_AUXCTL_ACTL_SMDSP_ENA | | ||
323 | MII_BCM54XX_AUXCTL_ACTL_TX_6DB); | ||
324 | if (err < 0) | ||
325 | return err; | ||
326 | |||
327 | reg = bcm54xx_exp_read(phydev, MII_BCM54XX_EXP_EXP75); | ||
328 | if (reg < 0) | ||
329 | goto error; | ||
330 | |||
331 | reg |= MII_BCM54XX_EXP_EXP75_CM_OSC; | ||
332 | err = bcm54xx_exp_write(phydev, MII_BCM54XX_EXP_EXP75, reg); | ||
333 | |||
334 | error: | ||
335 | err2 = bcm54xx_auxctl_write(phydev, | ||
336 | MII_BCM54XX_AUXCTL_SHDWSEL_AUXCTL, | ||
337 | MII_BCM54XX_AUXCTL_ACTL_TX_6DB); | ||
338 | if (err) | ||
339 | return err; | ||
340 | if (err2) | ||
341 | return err2; | ||
342 | } | ||
343 | 337 | ||
344 | return 0; | 338 | return 0; |
345 | } | 339 | } |