[PATCH] drm/qxl: Pin buffer objects for internal mappings
Daniel Vetter
daniel.vetter at ffwll.ch
Mon Jul 8 08:58:03 UTC 2024
On Mon, Jul 08, 2024 at 10:09:44AM +0200, Thomas Zimmermann wrote:
> Hi,
>
> ping for a review. This is a bugfix for a serious problem.
I tried to look around whether there's any place where we could WARN_ON if
we create a vmap but it's not pinned. But there's lots of places where we
want the vmap only for the duration of the dma_resv locked section, so
really can't do that. And your patch removes the unlocked vmap
implementation, which would be the only place really.
Reviewed-by: Daniel Vetter <daniel.vetter at ffwll.ch>
>
> Best regards
> Thomas
>
> Am 02.07.24 um 16:20 schrieb Thomas Zimmermann:
> > Add qxl_bo_pin_and_vmap() that pins and vmaps a buffer object in one
> > step. Update callers of the regular qxl_bo_vmap(). Fixes a bug where
> > qxl accesses an unpinned buffer object while it is being moved; such
> > as with the monitor-description BO. An typical error is shown below.
> >
> > [ 4.303586] [drm:drm_atomic_helper_commit_planes] *ERROR* head 1 wrong: 65376256x16777216+0+0
> > [ 4.586883] [drm:drm_atomic_helper_commit_planes] *ERROR* head 1 wrong: 65376256x16777216+0+0
> > [ 4.904036] [drm:drm_atomic_helper_commit_planes] *ERROR* head 1 wrong: 65335296x16777216+0+0
> > [ 5.374347] [drm:qxl_release_from_id_locked] *ERROR* failed to find id in release_idr
> >
> > Commit b33651a5c98d ("drm/qxl: Do not pin buffer objects for vmap")
> > removed the implicit pin operation from qxl's vmap code. This is the
> > correct behavior for GEM and PRIME interfaces, but the pin is still
> > needed for qxl internal operation.
> >
> > Also add a corresponding function qxl_bo_vunmap_and_unpin() and remove
> > the old qxl_bo_vmap() helpers.
> >
> > Future directions: BOs should not be pinned or vmapped unnecessarily.
> > The pin-and-vmap operation should be removed from the driver and a
> > temporary mapping should be established with a vmap_local-like helper.
> > See the client helper drm_client_buffer_vmap_local() for semantics.
> >
> > Signed-off-by: Thomas Zimmermann <tzimmermann at suse.de>
> > Fixes: b33651a5c98d ("drm/qxl: Do not pin buffer objects for vmap")
> > Reported-by: David Kaplan <david.kaplan at amd.com>
> > Closes: https://lore.kernel.org/dri-devel/ab0fb17d-0f96-4ee6-8b21-65d02bb02655@suse.de/
> > Tested-by: David Kaplan <david.kaplan at amd.com>
> > Cc: Thomas Zimmermann <tzimmermann at suse.de>
> > Cc: Dmitry Osipenko <dmitry.osipenko at collabora.com>
> > Cc: Christian König <christian.koenig at amd.com>
> > Cc: Zack Rusin <zack.rusin at broadcom.com>
> > Cc: Dave Airlie <airlied at redhat.com>
> > Cc: Gerd Hoffmann <kraxel at redhat.com>
> > Cc: virtualization at lists.linux.dev
> > Cc: spice-devel at lists.freedesktop.org
> > ---
> > drivers/gpu/drm/qxl/qxl_display.c | 14 +++++++-------
> > drivers/gpu/drm/qxl/qxl_object.c | 11 +++++++++--
> > drivers/gpu/drm/qxl/qxl_object.h | 4 ++--
> > 3 files changed, 18 insertions(+), 11 deletions(-)
> >
> > diff --git a/drivers/gpu/drm/qxl/qxl_display.c b/drivers/gpu/drm/qxl/qxl_display.c
> > index 86a5dea710c0..bc24af08dfcd 100644
> > --- a/drivers/gpu/drm/qxl/qxl_display.c
> > +++ b/drivers/gpu/drm/qxl/qxl_display.c
> > @@ -584,11 +584,11 @@ static struct qxl_bo *qxl_create_cursor(struct qxl_device *qdev,
> > if (ret)
> > goto err;
> > - ret = qxl_bo_vmap(cursor_bo, &cursor_map);
> > + ret = qxl_bo_pin_and_vmap(cursor_bo, &cursor_map);
> > if (ret)
> > goto err_unref;
> > - ret = qxl_bo_vmap(user_bo, &user_map);
> > + ret = qxl_bo_pin_and_vmap(user_bo, &user_map);
> > if (ret)
> > goto err_unmap;
> > @@ -614,12 +614,12 @@ static struct qxl_bo *qxl_create_cursor(struct qxl_device *qdev,
> > user_map.vaddr, size);
> > }
> > - qxl_bo_vunmap(user_bo);
> > - qxl_bo_vunmap(cursor_bo);
> > + qxl_bo_vunmap_and_unpin(user_bo);
> > + qxl_bo_vunmap_and_unpin(cursor_bo);
> > return cursor_bo;
> > err_unmap:
> > - qxl_bo_vunmap(cursor_bo);
> > + qxl_bo_vunmap_and_unpin(cursor_bo);
> > err_unref:
> > qxl_bo_unpin(cursor_bo);
> > qxl_bo_unref(&cursor_bo);
> > @@ -1205,7 +1205,7 @@ int qxl_create_monitors_object(struct qxl_device *qdev)
> > }
> > qdev->monitors_config_bo = gem_to_qxl_bo(gobj);
> > - ret = qxl_bo_vmap(qdev->monitors_config_bo, &map);
> > + ret = qxl_bo_pin_and_vmap(qdev->monitors_config_bo, &map);
> > if (ret)
> > return ret;
> > @@ -1236,7 +1236,7 @@ int qxl_destroy_monitors_object(struct qxl_device *qdev)
> > qdev->monitors_config = NULL;
> > qdev->ram_header->monitors_config = 0;
> > - ret = qxl_bo_vunmap(qdev->monitors_config_bo);
> > + ret = qxl_bo_vunmap_and_unpin(qdev->monitors_config_bo);
> > if (ret)
> > return ret;
> > diff --git a/drivers/gpu/drm/qxl/qxl_object.c b/drivers/gpu/drm/qxl/qxl_object.c
> > index 5893e27a7ae5..cb1b7c2580ae 100644
> > --- a/drivers/gpu/drm/qxl/qxl_object.c
> > +++ b/drivers/gpu/drm/qxl/qxl_object.c
> > @@ -182,7 +182,7 @@ int qxl_bo_vmap_locked(struct qxl_bo *bo, struct iosys_map *map)
> > return 0;
> > }
> > -int qxl_bo_vmap(struct qxl_bo *bo, struct iosys_map *map)
> > +int qxl_bo_pin_and_vmap(struct qxl_bo *bo, struct iosys_map *map)
> > {
> > int r;
> > @@ -190,7 +190,13 @@ int qxl_bo_vmap(struct qxl_bo *bo, struct iosys_map *map)
> > if (r)
> > return r;
> > + r = qxl_bo_pin_locked(bo);
> > + if (r)
> > + return r;
> > +
> > r = qxl_bo_vmap_locked(bo, map);
> > + if (r)
> > + qxl_bo_unpin_locked(bo);
> > qxl_bo_unreserve(bo);
> > return r;
> > }
> > @@ -241,7 +247,7 @@ void qxl_bo_vunmap_locked(struct qxl_bo *bo)
> > ttm_bo_vunmap(&bo->tbo, &bo->map);
> > }
> > -int qxl_bo_vunmap(struct qxl_bo *bo)
> > +int qxl_bo_vunmap_and_unpin(struct qxl_bo *bo)
> > {
> > int r;
> > @@ -250,6 +256,7 @@ int qxl_bo_vunmap(struct qxl_bo *bo)
> > return r;
> > qxl_bo_vunmap_locked(bo);
> > + qxl_bo_unpin_locked(bo);
> > qxl_bo_unreserve(bo);
> > return 0;
> > }
> > diff --git a/drivers/gpu/drm/qxl/qxl_object.h b/drivers/gpu/drm/qxl/qxl_object.h
> > index 1cf5bc759101..875f63221074 100644
> > --- a/drivers/gpu/drm/qxl/qxl_object.h
> > +++ b/drivers/gpu/drm/qxl/qxl_object.h
> > @@ -59,9 +59,9 @@ extern int qxl_bo_create(struct qxl_device *qdev,
> > u32 priority,
> > struct qxl_surface *surf,
> > struct qxl_bo **bo_ptr);
> > -int qxl_bo_vmap(struct qxl_bo *bo, struct iosys_map *map);
> > +int qxl_bo_pin_and_vmap(struct qxl_bo *bo, struct iosys_map *map);
> > int qxl_bo_vmap_locked(struct qxl_bo *bo, struct iosys_map *map);
> > -int qxl_bo_vunmap(struct qxl_bo *bo);
> > +int qxl_bo_vunmap_and_unpin(struct qxl_bo *bo);
> > void qxl_bo_vunmap_locked(struct qxl_bo *bo);
> > void *qxl_bo_kmap_atomic_page(struct qxl_device *qdev, struct qxl_bo *bo, int page_offset);
> > void qxl_bo_kunmap_atomic_page(struct qxl_device *qdev, struct qxl_bo *bo, void *map);
>
> --
> --
> Thomas Zimmermann
> Graphics Driver Developer
> SUSE Software Solutions Germany GmbH
> Frankenstrasse 146, 90461 Nuernberg, Germany
> GF: Ivo Totev, Andrew Myers, Andrew McDonald, Boudien Moerman
> HRB 36809 (AG Nuernberg)
>
--
Daniel Vetter
Software Engineer, Intel Corporation
http://blog.ffwll.ch
More information about the dri-devel
mailing list