drm/i915: use i915_sg_dma_sizes() for all backends
authorMatthew Auld <matthew.auld@intel.com>
Tue, 8 Nov 2022 10:32:38 +0000 (10:32 +0000)
committerMatthew Auld <matthew.auld@intel.com>
Wed, 9 Nov 2022 10:07:22 +0000 (10:07 +0000)
We rely on page_sizes.sg in setup_scratch_page() reporting the correct
value if the underlying sgl is not contiguous, however in
get_pages_internal() we are only looking at the layout of the created
pages when calculating the sg_page_sizes, and not the final sgl, which
could in theory be completely different. In such a situation we might
incorrectly think we have a 64K scratch page, when it is actually only
4K or similar split over multiple non-contiguous entries, which could
lead to broken behaviour when touching the scratch space within the
padding of a 64K GTT page-table. For most of the other backends we
already just call i915_sg_dma_sizes() on the final mapping, so rather
just move that into __i915_gem_object_set_pages() to avoid such issues
coming back to bite us later.

v2: Update missing conversion in gvt

Suggested-by: Tvrtko Ursulin <tvrtko.ursulin@linux.intel.com>
Signed-off-by: Matthew Auld <matthew.auld@intel.com>
Cc: Stuart Summers <stuart.summers@intel.com>
Cc: Andrzej Hajda <andrzej.hajda@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20221108103238.165447-1-matthew.auld@intel.com
14 files changed:
drivers/gpu/drm/i915/gem/i915_gem_dmabuf.c
drivers/gpu/drm/i915/gem/i915_gem_internal.c
drivers/gpu/drm/i915/gem/i915_gem_object.h
drivers/gpu/drm/i915/gem/i915_gem_pages.c
drivers/gpu/drm/i915/gem/i915_gem_phys.c
drivers/gpu/drm/i915/gem/i915_gem_shmem.c
drivers/gpu/drm/i915/gem/i915_gem_stolen.c
drivers/gpu/drm/i915/gem/i915_gem_ttm.c
drivers/gpu/drm/i915/gem/i915_gem_userptr.c
drivers/gpu/drm/i915/gem/selftests/huge_gem_object.c
drivers/gpu/drm/i915/gem/selftests/huge_pages.c
drivers/gpu/drm/i915/gvt/dmabuf.c
drivers/gpu/drm/i915/selftests/i915_gem_gtt.c
drivers/gpu/drm/i915/selftests/mock_region.c

index ec6f7ae47783aceaf7450e6c6dc7b19a4519435b..1df74f7aa3dcbdda1b5dbff0df8dc07e3fa58a5b 100644 (file)
@@ -238,7 +238,6 @@ static int i915_gem_object_get_pages_dmabuf(struct drm_i915_gem_object *obj)
 {
        struct drm_i915_private *i915 = to_i915(obj->base.dev);
        struct sg_table *sgt;
-       unsigned int sg_page_sizes;
 
        assert_object_held(obj);
 
@@ -262,8 +261,7 @@ static int i915_gem_object_get_pages_dmabuf(struct drm_i915_gem_object *obj)
            (!HAS_LLC(i915) && !IS_DG1(i915)))
                wbinvd_on_all_cpus();
 
-       sg_page_sizes = i915_sg_dma_sizes(sgt->sgl);
-       __i915_gem_object_set_pages(obj, sgt, sg_page_sizes);
+       __i915_gem_object_set_pages(obj, sgt);
 
        return 0;
 }
index 629acb403a2c975971a1406822b75c7e38961b75..f66bcefc09ec003f1232fab5a4e28f40b1b76024 100644 (file)
@@ -35,7 +35,6 @@ static int i915_gem_object_get_pages_internal(struct drm_i915_gem_object *obj)
        struct drm_i915_private *i915 = to_i915(obj->base.dev);
        struct sg_table *st;
        struct scatterlist *sg;
-       unsigned int sg_page_sizes;
        unsigned int npages;
        int max_order = MAX_ORDER;
        unsigned int max_segment;
@@ -64,7 +63,6 @@ create_st:
 
        sg = st->sgl;
        st->nents = 0;
-       sg_page_sizes = 0;
 
        do {
                int order = min(fls(npages) - 1, max_order);
@@ -83,7 +81,6 @@ create_st:
                } while (1);
 
                sg_set_page(sg, page, PAGE_SIZE << order, 0);
-               sg_page_sizes |= PAGE_SIZE << order;
                st->nents++;
 
                npages -= 1 << order;
@@ -105,7 +102,7 @@ create_st:
                goto err;
        }
 
-       __i915_gem_object_set_pages(obj, st, sg_page_sizes);
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 
index 6b9ecff42bb5cae980dd1993180f5967738202e6..3db53769864c243ecc3a00af63b4eaf7b51e4aea 100644 (file)
@@ -403,8 +403,7 @@ i915_gem_object_get_dma_address(struct drm_i915_gem_object *obj,
                                unsigned long n);
 
 void __i915_gem_object_set_pages(struct drm_i915_gem_object *obj,
-                                struct sg_table *pages,
-                                unsigned int sg_page_sizes);
+                                struct sg_table *pages);
 
 int ____i915_gem_object_get_pages(struct drm_i915_gem_object *obj);
 int __i915_gem_object_get_pages(struct drm_i915_gem_object *obj);
index 16f845663ff2cf1a709b4e3c8daea0eba45381eb..05a27723ebb8cb0ad24fc8ba95b711a73309cdd5 100644 (file)
@@ -16,8 +16,7 @@
 #include "i915_gem_mman.h"
 
 void __i915_gem_object_set_pages(struct drm_i915_gem_object *obj,
-                                struct sg_table *pages,
-                                unsigned int sg_page_sizes)
+                                struct sg_table *pages)
 {
        struct drm_i915_private *i915 = to_i915(obj->base.dev);
        unsigned long supported = RUNTIME_INFO(i915)->page_sizes;
@@ -45,8 +44,8 @@ void __i915_gem_object_set_pages(struct drm_i915_gem_object *obj,
 
        obj->mm.pages = pages;
 
-       GEM_BUG_ON(!sg_page_sizes);
-       obj->mm.page_sizes.phys = sg_page_sizes;
+       obj->mm.page_sizes.phys = i915_sg_dma_sizes(pages->sgl);
+       GEM_BUG_ON(!obj->mm.page_sizes.phys);
 
        /*
         * Calculate the supported page-sizes which fit into the given
index 0d0e46dae559c868a16cf637f51086f4544e0697..68453572275baf048564b8c21e4facb7e0ef5ba4 100644 (file)
@@ -79,7 +79,7 @@ static int i915_gem_object_get_pages_phys(struct drm_i915_gem_object *obj)
 
        /* We're no longer struct page backed */
        obj->mem_flags &= ~I915_BO_FLAG_STRUCT_PAGE;
-       __i915_gem_object_set_pages(obj, st, sg->length);
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 
@@ -209,11 +209,8 @@ static int i915_gem_object_shmem_to_phys(struct drm_i915_gem_object *obj)
        return 0;
 
 err_xfer:
-       if (!IS_ERR_OR_NULL(pages)) {
-               unsigned int sg_page_sizes = i915_sg_dma_sizes(pages->sgl);
-
-               __i915_gem_object_set_pages(obj, pages, sg_page_sizes);
-       }
+       if (!IS_ERR_OR_NULL(pages))
+               __i915_gem_object_set_pages(obj, pages);
        return err;
 }
 
index 2f7804492cd5cb6df70c284d9ec0e033af4f9829..9c759df700caaaf159cd7d3035d02fd3f2db0b4a 100644 (file)
@@ -247,7 +247,7 @@ rebuild_st:
        if (i915_gem_object_can_bypass_llc(obj))
                obj->cache_dirty = true;
 
-       __i915_gem_object_set_pages(obj, st, i915_sg_dma_sizes(st->sgl));
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 
index 0c70711818edd2354e580aaeb9c46b2503697ae3..bc9521078807340ad20bba08d378dd66d17bf958 100644 (file)
@@ -628,7 +628,7 @@ static int i915_gem_object_get_pages_stolen(struct drm_i915_gem_object *obj)
                   sg_dma_len(pages->sgl),
                   POISON_INUSE);
 
-       __i915_gem_object_set_pages(obj, pages, obj->stolen->size);
+       __i915_gem_object_set_pages(obj, pages);
 
        return 0;
 }
index 2c8b2d5ae90359b733ec218c08f3f0256c1417c7..e4e55e3f4e41181ae485c9d4516fef778adfb015 100644 (file)
@@ -815,8 +815,7 @@ static int __i915_ttm_get_pages(struct drm_i915_gem_object *obj,
 
                GEM_BUG_ON(obj->mm.rsgt);
                obj->mm.rsgt = rsgt;
-               __i915_gem_object_set_pages(obj, &rsgt->table,
-                                           i915_sg_dma_sizes(rsgt->table.sgl));
+               __i915_gem_object_set_pages(obj, &rsgt->table);
        }
 
        GEM_BUG_ON(bo->ttm && ((obj->base.size >> PAGE_SHIFT) < bo->ttm->num_pages));
index ca7a388ba2bf5512e54b5477da36d5e0b888eb9a..9348b1804d53ce1aefe6764a969734496c257f7e 100644 (file)
@@ -131,7 +131,6 @@ static int i915_gem_userptr_get_pages(struct drm_i915_gem_object *obj)
        const unsigned long num_pages = obj->base.size >> PAGE_SHIFT;
        unsigned int max_segment = i915_sg_segment_size(obj->base.dev->dev);
        struct sg_table *st;
-       unsigned int sg_page_sizes;
        struct page **pvec;
        int ret;
 
@@ -170,8 +169,7 @@ alloc_table:
        if (i915_gem_object_can_bypass_llc(obj))
                obj->cache_dirty = true;
 
-       sg_page_sizes = i915_sg_dma_sizes(st->sgl);
-       __i915_gem_object_set_pages(obj, st, sg_page_sizes);
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 
index f963b8e1e37b5d984ed1500bbb4607f17ab3492d..cbd9b624a78882abdb76c9d194ef850802ca257a 100644 (file)
@@ -68,7 +68,7 @@ static int huge_get_pages(struct drm_i915_gem_object *obj)
        if (i915_gem_gtt_prepare_pages(obj, pages))
                goto err;
 
-       __i915_gem_object_set_pages(obj, pages, PAGE_SIZE);
+       __i915_gem_object_set_pages(obj, pages);
 
        return 0;
 
index 0cb99e75b0bc294e08ade9b4ee132c4b24d724b8..beaf27e09e8a9d6fdebe9046d242a303d43d6deb 100644 (file)
@@ -136,7 +136,7 @@ static int get_huge_pages(struct drm_i915_gem_object *obj)
                goto err;
 
        GEM_BUG_ON(sg_page_sizes != obj->mm.page_mask);
-       __i915_gem_object_set_pages(obj, st, sg_page_sizes);
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 
@@ -210,7 +210,6 @@ static int fake_get_huge_pages(struct drm_i915_gem_object *obj)
        const u64 max_len = rounddown_pow_of_two(UINT_MAX);
        struct sg_table *st;
        struct scatterlist *sg;
-       unsigned int sg_page_sizes;
        u64 rem;
 
        st = kmalloc(sizeof(*st), GFP);
@@ -226,7 +225,6 @@ static int fake_get_huge_pages(struct drm_i915_gem_object *obj)
        rem = obj->base.size;
        sg = st->sgl;
        st->nents = 0;
-       sg_page_sizes = 0;
        do {
                unsigned int page_size = get_largest_page_size(i915, rem);
                unsigned int len = min(page_size * div_u64(rem, page_size),
@@ -239,8 +237,6 @@ static int fake_get_huge_pages(struct drm_i915_gem_object *obj)
                sg_dma_len(sg) = len;
                sg_dma_address(sg) = page_size;
 
-               sg_page_sizes |= len;
-
                st->nents++;
 
                rem -= len;
@@ -254,7 +250,7 @@ static int fake_get_huge_pages(struct drm_i915_gem_object *obj)
 
        i915_sg_trim(st);
 
-       __i915_gem_object_set_pages(obj, st, sg_page_sizes);
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 }
@@ -286,7 +282,7 @@ static int fake_get_huge_pages_single(struct drm_i915_gem_object *obj)
        sg_dma_len(sg) = obj->base.size;
        sg_dma_address(sg) = page_size;
 
-       __i915_gem_object_set_pages(obj, st, sg->length);
+       __i915_gem_object_set_pages(obj, st);
 
        return 0;
 #undef GFP
index 01e54b45c5c1b5cff646d70f420dfbefe0692af9..355f1c0e86641730b09a2851b8172325087b360d 100644 (file)
@@ -88,7 +88,7 @@ static int vgpu_gem_get_pages(
                sg_dma_address(sg) = dma_addr;
        }
 
-       __i915_gem_object_set_pages(obj, st, PAGE_SIZE);
+       __i915_gem_object_set_pages(obj, st);
 out:
        if (ret) {
                dma_addr_t dma_addr;
index 27c733b0097628bcecbc6f4ebc2f1ae7f688492c..eae7d947d7de28a3e2132ee5dfc299dcfbf106b8 100644 (file)
@@ -61,7 +61,6 @@ static int fake_get_pages(struct drm_i915_gem_object *obj)
 #define PFN_BIAS 0x1000
        struct sg_table *pages;
        struct scatterlist *sg;
-       unsigned int sg_page_sizes;
        typeof(obj->base.size) rem;
 
        pages = kmalloc(sizeof(*pages), GFP);
@@ -74,7 +73,6 @@ static int fake_get_pages(struct drm_i915_gem_object *obj)
                return -ENOMEM;
        }
 
-       sg_page_sizes = 0;
        rem = obj->base.size;
        for (sg = pages->sgl; sg; sg = sg_next(sg)) {
                unsigned long len = min_t(typeof(rem), rem, BIT(31));
@@ -83,13 +81,12 @@ static int fake_get_pages(struct drm_i915_gem_object *obj)
                sg_set_page(sg, pfn_to_page(PFN_BIAS), len, 0);
                sg_dma_address(sg) = page_to_phys(sg_page(sg));
                sg_dma_len(sg) = len;
-               sg_page_sizes |= len;
 
                rem -= len;
        }
        GEM_BUG_ON(rem);
 
-       __i915_gem_object_set_pages(obj, pages, sg_page_sizes);
+       __i915_gem_object_set_pages(obj, pages);
 
        return 0;
 #undef GFP
index bac21fe84ca5685dd91480e6bbabd54e4c55aeb5..6324eb32d4ddcb0a47043c540e5ee10a4b08d75c 100644 (file)
@@ -41,7 +41,7 @@ static int mock_region_get_pages(struct drm_i915_gem_object *obj)
        }
 
        pages = &obj->mm.rsgt->table;
-       __i915_gem_object_set_pages(obj, pages, i915_sg_dma_sizes(pages->sgl));
+       __i915_gem_object_set_pages(obj, pages);
 
        return 0;