i830_dma.c 39.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/* i830_dma.c -- DMA support for the I830 -*- linux-c -*-
 * Created: Mon Dec 13 01:50:01 1999 by jhartmann@precisioninsight.com
 *
 * Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
 * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
 * All Rights Reserved.
 *
 * 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:
D
Dave Airlie 已提交
14
 *
L
Linus Torvalds 已提交
15 16 17
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
D
Dave Airlie 已提交
18
 *
L
Linus Torvalds 已提交
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
 * 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
 * PRECISION INSIGHT 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.
 *
 * Authors: Rickard E. (Rik) Faith <faith@valinux.com>
 *	    Jeff Hartmann <jhartmann@valinux.com>
 *	    Keith Whitwell <keith@tungstengraphics.com>
 *	    Abraham vd Merwe <abraham@2d3d.co.za>
 *
 */

#include "drmP.h"
#include "drm.h"
#include "i830_drm.h"
#include "i830_drv.h"
#include <linux/interrupt.h>	/* For task queue support */
A
Arnd Bergmann 已提交
39
#include <linux/smp_lock.h>
40
#include <linux/pagemap.h>
L
Linus Torvalds 已提交
41
#include <linux/delay.h>
42
#include <linux/slab.h>
L
Linus Torvalds 已提交
43 44 45 46
#include <asm/uaccess.h>

#define I830_BUF_FREE		2
#define I830_BUF_CLIENT		1
D
Dave Airlie 已提交
47
#define I830_BUF_HARDWARE	0
L
Linus Torvalds 已提交
48 49 50 51

#define I830_BUF_UNMAPPED 0
#define I830_BUF_MAPPED   1

D
Dave Airlie 已提交
52
static struct drm_buf *i830_freelist_get(struct drm_device * dev)
L
Linus Torvalds 已提交
53
{
54
	struct drm_device_dma *dma = dev->dma;
D
Dave Airlie 已提交
55 56 57
	int i;
	int used;

L
Linus Torvalds 已提交
58 59
	/* Linear search might not be the best solution */

D
Dave Airlie 已提交
60
	for (i = 0; i < dma->buf_count; i++) {
D
Dave Airlie 已提交
61
		struct drm_buf *buf = dma->buflist[i];
D
Dave Airlie 已提交
62
		drm_i830_buf_priv_t *buf_priv = buf->dev_private;
L
Linus Torvalds 已提交
63
		/* In use is already a pointer */
D
Dave Airlie 已提交
64
		used = cmpxchg(buf_priv->in_use, I830_BUF_FREE,
L
Linus Torvalds 已提交
65
			       I830_BUF_CLIENT);
66
		if (used == I830_BUF_FREE)
L
Linus Torvalds 已提交
67 68
			return buf;
	}
D
Dave Airlie 已提交
69
	return NULL;
L
Linus Torvalds 已提交
70 71 72 73 74 75
}

/* This should only be called if the buffer is not sent to the hardware
 * yet, the hardware updates in use for us once its on the ring buffer.
 */

76
static int i830_freelist_put(struct drm_device *dev, struct drm_buf *buf)
L
Linus Torvalds 已提交
77
{
D
Dave Airlie 已提交
78 79 80 81 82 83 84 85
	drm_i830_buf_priv_t *buf_priv = buf->dev_private;
	int used;

	/* In use is already a pointer */
	used = cmpxchg(buf_priv->in_use, I830_BUF_CLIENT, I830_BUF_FREE);
	if (used != I830_BUF_CLIENT) {
		DRM_ERROR("Freeing buffer thats not in use : %d\n", buf->idx);
		return -EINVAL;
L
Linus Torvalds 已提交
86
	}
D
Dave Airlie 已提交
87 88

	return 0;
L
Linus Torvalds 已提交
89 90
}

D
Dave Airlie 已提交
91
static int i830_mmap_buffers(struct file *filp, struct vm_area_struct *vma)
L
Linus Torvalds 已提交
92
{
93 94
	struct drm_file *priv = filp->private_data;
	struct drm_device *dev;
D
Dave Airlie 已提交
95
	drm_i830_private_t *dev_priv;
D
Dave Airlie 已提交
96
	struct drm_buf *buf;
L
Linus Torvalds 已提交
97 98 99
	drm_i830_buf_priv_t *buf_priv;

	lock_kernel();
100
	dev = priv->minor->dev;
L
Linus Torvalds 已提交
101
	dev_priv = dev->dev_private;
D
Dave Airlie 已提交
102
	buf = dev_priv->mmap_buffer;
L
Linus Torvalds 已提交
103
	buf_priv = buf->dev_private;
D
Dave Airlie 已提交
104

L
Linus Torvalds 已提交
105 106
	vma->vm_flags |= (VM_IO | VM_DONTCOPY);
	vma->vm_file = filp;
D
Dave Airlie 已提交
107 108

	buf_priv->currently_mapped = I830_BUF_MAPPED;
L
Linus Torvalds 已提交
109 110 111
	unlock_kernel();

	if (io_remap_pfn_range(vma, vma->vm_start,
112
			       vma->vm_pgoff,
D
Dave Airlie 已提交
113 114
			       vma->vm_end - vma->vm_start, vma->vm_page_prot))
		return -EAGAIN;
L
Linus Torvalds 已提交
115 116 117
	return 0;
}

118
static const struct file_operations i830_buffer_fops = {
D
Dave Airlie 已提交
119
	.open = drm_open,
D
Dave Airlie 已提交
120
	.release = drm_release,
121
	.unlocked_ioctl = drm_ioctl,
D
Dave Airlie 已提交
122 123
	.mmap = i830_mmap_buffers,
	.fasync = drm_fasync,
D
Dave Airlie 已提交
124 125
};

126
static int i830_map_buffer(struct drm_buf *buf, struct drm_file *file_priv)
L
Linus Torvalds 已提交
127
{
128
	struct drm_device *dev = file_priv->minor->dev;
L
Linus Torvalds 已提交
129
	drm_i830_buf_priv_t *buf_priv = buf->dev_private;
D
Dave Airlie 已提交
130
	drm_i830_private_t *dev_priv = dev->dev_private;
131
	const struct file_operations *old_fops;
L
Linus Torvalds 已提交
132 133 134
	unsigned long virtual;
	int retcode = 0;

D
Dave Airlie 已提交
135 136
	if (buf_priv->currently_mapped == I830_BUF_MAPPED)
		return -EINVAL;
L
Linus Torvalds 已提交
137

D
Dave Airlie 已提交
138
	down_write(&current->mm->mmap_sem);
139 140
	old_fops = file_priv->filp->f_op;
	file_priv->filp->f_op = &i830_buffer_fops;
L
Linus Torvalds 已提交
141
	dev_priv->mmap_buffer = buf;
142
	virtual = do_mmap(file_priv->filp, 0, buf->total, PROT_READ | PROT_WRITE,
D
Dave Airlie 已提交
143
			  MAP_SHARED, buf->bus_address);
L
Linus Torvalds 已提交
144
	dev_priv->mmap_buffer = NULL;
145
	file_priv->filp->f_op = old_fops;
D
Dave Airlie 已提交
146
	if (IS_ERR((void *)virtual)) {	/* ugh */
L
Linus Torvalds 已提交
147 148
		/* Real error */
		DRM_ERROR("mmap error\n");
149
		retcode = PTR_ERR((void *)virtual);
L
Linus Torvalds 已提交
150 151 152 153
		buf_priv->virtual = NULL;
	} else {
		buf_priv->virtual = (void __user *)virtual;
	}
D
Dave Airlie 已提交
154
	up_write(&current->mm->mmap_sem);
L
Linus Torvalds 已提交
155 156 157 158

	return retcode;
}

159
static int i830_unmap_buffer(struct drm_buf *buf)
L
Linus Torvalds 已提交
160 161 162 163
{
	drm_i830_buf_priv_t *buf_priv = buf->dev_private;
	int retcode = 0;

D
Dave Airlie 已提交
164
	if (buf_priv->currently_mapped != I830_BUF_MAPPED)
L
Linus Torvalds 已提交
165 166 167 168 169 170 171 172
		return -EINVAL;

	down_write(&current->mm->mmap_sem);
	retcode = do_munmap(current->mm,
			    (unsigned long)buf_priv->virtual,
			    (size_t) buf->total);
	up_write(&current->mm->mmap_sem);

D
Dave Airlie 已提交
173 174
	buf_priv->currently_mapped = I830_BUF_UNMAPPED;
	buf_priv->virtual = NULL;
L
Linus Torvalds 已提交
175 176 177 178

	return retcode;
}

179
static int i830_dma_get_buffer(struct drm_device *dev, drm_i830_dma_t *d,
180
			       struct drm_file *file_priv)
L
Linus Torvalds 已提交
181
{
D
Dave Airlie 已提交
182
	struct drm_buf *buf;
L
Linus Torvalds 已提交
183 184 185 186 187 188
	drm_i830_buf_priv_t *buf_priv;
	int retcode = 0;

	buf = i830_freelist_get(dev);
	if (!buf) {
		retcode = -ENOMEM;
D
Dave Airlie 已提交
189
		DRM_DEBUG("retcode=%d\n", retcode);
L
Linus Torvalds 已提交
190 191
		return retcode;
	}
D
Dave Airlie 已提交
192

193
	retcode = i830_map_buffer(buf, file_priv);
D
Dave Airlie 已提交
194
	if (retcode) {
L
Linus Torvalds 已提交
195
		i830_freelist_put(dev, buf);
D
Dave Airlie 已提交
196
		DRM_ERROR("mapbuf failed, retcode %d\n", retcode);
L
Linus Torvalds 已提交
197 198
		return retcode;
	}
199
	buf->file_priv = file_priv;
D
Dave Airlie 已提交
200
	buf_priv = buf->dev_private;
L
Linus Torvalds 已提交
201
	d->granted = 1;
D
Dave Airlie 已提交
202 203 204
	d->request_idx = buf->idx;
	d->request_size = buf->total;
	d->virtual = buf_priv->virtual;
L
Linus Torvalds 已提交
205 206 207 208

	return retcode;
}

209
static int i830_dma_cleanup(struct drm_device *dev)
L
Linus Torvalds 已提交
210
{
211
	struct drm_device_dma *dma = dev->dma;
L
Linus Torvalds 已提交
212 213 214 215 216

	/* 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.
	 */
D
Dave Airlie 已提交
217 218
	if (dev->irq_enabled)
		drm_irq_uninstall(dev);
L
Linus Torvalds 已提交
219 220 221

	if (dev->dev_private) {
		int i;
D
Dave Airlie 已提交
222 223 224
		drm_i830_private_t *dev_priv =
		    (drm_i830_private_t *) dev->dev_private;

225
		if (dev_priv->ring.virtual_start)
226
			drm_core_ioremapfree(&dev_priv->ring.map, dev);
D
Dave Airlie 已提交
227
		if (dev_priv->hw_status_page) {
L
Linus Torvalds 已提交
228 229 230
			pci_free_consistent(dev->pdev, PAGE_SIZE,
					    dev_priv->hw_status_page,
					    dev_priv->dma_status_page);
D
Dave Airlie 已提交
231 232
			/* Need to rewrite hardware status page */
			I830_WRITE(0x02080, 0x1ffff000);
L
Linus Torvalds 已提交
233 234
		}

235
		kfree(dev->dev_private);
D
Dave Airlie 已提交
236
		dev->dev_private = NULL;
L
Linus Torvalds 已提交
237 238

		for (i = 0; i < dma->buf_count; i++) {
D
Dave Airlie 已提交
239
			struct drm_buf *buf = dma->buflist[i];
L
Linus Torvalds 已提交
240
			drm_i830_buf_priv_t *buf_priv = buf->dev_private;
D
Dave Airlie 已提交
241
			if (buf_priv->kernel_virtual && buf->total)
242
				drm_core_ioremapfree(&buf_priv->map, dev);
L
Linus Torvalds 已提交
243 244
		}
	}
D
Dave Airlie 已提交
245
	return 0;
L
Linus Torvalds 已提交
246 247
}

248
int i830_wait_ring(struct drm_device *dev, int n, const char *caller)
L
Linus Torvalds 已提交
249
{
D
Dave Airlie 已提交
250 251 252 253
	drm_i830_private_t *dev_priv = dev->dev_private;
	drm_i830_ring_buffer_t *ring = &(dev_priv->ring);
	int iters = 0;
	unsigned long end;
L
Linus Torvalds 已提交
254 255
	unsigned int last_head = I830_READ(LP_RING + RING_HEAD) & HEAD_ADDR;

D
Dave Airlie 已提交
256 257 258 259 260 261 262
	end = jiffies + (HZ * 3);
	while (ring->space < n) {
		ring->head = I830_READ(LP_RING + RING_HEAD) & HEAD_ADDR;
		ring->space = ring->head - (ring->tail + 8);
		if (ring->space < 0)
			ring->space += ring->Size;

L
Linus Torvalds 已提交
263
		if (ring->head != last_head) {
D
Dave Airlie 已提交
264
			end = jiffies + (HZ * 3);
L
Linus Torvalds 已提交
265 266
			last_head = ring->head;
		}
D
Dave Airlie 已提交
267 268 269 270 271 272

		iters++;
		if (time_before(end, jiffies)) {
			DRM_ERROR("space: %d wanted %d\n", ring->space, n);
			DRM_ERROR("lockup\n");
			goto out_wait_ring;
L
Linus Torvalds 已提交
273 274 275 276 277
		}
		udelay(1);
		dev_priv->sarea_priv->perf_boxes |= I830_BOX_WAIT;
	}

278
out_wait_ring:
D
Dave Airlie 已提交
279
	return iters;
L
Linus Torvalds 已提交
280 281
}

282
static void i830_kernel_lost_context(struct drm_device *dev)
L
Linus Torvalds 已提交
283
{
D
Dave Airlie 已提交
284 285 286 287 288 289 290 291
	drm_i830_private_t *dev_priv = dev->dev_private;
	drm_i830_ring_buffer_t *ring = &(dev_priv->ring);

	ring->head = I830_READ(LP_RING + RING_HEAD) & HEAD_ADDR;
	ring->tail = I830_READ(LP_RING + RING_TAIL) & TAIL_ADDR;
	ring->space = ring->head - (ring->tail + 8);
	if (ring->space < 0)
		ring->space += ring->Size;
L
Linus Torvalds 已提交
292 293 294 295 296

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

297
static int i830_freelist_init(struct drm_device *dev, drm_i830_private_t *dev_priv)
L
Linus Torvalds 已提交
298
{
299
	struct drm_device_dma *dma = dev->dma;
D
Dave Airlie 已提交
300 301 302 303 304 305 306
	int my_idx = 36;
	u32 *hw_status = (u32 *) (dev_priv->hw_status_page + my_idx);
	int i;

	if (dma->buf_count > 1019) {
		/* Not enough space in the status page for the freelist */
		return -EINVAL;
L
Linus Torvalds 已提交
307 308
	}

D
Dave Airlie 已提交
309
	for (i = 0; i < dma->buf_count; i++) {
D
Dave Airlie 已提交
310
		struct drm_buf *buf = dma->buflist[i];
D
Dave Airlie 已提交
311
		drm_i830_buf_priv_t *buf_priv = buf->dev_private;
L
Linus Torvalds 已提交
312

D
Dave Airlie 已提交
313 314 315
		buf_priv->in_use = hw_status++;
		buf_priv->my_use_idx = my_idx;
		my_idx += 4;
L
Linus Torvalds 已提交
316

D
Dave Airlie 已提交
317
		*buf_priv->in_use = I830_BUF_FREE;
L
Linus Torvalds 已提交
318

319 320 321 322 323 324 325 326
		buf_priv->map.offset = buf->bus_address;
		buf_priv->map.size = buf->total;
		buf_priv->map.type = _DRM_AGP;
		buf_priv->map.flags = 0;
		buf_priv->map.mtrr = 0;

		drm_core_ioremap(&buf_priv->map, dev);
		buf_priv->kernel_virtual = buf_priv->map.handle;
L
Linus Torvalds 已提交
327 328 329 330
	}
	return 0;
}

331 332 333
static int i830_dma_initialize(struct drm_device *dev,
			       drm_i830_private_t *dev_priv,
			       drm_i830_init_t *init)
L
Linus Torvalds 已提交
334
{
D
Dave Airlie 已提交
335
	struct drm_map_list *r_list;
L
Linus Torvalds 已提交
336

D
Dave Airlie 已提交
337
	memset(dev_priv, 0, sizeof(drm_i830_private_t));
L
Linus Torvalds 已提交
338

339
	list_for_each_entry(r_list, &dev->maplist, head) {
D
Dave Airlie 已提交
340
		if (r_list->map &&
L
Linus Torvalds 已提交
341
		    r_list->map->type == _DRM_SHM &&
D
Dave Airlie 已提交
342
		    r_list->map->flags & _DRM_CONTAINS_LOCK) {
L
Linus Torvalds 已提交
343
			dev_priv->sarea_map = r_list->map;
D
Dave Airlie 已提交
344 345 346
			break;
		}
	}
L
Linus Torvalds 已提交
347

D
Dave Airlie 已提交
348
	if (!dev_priv->sarea_map) {
L
Linus Torvalds 已提交
349 350 351 352 353 354
		dev->dev_private = (void *)dev_priv;
		i830_dma_cleanup(dev);
		DRM_ERROR("can not find sarea!\n");
		return -EINVAL;
	}
	dev_priv->mmio_map = drm_core_findmap(dev, init->mmio_offset);
D
Dave Airlie 已提交
355
	if (!dev_priv->mmio_map) {
L
Linus Torvalds 已提交
356 357 358 359 360
		dev->dev_private = (void *)dev_priv;
		i830_dma_cleanup(dev);
		DRM_ERROR("can not find mmio map!\n");
		return -EINVAL;
	}
361
	dev->agp_buffer_token = init->buffers_offset;
L
Linus Torvalds 已提交
362
	dev->agp_buffer_map = drm_core_findmap(dev, init->buffers_offset);
D
Dave Airlie 已提交
363
	if (!dev->agp_buffer_map) {
L
Linus Torvalds 已提交
364 365 366 367 368 369 370
		dev->dev_private = (void *)dev_priv;
		i830_dma_cleanup(dev);
		DRM_ERROR("can not find dma buffer map!\n");
		return -EINVAL;
	}

	dev_priv->sarea_priv = (drm_i830_sarea_t *)
D
Dave Airlie 已提交
371
	    ((u8 *) dev_priv->sarea_map->handle + init->sarea_priv_offset);
L
Linus Torvalds 已提交
372

D
Dave Airlie 已提交
373 374 375
	dev_priv->ring.Start = init->ring_start;
	dev_priv->ring.End = init->ring_end;
	dev_priv->ring.Size = init->ring_size;
L
Linus Torvalds 已提交
376

377 378 379 380 381 382 383
	dev_priv->ring.map.offset = dev->agp->base + init->ring_start;
	dev_priv->ring.map.size = init->ring_size;
	dev_priv->ring.map.type = _DRM_AGP;
	dev_priv->ring.map.flags = 0;
	dev_priv->ring.map.mtrr = 0;

	drm_core_ioremap(&dev_priv->ring.map, dev);
L
Linus Torvalds 已提交
384

385
	if (dev_priv->ring.map.handle == NULL) {
D
Dave Airlie 已提交
386 387 388
		dev->dev_private = (void *)dev_priv;
		i830_dma_cleanup(dev);
		DRM_ERROR("can not ioremap virtual address for"
L
Linus Torvalds 已提交
389
			  " ring buffer\n");
E
Eric Anholt 已提交
390
		return -ENOMEM;
L
Linus Torvalds 已提交
391 392
	}

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

D
Dave Airlie 已提交
395 396
	dev_priv->ring.tail_mask = dev_priv->ring.Size - 1;

L
Linus Torvalds 已提交
397 398 399 400 401 402 403 404 405 406 407
	dev_priv->w = init->w;
	dev_priv->h = init->h;
	dev_priv->pitch = init->pitch;
	dev_priv->back_offset = init->back_offset;
	dev_priv->depth_offset = init->depth_offset;
	dev_priv->front_offset = init->front_offset;

	dev_priv->front_di1 = init->front_offset | init->pitch_bits;
	dev_priv->back_di1 = init->back_offset | init->pitch_bits;
	dev_priv->zi1 = init->depth_offset | init->pitch_bits;

D
Dave Airlie 已提交
408
	DRM_DEBUG("front_di1 %x\n", dev_priv->front_di1);
L
Linus Torvalds 已提交
409
	DRM_DEBUG("back_offset %x\n", dev_priv->back_offset);
D
Dave Airlie 已提交
410 411
	DRM_DEBUG("back_di1 %x\n", dev_priv->back_di1);
	DRM_DEBUG("pitch_bits %x\n", init->pitch_bits);
L
Linus Torvalds 已提交
412 413 414 415 416 417 418 419 420 421 422

	dev_priv->cpp = init->cpp;
	/* We are using separate values as placeholders for mechanisms for
	 * private backbuffer/depthbuffer usage.
	 */

	dev_priv->back_pitch = init->back_pitch;
	dev_priv->depth_pitch = init->depth_pitch;
	dev_priv->do_boxes = 0;
	dev_priv->use_mi_batchbuffer_start = 0;

D
Dave Airlie 已提交
423 424 425 426 427
	/* Program Hardware Status Page */
	dev_priv->hw_status_page =
	    pci_alloc_consistent(dev->pdev, PAGE_SIZE,
				 &dev_priv->dma_status_page);
	if (!dev_priv->hw_status_page) {
L
Linus Torvalds 已提交
428 429 430 431 432
		dev->dev_private = (void *)dev_priv;
		i830_dma_cleanup(dev);
		DRM_ERROR("Can not allocate hardware status page\n");
		return -ENOMEM;
	}
D
Dave Airlie 已提交
433
	memset(dev_priv->hw_status_page, 0, PAGE_SIZE);
L
Linus Torvalds 已提交
434
	DRM_DEBUG("hw status page @ %p\n", dev_priv->hw_status_page);
D
Dave Airlie 已提交
435 436

	I830_WRITE(0x02080, dev_priv->dma_status_page);
L
Linus Torvalds 已提交
437
	DRM_DEBUG("Enabled hardware status page\n");
D
Dave Airlie 已提交
438 439 440

	/* Now we need to init our freelist */
	if (i830_freelist_init(dev, dev_priv) != 0) {
L
Linus Torvalds 已提交
441
		dev->dev_private = (void *)dev_priv;
D
Dave Airlie 已提交
442 443
		i830_dma_cleanup(dev);
		DRM_ERROR("Not enough space in the status page for"
L
Linus Torvalds 已提交
444
			  " the freelist\n");
D
Dave Airlie 已提交
445
		return -ENOMEM;
L
Linus Torvalds 已提交
446 447 448
	}
	dev->dev_private = (void *)dev_priv;

D
Dave Airlie 已提交
449
	return 0;
L
Linus Torvalds 已提交
450 451
}

452 453
static int i830_dma_init(struct drm_device *dev, void *data,
			 struct drm_file *file_priv)
L
Linus Torvalds 已提交
454
{
D
Dave Airlie 已提交
455
	drm_i830_private_t *dev_priv;
456
	drm_i830_init_t *init = data;
D
Dave Airlie 已提交
457 458
	int retcode = 0;

459
	switch (init->func) {
D
Dave Airlie 已提交
460
	case I830_INIT_DMA:
461
		dev_priv = kmalloc(sizeof(drm_i830_private_t), GFP_KERNEL);
D
Dave Airlie 已提交
462 463
		if (dev_priv == NULL)
			return -ENOMEM;
464
		retcode = i830_dma_initialize(dev, dev_priv, init);
D
Dave Airlie 已提交
465 466 467 468 469 470 471
		break;
	case I830_CLEANUP_DMA:
		retcode = i830_dma_cleanup(dev);
		break;
	default:
		retcode = -EINVAL;
		break;
L
Linus Torvalds 已提交
472
	}
D
Dave Airlie 已提交
473 474

	return retcode;
L
Linus Torvalds 已提交
475 476 477 478 479 480 481 482 483
}

#define GFX_OP_STIPPLE           ((0x3<<29)|(0x1d<<24)|(0x83<<16))
#define ST1_ENABLE               (1<<16)
#define ST1_MASK                 (0xffff)

/* Most efficient way to verify state for the i830 is as it is
 * emitted.  Non-conformant state is silently dropped.
 */
484
static void i830EmitContextVerified(struct drm_device *dev, unsigned int *code)
L
Linus Torvalds 已提交
485
{
D
Dave Airlie 已提交
486
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
487 488 489 490
	int i, j = 0;
	unsigned int tmp;
	RING_LOCALS;

D
Dave Airlie 已提交
491
	BEGIN_LP_RING(I830_CTX_SETUP_SIZE + 4);
L
Linus Torvalds 已提交
492

D
Dave Airlie 已提交
493
	for (i = 0; i < I830_CTXREG_BLENDCOLR0; i++) {
L
Linus Torvalds 已提交
494
		tmp = code[i];
D
Dave Airlie 已提交
495 496 497
		if ((tmp & (7 << 29)) == CMD_3D &&
		    (tmp & (0x1f << 24)) < (0x1d << 24)) {
			OUT_RING(tmp);
L
Linus Torvalds 已提交
498 499 500 501 502 503
			j++;
		} else {
			DRM_ERROR("Skipping %d\n", i);
		}
	}

D
Dave Airlie 已提交
504 505
	OUT_RING(STATE3D_CONST_BLEND_COLOR_CMD);
	OUT_RING(code[I830_CTXREG_BLENDCOLR]);
L
Linus Torvalds 已提交
506 507
	j += 2;

D
Dave Airlie 已提交
508
	for (i = I830_CTXREG_VF; i < I830_CTXREG_MCSB0; i++) {
L
Linus Torvalds 已提交
509
		tmp = code[i];
D
Dave Airlie 已提交
510 511 512
		if ((tmp & (7 << 29)) == CMD_3D &&
		    (tmp & (0x1f << 24)) < (0x1d << 24)) {
			OUT_RING(tmp);
L
Linus Torvalds 已提交
513 514 515 516 517 518
			j++;
		} else {
			DRM_ERROR("Skipping %d\n", i);
		}
	}

D
Dave Airlie 已提交
519 520
	OUT_RING(STATE3D_MAP_COORD_SETBIND_CMD);
	OUT_RING(code[I830_CTXREG_MCSB1]);
L
Linus Torvalds 已提交
521 522
	j += 2;

D
Dave Airlie 已提交
523 524
	if (j & 1)
		OUT_RING(0);
L
Linus Torvalds 已提交
525 526 527 528

	ADVANCE_LP_RING();
}

529
static void i830EmitTexVerified(struct drm_device *dev, unsigned int *code)
L
Linus Torvalds 已提交
530
{
D
Dave Airlie 已提交
531
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
532 533 534 535 536
	int i, j = 0;
	unsigned int tmp;
	RING_LOCALS;

	if (code[I830_TEXREG_MI0] == GFX_OP_MAP_INFO ||
D
Dave Airlie 已提交
537 538 539 540 541 542 543 544 545 546 547 548 549
	    (code[I830_TEXREG_MI0] & ~(0xf * LOAD_TEXTURE_MAP0)) ==
	    (STATE3D_LOAD_STATE_IMMEDIATE_2 | 4)) {

		BEGIN_LP_RING(I830_TEX_SETUP_SIZE);

		OUT_RING(code[I830_TEXREG_MI0]);	/* TM0LI */
		OUT_RING(code[I830_TEXREG_MI1]);	/* TM0S0 */
		OUT_RING(code[I830_TEXREG_MI2]);	/* TM0S1 */
		OUT_RING(code[I830_TEXREG_MI3]);	/* TM0S2 */
		OUT_RING(code[I830_TEXREG_MI4]);	/* TM0S3 */
		OUT_RING(code[I830_TEXREG_MI5]);	/* TM0S4 */

		for (i = 6; i < I830_TEX_SETUP_SIZE; i++) {
L
Linus Torvalds 已提交
550
			tmp = code[i];
D
Dave Airlie 已提交
551
			OUT_RING(tmp);
L
Linus Torvalds 已提交
552
			j++;
D
Dave Airlie 已提交
553
		}
L
Linus Torvalds 已提交
554

D
Dave Airlie 已提交
555 556
		if (j & 1)
			OUT_RING(0);
L
Linus Torvalds 已提交
557 558

		ADVANCE_LP_RING();
D
Dave Airlie 已提交
559
	} else
L
Linus Torvalds 已提交
560 561 562
		printk("rejected packet %x\n", code[0]);
}

563
static void i830EmitTexBlendVerified(struct drm_device *dev,
D
Dave Airlie 已提交
564
				     unsigned int *code, unsigned int num)
L
Linus Torvalds 已提交
565
{
D
Dave Airlie 已提交
566
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
567 568 569 570 571 572 573
	int i, j = 0;
	unsigned int tmp;
	RING_LOCALS;

	if (!num)
		return;

D
Dave Airlie 已提交
574
	BEGIN_LP_RING(num + 1);
L
Linus Torvalds 已提交
575

D
Dave Airlie 已提交
576
	for (i = 0; i < num; i++) {
L
Linus Torvalds 已提交
577
		tmp = code[i];
D
Dave Airlie 已提交
578
		OUT_RING(tmp);
L
Linus Torvalds 已提交
579 580 581
		j++;
	}

D
Dave Airlie 已提交
582 583
	if (j & 1)
		OUT_RING(0);
L
Linus Torvalds 已提交
584 585 586 587

	ADVANCE_LP_RING();
}

588
static void i830EmitTexPalette(struct drm_device *dev,
D
Dave Airlie 已提交
589
			       unsigned int *palette, int number, int is_shared)
L
Linus Torvalds 已提交
590
{
D
Dave Airlie 已提交
591
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
592 593 594 595 596
	int i;
	RING_LOCALS;

	return;

D
Dave Airlie 已提交
597
	BEGIN_LP_RING(258);
L
Linus Torvalds 已提交
598

D
Dave Airlie 已提交
599
	if (is_shared == 1) {
L
Linus Torvalds 已提交
600
		OUT_RING(CMD_OP_MAP_PALETTE_LOAD |
D
Dave Airlie 已提交
601
			 MAP_PALETTE_NUM(0) | MAP_PALETTE_BOTH);
L
Linus Torvalds 已提交
602 603 604
	} else {
		OUT_RING(CMD_OP_MAP_PALETTE_LOAD | MAP_PALETTE_NUM(number));
	}
605
	for (i = 0; i < 256; i++)
L
Linus Torvalds 已提交
606 607
		OUT_RING(palette[i]);
	OUT_RING(0);
D
Dave Airlie 已提交
608
	/* KW:  WHERE IS THE ADVANCE_LP_RING?  This is effectively a noop!
L
Linus Torvalds 已提交
609 610 611 612 613
	 */
}

/* Need to do some additional checking when setting the dest buffer.
 */
614
static void i830EmitDestVerified(struct drm_device *dev, unsigned int *code)
D
Dave Airlie 已提交
615 616
{
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
617 618 619
	unsigned int tmp;
	RING_LOCALS;

D
Dave Airlie 已提交
620
	BEGIN_LP_RING(I830_DEST_SETUP_SIZE + 10);
L
Linus Torvalds 已提交
621 622 623 624 625 626 627 628

	tmp = code[I830_DESTREG_CBUFADDR];
	if (tmp == dev_priv->front_di1 || tmp == dev_priv->back_di1) {
		if (((int)outring) & 8) {
			OUT_RING(0);
			OUT_RING(0);
		}

D
Dave Airlie 已提交
629 630 631 632 633 634 635 636 637 638 639 640
		OUT_RING(CMD_OP_DESTBUFFER_INFO);
		OUT_RING(BUF_3D_ID_COLOR_BACK |
			 BUF_3D_PITCH(dev_priv->back_pitch * dev_priv->cpp) |
			 BUF_3D_USE_FENCE);
		OUT_RING(tmp);
		OUT_RING(0);

		OUT_RING(CMD_OP_DESTBUFFER_INFO);
		OUT_RING(BUF_3D_ID_DEPTH | BUF_3D_USE_FENCE |
			 BUF_3D_PITCH(dev_priv->depth_pitch * dev_priv->cpp));
		OUT_RING(dev_priv->zi1);
		OUT_RING(0);
L
Linus Torvalds 已提交
641 642 643 644 645 646 647 648
	} else {
		DRM_ERROR("bad di1 %x (allow %x or %x)\n",
			  tmp, dev_priv->front_di1, dev_priv->back_di1);
	}

	/* invarient:
	 */

D
Dave Airlie 已提交
649 650
	OUT_RING(GFX_OP_DESTBUFFER_VARS);
	OUT_RING(code[I830_DESTREG_DV1]);
L
Linus Torvalds 已提交
651

D
Dave Airlie 已提交
652 653 654 655 656
	OUT_RING(GFX_OP_DRAWRECT_INFO);
	OUT_RING(code[I830_DESTREG_DR1]);
	OUT_RING(code[I830_DESTREG_DR2]);
	OUT_RING(code[I830_DESTREG_DR3]);
	OUT_RING(code[I830_DESTREG_DR4]);
L
Linus Torvalds 已提交
657 658 659

	/* Need to verify this */
	tmp = code[I830_DESTREG_SENABLE];
D
Dave Airlie 已提交
660 661
	if ((tmp & ~0x3) == GFX_OP_SCISSOR_ENABLE) {
		OUT_RING(tmp);
L
Linus Torvalds 已提交
662 663
	} else {
		DRM_ERROR("bad scissor enable\n");
D
Dave Airlie 已提交
664
		OUT_RING(0);
L
Linus Torvalds 已提交
665 666
	}

D
Dave Airlie 已提交
667 668 669 670
	OUT_RING(GFX_OP_SCISSOR_RECT);
	OUT_RING(code[I830_DESTREG_SR1]);
	OUT_RING(code[I830_DESTREG_SR2]);
	OUT_RING(0);
L
Linus Torvalds 已提交
671 672 673 674

	ADVANCE_LP_RING();
}

675
static void i830EmitStippleVerified(struct drm_device *dev, unsigned int *code)
L
Linus Torvalds 已提交
676
{
D
Dave Airlie 已提交
677
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
678 679
	RING_LOCALS;

D
Dave Airlie 已提交
680 681 682 683
	BEGIN_LP_RING(2);
	OUT_RING(GFX_OP_STIPPLE);
	OUT_RING(code[1]);
	ADVANCE_LP_RING();
L
Linus Torvalds 已提交
684 685
}

686
static void i830EmitState(struct drm_device *dev)
L
Linus Torvalds 已提交
687
{
D
Dave Airlie 已提交
688 689
	drm_i830_private_t *dev_priv = dev->dev_private;
	drm_i830_sarea_t *sarea_priv = dev_priv->sarea_priv;
L
Linus Torvalds 已提交
690 691
	unsigned int dirty = sarea_priv->dirty;

692
	DRM_DEBUG("%s %x\n", __func__, dirty);
L
Linus Torvalds 已提交
693 694

	if (dirty & I830_UPLOAD_BUFFERS) {
D
Dave Airlie 已提交
695
		i830EmitDestVerified(dev, sarea_priv->BufferState);
L
Linus Torvalds 已提交
696 697 698 699
		sarea_priv->dirty &= ~I830_UPLOAD_BUFFERS;
	}

	if (dirty & I830_UPLOAD_CTX) {
D
Dave Airlie 已提交
700
		i830EmitContextVerified(dev, sarea_priv->ContextState);
L
Linus Torvalds 已提交
701 702 703 704
		sarea_priv->dirty &= ~I830_UPLOAD_CTX;
	}

	if (dirty & I830_UPLOAD_TEX0) {
D
Dave Airlie 已提交
705
		i830EmitTexVerified(dev, sarea_priv->TexState[0]);
L
Linus Torvalds 已提交
706 707 708 709
		sarea_priv->dirty &= ~I830_UPLOAD_TEX0;
	}

	if (dirty & I830_UPLOAD_TEX1) {
D
Dave Airlie 已提交
710
		i830EmitTexVerified(dev, sarea_priv->TexState[1]);
L
Linus Torvalds 已提交
711 712 713 714
		sarea_priv->dirty &= ~I830_UPLOAD_TEX1;
	}

	if (dirty & I830_UPLOAD_TEXBLEND0) {
D
Dave Airlie 已提交
715 716
		i830EmitTexBlendVerified(dev, sarea_priv->TexBlendState[0],
					 sarea_priv->TexBlendStateWordsUsed[0]);
L
Linus Torvalds 已提交
717 718 719 720
		sarea_priv->dirty &= ~I830_UPLOAD_TEXBLEND0;
	}

	if (dirty & I830_UPLOAD_TEXBLEND1) {
D
Dave Airlie 已提交
721 722
		i830EmitTexBlendVerified(dev, sarea_priv->TexBlendState[1],
					 sarea_priv->TexBlendStateWordsUsed[1]);
L
Linus Torvalds 已提交
723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754
		sarea_priv->dirty &= ~I830_UPLOAD_TEXBLEND1;
	}

	if (dirty & I830_UPLOAD_TEX_PALETTE_SHARED) {
		i830EmitTexPalette(dev, sarea_priv->Palette[0], 0, 1);
	} else {
		if (dirty & I830_UPLOAD_TEX_PALETTE_N(0)) {
			i830EmitTexPalette(dev, sarea_priv->Palette[0], 0, 0);
			sarea_priv->dirty &= ~I830_UPLOAD_TEX_PALETTE_N(0);
		}
		if (dirty & I830_UPLOAD_TEX_PALETTE_N(1)) {
			i830EmitTexPalette(dev, sarea_priv->Palette[1], 1, 0);
			sarea_priv->dirty &= ~I830_UPLOAD_TEX_PALETTE_N(1);
		}

		/* 1.3:
		 */
#if 0
		if (dirty & I830_UPLOAD_TEX_PALETTE_N(2)) {
			i830EmitTexPalette(dev, sarea_priv->Palette2[0], 0, 0);
			sarea_priv->dirty &= ~I830_UPLOAD_TEX_PALETTE_N(2);
		}
		if (dirty & I830_UPLOAD_TEX_PALETTE_N(3)) {
			i830EmitTexPalette(dev, sarea_priv->Palette2[1], 1, 0);
			sarea_priv->dirty &= ~I830_UPLOAD_TEX_PALETTE_N(2);
		}
#endif
	}

	/* 1.3:
	 */
	if (dirty & I830_UPLOAD_STIPPLE) {
D
Dave Airlie 已提交
755
		i830EmitStippleVerified(dev, sarea_priv->StippleState);
L
Linus Torvalds 已提交
756 757 758 759
		sarea_priv->dirty &= ~I830_UPLOAD_STIPPLE;
	}

	if (dirty & I830_UPLOAD_TEX2) {
D
Dave Airlie 已提交
760
		i830EmitTexVerified(dev, sarea_priv->TexState2);
L
Linus Torvalds 已提交
761 762 763 764
		sarea_priv->dirty &= ~I830_UPLOAD_TEX2;
	}

	if (dirty & I830_UPLOAD_TEX3) {
D
Dave Airlie 已提交
765
		i830EmitTexVerified(dev, sarea_priv->TexState3);
L
Linus Torvalds 已提交
766 767 768 769
		sarea_priv->dirty &= ~I830_UPLOAD_TEX3;
	}

	if (dirty & I830_UPLOAD_TEXBLEND2) {
D
Dave Airlie 已提交
770 771 772
		i830EmitTexBlendVerified(dev,
					 sarea_priv->TexBlendState2,
					 sarea_priv->TexBlendStateWordsUsed2);
L
Linus Torvalds 已提交
773 774 775 776 777

		sarea_priv->dirty &= ~I830_UPLOAD_TEXBLEND2;
	}

	if (dirty & I830_UPLOAD_TEXBLEND3) {
D
Dave Airlie 已提交
778 779 780
		i830EmitTexBlendVerified(dev,
					 sarea_priv->TexBlendState3,
					 sarea_priv->TexBlendStateWordsUsed3);
L
Linus Torvalds 已提交
781 782 783 784 785 786 787 788
		sarea_priv->dirty &= ~I830_UPLOAD_TEXBLEND3;
	}
}

/* ================================================================
 * Performance monitoring functions
 */

789
static void i830_fill_box(struct drm_device *dev,
D
Dave Airlie 已提交
790
			  int x, int y, int w, int h, int r, int g, int b)
L
Linus Torvalds 已提交
791
{
D
Dave Airlie 已提交
792
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
793 794 795 796
	u32 color;
	unsigned int BR13, CMD;
	RING_LOCALS;

D
Dave Airlie 已提交
797
	BR13 = (0xF0 << 16) | (dev_priv->pitch * dev_priv->cpp) | (1 << 24);
L
Linus Torvalds 已提交
798 799 800 801 802
	CMD = XY_COLOR_BLT_CMD;
	x += dev_priv->sarea_priv->boxes[0].x1;
	y += dev_priv->sarea_priv->boxes[0].y1;

	if (dev_priv->cpp == 4) {
D
Dave Airlie 已提交
803
		BR13 |= (1 << 25);
L
Linus Torvalds 已提交
804
		CMD |= (XY_COLOR_BLT_WRITE_ALPHA | XY_COLOR_BLT_WRITE_RGB);
D
Dave Airlie 已提交
805
		color = (((0xff) << 24) | (r << 16) | (g << 8) | b);
L
Linus Torvalds 已提交
806 807
	} else {
		color = (((r & 0xf8) << 8) |
D
Dave Airlie 已提交
808
			 ((g & 0xfc) << 3) | ((b & 0xf8) >> 3));
L
Linus Torvalds 已提交
809 810
	}

D
Dave Airlie 已提交
811 812 813 814 815
	BEGIN_LP_RING(6);
	OUT_RING(CMD);
	OUT_RING(BR13);
	OUT_RING((y << 16) | x);
	OUT_RING(((y + h) << 16) | (x + w));
L
Linus Torvalds 已提交
816

817
	if (dev_priv->current_page == 1)
D
Dave Airlie 已提交
818
		OUT_RING(dev_priv->front_offset);
819
	else
D
Dave Airlie 已提交
820
		OUT_RING(dev_priv->back_offset);
L
Linus Torvalds 已提交
821

D
Dave Airlie 已提交
822
	OUT_RING(color);
L
Linus Torvalds 已提交
823 824 825
	ADVANCE_LP_RING();
}

826
static void i830_cp_performance_boxes(struct drm_device *dev)
L
Linus Torvalds 已提交
827
{
D
Dave Airlie 已提交
828
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
829 830 831

	/* Purple box for page flipping
	 */
D
Dave Airlie 已提交
832 833
	if (dev_priv->sarea_priv->perf_boxes & I830_BOX_FLIP)
		i830_fill_box(dev, 4, 4, 8, 8, 255, 0, 255);
L
Linus Torvalds 已提交
834 835 836

	/* Red box if we have to wait for idle at any point
	 */
D
Dave Airlie 已提交
837 838
	if (dev_priv->sarea_priv->perf_boxes & I830_BOX_WAIT)
		i830_fill_box(dev, 16, 4, 8, 8, 255, 0, 0);
L
Linus Torvalds 已提交
839 840 841

	/* Blue box: lost context?
	 */
D
Dave Airlie 已提交
842 843
	if (dev_priv->sarea_priv->perf_boxes & I830_BOX_LOST_CONTEXT)
		i830_fill_box(dev, 28, 4, 8, 8, 0, 0, 255);
L
Linus Torvalds 已提交
844 845 846

	/* Yellow box for texture swaps
	 */
D
Dave Airlie 已提交
847 848
	if (dev_priv->sarea_priv->perf_boxes & I830_BOX_TEXTURE_LOAD)
		i830_fill_box(dev, 40, 4, 8, 8, 255, 255, 0);
L
Linus Torvalds 已提交
849 850 851

	/* Green box if hardware never idles (as far as we can tell)
	 */
D
Dave Airlie 已提交
852 853
	if (!(dev_priv->sarea_priv->perf_boxes & I830_BOX_RING_EMPTY))
		i830_fill_box(dev, 64, 4, 8, 8, 0, 255, 0);
L
Linus Torvalds 已提交
854

D
Dave Airlie 已提交
855
	/* Draw bars indicating number of buffers allocated
L
Linus Torvalds 已提交
856 857 858 859
	 * (not a great measure, easily confused)
	 */
	if (dev_priv->dma_used) {
		int bar = dev_priv->dma_used / 10240;
D
Dave Airlie 已提交
860 861 862 863 864
		if (bar > 100)
			bar = 100;
		if (bar < 1)
			bar = 1;
		i830_fill_box(dev, 4, 16, bar, 4, 196, 128, 128);
L
Linus Torvalds 已提交
865 866 867 868 869 870
		dev_priv->dma_used = 0;
	}

	dev_priv->sarea_priv->perf_boxes = 0;
}

871
static void i830_dma_dispatch_clear(struct drm_device *dev, int flags,
L
Linus Torvalds 已提交
872 873 874 875
				    unsigned int clear_color,
				    unsigned int clear_zval,
				    unsigned int clear_depthmask)
{
D
Dave Airlie 已提交
876 877
	drm_i830_private_t *dev_priv = dev->dev_private;
	drm_i830_sarea_t *sarea_priv = dev_priv->sarea_priv;
L
Linus Torvalds 已提交
878
	int nbox = sarea_priv->nbox;
879
	struct drm_clip_rect *pbox = sarea_priv->boxes;
L
Linus Torvalds 已提交
880 881 882 883 884 885
	int pitch = dev_priv->pitch;
	int cpp = dev_priv->cpp;
	int i;
	unsigned int BR13, CMD, D_CMD;
	RING_LOCALS;

D
Dave Airlie 已提交
886
	if (dev_priv->current_page == 1) {
L
Linus Torvalds 已提交
887 888 889
		unsigned int tmp = flags;

		flags &= ~(I830_FRONT | I830_BACK);
D
Dave Airlie 已提交
890 891 892 893
		if (tmp & I830_FRONT)
			flags |= I830_BACK;
		if (tmp & I830_BACK)
			flags |= I830_FRONT;
L
Linus Torvalds 已提交
894 895
	}

D
Dave Airlie 已提交
896
	i830_kernel_lost_context(dev);
L
Linus Torvalds 已提交
897

D
Dave Airlie 已提交
898 899 900
	switch (cpp) {
	case 2:
		BR13 = (0xF0 << 16) | (pitch * cpp) | (1 << 24);
L
Linus Torvalds 已提交
901 902 903
		D_CMD = CMD = XY_COLOR_BLT_CMD;
		break;
	case 4:
D
Dave Airlie 已提交
904 905
		BR13 = (0xF0 << 16) | (pitch * cpp) | (1 << 24) | (1 << 25);
		CMD = (XY_COLOR_BLT_CMD | XY_COLOR_BLT_WRITE_ALPHA |
L
Linus Torvalds 已提交
906 907
		       XY_COLOR_BLT_WRITE_RGB);
		D_CMD = XY_COLOR_BLT_CMD;
D
Dave Airlie 已提交
908
		if (clear_depthmask & 0x00ffffff)
L
Linus Torvalds 已提交
909
			D_CMD |= XY_COLOR_BLT_WRITE_RGB;
D
Dave Airlie 已提交
910
		if (clear_depthmask & 0xff000000)
L
Linus Torvalds 已提交
911 912 913
			D_CMD |= XY_COLOR_BLT_WRITE_ALPHA;
		break;
	default:
D
Dave Airlie 已提交
914
		BR13 = (0xF0 << 16) | (pitch * cpp) | (1 << 24);
L
Linus Torvalds 已提交
915 916 917 918
		D_CMD = CMD = XY_COLOR_BLT_CMD;
		break;
	}

D
Dave Airlie 已提交
919 920
	if (nbox > I830_NR_SAREA_CLIPRECTS)
		nbox = I830_NR_SAREA_CLIPRECTS;
L
Linus Torvalds 已提交
921

D
Dave Airlie 已提交
922
	for (i = 0; i < nbox; i++, pbox++) {
L
Linus Torvalds 已提交
923 924
		if (pbox->x1 > pbox->x2 ||
		    pbox->y1 > pbox->y2 ||
D
Dave Airlie 已提交
925
		    pbox->x2 > dev_priv->w || pbox->y2 > dev_priv->h)
L
Linus Torvalds 已提交
926 927
			continue;

D
Dave Airlie 已提交
928 929 930 931 932 933 934 935 936
		if (flags & I830_FRONT) {
			DRM_DEBUG("clear front\n");
			BEGIN_LP_RING(6);
			OUT_RING(CMD);
			OUT_RING(BR13);
			OUT_RING((pbox->y1 << 16) | pbox->x1);
			OUT_RING((pbox->y2 << 16) | pbox->x2);
			OUT_RING(dev_priv->front_offset);
			OUT_RING(clear_color);
L
Linus Torvalds 已提交
937 938 939
			ADVANCE_LP_RING();
		}

D
Dave Airlie 已提交
940
		if (flags & I830_BACK) {
L
Linus Torvalds 已提交
941
			DRM_DEBUG("clear back\n");
D
Dave Airlie 已提交
942 943 944 945 946 947 948
			BEGIN_LP_RING(6);
			OUT_RING(CMD);
			OUT_RING(BR13);
			OUT_RING((pbox->y1 << 16) | pbox->x1);
			OUT_RING((pbox->y2 << 16) | pbox->x2);
			OUT_RING(dev_priv->back_offset);
			OUT_RING(clear_color);
L
Linus Torvalds 已提交
949 950 951
			ADVANCE_LP_RING();
		}

D
Dave Airlie 已提交
952
		if (flags & I830_DEPTH) {
L
Linus Torvalds 已提交
953
			DRM_DEBUG("clear depth\n");
D
Dave Airlie 已提交
954 955 956 957 958 959 960
			BEGIN_LP_RING(6);
			OUT_RING(D_CMD);
			OUT_RING(BR13);
			OUT_RING((pbox->y1 << 16) | pbox->x1);
			OUT_RING((pbox->y2 << 16) | pbox->x2);
			OUT_RING(dev_priv->depth_offset);
			OUT_RING(clear_zval);
L
Linus Torvalds 已提交
961 962 963 964 965
			ADVANCE_LP_RING();
		}
	}
}

966
static void i830_dma_dispatch_swap(struct drm_device *dev)
L
Linus Torvalds 已提交
967
{
D
Dave Airlie 已提交
968 969
	drm_i830_private_t *dev_priv = dev->dev_private;
	drm_i830_sarea_t *sarea_priv = dev_priv->sarea_priv;
L
Linus Torvalds 已提交
970
	int nbox = sarea_priv->nbox;
971
	struct drm_clip_rect *pbox = sarea_priv->boxes;
L
Linus Torvalds 已提交
972 973 974 975 976 977 978 979
	int pitch = dev_priv->pitch;
	int cpp = dev_priv->cpp;
	int i;
	unsigned int CMD, BR13;
	RING_LOCALS;

	DRM_DEBUG("swapbuffers\n");

D
Dave Airlie 已提交
980
	i830_kernel_lost_context(dev);
L
Linus Torvalds 已提交
981 982

	if (dev_priv->do_boxes)
D
Dave Airlie 已提交
983
		i830_cp_performance_boxes(dev);
L
Linus Torvalds 已提交
984

D
Dave Airlie 已提交
985 986 987
	switch (cpp) {
	case 2:
		BR13 = (pitch * cpp) | (0xCC << 16) | (1 << 24);
L
Linus Torvalds 已提交
988 989 990
		CMD = XY_SRC_COPY_BLT_CMD;
		break;
	case 4:
D
Dave Airlie 已提交
991
		BR13 = (pitch * cpp) | (0xCC << 16) | (1 << 24) | (1 << 25);
L
Linus Torvalds 已提交
992 993 994 995
		CMD = (XY_SRC_COPY_BLT_CMD | XY_SRC_COPY_BLT_WRITE_ALPHA |
		       XY_SRC_COPY_BLT_WRITE_RGB);
		break;
	default:
D
Dave Airlie 已提交
996
		BR13 = (pitch * cpp) | (0xCC << 16) | (1 << 24);
L
Linus Torvalds 已提交
997 998 999 1000
		CMD = XY_SRC_COPY_BLT_CMD;
		break;
	}

D
Dave Airlie 已提交
1001 1002
	if (nbox > I830_NR_SAREA_CLIPRECTS)
		nbox = I830_NR_SAREA_CLIPRECTS;
L
Linus Torvalds 已提交
1003

D
Dave Airlie 已提交
1004
	for (i = 0; i < nbox; i++, pbox++) {
L
Linus Torvalds 已提交
1005 1006
		if (pbox->x1 > pbox->x2 ||
		    pbox->y1 > pbox->y2 ||
D
Dave Airlie 已提交
1007
		    pbox->x2 > dev_priv->w || pbox->y2 > dev_priv->h)
L
Linus Torvalds 已提交
1008
			continue;
D
Dave Airlie 已提交
1009

L
Linus Torvalds 已提交
1010
		DRM_DEBUG("dispatch swap %d,%d-%d,%d!\n",
D
Dave Airlie 已提交
1011
			  pbox->x1, pbox->y1, pbox->x2, pbox->y2);
L
Linus Torvalds 已提交
1012

D
Dave Airlie 已提交
1013 1014 1015 1016 1017
		BEGIN_LP_RING(8);
		OUT_RING(CMD);
		OUT_RING(BR13);
		OUT_RING((pbox->y1 << 16) | pbox->x1);
		OUT_RING((pbox->y2 << 16) | pbox->x2);
L
Linus Torvalds 已提交
1018

D
Dave Airlie 已提交
1019 1020
		if (dev_priv->current_page == 0)
			OUT_RING(dev_priv->front_offset);
L
Linus Torvalds 已提交
1021
		else
D
Dave Airlie 已提交
1022
			OUT_RING(dev_priv->back_offset);
L
Linus Torvalds 已提交
1023

D
Dave Airlie 已提交
1024 1025
		OUT_RING((pbox->y1 << 16) | pbox->x1);
		OUT_RING(BR13 & 0xffff);
L
Linus Torvalds 已提交
1026

D
Dave Airlie 已提交
1027 1028
		if (dev_priv->current_page == 0)
			OUT_RING(dev_priv->back_offset);
L
Linus Torvalds 已提交
1029
		else
D
Dave Airlie 已提交
1030
			OUT_RING(dev_priv->front_offset);
L
Linus Torvalds 已提交
1031 1032 1033 1034 1035

		ADVANCE_LP_RING();
	}
}

1036
static void i830_dma_dispatch_flip(struct drm_device *dev)
L
Linus Torvalds 已提交
1037
{
D
Dave Airlie 已提交
1038
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
1039 1040
	RING_LOCALS;

D
Dave Airlie 已提交
1041
	DRM_DEBUG("%s: page=%d pfCurrentPage=%d\n",
1042
		  __func__,
D
Dave Airlie 已提交
1043 1044
		  dev_priv->current_page,
		  dev_priv->sarea_priv->pf_current_page);
L
Linus Torvalds 已提交
1045

D
Dave Airlie 已提交
1046
	i830_kernel_lost_context(dev);
L
Linus Torvalds 已提交
1047 1048 1049

	if (dev_priv->do_boxes) {
		dev_priv->sarea_priv->perf_boxes |= I830_BOX_FLIP;
D
Dave Airlie 已提交
1050
		i830_cp_performance_boxes(dev);
L
Linus Torvalds 已提交
1051 1052
	}

D
Dave Airlie 已提交
1053 1054 1055
	BEGIN_LP_RING(2);
	OUT_RING(INST_PARSER_CLIENT | INST_OP_FLUSH | INST_FLUSH_MAP_CACHE);
	OUT_RING(0);
L
Linus Torvalds 已提交
1056 1057
	ADVANCE_LP_RING();

D
Dave Airlie 已提交
1058 1059 1060 1061 1062
	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);
L
Linus Torvalds 已提交
1063 1064
		dev_priv->current_page = 1;
	} else {
D
Dave Airlie 已提交
1065
		OUT_RING(dev_priv->front_offset);
L
Linus Torvalds 已提交
1066 1067 1068 1069 1070
		dev_priv->current_page = 0;
	}
	OUT_RING(0);
	ADVANCE_LP_RING();

D
Dave Airlie 已提交
1071 1072 1073
	BEGIN_LP_RING(2);
	OUT_RING(MI_WAIT_FOR_EVENT | MI_WAIT_FOR_PLANE_A_FLIP);
	OUT_RING(0);
L
Linus Torvalds 已提交
1074 1075 1076 1077 1078
	ADVANCE_LP_RING();

	dev_priv->sarea_priv->pf_current_page = dev_priv->current_page;
}

1079 1080
static void i830_dma_dispatch_vertex(struct drm_device *dev,
				     struct drm_buf *buf, int discard, int used)
L
Linus Torvalds 已提交
1081
{
D
Dave Airlie 已提交
1082
	drm_i830_private_t *dev_priv = dev->dev_private;
L
Linus Torvalds 已提交
1083
	drm_i830_buf_priv_t *buf_priv = buf->dev_private;
D
Dave Airlie 已提交
1084
	drm_i830_sarea_t *sarea_priv = dev_priv->sarea_priv;
1085
	struct drm_clip_rect *box = sarea_priv->boxes;
D
Dave Airlie 已提交
1086
	int nbox = sarea_priv->nbox;
L
Linus Torvalds 已提交
1087
	unsigned long address = (unsigned long)buf->bus_address;
D
Dave Airlie 已提交
1088
	unsigned long start = address - dev->agp->base;
L
Linus Torvalds 已提交
1089
	int i = 0, u;
D
Dave Airlie 已提交
1090
	RING_LOCALS;
L
Linus Torvalds 已提交
1091

D
Dave Airlie 已提交
1092
	i830_kernel_lost_context(dev);
L
Linus Torvalds 已提交
1093

D
Dave Airlie 已提交
1094
	if (nbox > I830_NR_SAREA_CLIPRECTS)
L
Linus Torvalds 已提交
1095 1096 1097
		nbox = I830_NR_SAREA_CLIPRECTS;

	if (discard) {
D
Dave Airlie 已提交
1098
		u = cmpxchg(buf_priv->in_use, I830_BUF_CLIENT,
L
Linus Torvalds 已提交
1099
			    I830_BUF_HARDWARE);
1100
		if (u != I830_BUF_CLIENT)
L
Linus Torvalds 已提交
1101 1102 1103
			DRM_DEBUG("xxxx 2\n");
	}

D
Dave Airlie 已提交
1104
	if (used > 4 * 1023)
L
Linus Torvalds 已提交
1105 1106 1107
		used = 0;

	if (sarea_priv->dirty)
D
Dave Airlie 已提交
1108
		i830EmitState(dev);
L
Linus Torvalds 已提交
1109

D
Dave Airlie 已提交
1110
	DRM_DEBUG("dispatch vertex addr 0x%lx, used 0x%x nbox %d\n",
L
Linus Torvalds 已提交
1111 1112
		  address, used, nbox);

D
Dave Airlie 已提交
1113 1114 1115 1116 1117 1118
	dev_priv->counter++;
	DRM_DEBUG("dispatch counter : %ld\n", dev_priv->counter);
	DRM_DEBUG("i830_dma_dispatch\n");
	DRM_DEBUG("start : %lx\n", start);
	DRM_DEBUG("used : %d\n", used);
	DRM_DEBUG("start + used - 4 : %ld\n", start + used - 4);
L
Linus Torvalds 已提交
1119 1120 1121 1122 1123

	if (buf_priv->currently_mapped == I830_BUF_MAPPED) {
		u32 *vp = buf_priv->kernel_virtual;

		vp[0] = (GFX_OP_PRIMITIVE |
D
Dave Airlie 已提交
1124
			 sarea_priv->vertex_prim | ((used / 4) - 2));
L
Linus Torvalds 已提交
1125 1126

		if (dev_priv->use_mi_batchbuffer_start) {
D
Dave Airlie 已提交
1127 1128
			vp[used / 4] = MI_BATCH_BUFFER_END;
			used += 4;
L
Linus Torvalds 已提交
1129
		}
D
Dave Airlie 已提交
1130

L
Linus Torvalds 已提交
1131
		if (used & 4) {
D
Dave Airlie 已提交
1132
			vp[used / 4] = 0;
L
Linus Torvalds 已提交
1133 1134 1135 1136 1137
			used += 4;
		}

		i830_unmap_buffer(buf);
	}
D
Dave Airlie 已提交
1138

L
Linus Torvalds 已提交
1139 1140 1141 1142
	if (used) {
		do {
			if (i < nbox) {
				BEGIN_LP_RING(6);
D
Dave Airlie 已提交
1143 1144 1145 1146 1147 1148 1149 1150
				OUT_RING(GFX_OP_DRAWRECT_INFO);
				OUT_RING(sarea_priv->
					 BufferState[I830_DESTREG_DR1]);
				OUT_RING(box[i].x1 | (box[i].y1 << 16));
				OUT_RING(box[i].x2 | (box[i].y2 << 16));
				OUT_RING(sarea_priv->
					 BufferState[I830_DESTREG_DR4]);
				OUT_RING(0);
L
Linus Torvalds 已提交
1151 1152 1153 1154 1155
				ADVANCE_LP_RING();
			}

			if (dev_priv->use_mi_batchbuffer_start) {
				BEGIN_LP_RING(2);
D
Dave Airlie 已提交
1156 1157
				OUT_RING(MI_BATCH_BUFFER_START | (2 << 6));
				OUT_RING(start | MI_BATCH_NON_SECURE);
L
Linus Torvalds 已提交
1158
				ADVANCE_LP_RING();
D
Dave Airlie 已提交
1159
			} else {
L
Linus Torvalds 已提交
1160
				BEGIN_LP_RING(4);
D
Dave Airlie 已提交
1161 1162 1163 1164
				OUT_RING(MI_BATCH_BUFFER);
				OUT_RING(start | MI_BATCH_NON_SECURE);
				OUT_RING(start + used - 4);
				OUT_RING(0);
L
Linus Torvalds 已提交
1165 1166 1167 1168 1169 1170 1171 1172 1173
				ADVANCE_LP_RING();
			}

		} while (++i < nbox);
	}

	if (discard) {
		dev_priv->counter++;

D
Dave Airlie 已提交
1174 1175
		(void)cmpxchg(buf_priv->in_use, I830_BUF_CLIENT,
			      I830_BUF_HARDWARE);
L
Linus Torvalds 已提交
1176 1177

		BEGIN_LP_RING(8);
D
Dave Airlie 已提交
1178 1179 1180 1181 1182 1183 1184 1185
		OUT_RING(CMD_STORE_DWORD_IDX);
		OUT_RING(20);
		OUT_RING(dev_priv->counter);
		OUT_RING(CMD_STORE_DWORD_IDX);
		OUT_RING(buf_priv->my_use_idx);
		OUT_RING(I830_BUF_FREE);
		OUT_RING(CMD_REPORT_HEAD);
		OUT_RING(0);
L
Linus Torvalds 已提交
1186 1187 1188 1189
		ADVANCE_LP_RING();
	}
}

1190
static void i830_dma_quiescent(struct drm_device *dev)
L
Linus Torvalds 已提交
1191
{
D
Dave Airlie 已提交
1192 1193
	drm_i830_private_t *dev_priv = dev->dev_private;
	RING_LOCALS;
L
Linus Torvalds 已提交
1194

D
Dave Airlie 已提交
1195
	i830_kernel_lost_context(dev);
L
Linus Torvalds 已提交
1196

D
Dave Airlie 已提交
1197 1198 1199 1200 1201 1202
	BEGIN_LP_RING(4);
	OUT_RING(INST_PARSER_CLIENT | INST_OP_FLUSH | INST_FLUSH_MAP_CACHE);
	OUT_RING(CMD_REPORT_HEAD);
	OUT_RING(0);
	OUT_RING(0);
	ADVANCE_LP_RING();
L
Linus Torvalds 已提交
1203

1204
	i830_wait_ring(dev, dev_priv->ring.Size - 8, __func__);
L
Linus Torvalds 已提交
1205 1206
}

1207
static int i830_flush_queue(struct drm_device *dev)
L
Linus Torvalds 已提交
1208
{
D
Dave Airlie 已提交
1209
	drm_i830_private_t *dev_priv = dev->dev_private;
1210
	struct drm_device_dma *dma = dev->dma;
D
Dave Airlie 已提交
1211 1212 1213 1214 1215 1216 1217 1218 1219 1220
	int i, ret = 0;
	RING_LOCALS;

	i830_kernel_lost_context(dev);

	BEGIN_LP_RING(2);
	OUT_RING(CMD_REPORT_HEAD);
	OUT_RING(0);
	ADVANCE_LP_RING();

1221
	i830_wait_ring(dev, dev_priv->ring.Size - 8, __func__);
D
Dave Airlie 已提交
1222 1223

	for (i = 0; i < dma->buf_count; i++) {
D
Dave Airlie 已提交
1224
		struct drm_buf *buf = dma->buflist[i];
D
Dave Airlie 已提交
1225 1226 1227
		drm_i830_buf_priv_t *buf_priv = buf->dev_private;

		int used = cmpxchg(buf_priv->in_use, I830_BUF_HARDWARE,
L
Linus Torvalds 已提交
1228 1229 1230 1231 1232 1233 1234 1235
				   I830_BUF_FREE);

		if (used == I830_BUF_HARDWARE)
			DRM_DEBUG("reclaimed from HARDWARE\n");
		if (used == I830_BUF_CLIENT)
			DRM_DEBUG("still on client\n");
	}

D
Dave Airlie 已提交
1236
	return ret;
L
Linus Torvalds 已提交
1237 1238 1239
}

/* Must be called with the lock held */
1240
static void i830_reclaim_buffers(struct drm_device *dev, struct drm_file *file_priv)
L
Linus Torvalds 已提交
1241
{
1242
	struct drm_device_dma *dma = dev->dma;
D
Dave Airlie 已提交
1243
	int i;
L
Linus Torvalds 已提交
1244

D
Dave Airlie 已提交
1245 1246 1247 1248 1249 1250
	if (!dma)
		return;
	if (!dev->dev_private)
		return;
	if (!dma->buflist)
		return;
L
Linus Torvalds 已提交
1251

D
Dave Airlie 已提交
1252
	i830_flush_queue(dev);
L
Linus Torvalds 已提交
1253 1254

	for (i = 0; i < dma->buf_count; i++) {
D
Dave Airlie 已提交
1255
		struct drm_buf *buf = dma->buflist[i];
D
Dave Airlie 已提交
1256 1257
		drm_i830_buf_priv_t *buf_priv = buf->dev_private;

1258
		if (buf->file_priv == file_priv && buf_priv) {
D
Dave Airlie 已提交
1259
			int used = cmpxchg(buf_priv->in_use, I830_BUF_CLIENT,
L
Linus Torvalds 已提交
1260 1261 1262 1263
					   I830_BUF_FREE);

			if (used == I830_BUF_CLIENT)
				DRM_DEBUG("reclaimed from client\n");
D
Dave Airlie 已提交
1264 1265
			if (buf_priv->currently_mapped == I830_BUF_MAPPED)
				buf_priv->currently_mapped = I830_BUF_UNMAPPED;
L
Linus Torvalds 已提交
1266 1267 1268 1269
		}
	}
}

1270 1271
static int i830_flush_ioctl(struct drm_device *dev, void *data,
			    struct drm_file *file_priv)
L
Linus Torvalds 已提交
1272
{
1273
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
1274

D
Dave Airlie 已提交
1275 1276
	i830_flush_queue(dev);
	return 0;
L
Linus Torvalds 已提交
1277 1278
}

1279 1280
static int i830_dma_vertex(struct drm_device *dev, void *data,
			   struct drm_file *file_priv)
L
Linus Torvalds 已提交
1281
{
1282
	struct drm_device_dma *dma = dev->dma;
D
Dave Airlie 已提交
1283 1284 1285 1286
	drm_i830_private_t *dev_priv = (drm_i830_private_t *) dev->dev_private;
	u32 *hw_status = dev_priv->hw_status_page;
	drm_i830_sarea_t *sarea_priv = (drm_i830_sarea_t *)
	    dev_priv->sarea_priv;
1287
	drm_i830_vertex_t *vertex = data;
L
Linus Torvalds 已提交
1288

1289
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
1290 1291

	DRM_DEBUG("i830 dma vertex, idx %d used %d discard %d\n",
1292
		  vertex->idx, vertex->used, vertex->discard);
L
Linus Torvalds 已提交
1293

1294
	if (vertex->idx < 0 || vertex->idx > dma->buf_count)
D
Dave Airlie 已提交
1295
		return -EINVAL;
L
Linus Torvalds 已提交
1296

D
Dave Airlie 已提交
1297
	i830_dma_dispatch_vertex(dev,
1298 1299
				 dma->buflist[vertex->idx],
				 vertex->discard, vertex->used);
D
Dave Airlie 已提交
1300 1301 1302

	sarea_priv->last_enqueue = dev_priv->counter - 1;
	sarea_priv->last_dispatch = (int)hw_status[5];
L
Linus Torvalds 已提交
1303 1304 1305 1306

	return 0;
}

1307 1308
static int i830_clear_bufs(struct drm_device *dev, void *data,
			   struct drm_file *file_priv)
L
Linus Torvalds 已提交
1309
{
1310
	drm_i830_clear_t *clear = data;
D
Dave Airlie 已提交
1311

1312
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
1313 1314

	/* GH: Someone's doing nasty things... */
1315
	if (!dev->dev_private)
L
Linus Torvalds 已提交
1316 1317
		return -EINVAL;

1318 1319 1320
	i830_dma_dispatch_clear(dev, clear->flags,
				clear->clear_color,
				clear->clear_depth, clear->clear_depthmask);
D
Dave Airlie 已提交
1321
	return 0;
L
Linus Torvalds 已提交
1322 1323
}

1324 1325
static int i830_swap_bufs(struct drm_device *dev, void *data,
			  struct drm_file *file_priv)
L
Linus Torvalds 已提交
1326 1327 1328
{
	DRM_DEBUG("i830_swap_bufs\n");

1329
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
1330

D
Dave Airlie 已提交
1331 1332
	i830_dma_dispatch_swap(dev);
	return 0;
L
Linus Torvalds 已提交
1333 1334 1335
}

/* Not sure why this isn't set all the time:
D
Dave Airlie 已提交
1336
 */
1337
static void i830_do_init_pageflip(struct drm_device *dev)
L
Linus Torvalds 已提交
1338 1339 1340
{
	drm_i830_private_t *dev_priv = dev->dev_private;

1341
	DRM_DEBUG("%s\n", __func__);
L
Linus Torvalds 已提交
1342 1343 1344 1345 1346
	dev_priv->page_flipping = 1;
	dev_priv->current_page = 0;
	dev_priv->sarea_priv->pf_current_page = dev_priv->current_page;
}

1347
static int i830_do_cleanup_pageflip(struct drm_device *dev)
L
Linus Torvalds 已提交
1348 1349 1350
{
	drm_i830_private_t *dev_priv = dev->dev_private;

1351
	DRM_DEBUG("%s\n", __func__);
L
Linus Torvalds 已提交
1352
	if (dev_priv->current_page != 0)
D
Dave Airlie 已提交
1353
		i830_dma_dispatch_flip(dev);
L
Linus Torvalds 已提交
1354 1355 1356 1357 1358

	dev_priv->page_flipping = 0;
	return 0;
}

1359 1360
static int i830_flip_bufs(struct drm_device *dev, void *data,
			  struct drm_file *file_priv)
L
Linus Torvalds 已提交
1361 1362 1363
{
	drm_i830_private_t *dev_priv = dev->dev_private;

1364
	DRM_DEBUG("%s\n", __func__);
L
Linus Torvalds 已提交
1365

1366
	LOCK_TEST_WITH_RETURN(dev, file_priv);
L
Linus Torvalds 已提交
1367

D
Dave Airlie 已提交
1368 1369
	if (!dev_priv->page_flipping)
		i830_do_init_pageflip(dev);
L
Linus Torvalds 已提交
1370

D
Dave Airlie 已提交
1371 1372
	i830_dma_dispatch_flip(dev);
	return 0;
L
Linus Torvalds 已提交
1373 1374
}

1375 1376
static int i830_getage(struct drm_device *dev, void *data,
		       struct drm_file *file_priv)
L
Linus Torvalds 已提交
1377
{
D
Dave Airlie 已提交
1378 1379 1380 1381 1382 1383
	drm_i830_private_t *dev_priv = (drm_i830_private_t *) dev->dev_private;
	u32 *hw_status = dev_priv->hw_status_page;
	drm_i830_sarea_t *sarea_priv = (drm_i830_sarea_t *)
	    dev_priv->sarea_priv;

	sarea_priv->last_dispatch = (int)hw_status[5];
L
Linus Torvalds 已提交
1384 1385 1386
	return 0;
}

1387 1388
static int i830_getbuf(struct drm_device *dev, void *data,
		       struct drm_file *file_priv)
L
Linus Torvalds 已提交
1389
{
D
Dave Airlie 已提交
1390
	int retcode = 0;
1391
	drm_i830_dma_t *d = data;
D
Dave Airlie 已提交
1392 1393 1394 1395
	drm_i830_private_t *dev_priv = (drm_i830_private_t *) dev->dev_private;
	u32 *hw_status = dev_priv->hw_status_page;
	drm_i830_sarea_t *sarea_priv = (drm_i830_sarea_t *)
	    dev_priv->sarea_priv;
L
Linus Torvalds 已提交
1396 1397

	DRM_DEBUG("getbuf\n");
D
Dave Airlie 已提交
1398

1399
	LOCK_TEST_WITH_RETURN(dev, file_priv);
D
Dave Airlie 已提交
1400

1401
	d->granted = 0;
L
Linus Torvalds 已提交
1402

1403
	retcode = i830_dma_get_buffer(dev, d, file_priv);
L
Linus Torvalds 已提交
1404 1405

	DRM_DEBUG("i830_dma: %d returning %d, granted = %d\n",
1406
		  task_pid_nr(current), retcode, d->granted);
L
Linus Torvalds 已提交
1407

D
Dave Airlie 已提交
1408
	sarea_priv->last_dispatch = (int)hw_status[5];
L
Linus Torvalds 已提交
1409 1410 1411 1412

	return retcode;
}

1413 1414
static int i830_copybuf(struct drm_device *dev, void *data,
			struct drm_file *file_priv)
L
Linus Torvalds 已提交
1415 1416 1417 1418 1419
{
	/* Never copy - 2.4.x doesn't need it */
	return 0;
}

1420 1421
static int i830_docopy(struct drm_device *dev, void *data,
		       struct drm_file *file_priv)
L
Linus Torvalds 已提交
1422 1423 1424 1425
{
	return 0;
}

1426 1427
static int i830_getparam(struct drm_device *dev, void *data,
			 struct drm_file *file_priv)
L
Linus Torvalds 已提交
1428 1429
{
	drm_i830_private_t *dev_priv = dev->dev_private;
1430
	drm_i830_getparam_t *param = data;
L
Linus Torvalds 已提交
1431 1432
	int value;

D
Dave Airlie 已提交
1433
	if (!dev_priv) {
1434
		DRM_ERROR("%s called with no initialization\n", __func__);
L
Linus Torvalds 已提交
1435 1436 1437
		return -EINVAL;
	}

1438
	switch (param->param) {
L
Linus Torvalds 已提交
1439 1440 1441 1442 1443 1444 1445
	case I830_PARAM_IRQ_ACTIVE:
		value = dev->irq_enabled;
		break;
	default:
		return -EINVAL;
	}

1446
	if (copy_to_user(param->value, &value, sizeof(int))) {
D
Dave Airlie 已提交
1447
		DRM_ERROR("copy_to_user\n");
L
Linus Torvalds 已提交
1448 1449
		return -EFAULT;
	}
D
Dave Airlie 已提交
1450

L
Linus Torvalds 已提交
1451 1452 1453
	return 0;
}

1454 1455
static int i830_setparam(struct drm_device *dev, void *data,
			 struct drm_file *file_priv)
L
Linus Torvalds 已提交
1456 1457
{
	drm_i830_private_t *dev_priv = dev->dev_private;
1458
	drm_i830_setparam_t *param = data;
L
Linus Torvalds 已提交
1459

D
Dave Airlie 已提交
1460
	if (!dev_priv) {
1461
		DRM_ERROR("%s called with no initialization\n", __func__);
L
Linus Torvalds 已提交
1462 1463 1464
		return -EINVAL;
	}

1465
	switch (param->param) {
L
Linus Torvalds 已提交
1466
	case I830_SETPARAM_USE_MI_BATCHBUFFER_START:
1467
		dev_priv->use_mi_batchbuffer_start = param->value;
L
Linus Torvalds 已提交
1468 1469 1470 1471 1472 1473 1474 1475
		break;
	default:
		return -EINVAL;
	}

	return 0;
}

1476
int i830_driver_load(struct drm_device *dev, unsigned long flags)
1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487
{
	/* i830 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;

	return 0;
}

1488
void i830_driver_lastclose(struct drm_device *dev)
L
Linus Torvalds 已提交
1489
{
D
Dave Airlie 已提交
1490
	i830_dma_cleanup(dev);
L
Linus Torvalds 已提交
1491 1492
}

1493
void i830_driver_preclose(struct drm_device *dev, struct drm_file *file_priv)
L
Linus Torvalds 已提交
1494 1495 1496
{
	if (dev->dev_private) {
		drm_i830_private_t *dev_priv = dev->dev_private;
1497
		if (dev_priv->page_flipping)
L
Linus Torvalds 已提交
1498 1499 1500 1501
			i830_do_cleanup_pageflip(dev);
	}
}

1502
void i830_driver_reclaim_buffers_locked(struct drm_device *dev, struct drm_file *file_priv)
L
Linus Torvalds 已提交
1503
{
1504
	i830_reclaim_buffers(dev, file_priv);
L
Linus Torvalds 已提交
1505 1506
}

1507
int i830_driver_dma_quiescent(struct drm_device *dev)
L
Linus Torvalds 已提交
1508
{
D
Dave Airlie 已提交
1509
	i830_dma_quiescent(dev);
L
Linus Torvalds 已提交
1510 1511 1512
	return 0;
}

A
Arnd Bergmann 已提交
1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525
/*
 * call the drm_ioctl under the big kernel lock because
 * to lock against the i830_mmap_buffers function.
 */
long i830_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
{
	int ret;
	lock_kernel();
	ret = drm_ioctl(file, cmd, arg);
	unlock_kernel();
	return ret;
}

1526
struct drm_ioctl_desc i830_ioctls[] = {
1527 1528 1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540
	DRM_IOCTL_DEF_DRV(I830_INIT, i830_dma_init, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_VERTEX, i830_dma_vertex, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_CLEAR, i830_clear_bufs, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_FLUSH, i830_flush_ioctl, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_GETAGE, i830_getage, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_GETBUF, i830_getbuf, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_SWAP, i830_swap_bufs, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_COPY, i830_copybuf, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_DOCOPY, i830_docopy, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_FLIP, i830_flip_bufs, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_IRQ_EMIT, i830_irq_emit, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_IRQ_WAIT, i830_irq_wait, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_GETPARAM, i830_getparam, DRM_AUTH|DRM_UNLOCKED),
	DRM_IOCTL_DEF_DRV(I830_SETPARAM, i830_setparam, DRM_AUTH|DRM_UNLOCKED),
L
Linus Torvalds 已提交
1541 1542 1543
};

int i830_max_ioctl = DRM_ARRAY_SIZE(i830_ioctls);
1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554 1555

/**
 * 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 i8xx is AGP.
 */
1556
int i830_driver_device_is_agp(struct drm_device *dev)
1557 1558 1559
{
	return 1;
}