phy-ti-pipe3.c 13.0 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 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47

#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
48
#define	PLL_SD_SHIFT		10
49
#define	SET_PLL_GO		0x1
50 51
#define PLL_LDOPWDN		BIT(15)
#define PLL_TICOPWDN		BIT(16)
52 53 54 55 56
#define	PLL_LOCK		0x2
#define	PLL_IDLE		0x1

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

63 64 65 66 67 68 69 70
struct pipe3_dpll_params {
	u16	m;
	u8	n;
	u8	freq:3;
	u8	sd;
	u32	mf;
};

71 72 73 74 75
struct pipe3_dpll_map {
	unsigned long rate;
	struct pipe3_dpll_params params;
};

76 77 78 79 80 81
struct ti_pipe3 {
	void __iomem		*pll_ctrl_base;
	struct device		*dev;
	struct device		*control_dev;
	struct clk		*wkupclk;
	struct clk		*sys_clk;
82
	struct clk		*refclk;
83
	struct clk		*div_clk;
84
	struct pipe3_dpll_map	*dpll_map;
85 86
};

87
static struct pipe3_dpll_map dpll_map_usb[] = {
88 89 90 91 92 93
	{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 */
94 95 96 97 98 99 100 101 102 103 104
	{ },					/* 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 */
105 106
};

107 108 109 110 111 112 113 114 115 116 117
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);
}

118
static struct pipe3_dpll_params *ti_pipe3_get_dpll_params(struct ti_pipe3 *phy)
119
{
120 121
	unsigned long rate;
	struct pipe3_dpll_map *dpll_map = phy->dpll_map;
122

123 124 125 126 127
	rate = clk_get_rate(phy->sys_clk);

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

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

132
	return NULL;
133 134
}

R
Roger Quadros 已提交
135 136 137
static int ti_pipe3_enable_clocks(struct ti_pipe3 *phy);
static void ti_pipe3_disable_clocks(struct ti_pipe3 *phy);

138 139 140 141
static int ti_pipe3_power_off(struct phy *x)
{
	struct ti_pipe3 *phy = phy_get_drvdata(x);

142
	omap_control_phy_power(phy->control_dev, 0);
143 144 145 146 147

	return 0;
}

static int ti_pipe3_power_on(struct phy *x)
148
{
149
	struct ti_pipe3 *phy = phy_get_drvdata(x);
150

151
	omap_control_phy_power(phy->control_dev, 1);
152 153 154 155

	return 0;
}

156
static int ti_pipe3_dpll_wait_lock(struct ti_pipe3 *phy)
157 158 159 160
{
	u32		val;
	unsigned long	timeout;

161
	timeout = jiffies + msecs_to_jiffies(PLL_LOCK_TIME);
162
	do {
163
		cpu_relax();
164
		val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_STATUS);
165
		if (val & PLL_LOCK)
166
			return 0;
167 168
	} while (!time_after(jiffies, timeout));

169 170
	dev_err(phy->dev, "DPLL failed to lock\n");
	return -EBUSY;
171 172
}

173
static int ti_pipe3_dpll_program(struct ti_pipe3 *phy)
174 175
{
	u32			val;
176
	struct pipe3_dpll_params *dpll_params;
177

178 179
	dpll_params = ti_pipe3_get_dpll_params(phy);
	if (!dpll_params)
180 181
		return -EINVAL;

182
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION1);
183
	val &= ~PLL_REGN_MASK;
184
	val |= dpll_params->n << PLL_REGN_SHIFT;
185
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION1, val);
186

187
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION2);
188
	val &= ~PLL_SELFREQDCO_MASK;
189
	val |= dpll_params->freq << PLL_SELFREQDCO_SHIFT;
190
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION2, val);
191

192
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION1);
193
	val &= ~PLL_REGM_MASK;
194
	val |= dpll_params->m << PLL_REGM_SHIFT;
195
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION1, val);
196

197
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION4);
198
	val &= ~PLL_REGM_F_MASK;
199
	val |= dpll_params->mf << PLL_REGM_F_SHIFT;
200
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION4, val);
201

202
	val = ti_pipe3_readl(phy->pll_ctrl_base, PLL_CONFIGURATION3);
203
	val &= ~PLL_SD_MASK;
204
	val |= dpll_params->sd << PLL_SD_SHIFT;
205
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_CONFIGURATION3, val);
206

207
	ti_pipe3_writel(phy->pll_ctrl_base, PLL_GO, SET_PLL_GO);
208

209
	return ti_pipe3_dpll_wait_lock(phy);
210 211
}

212
static int ti_pipe3_init(struct phy *x)
213
{
214
	struct ti_pipe3 *phy = phy_get_drvdata(x);
215 216
	u32 val;
	int ret = 0;
217

R
Roger Quadros 已提交
218
	ti_pipe3_enable_clocks(phy);
219 220 221 222 223
	/*
	 * 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.
	 */
224
	if (of_device_is_compatible(phy->dev->of_node, "ti,phy-pipe3-pcie")) {
225
		omap_control_pcie_pcs(phy->control_dev, 0x96);
226
		return 0;
227
	}
228

229 230 231 232 233 234 235
	/* 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);
	}
236

237 238 239 240 241
	/* 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;
242

243
	return ret;
244 245
}

246 247 248 249 250 251
static int ti_pipe3_exit(struct phy *x)
{
	struct ti_pipe3 *phy = phy_get_drvdata(x);
	u32 val;
	unsigned long timeout;

R
Roger Quadros 已提交
252 253
	/* SATA DPLL can't be powered down due to Errata i783 */
	if (of_device_is_compatible(phy->dev->of_node, "ti,phy-pipe3-sata"))
254 255
		return 0;

R
Roger Quadros 已提交
256 257 258 259 260 261
	/* 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);
262

R
Roger Quadros 已提交
263 264 265 266 267 268 269 270 271 272 273 274 275 276
		/* 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;
		}
277 278
	}

R
Roger Quadros 已提交
279 280
	ti_pipe3_disable_clocks(phy);

281 282
	return 0;
}
283 284
static struct phy_ops ops = {
	.init		= ti_pipe3_init,
285
	.exit		= ti_pipe3_exit,
286 287 288 289 290
	.power_on	= ti_pipe3_power_on,
	.power_off	= ti_pipe3_power_off,
	.owner		= THIS_MODULE,
};

291 292
static const struct of_device_id ti_pipe3_id_table[];

293
static int ti_pipe3_probe(struct platform_device *pdev)
294
{
295 296 297
	struct ti_pipe3 *phy;
	struct phy *generic_phy;
	struct phy_provider *phy_provider;
298 299 300 301
	struct resource *res;
	struct device_node *node = pdev->dev.of_node;
	struct device_node *control_node;
	struct platform_device *control_pdev;
302
	const struct of_device_id *match;
303
	struct clk *clk;
304 305

	phy = devm_kzalloc(&pdev->dev, sizeof(*phy), GFP_KERNEL);
306
	if (!phy)
307
		return -ENOMEM;
308

309
	phy->dev		= &pdev->dev;
310

311
	if (!of_device_is_compatible(node, "ti,phy-pipe3-pcie")) {
312
		match = of_match_device(ti_pipe3_id_table, &pdev->dev);
313 314
		if (!match)
			return -EINVAL;
315

316 317 318 319 320
		phy->dpll_map = (struct pipe3_dpll_map *)match->data;
		if (!phy->dpll_map) {
			dev_err(&pdev->dev, "no DPLL data\n");
			return -EINVAL;
		}
321

322 323 324 325 326
		res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
						   "pll_ctrl");
		phy->pll_ctrl_base = devm_ioremap_resource(&pdev->dev, res);
		if (IS_ERR(phy->pll_ctrl_base))
			return PTR_ERR(phy->pll_ctrl_base);
327

328 329 330 331 332 333
		phy->sys_clk = devm_clk_get(phy->dev, "sysclk");
		if (IS_ERR(phy->sys_clk)) {
			dev_err(&pdev->dev, "unable to get sysclk\n");
			return -EINVAL;
		}
	}
334

335 336 337 338 339 340 341 342 343 344
	phy->refclk = devm_clk_get(phy->dev, "refclk");
	if (IS_ERR(phy->refclk)) {
		dev_err(&pdev->dev, "unable to get refclk\n");
		/* 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);
	}

345
	if (!of_device_is_compatible(node, "ti,phy-pipe3-sata")) {
346 347 348 349 350 351 352
		phy->wkupclk = devm_clk_get(phy->dev, "wkupclk");
		if (IS_ERR(phy->wkupclk)) {
			dev_err(&pdev->dev, "unable to get wkupclk\n");
			return PTR_ERR(phy->wkupclk);
		}
	} else {
		phy->wkupclk = ERR_PTR(-ENODEV);
353 354
	}

355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384
	if (of_device_is_compatible(node, "ti,phy-pipe3-pcie")) {

		clk = devm_clk_get(phy->dev, "dpll_ref");
		if (IS_ERR(clk)) {
			dev_err(&pdev->dev, "unable to get dpll ref clk\n");
			return PTR_ERR(clk);
		}
		clk_set_rate(clk, 1500000000);

		clk = devm_clk_get(phy->dev, "dpll_ref_m2");
		if (IS_ERR(clk)) {
			dev_err(&pdev->dev, "unable to get dpll ref m2 clk\n");
			return PTR_ERR(clk);
		}
		clk_set_rate(clk, 100000000);

		clk = devm_clk_get(phy->dev, "phy-div");
		if (IS_ERR(clk)) {
			dev_err(&pdev->dev, "unable to get phy-div clk\n");
			return PTR_ERR(clk);
		}
		clk_set_rate(clk, 100000000);

		phy->div_clk = devm_clk_get(phy->dev, "div-clk");
		if (IS_ERR(phy->div_clk)) {
			dev_err(&pdev->dev, "unable to get div-clk\n");
			return PTR_ERR(phy->div_clk);
		}
	} else {
		phy->div_clk = ERR_PTR(-ENODEV);
385 386
	}

387 388 389 390
	control_node = of_parse_phandle(node, "ctrl-module", 0);
	if (!control_node) {
		dev_err(&pdev->dev, "Failed to get control device phandle\n");
		return -EINVAL;
391
	}
392

393 394 395 396 397 398 399
	control_pdev = of_find_device_by_node(control_node);
	if (!control_pdev) {
		dev_err(&pdev->dev, "Failed to get control device\n");
		return -EINVAL;
	}

	phy->control_dev = &control_pdev->dev;
400

401
	omap_control_phy_power(phy->control_dev, 0);
402 403 404

	platform_set_drvdata(pdev, phy);
	pm_runtime_enable(phy->dev);
R
Roger Quadros 已提交
405 406 407 408
	/* 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))
			clk_prepare_enable(phy->refclk);
409

410
	generic_phy = devm_phy_create(phy->dev, NULL, &ops);
411 412 413 414 415 416 417 418 419
	if (IS_ERR(generic_phy))
		return PTR_ERR(generic_phy);

	phy_set_drvdata(generic_phy, phy);
	phy_provider = devm_of_phy_provider_register(phy->dev,
			of_phy_simple_xlate);
	if (IS_ERR(phy_provider))
		return PTR_ERR(phy_provider);

420 421 422
	return 0;
}

423
static int ti_pipe3_remove(struct platform_device *pdev)
424 425 426 427 428 429
{
	pm_runtime_disable(&pdev->dev);

	return 0;
}

R
Roger Quadros 已提交
430
static int ti_pipe3_enable_clocks(struct ti_pipe3 *phy)
431
{
R
Roger Quadros 已提交
432
	int ret = 0;
433

R
Roger Quadros 已提交
434
	if (!IS_ERR(phy->refclk)) {
435 436 437 438 439 440 441
		ret = clk_prepare_enable(phy->refclk);
		if (ret) {
			dev_err(phy->dev, "Failed to enable refclk %d\n", ret);
			return ret;
		}
	}

442 443 444 445
	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 已提交
446
			goto disable_refclk;
447
		}
448 449
	}

450 451 452 453
	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 已提交
454
			goto disable_wkupclk;
455 456
		}
	}
457

458 459
	return 0;

R
Roger Quadros 已提交
460
disable_wkupclk:
461 462 463
	if (!IS_ERR(phy->wkupclk))
		clk_disable_unprepare(phy->wkupclk);

R
Roger Quadros 已提交
464
disable_refclk:
465 466
	if (!IS_ERR(phy->refclk))
		clk_disable_unprepare(phy->refclk);
467

468 469 470 471 472 473 474
	return ret;
}

static void ti_pipe3_disable_clocks(struct ti_pipe3 *phy)
{
	if (!IS_ERR(phy->wkupclk))
		clk_disable_unprepare(phy->wkupclk);
R
Roger Quadros 已提交
475 476
	if (!IS_ERR(phy->refclk))
		clk_disable_unprepare(phy->refclk);
477 478 479 480
	if (!IS_ERR(phy->div_clk))
		clk_disable_unprepare(phy->div_clk);
}

481
static const struct of_device_id ti_pipe3_id_table[] = {
482 483 484 485 486 487 488 489 490 491 492 493
	{
		.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,
	},
494 495 496
	{
		.compatible = "ti,phy-pipe3-pcie",
	},
497 498
	{}
};
499
MODULE_DEVICE_TABLE(of, ti_pipe3_id_table);
500

501 502 503
static struct platform_driver ti_pipe3_driver = {
	.probe		= ti_pipe3_probe,
	.remove		= ti_pipe3_remove,
504
	.driver		= {
505
		.name	= "ti-pipe3",
506
		.of_match_table = ti_pipe3_id_table,
507 508 509
	},
};

510
module_platform_driver(ti_pipe3_driver);
511

A
Axel Lin 已提交
512
MODULE_ALIAS("platform:ti_pipe3");
513
MODULE_AUTHOR("Texas Instruments Inc.");
514
MODULE_DESCRIPTION("TI PIPE3 phy driver");
515
MODULE_LICENSE("GPL v2");