Commit 17ecd246 authored by Maxime Ripard's avatar Maxime Ripard Committed by Alexandre Belloni

rtc: sun6i: Add support for the external oscillator gate

The RTC can output its 32kHz clock outside of the SoC, for example to clock
a WiFi chip.

Create a new clock that other devices will be able to retrieve, while
maintaining the DT stability by providing a default name for that clock if
clock-output-names doesn't list one.
Signed-off-by: default avatarMaxime Ripard <maxime.ripard@free-electrons.com>
Signed-off-by: default avatarAlexandre Belloni <alexandre.belloni@free-electrons.com>
parent f22d9cdc
...@@ -10,7 +10,7 @@ Required properties: ...@@ -10,7 +10,7 @@ Required properties:
Required properties for new device trees Required properties for new device trees
- clocks : phandle to the 32kHz external oscillator - clocks : phandle to the 32kHz external oscillator
- clock-output-names : name of the LOSC clock created - clock-output-names : names of the LOSC and its external output clocks created
- #clock-cells : must be equals to 1. The RTC provides two clocks: the - #clock-cells : must be equals to 1. The RTC provides two clocks: the
LOSC and its external output, with index 0 and 1 LOSC and its external output, with index 0 and 1
respectively. respectively.
...@@ -21,7 +21,7 @@ rtc: rtc@01f00000 { ...@@ -21,7 +21,7 @@ rtc: rtc@01f00000 {
compatible = "allwinner,sun6i-a31-rtc"; compatible = "allwinner,sun6i-a31-rtc";
reg = <0x01f00000 0x54>; reg = <0x01f00000 0x54>;
interrupts = <0 40 4>, <0 41 4>; interrupts = <0 40 4>, <0 41 4>;
clock-output-names = "osc32k"; clock-output-names = "osc32k", "osc32k-out";
clocks = <&ext_osc32k>; clocks = <&ext_osc32k>;
#clock-cells = <1>; #clock-cells = <1>;
}; };
...@@ -73,6 +73,9 @@ ...@@ -73,6 +73,9 @@
#define SUN6I_ALARM_CONFIG 0x0050 #define SUN6I_ALARM_CONFIG 0x0050
#define SUN6I_ALARM_CONFIG_WAKEUP BIT(0) #define SUN6I_ALARM_CONFIG_WAKEUP BIT(0)
#define SUN6I_LOSC_OUT_GATING 0x0060
#define SUN6I_LOSC_OUT_GATING_EN BIT(0)
/* /*
* Get date values * Get date values
*/ */
...@@ -125,6 +128,7 @@ struct sun6i_rtc_dev { ...@@ -125,6 +128,7 @@ struct sun6i_rtc_dev {
struct clk_hw hw; struct clk_hw hw;
struct clk_hw *int_osc; struct clk_hw *int_osc;
struct clk *losc; struct clk *losc;
struct clk *ext_losc;
spinlock_t lock; spinlock_t lock;
}; };
...@@ -188,13 +192,14 @@ static void __init sun6i_rtc_clk_init(struct device_node *node) ...@@ -188,13 +192,14 @@ static void __init sun6i_rtc_clk_init(struct device_node *node)
struct clk_init_data init = { struct clk_init_data init = {
.ops = &sun6i_rtc_osc_ops, .ops = &sun6i_rtc_osc_ops,
}; };
const char *clkout_name = "osc32k-out";
const char *parents[2]; const char *parents[2];
rtc = kzalloc(sizeof(*rtc), GFP_KERNEL); rtc = kzalloc(sizeof(*rtc), GFP_KERNEL);
if (!rtc) if (!rtc)
return; return;
clk_data = kzalloc(sizeof(*clk_data) + sizeof(*clk_data->hws), clk_data = kzalloc(sizeof(*clk_data) + (sizeof(*clk_data->hws) * 2),
GFP_KERNEL); GFP_KERNEL);
if (!clk_data) if (!clk_data)
return; return;
...@@ -235,7 +240,8 @@ static void __init sun6i_rtc_clk_init(struct device_node *node) ...@@ -235,7 +240,8 @@ static void __init sun6i_rtc_clk_init(struct device_node *node)
init.parent_names = parents; init.parent_names = parents;
init.num_parents = of_clk_get_parent_count(node) + 1; init.num_parents = of_clk_get_parent_count(node) + 1;
of_property_read_string(node, "clock-output-names", &init.name); of_property_read_string_index(node, "clock-output-names", 0,
&init.name);
rtc->losc = clk_register(NULL, &rtc->hw); rtc->losc = clk_register(NULL, &rtc->hw);
if (IS_ERR(rtc->losc)) { if (IS_ERR(rtc->losc)) {
...@@ -243,8 +249,20 @@ static void __init sun6i_rtc_clk_init(struct device_node *node) ...@@ -243,8 +249,20 @@ static void __init sun6i_rtc_clk_init(struct device_node *node)
return; return;
} }
clk_data->num = 1; of_property_read_string_index(node, "clock-output-names", 1,
&clkout_name);
rtc->ext_losc = clk_register_gate(NULL, clkout_name, rtc->hw.init->name,
0, rtc->base + SUN6I_LOSC_OUT_GATING,
SUN6I_LOSC_OUT_GATING_EN, 0,
&rtc->lock);
if (IS_ERR(rtc->ext_losc)) {
pr_crit("Couldn't register the LOSC external gate\n");
return;
}
clk_data->num = 2;
clk_data->hws[0] = &rtc->hw; clk_data->hws[0] = &rtc->hw;
clk_data->hws[1] = __clk_get_hw(rtc->ext_losc);
of_clk_add_hw_provider(node, of_clk_hw_onecell_get, clk_data); of_clk_add_hw_provider(node, of_clk_hw_onecell_get, clk_data);
return; return;
......
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