1 // SPDX-License-Identifier: GPL-2.0+
3 * Copyright (C) 2016 Freescale Semiconductor, Inc.
4 * Copyright 2017~2018 NXP
6 * Author: Dong Aisheng <aisheng.dong@nxp.com>
10 #include <linux/clk-provider.h>
11 #include <linux/err.h>
13 #include <linux/iopoll.h>
14 #include <linux/slab.h>
19 * struct clk_pfdv2 - IMX PFD clock
21 * @reg: PFD register address
22 * @gate_bit: Gate bit offset
23 * @vld_bit: Valid bit offset
24 * @frac_off: PLL Fractional Divider offset
35 #define to_clk_pfdv2(_hw) container_of(_hw, struct clk_pfdv2, hw)
37 #define CLK_PFDV2_FRAC_MASK 0x3f
39 #define LOCK_TIMEOUT_US USEC_PER_MSEC
41 static DEFINE_SPINLOCK(pfd_lock
);
43 static int clk_pfdv2_wait(struct clk_pfdv2
*pfd
)
47 return readl_poll_timeout(pfd
->reg
, val
, val
& (1 << pfd
->vld_bit
),
51 static int clk_pfdv2_enable(struct clk_hw
*hw
)
53 struct clk_pfdv2
*pfd
= to_clk_pfdv2(hw
);
57 spin_lock_irqsave(&pfd_lock
, flags
);
58 val
= readl_relaxed(pfd
->reg
);
59 val
&= ~(1 << pfd
->gate_bit
);
60 writel_relaxed(val
, pfd
->reg
);
61 spin_unlock_irqrestore(&pfd_lock
, flags
);
63 return clk_pfdv2_wait(pfd
);
66 static void clk_pfdv2_disable(struct clk_hw
*hw
)
68 struct clk_pfdv2
*pfd
= to_clk_pfdv2(hw
);
72 spin_lock_irqsave(&pfd_lock
, flags
);
73 val
= readl_relaxed(pfd
->reg
);
74 val
|= (1 << pfd
->gate_bit
);
75 writel_relaxed(val
, pfd
->reg
);
76 spin_unlock_irqrestore(&pfd_lock
, flags
);
79 static unsigned long clk_pfdv2_recalc_rate(struct clk_hw
*hw
,
80 unsigned long parent_rate
)
82 struct clk_pfdv2
*pfd
= to_clk_pfdv2(hw
);
83 u64 tmp
= parent_rate
;
86 frac
= (readl_relaxed(pfd
->reg
) >> pfd
->frac_off
)
87 & CLK_PFDV2_FRAC_MASK
;
90 pr_debug("clk_pfdv2: %s invalid pfd frac value 0\n",
101 static int clk_pfdv2_determine_rate(struct clk_hw
*hw
,
102 struct clk_rate_request
*req
)
104 unsigned long parent_rates
[] = {
107 req
->best_parent_rate
109 unsigned long best_rate
= -1UL, rate
= req
->rate
;
110 unsigned long best_parent_rate
= req
->best_parent_rate
;
115 for (i
= 0; i
< ARRAY_SIZE(parent_rates
); i
++) {
116 tmp
= parent_rates
[i
];
117 tmp
= tmp
* 18 + rate
/ 2;
126 tmp
= parent_rates
[i
];
130 if (abs(tmp
- req
->rate
) < abs(best_rate
- req
->rate
)) {
132 best_parent_rate
= parent_rates
[i
];
136 req
->best_parent_rate
= best_parent_rate
;
137 req
->rate
= best_rate
;
142 static int clk_pfdv2_is_enabled(struct clk_hw
*hw
)
144 struct clk_pfdv2
*pfd
= to_clk_pfdv2(hw
);
146 if (readl_relaxed(pfd
->reg
) & (1 << pfd
->gate_bit
))
152 static int clk_pfdv2_set_rate(struct clk_hw
*hw
, unsigned long rate
,
153 unsigned long parent_rate
)
155 struct clk_pfdv2
*pfd
= to_clk_pfdv2(hw
);
157 u64 tmp
= parent_rate
;
165 * PFD can NOT change rate without gating.
166 * as the PFDs may enabled in HW by default but no
167 * consumer used it, the enable count is '0', so the
168 * 'SET_RATE_GATE' can NOT help on blocking the set_rate
169 * ops especially for 'assigned-clock-xxx'. In order
170 * to simplify the case, just disable the PFD if it is
171 * enabled in HW but not in SW.
173 if (clk_pfdv2_is_enabled(hw
))
174 clk_pfdv2_disable(hw
);
176 tmp
= tmp
* 18 + rate
/ 2;
184 spin_lock_irqsave(&pfd_lock
, flags
);
185 val
= readl_relaxed(pfd
->reg
);
186 val
&= ~(CLK_PFDV2_FRAC_MASK
<< pfd
->frac_off
);
187 val
|= frac
<< pfd
->frac_off
;
188 writel_relaxed(val
, pfd
->reg
);
189 spin_unlock_irqrestore(&pfd_lock
, flags
);
194 static const struct clk_ops clk_pfdv2_ops
= {
195 .enable
= clk_pfdv2_enable
,
196 .disable
= clk_pfdv2_disable
,
197 .recalc_rate
= clk_pfdv2_recalc_rate
,
198 .determine_rate
= clk_pfdv2_determine_rate
,
199 .set_rate
= clk_pfdv2_set_rate
,
200 .is_enabled
= clk_pfdv2_is_enabled
,
203 struct clk_hw
*imx_clk_hw_pfdv2(enum imx_pfdv2_type type
, const char *name
,
204 const char *parent_name
, void __iomem
*reg
, u8 idx
)
206 struct clk_init_data init
;
207 struct clk_pfdv2
*pfd
;
213 pfd
= kzalloc(sizeof(*pfd
), GFP_KERNEL
);
215 return ERR_PTR(-ENOMEM
);
218 pfd
->gate_bit
= (idx
+ 1) * 8 - 1;
219 pfd
->vld_bit
= pfd
->gate_bit
- 1;
220 pfd
->frac_off
= idx
* 8;
223 init
.ops
= &clk_pfdv2_ops
;
224 init
.parent_names
= &parent_name
;
225 init
.num_parents
= 1;
226 if (type
== IMX_PFDV2_IMX7ULP
)
227 init
.flags
= CLK_SET_RATE_GATE
| CLK_SET_RATE_PARENT
;
229 init
.flags
= CLK_SET_RATE_GATE
;
231 pfd
->hw
.init
= &init
;
234 ret
= clk_hw_register(NULL
, hw
);
242 EXPORT_SYMBOL_GPL(imx_clk_hw_pfdv2
);