1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546 |
- /* Copyright (c) 2013-2018, The Linux Foundation. All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 and
- * only version 2 as published by the Free Software Foundation.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- */
- #include <linux/init.h>
- #include <linux/err.h>
- #include <linux/module.h>
- #include <linux/time.h>
- #include <linux/wait.h>
- #include <linux/platform_device.h>
- #include <linux/slab.h>
- #include <linux/dma-mapping.h>
- #include <sound/core.h>
- #include <sound/soc.h>
- #include <sound/soc-dapm.h>
- #include <sound/pcm.h>
- #include <sound/initval.h>
- #include <sound/control.h>
- #include <asm/dma.h>
- #include <dsp/msm_audio_ion.h>
- #include <dsp/q6voice.h>
- #define HPCM_MAX_Q_LEN 2
- #define HPCM_MIN_VOC_PKT_SIZE 320
- #define HPCM_MAX_VOC_PKT_SIZE 640
- #define VHPCM_BLOCK_SIZE 4096
- #define CACHE_ALIGNMENT_SIZE 128
- #define CACHE_ALIGNMENT_MASK 0xFFFFFF80
- #define VOICE_TX_CAPTURE_DAI_ID "CS-VOICE HOST TX CAPTURE"
- #define VOICE_TX_PLAYBACK_DAI_ID "CS-VOICE HOST TX PLAYBACK"
- #define VOICE_RX_CAPTURE_DAI_ID "CS-VOICE HOST RX CAPTURE"
- #define VOICE_RX_PLAYBACK_DAI_ID "CS-VOICE HOST RX PLAYBACK"
- #define VOLTE_TX_CAPTURE_DAI_ID "VOLTE HOST TX CAPTURE"
- #define VOLTE_TX_PLAYBACK_DAI_ID "VOLTE HOST TX PLAYBACK"
- #define VOLTE_RX_CAPTURE_DAI_ID "VOLTE HOST RX CAPTURE"
- #define VOLTE_RX_PLAYBACK_DAI_ID "VOLTE HOST RX PLAYBACK"
- #define VoMMode1_TX_CAPTURE_DAI_ID "VoiceMMode1 HOST TX CAPTURE"
- #define VoMMode1_TX_PLAYBACK_DAI_ID "VoiceMMode1 HOST TX PLAYBACK"
- #define VoMMode1_RX_CAPTURE_DAI_ID "VoiceMMode1 HOST RX CAPTURE"
- #define VoMMode1_RX_PLAYBACK_DAI_ID "VoiceMMode1 HOST RX PLAYBACK"
- #define VoMMode2_TX_CAPTURE_DAI_ID "VoiceMMode2 HOST TX CAPTURE"
- #define VoMMode2_TX_PLAYBACK_DAI_ID "VoiceMMode2 HOST TX PLAYBACK"
- #define VoMMode2_RX_CAPTURE_DAI_ID "VoiceMMode2 HOST RX CAPTURE"
- #define VoMMode2_RX_PLAYBACK_DAI_ID "VoiceMMode2 HOST RX PLAYBACK"
- enum {
- RX = 1,
- TX,
- };
- enum {
- VOICE_INDEX = 0,
- VOLTE_INDEX,
- VOMMODE1_INDEX,
- VOMMODE2_INDEX,
- MAX_SESSION
- };
- enum hpcm_state {
- HPCM_STOPPED = 1,
- HPCM_CLOSED,
- HPCM_PREPARED,
- HPCM_STARTED,
- };
- struct hpcm_frame {
- uint32_t len;
- uint8_t voc_pkt[HPCM_MAX_VOC_PKT_SIZE];
- };
- struct hpcm_buf_node {
- struct list_head list;
- struct hpcm_frame frame;
- };
- struct vocpcm_ion_buffer {
- /* Physical address */
- phys_addr_t paddr;
- /* Kernel virtual address */
- void *kvaddr;
- };
- struct dai_data {
- enum hpcm_state state;
- struct snd_pcm_substream *substream;
- struct list_head filled_queue;
- struct list_head free_queue;
- wait_queue_head_t queue_wait;
- spinlock_t dsp_lock;
- uint32_t pcm_size;
- uint32_t pcm_count;
- /* IRQ position */
- uint32_t pcm_irq_pos;
- /* Position in buffer */
- uint32_t pcm_buf_pos;
- struct vocpcm_ion_buffer vocpcm_ion_buffer;
- };
- struct tap_point {
- struct dai_data playback_dai_data;
- struct dai_data capture_dai_data;
- };
- struct session {
- struct tap_point tx_tap_point;
- struct tap_point rx_tap_point;
- phys_addr_t sess_paddr;
- void *sess_kvaddr;
- struct ion_handle *ion_handle;
- struct mem_map_table tp_mem_table;
- };
- struct tappnt_mxr_data {
- bool enable;
- uint16_t direction;
- uint16_t sample_rate;
- };
- /* Values from mixer ctl are cached in this structure */
- struct mixer_conf {
- int8_t sess_indx;
- struct tappnt_mxr_data rx;
- struct tappnt_mxr_data tx;
- };
- struct start_cmd {
- struct vss_ivpcm_tap_point tap_pnt[2];
- uint32_t no_of_tapoints;
- };
- struct hpcm_drv {
- struct mutex lock;
- struct session session[MAX_SESSION];
- struct mixer_conf mixer_conf;
- struct ion_client *ion_client;
- struct start_cmd start_cmd;
- };
- static struct hpcm_drv hpcm_drv;
- static struct snd_pcm_hardware msm_pcm_hardware = {
- .info = (SNDRV_PCM_INFO_MMAP |
- SNDRV_PCM_INFO_BLOCK_TRANSFER |
- SNDRV_PCM_INFO_MMAP_VALID |
- SNDRV_PCM_INFO_INTERLEAVED),
- .formats = SNDRV_PCM_FMTBIT_S16_LE |
- SNDRV_PCM_FMTBIT_SPECIAL,
- .rates = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000,
- .rate_min = 8000,
- .rate_max = 16000,
- .channels_min = 1,
- .channels_max = 1,
- .buffer_bytes_max = sizeof(struct hpcm_buf_node) * HPCM_MAX_Q_LEN,
- .period_bytes_min = HPCM_MIN_VOC_PKT_SIZE,
- .period_bytes_max = HPCM_MAX_VOC_PKT_SIZE,
- .periods_min = HPCM_MAX_Q_LEN,
- .periods_max = HPCM_MAX_Q_LEN,
- .fifo_size = 0,
- };
- static char *hpcm_get_sess_name(int sess_indx)
- {
- char *sess_name = NULL;
- if (sess_indx == VOICE_INDEX)
- sess_name = VOICE_SESSION_NAME;
- else if (sess_indx == VOLTE_INDEX)
- sess_name = VOLTE_SESSION_NAME;
- else if (sess_indx == VOMMODE1_INDEX)
- sess_name = VOICEMMODE1_NAME;
- else if (sess_indx == VOMMODE2_INDEX)
- sess_name = VOICEMMODE2_NAME;
- else
- pr_err("%s:, Invalid sess_index\n", __func__);
- return sess_name;
- }
- static void hpcm_reset_mixer_config(struct hpcm_drv *prtd)
- {
- prtd->mixer_conf.sess_indx = -1;
- prtd->mixer_conf.rx.enable = false;
- prtd->mixer_conf.rx.direction = -1;
- prtd->mixer_conf.rx.sample_rate = 0;
- prtd->mixer_conf.tx.enable = false;
- prtd->mixer_conf.tx.direction = -1;
- prtd->mixer_conf.tx.sample_rate = 0;
- }
- /* Check for valid mixer control values */
- static bool hpcm_is_valid_config(int sess_indx, int tap_point,
- uint16_t direction, uint16_t samplerate)
- {
- if (sess_indx < VOICE_INDEX || sess_indx > VOMMODE2_INDEX) {
- pr_err("%s: invalid sess_indx :%d\n", __func__, sess_indx);
- goto error;
- }
- if (samplerate != VSS_IVPCM_SAMPLING_RATE_8K &&
- samplerate != VSS_IVPCM_SAMPLING_RATE_16K) {
- pr_err("%s: invalid sample rate :%d\n", __func__, samplerate);
- goto error;
- }
- if ((tap_point != RX) && (tap_point != TX)) {
- pr_err("%s: invalid tappoint :%d\n", __func__, tap_point);
- goto error;
- }
- if ((direction != VSS_IVPCM_TAP_POINT_DIR_IN) &&
- (direction != VSS_IVPCM_TAP_POINT_DIR_OUT) &&
- (direction != VSS_IVPCM_TAP_POINT_DIR_OUT_IN)) {
- pr_err("%s: invalid direction :%d\n", __func__, direction);
- goto error;
- }
- return true;
- error:
- return false;
- }
- static struct dai_data *hpcm_get_dai_data(char *pcm_id, struct hpcm_drv *prtd)
- {
- struct dai_data *dai_data = NULL;
- size_t size = 0;
- if (pcm_id) {
- size = strlen(pcm_id);
- /* Check for Voice DAI */
- if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOICE_INDEX].tx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOICE_INDEX].tx_tap_point.playback_dai_data;
- } else if (strnstr(pcm_id, VOICE_RX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOICE_INDEX].rx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VOICE_RX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOICE_INDEX].rx_tap_point.playback_dai_data;
- /* Check for VoLTE DAI */
- } else if (strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOLTE_INDEX].tx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOLTE_INDEX].tx_tap_point.playback_dai_data;
- } else if (strnstr(pcm_id, VOLTE_RX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOLTE_INDEX].rx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VOLTE_RX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOLTE_INDEX].rx_tap_point.playback_dai_data;
- /* check for VoiceMMode1 DAI */
- } else if (strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE1_INDEX].tx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE1_INDEX].tx_tap_point.playback_dai_data;
- } else if (strnstr(pcm_id, VoMMode1_RX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE1_INDEX].rx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VoMMode1_RX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE1_INDEX].rx_tap_point.playback_dai_data;
- /* check for VOiceMMode2 DAI */
- } else if (strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE2_INDEX].tx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE2_INDEX].tx_tap_point.playback_dai_data;
- } else if (strnstr(pcm_id, VoMMode2_RX_CAPTURE_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE2_INDEX].rx_tap_point.capture_dai_data;
- } else if (strnstr(pcm_id, VoMMode2_RX_PLAYBACK_DAI_ID, size)) {
- dai_data =
- &prtd->session[VOMMODE2_INDEX].rx_tap_point.playback_dai_data;
- } else {
- pr_err("%s: Wrong dai id\n", __func__);
- }
- }
- return dai_data;
- }
- static struct tap_point *hpcm_get_tappoint_data(char *pcm_id,
- struct hpcm_drv *prtd)
- {
- struct tap_point *tp = NULL;
- size_t size = 0;
- if (pcm_id) {
- size = strlen(pcm_id);
- /* Check for Voice DAI */
- if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOICE_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOICE_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VOICE_RX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOICE_INDEX].rx_tap_point;
- } else if (strnstr(pcm_id, VOICE_RX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOICE_INDEX].rx_tap_point;
- /* Check for VoLTE DAI */
- } else if (strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOLTE_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOLTE_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VOLTE_RX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOLTE_INDEX].rx_tap_point;
- } else if (strnstr(pcm_id, VOLTE_RX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOLTE_INDEX].rx_tap_point;
- /* check for VoiceMMode1 */
- } else if (strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE1_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE1_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VoMMode1_RX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE1_INDEX].rx_tap_point;
- } else if (strnstr(pcm_id, VoMMode1_RX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE1_INDEX].rx_tap_point;
- /* check for VoiceMMode2 */
- } else if (strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE2_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE2_INDEX].tx_tap_point;
- } else if (strnstr(pcm_id, VoMMode2_RX_CAPTURE_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE2_INDEX].rx_tap_point;
- } else if (strnstr(pcm_id, VoMMode2_RX_PLAYBACK_DAI_ID, size)) {
- tp = &prtd->session[VOMMODE2_INDEX].rx_tap_point;
- } else {
- pr_err("%s: wrong dai id\n", __func__);
- }
- }
- return tp;
- }
- static struct tappnt_mxr_data *hpcm_get_tappnt_mixer_data(char *pcm_id,
- struct hpcm_drv *prtd)
- {
- if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, strlen(pcm_id))) {
- return &prtd->mixer_conf.tx;
- } else {
- return &prtd->mixer_conf.rx;
- }
- }
- static int get_tappnt_value(char *pcm_id)
- {
- if (strnstr(pcm_id, VOICE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VOICE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VOLTE_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VOLTE_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode1_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode1_TX_PLAYBACK_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode2_TX_CAPTURE_DAI_ID, strlen(pcm_id)) ||
- strnstr(pcm_id, VoMMode2_TX_PLAYBACK_DAI_ID, strlen(pcm_id))) {
- return TX;
- } else {
- return RX;
- }
- }
- static bool hpcm_all_dais_are_ready(uint16_t direction, struct tap_point *tp,
- enum hpcm_state state)
- {
- bool dais_started = false;
- /*
- * Based on the direction set per tap point in the mixer control,
- * all the dais per tap point should meet the required state for the
- * commands such as vpcm_map_memory/vpcm_start to be executed.
- */
- switch (direction) {
- case VSS_IVPCM_TAP_POINT_DIR_OUT_IN:
- if ((tp->playback_dai_data.state >= state) &&
- (tp->capture_dai_data.state >= state)) {
- dais_started = true;
- }
- break;
- case VSS_IVPCM_TAP_POINT_DIR_IN:
- if (tp->playback_dai_data.state >= state)
- dais_started = true;
- break;
- case VSS_IVPCM_TAP_POINT_DIR_OUT:
- if (tp->capture_dai_data.state >= state)
- dais_started = true;
- break;
- default:
- pr_err("invalid direction\n");
- }
- return dais_started;
- }
- static void hpcm_create_free_queue(struct snd_dma_buffer *dma_buf,
- struct dai_data *dai_data)
- {
- struct hpcm_buf_node *buf_node = NULL;
- int i = 0, offset = 0;
- for (i = 0; i < HPCM_MAX_Q_LEN; i++) {
- buf_node = (void *)dma_buf->area + offset;
- list_add_tail(&buf_node->list,
- &dai_data->free_queue);
- offset = offset + sizeof(struct hpcm_buf_node);
- }
- }
- static void hpcm_free_allocated_mem(struct hpcm_drv *prtd)
- {
- phys_addr_t paddr = 0;
- struct tap_point *txtp = NULL;
- struct tap_point *rxtp = NULL;
- struct session *sess = NULL;
- sess = &prtd->session[prtd->mixer_conf.sess_indx];
- txtp = &sess->tx_tap_point;
- rxtp = &sess->rx_tap_point;
- paddr = sess->sess_paddr;
- if (paddr) {
- msm_audio_ion_free(prtd->ion_client, sess->ion_handle);
- prtd->ion_client = NULL;
- sess->ion_handle = NULL;
- msm_audio_ion_free(sess->tp_mem_table.client,
- sess->tp_mem_table.handle);
- sess->tp_mem_table.client = NULL;
- sess->tp_mem_table.handle = NULL;
- sess->sess_paddr = 0;
- sess->sess_kvaddr = 0;
- sess->ion_handle = 0;
- prtd->ion_client = 0;
- sess->tp_mem_table.client = 0;
- sess->tp_mem_table.handle = 0;
- txtp->capture_dai_data.vocpcm_ion_buffer.paddr = 0;
- txtp->capture_dai_data.vocpcm_ion_buffer.kvaddr = 0;
- txtp->playback_dai_data.vocpcm_ion_buffer.paddr = 0;
- txtp->playback_dai_data.vocpcm_ion_buffer.kvaddr = 0;
- rxtp->capture_dai_data.vocpcm_ion_buffer.paddr = 0;
- rxtp->capture_dai_data.vocpcm_ion_buffer.kvaddr = 0;
- rxtp->playback_dai_data.vocpcm_ion_buffer.paddr = 0;
- rxtp->playback_dai_data.vocpcm_ion_buffer.kvaddr = 0;
- } else {
- pr_debug("%s, paddr = 0, nothing to free\n", __func__);
- }
- }
- static void hpcm_unmap_and_free_shared_memory(struct hpcm_drv *prtd)
- {
- phys_addr_t paddr = 0;
- char *sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
- if (prtd->mixer_conf.sess_indx >= 0)
- paddr = prtd->session[prtd->mixer_conf.sess_indx].sess_paddr;
- else
- paddr = 0;
- if (paddr) {
- voc_send_cvp_unmap_vocpcm_memory(voc_get_session_id(sess_name));
- hpcm_free_allocated_mem(prtd);
- } else {
- pr_debug("%s, paddr = 0, nothing to unmap/free\n", __func__);
- }
- }
- static int hpcm_map_vocpcm_memory(struct hpcm_drv *prtd)
- {
- int ret = 0;
- char *sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
- struct session *sess = NULL;
- sess = &prtd->session[prtd->mixer_conf.sess_indx];
- ret = voc_send_cvp_map_vocpcm_memory(voc_get_session_id(sess_name),
- &sess->tp_mem_table,
- sess->sess_paddr,
- VHPCM_BLOCK_SIZE);
- return ret;
- }
- static int hpcm_allocate_shared_memory(struct hpcm_drv *prtd)
- {
- int result;
- int ret = 0;
- size_t mem_len;
- size_t len;
- struct tap_point *txtp = NULL;
- struct tap_point *rxtp = NULL;
- struct session *sess = NULL;
- sess = &prtd->session[prtd->mixer_conf.sess_indx];
- txtp = &sess->tx_tap_point;
- rxtp = &sess->rx_tap_point;
- result = msm_audio_ion_alloc("host_pcm_buffer",
- &prtd->ion_client,
- &sess->ion_handle,
- VHPCM_BLOCK_SIZE,
- &sess->sess_paddr,
- &mem_len,
- &sess->sess_kvaddr);
- if (result) {
- pr_err("%s: msm_audio_ion_alloc error, rc = %d\n",
- __func__, result);
- sess->sess_paddr = 0;
- sess->sess_kvaddr = 0;
- ret = -ENOMEM;
- goto done;
- }
- pr_debug("%s: Host PCM memory block allocated\n", __func__);
- /* Allocate mem_map_table for tap point */
- result = msm_audio_ion_alloc("host_pcm_table",
- &sess->tp_mem_table.client,
- &sess->tp_mem_table.handle,
- sizeof(struct vss_imemory_table_t),
- &sess->tp_mem_table.phys,
- &len,
- &sess->tp_mem_table.data);
- if (result) {
- pr_err("%s: msm_audio_ion_alloc error, rc = %d\n",
- __func__, result);
- msm_audio_ion_free(prtd->ion_client, sess->ion_handle);
- prtd->ion_client = NULL;
- sess->ion_handle = NULL;
- sess->sess_paddr = 0;
- sess->sess_kvaddr = 0;
- ret = -ENOMEM;
- goto done;
- }
- pr_debug("%s: Host PCM memory table allocated\n", __func__);
- memset(sess->tp_mem_table.data, 0,
- sizeof(struct vss_imemory_table_t));
- sess->tp_mem_table.size = sizeof(struct vss_imemory_table_t);
- pr_debug("%s: data %pK phys %pK\n", __func__,
- sess->tp_mem_table.data, &sess->tp_mem_table.phys);
- /* Split 4096 block into four 1024 byte blocks for each dai */
- txtp->capture_dai_data.vocpcm_ion_buffer.paddr =
- sess->sess_paddr;
- txtp->capture_dai_data.vocpcm_ion_buffer.kvaddr =
- sess->sess_kvaddr;
- txtp->playback_dai_data.vocpcm_ion_buffer.paddr =
- sess->sess_paddr + VHPCM_BLOCK_SIZE/4;
- txtp->playback_dai_data.vocpcm_ion_buffer.kvaddr =
- sess->sess_kvaddr + VHPCM_BLOCK_SIZE/4;
- rxtp->capture_dai_data.vocpcm_ion_buffer.paddr =
- sess->sess_paddr + (VHPCM_BLOCK_SIZE/4) * 2;
- rxtp->capture_dai_data.vocpcm_ion_buffer.kvaddr =
- sess->sess_kvaddr + (VHPCM_BLOCK_SIZE/4) * 2;
- rxtp->playback_dai_data.vocpcm_ion_buffer.paddr =
- sess->sess_paddr + (VHPCM_BLOCK_SIZE/4) * 3;
- rxtp->playback_dai_data.vocpcm_ion_buffer.kvaddr =
- sess->sess_kvaddr + (VHPCM_BLOCK_SIZE/4) * 3;
- done:
- return ret;
- }
- static int hpcm_start_vocpcm(char *pcm_id, struct hpcm_drv *prtd,
- struct tap_point *tp)
- {
- int indx = prtd->mixer_conf.sess_indx;
- uint32_t *no_of_tp = &prtd->start_cmd.no_of_tapoints;
- struct vss_ivpcm_tap_point *tap_pnt = &prtd->start_cmd.tap_pnt[0];
- uint32_t no_of_tp_req = 0;
- char *sess_name = hpcm_get_sess_name(indx);
- if (prtd->mixer_conf.rx.enable)
- no_of_tp_req++;
- if (prtd->mixer_conf.tx.enable)
- no_of_tp_req++;
- if (prtd->mixer_conf.rx.enable && (get_tappnt_value(pcm_id) == RX)) {
- if (hpcm_all_dais_are_ready(prtd->mixer_conf.rx.direction,
- tp, HPCM_PREPARED)) {
- pr_debug("%s: RX conditions met\n", __func__);
- tap_pnt[*no_of_tp].tap_point =
- VSS_IVPCM_TAP_POINT_RX_DEFAULT;
- tap_pnt[*no_of_tp].direction =
- prtd->mixer_conf.rx.direction;
- tap_pnt[*no_of_tp].sampling_rate =
- prtd->mixer_conf.rx.sample_rate;
- (*no_of_tp)++;
- }
- }
- if (prtd->mixer_conf.tx.enable && (get_tappnt_value(pcm_id) == TX)) {
- if (hpcm_all_dais_are_ready(prtd->mixer_conf.tx.direction,
- tp, HPCM_PREPARED)) {
- pr_debug("%s: TX conditions met\n", __func__);
- tap_pnt[*no_of_tp].tap_point =
- VSS_IVPCM_TAP_POINT_TX_DEFAULT;
- tap_pnt[*no_of_tp].direction =
- prtd->mixer_conf.tx.direction;
- tap_pnt[*no_of_tp].sampling_rate =
- prtd->mixer_conf.tx.sample_rate;
- (*no_of_tp)++;
- }
- }
- if ((prtd->mixer_conf.tx.enable || prtd->mixer_conf.rx.enable) &&
- *no_of_tp == no_of_tp_req) {
- voc_send_cvp_start_vocpcm(voc_get_session_id(sess_name),
- tap_pnt, *no_of_tp);
- /* Reset the start command so that it is not called twice */
- memset(&prtd->start_cmd, 0, sizeof(struct start_cmd));
- } else {
- pr_debug("%s: required pcm handles not opened yet\n", __func__);
- }
- return 0;
- }
- /* Playback path*/
- static void hpcm_copy_playback_data_from_queue(struct dai_data *dai_data,
- uint32_t *len)
- {
- struct hpcm_buf_node *buf_node = NULL;
- unsigned long dsp_flags;
- if (dai_data->substream == NULL)
- return;
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- if (!list_empty(&dai_data->filled_queue)) {
- buf_node = list_first_entry(&dai_data->filled_queue,
- struct hpcm_buf_node, list);
- list_del(&buf_node->list);
- *len = buf_node->frame.len;
- memcpy((u8 *)dai_data->vocpcm_ion_buffer.kvaddr,
- &buf_node->frame.voc_pkt[0],
- buf_node->frame.len);
- list_add_tail(&buf_node->list, &dai_data->free_queue);
- dai_data->pcm_irq_pos += dai_data->pcm_count;
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- snd_pcm_period_elapsed(dai_data->substream);
- } else {
- *len = 0;
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- pr_err("IN data not available\n");
- }
- wake_up(&dai_data->queue_wait);
- }
- /* Capture path*/
- static void hpcm_copy_capture_data_to_queue(struct dai_data *dai_data,
- uint32_t len)
- {
- struct hpcm_buf_node *buf_node = NULL;
- unsigned long dsp_flags;
- if (dai_data->substream == NULL)
- return;
- /* Copy out buffer packet into free_queue */
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- if (!list_empty(&dai_data->free_queue)) {
- buf_node = list_first_entry(&dai_data->free_queue,
- struct hpcm_buf_node, list);
- list_del(&buf_node->list);
- buf_node->frame.len = len;
- memcpy(&buf_node->frame.voc_pkt[0],
- (uint8_t *)dai_data->vocpcm_ion_buffer.kvaddr,
- buf_node->frame.len);
- list_add_tail(&buf_node->list, &dai_data->filled_queue);
- dai_data->pcm_irq_pos += dai_data->pcm_count;
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- snd_pcm_period_elapsed(dai_data->substream);
- } else {
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- pr_err("OUTPUT data dropped\n");
- }
- wake_up(&dai_data->queue_wait);
- }
- void hpcm_notify_evt_processing(uint8_t *data, char *session,
- void *private_data)
- {
- struct hpcm_drv *prtd = (struct hpcm_drv *)private_data;
- struct vss_ivpcm_evt_notify_v2_t *notify_evt =
- (struct vss_ivpcm_evt_notify_v2_t *)data;
- struct vss_ivpcm_evt_push_buffer_v2_t push_buff_event;
- struct tap_point *tp = NULL;
- int in_buf_len = 0;
- struct tappnt_mxr_data *tmd = NULL;
- char *sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
- /* If it's not a timetick, it's a error notification, drop the event */
- if ((notify_evt->notify_mask & VSS_IVPCM_NOTIFY_MASK_TIMETICK) == 0) {
- pr_err("%s: Error notification. mask=%d\n", __func__,
- notify_evt->notify_mask);
- return;
- }
- if (notify_evt->tap_point == VSS_IVPCM_TAP_POINT_TX_DEFAULT) {
- tp = &prtd->session[prtd->mixer_conf.sess_indx].tx_tap_point;
- tmd = &prtd->mixer_conf.tx;
- } else if (notify_evt->tap_point == VSS_IVPCM_TAP_POINT_RX_DEFAULT) {
- tp = &prtd->session[prtd->mixer_conf.sess_indx].rx_tap_point;
- tmd = &prtd->mixer_conf.rx;
- }
- if (tp == NULL || tmd == NULL) {
- pr_err("%s: tp = %pK or tmd = %pK is null\n", __func__,
- tp, tmd);
- return;
- }
- if (notify_evt->notify_mask & VSS_IVPCM_NOTIFY_MASK_OUTPUT_BUFFER) {
- hpcm_copy_capture_data_to_queue(&tp->capture_dai_data,
- notify_evt->filled_out_size);
- }
- if (notify_evt->notify_mask & VSS_IVPCM_NOTIFY_MASK_INPUT_BUFFER) {
- hpcm_copy_playback_data_from_queue(&tp->playback_dai_data,
- &in_buf_len);
- }
- switch (tmd->direction) {
- /*
- * When the dir is OUT_IN, for the first notify mask, pushbuf mask
- * should be set to VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER since we
- * atleast need one buffer's worth data before we can send IN buffer.
- * For the consecutive notify evts, the push buf mask will set for both
- * VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER and
- * VSS_IVPCM_PUSH_BUFFER_MASK_IN_BUFFER.
- */
- case VSS_IVPCM_TAP_POINT_DIR_OUT_IN:
- if (notify_evt->notify_mask ==
- VSS_IVPCM_NOTIFY_MASK_TIMETICK) {
- push_buff_event.push_buf_mask =
- VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER;
- } else {
- push_buff_event.push_buf_mask =
- VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER |
- VSS_IVPCM_PUSH_BUFFER_MASK_INPUT_BUFFER;
- }
- break;
- case VSS_IVPCM_TAP_POINT_DIR_IN:
- push_buff_event.push_buf_mask =
- VSS_IVPCM_PUSH_BUFFER_MASK_INPUT_BUFFER;
- break;
- case VSS_IVPCM_TAP_POINT_DIR_OUT:
- push_buff_event.push_buf_mask =
- VSS_IVPCM_PUSH_BUFFER_MASK_OUTPUT_BUFFER;
- break;
- }
- push_buff_event.tap_point = notify_evt->tap_point;
- push_buff_event.out_buf_mem_address =
- tp->capture_dai_data.vocpcm_ion_buffer.paddr;
- push_buff_event.in_buf_mem_address =
- tp->playback_dai_data.vocpcm_ion_buffer.paddr;
- push_buff_event.sampling_rate = notify_evt->sampling_rate;
- push_buff_event.num_in_channels = 1;
- /*
- * ADSP must read and write from a cache aligned (128 byte) location,
- * and in blocks of the cache alignment size. The 128 byte cache
- * alignment requirement is guaranteed due to 4096 byte memory
- * alignment requirement during memory allocation/mapping. The output
- * buffer (ADSP write) size mask ensures that a 128 byte multiple
- * worth of will be written. Internally, the input buffer (ADSP read)
- * size will also be a multiple of 128 bytes. However it is the
- * application's responsibility to ensure no other data is written in
- * the specified length of memory.
- */
- push_buff_event.out_buf_mem_size = ((notify_evt->request_buf_size) +
- CACHE_ALIGNMENT_SIZE) & CACHE_ALIGNMENT_MASK;
- push_buff_event.in_buf_mem_size = in_buf_len;
- voc_send_cvp_vocpcm_push_buf_evt(voc_get_session_id(sess_name),
- &push_buff_event);
- }
- static int msm_hpcm_configure_voice_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- int tap_point = ucontrol->value.integer.value[0];
- uint16_t direction = ucontrol->value.integer.value[1];
- uint16_t sample_rate = ucontrol->value.integer.value[2];
- struct tappnt_mxr_data *tmd = NULL;
- int ret = 0;
- mutex_lock(&hpcm_drv.lock);
- pr_debug("%s: tap_point = %d direction = %d sample_rate = %d\n",
- __func__, tap_point, direction, sample_rate);
- if (!hpcm_is_valid_config(VOICE_INDEX, tap_point, direction,
- sample_rate)) {
- pr_err("Invalid vpcm mixer control voice values\n");
- ret = -EINVAL;
- goto done;
- }
- if (tap_point == RX)
- tmd = &hpcm_drv.mixer_conf.rx;
- else
- tmd = &hpcm_drv.mixer_conf.tx;
- tmd->enable = true;
- tmd->direction = direction;
- tmd->sample_rate = sample_rate;
- hpcm_drv.mixer_conf.sess_indx = VOICE_INDEX;
- done:
- mutex_unlock(&hpcm_drv.lock);
- return ret;
- }
- static int msm_hpcm_configure_vmmode1_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- int tap_point = ucontrol->value.integer.value[0];
- uint16_t direction = ucontrol->value.integer.value[1];
- uint16_t sample_rate = ucontrol->value.integer.value[2];
- struct tappnt_mxr_data *tmd = NULL;
- int ret = 0;
- mutex_lock(&hpcm_drv.lock);
- pr_debug("%s: tap_point = %d direction = %d sample_rate = %d\n",
- __func__, tap_point, direction, sample_rate);
- if (!hpcm_is_valid_config(VOMMODE1_INDEX, tap_point, direction,
- sample_rate)) {
- pr_err("Invalid vpcm mixer control voice values\n");
- ret = -EINVAL;
- goto done;
- }
- if (tap_point == RX)
- tmd = &hpcm_drv.mixer_conf.rx;
- else
- tmd = &hpcm_drv.mixer_conf.tx;
- tmd->enable = true;
- tmd->direction = direction;
- tmd->sample_rate = sample_rate;
- hpcm_drv.mixer_conf.sess_indx = VOMMODE1_INDEX;
- done:
- mutex_unlock(&hpcm_drv.lock);
- return ret;
- }
- static int msm_hpcm_configure_vmmode2_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- int tap_point = ucontrol->value.integer.value[0];
- uint16_t direction = ucontrol->value.integer.value[1];
- uint16_t sample_rate = ucontrol->value.integer.value[2];
- struct tappnt_mxr_data *tmd = NULL;
- int ret = 0;
- mutex_lock(&hpcm_drv.lock);
- pr_debug("%s: tap_point = %d direction = %d sample_rate = %d\n",
- __func__, tap_point, direction, sample_rate);
- if (!hpcm_is_valid_config(VOMMODE2_INDEX, tap_point, direction,
- sample_rate)) {
- pr_err("Invalid vpcm mixer control voice values\n");
- ret = -EINVAL;
- goto done;
- }
- if (tap_point == RX)
- tmd = &hpcm_drv.mixer_conf.rx;
- else
- tmd = &hpcm_drv.mixer_conf.tx;
- tmd->enable = true;
- tmd->direction = direction;
- tmd->sample_rate = sample_rate;
- hpcm_drv.mixer_conf.sess_indx = VOMMODE2_INDEX;
- done:
- mutex_unlock(&hpcm_drv.lock);
- return ret;
- }
- static int msm_hpcm_configure_volte_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- int tap_point = ucontrol->value.integer.value[0];
- uint16_t direction = ucontrol->value.integer.value[1];
- uint16_t sample_rate = ucontrol->value.integer.value[2];
- struct tappnt_mxr_data *tmd = NULL;
- int ret = 0;
- mutex_lock(&hpcm_drv.lock);
- pr_debug("%s: tap_point=%d direction=%d sample_rate=%d\n",
- __func__, tap_point, direction, sample_rate);
- if (!hpcm_is_valid_config(VOLTE_INDEX, tap_point, direction,
- sample_rate)) {
- pr_err("Invalid vpcm mixer control volte values\n");
- ret = -EINVAL;
- goto done;
- }
- if (tap_point == RX)
- tmd = &hpcm_drv.mixer_conf.rx;
- else
- tmd = &hpcm_drv.mixer_conf.tx;
- tmd->enable = true;
- tmd->direction = direction;
- tmd->sample_rate = sample_rate;
- hpcm_drv.mixer_conf.sess_indx = VOLTE_INDEX;
- done:
- mutex_unlock(&hpcm_drv.lock);
- return ret;
- }
- static struct snd_kcontrol_new msm_hpcm_controls[] = {
- SOC_SINGLE_MULTI_EXT("HPCM_Voice tappoint direction samplerate",
- SND_SOC_NOPM, 0, 16000, 0, 3,
- NULL, msm_hpcm_configure_voice_put),
- SOC_SINGLE_MULTI_EXT("HPCM_VoLTE tappoint direction samplerate",
- SND_SOC_NOPM, 0, 16000, 0, 3,
- NULL, msm_hpcm_configure_volte_put),
- SOC_SINGLE_MULTI_EXT("HPCM_VMMode1 tappoint direction samplerate",
- SND_SOC_NOPM, 0, 16000, 0, 3,
- NULL, msm_hpcm_configure_vmmode1_put),
- SOC_SINGLE_MULTI_EXT("HPCM_VMMode2 tappoint direction samplerate",
- SND_SOC_NOPM, 0, 16000, 0, 3,
- NULL, msm_hpcm_configure_vmmode2_put),
- };
- /* Sample rates supported */
- static unsigned int supported_sample_rates[] = {8000, 16000};
- static struct snd_pcm_hw_constraint_list constraints_sample_rates = {
- .count = ARRAY_SIZE(supported_sample_rates),
- .list = supported_sample_rates,
- .mask = 0,
- };
- static int msm_pcm_close(struct snd_pcm_substream *substream)
- {
- int ret = 0;
- struct list_head *ptr = NULL;
- struct list_head *next = NULL;
- struct hpcm_buf_node *buf_node = NULL;
- struct snd_dma_buffer *dma_buf;
- struct snd_pcm_runtime *runtime;
- struct hpcm_drv *prtd;
- unsigned long dsp_flags;
- struct dai_data *dai_data = NULL;
- struct tap_point *tp = NULL;
- struct tappnt_mxr_data *tmd = NULL;
- char *sess_name = NULL;
- if (substream == NULL) {
- pr_err("substream is NULL\n");
- return -EINVAL;
- }
- pr_debug("%s, %s\n", __func__, substream->pcm->id);
- runtime = substream->runtime;
- prtd = runtime->private_data;
- sess_name = hpcm_get_sess_name(prtd->mixer_conf.sess_indx);
- dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
- if (dai_data == NULL) {
- pr_err("%s, dai_data is NULL\n", __func__);
- ret = -EINVAL;
- goto done;
- }
- wake_up(&dai_data->queue_wait);
- mutex_lock(&prtd->lock);
- tmd = hpcm_get_tappnt_mixer_data(substream->pcm->id, prtd);
- tp = hpcm_get_tappoint_data(substream->pcm->id, prtd);
- /* Send stop command */
- voc_send_cvp_stop_vocpcm(voc_get_session_id(sess_name));
- /* Memory unmap/free takes place only when called the first time */
- hpcm_unmap_and_free_shared_memory(prtd);
- /* Unregister host PCM event callback function */
- voc_deregister_hpcm_evt_cb();
- /* Reset the cached start cmd */
- memset(&prtd->start_cmd, 0, sizeof(struct start_cmd));
- /* Release all buffer */
- pr_debug("%s: Release all buffer\n", __func__);
- substream = dai_data->substream;
- if (substream == NULL) {
- pr_debug("%s: substream is NULL\n", __func__);
- goto done;
- }
- dma_buf = &substream->dma_buffer;
- if (dma_buf == NULL) {
- pr_debug("%s: dma_buf is NULL\n", __func__);
- goto done;
- }
- if (dma_buf->area != NULL) {
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- list_for_each_safe(ptr, next, &dai_data->filled_queue) {
- buf_node = list_entry(ptr,
- struct hpcm_buf_node, list);
- list_del(&buf_node->list);
- }
- list_for_each_safe(ptr, next, &dai_data->free_queue) {
- buf_node = list_entry(ptr,
- struct hpcm_buf_node, list);
- list_del(&buf_node->list);
- }
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- dma_free_coherent(substream->pcm->card->dev,
- runtime->hw.buffer_bytes_max, dma_buf->area,
- dma_buf->addr);
- dma_buf->area = NULL;
- }
- dai_data->substream = NULL;
- dai_data->pcm_buf_pos = 0;
- dai_data->pcm_count = 0;
- dai_data->pcm_irq_pos = 0;
- dai_data->pcm_size = 0;
- dai_data->state = HPCM_CLOSED;
- hpcm_reset_mixer_config(prtd);
- done:
- mutex_unlock(&prtd->lock);
- return ret;
- }
- static int msm_pcm_playback_copy(struct snd_pcm_substream *substream, int a,
- unsigned long hwoff, void __user *buf,
- unsigned long fbytes)
- {
- int ret = 0;
- struct hpcm_buf_node *buf_node = NULL;
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct hpcm_drv *prtd = runtime->private_data;
- struct dai_data *dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
- unsigned long dsp_flags;
- if (dai_data == NULL) {
- pr_err("%s, dai_data is null\n", __func__);
- ret = -EINVAL;
- goto done;
- }
- ret = wait_event_interruptible_timeout(dai_data->queue_wait,
- (!list_empty(&dai_data->free_queue) ||
- dai_data->state == HPCM_STOPPED),
- 1 * HZ);
- if (ret > 0) {
- if (fbytes <= HPCM_MAX_VOC_PKT_SIZE) {
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- buf_node =
- list_first_entry(&dai_data->free_queue,
- struct hpcm_buf_node, list);
- list_del(&buf_node->list);
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- ret = copy_from_user(&buf_node->frame.voc_pkt, buf,
- fbytes);
- buf_node->frame.len = fbytes;
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- list_add_tail(&buf_node->list, &dai_data->filled_queue);
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- } else {
- pr_err("%s: Write cnt %lu is > HPCM_MAX_VOC_PKT_SIZE\n",
- __func__, fbytes);
- ret = -ENOMEM;
- }
- } else if (ret == 0) {
- pr_err("%s: No free Playback buffer\n", __func__);
- ret = -ETIMEDOUT;
- } else {
- pr_err("%s: playback copy was interrupted\n", __func__);
- }
- done:
- return ret;
- }
- static int msm_pcm_capture_copy(struct snd_pcm_substream *substream,
- int channel, unsigned long hwoff,
- void __user *buf, unsigned long fbytes)
- {
- int ret = 0;
- struct hpcm_buf_node *buf_node = NULL;
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct hpcm_drv *prtd = runtime->private_data;
- struct dai_data *dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
- unsigned long dsp_flags;
- if (dai_data == NULL) {
- pr_err("%s, dai_data is null\n", __func__);
- ret = -EINVAL;
- goto done;
- }
- ret = wait_event_interruptible_timeout(dai_data->queue_wait,
- (!list_empty(&dai_data->filled_queue) ||
- dai_data->state == HPCM_STOPPED),
- 1 * HZ);
- if (ret > 0) {
- if (fbytes <= HPCM_MAX_VOC_PKT_SIZE) {
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- buf_node = list_first_entry(&dai_data->filled_queue,
- struct hpcm_buf_node, list);
- list_del(&buf_node->list);
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- ret = copy_to_user(buf, &buf_node->frame.voc_pkt,
- buf_node->frame.len);
- if (ret) {
- pr_err("%s: Copy to user returned %d\n",
- __func__, ret);
- ret = -EFAULT;
- }
- spin_lock_irqsave(&dai_data->dsp_lock, dsp_flags);
- list_add_tail(&buf_node->list, &dai_data->free_queue);
- spin_unlock_irqrestore(&dai_data->dsp_lock, dsp_flags);
- } else {
- pr_err("%s: Read count %lu > HPCM_MAX_VOC_PKT_SIZE\n",
- __func__, fbytes);
- ret = -ENOMEM;
- }
- } else if (ret == 0) {
- pr_err("%s: No Caputre data available\n", __func__);
- ret = -ETIMEDOUT;
- } else {
- pr_err("%s: Read was interrupted\n", __func__);
- ret = -ERESTARTSYS;
- }
- done:
- return ret;
- }
- static int msm_pcm_copy(struct snd_pcm_substream *substream, int channel,
- unsigned long hwoff, void __user *buf,
- unsigned long fbytes)
- {
- int ret = 0;
- if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
- ret = msm_pcm_playback_copy(substream, channel,
- hwoff, buf, fbytes);
- else if (substream->stream == SNDRV_PCM_STREAM_CAPTURE)
- ret = msm_pcm_capture_copy(substream, channel,
- hwoff, buf, fbytes);
- return ret;
- }
- static snd_pcm_uframes_t msm_pcm_pointer(struct snd_pcm_substream *substream)
- {
- struct dai_data *dai_data = NULL;
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct hpcm_drv *prtd = runtime->private_data;
- snd_pcm_uframes_t ret;
- dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
- if (dai_data == NULL) {
- pr_err("%s, dai_data is null\n", __func__);
- ret = 0;
- goto done;
- }
- if (dai_data->pcm_irq_pos >= dai_data->pcm_size)
- dai_data->pcm_irq_pos = 0;
- ret = bytes_to_frames(runtime, (dai_data->pcm_irq_pos));
- done:
- return ret;
- }
- static int msm_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
- {
- int ret = 0;
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct hpcm_drv *prtd = runtime->private_data;
- struct dai_data *dai_data =
- hpcm_get_dai_data(substream->pcm->id, prtd);
- if (dai_data == NULL) {
- pr_err("%s, dai_data is null\n", __func__);
- ret = -EINVAL;
- goto done;
- }
- pr_debug("%s, %s\n", __func__, substream->pcm->id);
- switch (cmd) {
- case SNDRV_PCM_TRIGGER_START:
- pr_debug("SNDRV_PCM_TRIGGER_START\n");
- dai_data->state = HPCM_STARTED;
- break;
- case SNDRV_PCM_TRIGGER_STOP:
- pr_debug("SNDRV_PCM_TRIGGER_STOP\n");
- dai_data->state = HPCM_STOPPED;
- break;
- default:
- ret = -EINVAL;
- break;
- }
- done:
- return ret;
- }
- static int msm_pcm_prepare(struct snd_pcm_substream *substream)
- {
- int ret = 0;
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct hpcm_drv *prtd = runtime->private_data;
- struct dai_data *dai_data = NULL;
- struct tap_point *tp = NULL;
- pr_debug("%s, %s\n", __func__, substream->pcm->id);
- mutex_lock(&prtd->lock);
- dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
- if (dai_data == NULL) {
- pr_err("%s, dai_data is null\n", __func__);
- ret = -EINVAL;
- goto done;
- }
- dai_data->pcm_size = snd_pcm_lib_buffer_bytes(substream);
- dai_data->pcm_count = snd_pcm_lib_period_bytes(substream);
- dai_data->pcm_irq_pos = 0;
- dai_data->pcm_buf_pos = 0;
- dai_data->state = HPCM_PREPARED;
- /* Register event notify processing callback in prepare instead of
- * init() as q6voice module's init() can be called at a later point
- */
- voc_register_hpcm_evt_cb(hpcm_notify_evt_processing, &hpcm_drv);
- tp = hpcm_get_tappoint_data(substream->pcm->id, prtd);
- if (tp != NULL) {
- ret = hpcm_start_vocpcm(substream->pcm->id, prtd, tp);
- if (ret) {
- pr_err("error sending start cmd err=%d\n", ret);
- goto done;
- }
- } else {
- pr_err("%s tp is NULL\n", __func__);
- }
- done:
- mutex_unlock(&prtd->lock);
- return ret;
- }
- static int msm_pcm_hw_params(struct snd_pcm_substream *substream,
- struct snd_pcm_hw_params *params)
- {
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct snd_dma_buffer *dma_buf = &substream->dma_buffer;
- struct hpcm_drv *prtd = (struct hpcm_drv *)runtime->private_data;
- int ret = 0;
- pr_debug("%s: %s\n", __func__, substream->pcm->id);
- mutex_lock(&prtd->lock);
- /* Allocate and map voice host PCM ion buffer */
- if (prtd->session[prtd->mixer_conf.sess_indx].sess_paddr == 0) {
- ret = hpcm_allocate_shared_memory(prtd);
- if (ret) {
- pr_err("error creating shared memory err=%d\n", ret);
- goto done;
- }
- ret = hpcm_map_vocpcm_memory(prtd);
- if (ret) {
- pr_err("error mapping shared memory err=%d\n", ret);
- hpcm_free_allocated_mem(prtd);
- goto done;
- }
- } else {
- pr_debug("%s, VHPCM memory allocation/mapping not performed\n"
- , __func__);
- }
- dma_buf->dev.type = SNDRV_DMA_TYPE_DEV;
- dma_buf->dev.dev = substream->pcm->card->dev;
- dma_buf->private_data = NULL;
- dma_buf->area = dma_alloc_coherent(substream->pcm->card->dev,
- runtime->hw.buffer_bytes_max,
- &dma_buf->addr, GFP_KERNEL);
- if (!dma_buf->area) {
- pr_err("%s:MSM dma_alloc failed\n", __func__);
- ret = -ENOMEM;
- goto done;
- }
- dma_buf->bytes = runtime->hw.buffer_bytes_max;
- memset(dma_buf->area, 0, runtime->hw.buffer_bytes_max);
- hpcm_create_free_queue(dma_buf,
- hpcm_get_dai_data(substream->pcm->id, prtd));
- snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer);
- done:
- mutex_unlock(&prtd->lock);
- return ret;
- }
- static int msm_pcm_open(struct snd_pcm_substream *substream)
- {
- struct snd_pcm_runtime *runtime = substream->runtime;
- struct hpcm_drv *prtd = &hpcm_drv;
- struct tappnt_mxr_data *tmd = NULL;
- struct dai_data *dai_data = NULL;
- int ret = 0;
- int tp_val = 0;
- pr_debug("%s, %s\n", __func__, substream->pcm->id);
- mutex_lock(&prtd->lock);
- dai_data = hpcm_get_dai_data(substream->pcm->id, prtd);
- if (dai_data == NULL) {
- pr_err("%s, dai_data is null\n", __func__);
- ret = -EINVAL;
- goto done;
- }
- runtime->hw = msm_pcm_hardware;
- ret = snd_pcm_hw_constraint_list(runtime, 0, SNDRV_PCM_HW_PARAM_RATE,
- &constraints_sample_rates);
- if (ret < 0)
- pr_debug("snd_pcm_hw_constraint_list failed\n");
- ret = snd_pcm_hw_constraint_integer(runtime,
- SNDRV_PCM_HW_PARAM_PERIODS);
- if (ret < 0) {
- pr_debug("snd_pcm_hw_constraint_integer failed\n");
- goto done;
- }
- tp_val = get_tappnt_value(substream->pcm->id);
- tmd = hpcm_get_tappnt_mixer_data(substream->pcm->id, prtd);
- /* Check wheather the kcontrol values set are valid */
- if (!tmd ||
- !(tmd->enable) ||
- !hpcm_is_valid_config(prtd->mixer_conf.sess_indx,
- tp_val, tmd->direction,
- tmd->sample_rate)) {
- ret = -EINVAL;
- goto done;
- }
- dai_data->substream = substream;
- runtime->private_data = prtd;
- done:
- mutex_unlock(&prtd->lock);
- return ret;
- }
- static const struct snd_pcm_ops msm_pcm_ops = {
- .open = msm_pcm_open,
- .hw_params = msm_pcm_hw_params,
- .prepare = msm_pcm_prepare,
- .trigger = msm_pcm_trigger,
- .pointer = msm_pcm_pointer,
- .copy_user = msm_pcm_copy,
- .close = msm_pcm_close,
- };
- static int msm_asoc_pcm_new(struct snd_soc_pcm_runtime *rtd)
- {
- struct snd_card *card = rtd->card->snd_card;
- pr_debug("%s:\n", __func__);
- if (!card->dev->coherent_dma_mask)
- card->dev->coherent_dma_mask = DMA_BIT_MASK(32);
- return 0;
- }
- static int msm_pcm_hpcm_probe(struct snd_soc_platform *platform)
- {
- snd_soc_add_platform_controls(platform, msm_hpcm_controls,
- ARRAY_SIZE(msm_hpcm_controls));
- return 0;
- }
- static struct snd_soc_platform_driver msm_soc_platform = {
- .ops = &msm_pcm_ops,
- .pcm_new = msm_asoc_pcm_new,
- .probe = msm_pcm_hpcm_probe,
- };
- static int msm_pcm_probe(struct platform_device *pdev)
- {
- pr_info("%s: dev name %s\n", __func__, dev_name(&pdev->dev));
- return snd_soc_register_platform(&pdev->dev, &msm_soc_platform);
- }
- static int msm_pcm_remove(struct platform_device *pdev)
- {
- snd_soc_unregister_platform(&pdev->dev);
- return 0;
- }
- static const struct of_device_id msm_voice_host_pcm_dt_match[] = {
- {.compatible = "qcom,msm-voice-host-pcm"},
- {}
- };
- MODULE_DEVICE_TABLE(of, msm_voice_host_pcm_dt_match);
- static struct platform_driver msm_pcm_driver = {
- .driver = {
- .name = "msm-voice-host-pcm",
- .owner = THIS_MODULE,
- .of_match_table = msm_voice_host_pcm_dt_match,
- },
- .probe = msm_pcm_probe,
- .remove = msm_pcm_remove,
- };
- int __init msm_voice_host_init(void)
- {
- int i = 0;
- struct session *s = NULL;
- memset(&hpcm_drv, 0, sizeof(hpcm_drv));
- mutex_init(&hpcm_drv.lock);
- for (i = 0; i < MAX_SESSION; i++) {
- s = &hpcm_drv.session[i];
- spin_lock_init(&s->rx_tap_point.capture_dai_data.dsp_lock);
- spin_lock_init(&s->rx_tap_point.playback_dai_data.dsp_lock);
- spin_lock_init(&s->tx_tap_point.capture_dai_data.dsp_lock);
- spin_lock_init(&s->tx_tap_point.playback_dai_data.dsp_lock);
- init_waitqueue_head(
- &s->rx_tap_point.capture_dai_data.queue_wait);
- init_waitqueue_head(
- &s->rx_tap_point.playback_dai_data.queue_wait);
- init_waitqueue_head(
- &s->tx_tap_point.capture_dai_data.queue_wait);
- init_waitqueue_head(
- &s->tx_tap_point.playback_dai_data.queue_wait);
- INIT_LIST_HEAD(&s->rx_tap_point.capture_dai_data.filled_queue);
- INIT_LIST_HEAD(&s->rx_tap_point.capture_dai_data.free_queue);
- INIT_LIST_HEAD(&s->rx_tap_point.playback_dai_data.filled_queue);
- INIT_LIST_HEAD(&s->rx_tap_point.playback_dai_data.free_queue);
- INIT_LIST_HEAD(&s->tx_tap_point.capture_dai_data.filled_queue);
- INIT_LIST_HEAD(&s->tx_tap_point.capture_dai_data.free_queue);
- INIT_LIST_HEAD(&s->tx_tap_point.playback_dai_data.filled_queue);
- INIT_LIST_HEAD(&s->tx_tap_point.playback_dai_data.free_queue);
- }
- return platform_driver_register(&msm_pcm_driver);
- }
- void msm_voice_host_exit(void)
- {
- platform_driver_unregister(&msm_pcm_driver);
- }
- MODULE_DESCRIPTION("PCM module platform driver");
- MODULE_LICENSE("GPL v2");
|