msm-pcm-host-voice-v2.c 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551
  1. /* Copyright (c) 2013-2017, 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/init.h>
  13. #include <linux/err.h>
  14. #include <linux/module.h>
  15. #include <linux/time.h>
  16. #include <linux/wait.h>
  17. #include <linux/platform_device.h>
  18. #include <linux/slab.h>
  19. #include <linux/dma-mapping.h>
  20. #include <sound/core.h>
  21. #include <sound/soc.h>
  22. #include <sound/soc-dapm.h>
  23. #include <sound/pcm.h>
  24. #include <sound/initval.h>
  25. #include <sound/control.h>
  26. #include <asm/dma.h>
  27. #include <dsp/msm_audio_ion.h>
  28. #include <dsp/q6voice.h>
  29. #define HPCM_MAX_Q_LEN 2
  30. #define HPCM_MIN_VOC_PKT_SIZE 320
  31. #define HPCM_MAX_VOC_PKT_SIZE 640
  32. #define VHPCM_BLOCK_SIZE 4096
  33. #define CACHE_ALIGNMENT_SIZE 128
  34. #define CACHE_ALIGNMENT_MASK 0xFFFFFF80
  35. #define VOICE_TX_CAPTURE_DAI_ID "CS-VOICE HOST TX CAPTURE"
  36. #define VOICE_TX_PLAYBACK_DAI_ID "CS-VOICE HOST TX PLAYBACK"
  37. #define VOICE_RX_CAPTURE_DAI_ID "CS-VOICE HOST RX CAPTURE"
  38. #define VOICE_RX_PLAYBACK_DAI_ID "CS-VOICE HOST RX PLAYBACK"
  39. #define VOLTE_TX_CAPTURE_DAI_ID "VOLTE HOST TX CAPTURE"
  40. #define VOLTE_TX_PLAYBACK_DAI_ID "VOLTE HOST TX PLAYBACK"
  41. #define VOLTE_RX_CAPTURE_DAI_ID "VOLTE HOST RX CAPTURE"
  42. #define VOLTE_RX_PLAYBACK_DAI_ID "VOLTE HOST RX PLAYBACK"
  43. #define VoMMode1_TX_CAPTURE_DAI_ID "VoiceMMode1 HOST TX CAPTURE"
  44. #define VoMMode1_TX_PLAYBACK_DAI_ID "VoiceMMode1 HOST TX PLAYBACK"
  45. #define VoMMode1_RX_CAPTURE_DAI_ID "VoiceMMode1 HOST RX CAPTURE"
  46. #define VoMMode1_RX_PLAYBACK_DAI_ID "VoiceMMode1 HOST RX PLAYBACK"
  47. #define VoMMode2_TX_CAPTURE_DAI_ID "VoiceMMode2 HOST TX CAPTURE"
  48. #define VoMMode2_TX_PLAYBACK_DAI_ID "VoiceMMode2 HOST TX PLAYBACK"
  49. #define VoMMode2_RX_CAPTURE_DAI_ID "VoiceMMode2 HOST RX CAPTURE"
  50. #define VoMMode2_RX_PLAYBACK_DAI_ID "VoiceMMode2 HOST RX PLAYBACK"
  51. enum {
  52. RX = 1,
  53. TX,
  54. };
  55. enum {
  56. VOICE_INDEX = 0,
  57. VOLTE_INDEX,
  58. VOMMODE1_INDEX,
  59. VOMMODE2_INDEX,
  60. MAX_SESSION
  61. };
  62. enum hpcm_state {
  63. HPCM_STOPPED = 1,
  64. HPCM_CLOSED,
  65. HPCM_PREPARED,
  66. HPCM_STARTED,
  67. };
  68. struct hpcm_frame {
  69. uint32_t len;
  70. uint8_t voc_pkt[HPCM_MAX_VOC_PKT_SIZE];
  71. };
  72. struct hpcm_buf_node {
  73. struct list_head list;
  74. struct hpcm_frame frame;
  75. };
  76. struct vocpcm_ion_buffer {
  77. /* Physical address */
  78. phys_addr_t paddr;
  79. /* Kernel virtual address */
  80. void *kvaddr;
  81. };
  82. struct dai_data {
  83. enum hpcm_state state;
  84. struct snd_pcm_substream *substream;
  85. struct list_head filled_queue;
  86. struct list_head free_queue;
  87. wait_queue_head_t queue_wait;
  88. spinlock_t dsp_lock;
  89. uint32_t pcm_size;
  90. uint32_t pcm_count;
  91. /* IRQ position */
  92. uint32_t pcm_irq_pos;
  93. /* Position in buffer */
  94. uint32_t pcm_buf_pos;
  95. struct vocpcm_ion_buffer vocpcm_ion_buffer;
  96. };
  97. struct tap_point {
  98. struct dai_data playback_dai_data;
  99. struct dai_data capture_dai_data;
  100. };
  101. struct session {
  102. struct tap_point tx_tap_point;
  103. struct tap_point rx_tap_point;
  104. phys_addr_t sess_paddr;
  105. void *sess_kvaddr;
  106. struct ion_handle *ion_handle;
  107. struct mem_map_table tp_mem_table;
  108. };
  109. struct tappnt_mxr_data {
  110. bool enable;
  111. uint16_t direction;
  112. uint16_t sample_rate;
  113. };
  114. /* Values from mixer ctl are cached in this structure */
  115. struct mixer_conf {
  116. int8_t sess_indx;
  117. struct tappnt_mxr_data rx;
  118. struct tappnt_mxr_data tx;
  119. };
  120. struct start_cmd {
  121. struct vss_ivpcm_tap_point tap_pnt[2];
  122. uint32_t no_of_tapoints;
  123. };
  124. struct hpcm_drv {
  125. struct mutex lock;
  126. struct session session[MAX_SESSION];
  127. struct mixer_conf mixer_conf;
  128. struct ion_client *ion_client;
  129. struct start_cmd start_cmd;
  130. };
  131. static struct hpcm_drv hpcm_drv;
  132. static struct snd_pcm_hardware msm_pcm_hardware = {
  133. .info = (SNDRV_PCM_INFO_MMAP |
  134. SNDRV_PCM_INFO_BLOCK_TRANSFER |
  135. SNDRV_PCM_INFO_MMAP_VALID |
  136. SNDRV_PCM_INFO_INTERLEAVED),
  137. .formats = SNDRV_PCM_FMTBIT_S16_LE |
  138. SNDRV_PCM_FMTBIT_SPECIAL,
  139. .rates = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000,
  140. .rate_min = 8000,
  141. .rate_max = 16000,
  142. .channels_min = 1,
  143. .channels_max = 1,
  144. .buffer_bytes_max = sizeof(struct hpcm_buf_node) * HPCM_MAX_Q_LEN,
  145. .period_bytes_min = HPCM_MIN_VOC_PKT_SIZE,
  146. .period_bytes_max = HPCM_MAX_VOC_PKT_SIZE,
  147. .periods_min = HPCM_MAX_Q_LEN,
  148. .periods_max = HPCM_MAX_Q_LEN,
  149. .fifo_size = 0,
  150. };
  151. static char *hpcm_get_sess_name(int sess_indx)
  152. {
  153. char *sess_name = NULL;
  154. if (sess_indx == VOICE_INDEX)
  155. sess_name = VOICE_SESSION_NAME;
  156. else if (sess_indx == VOLTE_INDEX)
  157. sess_name = VOLTE_SESSION_NAME;
  158. else if (sess_indx == VOMMODE1_INDEX)
  159. sess_name = VOICEMMODE1_NAME;
  160. else if (sess_indx == VOMMODE2_INDEX)
  161. sess_name = VOICEMMODE2_NAME;
  162. else
  163. pr_err("%s:, Invalid sess_index\n", __func__);
  164. return sess_name;
  165. }
  166. static void hpcm_reset_mixer_config(struct hpcm_drv *prtd)
  167. {
  168. prtd->mixer_conf.sess_indx = -1;
  169. prtd->mixer_conf.rx.enable = false;
  170. prtd->mixer_conf.rx.direction = -1;
  171. prtd->mixer_conf.rx.sample_rate = 0;
  172. prtd->mixer_conf.tx.enable = false;
  173. prtd->mixer_conf.tx.direction = -1;
  174. prtd->mixer_conf.tx.sample_rate = 0;
  175. }
  176. /* Check for valid mixer control values */
  177. static bool hpcm_is_valid_config(int sess_indx, int tap_point,
  178. uint16_t direction, uint16_t samplerate)
  179. {
  180. if (sess_indx < VOICE_INDEX || sess_indx > VOMMODE2_INDEX) {
  181. pr_err("%s: invalid sess_indx :%d\n", __func__, sess_indx);
  182. goto error;
  183. }
  184. if (samplerate != VSS_IVPCM_SAMPLING_RATE_8K &&
  185. samplerate != VSS_IVPCM_SAMPLING_RATE_16K) {
  186. pr_err("%s: invalid sample rate :%d\n", __func__, samplerate);
  187. goto error;
  188. }
  189. if ((tap_point != RX) && (tap_point != TX)) {
  190. pr_err("%s: invalid tappoint :%d\n", __func__, tap_point);
  191. goto error;
  192. }
  193. if ((direction != VSS_IVPCM_TAP_POINT_DIR_IN) &&
  194. (direction != VSS_IVPCM_TAP_POINT_DIR_OUT) &&
  195. (direction != VSS_IVPCM_TAP_POINT_DIR_OUT_IN)) {
  196. pr_err("%s: invalid direction :%d\n", __func__, direction);
  197. goto error;
  198. }
  199. return true;
  200. error:
  201. return false;
  202. }
  203. static struct dai_data *hpcm_get_dai_data(char *pcm_id, struct hpcm_drv *prtd)
  204. {
  205. struct dai_data *dai_data = NULL;
  206. size_t size = 0;
  207. if (pcm_id) {
  208. size = strlen(pcm_id);
  209. /* Check for Voice DAI */
  210. if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, size)) {
  211. dai_data =
  212. &prtd->session[VOICE_INDEX].tx_tap_point.capture_dai_data;
  213. } else if (strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, size)) {
  214. dai_data =
  215. &prtd->session[VOICE_INDEX].tx_tap_point.playback_dai_data;
  216. } else if (strnstr(pcm_id, VOICE_RX_CAPTURE_DAI_ID, size)) {
  217. dai_data =
  218. &prtd->session[VOICE_INDEX].rx_tap_point.capture_dai_data;
  219. } else if (strnstr(pcm_id, VOICE_RX_PLAYBACK_DAI_ID, size)) {
  220. dai_data =
  221. &prtd->session[VOICE_INDEX].rx_tap_point.playback_dai_data;
  222. /* Check for VoLTE DAI */
  223. } else if (strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, size)) {
  224. dai_data =
  225. &prtd->session[VOLTE_INDEX].tx_tap_point.capture_dai_data;
  226. } else if (strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, size)) {
  227. dai_data =
  228. &prtd->session[VOLTE_INDEX].tx_tap_point.playback_dai_data;
  229. } else if (strnstr(pcm_id, VOLTE_RX_CAPTURE_DAI_ID, size)) {
  230. dai_data =
  231. &prtd->session[VOLTE_INDEX].rx_tap_point.capture_dai_data;
  232. } else if (strnstr(pcm_id, VOLTE_RX_PLAYBACK_DAI_ID, size)) {
  233. dai_data =
  234. &prtd->session[VOLTE_INDEX].rx_tap_point.playback_dai_data;
  235. /* check for VoiceMMode1 DAI */
  236. } else if (strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, size)) {
  237. dai_data =
  238. &prtd->session[VOMMODE1_INDEX].tx_tap_point.capture_dai_data;
  239. } else if (strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, size)) {
  240. dai_data =
  241. &prtd->session[VOMMODE1_INDEX].tx_tap_point.playback_dai_data;
  242. } else if (strnstr(pcm_id, VoMMode1_RX_CAPTURE_DAI_ID, size)) {
  243. dai_data =
  244. &prtd->session[VOMMODE1_INDEX].rx_tap_point.capture_dai_data;
  245. } else if (strnstr(pcm_id, VoMMode1_RX_PLAYBACK_DAI_ID, size)) {
  246. dai_data =
  247. &prtd->session[VOMMODE1_INDEX].rx_tap_point.playback_dai_data;
  248. /* check for VOiceMMode2 DAI */
  249. } else if (strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, size)) {
  250. dai_data =
  251. &prtd->session[VOMMODE2_INDEX].tx_tap_point.capture_dai_data;
  252. } else if (strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, size)) {
  253. dai_data =
  254. &prtd->session[VOMMODE2_INDEX].tx_tap_point.playback_dai_data;
  255. } else if (strnstr(pcm_id, VoMMode2_RX_CAPTURE_DAI_ID, size)) {
  256. dai_data =
  257. &prtd->session[VOMMODE2_INDEX].rx_tap_point.capture_dai_data;
  258. } else if (strnstr(pcm_id, VoMMode2_RX_PLAYBACK_DAI_ID, size)) {
  259. dai_data =
  260. &prtd->session[VOMMODE2_INDEX].rx_tap_point.playback_dai_data;
  261. } else {
  262. pr_err("%s: Wrong dai id\n", __func__);
  263. }
  264. }
  265. return dai_data;
  266. }
  267. static struct tap_point *hpcm_get_tappoint_data(char *pcm_id,
  268. struct hpcm_drv *prtd)
  269. {
  270. struct tap_point *tp = NULL;
  271. size_t size = 0;
  272. if (pcm_id) {
  273. size = strlen(pcm_id);
  274. /* Check for Voice DAI */
  275. if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, size)) {
  276. tp = &prtd->session[VOICE_INDEX].tx_tap_point;
  277. } else if (strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, size)) {
  278. tp = &prtd->session[VOICE_INDEX].tx_tap_point;
  279. } else if (strnstr(pcm_id, VOICE_RX_CAPTURE_DAI_ID, size)) {
  280. tp = &prtd->session[VOICE_INDEX].rx_tap_point;
  281. } else if (strnstr(pcm_id, VOICE_RX_PLAYBACK_DAI_ID, size)) {
  282. tp = &prtd->session[VOICE_INDEX].rx_tap_point;
  283. /* Check for VoLTE DAI */
  284. } else if (strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, size)) {
  285. tp = &prtd->session[VOLTE_INDEX].tx_tap_point;
  286. } else if (strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, size)) {
  287. tp = &prtd->session[VOLTE_INDEX].tx_tap_point;
  288. } else if (strnstr(pcm_id, VOLTE_RX_CAPTURE_DAI_ID, size)) {
  289. tp = &prtd->session[VOLTE_INDEX].rx_tap_point;
  290. } else if (strnstr(pcm_id, VOLTE_RX_PLAYBACK_DAI_ID, size)) {
  291. tp = &prtd->session[VOLTE_INDEX].rx_tap_point;
  292. /* check for VoiceMMode1 */
  293. } else if (strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, size)) {
  294. tp = &prtd->session[VOMMODE1_INDEX].tx_tap_point;
  295. } else if (strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, size)) {
  296. tp = &prtd->session[VOMMODE1_INDEX].tx_tap_point;
  297. } else if (strnstr(pcm_id, VoMMode1_RX_CAPTURE_DAI_ID, size)) {
  298. tp = &prtd->session[VOMMODE1_INDEX].rx_tap_point;
  299. } else if (strnstr(pcm_id, VoMMode1_RX_PLAYBACK_DAI_ID, size)) {
  300. tp = &prtd->session[VOMMODE1_INDEX].rx_tap_point;
  301. /* check for VoiceMMode2 */
  302. } else if (strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, size)) {
  303. tp = &prtd->session[VOMMODE2_INDEX].tx_tap_point;
  304. } else if (strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, size)) {
  305. tp = &prtd->session[VOMMODE2_INDEX].tx_tap_point;
  306. } else if (strnstr(pcm_id, VoMMode2_RX_CAPTURE_DAI_ID, size)) {
  307. tp = &prtd->session[VOMMODE2_INDEX].rx_tap_point;
  308. } else if (strnstr(pcm_id, VoMMode2_RX_PLAYBACK_DAI_ID, size)) {
  309. tp = &prtd->session[VOMMODE2_INDEX].rx_tap_point;
  310. } else {
  311. pr_err("%s: wrong dai id\n", __func__);
  312. }
  313. }
  314. return tp;
  315. }
  316. static struct tappnt_mxr_data *hpcm_get_tappnt_mixer_data(char *pcm_id,
  317. struct hpcm_drv *prtd)
  318. {
  319. if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  320. strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
  321. strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  322. strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
  323. strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  324. strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
  325. strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  326. strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, strlen(pcm_id))) {
  327. return &prtd->mixer_conf.tx;
  328. } else {
  329. return &prtd->mixer_conf.rx;
  330. }
  331. }
  332. static int get_tappnt_value(char *pcm_id)
  333. {
  334. if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  335. strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
  336. strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  337. strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
  338. strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  339. strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
  340. strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
  341. strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, strlen(pcm_id))) {
  342. return TX;
  343. } else {
  344. return RX;
  345. }
  346. }
  347. static bool hpcm_all_dais_are_ready(uint16_t direction, struct tap_point *tp,
  348. enum hpcm_state state)
  349. {
  350. bool dais_started = false;
  351. /*
  352. * Based on the direction set per tap point in the mixer control,
  353. * all the dais per tap point should meet the required state for the
  354. * commands such as vpcm_map_memory/vpcm_start to be executed.
  355. */
  356. switch (direction) {
  357. case VSS_IVPCM_TAP_POINT_DIR_OUT_IN:
  358. if ((tp->playback_dai_data.state >= state) &&
  359. (tp->capture_dai_data.state >= state)) {
  360. dais_started = true;
  361. }
  362. break;
  363. case VSS_IVPCM_TAP_POINT_DIR_IN:
  364. if (tp->playback_dai_data.state >= state)
  365. dais_started = true;
  366. break;
  367. case VSS_IVPCM_TAP_POINT_DIR_OUT:
  368. if (tp->capture_dai_data.state >= state)
  369. dais_started = true;
  370. break;
  371. default:
  372. pr_err("invalid direction\n");
  373. }
  374. return dais_started;
  375. }
  376. static void hpcm_create_free_queue(struct snd_dma_buffer *dma_buf,
  377. struct dai_data *dai_data)
  378. {
  379. struct hpcm_buf_node *buf_node = NULL;
  380. int i = 0, offset = 0;
  381. for (i = 0; i < HPCM_MAX_Q_LEN; i++) {
  382. buf_node = (void *)dma_buf->area + offset;
  383. list_add_tail(&buf_node->list,
  384. &dai_data->free_queue);
  385. offset = offset + sizeof(struct hpcm_buf_node);
  386. }
  387. }
  388. static void hpcm_free_allocated_mem(struct hpcm_drv *prtd)
  389. {
  390. phys_addr_t paddr = 0;
  391. struct tap_point *txtp = NULL;
  392. struct tap_point *rxtp = NULL;
  393. struct session *sess = NULL;
  394. sess = &prtd->session[prtd->mixer_conf.sess_indx];
  395. txtp = &sess->tx_tap_point;
  396. rxtp = &sess->rx_tap_point;
  397. paddr = sess->sess_paddr;
  398. if (paddr) {
  399. msm_audio_ion_free(prtd->ion_client, sess->ion_handle);
  400. prtd->ion_client = NULL;
  401. sess->ion_handle = NULL;
  402. msm_audio_ion_free(sess->tp_mem_table.client,
  403. sess->tp_mem_table.handle);
  404. sess->tp_mem_table.client = NULL;
  405. sess->tp_mem_table.handle = NULL;
  406. sess->sess_paddr = 0;
  407. sess->sess_kvaddr = 0;
  408. sess->ion_handle = 0;
  409. prtd->ion_client = 0;
  410. sess->tp_mem_table.client = 0;
  411. sess->tp_mem_table.handle = 0;
  412. txtp->capture_dai_data.vocpcm_ion_buffer.paddr = 0;
  413. txtp->capture_dai_data.vocpcm_ion_buffer.kvaddr = 0;
  414. txtp->playback_dai_data.vocpcm_ion_buffer.paddr = 0;
  415. txtp->playback_dai_data.vocpcm_ion_buffer.kvaddr = 0;
  416. rxtp->capture_dai_data.vocpcm_ion_buffer.paddr = 0;
  417. rxtp->capture_dai_data.vocpcm_ion_buffer.kvaddr = 0;
  418. rxtp->playback_dai_data.vocpcm_ion_buffer.paddr = 0;
  419. rxtp->playback_dai_data.vocpcm_ion_buffer.kvaddr = 0;
  420. } else {
  421. pr_debug("%s, paddr = 0, nothing to free\n", __func__);
  422. }
  423. }
  424. static void hpcm_unmap_and_free_shared_memory(struct hpcm_drv *prtd)
  425. {
  426. phys_addr_t paddr = 0;
  427. char *sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
  428. if (prtd->mixer_conf.sess_indx >= 0)
  429. paddr = prtd->session[prtd->mixer_conf.sess_indx].sess_paddr;
  430. else
  431. paddr = 0;
  432. if (paddr) {
  433. voc_send_cvp_unmap_vocpcm_memory(voc_get_session_id(sess_name));
  434. hpcm_free_allocated_mem(prtd);
  435. } else {
  436. pr_debug("%s, paddr = 0, nothing to unmap/free\n", __func__);
  437. }
  438. }
  439. static int hpcm_map_vocpcm_memory(struct hpcm_drv *prtd)
  440. {
  441. int ret = 0;
  442. char *sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
  443. struct session *sess = NULL;
  444. sess = &prtd->session[prtd->mixer_conf.sess_indx];
  445. ret = voc_send_cvp_map_vocpcm_memory(voc_get_session_id(sess_name),
  446. &sess->tp_mem_table,
  447. sess->sess_paddr,
  448. VHPCM_BLOCK_SIZE);
  449. return ret;
  450. }
  451. static int hpcm_allocate_shared_memory(struct hpcm_drv *prtd)
  452. {
  453. int result;
  454. int ret = 0;
  455. size_t mem_len;
  456. size_t len;
  457. struct tap_point *txtp = NULL;
  458. struct tap_point *rxtp = NULL;
  459. struct session *sess = NULL;
  460. sess = &prtd->session[prtd->mixer_conf.sess_indx];
  461. txtp = &sess->tx_tap_point;
  462. rxtp = &sess->rx_tap_point;
  463. result = msm_audio_ion_alloc("host_pcm_buffer",
  464. &prtd->ion_client,
  465. &sess->ion_handle,
  466. VHPCM_BLOCK_SIZE,
  467. &sess->sess_paddr,
  468. &mem_len,
  469. &sess->sess_kvaddr);
  470. if (result) {
  471. pr_err("%s: msm_audio_ion_alloc error, rc = %d\n",
  472. __func__, result);
  473. sess->sess_paddr = 0;
  474. sess->sess_kvaddr = 0;
  475. ret = -ENOMEM;
  476. goto done;
  477. }
  478. pr_debug("%s: Host PCM memory block allocated\n", __func__);
  479. /* Allocate mem_map_table for tap point */
  480. result = msm_audio_ion_alloc("host_pcm_table",
  481. &sess->tp_mem_table.client,
  482. &sess->tp_mem_table.handle,
  483. sizeof(struct vss_imemory_table_t),
  484. &sess->tp_mem_table.phys,
  485. &len,
  486. &sess->tp_mem_table.data);
  487. if (result) {
  488. pr_err("%s: msm_audio_ion_alloc error, rc = %d\n",
  489. __func__, result);
  490. msm_audio_ion_free(prtd->ion_client, sess->ion_handle);
  491. prtd->ion_client = NULL;
  492. sess->ion_handle = NULL;
  493. sess->sess_paddr = 0;
  494. sess->sess_kvaddr = 0;
  495. ret = -ENOMEM;
  496. goto done;
  497. }
  498. pr_debug("%s: Host PCM memory table allocated\n", __func__);
  499. memset(sess->tp_mem_table.data, 0,
  500. sizeof(struct vss_imemory_table_t));
  501. sess->tp_mem_table.size = sizeof(struct vss_imemory_table_t);
  502. pr_debug("%s: data %pK phys %pK\n", __func__,
  503. sess->tp_mem_table.data, &sess->tp_mem_table.phys);
  504. /* Split 4096 block into four 1024 byte blocks for each dai */
  505. txtp->capture_dai_data.vocpcm_ion_buffer.paddr =
  506. sess->sess_paddr;
  507. txtp->capture_dai_data.vocpcm_ion_buffer.kvaddr =
  508. sess->sess_kvaddr;
  509. txtp->playback_dai_data.vocpcm_ion_buffer.paddr =
  510. sess->sess_paddr + VHPCM_BLOCK_SIZE/4;
  511. txtp->playback_dai_data.vocpcm_ion_buffer.kvaddr =
  512. sess->sess_kvaddr + VHPCM_BLOCK_SIZE/4;
  513. rxtp->capture_dai_data.vocpcm_ion_buffer.paddr =
  514. sess->sess_paddr + (VHPCM_BLOCK_SIZE/4) * 2;
  515. rxtp->capture_dai_data.vocpcm_ion_buffer.kvaddr =
  516. sess->sess_kvaddr + (VHPCM_BLOCK_SIZE/4) * 2;
  517. rxtp->playback_dai_data.vocpcm_ion_buffer.paddr =
  518. sess->sess_paddr + (VHPCM_BLOCK_SIZE/4) * 3;
  519. rxtp->playback_dai_data.vocpcm_ion_buffer.kvaddr =
  520. sess->sess_kvaddr + (VHPCM_BLOCK_SIZE/4) * 3;
  521. done:
  522. return ret;
  523. }
  524. static int hpcm_start_vocpcm(char *pcm_id, struct hpcm_drv *prtd,
  525. struct tap_point *tp)
  526. {
  527. int indx = prtd->mixer_conf.sess_indx;
  528. uint32_t *no_of_tp = &prtd->start_cmd.no_of_tapoints;
  529. struct vss_ivpcm_tap_point *tap_pnt = &prtd->start_cmd.tap_pnt[0];
  530. uint32_t no_of_tp_req = 0;
  531. char *sess_name = hpcm_get_sess_name(indx);
  532. if (prtd->mixer_conf.rx.enable)
  533. no_of_tp_req++;
  534. if (prtd->mixer_conf.tx.enable)
  535. no_of_tp_req++;
  536. if (prtd->mixer_conf.rx.enable && (get_tappnt_value(pcm_id) == RX)) {
  537. if (hpcm_all_dais_are_ready(prtd->mixer_conf.rx.direction,
  538. tp, HPCM_PREPARED)) {
  539. pr_debug("%s: RX conditions met\n", __func__);
  540. tap_pnt[*no_of_tp].tap_point =
  541. VSS_IVPCM_TAP_POINT_RX_DEFAULT;
  542. tap_pnt[*no_of_tp].direction =
  543. prtd->mixer_conf.rx.direction;
  544. tap_pnt[*no_of_tp].sampling_rate =
  545. prtd->mixer_conf.rx.sample_rate;
  546. (*no_of_tp)++;
  547. }
  548. }
  549. if (prtd->mixer_conf.tx.enable && (get_tappnt_value(pcm_id) == TX)) {
  550. if (hpcm_all_dais_are_ready(prtd->mixer_conf.tx.direction,
  551. tp, HPCM_PREPARED)) {
  552. pr_debug("%s: TX conditions met\n", __func__);
  553. tap_pnt[*no_of_tp].tap_point =
  554. VSS_IVPCM_TAP_POINT_TX_DEFAULT;
  555. tap_pnt[*no_of_tp].direction =
  556. prtd->mixer_conf.tx.direction;
  557. tap_pnt[*no_of_tp].sampling_rate =
  558. prtd->mixer_conf.tx.sample_rate;
  559. (*no_of_tp)++;
  560. }
  561. }
  562. if ((prtd->mixer_conf.tx.enable || prtd->mixer_conf.rx.enable) &&
  563. *no_of_tp == no_of_tp_req) {
  564. voc_send_cvp_start_vocpcm(voc_get_session_id(sess_name),
  565. tap_pnt, *no_of_tp);
  566. /* Reset the start command so that it is not called twice */
  567. memset(&prtd->start_cmd, 0, sizeof(struct start_cmd));
  568. } else {
  569. pr_debug("%s: required pcm handles not opened yet\n", __func__);
  570. }
  571. return 0;
  572. }
  573. /* Playback path*/
  574. static void hpcm_copy_playback_data_from_queue(struct dai_data *dai_data,
  575. uint32_t *len)
  576. {
  577. struct hpcm_buf_node *buf_node = NULL;
  578. unsigned long dsp_flags;
  579. if (dai_data->substream == NULL)
  580. return;
  581. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  582. if (!list_empty(&dai_data->filled_queue)) {
  583. buf_node = list_first_entry(&dai_data->filled_queue,
  584. struct hpcm_buf_node, list);
  585. list_del(&buf_node->list);
  586. *len = buf_node->frame.len;
  587. memcpy((u8 *)dai_data->vocpcm_ion_buffer.kvaddr,
  588. &buf_node->frame.voc_pkt[0],
  589. buf_node->frame.len);
  590. list_add_tail(&buf_node->list, &dai_data->free_queue);
  591. dai_data->pcm_irq_pos += dai_data->pcm_count;
  592. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  593. snd_pcm_period_elapsed(dai_data->substream);
  594. } else {
  595. *len = 0;
  596. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  597. pr_err("IN data not available\n");
  598. }
  599. wake_up(&dai_data->queue_wait);
  600. }
  601. /* Capture path*/
  602. static void hpcm_copy_capture_data_to_queue(struct dai_data *dai_data,
  603. uint32_t len)
  604. {
  605. struct hpcm_buf_node *buf_node = NULL;
  606. unsigned long dsp_flags;
  607. if (dai_data->substream == NULL)
  608. return;
  609. /* Copy out buffer packet into free_queue */
  610. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  611. if (!list_empty(&dai_data->free_queue)) {
  612. buf_node = list_first_entry(&dai_data->free_queue,
  613. struct hpcm_buf_node, list);
  614. list_del(&buf_node->list);
  615. buf_node->frame.len = len;
  616. memcpy(&buf_node->frame.voc_pkt[0],
  617. (uint8_t *)dai_data->vocpcm_ion_buffer.kvaddr,
  618. buf_node->frame.len);
  619. list_add_tail(&buf_node->list, &dai_data->filled_queue);
  620. dai_data->pcm_irq_pos += dai_data->pcm_count;
  621. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  622. snd_pcm_period_elapsed(dai_data->substream);
  623. } else {
  624. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  625. pr_err("OUTPUT data dropped\n");
  626. }
  627. wake_up(&dai_data->queue_wait);
  628. }
  629. void hpcm_notify_evt_processing(uint8_t *data, char *session,
  630. void *private_data)
  631. {
  632. struct hpcm_drv *prtd = (struct hpcm_drv *)private_data;
  633. struct vss_ivpcm_evt_notify_v2_t *notify_evt =
  634. (struct vss_ivpcm_evt_notify_v2_t *)data;
  635. struct vss_ivpcm_evt_push_buffer_v2_t push_buff_event;
  636. struct tap_point *tp = NULL;
  637. int in_buf_len = 0;
  638. struct tappnt_mxr_data *tmd = NULL;
  639. char *sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
  640. /* If it's not a timetick, it's a error notification, drop the event */
  641. if ((notify_evt->notify_mask & VSS_IVPCM_NOTIFY_MASK_TIMETICK) == 0) {
  642. pr_err("%s: Error notification. mask=%d\n", __func__,
  643. notify_evt->notify_mask);
  644. return;
  645. }
  646. if (notify_evt->tap_point == VSS_IVPCM_TAP_POINT_TX_DEFAULT) {
  647. tp = &prtd->session[prtd->mixer_conf.sess_indx].tx_tap_point;
  648. tmd = &prtd->mixer_conf.tx;
  649. } else if (notify_evt->tap_point == VSS_IVPCM_TAP_POINT_RX_DEFAULT) {
  650. tp = &prtd->session[prtd->mixer_conf.sess_indx].rx_tap_point;
  651. tmd = &prtd->mixer_conf.rx;
  652. }
  653. if (tp == NULL || tmd == NULL) {
  654. pr_err("%s: tp = %pK or tmd = %pK is null\n", __func__,
  655. tp, tmd);
  656. return;
  657. }
  658. if (notify_evt->notify_mask & VSS_IVPCM_NOTIFY_MASK_OUTPUT_BUFFER) {
  659. hpcm_copy_capture_data_to_queue(&tp->capture_dai_data,
  660. notify_evt->filled_out_size);
  661. }
  662. if (notify_evt->notify_mask & VSS_IVPCM_NOTIFY_MASK_INPUT_BUFFER) {
  663. hpcm_copy_playback_data_from_queue(&tp->playback_dai_data,
  664. &in_buf_len);
  665. }
  666. switch (tmd->direction) {
  667. /*
  668. * When the dir is OUT_IN, for the first notify mask, pushbuf mask
  669. * should be set to VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER since we
  670. * atleast need one buffer's worth data before we can send IN buffer.
  671. * For the consecutive notify evts, the push buf mask will set for both
  672. * VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER and
  673. * VSS_IVPCM_PUSH_BUFFER_MASK_IN_BUFFER.
  674. */
  675. case VSS_IVPCM_TAP_POINT_DIR_OUT_IN:
  676. if (notify_evt->notify_mask ==
  677. VSS_IVPCM_NOTIFY_MASK_TIMETICK) {
  678. push_buff_event.push_buf_mask =
  679. VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER;
  680. } else {
  681. push_buff_event.push_buf_mask =
  682. VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER |
  683. VSS_IVPCM_PUSH_BUFFER_MASK_INPUT_BUFFER;
  684. }
  685. break;
  686. case VSS_IVPCM_TAP_POINT_DIR_IN:
  687. push_buff_event.push_buf_mask =
  688. VSS_IVPCM_PUSH_BUFFER_MASK_INPUT_BUFFER;
  689. break;
  690. case VSS_IVPCM_TAP_POINT_DIR_OUT:
  691. push_buff_event.push_buf_mask =
  692. VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER;
  693. break;
  694. }
  695. push_buff_event.tap_point = notify_evt->tap_point;
  696. push_buff_event.out_buf_mem_address =
  697. tp->capture_dai_data.vocpcm_ion_buffer.paddr;
  698. push_buff_event.in_buf_mem_address =
  699. tp->playback_dai_data.vocpcm_ion_buffer.paddr;
  700. push_buff_event.sampling_rate = notify_evt->sampling_rate;
  701. push_buff_event.num_in_channels = 1;
  702. /*
  703. * ADSP must read and write from a cache aligned (128 byte) location,
  704. * and in blocks of the cache alignment size. The 128 byte cache
  705. * alignment requirement is guaranteed due to 4096 byte memory
  706. * alignment requirement during memory allocation/mapping. The output
  707. * buffer (ADSP write) size mask ensures that a 128 byte multiple
  708. * worth of will be written. Internally, the input buffer (ADSP read)
  709. * size will also be a multiple of 128 bytes. However it is the
  710. * application's responsibility to ensure no other data is written in
  711. * the specified length of memory.
  712. */
  713. push_buff_event.out_buf_mem_size = ((notify_evt->request_buf_size) +
  714. CACHE_ALIGNMENT_SIZE) & CACHE_ALIGNMENT_MASK;
  715. push_buff_event.in_buf_mem_size = in_buf_len;
  716. voc_send_cvp_vocpcm_push_buf_evt(voc_get_session_id(sess_name),
  717. &push_buff_event);
  718. }
  719. static int msm_hpcm_configure_voice_put(struct snd_kcontrol *kcontrol,
  720. struct snd_ctl_elem_value *ucontrol)
  721. {
  722. int tap_point = ucontrol->value.integer.value[0];
  723. uint16_t direction = ucontrol->value.integer.value[1];
  724. uint16_t sample_rate = ucontrol->value.integer.value[2];
  725. struct tappnt_mxr_data *tmd = NULL;
  726. int ret = 0;
  727. mutex_lock(&hpcm_drv.lock);
  728. pr_debug("%s: tap_point = %d direction = %d sample_rate = %d\n",
  729. __func__, tap_point, direction, sample_rate);
  730. if (!hpcm_is_valid_config(VOICE_INDEX, tap_point, direction,
  731. sample_rate)) {
  732. pr_err("Invalid vpcm mixer control voice values\n");
  733. ret = -EINVAL;
  734. goto done;
  735. }
  736. if (tap_point == RX)
  737. tmd = &hpcm_drv.mixer_conf.rx;
  738. else
  739. tmd = &hpcm_drv.mixer_conf.tx;
  740. tmd->enable = true;
  741. tmd->direction = direction;
  742. tmd->sample_rate = sample_rate;
  743. hpcm_drv.mixer_conf.sess_indx = VOICE_INDEX;
  744. done:
  745. mutex_unlock(&hpcm_drv.lock);
  746. return ret;
  747. }
  748. static int msm_hpcm_configure_vmmode1_put(struct snd_kcontrol *kcontrol,
  749. struct snd_ctl_elem_value *ucontrol)
  750. {
  751. int tap_point = ucontrol->value.integer.value[0];
  752. uint16_t direction = ucontrol->value.integer.value[1];
  753. uint16_t sample_rate = ucontrol->value.integer.value[2];
  754. struct tappnt_mxr_data *tmd = NULL;
  755. int ret = 0;
  756. mutex_lock(&hpcm_drv.lock);
  757. pr_debug("%s: tap_point = %d direction = %d sample_rate = %d\n",
  758. __func__, tap_point, direction, sample_rate);
  759. if (!hpcm_is_valid_config(VOMMODE1_INDEX, tap_point, direction,
  760. sample_rate)) {
  761. pr_err("Invalid vpcm mixer control voice values\n");
  762. ret = -EINVAL;
  763. goto done;
  764. }
  765. if (tap_point == RX)
  766. tmd = &hpcm_drv.mixer_conf.rx;
  767. else
  768. tmd = &hpcm_drv.mixer_conf.tx;
  769. tmd->enable = true;
  770. tmd->direction = direction;
  771. tmd->sample_rate = sample_rate;
  772. hpcm_drv.mixer_conf.sess_indx = VOMMODE1_INDEX;
  773. done:
  774. mutex_unlock(&hpcm_drv.lock);
  775. return ret;
  776. }
  777. static int msm_hpcm_configure_vmmode2_put(struct snd_kcontrol *kcontrol,
  778. struct snd_ctl_elem_value *ucontrol)
  779. {
  780. int tap_point = ucontrol->value.integer.value[0];
  781. uint16_t direction = ucontrol->value.integer.value[1];
  782. uint16_t sample_rate = ucontrol->value.integer.value[2];
  783. struct tappnt_mxr_data *tmd = NULL;
  784. int ret = 0;
  785. mutex_lock(&hpcm_drv.lock);
  786. pr_debug("%s: tap_point = %d direction = %d sample_rate = %d\n",
  787. __func__, tap_point, direction, sample_rate);
  788. if (!hpcm_is_valid_config(VOMMODE2_INDEX, tap_point, direction,
  789. sample_rate)) {
  790. pr_err("Invalid vpcm mixer control voice values\n");
  791. ret = -EINVAL;
  792. goto done;
  793. }
  794. if (tap_point == RX)
  795. tmd = &hpcm_drv.mixer_conf.rx;
  796. else
  797. tmd = &hpcm_drv.mixer_conf.tx;
  798. tmd->enable = true;
  799. tmd->direction = direction;
  800. tmd->sample_rate = sample_rate;
  801. hpcm_drv.mixer_conf.sess_indx = VOMMODE2_INDEX;
  802. done:
  803. mutex_unlock(&hpcm_drv.lock);
  804. return ret;
  805. }
  806. static int msm_hpcm_configure_volte_put(struct snd_kcontrol *kcontrol,
  807. struct snd_ctl_elem_value *ucontrol)
  808. {
  809. int tap_point = ucontrol->value.integer.value[0];
  810. uint16_t direction = ucontrol->value.integer.value[1];
  811. uint16_t sample_rate = ucontrol->value.integer.value[2];
  812. struct tappnt_mxr_data *tmd = NULL;
  813. int ret = 0;
  814. mutex_lock(&hpcm_drv.lock);
  815. pr_debug("%s: tap_point=%d direction=%d sample_rate=%d\n",
  816. __func__, tap_point, direction, sample_rate);
  817. if (!hpcm_is_valid_config(VOLTE_INDEX, tap_point, direction,
  818. sample_rate)) {
  819. pr_err("Invalid vpcm mixer control volte values\n");
  820. ret = -EINVAL;
  821. goto done;
  822. }
  823. if (tap_point == RX)
  824. tmd = &hpcm_drv.mixer_conf.rx;
  825. else
  826. tmd = &hpcm_drv.mixer_conf.tx;
  827. tmd->enable = true;
  828. tmd->direction = direction;
  829. tmd->sample_rate = sample_rate;
  830. hpcm_drv.mixer_conf.sess_indx = VOLTE_INDEX;
  831. done:
  832. mutex_unlock(&hpcm_drv.lock);
  833. return ret;
  834. }
  835. static struct snd_kcontrol_new msm_hpcm_controls[] = {
  836. SOC_SINGLE_MULTI_EXT("HPCM_Voice tappoint direction samplerate",
  837. SND_SOC_NOPM, 0, 16000, 0, 3,
  838. NULL, msm_hpcm_configure_voice_put),
  839. SOC_SINGLE_MULTI_EXT("HPCM_VoLTE tappoint direction samplerate",
  840. SND_SOC_NOPM, 0, 16000, 0, 3,
  841. NULL, msm_hpcm_configure_volte_put),
  842. SOC_SINGLE_MULTI_EXT("HPCM_VMMode1 tappoint direction samplerate",
  843. SND_SOC_NOPM, 0, 16000, 0, 3,
  844. NULL, msm_hpcm_configure_vmmode1_put),
  845. SOC_SINGLE_MULTI_EXT("HPCM_VMMode2 tappoint direction samplerate",
  846. SND_SOC_NOPM, 0, 16000, 0, 3,
  847. NULL, msm_hpcm_configure_vmmode2_put),
  848. };
  849. /* Sample rates supported */
  850. static unsigned int supported_sample_rates[] = {8000, 16000};
  851. static struct snd_pcm_hw_constraint_list constraints_sample_rates = {
  852. .count = ARRAY_SIZE(supported_sample_rates),
  853. .list = supported_sample_rates,
  854. .mask = 0,
  855. };
  856. static int msm_pcm_close(struct snd_pcm_substream *substream)
  857. {
  858. int ret = 0;
  859. struct list_head *ptr = NULL;
  860. struct list_head *next = NULL;
  861. struct hpcm_buf_node *buf_node = NULL;
  862. struct snd_dma_buffer *dma_buf;
  863. struct snd_pcm_runtime *runtime;
  864. struct hpcm_drv *prtd;
  865. unsigned long dsp_flags;
  866. struct dai_data *dai_data = NULL;
  867. struct tap_point *tp = NULL;
  868. struct tappnt_mxr_data *tmd = NULL;
  869. char *sess_name = NULL;
  870. if (substream == NULL) {
  871. pr_err("substream is NULL\n");
  872. return -EINVAL;
  873. }
  874. pr_debug("%s, %s\n", __func__, substream->pcm->id);
  875. runtime = substream->runtime;
  876. prtd = runtime->private_data;
  877. sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
  878. dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
  879. if (dai_data == NULL) {
  880. pr_err("%s, dai_data is NULL\n", __func__);
  881. ret = -EINVAL;
  882. goto done;
  883. }
  884. wake_up(&dai_data->queue_wait);
  885. mutex_lock(&prtd->lock);
  886. tmd = hpcm_get_tappnt_mixer_data(substream->pcm->id, prtd);
  887. tp = hpcm_get_tappoint_data(substream->pcm->id, prtd);
  888. /* Send stop command */
  889. voc_send_cvp_stop_vocpcm(voc_get_session_id(sess_name));
  890. /* Memory unmap/free takes place only when called the first time */
  891. hpcm_unmap_and_free_shared_memory(prtd);
  892. /* Unregister host PCM event callback function */
  893. voc_deregister_hpcm_evt_cb();
  894. /* Reset the cached start cmd */
  895. memset(&prtd->start_cmd, 0, sizeof(struct start_cmd));
  896. /* Release all buffer */
  897. pr_debug("%s: Release all buffer\n", __func__);
  898. substream = dai_data->substream;
  899. if (substream == NULL) {
  900. pr_debug("%s: substream is NULL\n", __func__);
  901. goto done;
  902. }
  903. dma_buf = &substream->dma_buffer;
  904. if (dma_buf == NULL) {
  905. pr_debug("%s: dma_buf is NULL\n", __func__);
  906. goto done;
  907. }
  908. if (dma_buf->area != NULL) {
  909. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  910. list_for_each_safe(ptr, next, &dai_data->filled_queue) {
  911. buf_node = list_entry(ptr,
  912. struct hpcm_buf_node, list);
  913. list_del(&buf_node->list);
  914. }
  915. list_for_each_safe(ptr, next, &dai_data->free_queue) {
  916. buf_node = list_entry(ptr,
  917. struct hpcm_buf_node, list);
  918. list_del(&buf_node->list);
  919. }
  920. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  921. dma_free_coherent(substream->pcm->card->dev,
  922. runtime->hw.buffer_bytes_max, dma_buf->area,
  923. dma_buf->addr);
  924. dma_buf->area = NULL;
  925. }
  926. dai_data->substream = NULL;
  927. dai_data->pcm_buf_pos = 0;
  928. dai_data->pcm_count = 0;
  929. dai_data->pcm_irq_pos = 0;
  930. dai_data->pcm_size = 0;
  931. dai_data->state = HPCM_CLOSED;
  932. hpcm_reset_mixer_config(prtd);
  933. done:
  934. mutex_unlock(&prtd->lock);
  935. return ret;
  936. }
  937. static int msm_pcm_playback_copy(struct snd_pcm_substream *substream, int a,
  938. snd_pcm_uframes_t hwoff, void __user *buf,
  939. snd_pcm_uframes_t frames)
  940. {
  941. int ret = 0;
  942. struct hpcm_buf_node *buf_node = NULL;
  943. struct snd_pcm_runtime *runtime = substream->runtime;
  944. struct hpcm_drv *prtd = runtime->private_data;
  945. struct dai_data *dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
  946. unsigned long dsp_flags;
  947. int count = frames_to_bytes(runtime, frames);
  948. if (dai_data == NULL) {
  949. pr_err("%s, dai_data is null\n", __func__);
  950. ret = -EINVAL;
  951. goto done;
  952. }
  953. ret = wait_event_interruptible_timeout(dai_data->queue_wait,
  954. (!list_empty(&dai_data->free_queue) ||
  955. dai_data->state == HPCM_STOPPED),
  956. 1 * HZ);
  957. if (ret > 0) {
  958. if (count <= HPCM_MAX_VOC_PKT_SIZE) {
  959. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  960. buf_node =
  961. list_first_entry(&dai_data->free_queue,
  962. struct hpcm_buf_node, list);
  963. list_del(&buf_node->list);
  964. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  965. ret = copy_from_user(&buf_node->frame.voc_pkt, buf,
  966. count);
  967. buf_node->frame.len = count;
  968. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  969. list_add_tail(&buf_node->list, &dai_data->filled_queue);
  970. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  971. } else {
  972. pr_err("%s: Write cnt %d is > HPCM_MAX_VOC_PKT_SIZE\n",
  973. __func__, count);
  974. ret = -ENOMEM;
  975. }
  976. } else if (ret == 0) {
  977. pr_err("%s: No free Playback buffer\n", __func__);
  978. ret = -ETIMEDOUT;
  979. } else {
  980. pr_err("%s: playback copy was interrupted\n", __func__);
  981. }
  982. done:
  983. return ret;
  984. }
  985. static int msm_pcm_capture_copy(struct snd_pcm_substream *substream,
  986. int channel, snd_pcm_uframes_t hwoff,
  987. void __user *buf, snd_pcm_uframes_t frames)
  988. {
  989. int ret = 0;
  990. int count = 0;
  991. struct hpcm_buf_node *buf_node = NULL;
  992. struct snd_pcm_runtime *runtime = substream->runtime;
  993. struct hpcm_drv *prtd = runtime->private_data;
  994. struct dai_data *dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
  995. unsigned long dsp_flags;
  996. if (dai_data == NULL) {
  997. pr_err("%s, dai_data is null\n", __func__);
  998. ret = -EINVAL;
  999. goto done;
  1000. }
  1001. count = frames_to_bytes(runtime, frames);
  1002. ret = wait_event_interruptible_timeout(dai_data->queue_wait,
  1003. (!list_empty(&dai_data->filled_queue) ||
  1004. dai_data->state == HPCM_STOPPED),
  1005. 1 * HZ);
  1006. if (ret > 0) {
  1007. if (count <= HPCM_MAX_VOC_PKT_SIZE) {
  1008. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  1009. buf_node = list_first_entry(&dai_data->filled_queue,
  1010. struct hpcm_buf_node, list);
  1011. list_del(&buf_node->list);
  1012. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  1013. ret = copy_to_user(buf, &buf_node->frame.voc_pkt,
  1014. buf_node->frame.len);
  1015. if (ret) {
  1016. pr_err("%s: Copy to user returned %d\n",
  1017. __func__, ret);
  1018. ret = -EFAULT;
  1019. }
  1020. spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
  1021. list_add_tail(&buf_node->list, &dai_data->free_queue);
  1022. spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
  1023. } else {
  1024. pr_err("%s: Read count %d > HPCM_MAX_VOC_PKT_SIZE\n",
  1025. __func__, count);
  1026. ret = -ENOMEM;
  1027. }
  1028. } else if (ret == 0) {
  1029. pr_err("%s: No Caputre data available\n", __func__);
  1030. ret = -ETIMEDOUT;
  1031. } else {
  1032. pr_err("%s: Read was interrupted\n", __func__);
  1033. ret = -ERESTARTSYS;
  1034. }
  1035. done:
  1036. return ret;
  1037. }
  1038. static int msm_pcm_copy(struct snd_pcm_substream *substream, int channel,
  1039. snd_pcm_uframes_t hwoff, void __user *buf,
  1040. snd_pcm_uframes_t frames)
  1041. {
  1042. int ret = 0;
  1043. if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
  1044. ret = msm_pcm_playback_copy(substream, channel,
  1045. hwoff, buf, frames);
  1046. else if (substream->stream == SNDRV_PCM_STREAM_CAPTURE)
  1047. ret = msm_pcm_capture_copy(substream, channel,
  1048. hwoff, buf, frames);
  1049. return ret;
  1050. }
  1051. static snd_pcm_uframes_t msm_pcm_pointer(struct snd_pcm_substream *substream)
  1052. {
  1053. struct dai_data *dai_data = NULL;
  1054. struct snd_pcm_runtime *runtime = substream->runtime;
  1055. struct hpcm_drv *prtd = runtime->private_data;
  1056. snd_pcm_uframes_t ret;
  1057. dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
  1058. if (dai_data == NULL) {
  1059. pr_err("%s, dai_data is null\n", __func__);
  1060. ret = 0;
  1061. goto done;
  1062. }
  1063. if (dai_data->pcm_irq_pos >= dai_data->pcm_size)
  1064. dai_data->pcm_irq_pos = 0;
  1065. ret = bytes_to_frames(runtime, (dai_data->pcm_irq_pos));
  1066. done:
  1067. return ret;
  1068. }
  1069. static int msm_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
  1070. {
  1071. int ret = 0;
  1072. struct snd_pcm_runtime *runtime = substream->runtime;
  1073. struct hpcm_drv *prtd = runtime->private_data;
  1074. struct dai_data *dai_data =
  1075. hpcm_get_dai_data(substream->pcm->id, prtd);
  1076. if (dai_data == NULL) {
  1077. pr_err("%s, dai_data is null\n", __func__);
  1078. ret = -EINVAL;
  1079. goto done;
  1080. }
  1081. pr_debug("%s, %s\n", __func__, substream->pcm->id);
  1082. switch (cmd) {
  1083. case SNDRV_PCM_TRIGGER_START:
  1084. pr_debug("SNDRV_PCM_TRIGGER_START\n");
  1085. dai_data->state = HPCM_STARTED;
  1086. break;
  1087. case SNDRV_PCM_TRIGGER_STOP:
  1088. pr_debug("SNDRV_PCM_TRIGGER_STOP\n");
  1089. dai_data->state = HPCM_STOPPED;
  1090. break;
  1091. default:
  1092. ret = -EINVAL;
  1093. break;
  1094. }
  1095. done:
  1096. return ret;
  1097. }
  1098. static int msm_pcm_prepare(struct snd_pcm_substream *substream)
  1099. {
  1100. int ret = 0;
  1101. struct snd_pcm_runtime *runtime = substream->runtime;
  1102. struct hpcm_drv *prtd = runtime->private_data;
  1103. struct dai_data *dai_data = NULL;
  1104. struct tap_point *tp = NULL;
  1105. pr_debug("%s, %s\n", __func__, substream->pcm->id);
  1106. mutex_lock(&prtd->lock);
  1107. dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
  1108. if (dai_data == NULL) {
  1109. pr_err("%s, dai_data is null\n", __func__);
  1110. ret = -EINVAL;
  1111. goto done;
  1112. }
  1113. dai_data->pcm_size = snd_pcm_lib_buffer_bytes(substream);
  1114. dai_data->pcm_count = snd_pcm_lib_period_bytes(substream);
  1115. dai_data->pcm_irq_pos = 0;
  1116. dai_data->pcm_buf_pos = 0;
  1117. dai_data->state = HPCM_PREPARED;
  1118. /* Register event notify processing callback in prepare instead of
  1119. * init() as q6voice module's init() can be called at a later point
  1120. */
  1121. voc_register_hpcm_evt_cb(hpcm_notify_evt_processing, &hpcm_drv);
  1122. tp = hpcm_get_tappoint_data(substream->pcm->id, prtd);
  1123. if (tp != NULL) {
  1124. ret = hpcm_start_vocpcm(substream->pcm->id, prtd, tp);
  1125. if (ret) {
  1126. pr_err("error sending start cmd err=%d\n", ret);
  1127. goto done;
  1128. }
  1129. } else {
  1130. pr_err("%s tp is NULL\n", __func__);
  1131. }
  1132. done:
  1133. mutex_unlock(&prtd->lock);
  1134. return ret;
  1135. }
  1136. static int msm_pcm_hw_params(struct snd_pcm_substream *substream,
  1137. struct snd_pcm_hw_params *params)
  1138. {
  1139. struct snd_pcm_runtime *runtime = substream->runtime;
  1140. struct snd_dma_buffer *dma_buf = &substream->dma_buffer;
  1141. struct hpcm_drv *prtd = (struct hpcm_drv *)runtime->private_data;
  1142. int ret = 0;
  1143. pr_debug("%s: %s\n", __func__, substream->pcm->id);
  1144. mutex_lock(&prtd->lock);
  1145. /* Allocate and map voice host PCM ion buffer */
  1146. if (prtd->session[prtd->mixer_conf.sess_indx].sess_paddr == 0) {
  1147. ret = hpcm_allocate_shared_memory(prtd);
  1148. if (ret) {
  1149. pr_err("error creating shared memory err=%d\n", ret);
  1150. goto done;
  1151. }
  1152. ret = hpcm_map_vocpcm_memory(prtd);
  1153. if (ret) {
  1154. pr_err("error mapping shared memory err=%d\n", ret);
  1155. hpcm_free_allocated_mem(prtd);
  1156. goto done;
  1157. }
  1158. } else {
  1159. pr_debug("%s, VHPCM memory allocation/mapping not performed\n"
  1160. , __func__);
  1161. }
  1162. dma_buf->dev.type = SNDRV_DMA_TYPE_DEV;
  1163. dma_buf->dev.dev = substream->pcm->card->dev;
  1164. dma_buf->private_data = NULL;
  1165. dma_buf->area = dma_alloc_coherent(substream->pcm->card->dev,
  1166. runtime->hw.buffer_bytes_max,
  1167. &dma_buf->addr, GFP_KERNEL);
  1168. if (!dma_buf->area) {
  1169. pr_err("%s:MSM dma_alloc failed\n", __func__);
  1170. ret = -ENOMEM;
  1171. goto done;
  1172. }
  1173. dma_buf->bytes = runtime->hw.buffer_bytes_max;
  1174. memset(dma_buf->area, 0, runtime->hw.buffer_bytes_max);
  1175. hpcm_create_free_queue(dma_buf,
  1176. hpcm_get_dai_data(substream->pcm->id, prtd));
  1177. snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer);
  1178. done:
  1179. mutex_unlock(&prtd->lock);
  1180. return ret;
  1181. }
  1182. static int msm_pcm_open(struct snd_pcm_substream *substream)
  1183. {
  1184. struct snd_pcm_runtime *runtime = substream->runtime;
  1185. struct hpcm_drv *prtd = &hpcm_drv;
  1186. struct tappnt_mxr_data *tmd = NULL;
  1187. struct dai_data *dai_data = NULL;
  1188. int ret = 0;
  1189. int tp_val = 0;
  1190. pr_debug("%s, %s\n", __func__, substream->pcm->id);
  1191. mutex_lock(&prtd->lock);
  1192. dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
  1193. if (dai_data == NULL) {
  1194. pr_err("%s, dai_data is null\n", __func__);
  1195. ret = -EINVAL;
  1196. goto done;
  1197. }
  1198. runtime->hw = msm_pcm_hardware;
  1199. ret = snd_pcm_hw_constraint_list(runtime, 0, SNDRV_PCM_HW_PARAM_RATE,
  1200. &constraints_sample_rates);
  1201. if (ret < 0)
  1202. pr_debug("snd_pcm_hw_constraint_list failed\n");
  1203. ret = snd_pcm_hw_constraint_integer(runtime,
  1204. SNDRV_PCM_HW_PARAM_PERIODS);
  1205. if (ret < 0) {
  1206. pr_debug("snd_pcm_hw_constraint_integer failed\n");
  1207. goto done;
  1208. }
  1209. tp_val = get_tappnt_value(substream->pcm->id);
  1210. tmd = hpcm_get_tappnt_mixer_data(substream->pcm->id, prtd);
  1211. /* Check wheather the kcontrol values set are valid */
  1212. if (!tmd ||
  1213. !(tmd->enable) ||
  1214. !hpcm_is_valid_config(prtd->mixer_conf.sess_indx,
  1215. tp_val, tmd->direction,
  1216. tmd->sample_rate)) {
  1217. ret = -EINVAL;
  1218. goto done;
  1219. }
  1220. dai_data->substream = substream;
  1221. runtime->private_data = prtd;
  1222. done:
  1223. mutex_unlock(&prtd->lock);
  1224. return ret;
  1225. }
  1226. static const struct snd_pcm_ops msm_pcm_ops = {
  1227. .open = msm_pcm_open,
  1228. .hw_params = msm_pcm_hw_params,
  1229. .prepare = msm_pcm_prepare,
  1230. .trigger = msm_pcm_trigger,
  1231. .pointer = msm_pcm_pointer,
  1232. .copy = msm_pcm_copy,
  1233. .close = msm_pcm_close,
  1234. };
  1235. static int msm_asoc_pcm_new(struct snd_soc_pcm_runtime *rtd)
  1236. {
  1237. struct snd_card *card = rtd->card->snd_card;
  1238. pr_debug("%s:\n", __func__);
  1239. if (!card->dev->coherent_dma_mask)
  1240. card->dev->coherent_dma_mask = DMA_BIT_MASK(32);
  1241. return 0;
  1242. }
  1243. static int msm_pcm_hpcm_probe(struct snd_soc_platform *platform)
  1244. {
  1245. snd_soc_add_platform_controls(platform, msm_hpcm_controls,
  1246. ARRAY_SIZE(msm_hpcm_controls));
  1247. return 0;
  1248. }
  1249. static struct snd_soc_platform_driver msm_soc_platform = {
  1250. .ops = &msm_pcm_ops,
  1251. .pcm_new = msm_asoc_pcm_new,
  1252. .probe = msm_pcm_hpcm_probe,
  1253. };
  1254. static int msm_pcm_probe(struct platform_device *pdev)
  1255. {
  1256. pr_info("%s: dev name %s\n", __func__, dev_name(&pdev->dev));
  1257. return snd_soc_register_platform(&pdev->dev, &msm_soc_platform);
  1258. }
  1259. static int msm_pcm_remove(struct platform_device *pdev)
  1260. {
  1261. snd_soc_unregister_platform(&pdev->dev);
  1262. return 0;
  1263. }
  1264. static const struct of_device_id msm_voice_host_pcm_dt_match[] = {
  1265. {.compatible = "qcom,msm-voice-host-pcm"},
  1266. {}
  1267. };
  1268. MODULE_DEVICE_TABLE(of, msm_voice_host_pcm_dt_match);
  1269. static struct platform_driver msm_pcm_driver = {
  1270. .driver = {
  1271. .name = "msm-voice-host-pcm",
  1272. .owner = THIS_MODULE,
  1273. .of_match_table = msm_voice_host_pcm_dt_match,
  1274. },
  1275. .probe = msm_pcm_probe,
  1276. .remove = msm_pcm_remove,
  1277. };
  1278. int __init msm_voice_host_init(void)
  1279. {
  1280. int i = 0;
  1281. struct session *s = NULL;
  1282. memset(&hpcm_drv, 0, sizeof(hpcm_drv));
  1283. mutex_init(&hpcm_drv.lock);
  1284. for (i = 0; i < MAX_SESSION; i++) {
  1285. s = &hpcm_drv.session[i];
  1286. spin_lock_init(&s->rx_tap_point.capture_dai_data.dsp_lock);
  1287. spin_lock_init(&s->rx_tap_point.playback_dai_data.dsp_lock);
  1288. spin_lock_init(&s->tx_tap_point.capture_dai_data.dsp_lock);
  1289. spin_lock_init(&s->tx_tap_point.playback_dai_data.dsp_lock);
  1290. init_waitqueue_head(
  1291. &s->rx_tap_point.capture_dai_data.queue_wait);
  1292. init_waitqueue_head(
  1293. &s->rx_tap_point.playback_dai_data.queue_wait);
  1294. init_waitqueue_head(
  1295. &s->tx_tap_point.capture_dai_data.queue_wait);
  1296. init_waitqueue_head(
  1297. &s->tx_tap_point.playback_dai_data.queue_wait);
  1298. INIT_LIST_HEAD(&s->rx_tap_point.capture_dai_data.filled_queue);
  1299. INIT_LIST_HEAD(&s->rx_tap_point.capture_dai_data.free_queue);
  1300. INIT_LIST_HEAD(&s->rx_tap_point.playback_dai_data.filled_queue);
  1301. INIT_LIST_HEAD(&s->rx_tap_point.playback_dai_data.free_queue);
  1302. INIT_LIST_HEAD(&s->tx_tap_point.capture_dai_data.filled_queue);
  1303. INIT_LIST_HEAD(&s->tx_tap_point.capture_dai_data.free_queue);
  1304. INIT_LIST_HEAD(&s->tx_tap_point.playback_dai_data.filled_queue);
  1305. INIT_LIST_HEAD(&s->tx_tap_point.playback_dai_data.free_queue);
  1306. }
  1307. return platform_driver_register(&msm_pcm_driver);
  1308. }
  1309. void msm_voice_host_exit(void)
  1310. {
  1311. platform_driver_unregister(&msm_pcm_driver);
  1312. }
  1313. MODULE_DESCRIPTION("PCM module platform driver");
  1314. MODULE_LICENSE("GPL v2");