Commit e0e44513 authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

Merge branches 'powercap', 'pm-x86', 'pm-opp' and 'pm-misc'

Merge power capping code updates, x86-specific power management pdate,
operating performance points library updates and miscellaneous power
management updates for 6.2-rc1:

 - Fix compiler warnings with make W=1 in the idle_inject power capping
   driver (Srinivas Pandruvada).

 - Use kstrtobool() instead of strtobool() in the power capping sysfs
   interface (Christophe JAILLET).

 - Add SCMI Powercap based power capping driver (Cristian Marussi).

 - Add Emerald Rapids support to the intel-uncore-freq driver (Artem
   Bityutskiy).

 - Repair slips in kernel-doc comments in the generic notifier code
   (Lukas Bulwahn).

 - Fix several DT issues in the OPP library reorganize code around
   opp-microvolt-<named> DT property (Viresh Kumar).

 - Allow any of opp-microvolt, opp-microamp, or opp-microwatt properties
   to be present without the others present (James Calligeros).

 - Fix clock-latency-ns property in DT example (Serge Semin).

* powercap:
  powercap: idle_inject: Fix warnings with make W=1
  powercap: Use kstrtobool() instead of strtobool()
  powercap: arm_scmi: Add SCMI Powercap based driver

* pm-x86:
  platform/x86: intel-uncore-freq: add Emerald Rapids support

* pm-opp:
  dt-bindings: opp-v2: Fix clock-latency-ns prop in example
  OPP: decouple dt properties in opp_parse_supplies()
  OPP: Simplify opp_parse_supplies() by restructuring it
  OPP: Parse named opp-microwatt property too
  dt-bindings: opp: Fix named microwatt property
  dt-bindings: opp: Fix usage of current in microwatt property

* pm-misc:
  notifier: repair slips in kernel-doc comments
...@@ -108,7 +108,7 @@ patternProperties: ...@@ -108,7 +108,7 @@ patternProperties:
The power for the OPP in micro-Watts. The power for the OPP in micro-Watts.
Entries for multiple regulators shall be provided in the same field Entries for multiple regulators shall be provided in the same field
separated by angular brackets <>. If current values aren't required separated by angular brackets <>. If power values aren't required
for a regulator, then it shall be filled with 0. If power values for a regulator, then it shall be filled with 0. If power values
aren't required for any of the regulators, then this field is not aren't required for any of the regulators, then this field is not
required. The OPP binding doesn't provide any provisions to relate the required. The OPP binding doesn't provide any provisions to relate the
...@@ -230,9 +230,9 @@ patternProperties: ...@@ -230,9 +230,9 @@ patternProperties:
minItems: 1 minItems: 1
maxItems: 8 # Should be enough regulators maxItems: 8 # Should be enough regulators
'^opp-microwatt': '^opp-microwatt-':
description: description:
Named opp-microwatt property. Similar to opp-microamp property, Named opp-microwatt property. Similar to opp-microamp-<name> property,
but for microwatt instead. but for microwatt instead.
$ref: /schemas/types.yaml#/definitions/uint32-array $ref: /schemas/types.yaml#/definitions/uint32-array
minItems: 1 minItems: 1
......
...@@ -155,7 +155,7 @@ examples: ...@@ -155,7 +155,7 @@ examples:
opp-hz = /bits/ 64 <1200000000>; opp-hz = /bits/ 64 <1200000000>;
opp-microvolt = <1025000>; opp-microvolt = <1025000>;
opp-microamp = <90000>; opp-microamp = <90000>;
lock-latency-ns = <290000>; clock-latency-ns = <290000>;
turbo-mode; turbo-mode;
}; };
}; };
......
...@@ -20004,6 +20004,7 @@ F: drivers/clk/clk-sc[mp]i.c ...@@ -20004,6 +20004,7 @@ F: drivers/clk/clk-sc[mp]i.c
F: drivers/cpufreq/sc[mp]i-cpufreq.c F: drivers/cpufreq/sc[mp]i-cpufreq.c
F: drivers/firmware/arm_scmi/ F: drivers/firmware/arm_scmi/
F: drivers/firmware/arm_scpi.c F: drivers/firmware/arm_scpi.c
F: drivers/powercap/arm_scmi_powercap.c
F: drivers/regulator/scmi-regulator.c F: drivers/regulator/scmi-regulator.c
F: drivers/reset/reset-scmi.c F: drivers/reset/reset-scmi.c
F: include/linux/sc[mp]i_protocol.h F: include/linux/sc[mp]i_protocol.h
......
...@@ -578,169 +578,140 @@ static bool _opp_is_supported(struct device *dev, struct opp_table *opp_table, ...@@ -578,169 +578,140 @@ static bool _opp_is_supported(struct device *dev, struct opp_table *opp_table,
return false; return false;
} }
static int opp_parse_supplies(struct dev_pm_opp *opp, struct device *dev, static u32 *_parse_named_prop(struct dev_pm_opp *opp, struct device *dev,
struct opp_table *opp_table) struct opp_table *opp_table,
const char *prop_type, bool *triplet)
{ {
u32 *microvolt, *microamp = NULL, *microwatt = NULL;
int supplies = opp_table->regulator_count;
int vcount, icount, pcount, ret, i, j;
struct property *prop = NULL; struct property *prop = NULL;
char name[NAME_MAX]; char name[NAME_MAX];
int count, ret;
u32 *out;
/* Search for "opp-microvolt-<name>" */ /* Search for "opp-<prop_type>-<name>" */
if (opp_table->prop_name) { if (opp_table->prop_name) {
snprintf(name, sizeof(name), "opp-microvolt-%s", snprintf(name, sizeof(name), "opp-%s-%s", prop_type,
opp_table->prop_name); opp_table->prop_name);
prop = of_find_property(opp->np, name, NULL); prop = of_find_property(opp->np, name, NULL);
} }
if (!prop) { if (!prop) {
/* Search for "opp-microvolt" */ /* Search for "opp-<prop_type>" */
sprintf(name, "opp-microvolt"); snprintf(name, sizeof(name), "opp-%s", prop_type);
prop = of_find_property(opp->np, name, NULL); prop = of_find_property(opp->np, name, NULL);
if (!prop)
/* Missing property isn't a problem, but an invalid entry is */ return NULL;
if (!prop) {
if (unlikely(supplies == -1)) {
/* Initialize regulator_count */
opp_table->regulator_count = 0;
return 0;
}
if (!supplies)
return 0;
dev_err(dev, "%s: opp-microvolt missing although OPP managing regulators\n",
__func__);
return -EINVAL;
}
} }
if (unlikely(supplies == -1)) { count = of_property_count_u32_elems(opp->np, name);
/* Initialize regulator_count */ if (count < 0) {
supplies = opp_table->regulator_count = 1; dev_err(dev, "%s: Invalid %s property (%d)\n", __func__, name,
} else if (unlikely(!supplies)) { count);
dev_err(dev, "%s: opp-microvolt wasn't expected\n", __func__); return ERR_PTR(count);
return -EINVAL;
} }
vcount = of_property_count_u32_elems(opp->np, name); /*
if (vcount < 0) { * Initialize regulator_count, if regulator information isn't provided
dev_err(dev, "%s: Invalid %s property (%d)\n", * by the platform. Now that one of the properties is available, fix the
__func__, name, vcount); * regulator_count to 1.
return vcount; */
} if (unlikely(opp_table->regulator_count == -1))
opp_table->regulator_count = 1;
/* There can be one or three elements per supply */
if (vcount != supplies && vcount != supplies * 3) { if (count != opp_table->regulator_count &&
dev_err(dev, "%s: Invalid number of elements in %s property (%d) with supplies (%d)\n", (!triplet || count != opp_table->regulator_count * 3)) {
__func__, name, vcount, supplies); dev_err(dev, "%s: Invalid number of elements in %s property (%u) with supplies (%d)\n",
return -EINVAL; __func__, prop_type, count, opp_table->regulator_count);
return ERR_PTR(-EINVAL);
} }
microvolt = kmalloc_array(vcount, sizeof(*microvolt), GFP_KERNEL); out = kmalloc_array(count, sizeof(*out), GFP_KERNEL);
if (!microvolt) if (!out)
return -ENOMEM; return ERR_PTR(-EINVAL);
ret = of_property_read_u32_array(opp->np, name, microvolt, vcount); ret = of_property_read_u32_array(opp->np, name, out, count);
if (ret) { if (ret) {
dev_err(dev, "%s: error parsing %s: %d\n", __func__, name, ret); dev_err(dev, "%s: error parsing %s: %d\n", __func__, name, ret);
ret = -EINVAL; kfree(out);
goto free_microvolt; return ERR_PTR(-EINVAL);
}
/* Search for "opp-microamp-<name>" */
prop = NULL;
if (opp_table->prop_name) {
snprintf(name, sizeof(name), "opp-microamp-%s",
opp_table->prop_name);
prop = of_find_property(opp->np, name, NULL);
} }
if (!prop) { if (triplet)
/* Search for "opp-microamp" */ *triplet = count != opp_table->regulator_count;
sprintf(name, "opp-microamp");
prop = of_find_property(opp->np, name, NULL);
}
if (prop) { return out;
icount = of_property_count_u32_elems(opp->np, name); }
if (icount < 0) {
dev_err(dev, "%s: Invalid %s property (%d)\n", __func__,
name, icount);
ret = icount;
goto free_microvolt;
}
if (icount != supplies) { static u32 *opp_parse_microvolt(struct dev_pm_opp *opp, struct device *dev,
dev_err(dev, "%s: Invalid number of elements in %s property (%d) with supplies (%d)\n", struct opp_table *opp_table, bool *triplet)
__func__, name, icount, supplies); {
ret = -EINVAL; u32 *microvolt;
goto free_microvolt;
}
microamp = kmalloc_array(icount, sizeof(*microamp), GFP_KERNEL); microvolt = _parse_named_prop(opp, dev, opp_table, "microvolt", triplet);
if (!microamp) { if (IS_ERR(microvolt))
ret = -EINVAL; return microvolt;
goto free_microvolt;
}
ret = of_property_read_u32_array(opp->np, name, microamp, if (!microvolt) {
icount); /*
if (ret) { * Missing property isn't a problem, but an invalid
dev_err(dev, "%s: error parsing %s: %d\n", __func__, * entry is. This property isn't optional if regulator
name, ret); * information is provided. Check only for the first OPP, as
ret = -EINVAL; * regulator_count may get initialized after that to a valid
goto free_microamp; * value.
*/
if (list_empty(&opp_table->opp_list) &&
opp_table->regulator_count > 0) {
dev_err(dev, "%s: opp-microvolt missing although OPP managing regulators\n",
__func__);
return ERR_PTR(-EINVAL);
} }
} }
/* Search for "opp-microwatt" */ return microvolt;
sprintf(name, "opp-microwatt"); }
prop = of_find_property(opp->np, name, NULL);
if (prop) {
pcount = of_property_count_u32_elems(opp->np, name);
if (pcount < 0) {
dev_err(dev, "%s: Invalid %s property (%d)\n", __func__,
name, pcount);
ret = pcount;
goto free_microamp;
}
if (pcount != supplies) { static int opp_parse_supplies(struct dev_pm_opp *opp, struct device *dev,
dev_err(dev, "%s: Invalid number of elements in %s property (%d) with supplies (%d)\n", struct opp_table *opp_table)
__func__, name, pcount, supplies); {
ret = -EINVAL; u32 *microvolt, *microamp, *microwatt;
goto free_microamp; int ret = 0, i, j;
} bool triplet;
microwatt = kmalloc_array(pcount, sizeof(*microwatt), microvolt = opp_parse_microvolt(opp, dev, opp_table, &triplet);
GFP_KERNEL); if (IS_ERR(microvolt))
if (!microwatt) { return PTR_ERR(microvolt);
ret = -EINVAL;
goto free_microamp;
}
ret = of_property_read_u32_array(opp->np, name, microwatt, microamp = _parse_named_prop(opp, dev, opp_table, "microamp", NULL);
pcount); if (IS_ERR(microamp)) {
if (ret) { ret = PTR_ERR(microamp);
dev_err(dev, "%s: error parsing %s: %d\n", __func__, goto free_microvolt;
name, ret);
ret = -EINVAL;
goto free_microwatt;
}
} }
for (i = 0, j = 0; i < supplies; i++) { microwatt = _parse_named_prop(opp, dev, opp_table, "microwatt", NULL);
opp->supplies[i].u_volt = microvolt[j++]; if (IS_ERR(microwatt)) {
ret = PTR_ERR(microwatt);
goto free_microamp;
}
if (vcount == supplies) { /*
opp->supplies[i].u_volt_min = opp->supplies[i].u_volt; * Initialize regulator_count if it is uninitialized and no properties
opp->supplies[i].u_volt_max = opp->supplies[i].u_volt; * are found.
} else { */
opp->supplies[i].u_volt_min = microvolt[j++]; if (unlikely(opp_table->regulator_count == -1)) {
opp->supplies[i].u_volt_max = microvolt[j++]; opp_table->regulator_count = 0;
return 0;
}
for (i = 0, j = 0; i < opp_table->regulator_count; i++) {
if (microvolt) {
opp->supplies[i].u_volt = microvolt[j++];
if (triplet) {
opp->supplies[i].u_volt_min = microvolt[j++];
opp->supplies[i].u_volt_max = microvolt[j++];
} else {
opp->supplies[i].u_volt_min = opp->supplies[i].u_volt;
opp->supplies[i].u_volt_max = opp->supplies[i].u_volt;
}
} }
if (microamp) if (microamp)
...@@ -750,7 +721,6 @@ static int opp_parse_supplies(struct dev_pm_opp *opp, struct device *dev, ...@@ -750,7 +721,6 @@ static int opp_parse_supplies(struct dev_pm_opp *opp, struct device *dev,
opp->supplies[i].u_watt = microwatt[i]; opp->supplies[i].u_watt = microwatt[i];
} }
free_microwatt:
kfree(microwatt); kfree(microwatt);
free_microamp: free_microamp:
kfree(microamp); kfree(microamp);
......
...@@ -203,6 +203,7 @@ static const struct x86_cpu_id intel_uncore_cpu_ids[] = { ...@@ -203,6 +203,7 @@ static const struct x86_cpu_id intel_uncore_cpu_ids[] = {
X86_MATCH_INTEL_FAM6_MODEL(ICELAKE_X, NULL), X86_MATCH_INTEL_FAM6_MODEL(ICELAKE_X, NULL),
X86_MATCH_INTEL_FAM6_MODEL(ICELAKE_D, NULL), X86_MATCH_INTEL_FAM6_MODEL(ICELAKE_D, NULL),
X86_MATCH_INTEL_FAM6_MODEL(SAPPHIRERAPIDS_X, NULL), X86_MATCH_INTEL_FAM6_MODEL(SAPPHIRERAPIDS_X, NULL),
X86_MATCH_INTEL_FAM6_MODEL(EMERALDRAPIDS_X, NULL),
{} {}
}; };
MODULE_DEVICE_TABLE(x86cpu, intel_uncore_cpu_ids); MODULE_DEVICE_TABLE(x86cpu, intel_uncore_cpu_ids);
......
...@@ -44,6 +44,19 @@ config IDLE_INJECT ...@@ -44,6 +44,19 @@ config IDLE_INJECT
synchronously on a set of specified CPUs or alternatively synchronously on a set of specified CPUs or alternatively
on a per CPU basis. on a per CPU basis.
config ARM_SCMI_POWERCAP
tristate "ARM SCMI Powercap driver"
depends on ARM_SCMI_PROTOCOL
help
This enables support for the ARM Powercap based on ARM SCMI
Powercap protocol.
ARM SCMI Powercap protocol allows power limits to be enforced
and monitored against the SCMI Powercap domains advertised as
available by the SCMI platform firmware.
When compiled as module it will be called arm_scmi_powercap.ko.
config DTPM config DTPM
bool "Power capping for Dynamic Thermal Power Management (EXPERIMENTAL)" bool "Power capping for Dynamic Thermal Power Management (EXPERIMENTAL)"
depends on OF depends on OF
......
...@@ -6,3 +6,4 @@ obj-$(CONFIG_POWERCAP) += powercap_sys.o ...@@ -6,3 +6,4 @@ obj-$(CONFIG_POWERCAP) += powercap_sys.o
obj-$(CONFIG_INTEL_RAPL_CORE) += intel_rapl_common.o obj-$(CONFIG_INTEL_RAPL_CORE) += intel_rapl_common.o
obj-$(CONFIG_INTEL_RAPL) += intel_rapl_msr.o obj-$(CONFIG_INTEL_RAPL) += intel_rapl_msr.o
obj-$(CONFIG_IDLE_INJECT) += idle_inject.o obj-$(CONFIG_IDLE_INJECT) += idle_inject.o
obj-$(CONFIG_ARM_SCMI_POWERCAP) += arm_scmi_powercap.o
This diff is collapsed.
...@@ -147,6 +147,7 @@ static void idle_inject_fn(unsigned int cpu) ...@@ -147,6 +147,7 @@ static void idle_inject_fn(unsigned int cpu)
/** /**
* idle_inject_set_duration - idle and run duration update helper * idle_inject_set_duration - idle and run duration update helper
* @ii_dev: idle injection control device structure
* @run_duration_us: CPU run time to allow in microseconds * @run_duration_us: CPU run time to allow in microseconds
* @idle_duration_us: CPU idle time to inject in microseconds * @idle_duration_us: CPU idle time to inject in microseconds
*/ */
...@@ -162,6 +163,7 @@ void idle_inject_set_duration(struct idle_inject_device *ii_dev, ...@@ -162,6 +163,7 @@ void idle_inject_set_duration(struct idle_inject_device *ii_dev,
/** /**
* idle_inject_get_duration - idle and run duration retrieval helper * idle_inject_get_duration - idle and run duration retrieval helper
* @ii_dev: idle injection control device structure
* @run_duration_us: memory location to store the current CPU run time * @run_duration_us: memory location to store the current CPU run time
* @idle_duration_us: memory location to store the current CPU idle time * @idle_duration_us: memory location to store the current CPU idle time
*/ */
...@@ -175,6 +177,7 @@ void idle_inject_get_duration(struct idle_inject_device *ii_dev, ...@@ -175,6 +177,7 @@ void idle_inject_get_duration(struct idle_inject_device *ii_dev,
/** /**
* idle_inject_set_latency - set the maximum latency allowed * idle_inject_set_latency - set the maximum latency allowed
* @ii_dev: idle injection control device structure
* @latency_us: set the latency requirement for the idle state * @latency_us: set the latency requirement for the idle state
*/ */
void idle_inject_set_latency(struct idle_inject_device *ii_dev, void idle_inject_set_latency(struct idle_inject_device *ii_dev,
......
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/err.h> #include <linux/err.h>
#include <linux/kstrtox.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/powercap.h> #include <linux/powercap.h>
...@@ -446,7 +447,7 @@ static ssize_t enabled_store(struct device *dev, ...@@ -446,7 +447,7 @@ static ssize_t enabled_store(struct device *dev,
{ {
bool mode; bool mode;
if (strtobool(buf, &mode)) if (kstrtobool(buf, &mode))
return -EINVAL; return -EINVAL;
if (dev->parent) { if (dev->parent) {
struct powercap_zone *power_zone = to_powercap_zone(dev); struct powercap_zone *power_zone = to_powercap_zone(dev);
......
...@@ -62,7 +62,7 @@ static int notifier_chain_unregister(struct notifier_block **nl, ...@@ -62,7 +62,7 @@ static int notifier_chain_unregister(struct notifier_block **nl,
* value of this parameter is -1. * value of this parameter is -1.
* @nr_calls: Records the number of notifications sent. Don't care * @nr_calls: Records the number of notifications sent. Don't care
* value of this field is NULL. * value of this field is NULL.
* @returns: notifier_call_chain returns the value returned by the * Return: notifier_call_chain returns the value returned by the
* last notifier function called. * last notifier function called.
*/ */
static int notifier_call_chain(struct notifier_block **nl, static int notifier_call_chain(struct notifier_block **nl,
...@@ -105,13 +105,13 @@ NOKPROBE_SYMBOL(notifier_call_chain); ...@@ -105,13 +105,13 @@ NOKPROBE_SYMBOL(notifier_call_chain);
* @val_up: Value passed unmodified to the notifier function * @val_up: Value passed unmodified to the notifier function
* @val_down: Value passed unmodified to the notifier function when recovering * @val_down: Value passed unmodified to the notifier function when recovering
* from an error on @val_up * from an error on @val_up
* @v Pointer passed unmodified to the notifier function * @v: Pointer passed unmodified to the notifier function
* *
* NOTE: It is important the @nl chain doesn't change between the two * NOTE: It is important the @nl chain doesn't change between the two
* invocations of notifier_call_chain() such that we visit the * invocations of notifier_call_chain() such that we visit the
* exact same notifier callbacks; this rules out any RCU usage. * exact same notifier callbacks; this rules out any RCU usage.
* *
* Returns: the return value of the @val_up call. * Return: the return value of the @val_up call.
*/ */
static int notifier_call_chain_robust(struct notifier_block **nl, static int notifier_call_chain_robust(struct notifier_block **nl,
unsigned long val_up, unsigned long val_down, unsigned long val_up, unsigned long val_down,
......
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