2015-07-29 17:45:14 +01:00
|
|
|
/*
|
|
|
|
|
Copyright(c) 2014-2015 Intel Corporation
|
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of version 2 of the GNU General Public License as
|
|
|
|
|
published by the Free Software Foundation.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful, but
|
|
|
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
General Public License for more details.
|
|
|
|
|
|
|
|
|
|
Authors: Mengdong Lin <mengdong.lin@intel.com>
|
|
|
|
|
Yao Jin <yao.jin@intel.com>
|
|
|
|
|
Liam Girdwood <liam.r.girdwood@linux.intel.com>
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "list.h"
|
|
|
|
|
#include "tplg_local.h"
|
|
|
|
|
|
|
|
|
|
int tplg_ref_add(struct tplg_elem *elem, int type, const char* id)
|
|
|
|
|
{
|
|
|
|
|
struct tplg_ref *ref;
|
|
|
|
|
|
|
|
|
|
ref = calloc(1, sizeof(*ref));
|
|
|
|
|
if (!ref)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
strncpy(ref->id, id, SNDRV_CTL_ELEM_ID_NAME_MAXLEN);
|
|
|
|
|
ref->id[SNDRV_CTL_ELEM_ID_NAME_MAXLEN - 1] = 0;
|
|
|
|
|
ref->type = type;
|
|
|
|
|
|
|
|
|
|
list_add_tail(&ref->list, &elem->ref_list);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-11 18:23:17 +01:00
|
|
|
/* directly add a reference elem */
|
|
|
|
|
int tplg_ref_add_elem(struct tplg_elem *elem, struct tplg_elem *elem_ref)
|
|
|
|
|
{
|
|
|
|
|
struct tplg_ref *ref;
|
|
|
|
|
|
|
|
|
|
ref = calloc(1, sizeof(*ref));
|
|
|
|
|
if (!ref)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
ref->type = elem_ref->type;
|
|
|
|
|
ref->elem = elem_ref;
|
|
|
|
|
elem_copy_text(ref->id, elem_ref->id, SNDRV_CTL_ELEM_ID_NAME_MAXLEN);
|
|
|
|
|
|
|
|
|
|
list_add_tail(&ref->list, &elem->ref_list);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-29 17:45:14 +01:00
|
|
|
void tplg_ref_free_list(struct list_head *base)
|
|
|
|
|
{
|
|
|
|
|
struct list_head *pos, *npos;
|
|
|
|
|
struct tplg_ref *ref;
|
|
|
|
|
|
|
|
|
|
list_for_each_safe(pos, npos, base) {
|
|
|
|
|
ref = list_entry(pos, struct tplg_ref, list);
|
|
|
|
|
list_del(&ref->list);
|
|
|
|
|
free(ref);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct tplg_elem *tplg_elem_new(void)
|
|
|
|
|
{
|
|
|
|
|
struct tplg_elem *elem;
|
|
|
|
|
|
|
|
|
|
elem = calloc(1, sizeof(*elem));
|
|
|
|
|
if (!elem)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&elem->ref_list);
|
|
|
|
|
return elem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void tplg_elem_free(struct tplg_elem *elem)
|
|
|
|
|
{
|
|
|
|
|
tplg_ref_free_list(&elem->ref_list);
|
|
|
|
|
|
|
|
|
|
/* free struct snd_tplg_ object,
|
|
|
|
|
* the union pointers share the same address
|
|
|
|
|
*/
|
2016-04-07 15:29:01 +08:00
|
|
|
if (elem->obj) {
|
|
|
|
|
if (elem->free)
|
|
|
|
|
elem->free(elem->obj);
|
|
|
|
|
|
2016-04-07 15:28:42 +08:00
|
|
|
free(elem->obj);
|
2016-04-07 15:29:01 +08:00
|
|
|
}
|
2015-07-29 17:45:14 +01:00
|
|
|
|
|
|
|
|
free(elem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void tplg_elem_free_list(struct list_head *base)
|
|
|
|
|
{
|
|
|
|
|
struct list_head *pos, *npos;
|
|
|
|
|
struct tplg_elem *elem;
|
|
|
|
|
|
|
|
|
|
list_for_each_safe(pos, npos, base) {
|
|
|
|
|
elem = list_entry(pos, struct tplg_elem, list);
|
|
|
|
|
list_del(&elem->list);
|
|
|
|
|
tplg_elem_free(elem);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct tplg_elem *tplg_elem_lookup(struct list_head *base, const char* id,
|
|
|
|
|
unsigned int type)
|
|
|
|
|
{
|
|
|
|
|
struct list_head *pos;
|
|
|
|
|
struct tplg_elem *elem;
|
|
|
|
|
|
2016-11-04 07:08:27 +08:00
|
|
|
if (!base || !id)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
2015-07-29 17:45:14 +01:00
|
|
|
list_for_each(pos, base) {
|
|
|
|
|
|
|
|
|
|
elem = list_entry(pos, struct tplg_elem, list);
|
|
|
|
|
|
|
|
|
|
if (!strcmp(elem->id, id) && elem->type == type)
|
|
|
|
|
return elem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* create a new common element and object */
|
|
|
|
|
struct tplg_elem* tplg_elem_new_common(snd_tplg_t *tplg,
|
2015-08-10 19:13:47 +01:00
|
|
|
snd_config_t *cfg, const char *name, enum snd_tplg_type type)
|
2015-07-29 17:45:14 +01:00
|
|
|
{
|
|
|
|
|
struct tplg_elem *elem;
|
|
|
|
|
const char *id;
|
|
|
|
|
int obj_size = 0;
|
|
|
|
|
void *obj;
|
|
|
|
|
|
2015-08-04 18:09:12 +01:00
|
|
|
if (!cfg && !name)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
2015-07-29 17:45:14 +01:00
|
|
|
elem = tplg_elem_new();
|
|
|
|
|
if (!elem)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
2015-08-04 18:09:12 +01:00
|
|
|
/* do we get name from cfg */
|
|
|
|
|
if (cfg) {
|
|
|
|
|
snd_config_get_id(cfg, &id);
|
|
|
|
|
elem_copy_text(elem->id, id, SNDRV_CTL_ELEM_ID_NAME_MAXLEN);
|
|
|
|
|
elem->id[SNDRV_CTL_ELEM_ID_NAME_MAXLEN - 1] = 0;
|
|
|
|
|
} else if (name != NULL)
|
|
|
|
|
elem_copy_text(elem->id, name, SNDRV_CTL_ELEM_ID_NAME_MAXLEN);
|
2015-07-29 17:45:14 +01:00
|
|
|
|
|
|
|
|
switch (type) {
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_DATA:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->pdata_list);
|
|
|
|
|
break;
|
2016-07-15 20:19:41 +08:00
|
|
|
case SND_TPLG_TYPE_MANIFEST:
|
|
|
|
|
list_add_tail(&elem->list, &tplg->manifest_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_manifest);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_TEXT:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->text_list);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_TLV:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->tlv_list);
|
|
|
|
|
elem->size = sizeof(struct snd_soc_tplg_ctl_tlv);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_BYTES:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->bytes_ext_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_bytes_control);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_ENUM:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->enum_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_enum_control);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_MIXER:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->mixer_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_mixer_control);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_DAPM_WIDGET:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->widget_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_dapm_widget);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_STREAM_CONFIG:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->pcm_config_list);
|
2015-11-05 20:48:30 +08:00
|
|
|
obj_size = sizeof(struct snd_soc_tplg_stream);
|
2015-07-29 17:45:14 +01:00
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_STREAM_CAPS:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->pcm_caps_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_stream_caps);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_PCM:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->pcm_list);
|
2015-11-05 20:48:30 +08:00
|
|
|
obj_size = sizeof(struct snd_soc_tplg_pcm);
|
2015-07-29 17:45:14 +01:00
|
|
|
break;
|
2016-11-16 14:42:33 +08:00
|
|
|
case SND_TPLG_TYPE_DAI:
|
|
|
|
|
list_add_tail(&elem->list, &tplg->dai_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_dai);
|
|
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_BE:
|
2016-11-06 13:13:46 +08:00
|
|
|
case SND_TPLG_TYPE_LINK:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->be_list);
|
2015-11-05 20:48:30 +08:00
|
|
|
obj_size = sizeof(struct snd_soc_tplg_link_config);
|
2015-07-29 17:45:14 +01:00
|
|
|
break;
|
2015-08-10 19:13:47 +01:00
|
|
|
case SND_TPLG_TYPE_CC:
|
2015-07-29 17:45:14 +01:00
|
|
|
list_add_tail(&elem->list, &tplg->cc_list);
|
2015-11-05 20:48:30 +08:00
|
|
|
obj_size = sizeof(struct snd_soc_tplg_link_config);
|
2015-07-29 17:45:14 +01:00
|
|
|
break;
|
2016-04-07 15:29:36 +08:00
|
|
|
case SND_TPLG_TYPE_TOKEN:
|
|
|
|
|
list_add_tail(&elem->list, &tplg->token_list);
|
|
|
|
|
break;
|
2016-04-07 15:29:43 +08:00
|
|
|
case SND_TPLG_TYPE_TUPLE:
|
|
|
|
|
list_add_tail(&elem->list, &tplg->tuple_list);
|
|
|
|
|
elem->free = tplg_free_tuples;
|
|
|
|
|
break;
|
2016-11-06 13:14:04 +08:00
|
|
|
case SND_TPLG_TYPE_HW_CONFIG:
|
|
|
|
|
list_add_tail(&elem->list, &tplg->hw_cfg_list);
|
|
|
|
|
obj_size = sizeof(struct snd_soc_tplg_hw_config);
|
|
|
|
|
break;
|
2015-07-29 17:45:14 +01:00
|
|
|
default:
|
|
|
|
|
free(elem);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* create new object too if required */
|
|
|
|
|
if (obj_size > 0) {
|
|
|
|
|
obj = calloc(1, obj_size);
|
|
|
|
|
if (obj == NULL) {
|
|
|
|
|
free(elem);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elem->obj = obj;
|
|
|
|
|
elem->size = obj_size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elem->type = type;
|
|
|
|
|
return elem;
|
|
|
|
|
}
|