[Intel-gfx] [PATCH] drm/i915/ggtt: do not set bits 1-11 in gen12 ptes

Mika Kuoppala mika.kuoppala at linux.intel.com
Thu Feb 27 10:51:46 UTC 2020


Daniele Ceraolo Spurio <daniele.ceraolospurio at intel.com> writes:

> On TGL, bits 2-4 in the GGTT PTE are not ignored anymore and are
> instead used for some extra VT-d capabilities. We don't (yet?) have
> support for those capabilities, but, given that we shared the pte_encode
> function betweed GGTT and PPGTT, we still set those bits to the PPGTT
> PPAT values. The DMA engine gets very confused when those bits are
> set while the iommu is enabled, leading to errors. E.g. when loading
> the GuC we get:
>
> [    9.796218] DMAR: DRHD: handling fault status reg 2
> [    9.796235] DMAR: [DMA Write] Request device [00:02.0] PASID ffffffff fault addr 0 [fault reason 02] Present bit in context entry is clear
> [    9.899215] [drm:intel_guc_fw_upload [i915]] *ERROR* GuC firmware signature verification failed
>
> To fix this, just have dedicated gen8_pte_encode function per type of
> gtt. Also, explicitly set vm->pte_encode for gen8_ppgtt, even if we
> don't use it, to make sure we don't accidentally assign it to the GGTT
> one, like we do for gen6_ppgtt, in case we need it in the future.

Nice find. Tho I feel that the subject and commit message needs
a bit massaging for future archeologists.

We are changing the gen8+ ggtt encoding and not only gen12 and
it should be noted.

>
> Reported-by: "Sodhi, Vunny" <vunny.sodhi at intel.com>
> Signed-off-by: Daniele Ceraolo Spurio <daniele.ceraolospurio at intel.com>
> Cc: Chris Wilson <chris at chris-wilson.co.uk>
> Cc: Mika Kuoppala <mika.kuoppala at linux.intel.com>
> Cc: Matthew Auld <matthew.auld at intel.com>
> Cc: Michal Wajdeczko <michal.wajdeczko at intel.com>
> ---
>  drivers/gpu/drm/i915/gt/gen8_ppgtt.c | 26 ++++++++++++++++++++++++++
>  drivers/gpu/drm/i915/gt/intel_ggtt.c | 13 ++++++++++---
>  drivers/gpu/drm/i915/gt/intel_gtt.c  | 24 ------------------------
>  drivers/gpu/drm/i915/gt/intel_gtt.h  |  4 ----
>  4 files changed, 36 insertions(+), 31 deletions(-)
>
> diff --git a/drivers/gpu/drm/i915/gt/gen8_ppgtt.c b/drivers/gpu/drm/i915/gt/gen8_ppgtt.c
> index 4d1de2d97d5c..9aabc5815d38 100644
> --- a/drivers/gpu/drm/i915/gt/gen8_ppgtt.c
> +++ b/drivers/gpu/drm/i915/gt/gen8_ppgtt.c
> @@ -25,6 +25,30 @@ static u64 gen8_pde_encode(const dma_addr_t addr,
>  	return pde;
>  }
>  
> +static u64 gen8_pte_encode(dma_addr_t addr,
> +			   enum i915_cache_level level,
> +			   u32 flags)
> +{
> +	gen8_pte_t pte = addr | _PAGE_PRESENT | _PAGE_RW;
> +
> +	if (unlikely(flags & PTE_READ_ONLY))
> +		pte &= ~_PAGE_RW;
> +
> +	switch (level) {
> +	case I915_CACHE_NONE:
> +		pte |= PPAT_UNCACHED;
> +		break;
> +	case I915_CACHE_WT:
> +		pte |= PPAT_DISPLAY_ELLC;
> +		break;
> +	default:
> +		pte |= PPAT_CACHED;
> +		break;
> +	}
> +
> +	return pte;
> +}
> +
>  static void gen8_ppgtt_notify_vgt(struct i915_ppgtt *ppgtt, bool create)
>  {
>  	struct drm_i915_private *i915 = ppgtt->vm.i915;
> @@ -706,6 +730,8 @@ struct i915_ppgtt *gen8_ppgtt_create(struct intel_gt *gt)
>  	ppgtt->vm.allocate_va_range = gen8_ppgtt_alloc;
>  	ppgtt->vm.clear_range = gen8_ppgtt_clear;
>  
> +	ppgtt->vm.pte_encode = gen8_pte_encode;
> +
>  	if (intel_vgpu_active(gt->i915))
>  		gen8_ppgtt_notify_vgt(ppgtt, true);
>  
> diff --git a/drivers/gpu/drm/i915/gt/intel_ggtt.c b/drivers/gpu/drm/i915/gt/intel_ggtt.c
> index 41a00281f364..e45eca985b14 100644
> --- a/drivers/gpu/drm/i915/gt/intel_ggtt.c
> +++ b/drivers/gpu/drm/i915/gt/intel_ggtt.c
> @@ -157,6 +157,13 @@ static void gmch_ggtt_invalidate(struct i915_ggtt *ggtt)
>  	intel_gtt_chipset_flush();
>  }
>  
> +static u64 gen8_ggtt_pte_encode(dma_addr_t addr,
> +				enum i915_cache_level level,
> +				u32 flags)
> +{
> +	return addr | _PAGE_PRESENT;

The bspec is bit ambivalent in here. First it says R/W is ignored but
further ahead it states that bit0 and bit1 defines the validity
and rest are ignored for aperture/display.

This said, I am leaning to give a minimal set a try first :)

> +}
> +
>  static void gen8_set_pte(void __iomem *addr, gen8_pte_t pte)
>  {
>  	writeq(pte, addr);
> @@ -172,7 +179,7 @@ static void gen8_ggtt_insert_page(struct i915_address_space *vm,
>  	gen8_pte_t __iomem *pte =
>  		(gen8_pte_t __iomem *)ggtt->gsm + offset / I915_GTT_PAGE_SIZE;
>  
> -	gen8_set_pte(pte, gen8_pte_encode(addr, level, 0));
> +	gen8_set_pte(pte, gen8_ggtt_pte_encode(addr, level, 0));

Make me ponder why we don't use the vm->pte_encode all the way
as we have it :P

Nevertheless, with s/gen12/gen8 ggtt encoding in subject/
commit msg and this is,

Reviewed-by: Mika Kuoppala <mika.kuoppala at linux.intel.com>

>  
>  	ggtt->invalidate(ggtt);
>  }
> @@ -185,7 +192,7 @@ static void gen8_ggtt_insert_entries(struct i915_address_space *vm,
>  	struct i915_ggtt *ggtt = i915_vm_to_ggtt(vm);
>  	struct sgt_iter sgt_iter;
>  	gen8_pte_t __iomem *gtt_entries;
> -	const gen8_pte_t pte_encode = gen8_pte_encode(0, level, 0);
> +	const gen8_pte_t pte_encode = gen8_ggtt_pte_encode(0, level, 0);
>  	dma_addr_t addr;
>  
>  	/*
> @@ -857,7 +864,7 @@ static int gen8_gmch_probe(struct i915_ggtt *ggtt)
>  	ggtt->vm.vma_ops.set_pages   = ggtt_set_pages;
>  	ggtt->vm.vma_ops.clear_pages = clear_pages;
>  
> -	ggtt->vm.pte_encode = gen8_pte_encode;
> +	ggtt->vm.pte_encode = gen8_ggtt_pte_encode;
>  
>  	setup_private_pat(ggtt->vm.gt->uncore);
>  
> diff --git a/drivers/gpu/drm/i915/gt/intel_gtt.c b/drivers/gpu/drm/i915/gt/intel_gtt.c
> index bb9a6e638175..c8db4f95c1b7 100644
> --- a/drivers/gpu/drm/i915/gt/intel_gtt.c
> +++ b/drivers/gpu/drm/i915/gt/intel_gtt.c
> @@ -484,30 +484,6 @@ void gtt_write_workarounds(struct intel_gt *gt)
>  	}
>  }
>  
> -u64 gen8_pte_encode(dma_addr_t addr,
> -		    enum i915_cache_level level,
> -		    u32 flags)
> -{
> -	gen8_pte_t pte = addr | _PAGE_PRESENT | _PAGE_RW;
> -
> -	if (unlikely(flags & PTE_READ_ONLY))
> -		pte &= ~_PAGE_RW;
> -
> -	switch (level) {
> -	case I915_CACHE_NONE:
> -		pte |= PPAT_UNCACHED;
> -		break;
> -	case I915_CACHE_WT:
> -		pte |= PPAT_DISPLAY_ELLC;
> -		break;
> -	default:
> -		pte |= PPAT_CACHED;
> -		break;
> -	}
> -
> -	return pte;
> -}
> -
>  static void tgl_setup_private_ppat(struct intel_uncore *uncore)
>  {
>  	/* TGL doesn't support LLC or AGE settings */
> diff --git a/drivers/gpu/drm/i915/gt/intel_gtt.h b/drivers/gpu/drm/i915/gt/intel_gtt.h
> index 23004445806a..9a185f4537e1 100644
> --- a/drivers/gpu/drm/i915/gt/intel_gtt.h
> +++ b/drivers/gpu/drm/i915/gt/intel_gtt.h
> @@ -515,10 +515,6 @@ struct i915_ppgtt *i915_ppgtt_create(struct intel_gt *gt);
>  void i915_ggtt_suspend(struct i915_ggtt *gtt);
>  void i915_ggtt_resume(struct i915_ggtt *ggtt);
>  
> -u64 gen8_pte_encode(dma_addr_t addr,
> -		    enum i915_cache_level level,
> -		    u32 flags);
> -
>  int setup_page_dma(struct i915_address_space *vm, struct i915_page_dma *p);
>  void cleanup_page_dma(struct i915_address_space *vm, struct i915_page_dma *p);
>  
> -- 
> 2.24.1


More information about the Intel-gfx mailing list