libdrm: intel: add DRM_RDWR flag in drm_intel_bo_gem_export_to_prime
[platform/upstream/libdrm.git] / intel / intel_bufmgr_gem.c
index 1eae898..023af61 100644 (file)
  *         Dave Airlie <airlied@linux.ie>
  */
 
-#ifdef HAVE_CONFIG_H
-#include "config.h"
-#endif
-
 #include <xf86drm.h>
 #include <xf86atomic.h>
 #include <fcntl.h>
@@ -64,8 +60,9 @@
 #include "string.h"
 
 #include "i915_drm.h"
+#include "uthash.h"
 
-#ifdef HAVE_VALGRIND
+#if HAVE_VALGRIND
 #include <valgrind.h>
 #include <memcheck.h>
 #define VG(x) x
@@ -130,7 +127,9 @@ typedef struct _drm_intel_bufmgr_gem {
 
        drmMMListHead managers;
 
-       drmMMListHead named;
+       drm_intel_bo_gem *name_table;
+       drm_intel_bo_gem *handle_table;
+
        drmMMListHead vma_cache;
        int vma_count, vma_open, vma_max;
 
@@ -146,6 +145,7 @@ typedef struct _drm_intel_bufmgr_gem {
        unsigned int bo_reuse : 1;
        unsigned int no_exec : 1;
        unsigned int has_vebox : 1;
+       unsigned int has_exec_async : 1;
        bool fenced_relocs;
 
        struct {
@@ -175,7 +175,9 @@ struct _drm_intel_bo_gem {
          * List contains both flink named and prime fd'd objects
         */
        unsigned int global_name;
-       drmMMListHead name_list;
+
+       UT_hash_handle handle_hh;
+       UT_hash_handle name_hh;
 
        /**
         * Index of the buffer within the validation list while preparing a
@@ -190,6 +192,8 @@ struct _drm_intel_bo_gem {
        uint32_t swizzle_mode;
        unsigned long stride;
 
+       unsigned long kflags;
+
        time_t free_time;
 
        /** Array passed to the DRM containing relocation information. */
@@ -200,10 +204,19 @@ struct _drm_intel_bo_gem {
        drm_intel_reloc_target *reloc_target_info;
        /** Number of entries in relocs */
        int reloc_count;
+       /** Array of BOs that are referenced by this buffer and will be softpinned */
+       drm_intel_bo **softpin_target;
+       /** Number softpinned BOs that are referenced by this buffer */
+       int softpin_target_count;
+       /** Maximum amount of softpinned BOs that are referenced by this buffer */
+       int softpin_target_size;
+
        /** Mapped address for the buffer, saved across map/unmap cycles */
        void *mem_virtual;
        /** GTT virtual address for the buffer, saved across map/unmap cycles */
        void *gtt_virtual;
+       /** WC CPU address for the buffer, saved across map/unmap cycles */
+       void *wc_virtual;
        /**
         * Virtual address of the buffer allocated by user, used for userptr
         * objects only.
@@ -242,7 +255,7 @@ struct _drm_intel_bo_gem {
         * Boolean of whether the GPU is definitely not accessing the buffer.
         *
         * This is only valid when reusable, since non-reusable
-        * buffers are those that have been shared wth other
+        * buffers are those that have been shared with other
         * processes, so we don't know their state.
         */
        bool idle;
@@ -253,15 +266,6 @@ struct _drm_intel_bo_gem {
        bool is_userptr;
 
        /**
-        * Boolean of whether this buffer can be placed in the full 48-bit
-        * address range on gen8+.
-        *
-        * By default, buffers will be keep in a 32-bit range, unless this
-        * flag is explicitly set.
-        */
-       bool use_48b_address_range;
-
-       /**
         * Size in bytes of this buffer and its relocation descendents.
         *
         * Used to avoid costly tree walking in
@@ -275,7 +279,7 @@ struct _drm_intel_bo_gem {
         */
        int reloc_tree_fences;
 
-       /** Flags that we may need to do the SW_FINSIH ioctl on unmap. */
+       /** Flags that we may need to do the SW_FINISH ioctl on unmap. */
        bool mapped_cpu_write;
 };
 
@@ -414,8 +418,9 @@ drm_intel_gem_dump_validation_list(drm_intel_bufmgr_gem *bufmgr_gem)
                drm_intel_bo *bo = bufmgr_gem->exec_bos[i];
                drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
 
-               if (bo_gem->relocs == NULL) {
-                       DBG("%2d: %d (%s)\n", i, bo_gem->gem_handle,
+               if (bo_gem->relocs == NULL && bo_gem->softpin_target == NULL) {
+                       DBG("%2d: %d %s(%s)\n", i, bo_gem->gem_handle,
+                           bo_gem->kflags & EXEC_OBJECT_PINNED ? "*" : "",
                            bo_gem->name);
                        continue;
                }
@@ -425,10 +430,12 @@ drm_intel_gem_dump_validation_list(drm_intel_bufmgr_gem *bufmgr_gem)
                        drm_intel_bo_gem *target_gem =
                            (drm_intel_bo_gem *) target_bo;
 
-                       DBG("%2d: %d (%s)@0x%08x %08x -> "
+                       DBG("%2d: %d %s(%s)@0x%08x %08x -> "
                            "%d (%s)@0x%08x %08x + 0x%08x\n",
                            i,
-                           bo_gem->gem_handle, bo_gem->name,
+                           bo_gem->gem_handle,
+                           bo_gem->kflags & EXEC_OBJECT_PINNED ? "*" : "",
+                           bo_gem->name,
                            upper_32_bits(bo_gem->relocs[j].offset),
                            lower_32_bits(bo_gem->relocs[j].offset),
                            target_gem->gem_handle,
@@ -437,6 +444,22 @@ drm_intel_gem_dump_validation_list(drm_intel_bufmgr_gem *bufmgr_gem)
                            lower_32_bits(target_bo->offset64),
                            bo_gem->relocs[j].delta);
                }
+
+               for (j = 0; j < bo_gem->softpin_target_count; j++) {
+                       drm_intel_bo *target_bo = bo_gem->softpin_target[j];
+                       drm_intel_bo_gem *target_gem =
+                           (drm_intel_bo_gem *) target_bo;
+                       DBG("%2d: %d %s(%s) -> "
+                           "%d *(%s)@0x%08x %08x\n",
+                           i,
+                           bo_gem->gem_handle,
+                           bo_gem->kflags & EXEC_OBJECT_PINNED ? "*" : "",
+                           bo_gem->name,
+                           target_gem->gem_handle,
+                           target_gem->name,
+                           upper_32_bits(target_bo->offset64),
+                           lower_32_bits(target_bo->offset64));
+               }
        }
 }
 
@@ -500,12 +523,11 @@ drm_intel_add_validate_buffer2(drm_intel_bo *bo, int need_fence)
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bo->bufmgr;
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *)bo;
        int index;
-       int flags = 0;
+       unsigned long flags;
 
+       flags = 0;
        if (need_fence)
                flags |= EXEC_OBJECT_NEEDS_FENCE;
-       if (bo_gem->use_48b_address_range)
-               flags |= EXEC_OBJECT_SUPPORTS_48B_ADDRESS;
 
        if (bo_gem->validate_index != -1) {
                bufmgr_gem->exec2_objects[bo_gem->validate_index].flags |= flags;
@@ -535,11 +557,11 @@ drm_intel_add_validate_buffer2(drm_intel_bo *bo, int need_fence)
        bufmgr_gem->exec2_objects[index].relocation_count = bo_gem->reloc_count;
        bufmgr_gem->exec2_objects[index].relocs_ptr = (uintptr_t)bo_gem->relocs;
        bufmgr_gem->exec2_objects[index].alignment = bo->align;
-       bufmgr_gem->exec2_objects[index].offset = 0;
-       bufmgr_gem->exec_bos[index] = bo;
-       bufmgr_gem->exec2_objects[index].flags = flags;
+       bufmgr_gem->exec2_objects[index].offset = bo->offset64;
+       bufmgr_gem->exec2_objects[index].flags = bo_gem->kflags | flags;
        bufmgr_gem->exec2_objects[index].rsvd1 = 0;
        bufmgr_gem->exec2_objects[index].rsvd2 = 0;
+       bufmgr_gem->exec_bos[index] = bo;
        bufmgr_gem->exec_count++;
 }
 
@@ -633,7 +655,6 @@ drm_intel_gem_bo_busy(drm_intel_bo *bo)
        } else {
                return false;
        }
-       return (ret == 0 && busy.busy);
 }
 
 static int
@@ -763,14 +784,17 @@ retry:
                        }
                }
        }
-       pthread_mutex_unlock(&bufmgr_gem->lock);
 
        if (!alloc_from_cache) {
                struct drm_i915_gem_create create;
 
                bo_gem = calloc(1, sizeof(*bo_gem));
                if (!bo_gem)
-                       return NULL;
+                       goto err;
+
+               /* drm_intel_gem_bo_free calls DRMLISTDEL() for an uninitialized
+                  list (vma_list), so better set the list head here */
+               DRMINITLISTHEAD(&bo_gem->vma_list);
 
                bo_gem->bo.size = bo_size;
 
@@ -780,12 +804,17 @@ retry:
                ret = drmIoctl(bufmgr_gem->fd,
                               DRM_IOCTL_I915_GEM_CREATE,
                               &create);
-               bo_gem->gem_handle = create.handle;
-               bo_gem->bo.handle = bo_gem->gem_handle;
                if (ret != 0) {
                        free(bo_gem);
-                       return NULL;
+                       goto err;
                }
+
+               bo_gem->gem_handle = create.handle;
+               HASH_ADD(handle_hh, bufmgr_gem->handle_table,
+                        gem_handle, sizeof(bo_gem->gem_handle),
+                        bo_gem);
+
+               bo_gem->bo.handle = bo_gem->gem_handle;
                bo_gem->bo.bufmgr = bufmgr;
                bo_gem->bo.align = alignment;
 
@@ -793,16 +822,10 @@ retry:
                bo_gem->swizzle_mode = I915_BIT_6_SWIZZLE_NONE;
                bo_gem->stride = 0;
 
-               /* drm_intel_gem_bo_free calls DRMLISTDEL() for an uninitialized
-                  list (vma_list), so better set the list head here */
-               DRMINITLISTHEAD(&bo_gem->name_list);
-               DRMINITLISTHEAD(&bo_gem->vma_list);
                if (drm_intel_gem_bo_set_tiling_internal(&bo_gem->bo,
                                                         tiling_mode,
-                                                        stride)) {
-                   drm_intel_gem_bo_free(&bo_gem->bo);
-                   return NULL;
-               }
+                                                        stride))
+                       goto err_free;
        }
 
        bo_gem->name = name;
@@ -812,14 +835,20 @@ retry:
        bo_gem->used_as_reloc_target = false;
        bo_gem->has_error = false;
        bo_gem->reusable = true;
-       bo_gem->use_48b_address_range = false;
 
        drm_intel_bo_gem_set_in_aperture_size(bufmgr_gem, bo_gem, alignment);
+       pthread_mutex_unlock(&bufmgr_gem->lock);
 
        DBG("bo_create: buf %d (%s) %ldb\n",
            bo_gem->gem_handle, bo_gem->name, size);
 
        return &bo_gem->bo;
+
+err_free:
+       drm_intel_gem_bo_free(&bo_gem->bo);
+err:
+       pthread_mutex_unlock(&bufmgr_gem->lock);
+       return NULL;
 }
 
 static drm_intel_bo *
@@ -920,6 +949,9 @@ drm_intel_gem_bo_alloc_userptr(drm_intel_bufmgr *bufmgr,
        if (!bo_gem)
                return NULL;
 
+       atomic_set(&bo_gem->refcount, 1);
+       DRMINITLISTHEAD(&bo_gem->vma_list);
+
        bo_gem->bo.size = size;
 
        memclear(userptr);
@@ -938,6 +970,8 @@ drm_intel_gem_bo_alloc_userptr(drm_intel_bufmgr *bufmgr,
                return NULL;
        }
 
+       pthread_mutex_lock(&bufmgr_gem->lock);
+
        bo_gem->gem_handle = userptr.handle;
        bo_gem->bo.handle = bo_gem->gem_handle;
        bo_gem->bo.bufmgr    = bufmgr;
@@ -949,19 +983,19 @@ drm_intel_gem_bo_alloc_userptr(drm_intel_bufmgr *bufmgr,
        bo_gem->swizzle_mode = I915_BIT_6_SWIZZLE_NONE;
        bo_gem->stride       = 0;
 
-       DRMINITLISTHEAD(&bo_gem->name_list);
-       DRMINITLISTHEAD(&bo_gem->vma_list);
+       HASH_ADD(handle_hh, bufmgr_gem->handle_table,
+                gem_handle, sizeof(bo_gem->gem_handle),
+                bo_gem);
 
        bo_gem->name = name;
-       atomic_set(&bo_gem->refcount, 1);
        bo_gem->validate_index = -1;
        bo_gem->reloc_tree_fences = 0;
        bo_gem->used_as_reloc_target = false;
        bo_gem->has_error = false;
        bo_gem->reusable = false;
-       bo_gem->use_48b_address_range = false;
 
        drm_intel_bo_gem_set_in_aperture_size(bufmgr_gem, bo_gem, 0);
+       pthread_mutex_unlock(&bufmgr_gem->lock);
 
        DBG("bo_create_userptr: "
            "ptr %p buf %d (%s) size %ldb, stride 0x%x, tile mode %d\n",
@@ -1035,13 +1069,35 @@ check_bo_alloc_userptr(drm_intel_bufmgr *bufmgr,
                                          tiling_mode, stride, size, flags);
 }
 
+static int get_tiling_mode(drm_intel_bufmgr_gem *bufmgr_gem,
+                          uint32_t gem_handle,
+                          uint32_t *tiling_mode,
+                          uint32_t *swizzle_mode)
+{
+       struct drm_i915_gem_get_tiling get_tiling = {
+               .handle = gem_handle,
+       };
+       int ret;
+
+       ret = drmIoctl(bufmgr_gem->fd,
+                      DRM_IOCTL_I915_GEM_GET_TILING,
+                      &get_tiling);
+       if (ret != 0 && errno != EOPNOTSUPP)
+               return ret;
+
+       *tiling_mode = get_tiling.tiling_mode;
+       *swizzle_mode = get_tiling.swizzle_mode;
+
+       return 0;
+}
+
 /**
  * Returns a drm_intel_bo wrapping the given buffer object handle.
  *
  * This can be used when one application needs to pass a buffer object
  * to another.
  */
-drm_intel_bo *
+drm_public drm_intel_bo *
 drm_intel_bo_gem_create_from_name(drm_intel_bufmgr *bufmgr,
                                  const char *name,
                                  unsigned int handle)
@@ -1050,8 +1106,6 @@ drm_intel_bo_gem_create_from_name(drm_intel_bufmgr *bufmgr,
        drm_intel_bo_gem *bo_gem;
        int ret;
        struct drm_gem_open open_arg;
-       struct drm_i915_gem_get_tiling get_tiling;
-       drmMMListHead *list;
 
        /* At the moment most applications only have a few named bo.
         * For instance, in a DRI client only the render buffers passed
@@ -1060,15 +1114,11 @@ drm_intel_bo_gem_create_from_name(drm_intel_bufmgr *bufmgr,
         * provides a sufficiently fast match.
         */
        pthread_mutex_lock(&bufmgr_gem->lock);
-       for (list = bufmgr_gem->named.next;
-            list != &bufmgr_gem->named;
-            list = list->next) {
-               bo_gem = DRMLISTENTRY(drm_intel_bo_gem, list, name_list);
-               if (bo_gem->global_name == handle) {
-                       drm_intel_gem_bo_reference(&bo_gem->bo);
-                       pthread_mutex_unlock(&bufmgr_gem->lock);
-                       return &bo_gem->bo;
-               }
+       HASH_FIND(name_hh, bufmgr_gem->name_table,
+                 &handle, sizeof(handle), bo_gem);
+       if (bo_gem) {
+               drm_intel_gem_bo_reference(&bo_gem->bo);
+               goto out;
        }
 
        memclear(open_arg);
@@ -1079,29 +1129,26 @@ drm_intel_bo_gem_create_from_name(drm_intel_bufmgr *bufmgr,
        if (ret != 0) {
                DBG("Couldn't reference %s handle 0x%08x: %s\n",
                    name, handle, strerror(errno));
-               pthread_mutex_unlock(&bufmgr_gem->lock);
-               return NULL;
+               bo_gem = NULL;
+               goto out;
        }
         /* Now see if someone has used a prime handle to get this
          * object from the kernel before by looking through the list
          * again for a matching gem_handle
          */
-       for (list = bufmgr_gem->named.next;
-            list != &bufmgr_gem->named;
-            list = list->next) {
-               bo_gem = DRMLISTENTRY(drm_intel_bo_gem, list, name_list);
-               if (bo_gem->gem_handle == open_arg.handle) {
-                       drm_intel_gem_bo_reference(&bo_gem->bo);
-                       pthread_mutex_unlock(&bufmgr_gem->lock);
-                       return &bo_gem->bo;
-               }
+       HASH_FIND(handle_hh, bufmgr_gem->handle_table,
+                 &open_arg.handle, sizeof(open_arg.handle), bo_gem);
+       if (bo_gem) {
+               drm_intel_gem_bo_reference(&bo_gem->bo);
+               goto out;
        }
 
        bo_gem = calloc(1, sizeof(*bo_gem));
-       if (!bo_gem) {
-               pthread_mutex_unlock(&bufmgr_gem->lock);
-               return NULL;
-       }
+       if (!bo_gem)
+               goto out;
+
+       atomic_set(&bo_gem->refcount, 1);
+       DRMINITLISTHEAD(&bo_gem->vma_list);
 
        bo_gem->bo.size = open_arg.size;
        bo_gem->bo.offset = 0;
@@ -1109,35 +1156,34 @@ drm_intel_bo_gem_create_from_name(drm_intel_bufmgr *bufmgr,
        bo_gem->bo.virtual = NULL;
        bo_gem->bo.bufmgr = bufmgr;
        bo_gem->name = name;
-       atomic_set(&bo_gem->refcount, 1);
        bo_gem->validate_index = -1;
        bo_gem->gem_handle = open_arg.handle;
        bo_gem->bo.handle = open_arg.handle;
        bo_gem->global_name = handle;
        bo_gem->reusable = false;
-       bo_gem->use_48b_address_range = false;
 
-       memclear(get_tiling);
-       get_tiling.handle = bo_gem->gem_handle;
-       ret = drmIoctl(bufmgr_gem->fd,
-                      DRM_IOCTL_I915_GEM_GET_TILING,
-                      &get_tiling);
-       if (ret != 0) {
-               drm_intel_gem_bo_unreference(&bo_gem->bo);
-               pthread_mutex_unlock(&bufmgr_gem->lock);
-               return NULL;
-       }
-       bo_gem->tiling_mode = get_tiling.tiling_mode;
-       bo_gem->swizzle_mode = get_tiling.swizzle_mode;
+       HASH_ADD(handle_hh, bufmgr_gem->handle_table,
+                gem_handle, sizeof(bo_gem->gem_handle), bo_gem);
+       HASH_ADD(name_hh, bufmgr_gem->name_table,
+                global_name, sizeof(bo_gem->global_name), bo_gem);
+
+       ret = get_tiling_mode(bufmgr_gem, bo_gem->gem_handle,
+                             &bo_gem->tiling_mode, &bo_gem->swizzle_mode);
+       if (ret != 0)
+               goto err_unref;
+
        /* XXX stride is unknown */
        drm_intel_bo_gem_set_in_aperture_size(bufmgr_gem, bo_gem, 0);
-
-       DRMINITLISTHEAD(&bo_gem->vma_list);
-       DRMLISTADDTAIL(&bo_gem->name_list, &bufmgr_gem->named);
-       pthread_mutex_unlock(&bufmgr_gem->lock);
        DBG("bo_create_from_handle: %d (%s)\n", handle, bo_gem->name);
 
+out:
+       pthread_mutex_unlock(&bufmgr_gem->lock);
        return &bo_gem->bo;
+
+err_unref:
+       drm_intel_gem_bo_free(&bo_gem->bo);
+       pthread_mutex_unlock(&bufmgr_gem->lock);
+       return NULL;
 }
 
 static void
@@ -1154,11 +1200,20 @@ drm_intel_gem_bo_free(drm_intel_bo *bo)
                drm_munmap(bo_gem->mem_virtual, bo_gem->bo.size);
                bufmgr_gem->vma_count--;
        }
+       if (bo_gem->wc_virtual) {
+               VG(VALGRIND_FREELIKE_BLOCK(bo_gem->wc_virtual, 0));
+               drm_munmap(bo_gem->wc_virtual, bo_gem->bo.size);
+               bufmgr_gem->vma_count--;
+       }
        if (bo_gem->gtt_virtual) {
                drm_munmap(bo_gem->gtt_virtual, bo_gem->bo.size);
                bufmgr_gem->vma_count--;
        }
 
+       if (bo_gem->global_name)
+               HASH_DELETE(name_hh, bufmgr_gem->name_table, bo_gem);
+       HASH_DELETE(handle_hh, bufmgr_gem->handle_table, bo_gem);
+
        /* Close this object */
        memclear(close);
        close.handle = bo_gem->gem_handle;
@@ -1179,6 +1234,9 @@ drm_intel_gem_bo_mark_mmaps_incoherent(drm_intel_bo *bo)
        if (bo_gem->mem_virtual)
                VALGRIND_MAKE_MEM_NOACCESS(bo_gem->mem_virtual, bo->size);
 
+       if (bo_gem->wc_virtual)
+               VALGRIND_MAKE_MEM_NOACCESS(bo_gem->wc_virtual, bo->size);
+
        if (bo_gem->gtt_virtual)
                VALGRIND_MAKE_MEM_NOACCESS(bo_gem->gtt_virtual, bo->size);
 #endif
@@ -1243,6 +1301,11 @@ static void drm_intel_gem_bo_purge_vma_cache(drm_intel_bufmgr_gem *bufmgr_gem)
                        bo_gem->mem_virtual = NULL;
                        bufmgr_gem->vma_count--;
                }
+               if (bo_gem->wc_virtual) {
+                       drm_munmap(bo_gem->wc_virtual, bo_gem->bo.size);
+                       bo_gem->wc_virtual = NULL;
+                       bufmgr_gem->vma_count--;
+               }
                if (bo_gem->gtt_virtual) {
                        drm_munmap(bo_gem->gtt_virtual, bo_gem->bo.size);
                        bo_gem->gtt_virtual = NULL;
@@ -1258,6 +1321,8 @@ static void drm_intel_gem_bo_close_vma(drm_intel_bufmgr_gem *bufmgr_gem,
        DRMLISTADDTAIL(&bo_gem->vma_list, &bufmgr_gem->vma_cache);
        if (bo_gem->mem_virtual)
                bufmgr_gem->vma_count++;
+       if (bo_gem->wc_virtual)
+               bufmgr_gem->vma_count++;
        if (bo_gem->gtt_virtual)
                bufmgr_gem->vma_count++;
        drm_intel_gem_bo_purge_vma_cache(bufmgr_gem);
@@ -1270,6 +1335,8 @@ static void drm_intel_gem_bo_open_vma(drm_intel_bufmgr_gem *bufmgr_gem,
        DRMLISTDEL(&bo_gem->vma_list);
        if (bo_gem->mem_virtual)
                bufmgr_gem->vma_count--;
+       if (bo_gem->wc_virtual)
+               bufmgr_gem->vma_count--;
        if (bo_gem->gtt_virtual)
                bufmgr_gem->vma_count--;
        drm_intel_gem_bo_purge_vma_cache(bufmgr_gem);
@@ -1291,8 +1358,13 @@ drm_intel_gem_bo_unreference_final(drm_intel_bo *bo, time_t time)
                                                                  time);
                }
        }
+       for (i = 0; i < bo_gem->softpin_target_count; i++)
+               drm_intel_gem_bo_unreference_locked_timed(bo_gem->softpin_target[i],
+                                                                 time);
+       bo_gem->kflags = 0;
        bo_gem->reloc_count = 0;
        bo_gem->used_as_reloc_target = false;
+       bo_gem->softpin_target_count = 0;
 
        DBG("bo_unreference final: %d (%s)\n",
            bo_gem->gem_handle, bo_gem->name);
@@ -1306,6 +1378,11 @@ drm_intel_gem_bo_unreference_final(drm_intel_bo *bo, time_t time)
                free(bo_gem->relocs);
                bo_gem->relocs = NULL;
        }
+       if (bo_gem->softpin_target) {
+               free(bo_gem->softpin_target);
+               bo_gem->softpin_target = NULL;
+               bo_gem->softpin_target_size = 0;
+       }
 
        /* Clear any left-over mappings */
        if (bo_gem->map_count) {
@@ -1315,8 +1392,6 @@ drm_intel_gem_bo_unreference_final(drm_intel_bo *bo, time_t time)
                drm_intel_gem_bo_mark_mmaps_incoherent(bo);
        }
 
-       DRMLISTDEL(&bo_gem->name_list);
-
        bucket = drm_intel_gem_bo_bucket_for_size(bufmgr_gem, bo->size);
        /* Put the buffer into our internal cache for reuse if we can. */
        if (bufmgr_gem->bo_reuse && bo_gem->reusable && bucket != NULL &&
@@ -1503,7 +1578,7 @@ map_gtt(drm_intel_bo *bo)
        return 0;
 }
 
-int
+drm_public int
 drm_intel_gem_bo_map_gtt(drm_intel_bo *bo)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
@@ -1562,11 +1637,11 @@ drm_intel_gem_bo_map_gtt(drm_intel_bo *bo)
  * undefined).
  */
 
-int
+drm_public int
 drm_intel_gem_bo_map_unsynchronized(drm_intel_bo *bo)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
-#ifdef HAVE_VALGRIND
+#if HAVE_VALGRIND
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
 #endif
        int ret;
@@ -1638,7 +1713,7 @@ static int drm_intel_gem_bo_unmap(drm_intel_bo *bo)
        }
 
        /* We need to unmap after every innovation as we cannot track
-        * an open vma for every bo as that will exhaasut the system
+        * an open vma for every bo as that will exhaust the system
         * limits and cause later failures.
         */
        if (--bo_gem->map_count == 0) {
@@ -1651,7 +1726,7 @@ static int drm_intel_gem_bo_unmap(drm_intel_bo *bo)
        return ret;
 }
 
-int
+drm_public int
 drm_intel_gem_bo_unmap_gtt(drm_intel_bo *bo)
 {
        return drm_intel_gem_bo_unmap(bo);
@@ -1776,7 +1851,7 @@ drm_intel_gem_bo_wait_rendering(drm_intel_bo *bo)
  * Note that some kernels have broken the inifite wait for negative values
  * promise, upgrade to latest stable kernels if this is the case.
  */
-int
+drm_public int
 drm_intel_gem_bo_wait(drm_intel_bo *bo, int64_t timeout_ns)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
@@ -1812,7 +1887,7 @@ drm_intel_gem_bo_wait(drm_intel_bo *bo, int64_t timeout_ns)
  * In combination with drm_intel_gem_bo_pin() and manual fence management, we
  * can do tiled pixmaps this way.
  */
-void
+drm_public void
 drm_intel_gem_bo_start_gtt_access(drm_intel_bo *bo, int write_enable)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
@@ -1943,14 +2018,6 @@ do_bo_emit_reloc(drm_intel_bo *bo, uint32_t offset,
                bo_gem->reloc_tree_fences += target_bo_gem->reloc_tree_fences;
        }
 
-       bo_gem->relocs[bo_gem->reloc_count].offset = offset;
-       bo_gem->relocs[bo_gem->reloc_count].delta = target_offset;
-       bo_gem->relocs[bo_gem->reloc_count].target_handle =
-           target_bo_gem->gem_handle;
-       bo_gem->relocs[bo_gem->reloc_count].read_domains = read_domains;
-       bo_gem->relocs[bo_gem->reloc_count].write_domain = write_domain;
-       bo_gem->relocs[bo_gem->reloc_count].presumed_offset = target_bo->offset64;
-
        bo_gem->reloc_target_info[bo_gem->reloc_count].bo = target_bo;
        if (target_bo != bo)
                drm_intel_gem_bo_reference(target_bo);
@@ -1960,6 +2027,13 @@ do_bo_emit_reloc(drm_intel_bo *bo, uint32_t offset,
        else
                bo_gem->reloc_target_info[bo_gem->reloc_count].flags = 0;
 
+       bo_gem->relocs[bo_gem->reloc_count].offset = offset;
+       bo_gem->relocs[bo_gem->reloc_count].delta = target_offset;
+       bo_gem->relocs[bo_gem->reloc_count].target_handle =
+           target_bo_gem->gem_handle;
+       bo_gem->relocs[bo_gem->reloc_count].read_domains = read_domains;
+       bo_gem->relocs[bo_gem->reloc_count].write_domain = write_domain;
+       bo_gem->relocs[bo_gem->reloc_count].presumed_offset = target_bo->offset64;
        bo_gem->reloc_count++;
 
        return 0;
@@ -1969,7 +2043,49 @@ static void
 drm_intel_gem_bo_use_48b_address_range(drm_intel_bo *bo, uint32_t enable)
 {
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
-       bo_gem->use_48b_address_range = enable;
+
+       if (enable)
+               bo_gem->kflags |= EXEC_OBJECT_SUPPORTS_48B_ADDRESS;
+       else
+               bo_gem->kflags &= ~EXEC_OBJECT_SUPPORTS_48B_ADDRESS;
+}
+
+static int
+drm_intel_gem_bo_add_softpin_target(drm_intel_bo *bo, drm_intel_bo *target_bo)
+{
+       drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+       drm_intel_bo_gem *target_bo_gem = (drm_intel_bo_gem *) target_bo;
+       if (bo_gem->has_error)
+               return -ENOMEM;
+
+       if (target_bo_gem->has_error) {
+               bo_gem->has_error = true;
+               return -ENOMEM;
+       }
+
+       if (!(target_bo_gem->kflags & EXEC_OBJECT_PINNED))
+               return -EINVAL;
+       if (target_bo_gem == bo_gem)
+               return -EINVAL;
+
+       if (bo_gem->softpin_target_count == bo_gem->softpin_target_size) {
+               int new_size = bo_gem->softpin_target_size * 2;
+               if (new_size == 0)
+                       new_size = bufmgr_gem->max_relocs;
+
+               bo_gem->softpin_target = realloc(bo_gem->softpin_target, new_size *
+                               sizeof(drm_intel_bo *));
+               if (!bo_gem->softpin_target)
+                       return -ENOMEM;
+
+               bo_gem->softpin_target_size = new_size;
+       }
+       bo_gem->softpin_target[bo_gem->softpin_target_count] = target_bo;
+       drm_intel_gem_bo_reference(target_bo);
+       bo_gem->softpin_target_count++;
+
+       return 0;
 }
 
 static int
@@ -1978,10 +2094,14 @@ drm_intel_gem_bo_emit_reloc(drm_intel_bo *bo, uint32_t offset,
                            uint32_t read_domains, uint32_t write_domain)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bo->bufmgr;
+       drm_intel_bo_gem *target_bo_gem = (drm_intel_bo_gem *)target_bo;
 
-       return do_bo_emit_reloc(bo, offset, target_bo, target_offset,
-                               read_domains, write_domain,
-                               !bufmgr_gem->fenced_relocs);
+       if (target_bo_gem->kflags & EXEC_OBJECT_PINNED)
+               return drm_intel_gem_bo_add_softpin_target(bo, target_bo);
+       else
+               return do_bo_emit_reloc(bo, offset, target_bo, target_offset,
+                                       read_domains, write_domain,
+                                       !bufmgr_gem->fenced_relocs);
 }
 
 static int
@@ -1994,7 +2114,7 @@ drm_intel_gem_bo_emit_reloc_fence(drm_intel_bo *bo, uint32_t offset,
                                read_domains, write_domain, true);
 }
 
-int
+drm_public int
 drm_intel_gem_bo_get_reloc_count(drm_intel_bo *bo)
 {
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
@@ -2014,8 +2134,10 @@ drm_intel_gem_bo_get_reloc_count(drm_intel_bo *bo)
  *
  * Any further drm_intel_bufmgr_check_aperture_space() queries
  * involving this buffer in the tree are undefined after this call.
+ *
+ * This also removes all softpinned targets being referenced by the BO.
  */
-void
+drm_public void
 drm_intel_gem_bo_clear_relocs(drm_intel_bo *bo, int start)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
@@ -2040,6 +2162,12 @@ drm_intel_gem_bo_clear_relocs(drm_intel_bo *bo, int start)
        }
        bo_gem->reloc_count = start;
 
+       for (i = 0; i < bo_gem->softpin_target_count; i++) {
+               drm_intel_bo_gem *target_bo_gem = (drm_intel_bo_gem *) bo_gem->softpin_target[i];
+               drm_intel_gem_bo_unreference_locked_timed(&target_bo_gem->bo, time.tv_sec);
+       }
+       bo_gem->softpin_target_count = 0;
+
        pthread_mutex_unlock(&bufmgr_gem->lock);
 
 }
@@ -2080,7 +2208,7 @@ drm_intel_gem_bo_process_reloc2(drm_intel_bo *bo)
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *)bo;
        int i;
 
-       if (bo_gem->relocs == NULL)
+       if (bo_gem->relocs == NULL && bo_gem->softpin_target == NULL)
                return;
 
        for (i = 0; i < bo_gem->reloc_count; i++) {
@@ -2101,6 +2229,17 @@ drm_intel_gem_bo_process_reloc2(drm_intel_bo *bo)
                /* Add the target to the validate list */
                drm_intel_add_validate_buffer2(target_bo, need_fence);
        }
+
+       for (i = 0; i < bo_gem->softpin_target_count; i++) {
+               drm_intel_bo *target_bo = bo_gem->softpin_target[i];
+
+               if (target_bo == bo)
+                       continue;
+
+               drm_intel_gem_bo_mark_mmaps_incoherent(bo);
+               drm_intel_gem_bo_process_reloc2(target_bo);
+               drm_intel_add_validate_buffer2(target_bo, false);
+       }
 }
 
 
@@ -2138,6 +2277,10 @@ drm_intel_update_buffer_offsets2 (drm_intel_bufmgr_gem *bufmgr_gem)
 
                /* Update the buffer offset */
                if (bufmgr_gem->exec2_objects[i].offset != bo->offset64) {
+                       /* If we're seeing softpinned object here it means that the kernel
+                        * has relocated our object... Indicating a programming error
+                        */
+                       assert(!(bo_gem->kflags & EXEC_OBJECT_PINNED));
                        DBG("BO %d (%s) migrated: 0x%08x %08x -> 0x%08x %08x\n",
                            bo_gem->gem_handle, bo_gem->name,
                            upper_32_bits(bo->offset64),
@@ -2150,7 +2293,7 @@ drm_intel_update_buffer_offsets2 (drm_intel_bufmgr_gem *bufmgr_gem)
        }
 }
 
-void
+drm_public void
 drm_intel_gem_bo_aub_dump_bmp(drm_intel_bo *bo,
                              int x1, int y1, int width, int height,
                              enum aub_dump_bmp_format format,
@@ -2228,6 +2371,7 @@ drm_intel_gem_bo_exec(drm_intel_bo *bo, int used,
 static int
 do_exec2(drm_intel_bo *bo, int used, drm_intel_context *ctx,
         drm_clip_rect_t *cliprects, int num_cliprects, int DR4,
+        int in_fence, int *out_fence,
         unsigned int flags)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bo->bufmgr;
@@ -2282,12 +2426,20 @@ do_exec2(drm_intel_bo *bo, int used, drm_intel_context *ctx,
        else
                i915_execbuffer2_set_context_id(execbuf, ctx->ctx_id);
        execbuf.rsvd2 = 0;
+       if (in_fence != -1) {
+               execbuf.rsvd2 = in_fence;
+               execbuf.flags |= I915_EXEC_FENCE_IN;
+       }
+       if (out_fence != NULL) {
+               *out_fence = -1;
+               execbuf.flags |= I915_EXEC_FENCE_OUT;
+       }
 
        if (bufmgr_gem->no_exec)
                goto skip_execution;
 
        ret = drmIoctl(bufmgr_gem->fd,
-                      DRM_IOCTL_I915_GEM_EXECBUFFER2,
+                      DRM_IOCTL_I915_GEM_EXECBUFFER2_WR,
                       &execbuf);
        if (ret != 0) {
                ret = -errno;
@@ -2303,6 +2455,9 @@ do_exec2(drm_intel_bo *bo, int used, drm_intel_context *ctx,
        }
        drm_intel_update_buffer_offsets2(bufmgr_gem);
 
+       if (ret == 0 && out_fence != NULL)
+               *out_fence = execbuf.rsvd2 >> 32;
+
 skip_execution:
        if (bufmgr_gem->bufmgr.debug)
                drm_intel_gem_dump_validation_list(bufmgr_gem);
@@ -2328,7 +2483,7 @@ drm_intel_gem_bo_exec2(drm_intel_bo *bo, int used,
                       int DR4)
 {
        return do_exec2(bo, used, NULL, cliprects, num_cliprects, DR4,
-                       I915_EXEC_RENDER);
+                       -1, NULL, I915_EXEC_RENDER);
 }
 
 static int
@@ -2337,14 +2492,25 @@ drm_intel_gem_bo_mrb_exec2(drm_intel_bo *bo, int used,
                        unsigned int flags)
 {
        return do_exec2(bo, used, NULL, cliprects, num_cliprects, DR4,
-                       flags);
+                       -1, NULL, flags);
 }
 
-int
+drm_public int
 drm_intel_gem_bo_context_exec(drm_intel_bo *bo, drm_intel_context *ctx,
                              int used, unsigned int flags)
 {
-       return do_exec2(bo, used, ctx, NULL, 0, 0, flags);
+       return do_exec2(bo, used, ctx, NULL, 0, 0, -1, NULL, flags);
+}
+
+drm_public int
+drm_intel_gem_bo_fence_exec(drm_intel_bo *bo,
+                           drm_intel_context *ctx,
+                           int used,
+                           int in_fence,
+                           int *out_fence,
+                           unsigned int flags)
+{
+       return do_exec2(bo, used, ctx, NULL, 0, 0, in_fence, out_fence, flags);
 }
 
 static int
@@ -2465,15 +2631,25 @@ drm_intel_gem_bo_get_tiling(drm_intel_bo *bo, uint32_t * tiling_mode,
        return 0;
 }
 
-drm_intel_bo *
+static int
+drm_intel_gem_bo_set_softpin_offset(drm_intel_bo *bo, uint64_t offset)
+{
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+
+       bo->offset64 = offset;
+       bo->offset = offset;
+       bo_gem->kflags |= EXEC_OBJECT_PINNED;
+
+       return 0;
+}
+
+drm_public drm_intel_bo *
 drm_intel_bo_gem_create_from_prime(drm_intel_bufmgr *bufmgr, int prime_fd, int size)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bufmgr;
        int ret;
        uint32_t handle;
        drm_intel_bo_gem *bo_gem;
-       struct drm_i915_gem_get_tiling get_tiling;
-       drmMMListHead *list;
 
        pthread_mutex_lock(&bufmgr_gem->lock);
        ret = drmPrimeFDToHandle(bufmgr_gem->fd, prime_fd, &handle);
@@ -2488,22 +2664,20 @@ drm_intel_bo_gem_create_from_prime(drm_intel_bufmgr *bufmgr, int prime_fd, int s
         * for named buffers, we must not create two bo's pointing at the same
         * kernel object
         */
-       for (list = bufmgr_gem->named.next;
-            list != &bufmgr_gem->named;
-            list = list->next) {
-               bo_gem = DRMLISTENTRY(drm_intel_bo_gem, list, name_list);
-               if (bo_gem->gem_handle == handle) {
-                       drm_intel_gem_bo_reference(&bo_gem->bo);
-                       pthread_mutex_unlock(&bufmgr_gem->lock);
-                       return &bo_gem->bo;
-               }
+       HASH_FIND(handle_hh, bufmgr_gem->handle_table,
+                 &handle, sizeof(handle), bo_gem);
+       if (bo_gem) {
+               drm_intel_gem_bo_reference(&bo_gem->bo);
+               goto out;
        }
 
        bo_gem = calloc(1, sizeof(*bo_gem));
-       if (!bo_gem) {
-               pthread_mutex_unlock(&bufmgr_gem->lock);
-               return NULL;
-       }
+       if (!bo_gem)
+               goto out;
+
+       atomic_set(&bo_gem->refcount, 1);
+       DRMINITLISTHEAD(&bo_gem->vma_list);
+
        /* Determine size of bo.  The fd-to-handle ioctl really should
         * return the size, but it doesn't.  If we have kernel 3.12 or
         * later, we can lseek on the prime fd to get the size.  Older
@@ -2519,8 +2693,8 @@ drm_intel_bo_gem_create_from_prime(drm_intel_bufmgr *bufmgr, int prime_fd, int s
        bo_gem->bo.bufmgr = bufmgr;
 
        bo_gem->gem_handle = handle;
-
-       atomic_set(&bo_gem->refcount, 1);
+       HASH_ADD(handle_hh, bufmgr_gem->handle_table,
+                gem_handle, sizeof(bo_gem->gem_handle), bo_gem);
 
        bo_gem->name = "prime";
        bo_gem->validate_index = -1;
@@ -2528,43 +2702,33 @@ drm_intel_bo_gem_create_from_prime(drm_intel_bufmgr *bufmgr, int prime_fd, int s
        bo_gem->used_as_reloc_target = false;
        bo_gem->has_error = false;
        bo_gem->reusable = false;
-       bo_gem->use_48b_address_range = false;
 
-       DRMINITLISTHEAD(&bo_gem->vma_list);
-       DRMLISTADDTAIL(&bo_gem->name_list, &bufmgr_gem->named);
-       pthread_mutex_unlock(&bufmgr_gem->lock);
+       ret = get_tiling_mode(bufmgr_gem, handle,
+                             &bo_gem->tiling_mode, &bo_gem->swizzle_mode);
+       if (ret)
+               goto err;
 
-       memclear(get_tiling);
-       get_tiling.handle = bo_gem->gem_handle;
-       ret = drmIoctl(bufmgr_gem->fd,
-                      DRM_IOCTL_I915_GEM_GET_TILING,
-                      &get_tiling);
-       if (ret != 0) {
-               DBG("create_from_prime: failed to get tiling: %s\n", strerror(errno));
-               drm_intel_gem_bo_unreference(&bo_gem->bo);
-               return NULL;
-       }
-       bo_gem->tiling_mode = get_tiling.tiling_mode;
-       bo_gem->swizzle_mode = get_tiling.swizzle_mode;
        /* XXX stride is unknown */
        drm_intel_bo_gem_set_in_aperture_size(bufmgr_gem, bo_gem, 0);
 
+out:
+       pthread_mutex_unlock(&bufmgr_gem->lock);
        return &bo_gem->bo;
+
+err:
+       drm_intel_gem_bo_free(&bo_gem->bo);
+       pthread_mutex_unlock(&bufmgr_gem->lock);
+       return NULL;
 }
 
-int
+drm_public int
 drm_intel_bo_gem_export_to_prime(drm_intel_bo *bo, int *prime_fd)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
 
-       pthread_mutex_lock(&bufmgr_gem->lock);
-        if (DRMLISTEMPTY(&bo_gem->name_list))
-                DRMLISTADDTAIL(&bo_gem->name_list, &bufmgr_gem->named);
-       pthread_mutex_unlock(&bufmgr_gem->lock);
-
        if (drmPrimeHandleToFD(bufmgr_gem->fd, bo_gem->gem_handle,
-                              DRM_CLOEXEC, prime_fd) != 0)
+                              DRM_CLOEXEC | DRM_RDWR, prime_fd) != 0)
                return -errno;
 
        bo_gem->reusable = false;
@@ -2577,27 +2741,24 @@ drm_intel_gem_bo_flink(drm_intel_bo *bo, uint32_t * name)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
        drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
-       int ret;
 
        if (!bo_gem->global_name) {
                struct drm_gem_flink flink;
 
                memclear(flink);
                flink.handle = bo_gem->gem_handle;
+               if (drmIoctl(bufmgr_gem->fd, DRM_IOCTL_GEM_FLINK, &flink))
+                       return -errno;
 
                pthread_mutex_lock(&bufmgr_gem->lock);
+               if (!bo_gem->global_name) {
+                       bo_gem->global_name = flink.name;
+                       bo_gem->reusable = false;
 
-               ret = drmIoctl(bufmgr_gem->fd, DRM_IOCTL_GEM_FLINK, &flink);
-               if (ret != 0) {
-                       pthread_mutex_unlock(&bufmgr_gem->lock);
-                       return -errno;
+                       HASH_ADD(name_hh, bufmgr_gem->name_table,
+                                global_name, sizeof(bo_gem->global_name),
+                                bo_gem);
                }
-
-               bo_gem->global_name = flink.name;
-               bo_gem->reusable = false;
-
-                if (DRMLISTEMPTY(&bo_gem->name_list))
-                        DRMLISTADDTAIL(&bo_gem->name_list, &bufmgr_gem->named);
                pthread_mutex_unlock(&bufmgr_gem->lock);
        }
 
@@ -2612,7 +2773,7 @@ drm_intel_gem_bo_flink(drm_intel_bo *bo, uint32_t * name)
  * size is only bounded by how many buffers of that size we've managed to have
  * in flight at once.
  */
-void
+drm_public void
 drm_intel_bufmgr_gem_enable_reuse(drm_intel_bufmgr *bufmgr)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bufmgr;
@@ -2621,13 +2782,66 @@ drm_intel_bufmgr_gem_enable_reuse(drm_intel_bufmgr *bufmgr)
 }
 
 /**
+ * Disables implicit synchronisation before executing the bo
+ *
+ * This will cause rendering corruption unless you correctly manage explicit
+ * fences for all rendering involving this buffer - including use by others.
+ * Disabling the implicit serialisation is only required if that serialisation
+ * is too coarse (for example, you have split the buffer into many
+ * non-overlapping regions and are sharing the whole buffer between concurrent
+ * independent command streams).
+ *
+ * Note the kernel must advertise support via I915_PARAM_HAS_EXEC_ASYNC,
+ * which can be checked using drm_intel_bufmgr_can_disable_implicit_sync,
+ * or subsequent execbufs involving the bo will generate EINVAL.
+ */
+drm_public void
+drm_intel_gem_bo_disable_implicit_sync(drm_intel_bo *bo)
+{
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+
+       bo_gem->kflags |= EXEC_OBJECT_ASYNC;
+}
+
+/**
+ * Enables implicit synchronisation before executing the bo
+ *
+ * This is the default behaviour of the kernel, to wait upon prior writes
+ * completing on the object before rendering with it, or to wait for prior
+ * reads to complete before writing into the object.
+ * drm_intel_gem_bo_disable_implicit_sync() can stop this behaviour, telling
+ * the kernel never to insert a stall before using the object. Then this
+ * function can be used to restore the implicit sync before subsequent
+ * rendering.
+ */
+drm_public void
+drm_intel_gem_bo_enable_implicit_sync(drm_intel_bo *bo)
+{
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+
+       bo_gem->kflags &= ~EXEC_OBJECT_ASYNC;
+}
+
+/**
+ * Query whether the kernel supports disabling of its implicit synchronisation
+ * before execbuf. See drm_intel_gem_bo_disable_implicit_sync()
+ */
+drm_public int
+drm_intel_bufmgr_gem_can_disable_implicit_sync(drm_intel_bufmgr *bufmgr)
+{
+       drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bufmgr;
+
+       return bufmgr_gem->has_exec_async;
+}
+
+/**
  * Enable use of fenced reloc type.
  *
  * New code should enable this to avoid unnecessary fence register
  * allocation.  If this option is not enabled, all relocs will have fence
  * register allocated.
  */
-void
+drm_public void
 drm_intel_bufmgr_gem_enable_fenced_relocs(drm_intel_bufmgr *bufmgr)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bufmgr;
@@ -2844,6 +3058,13 @@ _drm_intel_gem_bo_references(drm_intel_bo *bo, drm_intel_bo *target_bo)
                        return 1;
        }
 
+       for (i = 0; i< bo_gem->softpin_target_count; i++) {
+               if (bo_gem->softpin_target[i] == target_bo)
+                       return 1;
+               if (_drm_intel_gem_bo_references(bo_gem->softpin_target[i], target_bo))
+                       return 1;
+       }
+
        return 0;
 }
 
@@ -2899,7 +3120,7 @@ init_cache_buckets(drm_intel_bufmgr_gem *bufmgr_gem)
        }
 }
 
-void
+drm_public void
 drm_intel_bufmgr_gem_set_vma_cache_size(drm_intel_bufmgr *bufmgr, int limit)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bufmgr;
@@ -2909,6 +3130,34 @@ drm_intel_bufmgr_gem_set_vma_cache_size(drm_intel_bufmgr *bufmgr, int limit)
        drm_intel_gem_bo_purge_vma_cache(bufmgr_gem);
 }
 
+static int
+parse_devid_override(const char *devid_override)
+{
+       static const struct {
+               const char *name;
+               int pci_id;
+       } name_map[] = {
+               { "brw", PCI_CHIP_I965_GM },
+               { "g4x", PCI_CHIP_GM45_GM },
+               { "ilk", PCI_CHIP_ILD_G },
+               { "snb", PCI_CHIP_SANDYBRIDGE_M_GT2_PLUS },
+               { "ivb", PCI_CHIP_IVYBRIDGE_S_GT2 },
+               { "hsw", PCI_CHIP_HASWELL_CRW_E_GT3 },
+               { "byt", PCI_CHIP_VALLEYVIEW_3 },
+               { "bdw", 0x1620 | BDW_ULX },
+               { "skl", PCI_CHIP_SKYLAKE_DT_GT2 },
+               { "kbl", PCI_CHIP_KABYLAKE_DT_GT2 },
+       };
+       unsigned int i;
+
+       for (i = 0; i < ARRAY_SIZE(name_map); i++) {
+               if (!strcmp(name_map[i].name, devid_override))
+                       return name_map[i].pci_id;
+       }
+
+       return strtod(devid_override, NULL);
+}
+
 /**
  * Get the PCI ID for the device.  This can be overridden by setting the
  * INTEL_DEVID_OVERRIDE environment variable to the desired ID.
@@ -2925,7 +3174,7 @@ get_pci_device_id(drm_intel_bufmgr_gem *bufmgr_gem)
                devid_override = getenv("INTEL_DEVID_OVERRIDE");
                if (devid_override) {
                        bufmgr_gem->no_exec = true;
-                       return strtod(devid_override, NULL);
+                       return parse_devid_override(devid_override);
                }
        }
 
@@ -2940,7 +3189,7 @@ get_pci_device_id(drm_intel_bufmgr_gem *bufmgr_gem)
        return devid;
 }
 
-int
+drm_public int
 drm_intel_bufmgr_gem_get_devid(drm_intel_bufmgr *bufmgr)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bufmgr;
@@ -2954,7 +3203,7 @@ drm_intel_bufmgr_gem_get_devid(drm_intel_bufmgr *bufmgr)
  * This function has to be called before drm_intel_bufmgr_gem_set_aub_dump()
  * for it to have any effect.
  */
-void
+drm_public void
 drm_intel_bufmgr_gem_set_aub_filename(drm_intel_bufmgr *bufmgr,
                                      const char *filename)
 {
@@ -2968,7 +3217,7 @@ drm_intel_bufmgr_gem_set_aub_filename(drm_intel_bufmgr *bufmgr,
  * You can set up a GTT and upload your objects into the referenced
  * space, then send off batchbuffers and get BMPs out the other end.
  */
-void
+drm_public void
 drm_intel_bufmgr_gem_set_aub_dump(drm_intel_bufmgr *bufmgr, int enable)
 {
        fprintf(stderr, "libdrm aub dumping is deprecated.\n\n"
@@ -2978,7 +3227,7 @@ drm_intel_bufmgr_gem_set_aub_dump(drm_intel_bufmgr *bufmgr, int enable)
                "See the intel_aubdump man page for more details.\n");
 }
 
-drm_intel_context *
+drm_public drm_intel_context *
 drm_intel_gem_context_create(drm_intel_bufmgr *bufmgr)
 {
        drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *)bufmgr;
@@ -3005,7 +3254,18 @@ drm_intel_gem_context_create(drm_intel_bufmgr *bufmgr)
        return context;
 }
 
-void
+drm_public int
+drm_intel_gem_context_get_id(drm_intel_context *ctx, uint32_t *ctx_id)
+{
+       if (ctx == NULL)
+               return -EINVAL;
+
+       *ctx_id = ctx->ctx_id;
+
+       return 0;
+}
+
+drm_public void
 drm_intel_gem_context_destroy(drm_intel_context *ctx)
 {
        drm_intel_bufmgr_gem *bufmgr_gem;
@@ -3028,7 +3288,7 @@ drm_intel_gem_context_destroy(drm_intel_context *ctx)
        free(ctx);
 }
 
-int
+drm_public int
 drm_intel_get_reset_stats(drm_intel_context *ctx,
                          uint32_t *reset_count,
                          uint32_t *active,
@@ -3062,7 +3322,7 @@ drm_intel_get_reset_stats(drm_intel_context *ctx,
        return ret;
 }
 
-int
+drm_public int
 drm_intel_reg_read(drm_intel_bufmgr *bufmgr,
                   uint32_t offset,
                   uint64_t *result)
@@ -3080,7 +3340,7 @@ drm_intel_reg_read(drm_intel_bufmgr *bufmgr,
        return ret;
 }
 
-int
+drm_public int
 drm_intel_get_subslice_total(int fd, unsigned int *subslice_total)
 {
        drm_i915_getparam_t gp;
@@ -3096,7 +3356,7 @@ drm_intel_get_subslice_total(int fd, unsigned int *subslice_total)
        return 0;
 }
 
-int
+drm_public int
 drm_intel_get_eu_total(int fd, unsigned int *eu_total)
 {
        drm_i915_getparam_t gp;
@@ -3112,6 +3372,36 @@ drm_intel_get_eu_total(int fd, unsigned int *eu_total)
        return 0;
 }
 
+drm_public int
+drm_intel_get_pooled_eu(int fd)
+{
+       drm_i915_getparam_t gp;
+       int ret = -1;
+
+       memclear(gp);
+       gp.param = I915_PARAM_HAS_POOLED_EU;
+       gp.value = &ret;
+       if (drmIoctl(fd, DRM_IOCTL_I915_GETPARAM, &gp))
+               return -errno;
+
+       return ret;
+}
+
+drm_public int
+drm_intel_get_min_eu_in_pool(int fd)
+{
+       drm_i915_getparam_t gp;
+       int ret = -1;
+
+       memclear(gp);
+       gp.param = I915_PARAM_MIN_EU_IN_POOL;
+       gp.value = &ret;
+       if (drmIoctl(fd, DRM_IOCTL_I915_GETPARAM, &gp))
+               return -errno;
+
+       return ret;
+}
+
 /**
  * Annotate the given bo for use in aub dumping.
  *
@@ -3133,8 +3423,7 @@ drm_intel_get_eu_total(int fd, unsigned int *eu_total)
  * default state (no annotations), call this function with a \c count
  * of zero.
  */
-void
-drm_intel_bufmgr_gem_set_aub_annotations(drm_intel_bo *bo,
+drm_public void drm_intel_bufmgr_gem_set_aub_annotations(drm_intel_bo *bo,
                                         drm_intel_aub_annotation *annotations,
                                         unsigned count)
 {
@@ -3175,13 +3464,148 @@ drm_intel_bufmgr_gem_unref(drm_intel_bufmgr *bufmgr)
        }
 }
 
+drm_public void *drm_intel_gem_bo_map__gtt(drm_intel_bo *bo)
+{
+       drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+
+       if (bo_gem->gtt_virtual)
+               return bo_gem->gtt_virtual;
+
+       if (bo_gem->is_userptr)
+               return NULL;
+
+       pthread_mutex_lock(&bufmgr_gem->lock);
+       if (bo_gem->gtt_virtual == NULL) {
+               struct drm_i915_gem_mmap_gtt mmap_arg;
+               void *ptr;
+
+               DBG("bo_map_gtt: mmap %d (%s), map_count=%d\n",
+                   bo_gem->gem_handle, bo_gem->name, bo_gem->map_count);
+
+               if (bo_gem->map_count++ == 0)
+                       drm_intel_gem_bo_open_vma(bufmgr_gem, bo_gem);
+
+               memclear(mmap_arg);
+               mmap_arg.handle = bo_gem->gem_handle;
+
+               /* Get the fake offset back... */
+               ptr = MAP_FAILED;
+               if (drmIoctl(bufmgr_gem->fd,
+                            DRM_IOCTL_I915_GEM_MMAP_GTT,
+                            &mmap_arg) == 0) {
+                       /* and mmap it */
+                       ptr = drm_mmap(0, bo->size, PROT_READ | PROT_WRITE,
+                                      MAP_SHARED, bufmgr_gem->fd,
+                                      mmap_arg.offset);
+               }
+               if (ptr == MAP_FAILED) {
+                       if (--bo_gem->map_count == 0)
+                               drm_intel_gem_bo_close_vma(bufmgr_gem, bo_gem);
+                       ptr = NULL;
+               }
+
+               bo_gem->gtt_virtual = ptr;
+       }
+       pthread_mutex_unlock(&bufmgr_gem->lock);
+
+       return bo_gem->gtt_virtual;
+}
+
+drm_public void *drm_intel_gem_bo_map__cpu(drm_intel_bo *bo)
+{
+       drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+
+       if (bo_gem->mem_virtual)
+               return bo_gem->mem_virtual;
+
+       if (bo_gem->is_userptr) {
+               /* Return the same user ptr */
+               return bo_gem->user_virtual;
+       }
+
+       pthread_mutex_lock(&bufmgr_gem->lock);
+       if (!bo_gem->mem_virtual) {
+               struct drm_i915_gem_mmap mmap_arg;
+
+               if (bo_gem->map_count++ == 0)
+                       drm_intel_gem_bo_open_vma(bufmgr_gem, bo_gem);
+
+               DBG("bo_map: %d (%s), map_count=%d\n",
+                   bo_gem->gem_handle, bo_gem->name, bo_gem->map_count);
+
+               memclear(mmap_arg);
+               mmap_arg.handle = bo_gem->gem_handle;
+               mmap_arg.size = bo->size;
+               if (drmIoctl(bufmgr_gem->fd,
+                            DRM_IOCTL_I915_GEM_MMAP,
+                            &mmap_arg)) {
+                       DBG("%s:%d: Error mapping buffer %d (%s): %s .\n",
+                           __FILE__, __LINE__, bo_gem->gem_handle,
+                           bo_gem->name, strerror(errno));
+                       if (--bo_gem->map_count == 0)
+                               drm_intel_gem_bo_close_vma(bufmgr_gem, bo_gem);
+               } else {
+                       VG(VALGRIND_MALLOCLIKE_BLOCK(mmap_arg.addr_ptr, mmap_arg.size, 0, 1));
+                       bo_gem->mem_virtual = (void *)(uintptr_t) mmap_arg.addr_ptr;
+               }
+       }
+       pthread_mutex_unlock(&bufmgr_gem->lock);
+
+       return bo_gem->mem_virtual;
+}
+
+drm_public void *drm_intel_gem_bo_map__wc(drm_intel_bo *bo)
+{
+       drm_intel_bufmgr_gem *bufmgr_gem = (drm_intel_bufmgr_gem *) bo->bufmgr;
+       drm_intel_bo_gem *bo_gem = (drm_intel_bo_gem *) bo;
+
+       if (bo_gem->wc_virtual)
+               return bo_gem->wc_virtual;
+
+       if (bo_gem->is_userptr)
+               return NULL;
+
+       pthread_mutex_lock(&bufmgr_gem->lock);
+       if (!bo_gem->wc_virtual) {
+               struct drm_i915_gem_mmap mmap_arg;
+
+               if (bo_gem->map_count++ == 0)
+                       drm_intel_gem_bo_open_vma(bufmgr_gem, bo_gem);
+
+               DBG("bo_map: %d (%s), map_count=%d\n",
+                   bo_gem->gem_handle, bo_gem->name, bo_gem->map_count);
+
+               memclear(mmap_arg);
+               mmap_arg.handle = bo_gem->gem_handle;
+               mmap_arg.size = bo->size;
+               mmap_arg.flags = I915_MMAP_WC;
+               if (drmIoctl(bufmgr_gem->fd,
+                            DRM_IOCTL_I915_GEM_MMAP,
+                            &mmap_arg)) {
+                       DBG("%s:%d: Error mapping buffer %d (%s): %s .\n",
+                           __FILE__, __LINE__, bo_gem->gem_handle,
+                           bo_gem->name, strerror(errno));
+                       if (--bo_gem->map_count == 0)
+                               drm_intel_gem_bo_close_vma(bufmgr_gem, bo_gem);
+               } else {
+                       VG(VALGRIND_MALLOCLIKE_BLOCK(mmap_arg.addr_ptr, mmap_arg.size, 0, 1));
+                       bo_gem->wc_virtual = (void *)(uintptr_t) mmap_arg.addr_ptr;
+               }
+       }
+       pthread_mutex_unlock(&bufmgr_gem->lock);
+
+       return bo_gem->wc_virtual;
+}
+
 /**
  * Initializes the GEM buffer manager, which uses the kernel to allocate, map,
  * and manage map buffer objections.
  *
  * \param fd File descriptor of the opened DRM device.
  */
-drm_intel_bufmgr *
+drm_public drm_intel_bufmgr *
 drm_intel_bufmgr_gem_init(int fd, int batch_size)
 {
        drm_intel_bufmgr_gem *bufmgr_gem;
@@ -3242,9 +3666,7 @@ drm_intel_bufmgr_gem_init(int fd, int batch_size)
                bufmgr_gem->gen = 7;
        else if (IS_GEN8(bufmgr_gem->pci_device))
                bufmgr_gem->gen = 8;
-       else if (IS_GEN9(bufmgr_gem->pci_device))
-               bufmgr_gem->gen = 9;
-       else {
+       else if (!intel_get_genx(bufmgr_gem->pci_device, &bufmgr_gem->gen)) {
                free(bufmgr_gem);
                bufmgr_gem = NULL;
                goto exit;
@@ -3254,7 +3676,7 @@ drm_intel_bufmgr_gem_init(int fd, int batch_size)
            bufmgr_gem->gtt_size > 256*1024*1024) {
                /* The unmappable part of gtt on gen 3 (i.e. above 256MB) can't
                 * be used for tiled blits. To simplify the accounting, just
-                * substract the unmappable part (fixed to 256MB on all known
+                * subtract the unmappable part (fixed to 256MB on all known
                 * gen3 devices) if the kernel advertises it. */
                bufmgr_gem->gtt_size -= 256*1024*1024;
        }
@@ -3279,6 +3701,10 @@ drm_intel_bufmgr_gem_init(int fd, int batch_size)
        ret = drmIoctl(bufmgr_gem->fd, DRM_IOCTL_I915_GETPARAM, &gp);
        bufmgr_gem->has_relaxed_fencing = ret == 0;
 
+       gp.param = I915_PARAM_HAS_EXEC_ASYNC;
+       ret = drmIoctl(bufmgr_gem->fd, DRM_IOCTL_I915_GETPARAM, &gp);
+       bufmgr_gem->has_exec_async = ret == 0;
+
        bufmgr_gem->bufmgr.bo_alloc_userptr = check_bo_alloc_userptr;
 
        gp.param = I915_PARAM_HAS_WAIT_TIMEOUT;
@@ -3300,6 +3726,11 @@ drm_intel_bufmgr_gem_init(int fd, int batch_size)
        ret = drmIoctl(bufmgr_gem->fd, DRM_IOCTL_I915_GETPARAM, &gp);
        bufmgr_gem->has_vebox = (ret == 0) & (*gp.value > 0);
 
+       gp.param = I915_PARAM_HAS_EXEC_SOFTPIN;
+       ret = drmIoctl(bufmgr_gem->fd, DRM_IOCTL_I915_GETPARAM, &gp);
+       if (ret == 0 && *gp.value > 0)
+               bufmgr_gem->bufmgr.bo_set_softpin_offset = drm_intel_gem_bo_set_softpin_offset;
+
        if (bufmgr_gem->gen < 4) {
                gp.param = I915_PARAM_NUM_FENCES_AVAIL;
                gp.value = &bufmgr_gem->available_fences;
@@ -3377,7 +3808,6 @@ drm_intel_bufmgr_gem_init(int fd, int batch_size)
            drm_intel_gem_get_pipe_from_crtc_id;
        bufmgr_gem->bufmgr.bo_references = drm_intel_gem_bo_references;
 
-       DRMINITLISTHEAD(&bufmgr_gem->named);
        init_cache_buckets(bufmgr_gem);
 
        DRMINITLISTHEAD(&bufmgr_gem->vma_cache);