aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/idle/intel_idle.c75
1 files changed, 26 insertions, 49 deletions
diff --git a/drivers/idle/intel_idle.c b/drivers/idle/intel_idle.c
index 2df9414a72f7..b2cf489ba3e1 100644
--- a/drivers/idle/intel_idle.c
+++ b/drivers/idle/intel_idle.c
@@ -109,6 +109,16 @@ static struct cpuidle_state *cpuidle_state_table;
109#define CPUIDLE_FLAG_TLB_FLUSHED 0x10000 109#define CPUIDLE_FLAG_TLB_FLUSHED 0x10000
110 110
111/* 111/*
112 * MWAIT takes an 8-bit "hint" in EAX "suggesting"
113 * the C-state (top nibble) and sub-state (bottom nibble)
114 * 0x00 means "MWAIT(C1)", 0x10 means "MWAIT(C2)" etc.
115 *
116 * We store the hint at the top of our "flags" for each state.
117 */
118#define flg2MWAIT(flags) (((flags) >> 24) & 0xFF)
119#define MWAIT2flg(eax) ((eax & 0xFF) << 24)
120
121/*
112 * States are indexed by the cstate number, 122 * States are indexed by the cstate number,
113 * which is also the index into the MWAIT hint array. 123 * which is also the index into the MWAIT hint array.
114 * Thus C0 is a dummy. 124 * Thus C0 is a dummy.
@@ -118,21 +128,21 @@ static struct cpuidle_state nehalem_cstates[MWAIT_MAX_NUM_CSTATES] = {
118 { /* MWAIT C1 */ 128 { /* MWAIT C1 */
119 .name = "C1-NHM", 129 .name = "C1-NHM",
120 .desc = "MWAIT 0x00", 130 .desc = "MWAIT 0x00",
121 .flags = CPUIDLE_FLAG_TIME_VALID, 131 .flags = MWAIT2flg(0x00) | CPUIDLE_FLAG_TIME_VALID,
122 .exit_latency = 3, 132 .exit_latency = 3,
123 .target_residency = 6, 133 .target_residency = 6,
124 .enter = &intel_idle }, 134 .enter = &intel_idle },
125 { /* MWAIT C2 */ 135 { /* MWAIT C2 */
126 .name = "C3-NHM", 136 .name = "C3-NHM",
127 .desc = "MWAIT 0x10", 137 .desc = "MWAIT 0x10",
128 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 138 .flags = MWAIT2flg(0x10) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
129 .exit_latency = 20, 139 .exit_latency = 20,
130 .target_residency = 80, 140 .target_residency = 80,
131 .enter = &intel_idle }, 141 .enter = &intel_idle },
132 { /* MWAIT C3 */ 142 { /* MWAIT C3 */
133 .name = "C6-NHM", 143 .name = "C6-NHM",
134 .desc = "MWAIT 0x20", 144 .desc = "MWAIT 0x20",
135 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 145 .flags = MWAIT2flg(0x20) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
136 .exit_latency = 200, 146 .exit_latency = 200,
137 .target_residency = 800, 147 .target_residency = 800,
138 .enter = &intel_idle }, 148 .enter = &intel_idle },
@@ -143,28 +153,28 @@ static struct cpuidle_state snb_cstates[MWAIT_MAX_NUM_CSTATES] = {
143 { /* MWAIT C1 */ 153 { /* MWAIT C1 */
144 .name = "C1-SNB", 154 .name = "C1-SNB",
145 .desc = "MWAIT 0x00", 155 .desc = "MWAIT 0x00",
146 .flags = CPUIDLE_FLAG_TIME_VALID, 156 .flags = MWAIT2flg(0x00) | CPUIDLE_FLAG_TIME_VALID,
147 .exit_latency = 1, 157 .exit_latency = 1,
148 .target_residency = 1, 158 .target_residency = 1,
149 .enter = &intel_idle }, 159 .enter = &intel_idle },
150 { /* MWAIT C2 */ 160 { /* MWAIT C2 */
151 .name = "C3-SNB", 161 .name = "C3-SNB",
152 .desc = "MWAIT 0x10", 162 .desc = "MWAIT 0x10",
153 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 163 .flags = MWAIT2flg(0x10) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
154 .exit_latency = 80, 164 .exit_latency = 80,
155 .target_residency = 211, 165 .target_residency = 211,
156 .enter = &intel_idle }, 166 .enter = &intel_idle },
157 { /* MWAIT C3 */ 167 { /* MWAIT C3 */
158 .name = "C6-SNB", 168 .name = "C6-SNB",
159 .desc = "MWAIT 0x20", 169 .desc = "MWAIT 0x20",
160 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 170 .flags = MWAIT2flg(0x20) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
161 .exit_latency = 104, 171 .exit_latency = 104,
162 .target_residency = 345, 172 .target_residency = 345,
163 .enter = &intel_idle }, 173 .enter = &intel_idle },
164 { /* MWAIT C4 */ 174 { /* MWAIT C4 */
165 .name = "C7-SNB", 175 .name = "C7-SNB",
166 .desc = "MWAIT 0x30", 176 .desc = "MWAIT 0x30",
167 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 177 .flags = MWAIT2flg(0x30) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
168 .exit_latency = 109, 178 .exit_latency = 109,
169 .target_residency = 345, 179 .target_residency = 345,
170 .enter = &intel_idle }, 180 .enter = &intel_idle },
@@ -175,28 +185,28 @@ static struct cpuidle_state ivb_cstates[MWAIT_MAX_NUM_CSTATES] = {
175 { /* MWAIT C1 */ 185 { /* MWAIT C1 */
176 .name = "C1-IVB", 186 .name = "C1-IVB",
177 .desc = "MWAIT 0x00", 187 .desc = "MWAIT 0x00",
178 .flags = CPUIDLE_FLAG_TIME_VALID, 188 .flags = MWAIT2flg(0x00) | CPUIDLE_FLAG_TIME_VALID,
179 .exit_latency = 1, 189 .exit_latency = 1,
180 .target_residency = 1, 190 .target_residency = 1,
181 .enter = &intel_idle }, 191 .enter = &intel_idle },
182 { /* MWAIT C2 */ 192 { /* MWAIT C2 */
183 .name = "C3-IVB", 193 .name = "C3-IVB",
184 .desc = "MWAIT 0x10", 194 .desc = "MWAIT 0x10",
185 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 195 .flags = MWAIT2flg(0x10) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
186 .exit_latency = 59, 196 .exit_latency = 59,
187 .target_residency = 156, 197 .target_residency = 156,
188 .enter = &intel_idle }, 198 .enter = &intel_idle },
189 { /* MWAIT C3 */ 199 { /* MWAIT C3 */
190 .name = "C6-IVB", 200 .name = "C6-IVB",
191 .desc = "MWAIT 0x20", 201 .desc = "MWAIT 0x20",
192 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 202 .flags = MWAIT2flg(0x20) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
193 .exit_latency = 80, 203 .exit_latency = 80,
194 .target_residency = 300, 204 .target_residency = 300,
195 .enter = &intel_idle }, 205 .enter = &intel_idle },
196 { /* MWAIT C4 */ 206 { /* MWAIT C4 */
197 .name = "C7-IVB", 207 .name = "C7-IVB",
198 .desc = "MWAIT 0x30", 208 .desc = "MWAIT 0x30",
199 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 209 .flags = MWAIT2flg(0x30) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
200 .exit_latency = 87, 210 .exit_latency = 87,
201 .target_residency = 300, 211 .target_residency = 300,
202 .enter = &intel_idle }, 212 .enter = &intel_idle },
@@ -207,14 +217,14 @@ static struct cpuidle_state atom_cstates[MWAIT_MAX_NUM_CSTATES] = {
207 { /* MWAIT C1 */ 217 { /* MWAIT C1 */
208 .name = "C1-ATM", 218 .name = "C1-ATM",
209 .desc = "MWAIT 0x00", 219 .desc = "MWAIT 0x00",
210 .flags = CPUIDLE_FLAG_TIME_VALID, 220 .flags = MWAIT2flg(0x00) | CPUIDLE_FLAG_TIME_VALID,
211 .exit_latency = 1, 221 .exit_latency = 1,
212 .target_residency = 4, 222 .target_residency = 4,
213 .enter = &intel_idle }, 223 .enter = &intel_idle },
214 { /* MWAIT C2 */ 224 { /* MWAIT C2 */
215 .name = "C2-ATM", 225 .name = "C2-ATM",
216 .desc = "MWAIT 0x10", 226 .desc = "MWAIT 0x10",
217 .flags = CPUIDLE_FLAG_TIME_VALID, 227 .flags = MWAIT2flg(0x10) | CPUIDLE_FLAG_TIME_VALID,
218 .exit_latency = 20, 228 .exit_latency = 20,
219 .target_residency = 80, 229 .target_residency = 80,
220 .enter = &intel_idle }, 230 .enter = &intel_idle },
@@ -222,7 +232,7 @@ static struct cpuidle_state atom_cstates[MWAIT_MAX_NUM_CSTATES] = {
222 { /* MWAIT C4 */ 232 { /* MWAIT C4 */
223 .name = "C4-ATM", 233 .name = "C4-ATM",
224 .desc = "MWAIT 0x30", 234 .desc = "MWAIT 0x30",
225 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 235 .flags = MWAIT2flg(0x30) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
226 .exit_latency = 100, 236 .exit_latency = 100,
227 .target_residency = 400, 237 .target_residency = 400,
228 .enter = &intel_idle }, 238 .enter = &intel_idle },
@@ -230,41 +240,12 @@ static struct cpuidle_state atom_cstates[MWAIT_MAX_NUM_CSTATES] = {
230 { /* MWAIT C6 */ 240 { /* MWAIT C6 */
231 .name = "C6-ATM", 241 .name = "C6-ATM",
232 .desc = "MWAIT 0x52", 242 .desc = "MWAIT 0x52",
233 .flags = CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED, 243 .flags = MWAIT2flg(0x52) | CPUIDLE_FLAG_TIME_VALID | CPUIDLE_FLAG_TLB_FLUSHED,
234 .exit_latency = 140, 244 .exit_latency = 140,
235 .target_residency = 560, 245 .target_residency = 560,
236 .enter = &intel_idle }, 246 .enter = &intel_idle },
237}; 247};
238 248
239static long get_driver_data(int cstate)
240{
241 int driver_data;
242 switch (cstate) {
243
244 case 1: /* MWAIT C1 */
245 driver_data = 0x00;
246 break;
247 case 2: /* MWAIT C2 */
248 driver_data = 0x10;
249 break;
250 case 3: /* MWAIT C3 */
251 driver_data = 0x20;
252 break;
253 case 4: /* MWAIT C4 */
254 driver_data = 0x30;
255 break;
256 case 5: /* MWAIT C5 */
257 driver_data = 0x40;
258 break;
259 case 6: /* MWAIT C6 */
260 driver_data = 0x52;
261 break;
262 default:
263 driver_data = 0x00;
264 }
265 return driver_data;
266}
267
268/** 249/**
269 * intel_idle 250 * intel_idle
270 * @dev: cpuidle_device 251 * @dev: cpuidle_device
@@ -278,8 +259,7 @@ static int intel_idle(struct cpuidle_device *dev,
278{ 259{
279 unsigned long ecx = 1; /* break on interrupt flag */ 260 unsigned long ecx = 1; /* break on interrupt flag */
280 struct cpuidle_state *state = &drv->states[index]; 261 struct cpuidle_state *state = &drv->states[index];
281 struct cpuidle_state_usage *state_usage = &dev->states_usage[index]; 262 unsigned long eax = flg2MWAIT(state->flags);
282 unsigned long eax = (unsigned long)cpuidle_get_statedata(state_usage);
283 unsigned int cstate; 263 unsigned int cstate;
284 int cpu = smp_processor_id(); 264 int cpu = smp_processor_id();
285 265
@@ -558,9 +538,6 @@ static int intel_idle_cpu_init(int cpu)
558 if (cpuidle_state_table[cstate].enter == NULL) 538 if (cpuidle_state_table[cstate].enter == NULL)
559 continue; 539 continue;
560 540
561 dev->states_usage[dev->state_count].driver_data =
562 (void *)get_driver_data(cstate);
563
564 dev->state_count += 1; 541 dev->state_count += 1;
565 } 542 }
566 543