Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

cpuidle: pseries: Fixup exit latency for CEDE(0)

We are currently assuming that CEDE(0) has exit latency 10us, since
there is no way for us to query from the platform. However, if the
wakeup latency of an Extended CEDE state is smaller than 10us, then we
can be sure that the exit latency of CEDE(0) cannot be more than that.

In this patch, we fix the exit latency of CEDE(0) if we discover an
Extended CEDE state with wakeup latency smaller than 10us.

Benchmark results:

On POWER8, this patch does not have any impact since the advertized
latency of Extended CEDE (1) is 30us which is higher than the default
latency of CEDE (0) which is 10us.

On POWER9 we see improvement the single-threaded performance of
ebizzy, and no regression in the wakeup latency or the number of
context-switches.

ebizzy:
2 ebizzy threads bound to the same big-core. 25% improvement in the
avg records/s with patch.

x without_patch
* with_patch
N Min Max Median Avg Stddev
x 10 2491089 5834307 5398375 4244335 1596244.9
* 10 2893813 5834474 5832448 5327281.3 1055941.4

context_switch2:
There is no major regression observed with this patch as seen from the
context_switch2 benchmark.

context_switch2 across CPU0 CPU1 (Both belong to same big-core, but
different small cores). We observe a minor 0.14% regression in the
number of context-switches (higher is better).

x without_patch
* with_patch
N Min Max Median Avg Stddev
x 500 348872 362236 354712 354745.69 2711.827
* 500 349422 361452 353942 354215.4 2576.9258

Difference at 99.0% confidence
-530.288 +/- 430.963
-0.149484% +/- 0.121485%
(Student's t, pooled s = 2645.24)

context_switch2 across CPU0 CPU8 (Different big-cores). We observe a
0.37% improvement in the number of context-switches (higher is
better).

x without_patch
* with_patch
N Min Max Median Avg Stddev
x 500 287956 294940 288896 288977.23 646.59295
* 500 288300 294646 289582 290064.76 1161.9992

Difference at 99.0% confidence
1087.53 +/- 153.194
0.376337% +/- 0.0530125%
(Student's t, pooled s = 940.299)

schbench:
No major difference could be seen until the 99.9th percentile.

Without-patch:
Latency percentiles (usec)
50.0th: 29
75.0th: 39
90.0th: 49
95.0th: 59
*99.0th: 13104
99.5th: 14672
99.9th: 15824
min=0, max=17993

With-patch:
Latency percentiles (usec)
50.0th: 29
75.0th: 40
90.0th: 50
95.0th: 61
*99.0th: 13648
99.5th: 14768
99.9th: 15664
min=0, max=29812

Signed-off-by: Gautham R. Shenoy <ego@linux.vnet.ibm.com>
[mpe: Minor formatting]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1596087177-30329-4-git-send-email-ego@linux.vnet.ibm.com

authored by

Gautham R. Shenoy and committed by
Michael Ellerman
d947fb4c 054e44ba

+42 -3
+42 -3
drivers/cpuidle/cpuidle-pseries.c
··· 343 343 return 0; 344 344 } 345 345 346 - static void __init parse_xcede_idle_states(void) 346 + static void __init fixup_cede0_latency(void) 347 347 { 348 + struct xcede_latency_payload *payload; 349 + u64 min_latency_us; 350 + int i; 351 + 352 + min_latency_us = dedicated_states[1].exit_latency; // CEDE latency 353 + 348 354 if (parse_cede_parameters()) 349 355 return; 350 356 351 - pr_info("cpuidle : Skipping the %d Extended CEDE idle states\n", 357 + pr_info("cpuidle: Skipping the %d Extended CEDE idle states\n", 352 358 nr_xcede_records); 359 + 360 + payload = &xcede_latency_parameter.payload; 361 + for (i = 0; i < nr_xcede_records; i++) { 362 + struct xcede_latency_record *record = &payload->records[i]; 363 + u64 latency_tb = be64_to_cpu(record->latency_ticks); 364 + u64 latency_us = tb_to_ns(latency_tb) / NSEC_PER_USEC; 365 + 366 + if (latency_us < min_latency_us) 367 + min_latency_us = latency_us; 368 + } 369 + 370 + /* 371 + * By default, we assume that CEDE(0) has exit latency 10us, 372 + * since there is no way for us to query from the platform. 373 + * 374 + * However, if the wakeup latency of an Extended CEDE state is 375 + * smaller than 10us, then we can be sure that CEDE(0) 376 + * requires no more than that. 377 + * 378 + * Perform the fix-up. 379 + */ 380 + if (min_latency_us < dedicated_states[1].exit_latency) { 381 + u64 cede0_latency = min_latency_us - 1; 382 + 383 + if (cede0_latency <= 0) 384 + cede0_latency = min_latency_us; 385 + 386 + dedicated_states[1].exit_latency = cede0_latency; 387 + dedicated_states[1].target_residency = 10 * (cede0_latency); 388 + pr_info("cpuidle: Fixed up CEDE exit latency to %llu us\n", 389 + cede0_latency); 390 + } 391 + 353 392 } 354 393 355 394 /* ··· 412 373 cpuidle_state_table = shared_states; 413 374 max_idle_state = ARRAY_SIZE(shared_states); 414 375 } else { 415 - parse_xcede_idle_states(); 376 + fixup_cede0_latency(); 416 377 cpuidle_state_table = dedicated_states; 417 378 max_idle_state = NR_DEDICATED_STATES; 418 379 }