spitz.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  1. /*
  2. * spitz.c -- SoC audio for Sharp SL-Cxx00 models Spitz, Borzoi and Akita
  3. *
  4. * Copyright 2005 Wolfson Microelectronics PLC.
  5. * Copyright 2005 Openedhand Ltd.
  6. *
  7. * Authors: Liam Girdwood <liam.girdwood@wolfsonmicro.com>
  8. * Richard Purdie <richard@openedhand.com>
  9. *
  10. * This program is free software; you can redistribute it and/or modify it
  11. * under the terms of the GNU General Public License as published by the
  12. * Free Software Foundation; either version 2 of the License, or (at your
  13. * option) any later version.
  14. *
  15. */
  16. #include <linux/module.h>
  17. #include <linux/moduleparam.h>
  18. #include <linux/timer.h>
  19. #include <linux/interrupt.h>
  20. #include <linux/platform_device.h>
  21. #include <sound/core.h>
  22. #include <sound/pcm.h>
  23. #include <sound/soc.h>
  24. #include <sound/soc-dapm.h>
  25. #include <asm/mach-types.h>
  26. #include <asm/hardware/scoop.h>
  27. #include <asm/arch/pxa-regs.h>
  28. #include <asm/arch/hardware.h>
  29. #include <asm/arch/akita.h>
  30. #include <asm/arch/spitz.h>
  31. #include "../codecs/wm8750.h"
  32. #include "pxa2xx-pcm.h"
  33. #include "pxa2xx-i2s.h"
  34. #define SPITZ_HP 0
  35. #define SPITZ_MIC 1
  36. #define SPITZ_LINE 2
  37. #define SPITZ_HEADSET 3
  38. #define SPITZ_HP_OFF 4
  39. #define SPITZ_SPK_ON 0
  40. #define SPITZ_SPK_OFF 1
  41. /* audio clock in Hz - rounded from 12.235MHz */
  42. #define SPITZ_AUDIO_CLOCK 12288000
  43. static int spitz_jack_func;
  44. static int spitz_spk_func;
  45. static void spitz_ext_control(struct snd_soc_codec *codec)
  46. {
  47. if (spitz_spk_func == SPITZ_SPK_ON)
  48. snd_soc_dapm_enable_pin(codec, "Ext Spk");
  49. else
  50. snd_soc_dapm_disable_pin(codec, "Ext Spk");
  51. /* set up jack connection */
  52. switch (spitz_jack_func) {
  53. case SPITZ_HP:
  54. /* enable and unmute hp jack, disable mic bias */
  55. snd_soc_dapm_disable_pin(codec, "Headset Jack");
  56. snd_soc_dapm_disable_pin(codec, "Mic Jack");
  57. snd_soc_dapm_disable_pin(codec, "Line Jack");
  58. snd_soc_dapm_enable_pin(codec, "Headphone Jack");
  59. set_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L);
  60. set_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R);
  61. break;
  62. case SPITZ_MIC:
  63. /* enable mic jack and bias, mute hp */
  64. snd_soc_dapm_disable_pin(codec, "Headphone Jack");
  65. snd_soc_dapm_disable_pin(codec, "Headset Jack");
  66. snd_soc_dapm_disable_pin(codec, "Line Jack");
  67. snd_soc_dapm_enable_pin(codec, "Mic Jack");
  68. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L);
  69. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R);
  70. break;
  71. case SPITZ_LINE:
  72. /* enable line jack, disable mic bias and mute hp */
  73. snd_soc_dapm_disable_pin(codec, "Headphone Jack");
  74. snd_soc_dapm_disable_pin(codec, "Headset Jack");
  75. snd_soc_dapm_disable_pin(codec, "Mic Jack");
  76. snd_soc_dapm_enable_pin(codec, "Line Jack");
  77. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L);
  78. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R);
  79. break;
  80. case SPITZ_HEADSET:
  81. /* enable and unmute headset jack enable mic bias, mute L hp */
  82. snd_soc_dapm_disable_pin(codec, "Headphone Jack");
  83. snd_soc_dapm_enable_pin(codec, "Mic Jack");
  84. snd_soc_dapm_disable_pin(codec, "Line Jack");
  85. snd_soc_dapm_enable_pin(codec, "Headset Jack");
  86. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L);
  87. set_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R);
  88. break;
  89. case SPITZ_HP_OFF:
  90. /* jack removed, everything off */
  91. snd_soc_dapm_disable_pin(codec, "Headphone Jack");
  92. snd_soc_dapm_disable_pin(codec, "Headset Jack");
  93. snd_soc_dapm_disable_pin(codec, "Mic Jack");
  94. snd_soc_dapm_disable_pin(codec, "Line Jack");
  95. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_L);
  96. reset_scoop_gpio(&spitzscoop_device.dev, SPITZ_SCP_MUTE_R);
  97. break;
  98. }
  99. snd_soc_dapm_sync(codec);
  100. }
  101. static int spitz_startup(struct snd_pcm_substream *substream)
  102. {
  103. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  104. struct snd_soc_codec *codec = rtd->socdev->codec;
  105. /* check the jack status at stream startup */
  106. spitz_ext_control(codec);
  107. return 0;
  108. }
  109. static int spitz_hw_params(struct snd_pcm_substream *substream,
  110. struct snd_pcm_hw_params *params)
  111. {
  112. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  113. struct snd_soc_dai *codec_dai = rtd->dai->codec_dai;
  114. struct snd_soc_dai *cpu_dai = rtd->dai->cpu_dai;
  115. unsigned int clk = 0;
  116. int ret = 0;
  117. switch (params_rate(params)) {
  118. case 8000:
  119. case 16000:
  120. case 48000:
  121. case 96000:
  122. clk = 12288000;
  123. break;
  124. case 11025:
  125. case 22050:
  126. case 44100:
  127. clk = 11289600;
  128. break;
  129. }
  130. /* set codec DAI configuration */
  131. ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_I2S |
  132. SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBS_CFS);
  133. if (ret < 0)
  134. return ret;
  135. /* set cpu DAI configuration */
  136. ret = snd_soc_dai_set_fmt(cpu_dai, SND_SOC_DAIFMT_I2S |
  137. SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBS_CFS);
  138. if (ret < 0)
  139. return ret;
  140. /* set the codec system clock for DAC and ADC */
  141. ret = snd_soc_dai_set_sysclk(codec_dai, WM8750_SYSCLK, clk,
  142. SND_SOC_CLOCK_IN);
  143. if (ret < 0)
  144. return ret;
  145. /* set the I2S system clock as input (unused) */
  146. ret = snd_soc_dai_set_sysclk(cpu_dai, PXA2XX_I2S_SYSCLK, 0,
  147. SND_SOC_CLOCK_IN);
  148. if (ret < 0)
  149. return ret;
  150. return 0;
  151. }
  152. static struct snd_soc_ops spitz_ops = {
  153. .startup = spitz_startup,
  154. .hw_params = spitz_hw_params,
  155. };
  156. static int spitz_get_jack(struct snd_kcontrol *kcontrol,
  157. struct snd_ctl_elem_value *ucontrol)
  158. {
  159. ucontrol->value.integer.value[0] = spitz_jack_func;
  160. return 0;
  161. }
  162. static int spitz_set_jack(struct snd_kcontrol *kcontrol,
  163. struct snd_ctl_elem_value *ucontrol)
  164. {
  165. struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
  166. if (spitz_jack_func == ucontrol->value.integer.value[0])
  167. return 0;
  168. spitz_jack_func = ucontrol->value.integer.value[0];
  169. spitz_ext_control(codec);
  170. return 1;
  171. }
  172. static int spitz_get_spk(struct snd_kcontrol *kcontrol,
  173. struct snd_ctl_elem_value *ucontrol)
  174. {
  175. ucontrol->value.integer.value[0] = spitz_spk_func;
  176. return 0;
  177. }
  178. static int spitz_set_spk(struct snd_kcontrol *kcontrol,
  179. struct snd_ctl_elem_value *ucontrol)
  180. {
  181. struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol);
  182. if (spitz_spk_func == ucontrol->value.integer.value[0])
  183. return 0;
  184. spitz_spk_func = ucontrol->value.integer.value[0];
  185. spitz_ext_control(codec);
  186. return 1;
  187. }
  188. static int spitz_mic_bias(struct snd_soc_dapm_widget *w,
  189. struct snd_kcontrol *k, int event)
  190. {
  191. if (machine_is_borzoi() || machine_is_spitz()) {
  192. if (SND_SOC_DAPM_EVENT_ON(event))
  193. set_scoop_gpio(&spitzscoop2_device.dev,
  194. SPITZ_SCP2_MIC_BIAS);
  195. else
  196. reset_scoop_gpio(&spitzscoop2_device.dev,
  197. SPITZ_SCP2_MIC_BIAS);
  198. }
  199. if (machine_is_akita()) {
  200. if (SND_SOC_DAPM_EVENT_ON(event))
  201. akita_set_ioexp(&akitaioexp_device.dev,
  202. AKITA_IOEXP_MIC_BIAS);
  203. else
  204. akita_reset_ioexp(&akitaioexp_device.dev,
  205. AKITA_IOEXP_MIC_BIAS);
  206. }
  207. return 0;
  208. }
  209. /* spitz machine dapm widgets */
  210. static const struct snd_soc_dapm_widget wm8750_dapm_widgets[] = {
  211. SND_SOC_DAPM_HP("Headphone Jack", NULL),
  212. SND_SOC_DAPM_MIC("Mic Jack", spitz_mic_bias),
  213. SND_SOC_DAPM_SPK("Ext Spk", NULL),
  214. SND_SOC_DAPM_LINE("Line Jack", NULL),
  215. /* headset is a mic and mono headphone */
  216. SND_SOC_DAPM_HP("Headset Jack", NULL),
  217. };
  218. /* Spitz machine audio_map */
  219. static const struct snd_soc_dapm_route audio_map[] = {
  220. /* headphone connected to LOUT1, ROUT1 */
  221. {"Headphone Jack", NULL, "LOUT1"},
  222. {"Headphone Jack", NULL, "ROUT1"},
  223. /* headset connected to ROUT1 and LINPUT1 with bias (def below) */
  224. {"Headset Jack", NULL, "ROUT1"},
  225. /* ext speaker connected to LOUT2, ROUT2 */
  226. {"Ext Spk", NULL , "ROUT2"},
  227. {"Ext Spk", NULL , "LOUT2"},
  228. /* mic is connected to input 1 - with bias */
  229. {"LINPUT1", NULL, "Mic Bias"},
  230. {"Mic Bias", NULL, "Mic Jack"},
  231. /* line is connected to input 1 - no bias */
  232. {"LINPUT1", NULL, "Line Jack"},
  233. };
  234. static const char *jack_function[] = {"Headphone", "Mic", "Line", "Headset",
  235. "Off"};
  236. static const char *spk_function[] = {"On", "Off"};
  237. static const struct soc_enum spitz_enum[] = {
  238. SOC_ENUM_SINGLE_EXT(5, jack_function),
  239. SOC_ENUM_SINGLE_EXT(2, spk_function),
  240. };
  241. static const struct snd_kcontrol_new wm8750_spitz_controls[] = {
  242. SOC_ENUM_EXT("Jack Function", spitz_enum[0], spitz_get_jack,
  243. spitz_set_jack),
  244. SOC_ENUM_EXT("Speaker Function", spitz_enum[1], spitz_get_spk,
  245. spitz_set_spk),
  246. };
  247. /*
  248. * Logic for a wm8750 as connected on a Sharp SL-Cxx00 Device
  249. */
  250. static int spitz_wm8750_init(struct snd_soc_codec *codec)
  251. {
  252. int i, err;
  253. /* NC codec pins */
  254. snd_soc_dapm_disable_pin(codec, "RINPUT1");
  255. snd_soc_dapm_disable_pin(codec, "LINPUT2");
  256. snd_soc_dapm_disable_pin(codec, "RINPUT2");
  257. snd_soc_dapm_disable_pin(codec, "LINPUT3");
  258. snd_soc_dapm_disable_pin(codec, "RINPUT3");
  259. snd_soc_dapm_disable_pin(codec, "OUT3");
  260. snd_soc_dapm_disable_pin(codec, "MONO");
  261. /* Add spitz specific controls */
  262. for (i = 0; i < ARRAY_SIZE(wm8750_spitz_controls); i++) {
  263. err = snd_ctl_add(codec->card,
  264. snd_soc_cnew(&wm8750_spitz_controls[i], codec, NULL));
  265. if (err < 0)
  266. return err;
  267. }
  268. /* Add spitz specific widgets */
  269. snd_soc_dapm_new_controls(codec, wm8750_dapm_widgets,
  270. ARRAY_SIZE(wm8750_dapm_widgets));
  271. /* Set up spitz specific audio paths */
  272. snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map));
  273. snd_soc_dapm_sync(codec);
  274. return 0;
  275. }
  276. /* spitz digital audio interface glue - connects codec <--> CPU */
  277. static struct snd_soc_dai_link spitz_dai = {
  278. .name = "wm8750",
  279. .stream_name = "WM8750",
  280. .cpu_dai = &pxa_i2s_dai,
  281. .codec_dai = &wm8750_dai,
  282. .init = spitz_wm8750_init,
  283. .ops = &spitz_ops,
  284. };
  285. /* spitz audio machine driver */
  286. static struct snd_soc_machine snd_soc_machine_spitz = {
  287. .name = "Spitz",
  288. .dai_link = &spitz_dai,
  289. .num_links = 1,
  290. };
  291. /* spitz audio private data */
  292. static struct wm8750_setup_data spitz_wm8750_setup = {
  293. .i2c_address = 0x1b,
  294. };
  295. /* spitz audio subsystem */
  296. static struct snd_soc_device spitz_snd_devdata = {
  297. .machine = &snd_soc_machine_spitz,
  298. .platform = &pxa2xx_soc_platform,
  299. .codec_dev = &soc_codec_dev_wm8750,
  300. .codec_data = &spitz_wm8750_setup,
  301. };
  302. static struct platform_device *spitz_snd_device;
  303. static int __init spitz_init(void)
  304. {
  305. int ret;
  306. if (!(machine_is_spitz() || machine_is_borzoi() || machine_is_akita()))
  307. return -ENODEV;
  308. spitz_snd_device = platform_device_alloc("soc-audio", -1);
  309. if (!spitz_snd_device)
  310. return -ENOMEM;
  311. platform_set_drvdata(spitz_snd_device, &spitz_snd_devdata);
  312. spitz_snd_devdata.dev = &spitz_snd_device->dev;
  313. ret = platform_device_add(spitz_snd_device);
  314. if (ret)
  315. platform_device_put(spitz_snd_device);
  316. return ret;
  317. }
  318. static void __exit spitz_exit(void)
  319. {
  320. platform_device_unregister(spitz_snd_device);
  321. }
  322. module_init(spitz_init);
  323. module_exit(spitz_exit);
  324. MODULE_AUTHOR("Richard Purdie");
  325. MODULE_DESCRIPTION("ALSA SoC Spitz");
  326. MODULE_LICENSE("GPL");