1b97169daSJie Yang /* 2b97169daSJie Yang * sst-atom-controls.c - Intel MID Platform driver DPCM ALSA controls for Mrfld 3b97169daSJie Yang * 4b97169daSJie Yang * Copyright (C) 2013-14 Intel Corp 5b97169daSJie Yang * Author: Omair Mohammed Abdullah <omair.m.abdullah@intel.com> 6b97169daSJie Yang * Vinod Koul <vinod.koul@intel.com> 7b97169daSJie Yang * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 8b97169daSJie Yang * 9b97169daSJie Yang * This program is free software; you can redistribute it and/or modify 10b97169daSJie Yang * it under the terms of the GNU General Public License as published by 11b97169daSJie Yang * the Free Software Foundation; version 2 of the License. 12b97169daSJie Yang * 13b97169daSJie Yang * This program is distributed in the hope that it will be useful, but 14b97169daSJie Yang * WITHOUT ANY WARRANTY; without even the implied warranty of 15b97169daSJie Yang * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 16b97169daSJie Yang * General Public License for more details. 17b97169daSJie Yang * 18b97169daSJie Yang * In the dpcm driver modelling when a particular FE/BE/Mixer/Pipe is active 19b97169daSJie Yang * we forward the settings and parameters, rest we keep the values in 20b97169daSJie Yang * driver and forward when DAPM enables them 21b97169daSJie Yang * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 22b97169daSJie Yang */ 23b97169daSJie Yang #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt 24b97169daSJie Yang 25b97169daSJie Yang #include <linux/slab.h> 26b97169daSJie Yang #include <sound/soc.h> 27b97169daSJie Yang #include <sound/tlv.h> 28b97169daSJie Yang #include "sst-mfld-platform.h" 29b97169daSJie Yang #include "sst-atom-controls.h" 30b97169daSJie Yang 31b97169daSJie Yang static int sst_fill_byte_control(struct sst_data *drv, 32b97169daSJie Yang u8 ipc_msg, u8 block, 33b97169daSJie Yang u8 task_id, u8 pipe_id, 34b97169daSJie Yang u16 len, void *cmd_data) 35b97169daSJie Yang { 36b97169daSJie Yang struct snd_sst_bytes_v2 *byte_data = drv->byte_stream; 37b97169daSJie Yang 38b97169daSJie Yang byte_data->type = SST_CMD_BYTES_SET; 39b97169daSJie Yang byte_data->ipc_msg = ipc_msg; 40b97169daSJie Yang byte_data->block = block; 41b97169daSJie Yang byte_data->task_id = task_id; 42b97169daSJie Yang byte_data->pipe_id = pipe_id; 43b97169daSJie Yang 44b97169daSJie Yang if (len > SST_MAX_BIN_BYTES - sizeof(*byte_data)) { 45b97169daSJie Yang dev_err(&drv->pdev->dev, "command length too big (%u)", len); 46b97169daSJie Yang return -EINVAL; 47b97169daSJie Yang } 48b97169daSJie Yang byte_data->len = len; 49b97169daSJie Yang memcpy(byte_data->bytes, cmd_data, len); 50b97169daSJie Yang print_hex_dump_bytes("writing to lpe: ", DUMP_PREFIX_OFFSET, 51b97169daSJie Yang byte_data, len + sizeof(*byte_data)); 52b97169daSJie Yang return 0; 53b97169daSJie Yang } 54b97169daSJie Yang 55b97169daSJie Yang static int sst_fill_and_send_cmd_unlocked(struct sst_data *drv, 56b97169daSJie Yang u8 ipc_msg, u8 block, u8 task_id, u8 pipe_id, 57b97169daSJie Yang void *cmd_data, u16 len) 58b97169daSJie Yang { 59b97169daSJie Yang int ret = 0; 60b97169daSJie Yang 61b97169daSJie Yang ret = sst_fill_byte_control(drv, ipc_msg, 62b97169daSJie Yang block, task_id, pipe_id, len, cmd_data); 63b97169daSJie Yang if (ret < 0) 64b97169daSJie Yang return ret; 65b97169daSJie Yang return sst->ops->send_byte_stream(sst->dev, drv->byte_stream); 66b97169daSJie Yang } 67b97169daSJie Yang 68b97169daSJie Yang /** 69b97169daSJie Yang * sst_fill_and_send_cmd - generate the IPC message and send it to the FW 70b97169daSJie Yang * @ipc_msg: type of IPC (CMD, SET_PARAMS, GET_PARAMS) 71b97169daSJie Yang * @cmd_data: the IPC payload 72b97169daSJie Yang */ 73b97169daSJie Yang static int sst_fill_and_send_cmd(struct sst_data *drv, 74b97169daSJie Yang u8 ipc_msg, u8 block, u8 task_id, u8 pipe_id, 75b97169daSJie Yang void *cmd_data, u16 len) 76b97169daSJie Yang { 77b97169daSJie Yang int ret; 78b97169daSJie Yang 79b97169daSJie Yang mutex_lock(&drv->lock); 80b97169daSJie Yang ret = sst_fill_and_send_cmd_unlocked(drv, ipc_msg, block, 81b97169daSJie Yang task_id, pipe_id, cmd_data, len); 82b97169daSJie Yang mutex_unlock(&drv->lock); 83b97169daSJie Yang 84b97169daSJie Yang return ret; 85b97169daSJie Yang } 86b97169daSJie Yang 87b97169daSJie Yang /** 88b97169daSJie Yang * tx map value is a bitfield where each bit represents a FW channel 89b97169daSJie Yang * 90b97169daSJie Yang * 3 2 1 0 # 0 = codec0, 1 = codec1 91b97169daSJie Yang * RLRLRLRL # 3, 4 = reserved 92b97169daSJie Yang * 93b97169daSJie Yang * e.g. slot 0 rx map = 00001100b -> data from slot 0 goes into codec_in1 L,R 94b97169daSJie Yang */ 95b97169daSJie Yang static u8 sst_ssp_tx_map[SST_MAX_TDM_SLOTS] = { 96b97169daSJie Yang 0x1, 0x2, 0x4, 0x8, 0x10, 0x20, 0x40, 0x80, /* default rx map */ 97b97169daSJie Yang }; 98b97169daSJie Yang 99b97169daSJie Yang /** 100b97169daSJie Yang * rx map value is a bitfield where each bit represents a slot 101b97169daSJie Yang * 102b97169daSJie Yang * 76543210 # 0 = slot 0, 1 = slot 1 103b97169daSJie Yang * 104b97169daSJie Yang * e.g. codec1_0 tx map = 00000101b -> data from codec_out1_0 goes into slot 0, 2 105b97169daSJie Yang */ 106b97169daSJie Yang static u8 sst_ssp_rx_map[SST_MAX_TDM_SLOTS] = { 107b97169daSJie Yang 0x1, 0x2, 0x4, 0x8, 0x10, 0x20, 0x40, 0x80, /* default tx map */ 108b97169daSJie Yang }; 109b97169daSJie Yang 110b97169daSJie Yang /** 111b97169daSJie Yang * NOTE: this is invoked with lock held 112b97169daSJie Yang */ 113b97169daSJie Yang static int sst_send_slot_map(struct sst_data *drv) 114b97169daSJie Yang { 115b97169daSJie Yang struct sst_param_sba_ssp_slot_map cmd; 116b97169daSJie Yang 117b97169daSJie Yang SST_FILL_DEFAULT_DESTINATION(cmd.header.dst); 118b97169daSJie Yang cmd.header.command_id = SBA_SET_SSP_SLOT_MAP; 119b97169daSJie Yang cmd.header.length = sizeof(struct sst_param_sba_ssp_slot_map) 120b97169daSJie Yang - sizeof(struct sst_dsp_header); 121b97169daSJie Yang 122b97169daSJie Yang cmd.param_id = SBA_SET_SSP_SLOT_MAP; 123b97169daSJie Yang cmd.param_len = sizeof(cmd.rx_slot_map) + sizeof(cmd.tx_slot_map) 124b97169daSJie Yang + sizeof(cmd.ssp_index); 125b97169daSJie Yang cmd.ssp_index = SSP_CODEC; 126b97169daSJie Yang 127b97169daSJie Yang memcpy(cmd.rx_slot_map, &sst_ssp_tx_map[0], sizeof(cmd.rx_slot_map)); 128b97169daSJie Yang memcpy(cmd.tx_slot_map, &sst_ssp_rx_map[0], sizeof(cmd.tx_slot_map)); 129b97169daSJie Yang 130b97169daSJie Yang return sst_fill_and_send_cmd_unlocked(drv, SST_IPC_IA_SET_PARAMS, 131b97169daSJie Yang SST_FLAG_BLOCKED, SST_TASK_SBA, 0, &cmd, 132b97169daSJie Yang sizeof(cmd.header) + cmd.header.length); 133b97169daSJie Yang } 134b97169daSJie Yang 1359569909fSVinod Koul static int sst_slot_enum_info(struct snd_kcontrol *kcontrol, 136b97169daSJie Yang struct snd_ctl_elem_info *uinfo) 137b97169daSJie Yang { 138b97169daSJie Yang struct sst_enum *e = (struct sst_enum *)kcontrol->private_value; 139b97169daSJie Yang 140b97169daSJie Yang uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED; 141b97169daSJie Yang uinfo->count = 1; 142b97169daSJie Yang uinfo->value.enumerated.items = e->max; 143b97169daSJie Yang 144b97169daSJie Yang if (uinfo->value.enumerated.item > e->max - 1) 145b97169daSJie Yang uinfo->value.enumerated.item = e->max - 1; 146b97169daSJie Yang strcpy(uinfo->value.enumerated.name, 147b97169daSJie Yang e->texts[uinfo->value.enumerated.item]); 148b97169daSJie Yang 149b97169daSJie Yang return 0; 150b97169daSJie Yang } 151b97169daSJie Yang 152b97169daSJie Yang /** 153b97169daSJie Yang * sst_slot_get - get the status of the interleaver/deinterleaver control 154b97169daSJie Yang * 155b97169daSJie Yang * Searches the map where the control status is stored, and gets the 156b97169daSJie Yang * channel/slot which is currently set for this enumerated control. Since it is 157b97169daSJie Yang * an enumerated control, there is only one possible value. 158b97169daSJie Yang */ 159b97169daSJie Yang static int sst_slot_get(struct snd_kcontrol *kcontrol, 160b97169daSJie Yang struct snd_ctl_elem_value *ucontrol) 161b97169daSJie Yang { 162b97169daSJie Yang struct sst_enum *e = (void *)kcontrol->private_value; 163b97169daSJie Yang struct snd_soc_component *c = snd_kcontrol_chip(kcontrol); 164b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(c); 165b97169daSJie Yang unsigned int ctl_no = e->reg; 166b97169daSJie Yang unsigned int is_tx = e->tx; 167b97169daSJie Yang unsigned int val, mux; 168b97169daSJie Yang u8 *map = is_tx ? sst_ssp_rx_map : sst_ssp_tx_map; 169b97169daSJie Yang 170b97169daSJie Yang mutex_lock(&drv->lock); 171b97169daSJie Yang val = 1 << ctl_no; 172b97169daSJie Yang /* search which slot/channel has this bit set - there should be only one */ 173b97169daSJie Yang for (mux = e->max; mux > 0; mux--) 174b97169daSJie Yang if (map[mux - 1] & val) 175b97169daSJie Yang break; 176b97169daSJie Yang 177b97169daSJie Yang ucontrol->value.enumerated.item[0] = mux; 178b97169daSJie Yang mutex_unlock(&drv->lock); 179b97169daSJie Yang 180b97169daSJie Yang dev_dbg(c->dev, "%s - %s map = %#x\n", 181b97169daSJie Yang is_tx ? "tx channel" : "rx slot", 182b97169daSJie Yang e->texts[mux], mux ? map[mux - 1] : -1); 183b97169daSJie Yang return 0; 184b97169daSJie Yang } 185b97169daSJie Yang 186b97169daSJie Yang /* sst_check_and_send_slot_map - helper for checking power state and sending 187b97169daSJie Yang * slot map cmd 188b97169daSJie Yang * 189b97169daSJie Yang * called with lock held 190b97169daSJie Yang */ 191b97169daSJie Yang static int sst_check_and_send_slot_map(struct sst_data *drv, struct snd_kcontrol *kcontrol) 192b97169daSJie Yang { 193b97169daSJie Yang struct sst_enum *e = (void *)kcontrol->private_value; 194b97169daSJie Yang int ret = 0; 195b97169daSJie Yang 196b97169daSJie Yang if (e->w && e->w->power) 197b97169daSJie Yang ret = sst_send_slot_map(drv); 198fba0d706SVinod Koul else if (!e->w) 199b97169daSJie Yang dev_err(&drv->pdev->dev, "Slot control: %s doesn't have DAPM widget!!!\n", 200b97169daSJie Yang kcontrol->id.name); 201b97169daSJie Yang return ret; 202b97169daSJie Yang } 203b97169daSJie Yang 204b97169daSJie Yang /** 205b97169daSJie Yang * sst_slot_put - set the status of interleaver/deinterleaver control 206b97169daSJie Yang * 207b97169daSJie Yang * (de)interleaver controls are defined in opposite sense to be user-friendly 208b97169daSJie Yang * 209b97169daSJie Yang * Instead of the enum value being the value written to the register, it is the 210b97169daSJie Yang * register address; and the kcontrol number (register num) is the value written 211b97169daSJie Yang * to the register. This is so that there can be only one value for each 212b97169daSJie Yang * slot/channel since there is only one control for each slot/channel. 213b97169daSJie Yang * 214b97169daSJie Yang * This means that whenever an enum is set, we need to clear the bit 215b97169daSJie Yang * for that kcontrol_no for all the interleaver OR deinterleaver registers 216b97169daSJie Yang */ 217b97169daSJie Yang static int sst_slot_put(struct snd_kcontrol *kcontrol, 218b97169daSJie Yang struct snd_ctl_elem_value *ucontrol) 219b97169daSJie Yang { 220b97169daSJie Yang struct snd_soc_component *c = snd_soc_kcontrol_component(kcontrol); 221b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(c); 222b97169daSJie Yang struct sst_enum *e = (void *)kcontrol->private_value; 223b97169daSJie Yang int i, ret = 0; 224b97169daSJie Yang unsigned int ctl_no = e->reg; 225b97169daSJie Yang unsigned int is_tx = e->tx; 226b97169daSJie Yang unsigned int slot_channel_no; 227b97169daSJie Yang unsigned int val, mux; 228b97169daSJie Yang u8 *map; 229b97169daSJie Yang 230b97169daSJie Yang map = is_tx ? sst_ssp_rx_map : sst_ssp_tx_map; 231b97169daSJie Yang 232b97169daSJie Yang val = 1 << ctl_no; 233b97169daSJie Yang mux = ucontrol->value.enumerated.item[0]; 234b97169daSJie Yang if (mux > e->max - 1) 235b97169daSJie Yang return -EINVAL; 236b97169daSJie Yang 237b97169daSJie Yang mutex_lock(&drv->lock); 238b97169daSJie Yang /* first clear all registers of this bit */ 239b97169daSJie Yang for (i = 0; i < e->max; i++) 240b97169daSJie Yang map[i] &= ~val; 241b97169daSJie Yang 242b97169daSJie Yang if (mux == 0) { 243b97169daSJie Yang /* kctl set to 'none' and we reset the bits so send IPC */ 244b97169daSJie Yang ret = sst_check_and_send_slot_map(drv, kcontrol); 245b97169daSJie Yang 246b97169daSJie Yang mutex_unlock(&drv->lock); 247b97169daSJie Yang return ret; 248b97169daSJie Yang } 249b97169daSJie Yang 250b97169daSJie Yang /* offset by one to take "None" into account */ 251b97169daSJie Yang slot_channel_no = mux - 1; 252b97169daSJie Yang map[slot_channel_no] |= val; 253b97169daSJie Yang 254b97169daSJie Yang dev_dbg(c->dev, "%s %s map = %#x\n", 255b97169daSJie Yang is_tx ? "tx channel" : "rx slot", 256b97169daSJie Yang e->texts[mux], map[slot_channel_no]); 257b97169daSJie Yang 258b97169daSJie Yang ret = sst_check_and_send_slot_map(drv, kcontrol); 259b97169daSJie Yang 260b97169daSJie Yang mutex_unlock(&drv->lock); 261b97169daSJie Yang return ret; 262b97169daSJie Yang } 263b97169daSJie Yang 264b97169daSJie Yang static int sst_send_algo_cmd(struct sst_data *drv, 265b97169daSJie Yang struct sst_algo_control *bc) 266b97169daSJie Yang { 267b97169daSJie Yang int len, ret = 0; 268b97169daSJie Yang struct sst_cmd_set_params *cmd; 269b97169daSJie Yang 270b97169daSJie Yang /*bc->max includes sizeof algos + length field*/ 271b97169daSJie Yang len = sizeof(cmd->dst) + sizeof(cmd->command_id) + bc->max; 272b97169daSJie Yang 273b97169daSJie Yang cmd = kzalloc(len, GFP_KERNEL); 274b97169daSJie Yang if (cmd == NULL) 275b97169daSJie Yang return -ENOMEM; 276b97169daSJie Yang 277b97169daSJie Yang SST_FILL_DESTINATION(2, cmd->dst, bc->pipe_id, bc->module_id); 278b97169daSJie Yang cmd->command_id = bc->cmd_id; 279b97169daSJie Yang memcpy(cmd->params, bc->params, bc->max); 280b97169daSJie Yang 281b97169daSJie Yang ret = sst_fill_and_send_cmd_unlocked(drv, SST_IPC_IA_SET_PARAMS, 282b97169daSJie Yang SST_FLAG_BLOCKED, bc->task_id, 0, cmd, len); 283b97169daSJie Yang kfree(cmd); 284b97169daSJie Yang return ret; 285b97169daSJie Yang } 286b97169daSJie Yang 287b97169daSJie Yang /** 288b97169daSJie Yang * sst_find_and_send_pipe_algo - send all the algo parameters for a pipe 289b97169daSJie Yang * 290b97169daSJie Yang * The algos which are in each pipeline are sent to the firmware one by one 291b97169daSJie Yang * 292b97169daSJie Yang * Called with lock held 293b97169daSJie Yang */ 294b97169daSJie Yang static int sst_find_and_send_pipe_algo(struct sst_data *drv, 295b97169daSJie Yang const char *pipe, struct sst_ids *ids) 296b97169daSJie Yang { 297b97169daSJie Yang int ret = 0; 298b97169daSJie Yang struct sst_algo_control *bc; 299b97169daSJie Yang struct sst_module *algo = NULL; 300b97169daSJie Yang 301b97169daSJie Yang dev_dbg(&drv->pdev->dev, "Enter: widget=%s\n", pipe); 302b97169daSJie Yang 303b97169daSJie Yang list_for_each_entry(algo, &ids->algo_list, node) { 304b97169daSJie Yang bc = (void *)algo->kctl->private_value; 305b97169daSJie Yang 306b97169daSJie Yang dev_dbg(&drv->pdev->dev, "Found algo control name=%s pipe=%s\n", 307b97169daSJie Yang algo->kctl->id.name, pipe); 308b97169daSJie Yang ret = sst_send_algo_cmd(drv, bc); 309b97169daSJie Yang if (ret) 310b97169daSJie Yang return ret; 311b97169daSJie Yang } 312b97169daSJie Yang return ret; 313b97169daSJie Yang } 314b97169daSJie Yang 315b97169daSJie Yang static int sst_algo_bytes_ctl_info(struct snd_kcontrol *kcontrol, 316b97169daSJie Yang struct snd_ctl_elem_info *uinfo) 317b97169daSJie Yang { 318b97169daSJie Yang struct sst_algo_control *bc = (void *)kcontrol->private_value; 319b97169daSJie Yang 320b97169daSJie Yang uinfo->type = SNDRV_CTL_ELEM_TYPE_BYTES; 321b97169daSJie Yang uinfo->count = bc->max; 322b97169daSJie Yang 323b97169daSJie Yang return 0; 324b97169daSJie Yang } 325b97169daSJie Yang 326b97169daSJie Yang static int sst_algo_control_get(struct snd_kcontrol *kcontrol, 327b97169daSJie Yang struct snd_ctl_elem_value *ucontrol) 328b97169daSJie Yang { 329b97169daSJie Yang struct sst_algo_control *bc = (void *)kcontrol->private_value; 330b97169daSJie Yang struct snd_soc_component *component = snd_kcontrol_chip(kcontrol); 331b97169daSJie Yang 332b97169daSJie Yang switch (bc->type) { 333b97169daSJie Yang case SST_ALGO_PARAMS: 334b97169daSJie Yang memcpy(ucontrol->value.bytes.data, bc->params, bc->max); 335b97169daSJie Yang break; 336b97169daSJie Yang default: 337b97169daSJie Yang dev_err(component->dev, "Invalid Input- algo type:%d\n", 338b97169daSJie Yang bc->type); 339b97169daSJie Yang return -EINVAL; 340b97169daSJie Yang 341b97169daSJie Yang } 342b97169daSJie Yang return 0; 343b97169daSJie Yang } 344b97169daSJie Yang 345b97169daSJie Yang static int sst_algo_control_set(struct snd_kcontrol *kcontrol, 346b97169daSJie Yang struct snd_ctl_elem_value *ucontrol) 347b97169daSJie Yang { 348b97169daSJie Yang int ret = 0; 349b97169daSJie Yang struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 350b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(cmpnt); 351b97169daSJie Yang struct sst_algo_control *bc = (void *)kcontrol->private_value; 352b97169daSJie Yang 353b97169daSJie Yang dev_dbg(cmpnt->dev, "control_name=%s\n", kcontrol->id.name); 354b97169daSJie Yang mutex_lock(&drv->lock); 355b97169daSJie Yang switch (bc->type) { 356b97169daSJie Yang case SST_ALGO_PARAMS: 357b97169daSJie Yang memcpy(bc->params, ucontrol->value.bytes.data, bc->max); 358b97169daSJie Yang break; 359b97169daSJie Yang default: 360b97169daSJie Yang mutex_unlock(&drv->lock); 361b97169daSJie Yang dev_err(cmpnt->dev, "Invalid Input- algo type:%d\n", 362b97169daSJie Yang bc->type); 363b97169daSJie Yang return -EINVAL; 364b97169daSJie Yang } 365b97169daSJie Yang /*if pipe is enabled, need to send the algo params from here*/ 366b97169daSJie Yang if (bc->w && bc->w->power) 367b97169daSJie Yang ret = sst_send_algo_cmd(drv, bc); 368b97169daSJie Yang mutex_unlock(&drv->lock); 369b97169daSJie Yang 370b97169daSJie Yang return ret; 371b97169daSJie Yang } 372b97169daSJie Yang 373b97169daSJie Yang static int sst_gain_ctl_info(struct snd_kcontrol *kcontrol, 374b97169daSJie Yang struct snd_ctl_elem_info *uinfo) 375b97169daSJie Yang { 376b97169daSJie Yang struct sst_gain_mixer_control *mc = (void *)kcontrol->private_value; 377b97169daSJie Yang 378b97169daSJie Yang uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; 379b97169daSJie Yang uinfo->count = mc->stereo ? 2 : 1; 380b97169daSJie Yang uinfo->value.integer.min = mc->min; 381b97169daSJie Yang uinfo->value.integer.max = mc->max; 382b97169daSJie Yang 383b97169daSJie Yang return 0; 384b97169daSJie Yang } 385b97169daSJie Yang 386b97169daSJie Yang /** 387b97169daSJie Yang * sst_send_gain_cmd - send the gain algorithm IPC to the FW 388b97169daSJie Yang * @gv: the stored value of gain (also contains rampduration) 389b97169daSJie Yang * @mute: flag that indicates whether this was called from the 390b97169daSJie Yang * digital_mute callback or directly. If called from the 391b97169daSJie Yang * digital_mute callback, module will be muted/unmuted based on this 392b97169daSJie Yang * flag. The flag is always 0 if called directly. 393b97169daSJie Yang * 394b97169daSJie Yang * Called with sst_data.lock held 395b97169daSJie Yang * 396b97169daSJie Yang * The user-set gain value is sent only if the user-controllable 'mute' control 397b97169daSJie Yang * is OFF (indicated by gv->mute). Otherwise, the mute value (MIN value) is 398b97169daSJie Yang * sent. 399b97169daSJie Yang */ 400b97169daSJie Yang static int sst_send_gain_cmd(struct sst_data *drv, struct sst_gain_value *gv, 401b97169daSJie Yang u16 task_id, u16 loc_id, u16 module_id, int mute) 402b97169daSJie Yang { 403b97169daSJie Yang struct sst_cmd_set_gain_dual cmd; 404b97169daSJie Yang 405b97169daSJie Yang dev_dbg(&drv->pdev->dev, "Enter\n"); 406b97169daSJie Yang 407b97169daSJie Yang cmd.header.command_id = MMX_SET_GAIN; 408b97169daSJie Yang SST_FILL_DEFAULT_DESTINATION(cmd.header.dst); 409b97169daSJie Yang cmd.gain_cell_num = 1; 410b97169daSJie Yang 411b97169daSJie Yang if (mute || gv->mute) { 412b97169daSJie Yang cmd.cell_gains[0].cell_gain_left = SST_GAIN_MIN_VALUE; 413b97169daSJie Yang cmd.cell_gains[0].cell_gain_right = SST_GAIN_MIN_VALUE; 414b97169daSJie Yang } else { 415b97169daSJie Yang cmd.cell_gains[0].cell_gain_left = gv->l_gain; 416b97169daSJie Yang cmd.cell_gains[0].cell_gain_right = gv->r_gain; 417b97169daSJie Yang } 418b97169daSJie Yang 419b97169daSJie Yang SST_FILL_DESTINATION(2, cmd.cell_gains[0].dest, 420b97169daSJie Yang loc_id, module_id); 421b97169daSJie Yang cmd.cell_gains[0].gain_time_constant = gv->ramp_duration; 422b97169daSJie Yang 423b97169daSJie Yang cmd.header.length = sizeof(struct sst_cmd_set_gain_dual) 424b97169daSJie Yang - sizeof(struct sst_dsp_header); 425b97169daSJie Yang 426b97169daSJie Yang /* we are with lock held, so call the unlocked api to send */ 427b97169daSJie Yang return sst_fill_and_send_cmd_unlocked(drv, SST_IPC_IA_SET_PARAMS, 428b97169daSJie Yang SST_FLAG_BLOCKED, task_id, 0, &cmd, 429b97169daSJie Yang sizeof(cmd.header) + cmd.header.length); 430b97169daSJie Yang } 431b97169daSJie Yang 432b97169daSJie Yang static int sst_gain_get(struct snd_kcontrol *kcontrol, 433b97169daSJie Yang struct snd_ctl_elem_value *ucontrol) 434b97169daSJie Yang { 435b97169daSJie Yang struct snd_soc_component *component = snd_kcontrol_chip(kcontrol); 436b97169daSJie Yang struct sst_gain_mixer_control *mc = (void *)kcontrol->private_value; 437b97169daSJie Yang struct sst_gain_value *gv = mc->gain_val; 438b97169daSJie Yang 439b97169daSJie Yang switch (mc->type) { 440b97169daSJie Yang case SST_GAIN_TLV: 441b97169daSJie Yang ucontrol->value.integer.value[0] = gv->l_gain; 442b97169daSJie Yang ucontrol->value.integer.value[1] = gv->r_gain; 443b97169daSJie Yang break; 444b97169daSJie Yang 445b97169daSJie Yang case SST_GAIN_MUTE: 446940a5a01SPierre-Louis Bossart ucontrol->value.integer.value[0] = gv->mute ? 0 : 1; 447b97169daSJie Yang break; 448b97169daSJie Yang 449b97169daSJie Yang case SST_GAIN_RAMP_DURATION: 450b97169daSJie Yang ucontrol->value.integer.value[0] = gv->ramp_duration; 451b97169daSJie Yang break; 452b97169daSJie Yang 453b97169daSJie Yang default: 454b97169daSJie Yang dev_err(component->dev, "Invalid Input- gain type:%d\n", 455b97169daSJie Yang mc->type); 456b97169daSJie Yang return -EINVAL; 457b97169daSJie Yang } 458b97169daSJie Yang 459b97169daSJie Yang return 0; 460b97169daSJie Yang } 461b97169daSJie Yang 462b97169daSJie Yang static int sst_gain_put(struct snd_kcontrol *kcontrol, 463b97169daSJie Yang struct snd_ctl_elem_value *ucontrol) 464b97169daSJie Yang { 465b97169daSJie Yang int ret = 0; 466b97169daSJie Yang struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 467b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(cmpnt); 468b97169daSJie Yang struct sst_gain_mixer_control *mc = (void *)kcontrol->private_value; 469b97169daSJie Yang struct sst_gain_value *gv = mc->gain_val; 470b97169daSJie Yang 471b97169daSJie Yang mutex_lock(&drv->lock); 472b97169daSJie Yang 473b97169daSJie Yang switch (mc->type) { 474b97169daSJie Yang case SST_GAIN_TLV: 475b97169daSJie Yang gv->l_gain = ucontrol->value.integer.value[0]; 476b97169daSJie Yang gv->r_gain = ucontrol->value.integer.value[1]; 477b97169daSJie Yang dev_dbg(cmpnt->dev, "%s: Volume %d, %d\n", 478b97169daSJie Yang mc->pname, gv->l_gain, gv->r_gain); 479b97169daSJie Yang break; 480b97169daSJie Yang 481b97169daSJie Yang case SST_GAIN_MUTE: 482940a5a01SPierre-Louis Bossart gv->mute = !ucontrol->value.integer.value[0]; 483b97169daSJie Yang dev_dbg(cmpnt->dev, "%s: Mute %d\n", mc->pname, gv->mute); 484b97169daSJie Yang break; 485b97169daSJie Yang 486b97169daSJie Yang case SST_GAIN_RAMP_DURATION: 487b97169daSJie Yang gv->ramp_duration = ucontrol->value.integer.value[0]; 488b97169daSJie Yang dev_dbg(cmpnt->dev, "%s: Ramp Delay%d\n", 489b97169daSJie Yang mc->pname, gv->ramp_duration); 490b97169daSJie Yang break; 491b97169daSJie Yang 492b97169daSJie Yang default: 493b97169daSJie Yang mutex_unlock(&drv->lock); 494b97169daSJie Yang dev_err(cmpnt->dev, "Invalid Input- gain type:%d\n", 495b97169daSJie Yang mc->type); 496b97169daSJie Yang return -EINVAL; 497b97169daSJie Yang } 498b97169daSJie Yang 499b97169daSJie Yang if (mc->w && mc->w->power) 500b97169daSJie Yang ret = sst_send_gain_cmd(drv, gv, mc->task_id, 501b97169daSJie Yang mc->pipe_id | mc->instance_id, mc->module_id, 0); 502b97169daSJie Yang mutex_unlock(&drv->lock); 503b97169daSJie Yang 504b97169daSJie Yang return ret; 505b97169daSJie Yang } 506b97169daSJie Yang 507b97169daSJie Yang static int sst_set_pipe_gain(struct sst_ids *ids, 508b97169daSJie Yang struct sst_data *drv, int mute); 509b97169daSJie Yang 510b97169daSJie Yang static int sst_send_pipe_module_params(struct snd_soc_dapm_widget *w, 511b97169daSJie Yang struct snd_kcontrol *kcontrol) 512b97169daSJie Yang { 513b97169daSJie Yang struct snd_soc_component *c = snd_soc_dapm_to_component(w->dapm); 514b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(c); 515b97169daSJie Yang struct sst_ids *ids = w->priv; 516b97169daSJie Yang 517b97169daSJie Yang mutex_lock(&drv->lock); 518b97169daSJie Yang sst_find_and_send_pipe_algo(drv, w->name, ids); 519b97169daSJie Yang sst_set_pipe_gain(ids, drv, 0); 520b97169daSJie Yang mutex_unlock(&drv->lock); 521b97169daSJie Yang 522b97169daSJie Yang return 0; 523b97169daSJie Yang } 524b97169daSJie Yang 525b97169daSJie Yang static int sst_generic_modules_event(struct snd_soc_dapm_widget *w, 526b97169daSJie Yang struct snd_kcontrol *k, int event) 527b97169daSJie Yang { 528b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event)) 529b97169daSJie Yang return sst_send_pipe_module_params(w, k); 530b97169daSJie Yang return 0; 531b97169daSJie Yang } 532b97169daSJie Yang 533b97169daSJie Yang static const DECLARE_TLV_DB_SCALE(sst_gain_tlv_common, SST_GAIN_MIN_VALUE * 10, 10, 0); 534b97169daSJie Yang 535b97169daSJie Yang /* Look up table to convert MIXER SW bit regs to SWM inputs */ 536b97169daSJie Yang static const uint swm_mixer_input_ids[SST_SWM_INPUT_COUNT] = { 5373d240d73SPierre-Louis Bossart [SST_IP_MODEM] = SST_SWM_IN_MODEM, 538b97169daSJie Yang [SST_IP_CODEC0] = SST_SWM_IN_CODEC0, 539b97169daSJie Yang [SST_IP_CODEC1] = SST_SWM_IN_CODEC1, 540b97169daSJie Yang [SST_IP_LOOP0] = SST_SWM_IN_SPROT_LOOP, 541b97169daSJie Yang [SST_IP_LOOP1] = SST_SWM_IN_MEDIA_LOOP1, 542b97169daSJie Yang [SST_IP_LOOP2] = SST_SWM_IN_MEDIA_LOOP2, 543b97169daSJie Yang [SST_IP_PCM0] = SST_SWM_IN_PCM0, 544b97169daSJie Yang [SST_IP_PCM1] = SST_SWM_IN_PCM1, 545b97169daSJie Yang [SST_IP_MEDIA0] = SST_SWM_IN_MEDIA0, 546b97169daSJie Yang [SST_IP_MEDIA1] = SST_SWM_IN_MEDIA1, 547b97169daSJie Yang [SST_IP_MEDIA2] = SST_SWM_IN_MEDIA2, 548b97169daSJie Yang [SST_IP_MEDIA3] = SST_SWM_IN_MEDIA3, 549b97169daSJie Yang }; 550b97169daSJie Yang 551b97169daSJie Yang /** 552b97169daSJie Yang * fill_swm_input - fill in the SWM input ids given the register 553b97169daSJie Yang * 554b97169daSJie Yang * The register value is a bit-field inicated which mixer inputs are ON. Use the 555b97169daSJie Yang * lookup table to get the input-id and fill it in the structure. 556b97169daSJie Yang */ 557b97169daSJie Yang static int fill_swm_input(struct snd_soc_component *cmpnt, 558b97169daSJie Yang struct swm_input_ids *swm_input, unsigned int reg) 559b97169daSJie Yang { 560b97169daSJie Yang uint i, is_set, nb_inputs = 0; 561b97169daSJie Yang u16 input_loc_id; 562b97169daSJie Yang 563b97169daSJie Yang dev_dbg(cmpnt->dev, "reg: %#x\n", reg); 564b97169daSJie Yang for (i = 0; i < SST_SWM_INPUT_COUNT; i++) { 565b97169daSJie Yang is_set = reg & BIT(i); 566b97169daSJie Yang if (!is_set) 567b97169daSJie Yang continue; 568b97169daSJie Yang 569b97169daSJie Yang input_loc_id = swm_mixer_input_ids[i]; 570b97169daSJie Yang SST_FILL_DESTINATION(2, swm_input->input_id, 571b97169daSJie Yang input_loc_id, SST_DEFAULT_MODULE_ID); 572b97169daSJie Yang nb_inputs++; 573b97169daSJie Yang swm_input++; 574b97169daSJie Yang dev_dbg(cmpnt->dev, "input id: %#x, nb_inputs: %d\n", 575b97169daSJie Yang input_loc_id, nb_inputs); 576b97169daSJie Yang 577b97169daSJie Yang if (nb_inputs == SST_CMD_SWM_MAX_INPUTS) { 578b97169daSJie Yang dev_warn(cmpnt->dev, "SET_SWM cmd max inputs reached"); 579b97169daSJie Yang break; 580b97169daSJie Yang } 581b97169daSJie Yang } 582b97169daSJie Yang return nb_inputs; 583b97169daSJie Yang } 584b97169daSJie Yang 585b97169daSJie Yang 586b97169daSJie Yang /** 587b97169daSJie Yang * called with lock held 588b97169daSJie Yang */ 589b97169daSJie Yang static int sst_set_pipe_gain(struct sst_ids *ids, 590b97169daSJie Yang struct sst_data *drv, int mute) 591b97169daSJie Yang { 592b97169daSJie Yang int ret = 0; 593b97169daSJie Yang struct sst_gain_mixer_control *mc; 594b97169daSJie Yang struct sst_gain_value *gv; 595b97169daSJie Yang struct sst_module *gain = NULL; 596b97169daSJie Yang 597b97169daSJie Yang list_for_each_entry(gain, &ids->gain_list, node) { 598b97169daSJie Yang struct snd_kcontrol *kctl = gain->kctl; 599b97169daSJie Yang 600b97169daSJie Yang dev_dbg(&drv->pdev->dev, "control name=%s\n", kctl->id.name); 601b97169daSJie Yang mc = (void *)kctl->private_value; 602b97169daSJie Yang gv = mc->gain_val; 603b97169daSJie Yang 604b97169daSJie Yang ret = sst_send_gain_cmd(drv, gv, mc->task_id, 605b97169daSJie Yang mc->pipe_id | mc->instance_id, mc->module_id, mute); 606b97169daSJie Yang if (ret) 607b97169daSJie Yang return ret; 608b97169daSJie Yang } 609b97169daSJie Yang return ret; 610b97169daSJie Yang } 611b97169daSJie Yang 612b97169daSJie Yang static int sst_swm_mixer_event(struct snd_soc_dapm_widget *w, 613b97169daSJie Yang struct snd_kcontrol *k, int event) 614b97169daSJie Yang { 615b97169daSJie Yang struct sst_cmd_set_swm cmd; 616b97169daSJie Yang struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm); 617b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(cmpnt); 618b97169daSJie Yang struct sst_ids *ids = w->priv; 619b97169daSJie Yang bool set_mixer = false; 620b97169daSJie Yang struct soc_mixer_control *mc; 621b97169daSJie Yang int val = 0; 622b97169daSJie Yang int i = 0; 623b97169daSJie Yang 624b97169daSJie Yang dev_dbg(cmpnt->dev, "widget = %s\n", w->name); 625b97169daSJie Yang /* 626b97169daSJie Yang * Identify which mixer input is on and send the bitmap of the 627b97169daSJie Yang * inputs as an IPC to the DSP. 628b97169daSJie Yang */ 629b97169daSJie Yang for (i = 0; i < w->num_kcontrols; i++) { 630b97169daSJie Yang if (dapm_kcontrol_get_value(w->kcontrols[i])) { 631b97169daSJie Yang mc = (struct soc_mixer_control *)(w->kcontrols[i])->private_value; 632b97169daSJie Yang val |= 1 << mc->shift; 633b97169daSJie Yang } 634b97169daSJie Yang } 635b97169daSJie Yang dev_dbg(cmpnt->dev, "val = %#x\n", val); 636b97169daSJie Yang 637b97169daSJie Yang switch (event) { 638b97169daSJie Yang case SND_SOC_DAPM_PRE_PMU: 639b97169daSJie Yang case SND_SOC_DAPM_POST_PMD: 640b97169daSJie Yang set_mixer = true; 641b97169daSJie Yang break; 642b97169daSJie Yang case SND_SOC_DAPM_POST_REG: 643b97169daSJie Yang if (w->power) 644b97169daSJie Yang set_mixer = true; 645b97169daSJie Yang break; 646b97169daSJie Yang default: 647b97169daSJie Yang set_mixer = false; 648b97169daSJie Yang } 649b97169daSJie Yang 650b97169daSJie Yang if (set_mixer == false) 651b97169daSJie Yang return 0; 652b97169daSJie Yang 653b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event) || 654b97169daSJie Yang event == SND_SOC_DAPM_POST_REG) 655b97169daSJie Yang cmd.switch_state = SST_SWM_ON; 656b97169daSJie Yang else 657b97169daSJie Yang cmd.switch_state = SST_SWM_OFF; 658b97169daSJie Yang 659b97169daSJie Yang SST_FILL_DEFAULT_DESTINATION(cmd.header.dst); 660b97169daSJie Yang /* MMX_SET_SWM == SBA_SET_SWM */ 661b97169daSJie Yang cmd.header.command_id = SBA_SET_SWM; 662b97169daSJie Yang 663b97169daSJie Yang SST_FILL_DESTINATION(2, cmd.output_id, 664b97169daSJie Yang ids->location_id, SST_DEFAULT_MODULE_ID); 665b97169daSJie Yang cmd.nb_inputs = fill_swm_input(cmpnt, &cmd.input[0], val); 666b97169daSJie Yang cmd.header.length = offsetof(struct sst_cmd_set_swm, input) 667b97169daSJie Yang - sizeof(struct sst_dsp_header) 668b97169daSJie Yang + (cmd.nb_inputs * sizeof(cmd.input[0])); 669b97169daSJie Yang 670b97169daSJie Yang return sst_fill_and_send_cmd(drv, SST_IPC_IA_CMD, SST_FLAG_BLOCKED, 671b97169daSJie Yang ids->task_id, 0, &cmd, 672b97169daSJie Yang sizeof(cmd.header) + cmd.header.length); 673b97169daSJie Yang } 674b97169daSJie Yang 675b97169daSJie Yang /* SBA mixers - 16 inputs */ 676b97169daSJie Yang #define SST_SBA_DECLARE_MIX_CONTROLS(kctl_name) \ 677b97169daSJie Yang static const struct snd_kcontrol_new kctl_name[] = { \ 6783d240d73SPierre-Louis Bossart SOC_DAPM_SINGLE("modem_in Switch", SND_SOC_NOPM, SST_IP_MODEM, 1, 0), \ 679b97169daSJie Yang SOC_DAPM_SINGLE("codec_in0 Switch", SND_SOC_NOPM, SST_IP_CODEC0, 1, 0), \ 680b97169daSJie Yang SOC_DAPM_SINGLE("codec_in1 Switch", SND_SOC_NOPM, SST_IP_CODEC1, 1, 0), \ 681b97169daSJie Yang SOC_DAPM_SINGLE("sprot_loop_in Switch", SND_SOC_NOPM, SST_IP_LOOP0, 1, 0), \ 682b97169daSJie Yang SOC_DAPM_SINGLE("media_loop1_in Switch", SND_SOC_NOPM, SST_IP_LOOP1, 1, 0), \ 683b97169daSJie Yang SOC_DAPM_SINGLE("media_loop2_in Switch", SND_SOC_NOPM, SST_IP_LOOP2, 1, 0), \ 684b97169daSJie Yang SOC_DAPM_SINGLE("pcm0_in Switch", SND_SOC_NOPM, SST_IP_PCM0, 1, 0), \ 685b97169daSJie Yang SOC_DAPM_SINGLE("pcm1_in Switch", SND_SOC_NOPM, SST_IP_PCM1, 1, 0), \ 686b97169daSJie Yang } 687b97169daSJie Yang 688b97169daSJie Yang #define SST_SBA_MIXER_GRAPH_MAP(mix_name) \ 6893d240d73SPierre-Louis Bossart { mix_name, "modem_in Switch", "modem_in" }, \ 690b97169daSJie Yang { mix_name, "codec_in0 Switch", "codec_in0" }, \ 691b97169daSJie Yang { mix_name, "codec_in1 Switch", "codec_in1" }, \ 692b97169daSJie Yang { mix_name, "sprot_loop_in Switch", "sprot_loop_in" }, \ 693b97169daSJie Yang { mix_name, "media_loop1_in Switch", "media_loop1_in" }, \ 694b97169daSJie Yang { mix_name, "media_loop2_in Switch", "media_loop2_in" }, \ 695b97169daSJie Yang { mix_name, "pcm0_in Switch", "pcm0_in" }, \ 696b97169daSJie Yang { mix_name, "pcm1_in Switch", "pcm1_in" } 697b97169daSJie Yang 698b97169daSJie Yang #define SST_MMX_DECLARE_MIX_CONTROLS(kctl_name) \ 699b97169daSJie Yang static const struct snd_kcontrol_new kctl_name[] = { \ 700b97169daSJie Yang SOC_DAPM_SINGLE("media0_in Switch", SND_SOC_NOPM, SST_IP_MEDIA0, 1, 0), \ 701b97169daSJie Yang SOC_DAPM_SINGLE("media1_in Switch", SND_SOC_NOPM, SST_IP_MEDIA1, 1, 0), \ 702b97169daSJie Yang SOC_DAPM_SINGLE("media2_in Switch", SND_SOC_NOPM, SST_IP_MEDIA2, 1, 0), \ 703b97169daSJie Yang SOC_DAPM_SINGLE("media3_in Switch", SND_SOC_NOPM, SST_IP_MEDIA3, 1, 0), \ 704b97169daSJie Yang } 705b97169daSJie Yang 706b97169daSJie Yang SST_MMX_DECLARE_MIX_CONTROLS(sst_mix_media0_controls); 707b97169daSJie Yang SST_MMX_DECLARE_MIX_CONTROLS(sst_mix_media1_controls); 708b97169daSJie Yang 709b97169daSJie Yang /* 18 SBA mixers */ 710b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_pcm0_controls); 711b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_pcm1_controls); 712b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_pcm2_controls); 713b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_sprot_l0_controls); 714b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_media_l1_controls); 715b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_media_l2_controls); 716b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_voip_controls); 717b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_codec0_controls); 718b97169daSJie Yang SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_codec1_controls); 7193d240d73SPierre-Louis Bossart SST_SBA_DECLARE_MIX_CONTROLS(sst_mix_modem_controls); 720b97169daSJie Yang 721b97169daSJie Yang /* 722b97169daSJie Yang * sst_handle_vb_timer - Start/Stop the DSP scheduler 723b97169daSJie Yang * 724b97169daSJie Yang * The DSP expects first cmd to be SBA_VB_START, so at first startup send 725b97169daSJie Yang * that. 726b97169daSJie Yang * DSP expects last cmd to be SBA_VB_IDLE, so at last shutdown send that. 727b97169daSJie Yang * 728b97169daSJie Yang * Do refcount internally so that we send command only at first start 729b97169daSJie Yang * and last end. Since SST driver does its own ref count, invoke sst's 730b97169daSJie Yang * power ops always! 731b97169daSJie Yang */ 732b97169daSJie Yang int sst_handle_vb_timer(struct snd_soc_dai *dai, bool enable) 733b97169daSJie Yang { 734b97169daSJie Yang int ret = 0; 735b97169daSJie Yang struct sst_cmd_generic cmd; 736b97169daSJie Yang struct sst_data *drv = snd_soc_dai_get_drvdata(dai); 737b97169daSJie Yang static int timer_usage; 738b97169daSJie Yang 739b97169daSJie Yang if (enable) 740b97169daSJie Yang cmd.header.command_id = SBA_VB_START; 741b97169daSJie Yang else 742b97169daSJie Yang cmd.header.command_id = SBA_IDLE; 743b97169daSJie Yang dev_dbg(dai->dev, "enable=%u, usage=%d\n", enable, timer_usage); 744b97169daSJie Yang 745b97169daSJie Yang SST_FILL_DEFAULT_DESTINATION(cmd.header.dst); 746b97169daSJie Yang cmd.header.length = 0; 747b97169daSJie Yang 748b97169daSJie Yang if (enable) { 749b97169daSJie Yang ret = sst->ops->power(sst->dev, true); 750b97169daSJie Yang if (ret < 0) 751b97169daSJie Yang return ret; 752b97169daSJie Yang } 753b97169daSJie Yang 754b97169daSJie Yang mutex_lock(&drv->lock); 755b97169daSJie Yang if (enable) 756b97169daSJie Yang timer_usage++; 757b97169daSJie Yang else 758b97169daSJie Yang timer_usage--; 759b97169daSJie Yang 760b97169daSJie Yang /* 761b97169daSJie Yang * Send the command only if this call is the first enable or last 762b97169daSJie Yang * disable 763b97169daSJie Yang */ 764b97169daSJie Yang if ((enable && (timer_usage == 1)) || 765b97169daSJie Yang (!enable && (timer_usage == 0))) { 766b97169daSJie Yang ret = sst_fill_and_send_cmd_unlocked(drv, SST_IPC_IA_CMD, 767b97169daSJie Yang SST_FLAG_BLOCKED, SST_TASK_SBA, 0, &cmd, 768b97169daSJie Yang sizeof(cmd.header) + cmd.header.length); 769b97169daSJie Yang if (ret && enable) { 770b97169daSJie Yang timer_usage--; 771b97169daSJie Yang enable = false; 772b97169daSJie Yang } 773b97169daSJie Yang } 774b97169daSJie Yang mutex_unlock(&drv->lock); 775b97169daSJie Yang 776b97169daSJie Yang if (!enable) 777b97169daSJie Yang sst->ops->power(sst->dev, false); 778b97169daSJie Yang return ret; 779b97169daSJie Yang } 780b97169daSJie Yang 78183f125e2SVinod Koul int sst_fill_ssp_slot(struct snd_soc_dai *dai, unsigned int tx_mask, 78283f125e2SVinod Koul unsigned int rx_mask, int slots, int slot_width) 78383f125e2SVinod Koul { 78483f125e2SVinod Koul struct sst_data *ctx = snd_soc_dai_get_drvdata(dai); 78583f125e2SVinod Koul 78683f125e2SVinod Koul ctx->ssp_cmd.nb_slots = slots; 78783f125e2SVinod Koul ctx->ssp_cmd.active_tx_slot_map = tx_mask; 78883f125e2SVinod Koul ctx->ssp_cmd.active_rx_slot_map = rx_mask; 78983f125e2SVinod Koul ctx->ssp_cmd.nb_bits_per_slots = slot_width; 79083f125e2SVinod Koul 79183f125e2SVinod Koul return 0; 79283f125e2SVinod Koul } 79383f125e2SVinod Koul 7940b44e345SVinod Koul static int sst_get_frame_sync_polarity(struct snd_soc_dai *dai, 7950b44e345SVinod Koul unsigned int fmt) 7960b44e345SVinod Koul { 7970b44e345SVinod Koul int format; 7980b44e345SVinod Koul 7990b44e345SVinod Koul format = fmt & SND_SOC_DAIFMT_INV_MASK; 8000b44e345SVinod Koul dev_dbg(dai->dev, "Enter:%s, format=%x\n", __func__, format); 8010b44e345SVinod Koul 8020b44e345SVinod Koul switch (format) { 8030b44e345SVinod Koul case SND_SOC_DAIFMT_NB_NF: 8040b44e345SVinod Koul case SND_SOC_DAIFMT_IB_NF: 8050b44e345SVinod Koul return SSP_FS_ACTIVE_HIGH; 806f12f5c84SPierre-Louis Bossart case SND_SOC_DAIFMT_NB_IF: 807f12f5c84SPierre-Louis Bossart case SND_SOC_DAIFMT_IB_IF: 808f12f5c84SPierre-Louis Bossart return SSP_FS_ACTIVE_LOW; 8090b44e345SVinod Koul default: 8100b44e345SVinod Koul dev_err(dai->dev, "Invalid frame sync polarity %d\n", format); 8110b44e345SVinod Koul } 8120b44e345SVinod Koul 8130b44e345SVinod Koul return -EINVAL; 8140b44e345SVinod Koul } 8150b44e345SVinod Koul 8160b44e345SVinod Koul static int sst_get_ssp_mode(struct snd_soc_dai *dai, unsigned int fmt) 8170b44e345SVinod Koul { 8180b44e345SVinod Koul int format; 8190b44e345SVinod Koul 8200b44e345SVinod Koul format = (fmt & SND_SOC_DAIFMT_MASTER_MASK); 8210b44e345SVinod Koul dev_dbg(dai->dev, "Enter:%s, format=%x\n", __func__, format); 8220b44e345SVinod Koul 8230b44e345SVinod Koul switch (format) { 8240b44e345SVinod Koul case SND_SOC_DAIFMT_CBS_CFS: 8250b44e345SVinod Koul return SSP_MODE_MASTER; 8260b44e345SVinod Koul case SND_SOC_DAIFMT_CBM_CFM: 8270b44e345SVinod Koul return SSP_MODE_SLAVE; 8280b44e345SVinod Koul default: 8290b44e345SVinod Koul dev_err(dai->dev, "Invalid ssp protocol: %d\n", format); 8300b44e345SVinod Koul } 8310b44e345SVinod Koul 8320b44e345SVinod Koul return -EINVAL; 8330b44e345SVinod Koul } 8340b44e345SVinod Koul 8350b44e345SVinod Koul 8360b44e345SVinod Koul int sst_fill_ssp_config(struct snd_soc_dai *dai, unsigned int fmt) 8370b44e345SVinod Koul { 8380b44e345SVinod Koul unsigned int mode; 8390b44e345SVinod Koul int fs_polarity; 8400b44e345SVinod Koul struct sst_data *ctx = snd_soc_dai_get_drvdata(dai); 8410b44e345SVinod Koul 8420b44e345SVinod Koul mode = fmt & SND_SOC_DAIFMT_FORMAT_MASK; 8430b44e345SVinod Koul 8440b44e345SVinod Koul switch (mode) { 8450b44e345SVinod Koul case SND_SOC_DAIFMT_DSP_B: 8460b44e345SVinod Koul ctx->ssp_cmd.ssp_protocol = SSP_MODE_PCM; 8470b44e345SVinod Koul ctx->ssp_cmd.mode = sst_get_ssp_mode(dai, fmt) | (SSP_PCM_MODE_NETWORK << 1); 8480b44e345SVinod Koul ctx->ssp_cmd.start_delay = 0; 8490b44e345SVinod Koul ctx->ssp_cmd.data_polarity = 1; 8500b44e345SVinod Koul ctx->ssp_cmd.frame_sync_width = 1; 8510b44e345SVinod Koul break; 8520b44e345SVinod Koul 8530b44e345SVinod Koul case SND_SOC_DAIFMT_DSP_A: 8540b44e345SVinod Koul ctx->ssp_cmd.ssp_protocol = SSP_MODE_PCM; 8550b44e345SVinod Koul ctx->ssp_cmd.mode = sst_get_ssp_mode(dai, fmt) | (SSP_PCM_MODE_NETWORK << 1); 8560b44e345SVinod Koul ctx->ssp_cmd.start_delay = 1; 8570b44e345SVinod Koul ctx->ssp_cmd.data_polarity = 1; 8580b44e345SVinod Koul ctx->ssp_cmd.frame_sync_width = 1; 8590b44e345SVinod Koul break; 8600b44e345SVinod Koul 8610b44e345SVinod Koul case SND_SOC_DAIFMT_I2S: 8620b44e345SVinod Koul ctx->ssp_cmd.ssp_protocol = SSP_MODE_I2S; 8630b44e345SVinod Koul ctx->ssp_cmd.mode = sst_get_ssp_mode(dai, fmt) | (SSP_PCM_MODE_NORMAL << 1); 8640b44e345SVinod Koul ctx->ssp_cmd.start_delay = 1; 8650b44e345SVinod Koul ctx->ssp_cmd.data_polarity = 0; 8660b44e345SVinod Koul ctx->ssp_cmd.frame_sync_width = ctx->ssp_cmd.nb_bits_per_slots; 8670b44e345SVinod Koul break; 8680b44e345SVinod Koul 8690b44e345SVinod Koul case SND_SOC_DAIFMT_LEFT_J: 8700b44e345SVinod Koul ctx->ssp_cmd.ssp_protocol = SSP_MODE_I2S; 8710b44e345SVinod Koul ctx->ssp_cmd.mode = sst_get_ssp_mode(dai, fmt) | (SSP_PCM_MODE_NORMAL << 1); 8720b44e345SVinod Koul ctx->ssp_cmd.start_delay = 0; 8730b44e345SVinod Koul ctx->ssp_cmd.data_polarity = 0; 8740b44e345SVinod Koul ctx->ssp_cmd.frame_sync_width = ctx->ssp_cmd.nb_bits_per_slots; 8750b44e345SVinod Koul break; 8760b44e345SVinod Koul 8770b44e345SVinod Koul default: 8780b44e345SVinod Koul dev_dbg(dai->dev, "using default ssp configs\n"); 8790b44e345SVinod Koul } 8800b44e345SVinod Koul 8810b44e345SVinod Koul fs_polarity = sst_get_frame_sync_polarity(dai, fmt); 8820b44e345SVinod Koul if (fs_polarity < 0) 8830b44e345SVinod Koul return fs_polarity; 8840b44e345SVinod Koul 8850b44e345SVinod Koul ctx->ssp_cmd.frame_sync_polarity = fs_polarity; 8860b44e345SVinod Koul 8870b44e345SVinod Koul return 0; 8880b44e345SVinod Koul } 8890b44e345SVinod Koul 890b97169daSJie Yang /** 891b97169daSJie Yang * sst_ssp_config - contains SSP configuration for media UC 8920b44e345SVinod Koul * this can be overwritten by set_dai_xxx APIs 893b97169daSJie Yang */ 894b97169daSJie Yang static const struct sst_ssp_config sst_ssp_configs = { 895b97169daSJie Yang .ssp_id = SSP_CODEC, 896b97169daSJie Yang .bits_per_slot = 24, 897b97169daSJie Yang .slots = 4, 898b97169daSJie Yang .ssp_mode = SSP_MODE_MASTER, 899b97169daSJie Yang .pcm_mode = SSP_PCM_MODE_NETWORK, 900b97169daSJie Yang .duplex = SSP_DUPLEX, 901b97169daSJie Yang .ssp_protocol = SSP_MODE_PCM, 902b97169daSJie Yang .fs_width = 1, 903b97169daSJie Yang .fs_frequency = SSP_FS_48_KHZ, 904b97169daSJie Yang .active_slot_map = 0xF, 905b97169daSJie Yang .start_delay = 0, 9061f2d86f1SVinod Koul .frame_sync_polarity = SSP_FS_ACTIVE_HIGH, 9071f2d86f1SVinod Koul .data_polarity = 1, 908b97169daSJie Yang }; 909b97169daSJie Yang 9105749d70eSVinod Koul void sst_fill_ssp_defaults(struct snd_soc_dai *dai) 9115749d70eSVinod Koul { 9125749d70eSVinod Koul const struct sst_ssp_config *config; 9135749d70eSVinod Koul struct sst_data *ctx = snd_soc_dai_get_drvdata(dai); 9145749d70eSVinod Koul 9155749d70eSVinod Koul config = &sst_ssp_configs; 9165749d70eSVinod Koul 9175749d70eSVinod Koul ctx->ssp_cmd.selection = config->ssp_id; 9185749d70eSVinod Koul ctx->ssp_cmd.nb_bits_per_slots = config->bits_per_slot; 9195749d70eSVinod Koul ctx->ssp_cmd.nb_slots = config->slots; 9205749d70eSVinod Koul ctx->ssp_cmd.mode = config->ssp_mode | (config->pcm_mode << 1); 9215749d70eSVinod Koul ctx->ssp_cmd.duplex = config->duplex; 9225749d70eSVinod Koul ctx->ssp_cmd.active_tx_slot_map = config->active_slot_map; 9235749d70eSVinod Koul ctx->ssp_cmd.active_rx_slot_map = config->active_slot_map; 9245749d70eSVinod Koul ctx->ssp_cmd.frame_sync_frequency = config->fs_frequency; 9255749d70eSVinod Koul ctx->ssp_cmd.frame_sync_polarity = config->frame_sync_polarity; 9265749d70eSVinod Koul ctx->ssp_cmd.data_polarity = config->data_polarity; 9275749d70eSVinod Koul ctx->ssp_cmd.frame_sync_width = config->fs_width; 9285749d70eSVinod Koul ctx->ssp_cmd.ssp_protocol = config->ssp_protocol; 9295749d70eSVinod Koul ctx->ssp_cmd.start_delay = config->start_delay; 9305749d70eSVinod Koul ctx->ssp_cmd.reserved1 = ctx->ssp_cmd.reserved2 = 0xFF; 9315749d70eSVinod Koul } 9325749d70eSVinod Koul 933b97169daSJie Yang int send_ssp_cmd(struct snd_soc_dai *dai, const char *id, bool enable) 934b97169daSJie Yang { 935b97169daSJie Yang struct sst_data *drv = snd_soc_dai_get_drvdata(dai); 936a3f10de1SPierre-Louis Bossart int ssp_id; 937b97169daSJie Yang 93891197a92STakashi Iwai dev_dbg(dai->dev, "Enter: enable=%d port_name=%s\n", enable, id); 939b97169daSJie Yang 940a3f10de1SPierre-Louis Bossart if (strcmp(id, "ssp0-port") == 0) 941a3f10de1SPierre-Louis Bossart ssp_id = SSP_MODEM; 942a3f10de1SPierre-Louis Bossart else if (strcmp(id, "ssp2-port") == 0) 943a3f10de1SPierre-Louis Bossart ssp_id = SSP_CODEC; 944a3f10de1SPierre-Louis Bossart else { 945a3f10de1SPierre-Louis Bossart dev_dbg(dai->dev, "port %s is not supported\n", id); 946a3f10de1SPierre-Louis Bossart return -1; 947a3f10de1SPierre-Louis Bossart } 948a3f10de1SPierre-Louis Bossart 9495749d70eSVinod Koul SST_FILL_DEFAULT_DESTINATION(drv->ssp_cmd.header.dst); 9505749d70eSVinod Koul drv->ssp_cmd.header.command_id = SBA_HW_SET_SSP; 9515749d70eSVinod Koul drv->ssp_cmd.header.length = sizeof(struct sst_cmd_sba_hw_set_ssp) 952b97169daSJie Yang - sizeof(struct sst_dsp_header); 953b97169daSJie Yang 954a3f10de1SPierre-Louis Bossart drv->ssp_cmd.selection = ssp_id; 955a3f10de1SPierre-Louis Bossart dev_dbg(dai->dev, "ssp_id: %u\n", ssp_id); 956b97169daSJie Yang 957b97169daSJie Yang if (enable) 9585749d70eSVinod Koul drv->ssp_cmd.switch_state = SST_SWITCH_ON; 959b97169daSJie Yang else 9605749d70eSVinod Koul drv->ssp_cmd.switch_state = SST_SWITCH_OFF; 961b97169daSJie Yang 962b97169daSJie Yang return sst_fill_and_send_cmd(drv, SST_IPC_IA_CMD, SST_FLAG_BLOCKED, 9635749d70eSVinod Koul SST_TASK_SBA, 0, &drv->ssp_cmd, 9645749d70eSVinod Koul sizeof(drv->ssp_cmd.header) + drv->ssp_cmd.header.length); 965b97169daSJie Yang } 966b97169daSJie Yang 967b97169daSJie Yang static int sst_set_be_modules(struct snd_soc_dapm_widget *w, 968b97169daSJie Yang struct snd_kcontrol *k, int event) 969b97169daSJie Yang { 970b97169daSJie Yang int ret = 0; 971b97169daSJie Yang struct snd_soc_component *c = snd_soc_dapm_to_component(w->dapm); 972b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(c); 973b97169daSJie Yang 974b97169daSJie Yang dev_dbg(c->dev, "Enter: widget=%s\n", w->name); 975b97169daSJie Yang 976b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event)) { 977b97169daSJie Yang ret = sst_send_slot_map(drv); 978b97169daSJie Yang if (ret) 979b97169daSJie Yang return ret; 980b97169daSJie Yang ret = sst_send_pipe_module_params(w, k); 981b97169daSJie Yang } 982b97169daSJie Yang return ret; 983b97169daSJie Yang } 984b97169daSJie Yang 985b97169daSJie Yang static int sst_set_media_path(struct snd_soc_dapm_widget *w, 986b97169daSJie Yang struct snd_kcontrol *k, int event) 987b97169daSJie Yang { 988b97169daSJie Yang int ret = 0; 989b97169daSJie Yang struct sst_cmd_set_media_path cmd; 990b97169daSJie Yang struct snd_soc_component *c = snd_soc_dapm_to_component(w->dapm); 991b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(c); 992b97169daSJie Yang struct sst_ids *ids = w->priv; 993b97169daSJie Yang 994b97169daSJie Yang dev_dbg(c->dev, "widget=%s\n", w->name); 995b97169daSJie Yang dev_dbg(c->dev, "task=%u, location=%#x\n", 996b97169daSJie Yang ids->task_id, ids->location_id); 997b97169daSJie Yang 998b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event)) 999b97169daSJie Yang cmd.switch_state = SST_PATH_ON; 1000b97169daSJie Yang else 1001b97169daSJie Yang cmd.switch_state = SST_PATH_OFF; 1002b97169daSJie Yang 1003b97169daSJie Yang SST_FILL_DESTINATION(2, cmd.header.dst, 1004b97169daSJie Yang ids->location_id, SST_DEFAULT_MODULE_ID); 1005b97169daSJie Yang 1006b97169daSJie Yang /* MMX_SET_MEDIA_PATH == SBA_SET_MEDIA_PATH */ 1007b97169daSJie Yang cmd.header.command_id = MMX_SET_MEDIA_PATH; 1008b97169daSJie Yang cmd.header.length = sizeof(struct sst_cmd_set_media_path) 1009b97169daSJie Yang - sizeof(struct sst_dsp_header); 1010b97169daSJie Yang 1011b97169daSJie Yang ret = sst_fill_and_send_cmd(drv, SST_IPC_IA_CMD, SST_FLAG_BLOCKED, 1012b97169daSJie Yang ids->task_id, 0, &cmd, 1013b97169daSJie Yang sizeof(cmd.header) + cmd.header.length); 1014b97169daSJie Yang if (ret) 1015b97169daSJie Yang return ret; 1016b97169daSJie Yang 1017b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event)) 1018b97169daSJie Yang ret = sst_send_pipe_module_params(w, k); 1019b97169daSJie Yang return ret; 1020b97169daSJie Yang } 1021b97169daSJie Yang 1022b97169daSJie Yang static int sst_set_media_loop(struct snd_soc_dapm_widget *w, 1023b97169daSJie Yang struct snd_kcontrol *k, int event) 1024b97169daSJie Yang { 1025b97169daSJie Yang int ret = 0; 1026b97169daSJie Yang struct sst_cmd_sba_set_media_loop_map cmd; 1027b97169daSJie Yang struct snd_soc_component *c = snd_soc_dapm_to_component(w->dapm); 1028b97169daSJie Yang struct sst_data *drv = snd_soc_component_get_drvdata(c); 1029b97169daSJie Yang struct sst_ids *ids = w->priv; 1030b97169daSJie Yang 1031b97169daSJie Yang dev_dbg(c->dev, "Enter:widget=%s\n", w->name); 1032b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event)) 1033b97169daSJie Yang cmd.switch_state = SST_SWITCH_ON; 1034b97169daSJie Yang else 1035b97169daSJie Yang cmd.switch_state = SST_SWITCH_OFF; 1036b97169daSJie Yang 1037b97169daSJie Yang SST_FILL_DESTINATION(2, cmd.header.dst, 1038b97169daSJie Yang ids->location_id, SST_DEFAULT_MODULE_ID); 1039b97169daSJie Yang 1040b97169daSJie Yang cmd.header.command_id = SBA_SET_MEDIA_LOOP_MAP; 1041b97169daSJie Yang cmd.header.length = sizeof(struct sst_cmd_sba_set_media_loop_map) 1042b97169daSJie Yang - sizeof(struct sst_dsp_header); 1043b97169daSJie Yang cmd.param.part.cfg.rate = 2; /* 48khz */ 1044b97169daSJie Yang 1045b97169daSJie Yang cmd.param.part.cfg.format = ids->format; /* stereo/Mono */ 1046b97169daSJie Yang cmd.param.part.cfg.s_length = 1; /* 24bit left justified */ 1047b97169daSJie Yang cmd.map = 0; /* Algo sequence: Gain - DRP - FIR - IIR */ 1048b97169daSJie Yang 1049b97169daSJie Yang ret = sst_fill_and_send_cmd(drv, SST_IPC_IA_CMD, SST_FLAG_BLOCKED, 1050b97169daSJie Yang SST_TASK_SBA, 0, &cmd, 1051b97169daSJie Yang sizeof(cmd.header) + cmd.header.length); 1052b97169daSJie Yang if (ret) 1053b97169daSJie Yang return ret; 1054b97169daSJie Yang 1055b97169daSJie Yang if (SND_SOC_DAPM_EVENT_ON(event)) 1056b97169daSJie Yang ret = sst_send_pipe_module_params(w, k); 1057b97169daSJie Yang return ret; 1058b97169daSJie Yang } 1059b97169daSJie Yang 1060b97169daSJie Yang static const struct snd_soc_dapm_widget sst_dapm_widgets[] = { 10613d240d73SPierre-Louis Bossart SST_AIF_IN("modem_in", sst_set_be_modules), 1062b97169daSJie Yang SST_AIF_IN("codec_in0", sst_set_be_modules), 1063b97169daSJie Yang SST_AIF_IN("codec_in1", sst_set_be_modules), 10643d240d73SPierre-Louis Bossart SST_AIF_OUT("modem_out", sst_set_be_modules), 1065b97169daSJie Yang SST_AIF_OUT("codec_out0", sst_set_be_modules), 1066b97169daSJie Yang SST_AIF_OUT("codec_out1", sst_set_be_modules), 1067b97169daSJie Yang 1068b97169daSJie Yang /* Media Paths */ 1069b97169daSJie Yang /* MediaX IN paths are set via ALLOC, so no SET_MEDIA_PATH command */ 1070b97169daSJie Yang SST_PATH_INPUT("media0_in", SST_TASK_MMX, SST_SWM_IN_MEDIA0, sst_generic_modules_event), 1071b97169daSJie Yang SST_PATH_INPUT("media1_in", SST_TASK_MMX, SST_SWM_IN_MEDIA1, NULL), 1072b97169daSJie Yang SST_PATH_INPUT("media2_in", SST_TASK_MMX, SST_SWM_IN_MEDIA2, sst_set_media_path), 1073b97169daSJie Yang SST_PATH_INPUT("media3_in", SST_TASK_MMX, SST_SWM_IN_MEDIA3, NULL), 1074b97169daSJie Yang SST_PATH_OUTPUT("media0_out", SST_TASK_MMX, SST_SWM_OUT_MEDIA0, sst_set_media_path), 1075b97169daSJie Yang SST_PATH_OUTPUT("media1_out", SST_TASK_MMX, SST_SWM_OUT_MEDIA1, sst_set_media_path), 1076b97169daSJie Yang 1077b97169daSJie Yang /* SBA PCM Paths */ 1078b97169daSJie Yang SST_PATH_INPUT("pcm0_in", SST_TASK_SBA, SST_SWM_IN_PCM0, sst_set_media_path), 1079b97169daSJie Yang SST_PATH_INPUT("pcm1_in", SST_TASK_SBA, SST_SWM_IN_PCM1, sst_set_media_path), 1080b97169daSJie Yang SST_PATH_OUTPUT("pcm0_out", SST_TASK_SBA, SST_SWM_OUT_PCM0, sst_set_media_path), 1081b97169daSJie Yang SST_PATH_OUTPUT("pcm1_out", SST_TASK_SBA, SST_SWM_OUT_PCM1, sst_set_media_path), 1082b97169daSJie Yang SST_PATH_OUTPUT("pcm2_out", SST_TASK_SBA, SST_SWM_OUT_PCM2, sst_set_media_path), 1083b97169daSJie Yang 1084b97169daSJie Yang /* SBA Loops */ 1085b97169daSJie Yang SST_PATH_INPUT("sprot_loop_in", SST_TASK_SBA, SST_SWM_IN_SPROT_LOOP, NULL), 1086b97169daSJie Yang SST_PATH_INPUT("media_loop1_in", SST_TASK_SBA, SST_SWM_IN_MEDIA_LOOP1, NULL), 1087b97169daSJie Yang SST_PATH_INPUT("media_loop2_in", SST_TASK_SBA, SST_SWM_IN_MEDIA_LOOP2, NULL), 10885b101ab4SSebastien Guiriec SST_PATH_MEDIA_LOOP_OUTPUT("sprot_loop_out", SST_TASK_SBA, SST_SWM_OUT_SPROT_LOOP, SST_FMT_STEREO, sst_set_media_loop), 10895b101ab4SSebastien Guiriec SST_PATH_MEDIA_LOOP_OUTPUT("media_loop1_out", SST_TASK_SBA, SST_SWM_OUT_MEDIA_LOOP1, SST_FMT_STEREO, sst_set_media_loop), 1090b97169daSJie Yang SST_PATH_MEDIA_LOOP_OUTPUT("media_loop2_out", SST_TASK_SBA, SST_SWM_OUT_MEDIA_LOOP2, SST_FMT_STEREO, sst_set_media_loop), 1091b97169daSJie Yang 1092b97169daSJie Yang /* Media Mixers */ 1093b97169daSJie Yang SST_SWM_MIXER("media0_out mix 0", SND_SOC_NOPM, SST_TASK_MMX, SST_SWM_OUT_MEDIA0, 1094b97169daSJie Yang sst_mix_media0_controls, sst_swm_mixer_event), 1095b97169daSJie Yang SST_SWM_MIXER("media1_out mix 0", SND_SOC_NOPM, SST_TASK_MMX, SST_SWM_OUT_MEDIA1, 1096b97169daSJie Yang sst_mix_media1_controls, sst_swm_mixer_event), 1097b97169daSJie Yang 1098b97169daSJie Yang /* SBA PCM mixers */ 1099b97169daSJie Yang SST_SWM_MIXER("pcm0_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_PCM0, 1100b97169daSJie Yang sst_mix_pcm0_controls, sst_swm_mixer_event), 1101b97169daSJie Yang SST_SWM_MIXER("pcm1_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_PCM1, 1102b97169daSJie Yang sst_mix_pcm1_controls, sst_swm_mixer_event), 1103b97169daSJie Yang SST_SWM_MIXER("pcm2_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_PCM2, 1104b97169daSJie Yang sst_mix_pcm2_controls, sst_swm_mixer_event), 1105b97169daSJie Yang 1106b97169daSJie Yang /* SBA Loop mixers */ 1107b97169daSJie Yang SST_SWM_MIXER("sprot_loop_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_SPROT_LOOP, 1108b97169daSJie Yang sst_mix_sprot_l0_controls, sst_swm_mixer_event), 1109b97169daSJie Yang SST_SWM_MIXER("media_loop1_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_MEDIA_LOOP1, 1110b97169daSJie Yang sst_mix_media_l1_controls, sst_swm_mixer_event), 1111b97169daSJie Yang SST_SWM_MIXER("media_loop2_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_MEDIA_LOOP2, 1112b97169daSJie Yang sst_mix_media_l2_controls, sst_swm_mixer_event), 1113b97169daSJie Yang 1114b97169daSJie Yang /* SBA Backend mixers */ 1115b97169daSJie Yang SST_SWM_MIXER("codec_out0 mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_CODEC0, 1116b97169daSJie Yang sst_mix_codec0_controls, sst_swm_mixer_event), 1117b97169daSJie Yang SST_SWM_MIXER("codec_out1 mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_CODEC1, 1118b97169daSJie Yang sst_mix_codec1_controls, sst_swm_mixer_event), 11193d240d73SPierre-Louis Bossart SST_SWM_MIXER("modem_out mix 0", SND_SOC_NOPM, SST_TASK_SBA, SST_SWM_OUT_MODEM, 11203d240d73SPierre-Louis Bossart sst_mix_modem_controls, sst_swm_mixer_event), 11213d240d73SPierre-Louis Bossart 1122b97169daSJie Yang }; 1123b97169daSJie Yang 1124b97169daSJie Yang static const struct snd_soc_dapm_route intercon[] = { 1125b97169daSJie Yang {"media0_in", NULL, "Compress Playback"}, 1126b97169daSJie Yang {"media1_in", NULL, "Headset Playback"}, 1127b97169daSJie Yang {"media2_in", NULL, "pcm0_out"}, 11288788f839SPierre-Louis Bossart {"media3_in", NULL, "Deepbuffer Playback"}, 1129b97169daSJie Yang 1130b97169daSJie Yang {"media0_out mix 0", "media0_in Switch", "media0_in"}, 1131b97169daSJie Yang {"media0_out mix 0", "media1_in Switch", "media1_in"}, 1132b97169daSJie Yang {"media0_out mix 0", "media2_in Switch", "media2_in"}, 1133b97169daSJie Yang {"media0_out mix 0", "media3_in Switch", "media3_in"}, 1134b97169daSJie Yang {"media1_out mix 0", "media0_in Switch", "media0_in"}, 1135b97169daSJie Yang {"media1_out mix 0", "media1_in Switch", "media1_in"}, 1136b97169daSJie Yang {"media1_out mix 0", "media2_in Switch", "media2_in"}, 1137b97169daSJie Yang {"media1_out mix 0", "media3_in Switch", "media3_in"}, 1138b97169daSJie Yang 1139b97169daSJie Yang {"media0_out", NULL, "media0_out mix 0"}, 1140b97169daSJie Yang {"media1_out", NULL, "media1_out mix 0"}, 1141b97169daSJie Yang {"pcm0_in", NULL, "media0_out"}, 1142b97169daSJie Yang {"pcm1_in", NULL, "media1_out"}, 1143b97169daSJie Yang 1144b97169daSJie Yang {"Headset Capture", NULL, "pcm1_out"}, 1145b97169daSJie Yang {"Headset Capture", NULL, "pcm2_out"}, 1146b97169daSJie Yang {"pcm0_out", NULL, "pcm0_out mix 0"}, 1147b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("pcm0_out mix 0"), 1148b97169daSJie Yang {"pcm1_out", NULL, "pcm1_out mix 0"}, 1149b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("pcm1_out mix 0"), 1150b97169daSJie Yang {"pcm2_out", NULL, "pcm2_out mix 0"}, 1151b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("pcm2_out mix 0"), 1152b97169daSJie Yang 1153b97169daSJie Yang {"media_loop1_in", NULL, "media_loop1_out"}, 1154b97169daSJie Yang {"media_loop1_out", NULL, "media_loop1_out mix 0"}, 1155b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("media_loop1_out mix 0"), 1156b97169daSJie Yang {"media_loop2_in", NULL, "media_loop2_out"}, 1157b97169daSJie Yang {"media_loop2_out", NULL, "media_loop2_out mix 0"}, 1158b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("media_loop2_out mix 0"), 1159b97169daSJie Yang {"sprot_loop_in", NULL, "sprot_loop_out"}, 1160b97169daSJie Yang {"sprot_loop_out", NULL, "sprot_loop_out mix 0"}, 1161b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("sprot_loop_out mix 0"), 1162b97169daSJie Yang 1163b97169daSJie Yang {"codec_out0", NULL, "codec_out0 mix 0"}, 1164b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("codec_out0 mix 0"), 1165b97169daSJie Yang {"codec_out1", NULL, "codec_out1 mix 0"}, 1166b97169daSJie Yang SST_SBA_MIXER_GRAPH_MAP("codec_out1 mix 0"), 11673d240d73SPierre-Louis Bossart {"modem_out", NULL, "modem_out mix 0"}, 11683d240d73SPierre-Louis Bossart SST_SBA_MIXER_GRAPH_MAP("modem_out mix 0"), 11693d240d73SPierre-Louis Bossart 1170b97169daSJie Yang 1171b97169daSJie Yang }; 1172b97169daSJie Yang static const char * const slot_names[] = { 1173b97169daSJie Yang "none", 1174b97169daSJie Yang "slot 0", "slot 1", "slot 2", "slot 3", 1175b97169daSJie Yang "slot 4", "slot 5", "slot 6", "slot 7", /* not supported by FW */ 1176b97169daSJie Yang }; 1177b97169daSJie Yang 1178b97169daSJie Yang static const char * const channel_names[] = { 1179b97169daSJie Yang "none", 1180b97169daSJie Yang "codec_out0_0", "codec_out0_1", "codec_out1_0", "codec_out1_1", 1181b97169daSJie Yang "codec_out2_0", "codec_out2_1", "codec_out3_0", "codec_out3_1", /* not supported by FW */ 1182b97169daSJie Yang }; 1183b97169daSJie Yang 1184b97169daSJie Yang #define SST_INTERLEAVER(xpname, slot_name, slotno) \ 1185b97169daSJie Yang SST_SSP_SLOT_CTL(xpname, "tx interleaver", slot_name, slotno, true, \ 1186b97169daSJie Yang channel_names, sst_slot_get, sst_slot_put) 1187b97169daSJie Yang 1188b97169daSJie Yang #define SST_DEINTERLEAVER(xpname, channel_name, channel_no) \ 1189b97169daSJie Yang SST_SSP_SLOT_CTL(xpname, "rx deinterleaver", channel_name, channel_no, false, \ 1190b97169daSJie Yang slot_names, sst_slot_get, sst_slot_put) 1191b97169daSJie Yang 1192b97169daSJie Yang static const struct snd_kcontrol_new sst_slot_controls[] = { 1193b97169daSJie Yang SST_INTERLEAVER("codec_out", "slot 0", 0), 1194b97169daSJie Yang SST_INTERLEAVER("codec_out", "slot 1", 1), 1195b97169daSJie Yang SST_INTERLEAVER("codec_out", "slot 2", 2), 1196b97169daSJie Yang SST_INTERLEAVER("codec_out", "slot 3", 3), 1197b97169daSJie Yang SST_DEINTERLEAVER("codec_in", "codec_in0_0", 0), 1198b97169daSJie Yang SST_DEINTERLEAVER("codec_in", "codec_in0_1", 1), 1199b97169daSJie Yang SST_DEINTERLEAVER("codec_in", "codec_in1_0", 2), 1200b97169daSJie Yang SST_DEINTERLEAVER("codec_in", "codec_in1_1", 3), 1201b97169daSJie Yang }; 1202b97169daSJie Yang 1203b97169daSJie Yang /* Gain helper with min/max set */ 1204b97169daSJie Yang #define SST_GAIN(name, path_id, task_id, instance, gain_var) \ 1205b97169daSJie Yang SST_GAIN_KCONTROLS(name, "Gain", SST_GAIN_MIN_VALUE, SST_GAIN_MAX_VALUE, \ 1206b97169daSJie Yang SST_GAIN_TC_MIN, SST_GAIN_TC_MAX, \ 1207b97169daSJie Yang sst_gain_get, sst_gain_put, \ 1208b97169daSJie Yang SST_MODULE_ID_GAIN_CELL, path_id, instance, task_id, \ 1209b97169daSJie Yang sst_gain_tlv_common, gain_var) 1210b97169daSJie Yang 1211b97169daSJie Yang #define SST_VOLUME(name, path_id, task_id, instance, gain_var) \ 1212b97169daSJie Yang SST_GAIN_KCONTROLS(name, "Volume", SST_GAIN_MIN_VALUE, SST_GAIN_MAX_VALUE, \ 1213b97169daSJie Yang SST_GAIN_TC_MIN, SST_GAIN_TC_MAX, \ 1214b97169daSJie Yang sst_gain_get, sst_gain_put, \ 1215b97169daSJie Yang SST_MODULE_ID_VOLUME, path_id, instance, task_id, \ 1216b97169daSJie Yang sst_gain_tlv_common, gain_var) 1217b97169daSJie Yang 1218b97169daSJie Yang static struct sst_gain_value sst_gains[]; 1219b97169daSJie Yang 1220b97169daSJie Yang static const struct snd_kcontrol_new sst_gain_controls[] = { 1221b97169daSJie Yang SST_GAIN("media0_in", SST_PATH_INDEX_MEDIA0_IN, SST_TASK_MMX, 0, &sst_gains[0]), 1222b97169daSJie Yang SST_GAIN("media1_in", SST_PATH_INDEX_MEDIA1_IN, SST_TASK_MMX, 0, &sst_gains[1]), 1223b97169daSJie Yang SST_GAIN("media2_in", SST_PATH_INDEX_MEDIA2_IN, SST_TASK_MMX, 0, &sst_gains[2]), 1224b97169daSJie Yang SST_GAIN("media3_in", SST_PATH_INDEX_MEDIA3_IN, SST_TASK_MMX, 0, &sst_gains[3]), 1225b97169daSJie Yang 1226b97169daSJie Yang SST_GAIN("pcm0_in", SST_PATH_INDEX_PCM0_IN, SST_TASK_SBA, 0, &sst_gains[4]), 1227b97169daSJie Yang SST_GAIN("pcm1_in", SST_PATH_INDEX_PCM1_IN, SST_TASK_SBA, 0, &sst_gains[5]), 1228b97169daSJie Yang SST_GAIN("pcm1_out", SST_PATH_INDEX_PCM1_OUT, SST_TASK_SBA, 0, &sst_gains[6]), 1229b97169daSJie Yang SST_GAIN("pcm2_out", SST_PATH_INDEX_PCM2_OUT, SST_TASK_SBA, 0, &sst_gains[7]), 1230b97169daSJie Yang 1231b97169daSJie Yang SST_GAIN("codec_in0", SST_PATH_INDEX_CODEC_IN0, SST_TASK_SBA, 0, &sst_gains[8]), 1232b97169daSJie Yang SST_GAIN("codec_in1", SST_PATH_INDEX_CODEC_IN1, SST_TASK_SBA, 0, &sst_gains[9]), 1233b97169daSJie Yang SST_GAIN("codec_out0", SST_PATH_INDEX_CODEC_OUT0, SST_TASK_SBA, 0, &sst_gains[10]), 1234b97169daSJie Yang SST_GAIN("codec_out1", SST_PATH_INDEX_CODEC_OUT1, SST_TASK_SBA, 0, &sst_gains[11]), 1235b97169daSJie Yang SST_GAIN("media_loop1_out", SST_PATH_INDEX_MEDIA_LOOP1_OUT, SST_TASK_SBA, 0, &sst_gains[12]), 1236b97169daSJie Yang SST_GAIN("media_loop2_out", SST_PATH_INDEX_MEDIA_LOOP2_OUT, SST_TASK_SBA, 0, &sst_gains[13]), 1237b97169daSJie Yang SST_GAIN("sprot_loop_out", SST_PATH_INDEX_SPROT_LOOP_OUT, SST_TASK_SBA, 0, &sst_gains[14]), 1238b97169daSJie Yang SST_VOLUME("media0_in", SST_PATH_INDEX_MEDIA0_IN, SST_TASK_MMX, 0, &sst_gains[15]), 12393d240d73SPierre-Louis Bossart SST_GAIN("modem_in", SST_PATH_INDEX_MODEM_IN, SST_TASK_SBA, 0, &sst_gains[16]), 12403d240d73SPierre-Louis Bossart SST_GAIN("modem_out", SST_PATH_INDEX_MODEM_OUT, SST_TASK_SBA, 0, &sst_gains[17]), 12413d240d73SPierre-Louis Bossart 1242b97169daSJie Yang }; 1243b97169daSJie Yang 1244b97169daSJie Yang #define SST_GAIN_NUM_CONTROLS 3 1245b97169daSJie Yang /* the SST_GAIN macro above will create three alsa controls for each 1246b97169daSJie Yang * instance invoked, gain, mute and ramp duration, which use the same gain 1247b97169daSJie Yang * cell sst_gain to keep track of data 1248b97169daSJie Yang * To calculate number of gain cell instances we need to device by 3 in 1249b97169daSJie Yang * below caulcation for gain cell memory. 1250b97169daSJie Yang * This gets rid of static number and issues while adding new controls 1251b97169daSJie Yang */ 1252b97169daSJie Yang static struct sst_gain_value sst_gains[ARRAY_SIZE(sst_gain_controls)/SST_GAIN_NUM_CONTROLS]; 1253b97169daSJie Yang 1254b97169daSJie Yang static const struct snd_kcontrol_new sst_algo_controls[] = { 1255b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("media_loop1_out", "fir", 272, SST_MODULE_ID_FIR_24, 1256b97169daSJie Yang SST_PATH_INDEX_MEDIA_LOOP1_OUT, 0, SST_TASK_SBA, SBA_VB_SET_FIR), 1257b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("media_loop1_out", "iir", 300, SST_MODULE_ID_IIR_24, 1258b97169daSJie Yang SST_PATH_INDEX_MEDIA_LOOP1_OUT, 0, SST_TASK_SBA, SBA_VB_SET_IIR), 1259b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("media_loop1_out", "mdrp", 286, SST_MODULE_ID_MDRP, 1260b97169daSJie Yang SST_PATH_INDEX_MEDIA_LOOP1_OUT, 0, SST_TASK_SBA, SBA_SET_MDRP), 1261b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("media_loop2_out", "fir", 272, SST_MODULE_ID_FIR_24, 1262b97169daSJie Yang SST_PATH_INDEX_MEDIA_LOOP2_OUT, 0, SST_TASK_SBA, SBA_VB_SET_FIR), 1263b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("media_loop2_out", "iir", 300, SST_MODULE_ID_IIR_24, 1264b97169daSJie Yang SST_PATH_INDEX_MEDIA_LOOP2_OUT, 0, SST_TASK_SBA, SBA_VB_SET_IIR), 1265b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("media_loop2_out", "mdrp", 286, SST_MODULE_ID_MDRP, 1266b97169daSJie Yang SST_PATH_INDEX_MEDIA_LOOP2_OUT, 0, SST_TASK_SBA, SBA_SET_MDRP), 1267b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("sprot_loop_out", "lpro", 192, SST_MODULE_ID_SPROT, 1268b97169daSJie Yang SST_PATH_INDEX_SPROT_LOOP_OUT, 0, SST_TASK_SBA, SBA_VB_LPRO), 1269b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("codec_in0", "dcr", 52, SST_MODULE_ID_FILT_DCR, 1270b97169daSJie Yang SST_PATH_INDEX_CODEC_IN0, 0, SST_TASK_SBA, SBA_VB_SET_IIR), 1271b97169daSJie Yang SST_ALGO_KCONTROL_BYTES("codec_in1", "dcr", 52, SST_MODULE_ID_FILT_DCR, 1272b97169daSJie Yang SST_PATH_INDEX_CODEC_IN1, 0, SST_TASK_SBA, SBA_VB_SET_IIR), 1273b97169daSJie Yang 1274b97169daSJie Yang }; 1275b97169daSJie Yang 1276b97169daSJie Yang static int sst_algo_control_init(struct device *dev) 1277b97169daSJie Yang { 1278b97169daSJie Yang int i = 0; 1279b97169daSJie Yang struct sst_algo_control *bc; 1280b97169daSJie Yang /*allocate space to cache the algo parameters in the driver*/ 1281b97169daSJie Yang for (i = 0; i < ARRAY_SIZE(sst_algo_controls); i++) { 1282b97169daSJie Yang bc = (struct sst_algo_control *)sst_algo_controls[i].private_value; 1283b97169daSJie Yang bc->params = devm_kzalloc(dev, bc->max, GFP_KERNEL); 1284b97169daSJie Yang if (bc->params == NULL) 1285b97169daSJie Yang return -ENOMEM; 1286b97169daSJie Yang } 1287b97169daSJie Yang return 0; 1288b97169daSJie Yang } 1289b97169daSJie Yang 1290b97169daSJie Yang static bool is_sst_dapm_widget(struct snd_soc_dapm_widget *w) 1291b97169daSJie Yang { 1292b97169daSJie Yang switch (w->id) { 1293b97169daSJie Yang case snd_soc_dapm_pga: 1294b97169daSJie Yang case snd_soc_dapm_aif_in: 1295b97169daSJie Yang case snd_soc_dapm_aif_out: 1296b97169daSJie Yang case snd_soc_dapm_input: 1297b97169daSJie Yang case snd_soc_dapm_output: 1298b97169daSJie Yang case snd_soc_dapm_mixer: 1299b97169daSJie Yang return true; 1300b97169daSJie Yang default: 1301b97169daSJie Yang return false; 1302b97169daSJie Yang } 1303b97169daSJie Yang } 1304b97169daSJie Yang 1305b97169daSJie Yang /** 1306b97169daSJie Yang * sst_send_pipe_gains - send gains for the front-end DAIs 1307b97169daSJie Yang * 1308b97169daSJie Yang * The gains in the pipes connected to the front-ends are muted/unmuted 1309b97169daSJie Yang * automatically via the digital_mute() DAPM callback. This function sends the 1310b97169daSJie Yang * gains for the front-end pipes. 1311b97169daSJie Yang */ 1312b97169daSJie Yang int sst_send_pipe_gains(struct snd_soc_dai *dai, int stream, int mute) 1313b97169daSJie Yang { 1314b97169daSJie Yang struct sst_data *drv = snd_soc_dai_get_drvdata(dai); 1315b97169daSJie Yang struct snd_soc_dapm_widget *w; 1316b97169daSJie Yang struct snd_soc_dapm_path *p = NULL; 1317b97169daSJie Yang 1318b97169daSJie Yang dev_dbg(dai->dev, "enter, dai-name=%s dir=%d\n", dai->name, stream); 1319b97169daSJie Yang 1320b97169daSJie Yang if (stream == SNDRV_PCM_STREAM_PLAYBACK) { 1321b97169daSJie Yang dev_dbg(dai->dev, "Stream name=%s\n", 1322b97169daSJie Yang dai->playback_widget->name); 1323b97169daSJie Yang w = dai->playback_widget; 1324e63bfd45SLars-Peter Clausen snd_soc_dapm_widget_for_each_sink_path(w, p) { 1325b97169daSJie Yang if (p->connected && !p->connected(w, p->sink)) 1326b97169daSJie Yang continue; 1327b97169daSJie Yang 1328b97169daSJie Yang if (p->connect && p->sink->power && 1329b97169daSJie Yang is_sst_dapm_widget(p->sink)) { 1330b97169daSJie Yang struct sst_ids *ids = p->sink->priv; 1331b97169daSJie Yang 1332b97169daSJie Yang dev_dbg(dai->dev, "send gains for widget=%s\n", 1333b97169daSJie Yang p->sink->name); 1334b97169daSJie Yang mutex_lock(&drv->lock); 1335b97169daSJie Yang sst_set_pipe_gain(ids, drv, mute); 1336b97169daSJie Yang mutex_unlock(&drv->lock); 1337b97169daSJie Yang } 1338b97169daSJie Yang } 1339b97169daSJie Yang } else { 1340b97169daSJie Yang dev_dbg(dai->dev, "Stream name=%s\n", 1341b97169daSJie Yang dai->capture_widget->name); 1342b97169daSJie Yang w = dai->capture_widget; 1343e63bfd45SLars-Peter Clausen snd_soc_dapm_widget_for_each_source_path(w, p) { 1344b97169daSJie Yang if (p->connected && !p->connected(w, p->sink)) 1345b97169daSJie Yang continue; 1346b97169daSJie Yang 1347b97169daSJie Yang if (p->connect && p->source->power && 1348b97169daSJie Yang is_sst_dapm_widget(p->source)) { 1349b97169daSJie Yang struct sst_ids *ids = p->source->priv; 1350b97169daSJie Yang 1351b97169daSJie Yang dev_dbg(dai->dev, "send gain for widget=%s\n", 1352b97169daSJie Yang p->source->name); 1353b97169daSJie Yang mutex_lock(&drv->lock); 1354b97169daSJie Yang sst_set_pipe_gain(ids, drv, mute); 1355b97169daSJie Yang mutex_unlock(&drv->lock); 1356b97169daSJie Yang } 1357b97169daSJie Yang } 1358b97169daSJie Yang } 1359b97169daSJie Yang return 0; 1360b97169daSJie Yang } 1361b97169daSJie Yang 1362b97169daSJie Yang /** 1363b97169daSJie Yang * sst_fill_module_list - populate the list of modules/gains for a pipe 1364b97169daSJie Yang * 1365b97169daSJie Yang * 1366b97169daSJie Yang * Fills the widget pointer in the kcontrol private data, and also fills the 1367b97169daSJie Yang * kcontrol pointer in the widget private data. 1368b97169daSJie Yang * 1369b97169daSJie Yang * Widget pointer is used to send the algo/gain in the .put() handler if the 1370b97169daSJie Yang * widget is powerd on. 1371b97169daSJie Yang * 1372b97169daSJie Yang * Kcontrol pointer is used to send the algo/gain in the widget power ON/OFF 1373b97169daSJie Yang * event handler. Each widget (pipe) has multiple algos stored in the algo_list. 1374b97169daSJie Yang */ 1375b97169daSJie Yang static int sst_fill_module_list(struct snd_kcontrol *kctl, 1376b97169daSJie Yang struct snd_soc_dapm_widget *w, int type) 1377b97169daSJie Yang { 1378b97169daSJie Yang struct sst_module *module = NULL; 1379b97169daSJie Yang struct snd_soc_component *c = snd_soc_dapm_to_component(w->dapm); 1380b97169daSJie Yang struct sst_ids *ids = w->priv; 1381b97169daSJie Yang int ret = 0; 1382b97169daSJie Yang 1383b97169daSJie Yang module = devm_kzalloc(c->dev, sizeof(*module), GFP_KERNEL); 1384b97169daSJie Yang if (!module) 1385b97169daSJie Yang return -ENOMEM; 1386b97169daSJie Yang 1387b97169daSJie Yang if (type == SST_MODULE_GAIN) { 1388b97169daSJie Yang struct sst_gain_mixer_control *mc = (void *)kctl->private_value; 1389b97169daSJie Yang 1390b97169daSJie Yang mc->w = w; 1391b97169daSJie Yang module->kctl = kctl; 1392b97169daSJie Yang list_add_tail(&module->node, &ids->gain_list); 1393b97169daSJie Yang } else if (type == SST_MODULE_ALGO) { 1394b97169daSJie Yang struct sst_algo_control *bc = (void *)kctl->private_value; 1395b97169daSJie Yang 1396b97169daSJie Yang bc->w = w; 1397b97169daSJie Yang module->kctl = kctl; 1398b97169daSJie Yang list_add_tail(&module->node, &ids->algo_list); 1399b97169daSJie Yang } else { 1400b97169daSJie Yang dev_err(c->dev, "invoked for unknown type %d module %s", 1401b97169daSJie Yang type, kctl->id.name); 1402b97169daSJie Yang ret = -EINVAL; 1403b97169daSJie Yang } 1404b97169daSJie Yang 1405b97169daSJie Yang return ret; 1406b97169daSJie Yang } 1407b97169daSJie Yang 1408b97169daSJie Yang /** 1409b97169daSJie Yang * sst_fill_widget_module_info - fill list of gains/algos for the pipe 1410b97169daSJie Yang * @widget: pipe modelled as a DAPM widget 1411b97169daSJie Yang * 1412b97169daSJie Yang * Fill the list of gains/algos for the widget by looking at all the card 1413b97169daSJie Yang * controls and comparing the name of the widget with the first part of control 1414b97169daSJie Yang * name. First part of control name contains the pipe name (widget name). 1415b97169daSJie Yang */ 1416b97169daSJie Yang static int sst_fill_widget_module_info(struct snd_soc_dapm_widget *w, 14176840962bSKuninori Morimoto struct snd_soc_component *component) 1418b97169daSJie Yang { 1419b97169daSJie Yang struct snd_kcontrol *kctl; 1420b97169daSJie Yang int index, ret = 0; 14216840962bSKuninori Morimoto struct snd_card *card = component->card->snd_card; 1422b97169daSJie Yang char *idx; 1423b97169daSJie Yang 1424b97169daSJie Yang down_read(&card->controls_rwsem); 1425b97169daSJie Yang 1426b97169daSJie Yang list_for_each_entry(kctl, &card->controls, list) { 14276846e18aSRasmus Villemoes idx = strchr(kctl->id.name, ' '); 1428b97169daSJie Yang if (idx == NULL) 1429b97169daSJie Yang continue; 14306846e18aSRasmus Villemoes index = idx - (char*)kctl->id.name; 14316846e18aSRasmus Villemoes if (strncmp(kctl->id.name, w->name, index)) 14326846e18aSRasmus Villemoes continue; 1433b97169daSJie Yang 14346846e18aSRasmus Villemoes if (strstr(kctl->id.name, "Volume")) 1435b97169daSJie Yang ret = sst_fill_module_list(kctl, w, SST_MODULE_GAIN); 1436b97169daSJie Yang 14376846e18aSRasmus Villemoes else if (strstr(kctl->id.name, "params")) 1438b97169daSJie Yang ret = sst_fill_module_list(kctl, w, SST_MODULE_ALGO); 1439b97169daSJie Yang 1440b97169daSJie Yang else if (strstr(kctl->id.name, "Switch") && 1441b97169daSJie Yang strstr(kctl->id.name, "Gain")) { 1442b97169daSJie Yang struct sst_gain_mixer_control *mc = 1443b97169daSJie Yang (void *)kctl->private_value; 1444b97169daSJie Yang 1445b97169daSJie Yang mc->w = w; 1446b97169daSJie Yang 14476846e18aSRasmus Villemoes } else if (strstr(kctl->id.name, "interleaver")) { 1448b97169daSJie Yang struct sst_enum *e = (void *)kctl->private_value; 1449b97169daSJie Yang 1450b97169daSJie Yang e->w = w; 1451b97169daSJie Yang 14526846e18aSRasmus Villemoes } else if (strstr(kctl->id.name, "deinterleaver")) { 1453b97169daSJie Yang struct sst_enum *e = (void *)kctl->private_value; 1454b97169daSJie Yang 1455b97169daSJie Yang e->w = w; 1456b97169daSJie Yang } 1457b97169daSJie Yang 1458b97169daSJie Yang if (ret < 0) { 1459b97169daSJie Yang up_read(&card->controls_rwsem); 1460b97169daSJie Yang return ret; 1461b97169daSJie Yang } 1462b97169daSJie Yang } 1463b97169daSJie Yang 1464b97169daSJie Yang up_read(&card->controls_rwsem); 1465b97169daSJie Yang return 0; 1466b97169daSJie Yang } 1467b97169daSJie Yang 1468b97169daSJie Yang /** 1469b97169daSJie Yang * sst_fill_linked_widgets - fill the parent pointer for the linked widget 1470b97169daSJie Yang */ 14716840962bSKuninori Morimoto static void sst_fill_linked_widgets(struct snd_soc_component *component, 1472b97169daSJie Yang struct sst_ids *ids) 1473b97169daSJie Yang { 1474b97169daSJie Yang struct snd_soc_dapm_widget *w; 1475b97169daSJie Yang unsigned int len = strlen(ids->parent_wname); 1476b97169daSJie Yang 14776840962bSKuninori Morimoto list_for_each_entry(w, &component->card->widgets, list) { 1478b97169daSJie Yang if (!strncmp(ids->parent_wname, w->name, len)) { 1479b97169daSJie Yang ids->parent_w = w; 1480b97169daSJie Yang break; 1481b97169daSJie Yang } 1482b97169daSJie Yang } 1483b97169daSJie Yang } 1484b97169daSJie Yang 1485b97169daSJie Yang /** 1486b97169daSJie Yang * sst_map_modules_to_pipe - fill algo/gains list for all pipes 1487b97169daSJie Yang */ 14886840962bSKuninori Morimoto static int sst_map_modules_to_pipe(struct snd_soc_component *component) 1489b97169daSJie Yang { 1490b97169daSJie Yang struct snd_soc_dapm_widget *w; 1491b97169daSJie Yang int ret = 0; 1492b97169daSJie Yang 14936840962bSKuninori Morimoto list_for_each_entry(w, &component->card->widgets, list) { 1494b97169daSJie Yang if (is_sst_dapm_widget(w) && (w->priv)) { 1495b97169daSJie Yang struct sst_ids *ids = w->priv; 1496b97169daSJie Yang 14976840962bSKuninori Morimoto dev_dbg(component->dev, "widget type=%d name=%s\n", 1498b97169daSJie Yang w->id, w->name); 1499b97169daSJie Yang INIT_LIST_HEAD(&ids->algo_list); 1500b97169daSJie Yang INIT_LIST_HEAD(&ids->gain_list); 15016840962bSKuninori Morimoto ret = sst_fill_widget_module_info(w, component); 1502b97169daSJie Yang 1503b97169daSJie Yang if (ret < 0) 1504b97169daSJie Yang return ret; 1505b97169daSJie Yang 1506b97169daSJie Yang /* fill linked widgets */ 1507b97169daSJie Yang if (ids->parent_wname != NULL) 15086840962bSKuninori Morimoto sst_fill_linked_widgets(component, ids); 1509b97169daSJie Yang } 1510b97169daSJie Yang } 1511b97169daSJie Yang return 0; 1512b97169daSJie Yang } 1513b97169daSJie Yang 15146840962bSKuninori Morimoto int sst_dsp_init_v2_dpcm(struct snd_soc_component *component) 1515b97169daSJie Yang { 1516b97169daSJie Yang int i, ret = 0; 1517b97169daSJie Yang struct snd_soc_dapm_context *dapm = 15186840962bSKuninori Morimoto snd_soc_component_get_dapm(component); 15196840962bSKuninori Morimoto struct sst_data *drv = snd_soc_component_get_drvdata(component); 1520b97169daSJie Yang unsigned int gains = ARRAY_SIZE(sst_gain_controls)/3; 1521b97169daSJie Yang 15226840962bSKuninori Morimoto drv->byte_stream = devm_kzalloc(component->dev, 1523b97169daSJie Yang SST_MAX_BIN_BYTES, GFP_KERNEL); 1524b97169daSJie Yang if (!drv->byte_stream) 1525b97169daSJie Yang return -ENOMEM; 1526b97169daSJie Yang 1527b97169daSJie Yang snd_soc_dapm_new_controls(dapm, sst_dapm_widgets, 1528b97169daSJie Yang ARRAY_SIZE(sst_dapm_widgets)); 1529b97169daSJie Yang snd_soc_dapm_add_routes(dapm, intercon, 1530b97169daSJie Yang ARRAY_SIZE(intercon)); 1531b97169daSJie Yang snd_soc_dapm_new_widgets(dapm->card); 1532b97169daSJie Yang 1533b97169daSJie Yang for (i = 0; i < gains; i++) { 1534b97169daSJie Yang sst_gains[i].mute = SST_GAIN_MUTE_DEFAULT; 1535b97169daSJie Yang sst_gains[i].l_gain = SST_GAIN_VOLUME_DEFAULT; 1536b97169daSJie Yang sst_gains[i].r_gain = SST_GAIN_VOLUME_DEFAULT; 1537b97169daSJie Yang sst_gains[i].ramp_duration = SST_GAIN_RAMP_DURATION_DEFAULT; 1538b97169daSJie Yang } 1539b97169daSJie Yang 15406840962bSKuninori Morimoto ret = snd_soc_add_component_controls(component, sst_gain_controls, 1541b97169daSJie Yang ARRAY_SIZE(sst_gain_controls)); 1542b97169daSJie Yang if (ret) 1543b97169daSJie Yang return ret; 1544b97169daSJie Yang 1545b97169daSJie Yang /* Initialize algo control params */ 15466840962bSKuninori Morimoto ret = sst_algo_control_init(component->dev); 1547b97169daSJie Yang if (ret) 1548b97169daSJie Yang return ret; 15496840962bSKuninori Morimoto ret = snd_soc_add_component_controls(component, sst_algo_controls, 1550b97169daSJie Yang ARRAY_SIZE(sst_algo_controls)); 1551b97169daSJie Yang if (ret) 1552b97169daSJie Yang return ret; 1553b97169daSJie Yang 15546840962bSKuninori Morimoto ret = snd_soc_add_component_controls(component, sst_slot_controls, 1555b97169daSJie Yang ARRAY_SIZE(sst_slot_controls)); 1556b97169daSJie Yang if (ret) 1557b97169daSJie Yang return ret; 1558b97169daSJie Yang 15596840962bSKuninori Morimoto ret = sst_map_modules_to_pipe(component); 1560b97169daSJie Yang 1561b97169daSJie Yang return ret; 1562b97169daSJie Yang } 1563