1 // SPDX-License-Identifier: GPL-2.0
2 /* Driver for the Texas Instruments DP83822, DP83825 and DP83826 PHYs.
4 * Copyright (C) 2017 Texas Instruments Inc.
7 #include <linux/ethtool.h>
8 #include <linux/etherdevice.h>
9 #include <linux/kernel.h>
10 #include <linux/mii.h>
11 #include <linux/module.h>
13 #include <linux/phy.h>
14 #include <linux/netdevice.h>
15 #include <linux/bitfield.h>
17 #define DP83822_PHY_ID 0x2000a240
18 #define DP83825S_PHY_ID 0x2000a140
19 #define DP83825I_PHY_ID 0x2000a150
20 #define DP83825CM_PHY_ID 0x2000a160
21 #define DP83825CS_PHY_ID 0x2000a170
22 #define DP83826C_PHY_ID 0x2000a130
23 #define DP83826NC_PHY_ID 0x2000a110
25 #define DP83822_DEVADDR 0x1f
27 #define MII_DP83822_CTRL_2 0x0a
28 #define MII_DP83822_PHYSTS 0x10
29 #define MII_DP83822_PHYSCR 0x11
30 #define MII_DP83822_MISR1 0x12
31 #define MII_DP83822_MISR2 0x13
32 #define MII_DP83822_FCSCR 0x14
33 #define MII_DP83822_RCSR 0x17
34 #define MII_DP83822_RESET_CTRL 0x1f
35 #define MII_DP83822_GENCFG 0x465
36 #define MII_DP83822_SOR1 0x467
38 /* DP83826 specific registers */
39 #define MII_DP83826_VOD_CFG1 0x30b
40 #define MII_DP83826_VOD_CFG2 0x30c
43 #define DP83822_SIG_DET_LOW BIT(0)
45 /* Control Register 2 bits */
46 #define DP83822_FX_ENABLE BIT(14)
48 #define DP83822_SW_RESET BIT(15)
49 #define DP83822_DIG_RESTART BIT(14)
52 #define DP83822_PHYSTS_DUPLEX BIT(2)
53 #define DP83822_PHYSTS_10 BIT(1)
54 #define DP83822_PHYSTS_LINK BIT(0)
56 /* PHYSCR Register Fields */
57 #define DP83822_PHYSCR_INT_OE BIT(0) /* Interrupt Output Enable */
58 #define DP83822_PHYSCR_INTEN BIT(1) /* Interrupt Enable */
61 #define DP83822_RX_ERR_HF_INT_EN BIT(0)
62 #define DP83822_FALSE_CARRIER_HF_INT_EN BIT(1)
63 #define DP83822_ANEG_COMPLETE_INT_EN BIT(2)
64 #define DP83822_DUP_MODE_CHANGE_INT_EN BIT(3)
65 #define DP83822_SPEED_CHANGED_INT_EN BIT(4)
66 #define DP83822_LINK_STAT_INT_EN BIT(5)
67 #define DP83822_ENERGY_DET_INT_EN BIT(6)
68 #define DP83822_LINK_QUAL_INT_EN BIT(7)
71 #define DP83822_JABBER_DET_INT_EN BIT(0)
72 #define DP83822_WOL_PKT_INT_EN BIT(1)
73 #define DP83822_SLEEP_MODE_INT_EN BIT(2)
74 #define DP83822_MDI_XOVER_INT_EN BIT(3)
75 #define DP83822_LB_FIFO_INT_EN BIT(4)
76 #define DP83822_PAGE_RX_INT_EN BIT(5)
77 #define DP83822_ANEG_ERR_INT_EN BIT(6)
78 #define DP83822_EEE_ERROR_CHANGE_INT_EN BIT(7)
81 #define DP83822_WOL_INT_EN BIT(4)
82 #define DP83822_WOL_INT_STAT BIT(12)
84 #define MII_DP83822_RXSOP1 0x04a5
85 #define MII_DP83822_RXSOP2 0x04a6
86 #define MII_DP83822_RXSOP3 0x04a7
89 #define MII_DP83822_WOL_CFG 0x04a0
90 #define MII_DP83822_WOL_STAT 0x04a1
91 #define MII_DP83822_WOL_DA1 0x04a2
92 #define MII_DP83822_WOL_DA2 0x04a3
93 #define MII_DP83822_WOL_DA3 0x04a4
96 #define DP83822_WOL_MAGIC_EN BIT(0)
97 #define DP83822_WOL_SECURE_ON BIT(5)
98 #define DP83822_WOL_EN BIT(7)
99 #define DP83822_WOL_INDICATION_SEL BIT(8)
100 #define DP83822_WOL_CLR_INDICATION BIT(11)
103 #define DP83822_RMII_MODE_EN BIT(5)
104 #define DP83822_RMII_MODE_SEL BIT(7)
105 #define DP83822_RGMII_MODE_EN BIT(9)
106 #define DP83822_RX_CLK_SHIFT BIT(12)
107 #define DP83822_TX_CLK_SHIFT BIT(11)
110 #define DP83822_STRAP_MODE1 0
111 #define DP83822_STRAP_MODE2 BIT(0)
112 #define DP83822_STRAP_MODE3 BIT(1)
113 #define DP83822_STRAP_MODE4 GENMASK(1, 0)
115 #define DP83822_COL_STRAP_MASK GENMASK(11, 10)
116 #define DP83822_COL_SHIFT 10
117 #define DP83822_RX_ER_STR_MASK GENMASK(9, 8)
118 #define DP83822_RX_ER_SHIFT 8
120 /* DP83826: VOD_CFG1 & VOD_CFG2 */
121 #define DP83826_VOD_CFG1_MINUS_MDIX_MASK GENMASK(13, 12)
122 #define DP83826_VOD_CFG1_MINUS_MDI_MASK GENMASK(11, 6)
123 #define DP83826_VOD_CFG2_MINUS_MDIX_MASK GENMASK(15, 12)
124 #define DP83826_VOD_CFG2_PLUS_MDIX_MASK GENMASK(11, 6)
125 #define DP83826_VOD_CFG2_PLUS_MDI_MASK GENMASK(5, 0)
126 #define DP83826_CFG_DAC_MINUS_MDIX_5_TO_4 GENMASK(5, 4)
127 #define DP83826_CFG_DAC_MINUS_MDIX_3_TO_0 GENMASK(3, 0)
128 #define DP83826_CFG_DAC_PERCENT_PER_STEP 625
129 #define DP83826_CFG_DAC_PERCENT_DEFAULT 10000
130 #define DP83826_CFG_DAC_MINUS_DEFAULT 0x30
131 #define DP83826_CFG_DAC_PLUS_DEFAULT 0x10
133 #define MII_DP83822_FIBER_ADVERTISE (ADVERTISED_TP | ADVERTISED_MII | \
135 ADVERTISED_Pause | ADVERTISED_Asym_Pause)
137 struct dp83822_private
{
138 bool fx_signal_det_low
;
143 struct ethtool_wolinfo wol
;
146 static int dp83822_config_wol(struct phy_device
*phydev
,
147 struct ethtool_wolinfo
*wol
)
149 struct net_device
*ndev
= phydev
->attached_dev
;
153 if (wol
->wolopts
& (WAKE_MAGIC
| WAKE_MAGICSECURE
)) {
154 mac
= (const u8
*)ndev
->dev_addr
;
156 if (!is_valid_ether_addr(mac
))
159 /* MAC addresses start with byte 5, but stored in mac[0].
160 * 822 PHYs store bytes 4|5, 2|3, 0|1
162 phy_write_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_DA1
,
163 (mac
[1] << 8) | mac
[0]);
164 phy_write_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_DA2
,
165 (mac
[3] << 8) | mac
[2]);
166 phy_write_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_DA3
,
167 (mac
[5] << 8) | mac
[4]);
169 value
= phy_read_mmd(phydev
, DP83822_DEVADDR
,
170 MII_DP83822_WOL_CFG
);
171 if (wol
->wolopts
& WAKE_MAGIC
)
172 value
|= DP83822_WOL_MAGIC_EN
;
174 value
&= ~DP83822_WOL_MAGIC_EN
;
176 if (wol
->wolopts
& WAKE_MAGICSECURE
) {
177 phy_write_mmd(phydev
, DP83822_DEVADDR
,
179 (wol
->sopass
[1] << 8) | wol
->sopass
[0]);
180 phy_write_mmd(phydev
, DP83822_DEVADDR
,
182 (wol
->sopass
[3] << 8) | wol
->sopass
[2]);
183 phy_write_mmd(phydev
, DP83822_DEVADDR
,
185 (wol
->sopass
[5] << 8) | wol
->sopass
[4]);
186 value
|= DP83822_WOL_SECURE_ON
;
188 value
&= ~DP83822_WOL_SECURE_ON
;
191 /* Clear any pending WoL interrupt */
192 phy_read(phydev
, MII_DP83822_MISR2
);
194 value
|= DP83822_WOL_EN
| DP83822_WOL_INDICATION_SEL
|
195 DP83822_WOL_CLR_INDICATION
;
197 return phy_write_mmd(phydev
, DP83822_DEVADDR
,
198 MII_DP83822_WOL_CFG
, value
);
200 return phy_clear_bits_mmd(phydev
, DP83822_DEVADDR
,
203 DP83822_WOL_MAGIC_EN
|
204 DP83822_WOL_SECURE_ON
);
208 static int dp83822_set_wol(struct phy_device
*phydev
,
209 struct ethtool_wolinfo
*wol
)
211 struct dp83822_private
*dp83822
= phydev
->priv
;
214 ret
= dp83822_config_wol(phydev
, wol
);
216 memcpy(&dp83822
->wol
, wol
, sizeof(*wol
));
220 static void dp83822_get_wol(struct phy_device
*phydev
,
221 struct ethtool_wolinfo
*wol
)
226 wol
->supported
= (WAKE_MAGIC
| WAKE_MAGICSECURE
);
229 value
= phy_read_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_CFG
);
231 if (value
& DP83822_WOL_MAGIC_EN
)
232 wol
->wolopts
|= WAKE_MAGIC
;
234 if (value
& DP83822_WOL_SECURE_ON
) {
235 sopass_val
= phy_read_mmd(phydev
, DP83822_DEVADDR
,
237 wol
->sopass
[0] = (sopass_val
& 0xff);
238 wol
->sopass
[1] = (sopass_val
>> 8);
240 sopass_val
= phy_read_mmd(phydev
, DP83822_DEVADDR
,
242 wol
->sopass
[2] = (sopass_val
& 0xff);
243 wol
->sopass
[3] = (sopass_val
>> 8);
245 sopass_val
= phy_read_mmd(phydev
, DP83822_DEVADDR
,
247 wol
->sopass
[4] = (sopass_val
& 0xff);
248 wol
->sopass
[5] = (sopass_val
>> 8);
250 wol
->wolopts
|= WAKE_MAGICSECURE
;
253 /* WoL is not enabled so set wolopts to 0 */
254 if (!(value
& DP83822_WOL_EN
))
258 static int dp83822_config_intr(struct phy_device
*phydev
)
260 struct dp83822_private
*dp83822
= phydev
->priv
;
265 if (phydev
->interrupts
== PHY_INTERRUPT_ENABLED
) {
266 misr_status
= phy_read(phydev
, MII_DP83822_MISR1
);
270 misr_status
|= (DP83822_LINK_STAT_INT_EN
|
271 DP83822_ENERGY_DET_INT_EN
|
272 DP83822_LINK_QUAL_INT_EN
);
274 if (!dp83822
->fx_enabled
)
275 misr_status
|= DP83822_ANEG_COMPLETE_INT_EN
|
276 DP83822_DUP_MODE_CHANGE_INT_EN
|
277 DP83822_SPEED_CHANGED_INT_EN
;
280 err
= phy_write(phydev
, MII_DP83822_MISR1
, misr_status
);
284 misr_status
= phy_read(phydev
, MII_DP83822_MISR2
);
288 misr_status
|= (DP83822_JABBER_DET_INT_EN
|
289 DP83822_SLEEP_MODE_INT_EN
|
290 DP83822_LB_FIFO_INT_EN
|
291 DP83822_PAGE_RX_INT_EN
|
292 DP83822_EEE_ERROR_CHANGE_INT_EN
);
294 if (!dp83822
->fx_enabled
)
295 misr_status
|= DP83822_ANEG_ERR_INT_EN
|
296 DP83822_WOL_PKT_INT_EN
;
298 err
= phy_write(phydev
, MII_DP83822_MISR2
, misr_status
);
302 physcr_status
= phy_read(phydev
, MII_DP83822_PHYSCR
);
303 if (physcr_status
< 0)
304 return physcr_status
;
306 physcr_status
|= DP83822_PHYSCR_INT_OE
| DP83822_PHYSCR_INTEN
;
309 err
= phy_write(phydev
, MII_DP83822_MISR1
, 0);
313 err
= phy_write(phydev
, MII_DP83822_MISR2
, 0);
317 physcr_status
= phy_read(phydev
, MII_DP83822_PHYSCR
);
318 if (physcr_status
< 0)
319 return physcr_status
;
321 physcr_status
&= ~DP83822_PHYSCR_INTEN
;
324 return phy_write(phydev
, MII_DP83822_PHYSCR
, physcr_status
);
327 static irqreturn_t
dp83822_handle_interrupt(struct phy_device
*phydev
)
329 bool trigger_machine
= false;
332 /* The MISR1 and MISR2 registers are holding the interrupt status in
333 * the upper half (15:8), while the lower half (7:0) is used for
334 * controlling the interrupt enable state of those individual interrupt
335 * sources. To determine the possible interrupt sources, just read the
336 * MISR* register and use it directly to know which interrupts have
337 * been enabled previously or not.
339 irq_status
= phy_read(phydev
, MII_DP83822_MISR1
);
340 if (irq_status
< 0) {
344 if (irq_status
& ((irq_status
& GENMASK(7, 0)) << 8))
345 trigger_machine
= true;
347 irq_status
= phy_read(phydev
, MII_DP83822_MISR2
);
348 if (irq_status
< 0) {
352 if (irq_status
& ((irq_status
& GENMASK(7, 0)) << 8))
353 trigger_machine
= true;
355 if (!trigger_machine
)
358 phy_trigger_machine(phydev
);
363 static int dp83822_read_status(struct phy_device
*phydev
)
365 struct dp83822_private
*dp83822
= phydev
->priv
;
366 int status
= phy_read(phydev
, MII_DP83822_PHYSTS
);
370 if (dp83822
->fx_enabled
) {
371 if (status
& DP83822_PHYSTS_LINK
) {
372 phydev
->speed
= SPEED_UNKNOWN
;
373 phydev
->duplex
= DUPLEX_UNKNOWN
;
375 ctrl2
= phy_read(phydev
, MII_DP83822_CTRL_2
);
379 if (!(ctrl2
& DP83822_FX_ENABLE
)) {
380 ret
= phy_write(phydev
, MII_DP83822_CTRL_2
,
381 DP83822_FX_ENABLE
| ctrl2
);
388 ret
= genphy_read_status(phydev
);
395 if (status
& DP83822_PHYSTS_DUPLEX
)
396 phydev
->duplex
= DUPLEX_FULL
;
398 phydev
->duplex
= DUPLEX_HALF
;
400 if (status
& DP83822_PHYSTS_10
)
401 phydev
->speed
= SPEED_10
;
403 phydev
->speed
= SPEED_100
;
408 static int dp83822_config_init(struct phy_device
*phydev
)
410 struct dp83822_private
*dp83822
= phydev
->priv
;
411 struct device
*dev
= &phydev
->mdio
.dev
;
418 if (phy_interface_is_rgmii(phydev
)) {
419 rx_int_delay
= phy_get_internal_delay(phydev
, dev
, NULL
, 0,
422 /* Set DP83822_RX_CLK_SHIFT to enable rx clk internal delay */
423 if (rx_int_delay
> 0)
424 rgmii_delay
|= DP83822_RX_CLK_SHIFT
;
426 tx_int_delay
= phy_get_internal_delay(phydev
, dev
, NULL
, 0,
429 /* Set DP83822_TX_CLK_SHIFT to disable tx clk internal delay */
430 if (tx_int_delay
<= 0)
431 rgmii_delay
|= DP83822_TX_CLK_SHIFT
;
433 err
= phy_modify_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_RCSR
,
434 DP83822_RX_CLK_SHIFT
| DP83822_TX_CLK_SHIFT
, rgmii_delay
);
438 err
= phy_set_bits_mmd(phydev
, DP83822_DEVADDR
,
439 MII_DP83822_RCSR
, DP83822_RGMII_MODE_EN
);
444 err
= phy_clear_bits_mmd(phydev
, DP83822_DEVADDR
,
445 MII_DP83822_RCSR
, DP83822_RGMII_MODE_EN
);
451 if (dp83822
->fx_enabled
) {
452 err
= phy_modify(phydev
, MII_DP83822_CTRL_2
,
453 DP83822_FX_ENABLE
, 1);
457 /* Only allow advertising what this PHY supports */
458 linkmode_and(phydev
->advertising
, phydev
->advertising
,
461 linkmode_set_bit(ETHTOOL_LINK_MODE_FIBRE_BIT
,
463 linkmode_set_bit(ETHTOOL_LINK_MODE_FIBRE_BIT
,
464 phydev
->advertising
);
465 linkmode_set_bit(ETHTOOL_LINK_MODE_100baseFX_Full_BIT
,
467 linkmode_set_bit(ETHTOOL_LINK_MODE_100baseFX_Half_BIT
,
469 linkmode_set_bit(ETHTOOL_LINK_MODE_100baseFX_Full_BIT
,
470 phydev
->advertising
);
471 linkmode_set_bit(ETHTOOL_LINK_MODE_100baseFX_Half_BIT
,
472 phydev
->advertising
);
474 /* Auto neg is not supported in fiber mode */
475 bmcr
= phy_read(phydev
, MII_BMCR
);
479 if (bmcr
& BMCR_ANENABLE
) {
480 err
= phy_modify(phydev
, MII_BMCR
, BMCR_ANENABLE
, 0);
484 phydev
->autoneg
= AUTONEG_DISABLE
;
485 linkmode_clear_bit(ETHTOOL_LINK_MODE_Autoneg_BIT
,
487 linkmode_clear_bit(ETHTOOL_LINK_MODE_Autoneg_BIT
,
488 phydev
->advertising
);
490 /* Setup fiber advertisement */
491 err
= phy_modify_changed(phydev
, MII_ADVERTISE
,
492 MII_DP83822_FIBER_ADVERTISE
,
493 MII_DP83822_FIBER_ADVERTISE
);
498 if (dp83822
->fx_signal_det_low
) {
499 err
= phy_set_bits_mmd(phydev
, DP83822_DEVADDR
,
501 DP83822_SIG_DET_LOW
);
506 return dp83822_config_wol(phydev
, &dp83822
->wol
);
509 static int dp8382x_config_rmii_mode(struct phy_device
*phydev
)
511 struct device
*dev
= &phydev
->mdio
.dev
;
515 if (!device_property_read_string(dev
, "ti,rmii-mode", &of_val
)) {
516 if (strcmp(of_val
, "master") == 0) {
517 ret
= phy_clear_bits_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_RCSR
,
518 DP83822_RMII_MODE_SEL
);
519 } else if (strcmp(of_val
, "slave") == 0) {
520 ret
= phy_set_bits_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_RCSR
,
521 DP83822_RMII_MODE_SEL
);
523 phydev_err(phydev
, "Invalid value for ti,rmii-mode property (%s)\n",
535 static int dp83826_config_init(struct phy_device
*phydev
)
537 struct dp83822_private
*dp83822
= phydev
->priv
;
541 if (phydev
->interface
== PHY_INTERFACE_MODE_RMII
) {
542 ret
= phy_set_bits_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_RCSR
,
543 DP83822_RMII_MODE_EN
);
547 ret
= dp8382x_config_rmii_mode(phydev
);
551 ret
= phy_clear_bits_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_RCSR
,
552 DP83822_RMII_MODE_EN
);
557 if (dp83822
->cfg_dac_minus
!= DP83826_CFG_DAC_MINUS_DEFAULT
) {
558 val
= FIELD_PREP(DP83826_VOD_CFG1_MINUS_MDI_MASK
, dp83822
->cfg_dac_minus
) |
559 FIELD_PREP(DP83826_VOD_CFG1_MINUS_MDIX_MASK
,
560 FIELD_GET(DP83826_CFG_DAC_MINUS_MDIX_5_TO_4
,
561 dp83822
->cfg_dac_minus
));
562 mask
= DP83826_VOD_CFG1_MINUS_MDIX_MASK
| DP83826_VOD_CFG1_MINUS_MDI_MASK
;
563 ret
= phy_modify_mmd(phydev
, DP83822_DEVADDR
, MII_DP83826_VOD_CFG1
, mask
, val
);
567 val
= FIELD_PREP(DP83826_VOD_CFG2_MINUS_MDIX_MASK
,
568 FIELD_GET(DP83826_CFG_DAC_MINUS_MDIX_3_TO_0
,
569 dp83822
->cfg_dac_minus
));
570 mask
= DP83826_VOD_CFG2_MINUS_MDIX_MASK
;
571 ret
= phy_modify_mmd(phydev
, DP83822_DEVADDR
, MII_DP83826_VOD_CFG2
, mask
, val
);
576 if (dp83822
->cfg_dac_plus
!= DP83826_CFG_DAC_PLUS_DEFAULT
) {
577 val
= FIELD_PREP(DP83826_VOD_CFG2_PLUS_MDIX_MASK
, dp83822
->cfg_dac_plus
) |
578 FIELD_PREP(DP83826_VOD_CFG2_PLUS_MDI_MASK
, dp83822
->cfg_dac_plus
);
579 mask
= DP83826_VOD_CFG2_PLUS_MDIX_MASK
| DP83826_VOD_CFG2_PLUS_MDI_MASK
;
580 ret
= phy_modify_mmd(phydev
, DP83822_DEVADDR
, MII_DP83826_VOD_CFG2
, mask
, val
);
585 return dp83822_config_wol(phydev
, &dp83822
->wol
);
588 static int dp83825_config_init(struct phy_device
*phydev
)
590 struct dp83822_private
*dp83822
= phydev
->priv
;
593 ret
= dp8382x_config_rmii_mode(phydev
);
597 return dp83822_config_wol(phydev
, &dp83822
->wol
);
600 static int dp83822_phy_reset(struct phy_device
*phydev
)
604 err
= phy_write(phydev
, MII_DP83822_RESET_CTRL
, DP83822_SW_RESET
);
608 return phydev
->drv
->config_init(phydev
);
611 #ifdef CONFIG_OF_MDIO
612 static int dp83822_of_init(struct phy_device
*phydev
)
614 struct dp83822_private
*dp83822
= phydev
->priv
;
615 struct device
*dev
= &phydev
->mdio
.dev
;
617 /* Signal detection for the PHY is only enabled if the FX_EN and the
618 * SD_EN pins are strapped. Signal detection can only enabled if FX_EN
619 * is strapped otherwise signal detection is disabled for the PHY.
621 if (dp83822
->fx_enabled
&& dp83822
->fx_sd_enable
)
622 dp83822
->fx_signal_det_low
= device_property_present(dev
,
624 if (!dp83822
->fx_enabled
)
625 dp83822
->fx_enabled
= device_property_present(dev
,
631 static int dp83826_to_dac_minus_one_regval(int percent
)
633 int tmp
= DP83826_CFG_DAC_PERCENT_DEFAULT
- percent
;
635 return tmp
/ DP83826_CFG_DAC_PERCENT_PER_STEP
;
638 static int dp83826_to_dac_plus_one_regval(int percent
)
640 int tmp
= percent
- DP83826_CFG_DAC_PERCENT_DEFAULT
;
642 return tmp
/ DP83826_CFG_DAC_PERCENT_PER_STEP
;
645 static void dp83826_of_init(struct phy_device
*phydev
)
647 struct dp83822_private
*dp83822
= phydev
->priv
;
648 struct device
*dev
= &phydev
->mdio
.dev
;
651 dp83822
->cfg_dac_minus
= DP83826_CFG_DAC_MINUS_DEFAULT
;
652 if (!device_property_read_u32(dev
, "ti,cfg-dac-minus-one-bp", &val
))
653 dp83822
->cfg_dac_minus
+= dp83826_to_dac_minus_one_regval(val
);
655 dp83822
->cfg_dac_plus
= DP83826_CFG_DAC_PLUS_DEFAULT
;
656 if (!device_property_read_u32(dev
, "ti,cfg-dac-plus-one-bp", &val
))
657 dp83822
->cfg_dac_plus
+= dp83826_to_dac_plus_one_regval(val
);
660 static int dp83822_of_init(struct phy_device
*phydev
)
665 static void dp83826_of_init(struct phy_device
*phydev
)
668 #endif /* CONFIG_OF_MDIO */
670 static int dp83822_read_straps(struct phy_device
*phydev
)
672 struct dp83822_private
*dp83822
= phydev
->priv
;
673 int fx_enabled
, fx_sd_enable
;
676 val
= phy_read_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_SOR1
);
680 phydev_dbg(phydev
, "SOR1 strap register: 0x%04x\n", val
);
682 fx_enabled
= (val
& DP83822_COL_STRAP_MASK
) >> DP83822_COL_SHIFT
;
683 if (fx_enabled
== DP83822_STRAP_MODE2
||
684 fx_enabled
== DP83822_STRAP_MODE3
)
685 dp83822
->fx_enabled
= 1;
687 if (dp83822
->fx_enabled
) {
688 fx_sd_enable
= (val
& DP83822_RX_ER_STR_MASK
) >> DP83822_RX_ER_SHIFT
;
689 if (fx_sd_enable
== DP83822_STRAP_MODE3
||
690 fx_sd_enable
== DP83822_STRAP_MODE4
)
691 dp83822
->fx_sd_enable
= 1;
697 static int dp8382x_probe(struct phy_device
*phydev
)
699 struct dp83822_private
*dp83822
;
701 dp83822
= devm_kzalloc(&phydev
->mdio
.dev
, sizeof(*dp83822
),
706 phydev
->priv
= dp83822
;
711 static int dp83822_probe(struct phy_device
*phydev
)
713 struct dp83822_private
*dp83822
;
716 ret
= dp8382x_probe(phydev
);
720 dp83822
= phydev
->priv
;
722 ret
= dp83822_read_straps(phydev
);
726 dp83822_of_init(phydev
);
728 if (dp83822
->fx_enabled
)
729 phydev
->port
= PORT_FIBRE
;
734 static int dp83826_probe(struct phy_device
*phydev
)
738 ret
= dp8382x_probe(phydev
);
742 dp83826_of_init(phydev
);
747 static int dp83822_suspend(struct phy_device
*phydev
)
751 value
= phy_read_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_CFG
);
753 if (!(value
& DP83822_WOL_EN
))
754 genphy_suspend(phydev
);
759 static int dp83822_resume(struct phy_device
*phydev
)
763 genphy_resume(phydev
);
765 value
= phy_read_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_CFG
);
767 phy_write_mmd(phydev
, DP83822_DEVADDR
, MII_DP83822_WOL_CFG
, value
|
768 DP83822_WOL_CLR_INDICATION
);
773 #define DP83822_PHY_DRIVER(_id, _name) \
775 PHY_ID_MATCH_MODEL(_id), \
777 /* PHY_BASIC_FEATURES */ \
778 .probe = dp83822_probe, \
779 .soft_reset = dp83822_phy_reset, \
780 .config_init = dp83822_config_init, \
781 .read_status = dp83822_read_status, \
782 .get_wol = dp83822_get_wol, \
783 .set_wol = dp83822_set_wol, \
784 .config_intr = dp83822_config_intr, \
785 .handle_interrupt = dp83822_handle_interrupt, \
786 .suspend = dp83822_suspend, \
787 .resume = dp83822_resume, \
790 #define DP83825_PHY_DRIVER(_id, _name) \
792 PHY_ID_MATCH_MODEL(_id), \
794 /* PHY_BASIC_FEATURES */ \
795 .probe = dp8382x_probe, \
796 .soft_reset = dp83822_phy_reset, \
797 .config_init = dp83825_config_init, \
798 .get_wol = dp83822_get_wol, \
799 .set_wol = dp83822_set_wol, \
800 .config_intr = dp83822_config_intr, \
801 .handle_interrupt = dp83822_handle_interrupt, \
802 .suspend = dp83822_suspend, \
803 .resume = dp83822_resume, \
806 #define DP83826_PHY_DRIVER(_id, _name) \
808 PHY_ID_MATCH_MODEL(_id), \
810 /* PHY_BASIC_FEATURES */ \
811 .probe = dp83826_probe, \
812 .soft_reset = dp83822_phy_reset, \
813 .config_init = dp83826_config_init, \
814 .get_wol = dp83822_get_wol, \
815 .set_wol = dp83822_set_wol, \
816 .config_intr = dp83822_config_intr, \
817 .handle_interrupt = dp83822_handle_interrupt, \
818 .suspend = dp83822_suspend, \
819 .resume = dp83822_resume, \
822 static struct phy_driver dp83822_driver
[] = {
823 DP83822_PHY_DRIVER(DP83822_PHY_ID
, "TI DP83822"),
824 DP83825_PHY_DRIVER(DP83825I_PHY_ID
, "TI DP83825I"),
825 DP83825_PHY_DRIVER(DP83825S_PHY_ID
, "TI DP83825S"),
826 DP83825_PHY_DRIVER(DP83825CM_PHY_ID
, "TI DP83825M"),
827 DP83825_PHY_DRIVER(DP83825CS_PHY_ID
, "TI DP83825CS"),
828 DP83826_PHY_DRIVER(DP83826C_PHY_ID
, "TI DP83826C"),
829 DP83826_PHY_DRIVER(DP83826NC_PHY_ID
, "TI DP83826NC"),
831 module_phy_driver(dp83822_driver
);
833 static struct mdio_device_id __maybe_unused dp83822_tbl
[] = {
834 { DP83822_PHY_ID
, 0xfffffff0 },
835 { DP83825I_PHY_ID
, 0xfffffff0 },
836 { DP83826C_PHY_ID
, 0xfffffff0 },
837 { DP83826NC_PHY_ID
, 0xfffffff0 },
838 { DP83825S_PHY_ID
, 0xfffffff0 },
839 { DP83825CM_PHY_ID
, 0xfffffff0 },
840 { DP83825CS_PHY_ID
, 0xfffffff0 },
843 MODULE_DEVICE_TABLE(mdio
, dp83822_tbl
);
845 MODULE_DESCRIPTION("Texas Instruments DP83822 PHY driver");
846 MODULE_AUTHOR("Dan Murphy <dmurphy@ti.com");
847 MODULE_LICENSE("GPL v2");