ssm2602.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. //
  3. // File: sound/soc/codecs/ssm2602.c
  4. // Author: Cliff Cai <[email protected]>
  5. //
  6. // Created: Tue June 06 2008
  7. // Description: Driver for ssm2602 sound chip
  8. //
  9. // Modified:
  10. // Copyright 2008 Analog Devices Inc.
  11. //
  12. // Bugs: Enter bugs at http://blackfin.uclinux.org/
  13. #include <linux/delay.h>
  14. #include <linux/module.h>
  15. #include <linux/regmap.h>
  16. #include <linux/slab.h>
  17. #include <sound/pcm.h>
  18. #include <sound/pcm_params.h>
  19. #include <sound/soc.h>
  20. #include <sound/tlv.h>
  21. #include "ssm2602.h"
  22. /* codec private data */
  23. struct ssm2602_priv {
  24. unsigned int sysclk;
  25. const struct snd_pcm_hw_constraint_list *sysclk_constraints;
  26. struct regmap *regmap;
  27. enum ssm2602_type type;
  28. unsigned int clk_out_pwr;
  29. };
  30. /*
  31. * ssm2602 register cache
  32. * We can't read the ssm2602 register space when we are
  33. * using 2 wire for device control, so we cache them instead.
  34. * There is no point in caching the reset register
  35. */
  36. static const struct reg_default ssm2602_reg[SSM2602_CACHEREGNUM] = {
  37. { .reg = 0x00, .def = 0x0097 },
  38. { .reg = 0x01, .def = 0x0097 },
  39. { .reg = 0x02, .def = 0x0079 },
  40. { .reg = 0x03, .def = 0x0079 },
  41. { .reg = 0x04, .def = 0x000a },
  42. { .reg = 0x05, .def = 0x0008 },
  43. { .reg = 0x06, .def = 0x009f },
  44. { .reg = 0x07, .def = 0x000a },
  45. { .reg = 0x08, .def = 0x0000 },
  46. { .reg = 0x09, .def = 0x0000 }
  47. };
  48. /*
  49. * ssm2602 register patch
  50. * Workaround for playback distortions after power up: activates digital
  51. * core, and then powers on output, DAC, and whole chip at the same time
  52. */
  53. static const struct reg_sequence ssm2602_patch[] = {
  54. { SSM2602_ACTIVE, 0x01 },
  55. { SSM2602_PWR, 0x07 },
  56. { SSM2602_RESET, 0x00 },
  57. };
  58. /*Appending several "None"s just for OSS mixer use*/
  59. static const char *ssm2602_input_select[] = {
  60. "Line", "Mic",
  61. };
  62. static const char *ssm2602_deemph[] = {"None", "32Khz", "44.1Khz", "48Khz"};
  63. static const struct soc_enum ssm2602_enum[] = {
  64. SOC_ENUM_SINGLE(SSM2602_APANA, 2, ARRAY_SIZE(ssm2602_input_select),
  65. ssm2602_input_select),
  66. SOC_ENUM_SINGLE(SSM2602_APDIGI, 1, ARRAY_SIZE(ssm2602_deemph),
  67. ssm2602_deemph),
  68. };
  69. static const DECLARE_TLV_DB_RANGE(ssm260x_outmix_tlv,
  70. 0, 47, TLV_DB_SCALE_ITEM(TLV_DB_GAIN_MUTE, 0, 0),
  71. 48, 127, TLV_DB_SCALE_ITEM(-7400, 100, 0)
  72. );
  73. static const DECLARE_TLV_DB_SCALE(ssm260x_inpga_tlv, -3450, 150, 0);
  74. static const DECLARE_TLV_DB_SCALE(ssm260x_sidetone_tlv, -1500, 300, 0);
  75. static const struct snd_kcontrol_new ssm260x_snd_controls[] = {
  76. SOC_DOUBLE_R_TLV("Capture Volume", SSM2602_LINVOL, SSM2602_RINVOL, 0, 45, 0,
  77. ssm260x_inpga_tlv),
  78. SOC_DOUBLE_R("Capture Switch", SSM2602_LINVOL, SSM2602_RINVOL, 7, 1, 1),
  79. SOC_SINGLE("ADC High Pass Filter Switch", SSM2602_APDIGI, 0, 1, 1),
  80. SOC_SINGLE("Store DC Offset Switch", SSM2602_APDIGI, 4, 1, 0),
  81. SOC_ENUM("Playback De-emphasis", ssm2602_enum[1]),
  82. };
  83. static const struct snd_kcontrol_new ssm2602_snd_controls[] = {
  84. SOC_DOUBLE_R_TLV("Master Playback Volume", SSM2602_LOUT1V, SSM2602_ROUT1V,
  85. 0, 127, 0, ssm260x_outmix_tlv),
  86. SOC_DOUBLE_R("Master Playback ZC Switch", SSM2602_LOUT1V, SSM2602_ROUT1V,
  87. 7, 1, 0),
  88. SOC_SINGLE_TLV("Sidetone Playback Volume", SSM2602_APANA, 6, 3, 1,
  89. ssm260x_sidetone_tlv),
  90. SOC_SINGLE("Mic Boost (+20dB)", SSM2602_APANA, 0, 1, 0),
  91. SOC_SINGLE("Mic Boost2 (+20dB)", SSM2602_APANA, 8, 1, 0),
  92. };
  93. /* Output Mixer */
  94. static const struct snd_kcontrol_new ssm260x_output_mixer_controls[] = {
  95. SOC_DAPM_SINGLE("Line Bypass Switch", SSM2602_APANA, 3, 1, 0),
  96. SOC_DAPM_SINGLE("HiFi Playback Switch", SSM2602_APANA, 4, 1, 0),
  97. SOC_DAPM_SINGLE("Mic Sidetone Switch", SSM2602_APANA, 5, 1, 0),
  98. };
  99. static const struct snd_kcontrol_new mic_ctl =
  100. SOC_DAPM_SINGLE("Switch", SSM2602_APANA, 1, 1, 1);
  101. /* Input mux */
  102. static const struct snd_kcontrol_new ssm2602_input_mux_controls =
  103. SOC_DAPM_ENUM("Input Select", ssm2602_enum[0]);
  104. static int ssm2602_mic_switch_event(struct snd_soc_dapm_widget *w,
  105. struct snd_kcontrol *kcontrol, int event)
  106. {
  107. /*
  108. * According to the ssm2603 data sheet (control register sequencing),
  109. * the digital core should be activated only after all necessary bits
  110. * in the power register are enabled, and a delay determined by the
  111. * decoupling capacitor on the VMID pin has passed. If the digital core
  112. * is activated too early, or even before the ADC is powered up, audible
  113. * artifacts appear at the beginning and end of the recorded signal.
  114. *
  115. * In practice, audible artifacts disappear well over 500 ms.
  116. */
  117. msleep(500);
  118. return 0;
  119. }
  120. static const struct snd_soc_dapm_widget ssm260x_dapm_widgets[] = {
  121. SND_SOC_DAPM_DAC("DAC", "HiFi Playback", SSM2602_PWR, 3, 1),
  122. SND_SOC_DAPM_ADC("ADC", "HiFi Capture", SSM2602_PWR, 2, 1),
  123. SND_SOC_DAPM_PGA("Line Input", SSM2602_PWR, 0, 1, NULL, 0),
  124. SND_SOC_DAPM_SUPPLY("Digital Core Power", SSM2602_ACTIVE, 0, 0, NULL, 0),
  125. SND_SOC_DAPM_OUTPUT("LOUT"),
  126. SND_SOC_DAPM_OUTPUT("ROUT"),
  127. SND_SOC_DAPM_INPUT("RLINEIN"),
  128. SND_SOC_DAPM_INPUT("LLINEIN"),
  129. };
  130. static const struct snd_soc_dapm_widget ssm2602_dapm_widgets[] = {
  131. SND_SOC_DAPM_MIXER("Output Mixer", SSM2602_PWR, 4, 1,
  132. ssm260x_output_mixer_controls,
  133. ARRAY_SIZE(ssm260x_output_mixer_controls)),
  134. SND_SOC_DAPM_MUX("Input Mux", SND_SOC_NOPM, 0, 0, &ssm2602_input_mux_controls),
  135. SND_SOC_DAPM_MICBIAS("Mic Bias", SSM2602_PWR, 1, 1),
  136. SND_SOC_DAPM_SWITCH_E("Mic Switch", SSM2602_APANA, 1, 1, &mic_ctl,
  137. ssm2602_mic_switch_event, SND_SOC_DAPM_PRE_PMU),
  138. SND_SOC_DAPM_OUTPUT("LHPOUT"),
  139. SND_SOC_DAPM_OUTPUT("RHPOUT"),
  140. SND_SOC_DAPM_INPUT("MICIN"),
  141. };
  142. static const struct snd_soc_dapm_widget ssm2604_dapm_widgets[] = {
  143. SND_SOC_DAPM_MIXER("Output Mixer", SND_SOC_NOPM, 0, 0,
  144. ssm260x_output_mixer_controls,
  145. ARRAY_SIZE(ssm260x_output_mixer_controls) - 1), /* Last element is the mic */
  146. };
  147. static const struct snd_soc_dapm_route ssm260x_routes[] = {
  148. {"DAC", NULL, "Digital Core Power"},
  149. {"ADC", NULL, "Digital Core Power"},
  150. {"Output Mixer", "Line Bypass Switch", "Line Input"},
  151. {"Output Mixer", "HiFi Playback Switch", "DAC"},
  152. {"ROUT", NULL, "Output Mixer"},
  153. {"LOUT", NULL, "Output Mixer"},
  154. {"Line Input", NULL, "LLINEIN"},
  155. {"Line Input", NULL, "RLINEIN"},
  156. };
  157. static const struct snd_soc_dapm_route ssm2602_routes[] = {
  158. {"Output Mixer", "Mic Sidetone Switch", "Mic Bias"},
  159. {"RHPOUT", NULL, "Output Mixer"},
  160. {"LHPOUT", NULL, "Output Mixer"},
  161. {"Input Mux", "Line", "Line Input"},
  162. {"Input Mux", "Mic", "Mic Switch"},
  163. {"ADC", NULL, "Input Mux"},
  164. {"Mic Switch", NULL, "Mic Bias"},
  165. {"Mic Bias", NULL, "MICIN"},
  166. };
  167. static const struct snd_soc_dapm_route ssm2604_routes[] = {
  168. {"ADC", NULL, "Line Input"},
  169. };
  170. static const unsigned int ssm2602_rates_12288000[] = {
  171. 8000, 16000, 32000, 48000, 96000,
  172. };
  173. static const struct snd_pcm_hw_constraint_list ssm2602_constraints_12288000 = {
  174. .list = ssm2602_rates_12288000,
  175. .count = ARRAY_SIZE(ssm2602_rates_12288000),
  176. };
  177. static const unsigned int ssm2602_rates_11289600[] = {
  178. 8000, 11025, 22050, 44100, 88200,
  179. };
  180. static const struct snd_pcm_hw_constraint_list ssm2602_constraints_11289600 = {
  181. .list = ssm2602_rates_11289600,
  182. .count = ARRAY_SIZE(ssm2602_rates_11289600),
  183. };
  184. struct ssm2602_coeff {
  185. u32 mclk;
  186. u32 rate;
  187. u8 srate;
  188. };
  189. #define SSM2602_COEFF_SRATE(sr, bosr, usb) (((sr) << 2) | ((bosr) << 1) | (usb))
  190. /* codec mclk clock coefficients */
  191. static const struct ssm2602_coeff ssm2602_coeff_table[] = {
  192. /* 48k */
  193. {12288000, 48000, SSM2602_COEFF_SRATE(0x0, 0x0, 0x0)},
  194. {18432000, 48000, SSM2602_COEFF_SRATE(0x0, 0x1, 0x0)},
  195. {12000000, 48000, SSM2602_COEFF_SRATE(0x0, 0x0, 0x1)},
  196. /* 32k */
  197. {12288000, 32000, SSM2602_COEFF_SRATE(0x6, 0x0, 0x0)},
  198. {18432000, 32000, SSM2602_COEFF_SRATE(0x6, 0x1, 0x0)},
  199. {12000000, 32000, SSM2602_COEFF_SRATE(0x6, 0x0, 0x1)},
  200. /* 16k */
  201. {12288000, 16000, SSM2602_COEFF_SRATE(0x5, 0x0, 0x0)},
  202. {18432000, 16000, SSM2602_COEFF_SRATE(0x5, 0x1, 0x0)},
  203. {12000000, 16000, SSM2602_COEFF_SRATE(0xa, 0x0, 0x1)},
  204. /* 8k */
  205. {12288000, 8000, SSM2602_COEFF_SRATE(0x3, 0x0, 0x0)},
  206. {18432000, 8000, SSM2602_COEFF_SRATE(0x3, 0x1, 0x0)},
  207. {11289600, 8000, SSM2602_COEFF_SRATE(0xb, 0x0, 0x0)},
  208. {16934400, 8000, SSM2602_COEFF_SRATE(0xb, 0x1, 0x0)},
  209. {12000000, 8000, SSM2602_COEFF_SRATE(0x3, 0x0, 0x1)},
  210. /* 96k */
  211. {12288000, 96000, SSM2602_COEFF_SRATE(0x7, 0x0, 0x0)},
  212. {18432000, 96000, SSM2602_COEFF_SRATE(0x7, 0x1, 0x0)},
  213. {12000000, 96000, SSM2602_COEFF_SRATE(0x7, 0x0, 0x1)},
  214. /* 11.025k */
  215. {11289600, 11025, SSM2602_COEFF_SRATE(0xc, 0x0, 0x0)},
  216. {16934400, 11025, SSM2602_COEFF_SRATE(0xc, 0x1, 0x0)},
  217. {12000000, 11025, SSM2602_COEFF_SRATE(0xc, 0x1, 0x1)},
  218. /* 22.05k */
  219. {11289600, 22050, SSM2602_COEFF_SRATE(0xd, 0x0, 0x0)},
  220. {16934400, 22050, SSM2602_COEFF_SRATE(0xd, 0x1, 0x0)},
  221. {12000000, 22050, SSM2602_COEFF_SRATE(0xd, 0x1, 0x1)},
  222. /* 44.1k */
  223. {11289600, 44100, SSM2602_COEFF_SRATE(0x8, 0x0, 0x0)},
  224. {16934400, 44100, SSM2602_COEFF_SRATE(0x8, 0x1, 0x0)},
  225. {12000000, 44100, SSM2602_COEFF_SRATE(0x8, 0x1, 0x1)},
  226. /* 88.2k */
  227. {11289600, 88200, SSM2602_COEFF_SRATE(0xf, 0x0, 0x0)},
  228. {16934400, 88200, SSM2602_COEFF_SRATE(0xf, 0x1, 0x0)},
  229. {12000000, 88200, SSM2602_COEFF_SRATE(0xf, 0x1, 0x1)},
  230. };
  231. static inline int ssm2602_get_coeff(int mclk, int rate)
  232. {
  233. int i;
  234. for (i = 0; i < ARRAY_SIZE(ssm2602_coeff_table); i++) {
  235. if (ssm2602_coeff_table[i].rate == rate &&
  236. ssm2602_coeff_table[i].mclk == mclk)
  237. return ssm2602_coeff_table[i].srate;
  238. }
  239. return -EINVAL;
  240. }
  241. static int ssm2602_hw_params(struct snd_pcm_substream *substream,
  242. struct snd_pcm_hw_params *params,
  243. struct snd_soc_dai *dai)
  244. {
  245. struct snd_soc_component *component = dai->component;
  246. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  247. int srate = ssm2602_get_coeff(ssm2602->sysclk, params_rate(params));
  248. unsigned int iface;
  249. if (srate < 0)
  250. return srate;
  251. regmap_write(ssm2602->regmap, SSM2602_SRATE, srate);
  252. /* bit size */
  253. switch (params_width(params)) {
  254. case 16:
  255. iface = 0x0;
  256. break;
  257. case 20:
  258. iface = 0x4;
  259. break;
  260. case 24:
  261. iface = 0x8;
  262. break;
  263. case 32:
  264. iface = 0xc;
  265. break;
  266. default:
  267. return -EINVAL;
  268. }
  269. regmap_update_bits(ssm2602->regmap, SSM2602_IFACE,
  270. IFACE_AUDIO_DATA_LEN, iface);
  271. return 0;
  272. }
  273. static int ssm2602_startup(struct snd_pcm_substream *substream,
  274. struct snd_soc_dai *dai)
  275. {
  276. struct snd_soc_component *component = dai->component;
  277. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  278. if (ssm2602->sysclk_constraints) {
  279. snd_pcm_hw_constraint_list(substream->runtime, 0,
  280. SNDRV_PCM_HW_PARAM_RATE,
  281. ssm2602->sysclk_constraints);
  282. }
  283. return 0;
  284. }
  285. static int ssm2602_mute(struct snd_soc_dai *dai, int mute, int direction)
  286. {
  287. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(dai->component);
  288. if (mute)
  289. regmap_update_bits(ssm2602->regmap, SSM2602_APDIGI,
  290. APDIGI_ENABLE_DAC_MUTE,
  291. APDIGI_ENABLE_DAC_MUTE);
  292. else
  293. regmap_update_bits(ssm2602->regmap, SSM2602_APDIGI,
  294. APDIGI_ENABLE_DAC_MUTE, 0);
  295. return 0;
  296. }
  297. static int ssm2602_set_dai_sysclk(struct snd_soc_dai *codec_dai,
  298. int clk_id, unsigned int freq, int dir)
  299. {
  300. struct snd_soc_component *component = codec_dai->component;
  301. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  302. if (dir == SND_SOC_CLOCK_IN) {
  303. if (clk_id != SSM2602_SYSCLK)
  304. return -EINVAL;
  305. switch (freq) {
  306. case 12288000:
  307. case 18432000:
  308. ssm2602->sysclk_constraints = &ssm2602_constraints_12288000;
  309. break;
  310. case 11289600:
  311. case 16934400:
  312. ssm2602->sysclk_constraints = &ssm2602_constraints_11289600;
  313. break;
  314. case 12000000:
  315. ssm2602->sysclk_constraints = NULL;
  316. break;
  317. default:
  318. return -EINVAL;
  319. }
  320. ssm2602->sysclk = freq;
  321. } else {
  322. unsigned int mask;
  323. switch (clk_id) {
  324. case SSM2602_CLK_CLKOUT:
  325. mask = PWR_CLK_OUT_PDN;
  326. break;
  327. case SSM2602_CLK_XTO:
  328. mask = PWR_OSC_PDN;
  329. break;
  330. default:
  331. return -EINVAL;
  332. }
  333. if (freq == 0)
  334. ssm2602->clk_out_pwr |= mask;
  335. else
  336. ssm2602->clk_out_pwr &= ~mask;
  337. regmap_update_bits(ssm2602->regmap, SSM2602_PWR,
  338. PWR_CLK_OUT_PDN | PWR_OSC_PDN, ssm2602->clk_out_pwr);
  339. }
  340. return 0;
  341. }
  342. static int ssm2602_set_dai_fmt(struct snd_soc_dai *codec_dai,
  343. unsigned int fmt)
  344. {
  345. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(codec_dai->component);
  346. unsigned int iface = 0;
  347. /* set master/slave audio interface */
  348. switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {
  349. case SND_SOC_DAIFMT_CBP_CFP:
  350. iface |= 0x0040;
  351. break;
  352. case SND_SOC_DAIFMT_CBC_CFC:
  353. break;
  354. default:
  355. return -EINVAL;
  356. }
  357. /* interface format */
  358. switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
  359. case SND_SOC_DAIFMT_I2S:
  360. iface |= 0x0002;
  361. break;
  362. case SND_SOC_DAIFMT_RIGHT_J:
  363. break;
  364. case SND_SOC_DAIFMT_LEFT_J:
  365. iface |= 0x0001;
  366. break;
  367. case SND_SOC_DAIFMT_DSP_A:
  368. iface |= 0x0013;
  369. break;
  370. case SND_SOC_DAIFMT_DSP_B:
  371. iface |= 0x0003;
  372. break;
  373. default:
  374. return -EINVAL;
  375. }
  376. /* clock inversion */
  377. switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
  378. case SND_SOC_DAIFMT_NB_NF:
  379. break;
  380. case SND_SOC_DAIFMT_IB_IF:
  381. iface |= 0x0090;
  382. break;
  383. case SND_SOC_DAIFMT_IB_NF:
  384. iface |= 0x0080;
  385. break;
  386. case SND_SOC_DAIFMT_NB_IF:
  387. iface |= 0x0010;
  388. break;
  389. default:
  390. return -EINVAL;
  391. }
  392. /* set iface */
  393. regmap_write(ssm2602->regmap, SSM2602_IFACE, iface);
  394. return 0;
  395. }
  396. static int ssm2602_set_bias_level(struct snd_soc_component *component,
  397. enum snd_soc_bias_level level)
  398. {
  399. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  400. switch (level) {
  401. case SND_SOC_BIAS_ON:
  402. /* vref/mid on, osc and clkout on if enabled */
  403. regmap_update_bits(ssm2602->regmap, SSM2602_PWR,
  404. PWR_POWER_OFF | PWR_CLK_OUT_PDN | PWR_OSC_PDN,
  405. ssm2602->clk_out_pwr);
  406. break;
  407. case SND_SOC_BIAS_PREPARE:
  408. break;
  409. case SND_SOC_BIAS_STANDBY:
  410. /* everything off except vref/vmid, */
  411. regmap_update_bits(ssm2602->regmap, SSM2602_PWR,
  412. PWR_POWER_OFF | PWR_CLK_OUT_PDN | PWR_OSC_PDN,
  413. PWR_CLK_OUT_PDN | PWR_OSC_PDN);
  414. break;
  415. case SND_SOC_BIAS_OFF:
  416. /* everything off */
  417. regmap_update_bits(ssm2602->regmap, SSM2602_PWR,
  418. PWR_POWER_OFF, PWR_POWER_OFF);
  419. break;
  420. }
  421. return 0;
  422. }
  423. #define SSM2602_RATES (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 |\
  424. SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 |\
  425. SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 |\
  426. SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 |\
  427. SNDRV_PCM_RATE_96000)
  428. #define SSM2602_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\
  429. SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE)
  430. static const struct snd_soc_dai_ops ssm2602_dai_ops = {
  431. .startup = ssm2602_startup,
  432. .hw_params = ssm2602_hw_params,
  433. .mute_stream = ssm2602_mute,
  434. .set_sysclk = ssm2602_set_dai_sysclk,
  435. .set_fmt = ssm2602_set_dai_fmt,
  436. .no_capture_mute = 1,
  437. };
  438. static struct snd_soc_dai_driver ssm2602_dai = {
  439. .name = "ssm2602-hifi",
  440. .playback = {
  441. .stream_name = "Playback",
  442. .channels_min = 2,
  443. .channels_max = 2,
  444. .rates = SSM2602_RATES,
  445. .formats = SSM2602_FORMATS,},
  446. .capture = {
  447. .stream_name = "Capture",
  448. .channels_min = 2,
  449. .channels_max = 2,
  450. .rates = SSM2602_RATES,
  451. .formats = SSM2602_FORMATS,},
  452. .ops = &ssm2602_dai_ops,
  453. .symmetric_rate = 1,
  454. .symmetric_sample_bits = 1,
  455. };
  456. static int ssm2602_resume(struct snd_soc_component *component)
  457. {
  458. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  459. regcache_sync(ssm2602->regmap);
  460. return 0;
  461. }
  462. static int ssm2602_component_probe(struct snd_soc_component *component)
  463. {
  464. struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
  465. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  466. int ret;
  467. regmap_update_bits(ssm2602->regmap, SSM2602_LOUT1V,
  468. LOUT1V_LRHP_BOTH, LOUT1V_LRHP_BOTH);
  469. regmap_update_bits(ssm2602->regmap, SSM2602_ROUT1V,
  470. ROUT1V_RLHP_BOTH, ROUT1V_RLHP_BOTH);
  471. ret = snd_soc_add_component_controls(component, ssm2602_snd_controls,
  472. ARRAY_SIZE(ssm2602_snd_controls));
  473. if (ret)
  474. return ret;
  475. ret = snd_soc_dapm_new_controls(dapm, ssm2602_dapm_widgets,
  476. ARRAY_SIZE(ssm2602_dapm_widgets));
  477. if (ret)
  478. return ret;
  479. return snd_soc_dapm_add_routes(dapm, ssm2602_routes,
  480. ARRAY_SIZE(ssm2602_routes));
  481. }
  482. static int ssm2604_component_probe(struct snd_soc_component *component)
  483. {
  484. struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
  485. int ret;
  486. ret = snd_soc_dapm_new_controls(dapm, ssm2604_dapm_widgets,
  487. ARRAY_SIZE(ssm2604_dapm_widgets));
  488. if (ret)
  489. return ret;
  490. return snd_soc_dapm_add_routes(dapm, ssm2604_routes,
  491. ARRAY_SIZE(ssm2604_routes));
  492. }
  493. static int ssm260x_component_probe(struct snd_soc_component *component)
  494. {
  495. struct ssm2602_priv *ssm2602 = snd_soc_component_get_drvdata(component);
  496. int ret;
  497. ret = regmap_write(ssm2602->regmap, SSM2602_RESET, 0);
  498. if (ret < 0) {
  499. dev_err(component->dev, "Failed to issue reset: %d\n", ret);
  500. return ret;
  501. }
  502. regmap_register_patch(ssm2602->regmap, ssm2602_patch,
  503. ARRAY_SIZE(ssm2602_patch));
  504. /* set the update bits */
  505. regmap_update_bits(ssm2602->regmap, SSM2602_LINVOL,
  506. LINVOL_LRIN_BOTH, LINVOL_LRIN_BOTH);
  507. regmap_update_bits(ssm2602->regmap, SSM2602_RINVOL,
  508. RINVOL_RLIN_BOTH, RINVOL_RLIN_BOTH);
  509. /*select Line in as default input*/
  510. regmap_write(ssm2602->regmap, SSM2602_APANA, APANA_SELECT_DAC |
  511. APANA_ENABLE_MIC_BOOST);
  512. switch (ssm2602->type) {
  513. case SSM2602:
  514. ret = ssm2602_component_probe(component);
  515. break;
  516. case SSM2604:
  517. ret = ssm2604_component_probe(component);
  518. break;
  519. }
  520. return ret;
  521. }
  522. static const struct snd_soc_component_driver soc_component_dev_ssm2602 = {
  523. .probe = ssm260x_component_probe,
  524. .resume = ssm2602_resume,
  525. .set_bias_level = ssm2602_set_bias_level,
  526. .controls = ssm260x_snd_controls,
  527. .num_controls = ARRAY_SIZE(ssm260x_snd_controls),
  528. .dapm_widgets = ssm260x_dapm_widgets,
  529. .num_dapm_widgets = ARRAY_SIZE(ssm260x_dapm_widgets),
  530. .dapm_routes = ssm260x_routes,
  531. .num_dapm_routes = ARRAY_SIZE(ssm260x_routes),
  532. .suspend_bias_off = 1,
  533. .idle_bias_on = 1,
  534. .use_pmdown_time = 1,
  535. .endianness = 1,
  536. };
  537. static bool ssm2602_register_volatile(struct device *dev, unsigned int reg)
  538. {
  539. return reg == SSM2602_RESET;
  540. }
  541. const struct regmap_config ssm2602_regmap_config = {
  542. .val_bits = 9,
  543. .reg_bits = 7,
  544. .max_register = SSM2602_RESET,
  545. .volatile_reg = ssm2602_register_volatile,
  546. .cache_type = REGCACHE_RBTREE,
  547. .reg_defaults = ssm2602_reg,
  548. .num_reg_defaults = ARRAY_SIZE(ssm2602_reg),
  549. };
  550. EXPORT_SYMBOL_GPL(ssm2602_regmap_config);
  551. int ssm2602_probe(struct device *dev, enum ssm2602_type type,
  552. struct regmap *regmap)
  553. {
  554. struct ssm2602_priv *ssm2602;
  555. if (IS_ERR(regmap))
  556. return PTR_ERR(regmap);
  557. ssm2602 = devm_kzalloc(dev, sizeof(*ssm2602), GFP_KERNEL);
  558. if (ssm2602 == NULL)
  559. return -ENOMEM;
  560. dev_set_drvdata(dev, ssm2602);
  561. ssm2602->type = type;
  562. ssm2602->regmap = regmap;
  563. return devm_snd_soc_register_component(dev, &soc_component_dev_ssm2602,
  564. &ssm2602_dai, 1);
  565. }
  566. EXPORT_SYMBOL_GPL(ssm2602_probe);
  567. MODULE_DESCRIPTION("ASoC SSM2602/SSM2603/SSM2604 driver");
  568. MODULE_AUTHOR("Cliff Cai");
  569. MODULE_LICENSE("GPL");