1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * clkgen-mux.c: ST GEN-MUX Clock driver
5 * Copyright (C) 2014 STMicroelectronics (R&D) Limited
7 * Authors: Stephen Gallimore <stephen.gallimore@st.com>
8 * Pankaj Dev <pankaj.dev@st.com>
11 #include <linux/slab.h>
13 #include <linux/of_address.h>
14 #include <linux/clk.h>
15 #include <linux/clk-provider.h>
18 static const char ** __init
clkgen_mux_get_parents(struct device_node
*np
,
22 unsigned int nparents
;
24 nparents
= of_clk_get_parent_count(np
);
25 if (WARN_ON(!nparents
))
26 return ERR_PTR(-EINVAL
);
28 parents
= kcalloc(nparents
, sizeof(const char *), GFP_KERNEL
);
30 return ERR_PTR(-ENOMEM
);
32 *num_parents
= of_clk_parent_fill(np
, parents
, nparents
);
36 struct clkgen_mux_data
{
41 unsigned long clk_flags
;
45 static struct clkgen_mux_data stih407_a9_mux_data
= {
49 .lock
= &clkgen_a9_lock
,
52 static void __init
st_of_clkgen_mux_setup(struct device_node
*np
,
53 struct clkgen_mux_data
*data
)
59 struct device_node
*parent_np
;
62 * First check for reg property within the node to keep backward
63 * compatibility, then if reg doesn't exist look at the parent node
65 reg
= of_iomap(np
, 0);
67 parent_np
= of_get_parent(np
);
68 reg
= of_iomap(parent_np
, 0);
69 of_node_put(parent_np
);
71 pr_err("%s: Failed to get base address\n", __func__
);
76 parents
= clkgen_mux_get_parents(np
, &num_parents
);
77 if (IS_ERR(parents
)) {
78 pr_err("%s: Failed to get parents (%ld)\n",
79 __func__
, PTR_ERR(parents
));
83 clk
= clk_register_mux(NULL
, np
->name
, parents
, num_parents
,
84 data
->clk_flags
| CLK_SET_RATE_PARENT
,
86 data
->shift
, data
->width
, data
->mux_flags
,
91 pr_debug("%s: parent %s rate %u\n",
93 __clk_get_name(clk_get_parent(clk
)),
94 (unsigned int)clk_get_rate(clk
));
97 of_clk_add_provider(np
, of_clk_src_simple_get
, clk
);
106 static void __init
st_of_clkgen_a9_mux_setup(struct device_node
*np
)
108 st_of_clkgen_mux_setup(np
, &stih407_a9_mux_data
);
110 CLK_OF_DECLARE(clkgen_a9mux
, "st,stih407-clkgen-a9-mux",
111 st_of_clkgen_a9_mux_setup
);