rockchip_rt5645.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Rockchip machine ASoC driver for boards using a RT5645/RT5650 CODEC.
  4. *
  5. * Copyright (c) 2015, ROCKCHIP CORPORATION. All rights reserved.
  6. */
  7. #include <linux/module.h>
  8. #include <linux/platform_device.h>
  9. #include <linux/slab.h>
  10. #include <linux/gpio.h>
  11. #include <linux/of_gpio.h>
  12. #include <linux/delay.h>
  13. #include <sound/core.h>
  14. #include <sound/jack.h>
  15. #include <sound/pcm.h>
  16. #include <sound/pcm_params.h>
  17. #include <sound/soc.h>
  18. #include "rockchip_i2s.h"
  19. #include "../codecs/rt5645.h"
  20. #define DRV_NAME "rockchip-snd-rt5645"
  21. static struct snd_soc_jack headset_jack;
  22. static const struct snd_soc_dapm_widget rk_dapm_widgets[] = {
  23. SND_SOC_DAPM_HP("Headphones", NULL),
  24. SND_SOC_DAPM_SPK("Speakers", NULL),
  25. SND_SOC_DAPM_MIC("Headset Mic", NULL),
  26. SND_SOC_DAPM_MIC("Int Mic", NULL),
  27. };
  28. static const struct snd_soc_dapm_route rk_audio_map[] = {
  29. /* Input Lines */
  30. {"DMIC L2", NULL, "Int Mic"},
  31. {"DMIC R2", NULL, "Int Mic"},
  32. {"RECMIXL", NULL, "Headset Mic"},
  33. {"RECMIXR", NULL, "Headset Mic"},
  34. /* Output Lines */
  35. {"Headphones", NULL, "HPOR"},
  36. {"Headphones", NULL, "HPOL"},
  37. {"Speakers", NULL, "SPOL"},
  38. {"Speakers", NULL, "SPOR"},
  39. };
  40. static const struct snd_kcontrol_new rk_mc_controls[] = {
  41. SOC_DAPM_PIN_SWITCH("Headphones"),
  42. SOC_DAPM_PIN_SWITCH("Speakers"),
  43. SOC_DAPM_PIN_SWITCH("Headset Mic"),
  44. SOC_DAPM_PIN_SWITCH("Int Mic"),
  45. };
  46. static int rk_aif1_hw_params(struct snd_pcm_substream *substream,
  47. struct snd_pcm_hw_params *params)
  48. {
  49. int ret = 0;
  50. struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
  51. struct snd_soc_dai *cpu_dai = asoc_rtd_to_cpu(rtd, 0);
  52. struct snd_soc_dai *codec_dai = asoc_rtd_to_codec(rtd, 0);
  53. int mclk;
  54. switch (params_rate(params)) {
  55. case 8000:
  56. case 16000:
  57. case 24000:
  58. case 32000:
  59. case 48000:
  60. case 64000:
  61. case 96000:
  62. mclk = 12288000;
  63. break;
  64. case 11025:
  65. case 22050:
  66. case 44100:
  67. case 88200:
  68. mclk = 11289600;
  69. break;
  70. default:
  71. return -EINVAL;
  72. }
  73. ret = snd_soc_dai_set_sysclk(cpu_dai, 0, mclk,
  74. SND_SOC_CLOCK_OUT);
  75. if (ret < 0) {
  76. dev_err(codec_dai->dev, "Can't set codec clock %d\n", ret);
  77. return ret;
  78. }
  79. ret = snd_soc_dai_set_sysclk(codec_dai, 0, mclk,
  80. SND_SOC_CLOCK_IN);
  81. if (ret < 0) {
  82. dev_err(codec_dai->dev, "Can't set codec clock %d\n", ret);
  83. return ret;
  84. }
  85. return ret;
  86. }
  87. static int rk_init(struct snd_soc_pcm_runtime *runtime)
  88. {
  89. struct snd_soc_card *card = runtime->card;
  90. int ret;
  91. /* Enable Headset and 4 Buttons Jack detection */
  92. ret = snd_soc_card_jack_new(card, "Headset Jack",
  93. SND_JACK_HEADPHONE | SND_JACK_MICROPHONE |
  94. SND_JACK_BTN_0 | SND_JACK_BTN_1 |
  95. SND_JACK_BTN_2 | SND_JACK_BTN_3,
  96. &headset_jack);
  97. if (ret) {
  98. dev_err(card->dev, "New Headset Jack failed! (%d)\n", ret);
  99. return ret;
  100. }
  101. return rt5645_set_jack_detect(asoc_rtd_to_codec(runtime, 0)->component,
  102. &headset_jack,
  103. &headset_jack,
  104. &headset_jack);
  105. }
  106. static const struct snd_soc_ops rk_aif1_ops = {
  107. .hw_params = rk_aif1_hw_params,
  108. };
  109. SND_SOC_DAILINK_DEFS(pcm,
  110. DAILINK_COMP_ARRAY(COMP_EMPTY()),
  111. DAILINK_COMP_ARRAY(COMP_CODEC(NULL, "rt5645-aif1")),
  112. DAILINK_COMP_ARRAY(COMP_EMPTY()));
  113. static struct snd_soc_dai_link rk_dailink = {
  114. .name = "rt5645",
  115. .stream_name = "rt5645 PCM",
  116. .init = rk_init,
  117. .ops = &rk_aif1_ops,
  118. /* set rt5645 as slave */
  119. .dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
  120. SND_SOC_DAIFMT_CBS_CFS,
  121. SND_SOC_DAILINK_REG(pcm),
  122. };
  123. static struct snd_soc_card snd_soc_card_rk = {
  124. .name = "I2S-RT5650",
  125. .owner = THIS_MODULE,
  126. .dai_link = &rk_dailink,
  127. .num_links = 1,
  128. .dapm_widgets = rk_dapm_widgets,
  129. .num_dapm_widgets = ARRAY_SIZE(rk_dapm_widgets),
  130. .dapm_routes = rk_audio_map,
  131. .num_dapm_routes = ARRAY_SIZE(rk_audio_map),
  132. .controls = rk_mc_controls,
  133. .num_controls = ARRAY_SIZE(rk_mc_controls),
  134. };
  135. static int snd_rk_mc_probe(struct platform_device *pdev)
  136. {
  137. int ret = 0;
  138. struct snd_soc_card *card = &snd_soc_card_rk;
  139. struct device_node *np = pdev->dev.of_node;
  140. /* register the soc card */
  141. card->dev = &pdev->dev;
  142. rk_dailink.codecs->of_node = of_parse_phandle(np,
  143. "rockchip,audio-codec", 0);
  144. if (!rk_dailink.codecs->of_node) {
  145. dev_err(&pdev->dev,
  146. "Property 'rockchip,audio-codec' missing or invalid\n");
  147. return -EINVAL;
  148. }
  149. rk_dailink.cpus->of_node = of_parse_phandle(np,
  150. "rockchip,i2s-controller", 0);
  151. if (!rk_dailink.cpus->of_node) {
  152. dev_err(&pdev->dev,
  153. "Property 'rockchip,i2s-controller' missing or invalid\n");
  154. ret = -EINVAL;
  155. goto put_codec_of_node;
  156. }
  157. rk_dailink.platforms->of_node = rk_dailink.cpus->of_node;
  158. ret = snd_soc_of_parse_card_name(card, "rockchip,model");
  159. if (ret) {
  160. dev_err(&pdev->dev,
  161. "Soc parse card name failed %d\n", ret);
  162. goto put_cpu_of_node;
  163. }
  164. ret = devm_snd_soc_register_card(&pdev->dev, card);
  165. if (ret) {
  166. dev_err(&pdev->dev,
  167. "Soc register card failed %d\n", ret);
  168. goto put_cpu_of_node;
  169. }
  170. return ret;
  171. put_cpu_of_node:
  172. of_node_put(rk_dailink.cpus->of_node);
  173. rk_dailink.cpus->of_node = NULL;
  174. put_codec_of_node:
  175. of_node_put(rk_dailink.codecs->of_node);
  176. rk_dailink.codecs->of_node = NULL;
  177. return ret;
  178. }
  179. static int snd_rk_mc_remove(struct platform_device *pdev)
  180. {
  181. of_node_put(rk_dailink.cpus->of_node);
  182. rk_dailink.cpus->of_node = NULL;
  183. of_node_put(rk_dailink.codecs->of_node);
  184. rk_dailink.codecs->of_node = NULL;
  185. return 0;
  186. }
  187. static const struct of_device_id rockchip_rt5645_of_match[] = {
  188. { .compatible = "rockchip,rockchip-audio-rt5645", },
  189. {},
  190. };
  191. MODULE_DEVICE_TABLE(of, rockchip_rt5645_of_match);
  192. static struct platform_driver snd_rk_mc_driver = {
  193. .probe = snd_rk_mc_probe,
  194. .remove = snd_rk_mc_remove,
  195. .driver = {
  196. .name = DRV_NAME,
  197. .pm = &snd_soc_pm_ops,
  198. .of_match_table = rockchip_rt5645_of_match,
  199. },
  200. };
  201. module_platform_driver(snd_rk_mc_driver);
  202. MODULE_AUTHOR("Xing Zheng <[email protected]>");
  203. MODULE_DESCRIPTION("Rockchip rt5645 machine ASoC driver");
  204. MODULE_LICENSE("GPL v2");
  205. MODULE_ALIAS("platform:" DRV_NAME);