diff options
Diffstat (limited to 'arch/ppc64')
-rw-r--r-- | arch/ppc64/kernel/iSeries_setup.c | 81 | ||||
-rw-r--r-- | arch/ppc64/kernel/idle.c | 86 |
2 files changed, 81 insertions, 86 deletions
diff --git a/arch/ppc64/kernel/iSeries_setup.c b/arch/ppc64/kernel/iSeries_setup.c index b3f770f6d402..1139e27e1713 100644 --- a/arch/ppc64/kernel/iSeries_setup.c +++ b/arch/ppc64/kernel/iSeries_setup.c | |||
@@ -834,6 +834,87 @@ static int __init iSeries_src_init(void) | |||
834 | 834 | ||
835 | late_initcall(iSeries_src_init); | 835 | late_initcall(iSeries_src_init); |
836 | 836 | ||
837 | static unsigned long maxYieldTime = 0; | ||
838 | static unsigned long minYieldTime = 0xffffffffffffffffUL; | ||
839 | |||
840 | static inline void process_iSeries_events(void) | ||
841 | { | ||
842 | asm volatile ("li 0,0x5555; sc" : : : "r0", "r3"); | ||
843 | } | ||
844 | |||
845 | static void yield_shared_processor(void) | ||
846 | { | ||
847 | unsigned long tb; | ||
848 | unsigned long yieldTime; | ||
849 | |||
850 | HvCall_setEnabledInterrupts(HvCall_MaskIPI | | ||
851 | HvCall_MaskLpEvent | | ||
852 | HvCall_MaskLpProd | | ||
853 | HvCall_MaskTimeout); | ||
854 | |||
855 | tb = get_tb(); | ||
856 | /* Compute future tb value when yield should expire */ | ||
857 | HvCall_yieldProcessor(HvCall_YieldTimed, tb+tb_ticks_per_jiffy); | ||
858 | |||
859 | yieldTime = get_tb() - tb; | ||
860 | if (yieldTime > maxYieldTime) | ||
861 | maxYieldTime = yieldTime; | ||
862 | |||
863 | if (yieldTime < minYieldTime) | ||
864 | minYieldTime = yieldTime; | ||
865 | |||
866 | /* | ||
867 | * The decrementer stops during the yield. Force a fake decrementer | ||
868 | * here and let the timer_interrupt code sort out the actual time. | ||
869 | */ | ||
870 | get_paca()->lppaca.int_dword.fields.decr_int = 1; | ||
871 | process_iSeries_events(); | ||
872 | } | ||
873 | |||
874 | static int iSeries_idle(void) | ||
875 | { | ||
876 | struct paca_struct *lpaca; | ||
877 | long oldval; | ||
878 | |||
879 | /* ensure iSeries run light will be out when idle */ | ||
880 | ppc64_runlatch_off(); | ||
881 | |||
882 | lpaca = get_paca(); | ||
883 | |||
884 | while (1) { | ||
885 | if (lpaca->lppaca.shared_proc) { | ||
886 | if (hvlpevent_is_pending()) | ||
887 | process_iSeries_events(); | ||
888 | if (!need_resched()) | ||
889 | yield_shared_processor(); | ||
890 | } else { | ||
891 | oldval = test_and_clear_thread_flag(TIF_NEED_RESCHED); | ||
892 | |||
893 | if (!oldval) { | ||
894 | set_thread_flag(TIF_POLLING_NRFLAG); | ||
895 | |||
896 | while (!need_resched()) { | ||
897 | HMT_medium(); | ||
898 | if (hvlpevent_is_pending()) | ||
899 | process_iSeries_events(); | ||
900 | HMT_low(); | ||
901 | } | ||
902 | |||
903 | HMT_medium(); | ||
904 | clear_thread_flag(TIF_POLLING_NRFLAG); | ||
905 | } else { | ||
906 | set_need_resched(); | ||
907 | } | ||
908 | } | ||
909 | |||
910 | ppc64_runlatch_on(); | ||
911 | schedule(); | ||
912 | ppc64_runlatch_off(); | ||
913 | } | ||
914 | |||
915 | return 0; | ||
916 | } | ||
917 | |||
837 | #ifndef CONFIG_PCI | 918 | #ifndef CONFIG_PCI |
838 | void __init iSeries_init_IRQ(void) { } | 919 | void __init iSeries_init_IRQ(void) { } |
839 | #endif | 920 | #endif |
diff --git a/arch/ppc64/kernel/idle.c b/arch/ppc64/kernel/idle.c index e270055e73e2..226152467791 100644 --- a/arch/ppc64/kernel/idle.c +++ b/arch/ppc64/kernel/idle.c | |||
@@ -39,90 +39,6 @@ extern void power4_idle(void); | |||
39 | 39 | ||
40 | static int (*idle_loop)(void); | 40 | static int (*idle_loop)(void); |
41 | 41 | ||
42 | #ifdef CONFIG_PPC_ISERIES | ||
43 | static unsigned long maxYieldTime = 0; | ||
44 | static unsigned long minYieldTime = 0xffffffffffffffffUL; | ||
45 | |||
46 | static inline void process_iSeries_events(void) | ||
47 | { | ||
48 | asm volatile ("li 0,0x5555; sc" : : : "r0", "r3"); | ||
49 | } | ||
50 | |||
51 | static void yield_shared_processor(void) | ||
52 | { | ||
53 | unsigned long tb; | ||
54 | unsigned long yieldTime; | ||
55 | |||
56 | HvCall_setEnabledInterrupts(HvCall_MaskIPI | | ||
57 | HvCall_MaskLpEvent | | ||
58 | HvCall_MaskLpProd | | ||
59 | HvCall_MaskTimeout); | ||
60 | |||
61 | tb = get_tb(); | ||
62 | /* Compute future tb value when yield should expire */ | ||
63 | HvCall_yieldProcessor(HvCall_YieldTimed, tb+tb_ticks_per_jiffy); | ||
64 | |||
65 | yieldTime = get_tb() - tb; | ||
66 | if (yieldTime > maxYieldTime) | ||
67 | maxYieldTime = yieldTime; | ||
68 | |||
69 | if (yieldTime < minYieldTime) | ||
70 | minYieldTime = yieldTime; | ||
71 | |||
72 | /* | ||
73 | * The decrementer stops during the yield. Force a fake decrementer | ||
74 | * here and let the timer_interrupt code sort out the actual time. | ||
75 | */ | ||
76 | get_paca()->lppaca.int_dword.fields.decr_int = 1; | ||
77 | process_iSeries_events(); | ||
78 | } | ||
79 | |||
80 | static int iSeries_idle(void) | ||
81 | { | ||
82 | struct paca_struct *lpaca; | ||
83 | long oldval; | ||
84 | |||
85 | /* ensure iSeries run light will be out when idle */ | ||
86 | ppc64_runlatch_off(); | ||
87 | |||
88 | lpaca = get_paca(); | ||
89 | |||
90 | while (1) { | ||
91 | if (lpaca->lppaca.shared_proc) { | ||
92 | if (hvlpevent_is_pending()) | ||
93 | process_iSeries_events(); | ||
94 | if (!need_resched()) | ||
95 | yield_shared_processor(); | ||
96 | } else { | ||
97 | oldval = test_and_clear_thread_flag(TIF_NEED_RESCHED); | ||
98 | |||
99 | if (!oldval) { | ||
100 | set_thread_flag(TIF_POLLING_NRFLAG); | ||
101 | |||
102 | while (!need_resched()) { | ||
103 | HMT_medium(); | ||
104 | if (hvlpevent_is_pending()) | ||
105 | process_iSeries_events(); | ||
106 | HMT_low(); | ||
107 | } | ||
108 | |||
109 | HMT_medium(); | ||
110 | clear_thread_flag(TIF_POLLING_NRFLAG); | ||
111 | } else { | ||
112 | set_need_resched(); | ||
113 | } | ||
114 | } | ||
115 | |||
116 | ppc64_runlatch_on(); | ||
117 | schedule(); | ||
118 | ppc64_runlatch_off(); | ||
119 | } | ||
120 | |||
121 | return 0; | ||
122 | } | ||
123 | |||
124 | #else | ||
125 | |||
126 | int default_idle(void) | 42 | int default_idle(void) |
127 | { | 43 | { |
128 | long oldval; | 44 | long oldval; |
@@ -305,8 +221,6 @@ int native_idle(void) | |||
305 | return 0; | 221 | return 0; |
306 | } | 222 | } |
307 | 223 | ||
308 | #endif /* CONFIG_PPC_ISERIES */ | ||
309 | |||
310 | void cpu_idle(void) | 224 | void cpu_idle(void) |
311 | { | 225 | { |
312 | BUG_ON(NULL == ppc_md.idle_loop); | 226 | BUG_ON(NULL == ppc_md.idle_loop); |