[PATCH v3 2/6] drm/i915: Add support for asynchronous moving fence waiting

Thomas Hellström thomas.hellstrom at linux.intel.com
Mon Nov 15 13:29:49 UTC 2021


On 11/15/21 14:13, Matthew Auld wrote:
> On 15/11/2021 12:42, Thomas Hellström wrote:
>>
>> On 11/15/21 13:36, Matthew Auld wrote:
>>> On 14/11/2021 11:12, Thomas Hellström wrote:
>>>> From: Maarten Lankhorst <maarten.lankhorst at linux.intel.com>
>>>>
>>>> For now, we will only allow async migration when TTM is used,
>>>> so the paths we care about are related to TTM.
>>>>
>>>> The mmap path is handled by having the fence in ttm_bo->moving,
>>>> when pinning, the binding only becomes available after the moving
>>>> fence is signaled, and pinning a cpu map will only work after
>>>> the moving fence signals.
>>>>
>>>> This should close all holes where userspace can read a buffer
>>>> before it's fully migrated.
>>>>
>>>> v2:
>>>> - Fix a couple of SPARSE warnings
>>>> v3:
>>>> - Fix a NULL pointer dereference
>>>>
>>>> Co-developed-by: Thomas Hellström <thomas.hellstrom at linux.intel.com>
>>>> Signed-off-by: Thomas Hellström <thomas.hellstrom at linux.intel.com>
>>>> Signed-off-by: Maarten Lankhorst <maarten.lankhorst at linux.intel.com>
>>>> ---
>>>>   drivers/gpu/drm/i915/display/intel_fbdev.c    |  7 ++--
>>>>   drivers/gpu/drm/i915/display/intel_overlay.c  |  2 +-
>>>>   drivers/gpu/drm/i915/gem/i915_gem_pages.c     |  6 +++
>>>>   .../i915/gem/selftests/i915_gem_coherency.c   |  4 +-
>>>>   .../drm/i915/gem/selftests/i915_gem_mman.c    | 22 ++++++-----
>>>>   drivers/gpu/drm/i915/i915_vma.c               | 39 
>>>> ++++++++++++++++++-
>>>>   drivers/gpu/drm/i915/i915_vma.h               |  3 ++
>>>>   drivers/gpu/drm/i915/selftests/i915_vma.c     |  4 +-
>>>>   8 files changed, 69 insertions(+), 18 deletions(-)
>>>>
>>>> diff --git a/drivers/gpu/drm/i915/display/intel_fbdev.c 
>>>> b/drivers/gpu/drm/i915/display/intel_fbdev.c
>>>> index adc3a81be9f7..5902ad0c2bd8 100644
>>>> --- a/drivers/gpu/drm/i915/display/intel_fbdev.c
>>>> +++ b/drivers/gpu/drm/i915/display/intel_fbdev.c
>>>> @@ -265,11 +265,12 @@ static int intelfb_create(struct 
>>>> drm_fb_helper *helper,
>>>>           info->fix.smem_len = vma->node.size;
>>>>       }
>>>>   -    vaddr = i915_vma_pin_iomap(vma);
>>>> +    vaddr = i915_vma_pin_iomap_unlocked(vma);
>>>>       if (IS_ERR(vaddr)) {
>>>> -        drm_err(&dev_priv->drm,
>>>> -            "Failed to remap framebuffer into virtual memory\n");
>>>>           ret = PTR_ERR(vaddr);
>>>> +        if (ret != -EINTR && ret != -ERESTARTSYS)
>>>> +            drm_err(&dev_priv->drm,
>>>> +                "Failed to remap framebuffer into virtual memory\n");
>>>>           goto out_unpin;
>>>>       }
>>>>       info->screen_base = vaddr;
>>>> diff --git a/drivers/gpu/drm/i915/display/intel_overlay.c 
>>>> b/drivers/gpu/drm/i915/display/intel_overlay.c
>>>> index 7e3f5c6ca484..21593f3f2664 100644
>>>> --- a/drivers/gpu/drm/i915/display/intel_overlay.c
>>>> +++ b/drivers/gpu/drm/i915/display/intel_overlay.c
>>>> @@ -1357,7 +1357,7 @@ static int get_registers(struct intel_overlay 
>>>> *overlay, bool use_phys)
>>>>           overlay->flip_addr = sg_dma_address(obj->mm.pages->sgl);
>>>>       else
>>>>           overlay->flip_addr = i915_ggtt_offset(vma);
>>>> -    overlay->regs = i915_vma_pin_iomap(vma);
>>>> +    overlay->regs = i915_vma_pin_iomap_unlocked(vma);
>>>>       i915_vma_unpin(vma);
>>>>         if (IS_ERR(overlay->regs)) {
>>>> diff --git a/drivers/gpu/drm/i915/gem/i915_gem_pages.c 
>>>> b/drivers/gpu/drm/i915/gem/i915_gem_pages.c
>>>> index c4f684b7cc51..49c6e55c68ce 100644
>>>> --- a/drivers/gpu/drm/i915/gem/i915_gem_pages.c
>>>> +++ b/drivers/gpu/drm/i915/gem/i915_gem_pages.c
>>>> @@ -418,6 +418,12 @@ void *i915_gem_object_pin_map(struct 
>>>> drm_i915_gem_object *obj,
>>>>       }
>>>>         if (!ptr) {
>>>> +        err = i915_gem_object_wait_moving_fence(obj, true);
>>>> +        if (err) {
>>>> +            ptr = ERR_PTR(err);
>>>> +            goto err_unpin;
>>>> +        }
>>>> +
>>>>           if (GEM_WARN_ON(type == I915_MAP_WC &&
>>>>                   !static_cpu_has(X86_FEATURE_PAT)))
>>>>               ptr = ERR_PTR(-ENODEV);
>>>> diff --git 
>>>> a/drivers/gpu/drm/i915/gem/selftests/i915_gem_coherency.c 
>>>> b/drivers/gpu/drm/i915/gem/selftests/i915_gem_coherency.c
>>>> index 13b088cc787e..067c512961ba 100644
>>>> --- a/drivers/gpu/drm/i915/gem/selftests/i915_gem_coherency.c
>>>> +++ b/drivers/gpu/drm/i915/gem/selftests/i915_gem_coherency.c
>>>> @@ -101,7 +101,7 @@ static int gtt_set(struct context *ctx, 
>>>> unsigned long offset, u32 v)
>>>>         intel_gt_pm_get(vma->vm->gt);
>>>>   -    map = i915_vma_pin_iomap(vma);
>>>> +    map = i915_vma_pin_iomap_unlocked(vma);
>>>>       i915_vma_unpin(vma);
>>>>       if (IS_ERR(map)) {
>>>>           err = PTR_ERR(map);
>>>> @@ -134,7 +134,7 @@ static int gtt_get(struct context *ctx, 
>>>> unsigned long offset, u32 *v)
>>>>         intel_gt_pm_get(vma->vm->gt);
>>>>   -    map = i915_vma_pin_iomap(vma);
>>>> +    map = i915_vma_pin_iomap_unlocked(vma);
>>>>       i915_vma_unpin(vma);
>>>>       if (IS_ERR(map)) {
>>>>           err = PTR_ERR(map);
>>>> diff --git a/drivers/gpu/drm/i915/gem/selftests/i915_gem_mman.c 
>>>> b/drivers/gpu/drm/i915/gem/selftests/i915_gem_mman.c
>>>> index 6d30cdfa80f3..5d54181c2145 100644
>>>> --- a/drivers/gpu/drm/i915/gem/selftests/i915_gem_mman.c
>>>> +++ b/drivers/gpu/drm/i915/gem/selftests/i915_gem_mman.c
>>>> @@ -125,12 +125,13 @@ static int check_partial_mapping(struct 
>>>> drm_i915_gem_object *obj,
>>>>       n = page - view.partial.offset;
>>>>       GEM_BUG_ON(n >= view.partial.size);
>>>>   -    io = i915_vma_pin_iomap(vma);
>>>> +    io = i915_vma_pin_iomap_unlocked(vma);
>>>>       i915_vma_unpin(vma);
>>>>       if (IS_ERR(io)) {
>>>> -        pr_err("Failed to iomap partial view: offset=%lu; err=%d\n",
>>>> -               page, (int)PTR_ERR(io));
>>>>           err = PTR_ERR(io);
>>>> +        if (err != -EINTR && err != -ERESTARTSYS)
>>>> +            pr_err("Failed to iomap partial view: offset=%lu; 
>>>> err=%d\n",
>>>> +                   page, err);
>>>>           goto out;
>>>>       }
>>>>   @@ -219,12 +220,15 @@ static int check_partial_mappings(struct 
>>>> drm_i915_gem_object *obj,
>>>>           n = page - view.partial.offset;
>>>>           GEM_BUG_ON(n >= view.partial.size);
>>>>   -        io = i915_vma_pin_iomap(vma);
>>>> +        io = i915_vma_pin_iomap_unlocked(vma);
>>>>           i915_vma_unpin(vma);
>>>>           if (IS_ERR(io)) {
>>>> -            pr_err("Failed to iomap partial view: offset=%lu; 
>>>> err=%d\n",
>>>> -                   page, (int)PTR_ERR(io));
>>>> -            return PTR_ERR(io);
>>>> +            int err = PTR_ERR(io);
>>>> +
>>>> +            if (err != -EINTR && err != -ERESTARTSYS)
>>>> +                pr_err("Failed to iomap partial view: offset=%lu; 
>>>> err=%d\n",
>>>> +                       page, err);
>>>> +            return err;
>>>>           }
>>>>             iowrite32(page, io + n * PAGE_SIZE / sizeof(*io));
>>>> @@ -773,7 +777,7 @@ static int gtt_set(struct drm_i915_gem_object 
>>>> *obj)
>>>>           return PTR_ERR(vma);
>>>>         intel_gt_pm_get(vma->vm->gt);
>>>> -    map = i915_vma_pin_iomap(vma);
>>>> +    map = i915_vma_pin_iomap_unlocked(vma);
>>>>       i915_vma_unpin(vma);
>>>>       if (IS_ERR(map)) {
>>>>           err = PTR_ERR(map);
>>>> @@ -799,7 +803,7 @@ static int gtt_check(struct drm_i915_gem_object 
>>>> *obj)
>>>>           return PTR_ERR(vma);
>>>>         intel_gt_pm_get(vma->vm->gt);
>>>> -    map = i915_vma_pin_iomap(vma);
>>>> +    map = i915_vma_pin_iomap_unlocked(vma);
>>>>       i915_vma_unpin(vma);
>>>>       if (IS_ERR(map)) {
>>>>           err = PTR_ERR(map);
>>>> diff --git a/drivers/gpu/drm/i915/i915_vma.c 
>>>> b/drivers/gpu/drm/i915/i915_vma.c
>>>> index 8781c4f61952..069f22b3cd48 100644
>>>> --- a/drivers/gpu/drm/i915/i915_vma.c
>>>> +++ b/drivers/gpu/drm/i915/i915_vma.c
>>>> @@ -431,6 +431,13 @@ int i915_vma_bind(struct i915_vma *vma,
>>>>               work->pinned = i915_gem_object_get(vma->obj);
>>>>           }
>>>>       } else {
>>>> +        if (vma->obj) {
>>>> +            int ret;
>>>> +
>>>> +            ret = i915_gem_object_wait_moving_fence(vma->obj, true);
>>>> +            if (ret)
>>>> +                return ret;
>>>> +        }
>>>>           vma->ops->bind_vma(vma->vm, NULL, vma, cache_level, 
>>>> bind_flags);
>>>>       }
>>>>   @@ -455,6 +462,10 @@ void __iomem *i915_vma_pin_iomap(struct 
>>>> i915_vma *vma)
>>>>         ptr = READ_ONCE(vma->iomap);
>>>>       if (ptr == NULL) {
>>>> +        err = i915_gem_object_wait_moving_fence(vma->obj, true);
>>>> +        if (err)
>>>> +            goto err;
>>>> +
>>>>           /*
>>>>            * TODO: consider just using i915_gem_object_pin_map() 
>>>> for lmem
>>>>            * instead, which already supports mapping non-contiguous 
>>>> chunks
>>>> @@ -496,6 +507,25 @@ void __iomem *i915_vma_pin_iomap(struct 
>>>> i915_vma *vma)
>>>>       return IO_ERR_PTR(err);
>>>>   }
>>>>   +void __iomem *i915_vma_pin_iomap_unlocked(struct i915_vma *vma)
>>>> +{
>>>> +    struct i915_gem_ww_ctx ww;
>>>> +    void __iomem *map;
>>>> +    int err;
>>>> +
>>>> +    for_i915_gem_ww(&ww, err, true) {
>>>> +        err = i915_gem_object_lock(vma->obj, &ww);
>>>> +        if (err)
>>>> +            continue;
>>>> +
>>>> +        map = i915_vma_pin_iomap(vma);
>>>> +    }
>>>> +    if (err)
>>>> +        map = IO_ERR_PTR(err);
>>>> +
>>>> +    return map;
>>>> +}
>>>
>>> What is the reason for this change? Is this strictly related to this 
>>> series/commit?
>>
>> Yes, it's because pulling out the moving fence requires the dma_resv 
>> lock.
>
> Ok, I was thinking that vma_pin_iomap is only ever called on an 
> already bound GGTT vma, for which we do a syncronous wait_for_bind, 
> but maybe that's not always true?
>
Hmm, Good point. We should probably replace that vma_pin_iomap stuff 
with an assert that the
binding fence is indeed signaled and error free? Because if binding 
succeeded, no need to check the moving fence.

/Thomas



> Reviewed-by: Matthew Auld <matthew.auld at intel.com>
>
>>
>> /Thomas
>>
>>


More information about the dri-devel mailing list