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

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