* [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered
@ 2026-02-11 5:35 Aboorva Devarajan
2026-02-11 15:00 ` Christian Loehle
0 siblings, 1 reply; 6+ messages in thread
From: Aboorva Devarajan @ 2026-02-11 5:35 UTC (permalink / raw)
To: rafael, christian.loehle, daniel.lezcano; +Cc: aboorvad, linux-pm, linux-kernel
On certain platforms (PowerNV systems without a power-mgt DT node),
cpuidle may register only a single idle state. In cases where that
single state is a polling state (state 0), the ladder governor may
incorrectly treat state 1 as the first usable state and pass an
out-of-bounds index. This can lead to a NULL enter callback being
invoked, ultimately resulting in a system crash.
[ 13.342636] cpuidle-powernv : Only Snooze is available
[ 13.351854] Faulting instruction address: 0x00000000
[ 13.376489] NIP [0000000000000000] 0x0
[ 13.378351] LR [c000000001e01974] cpuidle_enter_state+0x2c4/0x668
Fix this by determining the first non-polling state index based on
the number of registered states, and by returning state 0 when only
one state is registered.
Fixes: dc2251bf98c6 ("cpuidle: Eliminate the CPUIDLE_DRIVER_STATE_START symbol")
Signed-off-by: Aboorva Devarajan <aboorvad@linux.ibm.com>
---
drivers/cpuidle/governors/ladder.c | 24 ++++++++++++++++++++----
1 file changed, 20 insertions(+), 4 deletions(-)
diff --git a/drivers/cpuidle/governors/ladder.c b/drivers/cpuidle/governors/ladder.c
index 6617eb494a11..294a688ed0bb 100644
--- a/drivers/cpuidle/governors/ladder.c
+++ b/drivers/cpuidle/governors/ladder.c
@@ -42,6 +42,21 @@ struct ladder_device {
static DEFINE_PER_CPU(struct ladder_device, ladder_devices);
+/**
+ * ladder_get_first_idx - get the first non-polling state index
+ * @drv: cpuidle driver
+ *
+ * Returns the index of the first non-polling state, or 0 if state 0 is not
+ * polling or if there's only one state available.
+ */
+static inline int ladder_get_first_idx(struct cpuidle_driver *drv)
+{
+ if (drv->state_count > 1 &&
+ drv->states[0].flags & CPUIDLE_FLAG_POLLING)
+ return 1;
+ return 0;
+}
+
/**
* ladder_do_selection - prepares private data for a state change
* @dev: the CPU
@@ -70,16 +85,17 @@ static int ladder_select_state(struct cpuidle_driver *drv,
struct ladder_device *ldev = this_cpu_ptr(&ladder_devices);
struct ladder_device_state *last_state;
int last_idx = dev->last_state_idx;
- int first_idx = drv->states[0].flags & CPUIDLE_FLAG_POLLING ? 1 : 0;
+ int first_idx;
s64 latency_req = cpuidle_governor_latency_req(dev->cpu);
s64 last_residency;
- /* Special case when user has set very strict latency requirement */
- if (unlikely(latency_req == 0)) {
+ /* Special case when there's only one state or strict latency requirement */
+ if (unlikely(drv->state_count <= 1 || latency_req == 0)) {
ladder_do_selection(dev, ldev, last_idx, 0);
return 0;
}
+ first_idx = ladder_get_first_idx(drv);
last_state = &ldev->states[last_idx];
last_residency = dev->last_residency_ns - drv->states[last_idx].exit_latency_ns;
@@ -134,7 +150,7 @@ static int ladder_enable_device(struct cpuidle_driver *drv,
struct cpuidle_device *dev)
{
int i;
- int first_idx = drv->states[0].flags & CPUIDLE_FLAG_POLLING ? 1 : 0;
+ int first_idx = ladder_get_first_idx(drv);
struct ladder_device *ldev = &per_cpu(ladder_devices, dev->cpu);
struct ladder_device_state *lstate;
struct cpuidle_state *state;
--
2.52.0
^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered
2026-02-11 5:35 [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered Aboorva Devarajan
@ 2026-02-11 15:00 ` Christian Loehle
2026-02-13 8:29 ` Aboorva Devarajan
0 siblings, 1 reply; 6+ messages in thread
From: Christian Loehle @ 2026-02-11 15:00 UTC (permalink / raw)
To: Aboorva Devarajan, rafael, daniel.lezcano; +Cc: linux-pm, linux-kernel
On 2/11/26 05:35, Aboorva Devarajan wrote:
> On certain platforms (PowerNV systems without a power-mgt DT node),
> cpuidle may register only a single idle state. In cases where that
> single state is a polling state (state 0), the ladder governor may
> incorrectly treat state 1 as the first usable state and pass an
> out-of-bounds index. This can lead to a NULL enter callback being
> invoked, ultimately resulting in a system crash.
>
> [ 13.342636] cpuidle-powernv : Only Snooze is available
> [ 13.351854] Faulting instruction address: 0x00000000
> [ 13.376489] NIP [0000000000000000] 0x0
> [ 13.378351] LR [c000000001e01974] cpuidle_enter_state+0x2c4/0x668
>
> Fix this by determining the first non-polling state index based on
> the number of registered states, and by returning state 0 when only
> one state is registered.
>
> Fixes: dc2251bf98c6 ("cpuidle: Eliminate the CPUIDLE_DRIVER_STATE_START symbol")
> Signed-off-by: Aboorva Devarajan <aboorvad@linux.ibm.com>
Agreed that the current behavior is a bug, but is there really much value
in using a cpuidle governor with just a polling state?
It's dead code and trivial to bail out of in cpuidle, right?
> ---
> drivers/cpuidle/governors/ladder.c | 24 ++++++++++++++++++++----
> 1 file changed, 20 insertions(+), 4 deletions(-)
>
> diff --git a/drivers/cpuidle/governors/ladder.c b/drivers/cpuidle/governors/ladder.c
> index 6617eb494a11..294a688ed0bb 100644
> --- a/drivers/cpuidle/governors/ladder.c
> +++ b/drivers/cpuidle/governors/ladder.c
> @@ -42,6 +42,21 @@ struct ladder_device {
>
> static DEFINE_PER_CPU(struct ladder_device, ladder_devices);
>
> +/**
> + * ladder_get_first_idx - get the first non-polling state index
> + * @drv: cpuidle driver
> + *
> + * Returns the index of the first non-polling state, or 0 if state 0 is not
> + * polling or if there's only one state available.
> + */
> +static inline int ladder_get_first_idx(struct cpuidle_driver *drv)
> +{
> + if (drv->state_count > 1 &&
> + drv->states[0].flags & CPUIDLE_FLAG_POLLING)
> + return 1;
> + return 0;
> +}
> +
> /**
> * ladder_do_selection - prepares private data for a state change
> * @dev: the CPU
> @@ -70,16 +85,17 @@ static int ladder_select_state(struct cpuidle_driver *drv,
> struct ladder_device *ldev = this_cpu_ptr(&ladder_devices);
> struct ladder_device_state *last_state;
> int last_idx = dev->last_state_idx;
> - int first_idx = drv->states[0].flags & CPUIDLE_FLAG_POLLING ? 1 : 0;
> + int first_idx;
> s64 latency_req = cpuidle_governor_latency_req(dev->cpu);
> s64 last_residency;
>
> - /* Special case when user has set very strict latency requirement */
> - if (unlikely(latency_req == 0)) {
> + /* Special case when there's only one state or strict latency requirement */
> + if (unlikely(drv->state_count <= 1 || latency_req == 0)) {
> ladder_do_selection(dev, ldev, last_idx, 0);
> return 0;
> }
>
> + first_idx = ladder_get_first_idx(drv);
> last_state = &ldev->states[last_idx];
>
> last_residency = dev->last_residency_ns - drv->states[last_idx].exit_latency_ns;
> @@ -134,7 +150,7 @@ static int ladder_enable_device(struct cpuidle_driver *drv,
> struct cpuidle_device *dev)
> {
> int i;
> - int first_idx = drv->states[0].flags & CPUIDLE_FLAG_POLLING ? 1 : 0;
> + int first_idx = ladder_get_first_idx(drv);
> struct ladder_device *ldev = &per_cpu(ladder_devices, dev->cpu);
> struct ladder_device_state *lstate;
> struct cpuidle_state *state;
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered
2026-02-11 15:00 ` Christian Loehle
@ 2026-02-13 8:29 ` Aboorva Devarajan
2026-02-13 13:44 ` Christian Loehle
0 siblings, 1 reply; 6+ messages in thread
From: Aboorva Devarajan @ 2026-02-13 8:29 UTC (permalink / raw)
To: Christian Loehle, rafael; +Cc: daniel.lezcano, linux-pm, linux-kernel
On Wed, 2026-02-11 at 15:00 +0000, Christian Loehle wrote:
> On 2/11/26 05:35, Aboorva Devarajan wrote:
> > On certain platforms (PowerNV systems without a power-mgt DT node),
> > cpuidle may register only a single idle state. In cases where that
> > single state is a polling state (state 0), the ladder governor may
> > incorrectly treat state 1 as the first usable state and pass an
> > out-of-bounds index. This can lead to a NULL enter callback being
> > invoked, ultimately resulting in a system crash.
> >
> > [ 13.342636] cpuidle-powernv : Only Snooze is available
> > [ 13.351854] Faulting instruction address: 0x00000000
> > [ 13.376489] NIP [0000000000000000] 0x0
> > [ 13.378351] LR [c000000001e01974] cpuidle_enter_state+0x2c4/0x668
> >
> > Fix this by determining the first non-polling state index based on
> > the number of registered states, and by returning state 0 when only
> > one state is registered.
> >
> > Fixes: dc2251bf98c6 ("cpuidle: Eliminate the CPUIDLE_DRIVER_STATE_START symbol")
> > Signed-off-by: Aboorva Devarajan <aboorvad@linux.ibm.com>
>
> Agreed that the current behavior is a bug, but is there really much value
> in using a cpuidle governor with just a polling state?
> It's dead code and trivial to bail out of in cpuidle, right?
>
Hi Christian,
Thanks for the review.
Other governors (teo, menu) already handle this single-state scenario
correctly. Fixing ladder's first_idx calculation seemed like the most
targeted fix, however since ladder is not widely used this is likely
to go unnoticed, it only popped up during testing with a missing
power-mgt device tree node.
yes, adding a bail-out in the core cpuidle_select() is also trivial and
would benefit all governors uniformly. Setting stop_tick to false keeps
the tick running, which is correct for a single state configuration.
Please let me know if you'd prefer this approach instead.
---
diff --git a/drivers/cpuidle/cpuidle.c b/drivers/cpuidle/cpuidle.c
index c7876e9e024f..ea082419f7db 100644
--- a/drivers/cpuidle/cpuidle.c
+++ b/drivers/cpuidle/cpuidle.c
@@ -359,6 +359,16 @@ noinstr int cpuidle_enter_state(struct
cpuidle_device *dev,
int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device
*dev,
bool *stop_tick)
{
+ /*
+ * If there is only a single idle state (or none), there is
nothing
+ * meaningful for the governor to choose. Skip the governor and
+ * always use state 0 with the tick running.
+ */
+ if (unlikely(drv->state_count <= 1)) {
+ *stop_tick = false;
+ return 0;
+ }
+
return cpuidle_curr_governor->select(drv, dev, stop_tick);
}
---
Regards,
Aboorva
> > ---
> > drivers/cpuidle/governors/ladder.c | 24 ++++++++++++++++++++----
> > 1 file changed, 20 insertions(+), 4 deletions(-)
> >
> > diff --git a/drivers/cpuidle/governors/ladder.c b/drivers/cpuidle/governors/ladder.c
> > index 6617eb494a11..294a688ed0bb 100644
> > --- a/drivers/cpuidle/governors/ladder.c
> > +++ b/drivers/cpuidle/governors/ladder.c
> > @@ -42,6 +42,21 @@ struct ladder_device {
> >
> > static DEFINE_PER_CPU(struct ladder_device, ladder_devices);
> >
> > +/**
> > + * ladder_get_first_idx - get the first non-polling state index
> > + * @drv: cpuidle driver
> > + *
> > + * Returns the index of the first non-polling state, or 0 if state 0 is not
> > + * polling or if there's only one state available.
> > + */
> > +static inline int ladder_get_first_idx(struct cpuidle_driver *drv)
> > +{
> > + if (drv->state_count > 1 &&
> > + drv->states[0].flags & CPUIDLE_FLAG_POLLING)
> > + return 1;
> > + return 0;
> > +}
> > +
> > /**
> > * ladder_do_selection - prepares private data for a state change
> > * @dev: the CPU
> > @@ -70,16 +85,17 @@ static int ladder_select_state(struct cpuidle_driver *drv,
> > struct ladder_device *ldev = this_cpu_ptr(&ladder_devices);
> > struct ladder_device_state *last_state;
> > int last_idx = dev->last_state_idx;
> > - int first_idx = drv->states[0].flags & CPUIDLE_FLAG_POLLING ? 1 : 0;
> > + int first_idx;
> > s64 latency_req = cpuidle_governor_latency_req(dev->cpu);
> > s64 last_residency;
> >
> > - /* Special case when user has set very strict latency requirement */
> > - if (unlikely(latency_req == 0)) {
> > + /* Special case when there's only one state or strict latency requirement */
> > + if (unlikely(drv->state_count <= 1 || latency_req == 0)) {
> > ladder_do_selection(dev, ldev, last_idx, 0);
> > return 0;
> > }
> >
> > + first_idx = ladder_get_first_idx(drv);
> > last_state = &ldev->states[last_idx];
> >
> > last_residency = dev->last_residency_ns - drv->states[last_idx].exit_latency_ns;
> > @@ -134,7 +150,7 @@ static int ladder_enable_device(struct cpuidle_driver *drv,
> > struct cpuidle_device *dev)
> > {
> > int i;
> > - int first_idx = drv->states[0].flags & CPUIDLE_FLAG_POLLING ? 1 : 0;
> > + int first_idx = ladder_get_first_idx(drv);
> > struct ladder_device *ldev = &per_cpu(ladder_devices, dev->cpu);
> > struct ladder_device_state *lstate;
> > struct cpuidle_state *state;
^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered
2026-02-13 8:29 ` Aboorva Devarajan
@ 2026-02-13 13:44 ` Christian Loehle
2026-02-16 11:05 ` Christian Loehle
0 siblings, 1 reply; 6+ messages in thread
From: Christian Loehle @ 2026-02-13 13:44 UTC (permalink / raw)
To: Aboorva Devarajan, rafael; +Cc: daniel.lezcano, linux-pm, linux-kernel
On 2/13/26 08:29, Aboorva Devarajan wrote:
> On Wed, 2026-02-11 at 15:00 +0000, Christian Loehle wrote:
>> On 2/11/26 05:35, Aboorva Devarajan wrote:
>>> On certain platforms (PowerNV systems without a power-mgt DT node),
>>> cpuidle may register only a single idle state. In cases where that
>>> single state is a polling state (state 0), the ladder governor may
>>> incorrectly treat state 1 as the first usable state and pass an
>>> out-of-bounds index. This can lead to a NULL enter callback being
>>> invoked, ultimately resulting in a system crash.
>>>
>>> [ 13.342636] cpuidle-powernv : Only Snooze is available
>>> [ 13.351854] Faulting instruction address: 0x00000000
>>> [ 13.376489] NIP [0000000000000000] 0x0
>>> [ 13.378351] LR [c000000001e01974] cpuidle_enter_state+0x2c4/0x668
>>>
>>> Fix this by determining the first non-polling state index based on
>>> the number of registered states, and by returning state 0 when only
>>> one state is registered.
>>>
>>> Fixes: dc2251bf98c6 ("cpuidle: Eliminate the CPUIDLE_DRIVER_STATE_START symbol")
>>> Signed-off-by: Aboorva Devarajan <aboorvad@linux.ibm.com>
>>
>> Agreed that the current behavior is a bug, but is there really much value
>> in using a cpuidle governor with just a polling state?
>> It's dead code and trivial to bail out of in cpuidle, right?
>>
>
> Hi Christian,
>
> Thanks for the review.
>
> Other governors (teo, menu) already handle this single-state scenario
> correctly. Fixing ladder's first_idx calculation seemed like the most
> targeted fix, however since ladder is not widely used this is likely
> to go unnoticed, it only popped up during testing with a missing
> power-mgt device tree node.
>
> yes, adding a bail-out in the core cpuidle_select() is also trivial and
> would benefit all governors uniformly. Setting stop_tick to false keeps
> the tick running, which is correct for a single state configuration.
>
> Please let me know if you'd prefer this approach instead.
>
> ---
>
> diff --git a/drivers/cpuidle/cpuidle.c b/drivers/cpuidle/cpuidle.c
> index c7876e9e024f..ea082419f7db 100644
> --- a/drivers/cpuidle/cpuidle.c
> +++ b/drivers/cpuidle/cpuidle.c
> @@ -359,6 +359,16 @@ noinstr int cpuidle_enter_state(struct
> cpuidle_device *dev,
> int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device
> *dev,
> bool *stop_tick)
> {
> + /*
> + * If there is only a single idle state (or none), there is
> nothing
> + * meaningful for the governor to choose. Skip the governor and
> + * always use state 0 with the tick running.
> + */
> + if (unlikely(drv->state_count <= 1)) {
I think the unlikely isn't helping here, this just let the branch predictor
handle this as it won't change anyway.
> + *stop_tick = false;
> + return 0;
> + }
> +
> return cpuidle_curr_governor->select(drv, dev, stop_tick);
> }
>
I prefer this, additionally of course:
-------8<-------
diff --git a/drivers/cpuidle/governors/menu.c b/drivers/cpuidle/governors/menu.c
index 64d6f7a1c776..fdfa5d7e10a6 100644
--- a/drivers/cpuidle/governors/menu.c
+++ b/drivers/cpuidle/governors/menu.c
@@ -271,7 +271,7 @@ static int menu_select(struct cpuidle_driver *drv, struct cpuidle_device *dev,
data->bucket = BUCKETS - 1;
}
- if (unlikely(drv->state_count <= 1 || latency_req == 0) ||
+ if (unlikely(latency_req == 0) ||
((data->next_timer_ns < drv->states[1].target_residency_ns ||
latency_req < drv->states[1].exit_latency_ns) &&
!dev->states_usage[0].disable)) {
diff --git a/drivers/cpuidle/governors/teo.c b/drivers/cpuidle/governors/teo.c
index 81ac5fd58a1c..9b5b8c617806 100644
--- a/drivers/cpuidle/governors/teo.c
+++ b/drivers/cpuidle/governors/teo.c
@@ -317,12 +317,6 @@ static int teo_select(struct cpuidle_driver *drv, struct cpuidle_device *dev,
*/
cpu_data->sleep_length_ns = KTIME_MAX;
- /* Check if there is any choice in the first place. */
- if (drv->state_count < 2) {
- idx = 0;
- goto out_tick;
- }
-
if (!dev->states_usage[0].disable)
idx = 0;
^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered
2026-02-13 13:44 ` Christian Loehle
@ 2026-02-16 11:05 ` Christian Loehle
2026-02-16 18:58 ` Aboorva Devarajan
0 siblings, 1 reply; 6+ messages in thread
From: Christian Loehle @ 2026-02-16 11:05 UTC (permalink / raw)
To: Aboorva Devarajan, rafael; +Cc: daniel.lezcano, linux-pm, linux-kernel
[-- Attachment #1: Type: text/plain, Size: 3245 bytes --]
On 2/13/26 13:44, Christian Loehle wrote:
> On 2/13/26 08:29, Aboorva Devarajan wrote:
>> On Wed, 2026-02-11 at 15:00 +0000, Christian Loehle wrote:
>>> On 2/11/26 05:35, Aboorva Devarajan wrote:
>>>> On certain platforms (PowerNV systems without a power-mgt DT node),
>>>> cpuidle may register only a single idle state. In cases where that
>>>> single state is a polling state (state 0), the ladder governor may
>>>> incorrectly treat state 1 as the first usable state and pass an
>>>> out-of-bounds index. This can lead to a NULL enter callback being
>>>> invoked, ultimately resulting in a system crash.
>>>>
>>>> [ 13.342636] cpuidle-powernv : Only Snooze is available
>>>> [ 13.351854] Faulting instruction address: 0x00000000
>>>> [ 13.376489] NIP [0000000000000000] 0x0
>>>> [ 13.378351] LR [c000000001e01974] cpuidle_enter_state+0x2c4/0x668
>>>>
>>>> Fix this by determining the first non-polling state index based on
>>>> the number of registered states, and by returning state 0 when only
>>>> one state is registered.
>>>>
>>>> Fixes: dc2251bf98c6 ("cpuidle: Eliminate the CPUIDLE_DRIVER_STATE_START symbol")
>>>> Signed-off-by: Aboorva Devarajan <aboorvad@linux.ibm.com>
>>>
>>> Agreed that the current behavior is a bug, but is there really much value
>>> in using a cpuidle governor with just a polling state?
>>> It's dead code and trivial to bail out of in cpuidle, right?
>>>
>>
>> Hi Christian,
>>
>> Thanks for the review.
>>
>> Other governors (teo, menu) already handle this single-state scenario
>> correctly. Fixing ladder's first_idx calculation seemed like the most
>> targeted fix, however since ladder is not widely used this is likely
>> to go unnoticed, it only popped up during testing with a missing
>> power-mgt device tree node.
>>
>> yes, adding a bail-out in the core cpuidle_select() is also trivial and
>> would benefit all governors uniformly. Setting stop_tick to false keeps
>> the tick running, which is correct for a single state configuration.
>>
>> Please let me know if you'd prefer this approach instead.
>>
>> ---
>>
>> diff --git a/drivers/cpuidle/cpuidle.c b/drivers/cpuidle/cpuidle.c
>> index c7876e9e024f..ea082419f7db 100644
>> --- a/drivers/cpuidle/cpuidle.c
>> +++ b/drivers/cpuidle/cpuidle.c
>> @@ -359,6 +359,16 @@ noinstr int cpuidle_enter_state(struct
>> cpuidle_device *dev,
>> int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device
>> *dev,
>> bool *stop_tick)
>> {
>> + /*
>> + * If there is only a single idle state (or none), there is
>> nothing
>> + * meaningful for the governor to choose. Skip the governor and
>> + * always use state 0 with the tick running.
>> + */
>> + if (unlikely(drv->state_count <= 1)) {
>
> I think the unlikely isn't helping here, this just let the branch predictor
> handle this as it won't change anyway.
>
>> + *stop_tick = false;
>> + return 0;
>> + }
>> +
>> return cpuidle_curr_governor->select(drv, dev, stop_tick);
>> }
>>
>
> I prefer this, additionally of course:
I've attached them as patches with a sign-off, feel free to pick them up as a series
or if you provide your signoff I can do that as well.
[-- Attachment #2: 0002-cpuidle-teo-Remove-single-state-handling.patch --]
[-- Type: text/x-patch, Size: 1073 bytes --]
From 56ba6cbec9aca645a51744261ecf0276072e001d Mon Sep 17 00:00:00 2001
From: Christian Loehle <christian.loehle@arm.com>
Date: Mon, 16 Feb 2026 11:02:55 +0000
Subject: [PATCH 2/2] cpuidle: teo: Remove single state handling
cpuidle systems where the governor has no choice because there's only
a single idle state are now handled by cpuidle core and bypass the
governor, so remove the related handling.
Signed-off-by: Christian Loehle <christian.loehle@arm.com>
---
drivers/cpuidle/governors/teo.c | 6 ------
1 file changed, 6 deletions(-)
diff --git a/drivers/cpuidle/governors/teo.c b/drivers/cpuidle/governors/teo.c
index 81ac5fd58a1c..9b5b8c617806 100644
--- a/drivers/cpuidle/governors/teo.c
+++ b/drivers/cpuidle/governors/teo.c
@@ -317,12 +317,6 @@ static int teo_select(struct cpuidle_driver *drv, struct cpuidle_device *dev,
*/
cpu_data->sleep_length_ns = KTIME_MAX;
- /* Check if there is any choice in the first place. */
- if (drv->state_count < 2) {
- idx = 0;
- goto out_tick;
- }
-
if (!dev->states_usage[0].disable)
idx = 0;
--
2.34.1
[-- Attachment #3: 0001-cpuidle-menu-Remove-single-state-handling.patch --]
[-- Type: text/x-patch, Size: 1167 bytes --]
From 1046bac618c05262d139071e55f4248d3121310d Mon Sep 17 00:00:00 2001
From: Christian Loehle <christian.loehle@arm.com>
Date: Mon, 16 Feb 2026 11:01:02 +0000
Subject: [PATCH 1/2] cpuidle: menu: Remove single state handling
cpuidle systems where the governor has no choice because there's only
a single idle state are now handled by cpuidle core and bypass the
governor, so remove the related handling.
Signed-off-by: Christian Loehle <christian.loehle@arm.com>
---
| 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
--git a/drivers/cpuidle/governors/menu.c b/drivers/cpuidle/governors/menu.c
index 64d6f7a1c776..fdfa5d7e10a6 100644
--- a/drivers/cpuidle/governors/menu.c
+++ b/drivers/cpuidle/governors/menu.c
@@ -271,7 +271,7 @@ static int menu_select(struct cpuidle_driver *drv, struct cpuidle_device *dev,
data->bucket = BUCKETS - 1;
}
- if (unlikely(drv->state_count <= 1 || latency_req == 0) ||
+ if (unlikely(latency_req == 0) ||
((data->next_timer_ns < drv->states[1].target_residency_ns ||
latency_req < drv->states[1].exit_latency_ns) &&
!dev->states_usage[0].disable)) {
--
2.34.1
^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered
2026-02-16 11:05 ` Christian Loehle
@ 2026-02-16 18:58 ` Aboorva Devarajan
0 siblings, 0 replies; 6+ messages in thread
From: Aboorva Devarajan @ 2026-02-16 18:58 UTC (permalink / raw)
To: Christian Loehle, rafael; +Cc: daniel.lezcano, linux-pm, linux-kernel
On Mon, 2026-02-16 at 11:05 +0000, Christian Loehle wrote:
> On 2/13/26 13:44, Christian Loehle wrote:
> > On 2/13/26 08:29, Aboorva Devarajan wrote:
> > > On Wed, 2026-02-11 at 15:00 +0000, Christian Loehle wrote:
> > > > On 2/11/26 05:35, Aboorva Devarajan wrote:
> > > > > On certain platforms (PowerNV systems without a power-mgt DT node),
> > > > > cpuidle may register only a single idle state. In cases where that
> > > > > single state is a polling state (state 0), the ladder governor may
> > > > > incorrectly treat state 1 as the first usable state and pass an
> > > > > out-of-bounds index. This can lead to a NULL enter callback being
> > > > > invoked, ultimately resulting in a system crash.
> > > > >
> > > > > [ 13.342636] cpuidle-powernv : Only Snooze is available
> > > > > [ 13.351854] Faulting instruction address: 0x00000000
> > > > > [ 13.376489] NIP [0000000000000000] 0x0
> > > > > [ 13.378351] LR [c000000001e01974] cpuidle_enter_state+0x2c4/0x668
> > > > >
> > > > > Fix this by determining the first non-polling state index based on
> > > > > the number of registered states, and by returning state 0 when only
> > > > > one state is registered.
> > > > >
> > > > > Fixes: dc2251bf98c6 ("cpuidle: Eliminate the CPUIDLE_DRIVER_STATE_START symbol")
> > > > > Signed-off-by: Aboorva Devarajan <aboorvad@linux.ibm.com>
> > > >
> > > > Agreed that the current behavior is a bug, but is there really much value
> > > > in using a cpuidle governor with just a polling state?
> > > > It's dead code and trivial to bail out of in cpuidle, right?
> > > >
> > >
> > > Hi Christian,
> > >
> > > Thanks for the review.
> > >
> > > Other governors (teo, menu) already handle this single-state scenario
> > > correctly. Fixing ladder's first_idx calculation seemed like the most
> > > targeted fix, however since ladder is not widely used this is likely
> > > to go unnoticed, it only popped up during testing with a missing
> > > power-mgt device tree node.
> > >
> > > yes, adding a bail-out in the core cpuidle_select() is also trivial and
> > > would benefit all governors uniformly. Setting stop_tick to false keeps
> > > the tick running, which is correct for a single state configuration.
> > >
> > > Please let me know if you'd prefer this approach instead.
> > >
> > > ---
> > >
> > > diff --git a/drivers/cpuidle/cpuidle.c b/drivers/cpuidle/cpuidle.c
> > > index c7876e9e024f..ea082419f7db 100644
> > > --- a/drivers/cpuidle/cpuidle.c
> > > +++ b/drivers/cpuidle/cpuidle.c
> > > @@ -359,6 +359,16 @@ noinstr int cpuidle_enter_state(struct
> > > cpuidle_device *dev,
> > > int cpuidle_select(struct cpuidle_driver *drv, struct cpuidle_device
> > > *dev,
> > > bool *stop_tick)
> > > {
> > > + /*
> > > + * If there is only a single idle state (or none), there is
> > > nothing
> > > + * meaningful for the governor to choose. Skip the governor and
> > > + * always use state 0 with the tick running.
> > > + */
> > > + if (unlikely(drv->state_count <= 1)) {
> >
> > I think the unlikely isn't helping here, this just let the branch predictor
> > handle this as it won't change anyway.
> >
> > > + *stop_tick = false;
> > > + return 0;
> > > + }
> > > +
> > > return cpuidle_curr_governor->select(drv, dev, stop_tick);
> > > }
> > >
> >
> > I prefer this, additionally of course:
>
> I've attached them as patches with a sign-off, feel free to pick them up as a series
> or if you provide your signoff I can do that as well.
Hi Christian,
Thanks, I've included your patches and sent them as a series:
https://lore.kernel.org/all/20260216185005.1131593-1-aboorvad@linux.ibm.com/
Regards,
Aboorva
^ permalink raw reply [flat|nested] 6+ messages in thread
end of thread, other threads:[~2026-02-16 18:58 UTC | newest]
Thread overview: 6+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2026-02-11 5:35 [PATCH] cpuidle: ladder: Fix state index when only one idle state is registered Aboorva Devarajan
2026-02-11 15:00 ` Christian Loehle
2026-02-13 8:29 ` Aboorva Devarajan
2026-02-13 13:44 ` Christian Loehle
2026-02-16 11:05 ` Christian Loehle
2026-02-16 18:58 ` Aboorva Devarajan
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox