Commit d29d41e2 authored by Jaska Uimonen's avatar Jaska Uimonen Committed by Mark Brown

ASoC: topology: Add support for multiple kcontrol types to a widget

Current dapm widget has a single variable to describe its kcontrol's
type. As there can be many kcontrols in one widget it is inherently
presumed that the types are the same.

Lately there has been use cases where different types of kcontrols would
be needed for a single widget. Thus add pointer to dapm widget to hold
an array for different kcontrol types and modify the kcontrol creation
to operate in a loop based on individual kcontrol type.

Change control creation and deletion to use individual kcontrol types in
SOF driver. This is done in the same patch for not breaking bisect. SOF
driver is also currently the only one using the dapm widget
kcontrol_type.
Signed-off-by: default avatarJaska Uimonen <jaska.uimonen@linux.intel.com>
Reviewed-by: default avatarGuennadi Liakhovetski <guennadi.liakhovetski@linux.intel.com>
Reviewed-by: default avatarRanjani Sridharan <ranjani.sridharan@linux.intel.com>
Link: https://lore.kernel.org/r/20210507070246.404446-1-jaska.uimonen@linux.intel.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent a75e5cdf
...@@ -54,7 +54,7 @@ struct snd_soc_dobj_control { ...@@ -54,7 +54,7 @@ struct snd_soc_dobj_control {
/* dynamic widget object */ /* dynamic widget object */
struct snd_soc_dobj_widget { struct snd_soc_dobj_widget {
unsigned int kcontrol_type; /* kcontrol type: mixer, enum, bytes */ unsigned int *kcontrol_type; /* kcontrol type: mixer, enum, bytes */
}; };
/* generic dynamic object - all dynamic objects belong to this struct */ /* generic dynamic object - all dynamic objects belong to this struct */
......
...@@ -1203,249 +1203,216 @@ static int soc_tplg_dapm_graph_elems_load(struct soc_tplg *tplg, ...@@ -1203,249 +1203,216 @@ static int soc_tplg_dapm_graph_elems_load(struct soc_tplg *tplg,
return ret; return ret;
} }
static struct snd_kcontrol_new *soc_tplg_dapm_widget_dmixer_create( static int soc_tplg_dapm_widget_dmixer_create(struct soc_tplg *tplg, struct snd_kcontrol_new *kc)
struct soc_tplg *tplg, int num_kcontrols)
{ {
struct snd_kcontrol_new *kc;
struct soc_mixer_control *sm; struct soc_mixer_control *sm;
struct snd_soc_tplg_mixer_control *mc; struct snd_soc_tplg_mixer_control *mc;
int i, err; int err;
kc = devm_kcalloc(tplg->dev, num_kcontrols, sizeof(*kc), GFP_KERNEL);
if (kc == NULL)
return NULL;
for (i = 0; i < num_kcontrols; i++) {
mc = (struct snd_soc_tplg_mixer_control *)tplg->pos;
/* validate kcontrol */
if (strnlen(mc->hdr.name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) ==
SNDRV_CTL_ELEM_ID_NAME_MAXLEN)
goto err_sm;
sm = devm_kzalloc(tplg->dev, sizeof(*sm), GFP_KERNEL); mc = (struct snd_soc_tplg_mixer_control *)tplg->pos;
if (sm == NULL)
goto err_sm;
tplg->pos += (sizeof(struct snd_soc_tplg_mixer_control) + /* validate kcontrol */
le32_to_cpu(mc->priv.size)); if (strnlen(mc->hdr.name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) ==
SNDRV_CTL_ELEM_ID_NAME_MAXLEN)
return -EINVAL;
dev_dbg(tplg->dev, " adding DAPM widget mixer control %s at %d\n", sm = devm_kzalloc(tplg->dev, sizeof(*sm), GFP_KERNEL);
mc->hdr.name, i); if (!sm)
return -ENOMEM;
kc[i].private_value = (long)sm; tplg->pos += sizeof(struct snd_soc_tplg_mixer_control) +
kc[i].name = devm_kstrdup(tplg->dev, mc->hdr.name, GFP_KERNEL); le32_to_cpu(mc->priv.size);
if (kc[i].name == NULL)
goto err_sm;
kc[i].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
kc[i].access = le32_to_cpu(mc->hdr.access);
/* we only support FL/FR channel mapping atm */ dev_dbg(tplg->dev, " adding DAPM widget mixer control %s\n",
sm->reg = tplc_chan_get_reg(tplg, mc->channel, mc->hdr.name);
SNDRV_CHMAP_FL);
sm->rreg = tplc_chan_get_reg(tplg, mc->channel,
SNDRV_CHMAP_FR);
sm->shift = tplc_chan_get_shift(tplg, mc->channel,
SNDRV_CHMAP_FL);
sm->rshift = tplc_chan_get_shift(tplg, mc->channel,
SNDRV_CHMAP_FR);
sm->max = le32_to_cpu(mc->max); kc->private_value = (long)sm;
sm->min = le32_to_cpu(mc->min); kc->name = devm_kstrdup(tplg->dev, mc->hdr.name, GFP_KERNEL);
sm->invert = le32_to_cpu(mc->invert); if (!kc->name)
sm->platform_max = le32_to_cpu(mc->platform_max); return -ENOMEM;
sm->dobj.index = tplg->index; kc->iface = SNDRV_CTL_ELEM_IFACE_MIXER;
INIT_LIST_HEAD(&sm->dobj.list); kc->access = le32_to_cpu(mc->hdr.access);
/* map io handlers */ /* we only support FL/FR channel mapping atm */
err = soc_tplg_kcontrol_bind_io(&mc->hdr, &kc[i], tplg); sm->reg = tplc_chan_get_reg(tplg, mc->channel,
if (err) { SNDRV_CHMAP_FL);
soc_control_err(tplg, &mc->hdr, mc->hdr.name); sm->rreg = tplc_chan_get_reg(tplg, mc->channel,
goto err_sm; SNDRV_CHMAP_FR);
} sm->shift = tplc_chan_get_shift(tplg, mc->channel,
SNDRV_CHMAP_FL);
sm->rshift = tplc_chan_get_shift(tplg, mc->channel,
SNDRV_CHMAP_FR);
sm->max = le32_to_cpu(mc->max);
sm->min = le32_to_cpu(mc->min);
sm->invert = le32_to_cpu(mc->invert);
sm->platform_max = le32_to_cpu(mc->platform_max);
sm->dobj.index = tplg->index;
INIT_LIST_HEAD(&sm->dobj.list);
/* map io handlers */
err = soc_tplg_kcontrol_bind_io(&mc->hdr, kc, tplg);
if (err) {
soc_control_err(tplg, &mc->hdr, mc->hdr.name);
return err;
}
/* create any TLV data */ /* create any TLV data */
err = soc_tplg_create_tlv(tplg, &kc[i], &mc->hdr); err = soc_tplg_create_tlv(tplg, kc, &mc->hdr);
if (err < 0) { if (err < 0) {
dev_err(tplg->dev, "ASoC: failed to create TLV %s\n", dev_err(tplg->dev, "ASoC: failed to create TLV %s\n",
mc->hdr.name); mc->hdr.name);
goto err_sm; return err;
} }
/* pass control to driver for optional further init */ /* pass control to driver for optional further init */
err = soc_tplg_init_kcontrol(tplg, &kc[i], err = soc_tplg_init_kcontrol(tplg, kc,
(struct snd_soc_tplg_ctl_hdr *)mc); (struct snd_soc_tplg_ctl_hdr *)mc);
if (err < 0) { if (err < 0) {
dev_err(tplg->dev, "ASoC: failed to init %s\n", dev_err(tplg->dev, "ASoC: failed to init %s\n",
mc->hdr.name); mc->hdr.name);
goto err_sm; return err;
}
} }
return kc;
err_sm: return 0;
return NULL;
} }
static struct snd_kcontrol_new *soc_tplg_dapm_widget_denum_create( static int soc_tplg_dapm_widget_denum_create(struct soc_tplg *tplg, struct snd_kcontrol_new *kc)
struct soc_tplg *tplg, int num_kcontrols)
{ {
struct snd_kcontrol_new *kc;
struct snd_soc_tplg_enum_control *ec; struct snd_soc_tplg_enum_control *ec;
struct soc_enum *se; struct soc_enum *se;
int i, err; int err;
kc = devm_kcalloc(tplg->dev, num_kcontrols, sizeof(*kc), GFP_KERNEL);
if (kc == NULL)
return NULL;
for (i = 0; i < num_kcontrols; i++) {
ec = (struct snd_soc_tplg_enum_control *)tplg->pos;
/* validate kcontrol */
if (strnlen(ec->hdr.name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) ==
SNDRV_CTL_ELEM_ID_NAME_MAXLEN)
goto err_se;
se = devm_kzalloc(tplg->dev, sizeof(*se), GFP_KERNEL); ec = (struct snd_soc_tplg_enum_control *)tplg->pos;
if (se == NULL) /* validate kcontrol */
goto err_se; if (strnlen(ec->hdr.name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) ==
SNDRV_CTL_ELEM_ID_NAME_MAXLEN)
return -EINVAL;
tplg->pos += (sizeof(struct snd_soc_tplg_enum_control) + se = devm_kzalloc(tplg->dev, sizeof(*se), GFP_KERNEL);
le32_to_cpu(ec->priv.size)); if (!se)
return -ENOMEM;
dev_dbg(tplg->dev, " adding DAPM widget enum control %s\n", tplg->pos += (sizeof(struct snd_soc_tplg_enum_control) +
ec->hdr.name); le32_to_cpu(ec->priv.size));
kc[i].private_value = (long)se; dev_dbg(tplg->dev, " adding DAPM widget enum control %s\n",
kc[i].name = devm_kstrdup(tplg->dev, ec->hdr.name, GFP_KERNEL); ec->hdr.name);
if (kc[i].name == NULL)
goto err_se;
kc[i].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
kc[i].access = le32_to_cpu(ec->hdr.access);
/* we only support FL/FR channel mapping atm */ kc->private_value = (long)se;
se->reg = tplc_chan_get_reg(tplg, ec->channel, SNDRV_CHMAP_FL); kc->name = devm_kstrdup(tplg->dev, ec->hdr.name, GFP_KERNEL);
se->shift_l = tplc_chan_get_shift(tplg, ec->channel, if (!kc->name)
SNDRV_CHMAP_FL); return -ENOMEM;
se->shift_r = tplc_chan_get_shift(tplg, ec->channel, kc->iface = SNDRV_CTL_ELEM_IFACE_MIXER;
SNDRV_CHMAP_FR); kc->access = le32_to_cpu(ec->hdr.access);
se->items = le32_to_cpu(ec->items); /* we only support FL/FR channel mapping atm */
se->mask = le32_to_cpu(ec->mask); se->reg = tplc_chan_get_reg(tplg, ec->channel, SNDRV_CHMAP_FL);
se->dobj.index = tplg->index; se->shift_l = tplc_chan_get_shift(tplg, ec->channel,
SNDRV_CHMAP_FL);
se->shift_r = tplc_chan_get_shift(tplg, ec->channel,
SNDRV_CHMAP_FR);
switch (le32_to_cpu(ec->hdr.ops.info)) { se->items = le32_to_cpu(ec->items);
case SND_SOC_TPLG_CTL_ENUM_VALUE: se->mask = le32_to_cpu(ec->mask);
case SND_SOC_TPLG_DAPM_CTL_ENUM_VALUE: se->dobj.index = tplg->index;
err = soc_tplg_denum_create_values(tplg, se, ec);
if (err < 0) {
dev_err(tplg->dev, "ASoC: could not create values for %s\n",
ec->hdr.name);
goto err_se;
}
fallthrough;
case SND_SOC_TPLG_CTL_ENUM:
case SND_SOC_TPLG_DAPM_CTL_ENUM_DOUBLE:
case SND_SOC_TPLG_DAPM_CTL_ENUM_VIRT:
err = soc_tplg_denum_create_texts(tplg, se, ec);
if (err < 0) {
dev_err(tplg->dev, "ASoC: could not create texts for %s\n",
ec->hdr.name);
goto err_se;
}
break;
default:
dev_err(tplg->dev, "ASoC: invalid enum control type %d for %s\n",
ec->hdr.ops.info, ec->hdr.name);
goto err_se;
}
/* map io handlers */ switch (le32_to_cpu(ec->hdr.ops.info)) {
err = soc_tplg_kcontrol_bind_io(&ec->hdr, &kc[i], tplg); case SND_SOC_TPLG_CTL_ENUM_VALUE:
if (err) { case SND_SOC_TPLG_DAPM_CTL_ENUM_VALUE:
soc_control_err(tplg, &ec->hdr, ec->hdr.name); err = soc_tplg_denum_create_values(tplg, se, ec);
goto err_se; if (err < 0) {
dev_err(tplg->dev, "ASoC: could not create values for %s\n",
ec->hdr.name);
return err;
} }
fallthrough;
/* pass control to driver for optional further init */ case SND_SOC_TPLG_CTL_ENUM:
err = soc_tplg_init_kcontrol(tplg, &kc[i], case SND_SOC_TPLG_DAPM_CTL_ENUM_DOUBLE:
(struct snd_soc_tplg_ctl_hdr *)ec); case SND_SOC_TPLG_DAPM_CTL_ENUM_VIRT:
err = soc_tplg_denum_create_texts(tplg, se, ec);
if (err < 0) { if (err < 0) {
dev_err(tplg->dev, "ASoC: failed to init %s\n", dev_err(tplg->dev, "ASoC: could not create texts for %s\n",
ec->hdr.name); ec->hdr.name);
goto err_se; return err;
} }
break;
default:
dev_err(tplg->dev, "ASoC: invalid enum control type %d for %s\n",
ec->hdr.ops.info, ec->hdr.name);
return -EINVAL;
} }
return kc; /* map io handlers */
err = soc_tplg_kcontrol_bind_io(&ec->hdr, kc, tplg);
if (err) {
soc_control_err(tplg, &ec->hdr, ec->hdr.name);
return err;
}
err_se: /* pass control to driver for optional further init */
return NULL; err = soc_tplg_init_kcontrol(tplg, kc,
(struct snd_soc_tplg_ctl_hdr *)ec);
if (err < 0) {
dev_err(tplg->dev, "ASoC: failed to init %s\n",
ec->hdr.name);
return err;
}
return 0;
} }
static struct snd_kcontrol_new *soc_tplg_dapm_widget_dbytes_create( static int soc_tplg_dapm_widget_dbytes_create(struct soc_tplg *tplg, struct snd_kcontrol_new *kc)
struct soc_tplg *tplg, int num_kcontrols)
{ {
struct snd_soc_tplg_bytes_control *be; struct snd_soc_tplg_bytes_control *be;
struct soc_bytes_ext *sbe; struct soc_bytes_ext *sbe;
struct snd_kcontrol_new *kc; int err;
int i, err;
kc = devm_kcalloc(tplg->dev, num_kcontrols, sizeof(*kc), GFP_KERNEL);
if (!kc)
return NULL;
for (i = 0; i < num_kcontrols; i++) { be = (struct snd_soc_tplg_bytes_control *)tplg->pos;
be = (struct snd_soc_tplg_bytes_control *)tplg->pos;
/* validate kcontrol */ /* validate kcontrol */
if (strnlen(be->hdr.name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) == if (strnlen(be->hdr.name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) ==
SNDRV_CTL_ELEM_ID_NAME_MAXLEN) SNDRV_CTL_ELEM_ID_NAME_MAXLEN)
goto err_sbe; return -EINVAL;
sbe = devm_kzalloc(tplg->dev, sizeof(*sbe), GFP_KERNEL); sbe = devm_kzalloc(tplg->dev, sizeof(*sbe), GFP_KERNEL);
if (sbe == NULL) if (!sbe)
goto err_sbe; return -ENOMEM;
tplg->pos += (sizeof(struct snd_soc_tplg_bytes_control) + tplg->pos += (sizeof(struct snd_soc_tplg_bytes_control) +
le32_to_cpu(be->priv.size)); le32_to_cpu(be->priv.size));
dev_dbg(tplg->dev, dev_dbg(tplg->dev,
"ASoC: adding bytes kcontrol %s with access 0x%x\n", "ASoC: adding bytes kcontrol %s with access 0x%x\n",
be->hdr.name, be->hdr.access); be->hdr.name, be->hdr.access);
kc[i].private_value = (long)sbe; kc->private_value = (long)sbe;
kc[i].name = devm_kstrdup(tplg->dev, be->hdr.name, GFP_KERNEL); kc->name = devm_kstrdup(tplg->dev, be->hdr.name, GFP_KERNEL);
if (kc[i].name == NULL) if (!kc->name)
goto err_sbe; return -ENOMEM;
kc[i].iface = SNDRV_CTL_ELEM_IFACE_MIXER; kc->iface = SNDRV_CTL_ELEM_IFACE_MIXER;
kc[i].access = le32_to_cpu(be->hdr.access); kc->access = le32_to_cpu(be->hdr.access);
sbe->max = le32_to_cpu(be->max); sbe->max = le32_to_cpu(be->max);
INIT_LIST_HEAD(&sbe->dobj.list); INIT_LIST_HEAD(&sbe->dobj.list);
/* map standard io handlers and check for external handlers */ /* map standard io handlers and check for external handlers */
err = soc_tplg_kcontrol_bind_io(&be->hdr, &kc[i], tplg); err = soc_tplg_kcontrol_bind_io(&be->hdr, kc, tplg);
if (err) { if (err) {
soc_control_err(tplg, &be->hdr, be->hdr.name); soc_control_err(tplg, &be->hdr, be->hdr.name);
goto err_sbe; return err;
}
/* pass control to driver for optional further init */
err = soc_tplg_init_kcontrol(tplg, &kc[i],
(struct snd_soc_tplg_ctl_hdr *)be);
if (err < 0) {
dev_err(tplg->dev, "ASoC: failed to init %s\n",
be->hdr.name);
goto err_sbe;
}
} }
return kc; /* pass control to driver for optional further init */
err = soc_tplg_init_kcontrol(tplg, kc,
err_sbe: (struct snd_soc_tplg_ctl_hdr *)be);
if (err < 0) {
dev_err(tplg->dev, "ASoC: failed to init %s\n",
be->hdr.name);
return err;
}
return NULL; return 0;
} }
static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg, static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg,
...@@ -1455,8 +1422,13 @@ static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg, ...@@ -1455,8 +1422,13 @@ static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg,
struct snd_soc_dapm_widget template, *widget; struct snd_soc_dapm_widget template, *widget;
struct snd_soc_tplg_ctl_hdr *control_hdr; struct snd_soc_tplg_ctl_hdr *control_hdr;
struct snd_soc_card *card = tplg->comp->card; struct snd_soc_card *card = tplg->comp->card;
unsigned int kcontrol_type; unsigned int *kcontrol_type;
struct snd_kcontrol_new *kc;
int mixer_count = 0;
int bytes_count = 0;
int enum_count = 0;
int ret = 0; int ret = 0;
int i;
if (strnlen(w->name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) == if (strnlen(w->name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN) ==
SNDRV_CTL_ELEM_ID_NAME_MAXLEN) SNDRV_CTL_ELEM_ID_NAME_MAXLEN)
...@@ -1499,7 +1471,6 @@ static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg, ...@@ -1499,7 +1471,6 @@ static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg,
le32_to_cpu(w->priv.size)); le32_to_cpu(w->priv.size));
if (w->num_kcontrols == 0) { if (w->num_kcontrols == 0) {
kcontrol_type = 0;
template.num_kcontrols = 0; template.num_kcontrols = 0;
goto widget; goto widget;
} }
...@@ -1508,57 +1479,66 @@ static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg, ...@@ -1508,57 +1479,66 @@ static int soc_tplg_dapm_widget_create(struct soc_tplg *tplg,
dev_dbg(tplg->dev, "ASoC: template %s has %d controls of type %x\n", dev_dbg(tplg->dev, "ASoC: template %s has %d controls of type %x\n",
w->name, w->num_kcontrols, control_hdr->type); w->name, w->num_kcontrols, control_hdr->type);
switch (le32_to_cpu(control_hdr->ops.info)) { template.num_kcontrols = le32_to_cpu(w->num_kcontrols);
case SND_SOC_TPLG_CTL_VOLSW: kc = devm_kcalloc(tplg->dev, le32_to_cpu(w->num_kcontrols), sizeof(*kc), GFP_KERNEL);
case SND_SOC_TPLG_CTL_STROBE: if (!kc)
case SND_SOC_TPLG_CTL_VOLSW_SX: goto err;
case SND_SOC_TPLG_CTL_VOLSW_XR_SX:
case SND_SOC_TPLG_CTL_RANGE: kcontrol_type = devm_kcalloc(tplg->dev, le32_to_cpu(w->num_kcontrols), sizeof(unsigned int),
case SND_SOC_TPLG_DAPM_CTL_VOLSW: GFP_KERNEL);
kcontrol_type = SND_SOC_TPLG_TYPE_MIXER; /* volume mixer */ if (!kcontrol_type)
template.num_kcontrols = le32_to_cpu(w->num_kcontrols); goto err;
template.kcontrol_news =
soc_tplg_dapm_widget_dmixer_create(tplg, for (i = 0; i < w->num_kcontrols; i++) {
template.num_kcontrols); control_hdr = (struct snd_soc_tplg_ctl_hdr *)tplg->pos;
if (!template.kcontrol_news) { switch (le32_to_cpu(control_hdr->ops.info)) {
ret = -ENOMEM; case SND_SOC_TPLG_CTL_VOLSW:
goto hdr_err; case SND_SOC_TPLG_CTL_STROBE:
} case SND_SOC_TPLG_CTL_VOLSW_SX:
break; case SND_SOC_TPLG_CTL_VOLSW_XR_SX:
case SND_SOC_TPLG_CTL_ENUM: case SND_SOC_TPLG_CTL_RANGE:
case SND_SOC_TPLG_CTL_ENUM_VALUE: case SND_SOC_TPLG_DAPM_CTL_VOLSW:
case SND_SOC_TPLG_DAPM_CTL_ENUM_DOUBLE: /* volume mixer */
case SND_SOC_TPLG_DAPM_CTL_ENUM_VIRT: kc[i].index = mixer_count;
case SND_SOC_TPLG_DAPM_CTL_ENUM_VALUE: kcontrol_type[i] = SND_SOC_TPLG_TYPE_MIXER;
kcontrol_type = SND_SOC_TPLG_TYPE_ENUM; /* enumerated mixer */ mixer_count++;
template.num_kcontrols = le32_to_cpu(w->num_kcontrols); ret = soc_tplg_dapm_widget_dmixer_create(tplg, &kc[i]);
template.kcontrol_news = if (ret < 0)
soc_tplg_dapm_widget_denum_create(tplg, goto hdr_err;
template.num_kcontrols); break;
if (!template.kcontrol_news) { case SND_SOC_TPLG_CTL_ENUM:
ret = -ENOMEM; case SND_SOC_TPLG_CTL_ENUM_VALUE:
goto hdr_err; case SND_SOC_TPLG_DAPM_CTL_ENUM_DOUBLE:
} case SND_SOC_TPLG_DAPM_CTL_ENUM_VIRT:
break; case SND_SOC_TPLG_DAPM_CTL_ENUM_VALUE:
case SND_SOC_TPLG_CTL_BYTES: /* enumerated mixer */
kcontrol_type = SND_SOC_TPLG_TYPE_BYTES; /* bytes control */ kc[i].index = enum_count;
template.num_kcontrols = le32_to_cpu(w->num_kcontrols); kcontrol_type[i] = SND_SOC_TPLG_TYPE_ENUM;
template.kcontrol_news = enum_count++;
soc_tplg_dapm_widget_dbytes_create(tplg, ret = soc_tplg_dapm_widget_denum_create(tplg, &kc[i]);
template.num_kcontrols); if (ret < 0)
if (!template.kcontrol_news) { goto hdr_err;
ret = -ENOMEM; break;
case SND_SOC_TPLG_CTL_BYTES:
/* bytes control */
kc[i].index = bytes_count;
kcontrol_type[i] = SND_SOC_TPLG_TYPE_BYTES;
bytes_count++;
ret = soc_tplg_dapm_widget_dbytes_create(tplg, &kc[i]);
if (ret < 0)
goto hdr_err;
break;
default:
dev_err(tplg->dev, "ASoC: invalid widget control type %d:%d:%d\n",
control_hdr->ops.get, control_hdr->ops.put,
le32_to_cpu(control_hdr->ops.info));
ret = -EINVAL;
goto hdr_err; goto hdr_err;
} }
break;
default:
dev_err(tplg->dev, "ASoC: invalid widget control type %d:%d:%d\n",
control_hdr->ops.get, control_hdr->ops.put,
le32_to_cpu(control_hdr->ops.info));
ret = -EINVAL;
goto hdr_err;
} }
template.kcontrol_news = kc;
widget: widget:
ret = soc_tplg_widget_load(tplg, &template, w); ret = soc_tplg_widget_load(tplg, &template, w);
if (ret < 0) if (ret < 0)
......
...@@ -1063,6 +1063,7 @@ static int sof_control_load_volume(struct snd_soc_component *scomp, ...@@ -1063,6 +1063,7 @@ static int sof_control_load_volume(struct snd_soc_component *scomp,
scontrol->min_volume_step = le32_to_cpu(mc->min); scontrol->min_volume_step = le32_to_cpu(mc->min);
scontrol->max_volume_step = le32_to_cpu(mc->max); scontrol->max_volume_step = le32_to_cpu(mc->max);
scontrol->num_channels = le32_to_cpu(mc->num_channels); scontrol->num_channels = le32_to_cpu(mc->num_channels);
scontrol->control_data->index = kc->index;
/* set cmd for mixer control */ /* set cmd for mixer control */
if (le32_to_cpu(mc->max) == 1) { if (le32_to_cpu(mc->max) == 1) {
...@@ -1140,7 +1141,7 @@ static int sof_control_load_enum(struct snd_soc_component *scomp, ...@@ -1140,7 +1141,7 @@ static int sof_control_load_enum(struct snd_soc_component *scomp,
scontrol->comp_id = sdev->next_comp_id; scontrol->comp_id = sdev->next_comp_id;
scontrol->num_channels = le32_to_cpu(ec->num_channels); scontrol->num_channels = le32_to_cpu(ec->num_channels);
scontrol->control_data->index = kc->index;
scontrol->cmd = SOF_CTRL_CMD_ENUM; scontrol->cmd = SOF_CTRL_CMD_ENUM;
dev_dbg(scomp->dev, "tplg: load kcontrol index %d chans %d comp_id %d\n", dev_dbg(scomp->dev, "tplg: load kcontrol index %d chans %d comp_id %d\n",
...@@ -1188,6 +1189,7 @@ static int sof_control_load_bytes(struct snd_soc_component *scomp, ...@@ -1188,6 +1189,7 @@ static int sof_control_load_bytes(struct snd_soc_component *scomp,
scontrol->comp_id = sdev->next_comp_id; scontrol->comp_id = sdev->next_comp_id;
scontrol->cmd = SOF_CTRL_CMD_BINARY; scontrol->cmd = SOF_CTRL_CMD_BINARY;
scontrol->control_data->index = kc->index;
dev_dbg(scomp->dev, "tplg: load kcontrol index %d chans %d\n", dev_dbg(scomp->dev, "tplg: load kcontrol index %d chans %d\n",
scontrol->comp_id, scontrol->num_channels); scontrol->comp_id, scontrol->num_channels);
...@@ -2133,7 +2135,7 @@ static int sof_get_control_data(struct snd_soc_component *scomp, ...@@ -2133,7 +2135,7 @@ static int sof_get_control_data(struct snd_soc_component *scomp,
for (i = 0; i < widget->num_kcontrols; i++) { for (i = 0; i < widget->num_kcontrols; i++) {
kc = &widget->kcontrol_news[i]; kc = &widget->kcontrol_news[i];
switch (widget->dobj.widget.kcontrol_type) { switch (widget->dobj.widget.kcontrol_type[i]) {
case SND_SOC_TPLG_TYPE_MIXER: case SND_SOC_TPLG_TYPE_MIXER:
sm = (struct soc_mixer_control *)kc->private_value; sm = (struct soc_mixer_control *)kc->private_value;
wdata[i].control = sm->dobj.private; wdata[i].control = sm->dobj.private;
...@@ -2147,8 +2149,8 @@ static int sof_get_control_data(struct snd_soc_component *scomp, ...@@ -2147,8 +2149,8 @@ static int sof_get_control_data(struct snd_soc_component *scomp,
wdata[i].control = se->dobj.private; wdata[i].control = se->dobj.private;
break; break;
default: default:
dev_err(scomp->dev, "error: unknown kcontrol type %d in widget %s\n", dev_err(scomp->dev, "error: unknown kcontrol type %u in widget %s\n",
widget->dobj.widget.kcontrol_type, widget->dobj.widget.kcontrol_type[i],
widget->name); widget->name);
return -EINVAL; return -EINVAL;
} }
...@@ -2164,7 +2166,8 @@ static int sof_get_control_data(struct snd_soc_component *scomp, ...@@ -2164,7 +2166,8 @@ static int sof_get_control_data(struct snd_soc_component *scomp,
return -EINVAL; return -EINVAL;
/* make sure data is valid - data can be updated at runtime */ /* make sure data is valid - data can be updated at runtime */
if (wdata[i].pdata->magic != SOF_ABI_MAGIC) if (widget->dobj.widget.kcontrol_type[i] == SND_SOC_TPLG_TYPE_BYTES &&
wdata[i].pdata->magic != SOF_ABI_MAGIC)
return -EINVAL; return -EINVAL;
*size += wdata[i].pdata->size; *size += wdata[i].pdata->size;
...@@ -2605,7 +2608,7 @@ static int sof_widget_unload(struct snd_soc_component *scomp, ...@@ -2605,7 +2608,7 @@ static int sof_widget_unload(struct snd_soc_component *scomp,
} }
for (i = 0; i < widget->num_kcontrols; i++) { for (i = 0; i < widget->num_kcontrols; i++) {
kc = &widget->kcontrol_news[i]; kc = &widget->kcontrol_news[i];
switch (dobj->widget.kcontrol_type) { switch (widget->dobj.widget.kcontrol_type[i]) {
case SND_SOC_TPLG_TYPE_MIXER: case SND_SOC_TPLG_TYPE_MIXER:
sm = (struct soc_mixer_control *)kc->private_value; sm = (struct soc_mixer_control *)kc->private_value;
scontrol = sm->dobj.private; scontrol = sm->dobj.private;
......
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