]> git.droids-corp.org - dpdk.git/commitdiff
power: add get/set min/max scaling frequencies API
authorKevin Laatz <kevin.laatz@intel.com>
Thu, 2 Jun 2022 15:13:38 +0000 (16:13 +0100)
committerThomas Monjalon <thomas@monjalon.net>
Sat, 4 Jun 2022 20:39:49 +0000 (22:39 +0200)
Add new get/set API to allow the user or application to set the minimum
and maximum frequencies to use when scaling.
Previously, the frequency range was determined by the HW capabilities of
the CPU. With this new API, the user or application can constrain this
if required.

Signed-off-by: Kevin Laatz <kevin.laatz@intel.com>
Acked-by: Ray Kinsella <mdr@ashroe.eu>
Acked-by: Anatoly Burakov <anatoly.burakov@intel.com>
Tested-by: David Hunt <david.hunt@intel.com>
doc/guides/prog_guide/power_man.rst
lib/power/power_pstate_cpufreq.c
lib/power/rte_power_pmd_mgmt.c
lib/power/rte_power_pmd_mgmt.h
lib/power/version.map

index 266c7544e5ddd50d07d6cd6ef3c035430cefe430..98cfd3c1f3efa40cde782ea6d26931499c5fd674 100644 (file)
@@ -264,6 +264,18 @@ API Overview for Ethernet PMD Power Management
 * **Set Pause Duration**: Set the duration of the pause (microseconds) used in
   the Pause mode callback.
 
+* **Get Scaling Min Freq**: Get the configured minimum frequency (kHz) to be used
+  in Frequency Scaling mode.
+
+* **Set Scaling Min Freq**: Set the minimum frequency (kHz) to be used in Frequency
+  Scaling mode.
+
+* **Get Scaling Max Freq**: Get the configured maximum frequency (kHz) to be used
+  in Frequency Scaling mode.
+
+* **Set Scaling Max Freq**: Set the maximum frequency (kHz) to be used in Frequency
+  Scaling mode.
+
 References
 ----------
 
index f4c36179ecd99b3d7eaa318b8e9d6aba33a0a9c4..78c9197695166bb118ff614f573fe2c270626928 100644 (file)
@@ -12,6 +12,7 @@
 
 #include <rte_memcpy.h>
 
+#include "rte_power_pmd_mgmt.h"
 #include "power_pstate_cpufreq.h"
 #include "power_common.h"
 
@@ -354,6 +355,7 @@ power_get_available_freqs(struct pstate_power_info *pi)
        FILE *f_min = NULL, *f_max = NULL;
        int ret = -1;
        uint32_t sys_min_freq = 0, sys_max_freq = 0, base_max_freq = 0;
+       int config_min_freq, config_max_freq;
        uint32_t i, num_freqs = 0;
 
        /* open all files */
@@ -388,6 +390,18 @@ power_get_available_freqs(struct pstate_power_info *pi)
                goto out;
        }
 
+       /* check for config set by user or application to limit frequency range */
+       config_min_freq = rte_power_pmd_mgmt_get_scaling_freq_min(pi->lcore_id);
+       if (config_min_freq < 0)
+               goto out;
+       config_max_freq = rte_power_pmd_mgmt_get_scaling_freq_max(pi->lcore_id);
+       if (config_max_freq < 0)
+               goto out;
+
+       sys_min_freq = RTE_MAX(sys_min_freq, (uint32_t)config_min_freq);
+       if (config_max_freq > 0) /* Only use config_max_freq if a value has been set */
+               sys_max_freq = RTE_MIN(sys_max_freq, (uint32_t)config_max_freq);
+
        if (sys_max_freq < sys_min_freq)
                goto out;
 
@@ -411,8 +425,8 @@ power_get_available_freqs(struct pstate_power_info *pi)
        /* If turbo is available then there is one extra freq bucket
         * to store the sys max freq which value is base_max +1
         */
-       num_freqs = (base_max_freq - sys_min_freq) / BUS_FREQ + 1 +
-               pi->turbo_available;
+       num_freqs = (RTE_MIN(base_max_freq, sys_max_freq) - sys_min_freq) / BUS_FREQ
+                       + 1 + pi->turbo_available;
        if (num_freqs >= RTE_MAX_LCORE_FREQS) {
                RTE_LOG(ERR, POWER, "Too many available frequencies: %d\n",
                                num_freqs);
@@ -427,10 +441,10 @@ power_get_available_freqs(struct pstate_power_info *pi)
         */
        for (i = 0, pi->nb_freqs = 0; i < num_freqs; i++) {
                if ((i == 0) && pi->turbo_available)
-                       pi->freqs[pi->nb_freqs++] = base_max_freq + 1;
+                       pi->freqs[pi->nb_freqs++] = RTE_MIN(base_max_freq, sys_max_freq) + 1;
                else
-                       pi->freqs[pi->nb_freqs++] =
-                       base_max_freq - (i - pi->turbo_available) * BUS_FREQ;
+                       pi->freqs[pi->nb_freqs++] = RTE_MIN(base_max_freq, sys_max_freq) -
+                                       (i - pi->turbo_available) * BUS_FREQ;
        }
 
        ret = 0;
index 1374cc213d2b309c82e05a830bff542b9e80d720..a66e08e574992a403427f47a98340cfc86f77cc5 100644 (file)
 #include <rte_power_intrinsics.h>
 
 #include "rte_power_pmd_mgmt.h"
+#include "power_common.h"
 
 unsigned int emptypoll_max;
 unsigned int pause_duration;
+unsigned int scale_freq_min[RTE_MAX_LCORE];
+unsigned int scale_freq_max[RTE_MAX_LCORE];
 
 /* store some internal state */
 static struct pmd_conf_data {
@@ -693,8 +696,77 @@ rte_power_pmd_mgmt_get_pause_duration(void)
        return pause_duration;
 }
 
+int
+rte_power_pmd_mgmt_set_scaling_freq_min(unsigned int lcore, unsigned int min)
+{
+       if (lcore >= RTE_MAX_LCORE) {
+               RTE_LOG(ERR, POWER, "Invalid lcore ID: %u\n", lcore);
+               return -EINVAL;
+       }
+
+       if (min > scale_freq_max[lcore]) {
+               RTE_LOG(ERR, POWER, "Invalid min frequency: Cannot be greater than max frequency");
+               return -EINVAL;
+       }
+       scale_freq_min[lcore] = min;
+
+       return 0;
+}
+
+int
+rte_power_pmd_mgmt_set_scaling_freq_max(unsigned int lcore, unsigned int max)
+{
+       if (lcore >= RTE_MAX_LCORE) {
+               RTE_LOG(ERR, POWER, "Invalid lcore ID: %u\n", lcore);
+               return -EINVAL;
+       }
+
+       /* Zero means 'not set'. Use UINT32_MAX to enable RTE_MIN/MAX macro use when scaling. */
+       if (max == 0)
+               max = UINT32_MAX;
+       if (max < scale_freq_min[lcore]) {
+               RTE_LOG(ERR, POWER, "Invalid max frequency: Cannot be less than min frequency");
+               return -EINVAL;
+       }
+
+       scale_freq_max[lcore] = max;
+
+       return 0;
+}
+
+int
+rte_power_pmd_mgmt_get_scaling_freq_min(unsigned int lcore)
+{
+       if (lcore >= RTE_MAX_LCORE) {
+               RTE_LOG(ERR, POWER, "Invalid lcore ID: %u\n", lcore);
+               return -EINVAL;
+       }
+
+       if (scale_freq_max[lcore] == 0)
+               RTE_LOG(DEBUG, POWER, "Scaling freq min config not set. Using sysfs min freq.\n");
+
+       return scale_freq_min[lcore];
+}
+
+int
+rte_power_pmd_mgmt_get_scaling_freq_max(unsigned int lcore)
+{
+       if (lcore >= RTE_MAX_LCORE) {
+               RTE_LOG(ERR, POWER, "Invalid lcore ID: %u\n", lcore);
+               return -EINVAL;
+       }
+
+       if (scale_freq_max[lcore] == UINT32_MAX) {
+               RTE_LOG(DEBUG, POWER, "Scaling freq max config not set. Using sysfs max freq.\n");
+               return 0;
+       }
+
+       return scale_freq_max[lcore];
+}
+
 RTE_INIT(rte_power_ethdev_pmgmt_init) {
        size_t i;
+       int j;
 
        /* initialize all tailqs */
        for (i = 0; i < RTE_DIM(lcore_cfgs); i++) {
@@ -705,4 +777,9 @@ RTE_INIT(rte_power_ethdev_pmgmt_init) {
        /* initialize config defaults */
        emptypoll_max = 512;
        pause_duration = 1;
+       /* scaling defaults out of range to ensure not used unless set by user or app */
+       for (j = 0; j < RTE_MAX_LCORE; j++) {
+               scale_freq_min[j] = 0;
+               scale_freq_max[j] = UINT32_MAX;
+       }
 }
index 18a9c3abb5371d9fbb5938f5852d6c50d2bd38f8..789fbe0e4df3fa6a104d6cc5b5825f166223ea60 100644 (file)
@@ -148,6 +148,87 @@ __rte_experimental
 unsigned int
 rte_power_pmd_mgmt_get_pause_duration(void);
 
+/**
+ * @warning
+ * @b EXPERIMENTAL: this API may change, or be removed, without prior notice.
+ *
+ * Set the min frequency to be used for frequency scaling or zero to use defaults.
+ *
+ * @note Supported by: Pstate mode.
+ *
+ * @param lcore
+ *   The ID of the lcore to set the min frequency for.
+ * @param min
+ *   The value, in KiloHertz, to set the minimum frequency to.
+ * @return
+ *   0 on success
+ *   <0 on error
+ */
+__rte_experimental
+int
+rte_power_pmd_mgmt_set_scaling_freq_min(unsigned int lcore, unsigned int min);
+
+/**
+ * @warning
+ * @b EXPERIMENTAL: this API may change, or be removed, without prior notice.
+ *
+ * Set the max frequency to be used for frequency scaling or zero to use defaults.
+ *
+ * @note Supported by: Pstate mode.
+ *
+ * @param lcore
+ *   The ID of the lcore to set the max frequency for.
+ * @param max
+ *   The value, in KiloHertz, to set the maximum frequency to.
+ *   If 'max' is 0, it is considered 'not set'.
+ * @return
+ *   0 on success
+ *   <0 on error
+ */
+__rte_experimental
+int
+rte_power_pmd_mgmt_set_scaling_freq_max(unsigned int lcore, unsigned int max);
+
+/**
+ * @warning
+ * @b EXPERIMENTAL: this API may change, or be removed, without prior notice.
+ *
+ * Get the current configured min frequency used for frequency scaling.
+ *
+ * @note Supported by: Pstate mode.
+ *
+ * @param lcore
+ *   The ID of the lcore to get the min frequency for.
+ * @return
+ *   0 if no value has been configured via the 'set' API.
+ *   >0 if a minimum frequency has been configured. Value is the minimum frequency
+ *   , in KiloHertz, used for frequency scaling.
+ *   <0 on error
+ */
+__rte_experimental
+int
+rte_power_pmd_mgmt_get_scaling_freq_min(unsigned int lcore);
+
+/**
+ * @warning
+ * @b EXPERIMENTAL: this API may change, or be removed, without prior notice.
+ *
+ * Get the current configured max frequency used for frequency scaling.
+ *
+ * @note Supported by: Pstate mode.
+ *
+ * @param lcore
+ *   The ID of the lcore to get the max frequency for.
+ * @return
+ *   0 if no value has been configured via the 'set' API.
+ *   >0 if a maximum frequency has been configured. Value is the maximum frequency
+ *   , in KiloHertz, used for frequency scaling.
+ *   <0 on error
+ */
+__rte_experimental
+int
+rte_power_pmd_mgmt_get_scaling_freq_max(unsigned int lcore);
+
 #ifdef __cplusplus
 }
 #endif
index 4673b719f9735461318c228637e2002a66e8a1ef..a687754f4aa4b595a338204198b1cd04dfee530e 100644 (file)
@@ -42,6 +42,10 @@ EXPERIMENTAL {
        # added in 22.07
        rte_power_pmd_mgmt_get_emptypoll_max;
        rte_power_pmd_mgmt_get_pause_duration;
+       rte_power_pmd_mgmt_get_scaling_freq_max;
+       rte_power_pmd_mgmt_get_scaling_freq_min;
        rte_power_pmd_mgmt_set_emptypoll_max;
        rte_power_pmd_mgmt_set_pause_duration;
+       rte_power_pmd_mgmt_set_scaling_freq_max;
+       rte_power_pmd_mgmt_set_scaling_freq_min;
 };