diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2010-06-04 16:04:39 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2010-06-04 16:04:39 -0400 |
commit | b8a3c6091a2337391ed878693604d712d6420241 (patch) | |
tree | aa6bad4575406465b88b0dfc03e1a166c363bde3 | |
parent | ed8319e9b269ed19449432db3aefc11eb7be7376 (diff) | |
parent | 85f1bb4ace038289d587bcff64128be10613f9f3 (diff) |
Merge branch 'release' of git://git.kernel.org/pub/scm/linux/kernel/git/lenb/linux-acpi-2.6
* 'release' of git://git.kernel.org/pub/scm/linux/kernel/git/lenb/linux-acpi-2.6:
ACPI: Eliminate us to pm ticks conversion in common path
ACPI: Fix the incorrect calculation about C-state idle time
ACPI: update feature-removal.txt to reflect deleted acpi=ht option
ACPI / EC / PM: Fix names of functions that block/unblock EC transactions
ACPI / EC / PM: Fix race between EC transactions and system suspend
-rw-r--r-- | Documentation/feature-removal-schedule.txt | 9 | ||||
-rw-r--r-- | drivers/acpi/ec.c | 22 | ||||
-rw-r--r-- | drivers/acpi/internal.h | 5 | ||||
-rw-r--r-- | drivers/acpi/processor_idle.c | 17 | ||||
-rw-r--r-- | drivers/acpi/sleep.c | 57 |
5 files changed, 55 insertions, 55 deletions
diff --git a/Documentation/feature-removal-schedule.txt b/Documentation/feature-removal-schedule.txt index 672be0109d02..c268783bc4e7 100644 --- a/Documentation/feature-removal-schedule.txt +++ b/Documentation/feature-removal-schedule.txt | |||
@@ -578,15 +578,6 @@ Who: Avi Kivity <avi@redhat.com> | |||
578 | 578 | ||
579 | ---------------------------- | 579 | ---------------------------- |
580 | 580 | ||
581 | What: "acpi=ht" boot option | ||
582 | When: 2.6.35 | ||
583 | Why: Useful in 2003, implementation is a hack. | ||
584 | Generally invoked by accident today. | ||
585 | Seen as doing more harm than good. | ||
586 | Who: Len Brown <len.brown@intel.com> | ||
587 | |||
588 | ---------------------------- | ||
589 | |||
590 | What: iwlwifi 50XX module parameters | 581 | What: iwlwifi 50XX module parameters |
591 | When: 2.6.40 | 582 | When: 2.6.40 |
592 | Why: The "..50" modules parameters were used to configure 5000 series and | 583 | Why: The "..50" modules parameters were used to configure 5000 series and |
diff --git a/drivers/acpi/ec.c b/drivers/acpi/ec.c index e61d4f8e62a5..5f2027d782e8 100644 --- a/drivers/acpi/ec.c +++ b/drivers/acpi/ec.c | |||
@@ -79,7 +79,7 @@ enum { | |||
79 | EC_FLAGS_GPE_STORM, /* GPE storm detected */ | 79 | EC_FLAGS_GPE_STORM, /* GPE storm detected */ |
80 | EC_FLAGS_HANDLERS_INSTALLED, /* Handlers for GPE and | 80 | EC_FLAGS_HANDLERS_INSTALLED, /* Handlers for GPE and |
81 | * OpReg are installed */ | 81 | * OpReg are installed */ |
82 | EC_FLAGS_FROZEN, /* Transactions are suspended */ | 82 | EC_FLAGS_BLOCKED, /* Transactions are blocked */ |
83 | }; | 83 | }; |
84 | 84 | ||
85 | /* If we find an EC via the ECDT, we need to keep a ptr to its context */ | 85 | /* If we find an EC via the ECDT, we need to keep a ptr to its context */ |
@@ -293,7 +293,7 @@ static int acpi_ec_transaction(struct acpi_ec *ec, struct transaction *t) | |||
293 | if (t->rdata) | 293 | if (t->rdata) |
294 | memset(t->rdata, 0, t->rlen); | 294 | memset(t->rdata, 0, t->rlen); |
295 | mutex_lock(&ec->lock); | 295 | mutex_lock(&ec->lock); |
296 | if (test_bit(EC_FLAGS_FROZEN, &ec->flags)) { | 296 | if (test_bit(EC_FLAGS_BLOCKED, &ec->flags)) { |
297 | status = -EINVAL; | 297 | status = -EINVAL; |
298 | goto unlock; | 298 | goto unlock; |
299 | } | 299 | } |
@@ -459,7 +459,7 @@ int ec_transaction(u8 command, | |||
459 | 459 | ||
460 | EXPORT_SYMBOL(ec_transaction); | 460 | EXPORT_SYMBOL(ec_transaction); |
461 | 461 | ||
462 | void acpi_ec_suspend_transactions(void) | 462 | void acpi_ec_block_transactions(void) |
463 | { | 463 | { |
464 | struct acpi_ec *ec = first_ec; | 464 | struct acpi_ec *ec = first_ec; |
465 | 465 | ||
@@ -468,11 +468,11 @@ void acpi_ec_suspend_transactions(void) | |||
468 | 468 | ||
469 | mutex_lock(&ec->lock); | 469 | mutex_lock(&ec->lock); |
470 | /* Prevent transactions from being carried out */ | 470 | /* Prevent transactions from being carried out */ |
471 | set_bit(EC_FLAGS_FROZEN, &ec->flags); | 471 | set_bit(EC_FLAGS_BLOCKED, &ec->flags); |
472 | mutex_unlock(&ec->lock); | 472 | mutex_unlock(&ec->lock); |
473 | } | 473 | } |
474 | 474 | ||
475 | void acpi_ec_resume_transactions(void) | 475 | void acpi_ec_unblock_transactions(void) |
476 | { | 476 | { |
477 | struct acpi_ec *ec = first_ec; | 477 | struct acpi_ec *ec = first_ec; |
478 | 478 | ||
@@ -481,10 +481,20 @@ void acpi_ec_resume_transactions(void) | |||
481 | 481 | ||
482 | mutex_lock(&ec->lock); | 482 | mutex_lock(&ec->lock); |
483 | /* Allow transactions to be carried out again */ | 483 | /* Allow transactions to be carried out again */ |
484 | clear_bit(EC_FLAGS_FROZEN, &ec->flags); | 484 | clear_bit(EC_FLAGS_BLOCKED, &ec->flags); |
485 | mutex_unlock(&ec->lock); | 485 | mutex_unlock(&ec->lock); |
486 | } | 486 | } |
487 | 487 | ||
488 | void acpi_ec_unblock_transactions_early(void) | ||
489 | { | ||
490 | /* | ||
491 | * Allow transactions to happen again (this function is called from | ||
492 | * atomic context during wakeup, so we don't need to acquire the mutex). | ||
493 | */ | ||
494 | if (first_ec) | ||
495 | clear_bit(EC_FLAGS_BLOCKED, &first_ec->flags); | ||
496 | } | ||
497 | |||
488 | static int acpi_ec_query_unlocked(struct acpi_ec *ec, u8 * data) | 498 | static int acpi_ec_query_unlocked(struct acpi_ec *ec, u8 * data) |
489 | { | 499 | { |
490 | int result; | 500 | int result; |
diff --git a/drivers/acpi/internal.h b/drivers/acpi/internal.h index e28411367239..f8f190ec066e 100644 --- a/drivers/acpi/internal.h +++ b/drivers/acpi/internal.h | |||
@@ -49,8 +49,9 @@ void acpi_early_processor_set_pdc(void); | |||
49 | int acpi_ec_init(void); | 49 | int acpi_ec_init(void); |
50 | int acpi_ec_ecdt_probe(void); | 50 | int acpi_ec_ecdt_probe(void); |
51 | int acpi_boot_ec_enable(void); | 51 | int acpi_boot_ec_enable(void); |
52 | void acpi_ec_suspend_transactions(void); | 52 | void acpi_ec_block_transactions(void); |
53 | void acpi_ec_resume_transactions(void); | 53 | void acpi_ec_unblock_transactions(void); |
54 | void acpi_ec_unblock_transactions_early(void); | ||
54 | 55 | ||
55 | /*-------------------------------------------------------------------------- | 56 | /*-------------------------------------------------------------------------- |
56 | Suspend/Resume | 57 | Suspend/Resume |
diff --git a/drivers/acpi/processor_idle.c b/drivers/acpi/processor_idle.c index 2e8c27d48f2b..b1b385692f46 100644 --- a/drivers/acpi/processor_idle.c +++ b/drivers/acpi/processor_idle.c | |||
@@ -80,7 +80,7 @@ module_param(nocst, uint, 0000); | |||
80 | static unsigned int latency_factor __read_mostly = 2; | 80 | static unsigned int latency_factor __read_mostly = 2; |
81 | module_param(latency_factor, uint, 0644); | 81 | module_param(latency_factor, uint, 0644); |
82 | 82 | ||
83 | static s64 us_to_pm_timer_ticks(s64 t) | 83 | static u64 us_to_pm_timer_ticks(s64 t) |
84 | { | 84 | { |
85 | return div64_u64(t * PM_TIMER_FREQUENCY, 1000000); | 85 | return div64_u64(t * PM_TIMER_FREQUENCY, 1000000); |
86 | } | 86 | } |
@@ -731,10 +731,10 @@ static int acpi_processor_power_seq_show(struct seq_file *seq, void *offset) | |||
731 | 731 | ||
732 | seq_puts(seq, "demotion[--] "); | 732 | seq_puts(seq, "demotion[--] "); |
733 | 733 | ||
734 | seq_printf(seq, "latency[%03d] usage[%08d] duration[%020llu]\n", | 734 | seq_printf(seq, "latency[%03d] usage[%08d] duration[%020Lu]\n", |
735 | pr->power.states[i].latency, | 735 | pr->power.states[i].latency, |
736 | pr->power.states[i].usage, | 736 | pr->power.states[i].usage, |
737 | (unsigned long long)pr->power.states[i].time); | 737 | us_to_pm_timer_ticks(pr->power.states[i].time)); |
738 | } | 738 | } |
739 | 739 | ||
740 | end: | 740 | end: |
@@ -861,7 +861,6 @@ static int acpi_idle_enter_simple(struct cpuidle_device *dev, | |||
861 | ktime_t kt1, kt2; | 861 | ktime_t kt1, kt2; |
862 | s64 idle_time_ns; | 862 | s64 idle_time_ns; |
863 | s64 idle_time; | 863 | s64 idle_time; |
864 | s64 sleep_ticks = 0; | ||
865 | 864 | ||
866 | pr = __get_cpu_var(processors); | 865 | pr = __get_cpu_var(processors); |
867 | 866 | ||
@@ -906,8 +905,6 @@ static int acpi_idle_enter_simple(struct cpuidle_device *dev, | |||
906 | idle_time = idle_time_ns; | 905 | idle_time = idle_time_ns; |
907 | do_div(idle_time, NSEC_PER_USEC); | 906 | do_div(idle_time, NSEC_PER_USEC); |
908 | 907 | ||
909 | sleep_ticks = us_to_pm_timer_ticks(idle_time); | ||
910 | |||
911 | /* Tell the scheduler how much we idled: */ | 908 | /* Tell the scheduler how much we idled: */ |
912 | sched_clock_idle_wakeup_event(idle_time_ns); | 909 | sched_clock_idle_wakeup_event(idle_time_ns); |
913 | 910 | ||
@@ -918,7 +915,7 @@ static int acpi_idle_enter_simple(struct cpuidle_device *dev, | |||
918 | cx->usage++; | 915 | cx->usage++; |
919 | 916 | ||
920 | lapic_timer_state_broadcast(pr, cx, 0); | 917 | lapic_timer_state_broadcast(pr, cx, 0); |
921 | cx->time += sleep_ticks; | 918 | cx->time += idle_time; |
922 | return idle_time; | 919 | return idle_time; |
923 | } | 920 | } |
924 | 921 | ||
@@ -940,7 +937,6 @@ static int acpi_idle_enter_bm(struct cpuidle_device *dev, | |||
940 | ktime_t kt1, kt2; | 937 | ktime_t kt1, kt2; |
941 | s64 idle_time_ns; | 938 | s64 idle_time_ns; |
942 | s64 idle_time; | 939 | s64 idle_time; |
943 | s64 sleep_ticks = 0; | ||
944 | 940 | ||
945 | 941 | ||
946 | pr = __get_cpu_var(processors); | 942 | pr = __get_cpu_var(processors); |
@@ -1022,11 +1018,10 @@ static int acpi_idle_enter_bm(struct cpuidle_device *dev, | |||
1022 | spin_unlock(&c3_lock); | 1018 | spin_unlock(&c3_lock); |
1023 | } | 1019 | } |
1024 | kt2 = ktime_get_real(); | 1020 | kt2 = ktime_get_real(); |
1025 | idle_time_ns = ktime_to_us(ktime_sub(kt2, kt1)); | 1021 | idle_time_ns = ktime_to_ns(ktime_sub(kt2, kt1)); |
1026 | idle_time = idle_time_ns; | 1022 | idle_time = idle_time_ns; |
1027 | do_div(idle_time, NSEC_PER_USEC); | 1023 | do_div(idle_time, NSEC_PER_USEC); |
1028 | 1024 | ||
1029 | sleep_ticks = us_to_pm_timer_ticks(idle_time); | ||
1030 | /* Tell the scheduler how much we idled: */ | 1025 | /* Tell the scheduler how much we idled: */ |
1031 | sched_clock_idle_wakeup_event(idle_time_ns); | 1026 | sched_clock_idle_wakeup_event(idle_time_ns); |
1032 | 1027 | ||
@@ -1037,7 +1032,7 @@ static int acpi_idle_enter_bm(struct cpuidle_device *dev, | |||
1037 | cx->usage++; | 1032 | cx->usage++; |
1038 | 1033 | ||
1039 | lapic_timer_state_broadcast(pr, cx, 0); | 1034 | lapic_timer_state_broadcast(pr, cx, 0); |
1040 | cx->time += sleep_ticks; | 1035 | cx->time += idle_time; |
1041 | return idle_time; | 1036 | return idle_time; |
1042 | } | 1037 | } |
1043 | 1038 | ||
diff --git a/drivers/acpi/sleep.c b/drivers/acpi/sleep.c index 4ab2275b4461..3fb4bdea7e06 100644 --- a/drivers/acpi/sleep.c +++ b/drivers/acpi/sleep.c | |||
@@ -94,11 +94,13 @@ void __init acpi_old_suspend_ordering(void) | |||
94 | } | 94 | } |
95 | 95 | ||
96 | /** | 96 | /** |
97 | * acpi_pm_disable_gpes - Disable the GPEs. | 97 | * acpi_pm_freeze - Disable the GPEs and suspend EC transactions. |
98 | */ | 98 | */ |
99 | static int acpi_pm_disable_gpes(void) | 99 | static int acpi_pm_freeze(void) |
100 | { | 100 | { |
101 | acpi_disable_all_gpes(); | 101 | acpi_disable_all_gpes(); |
102 | acpi_os_wait_events_complete(NULL); | ||
103 | acpi_ec_block_transactions(); | ||
102 | return 0; | 104 | return 0; |
103 | } | 105 | } |
104 | 106 | ||
@@ -126,7 +128,8 @@ static int acpi_pm_prepare(void) | |||
126 | int error = __acpi_pm_prepare(); | 128 | int error = __acpi_pm_prepare(); |
127 | 129 | ||
128 | if (!error) | 130 | if (!error) |
129 | acpi_disable_all_gpes(); | 131 | acpi_pm_freeze(); |
132 | |||
130 | return error; | 133 | return error; |
131 | } | 134 | } |
132 | 135 | ||
@@ -256,6 +259,8 @@ static int acpi_suspend_enter(suspend_state_t pm_state) | |||
256 | * acpi_leave_sleep_state will reenable specific GPEs later | 259 | * acpi_leave_sleep_state will reenable specific GPEs later |
257 | */ | 260 | */ |
258 | acpi_disable_all_gpes(); | 261 | acpi_disable_all_gpes(); |
262 | /* Allow EC transactions to happen. */ | ||
263 | acpi_ec_unblock_transactions_early(); | ||
259 | 264 | ||
260 | local_irq_restore(flags); | 265 | local_irq_restore(flags); |
261 | printk(KERN_DEBUG "Back to C!\n"); | 266 | printk(KERN_DEBUG "Back to C!\n"); |
@@ -267,6 +272,12 @@ static int acpi_suspend_enter(suspend_state_t pm_state) | |||
267 | return ACPI_SUCCESS(status) ? 0 : -EFAULT; | 272 | return ACPI_SUCCESS(status) ? 0 : -EFAULT; |
268 | } | 273 | } |
269 | 274 | ||
275 | static void acpi_suspend_finish(void) | ||
276 | { | ||
277 | acpi_ec_unblock_transactions(); | ||
278 | acpi_pm_finish(); | ||
279 | } | ||
280 | |||
270 | static int acpi_suspend_state_valid(suspend_state_t pm_state) | 281 | static int acpi_suspend_state_valid(suspend_state_t pm_state) |
271 | { | 282 | { |
272 | u32 acpi_state; | 283 | u32 acpi_state; |
@@ -288,7 +299,7 @@ static struct platform_suspend_ops acpi_suspend_ops = { | |||
288 | .begin = acpi_suspend_begin, | 299 | .begin = acpi_suspend_begin, |
289 | .prepare_late = acpi_pm_prepare, | 300 | .prepare_late = acpi_pm_prepare, |
290 | .enter = acpi_suspend_enter, | 301 | .enter = acpi_suspend_enter, |
291 | .wake = acpi_pm_finish, | 302 | .wake = acpi_suspend_finish, |
292 | .end = acpi_pm_end, | 303 | .end = acpi_pm_end, |
293 | }; | 304 | }; |
294 | 305 | ||
@@ -314,9 +325,9 @@ static int acpi_suspend_begin_old(suspend_state_t pm_state) | |||
314 | static struct platform_suspend_ops acpi_suspend_ops_old = { | 325 | static struct platform_suspend_ops acpi_suspend_ops_old = { |
315 | .valid = acpi_suspend_state_valid, | 326 | .valid = acpi_suspend_state_valid, |
316 | .begin = acpi_suspend_begin_old, | 327 | .begin = acpi_suspend_begin_old, |
317 | .prepare_late = acpi_pm_disable_gpes, | 328 | .prepare_late = acpi_pm_freeze, |
318 | .enter = acpi_suspend_enter, | 329 | .enter = acpi_suspend_enter, |
319 | .wake = acpi_pm_finish, | 330 | .wake = acpi_suspend_finish, |
320 | .end = acpi_pm_end, | 331 | .end = acpi_pm_end, |
321 | .recover = acpi_pm_finish, | 332 | .recover = acpi_pm_finish, |
322 | }; | 333 | }; |
@@ -433,6 +444,7 @@ static int acpi_hibernation_enter(void) | |||
433 | static void acpi_hibernation_finish(void) | 444 | static void acpi_hibernation_finish(void) |
434 | { | 445 | { |
435 | hibernate_nvs_free(); | 446 | hibernate_nvs_free(); |
447 | acpi_ec_unblock_transactions(); | ||
436 | acpi_pm_finish(); | 448 | acpi_pm_finish(); |
437 | } | 449 | } |
438 | 450 | ||
@@ -453,19 +465,13 @@ static void acpi_hibernation_leave(void) | |||
453 | } | 465 | } |
454 | /* Restore the NVS memory area */ | 466 | /* Restore the NVS memory area */ |
455 | hibernate_nvs_restore(); | 467 | hibernate_nvs_restore(); |
468 | /* Allow EC transactions to happen. */ | ||
469 | acpi_ec_unblock_transactions_early(); | ||
456 | } | 470 | } |
457 | 471 | ||
458 | static int acpi_pm_pre_restore(void) | 472 | static void acpi_pm_thaw(void) |
459 | { | ||
460 | acpi_disable_all_gpes(); | ||
461 | acpi_os_wait_events_complete(NULL); | ||
462 | acpi_ec_suspend_transactions(); | ||
463 | return 0; | ||
464 | } | ||
465 | |||
466 | static void acpi_pm_restore_cleanup(void) | ||
467 | { | 473 | { |
468 | acpi_ec_resume_transactions(); | 474 | acpi_ec_unblock_transactions(); |
469 | acpi_enable_all_runtime_gpes(); | 475 | acpi_enable_all_runtime_gpes(); |
470 | } | 476 | } |
471 | 477 | ||
@@ -477,8 +483,8 @@ static struct platform_hibernation_ops acpi_hibernation_ops = { | |||
477 | .prepare = acpi_pm_prepare, | 483 | .prepare = acpi_pm_prepare, |
478 | .enter = acpi_hibernation_enter, | 484 | .enter = acpi_hibernation_enter, |
479 | .leave = acpi_hibernation_leave, | 485 | .leave = acpi_hibernation_leave, |
480 | .pre_restore = acpi_pm_pre_restore, | 486 | .pre_restore = acpi_pm_freeze, |
481 | .restore_cleanup = acpi_pm_restore_cleanup, | 487 | .restore_cleanup = acpi_pm_thaw, |
482 | }; | 488 | }; |
483 | 489 | ||
484 | /** | 490 | /** |
@@ -510,12 +516,9 @@ static int acpi_hibernation_begin_old(void) | |||
510 | 516 | ||
511 | static int acpi_hibernation_pre_snapshot_old(void) | 517 | static int acpi_hibernation_pre_snapshot_old(void) |
512 | { | 518 | { |
513 | int error = acpi_pm_disable_gpes(); | 519 | acpi_pm_freeze(); |
514 | 520 | hibernate_nvs_save(); | |
515 | if (!error) | 521 | return 0; |
516 | hibernate_nvs_save(); | ||
517 | |||
518 | return error; | ||
519 | } | 522 | } |
520 | 523 | ||
521 | /* | 524 | /* |
@@ -527,11 +530,11 @@ static struct platform_hibernation_ops acpi_hibernation_ops_old = { | |||
527 | .end = acpi_pm_end, | 530 | .end = acpi_pm_end, |
528 | .pre_snapshot = acpi_hibernation_pre_snapshot_old, | 531 | .pre_snapshot = acpi_hibernation_pre_snapshot_old, |
529 | .finish = acpi_hibernation_finish, | 532 | .finish = acpi_hibernation_finish, |
530 | .prepare = acpi_pm_disable_gpes, | 533 | .prepare = acpi_pm_freeze, |
531 | .enter = acpi_hibernation_enter, | 534 | .enter = acpi_hibernation_enter, |
532 | .leave = acpi_hibernation_leave, | 535 | .leave = acpi_hibernation_leave, |
533 | .pre_restore = acpi_pm_pre_restore, | 536 | .pre_restore = acpi_pm_freeze, |
534 | .restore_cleanup = acpi_pm_restore_cleanup, | 537 | .restore_cleanup = acpi_pm_thaw, |
535 | .recover = acpi_pm_finish, | 538 | .recover = acpi_pm_finish, |
536 | }; | 539 | }; |
537 | #endif /* CONFIG_HIBERNATION */ | 540 | #endif /* CONFIG_HIBERNATION */ |