blob: 0be805ca423f06f604099cfb2c2a8e7f2d0a4e80 [file] [log] [blame]
Bryan Wu1394f032007-05-06 14:50:22 -07001/*
2 * File: arch/blackfin/mach-common/pm.c
3 * Based on: arm/mach-omap/pm.c
4 * Author: Cliff Brake <cbrake@accelent.com> Copyright (c) 2001
5 *
6 * Created: 2001
Michael Hennerichcfefe3c2008-02-09 04:12:37 +08007 * Description: Blackfin power management
Bryan Wu1394f032007-05-06 14:50:22 -07008 *
9 * Modified: Nicolas Pitre - PXA250 support
10 * Copyright (c) 2002 Monta Vista Software, Inc.
11 * David Singleton - OMAP1510
12 * Copyright (c) 2002 Monta Vista Software, Inc.
13 * Dirk Behme <dirk.behme@de.bosch.com> - OMAP1510/1610
14 * Copyright 2004
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080015 * Copyright 2004-2008 Analog Devices Inc.
Bryan Wu1394f032007-05-06 14:50:22 -070016 *
17 * Bugs: Enter bugs at http://blackfin.uclinux.org/
18 *
19 * This program is free software; you can redistribute it and/or modify
20 * it under the terms of the GNU General Public License as published by
21 * the Free Software Foundation; either version 2 of the License, or
22 * (at your option) any later version.
23 *
24 * This program is distributed in the hope that it will be useful,
25 * but WITHOUT ANY WARRANTY; without even the implied warranty of
26 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
27 * GNU General Public License for more details.
28 *
29 * You should have received a copy of the GNU General Public License
30 * along with this program; if not, see the file COPYING, or write
31 * to the Free Software Foundation, Inc.,
32 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
33 */
34
Rafael J. Wysocki95d9ffb2007-10-18 03:04:39 -070035#include <linux/suspend.h>
Bryan Wu1394f032007-05-06 14:50:22 -070036#include <linux/sched.h>
37#include <linux/proc_fs.h>
Mike Frysinger1f83b8f2007-07-12 22:58:21 +080038#include <linux/io.h>
39#include <linux/irq.h>
Bryan Wu1394f032007-05-06 14:50:22 -070040
Bryan Wu1394f032007-05-06 14:50:22 -070041#include <asm/dpmc.h>
Michael Hennerichfd923482007-06-11 16:39:40 +080042#include <asm/gpio.h>
Bryan Wu1394f032007-05-06 14:50:22 -070043
44#ifdef CONFIG_PM_WAKEUP_GPIO_POLAR_H
45#define WAKEUP_TYPE PM_WAKE_HIGH
46#endif
47
48#ifdef CONFIG_PM_WAKEUP_GPIO_POLAR_L
49#define WAKEUP_TYPE PM_WAKE_LOW
50#endif
51
52#ifdef CONFIG_PM_WAKEUP_GPIO_POLAR_EDGE_F
53#define WAKEUP_TYPE PM_WAKE_FALLING
54#endif
55
56#ifdef CONFIG_PM_WAKEUP_GPIO_POLAR_EDGE_R
57#define WAKEUP_TYPE PM_WAKE_RISING
58#endif
59
60#ifdef CONFIG_PM_WAKEUP_GPIO_POLAR_EDGE_B
61#define WAKEUP_TYPE PM_WAKE_BOTH_EDGES
62#endif
63
64void bfin_pm_suspend_standby_enter(void)
65{
66#ifdef CONFIG_PM_WAKEUP_BY_GPIO
67 gpio_pm_wakeup_request(CONFIG_PM_WAKEUP_GPIO_NUMBER, WAKEUP_TYPE);
68#endif
69
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080070 u32 flags;
Bryan Wu1394f032007-05-06 14:50:22 -070071
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080072 local_irq_save(flags);
73 bfin_pm_setup();
Bryan Wu1394f032007-05-06 14:50:22 -070074
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080075#ifdef CONFIG_PM_BFIN_SLEEP_DEEPER
76 sleep_deeper(bfin_sic_iwr[0], bfin_sic_iwr[1], bfin_sic_iwr[2]);
Sonic Zhangfb5f0042007-12-23 23:02:13 +080077#else
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080078 sleep_mode(bfin_sic_iwr[0], bfin_sic_iwr[1], bfin_sic_iwr[2]);
Sonic Zhangfb5f0042007-12-23 23:02:13 +080079#endif
Bryan Wu1394f032007-05-06 14:50:22 -070080
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080081 bfin_pm_restore();
Bryan Wu1394f032007-05-06 14:50:22 -070082
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080083#if defined(CONFIG_BF54x) || defined(CONFIG_BF52x) || defined(CONFIG_BF561)
Sonic Zhangfb5f0042007-12-23 23:02:13 +080084 bfin_write_SIC_IWR0(IWR_ENABLE_ALL);
85 bfin_write_SIC_IWR1(IWR_ENABLE_ALL);
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080086# ifdef CONFIG_BF54x
Sonic Zhangfb5f0042007-12-23 23:02:13 +080087 bfin_write_SIC_IWR2(IWR_ENABLE_ALL);
Sonic Zhangfb5f0042007-12-23 23:02:13 +080088# endif
Michael Hennerichcfefe3c2008-02-09 04:12:37 +080089#else
90 bfin_write_SIC_IWR(IWR_ENABLE_ALL);
91#endif
92
93 local_irq_restore(flags);
Bryan Wu1394f032007-05-06 14:50:22 -070094}
95
Bryan Wu1394f032007-05-06 14:50:22 -070096/*
Rafael J. Wysockie6c5eb92007-10-18 03:04:41 -070097 * bfin_pm_valid - Tell the PM core that we only support the standby sleep
98 * state
99 * @state: suspend state we're checking.
Bryan Wu1394f032007-05-06 14:50:22 -0700100 *
101 */
Rafael J. Wysockie6c5eb92007-10-18 03:04:41 -0700102static int bfin_pm_valid(suspend_state_t state)
Bryan Wu1394f032007-05-06 14:50:22 -0700103{
Rafael J. Wysockie6c5eb92007-10-18 03:04:41 -0700104 return (state == PM_SUSPEND_STANDBY);
Bryan Wu1394f032007-05-06 14:50:22 -0700105}
106
107/*
108 * bfin_pm_enter - Actually enter a sleep state.
109 * @state: State we're entering.
110 *
111 */
112static int bfin_pm_enter(suspend_state_t state)
113{
114 switch (state) {
115 case PM_SUSPEND_STANDBY:
116 bfin_pm_suspend_standby_enter();
117 break;
Bryan Wu1394f032007-05-06 14:50:22 -0700118
Bryan Wu9d7b6672007-05-21 18:09:37 +0800119 case PM_SUSPEND_MEM:
Bryan Wu1394f032007-05-06 14:50:22 -0700120 return -ENOTSUPP;
121
122 default:
123 return -EINVAL;
124 }
125
126 return 0;
127}
128
Rafael J. Wysocki26398a72007-10-18 03:04:40 -0700129struct platform_suspend_ops bfin_pm_ops = {
Bryan Wu1394f032007-05-06 14:50:22 -0700130 .enter = bfin_pm_enter,
Michael Hennerich4bbd10f2007-08-27 17:29:10 +0800131 .valid = bfin_pm_valid,
Bryan Wu1394f032007-05-06 14:50:22 -0700132};
133
134static int __init bfin_pm_init(void)
135{
Rafael J. Wysocki26398a72007-10-18 03:04:40 -0700136 suspend_set_ops(&bfin_pm_ops);
Bryan Wu1394f032007-05-06 14:50:22 -0700137 return 0;
138}
139
140__initcall(bfin_pm_init);