Commit 332f0a01 authored by Gautham R. Shenoy's avatar Gautham R. Shenoy Committed by Rafael J. Wysocki

powernv-cpufreq: Fix pstate_to_idx() to handle non-continguous pstates

The code in powernv-cpufreq, makes the following two assumptions which
are not guaranteed by the device-tree bindings:

    1) Pstate ids are continguous: This is used in pstate_to_idx() to
       obtain the reverse map from a pstate to it's corresponding
       entry into the cpufreq frequency table.

    2) Every Pstate should always lie between the max and the min
       pstates that are explicitly reported in the device tree: This
       is used to determine whether a pstate reported by the PMSR is
       out of bounds.

Both these assumptions are unwarranted and can change on future
platforms.

In this patch, we maintain the reverse map from a pstate to it's index
in the cpufreq frequency table and use this in pstate_to_idx(). This
does away with the assumptions (1) mentioned above, and will work with
non continguous pstate ids. If no entry exists for a particular
pstate, then such a pstate is treated as being out of bounds. This
gets rid of assumption (2).

On all the existing platforms, where the pstates are 8-bit long
values, the new implementation of pstate_to_idx() takes constant time.
Signed-off-by: default avatarGautham R. Shenoy <ego@linux.vnet.ibm.com>
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
parent ee1f4a7d
...@@ -29,6 +29,7 @@ ...@@ -29,6 +29,7 @@
#include <linux/reboot.h> #include <linux/reboot.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/cpu.h> #include <linux/cpu.h>
#include <linux/hashtable.h>
#include <trace/events/power.h> #include <trace/events/power.h>
#include <asm/cputhreads.h> #include <asm/cputhreads.h>
...@@ -38,7 +39,8 @@ ...@@ -38,7 +39,8 @@
#include <asm/opal.h> #include <asm/opal.h>
#include <linux/timer.h> #include <linux/timer.h>
#define POWERNV_MAX_PSTATES 256 #define POWERNV_MAX_PSTATES_ORDER 8
#define POWERNV_MAX_PSTATES (1UL << (POWERNV_MAX_PSTATES_ORDER))
#define PMSR_PSAFE_ENABLE (1UL << 30) #define PMSR_PSAFE_ENABLE (1UL << 30)
#define PMSR_SPR_EM_DISABLE (1UL << 31) #define PMSR_SPR_EM_DISABLE (1UL << 31)
#define MAX_PSTATE_SHIFT 32 #define MAX_PSTATE_SHIFT 32
...@@ -92,6 +94,27 @@ struct global_pstate_info { ...@@ -92,6 +94,27 @@ struct global_pstate_info {
}; };
static struct cpufreq_frequency_table powernv_freqs[POWERNV_MAX_PSTATES+1]; static struct cpufreq_frequency_table powernv_freqs[POWERNV_MAX_PSTATES+1];
DEFINE_HASHTABLE(pstate_revmap, POWERNV_MAX_PSTATES_ORDER);
/**
* struct pstate_idx_revmap_data: Entry in the hashmap pstate_revmap
* indexed by a function of pstate id.
*
* @pstate_id: pstate id for this entry.
*
* @cpufreq_table_idx: Index into the powernv_freqs
* cpufreq_frequency_table for frequency
* corresponding to pstate_id.
*
* @hentry: hlist_node that hooks this entry into the pstate_revmap
* hashtable
*/
struct pstate_idx_revmap_data {
int pstate_id;
unsigned int cpufreq_table_idx;
struct hlist_node hentry;
};
u32 pstate_sign_prefix; u32 pstate_sign_prefix;
static bool rebooting, throttled, occ_reset; static bool rebooting, throttled, occ_reset;
...@@ -161,39 +184,47 @@ static inline int extract_pstate(u64 pmsr_val, unsigned int shift) ...@@ -161,39 +184,47 @@ static inline int extract_pstate(u64 pmsr_val, unsigned int shift)
#define extract_global_pstate(x) extract_pstate(x, GPSTATE_SHIFT) #define extract_global_pstate(x) extract_pstate(x, GPSTATE_SHIFT)
#define extract_max_pstate(x) extract_pstate(x, MAX_PSTATE_SHIFT) #define extract_max_pstate(x) extract_pstate(x, MAX_PSTATE_SHIFT)
/* Use following macros for conversions between pstate_id and index */ /* Use following functions for conversions between pstate_id and index */
/**
* idx_to_pstate : Returns the pstate id corresponding to the
* frequency in the cpufreq frequency table
* powernv_freqs indexed by @i.
*
* If @i is out of bound, this will return the pstate
* corresponding to the nominal frequency.
*/
static inline int idx_to_pstate(unsigned int i) static inline int idx_to_pstate(unsigned int i)
{ {
if (unlikely(i >= powernv_pstate_info.nr_pstates)) { if (unlikely(i >= powernv_pstate_info.nr_pstates)) {
pr_warn_once("index %u is out of bound\n", i); pr_warn_once("idx_to_pstate: index %u is out of bound\n", i);
return powernv_freqs[powernv_pstate_info.nominal].driver_data; return powernv_freqs[powernv_pstate_info.nominal].driver_data;
} }
return powernv_freqs[i].driver_data; return powernv_freqs[i].driver_data;
} }
static inline unsigned int pstate_to_idx(int pstate) /**
* pstate_to_idx : Returns the index in the cpufreq frequencytable
* powernv_freqs for the frequency whose corresponding
* pstate id is @pstate.
*
* If no frequency corresponding to @pstate is found,
* this will return the index of the nominal
* frequency.
*/
static unsigned int pstate_to_idx(int pstate)
{ {
int min = powernv_freqs[powernv_pstate_info.min].driver_data; unsigned int key = pstate % POWERNV_MAX_PSTATES;
int max = powernv_freqs[powernv_pstate_info.max].driver_data; struct pstate_idx_revmap_data *revmap_data;
if (min > 0) { hash_for_each_possible(pstate_revmap, revmap_data, hentry, key) {
if (unlikely((pstate < max) || (pstate > min))) { if (revmap_data->pstate_id == pstate)
pr_warn_once("pstate %d is out of bound\n", pstate); return revmap_data->cpufreq_table_idx;
return powernv_pstate_info.nominal;
}
} else {
if (unlikely((pstate > max) || (pstate < min))) {
pr_warn_once("pstate %d is out of bound\n", pstate);
return powernv_pstate_info.nominal;
}
} }
/*
* abs() is deliberately used so that is works with pr_warn_once("pstate_to_idx: pstate %d not found\n", pstate);
* both monotonically increasing and decreasing return powernv_pstate_info.nominal;
* pstate values
*/
return abs(pstate - idx_to_pstate(powernv_pstate_info.max));
} }
static inline void reset_gpstates(struct cpufreq_policy *policy) static inline void reset_gpstates(struct cpufreq_policy *policy)
...@@ -297,11 +328,21 @@ static int init_powernv_pstates(void) ...@@ -297,11 +328,21 @@ static int init_powernv_pstates(void)
for (i = 0; i < nr_pstates; i++) { for (i = 0; i < nr_pstates; i++) {
u32 id = be32_to_cpu(pstate_ids[i]); u32 id = be32_to_cpu(pstate_ids[i]);
u32 freq = be32_to_cpu(pstate_freqs[i]); u32 freq = be32_to_cpu(pstate_freqs[i]);
struct pstate_idx_revmap_data *revmap_data;
unsigned int key;
pr_debug("PState id %d freq %d MHz\n", id, freq); pr_debug("PState id %d freq %d MHz\n", id, freq);
powernv_freqs[i].frequency = freq * 1000; /* kHz */ powernv_freqs[i].frequency = freq * 1000; /* kHz */
powernv_freqs[i].driver_data = id; powernv_freqs[i].driver_data = id;
revmap_data = (struct pstate_idx_revmap_data *)
kmalloc(sizeof(*revmap_data), GFP_KERNEL);
revmap_data->pstate_id = id;
revmap_data->cpufreq_table_idx = i;
key = id % POWERNV_MAX_PSTATES;
hash_add(pstate_revmap, &revmap_data->hentry, key);
if (id == pstate_max) if (id == pstate_max)
powernv_pstate_info.max = i; powernv_pstate_info.max = i;
else if (id == pstate_nominal) else if (id == pstate_nominal)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment