Re: [PATCH 4/4] drm/tegra: Refactor IOMMU attach/detach
On 04.05.2018 16:37, Thierry Reding wrote: > From: Thierry Reding > > Attaching to and detaching from an IOMMU uses the same code sequence in > every driver, so factor it out into separate helpers. > > Signed-off-by: Thierry Reding > --- > drivers/gpu/drm/tegra/dc.c | 42 ++-- > drivers/gpu/drm/tegra/drm.c | 42 > drivers/gpu/drm/tegra/drm.h | 4 > drivers/gpu/drm/tegra/gr2d.c | 32 +++ > drivers/gpu/drm/tegra/gr3d.c | 31 ++ > 5 files changed, 68 insertions(+), 83 deletions(-) > > diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c > index c843f11043db..3e7ec3937346 100644 > --- a/drivers/gpu/drm/tegra/dc.c > +++ b/drivers/gpu/drm/tegra/dc.c > @@ -1837,21 +1837,11 @@ static int tegra_dc_init(struct host1x_client *client) > if (!dc->syncpt) > dev_warn(dc->dev, "failed to allocate syncpoint\n"); > > - if (tegra->domain) { > - dc->group = iommu_group_get(client->dev); > - > - if (dc->group && dc->group != tegra->group) { > - err = iommu_attach_group(tegra->domain, dc->group); > - if (err < 0) { > - dev_err(dc->dev, > - "failed to attach to domain: %d\n", > - err); > - iommu_group_put(dc->group); > - return err; > - } > - > - tegra->group = dc->group; > - } > + dc->group = host1x_client_iommu_attach(client, true); > + if (IS_ERR(dc->group)) { > + err = PTR_ERR(dc->group); > + dev_err(client->dev, "failed to attach to domain: %d\n", err); > + return err; > } > > if (dc->soc->wgrps) > @@ -1916,15 +1906,7 @@ static int tegra_dc_init(struct host1x_client *client) > if (!IS_ERR(primary)) > drm_plane_cleanup(primary); > > - if (dc->group) { > - if (dc->group == tegra->group) { > - iommu_detach_group(tegra->domain, dc->group); > - tegra->group = NULL; > - } > - > - iommu_group_put(dc->group); > - } > - > + host1x_client_iommu_detach(client, dc->group); > host1x_syncpt_free(dc->syncpt); > > return err; > @@ -1932,9 +1914,7 @@ static int tegra_dc_init(struct host1x_client *client) > > static int tegra_dc_exit(struct host1x_client *client) > { > - struct drm_device *drm = dev_get_drvdata(client->parent); > struct tegra_dc *dc = host1x_client_to_dc(client); > - struct tegra_drm *tegra = drm->dev_private; > int err; > > devm_free_irq(dc->dev, dc->irq, dc); > @@ -1945,15 +1925,7 @@ static int tegra_dc_exit(struct host1x_client *client) > return err; > } > > - if (dc->group) { > - if (dc->group == tegra->group) { > - iommu_detach_group(tegra->domain, dc->group); > - tegra->group = NULL; > - } > - > - iommu_group_put(dc->group); > - } > - > + host1x_client_iommu_detach(client, dc->group); > host1x_syncpt_free(dc->syncpt); > > return 0; > diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c > index 7afe2f635f74..bc1008305e1e 100644 > --- a/drivers/gpu/drm/tegra/drm.c > +++ b/drivers/gpu/drm/tegra/drm.c > @@ -1114,6 +1114,48 @@ int tegra_drm_unregister_client(struct tegra_drm > *tegra, > return 0; > } > > +struct iommu_group *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; > + struct iommu_group *group = NULL; > + int err; > + > + if (tegra->domain) { > + group = iommu_group_get(client->dev); > + > + if (group && (!shared || (shared && (group != tegra->group { > + err = iommu_attach_group(tegra->domain, group); > + if (err < 0) { > + iommu_group_put(group); > + return ERR_PTR(err); > + } > + > + if (shared && !tegra->group) > + tegra->group = group; Nit: Probably it doesn't make much sense to have devices unattached to BO's AS, what about to error out here or at least display a error/warning message? @@ -1796,7 +1796,12 @@ struct iommu_group *host1x_client_iommu_attach(struct host1x_client *client, if (tegra->domain) { group = iommu_group_get(client->dev); - if (group && (!shared || (shared && (group != tegra->group { + if (!group) { + dev_err(client->dev, "Fa
Re: [PATCH 4/4] drm/tegra: Refactor IOMMU attach/detach
On 04.05.2018 18:10, Thierry Reding wrote: > On Fri, May 04, 2018 at 05:23:42PM +0300, Dmitry Osipenko wrote: >> On 04.05.2018 16:37, Thierry Reding wrote: >>> From: Thierry Reding >>> >>> Attaching to and detaching from an IOMMU uses the same code sequence in >>> every driver, so factor it out into separate helpers. >>> >>> Signed-off-by: Thierry Reding >>> --- >>> drivers/gpu/drm/tegra/dc.c | 42 ++-- >>> drivers/gpu/drm/tegra/drm.c | 42 >>> drivers/gpu/drm/tegra/drm.h | 4 >>> drivers/gpu/drm/tegra/gr2d.c | 32 +++ >>> drivers/gpu/drm/tegra/gr3d.c | 31 ++ >>> 5 files changed, 68 insertions(+), 83 deletions(-) >>> >>> diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c >>> index c843f11043db..3e7ec3937346 100644 >>> --- a/drivers/gpu/drm/tegra/dc.c >>> +++ b/drivers/gpu/drm/tegra/dc.c >>> @@ -1837,21 +1837,11 @@ static int tegra_dc_init(struct host1x_client >>> *client) >>> if (!dc->syncpt) >>> dev_warn(dc->dev, "failed to allocate syncpoint\n"); >>> >>> - if (tegra->domain) { >>> - dc->group = iommu_group_get(client->dev); >>> - >>> - if (dc->group && dc->group != tegra->group) { >>> - err = iommu_attach_group(tegra->domain, dc->group); >>> - if (err < 0) { >>> - dev_err(dc->dev, >>> - "failed to attach to domain: %d\n", >>> - err); >>> - iommu_group_put(dc->group); >>> - return err; >>> - } >>> - >>> - tegra->group = dc->group; >>> - } >>> + dc->group = host1x_client_iommu_attach(client, true); >>> + if (IS_ERR(dc->group)) { >>> + err = PTR_ERR(dc->group); >>> + dev_err(client->dev, "failed to attach to domain: %d\n", err); >>> + return err; >>> } >>> >>> if (dc->soc->wgrps) >>> @@ -1916,15 +1906,7 @@ static int tegra_dc_init(struct host1x_client >>> *client) >>> if (!IS_ERR(primary)) >>> drm_plane_cleanup(primary); >>> >>> - if (dc->group) { >>> - if (dc->group == tegra->group) { >>> - iommu_detach_group(tegra->domain, dc->group); >>> - tegra->group = NULL; >>> - } >>> - >>> - iommu_group_put(dc->group); >>> - } >>> - >>> + host1x_client_iommu_detach(client, dc->group); >>> host1x_syncpt_free(dc->syncpt); >>> >>> return err; >>> @@ -1932,9 +1914,7 @@ static int tegra_dc_init(struct host1x_client *client) >>> >>> static int tegra_dc_exit(struct host1x_client *client) >>> { >>> - struct drm_device *drm = dev_get_drvdata(client->parent); >>> struct tegra_dc *dc = host1x_client_to_dc(client); >>> - struct tegra_drm *tegra = drm->dev_private; >>> int err; >>> >>> devm_free_irq(dc->dev, dc->irq, dc); >>> @@ -1945,15 +1925,7 @@ static int tegra_dc_exit(struct host1x_client >>> *client) >>> return err; >>> } >>> >>> - if (dc->group) { >>> - if (dc->group == tegra->group) { >>> - iommu_detach_group(tegra->domain, dc->group); >>> - tegra->group = NULL; >>> - } >>> - >>> - iommu_group_put(dc->group); >>> - } >>> - >>> + host1x_client_iommu_detach(client, dc->group); >>> host1x_syncpt_free(dc->syncpt); >>> >>> return 0; >>> diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c >>> index 7afe2f635f74..bc1008305e1e 100644 >>> --- a/drivers/gpu/drm/tegra/drm.c >>> +++ b/drivers/gpu/drm/tegra/drm.c >>> @@ -1114,6 +1114,48 @@ int tegra_drm_unregister_client(struct tegra_drm >>> *tegra, >>> return 0; >>> } >>> >>> +struct iommu_group *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; >>> + struct iommu_group *group = NULL; >>> + int err; >>> + >>> + if (tegra->domain) { >>> + group = iommu_group_get(client->dev); >>> + >>> + if (group && (!shared || (shared && (group != tegra->group { >>> + err = iommu_attach_group(tegra->domain, group); >>> + if (err < 0) { >>> + iommu_group_put(group); >>> + return ERR_PTR(err); >>> + } >>> + >>> + if (shared && !tegra->group) >>> + tegra->group = group; >>> + } >>> + } >>> + >>> + return group; >>> +} >>> + >>> +void host1x_client_iommu_detach(struct host1x_client *client, >>> + struct iommu_group *group) >>> +{ >>> + struct drm_device *drm = dev_get_drvdata(client->parent);
Re: [PATCH 4/4] drm/tegra: Refactor IOMMU attach/detach
On 04.05.2018 16:37, Thierry Reding wrote: > From: Thierry Reding > > Attaching to and detaching from an IOMMU uses the same code sequence in > every driver, so factor it out into separate helpers. > > Signed-off-by: Thierry Reding > --- > drivers/gpu/drm/tegra/dc.c | 42 ++-- > drivers/gpu/drm/tegra/drm.c | 42 > drivers/gpu/drm/tegra/drm.h | 4 > drivers/gpu/drm/tegra/gr2d.c | 32 +++ > drivers/gpu/drm/tegra/gr3d.c | 31 ++ > 5 files changed, 68 insertions(+), 83 deletions(-) > > diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c > index c843f11043db..3e7ec3937346 100644 > --- a/drivers/gpu/drm/tegra/dc.c > +++ b/drivers/gpu/drm/tegra/dc.c > @@ -1837,21 +1837,11 @@ static int tegra_dc_init(struct host1x_client *client) > if (!dc->syncpt) > dev_warn(dc->dev, "failed to allocate syncpoint\n"); > > - if (tegra->domain) { > - dc->group = iommu_group_get(client->dev); > - > - if (dc->group && dc->group != tegra->group) { > - err = iommu_attach_group(tegra->domain, dc->group); > - if (err < 0) { > - dev_err(dc->dev, > - "failed to attach to domain: %d\n", > - err); > - iommu_group_put(dc->group); > - return err; > - } > - > - tegra->group = dc->group; > - } > + dc->group = host1x_client_iommu_attach(client, true); > + if (IS_ERR(dc->group)) { > + err = PTR_ERR(dc->group); > + dev_err(client->dev, "failed to attach to domain: %d\n", err); > + return err; > } > > if (dc->soc->wgrps) > @@ -1916,15 +1906,7 @@ static int tegra_dc_init(struct host1x_client *client) > if (!IS_ERR(primary)) > drm_plane_cleanup(primary); > > - if (dc->group) { > - if (dc->group == tegra->group) { > - iommu_detach_group(tegra->domain, dc->group); > - tegra->group = NULL; > - } > - > - iommu_group_put(dc->group); > - } > - > + host1x_client_iommu_detach(client, dc->group); > host1x_syncpt_free(dc->syncpt); > > return err; > @@ -1932,9 +1914,7 @@ static int tegra_dc_init(struct host1x_client *client) > > static int tegra_dc_exit(struct host1x_client *client) > { > - struct drm_device *drm = dev_get_drvdata(client->parent); > struct tegra_dc *dc = host1x_client_to_dc(client); > - struct tegra_drm *tegra = drm->dev_private; > int err; > > devm_free_irq(dc->dev, dc->irq, dc); > @@ -1945,15 +1925,7 @@ static int tegra_dc_exit(struct host1x_client *client) > return err; > } > > - if (dc->group) { > - if (dc->group == tegra->group) { > - iommu_detach_group(tegra->domain, dc->group); > - tegra->group = NULL; > - } > - > - iommu_group_put(dc->group); > - } > - > + host1x_client_iommu_detach(client, dc->group); > host1x_syncpt_free(dc->syncpt); > > return 0; > diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c > index 7afe2f635f74..bc1008305e1e 100644 > --- a/drivers/gpu/drm/tegra/drm.c > +++ b/drivers/gpu/drm/tegra/drm.c > @@ -1114,6 +1114,48 @@ int tegra_drm_unregister_client(struct tegra_drm > *tegra, > return 0; > } > > +struct iommu_group *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; > + struct iommu_group *group = NULL; > + int err; > + > + if (tegra->domain) { > + group = iommu_group_get(client->dev); > + > + if (group && (!shared || (shared && (group != tegra->group { > + err = iommu_attach_group(tegra->domain, group); > + if (err < 0) { > + iommu_group_put(group); > + return ERR_PTR(err); > + } > + > + if (shared && !tegra->group) > + tegra->group = group; > + } > + } > + > + return group; > +} > + > +void host1x_client_iommu_detach(struct host1x_client *client, > + struct iommu_group *group) > +{ > + 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); > + tegra->group = NULL; > +
Re: [PATCH 4/4] drm/tegra: Refactor IOMMU attach/detach
On Fri, May 04, 2018 at 06:17:45PM +0300, Dmitry Osipenko wrote: > On 04.05.2018 18:10, Thierry Reding wrote: > > On Fri, May 04, 2018 at 05:23:42PM +0300, Dmitry Osipenko wrote: > >> On 04.05.2018 16:37, Thierry Reding wrote: > >>> From: Thierry Reding > >>> > >>> Attaching to and detaching from an IOMMU uses the same code sequence in > >>> every driver, so factor it out into separate helpers. > >>> > >>> Signed-off-by: Thierry Reding > >>> --- > >>> drivers/gpu/drm/tegra/dc.c | 42 ++-- > >>> drivers/gpu/drm/tegra/drm.c | 42 > >>> drivers/gpu/drm/tegra/drm.h | 4 > >>> drivers/gpu/drm/tegra/gr2d.c | 32 +++ > >>> drivers/gpu/drm/tegra/gr3d.c | 31 ++ > >>> 5 files changed, 68 insertions(+), 83 deletions(-) > >>> > >>> diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c > >>> index c843f11043db..3e7ec3937346 100644 > >>> --- a/drivers/gpu/drm/tegra/dc.c > >>> +++ b/drivers/gpu/drm/tegra/dc.c > >>> @@ -1837,21 +1837,11 @@ static int tegra_dc_init(struct host1x_client > >>> *client) > >>> if (!dc->syncpt) > >>> dev_warn(dc->dev, "failed to allocate syncpoint\n"); > >>> > >>> - if (tegra->domain) { > >>> - dc->group = iommu_group_get(client->dev); > >>> - > >>> - if (dc->group && dc->group != tegra->group) { > >>> - err = iommu_attach_group(tegra->domain, dc->group); > >>> - if (err < 0) { > >>> - dev_err(dc->dev, > >>> - "failed to attach to domain: %d\n", > >>> - err); > >>> - iommu_group_put(dc->group); > >>> - return err; > >>> - } > >>> - > >>> - tegra->group = dc->group; > >>> - } > >>> + dc->group = host1x_client_iommu_attach(client, true); > >>> + if (IS_ERR(dc->group)) { > >>> + err = PTR_ERR(dc->group); > >>> + dev_err(client->dev, "failed to attach to domain: %d\n", err); > >>> + return err; > >>> } > >>> > >>> if (dc->soc->wgrps) > >>> @@ -1916,15 +1906,7 @@ static int tegra_dc_init(struct host1x_client > >>> *client) > >>> if (!IS_ERR(primary)) > >>> drm_plane_cleanup(primary); > >>> > >>> - if (dc->group) { > >>> - if (dc->group == tegra->group) { > >>> - iommu_detach_group(tegra->domain, dc->group); > >>> - tegra->group = NULL; > >>> - } > >>> - > >>> - iommu_group_put(dc->group); > >>> - } > >>> - > >>> + host1x_client_iommu_detach(client, dc->group); > >>> host1x_syncpt_free(dc->syncpt); > >>> > >>> return err; > >>> @@ -1932,9 +1914,7 @@ static int tegra_dc_init(struct host1x_client > >>> *client) > >>> > >>> static int tegra_dc_exit(struct host1x_client *client) > >>> { > >>> - struct drm_device *drm = dev_get_drvdata(client->parent); > >>> struct tegra_dc *dc = host1x_client_to_dc(client); > >>> - struct tegra_drm *tegra = drm->dev_private; > >>> int err; > >>> > >>> devm_free_irq(dc->dev, dc->irq, dc); > >>> @@ -1945,15 +1925,7 @@ static int tegra_dc_exit(struct host1x_client > >>> *client) > >>> return err; > >>> } > >>> > >>> - if (dc->group) { > >>> - if (dc->group == tegra->group) { > >>> - iommu_detach_group(tegra->domain, dc->group); > >>> - tegra->group = NULL; > >>> - } > >>> - > >>> - iommu_group_put(dc->group); > >>> - } > >>> - > >>> + host1x_client_iommu_detach(client, dc->group); > >>> host1x_syncpt_free(dc->syncpt); > >>> > >>> return 0; > >>> diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c > >>> index 7afe2f635f74..bc1008305e1e 100644 > >>> --- a/drivers/gpu/drm/tegra/drm.c > >>> +++ b/drivers/gpu/drm/tegra/drm.c > >>> @@ -1114,6 +1114,48 @@ int tegra_drm_unregister_client(struct tegra_drm > >>> *tegra, > >>> return 0; > >>> } > >>> > >>> +struct iommu_group *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; > >>> + struct iommu_group *group = NULL; > >>> + int err; > >>> + > >>> + if (tegra->domain) { > >>> + group = iommu_group_get(client->dev); > >>> + > >>> + if (group && (!shared || (shared && (group != tegra->group { > >>> + err = iommu_attach_group(tegra->domain, group); > >>> + if (err < 0) { > >>> + iommu_group_put(group); > >>> + return ERR_PTR(err); > >>> + } > >>> + > >>> + if (shared && !tegra->group) > >>> + tegra->group = group; > >>> + } > >>> + } > >>> + > >>> + return group; > >>> +} > >>> + > >>
Re: [PATCH 4/4] drm/tegra: Refactor IOMMU attach/detach
On Fri, May 04, 2018 at 05:23:42PM +0300, Dmitry Osipenko wrote: > On 04.05.2018 16:37, Thierry Reding wrote: > > From: Thierry Reding > > > > Attaching to and detaching from an IOMMU uses the same code sequence in > > every driver, so factor it out into separate helpers. > > > > Signed-off-by: Thierry Reding > > --- > > drivers/gpu/drm/tegra/dc.c | 42 ++-- > > drivers/gpu/drm/tegra/drm.c | 42 > > drivers/gpu/drm/tegra/drm.h | 4 > > drivers/gpu/drm/tegra/gr2d.c | 32 +++ > > drivers/gpu/drm/tegra/gr3d.c | 31 ++ > > 5 files changed, 68 insertions(+), 83 deletions(-) > > > > diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c > > index c843f11043db..3e7ec3937346 100644 > > --- a/drivers/gpu/drm/tegra/dc.c > > +++ b/drivers/gpu/drm/tegra/dc.c > > @@ -1837,21 +1837,11 @@ static int tegra_dc_init(struct host1x_client > > *client) > > if (!dc->syncpt) > > dev_warn(dc->dev, "failed to allocate syncpoint\n"); > > > > - if (tegra->domain) { > > - dc->group = iommu_group_get(client->dev); > > - > > - if (dc->group && dc->group != tegra->group) { > > - err = iommu_attach_group(tegra->domain, dc->group); > > - if (err < 0) { > > - dev_err(dc->dev, > > - "failed to attach to domain: %d\n", > > - err); > > - iommu_group_put(dc->group); > > - return err; > > - } > > - > > - tegra->group = dc->group; > > - } > > + dc->group = host1x_client_iommu_attach(client, true); > > + if (IS_ERR(dc->group)) { > > + err = PTR_ERR(dc->group); > > + dev_err(client->dev, "failed to attach to domain: %d\n", err); > > + return err; > > } > > > > if (dc->soc->wgrps) > > @@ -1916,15 +1906,7 @@ static int tegra_dc_init(struct host1x_client > > *client) > > if (!IS_ERR(primary)) > > drm_plane_cleanup(primary); > > > > - if (dc->group) { > > - if (dc->group == tegra->group) { > > - iommu_detach_group(tegra->domain, dc->group); > > - tegra->group = NULL; > > - } > > - > > - iommu_group_put(dc->group); > > - } > > - > > + host1x_client_iommu_detach(client, dc->group); > > host1x_syncpt_free(dc->syncpt); > > > > return err; > > @@ -1932,9 +1914,7 @@ static int tegra_dc_init(struct host1x_client *client) > > > > static int tegra_dc_exit(struct host1x_client *client) > > { > > - struct drm_device *drm = dev_get_drvdata(client->parent); > > struct tegra_dc *dc = host1x_client_to_dc(client); > > - struct tegra_drm *tegra = drm->dev_private; > > int err; > > > > devm_free_irq(dc->dev, dc->irq, dc); > > @@ -1945,15 +1925,7 @@ static int tegra_dc_exit(struct host1x_client > > *client) > > return err; > > } > > > > - if (dc->group) { > > - if (dc->group == tegra->group) { > > - iommu_detach_group(tegra->domain, dc->group); > > - tegra->group = NULL; > > - } > > - > > - iommu_group_put(dc->group); > > - } > > - > > + host1x_client_iommu_detach(client, dc->group); > > host1x_syncpt_free(dc->syncpt); > > > > return 0; > > diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c > > index 7afe2f635f74..bc1008305e1e 100644 > > --- a/drivers/gpu/drm/tegra/drm.c > > +++ b/drivers/gpu/drm/tegra/drm.c > > @@ -1114,6 +1114,48 @@ int tegra_drm_unregister_client(struct tegra_drm > > *tegra, > > return 0; > > } > > > > +struct iommu_group *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; > > + struct iommu_group *group = NULL; > > + int err; > > + > > + if (tegra->domain) { > > + group = iommu_group_get(client->dev); > > + > > + if (group && (!shared || (shared && (group != tegra->group { > > + err = iommu_attach_group(tegra->domain, group); > > + if (err < 0) { > > + iommu_group_put(group); > > + return ERR_PTR(err); > > + } > > + > > + if (shared && !tegra->group) > > + tegra->group = group; > > + } > > + } > > + > > + return group; > > +} > > + > > +void host1x_client_iommu_detach(struct host1x_client *client, > > + struct iommu_group *group) > > +{ > > + struct drm_device *drm = dev_get_drvdata(client->parent); > > + struct tegra_drm *tegra = drm->dev
[PATCH 4/4] drm/tegra: Refactor IOMMU attach/detach
From: Thierry Reding Attaching to and detaching from an IOMMU uses the same code sequence in every driver, so factor it out into separate helpers. Signed-off-by: Thierry Reding --- drivers/gpu/drm/tegra/dc.c | 42 ++-- drivers/gpu/drm/tegra/drm.c | 42 drivers/gpu/drm/tegra/drm.h | 4 drivers/gpu/drm/tegra/gr2d.c | 32 +++ drivers/gpu/drm/tegra/gr3d.c | 31 ++ 5 files changed, 68 insertions(+), 83 deletions(-) diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c index c843f11043db..3e7ec3937346 100644 --- a/drivers/gpu/drm/tegra/dc.c +++ b/drivers/gpu/drm/tegra/dc.c @@ -1837,21 +1837,11 @@ static int tegra_dc_init(struct host1x_client *client) if (!dc->syncpt) dev_warn(dc->dev, "failed to allocate syncpoint\n"); - if (tegra->domain) { - dc->group = iommu_group_get(client->dev); - - if (dc->group && dc->group != tegra->group) { - err = iommu_attach_group(tegra->domain, dc->group); - if (err < 0) { - dev_err(dc->dev, - "failed to attach to domain: %d\n", - err); - iommu_group_put(dc->group); - return err; - } - - tegra->group = dc->group; - } + dc->group = host1x_client_iommu_attach(client, true); + if (IS_ERR(dc->group)) { + err = PTR_ERR(dc->group); + dev_err(client->dev, "failed to attach to domain: %d\n", err); + return err; } if (dc->soc->wgrps) @@ -1916,15 +1906,7 @@ static int tegra_dc_init(struct host1x_client *client) if (!IS_ERR(primary)) drm_plane_cleanup(primary); - if (dc->group) { - if (dc->group == tegra->group) { - iommu_detach_group(tegra->domain, dc->group); - tegra->group = NULL; - } - - iommu_group_put(dc->group); - } - + host1x_client_iommu_detach(client, dc->group); host1x_syncpt_free(dc->syncpt); return err; @@ -1932,9 +1914,7 @@ static int tegra_dc_init(struct host1x_client *client) static int tegra_dc_exit(struct host1x_client *client) { - struct drm_device *drm = dev_get_drvdata(client->parent); struct tegra_dc *dc = host1x_client_to_dc(client); - struct tegra_drm *tegra = drm->dev_private; int err; devm_free_irq(dc->dev, dc->irq, dc); @@ -1945,15 +1925,7 @@ static int tegra_dc_exit(struct host1x_client *client) return err; } - if (dc->group) { - if (dc->group == tegra->group) { - iommu_detach_group(tegra->domain, dc->group); - tegra->group = NULL; - } - - iommu_group_put(dc->group); - } - + host1x_client_iommu_detach(client, dc->group); host1x_syncpt_free(dc->syncpt); return 0; diff --git a/drivers/gpu/drm/tegra/drm.c b/drivers/gpu/drm/tegra/drm.c index 7afe2f635f74..bc1008305e1e 100644 --- a/drivers/gpu/drm/tegra/drm.c +++ b/drivers/gpu/drm/tegra/drm.c @@ -1114,6 +1114,48 @@ int tegra_drm_unregister_client(struct tegra_drm *tegra, return 0; } +struct iommu_group *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; + struct iommu_group *group = NULL; + int err; + + if (tegra->domain) { + group = iommu_group_get(client->dev); + + if (group && (!shared || (shared && (group != tegra->group { + err = iommu_attach_group(tegra->domain, group); + if (err < 0) { + iommu_group_put(group); + return ERR_PTR(err); + } + + if (shared && !tegra->group) + tegra->group = group; + } + } + + return group; +} + +void host1x_client_iommu_detach(struct host1x_client *client, + struct iommu_group *group) +{ + 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); + tegra->group = NULL; + } + + iommu_group_put(group); + } +} + void *tegra_drm_alloc(struct tegra_drm *tegra, size_t size, dma_addr_t *dma) { st