drm/tegra: Move IOMMU group into host1x client
authorThierry Reding <treding@nvidia.com>
Fri, 8 Feb 2019 13:35:13 +0000 (14:35 +0100)
committerThierry Reding <treding@nvidia.com>
Mon, 28 Oct 2019 10:18:37 +0000 (11:18 +0100)
Handling of the IOMMU group attachment is common to all clients, so move
the group into the client to simplify code.

Signed-off-by: Thierry Reding <treding@nvidia.com>
drivers/gpu/drm/tegra/dc.c
drivers/gpu/drm/tegra/dc.h
drivers/gpu/drm/tegra/drm.c
drivers/gpu/drm/tegra/drm.h
drivers/gpu/drm/tegra/gr2d.c
drivers/gpu/drm/tegra/gr3d.c
drivers/gpu/drm/tegra/vic.c
include/linux/host1x.h

index f7f7984..3ac535b 100644 (file)
@@ -2014,9 +2014,8 @@ static int tegra_dc_init(struct host1x_client *client)
        if (!dc->syncpt)
                dev_warn(dc->dev, "failed to allocate syncpoint\n");
 
-       dc->group = host1x_client_iommu_attach(client, true);
-       if (IS_ERR(dc->group)) {
-               err = PTR_ERR(dc->group);
+       err = host1x_client_iommu_attach(client, true);
+       if (err < 0) {
                dev_err(client->dev, "failed to attach to domain: %d\n", err);
                return err;
        }
@@ -2089,7 +2088,7 @@ cleanup:
        if (!IS_ERR(primary))
                drm_plane_cleanup(primary);
 
-       host1x_client_iommu_detach(client, dc->group);
+       host1x_client_iommu_detach(client);
        host1x_syncpt_free(dc->syncpt);
 
        return err;
@@ -2114,7 +2113,7 @@ static int tegra_dc_exit(struct host1x_client *client)
                return err;
        }
 
-       host1x_client_iommu_detach(client, dc->group);
+       host1x_client_iommu_detach(client);
        host1x_syncpt_free(dc->syncpt);
 
        return 0;
index 0c4d178..3d8ddcc 100644 (file)
@@ -90,8 +90,6 @@ struct tegra_dc {
        struct drm_info_list *debugfs_files;
 
        const struct tegra_dc_soc_info *soc;
-
-       struct iommu_group *group;
 };
 
 static inline struct tegra_dc *
index 498d22a..b74362c 100644 (file)
@@ -1068,8 +1068,7 @@ int tegra_drm_unregister_client(struct tegra_drm *tegra,
        return 0;
 }
 
-struct iommu_group *host1x_client_iommu_attach(struct host1x_client *client,
-                                              bool shared)
+int host1x_client_iommu_attach(struct host1x_client *client, bool shared)
 {
        struct drm_device *drm = dev_get_drvdata(client->parent);
        struct tegra_drm *tegra = drm->dev_private;
@@ -1080,7 +1079,7 @@ struct iommu_group *host1x_client_iommu_attach(struct host1x_client *client,
                group = iommu_group_get(client->dev);
                if (!group) {
                        dev_err(client->dev, "failed to get IOMMU group\n");
-                       return ERR_PTR(-ENODEV);
+                       return -ENODEV;
                }
 
                if (!shared || (shared && (group != tegra->group))) {
@@ -1095,7 +1094,7 @@ struct iommu_group *host1x_client_iommu_attach(struct host1x_client *client,
                        err = iommu_attach_group(tegra->domain, group);
                        if (err < 0) {
                                iommu_group_put(group);
-                               return ERR_PTR(err);
+                               return err;
                        }
 
                        if (shared && !tegra->group)
@@ -1103,22 +1102,23 @@ struct iommu_group *host1x_client_iommu_attach(struct host1x_client *client,
                }
        }
 
-       return group;
+       client->group = group;
+
+       return 0;
 }
 
-void host1x_client_iommu_detach(struct host1x_client *client,
-                               struct iommu_group *group)
+void host1x_client_iommu_detach(struct host1x_client *client)
 {
        struct drm_device *drm = dev_get_drvdata(client->parent);
        struct tegra_drm *tegra = drm->dev_private;
 
-       if (group) {
-               if (group == tegra->group) {
-                       iommu_detach_group(tegra->domain, group);
+       if (client->group) {
+               if (client->group == tegra->group) {
+                       iommu_detach_group(tegra->domain, client->group);
                        tegra->group = NULL;
                }
 
-               iommu_group_put(group);
+               iommu_group_put(client->group);
        }
 }
 
index 29911ef..6a06d63 100644 (file)
@@ -100,10 +100,8 @@ int tegra_drm_register_client(struct tegra_drm *tegra,
                              struct tegra_drm_client *client);
 int tegra_drm_unregister_client(struct tegra_drm *tegra,
                                struct tegra_drm_client *client);
-struct iommu_group *host1x_client_iommu_attach(struct host1x_client *client,
-                                              bool shared);
-void host1x_client_iommu_detach(struct host1x_client *client,
-                               struct iommu_group *group);
+int host1x_client_iommu_attach(struct host1x_client *client, bool shared);
+void host1x_client_iommu_detach(struct host1x_client *client);
 
 int tegra_drm_init(struct tegra_drm *tegra, struct drm_device *drm);
 int tegra_drm_exit(struct tegra_drm *tegra);
index 3cbb4a0..5d5af9a 100644 (file)
@@ -17,7 +17,6 @@ struct gr2d_soc {
 };
 
 struct gr2d {
-       struct iommu_group *group;
        struct tegra_drm_client client;
        struct host1x_channel *channel;
        struct clk *clk;
@@ -51,9 +50,8 @@ static int gr2d_init(struct host1x_client *client)
                goto put;
        }
 
-       gr2d->group = host1x_client_iommu_attach(client, false);
-       if (IS_ERR(gr2d->group)) {
-               err = PTR_ERR(gr2d->group);
+       err = host1x_client_iommu_attach(client, false);
+       if (err < 0) {
                dev_err(client->dev, "failed to attach to domain: %d\n", err);
                goto free;
        }
@@ -67,7 +65,7 @@ static int gr2d_init(struct host1x_client *client)
        return 0;
 
 detach:
-       host1x_client_iommu_detach(client, gr2d->group);
+       host1x_client_iommu_detach(client);
 free:
        host1x_syncpt_free(client->syncpts[0]);
 put:
@@ -87,7 +85,7 @@ static int gr2d_exit(struct host1x_client *client)
        if (err < 0)
                return err;
 
-       host1x_client_iommu_detach(client, gr2d->group);
+       host1x_client_iommu_detach(client);
        host1x_syncpt_free(client->syncpts[0]);
        host1x_channel_put(gr2d->channel);
 
index 87a3861..c249a6b 100644 (file)
@@ -23,7 +23,6 @@ struct gr3d_soc {
 };
 
 struct gr3d {
-       struct iommu_group *group;
        struct tegra_drm_client client;
        struct host1x_channel *channel;
        struct clk *clk_secondary;
@@ -60,9 +59,8 @@ static int gr3d_init(struct host1x_client *client)
                goto put;
        }
 
-       gr3d->group = host1x_client_iommu_attach(client, false);
-       if (IS_ERR(gr3d->group)) {
-               err = PTR_ERR(gr3d->group);
+       err = host1x_client_iommu_attach(client, false);
+       if (err < 0) {
                dev_err(client->dev, "failed to attach to domain: %d\n", err);
                goto free;
        }
@@ -76,7 +74,7 @@ static int gr3d_init(struct host1x_client *client)
        return 0;
 
 detach:
-       host1x_client_iommu_detach(client, gr3d->group);
+       host1x_client_iommu_detach(client);
 free:
        host1x_syncpt_free(client->syncpts[0]);
 put:
@@ -95,7 +93,7 @@ static int gr3d_exit(struct host1x_client *client)
        if (err < 0)
                return err;
 
-       host1x_client_iommu_detach(client, gr3d->group);
+       host1x_client_iommu_detach(client);
        host1x_syncpt_free(client->syncpts[0]);
        host1x_channel_put(gr3d->channel);
 
index e4b17c7..d34b1ad 100644 (file)
@@ -34,7 +34,6 @@ struct vic {
        void __iomem *regs;
        struct tegra_drm_client client;
        struct host1x_channel *channel;
-       struct iommu_group *group;
        struct device *dev;
        struct clk *clk;
        struct reset_control *rst;
@@ -188,9 +187,8 @@ static int vic_init(struct host1x_client *client)
        struct vic *vic = to_vic(drm);
        int err;
 
-       vic->group = host1x_client_iommu_attach(client, false);
-       if (IS_ERR(vic->group)) {
-               err = PTR_ERR(vic->group);
+       err = host1x_client_iommu_attach(client, false);
+       if (err < 0) {
                dev_err(vic->dev, "failed to attach to domain: %d\n", err);
                return err;
        }
@@ -224,7 +222,7 @@ free_syncpt:
 free_channel:
        host1x_channel_put(vic->channel);
 detach:
-       host1x_client_iommu_detach(client, vic->group);
+       host1x_client_iommu_detach(client);
 
        return err;
 }
@@ -246,7 +244,7 @@ static int vic_exit(struct host1x_client *client)
 
        host1x_syncpt_free(client->syncpts[0]);
        host1x_channel_put(vic->channel);
-       host1x_client_iommu_detach(client, vic->group);
+       host1x_client_iommu_detach(client);
 
        return 0;
 }
index 4396cd5..df6e613 100644 (file)
@@ -18,6 +18,7 @@ enum host1x_class {
 };
 
 struct host1x_client;
+struct iommu_group;
 
 /**
  * struct host1x_client_ops - host1x client operations
@@ -34,6 +35,7 @@ struct host1x_client_ops {
  * @list: list node for the host1x client
  * @parent: pointer to struct device representing the host1x controller
  * @dev: pointer to struct device backing this host1x client
+ * @group: IOMMU group that this client is a member of
  * @ops: host1x client operations
  * @class: host1x class represented by this client
  * @channel: host1x channel associated with this client
@@ -44,6 +46,7 @@ struct host1x_client {
        struct list_head list;
        struct device *parent;
        struct device *dev;
+       struct iommu_group *group;
 
        const struct host1x_client_ops *ops;