staticbool regulator_ops_is_valid(struct regulator_dev *rdev, int ops)
{ if (!rdev->constraints) {
rdev_err(rdev, "no constraints\n"); returnfalse;
}
if (rdev->constraints->valid_ops_mask & ops) returntrue;
returnfalse;
}
/** * regulator_lock_nested - lock a single regulator * @rdev: regulator source * @ww_ctx: w/w mutex acquire context * * This function can be called many times by one task on * a single regulator and its mutex will be locked only * once. If a task, which is calling this function is other * than the one, which initially locked the mutex, it will * wait on mutex. * * Return: 0 on success or a negative error number on failure.
*/ staticinlineint regulator_lock_nested(struct regulator_dev *rdev, struct ww_acquire_ctx *ww_ctx)
{ bool lock = false; int ret = 0;
mutex_lock(®ulator_nesting_mutex);
if (!ww_mutex_trylock(&rdev->mutex, ww_ctx)) { if (rdev->mutex_owner == current)
rdev->ref_cnt++; else
lock = true;
if (lock) {
mutex_unlock(®ulator_nesting_mutex);
ret = ww_mutex_lock(&rdev->mutex, ww_ctx);
mutex_lock(®ulator_nesting_mutex);
}
} else {
lock = true;
}
if (lock && ret != -EDEADLK) {
rdev->ref_cnt++;
rdev->mutex_owner = current;
}
mutex_unlock(®ulator_nesting_mutex);
return ret;
}
/** * regulator_lock - lock a single regulator * @rdev: regulator source * * This function can be called many times by one task on * a single regulator and its mutex will be locked only * once. If a task, which is calling this function is other * than the one, which initially locked the mutex, it will * wait on mutex.
*/ staticvoid regulator_lock(struct regulator_dev *rdev)
{
regulator_lock_nested(rdev, NULL);
}
/** * regulator_unlock - unlock a single regulator * @rdev: regulator_source * * This function unlocks the mutex when the * reference counter reaches 0.
*/ staticvoid regulator_unlock(struct regulator_dev *rdev)
{
mutex_lock(®ulator_nesting_mutex);
if (--rdev->ref_cnt == 0) {
rdev->mutex_owner = NULL;
ww_mutex_unlock(&rdev->mutex);
}
WARN_ON_ONCE(rdev->ref_cnt < 0);
mutex_unlock(®ulator_nesting_mutex);
}
/** * regulator_lock_two - lock two regulators * @rdev1: first regulator * @rdev2: second regulator * @ww_ctx: w/w mutex acquire context * * Locks both rdevs using the regulator_ww_class.
*/ staticvoid regulator_lock_two(struct regulator_dev *rdev1, struct regulator_dev *rdev2, struct ww_acquire_ctx *ww_ctx)
{ struct regulator_dev *held, *contended; int ret;
ww_acquire_init(ww_ctx, ®ulator_ww_class);
/* Try to just grab both of them */
ret = regulator_lock_nested(rdev1, ww_ctx);
WARN_ON(ret);
ret = regulator_lock_nested(rdev2, ww_ctx); if (ret != -EDEADLOCK) {
WARN_ON(ret); gotoexit;
}
held = rdev1;
contended = rdev2; while (true) {
regulator_unlock(held);
if (c_rdev->supply && !regulator_supply_is_couple(c_rdev)) {
err = regulator_lock_recursive(c_rdev->supply->rdev,
new_contended_rdev,
old_contended_rdev,
ww_ctx); if (err) {
regulator_unlock(c_rdev); goto err_unlock;
}
}
}
return 0;
err_unlock:
regulator_unlock_recursive(rdev, i);
return err;
}
/** * regulator_unlock_dependent - unlock regulator's suppliers and coupled * regulators * @rdev: regulator source * @ww_ctx: w/w mutex acquire context * * Unlock all regulators related with rdev by coupling or supplying.
*/ staticvoid regulator_unlock_dependent(struct regulator_dev *rdev, struct ww_acquire_ctx *ww_ctx)
{
regulator_unlock_recursive(rdev, rdev->coupling_desc.n_coupled);
ww_acquire_fini(ww_ctx);
}
/** * regulator_lock_dependent - lock regulator's suppliers and coupled regulators * @rdev: regulator source * @ww_ctx: w/w mutex acquire context * * This function as a wrapper on regulator_lock_recursive(), which locks * all regulators related with rdev by coupling or supplying.
*/ staticvoid regulator_lock_dependent(struct regulator_dev *rdev, struct ww_acquire_ctx *ww_ctx)
{ struct regulator_dev *new_contended_rdev = NULL; struct regulator_dev *old_contended_rdev = NULL; int err;
mutex_lock(®ulator_list_mutex);
ww_acquire_init(ww_ctx, ®ulator_ww_class);
do { if (new_contended_rdev) {
ww_mutex_lock_slow(&new_contended_rdev->mutex, ww_ctx);
old_contended_rdev = new_contended_rdev;
old_contended_rdev->ref_cnt++;
old_contended_rdev->mutex_owner = current;
}
if (old_contended_rdev)
regulator_unlock(old_contended_rdev);
} while (err == -EDEADLK);
ww_acquire_done(ww_ctx);
mutex_unlock(®ulator_list_mutex);
}
/* Platform voltage constraint check */ int regulator_check_voltage(struct regulator_dev *rdev, int *min_uV, int *max_uV)
{
BUG_ON(*min_uV > *max_uV);
if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_VOLTAGE)) {
rdev_err(rdev, "voltage operation not allowed\n"); return -EPERM;
}
if (*max_uV > rdev->constraints->max_uV)
*max_uV = rdev->constraints->max_uV; if (*min_uV < rdev->constraints->min_uV)
*min_uV = rdev->constraints->min_uV;
if (*min_uV > *max_uV) {
rdev_err(rdev, "unsupportable voltage range: %d-%duV\n",
*min_uV, *max_uV); return -EINVAL;
}
return 0;
}
/* return 0 if the state is valid */ staticint 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 * regulator consumers
*/ int regulator_check_consumers(struct regulator_dev *rdev, int *min_uV, int *max_uV,
suspend_state_t state)
{ struct regulator *regulator; struct regulator_voltage *voltage;
list_for_each_entry(regulator, &rdev->consumer_list, list) {
voltage = ®ulator->voltage[state]; /* * Assume consumers that didn't say anything are OK * with anything in the constraint range.
*/ if (!voltage->min_uV && !voltage->max_uV) continue;
if (*max_uV > voltage->max_uV)
*max_uV = voltage->max_uV; if (*min_uV < voltage->min_uV)
*min_uV = voltage->min_uV;
}
/* current constraint check */ staticint regulator_check_current_limit(struct regulator_dev *rdev, int *min_uA, int *max_uA)
{
BUG_ON(*min_uA > *max_uA);
if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_CURRENT)) {
rdev_err(rdev, "current operation not allowed\n"); return -EPERM;
}
if (*max_uA > rdev->constraints->max_uA &&
rdev->constraints->max_uA)
*max_uA = rdev->constraints->max_uA; if (*min_uA < rdev->constraints->min_uA)
*min_uA = rdev->constraints->min_uA;
if (*min_uA > *max_uA) {
rdev_err(rdev, "unsupportable current range: %d-%duA\n",
*min_uA, *max_uA); return -EINVAL;
}
return 0;
}
/* operating mode constraint check */ staticint regulator_mode_constrain(struct regulator_dev *rdev, unsignedint *mode)
{ switch (*mode) { case REGULATOR_MODE_FAST: case REGULATOR_MODE_NORMAL: case REGULATOR_MODE_IDLE: case REGULATOR_MODE_STANDBY: break; default:
rdev_err(rdev, "invalid mode %x specified\n", *mode); return -EINVAL;
}
if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_MODE)) {
rdev_err(rdev, "mode operation not allowed\n"); return -EPERM;
}
/* The modes are bitmasks, the most power hungry modes having * the lowest values. If the requested mode isn't supported * try higher modes.
*/ while (*mode) { if (rdev->constraints->valid_modes_mask & *mode) return 0;
*mode /= 2;
}
rstate = regulator_get_suspend_state(rdev, state); if (rstate == NULL) return NULL;
/* If we have no suspend mode configuration don't set anything; * only warn if the driver implements set_suspend_voltage or * set_suspend_mode callback.
*/ if (rstate->enabled != ENABLE_IN_SUSPEND &&
rstate->enabled != DISABLE_IN_SUSPEND) { if (rdev->desc->ops->set_suspend_voltage ||
rdev->desc->ops->set_suspend_mode)
rdev_warn(rdev, "No configuration\n"); return NULL;
}
/* Calculate the new optimum regulator operating mode based on the new total * consumer load. All locks held by caller
*/ staticint drms_uA_update(struct regulator_dev *rdev)
{ struct regulator *sibling; int current_uA = 0, output_uV, input_uV, err; unsignedint mode;
/* * first check to see if we can set modes at all, otherwise just * tell the consumer everything is OK.
*/ if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_DRMS)) {
rdev_dbg(rdev, "DRMS operation not allowed\n"); return 0;
}
if (!rdev->desc->ops->get_optimum_mode &&
!rdev->desc->ops->set_load) return 0;
if (!rdev->desc->ops->set_mode &&
!rdev->desc->ops->set_load) return -EINVAL;
/* calc total requested load */
list_for_each_entry(sibling, &rdev->consumer_list, list) { if (sibling->enable_count)
current_uA += sibling->uA_load;
}
current_uA += rdev->constraints->system_load;
if (rdev->desc->ops->set_load) { /* set the optimum mode for our new total regulator load */
err = rdev->desc->ops->set_load(rdev, current_uA); if (err < 0)
rdev_err(rdev, "failed to set load %d: %pe\n",
current_uA, ERR_PTR(err));
} else { /* * Unfortunately in some cases the constraints->valid_ops has * REGULATOR_CHANGE_DRMS but there are no valid modes listed. * That's not really legit but we won't consider it a fatal * error here. We'll treat it as if REGULATOR_CHANGE_DRMS * wasn't set.
*/ if (!rdev->constraints->valid_modes_mask) {
rdev_dbg(rdev, "Can change modes; but no valid mode\n"); return 0;
}
/* get output voltage */
output_uV = regulator_get_voltage_rdev(rdev);
/* * Don't return an error; if regulator driver cares about * output_uV then it's up to the driver to validate.
*/ if (output_uV <= 0)
rdev_dbg(rdev, "invalid output voltage found\n");
/* get input voltage */
input_uV = 0; if (rdev->supply)
input_uV = regulator_get_voltage_rdev(rdev->supply->rdev); if (input_uV <= 0)
input_uV = rdev->constraints->input_uV;
/* * Don't return an error; if regulator driver cares about * input_uV then it's up to the driver to validate.
*/ if (input_uV <= 0)
rdev_dbg(rdev, "invalid input voltage found\n");
/* 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: %pe\n",
current_uA, input_uV, output_uV, ERR_PTR(err)); return err;
}
err = rdev->desc->ops->set_mode(rdev, mode); if (err < 0)
rdev_err(rdev, "failed to set optimum mode %x: %pe\n",
mode, ERR_PTR(err));
}
return err;
}
staticint __suspend_set_state(struct regulator_dev *rdev, conststruct regulator_state *rstate)
{ int ret = 0;
if (rstate->enabled == ENABLE_IN_SUSPEND &&
rdev->desc->ops->set_suspend_enable)
ret = rdev->desc->ops->set_suspend_enable(rdev); elseif (rstate->enabled == DISABLE_IN_SUSPEND &&
rdev->desc->ops->set_suspend_disable)
ret = rdev->desc->ops->set_suspend_disable(rdev); else/* OK if set_suspend_enable or set_suspend_disable is NULL */
ret = 0;
if (ret < 0) {
rdev_err(rdev, "failed to enabled/disable: %pe\n", ERR_PTR(ret)); return ret;
}
if (rdev->desc->ops->set_suspend_voltage && rstate->uV > 0) {
ret = rdev->desc->ops->set_suspend_voltage(rdev, rstate->uV); if (ret < 0) {
rdev_err(rdev, "failed to set voltage: %pe\n", ERR_PTR(ret)); return ret;
}
}
if (rdev->desc->ops->set_suspend_mode && rstate->mode > 0) {
ret = rdev->desc->ops->set_suspend_mode(rdev, rstate->mode); if (ret < 0) {
rdev_err(rdev, "failed to set mode: %pe\n", ERR_PTR(ret)); return ret;
}
}
if (!constraints->min_uV ||
constraints->min_uV != constraints->max_uV) {
ret = regulator_get_voltage_rdev(rdev); if (ret > 0)
count += scnprintf(buf + count, len - count, "at %d mV ", ret / 1000);
}
if (constraints->uV_offset)
count += scnprintf(buf + count, len - count, "%dmV offset ",
constraints->uV_offset / 1000);
if (constraints->min_uA && constraints->max_uA) { if (constraints->min_uA == constraints->max_uA)
count += scnprintf(buf + count, len - count, "%d mA ",
constraints->min_uA / 1000); else
count += scnprintf(buf + count, len - count, "%d <--> %d mA ",
constraints->min_uA / 1000,
constraints->max_uA / 1000);
}
if (!constraints->min_uA ||
constraints->min_uA != constraints->max_uA) {
ret = _regulator_get_current_limit(rdev); if (ret > 0)
count += scnprintf(buf + count, len - count, "at %d mA ", ret / 1000);
}
if (constraints->valid_modes_mask & REGULATOR_MODE_FAST)
count += scnprintf(buf + count, len - count, "fast "); if (constraints->valid_modes_mask & REGULATOR_MODE_NORMAL)
count += scnprintf(buf + count, len - count, "normal "); if (constraints->valid_modes_mask & REGULATOR_MODE_IDLE)
count += scnprintf(buf + count, len - count, "idle "); if (constraints->valid_modes_mask & REGULATOR_MODE_STANDBY)
count += scnprintf(buf + count, len - count, "standby ");
if (constraints->pw_budget_mW)
count += scnprintf(buf + count, len - count, "%d mW budget",
constraints->pw_budget_mW);
if (!count)
count = scnprintf(buf, len, "no parameters"); else
--count;
if ((constraints->min_uV != constraints->max_uV) &&
!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_VOLTAGE))
rdev_warn(rdev, "Voltage range but no REGULATOR_CHANGE_VOLTAGE\n");
}
/* do we need to apply the constraint voltage */ if (rdev->constraints->apply_uV &&
rdev->constraints->min_uV && rdev->constraints->max_uV) { int target_min, target_max; int current_uV = regulator_get_voltage_rdev(rdev);
if (current_uV == -ENOTRECOVERABLE) { /* This regulator can't be read and must be initialized */
rdev_info(rdev, "Setting %d-%duV\n",
rdev->constraints->min_uV,
rdev->constraints->max_uV);
_regulator_do_set_voltage(rdev,
rdev->constraints->min_uV,
rdev->constraints->max_uV);
current_uV = regulator_get_voltage_rdev(rdev);
}
if (current_uV < 0) { if (current_uV != -EPROBE_DEFER)
rdev_err(rdev, "failed to get the current voltage: %pe\n",
ERR_PTR(current_uV)); return current_uV;
}
/* * If we're below the minimum voltage move up to the * minimum voltage, if we're above the maximum voltage * then move down to the maximum.
*/
target_min = current_uV;
target_max = current_uV;
if (target_min != current_uV || target_max != current_uV) {
rdev_info(rdev, "Bringing %duV into %d-%duV\n",
current_uV, target_min, target_max);
ret = _regulator_do_set_voltage(
rdev, target_min, target_max); if (ret < 0) {
rdev_err(rdev, "failed to apply %d-%duV constraint: %pe\n",
target_min, target_max, ERR_PTR(ret)); return ret;
}
}
}
/* constrain machine-level voltage specs to fit * the actual range supported by this regulator.
*/ if (ops->list_voltage && rdev->desc->n_voltages) { int count = rdev->desc->n_voltages; int i; int min_uV = INT_MAX; int max_uV = INT_MIN; int cmin = constraints->min_uV; int cmax = constraints->max_uV;
/* it's safe to autoconfigure fixed-voltage supplies * and the constraints are used by list_voltage.
*/ if (count == 1 && !cmin) {
cmin = 1;
cmax = INT_MAX;
constraints->min_uV = cmin;
constraints->max_uV = cmax;
}
/* voltage constraints are optional */ if ((cmin == 0) && (cmax == 0)) return 0;
if (!constraints->min_uA && !constraints->max_uA) return 0;
if (constraints->min_uA > constraints->max_uA) {
rdev_err(rdev, "Invalid current constraints\n"); return -EINVAL;
}
if (!ops->set_current_limit || !ops->get_current_limit) {
rdev_warn(rdev, "Operation of current configuration missing\n"); return 0;
}
/* Set regulator current in constraints range */
ret = ops->set_current_limit(rdev, constraints->min_uA,
constraints->max_uA); if (ret < 0) {
rdev_err(rdev, "Failed to set current constraint, %d\n", ret); return ret;
}
if (limit == REGULATOR_NOTIF_LIMIT_ENABLE)
limit = 0;
return set(rdev, limit, severity, enable);
}
staticint handle_notify_limits(struct regulator_dev *rdev, int (*set)(struct regulator_dev *, int, int, bool), struct notification_limit *limits)
{ int ret = 0;
if (!set) return -EOPNOTSUPP;
if (limits->prot)
ret = notif_set_limit(rdev, set, limits->prot,
REGULATOR_SEVERITY_PROT); if (ret) return ret;
if (limits->err)
ret = notif_set_limit(rdev, set, limits->err,
REGULATOR_SEVERITY_ERR); if (ret) return ret;
if (limits->warn)
ret = notif_set_limit(rdev, set, limits->warn,
REGULATOR_SEVERITY_WARN);
return ret;
} /** * set_machine_constraints - sets regulator constraints * @rdev: regulator source * * Allows platform initialisation code to define and constrain * regulator circuits e.g. valid voltage/current ranges, etc. NOTE: * Constraints *must* be set by platform code in order for some * regulator operations to proceed i.e. set_voltage, set_current_limit, * set_mode. * * Return: 0 on success or a negative error number on failure.
*/ staticint set_machine_constraints(struct regulator_dev *rdev)
{ int ret = 0; conststruct regulator_ops *ops = rdev->desc->ops;
ret = machine_constraints_voltage(rdev, rdev->constraints); if (ret != 0) return ret;
ret = machine_constraints_current(rdev, rdev->constraints); if (ret != 0) return ret;
if (rdev->constraints->ilim_uA && ops->set_input_current_limit) {
ret = ops->set_input_current_limit(rdev,
rdev->constraints->ilim_uA); if (ret < 0) {
rdev_err(rdev, "failed to set input limit: %pe\n", ERR_PTR(ret)); return ret;
}
}
/* do we need to setup our suspend state */ if (rdev->constraints->initial_state) {
ret = suspend_set_initial_state(rdev); if (ret < 0) {
rdev_err(rdev, "failed to set suspend state: %pe\n", ERR_PTR(ret)); return ret;
}
}
if (rdev->constraints->initial_mode) { if (!ops->set_mode) {
rdev_err(rdev, "no set_mode operation\n"); return -EINVAL;
}
ret = ops->set_mode(rdev, rdev->constraints->initial_mode); if (ret < 0) {
rdev_err(rdev, "failed to set initial mode: %pe\n", ERR_PTR(ret)); return ret;
}
} elseif (rdev->constraints->system_load) { /* * We'll only apply the initial system load if an * initial mode wasn't specified.
*/
drms_uA_update(rdev);
}
if ((rdev->constraints->ramp_delay || rdev->constraints->ramp_disable)
&& ops->set_ramp_delay) {
ret = ops->set_ramp_delay(rdev, rdev->constraints->ramp_delay); if (ret < 0) {
rdev_err(rdev, "failed to set ramp_delay: %pe\n", ERR_PTR(ret)); return ret;
}
}
if (rdev->constraints->pull_down && ops->set_pull_down) {
ret = ops->set_pull_down(rdev); if (ret < 0) {
rdev_err(rdev, "failed to set pull down: %pe\n", ERR_PTR(ret)); return ret;
}
}
if (rdev->constraints->soft_start && ops->set_soft_start) {
ret = ops->set_soft_start(rdev); if (ret < 0) {
rdev_err(rdev, "failed to set soft start: %pe\n", ERR_PTR(ret)); return ret;
}
}
/* * Existing logic does not warn if over_current_protection is given as * a constraint but driver does not support that. I think we should * warn about this type of issues as it is possible someone changes * PMIC on board to another type - and the another PMIC's driver does * not support setting protection. Board composer may happily believe * the DT limits are respected - especially if the new PMIC HW also * supports protection but the driver does not. I won't change the logic * without hearing more experienced opinion on this though. * * If warning is seen as a good idea then we can merge handling the * over-curret protection and detection and get rid of this special * handling.
*/ if (rdev->constraints->over_current_protection
&& ops->set_over_current_protection) { int lim = rdev->constraints->over_curr_limits.prot;
ret = ops->set_over_current_protection(rdev, lim,
REGULATOR_SEVERITY_PROT, true); if (ret < 0) {
rdev_err(rdev, "failed to set over current protection: %pe\n",
ERR_PTR(ret)); return ret;
}
}
if (rdev->constraints->over_current_detection)
ret = handle_notify_limits(rdev,
ops->set_over_current_protection,
&rdev->constraints->over_curr_limits); if (ret) { if (ret != -EOPNOTSUPP) {
rdev_err(rdev, "failed to set over current limits: %pe\n",
ERR_PTR(ret)); return ret;
}
rdev_warn(rdev, "IC does not support requested over-current limits\n");
}
if (rdev->constraints->over_voltage_detection)
ret = handle_notify_limits(rdev,
ops->set_over_voltage_protection,
&rdev->constraints->over_voltage_limits); if (ret) { if (ret != -EOPNOTSUPP) {
rdev_err(rdev, "failed to set over voltage limits %pe\n",
ERR_PTR(ret)); return ret;
}
rdev_warn(rdev, "IC does not support requested over voltage limits\n");
}
if (rdev->constraints->under_voltage_detection)
ret = handle_notify_limits(rdev,
ops->set_under_voltage_protection,
&rdev->constraints->under_voltage_limits); if (ret) { if (ret != -EOPNOTSUPP) {
rdev_err(rdev, "failed to set under voltage limits %pe\n",
ERR_PTR(ret)); return ret;
}
rdev_warn(rdev, "IC does not support requested under voltage limits\n");
}
if (rdev->constraints->over_temp_detection)
ret = handle_notify_limits(rdev,
ops->set_thermal_protection,
&rdev->constraints->temp_limits); if (ret) { if (ret != -EOPNOTSUPP) {
rdev_err(rdev, "failed to set temperature limits %pe\n",
ERR_PTR(ret)); return ret;
}
rdev_warn(rdev, "IC does not support requested temperature limits\n");
}
ret = ops->set_active_discharge(rdev, ad_state); if (ret < 0) {
rdev_err(rdev, "failed to set active discharge: %pe\n", ERR_PTR(ret)); return ret;
}
}
/* * If there is no mechanism for controlling the regulator then * flag it as always_on so we don't end up duplicating checks * for this so much. Note that we could control the state of * a supply to control the output on a regulator that has no * direct control.
*/ if (!rdev->ena_pin && !ops->enable) { if (rdev->supply_name && !rdev->supply) return -EPROBE_DEFER;
if (rdev->supply)
rdev->constraints->always_on =
rdev->supply->rdev->constraints->always_on; else
rdev->constraints->always_on = true;
}
/* If the constraints say the regulator should be on at this point * and we have control then make sure it is enabled.
*/ if (rdev->constraints->always_on || rdev->constraints->boot_on) { /* If we want to enable this regulator, make sure that we know * the supplying regulator.
*/ if (rdev->supply_name && !rdev->supply) return -EPROBE_DEFER;
/* If supplying regulator has already been enabled, * it's not intended to have use_count increment * when rdev is only boot-on.
*/ if (rdev->supply &&
(rdev->constraints->always_on ||
!regulator_is_enabled(rdev->supply))) {
ret = regulator_enable(rdev->supply); if (ret < 0) {
_regulator_put(rdev->supply);
rdev->supply = NULL; return ret;
}
}
ret = _regulator_do_enable(rdev); if (ret < 0 && ret != -EINVAL) {
rdev_err(rdev, "failed to enable: %pe\n", ERR_PTR(ret)); return ret;
}
if (!rdev->constraints->pw_budget_mW)
rdev->constraints->pw_budget_mW = INT_MAX;
print_constraints(rdev); return 0;
}
/** * set_supply - set regulator supply regulator * @rdev: regulator (locked) * @supply_rdev: supply regulator (locked)) * * Called by platform initialisation code to set the supply regulator for this * regulator. This ensures that a regulators supply will also be enabled by the * core if it's child is enabled. * * Return: 0 on success or a negative error number on failure.
*/ staticint set_supply(struct regulator_dev *rdev, struct regulator_dev *supply_rdev)
{ int err;
rdev_dbg(rdev, "supplied by %s\n", rdev_get_name(supply_rdev));
if (!try_module_get(supply_rdev->owner)) return -ENODEV;
/** * set_consumer_device_supply - Bind a regulator to a symbolic supply * @rdev: regulator source * @consumer_dev_name: dev_name() string for device supply applies to * @supply: symbolic name for supply * * Allows platform initialisation code to map physical regulator * sources to symbolic names for supplies for use by devices. Devices * should use these symbolic names to request regulators, avoiding the * need to provide board-specific regulator names as platform data. * * Return: 0 on success or a negative error number on failure.
*/ staticint set_consumer_device_supply(struct regulator_dev *rdev, constchar *consumer_dev_name, constchar *supply)
{ struct regulator_map *node, *new_node; int has_dev;
/* Add a link to the device sysfs entry */
err = sysfs_create_link_nowarn(&rdev->dev.kobj, &dev->kobj,
regulator->supply_name); if (err) {
rdev_dbg(rdev, "could not add device link %s: %pe\n",
dev->kobj.name, ERR_PTR(err)); /* non-fatal */
}
}
if (err != -EEXIST) {
regulator->debugfs = debugfs_create_dir(regulator->supply_name, rdev->debugfs); if (IS_ERR(regulator->debugfs)) {
rdev_dbg(rdev, "Failed to create debugfs directory\n");
regulator->debugfs = NULL;
}
}
/* * Check now if the regulator is an always on regulator - if * it is then we don't need to do nearly so much work for * enable/disable calls.
*/ if (!regulator_ops_is_valid(rdev, REGULATOR_CHANGE_STATUS) &&
_regulator_is_enabled(rdev))
regulator->always_on = true;
return regulator;
}
staticint _regulator_get_enable_time(struct regulator_dev *rdev)
{ if (rdev->constraints && rdev->constraints->enable_time) return rdev->constraints->enable_time; if (rdev->desc->ops->enable_time) return rdev->desc->ops->enable_time(rdev); return rdev->desc->enable_time;
}
if (dev_of_node(dev)) {
r = of_regulator_dev_lookup(dev, dev_of_node(dev), supply); if (PTR_ERR(r) == -ENODEV)
r = NULL;
}
return r;
}
/** * regulator_dev_lookup - lookup a regulator device. * @dev: device for regulator "consumer". * @supply: Supply name or regulator ID. * * Return: pointer to &struct regulator_dev or ERR_PTR() encoded negative error number. * * If successful, returns a struct regulator_dev that corresponds to the name * @supply and with the embedded struct device refcount incremented by one. * The refcount must be dropped by calling put_device(). * On failure one of the following ERR_PTR() encoded values is returned: * -%ENODEV if lookup fails permanently, -%EPROBE_DEFER if lookup could succeed * in the future.
*/ staticstruct regulator_dev *regulator_dev_lookup(struct device *dev, constchar *supply)
{ struct regulator_dev *r = NULL; struct regulator_map *map; constchar *devname = NULL;
regulator_supply_alias(&dev, &supply);
/* first do a dt based lookup */
r = regulator_dt_lookup(dev, supply); if (r) return r;
/* if not found, try doing it non-dt way */ if (dev)
devname = dev_name(dev);
mutex_lock(®ulator_list_mutex);
list_for_each_entry(map, ®ulator_map_list, list) { /* If the mapping has a device set up it must match */ if (map->dev_name &&
(!devname || strcmp(map->dev_name, devname))) continue;
if (strcmp(map->supply, supply) == 0 &&
get_device(&map->regulator->dev)) {
r = map->regulator; break;
}
}
mutex_unlock(®ulator_list_mutex);
if (r) return r;
r = regulator_lookup_by_name(supply); if (r) return r;
return ERR_PTR(-ENODEV);
}
staticint regulator_resolve_supply(struct regulator_dev *rdev)
{ struct regulator_dev *r; struct device *dev = rdev->dev.parent; struct ww_acquire_ctx ww_ctx; int ret = 0;
/* No supply to resolve? */ if (!rdev->supply_name) return 0;
/* Supply already resolved? (fast-path without locking contention) */ if (rdev->supply) return 0;
/* first do a dt based lookup on the node described in the virtual * device.
*/
r = regulator_dt_lookup(&rdev->dev, rdev->supply_name);
/* If regulator not found use usual search path in the parent * device.
*/ if (!r)
r = regulator_dev_lookup(dev, rdev->supply_name);
if (IS_ERR(r)) {
ret = PTR_ERR(r);
/* Did the lookup explicitly defer for us? */ if (ret == -EPROBE_DEFER) goto out;
if (have_full_constraints()) {
r = dummy_regulator_rdev; if (!r) {
ret = -EPROBE_DEFER; goto out;
}
get_device(&r->dev);
} else {
dev_err(dev, "Failed to resolve %s-supply for %s\n",
rdev->supply_name, rdev->desc->name);
ret = -EPROBE_DEFER; goto out;
}
}
if (r == rdev) {
dev_err(dev, "Supply for %s (%s) resolved to itself\n",
rdev->desc->name, rdev->supply_name); if (!have_full_constraints()) {
ret = -EINVAL; goto out;
}
r = dummy_regulator_rdev; if (!r) {
ret = -EPROBE_DEFER; goto out;
}
get_device(&r->dev);
}
/* * If the supply's parent device is not the same as the * regulator's parent device, then ensure the parent device * is bound before we resolve the supply, in case the parent * device get probe deferred and unregisters the supply.
*/ if (r->dev.parent && r->dev.parent != rdev->dev.parent) { if (!device_is_bound(r->dev.parent)) {
put_device(&r->dev);
ret = -EPROBE_DEFER; goto out;
}
}
/* Recursively resolve the supply of the supply */
ret = regulator_resolve_supply(r); if (ret < 0) {
put_device(&r->dev); goto out;
}
/* * Recheck rdev->supply with rdev->mutex lock held to avoid a race * between rdev->supply null check and setting rdev->supply in * set_supply() from concurrent tasks.
*/
regulator_lock_two(rdev, r, &ww_ctx);
/* Supply just resolved by a concurrent task? */ if (rdev->supply) {
regulator_unlock_two(rdev, r, &ww_ctx);
put_device(&r->dev); goto out;
}
ret = set_supply(rdev, r); if (ret < 0) {
regulator_unlock_two(rdev, r, &ww_ctx);
put_device(&r->dev); goto out;
}
regulator_unlock_two(rdev, r, &ww_ctx);
/* rdev->supply was created in set_supply() */
link_and_create_debugfs(rdev->supply, r, &rdev->dev);
/* * In set_machine_constraints() we may have turned this regulator on * but we couldn't propagate to the supply if it hadn't been resolved * yet. Do it now.
*/ if (rdev->use_count) {
ret = regulator_enable(rdev->supply); if (ret < 0) {
_regulator_put(rdev->supply);
rdev->supply = NULL; goto out;
}
}
out: return ret;
}
/* common pre-checks for regulator requests */ int _regulator_get_common_check(struct device *dev, constchar *id, enum regulator_get_type get_type)
{ if (get_type >= MAX_GET_TYPE) {
dev_err(dev, "invalid type %d in %s\n", get_type, __func__); return -EINVAL;
}
if (id == NULL) {
dev_err(dev, "regulator request with no identifier\n"); return -EINVAL;
}
return 0;
}
/** * _regulator_get_common - Common code for regulator requests * @rdev: regulator device pointer as returned by *regulator_dev_lookup() * Its reference count is expected to have been incremented. * @dev: device used for dev_printk messages * @id: Supply name or regulator ID * @get_type: enum regulator_get_type value corresponding to type of request * * Returns: pointer to struct regulator corresponding to @rdev, or ERR_PTR() * encoded error. * * This function should be chained with *regulator_dev_lookup() functions.
*/ struct regulator *_regulator_get_common(struct regulator_dev *rdev, struct device *dev, constchar *id, enum regulator_get_type get_type)
{ struct regulator *regulator; struct device_link *link; int ret;
if (IS_ERR(rdev)) {
ret = PTR_ERR(rdev);
/* * If regulator_dev_lookup() fails with error other * than -ENODEV our job here is done, we simply return it.
*/ if (ret != -ENODEV) return ERR_PTR(ret);
if (!have_full_constraints()) {
dev_warn(dev, "incomplete constraints, dummy supplies not allowed (id=%s)\n", id); return ERR_PTR(-ENODEV);
}
switch (get_type) { case NORMAL_GET: /* * Assume that a regulator is physically present and * enabled, even if it isn't hooked up, and just * provide a dummy.
*/
rdev = dummy_regulator_rdev; if (!rdev) return ERR_PTR(-EPROBE_DEFER);
dev_warn(dev, "supply %s not found, using dummy regulator\n", id);
get_device(&rdev->dev); break;
case EXCLUSIVE_GET:
dev_warn(dev, "dummy supplies not allowed for exclusive requests (id=%s)\n", id);
fallthrough;
default: return ERR_PTR(-ENODEV);
}
}
if (rdev->exclusive) {
regulator = ERR_PTR(-EPERM);
put_device(&rdev->dev); return regulator;
}
/** * regulator_get - lookup and obtain a reference to a regulator. * @dev: device for regulator "consumer" * @id: Supply name or regulator ID. * * Use of supply names configured via set_consumer_device_supply() is * strongly encouraged. It is recommended that the supply name used * should match the name used for the supply and/or the relevant * device pins in the datasheet. * * Return: Pointer to a &struct regulator corresponding to the regulator * producer, or an ERR_PTR() encoded negative error number.
*/ struct regulator *regulator_get(struct device *dev, constchar *id)
{ return _regulator_get(dev, id, NORMAL_GET);
}
EXPORT_SYMBOL_GPL(regulator_get);
/** * regulator_get_exclusive - obtain exclusive access to a regulator. * @dev: device for regulator "consumer" * @id: Supply name or regulator ID. * * Other consumers will be unable to obtain this regulator while this * reference is held and the use count for the regulator will be * initialised to reflect the current state of the regulator. * * This is intended for use by consumers which cannot tolerate shared * use of the regulator such as those which need to force the * regulator off for correct operation of the hardware they are * controlling. * * Use of supply names configured via set_consumer_device_supply() is * strongly encouraged. It is recommended that the supply name used * should match the name used for the supply and/or the relevant * device pins in the datasheet. * * Return: Pointer to a &struct regulator corresponding to the regulator * producer, or an ERR_PTR() encoded negative error number.
*/ struct regulator *regulator_get_exclusive(struct device *dev, constchar *id)
{ return _regulator_get(dev, id, EXCLUSIVE_GET);
}
EXPORT_SYMBOL_GPL(regulator_get_exclusive);
/** * regulator_get_optional - obtain optional access to a regulator. * @dev: device for regulator "consumer" * @id: Supply name or regulator ID. * * This is intended for use by consumers for devices which can have * some supplies unconnected in normal use, such as some MMC devices. * It can allow the regulator core to provide stub supplies for other * supplies requested using normal regulator_get() calls without * disrupting the operation of drivers that can handle absent * supplies. * * Use of supply names configured via set_consumer_device_supply() is * strongly encouraged. It is recommended that the supply name used * should match the name used for the supply and/or the relevant * device pins in the datasheet. * * Return: Pointer to a &struct regulator corresponding to the regulator * producer, or an ERR_PTR() encoded negative error number.
*/ struct regulator *regulator_get_optional(struct device *dev, constchar *id)
{ return _regulator_get(dev, id, OPTIONAL_GET);
}
EXPORT_SYMBOL_GPL(regulator_get_optional);
/** * regulator_put - "free" the regulator source * @regulator: regulator source * * Note: drivers must ensure that all regulator_enable calls made on this * regulator source are balanced by regulator_disable calls prior to calling * this function.
*/ void regulator_put(struct regulator *regulator)
{
mutex_lock(®ulator_list_mutex);
_regulator_put(regulator);
mutex_unlock(®ulator_list_mutex);
}
EXPORT_SYMBOL_GPL(regulator_put);
/** * regulator_register_supply_alias - Provide device alias for supply lookup * * @dev: device that will be given as the regulator "consumer" * @id: Supply name or regulator ID * @alias_dev: device that should be used to lookup the supply * @alias_id: Supply name or regulator ID that should be used to lookup the * supply * * All lookups for id on dev will instead be conducted for alias_id on * alias_dev. *
--> --------------------
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.