PM: EM: Fix inefficient states detection
authorVincent Donnefort <vincent.donnefort@arm.com>
Wed, 8 Sep 2021 14:05:22 +0000 (15:05 +0100)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Tue, 5 Oct 2021 14:33:05 +0000 (16:33 +0200)
Currently, a debug message is printed if an inefficient state is detected
in the Energy Model. Unfortunately, it won't detect if the first state is
inefficient or if two successive states are. Fix this behavior.

Fixes: 27871f7a8a34 (PM: Introduce an Energy Model management framework)
Signed-off-by: Vincent Donnefort <vincent.donnefort@arm.com>
Reviewed-by: Quentin Perret <qperret@google.com>
Reviewed-by: Lukasz Luba <lukasz.luba@arm.com>
Reviewed-by: Matthias Kaehlcke <mka@chromium.org>
Acked-by: Viresh Kumar <viresh.kumar@linaro.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
kernel/power/energy_model.c

index a332ccd829e24af2b5169b11d5a25112d9c55a58..97e62469a6b32fc2ceb14c71473270106eca5cf2 100644 (file)
@@ -107,8 +107,7 @@ static void em_debug_remove_pd(struct device *dev) {}
 static int em_create_perf_table(struct device *dev, struct em_perf_domain *pd,
                                int nr_states, struct em_data_callback *cb)
 {
-       unsigned long opp_eff, prev_opp_eff = ULONG_MAX;
-       unsigned long power, freq, prev_freq = 0;
+       unsigned long power, freq, prev_freq = 0, prev_cost = ULONG_MAX;
        struct em_perf_state *table;
        int i, ret;
        u64 fmax;
@@ -153,27 +152,21 @@ static int em_create_perf_table(struct device *dev, struct em_perf_domain *pd,
 
                table[i].power = power;
                table[i].frequency = prev_freq = freq;
-
-               /*
-                * The hertz/watts efficiency ratio should decrease as the
-                * frequency grows on sane platforms. But this isn't always
-                * true in practice so warn the user if a higher OPP is more
-                * power efficient than a lower one.
-                */
-               opp_eff = freq / power;
-               if (opp_eff >= prev_opp_eff)
-                       dev_dbg(dev, "EM: hertz/watts ratio non-monotonically decreasing: em_perf_state %d >= em_perf_state%d\n",
-                                       i, i - 1);
-               prev_opp_eff = opp_eff;
        }
 
        /* Compute the cost of each performance state. */
        fmax = (u64) table[nr_states - 1].frequency;
-       for (i = 0; i < nr_states; i++) {
+       for (i = nr_states - 1; i >= 0; i--) {
                unsigned long power_res = em_scale_power(table[i].power);
 
                table[i].cost = div64_u64(fmax * power_res,
                                          table[i].frequency);
+               if (table[i].cost >= prev_cost) {
+                       dev_dbg(dev, "EM: OPP:%lu is inefficient\n",
+                               table[i].frequency);
+               } else {
+                       prev_cost = table[i].cost;
+               }
        }
 
        pd->table = table;