aboutsummaryrefslogtreecommitdiffstats
path: root/arch/powerpc/sysdev
diff options
context:
space:
mode:
authorJohannes Berg <johannes@sipsolutions.net>2007-03-19 06:53:53 -0400
committerPaul Mackerras <paulus@samba.org>2007-03-25 22:35:14 -0400
commit17e638bc28f2fdc9c0d3eebfb80fce43827b8d12 (patch)
treebaedb1e3e83358ea41331d763dcca560ea4f8e2a /arch/powerpc/sysdev
parentec5f77e789a02adf7c45f03a76455b4e71ae1c5b (diff)
[POWERPC] Generic time suspend/resume code
This removes the time suspend/restore code that was done through a PMU notifier in arch/platforms/powermac/time.c. Instead, introduce arch/powerpc/sysdev/timer.c which creates a sys device and handles time of day suspend/resume through that. This should probably be replaced by using the generic RTC framework but for now it gets rid of the arcane powermac specific hack. Signed-off-by: Johannes Berg <johannes@sipsolutions.net> Acked-by: Benjamin Herrenschmidt <benh@kernel.crashing.org> Signed-off-by: Paul Mackerras <paulus@samba.org>
Diffstat (limited to 'arch/powerpc/sysdev')
-rw-r--r--arch/powerpc/sysdev/Makefile3
-rw-r--r--arch/powerpc/sysdev/timer.c70
2 files changed, 73 insertions, 0 deletions
diff --git a/arch/powerpc/sysdev/Makefile b/arch/powerpc/sysdev/Makefile
index 26ca3ffbc1de..e57379d22b61 100644
--- a/arch/powerpc/sysdev/Makefile
+++ b/arch/powerpc/sysdev/Makefile
@@ -14,6 +14,9 @@ obj-$(CONFIG_FSL_SOC) += fsl_soc.o
14obj-$(CONFIG_TSI108_BRIDGE) += tsi108_pci.o tsi108_dev.o 14obj-$(CONFIG_TSI108_BRIDGE) += tsi108_pci.o tsi108_dev.o
15obj-$(CONFIG_QUICC_ENGINE) += qe_lib/ 15obj-$(CONFIG_QUICC_ENGINE) += qe_lib/
16 16
17# contains only the suspend handler for time
18obj-$(CONFIG_PM) += timer.o
19
17ifeq ($(CONFIG_PPC_MERGE),y) 20ifeq ($(CONFIG_PPC_MERGE),y)
18obj-$(CONFIG_PPC_I8259) += i8259.o 21obj-$(CONFIG_PPC_I8259) += i8259.o
19obj-$(CONFIG_PPC_83xx) += ipic.o 22obj-$(CONFIG_PPC_83xx) += ipic.o
diff --git a/arch/powerpc/sysdev/timer.c b/arch/powerpc/sysdev/timer.c
new file mode 100644
index 000000000000..bdbf8fe520e4
--- /dev/null
+++ b/arch/powerpc/sysdev/timer.c
@@ -0,0 +1,70 @@
1/*
2 * Common code to keep time when machine suspends.
3 *
4 * Copyright 2007 Johannes Berg <johannes@sipsolutions.net>
5 *
6 * GPLv2
7 */
8
9#include <linux/time.h>
10#include <asm/rtc.h>
11
12static unsigned long suspend_rtc_time;
13
14/*
15 * Reset the time after a sleep.
16 */
17static int timer_resume(struct sys_device *dev)
18{
19 struct timeval tv;
20 struct timespec ts;
21 struct rtc_time cur_rtc_tm;
22 unsigned long cur_rtc_time, diff;
23
24 /* get current RTC time and convert to seconds */
25 get_rtc_time(&cur_rtc_tm);
26 rtc_tm_to_time(&cur_rtc_tm, &cur_rtc_time);
27
28 diff = cur_rtc_time - suspend_rtc_time;
29
30 /* adjust time of day by seconds that elapsed while
31 * we were suspended */
32 do_gettimeofday(&tv);
33 ts.tv_sec = tv.tv_sec + diff;
34 ts.tv_nsec = tv.tv_usec * NSEC_PER_USEC;
35 do_settimeofday(&ts);
36
37 return 0;
38}
39
40static int timer_suspend(struct sys_device *dev, pm_message_t state)
41{
42 struct rtc_time suspend_rtc_tm;
43 WARN_ON(!ppc_md.get_rtc_time);
44
45 get_rtc_time(&suspend_rtc_tm);
46 rtc_tm_to_time(&suspend_rtc_tm, &suspend_rtc_time);
47
48 return 0;
49}
50
51static struct sysdev_class timer_sysclass = {
52 .resume = timer_resume,
53 .suspend = timer_suspend,
54 set_kset_name("timer"),
55};
56
57static struct sys_device device_timer = {
58 .id = 0,
59 .cls = &timer_sysclass,
60};
61
62static int time_init_device(void)
63{
64 int error = sysdev_class_register(&timer_sysclass);
65 if (!error)
66 error = sysdev_register(&device_timer);
67 return error;
68}
69
70device_initcall(time_init_device);