Skip to content

Commit 4cfe6ae

Browse files
agnersShawn Guo
authored andcommitted
clk: imx: vf610: add suspend/resume support
The clock register are lost when enterying LPSTOPx, hence provide suspend/resume functions restoring them. The clock gates get restored by the individual driver, hence we do not need to restore them here. Signed-off-by: Stefan Agner <stefan@agner.ch> Signed-off-by: Shawn Guo <shawnguo@kernel.org>
1 parent 349efbe commit 4cfe6ae

File tree

1 file changed

+48
-0
lines changed

1 file changed

+48
-0
lines changed

drivers/clk/imx/clk-vf610.c

Lines changed: 48 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010

1111
#include <linux/of_address.h>
1212
#include <linux/clk.h>
13+
#include <linux/syscore_ops.h>
1314
#include <dt-bindings/clock/vf610-clock.h>
1415

1516
#include "clk.h"
@@ -40,6 +41,7 @@
4041
#define CCM_CCGR9 (ccm_base + 0x64)
4142
#define CCM_CCGR10 (ccm_base + 0x68)
4243
#define CCM_CCGR11 (ccm_base + 0x6c)
44+
#define CCM_CCGRx(x) (CCM_CCGR0 + (x) * 4)
4345
#define CCM_CMEOR0 (ccm_base + 0x70)
4446
#define CCM_CMEOR1 (ccm_base + 0x74)
4547
#define CCM_CMEOR2 (ccm_base + 0x78)
@@ -115,6 +117,13 @@ static struct clk_div_table pll4_audio_div_table[] = {
115117
static struct clk *clk[VF610_CLK_END];
116118
static struct clk_onecell_data clk_data;
117119

120+
static u32 cscmr1;
121+
static u32 cscmr2;
122+
static u32 cscdr1;
123+
static u32 cscdr2;
124+
static u32 cscdr3;
125+
static u32 ccgr[12];
126+
118127
static unsigned int const clks_init_on[] __initconst = {
119128
VF610_CLK_SYS_BUS,
120129
VF610_CLK_DDR_SEL,
@@ -134,6 +143,43 @@ static struct clk * __init vf610_get_fixed_clock(
134143
return clk;
135144
};
136145

146+
static int vf610_clk_suspend(void)
147+
{
148+
int i;
149+
150+
cscmr1 = readl_relaxed(CCM_CSCMR1);
151+
cscmr2 = readl_relaxed(CCM_CSCMR2);
152+
153+
cscdr1 = readl_relaxed(CCM_CSCDR1);
154+
cscdr2 = readl_relaxed(CCM_CSCDR2);
155+
cscdr3 = readl_relaxed(CCM_CSCDR3);
156+
157+
for (i = 0; i < 12; i++)
158+
ccgr[i] = readl_relaxed(CCM_CCGRx(i));
159+
160+
return 0;
161+
}
162+
163+
static void vf610_clk_resume(void)
164+
{
165+
int i;
166+
167+
writel_relaxed(cscmr1, CCM_CSCMR1);
168+
writel_relaxed(cscmr2, CCM_CSCMR2);
169+
170+
writel_relaxed(cscdr1, CCM_CSCDR1);
171+
writel_relaxed(cscdr2, CCM_CSCDR2);
172+
writel_relaxed(cscdr3, CCM_CSCDR3);
173+
174+
for (i = 0; i < 12; i++)
175+
writel_relaxed(ccgr[i], CCM_CCGRx(i));
176+
}
177+
178+
static struct syscore_ops vf610_clk_syscore_ops = {
179+
.suspend = vf610_clk_suspend,
180+
.resume = vf610_clk_resume,
181+
};
182+
137183
static void __init vf610_clocks_init(struct device_node *ccm_node)
138184
{
139185
struct device_node *np;
@@ -414,6 +460,8 @@ static void __init vf610_clocks_init(struct device_node *ccm_node)
414460
for (i = 0; i < ARRAY_SIZE(clks_init_on); i++)
415461
clk_prepare_enable(clk[clks_init_on[i]]);
416462

463+
register_syscore_ops(&vf610_clk_syscore_ops);
464+
417465
/* Add the clocks to provider list */
418466
clk_data.clks = clk;
419467
clk_data.clk_num = ARRAY_SIZE(clk);

0 commit comments

Comments
 (0)