Commit 8f4490e0 authored by Bjorn Andersson's avatar Bjorn Andersson Committed by Mark Brown

regulator: core: Introduce set_load op

Expose the requested load directly to the regulator implementation for
hardware that does not support the normal enum based set_mode().
Signed-off-by: default avatarBjorn Andersson <bjorn.andersson@sonymobile.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent c517d838
...@@ -648,10 +648,12 @@ static int drms_uA_update(struct regulator_dev *rdev) ...@@ -648,10 +648,12 @@ static int drms_uA_update(struct regulator_dev *rdev)
if (err < 0) if (err < 0)
return 0; return 0;
if (!rdev->desc->ops->get_optimum_mode) if (!rdev->desc->ops->get_optimum_mode &&
!rdev->desc->ops->set_load)
return 0; return 0;
if (!rdev->desc->ops->set_mode) if (!rdev->desc->ops->set_mode &&
!rdev->desc->ops->set_load)
return -EINVAL; return -EINVAL;
/* get output voltage */ /* get output voltage */
...@@ -676,22 +678,29 @@ static int drms_uA_update(struct regulator_dev *rdev) ...@@ -676,22 +678,29 @@ static int drms_uA_update(struct regulator_dev *rdev)
list_for_each_entry(sibling, &rdev->consumer_list, list) list_for_each_entry(sibling, &rdev->consumer_list, list)
current_uA += sibling->uA_load; current_uA += sibling->uA_load;
/* now get the optimum mode for our new total regulator load */ if (rdev->desc->ops->set_load) {
mode = rdev->desc->ops->get_optimum_mode(rdev, input_uV, /* set the optimum mode for our new total regulator load */
output_uV, current_uA); err = rdev->desc->ops->set_load(rdev, current_uA);
if (err < 0)
rdev_err(rdev, "failed to set load %d\n", current_uA);
} else {
/* now get the optimum mode for our new total regulator load */
mode = rdev->desc->ops->get_optimum_mode(rdev, input_uV,
output_uV, current_uA);
/* check the new mode is allowed */
err = regulator_mode_constrain(rdev, &mode);
if (err < 0) {
rdev_err(rdev, "failed to get optimum mode @ %d uA %d -> %d uV\n",
current_uA, input_uV, output_uV);
return err;
}
/* check the new mode is allowed */ err = rdev->desc->ops->set_mode(rdev, mode);
err = regulator_mode_constrain(rdev, &mode); if (err < 0)
if (err < 0) { rdev_err(rdev, "failed to set optimum mode %x\n", mode);
rdev_err(rdev, "failed to get optimum mode @ %d uA %d -> %d uV\n",
current_uA, input_uV, output_uV);
return err;
} }
err = rdev->desc->ops->set_mode(rdev, mode);
if (err < 0)
rdev_err(rdev, "failed to set optimum mode %x\n", mode);
return err; return err;
} }
......
...@@ -98,6 +98,7 @@ struct regulator_linear_range { ...@@ -98,6 +98,7 @@ struct regulator_linear_range {
* REGULATOR_STATUS value (or negative errno) * REGULATOR_STATUS value (or negative errno)
* @get_optimum_mode: Get the most efficient operating mode for the regulator * @get_optimum_mode: Get the most efficient operating mode for the regulator
* when running with the specified parameters. * when running with the specified parameters.
* @set_load: Set the load for the regulator.
* *
* @set_bypass: Set the regulator in bypass mode. * @set_bypass: Set the regulator in bypass mode.
* @get_bypass: Get the regulator bypass mode state. * @get_bypass: Get the regulator bypass mode state.
...@@ -167,6 +168,8 @@ struct regulator_ops { ...@@ -167,6 +168,8 @@ struct regulator_ops {
/* get most efficient regulator operating mode for load */ /* get most efficient regulator operating mode for load */
unsigned int (*get_optimum_mode) (struct regulator_dev *, int input_uV, unsigned int (*get_optimum_mode) (struct regulator_dev *, int input_uV,
int output_uV, int load_uA); int output_uV, int load_uA);
/* set the load on the regulator */
int (*set_load)(struct regulator_dev *, int load_uA);
/* control and report on bypass mode */ /* control and report on bypass mode */
int (*set_bypass)(struct regulator_dev *dev, bool enable); int (*set_bypass)(struct regulator_dev *dev, bool enable);
......
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