drm/etnaviv: allocate unique ID per drm_file
authorLucas Stach <l.stach@pengutronix.de>
Wed, 1 Feb 2023 15:26:08 +0000 (16:26 +0100)
committerLucas Stach <l.stach@pengutronix.de>
Tue, 7 Feb 2023 19:49:54 +0000 (20:49 +0100)
Allows to easily track if several fd are pointing to the same
execution context due to being dup'ed.

Signed-off-by: Lucas Stach <l.stach@pengutronix.de>
Reviewed-by: Philipp Zabel <p.zabel@pengutronix.de>
drivers/gpu/drm/etnaviv/etnaviv_drv.c
drivers/gpu/drm/etnaviv/etnaviv_drv.h

index 1d2b4fb..31a7f59 100644 (file)
@@ -56,6 +56,11 @@ static int etnaviv_open(struct drm_device *dev, struct drm_file *file)
        if (!ctx)
                return -ENOMEM;
 
+       ret = xa_alloc_cyclic(&priv->active_contexts, &ctx->id, ctx,
+                             xa_limit_32b, &priv->next_context_id, GFP_KERNEL);
+       if (ret < 0)
+               goto out_free;
+
        ctx->mmu = etnaviv_iommu_context_init(priv->mmu_global,
                                              priv->cmdbuf_suballoc);
        if (!ctx->mmu) {
@@ -99,6 +104,8 @@ static void etnaviv_postclose(struct drm_device *dev, struct drm_file *file)
 
        etnaviv_iommu_context_put(ctx->mmu);
 
+       xa_erase(&priv->active_contexts, ctx->id);
+
        kfree(ctx);
 }
 
@@ -514,6 +521,8 @@ static int etnaviv_bind(struct device *dev)
 
        dma_set_max_seg_size(dev, SZ_2G);
 
+       xa_init_flags(&priv->active_contexts, XA_FLAGS_ALLOC);
+
        mutex_init(&priv->gem_lock);
        INIT_LIST_HEAD(&priv->gem_list);
        priv->num_gpus = 0;
@@ -563,6 +572,8 @@ static void etnaviv_unbind(struct device *dev)
 
        etnaviv_cmdbuf_suballoc_destroy(priv->cmdbuf_suballoc);
 
+       xa_destroy(&priv->active_contexts);
+
        drm->dev_private = NULL;
        kfree(priv);
 
index 0b311af..b3eb166 100644 (file)
@@ -29,6 +29,7 @@ struct etnaviv_iommu_global;
 #define ETNAVIV_SOFTPIN_START_ADDRESS  SZ_4M /* must be >= SUBALLOC_SIZE */
 
 struct etnaviv_file_private {
+       int id;
        struct etnaviv_iommu_context    *mmu;
        struct drm_sched_entity         sched_entity[ETNA_MAX_PIPES];
 };
@@ -41,6 +42,9 @@ struct etnaviv_drm_private {
        struct etnaviv_cmdbuf_suballoc *cmdbuf_suballoc;
        struct etnaviv_iommu_global *mmu_global;
 
+       struct xarray active_contexts;
+       u32 next_context_id;
+
        /* list of GEM objects: */
        struct mutex gem_lock;
        struct list_head gem_list;