Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / drivers / clk / imx / clk-gate-93.c
blobceb56b290394e4951286792309bfbca69cee89f7
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3 * Copyright 2022 NXP
5 * Peng Fan <peng.fan@nxp.com>
6 */
8 #include <linux/clk-provider.h>
9 #include <linux/errno.h>
10 #include <linux/export.h>
11 #include <linux/io.h>
12 #include <linux/iopoll.h>
13 #include <linux/slab.h>
15 #include "clk.h"
17 #define DIRECT_OFFSET 0x0
20 * 0b000 - LPCG will be OFF in any CPU mode.
21 * 0b100 - LPCG will be ON in any CPU mode.
23 #define LPM_SETTING_OFF 0x0
24 #define LPM_SETTING_ON 0x4
26 #define LPM_CUR_OFFSET 0x1c
28 #define AUTHEN_OFFSET 0x30
29 #define CPULPM_EN BIT(2)
30 #define TZ_NS_SHIFT 9
31 #define TZ_NS_MASK BIT(9)
33 #define WHITE_LIST_SHIFT 16
35 struct imx93_clk_gate {
36 struct clk_hw hw;
37 void __iomem *reg;
38 u32 bit_idx;
39 u32 val;
40 u32 mask;
41 spinlock_t *lock;
42 unsigned int *share_count;
45 #define to_imx93_clk_gate(_hw) container_of(_hw, struct imx93_clk_gate, hw)
47 static void imx93_clk_gate_do_hardware(struct clk_hw *hw, bool enable)
49 struct imx93_clk_gate *gate = to_imx93_clk_gate(hw);
50 u32 val;
52 val = readl(gate->reg + AUTHEN_OFFSET);
53 if (val & CPULPM_EN) {
54 val = enable ? LPM_SETTING_ON : LPM_SETTING_OFF;
55 writel(val, gate->reg + LPM_CUR_OFFSET);
56 } else {
57 val = readl(gate->reg + DIRECT_OFFSET);
58 val &= ~(gate->mask << gate->bit_idx);
59 if (enable)
60 val |= (gate->val & gate->mask) << gate->bit_idx;
61 writel(val, gate->reg + DIRECT_OFFSET);
65 static int imx93_clk_gate_enable(struct clk_hw *hw)
67 struct imx93_clk_gate *gate = to_imx93_clk_gate(hw);
68 unsigned long flags;
70 spin_lock_irqsave(gate->lock, flags);
72 if (gate->share_count && (*gate->share_count)++ > 0)
73 goto out;
75 imx93_clk_gate_do_hardware(hw, true);
76 out:
77 spin_unlock_irqrestore(gate->lock, flags);
79 return 0;
82 static void imx93_clk_gate_disable(struct clk_hw *hw)
84 struct imx93_clk_gate *gate = to_imx93_clk_gate(hw);
85 unsigned long flags;
87 spin_lock_irqsave(gate->lock, flags);
89 if (gate->share_count) {
90 if (WARN_ON(*gate->share_count == 0))
91 goto out;
92 else if (--(*gate->share_count) > 0)
93 goto out;
96 imx93_clk_gate_do_hardware(hw, false);
97 out:
98 spin_unlock_irqrestore(gate->lock, flags);
101 static int imx93_clk_gate_reg_is_enabled(struct imx93_clk_gate *gate)
103 u32 val = readl(gate->reg + AUTHEN_OFFSET);
105 if (val & CPULPM_EN) {
106 val = readl(gate->reg + LPM_CUR_OFFSET);
107 if (val == LPM_SETTING_ON)
108 return 1;
109 } else {
110 val = readl(gate->reg);
111 if (((val >> gate->bit_idx) & gate->mask) == gate->val)
112 return 1;
115 return 0;
118 static int imx93_clk_gate_is_enabled(struct clk_hw *hw)
120 struct imx93_clk_gate *gate = to_imx93_clk_gate(hw);
121 unsigned long flags;
122 int ret;
124 spin_lock_irqsave(gate->lock, flags);
126 ret = imx93_clk_gate_reg_is_enabled(gate);
128 spin_unlock_irqrestore(gate->lock, flags);
130 return ret;
133 static void imx93_clk_gate_disable_unused(struct clk_hw *hw)
135 struct imx93_clk_gate *gate = to_imx93_clk_gate(hw);
136 unsigned long flags;
138 spin_lock_irqsave(gate->lock, flags);
140 if (!gate->share_count || *gate->share_count == 0)
141 imx93_clk_gate_do_hardware(hw, false);
143 spin_unlock_irqrestore(gate->lock, flags);
146 static const struct clk_ops imx93_clk_gate_ops = {
147 .enable = imx93_clk_gate_enable,
148 .disable = imx93_clk_gate_disable,
149 .disable_unused = imx93_clk_gate_disable_unused,
150 .is_enabled = imx93_clk_gate_is_enabled,
153 static const struct clk_ops imx93_clk_gate_ro_ops = {
154 .is_enabled = imx93_clk_gate_is_enabled,
157 struct clk_hw *imx93_clk_gate(struct device *dev, const char *name, const char *parent_name,
158 unsigned long flags, void __iomem *reg, u32 bit_idx, u32 val,
159 u32 mask, u32 domain_id, unsigned int *share_count)
161 struct imx93_clk_gate *gate;
162 struct clk_hw *hw;
163 struct clk_init_data init;
164 int ret;
165 u32 authen;
167 gate = kzalloc(sizeof(struct imx93_clk_gate), GFP_KERNEL);
168 if (!gate)
169 return ERR_PTR(-ENOMEM);
171 gate->reg = reg;
172 gate->lock = &imx_ccm_lock;
173 gate->bit_idx = bit_idx;
174 gate->val = val;
175 gate->mask = mask;
176 gate->share_count = share_count;
178 init.name = name;
179 init.ops = &imx93_clk_gate_ops;
180 init.flags = flags | CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE;
181 init.parent_names = parent_name ? &parent_name : NULL;
182 init.num_parents = parent_name ? 1 : 0;
184 gate->hw.init = &init;
185 hw = &gate->hw;
187 authen = readl(reg + AUTHEN_OFFSET);
188 if (!(authen & TZ_NS_MASK) || !(authen & BIT(WHITE_LIST_SHIFT + domain_id)))
189 init.ops = &imx93_clk_gate_ro_ops;
191 ret = clk_hw_register(dev, hw);
192 if (ret) {
193 kfree(gate);
194 return ERR_PTR(ret);
197 return hw;
199 EXPORT_SYMBOL_GPL(imx93_clk_gate);