1219 lines
33 KiB
C
1219 lines
33 KiB
C
|
/* Copyright (c) 2009-2011, 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/module.h>
|
||
|
#include <linux/platform_device.h>
|
||
|
#include <linux/mfd/msm-adie-codec.h>
|
||
|
#include <linux/clk.h>
|
||
|
#include <linux/err.h>
|
||
|
#include <linux/debugfs.h>
|
||
|
#include <linux/pm_qos.h>
|
||
|
|
||
|
#include <asm/uaccess.h>
|
||
|
#include <mach/qdsp5v2/snddev_icodec.h>
|
||
|
#include <mach/qdsp5v2/audio_dev_ctl.h>
|
||
|
#include <mach/qdsp5v2/audio_interct.h>
|
||
|
#include <mach/qdsp5v2/mi2s.h>
|
||
|
#include <mach/qdsp5v2/afe.h>
|
||
|
#include <mach/qdsp5v2/lpa.h>
|
||
|
#include <mach/qdsp5v2/marimba_profile.h>
|
||
|
#include <mach/vreg.h>
|
||
|
#include <mach/pmic.h>
|
||
|
#include <linux/wakelock.h>
|
||
|
#include <mach/debug_mm.h>
|
||
|
#include <mach/rpc_pmapp.h>
|
||
|
#include <mach/qdsp5v2/audio_acdb_def.h>
|
||
|
#include <linux/slab.h>
|
||
|
#include <mach/cpuidle.h>
|
||
|
|
||
|
#define SMPS_AUDIO_PLAYBACK_ID "AUPB"
|
||
|
#define SMPS_AUDIO_RECORD_ID "AURC"
|
||
|
|
||
|
#define SNDDEV_ICODEC_PCM_SZ 32 /* 16 bit / sample stereo mode */
|
||
|
#define SNDDEV_ICODEC_MUL_FACTOR 3 /* Multi by 8 Shift by 3 */
|
||
|
#define SNDDEV_ICODEC_CLK_RATE(freq) \
|
||
|
(((freq) * (SNDDEV_ICODEC_PCM_SZ)) << (SNDDEV_ICODEC_MUL_FACTOR))
|
||
|
|
||
|
#ifdef CONFIG_DEBUG_FS
|
||
|
static struct adie_codec_action_unit debug_rx_actions[] =
|
||
|
HANDSET_RX_8000_OSR_256;
|
||
|
|
||
|
static struct adie_codec_action_unit debug_tx_lb_actions[] = {
|
||
|
{ ADIE_CODEC_ACTION_STAGE_REACHED, ADIE_CODEC_DIGITAL_OFF },
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x80, 0x01, 0x01)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x80, 0x01, 0x00) },
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x8A, 0x30, 0x30)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x11, 0xfc, 0xfc)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x13, 0xfc, 0x58)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x14, 0xff, 0x65)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x15, 0xff, 0x64)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x82, 0xff, 0x5C)},
|
||
|
{ ADIE_CODEC_ACTION_STAGE_REACHED, ADIE_CODEC_DIGITAL_READY },
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x0D, 0xF0, 0xd0)},
|
||
|
{ ADIE_CODEC_ACTION_DELAY_WAIT, 0xbb8},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x83, 0x14, 0x14)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x86, 0xff, 0x00)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x8A, 0x50, 0x40)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x91, 0xFF, 0x01)}, /* Start loop back */
|
||
|
{ ADIE_CODEC_ACTION_STAGE_REACHED, ADIE_CODEC_DIGITAL_ANALOG_READY},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x8A, 0x10, 0x30)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x0D, 0xFF, 0x00)},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x83, 0x14, 0x00)},
|
||
|
{ ADIE_CODEC_ACTION_STAGE_REACHED, ADIE_CODEC_ANALOG_OFF},
|
||
|
{ ADIE_CODEC_ACTION_ENTRY,
|
||
|
ADIE_CODEC_PACK_ENTRY(0x11, 0xff, 0x00)}
|
||
|
};
|
||
|
|
||
|
static struct adie_codec_action_unit debug_tx_actions[] =
|
||
|
HANDSET_TX_8000_OSR_256;
|
||
|
|
||
|
static struct adie_codec_hwsetting_entry debug_rx_settings[] = {
|
||
|
{
|
||
|
.freq_plan = 8000,
|
||
|
.osr = 256,
|
||
|
.actions = debug_rx_actions,
|
||
|
.action_sz = ARRAY_SIZE(debug_rx_actions),
|
||
|
}
|
||
|
};
|
||
|
|
||
|
static struct adie_codec_hwsetting_entry debug_tx_settings[] = {
|
||
|
{
|
||
|
.freq_plan = 8000,
|
||
|
.osr = 256,
|
||
|
.actions = debug_tx_actions,
|
||
|
.action_sz = ARRAY_SIZE(debug_tx_actions),
|
||
|
}
|
||
|
};
|
||
|
|
||
|
static struct adie_codec_hwsetting_entry debug_tx_lb_settings[] = {
|
||
|
{
|
||
|
.freq_plan = 8000,
|
||
|
.osr = 256,
|
||
|
.actions = debug_tx_lb_actions,
|
||
|
.action_sz = ARRAY_SIZE(debug_tx_lb_actions),
|
||
|
}
|
||
|
};
|
||
|
|
||
|
static struct adie_codec_dev_profile debug_rx_profile = {
|
||
|
.path_type = ADIE_CODEC_RX,
|
||
|
.settings = debug_rx_settings,
|
||
|
.setting_sz = ARRAY_SIZE(debug_rx_settings),
|
||
|
};
|
||
|
|
||
|
static struct adie_codec_dev_profile debug_tx_profile = {
|
||
|
.path_type = ADIE_CODEC_TX,
|
||
|
.settings = debug_tx_settings,
|
||
|
.setting_sz = ARRAY_SIZE(debug_tx_settings),
|
||
|
};
|
||
|
|
||
|
static struct adie_codec_dev_profile debug_tx_lb_profile = {
|
||
|
.path_type = ADIE_CODEC_TX,
|
||
|
.settings = debug_tx_lb_settings,
|
||
|
.setting_sz = ARRAY_SIZE(debug_tx_lb_settings),
|
||
|
};
|
||
|
#endif /* CONFIG_DEBUG_FS */
|
||
|
|
||
|
/* Context for each internal codec sound device */
|
||
|
struct snddev_icodec_state {
|
||
|
struct snddev_icodec_data *data;
|
||
|
struct adie_codec_path *adie_path;
|
||
|
u32 sample_rate;
|
||
|
u32 enabled;
|
||
|
};
|
||
|
|
||
|
/* Global state for the driver */
|
||
|
struct snddev_icodec_drv_state {
|
||
|
struct mutex rx_lock;
|
||
|
struct mutex lb_lock;
|
||
|
struct mutex tx_lock;
|
||
|
u32 rx_active; /* ensure one rx device at a time */
|
||
|
u32 tx_active; /* ensure one tx device at a time */
|
||
|
struct clk *rx_mclk;
|
||
|
struct clk *rx_sclk;
|
||
|
struct clk *tx_mclk;
|
||
|
struct clk *tx_sclk;
|
||
|
struct clk *lpa_codec_clk;
|
||
|
struct clk *lpa_core_clk;
|
||
|
struct clk *lpa_p_clk;
|
||
|
struct lpa_drv *lpa;
|
||
|
|
||
|
struct pm_qos_request rx_pm_qos_req;
|
||
|
struct pm_qos_request tx_pm_qos_req;
|
||
|
};
|
||
|
|
||
|
static struct snddev_icodec_drv_state snddev_icodec_drv;
|
||
|
|
||
|
static int snddev_icodec_open_rx(struct snddev_icodec_state *icodec)
|
||
|
{
|
||
|
int trc, err;
|
||
|
int smps_mode = PMAPP_SMPS_MODE_VOTE_PWM;
|
||
|
struct msm_afe_config afe_config;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
struct lpa_codec_config lpa_config;
|
||
|
|
||
|
pm_qos_update_request(&drv->rx_pm_qos_req,
|
||
|
msm_cpuidle_get_deep_idle_latency());
|
||
|
|
||
|
if ((icodec->data->acdb_id == ACDB_ID_HEADSET_SPKR_MONO) ||
|
||
|
(icodec->data->acdb_id == ACDB_ID_HEADSET_SPKR_STEREO)) {
|
||
|
/* Vote PMAPP_SMPS_MODE_VOTE_PFM for headset */
|
||
|
smps_mode = PMAPP_SMPS_MODE_VOTE_PFM;
|
||
|
MM_DBG("snddev_icodec_open_rx: PMAPP_SMPS_MODE_VOTE_PFM \n");
|
||
|
} else
|
||
|
MM_DBG("snddev_icodec_open_rx: PMAPP_SMPS_MODE_VOTE_PWM \n");
|
||
|
|
||
|
/* Vote for SMPS mode*/
|
||
|
err = pmapp_smps_mode_vote(SMPS_AUDIO_PLAYBACK_ID,
|
||
|
PMAPP_VREG_S4, smps_mode);
|
||
|
if (err != 0)
|
||
|
MM_ERR("pmapp_smps_mode_vote error %d\n", err);
|
||
|
|
||
|
/* enable MI2S RX master block */
|
||
|
/* enable MI2S RX bit clock */
|
||
|
trc = clk_set_rate(drv->rx_mclk,
|
||
|
SNDDEV_ICODEC_CLK_RATE(icodec->sample_rate));
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
goto error_invalid_freq;
|
||
|
clk_prepare_enable(drv->rx_mclk);
|
||
|
clk_prepare_enable(drv->rx_sclk);
|
||
|
/* clk_set_rate(drv->lpa_codec_clk, 1); */ /* Remove if use pcom */
|
||
|
clk_prepare_enable(drv->lpa_p_clk);
|
||
|
clk_prepare_enable(drv->lpa_codec_clk);
|
||
|
clk_prepare_enable(drv->lpa_core_clk);
|
||
|
|
||
|
/* Enable LPA sub system
|
||
|
*/
|
||
|
drv->lpa = lpa_get();
|
||
|
if (!drv->lpa)
|
||
|
goto error_lpa;
|
||
|
lpa_config.sample_rate = icodec->sample_rate;
|
||
|
lpa_config.sample_width = 16;
|
||
|
lpa_config.output_interface = LPA_OUTPUT_INTF_WB_CODEC;
|
||
|
lpa_config.num_channels = icodec->data->channel_mode;
|
||
|
lpa_cmd_codec_config(drv->lpa, &lpa_config);
|
||
|
|
||
|
/* Set audio interconnect reg to LPA */
|
||
|
audio_interct_codec(AUDIO_INTERCT_LPA);
|
||
|
|
||
|
/* Set MI2S */
|
||
|
mi2s_set_codec_output_path((icodec->data->channel_mode == 2 ?
|
||
|
MI2S_CHAN_STEREO : MI2S_CHAN_MONO_PACKED), WT_16_BIT);
|
||
|
|
||
|
if (icodec->data->voltage_on)
|
||
|
icodec->data->voltage_on();
|
||
|
|
||
|
/* Configure ADIE */
|
||
|
trc = adie_codec_open(icodec->data->profile, &icodec->adie_path);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
goto error_adie;
|
||
|
/* OSR default to 256, can be changed for power optimization
|
||
|
* If OSR is to be changed, need clock API for setting the divider
|
||
|
*/
|
||
|
adie_codec_setpath(icodec->adie_path, icodec->sample_rate, 256);
|
||
|
/* Start AFE */
|
||
|
afe_config.sample_rate = icodec->sample_rate / 1000;
|
||
|
afe_config.channel_mode = icodec->data->channel_mode;
|
||
|
afe_config.volume = AFE_VOLUME_UNITY;
|
||
|
trc = afe_enable(AFE_HW_PATH_CODEC_RX, &afe_config);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
goto error_afe;
|
||
|
lpa_cmd_enable_codec(drv->lpa, 1);
|
||
|
/* Enable ADIE */
|
||
|
adie_codec_proceed_stage(icodec->adie_path, ADIE_CODEC_DIGITAL_READY);
|
||
|
adie_codec_proceed_stage(icodec->adie_path,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
|
||
|
/* Enable power amplifier */
|
||
|
if (icodec->data->pamp_on)
|
||
|
icodec->data->pamp_on();
|
||
|
|
||
|
icodec->enabled = 1;
|
||
|
|
||
|
pm_qos_update_request(&drv->rx_pm_qos_req, PM_QOS_DEFAULT_VALUE);
|
||
|
return 0;
|
||
|
|
||
|
error_afe:
|
||
|
adie_codec_close(icodec->adie_path);
|
||
|
icodec->adie_path = NULL;
|
||
|
error_adie:
|
||
|
lpa_put(drv->lpa);
|
||
|
error_lpa:
|
||
|
clk_disable_unprepare(drv->lpa_p_clk);
|
||
|
clk_disable_unprepare(drv->lpa_codec_clk);
|
||
|
clk_disable_unprepare(drv->lpa_core_clk);
|
||
|
clk_disable_unprepare(drv->rx_sclk);
|
||
|
clk_disable_unprepare(drv->rx_mclk);
|
||
|
error_invalid_freq:
|
||
|
|
||
|
MM_ERR("encounter error\n");
|
||
|
|
||
|
pm_qos_update_request(&drv->rx_pm_qos_req, PM_QOS_DEFAULT_VALUE);
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_open_tx(struct snddev_icodec_state *icodec)
|
||
|
{
|
||
|
int trc;
|
||
|
int i, err;
|
||
|
struct msm_afe_config afe_config;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;;
|
||
|
|
||
|
pm_qos_update_request(&drv->tx_pm_qos_req,
|
||
|
msm_cpuidle_get_deep_idle_latency());
|
||
|
|
||
|
/* Vote for PWM mode*/
|
||
|
err = pmapp_smps_mode_vote(SMPS_AUDIO_RECORD_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_PWM);
|
||
|
if (err != 0)
|
||
|
MM_ERR("pmapp_smps_mode_vote error %d\n", err);
|
||
|
|
||
|
/* Reuse pamp_on for TX platform-specific setup */
|
||
|
if (icodec->data->pamp_on)
|
||
|
icodec->data->pamp_on();
|
||
|
|
||
|
for (i = 0; i < icodec->data->pmctl_id_sz; i++) {
|
||
|
pmic_hsed_enable(icodec->data->pmctl_id[i],
|
||
|
PM_HSED_ENABLE_PWM_TCXO);
|
||
|
}
|
||
|
|
||
|
/* enable MI2S TX master block */
|
||
|
/* enable MI2S TX bit clock */
|
||
|
trc = clk_set_rate(drv->tx_mclk,
|
||
|
SNDDEV_ICODEC_CLK_RATE(icodec->sample_rate));
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
goto error_invalid_freq;
|
||
|
clk_prepare_enable(drv->tx_mclk);
|
||
|
clk_prepare_enable(drv->tx_sclk);
|
||
|
|
||
|
/* Set MI2S */
|
||
|
mi2s_set_codec_input_path((icodec->data->channel_mode ==
|
||
|
REAL_STEREO_CHANNEL_MODE ? MI2S_CHAN_STEREO :
|
||
|
(icodec->data->channel_mode == 2 ?
|
||
|
MI2S_CHAN_STEREO : MI2S_CHAN_MONO_RAW)),
|
||
|
WT_16_BIT);
|
||
|
/* Configure ADIE */
|
||
|
trc = adie_codec_open(icodec->data->profile, &icodec->adie_path);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
goto error_adie;
|
||
|
/* Enable ADIE */
|
||
|
adie_codec_setpath(icodec->adie_path, icodec->sample_rate, 256);
|
||
|
adie_codec_proceed_stage(icodec->adie_path, ADIE_CODEC_DIGITAL_READY);
|
||
|
adie_codec_proceed_stage(icodec->adie_path,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
|
||
|
/* Start AFE */
|
||
|
afe_config.sample_rate = icodec->sample_rate / 1000;
|
||
|
afe_config.channel_mode = icodec->data->channel_mode;
|
||
|
afe_config.volume = AFE_VOLUME_UNITY;
|
||
|
trc = afe_enable(AFE_HW_PATH_CODEC_TX, &afe_config);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
goto error_afe;
|
||
|
|
||
|
|
||
|
icodec->enabled = 1;
|
||
|
|
||
|
pm_qos_update_request(&drv->tx_pm_qos_req, PM_QOS_DEFAULT_VALUE);
|
||
|
return 0;
|
||
|
|
||
|
error_afe:
|
||
|
adie_codec_close(icodec->adie_path);
|
||
|
icodec->adie_path = NULL;
|
||
|
error_adie:
|
||
|
clk_disable_unprepare(drv->tx_sclk);
|
||
|
clk_disable_unprepare(drv->tx_mclk);
|
||
|
error_invalid_freq:
|
||
|
|
||
|
/* Disable mic bias */
|
||
|
for (i = 0; i < icodec->data->pmctl_id_sz; i++) {
|
||
|
pmic_hsed_enable(icodec->data->pmctl_id[i],
|
||
|
PM_HSED_ENABLE_OFF);
|
||
|
}
|
||
|
|
||
|
if (icodec->data->pamp_off)
|
||
|
icodec->data->pamp_off();
|
||
|
|
||
|
MM_ERR("encounter error\n");
|
||
|
|
||
|
pm_qos_update_request(&drv->tx_pm_qos_req, PM_QOS_DEFAULT_VALUE);
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_close_lb(struct snddev_icodec_state *icodec)
|
||
|
{
|
||
|
/* Disable power amplifier */
|
||
|
if (icodec->data->pamp_off)
|
||
|
icodec->data->pamp_off();
|
||
|
|
||
|
if (icodec->adie_path) {
|
||
|
adie_codec_proceed_stage(icodec->adie_path,
|
||
|
ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(icodec->adie_path);
|
||
|
icodec->adie_path = NULL;
|
||
|
}
|
||
|
if (icodec->data->voltage_off)
|
||
|
icodec->data->voltage_off();
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_close_rx(struct snddev_icodec_state *icodec)
|
||
|
{
|
||
|
int err;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
|
||
|
pm_qos_update_request(&drv->rx_pm_qos_req,
|
||
|
msm_cpuidle_get_deep_idle_latency());
|
||
|
|
||
|
/* Remove the vote for SMPS mode*/
|
||
|
err = pmapp_smps_mode_vote(SMPS_AUDIO_PLAYBACK_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_DONTCARE);
|
||
|
if (err != 0)
|
||
|
MM_ERR("pmapp_smps_mode_vote error %d\n", err);
|
||
|
|
||
|
/* Disable power amplifier */
|
||
|
if (icodec->data->pamp_off)
|
||
|
icodec->data->pamp_off();
|
||
|
|
||
|
/* Disable ADIE */
|
||
|
adie_codec_proceed_stage(icodec->adie_path, ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(icodec->adie_path);
|
||
|
icodec->adie_path = NULL;
|
||
|
|
||
|
afe_disable(AFE_HW_PATH_CODEC_RX);
|
||
|
|
||
|
if (icodec->data->voltage_off)
|
||
|
icodec->data->voltage_off();
|
||
|
|
||
|
/* Disable LPA Sub system */
|
||
|
lpa_cmd_enable_codec(drv->lpa, 0);
|
||
|
lpa_put(drv->lpa);
|
||
|
|
||
|
/* Disable LPA clocks */
|
||
|
clk_disable_unprepare(drv->lpa_p_clk);
|
||
|
clk_disable_unprepare(drv->lpa_codec_clk);
|
||
|
clk_disable_unprepare(drv->lpa_core_clk);
|
||
|
|
||
|
/* Disable MI2S RX master block */
|
||
|
/* Disable MI2S RX bit clock */
|
||
|
clk_disable_unprepare(drv->rx_sclk);
|
||
|
clk_disable_unprepare(drv->rx_mclk);
|
||
|
|
||
|
icodec->enabled = 0;
|
||
|
|
||
|
pm_qos_update_request(&drv->rx_pm_qos_req, PM_QOS_DEFAULT_VALUE);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_close_tx(struct snddev_icodec_state *icodec)
|
||
|
{
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
int i, err;
|
||
|
|
||
|
pm_qos_update_request(&drv->tx_pm_qos_req,
|
||
|
msm_cpuidle_get_deep_idle_latency());
|
||
|
|
||
|
/* Remove the vote for SMPS mode*/
|
||
|
err = pmapp_smps_mode_vote(SMPS_AUDIO_RECORD_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_DONTCARE);
|
||
|
if (err != 0)
|
||
|
MM_ERR("pmapp_smps_mode_vote error %d\n", err);
|
||
|
|
||
|
afe_disable(AFE_HW_PATH_CODEC_TX);
|
||
|
|
||
|
/* Disable ADIE */
|
||
|
adie_codec_proceed_stage(icodec->adie_path, ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(icodec->adie_path);
|
||
|
icodec->adie_path = NULL;
|
||
|
|
||
|
/* Disable MI2S TX master block */
|
||
|
/* Disable MI2S TX bit clock */
|
||
|
clk_disable_unprepare(drv->tx_sclk);
|
||
|
clk_disable_unprepare(drv->tx_mclk);
|
||
|
|
||
|
/* Disable mic bias */
|
||
|
for (i = 0; i < icodec->data->pmctl_id_sz; i++) {
|
||
|
pmic_hsed_enable(icodec->data->pmctl_id[i],
|
||
|
PM_HSED_ENABLE_OFF);
|
||
|
}
|
||
|
|
||
|
/* Reuse pamp_off for TX platform-specific setup */
|
||
|
if (icodec->data->pamp_off)
|
||
|
icodec->data->pamp_off();
|
||
|
|
||
|
icodec->enabled = 0;
|
||
|
|
||
|
pm_qos_update_request(&drv->tx_pm_qos_req, PM_QOS_DEFAULT_VALUE);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_open_lb(struct snddev_icodec_state *icodec)
|
||
|
{
|
||
|
int trc;
|
||
|
trc = adie_codec_open(icodec->data->profile, &icodec->adie_path);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
pr_err("%s: adie codec open failed\n", __func__);
|
||
|
else
|
||
|
adie_codec_setpath(icodec->adie_path,
|
||
|
icodec->sample_rate, 256);
|
||
|
|
||
|
if (icodec->adie_path)
|
||
|
adie_codec_proceed_stage(icodec->adie_path,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
if (icodec->data->pamp_on)
|
||
|
icodec->data->pamp_on();
|
||
|
|
||
|
icodec->enabled = 1;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_set_device_volume_impl(
|
||
|
struct msm_snddev_info *dev_info, u32 volume)
|
||
|
{
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
u8 afe_path_id;
|
||
|
|
||
|
int rc = 0;
|
||
|
|
||
|
icodec = dev_info->private_data;
|
||
|
|
||
|
if (icodec->data->capability & SNDDEV_CAP_RX)
|
||
|
afe_path_id = AFE_HW_PATH_CODEC_RX;
|
||
|
else
|
||
|
afe_path_id = AFE_HW_PATH_CODEC_TX;
|
||
|
|
||
|
if (icodec->data->dev_vol_type & SNDDEV_DEV_VOL_DIGITAL) {
|
||
|
|
||
|
rc = adie_codec_set_device_digital_volume(icodec->adie_path,
|
||
|
icodec->data->channel_mode ==
|
||
|
REAL_STEREO_CHANNEL_MODE ?
|
||
|
2 : icodec->data->channel_mode, volume);
|
||
|
if (rc < 0) {
|
||
|
MM_ERR("unable to set_device_digital_volume for"
|
||
|
"%s volume in percentage = %u\n",
|
||
|
dev_info->name, volume);
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
} else if (icodec->data->dev_vol_type & SNDDEV_DEV_VOL_ANALOG) {
|
||
|
rc = adie_codec_set_device_analog_volume(icodec->adie_path,
|
||
|
icodec->data->channel_mode ==
|
||
|
REAL_STEREO_CHANNEL_MODE ?
|
||
|
2 : icodec->data->channel_mode, volume);
|
||
|
if (rc < 0) {
|
||
|
MM_ERR("unable to set_device_analog_volume for"
|
||
|
"%s volume in percentage = %u\n",
|
||
|
dev_info->name, volume);
|
||
|
return rc;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
MM_ERR("Invalid device volume control\n");
|
||
|
return -EPERM;
|
||
|
}
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_close(struct msm_snddev_info *dev_info)
|
||
|
{
|
||
|
int rc = 0;
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
if (!dev_info) {
|
||
|
rc = -EINVAL;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
icodec = dev_info->private_data;
|
||
|
|
||
|
if (icodec->data->capability & SNDDEV_CAP_RX) {
|
||
|
mutex_lock(&drv->rx_lock);
|
||
|
if (!drv->rx_active) {
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
rc = -EPERM;
|
||
|
goto error;
|
||
|
}
|
||
|
rc = snddev_icodec_close_rx(icodec);
|
||
|
if (!IS_ERR_VALUE(rc))
|
||
|
drv->rx_active = 0;
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
} else if (icodec->data->capability & SNDDEV_CAP_LB) {
|
||
|
mutex_lock(&drv->lb_lock);
|
||
|
rc = snddev_icodec_close_lb(icodec);
|
||
|
mutex_unlock(&drv->lb_lock);
|
||
|
} else {
|
||
|
mutex_lock(&drv->tx_lock);
|
||
|
if (!drv->tx_active) {
|
||
|
mutex_unlock(&drv->tx_lock);
|
||
|
rc = -EPERM;
|
||
|
goto error;
|
||
|
}
|
||
|
rc = snddev_icodec_close_tx(icodec);
|
||
|
if (!IS_ERR_VALUE(rc))
|
||
|
drv->tx_active = 0;
|
||
|
mutex_unlock(&drv->tx_lock);
|
||
|
}
|
||
|
|
||
|
error:
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_open(struct msm_snddev_info *dev_info)
|
||
|
{
|
||
|
int rc = 0;
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
|
||
|
if (!dev_info) {
|
||
|
rc = -EINVAL;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
icodec = dev_info->private_data;
|
||
|
|
||
|
if (icodec->data->capability & SNDDEV_CAP_RX) {
|
||
|
mutex_lock(&drv->rx_lock);
|
||
|
if (drv->rx_active) {
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
rc = -EBUSY;
|
||
|
goto error;
|
||
|
}
|
||
|
rc = snddev_icodec_open_rx(icodec);
|
||
|
|
||
|
if (!IS_ERR_VALUE(rc)) {
|
||
|
drv->rx_active = 1;
|
||
|
if ((icodec->data->dev_vol_type & (
|
||
|
SNDDEV_DEV_VOL_DIGITAL |
|
||
|
SNDDEV_DEV_VOL_ANALOG)))
|
||
|
rc = snddev_icodec_set_device_volume_impl(
|
||
|
dev_info, dev_info->dev_volume);
|
||
|
if (IS_ERR_VALUE(rc)) {
|
||
|
MM_ERR("Failed to set device volume"
|
||
|
" impl for rx device\n");
|
||
|
snddev_icodec_close(dev_info);
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
goto error;
|
||
|
}
|
||
|
}
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
} else if (icodec->data->capability & SNDDEV_CAP_LB) {
|
||
|
mutex_lock(&drv->lb_lock);
|
||
|
rc = snddev_icodec_open_lb(icodec);
|
||
|
if (!IS_ERR_VALUE(rc)) {
|
||
|
if ((icodec->data->dev_vol_type & (
|
||
|
SNDDEV_DEV_VOL_DIGITAL |
|
||
|
SNDDEV_DEV_VOL_ANALOG)))
|
||
|
rc = snddev_icodec_set_device_volume_impl(
|
||
|
dev_info,
|
||
|
dev_info->dev_volume);
|
||
|
if (rc < 0)
|
||
|
MM_ERR("failed to set device volume\n");
|
||
|
}
|
||
|
mutex_unlock(&drv->lb_lock);
|
||
|
} else {
|
||
|
mutex_lock(&drv->tx_lock);
|
||
|
if (drv->tx_active) {
|
||
|
mutex_unlock(&drv->tx_lock);
|
||
|
rc = -EBUSY;
|
||
|
goto error;
|
||
|
}
|
||
|
rc = snddev_icodec_open_tx(icodec);
|
||
|
|
||
|
if (!IS_ERR_VALUE(rc)) {
|
||
|
drv->tx_active = 1;
|
||
|
if ((icodec->data->dev_vol_type & (
|
||
|
SNDDEV_DEV_VOL_DIGITAL |
|
||
|
SNDDEV_DEV_VOL_ANALOG)))
|
||
|
rc = snddev_icodec_set_device_volume_impl(
|
||
|
dev_info, dev_info->dev_volume);
|
||
|
if (IS_ERR_VALUE(rc)) {
|
||
|
MM_ERR("Failed to set device volume"
|
||
|
" impl for tx device\n");
|
||
|
snddev_icodec_close(dev_info);
|
||
|
mutex_unlock(&drv->tx_lock);
|
||
|
goto error;
|
||
|
}
|
||
|
}
|
||
|
mutex_unlock(&drv->tx_lock);
|
||
|
}
|
||
|
error:
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_check_freq(u32 req_freq)
|
||
|
{
|
||
|
int rc = -EINVAL;
|
||
|
|
||
|
if ((req_freq != 0) && (req_freq >= 8000) && (req_freq <= 48000)) {
|
||
|
if ((req_freq == 8000) || (req_freq == 11025) ||
|
||
|
(req_freq == 12000) || (req_freq == 16000) ||
|
||
|
(req_freq == 22050) || (req_freq == 24000) ||
|
||
|
(req_freq == 32000) || (req_freq == 44100) ||
|
||
|
(req_freq == 48000)) {
|
||
|
rc = 0;
|
||
|
} else
|
||
|
MM_INFO("Unsupported Frequency:%d\n", req_freq);
|
||
|
}
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_set_freq(struct msm_snddev_info *dev_info, u32 rate)
|
||
|
{
|
||
|
int rc;
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
|
||
|
if (!dev_info) {
|
||
|
rc = -EINVAL;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
icodec = dev_info->private_data;
|
||
|
if (adie_codec_freq_supported(icodec->data->profile, rate) != 0) {
|
||
|
rc = -EINVAL;
|
||
|
goto error;
|
||
|
} else {
|
||
|
if (snddev_icodec_check_freq(rate) != 0) {
|
||
|
rc = -EINVAL;
|
||
|
goto error;
|
||
|
} else
|
||
|
icodec->sample_rate = rate;
|
||
|
}
|
||
|
|
||
|
if (icodec->enabled) {
|
||
|
snddev_icodec_close(dev_info);
|
||
|
snddev_icodec_open(dev_info);
|
||
|
}
|
||
|
|
||
|
return icodec->sample_rate;
|
||
|
|
||
|
error:
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_enable_sidetone(struct msm_snddev_info *dev_info,
|
||
|
u32 enable)
|
||
|
{
|
||
|
int rc = 0;
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
|
||
|
if (!dev_info) {
|
||
|
MM_ERR("invalid dev_info\n");
|
||
|
rc = -EINVAL;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
icodec = dev_info->private_data;
|
||
|
|
||
|
if (icodec->data->capability & SNDDEV_CAP_RX) {
|
||
|
mutex_lock(&drv->rx_lock);
|
||
|
if (!drv->rx_active || !dev_info->opened) {
|
||
|
MM_ERR("dev not active\n");
|
||
|
rc = -EPERM;
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
goto error;
|
||
|
}
|
||
|
rc = adie_codec_enable_sidetone(icodec->adie_path, enable);
|
||
|
mutex_unlock(&drv->rx_lock);
|
||
|
} else {
|
||
|
rc = -EINVAL;
|
||
|
MM_ERR("rx device only\n");
|
||
|
}
|
||
|
|
||
|
error:
|
||
|
return rc;
|
||
|
|
||
|
}
|
||
|
|
||
|
int snddev_icodec_set_device_volume(struct msm_snddev_info *dev_info,
|
||
|
u32 volume)
|
||
|
{
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
struct mutex *lock;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
int rc = -EPERM;
|
||
|
|
||
|
if (!dev_info) {
|
||
|
MM_INFO("device not intilized.\n");
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
icodec = dev_info->private_data;
|
||
|
|
||
|
if (!(icodec->data->dev_vol_type & (SNDDEV_DEV_VOL_DIGITAL
|
||
|
| SNDDEV_DEV_VOL_ANALOG))) {
|
||
|
|
||
|
MM_INFO("device %s does not support device volume "
|
||
|
"control.", dev_info->name);
|
||
|
return -EPERM;
|
||
|
}
|
||
|
dev_info->dev_volume = volume;
|
||
|
|
||
|
if (icodec->data->capability & SNDDEV_CAP_RX)
|
||
|
lock = &drv->rx_lock;
|
||
|
else if (icodec->data->capability & SNDDEV_CAP_LB)
|
||
|
lock = &drv->lb_lock;
|
||
|
else
|
||
|
lock = &drv->tx_lock;
|
||
|
|
||
|
mutex_lock(lock);
|
||
|
|
||
|
rc = snddev_icodec_set_device_volume_impl(dev_info,
|
||
|
dev_info->dev_volume);
|
||
|
mutex_unlock(lock);
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_probe(struct platform_device *pdev)
|
||
|
{
|
||
|
int rc = 0, i;
|
||
|
struct snddev_icodec_data *pdata;
|
||
|
struct msm_snddev_info *dev_info;
|
||
|
struct snddev_icodec_state *icodec;
|
||
|
|
||
|
if (!pdev || !pdev->dev.platform_data) {
|
||
|
printk(KERN_ALERT "Invalid caller \n");
|
||
|
rc = -1;
|
||
|
goto error;
|
||
|
}
|
||
|
pdata = pdev->dev.platform_data;
|
||
|
if ((pdata->capability & SNDDEV_CAP_RX) &&
|
||
|
(pdata->capability & SNDDEV_CAP_TX)) {
|
||
|
MM_ERR("invalid device data either RX or TX\n");
|
||
|
goto error;
|
||
|
}
|
||
|
icodec = kzalloc(sizeof(struct snddev_icodec_state), GFP_KERNEL);
|
||
|
if (!icodec) {
|
||
|
rc = -ENOMEM;
|
||
|
goto error;
|
||
|
}
|
||
|
dev_info = kmalloc(sizeof(struct msm_snddev_info), GFP_KERNEL);
|
||
|
if (!dev_info) {
|
||
|
kfree(icodec);
|
||
|
rc = -ENOMEM;
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
dev_info->name = pdata->name;
|
||
|
dev_info->copp_id = pdata->copp_id;
|
||
|
dev_info->acdb_id = pdata->acdb_id;
|
||
|
dev_info->private_data = (void *) icodec;
|
||
|
dev_info->dev_ops.open = snddev_icodec_open;
|
||
|
dev_info->dev_ops.close = snddev_icodec_close;
|
||
|
dev_info->dev_ops.set_freq = snddev_icodec_set_freq;
|
||
|
dev_info->dev_ops.set_device_volume = snddev_icodec_set_device_volume;
|
||
|
dev_info->capability = pdata->capability;
|
||
|
dev_info->opened = 0;
|
||
|
msm_snddev_register(dev_info);
|
||
|
icodec->data = pdata;
|
||
|
icodec->sample_rate = pdata->default_sample_rate;
|
||
|
dev_info->sample_rate = pdata->default_sample_rate;
|
||
|
if (pdata->capability & SNDDEV_CAP_RX) {
|
||
|
for (i = 0; i < VOC_RX_VOL_ARRAY_NUM; i++) {
|
||
|
dev_info->max_voc_rx_vol[i] =
|
||
|
pdata->max_voice_rx_vol[i];
|
||
|
dev_info->min_voc_rx_vol[i] =
|
||
|
pdata->min_voice_rx_vol[i];
|
||
|
}
|
||
|
/*sidetone is enabled only for the device which
|
||
|
property set for side tone*/
|
||
|
if (pdata->property & SIDE_TONE_MASK)
|
||
|
dev_info->dev_ops.enable_sidetone =
|
||
|
snddev_icodec_enable_sidetone;
|
||
|
else
|
||
|
dev_info->dev_ops.enable_sidetone = NULL;
|
||
|
} else {
|
||
|
dev_info->dev_ops.enable_sidetone = NULL;
|
||
|
}
|
||
|
|
||
|
error:
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
static int snddev_icodec_remove(struct platform_device *pdev)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static struct platform_driver snddev_icodec_driver = {
|
||
|
.probe = snddev_icodec_probe,
|
||
|
.remove = snddev_icodec_remove,
|
||
|
.driver = { .name = "snddev_icodec" }
|
||
|
};
|
||
|
|
||
|
#ifdef CONFIG_DEBUG_FS
|
||
|
static struct dentry *debugfs_sdev_dent;
|
||
|
static struct dentry *debugfs_afelb;
|
||
|
static struct dentry *debugfs_adielb;
|
||
|
static struct adie_codec_path *debugfs_rx_adie;
|
||
|
static struct adie_codec_path *debugfs_tx_adie;
|
||
|
|
||
|
static int snddev_icodec_debug_open(struct inode *inode, struct file *file)
|
||
|
{
|
||
|
file->private_data = inode->i_private;
|
||
|
MM_INFO("snddev_icodec: debug intf %s\n", (char *) file->private_data);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void debugfs_adie_loopback(u32 loop)
|
||
|
{
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
|
||
|
if (loop) {
|
||
|
|
||
|
/* enable MI2S RX master block */
|
||
|
/* enable MI2S RX bit clock */
|
||
|
clk_set_rate(drv->rx_mclk,
|
||
|
SNDDEV_ICODEC_CLK_RATE(8000));
|
||
|
clk_prepare_enable(drv->rx_mclk);
|
||
|
clk_prepare_enable(drv->rx_sclk);
|
||
|
|
||
|
MM_INFO("configure ADIE RX path\n");
|
||
|
/* Configure ADIE */
|
||
|
adie_codec_open(&debug_rx_profile, &debugfs_rx_adie);
|
||
|
adie_codec_setpath(debugfs_rx_adie, 8000, 256);
|
||
|
adie_codec_proceed_stage(debugfs_rx_adie,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
|
||
|
MM_INFO("Enable Handset Mic bias\n");
|
||
|
pmic_hsed_enable(PM_HSED_CONTROLLER_0, PM_HSED_ENABLE_PWM_TCXO);
|
||
|
/* enable MI2S TX master block */
|
||
|
/* enable MI2S TX bit clock */
|
||
|
clk_set_rate(drv->tx_mclk,
|
||
|
SNDDEV_ICODEC_CLK_RATE(8000));
|
||
|
clk_prepare_enable(drv->tx_mclk);
|
||
|
clk_prepare_enable(drv->tx_sclk);
|
||
|
|
||
|
MM_INFO("configure ADIE TX path\n");
|
||
|
/* Configure ADIE */
|
||
|
adie_codec_open(&debug_tx_lb_profile, &debugfs_tx_adie);
|
||
|
adie_codec_setpath(debugfs_tx_adie, 8000, 256);
|
||
|
adie_codec_proceed_stage(debugfs_tx_adie,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
} else {
|
||
|
/* Disable ADIE */
|
||
|
adie_codec_proceed_stage(debugfs_rx_adie,
|
||
|
ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(debugfs_rx_adie);
|
||
|
adie_codec_proceed_stage(debugfs_tx_adie,
|
||
|
ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(debugfs_tx_adie);
|
||
|
|
||
|
pmic_hsed_enable(PM_HSED_CONTROLLER_0, PM_HSED_ENABLE_OFF);
|
||
|
|
||
|
/* Disable MI2S RX master block */
|
||
|
/* Disable MI2S RX bit clock */
|
||
|
clk_disable_unprepare(drv->rx_sclk);
|
||
|
clk_disable_unprepare(drv->rx_mclk);
|
||
|
|
||
|
/* Disable MI2S TX master block */
|
||
|
/* Disable MI2S TX bit clock */
|
||
|
clk_disable_unprepare(drv->tx_sclk);
|
||
|
clk_disable_unprepare(drv->tx_mclk);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void debugfs_afe_loopback(u32 loop)
|
||
|
{
|
||
|
int trc;
|
||
|
struct msm_afe_config afe_config;
|
||
|
struct snddev_icodec_drv_state *drv = &snddev_icodec_drv;
|
||
|
struct lpa_codec_config lpa_config;
|
||
|
|
||
|
if (loop) {
|
||
|
/* Vote for SMPS mode*/
|
||
|
pmapp_smps_mode_vote(SMPS_AUDIO_PLAYBACK_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_PWM);
|
||
|
|
||
|
/* enable MI2S RX master block */
|
||
|
/* enable MI2S RX bit clock */
|
||
|
trc = clk_set_rate(drv->rx_mclk,
|
||
|
SNDDEV_ICODEC_CLK_RATE(8000));
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
MM_ERR("failed to set clk rate\n");
|
||
|
clk_prepare_enable(drv->rx_mclk);
|
||
|
clk_prepare_enable(drv->rx_sclk);
|
||
|
clk_prepare_enable(drv->lpa_p_clk);
|
||
|
clk_prepare_enable(drv->lpa_codec_clk);
|
||
|
clk_prepare_enable(drv->lpa_core_clk);
|
||
|
/* Enable LPA sub system
|
||
|
*/
|
||
|
drv->lpa = lpa_get();
|
||
|
if (!drv->lpa)
|
||
|
MM_ERR("failed to enable lpa\n");
|
||
|
lpa_config.sample_rate = 8000;
|
||
|
lpa_config.sample_width = 16;
|
||
|
lpa_config.output_interface = LPA_OUTPUT_INTF_WB_CODEC;
|
||
|
lpa_config.num_channels = 1;
|
||
|
lpa_cmd_codec_config(drv->lpa, &lpa_config);
|
||
|
/* Set audio interconnect reg to LPA */
|
||
|
audio_interct_codec(AUDIO_INTERCT_LPA);
|
||
|
mi2s_set_codec_output_path(MI2S_CHAN_MONO_PACKED, WT_16_BIT);
|
||
|
MM_INFO("configure ADIE RX path\n");
|
||
|
/* Configure ADIE */
|
||
|
adie_codec_open(&debug_rx_profile, &debugfs_rx_adie);
|
||
|
adie_codec_setpath(debugfs_rx_adie, 8000, 256);
|
||
|
lpa_cmd_enable_codec(drv->lpa, 1);
|
||
|
|
||
|
/* Start AFE for RX */
|
||
|
afe_config.sample_rate = 0x8;
|
||
|
afe_config.channel_mode = 1;
|
||
|
afe_config.volume = AFE_VOLUME_UNITY;
|
||
|
MM_INFO("enable afe\n");
|
||
|
trc = afe_enable(AFE_HW_PATH_CODEC_RX, &afe_config);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
MM_ERR("fail to enable afe RX\n");
|
||
|
adie_codec_proceed_stage(debugfs_rx_adie,
|
||
|
ADIE_CODEC_DIGITAL_READY);
|
||
|
adie_codec_proceed_stage(debugfs_rx_adie,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
|
||
|
/* Vote for PWM mode*/
|
||
|
pmapp_smps_mode_vote(SMPS_AUDIO_RECORD_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_PWM);
|
||
|
|
||
|
MM_INFO("Enable Handset Mic bias\n");
|
||
|
pmic_hsed_enable(PM_HSED_CONTROLLER_0, PM_HSED_ENABLE_PWM_TCXO);
|
||
|
|
||
|
/* enable MI2S TX master block */
|
||
|
/* enable MI2S TX bit clock */
|
||
|
clk_set_rate(drv->tx_mclk,
|
||
|
SNDDEV_ICODEC_CLK_RATE(8000));
|
||
|
clk_prepare_enable(drv->tx_mclk);
|
||
|
clk_prepare_enable(drv->tx_sclk);
|
||
|
/* Set MI2S */
|
||
|
mi2s_set_codec_input_path(MI2S_CHAN_MONO_PACKED, WT_16_BIT);
|
||
|
MM_INFO("configure ADIE TX path\n");
|
||
|
/* Configure ADIE */
|
||
|
adie_codec_open(&debug_tx_profile, &debugfs_tx_adie);
|
||
|
adie_codec_setpath(debugfs_tx_adie, 8000, 256);
|
||
|
adie_codec_proceed_stage(debugfs_tx_adie,
|
||
|
ADIE_CODEC_DIGITAL_READY);
|
||
|
adie_codec_proceed_stage(debugfs_tx_adie,
|
||
|
ADIE_CODEC_DIGITAL_ANALOG_READY);
|
||
|
/* Start AFE for TX */
|
||
|
afe_config.sample_rate = 0x8;
|
||
|
afe_config.channel_mode = 1;
|
||
|
afe_config.volume = AFE_VOLUME_UNITY;
|
||
|
trc = afe_enable(AFE_HW_PATH_CODEC_TX, &afe_config);
|
||
|
if (IS_ERR_VALUE(trc))
|
||
|
MM_ERR("failed to enable AFE TX\n");
|
||
|
/* Set the volume level to non unity, to avoid
|
||
|
loopback effect */
|
||
|
afe_device_volume_ctrl(AFE_HW_PATH_CODEC_RX, 0x0500);
|
||
|
|
||
|
/* enable afe loopback */
|
||
|
afe_loopback(1);
|
||
|
MM_INFO("AFE loopback enabled\n");
|
||
|
} else {
|
||
|
/* disable afe loopback */
|
||
|
afe_loopback(0);
|
||
|
/* Remove the vote for SMPS mode*/
|
||
|
pmapp_smps_mode_vote(SMPS_AUDIO_PLAYBACK_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_DONTCARE);
|
||
|
|
||
|
/* Disable ADIE */
|
||
|
adie_codec_proceed_stage(debugfs_rx_adie,
|
||
|
ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(debugfs_rx_adie);
|
||
|
/* Disable AFE for RX */
|
||
|
afe_disable(AFE_HW_PATH_CODEC_RX);
|
||
|
|
||
|
/* Disable LPA Sub system */
|
||
|
lpa_cmd_enable_codec(drv->lpa, 0);
|
||
|
lpa_put(drv->lpa);
|
||
|
|
||
|
/* Disable LPA clocks */
|
||
|
clk_disable_unprepare(drv->lpa_p_clk);
|
||
|
clk_disable_unprepare(drv->lpa_codec_clk);
|
||
|
clk_disable_unprepare(drv->lpa_core_clk);
|
||
|
|
||
|
/* Disable MI2S RX master block */
|
||
|
/* Disable MI2S RX bit clock */
|
||
|
clk_disable_unprepare(drv->rx_sclk);
|
||
|
clk_disable_unprepare(drv->rx_mclk);
|
||
|
|
||
|
pmapp_smps_mode_vote(SMPS_AUDIO_RECORD_ID,
|
||
|
PMAPP_VREG_S4, PMAPP_SMPS_MODE_VOTE_DONTCARE);
|
||
|
|
||
|
/* Disable AFE for TX */
|
||
|
afe_disable(AFE_HW_PATH_CODEC_TX);
|
||
|
|
||
|
/* Disable ADIE */
|
||
|
adie_codec_proceed_stage(debugfs_tx_adie,
|
||
|
ADIE_CODEC_DIGITAL_OFF);
|
||
|
adie_codec_close(debugfs_tx_adie);
|
||
|
/* Disable MI2S TX master block */
|
||
|
/* Disable MI2S TX bit clock */
|
||
|
clk_disable_unprepare(drv->tx_sclk);
|
||
|
clk_disable_unprepare(drv->tx_mclk);
|
||
|
pmic_hsed_enable(PM_HSED_CONTROLLER_0, PM_HSED_ENABLE_OFF);
|
||
|
MM_INFO("AFE loopback disabled\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static ssize_t snddev_icodec_debug_write(struct file *filp,
|
||
|
const char __user *ubuf, size_t cnt, loff_t *ppos)
|
||
|
{
|
||
|
char *lb_str = filp->private_data;
|
||
|
char cmd;
|
||
|
|
||
|
if (get_user(cmd, ubuf))
|
||
|
return -EFAULT;
|
||
|
|
||
|
MM_INFO("%s %c\n", lb_str, cmd);
|
||
|
|
||
|
if (!strcmp(lb_str, "adie_loopback")) {
|
||
|
switch (cmd) {
|
||
|
case '1':
|
||
|
debugfs_adie_loopback(1);
|
||
|
break;
|
||
|
case '0':
|
||
|
debugfs_adie_loopback(0);
|
||
|
break;
|
||
|
}
|
||
|
} else if (!strcmp(lb_str, "afe_loopback")) {
|
||
|
switch (cmd) {
|
||
|
case '1':
|
||
|
debugfs_afe_loopback(1);
|
||
|
break;
|
||
|
case '0':
|
||
|
debugfs_afe_loopback(0);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return cnt;
|
||
|
}
|
||
|
|
||
|
static const struct file_operations snddev_icodec_debug_fops = {
|
||
|
.open = snddev_icodec_debug_open,
|
||
|
.write = snddev_icodec_debug_write
|
||
|
};
|
||
|
#endif
|
||
|
|
||
|
static int __init snddev_icodec_init(void)
|
||
|
{
|
||
|
s32 rc;
|
||
|
struct snddev_icodec_drv_state *icodec_drv = &snddev_icodec_drv;
|
||
|
|
||
|
rc = platform_driver_register(&snddev_icodec_driver);
|
||
|
if (IS_ERR_VALUE(rc))
|
||
|
goto error_platform_driver;
|
||
|
icodec_drv->rx_mclk = clk_get(NULL, "mi2s_codec_rx_m_clk");
|
||
|
if (IS_ERR(icodec_drv->rx_mclk))
|
||
|
goto error_rx_mclk;
|
||
|
icodec_drv->rx_sclk = clk_get(NULL, "mi2s_codec_rx_s_clk");
|
||
|
if (IS_ERR(icodec_drv->rx_sclk))
|
||
|
goto error_rx_sclk;
|
||
|
icodec_drv->tx_mclk = clk_get(NULL, "mi2s_codec_tx_m_clk");
|
||
|
if (IS_ERR(icodec_drv->tx_mclk))
|
||
|
goto error_tx_mclk;
|
||
|
icodec_drv->tx_sclk = clk_get(NULL, "mi2s_codec_tx_s_clk");
|
||
|
if (IS_ERR(icodec_drv->tx_sclk))
|
||
|
goto error_tx_sclk;
|
||
|
icodec_drv->lpa_codec_clk = clk_get(NULL, "lpa_codec_clk");
|
||
|
if (IS_ERR(icodec_drv->lpa_codec_clk))
|
||
|
goto error_lpa_codec_clk;
|
||
|
icodec_drv->lpa_core_clk = clk_get(NULL, "lpa_core_clk");
|
||
|
if (IS_ERR(icodec_drv->lpa_core_clk))
|
||
|
goto error_lpa_core_clk;
|
||
|
icodec_drv->lpa_p_clk = clk_get(NULL, "lpa_pclk");
|
||
|
if (IS_ERR(icodec_drv->lpa_p_clk))
|
||
|
goto error_lpa_p_clk;
|
||
|
|
||
|
#ifdef CONFIG_DEBUG_FS
|
||
|
debugfs_sdev_dent = debugfs_create_dir("snddev_icodec", 0);
|
||
|
if (debugfs_sdev_dent) {
|
||
|
debugfs_afelb = debugfs_create_file("afe_loopback",
|
||
|
S_IFREG | S_IWUGO, debugfs_sdev_dent,
|
||
|
(void *) "afe_loopback", &snddev_icodec_debug_fops);
|
||
|
debugfs_adielb = debugfs_create_file("adie_loopback",
|
||
|
S_IFREG | S_IWUGO, debugfs_sdev_dent,
|
||
|
(void *) "adie_loopback", &snddev_icodec_debug_fops);
|
||
|
}
|
||
|
#endif
|
||
|
mutex_init(&icodec_drv->rx_lock);
|
||
|
mutex_init(&icodec_drv->lb_lock);
|
||
|
mutex_init(&icodec_drv->tx_lock);
|
||
|
icodec_drv->rx_active = 0;
|
||
|
icodec_drv->tx_active = 0;
|
||
|
icodec_drv->lpa = NULL;
|
||
|
pm_qos_add_request(&icodec_drv->tx_pm_qos_req, PM_QOS_CPU_DMA_LATENCY,
|
||
|
PM_QOS_DEFAULT_VALUE);
|
||
|
pm_qos_add_request(&icodec_drv->rx_pm_qos_req, PM_QOS_CPU_DMA_LATENCY,
|
||
|
PM_QOS_DEFAULT_VALUE);
|
||
|
return 0;
|
||
|
|
||
|
error_lpa_p_clk:
|
||
|
clk_put(icodec_drv->lpa_core_clk);
|
||
|
error_lpa_core_clk:
|
||
|
clk_put(icodec_drv->lpa_codec_clk);
|
||
|
error_lpa_codec_clk:
|
||
|
clk_put(icodec_drv->tx_sclk);
|
||
|
error_tx_sclk:
|
||
|
clk_put(icodec_drv->tx_mclk);
|
||
|
error_tx_mclk:
|
||
|
clk_put(icodec_drv->rx_sclk);
|
||
|
error_rx_sclk:
|
||
|
clk_put(icodec_drv->rx_mclk);
|
||
|
error_rx_mclk:
|
||
|
platform_driver_unregister(&snddev_icodec_driver);
|
||
|
error_platform_driver:
|
||
|
|
||
|
MM_ERR("encounter error\n");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
static void __exit snddev_icodec_exit(void)
|
||
|
{
|
||
|
struct snddev_icodec_drv_state *icodec_drv = &snddev_icodec_drv;
|
||
|
|
||
|
#ifdef CONFIG_DEBUG_FS
|
||
|
if (debugfs_afelb)
|
||
|
debugfs_remove(debugfs_afelb);
|
||
|
if (debugfs_adielb)
|
||
|
debugfs_remove(debugfs_adielb);
|
||
|
if (debugfs_sdev_dent)
|
||
|
debugfs_remove(debugfs_sdev_dent);
|
||
|
#endif
|
||
|
platform_driver_unregister(&snddev_icodec_driver);
|
||
|
|
||
|
clk_put(icodec_drv->rx_sclk);
|
||
|
clk_put(icodec_drv->rx_mclk);
|
||
|
clk_put(icodec_drv->tx_sclk);
|
||
|
clk_put(icodec_drv->tx_mclk);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
module_init(snddev_icodec_init);
|
||
|
module_exit(snddev_icodec_exit);
|
||
|
|
||
|
MODULE_DESCRIPTION("ICodec Sound Device driver");
|
||
|
MODULE_VERSION("1.0");
|
||
|
MODULE_LICENSE("GPL v2");
|