kernel_samsung_a34x-permissive/sound/soc/codecs/aw882xx/aw882xx.c
2024-04-28 15:49:01 +02:00

2350 lines
59 KiB
C
Executable file

/*
* aw882xx.c aw882xx codec module
*
*
* Copyright (c) 2020 AWINIC Technology CO., LTD
*
* Author: Nick Li <liweilei@awinic.com.cn>
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at your
* option) any later version.
*/
/*#define DEBUG*/
#include <linux/module.h>
#include <linux/i2c.h>
#include <sound/core.h>
#include <sound/pcm.h>
#include <sound/pcm_params.h>
#include <sound/soc.h>
#include <linux/of_gpio.h>
#include <linux/delay.h>
#include <linux/device.h>
#include <linux/firmware.h>
#include <linux/debugfs.h>
#include <linux/version.h>
#include <linux/workqueue.h>
#include <linux/syscalls.h>
#include <sound/control.h>
#include <linux/uaccess.h>
#include <linux/vmalloc.h>
#if IS_ENABLED(CONFIG_SEC_ABC)
#include <linux/sti/abc_common.h>
#endif
#include "aw882xx.h"
#include "aw_log.h"
#include "aw_dsp.h"
#define AW882XX_DRIVER_VERSION "v1.7.0.4_MTK"
#define AW882XX_I2C_NAME "aw882xx_smartpa"
#define AW_READ_CHIPID_RETRIES 5 /* 5 times */
#define AW_READ_CHIPID_RETRY_DELAY 5 /* 5 ms */
static unsigned int g_aw882xx_dev_cnt = 0;
static unsigned int g_print_dbg = 0;
static unsigned int g_algo_rx_en = false;
static unsigned int g_algo_tx_en = false;
static unsigned int g_algo_copp_en = false;
#ifdef AW_SPIN_ENABLE
static unsigned int g_spin_value = 0;
#endif
static DEFINE_MUTEX(g_aw882xx_lock);
struct aw_container *g_awinic_cfg = NULL;
static const char *const aw882xx_switch[] = {"Disable", "Enable"};
#ifdef AW_SPIN_ENABLE
static const char *const aw882xx_spin[] = {"spin_0", "spin_90",
"spin_180", "spin_270"};
#endif
/******************************************************
*
* aw882xx distinguish between codecs and components by version
*
******************************************************/
#ifdef AW_KERNEL_VER_OVER_4_19_1
static struct aw_componet_codec_ops aw_componet_codec_ops = {
.kcontrol_codec = snd_soc_kcontrol_component,
.codec_get_drvdata = snd_soc_component_get_drvdata,
.add_codec_controls = snd_soc_add_component_controls,
.unregister_codec = snd_soc_unregister_component,
.register_codec = snd_soc_register_component,
};
#else
static struct aw_componet_codec_ops aw_componet_codec_ops = {
.kcontrol_codec = snd_soc_kcontrol_codec,
.codec_get_drvdata = snd_soc_codec_get_drvdata,
.add_codec_controls = snd_soc_add_codec_controls,
.unregister_codec = snd_soc_unregister_codec,
.register_codec = snd_soc_register_codec,
};
#endif
static aw_snd_soc_codec_t *aw_get_codec(struct snd_soc_dai *dai)
{
#ifdef AW_KERNEL_VER_OVER_4_19_1
return dai->component;
#else
return dai->codec;
#endif
}
/******************************************************
*
* aw882xx i2c err report
*
******************************************************/
void aw_i2c_err_report(struct aw882xx *aw882xx, int err_type)
{
aw_dev_err(aw882xx->dev, "aw882xx i2c rw error, err_type = %d", err_type);
//custom add API to get err info
#if IS_ENABLED(CONFIG_SEC_ABC)
sec_abc_send_event("MODULE=audio@ERROR=spk_amp");
#endif
}
/******************************************************
*
* aw882xx i2c write/read
*
******************************************************/
int aw882xx_get_version(char *buf, int size)
{
if (size > strlen(AW882XX_DRIVER_VERSION)) {
memcpy(buf, AW882XX_DRIVER_VERSION, strlen(AW882XX_DRIVER_VERSION));
return strlen(AW882XX_DRIVER_VERSION);
} else {
return -ENOMEM;
}
}
int aw882xx_get_dev_num(void) {
return g_aw882xx_dev_cnt;
}
static int aw882xx_i2c_writes(struct aw882xx *aw882xx,
unsigned char reg_addr, unsigned char *buf, unsigned int len)
{
int ret = -1;
unsigned char *data = NULL;
data = kmalloc(len+1, GFP_KERNEL);
if (data == NULL) {
aw_dev_err(aw882xx->dev, "can not allocate memory");
return -ENOMEM;
}
data[0] = reg_addr;
memcpy(&data[1], buf, len);
ret = i2c_master_send(aw882xx->i2c, data, len+1);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "i2c master send error");
aw_i2c_err_report(aw882xx, ret);
}
kfree(data);
return ret;
}
static int aw882xx_i2c_reads(struct aw882xx *aw882xx,
unsigned char reg_addr, unsigned char *data_buf, unsigned int data_len)
{
int ret;
struct i2c_msg msg[] = {
[0] = {
.addr = aw882xx->i2c->addr,
.flags = 0,
.len = sizeof(uint8_t),
.buf = &reg_addr,
},
[1] = {
.addr = aw882xx->i2c->addr,
.flags = I2C_M_RD,
.len = data_len,
.buf = data_buf,
},
};
ret = i2c_transfer(aw882xx->i2c->adapter, msg, ARRAY_SIZE(msg));
if (ret < 0) {
aw_dev_err(aw882xx->dev, "transfer failed.");
aw_i2c_err_report(aw882xx, ret);
return ret;
} else if (ret != AW882XX_I2C_READ_MSG_NUM) {
aw_dev_err(aw882xx->dev, "transfer failed(size error).");
aw_i2c_err_report(aw882xx, ret);
return -ENXIO;
}
return 0;
}
int aw882xx_i2c_write(struct aw882xx *aw882xx,
unsigned char reg_addr, unsigned int reg_data)
{
int ret = -1;
unsigned char cnt = 0;
unsigned char buf[2];
buf[0] = (reg_data&0xff00)>>8;
buf[1] = (reg_data&0x00ff)>>0;
while (cnt < AW_I2C_RETRIES) {
ret = aw882xx_i2c_writes(aw882xx, reg_addr, buf, 2);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "i2c_write cnt=%d error=%d",
cnt, ret);
} else {
if (g_print_dbg)
aw_dev_info(aw882xx->dev, "reg_addr: 0x%04x, reg_data :0x%04x",
(uint16_t)reg_addr, (uint16_t)reg_data);
break;
}
cnt++;
}
return ret;
}
int aw882xx_i2c_read(struct aw882xx *aw882xx,
unsigned char reg_addr, unsigned int *reg_data)
{
int ret = -1;
unsigned char cnt = 0;
unsigned char buf[2];
while (cnt < AW_I2C_RETRIES) {
ret = aw882xx_i2c_reads(aw882xx, reg_addr, buf, 2);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "i2c_read cnt=%d error=%d",
cnt, ret);
} else {
*reg_data = (buf[0]<<8) | (buf[1]<<0);
if (g_print_dbg)
aw_dev_info(aw882xx->dev, "reg_addr: 0x%04x, reg_data :0x%04x",
(uint16_t)reg_addr, (uint16_t)(*reg_data));
break;
}
cnt++;
}
return ret;
}
int aw882xx_i2c_write_bits(struct aw882xx *aw882xx,
unsigned char reg_addr, unsigned int mask, unsigned int reg_data)
{
int ret = -1;
unsigned int reg_val = 0;
ret = aw882xx_i2c_read(aw882xx, reg_addr, &reg_val);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "i2c read error, ret=%d", ret);
return ret;
}
reg_val &= mask;
reg_val |= reg_data;
ret = aw882xx_i2c_write(aw882xx, reg_addr, reg_val);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "i2c read error, ret=%d", ret);
return ret;
}
return 0;
}
static void *aw882xx_devm_kstrdup(struct device *dev, char *buf)
{
char *str = NULL;
str = devm_kzalloc(dev, strlen(buf) + 1, GFP_KERNEL);
if (!str)
return str;
memcpy(str, buf, strlen(buf));
return str;
}
/*****************************************************
*
* snd_soc_dai_driver ops
*
*****************************************************/
static int aw882xx_startup(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai)
{
aw_snd_soc_codec_t *codec = aw_get_codec(dai);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
aw_dev_info(aw882xx->dev, "playback enter");
/*load cali re*/
aw_dev_init_cali_re(aw882xx->aw_pa);
} else {
aw_dev_info(aw882xx->dev, "capture enter");
}
return 0;
}
static int aw882xx_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
{
/*struct aw882xx *aw882xx = aw_snd_soc_codec_get_drvdata(dai->codec);*/
aw_snd_soc_codec_t *codec = aw_get_codec(dai);
aw_dev_info(codec->dev, "fmt=0x%x", fmt);
/* Supported mode: regular I2S, slave, or PDM */
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
case SND_SOC_DAIFMT_I2S:
if ((fmt & SND_SOC_DAIFMT_MASTER_MASK) !=
SND_SOC_DAIFMT_CBS_CFS) {
aw_dev_err(codec->dev, "invalid codec master mode");
return -EINVAL;
}
break;
default:
aw_dev_err(codec->dev, "unsupported DAI format %d",
fmt & SND_SOC_DAIFMT_FORMAT_MASK);
return -EINVAL;
}
return 0;
}
static int aw882xx_set_dai_sysclk(struct snd_soc_dai *dai,
int clk_id, unsigned int freq, int dir)
{
aw_snd_soc_codec_t *codec = aw_get_codec(dai);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev_info(aw882xx->dev, "freq=%d", freq);
aw882xx->sysclk = freq;
return 0;
}
static int aw882xx_hw_params(struct snd_pcm_substream *substream,
struct snd_pcm_hw_params *params,
struct snd_soc_dai *dai)
{
aw_snd_soc_codec_t *codec = aw_get_codec(dai);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) {
aw_dev_dbg(aw882xx->dev, "stream capture requested rate: %d, sample size: %d",
params_rate(params), params_width(params));
}
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
aw_dev_dbg(aw882xx->dev, "stream playback requested rate: %d, sample size: %d",
params_rate(params), params_width(params));
}
return 0;
}
static void aw882xx_shutdown(struct snd_pcm_substream *substream,
struct snd_soc_dai *dai)
{
aw_snd_soc_codec_t *codec = aw_get_codec(dai);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
aw882xx->rate = 0;
aw_dev_info(aw882xx->dev, "stream playback");
} else {
aw_dev_info(aw882xx->dev, "stream capture");
}
}
static void aw882xx_start_pa(struct aw882xx *aw882xx)
{
int ret;
int i;
aw_dev_info(aw882xx->dev, "enter");
if (aw882xx->fw_status == AW_DEV_FW_OK) {
if (aw882xx->allow_pw == false) {
aw_dev_info(aw882xx->dev, "dev can not allow power ");
return;
}
for (i = 0; i < AW_START_RETRIES; i++) {
/*if PA already power ,stop PA then start*/
if (aw882xx->aw_pa->status) {
aw_dev_info(aw882xx->dev, "already start");
return;
}
ret = aw_dev_reg_update(aw882xx->aw_pa, aw882xx->phase_sync);
if (ret) {
aw_dev_err(aw882xx->dev, "fw update failed, cnt:%d", i);
continue;
}
ret = aw_device_start(aw882xx->aw_pa);
if (ret) {
aw_dev_err(aw882xx->dev, "start failed, cnt:%d", i);
continue;
} else {
if (aw882xx->dc_flag)
queue_delayed_work(aw882xx->work_queue,
&aw882xx->dc_work,
msecs_to_jiffies(AW882XX_DC_DELAY_TIME));
aw_dev_info(aw882xx->dev, "start success");
break;
}
}
} else {
aw_dev_info(aw882xx->dev, "dev acf load failed");
}
}
static int aw882xx_mute(struct snd_soc_dai *dai, int mute, int stream)
{
aw_snd_soc_codec_t *codec = aw_get_codec(dai);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
int ret = 0;
aw_dev_info(aw882xx->dev, "mute state=%d", mute);
if (stream != SNDRV_PCM_STREAM_PLAYBACK) {
aw_dev_info(aw882xx->dev, "capture");
return 0;
}
if (mute) {
aw882xx->pstream = false;
cancel_delayed_work_sync(&aw882xx->dc_work);
cancel_delayed_work_sync(&aw882xx->start_work);
mutex_lock(&aw882xx->lock);
aw_device_stop(aw882xx->aw_pa);
mutex_unlock(&aw882xx->lock);
} else {
if (aw882xx->fw_status == AW_DEV_FW_FAILED) {
aw_dev_info(aw882xx->dev, "fw_load failed ,can not start PA");
return 0;
}
aw882xx->pstream = true;
mutex_lock(&aw882xx->lock);
/*aw882xx_start_pa(aw882xx);*/
queue_delayed_work(aw882xx->work_queue,
&aw882xx->start_work, 0);
#ifdef AW_SPIN_ENABLE
if (aw882xx->index == 0) {
ret = aw_dev_set_spin(g_spin_value);
if (ret)
aw_dev_err(aw882xx->dev, "set spin error, ret=%d", ret);
}
#endif
mutex_unlock(&aw882xx->lock);
}
return ret;
}
static const struct snd_soc_dai_ops aw882xx_dai_ops = {
.startup = aw882xx_startup,
.set_fmt = aw882xx_set_fmt,
.set_sysclk = aw882xx_set_dai_sysclk,
.hw_params = aw882xx_hw_params,
.mute_stream = aw882xx_mute,
.shutdown = aw882xx_shutdown,
};
/*****************************************************
*
* snd_soc_codec_driver | snd_soc_component_driver|
*
*****************************************************/
static int aw882xx_profile_info(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_info *uinfo)
{
int count, ret;
char *name = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
uinfo->count = 1;
count = aw_dev_get_profile_count(aw882xx->aw_pa);
if (count <= 0) {
uinfo->value.enumerated.items = 0;
aw_dev_err(aw882xx->dev, "get count[%d] failed ", count);
return 0;
}
uinfo->value.enumerated.items = count;
if (uinfo->value.enumerated.item >= count)
uinfo->value.enumerated.item = count - 1;
name = uinfo->value.enumerated.name;
count = uinfo->value.enumerated.item;
ret = aw_dev_get_profile_name(aw882xx->aw_pa, name, count);
if (ret) {
strlcpy(uinfo->value.enumerated.name, "null", strlen("null") + 1);
return 0;
}
return 0;
}
static int aw882xx_profile_get(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
ucontrol->value.integer.value[0] = aw_dev_get_profile_index(aw882xx->aw_pa);
aw_dev_dbg(codec->dev, "profile index [%d]", aw_dev_get_profile_index(aw882xx->aw_pa));
return 0;
}
static int aw882xx_profile_set(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
int ret;
int cur_index;
if (aw882xx->dbg_en_prof == false) {
aw_dev_info(codec->dev, "profile close ");
return 0;
}
/* check value valid */
ret = aw_dev_check_profile_index(aw882xx->aw_pa, ucontrol->value.integer.value[0]);
if (ret) {
aw_dev_info(codec->dev, "unsupported index %d",
(int)ucontrol->value.integer.value[0]);
return 0;
}
/*check cur_index == set value*/
cur_index = aw_dev_get_profile_index(aw882xx->aw_pa);
if (cur_index == ucontrol->value.integer.value[0]) {
aw_dev_info(codec->dev, "index no change");
return 0;
}
mutex_lock(&aw882xx->lock);
aw_dev_set_profile_index(aw882xx->aw_pa, ucontrol->value.integer.value[0]);
/*pstream = 0 no pcm just set status*/
if (aw882xx->pstream && aw882xx->allow_pw)
aw_dev_prof_update(aw882xx->aw_pa, aw882xx->phase_sync);
mutex_unlock(&aw882xx->lock);
aw_dev_info(codec->dev, "prof id %d", (int)ucontrol->value.integer.value[0]);
return 0;
}
static int aw882xx_switch_info(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_info *uinfo)
{
int count;
uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
uinfo->count = 1;
count = 2;
uinfo->value.enumerated.items = count;
if (uinfo->value.enumerated.item >= count)
uinfo->value.enumerated.item = count - 1;
strlcpy(uinfo->value.enumerated.name,
aw882xx_switch[uinfo->value.enumerated.item],
strlen(aw882xx_switch[uinfo->value.enumerated.item]) + 1);
return 0;
}
static int aw882xx_switch_get(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
ucontrol->value.integer.value[0] = aw882xx->allow_pw;
return 0;
}
static int aw882xx_switch_set(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
if (aw882xx->pstream) {
if (ucontrol->value.integer.value[0] == 0) {
cancel_delayed_work_sync(&aw882xx->dc_work);
cancel_delayed_work_sync(&aw882xx->start_work);
mutex_lock(&aw882xx->lock);
aw_device_stop(aw882xx->aw_pa);
aw882xx->allow_pw = false;
mutex_unlock(&aw882xx->lock);
aw_dev_info(aw882xx->dev, "stop pa");
} else {
cancel_delayed_work_sync(&aw882xx->start_work);
mutex_lock(&aw882xx->lock);
aw882xx->allow_pw = true;
if (aw882xx->fw_status == AW_DEV_FW_OK)
aw882xx_start_pa(aw882xx);
else
aw_dev_info(aw882xx->dev, "fw_load failed ,can not start PA");
mutex_unlock(&aw882xx->lock);
}
} else {
mutex_lock(&aw882xx->lock);
if (ucontrol->value.integer.value[0])
aw882xx->allow_pw = true;
else
aw882xx->allow_pw = false;
mutex_unlock(&aw882xx->lock);
}
return 0;
}
static int aw882xx_dynamic_create_controls(struct aw882xx *aw882xx)
{
struct snd_kcontrol_new *aw882xx_dev_control = NULL;
char *kctl_name = NULL;
aw882xx_dev_control = devm_kzalloc(aw882xx->codec->dev, sizeof(struct snd_kcontrol_new) * 2, GFP_KERNEL);
if (aw882xx_dev_control == NULL) {
aw_dev_err(aw882xx->codec->dev, "kcontrol malloc failed!");
return -ENOMEM;
}
kctl_name = devm_kzalloc(aw882xx->codec->dev, AW_NAME_BUF_MAX, GFP_KERNEL);
if (!kctl_name)
return -ENOMEM;
snprintf(kctl_name, AW_NAME_BUF_MAX, "aw_dev_%d_prof", aw882xx->index);
aw882xx_dev_control[0].name = kctl_name;
aw882xx_dev_control[0].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
aw882xx_dev_control[0].info = aw882xx_profile_info;
aw882xx_dev_control[0].get = aw882xx_profile_get;
aw882xx_dev_control[0].put = aw882xx_profile_set;
kctl_name = devm_kzalloc(aw882xx->codec->dev, AW_NAME_BUF_MAX, GFP_KERNEL);
if (!kctl_name)
return -ENOMEM;
snprintf(kctl_name, AW_NAME_BUF_MAX, "aw_dev_%d_switch", aw882xx->index);
aw882xx_dev_control[1].name = kctl_name;
aw882xx_dev_control[1].iface = SNDRV_CTL_ELEM_IFACE_MIXER;
aw882xx_dev_control[1].info = aw882xx_switch_info;
aw882xx_dev_control[1].get = aw882xx_switch_get;
aw882xx_dev_control[1].put = aw882xx_switch_set;
aw_componet_codec_ops.add_codec_controls(aw882xx->codec,
aw882xx_dev_control, 2);
return 0;
}
static void aw882xx_request_firmware(struct work_struct *work)
{
struct aw882xx *aw882xx =
container_of(work, struct aw882xx, fw_work.work);
const struct firmware *cont = NULL;
struct aw_container *aw_cfg = NULL;
int ret = -1;
aw882xx->fw_status = AW_DEV_FW_FAILED;
ret = request_firmware(&cont, aw882xx->aw_pa->acf_name, aw882xx->dev);
if ((ret) || (!cont)) {
aw_dev_info(aw882xx->dev, "load [%s] failed!", aw882xx->aw_pa->acf_name);
if (aw882xx->fw_retry_cnt == AW_READ_CHIPID_RETRIES) {
aw882xx->fw_retry_cnt = 0;
} else {
aw882xx->fw_retry_cnt++;
/* sleep 1s */
msleep(1000);
aw_dev_info(aw882xx->dev, "load [%s] try [%d]!",
aw882xx->aw_pa->acf_name, aw882xx->fw_retry_cnt);
aw882xx_request_firmware(work);
}
return;
}
aw_dev_info(aw882xx->dev, "load [%s] , file size: [%zu]",
aw882xx->aw_pa->acf_name, cont ? cont->size : 0);
mutex_lock(&g_aw882xx_lock);
if (g_awinic_cfg == NULL) {
aw_cfg = vzalloc(cont->size + sizeof(int));
if (aw_cfg == NULL) {
release_firmware(cont);
aw_dev_err(aw882xx->dev, "malloc failed");
mutex_unlock(&g_aw882xx_lock);
return;
}
aw_cfg->len = cont->size;
memcpy(aw_cfg->data, cont->data, cont->size);
release_firmware(cont);
ret = aw_dev_load_acf_check(aw_cfg);
if (ret) {
aw_dev_err(aw882xx->dev, "Load [%s] failed ....!", aw882xx->aw_pa->acf_name);
vfree(aw_cfg);
aw_cfg = NULL;
mutex_unlock(&g_aw882xx_lock);
return;
}
g_awinic_cfg = aw_cfg;
} else {
aw_cfg = g_awinic_cfg;
release_firmware(cont);
aw_dev_info(aw882xx->dev, "[%s] already loaded...", aw882xx->aw_pa->acf_name);
}
mutex_unlock(&g_aw882xx_lock);
mutex_lock(&aw882xx->lock);
/*aw device init*/
ret = aw_device_init(aw882xx->aw_pa, aw_cfg);
if (ret < 0) {
aw_dev_info(aw882xx->dev, "dev init failed");
mutex_unlock(&aw882xx->lock);
return;
}
/*create kcontrol by profile*/
aw882xx_dynamic_create_controls(aw882xx);
aw882xx->fw_status = AW_DEV_FW_OK;
aw882xx->fw_retry_cnt = 0;
mutex_unlock(&aw882xx->lock);
}
static void aw882xx_startup_work(struct work_struct *work)
{
struct aw882xx *aw882xx = container_of(work, struct aw882xx, start_work.work);
aw_dev_info(aw882xx->dev, "enter");
mutex_lock(&aw882xx->lock);
aw882xx_start_pa(aw882xx);
mutex_unlock(&aw882xx->lock);
}
static void aw882xx_dc_prot_work(struct work_struct *work)
{
struct aw882xx *aw882xx = container_of(work, struct aw882xx, dc_work.work);
int dc_status = -1;
int dev_status = aw_dev_status(aw882xx->aw_pa);
if (aw882xx->dc_flag) {
if (dev_status) {
dc_status = aw_dev_dc_status(aw882xx->aw_pa);
if (dc_status > 0) {
cancel_delayed_work_sync(&aw882xx->start_work);
mutex_lock(&aw882xx->lock);
aw_device_stop(aw882xx->aw_pa);
mutex_unlock(&aw882xx->lock);
} else {
queue_delayed_work(aw882xx->work_queue,
&aw882xx->dc_work,
msecs_to_jiffies(AW882XX_DC_DELAY_TIME));
}
}
}
}
static void aw882xx_irq_restart(struct aw882xx *aw882xx)
{
int ret;
aw_dev_dbg(aw882xx->dev, "enter");
mutex_lock(&aw882xx->lock);
/*stop pa*/
aw_device_stop(aw882xx->aw_pa);
/*hw reset*/
aw882xx_hw_reset(aw882xx);
/*aw reinit*/
if (aw882xx->fw_status == AW_DEV_FW_OK) {
ret = aw_device_irq_reinit(aw882xx->aw_pa);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "irq reinit failed");
goto failed_exit;
}
if (aw882xx->allow_pw && aw882xx->pstream) {
ret = aw_device_start(aw882xx->aw_pa);
if (ret) {
aw_dev_err(aw882xx->dev, "start failed");
goto failed_exit;
}
} else {
aw_dev_info(aw882xx->dev, "allow_pw [%d] ,pstream[%d], not start",
aw882xx->allow_pw, aw882xx->pstream);
}
} else {
aw_dev_err(aw882xx->dev, "fw not load ,cannot init device");
}
failed_exit:
mutex_unlock(&aw882xx->lock);
}
static void aw882xx_interrupt_work(struct work_struct *work)
{
struct aw882xx *aw882xx = container_of(work, struct aw882xx, interrupt_work.work);
int16_t reg_value;
int ret;
aw_dev_info(aw882xx->dev, "enter");
/*read reg value*/
ret = aw_dev_get_int_status(aw882xx->aw_pa, &reg_value);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "get init_reg value failed");
} else {
aw_dev_info(aw882xx->dev, "int value 0x%x", reg_value);
if (aw882xx->aw_pa->ops.aw_get_irq_type) {
ret = aw882xx->aw_pa->ops.aw_get_irq_type(aw882xx->aw_pa, reg_value);
if (ret != INT_TYPE_NONE) {
aw882xx_irq_restart(aw882xx);
return;
}
}
}
/*clear init reg*/
aw_dev_clear_int_status(aw882xx->aw_pa);
/*unmask interrupt*/
aw_dev_set_intmask(aw882xx->aw_pa, true);
}
static int aw882xx_set_rx_en(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
int ret = -EINVAL;
uint32_t ctrl_value = 0;
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev_dbg(aw882xx->dev, "ucontrol->value.integer.value[0]=%ld",
ucontrol->value.integer.value[0]);
aw_dev = aw882xx->aw_pa;
ctrl_value = ucontrol->value.integer.value[0];
if (aw882xx->pstream) {
ret = aw_dev_set_afe_module_en(AW_RX_MODULE, ctrl_value);
if (ret)
aw_dev_err(aw882xx->dev, "dsp_msg error, ret=%d", ret);
} else {
aw_dev_info(aw882xx->dev, "stream no start only record");
}
g_algo_rx_en = ctrl_value;
aw_dev_info(aw882xx->dev, "set value %d", ctrl_value);
return 0;
}
static int aw882xx_get_rx_en(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
int ret = -EINVAL;
uint32_t ctrl_value = 0;
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev = aw882xx->aw_pa;
if (aw882xx->pstream) {
ret = aw_dev_get_afe_module_en(AW_RX_MODULE, &ctrl_value);
if (ret)
aw_dev_err(aw882xx->dev, "dsp_msg error, ret=%d", ret);
ucontrol->value.integer.value[0] = ctrl_value;
} else {
ucontrol->value.integer.value[0] = g_algo_rx_en;
aw_dev_info(aw882xx->dev, "no stream, use record value");
}
aw_dev_dbg(aw882xx->dev, "aw882xx_rx_enable %ld",
ucontrol->value.integer.value[0]);
return 0;
}
static int aw882xx_set_tx_en(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
int ret = -EINVAL;
uint32_t ctrl_value = 0;
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev_dbg(aw882xx->dev, "ucontrol->value.integer.value[0]=%ld",
ucontrol->value.integer.value[0]);
aw_dev = aw882xx->aw_pa;
ctrl_value = ucontrol->value.integer.value[0];
if (aw882xx->pstream) {
ret = aw_dev_set_afe_module_en(AW_TX_MODULE, ctrl_value);
if (ret)
aw_dev_err(aw882xx->dev, "dsp_msg error, ret=%d", ret);
} else {
aw_dev_info(aw882xx->dev, "stream no start only record");
}
g_algo_tx_en = ctrl_value;
aw_dev_info(aw882xx->dev, "set value %d", ctrl_value);
return 0;
}
static int aw882xx_get_tx_en(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
int ret = -EINVAL;
uint32_t ctrl_value = 0;
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev = aw882xx->aw_pa;
if (aw882xx->pstream) {
ret = aw_dev_get_afe_module_en(AW_TX_MODULE, &ctrl_value);
if (ret) {
aw_dev_err(aw882xx->dev, "dsp_msg error, ret=%d", ret);
ctrl_value = 0;
}
ucontrol->value.integer.value[0] = ctrl_value;
} else {
ucontrol->value.integer.value[0] = g_algo_tx_en;
aw_dev_info(aw882xx->dev, "no stream, use record value");
}
aw_dev_dbg(aw882xx->dev, "aw882xx_tx_enable %ld",
ucontrol->value.integer.value[0]);
return 0;
}
static int aw882xx_set_copp_en(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
int ret = -EINVAL;
uint32_t ctrl_value = 0;
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev_dbg(aw882xx->dev, "ucontrol->value.integer.value[0]=%ld",
ucontrol->value.integer.value[0]);
aw_dev = aw882xx->aw_pa;
ctrl_value = ucontrol->value.integer.value[0];
if (aw882xx->pstream) {
ret = aw_dev_set_copp_module_en(ctrl_value);
if (ret)
aw_dev_err(aw882xx->dev, "dsp_msg error, ret=%d", ret);
} else {
aw_dev_info(aw882xx->dev, "stream no start only record");
}
g_algo_copp_en = ctrl_value;
aw_dev_info(aw882xx->dev, "set value %d", ctrl_value);
return 0;
}
static int aw882xx_get_copp_en(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
ucontrol->value.integer.value[0] = g_algo_copp_en;
aw_dev_dbg(aw882xx->dev, "done nothing");
return 0;
}
void aw882xx_kcontorl_set(struct aw882xx *aw882xx)
{
int ret;
ret = aw_dev_set_afe_module_en(AW_RX_MODULE, g_algo_rx_en);
if (ret)
aw_dev_err(aw882xx->dev, "afe set, ret=%d", ret);
ret = aw_dev_set_copp_module_en(g_algo_copp_en);
if (ret)
aw_dev_err(aw882xx->dev, "copp set error, ret=%d", ret);
}
#ifdef AW_SPIN_ENABLE
static int aw882xx_set_spin(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
int ret = -EINVAL;
uint32_t ctrl_value = 0;
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
aw_dev_dbg(aw882xx->dev, "ucontrol->value.integer.value[0]=%ld",
ucontrol->value.integer.value[0]);
aw_dev = aw882xx->aw_pa;
ctrl_value = ucontrol->value.integer.value[0];
if (aw882xx->pstream) {
ret = aw_dev_set_spin(ctrl_value);
if (ret)
aw_dev_err(aw882xx->dev, "set spin error, ret=%d", ret);
} else {
aw_dev_info(aw882xx->dev, "stream no start only record");
}
g_spin_value = ctrl_value;
return 0;
}
static int aw882xx_get_spin(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct aw_device *aw_dev = NULL;
aw_snd_soc_codec_t *codec =
aw_componet_codec_ops.kcontrol_codec(kcontrol);
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(codec);
int ctrl_value;
int ret = -EINVAL;
aw_dev = aw882xx->aw_pa;
if (aw882xx->pstream) {
ret = aw_dev_get_spin(&ctrl_value);
if (ret) {
aw_dev_err(aw882xx->dev, "get spin failed!, ret = %d", ret);
ctrl_value = 0;
}
ucontrol->value.integer.value[0] = ctrl_value;
} else {
ucontrol->value.integer.value[0] = g_spin_value;
aw_dev_info(aw882xx->dev, "no stream, use record value");
}
aw_dev_dbg(aw882xx->dev, "spin value is %ld", ucontrol->value.integer.value[0]);
return 0;
}
#endif
static int aw882xx_get_fade_in_time(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
unsigned int time;
aw_dev_get_fade_time(&time, true);
ucontrol->value.integer.value[0] = time;
aw_pr_dbg("step time %ld", ucontrol->value.integer.value[0]);
return 0;
}
static int aw882xx_set_fade_in_time(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct soc_mixer_control *mc =
(struct soc_mixer_control *)kcontrol->private_value;
if (ucontrol->value.integer.value[0] > mc->max) {
aw_pr_dbg("set val %ld overflow %d",
ucontrol->value.integer.value[0], mc->max);
return 0;
}
aw_dev_set_fade_time(ucontrol->value.integer.value[0], true);
aw_pr_dbg("step time %ld", ucontrol->value.integer.value[0]);
return 0;
}
static int aw882xx_get_fade_out_time(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
unsigned int time;
aw_dev_get_fade_time(&time, false);
ucontrol->value.integer.value[0] = time;
aw_pr_dbg("step time %ld", ucontrol->value.integer.value[0]);
return 0;
}
static int aw882xx_set_fade_out_time(struct snd_kcontrol *kcontrol,
struct snd_ctl_elem_value *ucontrol)
{
struct soc_mixer_control *mc =
(struct soc_mixer_control *)kcontrol->private_value;
if (ucontrol->value.integer.value[0] > mc->max) {
aw_pr_dbg("set val %ld overflow %d",
ucontrol->value.integer.value[0], mc->max);
return 0;
}
aw_dev_set_fade_time(ucontrol->value.integer.value[0], false);
aw_pr_dbg("step time %ld", ucontrol->value.integer.value[0]);
return 0;
}
static const struct soc_enum aw882xx_snd_enum[] = {
SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(aw882xx_switch), aw882xx_switch),
#ifdef AW_SPIN_ENABLE
SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(aw882xx_spin), aw882xx_spin),
#endif
};
static struct snd_kcontrol_new aw882xx_controls[] = {
SOC_ENUM_EXT("aw882xx_rx_switch", aw882xx_snd_enum[0],
aw882xx_get_rx_en, aw882xx_set_rx_en),
SOC_ENUM_EXT("aw882xx_tx_switch", aw882xx_snd_enum[0],
aw882xx_get_tx_en, aw882xx_set_tx_en),
SOC_ENUM_EXT("aw882xx_copp_switch", aw882xx_snd_enum[0],
aw882xx_get_copp_en, aw882xx_set_copp_en),
#ifdef AW_SPIN_ENABLE
SOC_ENUM_EXT("aw882xx_spin_switch", aw882xx_snd_enum[1],
aw882xx_get_spin, aw882xx_set_spin),
#endif
SOC_SINGLE_EXT("aw882xx_fadein_us", 0, 0, 1000000, 0,
aw882xx_get_fade_in_time, aw882xx_set_fade_in_time),
SOC_SINGLE_EXT("aw882xx_fadeout_us", 0, 0, 1000000, 0,
aw882xx_get_fade_out_time, aw882xx_set_fade_out_time),
};
static void aw882xx_add_codec_controls(struct aw882xx *aw882xx)
{
aw_dev_info(aw882xx->dev, "enter");
aw_componet_codec_ops.add_codec_controls(aw882xx->codec,
&aw882xx_controls[0], ARRAY_SIZE(aw882xx_controls));
}
#ifdef AW_MTK_PLATFORM_WITH_DSP
static int aw882xx_name_append_suffix(struct aw882xx *aw882xx, const char **name)
{
char buf[50];
int i2cbus = aw882xx->i2c->adapter->nr;
int i2caddr = aw882xx->i2c->addr;
snprintf(buf, 50, "%s-%x-%x", *name, i2cbus, i2caddr);
(*name) = aw882xx_devm_kstrdup(aw882xx->dev, buf);
if (!(*name))
return -ENOMEM;
aw_dev_info(aw882xx->dev, "name is %s", (*name));
return 0;
}
static const struct snd_soc_dapm_widget aw882xx_dapm_widgets[] = {
/* playback */
SND_SOC_DAPM_AIF_IN("AIF_RX", "Speaker_Playback", 0, SND_SOC_NOPM, 0, 0),
SND_SOC_DAPM_OUTPUT("audio_out"),
/* capture */
SND_SOC_DAPM_AIF_OUT("AIF_TX", "Speaker_Capture", 0, SND_SOC_NOPM, 0, 0),
SND_SOC_DAPM_INPUT("iv_in"),
};
static const struct snd_soc_dapm_route aw882xx_audio_map[] = {
{"audio_out", NULL, "AIF_RX"},
{"AIF_TX", NULL, "iv_in"},
};
#if LINUX_VERSION_CODE < KERNEL_VERSION(4,2,0)
static struct snd_soc_dapm_context *snd_soc_codec_get_dapm(struct snd_soc_codec *codec)
{
return &codec->dapm;
}
#endif
#endif
static void aw882xx_add_widgets(struct aw882xx *aw882xx)
{
#ifdef AW_MTK_PLATFORM_WITH_DSP
int i = 0;
int ret;
struct snd_soc_dapm_widget *aw_widgets = NULL;
struct snd_soc_dapm_route *aw_route = NULL;
#ifdef AW_KERNEL_VER_OVER_4_19_1
struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(aw882xx->codec);
#else
struct snd_soc_dapm_context *dapm = snd_soc_codec_get_dapm(aw882xx->codec);
#endif
/*add widgets*/
aw_widgets = devm_kzalloc(aw882xx->dev,
sizeof(struct snd_soc_dapm_widget) * ARRAY_SIZE(aw882xx_dapm_widgets),
GFP_KERNEL);
if (!aw_widgets) {
aw_dev_err(aw882xx->dev, "alloc widget memory failed!");
return;
}
memcpy(aw_widgets, aw882xx_dapm_widgets,
sizeof(struct snd_soc_dapm_widget) * ARRAY_SIZE(aw882xx_dapm_widgets));
for (i = 0; i < ARRAY_SIZE(aw882xx_dapm_widgets); i++) {
if (aw_widgets[i].name) {
ret = aw882xx_name_append_suffix(aw882xx, &aw_widgets[i].name);
if (ret) {
aw_dev_err(aw882xx->dev, "append widget name suffix failed!");
return;
}
}
if (aw_widgets[i].sname) {
ret = aw882xx_name_append_suffix(aw882xx, &aw_widgets[i].sname);
if (ret) {
aw_dev_err(aw882xx->dev, "append widget sname suffix failed!");
return;
}
}
}
snd_soc_dapm_new_controls(dapm, aw_widgets, ARRAY_SIZE(aw882xx_dapm_widgets));
/*add route*/
aw_route = devm_kzalloc(aw882xx->dev,
sizeof(struct snd_soc_dapm_route) * ARRAY_SIZE(aw882xx_audio_map),
GFP_KERNEL);
if (!aw_route) {
aw_dev_err(aw882xx->dev, "alloc route memory failed!");
return;
}
memcpy(aw_route, aw882xx_audio_map,
sizeof(struct snd_soc_dapm_route) * ARRAY_SIZE(aw882xx_audio_map));
for (i = 0; i < ARRAY_SIZE(aw882xx_audio_map); i++) {
if (aw_route[i].sink) {
ret = aw882xx_name_append_suffix(aw882xx, &aw_route[i].sink);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "append sink name suffix failed!");
return;
}
}
if (aw_route[i].source) {
ret = aw882xx_name_append_suffix(aw882xx, &aw_route[i].source);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "append source name suffix failed!");
return;
}
}
}
snd_soc_dapm_add_routes(dapm, aw_route, ARRAY_SIZE(aw882xx_audio_map));
#endif
}
static void aw882xx_load_fw(struct aw882xx *aw882xx)
{
#ifdef AW_QCOM_PLATFORM
aw882xx_request_firmware(&aw882xx->fw_work.work);
#else
queue_delayed_work(aw882xx->work_queue,
&aw882xx->fw_work,
msecs_to_jiffies(AW882XX_LOAD_FW_DELAY_TIME));
#endif
}
static int aw882xx_codec_probe(aw_snd_soc_codec_t *aw_codec)
{
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(aw_codec);
aw_dev_info(aw882xx->dev, "enter");
aw882xx->work_queue = create_singlethread_workqueue("aw882xx");
if (!aw882xx->work_queue) {
aw_dev_err(aw882xx->dev, "create workqueue failed !");
return -EINVAL;
}
INIT_DELAYED_WORK(&aw882xx->start_work, aw882xx_startup_work);
INIT_DELAYED_WORK(&aw882xx->interrupt_work, aw882xx_interrupt_work);
INIT_DELAYED_WORK(&aw882xx->dc_work, aw882xx_dc_prot_work);
INIT_DELAYED_WORK(&aw882xx->fw_work, aw882xx_request_firmware);
aw882xx->codec = aw_codec;
if (aw882xx->index == 0)
aw882xx_add_codec_controls(aw882xx);
aw882xx_add_widgets(aw882xx);
/*load fw bin*/
aw882xx_load_fw(aw882xx);
/*load cali re*/
aw_dev_init_cali_re(aw882xx->aw_pa);
return 0;
}
#ifdef AW_KERNEL_VER_OVER_4_19_1
static void aw882xx_codec_remove(aw_snd_soc_codec_t *aw_codec)
{
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(aw_codec);
aw_dev_info(aw882xx->dev, "enter");
aw_dev_deinit(aw882xx->aw_pa);
destroy_workqueue(aw882xx->work_queue);
aw882xx->work_queue = NULL;
}
#else
static int aw882xx_codec_remove(aw_snd_soc_codec_t *aw_codec)
{
struct aw882xx *aw882xx =
aw_componet_codec_ops.codec_get_drvdata(aw_codec);
aw_dev_info(aw882xx->dev, "enter");
aw_dev_deinit(aw882xx->aw_pa);
destroy_workqueue(aw882xx->work_queue);
aw882xx->work_queue = NULL;
return 0;
}
#endif
static int aw882xx_dai_drv_append_suffix(struct aw882xx *aw882xx,
struct snd_soc_dai_driver *dai_drv,
int num_dai)
{
char buf[50];
int i;
int i2cbus = aw882xx->i2c->adapter->nr;
int addr = aw882xx->i2c->addr;
if ((dai_drv != NULL) && (num_dai > 0))
for (i = 0; i < num_dai; i++) {
snprintf(buf, 50, "%s-%x-%x", dai_drv[i].name, i2cbus,
addr);
dai_drv[i].name = aw882xx_devm_kstrdup(aw882xx->dev, buf);
snprintf(buf, 50, "%s-%x-%x",
dai_drv[i].playback.stream_name,
i2cbus, addr);
dai_drv[i].playback.stream_name = aw882xx_devm_kstrdup(aw882xx->dev, buf);
snprintf(buf, 50, "%s-%x-%x",
dai_drv[i].capture.stream_name,
i2cbus, addr);
dai_drv[i].capture.stream_name = aw882xx_devm_kstrdup(aw882xx->dev, buf);
aw_dev_info(aw882xx->dev, "dai name [%s]", dai_drv[i].name);
aw_dev_info(aw882xx->dev, "pstream_name name [%s]", dai_drv[i].playback.stream_name);
aw_dev_info(aw882xx->dev, "cstream_name name [%s]", dai_drv[i].capture.stream_name);
}
return 0;
}
static struct snd_soc_dai_driver aw882xx_dai[] = {
{
.name = "aw882xx-aif",
.id = 1,
.playback = {
.stream_name = "Speaker_Playback",
.channels_min = 1,
.channels_max = 2,
.rates = SNDRV_PCM_RATE_8000_48000 | SNDRV_PCM_RATE_96000,
.formats = (SNDRV_PCM_FMTBIT_S16_LE | \
SNDRV_PCM_FMTBIT_S24_LE | \
SNDRV_PCM_FMTBIT_S32_LE),
},
.capture = {
.stream_name = "Speaker_Capture",
.channels_min = 1,
.channels_max = 2,
.rates = SNDRV_PCM_RATE_8000_48000 | SNDRV_PCM_RATE_96000,
.formats = (SNDRV_PCM_FMTBIT_S16_LE | \
SNDRV_PCM_FMTBIT_S24_LE | \
SNDRV_PCM_FMTBIT_S32_LE),
},
.ops = &aw882xx_dai_ops,
/* .symmetric_rates = 1,*/
},
};
#ifdef AW_KERNEL_VER_OVER_4_19_1
static struct snd_soc_component_driver soc_codec_dev_aw882xx = {
.probe = aw882xx_codec_probe,
.remove = aw882xx_codec_remove,
};
#else
static struct snd_soc_codec_driver soc_codec_dev_aw882xx = {
.probe = aw882xx_codec_probe,
.remove = aw882xx_codec_remove,
};
#endif
int aw_componet_codec_register(struct aw882xx *aw882xx)
{
struct snd_soc_dai_driver *dai_drv;
int ret;
dai_drv = devm_kzalloc(aw882xx->dev, sizeof(aw882xx_dai), GFP_KERNEL);
if (dai_drv == NULL) {
aw_dev_err(aw882xx->dev, "dai_driver malloc failed");
return -ENOMEM;
}
memcpy(dai_drv, aw882xx_dai, sizeof(aw882xx_dai));
aw882xx_dai_drv_append_suffix(aw882xx, dai_drv, ARRAY_SIZE(aw882xx_dai));
ret = aw882xx->codec_ops->register_codec(aw882xx->dev,
&soc_codec_dev_aw882xx,
dai_drv, ARRAY_SIZE(aw882xx_dai));
if (ret < 0) {
aw_dev_err(aw882xx->dev, "failed to register aw882xx: %d", ret);
return -EINVAL;
}
return 0;
}
/*****************************************************
*
* device tree
*
*****************************************************/
static int aw882xx_parse_gpio_dt(struct aw882xx *aw882xx,
struct device_node *np)
{
if (!np) {
aw882xx->reset_gpio = -1;
aw882xx->irq_gpio = -1;
return -EINVAL;
}
aw882xx->reset_gpio = of_get_named_gpio(np, "reset-gpio", 0);
if (aw882xx->reset_gpio < 0) {
aw_dev_info(aw882xx->dev, "no reset gpio provided, will not HW reset device");
} else {
aw_dev_info(aw882xx->dev, "reset gpio provided ok");
}
aw882xx->irq_gpio = of_get_named_gpio(np, "irq-gpio", 0);
if (aw882xx->irq_gpio < 0) {
aw_dev_info(aw882xx->dev, "no irq gpio provided.");
} else {
aw_dev_info(aw882xx->dev, "irq gpio provided ok.");
}
return 0;
}
static struct aw882xx *aw882xx_malloc_init(struct i2c_client *i2c)
{
struct aw882xx *aw882xx = devm_kzalloc(&i2c->dev, sizeof(struct aw882xx), GFP_KERNEL);
if (aw882xx == NULL) {
dev_err(&i2c->dev, "devm_kzalloc failed.");
return NULL;
}
aw882xx->dev = &i2c->dev;
aw882xx->i2c = i2c;
aw882xx->aw_pa = NULL;
aw882xx->codec = NULL;
aw882xx->codec_ops = &aw_componet_codec_ops;
aw882xx->fw_status = AW_DEV_FW_FAILED;
aw882xx->fw_retry_cnt = 0;
aw882xx->dbg_en_prof = true;
aw882xx->allow_pw = true;
aw882xx->work_queue = NULL;
mutex_init(&aw882xx->lock);
return aw882xx;
}
static int aw882xx_gpio_request(struct aw882xx *aw882xx)
{
int ret;
if (gpio_is_valid(aw882xx->reset_gpio)) {
ret = devm_gpio_request_one(aw882xx->dev, aw882xx->reset_gpio,
GPIOF_OUT_INIT_LOW, "aw882xx_rst");
if (ret) {
aw_dev_err(aw882xx->dev, "rst request failed");
return ret;
}
}
if (gpio_is_valid(aw882xx->irq_gpio)) {
ret = devm_gpio_request_one(aw882xx->dev, aw882xx->irq_gpio,
GPIOF_DIR_IN, "aw882xx_int");
if (ret) {
aw_dev_err(aw882xx->dev, "int request failed");
return ret;
}
}
return 0;
}
static int aw882xx_parse_dt(struct device *dev, struct aw882xx *aw882xx,
struct device_node *np)
{
int ret;
int32_t dc_enable = 0;
int32_t sync_enable = 0;
/*gpio dts parser*/
ret = aw882xx_parse_gpio_dt(aw882xx, np);
if (ret)
return ret;
ret = of_property_read_u32(np, "dc-flag", &dc_enable);
if (ret) {
dc_enable = false;
aw_dev_info(aw882xx->dev, "close dc protect!");
} else {
aw_dev_info(aw882xx->dev, "dc-flag = %d", dc_enable);
}
aw882xx->dc_flag = dc_enable;
ret = of_property_read_u32(np, "sync-flag", &sync_enable);
if (ret < 0) {
aw_dev_info(aw882xx->dev,
"read sync flag failed,default phase sync off");
sync_enable = false;
} else {
aw_dev_info(aw882xx->dev,
"sync flag is %d", sync_enable);
}
aw882xx->phase_sync = sync_enable;
return 0;
}
int aw882xx_hw_reset(struct aw882xx *aw882xx)
{
aw_dev_info(aw882xx->dev, "enter");
if (gpio_is_valid(aw882xx->reset_gpio)) {
gpio_set_value_cansleep(aw882xx->reset_gpio, 0);
mdelay(1);
gpio_set_value_cansleep(aw882xx->reset_gpio, 1);
mdelay(2);
} else {
aw_dev_info(aw882xx->dev, "has no reset gpio");
}
return 0;
}
static int aw882xx_read_chipid(struct aw882xx *aw882xx)
{
int ret = -1;
unsigned int cnt = 0;
unsigned int reg_value = 0;
while (cnt < AW_READ_CHIPID_RETRIES) {
ret = aw882xx_i2c_read(aw882xx, AW882XX_CHIP_ID_REG, &reg_value);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "failed to read REG_ID: %d", ret);
return -EIO;
}
switch (reg_value) {
case PID_1852_ID: {
aw_dev_info(aw882xx->dev, "aw882xx 1852 detected");
aw882xx->chip_id = reg_value;
return 0;
}
case PID_2013_ID: {
aw_dev_info(aw882xx->dev, "aw882xx 2013 detected");
aw882xx->chip_id = reg_value;
return 0;
}
case PID_2032_ID: {
aw_dev_info(aw882xx->dev, "aw882xx 2032 detected");
aw882xx->chip_id = reg_value;
return 0;
}
case PID_2055_ID: {
aw_dev_info(aw882xx->dev, "aw882xx 2055 detected");
aw882xx->chip_id = reg_value;
return 0;
}
case PID_2071_ID: {
aw_dev_info(aw882xx->dev, "aw882xx 2071 detected");
aw882xx->chip_id = reg_value;
return 0;
}
case PID_2113_ID: {
aw_dev_info(aw882xx->dev, "aw882xx 2113 detected");
aw882xx->chip_id = reg_value;
return 0;
}
default:
aw_dev_info(aw882xx->dev, "unsupported device revision (0x%x)",
reg_value);
break;
}
cnt++;
msleep(AW_READ_CHIPID_RETRY_DELAY);
}
return -EINVAL;
}
static irqreturn_t aw882xx_irq(int irq, void *data)
{
struct aw882xx *aw882xx = (struct aw882xx *)data;
if (!aw882xx) {
aw_pr_err("pointer is NULL");
return -EINVAL;
}
aw_dev_info(aw882xx->dev, "enter");
/* mask all irq */
aw_dev_set_intmask(aw882xx->aw_pa, false);
/* upload workqueue */
if (aw882xx->work_queue)
queue_delayed_work(aw882xx->work_queue, &aw882xx->interrupt_work, 0);
return IRQ_HANDLED;
}
static int aw882xx_interrupt_init(struct aw882xx *aw882xx)
{
int irq_flags;
int ret;
if (gpio_is_valid(aw882xx->irq_gpio)) {
irq_flags = IRQF_TRIGGER_FALLING | IRQF_ONESHOT;
ret = devm_request_threaded_irq(aw882xx->dev,
gpio_to_irq(aw882xx->irq_gpio),
NULL, aw882xx_irq, irq_flags,
"aw882xx", aw882xx);
if (ret != 0) {
aw_dev_err(aw882xx->dev, "Failed to request IRQ %d: %d",
gpio_to_irq(aw882xx->irq_gpio), ret);
return ret;
}
} else {
aw_dev_info(aw882xx->dev, "gpio invalid");
/* disable interrupt */
}
return 0;
}
static ssize_t aw882xx_reg_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
unsigned int databuf[2] = {0};
if (2 == sscanf(buf, "%x %x", &databuf[0], &databuf[1]))
aw882xx_i2c_write(aw882xx, databuf[0], databuf[1]);
return count;
}
static ssize_t aw882xx_reg_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
ssize_t len = 0;
unsigned char i = 0;
unsigned int reg_val = 0;
int reg_num = aw882xx->aw_pa->ops.aw_get_reg_num();
for (i = 0; i < reg_num; i++) {
if (aw882xx->aw_pa->ops.aw_check_rd_access(i)) {
aw882xx_i2c_read(aw882xx, i, &reg_val);
len += snprintf(buf+len, PAGE_SIZE-len,
"reg:0x%02x=0x%04x\n", i, reg_val);
}
}
return len;
}
static ssize_t aw882xx_rw_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
unsigned int databuf[2] = {0};
if (2 == sscanf(buf, "%x %x", &databuf[0], &databuf[1])) {
aw882xx->rw_reg_addr = (unsigned char)databuf[0];
if (aw882xx->aw_pa->ops.aw_check_rd_access(databuf[0]))
aw882xx_i2c_write(aw882xx, databuf[0], databuf[1]);
} else if (1 == sscanf(buf, "%x", &databuf[0])) {
aw882xx->rw_reg_addr = (unsigned char)databuf[0];
}
return count;
}
static ssize_t aw882xx_rw_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
ssize_t len = 0;
unsigned int reg_val = 0;
if (aw882xx->aw_pa->ops.aw_check_rd_access(aw882xx->rw_reg_addr)) {
aw882xx_i2c_read(aw882xx, aw882xx->rw_reg_addr, &reg_val);
len += snprintf(buf + len, PAGE_SIZE - len,
"reg:0x%02x=0x%04x\n", aw882xx->rw_reg_addr, reg_val);
}
return len;
}
int aw882xx_awrw_write(struct aw882xx *aw882xx, const char *buf, size_t count)
{
int i, ret;
char *data_buf = NULL;
int str_len, data_len, temp_data;
char *reg_data = NULL;
struct aw882xx_i2c_packet *packet = &aw882xx->i2c_packet;
aw_dev_info(aw882xx->dev, "write:reg_addr[0x%02x], reg_num[%d]",
packet->reg_addr, packet->reg_num);
data_len = AWRW_DATA_BYTES * packet->reg_num;
str_len = count - AWRW_HDR_LEN - 1;
if ((data_len * 5 - 1) > str_len) {
aw_dev_err(aw882xx->dev, "data_str_len [%d], requeset len [%d]",
str_len, (data_len * 5 - 1));
return -EINVAL;
}
data_buf = kmalloc(data_len + 1, GFP_KERNEL);
if (data_buf == NULL) {
aw_dev_err(aw882xx->dev, "alloc memory failed");
return -ENOMEM;
}
data_buf[0] = packet->reg_addr;
reg_data = data_buf + 1;
aw_dev_dbg(aw882xx->dev, "reg_addr: 0x%02x", data_buf[0]);
for (i = 0; i < data_len; i++) {
sscanf(buf + AWRW_HDR_LEN + 1 + i * 5, "0x%02x", &temp_data);
reg_data[i] = temp_data;
aw_dev_dbg(aw882xx->dev, "[%d] : 0x%02x", i, reg_data[i]);
}
ret = i2c_master_send(aw882xx->i2c, data_buf, data_len + 1);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "write failed");
kfree(data_buf);
data_buf = NULL;
return -EFAULT;
}
kfree(data_buf);
data_buf = NULL;
aw_dev_info(aw882xx->dev, "write success");
return 0;
}
static int aw882xx_awrw_data_check(struct aw882xx *aw882xx, int *data)
{
int reg_num_max = aw882xx->aw_pa->ops.aw_get_reg_num();
if ((data[AWRW_HDR_ADDR_BYTES] != AWRW_ADDR_BYTES) ||
(data[AWRW_HDR_DATA_BYTES] != AWRW_DATA_BYTES)) {
aw_dev_err(aw882xx->dev, "addr_bytes [%d] or data_bytes [%d] unsupport",
data[AWRW_HDR_ADDR_BYTES], data[AWRW_HDR_DATA_BYTES]);
return -EINVAL;
}
if (data[AWRW_HDR_REG_ADDR] >= reg_num_max) {
aw_dev_err(aw882xx->dev, "reg_addr[%d] > reg_max[%d]",
data[AWRW_HDR_REG_ADDR], reg_num_max);
return -EINVAL;
}
return 0;
}
/* flag addr_bytes data_bytes reg_num reg_addr*/
static int aw882xx_awrw_parse_buf(struct aw882xx *aw882xx,
const char *buf, size_t count)
{
int data[AWRW_HDR_MAX] = {0};
struct aw882xx_i2c_packet *packet = &aw882xx->i2c_packet;
int ret;
if (sscanf(buf, "0x%02x 0x%02x 0x%02x 0x%02x 0x%02x",
&data[AWRW_HDR_WR_FLAG],
&data[AWRW_HDR_ADDR_BYTES],
&data[AWRW_HDR_DATA_BYTES],
&data[AWRW_HDR_REG_NUM],
&data[AWRW_HDR_REG_ADDR]) == 5) {
ret = aw882xx_awrw_data_check(aw882xx, data);
if (ret < 0)
return ret;
packet->reg_addr = data[AWRW_HDR_REG_ADDR];
packet->reg_num = data[AWRW_HDR_REG_NUM];
if (data[AWRW_HDR_WR_FLAG] == AWRW_FLAG_WRITE) {
return aw882xx_awrw_write(aw882xx, buf, count);
} else if (data[AWRW_HDR_WR_FLAG] == AWRW_FLAG_READ) {
packet->status = AWRW_I2C_ST_READ;
aw_dev_info(aw882xx->dev, "read_cmd:reg_addr[0x%02x], reg_num[%d]",
packet->reg_addr, packet->reg_num);
return 0;
} else {
aw_dev_err(aw882xx->dev, "please check str format, unsupport flag %d",
data[AWRW_HDR_WR_FLAG]);
return -EINVAL;
}
} else {
aw_dev_err(aw882xx->dev, "can not parse string");
return -EINVAL;
}
}
static ssize_t aw882xx_awrw_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
int ret;
if (count < AWRW_HDR_LEN) {
aw_dev_err(dev, "data count too smaller, please check write format");
aw_dev_err(dev, "string %s", buf);
return -EINVAL;
}
ret = aw882xx_awrw_parse_buf(aw882xx, buf, count);
if (ret)
return -EINVAL;
return count;
}
static ssize_t aw882xx_awrw_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
struct aw882xx_i2c_packet *packet = &aw882xx->i2c_packet;
int data_len, len = 0;
int ret, i;
char *reg_data = NULL;
if (packet->status != AWRW_I2C_ST_READ) {
aw_dev_err(aw882xx->dev, "please write read cmd first");
return -EINVAL;
}
data_len = AWRW_DATA_BYTES * packet->reg_num;
reg_data = (char *)kmalloc(data_len, GFP_KERNEL);
if (reg_data == NULL) {
aw_dev_err(aw882xx->dev, "memory alloc failed");
ret = -EINVAL;
goto exit;
}
ret = aw882xx_i2c_reads(aw882xx, packet->reg_addr, (char *)reg_data, data_len);
if (ret < 0) {
ret = -EFAULT;
goto exit;
}
aw_dev_info(aw882xx->dev, "reg_addr 0x%02x, reg_num %d",
packet->reg_addr, packet->reg_num);
for (i = 0; i < data_len; i++) {
len += snprintf(buf + len, PAGE_SIZE - len,
"0x%02x,", reg_data[i]);
aw_dev_dbg(aw882xx->dev, "0x%02x", reg_data[i]);
}
ret = len;
exit:
if (reg_data)
kfree(reg_data);
packet->status = AWRW_I2C_ST_NONE;
return ret;
}
static ssize_t aw882xx_drv_ver_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
ssize_t len = 0;
len += snprintf(buf + len, PAGE_SIZE - len,
"driver_ver: %s \n", AW882XX_DRIVER_VERSION);
return len;
}
static ssize_t aw882xx_dsp_re_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
ssize_t len = 0;
int cali_re;
int ret;
struct aw882xx *aw882xx = dev_get_drvdata(dev);
ret = aw_dev_get_cali_re(aw882xx->aw_pa, &cali_re);
if (ret) {
len += snprintf(buf + len, PAGE_SIZE - len,
"read dsp_re failed!\n");
return len;
}
len += snprintf(buf + len, PAGE_SIZE - len,
"%d \n", cali_re);
return len;
}
static ssize_t aw882xx_fade_step_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
unsigned int databuf[2] = {0};
/*step 0 - 12*/
if (1 == sscanf(buf, "%d", &databuf[0])) {
if (databuf[0] > (aw882xx->aw_pa->volume_desc.mute_volume)) {
aw_dev_info(aw882xx->dev, "step overflow %d Db", databuf[0]);
return count;
}
aw_dev_set_fade_vol_step(aw882xx->aw_pa, databuf[0]);
}
aw_dev_info(aw882xx->dev, "set step %d Done", databuf[0]);
return count;
}
static ssize_t aw882xx_fade_step_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
ssize_t len = 0;
struct aw882xx *aw882xx = dev_get_drvdata(dev);
len += snprintf(buf + len, PAGE_SIZE - len,
"step: %d \n", aw_dev_get_fade_vol_step(aw882xx->aw_pa));
return len;
}
static ssize_t aw882xx_dbg_prof_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
unsigned int databuf[2] = {0};
if (1 == sscanf(buf, "%d", &databuf[0])) {
if (databuf[0])
aw882xx->dbg_en_prof = true;
else
aw882xx->dbg_en_prof = false;
}
aw_dev_info(aw882xx->dev, "en_prof %d Done", databuf[0]);
return count;
}
static ssize_t aw882xx_dbg_prof_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
ssize_t len = 0;
len += snprintf(buf + len, PAGE_SIZE - len,
" %d \n", aw882xx->dbg_en_prof);
return len;
}
static ssize_t aw882xx_sync_flag_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
unsigned int flag = 0;
int ret;
ret = kstrtouint(buf, 0, &flag);
if (ret < 0)
return ret;
flag = ((flag == false) ? false : true);
aw_dev_info(aw882xx->dev, "set phase sync flag : [%d]", flag);
aw882xx->phase_sync = flag;
return count;
}
static ssize_t aw882xx_sync_flag_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
ssize_t len = 0;
len += snprintf(buf + len, PAGE_SIZE - len,
"sync flag : %d\n", aw882xx->phase_sync);
return len;
}
static ssize_t aw882xx_print_dbg_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct aw882xx *aw882xx = dev_get_drvdata(dev);
int ret;
ret = kstrtouint(buf, 0, &g_print_dbg);
if (ret < 0)
return ret;
g_print_dbg = ((g_print_dbg == false) ? false : true);
aw_dev_info(aw882xx->dev, "set g_print_dbg : [%d]", g_print_dbg);
return count;
}
static ssize_t aw882xx_print_dbg_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
ssize_t len = 0;
len += snprintf(buf + len, PAGE_SIZE - len,
"g_print_dbg : %d\n", g_print_dbg);
return len;
}
static ssize_t aw882xx_algo_ver_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
ssize_t len = 0;
int ret;
char algo_ver_buf[ALGO_VERSION_MAX] = { 0 };
struct aw882xx *aw882xx = dev_get_drvdata(dev);
ret = aw_get_algo_version(aw882xx->aw_pa, algo_ver_buf);
if (ret < 0) {
len += snprintf(buf + len, PAGE_SIZE - len,
"read algo version failed!\n");
return len;
}
len += snprintf(buf + len, PAGE_SIZE - len, "%s\n", algo_ver_buf);
return len;
}
static DEVICE_ATTR(reg, S_IWUSR | S_IRUGO,
aw882xx_reg_show, aw882xx_reg_store);
static DEVICE_ATTR(rw, S_IWUSR | S_IRUGO,
aw882xx_rw_show, aw882xx_rw_store);
static DEVICE_ATTR(awrw, S_IWUSR | S_IRUGO,
aw882xx_awrw_show, aw882xx_awrw_store);
static DEVICE_ATTR(drv_ver, S_IRUGO,
aw882xx_drv_ver_show, NULL);
static DEVICE_ATTR(dsp_re, S_IRUGO,
aw882xx_dsp_re_show, NULL);
static DEVICE_ATTR(fade_step, S_IWUSR | S_IRUGO,
aw882xx_fade_step_show, aw882xx_fade_step_store);
static DEVICE_ATTR(dbg_prof, S_IWUSR | S_IRUGO,
aw882xx_dbg_prof_show, aw882xx_dbg_prof_store);
static DEVICE_ATTR(phase_sync, S_IWUSR | S_IRUGO,
aw882xx_sync_flag_show, aw882xx_sync_flag_store);
static DEVICE_ATTR(print_dbg, S_IWUSR | S_IRUGO,
aw882xx_print_dbg_show, aw882xx_print_dbg_store);
static DEVICE_ATTR(algo_ver, S_IRUGO,
aw882xx_algo_ver_show, NULL);
static struct attribute *aw882xx_attributes[] = {
&dev_attr_reg.attr,
&dev_attr_rw.attr,
&dev_attr_awrw.attr,
&dev_attr_drv_ver.attr,
&dev_attr_fade_step.attr,
&dev_attr_dbg_prof.attr,
&dev_attr_dsp_re.attr,
&dev_attr_phase_sync.attr,
&dev_attr_print_dbg.attr,
&dev_attr_algo_ver.attr,
NULL
};
static struct attribute_group aw882xx_attribute_group = {
.attrs = aw882xx_attributes,
};
int aw882xx_i2c_probe(struct i2c_client *i2c,
const struct i2c_device_id *id)
{
struct aw882xx *aw882xx = NULL;
struct device_node *np = i2c->dev.of_node;
int ret;
aw_pr_info("enter addr=0x%x", i2c->addr);
if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_I2C)) {
aw_dev_err(&i2c->dev, "check_functionality failed");
return -EIO;
}
/*dev free all auto free*/
aw882xx = aw882xx_malloc_init(i2c);
if (aw882xx == NULL) {
aw_dev_err(&i2c->dev, "malloc aw882xx failed");
return -ENOMEM;
}
i2c_set_clientdata(i2c, aw882xx);
ret = aw882xx_parse_dt(&i2c->dev, aw882xx, np);
if (ret) {
aw_dev_err(&i2c->dev, "failed to parse device tree node");
return ret;
}
/*get gpio resource*/
ret = aw882xx_gpio_request(aw882xx);
if (ret)
return ret;
/* hardware reset */
aw882xx_hw_reset(aw882xx);
/* aw882xx chip id */
ret = aw882xx_read_chipid(aw882xx);
if (ret < 0) {
aw_dev_err(&i2c->dev, "aw882xx_read_chipid failed ret=%d", ret);
return ret;
}
/*aw pa init*/
ret = aw882xx_init(aw882xx, g_aw882xx_dev_cnt);
if (ret)
return ret;
/*aw882xx irq*/
aw882xx_interrupt_init(aw882xx);
/*codec register*/
ret = aw_componet_codec_register(aw882xx);
if (ret) {
aw_dev_err(&i2c->dev, "codec register failde");
return ret;
}
/*create attr*/
ret = sysfs_create_group(&i2c->dev.kobj, &aw882xx_attribute_group);
if (ret < 0) {
aw_dev_err(aw882xx->dev, "error creating sysfs attr files");
goto err_sysfs;
}
/*set aw882xx to dev private*/
dev_set_drvdata(&i2c->dev, aw882xx);
/*i2c packet init*/
aw882xx->i2c_packet.status = AWRW_I2C_ST_NONE;
aw882xx->i2c_packet.reg_num = 0;
aw882xx->i2c_packet.reg_addr = 0xff;
aw882xx->i2c_packet.reg_data = NULL;
aw882xx->index = g_aw882xx_dev_cnt;
/*add device to total list*/
mutex_lock(&g_aw882xx_lock);
g_aw882xx_dev_cnt++;
mutex_unlock(&g_aw882xx_lock);
aw_dev_info(&i2c->dev, "dev_cnt %d", g_aw882xx_dev_cnt);
return ret;
err_sysfs:
aw_componet_codec_ops.unregister_codec(&i2c->dev);
return ret;
}
int aw882xx_i2c_remove(struct i2c_client *i2c)
{
struct aw882xx *aw882xx = i2c_get_clientdata(i2c);
aw_dev_info(aw882xx->dev, "enter");
/*rm irq*/
if (gpio_to_irq(aw882xx->irq_gpio))
devm_free_irq(&i2c->dev,
gpio_to_irq(aw882xx->irq_gpio),
aw882xx);
/*free gpio*/
if (gpio_is_valid(aw882xx->irq_gpio))
devm_gpio_free(&i2c->dev, aw882xx->irq_gpio);
if (gpio_is_valid(aw882xx->reset_gpio))
devm_gpio_free(&i2c->dev, aw882xx->reset_gpio);
/*rm attr node*/
sysfs_remove_group(&i2c->dev.kobj, &aw882xx_attribute_group);
/*free device resource */
aw_device_remove(aw882xx->aw_pa);
/*unregister codec*/
aw882xx->codec_ops->unregister_codec(&i2c->dev);
/*remove device to total list*/
mutex_lock(&g_aw882xx_lock);
g_aw882xx_dev_cnt--;
if (g_aw882xx_dev_cnt == 0) {
if (g_awinic_cfg) {
vfree(g_awinic_cfg);
g_awinic_cfg = NULL;
}
}
mutex_unlock(&g_aw882xx_lock);
return 0;
}
#if 0
static void aw882xx_i2c_shutdown(struct i2c_client *i2c)
{
struct aw882xx *aw882xx = i2c_get_clientdata(i2c);
aw_dev_info(aw882xx->dev, "enter");
mutex_lock(&aw882xx->lock);
aw_device_stop(aw882xx->aw_pa);
mutex_unlock(&aw882xx->lock);
}
static const struct i2c_device_id aw882xx_i2c_id[] = {
{ AW882XX_I2C_NAME, 0 },
{ }
};
MODULE_DEVICE_TABLE(i2c, aw882xx_i2c_id);
static struct of_device_id aw882xx_dt_match[] = {
{ .compatible = "awinic,aw882xx_smartpa" },
{ },
};
static struct i2c_driver aw882xx_i2c_driver = {
.driver = {
.name = AW882XX_I2C_NAME,
.owner = THIS_MODULE,
.of_match_table = of_match_ptr(aw882xx_dt_match),
},
.probe = aw882xx_i2c_probe,
.remove = aw882xx_i2c_remove,
.shutdown = aw882xx_i2c_shutdown,
.id_table = aw882xx_i2c_id,
};
static int __init aw882xx_i2c_init(void)
{
int ret = -1;
aw_pr_info("aw882xx driver version %s", AW882XX_DRIVER_VERSION);
ret = i2c_add_driver(&aw882xx_i2c_driver);
if (ret)
aw_pr_err("fail to add aw882xx device into i2c");
return ret;
}
module_init(aw882xx_i2c_init);
static void __exit aw882xx_i2c_exit(void)
{
i2c_del_driver(&aw882xx_i2c_driver);
}
module_exit(aw882xx_i2c_exit);
#endif
MODULE_DESCRIPTION("ASoC AW882XX Smart PA Driver");
MODULE_LICENSE("GPL v2");