hpm_wm8960.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525
  1. /*
  2. * Copyright (c) 2015, Freescale Semiconductor, Inc.
  3. * Copyright 2016-2021 NXP
  4. * Copyright (c) 2022 HPMicro
  5. *
  6. * SPDX-License-Identifier: BSD-3-Clause
  7. *
  8. */
  9. #include <stdio.h>
  10. #include "hpm_wm8960.h"
  11. /* wm8960 register default value */
  12. static const uint16_t wm8960_default_reg_val[WM8960_REG_NUM] = {
  13. 0x0097, 0x0097, 0x0000, 0x0000, 0x0000, 0x0008, 0x0000, 0x000a, 0x01c0, 0x0000, 0x00ff, 0x00ff, 0x0000, 0x0000,
  14. 0x0000, 0x0000, 0x0000, 0x007b, 0x0100, 0x0032, 0x0000, 0x00c3, 0x00c3, 0x01c0, 0x0000, 0x0000, 0x0000, 0x0000,
  15. 0x0000, 0x0000, 0x0000, 0x0000, 0x0100, 0x0100, 0x0050, 0x0050, 0x0050, 0x0050, 0x0000, 0x0000, 0x0000, 0x0000,
  16. 0x0040, 0x0000, 0x0000, 0x0050, 0x0050, 0x0000, 0x0002, 0x0037, 0x004d, 0x0080, 0x0008, 0x0031, 0x0026, 0x00e9,
  17. };
  18. /* store reg value */
  19. static uint16_t wm8960_reg_val[WM8960_REG_NUM];
  20. hpm_stat_t wm8960_init(wm8960_control_t *control, wm8960_config_t *config)
  21. {
  22. assert(control != NULL);
  23. assert(config != NULL);
  24. hpm_stat_t stat = status_success;
  25. (void)memcpy(wm8960_reg_val, wm8960_default_reg_val, sizeof(wm8960_default_reg_val));
  26. /* Reset */
  27. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RESET, 0x00));
  28. /* Power on input modules */
  29. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, 0xFE));
  30. /* Power on output modules */
  31. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER2, 0x1F8));
  32. /* Power on PGA and mixer */
  33. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER3, 0x3C));
  34. /* ADC and DAC uses same clock */
  35. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_IFACE2, 0x40));
  36. /* set data protocol */
  37. HPM_CHECK_RET(wm8960_set_protocol(control, config->bus));
  38. /* set wm8960 as slave */
  39. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_IFACE1, WM8960_IFACE1_MS_MASK, WM8960_IFACE1_MS_SET(0)));
  40. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ADDCTL1, 0xC0));
  41. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ADDCTL4, 0x40));
  42. /* ADC volume, 8dB */
  43. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LADC, 0x1D3));
  44. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RADC, 0x1D3));
  45. /* Digital DAC volume, 0dB */
  46. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LDAC, 0x1E0));
  47. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RDAC, 0x1E0));
  48. /* Headphone volume, LOUT1 and ROUT1, 6dB */
  49. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LOUT1, 0x17F));
  50. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ROUT1, 0x17F));
  51. /* speaker volume 6dB */
  52. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LOUT2, 0x1ff));
  53. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ROUT2, 0x1ff));
  54. /* enable class D output */
  55. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_CLASSD1, 0xf7));
  56. /* Unmute DAC. */
  57. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_DACCTL1, 0x0000));
  58. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINVOL, 0x117));
  59. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINVOL, 0x117));
  60. HPM_CHECK_RET(wm8960_set_data_format(control, config->format.mclk_hz, config->format.sample_rate, config->format.bit_width));
  61. /* set data route */
  62. HPM_CHECK_RET(wm8960_set_data_route(control, config));
  63. return status_success;
  64. }
  65. hpm_stat_t wm8960_deinit(wm8960_control_t *control)
  66. {
  67. hpm_stat_t stat = status_success;
  68. /* power off all modules */
  69. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, 0x00U));
  70. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER2, 0x00U));
  71. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER3, 0x00U));
  72. return status_success;
  73. }
  74. hpm_stat_t wm8960_set_protocol(wm8960_control_t *control, wm8960_protocol_t protocol)
  75. {
  76. return wm8960_modify_reg(control, WM8960_IFACE1, WM8960_IFACE1_FORMAT_MASK, (uint16_t)protocol);
  77. }
  78. hpm_stat_t wm8960_set_module(wm8960_control_t *control, wm8960_module_t module, bool enable)
  79. {
  80. hpm_stat_t stat = status_success;
  81. switch (module) {
  82. case wm8960_module_adc:
  83. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER1, WM8960_POWER1_ADCL_MASK,
  84. ((uint16_t)enable << WM8960_POWER1_ADCL_SHIFT)));
  85. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER1, WM8960_POWER1_ADCR_MASK,
  86. ((uint16_t)enable << WM8960_POWER1_ADCR_SHIFT)));
  87. break;
  88. case wm8960_module_dac:
  89. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER2, WM8960_POWER2_DACL_MASK,
  90. ((uint16_t)enable << WM8960_POWER2_DACL_SHIFT)));
  91. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER2, WM8960_POWER2_DACR_MASK,
  92. ((uint16_t)enable << WM8960_POWER2_DACR_SHIFT)));
  93. break;
  94. case wm8960_module_vref:
  95. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER1, WM8960_POWER1_VREF_MASK,
  96. ((uint16_t)enable << WM8960_POWER1_VREF_SHIFT)));
  97. break;
  98. case wm8960_module_ana_in:
  99. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER1, WM8960_POWER1_AINL_MASK,
  100. ((uint16_t)enable << WM8960_POWER1_AINL_SHIFT)));
  101. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER1, WM8960_POWER1_AINR_MASK,
  102. ((uint16_t)enable << WM8960_POWER1_AINR_SHIFT)));
  103. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER3, WM8960_POWER3_LMIC_MASK,
  104. ((uint16_t)enable << WM8960_POWER3_LMIC_SHIFT)));
  105. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER3, WM8960_POWER3_RMIC_MASK,
  106. ((uint16_t)enable << WM8960_POWER3_RMIC_SHIFT)));
  107. break;
  108. case wm8960_module_lineout:
  109. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER2, WM8960_POWER2_LOUT1_MASK,
  110. ((uint16_t)enable << WM8960_POWER2_LOUT1_SHIFT)));
  111. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER2, WM8960_POWER2_ROUT1_MASK,
  112. ((uint16_t)enable << WM8960_POWER2_ROUT1_SHIFT)));
  113. break;
  114. case wm8960_module_micbais:
  115. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER1, WM8960_POWER1_MICB_MASK,
  116. ((uint16_t)enable << WM8960_POWER1_MICB_SHIFT)));
  117. break;
  118. case wm8960_module_speaker:
  119. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER2, WM8960_POWER2_SPKL_MASK,
  120. ((uint16_t)enable << WM8960_POWER2_SPKL_SHIFT)));
  121. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER2, WM8960_POWER2_SPKR_MASK,
  122. ((uint16_t)enable << WM8960_POWER2_SPKR_SHIFT)));
  123. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_CLASSD1, 0xF7));
  124. break;
  125. case wm8960_module_output_mixer:
  126. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER3, WM8960_POWER3_LOMIX_MASK,
  127. ((uint16_t)enable << WM8960_POWER3_LOMIX_SHIFT)));
  128. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_POWER3, WM8960_POWER3_ROMIX_MASK,
  129. ((uint16_t)enable << WM8960_POWER3_ROMIX_SHIFT)));
  130. break;
  131. default:
  132. stat = status_invalid_argument;
  133. break;
  134. }
  135. return stat;
  136. }
  137. hpm_stat_t wm8960_set_data_route(wm8960_control_t *control, wm8960_config_t *config)
  138. {
  139. hpm_stat_t stat = status_success;
  140. /* select left input */
  141. HPM_CHECK_RET(wm8960_set_left_input(control, config->left_input));
  142. /* select right input */
  143. HPM_CHECK_RET(wm8960_set_right_input(control, config->right_input));
  144. /* select source to output mixer */
  145. HPM_CHECK_RET(wm8960_config_input_to_output_mixer(control, config->play_source));
  146. switch (config->route) {
  147. case wm8960_route_bypass:
  148. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_micbais, true));
  149. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_ana_in, true));
  150. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_output_mixer, true));
  151. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_lineout, true));
  152. break;
  153. case wm8960_route_playback:
  154. /* I2S_IN-> DAC-> HP */
  155. /* Set power for DAC */
  156. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER3, 0x0C));
  157. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_dac, true));
  158. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_output_mixer, true));
  159. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_lineout, true));
  160. break;
  161. case wm8960_route_playback_and_record:
  162. /* Set power */
  163. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER3, 0x3C));
  164. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_adc, true));
  165. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_micbais, true));
  166. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_ana_in, true));
  167. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_dac, true));
  168. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_output_mixer, true));
  169. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_lineout, true));
  170. break;
  171. case wm8960_route_record:
  172. /* ANA_IN->ADC->I2S_OUT */
  173. /* Power up ADC and AIN */
  174. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER3, 0x30));
  175. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_micbais, true));
  176. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_ana_in, true));
  177. HPM_CHECK_RET(wm8960_set_module(control, wm8960_module_adc, true));
  178. break;
  179. default:
  180. stat = status_invalid_argument;
  181. break;
  182. }
  183. return stat;
  184. }
  185. hpm_stat_t wm8960_set_left_input(wm8960_control_t *control, wm8960_input_t input)
  186. {
  187. hpm_stat_t stat = status_success;
  188. uint16_t val = 0;
  189. switch (input) {
  190. case wm8960_input_closed:
  191. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  192. val &= (uint16_t) ~(WM8960_POWER1_AINL_MASK | WM8960_POWER1_ADCL_MASK);
  193. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  194. break;
  195. case wm8960_input_single_ended_mic:
  196. /* Only LMN1 enabled, LMICBOOST to 13db, LMIC2B enabled */
  197. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  198. val |= (WM8960_POWER1_AINL_MASK | WM8960_POWER1_ADCL_MASK | WM8960_POWER1_MICB_MASK);
  199. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  200. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINPATH, 0x138));
  201. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINVOL, 0x117));
  202. break;
  203. case wm8960_input_differential_mic_input2:
  204. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  205. val |= (WM8960_POWER1_AINL_MASK | WM8960_POWER1_ADCL_MASK | WM8960_POWER1_MICB_MASK);
  206. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  207. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINPATH, 0x178));
  208. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINVOL, 0x117));
  209. break;
  210. case wm8960_input_differential_mic_input3:
  211. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  212. val |= (WM8960_POWER1_AINL_MASK | WM8960_POWER1_ADCL_MASK | WM8960_POWER1_MICB_MASK);
  213. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  214. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINPATH, 0x1B8));
  215. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINVOL, 0x117));
  216. break;
  217. case wm8960_input_line_input2:
  218. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  219. val |= (WM8960_POWER1_AINL_MASK | WM8960_POWER1_ADCL_MASK);
  220. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  221. HPM_CHECK_RET(wm8960_read_reg(WM8960_INBMIX1, &val));
  222. val |= 0xEU;
  223. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_INBMIX1, val));
  224. break;
  225. case wm8960_input_line_input3:
  226. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  227. val |= (WM8960_POWER1_AINL_MASK | WM8960_POWER1_ADCL_MASK);
  228. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  229. HPM_CHECK_RET(wm8960_read_reg(WM8960_INBMIX1, &val));
  230. val |= 0x70U;
  231. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_INBMIX1, val));
  232. break;
  233. default:
  234. stat = status_invalid_argument;
  235. break;
  236. }
  237. return stat;
  238. }
  239. hpm_stat_t wm8960_set_right_input(wm8960_control_t *control, wm8960_input_t input)
  240. {
  241. hpm_stat_t stat = status_success;
  242. uint16_t val = 0;
  243. switch (input) {
  244. case wm8960_input_closed:
  245. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  246. val &= (uint16_t) ~(WM8960_POWER1_AINR_MASK | WM8960_POWER1_ADCR_MASK);
  247. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  248. break;
  249. case wm8960_input_single_ended_mic:
  250. /* Only LMN1 enabled, LMICBOOST to 13db, LMIC2B enabled */
  251. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  252. val |= (WM8960_POWER1_AINR_MASK | WM8960_POWER1_ADCR_MASK | WM8960_POWER1_MICB_MASK);
  253. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  254. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINPATH, 0x138));
  255. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINVOL, 0x117));
  256. break;
  257. case wm8960_input_differential_mic_input2:
  258. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  259. val |= (WM8960_POWER1_AINR_MASK | WM8960_POWER1_ADCR_MASK | WM8960_POWER1_MICB_MASK);
  260. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  261. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINPATH, 0x178));
  262. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINVOL, 0x117));
  263. break;
  264. case wm8960_input_differential_mic_input3:
  265. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  266. val |= (WM8960_POWER1_AINR_MASK | WM8960_POWER1_ADCR_MASK | WM8960_POWER1_MICB_MASK);
  267. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  268. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINPATH, 0x1B8));
  269. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINVOL, 0x117));
  270. break;
  271. case wm8960_input_line_input2:
  272. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  273. val |= (WM8960_POWER1_AINR_MASK | WM8960_POWER1_ADCR_MASK);
  274. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  275. HPM_CHECK_RET(wm8960_read_reg(WM8960_INBMIX2, &val));
  276. val |= 0xEU;
  277. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_INBMIX2, val));
  278. break;
  279. case wm8960_input_line_input3:
  280. HPM_CHECK_RET(wm8960_read_reg(WM8960_POWER1, &val));
  281. val |= (WM8960_POWER1_AINR_MASK | WM8960_POWER1_ADCR_MASK);
  282. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_POWER1, val));
  283. HPM_CHECK_RET(wm8960_read_reg(WM8960_INBMIX2, &val));
  284. val |= 0x70U;
  285. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_INBMIX2, val));
  286. break;
  287. default:
  288. stat = status_invalid_argument;
  289. break;
  290. }
  291. return stat;
  292. }
  293. hpm_stat_t wm8960_set_volume(wm8960_control_t *control, wm8960_module_t module, uint32_t volume)
  294. {
  295. uint16_t vol = 0;
  296. hpm_stat_t stat = status_success;
  297. switch (module) {
  298. case wm8960_module_adc:
  299. if (volume > 255U) {
  300. stat = status_invalid_argument;
  301. } else {
  302. vol = (uint16_t)volume;
  303. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LADC, vol));
  304. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RADC, vol));
  305. /* Update volume */
  306. vol = (uint16_t)(0x100U | volume);
  307. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LADC, vol));
  308. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RADC, vol));
  309. }
  310. break;
  311. case wm8960_module_dac:
  312. if (volume > 255U) {
  313. stat = status_invalid_argument;
  314. } else {
  315. vol = (uint16_t)volume;
  316. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LDAC, vol));
  317. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RDAC, vol));
  318. vol = 0x100U | (uint16_t)volume;
  319. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LDAC, vol));
  320. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RDAC, vol));
  321. }
  322. break;
  323. case wm8960_module_headphone:
  324. if (volume > 0x7FU) {
  325. stat = status_invalid_argument;
  326. } else {
  327. vol = (uint16_t)volume;
  328. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LOUT1, vol));
  329. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ROUT1, vol));
  330. vol = 0x100U | (uint16_t)volume;
  331. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LOUT1, vol));
  332. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ROUT1, vol));
  333. }
  334. break;
  335. case wm8960_module_ana_in:
  336. if (volume > 0x3FU) {
  337. stat = status_invalid_argument;
  338. } else {
  339. vol = (uint16_t)volume;
  340. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINVOL, vol));
  341. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINVOL, vol));
  342. vol = 0x100U | (uint16_t)volume;
  343. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LINVOL, vol));
  344. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_RINVOL, vol));
  345. }
  346. break;
  347. case wm8960_module_speaker:
  348. if (volume > 0x7FU) {
  349. stat = status_invalid_argument;
  350. } else {
  351. vol = (uint16_t)volume;
  352. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LOUT2, vol));
  353. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ROUT2, vol));
  354. vol = 0x100U | (uint16_t)volume;
  355. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_LOUT2, vol));
  356. HPM_CHECK_RET(wm8960_write_reg(control, WM8960_ROUT2, vol));
  357. }
  358. break;
  359. default:
  360. stat = status_invalid_argument;
  361. break;
  362. }
  363. return stat;
  364. }
  365. hpm_stat_t wm8960_set_data_format(wm8960_control_t *control, uint32_t sysclk, uint32_t sample_rate, uint32_t bits)
  366. {
  367. hpm_stat_t stat = status_success;
  368. uint32_t divider = 0;
  369. uint16_t val = 0;
  370. /* Compute sample rate divider and SYSCLK Pre-divider, dac and adc are the same sample rate */
  371. divider = sysclk / sample_rate;
  372. if (divider >= 512) {
  373. divider = divider / 2; /* SYSCLK Pre-divider */
  374. val |= WM8960_CLOCK1_SYSCLKDIV_SET(2U);
  375. }
  376. if (divider < 256U) {
  377. return status_invalid_argument;
  378. } else if (divider == 256U) {
  379. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_CLOCK1, 0x1FEU, val));
  380. } else {
  381. val |= (((divider / 256U) << WM8960_CLOCK1_ADCDIV_SHIFT) | ((divider / 256U) << WM8960_CLOCK1_DACDIV_SHIFT));
  382. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_CLOCK1, 0x1FEU, val));
  383. }
  384. /* set sample bit */
  385. switch (bits) {
  386. case 16:
  387. stat = wm8960_modify_reg(control, WM8960_IFACE1, WM8960_IFACE1_WL_MASK, WM8960_IFACE1_WL_SET(0U));
  388. break;
  389. case 20:
  390. stat = wm8960_modify_reg(control, WM8960_IFACE1, WM8960_IFACE1_WL_MASK, WM8960_IFACE1_WL_SET(1U));
  391. break;
  392. case 24:
  393. stat = wm8960_modify_reg(control, WM8960_IFACE1, WM8960_IFACE1_WL_MASK, WM8960_IFACE1_WL_SET(2U));
  394. break;
  395. case 32:
  396. stat = wm8960_modify_reg(control, WM8960_IFACE1, WM8960_IFACE1_WL_MASK, WM8960_IFACE1_WL_SET(3U));
  397. break;
  398. default:
  399. stat = status_invalid_argument;
  400. break;
  401. }
  402. return stat;
  403. }
  404. hpm_stat_t wm8960_config_input_to_output_mixer(wm8960_control_t *control, uint32_t play_source)
  405. {
  406. hpm_stat_t stat = status_success;
  407. if ((play_source & (uint32_t)wm8960_play_source_input_mixer) != 0U) {
  408. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_BYPASS1, 0x80U, 0x80U));
  409. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_BYPASS2, 0x80U, 0x80U));
  410. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_LOUTMIX, 0x180U, 0U));
  411. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_ROUTMIX, 0x180U, 0U));
  412. }
  413. if ((play_source & (uint32_t)wm8960_play_source_dac) != 0U) {
  414. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_BYPASS1, 0x80U, 0x00U));
  415. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_BYPASS2, 0x80U, 0x00U));
  416. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_LOUTMIX, 0x180U, 0x100U));
  417. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_ROUTMIX, 0x180U, 0x100U));
  418. }
  419. if ((play_source & (uint32_t)wm8960_play_source_input3) != 0U) {
  420. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_BYPASS1, 0x80U, 0x0U));
  421. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_BYPASS2, 0x80U, 0x0U));
  422. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_LOUTMIX, 0x180U, 0x80U));
  423. HPM_CHECK_RET(wm8960_modify_reg(control, WM8960_ROUTMIX, 0x180U, 0x80U));
  424. }
  425. return stat;
  426. }
  427. hpm_stat_t wm8960_write_reg(wm8960_control_t *control, uint8_t reg, uint16_t val)
  428. {
  429. /* The first 7 bits (B15 to B9) are address bits that select which control register */
  430. /* is accessed. The remaining 9 bits (B8 to B0) are data bits */
  431. rt_size_t size;
  432. rt_uint8_t data[2];
  433. data[0] = (reg << 1) | (uint8_t)((val >> 8U) & 0x0001U);
  434. data[1] = (uint8_t)(val & 0xFFU);
  435. size = rt_i2c_master_send(control->i2c_bus, control->slave_address, RT_I2C_WR, data, 2U);
  436. if (size != 2) {
  437. return status_fail;
  438. }
  439. wm8960_reg_val[reg] = val;
  440. return status_success;
  441. }
  442. hpm_stat_t wm8960_read_reg(uint8_t reg, uint16_t *val)
  443. {
  444. if (reg >= WM8960_REG_NUM) {
  445. return status_invalid_argument;
  446. }
  447. *val = wm8960_reg_val[reg];
  448. return status_success;
  449. }
  450. hpm_stat_t wm8960_modify_reg(wm8960_control_t *control, uint8_t reg, uint16_t mask, uint16_t val)
  451. {
  452. hpm_stat_t stat = 0;
  453. uint16_t reg_val;
  454. /* Read the register value out */
  455. stat = wm8960_read_reg(reg, &reg_val);
  456. if (stat != status_success) {
  457. return status_fail;
  458. }
  459. /* Modify the value */
  460. reg_val &= (uint16_t)~mask;
  461. reg_val |= val;
  462. /* Write the data to register */
  463. stat = wm8960_write_reg(control, reg, reg_val);
  464. if (stat != status_success) {
  465. return status_fail;
  466. }
  467. return status_success;
  468. }