i915_dma.c 21.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2
/* i915_dma.c -- DMA support for the I915 -*- linux-c -*-
 */
D
Dave Airlie 已提交
3
/*
L
Linus Torvalds 已提交
4 5
 * Copyright 2003 Tungsten Graphics, Inc., Cedar Park, Texas.
 * All Rights Reserved.
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
 *
 * 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, sub license, 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 NON-INFRINGEMENT.
 * IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS 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.
 *
D
Dave Airlie 已提交
27
 */
L
Linus Torvalds 已提交
28 29 30 31 32 33 34 35 36 37 38

#include "drmP.h"
#include "drm.h"
#include "i915_drm.h"
#include "i915_drv.h"

/* Really want an OS-independent resettable timer.  Would like to have
 * this loop run for (eg) 3 sec, but have the timer reset every time
 * the head pointer changes, so that EBUSY only happens if the ring
 * actually stalls for (eg) 3 seconds.
 */
39
int i915_wait_ring(struct drm_device * dev, int n, const char *caller)
L
Linus Torvalds 已提交
40 41 42
{
	drm_i915_private_t *dev_priv = dev->dev_private;
	drm_i915_ring_buffer_t *ring = &(dev_priv->ring);
43 44 45
	u32 acthd_reg = IS_I965G(dev) ? ACTHD_I965 : ACTHD;
	u32 last_acthd = I915_READ(acthd_reg);
	u32 acthd;
46
	u32 last_head = I915_READ(PRB0_HEAD) & HEAD_ADDR;
L
Linus Torvalds 已提交
47 48
	int i;

49
	for (i = 0; i < 100000; i++) {
50
		ring->head = I915_READ(PRB0_HEAD) & HEAD_ADDR;
51
		acthd = I915_READ(acthd_reg);
L
Linus Torvalds 已提交
52 53 54 55 56 57 58 59 60 61
		ring->space = ring->head - (ring->tail + 8);
		if (ring->space < 0)
			ring->space += ring->Size;
		if (ring->space >= n)
			return 0;

		dev_priv->sarea_priv->perf_boxes |= I915_BOX_WAIT;

		if (ring->head != last_head)
			i = 0;
62 63
		if (acthd != last_acthd)
			i = 0;
L
Linus Torvalds 已提交
64 65

		last_head = ring->head;
66 67 68
		last_acthd = acthd;
		msleep_interruptible(10);

L
Linus Torvalds 已提交
69 70
	}

E
Eric Anholt 已提交
71
	return -EBUSY;
L
Linus Torvalds 已提交
72 73
}

74
void i915_kernel_lost_context(struct drm_device * dev)
L
Linus Torvalds 已提交
75 76 77 78
{
	drm_i915_private_t *dev_priv = dev->dev_private;
	drm_i915_ring_buffer_t *ring = &(dev_priv->ring);

79 80
	ring->head = I915_READ(PRB0_HEAD) & HEAD_ADDR;
	ring->tail = I915_READ(PRB0_TAIL) & TAIL_ADDR;
L
Linus Torvalds 已提交
81 82 83 84 85 86 87 88
	ring->space = ring->head - (ring->tail + 8);
	if (ring->space < 0)
		ring->space += ring->Size;

	if (ring->head == ring->tail)
		dev_priv->sarea_priv->perf_boxes |= I915_BOX_RING_EMPTY;
}

89
static int i915_dma_cleanup(struct drm_device * dev)
L
Linus Torvalds 已提交
90
{
J
Jesse Barnes 已提交
91
	drm_i915_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
92 93 94 95
	/* Make sure interrupts are disabled here because the uninstall ioctl
	 * may not have been called from userspace and after dev_private
	 * is freed, it's too late.
	 */
96
	if (dev->irq_enabled)
D
Dave Airlie 已提交
97
		drm_irq_uninstall(dev);
L
Linus Torvalds 已提交
98

J
Jesse Barnes 已提交
99 100 101 102 103 104
	if (dev_priv->ring.virtual_start) {
		drm_core_ioremapfree(&dev_priv->ring.map, dev);
		dev_priv->ring.virtual_start = 0;
		dev_priv->ring.map.handle = 0;
		dev_priv->ring.map.size = 0;
	}
105

J
Jesse Barnes 已提交
106 107 108 109
	if (dev_priv->status_page_dmah) {
		drm_pci_free(dev, dev_priv->status_page_dmah);
		dev_priv->status_page_dmah = NULL;
		/* Need to rewrite hardware status page */
110
		I915_WRITE(HWS_PGA, 0x1ffff000);
J
Jesse Barnes 已提交
111
	}
L
Linus Torvalds 已提交
112

J
Jesse Barnes 已提交
113 114 115
	if (dev_priv->status_gfx_addr) {
		dev_priv->status_gfx_addr = 0;
		drm_core_ioremapfree(&dev_priv->hws_map, dev);
116
		I915_WRITE(HWS_PGA, 0x1ffff000);
L
Linus Torvalds 已提交
117 118 119 120 121
	}

	return 0;
}

J
Jesse Barnes 已提交
122
static int i915_initialize(struct drm_device * dev, drm_i915_init_t * init)
L
Linus Torvalds 已提交
123
{
J
Jesse Barnes 已提交
124
	drm_i915_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
125

126
	dev_priv->sarea = drm_getsarea(dev);
L
Linus Torvalds 已提交
127 128 129
	if (!dev_priv->sarea) {
		DRM_ERROR("can not find sarea!\n");
		i915_dma_cleanup(dev);
E
Eric Anholt 已提交
130
		return -EINVAL;
L
Linus Torvalds 已提交
131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146
	}

	dev_priv->sarea_priv = (drm_i915_sarea_t *)
	    ((u8 *) dev_priv->sarea->handle + init->sarea_priv_offset);

	dev_priv->ring.Start = init->ring_start;
	dev_priv->ring.End = init->ring_end;
	dev_priv->ring.Size = init->ring_size;
	dev_priv->ring.tail_mask = dev_priv->ring.Size - 1;

	dev_priv->ring.map.offset = init->ring_start;
	dev_priv->ring.map.size = init->ring_size;
	dev_priv->ring.map.type = 0;
	dev_priv->ring.map.flags = 0;
	dev_priv->ring.map.mtrr = 0;

D
Dave Airlie 已提交
147
	drm_core_ioremap(&dev_priv->ring.map, dev);
L
Linus Torvalds 已提交
148 149 150 151 152

	if (dev_priv->ring.map.handle == NULL) {
		i915_dma_cleanup(dev);
		DRM_ERROR("can not ioremap virtual address for"
			  " ring buffer\n");
E
Eric Anholt 已提交
153
		return -ENOMEM;
L
Linus Torvalds 已提交
154 155 156 157
	}

	dev_priv->ring.virtual_start = dev_priv->ring.map.handle;

158
	dev_priv->cpp = init->cpp;
L
Linus Torvalds 已提交
159 160 161 162 163 164 165 166 167 168
	dev_priv->back_offset = init->back_offset;
	dev_priv->front_offset = init->front_offset;
	dev_priv->current_page = 0;
	dev_priv->sarea_priv->pf_current_page = dev_priv->current_page;

	/* Allow hardware batchbuffers unless told otherwise.
	 */
	dev_priv->allow_batchbuffer = 1;

	/* Program Hardware Status Page */
169
	if (!I915_NEED_GFX_HWS(dev)) {
170 171 172 173 174 175
		dev_priv->status_page_dmah =
			drm_pci_alloc(dev, PAGE_SIZE, PAGE_SIZE, 0xffffffff);

		if (!dev_priv->status_page_dmah) {
			i915_dma_cleanup(dev);
			DRM_ERROR("Can not allocate hardware status page\n");
E
Eric Anholt 已提交
176
			return -ENOMEM;
177 178 179
		}
		dev_priv->hw_status_page = dev_priv->status_page_dmah->vaddr;
		dev_priv->dma_status_page = dev_priv->status_page_dmah->busaddr;
L
Linus Torvalds 已提交
180

181
		memset(dev_priv->hw_status_page, 0, PAGE_SIZE);
182
		I915_WRITE(HWS_PGA, dev_priv->dma_status_page);
L
Linus Torvalds 已提交
183 184 185 186 187
	}
	DRM_DEBUG("Enabled hardware status page\n");
	return 0;
}

188
static int i915_dma_resume(struct drm_device * dev)
L
Linus Torvalds 已提交
189 190 191
{
	drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;

192
	DRM_DEBUG("%s\n", __func__);
L
Linus Torvalds 已提交
193 194 195

	if (!dev_priv->sarea) {
		DRM_ERROR("can not find sarea!\n");
E
Eric Anholt 已提交
196
		return -EINVAL;
L
Linus Torvalds 已提交
197 198 199 200 201
	}

	if (dev_priv->ring.map.handle == NULL) {
		DRM_ERROR("can not ioremap virtual address for"
			  " ring buffer\n");
E
Eric Anholt 已提交
202
		return -ENOMEM;
L
Linus Torvalds 已提交
203 204 205 206 207
	}

	/* Program Hardware Status Page */
	if (!dev_priv->hw_status_page) {
		DRM_ERROR("Can not find hardware status page\n");
E
Eric Anholt 已提交
208
		return -EINVAL;
L
Linus Torvalds 已提交
209 210 211
	}
	DRM_DEBUG("hw status page @ %p\n", dev_priv->hw_status_page);

212
	if (dev_priv->status_gfx_addr != 0)
213
		I915_WRITE(HWS_PGA, dev_priv->status_gfx_addr);
214
	else
215
		I915_WRITE(HWS_PGA, dev_priv->dma_status_page);
L
Linus Torvalds 已提交
216 217 218 219 220
	DRM_DEBUG("Enabled hardware status page\n");

	return 0;
}

221 222
static int i915_dma_init(struct drm_device *dev, void *data,
			 struct drm_file *file_priv)
L
Linus Torvalds 已提交
223
{
224
	drm_i915_init_t *init = data;
L
Linus Torvalds 已提交
225 226
	int retcode = 0;

227
	switch (init->func) {
L
Linus Torvalds 已提交
228
	case I915_INIT_DMA:
J
Jesse Barnes 已提交
229
		retcode = i915_initialize(dev, init);
L
Linus Torvalds 已提交
230 231 232 233 234
		break;
	case I915_CLEANUP_DMA:
		retcode = i915_dma_cleanup(dev);
		break;
	case I915_RESUME_DMA:
D
Dave Airlie 已提交
235
		retcode = i915_dma_resume(dev);
L
Linus Torvalds 已提交
236 237
		break;
	default:
E
Eric Anholt 已提交
238
		retcode = -EINVAL;
L
Linus Torvalds 已提交
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 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
		break;
	}

	return retcode;
}

/* Implement basically the same security restrictions as hardware does
 * for MI_BATCH_NON_SECURE.  These can be made stricter at any time.
 *
 * Most of the calculations below involve calculating the size of a
 * particular instruction.  It's important to get the size right as
 * that tells us where the next instruction to check is.  Any illegal
 * instruction detected will be given a size of zero, which is a
 * signal to abort the rest of the buffer.
 */
static int do_validate_cmd(int cmd)
{
	switch (((cmd >> 29) & 0x7)) {
	case 0x0:
		switch ((cmd >> 23) & 0x3f) {
		case 0x0:
			return 1;	/* MI_NOOP */
		case 0x4:
			return 1;	/* MI_FLUSH */
		default:
			return 0;	/* disallow everything else */
		}
		break;
	case 0x1:
		return 0;	/* reserved */
	case 0x2:
		return (cmd & 0xff) + 2;	/* 2d commands */
	case 0x3:
		if (((cmd >> 24) & 0x1f) <= 0x18)
			return 1;

		switch ((cmd >> 24) & 0x1f) {
		case 0x1c:
			return 1;
		case 0x1d:
D
Dave Airlie 已提交
279
			switch ((cmd >> 16) & 0xff) {
L
Linus Torvalds 已提交
280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315
			case 0x3:
				return (cmd & 0x1f) + 2;
			case 0x4:
				return (cmd & 0xf) + 2;
			default:
				return (cmd & 0xffff) + 2;
			}
		case 0x1e:
			if (cmd & (1 << 23))
				return (cmd & 0xffff) + 1;
			else
				return 1;
		case 0x1f:
			if ((cmd & (1 << 23)) == 0)	/* inline vertices */
				return (cmd & 0x1ffff) + 2;
			else if (cmd & (1 << 17))	/* indirect random */
				if ((cmd & 0xffff) == 0)
					return 0;	/* unknown length, too hard */
				else
					return (((cmd & 0xffff) + 1) / 2) + 1;
			else
				return 2;	/* indirect sequential */
		default:
			return 0;
		}
	default:
		return 0;
	}

	return 0;
}

static int validate_cmd(int cmd)
{
	int ret = do_validate_cmd(cmd);

D
Dave Airlie 已提交
316
/*	printk("validate_cmd( %x ): %d\n", cmd, ret); */
L
Linus Torvalds 已提交
317 318 319 320

	return ret;
}

321
static int i915_emit_cmds(struct drm_device * dev, int __user * buffer, int dwords)
L
Linus Torvalds 已提交
322 323 324 325 326
{
	drm_i915_private_t *dev_priv = dev->dev_private;
	int i;
	RING_LOCALS;

327
	if ((dwords+1) * sizeof(int) >= dev_priv->ring.Size - 8)
E
Eric Anholt 已提交
328
		return -EINVAL;
329

330
	BEGIN_LP_RING((dwords+1)&~1);
331

L
Linus Torvalds 已提交
332 333 334 335
	for (i = 0; i < dwords;) {
		int cmd, sz;

		if (DRM_COPY_FROM_USER_UNCHECKED(&cmd, &buffer[i], sizeof(cmd)))
E
Eric Anholt 已提交
336
			return -EINVAL;
L
Linus Torvalds 已提交
337 338

		if ((sz = validate_cmd(cmd)) == 0 || i + sz > dwords)
E
Eric Anholt 已提交
339
			return -EINVAL;
L
Linus Torvalds 已提交
340 341 342 343 344 345

		OUT_RING(cmd);

		while (++i, --sz) {
			if (DRM_COPY_FROM_USER_UNCHECKED(&cmd, &buffer[i],
							 sizeof(cmd))) {
E
Eric Anholt 已提交
346
				return -EINVAL;
L
Linus Torvalds 已提交
347 348 349 350 351
			}
			OUT_RING(cmd);
		}
	}

352 353 354 355 356
	if (dwords & 1)
		OUT_RING(0);

	ADVANCE_LP_RING();

L
Linus Torvalds 已提交
357 358 359
	return 0;
}

360
static int i915_emit_box(struct drm_device * dev,
361
			 struct drm_clip_rect __user * boxes,
L
Linus Torvalds 已提交
362 363 364
			 int i, int DR1, int DR4)
{
	drm_i915_private_t *dev_priv = dev->dev_private;
365
	struct drm_clip_rect box;
L
Linus Torvalds 已提交
366 367 368
	RING_LOCALS;

	if (DRM_COPY_FROM_USER_UNCHECKED(&box, &boxes[i], sizeof(box))) {
E
Eric Anholt 已提交
369
		return -EFAULT;
L
Linus Torvalds 已提交
370 371 372 373 374
	}

	if (box.y2 <= box.y1 || box.x2 <= box.x1 || box.y2 <= 0 || box.x2 <= 0) {
		DRM_ERROR("Bad box %d,%d..%d,%d\n",
			  box.x1, box.y1, box.x2, box.y2);
E
Eric Anholt 已提交
375
		return -EINVAL;
L
Linus Torvalds 已提交
376 377
	}

378 379 380 381
	if (IS_I965G(dev)) {
		BEGIN_LP_RING(4);
		OUT_RING(GFX_OP_DRAWRECT_INFO_I965);
		OUT_RING((box.x1 & 0xffff) | (box.y1 << 16));
A
Andrew Morton 已提交
382
		OUT_RING(((box.x2 - 1) & 0xffff) | ((box.y2 - 1) << 16));
383 384 385 386 387 388 389 390 391 392 393 394
		OUT_RING(DR4);
		ADVANCE_LP_RING();
	} else {
		BEGIN_LP_RING(6);
		OUT_RING(GFX_OP_DRAWRECT_INFO);
		OUT_RING(DR1);
		OUT_RING((box.x1 & 0xffff) | (box.y1 << 16));
		OUT_RING(((box.x2 - 1) & 0xffff) | ((box.y2 - 1) << 16));
		OUT_RING(DR4);
		OUT_RING(0);
		ADVANCE_LP_RING();
	}
L
Linus Torvalds 已提交
395 396 397 398

	return 0;
}

399 400 401 402
/* XXX: Emitting the counter should really be moved to part of the IRQ
 * emit. For now, do it in both places:
 */

403
static void i915_emit_breadcrumb(struct drm_device *dev)
404 405 406 407
{
	drm_i915_private_t *dev_priv = dev->dev_private;
	RING_LOCALS;

408
	dev_priv->sarea_priv->last_enqueue = ++dev_priv->counter;
409

410 411
	if (dev_priv->counter > 0x7FFFFFFFUL)
		dev_priv->sarea_priv->last_enqueue = dev_priv->counter = 1;
412 413

	BEGIN_LP_RING(4);
414 415
	OUT_RING(MI_STORE_DWORD_INDEX);
	OUT_RING(5 << MI_STORE_DWORD_INDEX_SHIFT);
416 417 418 419 420
	OUT_RING(dev_priv->counter);
	OUT_RING(0);
	ADVANCE_LP_RING();
}

421
static int i915_dispatch_cmdbuffer(struct drm_device * dev,
L
Linus Torvalds 已提交
422 423 424 425 426 427 428
				   drm_i915_cmdbuffer_t * cmd)
{
	int nbox = cmd->num_cliprects;
	int i = 0, count, ret;

	if (cmd->sz & 0x3) {
		DRM_ERROR("alignment");
E
Eric Anholt 已提交
429
		return -EINVAL;
L
Linus Torvalds 已提交
430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448
	}

	i915_kernel_lost_context(dev);

	count = nbox ? nbox : 1;

	for (i = 0; i < count; i++) {
		if (i < nbox) {
			ret = i915_emit_box(dev, cmd->cliprects, i,
					    cmd->DR1, cmd->DR4);
			if (ret)
				return ret;
		}

		ret = i915_emit_cmds(dev, (int __user *)cmd->buf, cmd->sz / 4);
		if (ret)
			return ret;
	}

449
	i915_emit_breadcrumb(dev);
L
Linus Torvalds 已提交
450 451 452
	return 0;
}

453
static int i915_dispatch_batchbuffer(struct drm_device * dev,
L
Linus Torvalds 已提交
454 455 456
				     drm_i915_batchbuffer_t * batch)
{
	drm_i915_private_t *dev_priv = dev->dev_private;
457
	struct drm_clip_rect __user *boxes = batch->cliprects;
L
Linus Torvalds 已提交
458 459 460 461 462 463
	int nbox = batch->num_cliprects;
	int i = 0, count;
	RING_LOCALS;

	if ((batch->start | batch->used) & 0x7) {
		DRM_ERROR("alignment");
E
Eric Anholt 已提交
464
		return -EINVAL;
L
Linus Torvalds 已提交
465 466 467 468 469 470 471 472 473 474 475 476 477 478
	}

	i915_kernel_lost_context(dev);

	count = nbox ? nbox : 1;

	for (i = 0; i < count; i++) {
		if (i < nbox) {
			int ret = i915_emit_box(dev, boxes, i,
						batch->DR1, batch->DR4);
			if (ret)
				return ret;
		}

479
		if (!IS_I830(dev) && !IS_845G(dev)) {
L
Linus Torvalds 已提交
480
			BEGIN_LP_RING(2);
481 482 483 484 485 486 487
			if (IS_I965G(dev)) {
				OUT_RING(MI_BATCH_BUFFER_START | (2 << 6) | MI_BATCH_NON_SECURE_I965);
				OUT_RING(batch->start);
			} else {
				OUT_RING(MI_BATCH_BUFFER_START | (2 << 6));
				OUT_RING(batch->start | MI_BATCH_NON_SECURE);
			}
L
Linus Torvalds 已提交
488 489 490 491 492 493 494 495 496 497 498
			ADVANCE_LP_RING();
		} else {
			BEGIN_LP_RING(4);
			OUT_RING(MI_BATCH_BUFFER);
			OUT_RING(batch->start | MI_BATCH_NON_SECURE);
			OUT_RING(batch->start + batch->used - 4);
			OUT_RING(0);
			ADVANCE_LP_RING();
		}
	}

499
	i915_emit_breadcrumb(dev);
L
Linus Torvalds 已提交
500 501 502 503

	return 0;
}

504
static int i915_dispatch_flip(struct drm_device * dev)
L
Linus Torvalds 已提交
505 506 507 508
{
	drm_i915_private_t *dev_priv = dev->dev_private;
	RING_LOCALS;

509
	DRM_DEBUG("%s: page=%d pfCurrentPage=%d\n",
510
		  __func__,
511 512
		  dev_priv->current_page,
		  dev_priv->sarea_priv->pf_current_page);
L
Linus Torvalds 已提交
513

514 515 516
	i915_kernel_lost_context(dev);

	BEGIN_LP_RING(2);
517
	OUT_RING(MI_FLUSH | MI_READ_FLUSH);
518 519
	OUT_RING(0);
	ADVANCE_LP_RING();
L
Linus Torvalds 已提交
520

521 522 523 524 525 526
	BEGIN_LP_RING(6);
	OUT_RING(CMD_OP_DISPLAYBUFFER_INFO | ASYNC_FLIP);
	OUT_RING(0);
	if (dev_priv->current_page == 0) {
		OUT_RING(dev_priv->back_offset);
		dev_priv->current_page = 1;
L
Linus Torvalds 已提交
527
	} else {
528 529
		OUT_RING(dev_priv->front_offset);
		dev_priv->current_page = 0;
L
Linus Torvalds 已提交
530
	}
531 532
	OUT_RING(0);
	ADVANCE_LP_RING();
L
Linus Torvalds 已提交
533

534 535 536 537
	BEGIN_LP_RING(2);
	OUT_RING(MI_WAIT_FOR_EVENT | MI_WAIT_FOR_PLANE_A_FLIP);
	OUT_RING(0);
	ADVANCE_LP_RING();
L
Linus Torvalds 已提交
538

539
	dev_priv->sarea_priv->last_enqueue = dev_priv->counter++;
L
Linus Torvalds 已提交
540 541

	BEGIN_LP_RING(4);
542 543
	OUT_RING(MI_STORE_DWORD_INDEX);
	OUT_RING(5 << MI_STORE_DWORD_INDEX_SHIFT);
544 545
	OUT_RING(dev_priv->counter);
	OUT_RING(0);
L
Linus Torvalds 已提交
546 547
	ADVANCE_LP_RING();

548 549
	dev_priv->sarea_priv->pf_current_page = dev_priv->current_page;
	return 0;
L
Linus Torvalds 已提交
550 551
}

552
static int i915_quiescent(struct drm_device * dev)
L
Linus Torvalds 已提交
553 554 555 556
{
	drm_i915_private_t *dev_priv = dev->dev_private;

	i915_kernel_lost_context(dev);
557
	return i915_wait_ring(dev, dev_priv->ring.Size - 8, __func__);
L
Linus Torvalds 已提交
558 559
}

560 561
static int i915_flush_ioctl(struct drm_device *dev, void *data,
			    struct drm_file *file_priv)
L
Linus Torvalds 已提交
562
{
563
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
564 565 566 567

	return i915_quiescent(dev);
}

568 569
static int i915_batchbuffer(struct drm_device *dev, void *data,
			    struct drm_file *file_priv)
L
Linus Torvalds 已提交
570 571
{
	drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
572
	u32 *hw_status = dev_priv->hw_status_page;
L
Linus Torvalds 已提交
573 574
	drm_i915_sarea_t *sarea_priv = (drm_i915_sarea_t *)
	    dev_priv->sarea_priv;
575
	drm_i915_batchbuffer_t *batch = data;
L
Linus Torvalds 已提交
576 577 578 579
	int ret;

	if (!dev_priv->allow_batchbuffer) {
		DRM_ERROR("Batchbuffer ioctl disabled\n");
E
Eric Anholt 已提交
580
		return -EINVAL;
L
Linus Torvalds 已提交
581 582 583
	}

	DRM_DEBUG("i915 batchbuffer, start %x used %d cliprects %d\n",
584
		  batch->start, batch->used, batch->num_cliprects);
L
Linus Torvalds 已提交
585

586
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
587

588 589
	if (batch->num_cliprects && DRM_VERIFYAREA_READ(batch->cliprects,
						       batch->num_cliprects *
590
						       sizeof(struct drm_clip_rect)))
E
Eric Anholt 已提交
591
		return -EFAULT;
L
Linus Torvalds 已提交
592

593
	ret = i915_dispatch_batchbuffer(dev, batch);
L
Linus Torvalds 已提交
594

595
	sarea_priv->last_dispatch = (int)hw_status[5];
L
Linus Torvalds 已提交
596 597 598
	return ret;
}

599 600
static int i915_cmdbuffer(struct drm_device *dev, void *data,
			  struct drm_file *file_priv)
L
Linus Torvalds 已提交
601 602
{
	drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
603
	u32 *hw_status = dev_priv->hw_status_page;
L
Linus Torvalds 已提交
604 605
	drm_i915_sarea_t *sarea_priv = (drm_i915_sarea_t *)
	    dev_priv->sarea_priv;
606
	drm_i915_cmdbuffer_t *cmdbuf = data;
L
Linus Torvalds 已提交
607 608 609
	int ret;

	DRM_DEBUG("i915 cmdbuffer, buf %p sz %d cliprects %d\n",
610
		  cmdbuf->buf, cmdbuf->sz, cmdbuf->num_cliprects);
L
Linus Torvalds 已提交
611

612
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
613

614 615 616
	if (cmdbuf->num_cliprects &&
	    DRM_VERIFYAREA_READ(cmdbuf->cliprects,
				cmdbuf->num_cliprects *
617
				sizeof(struct drm_clip_rect))) {
L
Linus Torvalds 已提交
618
		DRM_ERROR("Fault accessing cliprects\n");
E
Eric Anholt 已提交
619
		return -EFAULT;
L
Linus Torvalds 已提交
620 621
	}

622
	ret = i915_dispatch_cmdbuffer(dev, cmdbuf);
L
Linus Torvalds 已提交
623 624 625 626 627
	if (ret) {
		DRM_ERROR("i915_dispatch_cmdbuffer failed\n");
		return ret;
	}

628
	sarea_priv->last_dispatch = (int)hw_status[5];
L
Linus Torvalds 已提交
629 630 631
	return 0;
}

632 633
static int i915_flip_bufs(struct drm_device *dev, void *data,
			  struct drm_file *file_priv)
L
Linus Torvalds 已提交
634
{
635
	DRM_DEBUG("%s\n", __func__);
L
Linus Torvalds 已提交
636

637
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
638

639
	return i915_dispatch_flip(dev);
L
Linus Torvalds 已提交
640 641
}

642 643
static int i915_getparam(struct drm_device *dev, void *data,
			 struct drm_file *file_priv)
L
Linus Torvalds 已提交
644 645
{
	drm_i915_private_t *dev_priv = dev->dev_private;
646
	drm_i915_getparam_t *param = data;
L
Linus Torvalds 已提交
647 648 649
	int value;

	if (!dev_priv) {
650
		DRM_ERROR("called with no initialization\n");
E
Eric Anholt 已提交
651
		return -EINVAL;
L
Linus Torvalds 已提交
652 653
	}

654
	switch (param->param) {
L
Linus Torvalds 已提交
655
	case I915_PARAM_IRQ_ACTIVE:
656
		value = dev->irq_enabled;
L
Linus Torvalds 已提交
657 658 659 660
		break;
	case I915_PARAM_ALLOW_BATCHBUFFER:
		value = dev_priv->allow_batchbuffer ? 1 : 0;
		break;
D
Dave Airlie 已提交
661 662 663
	case I915_PARAM_LAST_DISPATCH:
		value = READ_BREADCRUMB(dev_priv);
		break;
L
Linus Torvalds 已提交
664
	default:
665
		DRM_ERROR("Unknown parameter %d\n", param->param);
E
Eric Anholt 已提交
666
		return -EINVAL;
L
Linus Torvalds 已提交
667 668
	}

669
	if (DRM_COPY_TO_USER(param->value, &value, sizeof(int))) {
L
Linus Torvalds 已提交
670
		DRM_ERROR("DRM_COPY_TO_USER failed\n");
E
Eric Anholt 已提交
671
		return -EFAULT;
L
Linus Torvalds 已提交
672 673 674 675 676
	}

	return 0;
}

677 678
static int i915_setparam(struct drm_device *dev, void *data,
			 struct drm_file *file_priv)
L
Linus Torvalds 已提交
679 680
{
	drm_i915_private_t *dev_priv = dev->dev_private;
681
	drm_i915_setparam_t *param = data;
L
Linus Torvalds 已提交
682 683

	if (!dev_priv) {
684
		DRM_ERROR("called with no initialization\n");
E
Eric Anholt 已提交
685
		return -EINVAL;
L
Linus Torvalds 已提交
686 687
	}

688
	switch (param->param) {
L
Linus Torvalds 已提交
689 690 691
	case I915_SETPARAM_USE_MI_BATCHBUFFER_START:
		break;
	case I915_SETPARAM_TEX_LRU_LOG_GRANULARITY:
692
		dev_priv->tex_lru_log_granularity = param->value;
L
Linus Torvalds 已提交
693 694
		break;
	case I915_SETPARAM_ALLOW_BATCHBUFFER:
695
		dev_priv->allow_batchbuffer = param->value;
L
Linus Torvalds 已提交
696 697
		break;
	default:
698
		DRM_ERROR("unknown parameter %d\n", param->param);
E
Eric Anholt 已提交
699
		return -EINVAL;
L
Linus Torvalds 已提交
700 701 702 703 704
	}

	return 0;
}

705 706
static int i915_set_status_page(struct drm_device *dev, void *data,
				struct drm_file *file_priv)
707 708
{
	drm_i915_private_t *dev_priv = dev->dev_private;
709
	drm_i915_hws_addr_t *hws = data;
710 711 712

	if (!I915_NEED_GFX_HWS(dev))
		return -EINVAL;
713 714

	if (!dev_priv) {
715
		DRM_ERROR("called with no initialization\n");
E
Eric Anholt 已提交
716
		return -EINVAL;
717 718
	}

719 720 721
	printk(KERN_DEBUG "set status page addr 0x%08x\n", (u32)hws->addr);

	dev_priv->status_gfx_addr = hws->addr & (0x1ffff<<12);
722

723
	dev_priv->hws_map.offset = dev->agp->base + hws->addr;
724 725 726 727 728 729 730 731 732 733 734
	dev_priv->hws_map.size = 4*1024;
	dev_priv->hws_map.type = 0;
	dev_priv->hws_map.flags = 0;
	dev_priv->hws_map.mtrr = 0;

	drm_core_ioremap(&dev_priv->hws_map, dev);
	if (dev_priv->hws_map.handle == NULL) {
		i915_dma_cleanup(dev);
		dev_priv->status_gfx_addr = 0;
		DRM_ERROR("can not ioremap virtual address for"
				" G33 hw status page\n");
E
Eric Anholt 已提交
735
		return -ENOMEM;
736 737 738 739
	}
	dev_priv->hw_status_page = dev_priv->hws_map.handle;

	memset(dev_priv->hw_status_page, 0, PAGE_SIZE);
740 741
	I915_WRITE(HWS_PGA, dev_priv->status_gfx_addr);
	DRM_DEBUG("load hws HWS_PGA with gfx mem 0x%x\n",
742 743 744 745 746
			dev_priv->status_gfx_addr);
	DRM_DEBUG("load hws at %p\n", dev_priv->hw_status_page);
	return 0;
}

747
int i915_driver_load(struct drm_device *dev, unsigned long flags)
748
{
J
Jesse Barnes 已提交
749 750 751 752
	struct drm_i915_private *dev_priv = dev->dev_private;
	unsigned long base, size;
	int ret = 0, mmio_bar = IS_I9XX(dev) ? 0 : 1;

753 754 755 756 757 758 759
	/* i915 has 4 more counters */
	dev->counters += 4;
	dev->types[6] = _DRM_STAT_IRQ;
	dev->types[7] = _DRM_STAT_PRIMARY;
	dev->types[8] = _DRM_STAT_SECONDARY;
	dev->types[9] = _DRM_STAT_DMA;

J
Jesse Barnes 已提交
760 761 762 763 764 765 766 767 768 769 770 771
	dev_priv = drm_alloc(sizeof(drm_i915_private_t), DRM_MEM_DRIVER);
	if (dev_priv == NULL)
		return -ENOMEM;

	memset(dev_priv, 0, sizeof(drm_i915_private_t));

	dev->dev_private = (void *)dev_priv;

	/* Add register map (needed for suspend/resume) */
	base = drm_get_resource_start(dev, mmio_bar);
	size = drm_get_resource_len(dev, mmio_bar);

772 773
	ret = drm_addmap(dev, base, size, _DRM_REGISTERS,
			 _DRM_KERNEL | _DRM_DRIVER,
J
Jesse Barnes 已提交
774
			 &dev_priv->mmio_map);
775 776 777 778 779 780 781 782 783 784 785 786 787 788


	/* On the 945G/GM, the chipset reports the MSI capability on the
	 * integrated graphics even though the support isn't actually there
	 * according to the published specs.  It doesn't appear to function
	 * correctly in testing on 945G.
	 * This may be a side effect of MSI having been made available for PEG
	 * and the registers being closely associated.
	 */
	if (!IS_I945G(dev) && !IS_I945GM(dev))
		pci_enable_msi(dev->pdev);

	spin_lock_init(&dev_priv->user_irq_lock);

J
Jesse Barnes 已提交
789 790 791 792 793 794 795
	return ret;
}

int i915_driver_unload(struct drm_device *dev)
{
	struct drm_i915_private *dev_priv = dev->dev_private;

796 797 798
	if (dev->pdev->msi_enabled)
		pci_disable_msi(dev->pdev);

J
Jesse Barnes 已提交
799 800 801 802 803 804
	if (dev_priv->mmio_map)
		drm_rmmap(dev, dev_priv->mmio_map);

	drm_free(dev->dev_private, sizeof(drm_i915_private_t),
		 DRM_MEM_DRIVER);

805 806 807
	return 0;
}

808
void i915_driver_lastclose(struct drm_device * dev)
L
Linus Torvalds 已提交
809
{
J
Jesse Barnes 已提交
810 811
	drm_i915_private_t *dev_priv = dev->dev_private;

D
Dave Airlie 已提交
812 813 814
	if (!dev_priv)
		return;

J
Jesse Barnes 已提交
815
	if (dev_priv->agp_heap)
D
Dave Airlie 已提交
816
		i915_mem_takedown(&(dev_priv->agp_heap));
J
Jesse Barnes 已提交
817

D
Dave Airlie 已提交
818
	i915_dma_cleanup(dev);
L
Linus Torvalds 已提交
819 820
}

821
void i915_driver_preclose(struct drm_device * dev, struct drm_file *file_priv)
L
Linus Torvalds 已提交
822
{
J
Jesse Barnes 已提交
823 824
	drm_i915_private_t *dev_priv = dev->dev_private;
	i915_mem_release(dev, file_priv, dev_priv->agp_heap);
L
Linus Torvalds 已提交
825 826
}

827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844
struct drm_ioctl_desc i915_ioctls[] = {
	DRM_IOCTL_DEF(DRM_I915_INIT, i915_dma_init, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
	DRM_IOCTL_DEF(DRM_I915_FLUSH, i915_flush_ioctl, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_FLIP, i915_flip_bufs, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_BATCHBUFFER, i915_batchbuffer, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_IRQ_EMIT, i915_irq_emit, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_IRQ_WAIT, i915_irq_wait, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_GETPARAM, i915_getparam, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_SETPARAM, i915_setparam, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
	DRM_IOCTL_DEF(DRM_I915_ALLOC, i915_mem_alloc, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_FREE, i915_mem_free, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_INIT_HEAP, i915_mem_init_heap, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
	DRM_IOCTL_DEF(DRM_I915_CMDBUFFER, i915_cmdbuffer, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_DESTROY_HEAP,  i915_mem_destroy_heap, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY ),
	DRM_IOCTL_DEF(DRM_I915_SET_VBLANK_PIPE,  i915_vblank_pipe_set, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY ),
	DRM_IOCTL_DEF(DRM_I915_GET_VBLANK_PIPE,  i915_vblank_pipe_get, DRM_AUTH ),
	DRM_IOCTL_DEF(DRM_I915_VBLANK_SWAP, i915_vblank_swap, DRM_AUTH),
	DRM_IOCTL_DEF(DRM_I915_HWS_ADDR, i915_set_status_page, DRM_AUTH),
D
Dave Airlie 已提交
845 846 847
};

int i915_max_ioctl = DRM_ARRAY_SIZE(i915_ioctls);
848 849 850 851 852 853 854 855 856 857 858 859

/**
 * Determine if the device really is AGP or not.
 *
 * All Intel graphics chipsets are treated as AGP, even if they are really
 * PCI-e.
 *
 * \param dev   The device to be tested.
 *
 * \returns
 * A value of 1 is always retured to indictate every i9x5 is AGP.
 */
860
int i915_driver_device_is_agp(struct drm_device * dev)
861 862 863
{
	return 1;
}