blob: 020e6deb67c8c6e9de192905d839b84ec994837d [file] [log] [blame]
Andrew Scullb4b6d4a2019-01-02 15:54:55 +00001/*
2 * i.MX27 Power Management Routines
3 *
4 * Based on Freescale's BSP
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License.
8 */
9
Olivier Deprez157378f2022-04-04 15:47:50 +020010#include <linux/of_address.h>
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000011#include <linux/kernel.h>
12#include <linux/suspend.h>
13#include <linux/io.h>
14
15#include "hardware.h"
16
17static int mx27_suspend_enter(suspend_state_t state)
18{
Olivier Deprez157378f2022-04-04 15:47:50 +020019 void __iomem *ccm_base;
20 struct device_node *np;
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000021 u32 cscr;
Olivier Deprez157378f2022-04-04 15:47:50 +020022
23 np = of_find_compatible_node(NULL, NULL, "fsl,imx27-ccm");
24 ccm_base = of_iomap(np, 0);
25 BUG_ON(!ccm_base);
26
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000027 switch (state) {
28 case PM_SUSPEND_MEM:
29 /* Clear MPEN and SPEN to disable MPLL/SPLL */
Olivier Deprez157378f2022-04-04 15:47:50 +020030 cscr = imx_readl(ccm_base);
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000031 cscr &= 0xFFFFFFFC;
Olivier Deprez157378f2022-04-04 15:47:50 +020032 imx_writel(cscr, ccm_base);
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000033 /* Executes WFI */
34 cpu_do_idle();
35 break;
36
37 default:
38 return -EINVAL;
39 }
40 return 0;
41}
42
43static const struct platform_suspend_ops mx27_suspend_ops = {
44 .enter = mx27_suspend_enter,
45 .valid = suspend_valid_only_mem,
46};
47
48void __init imx27_pm_init(void)
49{
50 suspend_set_ops(&mx27_suspend_ops);
51}