162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Driver for generic Bluetooth SCO link
462306a36Sopenharmony_ci * Copyright 2011 Lars-Peter Clausen <lars@metafoo.de>
562306a36Sopenharmony_ci */
662306a36Sopenharmony_ci
762306a36Sopenharmony_ci#include <linux/init.h>
862306a36Sopenharmony_ci#include <linux/module.h>
962306a36Sopenharmony_ci#include <linux/platform_device.h>
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_ci#include <sound/soc.h>
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_cistatic const struct snd_soc_dapm_widget bt_sco_widgets[] = {
1462306a36Sopenharmony_ci	SND_SOC_DAPM_INPUT("RX"),
1562306a36Sopenharmony_ci	SND_SOC_DAPM_OUTPUT("TX"),
1662306a36Sopenharmony_ci	SND_SOC_DAPM_AIF_IN("BT_SCO_RX", "Playback", 0,
1762306a36Sopenharmony_ci			    SND_SOC_NOPM, 0, 0),
1862306a36Sopenharmony_ci	SND_SOC_DAPM_AIF_OUT("BT_SCO_TX", "Capture", 0,
1962306a36Sopenharmony_ci			     SND_SOC_NOPM, 0, 0),
2062306a36Sopenharmony_ci};
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic const struct snd_soc_dapm_route bt_sco_routes[] = {
2362306a36Sopenharmony_ci	{ "BT_SCO_TX", NULL, "RX" },
2462306a36Sopenharmony_ci	{ "TX", NULL, "BT_SCO_RX" },
2562306a36Sopenharmony_ci};
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_cistatic struct snd_soc_dai_driver bt_sco_dai[] = {
2862306a36Sopenharmony_ci	{
2962306a36Sopenharmony_ci		.name = "bt-sco-pcm",
3062306a36Sopenharmony_ci		.playback = {
3162306a36Sopenharmony_ci			.stream_name = "Playback",
3262306a36Sopenharmony_ci			.channels_min = 1,
3362306a36Sopenharmony_ci			.channels_max = 1,
3462306a36Sopenharmony_ci			.rates = SNDRV_PCM_RATE_8000,
3562306a36Sopenharmony_ci			.formats = SNDRV_PCM_FMTBIT_S16_LE,
3662306a36Sopenharmony_ci		},
3762306a36Sopenharmony_ci		.capture = {
3862306a36Sopenharmony_ci			 .stream_name = "Capture",
3962306a36Sopenharmony_ci			.channels_min = 1,
4062306a36Sopenharmony_ci			.channels_max = 1,
4162306a36Sopenharmony_ci			.rates = SNDRV_PCM_RATE_8000,
4262306a36Sopenharmony_ci			.formats = SNDRV_PCM_FMTBIT_S16_LE,
4362306a36Sopenharmony_ci		},
4462306a36Sopenharmony_ci	},
4562306a36Sopenharmony_ci	{
4662306a36Sopenharmony_ci		.name = "bt-sco-pcm-wb",
4762306a36Sopenharmony_ci		.playback = {
4862306a36Sopenharmony_ci			.stream_name = "Playback",
4962306a36Sopenharmony_ci			.channels_min = 1,
5062306a36Sopenharmony_ci			.channels_max = 1,
5162306a36Sopenharmony_ci			.rates = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000,
5262306a36Sopenharmony_ci			.formats = SNDRV_PCM_FMTBIT_S16_LE,
5362306a36Sopenharmony_ci		},
5462306a36Sopenharmony_ci		.capture = {
5562306a36Sopenharmony_ci			 .stream_name = "Capture",
5662306a36Sopenharmony_ci			.channels_min = 1,
5762306a36Sopenharmony_ci			.channels_max = 1,
5862306a36Sopenharmony_ci			.rates = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000,
5962306a36Sopenharmony_ci			.formats = SNDRV_PCM_FMTBIT_S16_LE,
6062306a36Sopenharmony_ci		},
6162306a36Sopenharmony_ci	}
6262306a36Sopenharmony_ci};
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_cistatic const struct snd_soc_component_driver soc_component_dev_bt_sco = {
6562306a36Sopenharmony_ci	.dapm_widgets		= bt_sco_widgets,
6662306a36Sopenharmony_ci	.num_dapm_widgets	= ARRAY_SIZE(bt_sco_widgets),
6762306a36Sopenharmony_ci	.dapm_routes		= bt_sco_routes,
6862306a36Sopenharmony_ci	.num_dapm_routes	= ARRAY_SIZE(bt_sco_routes),
6962306a36Sopenharmony_ci	.idle_bias_on		= 1,
7062306a36Sopenharmony_ci	.use_pmdown_time	= 1,
7162306a36Sopenharmony_ci	.endianness		= 1,
7262306a36Sopenharmony_ci};
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_cistatic int bt_sco_probe(struct platform_device *pdev)
7562306a36Sopenharmony_ci{
7662306a36Sopenharmony_ci	return devm_snd_soc_register_component(&pdev->dev,
7762306a36Sopenharmony_ci				      &soc_component_dev_bt_sco,
7862306a36Sopenharmony_ci				      bt_sco_dai, ARRAY_SIZE(bt_sco_dai));
7962306a36Sopenharmony_ci}
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_cistatic const struct platform_device_id bt_sco_driver_ids[] = {
8262306a36Sopenharmony_ci	{
8362306a36Sopenharmony_ci		.name		= "dfbmcs320",
8462306a36Sopenharmony_ci	},
8562306a36Sopenharmony_ci	{
8662306a36Sopenharmony_ci		.name		= "bt-sco",
8762306a36Sopenharmony_ci	},
8862306a36Sopenharmony_ci	{},
8962306a36Sopenharmony_ci};
9062306a36Sopenharmony_ciMODULE_DEVICE_TABLE(platform, bt_sco_driver_ids);
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci#if defined(CONFIG_OF)
9362306a36Sopenharmony_cistatic const struct of_device_id bt_sco_codec_of_match[] = {
9462306a36Sopenharmony_ci	{ .compatible = "delta,dfbmcs320", },
9562306a36Sopenharmony_ci	{ .compatible = "linux,bt-sco", },
9662306a36Sopenharmony_ci	{},
9762306a36Sopenharmony_ci};
9862306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, bt_sco_codec_of_match);
9962306a36Sopenharmony_ci#endif
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_cistatic struct platform_driver bt_sco_driver = {
10262306a36Sopenharmony_ci	.driver = {
10362306a36Sopenharmony_ci		.name = "bt-sco",
10462306a36Sopenharmony_ci		.of_match_table = of_match_ptr(bt_sco_codec_of_match),
10562306a36Sopenharmony_ci	},
10662306a36Sopenharmony_ci	.probe = bt_sco_probe,
10762306a36Sopenharmony_ci	.id_table = bt_sco_driver_ids,
10862306a36Sopenharmony_ci};
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_cimodule_platform_driver(bt_sco_driver);
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ciMODULE_AUTHOR("Lars-Peter Clausen <lars@metafoo.de>");
11362306a36Sopenharmony_ciMODULE_DESCRIPTION("ASoC generic bluetooth sco link driver");
11462306a36Sopenharmony_ciMODULE_LICENSE("GPL");
115