Commit d3e4ebc1 authored by Rajan Vaja's avatar Rajan Vaja Committed by Stephen Boyd

drivers: clk: Update clock driver to handle clock attribute

Versal EEMI APIs uses clock device ID which is combination of class,
subclass, type and clock index (e.g. 0x8104006 in which 0-13 bits are
for index(6 in given example), 14-19 bits are for clock type (i.e pll,
out or ref, 1 in given example), 20-25 bits are for subclass which is
nothing but clock type only), 26-32 bits are for device class, which
is clock(0x2) for all clocks) while zynqmp firmware uses clock ID
which is index only (e.g 0, 1, to n, where n is max_clock id).

To use zynqmp clock driver for versal platform also, extend use
of QueryAttribute API to fetch device class, subclass and clock type
to create clock device ID. In case of zynqmp this attributes would be
0 only, so there won't be any effect on clock id as it would use
clock index only.
Signed-off-by: default avatarTejas Patel <tejas.patel@xilinx.com>
Signed-off-by: default avatarRajan Vaja <rajan.vaja@xilinx.com>
Signed-off-by: default avatarMichal Simek <michal.simek@xilinx.com>
Signed-off-by: default avatarJolly Shah <jollys@xilinx.com>
Signed-off-by: default avatarStephen Boyd <sboyd@kernel.org>
parent 60d74e01
...@@ -53,6 +53,10 @@ ...@@ -53,6 +53,10 @@
#define RESERVED_CLK_NAME "" #define RESERVED_CLK_NAME ""
#define CLK_VALID_MASK 0x1 #define CLK_VALID_MASK 0x1
#define NODE_CLASS_SHIFT 26U
#define NODE_SUBCLASS_SHIFT 20U
#define NODE_TYPE_SHIFT 14U
#define NODE_INDEX_SHIFT 0U
enum clk_type { enum clk_type {
CLK_TYPE_OUTPUT, CLK_TYPE_OUTPUT,
...@@ -80,6 +84,7 @@ struct clock_parent { ...@@ -80,6 +84,7 @@ struct clock_parent {
* @num_nodes: Number of nodes present in topology * @num_nodes: Number of nodes present in topology
* @parent: Parent of clock * @parent: Parent of clock
* @num_parents: Number of parents of clock * @num_parents: Number of parents of clock
* @clk_id: Clock id
*/ */
struct zynqmp_clock { struct zynqmp_clock {
char clk_name[MAX_NAME_LEN]; char clk_name[MAX_NAME_LEN];
...@@ -89,6 +94,7 @@ struct zynqmp_clock { ...@@ -89,6 +94,7 @@ struct zynqmp_clock {
u32 num_nodes; u32 num_nodes;
struct clock_parent parent[MAX_PARENT]; struct clock_parent parent[MAX_PARENT];
u32 num_parents; u32 num_parents;
u32 clk_id;
}; };
static const char clk_type_postfix[][10] = { static const char clk_type_postfix[][10] = {
...@@ -396,7 +402,8 @@ static int zynqmp_clock_get_topology(u32 clk_id, ...@@ -396,7 +402,8 @@ static int zynqmp_clock_get_topology(u32 clk_id,
*num_nodes = 0; *num_nodes = 0;
for (j = 0; j <= MAX_NODES; j += 3) { for (j = 0; j <= MAX_NODES; j += 3) {
ret = zynqmp_pm_clock_get_topology(clk_id, j, pm_resp); ret = zynqmp_pm_clock_get_topology(clock[clk_id].clk_id, j,
pm_resp);
if (ret) if (ret)
return ret; return ret;
ret = __zynqmp_clock_get_topology(topology, pm_resp, num_nodes); ret = __zynqmp_clock_get_topology(topology, pm_resp, num_nodes);
...@@ -459,7 +466,8 @@ static int zynqmp_clock_get_parents(u32 clk_id, struct clock_parent *parents, ...@@ -459,7 +466,8 @@ static int zynqmp_clock_get_parents(u32 clk_id, struct clock_parent *parents,
*num_parents = 0; *num_parents = 0;
do { do {
/* Get parents from firmware */ /* Get parents from firmware */
ret = zynqmp_pm_clock_get_parents(clk_id, j, pm_resp); ret = zynqmp_pm_clock_get_parents(clock[clk_id].clk_id, j,
pm_resp);
if (ret) if (ret)
return ret; return ret;
...@@ -528,13 +536,14 @@ static struct clk_hw *zynqmp_register_clk_topology(int clk_id, char *clk_name, ...@@ -528,13 +536,14 @@ static struct clk_hw *zynqmp_register_clk_topology(int clk_id, char *clk_name,
const char **parent_names) const char **parent_names)
{ {
int j; int j;
u32 num_nodes; u32 num_nodes, clk_dev_id;
char *clk_out = NULL; char *clk_out = NULL;
struct clock_topology *nodes; struct clock_topology *nodes;
struct clk_hw *hw = NULL; struct clk_hw *hw = NULL;
nodes = clock[clk_id].node; nodes = clock[clk_id].node;
num_nodes = clock[clk_id].num_nodes; num_nodes = clock[clk_id].num_nodes;
clk_dev_id = clock[clk_id].clk_id;
for (j = 0; j < num_nodes; j++) { for (j = 0; j < num_nodes; j++) {
/* /*
...@@ -551,13 +560,14 @@ static struct clk_hw *zynqmp_register_clk_topology(int clk_id, char *clk_name, ...@@ -551,13 +560,14 @@ static struct clk_hw *zynqmp_register_clk_topology(int clk_id, char *clk_name,
if (!clk_topology[nodes[j].type]) if (!clk_topology[nodes[j].type])
continue; continue;
hw = (*clk_topology[nodes[j].type])(clk_out, clk_id, hw = (*clk_topology[nodes[j].type])(clk_out, clk_dev_id,
parent_names, parent_names,
num_parents, num_parents,
&nodes[j]); &nodes[j]);
if (IS_ERR(hw)) if (IS_ERR(hw))
pr_warn_once("%s() %s register fail with %ld\n", pr_warn_once("%s() 0x%x: %s register fail with %ld\n",
__func__, clk_name, PTR_ERR(hw)); __func__, clk_dev_id, clk_name,
PTR_ERR(hw));
parent_names[0] = clk_out; parent_names[0] = clk_out;
} }
...@@ -621,20 +631,26 @@ static int zynqmp_register_clocks(struct device_node *np) ...@@ -621,20 +631,26 @@ static int zynqmp_register_clocks(struct device_node *np)
static void zynqmp_get_clock_info(void) static void zynqmp_get_clock_info(void)
{ {
int i, ret; int i, ret;
u32 attr, type = 0; u32 attr, type = 0, nodetype, subclass, class;
for (i = 0; i < clock_max_idx; i++) { for (i = 0; i < clock_max_idx; i++) {
zynqmp_pm_clock_get_name(i, clock[i].clk_name);
if (!strcmp(clock[i].clk_name, RESERVED_CLK_NAME))
continue;
ret = zynqmp_pm_clock_get_attributes(i, &attr); ret = zynqmp_pm_clock_get_attributes(i, &attr);
if (ret) if (ret)
continue; continue;
clock[i].valid = attr & CLK_VALID_MASK; clock[i].valid = attr & CLK_VALID_MASK;
clock[i].type = attr >> CLK_TYPE_SHIFT ? CLK_TYPE_EXTERNAL : clock[i].type = ((attr >> CLK_TYPE_SHIFT) & 0x1) ?
CLK_TYPE_OUTPUT; CLK_TYPE_EXTERNAL : CLK_TYPE_OUTPUT;
nodetype = (attr >> NODE_TYPE_SHIFT) & 0x3F;
subclass = (attr >> NODE_SUBCLASS_SHIFT) & 0x3F;
class = (attr >> NODE_CLASS_SHIFT) & 0x3F;
clock[i].clk_id = (class << NODE_CLASS_SHIFT) |
(subclass << NODE_SUBCLASS_SHIFT) |
(nodetype << NODE_TYPE_SHIFT) |
(i << NODE_INDEX_SHIFT);
zynqmp_pm_clock_get_name(clock[i].clk_id, clock[i].clk_name);
} }
/* Get topology of all clock */ /* Get topology of all clock */
......
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