blob: 925a3227a1142fb697b5031ee03cd1e6ab32624c [file] [log] [blame]
/*
** =============================================================================
** Copyright (c) 2016 Texas Instruments Inc.
**
** 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; version 2.
**
** 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.
**
** File:
** tas2557-misc.c
**
** Description:
** misc driver for Texas Instruments TAS2557 High Performance 4W Smart Amplifier
**
** =============================================================================
*/
#ifdef CONFIG_TAS2557_MISC
#define DEBUG
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/pm.h>
#include <linux/i2c.h>
#include <linux/gpio.h>
#include <linux/regulator/consumer.h>
#include <linux/firmware.h>
#include <linux/regmap.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/slab.h>
#include <linux/syscalls.h>
#include <linux/fcntl.h>
#include <linux/miscdevice.h>
#include <linux/uaccess.h>
#include "tas2557.h"
#include "tas2557-core.h"
#include "tas2557-misc.h"
#include <linux/dma-mapping.h>
static int g_logEnable = 1;
static struct tas2557_priv *g_tas2557;
static int tas2557_file_open(struct inode *inode, struct file *file)
{
struct tas2557_priv *pTAS2557 = g_tas2557;
if (!try_module_get(THIS_MODULE))
return -ENODEV;
file->private_data = (void *)pTAS2557;
if (g_logEnable)
dev_info(pTAS2557->dev, "%s\n", __func__);
return 0;
}
static int tas2557_file_release(struct inode *inode, struct file *file)
{
struct tas2557_priv *pTAS2557 = (struct tas2557_priv *)file->private_data;
if (g_logEnable)
dev_info(pTAS2557->dev, "%s\n", __func__);
file->private_data = (void *)NULL;
module_put(THIS_MODULE);
return 0;
}
static ssize_t tas2557_file_read(struct file *file, char *buf, size_t count, loff_t *ppos)
{
struct tas2557_priv *pTAS2557 = (struct tas2557_priv *)file->private_data;
int ret = 0;
unsigned int nValue = 0;
unsigned char value = 0;
unsigned char *p_kBuf = NULL;
mutex_lock(&pTAS2557->file_lock);
switch (pTAS2557->mnDBGCmd) {
case TIAUDIO_CMD_REG_READ: {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_REG_READ: current_reg = 0x%x, count=%d\n",
pTAS2557->mnCurrentReg, (int)count);
if (count == 1) {
ret = pTAS2557->read(pTAS2557, pTAS2557->mnCurrentReg, &nValue);
if (ret < 0)
break;
value = (u8)nValue;
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_REG_READ: nValue=0x%x, value=0x%x\n", nValue, value);
ret = copy_to_user(buf, &value, 1);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
} else if (count > 1) {
p_kBuf = kzalloc(count, GFP_KERNEL);
if (p_kBuf != NULL) {
ret = pTAS2557->bulk_read(pTAS2557, pTAS2557->mnCurrentReg, p_kBuf, count);
if (ret < 0)
break;
ret = copy_to_user(buf, p_kBuf, count);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
kfree(p_kBuf);
} else
dev_err(pTAS2557->dev, "read no mem\n");
}
}
break;
case TIAUDIO_CMD_PROGRAM: {
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0)) {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_PROGRAM: count = %d\n", (int)count);
if (count == PROGRAM_BUF_SIZE) {
p_kBuf = kzalloc(count, GFP_KERNEL);
if (p_kBuf != NULL) {
struct TProgram *pProgram =
&(pTAS2557->mpFirmware->mpPrograms[pTAS2557->mnCurrentProgram]);
p_kBuf[0] = pTAS2557->mpFirmware->mnPrograms;
p_kBuf[1] = pTAS2557->mnCurrentProgram;
p_kBuf[2] = pProgram->mnAppMode;
p_kBuf[3] = (pProgram->mnBoost&0xff00)>>8;
p_kBuf[4] = (pProgram->mnBoost&0x00ff);
memcpy(&p_kBuf[5], pProgram->mpName, FW_NAME_SIZE);
strlcpy(&p_kBuf[5+FW_NAME_SIZE], pProgram->mpDescription, strlen(pProgram->mpDescription) + 1);
ret = copy_to_user(buf, p_kBuf, count);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
kfree(p_kBuf);
} else
dev_err(pTAS2557->dev, "read no mem\n");
} else
dev_err(pTAS2557->dev, "read buffer not sufficient\n");
} else
dev_err(pTAS2557->dev, "%s, firmware not loaded\n", __func__);
}
break;
case TIAUDIO_CMD_CONFIGURATION: {
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0)) {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_CONFIGURATION: count = %d\n", (int)count);
if (count == CONFIGURATION_BUF_SIZE) {
p_kBuf = kzalloc(count, GFP_KERNEL);
if (p_kBuf != NULL) {
struct TConfiguration *pConfiguration = &(pTAS2557->mpFirmware->mpConfigurations[pTAS2557->mnCurrentConfiguration]);
p_kBuf[0] = pTAS2557->mpFirmware->mnConfigurations;
p_kBuf[1] = pTAS2557->mnCurrentConfiguration;
memcpy(&p_kBuf[2], pConfiguration->mpName, FW_NAME_SIZE);
p_kBuf[2+FW_NAME_SIZE] = pConfiguration->mnProgram;
p_kBuf[3+FW_NAME_SIZE] = pConfiguration->mnPLL;
p_kBuf[4+FW_NAME_SIZE] = (pConfiguration->mnSamplingRate&0x000000ff);
p_kBuf[5+FW_NAME_SIZE] = ((pConfiguration->mnSamplingRate&0x0000ff00)>>8);
p_kBuf[6+FW_NAME_SIZE] = ((pConfiguration->mnSamplingRate&0x00ff0000)>>16);
p_kBuf[7+FW_NAME_SIZE] = ((pConfiguration->mnSamplingRate&0xff000000)>>24);
strlcpy(&p_kBuf[8+FW_NAME_SIZE], pConfiguration->mpDescription, strlen(pConfiguration->mpDescription)+1);
ret = copy_to_user(buf, p_kBuf, count);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
kfree(p_kBuf);
} else
dev_err(pTAS2557->dev, "read no mem\n");
} else
dev_err(pTAS2557->dev, "read buffer not sufficient\n");
} else
dev_err(pTAS2557->dev, "%s, firmware not loaded\n", __func__);
}
break;
case TIAUDIO_CMD_FW_TIMESTAMP: {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_FW_TIMESTAMP: count = %d\n", (int)count);
if (count == 4) {
p_kBuf = kzalloc(count, GFP_KERNEL);
if (p_kBuf != NULL) {
p_kBuf[0] = (pTAS2557->mpFirmware->mnTimeStamp&0x000000ff);
p_kBuf[1] = ((pTAS2557->mpFirmware->mnTimeStamp&0x0000ff00)>>8);
p_kBuf[2] = ((pTAS2557->mpFirmware->mnTimeStamp&0x00ff0000)>>16);
p_kBuf[3] = ((pTAS2557->mpFirmware->mnTimeStamp&0xff000000)>>24);
ret = copy_to_user(buf, p_kBuf, count);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
kfree(p_kBuf);
} else
dev_err(pTAS2557->dev, "read no mem\n");
}
}
break;
case TIAUDIO_CMD_CALIBRATION: {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_CALIBRATION: count = %d\n", (int)count);
if (count == 1) {
unsigned char curCal = pTAS2557->mnCurrentCalibration;
ret = copy_to_user(buf, &curCal, 1);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
}
}
break;
case TIAUDIO_CMD_SAMPLERATE: {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_SAMPLERATE: count = %d\n", (int)count);
if (count == 4) {
p_kBuf = kzalloc(count, GFP_KERNEL);
if (p_kBuf != NULL) {
struct TConfiguration *pConfiguration =
&(pTAS2557->mpFirmware->mpConfigurations[pTAS2557->mnCurrentConfiguration]);
p_kBuf[0] = (pConfiguration->mnSamplingRate&0x000000ff);
p_kBuf[1] = ((pConfiguration->mnSamplingRate&0x0000ff00)>>8);
p_kBuf[2] = ((pConfiguration->mnSamplingRate&0x00ff0000)>>16);
p_kBuf[3] = ((pConfiguration->mnSamplingRate&0xff000000)>>24);
ret = copy_to_user(buf, p_kBuf, count);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
kfree(p_kBuf);
} else
dev_err(pTAS2557->dev, "read no mem\n");
}
}
break;
case TIAUDIO_CMD_BITRATE: {
if (g_logEnable)
dev_info(pTAS2557->dev,
"TIAUDIO_CMD_BITRATE: count = %d\n", (int)count);
if (count == 1) {
unsigned char bitRate = 0;
ret = tas2557_get_bit_rate(pTAS2557, &bitRate);
if (ret >= 0) {
ret = copy_to_user(buf, &bitRate, 1);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
}
}
}
break;
case TIAUDIO_CMD_DACVOLUME: {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_DACVOLUME: count = %d\n", (int)count);
if (count == 1) {
unsigned char volume = 0;
ret = tas2557_get_DAC_gain(pTAS2557, &volume);
if (ret >= 0) {
ret = copy_to_user(buf, &volume, 1);
if (ret != 0) {
/* Failed to copy all the data, exit */
dev_err(pTAS2557->dev, "copy to user fail %d\n", ret);
}
}
}
}
break;
}
pTAS2557->mnDBGCmd = 0;
mutex_unlock(&pTAS2557->file_lock);
return count;
}
static ssize_t tas2557_file_write(struct file *file, const char *buf, size_t count, loff_t *ppos)
{
struct tas2557_priv *pTAS2557 = (struct tas2557_priv *)file->private_data;
int ret = 0;
unsigned char *p_kBuf = NULL;
unsigned int reg = 0;
unsigned int len = 0;
mutex_lock(&pTAS2557->file_lock);
p_kBuf = kzalloc(count, GFP_KERNEL);
if (p_kBuf == NULL) {
dev_err(pTAS2557->dev, "write no mem\n");
goto err;
}
ret = copy_from_user(p_kBuf, buf, count);
if (ret != 0) {
dev_err(pTAS2557->dev, "copy_from_user failed.\n");
goto err;
}
pTAS2557->mnDBGCmd = p_kBuf[0];
switch (pTAS2557->mnDBGCmd) {
case TIAUDIO_CMD_REG_WITE:
if (count > 5) {
reg = ((unsigned int)p_kBuf[1] << 24)
+ ((unsigned int)p_kBuf[2] << 16)
+ ((unsigned int)p_kBuf[3] << 8)
+ (unsigned int)p_kBuf[4];
len = count - 5;
if (len == 1) {
ret = pTAS2557->write(pTAS2557, reg, p_kBuf[5]);
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_REG_WITE, Reg=0x%x, Val=0x%x\n", reg, p_kBuf[5]);
} else
ret = pTAS2557->bulk_write(pTAS2557, reg, &p_kBuf[5], len);
} else
dev_err(pTAS2557->dev, "%s, write len fail, count=%d.\n", __func__, (int)count);
pTAS2557->mnDBGCmd = 0;
break;
case TIAUDIO_CMD_REG_READ:
if (count == 5) {
pTAS2557->mnCurrentReg = ((unsigned int)p_kBuf[1] << 24)
+ ((unsigned int)p_kBuf[2] << 16)
+ ((unsigned int)p_kBuf[3] << 8)
+ (unsigned int)p_kBuf[4];
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_REG_READ whole=0x%x\n", pTAS2557->mnCurrentReg);
} else
dev_err(pTAS2557->dev, "read len fail.\n");
break;
case TIAUDIO_CMD_DEBUG_ON:
if (count == 2)
g_logEnable = p_kBuf[1];
pTAS2557->mnDBGCmd = 0;
break;
case TIAUDIO_CMD_PROGRAM:
{
if (count == 2) {
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0)) {
int config = -1;
if (p_kBuf[1] == pTAS2557->mnCurrentProgram)
config = pTAS2557->mnCurrentConfiguration;
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_PROGRAM, set to %d, cfg=%d\n", p_kBuf[1], config);
tas2557_set_program(pTAS2557, p_kBuf[1], config);
pTAS2557->mnDBGCmd = 0;
} else
dev_err(pTAS2557->dev, "%s, firmware not loaded\n", __func__);
}
}
break;
case TIAUDIO_CMD_CONFIGURATION:
{
if (count == 2) {
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0)) {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_CONFIGURATION, set to %d\n", p_kBuf[1]);
tas2557_set_config(pTAS2557, p_kBuf[1]);
pTAS2557->mnDBGCmd = 0;
} else
dev_err(pTAS2557->dev, "%s, firmware not loaded\n", __func__);
}
}
break;
case TIAUDIO_CMD_FW_TIMESTAMP:
/*let go*/
break;
case TIAUDIO_CMD_CALIBRATION:
{
if (count == 2) {
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0)) {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_CALIBRATION, set to %d\n", p_kBuf[1]);
tas2557_set_calibration(pTAS2557, p_kBuf[1]);
pTAS2557->mnDBGCmd = 0;
}
}
}
break;
case TIAUDIO_CMD_SAMPLERATE:
if (count == 5) {
unsigned int nSampleRate = ((unsigned int)p_kBuf[1] << 24) +
((unsigned int)p_kBuf[2] << 16) +
((unsigned int)p_kBuf[3] << 8) +
(unsigned int)p_kBuf[4];
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_SAMPLERATE, set to %d\n", nSampleRate);
tas2557_set_sampling_rate(pTAS2557, nSampleRate);
}
break;
case TIAUDIO_CMD_BITRATE:
if (count == 2) {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_BITRATE, set to %d\n", p_kBuf[1]);
tas2557_set_bit_rate(pTAS2557, p_kBuf[1]);
}
break;
case TIAUDIO_CMD_DACVOLUME:
if (count == 2) {
unsigned char volume;
volume = (p_kBuf[1] & 0x0f);
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_DACVOLUME, set to %d\n", volume);
ret = tas2557_set_DAC_gain(pTAS2557, volume);
if (ret < 0)
goto err;
}
break;
case TIAUDIO_CMD_SPEAKER:
if (count == 2) {
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_SPEAKER, set to %d\n", p_kBuf[1]);
tas2557_enable(pTAS2557, (p_kBuf[1] > 0));
}
break;
case TIAUDIO_CMD_FW_RELOAD:
if (count == 1) {
const char *pFWName;
if (pTAS2557->mnPGID == TAS2557_PG_VERSION_2P1)
pFWName = TAS2557_FW_NAME;
else if (pTAS2557->mnPGID == TAS2557_PG_VERSION_1P0)
pFWName = TAS2557_PG1P0_FW_NAME;
else
break;
ret = request_firmware_nowait(THIS_MODULE, 1, pFWName,
pTAS2557->dev, GFP_KERNEL, pTAS2557, tas2557_fw_ready);
if (g_logEnable)
dev_info(pTAS2557->dev, "TIAUDIO_CMD_FW_RELOAD: ret = %d\n", ret);
}
break;
default:
pTAS2557->mnDBGCmd = 0;
break;
}
err:
if (p_kBuf != NULL)
kfree(p_kBuf);
mutex_unlock(&pTAS2557->file_lock);
return count;
}
static long tas2557_file_unlocked_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
{
struct tas2557_priv *pTAS2557 = file->private_data;
int ret = 0;
mutex_lock(&pTAS2557->file_lock);
switch (cmd) {
case SMARTPA_SPK_DAC_VOLUME:
{
}
break;
case SMARTPA_SPK_POWER_ON:
{
printk("[tas2557] SMARTPA_SPK_POWER_ON \n"); //[jinjia]Trace
tas2557_enable(pTAS2557, true);
}
break;
case SMARTPA_SPK_POWER_OFF:
{
printk("[tas2557] SMARTPA_SPK_POWER_OFF \n"); //[jinjia]Trace
tas2557_enable(pTAS2557, false);
}
break;
case SMARTPA_SPK_SWITCH_PROGRAM:
{
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0))
tas2557_set_program(pTAS2557, arg, -1);
}
break;
case SMARTPA_SPK_SWITCH_CONFIGURATION:
{
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0))
tas2557_set_config(pTAS2557, arg);
}
break;
case SMARTPA_SPK_SWITCH_CALIBRATION:
{
if ((pTAS2557->mpFirmware->mnConfigurations > 0)
&& (pTAS2557->mpFirmware->mnPrograms > 0))
tas2557_set_calibration(pTAS2557, arg);
}
break;
case SMARTPA_SPK_SET_SAMPLERATE:
{
tas2557_set_sampling_rate(pTAS2557, arg);
}
break;
case SMARTPA_SPK_SET_BITRATE:
{
tas2557_set_bit_rate(pTAS2557, arg);
}
break;
}
mutex_unlock(&pTAS2557->file_lock);
return ret;
}
static const struct file_operations fops = {
.owner = THIS_MODULE,
.read = tas2557_file_read,
.write = tas2557_file_write,
.unlocked_ioctl = tas2557_file_unlocked_ioctl,
.open = tas2557_file_open,
.release = tas2557_file_release,
};
#define MODULE_NAME "tas2557"
static struct miscdevice tas2557_misc = {
.minor = MISC_DYNAMIC_MINOR,
.name = MODULE_NAME,
.fops = &fops,
};
int tas2557_register_misc(struct tas2557_priv *pTAS2557)
{
int ret = 0;
g_tas2557 = pTAS2557;
ret = misc_register(&tas2557_misc);
if (ret)
dev_err(pTAS2557->dev, "TAS2557 misc fail: %d\n", ret);
dev_info(pTAS2557->dev, "%s, leave\n", __func__);
return ret;
}
int tas2557_deregister_misc(struct tas2557_priv *pTAS2557)
{
misc_deregister(&tas2557_misc);
return 0;
}
MODULE_AUTHOR("Texas Instruments Inc.");
MODULE_DESCRIPTION("TAS2557 Misc Smart Amplifier driver");
MODULE_LICENSE("GPL v2");
#endif