Commit 1a079560 authored by Stephen Boyd's avatar Stephen Boyd

clk: Cache core in clk_fetch_parent_index() without names

If a clk has specified parents via clk_hw pointers it won't specify the
globally unique names for the parents. Without the unique names, we
can't fallback to comparing them against the name of the 'parent'
pointer here. Therefore, do a pointer comparison against the clk_hw
pointers too and cache the clk_core structure if they match. This fixes
parent lookup code for clks that only specify clk_hw pointers and
nothing else, like muxes that are purely inside a clk controller.

Similarly, if the parent pointer isn't cached after trying to match
clk_core or clk_hw pointers, lookup the pointer from DT or via clkdev
lookups instead of relying purely on the globally unique clk name match.
This should allow us to move away from having to specify global names
for clk parents entirely.

While we're in the area, add some comments so it's clearer what's going
on. The if statements don't lend themselves to much clarity in their raw
form.

Fixes: fc0c209c ("clk: Allow parents to be specified without string names")
Reported-by: default avatarCharles Keepax <ckeepax@opensource.cirrus.com>
Signed-off-by: default avatarStephen Boyd <sboyd@kernel.org>
parent e4818d61
...@@ -327,8 +327,7 @@ static struct clk_core *clk_core_lookup(const char *name) ...@@ -327,8 +327,7 @@ static struct clk_core *clk_core_lookup(const char *name)
/** /**
* clk_core_get - Find the clk_core parent of a clk * clk_core_get - Find the clk_core parent of a clk
* @core: clk to find parent of * @core: clk to find parent of
* @name: name to search for (if string based) * @p_index: parent index to search for
* @index: index to use for search (if DT index based)
* *
* This is the preferred method for clk providers to find the parent of a * This is the preferred method for clk providers to find the parent of a
* clk when that parent is external to the clk controller. The parent_names * clk when that parent is external to the clk controller. The parent_names
...@@ -360,9 +359,10 @@ static struct clk_core *clk_core_lookup(const char *name) ...@@ -360,9 +359,10 @@ static struct clk_core *clk_core_lookup(const char *name)
* provider knows about the clk but it isn't provided on this system. * provider knows about the clk but it isn't provided on this system.
* A valid clk_core pointer when the clk can be found in the provider. * A valid clk_core pointer when the clk can be found in the provider.
*/ */
static struct clk_core *clk_core_get(struct clk_core *core, const char *name, static struct clk_core *clk_core_get(struct clk_core *core, u8 p_index)
int index)
{ {
const char *name = core->parents[p_index].fw_name;
int index = core->parents[p_index].index;
struct clk_hw *hw = ERR_PTR(-ENOENT); struct clk_hw *hw = ERR_PTR(-ENOENT);
struct device *dev = core->dev; struct device *dev = core->dev;
const char *dev_id = dev ? dev_name(dev) : NULL; const char *dev_id = dev ? dev_name(dev) : NULL;
...@@ -400,7 +400,7 @@ static void clk_core_fill_parent_index(struct clk_core *core, u8 index) ...@@ -400,7 +400,7 @@ static void clk_core_fill_parent_index(struct clk_core *core, u8 index)
if (!parent) if (!parent)
parent = ERR_PTR(-EPROBE_DEFER); parent = ERR_PTR(-EPROBE_DEFER);
} else { } else {
parent = clk_core_get(core, entry->fw_name, entry->index); parent = clk_core_get(core, index);
if (IS_ERR(parent) && PTR_ERR(parent) == -ENOENT) if (IS_ERR(parent) && PTR_ERR(parent) == -ENOENT)
parent = clk_core_lookup(entry->name); parent = clk_core_lookup(entry->name);
} }
...@@ -1612,20 +1612,37 @@ static int clk_fetch_parent_index(struct clk_core *core, ...@@ -1612,20 +1612,37 @@ static int clk_fetch_parent_index(struct clk_core *core,
return -EINVAL; return -EINVAL;
for (i = 0; i < core->num_parents; i++) { for (i = 0; i < core->num_parents; i++) {
/* Found it first try! */
if (core->parents[i].core == parent) if (core->parents[i].core == parent)
return i; return i;
/* Something else is here, so keep looking */
if (core->parents[i].core) if (core->parents[i].core)
continue; continue;
/* Fallback to comparing globally unique names */ /* Maybe core hasn't been cached but the hw is all we know? */
if (!strcmp(parent->name, core->parents[i].name)) { if (core->parents[i].hw) {
core->parents[i].core = parent; if (core->parents[i].hw == parent->hw)
return i; break;
/* Didn't match, but we're expecting a clk_hw */
continue;
} }
/* Maybe it hasn't been cached (clk_set_parent() path) */
if (parent == clk_core_get(core, i))
break;
/* Fallback to comparing globally unique names */
if (!strcmp(parent->name, core->parents[i].name))
break;
} }
if (i == core->num_parents)
return -EINVAL; return -EINVAL;
core->parents[i].core = parent;
return i;
} }
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment