Commit f6cd1f15 authored by Ben Widawsky's avatar Ben Widawsky Committed by Daniel Vetter

drm/i915: Use new bind/unbind in eviction code

Eviction code, like the rest of the converted code needs to be aware of
the address space for which it is evicting (or the everything case, all
addresses). With the updated bind/unbind interfaces of the last patch,
we can now safely move the eviction code over.
Signed-off-by: default avatarBen Widawsky <ben@bwidawsk.net>
Signed-off-by: default avatarDaniel Vetter <daniel.vetter@ffwll.ch>
parent 07fe0b12
...@@ -1983,7 +1983,9 @@ static inline void i915_gem_chipset_flush(struct drm_device *dev) ...@@ -1983,7 +1983,9 @@ static inline void i915_gem_chipset_flush(struct drm_device *dev)
/* i915_gem_evict.c */ /* i915_gem_evict.c */
int __must_check i915_gem_evict_something(struct drm_device *dev, int min_size, int __must_check i915_gem_evict_something(struct drm_device *dev,
struct i915_address_space *vm,
int min_size,
unsigned alignment, unsigned alignment,
unsigned cache_level, unsigned cache_level,
bool mappable, bool mappable,
......
...@@ -3166,7 +3166,7 @@ i915_gem_object_bind_to_vm(struct drm_i915_gem_object *obj, ...@@ -3166,7 +3166,7 @@ i915_gem_object_bind_to_vm(struct drm_i915_gem_object *obj,
size, alignment, size, alignment,
obj->cache_level, 0, gtt_max); obj->cache_level, 0, gtt_max);
if (ret) { if (ret) {
ret = i915_gem_evict_something(dev, size, alignment, ret = i915_gem_evict_something(dev, vm, size, alignment,
obj->cache_level, obj->cache_level,
map_and_fenceable, map_and_fenceable,
nonblocking); nonblocking);
......
...@@ -32,26 +32,21 @@ ...@@ -32,26 +32,21 @@
#include "i915_trace.h" #include "i915_trace.h"
static bool static bool
mark_free(struct drm_i915_gem_object *obj, struct list_head *unwind) mark_free(struct i915_vma *vma, struct list_head *unwind)
{ {
struct drm_device *dev = obj->base.dev; if (vma->obj->pin_count)
struct drm_i915_private *dev_priv = dev->dev_private;
struct i915_vma *vma = i915_gem_obj_to_vma(obj, &dev_priv->gtt.base);
if (obj->pin_count)
return false; return false;
list_add(&obj->exec_list, unwind); list_add(&vma->obj->exec_list, unwind);
return drm_mm_scan_add_block(&vma->node); return drm_mm_scan_add_block(&vma->node);
} }
int int
i915_gem_evict_something(struct drm_device *dev, int min_size, i915_gem_evict_something(struct drm_device *dev, struct i915_address_space *vm,
unsigned alignment, unsigned cache_level, int min_size, unsigned alignment, unsigned cache_level,
bool mappable, bool nonblocking) bool mappable, bool nonblocking)
{ {
drm_i915_private_t *dev_priv = dev->dev_private; drm_i915_private_t *dev_priv = dev->dev_private;
struct i915_address_space *vm = &dev_priv->gtt.base;
struct list_head eviction_list, unwind_list; struct list_head eviction_list, unwind_list;
struct i915_vma *vma; struct i915_vma *vma;
struct drm_i915_gem_object *obj; struct drm_i915_gem_object *obj;
...@@ -83,16 +78,18 @@ i915_gem_evict_something(struct drm_device *dev, int min_size, ...@@ -83,16 +78,18 @@ i915_gem_evict_something(struct drm_device *dev, int min_size,
*/ */
INIT_LIST_HEAD(&unwind_list); INIT_LIST_HEAD(&unwind_list);
if (mappable) if (mappable) {
BUG_ON(!i915_is_ggtt(vm));
drm_mm_init_scan_with_range(&vm->mm, min_size, drm_mm_init_scan_with_range(&vm->mm, min_size,
alignment, cache_level, 0, alignment, cache_level, 0,
dev_priv->gtt.mappable_end); dev_priv->gtt.mappable_end);
else } else
drm_mm_init_scan(&vm->mm, min_size, alignment, cache_level); drm_mm_init_scan(&vm->mm, min_size, alignment, cache_level);
/* First see if there is a large enough contiguous idle region... */ /* First see if there is a large enough contiguous idle region... */
list_for_each_entry(obj, &vm->inactive_list, mm_list) { list_for_each_entry(obj, &vm->inactive_list, mm_list) {
if (mark_free(obj, &unwind_list)) struct i915_vma *vma = i915_gem_obj_to_vma(obj, vm);
if (mark_free(vma, &unwind_list))
goto found; goto found;
} }
...@@ -101,7 +98,8 @@ i915_gem_evict_something(struct drm_device *dev, int min_size, ...@@ -101,7 +98,8 @@ i915_gem_evict_something(struct drm_device *dev, int min_size,
/* Now merge in the soon-to-be-expired objects... */ /* Now merge in the soon-to-be-expired objects... */
list_for_each_entry(obj, &vm->active_list, mm_list) { list_for_each_entry(obj, &vm->active_list, mm_list) {
if (mark_free(obj, &unwind_list)) struct i915_vma *vma = i915_gem_obj_to_vma(obj, vm);
if (mark_free(vma, &unwind_list))
goto found; goto found;
} }
...@@ -111,7 +109,7 @@ i915_gem_evict_something(struct drm_device *dev, int min_size, ...@@ -111,7 +109,7 @@ i915_gem_evict_something(struct drm_device *dev, int min_size,
obj = list_first_entry(&unwind_list, obj = list_first_entry(&unwind_list,
struct drm_i915_gem_object, struct drm_i915_gem_object,
exec_list); exec_list);
vma = i915_gem_obj_to_vma(obj, &dev_priv->gtt.base); vma = i915_gem_obj_to_vma(obj, vm);
ret = drm_mm_scan_remove_block(&vma->node); ret = drm_mm_scan_remove_block(&vma->node);
BUG_ON(ret); BUG_ON(ret);
...@@ -132,7 +130,7 @@ i915_gem_evict_something(struct drm_device *dev, int min_size, ...@@ -132,7 +130,7 @@ i915_gem_evict_something(struct drm_device *dev, int min_size,
obj = list_first_entry(&unwind_list, obj = list_first_entry(&unwind_list,
struct drm_i915_gem_object, struct drm_i915_gem_object,
exec_list); exec_list);
vma = i915_gem_obj_to_vma(obj, &dev_priv->gtt.base); vma = i915_gem_obj_to_vma(obj, vm);
if (drm_mm_scan_remove_block(&vma->node)) { if (drm_mm_scan_remove_block(&vma->node)) {
list_move(&obj->exec_list, &eviction_list); list_move(&obj->exec_list, &eviction_list);
drm_gem_object_reference(&obj->base); drm_gem_object_reference(&obj->base);
...@@ -147,7 +145,7 @@ i915_gem_evict_something(struct drm_device *dev, int min_size, ...@@ -147,7 +145,7 @@ i915_gem_evict_something(struct drm_device *dev, int min_size,
struct drm_i915_gem_object, struct drm_i915_gem_object,
exec_list); exec_list);
if (ret == 0) if (ret == 0)
ret = i915_gem_object_ggtt_unbind(obj); ret = i915_vma_unbind(i915_gem_obj_to_vma(obj, vm));
list_del_init(&obj->exec_list); list_del_init(&obj->exec_list);
drm_gem_object_unreference(&obj->base); drm_gem_object_unreference(&obj->base);
...@@ -160,13 +158,18 @@ int ...@@ -160,13 +158,18 @@ int
i915_gem_evict_everything(struct drm_device *dev) i915_gem_evict_everything(struct drm_device *dev)
{ {
drm_i915_private_t *dev_priv = dev->dev_private; drm_i915_private_t *dev_priv = dev->dev_private;
struct i915_address_space *vm = &dev_priv->gtt.base; struct i915_address_space *vm;
struct drm_i915_gem_object *obj, *next; struct drm_i915_gem_object *obj, *next;
bool lists_empty; bool lists_empty = true;
int ret; int ret;
lists_empty = (list_empty(&vm->inactive_list) && list_for_each_entry(vm, &dev_priv->vm_list, global_link) {
list_empty(&vm->active_list)); lists_empty = (list_empty(&vm->inactive_list) &&
list_empty(&vm->active_list));
if (!lists_empty)
lists_empty = false;
}
if (lists_empty) if (lists_empty)
return -ENOSPC; return -ENOSPC;
...@@ -183,9 +186,11 @@ i915_gem_evict_everything(struct drm_device *dev) ...@@ -183,9 +186,11 @@ i915_gem_evict_everything(struct drm_device *dev)
i915_gem_retire_requests(dev); i915_gem_retire_requests(dev);
/* Having flushed everything, unbind() should never raise an error */ /* Having flushed everything, unbind() should never raise an error */
list_for_each_entry_safe(obj, next, &vm->inactive_list, mm_list) list_for_each_entry(vm, &dev_priv->vm_list, global_link) {
if (obj->pin_count == 0) list_for_each_entry_safe(obj, next, &vm->inactive_list, mm_list)
WARN_ON(i915_gem_object_ggtt_unbind(obj)); if (obj->pin_count == 0)
WARN_ON(i915_vma_unbind(i915_gem_obj_to_vma(obj, vm)));
}
return 0; return 0;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment