diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2014-10-09 16:07:43 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-10-09 16:07:43 -0400 |
commit | b528392669415dc1e53a047215e5ad6c2de879fc (patch) | |
tree | d19aa6e1464ef7c7d9f399ac8ec9a7707e5ba6b4 /kernel/power | |
parent | 80213c03c4151d900cf293ef0fc51f8d88495e14 (diff) | |
parent | 9f1a053296953c69d7f23511db9441290cb89e2c (diff) |
Merge tag 'pm+acpi-3.18-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/rafael/linux-pm
Pull ACPI and power management updates from Rafael Wysocki:
"Features-wise, to me the most important this time is a rework of
wakeup interrupts handling in the core that makes them work
consistently across all of the available sleep states, including
suspend-to-idle. Many thanks to Thomas Gleixner for his help with
this work.
Second is an update of the generic PM domains code that has been in
need of some care for quite a while. Unused code is being removed, DT
support is being added and domains are now going to be attached to
devices in bus type code in analogy with the ACPI PM domain. The
majority of work here was done by Ulf Hansson who also has been the
most active developer this time.
Apart from this we have a traditional ACPICA update, this time to
upstream version 20140828 and a few ACPI wakeup interrupts handling
patches on top of the general rework mentioned above. There also are
several cpufreq commits including renaming the cpufreq-cpu0 driver to
cpufreq-dt, as this is what implements generic DT-based cpufreq
support, and a new DT-based idle states infrastructure for cpuidle.
In addition to that, the ACPI LPSS driver is updated, ACPI support for
Apple machines is improved, a few bugs are fixed and a few cleanups
are made all over.
Finally, the Adaptive Voltage Scaling (AVS) subsystem now has a tree
maintained by Kevin Hilman that will be merged through the PM tree.
Numbers-wise, the generic PM domains update takes the lead this time
with 32 non-merge commits, second is cpufreq (15 commits) and the 3rd
place goes to the wakeup interrupts handling rework (13 commits).
Specifics:
- Rework the handling of wakeup IRQs by the IRQ core such that all of
them will be switched over to "wakeup" mode in suspend_device_irqs()
and in that mode the first interrupt will abort system suspend in
progress or wake up the system if already in suspend-to-idle (or
equivalent) without executing any interrupt handlers. Among other
things that eliminates the wakeup-related motivation to use the
IRQF_NO_SUSPEND interrupt flag with interrupts which don't really
need it and should not use it (Thomas Gleixner and Rafael Wysocki)
- Switch over ACPI to handling wakeup interrupts with the help of the
new mechanism introduced by the above IRQ core rework (Rafael Wysocki)
- Rework the core generic PM domains code to eliminate code that's
not used, add DT support and add a generic mechanism by which
devices can be added to PM domains automatically during enumeration
(Ulf Hansson, Geert Uytterhoeven and Tomasz Figa).
- Add debugfs-based mechanics for debugging generic PM domains
(Maciej Matraszek).
- ACPICA update to upstream version 20140828. Included are updates
related to the SRAT and GTDT tables and the _PSx methods are in the
METHOD_NAME list now (Bob Moore and Hanjun Guo).
- Add _OSI("Darwin") support to the ACPI core (unfortunately, that
can't really be done in a straightforward way) to prevent
Thunderbolt from being turned off on Apple systems after boot (or
after resume from system suspend) and rework the ACPI Smart Battery
Subsystem (SBS) driver to work correctly with Apple platforms
(Matthew Garrett and Andreas Noever).
- ACPI LPSS (Low-Power Subsystem) driver update cleaning up the code,
adding support for 133MHz I2C source clock on Intel Baytrail to it
and making it avoid using UART RTS override with Auto Flow Control
(Heikki Krogerus).
- ACPI backlight updates removing the video_set_use_native_backlight
quirk which is not necessary any more, making the code check the
list of output devices returned by the _DOD method to avoid
creating acpi_video interfaces that won't work and adding a quirk
for Lenovo Ideapad Z570 (Hans de Goede, Aaron Lu and Stepan Bujnak)
- New Win8 ACPI OSI quirks for some Dell laptops (Edward Lin)
- Assorted ACPI code cleanups (Fabian Frederick, Rasmus Villemoes,
Sudip Mukherjee, Yijing Wang, and Zhang Rui)
- cpufreq core updates and cleanups (Viresh Kumar, Preeti U Murthy,
Rasmus Villemoes)
- cpufreq driver updates: cpufreq-cpu0/cpufreq-dt (driver name change
among other things), ppc-corenet, powernv (Viresh Kumar, Preeti U
Murthy, Shilpasri G Bhat, Lucas Stach)
- cpuidle support for DT-based idle states infrastructure, new ARM64
cpuidle driver, cpuidle core cleanups (Lorenzo Pieralisi, Rasmus
Villemoes)
- ARM big.LITTLE cpuidle driver updates: support for DT-based
initialization and Exynos5800 compatible string (Lorenzo Pieralisi,
Kevin Hilman)
- Rework of the test_suspend kernel command line argument and a new
trace event for console resume (Srinivas Pandruvada, Todd E Brandt)
- Second attempt to optimize swsusp_free() (hibernation core) to make
it avoid going through all PFNs which may be way too slow on some
systems (Joerg Roedel)
- devfreq updates (Paul Bolle, Punit Agrawal, Ãrjan Eide).
- rockchip-io Adaptive Voltage Scaling (AVS) driver and AVS entry
update in MAINTAINERS (Heiko Stübner, Kevin Hilman)
- PM core fix related to clock management (Geert Uytterhoeven)
- PM core's sysfs code cleanup (Johannes Berg)"
* tag 'pm+acpi-3.18-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/rafael/linux-pm: (105 commits)
ACPI / fan: printk replacement
PM / clk: Fix crash in clocks management code if !CONFIG_PM_RUNTIME
PM / Domains: Rename cpu_data to cpuidle_data
cpufreq: cpufreq-dt: fix potential double put of cpu OF node
cpufreq: cpu0: rename driver and internals to 'cpufreq_dt'
PM / hibernate: Iterate over set bits instead of PFNs in swsusp_free()
cpufreq: ppc-corenet: remove duplicate update of cpu_data
ACPI / sleep: Rework the handling of ACPI GPE wakeup from suspend-to-idle
PM / sleep: Rename platform suspend/resume functions in suspend.c
PM / sleep: Export dpm_suspend_late/noirq() and dpm_resume_early/noirq()
ACPICA: Introduce acpi_enable_all_wakeup_gpes()
ACPICA: Clear all non-wakeup GPEs in acpi_hw_enable_wakeup_gpe_block()
ACPI / video: check _DOD list when creating backlight devices
PM / Domains: Move dev_pm_domain_attach|detach() to pm_domain.h
cpufreq: Replace strnicmp with strncasecmp
cpufreq: powernv: Set the cpus to nominal frequency during reboot/kexec
cpufreq: powernv: Set the pstate of the last hotplugged out cpu in policy->cpus to minimum
cpufreq: Allow stop CPU callback to be used by all cpufreq drivers
PM / devfreq: exynos: Enable building exynos PPMU as module
PM / devfreq: Export helper functions for drivers
...
Diffstat (limited to 'kernel/power')
-rw-r--r-- | kernel/power/Kconfig | 4 | ||||
-rw-r--r-- | kernel/power/process.c | 1 | ||||
-rw-r--r-- | kernel/power/snapshot.c | 54 | ||||
-rw-r--r-- | kernel/power/suspend.c | 51 | ||||
-rw-r--r-- | kernel/power/suspend_test.c | 32 |
5 files changed, 113 insertions, 29 deletions
diff --git a/kernel/power/Kconfig b/kernel/power/Kconfig index e4e4121fa327..bbef57f5bdfd 100644 --- a/kernel/power/Kconfig +++ b/kernel/power/Kconfig | |||
@@ -302,6 +302,10 @@ config PM_GENERIC_DOMAINS_RUNTIME | |||
302 | def_bool y | 302 | def_bool y |
303 | depends on PM_RUNTIME && PM_GENERIC_DOMAINS | 303 | depends on PM_RUNTIME && PM_GENERIC_DOMAINS |
304 | 304 | ||
305 | config PM_GENERIC_DOMAINS_OF | ||
306 | def_bool y | ||
307 | depends on PM_GENERIC_DOMAINS && OF | ||
308 | |||
305 | config CPU_PM | 309 | config CPU_PM |
306 | bool | 310 | bool |
307 | depends on SUSPEND || CPU_IDLE | 311 | depends on SUSPEND || CPU_IDLE |
diff --git a/kernel/power/process.c b/kernel/power/process.c index 4ee194eb524b..7b323221b9ee 100644 --- a/kernel/power/process.c +++ b/kernel/power/process.c | |||
@@ -129,6 +129,7 @@ int freeze_processes(void) | |||
129 | if (!pm_freezing) | 129 | if (!pm_freezing) |
130 | atomic_inc(&system_freezing_cnt); | 130 | atomic_inc(&system_freezing_cnt); |
131 | 131 | ||
132 | pm_wakeup_clear(); | ||
132 | printk("Freezing user space processes ... "); | 133 | printk("Freezing user space processes ... "); |
133 | pm_freezing = true; | 134 | pm_freezing = true; |
134 | error = try_to_freeze_tasks(true); | 135 | error = try_to_freeze_tasks(true); |
diff --git a/kernel/power/snapshot.c b/kernel/power/snapshot.c index f1604d8cf489..791a61892bb5 100644 --- a/kernel/power/snapshot.c +++ b/kernel/power/snapshot.c | |||
@@ -725,6 +725,14 @@ static void memory_bm_clear_bit(struct memory_bitmap *bm, unsigned long pfn) | |||
725 | clear_bit(bit, addr); | 725 | clear_bit(bit, addr); |
726 | } | 726 | } |
727 | 727 | ||
728 | static void memory_bm_clear_current(struct memory_bitmap *bm) | ||
729 | { | ||
730 | int bit; | ||
731 | |||
732 | bit = max(bm->cur.node_bit - 1, 0); | ||
733 | clear_bit(bit, bm->cur.node->data); | ||
734 | } | ||
735 | |||
728 | static int memory_bm_test_bit(struct memory_bitmap *bm, unsigned long pfn) | 736 | static int memory_bm_test_bit(struct memory_bitmap *bm, unsigned long pfn) |
729 | { | 737 | { |
730 | void *addr; | 738 | void *addr; |
@@ -1333,23 +1341,39 @@ static struct memory_bitmap copy_bm; | |||
1333 | 1341 | ||
1334 | void swsusp_free(void) | 1342 | void swsusp_free(void) |
1335 | { | 1343 | { |
1336 | struct zone *zone; | 1344 | unsigned long fb_pfn, fr_pfn; |
1337 | unsigned long pfn, max_zone_pfn; | ||
1338 | 1345 | ||
1339 | for_each_populated_zone(zone) { | 1346 | if (!forbidden_pages_map || !free_pages_map) |
1340 | max_zone_pfn = zone_end_pfn(zone); | 1347 | goto out; |
1341 | for (pfn = zone->zone_start_pfn; pfn < max_zone_pfn; pfn++) | 1348 | |
1342 | if (pfn_valid(pfn)) { | 1349 | memory_bm_position_reset(forbidden_pages_map); |
1343 | struct page *page = pfn_to_page(pfn); | 1350 | memory_bm_position_reset(free_pages_map); |
1344 | 1351 | ||
1345 | if (swsusp_page_is_forbidden(page) && | 1352 | loop: |
1346 | swsusp_page_is_free(page)) { | 1353 | fr_pfn = memory_bm_next_pfn(free_pages_map); |
1347 | swsusp_unset_page_forbidden(page); | 1354 | fb_pfn = memory_bm_next_pfn(forbidden_pages_map); |
1348 | swsusp_unset_page_free(page); | 1355 | |
1349 | __free_page(page); | 1356 | /* |
1350 | } | 1357 | * Find the next bit set in both bitmaps. This is guaranteed to |
1351 | } | 1358 | * terminate when fb_pfn == fr_pfn == BM_END_OF_MAP. |
1359 | */ | ||
1360 | do { | ||
1361 | if (fb_pfn < fr_pfn) | ||
1362 | fb_pfn = memory_bm_next_pfn(forbidden_pages_map); | ||
1363 | if (fr_pfn < fb_pfn) | ||
1364 | fr_pfn = memory_bm_next_pfn(free_pages_map); | ||
1365 | } while (fb_pfn != fr_pfn); | ||
1366 | |||
1367 | if (fr_pfn != BM_END_OF_MAP && pfn_valid(fr_pfn)) { | ||
1368 | struct page *page = pfn_to_page(fr_pfn); | ||
1369 | |||
1370 | memory_bm_clear_current(forbidden_pages_map); | ||
1371 | memory_bm_clear_current(free_pages_map); | ||
1372 | __free_page(page); | ||
1373 | goto loop; | ||
1352 | } | 1374 | } |
1375 | |||
1376 | out: | ||
1353 | nr_copy_pages = 0; | 1377 | nr_copy_pages = 0; |
1354 | nr_meta_pages = 0; | 1378 | nr_meta_pages = 0; |
1355 | restore_pblist = NULL; | 1379 | restore_pblist = NULL; |
diff --git a/kernel/power/suspend.c b/kernel/power/suspend.c index 18c62195660f..4ca9a33ff620 100644 --- a/kernel/power/suspend.c +++ b/kernel/power/suspend.c | |||
@@ -146,17 +146,29 @@ static int platform_suspend_prepare(suspend_state_t state) | |||
146 | 146 | ||
147 | static int platform_suspend_prepare_late(suspend_state_t state) | 147 | static int platform_suspend_prepare_late(suspend_state_t state) |
148 | { | 148 | { |
149 | return state == PM_SUSPEND_FREEZE && freeze_ops->prepare ? | ||
150 | freeze_ops->prepare() : 0; | ||
151 | } | ||
152 | |||
153 | static int platform_suspend_prepare_noirq(suspend_state_t state) | ||
154 | { | ||
149 | return state != PM_SUSPEND_FREEZE && suspend_ops->prepare_late ? | 155 | return state != PM_SUSPEND_FREEZE && suspend_ops->prepare_late ? |
150 | suspend_ops->prepare_late() : 0; | 156 | suspend_ops->prepare_late() : 0; |
151 | } | 157 | } |
152 | 158 | ||
153 | static void platform_suspend_wake(suspend_state_t state) | 159 | static void platform_resume_noirq(suspend_state_t state) |
154 | { | 160 | { |
155 | if (state != PM_SUSPEND_FREEZE && suspend_ops->wake) | 161 | if (state != PM_SUSPEND_FREEZE && suspend_ops->wake) |
156 | suspend_ops->wake(); | 162 | suspend_ops->wake(); |
157 | } | 163 | } |
158 | 164 | ||
159 | static void platform_suspend_finish(suspend_state_t state) | 165 | static void platform_resume_early(suspend_state_t state) |
166 | { | ||
167 | if (state == PM_SUSPEND_FREEZE && freeze_ops->restore) | ||
168 | freeze_ops->restore(); | ||
169 | } | ||
170 | |||
171 | static void platform_resume_finish(suspend_state_t state) | ||
160 | { | 172 | { |
161 | if (state != PM_SUSPEND_FREEZE && suspend_ops->finish) | 173 | if (state != PM_SUSPEND_FREEZE && suspend_ops->finish) |
162 | suspend_ops->finish(); | 174 | suspend_ops->finish(); |
@@ -172,7 +184,7 @@ static int platform_suspend_begin(suspend_state_t state) | |||
172 | return 0; | 184 | return 0; |
173 | } | 185 | } |
174 | 186 | ||
175 | static void platform_suspend_end(suspend_state_t state) | 187 | static void platform_resume_end(suspend_state_t state) |
176 | { | 188 | { |
177 | if (state == PM_SUSPEND_FREEZE && freeze_ops && freeze_ops->end) | 189 | if (state == PM_SUSPEND_FREEZE && freeze_ops && freeze_ops->end) |
178 | freeze_ops->end(); | 190 | freeze_ops->end(); |
@@ -180,7 +192,7 @@ static void platform_suspend_end(suspend_state_t state) | |||
180 | suspend_ops->end(); | 192 | suspend_ops->end(); |
181 | } | 193 | } |
182 | 194 | ||
183 | static void platform_suspend_recover(suspend_state_t state) | 195 | static void platform_recover(suspend_state_t state) |
184 | { | 196 | { |
185 | if (state != PM_SUSPEND_FREEZE && suspend_ops->recover) | 197 | if (state != PM_SUSPEND_FREEZE && suspend_ops->recover) |
186 | suspend_ops->recover(); | 198 | suspend_ops->recover(); |
@@ -265,13 +277,22 @@ static int suspend_enter(suspend_state_t state, bool *wakeup) | |||
265 | if (error) | 277 | if (error) |
266 | goto Platform_finish; | 278 | goto Platform_finish; |
267 | 279 | ||
268 | error = dpm_suspend_end(PMSG_SUSPEND); | 280 | error = dpm_suspend_late(PMSG_SUSPEND); |
269 | if (error) { | 281 | if (error) { |
270 | printk(KERN_ERR "PM: Some devices failed to power down\n"); | 282 | printk(KERN_ERR "PM: late suspend of devices failed\n"); |
271 | goto Platform_finish; | 283 | goto Platform_finish; |
272 | } | 284 | } |
273 | error = platform_suspend_prepare_late(state); | 285 | error = platform_suspend_prepare_late(state); |
274 | if (error) | 286 | if (error) |
287 | goto Devices_early_resume; | ||
288 | |||
289 | error = dpm_suspend_noirq(PMSG_SUSPEND); | ||
290 | if (error) { | ||
291 | printk(KERN_ERR "PM: noirq suspend of devices failed\n"); | ||
292 | goto Platform_early_resume; | ||
293 | } | ||
294 | error = platform_suspend_prepare_noirq(state); | ||
295 | if (error) | ||
275 | goto Platform_wake; | 296 | goto Platform_wake; |
276 | 297 | ||
277 | if (suspend_test(TEST_PLATFORM)) | 298 | if (suspend_test(TEST_PLATFORM)) |
@@ -318,11 +339,17 @@ static int suspend_enter(suspend_state_t state, bool *wakeup) | |||
318 | enable_nonboot_cpus(); | 339 | enable_nonboot_cpus(); |
319 | 340 | ||
320 | Platform_wake: | 341 | Platform_wake: |
321 | platform_suspend_wake(state); | 342 | platform_resume_noirq(state); |
322 | dpm_resume_start(PMSG_RESUME); | 343 | dpm_resume_noirq(PMSG_RESUME); |
344 | |||
345 | Platform_early_resume: | ||
346 | platform_resume_early(state); | ||
347 | |||
348 | Devices_early_resume: | ||
349 | dpm_resume_early(PMSG_RESUME); | ||
323 | 350 | ||
324 | Platform_finish: | 351 | Platform_finish: |
325 | platform_suspend_finish(state); | 352 | platform_resume_finish(state); |
326 | return error; | 353 | return error; |
327 | } | 354 | } |
328 | 355 | ||
@@ -361,14 +388,16 @@ int suspend_devices_and_enter(suspend_state_t state) | |||
361 | suspend_test_start(); | 388 | suspend_test_start(); |
362 | dpm_resume_end(PMSG_RESUME); | 389 | dpm_resume_end(PMSG_RESUME); |
363 | suspend_test_finish("resume devices"); | 390 | suspend_test_finish("resume devices"); |
391 | trace_suspend_resume(TPS("resume_console"), state, true); | ||
364 | resume_console(); | 392 | resume_console(); |
393 | trace_suspend_resume(TPS("resume_console"), state, false); | ||
365 | 394 | ||
366 | Close: | 395 | Close: |
367 | platform_suspend_end(state); | 396 | platform_resume_end(state); |
368 | return error; | 397 | return error; |
369 | 398 | ||
370 | Recover_platform: | 399 | Recover_platform: |
371 | platform_suspend_recover(state); | 400 | platform_recover(state); |
372 | goto Resume_devices; | 401 | goto Resume_devices; |
373 | } | 402 | } |
374 | 403 | ||
diff --git a/kernel/power/suspend_test.c b/kernel/power/suspend_test.c index bd91bc177c93..084452e34a12 100644 --- a/kernel/power/suspend_test.c +++ b/kernel/power/suspend_test.c | |||
@@ -22,6 +22,8 @@ | |||
22 | #define TEST_SUSPEND_SECONDS 10 | 22 | #define TEST_SUSPEND_SECONDS 10 |
23 | 23 | ||
24 | static unsigned long suspend_test_start_time; | 24 | static unsigned long suspend_test_start_time; |
25 | static u32 test_repeat_count_max = 1; | ||
26 | static u32 test_repeat_count_current; | ||
25 | 27 | ||
26 | void suspend_test_start(void) | 28 | void suspend_test_start(void) |
27 | { | 29 | { |
@@ -74,6 +76,7 @@ static void __init test_wakealarm(struct rtc_device *rtc, suspend_state_t state) | |||
74 | int status; | 76 | int status; |
75 | 77 | ||
76 | /* this may fail if the RTC hasn't been initialized */ | 78 | /* this may fail if the RTC hasn't been initialized */ |
79 | repeat: | ||
77 | status = rtc_read_time(rtc, &alm.time); | 80 | status = rtc_read_time(rtc, &alm.time); |
78 | if (status < 0) { | 81 | if (status < 0) { |
79 | printk(err_readtime, dev_name(&rtc->dev), status); | 82 | printk(err_readtime, dev_name(&rtc->dev), status); |
@@ -100,10 +103,21 @@ static void __init test_wakealarm(struct rtc_device *rtc, suspend_state_t state) | |||
100 | if (state == PM_SUSPEND_STANDBY) { | 103 | if (state == PM_SUSPEND_STANDBY) { |
101 | printk(info_test, pm_states[state]); | 104 | printk(info_test, pm_states[state]); |
102 | status = pm_suspend(state); | 105 | status = pm_suspend(state); |
106 | if (status < 0) | ||
107 | state = PM_SUSPEND_FREEZE; | ||
103 | } | 108 | } |
109 | if (state == PM_SUSPEND_FREEZE) { | ||
110 | printk(info_test, pm_states[state]); | ||
111 | status = pm_suspend(state); | ||
112 | } | ||
113 | |||
104 | if (status < 0) | 114 | if (status < 0) |
105 | printk(err_suspend, status); | 115 | printk(err_suspend, status); |
106 | 116 | ||
117 | test_repeat_count_current++; | ||
118 | if (test_repeat_count_current < test_repeat_count_max) | ||
119 | goto repeat; | ||
120 | |||
107 | /* Some platforms can't detect that the alarm triggered the | 121 | /* Some platforms can't detect that the alarm triggered the |
108 | * wakeup, or (accordingly) disable it after it afterwards. | 122 | * wakeup, or (accordingly) disable it after it afterwards. |
109 | * It's supposed to give oneshot behavior; cope. | 123 | * It's supposed to give oneshot behavior; cope. |
@@ -137,16 +151,28 @@ static char warn_bad_state[] __initdata = | |||
137 | static int __init setup_test_suspend(char *value) | 151 | static int __init setup_test_suspend(char *value) |
138 | { | 152 | { |
139 | int i; | 153 | int i; |
154 | char *repeat; | ||
155 | char *suspend_type; | ||
140 | 156 | ||
141 | /* "=mem" ==> "mem" */ | 157 | /* example : "=mem[,N]" ==> "mem[,N]" */ |
142 | value++; | 158 | value++; |
159 | suspend_type = strsep(&value, ","); | ||
160 | if (!suspend_type) | ||
161 | return 0; | ||
162 | |||
163 | repeat = strsep(&value, ","); | ||
164 | if (repeat) { | ||
165 | if (kstrtou32(repeat, 0, &test_repeat_count_max)) | ||
166 | return 0; | ||
167 | } | ||
168 | |||
143 | for (i = 0; pm_labels[i]; i++) | 169 | for (i = 0; pm_labels[i]; i++) |
144 | if (!strcmp(pm_labels[i], value)) { | 170 | if (!strcmp(pm_labels[i], suspend_type)) { |
145 | test_state_label = pm_labels[i]; | 171 | test_state_label = pm_labels[i]; |
146 | return 0; | 172 | return 0; |
147 | } | 173 | } |
148 | 174 | ||
149 | printk(warn_bad_state, value); | 175 | printk(warn_bad_state, suspend_type); |
150 | return 0; | 176 | return 0; |
151 | } | 177 | } |
152 | __setup("test_suspend", setup_test_suspend); | 178 | __setup("test_suspend", setup_test_suspend); |