drm/i915: Split batch pool into size buckets
Now with the trimmed memcpy before the command parser, we try to allocate many different sizes of batches, predominantly one or two pages. We can therefore speed up searching for a good sized batch by keeping the objects of buckets of roughly the same size. v2: Add a comment about bucket sizes Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
This commit is contained in:
parent
35c94185c5
commit
8d9d5744c6
@ -378,15 +378,17 @@ static void print_batch_pool_stats(struct seq_file *m,
|
|||||||
struct drm_i915_gem_object *obj;
|
struct drm_i915_gem_object *obj;
|
||||||
struct file_stats stats;
|
struct file_stats stats;
|
||||||
struct intel_engine_cs *ring;
|
struct intel_engine_cs *ring;
|
||||||
int i;
|
int i, j;
|
||||||
|
|
||||||
memset(&stats, 0, sizeof(stats));
|
memset(&stats, 0, sizeof(stats));
|
||||||
|
|
||||||
for_each_ring(ring, dev_priv, i) {
|
for_each_ring(ring, dev_priv, i) {
|
||||||
list_for_each_entry(obj,
|
for (j = 0; j < ARRAY_SIZE(ring->batch_pool.cache_list); j++) {
|
||||||
&ring->batch_pool.cache_list,
|
list_for_each_entry(obj,
|
||||||
batch_pool_list)
|
&ring->batch_pool.cache_list[j],
|
||||||
per_file_stats(0, obj, &stats);
|
batch_pool_link)
|
||||||
|
per_file_stats(0, obj, &stats);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
print_file_stats(m, "batch pool", stats);
|
print_file_stats(m, "batch pool", stats);
|
||||||
@ -618,26 +620,38 @@ static int i915_gem_batch_pool_info(struct seq_file *m, void *data)
|
|||||||
struct drm_i915_private *dev_priv = dev->dev_private;
|
struct drm_i915_private *dev_priv = dev->dev_private;
|
||||||
struct drm_i915_gem_object *obj;
|
struct drm_i915_gem_object *obj;
|
||||||
struct intel_engine_cs *ring;
|
struct intel_engine_cs *ring;
|
||||||
int count = 0;
|
int total = 0;
|
||||||
int ret, i;
|
int ret, i, j;
|
||||||
|
|
||||||
ret = mutex_lock_interruptible(&dev->struct_mutex);
|
ret = mutex_lock_interruptible(&dev->struct_mutex);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
for_each_ring(ring, dev_priv, i) {
|
for_each_ring(ring, dev_priv, i) {
|
||||||
seq_printf(m, "%s cache:\n", ring->name);
|
for (j = 0; j < ARRAY_SIZE(ring->batch_pool.cache_list); j++) {
|
||||||
list_for_each_entry(obj,
|
int count;
|
||||||
&ring->batch_pool.cache_list,
|
|
||||||
batch_pool_list) {
|
count = 0;
|
||||||
seq_puts(m, " ");
|
list_for_each_entry(obj,
|
||||||
describe_obj(m, obj);
|
&ring->batch_pool.cache_list[j],
|
||||||
seq_putc(m, '\n');
|
batch_pool_link)
|
||||||
count++;
|
count++;
|
||||||
|
seq_printf(m, "%s cache[%d]: %d objects\n",
|
||||||
|
ring->name, j, count);
|
||||||
|
|
||||||
|
list_for_each_entry(obj,
|
||||||
|
&ring->batch_pool.cache_list[j],
|
||||||
|
batch_pool_link) {
|
||||||
|
seq_puts(m, " ");
|
||||||
|
describe_obj(m, obj);
|
||||||
|
seq_putc(m, '\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
total += count;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
seq_printf(m, "total: %d\n", count);
|
seq_printf(m, "total: %d\n", total);
|
||||||
|
|
||||||
mutex_unlock(&dev->struct_mutex);
|
mutex_unlock(&dev->struct_mutex);
|
||||||
|
|
||||||
|
@ -1910,7 +1910,7 @@ struct drm_i915_gem_object {
|
|||||||
/** Used in execbuf to temporarily hold a ref */
|
/** Used in execbuf to temporarily hold a ref */
|
||||||
struct list_head obj_exec_link;
|
struct list_head obj_exec_link;
|
||||||
|
|
||||||
struct list_head batch_pool_list;
|
struct list_head batch_pool_link;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is set if the object is on the active lists (has pending
|
* This is set if the object is on the active lists (has pending
|
||||||
|
@ -4409,7 +4409,7 @@ void i915_gem_object_init(struct drm_i915_gem_object *obj,
|
|||||||
INIT_LIST_HEAD(&obj->ring_list);
|
INIT_LIST_HEAD(&obj->ring_list);
|
||||||
INIT_LIST_HEAD(&obj->obj_exec_link);
|
INIT_LIST_HEAD(&obj->obj_exec_link);
|
||||||
INIT_LIST_HEAD(&obj->vma_list);
|
INIT_LIST_HEAD(&obj->vma_list);
|
||||||
INIT_LIST_HEAD(&obj->batch_pool_list);
|
INIT_LIST_HEAD(&obj->batch_pool_link);
|
||||||
|
|
||||||
obj->ops = ops;
|
obj->ops = ops;
|
||||||
|
|
||||||
|
@ -47,8 +47,12 @@
|
|||||||
void i915_gem_batch_pool_init(struct drm_device *dev,
|
void i915_gem_batch_pool_init(struct drm_device *dev,
|
||||||
struct i915_gem_batch_pool *pool)
|
struct i915_gem_batch_pool *pool)
|
||||||
{
|
{
|
||||||
|
int n;
|
||||||
|
|
||||||
pool->dev = dev;
|
pool->dev = dev;
|
||||||
INIT_LIST_HEAD(&pool->cache_list);
|
|
||||||
|
for (n = 0; n < ARRAY_SIZE(pool->cache_list); n++)
|
||||||
|
INIT_LIST_HEAD(&pool->cache_list[n]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -59,16 +63,20 @@ void i915_gem_batch_pool_init(struct drm_device *dev,
|
|||||||
*/
|
*/
|
||||||
void i915_gem_batch_pool_fini(struct i915_gem_batch_pool *pool)
|
void i915_gem_batch_pool_fini(struct i915_gem_batch_pool *pool)
|
||||||
{
|
{
|
||||||
|
int n;
|
||||||
|
|
||||||
WARN_ON(!mutex_is_locked(&pool->dev->struct_mutex));
|
WARN_ON(!mutex_is_locked(&pool->dev->struct_mutex));
|
||||||
|
|
||||||
while (!list_empty(&pool->cache_list)) {
|
for (n = 0; n < ARRAY_SIZE(pool->cache_list); n++) {
|
||||||
struct drm_i915_gem_object *obj =
|
while (!list_empty(&pool->cache_list[n])) {
|
||||||
list_first_entry(&pool->cache_list,
|
struct drm_i915_gem_object *obj =
|
||||||
struct drm_i915_gem_object,
|
list_first_entry(&pool->cache_list[n],
|
||||||
batch_pool_list);
|
struct drm_i915_gem_object,
|
||||||
|
batch_pool_link);
|
||||||
|
|
||||||
list_del(&obj->batch_pool_list);
|
list_del(&obj->batch_pool_link);
|
||||||
drm_gem_object_unreference(&obj->base);
|
drm_gem_object_unreference(&obj->base);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,28 +99,33 @@ i915_gem_batch_pool_get(struct i915_gem_batch_pool *pool,
|
|||||||
{
|
{
|
||||||
struct drm_i915_gem_object *obj = NULL;
|
struct drm_i915_gem_object *obj = NULL;
|
||||||
struct drm_i915_gem_object *tmp, *next;
|
struct drm_i915_gem_object *tmp, *next;
|
||||||
|
struct list_head *list;
|
||||||
|
int n;
|
||||||
|
|
||||||
WARN_ON(!mutex_is_locked(&pool->dev->struct_mutex));
|
WARN_ON(!mutex_is_locked(&pool->dev->struct_mutex));
|
||||||
|
|
||||||
list_for_each_entry_safe(tmp, next,
|
/* Compute a power-of-two bucket, but throw everything greater than
|
||||||
&pool->cache_list, batch_pool_list) {
|
* 16KiB into the same bucket: i.e. the the buckets hold objects of
|
||||||
|
* (1 page, 2 pages, 4 pages, 8+ pages).
|
||||||
|
*/
|
||||||
|
n = fls(size >> PAGE_SHIFT) - 1;
|
||||||
|
if (n >= ARRAY_SIZE(pool->cache_list))
|
||||||
|
n = ARRAY_SIZE(pool->cache_list) - 1;
|
||||||
|
list = &pool->cache_list[n];
|
||||||
|
|
||||||
|
list_for_each_entry_safe(tmp, next, list, batch_pool_link) {
|
||||||
/* The batches are strictly LRU ordered */
|
/* The batches are strictly LRU ordered */
|
||||||
if (tmp->active)
|
if (tmp->active)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* While we're looping, do some clean up */
|
/* While we're looping, do some clean up */
|
||||||
if (tmp->madv == __I915_MADV_PURGED) {
|
if (tmp->madv == __I915_MADV_PURGED) {
|
||||||
list_del(&tmp->batch_pool_list);
|
list_del(&tmp->batch_pool_link);
|
||||||
drm_gem_object_unreference(&tmp->base);
|
drm_gem_object_unreference(&tmp->base);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
if (tmp->base.size >= size) {
|
||||||
* Select a buffer that is at least as big as needed
|
|
||||||
* but not 'too much' bigger. A better way to do this
|
|
||||||
* might be to bucket the pool objects based on size.
|
|
||||||
*/
|
|
||||||
if (tmp->base.size >= size && tmp->base.size <= 2 * size) {
|
|
||||||
obj = tmp;
|
obj = tmp;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -132,7 +145,7 @@ i915_gem_batch_pool_get(struct i915_gem_batch_pool *pool,
|
|||||||
obj->madv = I915_MADV_DONTNEED;
|
obj->madv = I915_MADV_DONTNEED;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_move_tail(&obj->batch_pool_list, &pool->cache_list);
|
list_move_tail(&obj->batch_pool_link, list);
|
||||||
i915_gem_object_pin_pages(obj);
|
i915_gem_object_pin_pages(obj);
|
||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
|
|
||||||
struct i915_gem_batch_pool {
|
struct i915_gem_batch_pool {
|
||||||
struct drm_device *dev;
|
struct drm_device *dev;
|
||||||
struct list_head cache_list;
|
struct list_head cache_list[4];
|
||||||
};
|
};
|
||||||
|
|
||||||
/* i915_gem_batch_pool.c */
|
/* i915_gem_batch_pool.c */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user