msm-dts-srs-tm-config.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2012-2014, 2016-2018, 2020, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/err.h>
  6. #include <linux/module.h>
  7. #include <linux/bitops.h>
  8. #include <linux/mutex.h>
  9. #include <linux/atomic.h>
  10. #include <sound/control.h>
  11. #include <dsp/msm_audio_ion.h>
  12. #include <dsp/q6adm-v2.h>
  13. #include <dsp/msm-dts-srs-tm-config.h>
  14. static int srs_port_id[AFE_MAX_PORTS] = {-1};
  15. static int srs_copp_idx[AFE_MAX_PORTS] = {-1};
  16. static union srs_trumedia_params_u msm_srs_trumedia_params;
  17. struct dma_buf *dma_buf;
  18. static struct param_outband po;
  19. static atomic_t ref_cnt;
  20. #define ION_MEM_SIZE (8 * 1024)
  21. static int set_port_id(int port_id, int copp_idx)
  22. {
  23. int index = adm_validate_and_get_port_index(port_id);
  24. if (index < 0) {
  25. pr_err("%s: Invalid port idx %d port_id %#x\n", __func__, index,
  26. port_id);
  27. return -EINVAL;
  28. }
  29. srs_port_id[index] = port_id;
  30. srs_copp_idx[index] = copp_idx;
  31. return 0;
  32. }
  33. static void msm_dts_srs_tm_send_params(__s32 port_id, __u32 techs)
  34. {
  35. __s32 index = adm_validate_and_get_port_index(port_id);
  36. if (index < 0) {
  37. pr_err("%s: Invalid port idx %d port_id 0x%x\n",
  38. __func__, index, port_id);
  39. return;
  40. }
  41. if ((srs_copp_idx[index] < 0) ||
  42. (srs_copp_idx[index] >= MAX_COPPS_PER_PORT)) {
  43. pr_debug("%s: send params called before copp open. so, caching\n",
  44. __func__);
  45. return;
  46. }
  47. pr_debug("SRS %s: called, port_id = %d, techs flags = %u\n",
  48. __func__, port_id, techs);
  49. /* force all if techs is set to 1 */
  50. if (techs == 1)
  51. techs = 0xFFFFFFFF;
  52. if (techs & (1 << SRS_ID_WOWHD))
  53. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_WOWHD,
  54. (void *)&msm_srs_trumedia_params.srs_params.wowhd);
  55. if (techs & (1 << SRS_ID_CSHP))
  56. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_CSHP,
  57. (void *)&msm_srs_trumedia_params.srs_params.cshp);
  58. if (techs & (1 << SRS_ID_HPF))
  59. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_HPF,
  60. (void *)&msm_srs_trumedia_params.srs_params.hpf);
  61. if (techs & (1 << SRS_ID_AEQ))
  62. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_AEQ,
  63. (void *)&msm_srs_trumedia_params.srs_params.aeq);
  64. if (techs & (1 << SRS_ID_HL))
  65. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_HL,
  66. (void *)&msm_srs_trumedia_params.srs_params.hl);
  67. if (techs & (1 << SRS_ID_GEQ))
  68. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_GEQ,
  69. (void *)&msm_srs_trumedia_params.srs_params.geq);
  70. if (techs & (1 << SRS_ID_GLOBAL))
  71. srs_trumedia_open(port_id, srs_copp_idx[index], SRS_ID_GLOBAL,
  72. (void *)&msm_srs_trumedia_params.srs_params.global);
  73. }
  74. static int msm_dts_srs_trumedia_control_get(struct snd_kcontrol *kcontrol,
  75. struct snd_ctl_elem_value *ucontrol)
  76. {
  77. ucontrol->value.integer.value[0] = 0;
  78. return 0;
  79. }
  80. static int msm_dts_srs_trumedia_control_set_(int port_id,
  81. struct snd_kcontrol *kcontrol,
  82. struct snd_ctl_elem_value *ucontrol)
  83. {
  84. __u16 offset, value, max = sizeof(msm_srs_trumedia_params) >> 1;
  85. if (SRS_CMD_UPLOAD ==
  86. (ucontrol->value.integer.value[0] & SRS_CMD_UPLOAD)) {
  87. __u32 techs = ucontrol->value.integer.value[0] & 0xFF;
  88. __s32 index = adm_validate_and_get_port_index(port_id);
  89. if (index < 0) {
  90. pr_err("%s: Invalid port idx %d port_id 0x%x\n",
  91. __func__, index, port_id);
  92. return -EINVAL;
  93. }
  94. pr_debug("SRS %s: send params request, flag = %u\n",
  95. __func__, techs);
  96. if (srs_port_id[index] >= 0 && techs)
  97. msm_dts_srs_tm_send_params(port_id, techs);
  98. return 0;
  99. }
  100. offset = (__u16)((ucontrol->value.integer.value[0] &
  101. SRS_PARAM_OFFSET_MASK) >> 16);
  102. value = (__u16)(ucontrol->value.integer.value[0] &
  103. SRS_PARAM_VALUE_MASK);
  104. if (offset < max) {
  105. msm_srs_trumedia_params.raw_params[offset] = value;
  106. pr_debug("SRS %s: index set... (max %d, requested %d, value 0x%X)\n",
  107. __func__, max, offset, value);
  108. } else {
  109. pr_err("SRS %s: index out of bounds! (max %d, requested %d)\n",
  110. __func__, max, offset);
  111. }
  112. return 0;
  113. }
  114. static int msm_dts_srs_trumedia_control_set(struct snd_kcontrol *kcontrol,
  115. struct snd_ctl_elem_value *ucontrol)
  116. {
  117. int ret, port_id;
  118. pr_debug("SRS control normal called\n");
  119. msm_dts_srs_acquire_lock();
  120. port_id = SLIMBUS_0_RX;
  121. ret = msm_dts_srs_trumedia_control_set_(port_id, kcontrol, ucontrol);
  122. msm_dts_srs_release_lock();
  123. return ret;
  124. }
  125. static int msm_dts_srs_trumedia_control_i2s_set(struct snd_kcontrol *kcontrol,
  126. struct snd_ctl_elem_value *ucontrol)
  127. {
  128. int ret, port_id;
  129. pr_debug("SRS control I2S called\n");
  130. msm_dts_srs_acquire_lock();
  131. port_id = PRIMARY_I2S_RX;
  132. ret = msm_dts_srs_trumedia_control_set_(port_id, kcontrol, ucontrol);
  133. msm_dts_srs_release_lock();
  134. return ret;
  135. }
  136. static int msm_dts_srs_trumedia_control_mi2s_set(struct snd_kcontrol *kcontrol,
  137. struct snd_ctl_elem_value *ucontrol)
  138. {
  139. int ret, port_id;
  140. pr_debug("SRS control MI2S called\n");
  141. msm_dts_srs_acquire_lock();
  142. port_id = AFE_PORT_ID_PRIMARY_MI2S_RX;
  143. ret = msm_dts_srs_trumedia_control_set_(port_id, kcontrol, ucontrol);
  144. msm_dts_srs_release_lock();
  145. return ret;
  146. }
  147. static int msm_dts_srs_trumedia_control_hdmi_set(struct snd_kcontrol *kcontrol,
  148. struct snd_ctl_elem_value *ucontrol)
  149. {
  150. int ret, port_id;
  151. pr_debug("SRS control HDMI called\n");
  152. msm_dts_srs_acquire_lock();
  153. port_id = HDMI_RX;
  154. ret = msm_dts_srs_trumedia_control_set_(port_id, kcontrol, ucontrol);
  155. msm_dts_srs_release_lock();
  156. return ret;
  157. }
  158. static const struct snd_kcontrol_new lpa_srs_trumedia_controls[] = {
  159. {.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
  160. .name = "SRS TruMedia",
  161. .access = SNDRV_CTL_ELEM_ACCESS_TLV_READ |
  162. SNDRV_CTL_ELEM_ACCESS_READWRITE,
  163. .info = snd_soc_info_volsw,
  164. .get = msm_dts_srs_trumedia_control_get,
  165. .put = msm_dts_srs_trumedia_control_set,
  166. .private_value = ((unsigned long)&(struct soc_mixer_control)
  167. {.reg = SND_SOC_NOPM,
  168. .rreg = SND_SOC_NOPM,
  169. .shift = 0,
  170. .rshift = 0,
  171. .max = 0xFFFFFFFF,
  172. .platform_max = 0xFFFFFFFF,
  173. .invert = 0
  174. })
  175. }
  176. };
  177. static const struct snd_kcontrol_new lpa_srs_trumedia_controls_hdmi[] = {
  178. {.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
  179. .name = "SRS TruMedia HDMI",
  180. .access = SNDRV_CTL_ELEM_ACCESS_TLV_READ |
  181. SNDRV_CTL_ELEM_ACCESS_READWRITE,
  182. .info = snd_soc_info_volsw,
  183. .get = msm_dts_srs_trumedia_control_get,
  184. .put = msm_dts_srs_trumedia_control_hdmi_set,
  185. .private_value = ((unsigned long)&(struct soc_mixer_control)
  186. {.reg = SND_SOC_NOPM,
  187. .rreg = SND_SOC_NOPM,
  188. .shift = 0,
  189. .rshift = 0,
  190. .max = 0xFFFFFFFF,
  191. .platform_max = 0xFFFFFFFF,
  192. .invert = 0
  193. })
  194. }
  195. };
  196. static const struct snd_kcontrol_new lpa_srs_trumedia_controls_i2s[] = {
  197. {.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
  198. .name = "SRS TruMedia I2S",
  199. .access = SNDRV_CTL_ELEM_ACCESS_TLV_READ |
  200. SNDRV_CTL_ELEM_ACCESS_READWRITE,
  201. .info = snd_soc_info_volsw,
  202. .get = msm_dts_srs_trumedia_control_get,
  203. .put = msm_dts_srs_trumedia_control_i2s_set,
  204. .private_value = ((unsigned long)&(struct soc_mixer_control)
  205. {.reg = SND_SOC_NOPM,
  206. .rreg = SND_SOC_NOPM,
  207. .shift = 0,
  208. .rshift = 0,
  209. .max = 0xFFFFFFFF,
  210. .platform_max = 0xFFFFFFFF,
  211. .invert = 0
  212. })
  213. }
  214. };
  215. static const struct snd_kcontrol_new lpa_srs_trumedia_controls_mi2s[] = {
  216. {
  217. .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
  218. .name = "SRS TruMedia MI2S",
  219. .access = SNDRV_CTL_ELEM_ACCESS_TLV_READ |
  220. SNDRV_CTL_ELEM_ACCESS_READWRITE,
  221. .info = snd_soc_info_volsw,
  222. .get = msm_dts_srs_trumedia_control_get,
  223. .put = msm_dts_srs_trumedia_control_mi2s_set,
  224. .private_value = ((unsigned long)&(struct soc_mixer_control)
  225. {
  226. .reg = SND_SOC_NOPM,
  227. .rreg = SND_SOC_NOPM,
  228. .shift = 0,
  229. .rshift = 0,
  230. .max = 0xFFFFFFFF,
  231. .platform_max = 0xFFFFFFFF,
  232. .invert = 0
  233. })
  234. }
  235. };
  236. /**
  237. * msm_dts_srs_tm_add_controls -
  238. * Add DTS SRS module controls
  239. *
  240. * @component: component to which controls can be registered
  241. *
  242. */
  243. void msm_dts_srs_tm_add_controls(struct snd_soc_component *component)
  244. {
  245. snd_soc_add_component_controls(component,
  246. lpa_srs_trumedia_controls,
  247. ARRAY_SIZE(lpa_srs_trumedia_controls));
  248. snd_soc_add_component_controls(component,
  249. lpa_srs_trumedia_controls_hdmi,
  250. ARRAY_SIZE(lpa_srs_trumedia_controls_hdmi));
  251. snd_soc_add_component_controls(component,
  252. lpa_srs_trumedia_controls_i2s,
  253. ARRAY_SIZE(lpa_srs_trumedia_controls_i2s));
  254. snd_soc_add_component_controls(component,
  255. lpa_srs_trumedia_controls_mi2s,
  256. ARRAY_SIZE(lpa_srs_trumedia_controls_mi2s));
  257. }
  258. EXPORT_SYMBOL(msm_dts_srs_tm_add_controls);
  259. static int reg_ion_mem(void)
  260. {
  261. int rc;
  262. rc = msm_audio_ion_alloc(&dma_buf, ION_MEM_SIZE,
  263. &po.paddr, (size_t *)&po.size,
  264. &po.kvaddr);
  265. if (rc != 0)
  266. pr_err("%s: failed to allocate memory.\n", __func__);
  267. pr_debug("%s: exited dma_buf = %pK, phys_addr = %lu, length = %d, vaddr = %pK, rc = 0x%x\n",
  268. __func__, dma_buf, (long)po.paddr,
  269. (unsigned int)po.size, po.kvaddr, rc);
  270. return rc;
  271. }
  272. void msm_dts_srs_tm_ion_memmap(struct param_outband *po_)
  273. {
  274. if (po.kvaddr == NULL) {
  275. pr_debug("%s: callingreg_ion_mem()\n", __func__);
  276. reg_ion_mem();
  277. }
  278. po_->size = ION_MEM_SIZE;
  279. po_->kvaddr = po.kvaddr;
  280. po_->paddr = po.paddr;
  281. }
  282. static void unreg_ion_mem(void)
  283. {
  284. msm_audio_ion_free(dma_buf);
  285. dma_buf = NULL;
  286. po.kvaddr = NULL;
  287. po.paddr = 0;
  288. po.size = 0;
  289. }
  290. /**
  291. * msm_dts_srs_tm_deinit -
  292. * De-Initializes DTS SRS module
  293. *
  294. * @port_id: Port ID number
  295. *
  296. */
  297. void msm_dts_srs_tm_deinit(int port_id)
  298. {
  299. set_port_id(port_id, -1);
  300. atomic_dec(&ref_cnt);
  301. if (po.kvaddr != NULL) {
  302. if (!atomic_read(&ref_cnt)) {
  303. pr_debug("%s: calling unreg_ion_mem()\n", __func__);
  304. unreg_ion_mem();
  305. }
  306. }
  307. }
  308. EXPORT_SYMBOL(msm_dts_srs_tm_deinit);
  309. /**
  310. * msm_dts_srs_tm_init -
  311. * Initializes DTS SRS module
  312. *
  313. * @port_id: Port ID number
  314. * @copp_idx: COPP index
  315. *
  316. */
  317. void msm_dts_srs_tm_init(int port_id, int copp_idx)
  318. {
  319. int cur_ref_cnt = 0;
  320. if (set_port_id(port_id, copp_idx) < 0) {
  321. pr_err("%s: Invalid port_id: %d\n", __func__, port_id);
  322. return;
  323. }
  324. cur_ref_cnt = atomic_read(&ref_cnt);
  325. atomic_inc(&ref_cnt);
  326. if (!cur_ref_cnt && po.kvaddr == NULL) {
  327. pr_debug("%s: calling reg_ion_mem()\n", __func__);
  328. if (reg_ion_mem() != 0) {
  329. atomic_dec(&ref_cnt);
  330. po.kvaddr = NULL;
  331. return;
  332. }
  333. }
  334. msm_dts_srs_tm_send_params(port_id, 1);
  335. }
  336. EXPORT_SYMBOL(msm_dts_srs_tm_init);