at803x.c 11.4 KB
Newer Older
M
Matus Ujhelyi 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * drivers/net/phy/at803x.c
 *
 * Driver for Atheros 803x PHY
 *
 * Author: Matus Ujhelyi <ujhelyi.m@gmail.com>
 *
 * This program is free software; you can redistribute  it and/or modify it
 * under  the terms of  the GNU General  Public License as published by the
 * Free Software Foundation;  either version 2 of the  License, or (at your
 * option) any later version.
 */

#include <linux/phy.h>
#include <linux/module.h>
#include <linux/string.h>
#include <linux/netdevice.h>
#include <linux/etherdevice.h>
19 20
#include <linux/of_gpio.h>
#include <linux/gpio/consumer.h>
M
Matus Ujhelyi 已提交
21 22

#define AT803X_INTR_ENABLE			0x12
23 24 25 26 27 28 29 30 31 32
#define AT803X_INTR_ENABLE_AUTONEG_ERR		BIT(15)
#define AT803X_INTR_ENABLE_SPEED_CHANGED	BIT(14)
#define AT803X_INTR_ENABLE_DUPLEX_CHANGED	BIT(13)
#define AT803X_INTR_ENABLE_PAGE_RECEIVED	BIT(12)
#define AT803X_INTR_ENABLE_LINK_FAIL		BIT(11)
#define AT803X_INTR_ENABLE_LINK_SUCCESS		BIT(10)
#define AT803X_INTR_ENABLE_WIRESPEED_DOWNGRADE	BIT(5)
#define AT803X_INTR_ENABLE_POLARITY_CHANGED	BIT(1)
#define AT803X_INTR_ENABLE_WOL			BIT(0)

M
Matus Ujhelyi 已提交
33
#define AT803X_INTR_STATUS			0x13
34

35 36
#define AT803X_SMART_SPEED			0x14
#define AT803X_LED_CONTROL			0x18
37

M
Matus Ujhelyi 已提交
38 39 40 41 42 43 44
#define AT803X_DEVICE_ADDR			0x03
#define AT803X_LOC_MAC_ADDR_0_15_OFFSET		0x804C
#define AT803X_LOC_MAC_ADDR_16_31_OFFSET	0x804B
#define AT803X_LOC_MAC_ADDR_32_47_OFFSET	0x804A
#define AT803X_MMD_ACCESS_CONTROL		0x0D
#define AT803X_MMD_ACCESS_CONTROL_DATA		0x0E
#define AT803X_FUNC_DATA			0x4003
45

46 47
#define AT803X_DEBUG_ADDR			0x1D
#define AT803X_DEBUG_DATA			0x1E
48

49 50
#define AT803X_DEBUG_REG_0			0x00
#define AT803X_DEBUG_RX_CLK_DLY_EN		BIT(15)
51

52 53
#define AT803X_DEBUG_REG_5			0x05
#define AT803X_DEBUG_TX_CLK_DLY_EN		BIT(8)
M
Matus Ujhelyi 已提交
54

55 56 57
#define AT803X_REG_CHIP_CONFIG			0x1f
#define AT803X_BT_BX_REG_SEL			0x8000

58 59 60
#define ATH8030_PHY_ID 0x004dd076
#define ATH8031_PHY_ID 0x004dd074
#define ATH8035_PHY_ID 0x004dd072
61
#define AT803X_PHY_ID_MASK			0xffffffef
62

M
Matus Ujhelyi 已提交
63 64 65 66
MODULE_DESCRIPTION("Atheros 803x PHY driver");
MODULE_AUTHOR("Matus Ujhelyi");
MODULE_LICENSE("GPL");

67 68 69 70 71 72 73 74 75 76 77 78 79 80
struct at803x_priv {
	bool phy_reset:1;
	struct gpio_desc *gpiod_reset;
};

struct at803x_context {
	u16 bmcr;
	u16 advertise;
	u16 control1000;
	u16 int_enable;
	u16 smart_speed;
	u16 led_control;
};

81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
static int at803x_debug_reg_read(struct phy_device *phydev, u16 reg)
{
	int ret;

	ret = phy_write(phydev, AT803X_DEBUG_ADDR, reg);
	if (ret < 0)
		return ret;

	return phy_read(phydev, AT803X_DEBUG_DATA);
}

static int at803x_debug_reg_mask(struct phy_device *phydev, u16 reg,
				 u16 clear, u16 set)
{
	u16 val;
	int ret;

	ret = at803x_debug_reg_read(phydev, reg);
	if (ret < 0)
		return ret;

	val = ret & 0xffff;
	val &= ~clear;
	val |= set;

	return phy_write(phydev, AT803X_DEBUG_DATA, val);
}

static inline int at803x_enable_rx_delay(struct phy_device *phydev)
{
	return at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_0, 0,
					AT803X_DEBUG_RX_CLK_DLY_EN);
}

static inline int at803x_enable_tx_delay(struct phy_device *phydev)
{
	return at803x_debug_reg_mask(phydev, AT803X_DEBUG_REG_5, 0,
					AT803X_DEBUG_TX_CLK_DLY_EN);
}

121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144
/* save relevant PHY registers to private copy */
static void at803x_context_save(struct phy_device *phydev,
				struct at803x_context *context)
{
	context->bmcr = phy_read(phydev, MII_BMCR);
	context->advertise = phy_read(phydev, MII_ADVERTISE);
	context->control1000 = phy_read(phydev, MII_CTRL1000);
	context->int_enable = phy_read(phydev, AT803X_INTR_ENABLE);
	context->smart_speed = phy_read(phydev, AT803X_SMART_SPEED);
	context->led_control = phy_read(phydev, AT803X_LED_CONTROL);
}

/* restore relevant PHY registers from private copy */
static void at803x_context_restore(struct phy_device *phydev,
				   const struct at803x_context *context)
{
	phy_write(phydev, MII_BMCR, context->bmcr);
	phy_write(phydev, MII_ADVERTISE, context->advertise);
	phy_write(phydev, MII_CTRL1000, context->control1000);
	phy_write(phydev, AT803X_INTR_ENABLE, context->int_enable);
	phy_write(phydev, AT803X_SMART_SPEED, context->smart_speed);
	phy_write(phydev, AT803X_LED_CONTROL, context->led_control);
}

145 146
static int at803x_set_wol(struct phy_device *phydev,
			  struct ethtool_wolinfo *wol)
M
Matus Ujhelyi 已提交
147 148 149
{
	struct net_device *ndev = phydev->attached_dev;
	const u8 *mac;
150 151
	int ret;
	u32 value;
M
Matus Ujhelyi 已提交
152 153 154 155 156 157 158
	unsigned int i, offsets[] = {
		AT803X_LOC_MAC_ADDR_32_47_OFFSET,
		AT803X_LOC_MAC_ADDR_16_31_OFFSET,
		AT803X_LOC_MAC_ADDR_0_15_OFFSET,
	};

	if (!ndev)
159
		return -ENODEV;
M
Matus Ujhelyi 已提交
160

161 162
	if (wol->wolopts & WAKE_MAGIC) {
		mac = (const u8 *) ndev->dev_addr;
M
Matus Ujhelyi 已提交
163

164 165
		if (!is_valid_ether_addr(mac))
			return -EFAULT;
M
Matus Ujhelyi 已提交
166

167 168
		for (i = 0; i < 3; i++) {
			phy_write(phydev, AT803X_MMD_ACCESS_CONTROL,
M
Matus Ujhelyi 已提交
169
				  AT803X_DEVICE_ADDR);
170
			phy_write(phydev, AT803X_MMD_ACCESS_CONTROL_DATA,
M
Matus Ujhelyi 已提交
171
				  offsets[i]);
172
			phy_write(phydev, AT803X_MMD_ACCESS_CONTROL,
M
Matus Ujhelyi 已提交
173
				  AT803X_FUNC_DATA);
174
			phy_write(phydev, AT803X_MMD_ACCESS_CONTROL_DATA,
M
Matus Ujhelyi 已提交
175
				  mac[(i * 2) + 1] | (mac[(i * 2)] << 8));
176 177 178
		}

		value = phy_read(phydev, AT803X_INTR_ENABLE);
179
		value |= AT803X_INTR_ENABLE_WOL;
180 181 182 183 184 185
		ret = phy_write(phydev, AT803X_INTR_ENABLE, value);
		if (ret)
			return ret;
		value = phy_read(phydev, AT803X_INTR_STATUS);
	} else {
		value = phy_read(phydev, AT803X_INTR_ENABLE);
186
		value &= (~AT803X_INTR_ENABLE_WOL);
187 188 189 190
		ret = phy_write(phydev, AT803X_INTR_ENABLE, value);
		if (ret)
			return ret;
		value = phy_read(phydev, AT803X_INTR_STATUS);
M
Matus Ujhelyi 已提交
191
	}
192 193 194 195 196 197 198 199 200 201 202 203 204

	return ret;
}

static void at803x_get_wol(struct phy_device *phydev,
			   struct ethtool_wolinfo *wol)
{
	u32 value;

	wol->supported = WAKE_MAGIC;
	wol->wolopts = 0;

	value = phy_read(phydev, AT803X_INTR_ENABLE);
205
	if (value & AT803X_INTR_ENABLE_WOL)
206
		wol->wolopts |= WAKE_MAGIC;
M
Matus Ujhelyi 已提交
207 208
}

209 210 211 212
static int at803x_suspend(struct phy_device *phydev)
{
	int value;
	int wol_enabled;
213
	int ccr;
214 215 216 217

	mutex_lock(&phydev->lock);

	value = phy_read(phydev, AT803X_INTR_ENABLE);
218
	wol_enabled = value & AT803X_INTR_ENABLE_WOL;
219 220 221 222 223 224 225 226 227 228

	value = phy_read(phydev, MII_BMCR);

	if (wol_enabled)
		value |= BMCR_ISOLATE;
	else
		value |= BMCR_PDOWN;

	phy_write(phydev, MII_BMCR, value);

229 230 231 232 233 234 235 236 237 238
	if (phydev->interface != PHY_INTERFACE_MODE_SGMII)
		goto done;

	/* also power-down SGMII interface */
	ccr = phy_read(phydev, AT803X_REG_CHIP_CONFIG);
	phy_write(phydev, AT803X_REG_CHIP_CONFIG, ccr & ~AT803X_BT_BX_REG_SEL);
	phy_write(phydev, MII_BMCR, phy_read(phydev, MII_BMCR) | BMCR_PDOWN);
	phy_write(phydev, AT803X_REG_CHIP_CONFIG, ccr | AT803X_BT_BX_REG_SEL);

done:
239 240 241 242 243 244 245 246
	mutex_unlock(&phydev->lock);

	return 0;
}

static int at803x_resume(struct phy_device *phydev)
{
	int value;
247
	int ccr;
248 249 250 251 252 253 254

	mutex_lock(&phydev->lock);

	value = phy_read(phydev, MII_BMCR);
	value &= ~(BMCR_PDOWN | BMCR_ISOLATE);
	phy_write(phydev, MII_BMCR, value);

255 256 257 258 259 260 261 262 263 264 265
	if (phydev->interface != PHY_INTERFACE_MODE_SGMII)
		goto done;

	/* also power-up SGMII interface */
	ccr = phy_read(phydev, AT803X_REG_CHIP_CONFIG);
	phy_write(phydev, AT803X_REG_CHIP_CONFIG, ccr & ~AT803X_BT_BX_REG_SEL);
	value = phy_read(phydev, MII_BMCR) & ~(BMCR_PDOWN | BMCR_ISOLATE);
	phy_write(phydev, MII_BMCR, value);
	phy_write(phydev, AT803X_REG_CHIP_CONFIG, ccr | AT803X_BT_BX_REG_SEL);

done:
266 267 268 269 270
	mutex_unlock(&phydev->lock);

	return 0;
}

271 272
static int at803x_probe(struct phy_device *phydev)
{
A
Andrew Lunn 已提交
273
	struct device *dev = &phydev->mdio.dev;
274
	struct at803x_priv *priv;
275
	struct gpio_desc *gpiod_reset;
276

277
	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
278 279 280
	if (!priv)
		return -ENOMEM;

281 282 283
	if (phydev->drv->phy_id != ATH8030_PHY_ID)
		goto does_not_require_reset_workaround;

S
Sergei Shtylyov 已提交
284
	gpiod_reset = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_LOW);
285 286 287 288
	if (IS_ERR(gpiod_reset))
		return PTR_ERR(gpiod_reset);

	priv->gpiod_reset = gpiod_reset;
289

290
does_not_require_reset_workaround:
291 292 293 294 295
	phydev->priv = priv;

	return 0;
}

M
Matus Ujhelyi 已提交
296 297
static int at803x_config_init(struct phy_device *phydev)
{
298
	int ret;
M
Matus Ujhelyi 已提交
299

300 301 302
	ret = genphy_config_init(phydev);
	if (ret < 0)
		return ret;
M
Matus Ujhelyi 已提交
303

304 305 306 307
	if (phydev->interface == PHY_INTERFACE_MODE_RGMII_RXID ||
			phydev->interface == PHY_INTERFACE_MODE_RGMII_ID) {
		ret = at803x_enable_rx_delay(phydev);
		if (ret < 0)
308
			return ret;
309 310 311 312 313 314
	}

	if (phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID ||
			phydev->interface == PHY_INTERFACE_MODE_RGMII_ID) {
		ret = at803x_enable_tx_delay(phydev);
		if (ret < 0)
315 316 317
			return ret;
	}

M
Matus Ujhelyi 已提交
318 319 320
	return 0;
}

321 322 323 324
static int at803x_ack_interrupt(struct phy_device *phydev)
{
	int err;

325
	err = phy_read(phydev, AT803X_INTR_STATUS);
326 327 328 329 330 331 332 333 334

	return (err < 0) ? err : 0;
}

static int at803x_config_intr(struct phy_device *phydev)
{
	int err;
	int value;

335
	value = phy_read(phydev, AT803X_INTR_ENABLE);
336

337 338 339 340 341 342 343 344 345
	if (phydev->interrupts == PHY_INTERRUPT_ENABLED) {
		value |= AT803X_INTR_ENABLE_AUTONEG_ERR;
		value |= AT803X_INTR_ENABLE_SPEED_CHANGED;
		value |= AT803X_INTR_ENABLE_DUPLEX_CHANGED;
		value |= AT803X_INTR_ENABLE_LINK_FAIL;
		value |= AT803X_INTR_ENABLE_LINK_SUCCESS;

		err = phy_write(phydev, AT803X_INTR_ENABLE, value);
	}
346
	else
347
		err = phy_write(phydev, AT803X_INTR_ENABLE, 0);
348 349 350 351

	return err;
}

352 353 354 355 356 357 358 359 360 361 362
static void at803x_link_change_notify(struct phy_device *phydev)
{
	struct at803x_priv *priv = phydev->priv;

	/*
	 * Conduct a hardware reset for AT8030 every time a link loss is
	 * signalled. This is necessary to circumvent a hardware bug that
	 * occurs when the cable is unplugged while TX packets are pending
	 * in the FIFO. In such cases, the FIFO enters an error mode it
	 * cannot recover from by software.
	 */
363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378
	if (phydev->state == PHY_NOLINK) {
		if (priv->gpiod_reset && !priv->phy_reset) {
			struct at803x_context context;

			at803x_context_save(phydev, &context);

			gpiod_set_value(priv->gpiod_reset, 1);
			msleep(1);
			gpiod_set_value(priv->gpiod_reset, 0);
			msleep(1);

			at803x_context_restore(phydev, &context);

			phydev_dbg(phydev, "%s(): phy was reset\n",
				   __func__);
			priv->phy_reset = true;
379
		}
380 381
	} else {
		priv->phy_reset = false;
382 383 384
	}
}

385 386 387
static struct phy_driver at803x_driver[] = {
{
	/* ATHEROS 8035 */
388 389
	.phy_id			= ATH8035_PHY_ID,
	.name			= "Atheros 8035 ethernet",
390
	.phy_id_mask		= AT803X_PHY_ID_MASK,
391 392 393 394 395 396 397 398 399 400
	.probe			= at803x_probe,
	.config_init		= at803x_config_init,
	.set_wol		= at803x_set_wol,
	.get_wol		= at803x_get_wol,
	.suspend		= at803x_suspend,
	.resume			= at803x_resume,
	.features		= PHY_GBIT_FEATURES,
	.flags			= PHY_HAS_INTERRUPT,
	.config_aneg		= genphy_config_aneg,
	.read_status		= genphy_read_status,
401 402
	.ack_interrupt		= at803x_ack_interrupt,
	.config_intr		= at803x_config_intr,
403 404
}, {
	/* ATHEROS 8030 */
405 406
	.phy_id			= ATH8030_PHY_ID,
	.name			= "Atheros 8030 ethernet",
407
	.phy_id_mask		= AT803X_PHY_ID_MASK,
408 409 410 411 412 413 414
	.probe			= at803x_probe,
	.config_init		= at803x_config_init,
	.link_change_notify	= at803x_link_change_notify,
	.set_wol		= at803x_set_wol,
	.get_wol		= at803x_get_wol,
	.suspend		= at803x_suspend,
	.resume			= at803x_resume,
415
	.features		= PHY_BASIC_FEATURES,
416 417 418
	.flags			= PHY_HAS_INTERRUPT,
	.config_aneg		= genphy_config_aneg,
	.read_status		= genphy_read_status,
419 420
	.ack_interrupt		= at803x_ack_interrupt,
	.config_intr		= at803x_config_intr,
421 422
}, {
	/* ATHEROS 8031 */
423 424
	.phy_id			= ATH8031_PHY_ID,
	.name			= "Atheros 8031 ethernet",
425
	.phy_id_mask		= AT803X_PHY_ID_MASK,
426 427 428 429 430 431 432 433 434 435 436 437
	.probe			= at803x_probe,
	.config_init		= at803x_config_init,
	.set_wol		= at803x_set_wol,
	.get_wol		= at803x_get_wol,
	.suspend		= at803x_suspend,
	.resume			= at803x_resume,
	.features		= PHY_GBIT_FEATURES,
	.flags			= PHY_HAS_INTERRUPT,
	.config_aneg		= genphy_config_aneg,
	.read_status		= genphy_read_status,
	.ack_interrupt		= &at803x_ack_interrupt,
	.config_intr		= &at803x_config_intr,
438
} };
M
Matus Ujhelyi 已提交
439

440
module_phy_driver(at803x_driver);
M
Matus Ujhelyi 已提交
441 442

static struct mdio_device_id __maybe_unused atheros_tbl[] = {
443 444 445
	{ ATH8030_PHY_ID, AT803X_PHY_ID_MASK },
	{ ATH8031_PHY_ID, AT803X_PHY_ID_MASK },
	{ ATH8035_PHY_ID, AT803X_PHY_ID_MASK },
M
Matus Ujhelyi 已提交
446 447 448 449
	{ }
};

MODULE_DEVICE_TABLE(mdio, atheros_tbl);