diff options
author | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2015-02-03 15:55:11 -0500 |
---|---|---|
committer | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2015-02-03 15:55:11 -0500 |
commit | 6491bc0c616969f7ae1fcb30a8823c333e2944c7 (patch) | |
tree | 8f055537c917f4caa8ec9ccc6b764e4224b21418 | |
parent | d2cecb3d66e39765f3dec8adfb88c322e709b06d (diff) |
ACPI / cpuidle: Common callback routine for entering states
Introduce a common ->enter callback routine for the ACPI cpuidle
driver, acpi_idle_enter(), which helps to reduce code complexity,
size and duplication and prevents theoretically possible failues that
an incorrect routine may be run to enter the given idle state due to
a firmware bug (eg. when _CST returns a different set of states for
each processor).
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
-rw-r--r-- | drivers/acpi/processor_idle.c | 118 |
1 files changed, 45 insertions, 73 deletions
diff --git a/drivers/acpi/processor_idle.c b/drivers/acpi/processor_idle.c index 58113a6fa1d3..c256bd7fbd78 100644 --- a/drivers/acpi/processor_idle.c +++ b/drivers/acpi/processor_idle.c | |||
@@ -737,74 +737,17 @@ static bool acpi_idle_fallback_to_c1(struct acpi_processor *pr) | |||
737 | !pr->flags.has_cst; | 737 | !pr->flags.has_cst; |
738 | } | 738 | } |
739 | 739 | ||
740 | /** | ||
741 | * acpi_idle_enter_simple - enters a CPU idle state without BM handling | ||
742 | * @dev: the target CPU | ||
743 | * @drv: cpuidle driver with cpuidle state information | ||
744 | * @index: the index of suggested state | ||
745 | */ | ||
746 | static int acpi_idle_enter_simple(struct cpuidle_device *dev, | ||
747 | struct cpuidle_driver *drv, int index) | ||
748 | { | ||
749 | struct acpi_processor *pr; | ||
750 | struct acpi_processor_cx *cx = per_cpu(acpi_cstate[index], dev->cpu); | ||
751 | |||
752 | pr = __this_cpu_read(processors); | ||
753 | |||
754 | if (unlikely(!pr)) | ||
755 | return -EINVAL; | ||
756 | |||
757 | if (cx->type != ACPI_STATE_C1 && acpi_idle_fallback_to_c1(pr)) { | ||
758 | index = CPUIDLE_DRIVER_STATE_START; | ||
759 | cx = per_cpu(acpi_cstate[index], dev->cpu); | ||
760 | } | ||
761 | |||
762 | lapic_timer_state_broadcast(pr, cx, 1); | ||
763 | |||
764 | if (cx->type == ACPI_STATE_C3) | ||
765 | ACPI_FLUSH_CPU_CACHE(); | ||
766 | |||
767 | acpi_idle_do_entry(cx); | ||
768 | |||
769 | lapic_timer_state_broadcast(pr, cx, 0); | ||
770 | return index; | ||
771 | } | ||
772 | |||
773 | static int c3_cpu_count; | 740 | static int c3_cpu_count; |
774 | static DEFINE_RAW_SPINLOCK(c3_lock); | 741 | static DEFINE_RAW_SPINLOCK(c3_lock); |
775 | 742 | ||
776 | /** | 743 | /** |
777 | * acpi_idle_enter_bm - enters C3 with proper BM handling | 744 | * acpi_idle_enter_bm - enters C3 with proper BM handling |
778 | * @dev: the target CPU | 745 | * @pr: Target processor |
779 | * @drv: cpuidle driver containing state data | 746 | * @cx: Target state context |
780 | * @index: the index of suggested state | ||
781 | * | ||
782 | * If BM is detected, the deepest non-C3 idle state is entered instead. | ||
783 | */ | 747 | */ |
784 | static int acpi_idle_enter_bm(struct cpuidle_device *dev, | 748 | static void acpi_idle_enter_bm(struct acpi_processor *pr, |
785 | struct cpuidle_driver *drv, int index) | 749 | struct acpi_processor_cx *cx) |
786 | { | 750 | { |
787 | struct acpi_processor *pr; | ||
788 | struct acpi_processor_cx *cx = per_cpu(acpi_cstate[index], dev->cpu); | ||
789 | |||
790 | pr = __this_cpu_read(processors); | ||
791 | |||
792 | if (unlikely(!pr)) | ||
793 | return -EINVAL; | ||
794 | |||
795 | if (acpi_idle_fallback_to_c1(pr)) | ||
796 | return acpi_idle_enter_simple(dev, drv, CPUIDLE_DRIVER_STATE_START); | ||
797 | |||
798 | if (!cx->bm_sts_skip && acpi_idle_bm_check()) { | ||
799 | if (drv->safe_state_index >= 0) { | ||
800 | return drv->states[drv->safe_state_index].enter(dev, | ||
801 | drv, drv->safe_state_index); | ||
802 | } else { | ||
803 | acpi_safe_halt(); | ||
804 | return -EBUSY; | ||
805 | } | ||
806 | } | ||
807 | |||
808 | acpi_unlazy_tlb(smp_processor_id()); | 751 | acpi_unlazy_tlb(smp_processor_id()); |
809 | 752 | ||
810 | /* | 753 | /* |
@@ -842,6 +785,45 @@ static int acpi_idle_enter_bm(struct cpuidle_device *dev, | |||
842 | } | 785 | } |
843 | 786 | ||
844 | lapic_timer_state_broadcast(pr, cx, 0); | 787 | lapic_timer_state_broadcast(pr, cx, 0); |
788 | } | ||
789 | |||
790 | static int acpi_idle_enter(struct cpuidle_device *dev, | ||
791 | struct cpuidle_driver *drv, int index) | ||
792 | { | ||
793 | struct acpi_processor_cx *cx = per_cpu(acpi_cstate[index], dev->cpu); | ||
794 | struct acpi_processor *pr; | ||
795 | |||
796 | pr = __this_cpu_read(processors); | ||
797 | if (unlikely(!pr)) | ||
798 | return -EINVAL; | ||
799 | |||
800 | if (cx->type != ACPI_STATE_C1) { | ||
801 | if (acpi_idle_fallback_to_c1(pr)) { | ||
802 | index = CPUIDLE_DRIVER_STATE_START; | ||
803 | cx = per_cpu(acpi_cstate[index], dev->cpu); | ||
804 | } else if (cx->type == ACPI_STATE_C3 && pr->flags.bm_check) { | ||
805 | if (cx->bm_sts_skip || !acpi_idle_bm_check()) { | ||
806 | acpi_idle_enter_bm(pr, cx); | ||
807 | return index; | ||
808 | } else if (drv->safe_state_index >= 0) { | ||
809 | index = drv->safe_state_index; | ||
810 | cx = per_cpu(acpi_cstate[index], dev->cpu); | ||
811 | } else { | ||
812 | acpi_safe_halt(); | ||
813 | return -EBUSY; | ||
814 | } | ||
815 | } | ||
816 | } | ||
817 | |||
818 | lapic_timer_state_broadcast(pr, cx, 1); | ||
819 | |||
820 | if (cx->type == ACPI_STATE_C3) | ||
821 | ACPI_FLUSH_CPU_CACHE(); | ||
822 | |||
823 | acpi_idle_do_entry(cx); | ||
824 | |||
825 | lapic_timer_state_broadcast(pr, cx, 0); | ||
826 | |||
845 | return index; | 827 | return index; |
846 | } | 828 | } |
847 | 829 | ||
@@ -936,22 +918,12 @@ static int acpi_processor_setup_cpuidle_states(struct acpi_processor *pr) | |||
936 | strncpy(state->desc, cx->desc, CPUIDLE_DESC_LEN); | 918 | strncpy(state->desc, cx->desc, CPUIDLE_DESC_LEN); |
937 | state->exit_latency = cx->latency; | 919 | state->exit_latency = cx->latency; |
938 | state->target_residency = cx->latency * latency_factor; | 920 | state->target_residency = cx->latency * latency_factor; |
921 | state->enter = acpi_idle_enter; | ||
939 | 922 | ||
940 | state->flags = 0; | 923 | state->flags = 0; |
941 | switch (cx->type) { | 924 | if (cx->type == ACPI_STATE_C1 || cx->type == ACPI_STATE_C2) { |
942 | |||
943 | case ACPI_STATE_C1: | ||
944 | case ACPI_STATE_C2: | ||
945 | state->enter = acpi_idle_enter_simple; | ||
946 | state->enter_dead = acpi_idle_play_dead; | 925 | state->enter_dead = acpi_idle_play_dead; |
947 | drv->safe_state_index = count; | 926 | drv->safe_state_index = count; |
948 | break; | ||
949 | |||
950 | case ACPI_STATE_C3: | ||
951 | state->enter = pr->flags.bm_check ? | ||
952 | acpi_idle_enter_bm : | ||
953 | acpi_idle_enter_simple; | ||
954 | break; | ||
955 | } | 927 | } |
956 | 928 | ||
957 | count++; | 929 | count++; |