1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (c) 2024 Neil Armstrong <neil.armstrong@linaro.org>
4  */
5 
6 #include <linux/module.h>
7 #include "vclk.h"
8 
9 /* The VCLK gate has a supplementary reset bit to pulse after ungating */
10 
11 static inline struct meson_vclk_gate_data *
clk_get_meson_vclk_gate_data(struct clk_regmap * clk)12 clk_get_meson_vclk_gate_data(struct clk_regmap *clk)
13 {
14 	return (struct meson_vclk_gate_data *)clk->data;
15 }
16 
meson_vclk_gate_enable(struct clk_hw * hw)17 static int meson_vclk_gate_enable(struct clk_hw *hw)
18 {
19 	struct clk_regmap *clk = to_clk_regmap(hw);
20 	struct meson_vclk_gate_data *vclk = clk_get_meson_vclk_gate_data(clk);
21 
22 	meson_parm_write(clk->map, &vclk->enable, 1);
23 
24 	/* Do a reset pulse */
25 	meson_parm_write(clk->map, &vclk->reset, 1);
26 	meson_parm_write(clk->map, &vclk->reset, 0);
27 
28 	return 0;
29 }
30 
meson_vclk_gate_disable(struct clk_hw * hw)31 static void meson_vclk_gate_disable(struct clk_hw *hw)
32 {
33 	struct clk_regmap *clk = to_clk_regmap(hw);
34 	struct meson_vclk_gate_data *vclk = clk_get_meson_vclk_gate_data(clk);
35 
36 	meson_parm_write(clk->map, &vclk->enable, 0);
37 }
38 
meson_vclk_gate_is_enabled(struct clk_hw * hw)39 static int meson_vclk_gate_is_enabled(struct clk_hw *hw)
40 {
41 	struct clk_regmap *clk = to_clk_regmap(hw);
42 	struct meson_vclk_gate_data *vclk = clk_get_meson_vclk_gate_data(clk);
43 
44 	return meson_parm_read(clk->map, &vclk->enable);
45 }
46 
47 const struct clk_ops meson_vclk_gate_ops = {
48 	.enable = meson_vclk_gate_enable,
49 	.disable = meson_vclk_gate_disable,
50 	.is_enabled = meson_vclk_gate_is_enabled,
51 };
52 EXPORT_SYMBOL_NS_GPL(meson_vclk_gate_ops, CLK_MESON);
53 
54 /* The VCLK Divider has supplementary reset & enable bits */
55 
56 static inline struct meson_vclk_div_data *
clk_get_meson_vclk_div_data(struct clk_regmap * clk)57 clk_get_meson_vclk_div_data(struct clk_regmap *clk)
58 {
59 	return (struct meson_vclk_div_data *)clk->data;
60 }
61 
meson_vclk_div_recalc_rate(struct clk_hw * hw,unsigned long prate)62 static unsigned long meson_vclk_div_recalc_rate(struct clk_hw *hw,
63 						unsigned long prate)
64 {
65 	struct clk_regmap *clk = to_clk_regmap(hw);
66 	struct meson_vclk_div_data *vclk = clk_get_meson_vclk_div_data(clk);
67 
68 	return divider_recalc_rate(hw, prate, meson_parm_read(clk->map, &vclk->div),
69 				   vclk->table, vclk->flags, vclk->div.width);
70 }
71 
meson_vclk_div_determine_rate(struct clk_hw * hw,struct clk_rate_request * req)72 static int meson_vclk_div_determine_rate(struct clk_hw *hw,
73 					 struct clk_rate_request *req)
74 {
75 	struct clk_regmap *clk = to_clk_regmap(hw);
76 	struct meson_vclk_div_data *vclk = clk_get_meson_vclk_div_data(clk);
77 
78 	return divider_determine_rate(hw, req, vclk->table, vclk->div.width,
79 				      vclk->flags);
80 }
81 
meson_vclk_div_set_rate(struct clk_hw * hw,unsigned long rate,unsigned long parent_rate)82 static int meson_vclk_div_set_rate(struct clk_hw *hw, unsigned long rate,
83 				   unsigned long parent_rate)
84 {
85 	struct clk_regmap *clk = to_clk_regmap(hw);
86 	struct meson_vclk_div_data *vclk = clk_get_meson_vclk_div_data(clk);
87 	int ret;
88 
89 	ret = divider_get_val(rate, parent_rate, vclk->table, vclk->div.width,
90 			      vclk->flags);
91 	if (ret < 0)
92 		return ret;
93 
94 	meson_parm_write(clk->map, &vclk->div, ret);
95 
96 	return 0;
97 };
98 
meson_vclk_div_enable(struct clk_hw * hw)99 static int meson_vclk_div_enable(struct clk_hw *hw)
100 {
101 	struct clk_regmap *clk = to_clk_regmap(hw);
102 	struct meson_vclk_div_data *vclk = clk_get_meson_vclk_div_data(clk);
103 
104 	/* Unreset the divider when ungating */
105 	meson_parm_write(clk->map, &vclk->reset, 0);
106 	meson_parm_write(clk->map, &vclk->enable, 1);
107 
108 	return 0;
109 }
110 
meson_vclk_div_disable(struct clk_hw * hw)111 static void meson_vclk_div_disable(struct clk_hw *hw)
112 {
113 	struct clk_regmap *clk = to_clk_regmap(hw);
114 	struct meson_vclk_div_data *vclk = clk_get_meson_vclk_div_data(clk);
115 
116 	/* Reset the divider when gating */
117 	meson_parm_write(clk->map, &vclk->enable, 0);
118 	meson_parm_write(clk->map, &vclk->reset, 1);
119 }
120 
meson_vclk_div_is_enabled(struct clk_hw * hw)121 static int meson_vclk_div_is_enabled(struct clk_hw *hw)
122 {
123 	struct clk_regmap *clk = to_clk_regmap(hw);
124 	struct meson_vclk_div_data *vclk = clk_get_meson_vclk_div_data(clk);
125 
126 	return meson_parm_read(clk->map, &vclk->enable);
127 }
128 
129 const struct clk_ops meson_vclk_div_ops = {
130 	.recalc_rate = meson_vclk_div_recalc_rate,
131 	.determine_rate = meson_vclk_div_determine_rate,
132 	.set_rate = meson_vclk_div_set_rate,
133 	.enable = meson_vclk_div_enable,
134 	.disable = meson_vclk_div_disable,
135 	.is_enabled = meson_vclk_div_is_enabled,
136 };
137 EXPORT_SYMBOL_NS_GPL(meson_vclk_div_ops, CLK_MESON);
138 
139 MODULE_DESCRIPTION("Amlogic vclk clock driver");
140 MODULE_AUTHOR("Neil Armstrong <neil.armstrong@linaro.org>");
141 MODULE_LICENSE("GPL");
142 MODULE_IMPORT_NS(CLK_MESON);
143