Commit 68bf66a1 authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

Merge tag 'opp-updates-6.2' of git://git.kernel.org/pub/scm/linux/kernel/git/vireshk/pm

Pull OPP updates for 6.2 from Viresh Kumar:

"- Several DT fixes and code reorganization 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 prop in DT example (Serge Semin)."

* tag 'opp-updates-6.2' of git://git.kernel.org/pub/scm/linux/kernel/git/vireshk/pm:
  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
parents b7b275e6 dba79b78
...@@ -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;
}; };
}; };
......
...@@ -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 (count != opp_table->regulator_count &&
if (vcount != supplies && vcount != supplies * 3) { (!triplet || count != opp_table->regulator_count * 3)) {
dev_err(dev, "%s: Invalid number of elements in %s property (%d) with supplies (%d)\n", dev_err(dev, "%s: Invalid number of elements in %s property (%u) with supplies (%d)\n",
__func__, name, vcount, supplies); __func__, prop_type, count, opp_table->regulator_count);
return -EINVAL; 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>" */ if (triplet)
prop = NULL; *triplet = count != opp_table->regulator_count;
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) { return out;
/* Search for "opp-microamp" */ }
sprintf(name, "opp-microamp");
prop = of_find_property(opp->np, name, NULL);
}
if (prop) {
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) { static int opp_parse_supplies(struct dev_pm_opp *opp, struct device *dev,
pcount = of_property_count_u32_elems(opp->np, name); struct opp_table *opp_table)
if (pcount < 0) { {
dev_err(dev, "%s: Invalid %s property (%d)\n", __func__, u32 *microvolt, *microamp, *microwatt;
name, pcount); int ret = 0, i, j;
ret = pcount; bool triplet;
goto free_microamp;
}
if (pcount != supplies) { microvolt = opp_parse_microvolt(opp, dev, opp_table, &triplet);
dev_err(dev, "%s: Invalid number of elements in %s property (%d) with supplies (%d)\n", if (IS_ERR(microvolt))
__func__, name, pcount, supplies); return PTR_ERR(microvolt);
ret = -EINVAL;
goto free_microamp; microamp = _parse_named_prop(opp, dev, opp_table, "microamp", NULL);
if (IS_ERR(microamp)) {
ret = PTR_ERR(microamp);
goto free_microvolt;
} }
microwatt = kmalloc_array(pcount, sizeof(*microwatt), microwatt = _parse_named_prop(opp, dev, opp_table, "microwatt", NULL);
GFP_KERNEL); if (IS_ERR(microwatt)) {
if (!microwatt) { ret = PTR_ERR(microwatt);
ret = -EINVAL;
goto free_microamp; goto free_microamp;
} }
ret = of_property_read_u32_array(opp->np, name, microwatt, /*
pcount); * Initialize regulator_count if it is uninitialized and no properties
if (ret) { * are found.
dev_err(dev, "%s: error parsing %s: %d\n", __func__, */
name, ret); if (unlikely(opp_table->regulator_count == -1)) {
ret = -EINVAL; opp_table->regulator_count = 0;
goto free_microwatt; return 0;
}
} }
for (i = 0, j = 0; i < supplies; i++) { for (i = 0, j = 0; i < opp_table->regulator_count; i++) {
if (microvolt) {
opp->supplies[i].u_volt = microvolt[j++]; opp->supplies[i].u_volt = microvolt[j++];
if (vcount == supplies) { if (triplet) {
opp->supplies[i].u_volt_min = opp->supplies[i].u_volt;
opp->supplies[i].u_volt_max = opp->supplies[i].u_volt;
} else {
opp->supplies[i].u_volt_min = microvolt[j++]; opp->supplies[i].u_volt_min = microvolt[j++];
opp->supplies[i].u_volt_max = 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);
......
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