2004-06-10 06:45:38 -06:00
|
|
|
/* i915_dma.c -- DMA support for the I915 -*- linux-c -*-
|
|
|
|
*/
|
2005-11-28 16:10:41 -07:00
|
|
|
/*
|
2004-06-10 06:45:38 -06:00
|
|
|
* Copyright 2003 Tungsten Graphics, Inc., Cedar Park, Texas.
|
|
|
|
* All Rights Reserved.
|
2007-11-04 19:42:22 -07:00
|
|
|
*
|
2005-06-06 03:18:44 -06:00
|
|
|
* 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:
|
2007-11-04 19:42:22 -07:00
|
|
|
*
|
2005-06-06 03:18:44 -06:00
|
|
|
* The above copyright notice and this permission notice (including the
|
|
|
|
* next paragraph) shall be included in all copies or substantial portions
|
|
|
|
* of the Software.
|
2007-11-04 19:42:22 -07:00
|
|
|
*
|
2005-06-06 03:18:44 -06:00
|
|
|
* 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.
|
2007-11-04 19:42:22 -07:00
|
|
|
*
|
2005-11-28 16:10:41 -07:00
|
|
|
*/
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
#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.
|
|
|
|
*/
|
2007-07-15 20:32:51 -06:00
|
|
|
int i915_wait_ring(struct drm_device * dev, int n, const char *caller)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
drm_i915_ring_buffer_t *ring = &(dev_priv->ring);
|
2004-06-10 06:45:38 -06:00
|
|
|
u32 last_head = I915_READ(LP_RING + RING_HEAD) & HEAD_ADDR;
|
|
|
|
int i;
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
for (i = 0; i < 10000; i++) {
|
2004-06-10 06:45:38 -06:00
|
|
|
ring->head = I915_READ(LP_RING + RING_HEAD) & HEAD_ADDR;
|
2004-08-27 03:14:30 -06:00
|
|
|
ring->space = ring->head - (ring->tail + 8);
|
|
|
|
if (ring->space < 0)
|
|
|
|
ring->space += ring->Size;
|
|
|
|
if (ring->space >= n)
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
2004-08-27 03:14:30 -06:00
|
|
|
|
|
|
|
if (ring->head != last_head)
|
2004-06-10 06:45:38 -06:00
|
|
|
i = 0;
|
|
|
|
|
|
|
|
last_head = ring->head;
|
2007-01-12 12:24:14 -07:00
|
|
|
DRM_UDELAY(1);
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EBUSY;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
void i915_kernel_lost_context(struct drm_device * dev)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
drm_i915_ring_buffer_t *ring = &(dev_priv->ring);
|
|
|
|
|
|
|
|
ring->head = I915_READ(LP_RING + RING_HEAD) & HEAD_ADDR;
|
|
|
|
ring->tail = I915_READ(LP_RING + RING_TAIL) & TAIL_ADDR;
|
|
|
|
ring->space = ring->head - (ring->tail + 8);
|
|
|
|
if (ring->space < 0)
|
|
|
|
ring->space += ring->Size;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
static int i915_dma_cleanup(struct drm_device * dev)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2007-10-26 17:10:02 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2004-06-10 06:45:38 -06:00
|
|
|
/* 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.
|
|
|
|
*/
|
2004-08-27 03:14:30 -06:00
|
|
|
if (dev->irq)
|
2004-09-30 15:12:10 -06:00
|
|
|
drm_irq_uninstall(dev);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-10-26 17:10:02 -06:00
|
|
|
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;
|
|
|
|
}
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-10-26 17:10:02 -06:00
|
|
|
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 */
|
|
|
|
I915_WRITE(0x02080, 0x1ffff000);
|
|
|
|
}
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-10-26 17:10:02 -06:00
|
|
|
if (dev_priv->status_gfx_addr) {
|
|
|
|
dev_priv->status_gfx_addr = 0;
|
|
|
|
drm_core_ioremapfree(&dev_priv->hws_map, dev);
|
|
|
|
I915_WRITE(0x02080, 0x1ffff000);
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
return 0;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-10-26 17:10:02 -06:00
|
|
|
static int i915_initialize(struct drm_device * dev, drm_i915_init_t * init)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2007-10-26 17:10:02 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-04-27 22:49:27 -06:00
|
|
|
dev_priv->sarea = drm_getsarea(dev);
|
2004-08-27 03:14:30 -06:00
|
|
|
if (!dev_priv->sarea) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("can not find sarea!\n");
|
2004-07-29 05:09:22 -06:00
|
|
|
i915_dma_cleanup(dev);
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
|
2004-08-17 07:10:05 -06:00
|
|
|
dev_priv->mmio_map = drm_core_findmap(dev, init->mmio_offset);
|
2004-08-27 03:14:30 -06:00
|
|
|
if (!dev_priv->mmio_map) {
|
2004-07-29 05:09:22 -06:00
|
|
|
i915_dma_cleanup(dev);
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("can not find mmio map!\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
#ifdef I915_HAVE_BUFFER
|
|
|
|
dev_priv->max_validate_buffers = I915_MAX_VALIDATE_BUFFERS;
|
|
|
|
#endif
|
|
|
|
|
2008-02-05 10:25:22 -07:00
|
|
|
if (init->sarea_priv_offset)
|
|
|
|
dev_priv->sarea_priv = (drm_i915_sarea_t *)
|
|
|
|
((u8 *) dev_priv->sarea->handle +
|
|
|
|
init->sarea_priv_offset);
|
|
|
|
else {
|
|
|
|
/* No sarea_priv for you! */
|
|
|
|
dev_priv->sarea_priv = NULL;
|
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2004-06-10 06:45:38 -06:00
|
|
|
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;
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_core_ioremap(&dev_priv->ring.map, dev);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (dev_priv->ring.map.handle == NULL) {
|
|
|
|
i915_dma_cleanup(dev);
|
|
|
|
DRM_ERROR("can not ioremap virtual address for"
|
2004-06-10 06:45:38 -06:00
|
|
|
" ring buffer\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -ENOMEM;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
dev_priv->ring.virtual_start = dev_priv->ring.map.handle;
|
|
|
|
|
2006-08-25 11:01:05 -06:00
|
|
|
dev_priv->cpp = init->cpp;
|
2008-02-05 10:25:22 -07:00
|
|
|
|
|
|
|
if (dev_priv->sarea_priv)
|
|
|
|
dev_priv->sarea_priv->pf_current_page = 0;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
/* We are using separate values as placeholders for mechanisms for
|
|
|
|
* private backbuffer/depthbuffer usage.
|
|
|
|
*/
|
|
|
|
dev_priv->use_mi_batchbuffer_start = 0;
|
2007-11-28 16:37:51 -07:00
|
|
|
if (IS_I965G(dev)) /* 965 doesn't support older method */
|
2007-11-28 05:46:06 -07:00
|
|
|
dev_priv->use_mi_batchbuffer_start = 1;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
/* Allow hardware batchbuffers unless told otherwise.
|
|
|
|
*/
|
|
|
|
dev_priv->allow_batchbuffer = 1;
|
|
|
|
|
2007-05-07 07:07:48 -06:00
|
|
|
/* Enable vblank on pipe A for older X servers
|
|
|
|
*/
|
2007-11-04 19:42:22 -07:00
|
|
|
dev_priv->vblank_pipe = DRM_I915_VBLANK_PIPE_A;
|
2007-05-07 07:07:48 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
/* Program Hardware Status Page */
|
2007-06-05 12:15:29 -06:00
|
|
|
if (!IS_G33(dev)) {
|
2007-11-04 19:42:22 -07:00
|
|
|
dev_priv->status_page_dmah =
|
2007-06-05 12:15:29 -06:00
|
|
|
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");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -ENOMEM;
|
2007-06-05 12:15:29 -06:00
|
|
|
}
|
|
|
|
dev_priv->hw_status_page = dev_priv->status_page_dmah->vaddr;
|
|
|
|
dev_priv->dma_status_page = dev_priv->status_page_dmah->busaddr;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-06-05 12:15:29 -06:00
|
|
|
memset(dev_priv->hw_status_page, 0, PAGE_SIZE);
|
2004-08-27 03:14:30 -06:00
|
|
|
|
2007-06-05 12:15:29 -06:00
|
|
|
I915_WRITE(0x02080, dev_priv->dma_status_page);
|
|
|
|
}
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_DEBUG("Enabled hardware status page\n");
|
2007-11-13 15:50:46 -07:00
|
|
|
#ifdef I915_HAVE_BUFFER
|
2007-10-22 10:59:37 -06:00
|
|
|
mutex_init(&dev_priv->cmdbuf_mutex);
|
2007-11-13 15:50:46 -07:00
|
|
|
#endif
|
2004-08-27 03:14:30 -06:00
|
|
|
return 0;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
static int i915_dma_resume(struct drm_device * dev)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-11-23 23:56:05 -07:00
|
|
|
DRM_DEBUG("\n");
|
2004-08-27 03:14:30 -06:00
|
|
|
|
|
|
|
if (!dev_priv->sarea) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("can not find sarea!\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (!dev_priv->mmio_map) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("can not find mmio map!\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (dev_priv->ring.map.handle == NULL) {
|
|
|
|
DRM_ERROR("can not ioremap virtual address for"
|
2004-06-10 06:45:38 -06:00
|
|
|
" ring buffer\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -ENOMEM;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
/* Program Hardware Status Page */
|
|
|
|
if (!dev_priv->hw_status_page) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("Can not find hardware status page\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
DRM_DEBUG("hw status page @ %p\n", dev_priv->hw_status_page);
|
2004-08-27 03:14:30 -06:00
|
|
|
|
2007-06-05 12:15:29 -06:00
|
|
|
if (dev_priv->status_gfx_addr != 0)
|
|
|
|
I915_WRITE(0x02080, dev_priv->status_gfx_addr);
|
|
|
|
else
|
|
|
|
I915_WRITE(0x02080, dev_priv->dma_status_page);
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_DEBUG("Enabled hardware status page\n");
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
return 0;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_dma_init(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_init_t *init = data;
|
2004-08-27 03:14:30 -06:00
|
|
|
int retcode = 0;
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
switch (init->func) {
|
2004-08-27 03:14:30 -06:00
|
|
|
case I915_INIT_DMA:
|
2007-10-26 17:10:02 -06:00
|
|
|
retcode = i915_initialize(dev, init);
|
2004-08-27 03:14:30 -06:00
|
|
|
break;
|
|
|
|
case I915_CLEANUP_DMA:
|
|
|
|
retcode = i915_dma_cleanup(dev);
|
|
|
|
break;
|
|
|
|
case I915_RESUME_DMA:
|
2005-01-06 10:51:32 -07:00
|
|
|
retcode = i915_dma_resume(dev);
|
2004-08-27 03:14:30 -06:00
|
|
|
break;
|
|
|
|
default:
|
2007-07-19 18:00:17 -06:00
|
|
|
retcode = -EINVAL;
|
2004-08-27 03:14:30 -06:00
|
|
|
break;
|
|
|
|
}
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
return retcode;
|
|
|
|
}
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
/* 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.
|
|
|
|
*/
|
2004-08-27 03:14:30 -06:00
|
|
|
static int do_validate_cmd(int cmd)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
switch (((cmd >> 29) & 0x7)) {
|
2004-06-10 06:45:38 -06:00
|
|
|
case 0x0:
|
2004-08-27 03:14:30 -06:00
|
|
|
switch ((cmd >> 23) & 0x3f) {
|
|
|
|
case 0x0:
|
|
|
|
return 1; /* MI_NOOP */
|
|
|
|
case 0x4:
|
|
|
|
return 1; /* MI_FLUSH */
|
|
|
|
default:
|
|
|
|
return 0; /* disallow everything else */
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
break;
|
2004-08-27 03:14:30 -06:00
|
|
|
case 0x1:
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0; /* reserved */
|
2004-08-27 03:14:30 -06:00
|
|
|
case 0x2:
|
|
|
|
return (cmd & 0xff) + 2; /* 2d commands */
|
2004-06-10 06:45:38 -06:00
|
|
|
case 0x3:
|
2004-08-27 03:14:30 -06:00
|
|
|
if (((cmd >> 24) & 0x1f) <= 0x18)
|
2004-06-10 06:45:38 -06:00
|
|
|
return 1;
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
switch ((cmd >> 24) & 0x1f) {
|
|
|
|
case 0x1c:
|
2004-06-10 06:45:38 -06:00
|
|
|
return 1;
|
|
|
|
case 0x1d:
|
2004-08-27 03:14:30 -06:00
|
|
|
switch ((cmd >> 16) & 0xff) {
|
|
|
|
case 0x3:
|
2004-07-23 10:12:27 -06:00
|
|
|
return (cmd & 0x1f) + 2;
|
2004-08-27 03:14:30 -06:00
|
|
|
case 0x4:
|
2004-07-23 10:12:27 -06:00
|
|
|
return (cmd & 0xf) + 2;
|
2004-08-27 03:14:30 -06:00
|
|
|
default:
|
2004-07-23 10:12:27 -06:00
|
|
|
return (cmd & 0xffff) + 2;
|
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
case 0x1e:
|
|
|
|
if (cmd & (1 << 23))
|
2004-06-10 06:45:38 -06:00
|
|
|
return (cmd & 0xffff) + 1;
|
|
|
|
else
|
|
|
|
return 1;
|
|
|
|
case 0x1f:
|
2004-08-27 03:14:30 -06:00
|
|
|
if ((cmd & (1 << 23)) == 0) /* inline vertices */
|
2004-06-10 06:45:38 -06:00
|
|
|
return (cmd & 0x1ffff) + 2;
|
2004-08-27 03:14:30 -06:00
|
|
|
else if (cmd & (1 << 17)) /* indirect random */
|
2004-06-10 06:45:38 -06:00
|
|
|
if ((cmd & 0xffff) == 0)
|
2004-08-27 03:14:30 -06:00
|
|
|
return 0; /* unknown length, too hard */
|
2004-06-10 06:45:38 -06:00
|
|
|
else
|
|
|
|
return (((cmd & 0xffff) + 1) / 2) + 1;
|
|
|
|
else
|
2004-08-27 03:14:30 -06:00
|
|
|
return 2; /* indirect sequential */
|
|
|
|
default:
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
static int validate_cmd(int cmd)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
int ret = do_validate_cmd(cmd);
|
|
|
|
|
2007-11-04 19:42:22 -07:00
|
|
|
/* printk("validate_cmd( %x ): %d\n", cmd, ret); */
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-11-28 16:37:51 -07:00
|
|
|
static int i915_emit_cmds(struct drm_device *dev, int __user *buffer,
|
2007-07-19 18:11:11 -06:00
|
|
|
int dwords)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2004-06-10 06:45:38 -06:00
|
|
|
int i;
|
|
|
|
RING_LOCALS;
|
|
|
|
|
2006-01-23 03:05:22 -07:00
|
|
|
if ((dwords+1) * sizeof(int) >= dev_priv->ring.Size - 8)
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2006-01-23 03:05:22 -07:00
|
|
|
|
2006-08-08 16:05:54 -06:00
|
|
|
BEGIN_LP_RING((dwords+1)&~1);
|
2006-01-23 03:05:22 -07:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
for (i = 0; i < dwords;) {
|
2004-06-10 06:45:38 -06:00
|
|
|
int cmd, sz;
|
|
|
|
|
2006-12-19 03:48:18 -07:00
|
|
|
if (DRM_COPY_FROM_USER_UNCHECKED(&cmd, &buffer[i], sizeof(cmd)))
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2006-12-19 03:48:18 -07:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if ((sz = validate_cmd(cmd)) == 0 || i + sz > dwords)
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
OUT_RING(cmd);
|
|
|
|
|
|
|
|
while (++i, --sz) {
|
2004-08-27 03:14:30 -06:00
|
|
|
if (DRM_COPY_FROM_USER_UNCHECKED(&cmd, &buffer[i],
|
|
|
|
sizeof(cmd))) {
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
OUT_RING(cmd);
|
|
|
|
}
|
|
|
|
}
|
2007-11-04 19:42:22 -07:00
|
|
|
|
2006-01-23 03:05:22 -07:00
|
|
|
if (dwords & 1)
|
|
|
|
OUT_RING(0);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2006-01-23 03:05:22 -07:00
|
|
|
ADVANCE_LP_RING();
|
2007-11-04 19:42:22 -07:00
|
|
|
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
static int i915_emit_box(struct drm_device * dev,
|
2007-07-15 19:22:15 -06:00
|
|
|
struct drm_clip_rect __user * boxes,
|
2004-08-27 03:14:30 -06:00
|
|
|
int i, int DR1, int DR4)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-07-15 19:22:15 -06:00
|
|
|
struct drm_clip_rect box;
|
2004-08-27 03:14:30 -06:00
|
|
|
RING_LOCALS;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (DRM_COPY_FROM_USER_UNCHECKED(&box, &boxes[i], sizeof(box))) {
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EFAULT;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (box.y2 <= box.y1 || box.x2 <= box.x1 || box.y2 <= 0 || box.x2 <= 0) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("Bad box %d,%d..%d,%d\n",
|
|
|
|
box.x1, box.y1, box.x2, box.y2);
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2006-08-09 22:38:50 -06:00
|
|
|
if (IS_I965G(dev)) {
|
|
|
|
BEGIN_LP_RING(4);
|
|
|
|
OUT_RING(GFX_OP_DRAWRECT_INFO_I965);
|
|
|
|
OUT_RING((box.x1 & 0xffff) | (box.y1 << 16));
|
|
|
|
OUT_RING(((box.x2 - 1) & 0xffff) | ((box.y2 - 1) << 16));
|
|
|
|
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();
|
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-08-08 16:05:54 -06:00
|
|
|
/* XXX: Emitting the counter should really be moved to part of the IRQ
|
2007-11-21 23:10:36 -07:00
|
|
|
* emit. For now, do it in both places:
|
2006-08-08 16:05:54 -06:00
|
|
|
*/
|
2006-01-23 03:05:22 -07:00
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
void i915_emit_breadcrumb(struct drm_device *dev)
|
2006-01-23 03:05:22 -07:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
RING_LOCALS;
|
|
|
|
|
2007-06-15 09:13:11 -06:00
|
|
|
if (++dev_priv->counter > BREADCRUMB_MASK) {
|
|
|
|
dev_priv->counter = 1;
|
|
|
|
DRM_DEBUG("Breadcrumb counter wrapped around\n");
|
|
|
|
}
|
2006-08-08 16:05:54 -06:00
|
|
|
|
2008-02-05 10:25:22 -07:00
|
|
|
if (dev_priv->sarea_priv)
|
|
|
|
dev_priv->sarea_priv->last_enqueue = dev_priv->counter;
|
2007-02-02 09:23:42 -07:00
|
|
|
|
2006-01-23 03:05:22 -07:00
|
|
|
BEGIN_LP_RING(4);
|
|
|
|
OUT_RING(CMD_STORE_DWORD_IDX);
|
|
|
|
OUT_RING(20);
|
|
|
|
OUT_RING(dev_priv->counter);
|
|
|
|
OUT_RING(0);
|
|
|
|
ADVANCE_LP_RING();
|
|
|
|
}
|
|
|
|
|
2006-08-31 13:42:29 -06:00
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
int i915_emit_mi_flush(struct drm_device *dev, uint32_t flush)
|
2006-08-31 13:42:29 -06:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
uint32_t flush_cmd = CMD_MI_FLUSH;
|
|
|
|
RING_LOCALS;
|
|
|
|
|
|
|
|
flush_cmd |= flush;
|
|
|
|
|
|
|
|
i915_kernel_lost_context(dev);
|
|
|
|
|
|
|
|
BEGIN_LP_RING(4);
|
|
|
|
OUT_RING(flush_cmd);
|
|
|
|
OUT_RING(0);
|
|
|
|
OUT_RING(0);
|
|
|
|
OUT_RING(0);
|
|
|
|
ADVANCE_LP_RING();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
static int i915_dispatch_cmdbuffer(struct drm_device * dev,
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_cmdbuffer_t * cmd)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2007-05-15 14:35:33 -06:00
|
|
|
#ifdef I915_HAVE_FENCE
|
2007-02-02 09:23:42 -07:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-05-15 14:35:33 -06:00
|
|
|
#endif
|
2004-08-27 03:14:30 -06:00
|
|
|
int nbox = cmd->num_cliprects;
|
2004-06-10 06:45:38 -06:00
|
|
|
int i = 0, count, ret;
|
|
|
|
|
|
|
|
if (cmd->sz & 0x3) {
|
2007-10-24 17:27:46 -06:00
|
|
|
DRM_ERROR("alignment\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
|
|
|
|
i915_kernel_lost_context(dev);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
count = nbox ? nbox : 1;
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
for (i = 0; i < count; i++) {
|
2004-06-10 06:45:38 -06:00
|
|
|
if (i < nbox) {
|
2004-08-27 03:14:30 -06:00
|
|
|
ret = i915_emit_box(dev, cmd->cliprects, i,
|
|
|
|
cmd->DR1, cmd->DR4);
|
|
|
|
if (ret)
|
2004-06-10 06:45:38 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
ret = i915_emit_cmds(dev, (int __user *)cmd->buf, cmd->sz / 4);
|
|
|
|
if (ret)
|
2004-06-10 06:45:38 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-11-28 16:37:51 -07:00
|
|
|
i915_emit_breadcrumb(dev);
|
2007-02-02 09:23:42 -07:00
|
|
|
#ifdef I915_HAVE_FENCE
|
2008-01-30 14:14:02 -07:00
|
|
|
if (unlikely((dev_priv->counter & 0xFF) == 0))
|
|
|
|
drm_fence_flush_old(dev, 0, dev_priv->counter);
|
2007-02-02 09:23:42 -07:00
|
|
|
#endif
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
static int i915_dispatch_batchbuffer(struct drm_device * dev,
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_batchbuffer_t * batch)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-07-15 19:22:15 -06:00
|
|
|
struct drm_clip_rect __user *boxes = batch->cliprects;
|
2004-08-27 03:14:30 -06:00
|
|
|
int nbox = batch->num_cliprects;
|
2004-06-10 06:45:38 -06:00
|
|
|
int i = 0, count;
|
2004-08-27 03:14:30 -06:00
|
|
|
RING_LOCALS;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
if ((batch->start | batch->used) & 0x7) {
|
2007-10-24 17:27:46 -06:00
|
|
|
DRM_ERROR("alignment\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
|
|
|
|
i915_kernel_lost_context(dev);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
count = nbox ? nbox : 1;
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
for (i = 0; i < count; i++) {
|
2004-06-10 06:45:38 -06:00
|
|
|
if (i < nbox) {
|
2004-08-27 03:14:30 -06:00
|
|
|
int ret = i915_emit_box(dev, boxes, i,
|
|
|
|
batch->DR1, batch->DR4);
|
|
|
|
if (ret)
|
2004-06-10 06:45:38 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dev_priv->use_mi_batchbuffer_start) {
|
|
|
|
BEGIN_LP_RING(2);
|
2007-08-06 02:33:29 -06:00
|
|
|
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);
|
|
|
|
}
|
2004-06-10 06:45:38 -06:00
|
|
|
ADVANCE_LP_RING();
|
2007-08-06 02:33:29 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
} else {
|
2004-06-10 06:45:38 -06:00
|
|
|
BEGIN_LP_RING(4);
|
2004-08-27 03:14:30 -06:00
|
|
|
OUT_RING(MI_BATCH_BUFFER);
|
|
|
|
OUT_RING(batch->start | MI_BATCH_NON_SECURE);
|
|
|
|
OUT_RING(batch->start + batch->used - 4);
|
|
|
|
OUT_RING(0);
|
2004-06-10 06:45:38 -06:00
|
|
|
ADVANCE_LP_RING();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-28 16:37:51 -07:00
|
|
|
i915_emit_breadcrumb(dev);
|
2007-02-02 09:23:42 -07:00
|
|
|
#ifdef I915_HAVE_FENCE
|
2008-01-30 14:14:02 -07:00
|
|
|
if (unlikely((dev_priv->counter & 0xFF) == 0))
|
|
|
|
drm_fence_flush_old(dev, 0, dev_priv->counter);
|
2007-02-02 09:23:42 -07:00
|
|
|
#endif
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
static void i915_do_dispatch_flip(struct drm_device * dev, int plane, int sync)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-02-19 04:27:54 -07:00
|
|
|
u32 num_pages, current_page, next_page, dspbase;
|
2007-09-11 04:48:46 -06:00
|
|
|
int shift = 2 * plane, x, y;
|
2004-06-10 06:45:38 -06:00
|
|
|
RING_LOCALS;
|
|
|
|
|
2007-02-19 04:27:54 -07:00
|
|
|
/* Calculate display base offset */
|
|
|
|
num_pages = dev_priv->sarea_priv->third_handle ? 3 : 2;
|
2007-02-28 09:48:56 -07:00
|
|
|
current_page = (dev_priv->sarea_priv->pf_current_page >> shift) & 0x3;
|
2007-02-19 04:27:54 -07:00
|
|
|
next_page = (current_page + 1) % num_pages;
|
|
|
|
|
|
|
|
switch (next_page) {
|
|
|
|
default:
|
|
|
|
case 0:
|
|
|
|
dspbase = dev_priv->sarea_priv->front_offset;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
dspbase = dev_priv->sarea_priv->back_offset;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
dspbase = dev_priv->sarea_priv->third_offset;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
if (plane == 0) {
|
|
|
|
x = dev_priv->sarea_priv->planeA_x;
|
|
|
|
y = dev_priv->sarea_priv->planeA_y;
|
2007-02-19 04:27:54 -07:00
|
|
|
} else {
|
2007-09-11 04:48:46 -06:00
|
|
|
x = dev_priv->sarea_priv->planeB_x;
|
|
|
|
y = dev_priv->sarea_priv->planeB_y;
|
2007-02-19 04:27:54 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
dspbase += (y * dev_priv->sarea_priv->pitch + x) * dev_priv->cpp;
|
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
DRM_DEBUG("plane=%d current_page=%d dspbase=0x%x\n", plane, current_page,
|
2007-02-19 04:27:54 -07:00
|
|
|
dspbase);
|
|
|
|
|
|
|
|
BEGIN_LP_RING(4);
|
2007-03-09 15:34:11 -07:00
|
|
|
OUT_RING(sync ? 0 :
|
2007-09-11 04:48:46 -06:00
|
|
|
(MI_WAIT_FOR_EVENT | (plane ? MI_WAIT_FOR_PLANE_B_FLIP :
|
2007-03-09 15:34:11 -07:00
|
|
|
MI_WAIT_FOR_PLANE_A_FLIP)));
|
2007-02-22 09:21:18 -07:00
|
|
|
OUT_RING(CMD_OP_DISPLAYBUFFER_INFO | (sync ? 0 : ASYNC_FLIP) |
|
2007-09-11 04:48:46 -06:00
|
|
|
(plane ? DISPLAY_PLANE_B : DISPLAY_PLANE_A));
|
2007-02-22 09:21:18 -07:00
|
|
|
OUT_RING(dev_priv->sarea_priv->pitch * dev_priv->cpp);
|
2007-02-19 04:27:54 -07:00
|
|
|
OUT_RING(dspbase);
|
|
|
|
ADVANCE_LP_RING();
|
|
|
|
|
2007-02-28 09:48:56 -07:00
|
|
|
dev_priv->sarea_priv->pf_current_page &= ~(0x3 << shift);
|
|
|
|
dev_priv->sarea_priv->pf_current_page |= next_page << shift;
|
2007-02-19 04:27:54 -07:00
|
|
|
}
|
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
void i915_dispatch_flip(struct drm_device * dev, int planes, int sync)
|
2007-02-19 04:27:54 -07:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
int i;
|
|
|
|
|
2007-11-23 23:56:05 -07:00
|
|
|
DRM_DEBUG("planes=0x%x pfCurrentPage=%d\n",
|
2007-09-11 04:48:46 -06:00
|
|
|
planes, dev_priv->sarea_priv->pf_current_page);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-02-28 07:23:19 -07:00
|
|
|
i915_emit_mi_flush(dev, MI_READ_FLUSH | MI_EXE_FLUSH);
|
2007-02-19 04:27:54 -07:00
|
|
|
|
|
|
|
for (i = 0; i < 2; i++)
|
2007-09-11 04:48:46 -06:00
|
|
|
if (planes & (1 << i))
|
2007-02-22 09:21:18 -07:00
|
|
|
i915_do_dispatch_flip(dev, i, sync);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-02-02 09:23:42 -07:00
|
|
|
i915_emit_breadcrumb(dev);
|
2006-08-21 13:36:00 -06:00
|
|
|
#ifdef I915_HAVE_FENCE
|
2008-01-30 14:14:02 -07:00
|
|
|
if (unlikely(!sync && ((dev_priv->counter & 0xFF) == 0)))
|
2007-02-22 09:21:18 -07:00
|
|
|
drm_fence_flush_old(dev, 0, dev_priv->counter);
|
2006-08-21 13:36:00 -06:00
|
|
|
#endif
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-11-28 16:37:51 -07:00
|
|
|
static int i915_quiescent(struct drm_device *dev)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
i915_kernel_lost_context(dev);
|
|
|
|
return i915_wait_ring(dev, dev_priv->ring.Size - 8, __FUNCTION__);
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_flush_ioctl(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
|
|
|
|
2007-07-20 07:39:25 -06:00
|
|
|
LOCK_TEST_WITH_RETURN(dev, file_priv);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
return i915_quiescent(dev);
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_batchbuffer(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
|
|
|
|
drm_i915_sarea_t *sarea_priv = (drm_i915_sarea_t *)
|
|
|
|
dev_priv->sarea_priv;
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_batchbuffer_t *batch = data;
|
2004-06-10 06:45:38 -06:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!dev_priv->allow_batchbuffer) {
|
|
|
|
DRM_ERROR("Batchbuffer ioctl disabled\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
DRM_DEBUG("i915 batchbuffer, start %x used %d cliprects %d\n",
|
2007-07-19 18:11:11 -06:00
|
|
|
batch->start, batch->used, batch->num_cliprects);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-07-20 07:39:25 -06:00
|
|
|
LOCK_TEST_WITH_RETURN(dev, file_priv);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
if (batch->num_cliprects && DRM_VERIFYAREA_READ(batch->cliprects,
|
|
|
|
batch->num_cliprects *
|
|
|
|
sizeof(struct drm_clip_rect)))
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EFAULT;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
ret = i915_dispatch_batchbuffer(dev, batch);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-04-01 00:30:52 -06:00
|
|
|
sarea_priv->last_dispatch = READ_BREADCRUMB(dev_priv);
|
2004-06-10 06:45:38 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_cmdbuffer(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2004-08-27 03:14:30 -06:00
|
|
|
drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
|
|
|
|
drm_i915_sarea_t *sarea_priv = (drm_i915_sarea_t *)
|
|
|
|
dev_priv->sarea_priv;
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_cmdbuffer_t *cmdbuf = data;
|
2004-06-10 06:45:38 -06:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
DRM_DEBUG("i915 cmdbuffer, buf %p sz %d cliprects %d\n",
|
2007-07-19 18:11:11 -06:00
|
|
|
cmdbuf->buf, cmdbuf->sz, cmdbuf->num_cliprects);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-07-20 07:39:25 -06:00
|
|
|
LOCK_TEST_WITH_RETURN(dev, file_priv);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
if (cmdbuf->num_cliprects &&
|
|
|
|
DRM_VERIFYAREA_READ(cmdbuf->cliprects,
|
|
|
|
cmdbuf->num_cliprects *
|
2007-07-15 19:22:15 -06:00
|
|
|
sizeof(struct drm_clip_rect))) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("Fault accessing cliprects\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EFAULT;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
ret = i915_dispatch_cmdbuffer(dev, cmdbuf);
|
2004-06-10 06:45:38 -06:00
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("i915_dispatch_cmdbuffer failed\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-04-01 00:30:52 -06:00
|
|
|
sarea_priv->last_dispatch = READ_BREADCRUMB(dev_priv);
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-12-11 21:21:23 -07:00
|
|
|
#if DRM_DEBUG_CODE
|
|
|
|
#define DRM_DEBUG_RELOCATION (drm_debug != 0)
|
|
|
|
#else
|
|
|
|
#define DRM_DEBUG_RELOCATION 0
|
|
|
|
#endif
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
#ifdef I915_HAVE_BUFFER
|
2007-12-11 21:21:23 -07:00
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
struct i915_relocatee_info {
|
|
|
|
struct drm_buffer_object *buf;
|
|
|
|
unsigned long offset;
|
|
|
|
u32 *data_page;
|
|
|
|
unsigned page_offset;
|
|
|
|
struct drm_bo_kmap_obj kmap;
|
|
|
|
int is_iomem;
|
|
|
|
};
|
|
|
|
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
struct drm_i915_validate_buffer {
|
|
|
|
struct drm_buffer_object *buffer;
|
|
|
|
int presumed_offset_correct;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void i915_dereference_buffers_locked(struct drm_i915_validate_buffer *buffers,
|
2007-10-11 18:54:38 -06:00
|
|
|
unsigned num_buffers)
|
|
|
|
{
|
|
|
|
while (num_buffers--)
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
drm_bo_usage_deref_locked(&buffers[num_buffers].buffer);
|
2007-10-11 18:54:38 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
int i915_apply_reloc(struct drm_file *file_priv, int num_buffers,
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
struct drm_i915_validate_buffer *buffers,
|
2007-10-11 18:54:38 -06:00
|
|
|
struct i915_relocatee_info *relocatee,
|
|
|
|
uint32_t *reloc)
|
|
|
|
{
|
|
|
|
unsigned index;
|
|
|
|
unsigned long new_cmd_offset;
|
|
|
|
u32 val;
|
2008-01-23 21:37:40 -07:00
|
|
|
int ret, i;
|
|
|
|
int buf_index = -1;
|
|
|
|
|
|
|
|
for (i = 0; i <= num_buffers; i++)
|
|
|
|
if (buffers[i].buffer)
|
|
|
|
if (reloc[2] == buffers[i].buffer->base.hash.key)
|
|
|
|
buf_index = i;
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
if (buf_index == -1) {
|
2007-10-11 18:54:38 -06:00
|
|
|
DRM_ERROR("Illegal relocation buffer %08X\n", reloc[2]);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
/*
|
|
|
|
* Short-circuit relocations that were correctly
|
|
|
|
* guessed by the client
|
|
|
|
*/
|
2008-01-23 21:37:40 -07:00
|
|
|
if (buffers[buf_index].presumed_offset_correct && !DRM_DEBUG_RELOCATION)
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
return 0;
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
new_cmd_offset = reloc[0];
|
|
|
|
if (!relocatee->data_page ||
|
|
|
|
!drm_bo_same_page(relocatee->offset, new_cmd_offset)) {
|
|
|
|
drm_bo_kunmap(&relocatee->kmap);
|
|
|
|
relocatee->offset = new_cmd_offset;
|
2007-12-06 16:12:21 -07:00
|
|
|
mutex_lock (&relocatee->buf->mutex);
|
|
|
|
ret = drm_bo_wait (relocatee->buf, 0, 0, FALSE);
|
|
|
|
mutex_unlock (&relocatee->buf->mutex);
|
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("Could not wait for buffer to apply relocs\n %08lx", new_cmd_offset);
|
|
|
|
return ret;
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
ret = drm_bo_kmap(relocatee->buf, new_cmd_offset >> PAGE_SHIFT,
|
|
|
|
1, &relocatee->kmap);
|
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("Could not map command buffer to apply relocs\n %08lx", new_cmd_offset);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
relocatee->data_page = drm_bmo_virtual(&relocatee->kmap,
|
|
|
|
&relocatee->is_iomem);
|
|
|
|
relocatee->page_offset = (relocatee->offset & PAGE_MASK);
|
|
|
|
}
|
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
val = buffers[buf_index].buffer->offset;
|
2007-10-11 18:54:38 -06:00
|
|
|
index = (reloc[0] - relocatee->page_offset) >> 2;
|
|
|
|
|
|
|
|
/* add in validate */
|
|
|
|
val = val + reloc[1];
|
|
|
|
|
2007-12-11 21:21:23 -07:00
|
|
|
if (DRM_DEBUG_RELOCATION) {
|
2008-01-23 21:37:40 -07:00
|
|
|
if (buffers[buf_index].presumed_offset_correct &&
|
2007-12-11 21:21:23 -07:00
|
|
|
relocatee->data_page[index] != val) {
|
|
|
|
DRM_DEBUG ("Relocation mismatch source %d target %d buffer %d user %08x kernel %08x\n",
|
2008-01-23 21:37:40 -07:00
|
|
|
reloc[0], reloc[1], buf_index, relocatee->data_page[index], val);
|
2007-12-11 21:21:23 -07:00
|
|
|
}
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
relocatee->data_page[index] = val;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int i915_process_relocs(struct drm_file *file_priv,
|
|
|
|
uint32_t buf_handle,
|
2008-01-23 21:37:40 -07:00
|
|
|
uint32_t __user **reloc_user_ptr,
|
2007-10-11 18:54:38 -06:00
|
|
|
struct i915_relocatee_info *relocatee,
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
struct drm_i915_validate_buffer *buffers,
|
2007-10-11 18:54:38 -06:00
|
|
|
uint32_t num_buffers)
|
|
|
|
{
|
2008-01-23 21:37:40 -07:00
|
|
|
int ret, reloc_stride;
|
|
|
|
uint32_t cur_offset;
|
|
|
|
uint32_t reloc_count;
|
|
|
|
uint32_t reloc_type;
|
|
|
|
uint32_t reloc_buf_size;
|
|
|
|
uint32_t *reloc_buf = NULL;
|
|
|
|
int i;
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
/* do a copy from user from the user ptr */
|
|
|
|
ret = get_user(reloc_count, *reloc_user_ptr);
|
2007-10-11 18:54:38 -06:00
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("Could not map relocation buffer.\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
ret = get_user(reloc_type, (*reloc_user_ptr)+1);
|
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("Could not map relocation buffer.\n");
|
|
|
|
goto out;
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
if (reloc_type != 0) {
|
2007-10-11 18:54:38 -06:00
|
|
|
DRM_ERROR("Unsupported relocation type requested\n");
|
2008-01-23 21:37:40 -07:00
|
|
|
ret = -EINVAL;
|
2007-10-11 18:54:38 -06:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2008-01-23 22:18:09 -07:00
|
|
|
reloc_buf_size = (I915_RELOC_HEADER + (reloc_count * I915_RELOC0_STRIDE)) * sizeof(uint32_t);
|
2008-01-23 21:37:40 -07:00
|
|
|
reloc_buf = kmalloc(reloc_buf_size, GFP_KERNEL);
|
|
|
|
if (!reloc_buf) {
|
|
|
|
DRM_ERROR("Out of memory for reloc buffer\n");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto out;
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
if (copy_from_user(reloc_buf, *reloc_user_ptr, reloc_buf_size)) {
|
|
|
|
ret = -EFAULT;
|
|
|
|
goto out;
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
/* get next relocate buffer handle */
|
|
|
|
*reloc_user_ptr = (uint32_t *)*(unsigned long *)&reloc_buf[2];
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
reloc_stride = I915_RELOC0_STRIDE * sizeof(uint32_t); /* may be different for other types of relocs */
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
DRM_DEBUG("num relocs is %d, next is %p\n", reloc_count, *reloc_user_ptr);
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
for (i = 0; i < reloc_count; i++) {
|
|
|
|
cur_offset = I915_RELOC_HEADER + (i * I915_RELOC0_STRIDE);
|
|
|
|
|
|
|
|
ret = i915_apply_reloc(file_priv, num_buffers, buffers,
|
|
|
|
relocatee, reloc_buf + cur_offset);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
out:
|
2008-01-23 21:37:40 -07:00
|
|
|
|
|
|
|
if (reloc_buf)
|
|
|
|
kfree(reloc_buf);
|
2007-10-25 00:52:33 -06:00
|
|
|
drm_bo_kunmap(&relocatee->kmap);
|
|
|
|
relocatee->data_page = NULL;
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-10-21 18:54:41 -06:00
|
|
|
static int i915_exec_reloc(struct drm_file *file_priv, drm_handle_t buf_handle,
|
2008-01-23 21:37:40 -07:00
|
|
|
uint32_t __user *reloc_user_ptr,
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
struct drm_i915_validate_buffer *buffers,
|
2007-10-21 18:54:41 -06:00
|
|
|
uint32_t buf_count)
|
|
|
|
{
|
|
|
|
struct drm_device *dev = file_priv->head->dev;
|
|
|
|
struct i915_relocatee_info relocatee;
|
|
|
|
int ret = 0;
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
int b;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Short circuit relocations when all previous
|
|
|
|
* buffers offsets were correctly guessed by
|
|
|
|
* the client
|
|
|
|
*/
|
2007-12-11 21:21:23 -07:00
|
|
|
if (!DRM_DEBUG_RELOCATION) {
|
|
|
|
for (b = 0; b < buf_count; b++)
|
|
|
|
if (!buffers[b].presumed_offset_correct)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (b == buf_count)
|
|
|
|
return 0;
|
|
|
|
}
|
2007-10-21 18:54:41 -06:00
|
|
|
|
|
|
|
memset(&relocatee, 0, sizeof(relocatee));
|
2007-11-04 19:42:22 -07:00
|
|
|
|
2007-10-21 18:54:41 -06:00
|
|
|
mutex_lock(&dev->struct_mutex);
|
|
|
|
relocatee.buf = drm_lookup_buffer_object(file_priv, buf_handle, 1);
|
|
|
|
mutex_unlock(&dev->struct_mutex);
|
|
|
|
if (!relocatee.buf) {
|
|
|
|
DRM_DEBUG("relocatee buffer invalid %08x\n", buf_handle);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out_err;
|
|
|
|
}
|
2007-11-04 19:42:22 -07:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
while (reloc_user_ptr) {
|
|
|
|
ret = i915_process_relocs(file_priv, buf_handle, &reloc_user_ptr, &relocatee, buffers, buf_count);
|
2007-10-21 18:54:41 -06:00
|
|
|
if (ret) {
|
|
|
|
DRM_ERROR("process relocs failed\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-11-04 19:42:22 -07:00
|
|
|
|
2007-10-21 18:54:41 -06:00
|
|
|
mutex_lock(&dev->struct_mutex);
|
|
|
|
drm_bo_usage_deref_locked(&relocatee.buf);
|
|
|
|
mutex_unlock(&dev->struct_mutex);
|
2007-11-04 19:42:22 -07:00
|
|
|
|
2007-10-21 18:54:41 -06:00
|
|
|
out_err:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
/*
|
|
|
|
* Validate, add fence and relocate a block of bos from a userspace list
|
|
|
|
*/
|
|
|
|
int i915_validate_buffer_list(struct drm_file *file_priv,
|
|
|
|
unsigned int fence_class, uint64_t data,
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
struct drm_i915_validate_buffer *buffers,
|
2007-10-11 18:54:38 -06:00
|
|
|
uint32_t *num_buffers)
|
|
|
|
{
|
|
|
|
struct drm_i915_op_arg arg;
|
|
|
|
struct drm_bo_op_req *req = &arg.d.req;
|
|
|
|
struct drm_bo_arg_rep rep;
|
|
|
|
unsigned long next = 0;
|
|
|
|
int ret = 0;
|
|
|
|
unsigned buf_count = 0;
|
|
|
|
struct drm_device *dev = file_priv->head->dev;
|
2008-01-23 21:37:40 -07:00
|
|
|
uint32_t buf_handle;
|
|
|
|
uint32_t __user *reloc_user_ptr;
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
do {
|
|
|
|
if (buf_count >= *num_buffers) {
|
|
|
|
DRM_ERROR("Buffer count exceeded %d\n.", *num_buffers);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
buffers[buf_count].buffer = NULL;
|
|
|
|
buffers[buf_count].presumed_offset_correct = 0;
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2007-11-28 16:55:38 -07:00
|
|
|
if (copy_from_user(&arg, (void __user *)(unsigned long)data, sizeof(arg))) {
|
2007-10-11 18:54:38 -06:00
|
|
|
ret = -EFAULT;
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (arg.handled) {
|
|
|
|
data = arg.next;
|
2007-10-16 23:36:14 -06:00
|
|
|
mutex_lock(&dev->struct_mutex);
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
buffers[buf_count].buffer = drm_lookup_buffer_object(file_priv, req->arg_handle, 1);
|
2007-10-16 23:36:14 -06:00
|
|
|
mutex_unlock(&dev->struct_mutex);
|
2007-10-11 18:54:38 -06:00
|
|
|
buf_count++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
rep.ret = 0;
|
|
|
|
if (req->op != drm_bo_validate) {
|
|
|
|
DRM_ERROR
|
|
|
|
("Buffer object operation wasn't \"validate\".\n");
|
|
|
|
rep.ret = -EINVAL;
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf_handle = req->bo_req.handle;
|
2008-01-23 21:37:40 -07:00
|
|
|
reloc_user_ptr = (uint32_t *)(unsigned long)arg.reloc_ptr;
|
2007-10-11 18:54:38 -06:00
|
|
|
|
2008-01-23 21:37:40 -07:00
|
|
|
if (reloc_user_ptr) {
|
|
|
|
ret = i915_exec_reloc(file_priv, buf_handle, reloc_user_ptr, buffers, buf_count);
|
2007-10-25 00:53:18 -06:00
|
|
|
if (ret)
|
|
|
|
goto out_err;
|
|
|
|
DRM_MEMORYBARRIER();
|
|
|
|
}
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
rep.ret = drm_bo_handle_validate(file_priv, req->bo_req.handle,
|
2007-12-14 13:45:55 -07:00
|
|
|
req->bo_req.flags, req->bo_req.mask,
|
2007-10-11 18:54:38 -06:00
|
|
|
req->bo_req.hint,
|
2007-12-14 13:45:55 -07:00
|
|
|
req->bo_req.fence_class, 0,
|
2007-10-11 18:54:38 -06:00
|
|
|
&rep.bo_info,
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
&buffers[buf_count].buffer);
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
if (rep.ret) {
|
|
|
|
DRM_ERROR("error on handle validate %d\n", rep.ret);
|
|
|
|
goto out_err;
|
|
|
|
}
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
/*
|
|
|
|
* If the user provided a presumed offset hint, check whether
|
|
|
|
* the buffer is in the same place, if so, relocations relative to
|
|
|
|
* this buffer need not be performed
|
|
|
|
*/
|
|
|
|
if ((req->bo_req.hint & DRM_BO_HINT_PRESUMED_OFFSET) &&
|
|
|
|
buffers[buf_count].buffer->offset == req->bo_req.presumed_offset) {
|
|
|
|
buffers[buf_count].presumed_offset_correct = 1;
|
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
next = arg.next;
|
|
|
|
arg.handled = 1;
|
|
|
|
arg.d.rep = rep;
|
|
|
|
|
2007-11-28 16:55:38 -07:00
|
|
|
if (copy_to_user((void __user *)(unsigned long)data, &arg, sizeof(arg)))
|
2007-10-11 18:54:38 -06:00
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
data = next;
|
|
|
|
buf_count++;
|
|
|
|
|
|
|
|
} while (next != 0);
|
|
|
|
*num_buffers = buf_count;
|
|
|
|
return 0;
|
|
|
|
out_err:
|
|
|
|
mutex_lock(&dev->struct_mutex);
|
|
|
|
i915_dereference_buffers_locked(buffers, buf_count);
|
|
|
|
mutex_unlock(&dev->struct_mutex);
|
|
|
|
*num_buffers = 0;
|
|
|
|
return (ret) ? ret : rep.ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int i915_execbuffer(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
|
|
|
|
drm_i915_sarea_t *sarea_priv = (drm_i915_sarea_t *)
|
|
|
|
dev_priv->sarea_priv;
|
|
|
|
struct drm_i915_execbuffer *exec_buf = data;
|
|
|
|
struct _drm_i915_batchbuffer *batch = &exec_buf->batch;
|
|
|
|
struct drm_fence_arg *fence_arg = &exec_buf->fence_arg;
|
|
|
|
int num_buffers;
|
|
|
|
int ret;
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
struct drm_i915_validate_buffer *buffers;
|
2007-10-11 18:54:38 -06:00
|
|
|
struct drm_fence_object *fence;
|
|
|
|
|
|
|
|
if (!dev_priv->allow_batchbuffer) {
|
|
|
|
DRM_ERROR("Batchbuffer ioctl disabled\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (batch->num_cliprects && DRM_VERIFYAREA_READ(batch->cliprects,
|
|
|
|
batch->num_cliprects *
|
|
|
|
sizeof(struct drm_clip_rect)))
|
|
|
|
return -EFAULT;
|
|
|
|
|
|
|
|
if (exec_buf->num_buffers > dev_priv->max_validate_buffers)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2007-10-21 04:26:26 -06:00
|
|
|
|
|
|
|
ret = drm_bo_read_lock(&dev->bm.bm_lock);
|
2007-11-04 19:42:22 -07:00
|
|
|
if (ret)
|
2007-10-21 04:26:26 -06:00
|
|
|
return ret;
|
|
|
|
|
2007-10-22 10:59:37 -06:00
|
|
|
/*
|
|
|
|
* The cmdbuf_mutex makes sure the validate-submit-fence
|
2007-11-04 19:42:22 -07:00
|
|
|
* operation is atomic.
|
2007-10-22 10:59:37 -06:00
|
|
|
*/
|
|
|
|
|
|
|
|
ret = mutex_lock_interruptible(&dev_priv->cmdbuf_mutex);
|
|
|
|
if (ret) {
|
|
|
|
drm_bo_read_unlock(&dev->bm.bm_lock);
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
num_buffers = exec_buf->num_buffers;
|
|
|
|
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
buffers = drm_calloc(num_buffers, sizeof(struct drm_i915_validate_buffer), DRM_MEM_DRIVER);
|
2007-10-21 04:26:26 -06:00
|
|
|
if (!buffers) {
|
2007-11-28 16:37:51 -07:00
|
|
|
drm_bo_read_unlock(&dev->bm.bm_lock);
|
2007-10-22 10:59:37 -06:00
|
|
|
mutex_unlock(&dev_priv->cmdbuf_mutex);
|
2007-10-11 18:54:38 -06:00
|
|
|
return -ENOMEM;
|
2007-11-28 16:37:51 -07:00
|
|
|
}
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
/* validate buffer list + fixup relocations */
|
|
|
|
ret = i915_validate_buffer_list(file_priv, 0, exec_buf->ops_list,
|
|
|
|
buffers, &num_buffers);
|
|
|
|
if (ret)
|
|
|
|
goto out_free;
|
|
|
|
|
2007-10-23 01:54:07 -06:00
|
|
|
/* make sure all previous memory operations have passed */
|
2007-10-23 18:13:15 -06:00
|
|
|
DRM_MEMORYBARRIER();
|
2007-10-30 18:33:34 -06:00
|
|
|
drm_agp_chipset_flush(dev);
|
2007-10-23 01:54:07 -06:00
|
|
|
|
2007-10-11 18:54:38 -06:00
|
|
|
/* submit buffer */
|
Allow relocation to be skipped when buffers don't move.
One of the costs of superioctl has been the need to perform relocations
inside the kernel. The cost of mapping the buffers to the CPU and writing
data is fairly high, especially if those buffers have been mapped and read
by the GPU.
If we assume that buffers don't move around very often, we can have the
client compute the relocations itself using the previous GPU address. When
that object doesn't move, the kernel can skip computing and writing the
updated data.
Here's a patch which adds a new field to struct drm_bo_info_req called
'presumed_offset', and a new DRM_BO_HINT_PRESUMED_OFFSET that is set when
this field has been filled in by the client.
There are two separate optimizations performed when the presumed_offset is
correct:
1. i915_exec_reloc checks to see if all previous buffer offsets were guessed
correctly. If so, there's no need for it to look at *any* of the
relocations for a buffer. When this happens, it skips the whole
relocation process, simply returning success.
2. i915_apply_reloc checks to see if the target buffer offset was guessed
correctly. If so, it skips mapping the relocatee, computing the
relocation and writing the value. If no relocations are needed, the
relocatee should never be mapped to the CPU, and so the kernel shouldn't
need to wait for any fences to pass.
2007-12-04 13:22:30 -07:00
|
|
|
batch->start = buffers[num_buffers-1].buffer->offset;
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
DRM_DEBUG("i915 exec batchbuffer, start %x used %d cliprects %d\n",
|
|
|
|
batch->start, batch->used, batch->num_cliprects);
|
|
|
|
|
|
|
|
ret = i915_dispatch_batchbuffer(dev, batch);
|
|
|
|
if (ret)
|
|
|
|
goto out_err0;
|
|
|
|
|
2008-02-05 10:25:22 -07:00
|
|
|
if (sarea_priv)
|
|
|
|
sarea_priv->last_dispatch = READ_BREADCRUMB(dev_priv);
|
2007-10-11 18:54:38 -06:00
|
|
|
|
|
|
|
/* fence */
|
2008-01-15 02:03:41 -07:00
|
|
|
ret = drm_fence_buffer_objects(dev, NULL, fence_arg->flags,
|
|
|
|
NULL, &fence);
|
2007-10-11 18:54:38 -06:00
|
|
|
if (ret)
|
|
|
|
goto out_err0;
|
|
|
|
|
|
|
|
if (!(fence_arg->flags & DRM_FENCE_FLAG_NO_USER)) {
|
|
|
|
ret = drm_fence_add_user_object(file_priv, fence, fence_arg->flags & DRM_FENCE_FLAG_SHAREABLE);
|
|
|
|
if (!ret) {
|
|
|
|
fence_arg->handle = fence->base.hash.key;
|
|
|
|
fence_arg->fence_class = fence->fence_class;
|
|
|
|
fence_arg->type = fence->type;
|
2008-01-30 14:06:02 -07:00
|
|
|
fence_arg->signaled = fence->signaled_types;
|
2007-10-11 18:54:38 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
drm_fence_usage_deref_unlocked(&fence);
|
|
|
|
out_err0:
|
|
|
|
|
|
|
|
/* handle errors */
|
|
|
|
mutex_lock(&dev->struct_mutex);
|
|
|
|
i915_dereference_buffers_locked(buffers, num_buffers);
|
|
|
|
mutex_unlock(&dev->struct_mutex);
|
|
|
|
|
|
|
|
out_free:
|
|
|
|
drm_free(buffers, (exec_buf->num_buffers * sizeof(struct drm_buffer_object *)), DRM_MEM_DRIVER);
|
2007-10-22 10:59:37 -06:00
|
|
|
|
|
|
|
mutex_unlock(&dev_priv->cmdbuf_mutex);
|
2007-10-21 04:26:26 -06:00
|
|
|
drm_bo_read_unlock(&dev->bm.bm_lock);
|
2007-10-11 18:54:38 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
static int i915_do_cleanup_pageflip(struct drm_device * dev)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-09-11 04:48:46 -06:00
|
|
|
int i, planes, num_pages = dev_priv->sarea_priv->third_handle ? 3 : 2;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-11-23 23:56:05 -07:00
|
|
|
DRM_DEBUG("\n");
|
2007-02-19 04:27:54 -07:00
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
for (i = 0, planes = 0; i < 2; i++)
|
2007-02-28 09:48:56 -07:00
|
|
|
if (dev_priv->sarea_priv->pf_current_page & (0x3 << (2 * i))) {
|
|
|
|
dev_priv->sarea_priv->pf_current_page =
|
|
|
|
(dev_priv->sarea_priv->pf_current_page &
|
2007-11-25 10:50:07 -07:00
|
|
|
~(0x3 << (2 * i))) | ((num_pages - 1) << (2 * i));
|
2007-02-19 04:27:54 -07:00
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
planes |= 1 << i;
|
2007-02-28 09:48:56 -07:00
|
|
|
}
|
2007-02-19 04:27:54 -07:00
|
|
|
|
2007-09-11 04:48:46 -06:00
|
|
|
if (planes)
|
|
|
|
i915_dispatch_flip(dev, planes, 0);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_flip_bufs(struct drm_device *dev, void *data, struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_flip_t *param = data;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-11-23 23:56:05 -07:00
|
|
|
DRM_DEBUG("\n");
|
2004-11-11 04:09:11 -07:00
|
|
|
|
2007-07-20 07:39:25 -06:00
|
|
|
LOCK_TEST_WITH_RETURN(dev, file_priv);
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2007-09-28 11:10:08 -06:00
|
|
|
/* This is really planes */
|
|
|
|
if (param->pipes & ~0x3) {
|
2007-09-11 04:48:46 -06:00
|
|
|
DRM_ERROR("Invalid planes 0x%x, only <= 0x3 is valid\n",
|
2007-09-28 11:10:08 -06:00
|
|
|
param->pipes);
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2007-02-19 04:27:54 -07:00
|
|
|
}
|
|
|
|
|
2007-09-28 11:10:08 -06:00
|
|
|
i915_dispatch_flip(dev, param->pipes, 0);
|
2007-02-19 04:27:54 -07:00
|
|
|
|
|
|
|
return 0;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2006-08-31 13:42:29 -06:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_getparam(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_getparam_t *param = data;
|
2004-06-10 06:45:38 -06:00
|
|
|
int value;
|
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (!dev_priv) {
|
2008-01-02 23:56:04 -07:00
|
|
|
DRM_ERROR("called with no initialization\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
switch (param->param) {
|
2004-06-10 06:45:38 -06:00
|
|
|
case I915_PARAM_IRQ_ACTIVE:
|
|
|
|
value = dev->irq ? 1 : 0;
|
|
|
|
break;
|
|
|
|
case I915_PARAM_ALLOW_BATCHBUFFER:
|
|
|
|
value = dev_priv->allow_batchbuffer ? 1 : 0;
|
|
|
|
break;
|
2006-01-24 14:16:54 -07:00
|
|
|
case I915_PARAM_LAST_DISPATCH:
|
|
|
|
value = READ_BREADCRUMB(dev_priv);
|
|
|
|
break;
|
2004-06-10 06:45:38 -06:00
|
|
|
default:
|
2007-07-19 18:11:11 -06:00
|
|
|
DRM_ERROR("Unknown parameter %d\n", param->param);
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
if (DRM_COPY_TO_USER(param->value, &value, sizeof(int))) {
|
2004-06-10 06:45:38 -06:00
|
|
|
DRM_ERROR("DRM_COPY_TO_USER failed\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EFAULT;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
2004-08-27 03:14:30 -06:00
|
|
|
|
2004-06-10 06:45:38 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_setparam(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2004-06-10 06:45:38 -06:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_setparam_t *param = data;
|
2004-06-10 06:45:38 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
if (!dev_priv) {
|
2008-01-02 23:56:04 -07:00
|
|
|
DRM_ERROR("called with no initialization\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
switch (param->param) {
|
2004-06-10 06:45:38 -06:00
|
|
|
case I915_SETPARAM_USE_MI_BATCHBUFFER_START:
|
2007-11-21 23:10:36 -07:00
|
|
|
if (!IS_I965G(dev))
|
|
|
|
dev_priv->use_mi_batchbuffer_start = param->value;
|
2004-06-10 06:45:38 -06:00
|
|
|
break;
|
|
|
|
case I915_SETPARAM_TEX_LRU_LOG_GRANULARITY:
|
2007-07-19 18:11:11 -06:00
|
|
|
dev_priv->tex_lru_log_granularity = param->value;
|
2004-06-10 06:45:38 -06:00
|
|
|
break;
|
|
|
|
case I915_SETPARAM_ALLOW_BATCHBUFFER:
|
2007-07-19 18:11:11 -06:00
|
|
|
dev_priv->allow_batchbuffer = param->value;
|
2004-06-10 06:45:38 -06:00
|
|
|
break;
|
|
|
|
default:
|
2007-07-19 18:11:11 -06:00
|
|
|
DRM_ERROR("unknown parameter %d\n", param->param);
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2004-06-10 06:45:38 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2004-08-17 07:10:05 -06:00
|
|
|
|
2006-12-04 00:48:04 -07:00
|
|
|
drm_i915_mmio_entry_t mmio_table[] = {
|
|
|
|
[MMIO_REGS_PS_DEPTH_COUNT] = {
|
|
|
|
I915_MMIO_MAY_READ|I915_MMIO_MAY_WRITE,
|
|
|
|
0x2350,
|
|
|
|
8
|
2007-11-04 19:42:22 -07:00
|
|
|
}
|
2006-12-04 00:48:04 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static int mmio_table_size = sizeof(mmio_table)/sizeof(drm_i915_mmio_entry_t);
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_mmio(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2006-12-04 00:48:04 -07:00
|
|
|
{
|
2007-07-16 02:53:06 -06:00
|
|
|
uint32_t buf[8];
|
2006-12-04 00:48:04 -07:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-11-04 19:42:22 -07:00
|
|
|
drm_i915_mmio_entry_t *e;
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_mmio_t *mmio = data;
|
2006-12-04 00:48:04 -07:00
|
|
|
void __iomem *base;
|
2007-07-16 02:53:06 -06:00
|
|
|
int i;
|
|
|
|
|
2006-12-04 00:48:04 -07:00
|
|
|
if (!dev_priv) {
|
2008-01-02 23:56:04 -07:00
|
|
|
DRM_ERROR("called with no initialization\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2006-12-04 00:48:04 -07:00
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
if (mmio->reg >= mmio_table_size)
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2006-12-04 00:48:04 -07:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
e = &mmio_table[mmio->reg];
|
2007-05-15 14:35:33 -06:00
|
|
|
base = (u8 *) dev_priv->mmio_map->handle + e->offset;
|
2006-12-04 00:48:04 -07:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
switch (mmio->read_write) {
|
2007-11-21 23:10:36 -07:00
|
|
|
case I915_MMIO_READ:
|
|
|
|
if (!(e->flag & I915_MMIO_MAY_READ))
|
|
|
|
return -EINVAL;
|
|
|
|
for (i = 0; i < e->size / 4; i++)
|
|
|
|
buf[i] = I915_READ(e->offset + i * 4);
|
|
|
|
if (DRM_COPY_TO_USER(mmio->data, buf, e->size)) {
|
|
|
|
DRM_ERROR("DRM_COPY_TO_USER failed\n");
|
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case I915_MMIO_WRITE:
|
|
|
|
if (!(e->flag & I915_MMIO_MAY_WRITE))
|
|
|
|
return -EINVAL;
|
2007-11-28 16:37:51 -07:00
|
|
|
if (DRM_COPY_FROM_USER(buf, mmio->data, e->size)) {
|
2007-11-21 23:10:36 -07:00
|
|
|
DRM_ERROR("DRM_COPY_TO_USER failed\n");
|
2007-11-28 16:37:51 -07:00
|
|
|
return -EFAULT;
|
2007-11-21 23:10:36 -07:00
|
|
|
}
|
|
|
|
for (i = 0; i < e->size / 4; i++)
|
|
|
|
I915_WRITE(e->offset + i * 4, buf[i]);
|
|
|
|
break;
|
2006-12-04 00:48:04 -07:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
static int i915_set_status_page(struct drm_device *dev, void *data,
|
|
|
|
struct drm_file *file_priv)
|
2007-06-05 12:15:29 -06:00
|
|
|
{
|
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
2007-07-19 18:11:11 -06:00
|
|
|
drm_i915_hws_addr_t *hws = data;
|
2007-06-05 12:15:29 -06:00
|
|
|
|
|
|
|
if (!dev_priv) {
|
2008-01-02 23:56:04 -07:00
|
|
|
DRM_ERROR("called with no initialization\n");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -EINVAL;
|
2007-06-05 12:15:29 -06:00
|
|
|
}
|
2007-07-19 18:11:11 -06:00
|
|
|
DRM_DEBUG("set status page addr 0x%08x\n", (u32)hws->addr);
|
2007-06-05 12:15:29 -06:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
dev_priv->status_gfx_addr = hws->addr & (0x1ffff<<12);
|
2007-06-05 12:15:29 -06:00
|
|
|
|
2007-07-19 18:11:11 -06:00
|
|
|
dev_priv->hws_map.offset = dev->agp->base + hws->addr;
|
2007-06-05 12:15:29 -06:00
|
|
|
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");
|
2007-07-19 18:00:17 -06:00
|
|
|
return -ENOMEM;
|
2007-06-05 12:15:29 -06:00
|
|
|
}
|
|
|
|
dev_priv->hw_status_page = dev_priv->hws_map.handle;
|
|
|
|
|
|
|
|
memset(dev_priv->hw_status_page, 0, PAGE_SIZE);
|
|
|
|
I915_WRITE(0x02080, dev_priv->status_gfx_addr);
|
|
|
|
DRM_DEBUG("load hws 0x2080 with gfx mem 0x%x\n",
|
|
|
|
dev_priv->status_gfx_addr);
|
|
|
|
DRM_DEBUG("load hws at %p\n", dev_priv->hw_status_page);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
int i915_driver_load(struct drm_device *dev, unsigned long flags)
|
2005-08-04 21:50:23 -06:00
|
|
|
{
|
2007-10-26 17:10:02 -06:00
|
|
|
struct drm_i915_private *dev_priv = dev->dev_private;
|
|
|
|
unsigned long base, size;
|
|
|
|
int ret = 0, mmio_bar = IS_I9XX(dev) ? 0 : 1;
|
|
|
|
|
2005-08-04 21:50:23 -06:00
|
|
|
/* 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;
|
|
|
|
|
2007-10-26 17:10:02 -06:00
|
|
|
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);
|
|
|
|
|
2007-12-01 00:40:13 -07:00
|
|
|
ret = drm_addmap(dev, base, size, _DRM_REGISTERS,
|
|
|
|
_DRM_KERNEL | _DRM_DRIVER, &dev_priv->mmio_map);
|
2007-10-26 17:10:02 -06:00
|
|
|
|
2007-11-19 09:41:23 -07:00
|
|
|
#ifdef __linux__
|
2007-10-30 18:27:44 -06:00
|
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,25)
|
|
|
|
intel_init_chipset_flush_compat(dev);
|
2007-11-13 15:50:46 -07:00
|
|
|
#endif
|
2007-10-30 18:27:44 -06:00
|
|
|
#endif
|
2007-12-01 00:40:13 -07:00
|
|
|
|
2007-10-26 17:10:02 -06:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int i915_driver_unload(struct drm_device *dev)
|
|
|
|
{
|
|
|
|
struct drm_i915_private *dev_priv = dev->dev_private;
|
|
|
|
|
|
|
|
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);
|
2007-11-19 09:41:23 -07:00
|
|
|
#ifdef __linux__
|
2007-10-30 18:27:44 -06:00
|
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,25)
|
|
|
|
intel_fini_chipset_flush_compat(dev);
|
2007-11-13 15:50:46 -07:00
|
|
|
#endif
|
2007-10-30 18:27:44 -06:00
|
|
|
#endif
|
2005-08-04 21:50:23 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
void i915_driver_lastclose(struct drm_device * dev)
|
2004-08-17 07:10:05 -06:00
|
|
|
{
|
2007-10-26 17:10:02 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
|
|
|
|
if (drm_getsarea(dev) && dev_priv->sarea_priv)
|
2007-02-28 07:57:08 -07:00
|
|
|
i915_do_cleanup_pageflip(dev);
|
2007-10-26 17:10:02 -06:00
|
|
|
if (dev_priv->agp_heap)
|
2004-08-27 03:14:30 -06:00
|
|
|
i915_mem_takedown(&(dev_priv->agp_heap));
|
2007-10-26 17:10:02 -06:00
|
|
|
|
2004-08-27 03:14:30 -06:00
|
|
|
i915_dma_cleanup(dev);
|
2004-08-17 07:10:05 -06:00
|
|
|
}
|
|
|
|
|
2007-07-20 07:39:25 -06:00
|
|
|
void i915_driver_preclose(struct drm_device * dev, struct drm_file *file_priv)
|
2004-08-17 07:10:05 -06:00
|
|
|
{
|
2007-10-26 17:10:02 -06:00
|
|
|
drm_i915_private_t *dev_priv = dev->dev_private;
|
|
|
|
i915_mem_release(dev, file_priv, dev_priv->agp_heap);
|
2004-08-17 07:10:05 -06:00
|
|
|
}
|
2005-02-01 03:43:42 -07:00
|
|
|
|
2007-07-15 20:32:51 -06:00
|
|
|
struct drm_ioctl_desc i915_ioctls[] = {
|
2007-07-19 18:11:11 -06:00
|
|
|
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_MMIO, i915_mmio, DRM_AUTH),
|
|
|
|
DRM_IOCTL_DEF(DRM_I915_HWS_ADDR, i915_set_status_page, DRM_AUTH),
|
2007-10-11 18:54:38 -06:00
|
|
|
#ifdef I915_HAVE_BUFFER
|
|
|
|
DRM_IOCTL_DEF(DRM_I915_EXECBUFFER, i915_execbuffer, DRM_AUTH),
|
|
|
|
#endif
|
2005-02-01 03:43:42 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
int i915_max_ioctl = DRM_ARRAY_SIZE(i915_ioctls);
|
2005-05-27 17:42:11 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2007-07-15 20:32:51 -06:00
|
|
|
int i915_driver_device_is_agp(struct drm_device * dev)
|
2005-05-27 17:42:11 -06:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
2007-03-27 02:01:31 -06:00
|
|
|
|
|
|
|
int i915_driver_firstopen(struct drm_device *dev)
|
|
|
|
{
|
|
|
|
#ifdef I915_HAVE_BUFFER
|
2007-11-14 19:29:55 -07:00
|
|
|
drm_bo_driver_init(dev);
|
2007-03-27 02:01:31 -06:00
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|