[PATCH v2 1/2] drm/i915: enable PS64 support for DG2
Das, Nirmoy
nirmoy.das at linux.intel.com
Wed Sep 28 11:00:27 UTC 2022
On 9/27/2022 5:39 PM, Matthew Auld wrote:
> It turns out that on production DG2/ATS HW we should have support for
> PS64. This feature allows to provide a 64K TLB hint at the PTE level,
> which is a lot more flexible than the current method of enabling 64K GTT
> pages for the entire page-table, since that leads to all kinds of
> annoying restrictions, as documented in:
>
> commit caa574ffc4aaf4f29b890223878c63e2e7772f62
> Author: Matthew Auld <matthew.auld at intel.com>
> Date: Sat Feb 19 00:17:49 2022 +0530
>
> drm/i915/uapi: document behaviour for DG2 64K support
>
> On discrete platforms like DG2, we need to support a minimum page size
> of 64K when dealing with device local-memory. This is quite tricky for
> various reasons, so try to document the new implicit uapi for this.
>
> With PS64, we can now drop the 2M GTT alignment restriction, and instead
> only require 64K or larger when dealing with lmem. We still use the
> compact-pt layout when possible, but only when we are certain that this
> doesn't interfere with userspace.
>
> Note that this is a change in uAPI behaviour, but hopefully shouldn't be
> a concern (IGT is at least able to autodetect the alignment), since we
> are only making the GTT alignment constraint less restrictive.
>
> Based on a patch from CQ Tang.
>
> v2: update the comment wrt scratch page
>
> Reported-by: Michal Mrozek <michal.mrozek at intel.com>
> Signed-off-by: Matthew Auld <matthew.auld at intel.com>
> Cc: Lionel Landwerlin <lionel.g.landwerlin at intel.com>
> Cc: Thomas Hellström <thomas.hellstrom at linux.intel.com>
> Cc: Stuart Summers <stuart.summers at intel.com>
> Cc: Jordan Justen <jordan.l.justen at intel.com>
> Cc: Yang A Shi <yang.a.shi at intel.com>
> Cc: Nirmoy Das <nirmoy.das at intel.com>
> ---
> .../gpu/drm/i915/gem/selftests/huge_pages.c | 159 +++++++++++++++++-
> drivers/gpu/drm/i915/gt/gen8_ppgtt.c | 81 +++++----
> drivers/gpu/drm/i915/gt/intel_gtt.c | 21 +--
> drivers/gpu/drm/i915/gt/intel_gtt.h | 1 +
> drivers/gpu/drm/i915/i915_drv.h | 7 -
> drivers/gpu/drm/i915/i915_pci.c | 2 -
> drivers/gpu/drm/i915/i915_vma.c | 9 +-
> drivers/gpu/drm/i915/intel_device_info.h | 1 -
> drivers/gpu/drm/i915/selftests/i915_gem_gtt.c | 9 +-
> include/uapi/drm/i915_drm.h | 36 ++--
> 10 files changed, 220 insertions(+), 106 deletions(-)
>
> diff --git a/drivers/gpu/drm/i915/gem/selftests/huge_pages.c b/drivers/gpu/drm/i915/gem/selftests/huge_pages.c
> index c570cf780079..cc26c1293208 100644
> --- a/drivers/gpu/drm/i915/gem/selftests/huge_pages.c
> +++ b/drivers/gpu/drm/i915/gem/selftests/huge_pages.c
> @@ -1161,7 +1161,8 @@ static int igt_write_huge(struct drm_i915_private *i915,
> GEM_BUG_ON(!i915_gem_object_has_pinned_pages(obj));
>
> size = obj->base.size;
> - if (obj->mm.page_sizes.sg & I915_GTT_PAGE_SIZE_64K)
> + if (obj->mm.page_sizes.sg & I915_GTT_PAGE_SIZE_64K &&
> + !HAS_64K_PAGES(i915))
> size = round_up(size, I915_GTT_PAGE_SIZE_2M);
>
> n = 0;
> @@ -1214,6 +1215,10 @@ static int igt_write_huge(struct drm_i915_private *i915,
> * size and ensure the vma offset is at the start of the pt
> * boundary, however to improve coverage we opt for testing both
> * aligned and unaligned offsets.
> + *
> + * With PS64 this is no longer the case, but to ensure we
> + * sometimes get the compact layout for smaller objects, apply
> + * the round_up anyway.
> */
> if (obj->mm.page_sizes.sg & I915_GTT_PAGE_SIZE_64K)
> offset_low = round_down(offset_low,
> @@ -1411,6 +1416,7 @@ static int igt_ppgtt_sanity_check(void *arg)
> { SZ_2M + SZ_4K, SZ_64K | SZ_4K },
> { SZ_2M + SZ_4K, SZ_2M | SZ_4K },
> { SZ_2M + SZ_64K, SZ_2M | SZ_64K },
> + { SZ_2M + SZ_64K, SZ_64K },
> };
> int i, j;
> int err;
> @@ -1540,6 +1546,156 @@ static int igt_ppgtt_compact(void *arg)
> return err;
> }
>
> +static int igt_ppgtt_mixed(void *arg)
> +{
> + struct drm_i915_private *i915 = arg;
> + const unsigned long flags = PIN_OFFSET_FIXED | PIN_USER;
> + struct drm_i915_gem_object *obj, *on;
> + struct i915_gem_engines *engines;
> + struct i915_gem_engines_iter it;
> + struct i915_address_space *vm;
> + struct i915_gem_context *ctx;
> + struct intel_context *ce;
> + struct file *file;
> + I915_RND_STATE(prng);
> + LIST_HEAD(objects);
> + struct intel_memory_region *mr;
> + struct i915_vma *vma;
> + unsigned int count;
> + u32 i, rem, addr;
> + int *order;
> + int n, err;
> +
> + /*
> + * Sanity check mixing 4K and 64K pages within the same page-table via
> + * the new PS64 TLB hint.
> + */
> +
> + if (!HAS_64K_PAGES(i915)) {
> + pr_info("device lacks PS64, skipping\n");
> + return 0;
> + }
> +
> + file = mock_file(i915);
> + if (IS_ERR(file))
> + return PTR_ERR(file);
> +
> + ctx = hugepage_ctx(i915, file);
> + if (IS_ERR(ctx)) {
> + err = PTR_ERR(ctx);
> + goto out;
> + }
> + vm = i915_gem_context_get_eb_vm(ctx);
> +
> + i = 0;
> + rem = SZ_8M;
> + addr = 0;
> + do {
> + u32 sz;
> +
> + sz = i915_prandom_u32_max_state(min_t(u32, rem, SZ_4M), &prng);
> + sz = min_t(u32, rem, SZ_4K);
sz is not using the random size generated previously.
> +
> + mr = i915->mm.regions[INTEL_REGION_LMEM_0];
> + if (i & 1)
> + mr = i915->mm.regions[INTEL_REGION_SMEM];
> +
> + obj = i915_gem_object_create_region(mr, sz, 0, 0);
> + if (IS_ERR(obj)) {
> + err = PTR_ERR(obj);
> + goto out_vm;
> + }
> +
> + list_add_tail(&obj->st_link, &objects);
> +
> + vma = i915_vma_instance(obj, vm, NULL);
> + if (IS_ERR(vma)) {
> + err = PTR_ERR(vma);
> + goto err_put;
> + }
> +
> + addr = round_up(addr, mr->min_page_size);
> + err = i915_vma_pin(vma, 0, 0, addr | flags);
> + if (err)
> + goto err_put;
> +
> + if (mr->type == INTEL_MEMORY_LOCAL &&
> + (vma->resource->page_sizes_gtt & I915_GTT_PAGE_SIZE_4K)) {
> + err = -EINVAL;
> + goto err_put;
> + }
> +
> + addr += obj->base.size;
> + rem -= sz;
> + i++;
> + } while (rem);
> +
> + n = 0;
> + count = 0;
> + for_each_gem_engine(ce, i915_gem_context_lock_engines(ctx), it) {
> + count++;
> + if (!intel_engine_can_store_dword(ce->engine))
> + continue;
> +
> + n++;
> + }
> + i915_gem_context_unlock_engines(ctx);
> + if (!n)
> + goto err_put;
> +
> + order = i915_random_order(count * count, &prng);
> + if (!order) {
> + err = -ENOMEM;
> + goto err_put;
> + }
> +
> + i = 0;
> + addr = 0;
> + engines = i915_gem_context_lock_engines(ctx);
> + list_for_each_entry(obj, &objects, st_link) {
> + u32 rnd = i915_prandom_u32_max_state(UINT_MAX, &prng);
> +
> + addr = round_up(addr, obj->mm.region->min_page_size);
> +
> + ce = engines->engines[order[i] % engines->num_engines];
> + i = (i + 1) % (count * count);
> + if (!ce || !intel_engine_can_store_dword(ce->engine))
> + continue;
> +
> + err = __igt_write_huge(ce, obj, obj->base.size, addr, 0, rnd);
> + if (err)
> + break;
> +
> + err = __igt_write_huge(ce, obj, obj->base.size, addr,
> + offset_in_page(rnd) / sizeof(u32), rnd + 1);
> + if (err)
> + break;
> +
> + err = __igt_write_huge(ce, obj, obj->base.size, addr,
> + (PAGE_SIZE / sizeof(u32)) - 1,
> + rnd + 2);
> + if (err)
> + break;
> +
> + addr += obj->base.size;
> +
> + cond_resched();
> + }
> +
> + i915_gem_context_unlock_engines(ctx);
> + kfree(order);
> +err_put:
> + list_for_each_entry_safe(obj, on, &objects, st_link) {
> + list_del(&obj->st_link);
> + i915_gem_object_put(obj);
> + }
> +out_vm:
> + i915_vm_put(vm);
> +out:
> + fput(file);
> + return err;
> +}
> +
> static int igt_tmpfs_fallback(void *arg)
> {
> struct drm_i915_private *i915 = arg;
> @@ -1803,6 +1959,7 @@ int i915_gem_huge_page_live_selftests(struct drm_i915_private *i915)
> SUBTEST(igt_ppgtt_smoke_huge),
> SUBTEST(igt_ppgtt_sanity_check),
> SUBTEST(igt_ppgtt_compact),
> + SUBTEST(igt_ppgtt_mixed),
> };
>
> if (!HAS_PPGTT(i915)) {
> diff --git a/drivers/gpu/drm/i915/gt/gen8_ppgtt.c b/drivers/gpu/drm/i915/gt/gen8_ppgtt.c
> index 8dd60d5ef905..4daaa6f55668 100644
> --- a/drivers/gpu/drm/i915/gt/gen8_ppgtt.c
> +++ b/drivers/gpu/drm/i915/gt/gen8_ppgtt.c
> @@ -476,6 +476,7 @@ xehpsdv_ppgtt_insert_huge(struct i915_address_space *vm,
> const gen8_pte_t pte_encode = vm->pte_encode(0, cache_level, flags);
> unsigned int rem = sg_dma_len(iter->sg);
> u64 start = vma_res->start;
> + u64 end = start + vma_res->vma_size;
>
> GEM_BUG_ON(!i915_vm_is_4lvl(vm));
>
> @@ -489,9 +490,10 @@ xehpsdv_ppgtt_insert_huge(struct i915_address_space *vm,
> gen8_pte_t encode = pte_encode;
> unsigned int page_size;
> gen8_pte_t *vaddr;
> - u16 index, max;
> + u16 index, max, nent, i;
>
> max = I915_PDES;
> + nent = 1;
>
> if (vma_res->bi.page_sizes.sg & I915_GTT_PAGE_SIZE_2M &&
> IS_ALIGNED(iter->dma, I915_GTT_PAGE_SIZE_2M) &&
> @@ -503,25 +505,37 @@ xehpsdv_ppgtt_insert_huge(struct i915_address_space *vm,
>
> vaddr = px_vaddr(pd);
> } else {
> - if (encode & GEN12_PPGTT_PTE_LM) {
> - GEM_BUG_ON(__gen8_pte_index(start, 0) % 16);
> - GEM_BUG_ON(rem < I915_GTT_PAGE_SIZE_64K);
> - GEM_BUG_ON(!IS_ALIGNED(iter->dma,
> - I915_GTT_PAGE_SIZE_64K));
> -
> - index = __gen8_pte_index(start, 0) / 16;
> - page_size = I915_GTT_PAGE_SIZE_64K;
> -
> - max /= 16;
> -
> - vaddr = px_vaddr(pd);
> - vaddr[__gen8_pte_index(start, 1)] |= GEN12_PDE_64K;
> + index = __gen8_pte_index(start, 0);
> + page_size = I915_GTT_PAGE_SIZE;
>
> - pt->is_compact = true;
> - } else {
> - GEM_BUG_ON(pt->is_compact);
> - index = __gen8_pte_index(start, 0);
> - page_size = I915_GTT_PAGE_SIZE;
> + if (vma_res->bi.page_sizes.sg & I915_GTT_PAGE_SIZE_64K) {
> + /*
> + * Device local-memory on these platforms should
> + * always use 64K pages or larger (including GTT
> + * alignment), therefore if we know the whole
> + * page-table needs to be filled we can always
> + * safely use the compact-layout. Otherwise fall
> + * back to the TLB hint with PS64. If this is
> + * system memory we only bother with PS64.
> + */
> + if ((encode & GEN12_PPGTT_PTE_LM) &&
> + end - start >= SZ_2M && !index) {
> + index = __gen8_pte_index(start, 0) / 16;
> + page_size = I915_GTT_PAGE_SIZE_64K;
> +
> + max /= 16;
> +
> + vaddr = px_vaddr(pd);
> + vaddr[__gen8_pte_index(start, 1)] |= GEN12_PDE_64K;
> +
> + pt->is_compact = true;
> + } else if (IS_ALIGNED(iter->dma, I915_GTT_PAGE_SIZE_64K) &&
> + rem >= I915_GTT_PAGE_SIZE_64K &&
> + !(index % 16)) {
> + encode |= GEN12_PTE_PS64;
> + page_size = I915_GTT_PAGE_SIZE_64K;
> + nent = 16;
> + }
> }
>
> vaddr = px_vaddr(pt);
> @@ -529,7 +543,12 @@ xehpsdv_ppgtt_insert_huge(struct i915_address_space *vm,
>
> do {
> GEM_BUG_ON(rem < page_size);
> - vaddr[index++] = encode | iter->dma;
> +
> + for (i = 0; i < nent; i++) {
> + vaddr[index++] =
> + encode | (iter->dma + i *
> + I915_GTT_PAGE_SIZE);
> + }
>
> start += page_size;
> iter->dma += page_size;
> @@ -745,6 +764,8 @@ static void __xehpsdv_ppgtt_insert_entry_lm(struct i915_address_space *vm,
> GEM_BUG_ON(!IS_ALIGNED(addr, SZ_64K));
> GEM_BUG_ON(!IS_ALIGNED(offset, SZ_64K));
>
> + /* XXX: we don't strictly need to use this layout */
> +
> if (!pt->is_compact) {
> vaddr = px_vaddr(pd);
> vaddr[gen8_pd_index(idx, 1)] |= GEN12_PDE_64K;
> @@ -935,22 +956,10 @@ struct i915_ppgtt *gen8_ppgtt_create(struct intel_gt *gt,
> ppgtt->vm.alloc_pt_dma = alloc_pt_dma;
>
> /*
> - * On some platforms the hw has dropped support for 4K GTT pages
> - * when dealing with LMEM, and due to the design of 64K GTT
> - * pages in the hw, we can only mark the *entire* page-table as
> - * operating in 64K GTT mode, since the enable bit is still on
> - * the pde, and not the pte. And since we still need to allow
> - * 4K GTT pages for SMEM objects, we can't have a "normal" 4K
> - * page-table with scratch pointing to LMEM, since that's
> - * undefined from the hw pov. The simplest solution is to just
> - * move the 64K scratch page to SMEM on all platforms and call
> - * it a day, since that should work for all configurations.
> - *
> - * Using SMEM instead of LMEM has the additional advantage of
> - * not reserving high performance memory for a "never" used
> - * filler page. It also removes the device access that would
> - * be required to initialise the scratch page, reducing pressure
> - * on an even scarcer resource.
> + * Using SMEM here instead of LMEM has the advantage of not reserving
> + * high performance memory for a "never" used filler page. It also
> + * removes the device access that would be required to initialise the
> + * scratch page, reducing pressure on an even scarcer resource.
> */
> ppgtt->vm.alloc_scratch_dma = alloc_pt_dma;
>
> diff --git a/drivers/gpu/drm/i915/gt/intel_gtt.c b/drivers/gpu/drm/i915/gt/intel_gtt.c
> index 2eaeba14319e..13e411187fd5 100644
> --- a/drivers/gpu/drm/i915/gt/intel_gtt.c
> +++ b/drivers/gpu/drm/i915/gt/intel_gtt.c
> @@ -269,11 +269,7 @@ void i915_address_space_init(struct i915_address_space *vm, int subclass)
> memset64(vm->min_alignment, I915_GTT_MIN_ALIGNMENT,
> ARRAY_SIZE(vm->min_alignment));
>
> - if (HAS_64K_PAGES(vm->i915) && NEEDS_COMPACT_PT(vm->i915) &&
> - subclass == VM_CLASS_PPGTT) {
> - vm->min_alignment[INTEL_MEMORY_LOCAL] = I915_GTT_PAGE_SIZE_2M;
> - vm->min_alignment[INTEL_MEMORY_STOLEN_LOCAL] = I915_GTT_PAGE_SIZE_2M;
> - } else if (HAS_64K_PAGES(vm->i915)) {
> + if (HAS_64K_PAGES(vm->i915)) {
> vm->min_alignment[INTEL_MEMORY_LOCAL] = I915_GTT_PAGE_SIZE_64K;
> vm->min_alignment[INTEL_MEMORY_STOLEN_LOCAL] = I915_GTT_PAGE_SIZE_64K;
> }
> @@ -343,7 +339,8 @@ int setup_scratch_page(struct i915_address_space *vm)
> */
> size = I915_GTT_PAGE_SIZE_4K;
> if (i915_vm_is_4lvl(vm) &&
> - HAS_PAGE_SIZES(vm->i915, I915_GTT_PAGE_SIZE_64K))
> + HAS_PAGE_SIZES(vm->i915, I915_GTT_PAGE_SIZE_64K) &&
> + !HAS_64K_PAGES(vm->i915))
> size = I915_GTT_PAGE_SIZE_64K;
>
> do {
> @@ -385,18 +382,6 @@ int setup_scratch_page(struct i915_address_space *vm)
> if (size == I915_GTT_PAGE_SIZE_4K)
> return -ENOMEM;
>
> - /*
> - * If we need 64K minimum GTT pages for device local-memory,
> - * like on XEHPSDV, then we need to fail the allocation here,
> - * otherwise we can't safely support the insertion of
> - * local-memory pages for this vm, since the HW expects the
> - * correct physical alignment and size when the page-table is
> - * operating in 64K GTT mode, which includes any scratch PTEs,
> - * since userspace can still touch them.
> - */
> - if (HAS_64K_PAGES(vm->i915))
> - return -ENOMEM;
> -
> size = I915_GTT_PAGE_SIZE_4K;
> } while (1);
> }
> diff --git a/drivers/gpu/drm/i915/gt/intel_gtt.h b/drivers/gpu/drm/i915/gt/intel_gtt.h
> index c0ca53cba9f0..062b78333fb2 100644
> --- a/drivers/gpu/drm/i915/gt/intel_gtt.h
> +++ b/drivers/gpu/drm/i915/gt/intel_gtt.h
> @@ -93,6 +93,7 @@ typedef u64 gen8_pte_t;
> #define GEN12_GGTT_PTE_LM BIT_ULL(1)
>
> #define GEN12_PDE_64K BIT(6)
> +#define GEN12_PTE_PS64 BIT(8)
>
> /*
> * Cacheability Control is a 4-bit value. The low three bits are stored in bits
> diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h
> index 84a2f6b16f57..eb841daa0d63 100644
> --- a/drivers/gpu/drm/i915/i915_drv.h
> +++ b/drivers/gpu/drm/i915/i915_drv.h
> @@ -902,13 +902,6 @@ IS_SUBPLATFORM(const struct drm_i915_private *i915,
> */
> #define HAS_64K_PAGES(dev_priv) (INTEL_INFO(dev_priv)->has_64k_pages)
>
> -/*
> - * Set this flag when platform doesn't allow both 64k pages and 4k pages in
> - * the same PT. this flag means we need to support compact PT layout for the
> - * ppGTT when using the 64K GTT pages.
> - */
> -#define NEEDS_COMPACT_PT(dev_priv) (INTEL_INFO(dev_priv)->needs_compact_pt)
> -
> #define HAS_IPC(dev_priv) (INTEL_INFO(dev_priv)->display.has_ipc)
>
> #define HAS_REGION(i915, i) (RUNTIME_INFO(i915)->memory_regions & (i))
> diff --git a/drivers/gpu/drm/i915/i915_pci.c b/drivers/gpu/drm/i915/i915_pci.c
> index 38460a0bd7cb..40bb06c5cdc0 100644
> --- a/drivers/gpu/drm/i915/i915_pci.c
> +++ b/drivers/gpu/drm/i915/i915_pci.c
> @@ -1042,7 +1042,6 @@ static const struct intel_device_info xehpsdv_info = {
> PLATFORM(INTEL_XEHPSDV),
> NO_DISPLAY,
> .has_64k_pages = 1,
> - .needs_compact_pt = 1,
> .has_media_ratio_mode = 1,
> .__runtime.platform_engine_mask =
> BIT(RCS0) | BIT(BCS0) |
> @@ -1064,7 +1063,6 @@ static const struct intel_device_info xehpsdv_info = {
> .has_64k_pages = 1, \
> .has_guc_deprivilege = 1, \
> .has_heci_pxp = 1, \
> - .needs_compact_pt = 1, \
> .has_media_ratio_mode = 1, \
> .__runtime.platform_engine_mask = \
> BIT(RCS0) | BIT(BCS0) | \
> diff --git a/drivers/gpu/drm/i915/i915_vma.c b/drivers/gpu/drm/i915/i915_vma.c
> index f17c09ead7d7..c39488eb9eeb 100644
> --- a/drivers/gpu/drm/i915/i915_vma.c
> +++ b/drivers/gpu/drm/i915/i915_vma.c
> @@ -776,12 +776,6 @@ i915_vma_insert(struct i915_vma *vma, struct i915_gem_ww_ctx *ww,
> GEM_BUG_ON(!IS_ALIGNED(end, I915_GTT_PAGE_SIZE));
>
> alignment = max(alignment, i915_vm_obj_min_alignment(vma->vm, vma->obj));
> - /*
> - * for compact-pt we round up the reservation to prevent
> - * any smaller pages being used within the same PDE
> - */
> - if (NEEDS_COMPACT_PT(vma->vm->i915))
> - size = round_up(size, alignment);
>
> /* If binding the object/GGTT view requires more space than the entire
> * aperture has, reject it early before evicting everything in a vain
> @@ -820,7 +814,8 @@ i915_vma_insert(struct i915_vma *vma, struct i915_gem_ww_ctx *ww,
> * forseeable future. See also i915_ggtt_offset().
> */
> if (upper_32_bits(end - 1) &&
> - vma->page_sizes.sg > I915_GTT_PAGE_SIZE) {
> + vma->page_sizes.sg > I915_GTT_PAGE_SIZE &&
> + !HAS_64K_PAGES(vma->vm->i915)) {
> /*
> * We can't mix 64K and 4K PTEs in the same page-table
> * (2M block), and so to avoid the ugliness and
> diff --git a/drivers/gpu/drm/i915/intel_device_info.h b/drivers/gpu/drm/i915/intel_device_info.h
> index bc87d3156b14..58539660a1ec 100644
> --- a/drivers/gpu/drm/i915/intel_device_info.h
> +++ b/drivers/gpu/drm/i915/intel_device_info.h
> @@ -146,7 +146,6 @@ enum intel_ppgtt_type {
> /* Keep has_* in alphabetical order */ \
> func(has_64bit_reloc); \
> func(has_64k_pages); \
> - func(needs_compact_pt); \
> func(gpu_reset_clobbers_display); \
> func(has_reset_engine); \
> func(has_3d_pipeline); \
> diff --git a/drivers/gpu/drm/i915/selftests/i915_gem_gtt.c b/drivers/gpu/drm/i915/selftests/i915_gem_gtt.c
> index ea2cf1080979..27c733b00976 100644
> --- a/drivers/gpu/drm/i915/selftests/i915_gem_gtt.c
> +++ b/drivers/gpu/drm/i915/selftests/i915_gem_gtt.c
> @@ -1114,15 +1114,8 @@ static int misaligned_case(struct i915_address_space *vm, struct intel_memory_re
> expected_node_size = expected_vma_size;
>
> if (HAS_64K_PAGES(vm->i915) && i915_gem_object_is_lmem(obj)) {
> - /*
> - * The compact-pt should expand lmem node to 2MB for the ppGTT,
> - * for all other cases we should only expect 64K.
> - */
> expected_vma_size = round_up(size, I915_GTT_PAGE_SIZE_64K);
> - if (NEEDS_COMPACT_PT(vm->i915) && !i915_is_ggtt(vm))
> - expected_node_size = round_up(size, I915_GTT_PAGE_SIZE_2M);
> - else
> - expected_node_size = round_up(size, I915_GTT_PAGE_SIZE_64K);
> + expected_node_size = round_up(size, I915_GTT_PAGE_SIZE_64K);
> }
>
> if (vma->size != expected_vma_size || vma->node.size != expected_node_size) {
> diff --git a/include/uapi/drm/i915_drm.h b/include/uapi/drm/i915_drm.h
> index 520ad2691a99..791b4a8b6f97 100644
> --- a/include/uapi/drm/i915_drm.h
> +++ b/include/uapi/drm/i915_drm.h
> @@ -3493,27 +3493,13 @@ struct drm_i915_gem_create_ext {
> *
> * The (page-aligned) allocated size for the object will be returned.
> *
> - * DG2 64K min page size implications:
> + * On platforms like DG2/ATS the kernel will always use 64K or larger
> + * pages for I915_MEMORY_CLASS_DEVICE. The kernel also requires a
> + * minimum of 64K GTT alignment for such objects.
> *
> - * On discrete platforms, starting from DG2, we have to contend with GTT
> - * page size restrictions when dealing with I915_MEMORY_CLASS_DEVICE
> - * objects. Specifically the hardware only supports 64K or larger GTT
> - * page sizes for such memory. The kernel will already ensure that all
> - * I915_MEMORY_CLASS_DEVICE memory is allocated using 64K or larger page
> - * sizes underneath.
> - *
> - * Note that the returned size here will always reflect any required
> - * rounding up done by the kernel, i.e 4K will now become 64K on devices
> - * such as DG2. The kernel will always select the largest minimum
> - * page-size for the set of possible placements as the value to use when
> - * rounding up the @size.
> - *
> - * Special DG2 GTT address alignment requirement:
> - *
> - * The GTT alignment will also need to be at least 2M for such objects.
> - *
> - * Note that due to how the hardware implements 64K GTT page support, we
> - * have some further complications:
> + * NOTE: Previously the ABI here required a minimum GTT alignment of 2M
> + * due to how to how the hardware implemented 64K GTT page support,
Double "how to".
With above two fixes this is Reviewed-by: Nirmoy Das <nirmoy.das at intel.com>
Regards,
Nirmoy
> + * where we had the following complications:
> *
> * 1) The entire PDE (which covers a 2MB virtual address range), must
> * contain only 64K PTEs, i.e mixing 4K and 64K PTEs in the same
> @@ -3522,12 +3508,10 @@ struct drm_i915_gem_create_ext {
> * 2) We still need to support 4K PTEs for I915_MEMORY_CLASS_SYSTEM
> * objects.
> *
> - * To keep things simple for userland, we mandate that any GTT mappings
> - * must be aligned to and rounded up to 2MB. The kernel will internally
> - * pad them out to the next 2MB boundary. As this only wastes virtual
> - * address space and avoids userland having to copy any needlessly
> - * complicated PDE sharing scheme (coloring) and only affects DG2, this
> - * is deemed to be a good compromise.
> + * However on actual production HW this was completely changed to now
> + * allow setting a TLB hint at the PTE level (see PS64), which is a lot
> + * more flexible than the above. With this the 2M restriction was
> + * dropped where we now only require 64K.
> */
> __u64 size;
>
More information about the dri-devel
mailing list