Files
android_kernel_samsung_sm86…/asoc/codecs/swr-haptics.c
Fenglin Wu 84d85e958f asoc: swr-haptics: close SWR data channel after stoping playing
Per HW recommendation, when stopping SWR playing, SWR_PLAY register need
to get deasserted before disconnecting SWR data channel. Hence adjust the
sequence.

Change-Id: I15f33fefc8b847c9a7f6abd627217f21f8ddee89
Signed-off-by: Fenglin Wu <fenglinw@codeaurora.org>
2020-05-19 15:50:00 +08:00

552 lines
13 KiB
C

// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2020, The Linux Foundation. All rights reserved.
*/
#include <linux/device.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/regulator/consumer.h>
#include <linux/slab.h>
#include <soc/soundwire.h>
#include <sound/soc.h>
#include <sound/soc-dapm.h>
#define HAPTICS_RATES (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 |\
SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_48000 |\
SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_192000 |\
SNDRV_PCM_RATE_384000)
#define HAPTICS_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\
SNDRV_PCM_FMTBIT_S24_LE |\
SNDRV_PCM_FMTBIT_S24_3LE | SNDRV_PCM_FMTBIT_S32_LE)
/* SWR register definition */
#define SWR_HAP_ACCESS_BASE 0x3000
#define FIFO_WR_READY_REG (SWR_HAP_ACCESS_BASE + 0x8)
#define NUM_PAT_SMPL_REG (SWR_HAP_ACCESS_BASE + 0x9)
#define SWR_WR_ACCESS_REG (SWR_HAP_ACCESS_BASE + 0xa)
#define CAL_TLRA_STATUS_MSB_REG (SWR_HAP_ACCESS_BASE + 0xb)
#define CAL_TLRA_STATUS_LSB_REG (SWR_HAP_ACCESS_BASE + 0xc)
#define AUTO_RES_CAL_DONE_REG (SWR_HAP_ACCESS_BASE + 0xd)
#define SWR_READ_DATA_REG (SWR_HAP_ACCESS_BASE + 0x80)
#define SWR_PLAY_REG (SWR_HAP_ACCESS_BASE + 0x81)
#define SWR_VMAX_REG (SWR_HAP_ACCESS_BASE + 0x82)
#define SWR_PLAY_BIT BIT(7)
#define SWR_BRAKE_EN_BIT BIT(3)
#define SWR_PLAY_SRC_MASK GENMASK(2, 0)
#define SWR_PLAY_SRC_VAL_SWR 4
#define SWR_HAP_REG_MAX (SWR_HAP_ACCESS_BASE + 0xff)
static struct reg_default swr_hap_reg_defaults[] = {
{FIFO_WR_READY_REG, 1},
{NUM_PAT_SMPL_REG, 8},
{SWR_WR_ACCESS_REG, 1},
{CAL_TLRA_STATUS_MSB_REG, 0},
{CAL_TLRA_STATUS_LSB_REG, 0},
{AUTO_RES_CAL_DONE_REG, 0},
{SWR_READ_DATA_REG, 0},
{SWR_PLAY_REG, 4},
{SWR_VMAX_REG, 0},
};
enum {
PORT_ID_DT_IDX,
NUM_CH_DT_IDX,
CH_MASK_DT_IDX,
CH_RATE_DT_IDX,
PORT_TYPE_DT_IDX,
NUM_SWR_PORT_DT_PARAMS,
};
struct swr_port {
u8 port_id;
u8 ch_mask;
u32 ch_rate;
u8 num_ch;
u8 port_type;
};
struct swr_haptics_dev {
struct device *dev;
struct swr_device *swr_slave;
struct snd_soc_component *component;
struct regmap *regmap;
struct swr_port port;
struct regulator *vdd;
};
static bool swr_hap_volatile_register(struct device *dev, unsigned int reg)
{
switch (reg) {
case SWR_READ_DATA_REG:
case SWR_PLAY_REG:
case SWR_VMAX_REG:
return 1;
default:
return 0;
}
}
static bool swr_hap_readable_register(struct device *dev, unsigned int reg)
{
if (reg <= SWR_HAP_ACCESS_BASE)
return 0;
return 1;
}
static bool swr_hap_writeable_register(struct device *dev, unsigned int reg)
{
if (reg <= SWR_HAP_ACCESS_BASE)
return 0;
switch (reg) {
case FIFO_WR_READY_REG:
case NUM_PAT_SMPL_REG:
case SWR_WR_ACCESS_REG:
case CAL_TLRA_STATUS_MSB_REG:
case CAL_TLRA_STATUS_LSB_REG:
case AUTO_RES_CAL_DONE_REG:
case SWR_READ_DATA_REG:
return 0;
}
return 1;
}
struct regmap_config swr_hap_regmap_config = {
.reg_bits = 16,
.val_bits = 8,
.cache_type = REGCACHE_RBTREE,
.reg_defaults = swr_hap_reg_defaults,
.num_reg_defaults = ARRAY_SIZE(swr_hap_reg_defaults),
.max_register = SWR_HAP_REG_MAX,
.volatile_reg = swr_hap_volatile_register,
.readable_reg = swr_hap_readable_register,
.writeable_reg = swr_hap_writeable_register,
.reg_format_endian = REGMAP_ENDIAN_NATIVE,
.val_format_endian = REGMAP_ENDIAN_NATIVE,
.can_multi_write = true,
};
static const struct snd_kcontrol_new hap_swr_dac_port[] = {
SOC_DAPM_SINGLE("Switch", SND_SOC_NOPM, 0, 1, 0)
};
static int hap_enable_swr_dac_port(struct snd_soc_dapm_widget *w,
struct snd_kcontrol *kcontrol, int event)
{
struct snd_soc_component *swr_hap_comp =
snd_soc_dapm_to_component(w->dapm);
struct swr_haptics_dev *swr_hap;
u8 port_id, ch_mask, num_ch, port_type, num_port;
u32 ch_rate;
unsigned int val;
int rc;
if (!swr_hap_comp) {
pr_err("%s: swr_hap_component is NULL\n", __func__);
return -EINVAL;
}
swr_hap = snd_soc_component_get_drvdata(swr_hap_comp);
if (!swr_hap) {
pr_err("%s: get swr_haptics_dev failed\n", __func__);
return -ENODEV;
}
dev_dbg(swr_hap->dev, "%s: %s event %d\n", __func__, w->name, event);
num_port = 1;
port_id = swr_hap->port.port_id;
ch_mask = swr_hap->port.ch_mask;
ch_rate = swr_hap->port.ch_rate;
num_ch = swr_hap->port.num_ch;
port_type = swr_hap->port.port_type;
switch (event) {
case SND_SOC_DAPM_PRE_PMU:
swr_device_wakeup_vote(swr_hap->swr_slave);
swr_connect_port(swr_hap->swr_slave, &port_id, num_port,
&ch_mask, &ch_rate, &num_ch, &port_type);
break;
case SND_SOC_DAPM_POST_PMU:
swr_slvdev_datapath_control(swr_hap->swr_slave,
swr_hap->swr_slave->dev_num, true);
/* trigger SWR play */
val = SWR_PLAY_BIT | SWR_PLAY_SRC_VAL_SWR;
rc = regmap_write(swr_hap->regmap, SWR_PLAY_REG, val);
if (rc) {
dev_err(swr_hap->dev, "%s: Enable SWR_PLAY failed, rc=%d\n",
__func__, rc);
return rc;
}
break;
case SND_SOC_DAPM_PRE_PMD:
/* stop SWR play */
val = 0;
rc = regmap_write(swr_hap->regmap, SWR_PLAY_REG, val);
if (rc) {
dev_err(swr_hap->dev, "%s: Enable SWR_PLAY failed, rc=%d\n",
__func__, rc);
return rc;
}
break;
case SND_SOC_DAPM_POST_PMD:
swr_disconnect_port(swr_hap->swr_slave, &port_id, num_port,
&ch_mask, &port_type);
swr_slvdev_datapath_control(swr_hap->swr_slave,
swr_hap->swr_slave->dev_num, false);
swr_device_wakeup_unvote(swr_hap->swr_slave);
break;
default:
break;
}
return 0;
}
static const struct snd_soc_dapm_widget haptics_comp_dapm_widgets[] = {
SND_SOC_DAPM_INPUT("HAP_IN"),
SND_SOC_DAPM_MIXER_E("SWR DAC_Port", SND_SOC_NOPM, 0, 0,
hap_swr_dac_port, ARRAY_SIZE(hap_swr_dac_port),
hap_enable_swr_dac_port,
SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
SND_SOC_DAPM_SPK("HAP_OUT", NULL),
};
static const struct snd_soc_dapm_route haptics_comp_dapm_route[] = {
{"SWR DAC_Port", "Switch", "HAP_IN"},
{"HAP_OUT", NULL, "SWR DAC_Port"},
};
static int haptics_comp_probe(struct snd_soc_component *component)
{
struct snd_soc_dapm_context *dapm;
struct swr_haptics_dev *swr_hap =
snd_soc_component_get_drvdata(component);
if (!swr_hap) {
pr_err("%s: get swr_haptics_dev failed\n", __func__);
return -EINVAL;
}
snd_soc_component_init_regmap(component, swr_hap->regmap);
dapm = snd_soc_component_get_dapm(component);
if (dapm && dapm->component) {
snd_soc_dapm_ignore_suspend(dapm, "HAP_IN");
snd_soc_dapm_ignore_suspend(dapm, "HAP_OUT");
}
return 0;
}
static void haptics_comp_remove(struct snd_soc_component *component)
{
}
static const struct snd_soc_component_driver swr_haptics_component = {
.name = "swr-haptics",
.probe = haptics_comp_probe,
.remove = haptics_comp_remove,
.dapm_widgets = haptics_comp_dapm_widgets,
.num_dapm_widgets = ARRAY_SIZE(haptics_comp_dapm_widgets),
.dapm_routes = haptics_comp_dapm_route,
.num_dapm_routes = ARRAY_SIZE(haptics_comp_dapm_route),
};
static struct snd_soc_dai_driver haptics_dai[] = {
{
.name = "swr_haptics",
.playback = {
.stream_name = "HAPTICS_AIF Playback",
.rates = HAPTICS_RATES,
.formats = HAPTICS_FORMATS,
.rate_max = 192000,
.rate_min = 8000,
.channels_min = 1,
.channels_max = 1,
},
},
};
static int swr_haptics_parse_port_mapping(struct swr_device *sdev)
{
struct swr_haptics_dev *swr_hap = swr_get_dev_data(sdev);
u32 port_cfg[NUM_SWR_PORT_DT_PARAMS];
int rc;
if (!swr_hap) {
dev_err(&sdev->dev, "%s: get swr_haptics_dev failed\n",
__func__);
return -EINVAL;
}
rc = of_property_read_u32_array(sdev->dev.of_node, "qcom,rx_swr_ch_map",
port_cfg, NUM_SWR_PORT_DT_PARAMS);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: Get qcom,rx_swr_ch_map failed, rc=%d\n",
__func__, rc);
return -EINVAL;
}
swr_hap->port.port_id = (u8) port_cfg[PORT_ID_DT_IDX];
swr_hap->port.num_ch = (u8) port_cfg[NUM_CH_DT_IDX];
swr_hap->port.ch_mask = (u8) port_cfg[CH_MASK_DT_IDX];
swr_hap->port.ch_rate = port_cfg[CH_RATE_DT_IDX];
swr_hap->port.port_type = (u8) port_cfg[PORT_TYPE_DT_IDX];
dev_dbg(swr_hap->dev, "%s: port_id = %d, ch_mask = %d, ch_rate = %d, num_ch = %d, port_type = %d\n",
__func__, swr_hap->port.port_id,
swr_hap->port.ch_mask, swr_hap->port.ch_rate,
swr_hap->port.num_ch, swr_hap->port.port_type);
return 0;
}
static int swr_haptics_probe(struct swr_device *sdev)
{
struct swr_haptics_dev *swr_hap;
int rc;
u8 devnum;
swr_hap = devm_kzalloc(&sdev->dev,
sizeof(struct swr_haptics_dev), GFP_KERNEL);
if (!swr_hap)
return -ENOMEM;
swr_hap->swr_slave = sdev;
swr_hap->dev = &sdev->dev;
swr_set_dev_data(sdev, swr_hap);
rc = swr_haptics_parse_port_mapping(sdev);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: failed to parse swr port mapping, rc=%d\n",
__func__, rc);
goto clean;
}
swr_hap->vdd = devm_regulator_get(swr_hap->dev, "swr-slave");
if (IS_ERR(swr_hap->vdd)) {
rc = PTR_ERR(swr_hap->vdd);
if (rc != -EPROBE_DEFER)
dev_err(swr_hap->dev, "%s: get swr-slave-supply failed, rc=%d\n",
__func__, rc);
goto clean;
}
rc = regulator_enable(swr_hap->vdd);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: enable swr-slave-vdd failed, rc=%d\n",
__func__, rc);
goto clean;
}
rc = swr_get_logical_dev_num(sdev, sdev->addr, &devnum);
if (rc) {
dev_err(swr_hap->dev, "%s: failed to get devnum for swr-haptics, rc=%d\n",
__func__, rc);
rc = -EPROBE_DEFER;
goto clean;
}
sdev->dev_num = devnum;
swr_hap->regmap = devm_regmap_init_swr(sdev, &swr_hap_regmap_config);
if (IS_ERR(swr_hap->regmap)) {
rc = PTR_ERR(swr_hap->regmap);
dev_err(swr_hap->dev, "%s: init regmap failed, rc=%d\n",
__func__, rc);
goto dev_err;
}
rc = snd_soc_register_component(&sdev->dev,
&swr_haptics_component, haptics_dai, ARRAY_SIZE(haptics_dai));
if (rc) {
dev_err(swr_hap->dev, "%s: register swr_haptics component failed, rc=%d\n",
__func__, rc);
goto dev_err;
}
return 0;
dev_err:
regulator_disable(swr_hap->vdd);
swr_remove_device(sdev);
clean:
swr_set_dev_data(sdev, NULL);
return rc;
}
static int swr_haptics_remove(struct swr_device *sdev)
{
struct swr_haptics_dev *swr_hap;
int rc = 0;
swr_hap = swr_get_dev_data(sdev);
if (!swr_hap) {
dev_err(&sdev->dev, "%s: no data for swr_hap\n", __func__);
rc = -ENODEV;
goto clean;
}
rc = regulator_disable(swr_hap->vdd);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: disable swr-slave failed, rc=%d\n",
__func__, rc);
goto clean;
}
clean:
snd_soc_unregister_component(&sdev->dev);
swr_set_dev_data(sdev, NULL);
return rc;
}
static int swr_haptics_device_up(struct swr_device *sdev)
{
struct swr_haptics_dev *swr_hap;
int rc;
swr_hap = swr_get_dev_data(sdev);
if (!swr_hap) {
dev_err(&sdev->dev, "%s: no data for swr_hap\n", __func__);
return -ENODEV;
}
/* Take SWR slave out of reset */
rc = regulator_enable(swr_hap->vdd);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: enable swr-slave failed, rc=%d\n",
__func__, rc);
return rc;
}
return 0;
}
static int swr_haptics_device_down(struct swr_device *sdev)
{
struct swr_haptics_dev *swr_hap = swr_get_dev_data(sdev);
int rc;
unsigned int val;
if (!swr_hap) {
dev_err(&sdev->dev, "%s: no data for swr_hap\n", __func__);
return -ENODEV;
}
/* Stop SWR play in SSR */
val = 0;
rc = regmap_write(swr_hap->regmap, SWR_PLAY_REG, val);
if (rc) {
dev_err(swr_hap->dev, "%s: disable SWR_PLAY failed, rc=%d\n",
__func__, rc);
return rc;
}
/* Put SWR slave into reset */
rc = regulator_disable(swr_hap->vdd);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: disable swr-slave failed, rc=%d\n",
__func__, rc);
return rc;
}
return 0;
}
static int swr_haptics_suspend(struct device *dev)
{
struct swr_haptics_dev *swr_hap;
int rc;
swr_hap = swr_get_dev_data(to_swr_device(dev));
if (!swr_hap) {
dev_err(dev, "%s: no data for swr_hap\n", __func__);
return -ENODEV;
}
/* Put SWR slave into reset */
rc = regulator_disable(swr_hap->vdd);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: disable swr-slave failed, rc=%d\n",
__func__, rc);
return rc;
}
return 0;
}
static int swr_haptics_resume(struct device *dev)
{
struct swr_haptics_dev *swr_hap;
int rc;
swr_hap = swr_get_dev_data(to_swr_device(dev));
if (!swr_hap) {
dev_err(dev, "%s: no data for swr_hap\n", __func__);
return -ENODEV;
}
/* Take SWR slave out of reset */
rc = regulator_enable(swr_hap->vdd);
if (rc < 0) {
dev_err(swr_hap->dev, "%s: enable swr-slave failed, rc=%d\n",
__func__, rc);
return rc;
}
return 0;
}
static const struct of_device_id swr_haptics_match_table[] = {
{ .compatible = "qcom,swr-haptics", },
{ .compatible = "qcom,pm8350b-swr-haptics", },
{ },
};
static const struct swr_device_id swr_haptics_id[] = {
{"swr-haptics", 0},
{"pm8350b-swr-haptics", 0},
{},
};
static const struct dev_pm_ops swr_haptics_pm_ops = {
.suspend = swr_haptics_suspend,
.resume = swr_haptics_resume,
};
static struct swr_driver swr_haptics_driver = {
.driver = {
.name = "swr-haptics",
.owner = THIS_MODULE,
.pm = &swr_haptics_pm_ops,
.of_match_table = swr_haptics_match_table,
},
.probe = swr_haptics_probe,
.remove = swr_haptics_remove,
.id_table = swr_haptics_id,
.device_up = swr_haptics_device_up,
.device_down = swr_haptics_device_down,
};
static int __init swr_haptics_init(void)
{
return swr_driver_register(&swr_haptics_driver);
}
static void __exit swr_haptics_exit(void)
{
swr_driver_unregister(&swr_haptics_driver);
}
module_init(swr_haptics_init);
module_exit(swr_haptics_exit);
MODULE_DESCRIPTION("SWR haptics driver");
MODULE_LICENSE("GPL v2");