intel_idle: improve C-state flags handling robustness
authorArtem Bityutskiy <artem.bityutskiy@linux.intel.com>
Wed, 19 Apr 2023 14:39:44 +0000 (17:39 +0300)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Thu, 27 Apr 2023 17:37:36 +0000 (19:37 +0200)
The following C-state flags are currently mutually-exclusive and should not
be combined:
  * IRQ_ENABLE
  * IBRS
  * XSTATE

There is a warning for the situation when the IRQ_ENABLE flag
is combined with the IBRS flag, but no warnings for other combinations.
This is inconsistent and prone to errors.

Improve the situation by adding warnings for all the unexpected
combinations. Add a couple of helpful commentaries too.

Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Reviewed-by: Zhang Rui <rui.zhang@intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
drivers/idle/intel_idle.c

index 73ddb1d8cfcf8eaebb5877a8536e26cf2da6b6ce..1de36df15d5a9de6602b0f05ffb32b0bda34e10b 100644 (file)
@@ -1896,20 +1896,28 @@ static void __init intel_idle_init_cstates_icpu(struct cpuidle_driver *drv)
                drv->states[drv->state_count] = cpuidle_state_table[cstate];
                state = &drv->states[drv->state_count];
 
-               if ((state->flags & CPUIDLE_FLAG_IRQ_ENABLE) || force_irq_on) {
-                       pr_info("forced intel_idle_irq for state %d\n", cstate);
-                       state->enter = intel_idle_irq;
-               }
-
-               if (cpu_feature_enabled(X86_FEATURE_KERNEL_IBRS) &&
-                   state->flags & CPUIDLE_FLAG_IBRS) {
+               if (state->flags & CPUIDLE_FLAG_INIT_XSTATE) {
+                       /*
+                        * Combining with XSTATE with IBRS or IRQ_ENABLE flags
+                        * is not currently supported but this driver.
+                        */
+                       WARN_ON_ONCE(state->flags & CPUIDLE_FLAG_IBRS);
+                       WARN_ON_ONCE(state->flags & CPUIDLE_FLAG_IRQ_ENABLE);
+                       state->enter = intel_idle_xstate;
+               } else if (cpu_feature_enabled(X86_FEATURE_KERNEL_IBRS) &&
+                          state->flags & CPUIDLE_FLAG_IBRS) {
+                       /*
+                        * IBRS mitigation requires that C-states are entered
+                        * with interrupts disabled.
+                        */
                        WARN_ON_ONCE(state->flags & CPUIDLE_FLAG_IRQ_ENABLE);
                        state->enter = intel_idle_ibrs;
+               } else if ((state->flags & CPUIDLE_FLAG_IRQ_ENABLE) ||
+                          force_irq_on) {
+                       pr_info("forced intel_idle_irq for state %d\n", cstate);
+                       state->enter = intel_idle_irq;
                }
 
-               if (state->flags & CPUIDLE_FLAG_INIT_XSTATE)
-                       state->enter = intel_idle_xstate;
-
                if ((disabled_states_mask & BIT(drv->state_count)) ||
                    ((icpu->use_acpi || force_use_acpi) &&
                     intel_idle_off_by_default(mwait_hint) &&