diff options
author | Zhonghui Fu <zhonghui.fu@linux.intel.com> | 2015-03-18 10:54:27 -0400 |
---|---|---|
committer | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2015-03-18 10:54:27 -0400 |
commit | 431d452af13720463dda498999b2e9a08729c03a (patch) | |
tree | 581e00b8b20d6fbc60df1d0d0c4390141abe1f4f | |
parent | 1d4a9c17d4d204a159139361e8d4db7f9f267879 (diff) |
PM / sleep: add pm-trace support for suspending phase
Occasionally, the system can't come back up after suspend/resume
due to problems of device suspending phase. This patch make
PM_TRACE infrastructure cover device suspending phase of
suspend/resume process, and the information in RTC can tell
developers which device suspending function make system hang.
Signed-off-by: Zhonghui Fu <zhonghui.fu@linux.intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
-rw-r--r-- | arch/x86/include/asm/pm-trace.h (renamed from arch/x86/include/asm/resume-trace.h) | 10 | ||||
-rw-r--r-- | drivers/base/power/main.c | 20 | ||||
-rw-r--r-- | drivers/base/power/trace.c | 6 | ||||
-rw-r--r-- | include/linux/pm-trace.h (renamed from include/linux/resume-trace.h) | 9 | ||||
-rw-r--r-- | kernel/power/main.c | 2 |
5 files changed, 31 insertions, 16 deletions
diff --git a/arch/x86/include/asm/resume-trace.h b/arch/x86/include/asm/pm-trace.h index 3ff1c2cb1da5..7b7ac42c3661 100644 --- a/arch/x86/include/asm/resume-trace.h +++ b/arch/x86/include/asm/pm-trace.h | |||
@@ -1,5 +1,5 @@ | |||
1 | #ifndef _ASM_X86_RESUME_TRACE_H | 1 | #ifndef _ASM_X86_PM_TRACE_H |
2 | #define _ASM_X86_RESUME_TRACE_H | 2 | #define _ASM_X86_PM_TRACE_H |
3 | 3 | ||
4 | #include <asm/asm.h> | 4 | #include <asm/asm.h> |
5 | 5 | ||
@@ -14,8 +14,10 @@ do { \ | |||
14 | ".previous" \ | 14 | ".previous" \ |
15 | :"=r" (tracedata) \ | 15 | :"=r" (tracedata) \ |
16 | : "i" (__LINE__), "i" (__FILE__)); \ | 16 | : "i" (__LINE__), "i" (__FILE__)); \ |
17 | generate_resume_trace(tracedata, user); \ | 17 | generate_pm_trace(tracedata, user); \ |
18 | } \ | 18 | } \ |
19 | } while (0) | 19 | } while (0) |
20 | 20 | ||
21 | #endif /* _ASM_X86_RESUME_TRACE_H */ | 21 | #define TRACE_SUSPEND(user) TRACE_RESUME(user) |
22 | |||
23 | #endif /* _ASM_X86_PM_TRACE_H */ | ||
diff --git a/drivers/base/power/main.c b/drivers/base/power/main.c index 9717d5f20139..3d874eca7104 100644 --- a/drivers/base/power/main.c +++ b/drivers/base/power/main.c | |||
@@ -23,7 +23,7 @@ | |||
23 | #include <linux/mutex.h> | 23 | #include <linux/mutex.h> |
24 | #include <linux/pm.h> | 24 | #include <linux/pm.h> |
25 | #include <linux/pm_runtime.h> | 25 | #include <linux/pm_runtime.h> |
26 | #include <linux/resume-trace.h> | 26 | #include <linux/pm-trace.h> |
27 | #include <linux/interrupt.h> | 27 | #include <linux/interrupt.h> |
28 | #include <linux/sched.h> | 28 | #include <linux/sched.h> |
29 | #include <linux/async.h> | 29 | #include <linux/async.h> |
@@ -1017,6 +1017,9 @@ static int __device_suspend_noirq(struct device *dev, pm_message_t state, bool a | |||
1017 | char *info = NULL; | 1017 | char *info = NULL; |
1018 | int error = 0; | 1018 | int error = 0; |
1019 | 1019 | ||
1020 | TRACE_DEVICE(dev); | ||
1021 | TRACE_SUSPEND(0); | ||
1022 | |||
1020 | if (async_error) | 1023 | if (async_error) |
1021 | goto Complete; | 1024 | goto Complete; |
1022 | 1025 | ||
@@ -1057,6 +1060,7 @@ static int __device_suspend_noirq(struct device *dev, pm_message_t state, bool a | |||
1057 | 1060 | ||
1058 | Complete: | 1061 | Complete: |
1059 | complete_all(&dev->power.completion); | 1062 | complete_all(&dev->power.completion); |
1063 | TRACE_SUSPEND(error); | ||
1060 | return error; | 1064 | return error; |
1061 | } | 1065 | } |
1062 | 1066 | ||
@@ -1078,7 +1082,7 @@ static int device_suspend_noirq(struct device *dev) | |||
1078 | { | 1082 | { |
1079 | reinit_completion(&dev->power.completion); | 1083 | reinit_completion(&dev->power.completion); |
1080 | 1084 | ||
1081 | if (pm_async_enabled && dev->power.async_suspend) { | 1085 | if (is_async(dev)) { |
1082 | get_device(dev); | 1086 | get_device(dev); |
1083 | async_schedule(async_suspend_noirq, dev); | 1087 | async_schedule(async_suspend_noirq, dev); |
1084 | return 0; | 1088 | return 0; |
@@ -1157,6 +1161,9 @@ static int __device_suspend_late(struct device *dev, pm_message_t state, bool as | |||
1157 | char *info = NULL; | 1161 | char *info = NULL; |
1158 | int error = 0; | 1162 | int error = 0; |
1159 | 1163 | ||
1164 | TRACE_DEVICE(dev); | ||
1165 | TRACE_SUSPEND(0); | ||
1166 | |||
1160 | __pm_runtime_disable(dev, false); | 1167 | __pm_runtime_disable(dev, false); |
1161 | 1168 | ||
1162 | if (async_error) | 1169 | if (async_error) |
@@ -1198,6 +1205,7 @@ static int __device_suspend_late(struct device *dev, pm_message_t state, bool as | |||
1198 | async_error = error; | 1205 | async_error = error; |
1199 | 1206 | ||
1200 | Complete: | 1207 | Complete: |
1208 | TRACE_SUSPEND(error); | ||
1201 | complete_all(&dev->power.completion); | 1209 | complete_all(&dev->power.completion); |
1202 | return error; | 1210 | return error; |
1203 | } | 1211 | } |
@@ -1219,7 +1227,7 @@ static int device_suspend_late(struct device *dev) | |||
1219 | { | 1227 | { |
1220 | reinit_completion(&dev->power.completion); | 1228 | reinit_completion(&dev->power.completion); |
1221 | 1229 | ||
1222 | if (pm_async_enabled && dev->power.async_suspend) { | 1230 | if (is_async(dev)) { |
1223 | get_device(dev); | 1231 | get_device(dev); |
1224 | async_schedule(async_suspend_late, dev); | 1232 | async_schedule(async_suspend_late, dev); |
1225 | return 0; | 1233 | return 0; |
@@ -1338,6 +1346,9 @@ static int __device_suspend(struct device *dev, pm_message_t state, bool async) | |||
1338 | int error = 0; | 1346 | int error = 0; |
1339 | DECLARE_DPM_WATCHDOG_ON_STACK(wd); | 1347 | DECLARE_DPM_WATCHDOG_ON_STACK(wd); |
1340 | 1348 | ||
1349 | TRACE_DEVICE(dev); | ||
1350 | TRACE_SUSPEND(0); | ||
1351 | |||
1341 | dpm_wait_for_children(dev, async); | 1352 | dpm_wait_for_children(dev, async); |
1342 | 1353 | ||
1343 | if (async_error) | 1354 | if (async_error) |
@@ -1444,6 +1455,7 @@ static int __device_suspend(struct device *dev, pm_message_t state, bool async) | |||
1444 | if (error) | 1455 | if (error) |
1445 | async_error = error; | 1456 | async_error = error; |
1446 | 1457 | ||
1458 | TRACE_SUSPEND(error); | ||
1447 | return error; | 1459 | return error; |
1448 | } | 1460 | } |
1449 | 1461 | ||
@@ -1465,7 +1477,7 @@ static int device_suspend(struct device *dev) | |||
1465 | { | 1477 | { |
1466 | reinit_completion(&dev->power.completion); | 1478 | reinit_completion(&dev->power.completion); |
1467 | 1479 | ||
1468 | if (pm_async_enabled && dev->power.async_suspend) { | 1480 | if (is_async(dev)) { |
1469 | get_device(dev); | 1481 | get_device(dev); |
1470 | async_schedule(async_suspend, dev); | 1482 | async_schedule(async_suspend, dev); |
1471 | return 0; | 1483 | return 0; |
diff --git a/drivers/base/power/trace.c b/drivers/base/power/trace.c index d94a1f5121cf..a311cfa4c5bd 100644 --- a/drivers/base/power/trace.c +++ b/drivers/base/power/trace.c | |||
@@ -7,7 +7,7 @@ | |||
7 | * devices may be working. | 7 | * devices may be working. |
8 | */ | 8 | */ |
9 | 9 | ||
10 | #include <linux/resume-trace.h> | 10 | #include <linux/pm-trace.h> |
11 | #include <linux/export.h> | 11 | #include <linux/export.h> |
12 | #include <linux/rtc.h> | 12 | #include <linux/rtc.h> |
13 | 13 | ||
@@ -154,7 +154,7 @@ EXPORT_SYMBOL(set_trace_device); | |||
154 | * it's not any guarantee, but it's a high _likelihood_ that | 154 | * it's not any guarantee, but it's a high _likelihood_ that |
155 | * the match is valid). | 155 | * the match is valid). |
156 | */ | 156 | */ |
157 | void generate_resume_trace(const void *tracedata, unsigned int user) | 157 | void generate_pm_trace(const void *tracedata, unsigned int user) |
158 | { | 158 | { |
159 | unsigned short lineno = *(unsigned short *)tracedata; | 159 | unsigned short lineno = *(unsigned short *)tracedata; |
160 | const char *file = *(const char **)(tracedata + 2); | 160 | const char *file = *(const char **)(tracedata + 2); |
@@ -164,7 +164,7 @@ void generate_resume_trace(const void *tracedata, unsigned int user) | |||
164 | file_hash_value = hash_string(lineno, file, FILEHASH); | 164 | file_hash_value = hash_string(lineno, file, FILEHASH); |
165 | set_magic_time(user_hash_value, file_hash_value, dev_hash_value); | 165 | set_magic_time(user_hash_value, file_hash_value, dev_hash_value); |
166 | } | 166 | } |
167 | EXPORT_SYMBOL(generate_resume_trace); | 167 | EXPORT_SYMBOL(generate_pm_trace); |
168 | 168 | ||
169 | extern char __tracedata_start, __tracedata_end; | 169 | extern char __tracedata_start, __tracedata_end; |
170 | static int show_file_hash(unsigned int value) | 170 | static int show_file_hash(unsigned int value) |
diff --git a/include/linux/resume-trace.h b/include/linux/pm-trace.h index f31db2368782..ecbde7a5548e 100644 --- a/include/linux/resume-trace.h +++ b/include/linux/pm-trace.h | |||
@@ -1,8 +1,8 @@ | |||
1 | #ifndef RESUME_TRACE_H | 1 | #ifndef PM_TRACE_H |
2 | #define RESUME_TRACE_H | 2 | #define PM_TRACE_H |
3 | 3 | ||
4 | #ifdef CONFIG_PM_TRACE | 4 | #ifdef CONFIG_PM_TRACE |
5 | #include <asm/resume-trace.h> | 5 | #include <asm/pm-trace.h> |
6 | #include <linux/types.h> | 6 | #include <linux/types.h> |
7 | 7 | ||
8 | extern int pm_trace_enabled; | 8 | extern int pm_trace_enabled; |
@@ -14,7 +14,7 @@ static inline int pm_trace_is_enabled(void) | |||
14 | 14 | ||
15 | struct device; | 15 | struct device; |
16 | extern void set_trace_device(struct device *); | 16 | extern void set_trace_device(struct device *); |
17 | extern void generate_resume_trace(const void *tracedata, unsigned int user); | 17 | extern void generate_pm_trace(const void *tracedata, unsigned int user); |
18 | extern int show_trace_dev_match(char *buf, size_t size); | 18 | extern int show_trace_dev_match(char *buf, size_t size); |
19 | 19 | ||
20 | #define TRACE_DEVICE(dev) do { \ | 20 | #define TRACE_DEVICE(dev) do { \ |
@@ -28,6 +28,7 @@ static inline int pm_trace_is_enabled(void) { return 0; } | |||
28 | 28 | ||
29 | #define TRACE_DEVICE(dev) do { } while (0) | 29 | #define TRACE_DEVICE(dev) do { } while (0) |
30 | #define TRACE_RESUME(dev) do { } while (0) | 30 | #define TRACE_RESUME(dev) do { } while (0) |
31 | #define TRACE_SUSPEND(dev) do { } while (0) | ||
31 | 32 | ||
32 | #endif | 33 | #endif |
33 | 34 | ||
diff --git a/kernel/power/main.c b/kernel/power/main.c index 9a59d042ea84..86e8157a450f 100644 --- a/kernel/power/main.c +++ b/kernel/power/main.c | |||
@@ -11,7 +11,7 @@ | |||
11 | #include <linux/export.h> | 11 | #include <linux/export.h> |
12 | #include <linux/kobject.h> | 12 | #include <linux/kobject.h> |
13 | #include <linux/string.h> | 13 | #include <linux/string.h> |
14 | #include <linux/resume-trace.h> | 14 | #include <linux/pm-trace.h> |
15 | #include <linux/workqueue.h> | 15 | #include <linux/workqueue.h> |
16 | #include <linux/debugfs.h> | 16 | #include <linux/debugfs.h> |
17 | #include <linux/seq_file.h> | 17 | #include <linux/seq_file.h> |