Commit 07140abb authored by Bard Liao's avatar Bard Liao Committed by Mark Brown

ASoC: Intel: sof_sdw: add dai info

The existing code create a dailink for a codec. However, we may need
multi dailinks for a codec. This commit adds a new struct in
sof_sdw_codec_info{} to store the dai info of a codec.
The initial assumption if that we will create at most 3 dailink types
for a codec, since this is the max known with upcoming SDCA devices. We
may need to increase this number as new SDCA 'functions' become available.

One strong assumption is that all dailinks exposed are independent, as per
SDCA directions.

This commit just moves some items into the new sof_sdw_dai_info struct.
There is no function changed. Multi dais supported will be added in the
follow up commits.
Reviewed-by: default avatarRanjani Sridharan <ranjani.sridharan@linux.intel.com>
Signed-off-by: default avatarBard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: default avatarPierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20230602202225.249209-3-pierre-louis.bossart@linux.intel.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent ba032909
...@@ -563,134 +563,231 @@ static const struct snd_soc_ops sdw_ops = { ...@@ -563,134 +563,231 @@ static const struct snd_soc_ops sdw_ops = {
static struct sof_sdw_codec_info codec_info_list[] = { static struct sof_sdw_codec_info codec_info_list[] = {
{ {
.part_id = 0x700, .part_id = 0x700,
.direction = {true, true}, .dais = {
.dai_name = "rt700-aif1", {
.init = sof_sdw_rt700_init, .direction = {true, true},
.dai_name = "rt700-aif1",
.dai_type = SOF_SDW_DAI_TYPE_JACK,
.init = sof_sdw_rt700_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_JACK, .codec_type = SOF_SDW_CODEC_TYPE_JACK,
}, },
{ {
.part_id = 0x711, .part_id = 0x711,
.version_id = 3, .version_id = 3,
.direction = {true, true}, .dais = {
.dai_name = "rt711-sdca-aif1", {
.init = sof_sdw_rt711_sdca_init, .direction = {true, true},
.exit = sof_sdw_rt711_sdca_exit, .dai_name = "rt711-sdca-aif1",
.dai_type = SOF_SDW_DAI_TYPE_JACK,
.init = sof_sdw_rt711_sdca_init,
.exit = sof_sdw_rt711_sdca_exit,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_JACK, .codec_type = SOF_SDW_CODEC_TYPE_JACK,
}, },
{ {
.part_id = 0x711, .part_id = 0x711,
.version_id = 2, .version_id = 2,
.direction = {true, true}, .dais = {
.dai_name = "rt711-aif1", {
.init = sof_sdw_rt711_init, .direction = {true, true},
.exit = sof_sdw_rt711_exit, .dai_name = "rt711-aif1",
.dai_type = SOF_SDW_DAI_TYPE_JACK,
.init = sof_sdw_rt711_init,
.exit = sof_sdw_rt711_exit,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_JACK, .codec_type = SOF_SDW_CODEC_TYPE_JACK,
}, },
{ {
.part_id = 0x1308, .part_id = 0x1308,
.acpi_id = "10EC1308", .acpi_id = "10EC1308",
.direction = {true, false}, .dais = {
.dai_name = "rt1308-aif", {
.direction = {true, false},
.dai_name = "rt1308-aif",
.dai_type = SOF_SDW_DAI_TYPE_AMP,
.init = sof_sdw_rt_amp_init,
.exit = sof_sdw_rt_amp_exit,
},
},
.dai_num = 1,
.ops = &sof_sdw_rt1308_i2s_ops, .ops = &sof_sdw_rt1308_i2s_ops,
.init = sof_sdw_rt_amp_init,
.exit = sof_sdw_rt_amp_exit,
.codec_type = SOF_SDW_CODEC_TYPE_AMP, .codec_type = SOF_SDW_CODEC_TYPE_AMP,
}, },
{ {
.part_id = 0x1316, .part_id = 0x1316,
.direction = {true, true}, .dais = {
.dai_name = "rt1316-aif", {
.init = sof_sdw_rt_amp_init, .direction = {true, true},
.exit = sof_sdw_rt_amp_exit, .dai_name = "rt1316-aif",
.dai_type = SOF_SDW_DAI_TYPE_AMP,
.init = sof_sdw_rt_amp_init,
.exit = sof_sdw_rt_amp_exit,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_AMP, .codec_type = SOF_SDW_CODEC_TYPE_AMP,
}, },
{ {
.part_id = 0x1318, .part_id = 0x1318,
.direction = {true, true}, .dais = {
.dai_name = "rt1318-aif", {
.init = sof_sdw_rt_amp_init, .direction = {true, true},
.exit = sof_sdw_rt_amp_exit, .dai_name = "rt1318-aif",
.dai_type = SOF_SDW_DAI_TYPE_AMP,
.init = sof_sdw_rt_amp_init,
.exit = sof_sdw_rt_amp_exit,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_AMP, .codec_type = SOF_SDW_CODEC_TYPE_AMP,
}, },
{ {
.part_id = 0x714, .part_id = 0x714,
.version_id = 3, .version_id = 3,
.direction = {false, true},
.ignore_pch_dmic = true, .ignore_pch_dmic = true,
.dai_name = "rt715-aif2", .dais = {
.init = sof_sdw_rt715_sdca_init, {
.direction = {false, true},
.dai_name = "rt715-aif2",
.dai_type = SOF_SDW_DAI_TYPE_MIC,
.init = sof_sdw_rt715_sdca_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_MIC, .codec_type = SOF_SDW_CODEC_TYPE_MIC,
}, },
{ {
.part_id = 0x715, .part_id = 0x715,
.version_id = 3, .version_id = 3,
.direction = {false, true},
.ignore_pch_dmic = true, .ignore_pch_dmic = true,
.dai_name = "rt715-aif2", .dais = {
.init = sof_sdw_rt715_sdca_init, {
.direction = {false, true},
.dai_name = "rt715-aif2",
.dai_type = SOF_SDW_DAI_TYPE_MIC,
.init = sof_sdw_rt715_sdca_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_MIC, .codec_type = SOF_SDW_CODEC_TYPE_MIC,
}, },
{ {
.part_id = 0x714, .part_id = 0x714,
.version_id = 2, .version_id = 2,
.direction = {false, true},
.ignore_pch_dmic = true, .ignore_pch_dmic = true,
.dai_name = "rt715-aif2", .dais = {
.init = sof_sdw_rt715_init, {
.direction = {false, true},
.dai_name = "rt715-aif2",
.dai_type = SOF_SDW_DAI_TYPE_MIC,
.init = sof_sdw_rt715_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_MIC, .codec_type = SOF_SDW_CODEC_TYPE_MIC,
}, },
{ {
.part_id = 0x715, .part_id = 0x715,
.version_id = 2, .version_id = 2,
.direction = {false, true},
.ignore_pch_dmic = true, .ignore_pch_dmic = true,
.dai_name = "rt715-aif2", .dais = {
.init = sof_sdw_rt715_init, {
.direction = {false, true},
.dai_name = "rt715-aif2",
.dai_type = SOF_SDW_DAI_TYPE_MIC,
.init = sof_sdw_rt715_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_MIC, .codec_type = SOF_SDW_CODEC_TYPE_MIC,
}, },
{ {
.part_id = 0x8373, .part_id = 0x8373,
.direction = {true, true}, .dais = {
.dai_name = "max98373-aif1", {
.init = sof_sdw_mx8373_init, .direction = {true, true},
.dai_name = "max98373-aif1",
.dai_type = SOF_SDW_DAI_TYPE_AMP,
.init = sof_sdw_mx8373_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_AMP, .codec_type = SOF_SDW_CODEC_TYPE_AMP,
}, },
{ {
.part_id = 0x5682, .part_id = 0x5682,
.direction = {true, true}, .dais = {
.dai_name = "rt5682-sdw", {
.init = sof_sdw_rt5682_init, .direction = {true, true},
.dai_name = "rt5682-sdw",
.dai_type = SOF_SDW_DAI_TYPE_JACK,
.init = sof_sdw_rt5682_init,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_JACK, .codec_type = SOF_SDW_CODEC_TYPE_JACK,
}, },
{ {
.part_id = 0xaaaa, /* generic codec mockup */ .part_id = 0xaaaa, /* generic codec mockup */
.version_id = 0, .version_id = 0,
.direction = {true, true}, .dais = {
.dai_name = "sdw-mockup-aif1", {
.init = NULL, .direction = {true, true},
.dai_name = "sdw-mockup-aif1",
.dai_type = SOF_SDW_DAI_TYPE_JACK,
.init = NULL,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_JACK, .codec_type = SOF_SDW_CODEC_TYPE_JACK,
}, },
{ {
.part_id = 0xaa55, /* headset codec mockup */ .part_id = 0xaa55, /* headset codec mockup */
.version_id = 0, .version_id = 0,
.direction = {true, true}, .dais = {
.dai_name = "sdw-mockup-aif1", {
.init = NULL, .direction = {true, true},
.dai_name = "sdw-mockup-aif1",
.dai_type = SOF_SDW_DAI_TYPE_JACK,
.init = NULL,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_JACK, .codec_type = SOF_SDW_CODEC_TYPE_JACK,
}, },
{ {
.part_id = 0x55aa, /* amplifier mockup */ .part_id = 0x55aa, /* amplifier mockup */
.version_id = 0, .version_id = 0,
.direction = {true, false}, .dais = {
.dai_name = "sdw-mockup-aif1", {
.init = NULL, .direction = {true, false},
.dai_name = "sdw-mockup-aif1",
.dai_type = SOF_SDW_DAI_TYPE_AMP,
.init = NULL,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_AMP, .codec_type = SOF_SDW_CODEC_TYPE_AMP,
}, },
{ {
.part_id = 0x5555, .part_id = 0x5555,
.version_id = 0, .version_id = 0,
.direction = {false, true}, .dais = {
.dai_name = "sdw-mockup-aif1", {
.dai_name = "sdw-mockup-aif1",
.direction = {false, true},
.dai_type = SOF_SDW_DAI_TYPE_MIC,
.init = NULL,
},
},
.dai_num = 1,
.codec_type = SOF_SDW_CODEC_TYPE_MIC, .codec_type = SOF_SDW_CODEC_TYPE_MIC,
}, },
}; };
...@@ -780,7 +877,7 @@ static int get_sdw_dailink_info(struct device *dev, const struct snd_soc_acpi_li ...@@ -780,7 +877,7 @@ static int get_sdw_dailink_info(struct device *dev, const struct snd_soc_acpi_li
/* count DAI number for playback and capture */ /* count DAI number for playback and capture */
for_each_pcm_streams(stream) { for_each_pcm_streams(stream) {
if (!codec_info_list[codec_index].direction[stream]) if (!codec_info_list[codec_index].dais[0].direction[stream])
continue; continue;
(*sdw_cpu_dai_num)++; (*sdw_cpu_dai_num)++;
...@@ -920,7 +1017,7 @@ static int create_codec_dai_name(struct device *dev, ...@@ -920,7 +1017,7 @@ static int create_codec_dai_name(struct device *dev,
_codec_index = codec_index; _codec_index = codec_index;
codec[comp_index].dai_name = codec[comp_index].dai_name =
codec_info_list[codec_index].dai_name; codec_info_list[codec_index].dais[0].dai_name;
codec_conf[*codec_conf_index].dlc = codec[comp_index]; codec_conf[*codec_conf_index].dlc = codec[comp_index];
codec_conf[*codec_conf_index].name_prefix = link->adr_d[i].name_prefix; codec_conf[*codec_conf_index].name_prefix = link->adr_d[i].name_prefix;
...@@ -957,8 +1054,8 @@ static int set_codec_init_func(struct snd_soc_card *card, ...@@ -957,8 +1054,8 @@ static int set_codec_init_func(struct snd_soc_card *card,
/* The group_id is > 0 iff the codec is aggregated */ /* The group_id is > 0 iff the codec is aggregated */
if (link->adr_d[i].endpoints->group_id != group_id) if (link->adr_d[i].endpoints->group_id != group_id)
continue; continue;
if (codec_info_list[codec_index].init) if (codec_info_list[codec_index].dais[0].init)
codec_info_list[codec_index].init(card, codec_info_list[codec_index].dais[0].init(card,
link, link,
dai_links, dai_links,
&codec_info_list[codec_index], &codec_info_list[codec_index],
...@@ -1154,7 +1251,7 @@ static int create_sdw_dailink(struct snd_soc_card *card, ...@@ -1154,7 +1251,7 @@ static int create_sdw_dailink(struct snd_soc_card *card,
"SDW%d-Capture-%s", "SDW%d-Capture-%s",
}; };
if (!codec_info_list[codec_index].direction[stream]) if (!codec_info_list[codec_index].dais[0].direction[stream])
continue; continue;
/* create stream name according to first link id */ /* create stream name according to first link id */
...@@ -1458,18 +1555,18 @@ static int sof_card_dai_links_create(struct device *dev, ...@@ -1458,18 +1555,18 @@ static int sof_card_dai_links_create(struct device *dev,
return -ENOMEM; return -ENOMEM;
ssp_components->name = codec_name; ssp_components->name = codec_name;
ssp_components->dai_name = info->dai_name; ssp_components->dai_name = info->dais[0].dai_name;
cpus[cpu_id].dai_name = cpu_name; cpus[cpu_id].dai_name = cpu_name;
playback = info->direction[SNDRV_PCM_STREAM_PLAYBACK]; playback = info->dais[0].direction[SNDRV_PCM_STREAM_PLAYBACK];
capture = info->direction[SNDRV_PCM_STREAM_CAPTURE]; capture = info->dais[0].direction[SNDRV_PCM_STREAM_CAPTURE];
init_dai_link(dev, links + link_index, be_id, name, init_dai_link(dev, links + link_index, be_id, name,
playback, capture, playback, capture,
cpus + cpu_id, 1, cpus + cpu_id, 1,
ssp_components, 1, ssp_components, 1,
NULL, info->ops); NULL, info->ops);
ret = info->init(card, NULL, links + link_index, info, 0); ret = info->dais[0].init(card, NULL, links + link_index, info, 0);
if (ret < 0) if (ret < 0)
return ret; return ret;
...@@ -1606,7 +1703,7 @@ static void mc_dailink_exit_loop(struct snd_soc_card *card) ...@@ -1606,7 +1703,7 @@ static void mc_dailink_exit_loop(struct snd_soc_card *card)
int i, j; int i, j;
for (i = 0; i < ARRAY_SIZE(codec_info_list); i++) { for (i = 0; i < ARRAY_SIZE(codec_info_list); i++) {
if (!codec_info_list[i].exit) if (!codec_info_list[i].dais[0].exit)
continue; continue;
/* /*
* We don't need to call .exit function if there is no matched * We don't need to call .exit function if there is no matched
...@@ -1614,8 +1711,8 @@ static void mc_dailink_exit_loop(struct snd_soc_card *card) ...@@ -1614,8 +1711,8 @@ static void mc_dailink_exit_loop(struct snd_soc_card *card)
*/ */
for_each_card_prelinks(card, j, link) { for_each_card_prelinks(card, j, link) {
if (!strcmp(link->codecs[0].dai_name, if (!strcmp(link->codecs[0].dai_name,
codec_info_list[i].dai_name)) { codec_info_list[i].dais[0].dai_name)) {
ret = codec_info_list[i].exit(card, link); ret = codec_info_list[i].dais[0].exit(card, link);
if (ret) if (ret)
dev_warn(card->dev, dev_warn(card->dev,
"codec exit failed %d\n", "codec exit failed %d\n",
......
...@@ -56,24 +56,37 @@ enum { ...@@ -56,24 +56,37 @@ enum {
#define SOF_SDW_CODEC_TYPE_AMP 1 #define SOF_SDW_CODEC_TYPE_AMP 1
#define SOF_SDW_CODEC_TYPE_MIC 2 #define SOF_SDW_CODEC_TYPE_MIC 2
#define SOF_SDW_DAI_TYPE_JACK 0
#define SOF_SDW_DAI_TYPE_AMP 1
#define SOF_SDW_DAI_TYPE_MIC 2
#define SOF_SDW_MAX_DAI_NUM 3
struct sof_sdw_codec_info;
struct sof_sdw_dai_info {
const bool direction[2]; /* playback & capture support */
const char *dai_name;
const int dai_type;
int (*init)(struct snd_soc_card *card,
const struct snd_soc_acpi_link_adr *link,
struct snd_soc_dai_link *dai_links,
struct sof_sdw_codec_info *info,
bool playback);
int (*exit)(struct snd_soc_card *card, struct snd_soc_dai_link *dai_link);
};
struct sof_sdw_codec_info { struct sof_sdw_codec_info {
const int part_id; const int part_id;
const int version_id; const int version_id;
const int codec_type; const int codec_type;
int amp_num; int amp_num;
const u8 acpi_id[ACPI_ID_LEN]; const u8 acpi_id[ACPI_ID_LEN];
const bool direction[2]; // playback & capture support
const bool ignore_pch_dmic; const bool ignore_pch_dmic;
const char *dai_name;
const struct snd_soc_ops *ops; const struct snd_soc_ops *ops;
struct sof_sdw_dai_info dais[SOF_SDW_MAX_DAI_NUM];
const int dai_num;
int (*init)(struct snd_soc_card *card,
const struct snd_soc_acpi_link_adr *link,
struct snd_soc_dai_link *dai_links,
struct sof_sdw_codec_info *info,
bool playback);
int (*exit)(struct snd_soc_card *card, struct snd_soc_dai_link *dai_link);
int (*codec_card_late_probe)(struct snd_soc_card *card); int (*codec_card_late_probe)(struct snd_soc_card *card);
}; };
......
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