msm_common.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. /* Copyright (c) 2020, The Linux Foundation. All rights reserved.
  2. *
  3. * This program is free software; you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License version 2 and
  5. * only version 2 as published by the Free Software Foundation.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU General Public License for more details.
  11. */
  12. #include <linux/gpio.h>
  13. #include <linux/of_gpio.h>
  14. #include <linux/platform_device.h>
  15. #include <linux/slab.h>
  16. #include <linux/of_device.h>
  17. #include <asoc/msm-cdc-pinctrl.h>
  18. #include <dsp/gecko-core.h>
  19. #include <dsp/msm_audio_ion.h>
  20. #include "msm_common.h"
  21. #define to_asoc_mach_common_pdata(kobj) \
  22. container_of((kobj), struct msm_common_pdata, aud_dev_kobj)
  23. #define DEVICE_ENABLE 1
  24. #define DEVICE_DISABLE 0
  25. static struct attribute device_state_attr = {
  26. .name = "state",
  27. .mode = 0660,
  28. };
  29. #define MAX_USR_INPUT 10
  30. static ssize_t aud_dev_sysfs_store(struct kobject *kobj,
  31. struct attribute *attr,
  32. const char *buf, size_t count)
  33. {
  34. ssize_t ret = -EINVAL;
  35. struct msm_common_pdata *pdata = to_asoc_mach_common_pdata(kobj);
  36. uint32_t pcm_id, state = 0;
  37. if (count > MAX_USR_INPUT) {
  38. pr_err("%s: invalid string written", __func__);
  39. goto done;
  40. }
  41. sscanf(buf, "%d %d", &pcm_id, &state);
  42. if ((pcm_id > pdata->num_aud_devs) || (pcm_id < 0)) {
  43. pr_err("%s: invalid pcm id %d \n", __func__, pcm_id);
  44. goto done;
  45. }
  46. if ((state > DEVICE_ENABLE) || (state < DEVICE_DISABLE)) {
  47. pr_err("%s: invalid state %d \n", __func__, state);
  48. goto done;
  49. }
  50. pr_info("%s: pcm_id %d state %d \n", __func__, pcm_id, state);
  51. pdata->aud_dev_state[pcm_id] = state;
  52. if ( state == DEVICE_ENABLE && (pdata->dsp_sessions_closed != 0))
  53. pdata->dsp_sessions_closed = 0;
  54. ret = count;
  55. done:
  56. return ret;
  57. }
  58. static const struct sysfs_ops aud_dev_sysfs_ops = {
  59. .store = aud_dev_sysfs_store,
  60. };
  61. static struct kobj_type aud_dev_ktype = {
  62. .sysfs_ops = &aud_dev_sysfs_ops,
  63. };
  64. static int aud_dev_sysfs_init(struct msm_common_pdata *pdata)
  65. {
  66. int ret = 0;
  67. char dir[10] = "aud_dev";
  68. ret = kobject_init_and_add(&pdata->aud_dev_kobj, &aud_dev_ktype,
  69. kernel_kobj, dir);
  70. if (ret < 0) {
  71. pr_err("%s: Failed to add kobject %s, err = %d\n",
  72. __func__, dir, ret);
  73. goto done;
  74. }
  75. ret = sysfs_create_file(&pdata->aud_dev_kobj, &device_state_attr);
  76. if (ret < 0) {
  77. pr_err("%s: Failed to add wdsp_boot sysfs entry to %s\n",
  78. __func__, dir);
  79. goto fail_create_file;
  80. }
  81. return ret;
  82. fail_create_file:
  83. kobject_put(&pdata->aud_dev_kobj);
  84. done:
  85. return ret;
  86. }
  87. static void check_userspace_service_state(struct snd_soc_pcm_runtime *rtd,
  88. struct msm_common_pdata *pdata)
  89. {
  90. dev_info(rtd->card->dev,"%s: pcm_id %d state %d\n", __func__,
  91. rtd->num, pdata->aud_dev_state[rtd->num]);
  92. if (pdata->aud_dev_state[rtd->num] == DEVICE_ENABLE) {
  93. dev_info(rtd->card->dev, "%s userspace service crashed\n",
  94. __func__);
  95. if (pdata->dsp_sessions_closed == 0) {
  96. /*Issue close all graph cmd to DSP*/
  97. gecko_core_apm_close_all();
  98. /*unmap all dma mapped buffers*/
  99. msm_audio_ion_crash_handler();
  100. pdata->dsp_sessions_closed = 1;
  101. }
  102. /*Reset the state as sysfs node wont be triggred*/
  103. pdata->aud_dev_state[rtd->num] = 0;
  104. }
  105. }
  106. static int get_intf_index(const char *stream_name)
  107. {
  108. if (strnstr(stream_name, "PRIMARY", strlen("PRIMARY")))
  109. return PRI_MI2S_TDM_AUXPCM;
  110. else if (strnstr(stream_name, "SECONDARY", strlen("SECONDARY")))
  111. return SEC_MI2S_TDM_AUXPCM;
  112. else if (strnstr(stream_name, "TERTIARY", strlen("TERTIARY")))
  113. return TER_MI2S_TDM_AUXPCM;
  114. else if (strnstr(stream_name, "QUATERNARY", strlen("QUATERNARY")))
  115. return QUAT_MI2S_TDM_AUXPCM;
  116. else if (strnstr(stream_name, "QUINARY", strlen("QUINARY")))
  117. return QUIN_MI2S_TDM_AUXPCM;
  118. else if (strnstr(stream_name, "SENARY", strlen("SENARY")))
  119. return SEN_MI2S_TDM_AUXPCM;
  120. else
  121. return -EINVAL;
  122. }
  123. int msm_common_snd_startup(struct snd_pcm_substream *substream)
  124. {
  125. int ret = 0;
  126. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  127. struct snd_soc_card *card = rtd->card;
  128. struct msm_common_pdata *pdata = msm_common_get_pdata(card);
  129. const char *stream_name = rtd->dai_link->stream_name;
  130. int index = get_intf_index(stream_name);
  131. dev_dbg(rtd->card->dev,
  132. "%s: substream = %s stream = %d\n",
  133. __func__, substream->name, substream->stream);
  134. if (!pdata) {
  135. dev_err(rtd->card->dev, "%s: pdata is NULL\n", __func__);
  136. return -EINVAL;
  137. }
  138. if (index >= 0) {
  139. mutex_lock(&pdata->lock[index]);
  140. if (pdata->mi2s_gpio_p[index]) {
  141. if (atomic_read(&(pdata->mi2s_gpio_ref_cnt[index])) == 0) {
  142. ret = msm_cdc_pinctrl_select_active_state(
  143. pdata->mi2s_gpio_p[index]);
  144. if (ret) {
  145. pr_err("%s:pinctrl set actve fail with %d\n",
  146. __func__, ret);
  147. goto done;
  148. }
  149. }
  150. atomic_inc(&(pdata->mi2s_gpio_ref_cnt[index]));
  151. }
  152. done:
  153. mutex_unlock(&pdata->lock[index]);
  154. }
  155. return ret;
  156. }
  157. void msm_common_snd_shutdown(struct snd_pcm_substream *substream)
  158. {
  159. int ret;
  160. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  161. struct snd_soc_card *card = rtd->card;
  162. struct msm_common_pdata *pdata = msm_common_get_pdata(card);
  163. const char *stream_name = rtd->dai_link->stream_name;
  164. int index = get_intf_index(stream_name);
  165. pr_debug("%s(): substream = %s stream = %d\n", __func__,
  166. substream->name, substream->stream);
  167. if (!pdata) {
  168. dev_err(card->dev, "%s: pdata is NULL\n", __func__);
  169. return;
  170. }
  171. check_userspace_service_state(rtd, pdata);
  172. if (index >= 0) {
  173. mutex_lock(&pdata->lock[index]);
  174. if (pdata->mi2s_gpio_p[index]) {
  175. atomic_dec(&pdata->mi2s_gpio_ref_cnt[index]);
  176. if (atomic_read(&pdata->mi2s_gpio_ref_cnt[index]) == 0) {
  177. ret = msm_cdc_pinctrl_select_active_state(
  178. pdata->mi2s_gpio_p[index]);
  179. if (ret)
  180. dev_err(card->dev,
  181. "%s: pinctrl set actv fail %d\n",
  182. __func__, ret);
  183. }
  184. }
  185. mutex_unlock(&pdata->lock[index]);
  186. }
  187. }
  188. int msm_common_snd_init(struct platform_device *pdev, struct snd_soc_card *card)
  189. {
  190. struct msm_common_pdata *common_pdata = NULL;
  191. int count;
  192. common_pdata = kcalloc(1, sizeof(struct msm_common_pdata), GFP_KERNEL);
  193. if (!common_pdata)
  194. return -ENOMEM;
  195. for (count = 0; count < MI2S_TDM_AUXPCM_MAX; count++) {
  196. mutex_init(&common_pdata->lock[count]);
  197. atomic_set(&common_pdata->mi2s_gpio_ref_cnt[count], 0);
  198. }
  199. common_pdata->mi2s_gpio_p[PRI_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  200. "qcom,pri-mi2s-gpios", 0);
  201. common_pdata->mi2s_gpio_p[SEC_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  202. "qcom,sec-mi2s-gpios", 0);
  203. common_pdata->mi2s_gpio_p[TER_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  204. "qcom,tert-mi2s-gpios", 0);
  205. common_pdata->mi2s_gpio_p[QUAT_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  206. "qcom,quat-mi2s-gpios", 0);
  207. common_pdata->mi2s_gpio_p[QUIN_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  208. "qcom,quin-mi2s-gpios", 0);
  209. common_pdata->mi2s_gpio_p[SEN_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  210. "qcom,sen-mi2s-gpios", 0);
  211. common_pdata->aud_dev_state = devm_kcalloc(&pdev->dev, card->num_links,
  212. sizeof(uint8_t), GFP_KERNEL);
  213. dev_info(&pdev->dev, "num_links %d \n", card->num_links);
  214. common_pdata->num_aud_devs = card->num_links;
  215. aud_dev_sysfs_init(common_pdata);
  216. msm_common_set_pdata(card, common_pdata);
  217. return 0;
  218. };
  219. void msm_common_snd_deinit(struct msm_common_pdata *common_pdata)
  220. {
  221. int count;
  222. if (!common_pdata)
  223. return;
  224. for (count = 0; count < MI2S_TDM_AUXPCM_MAX; count++) {
  225. mutex_destroy(&common_pdata->lock[count]);
  226. }
  227. }