phy-ti-pipe3.c 15.1 KB
Newer Older
1
/*
2
 * phy-ti-pipe3 - PIPE3 PHY driver.
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 *
 * Copyright (C) 2013 Texas Instruments Incorporated - http://www.ti.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.
 *
 * Author: Kishon Vijay Abraham I <kishon@ti.com>
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 */

#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/slab.h>
22
#include <linux/phy/phy.h>
23 24 25
#include <linux/of.h>
#include <linux/clk.h>
#include <linux/err.h>
26
#include <linux/io.h>
27 28
#include <linux/pm_runtime.h>
#include <linux/delay.h>
29
#include <linux/phy/omap_control_phy.h>
30
#include <linux/of_platform.h>
31 32
#include <linux/mfd/syscon.h>
#include <linux/regmap.h>
33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49

#define	PLL_STATUS		0x00000004
#define	PLL_GO			0x00000008
#define	PLL_CONFIGURATION1	0x0000000C
#define	PLL_CONFIGURATION2	0x00000010
#define	PLL_CONFIGURATION3	0x00000014
#define	PLL_CONFIGURATION4	0x00000020

#define	PLL_REGM_MASK		0x001FFE00
#define	PLL_REGM_SHIFT		0x9
#define	PLL_REGM_F_MASK		0x0003FFFF
#define	PLL_REGM_F_SHIFT	0x0
#define	PLL_REGN_MASK		0x000001FE
#define	PLL_REGN_SHIFT		0x1
#define	PLL_SELFREQDCO_MASK	0x0000000E
#define	PLL_SELFREQDCO_SHIFT	0x1
#define	PLL_SD_MASK		0x0003FC00
50
#define	PLL_SD_SHIFT		10
51
#define	SET_PLL_GO		0x1
52 53
#define PLL_LDOPWDN		BIT(15)
#define PLL_TICOPWDN		BIT(16)
54 55 56
#define	PLL_LOCK		0x2
#define	PLL_IDLE		0x1

57 58
#define SATA_PLL_SOFT_RESET	BIT(18)

59 60
/*
 * This is an Empirical value that works, need to confirm the actual
61 62
 * value required for the PIPE3PHY_PLL_CONFIGURATION2.PLL_IDLE status
 * to be correctly reflected in the PIPE3PHY_PLL_STATUS register.
63
 */
64 65
#define PLL_IDLE_TIME	100	/* in milliseconds */
#define PLL_LOCK_TIME	100	/* in milliseconds */
66

67 68 69 70 71 72 73 74
struct pipe3_dpll_params {
	u16	m;
	u8	n;
	u8	freq:3;
	u8	sd;
	u32	mf;
};

75 76 77 78 79
struct pipe3_dpll_map {
	unsigned long rate;
	struct pipe3_dpll_params params;
};

80 81 82 83 84 85
struct ti_pipe3 {
	void __iomem		*pll_ctrl_base;
	struct device		*dev;
	struct device		*control_dev;
	struct clk		*wkupclk;
	struct clk		*sys_clk;
86
	struct clk		*refclk;
87
	struct clk		*div_clk;
88
	struct pipe3_dpll_map	*dpll_map;
89 90 91
	struct regmap		*dpll_reset_syscon; /* ctrl. reg. acces */
	unsigned int		dpll_reset_reg; /* reg. index within syscon */
	bool			sata_refclk_enabled;
92 93
};

94
static struct pipe3_dpll_map dpll_map_usb[] = {
95 96 97 98 99 100
	{12000000, {1250, 5, 4, 20, 0} },	/* 12 MHz */
	{16800000, {3125, 20, 4, 20, 0} },	/* 16.8 MHz */
	{19200000, {1172, 8, 4, 20, 65537} },	/* 19.2 MHz */
	{20000000, {1000, 7, 4, 10, 0} },	/* 20 MHz */
	{26000000, {1250, 12, 4, 20, 0} },	/* 26 MHz */
	{38400000, {3125, 47, 4, 20, 92843} },	/* 38.4 MHz */
101 102 103 104 105 106 107 108 109 110 111
	{ },					/* Terminator */
};

static struct pipe3_dpll_map dpll_map_sata[] = {
	{12000000, {1000, 7, 4, 6, 0} },	/* 12 MHz */
	{16800000, {714, 7, 4, 6, 0} },		/* 16.8 MHz */
	{19200000, {625, 7, 4, 6, 0} },		/* 19.2 MHz */
	{20000000, {600, 7, 4, 6, 0} },		/* 20 MHz */
	{26000000, {461, 7, 4, 6, 0} },		/* 26 MHz */
	{38400000, {312, 7, 4, 6, 0} },		/* 38.4 MHz */
	{ },					/* Terminator */
112 113
};

114 115 116 117 118 119 120 121 122 123 124
static inline u32 ti_pipe3_readl(void __iomem *addr, unsigned offset)
{
	return __raw_readl(addr + offset);
}

static inline void ti_pipe3_writel(void __iomem *addr, unsigned offset,
	u32 data)
{
	__raw_writel(data, addr + offset);
}

125
static struct pipe3_dpll_params *ti_pipe3_get_dpll_params(struct ti_pipe3 *phy)
126
{
127 128
	unsigned long rate;
	struct pipe3_dpll_map *dpll_map = phy->dpll_map;
129

130 131 132 133 134
	rate = clk_get_rate(phy->sys_clk);

	for (; dpll_map->rate; dpll_map++) {
		if (rate == dpll_map->rate)
			return &dpll_map->params;
135 136
	}

137 138
	dev_err(phy->dev, "No DPLL configuration for %lu Hz SYS CLK\n", rate);

139
	return NULL;
140 141
}

R
Roger Quadros 已提交
142 143 144
static int ti_pipe3_enable_clocks(struct ti_pipe3 *phy);
static void ti_pipe3_disable_clocks(struct ti_pipe3 *phy);

145 146 147 148
static int ti_pipe3_power_off(struct phy *x)
{
	struct ti_pipe3 *phy = phy_get_drvdata(x);

149
	omap_control_phy_power(phy->control_dev, 0);
150 151 152 153 154

	return 0;
}

static int ti_pipe3_power_on(struct phy *x)
155
{
156
	struct ti_pipe3 *phy = phy_get_drvdata(x);
157

158
	omap_control_phy_power(phy->control_dev, 1);
159 160 161 162

	return 0;
}

163
static int ti_pipe3_dpll_wait_lock(struct ti_pipe3 *phy)
164 165 166 167
{
	u32		val;
	unsigned long	timeout;

168
	timeout = jiffies + msecs_to_jiffies(PLL_LOCK_TIME);
169
	do {
170
		cpu_relax();
171
		val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_STATUS);
172
		if (val & PLL_LOCK)
173
			return 0;
174 175
	} while (!time_after(jiffies, timeout));

176 177
	dev_err(phy->dev, "DPLL failed to lock\n");
	return -EBUSY;
178 179
}

180
static int ti_pipe3_dpll_program(struct ti_pipe3 *phy)
181 182
{
	u32			val;
183
	struct pipe3_dpll_params *dpll_params;
184

185 186
	dpll_params = ti_pipe3_get_dpll_params(phy);
	if (!dpll_params)
187 188
		return -EINVAL;

189
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION1);
190
	val &= ~PLL_REGN_MASK;
191
	val |= dpll_params->n << PLL_REGN_SHIFT;
192
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION1, val);
193

194
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION2);
195
	val &= ~PLL_SELFREQDCO_MASK;
196
	val |= dpll_params->freq << PLL_SELFREQDCO_SHIFT;
197
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION2, val);
198

199
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION1);
200
	val &= ~PLL_REGM_MASK;
201
	val |= dpll_params->m << PLL_REGM_SHIFT;
202
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION1, val);
203

204
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION4);
205
	val &= ~PLL_REGM_F_MASK;
206
	val |= dpll_params->mf << PLL_REGM_F_SHIFT;
207
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION4, val);
208

209
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION3);
210
	val &= ~PLL_SD_MASK;
211
	val |= dpll_params->sd << PLL_SD_SHIFT;
212
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION3, val);
213

214
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_GO, SET_PLL_GO);
215

216
	return ti_pipe3_dpll_wait_lock(phy);
217 218
}

219
static int ti_pipe3_init(struct phy *x)
220
{
221
	struct ti_pipe3 *phy = phy_get_drvdata(x);
222 223
	u32 val;
	int ret = 0;
224

R
Roger Quadros 已提交
225
	ti_pipe3_enable_clocks(phy);
226 227 228 229 230
	/*
	 * Set pcie_pcs register to 0x96 for proper functioning of phy
	 * as recommended in AM572x TRM SPRUHZ6, section 18.5.2.2, table
	 * 18-1804.
	 */
231
	if (of_device_is_compatible(phy->dev->of_node, "ti,phy-pipe3-pcie")) {
232
		omap_control_pcie_pcs(phy->control_dev, 0x96);
233
		return 0;
234
	}
235

236 237 238 239 240 241 242
	/* Bring it out of IDLE if it is IDLE */
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION2);
	if (val & PLL_IDLE) {
		val &= ~PLL_IDLE;
		ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION2, val);
		ret = ti_pipe3_dpll_wait_lock(phy);
	}
243

244 245 246 247 248
	/* Program the DPLL only if not locked */
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_STATUS);
	if (!(val & PLL_LOCK))
		if (ti_pipe3_dpll_program(phy))
			return -EINVAL;
249

250
	return ret;
251 252
}

253 254 255 256 257 258
static int ti_pipe3_exit(struct phy *x)
{
	struct ti_pipe3 *phy = phy_get_drvdata(x);
	u32 val;
	unsigned long timeout;

259 260 261 262 263
	/* If dpll_reset_syscon is not present we wont power down SATA DPLL
	 * due to Errata i783
	 */
	if (of_device_is_compatible(phy->dev->of_node, "ti,phy-pipe3-sata") &&
	    !phy->dpll_reset_syscon)
264 265
		return 0;

R
Roger Quadros 已提交
266 267 268 269 270 271
	/* PCIe doesn't have internal DPLL */
	if (!of_device_is_compatible(phy->dev->of_node, "ti,phy-pipe3-pcie")) {
		/* Put DPLL in IDLE mode */
		val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION2);
		val |= PLL_IDLE;
		ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION2, val);
272

R
Roger Quadros 已提交
273 274 275 276 277 278 279 280 281 282 283 284 285 286
		/* wait for LDO and Oscillator to power down */
		timeout = jiffies + msecs_to_jiffies(PLL_IDLE_TIME);
		do {
			cpu_relax();
			val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_STATUS);
			if ((val & PLL_TICOPWDN) && (val & PLL_LDOPWDN))
				break;
		} while (!time_after(jiffies, timeout));

		if (!(val & PLL_TICOPWDN) || !(val & PLL_LDOPWDN)) {
			dev_err(phy->dev, "Failed to power down: PLL_STATUS 0x%x\n",
				val);
			return -EBUSY;
		}
287 288
	}

289 290 291 292 293 294 295 296
	/* i783: SATA needs control bit toggle after PLL unlock */
	if (of_device_is_compatible(phy->dev->of_node, "ti,phy-pipe3-sata")) {
		regmap_update_bits(phy->dpll_reset_syscon, phy->dpll_reset_reg,
				   SATA_PLL_SOFT_RESET, SATA_PLL_SOFT_RESET);
		regmap_update_bits(phy->dpll_reset_syscon, phy->dpll_reset_reg,
				   SATA_PLL_SOFT_RESET, 0);
	}

R
Roger Quadros 已提交
297 298
	ti_pipe3_disable_clocks(phy);

299 300
	return 0;
}
301
static const struct phy_ops ops = {
302
	.init		= ti_pipe3_init,
303
	.exit		= ti_pipe3_exit,
304 305 306 307 308
	.power_on	= ti_pipe3_power_on,
	.power_off	= ti_pipe3_power_off,
	.owner		= THIS_MODULE,
};

309 310
static const struct of_device_id ti_pipe3_id_table[];

311
static int ti_pipe3_get_clk(struct ti_pipe3 *phy)
312
{
313
	struct clk *clk;
314 315
	struct device *dev = phy->dev;
	struct device_node *node = dev->of_node;
316

317
	phy->refclk = devm_clk_get(dev, "refclk");
318
	if (IS_ERR(phy->refclk)) {
319
		dev_err(dev, "unable to get refclk\n");
320 321 322 323 324 325 326
		/* older DTBs have missing refclk in SATA PHY
		 * so don't bail out in case of SATA PHY.
		 */
		if (!of_device_is_compatible(node, "ti,phy-pipe3-sata"))
			return PTR_ERR(phy->refclk);
	}

327
	if (!of_device_is_compatible(node, "ti,phy-pipe3-sata")) {
328
		phy->wkupclk = devm_clk_get(dev, "wkupclk");
329
		if (IS_ERR(phy->wkupclk)) {
330
			dev_err(dev, "unable to get wkupclk\n");
331 332 333 334
			return PTR_ERR(phy->wkupclk);
		}
	} else {
		phy->wkupclk = ERR_PTR(-ENODEV);
335 336 337 338 339 340 341
	}

	if (!of_device_is_compatible(node, "ti,phy-pipe3-pcie")) {
		phy->sys_clk = devm_clk_get(dev, "sysclk");
		if (IS_ERR(phy->sys_clk)) {
			dev_err(dev, "unable to get sysclk\n");
			return -EINVAL;
342
		}
343 344
	}

345
	if (of_device_is_compatible(node, "ti,phy-pipe3-pcie")) {
346
		clk = devm_clk_get(dev, "dpll_ref");
347
		if (IS_ERR(clk)) {
348
			dev_err(dev, "unable to get dpll ref clk\n");
349 350 351 352
			return PTR_ERR(clk);
		}
		clk_set_rate(clk, 1500000000);

353
		clk = devm_clk_get(dev, "dpll_ref_m2");
354
		if (IS_ERR(clk)) {
355
			dev_err(dev, "unable to get dpll ref m2 clk\n");
356 357 358 359
			return PTR_ERR(clk);
		}
		clk_set_rate(clk, 100000000);

360
		clk = devm_clk_get(dev, "phy-div");
361
		if (IS_ERR(clk)) {
362
			dev_err(dev, "unable to get phy-div clk\n");
363 364 365 366
			return PTR_ERR(clk);
		}
		clk_set_rate(clk, 100000000);

367
		phy->div_clk = devm_clk_get(dev, "div-clk");
368
		if (IS_ERR(phy->div_clk)) {
369
			dev_err(dev, "unable to get div-clk\n");
370 371 372 373
			return PTR_ERR(phy->div_clk);
		}
	} else {
		phy->div_clk = ERR_PTR(-ENODEV);
374 375
	}

376 377 378
	return 0;
}

379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420
static int ti_pipe3_get_sysctrl(struct ti_pipe3 *phy)
{
	struct device *dev = phy->dev;
	struct device_node *node = dev->of_node;
	struct device_node *control_node;
	struct platform_device *control_pdev;

	control_node = of_parse_phandle(node, "ctrl-module", 0);
	if (!control_node) {
		dev_err(dev, "Failed to get control device phandle\n");
		return -EINVAL;
	}

	control_pdev = of_find_device_by_node(control_node);
	if (!control_pdev) {
		dev_err(dev, "Failed to get control device\n");
		return -EINVAL;
	}

	phy->control_dev = &control_pdev->dev;

	if (of_device_is_compatible(node, "ti,phy-pipe3-sata")) {
		phy->dpll_reset_syscon = syscon_regmap_lookup_by_phandle(node,
							"syscon-pllreset");
		if (IS_ERR(phy->dpll_reset_syscon)) {
			dev_info(dev,
				 "can't get syscon-pllreset, sata dpll won't idle\n");
			phy->dpll_reset_syscon = NULL;
		} else {
			if (of_property_read_u32_index(node,
						       "syscon-pllreset", 1,
						       &phy->dpll_reset_reg)) {
				dev_err(dev,
					"couldn't get pllreset reg. offset\n");
				return -EINVAL;
			}
		}
	}

	return 0;
}

421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450
static int ti_pipe3_get_pll_base(struct ti_pipe3 *phy)
{
	struct resource *res;
	const struct of_device_id *match;
	struct device *dev = phy->dev;
	struct device_node *node = dev->of_node;
	struct platform_device *pdev = to_platform_device(dev);

	if (of_device_is_compatible(node, "ti,phy-pipe3-pcie"))
		return 0;

	match = of_match_device(ti_pipe3_id_table, dev);
	if (!match)
		return -EINVAL;

	phy->dpll_map = (struct pipe3_dpll_map *)match->data;
	if (!phy->dpll_map) {
		dev_err(dev, "no DPLL data\n");
		return -EINVAL;
	}

	res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
					   "pll_ctrl");
	phy->pll_ctrl_base = devm_ioremap_resource(dev, res);
	if (IS_ERR(phy->pll_ctrl_base))
		return PTR_ERR(phy->pll_ctrl_base);

	return 0;
}

451 452 453 454 455 456 457 458 459 460 461 462 463 464 465
static int ti_pipe3_probe(struct platform_device *pdev)
{
	struct ti_pipe3 *phy;
	struct phy *generic_phy;
	struct phy_provider *phy_provider;
	struct device_node *node = pdev->dev.of_node;
	struct device *dev = &pdev->dev;
	int ret;

	phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL);
	if (!phy)
		return -ENOMEM;

	phy->dev		= dev;

466 467 468
	ret = ti_pipe3_get_pll_base(phy);
	if (ret)
		return ret;
469

470 471 472
	ret = ti_pipe3_get_sysctrl(phy);
	if (ret)
		return ret;
473 474 475 476 477

	ret = ti_pipe3_get_clk(phy);
	if (ret)
		return ret;

478
	omap_control_phy_power(phy->control_dev, 0);
479 480

	platform_set_drvdata(pdev, phy);
481
	pm_runtime_enable(dev);
482 483 484 485 486 487

	/*
	 * Prevent auto-disable of refclk for SATA PHY due to Errata i783
	 */
	if (of_device_is_compatible(node, "ti,phy-pipe3-sata")) {
		if (!IS_ERR(phy->refclk)) {
R
Roger Quadros 已提交
488
			clk_prepare_enable(phy->refclk);
489 490 491
			phy->sata_refclk_enabled = true;
		}
	}
492

493
	generic_phy = devm_phy_create(dev, NULL, &ops);
494 495 496 497
	if (IS_ERR(generic_phy))
		return PTR_ERR(generic_phy);

	phy_set_drvdata(generic_phy, phy);
498
	phy_provider = devm_of_phy_provider_register(dev, of_phy_simple_xlate);
499 500 501
	if (IS_ERR(phy_provider))
		return PTR_ERR(phy_provider);

502 503 504
	return 0;
}

505
static int ti_pipe3_remove(struct platform_device *pdev)
506 507 508 509 510 511
{
	pm_runtime_disable(&pdev->dev);

	return 0;
}

R
Roger Quadros 已提交
512
static int ti_pipe3_enable_clocks(struct ti_pipe3 *phy)
513
{
R
Roger Quadros 已提交
514
	int ret = 0;
515

R
Roger Quadros 已提交
516
	if (!IS_ERR(phy->refclk)) {
517 518 519 520 521 522 523
		ret = clk_prepare_enable(phy->refclk);
		if (ret) {
			dev_err(phy->dev, "Failed to enable refclk %d\n", ret);
			return ret;
		}
	}

524 525 526 527
	if (!IS_ERR(phy->wkupclk)) {
		ret = clk_prepare_enable(phy->wkupclk);
		if (ret) {
			dev_err(phy->dev, "Failed to enable wkupclk %d\n", ret);
R
Roger Quadros 已提交
528
			goto disable_refclk;
529
		}
530 531
	}

532 533 534 535
	if (!IS_ERR(phy->div_clk)) {
		ret = clk_prepare_enable(phy->div_clk);
		if (ret) {
			dev_err(phy->dev, "Failed to enable div_clk %d\n", ret);
R
Roger Quadros 已提交
536
			goto disable_wkupclk;
537 538
		}
	}
539

540 541
	return 0;

R
Roger Quadros 已提交
542
disable_wkupclk:
543 544 545
	if (!IS_ERR(phy->wkupclk))
		clk_disable_unprepare(phy->wkupclk);

R
Roger Quadros 已提交
546
disable_refclk:
547 548
	if (!IS_ERR(phy->refclk))
		clk_disable_unprepare(phy->refclk);
549

550 551 552 553 554 555 556
	return ret;
}

static void ti_pipe3_disable_clocks(struct ti_pipe3 *phy)
{
	if (!IS_ERR(phy->wkupclk))
		clk_disable_unprepare(phy->wkupclk);
557
	if (!IS_ERR(phy->refclk)) {
R
Roger Quadros 已提交
558
		clk_disable_unprepare(phy->refclk);
559 560 561 562 563 564 565 566 567 568
		/*
		 * SATA refclk needs an additional disable as we left it
		 * on in probe to avoid Errata i783
		 */
		if (phy->sata_refclk_enabled) {
			clk_disable_unprepare(phy->refclk);
			phy->sata_refclk_enabled = false;
		}
	}

569 570 571 572
	if (!IS_ERR(phy->div_clk))
		clk_disable_unprepare(phy->div_clk);
}

573
static const struct of_device_id ti_pipe3_id_table[] = {
574 575 576 577 578 579 580 581 582 583 584 585
	{
		.compatible = "ti,phy-usb3",
		.data = dpll_map_usb,
	},
	{
		.compatible = "ti,omap-usb3",
		.data = dpll_map_usb,
	},
	{
		.compatible = "ti,phy-pipe3-sata",
		.data = dpll_map_sata,
	},
586 587 588
	{
		.compatible = "ti,phy-pipe3-pcie",
	},
589 590
	{}
};
591
MODULE_DEVICE_TABLE(of, ti_pipe3_id_table);
592

593 594 595
static struct platform_driver ti_pipe3_driver = {
	.probe		= ti_pipe3_probe,
	.remove		= ti_pipe3_remove,
596
	.driver		= {
597
		.name	= "ti-pipe3",
598
		.of_match_table = ti_pipe3_id_table,
599 600 601
	},
};

602
module_platform_driver(ti_pipe3_driver);
603

A
Axel Lin 已提交
604
MODULE_ALIAS("platform:ti_pipe3");
605
MODULE_AUTHOR("Texas Instruments Inc.");
606
MODULE_DESCRIPTION("TI PIPE3 phy driver");
607
MODULE_LICENSE("GPL v2");