intel_sprite.c 37.0 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 26 27 28 29 30 31
/*
 * Copyright © 2011 Intel Corporation
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 *
 * Authors:
 *   Jesse Barnes <jbarnes@virtuousgeek.org>
 *
 * New plane/sprite handling.
 *
 * The older chips had a separate interface for programming plane related
 * registers; newer ones are much simpler and we can use the new DRM plane
 * support.
 */
32 33 34
#include <drm/drmP.h>
#include <drm/drm_crtc.h>
#include <drm/drm_fourcc.h>
35
#include <drm/drm_rect.h>
36
#include <drm/drm_plane_helper.h>
37
#include "intel_drv.h"
38
#include <drm/i915_drm.h>
39 40
#include "i915_drv.h"

41 42 43 44 45 46 47 48 49 50 51 52 53 54
static bool
format_is_yuv(uint32_t format)
{
	switch (format) {
	case DRM_FORMAT_YUYV:
	case DRM_FORMAT_UYVY:
	case DRM_FORMAT_VYUY:
	case DRM_FORMAT_YVYU:
		return true;
	default:
		return false;
	}
}

55 56 57 58 59 60 61 62 63
static int usecs_to_scanlines(const struct drm_display_mode *mode, int usecs)
{
	/* paranoia */
	if (!mode->crtc_htotal)
		return 1;

	return DIV_ROUND_UP(usecs * mode->crtc_clock, 1000 * mode->crtc_htotal);
}

64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
/**
 * intel_pipe_update_start() - start update of a set of display registers
 * @crtc: the crtc of which the registers are going to be updated
 * @start_vbl_count: vblank counter return pointer used for error checking
 *
 * Mark the start of an update to pipe registers that should be updated
 * atomically regarding vblank. If the next vblank will happens within
 * the next 100 us, this function waits until the vblank passes.
 *
 * After a successful call to this function, interrupts will be disabled
 * until a subsequent call to intel_pipe_update_end(). That is done to
 * avoid random delays. The value written to @start_vbl_count should be
 * supplied to intel_pipe_update_end() for error checking.
 *
 * Return: true if the call was successful
 */
80
bool intel_pipe_update_start(struct intel_crtc *crtc, uint32_t *start_vbl_count)
81 82
{
	struct drm_device *dev = crtc->base.dev;
83
	const struct drm_display_mode *mode = &crtc->config->base.adjusted_mode;
84 85 86
	enum pipe pipe = crtc->pipe;
	long timeout = msecs_to_jiffies_timeout(1);
	int scanline, min, max, vblank_start;
87
	wait_queue_head_t *wq = drm_crtc_vblank_waitqueue(&crtc->base);
88 89 90 91 92 93 94 95 96 97 98 99 100
	DEFINE_WAIT(wait);

	vblank_start = mode->crtc_vblank_start;
	if (mode->flags & DRM_MODE_FLAG_INTERLACE)
		vblank_start = DIV_ROUND_UP(vblank_start, 2);

	/* FIXME needs to be calibrated sensibly */
	min = vblank_start - usecs_to_scanlines(mode, 100);
	max = vblank_start - 1;

	if (min <= 0 || max <= 0)
		return false;

101
	if (WARN_ON(drm_crtc_vblank_get(&crtc->base)))
102 103 104 105
		return false;

	local_irq_disable();

106 107
	trace_i915_pipe_update_start(crtc, min, max);

108 109 110 111 112 113
	for (;;) {
		/*
		 * prepare_to_wait() has a memory barrier, which guarantees
		 * other CPUs can see the task state update by the time we
		 * read the scanline.
		 */
114
		prepare_to_wait(wq, &wait, TASK_UNINTERRUPTIBLE);
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132

		scanline = intel_get_crtc_scanline(crtc);
		if (scanline < min || scanline > max)
			break;

		if (timeout <= 0) {
			DRM_ERROR("Potential atomic update failure on pipe %c\n",
				  pipe_name(crtc->pipe));
			break;
		}

		local_irq_enable();

		timeout = schedule_timeout(timeout);

		local_irq_disable();
	}

133
	finish_wait(wq, &wait);
134

135
	drm_crtc_vblank_put(&crtc->base);
136 137 138

	*start_vbl_count = dev->driver->get_vblank_counter(dev, pipe);

139 140
	trace_i915_pipe_update_vblank_evaded(crtc, min, max, *start_vbl_count);

141 142 143
	return true;
}

144 145 146 147 148 149 150 151 152
/**
 * intel_pipe_update_end() - end update of a set of display registers
 * @crtc: the crtc of which the registers were updated
 * @start_vbl_count: start vblank counter (used for error checking)
 *
 * Mark the end of an update started with intel_pipe_update_start(). This
 * re-enables interrupts and verifies the update was actually completed
 * before a vblank using the value of @start_vbl_count.
 */
153
void intel_pipe_update_end(struct intel_crtc *crtc, u32 start_vbl_count)
154 155 156 157 158
{
	struct drm_device *dev = crtc->base.dev;
	enum pipe pipe = crtc->pipe;
	u32 end_vbl_count = dev->driver->get_vblank_counter(dev, pipe);

159 160
	trace_i915_pipe_update_end(crtc, end_vbl_count);

161 162 163 164 165 166 167
	local_irq_enable();

	if (start_vbl_count != end_vbl_count)
		DRM_ERROR("Atomic update failure on pipe %c (start=%u end=%u)\n",
			  pipe_name(pipe), start_vbl_count, end_vbl_count);
}

168 169 170 171 172 173 174 175 176 177 178
static void intel_update_primary_plane(struct intel_crtc *crtc)
{
	struct drm_i915_private *dev_priv = crtc->base.dev->dev_private;
	int reg = DSPCNTR(crtc->plane);

	if (crtc->primary_enabled)
		I915_WRITE(reg, I915_READ(reg) | DISPLAY_PLANE_ENABLE);
	else
		I915_WRITE(reg, I915_READ(reg) & ~DISPLAY_PLANE_ENABLE);
}

179 180 181
static void
skl_update_plane(struct drm_plane *drm_plane, struct drm_crtc *crtc,
		 struct drm_framebuffer *fb,
182
		 int crtc_x, int crtc_y,
183 184 185 186 187 188 189
		 unsigned int crtc_w, unsigned int crtc_h,
		 uint32_t x, uint32_t y,
		 uint32_t src_w, uint32_t src_h)
{
	struct drm_device *dev = drm_plane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(drm_plane);
190
	struct drm_i915_gem_object *obj = intel_fb_obj(fb);
191 192
	const int pipe = intel_plane->pipe;
	const int plane = intel_plane->plane + 1;
193
	u32 plane_ctl, stride_div, stride;
194
	int pixel_size = drm_format_plane_cpp(fb->pixel_format, 0);
195
	const struct drm_intel_sprite_colorkey *key = &intel_plane->ckey;
196
	unsigned long surf_addr;
197 198 199
	u32 tile_height, plane_offset, plane_size;
	unsigned int rotation;
	int x_offset, y_offset;
200

201 202
	plane_ctl = PLANE_CTL_ENABLE |
		PLANE_CTL_PIPE_CSC_ENABLE;
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

	switch (fb->pixel_format) {
	case DRM_FORMAT_RGB565:
		plane_ctl |= PLANE_CTL_FORMAT_RGB_565;
		break;
	case DRM_FORMAT_XBGR8888:
		plane_ctl |= PLANE_CTL_FORMAT_XRGB_8888 | PLANE_CTL_ORDER_RGBX;
		break;
	case DRM_FORMAT_XRGB8888:
		plane_ctl |= PLANE_CTL_FORMAT_XRGB_8888;
		break;
	/*
	 * XXX: For ARBG/ABGR formats we default to expecting scanout buffers
	 * to be already pre-multiplied. We need to add a knob (or a different
	 * DRM_FORMAT) for user-space to configure that.
	 */
	case DRM_FORMAT_ABGR8888:
		plane_ctl |= PLANE_CTL_FORMAT_XRGB_8888 |
			     PLANE_CTL_ORDER_RGBX |
			     PLANE_CTL_ALPHA_SW_PREMULTIPLY;
		break;
	case DRM_FORMAT_ARGB8888:
		plane_ctl |= PLANE_CTL_FORMAT_XRGB_8888 |
			     PLANE_CTL_ALPHA_SW_PREMULTIPLY;
		break;
	case DRM_FORMAT_YUYV:
		plane_ctl |= PLANE_CTL_FORMAT_YUV422 | PLANE_CTL_YUV422_YUYV;
		break;
	case DRM_FORMAT_YVYU:
		plane_ctl |= PLANE_CTL_FORMAT_YUV422 | PLANE_CTL_YUV422_YVYU;
		break;
	case DRM_FORMAT_UYVY:
		plane_ctl |= PLANE_CTL_FORMAT_YUV422 | PLANE_CTL_YUV422_UYVY;
		break;
	case DRM_FORMAT_VYUY:
		plane_ctl |= PLANE_CTL_FORMAT_YUV422 | PLANE_CTL_YUV422_VYUY;
		break;
	default:
		BUG();
	}

244 245
	switch (fb->modifier[0]) {
	case DRM_FORMAT_MOD_NONE:
246
		break;
247
	case I915_FORMAT_MOD_X_TILED:
248
		plane_ctl |= PLANE_CTL_TILED_X;
249 250 251 252 253 254
		break;
	case I915_FORMAT_MOD_Y_TILED:
		plane_ctl |= PLANE_CTL_TILED_Y;
		break;
	case I915_FORMAT_MOD_Yf_TILED:
		plane_ctl |= PLANE_CTL_TILED_YF;
255 256
		break;
	default:
257
		MISSING_CASE(fb->modifier[0]);
258
	}
259

260 261 262 263 264 265 266
	rotation = drm_plane->state->rotation;
	switch (rotation) {
	case BIT(DRM_ROTATE_90):
		plane_ctl |= PLANE_CTL_ROTATE_90;
		break;

	case BIT(DRM_ROTATE_180):
267
		plane_ctl |= PLANE_CTL_ROTATE_180;
268 269 270 271 272 273
		break;

	case BIT(DRM_ROTATE_270):
		plane_ctl |= PLANE_CTL_ROTATE_270;
		break;
	}
274 275 276 277 278

	intel_update_sprite_watermarks(drm_plane, crtc, src_w, src_h,
				       pixel_size, true,
				       src_w != crtc_w || src_h != crtc_h);

279 280 281
	stride_div = intel_fb_stride_alignment(dev, fb->modifier[0],
					       fb->pixel_format);

282 283 284 285 286 287
	/* Sizes are 0 based */
	src_w--;
	src_h--;
	crtc_w--;
	crtc_h--;

288 289 290 291 292 293 294 295 296 297 298
	if (key->flags) {
		I915_WRITE(PLANE_KEYVAL(pipe, plane), key->min_value);
		I915_WRITE(PLANE_KEYMAX(pipe, plane), key->max_value);
		I915_WRITE(PLANE_KEYMSK(pipe, plane), key->channel_mask);
	}

	if (key->flags & I915_SET_COLORKEY_DESTINATION)
		plane_ctl |= PLANE_CTL_KEY_ENABLE_DESTINATION;
	else if (key->flags & I915_SET_COLORKEY_SOURCE)
		plane_ctl |= PLANE_CTL_KEY_ENABLE_SOURCE;

299 300
	surf_addr = intel_plane_obj_offset(intel_plane, obj);

301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318
	if (intel_rotation_90_or_270(rotation)) {
		/* stride: Surface height in tiles */
		tile_height = intel_tile_height(dev, fb->bits_per_pixel,
							fb->modifier[0]);
		stride = DIV_ROUND_UP(fb->height, tile_height);
		plane_size = (src_w << 16) | src_h;
		x_offset = stride * tile_height - y - (src_h + 1);
		y_offset = x;
	} else {
		stride = fb->pitches[0] / stride_div;
		plane_size = (src_h << 16) | src_w;
		x_offset = x;
		y_offset = y;
	}
	plane_offset = y_offset << 16 | x_offset;

	I915_WRITE(PLANE_OFFSET(pipe, plane), plane_offset);
	I915_WRITE(PLANE_STRIDE(pipe, plane), stride);
319
	I915_WRITE(PLANE_POS(pipe, plane), (crtc_y << 16) | crtc_x);
320
	I915_WRITE(PLANE_SIZE(pipe, plane), plane_size);
321
	I915_WRITE(PLANE_CTL(pipe, plane), plane_ctl);
322
	I915_WRITE(PLANE_SURF(pipe, plane), surf_addr);
323 324 325 326 327 328 329 330 331 332 333 334
	POSTING_READ(PLANE_SURF(pipe, plane));
}

static void
skl_disable_plane(struct drm_plane *drm_plane, struct drm_crtc *crtc)
{
	struct drm_device *dev = drm_plane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(drm_plane);
	const int pipe = intel_plane->pipe;
	const int plane = intel_plane->plane + 1;

335
	I915_WRITE(PLANE_CTL(pipe, plane), 0);
336 337

	/* Activate double buffered register update */
338 339
	I915_WRITE(PLANE_SURF(pipe, plane), 0);
	POSTING_READ(PLANE_SURF(pipe, plane));
340 341 342 343

	intel_update_sprite_watermarks(drm_plane, crtc, 0, 0, 0, false, false);
}

344 345 346 347 348 349 350 351 352 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
static void
chv_update_csc(struct intel_plane *intel_plane, uint32_t format)
{
	struct drm_i915_private *dev_priv = intel_plane->base.dev->dev_private;
	int plane = intel_plane->plane;

	/* Seems RGB data bypasses the CSC always */
	if (!format_is_yuv(format))
		return;

	/*
	 * BT.601 limited range YCbCr -> full range RGB
	 *
	 * |r|   | 6537 4769     0|   |cr  |
	 * |g| = |-3330 4769 -1605| x |y-64|
	 * |b|   |    0 4769  8263|   |cb  |
	 *
	 * Cb and Cr apparently come in as signed already, so no
	 * need for any offset. For Y we need to remove the offset.
	 */
	I915_WRITE(SPCSCYGOFF(plane), SPCSC_OOFF(0) | SPCSC_IOFF(-64));
	I915_WRITE(SPCSCCBOFF(plane), SPCSC_OOFF(0) | SPCSC_IOFF(0));
	I915_WRITE(SPCSCCROFF(plane), SPCSC_OOFF(0) | SPCSC_IOFF(0));

	I915_WRITE(SPCSCC01(plane), SPCSC_C1(4769) | SPCSC_C0(6537));
	I915_WRITE(SPCSCC23(plane), SPCSC_C1(-3330) | SPCSC_C0(0));
	I915_WRITE(SPCSCC45(plane), SPCSC_C1(-1605) | SPCSC_C0(4769));
	I915_WRITE(SPCSCC67(plane), SPCSC_C1(4769) | SPCSC_C0(0));
	I915_WRITE(SPCSCC8(plane), SPCSC_C0(8263));

	I915_WRITE(SPCSCYGICLAMP(plane), SPCSC_IMAX(940) | SPCSC_IMIN(64));
	I915_WRITE(SPCSCCBICLAMP(plane), SPCSC_IMAX(448) | SPCSC_IMIN(-448));
	I915_WRITE(SPCSCCRICLAMP(plane), SPCSC_IMAX(448) | SPCSC_IMIN(-448));

	I915_WRITE(SPCSCYGOCLAMP(plane), SPCSC_OMAX(1023) | SPCSC_OMIN(0));
	I915_WRITE(SPCSCCBOCLAMP(plane), SPCSC_OMAX(1023) | SPCSC_OMIN(0));
	I915_WRITE(SPCSCCROCLAMP(plane), SPCSC_OMAX(1023) | SPCSC_OMIN(0));
}

383
static void
384 385
vlv_update_plane(struct drm_plane *dplane, struct drm_crtc *crtc,
		 struct drm_framebuffer *fb,
386
		 int crtc_x, int crtc_y,
387 388 389 390 391 392 393
		 unsigned int crtc_w, unsigned int crtc_h,
		 uint32_t x, uint32_t y,
		 uint32_t src_w, uint32_t src_h)
{
	struct drm_device *dev = dplane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(dplane);
394
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
395
	struct drm_i915_gem_object *obj = intel_fb_obj(fb);
396 397 398 399 400
	int pipe = intel_plane->pipe;
	int plane = intel_plane->plane;
	u32 sprctl;
	unsigned long sprsurf_offset, linear_offset;
	int pixel_size = drm_format_plane_cpp(fb->pixel_format, 0);
401
	const struct drm_intel_sprite_colorkey *key = &intel_plane->ckey;
402

403
	sprctl = SP_ENABLE;
404 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

	switch (fb->pixel_format) {
	case DRM_FORMAT_YUYV:
		sprctl |= SP_FORMAT_YUV422 | SP_YUV_ORDER_YUYV;
		break;
	case DRM_FORMAT_YVYU:
		sprctl |= SP_FORMAT_YUV422 | SP_YUV_ORDER_YVYU;
		break;
	case DRM_FORMAT_UYVY:
		sprctl |= SP_FORMAT_YUV422 | SP_YUV_ORDER_UYVY;
		break;
	case DRM_FORMAT_VYUY:
		sprctl |= SP_FORMAT_YUV422 | SP_YUV_ORDER_VYUY;
		break;
	case DRM_FORMAT_RGB565:
		sprctl |= SP_FORMAT_BGR565;
		break;
	case DRM_FORMAT_XRGB8888:
		sprctl |= SP_FORMAT_BGRX8888;
		break;
	case DRM_FORMAT_ARGB8888:
		sprctl |= SP_FORMAT_BGRA8888;
		break;
	case DRM_FORMAT_XBGR2101010:
		sprctl |= SP_FORMAT_RGBX1010102;
		break;
	case DRM_FORMAT_ABGR2101010:
		sprctl |= SP_FORMAT_RGBA1010102;
		break;
	case DRM_FORMAT_XBGR8888:
		sprctl |= SP_FORMAT_RGBX8888;
		break;
	case DRM_FORMAT_ABGR8888:
		sprctl |= SP_FORMAT_RGBA8888;
		break;
	default:
		/*
		 * If we get here one of the upper layers failed to filter
		 * out the unsupported plane formats
		 */
		BUG();
		break;
	}

448 449 450 451 452 453
	/*
	 * Enable gamma to match primary/cursor plane behaviour.
	 * FIXME should be user controllable via propertiesa.
	 */
	sprctl |= SP_GAMMA_ENABLE;

454 455 456
	if (obj->tiling_mode != I915_TILING_NONE)
		sprctl |= SP_TILED;

457 458
	intel_update_sprite_watermarks(dplane, crtc, src_w, src_h,
				       pixel_size, true,
459 460
				       src_w != crtc_w || src_h != crtc_h);

461 462 463 464 465 466 467 468 469 470 471 472 473
	/* Sizes are 0 based */
	src_w--;
	src_h--;
	crtc_w--;
	crtc_h--;

	linear_offset = y * fb->pitches[0] + x * pixel_size;
	sprsurf_offset = intel_gen4_compute_page_offset(&x, &y,
							obj->tiling_mode,
							pixel_size,
							fb->pitches[0]);
	linear_offset -= sprsurf_offset;

474
	if (dplane->state->rotation == BIT(DRM_ROTATE_180)) {
475 476 477 478 479 480 481
		sprctl |= SP_ROTATE_180;

		x += src_w;
		y += src_h;
		linear_offset += src_h * fb->pitches[0] + src_w * pixel_size;
	}

482 483
	intel_update_primary_plane(intel_crtc);

484 485 486 487 488 489 490 491 492
	if (key->flags) {
		I915_WRITE(SPKEYMINVAL(pipe, plane), key->min_value);
		I915_WRITE(SPKEYMAXVAL(pipe, plane), key->max_value);
		I915_WRITE(SPKEYMSK(pipe, plane), key->channel_mask);
	}

	if (key->flags & I915_SET_COLORKEY_SOURCE)
		sprctl |= SP_SOURCE_KEY;

493 494 495
	if (IS_CHERRYVIEW(dev) && pipe == PIPE_B)
		chv_update_csc(intel_plane, fb->pixel_format);

496 497 498
	I915_WRITE(SPSTRIDE(pipe, plane), fb->pitches[0]);
	I915_WRITE(SPPOS(pipe, plane), (crtc_y << 16) | crtc_x);

499 500 501 502 503
	if (obj->tiling_mode != I915_TILING_NONE)
		I915_WRITE(SPTILEOFF(pipe, plane), (y << 16) | x);
	else
		I915_WRITE(SPLINOFF(pipe, plane), linear_offset);

504 505
	I915_WRITE(SPCONSTALPHA(pipe, plane), 0);

506 507
	I915_WRITE(SPSIZE(pipe, plane), (crtc_h << 16) | crtc_w);
	I915_WRITE(SPCNTR(pipe, plane), sprctl);
508 509
	I915_WRITE(SPSURF(pipe, plane), i915_gem_obj_ggtt_offset(obj) +
		   sprsurf_offset);
510 511

	intel_flush_primary_plane(dev_priv, intel_crtc->plane);
512 513 514
}

static void
515
vlv_disable_plane(struct drm_plane *dplane, struct drm_crtc *crtc)
516 517 518 519
{
	struct drm_device *dev = dplane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(dplane);
520
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
521 522 523
	int pipe = intel_plane->pipe;
	int plane = intel_plane->plane;

524 525
	intel_update_primary_plane(intel_crtc);

526 527
	I915_WRITE(SPCNTR(pipe, plane), 0);

528
	/* Activate double buffered register update */
529
	I915_WRITE(SPSURF(pipe, plane), 0);
530 531

	intel_flush_primary_plane(dev_priv, intel_crtc->plane);
532

533
	intel_update_sprite_watermarks(dplane, crtc, 0, 0, 0, false, false);
534 535 536
}


537
static void
538 539
ivb_update_plane(struct drm_plane *plane, struct drm_crtc *crtc,
		 struct drm_framebuffer *fb,
540
		 int crtc_x, int crtc_y,
541 542 543 544 545 546 547
		 unsigned int crtc_w, unsigned int crtc_h,
		 uint32_t x, uint32_t y,
		 uint32_t src_w, uint32_t src_h)
{
	struct drm_device *dev = plane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(plane);
548
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
549
	struct drm_i915_gem_object *obj = intel_fb_obj(fb);
550
	enum pipe pipe = intel_plane->pipe;
551
	u32 sprctl, sprscale = 0;
552
	unsigned long sprsurf_offset, linear_offset;
V
Ville Syrjälä 已提交
553
	int pixel_size = drm_format_plane_cpp(fb->pixel_format, 0);
554
	const struct drm_intel_sprite_colorkey *key = &intel_plane->ckey;
555

556
	sprctl = SPRITE_ENABLE;
557 558 559

	switch (fb->pixel_format) {
	case DRM_FORMAT_XBGR8888:
560
		sprctl |= SPRITE_FORMAT_RGBX888 | SPRITE_RGB_ORDER_RGBX;
561 562
		break;
	case DRM_FORMAT_XRGB8888:
563
		sprctl |= SPRITE_FORMAT_RGBX888;
564 565 566 567 568 569 570 571 572 573 574 575 576 577
		break;
	case DRM_FORMAT_YUYV:
		sprctl |= SPRITE_FORMAT_YUV422 | SPRITE_YUV_ORDER_YUYV;
		break;
	case DRM_FORMAT_YVYU:
		sprctl |= SPRITE_FORMAT_YUV422 | SPRITE_YUV_ORDER_YVYU;
		break;
	case DRM_FORMAT_UYVY:
		sprctl |= SPRITE_FORMAT_YUV422 | SPRITE_YUV_ORDER_UYVY;
		break;
	case DRM_FORMAT_VYUY:
		sprctl |= SPRITE_FORMAT_YUV422 | SPRITE_YUV_ORDER_VYUY;
		break;
	default:
578
		BUG();
579 580
	}

581 582 583 584 585 586
	/*
	 * Enable gamma to match primary/cursor plane behaviour.
	 * FIXME should be user controllable via propertiesa.
	 */
	sprctl |= SPRITE_GAMMA_ENABLE;

587 588 589
	if (obj->tiling_mode != I915_TILING_NONE)
		sprctl |= SPRITE_TILED;

590
	if (IS_HASWELL(dev) || IS_BROADWELL(dev))
591 592 593 594
		sprctl &= ~SPRITE_TRICKLE_FEED_DISABLE;
	else
		sprctl |= SPRITE_TRICKLE_FEED_DISABLE;

595
	if (IS_HASWELL(dev) || IS_BROADWELL(dev))
596 597
		sprctl |= SPRITE_PIPE_CSC_ENABLE;

598 599
	intel_update_sprite_watermarks(plane, crtc, src_w, src_h, pixel_size,
				       true,
600 601
				       src_w != crtc_w || src_h != crtc_h);

602 603 604 605 606 607
	/* Sizes are 0 based */
	src_w--;
	src_h--;
	crtc_w--;
	crtc_h--;

608
	if (crtc_w != src_w || crtc_h != src_h)
609 610
		sprscale = SPRITE_SCALE_ENABLE | (src_w << 16) | src_h;

611
	linear_offset = y * fb->pitches[0] + x * pixel_size;
612
	sprsurf_offset =
613 614
		intel_gen4_compute_page_offset(&x, &y, obj->tiling_mode,
					       pixel_size, fb->pitches[0]);
615 616
	linear_offset -= sprsurf_offset;

617
	if (plane->state->rotation == BIT(DRM_ROTATE_180)) {
618 619 620 621 622 623 624 625 626 627 628
		sprctl |= SPRITE_ROTATE_180;

		/* HSW and BDW does this automagically in hardware */
		if (!IS_HASWELL(dev) && !IS_BROADWELL(dev)) {
			x += src_w;
			y += src_h;
			linear_offset += src_h * fb->pitches[0] +
				src_w * pixel_size;
		}
	}

629 630
	intel_update_primary_plane(intel_crtc);

631 632 633 634 635 636 637 638 639 640 641
	if (key->flags) {
		I915_WRITE(SPRKEYVAL(pipe), key->min_value);
		I915_WRITE(SPRKEYMAX(pipe), key->max_value);
		I915_WRITE(SPRKEYMSK(pipe), key->channel_mask);
	}

	if (key->flags & I915_SET_COLORKEY_DESTINATION)
		sprctl |= SPRITE_DEST_KEY;
	else if (key->flags & I915_SET_COLORKEY_SOURCE)
		sprctl |= SPRITE_SOURCE_KEY;

642 643 644
	I915_WRITE(SPRSTRIDE(pipe), fb->pitches[0]);
	I915_WRITE(SPRPOS(pipe), (crtc_y << 16) | crtc_x);

645 646
	/* HSW consolidates SPRTILEOFF and SPRLINOFF into a single SPROFFSET
	 * register */
647
	if (IS_HASWELL(dev) || IS_BROADWELL(dev))
648
		I915_WRITE(SPROFFSET(pipe), (y << 16) | x);
649
	else if (obj->tiling_mode != I915_TILING_NONE)
650
		I915_WRITE(SPRTILEOFF(pipe), (y << 16) | x);
651 652
	else
		I915_WRITE(SPRLINOFF(pipe), linear_offset);
653

654
	I915_WRITE(SPRSIZE(pipe), (crtc_h << 16) | crtc_w);
655 656
	if (intel_plane->can_scale)
		I915_WRITE(SPRSCALE(pipe), sprscale);
657
	I915_WRITE(SPRCTL(pipe), sprctl);
658 659
	I915_WRITE(SPRSURF(pipe),
		   i915_gem_obj_ggtt_offset(obj) + sprsurf_offset);
660 661

	intel_flush_primary_plane(dev_priv, intel_crtc->plane);
662 663 664
}

static void
665
ivb_disable_plane(struct drm_plane *plane, struct drm_crtc *crtc)
666 667 668 669
{
	struct drm_device *dev = plane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(plane);
670
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
671 672
	int pipe = intel_plane->pipe;

673 674
	intel_update_primary_plane(intel_crtc);

675 676
	I915_WRITE(SPRCTL(pipe), I915_READ(SPRCTL(pipe)) & ~SPRITE_ENABLE);
	/* Can't leave the scaler enabled... */
677 678
	if (intel_plane->can_scale)
		I915_WRITE(SPRSCALE(pipe), 0);
679
	/* Activate double buffered register update */
680
	I915_WRITE(SPRSURF(pipe), 0);
681 682

	intel_flush_primary_plane(dev_priv, intel_crtc->plane);
683 684 685
}

static void
686 687
ilk_update_plane(struct drm_plane *plane, struct drm_crtc *crtc,
		 struct drm_framebuffer *fb,
688
		 int crtc_x, int crtc_y,
689 690 691 692 693 694 695
		 unsigned int crtc_w, unsigned int crtc_h,
		 uint32_t x, uint32_t y,
		 uint32_t src_w, uint32_t src_h)
{
	struct drm_device *dev = plane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(plane);
696
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
697
	struct drm_i915_gem_object *obj = intel_fb_obj(fb);
V
Ville Syrjälä 已提交
698
	int pipe = intel_plane->pipe;
699
	unsigned long dvssurf_offset, linear_offset;
700
	u32 dvscntr, dvsscale;
V
Ville Syrjälä 已提交
701
	int pixel_size = drm_format_plane_cpp(fb->pixel_format, 0);
702
	const struct drm_intel_sprite_colorkey *key = &intel_plane->ckey;
703

704
	dvscntr = DVS_ENABLE;
705 706 707

	switch (fb->pixel_format) {
	case DRM_FORMAT_XBGR8888:
708
		dvscntr |= DVS_FORMAT_RGBX888 | DVS_RGB_ORDER_XBGR;
709 710
		break;
	case DRM_FORMAT_XRGB8888:
711
		dvscntr |= DVS_FORMAT_RGBX888;
712 713 714 715 716 717 718 719 720 721 722 723 724 725
		break;
	case DRM_FORMAT_YUYV:
		dvscntr |= DVS_FORMAT_YUV422 | DVS_YUV_ORDER_YUYV;
		break;
	case DRM_FORMAT_YVYU:
		dvscntr |= DVS_FORMAT_YUV422 | DVS_YUV_ORDER_YVYU;
		break;
	case DRM_FORMAT_UYVY:
		dvscntr |= DVS_FORMAT_YUV422 | DVS_YUV_ORDER_UYVY;
		break;
	case DRM_FORMAT_VYUY:
		dvscntr |= DVS_FORMAT_YUV422 | DVS_YUV_ORDER_VYUY;
		break;
	default:
726
		BUG();
727 728
	}

729 730 731 732 733 734
	/*
	 * Enable gamma to match primary/cursor plane behaviour.
	 * FIXME should be user controllable via propertiesa.
	 */
	dvscntr |= DVS_GAMMA_ENABLE;

735 736 737
	if (obj->tiling_mode != I915_TILING_NONE)
		dvscntr |= DVS_TILED;

738 739
	if (IS_GEN6(dev))
		dvscntr |= DVS_TRICKLE_FEED_DISABLE; /* must disable */
740

741 742
	intel_update_sprite_watermarks(plane, crtc, src_w, src_h,
				       pixel_size, true,
743 744
				       src_w != crtc_w || src_h != crtc_h);

745 746 747 748 749 750
	/* Sizes are 0 based */
	src_w--;
	src_h--;
	crtc_w--;
	crtc_h--;

751
	dvsscale = 0;
752
	if (crtc_w != src_w || crtc_h != src_h)
753 754
		dvsscale = DVS_SCALE_ENABLE | (src_w << 16) | src_h;

755
	linear_offset = y * fb->pitches[0] + x * pixel_size;
756
	dvssurf_offset =
757 758
		intel_gen4_compute_page_offset(&x, &y, obj->tiling_mode,
					       pixel_size, fb->pitches[0]);
759 760
	linear_offset -= dvssurf_offset;

761
	if (plane->state->rotation == BIT(DRM_ROTATE_180)) {
762 763 764 765 766 767 768
		dvscntr |= DVS_ROTATE_180;

		x += src_w;
		y += src_h;
		linear_offset += src_h * fb->pitches[0] + src_w * pixel_size;
	}

769 770
	intel_update_primary_plane(intel_crtc);

771 772 773 774 775 776 777 778 779 780 781
	if (key->flags) {
		I915_WRITE(DVSKEYVAL(pipe), key->min_value);
		I915_WRITE(DVSKEYMAX(pipe), key->max_value);
		I915_WRITE(DVSKEYMSK(pipe), key->channel_mask);
	}

	if (key->flags & I915_SET_COLORKEY_DESTINATION)
		dvscntr |= DVS_DEST_KEY;
	else if (key->flags & I915_SET_COLORKEY_SOURCE)
		dvscntr |= DVS_SOURCE_KEY;

782 783 784
	I915_WRITE(DVSSTRIDE(pipe), fb->pitches[0]);
	I915_WRITE(DVSPOS(pipe), (crtc_y << 16) | crtc_x);

785
	if (obj->tiling_mode != I915_TILING_NONE)
786
		I915_WRITE(DVSTILEOFF(pipe), (y << 16) | x);
787 788
	else
		I915_WRITE(DVSLINOFF(pipe), linear_offset);
789 790 791 792

	I915_WRITE(DVSSIZE(pipe), (crtc_h << 16) | crtc_w);
	I915_WRITE(DVSSCALE(pipe), dvsscale);
	I915_WRITE(DVSCNTR(pipe), dvscntr);
793 794
	I915_WRITE(DVSSURF(pipe),
		   i915_gem_obj_ggtt_offset(obj) + dvssurf_offset);
795 796

	intel_flush_primary_plane(dev_priv, intel_crtc->plane);
797 798 799
}

static void
800
ilk_disable_plane(struct drm_plane *plane, struct drm_crtc *crtc)
801 802 803 804
{
	struct drm_device *dev = plane->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_plane *intel_plane = to_intel_plane(plane);
805
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
806 807
	int pipe = intel_plane->pipe;

808 809
	intel_update_primary_plane(intel_crtc);

810
	I915_WRITE(DVSCNTR(pipe), 0);
811 812
	/* Disable the scaler */
	I915_WRITE(DVSSCALE(pipe), 0);
813

814
	/* Flush double buffered register updates */
815
	I915_WRITE(DVSSURF(pipe), 0);
816 817

	intel_flush_primary_plane(dev_priv, intel_crtc->plane);
818 819
}

820 821 822 823 824 825 826 827 828 829 830
/**
 * intel_post_enable_primary - Perform operations after enabling primary plane
 * @crtc: the CRTC whose primary plane was just enabled
 *
 * Performs potentially sleeping operations that must be done after the primary
 * plane is enabled, such as updating FBC and IPS.  Note that this may be
 * called due to an explicit primary plane update, or due to an implicit
 * re-enable that is caused when a sprite plane is updated to no longer
 * completely hide the primary plane.
 */
void
831
intel_post_enable_primary(struct drm_crtc *crtc)
832 833 834
{
	struct drm_device *dev = crtc->dev;
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
835

836 837 838 839 840 841 842 843
	/*
	 * BDW signals flip done immediately if the plane
	 * is disabled, even if the plane enable is already
	 * armed to occur at the next vblank :(
	 */
	if (IS_BROADWELL(dev))
		intel_wait_for_vblank(dev, intel_crtc->pipe);

844 845 846 847 848 849
	/*
	 * FIXME IPS should be fine as long as one plane is
	 * enabled, but in practice it seems to have problems
	 * when going from primary only to sprite only and vice
	 * versa.
	 */
850
	hsw_enable_ips(intel_crtc);
851

852
	mutex_lock(&dev->struct_mutex);
853
	intel_fbc_update(dev);
854
	mutex_unlock(&dev->struct_mutex);
855 856
}

857 858 859 860 861 862 863 864 865 866 867
/**
 * intel_pre_disable_primary - Perform operations before disabling primary plane
 * @crtc: the CRTC whose primary plane is to be disabled
 *
 * Performs potentially sleeping operations that must be done before the
 * primary plane is enabled, such as updating FBC and IPS.  Note that this may
 * be called due to an explicit primary plane update, or due to an implicit
 * disable that is caused when a sprite plane completely hides the primary
 * plane.
 */
void
868
intel_pre_disable_primary(struct drm_crtc *crtc)
869 870 871 872
{
	struct drm_device *dev = crtc->dev;
	struct drm_i915_private *dev_priv = dev->dev_private;
	struct intel_crtc *intel_crtc = to_intel_crtc(crtc);
873 874

	mutex_lock(&dev->struct_mutex);
875
	if (dev_priv->fbc.crtc == intel_crtc)
876
		intel_fbc_disable(dev);
877
	mutex_unlock(&dev->struct_mutex);
878

879 880 881 882 883 884 885
	/*
	 * FIXME IPS should be fine as long as one plane is
	 * enabled, but in practice it seems to have problems
	 * when going from primary only to sprite only and vice
	 * versa.
	 */
	hsw_disable_ips(intel_crtc);
886 887
}

888 889
static bool colorkey_enabled(struct intel_plane *intel_plane)
{
890
	return intel_plane->ckey.flags != I915_SET_COLORKEY_NONE;
891 892
}

893
static int
894 895
intel_check_sprite_plane(struct drm_plane *plane,
			 struct intel_plane_state *state)
896
{
897
	struct intel_crtc *intel_crtc = to_intel_crtc(state->base.crtc);
898
	struct intel_plane *intel_plane = to_intel_plane(plane);
899
	struct drm_framebuffer *fb = state->base.fb;
900 901 902 903 904 905
	int crtc_x, crtc_y;
	unsigned int crtc_w, crtc_h;
	uint32_t src_x, src_y, src_w, src_h;
	struct drm_rect *src = &state->src;
	struct drm_rect *dst = &state->dst;
	const struct drm_rect *clip = &state->clip;
906 907
	int hscale, vscale;
	int max_scale, min_scale;
908 909
	int pixel_size;

910 911
	intel_crtc = intel_crtc ? intel_crtc : to_intel_crtc(plane->crtc);

912 913
	if (!fb) {
		state->visible = false;
914
		goto finish;
915
	}
916

917 918 919
	/* Don't modify another pipe's plane */
	if (intel_plane->pipe != intel_crtc->pipe) {
		DRM_DEBUG_KMS("Wrong plane <-> crtc mapping\n");
920
		return -EINVAL;
921
	}
922

923 924 925
	/* FIXME check all gen limits */
	if (fb->width < 3 || fb->height < 3 || fb->pitches[0] > 16384) {
		DRM_DEBUG_KMS("Unsuitable framebuffer for plane\n");
926
		return -EINVAL;
927
	}
928

929 930 931 932 933
	/*
	 * FIXME the following code does a bunch of fuzzy adjustments to the
	 * coordinates and sizes. We probably need some way to decide whether
	 * more strict checking should be done instead.
	 */
934 935 936
	max_scale = intel_plane->max_downscale << 16;
	min_scale = intel_plane->can_scale ? 1 : (1 << 16);

937
	drm_rect_rotate(src, fb->width << 16, fb->height << 16,
938
			state->base.rotation);
939

940
	hscale = drm_rect_calc_hscale_relaxed(src, dst, min_scale, max_scale);
941
	BUG_ON(hscale < 0);
942

943
	vscale = drm_rect_calc_vscale_relaxed(src, dst, min_scale, max_scale);
944
	BUG_ON(vscale < 0);
945

946
	state->visible =  drm_rect_clip_scaled(src, dst, clip, hscale, vscale);
947

948 949 950 951
	crtc_x = dst->x1;
	crtc_y = dst->y1;
	crtc_w = drm_rect_width(dst);
	crtc_h = drm_rect_height(dst);
952

953
	if (state->visible) {
954
		/* check again in case clipping clamped the results */
955
		hscale = drm_rect_calc_hscale(src, dst, min_scale, max_scale);
956 957
		if (hscale < 0) {
			DRM_DEBUG_KMS("Horizontal scaling factor out of limits\n");
958 959
			drm_rect_debug_print(src, true);
			drm_rect_debug_print(dst, false);
960 961 962 963

			return hscale;
		}

964
		vscale = drm_rect_calc_vscale(src, dst, min_scale, max_scale);
965 966
		if (vscale < 0) {
			DRM_DEBUG_KMS("Vertical scaling factor out of limits\n");
967 968
			drm_rect_debug_print(src, true);
			drm_rect_debug_print(dst, false);
969 970 971 972

			return vscale;
		}

973
		/* Make the source viewport size an exact multiple of the scaling factors. */
974 975 976
		drm_rect_adjust_size(src,
				     drm_rect_width(dst) * hscale - drm_rect_width(src),
				     drm_rect_height(dst) * vscale - drm_rect_height(src));
977

978
		drm_rect_rotate_inv(src, fb->width << 16, fb->height << 16,
979
				    state->base.rotation);
980

981
		/* sanity check to make sure the src viewport wasn't enlarged */
982 983 984 985
		WARN_ON(src->x1 < (int) state->base.src_x ||
			src->y1 < (int) state->base.src_y ||
			src->x2 > (int) state->base.src_x + state->base.src_w ||
			src->y2 > (int) state->base.src_y + state->base.src_h);
986 987 988 989 990 991 992

		/*
		 * Hardware doesn't handle subpixel coordinates.
		 * Adjust to (macro)pixel boundary, but be careful not to
		 * increase the source viewport size, because that could
		 * push the downscaling factor out of bounds.
		 */
993 994 995 996
		src_x = src->x1 >> 16;
		src_w = drm_rect_width(src) >> 16;
		src_y = src->y1 >> 16;
		src_h = drm_rect_height(src) >> 16;
997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009

		if (format_is_yuv(fb->pixel_format)) {
			src_x &= ~1;
			src_w &= ~1;

			/*
			 * Must keep src and dst the
			 * same if we can't scale.
			 */
			if (!intel_plane->can_scale)
				crtc_w &= ~1;

			if (crtc_w == 0)
1010
				state->visible = false;
1011 1012 1013 1014
		}
	}

	/* Check size restrictions when scaling */
1015
	if (state->visible && (src_w != crtc_w || src_h != crtc_h)) {
1016 1017 1018 1019 1020 1021 1022
		unsigned int width_bytes;

		WARN_ON(!intel_plane->can_scale);

		/* FIXME interlacing min height is 6 */

		if (crtc_w < 3 || crtc_h < 3)
1023
			state->visible = false;
1024 1025

		if (src_w < 3 || src_h < 3)
1026
			state->visible = false;
1027

1028
		pixel_size = drm_format_plane_cpp(fb->pixel_format, 0);
1029 1030
		width_bytes = ((src_x * pixel_size) & 63) +
					src_w * pixel_size;
1031 1032 1033 1034 1035 1036 1037 1038

		if (src_w > 2048 || src_h > 2048 ||
		    width_bytes > 4096 || fb->pitches[0] > 4096) {
			DRM_DEBUG_KMS("Source dimensions exceed hardware limits\n");
			return -EINVAL;
		}
	}

1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050
	if (state->visible) {
		src->x1 = src_x;
		src->x2 = src_x + src_w;
		src->y1 = src_y;
		src->y2 = src_y + src_h;
	}

	dst->x1 = crtc_x;
	dst->x2 = crtc_x + crtc_w;
	dst->y1 = crtc_y;
	dst->y2 = crtc_y + crtc_h;

1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071
finish:
	/*
	 * If the sprite is completely covering the primary plane,
	 * we can disable the primary and save power.
	 */
	state->hides_primary = fb != NULL && drm_rect_equals(dst, clip) &&
		!colorkey_enabled(intel_plane);
	WARN_ON(state->hides_primary && !state->visible && intel_crtc->active);

	if (intel_crtc->active) {
		if (intel_crtc->primary_enabled == state->hides_primary)
			intel_crtc->atomic.wait_for_flips = true;

		if (intel_crtc->primary_enabled && state->hides_primary)
			intel_crtc->atomic.pre_disable_primary = true;

		intel_crtc->atomic.fb_bits |=
			INTEL_FRONTBUFFER_SPRITE(intel_crtc->pipe);

		if (!intel_crtc->primary_enabled && !state->hides_primary)
			intel_crtc->atomic.post_enable_primary = true;
1072

1073
		if (intel_wm_need_update(plane, &state->base))
1074
			intel_crtc->atomic.update_wm = true;
1075 1076 1077 1078 1079 1080 1081 1082 1083 1084

		if (!state->visible) {
			/*
			 * Avoid underruns when disabling the sprite.
			 * FIXME remove once watermark updates are done properly.
			 */
			intel_crtc->atomic.wait_vblank = true;
			intel_crtc->atomic.update_sprite_watermarks |=
				(1 << drm_plane_index(plane));
		}
1085 1086
	}

1087 1088 1089
	return 0;
}

1090 1091 1092 1093
static void
intel_commit_sprite_plane(struct drm_plane *plane,
			  struct intel_plane_state *state)
{
1094
	struct drm_crtc *crtc = state->base.crtc;
1095
	struct intel_crtc *intel_crtc;
1096
	struct intel_plane *intel_plane = to_intel_plane(plane);
1097
	struct drm_framebuffer *fb = state->base.fb;
1098 1099 1100 1101
	int crtc_x, crtc_y;
	unsigned int crtc_w, crtc_h;
	uint32_t src_x, src_y, src_w, src_h;

1102 1103 1104
	crtc = crtc ? crtc : plane->crtc;
	intel_crtc = to_intel_crtc(crtc);

1105
	plane->fb = fb;
1106

1107
	if (intel_crtc->active) {
1108
		intel_crtc->primary_enabled = !state->hides_primary;
1109

1110 1111
		if (state->visible) {
			crtc_x = state->dst.x1;
1112
			crtc_y = state->dst.y1;
1113 1114 1115 1116 1117 1118
			crtc_w = drm_rect_width(&state->dst);
			crtc_h = drm_rect_height(&state->dst);
			src_x = state->src.x1;
			src_y = state->src.y1;
			src_w = drm_rect_width(&state->src);
			src_h = drm_rect_height(&state->src);
1119
			intel_plane->update_plane(plane, crtc, fb,
1120 1121
						  crtc_x, crtc_y, crtc_w, crtc_h,
						  src_x, src_y, src_w, src_h);
1122
		} else {
1123
			intel_plane->disable_plane(plane, crtc);
1124
		}
1125
	}
1126 1127
}

1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139
int intel_sprite_set_colorkey(struct drm_device *dev, void *data,
			      struct drm_file *file_priv)
{
	struct drm_intel_sprite_colorkey *set = data;
	struct drm_plane *plane;
	struct intel_plane *intel_plane;
	int ret = 0;

	/* Make sure we don't try to enable both src & dest simultaneously */
	if ((set->flags & (I915_SET_COLORKEY_DESTINATION | I915_SET_COLORKEY_SOURCE)) == (I915_SET_COLORKEY_DESTINATION | I915_SET_COLORKEY_SOURCE))
		return -EINVAL;

1140 1141 1142 1143
	if (IS_VALLEYVIEW(dev) &&
	    set->flags & I915_SET_COLORKEY_DESTINATION)
		return -EINVAL;

1144
	drm_modeset_lock_all(dev);
1145

R
Rob Clark 已提交
1146 1147
	plane = drm_plane_find(dev, set->plane_id);
	if (!plane) {
1148
		ret = -ENOENT;
1149 1150 1151 1152
		goto out_unlock;
	}

	intel_plane = to_intel_plane(plane);
1153 1154 1155 1156 1157 1158 1159 1160 1161
	intel_plane->ckey = *set;

	/*
	 * The only way this could fail would be due to
	 * the current plane state being unsupportable already,
	 * and we dont't consider that an error for the
	 * colorkey ioctl. So just ignore any error.
	 */
	intel_plane_restore(plane);
1162 1163

out_unlock:
1164
	drm_modeset_unlock_all(dev);
1165 1166 1167
	return ret;
}

1168
int intel_plane_restore(struct drm_plane *plane)
1169
{
1170
	if (!plane->crtc || !plane->state->fb)
1171
		return 0;
1172

1173 1174 1175 1176 1177
	return drm_plane_helper_update(plane, plane->crtc, plane->state->fb,
				       plane->state->crtc_x, plane->state->crtc_y,
				       plane->state->crtc_w, plane->state->crtc_h,
				       plane->state->src_x, plane->state->src_y,
				       plane->state->src_w, plane->state->src_h);
1178 1179
}

1180 1181 1182 1183 1184 1185 1186 1187
static uint32_t ilk_plane_formats[] = {
	DRM_FORMAT_XRGB8888,
	DRM_FORMAT_YUYV,
	DRM_FORMAT_YVYU,
	DRM_FORMAT_UYVY,
	DRM_FORMAT_VYUY,
};

1188 1189 1190 1191 1192 1193 1194 1195 1196
static uint32_t snb_plane_formats[] = {
	DRM_FORMAT_XBGR8888,
	DRM_FORMAT_XRGB8888,
	DRM_FORMAT_YUYV,
	DRM_FORMAT_YVYU,
	DRM_FORMAT_UYVY,
	DRM_FORMAT_VYUY,
};

1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210
static uint32_t vlv_plane_formats[] = {
	DRM_FORMAT_RGB565,
	DRM_FORMAT_ABGR8888,
	DRM_FORMAT_ARGB8888,
	DRM_FORMAT_XBGR8888,
	DRM_FORMAT_XRGB8888,
	DRM_FORMAT_XBGR2101010,
	DRM_FORMAT_ABGR2101010,
	DRM_FORMAT_YUYV,
	DRM_FORMAT_YVYU,
	DRM_FORMAT_UYVY,
	DRM_FORMAT_VYUY,
};

1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222
static uint32_t skl_plane_formats[] = {
	DRM_FORMAT_RGB565,
	DRM_FORMAT_ABGR8888,
	DRM_FORMAT_ARGB8888,
	DRM_FORMAT_XBGR8888,
	DRM_FORMAT_XRGB8888,
	DRM_FORMAT_YUYV,
	DRM_FORMAT_YVYU,
	DRM_FORMAT_UYVY,
	DRM_FORMAT_VYUY,
};

1223
int
1224
intel_plane_init(struct drm_device *dev, enum pipe pipe, int plane)
1225 1226
{
	struct intel_plane *intel_plane;
1227
	struct intel_plane_state *state;
1228
	unsigned long possible_crtcs;
1229 1230
	const uint32_t *plane_formats;
	int num_plane_formats;
1231 1232
	int ret;

1233
	if (INTEL_INFO(dev)->gen < 5)
1234 1235
		return -ENODEV;

1236
	intel_plane = kzalloc(sizeof(*intel_plane), GFP_KERNEL);
1237 1238 1239
	if (!intel_plane)
		return -ENOMEM;

1240 1241
	state = intel_create_plane_state(&intel_plane->base);
	if (!state) {
1242 1243 1244
		kfree(intel_plane);
		return -ENOMEM;
	}
1245
	intel_plane->base.state = &state->base;
1246

1247 1248 1249
	switch (INTEL_INFO(dev)->gen) {
	case 5:
	case 6:
1250
		intel_plane->can_scale = true;
1251
		intel_plane->max_downscale = 16;
1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264
		intel_plane->update_plane = ilk_update_plane;
		intel_plane->disable_plane = ilk_disable_plane;

		if (IS_GEN6(dev)) {
			plane_formats = snb_plane_formats;
			num_plane_formats = ARRAY_SIZE(snb_plane_formats);
		} else {
			plane_formats = ilk_plane_formats;
			num_plane_formats = ARRAY_SIZE(ilk_plane_formats);
		}
		break;

	case 7:
B
Ben Widawsky 已提交
1265
	case 8:
1266
		if (IS_IVYBRIDGE(dev)) {
1267
			intel_plane->can_scale = true;
1268 1269 1270 1271 1272
			intel_plane->max_downscale = 2;
		} else {
			intel_plane->can_scale = false;
			intel_plane->max_downscale = 1;
		}
1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286

		if (IS_VALLEYVIEW(dev)) {
			intel_plane->update_plane = vlv_update_plane;
			intel_plane->disable_plane = vlv_disable_plane;

			plane_formats = vlv_plane_formats;
			num_plane_formats = ARRAY_SIZE(vlv_plane_formats);
		} else {
			intel_plane->update_plane = ivb_update_plane;
			intel_plane->disable_plane = ivb_disable_plane;

			plane_formats = snb_plane_formats;
			num_plane_formats = ARRAY_SIZE(snb_plane_formats);
		}
1287
		break;
1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300
	case 9:
		/*
		 * FIXME: Skylake planes can be scaled (with some restrictions),
		 * but this is for another time.
		 */
		intel_plane->can_scale = false;
		intel_plane->max_downscale = 1;
		intel_plane->update_plane = skl_update_plane;
		intel_plane->disable_plane = skl_disable_plane;

		plane_formats = skl_plane_formats;
		num_plane_formats = ARRAY_SIZE(skl_plane_formats);
		break;
1301
	default:
1302
		kfree(intel_plane);
1303
		return -ENODEV;
1304 1305 1306
	}

	intel_plane->pipe = pipe;
1307
	intel_plane->plane = plane;
1308 1309
	intel_plane->check_plane = intel_check_sprite_plane;
	intel_plane->commit_plane = intel_commit_sprite_plane;
1310
	possible_crtcs = (1 << pipe);
1311
	ret = drm_universal_plane_init(dev, &intel_plane->base, possible_crtcs,
1312
				       &intel_plane_funcs,
1313 1314
				       plane_formats, num_plane_formats,
				       DRM_PLANE_TYPE_OVERLAY);
1315
	if (ret) {
1316
		kfree(intel_plane);
1317 1318 1319
		goto out;
	}

1320
	intel_create_rotation_property(dev, intel_plane);
1321

1322 1323
	drm_plane_helper_add(&intel_plane->base, &intel_plane_helper_funcs);

1324
 out:
1325 1326
	return ret;
}