ALSA: vmaster: Add snd_ctl_sync_vmaster() helper function
Introduce a new helper function, snd_ctl_sync_vmaster(), which updates the slave put callbacks forcibly as well as calling the hook. This will be used in the upcoming patch in HD-audio codec driver for toggling the mute in vmaster slaves. Along with the new function, the old snd_ctl_sync_vmaster_hook() is replaced as a macro calling with the argument hook_only=true. Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
parent
e1a4dca671
commit
1ca2f2ec9e
@ -233,7 +233,8 @@ snd_ctl_add_slave_uncached(struct snd_kcontrol *master,
|
||||
int snd_ctl_add_vmaster_hook(struct snd_kcontrol *kctl,
|
||||
void (*hook)(void *private_data, int),
|
||||
void *private_data);
|
||||
void snd_ctl_sync_vmaster_hook(struct snd_kcontrol *kctl);
|
||||
void snd_ctl_sync_vmaster(struct snd_kcontrol *kctl, bool hook_only);
|
||||
#define snd_ctl_sync_vmaster_hook(kctl) snd_ctl_sync_vmaster(kctl, true)
|
||||
|
||||
/*
|
||||
* Helper functions for jack-detection controls
|
||||
|
@ -310,20 +310,10 @@ static int master_get(struct snd_kcontrol *kcontrol,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int master_put(struct snd_kcontrol *kcontrol,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
static int sync_slaves(struct link_master *master, int old_val, int new_val)
|
||||
{
|
||||
struct link_master *master = snd_kcontrol_chip(kcontrol);
|
||||
struct link_slave *slave;
|
||||
struct snd_ctl_elem_value *uval;
|
||||
int err, old_val;
|
||||
|
||||
err = master_init(master);
|
||||
if (err < 0)
|
||||
return err;
|
||||
old_val = master->val;
|
||||
if (ucontrol->value.integer.value[0] == old_val)
|
||||
return 0;
|
||||
|
||||
uval = kmalloc(sizeof(*uval), GFP_KERNEL);
|
||||
if (!uval)
|
||||
@ -332,11 +322,33 @@ static int master_put(struct snd_kcontrol *kcontrol,
|
||||
master->val = old_val;
|
||||
uval->id = slave->slave.id;
|
||||
slave_get_val(slave, uval);
|
||||
master->val = ucontrol->value.integer.value[0];
|
||||
master->val = new_val;
|
||||
slave_put_val(slave, uval);
|
||||
}
|
||||
kfree(uval);
|
||||
if (master->hook && !err)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int master_put(struct snd_kcontrol *kcontrol,
|
||||
struct snd_ctl_elem_value *ucontrol)
|
||||
{
|
||||
struct link_master *master = snd_kcontrol_chip(kcontrol);
|
||||
int err, new_val, old_val;
|
||||
bool first_init;
|
||||
|
||||
err = master_init(master);
|
||||
if (err < 0)
|
||||
return err;
|
||||
first_init = err;
|
||||
old_val = master->val;
|
||||
new_val = ucontrol->value.integer.value[0];
|
||||
if (new_val == old_val)
|
||||
return 0;
|
||||
|
||||
err = sync_slaves(master, old_val, new_val);
|
||||
if (err < 0)
|
||||
return err;
|
||||
if (master->hook && first_init)
|
||||
master->hook(master->hook_private_data, master->val);
|
||||
return 1;
|
||||
}
|
||||
@ -442,20 +454,33 @@ int snd_ctl_add_vmaster_hook(struct snd_kcontrol *kcontrol,
|
||||
EXPORT_SYMBOL_GPL(snd_ctl_add_vmaster_hook);
|
||||
|
||||
/**
|
||||
* snd_ctl_sync_vmaster_hook - Sync the vmaster hook
|
||||
* snd_ctl_sync_vmaster - Sync the vmaster slaves and hook
|
||||
* @kcontrol: vmaster kctl element
|
||||
* @hook_only: sync only the hook
|
||||
*
|
||||
* Call the hook function to synchronize with the current value of the given
|
||||
* vmaster element. NOP when NULL is passed to @kcontrol or the hook doesn't
|
||||
* exist.
|
||||
* Forcibly call the put callback of each slave and call the hook function
|
||||
* to synchronize with the current value of the given vmaster element.
|
||||
* NOP when NULL is passed to @kcontrol.
|
||||
*/
|
||||
void snd_ctl_sync_vmaster_hook(struct snd_kcontrol *kcontrol)
|
||||
void snd_ctl_sync_vmaster(struct snd_kcontrol *kcontrol, bool hook_only)
|
||||
{
|
||||
struct link_master *master;
|
||||
bool first_init = false;
|
||||
|
||||
if (!kcontrol)
|
||||
return;
|
||||
master = snd_kcontrol_chip(kcontrol);
|
||||
if (master->hook)
|
||||
if (!hook_only) {
|
||||
int err = master_init(master);
|
||||
if (err < 0)
|
||||
return;
|
||||
first_init = err;
|
||||
err = sync_slaves(master, master->val, master->val);
|
||||
if (err < 0)
|
||||
return;
|
||||
}
|
||||
|
||||
if (master->hook && !first_init)
|
||||
master->hook(master->hook_private_data, master->val);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(snd_ctl_sync_vmaster_hook);
|
||||
EXPORT_SYMBOL_GPL(snd_ctl_sync_vmaster);
|
||||
|
Loading…
Reference in New Issue
Block a user