/* * The idle-as-is "state" is not an actual state that may be selected, it * only implies that the state should not be changed. So, use that state * as indication that the cached state of the multiplexer is unknown.
*/ #define MUX_CACHE_UNKNOWN MUX_IDLE_AS_IS
/** * struct mux_state - Represents a mux controller state specific to a given * consumer. * @mux: Pointer to a mux controller. * @state: State of the mux to be selected. * * This structure is specific to the consumer that acquires it and has * information specific to that consumer.
*/ struct mux_state { struct mux_control *mux; unsignedint state;
};
/** * mux_chip_alloc() - Allocate a mux-chip. * @dev: The parent device implementing the mux interface. * @controllers: The number of mux controllers to allocate for this chip. * @sizeof_priv: Size of extra memory area for private use by the caller. * * After allocating the mux-chip with the desired number of mux controllers * but before registering the chip, the mux driver is required to configure * the number of valid mux states in the mux_chip->mux[N].states members and * the desired idle state in the returned mux_chip->mux[N].idle_state members. * The default idle state is MUX_IDLE_AS_IS. The mux driver also needs to * provide a pointer to the operations struct in the mux_chip->ops member * before registering the mux-chip with mux_chip_register. * * Return: A pointer to the new mux-chip, or an ERR_PTR with a negative errno.
*/ struct mux_chip *mux_chip_alloc(struct device *dev, unsignedint controllers, size_t sizeof_priv)
{ struct mux_chip *mux_chip; int i;
if (WARN_ON(!dev || !controllers)) return ERR_PTR(-EINVAL);
staticint mux_control_set(struct mux_control *mux, int state)
{ int ret = mux->chip->ops->set(mux, state);
mux->cached_state = ret < 0 ? MUX_CACHE_UNKNOWN : state; if (ret >= 0)
mux->last_change = ktime_get();
return ret;
}
/** * mux_chip_register() - Register a mux-chip, thus readying the controllers * for use. * @mux_chip: The mux-chip to register. * * Do not retry registration of the same mux-chip on failure. You should * instead put it away with mux_chip_free() and allocate a new one, if you * for some reason would like to retry registration. * * Return: Zero on success or a negative errno on error.
*/ int mux_chip_register(struct mux_chip *mux_chip)
{ int i; int ret;
for (i = 0; i < mux_chip->controllers; ++i) { struct mux_control *mux = &mux_chip->mux[i];
if (mux->idle_state == mux->cached_state) continue;
ret = mux_control_set(mux, mux->idle_state); if (ret < 0) {
dev_err(&mux_chip->dev, "unable to set idle state\n"); return ret;
}
}
ret = device_add(&mux_chip->dev); if (ret < 0)
dev_err(&mux_chip->dev, "device_add failed in %s: %d\n", __func__, ret); return ret;
}
EXPORT_SYMBOL_GPL(mux_chip_register);
/** * mux_chip_unregister() - Take the mux-chip off-line. * @mux_chip: The mux-chip to unregister. * * mux_chip_unregister() reverses the effects of mux_chip_register(). * But not completely, you should not try to call mux_chip_register() * on a mux-chip that has been registered before.
*/ void mux_chip_unregister(struct mux_chip *mux_chip)
{
device_del(&mux_chip->dev);
}
EXPORT_SYMBOL_GPL(mux_chip_unregister);
/** * mux_chip_free() - Free the mux-chip for good. * @mux_chip: The mux-chip to free. * * mux_chip_free() reverses the effects of mux_chip_alloc().
*/ void mux_chip_free(struct mux_chip *mux_chip)
{ if (!mux_chip) return;
/** * devm_mux_chip_alloc() - Resource-managed version of mux_chip_alloc(). * @dev: The parent device implementing the mux interface. * @controllers: The number of mux controllers to allocate for this chip. * @sizeof_priv: Size of extra memory area for private use by the caller. * * See mux_chip_alloc() for more details. * * Return: A pointer to the new mux-chip, or an ERR_PTR with a negative errno.
*/ struct mux_chip *devm_mux_chip_alloc(struct device *dev, unsignedint controllers,
size_t sizeof_priv)
{ struct mux_chip **ptr, *mux_chip;
ptr = devres_alloc(devm_mux_chip_release, sizeof(*ptr), GFP_KERNEL); if (!ptr) return ERR_PTR(-ENOMEM);
/** * devm_mux_chip_register() - Resource-managed version mux_chip_register(). * @dev: The parent device implementing the mux interface. * @mux_chip: The mux-chip to register. * * See mux_chip_register() for more details. * * Return: Zero on success or a negative errno on error.
*/ int devm_mux_chip_register(struct device *dev, struct mux_chip *mux_chip)
{ struct mux_chip **ptr; int res;
ptr = devres_alloc(devm_mux_chip_reg_release, sizeof(*ptr), GFP_KERNEL); if (!ptr) return -ENOMEM;
res = mux_chip_register(mux_chip); if (res) {
devres_free(ptr); return res;
}
/** * mux_control_states() - Query the number of multiplexer states. * @mux: The mux-control to query. * * Return: The number of multiplexer states.
*/ unsignedint mux_control_states(struct mux_control *mux)
{ return mux->states;
}
EXPORT_SYMBOL_GPL(mux_control_states);
/* * The mux->lock must be down when calling this function.
*/ staticint __mux_control_select(struct mux_control *mux, int state)
{ int ret;
if (WARN_ON(state < 0 || state >= mux->states)) return -EINVAL;
if (mux->cached_state == state) return 0;
ret = mux_control_set(mux, state); if (ret >= 0) return 0;
/* The mux update failed, try to revert if appropriate... */ if (mux->idle_state != MUX_IDLE_AS_IS)
mux_control_set(mux, mux->idle_state);
/** * mux_control_select_delay() - Select the given multiplexer state. * @mux: The mux-control to request a change of state from. * @state: The new requested state. * @delay_us: The time to delay (in microseconds) if the mux state is changed. * * On successfully selecting the mux-control state, it will be locked until * there is a call to mux_control_deselect(). If the mux-control is already * selected when mux_control_select() is called, the caller will be blocked * until mux_control_deselect() or mux_state_deselect() is called (by someone * else). * * Therefore, make sure to call mux_control_deselect() when the operation is * complete and the mux-control is free for others to use, but do not call * mux_control_deselect() if mux_control_select() fails. * * Return: 0 when the mux-control state has the requested state or a negative * errno on error.
*/ int mux_control_select_delay(struct mux_control *mux, unsignedint state, unsignedint delay_us)
{ int ret;
ret = down_killable(&mux->lock); if (ret < 0) return ret;
ret = __mux_control_select(mux, state); if (ret >= 0)
mux_control_delay(mux, delay_us);
/** * mux_state_select_delay() - Select the given multiplexer state. * @mstate: The mux-state to select. * @delay_us: The time to delay (in microseconds) if the mux state is changed. * * On successfully selecting the mux-state, its mux-control will be locked * until there is a call to mux_state_deselect(). If the mux-control is already * selected when mux_state_select() is called, the caller will be blocked * until mux_state_deselect() or mux_control_deselect() is called (by someone * else). * * Therefore, make sure to call mux_state_deselect() when the operation is * complete and the mux-control is free for others to use, but do not call * mux_state_deselect() if mux_state_select() fails. * * Return: 0 when the mux-state has been selected or a negative * errno on error.
*/ int mux_state_select_delay(struct mux_state *mstate, unsignedint delay_us)
{ return mux_control_select_delay(mstate->mux, mstate->state, delay_us);
}
EXPORT_SYMBOL_GPL(mux_state_select_delay);
/** * mux_control_try_select_delay() - Try to select the given multiplexer state. * @mux: The mux-control to request a change of state from. * @state: The new requested state. * @delay_us: The time to delay (in microseconds) if the mux state is changed. * * On successfully selecting the mux-control state, it will be locked until * mux_control_deselect() is called. * * Therefore, make sure to call mux_control_deselect() when the operation is * complete and the mux-control is free for others to use, but do not call * mux_control_deselect() if mux_control_try_select() fails. * * Return: 0 when the mux-control state has the requested state or a negative * errno on error. Specifically -EBUSY if the mux-control is contended.
*/ int mux_control_try_select_delay(struct mux_control *mux, unsignedint state, unsignedint delay_us)
{ int ret;
if (down_trylock(&mux->lock)) return -EBUSY;
ret = __mux_control_select(mux, state); if (ret >= 0)
mux_control_delay(mux, delay_us);
/** * mux_state_try_select_delay() - Try to select the given multiplexer state. * @mstate: The mux-state to select. * @delay_us: The time to delay (in microseconds) if the mux state is changed. * * On successfully selecting the mux-state, its mux-control will be locked * until mux_state_deselect() is called. * * Therefore, make sure to call mux_state_deselect() when the operation is * complete and the mux-control is free for others to use, but do not call * mux_state_deselect() if mux_state_try_select() fails. * * Return: 0 when the mux-state has been selected or a negative errno on * error. Specifically -EBUSY if the mux-control is contended.
*/ int mux_state_try_select_delay(struct mux_state *mstate, unsignedint delay_us)
{ return mux_control_try_select_delay(mstate->mux, mstate->state, delay_us);
}
EXPORT_SYMBOL_GPL(mux_state_try_select_delay);
/** * mux_control_deselect() - Deselect the previously selected multiplexer state. * @mux: The mux-control to deselect. * * It is required that a single call is made to mux_control_deselect() for * each and every successful call made to either of mux_control_select() or * mux_control_try_select(). * * Return: 0 on success and a negative errno on error. An error can only * occur if the mux has an idle state. Note that even if an error occurs, the * mux-control is unlocked and is thus free for the next access.
*/ int mux_control_deselect(struct mux_control *mux)
{ int ret = 0;
if (mux->idle_state != MUX_IDLE_AS_IS &&
mux->idle_state != mux->cached_state)
ret = mux_control_set(mux, mux->idle_state);
/** * mux_state_deselect() - Deselect the previously selected multiplexer state. * @mstate: The mux-state to deselect. * * It is required that a single call is made to mux_state_deselect() for * each and every successful call made to either of mux_state_select() or * mux_state_try_select(). * * Return: 0 on success and a negative errno on error. An error can only * occur if the mux has an idle state. Note that even if an error occurs, the * mux-control is unlocked and is thus free for the next access.
*/ int mux_state_deselect(struct mux_state *mstate)
{ return mux_control_deselect(mstate->mux);
}
EXPORT_SYMBOL_GPL(mux_state_deselect);
/* Note this function returns a reference to the mux_chip dev. */ staticstruct mux_chip *of_find_mux_chip_by_node(struct device_node *np)
{ struct device *dev;
dev = class_find_device_by_of_node(&mux_class, np);
return dev ? to_mux_chip(dev) : NULL;
}
/* * mux_get() - Get the mux-control for a device. * @dev: The device that needs a mux-control. * @mux_name: The name identifying the mux-control. * @state: Pointer to where the requested state is returned, or NULL when * the required multiplexer states are handled by other means. * * Return: A pointer to the mux-control, or an ERR_PTR with a negative errno.
*/ staticstruct mux_control *mux_get(struct device *dev, constchar *mux_name, unsignedint *state)
{ struct device_node *np = dev->of_node; struct of_phandle_args args; struct mux_chip *mux_chip; unsignedint controller; int index = 0; int ret;
if (mux_name) { if (state)
index = of_property_match_string(np, "mux-state-names",
mux_name); else
index = of_property_match_string(np, "mux-control-names",
mux_name); if (index < 0) {
dev_err(dev, "mux controller '%s' not found\n",
mux_name); return ERR_PTR(index);
}
}
if (state)
ret = of_parse_phandle_with_args(np, "mux-states", "#mux-state-cells",
index, &args); else
ret = of_parse_phandle_with_args(np, "mux-controls", "#mux-control-cells",
index, &args); if (ret) {
dev_err(dev, "%pOF: failed to get mux-%s %s(%i)\n",
np, state ? "state" : "control", mux_name ?: "", index); return ERR_PTR(ret);
}
mux_chip = of_find_mux_chip_by_node(args.np);
of_node_put(args.np); if (!mux_chip) return ERR_PTR(-EPROBE_DEFER);
if (controller >= mux_chip->controllers) {
dev_err(dev, "%pOF: bad mux controller %u specified in %pOF\n",
np, controller, args.np);
put_device(&mux_chip->dev); return ERR_PTR(-EINVAL);
}
return &mux_chip->mux[controller];
}
/** * mux_control_get() - Get the mux-control for a device. * @dev: The device that needs a mux-control. * @mux_name: The name identifying the mux-control. * * Return: A pointer to the mux-control, or an ERR_PTR with a negative errno.
*/ struct mux_control *mux_control_get(struct device *dev, constchar *mux_name)
{ return mux_get(dev, mux_name, NULL);
}
EXPORT_SYMBOL_GPL(mux_control_get);
/** * mux_control_put() - Put away the mux-control for good. * @mux: The mux-control to put away. * * mux_control_put() reverses the effects of mux_control_get().
*/ void mux_control_put(struct mux_control *mux)
{
put_device(&mux->chip->dev);
}
EXPORT_SYMBOL_GPL(mux_control_put);
/** * devm_mux_control_get() - Get the mux-control for a device, with resource * management. * @dev: The device that needs a mux-control. * @mux_name: The name identifying the mux-control. * * Return: Pointer to the mux-control, or an ERR_PTR with a negative errno.
*/ struct mux_control *devm_mux_control_get(struct device *dev, constchar *mux_name)
{ struct mux_control **ptr, *mux;
ptr = devres_alloc(devm_mux_control_release, sizeof(*ptr), GFP_KERNEL); if (!ptr) return ERR_PTR(-ENOMEM);
/* * mux_state_get() - Get the mux-state for a device. * @dev: The device that needs a mux-state. * @mux_name: The name identifying the mux-state. * * Return: A pointer to the mux-state, or an ERR_PTR with a negative errno.
*/ staticstruct mux_state *mux_state_get(struct device *dev, constchar *mux_name)
{ struct mux_state *mstate;
mstate = kzalloc(sizeof(*mstate), GFP_KERNEL); if (!mstate) return ERR_PTR(-ENOMEM);
mstate->mux = mux_get(dev, mux_name, &mstate->state); if (IS_ERR(mstate->mux)) { int err = PTR_ERR(mstate->mux);
kfree(mstate); return ERR_PTR(err);
}
return mstate;
}
/* * mux_state_put() - Put away the mux-state for good. * @mstate: The mux-state to put away. * * mux_state_put() reverses the effects of mux_state_get().
*/ staticvoid mux_state_put(struct mux_state *mstate)
{
mux_control_put(mstate->mux);
kfree(mstate);
}
/** * devm_mux_state_get() - Get the mux-state for a device, with resource * management. * @dev: The device that needs a mux-control. * @mux_name: The name identifying the mux-control. * * Return: Pointer to the mux-state, or an ERR_PTR with a negative errno.
*/ struct mux_state *devm_mux_state_get(struct device *dev, constchar *mux_name)
{ struct mux_state **ptr, *mstate;
ptr = devres_alloc(devm_mux_state_release, sizeof(*ptr), GFP_KERNEL); if (!ptr) return ERR_PTR(-ENOMEM);
/* * Using subsys_initcall instead of module_init here to try to ensure - for * the non-modular case - that the subsystem is initialized when mux consumers * and mux controllers start to use it. * For the modular case, the ordering is ensured with module dependencies.
*/
subsys_initcall(mux_init);
module_exit(mux_exit);
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.