xref: /linux/drivers/clk/rockchip/clk-ddr.c (revision cbecf716ca618fd44feda6bd9a64a8179d031fc5)
1c942fddfSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
2a4f182bfSLin Huang /*
3a4f182bfSLin Huang  * Copyright (c) 2016 Rockchip Electronics Co. Ltd.
4a4f182bfSLin Huang  * Author: Lin Huang <hl@rock-chips.com>
5a4f182bfSLin Huang  */
6a4f182bfSLin Huang 
7a4f182bfSLin Huang #include <linux/arm-smccc.h>
8a4f182bfSLin Huang #include <linux/clk.h>
9a4f182bfSLin Huang #include <linux/clk-provider.h>
10a4f182bfSLin Huang #include <linux/io.h>
11a4f182bfSLin Huang #include <linux/slab.h>
12a4f182bfSLin Huang #include <soc/rockchip/rockchip_sip.h>
13a4f182bfSLin Huang #include "clk.h"
14a4f182bfSLin Huang 
15a4f182bfSLin Huang struct rockchip_ddrclk {
16a4f182bfSLin Huang 	struct clk_hw	hw;
17a4f182bfSLin Huang 	void __iomem	*reg_base;
18a4f182bfSLin Huang 	int		mux_offset;
19a4f182bfSLin Huang 	int		mux_shift;
20a4f182bfSLin Huang 	int		mux_width;
21a4f182bfSLin Huang 	int		div_shift;
22a4f182bfSLin Huang 	int		div_width;
23a4f182bfSLin Huang 	int		ddr_flag;
24a4f182bfSLin Huang 	spinlock_t	*lock;
25a4f182bfSLin Huang };
26a4f182bfSLin Huang 
27a4f182bfSLin Huang #define to_rockchip_ddrclk_hw(hw) container_of(hw, struct rockchip_ddrclk, hw)
28a4f182bfSLin Huang 
rockchip_ddrclk_sip_set_rate(struct clk_hw * hw,unsigned long drate,unsigned long prate)29a4f182bfSLin Huang static int rockchip_ddrclk_sip_set_rate(struct clk_hw *hw, unsigned long drate,
30a4f182bfSLin Huang 					unsigned long prate)
31a4f182bfSLin Huang {
32a4f182bfSLin Huang 	struct rockchip_ddrclk *ddrclk = to_rockchip_ddrclk_hw(hw);
33a4f182bfSLin Huang 	unsigned long flags;
34a4f182bfSLin Huang 	struct arm_smccc_res res;
35a4f182bfSLin Huang 
36a4f182bfSLin Huang 	spin_lock_irqsave(ddrclk->lock, flags);
37a4f182bfSLin Huang 	arm_smccc_smc(ROCKCHIP_SIP_DRAM_FREQ, drate, 0,
38a4f182bfSLin Huang 		      ROCKCHIP_SIP_CONFIG_DRAM_SET_RATE,
39a4f182bfSLin Huang 		      0, 0, 0, 0, &res);
40a4f182bfSLin Huang 	spin_unlock_irqrestore(ddrclk->lock, flags);
41a4f182bfSLin Huang 
42a4f182bfSLin Huang 	return res.a0;
43a4f182bfSLin Huang }
44a4f182bfSLin Huang 
45a4f182bfSLin Huang static unsigned long
rockchip_ddrclk_sip_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)46a4f182bfSLin Huang rockchip_ddrclk_sip_recalc_rate(struct clk_hw *hw,
47a4f182bfSLin Huang 				unsigned long parent_rate)
48a4f182bfSLin Huang {
49a4f182bfSLin Huang 	struct arm_smccc_res res;
50a4f182bfSLin Huang 
51a4f182bfSLin Huang 	arm_smccc_smc(ROCKCHIP_SIP_DRAM_FREQ, 0, 0,
52a4f182bfSLin Huang 		      ROCKCHIP_SIP_CONFIG_DRAM_GET_RATE,
53a4f182bfSLin Huang 		      0, 0, 0, 0, &res);
54a4f182bfSLin Huang 
55a4f182bfSLin Huang 	return res.a0;
56a4f182bfSLin Huang }
57a4f182bfSLin Huang 
rockchip_ddrclk_sip_round_rate(struct clk_hw * hw,unsigned long rate,unsigned long * prate)58a4f182bfSLin Huang static long rockchip_ddrclk_sip_round_rate(struct clk_hw *hw,
59a4f182bfSLin Huang 					   unsigned long rate,
60a4f182bfSLin Huang 					   unsigned long *prate)
61a4f182bfSLin Huang {
62a4f182bfSLin Huang 	struct arm_smccc_res res;
63a4f182bfSLin Huang 
64a4f182bfSLin Huang 	arm_smccc_smc(ROCKCHIP_SIP_DRAM_FREQ, rate, 0,
65a4f182bfSLin Huang 		      ROCKCHIP_SIP_CONFIG_DRAM_ROUND_RATE,
66a4f182bfSLin Huang 		      0, 0, 0, 0, &res);
67a4f182bfSLin Huang 
68a4f182bfSLin Huang 	return res.a0;
69a4f182bfSLin Huang }
70a4f182bfSLin Huang 
rockchip_ddrclk_get_parent(struct clk_hw * hw)71a4f182bfSLin Huang static u8 rockchip_ddrclk_get_parent(struct clk_hw *hw)
72a4f182bfSLin Huang {
73a4f182bfSLin Huang 	struct rockchip_ddrclk *ddrclk = to_rockchip_ddrclk_hw(hw);
74a4f182bfSLin Huang 	u32 val;
75a4f182bfSLin Huang 
765834fd75SJonas Gorski 	val = readl(ddrclk->reg_base +
77a4f182bfSLin Huang 			ddrclk->mux_offset) >> ddrclk->mux_shift;
78a4f182bfSLin Huang 	val &= GENMASK(ddrclk->mux_width - 1, 0);
79a4f182bfSLin Huang 
80a4f182bfSLin Huang 	return val;
81a4f182bfSLin Huang }
82a4f182bfSLin Huang 
83a4f182bfSLin Huang static const struct clk_ops rockchip_ddrclk_sip_ops = {
84a4f182bfSLin Huang 	.recalc_rate = rockchip_ddrclk_sip_recalc_rate,
85a4f182bfSLin Huang 	.set_rate = rockchip_ddrclk_sip_set_rate,
86a4f182bfSLin Huang 	.round_rate = rockchip_ddrclk_sip_round_rate,
87a4f182bfSLin Huang 	.get_parent = rockchip_ddrclk_get_parent,
88a4f182bfSLin Huang };
89a4f182bfSLin Huang 
rockchip_clk_register_ddrclk(const char * name,int flags,const char * const * parent_names,u8 num_parents,int mux_offset,int mux_shift,int mux_width,int div_shift,int div_width,int ddr_flag,void __iomem * reg_base,spinlock_t * lock)90a4f182bfSLin Huang struct clk *rockchip_clk_register_ddrclk(const char *name, int flags,
91a4f182bfSLin Huang 					 const char *const *parent_names,
92a4f182bfSLin Huang 					 u8 num_parents, int mux_offset,
93a4f182bfSLin Huang 					 int mux_shift, int mux_width,
94a4f182bfSLin Huang 					 int div_shift, int div_width,
95a4f182bfSLin Huang 					 int ddr_flag, void __iomem *reg_base,
96a4f182bfSLin Huang 					 spinlock_t *lock)
97a4f182bfSLin Huang {
98a4f182bfSLin Huang 	struct rockchip_ddrclk *ddrclk;
99a4f182bfSLin Huang 	struct clk_init_data init;
100a4f182bfSLin Huang 	struct clk *clk;
101a4f182bfSLin Huang 
102a4f182bfSLin Huang 	ddrclk = kzalloc(sizeof(*ddrclk), GFP_KERNEL);
103a4f182bfSLin Huang 	if (!ddrclk)
104a4f182bfSLin Huang 		return ERR_PTR(-ENOMEM);
105a4f182bfSLin Huang 
106a4f182bfSLin Huang 	init.name = name;
107a4f182bfSLin Huang 	init.parent_names = parent_names;
108a4f182bfSLin Huang 	init.num_parents = num_parents;
109a4f182bfSLin Huang 
110a4f182bfSLin Huang 	init.flags = flags;
111a4f182bfSLin Huang 	init.flags |= CLK_SET_RATE_NO_REPARENT;
112a4f182bfSLin Huang 
113a4f182bfSLin Huang 	switch (ddr_flag) {
114a4f182bfSLin Huang 	case ROCKCHIP_DDRCLK_SIP:
115a4f182bfSLin Huang 		init.ops = &rockchip_ddrclk_sip_ops;
116a4f182bfSLin Huang 		break;
117a4f182bfSLin Huang 	default:
118a4f182bfSLin Huang 		pr_err("%s: unsupported ddrclk type %d\n", __func__, ddr_flag);
119a4f182bfSLin Huang 		kfree(ddrclk);
120a4f182bfSLin Huang 		return ERR_PTR(-EINVAL);
121a4f182bfSLin Huang 	}
122a4f182bfSLin Huang 
123a4f182bfSLin Huang 	ddrclk->reg_base = reg_base;
124a4f182bfSLin Huang 	ddrclk->lock = lock;
125a4f182bfSLin Huang 	ddrclk->hw.init = &init;
126a4f182bfSLin Huang 	ddrclk->mux_offset = mux_offset;
127a4f182bfSLin Huang 	ddrclk->mux_shift = mux_shift;
128a4f182bfSLin Huang 	ddrclk->mux_width = mux_width;
129a4f182bfSLin Huang 	ddrclk->div_shift = div_shift;
130a4f182bfSLin Huang 	ddrclk->div_width = div_width;
131a4f182bfSLin Huang 	ddrclk->ddr_flag = ddr_flag;
132a4f182bfSLin Huang 
133a4f182bfSLin Huang 	clk = clk_register(NULL, &ddrclk->hw);
1344a262b14SShawn Lin 	if (IS_ERR(clk))
135a4f182bfSLin Huang 		kfree(ddrclk);
136a4f182bfSLin Huang 
137a4f182bfSLin Huang 	return clk;
138a4f182bfSLin Huang }
139*f73907deSElaine Zhang EXPORT_SYMBOL_GPL(rockchip_clk_register_ddrclk);
140