From: Ekansh Gupta <[email protected]>

Introduce the QDA memory manager (qda_memory_manager) to track and
manage the IOMMU devices that back each compute context bank (CB).

Each CB device registered on the qda-compute-cb bus is assigned a
unique ID via an XArray and wrapped in a qda_iommu_device descriptor
that records the device pointer and its stream ID. This registry
allows the driver to look up the correct IOMMU domain for a given
session when mapping DSP buffers.

The memory manager is initialised in qda_init_device() before CB
devices are populated and torn down in qda_deinit_device() after they
are destroyed, ensuring no dangling references remain in the XArray.

qda_cb.c is extended with qda_cb_setup_device(), which is called
immediately after a CB device is registered on the bus. It allocates
a qda_iommu_device, registers it with the memory manager, and stores
it as the CB device's driver data so that qda_destroy_cb_device() can
retrieve and unregister it during teardown.

Assisted-by: Claude:claude-4-6-sonnet
Signed-off-by: Ekansh Gupta <[email protected]>
---
 drivers/accel/qda/Makefile             |   1 +
 drivers/accel/qda/qda_cb.c             |  47 ++++++++++++++
 drivers/accel/qda/qda_drv.c            |  34 ++++++++++
 drivers/accel/qda/qda_drv.h            |   5 ++
 drivers/accel/qda/qda_memory_manager.c | 111 +++++++++++++++++++++++++++++++++
 drivers/accel/qda/qda_memory_manager.h |  49 +++++++++++++++
 drivers/accel/qda/qda_rpmsg.c          |   7 +++
 7 files changed, 254 insertions(+)

diff --git a/drivers/accel/qda/Makefile b/drivers/accel/qda/Makefile
index 143c9e4e789e..701fad5ffb50 100644
--- a/drivers/accel/qda/Makefile
+++ b/drivers/accel/qda/Makefile
@@ -8,6 +8,7 @@ obj-$(CONFIG_DRM_ACCEL_QDA)     := qda.o
 qda-y := \
        qda_cb.o \
        qda_drv.o \
+       qda_memory_manager.o \
        qda_rpmsg.o
 
 obj-$(CONFIG_DRM_ACCEL_QDA_COMPUTE_BUS) += qda_compute_bus.o
diff --git a/drivers/accel/qda/qda_cb.c b/drivers/accel/qda/qda_cb.c
index 77caf8438c67..6d540bb0ec7b 100644
--- a/drivers/accel/qda/qda_cb.c
+++ b/drivers/accel/qda/qda_cb.c
@@ -8,11 +8,42 @@
 #include <linux/slab.h>
 #include <drm/drm_print.h>
 #include "qda_drv.h"
+#include "qda_memory_manager.h"
 #include "qda_cb.h"
 
+static int qda_cb_setup_device(struct qda_dev *qdev, struct device *cb_dev, 
u32 sid)
+{
+       struct qda_iommu_device *iommu_dev;
+       int rc;
+
+       drm_dbg_driver(&qdev->drm_dev, "Setting up CB device %s\n", 
dev_name(cb_dev));
+
+       iommu_dev = kzalloc_obj(*iommu_dev);
+       if (!iommu_dev)
+               return -ENOMEM;
+
+       iommu_dev->dev = cb_dev;
+       iommu_dev->qdev = qdev;
+       iommu_dev->sid = sid;
+
+       rc = qda_memory_manager_register_device(qdev->iommu_mgr, iommu_dev);
+       if (rc) {
+               drm_err(&qdev->drm_dev, "Failed to register IOMMU device: 
%d\n", rc);
+               kfree(iommu_dev);
+               return rc;
+       }
+
+       dev_set_drvdata(cb_dev, iommu_dev);
+
+       drm_dbg_driver(&qdev->drm_dev, "CB device setup complete - SID: %u\n", 
sid);
+
+       return 0;
+}
+
 int qda_create_cb_device(struct qda_dev *qdev, struct device_node *cb_node)
 {
        struct device *cb_dev;
+       int ret;
        u32 sid = 0;
        char name[64];
        struct qda_cb_dev *entry;
@@ -30,6 +61,13 @@ int qda_create_cb_device(struct qda_dev *qdev, struct 
device_node *cb_node)
                return PTR_ERR(cb_dev);
        }
 
+       ret = qda_cb_setup_device(qdev, cb_dev, sid);
+       if (ret) {
+               drm_err(&qdev->drm_dev, "CB device setup failed: %d\n", ret);
+               device_unregister(cb_dev);
+               return ret;
+       }
+
        entry = kzalloc_obj(*entry);
        if (!entry) {
                device_unregister(cb_dev);
@@ -80,6 +118,7 @@ int qda_cb_populate(struct qda_dev *qdev, struct device_node 
*parent_node)
 void qda_destroy_cb_device(struct device *cb_dev)
 {
        struct iommu_group *group;
+       struct qda_iommu_device *iommu_dev;
 
        if (!cb_dev) {
                pr_debug("qda: NULL CB device passed to destroy\n");
@@ -88,6 +127,14 @@ void qda_destroy_cb_device(struct device *cb_dev)
 
        dev_dbg(cb_dev, "Destroying CB device %s\n", dev_name(cb_dev));
 
+       iommu_dev = dev_get_drvdata(cb_dev);
+       if (iommu_dev && iommu_dev->qdev && iommu_dev->qdev->iommu_mgr) {
+               dev_dbg(cb_dev, "Unregistering IOMMU device for %s\n",
+                       dev_name(cb_dev));
+               qda_memory_manager_unregister_device(iommu_dev->qdev->iommu_mgr,
+                                                    iommu_dev);
+       }
+
        group = iommu_group_get(cb_dev);
        if (group) {
                dev_dbg(cb_dev, "Removing %s from IOMMU group\n", 
dev_name(cb_dev));
diff --git a/drivers/accel/qda/qda_drv.c b/drivers/accel/qda/qda_drv.c
index 6c20d6a2fc47..0ad5d9873d7e 100644
--- a/drivers/accel/qda/qda_drv.c
+++ b/drivers/accel/qda/qda_drv.c
@@ -57,6 +57,40 @@ struct qda_dev *qda_alloc_device(struct device *dev)
        return qdev;
 }
 
+static void cleanup_memory_manager(struct qda_dev *qdev)
+{
+       if (qdev->iommu_mgr) {
+               qda_memory_manager_exit(qdev->iommu_mgr);
+               kfree(qdev->iommu_mgr);
+               qdev->iommu_mgr = NULL;
+       }
+}
+
+static int init_memory_manager(struct qda_dev *qdev)
+{
+       qdev->iommu_mgr = kzalloc_obj(*qdev->iommu_mgr);
+       if (!qdev->iommu_mgr)
+               return -ENOMEM;
+
+       return qda_memory_manager_init(qdev->iommu_mgr);
+}
+
+void qda_deinit_device(struct qda_dev *qdev)
+{
+       cleanup_memory_manager(qdev);
+}
+
+int qda_init_device(struct qda_dev *qdev)
+{
+       int ret;
+
+       ret = init_memory_manager(qdev);
+       if (ret)
+               drm_err(&qdev->drm_dev, "Failed to initialize memory manager: 
%d\n", ret);
+
+       return ret;
+}
+
 void qda_unregister_device(struct qda_dev *qdev)
 {
        drm_dev_unregister(&qdev->drm_dev);
diff --git a/drivers/accel/qda/qda_drv.h b/drivers/accel/qda/qda_drv.h
index 2715f378775d..eb089e586b17 100644
--- a/drivers/accel/qda/qda_drv.h
+++ b/drivers/accel/qda/qda_drv.h
@@ -13,6 +13,7 @@
 #include <drm/drm_device.h>
 #include <drm/drm_drv.h>
 #include <drm/drm_file.h>
+#include "qda_memory_manager.h"
 
 /* Driver identification */
 #define QDA_DRIVER_NAME "qda"
@@ -40,6 +41,8 @@ struct qda_dev {
        struct device *dev;
        /** @cb_devs: Compute context-bank (CB) child devices */
        struct list_head cb_devs;
+       /** @iommu_mgr: IOMMU/memory manager instance */
+       struct qda_memory_manager *iommu_mgr;
        /** @dsp_name: Name of the DSP domain (e.g. "cdsp", "adsp") */
        const char *dsp_name;
 };
@@ -59,6 +62,8 @@ static inline struct qda_dev *qda_dev_from_drm(struct 
drm_device *dev)
 struct qda_dev *qda_alloc_device(struct device *dev);
 
 /* Core device lifecycle */
+int qda_init_device(struct qda_dev *qdev);
+void qda_deinit_device(struct qda_dev *qdev);
 int qda_register_device(struct qda_dev *qdev);
 void qda_unregister_device(struct qda_dev *qdev);
 
diff --git a/drivers/accel/qda/qda_memory_manager.c 
b/drivers/accel/qda/qda_memory_manager.c
new file mode 100644
index 000000000000..00a9c0ae4224
--- /dev/null
+++ b/drivers/accel/qda/qda_memory_manager.c
@@ -0,0 +1,111 @@
+// SPDX-License-Identifier: GPL-2.0-only
+// Copyright (c) Qualcomm Technologies, Inc. and/or its subsidiaries.
+
+#include <linux/refcount.h>
+#include <linux/slab.h>
+#include <linux/spinlock.h>
+#include <linux/xarray.h>
+#include <drm/drm_file.h>
+#include "qda_drv.h"
+#include "qda_memory_manager.h"
+
+static void cleanup_all_memory_devices(struct qda_memory_manager *mem_mgr)
+{
+       unsigned long index;
+       void *entry;
+
+       pr_debug("qda: Starting cleanup of all memory devices\n");
+
+       xa_for_each(&mem_mgr->device_xa, index, entry) {
+               struct qda_iommu_device *iommu_dev = entry;
+
+               pr_debug("qda: Cleaning up device id=%lu\n", index);
+
+               xa_erase(&mem_mgr->device_xa, index);
+               kfree(iommu_dev);
+       }
+
+       pr_debug("qda: Completed cleanup of all memory devices\n");
+}
+
+static int allocate_device_id(struct qda_memory_manager *mem_mgr,
+                             struct qda_iommu_device *iommu_dev, u32 *id)
+{
+       int ret;
+
+       ret = xa_alloc(&mem_mgr->device_xa, id, iommu_dev,
+                      xa_limit_31b, GFP_KERNEL);
+       if (ret) {
+               dev_err(iommu_dev->dev, "Failed to allocate XArray ID: %d\n", 
ret);
+               return ret;
+       }
+
+       dev_dbg(iommu_dev->dev, "Allocated device id=%u\n", *id);
+       return 0;
+}
+
+/**
+ * qda_memory_manager_register_device() - Register an IOMMU device
+ * @mem_mgr: Pointer to memory manager
+ * @iommu_dev: Pointer to IOMMU device to register
+ *
+ * Return: 0 on success, negative error code on failure
+ */
+int qda_memory_manager_register_device(struct qda_memory_manager *mem_mgr,
+                                      struct qda_iommu_device *iommu_dev)
+{
+       int ret;
+       u32 id;
+
+       ret = allocate_device_id(mem_mgr, iommu_dev, &id);
+       if (ret) {
+               dev_err(iommu_dev->dev,
+                       "Failed to allocate device ID: %d (sid=%u)\n",
+                       ret, iommu_dev->sid);
+               return ret;
+       }
+
+       iommu_dev->id = id;
+
+       dev_dbg(iommu_dev->dev, "Registered device id=%u (sid=%u)\n", id, 
iommu_dev->sid);
+
+       return 0;
+}
+
+/**
+ * qda_memory_manager_unregister_device() - Unregister an IOMMU device
+ * @mem_mgr: Pointer to memory manager
+ * @iommu_dev: Pointer to IOMMU device to unregister
+ */
+void qda_memory_manager_unregister_device(struct qda_memory_manager *mem_mgr,
+                                         struct qda_iommu_device *iommu_dev)
+{
+       xa_erase(&mem_mgr->device_xa, iommu_dev->id);
+       kfree(iommu_dev);
+}
+
+/**
+ * qda_memory_manager_init() - Initialize the memory manager
+ * @mem_mgr: Pointer to memory manager structure to initialize
+ *
+ * Return: 0 on success, negative error code on failure
+ */
+int qda_memory_manager_init(struct qda_memory_manager *mem_mgr)
+{
+       pr_debug("qda: Initializing memory manager\n");
+
+       xa_init_flags(&mem_mgr->device_xa, XA_FLAGS_ALLOC);
+
+       pr_debug("qda: Memory manager initialized successfully\n");
+       return 0;
+}
+
+/**
+ * qda_memory_manager_exit() - Clean up the memory manager
+ * @mem_mgr: Pointer to memory manager structure to clean up
+ */
+void qda_memory_manager_exit(struct qda_memory_manager *mem_mgr)
+{
+       cleanup_all_memory_devices(mem_mgr);
+       pr_debug("qda: Memory manager exited\n");
+}
diff --git a/drivers/accel/qda/qda_memory_manager.h 
b/drivers/accel/qda/qda_memory_manager.h
new file mode 100644
index 000000000000..0243f9c0c5aa
--- /dev/null
+++ b/drivers/accel/qda/qda_memory_manager.h
@@ -0,0 +1,49 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+/*
+ * Copyright (c) Qualcomm Technologies, Inc. and/or its subsidiaries.
+ */
+
+#ifndef __QDA_MEMORY_MANAGER_H__
+#define __QDA_MEMORY_MANAGER_H__
+
+#include <linux/device.h>
+#include <linux/xarray.h>
+#include "qda_drv.h"
+
+/**
+ * struct qda_iommu_device - IOMMU device instance for memory management
+ *
+ * Represents a single IOMMU-enabled device managed by the memory manager.
+ * Each device can be assigned to a specific process session.
+ */
+struct qda_iommu_device {
+       /** @dev: Pointer to the underlying device */
+       struct device *dev;
+       /** @qdev: Back-pointer to the parent QDA device */
+       struct qda_dev *qdev;
+       /** @id: Unique identifier assigned by the memory manager XArray */
+       u32 id;
+       /** @sid: Stream ID for IOMMU transactions */
+       u32 sid;
+};
+
+/**
+ * struct qda_memory_manager - Central memory management coordinator
+ *
+ * Coordinates memory management across multiple IOMMU devices. Maintains
+ * a registry of devices using an XArray for O(1) lookup by ID.
+ */
+struct qda_memory_manager {
+       /** @device_xa: XArray storing all registered IOMMU devices */
+       struct xarray device_xa;
+};
+
+int qda_memory_manager_init(struct qda_memory_manager *mem_mgr);
+void qda_memory_manager_exit(struct qda_memory_manager *mem_mgr);
+
+int qda_memory_manager_register_device(struct qda_memory_manager *mem_mgr,
+                                      struct qda_iommu_device *iommu_dev);
+void qda_memory_manager_unregister_device(struct qda_memory_manager *mem_mgr,
+                                         struct qda_iommu_device *iommu_dev);
+
+#endif /* __QDA_MEMORY_MANAGER_H__ */
diff --git a/drivers/accel/qda/qda_rpmsg.c b/drivers/accel/qda/qda_rpmsg.c
index afd9e851d00e..719dabb028c5 100644
--- a/drivers/accel/qda/qda_rpmsg.c
+++ b/drivers/accel/qda/qda_rpmsg.c
@@ -39,6 +39,7 @@ static void qda_rpmsg_remove(struct rpmsg_device *rpdev)
        drm_dev_unplug(&qdev->drm_dev);
        qdev->rpdev = NULL;
        qda_unregister_device(qdev);
+       qda_deinit_device(qdev);
        dev_info(qdev->dev, "RPMsg device removed\n");
 }
 
@@ -61,14 +62,20 @@ static int qda_rpmsg_probe(struct rpmsg_device *rpdev)
        }
        qdev->dsp_name = label;
 
+       ret = qda_init_device(qdev);
+       if (ret)
+               return ret;
+
        ret = qda_cb_populate(qdev, rpdev->dev.of_node);
        if (ret) {
                dev_err(qdev->dev, "Failed to populate child devices: %d\n", 
ret);
+               qda_deinit_device(qdev);
                return ret;
        }
 
        ret = qda_register_device(qdev);
        if (ret) {
+               qda_deinit_device(qdev);
                qda_cb_unpopulate(qdev);
                return ret;
        }

-- 
2.34.1


Reply via email to