dss.c 22.3 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
/*
 * linux/drivers/video/omap2/dss/dss.c
 *
 * Copyright (C) 2009 Nokia Corporation
 * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com>
 *
 * Some code and ideas taken from drivers/video/omap/ driver
 * by Imre Deak.
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 as published by
 * the Free Software Foundation.
 *
 * 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.
 *
 * You should have received a copy of the GNU General Public License along with
 * this program.  If not, see <http://www.gnu.org/licenses/>.
 */

#define DSS_SUBSYS_NAME "DSS"

#include <linux/kernel.h>
T
Tomi Valkeinen 已提交
26
#include <linux/module.h>
27
#include <linux/io.h>
28
#include <linux/export.h>
29 30 31 32
#include <linux/err.h>
#include <linux/delay.h>
#include <linux/seq_file.h>
#include <linux/clk.h>
33
#include <linux/platform_device.h>
34
#include <linux/pm_runtime.h>
35
#include <linux/gfp.h>
36
#include <linux/sizes.h>
T
Tomi Valkeinen 已提交
37
#include <linux/of.h>
38

39
#include <video/omapdss.h>
40

41
#include "dss.h"
42
#include "dss_features.h"
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65

#define DSS_SZ_REGS			SZ_512

struct dss_reg {
	u16 idx;
};

#define DSS_REG(idx)			((const struct dss_reg) { idx })

#define DSS_REVISION			DSS_REG(0x0000)
#define DSS_SYSCONFIG			DSS_REG(0x0010)
#define DSS_SYSSTATUS			DSS_REG(0x0014)
#define DSS_CONTROL			DSS_REG(0x0040)
#define DSS_SDI_CONTROL			DSS_REG(0x0044)
#define DSS_PLL_CONTROL			DSS_REG(0x0048)
#define DSS_SDI_STATUS			DSS_REG(0x005C)

#define REG_GET(idx, start, end) \
	FLD_GET(dss_read_reg(idx), start, end)

#define REG_FLD_MOD(idx, val, start, end) \
	dss_write_reg(idx, FLD_MOD(dss_read_reg(idx), val, start, end))

66 67 68
static int dss_runtime_get(void);
static void dss_runtime_put(void);

69 70 71
struct dss_features {
	u8 fck_div_max;
	u8 dss_fck_multiplier;
72
	const char *parent_clk_name;
73 74
	enum omap_display_type *ports;
	int num_ports;
75
	int (*dpi_select_source)(int port, enum omap_channel channel);
76 77
};

78
static struct {
79
	struct platform_device *pdev;
80
	void __iomem    *base;
81

82
	struct clk	*parent_clk;
83
	struct clk	*dss_clk;
84
	unsigned long	dss_clk_rate;
85 86 87 88 89

	unsigned long	cache_req_pck;
	unsigned long	cache_prate;
	struct dispc_clock_info cache_dispc_cinfo;

90
	enum omap_dss_clk_source dsi_clk_source[MAX_NUM_DSI];
91 92
	enum omap_dss_clk_source dispc_clk_source;
	enum omap_dss_clk_source lcd_clk_source[MAX_DSS_LCD_MANAGERS];
93

94
	bool		ctx_valid;
95
	u32		ctx[DSS_SZ_REGS / sizeof(u32)];
96 97

	const struct dss_features *feat;
98 99
} dss;

100
static const char * const dss_generic_clk_source_names[] = {
101 102 103
	[OMAP_DSS_CLK_SRC_DSI_PLL_HSDIV_DISPC]	= "DSI_PLL_HSDIV_DISPC",
	[OMAP_DSS_CLK_SRC_DSI_PLL_HSDIV_DSI]	= "DSI_PLL_HSDIV_DSI",
	[OMAP_DSS_CLK_SRC_FCK]			= "DSS_FCK",
T
Tomi Valkeinen 已提交
104 105
	[OMAP_DSS_CLK_SRC_DSI2_PLL_HSDIV_DISPC]	= "DSI_PLL2_HSDIV_DISPC",
	[OMAP_DSS_CLK_SRC_DSI2_PLL_HSDIV_DSI]	= "DSI_PLL2_HSDIV_DSI",
106 107
};

108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
static inline void dss_write_reg(const struct dss_reg idx, u32 val)
{
	__raw_writel(val, dss.base + idx.idx);
}

static inline u32 dss_read_reg(const struct dss_reg idx)
{
	return __raw_readl(dss.base + idx.idx);
}

#define SR(reg) \
	dss.ctx[(DSS_##reg).idx / sizeof(u32)] = dss_read_reg(DSS_##reg)
#define RR(reg) \
	dss_write_reg(DSS_##reg, dss.ctx[(DSS_##reg).idx / sizeof(u32)])

123
static void dss_save_context(void)
124
{
125
	DSSDBG("dss_save_context\n");
126 127 128

	SR(CONTROL);

129 130 131 132 133
	if (dss_feat_get_supported_displays(OMAP_DSS_CHANNEL_LCD) &
			OMAP_DISPLAY_TYPE_SDI) {
		SR(SDI_CONTROL);
		SR(PLL_CONTROL);
	}
134 135 136 137

	dss.ctx_valid = true;

	DSSDBG("context saved\n");
138 139
}

140
static void dss_restore_context(void)
141
{
142
	DSSDBG("dss_restore_context\n");
143

144 145 146
	if (!dss.ctx_valid)
		return;

147 148
	RR(CONTROL);

149 150 151 152 153
	if (dss_feat_get_supported_displays(OMAP_DSS_CHANNEL_LCD) &
			OMAP_DISPLAY_TYPE_SDI) {
		RR(SDI_CONTROL);
		RR(PLL_CONTROL);
	}
154 155

	DSSDBG("context restored\n");
156 157 158 159 160
}

#undef SR
#undef RR

161
void dss_sdi_init(int datapairs)
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247
{
	u32 l;

	BUG_ON(datapairs > 3 || datapairs < 1);

	l = dss_read_reg(DSS_SDI_CONTROL);
	l = FLD_MOD(l, 0xf, 19, 15);		/* SDI_PDIV */
	l = FLD_MOD(l, datapairs-1, 3, 2);	/* SDI_PRSEL */
	l = FLD_MOD(l, 2, 1, 0);		/* SDI_BWSEL */
	dss_write_reg(DSS_SDI_CONTROL, l);

	l = dss_read_reg(DSS_PLL_CONTROL);
	l = FLD_MOD(l, 0x7, 25, 22);	/* SDI_PLL_FREQSEL */
	l = FLD_MOD(l, 0xb, 16, 11);	/* SDI_PLL_REGN */
	l = FLD_MOD(l, 0xb4, 10, 1);	/* SDI_PLL_REGM */
	dss_write_reg(DSS_PLL_CONTROL, l);
}

int dss_sdi_enable(void)
{
	unsigned long timeout;

	dispc_pck_free_enable(1);

	/* Reset SDI PLL */
	REG_FLD_MOD(DSS_PLL_CONTROL, 1, 18, 18); /* SDI_PLL_SYSRESET */
	udelay(1);	/* wait 2x PCLK */

	/* Lock SDI PLL */
	REG_FLD_MOD(DSS_PLL_CONTROL, 1, 28, 28); /* SDI_PLL_GOBIT */

	/* Waiting for PLL lock request to complete */
	timeout = jiffies + msecs_to_jiffies(500);
	while (dss_read_reg(DSS_SDI_STATUS) & (1 << 6)) {
		if (time_after_eq(jiffies, timeout)) {
			DSSERR("PLL lock request timed out\n");
			goto err1;
		}
	}

	/* Clearing PLL_GO bit */
	REG_FLD_MOD(DSS_PLL_CONTROL, 0, 28, 28);

	/* Waiting for PLL to lock */
	timeout = jiffies + msecs_to_jiffies(500);
	while (!(dss_read_reg(DSS_SDI_STATUS) & (1 << 5))) {
		if (time_after_eq(jiffies, timeout)) {
			DSSERR("PLL lock timed out\n");
			goto err1;
		}
	}

	dispc_lcd_enable_signal(1);

	/* Waiting for SDI reset to complete */
	timeout = jiffies + msecs_to_jiffies(500);
	while (!(dss_read_reg(DSS_SDI_STATUS) & (1 << 2))) {
		if (time_after_eq(jiffies, timeout)) {
			DSSERR("SDI reset timed out\n");
			goto err2;
		}
	}

	return 0;

 err2:
	dispc_lcd_enable_signal(0);
 err1:
	/* Reset SDI PLL */
	REG_FLD_MOD(DSS_PLL_CONTROL, 0, 18, 18); /* SDI_PLL_SYSRESET */

	dispc_pck_free_enable(0);

	return -ETIMEDOUT;
}

void dss_sdi_disable(void)
{
	dispc_lcd_enable_signal(0);

	dispc_pck_free_enable(0);

	/* Reset SDI PLL */
	REG_FLD_MOD(DSS_PLL_CONTROL, 0, 18, 18); /* SDI_PLL_SYSRESET */
}

248
const char *dss_get_generic_clk_source_name(enum omap_dss_clk_source clk_src)
249
{
250
	return dss_generic_clk_source_names[clk_src];
251 252
}

253 254
void dss_dump_clocks(struct seq_file *s)
{
255 256
	const char *fclk_name, *fclk_real_name;
	unsigned long fclk_rate;
257

258 259
	if (dss_runtime_get())
		return;
260 261 262

	seq_printf(s, "- DSS -\n");

263 264
	fclk_name = dss_get_generic_clk_source_name(OMAP_DSS_CLK_SRC_FCK);
	fclk_real_name = dss_feat_get_clk_source_name(OMAP_DSS_CLK_SRC_FCK);
265
	fclk_rate = clk_get_rate(dss.dss_clk);
266

T
Tomi Valkeinen 已提交
267 268 269
	seq_printf(s, "%s (%s) = %lu\n",
			fclk_name, fclk_real_name,
			fclk_rate);
270

271
	dss_runtime_put();
272 273
}

274
static void dss_dump_regs(struct seq_file *s)
275 276 277
{
#define DUMPREG(r) seq_printf(s, "%-35s %08x\n", #r, dss_read_reg(r))

278 279
	if (dss_runtime_get())
		return;
280 281 282 283 284

	DUMPREG(DSS_REVISION);
	DUMPREG(DSS_SYSCONFIG);
	DUMPREG(DSS_SYSSTATUS);
	DUMPREG(DSS_CONTROL);
285 286 287 288 289 290 291

	if (dss_feat_get_supported_displays(OMAP_DSS_CHANNEL_LCD) &
			OMAP_DISPLAY_TYPE_SDI) {
		DUMPREG(DSS_SDI_CONTROL);
		DUMPREG(DSS_PLL_CONTROL);
		DUMPREG(DSS_SDI_STATUS);
	}
292

293
	dss_runtime_put();
294 295 296
#undef DUMPREG
}

297
static void dss_select_dispc_clk_source(enum omap_dss_clk_source clk_src)
298
{
299
	struct platform_device *dsidev;
300
	int b;
301
	u8 start, end;
302

303
	switch (clk_src) {
304
	case OMAP_DSS_CLK_SRC_FCK:
305 306
		b = 0;
		break;
307
	case OMAP_DSS_CLK_SRC_DSI_PLL_HSDIV_DISPC:
308
		b = 1;
309 310
		dsidev = dsi_get_dsidev_from_id(0);
		dsi_wait_pll_hsdiv_dispc_active(dsidev);
311
		break;
312 313 314 315 316
	case OMAP_DSS_CLK_SRC_DSI2_PLL_HSDIV_DISPC:
		b = 2;
		dsidev = dsi_get_dsidev_from_id(1);
		dsi_wait_pll_hsdiv_dispc_active(dsidev);
		break;
317 318
	default:
		BUG();
319
		return;
320
	}
321

322 323 324
	dss_feat_get_reg_field(FEAT_REG_DISPC_CLK_SWITCH, &start, &end);

	REG_FLD_MOD(DSS_CONTROL, b, start, end);	/* DISPC_CLK_SWITCH */
325 326 327 328

	dss.dispc_clk_source = clk_src;
}

329 330
void dss_select_dsi_clk_source(int dsi_module,
		enum omap_dss_clk_source clk_src)
331
{
332
	struct platform_device *dsidev;
333
	int b, pos;
334

335
	switch (clk_src) {
336
	case OMAP_DSS_CLK_SRC_FCK:
337 338
		b = 0;
		break;
339
	case OMAP_DSS_CLK_SRC_DSI_PLL_HSDIV_DSI:
340
		BUG_ON(dsi_module != 0);
341
		b = 1;
342 343
		dsidev = dsi_get_dsidev_from_id(0);
		dsi_wait_pll_hsdiv_dsi_active(dsidev);
344
		break;
345 346 347 348 349 350
	case OMAP_DSS_CLK_SRC_DSI2_PLL_HSDIV_DSI:
		BUG_ON(dsi_module != 1);
		b = 1;
		dsidev = dsi_get_dsidev_from_id(1);
		dsi_wait_pll_hsdiv_dsi_active(dsidev);
		break;
351 352
	default:
		BUG();
353
		return;
354
	}
355

356 357
	pos = dsi_module == 0 ? 1 : 10;
	REG_FLD_MOD(DSS_CONTROL, b, pos, pos);	/* DSIx_CLK_SWITCH */
358

359
	dss.dsi_clk_source[dsi_module] = clk_src;
360 361
}

362
void dss_select_lcd_clk_source(enum omap_channel channel,
363
		enum omap_dss_clk_source clk_src)
364
{
365
	struct platform_device *dsidev;
366 367
	int b, ix, pos;

368 369
	if (!dss_has_feature(FEAT_LCD_CLK_SRC)) {
		dss_select_dispc_clk_source(clk_src);
370
		return;
371
	}
372 373

	switch (clk_src) {
374
	case OMAP_DSS_CLK_SRC_FCK:
375 376
		b = 0;
		break;
377
	case OMAP_DSS_CLK_SRC_DSI_PLL_HSDIV_DISPC:
378 379
		BUG_ON(channel != OMAP_DSS_CHANNEL_LCD);
		b = 1;
380 381
		dsidev = dsi_get_dsidev_from_id(0);
		dsi_wait_pll_hsdiv_dispc_active(dsidev);
382
		break;
383
	case OMAP_DSS_CLK_SRC_DSI2_PLL_HSDIV_DISPC:
384 385
		BUG_ON(channel != OMAP_DSS_CHANNEL_LCD2 &&
		       channel != OMAP_DSS_CHANNEL_LCD3);
386 387 388 389
		b = 1;
		dsidev = dsi_get_dsidev_from_id(1);
		dsi_wait_pll_hsdiv_dispc_active(dsidev);
		break;
390 391
	default:
		BUG();
392
		return;
393 394
	}

395 396
	pos = channel == OMAP_DSS_CHANNEL_LCD ? 0 :
	     (channel == OMAP_DSS_CHANNEL_LCD2 ? 12 : 19);
397 398
	REG_FLD_MOD(DSS_CONTROL, b, pos, pos);	/* LCDx_CLK_SWITCH */

399 400
	ix = channel == OMAP_DSS_CHANNEL_LCD ? 0 :
	    (channel == OMAP_DSS_CHANNEL_LCD2 ? 1 : 2);
401 402 403
	dss.lcd_clk_source[ix] = clk_src;
}

404
enum omap_dss_clk_source dss_get_dispc_clk_source(void)
405
{
406
	return dss.dispc_clk_source;
407 408
}

409
enum omap_dss_clk_source dss_get_dsi_clk_source(int dsi_module)
410
{
411
	return dss.dsi_clk_source[dsi_module];
412 413
}

414
enum omap_dss_clk_source dss_get_lcd_clk_source(enum omap_channel channel)
415
{
416
	if (dss_has_feature(FEAT_LCD_CLK_SRC)) {
417 418
		int ix = channel == OMAP_DSS_CHANNEL_LCD ? 0 :
			(channel == OMAP_DSS_CHANNEL_LCD2 ? 1 : 2);
419 420 421 422 423 424
		return dss.lcd_clk_source[ix];
	} else {
		/* LCD_CLK source is the same as DISPC_FCLK source for
		 * OMAP2 and OMAP3 */
		return dss.dispc_clk_source;
	}
425 426
}

427 428
bool dss_div_calc(unsigned long pck, unsigned long fck_min,
		dss_div_calc_func func, void *data)
429 430 431 432 433 434
{
	int fckd, fckd_start, fckd_stop;
	unsigned long fck;
	unsigned long fck_hw_max;
	unsigned long fckd_hw_max;
	unsigned long prate;
435
	unsigned m;
436

437 438
	fck_hw_max = dss_feat_get_param_max(FEAT_PARAM_DSS_FCK);

439
	if (dss.parent_clk == NULL) {
440 441 442 443 444 445 446 447
		unsigned pckd;

		pckd = fck_hw_max / pck;

		fck = pck * pckd;

		fck = clk_round_rate(dss.dss_clk, fck);

448
		return func(fck, data);
449 450 451 452
	}

	fckd_hw_max = dss.feat->fck_div_max;

453
	m = dss.feat->dss_fck_multiplier;
454
	prate = clk_get_rate(dss.parent_clk);
455 456 457

	fck_min = fck_min ? fck_min : 1;

458 459
	fckd_start = min(prate * m / fck_min, fckd_hw_max);
	fckd_stop = max(DIV_ROUND_UP(prate * m, fck_hw_max), 1ul);
460 461

	for (fckd = fckd_start; fckd >= fckd_stop; --fckd) {
462
		fck = DIV_ROUND_UP(prate, fckd) * m;
463

464
		if (func(fck, data))
465 466 467 468 469 470
			return true;
	}

	return false;
}

471
int dss_set_fck_rate(unsigned long rate)
472
{
473
	int r;
474

475
	DSSDBG("set fck to %lu\n", rate);
476

477 478 479
	r = clk_set_rate(dss.dss_clk, rate);
	if (r)
		return r;
480

481 482
	dss.dss_clk_rate = clk_get_rate(dss.dss_clk);

483
	WARN_ONCE(dss.dss_clk_rate != rate,
484
			"clk rate mismatch: %lu != %lu", dss.dss_clk_rate,
485
			rate);
486 487 488 489

	return 0;
}

490 491 492 493 494
unsigned long dss_get_dispc_clk_rate(void)
{
	return dss.dss_clk_rate;
}

T
Tomi Valkeinen 已提交
495 496 497
static int dss_setup_default_clock(void)
{
	unsigned long max_dss_fck, prate;
498
	unsigned long fck;
T
Tomi Valkeinen 已提交
499 500 501 502 503
	unsigned fck_div;
	int r;

	max_dss_fck = dss_feat_get_param_max(FEAT_PARAM_DSS_FCK);

504 505 506 507
	if (dss.parent_clk == NULL) {
		fck = clk_round_rate(dss.dss_clk, max_dss_fck);
	} else {
		prate = clk_get_rate(dss.parent_clk);
T
Tomi Valkeinen 已提交
508

509 510
		fck_div = DIV_ROUND_UP(prate * dss.feat->dss_fck_multiplier,
				max_dss_fck);
511
		fck = DIV_ROUND_UP(prate, fck_div) * dss.feat->dss_fck_multiplier;
512
	}
T
Tomi Valkeinen 已提交
513

514
	r = dss_set_fck_rate(fck);
T
Tomi Valkeinen 已提交
515 516 517 518 519 520
	if (r)
		return r;

	return 0;
}

521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540
void dss_set_venc_output(enum omap_dss_venc_type type)
{
	int l = 0;

	if (type == OMAP_DSS_VENC_TYPE_COMPOSITE)
		l = 0;
	else if (type == OMAP_DSS_VENC_TYPE_SVIDEO)
		l = 1;
	else
		BUG();

	/* venc out selection. 0 = comp, 1 = svideo */
	REG_FLD_MOD(DSS_CONTROL, l, 6, 6);
}

void dss_set_dac_pwrdn_bgz(bool enable)
{
	REG_FLD_MOD(DSS_CONTROL, enable, 5, 5);	/* DAC Power-Down Control */
}

541
void dss_select_hdmi_venc_clk_source(enum dss_hdmi_venc_clk_source_select src)
542
{
543 544 545 546 547 548 549 550 551 552
	enum omap_display_type dp;
	dp = dss_feat_get_supported_displays(OMAP_DSS_CHANNEL_DIGIT);

	/* Complain about invalid selections */
	WARN_ON((src == DSS_VENC_TV_CLK) && !(dp & OMAP_DISPLAY_TYPE_VENC));
	WARN_ON((src == DSS_HDMI_M_PCLK) && !(dp & OMAP_DISPLAY_TYPE_HDMI));

	/* Select only if we have options */
	if ((dp & OMAP_DISPLAY_TYPE_VENC) && (dp & OMAP_DISPLAY_TYPE_HDMI))
		REG_FLD_MOD(DSS_CONTROL, src, 15, 15);	/* VENC_HDMI_SWITCH */
553 554
}

555 556 557 558 559 560 561 562
enum dss_hdmi_venc_clk_source_select dss_get_hdmi_venc_clk_source(void)
{
	enum omap_display_type displays;

	displays = dss_feat_get_supported_displays(OMAP_DSS_CHANNEL_DIGIT);
	if ((displays & OMAP_DISPLAY_TYPE_HDMI) == 0)
		return DSS_VENC_TV_CLK;

563 564 565
	if ((displays & OMAP_DISPLAY_TYPE_VENC) == 0)
		return DSS_HDMI_M_PCLK;

566 567 568
	return REG_GET(DSS_CONTROL, 15, 15);
}

569
static int dss_dpi_select_source_omap2_omap3(int port, enum omap_channel channel)
570 571 572 573 574 575 576
{
	if (channel != OMAP_DSS_CHANNEL_LCD)
		return -EINVAL;

	return 0;
}

577
static int dss_dpi_select_source_omap4(int port, enum omap_channel channel)
578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596
{
	int val;

	switch (channel) {
	case OMAP_DSS_CHANNEL_LCD2:
		val = 0;
		break;
	case OMAP_DSS_CHANNEL_DIGIT:
		val = 1;
		break;
	default:
		return -EINVAL;
	}

	REG_FLD_MOD(DSS_CONTROL, val, 17, 17);

	return 0;
}

597
static int dss_dpi_select_source_omap5(int port, enum omap_channel channel)
598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622
{
	int val;

	switch (channel) {
	case OMAP_DSS_CHANNEL_LCD:
		val = 1;
		break;
	case OMAP_DSS_CHANNEL_LCD2:
		val = 2;
		break;
	case OMAP_DSS_CHANNEL_LCD3:
		val = 3;
		break;
	case OMAP_DSS_CHANNEL_DIGIT:
		val = 0;
		break;
	default:
		return -EINVAL;
	}

	REG_FLD_MOD(DSS_CONTROL, val, 17, 16);

	return 0;
}

623
int dss_dpi_select_source(int port, enum omap_channel channel)
624
{
625
	return dss.feat->dpi_select_source(port, channel);
626 627
}

628 629
static int dss_get_clocks(void)
{
630
	struct clk *clk;
631

A
Archit Taneja 已提交
632
	clk = devm_clk_get(&dss.pdev->dev, "fck");
633 634
	if (IS_ERR(clk)) {
		DSSERR("can't get clock fck\n");
A
Archit Taneja 已提交
635
		return PTR_ERR(clk);
636
	}
637

638
	dss.dss_clk = clk;
639

640 641
	if (dss.feat->parent_clk_name) {
		clk = clk_get(NULL, dss.feat->parent_clk_name);
642
		if (IS_ERR(clk)) {
643
			DSSERR("Failed to get %s\n", dss.feat->parent_clk_name);
A
Archit Taneja 已提交
644
			return PTR_ERR(clk);
645 646 647
		}
	} else {
		clk = NULL;
648 649
	}

650
	dss.parent_clk = clk;
651

652 653 654 655 656
	return 0;
}

static void dss_put_clocks(void)
{
657 658
	if (dss.parent_clk)
		clk_put(dss.parent_clk);
659 660
}

661
static int dss_runtime_get(void)
662
{
663
	int r;
664

665
	DSSDBG("dss_runtime_get\n");
666

667 668 669
	r = pm_runtime_get_sync(&dss.pdev->dev);
	WARN_ON(r < 0);
	return r < 0 ? r : 0;
670 671
}

672
static void dss_runtime_put(void)
673
{
674
	int r;
675

676
	DSSDBG("dss_runtime_put\n");
677

678
	r = pm_runtime_put_sync(&dss.pdev->dev);
679
	WARN_ON(r < 0 && r != -ENOSYS && r != -EBUSY);
680 681 682
}

/* DEBUGFS */
683
#if defined(CONFIG_OMAP2_DSS_DEBUGFS)
684 685 686 687 688 689 690 691 692 693
void dss_debug_dump_clocks(struct seq_file *s)
{
	dss_dump_clocks(s);
	dispc_dump_clocks(s);
#ifdef CONFIG_OMAP2_DSS_DSI
	dsi_dump_clocks(s);
#endif
}
#endif

694 695 696 697 698 699 700 701 702 703

static enum omap_display_type omap2plus_ports[] = {
	OMAP_DISPLAY_TYPE_DPI,
};

static enum omap_display_type omap34xx_ports[] = {
	OMAP_DISPLAY_TYPE_DPI,
	OMAP_DISPLAY_TYPE_SDI,
};

704
static const struct dss_features omap24xx_dss_feats __initconst = {
705 706 707 708 709
	/*
	 * fck div max is really 16, but the divider range has gaps. The range
	 * from 1 to 6 has no gaps, so let's use that as a max.
	 */
	.fck_div_max		=	6,
710
	.dss_fck_multiplier	=	2,
711
	.parent_clk_name	=	"core_ck",
712
	.dpi_select_source	=	&dss_dpi_select_source_omap2_omap3,
713 714
	.ports			=	omap2plus_ports,
	.num_ports		=	ARRAY_SIZE(omap2plus_ports),
715 716 717 718 719
};

static const struct dss_features omap34xx_dss_feats __initconst = {
	.fck_div_max		=	16,
	.dss_fck_multiplier	=	2,
720
	.parent_clk_name	=	"dpll4_ck",
721
	.dpi_select_source	=	&dss_dpi_select_source_omap2_omap3,
722 723
	.ports			=	omap34xx_ports,
	.num_ports		=	ARRAY_SIZE(omap34xx_ports),
724 725 726 727 728
};

static const struct dss_features omap3630_dss_feats __initconst = {
	.fck_div_max		=	32,
	.dss_fck_multiplier	=	1,
729
	.parent_clk_name	=	"dpll4_ck",
730
	.dpi_select_source	=	&dss_dpi_select_source_omap2_omap3,
731 732
	.ports			=	omap2plus_ports,
	.num_ports		=	ARRAY_SIZE(omap2plus_ports),
733 734 735 736 737
};

static const struct dss_features omap44xx_dss_feats __initconst = {
	.fck_div_max		=	32,
	.dss_fck_multiplier	=	1,
738
	.parent_clk_name	=	"dpll_per_x2_ck",
739
	.dpi_select_source	=	&dss_dpi_select_source_omap4,
740 741
	.ports			=	omap2plus_ports,
	.num_ports		=	ARRAY_SIZE(omap2plus_ports),
742 743 744 745 746
};

static const struct dss_features omap54xx_dss_feats __initconst = {
	.fck_div_max		=	64,
	.dss_fck_multiplier	=	1,
747
	.parent_clk_name	=	"dpll_per_x2_ck",
748
	.dpi_select_source	=	&dss_dpi_select_source_omap5,
749 750
	.ports			=	omap2plus_ports,
	.num_ports		=	ARRAY_SIZE(omap2plus_ports),
751 752
};

753 754 755 756 757
static const struct dss_features am43xx_dss_feats __initconst = {
	.fck_div_max		=	0,
	.dss_fck_multiplier	=	0,
	.parent_clk_name	=	NULL,
	.dpi_select_source	=	&dss_dpi_select_source_omap2_omap3,
758 759
	.ports			=	omap2plus_ports,
	.num_ports		=	ARRAY_SIZE(omap2plus_ports),
760 761
};

T
Tomi Valkeinen 已提交
762
static int __init dss_init_features(struct platform_device *pdev)
763 764 765 766
{
	const struct dss_features *src;
	struct dss_features *dst;

T
Tomi Valkeinen 已提交
767
	dst = devm_kzalloc(&pdev->dev, sizeof(*dst), GFP_KERNEL);
768
	if (!dst) {
T
Tomi Valkeinen 已提交
769
		dev_err(&pdev->dev, "Failed to allocate local DSS Features\n");
770 771 772
		return -ENOMEM;
	}

773
	switch (omapdss_get_version()) {
T
Tomi Valkeinen 已提交
774
	case OMAPDSS_VER_OMAP24xx:
775
		src = &omap24xx_dss_feats;
T
Tomi Valkeinen 已提交
776 777 778 779 780
		break;

	case OMAPDSS_VER_OMAP34xx_ES1:
	case OMAPDSS_VER_OMAP34xx_ES3:
	case OMAPDSS_VER_AM35xx:
781
		src = &omap34xx_dss_feats;
T
Tomi Valkeinen 已提交
782 783 784
		break;

	case OMAPDSS_VER_OMAP3630:
785
		src = &omap3630_dss_feats;
T
Tomi Valkeinen 已提交
786 787 788 789 790
		break;

	case OMAPDSS_VER_OMAP4430_ES1:
	case OMAPDSS_VER_OMAP4430_ES2:
	case OMAPDSS_VER_OMAP4:
791
		src = &omap44xx_dss_feats;
T
Tomi Valkeinen 已提交
792 793 794
		break;

	case OMAPDSS_VER_OMAP5:
795
		src = &omap54xx_dss_feats;
T
Tomi Valkeinen 已提交
796 797
		break;

798 799 800 801
	case OMAPDSS_VER_AM43xx:
		src = &am43xx_dss_feats;
		break;

T
Tomi Valkeinen 已提交
802
	default:
803
		return -ENODEV;
T
Tomi Valkeinen 已提交
804
	}
805 806 807 808 809 810 811

	memcpy(dst, src, sizeof(*dst));
	dss.feat = dst;

	return 0;
}

812
static int __init dss_init_ports(struct platform_device *pdev)
T
Tomi Valkeinen 已提交
813 814 815 816 817 818 819 820 821
{
	struct device_node *parent = pdev->dev.of_node;
	struct device_node *port;
	int r;

	if (parent == NULL)
		return 0;

	port = omapdss_of_get_next_port(parent, NULL);
822
	if (!port)
T
Tomi Valkeinen 已提交
823 824
		return 0;

825 826 827
	if (dss.feat->num_ports == 0)
		return 0;

T
Tomi Valkeinen 已提交
828
	do {
829
		enum omap_display_type port_type;
T
Tomi Valkeinen 已提交
830 831 832 833 834 835
		u32 reg;

		r = of_property_read_u32(port, "reg", &reg);
		if (r)
			reg = 0;

836 837
		if (reg >= dss.feat->num_ports)
			continue;
T
Tomi Valkeinen 已提交
838

839
		port_type = dss.feat->ports[reg];
T
Tomi Valkeinen 已提交
840

841 842 843 844 845 846 847 848 849 850
		switch (port_type) {
		case OMAP_DISPLAY_TYPE_DPI:
			dpi_init_port(pdev, port);
			break;
		case OMAP_DISPLAY_TYPE_SDI:
			sdi_init_port(pdev, port);
			break;
		default:
			break;
		}
T
Tomi Valkeinen 已提交
851 852 853 854 855
	} while ((port = omapdss_of_get_next_port(parent, port)) != NULL);

	return 0;
}

A
Archit Taneja 已提交
856
static void __exit dss_uninit_ports(struct platform_device *pdev)
T
Tomi Valkeinen 已提交
857
{
858 859 860 861 862 863 864 865 866 867
	struct device_node *parent = pdev->dev.of_node;
	struct device_node *port;

	if (parent == NULL)
		return;

	port = omapdss_of_get_next_port(parent, NULL);
	if (!port)
		return;

868 869
	if (dss.feat->num_ports == 0)
		return;
T
Tomi Valkeinen 已提交
870

871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895
	do {
		enum omap_display_type port_type;
		u32 reg;
		int r;

		r = of_property_read_u32(port, "reg", &reg);
		if (r)
			reg = 0;

		if (reg >= dss.feat->num_ports)
			continue;

		port_type = dss.feat->ports[reg];

		switch (port_type) {
		case OMAP_DISPLAY_TYPE_DPI:
			dpi_uninit_port(port);
			break;
		case OMAP_DISPLAY_TYPE_SDI:
			sdi_uninit_port(port);
			break;
		default:
			break;
		}
	} while ((port = omapdss_of_get_next_port(parent, port)) != NULL);
T
Tomi Valkeinen 已提交
896 897
}

898
/* DSS HW IP initialisation */
T
Tomi Valkeinen 已提交
899
static int __init omap_dsshw_probe(struct platform_device *pdev)
900
{
901 902
	struct resource *dss_mem;
	u32 rev;
903 904 905 906
	int r;

	dss.pdev = pdev;

T
Tomi Valkeinen 已提交
907
	r = dss_init_features(dss.pdev);
908 909 910
	if (r)
		return r;

911 912 913
	dss_mem = platform_get_resource(dss.pdev, IORESOURCE_MEM, 0);
	if (!dss_mem) {
		DSSERR("can't get IORESOURCE_MEM DSS\n");
914
		return -EINVAL;
915
	}
916

J
Julia Lawall 已提交
917 918
	dss.base = devm_ioremap(&pdev->dev, dss_mem->start,
				resource_size(dss_mem));
919 920
	if (!dss.base) {
		DSSERR("can't ioremap DSS\n");
921
		return -ENOMEM;
922 923
	}

924 925
	r = dss_get_clocks();
	if (r)
926
		return r;
927

T
Tomi Valkeinen 已提交
928 929 930 931
	r = dss_setup_default_clock();
	if (r)
		goto err_setup_clocks;

932
	pm_runtime_enable(&pdev->dev);
933

934 935 936
	r = dss_runtime_get();
	if (r)
		goto err_runtime_get;
937

938 939
	dss.dss_clk_rate = clk_get_rate(dss.dss_clk);

940 941 942
	/* Select DPLL */
	REG_FLD_MOD(DSS_CONTROL, 0, 0, 0);

943 944
	dss_select_dispc_clk_source(OMAP_DSS_CLK_SRC_FCK);

945 946 947 948 949 950 951 952 953 954
#ifdef CONFIG_OMAP2_DSS_VENC
	REG_FLD_MOD(DSS_CONTROL, 1, 4, 4);	/* venc dac demen */
	REG_FLD_MOD(DSS_CONTROL, 1, 3, 3);	/* venc clock 4x enable */
	REG_FLD_MOD(DSS_CONTROL, 0, 2, 2);	/* venc clock mode = normal */
#endif
	dss.dsi_clk_source[0] = OMAP_DSS_CLK_SRC_FCK;
	dss.dsi_clk_source[1] = OMAP_DSS_CLK_SRC_FCK;
	dss.dispc_clk_source = OMAP_DSS_CLK_SRC_FCK;
	dss.lcd_clk_source[0] = OMAP_DSS_CLK_SRC_FCK;
	dss.lcd_clk_source[1] = OMAP_DSS_CLK_SRC_FCK;
955

T
Tomi Valkeinen 已提交
956 957
	dss_init_ports(pdev);

958 959 960 961
	rev = dss_read_reg(DSS_REVISION);
	printk(KERN_INFO "OMAP DSS rev %d.%d\n",
			FLD_GET(rev, 7, 4), FLD_GET(rev, 3, 0));

962
	dss_runtime_put();
963

964 965
	dss_debugfs_create_file("dss", dss_dump_regs);

966
	return 0;
967

968 969
err_runtime_get:
	pm_runtime_disable(&pdev->dev);
T
Tomi Valkeinen 已提交
970
err_setup_clocks:
971
	dss_put_clocks();
972 973 974
	return r;
}

T
Tomi Valkeinen 已提交
975
static int __exit omap_dsshw_remove(struct platform_device *pdev)
976
{
A
Archit Taneja 已提交
977
	dss_uninit_ports(pdev);
T
Tomi Valkeinen 已提交
978

979
	pm_runtime_disable(&pdev->dev);
980 981

	dss_put_clocks();
982

983 984 985
	return 0;
}

986 987 988
static int dss_runtime_suspend(struct device *dev)
{
	dss_save_context();
989
	dss_set_min_bus_tput(dev, 0);
990 991 992 993 994
	return 0;
}

static int dss_runtime_resume(struct device *dev)
{
995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006
	int r;
	/*
	 * Set an arbitrarily high tput request to ensure OPP100.
	 * What we should really do is to make a request to stay in OPP100,
	 * without any tput requirements, but that is not currently possible
	 * via the PM layer.
	 */

	r = dss_set_min_bus_tput(dev, 1000000000);
	if (r)
		return r;

1007
	dss_restore_context();
1008 1009 1010 1011 1012 1013 1014 1015
	return 0;
}

static const struct dev_pm_ops dss_pm_ops = {
	.runtime_suspend = dss_runtime_suspend,
	.runtime_resume = dss_runtime_resume,
};

T
Tomi Valkeinen 已提交
1016 1017 1018 1019
static const struct of_device_id dss_of_match[] = {
	{ .compatible = "ti,omap2-dss", },
	{ .compatible = "ti,omap3-dss", },
	{ .compatible = "ti,omap4-dss", },
1020
	{ .compatible = "ti,omap5-dss", },
T
Tomi Valkeinen 已提交
1021 1022 1023 1024 1025
	{},
};

MODULE_DEVICE_TABLE(of, dss_of_match);

1026
static struct platform_driver omap_dsshw_driver = {
T
Tomi Valkeinen 已提交
1027
	.remove         = __exit_p(omap_dsshw_remove),
1028 1029 1030
	.driver         = {
		.name   = "omapdss_dss",
		.owner  = THIS_MODULE,
1031
		.pm	= &dss_pm_ops,
T
Tomi Valkeinen 已提交
1032
		.of_match_table = dss_of_match,
T
Tomi Valkeinen 已提交
1033
		.suppress_bind_attrs = true,
1034 1035 1036
	},
};

T
Tomi Valkeinen 已提交
1037
int __init dss_init_platform_driver(void)
1038
{
1039
	return platform_driver_probe(&omap_dsshw_driver, omap_dsshw_probe);
1040 1041 1042 1043
}

void dss_uninit_platform_driver(void)
{
1044
	platform_driver_unregister(&omap_dsshw_driver);
1045
}