Commit a9ded785 authored by Chris Wilson's avatar Chris Wilson

drm/i915/gtt: Onionify error handling for gen6_ppgtt_create

Pull the empty stubs together into the top level gen6_ppgtt_create, and
tear each one down on error in proper onion order (rather than use
Joonas' pet hate of calling the cleanup function in indeterminable
state).
Signed-off-by: default avatarChris Wilson <chris@chris-wilson.co.uk>
Cc: Joonas Lahtinen <joonas.lahtinen@linux.intel.com>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Cc: Matthew Auld <matthew.william.auld@gmail.com>
Reviewed-by: default avatarMatthew Auld <matthew.william.auld@gmail.com>
Reviewed-by: default avatarJoonas Lahtinen <joonas.lahtinen@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20180612081815.3585-2-chris@chris-wilson.co.uk
parent 35ac40d8
...@@ -1875,7 +1875,7 @@ static int gen6_alloc_va_range(struct i915_address_space *vm, ...@@ -1875,7 +1875,7 @@ static int gen6_alloc_va_range(struct i915_address_space *vm,
return -ENOMEM; return -ENOMEM;
} }
static int gen6_init_scratch(struct i915_address_space *vm) static int gen6_ppgtt_init_scratch(struct i915_address_space *vm)
{ {
int ret; int ret;
...@@ -1894,33 +1894,37 @@ static int gen6_init_scratch(struct i915_address_space *vm) ...@@ -1894,33 +1894,37 @@ static int gen6_init_scratch(struct i915_address_space *vm)
return 0; return 0;
} }
static void gen6_free_scratch(struct i915_address_space *vm) static void gen6_ppgtt_free_scratch(struct i915_address_space *vm)
{ {
free_pt(vm, vm->scratch_pt); free_pt(vm, vm->scratch_pt);
cleanup_scratch_page(vm); cleanup_scratch_page(vm);
} }
static void gen6_ppgtt_cleanup(struct i915_address_space *vm) static void gen6_ppgtt_free_pd(struct gen6_hw_ppgtt *ppgtt)
{ {
struct gen6_hw_ppgtt *ppgtt = to_gen6_ppgtt(i915_vm_to_ppgtt(vm));
struct i915_page_table *pt; struct i915_page_table *pt;
u32 pde; u32 pde;
drm_mm_remove_node(&ppgtt->node);
gen6_for_all_pdes(pt, &ppgtt->base.pd, pde) gen6_for_all_pdes(pt, &ppgtt->base.pd, pde)
if (pt != vm->scratch_pt) if (pt != ppgtt->base.vm.scratch_pt)
free_pt(vm, pt); free_pt(&ppgtt->base.vm, pt);
}
gen6_free_scratch(vm); static void gen6_ppgtt_cleanup(struct i915_address_space *vm)
{
struct gen6_hw_ppgtt *ppgtt = to_gen6_ppgtt(i915_vm_to_ppgtt(vm));
drm_mm_remove_node(&ppgtt->node);
gen6_ppgtt_free_pd(ppgtt);
gen6_ppgtt_free_scratch(vm);
} }
static int gen6_ppgtt_allocate_page_directories(struct gen6_hw_ppgtt *ppgtt) static int gen6_ppgtt_allocate_page_directories(struct gen6_hw_ppgtt *ppgtt)
{ {
struct i915_address_space *vm = &ppgtt->base.vm;
struct drm_i915_private *dev_priv = ppgtt->base.vm.i915; struct drm_i915_private *dev_priv = ppgtt->base.vm.i915;
struct i915_ggtt *ggtt = &dev_priv->ggtt; struct i915_ggtt *ggtt = &dev_priv->ggtt;
int ret; int err;
/* PPGTT PDEs reside in the GGTT and consists of 512 entries. The /* PPGTT PDEs reside in the GGTT and consists of 512 entries. The
* allocator works in address space sizes, so it's multiplied by page * allocator works in address space sizes, so it's multiplied by page
...@@ -1928,17 +1932,13 @@ static int gen6_ppgtt_allocate_page_directories(struct gen6_hw_ppgtt *ppgtt) ...@@ -1928,17 +1932,13 @@ static int gen6_ppgtt_allocate_page_directories(struct gen6_hw_ppgtt *ppgtt)
*/ */
BUG_ON(!drm_mm_initialized(&ggtt->vm.mm)); BUG_ON(!drm_mm_initialized(&ggtt->vm.mm));
ret = gen6_init_scratch(vm); err = i915_gem_gtt_insert(&ggtt->vm, &ppgtt->node,
if (ret)
return ret;
ret = i915_gem_gtt_insert(&ggtt->vm, &ppgtt->node,
GEN6_PD_SIZE, GEN6_PD_ALIGN, GEN6_PD_SIZE, GEN6_PD_ALIGN,
I915_COLOR_UNEVICTABLE, I915_COLOR_UNEVICTABLE,
0, ggtt->vm.total, 0, ggtt->vm.total,
PIN_HIGH); PIN_HIGH);
if (ret) if (err)
goto err_out; return err;
if (ppgtt->node.start < ggtt->mappable_end) if (ppgtt->node.start < ggtt->mappable_end)
DRM_DEBUG("Forced to use aperture for PDEs\n"); DRM_DEBUG("Forced to use aperture for PDEs\n");
...@@ -1950,15 +1950,6 @@ static int gen6_ppgtt_allocate_page_directories(struct gen6_hw_ppgtt *ppgtt) ...@@ -1950,15 +1950,6 @@ static int gen6_ppgtt_allocate_page_directories(struct gen6_hw_ppgtt *ppgtt)
ppgtt->base.pd.base.ggtt_offset / sizeof(gen6_pte_t); ppgtt->base.pd.base.ggtt_offset / sizeof(gen6_pte_t);
return 0; return 0;
err_out:
gen6_free_scratch(vm);
return ret;
}
static int gen6_ppgtt_alloc(struct gen6_hw_ppgtt *ppgtt)
{
return gen6_ppgtt_allocate_page_directories(ppgtt);
} }
static void gen6_scratch_va_range(struct gen6_hw_ppgtt *ppgtt, static void gen6_scratch_va_range(struct gen6_hw_ppgtt *ppgtt,
...@@ -1984,21 +1975,8 @@ static struct i915_hw_ppgtt *gen6_ppgtt_create(struct drm_i915_private *i915) ...@@ -1984,21 +1975,8 @@ static struct i915_hw_ppgtt *gen6_ppgtt_create(struct drm_i915_private *i915)
ppgtt->base.vm.i915 = i915; ppgtt->base.vm.i915 = i915;
ppgtt->base.vm.dma = &i915->drm.pdev->dev; ppgtt->base.vm.dma = &i915->drm.pdev->dev;
ppgtt->base.vm.pte_encode = ggtt->vm.pte_encode;
err = gen6_ppgtt_alloc(ppgtt);
if (err)
goto err_free;
ppgtt->base.vm.total = I915_PDES * GEN6_PTES * PAGE_SIZE; ppgtt->base.vm.total = I915_PDES * GEN6_PTES * PAGE_SIZE;
gen6_scratch_va_range(ppgtt, 0, ppgtt->base.vm.total);
gen6_write_page_range(&ppgtt->base, 0, ppgtt->base.vm.total);
err = gen6_alloc_va_range(&ppgtt->base.vm, 0, ppgtt->base.vm.total);
if (err)
goto err_cleanup;
ppgtt->base.vm.clear_range = gen6_ppgtt_clear_range; ppgtt->base.vm.clear_range = gen6_ppgtt_clear_range;
ppgtt->base.vm.insert_entries = gen6_ppgtt_insert_entries; ppgtt->base.vm.insert_entries = gen6_ppgtt_insert_entries;
ppgtt->base.vm.cleanup = gen6_ppgtt_cleanup; ppgtt->base.vm.cleanup = gen6_ppgtt_cleanup;
...@@ -2009,6 +1987,23 @@ static struct i915_hw_ppgtt *gen6_ppgtt_create(struct drm_i915_private *i915) ...@@ -2009,6 +1987,23 @@ static struct i915_hw_ppgtt *gen6_ppgtt_create(struct drm_i915_private *i915)
ppgtt->base.vm.vma_ops.set_pages = ppgtt_set_pages; ppgtt->base.vm.vma_ops.set_pages = ppgtt_set_pages;
ppgtt->base.vm.vma_ops.clear_pages = clear_pages; ppgtt->base.vm.vma_ops.clear_pages = clear_pages;
ppgtt->base.vm.pte_encode = ggtt->vm.pte_encode;
err = gen6_ppgtt_init_scratch(&ppgtt->base.vm);
if (err)
goto err_free;
err = gen6_ppgtt_allocate_page_directories(ppgtt);
if (err)
goto err_scratch;
gen6_scratch_va_range(ppgtt, 0, ppgtt->base.vm.total);
gen6_write_page_range(&ppgtt->base, 0, ppgtt->base.vm.total);
err = gen6_alloc_va_range(&ppgtt->base.vm, 0, ppgtt->base.vm.total);
if (err)
goto err_pd;
DRM_DEBUG_DRIVER("Allocated pde space (%lldM) at GTT entry: %llx\n", DRM_DEBUG_DRIVER("Allocated pde space (%lldM) at GTT entry: %llx\n",
ppgtt->node.size >> 20, ppgtt->node.size >> 20,
ppgtt->node.start / PAGE_SIZE); ppgtt->node.start / PAGE_SIZE);
...@@ -2018,8 +2013,10 @@ static struct i915_hw_ppgtt *gen6_ppgtt_create(struct drm_i915_private *i915) ...@@ -2018,8 +2013,10 @@ static struct i915_hw_ppgtt *gen6_ppgtt_create(struct drm_i915_private *i915)
return &ppgtt->base; return &ppgtt->base;
err_cleanup: err_pd:
gen6_ppgtt_cleanup(&ppgtt->base.vm); gen6_ppgtt_free_pd(ppgtt);
err_scratch:
gen6_ppgtt_free_scratch(&ppgtt->base.vm);
err_free: err_free:
kfree(ppgtt); kfree(ppgtt);
return ERR_PTR(err); return ERR_PTR(err);
......
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