All of lore.kernel.org
 help / color / mirror / Atom feed
From: Liviu Dudau <Liviu.Dudau@arm.com>
To: Russell King - ARM Linux <linux@arm.linux.org.uk>
Cc: Mark Yao <mark.yao@rock-chips.com>,
	Heiko Stuebner <heiko@sntech.de>,
	Philipp Zabel <p.zabel@pengutronix.de>,
	Daniel Vetter <daniel.vetter@ffwll.ch>,
	David Airlie <airlied@linux.ie>, Eric Anholt <eric@anholt.net>,
	linux-rockchip <linux-rockchip@lists.infradead.org>,
	LAKML <linux-arm-kernel@lists.infradead.org>,
	dri-devel <dri-devel@lists.freedesktop.org>,
	LKML <linux-kernel@vger.kernel.org>
Subject: Re: [PATCH 1/2] drm: Improve drm_of_component_probe() to correctly handle ports and remote ports.
Date: Mon, 16 Nov 2015 16:49:07 +0000	[thread overview]
Message-ID: <20151116164907.GA4158@e106497-lin.cambridge.arm.com> (raw)
In-Reply-To: <20151116162241.GN8644@n2100.arm.linux.org.uk>

On Mon, Nov 16, 2015 at 04:22:41PM +0000, Russell King - ARM Linux wrote:
> On Mon, Nov 16, 2015 at 02:44:52PM +0000, Liviu Dudau wrote:
> > Rockchip DRM driver cannot use the same compare_of() function to match
> > ports and remote ports (aka encoders) as their OF sub-trees look different.
> > Add a second compare function to be used when encoders are added to the
> > component framework and patch the existing users of the function
> > accordingly.
> > 
> > Signed-off-by: Liviu Dudau <Liviu.Dudau@arm.com>
> > ---
> >  drivers/gpu/drm/armada/armada_drv.c |  3 ++-
> >  drivers/gpu/drm/drm_of.c            | 23 ++++++++++++++++++-----
> >  drivers/gpu/drm/imx/imx-drm-core.c  |  3 ++-
> >  include/drm/drm_of.h                |  6 ++++--
> >  4 files changed, 26 insertions(+), 9 deletions(-)
> > 
> > diff --git a/drivers/gpu/drm/armada/armada_drv.c b/drivers/gpu/drm/armada/armada_drv.c
> > index 77ab93d..3a2a929 100644
> > --- a/drivers/gpu/drm/armada/armada_drv.c
> > +++ b/drivers/gpu/drm/armada/armada_drv.c
> > @@ -274,7 +274,8 @@ static int armada_drm_probe(struct platform_device *pdev)
> >  	struct device *dev = &pdev->dev;
> >  	int ret;
> >  
> > -	ret = drm_of_component_probe(dev, compare_dev_name, &armada_master_ops);
> > +	ret = drm_of_component_probe(dev, compare_dev_name, compare_dev_name,
> > +				     &armada_master_ops);
> >  	if (ret != -EINVAL)
> >  		return ret;
> >  
> > diff --git a/drivers/gpu/drm/drm_of.c b/drivers/gpu/drm/drm_of.c
> > index 493c05c..58fafd7 100644
> > --- a/drivers/gpu/drm/drm_of.c
> > +++ b/drivers/gpu/drm/drm_of.c
> > @@ -77,7 +77,8 @@ EXPORT_SYMBOL(drm_of_find_possible_crtcs);
> >   * Returns zero if successful, or one of the standard error codes if it fails.
> >   */
> >  int drm_of_component_probe(struct device *dev,
> > -			   int (*compare_of)(struct device *, void *),
> > +			   int (*compare_port)(struct device *, void *),
> > +			   int (*compare_encoder)(struct device *, void *),
> >  			   const struct component_master_ops *m_ops)
> >  {
> >  	struct device_node *ep, *port, *remote;
> > @@ -101,8 +102,14 @@ int drm_of_component_probe(struct device *dev,
> >  			continue;
> >  		}
> >  
> > -		component_match_add(dev, &match, compare_of, port);
> > -		of_node_put(port);
> > +		component_match_add(dev, &match, compare_port, port);
> > +		/*
> > +		 * component_match_add keeps a reference to the port
> > +		 * variable but does not do proper reference counting,
> > +		 * so we cannot release the reference here. If that
> > +		 * gets fixed, the following line should be uncommented
> > +		 */
> > +		/* of_node_put(port); */
> 
> Even if it is fixed, this line should _never_ be uncommented.  This is
> totally the wrong place to drop the reference.

What if (as implied by the comment) component_match_add() does some reference counting
of sorts? (I know it doesn't get used only with OF nodes, it is more generic). I feel
that holding onto a reference to a counted resource without incrementing the use count
is not the right way of doing things, or at least it should be clearly documented in
the interface of component_match_add() so that people understand the mess they are
getting into.


> 
> >  	}
> >  
> >  	if (i == 0) {
> > @@ -140,8 +147,14 @@ int drm_of_component_probe(struct device *dev,
> >  				continue;
> >  			}
> >  
> > -			component_match_add(dev, &match, compare_of, remote);
> > -			of_node_put(remote);
> > +			component_match_add(dev, &match, compare_encoder, remote);
> > +			/*
> > +			 * component_match_add keeps a reference to the port
> > +			 * variable but does not do proper reference counting,
> > +			 * so we cannot release the reference here. If that
> > +			 * gets fixed, the following line should be uncommented
> > +			 */
> > +			/* of_node_put(remote); */
> 
> Ditto.
> 
> The component helper retains a reference (by pointer value) to the 'port'
> or 'remote', which is then subsequently passed into the supplied
> 'compare_encoder' or 'compare_port' method.
> 
> If you drop the reference after adding the match, then that pointer can
> go away and be re-used for something else - and that means it's totally
> useless to the compare functions, since the memory pointed to by it may
> not contain an device_node struct anymore.
> 
> So, it _never_ makes sense to drop the reference at this point.

See my comment above. Keeping a reference to a resource and passing it
on to other functions (even if they are callbacks) should clearly flag the
component framework as one of the refcounters.

> 
> Where it does make sense is when the array of matches is destroyed.  For
> that, we'd need to add a callback to the master ops struct so that the
> master driver can properly release these pointers.
> 
> I'd keep most of the big comments though (up to "... varable") to
> explain why we don't drop the reference.

Sorry, if I understand you correctly, you're saying that we should keep
the following comment:

			/*
			 * component_match_add keeps a reference to the port
			 * variable.
			 */

How does that explain why we don't drop the reference? Did you mean the comment
should be truncated somewhere else by chance (like including the fact the
reference counting is not done)?

Best regards,
Liviu

> 
> -- 
> FTTC broadband for 0.8mile line: currently at 9.6Mbps down 400kbps up
> according to speedtest.net.
> 

-- 
====================
| I would like to |
| fix the world,  |
| but they're not |
| giving me the   |
 \ source code!  /
  ---------------
    ¯\_(ツ)_/¯

WARNING: multiple messages have this Message-ID (diff)
From: Liviu.Dudau@arm.com (Liviu Dudau)
To: linux-arm-kernel@lists.infradead.org
Subject: [PATCH 1/2] drm: Improve drm_of_component_probe() to correctly handle ports and remote ports.
Date: Mon, 16 Nov 2015 16:49:07 +0000	[thread overview]
Message-ID: <20151116164907.GA4158@e106497-lin.cambridge.arm.com> (raw)
In-Reply-To: <20151116162241.GN8644@n2100.arm.linux.org.uk>

On Mon, Nov 16, 2015 at 04:22:41PM +0000, Russell King - ARM Linux wrote:
> On Mon, Nov 16, 2015 at 02:44:52PM +0000, Liviu Dudau wrote:
> > Rockchip DRM driver cannot use the same compare_of() function to match
> > ports and remote ports (aka encoders) as their OF sub-trees look different.
> > Add a second compare function to be used when encoders are added to the
> > component framework and patch the existing users of the function
> > accordingly.
> > 
> > Signed-off-by: Liviu Dudau <Liviu.Dudau@arm.com>
> > ---
> >  drivers/gpu/drm/armada/armada_drv.c |  3 ++-
> >  drivers/gpu/drm/drm_of.c            | 23 ++++++++++++++++++-----
> >  drivers/gpu/drm/imx/imx-drm-core.c  |  3 ++-
> >  include/drm/drm_of.h                |  6 ++++--
> >  4 files changed, 26 insertions(+), 9 deletions(-)
> > 
> > diff --git a/drivers/gpu/drm/armada/armada_drv.c b/drivers/gpu/drm/armada/armada_drv.c
> > index 77ab93d..3a2a929 100644
> > --- a/drivers/gpu/drm/armada/armada_drv.c
> > +++ b/drivers/gpu/drm/armada/armada_drv.c
> > @@ -274,7 +274,8 @@ static int armada_drm_probe(struct platform_device *pdev)
> >  	struct device *dev = &pdev->dev;
> >  	int ret;
> >  
> > -	ret = drm_of_component_probe(dev, compare_dev_name, &armada_master_ops);
> > +	ret = drm_of_component_probe(dev, compare_dev_name, compare_dev_name,
> > +				     &armada_master_ops);
> >  	if (ret != -EINVAL)
> >  		return ret;
> >  
> > diff --git a/drivers/gpu/drm/drm_of.c b/drivers/gpu/drm/drm_of.c
> > index 493c05c..58fafd7 100644
> > --- a/drivers/gpu/drm/drm_of.c
> > +++ b/drivers/gpu/drm/drm_of.c
> > @@ -77,7 +77,8 @@ EXPORT_SYMBOL(drm_of_find_possible_crtcs);
> >   * Returns zero if successful, or one of the standard error codes if it fails.
> >   */
> >  int drm_of_component_probe(struct device *dev,
> > -			   int (*compare_of)(struct device *, void *),
> > +			   int (*compare_port)(struct device *, void *),
> > +			   int (*compare_encoder)(struct device *, void *),
> >  			   const struct component_master_ops *m_ops)
> >  {
> >  	struct device_node *ep, *port, *remote;
> > @@ -101,8 +102,14 @@ int drm_of_component_probe(struct device *dev,
> >  			continue;
> >  		}
> >  
> > -		component_match_add(dev, &match, compare_of, port);
> > -		of_node_put(port);
> > +		component_match_add(dev, &match, compare_port, port);
> > +		/*
> > +		 * component_match_add keeps a reference to the port
> > +		 * variable but does not do proper reference counting,
> > +		 * so we cannot release the reference here. If that
> > +		 * gets fixed, the following line should be uncommented
> > +		 */
> > +		/* of_node_put(port); */
> 
> Even if it is fixed, this line should _never_ be uncommented.  This is
> totally the wrong place to drop the reference.

What if (as implied by the comment) component_match_add() does some reference counting
of sorts? (I know it doesn't get used only with OF nodes, it is more generic). I feel
that holding onto a reference to a counted resource without incrementing the use count
is not the right way of doing things, or at least it should be clearly documented in
the interface of component_match_add() so that people understand the mess they are
getting into.


> 
> >  	}
> >  
> >  	if (i == 0) {
> > @@ -140,8 +147,14 @@ int drm_of_component_probe(struct device *dev,
> >  				continue;
> >  			}
> >  
> > -			component_match_add(dev, &match, compare_of, remote);
> > -			of_node_put(remote);
> > +			component_match_add(dev, &match, compare_encoder, remote);
> > +			/*
> > +			 * component_match_add keeps a reference to the port
> > +			 * variable but does not do proper reference counting,
> > +			 * so we cannot release the reference here. If that
> > +			 * gets fixed, the following line should be uncommented
> > +			 */
> > +			/* of_node_put(remote); */
> 
> Ditto.
> 
> The component helper retains a reference (by pointer value) to the 'port'
> or 'remote', which is then subsequently passed into the supplied
> 'compare_encoder' or 'compare_port' method.
> 
> If you drop the reference after adding the match, then that pointer can
> go away and be re-used for something else - and that means it's totally
> useless to the compare functions, since the memory pointed to by it may
> not contain an device_node struct anymore.
> 
> So, it _never_ makes sense to drop the reference at this point.

See my comment above. Keeping a reference to a resource and passing it
on to other functions (even if they are callbacks) should clearly flag the
component framework as one of the refcounters.

> 
> Where it does make sense is when the array of matches is destroyed.  For
> that, we'd need to add a callback to the master ops struct so that the
> master driver can properly release these pointers.
> 
> I'd keep most of the big comments though (up to "... varable") to
> explain why we don't drop the reference.

Sorry, if I understand you correctly, you're saying that we should keep
the following comment:

			/*
			 * component_match_add keeps a reference to the port
			 * variable.
			 */

How does that explain why we don't drop the reference? Did you mean the comment
should be truncated somewhere else by chance (like including the fact the
reference counting is not done)?

Best regards,
Liviu

> 
> -- 
> FTTC broadband for 0.8mile line: currently at 9.6Mbps down 400kbps up
> according to speedtest.net.
> 

-- 
====================
| I would like to |
| fix the world,  |
| but they're not |
| giving me the   |
 \ source code!  /
  ---------------
    ?\_(?)_/?

  reply	other threads:[~2015-11-16 16:49 UTC|newest]

Thread overview: 48+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2015-11-16 14:44 [PATCH 0/2] Improve drm_of_component_probe() and move rockchip to use it Liviu Dudau
2015-11-16 14:44 ` Liviu Dudau
2015-11-16 14:44 ` Liviu Dudau
2015-11-16 14:44 ` [PATCH 1/2] drm: Improve drm_of_component_probe() to correctly handle ports and remote ports Liviu Dudau
2015-11-16 14:44   ` Liviu Dudau
2015-11-16 14:44   ` Liviu Dudau
2015-11-16 16:22   ` Russell King - ARM Linux
2015-11-16 16:22     ` Russell King - ARM Linux
2015-11-16 16:22     ` Russell King - ARM Linux
2015-11-16 16:49     ` Liviu Dudau [this message]
2015-11-16 16:49       ` Liviu Dudau
2015-11-16 17:22       ` Russell King - ARM Linux
2015-11-16 17:22         ` Russell King - ARM Linux
2015-11-16 17:22         ` Russell King - ARM Linux
2015-11-16 17:32         ` Daniel Stone
2015-11-16 17:32           ` Daniel Stone
2015-11-16 17:32           ` Daniel Stone
2015-11-16 17:43           ` Russell King - ARM Linux
2015-11-16 17:43             ` Russell King - ARM Linux
2015-11-16 17:43             ` Russell King - ARM Linux
2015-11-16 17:48             ` Daniel Stone
2015-11-16 17:48               ` Daniel Stone
2015-11-16 17:48               ` Daniel Stone
2015-11-16 17:53               ` Russell King - ARM Linux
2015-11-16 17:53                 ` Russell King - ARM Linux
2015-11-16 21:17         ` Liviu Dudau
2015-11-16 21:17           ` Liviu Dudau
2015-11-16 21:17           ` Liviu Dudau
2015-11-16 22:33           ` Russell King - ARM Linux
2015-11-16 22:33             ` Russell King - ARM Linux
2015-11-16 14:44 ` [PATCH 2/2] drm/rockchip: Convert the probe function to the generic drm_of_component_probe() Liviu Dudau
2015-11-16 14:44   ` Liviu Dudau
2015-11-16 14:44   ` Liviu Dudau
2015-11-16 16:30   ` Russell King - ARM Linux
2015-11-16 16:30     ` Russell King - ARM Linux
2015-11-16 16:30     ` Russell King - ARM Linux
2015-11-16 16:52     ` Liviu Dudau
2015-11-16 16:52       ` Liviu Dudau
2015-11-16 16:52       ` Liviu Dudau
2015-11-16 17:01       ` Heiko Stübner
2015-11-16 17:01         ` Heiko Stübner
2015-11-16 17:01         ` Heiko Stübner
2015-11-16 16:07 ` [PATCH 0/2] Improve drm_of_component_probe() and move rockchip to use it Heiko Stübner
2015-11-16 16:07   ` Heiko Stübner
2015-11-16 16:07   ` Heiko Stübner
2015-11-16 16:50   ` Liviu Dudau
2015-11-16 16:50     ` Liviu Dudau
2015-11-16 16:50     ` Liviu Dudau

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20151116164907.GA4158@e106497-lin.cambridge.arm.com \
    --to=liviu.dudau@arm.com \
    --cc=airlied@linux.ie \
    --cc=daniel.vetter@ffwll.ch \
    --cc=dri-devel@lists.freedesktop.org \
    --cc=eric@anholt.net \
    --cc=heiko@sntech.de \
    --cc=linux-arm-kernel@lists.infradead.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=linux-rockchip@lists.infradead.org \
    --cc=linux@arm.linux.org.uk \
    --cc=mark.yao@rock-chips.com \
    --cc=p.zabel@pengutronix.de \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.