mirror of
https://github.com/FEX-Emu/linux.git
synced 2024-12-28 20:37:27 +00:00
Merge branch 'topic/suspend' of https://git.kernel.org/pub/scm/linux/kernel/git/broonie/regulator into regulator-core
This commit is contained in:
commit
285c22de37
@ -42,8 +42,16 @@ Optional properties:
|
|||||||
- regulator-state-[mem/disk] node has following common properties:
|
- regulator-state-[mem/disk] node has following common properties:
|
||||||
- regulator-on-in-suspend: regulator should be on in suspend state.
|
- regulator-on-in-suspend: regulator should be on in suspend state.
|
||||||
- regulator-off-in-suspend: regulator should be off in suspend state.
|
- regulator-off-in-suspend: regulator should be off in suspend state.
|
||||||
- regulator-suspend-microvolt: regulator should be set to this voltage
|
- regulator-suspend-min-microvolt: minimum voltage may be set in
|
||||||
in suspend.
|
suspend state.
|
||||||
|
- regulator-suspend-max-microvolt: maximum voltage may be set in
|
||||||
|
suspend state.
|
||||||
|
- regulator-suspend-microvolt: the default voltage which regulator
|
||||||
|
would be set in suspend. This property is now deprecated, instead
|
||||||
|
setting voltage for suspend mode via the API which regulator
|
||||||
|
driver provides is recommended.
|
||||||
|
- regulator-changeable-in-suspend: whether the default voltage and
|
||||||
|
the regulator on/off in suspend can be changed in runtime.
|
||||||
- regulator-mode: operating mode in the given suspend state.
|
- regulator-mode: operating mode in the given suspend state.
|
||||||
The set of possible operating modes depends on the capabilities of
|
The set of possible operating modes depends on the capabilities of
|
||||||
every hardware so the valid modes are documented on each regulator
|
every hardware so the valid modes are documented on each regulator
|
||||||
|
@ -229,26 +229,35 @@ static int regulator_check_voltage(struct regulator_dev *rdev,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* return 0 if the state is valid */
|
||||||
|
static int regulator_check_states(suspend_state_t state)
|
||||||
|
{
|
||||||
|
return (state > PM_SUSPEND_MAX || state == PM_SUSPEND_TO_IDLE);
|
||||||
|
}
|
||||||
|
|
||||||
/* Make sure we select a voltage that suits the needs of all
|
/* Make sure we select a voltage that suits the needs of all
|
||||||
* regulator consumers
|
* regulator consumers
|
||||||
*/
|
*/
|
||||||
static int regulator_check_consumers(struct regulator_dev *rdev,
|
static int regulator_check_consumers(struct regulator_dev *rdev,
|
||||||
int *min_uV, int *max_uV)
|
int *min_uV, int *max_uV,
|
||||||
|
suspend_state_t state)
|
||||||
{
|
{
|
||||||
struct regulator *regulator;
|
struct regulator *regulator;
|
||||||
|
struct regulator_voltage *voltage;
|
||||||
|
|
||||||
list_for_each_entry(regulator, &rdev->consumer_list, list) {
|
list_for_each_entry(regulator, &rdev->consumer_list, list) {
|
||||||
|
voltage = ®ulator->voltage[state];
|
||||||
/*
|
/*
|
||||||
* Assume consumers that didn't say anything are OK
|
* Assume consumers that didn't say anything are OK
|
||||||
* with anything in the constraint range.
|
* with anything in the constraint range.
|
||||||
*/
|
*/
|
||||||
if (!regulator->min_uV && !regulator->max_uV)
|
if (!voltage->min_uV && !voltage->max_uV)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (*max_uV > regulator->max_uV)
|
if (*max_uV > voltage->max_uV)
|
||||||
*max_uV = regulator->max_uV;
|
*max_uV = voltage->max_uV;
|
||||||
if (*min_uV < regulator->min_uV)
|
if (*min_uV < voltage->min_uV)
|
||||||
*min_uV = regulator->min_uV;
|
*min_uV = voltage->min_uV;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (*min_uV > *max_uV) {
|
if (*min_uV > *max_uV) {
|
||||||
@ -317,6 +326,24 @@ static int regulator_mode_constrain(struct regulator_dev *rdev,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline struct regulator_state *
|
||||||
|
regulator_get_suspend_state(struct regulator_dev *rdev, suspend_state_t state)
|
||||||
|
{
|
||||||
|
if (rdev->constraints == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
switch (state) {
|
||||||
|
case PM_SUSPEND_STANDBY:
|
||||||
|
return &rdev->constraints->state_standby;
|
||||||
|
case PM_SUSPEND_MEM:
|
||||||
|
return &rdev->constraints->state_mem;
|
||||||
|
case PM_SUSPEND_MAX:
|
||||||
|
return &rdev->constraints->state_disk;
|
||||||
|
default:
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static ssize_t regulator_uV_show(struct device *dev,
|
static ssize_t regulator_uV_show(struct device *dev,
|
||||||
struct device_attribute *attr, char *buf)
|
struct device_attribute *attr, char *buf)
|
||||||
{
|
{
|
||||||
@ -724,29 +751,32 @@ static int drms_uA_update(struct regulator_dev *rdev)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int suspend_set_state(struct regulator_dev *rdev,
|
static int suspend_set_state(struct regulator_dev *rdev,
|
||||||
struct regulator_state *rstate)
|
suspend_state_t state)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
struct regulator_state *rstate;
|
||||||
|
|
||||||
|
rstate = regulator_get_suspend_state(rdev, state);
|
||||||
|
if (rstate == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
/* If we have no suspend mode configration don't set anything;
|
/* If we have no suspend mode configration don't set anything;
|
||||||
* only warn if the driver implements set_suspend_voltage or
|
* only warn if the driver implements set_suspend_voltage or
|
||||||
* set_suspend_mode callback.
|
* set_suspend_mode callback.
|
||||||
*/
|
*/
|
||||||
if (!rstate->enabled && !rstate->disabled) {
|
if (rstate->enabled != ENABLE_IN_SUSPEND &&
|
||||||
|
rstate->enabled != DISABLE_IN_SUSPEND) {
|
||||||
if (rdev->desc->ops->set_suspend_voltage ||
|
if (rdev->desc->ops->set_suspend_voltage ||
|
||||||
rdev->desc->ops->set_suspend_mode)
|
rdev->desc->ops->set_suspend_mode)
|
||||||
rdev_warn(rdev, "No configuration\n");
|
rdev_warn(rdev, "No configuration\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (rstate->enabled && rstate->disabled) {
|
if (rstate->enabled == ENABLE_IN_SUSPEND &&
|
||||||
rdev_err(rdev, "invalid configuration\n");
|
rdev->desc->ops->set_suspend_enable)
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rstate->enabled && rdev->desc->ops->set_suspend_enable)
|
|
||||||
ret = rdev->desc->ops->set_suspend_enable(rdev);
|
ret = rdev->desc->ops->set_suspend_enable(rdev);
|
||||||
else if (rstate->disabled && rdev->desc->ops->set_suspend_disable)
|
else if (rstate->enabled == DISABLE_IN_SUSPEND &&
|
||||||
|
rdev->desc->ops->set_suspend_disable)
|
||||||
ret = rdev->desc->ops->set_suspend_disable(rdev);
|
ret = rdev->desc->ops->set_suspend_disable(rdev);
|
||||||
else /* OK if set_suspend_enable or set_suspend_disable is NULL */
|
else /* OK if set_suspend_enable or set_suspend_disable is NULL */
|
||||||
ret = 0;
|
ret = 0;
|
||||||
@ -771,30 +801,10 @@ static int suspend_set_state(struct regulator_dev *rdev,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* locks held by caller */
|
|
||||||
static int suspend_prepare(struct regulator_dev *rdev, suspend_state_t state)
|
|
||||||
{
|
|
||||||
if (!rdev->constraints)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
switch (state) {
|
|
||||||
case PM_SUSPEND_STANDBY:
|
|
||||||
return suspend_set_state(rdev,
|
|
||||||
&rdev->constraints->state_standby);
|
|
||||||
case PM_SUSPEND_MEM:
|
|
||||||
return suspend_set_state(rdev,
|
|
||||||
&rdev->constraints->state_mem);
|
|
||||||
case PM_SUSPEND_MAX:
|
|
||||||
return suspend_set_state(rdev,
|
|
||||||
&rdev->constraints->state_disk);
|
|
||||||
default:
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void print_constraints(struct regulator_dev *rdev)
|
static void print_constraints(struct regulator_dev *rdev)
|
||||||
{
|
{
|
||||||
struct regulation_constraints *constraints = rdev->constraints;
|
struct regulation_constraints *constraints = rdev->constraints;
|
||||||
@ -1061,7 +1071,7 @@ static int set_machine_constraints(struct regulator_dev *rdev,
|
|||||||
|
|
||||||
/* do we need to setup our suspend state */
|
/* do we need to setup our suspend state */
|
||||||
if (rdev->constraints->initial_state) {
|
if (rdev->constraints->initial_state) {
|
||||||
ret = suspend_prepare(rdev, rdev->constraints->initial_state);
|
ret = suspend_set_state(rdev, rdev->constraints->initial_state);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
rdev_err(rdev, "failed to set suspend state\n");
|
rdev_err(rdev, "failed to set suspend state\n");
|
||||||
return ret;
|
return ret;
|
||||||
@ -1349,9 +1359,9 @@ static struct regulator *create_regulator(struct regulator_dev *rdev,
|
|||||||
debugfs_create_u32("uA_load", 0444, regulator->debugfs,
|
debugfs_create_u32("uA_load", 0444, regulator->debugfs,
|
||||||
®ulator->uA_load);
|
®ulator->uA_load);
|
||||||
debugfs_create_u32("min_uV", 0444, regulator->debugfs,
|
debugfs_create_u32("min_uV", 0444, regulator->debugfs,
|
||||||
®ulator->min_uV);
|
®ulator->voltage[PM_SUSPEND_ON].min_uV);
|
||||||
debugfs_create_u32("max_uV", 0444, regulator->debugfs,
|
debugfs_create_u32("max_uV", 0444, regulator->debugfs,
|
||||||
®ulator->max_uV);
|
®ulator->voltage[PM_SUSPEND_ON].max_uV);
|
||||||
debugfs_create_file("constraint_flags", 0444,
|
debugfs_create_file("constraint_flags", 0444,
|
||||||
regulator->debugfs, regulator,
|
regulator->debugfs, regulator,
|
||||||
&constraint_flags_fops);
|
&constraint_flags_fops);
|
||||||
@ -2876,10 +2886,38 @@ out:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int _regulator_do_set_suspend_voltage(struct regulator_dev *rdev,
|
||||||
|
int min_uV, int max_uV, suspend_state_t state)
|
||||||
|
{
|
||||||
|
struct regulator_state *rstate;
|
||||||
|
int uV, sel;
|
||||||
|
|
||||||
|
rstate = regulator_get_suspend_state(rdev, state);
|
||||||
|
if (rstate == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (min_uV < rstate->min_uV)
|
||||||
|
min_uV = rstate->min_uV;
|
||||||
|
if (max_uV > rstate->max_uV)
|
||||||
|
max_uV = rstate->max_uV;
|
||||||
|
|
||||||
|
sel = regulator_map_voltage(rdev, min_uV, max_uV);
|
||||||
|
if (sel < 0)
|
||||||
|
return sel;
|
||||||
|
|
||||||
|
uV = rdev->desc->ops->list_voltage(rdev, sel);
|
||||||
|
if (uV >= min_uV && uV <= max_uV)
|
||||||
|
rstate->uV = uV;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
||||||
int min_uV, int max_uV)
|
int min_uV, int max_uV,
|
||||||
|
suspend_state_t state)
|
||||||
{
|
{
|
||||||
struct regulator_dev *rdev = regulator->rdev;
|
struct regulator_dev *rdev = regulator->rdev;
|
||||||
|
struct regulator_voltage *voltage = ®ulator->voltage[state];
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
int old_min_uV, old_max_uV;
|
int old_min_uV, old_max_uV;
|
||||||
int current_uV;
|
int current_uV;
|
||||||
@ -2890,7 +2928,7 @@ static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
|||||||
* should be a noop (some cpufreq implementations use the same
|
* should be a noop (some cpufreq implementations use the same
|
||||||
* voltage for multiple frequencies, for example).
|
* voltage for multiple frequencies, for example).
|
||||||
*/
|
*/
|
||||||
if (regulator->min_uV == min_uV && regulator->max_uV == max_uV)
|
if (voltage->min_uV == min_uV && voltage->max_uV == max_uV)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
/* If we're trying to set a range that overlaps the current voltage,
|
/* If we're trying to set a range that overlaps the current voltage,
|
||||||
@ -2900,8 +2938,8 @@ static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
|||||||
if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_VOLTAGE)) {
|
if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_VOLTAGE)) {
|
||||||
current_uV = _regulator_get_voltage(rdev);
|
current_uV = _regulator_get_voltage(rdev);
|
||||||
if (min_uV <= current_uV && current_uV <= max_uV) {
|
if (min_uV <= current_uV && current_uV <= max_uV) {
|
||||||
regulator->min_uV = min_uV;
|
voltage->min_uV = min_uV;
|
||||||
regulator->max_uV = max_uV;
|
voltage->max_uV = max_uV;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2919,12 +2957,12 @@ static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
|||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
/* restore original values in case of error */
|
/* restore original values in case of error */
|
||||||
old_min_uV = regulator->min_uV;
|
old_min_uV = voltage->min_uV;
|
||||||
old_max_uV = regulator->max_uV;
|
old_max_uV = voltage->max_uV;
|
||||||
regulator->min_uV = min_uV;
|
voltage->min_uV = min_uV;
|
||||||
regulator->max_uV = max_uV;
|
voltage->max_uV = max_uV;
|
||||||
|
|
||||||
ret = regulator_check_consumers(rdev, &min_uV, &max_uV);
|
ret = regulator_check_consumers(rdev, &min_uV, &max_uV, state);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out2;
|
goto out2;
|
||||||
|
|
||||||
@ -2961,7 +2999,7 @@ static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
|||||||
|
|
||||||
if (supply_change_uV > 0) {
|
if (supply_change_uV > 0) {
|
||||||
ret = regulator_set_voltage_unlocked(rdev->supply,
|
ret = regulator_set_voltage_unlocked(rdev->supply,
|
||||||
best_supply_uV, INT_MAX);
|
best_supply_uV, INT_MAX, state);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(&rdev->dev, "Failed to increase supply voltage: %d\n",
|
dev_err(&rdev->dev, "Failed to increase supply voltage: %d\n",
|
||||||
ret);
|
ret);
|
||||||
@ -2969,13 +3007,17 @@ static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = _regulator_do_set_voltage(rdev, min_uV, max_uV);
|
if (state == PM_SUSPEND_ON)
|
||||||
|
ret = _regulator_do_set_voltage(rdev, min_uV, max_uV);
|
||||||
|
else
|
||||||
|
ret = _regulator_do_set_suspend_voltage(rdev, min_uV,
|
||||||
|
max_uV, state);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out2;
|
goto out2;
|
||||||
|
|
||||||
if (supply_change_uV < 0) {
|
if (supply_change_uV < 0) {
|
||||||
ret = regulator_set_voltage_unlocked(rdev->supply,
|
ret = regulator_set_voltage_unlocked(rdev->supply,
|
||||||
best_supply_uV, INT_MAX);
|
best_supply_uV, INT_MAX, state);
|
||||||
if (ret)
|
if (ret)
|
||||||
dev_warn(&rdev->dev, "Failed to decrease supply voltage: %d\n",
|
dev_warn(&rdev->dev, "Failed to decrease supply voltage: %d\n",
|
||||||
ret);
|
ret);
|
||||||
@ -2986,8 +3028,8 @@ static int regulator_set_voltage_unlocked(struct regulator *regulator,
|
|||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
out2:
|
out2:
|
||||||
regulator->min_uV = old_min_uV;
|
voltage->min_uV = old_min_uV;
|
||||||
regulator->max_uV = old_max_uV;
|
voltage->max_uV = old_max_uV;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -3016,7 +3058,8 @@ int regulator_set_voltage(struct regulator *regulator, int min_uV, int max_uV)
|
|||||||
|
|
||||||
regulator_lock_supply(regulator->rdev);
|
regulator_lock_supply(regulator->rdev);
|
||||||
|
|
||||||
ret = regulator_set_voltage_unlocked(regulator, min_uV, max_uV);
|
ret = regulator_set_voltage_unlocked(regulator, min_uV, max_uV,
|
||||||
|
PM_SUSPEND_ON);
|
||||||
|
|
||||||
regulator_unlock_supply(regulator->rdev);
|
regulator_unlock_supply(regulator->rdev);
|
||||||
|
|
||||||
@ -3024,6 +3067,89 @@ int regulator_set_voltage(struct regulator *regulator, int min_uV, int max_uV)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(regulator_set_voltage);
|
EXPORT_SYMBOL_GPL(regulator_set_voltage);
|
||||||
|
|
||||||
|
static inline int regulator_suspend_toggle(struct regulator_dev *rdev,
|
||||||
|
suspend_state_t state, bool en)
|
||||||
|
{
|
||||||
|
struct regulator_state *rstate;
|
||||||
|
|
||||||
|
rstate = regulator_get_suspend_state(rdev, state);
|
||||||
|
if (rstate == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (!rstate->changeable)
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
rstate->enabled = en;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int regulator_suspend_enable(struct regulator_dev *rdev,
|
||||||
|
suspend_state_t state)
|
||||||
|
{
|
||||||
|
return regulator_suspend_toggle(rdev, state, true);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(regulator_suspend_enable);
|
||||||
|
|
||||||
|
int regulator_suspend_disable(struct regulator_dev *rdev,
|
||||||
|
suspend_state_t state)
|
||||||
|
{
|
||||||
|
struct regulator *regulator;
|
||||||
|
struct regulator_voltage *voltage;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* if any consumer wants this regulator device keeping on in
|
||||||
|
* suspend states, don't set it as disabled.
|
||||||
|
*/
|
||||||
|
list_for_each_entry(regulator, &rdev->consumer_list, list) {
|
||||||
|
voltage = ®ulator->voltage[state];
|
||||||
|
if (voltage->min_uV || voltage->max_uV)
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return regulator_suspend_toggle(rdev, state, false);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(regulator_suspend_disable);
|
||||||
|
|
||||||
|
static int _regulator_set_suspend_voltage(struct regulator *regulator,
|
||||||
|
int min_uV, int max_uV,
|
||||||
|
suspend_state_t state)
|
||||||
|
{
|
||||||
|
struct regulator_dev *rdev = regulator->rdev;
|
||||||
|
struct regulator_state *rstate;
|
||||||
|
|
||||||
|
rstate = regulator_get_suspend_state(rdev, state);
|
||||||
|
if (rstate == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (rstate->min_uV == rstate->max_uV) {
|
||||||
|
rdev_err(rdev, "The suspend voltage can't be changed!\n");
|
||||||
|
return -EPERM;
|
||||||
|
}
|
||||||
|
|
||||||
|
return regulator_set_voltage_unlocked(regulator, min_uV, max_uV, state);
|
||||||
|
}
|
||||||
|
|
||||||
|
int regulator_set_suspend_voltage(struct regulator *regulator, int min_uV,
|
||||||
|
int max_uV, suspend_state_t state)
|
||||||
|
{
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
/* PM_SUSPEND_ON is handled by regulator_set_voltage() */
|
||||||
|
if (regulator_check_states(state) || state == PM_SUSPEND_ON)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
regulator_lock_supply(regulator->rdev);
|
||||||
|
|
||||||
|
ret = _regulator_set_suspend_voltage(regulator, min_uV,
|
||||||
|
max_uV, state);
|
||||||
|
|
||||||
|
regulator_unlock_supply(regulator->rdev);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(regulator_set_suspend_voltage);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* regulator_set_voltage_time - get raise/fall time
|
* regulator_set_voltage_time - get raise/fall time
|
||||||
* @regulator: regulator source
|
* @regulator: regulator source
|
||||||
@ -3117,6 +3243,7 @@ EXPORT_SYMBOL_GPL(regulator_set_voltage_time_sel);
|
|||||||
int regulator_sync_voltage(struct regulator *regulator)
|
int regulator_sync_voltage(struct regulator *regulator)
|
||||||
{
|
{
|
||||||
struct regulator_dev *rdev = regulator->rdev;
|
struct regulator_dev *rdev = regulator->rdev;
|
||||||
|
struct regulator_voltage *voltage = ®ulator->voltage[PM_SUSPEND_ON];
|
||||||
int ret, min_uV, max_uV;
|
int ret, min_uV, max_uV;
|
||||||
|
|
||||||
mutex_lock(&rdev->mutex);
|
mutex_lock(&rdev->mutex);
|
||||||
@ -3128,20 +3255,20 @@ int regulator_sync_voltage(struct regulator *regulator)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* This is only going to work if we've had a voltage configured. */
|
/* This is only going to work if we've had a voltage configured. */
|
||||||
if (!regulator->min_uV && !regulator->max_uV) {
|
if (!voltage->min_uV && !voltage->max_uV) {
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
min_uV = regulator->min_uV;
|
min_uV = voltage->min_uV;
|
||||||
max_uV = regulator->max_uV;
|
max_uV = voltage->max_uV;
|
||||||
|
|
||||||
/* This should be a paranoia check... */
|
/* This should be a paranoia check... */
|
||||||
ret = regulator_check_voltage(rdev, &min_uV, &max_uV);
|
ret = regulator_check_voltage(rdev, &min_uV, &max_uV);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
ret = regulator_check_consumers(rdev, &min_uV, &max_uV);
|
ret = regulator_check_consumers(rdev, &min_uV, &max_uV, 0);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
@ -3897,12 +4024,6 @@ static void regulator_dev_release(struct device *dev)
|
|||||||
kfree(rdev);
|
kfree(rdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct class regulator_class = {
|
|
||||||
.name = "regulator",
|
|
||||||
.dev_release = regulator_dev_release,
|
|
||||||
.dev_groups = regulator_dev_groups,
|
|
||||||
};
|
|
||||||
|
|
||||||
static void rdev_init_debugfs(struct regulator_dev *rdev)
|
static void rdev_init_debugfs(struct regulator_dev *rdev)
|
||||||
{
|
{
|
||||||
struct device *parent = rdev->dev.parent;
|
struct device *parent = rdev->dev.parent;
|
||||||
@ -4153,81 +4274,86 @@ void regulator_unregister(struct regulator_dev *rdev)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(regulator_unregister);
|
EXPORT_SYMBOL_GPL(regulator_unregister);
|
||||||
|
|
||||||
static int _regulator_suspend_prepare(struct device *dev, void *data)
|
#ifdef CONFIG_SUSPEND
|
||||||
|
static int _regulator_suspend_late(struct device *dev, void *data)
|
||||||
{
|
{
|
||||||
struct regulator_dev *rdev = dev_to_rdev(dev);
|
struct regulator_dev *rdev = dev_to_rdev(dev);
|
||||||
const suspend_state_t *state = data;
|
suspend_state_t *state = data;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
mutex_lock(&rdev->mutex);
|
mutex_lock(&rdev->mutex);
|
||||||
ret = suspend_prepare(rdev, *state);
|
ret = suspend_set_state(rdev, *state);
|
||||||
mutex_unlock(&rdev->mutex);
|
mutex_unlock(&rdev->mutex);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* regulator_suspend_prepare - prepare regulators for system wide suspend
|
* regulator_suspend_late - prepare regulators for system wide suspend
|
||||||
* @state: system suspend state
|
* @state: system suspend state
|
||||||
*
|
*
|
||||||
* Configure each regulator with it's suspend operating parameters for state.
|
* Configure each regulator with it's suspend operating parameters for state.
|
||||||
* This will usually be called by machine suspend code prior to supending.
|
|
||||||
*/
|
*/
|
||||||
int regulator_suspend_prepare(suspend_state_t state)
|
static int regulator_suspend_late(struct device *dev)
|
||||||
{
|
{
|
||||||
/* ON is handled by regulator active state */
|
suspend_state_t state = pm_suspend_target_state;
|
||||||
if (state == PM_SUSPEND_ON)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
return class_for_each_device(®ulator_class, NULL, &state,
|
return class_for_each_device(®ulator_class, NULL, &state,
|
||||||
_regulator_suspend_prepare);
|
_regulator_suspend_late);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(regulator_suspend_prepare);
|
static int _regulator_resume_early(struct device *dev, void *data)
|
||||||
|
|
||||||
static int _regulator_suspend_finish(struct device *dev, void *data)
|
|
||||||
{
|
{
|
||||||
|
int ret = 0;
|
||||||
struct regulator_dev *rdev = dev_to_rdev(dev);
|
struct regulator_dev *rdev = dev_to_rdev(dev);
|
||||||
int ret;
|
suspend_state_t *state = data;
|
||||||
|
struct regulator_state *rstate;
|
||||||
|
|
||||||
|
rstate = regulator_get_suspend_state(rdev, *state);
|
||||||
|
if (rstate == NULL)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
mutex_lock(&rdev->mutex);
|
mutex_lock(&rdev->mutex);
|
||||||
if (rdev->use_count > 0 || rdev->constraints->always_on) {
|
|
||||||
if (!_regulator_is_enabled(rdev)) {
|
|
||||||
ret = _regulator_do_enable(rdev);
|
|
||||||
if (ret)
|
|
||||||
dev_err(dev,
|
|
||||||
"Failed to resume regulator %d\n",
|
|
||||||
ret);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (!have_full_constraints())
|
|
||||||
goto unlock;
|
|
||||||
if (!_regulator_is_enabled(rdev))
|
|
||||||
goto unlock;
|
|
||||||
|
|
||||||
ret = _regulator_do_disable(rdev);
|
if (rdev->desc->ops->resume_early &&
|
||||||
if (ret)
|
(rstate->enabled == ENABLE_IN_SUSPEND ||
|
||||||
dev_err(dev, "Failed to suspend regulator %d\n", ret);
|
rstate->enabled == DISABLE_IN_SUSPEND))
|
||||||
}
|
ret = rdev->desc->ops->resume_early(rdev);
|
||||||
unlock:
|
|
||||||
mutex_unlock(&rdev->mutex);
|
mutex_unlock(&rdev->mutex);
|
||||||
|
|
||||||
/* Keep processing regulators in spite of any errors */
|
return ret;
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
static int regulator_resume_early(struct device *dev)
|
||||||
* regulator_suspend_finish - resume regulators from system wide suspend
|
|
||||||
*
|
|
||||||
* Turn on regulators that might be turned off by regulator_suspend_prepare
|
|
||||||
* and that should be turned on according to the regulators properties.
|
|
||||||
*/
|
|
||||||
int regulator_suspend_finish(void)
|
|
||||||
{
|
{
|
||||||
return class_for_each_device(®ulator_class, NULL, NULL,
|
suspend_state_t state = pm_suspend_target_state;
|
||||||
_regulator_suspend_finish);
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(regulator_suspend_finish);
|
|
||||||
|
|
||||||
|
return class_for_each_device(®ulator_class, NULL, &state,
|
||||||
|
_regulator_resume_early);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* !CONFIG_SUSPEND */
|
||||||
|
|
||||||
|
#define regulator_suspend_late NULL
|
||||||
|
#define regulator_resume_early NULL
|
||||||
|
|
||||||
|
#endif /* !CONFIG_SUSPEND */
|
||||||
|
|
||||||
|
#ifdef CONFIG_PM
|
||||||
|
static const struct dev_pm_ops __maybe_unused regulator_pm_ops = {
|
||||||
|
.suspend_late = regulator_suspend_late,
|
||||||
|
.resume_early = regulator_resume_early,
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct class regulator_class = {
|
||||||
|
.name = "regulator",
|
||||||
|
.dev_release = regulator_dev_release,
|
||||||
|
.dev_groups = regulator_dev_groups,
|
||||||
|
#ifdef CONFIG_PM
|
||||||
|
.pm = ®ulator_pm_ops,
|
||||||
|
#endif
|
||||||
|
};
|
||||||
/**
|
/**
|
||||||
* regulator_has_full_constraints - the system has fully specified constraints
|
* regulator_has_full_constraints - the system has fully specified constraints
|
||||||
*
|
*
|
||||||
@ -4403,8 +4529,8 @@ static void regulator_summary_show_subtree(struct seq_file *s,
|
|||||||
switch (rdev->desc->type) {
|
switch (rdev->desc->type) {
|
||||||
case REGULATOR_VOLTAGE:
|
case REGULATOR_VOLTAGE:
|
||||||
seq_printf(s, "%37dmV %5dmV",
|
seq_printf(s, "%37dmV %5dmV",
|
||||||
consumer->min_uV / 1000,
|
consumer->voltage[PM_SUSPEND_ON].min_uV / 1000,
|
||||||
consumer->max_uV / 1000);
|
consumer->voltage[PM_SUSPEND_ON].max_uV / 1000);
|
||||||
break;
|
break;
|
||||||
case REGULATOR_CURRENT:
|
case REGULATOR_CURRENT:
|
||||||
break;
|
break;
|
||||||
|
@ -16,10 +16,25 @@
|
|||||||
#ifndef __REGULATOR_INTERNAL_H
|
#ifndef __REGULATOR_INTERNAL_H
|
||||||
#define __REGULATOR_INTERNAL_H
|
#define __REGULATOR_INTERNAL_H
|
||||||
|
|
||||||
|
#include <linux/suspend.h>
|
||||||
|
|
||||||
|
#define REGULATOR_STATES_NUM (PM_SUSPEND_MAX + 1)
|
||||||
|
|
||||||
|
struct regulator_voltage {
|
||||||
|
int min_uV;
|
||||||
|
int max_uV;
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* struct regulator
|
* struct regulator
|
||||||
*
|
*
|
||||||
* One for each consumer device.
|
* One for each consumer device.
|
||||||
|
* @voltage - a voltage array for each state of runtime, i.e.:
|
||||||
|
* PM_SUSPEND_ON
|
||||||
|
* PM_SUSPEND_TO_IDLE
|
||||||
|
* PM_SUSPEND_STANDBY
|
||||||
|
* PM_SUSPEND_MEM
|
||||||
|
* PM_SUSPEND_MAX
|
||||||
*/
|
*/
|
||||||
struct regulator {
|
struct regulator {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
@ -27,8 +42,7 @@ struct regulator {
|
|||||||
unsigned int always_on:1;
|
unsigned int always_on:1;
|
||||||
unsigned int bypass:1;
|
unsigned int bypass:1;
|
||||||
int uA_load;
|
int uA_load;
|
||||||
int min_uV;
|
struct regulator_voltage voltage[REGULATOR_STATES_NUM];
|
||||||
int max_uV;
|
|
||||||
const char *supply_name;
|
const char *supply_name;
|
||||||
struct device_attribute dev_attr;
|
struct device_attribute dev_attr;
|
||||||
struct regulator_dev *rdev;
|
struct regulator_dev *rdev;
|
||||||
|
@ -177,14 +177,30 @@ static void of_get_regulation_constraints(struct device_node *np,
|
|||||||
|
|
||||||
if (of_property_read_bool(suspend_np,
|
if (of_property_read_bool(suspend_np,
|
||||||
"regulator-on-in-suspend"))
|
"regulator-on-in-suspend"))
|
||||||
suspend_state->enabled = true;
|
suspend_state->enabled = ENABLE_IN_SUSPEND;
|
||||||
else if (of_property_read_bool(suspend_np,
|
else if (of_property_read_bool(suspend_np,
|
||||||
"regulator-off-in-suspend"))
|
"regulator-off-in-suspend"))
|
||||||
suspend_state->disabled = true;
|
suspend_state->enabled = DISABLE_IN_SUSPEND;
|
||||||
|
else
|
||||||
|
suspend_state->enabled = DO_NOTHING_IN_SUSPEND;
|
||||||
|
|
||||||
|
if (!of_property_read_u32(np, "regulator-suspend-min-microvolt",
|
||||||
|
&pval))
|
||||||
|
suspend_state->min_uV = pval;
|
||||||
|
|
||||||
|
if (!of_property_read_u32(np, "regulator-suspend-max-microvolt",
|
||||||
|
&pval))
|
||||||
|
suspend_state->max_uV = pval;
|
||||||
|
|
||||||
if (!of_property_read_u32(suspend_np,
|
if (!of_property_read_u32(suspend_np,
|
||||||
"regulator-suspend-microvolt", &pval))
|
"regulator-suspend-microvolt", &pval))
|
||||||
suspend_state->uV = pval;
|
suspend_state->uV = pval;
|
||||||
|
else /* otherwise use min_uV as default suspend voltage */
|
||||||
|
suspend_state->uV = suspend_state->min_uV;
|
||||||
|
|
||||||
|
if (of_property_read_bool(suspend_np,
|
||||||
|
"regulator-changeable-in-suspend"))
|
||||||
|
suspend_state->changeable = true;
|
||||||
|
|
||||||
if (i == PM_SUSPEND_MEM)
|
if (i == PM_SUSPEND_MEM)
|
||||||
constraints->initial_state = PM_SUSPEND_MEM;
|
constraints->initial_state = PM_SUSPEND_MEM;
|
||||||
|
@ -214,6 +214,8 @@ struct regulator_ops {
|
|||||||
/* set regulator suspend operating mode (defined in consumer.h) */
|
/* set regulator suspend operating mode (defined in consumer.h) */
|
||||||
int (*set_suspend_mode) (struct regulator_dev *, unsigned int mode);
|
int (*set_suspend_mode) (struct regulator_dev *, unsigned int mode);
|
||||||
|
|
||||||
|
int (*resume_early)(struct regulator_dev *rdev);
|
||||||
|
|
||||||
int (*set_pull_down) (struct regulator_dev *);
|
int (*set_pull_down) (struct regulator_dev *);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -42,6 +42,16 @@ struct regulator;
|
|||||||
#define REGULATOR_CHANGE_DRMS 0x10
|
#define REGULATOR_CHANGE_DRMS 0x10
|
||||||
#define REGULATOR_CHANGE_BYPASS 0x20
|
#define REGULATOR_CHANGE_BYPASS 0x20
|
||||||
|
|
||||||
|
/*
|
||||||
|
* operations in suspend mode
|
||||||
|
* DO_NOTHING_IN_SUSPEND - the default value
|
||||||
|
* DISABLE_IN_SUSPEND - turn off regulator in suspend states
|
||||||
|
* ENABLE_IN_SUSPEND - keep regulator on in suspend states
|
||||||
|
*/
|
||||||
|
#define DO_NOTHING_IN_SUSPEND (-1)
|
||||||
|
#define DISABLE_IN_SUSPEND 0
|
||||||
|
#define ENABLE_IN_SUSPEND 1
|
||||||
|
|
||||||
/* Regulator active discharge flags */
|
/* Regulator active discharge flags */
|
||||||
enum regulator_active_discharge {
|
enum regulator_active_discharge {
|
||||||
REGULATOR_ACTIVE_DISCHARGE_DEFAULT,
|
REGULATOR_ACTIVE_DISCHARGE_DEFAULT,
|
||||||
@ -56,16 +66,24 @@ enum regulator_active_discharge {
|
|||||||
* state. One of enabled or disabled must be set for the
|
* state. One of enabled or disabled must be set for the
|
||||||
* configuration to be applied.
|
* configuration to be applied.
|
||||||
*
|
*
|
||||||
* @uV: Operating voltage during suspend.
|
* @uV: Default operating voltage during suspend, it can be adjusted
|
||||||
|
* among <min_uV, max_uV>.
|
||||||
|
* @min_uV: Minimum suspend voltage may be set.
|
||||||
|
* @max_uV: Maximum suspend voltage may be set.
|
||||||
* @mode: Operating mode during suspend.
|
* @mode: Operating mode during suspend.
|
||||||
* @enabled: Enabled during suspend.
|
* @enabled: operations during suspend.
|
||||||
* @disabled: Disabled during suspend.
|
* - DO_NOTHING_IN_SUSPEND
|
||||||
|
* - DISABLE_IN_SUSPEND
|
||||||
|
* - ENABLE_IN_SUSPEND
|
||||||
|
* @changeable: Is this state can be switched between enabled/disabled,
|
||||||
*/
|
*/
|
||||||
struct regulator_state {
|
struct regulator_state {
|
||||||
int uV; /* suspend voltage */
|
int uV;
|
||||||
unsigned int mode; /* suspend regulator operating mode */
|
int min_uV;
|
||||||
int enabled; /* is regulator enabled in this suspend state */
|
int max_uV;
|
||||||
int disabled; /* is the regulator disabled in this suspend state */
|
unsigned int mode;
|
||||||
|
int enabled;
|
||||||
|
bool changeable;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -225,12 +243,12 @@ struct regulator_init_data {
|
|||||||
|
|
||||||
#ifdef CONFIG_REGULATOR
|
#ifdef CONFIG_REGULATOR
|
||||||
void regulator_has_full_constraints(void);
|
void regulator_has_full_constraints(void);
|
||||||
int regulator_suspend_prepare(suspend_state_t state);
|
|
||||||
int regulator_suspend_finish(void);
|
|
||||||
#else
|
#else
|
||||||
static inline void regulator_has_full_constraints(void)
|
static inline void regulator_has_full_constraints(void)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
static inline int regulator_suspend_prepare(suspend_state_t state)
|
static inline int regulator_suspend_prepare(suspend_state_t state)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
@ -239,6 +257,5 @@ static inline int regulator_suspend_finish(void)
|
|||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
Reference in New Issue
Block a user