1 commit c97398223c6a505fac2c783a624dc80e0aa5d5d0
2 Author: Eric Anholt <eric@anholt.net>
3 Date: Wed Jul 30 12:06:12 2008 -0700
5 drm: Add GEM ("graphics execution manager") to i915 driver.
7 GEM allows the creation of persistent buffer objects accessible by the
8 graphics device through new ioctls for managing execution of commands on the
9 device. The userland API is almost entirely driver-specific to ensure that
10 any driver building on this model can easily map the interface to individual
13 GEM is used by the 2d driver for managing its internal state allocations and
14 will be used for pixmap storage to reduce memory consumption and enable
15 zero-copy GLX_EXT_texture_from_pixmap, and in the 3d driver is used to enable
16 GL_EXT_framebuffer_object and GL_ARB_pixel_buffer_object.
18 Signed-off-by: Eric Anholt <eric@anholt.net>
20 diff --git a/drivers/gpu/drm/Makefile b/drivers/gpu/drm/Makefile
21 index e9f9a97..74da994 100644
22 --- a/drivers/gpu/drm/Makefile
23 +++ b/drivers/gpu/drm/Makefile
26 ccflags-y := -Iinclude/drm
28 -drm-y := drm_auth.o drm_bufs.o drm_context.o drm_dma.o drm_drawable.o \
29 - drm_drv.o drm_fops.o drm_ioctl.o drm_irq.o \
30 +drm-y := drm_auth.o drm_bufs.o drm_cache.o \
31 + drm_context.o drm_dma.o drm_drawable.o \
32 + drm_drv.o drm_fops.o drm_gem.o drm_ioctl.o drm_irq.o \
33 drm_lock.o drm_memory.o drm_proc.o drm_stub.o drm_vm.o \
34 drm_agpsupport.o drm_scatter.o ati_pcigart.o drm_pci.o \
35 drm_sysfs.o drm_hashtab.o drm_sman.o drm_mm.o
36 diff --git a/drivers/gpu/drm/drm_agpsupport.c b/drivers/gpu/drm/drm_agpsupport.c
37 index aefa5ac..2639be2 100644
38 --- a/drivers/gpu/drm/drm_agpsupport.c
39 +++ b/drivers/gpu/drm/drm_agpsupport.c
43 #include <linux/module.h>
48 @@ -452,4 +453,52 @@ int drm_agp_unbind_memory(DRM_AGP_MEM * handle)
49 return agp_unbind_memory(handle);
52 -#endif /* __OS_HAS_AGP */
54 + * Binds a collection of pages into AGP memory at the given offset, returning
55 + * the AGP memory structure containing them.
57 + * No reference is held on the pages during this time -- it is up to the
58 + * caller to handle that.
61 +drm_agp_bind_pages(struct drm_device *dev,
62 + struct page **pages,
63 + unsigned long num_pages,
64 + uint32_t gtt_offset)
71 + mem = drm_agp_allocate_memory(dev->agp->bridge, num_pages,
74 + DRM_ERROR("Failed to allocate memory for %ld pages\n",
79 + for (i = 0; i < num_pages; i++)
80 + mem->memory[i] = phys_to_gart(page_to_phys(pages[i]));
81 + mem->page_count = num_pages;
83 + mem->is_flushed = true;
84 + ret = drm_agp_bind_memory(mem, gtt_offset / PAGE_SIZE);
86 + DRM_ERROR("Failed to bind AGP memory: %d\n", ret);
87 + agp_free_memory(mem);
93 +EXPORT_SYMBOL(drm_agp_bind_pages);
95 +void drm_agp_chipset_flush(struct drm_device *dev)
97 + agp_flush_chipset(dev->agp->bridge);
99 +EXPORT_SYMBOL(drm_agp_chipset_flush);
101 +#endif /* __OS_HAS_AGP */
102 diff --git a/drivers/gpu/drm/drm_cache.c b/drivers/gpu/drm/drm_cache.c
104 index 0000000..9475f7d
106 +++ b/drivers/gpu/drm/drm_cache.c
108 +/**************************************************************************
110 + * Copyright (c) 2006-2007 Tungsten Graphics, Inc., Cedar Park, TX., USA
111 + * All Rights Reserved.
113 + * Permission is hereby granted, free of charge, to any person obtaining a
114 + * copy of this software and associated documentation files (the
115 + * "Software"), to deal in the Software without restriction, including
116 + * without limitation the rights to use, copy, modify, merge, publish,
117 + * distribute, sub license, and/or sell copies of the Software, and to
118 + * permit persons to whom the Software is furnished to do so, subject to
119 + * the following conditions:
121 + * The above copyright notice and this permission notice (including the
122 + * next paragraph) shall be included in all copies or substantial portions
125 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
126 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
127 + * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
128 + * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM,
129 + * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
130 + * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
131 + * USE OR OTHER DEALINGS IN THE SOFTWARE.
133 + **************************************************************************/
135 + * Authors: Thomas Hellström <thomas-at-tungstengraphics-dot-com>
140 +#if defined(CONFIG_X86)
142 +drm_clflush_page(struct page *page)
144 + uint8_t *page_virtual;
147 + if (unlikely(page == NULL))
150 + page_virtual = kmap_atomic(page, KM_USER0);
151 + for (i = 0; i < PAGE_SIZE; i += boot_cpu_data.x86_clflush_size)
152 + clflush(page_virtual + i);
153 + kunmap_atomic(page_virtual, KM_USER0);
158 +drm_clflush_ipi_handler(void *null)
164 +drm_clflush_pages(struct page *pages[], unsigned long num_pages)
167 +#if defined(CONFIG_X86)
168 + if (cpu_has_clflush) {
172 + for (i = 0; i < num_pages; ++i)
173 + drm_clflush_page(*pages++);
180 + if (on_each_cpu(drm_clflush_ipi_handler, NULL, 1) != 0)
181 + DRM_ERROR("Timed out waiting for cache flush.\n");
183 +EXPORT_SYMBOL(drm_clflush_pages);
184 diff --git a/drivers/gpu/drm/drm_drv.c b/drivers/gpu/drm/drm_drv.c
185 index fb45fe7..96f416a 100644
186 --- a/drivers/gpu/drm/drm_drv.c
187 +++ b/drivers/gpu/drm/drm_drv.c
188 @@ -119,6 +119,10 @@ static struct drm_ioctl_desc drm_ioctls[] = {
189 DRM_IOCTL_DEF(DRM_IOCTL_MODESET_CTL, drm_modeset_ctl, 0),
191 DRM_IOCTL_DEF(DRM_IOCTL_UPDATE_DRAW, drm_update_drawable_info, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
193 + DRM_IOCTL_DEF(DRM_IOCTL_GEM_CLOSE, drm_gem_close_ioctl, 0),
194 + DRM_IOCTL_DEF(DRM_IOCTL_GEM_FLINK, drm_gem_flink_ioctl, DRM_AUTH),
195 + DRM_IOCTL_DEF(DRM_IOCTL_GEM_OPEN, drm_gem_open_ioctl, DRM_AUTH),
198 #define DRM_CORE_IOCTL_COUNT ARRAY_SIZE( drm_ioctls )
199 diff --git a/drivers/gpu/drm/drm_fops.c b/drivers/gpu/drm/drm_fops.c
200 index dcf8b4d..0d46627 100644
201 --- a/drivers/gpu/drm/drm_fops.c
202 +++ b/drivers/gpu/drm/drm_fops.c
203 @@ -256,6 +256,9 @@ static int drm_open_helper(struct inode *inode, struct file *filp,
205 INIT_LIST_HEAD(&priv->lhead);
207 + if (dev->driver->driver_features & DRIVER_GEM)
208 + drm_gem_open(dev, priv);
210 if (dev->driver->open) {
211 ret = dev->driver->open(dev, priv);
213 @@ -400,6 +403,9 @@ int drm_release(struct inode *inode, struct file *filp)
214 dev->driver->reclaim_buffers(dev, file_priv);
217 + if (dev->driver->driver_features & DRIVER_GEM)
218 + drm_gem_release(dev, file_priv);
220 drm_fasync(-1, filp, 0);
222 mutex_lock(&dev->ctxlist_mutex);
223 diff --git a/drivers/gpu/drm/drm_gem.c b/drivers/gpu/drm/drm_gem.c
225 index 0000000..434155b
227 +++ b/drivers/gpu/drm/drm_gem.c
230 + * Copyright © 2008 Intel Corporation
232 + * Permission is hereby granted, free of charge, to any person obtaining a
233 + * copy of this software and associated documentation files (the "Software"),
234 + * to deal in the Software without restriction, including without limitation
235 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
236 + * and/or sell copies of the Software, and to permit persons to whom the
237 + * Software is furnished to do so, subject to the following conditions:
239 + * The above copyright notice and this permission notice (including the next
240 + * paragraph) shall be included in all copies or substantial portions of the
243 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
244 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
245 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
246 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
247 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
248 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
252 + * Eric Anholt <eric@anholt.net>
256 +#include <linux/types.h>
257 +#include <linux/slab.h>
258 +#include <linux/mm.h>
259 +#include <linux/uaccess.h>
260 +#include <linux/fs.h>
261 +#include <linux/file.h>
262 +#include <linux/module.h>
263 +#include <linux/mman.h>
264 +#include <linux/pagemap.h>
269 + * This file provides some of the base ioctls and library routines for
270 + * the graphics memory manager implemented by each device driver.
272 + * Because various devices have different requirements in terms of
273 + * synchronization and migration strategies, implementing that is left up to
274 + * the driver, and all that the general API provides should be generic --
275 + * allocating objects, reading/writing data with the cpu, freeing objects.
276 + * Even there, platform-dependent optimizations for reading/writing data with
277 + * the CPU mean we'll likely hook those out to driver-specific calls. However,
278 + * the DRI2 implementation wants to have at least allocate/mmap be generic.
280 + * The goal was to have swap-backed object allocation managed through
281 + * struct file. However, file descriptors as handles to a struct file have
282 + * two major failings:
283 + * - Process limits prevent more than 1024 or so being used at a time by
285 + * - Inability to allocate high fds will aggravate the X Server's select()
286 + * handling, and likely that of many GL client applications as well.
288 + * This led to a plan of using our own integer IDs (called handles, following
289 + * DRM terminology) to mimic fds, and implement the fd syscalls we need as
290 + * ioctls. The objects themselves will still include the struct file so
291 + * that we can transition to fds if the required kernel infrastructure shows
292 + * up at a later date, and as our interface with shmfs for memory allocation.
296 + * Initialize the GEM device fields
300 +drm_gem_init(struct drm_device *dev)
302 + spin_lock_init(&dev->object_name_lock);
303 + idr_init(&dev->object_name_idr);
304 + atomic_set(&dev->object_count, 0);
305 + atomic_set(&dev->object_memory, 0);
306 + atomic_set(&dev->pin_count, 0);
307 + atomic_set(&dev->pin_memory, 0);
308 + atomic_set(&dev->gtt_count, 0);
309 + atomic_set(&dev->gtt_memory, 0);
314 + * Allocate a GEM object of the specified size with shmfs backing store
316 +struct drm_gem_object *
317 +drm_gem_object_alloc(struct drm_device *dev, size_t size)
319 + struct drm_gem_object *obj;
321 + BUG_ON((size & (PAGE_SIZE - 1)) != 0);
323 + obj = kcalloc(1, sizeof(*obj), GFP_KERNEL);
326 + obj->filp = shmem_file_setup("drm mm object", size, 0);
327 + if (IS_ERR(obj->filp)) {
332 + kref_init(&obj->refcount);
333 + kref_init(&obj->handlecount);
335 + if (dev->driver->gem_init_object != NULL &&
336 + dev->driver->gem_init_object(obj) != 0) {
341 + atomic_inc(&dev->object_count);
342 + atomic_add(obj->size, &dev->object_memory);
345 +EXPORT_SYMBOL(drm_gem_object_alloc);
348 + * Removes the mapping from handle to filp for this object.
351 +drm_gem_handle_delete(struct drm_file *filp, int handle)
353 + struct drm_device *dev;
354 + struct drm_gem_object *obj;
356 + /* This is gross. The idr system doesn't let us try a delete and
357 + * return an error code. It just spews if you fail at deleting.
358 + * So, we have to grab a lock around finding the object and then
359 + * doing the delete on it and dropping the refcount, or the user
360 + * could race us to double-decrement the refcount and cause a
361 + * use-after-free later. Given the frequency of our handle lookups,
362 + * we may want to use ida for number allocation and a hash table
363 + * for the pointers, anyway.
365 + spin_lock(&filp->table_lock);
367 + /* Check if we currently have a reference on the object */
368 + obj = idr_find(&filp->object_idr, handle);
370 + spin_unlock(&filp->table_lock);
375 + /* Release reference and decrement refcount. */
376 + idr_remove(&filp->object_idr, handle);
377 + spin_unlock(&filp->table_lock);
379 + mutex_lock(&dev->struct_mutex);
380 + drm_gem_object_handle_unreference(obj);
381 + mutex_unlock(&dev->struct_mutex);
387 + * Create a handle for this object. This adds a handle reference
388 + * to the object, which includes a regular reference count. Callers
389 + * will likely want to dereference the object afterwards.
392 +drm_gem_handle_create(struct drm_file *file_priv,
393 + struct drm_gem_object *obj,
399 + * Get the user-visible handle using idr.
402 + /* ensure there is space available to allocate a handle */
403 + if (idr_pre_get(&file_priv->object_idr, GFP_KERNEL) == 0)
406 + /* do the allocation under our spinlock */
407 + spin_lock(&file_priv->table_lock);
408 + ret = idr_get_new_above(&file_priv->object_idr, obj, 1, handlep);
409 + spin_unlock(&file_priv->table_lock);
410 + if (ret == -EAGAIN)
416 + drm_gem_object_handle_reference(obj);
419 +EXPORT_SYMBOL(drm_gem_handle_create);
421 +/** Returns a reference to the object named by the handle. */
422 +struct drm_gem_object *
423 +drm_gem_object_lookup(struct drm_device *dev, struct drm_file *filp,
426 + struct drm_gem_object *obj;
428 + spin_lock(&filp->table_lock);
430 + /* Check if we currently have a reference on the object */
431 + obj = idr_find(&filp->object_idr, handle);
433 + spin_unlock(&filp->table_lock);
437 + drm_gem_object_reference(obj);
439 + spin_unlock(&filp->table_lock);
443 +EXPORT_SYMBOL(drm_gem_object_lookup);
446 + * Releases the handle to an mm object.
449 +drm_gem_close_ioctl(struct drm_device *dev, void *data,
450 + struct drm_file *file_priv)
452 + struct drm_gem_close *args = data;
455 + if (!(dev->driver->driver_features & DRIVER_GEM))
458 + ret = drm_gem_handle_delete(file_priv, args->handle);
464 + * Create a global name for an object, returning the name.
466 + * Note that the name does not hold a reference; when the object
467 + * is freed, the name goes away.
470 +drm_gem_flink_ioctl(struct drm_device *dev, void *data,
471 + struct drm_file *file_priv)
473 + struct drm_gem_flink *args = data;
474 + struct drm_gem_object *obj;
477 + if (!(dev->driver->driver_features & DRIVER_GEM))
480 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
485 + if (idr_pre_get(&dev->object_name_idr, GFP_KERNEL) == 0)
488 + spin_lock(&dev->object_name_lock);
490 + spin_unlock(&dev->object_name_lock);
493 + ret = idr_get_new_above(&dev->object_name_idr, obj, 1,
495 + spin_unlock(&dev->object_name_lock);
496 + if (ret == -EAGAIN)
500 + mutex_lock(&dev->struct_mutex);
501 + drm_gem_object_unreference(obj);
502 + mutex_unlock(&dev->struct_mutex);
507 + * Leave the reference from the lookup around as the
508 + * name table now holds one
510 + args->name = (uint64_t) obj->name;
516 + * Open an object using the global name, returning a handle and the size.
518 + * This handle (of course) holds a reference to the object, so the object
519 + * will not go away until the handle is deleted.
522 +drm_gem_open_ioctl(struct drm_device *dev, void *data,
523 + struct drm_file *file_priv)
525 + struct drm_gem_open *args = data;
526 + struct drm_gem_object *obj;
530 + if (!(dev->driver->driver_features & DRIVER_GEM))
533 + spin_lock(&dev->object_name_lock);
534 + obj = idr_find(&dev->object_name_idr, (int) args->name);
536 + drm_gem_object_reference(obj);
537 + spin_unlock(&dev->object_name_lock);
541 + ret = drm_gem_handle_create(file_priv, obj, &handle);
542 + mutex_lock(&dev->struct_mutex);
543 + drm_gem_object_unreference(obj);
544 + mutex_unlock(&dev->struct_mutex);
548 + args->handle = handle;
549 + args->size = obj->size;
555 + * Called at device open time, sets up the structure for handling refcounting
559 +drm_gem_open(struct drm_device *dev, struct drm_file *file_private)
561 + idr_init(&file_private->object_idr);
562 + spin_lock_init(&file_private->table_lock);
566 + * Called at device close to release the file's
567 + * handle references on objects.
570 +drm_gem_object_release_handle(int id, void *ptr, void *data)
572 + struct drm_gem_object *obj = ptr;
574 + drm_gem_object_handle_unreference(obj);
580 + * Called at close time when the filp is going away.
582 + * Releases any remaining references on objects by this filp.
585 +drm_gem_release(struct drm_device *dev, struct drm_file *file_private)
587 + mutex_lock(&dev->struct_mutex);
588 + idr_for_each(&file_private->object_idr,
589 + &drm_gem_object_release_handle, NULL);
591 + idr_destroy(&file_private->object_idr);
592 + mutex_unlock(&dev->struct_mutex);
596 + * Called after the last reference to the object has been lost.
601 +drm_gem_object_free(struct kref *kref)
603 + struct drm_gem_object *obj = (struct drm_gem_object *) kref;
604 + struct drm_device *dev = obj->dev;
606 + BUG_ON(!mutex_is_locked(&dev->struct_mutex));
608 + if (dev->driver->gem_free_object != NULL)
609 + dev->driver->gem_free_object(obj);
612 + atomic_dec(&dev->object_count);
613 + atomic_sub(obj->size, &dev->object_memory);
616 +EXPORT_SYMBOL(drm_gem_object_free);
619 + * Called after the last handle to the object has been closed
621 + * Removes any name for the object. Note that this must be
622 + * called before drm_gem_object_free or we'll be touching
626 +drm_gem_object_handle_free(struct kref *kref)
628 + struct drm_gem_object *obj = container_of(kref,
629 + struct drm_gem_object,
631 + struct drm_device *dev = obj->dev;
633 + /* Remove any name for this object */
634 + spin_lock(&dev->object_name_lock);
636 + idr_remove(&dev->object_name_idr, obj->name);
637 + spin_unlock(&dev->object_name_lock);
639 + * The object name held a reference to this object, drop
642 + drm_gem_object_unreference(obj);
644 + spin_unlock(&dev->object_name_lock);
647 +EXPORT_SYMBOL(drm_gem_object_handle_free);
649 diff --git a/drivers/gpu/drm/drm_memory.c b/drivers/gpu/drm/drm_memory.c
650 index 0177012..803bc9e 100644
651 --- a/drivers/gpu/drm/drm_memory.c
652 +++ b/drivers/gpu/drm/drm_memory.c
653 @@ -133,6 +133,7 @@ int drm_free_agp(DRM_AGP_MEM * handle, int pages)
655 return drm_agp_free_memory(handle) ? 0 : -EINVAL;
657 +EXPORT_SYMBOL(drm_free_agp);
659 /** Wrapper around agp_bind_memory() */
660 int drm_bind_agp(DRM_AGP_MEM * handle, unsigned int start)
661 @@ -145,6 +146,7 @@ int drm_unbind_agp(DRM_AGP_MEM * handle)
663 return drm_agp_unbind_memory(handle);
665 +EXPORT_SYMBOL(drm_unbind_agp);
667 #else /* __OS_HAS_AGP */
668 static inline void *agp_remap(unsigned long offset, unsigned long size,
669 diff --git a/drivers/gpu/drm/drm_mm.c b/drivers/gpu/drm/drm_mm.c
670 index dcff9e9..217ad7d 100644
671 --- a/drivers/gpu/drm/drm_mm.c
672 +++ b/drivers/gpu/drm/drm_mm.c
673 @@ -169,6 +169,7 @@ struct drm_mm_node *drm_mm_get_block(struct drm_mm_node * parent,
677 +EXPORT_SYMBOL(drm_mm_get_block);
680 * Put a block. Merge with the previous and / or next block if they are free.
681 @@ -217,6 +218,7 @@ void drm_mm_put_block(struct drm_mm_node * cur)
682 drm_free(cur, sizeof(*cur), DRM_MEM_MM);
685 +EXPORT_SYMBOL(drm_mm_put_block);
687 struct drm_mm_node *drm_mm_search_free(const struct drm_mm * mm,
689 @@ -265,6 +267,7 @@ int drm_mm_clean(struct drm_mm * mm)
691 return (head->next->next == head);
693 +EXPORT_SYMBOL(drm_mm_search_free);
695 int drm_mm_init(struct drm_mm * mm, unsigned long start, unsigned long size)
697 @@ -273,7 +276,7 @@ int drm_mm_init(struct drm_mm * mm, unsigned long start, unsigned long size)
699 return drm_mm_create_tail_node(mm, start, size);
702 +EXPORT_SYMBOL(drm_mm_init);
704 void drm_mm_takedown(struct drm_mm * mm)
706 diff --git a/drivers/gpu/drm/drm_proc.c b/drivers/gpu/drm/drm_proc.c
707 index 93b1e04..d490db4 100644
708 --- a/drivers/gpu/drm/drm_proc.c
709 +++ b/drivers/gpu/drm/drm_proc.c
710 @@ -49,6 +49,10 @@ static int drm_queues_info(char *buf, char **start, off_t offset,
711 int request, int *eof, void *data);
712 static int drm_bufs_info(char *buf, char **start, off_t offset,
713 int request, int *eof, void *data);
714 +static int drm_gem_name_info(char *buf, char **start, off_t offset,
715 + int request, int *eof, void *data);
716 +static int drm_gem_object_info(char *buf, char **start, off_t offset,
717 + int request, int *eof, void *data);
719 static int drm_vma_info(char *buf, char **start, off_t offset,
720 int request, int *eof, void *data);
721 @@ -60,13 +64,16 @@ static int drm_vma_info(char *buf, char **start, off_t offset,
722 static struct drm_proc_list {
723 const char *name; /**< file name */
724 int (*f) (char *, char **, off_t, int, int *, void *); /**< proc callback*/
725 + u32 driver_features; /**< Required driver features for this entry */
726 } drm_proc_list[] = {
727 - {"name", drm_name_info},
728 - {"mem", drm_mem_info},
729 - {"vm", drm_vm_info},
730 - {"clients", drm_clients_info},
731 - {"queues", drm_queues_info},
732 - {"bufs", drm_bufs_info},
733 + {"name", drm_name_info, 0},
734 + {"mem", drm_mem_info, 0},
735 + {"vm", drm_vm_info, 0},
736 + {"clients", drm_clients_info, 0},
737 + {"queues", drm_queues_info, 0},
738 + {"bufs", drm_bufs_info, 0},
739 + {"gem_names", drm_gem_name_info, DRIVER_GEM},
740 + {"gem_objects", drm_gem_object_info, DRIVER_GEM},
742 {"vma", drm_vma_info},
744 @@ -90,8 +97,9 @@ static struct drm_proc_list {
745 int drm_proc_init(struct drm_minor *minor, int minor_id,
746 struct proc_dir_entry *root)
748 + struct drm_device *dev = minor->dev;
749 struct proc_dir_entry *ent;
754 sprintf(name, "%d", minor_id);
755 @@ -102,23 +110,42 @@ int drm_proc_init(struct drm_minor *minor, int minor_id,
758 for (i = 0; i < DRM_PROC_ENTRIES; i++) {
759 + u32 features = drm_proc_list[i].driver_features;
761 + if (features != 0 &&
762 + (dev->driver->driver_features & features) != features)
765 ent = create_proc_entry(drm_proc_list[i].name,
766 S_IFREG | S_IRUGO, minor->dev_root);
768 DRM_ERROR("Cannot create /proc/dri/%s/%s\n",
769 name, drm_proc_list[i].name);
770 - for (j = 0; j < i; j++)
771 - remove_proc_entry(drm_proc_list[i].name,
773 - remove_proc_entry(name, root);
774 - minor->dev_root = NULL;
779 ent->read_proc = drm_proc_list[i].f;
783 + if (dev->driver->proc_init) {
784 + ret = dev->driver->proc_init(minor);
786 + DRM_ERROR("DRM: Driver failed to initialize "
795 + for (j = 0; j < i; j++)
796 + remove_proc_entry(drm_proc_list[i].name,
798 + remove_proc_entry(name, root);
799 + minor->dev_root = NULL;
804 @@ -133,12 +160,16 @@ int drm_proc_init(struct drm_minor *minor, int minor_id,
806 int drm_proc_cleanup(struct drm_minor *minor, struct proc_dir_entry *root)
808 + struct drm_device *dev = minor->dev;
812 if (!root || !minor->dev_root)
815 + if (dev->driver->proc_cleanup)
816 + dev->driver->proc_cleanup(minor);
818 for (i = 0; i < DRM_PROC_ENTRIES; i++)
819 remove_proc_entry(drm_proc_list[i].name, minor->dev_root);
820 sprintf(name, "%d", minor->index);
821 @@ -480,6 +511,84 @@ static int drm_clients_info(char *buf, char **start, off_t offset,
825 +struct drm_gem_name_info_data {
831 +static int drm_gem_one_name_info(int id, void *ptr, void *data)
833 + struct drm_gem_object *obj = ptr;
834 + struct drm_gem_name_info_data *nid = data;
836 + DRM_INFO("name %d size %d\n", obj->name, obj->size);
840 + nid->len += sprintf(&nid->buf[nid->len],
842 + obj->name, obj->size,
843 + atomic_read(&obj->handlecount.refcount),
844 + atomic_read(&obj->refcount.refcount));
845 + if (nid->len > DRM_PROC_LIMIT) {
852 +static int drm_gem_name_info(char *buf, char **start, off_t offset,
853 + int request, int *eof, void *data)
855 + struct drm_minor *minor = (struct drm_minor *) data;
856 + struct drm_device *dev = minor->dev;
857 + struct drm_gem_name_info_data nid;
859 + if (offset > DRM_PROC_LIMIT) {
864 + nid.len = sprintf(buf, " name size handles refcount\n");
867 + idr_for_each(&dev->object_name_idr, drm_gem_one_name_info, &nid);
869 + *start = &buf[offset];
871 + if (nid.len > request + offset)
874 + return nid.len - offset;
877 +static int drm_gem_object_info(char *buf, char **start, off_t offset,
878 + int request, int *eof, void *data)
880 + struct drm_minor *minor = (struct drm_minor *) data;
881 + struct drm_device *dev = minor->dev;
884 + if (offset > DRM_PROC_LIMIT) {
889 + *start = &buf[offset];
891 + DRM_PROC_PRINT("%d objects\n", atomic_read(&dev->object_count));
892 + DRM_PROC_PRINT("%d object bytes\n", atomic_read(&dev->object_memory));
893 + DRM_PROC_PRINT("%d pinned\n", atomic_read(&dev->pin_count));
894 + DRM_PROC_PRINT("%d pin bytes\n", atomic_read(&dev->pin_memory));
895 + DRM_PROC_PRINT("%d gtt bytes\n", atomic_read(&dev->gtt_memory));
896 + DRM_PROC_PRINT("%d gtt total\n", dev->gtt_total);
897 + if (len > request + offset)
900 + return len - offset;
905 static int drm__vma_info(char *buf, char **start, off_t offset, int request,
906 diff --git a/drivers/gpu/drm/drm_stub.c b/drivers/gpu/drm/drm_stub.c
907 index c2f584f..82f4657 100644
908 --- a/drivers/gpu/drm/drm_stub.c
909 +++ b/drivers/gpu/drm/drm_stub.c
910 @@ -152,6 +152,15 @@ static int drm_fill_in_dev(struct drm_device * dev, struct pci_dev *pdev,
911 goto error_out_unreg;
914 + if (driver->driver_features & DRIVER_GEM) {
915 + retcode = drm_gem_init(dev);
917 + DRM_ERROR("Cannot initialize graphics execution "
918 + "manager (GEM)\n");
919 + goto error_out_unreg;
926 @@ -317,6 +326,7 @@ int drm_put_dev(struct drm_device * dev)
927 int drm_put_minor(struct drm_minor **minor_p)
929 struct drm_minor *minor = *minor_p;
931 DRM_DEBUG("release secondary minor %d\n", minor->index);
933 if (minor->type == DRM_MINOR_LEGACY)
934 diff --git a/drivers/gpu/drm/i915/Makefile b/drivers/gpu/drm/i915/Makefile
935 index c4bbda6..5ba78e4 100644
936 --- a/drivers/gpu/drm/i915/Makefile
937 +++ b/drivers/gpu/drm/i915/Makefile
940 ccflags-y := -Iinclude/drm
941 i915-y := i915_drv.o i915_dma.o i915_irq.o i915_mem.o i915_opregion.o \
949 i915-$(CONFIG_COMPAT) += i915_ioc32.o
951 diff --git a/drivers/gpu/drm/i915/i915_dma.c b/drivers/gpu/drm/i915/i915_dma.c
952 index 8609ec2..3b5aa74 100644
953 --- a/drivers/gpu/drm/i915/i915_dma.c
954 +++ b/drivers/gpu/drm/i915/i915_dma.c
955 @@ -170,24 +170,31 @@ static int i915_initialize(struct drm_device * dev, drm_i915_init_t * init)
956 dev_priv->sarea_priv = (drm_i915_sarea_t *)
957 ((u8 *) dev_priv->sarea->handle + init->sarea_priv_offset);
959 - dev_priv->ring.Start = init->ring_start;
960 - dev_priv->ring.End = init->ring_end;
961 - dev_priv->ring.Size = init->ring_size;
962 - dev_priv->ring.tail_mask = dev_priv->ring.Size - 1;
963 + if (init->ring_size != 0) {
964 + if (dev_priv->ring.ring_obj != NULL) {
965 + i915_dma_cleanup(dev);
966 + DRM_ERROR("Client tried to initialize ringbuffer in "
971 - dev_priv->ring.map.offset = init->ring_start;
972 - dev_priv->ring.map.size = init->ring_size;
973 - dev_priv->ring.map.type = 0;
974 - dev_priv->ring.map.flags = 0;
975 - dev_priv->ring.map.mtrr = 0;
976 + dev_priv->ring.Size = init->ring_size;
977 + dev_priv->ring.tail_mask = dev_priv->ring.Size - 1;
979 - drm_core_ioremap(&dev_priv->ring.map, dev);
980 + dev_priv->ring.map.offset = init->ring_start;
981 + dev_priv->ring.map.size = init->ring_size;
982 + dev_priv->ring.map.type = 0;
983 + dev_priv->ring.map.flags = 0;
984 + dev_priv->ring.map.mtrr = 0;
986 - if (dev_priv->ring.map.handle == NULL) {
987 - i915_dma_cleanup(dev);
988 - DRM_ERROR("can not ioremap virtual address for"
991 + drm_core_ioremap(&dev_priv->ring.map, dev);
993 + if (dev_priv->ring.map.handle == NULL) {
994 + i915_dma_cleanup(dev);
995 + DRM_ERROR("can not ioremap virtual address for"
1001 dev_priv->ring.virtual_start = dev_priv->ring.map.handle;
1002 @@ -377,9 +384,10 @@ static int i915_emit_cmds(struct drm_device * dev, int __user * buffer, int dwor
1006 -static int i915_emit_box(struct drm_device * dev,
1007 - struct drm_clip_rect __user * boxes,
1008 - int i, int DR1, int DR4)
1010 +i915_emit_box(struct drm_device *dev,
1011 + struct drm_clip_rect __user *boxes,
1012 + int i, int DR1, int DR4)
1014 drm_i915_private_t *dev_priv = dev->dev_private;
1015 struct drm_clip_rect box;
1016 @@ -681,6 +689,9 @@ static int i915_getparam(struct drm_device *dev, void *data,
1017 case I915_PARAM_LAST_DISPATCH:
1018 value = READ_BREADCRUMB(dev_priv);
1020 + case I915_PARAM_HAS_GEM:
1024 DRM_ERROR("Unknown parameter %d\n", param->param);
1026 @@ -784,6 +795,7 @@ int i915_driver_load(struct drm_device *dev, unsigned long flags)
1027 memset(dev_priv, 0, sizeof(drm_i915_private_t));
1029 dev->dev_private = (void *)dev_priv;
1030 + dev_priv->dev = dev;
1032 /* Add register map (needed for suspend/resume) */
1033 base = drm_get_resource_start(dev, mmio_bar);
1034 @@ -793,6 +805,8 @@ int i915_driver_load(struct drm_device *dev, unsigned long flags)
1035 _DRM_KERNEL | _DRM_DRIVER,
1036 &dev_priv->mmio_map);
1038 + i915_gem_load(dev);
1041 if (!I915_NEED_GFX_HWS(dev)) {
1042 ret = i915_init_phys_hws(dev);
1043 @@ -838,6 +852,25 @@ int i915_driver_unload(struct drm_device *dev)
1047 +int i915_driver_open(struct drm_device *dev, struct drm_file *file_priv)
1049 + struct drm_i915_file_private *i915_file_priv;
1052 + i915_file_priv = (struct drm_i915_file_private *)
1053 + drm_alloc(sizeof(*i915_file_priv), DRM_MEM_FILES);
1055 + if (!i915_file_priv)
1058 + file_priv->driver_priv = i915_file_priv;
1060 + i915_file_priv->mm.last_gem_seqno = 0;
1061 + i915_file_priv->mm.last_gem_throttle_seqno = 0;
1066 void i915_driver_lastclose(struct drm_device * dev)
1068 drm_i915_private_t *dev_priv = dev->dev_private;
1069 @@ -845,6 +878,8 @@ void i915_driver_lastclose(struct drm_device * dev)
1073 + i915_gem_lastclose(dev);
1075 if (dev_priv->agp_heap)
1076 i915_mem_takedown(&(dev_priv->agp_heap));
1078 @@ -857,6 +892,13 @@ void i915_driver_preclose(struct drm_device * dev, struct drm_file *file_priv)
1079 i915_mem_release(dev, file_priv, dev_priv->agp_heap);
1082 +void i915_driver_postclose(struct drm_device *dev, struct drm_file *file_priv)
1084 + struct drm_i915_file_private *i915_file_priv = file_priv->driver_priv;
1086 + drm_free(i915_file_priv, sizeof(*i915_file_priv), DRM_MEM_FILES);
1089 struct drm_ioctl_desc i915_ioctls[] = {
1090 DRM_IOCTL_DEF(DRM_I915_INIT, i915_dma_init, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY),
1091 DRM_IOCTL_DEF(DRM_I915_FLUSH, i915_flush_ioctl, DRM_AUTH),
1092 @@ -875,6 +917,22 @@ struct drm_ioctl_desc i915_ioctls[] = {
1093 DRM_IOCTL_DEF(DRM_I915_GET_VBLANK_PIPE, i915_vblank_pipe_get, DRM_AUTH ),
1094 DRM_IOCTL_DEF(DRM_I915_VBLANK_SWAP, i915_vblank_swap, DRM_AUTH),
1095 DRM_IOCTL_DEF(DRM_I915_HWS_ADDR, i915_set_status_page, DRM_AUTH),
1096 + DRM_IOCTL_DEF(DRM_I915_GEM_INIT, i915_gem_init_ioctl, DRM_AUTH),
1097 + DRM_IOCTL_DEF(DRM_I915_GEM_EXECBUFFER, i915_gem_execbuffer, DRM_AUTH),
1098 + DRM_IOCTL_DEF(DRM_I915_GEM_PIN, i915_gem_pin_ioctl, DRM_AUTH|DRM_ROOT_ONLY),
1099 + DRM_IOCTL_DEF(DRM_I915_GEM_UNPIN, i915_gem_unpin_ioctl, DRM_AUTH|DRM_ROOT_ONLY),
1100 + DRM_IOCTL_DEF(DRM_I915_GEM_BUSY, i915_gem_busy_ioctl, DRM_AUTH),
1101 + DRM_IOCTL_DEF(DRM_I915_GEM_THROTTLE, i915_gem_throttle_ioctl, DRM_AUTH),
1102 + DRM_IOCTL_DEF(DRM_I915_GEM_ENTERVT, i915_gem_entervt_ioctl, DRM_AUTH),
1103 + DRM_IOCTL_DEF(DRM_I915_GEM_LEAVEVT, i915_gem_leavevt_ioctl, DRM_AUTH),
1104 + DRM_IOCTL_DEF(DRM_I915_GEM_CREATE, i915_gem_create_ioctl, 0),
1105 + DRM_IOCTL_DEF(DRM_I915_GEM_PREAD, i915_gem_pread_ioctl, 0),
1106 + DRM_IOCTL_DEF(DRM_I915_GEM_PWRITE, i915_gem_pwrite_ioctl, 0),
1107 + DRM_IOCTL_DEF(DRM_I915_GEM_MMAP, i915_gem_mmap_ioctl, 0),
1108 + DRM_IOCTL_DEF(DRM_I915_GEM_SET_DOMAIN, i915_gem_set_domain_ioctl, 0),
1109 + DRM_IOCTL_DEF(DRM_I915_GEM_SW_FINISH, i915_gem_sw_finish_ioctl, 0),
1110 + DRM_IOCTL_DEF(DRM_I915_GEM_SET_TILING, i915_gem_set_tiling, 0),
1111 + DRM_IOCTL_DEF(DRM_I915_GEM_GET_TILING, i915_gem_get_tiling, 0),
1114 int i915_max_ioctl = DRM_ARRAY_SIZE(i915_ioctls);
1115 diff --git a/drivers/gpu/drm/i915/i915_drv.c b/drivers/gpu/drm/i915/i915_drv.c
1116 index 37af03f..a80ead2 100644
1117 --- a/drivers/gpu/drm/i915/i915_drv.c
1118 +++ b/drivers/gpu/drm/i915/i915_drv.c
1119 @@ -85,12 +85,15 @@ static struct drm_driver driver = {
1120 /* don't use mtrr's here, the Xserver or user space app should
1121 * deal with them for intel hardware.
1123 - .driver_features = DRIVER_USE_AGP | DRIVER_REQUIRE_AGP |
1124 - DRIVER_HAVE_IRQ | DRIVER_IRQ_SHARED,
1125 + .driver_features =
1126 + DRIVER_USE_AGP | DRIVER_REQUIRE_AGP | /* DRIVER_USE_MTRR |*/
1127 + DRIVER_HAVE_IRQ | DRIVER_IRQ_SHARED | DRIVER_GEM,
1128 .load = i915_driver_load,
1129 .unload = i915_driver_unload,
1130 + .open = i915_driver_open,
1131 .lastclose = i915_driver_lastclose,
1132 .preclose = i915_driver_preclose,
1133 + .postclose = i915_driver_postclose,
1134 .suspend = i915_suspend,
1135 .resume = i915_resume,
1136 .device_is_agp = i915_driver_device_is_agp,
1137 @@ -104,6 +107,10 @@ static struct drm_driver driver = {
1138 .reclaim_buffers = drm_core_reclaim_buffers,
1139 .get_map_ofs = drm_core_get_map_ofs,
1140 .get_reg_ofs = drm_core_get_reg_ofs,
1141 + .proc_init = i915_gem_proc_init,
1142 + .proc_cleanup = i915_gem_proc_cleanup,
1143 + .gem_init_object = i915_gem_init_object,
1144 + .gem_free_object = i915_gem_free_object,
1145 .ioctls = i915_ioctls,
1147 .owner = THIS_MODULE,
1148 diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h
1149 index d1a02be..87b071a 100644
1150 --- a/drivers/gpu/drm/i915/i915_drv.h
1151 +++ b/drivers/gpu/drm/i915/i915_drv.h
1154 #define DRIVER_NAME "i915"
1155 #define DRIVER_DESC "Intel Graphics"
1156 -#define DRIVER_DATE "20060119"
1157 +#define DRIVER_DATE "20080730"
1161 @@ -60,16 +60,23 @@ enum pipe {
1162 #define DRIVER_MINOR 6
1163 #define DRIVER_PATCHLEVEL 0
1165 +#define WATCH_COHERENCY 0
1166 +#define WATCH_BUF 0
1167 +#define WATCH_EXEC 0
1168 +#define WATCH_LRU 0
1169 +#define WATCH_RELOC 0
1170 +#define WATCH_INACTIVE 0
1171 +#define WATCH_PWRITE 0
1173 typedef struct _drm_i915_ring_buffer {
1175 - unsigned long Start;
1176 - unsigned long End;
1182 drm_local_map_t map;
1183 + struct drm_gem_object *ring_obj;
1184 } drm_i915_ring_buffer_t;
1187 @@ -101,6 +108,8 @@ struct intel_opregion {
1190 typedef struct drm_i915_private {
1191 + struct drm_device *dev;
1193 drm_local_map_t *sarea;
1194 drm_local_map_t *mmio_map;
1196 @@ -113,6 +122,7 @@ typedef struct drm_i915_private {
1198 unsigned int status_gfx_addr;
1199 drm_local_map_t hws_map;
1200 + struct drm_gem_object *hws_obj;
1204 @@ -122,7 +132,6 @@ typedef struct drm_i915_private {
1206 wait_queue_head_t irq_queue;
1207 atomic_t irq_received;
1208 - atomic_t irq_emitted;
1209 /** Protects user_irq_refcount and irq_mask_reg */
1210 spinlock_t user_irq_lock;
1211 /** Refcount for i915_user_irq_get() versus i915_user_irq_put(). */
1212 @@ -230,8 +239,174 @@ typedef struct drm_i915_private {
1214 u8 saveDACDATA[256*3]; /* 256 3-byte colors */
1218 + struct drm_mm gtt_space;
1221 + * List of objects currently involved in rendering from the
1224 + * A reference is held on the buffer while on this list.
1226 + struct list_head active_list;
1229 + * List of objects which are not in the ringbuffer but which
1230 + * still have a write_domain which needs to be flushed before
1233 + * A reference is held on the buffer while on this list.
1235 + struct list_head flushing_list;
1238 + * LRU list of objects which are not in the ringbuffer and
1239 + * are ready to unbind, but are still in the GTT.
1241 + * A reference is not held on the buffer while on this list,
1242 + * as merely being GTT-bound shouldn't prevent its being
1243 + * freed, and we'll pull it off the list in the free path.
1245 + struct list_head inactive_list;
1248 + * List of breadcrumbs associated with GPU requests currently
1251 + struct list_head request_list;
1254 + * We leave the user IRQ off as much as possible,
1255 + * but this means that requests will finish and never
1256 + * be retired once the system goes idle. Set a timer to
1257 + * fire periodically while the ring is running. When it
1258 + * fires, go retire requests.
1260 + struct delayed_work retire_work;
1262 + uint32_t next_gem_seqno;
1265 + * Waiting sequence number, if any
1267 + uint32_t waiting_gem_seqno;
1270 + * Last seq seen at irq time
1272 + uint32_t irq_gem_seqno;
1275 + * Flag if the X Server, and thus DRM, is not currently in
1276 + * control of the device.
1278 + * This is set between LeaveVT and EnterVT. It needs to be
1279 + * replaced with a semaphore. It also needs to be
1280 + * transitioned away from for kernel modesetting.
1285 + * Flag if the hardware appears to be wedged.
1287 + * This is set when attempts to idle the device timeout.
1288 + * It prevents command submission from occuring and makes
1289 + * every pending request fail
1293 + /** Bit 6 swizzling required for X tiling */
1294 + uint32_t bit_6_swizzle_x;
1295 + /** Bit 6 swizzling required for Y tiling */
1296 + uint32_t bit_6_swizzle_y;
1298 } drm_i915_private_t;
1300 +/** driver private structure attached to each drm_gem_object */
1301 +struct drm_i915_gem_object {
1302 + struct drm_gem_object *obj;
1304 + /** Current space allocated to this object in the GTT, if any. */
1305 + struct drm_mm_node *gtt_space;
1307 + /** This object's place on the active/flushing/inactive lists */
1308 + struct list_head list;
1311 + * This is set if the object is on the active or flushing lists
1312 + * (has pending rendering), and is not set if it's on inactive (ready
1318 + * This is set if the object has been written to since last bound
1323 + /** AGP memory structure for our GTT binding. */
1324 + DRM_AGP_MEM *agp_mem;
1326 + struct page **page_list;
1329 + * Current offset of the object in GTT space.
1331 + * This is the same as gtt_space->start
1333 + uint32_t gtt_offset;
1335 + /** Boolean whether this object has a valid gtt offset. */
1338 + /** How many users have pinned this object in GTT space */
1341 + /** Breadcrumb of last rendering to the buffer. */
1342 + uint32_t last_rendering_seqno;
1344 + /** Current tiling mode for the object. */
1345 + uint32_t tiling_mode;
1348 + * Flagging of which individual pages are valid in GEM_DOMAIN_CPU when
1349 + * GEM_DOMAIN_CPU is not in the object's read domain.
1351 + uint8_t *page_cpu_valid;
1355 + * Request queue structure.
1357 + * The request queue allows us to note sequence numbers that have been emitted
1358 + * and may be associated with active buffers to be retired.
1360 + * By keeping this list, we can avoid having to do questionable
1361 + * sequence-number comparisons on buffer last_rendering_seqnos, and associate
1362 + * an emission time with seqnos for tracking how far ahead of the GPU we are.
1364 +struct drm_i915_gem_request {
1365 + /** GEM sequence number associated with this request. */
1368 + /** Time at which this request was emitted, in jiffies. */
1369 + unsigned long emitted_jiffies;
1371 + /** Cache domains that were flushed at the start of the request. */
1372 + uint32_t flush_domains;
1374 + struct list_head list;
1377 +struct drm_i915_file_private {
1379 + uint32_t last_gem_seqno;
1380 + uint32_t last_gem_throttle_seqno;
1384 extern struct drm_ioctl_desc i915_ioctls[];
1385 extern int i915_max_ioctl;
1387 @@ -239,18 +414,26 @@ extern int i915_max_ioctl;
1388 extern void i915_kernel_lost_context(struct drm_device * dev);
1389 extern int i915_driver_load(struct drm_device *, unsigned long flags);
1390 extern int i915_driver_unload(struct drm_device *);
1391 +extern int i915_driver_open(struct drm_device *dev, struct drm_file *file_priv);
1392 extern void i915_driver_lastclose(struct drm_device * dev);
1393 extern void i915_driver_preclose(struct drm_device *dev,
1394 struct drm_file *file_priv);
1395 +extern void i915_driver_postclose(struct drm_device *dev,
1396 + struct drm_file *file_priv);
1397 extern int i915_driver_device_is_agp(struct drm_device * dev);
1398 extern long i915_compat_ioctl(struct file *filp, unsigned int cmd,
1400 +extern int i915_emit_box(struct drm_device *dev,
1401 + struct drm_clip_rect __user *boxes,
1402 + int i, int DR1, int DR4);
1405 extern int i915_irq_emit(struct drm_device *dev, void *data,
1406 struct drm_file *file_priv);
1407 extern int i915_irq_wait(struct drm_device *dev, void *data,
1408 struct drm_file *file_priv);
1409 +void i915_user_irq_get(struct drm_device *dev);
1410 +void i915_user_irq_put(struct drm_device *dev);
1412 extern irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS);
1413 extern void i915_driver_irq_preinstall(struct drm_device * dev);
1414 @@ -279,6 +462,67 @@ extern int i915_mem_destroy_heap(struct drm_device *dev, void *data,
1415 extern void i915_mem_takedown(struct mem_block **heap);
1416 extern void i915_mem_release(struct drm_device * dev,
1417 struct drm_file *file_priv, struct mem_block *heap);
1419 +int i915_gem_init_ioctl(struct drm_device *dev, void *data,
1420 + struct drm_file *file_priv);
1421 +int i915_gem_create_ioctl(struct drm_device *dev, void *data,
1422 + struct drm_file *file_priv);
1423 +int i915_gem_pread_ioctl(struct drm_device *dev, void *data,
1424 + struct drm_file *file_priv);
1425 +int i915_gem_pwrite_ioctl(struct drm_device *dev, void *data,
1426 + struct drm_file *file_priv);
1427 +int i915_gem_mmap_ioctl(struct drm_device *dev, void *data,
1428 + struct drm_file *file_priv);
1429 +int i915_gem_set_domain_ioctl(struct drm_device *dev, void *data,
1430 + struct drm_file *file_priv);
1431 +int i915_gem_sw_finish_ioctl(struct drm_device *dev, void *data,
1432 + struct drm_file *file_priv);
1433 +int i915_gem_execbuffer(struct drm_device *dev, void *data,
1434 + struct drm_file *file_priv);
1435 +int i915_gem_pin_ioctl(struct drm_device *dev, void *data,
1436 + struct drm_file *file_priv);
1437 +int i915_gem_unpin_ioctl(struct drm_device *dev, void *data,
1438 + struct drm_file *file_priv);
1439 +int i915_gem_busy_ioctl(struct drm_device *dev, void *data,
1440 + struct drm_file *file_priv);
1441 +int i915_gem_throttle_ioctl(struct drm_device *dev, void *data,
1442 + struct drm_file *file_priv);
1443 +int i915_gem_entervt_ioctl(struct drm_device *dev, void *data,
1444 + struct drm_file *file_priv);
1445 +int i915_gem_leavevt_ioctl(struct drm_device *dev, void *data,
1446 + struct drm_file *file_priv);
1447 +int i915_gem_set_tiling(struct drm_device *dev, void *data,
1448 + struct drm_file *file_priv);
1449 +int i915_gem_get_tiling(struct drm_device *dev, void *data,
1450 + struct drm_file *file_priv);
1451 +void i915_gem_load(struct drm_device *dev);
1452 +int i915_gem_proc_init(struct drm_minor *minor);
1453 +void i915_gem_proc_cleanup(struct drm_minor *minor);
1454 +int i915_gem_init_object(struct drm_gem_object *obj);
1455 +void i915_gem_free_object(struct drm_gem_object *obj);
1456 +int i915_gem_object_pin(struct drm_gem_object *obj, uint32_t alignment);
1457 +void i915_gem_object_unpin(struct drm_gem_object *obj);
1458 +void i915_gem_lastclose(struct drm_device *dev);
1459 +uint32_t i915_get_gem_seqno(struct drm_device *dev);
1460 +void i915_gem_retire_requests(struct drm_device *dev);
1461 +void i915_gem_retire_work_handler(struct work_struct *work);
1462 +void i915_gem_clflush_object(struct drm_gem_object *obj);
1464 +/* i915_gem_tiling.c */
1465 +void i915_gem_detect_bit_6_swizzle(struct drm_device *dev);
1467 +/* i915_gem_debug.c */
1468 +void i915_gem_dump_object(struct drm_gem_object *obj, int len,
1469 + const char *where, uint32_t mark);
1471 +void i915_verify_inactive(struct drm_device *dev, char *file, int line);
1473 +#define i915_verify_inactive(dev, file, line)
1475 +void i915_gem_object_check_coherency(struct drm_gem_object *obj, int handle);
1476 +void i915_gem_dump_object(struct drm_gem_object *obj, int len,
1477 + const char *where, uint32_t mark);
1478 +void i915_dump_lru(struct drm_device *dev, const char *where);
1480 /* i915_suspend.c */
1481 extern int i915_save_state(struct drm_device *dev);
1482 @@ -347,6 +591,7 @@ extern void opregion_enable_asle(struct drm_device *dev);
1484 #define READ_HWSP(dev_priv, reg) (((volatile u32*)(dev_priv->hw_status_page))[reg])
1485 #define READ_BREADCRUMB(dev_priv) READ_HWSP(dev_priv, 5)
1486 +#define I915_GEM_HWS_INDEX 0x10
1488 extern int i915_wait_ring(struct drm_device * dev, int n, const char *caller);
1490 diff --git a/drivers/gpu/drm/i915/i915_gem.c b/drivers/gpu/drm/i915/i915_gem.c
1491 new file mode 100644
1492 index 0000000..90ae8a0
1494 +++ b/drivers/gpu/drm/i915/i915_gem.c
1497 + * Copyright © 2008 Intel Corporation
1499 + * Permission is hereby granted, free of charge, to any person obtaining a
1500 + * copy of this software and associated documentation files (the "Software"),
1501 + * to deal in the Software without restriction, including without limitation
1502 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1503 + * and/or sell copies of the Software, and to permit persons to whom the
1504 + * Software is furnished to do so, subject to the following conditions:
1506 + * The above copyright notice and this permission notice (including the next
1507 + * paragraph) shall be included in all copies or substantial portions of the
1510 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1511 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1512 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1513 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1514 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1515 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
1516 + * IN THE SOFTWARE.
1519 + * Eric Anholt <eric@anholt.net>
1525 +#include "i915_drm.h"
1526 +#include "i915_drv.h"
1527 +#include <linux/swap.h>
1530 +i915_gem_object_set_domain(struct drm_gem_object *obj,
1531 + uint32_t read_domains,
1532 + uint32_t write_domain);
1534 +i915_gem_object_set_domain_range(struct drm_gem_object *obj,
1537 + uint32_t read_domains,
1538 + uint32_t write_domain);
1540 +i915_gem_set_domain(struct drm_gem_object *obj,
1541 + struct drm_file *file_priv,
1542 + uint32_t read_domains,
1543 + uint32_t write_domain);
1544 +static int i915_gem_object_get_page_list(struct drm_gem_object *obj);
1545 +static void i915_gem_object_free_page_list(struct drm_gem_object *obj);
1546 +static int i915_gem_object_wait_rendering(struct drm_gem_object *obj);
1549 +i915_gem_init_ioctl(struct drm_device *dev, void *data,
1550 + struct drm_file *file_priv)
1552 + drm_i915_private_t *dev_priv = dev->dev_private;
1553 + struct drm_i915_gem_init *args = data;
1555 + mutex_lock(&dev->struct_mutex);
1557 + if (args->gtt_start >= args->gtt_end ||
1558 + (args->gtt_start & (PAGE_SIZE - 1)) != 0 ||
1559 + (args->gtt_end & (PAGE_SIZE - 1)) != 0) {
1560 + mutex_unlock(&dev->struct_mutex);
1564 + drm_mm_init(&dev_priv->mm.gtt_space, args->gtt_start,
1565 + args->gtt_end - args->gtt_start);
1567 + dev->gtt_total = (uint32_t) (args->gtt_end - args->gtt_start);
1569 + mutex_unlock(&dev->struct_mutex);
1576 + * Creates a new mm object and returns a handle to it.
1579 +i915_gem_create_ioctl(struct drm_device *dev, void *data,
1580 + struct drm_file *file_priv)
1582 + struct drm_i915_gem_create *args = data;
1583 + struct drm_gem_object *obj;
1586 + args->size = roundup(args->size, PAGE_SIZE);
1588 + /* Allocate the new object */
1589 + obj = drm_gem_object_alloc(dev, args->size);
1593 + ret = drm_gem_handle_create(file_priv, obj, &handle);
1594 + mutex_lock(&dev->struct_mutex);
1595 + drm_gem_object_handle_unreference(obj);
1596 + mutex_unlock(&dev->struct_mutex);
1601 + args->handle = handle;
1607 + * Reads data from the object referenced by handle.
1609 + * On error, the contents of *data are undefined.
1612 +i915_gem_pread_ioctl(struct drm_device *dev, void *data,
1613 + struct drm_file *file_priv)
1615 + struct drm_i915_gem_pread *args = data;
1616 + struct drm_gem_object *obj;
1617 + struct drm_i915_gem_object *obj_priv;
1622 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
1625 + obj_priv = obj->driver_private;
1627 + /* Bounds check source.
1629 + * XXX: This could use review for overflow issues...
1631 + if (args->offset > obj->size || args->size > obj->size ||
1632 + args->offset + args->size > obj->size) {
1633 + drm_gem_object_unreference(obj);
1637 + mutex_lock(&dev->struct_mutex);
1639 + ret = i915_gem_object_set_domain_range(obj, args->offset, args->size,
1640 + I915_GEM_DOMAIN_CPU, 0);
1642 + drm_gem_object_unreference(obj);
1643 + mutex_unlock(&dev->struct_mutex);
1646 + offset = args->offset;
1648 + read = vfs_read(obj->filp, (char __user *)(uintptr_t)args->data_ptr,
1649 + args->size, &offset);
1650 + if (read != args->size) {
1651 + drm_gem_object_unreference(obj);
1652 + mutex_unlock(&dev->struct_mutex);
1659 + drm_gem_object_unreference(obj);
1660 + mutex_unlock(&dev->struct_mutex);
1666 +i915_gem_gtt_pwrite(struct drm_device *dev, struct drm_gem_object *obj,
1667 + struct drm_i915_gem_pwrite *args,
1668 + struct drm_file *file_priv)
1670 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
1673 + char __user *user_data;
1677 + unsigned long pfn;
1678 + unsigned long unwritten;
1680 + user_data = (char __user *) (uintptr_t) args->data_ptr;
1681 + remain = args->size;
1682 + if (!access_ok(VERIFY_READ, user_data, remain))
1686 + mutex_lock(&dev->struct_mutex);
1687 + ret = i915_gem_object_pin(obj, 0);
1689 + mutex_unlock(&dev->struct_mutex);
1692 + ret = i915_gem_set_domain(obj, file_priv,
1693 + I915_GEM_DOMAIN_GTT, I915_GEM_DOMAIN_GTT);
1697 + obj_priv = obj->driver_private;
1698 + offset = obj_priv->gtt_offset + args->offset;
1699 + obj_priv->dirty = 1;
1701 + while (remain > 0) {
1702 + /* Operation in this page
1705 + * o = offset within page
1706 + * l = bytes to copy
1708 + i = offset >> PAGE_SHIFT;
1709 + o = offset & (PAGE_SIZE-1);
1711 + if ((o + l) > PAGE_SIZE)
1712 + l = PAGE_SIZE - o;
1714 + pfn = (dev->agp->base >> PAGE_SHIFT) + i;
1716 +#ifdef CONFIG_HIGHMEM
1717 + /* kmap_atomic can't map IO pages on non-HIGHMEM kernels
1719 + vaddr = kmap_atomic_pfn(pfn, KM_USER0);
1721 + DRM_INFO("pwrite i %d o %d l %d pfn %ld vaddr %p\n",
1722 + i, o, l, pfn, vaddr);
1724 + unwritten = __copy_from_user_inatomic_nocache(vaddr + o,
1726 + kunmap_atomic(vaddr, KM_USER0);
1729 +#endif /* CONFIG_HIGHMEM */
1731 + vaddr = ioremap(pfn << PAGE_SHIFT, PAGE_SIZE);
1733 + DRM_INFO("pwrite slow i %d o %d l %d "
1734 + "pfn %ld vaddr %p\n",
1735 + i, o, l, pfn, vaddr);
1737 + if (vaddr == NULL) {
1741 + unwritten = __copy_from_user(vaddr + o, user_data, l);
1743 + DRM_INFO("unwritten %ld\n", unwritten);
1756 +#if WATCH_PWRITE && 1
1757 + i915_gem_clflush_object(obj);
1758 + i915_gem_dump_object(obj, args->offset + args->size, __func__, ~0);
1759 + i915_gem_clflush_object(obj);
1763 + i915_gem_object_unpin(obj);
1764 + mutex_unlock(&dev->struct_mutex);
1770 +i915_gem_shmem_pwrite(struct drm_device *dev, struct drm_gem_object *obj,
1771 + struct drm_i915_gem_pwrite *args,
1772 + struct drm_file *file_priv)
1778 + mutex_lock(&dev->struct_mutex);
1780 + ret = i915_gem_set_domain(obj, file_priv,
1781 + I915_GEM_DOMAIN_CPU, I915_GEM_DOMAIN_CPU);
1783 + mutex_unlock(&dev->struct_mutex);
1787 + offset = args->offset;
1789 + written = vfs_write(obj->filp,
1790 + (char __user *)(uintptr_t) args->data_ptr,
1791 + args->size, &offset);
1792 + if (written != args->size) {
1793 + mutex_unlock(&dev->struct_mutex);
1800 + mutex_unlock(&dev->struct_mutex);
1806 + * Writes data to the object referenced by handle.
1808 + * On error, the contents of the buffer that were to be modified are undefined.
1811 +i915_gem_pwrite_ioctl(struct drm_device *dev, void *data,
1812 + struct drm_file *file_priv)
1814 + struct drm_i915_gem_pwrite *args = data;
1815 + struct drm_gem_object *obj;
1816 + struct drm_i915_gem_object *obj_priv;
1819 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
1822 + obj_priv = obj->driver_private;
1824 + /* Bounds check destination.
1826 + * XXX: This could use review for overflow issues...
1828 + if (args->offset > obj->size || args->size > obj->size ||
1829 + args->offset + args->size > obj->size) {
1830 + drm_gem_object_unreference(obj);
1834 + /* We can only do the GTT pwrite on untiled buffers, as otherwise
1835 + * it would end up going through the fenced access, and we'll get
1836 + * different detiling behavior between reading and writing.
1837 + * pread/pwrite currently are reading and writing from the CPU
1838 + * perspective, requiring manual detiling by the client.
1840 + if (obj_priv->tiling_mode == I915_TILING_NONE &&
1841 + dev->gtt_total != 0)
1842 + ret = i915_gem_gtt_pwrite(dev, obj, args, file_priv);
1844 + ret = i915_gem_shmem_pwrite(dev, obj, args, file_priv);
1848 + DRM_INFO("pwrite failed %d\n", ret);
1851 + drm_gem_object_unreference(obj);
1857 + * Called when user space prepares to use an object
1860 +i915_gem_set_domain_ioctl(struct drm_device *dev, void *data,
1861 + struct drm_file *file_priv)
1863 + struct drm_i915_gem_set_domain *args = data;
1864 + struct drm_gem_object *obj;
1867 + if (!(dev->driver->driver_features & DRIVER_GEM))
1870 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
1874 + mutex_lock(&dev->struct_mutex);
1876 + DRM_INFO("set_domain_ioctl %p(%d), %08x %08x\n",
1877 + obj, obj->size, args->read_domains, args->write_domain);
1879 + ret = i915_gem_set_domain(obj, file_priv,
1880 + args->read_domains, args->write_domain);
1881 + drm_gem_object_unreference(obj);
1882 + mutex_unlock(&dev->struct_mutex);
1887 + * Called when user space has done writes to this buffer
1890 +i915_gem_sw_finish_ioctl(struct drm_device *dev, void *data,
1891 + struct drm_file *file_priv)
1893 + struct drm_i915_gem_sw_finish *args = data;
1894 + struct drm_gem_object *obj;
1895 + struct drm_i915_gem_object *obj_priv;
1898 + if (!(dev->driver->driver_features & DRIVER_GEM))
1901 + mutex_lock(&dev->struct_mutex);
1902 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
1903 + if (obj == NULL) {
1904 + mutex_unlock(&dev->struct_mutex);
1909 + DRM_INFO("%s: sw_finish %d (%p %d)\n",
1910 + __func__, args->handle, obj, obj->size);
1912 + obj_priv = obj->driver_private;
1914 + /* Pinned buffers may be scanout, so flush the cache */
1915 + if ((obj->write_domain & I915_GEM_DOMAIN_CPU) && obj_priv->pin_count) {
1916 + i915_gem_clflush_object(obj);
1917 + drm_agp_chipset_flush(dev);
1919 + drm_gem_object_unreference(obj);
1920 + mutex_unlock(&dev->struct_mutex);
1925 + * Maps the contents of an object, returning the address it is mapped
1928 + * While the mapping holds a reference on the contents of the object, it doesn't
1929 + * imply a ref on the object itself.
1932 +i915_gem_mmap_ioctl(struct drm_device *dev, void *data,
1933 + struct drm_file *file_priv)
1935 + struct drm_i915_gem_mmap *args = data;
1936 + struct drm_gem_object *obj;
1938 + unsigned long addr;
1940 + if (!(dev->driver->driver_features & DRIVER_GEM))
1943 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
1947 + offset = args->offset;
1949 + down_write(¤t->mm->mmap_sem);
1950 + addr = do_mmap(obj->filp, 0, args->size,
1951 + PROT_READ | PROT_WRITE, MAP_SHARED,
1953 + up_write(¤t->mm->mmap_sem);
1954 + mutex_lock(&dev->struct_mutex);
1955 + drm_gem_object_unreference(obj);
1956 + mutex_unlock(&dev->struct_mutex);
1957 + if (IS_ERR((void *)addr))
1960 + args->addr_ptr = (uint64_t) addr;
1966 +i915_gem_object_free_page_list(struct drm_gem_object *obj)
1968 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
1969 + int page_count = obj->size / PAGE_SIZE;
1972 + if (obj_priv->page_list == NULL)
1976 + for (i = 0; i < page_count; i++)
1977 + if (obj_priv->page_list[i] != NULL) {
1978 + if (obj_priv->dirty)
1979 + set_page_dirty(obj_priv->page_list[i]);
1980 + mark_page_accessed(obj_priv->page_list[i]);
1981 + page_cache_release(obj_priv->page_list[i]);
1983 + obj_priv->dirty = 0;
1985 + drm_free(obj_priv->page_list,
1986 + page_count * sizeof(struct page *),
1988 + obj_priv->page_list = NULL;
1992 +i915_gem_object_move_to_active(struct drm_gem_object *obj)
1994 + struct drm_device *dev = obj->dev;
1995 + drm_i915_private_t *dev_priv = dev->dev_private;
1996 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
1998 + /* Add a reference if we're newly entering the active list. */
1999 + if (!obj_priv->active) {
2000 + drm_gem_object_reference(obj);
2001 + obj_priv->active = 1;
2003 + /* Move from whatever list we were on to the tail of execution. */
2004 + list_move_tail(&obj_priv->list,
2005 + &dev_priv->mm.active_list);
2010 +i915_gem_object_move_to_inactive(struct drm_gem_object *obj)
2012 + struct drm_device *dev = obj->dev;
2013 + drm_i915_private_t *dev_priv = dev->dev_private;
2014 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2016 + i915_verify_inactive(dev, __FILE__, __LINE__);
2017 + if (obj_priv->pin_count != 0)
2018 + list_del_init(&obj_priv->list);
2020 + list_move_tail(&obj_priv->list, &dev_priv->mm.inactive_list);
2022 + if (obj_priv->active) {
2023 + obj_priv->active = 0;
2024 + drm_gem_object_unreference(obj);
2026 + i915_verify_inactive(dev, __FILE__, __LINE__);
2030 + * Creates a new sequence number, emitting a write of it to the status page
2031 + * plus an interrupt, which will trigger i915_user_interrupt_handler.
2033 + * Must be called with struct_lock held.
2035 + * Returned sequence numbers are nonzero on success.
2038 +i915_add_request(struct drm_device *dev, uint32_t flush_domains)
2040 + drm_i915_private_t *dev_priv = dev->dev_private;
2041 + struct drm_i915_gem_request *request;
2046 + request = drm_calloc(1, sizeof(*request), DRM_MEM_DRIVER);
2047 + if (request == NULL)
2050 + /* Grab the seqno we're going to make this request be, and bump the
2051 + * next (skipping 0 so it can be the reserved no-seqno value).
2053 + seqno = dev_priv->mm.next_gem_seqno;
2054 + dev_priv->mm.next_gem_seqno++;
2055 + if (dev_priv->mm.next_gem_seqno == 0)
2056 + dev_priv->mm.next_gem_seqno++;
2059 + OUT_RING(MI_STORE_DWORD_INDEX);
2060 + OUT_RING(I915_GEM_HWS_INDEX << MI_STORE_DWORD_INDEX_SHIFT);
2063 + OUT_RING(MI_USER_INTERRUPT);
2064 + ADVANCE_LP_RING();
2066 + DRM_DEBUG("%d\n", seqno);
2068 + request->seqno = seqno;
2069 + request->emitted_jiffies = jiffies;
2070 + request->flush_domains = flush_domains;
2071 + was_empty = list_empty(&dev_priv->mm.request_list);
2072 + list_add_tail(&request->list, &dev_priv->mm.request_list);
2075 + schedule_delayed_work(&dev_priv->mm.retire_work, HZ);
2080 + * Command execution barrier
2082 + * Ensures that all commands in the ring are finished
2083 + * before signalling the CPU
2086 +i915_retire_commands(struct drm_device *dev)
2088 + drm_i915_private_t *dev_priv = dev->dev_private;
2089 + uint32_t cmd = MI_FLUSH | MI_NO_WRITE_FLUSH;
2090 + uint32_t flush_domains = 0;
2093 + /* The sampler always gets flushed on i965 (sigh) */
2094 + if (IS_I965G(dev))
2095 + flush_domains |= I915_GEM_DOMAIN_SAMPLER;
2098 + OUT_RING(0); /* noop */
2099 + ADVANCE_LP_RING();
2100 + return flush_domains;
2104 + * Moves buffers associated only with the given active seqno from the active
2105 + * to inactive list, potentially freeing them.
2108 +i915_gem_retire_request(struct drm_device *dev,
2109 + struct drm_i915_gem_request *request)
2111 + drm_i915_private_t *dev_priv = dev->dev_private;
2113 + /* Move any buffers on the active list that are no longer referenced
2114 + * by the ringbuffer to the flushing/inactive lists as appropriate.
2116 + while (!list_empty(&dev_priv->mm.active_list)) {
2117 + struct drm_gem_object *obj;
2118 + struct drm_i915_gem_object *obj_priv;
2120 + obj_priv = list_first_entry(&dev_priv->mm.active_list,
2121 + struct drm_i915_gem_object,
2123 + obj = obj_priv->obj;
2125 + /* If the seqno being retired doesn't match the oldest in the
2126 + * list, then the oldest in the list must still be newer than
2129 + if (obj_priv->last_rendering_seqno != request->seqno)
2132 + DRM_INFO("%s: retire %d moves to inactive list %p\n",
2133 + __func__, request->seqno, obj);
2136 + if (obj->write_domain != 0) {
2137 + list_move_tail(&obj_priv->list,
2138 + &dev_priv->mm.flushing_list);
2140 + i915_gem_object_move_to_inactive(obj);
2144 + if (request->flush_domains != 0) {
2145 + struct drm_i915_gem_object *obj_priv, *next;
2147 + /* Clear the write domain and activity from any buffers
2148 + * that are just waiting for a flush matching the one retired.
2150 + list_for_each_entry_safe(obj_priv, next,
2151 + &dev_priv->mm.flushing_list, list) {
2152 + struct drm_gem_object *obj = obj_priv->obj;
2154 + if (obj->write_domain & request->flush_domains) {
2155 + obj->write_domain = 0;
2156 + i915_gem_object_move_to_inactive(obj);
2164 + * Returns true if seq1 is later than seq2.
2167 +i915_seqno_passed(uint32_t seq1, uint32_t seq2)
2169 + return (int32_t)(seq1 - seq2) >= 0;
2173 +i915_get_gem_seqno(struct drm_device *dev)
2175 + drm_i915_private_t *dev_priv = dev->dev_private;
2177 + return READ_HWSP(dev_priv, I915_GEM_HWS_INDEX);
2181 + * This function clears the request list as sequence numbers are passed.
2184 +i915_gem_retire_requests(struct drm_device *dev)
2186 + drm_i915_private_t *dev_priv = dev->dev_private;
2189 + seqno = i915_get_gem_seqno(dev);
2191 + while (!list_empty(&dev_priv->mm.request_list)) {
2192 + struct drm_i915_gem_request *request;
2193 + uint32_t retiring_seqno;
2195 + request = list_first_entry(&dev_priv->mm.request_list,
2196 + struct drm_i915_gem_request,
2198 + retiring_seqno = request->seqno;
2200 + if (i915_seqno_passed(seqno, retiring_seqno) ||
2201 + dev_priv->mm.wedged) {
2202 + i915_gem_retire_request(dev, request);
2204 + list_del(&request->list);
2205 + drm_free(request, sizeof(*request), DRM_MEM_DRIVER);
2212 +i915_gem_retire_work_handler(struct work_struct *work)
2214 + drm_i915_private_t *dev_priv;
2215 + struct drm_device *dev;
2217 + dev_priv = container_of(work, drm_i915_private_t,
2218 + mm.retire_work.work);
2219 + dev = dev_priv->dev;
2221 + mutex_lock(&dev->struct_mutex);
2222 + i915_gem_retire_requests(dev);
2223 + if (!list_empty(&dev_priv->mm.request_list))
2224 + schedule_delayed_work(&dev_priv->mm.retire_work, HZ);
2225 + mutex_unlock(&dev->struct_mutex);
2229 + * Waits for a sequence number to be signaled, and cleans up the
2230 + * request and object lists appropriately for that event.
2233 +i915_wait_request(struct drm_device *dev, uint32_t seqno)
2235 + drm_i915_private_t *dev_priv = dev->dev_private;
2238 + BUG_ON(seqno == 0);
2240 + if (!i915_seqno_passed(i915_get_gem_seqno(dev), seqno)) {
2241 + dev_priv->mm.waiting_gem_seqno = seqno;
2242 + i915_user_irq_get(dev);
2243 + ret = wait_event_interruptible(dev_priv->irq_queue,
2244 + i915_seqno_passed(i915_get_gem_seqno(dev),
2246 + dev_priv->mm.wedged);
2247 + i915_user_irq_put(dev);
2248 + dev_priv->mm.waiting_gem_seqno = 0;
2250 + if (dev_priv->mm.wedged)
2253 + if (ret && ret != -ERESTARTSYS)
2254 + DRM_ERROR("%s returns %d (awaiting %d at %d)\n",
2255 + __func__, ret, seqno, i915_get_gem_seqno(dev));
2257 + /* Directly dispatch request retiring. While we have the work queue
2258 + * to handle this, the waiter on a request often wants an associated
2259 + * buffer to have made it to the inactive list, and we would need
2260 + * a separate wait queue to handle that.
2263 + i915_gem_retire_requests(dev);
2269 +i915_gem_flush(struct drm_device *dev,
2270 + uint32_t invalidate_domains,
2271 + uint32_t flush_domains)
2273 + drm_i915_private_t *dev_priv = dev->dev_private;
2278 + DRM_INFO("%s: invalidate %08x flush %08x\n", __func__,
2279 + invalidate_domains, flush_domains);
2282 + if (flush_domains & I915_GEM_DOMAIN_CPU)
2283 + drm_agp_chipset_flush(dev);
2285 + if ((invalidate_domains | flush_domains) & ~(I915_GEM_DOMAIN_CPU |
2286 + I915_GEM_DOMAIN_GTT)) {
2288 + * read/write caches:
2290 + * I915_GEM_DOMAIN_RENDER is always invalidated, but is
2291 + * only flushed if MI_NO_WRITE_FLUSH is unset. On 965, it is
2292 + * also flushed at 2d versus 3d pipeline switches.
2294 + * read-only caches:
2296 + * I915_GEM_DOMAIN_SAMPLER is flushed on pre-965 if
2297 + * MI_READ_FLUSH is set, and is always flushed on 965.
2299 + * I915_GEM_DOMAIN_COMMAND may not exist?
2301 + * I915_GEM_DOMAIN_INSTRUCTION, which exists on 965, is
2302 + * invalidated when MI_EXE_FLUSH is set.
2304 + * I915_GEM_DOMAIN_VERTEX, which exists on 965, is
2305 + * invalidated with every MI_FLUSH.
2309 + * On 965, TLBs associated with I915_GEM_DOMAIN_COMMAND
2310 + * and I915_GEM_DOMAIN_CPU in are invalidated at PTE write and
2311 + * I915_GEM_DOMAIN_RENDER and I915_GEM_DOMAIN_SAMPLER
2312 + * are flushed at any MI_FLUSH.
2315 + cmd = MI_FLUSH | MI_NO_WRITE_FLUSH;
2316 + if ((invalidate_domains|flush_domains) &
2317 + I915_GEM_DOMAIN_RENDER)
2318 + cmd &= ~MI_NO_WRITE_FLUSH;
2319 + if (!IS_I965G(dev)) {
2321 + * On the 965, the sampler cache always gets flushed
2322 + * and this bit is reserved.
2324 + if (invalidate_domains & I915_GEM_DOMAIN_SAMPLER)
2325 + cmd |= MI_READ_FLUSH;
2327 + if (invalidate_domains & I915_GEM_DOMAIN_INSTRUCTION)
2328 + cmd |= MI_EXE_FLUSH;
2331 + DRM_INFO("%s: queue flush %08x to ring\n", __func__, cmd);
2335 + OUT_RING(0); /* noop */
2336 + ADVANCE_LP_RING();
2341 + * Ensures that all rendering to the object has completed and the object is
2342 + * safe to unbind from the GTT or access from the CPU.
2345 +i915_gem_object_wait_rendering(struct drm_gem_object *obj)
2347 + struct drm_device *dev = obj->dev;
2348 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2351 + /* If there are writes queued to the buffer, flush and
2352 + * create a new seqno to wait for.
2354 + if (obj->write_domain & ~(I915_GEM_DOMAIN_CPU|I915_GEM_DOMAIN_GTT)) {
2355 + uint32_t write_domain = obj->write_domain;
2357 + DRM_INFO("%s: flushing object %p from write domain %08x\n",
2358 + __func__, obj, write_domain);
2360 + i915_gem_flush(dev, 0, write_domain);
2362 + i915_gem_object_move_to_active(obj);
2363 + obj_priv->last_rendering_seqno = i915_add_request(dev,
2365 + BUG_ON(obj_priv->last_rendering_seqno == 0);
2367 + DRM_INFO("%s: flush moves to exec list %p\n", __func__, obj);
2371 + /* If there is rendering queued on the buffer being evicted, wait for
2374 + if (obj_priv->active) {
2376 + DRM_INFO("%s: object %p wait for seqno %08x\n",
2377 + __func__, obj, obj_priv->last_rendering_seqno);
2379 + ret = i915_wait_request(dev, obj_priv->last_rendering_seqno);
2388 + * Unbinds an object from the GTT aperture.
2391 +i915_gem_object_unbind(struct drm_gem_object *obj)
2393 + struct drm_device *dev = obj->dev;
2394 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2398 + DRM_INFO("%s:%d %p\n", __func__, __LINE__, obj);
2399 + DRM_INFO("gtt_space %p\n", obj_priv->gtt_space);
2401 + if (obj_priv->gtt_space == NULL)
2404 + if (obj_priv->pin_count != 0) {
2405 + DRM_ERROR("Attempting to unbind pinned buffer\n");
2409 + /* Wait for any rendering to complete
2411 + ret = i915_gem_object_wait_rendering(obj);
2413 + DRM_ERROR("wait_rendering failed: %d\n", ret);
2417 + /* Move the object to the CPU domain to ensure that
2418 + * any possible CPU writes while it's not in the GTT
2419 + * are flushed when we go to remap it. This will
2420 + * also ensure that all pending GPU writes are finished
2421 + * before we unbind.
2423 + ret = i915_gem_object_set_domain(obj, I915_GEM_DOMAIN_CPU,
2424 + I915_GEM_DOMAIN_CPU);
2426 + DRM_ERROR("set_domain failed: %d\n", ret);
2430 + if (obj_priv->agp_mem != NULL) {
2431 + drm_unbind_agp(obj_priv->agp_mem);
2432 + drm_free_agp(obj_priv->agp_mem, obj->size / PAGE_SIZE);
2433 + obj_priv->agp_mem = NULL;
2436 + BUG_ON(obj_priv->active);
2438 + i915_gem_object_free_page_list(obj);
2440 + if (obj_priv->gtt_space) {
2441 + atomic_dec(&dev->gtt_count);
2442 + atomic_sub(obj->size, &dev->gtt_memory);
2444 + drm_mm_put_block(obj_priv->gtt_space);
2445 + obj_priv->gtt_space = NULL;
2448 + /* Remove ourselves from the LRU list if present. */
2449 + if (!list_empty(&obj_priv->list))
2450 + list_del_init(&obj_priv->list);
2456 +i915_gem_evict_something(struct drm_device *dev)
2458 + drm_i915_private_t *dev_priv = dev->dev_private;
2459 + struct drm_gem_object *obj;
2460 + struct drm_i915_gem_object *obj_priv;
2464 + /* If there's an inactive buffer available now, grab it
2467 + if (!list_empty(&dev_priv->mm.inactive_list)) {
2468 + obj_priv = list_first_entry(&dev_priv->mm.inactive_list,
2469 + struct drm_i915_gem_object,
2471 + obj = obj_priv->obj;
2472 + BUG_ON(obj_priv->pin_count != 0);
2474 + DRM_INFO("%s: evicting %p\n", __func__, obj);
2476 + BUG_ON(obj_priv->active);
2478 + /* Wait on the rendering and unbind the buffer. */
2479 + ret = i915_gem_object_unbind(obj);
2483 + /* If we didn't get anything, but the ring is still processing
2484 + * things, wait for one of those things to finish and hopefully
2485 + * leave us a buffer to evict.
2487 + if (!list_empty(&dev_priv->mm.request_list)) {
2488 + struct drm_i915_gem_request *request;
2490 + request = list_first_entry(&dev_priv->mm.request_list,
2491 + struct drm_i915_gem_request,
2494 + ret = i915_wait_request(dev, request->seqno);
2498 + /* if waiting caused an object to become inactive,
2499 + * then loop around and wait for it. Otherwise, we
2500 + * assume that waiting freed and unbound something,
2501 + * so there should now be some space in the GTT
2503 + if (!list_empty(&dev_priv->mm.inactive_list))
2508 + /* If we didn't have anything on the request list but there
2509 + * are buffers awaiting a flush, emit one and try again.
2510 + * When we wait on it, those buffers waiting for that flush
2511 + * will get moved to inactive.
2513 + if (!list_empty(&dev_priv->mm.flushing_list)) {
2514 + obj_priv = list_first_entry(&dev_priv->mm.flushing_list,
2515 + struct drm_i915_gem_object,
2517 + obj = obj_priv->obj;
2519 + i915_gem_flush(dev,
2520 + obj->write_domain,
2521 + obj->write_domain);
2522 + i915_add_request(dev, obj->write_domain);
2528 + DRM_ERROR("inactive empty %d request empty %d "
2529 + "flushing empty %d\n",
2530 + list_empty(&dev_priv->mm.inactive_list),
2531 + list_empty(&dev_priv->mm.request_list),
2532 + list_empty(&dev_priv->mm.flushing_list));
2533 + /* If we didn't do any of the above, there's nothing to be done
2534 + * and we just can't fit it in.
2542 +i915_gem_object_get_page_list(struct drm_gem_object *obj)
2544 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2545 + int page_count, i;
2546 + struct address_space *mapping;
2547 + struct inode *inode;
2548 + struct page *page;
2551 + if (obj_priv->page_list)
2554 + /* Get the list of pages out of our struct file. They'll be pinned
2555 + * at this point until we release them.
2557 + page_count = obj->size / PAGE_SIZE;
2558 + BUG_ON(obj_priv->page_list != NULL);
2559 + obj_priv->page_list = drm_calloc(page_count, sizeof(struct page *),
2561 + if (obj_priv->page_list == NULL) {
2562 + DRM_ERROR("Faled to allocate page list\n");
2566 + inode = obj->filp->f_path.dentry->d_inode;
2567 + mapping = inode->i_mapping;
2568 + for (i = 0; i < page_count; i++) {
2569 + page = read_mapping_page(mapping, i, NULL);
2570 + if (IS_ERR(page)) {
2571 + ret = PTR_ERR(page);
2572 + DRM_ERROR("read_mapping_page failed: %d\n", ret);
2573 + i915_gem_object_free_page_list(obj);
2576 + obj_priv->page_list[i] = page;
2582 + * Finds free space in the GTT aperture and binds the object there.
2585 +i915_gem_object_bind_to_gtt(struct drm_gem_object *obj, unsigned alignment)
2587 + struct drm_device *dev = obj->dev;
2588 + drm_i915_private_t *dev_priv = dev->dev_private;
2589 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2590 + struct drm_mm_node *free_space;
2591 + int page_count, ret;
2593 + if (alignment == 0)
2594 + alignment = PAGE_SIZE;
2595 + if (alignment & (PAGE_SIZE - 1)) {
2596 + DRM_ERROR("Invalid object alignment requested %u\n", alignment);
2601 + free_space = drm_mm_search_free(&dev_priv->mm.gtt_space,
2602 + obj->size, alignment, 0);
2603 + if (free_space != NULL) {
2604 + obj_priv->gtt_space = drm_mm_get_block(free_space, obj->size,
2606 + if (obj_priv->gtt_space != NULL) {
2607 + obj_priv->gtt_space->private = obj;
2608 + obj_priv->gtt_offset = obj_priv->gtt_space->start;
2611 + if (obj_priv->gtt_space == NULL) {
2612 + /* If the gtt is empty and we're still having trouble
2613 + * fitting our object in, we're out of memory.
2616 + DRM_INFO("%s: GTT full, evicting something\n", __func__);
2618 + if (list_empty(&dev_priv->mm.inactive_list) &&
2619 + list_empty(&dev_priv->mm.flushing_list) &&
2620 + list_empty(&dev_priv->mm.active_list)) {
2621 + DRM_ERROR("GTT full, but LRU list empty\n");
2625 + ret = i915_gem_evict_something(dev);
2627 + DRM_ERROR("Failed to evict a buffer %d\n", ret);
2634 + DRM_INFO("Binding object of size %d at 0x%08x\n",
2635 + obj->size, obj_priv->gtt_offset);
2637 + ret = i915_gem_object_get_page_list(obj);
2639 + drm_mm_put_block(obj_priv->gtt_space);
2640 + obj_priv->gtt_space = NULL;
2644 + page_count = obj->size / PAGE_SIZE;
2645 + /* Create an AGP memory structure pointing at our pages, and bind it
2648 + obj_priv->agp_mem = drm_agp_bind_pages(dev,
2649 + obj_priv->page_list,
2651 + obj_priv->gtt_offset);
2652 + if (obj_priv->agp_mem == NULL) {
2653 + i915_gem_object_free_page_list(obj);
2654 + drm_mm_put_block(obj_priv->gtt_space);
2655 + obj_priv->gtt_space = NULL;
2658 + atomic_inc(&dev->gtt_count);
2659 + atomic_add(obj->size, &dev->gtt_memory);
2661 + /* Assert that the object is not currently in any GPU domain. As it
2662 + * wasn't in the GTT, there shouldn't be any way it could have been in
2665 + BUG_ON(obj->read_domains & ~(I915_GEM_DOMAIN_CPU|I915_GEM_DOMAIN_GTT));
2666 + BUG_ON(obj->write_domain & ~(I915_GEM_DOMAIN_CPU|I915_GEM_DOMAIN_GTT));
2672 +i915_gem_clflush_object(struct drm_gem_object *obj)
2674 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2676 + /* If we don't have a page list set up, then we're not pinned
2677 + * to GPU, and we can ignore the cache flush because it'll happen
2678 + * again at bind time.
2680 + if (obj_priv->page_list == NULL)
2683 + drm_clflush_pages(obj_priv->page_list, obj->size / PAGE_SIZE);
2687 + * Set the next domain for the specified object. This
2688 + * may not actually perform the necessary flushing/invaliding though,
2689 + * as that may want to be batched with other set_domain operations
2691 + * This is (we hope) the only really tricky part of gem. The goal
2692 + * is fairly simple -- track which caches hold bits of the object
2693 + * and make sure they remain coherent. A few concrete examples may
2694 + * help to explain how it works. For shorthand, we use the notation
2695 + * (read_domains, write_domain), e.g. (CPU, CPU) to indicate the
2696 + * a pair of read and write domain masks.
2698 + * Case 1: the batch buffer
2701 + * 2. Written by CPU
2702 + * 3. Mapped to GTT
2704 + * 5. Unmapped from GTT
2707 + * Let's take these a step at a time
2710 + * Pages allocated from the kernel may still have
2711 + * cache contents, so we set them to (CPU, CPU) always.
2712 + * 2. Written by CPU (using pwrite)
2713 + * The pwrite function calls set_domain (CPU, CPU) and
2714 + * this function does nothing (as nothing changes)
2715 + * 3. Mapped by GTT
2716 + * This function asserts that the object is not
2717 + * currently in any GPU-based read or write domains
2719 + * i915_gem_execbuffer calls set_domain (COMMAND, 0).
2720 + * As write_domain is zero, this function adds in the
2721 + * current read domains (CPU+COMMAND, 0).
2722 + * flush_domains is set to CPU.
2723 + * invalidate_domains is set to COMMAND
2724 + * clflush is run to get data out of the CPU caches
2725 + * then i915_dev_set_domain calls i915_gem_flush to
2726 + * emit an MI_FLUSH and drm_agp_chipset_flush
2727 + * 5. Unmapped from GTT
2728 + * i915_gem_object_unbind calls set_domain (CPU, CPU)
2729 + * flush_domains and invalidate_domains end up both zero
2730 + * so no flushing/invalidating happens
2734 + * Case 2: The shared render buffer
2737 + * 2. Mapped to GTT
2738 + * 3. Read/written by GPU
2739 + * 4. set_domain to (CPU,CPU)
2740 + * 5. Read/written by CPU
2741 + * 6. Read/written by GPU
2744 + * Same as last example, (CPU, CPU)
2745 + * 2. Mapped to GTT
2746 + * Nothing changes (assertions find that it is not in the GPU)
2747 + * 3. Read/written by GPU
2748 + * execbuffer calls set_domain (RENDER, RENDER)
2749 + * flush_domains gets CPU
2750 + * invalidate_domains gets GPU
2752 + * MI_FLUSH and drm_agp_chipset_flush
2753 + * 4. set_domain (CPU, CPU)
2754 + * flush_domains gets GPU
2755 + * invalidate_domains gets CPU
2756 + * wait_rendering (obj) to make sure all drawing is complete.
2757 + * This will include an MI_FLUSH to get the data from GPU
2759 + * clflush (obj) to invalidate the CPU cache
2760 + * Another MI_FLUSH in i915_gem_flush (eliminate this somehow?)
2761 + * 5. Read/written by CPU
2762 + * cache lines are loaded and dirtied
2763 + * 6. Read written by GPU
2764 + * Same as last GPU access
2766 + * Case 3: The constant buffer
2769 + * 2. Written by CPU
2771 + * 4. Updated (written) by CPU again
2776 + * 2. Written by CPU
2780 + * flush_domains = CPU
2781 + * invalidate_domains = RENDER
2784 + * drm_agp_chipset_flush
2785 + * 4. Updated (written) by CPU again
2787 + * flush_domains = 0 (no previous write domain)
2788 + * invalidate_domains = 0 (no new read domains)
2791 + * flush_domains = CPU
2792 + * invalidate_domains = RENDER
2795 + * drm_agp_chipset_flush
2798 +i915_gem_object_set_domain(struct drm_gem_object *obj,
2799 + uint32_t read_domains,
2800 + uint32_t write_domain)
2802 + struct drm_device *dev = obj->dev;
2803 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2804 + uint32_t invalidate_domains = 0;
2805 + uint32_t flush_domains = 0;
2809 + DRM_INFO("%s: object %p read %08x -> %08x write %08x -> %08x\n",
2811 + obj->read_domains, read_domains,
2812 + obj->write_domain, write_domain);
2815 + * If the object isn't moving to a new write domain,
2816 + * let the object stay in multiple read domains
2818 + if (write_domain == 0)
2819 + read_domains |= obj->read_domains;
2821 + obj_priv->dirty = 1;
2824 + * Flush the current write domain if
2825 + * the new read domains don't match. Invalidate
2826 + * any read domains which differ from the old
2829 + if (obj->write_domain && obj->write_domain != read_domains) {
2830 + flush_domains |= obj->write_domain;
2831 + invalidate_domains |= read_domains & ~obj->write_domain;
2834 + * Invalidate any read caches which may have
2835 + * stale data. That is, any new read domains.
2837 + invalidate_domains |= read_domains & ~obj->read_domains;
2838 + if ((flush_domains | invalidate_domains) & I915_GEM_DOMAIN_CPU) {
2840 + DRM_INFO("%s: CPU domain flush %08x invalidate %08x\n",
2841 + __func__, flush_domains, invalidate_domains);
2844 + * If we're invaliding the CPU cache and flushing a GPU cache,
2845 + * then pause for rendering so that the GPU caches will be
2846 + * flushed before the cpu cache is invalidated
2848 + if ((invalidate_domains & I915_GEM_DOMAIN_CPU) &&
2849 + (flush_domains & ~(I915_GEM_DOMAIN_CPU |
2850 + I915_GEM_DOMAIN_GTT))) {
2851 + ret = i915_gem_object_wait_rendering(obj);
2855 + i915_gem_clflush_object(obj);
2858 + if ((write_domain | flush_domains) != 0)
2859 + obj->write_domain = write_domain;
2861 + /* If we're invalidating the CPU domain, clear the per-page CPU
2862 + * domain list as well.
2864 + if (obj_priv->page_cpu_valid != NULL &&
2865 + (write_domain != 0 ||
2866 + read_domains & I915_GEM_DOMAIN_CPU)) {
2867 + drm_free(obj_priv->page_cpu_valid, obj->size / PAGE_SIZE,
2869 + obj_priv->page_cpu_valid = NULL;
2871 + obj->read_domains = read_domains;
2873 + dev->invalidate_domains |= invalidate_domains;
2874 + dev->flush_domains |= flush_domains;
2876 + DRM_INFO("%s: read %08x write %08x invalidate %08x flush %08x\n",
2878 + obj->read_domains, obj->write_domain,
2879 + dev->invalidate_domains, dev->flush_domains);
2885 + * Set the read/write domain on a range of the object.
2887 + * Currently only implemented for CPU reads, otherwise drops to normal
2888 + * i915_gem_object_set_domain().
2891 +i915_gem_object_set_domain_range(struct drm_gem_object *obj,
2894 + uint32_t read_domains,
2895 + uint32_t write_domain)
2897 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2900 + if (obj->read_domains & I915_GEM_DOMAIN_CPU)
2903 + if (read_domains != I915_GEM_DOMAIN_CPU ||
2904 + write_domain != 0)
2905 + return i915_gem_object_set_domain(obj,
2906 + read_domains, write_domain);
2908 + /* Wait on any GPU rendering to the object to be flushed. */
2909 + if (obj->write_domain & ~(I915_GEM_DOMAIN_CPU | I915_GEM_DOMAIN_GTT)) {
2910 + ret = i915_gem_object_wait_rendering(obj);
2915 + if (obj_priv->page_cpu_valid == NULL) {
2916 + obj_priv->page_cpu_valid = drm_calloc(1, obj->size / PAGE_SIZE,
2920 + /* Flush the cache on any pages that are still invalid from the CPU's
2923 + for (i = offset / PAGE_SIZE; i <= (offset + size - 1) / PAGE_SIZE; i++) {
2924 + if (obj_priv->page_cpu_valid[i])
2927 + drm_clflush_pages(obj_priv->page_list + i, 1);
2929 + obj_priv->page_cpu_valid[i] = 1;
2936 + * Once all of the objects have been set in the proper domain,
2937 + * perform the necessary flush and invalidate operations.
2939 + * Returns the write domains flushed, for use in flush tracking.
2942 +i915_gem_dev_set_domain(struct drm_device *dev)
2944 + uint32_t flush_domains = dev->flush_domains;
2947 + * Now that all the buffers are synced to the proper domains,
2948 + * flush and invalidate the collected domains
2950 + if (dev->invalidate_domains | dev->flush_domains) {
2952 + DRM_INFO("%s: invalidate_domains %08x flush_domains %08x\n",
2954 + dev->invalidate_domains,
2955 + dev->flush_domains);
2957 + i915_gem_flush(dev,
2958 + dev->invalidate_domains,
2959 + dev->flush_domains);
2960 + dev->invalidate_domains = 0;
2961 + dev->flush_domains = 0;
2964 + return flush_domains;
2968 + * Pin an object to the GTT and evaluate the relocations landing in it.
2971 +i915_gem_object_pin_and_relocate(struct drm_gem_object *obj,
2972 + struct drm_file *file_priv,
2973 + struct drm_i915_gem_exec_object *entry)
2975 + struct drm_device *dev = obj->dev;
2976 + struct drm_i915_gem_relocation_entry reloc;
2977 + struct drm_i915_gem_relocation_entry __user *relocs;
2978 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
2980 + uint32_t last_reloc_offset = -1;
2981 + void *reloc_page = NULL;
2983 + /* Choose the GTT offset for our buffer and put it there. */
2984 + ret = i915_gem_object_pin(obj, (uint32_t) entry->alignment);
2988 + entry->offset = obj_priv->gtt_offset;
2990 + relocs = (struct drm_i915_gem_relocation_entry __user *)
2991 + (uintptr_t) entry->relocs_ptr;
2992 + /* Apply the relocations, using the GTT aperture to avoid cache
2993 + * flushing requirements.
2995 + for (i = 0; i < entry->relocation_count; i++) {
2996 + struct drm_gem_object *target_obj;
2997 + struct drm_i915_gem_object *target_obj_priv;
2998 + uint32_t reloc_val, reloc_offset, *reloc_entry;
3001 + ret = copy_from_user(&reloc, relocs + i, sizeof(reloc));
3003 + i915_gem_object_unpin(obj);
3007 + target_obj = drm_gem_object_lookup(obj->dev, file_priv,
3008 + reloc.target_handle);
3009 + if (target_obj == NULL) {
3010 + i915_gem_object_unpin(obj);
3013 + target_obj_priv = target_obj->driver_private;
3015 + /* The target buffer should have appeared before us in the
3016 + * exec_object list, so it should have a GTT space bound by now.
3018 + if (target_obj_priv->gtt_space == NULL) {
3019 + DRM_ERROR("No GTT space found for object %d\n",
3020 + reloc.target_handle);
3021 + drm_gem_object_unreference(target_obj);
3022 + i915_gem_object_unpin(obj);
3026 + if (reloc.offset > obj->size - 4) {
3027 + DRM_ERROR("Relocation beyond object bounds: "
3028 + "obj %p target %d offset %d size %d.\n",
3029 + obj, reloc.target_handle,
3030 + (int) reloc.offset, (int) obj->size);
3031 + drm_gem_object_unreference(target_obj);
3032 + i915_gem_object_unpin(obj);
3035 + if (reloc.offset & 3) {
3036 + DRM_ERROR("Relocation not 4-byte aligned: "
3037 + "obj %p target %d offset %d.\n",
3038 + obj, reloc.target_handle,
3039 + (int) reloc.offset);
3040 + drm_gem_object_unreference(target_obj);
3041 + i915_gem_object_unpin(obj);
3045 + if (reloc.write_domain && target_obj->pending_write_domain &&
3046 + reloc.write_domain != target_obj->pending_write_domain) {
3047 + DRM_ERROR("Write domain conflict: "
3048 + "obj %p target %d offset %d "
3049 + "new %08x old %08x\n",
3050 + obj, reloc.target_handle,
3051 + (int) reloc.offset,
3052 + reloc.write_domain,
3053 + target_obj->pending_write_domain);
3054 + drm_gem_object_unreference(target_obj);
3055 + i915_gem_object_unpin(obj);
3060 + DRM_INFO("%s: obj %p offset %08x target %d "
3061 + "read %08x write %08x gtt %08x "
3062 + "presumed %08x delta %08x\n",
3065 + (int) reloc.offset,
3066 + (int) reloc.target_handle,
3067 + (int) reloc.read_domains,
3068 + (int) reloc.write_domain,
3069 + (int) target_obj_priv->gtt_offset,
3070 + (int) reloc.presumed_offset,
3074 + target_obj->pending_read_domains |= reloc.read_domains;
3075 + target_obj->pending_write_domain |= reloc.write_domain;
3077 + /* If the relocation already has the right value in it, no
3078 + * more work needs to be done.
3080 + if (target_obj_priv->gtt_offset == reloc.presumed_offset) {
3081 + drm_gem_object_unreference(target_obj);
3085 + /* Now that we're going to actually write some data in,
3086 + * make sure that any rendering using this buffer's contents
3089 + i915_gem_object_wait_rendering(obj);
3091 + /* As we're writing through the gtt, flush
3092 + * any CPU writes before we write the relocations
3094 + if (obj->write_domain & I915_GEM_DOMAIN_CPU) {
3095 + i915_gem_clflush_object(obj);
3096 + drm_agp_chipset_flush(dev);
3097 + obj->write_domain = 0;
3100 + /* Map the page containing the relocation we're going to
3103 + reloc_offset = obj_priv->gtt_offset + reloc.offset;
3104 + if (reloc_page == NULL ||
3105 + (last_reloc_offset & ~(PAGE_SIZE - 1)) !=
3106 + (reloc_offset & ~(PAGE_SIZE - 1))) {
3107 + if (reloc_page != NULL)
3108 + iounmap(reloc_page);
3110 + reloc_page = ioremap(dev->agp->base +
3111 + (reloc_offset & ~(PAGE_SIZE - 1)),
3113 + last_reloc_offset = reloc_offset;
3114 + if (reloc_page == NULL) {
3115 + drm_gem_object_unreference(target_obj);
3116 + i915_gem_object_unpin(obj);
3121 + reloc_entry = (uint32_t *)((char *)reloc_page +
3122 + (reloc_offset & (PAGE_SIZE - 1)));
3123 + reloc_val = target_obj_priv->gtt_offset + reloc.delta;
3126 + DRM_INFO("Applied relocation: %p@0x%08x %08x -> %08x\n",
3127 + obj, (unsigned int) reloc.offset,
3128 + readl(reloc_entry), reloc_val);
3130 + writel(reloc_val, reloc_entry);
3132 + /* Write the updated presumed offset for this entry back out
3135 + reloc.presumed_offset = target_obj_priv->gtt_offset;
3136 + ret = copy_to_user(relocs + i, &reloc, sizeof(reloc));
3138 + drm_gem_object_unreference(target_obj);
3139 + i915_gem_object_unpin(obj);
3143 + drm_gem_object_unreference(target_obj);
3146 + if (reloc_page != NULL)
3147 + iounmap(reloc_page);
3151 + i915_gem_dump_object(obj, 128, __func__, ~0);
3156 +/** Dispatch a batchbuffer to the ring
3159 +i915_dispatch_gem_execbuffer(struct drm_device *dev,
3160 + struct drm_i915_gem_execbuffer *exec,
3161 + uint64_t exec_offset)
3163 + drm_i915_private_t *dev_priv = dev->dev_private;
3164 + struct drm_clip_rect __user *boxes = (struct drm_clip_rect __user *)
3165 + (uintptr_t) exec->cliprects_ptr;
3166 + int nbox = exec->num_cliprects;
3168 + uint32_t exec_start, exec_len;
3171 + exec_start = (uint32_t) exec_offset + exec->batch_start_offset;
3172 + exec_len = (uint32_t) exec->batch_len;
3174 + if ((exec_start | exec_len) & 0x7) {
3175 + DRM_ERROR("alignment\n");
3182 + count = nbox ? nbox : 1;
3184 + for (i = 0; i < count; i++) {
3186 + int ret = i915_emit_box(dev, boxes, i,
3187 + exec->DR1, exec->DR4);
3192 + if (IS_I830(dev) || IS_845G(dev)) {
3194 + OUT_RING(MI_BATCH_BUFFER);
3195 + OUT_RING(exec_start | MI_BATCH_NON_SECURE);
3196 + OUT_RING(exec_start + exec_len - 4);
3198 + ADVANCE_LP_RING();
3201 + if (IS_I965G(dev)) {
3202 + OUT_RING(MI_BATCH_BUFFER_START |
3204 + MI_BATCH_NON_SECURE_I965);
3205 + OUT_RING(exec_start);
3207 + OUT_RING(MI_BATCH_BUFFER_START |
3209 + OUT_RING(exec_start | MI_BATCH_NON_SECURE);
3211 + ADVANCE_LP_RING();
3215 + /* XXX breadcrumb */
3219 +/* Throttle our rendering by waiting until the ring has completed our requests
3220 + * emitted over 20 msec ago.
3222 + * This should get us reasonable parallelism between CPU and GPU but also
3223 + * relatively low latency when blocking on a particular request to finish.
3226 +i915_gem_ring_throttle(struct drm_device *dev, struct drm_file *file_priv)
3228 + struct drm_i915_file_private *i915_file_priv = file_priv->driver_priv;
3232 + mutex_lock(&dev->struct_mutex);
3233 + seqno = i915_file_priv->mm.last_gem_throttle_seqno;
3234 + i915_file_priv->mm.last_gem_throttle_seqno =
3235 + i915_file_priv->mm.last_gem_seqno;
3237 + ret = i915_wait_request(dev, seqno);
3238 + mutex_unlock(&dev->struct_mutex);
3243 +i915_gem_execbuffer(struct drm_device *dev, void *data,
3244 + struct drm_file *file_priv)
3246 + drm_i915_private_t *dev_priv = dev->dev_private;
3247 + struct drm_i915_file_private *i915_file_priv = file_priv->driver_priv;
3248 + struct drm_i915_gem_execbuffer *args = data;
3249 + struct drm_i915_gem_exec_object *exec_list = NULL;
3250 + struct drm_gem_object **object_list = NULL;
3251 + struct drm_gem_object *batch_obj;
3252 + int ret, i, pinned = 0;
3253 + uint64_t exec_offset;
3254 + uint32_t seqno, flush_domains;
3257 + DRM_INFO("buffers_ptr %d buffer_count %d len %08x\n",
3258 + (int) args->buffers_ptr, args->buffer_count, args->batch_len);
3261 + /* Copy in the exec list from userland */
3262 + exec_list = drm_calloc(sizeof(*exec_list), args->buffer_count,
3264 + object_list = drm_calloc(sizeof(*object_list), args->buffer_count,
3266 + if (exec_list == NULL || object_list == NULL) {
3267 + DRM_ERROR("Failed to allocate exec or object list "
3268 + "for %d buffers\n",
3269 + args->buffer_count);
3271 + goto pre_mutex_err;
3273 + ret = copy_from_user(exec_list,
3274 + (struct drm_i915_relocation_entry __user *)
3275 + (uintptr_t) args->buffers_ptr,
3276 + sizeof(*exec_list) * args->buffer_count);
3278 + DRM_ERROR("copy %d exec entries failed %d\n",
3279 + args->buffer_count, ret);
3280 + goto pre_mutex_err;
3283 + mutex_lock(&dev->struct_mutex);
3285 + i915_verify_inactive(dev, __FILE__, __LINE__);
3287 + if (dev_priv->mm.wedged) {
3288 + DRM_ERROR("Execbuf while wedged\n");
3289 + mutex_unlock(&dev->struct_mutex);
3293 + if (dev_priv->mm.suspended) {
3294 + DRM_ERROR("Execbuf while VT-switched.\n");
3295 + mutex_unlock(&dev->struct_mutex);
3299 + /* Zero the gloabl flush/invalidate flags. These
3300 + * will be modified as each object is bound to the
3303 + dev->invalidate_domains = 0;
3304 + dev->flush_domains = 0;
3306 + /* Look up object handles and perform the relocations */
3307 + for (i = 0; i < args->buffer_count; i++) {
3308 + object_list[i] = drm_gem_object_lookup(dev, file_priv,
3309 + exec_list[i].handle);
3310 + if (object_list[i] == NULL) {
3311 + DRM_ERROR("Invalid object handle %d at index %d\n",
3312 + exec_list[i].handle, i);
3317 + object_list[i]->pending_read_domains = 0;
3318 + object_list[i]->pending_write_domain = 0;
3319 + ret = i915_gem_object_pin_and_relocate(object_list[i],
3323 + DRM_ERROR("object bind and relocate failed %d\n", ret);
3329 + /* Set the pending read domains for the batch buffer to COMMAND */
3330 + batch_obj = object_list[args->buffer_count-1];
3331 + batch_obj->pending_read_domains = I915_GEM_DOMAIN_COMMAND;
3332 + batch_obj->pending_write_domain = 0;
3334 + i915_verify_inactive(dev, __FILE__, __LINE__);
3336 + for (i = 0; i < args->buffer_count; i++) {
3337 + struct drm_gem_object *obj = object_list[i];
3338 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
3340 + if (obj_priv->gtt_space == NULL) {
3341 + /* We evicted the buffer in the process of validating
3342 + * our set of buffers in. We could try to recover by
3343 + * kicking them everything out and trying again from
3350 + /* make sure all previous memory operations have passed */
3351 + ret = i915_gem_object_set_domain(obj,
3352 + obj->pending_read_domains,
3353 + obj->pending_write_domain);
3358 + i915_verify_inactive(dev, __FILE__, __LINE__);
3360 + /* Flush/invalidate caches and chipset buffer */
3361 + flush_domains = i915_gem_dev_set_domain(dev);
3363 + i915_verify_inactive(dev, __FILE__, __LINE__);
3365 +#if WATCH_COHERENCY
3366 + for (i = 0; i < args->buffer_count; i++) {
3367 + i915_gem_object_check_coherency(object_list[i],
3368 + exec_list[i].handle);
3372 + exec_offset = exec_list[args->buffer_count - 1].offset;
3375 + i915_gem_dump_object(object_list[args->buffer_count - 1],
3381 + (void)i915_add_request(dev, flush_domains);
3383 + /* Exec the batchbuffer */
3384 + ret = i915_dispatch_gem_execbuffer(dev, args, exec_offset);
3386 + DRM_ERROR("dispatch failed %d\n", ret);
3391 + * Ensure that the commands in the batch buffer are
3392 + * finished before the interrupt fires
3394 + flush_domains = i915_retire_commands(dev);
3396 + i915_verify_inactive(dev, __FILE__, __LINE__);
3399 + * Get a seqno representing the execution of the current buffer,
3400 + * which we can wait on. We would like to mitigate these interrupts,
3401 + * likely by only creating seqnos occasionally (so that we have
3402 + * *some* interrupts representing completion of buffers that we can
3403 + * wait on when trying to clear up gtt space).
3405 + seqno = i915_add_request(dev, flush_domains);
3406 + BUG_ON(seqno == 0);
3407 + i915_file_priv->mm.last_gem_seqno = seqno;
3408 + for (i = 0; i < args->buffer_count; i++) {
3409 + struct drm_gem_object *obj = object_list[i];
3410 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
3412 + i915_gem_object_move_to_active(obj);
3413 + obj_priv->last_rendering_seqno = seqno;
3415 + DRM_INFO("%s: move to exec list %p\n", __func__, obj);
3419 + i915_dump_lru(dev, __func__);
3422 + i915_verify_inactive(dev, __FILE__, __LINE__);
3424 + /* Copy the new buffer offsets back to the user's exec list. */
3425 + ret = copy_to_user((struct drm_i915_relocation_entry __user *)
3426 + (uintptr_t) args->buffers_ptr,
3428 + sizeof(*exec_list) * args->buffer_count);
3430 + DRM_ERROR("failed to copy %d exec entries "
3431 + "back to user (%d)\n",
3432 + args->buffer_count, ret);
3434 + if (object_list != NULL) {
3435 + for (i = 0; i < pinned; i++)
3436 + i915_gem_object_unpin(object_list[i]);
3438 + for (i = 0; i < args->buffer_count; i++)
3439 + drm_gem_object_unreference(object_list[i]);
3441 + mutex_unlock(&dev->struct_mutex);
3444 + drm_free(object_list, sizeof(*object_list) * args->buffer_count,
3446 + drm_free(exec_list, sizeof(*exec_list) * args->buffer_count,
3453 +i915_gem_object_pin(struct drm_gem_object *obj, uint32_t alignment)
3455 + struct drm_device *dev = obj->dev;
3456 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
3459 + i915_verify_inactive(dev, __FILE__, __LINE__);
3460 + if (obj_priv->gtt_space == NULL) {
3461 + ret = i915_gem_object_bind_to_gtt(obj, alignment);
3463 + DRM_ERROR("Failure to bind: %d", ret);
3467 + obj_priv->pin_count++;
3469 + /* If the object is not active and not pending a flush,
3470 + * remove it from the inactive list
3472 + if (obj_priv->pin_count == 1) {
3473 + atomic_inc(&dev->pin_count);
3474 + atomic_add(obj->size, &dev->pin_memory);
3475 + if (!obj_priv->active &&
3476 + (obj->write_domain & ~(I915_GEM_DOMAIN_CPU |
3477 + I915_GEM_DOMAIN_GTT)) == 0 &&
3478 + !list_empty(&obj_priv->list))
3479 + list_del_init(&obj_priv->list);
3481 + i915_verify_inactive(dev, __FILE__, __LINE__);
3487 +i915_gem_object_unpin(struct drm_gem_object *obj)
3489 + struct drm_device *dev = obj->dev;
3490 + drm_i915_private_t *dev_priv = dev->dev_private;
3491 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
3493 + i915_verify_inactive(dev, __FILE__, __LINE__);
3494 + obj_priv->pin_count--;
3495 + BUG_ON(obj_priv->pin_count < 0);
3496 + BUG_ON(obj_priv->gtt_space == NULL);
3498 + /* If the object is no longer pinned, and is
3499 + * neither active nor being flushed, then stick it on
3500 + * the inactive list
3502 + if (obj_priv->pin_count == 0) {
3503 + if (!obj_priv->active &&
3504 + (obj->write_domain & ~(I915_GEM_DOMAIN_CPU |
3505 + I915_GEM_DOMAIN_GTT)) == 0)
3506 + list_move_tail(&obj_priv->list,
3507 + &dev_priv->mm.inactive_list);
3508 + atomic_dec(&dev->pin_count);
3509 + atomic_sub(obj->size, &dev->pin_memory);
3511 + i915_verify_inactive(dev, __FILE__, __LINE__);
3515 +i915_gem_pin_ioctl(struct drm_device *dev, void *data,
3516 + struct drm_file *file_priv)
3518 + struct drm_i915_gem_pin *args = data;
3519 + struct drm_gem_object *obj;
3520 + struct drm_i915_gem_object *obj_priv;
3523 + mutex_lock(&dev->struct_mutex);
3525 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
3526 + if (obj == NULL) {
3527 + DRM_ERROR("Bad handle in i915_gem_pin_ioctl(): %d\n",
3529 + mutex_unlock(&dev->struct_mutex);
3532 + obj_priv = obj->driver_private;
3534 + ret = i915_gem_object_pin(obj, args->alignment);
3536 + drm_gem_object_unreference(obj);
3537 + mutex_unlock(&dev->struct_mutex);
3541 + /* XXX - flush the CPU caches for pinned objects
3542 + * as the X server doesn't manage domains yet
3544 + if (obj->write_domain & I915_GEM_DOMAIN_CPU) {
3545 + i915_gem_clflush_object(obj);
3546 + drm_agp_chipset_flush(dev);
3547 + obj->write_domain = 0;
3549 + args->offset = obj_priv->gtt_offset;
3550 + drm_gem_object_unreference(obj);
3551 + mutex_unlock(&dev->struct_mutex);
3557 +i915_gem_unpin_ioctl(struct drm_device *dev, void *data,
3558 + struct drm_file *file_priv)
3560 + struct drm_i915_gem_pin *args = data;
3561 + struct drm_gem_object *obj;
3563 + mutex_lock(&dev->struct_mutex);
3565 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
3566 + if (obj == NULL) {
3567 + DRM_ERROR("Bad handle in i915_gem_unpin_ioctl(): %d\n",
3569 + mutex_unlock(&dev->struct_mutex);
3573 + i915_gem_object_unpin(obj);
3575 + drm_gem_object_unreference(obj);
3576 + mutex_unlock(&dev->struct_mutex);
3581 +i915_gem_busy_ioctl(struct drm_device *dev, void *data,
3582 + struct drm_file *file_priv)
3584 + struct drm_i915_gem_busy *args = data;
3585 + struct drm_gem_object *obj;
3586 + struct drm_i915_gem_object *obj_priv;
3588 + mutex_lock(&dev->struct_mutex);
3589 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
3590 + if (obj == NULL) {
3591 + DRM_ERROR("Bad handle in i915_gem_busy_ioctl(): %d\n",
3593 + mutex_unlock(&dev->struct_mutex);
3597 + obj_priv = obj->driver_private;
3598 + args->busy = obj_priv->active;
3600 + drm_gem_object_unreference(obj);
3601 + mutex_unlock(&dev->struct_mutex);
3606 +i915_gem_throttle_ioctl(struct drm_device *dev, void *data,
3607 + struct drm_file *file_priv)
3609 + return i915_gem_ring_throttle(dev, file_priv);
3612 +int i915_gem_init_object(struct drm_gem_object *obj)
3614 + struct drm_i915_gem_object *obj_priv;
3616 + obj_priv = drm_calloc(1, sizeof(*obj_priv), DRM_MEM_DRIVER);
3617 + if (obj_priv == NULL)
3621 + * We've just allocated pages from the kernel,
3622 + * so they've just been written by the CPU with
3623 + * zeros. They'll need to be clflushed before we
3624 + * use them with the GPU.
3626 + obj->write_domain = I915_GEM_DOMAIN_CPU;
3627 + obj->read_domains = I915_GEM_DOMAIN_CPU;
3629 + obj->driver_private = obj_priv;
3630 + obj_priv->obj = obj;
3631 + INIT_LIST_HEAD(&obj_priv->list);
3635 +void i915_gem_free_object(struct drm_gem_object *obj)
3637 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
3639 + while (obj_priv->pin_count > 0)
3640 + i915_gem_object_unpin(obj);
3642 + i915_gem_object_unbind(obj);
3644 + drm_free(obj_priv->page_cpu_valid, 1, DRM_MEM_DRIVER);
3645 + drm_free(obj->driver_private, 1, DRM_MEM_DRIVER);
3649 +i915_gem_set_domain(struct drm_gem_object *obj,
3650 + struct drm_file *file_priv,
3651 + uint32_t read_domains,
3652 + uint32_t write_domain)
3654 + struct drm_device *dev = obj->dev;
3656 + uint32_t flush_domains;
3658 + BUG_ON(!mutex_is_locked(&dev->struct_mutex));
3660 + ret = i915_gem_object_set_domain(obj, read_domains, write_domain);
3663 + flush_domains = i915_gem_dev_set_domain(obj->dev);
3665 + if (flush_domains & ~(I915_GEM_DOMAIN_CPU|I915_GEM_DOMAIN_GTT))
3666 + (void) i915_add_request(dev, flush_domains);
3671 +/** Unbinds all objects that are on the given buffer list. */
3673 +i915_gem_evict_from_list(struct drm_device *dev, struct list_head *head)
3675 + struct drm_gem_object *obj;
3676 + struct drm_i915_gem_object *obj_priv;
3679 + while (!list_empty(head)) {
3680 + obj_priv = list_first_entry(head,
3681 + struct drm_i915_gem_object,
3683 + obj = obj_priv->obj;
3685 + if (obj_priv->pin_count != 0) {
3686 + DRM_ERROR("Pinned object in unbind list\n");
3687 + mutex_unlock(&dev->struct_mutex);
3691 + ret = i915_gem_object_unbind(obj);
3693 + DRM_ERROR("Error unbinding object in LeaveVT: %d\n",
3695 + mutex_unlock(&dev->struct_mutex);
3705 +i915_gem_idle(struct drm_device *dev)
3707 + drm_i915_private_t *dev_priv = dev->dev_private;
3708 + uint32_t seqno, cur_seqno, last_seqno;
3711 + if (dev_priv->mm.suspended)
3714 + /* Hack! Don't let anybody do execbuf while we don't control the chip.
3715 + * We need to replace this with a semaphore, or something.
3717 + dev_priv->mm.suspended = 1;
3719 + i915_kernel_lost_context(dev);
3721 + /* Flush the GPU along with all non-CPU write domains
3723 + i915_gem_flush(dev, ~(I915_GEM_DOMAIN_CPU|I915_GEM_DOMAIN_GTT),
3724 + ~(I915_GEM_DOMAIN_CPU|I915_GEM_DOMAIN_GTT));
3725 + seqno = i915_add_request(dev, ~(I915_GEM_DOMAIN_CPU |
3726 + I915_GEM_DOMAIN_GTT));
3729 + mutex_unlock(&dev->struct_mutex);
3733 + dev_priv->mm.waiting_gem_seqno = seqno;
3737 + cur_seqno = i915_get_gem_seqno(dev);
3738 + if (i915_seqno_passed(cur_seqno, seqno))
3740 + if (last_seqno == cur_seqno) {
3741 + if (stuck++ > 100) {
3742 + DRM_ERROR("hardware wedged\n");
3743 + dev_priv->mm.wedged = 1;
3744 + DRM_WAKEUP(&dev_priv->irq_queue);
3749 + last_seqno = cur_seqno;
3751 + dev_priv->mm.waiting_gem_seqno = 0;
3753 + i915_gem_retire_requests(dev);
3755 + /* Active and flushing should now be empty as we've
3756 + * waited for a sequence higher than any pending execbuffer
3758 + BUG_ON(!list_empty(&dev_priv->mm.active_list));
3759 + BUG_ON(!list_empty(&dev_priv->mm.flushing_list));
3761 + /* Request should now be empty as we've also waited
3762 + * for the last request in the list
3764 + BUG_ON(!list_empty(&dev_priv->mm.request_list));
3766 + /* Move all buffers out of the GTT. */
3767 + ret = i915_gem_evict_from_list(dev, &dev_priv->mm.inactive_list);
3771 + BUG_ON(!list_empty(&dev_priv->mm.active_list));
3772 + BUG_ON(!list_empty(&dev_priv->mm.flushing_list));
3773 + BUG_ON(!list_empty(&dev_priv->mm.inactive_list));
3774 + BUG_ON(!list_empty(&dev_priv->mm.request_list));
3779 +i915_gem_init_hws(struct drm_device *dev)
3781 + drm_i915_private_t *dev_priv = dev->dev_private;
3782 + struct drm_gem_object *obj;
3783 + struct drm_i915_gem_object *obj_priv;
3786 + /* If we need a physical address for the status page, it's already
3787 + * initialized at driver load time.
3789 + if (!I915_NEED_GFX_HWS(dev))
3792 + obj = drm_gem_object_alloc(dev, 4096);
3793 + if (obj == NULL) {
3794 + DRM_ERROR("Failed to allocate status page\n");
3797 + obj_priv = obj->driver_private;
3799 + ret = i915_gem_object_pin(obj, 4096);
3801 + drm_gem_object_unreference(obj);
3805 + dev_priv->status_gfx_addr = obj_priv->gtt_offset;
3806 + dev_priv->hws_map.offset = dev->agp->base + obj_priv->gtt_offset;
3807 + dev_priv->hws_map.size = 4096;
3808 + dev_priv->hws_map.type = 0;
3809 + dev_priv->hws_map.flags = 0;
3810 + dev_priv->hws_map.mtrr = 0;
3812 + drm_core_ioremap(&dev_priv->hws_map, dev);
3813 + if (dev_priv->hws_map.handle == NULL) {
3814 + DRM_ERROR("Failed to map status page.\n");
3815 + memset(&dev_priv->hws_map, 0, sizeof(dev_priv->hws_map));
3816 + drm_gem_object_unreference(obj);
3819 + dev_priv->hws_obj = obj;
3820 + dev_priv->hw_status_page = dev_priv->hws_map.handle;
3821 + memset(dev_priv->hw_status_page, 0, PAGE_SIZE);
3822 + I915_WRITE(HWS_PGA, dev_priv->status_gfx_addr);
3823 + DRM_DEBUG("hws offset: 0x%08x\n", dev_priv->status_gfx_addr);
3829 +i915_gem_init_ringbuffer(struct drm_device *dev)
3831 + drm_i915_private_t *dev_priv = dev->dev_private;
3832 + struct drm_gem_object *obj;
3833 + struct drm_i915_gem_object *obj_priv;
3836 + ret = i915_gem_init_hws(dev);
3840 + obj = drm_gem_object_alloc(dev, 128 * 1024);
3841 + if (obj == NULL) {
3842 + DRM_ERROR("Failed to allocate ringbuffer\n");
3845 + obj_priv = obj->driver_private;
3847 + ret = i915_gem_object_pin(obj, 4096);
3849 + drm_gem_object_unreference(obj);
3853 + /* Set up the kernel mapping for the ring. */
3854 + dev_priv->ring.Size = obj->size;
3855 + dev_priv->ring.tail_mask = obj->size - 1;
3857 + dev_priv->ring.map.offset = dev->agp->base + obj_priv->gtt_offset;
3858 + dev_priv->ring.map.size = obj->size;
3859 + dev_priv->ring.map.type = 0;
3860 + dev_priv->ring.map.flags = 0;
3861 + dev_priv->ring.map.mtrr = 0;
3863 + drm_core_ioremap(&dev_priv->ring.map, dev);
3864 + if (dev_priv->ring.map.handle == NULL) {
3865 + DRM_ERROR("Failed to map ringbuffer.\n");
3866 + memset(&dev_priv->ring, 0, sizeof(dev_priv->ring));
3867 + drm_gem_object_unreference(obj);
3870 + dev_priv->ring.ring_obj = obj;
3871 + dev_priv->ring.virtual_start = dev_priv->ring.map.handle;
3873 + /* Stop the ring if it's running. */
3874 + I915_WRITE(PRB0_CTL, 0);
3875 + I915_WRITE(PRB0_HEAD, 0);
3876 + I915_WRITE(PRB0_TAIL, 0);
3877 + I915_WRITE(PRB0_START, 0);
3879 + /* Initialize the ring. */
3880 + I915_WRITE(PRB0_START, obj_priv->gtt_offset);
3881 + I915_WRITE(PRB0_CTL,
3882 + ((obj->size - 4096) & RING_NR_PAGES) |
3886 + /* Update our cache of the ring state */
3887 + i915_kernel_lost_context(dev);
3893 +i915_gem_cleanup_ringbuffer(struct drm_device *dev)
3895 + drm_i915_private_t *dev_priv = dev->dev_private;
3897 + if (dev_priv->ring.ring_obj == NULL)
3900 + drm_core_ioremapfree(&dev_priv->ring.map, dev);
3902 + i915_gem_object_unpin(dev_priv->ring.ring_obj);
3903 + drm_gem_object_unreference(dev_priv->ring.ring_obj);
3904 + dev_priv->ring.ring_obj = NULL;
3905 + memset(&dev_priv->ring, 0, sizeof(dev_priv->ring));
3907 + if (dev_priv->hws_obj != NULL) {
3908 + i915_gem_object_unpin(dev_priv->hws_obj);
3909 + drm_gem_object_unreference(dev_priv->hws_obj);
3910 + dev_priv->hws_obj = NULL;
3911 + memset(&dev_priv->hws_map, 0, sizeof(dev_priv->hws_map));
3913 + /* Write high address into HWS_PGA when disabling. */
3914 + I915_WRITE(HWS_PGA, 0x1ffff000);
3919 +i915_gem_entervt_ioctl(struct drm_device *dev, void *data,
3920 + struct drm_file *file_priv)
3922 + drm_i915_private_t *dev_priv = dev->dev_private;
3925 + if (dev_priv->mm.wedged) {
3926 + DRM_ERROR("Reenabling wedged hardware, good luck\n");
3927 + dev_priv->mm.wedged = 0;
3930 + ret = i915_gem_init_ringbuffer(dev);
3934 + mutex_lock(&dev->struct_mutex);
3935 + BUG_ON(!list_empty(&dev_priv->mm.active_list));
3936 + BUG_ON(!list_empty(&dev_priv->mm.flushing_list));
3937 + BUG_ON(!list_empty(&dev_priv->mm.inactive_list));
3938 + BUG_ON(!list_empty(&dev_priv->mm.request_list));
3939 + dev_priv->mm.suspended = 0;
3940 + mutex_unlock(&dev->struct_mutex);
3945 +i915_gem_leavevt_ioctl(struct drm_device *dev, void *data,
3946 + struct drm_file *file_priv)
3950 + mutex_lock(&dev->struct_mutex);
3951 + ret = i915_gem_idle(dev);
3953 + i915_gem_cleanup_ringbuffer(dev);
3954 + mutex_unlock(&dev->struct_mutex);
3960 +i915_gem_lastclose(struct drm_device *dev)
3963 + drm_i915_private_t *dev_priv = dev->dev_private;
3965 + mutex_lock(&dev->struct_mutex);
3967 + if (dev_priv->ring.ring_obj != NULL) {
3968 + ret = i915_gem_idle(dev);
3970 + DRM_ERROR("failed to idle hardware: %d\n", ret);
3972 + i915_gem_cleanup_ringbuffer(dev);
3975 + mutex_unlock(&dev->struct_mutex);
3979 +i915_gem_load(struct drm_device *dev)
3981 + drm_i915_private_t *dev_priv = dev->dev_private;
3983 + INIT_LIST_HEAD(&dev_priv->mm.active_list);
3984 + INIT_LIST_HEAD(&dev_priv->mm.flushing_list);
3985 + INIT_LIST_HEAD(&dev_priv->mm.inactive_list);
3986 + INIT_LIST_HEAD(&dev_priv->mm.request_list);
3987 + INIT_DELAYED_WORK(&dev_priv->mm.retire_work,
3988 + i915_gem_retire_work_handler);
3989 + dev_priv->mm.next_gem_seqno = 1;
3991 + i915_gem_detect_bit_6_swizzle(dev);
3993 diff --git a/drivers/gpu/drm/i915/i915_gem_debug.c b/drivers/gpu/drm/i915/i915_gem_debug.c
3994 new file mode 100644
3995 index 0000000..131c088
3997 +++ b/drivers/gpu/drm/i915/i915_gem_debug.c
4000 + * Copyright © 2008 Intel Corporation
4002 + * Permission is hereby granted, free of charge, to any person obtaining a
4003 + * copy of this software and associated documentation files (the "Software"),
4004 + * to deal in the Software without restriction, including without limitation
4005 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
4006 + * and/or sell copies of the Software, and to permit persons to whom the
4007 + * Software is furnished to do so, subject to the following conditions:
4009 + * The above copyright notice and this permission notice (including the next
4010 + * paragraph) shall be included in all copies or substantial portions of the
4013 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
4014 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
4015 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
4016 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
4017 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
4018 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
4019 + * IN THE SOFTWARE.
4022 + * Keith Packard <keithp@keithp.com>
4028 +#include "i915_drm.h"
4029 +#include "i915_drv.h"
4033 +i915_verify_inactive(struct drm_device *dev, char *file, int line)
4035 + drm_i915_private_t *dev_priv = dev->dev_private;
4036 + struct drm_gem_object *obj;
4037 + struct drm_i915_gem_object *obj_priv;
4039 + list_for_each_entry(obj_priv, &dev_priv->mm.inactive_list, list) {
4040 + obj = obj_priv->obj;
4041 + if (obj_priv->pin_count || obj_priv->active ||
4042 + (obj->write_domain & ~(I915_GEM_DOMAIN_CPU |
4043 + I915_GEM_DOMAIN_GTT)))
4044 + DRM_ERROR("inactive %p (p %d a %d w %x) %s:%d\n",
4046 + obj_priv->pin_count, obj_priv->active,
4047 + obj->write_domain, file, line);
4050 +#endif /* WATCH_INACTIVE */
4053 +#if WATCH_BUF | WATCH_EXEC | WATCH_PWRITE
4055 +i915_gem_dump_page(struct page *page, uint32_t start, uint32_t end,
4056 + uint32_t bias, uint32_t mark)
4058 + uint32_t *mem = kmap_atomic(page, KM_USER0);
4060 + for (i = start; i < end; i += 4)
4061 + DRM_INFO("%08x: %08x%s\n",
4062 + (int) (bias + i), mem[i / 4],
4063 + (bias + i == mark) ? " ********" : "");
4064 + kunmap_atomic(mem, KM_USER0);
4065 + /* give syslog time to catch up */
4070 +i915_gem_dump_object(struct drm_gem_object *obj, int len,
4071 + const char *where, uint32_t mark)
4073 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
4076 + DRM_INFO("%s: object at offset %08x\n", where, obj_priv->gtt_offset);
4077 + for (page = 0; page < (len + PAGE_SIZE-1) / PAGE_SIZE; page++) {
4078 + int page_len, chunk, chunk_len;
4080 + page_len = len - page * PAGE_SIZE;
4081 + if (page_len > PAGE_SIZE)
4082 + page_len = PAGE_SIZE;
4084 + for (chunk = 0; chunk < page_len; chunk += 128) {
4085 + chunk_len = page_len - chunk;
4086 + if (chunk_len > 128)
4088 + i915_gem_dump_page(obj_priv->page_list[page],
4089 + chunk, chunk + chunk_len,
4090 + obj_priv->gtt_offset +
4100 +i915_dump_lru(struct drm_device *dev, const char *where)
4102 + drm_i915_private_t *dev_priv = dev->dev_private;
4103 + struct drm_i915_gem_object *obj_priv;
4105 + DRM_INFO("active list %s {\n", where);
4106 + list_for_each_entry(obj_priv, &dev_priv->mm.active_list,
4109 + DRM_INFO(" %p: %08x\n", obj_priv,
4110 + obj_priv->last_rendering_seqno);
4113 + DRM_INFO("flushing list %s {\n", where);
4114 + list_for_each_entry(obj_priv, &dev_priv->mm.flushing_list,
4117 + DRM_INFO(" %p: %08x\n", obj_priv,
4118 + obj_priv->last_rendering_seqno);
4121 + DRM_INFO("inactive %s {\n", where);
4122 + list_for_each_entry(obj_priv, &dev_priv->mm.inactive_list, list) {
4123 + DRM_INFO(" %p: %08x\n", obj_priv,
4124 + obj_priv->last_rendering_seqno);
4131 +#if WATCH_COHERENCY
4133 +i915_gem_object_check_coherency(struct drm_gem_object *obj, int handle)
4135 + struct drm_device *dev = obj->dev;
4136 + struct drm_i915_gem_object *obj_priv = obj->driver_private;
4138 + uint32_t *gtt_mapping;
4139 + uint32_t *backing_map = NULL;
4140 + int bad_count = 0;
4142 + DRM_INFO("%s: checking coherency of object %p@0x%08x (%d, %dkb):\n",
4143 + __func__, obj, obj_priv->gtt_offset, handle,
4144 + obj->size / 1024);
4146 + gtt_mapping = ioremap(dev->agp->base + obj_priv->gtt_offset,
4148 + if (gtt_mapping == NULL) {
4149 + DRM_ERROR("failed to map GTT space\n");
4153 + for (page = 0; page < obj->size / PAGE_SIZE; page++) {
4156 + backing_map = kmap_atomic(obj_priv->page_list[page], KM_USER0);
4158 + if (backing_map == NULL) {
4159 + DRM_ERROR("failed to map backing page\n");
4163 + for (i = 0; i < PAGE_SIZE / 4; i++) {
4164 + uint32_t cpuval = backing_map[i];
4165 + uint32_t gttval = readl(gtt_mapping +
4168 + if (cpuval != gttval) {
4169 + DRM_INFO("incoherent CPU vs GPU at 0x%08x: "
4170 + "0x%08x vs 0x%08x\n",
4171 + (int)(obj_priv->gtt_offset +
4172 + page * PAGE_SIZE + i * 4),
4174 + if (bad_count++ >= 8) {
4175 + DRM_INFO("...\n");
4180 + kunmap_atomic(backing_map, KM_USER0);
4181 + backing_map = NULL;
4185 + if (backing_map != NULL)
4186 + kunmap_atomic(backing_map, KM_USER0);
4187 + iounmap(gtt_mapping);
4189 + /* give syslog time to catch up */
4192 + /* Directly flush the object, since we just loaded values with the CPU
4193 + * from the backing pages and we don't want to disturb the cache
4194 + * management that we're trying to observe.
4197 + i915_gem_clflush_object(obj);
4200 diff --git a/drivers/gpu/drm/i915/i915_gem_proc.c b/drivers/gpu/drm/i915/i915_gem_proc.c
4201 new file mode 100644
4202 index 0000000..15d4160
4204 +++ b/drivers/gpu/drm/i915/i915_gem_proc.c
4207 + * Copyright © 2008 Intel Corporation
4209 + * Permission is hereby granted, free of charge, to any person obtaining a
4210 + * copy of this software and associated documentation files (the "Software"),
4211 + * to deal in the Software without restriction, including without limitation
4212 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
4213 + * and/or sell copies of the Software, and to permit persons to whom the
4214 + * Software is furnished to do so, subject to the following conditions:
4216 + * The above copyright notice and this permission notice (including the next
4217 + * paragraph) shall be included in all copies or substantial portions of the
4220 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
4221 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
4222 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
4223 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
4224 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
4225 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
4226 + * IN THE SOFTWARE.
4229 + * Eric Anholt <eric@anholt.net>
4230 + * Keith Packard <keithp@keithp.com>
4236 +#include "i915_drm.h"
4237 +#include "i915_drv.h"
4239 +static int i915_gem_active_info(char *buf, char **start, off_t offset,
4240 + int request, int *eof, void *data)
4242 + struct drm_minor *minor = (struct drm_minor *) data;
4243 + struct drm_device *dev = minor->dev;
4244 + drm_i915_private_t *dev_priv = dev->dev_private;
4245 + struct drm_i915_gem_object *obj_priv;
4248 + if (offset > DRM_PROC_LIMIT) {
4253 + *start = &buf[offset];
4255 + DRM_PROC_PRINT("Active:\n");
4256 + list_for_each_entry(obj_priv, &dev_priv->mm.active_list,
4259 + struct drm_gem_object *obj = obj_priv->obj;
4261 + DRM_PROC_PRINT(" %p(%d): %08x %08x %d\n",
4263 + obj->read_domains, obj->write_domain,
4264 + obj_priv->last_rendering_seqno);
4266 + DRM_PROC_PRINT(" %p: %08x %08x %d\n",
4268 + obj->read_domains, obj->write_domain,
4269 + obj_priv->last_rendering_seqno);
4272 + if (len > request + offset)
4275 + return len - offset;
4278 +static int i915_gem_flushing_info(char *buf, char **start, off_t offset,
4279 + int request, int *eof, void *data)
4281 + struct drm_minor *minor = (struct drm_minor *) data;
4282 + struct drm_device *dev = minor->dev;
4283 + drm_i915_private_t *dev_priv = dev->dev_private;
4284 + struct drm_i915_gem_object *obj_priv;
4287 + if (offset > DRM_PROC_LIMIT) {
4292 + *start = &buf[offset];
4294 + DRM_PROC_PRINT("Flushing:\n");
4295 + list_for_each_entry(obj_priv, &dev_priv->mm.flushing_list,
4298 + struct drm_gem_object *obj = obj_priv->obj;
4300 + DRM_PROC_PRINT(" %p(%d): %08x %08x %d\n",
4302 + obj->read_domains, obj->write_domain,
4303 + obj_priv->last_rendering_seqno);
4305 + DRM_PROC_PRINT(" %p: %08x %08x %d\n", obj,
4306 + obj->read_domains, obj->write_domain,
4307 + obj_priv->last_rendering_seqno);
4310 + if (len > request + offset)
4313 + return len - offset;
4316 +static int i915_gem_inactive_info(char *buf, char **start, off_t offset,
4317 + int request, int *eof, void *data)
4319 + struct drm_minor *minor = (struct drm_minor *) data;
4320 + struct drm_device *dev = minor->dev;
4321 + drm_i915_private_t *dev_priv = dev->dev_private;
4322 + struct drm_i915_gem_object *obj_priv;
4325 + if (offset > DRM_PROC_LIMIT) {
4330 + *start = &buf[offset];
4332 + DRM_PROC_PRINT("Inactive:\n");
4333 + list_for_each_entry(obj_priv, &dev_priv->mm.inactive_list,
4336 + struct drm_gem_object *obj = obj_priv->obj;
4338 + DRM_PROC_PRINT(" %p(%d): %08x %08x %d\n",
4340 + obj->read_domains, obj->write_domain,
4341 + obj_priv->last_rendering_seqno);
4343 + DRM_PROC_PRINT(" %p: %08x %08x %d\n", obj,
4344 + obj->read_domains, obj->write_domain,
4345 + obj_priv->last_rendering_seqno);
4348 + if (len > request + offset)
4351 + return len - offset;
4354 +static int i915_gem_request_info(char *buf, char **start, off_t offset,
4355 + int request, int *eof, void *data)
4357 + struct drm_minor *minor = (struct drm_minor *) data;
4358 + struct drm_device *dev = minor->dev;
4359 + drm_i915_private_t *dev_priv = dev->dev_private;
4360 + struct drm_i915_gem_request *gem_request;
4363 + if (offset > DRM_PROC_LIMIT) {
4368 + *start = &buf[offset];
4370 + DRM_PROC_PRINT("Request:\n");
4371 + list_for_each_entry(gem_request, &dev_priv->mm.request_list,
4374 + DRM_PROC_PRINT(" %d @ %d %08x\n",
4375 + gem_request->seqno,
4376 + (int) (jiffies - gem_request->emitted_jiffies),
4377 + gem_request->flush_domains);
4379 + if (len > request + offset)
4382 + return len - offset;
4385 +static int i915_gem_seqno_info(char *buf, char **start, off_t offset,
4386 + int request, int *eof, void *data)
4388 + struct drm_minor *minor = (struct drm_minor *) data;
4389 + struct drm_device *dev = minor->dev;
4390 + drm_i915_private_t *dev_priv = dev->dev_private;
4393 + if (offset > DRM_PROC_LIMIT) {
4398 + *start = &buf[offset];
4400 + DRM_PROC_PRINT("Current sequence: %d\n", i915_get_gem_seqno(dev));
4401 + DRM_PROC_PRINT("Waiter sequence: %d\n",
4402 + dev_priv->mm.waiting_gem_seqno);
4403 + DRM_PROC_PRINT("IRQ sequence: %d\n", dev_priv->mm.irq_gem_seqno);
4404 + if (len > request + offset)
4407 + return len - offset;
4411 +static int i915_interrupt_info(char *buf, char **start, off_t offset,
4412 + int request, int *eof, void *data)
4414 + struct drm_minor *minor = (struct drm_minor *) data;
4415 + struct drm_device *dev = minor->dev;
4416 + drm_i915_private_t *dev_priv = dev->dev_private;
4419 + if (offset > DRM_PROC_LIMIT) {
4424 + *start = &buf[offset];
4426 + DRM_PROC_PRINT("Interrupt enable: %08x\n",
4428 + DRM_PROC_PRINT("Interrupt identity: %08x\n",
4430 + DRM_PROC_PRINT("Interrupt mask: %08x\n",
4432 + DRM_PROC_PRINT("Pipe A stat: %08x\n",
4433 + I915_READ(PIPEASTAT));
4434 + DRM_PROC_PRINT("Pipe B stat: %08x\n",
4435 + I915_READ(PIPEBSTAT));
4436 + DRM_PROC_PRINT("Interrupts received: %d\n",
4437 + atomic_read(&dev_priv->irq_received));
4438 + DRM_PROC_PRINT("Current sequence: %d\n",
4439 + i915_get_gem_seqno(dev));
4440 + DRM_PROC_PRINT("Waiter sequence: %d\n",
4441 + dev_priv->mm.waiting_gem_seqno);
4442 + DRM_PROC_PRINT("IRQ sequence: %d\n",
4443 + dev_priv->mm.irq_gem_seqno);
4444 + if (len > request + offset)
4447 + return len - offset;
4450 +static struct drm_proc_list {
4453 + /** proc callback*/
4454 + int (*f) (char *, char **, off_t, int, int *, void *);
4455 +} i915_gem_proc_list[] = {
4456 + {"i915_gem_active", i915_gem_active_info},
4457 + {"i915_gem_flushing", i915_gem_flushing_info},
4458 + {"i915_gem_inactive", i915_gem_inactive_info},
4459 + {"i915_gem_request", i915_gem_request_info},
4460 + {"i915_gem_seqno", i915_gem_seqno_info},
4461 + {"i915_gem_interrupt", i915_interrupt_info},
4464 +#define I915_GEM_PROC_ENTRIES ARRAY_SIZE(i915_gem_proc_list)
4466 +int i915_gem_proc_init(struct drm_minor *minor)
4468 + struct proc_dir_entry *ent;
4471 + for (i = 0; i < I915_GEM_PROC_ENTRIES; i++) {
4472 + ent = create_proc_entry(i915_gem_proc_list[i].name,
4473 + S_IFREG | S_IRUGO, minor->dev_root);
4475 + DRM_ERROR("Cannot create /proc/dri/.../%s\n",
4476 + i915_gem_proc_list[i].name);
4477 + for (j = 0; j < i; j++)
4478 + remove_proc_entry(i915_gem_proc_list[i].name,
4482 + ent->read_proc = i915_gem_proc_list[i].f;
4483 + ent->data = minor;
4488 +void i915_gem_proc_cleanup(struct drm_minor *minor)
4492 + if (!minor->dev_root)
4495 + for (i = 0; i < I915_GEM_PROC_ENTRIES; i++)
4496 + remove_proc_entry(i915_gem_proc_list[i].name, minor->dev_root);
4498 diff --git a/drivers/gpu/drm/i915/i915_gem_tiling.c b/drivers/gpu/drm/i915/i915_gem_tiling.c
4499 new file mode 100644
4500 index 0000000..0c1b3a0
4502 +++ b/drivers/gpu/drm/i915/i915_gem_tiling.c
4505 + * Copyright © 2008 Intel Corporation
4507 + * Permission is hereby granted, free of charge, to any person obtaining a
4508 + * copy of this software and associated documentation files (the "Software"),
4509 + * to deal in the Software without restriction, including without limitation
4510 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
4511 + * and/or sell copies of the Software, and to permit persons to whom the
4512 + * Software is furnished to do so, subject to the following conditions:
4514 + * The above copyright notice and this permission notice (including the next
4515 + * paragraph) shall be included in all copies or substantial portions of the
4518 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
4519 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
4520 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
4521 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
4522 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
4523 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
4524 + * IN THE SOFTWARE.
4527 + * Eric Anholt <eric@anholt.net>
4533 +#include "i915_drm.h"
4534 +#include "i915_drv.h"
4536 +/** @file i915_gem_tiling.c
4538 + * Support for managing tiling state of buffer objects.
4540 + * The idea behind tiling is to increase cache hit rates by rearranging
4541 + * pixel data so that a group of pixel accesses are in the same cacheline.
4542 + * Performance improvement from doing this on the back/depth buffer are on
4543 + * the order of 30%.
4545 + * Intel architectures make this somewhat more complicated, though, by
4546 + * adjustments made to addressing of data when the memory is in interleaved
4547 + * mode (matched pairs of DIMMS) to improve memory bandwidth.
4548 + * For interleaved memory, the CPU sends every sequential 64 bytes
4549 + * to an alternate memory channel so it can get the bandwidth from both.
4551 + * The GPU also rearranges its accesses for increased bandwidth to interleaved
4552 + * memory, and it matches what the CPU does for non-tiled. However, when tiled
4553 + * it does it a little differently, since one walks addresses not just in the
4554 + * X direction but also Y. So, along with alternating channels when bit
4555 + * 6 of the address flips, it also alternates when other bits flip -- Bits 9
4556 + * (every 512 bytes, an X tile scanline) and 10 (every two X tile scanlines)
4557 + * are common to both the 915 and 965-class hardware.
4559 + * The CPU also sometimes XORs in higher bits as well, to improve
4560 + * bandwidth doing strided access like we do so frequently in graphics. This
4561 + * is called "Channel XOR Randomization" in the MCH documentation. The result
4562 + * is that the CPU is XORing in either bit 11 or bit 17 to bit 6 of its address
4565 + * All of this bit 6 XORing has an effect on our memory management,
4566 + * as we need to make sure that the 3d driver can correctly address object
4569 + * If we don't have interleaved memory, all tiling is safe and no swizzling is
4572 + * When bit 17 is XORed in, we simply refuse to tile at all. Bit
4573 + * 17 is not just a page offset, so as we page an objet out and back in,
4574 + * individual pages in it will have different bit 17 addresses, resulting in
4575 + * each 64 bytes being swapped with its neighbor!
4577 + * Otherwise, if interleaved, we have to tell the 3d driver what the address
4578 + * swizzling it needs to do is, since it's writing with the CPU to the pages
4579 + * (bit 6 and potentially bit 11 XORed in), and the GPU is reading from the
4580 + * pages (bit 6, 9, and 10 XORed in), resulting in a cumulative bit swizzling
4581 + * required by the CPU of XORing in bit 6, 9, 10, and potentially 11, in order
4582 + * to match what the GPU expects.
4586 + * Detects bit 6 swizzling of address lookup between IGD access and CPU
4587 + * access through main memory.
4590 +i915_gem_detect_bit_6_swizzle(struct drm_device *dev)
4592 + drm_i915_private_t *dev_priv = dev->dev_private;
4593 + uint32_t swizzle_x = I915_BIT_6_SWIZZLE_UNKNOWN;
4594 + uint32_t swizzle_y = I915_BIT_6_SWIZZLE_UNKNOWN;
4596 + if (!IS_I9XX(dev)) {
4597 + /* As far as we know, the 865 doesn't have these bit 6
4598 + * swizzling issues.
4600 + swizzle_x = I915_BIT_6_SWIZZLE_NONE;
4601 + swizzle_y = I915_BIT_6_SWIZZLE_NONE;
4602 + } else if (!IS_I965G(dev) || IS_I965GM(dev)) {
4605 + /* On 915-945 and GM965, channel interleave by the CPU is
4606 + * determined by DCC. The CPU will alternate based on bit 6
4607 + * in interleaved mode, and the GPU will then also alternate
4608 + * on bit 6, 9, and 10 for X, but the CPU may also optionally
4609 + * alternate based on bit 17 (XOR not disabled and XOR
4612 + dcc = I915_READ(DCC);
4613 + switch (dcc & DCC_ADDRESSING_MODE_MASK) {
4614 + case DCC_ADDRESSING_MODE_SINGLE_CHANNEL:
4615 + case DCC_ADDRESSING_MODE_DUAL_CHANNEL_ASYMMETRIC:
4616 + swizzle_x = I915_BIT_6_SWIZZLE_NONE;
4617 + swizzle_y = I915_BIT_6_SWIZZLE_NONE;
4619 + case DCC_ADDRESSING_MODE_DUAL_CHANNEL_INTERLEAVED:
4620 + if (IS_I915G(dev) || IS_I915GM(dev) ||
4621 + dcc & DCC_CHANNEL_XOR_DISABLE) {
4622 + swizzle_x = I915_BIT_6_SWIZZLE_9_10;
4623 + swizzle_y = I915_BIT_6_SWIZZLE_9;
4624 + } else if (IS_I965GM(dev)) {
4625 + /* GM965 only does bit 11-based channel
4628 + swizzle_x = I915_BIT_6_SWIZZLE_9_10_11;
4629 + swizzle_y = I915_BIT_6_SWIZZLE_9_11;
4631 + /* Bit 17 or perhaps other swizzling */
4632 + swizzle_x = I915_BIT_6_SWIZZLE_UNKNOWN;
4633 + swizzle_y = I915_BIT_6_SWIZZLE_UNKNOWN;
4637 + if (dcc == 0xffffffff) {
4638 + DRM_ERROR("Couldn't read from MCHBAR. "
4639 + "Disabling tiling.\n");
4640 + swizzle_x = I915_BIT_6_SWIZZLE_UNKNOWN;
4641 + swizzle_y = I915_BIT_6_SWIZZLE_UNKNOWN;
4644 + /* The 965, G33, and newer, have a very flexible memory
4645 + * configuration. It will enable dual-channel mode
4646 + * (interleaving) on as much memory as it can, and the GPU
4647 + * will additionally sometimes enable different bit 6
4648 + * swizzling for tiled objects from the CPU.
4650 + * Here's what I found on the G965:
4651 + * slot fill memory size swizzling
4652 + * 0A 0B 1A 1B 1-ch 2-ch
4653 + * 512 0 0 0 512 0 O
4654 + * 512 0 512 0 16 1008 X
4655 + * 512 0 0 512 16 1008 X
4656 + * 0 512 0 512 16 1008 X
4657 + * 1024 1024 1024 0 2048 1024 O
4659 + * We could probably detect this based on either the DRB
4660 + * matching, which was the case for the swizzling required in
4661 + * the table above, or from the 1-ch value being less than
4662 + * the minimum size of a rank.
4664 + if (I915_READ16(C0DRB3) != I915_READ16(C1DRB3)) {
4665 + swizzle_x = I915_BIT_6_SWIZZLE_NONE;
4666 + swizzle_y = I915_BIT_6_SWIZZLE_NONE;
4668 + swizzle_x = I915_BIT_6_SWIZZLE_9_10;
4669 + swizzle_y = I915_BIT_6_SWIZZLE_9;
4673 + dev_priv->mm.bit_6_swizzle_x = swizzle_x;
4674 + dev_priv->mm.bit_6_swizzle_y = swizzle_y;
4678 + * Sets the tiling mode of an object, returning the required swizzling of
4679 + * bit 6 of addresses in the object.
4682 +i915_gem_set_tiling(struct drm_device *dev, void *data,
4683 + struct drm_file *file_priv)
4685 + struct drm_i915_gem_set_tiling *args = data;
4686 + drm_i915_private_t *dev_priv = dev->dev_private;
4687 + struct drm_gem_object *obj;
4688 + struct drm_i915_gem_object *obj_priv;
4690 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
4693 + obj_priv = obj->driver_private;
4695 + mutex_lock(&dev->struct_mutex);
4697 + if (args->tiling_mode == I915_TILING_NONE) {
4698 + obj_priv->tiling_mode = I915_TILING_NONE;
4699 + args->swizzle_mode = I915_BIT_6_SWIZZLE_NONE;
4701 + if (args->tiling_mode == I915_TILING_X)
4702 + args->swizzle_mode = dev_priv->mm.bit_6_swizzle_x;
4704 + args->swizzle_mode = dev_priv->mm.bit_6_swizzle_y;
4705 + /* If we can't handle the swizzling, make it untiled. */
4706 + if (args->swizzle_mode == I915_BIT_6_SWIZZLE_UNKNOWN) {
4707 + args->tiling_mode = I915_TILING_NONE;
4708 + args->swizzle_mode = I915_BIT_6_SWIZZLE_NONE;
4711 + obj_priv->tiling_mode = args->tiling_mode;
4713 + mutex_unlock(&dev->struct_mutex);
4715 + drm_gem_object_unreference(obj);
4721 + * Returns the current tiling mode and required bit 6 swizzling for the object.
4724 +i915_gem_get_tiling(struct drm_device *dev, void *data,
4725 + struct drm_file *file_priv)
4727 + struct drm_i915_gem_get_tiling *args = data;
4728 + drm_i915_private_t *dev_priv = dev->dev_private;
4729 + struct drm_gem_object *obj;
4730 + struct drm_i915_gem_object *obj_priv;
4732 + obj = drm_gem_object_lookup(dev, file_priv, args->handle);
4735 + obj_priv = obj->driver_private;
4737 + mutex_lock(&dev->struct_mutex);
4739 + args->tiling_mode = obj_priv->tiling_mode;
4740 + switch (obj_priv->tiling_mode) {
4741 + case I915_TILING_X:
4742 + args->swizzle_mode = dev_priv->mm.bit_6_swizzle_x;
4744 + case I915_TILING_Y:
4745 + args->swizzle_mode = dev_priv->mm.bit_6_swizzle_y;
4747 + case I915_TILING_NONE:
4748 + args->swizzle_mode = I915_BIT_6_SWIZZLE_NONE;
4751 + DRM_ERROR("unknown tiling mode\n");
4754 + mutex_unlock(&dev->struct_mutex);
4756 + drm_gem_object_unreference(obj);
4760 diff --git a/drivers/gpu/drm/i915/i915_irq.c b/drivers/gpu/drm/i915/i915_irq.c
4761 index f875959..f295bdf 100644
4762 --- a/drivers/gpu/drm/i915/i915_irq.c
4763 +++ b/drivers/gpu/drm/i915/i915_irq.c
4764 @@ -407,15 +407,20 @@ irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS)
4765 I915_WRITE(PIPEBSTAT, pipeb_stats);
4768 - if (iir & I915_ASLE_INTERRUPT)
4769 - opregion_asle_intr(dev);
4770 + I915_WRITE(IIR, iir);
4771 + if (dev->pdev->msi_enabled)
4772 + I915_WRITE(IMR, dev_priv->irq_mask_reg);
4773 + (void) I915_READ(IIR); /* Flush posted writes */
4775 dev_priv->sarea_priv->last_dispatch = READ_BREADCRUMB(dev_priv);
4777 - if (dev->pdev->msi_enabled)
4778 - I915_WRITE(IMR, dev_priv->irq_mask_reg);
4779 - I915_WRITE(IIR, iir);
4780 - (void) I915_READ(IIR);
4781 + if (iir & I915_USER_INTERRUPT) {
4782 + dev_priv->mm.irq_gem_seqno = i915_get_gem_seqno(dev);
4783 + DRM_WAKEUP(&dev_priv->irq_queue);
4786 + if (iir & I915_ASLE_INTERRUPT)
4787 + opregion_asle_intr(dev);
4789 if (vblank && dev_priv->swaps_pending > 0)
4790 drm_locked_tasklet(dev, i915_vblank_tasklet);
4791 @@ -449,7 +454,7 @@ static int i915_emit_irq(struct drm_device * dev)
4792 return dev_priv->counter;
4795 -static void i915_user_irq_get(struct drm_device *dev)
4796 +void i915_user_irq_get(struct drm_device *dev)
4798 drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
4800 diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h
4801 index 43ad2cb..5c2d9f2 100644
4802 --- a/drivers/gpu/drm/i915/i915_reg.h
4803 +++ b/drivers/gpu/drm/i915/i915_reg.h
4805 #ifndef _I915_REG_H_
4806 #define _I915_REG_H_
4808 -/* MCH MMIO space */
4809 -/** 915-945 and GM965 MCH register controlling DRAM channel access */
4811 -#define DCC_ADDRESSING_MODE_SINGLE_CHANNEL (0 << 0)
4812 -#define DCC_ADDRESSING_MODE_DUAL_CHANNEL_ASYMMETRIC (1 << 0)
4813 -#define DCC_ADDRESSING_MODE_DUAL_CHANNEL_INTERLEAVED (2 << 0)
4814 -#define DCC_ADDRESSING_MODE_MASK (3 << 0)
4815 -#define DCC_CHANNEL_XOR_DISABLE (1 << 10)
4817 -/** 965 MCH register controlling DRAM channel configuration */
4818 -#define CHDECMISC 0x111
4819 -#define CHDECMISC_FLEXMEMORY (1 << 1)
4822 * The Bridge device's PCI config space has information about the
4823 * fb aperture size and the amount of pre-reserved memory.
4824 @@ -516,6 +503,30 @@
4825 #define PALETTE_A 0x0a000
4826 #define PALETTE_B 0x0a800
4828 +/* MCH MMIO space */
4833 + * This mirrors the MCHBAR MMIO space whose location is determined by
4834 + * device 0 function 0's pci config register 0x44 or 0x48 and matches it in
4835 + * every way. It is not accessible from the CP register read instructions.
4838 +#define MCHBAR_MIRROR_BASE 0x10000
4840 +/** 915-945 and GM965 MCH register controlling DRAM channel access */
4841 +#define DCC 0x10200
4842 +#define DCC_ADDRESSING_MODE_SINGLE_CHANNEL (0 << 0)
4843 +#define DCC_ADDRESSING_MODE_DUAL_CHANNEL_ASYMMETRIC (1 << 0)
4844 +#define DCC_ADDRESSING_MODE_DUAL_CHANNEL_INTERLEAVED (2 << 0)
4845 +#define DCC_ADDRESSING_MODE_MASK (3 << 0)
4846 +#define DCC_CHANNEL_XOR_DISABLE (1 << 10)
4848 +/** 965 MCH register controlling DRAM channel configuration */
4849 +#define C0DRB3 0x10206
4850 +#define C1DRB3 0x10606
4855 diff --git a/include/drm/drm.h b/include/drm/drm.h
4856 index 15e5503..f46ba4b 100644
4857 --- a/include/drm/drm.h
4858 +++ b/include/drm/drm.h
4859 @@ -570,6 +570,34 @@ struct drm_set_version {
4863 +/** DRM_IOCTL_GEM_CLOSE ioctl argument type */
4864 +struct drm_gem_close {
4865 + /** Handle of the object to be closed. */
4870 +/** DRM_IOCTL_GEM_FLINK ioctl argument type */
4871 +struct drm_gem_flink {
4872 + /** Handle for the object being named */
4875 + /** Returned global name */
4879 +/** DRM_IOCTL_GEM_OPEN ioctl argument type */
4880 +struct drm_gem_open {
4881 + /** Name of object being opened */
4884 + /** Returned handle for the object */
4887 + /** Returned size of the object */
4891 #define DRM_IOCTL_BASE 'd'
4892 #define DRM_IO(nr) _IO(DRM_IOCTL_BASE,nr)
4893 #define DRM_IOR(nr,type) _IOR(DRM_IOCTL_BASE,nr,type)
4894 @@ -585,6 +613,9 @@ struct drm_set_version {
4895 #define DRM_IOCTL_GET_STATS DRM_IOR( 0x06, struct drm_stats)
4896 #define DRM_IOCTL_SET_VERSION DRM_IOWR(0x07, struct drm_set_version)
4897 #define DRM_IOCTL_MODESET_CTL DRM_IOW(0x08, struct drm_modeset_ctl)
4898 +#define DRM_IOCTL_GEM_CLOSE DRM_IOW (0x09, struct drm_gem_close)
4899 +#define DRM_IOCTL_GEM_FLINK DRM_IOWR(0x0a, struct drm_gem_flink)
4900 +#define DRM_IOCTL_GEM_OPEN DRM_IOWR(0x0b, struct drm_gem_open)
4902 #define DRM_IOCTL_SET_UNIQUE DRM_IOW( 0x10, struct drm_unique)
4903 #define DRM_IOCTL_AUTH_MAGIC DRM_IOW( 0x11, struct drm_auth)
4904 diff --git a/include/drm/drmP.h b/include/drm/drmP.h
4905 index e79ce07..1469a1b 100644
4906 --- a/include/drm/drmP.h
4907 +++ b/include/drm/drmP.h
4908 @@ -104,6 +104,7 @@ struct drm_device;
4909 #define DRIVER_DMA_QUEUE 0x200
4910 #define DRIVER_FB_DMA 0x400
4911 #define DRIVER_IRQ_VBL2 0x800
4912 +#define DRIVER_GEM 0x1000
4914 /***********************************************************************/
4915 /** \name Begin the DRM... */
4916 @@ -387,6 +388,10 @@ struct drm_file {
4917 struct drm_minor *minor;
4918 int remove_auth_on_close;
4919 unsigned long lock_count;
4920 + /** Mapping of mm object handles to object pointers. */
4921 + struct idr object_idr;
4922 + /** Lock for synchronization of access to object_idr. */
4923 + spinlock_t table_lock;
4927 @@ -558,6 +563,56 @@ struct drm_ati_pcigart_info {
4931 + * This structure defines the drm_mm memory object, which will be used by the
4932 + * DRM for its buffer objects.
4934 +struct drm_gem_object {
4935 + /** Reference count of this object */
4936 + struct kref refcount;
4938 + /** Handle count of this object. Each handle also holds a reference */
4939 + struct kref handlecount;
4941 + /** Related drm device */
4942 + struct drm_device *dev;
4944 + /** File representing the shmem storage */
4945 + struct file *filp;
4948 + * Size of the object, in bytes. Immutable over the object's
4954 + * Global name for this object, starts at 1. 0 means unnamed.
4955 + * Access is covered by the object_name_lock in the related drm_device
4960 + * Memory domains. These monitor which caches contain read/write data
4961 + * related to the object. When transitioning from one set of domains
4962 + * to another, the driver is called to ensure that caches are suitably
4963 + * flushed and invalidated
4965 + uint32_t read_domains;
4966 + uint32_t write_domain;
4969 + * While validating an exec operation, the
4970 + * new read/write domain values are computed here.
4971 + * They will be transferred to the above values
4972 + * at the point that any cache flushing occurs
4974 + uint32_t pending_read_domains;
4975 + uint32_t pending_write_domain;
4977 + void *driver_private;
4981 * DRM driver structure. This structure represent the common code for
4982 * a family of cards. There will one drm_device for each card present
4984 @@ -657,6 +712,18 @@ struct drm_driver {
4985 void (*set_version) (struct drm_device *dev,
4986 struct drm_set_version *sv);
4988 + int (*proc_init)(struct drm_minor *minor);
4989 + void (*proc_cleanup)(struct drm_minor *minor);
4992 + * Driver-specific constructor for drm_gem_objects, to set up
4993 + * obj->driver_private.
4995 + * Returns 0 on success.
4997 + int (*gem_init_object) (struct drm_gem_object *obj);
4998 + void (*gem_free_object) (struct drm_gem_object *obj);
5003 @@ -830,6 +897,22 @@ struct drm_device {
5004 spinlock_t drw_lock;
5008 + /** \name GEM information */
5010 + spinlock_t object_name_lock;
5011 + struct idr object_name_idr;
5012 + atomic_t object_count;
5013 + atomic_t object_memory;
5014 + atomic_t pin_count;
5015 + atomic_t pin_memory;
5016 + atomic_t gtt_count;
5017 + atomic_t gtt_memory;
5018 + uint32_t gtt_total;
5019 + uint32_t invalidate_domains; /* domains pending invalidation */
5020 + uint32_t flush_domains; /* domains pending flush */
5025 static __inline__ int drm_core_check_feature(struct drm_device *dev,
5026 @@ -926,6 +1009,10 @@ extern void *drm_realloc(void *oldpt, size_t oldsize, size_t size, int area);
5027 extern DRM_AGP_MEM *drm_alloc_agp(struct drm_device *dev, int pages, u32 type);
5028 extern int drm_free_agp(DRM_AGP_MEM * handle, int pages);
5029 extern int drm_bind_agp(DRM_AGP_MEM * handle, unsigned int start);
5030 +extern DRM_AGP_MEM *drm_agp_bind_pages(struct drm_device *dev,
5031 + struct page **pages,
5032 + unsigned long num_pages,
5033 + uint32_t gtt_offset);
5034 extern int drm_unbind_agp(DRM_AGP_MEM * handle);
5036 /* Misc. IOCTL support (drm_ioctl.h) */
5037 @@ -988,6 +1075,9 @@ extern int drm_getmagic(struct drm_device *dev, void *data,
5038 extern int drm_authmagic(struct drm_device *dev, void *data,
5039 struct drm_file *file_priv);
5041 +/* Cache management (drm_cache.c) */
5042 +void drm_clflush_pages(struct page *pages[], unsigned long num_pages);
5044 /* Locking IOCTL support (drm_lock.h) */
5045 extern int drm_lock(struct drm_device *dev, void *data,
5046 struct drm_file *file_priv);
5047 @@ -1094,6 +1184,7 @@ extern DRM_AGP_MEM *drm_agp_allocate_memory(struct agp_bridge_data *bridge, size
5048 extern int drm_agp_free_memory(DRM_AGP_MEM * handle);
5049 extern int drm_agp_bind_memory(DRM_AGP_MEM * handle, off_t start);
5050 extern int drm_agp_unbind_memory(DRM_AGP_MEM * handle);
5051 +extern void drm_agp_chipset_flush(struct drm_device *dev);
5053 /* Stub support (drm_stub.h) */
5054 extern int drm_get_dev(struct pci_dev *pdev, const struct pci_device_id *ent,
5055 @@ -1156,6 +1247,66 @@ extern unsigned long drm_mm_tail_space(struct drm_mm *mm);
5056 extern int drm_mm_remove_space_from_tail(struct drm_mm *mm, unsigned long size);
5057 extern int drm_mm_add_space_to_tail(struct drm_mm *mm, unsigned long size);
5059 +/* Graphics Execution Manager library functions (drm_gem.c) */
5060 +int drm_gem_init(struct drm_device *dev);
5061 +void drm_gem_object_free(struct kref *kref);
5062 +struct drm_gem_object *drm_gem_object_alloc(struct drm_device *dev,
5064 +void drm_gem_object_handle_free(struct kref *kref);
5067 +drm_gem_object_reference(struct drm_gem_object *obj)
5069 + kref_get(&obj->refcount);
5073 +drm_gem_object_unreference(struct drm_gem_object *obj)
5078 + kref_put(&obj->refcount, drm_gem_object_free);
5081 +int drm_gem_handle_create(struct drm_file *file_priv,
5082 + struct drm_gem_object *obj,
5086 +drm_gem_object_handle_reference(struct drm_gem_object *obj)
5088 + drm_gem_object_reference(obj);
5089 + kref_get(&obj->handlecount);
5093 +drm_gem_object_handle_unreference(struct drm_gem_object *obj)
5099 + * Must bump handle count first as this may be the last
5100 + * ref, in which case the object would disappear before we
5101 + * checked for a name
5103 + kref_put(&obj->handlecount, drm_gem_object_handle_free);
5104 + drm_gem_object_unreference(obj);
5107 +struct drm_gem_object *drm_gem_object_lookup(struct drm_device *dev,
5108 + struct drm_file *filp,
5110 +int drm_gem_close_ioctl(struct drm_device *dev, void *data,
5111 + struct drm_file *file_priv);
5112 +int drm_gem_flink_ioctl(struct drm_device *dev, void *data,
5113 + struct drm_file *file_priv);
5114 +int drm_gem_open_ioctl(struct drm_device *dev, void *data,
5115 + struct drm_file *file_priv);
5116 +void drm_gem_open(struct drm_device *dev, struct drm_file *file_private);
5117 +void drm_gem_release(struct drm_device *dev, struct drm_file *file_private);
5119 extern void drm_core_ioremap(struct drm_map *map, struct drm_device *dev);
5120 extern void drm_core_ioremap_wc(struct drm_map *map, struct drm_device *dev);
5121 extern void drm_core_ioremapfree(struct drm_map *map, struct drm_device *dev);
5122 diff --git a/include/drm/i915_drm.h b/include/drm/i915_drm.h
5123 index 05c66cf..59d08fc 100644
5124 --- a/include/drm/i915_drm.h
5125 +++ b/include/drm/i915_drm.h
5126 @@ -143,6 +143,22 @@ typedef struct _drm_i915_sarea {
5127 #define DRM_I915_GET_VBLANK_PIPE 0x0e
5128 #define DRM_I915_VBLANK_SWAP 0x0f
5129 #define DRM_I915_HWS_ADDR 0x11
5130 +#define DRM_I915_GEM_INIT 0x13
5131 +#define DRM_I915_GEM_EXECBUFFER 0x14
5132 +#define DRM_I915_GEM_PIN 0x15
5133 +#define DRM_I915_GEM_UNPIN 0x16
5134 +#define DRM_I915_GEM_BUSY 0x17
5135 +#define DRM_I915_GEM_THROTTLE 0x18
5136 +#define DRM_I915_GEM_ENTERVT 0x19
5137 +#define DRM_I915_GEM_LEAVEVT 0x1a
5138 +#define DRM_I915_GEM_CREATE 0x1b
5139 +#define DRM_I915_GEM_PREAD 0x1c
5140 +#define DRM_I915_GEM_PWRITE 0x1d
5141 +#define DRM_I915_GEM_MMAP 0x1e
5142 +#define DRM_I915_GEM_SET_DOMAIN 0x1f
5143 +#define DRM_I915_GEM_SW_FINISH 0x20
5144 +#define DRM_I915_GEM_SET_TILING 0x21
5145 +#define DRM_I915_GEM_GET_TILING 0x22
5147 #define DRM_IOCTL_I915_INIT DRM_IOW( DRM_COMMAND_BASE + DRM_I915_INIT, drm_i915_init_t)
5148 #define DRM_IOCTL_I915_FLUSH DRM_IO ( DRM_COMMAND_BASE + DRM_I915_FLUSH)
5149 @@ -160,6 +176,20 @@ typedef struct _drm_i915_sarea {
5150 #define DRM_IOCTL_I915_SET_VBLANK_PIPE DRM_IOW( DRM_COMMAND_BASE + DRM_I915_SET_VBLANK_PIPE, drm_i915_vblank_pipe_t)
5151 #define DRM_IOCTL_I915_GET_VBLANK_PIPE DRM_IOR( DRM_COMMAND_BASE + DRM_I915_GET_VBLANK_PIPE, drm_i915_vblank_pipe_t)
5152 #define DRM_IOCTL_I915_VBLANK_SWAP DRM_IOWR(DRM_COMMAND_BASE + DRM_I915_VBLANK_SWAP, drm_i915_vblank_swap_t)
5153 +#define DRM_IOCTL_I915_GEM_PIN DRM_IOWR(DRM_COMMAND_BASE + DRM_I915_GEM_PIN, struct drm_i915_gem_pin)
5154 +#define DRM_IOCTL_I915_GEM_UNPIN DRM_IOW(DRM_COMMAND_BASE + DRM_I915_GEM_UNPIN, struct drm_i915_gem_unpin)
5155 +#define DRM_IOCTL_I915_GEM_BUSY DRM_IOWR(DRM_COMMAND_BASE + DRM_I915_GEM_BUSY, struct drm_i915_gem_busy)
5156 +#define DRM_IOCTL_I915_GEM_THROTTLE DRM_IO ( DRM_COMMAND_BASE + DRM_I915_GEM_THROTTLE)
5157 +#define DRM_IOCTL_I915_GEM_ENTERVT DRM_IO(DRM_COMMAND_BASE + DRM_I915_GEM_ENTERVT)
5158 +#define DRM_IOCTL_I915_GEM_LEAVEVT DRM_IO(DRM_COMMAND_BASE + DRM_I915_GEM_LEAVEVT)
5159 +#define DRM_IOCTL_I915_GEM_CREATE DRM_IOWR(DRM_COMMAND_BASE + DRM_I915_GEM_CREATE, struct drm_i915_gem_create)
5160 +#define DRM_IOCTL_I915_GEM_PREAD DRM_IOW (DRM_COMMAND_BASE + DRM_I915_GEM_PREAD, struct drm_i915_gem_pread)
5161 +#define DRM_IOCTL_I915_GEM_PWRITE DRM_IOW (DRM_COMMAND_BASE + DRM_I915_GEM_PWRITE, struct drm_i915_gem_pwrite)
5162 +#define DRM_IOCTL_I915_GEM_MMAP DRM_IOWR(DRM_COMMAND_BASE + DRM_I915_GEM_MMAP, struct drm_i915_gem_mmap)
5163 +#define DRM_IOCTL_I915_GEM_SET_DOMAIN DRM_IOW (DRM_COMMAND_BASE + DRM_I915_GEM_SET_DOMAIN, struct drm_i915_gem_set_domain)
5164 +#define DRM_IOCTL_I915_GEM_SW_FINISH DRM_IOW (DRM_COMMAND_BASE + DRM_I915_GEM_SW_FINISH, struct drm_i915_gem_sw_finish)
5165 +#define DRM_IOCTL_I915_GEM_SET_TILING DRM_IOWR (DRM_COMMAND_BASE + DRM_I915_GEM_SET_TILING, struct drm_i915_gem_set_tiling)
5166 +#define DRM_IOCTL_I915_GEM_GET_TILING DRM_IOWR (DRM_COMMAND_BASE + DRM_I915_GEM_GET_TILING, struct drm_i915_gem_get_tiling)
5168 /* Allow drivers to submit batchbuffers directly to hardware, relying
5169 * on the security mechanisms provided by hardware.
5170 @@ -200,6 +230,7 @@ typedef struct drm_i915_irq_wait {
5171 #define I915_PARAM_IRQ_ACTIVE 1
5172 #define I915_PARAM_ALLOW_BATCHBUFFER 2
5173 #define I915_PARAM_LAST_DISPATCH 3
5174 +#define I915_PARAM_HAS_GEM 5
5176 typedef struct drm_i915_getparam {
5178 @@ -267,4 +298,305 @@ typedef struct drm_i915_hws_addr {
5180 } drm_i915_hws_addr_t;
5182 +struct drm_i915_gem_init {
5184 + * Beginning offset in the GTT to be managed by the DRM memory
5187 + uint64_t gtt_start;
5189 + * Ending offset in the GTT to be managed by the DRM memory
5195 +struct drm_i915_gem_create {
5197 + * Requested size for the object.
5199 + * The (page-aligned) allocated size for the object will be returned.
5203 + * Returned handle for the object.
5205 + * Object handles are nonzero.
5211 +struct drm_i915_gem_pread {
5212 + /** Handle for the object being read. */
5215 + /** Offset into the object to read from */
5217 + /** Length of data to read */
5220 + * Pointer to write the data into.
5222 + * This is a fixed-size type for 32/64 compatibility.
5224 + uint64_t data_ptr;
5227 +struct drm_i915_gem_pwrite {
5228 + /** Handle for the object being written to. */
5231 + /** Offset into the object to write to */
5233 + /** Length of data to write */
5236 + * Pointer to read the data from.
5238 + * This is a fixed-size type for 32/64 compatibility.
5240 + uint64_t data_ptr;
5243 +struct drm_i915_gem_mmap {
5244 + /** Handle for the object being mapped. */
5247 + /** Offset in the object to map. */
5250 + * Length of data to map.
5252 + * The value will be page-aligned.
5256 + * Returned pointer the data was mapped at.
5258 + * This is a fixed-size type for 32/64 compatibility.
5260 + uint64_t addr_ptr;
5263 +struct drm_i915_gem_set_domain {
5264 + /** Handle for the object */
5267 + /** New read domains */
5268 + uint32_t read_domains;
5270 + /** New write domain */
5271 + uint32_t write_domain;
5274 +struct drm_i915_gem_sw_finish {
5275 + /** Handle for the object */
5279 +struct drm_i915_gem_relocation_entry {
5281 + * Handle of the buffer being pointed to by this relocation entry.
5283 + * It's appealing to make this be an index into the mm_validate_entry
5284 + * list to refer to the buffer, but this allows the driver to create
5285 + * a relocation list for state buffers and not re-write it per
5286 + * exec using the buffer.
5288 + uint32_t target_handle;
5291 + * Value to be added to the offset of the target buffer to make up
5292 + * the relocation entry.
5296 + /** Offset in the buffer the relocation entry will be written into */
5300 + * Offset value of the target buffer that the relocation entry was last
5303 + * If the buffer has the same offset as last time, we can skip syncing
5304 + * and writing the relocation. This value is written back out by
5305 + * the execbuffer ioctl when the relocation is written.
5307 + uint64_t presumed_offset;
5310 + * Target memory domains read by this operation.
5312 + uint32_t read_domains;
5315 + * Target memory domains written by this operation.
5317 + * Note that only one domain may be written by the whole
5318 + * execbuffer operation, so that where there are conflicts,
5319 + * the application will get -EINVAL back.
5321 + uint32_t write_domain;
5325 + * Intel memory domains
5327 + * Most of these just align with the various caches in
5328 + * the system and are used to flush and invalidate as
5329 + * objects end up cached in different domains.
5332 +#define I915_GEM_DOMAIN_CPU 0x00000001
5333 +/** Render cache, used by 2D and 3D drawing */
5334 +#define I915_GEM_DOMAIN_RENDER 0x00000002
5335 +/** Sampler cache, used by texture engine */
5336 +#define I915_GEM_DOMAIN_SAMPLER 0x00000004
5337 +/** Command queue, used to load batch buffers */
5338 +#define I915_GEM_DOMAIN_COMMAND 0x00000008
5339 +/** Instruction cache, used by shader programs */
5340 +#define I915_GEM_DOMAIN_INSTRUCTION 0x00000010
5341 +/** Vertex address cache */
5342 +#define I915_GEM_DOMAIN_VERTEX 0x00000020
5343 +/** GTT domain - aperture and scanout */
5344 +#define I915_GEM_DOMAIN_GTT 0x00000040
5347 +struct drm_i915_gem_exec_object {
5349 + * User's handle for a buffer to be bound into the GTT for this
5354 + /** Number of relocations to be performed on this buffer */
5355 + uint32_t relocation_count;
5357 + * Pointer to array of struct drm_i915_gem_relocation_entry containing
5358 + * the relocations to be performed in this buffer.
5360 + uint64_t relocs_ptr;
5362 + /** Required alignment in graphics aperture */
5363 + uint64_t alignment;
5366 + * Returned value of the updated offset of the object, for future
5367 + * presumed_offset writes.
5372 +struct drm_i915_gem_execbuffer {
5374 + * List of buffers to be validated with their relocations to be
5375 + * performend on them.
5377 + * This is a pointer to an array of struct drm_i915_gem_validate_entry.
5379 + * These buffers must be listed in an order such that all relocations
5380 + * a buffer is performing refer to buffers that have already appeared
5381 + * in the validate list.
5383 + uint64_t buffers_ptr;
5384 + uint32_t buffer_count;
5386 + /** Offset in the batchbuffer to start execution from. */
5387 + uint32_t batch_start_offset;
5388 + /** Bytes used in batchbuffer from batch_start_offset */
5389 + uint32_t batch_len;
5392 + uint32_t num_cliprects;
5393 + /** This is a struct drm_clip_rect *cliprects */
5394 + uint64_t cliprects_ptr;
5397 +struct drm_i915_gem_pin {
5398 + /** Handle of the buffer to be pinned. */
5402 + /** alignment required within the aperture */
5403 + uint64_t alignment;
5405 + /** Returned GTT offset of the buffer. */
5409 +struct drm_i915_gem_unpin {
5410 + /** Handle of the buffer to be unpinned. */
5415 +struct drm_i915_gem_busy {
5416 + /** Handle of the buffer to check for busy */
5419 + /** Return busy status (1 if busy, 0 if idle) */
5423 +#define I915_TILING_NONE 0
5424 +#define I915_TILING_X 1
5425 +#define I915_TILING_Y 2
5427 +#define I915_BIT_6_SWIZZLE_NONE 0
5428 +#define I915_BIT_6_SWIZZLE_9 1
5429 +#define I915_BIT_6_SWIZZLE_9_10 2
5430 +#define I915_BIT_6_SWIZZLE_9_11 3
5431 +#define I915_BIT_6_SWIZZLE_9_10_11 4
5432 +/* Not seen by userland */
5433 +#define I915_BIT_6_SWIZZLE_UNKNOWN 5
5435 +struct drm_i915_gem_set_tiling {
5436 + /** Handle of the buffer to have its tiling state updated */
5440 + * Tiling mode for the object (I915_TILING_NONE, I915_TILING_X,
5443 + * This value is to be set on request, and will be updated by the
5444 + * kernel on successful return with the actual chosen tiling layout.
5446 + * The tiling mode may be demoted to I915_TILING_NONE when the system
5447 + * has bit 6 swizzling that can't be managed correctly by GEM.
5449 + * Buffer contents become undefined when changing tiling_mode.
5451 + uint32_t tiling_mode;
5454 + * Stride in bytes for the object when in I915_TILING_X or
5460 + * Returned address bit 6 swizzling required for CPU access through
5463 + uint32_t swizzle_mode;
5466 +struct drm_i915_gem_get_tiling {
5467 + /** Handle of the buffer to get tiling state for. */
5471 + * Current tiling mode for the object (I915_TILING_NONE, I915_TILING_X,
5474 + uint32_t tiling_mode;
5477 + * Returned address bit 6 swizzling required for CPU access through
5480 + uint32_t swizzle_mode;
5483 #endif /* _I915_DRM_H_ */