Commit d209b9c3 authored by Michał Winiarski's avatar Michał Winiarski Committed by Chris Wilson

drm/i915/gtt: Split gen8_ppgtt_clear_pte_range

Let's use more top-down approach, where each gen8_ppgtt_clear_* function
is responsible for clearing the struct passed as an argument and calling
relevant clear_range functions on lower-level tables.
Doing this rather than operating on PTE ranges makes the implementation
of shrinking page tables quite simple.

v2: Drop min when calculating num_entries, no negation in 48b ppgtt
check, no newlines in vars block (Joonas)

Cc: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Joonas Lahtinen <joonas.lahtinen@linux.intel.com>
Cc: Michel Thierry <michel.thierry@intel.com>
Cc: Mika Kuoppala <mika.kuoppala@intel.com>
Signed-off-by: default avatarMichał Winiarski <michal.winiarski@intel.com>
Signed-off-by: default avatarChris Wilson <chris@chris-wilson.co.uk>
Link: http://patchwork.freedesktop.org/patch/msgid/1476360162-24062-2-git-send-email-michal.winiarski@intel.com
parent 4fb84d99
...@@ -706,59 +706,78 @@ static int gen8_48b_mm_switch(struct i915_hw_ppgtt *ppgtt, ...@@ -706,59 +706,78 @@ static int gen8_48b_mm_switch(struct i915_hw_ppgtt *ppgtt,
return gen8_write_pdp(req, 0, px_dma(&ppgtt->pml4)); return gen8_write_pdp(req, 0, px_dma(&ppgtt->pml4));
} }
static void gen8_ppgtt_clear_pte_range(struct i915_address_space *vm, static void gen8_ppgtt_clear_pt(struct i915_address_space *vm,
struct i915_page_directory_pointer *pdp, struct i915_page_table *pt,
uint64_t start, uint64_t start,
uint64_t length, uint64_t length)
gen8_pte_t scratch_pte)
{ {
struct i915_hw_ppgtt *ppgtt = i915_vm_to_ppgtt(vm); struct i915_hw_ppgtt *ppgtt = i915_vm_to_ppgtt(vm);
unsigned int pte_start = gen8_pte_index(start);
unsigned int num_entries = gen8_pte_count(start, length);
uint64_t pte;
gen8_pte_t *pt_vaddr; gen8_pte_t *pt_vaddr;
unsigned pdpe = gen8_pdpe_index(start); gen8_pte_t scratch_pte = gen8_pte_encode(vm->scratch_page.daddr,
unsigned pde = gen8_pde_index(start); I915_CACHE_LLC);
unsigned pte = gen8_pte_index(start);
unsigned num_entries = length >> PAGE_SHIFT;
unsigned last_pte, i;
if (WARN_ON(!pdp)) if (WARN_ON(!px_page(pt)))
return; return;
while (num_entries) { bitmap_clear(pt->used_ptes, pte_start, num_entries);
struct i915_page_directory *pd;
struct i915_page_table *pt;
if (WARN_ON(!pdp->page_directory[pdpe])) pt_vaddr = kmap_px(pt);
break;
pd = pdp->page_directory[pdpe]; for (pte = pte_start; pte < num_entries; pte++)
pt_vaddr[pte] = scratch_pte;
if (WARN_ON(!pd->page_table[pde])) kunmap_px(ppgtt, pt_vaddr);
break; }
pt = pd->page_table[pde]; static void gen8_ppgtt_clear_pd(struct i915_address_space *vm,
struct i915_page_directory *pd,
uint64_t start,
uint64_t length)
{
struct i915_page_table *pt;
uint64_t pde;
if (WARN_ON(!px_page(pt))) gen8_for_each_pde(pt, pd, start, length, pde) {
if (WARN_ON(!pd->page_table[pde]))
break; break;
last_pte = pte + num_entries; gen8_ppgtt_clear_pt(vm, pt, start, length);
if (last_pte > GEN8_PTES) }
last_pte = GEN8_PTES; }
pt_vaddr = kmap_px(pt); static void gen8_ppgtt_clear_pdp(struct i915_address_space *vm,
struct i915_page_directory_pointer *pdp,
uint64_t start,
uint64_t length)
{
struct i915_page_directory *pd;
uint64_t pdpe;
for (i = pte; i < last_pte; i++) { gen8_for_each_pdpe(pd, pdp, start, length, pdpe) {
pt_vaddr[i] = scratch_pte; if (WARN_ON(!pdp->page_directory[pdpe]))
num_entries--; break;
gen8_ppgtt_clear_pd(vm, pd, start, length);
} }
}
kunmap_px(ppgtt, pt_vaddr); static void gen8_ppgtt_clear_pml4(struct i915_address_space *vm,
struct i915_pml4 *pml4,
uint64_t start,
uint64_t length)
{
struct i915_page_directory_pointer *pdp;
uint64_t pml4e;
pte = 0; gen8_for_each_pml4e(pdp, pml4, start, length, pml4e) {
if (++pde == I915_PDES) { if (WARN_ON(!pml4->pdps[pml4e]))
if (++pdpe == I915_PDPES_PER_PDP(vm->dev))
break; break;
pde = 0;
} gen8_ppgtt_clear_pdp(vm, pdp, start, length);
} }
} }
...@@ -766,21 +785,11 @@ static void gen8_ppgtt_clear_range(struct i915_address_space *vm, ...@@ -766,21 +785,11 @@ static void gen8_ppgtt_clear_range(struct i915_address_space *vm,
uint64_t start, uint64_t length) uint64_t start, uint64_t length)
{ {
struct i915_hw_ppgtt *ppgtt = i915_vm_to_ppgtt(vm); struct i915_hw_ppgtt *ppgtt = i915_vm_to_ppgtt(vm);
gen8_pte_t scratch_pte = gen8_pte_encode(vm->scratch_page.daddr,
I915_CACHE_LLC);
if (!USES_FULL_48BIT_PPGTT(vm->dev)) { if (USES_FULL_48BIT_PPGTT(vm->dev))
gen8_ppgtt_clear_pte_range(vm, &ppgtt->pdp, start, length, gen8_ppgtt_clear_pml4(vm, &ppgtt->pml4, start, length);
scratch_pte); else
} else { gen8_ppgtt_clear_pdp(vm, &ppgtt->pdp, start, length);
uint64_t pml4e;
struct i915_page_directory_pointer *pdp;
gen8_for_each_pml4e(pdp, &ppgtt->pml4, start, length, pml4e) {
gen8_ppgtt_clear_pte_range(vm, pdp, start, length,
scratch_pte);
}
}
} }
static void static void
......
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