]> git.proxmox.com Git - mirror_ubuntu-kernels.git/commitdiff
drm/i915: Sleep around performing iommu unmaps on Tigerlake
authorChris Wilson <chris@chris-wilson.co.uk>
Wed, 9 Dec 2020 16:40:07 +0000 (16:40 +0000)
committerChris Wilson <chris@chris-wilson.co.uk>
Wed, 9 Dec 2020 19:43:18 +0000 (19:43 +0000)
Tigerlake is plagued by spontaneous DMAR faults [reason 7, next page
table ptr is invalid] which lead to GPU hangs. These faults occur when
an iommu map is immediately reused. Adding further clflushes and
barriers around either the GTT PTE or iommu PTE updates do not prevent
the faults. So far the only effect has been from inducing a delay
between reuse of the iommu on the GPU, and applying the delay at the
iommu map allows for the smallest stable delay.

Note that such a delay is hideous and clearly does not fix the root cause,
and so should only be a bandaid until a complete solution is found. The
delay was determined by running igt/gem_exec_fence/parallel in a loop for
a few hours (unpatched MTBF is about 10s).

We have also seen such DMAR fault [reason 7] errors on other platforms,
notably gen9-gen11, but so far it has only been trivially and
consistently reproduced on Tigerlake.

v2: Leave a tell-tale to know when we apply the vt'd quirk, and as a
reminder to remove it again. Hopefully.

Testcase: igt/gem_exec_fence/parallel
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Acked-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20201209164008.5487-2-chris@chris-wilson.co.uk
drivers/gpu/drm/i915/gt/intel_ggtt.c

index cf94525be2c18c5f19abe623c6143810b83d5182..eece0844fbe974ccaba6ea10a69cefd95264a60a 100644 (file)
@@ -101,7 +101,16 @@ static bool needs_idle_maps(struct drm_i915_private *i915)
         * Query intel_iommu to see if we need the workaround. Presumably that
         * was loaded first.
         */
-       return IS_GEN(i915, 5) && IS_MOBILE(i915) && intel_vtd_active();
+       if (!intel_vtd_active())
+               return false;
+
+       if (IS_GEN(i915, 5) && IS_MOBILE(i915))
+               return true;
+
+       if (IS_GEN(i915, 12))
+               return true; /* XXX DMAR fault reason 7 */
+
+       return false;
 }
 
 void i915_ggtt_suspend(struct i915_ggtt *ggtt)
@@ -1050,7 +1059,12 @@ static int i915_gmch_probe(struct i915_ggtt *ggtt)
 
        ggtt->vm.alloc_pt_dma = alloc_pt_dma;
 
-       ggtt->do_idle_maps = needs_idle_maps(i915);
+       if (needs_idle_maps(i915)) {
+               drm_notice(&i915->drm,
+                          "Flushing DMA requests before IOMMU unmaps; performance may be degraded\n");
+               ggtt->do_idle_maps = true;
+       }
+
        ggtt->vm.insert_page = i915_ggtt_insert_page;
        ggtt->vm.insert_entries = i915_ggtt_insert_entries;
        ggtt->vm.clear_range = i915_ggtt_clear_range;