162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *  bytcht_nocodec.c - ASoc Machine driver for MinnowBoard Max and Up
462306a36Sopenharmony_ci *  to make I2S signals observable on the Low-Speed connector. Audio codec
562306a36Sopenharmony_ci *  is not managed by ASoC/DAPM
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci *  Copyright (C) 2015-2017 Intel Corp
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci *  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
1062306a36Sopenharmony_ci *
1162306a36Sopenharmony_ci * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
1262306a36Sopenharmony_ci */
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include <linux/module.h>
1562306a36Sopenharmony_ci#include <sound/pcm.h>
1662306a36Sopenharmony_ci#include <sound/pcm_params.h>
1762306a36Sopenharmony_ci#include <sound/soc.h>
1862306a36Sopenharmony_ci#include "../atom/sst-atom-controls.h"
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_cistatic const struct snd_soc_dapm_widget widgets[] = {
2162306a36Sopenharmony_ci	SND_SOC_DAPM_MIC("Mic", NULL),
2262306a36Sopenharmony_ci	SND_SOC_DAPM_SPK("Speaker", NULL),
2362306a36Sopenharmony_ci};
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_cistatic const struct snd_kcontrol_new controls[] = {
2662306a36Sopenharmony_ci	SOC_DAPM_PIN_SWITCH("Mic"),
2762306a36Sopenharmony_ci	SOC_DAPM_PIN_SWITCH("Speaker"),
2862306a36Sopenharmony_ci};
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_cistatic const struct snd_soc_dapm_route audio_map[] = {
3162306a36Sopenharmony_ci	{"ssp2 Tx", NULL, "codec_out0"},
3262306a36Sopenharmony_ci	{"ssp2 Tx", NULL, "codec_out1"},
3362306a36Sopenharmony_ci	{"codec_in0", NULL, "ssp2 Rx"},
3462306a36Sopenharmony_ci	{"codec_in1", NULL, "ssp2 Rx"},
3562306a36Sopenharmony_ci
3662306a36Sopenharmony_ci	{"ssp2 Rx", NULL, "Mic"},
3762306a36Sopenharmony_ci	{"Speaker", NULL, "ssp2 Tx"},
3862306a36Sopenharmony_ci};
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_cistatic int codec_fixup(struct snd_soc_pcm_runtime *rtd,
4162306a36Sopenharmony_ci			    struct snd_pcm_hw_params *params)
4262306a36Sopenharmony_ci{
4362306a36Sopenharmony_ci	struct snd_interval *rate = hw_param_interval(params,
4462306a36Sopenharmony_ci			SNDRV_PCM_HW_PARAM_RATE);
4562306a36Sopenharmony_ci	struct snd_interval *channels = hw_param_interval(params,
4662306a36Sopenharmony_ci						SNDRV_PCM_HW_PARAM_CHANNELS);
4762306a36Sopenharmony_ci	int ret;
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	/* The DSP will convert the FE rate to 48k, stereo, 24bits */
5062306a36Sopenharmony_ci	rate->min = rate->max = 48000;
5162306a36Sopenharmony_ci	channels->min = channels->max = 2;
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	/* set SSP2 to 24-bit */
5462306a36Sopenharmony_ci	params_set_format(params, SNDRV_PCM_FORMAT_S24_LE);
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci	/*
5762306a36Sopenharmony_ci	 * Default mode for SSP configuration is TDM 4 slot, override config
5862306a36Sopenharmony_ci	 * with explicit setting to I2S 2ch 24-bit. The word length is set with
5962306a36Sopenharmony_ci	 * dai_set_tdm_slot() since there is no other API exposed
6062306a36Sopenharmony_ci	 */
6162306a36Sopenharmony_ci	ret = snd_soc_dai_set_fmt(asoc_rtd_to_cpu(rtd, 0),
6262306a36Sopenharmony_ci				  SND_SOC_DAIFMT_I2S     |
6362306a36Sopenharmony_ci				  SND_SOC_DAIFMT_NB_NF   |
6462306a36Sopenharmony_ci				  SND_SOC_DAIFMT_BP_FP);
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci	if (ret < 0) {
6762306a36Sopenharmony_ci		dev_err(rtd->dev, "can't set format to I2S, err %d\n", ret);
6862306a36Sopenharmony_ci		return ret;
6962306a36Sopenharmony_ci	}
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	ret = snd_soc_dai_set_tdm_slot(asoc_rtd_to_cpu(rtd, 0), 0x3, 0x3, 2, 24);
7262306a36Sopenharmony_ci	if (ret < 0) {
7362306a36Sopenharmony_ci		dev_err(rtd->dev, "can't set I2S config, err %d\n", ret);
7462306a36Sopenharmony_ci		return ret;
7562306a36Sopenharmony_ci	}
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_ci	return 0;
7862306a36Sopenharmony_ci}
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_cistatic const unsigned int rates_48000[] = {
8162306a36Sopenharmony_ci	48000,
8262306a36Sopenharmony_ci};
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_cistatic const struct snd_pcm_hw_constraint_list constraints_48000 = {
8562306a36Sopenharmony_ci	.count = ARRAY_SIZE(rates_48000),
8662306a36Sopenharmony_ci	.list  = rates_48000,
8762306a36Sopenharmony_ci};
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cistatic int aif1_startup(struct snd_pcm_substream *substream)
9062306a36Sopenharmony_ci{
9162306a36Sopenharmony_ci	return snd_pcm_hw_constraint_list(substream->runtime, 0,
9262306a36Sopenharmony_ci			SNDRV_PCM_HW_PARAM_RATE,
9362306a36Sopenharmony_ci			&constraints_48000);
9462306a36Sopenharmony_ci}
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_cistatic const struct snd_soc_ops aif1_ops = {
9762306a36Sopenharmony_ci	.startup = aif1_startup,
9862306a36Sopenharmony_ci};
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_ciSND_SOC_DAILINK_DEF(dummy,
10162306a36Sopenharmony_ci	DAILINK_COMP_ARRAY(COMP_DUMMY()));
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ciSND_SOC_DAILINK_DEF(media,
10462306a36Sopenharmony_ci	DAILINK_COMP_ARRAY(COMP_CPU("media-cpu-dai")));
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ciSND_SOC_DAILINK_DEF(deepbuffer,
10762306a36Sopenharmony_ci	DAILINK_COMP_ARRAY(COMP_CPU("deepbuffer-cpu-dai")));
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_ciSND_SOC_DAILINK_DEF(ssp2_port,
11062306a36Sopenharmony_ci	DAILINK_COMP_ARRAY(COMP_CPU("ssp2-port")));
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ciSND_SOC_DAILINK_DEF(platform,
11362306a36Sopenharmony_ci	DAILINK_COMP_ARRAY(COMP_PLATFORM("sst-mfld-platform")));
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_cistatic struct snd_soc_dai_link dais[] = {
11662306a36Sopenharmony_ci	[MERR_DPCM_AUDIO] = {
11762306a36Sopenharmony_ci		.name = "Audio Port",
11862306a36Sopenharmony_ci		.stream_name = "Audio",
11962306a36Sopenharmony_ci		.ignore_suspend = 1,
12062306a36Sopenharmony_ci		.nonatomic = true,
12162306a36Sopenharmony_ci		.dynamic = 1,
12262306a36Sopenharmony_ci		.dpcm_playback = 1,
12362306a36Sopenharmony_ci		.dpcm_capture = 1,
12462306a36Sopenharmony_ci		.ops = &aif1_ops,
12562306a36Sopenharmony_ci		SND_SOC_DAILINK_REG(media, dummy, platform),
12662306a36Sopenharmony_ci	},
12762306a36Sopenharmony_ci	[MERR_DPCM_DEEP_BUFFER] = {
12862306a36Sopenharmony_ci		.name = "Deep-Buffer Audio Port",
12962306a36Sopenharmony_ci		.stream_name = "Deep-Buffer Audio",
13062306a36Sopenharmony_ci		.ignore_suspend = 1,
13162306a36Sopenharmony_ci		.nonatomic = true,
13262306a36Sopenharmony_ci		.dynamic = 1,
13362306a36Sopenharmony_ci		.dpcm_playback = 1,
13462306a36Sopenharmony_ci		.ops = &aif1_ops,
13562306a36Sopenharmony_ci		SND_SOC_DAILINK_REG(deepbuffer, dummy, platform),
13662306a36Sopenharmony_ci	},
13762306a36Sopenharmony_ci	/* CODEC<->CODEC link */
13862306a36Sopenharmony_ci	/* back ends */
13962306a36Sopenharmony_ci	{
14062306a36Sopenharmony_ci		.name = "SSP2-LowSpeed Connector",
14162306a36Sopenharmony_ci		.id = 0,
14262306a36Sopenharmony_ci		.no_pcm = 1,
14362306a36Sopenharmony_ci		.dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF
14462306a36Sopenharmony_ci						| SND_SOC_DAIFMT_CBC_CFC,
14562306a36Sopenharmony_ci		.be_hw_params_fixup = codec_fixup,
14662306a36Sopenharmony_ci		.ignore_suspend = 1,
14762306a36Sopenharmony_ci		.dpcm_playback = 1,
14862306a36Sopenharmony_ci		.dpcm_capture = 1,
14962306a36Sopenharmony_ci		SND_SOC_DAILINK_REG(ssp2_port, dummy, platform),
15062306a36Sopenharmony_ci	},
15162306a36Sopenharmony_ci};
15262306a36Sopenharmony_ci
15362306a36Sopenharmony_ci/* SoC card */
15462306a36Sopenharmony_cistatic struct snd_soc_card bytcht_nocodec_card = {
15562306a36Sopenharmony_ci	.name = "bytcht-nocodec",
15662306a36Sopenharmony_ci	.owner = THIS_MODULE,
15762306a36Sopenharmony_ci	.dai_link = dais,
15862306a36Sopenharmony_ci	.num_links = ARRAY_SIZE(dais),
15962306a36Sopenharmony_ci	.dapm_widgets = widgets,
16062306a36Sopenharmony_ci	.num_dapm_widgets = ARRAY_SIZE(widgets),
16162306a36Sopenharmony_ci	.dapm_routes = audio_map,
16262306a36Sopenharmony_ci	.num_dapm_routes = ARRAY_SIZE(audio_map),
16362306a36Sopenharmony_ci	.controls = controls,
16462306a36Sopenharmony_ci	.num_controls = ARRAY_SIZE(controls),
16562306a36Sopenharmony_ci	.fully_routed = true,
16662306a36Sopenharmony_ci};
16762306a36Sopenharmony_ci
16862306a36Sopenharmony_cistatic int snd_bytcht_nocodec_mc_probe(struct platform_device *pdev)
16962306a36Sopenharmony_ci{
17062306a36Sopenharmony_ci	int ret_val = 0;
17162306a36Sopenharmony_ci
17262306a36Sopenharmony_ci	/* register the soc card */
17362306a36Sopenharmony_ci	bytcht_nocodec_card.dev = &pdev->dev;
17462306a36Sopenharmony_ci
17562306a36Sopenharmony_ci	ret_val = devm_snd_soc_register_card(&pdev->dev, &bytcht_nocodec_card);
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ci	if (ret_val) {
17862306a36Sopenharmony_ci		dev_err(&pdev->dev, "devm_snd_soc_register_card failed %d\n",
17962306a36Sopenharmony_ci			ret_val);
18062306a36Sopenharmony_ci		return ret_val;
18162306a36Sopenharmony_ci	}
18262306a36Sopenharmony_ci	platform_set_drvdata(pdev, &bytcht_nocodec_card);
18362306a36Sopenharmony_ci	return ret_val;
18462306a36Sopenharmony_ci}
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_cistatic struct platform_driver snd_bytcht_nocodec_mc_driver = {
18762306a36Sopenharmony_ci	.driver = {
18862306a36Sopenharmony_ci		.name = "bytcht_nocodec",
18962306a36Sopenharmony_ci	},
19062306a36Sopenharmony_ci	.probe = snd_bytcht_nocodec_mc_probe,
19162306a36Sopenharmony_ci};
19262306a36Sopenharmony_cimodule_platform_driver(snd_bytcht_nocodec_mc_driver);
19362306a36Sopenharmony_ci
19462306a36Sopenharmony_ciMODULE_DESCRIPTION("ASoC Intel(R) Baytrail/Cherrytrail Nocodec Machine driver");
19562306a36Sopenharmony_ciMODULE_AUTHOR("Pierre-Louis Bossart <pierre-louis.bossart at linux.intel.com>");
19662306a36Sopenharmony_ciMODULE_LICENSE("GPL v2");
19762306a36Sopenharmony_ciMODULE_ALIAS("platform:bytcht_nocodec");
198