Re: [PATCH] of: require a match on all fields of of_device_id
On 07/22/2012 08:56 PM, Rob Herring wrote: On 07/18/2012 11:04 AM, Scott Wood wrote: On 07/17/2012 09:38 PM, Rob Herring wrote: On 07/17/2012 08:11 PM, Scott Wood wrote: Commit 107a84e61cdd3406c842a0e4be7efffd3a05dba6 (of: match by compatible property first) breaks the gianfar ethernet driver found on various Freescale PPC chips. You do know this is reverted, right? No, I didn't. I got it via Kumar's next branch, and saw that it was still in your fixes-for-grant branch, and didn't see any revert-related e-mail activity on the devicetree-discuss list about it. I now see that it was reverted directly in Linus's tree (I didn't see either the original or the revert in Linus's tree when I checked, but apparently I hadn't fetched that as recently as I thought). Here's my fix (untested) which is a bit simpler. I'm assuming if we care about which compatible string we are matching to, then we require name and type are blank and we only care about compatible strings. Any particular reason for making that assumption? We should be avoiding the need for .name or .type matching in new bindings, but this seems like unnecessarily inconsistent behavior. Only to ensure we don't change existing behavior and I think trying to cover all possibilities will be nearly impossible. For example, what if we match on both a specific compatible prop alone and a less specific compatible prop plus name and/or type. Which one do we pick as the better match? Whichever one has a compatible string that is listed earlier. Having an additional type/name field doesn't necessarily make it a better match -- it's just there to resolve ambiguity (or sometimes for no good reason at all). You're going to change existing behavior in some case no matter what, short of a match table flag saying it's OK to not keep depending on link order, or just not making the change. Might as well change it to something sane. -Scott ___ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev
Re: [PATCH] of: require a match on all fields of of_device_id
On 07/18/2012 11:04 AM, Scott Wood wrote: On 07/17/2012 09:38 PM, Rob Herring wrote: On 07/17/2012 08:11 PM, Scott Wood wrote: Commit 107a84e61cdd3406c842a0e4be7efffd3a05dba6 (of: match by compatible property first) breaks the gianfar ethernet driver found on various Freescale PPC chips. You do know this is reverted, right? No, I didn't. I got it via Kumar's next branch, and saw that it was still in your fixes-for-grant branch, and didn't see any revert-related e-mail activity on the devicetree-discuss list about it. I now see that it was reverted directly in Linus's tree (I didn't see either the original or the revert in Linus's tree when I checked, but apparently I hadn't fetched that as recently as I thought). Here's my fix (untested) which is a bit simpler. I'm assuming if we care about which compatible string we are matching to, then we require name and type are blank and we only care about compatible strings. Any particular reason for making that assumption? We should be avoiding the need for .name or .type matching in new bindings, but this seems like unnecessarily inconsistent behavior. Only to ensure we don't change existing behavior and I think trying to cover all possibilities will be nearly impossible. For example, what if we match on both a specific compatible prop alone and a less specific compatible prop plus name and/or type. Which one do we pick as the better match? Rob ___ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev
Re: [PATCH] of: require a match on all fields of of_device_id
On 07/17/2012 09:38 PM, Rob Herring wrote: On 07/17/2012 08:11 PM, Scott Wood wrote: Commit 107a84e61cdd3406c842a0e4be7efffd3a05dba6 (of: match by compatible property first) breaks the gianfar ethernet driver found on various Freescale PPC chips. You do know this is reverted, right? No, I didn't. I got it via Kumar's next branch, and saw that it was still in your fixes-for-grant branch, and didn't see any revert-related e-mail activity on the devicetree-discuss list about it. I now see that it was reverted directly in Linus's tree (I didn't see either the original or the revert in Linus's tree when I checked, but apparently I hadn't fetched that as recently as I thought). Here's my fix (untested) which is a bit simpler. I'm assuming if we care about which compatible string we are matching to, then we require name and type are blank and we only care about compatible strings. Any particular reason for making that assumption? We should be avoiding the need for .name or .type matching in new bindings, but this seems like unnecessarily inconsistent behavior. -Scott ___ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev
[PATCH] of: require a match on all fields of of_device_id
Commit 107a84e61cdd3406c842a0e4be7efffd3a05dba6 (of: match by compatible property first) breaks the gianfar ethernet driver found on various Freescale PPC chips. There are, for unfortunate historical reasons, two nodes with a compatible of gianfar. One has a device_type of network and the other has device_type of mdio. The match entries look like this: { .type = mdio, .compatible = gianfar, }, and { .type = network, .compatible = gianfar, }, With the above patch, both nodes get probed by the first driver, because nothing else in the match struct is looked at if there's a compatible match. Signed-off-by: Scott Wood scottw...@freescale.com --- drivers/of/base.c | 44 1 file changed, 32 insertions(+), 12 deletions(-) diff --git a/drivers/of/base.c b/drivers/of/base.c index bc86ea2..4e707cc 100644 --- a/drivers/of/base.c +++ b/drivers/of/base.c @@ -511,14 +511,37 @@ out: } EXPORT_SYMBOL(of_find_node_with_property); -static const struct of_device_id *of_match_compat(const struct of_device_id *matches, - const char *compat) +/* + * Tell if an device_node matches the non-compatible fields of + * a specific of_match element. + */ +static bool of_match_one_noncompat(const struct of_device_id *match, + const struct device_node *node) +{ + bool is_match = true; + + if (match-name[0]) + is_match = node-name !strcmp(match-name, node-name); + if (match-type[0]) + is_match = node-type !strcmp(match-type, node-type); + + return is_match; +} + +/* + * Find an OF match using the supplied compatible string, rather than + * the node's entire string list. + */ +static const struct of_device_id *of_match_compat( + const struct of_device_id *matches, const char *compat, + const struct device_node *node) { while (matches-name[0] || matches-type[0] || matches-compatible[0]) { const char *cp = matches-compatible; int len = strlen(cp); - if (len 0 of_compat_cmp(compat, cp, len) == 0) + if (len 0 of_compat_cmp(compat, cp, len) == 0 + of_match_one_noncompat(matches, node)) return matches; matches++; @@ -544,23 +567,20 @@ const struct of_device_id *of_match_node(const struct of_device_id *matches, return NULL; of_property_for_each_string(node, compatible, prop, cp) { - const struct of_device_id *match = of_match_compat(matches, cp); + const struct of_device_id *match = + of_match_compat(matches, cp, node); if (match) return match; } while (matches-name[0] || matches-type[0] || matches-compatible[0]) { - int match = 1; - if (matches-name[0]) - match = node-name -!strcmp(matches-name, node-name); - if (matches-type[0]) - match = node-type -!strcmp(matches-type, node-type); - if (match !matches-compatible[0]) + if (of_match_one_noncompat(matches, node) + !matches-compatible[0]) return matches; + matches++; } + return NULL; } EXPORT_SYMBOL(of_match_node); -- 1.7.9.5 ___ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev
Re: [PATCH] of: require a match on all fields of of_device_id
On Tue, Jul 17, 2012 at 8:11 PM, Scott Wood scottw...@freescale.com wrote: Commit 107a84e61cdd3406c842a0e4be7efffd3a05dba6 (of: match by compatible property first) breaks the gianfar ethernet driver found on various Freescale PPC chips. There are, for unfortunate historical reasons, two nodes with a compatible of gianfar. Would it be worth updating the binding for the two nodes to make the compatible property different? We could do something like this: ethernet@24000 { reg = 0x24000 0x1000; compatible = fsl,gianfar-eth, gianfar; ... }; (and something similar for MDIO nodes) and update all the drivers to look for both strings. After a few years, we can delete gianfar. -- Timur Tabi Linux kernel developer at Freescale ___ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev
Re: [PATCH] of: require a match on all fields of of_device_id
On 07/17/2012 08:11 PM, Scott Wood wrote: Commit 107a84e61cdd3406c842a0e4be7efffd3a05dba6 (of: match by compatible property first) breaks the gianfar ethernet driver found on various Freescale PPC chips. You do know this is reverted, right? There are, for unfortunate historical reasons, two nodes with a compatible of gianfar. One has a device_type of network and the other has device_type of mdio. The match entries look like this: { .type = mdio, .compatible = gianfar, }, and { .type = network, .compatible = gianfar, }, With the above patch, both nodes get probed by the first driver, because nothing else in the match struct is looked at if there's a compatible match. Signed-off-by: Scott Wood scottw...@freescale.com Here's my fix (untested) which is a bit simpler. I'm assuming if we care about which compatible string we are matching to, then we require name and type are blank and we only care about compatible strings. diff --git a/drivers/of/base.c b/drivers/of/base.c index eada3f4..6e10004 100644 --- a/drivers/of/base.c +++ b/drivers/of/base.c @@ -518,6 +518,9 @@ static const struct of_device_id *of_match_compat(const struct of_device_id *mat const char *cp = matches-compatible; int len = strlen(cp); + if (matches-name[0] || matches-type[0]) + continue; + if (len 0 of_compat_cmp(compat, cp, len) == 0) return matches; @@ -557,7 +560,10 @@ const struct of_device_id *of_match_node(const struct of_device_id *matches, if (matches-type[0]) match = node-type !strcmp(matches-type, node-type); - if (match !matches-compatible[0]) + if (matches-compatible[0]) + match = of_device_is_compatible(node, + matches-compatible); + if (match) return matches; matches++; } --- drivers/of/base.c | 44 1 file changed, 32 insertions(+), 12 deletions(-) diff --git a/drivers/of/base.c b/drivers/of/base.c index bc86ea2..4e707cc 100644 --- a/drivers/of/base.c +++ b/drivers/of/base.c @@ -511,14 +511,37 @@ out: } EXPORT_SYMBOL(of_find_node_with_property); -static const struct of_device_id *of_match_compat(const struct of_device_id *matches, - const char *compat) +/* + * Tell if an device_node matches the non-compatible fields of + * a specific of_match element. + */ +static bool of_match_one_noncompat(const struct of_device_id *match, +const struct device_node *node) +{ + bool is_match = true; + + if (match-name[0]) + is_match = node-name !strcmp(match-name, node-name); + if (match-type[0]) + is_match = node-type !strcmp(match-type, node-type); + + return is_match; +} + +/* + * Find an OF match using the supplied compatible string, rather than + * the node's entire string list. + */ +static const struct of_device_id *of_match_compat( + const struct of_device_id *matches, const char *compat, + const struct device_node *node) { while (matches-name[0] || matches-type[0] || matches-compatible[0]) { const char *cp = matches-compatible; int len = strlen(cp); - if (len 0 of_compat_cmp(compat, cp, len) == 0) + if (len 0 of_compat_cmp(compat, cp, len) == 0 + of_match_one_noncompat(matches, node)) return matches; matches++; @@ -544,23 +567,20 @@ const struct of_device_id *of_match_node(const struct of_device_id *matches, return NULL; of_property_for_each_string(node, compatible, prop, cp) { - const struct of_device_id *match = of_match_compat(matches, cp); + const struct of_device_id *match = + of_match_compat(matches, cp, node); if (match) return match; } while (matches-name[0] || matches-type[0] || matches-compatible[0]) { - int match = 1; - if (matches-name[0]) - match = node-name - !strcmp(matches-name, node-name); - if (matches-type[0]) - match = node-type - !strcmp(matches-type, node-type); - if (match !matches-compatible[0]) + if (of_match_one_noncompat(matches, node) + !matches-compatible[0]) return matches; + matches++; } + return NULL; }