forked from mirrors/linux
		
	net: dsa: vsc73xx: check busy flag in MDIO operations
The VSC73xx has a busy flag used during MDIO operations. It is raised
when MDIO read/write operations are in progress. Without it, PHYs are
misconfigured and bus operations do not work as expected.
Fixes: 05bd97fc55 ("net: dsa: Add Vitesse VSC73xx DSA router driver")
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
Reviewed-by: Florian Fainelli <florian.fainelli@broadcom.com>
Signed-off-by: Pawel Dembicki <paweldembicki@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
			
			
This commit is contained in:
		
							parent
							
								
									5b9eebc2c7
								
							
						
					
					
						commit
						fa63c6434b
					
				
					 1 changed files with 36 additions and 1 deletions
				
			
		|  | @ -40,6 +40,10 @@ | ||||||
| #define VSC73XX_BLOCK_ARBITER	0x5 /* Only subblock 0 */ | #define VSC73XX_BLOCK_ARBITER	0x5 /* Only subblock 0 */ | ||||||
| #define VSC73XX_BLOCK_SYSTEM	0x7 /* Only subblock 0 */ | #define VSC73XX_BLOCK_SYSTEM	0x7 /* Only subblock 0 */ | ||||||
| 
 | 
 | ||||||
|  | /* MII Block subblock */ | ||||||
|  | #define VSC73XX_BLOCK_MII_INTERNAL	0x0 /* Internal MDIO subblock */ | ||||||
|  | #define VSC73XX_BLOCK_MII_EXTERNAL	0x1 /* External MDIO subblock */ | ||||||
|  | 
 | ||||||
| #define CPU_PORT	6 /* CPU port */ | #define CPU_PORT	6 /* CPU port */ | ||||||
| 
 | 
 | ||||||
| /* MAC Block registers */ | /* MAC Block registers */ | ||||||
|  | @ -225,6 +229,8 @@ | ||||||
| #define VSC73XX_MII_CMD		0x1 | #define VSC73XX_MII_CMD		0x1 | ||||||
| #define VSC73XX_MII_DATA	0x2 | #define VSC73XX_MII_DATA	0x2 | ||||||
| 
 | 
 | ||||||
|  | #define VSC73XX_MII_STAT_BUSY	BIT(3) | ||||||
|  | 
 | ||||||
| /* Arbiter block 5 registers */ | /* Arbiter block 5 registers */ | ||||||
| #define VSC73XX_ARBEMPTY		0x0c | #define VSC73XX_ARBEMPTY		0x0c | ||||||
| #define VSC73XX_ARBDISC			0x0e | #define VSC73XX_ARBDISC			0x0e | ||||||
|  | @ -299,6 +305,7 @@ | ||||||
| #define IS_739X(a) (IS_7395(a) || IS_7398(a)) | #define IS_739X(a) (IS_7395(a) || IS_7398(a)) | ||||||
| 
 | 
 | ||||||
| #define VSC73XX_POLL_SLEEP_US		1000 | #define VSC73XX_POLL_SLEEP_US		1000 | ||||||
|  | #define VSC73XX_MDIO_POLL_SLEEP_US	5 | ||||||
| #define VSC73XX_POLL_TIMEOUT_US		10000 | #define VSC73XX_POLL_TIMEOUT_US		10000 | ||||||
| 
 | 
 | ||||||
| struct vsc73xx_counter { | struct vsc73xx_counter { | ||||||
|  | @ -527,6 +534,22 @@ static int vsc73xx_detect(struct vsc73xx *vsc) | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static int vsc73xx_mdio_busy_check(struct vsc73xx *vsc) | ||||||
|  | { | ||||||
|  | 	int ret, err; | ||||||
|  | 	u32 val; | ||||||
|  | 
 | ||||||
|  | 	ret = read_poll_timeout(vsc73xx_read, err, | ||||||
|  | 				err < 0 || !(val & VSC73XX_MII_STAT_BUSY), | ||||||
|  | 				VSC73XX_MDIO_POLL_SLEEP_US, | ||||||
|  | 				VSC73XX_POLL_TIMEOUT_US, false, vsc, | ||||||
|  | 				VSC73XX_BLOCK_MII, VSC73XX_BLOCK_MII_INTERNAL, | ||||||
|  | 				VSC73XX_MII_STAT, &val); | ||||||
|  | 	if (ret) | ||||||
|  | 		return ret; | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int vsc73xx_phy_read(struct dsa_switch *ds, int phy, int regnum) | static int vsc73xx_phy_read(struct dsa_switch *ds, int phy, int regnum) | ||||||
| { | { | ||||||
| 	struct vsc73xx *vsc = ds->priv; | 	struct vsc73xx *vsc = ds->priv; | ||||||
|  | @ -534,12 +557,20 @@ static int vsc73xx_phy_read(struct dsa_switch *ds, int phy, int regnum) | ||||||
| 	u32 val; | 	u32 val; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
|  | 	ret = vsc73xx_mdio_busy_check(vsc); | ||||||
|  | 	if (ret) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
| 	/* Setting bit 26 means "read" */ | 	/* Setting bit 26 means "read" */ | ||||||
| 	cmd = BIT(26) | (phy << 21) | (regnum << 16); | 	cmd = BIT(26) | (phy << 21) | (regnum << 16); | ||||||
| 	ret = vsc73xx_write(vsc, VSC73XX_BLOCK_MII, 0, 1, cmd); | 	ret = vsc73xx_write(vsc, VSC73XX_BLOCK_MII, 0, 1, cmd); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		return ret; | 		return ret; | ||||||
| 	msleep(2); | 
 | ||||||
|  | 	ret = vsc73xx_mdio_busy_check(vsc); | ||||||
|  | 	if (ret) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
| 	ret = vsc73xx_read(vsc, VSC73XX_BLOCK_MII, 0, 2, &val); | 	ret = vsc73xx_read(vsc, VSC73XX_BLOCK_MII, 0, 2, &val); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		return ret; | 		return ret; | ||||||
|  | @ -563,6 +594,10 @@ static int vsc73xx_phy_write(struct dsa_switch *ds, int phy, int regnum, | ||||||
| 	u32 cmd; | 	u32 cmd; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
|  | 	ret = vsc73xx_mdio_busy_check(vsc); | ||||||
|  | 	if (ret) | ||||||
|  | 		return ret; | ||||||
|  | 
 | ||||||
| 	/* It was found through tedious experiments that this router
 | 	/* It was found through tedious experiments that this router
 | ||||||
| 	 * chip really hates to have it's PHYs reset. They | 	 * chip really hates to have it's PHYs reset. They | ||||||
| 	 * never recover if that happens: autonegotiation stops | 	 * never recover if that happens: autonegotiation stops | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Pawel Dembicki
						Pawel Dembicki