exynos_drm_fimd.c 29.2 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
/* exynos_drm_fimd.c
 *
 * Copyright (C) 2011 Samsung Electronics Co.Ltd
 * Authors:
 *	Joonyoung Shim <jy0922.shim@samsung.com>
 *	Inki Dae <inki.dae@samsung.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.
 *
 */
14
#include <drm/drmP.h>
15 16 17 18

#include <linux/kernel.h>
#include <linux/platform_device.h>
#include <linux/clk.h>
19
#include <linux/of.h>
20
#include <linux/of_device.h>
21
#include <linux/pm_runtime.h>
22
#include <linux/component.h>
23 24
#include <linux/mfd/syscon.h>
#include <linux/regmap.h>
25

26
#include <video/of_display_timing.h>
27
#include <video/of_videomode.h>
28
#include <video/samsung_fimd.h>
29 30 31
#include <drm/exynos_drm.h>

#include "exynos_drm_drv.h"
32
#include "exynos_drm_fb.h"
33 34
#include "exynos_drm_fbdev.h"
#include "exynos_drm_crtc.h"
35
#include "exynos_drm_plane.h"
36
#include "exynos_drm_iommu.h"
37 38

/*
39
 * FIMD stands for Fully Interactive Mobile Display and
40 41 42 43 44
 * as a display controller, it transfers contents drawn on memory
 * to a LCD Panel through Display Interfaces such as RGB or
 * CPU Interface.
 */

45
#define MIN_FB_WIDTH_FOR_16WORD_BURST 128
46

47 48 49
/* position control register for hardware window 0, 2 ~ 4.*/
#define VIDOSD_A(win)		(VIDOSD_BASE + 0x00 + (win) * 16)
#define VIDOSD_B(win)		(VIDOSD_BASE + 0x04 + (win) * 16)
50 51 52 53 54 55
/*
 * size control register for hardware windows 0 and alpha control register
 * for hardware windows 1 ~ 4
 */
#define VIDOSD_C(win)		(VIDOSD_BASE + 0x08 + (win) * 16)
/* size control register for hardware windows 1 ~ 2. */
56 57
#define VIDOSD_D(win)		(VIDOSD_BASE + 0x0C + (win) * 16)

58 59 60
#define VIDWnALPHA0(win)	(VIDW_ALPHA + 0x00 + (win) * 8)
#define VIDWnALPHA1(win)	(VIDW_ALPHA + 0x04 + (win) * 8)

61
#define VIDWx_BUF_START(win, buf)	(VIDW_BUF_START(buf) + (win) * 8)
62
#define VIDWx_BUF_START_S(win, buf)	(VIDW_BUF_START_S(buf) + (win) * 8)
63 64 65 66
#define VIDWx_BUF_END(win, buf)		(VIDW_BUF_END(buf) + (win) * 8)
#define VIDWx_BUF_SIZE(win, buf)	(VIDW_BUF_SIZE(buf) + (win) * 4)

/* color key control register for hardware window 1 ~ 4. */
67
#define WKEYCON0_BASE(x)		((WKEYCON0 + 0x140) + ((x - 1) * 8))
68
/* color key value register for hardware window 1 ~ 4. */
69
#define WKEYCON1_BASE(x)		((WKEYCON1 + 0x140) + ((x - 1) * 8))
70

71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
/* I80 / RGB trigger control register */
#define TRIGCON				0x1A4
#define TRGMODE_I80_RGB_ENABLE_I80	(1 << 0)
#define SWTRGCMD_I80_RGB_ENABLE		(1 << 1)

/* display mode change control register except exynos4 */
#define VIDOUT_CON			0x000
#define VIDOUT_CON_F_I80_LDI0		(0x2 << 8)

/* I80 interface control for main LDI register */
#define I80IFCONFAx(x)			(0x1B0 + (x) * 4)
#define I80IFCONFBx(x)			(0x1B8 + (x) * 4)
#define LCD_CS_SETUP(x)			((x) << 16)
#define LCD_WR_SETUP(x)			((x) << 12)
#define LCD_WR_ACTIVE(x)		((x) << 8)
#define LCD_WR_HOLD(x)			((x) << 4)
#define I80IFEN_ENABLE			(1 << 0)

89 90
/* FIMD has totally five hardware windows. */
#define WINDOWS_NR	5
91
#define CURSOR_WIN	4
92

93 94
struct fimd_driver_data {
	unsigned int timing_base;
95 96 97
	unsigned int lcdblk_offset;
	unsigned int lcdblk_vt_shift;
	unsigned int lcdblk_bypass_shift;
98 99

	unsigned int has_shadowcon:1;
100
	unsigned int has_clksel:1;
101
	unsigned int has_limited_fmt:1;
102
	unsigned int has_vidoutcon:1;
J
Joonyoung Shim 已提交
103
	unsigned int has_vtsel:1;
104 105
};

106 107 108
static struct fimd_driver_data s3c64xx_fimd_driver_data = {
	.timing_base = 0x0,
	.has_clksel = 1,
109
	.has_limited_fmt = 1,
110 111
};

112 113 114 115 116 117 118 119
static struct fimd_driver_data exynos3_fimd_driver_data = {
	.timing_base = 0x20000,
	.lcdblk_offset = 0x210,
	.lcdblk_bypass_shift = 1,
	.has_shadowcon = 1,
	.has_vidoutcon = 1,
};

120
static struct fimd_driver_data exynos4_fimd_driver_data = {
121
	.timing_base = 0x0,
122 123 124
	.lcdblk_offset = 0x210,
	.lcdblk_vt_shift = 10,
	.lcdblk_bypass_shift = 1,
125
	.has_shadowcon = 1,
J
Joonyoung Shim 已提交
126
	.has_vtsel = 1,
127 128
};

129 130 131 132 133 134 135
static struct fimd_driver_data exynos4415_fimd_driver_data = {
	.timing_base = 0x20000,
	.lcdblk_offset = 0x210,
	.lcdblk_vt_shift = 10,
	.lcdblk_bypass_shift = 1,
	.has_shadowcon = 1,
	.has_vidoutcon = 1,
J
Joonyoung Shim 已提交
136
	.has_vtsel = 1,
137 138
};

139
static struct fimd_driver_data exynos5_fimd_driver_data = {
140
	.timing_base = 0x20000,
141 142 143
	.lcdblk_offset = 0x214,
	.lcdblk_vt_shift = 24,
	.lcdblk_bypass_shift = 15,
144
	.has_shadowcon = 1,
145
	.has_vidoutcon = 1,
J
Joonyoung Shim 已提交
146
	.has_vtsel = 1,
147 148
};

149
struct fimd_context {
150
	struct device			*dev;
151
	struct drm_device		*drm_dev;
152
	struct exynos_drm_crtc		*crtc;
153
	struct exynos_drm_plane		planes[WINDOWS_NR];
154 155 156
	struct clk			*bus_clk;
	struct clk			*lcd_clk;
	void __iomem			*regs;
157
	struct regmap			*sysreg;
158
	unsigned long			irq_flags;
159
	u32				vidcon0;
160
	u32				vidcon1;
161 162 163
	u32				vidout_con;
	u32				i80ifcon;
	bool				i80_if;
164
	bool				suspended;
165
	int				pipe;
166 167
	wait_queue_head_t		wait_vsync_queue;
	atomic_t			wait_vsync_event;
168 169
	atomic_t			win_updated;
	atomic_t			triggering;
170

171
	struct exynos_drm_panel_info panel;
172
	struct fimd_driver_data *driver_data;
173
	struct drm_encoder *encoder;
174 175
};

176
static const struct of_device_id fimd_driver_dt_match[] = {
177 178
	{ .compatible = "samsung,s3c6400-fimd",
	  .data = &s3c64xx_fimd_driver_data },
179 180
	{ .compatible = "samsung,exynos3250-fimd",
	  .data = &exynos3_fimd_driver_data },
181
	{ .compatible = "samsung,exynos4210-fimd",
182
	  .data = &exynos4_fimd_driver_data },
183 184
	{ .compatible = "samsung,exynos4415-fimd",
	  .data = &exynos4415_fimd_driver_data },
185
	{ .compatible = "samsung,exynos5250-fimd",
186 187 188
	  .data = &exynos5_fimd_driver_data },
	{},
};
189
MODULE_DEVICE_TABLE(of, fimd_driver_dt_match);
190

191 192 193 194 195 196 197 198
static const uint32_t fimd_formats[] = {
	DRM_FORMAT_C8,
	DRM_FORMAT_XRGB1555,
	DRM_FORMAT_RGB565,
	DRM_FORMAT_XRGB8888,
	DRM_FORMAT_ARGB8888,
};

199 200 201
static inline struct fimd_driver_data *drm_fimd_get_driver_data(
	struct platform_device *pdev)
{
202 203 204
	const struct of_device_id *of_id =
			of_match_device(fimd_driver_dt_match, &pdev->dev);

205
	return (struct fimd_driver_data *)of_id->data;
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 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263
static int fimd_enable_vblank(struct exynos_drm_crtc *crtc)
{
	struct fimd_context *ctx = crtc->ctx;
	u32 val;

	if (ctx->suspended)
		return -EPERM;

	if (!test_and_set_bit(0, &ctx->irq_flags)) {
		val = readl(ctx->regs + VIDINTCON0);

		val |= VIDINTCON0_INT_ENABLE;

		if (ctx->i80_if) {
			val |= VIDINTCON0_INT_I80IFDONE;
			val |= VIDINTCON0_INT_SYSMAINCON;
			val &= ~VIDINTCON0_INT_SYSSUBCON;
		} else {
			val |= VIDINTCON0_INT_FRAME;

			val &= ~VIDINTCON0_FRAMESEL0_MASK;
			val |= VIDINTCON0_FRAMESEL0_VSYNC;
			val &= ~VIDINTCON0_FRAMESEL1_MASK;
			val |= VIDINTCON0_FRAMESEL1_NONE;
		}

		writel(val, ctx->regs + VIDINTCON0);
	}

	return 0;
}

static void fimd_disable_vblank(struct exynos_drm_crtc *crtc)
{
	struct fimd_context *ctx = crtc->ctx;
	u32 val;

	if (ctx->suspended)
		return;

	if (test_and_clear_bit(0, &ctx->irq_flags)) {
		val = readl(ctx->regs + VIDINTCON0);

		val &= ~VIDINTCON0_INT_ENABLE;

		if (ctx->i80_if) {
			val &= ~VIDINTCON0_INT_I80IFDONE;
			val &= ~VIDINTCON0_INT_SYSMAINCON;
			val &= ~VIDINTCON0_INT_SYSSUBCON;
		} else
			val &= ~VIDINTCON0_INT_FRAME;

		writel(val, ctx->regs + VIDINTCON0);
	}
}

264
static void fimd_wait_for_vblank(struct exynos_drm_crtc *crtc)
265
{
266
	struct fimd_context *ctx = crtc->ctx;
267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282

	if (ctx->suspended)
		return;

	atomic_set(&ctx->wait_vsync_event, 1);

	/*
	 * wait for FIMD to signal VSYNC interrupt or return after
	 * timeout which is set to 50ms (refresh rate of 20).
	 */
	if (!wait_event_timeout(ctx->wait_vsync_queue,
				!atomic_read(&ctx->wait_vsync_event),
				HZ/20))
		DRM_DEBUG_KMS("vblank wait timed out.\n");
}

283
static void fimd_enable_video_output(struct fimd_context *ctx, unsigned int win,
284 285 286 287 288 289 290 291 292 293 294 295
					bool enable)
{
	u32 val = readl(ctx->regs + WINCON(win));

	if (enable)
		val |= WINCONx_ENWIN;
	else
		val &= ~WINCONx_ENWIN;

	writel(val, ctx->regs + WINCON(win));
}

296 297
static void fimd_enable_shadow_channel_path(struct fimd_context *ctx,
						unsigned int win,
298 299 300 301 302 303 304 305 306 307 308 309
						bool enable)
{
	u32 val = readl(ctx->regs + SHADOWCON);

	if (enable)
		val |= SHADOWCON_CHx_ENABLE(win);
	else
		val &= ~SHADOWCON_CHx_ENABLE(win);

	writel(val, ctx->regs + SHADOWCON);
}

310
static void fimd_clear_channels(struct exynos_drm_crtc *crtc)
311
{
312
	struct fimd_context *ctx = crtc->ctx;
313
	unsigned int win, ch_enabled = 0;
314 315 316

	DRM_DEBUG_KMS("%s\n", __FILE__);

317 318 319 320 321 322
	/* Hardware is in unknown state, so ensure it gets enabled properly */
	pm_runtime_get_sync(ctx->dev);

	clk_prepare_enable(ctx->bus_clk);
	clk_prepare_enable(ctx->lcd_clk);

323 324
	/* Check if any channel is enabled. */
	for (win = 0; win < WINDOWS_NR; win++) {
325 326 327
		u32 val = readl(ctx->regs + WINCON(win));

		if (val & WINCONx_ENWIN) {
328
			fimd_enable_video_output(ctx, win, false);
329

330 331 332 333
			if (ctx->driver_data->has_shadowcon)
				fimd_enable_shadow_channel_path(ctx, win,
								false);

334 335 336 337 338
			ch_enabled = 1;
		}
	}

	/* Wait for vsync, as disable channel takes effect at next vsync */
339
	if (ch_enabled) {
340 341 342 343 344
		int pipe = ctx->pipe;

		/* ensure that vblank interrupt won't be reported to core */
		ctx->suspended = false;
		ctx->pipe = -1;
345

346
		fimd_enable_vblank(ctx->crtc);
347
		fimd_wait_for_vblank(ctx->crtc);
348 349 350 351
		fimd_disable_vblank(ctx->crtc);

		ctx->suspended = true;
		ctx->pipe = pipe;
352
	}
353 354 355 356 357

	clk_disable_unprepare(ctx->lcd_clk);
	clk_disable_unprepare(ctx->bus_clk);

	pm_runtime_put(ctx->dev);
358 359
}

360 361 362 363 364 365
static u32 fimd_calc_clkdiv(struct fimd_context *ctx,
		const struct drm_display_mode *mode)
{
	unsigned long ideal_clk = mode->htotal * mode->vtotal * mode->vrefresh;
	u32 clkdiv;

366 367 368 369 370 371 372 373
	if (ctx->i80_if) {
		/*
		 * The frame done interrupt should be occurred prior to the
		 * next TE signal.
		 */
		ideal_clk *= 2;
	}

374 375 376 377 378 379
	/* Find the clock divider value that gets us closest to ideal_clk */
	clkdiv = DIV_ROUND_UP(clk_get_rate(ctx->lcd_clk), ideal_clk);

	return (clkdiv < 0x100) ? clkdiv : 0xff;
}

380
static void fimd_commit(struct exynos_drm_crtc *crtc)
381
{
382
	struct fimd_context *ctx = crtc->ctx;
383
	struct drm_display_mode *mode = &crtc->base.state->adjusted_mode;
384 385 386
	struct fimd_driver_data *driver_data = ctx->driver_data;
	void *timing_base = ctx->regs + driver_data->timing_base;
	u32 val, clkdiv;
387

I
Inki Dae 已提交
388 389 390
	if (ctx->suspended)
		return;

391 392 393 394
	/* nothing to do if we haven't set the mode yet */
	if (mode->htotal == 0 || mode->vtotal == 0)
		return;

395 396 397 398 399 400 401 402
	if (ctx->i80_if) {
		val = ctx->i80ifcon | I80IFEN_ENABLE;
		writel(val, timing_base + I80IFCONFAx(0));

		/* disable auto frame rate */
		writel(0, timing_base + I80IFCONFBx(0));

		/* set video type selection to I80 interface */
J
Joonyoung Shim 已提交
403 404
		if (driver_data->has_vtsel && ctx->sysreg &&
				regmap_update_bits(ctx->sysreg,
405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 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 451 452 453 454
					driver_data->lcdblk_offset,
					0x3 << driver_data->lcdblk_vt_shift,
					0x1 << driver_data->lcdblk_vt_shift)) {
			DRM_ERROR("Failed to update sysreg for I80 i/f.\n");
			return;
		}
	} else {
		int vsync_len, vbpd, vfpd, hsync_len, hbpd, hfpd;
		u32 vidcon1;

		/* setup polarity values */
		vidcon1 = ctx->vidcon1;
		if (mode->flags & DRM_MODE_FLAG_NVSYNC)
			vidcon1 |= VIDCON1_INV_VSYNC;
		if (mode->flags & DRM_MODE_FLAG_NHSYNC)
			vidcon1 |= VIDCON1_INV_HSYNC;
		writel(vidcon1, ctx->regs + driver_data->timing_base + VIDCON1);

		/* setup vertical timing values. */
		vsync_len = mode->crtc_vsync_end - mode->crtc_vsync_start;
		vbpd = mode->crtc_vtotal - mode->crtc_vsync_end;
		vfpd = mode->crtc_vsync_start - mode->crtc_vdisplay;

		val = VIDTCON0_VBPD(vbpd - 1) |
			VIDTCON0_VFPD(vfpd - 1) |
			VIDTCON0_VSPW(vsync_len - 1);
		writel(val, ctx->regs + driver_data->timing_base + VIDTCON0);

		/* setup horizontal timing values.  */
		hsync_len = mode->crtc_hsync_end - mode->crtc_hsync_start;
		hbpd = mode->crtc_htotal - mode->crtc_hsync_end;
		hfpd = mode->crtc_hsync_start - mode->crtc_hdisplay;

		val = VIDTCON1_HBPD(hbpd - 1) |
			VIDTCON1_HFPD(hfpd - 1) |
			VIDTCON1_HSPW(hsync_len - 1);
		writel(val, ctx->regs + driver_data->timing_base + VIDTCON1);
	}

	if (driver_data->has_vidoutcon)
		writel(ctx->vidout_con, timing_base + VIDOUT_CON);

	/* set bypass selection */
	if (ctx->sysreg && regmap_update_bits(ctx->sysreg,
				driver_data->lcdblk_offset,
				0x1 << driver_data->lcdblk_bypass_shift,
				0x1 << driver_data->lcdblk_bypass_shift)) {
		DRM_ERROR("Failed to update sysreg for bypass setting.\n");
		return;
	}
455 456

	/* setup horizontal and vertical display size. */
457 458 459 460
	val = VIDTCON2_LINEVAL(mode->vdisplay - 1) |
	       VIDTCON2_HOZVAL(mode->hdisplay - 1) |
	       VIDTCON2_LINEVAL_E(mode->vdisplay - 1) |
	       VIDTCON2_HOZVAL_E(mode->hdisplay - 1);
461
	writel(val, ctx->regs + driver_data->timing_base + VIDTCON2);
462

463 464 465 466
	/*
	 * fields of register with prefix '_F' would be updated
	 * at vsync(same as dma start)
	 */
467 468
	val = ctx->vidcon0;
	val |= VIDCON0_ENVID | VIDCON0_ENVID_F;
469

470
	if (ctx->driver_data->has_clksel)
471 472
		val |= VIDCON0_CLKSEL_LCD;

473 474 475
	clkdiv = fimd_calc_clkdiv(ctx, mode);
	if (clkdiv > 1)
		val |= VIDCON0_CLKVAL_F(clkdiv - 1) | VIDCON0_CLKDIR;
476 477 478 479 480

	writel(val, ctx->regs + VIDCON0);
}


481 482
static void fimd_win_set_pixfmt(struct fimd_context *ctx, unsigned int win,
				struct drm_framebuffer *fb)
483 484 485 486 487
{
	unsigned long val;

	val = WINCONx_ENWIN;

488 489 490 491 492
	/*
	 * In case of s3c64xx, window 0 doesn't support alpha channel.
	 * So the request format is ARGB8888 then change it to XRGB8888.
	 */
	if (ctx->driver_data->has_limited_fmt && !win) {
493 494
		if (fb->pixel_format == DRM_FORMAT_ARGB8888)
			fb->pixel_format = DRM_FORMAT_XRGB8888;
495 496
	}

497
	switch (fb->pixel_format) {
498
	case DRM_FORMAT_C8:
499 500 501 502
		val |= WINCON0_BPPMODE_8BPP_PALETTE;
		val |= WINCONx_BURSTLEN_8WORD;
		val |= WINCONx_BYTSWP;
		break;
503 504 505 506 507 508
	case DRM_FORMAT_XRGB1555:
		val |= WINCON0_BPPMODE_16BPP_1555;
		val |= WINCONx_HAWSWP;
		val |= WINCONx_BURSTLEN_16WORD;
		break;
	case DRM_FORMAT_RGB565:
509 510 511 512
		val |= WINCON0_BPPMODE_16BPP_565;
		val |= WINCONx_HAWSWP;
		val |= WINCONx_BURSTLEN_16WORD;
		break;
513
	case DRM_FORMAT_XRGB8888:
514 515 516 517
		val |= WINCON0_BPPMODE_24BPP_888;
		val |= WINCONx_WSWP;
		val |= WINCONx_BURSTLEN_16WORD;
		break;
518 519
	case DRM_FORMAT_ARGB8888:
		val |= WINCON1_BPPMODE_25BPP_A1888
520 521 522 523 524 525 526 527 528 529 530 531 532
			| WINCON1_BLD_PIX | WINCON1_ALPHA_SEL;
		val |= WINCONx_WSWP;
		val |= WINCONx_BURSTLEN_16WORD;
		break;
	default:
		DRM_DEBUG_KMS("invalid pixel size so using unpacked 24bpp.\n");

		val |= WINCON0_BPPMODE_24BPP_888;
		val |= WINCONx_WSWP;
		val |= WINCONx_BURSTLEN_16WORD;
		break;
	}

533
	DRM_DEBUG_KMS("bpp = %d\n", fb->bits_per_pixel);
534

535 536 537
	/*
	 * In case of exynos, setting dma-burst to 16Word causes permanent
	 * tearing for very small buffers, e.g. cursor buffer. Burst Mode
538 539
	 * switching which is based on plane size is not recommended as
	 * plane size varies alot towards the end of the screen and rapid
540 541 542
	 * movement causes unstable DMA which results into iommu crash/tear.
	 */

543
	if (fb->width < MIN_FB_WIDTH_FOR_16WORD_BURST) {
544 545 546 547
		val &= ~WINCONx_BURSTLEN_MASK;
		val |= WINCONx_BURSTLEN_4WORD;
	}

548
	writel(val, ctx->regs + WINCON(win));
549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566

	/* hardware window 0 doesn't support alpha channel. */
	if (win != 0) {
		/* OSD alpha */
		val = VIDISD14C_ALPHA0_R(0xf) |
			VIDISD14C_ALPHA0_G(0xf) |
			VIDISD14C_ALPHA0_B(0xf) |
			VIDISD14C_ALPHA1_R(0xf) |
			VIDISD14C_ALPHA1_G(0xf) |
			VIDISD14C_ALPHA1_B(0xf);

		writel(val, ctx->regs + VIDOSD_C(win));

		val = VIDW_ALPHA_R(0xf) | VIDW_ALPHA_G(0xf) |
			VIDW_ALPHA_G(0xf);
		writel(val, ctx->regs + VIDWnALPHA0(win));
		writel(val, ctx->regs + VIDWnALPHA1(win));
	}
567 568
}

569
static void fimd_win_set_colkey(struct fimd_context *ctx, unsigned int win)
570 571 572 573 574 575 576 577 578 579 580 581
{
	unsigned int keycon0 = 0, keycon1 = 0;

	keycon0 = ~(WxKEYCON0_KEYBL_EN | WxKEYCON0_KEYEN_F |
			WxKEYCON0_DIRCON) | WxKEYCON0_COMPKEY(0);

	keycon1 = WxKEYCON1_COLVAL(0xffffffff);

	writel(keycon0, ctx->regs + WKEYCON0_BASE(win));
	writel(keycon1, ctx->regs + WKEYCON1_BASE(win));
}

582 583 584 585 586 587 588
/**
 * shadow_protect_win() - disable updating values from shadow registers at vsync
 *
 * @win: window to protect registers for
 * @protect: 1 to protect (disable updates)
 */
static void fimd_shadow_protect_win(struct fimd_context *ctx,
589
				    unsigned int win, bool protect)
590 591 592
{
	u32 reg, bits, val;

593 594 595 596 597 598 599 600 601 602
	/*
	 * SHADOWCON/PRTCON register is used for enabling timing.
	 *
	 * for example, once only width value of a register is set,
	 * if the dma is started then fimd hardware could malfunction so
	 * with protect window setting, the register fields with prefix '_F'
	 * wouldn't be updated at vsync also but updated once unprotect window
	 * is set.
	 */

603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618
	if (ctx->driver_data->has_shadowcon) {
		reg = SHADOWCON;
		bits = SHADOWCON_WINx_PROTECT(win);
	} else {
		reg = PRTCON;
		bits = PRTCON_PROTECT;
	}

	val = readl(ctx->regs + reg);
	if (protect)
		val |= bits;
	else
		val &= ~bits;
	writel(val, ctx->regs + reg);
}

619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640
static void fimd_atomic_begin(struct exynos_drm_crtc *crtc,
			       struct exynos_drm_plane *plane)
{
	struct fimd_context *ctx = crtc->ctx;

	if (ctx->suspended)
		return;

	fimd_shadow_protect_win(ctx, plane->zpos, true);
}

static void fimd_atomic_flush(struct exynos_drm_crtc *crtc,
			       struct exynos_drm_plane *plane)
{
	struct fimd_context *ctx = crtc->ctx;

	if (ctx->suspended)
		return;

	fimd_shadow_protect_win(ctx, plane->zpos, false);
}

641 642
static void fimd_update_plane(struct exynos_drm_crtc *crtc,
			      struct exynos_drm_plane *plane)
643
{
644 645
	struct exynos_drm_plane_state *state =
				to_exynos_plane_state(plane->base.state);
646
	struct fimd_context *ctx = crtc->ctx;
647
	struct drm_framebuffer *fb = state->base.fb;
648 649 650
	dma_addr_t dma_addr;
	unsigned long val, size, offset;
	unsigned int last_x, last_y, buf_offsize, line_size;
651
	unsigned int win = plane->zpos;
652 653
	unsigned int bpp = fb->bits_per_pixel >> 3;
	unsigned int pitch = fb->pitches[0];
654

I
Inki Dae 已提交
655 656 657
	if (ctx->suspended)
		return;

658 659
	offset = state->src.x * bpp;
	offset += state->src.y * pitch;
660

661
	/* buffer start address */
662
	dma_addr = exynos_drm_fb_dma_addr(fb, 0) + offset;
663
	val = (unsigned long)dma_addr;
664 665 666
	writel(val, ctx->regs + VIDWx_BUF_START(win, 0));

	/* buffer end address */
667
	size = pitch * state->crtc.h;
668
	val = (unsigned long)(dma_addr + size);
669 670 671
	writel(val, ctx->regs + VIDWx_BUF_END(win, 0));

	DRM_DEBUG_KMS("start addr = 0x%lx, end addr = 0x%lx, size = 0x%lx\n",
672
			(unsigned long)dma_addr, val, size);
673
	DRM_DEBUG_KMS("ovl_width = %d, ovl_height = %d\n",
674
			state->crtc.w, state->crtc.h);
675 676

	/* buffer size */
677 678
	buf_offsize = pitch - (state->crtc.w * bpp);
	line_size = state->crtc.w * bpp;
679 680 681 682
	val = VIDW_BUF_SIZE_OFFSET(buf_offsize) |
		VIDW_BUF_SIZE_PAGEWIDTH(line_size) |
		VIDW_BUF_SIZE_OFFSET_E(buf_offsize) |
		VIDW_BUF_SIZE_PAGEWIDTH_E(line_size);
683 684 685
	writel(val, ctx->regs + VIDWx_BUF_SIZE(win, 0));

	/* OSD position */
686 687 688 689
	val = VIDOSDxA_TOPLEFT_X(state->crtc.x) |
		VIDOSDxA_TOPLEFT_Y(state->crtc.y) |
		VIDOSDxA_TOPLEFT_X_E(state->crtc.x) |
		VIDOSDxA_TOPLEFT_Y_E(state->crtc.y);
690 691
	writel(val, ctx->regs + VIDOSD_A(win));

692
	last_x = state->crtc.x + state->crtc.w;
693 694
	if (last_x)
		last_x--;
695
	last_y = state->crtc.y + state->crtc.h;
696 697 698
	if (last_y)
		last_y--;

699 700 701
	val = VIDOSDxB_BOTRIGHT_X(last_x) | VIDOSDxB_BOTRIGHT_Y(last_y) |
		VIDOSDxB_BOTRIGHT_X_E(last_x) | VIDOSDxB_BOTRIGHT_Y_E(last_y);

702 703
	writel(val, ctx->regs + VIDOSD_B(win));

704
	DRM_DEBUG_KMS("osd pos: tx = %d, ty = %d, bx = %d, by = %d\n",
705
			state->crtc.x, state->crtc.y, last_x, last_y);
706 707 708 709 710

	/* OSD size */
	if (win != 3 && win != 4) {
		u32 offset = VIDOSD_D(win);
		if (win == 0)
711
			offset = VIDOSD_C(win);
712
		val = state->crtc.w * state->crtc.h;
713 714 715 716 717
		writel(val, ctx->regs + offset);

		DRM_DEBUG_KMS("osd size = 0x%x\n", (unsigned int)val);
	}

718
	fimd_win_set_pixfmt(ctx, win, fb);
719 720 721

	/* hardware window 0 doesn't support color key. */
	if (win != 0)
722
		fimd_win_set_colkey(ctx, win);
723

724
	fimd_enable_video_output(ctx, win, true);
725

726 727
	if (ctx->driver_data->has_shadowcon)
		fimd_enable_shadow_channel_path(ctx, win, true);
728

729 730
	if (ctx->i80_if)
		atomic_set(&ctx->win_updated, 1);
731 732
}

733 734
static void fimd_disable_plane(struct exynos_drm_crtc *crtc,
			       struct exynos_drm_plane *plane)
735
{
736
	struct fimd_context *ctx = crtc->ctx;
737
	unsigned int win = plane->zpos;
738

739
	if (ctx->suspended)
740 741
		return;

742
	fimd_enable_video_output(ctx, win, false);
743

744 745
	if (ctx->driver_data->has_shadowcon)
		fimd_enable_shadow_channel_path(ctx, win, false);
746 747
}

748
static void fimd_enable(struct exynos_drm_crtc *crtc)
749
{
750
	struct fimd_context *ctx = crtc->ctx;
751 752

	if (!ctx->suspended)
753
		return;
754 755 756

	ctx->suspended = false;

757 758
	pm_runtime_get_sync(ctx->dev);

759
	/* if vblank was enabled status, enable it again. */
760 761
	if (test_and_clear_bit(0, &ctx->irq_flags))
		fimd_enable_vblank(ctx->crtc);
762

763
	fimd_commit(ctx->crtc);
764 765
}

766
static void fimd_disable(struct exynos_drm_crtc *crtc)
767
{
768
	struct fimd_context *ctx = crtc->ctx;
769
	int i;
770

771
	if (ctx->suspended)
772
		return;
773 774 775 776 777 778

	/*
	 * We need to make sure that all windows are disabled before we
	 * suspend that connector. Otherwise we might try to scan from
	 * a destroyed buffer later.
	 */
779
	for (i = 0; i < WINDOWS_NR; i++)
780
		fimd_disable_plane(crtc, &ctx->planes[i]);
781

782 783 784 785
	fimd_enable_vblank(crtc);
	fimd_wait_for_vblank(crtc);
	fimd_disable_vblank(crtc);

786 787
	writel(0, ctx->regs + VIDCON0);

788
	pm_runtime_put_sync(ctx->dev);
789
	ctx->suspended = true;
790 791
}

792 793
static void fimd_trigger(struct device *dev)
{
794
	struct fimd_context *ctx = dev_get_drvdata(dev);
795 796 797 798
	struct fimd_driver_data *driver_data = ctx->driver_data;
	void *timing_base = ctx->regs + driver_data->timing_base;
	u32 reg;

799
	 /*
800 801 802
	  * Skips triggering if in triggering state, because multiple triggering
	  * requests can cause panel reset.
	  */
803 804 805
	if (atomic_read(&ctx->triggering))
		return;

806
	/* Enters triggering mode */
807 808 809 810 811
	atomic_set(&ctx->triggering, 1);

	reg = readl(timing_base + TRIGCON);
	reg |= (TRGMODE_I80_RGB_ENABLE_I80 | SWTRGCMD_I80_RGB_ENABLE);
	writel(reg, timing_base + TRIGCON);
812 813 814 815 816 817 818

	/*
	 * Exits triggering mode if vblank is not enabled yet, because when the
	 * VIDINTCON0 register is not set, it can not exit from triggering mode.
	 */
	if (!test_bit(0, &ctx->irq_flags))
		atomic_set(&ctx->triggering, 0);
819 820
}

821
static void fimd_te_handler(struct exynos_drm_crtc *crtc)
822
{
823
	struct fimd_context *ctx = crtc->ctx;
824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840

	/* Checks the crtc is detached already from encoder */
	if (ctx->pipe < 0 || !ctx->drm_dev)
		return;

	/*
	 * If there is a page flip request, triggers and handles the page flip
	 * event so that current fb can be updated into panel GRAM.
	 */
	if (atomic_add_unless(&ctx->win_updated, -1, 0))
		fimd_trigger(ctx->dev);

	/* Wakes up vsync event queue */
	if (atomic_read(&ctx->wait_vsync_event)) {
		atomic_set(&ctx->wait_vsync_event, 0);
		wake_up(&ctx->wait_vsync_queue);
	}
841

842
	if (test_bit(0, &ctx->irq_flags))
843
		drm_crtc_handle_vblank(&ctx->crtc->base);
844 845
}

846 847 848 849 850 851 852 853 854 855 856 857 858 859
static void fimd_dp_clock_enable(struct exynos_drm_crtc *crtc, bool enable)
{
	struct fimd_context *ctx = crtc->ctx;
	u32 val;

	/*
	 * Only Exynos 5250, 5260, 5410 and 542x requires enabling DP/MIE
	 * clock. On these SoCs the bootloader may enable it but any
	 * power domain off/on will reset it to disable state.
	 */
	if (ctx->driver_data != &exynos5_fimd_driver_data)
		return;

	val = enable ? DP_MIE_CLK_DP_ENABLE : DP_MIE_CLK_DISABLE;
860
	writel(val, ctx->regs + DP_MIE_CLKCON);
861 862
}

863
static const struct exynos_drm_crtc_ops fimd_crtc_ops = {
864 865
	.enable = fimd_enable,
	.disable = fimd_disable,
866 867 868 869
	.commit = fimd_commit,
	.enable_vblank = fimd_enable_vblank,
	.disable_vblank = fimd_disable_vblank,
	.wait_for_vblank = fimd_wait_for_vblank,
870
	.atomic_begin = fimd_atomic_begin,
871 872
	.update_plane = fimd_update_plane,
	.disable_plane = fimd_disable_plane,
873
	.atomic_flush = fimd_atomic_flush,
874
	.te_handler = fimd_te_handler,
875
	.clock_enable = fimd_dp_clock_enable,
876 877 878 879 880
};

static irqreturn_t fimd_irq_handler(int irq, void *dev_id)
{
	struct fimd_context *ctx = (struct fimd_context *)dev_id;
881
	u32 val, clear_bit, start, start_s;
882
	int win;
883 884 885

	val = readl(ctx->regs + VIDINTCON1);

886 887 888
	clear_bit = ctx->i80_if ? VIDINTCON1_INT_I80 : VIDINTCON1_INT_FRAME;
	if (val & clear_bit)
		writel(clear_bit, ctx->regs + VIDINTCON1);
889

890
	/* check the crtc is detached already from encoder */
891
	if (ctx->pipe < 0 || !ctx->drm_dev)
892
		goto out;
I
Inki Dae 已提交
893

894 895 896
	if (!ctx->i80_if)
		drm_crtc_handle_vblank(&ctx->crtc->base);

897 898 899 900 901 902
	for (win = 0 ; win < WINDOWS_NR ; win++) {
		struct exynos_drm_plane *plane = &ctx->planes[win];

		if (!plane->pending_fb)
			continue;

903 904 905 906
		start = readl(ctx->regs + VIDWx_BUF_START(win, 0));
		start_s = readl(ctx->regs + VIDWx_BUF_START_S(win, 0));
		if (start == start_s)
			exynos_drm_crtc_finish_update(ctx->crtc, plane);
907
	}
908

909
	if (ctx->i80_if) {
910
		/* Exits triggering mode */
911 912 913 914 915 916 917
		atomic_set(&ctx->triggering, 0);
	} else {
		/* set wait vsync event to zero and wake up queue. */
		if (atomic_read(&ctx->wait_vsync_event)) {
			atomic_set(&ctx->wait_vsync_event, 0);
			wake_up(&ctx->wait_vsync_queue);
		}
918
	}
919

920
out:
921 922 923
	return IRQ_HANDLED;
}

924
static int fimd_bind(struct device *dev, struct device *master, void *data)
925
{
926
	struct fimd_context *ctx = dev_get_drvdata(dev);
927
	struct drm_device *drm_dev = data;
928
	struct exynos_drm_private *priv = drm_dev->dev_private;
929 930
	struct exynos_drm_plane *exynos_plane;
	enum drm_plane_type type;
931 932
	unsigned int zpos;
	int ret;
933

934 935
	ctx->drm_dev = drm_dev;
	ctx->pipe = priv->pipe++;
936

937
	for (zpos = 0; zpos < WINDOWS_NR; zpos++) {
938
		type = exynos_plane_get_type(zpos, CURSOR_WIN);
939
		ret = exynos_plane_init(drm_dev, &ctx->planes[zpos],
940 941
					1 << ctx->pipe, type, fimd_formats,
					ARRAY_SIZE(fimd_formats), zpos);
942 943 944 945
		if (ret)
			return ret;
	}

946
	exynos_plane = &ctx->planes[DEFAULT_WIN];
947 948
	ctx->crtc = exynos_drm_crtc_create(drm_dev, &exynos_plane->base,
					   ctx->pipe, EXYNOS_DISPLAY_TYPE_LCD,
949
					   &fimd_crtc_ops, ctx);
950 951
	if (IS_ERR(ctx->crtc))
		return PTR_ERR(ctx->crtc);
952

953
	if (ctx->encoder)
954
		exynos_dpi_bind(drm_dev, ctx->encoder);
955

956 957
	if (is_drm_iommu_supported(drm_dev))
		fimd_clear_channels(ctx->crtc);
958 959

	ret = drm_iommu_attach_device(drm_dev, dev);
960 961 962 963
	if (ret)
		priv->pipe--;

	return ret;
964 965 966 967 968
}

static void fimd_unbind(struct device *dev, struct device *master,
			void *data)
{
969
	struct fimd_context *ctx = dev_get_drvdata(dev);
970

971
	fimd_disable(ctx->crtc);
972

973
	drm_iommu_detach_device(ctx->drm_dev, ctx->dev);
974

975 976
	if (ctx->encoder)
		exynos_dpi_remove(ctx->encoder);
977 978 979 980 981 982 983 984 985 986
}

static const struct component_ops fimd_component_ops = {
	.bind	= fimd_bind,
	.unbind = fimd_unbind,
};

static int fimd_probe(struct platform_device *pdev)
{
	struct device *dev = &pdev->dev;
987
	struct fimd_context *ctx;
988
	struct device_node *i80_if_timings;
989
	struct resource *res;
990
	int ret;
991

992 993
	if (!dev->of_node)
		return -ENODEV;
994

995
	ctx = devm_kzalloc(dev, sizeof(*ctx), GFP_KERNEL);
996 997 998
	if (!ctx)
		return -ENOMEM;

999
	ctx->dev = dev;
1000
	ctx->suspended = true;
1001
	ctx->driver_data = drm_fimd_get_driver_data(pdev);
1002

1003 1004 1005 1006
	if (of_property_read_bool(dev->of_node, "samsung,invert-vden"))
		ctx->vidcon1 |= VIDCON1_INV_VDEN;
	if (of_property_read_bool(dev->of_node, "samsung,invert-vclk"))
		ctx->vidcon1 |= VIDCON1_INV_VCLK;
1007

1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045
	i80_if_timings = of_get_child_by_name(dev->of_node, "i80-if-timings");
	if (i80_if_timings) {
		u32 val;

		ctx->i80_if = true;

		if (ctx->driver_data->has_vidoutcon)
			ctx->vidout_con |= VIDOUT_CON_F_I80_LDI0;
		else
			ctx->vidcon0 |= VIDCON0_VIDOUT_I80_LDI0;
		/*
		 * The user manual describes that this "DSI_EN" bit is required
		 * to enable I80 24-bit data interface.
		 */
		ctx->vidcon0 |= VIDCON0_DSI_EN;

		if (of_property_read_u32(i80_if_timings, "cs-setup", &val))
			val = 0;
		ctx->i80ifcon = LCD_CS_SETUP(val);
		if (of_property_read_u32(i80_if_timings, "wr-setup", &val))
			val = 0;
		ctx->i80ifcon |= LCD_WR_SETUP(val);
		if (of_property_read_u32(i80_if_timings, "wr-active", &val))
			val = 1;
		ctx->i80ifcon |= LCD_WR_ACTIVE(val);
		if (of_property_read_u32(i80_if_timings, "wr-hold", &val))
			val = 0;
		ctx->i80ifcon |= LCD_WR_HOLD(val);
	}
	of_node_put(i80_if_timings);

	ctx->sysreg = syscon_regmap_lookup_by_phandle(dev->of_node,
							"samsung,sysreg");
	if (IS_ERR(ctx->sysreg)) {
		dev_warn(dev, "failed to get system register.\n");
		ctx->sysreg = NULL;
	}

1046 1047 1048
	ctx->bus_clk = devm_clk_get(dev, "fimd");
	if (IS_ERR(ctx->bus_clk)) {
		dev_err(dev, "failed to get bus clock\n");
1049
		return PTR_ERR(ctx->bus_clk);
1050 1051 1052 1053 1054
	}

	ctx->lcd_clk = devm_clk_get(dev, "sclk_fimd");
	if (IS_ERR(ctx->lcd_clk)) {
		dev_err(dev, "failed to get lcd clock\n");
1055
		return PTR_ERR(ctx->lcd_clk);
1056
	}
1057 1058 1059

	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);

1060
	ctx->regs = devm_ioremap_resource(dev, res);
1061 1062
	if (IS_ERR(ctx->regs))
		return PTR_ERR(ctx->regs);
1063

1064 1065
	res = platform_get_resource_byname(pdev, IORESOURCE_IRQ,
					   ctx->i80_if ? "lcd_sys" : "vsync");
1066 1067
	if (!res) {
		dev_err(dev, "irq request failed.\n");
1068
		return -ENXIO;
1069 1070
	}

1071
	ret = devm_request_irq(dev, res->start, fimd_irq_handler,
1072 1073
							0, "drm_fimd", ctx);
	if (ret) {
1074
		dev_err(dev, "irq request failed.\n");
1075
		return ret;
1076 1077
	}

1078
	init_waitqueue_head(&ctx->wait_vsync_queue);
1079
	atomic_set(&ctx->wait_vsync_event, 0);
1080

1081
	platform_set_drvdata(pdev, ctx);
1082

1083 1084 1085
	ctx->encoder = exynos_dpi_probe(dev);
	if (IS_ERR(ctx->encoder))
		return PTR_ERR(ctx->encoder);
1086

1087
	pm_runtime_enable(dev);
1088

1089
	ret = component_add(dev, &fimd_component_ops);
1090 1091 1092 1093 1094 1095
	if (ret)
		goto err_disable_pm_runtime;

	return ret;

err_disable_pm_runtime:
1096
	pm_runtime_disable(dev);
1097 1098

	return ret;
1099
}
1100

1101 1102
static int fimd_remove(struct platform_device *pdev)
{
1103
	pm_runtime_disable(&pdev->dev);
1104

1105 1106
	component_del(&pdev->dev, &fimd_component_ops);

1107
	return 0;
I
Inki Dae 已提交
1108 1109
}

1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145
#ifdef CONFIG_PM
static int exynos_fimd_suspend(struct device *dev)
{
	struct fimd_context *ctx = dev_get_drvdata(dev);

	clk_disable_unprepare(ctx->lcd_clk);
	clk_disable_unprepare(ctx->bus_clk);

	return 0;
}

static int exynos_fimd_resume(struct device *dev)
{
	struct fimd_context *ctx = dev_get_drvdata(dev);
	int ret;

	ret = clk_prepare_enable(ctx->bus_clk);
	if (ret < 0) {
		DRM_ERROR("Failed to prepare_enable the bus clk [%d]\n", ret);
		return ret;
	}

	ret = clk_prepare_enable(ctx->lcd_clk);
	if  (ret < 0) {
		DRM_ERROR("Failed to prepare_enable the lcd clk [%d]\n", ret);
		return ret;
	}

	return 0;
}
#endif

static const struct dev_pm_ops exynos_fimd_pm_ops = {
	SET_RUNTIME_PM_OPS(exynos_fimd_suspend, exynos_fimd_resume, NULL)
};

1146
struct platform_driver fimd_driver = {
1147
	.probe		= fimd_probe,
1148
	.remove		= fimd_remove,
1149 1150 1151
	.driver		= {
		.name	= "exynos4-fb",
		.owner	= THIS_MODULE,
1152
		.pm	= &exynos_fimd_pm_ops,
1153
		.of_match_table = fimd_driver_dt_match,
1154 1155
	},
};