lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [thread-next>] [day] [month] [year] [list]
Date:   Tue, 20 Dec 2022 23:11:57 +0800
From:   Kevin Lu <luminlong@....com>
To:     lgirdwood@...il.com, broonie@...nel.org, perex@...ex.cz,
        tiwai@...e.com
Cc:     alsa-devel@...a-project.org, linux-kernel@...r.kernel.org,
        shenghao-ding@...com, kevin-lu@...com, navada@...com,
        peeyush@...com, Kevin Lu <luminlong@....com>
Subject: [PATCH v1] ALSA SoC: Texas Instruments TAS2781 Audio Smart Amp

The TAS2781 driver implements a flexible and configurable register setting
for one, two, even multiple TAS2781 chips. All the register setting are in
a bin file. Almost no specific register setting can be found in the code.

Signed-off-by: Kevin Lu <luminlong@....com>
---
 sound/soc/codecs/Kconfig       |   13 +
 sound/soc/codecs/Makefile      |    2 +
 sound/soc/codecs/tas2781-dsp.c | 2483 ++++++++++++++++++++++++++++++++
 sound/soc/codecs/tas2781-dsp.h |  213 +++
 sound/soc/codecs/tas2781-i2c.c | 2143 +++++++++++++++++++++++++++
 sound/soc/codecs/tas2781.h     |  208 +++
 6 files changed, 5062 insertions(+)
 create mode 100644 sound/soc/codecs/tas2781-dsp.c
 create mode 100644 sound/soc/codecs/tas2781-dsp.h
 create mode 100644 sound/soc/codecs/tas2781-i2c.c
 create mode 100644 sound/soc/codecs/tas2781.h

diff --git a/sound/soc/codecs/Kconfig b/sound/soc/codecs/Kconfig
index 7022e6286..31d2d9594 100644
--- a/sound/soc/codecs/Kconfig
+++ b/sound/soc/codecs/Kconfig
@@ -222,6 +222,7 @@ config SND_SOC_ALL_CODECS
 	imply SND_SOC_TAS2764
 	imply SND_SOC_TAS2770
 	imply SND_SOC_TAS2780
+	imply SND_SOC_TAS2781
 	imply SND_SOC_TAS5086
 	imply SND_SOC_TAS571X
 	imply SND_SOC_TAS5720
@@ -1573,6 +1574,18 @@ config SND_SOC_TAS2780
 	  Enable support for Texas Instruments TAS2780 high-efficiency
 	  digital input mono Class-D audio power amplifiers.
 
+config SND_SOC_TAS2781
+	tristate "Texas Instruments TAS2781 speaker amplifier"
+	depends on I2C
+	select REGMAP_I2C
+	help
+	  Enable support for Texas Instruments TAS2781 Smart Amplifier
+	  Digital input mono Class-D and DSP-inside audio power amplifiers.
+	  Note the TAS2781 driver implements a flexible and configurable
+	  register setting, for one, two, even multiple TAS2781 chips.
+	  All the register setting are in a bin file. Almost no specific
+	  register setting can be found in the code.
+
 config SND_SOC_TAS5086
 	tristate "Texas Instruments TAS5086 speaker amplifier"
 	depends on I2C
diff --git a/sound/soc/codecs/Makefile b/sound/soc/codecs/Makefile
index 9170ee144..088fed9e7 100644
--- a/sound/soc/codecs/Makefile
+++ b/sound/soc/codecs/Makefile
@@ -354,6 +354,7 @@ snd-soc-tas2552-objs := tas2552.o
 snd-soc-tas2562-objs := tas2562.o
 snd-soc-tas2764-objs := tas2764.o
 snd-soc-tas2780-objs := tas2780.o
+snd-soc-tas2781-objs :=	tas2781-i2c.o tas2781-dsp.o
 # Mux
 snd-soc-simple-mux-objs := simple-mux.o
 
@@ -604,6 +605,7 @@ obj-$(CONFIG_SND_SOC_TAS2552)	+= snd-soc-tas2552.o
 obj-$(CONFIG_SND_SOC_TAS2562)	+= snd-soc-tas2562.o
 obj-$(CONFIG_SND_SOC_TAS2764)	+= snd-soc-tas2764.o
 obj-$(CONFIG_SND_SOC_TAS2780)	+= snd-soc-tas2780.o
+obj-$(CONFIG_SND_SOC_TAS2781)	+= snd-soc-tas2781.o
 obj-$(CONFIG_SND_SOC_TAS5086)	+= snd-soc-tas5086.o
 obj-$(CONFIG_SND_SOC_TAS571X)	+= snd-soc-tas571x.o
 obj-$(CONFIG_SND_SOC_TAS5720)	+= snd-soc-tas5720.o
diff --git a/sound/soc/codecs/tas2781-dsp.c b/sound/soc/codecs/tas2781-dsp.c
new file mode 100644
index 000000000..56c7efd30
--- /dev/null
+++ b/sound/soc/codecs/tas2781-dsp.c
@@ -0,0 +1,2483 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * ALSA SoC Texas Instruments TAS2781 Audio Smart Amplifier
+ *
+ * Copyright (C) 2022 - 2023 Texas Instruments Incorporated
+ * https://www.ti.com
+ *
+ * The TAS2781 driver implements a flexible and configurable register setting
+ * for one, two, even multiple TAS2781 chips. All the register setting are in
+ * a bin file. Almost no specific register setting can be found in the code.
+ *
+ * Author: Shenghao Ding <shenghao-ding@...com>
+ *         Kevin Lu <kevin-lu@...com>
+ */
+
+#include <linux/regmap.h>
+#include <linux/i2c.h>
+#include <linux/string.h>
+#include <linux/crc8.h>
+#include <linux/slab.h>
+#include <linux/fs.h>
+#include <linux/firmware.h>
+
+#include "tas2781.h"
+
+#define TAS2781_CAL_BIN_PATH	("/lib/firmware/")
+
+#define ERROR_PRAM_CRCCHK			(0x0000000)
+#define ERROR_YRAM_CRCCHK			(0x0000001)
+#define BINFILEDOCVER				(0)
+#define DRVFWVER					(1)
+#define	PPC_DRIVER_CRCCHK			(0x00000200)
+
+#define TAS2781_SA_COEFF_SWAP_REG	TASDEVICE_REG(0, 0x35, 0x2c)
+#define TAS2781_YRAM_BOOK1			(140)
+#define TAS2781_YRAM1_PAGE			(42)
+#define TAS2781_YRAM1_START_REG		(88)
+
+#define TAS2781_YRAM2_START_PAGE	(43)
+#define TAS2781_YRAM2_END_PAGE		(49)
+#define TAS2781_YRAM2_START_REG		(8)
+#define TAS2781_YRAM2_END_REG		(127)
+
+#define TAS2781_YRAM3_PAGE			(50)
+#define TAS2781_YRAM3_START_REG		(8)
+#define TAS2781_YRAM3_END_REG		(27)
+
+/*should not include B0_P53_R44-R47 */
+#define TAS2781_YRAM_BOOK2			(0)
+#define TAS2781_YRAM4_START_PAGE	(50)
+#define TAS2781_YRAM4_END_PAGE		(60)
+
+#define TAS2781_YRAM5_PAGE			(61)
+#define TAS2781_YRAM5_START_REG		(8)
+#define TAS2781_YRAM5_END_REG		(27)
+
+#define TASDEVICE_MAXPROGRAM_NUM_KERNEL			(5)
+#define TASDEVICE_MAXCONFIG_NUM_KERNEL_MULTIPLE_AMPS	(64)
+#define TASDEVICE_MAXCONFIG_NUM_KERNEL			(10)
+#define MAIN_ALL_DEVICES_1X				(0x01)
+#define MAIN_DEVICE_A_1X				(0x02)
+#define MAIN_DEVICE_B_1X				(0x03)
+#define MAIN_DEVICE_C_1X				(0x04)
+#define MAIN_DEVICE_D_1X				(0x05)
+#define COEFF_DEVICE_A_1X				(0x12)
+#define COEFF_DEVICE_B_1X				(0x13)
+#define COEFF_DEVICE_C_1X				(0x14)
+#define COEFF_DEVICE_D_1X				(0x15)
+#define PRE_DEVICE_A_1X					(0x22)
+#define PRE_DEVICE_B_1X					(0x23)
+#define PRE_DEVICE_C_1X					(0x24)
+#define PRE_DEVICE_D_1X					(0x25)
+
+struct TYCRC {
+	unsigned char mnOffset;
+	unsigned char mnLen;
+};
+
+const unsigned int BinFileformatVerInfo[][2] = {
+	{0x100, 0x100},
+	{0x110, 0x200},
+	{0x200, 0x300},
+	{0x210, 0x310},
+	{0x230, 0x320},
+	{0x300, 0x400}
+};
+
+const char *devicefamily[1] = {
+	"TAS Devices" };
+
+const char *devicelist[TASDEVICE_DSP_TAS_MAX_DEVICE] = {
+	"TAS2555",
+	"TAS2555 Stereo",
+	"TAS2557 Mono",
+	"TAS2557 Dual Mono",
+	"TAS2559",
+	"TAS2563",
+	NULL,
+	"TAS2563 Dual Mono",
+	"TAS2563 Quad",
+	"TAS2563 2.1",
+	"TAS2781",
+	"TAS2781 Stereo",
+	"TAS2781 2.1",
+	"TAS2781 Quad"
+};
+
+const char deviceNumber[TASDEVICE_DSP_TAS_MAX_DEVICE] = {
+	1, 2, 1, 2, 1, 1, 0, 2, 4, 3, 1, 2, 3, 4
+};
+
+static int fw_parse_block_data_kernel(struct TFirmware *pFirmware,
+	struct TBlock *pBlock, const struct firmware *pFW, int offset)
+{
+	const unsigned char *pData = pFW->data;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mnType = SMS_HTONL(pData[offset],
+			pData[offset + 1], pData[offset + 2],
+			pData[offset + 3]);
+	offset  += 4;
+
+	if (offset + 1 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mbPChkSumPresent error\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mbPChkSumPresent = pData[offset];
+	offset++;
+
+	if (offset + 1 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mnPChkSum error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mnPChkSum = pData[offset];
+	offset++;
+
+	if (offset + 1 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mbYChkSumPresent error\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mbYChkSumPresent = pData[offset];
+	offset++;
+
+	if (offset + 1 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mnYChkSum error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mnYChkSum = pData[offset];
+	offset++;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: blk_size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->blk_size = SMS_HTONL(pData[offset],
+			pData[offset + 1], pData[offset + 2],
+			pData[offset + 3]);
+	offset  += 4;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: nSublocks error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->nSublocks = SMS_HTONL(pData[offset],
+		pData[offset + 1], pData[offset + 2], pData[offset + 3]);
+	offset  += 4;
+
+	pBlock->mpData = kzalloc(pBlock->blk_size, GFP_KERNEL);
+	if (pBlock->mpData == NULL) {
+		offset = -1;
+		goto out;
+	}
+	memcpy(pBlock->mpData, &pData[offset], pBlock->blk_size);
+	offset  += pBlock->blk_size;
+out:
+	return offset;
+}
+
+static int fw_parse_data_kernel(struct TFirmware *pFirmware,
+	struct TData *pImageData, const struct firmware *pFW, int offset)
+{
+	const unsigned char *pData = pFW->data;
+	unsigned int nBlock;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pImageData->mnBlocks = SMS_HTONL(pData[offset],
+		pData[offset + 1], pData[offset + 2], pData[offset + 3]);
+	offset  += 4;
+
+	pImageData->mpBlocks =
+		kcalloc(pImageData->mnBlocks, sizeof(struct TBlock),
+			GFP_KERNEL);
+	if (pImageData->mpBlocks == NULL) {
+		dev_err(pFirmware->dev, "%s: FW memory failed!\n", __func__);
+		goto out;
+	}
+
+	for (nBlock = 0; nBlock < pImageData->mnBlocks; nBlock++) {
+		offset = fw_parse_block_data_kernel(pFirmware,
+			&(pImageData->mpBlocks[nBlock]), pFW, offset);
+		if (offset < 0) {
+			offset = -1;
+			goto out;
+		}
+	}
+out:
+	return offset;
+}
+
+int fw_parse_program_data_kernel(struct tasdevice_priv *tas_dev,
+	struct TFirmware *pFirmware,
+	const struct firmware *pFW, int offset)
+{
+	struct TProgram *pProgram;
+	const unsigned char *buf = pFW->data;
+	unsigned int  nProgram = 0;
+
+	for (nProgram = 0; nProgram < pFirmware->mnPrograms; nProgram++) {
+		pProgram = &(pFirmware->mpPrograms[nProgram]);
+		if (offset + 64 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpName error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		memcpy(pProgram->mpName, &buf[offset], 64);
+		offset  += 64;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnAppMode error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnAppMode = buf[offset];
+		offset++;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPDMI2SMode error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnPDMI2SMode = buf[offset];
+		offset++;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnISnsPD error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnISnsPD = buf[offset];
+		offset++;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnVSnsPD error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnVSnsPD = buf[offset];
+		offset++;
+		//skip 3-byte reserved
+		offset  += 3;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPowerLDG error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnPowerLDG = buf[offset];
+		offset++;
+
+		offset = fw_parse_data_kernel(pFirmware, &(pProgram->mData),
+			pFW, offset);
+		if (offset < 0)
+			goto out;
+	}
+out:
+	return offset;
+}
+
+int fw_parse_configuration_data_kernel(
+	struct tasdevice_priv *tas_dev,
+	struct TFirmware *pFirmware, const struct firmware *pFW, int offset)
+{
+	const unsigned char *pData = pFW->data;
+
+	unsigned int nConfiguration;
+	struct TConfiguration *pConfiguration;
+
+	for (nConfiguration = 0; nConfiguration < pFirmware->mnConfigurations;
+		nConfiguration++) {
+		pConfiguration =
+			&(pFirmware->mpConfigurations[nConfiguration]);
+		if (offset + 64 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpName error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		memcpy(pConfiguration->mpName, &pData[offset], 64);
+		offset  += 64;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev,
+				"%s: mnDevice_orientation error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnDevice_orientation = pData[offset];
+		offset++;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnDevices error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnDevices = pData[offset + 1];
+		offset  += 1;
+
+		if (offset + 2 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mProgram error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mProgram = SMS_HTONS(pData[offset],
+			pData[offset + 1]);
+		offset  += 2;
+
+		if (offset + 4 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnSamplingRate error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnSamplingRate = SMS_HTONL(pData[offset],
+			pData[offset + 1], pData[offset + 2],
+			pData[offset + 3]);
+		offset  += 4;
+
+		if (offset + 2 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPLLSrc error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnPLLSrc = SMS_HTONS(pData[offset],
+			pData[offset + 1]);
+		offset  += 2;
+
+		if (offset + 2 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnFsRate error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnFsRate = SMS_HTONS(pData[offset],
+			pData[offset + 1]);
+		offset  += 2;
+
+		if (offset + 4 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPLLSrcRate error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnPLLSrcRate = SMS_HTONL(pData[offset],
+			pData[offset + 1], pData[offset + 2],
+			pData[offset + 3]);
+		offset  += 4;
+		offset = fw_parse_data_kernel(pFirmware,
+			&(pConfiguration->mData), pFW, offset);
+		if (offset < 0)
+			goto out;
+	}
+out:
+	return offset;
+}
+
+int fw_parse_variable_header_kernel(struct tasdevice_priv *tas_dev,
+	const struct firmware *pFW, int offset)
+{
+	struct TFirmware *pFirmware = tas_dev->mpFirmware;
+	struct tasdevice_dspfw_hdr *pFw_hdr = &(pFirmware->fw_hdr);
+	const unsigned char *buf = pFW->data;
+	struct TProgram *pProgram;
+	struct TConfiguration *pConfiguration;
+	unsigned int  nProgram = 0, nConfiguration = 0;
+	unsigned short maxConf = TASDEVICE_MAXCONFIG_NUM_KERNEL;
+
+	if (offset + 2 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_hdr->mnDeviceFamily = SMS_HTONS(buf[offset], buf[offset + 1]);
+	if (pFw_hdr->mnDeviceFamily != 0) {
+		dev_err(tas_dev->dev, "ERROR:%s:not TAS device\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	offset  += 2;
+	if (offset + 2 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnDevice error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_hdr->mnDevice = SMS_HTONS(buf[offset], buf[offset + 1]);
+	if (pFw_hdr->mnDevice >= TASDEVICE_DSP_TAS_MAX_DEVICE ||
+		pFw_hdr->mnDevice == 6) {
+		dev_err(tas_dev->dev, "ERROR:%s: not support device %d\n",
+			__func__, pFw_hdr->mnDevice);
+		offset = -1;
+		goto out;
+	}
+	offset  += 2;
+	pFw_hdr->ndev = deviceNumber[pFw_hdr->mnDevice];
+
+	if (pFw_hdr->ndev != tas_dev->ndev) {
+		dev_err(tas_dev->dev,
+			"%s: ndev(%u) in dspbin dismatch ndev(%u) in DTS\n",
+			__func__, pFw_hdr->ndev, tas_dev->ndev);
+		offset = -1;
+		goto out;
+	}
+
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnPrograms error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFirmware->mnPrograms = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+
+	if (pFirmware->mnPrograms == 0 || pFirmware->mnPrograms >
+		TASDEVICE_MAXPROGRAM_NUM_KERNEL) {
+		dev_err(tas_dev->dev, "%s: mnPrograms is invalid\n", __func__);
+		offset = -1;
+		goto out;
+	}
+
+	if (offset + 4 * TASDEVICE_MAXPROGRAM_NUM_KERNEL > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mpPrograms error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+
+	pFirmware->mpPrograms =
+		kcalloc(pFirmware->mnPrograms,
+		sizeof(struct TProgram), GFP_KERNEL);
+	if (pFirmware->mpPrograms == NULL) {
+		dev_err(tas_dev->dev, "%s: mpPrograms memory failed!\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+
+	for (nProgram = 0; nProgram < pFirmware->mnPrograms; nProgram++) {
+		pProgram = &(pFirmware->mpPrograms[nProgram]);
+		pProgram->prog_size = SMS_HTONL(buf[offset], buf[offset + 1],
+			buf[offset + 2], buf[offset + 3]);
+		pFirmware->cfg_start_offset  += pProgram->prog_size;
+		offset  += 4;
+	}
+	offset  += (4 * (5 - nProgram));
+
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnConfigurations error\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+	pFirmware->mnConfigurations = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	maxConf = (pFw_hdr->ndev >= 4) ?
+		TASDEVICE_MAXCONFIG_NUM_KERNEL_MULTIPLE_AMPS :
+		TASDEVICE_MAXCONFIG_NUM_KERNEL;
+	if (pFirmware->mnConfigurations == 0 ||
+		pFirmware->mnConfigurations > maxConf) {
+		dev_err(tas_dev->dev, "%s: mnConfigurations is invalid\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+
+	if (offset + 4 * maxConf > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mpConfigurations error\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+
+	pFirmware->mpConfigurations = kcalloc(pFirmware->mnConfigurations,
+		sizeof(struct TConfiguration), GFP_KERNEL);
+	if (pFirmware->mpConfigurations == NULL) {
+		offset = -1;
+		goto out;
+	}
+
+	for (nConfiguration = 0; nConfiguration < pFirmware->mnPrograms;
+		nConfiguration++) {
+		pConfiguration =
+			&(pFirmware->mpConfigurations[nConfiguration]);
+		pConfiguration->cfg_size = SMS_HTONL(buf[offset],
+			buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+		offset  += 4;
+	}
+
+	offset  += (4 * (maxConf - nConfiguration));
+	pFirmware->prog_start_offset = offset;
+	pFirmware->cfg_start_offset  += offset;
+out:
+	return offset;
+}
+
+int tasdevice_load_block_kernel(struct tasdevice_priv *pTAS2781,
+	struct TBlock *pBlock)
+{
+	int nResult = 0;
+
+	unsigned char *pData = pBlock->mpData;
+	unsigned int i = 0, length = 0;
+	const unsigned int blk_size = pBlock->blk_size;
+	unsigned char dev_idx = 0;
+	struct tasdevice_dspfw_hdr *pFw_hdr = &(pTAS2781->mpFirmware->fw_hdr);
+	struct tasdevice_fw_fixed_hdr *pFw_fixed_hdr = &(pFw_hdr->mnFixedHdr);
+
+	if (pFw_fixed_hdr->mnPPCVersion >= PPC3_VERSION) {
+		switch (pBlock->mnType) {
+		case MAIN_ALL_DEVICES_1X:
+			dev_idx = 0|0x80;
+			break;
+		case MAIN_DEVICE_A_1X:
+			dev_idx = 1|0x80;
+			break;
+		case COEFF_DEVICE_A_1X:
+		case PRE_DEVICE_A_1X:
+			dev_idx = 1|0xC0;
+			break;
+		case MAIN_DEVICE_B_1X:
+			dev_idx = 2|0x80;
+			break;
+		case COEFF_DEVICE_B_1X:
+		case PRE_DEVICE_B_1X:
+			dev_idx = 2|0xC0;
+			break;
+		case MAIN_DEVICE_C_1X:
+			dev_idx = 3|0x80;
+			break;
+		case COEFF_DEVICE_C_1X:
+		case PRE_DEVICE_C_1X:
+			dev_idx = 3|0xC0;
+			break;
+		case MAIN_DEVICE_D_1X:
+			dev_idx = 4|0x80;
+			break;
+		case COEFF_DEVICE_D_1X:
+		case PRE_DEVICE_D_1X:
+			dev_idx = 4|0xC0;
+			break;
+		default:
+			dev_info(pTAS2781->dev,
+				"%s: TAS2781 load block: Other Type = 0x%02x\n",
+				__func__, pBlock->mnType);
+			break;
+		}
+	} else {
+		switch (pBlock->mnType) {
+		case MAIN_ALL_DEVICES:
+			dev_idx = 0|0x80;
+			break;
+		case MAIN_DEVICE_A:
+			dev_idx = 1|0x80;
+			break;
+		case COEFF_DEVICE_A:
+		case PRE_DEVICE_A:
+			dev_idx = 1|0xC0;
+			break;
+		case MAIN_DEVICE_B:
+			dev_idx = 2|0x80;
+			break;
+		case COEFF_DEVICE_B:
+		case PRE_DEVICE_B:
+			dev_idx = 2|0xC0;
+			break;
+		case MAIN_DEVICE_C:
+			dev_idx = 3|0x80;
+			break;
+		case COEFF_DEVICE_C:
+		case PRE_DEVICE_C:
+			dev_idx = 3|0xC0;
+			break;
+		case MAIN_DEVICE_D:
+			dev_idx = 4|0x80;
+			break;
+		case COEFF_DEVICE_D:
+		case PRE_DEVICE_D:
+			dev_idx = 4|0xC0;
+			break;
+		default:
+			dev_info(pTAS2781->dev,
+				"%s: TAS2781 load block: Other Type = 0x%02x\n",
+				__func__,
+				pBlock->mnType);
+			break;
+		}
+	}
+
+	for (i = 0; i < pBlock->nSublocks; i++) {
+		int rc = tasdevice_process_block(pTAS2781, pData + length,
+			dev_idx, blk_size - length);
+		if (rc < 0) {
+			dev_err(pTAS2781->dev,
+				"%s: ERROR:%u %u sublock write error\n",
+				__func__, length, blk_size);
+			break;
+		}
+		length  += (unsigned int)rc;
+		if (blk_size < length) {
+			dev_err(pTAS2781->dev,
+				"%s: ERROR:%u %u out of memory\n",
+				__func__, length, blk_size);
+			break;
+		}
+	}
+
+	return nResult;
+}
+
+int fw_parse_variable_header_git(struct tasdevice_priv *tas_dev,
+	const struct firmware *pFW, int offset)
+{
+	const unsigned char *buf = pFW->data;
+	struct TFirmware *pFirmware = tas_dev->mpFirmware;
+	struct tasdevice_dspfw_hdr *pFw_hdr = &(pFirmware->fw_hdr);
+	int i = strlen((char *)&buf[offset]);
+
+	i++;
+
+	if (offset + i > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+
+	pFw_hdr->mpDescription = kmemdup(&buf[offset], i, GFP_KERNEL);
+	if (pFw_hdr->mpDescription == NULL)
+		goto out;
+	offset  += i;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_hdr->mnDeviceFamily = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	if (pFw_hdr->mnDeviceFamily != 0) {
+		dev_err(tas_dev->dev, "ERROR:%s: not TAS device\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_hdr->mnDevice = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	if (pFw_hdr->mnDevice >= TASDEVICE_DSP_TAS_MAX_DEVICE ||
+		pFw_hdr->mnDevice == 6) {
+		dev_err(tas_dev->dev, "ERROR:%s: not support device %d\n",
+			__func__, pFw_hdr->mnDevice);
+		offset = -1;
+		goto out;
+	}
+	offset  += 4;
+	pFw_hdr->ndev = deviceNumber[pFw_hdr->mnDevice];
+	if (pFw_hdr->ndev != tas_dev->ndev) {
+		dev_err(tas_dev->dev,
+			"%s: ndev(%u) in dspbin dismatch ndev(%u) in DTS\n",
+			__func__, pFw_hdr->ndev,
+			tas_dev->ndev);
+		offset = -1;
+	}
+
+out:
+	return offset;
+}
+
+int fw_parse_variable_header_cal(struct tasdevice_priv *tas_dev,
+	struct TFirmware *pCalFirmware, const struct firmware *pFW, int offset)
+{
+	const unsigned char *buf = pFW->data;
+	struct tasdevice_dspfw_hdr *pFw_hdr = &(pCalFirmware->fw_hdr);
+	int i = strlen((char *)&buf[offset]);
+
+	i++;
+
+	if (offset + i > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+
+	pFw_hdr->mpDescription = kmemdup(&buf[offset], i, GFP_KERNEL);
+	if (pFw_hdr->mpDescription == NULL)
+		goto out;
+	offset  += i;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnDeviceFamily error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_hdr->mnDeviceFamily = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	if (pFw_hdr->mnDeviceFamily != 0) {
+		dev_err(tas_dev->dev, "ERROR:%s: not TAS device\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnDevice error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_hdr->mnDevice = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	if (pFw_hdr->mnDevice >= TASDEVICE_DSP_TAS_MAX_DEVICE ||
+		pFw_hdr->mnDevice == 6) {
+		dev_err(tas_dev->dev, "ERROR:%s: not support device %d\n",
+			__func__, pFw_hdr->mnDevice);
+		offset = -1;
+		goto out;
+	}
+	offset  += 4;
+	pFw_hdr->ndev = deviceNumber[pFw_hdr->mnDevice];
+	if (pFw_hdr->ndev != 1) {
+		dev_err(tas_dev->dev,
+			"%s: calbin must be 1, but currently ndev(%u)\n",
+			__func__, pFw_hdr->ndev);
+		offset = -1;
+	}
+
+out:
+	return offset;
+}
+
+static inline void tas2781_clear_Calfirmware(struct TFirmware
+	*mpCalFirmware)
+{
+	int i = 0;
+	unsigned int nBlock = 0;
+
+	if (mpCalFirmware->mpCalibrations) {
+		struct TCalibration *pCalibration;
+
+		for (i = 0; i < mpCalFirmware->mnCalibrations; i++) {
+			pCalibration = &(mpCalFirmware->mpCalibrations[i]);
+			if (pCalibration) {
+				struct TData *pImageData =
+					&(pCalibration->mData);
+
+				if (pImageData->mpBlocks) {
+					struct TBlock *pBlock;
+
+					for (nBlock = 0; nBlock <
+						pImageData->mnBlocks;
+						nBlock++) {
+						pBlock =
+						&(pImageData->mpBlocks[nBlock]);
+						kfree(pBlock->mpData);
+					}
+				kfree(pImageData->mpBlocks);
+				}
+				kfree(pCalibration->mpDescription);
+			}
+		}
+		kfree(mpCalFirmware->mpCalibrations);
+	}
+	kfree(mpCalFirmware);
+}
+
+static int fw_parse_block_data(struct TFirmware *pFirmware,
+	struct TBlock *pBlock, const struct firmware *pFW, int offset)
+{
+	unsigned char *pData = (unsigned char *)pFW->data;
+	int n;
+
+	if (offset + 4 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mnType error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mnType = SMS_HTONL(pData[offset],
+		pData[offset + 1], pData[offset + 2], pData[offset + 3]);
+	offset  += 4;
+
+	if (pFirmware->fw_hdr.mnFixedHdr.mnDriverVersion >=
+		PPC_DRIVER_CRCCHK) {
+		if (offset + 1 > pFW->size) {
+			dev_err(pFirmware->dev, "%s: mbPChkSumPresent error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pBlock->mbPChkSumPresent = pData[offset];
+		offset++;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(pFirmware->dev, "%s: mnPChkSum error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pBlock->mnPChkSum = pData[offset];
+		offset++;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(pFirmware->dev, "%s: mbYChkSumPresent error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pBlock->mbYChkSumPresent = pData[offset];
+		offset++;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(pFirmware->dev, "%s: mnYChkSum error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pBlock->mnYChkSum = pData[offset];
+		offset++;
+	} else {
+		pBlock->mbPChkSumPresent = 0;
+		pBlock->mbYChkSumPresent = 0;
+	}
+	if (offset + 4 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mnCommands error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mnCommands = SMS_HTONL(pData[offset],
+		pData[offset + 1], pData[offset + 2], pData[offset + 3]);
+	offset  += 4;
+
+	n = pBlock->mnCommands * 4;
+	if (offset + n > pFW->size) {
+		dev_err(pFirmware->dev,
+			"%s: File Size(%u) error offset = %d n = %d\n",
+			__func__, pFW->size, offset, n);
+		offset = -1;
+		goto out;
+	}
+	pBlock->mpData = kmemdup(&pData[offset], n, GFP_KERNEL);
+	if (pBlock->mpData == NULL) {
+		offset = -1;
+		goto out;
+	}
+	offset  += n;
+out:
+	return offset;
+}
+
+static int fw_parse_data(struct TFirmware *pFirmware,
+	struct TData *pImageData, const struct firmware *pFW, int offset)
+{
+	const unsigned char *pData = (unsigned char *)pFW->data;
+	int n = 0;
+	unsigned int nBlock;
+
+	if (offset + 64 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mpName error\n", __func__);
+		n = -1;
+		goto out;
+	}
+	memcpy(pImageData->mpName, &pData[offset], 64);
+	offset  += 64;
+
+	n = strlen((char *)&pData[offset]);
+	n++;
+	if (offset + n > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mpDescription error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pImageData->mpDescription = kmemdup(pData, n, GFP_KERNEL);
+	if (pImageData->mpDescription == NULL)
+		goto out;
+	offset  += n;
+
+	if (offset + 2 > pFW->size) {
+		dev_err(pFirmware->dev, "%s: mnBlocks error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pImageData->mnBlocks = SMS_HTONS(pData[offset], pData[offset + 1]);
+	offset  += 2;
+
+	pImageData->mpBlocks =
+		kcalloc(pImageData->mnBlocks, sizeof(struct TBlock),
+			GFP_KERNEL);
+	if (pImageData->mpBlocks == NULL) {
+		dev_err(pFirmware->dev, "%s: FW memory failed!\n", __func__);
+		goto out;
+	}
+	for (nBlock = 0; nBlock < pImageData->mnBlocks; nBlock++) {
+		offset = fw_parse_block_data(pFirmware,
+			&(pImageData->mpBlocks[nBlock]), pFW, offset);
+		if (offset < 0) {
+			offset = -1;
+			goto out;
+		}
+	}
+out:
+	return offset;
+}
+
+static int fw_parse_calibration_data(struct tasdevice_priv *tas_dev,
+	struct TFirmware *pFirmware, const struct firmware *pFW, int offset)
+{
+	unsigned char *pData = (unsigned char *)pFW->data;
+	unsigned int n = 0;
+	unsigned int nCalibration = 0;
+	struct TCalibration *pCalibration = NULL;
+
+	if (offset + 2 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnCalibrations error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFirmware->mnCalibrations = SMS_HTONS(pData[offset],
+		pData[offset + 1]);
+	offset  += 2;
+
+	if (pFirmware->mnCalibrations != 1) {
+		dev_err(tas_dev->dev,
+			"%s: only support one calibraiton(%d)!\n",
+			__func__, pFirmware->mnCalibrations);
+		goto out;
+	}
+
+	pFirmware->mpCalibrations =
+		kcalloc(pFirmware->mnCalibrations, sizeof(struct TCalibration),
+			GFP_KERNEL);
+	if (pFirmware->mpCalibrations == NULL) {
+		dev_err(tas_dev->dev, "%s: mpCalibrations memory failed!\n",
+			__func__);
+		offset = -1;
+		goto out;
+	}
+	for (nCalibration = 0; nCalibration < pFirmware->mnCalibrations;
+		nCalibration++) {
+		if (offset + 64 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpCalibrations error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pCalibration = &(pFirmware->mpCalibrations[nCalibration]);
+		memcpy(pCalibration->mpName, &pData[offset], 64);
+		offset  += 64;
+
+		n = strlen((char *)&pData[offset]);
+		n++;
+		if (offset + n > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpDescription error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pCalibration->mpDescription = kmemdup(&pData[offset], n,
+			GFP_KERNEL);
+		if (pCalibration->mpDescription == NULL) {
+			offset = -1;
+			goto out;
+		}
+		offset  += n;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev,
+				"%s: mnProgram error, offset = %d\n", __func__,
+				offset);
+			offset = -1;
+			goto out;
+		}
+		pCalibration->mnProgram = pData[offset];
+		offset++;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev,
+				"%s: mnConfiguration error, offset = %d\n",
+				__func__,
+				offset);
+			offset = -1;
+			goto out;
+		}
+		pCalibration->mnConfiguration = pData[offset];
+		offset++;
+
+		offset = fw_parse_data(pFirmware, &(pCalibration->mData), pFW,
+			offset);
+		if (offset < 0)
+			goto out;
+	}
+
+out:
+
+	return offset;
+}
+
+static int fw_parse_program_data(struct tasdevice_priv *tas_dev,
+	struct TFirmware *pFirmware, const struct firmware *pFW, int offset)
+{
+	struct TProgram *pProgram;
+	unsigned char *buf = (unsigned char *)pFW->data;
+	int nProgram = 0;
+
+	if (offset + 2 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFirmware->mnPrograms = SMS_HTONS(buf[offset], buf[offset + 1]);
+	offset  += 2;
+
+	if (pFirmware->mnPrograms == 0) {
+		dev_err(tas_dev->dev, "%s: mnPrograms is null, maybe calbin\n",
+			__func__);
+		//Do not "offset = -1;", because of calbin
+		goto out;
+	}
+
+	pFirmware->mpPrograms =
+		kcalloc(pFirmware->mnPrograms, sizeof(struct TProgram),
+			GFP_KERNEL);
+	if (pFirmware->mpPrograms == NULL) {
+		offset = -1;
+		goto out;
+	}
+	for (nProgram = 0; nProgram < pFirmware->mnPrograms; nProgram++) {
+		int n = 0;
+
+		pProgram = &(pFirmware->mpPrograms[nProgram]);
+		if (offset + 64 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpName error\n", __func__);
+			offset = -1;
+			goto out;
+		}
+		memcpy(pProgram->mpName, &buf[offset], 64);
+		offset  += 64;
+
+		n = strlen((char *)&buf[offset]);
+		n++;
+		if (offset + n > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpDescription error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mpDescription = kmemdup(&buf[offset], n, GFP_KERNEL);
+		if (pProgram->mpDescription == NULL) {
+			offset = -1;
+			goto out;
+		}
+
+		offset  += n;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnAppMode error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnAppMode = buf[offset];
+		offset++;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPDMI2SMode error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnPDMI2SMode = buf[offset];
+		offset++;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnISnsPD error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnISnsPD = buf[offset];
+		offset++;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnVSnsPD error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnVSnsPD = buf[offset];
+		offset++;
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPowerLDG error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pProgram->mnPowerLDG = buf[offset];
+		offset++;
+
+		offset = fw_parse_data(pFirmware, &(pProgram->mData), pFW,
+			offset);
+		if (offset < 0)
+			goto out;
+	}
+out:
+	return offset;
+}
+
+static int fw_parse_configuration_data(
+	struct tasdevice_priv *tas_dev,
+	struct TFirmware *pFirmware,
+	const struct firmware *pFW, int offset)
+{
+	unsigned char *pData = (unsigned char *)pFW->data;
+	int n;
+	unsigned int nConfiguration;
+	struct TConfiguration *pConfiguration;
+
+	if (offset + 2 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFirmware->mnConfigurations = SMS_HTONS(pData[offset],
+		pData[offset + 1]);
+	offset  += 2;
+
+	if (pFirmware->mnConfigurations == 0) {
+		dev_err(tas_dev->dev, "%s: mnConfigurations is zero\n",
+			__func__);
+		//Do not "offset = -1;", because of calbin
+		goto out;
+	}
+	pFirmware->mpConfigurations =
+		kcalloc(pFirmware->mnConfigurations,
+				sizeof(struct TConfiguration), GFP_KERNEL);
+
+	for (nConfiguration = 0; nConfiguration < pFirmware->mnConfigurations;
+		nConfiguration++) {
+		pConfiguration =
+			&(pFirmware->mpConfigurations[nConfiguration]);
+		if (offset + 64 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: File Size error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		memcpy(pConfiguration->mpName, &pData[offset], 64);
+		offset  += 64;
+
+		n = strlen((char *)&pData[offset]);
+		n++;
+		if (offset + n > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mpDescription error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mpDescription = kmemdup(&pData[offset], n,
+			GFP_KERNEL);
+
+		if (pConfiguration->mpDescription == NULL) {
+			dev_err(tas_dev->dev, "%s: FW memory failed!\n",
+				__func__);
+			goto out;
+		}
+		offset  += n;
+		if (offset + 2 > pFW->size) {
+			dev_err(tas_dev->dev,
+				"%s: mnDevice_orientation error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnDevice_orientation = pData[offset];
+
+		pConfiguration->mnDevices = pData[offset + 1];
+		offset  += 2;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mProgram error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mProgram = pData[offset];
+		offset++;
+
+		if (offset + 4 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnSamplingRate error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnSamplingRate = SMS_HTONL(pData[offset],
+			pData[offset + 1], pData[offset + 2],
+			pData[offset + 3]);
+		offset  += 4;
+
+		if (offset + 1 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPLLSrc error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnPLLSrc = pData[offset];
+		offset++;
+
+		if (offset + 4 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnPLLSrcRate error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnPLLSrcRate = SMS_HTONL(pData[offset],
+			pData[offset + 1], pData[offset + 2],
+			pData[offset + 3]);
+		offset  += 4;
+
+		if (offset + 2 > pFW->size) {
+			dev_err(tas_dev->dev, "%s: mnFsRate error\n",
+				__func__);
+			offset = -1;
+			goto out;
+		}
+		pConfiguration->mnFsRate = SMS_HTONS(pData[offset],
+			pData[offset + 1]);
+		offset  += 2;
+
+		offset = fw_parse_data(pFirmware, &(pConfiguration->mData),
+			pFW, offset);
+		if (offset < 0)
+			goto out;
+	}
+out:
+	return offset;
+}
+
+static int fw_parse_header(struct tasdevice_priv *tas_dev,
+	struct TFirmware *pFirmware, const struct firmware *pFW, int offset)
+{
+	struct tasdevice_dspfw_hdr *pFw_hdr = &(pFirmware->fw_hdr);
+	struct tasdevice_fw_fixed_hdr *pFw_fixed_hdr = &(pFw_hdr->mnFixedHdr);
+	const unsigned char *buf = (unsigned char *)pFW->data;
+	int i = 0;
+	unsigned char pMagicNumber[] = { 0x35, 0x35, 0x35, 0x32 };
+
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -EINVAL;
+		goto out;
+	}
+	if (memcmp(&buf[offset], pMagicNumber, 4)) {
+		dev_err(tas_dev->dev, "%s: Magic number doesn't match",
+			__func__);
+		offset = -EINVAL;
+		goto out;
+	}
+	pFw_fixed_hdr->mnMagicNumber = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnFWSize error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_fixed_hdr->mnFWSize = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: File Size error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	if (pFw_fixed_hdr->mnFWSize != pFW->size) {
+		dev_err(tas_dev->dev, "File size not match, %d %d", pFW->size,
+			pFw_fixed_hdr->mnFWSize);
+		offset = -1;
+		goto out;
+	}
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnChecksum error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_fixed_hdr->mnChecksum = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnPPCVersion error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_fixed_hdr->mnPPCVersion = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnFWVersion error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_fixed_hdr->mnFWVersion = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnDriverVersion error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	pFw_fixed_hdr->mnDriverVersion = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 4 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mnTimeStamp error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	for (i = 0; i < sizeof(BinFileformatVerInfo) /
+		sizeof(BinFileformatVerInfo[0]); i++) {
+		if (BinFileformatVerInfo[i][DRVFWVER] ==
+			pFw_fixed_hdr->mnDriverVersion) {
+			pFw_hdr->mnBinFileDocVer =
+				BinFileformatVerInfo[i][BINFILEDOCVER];
+			break;
+		}
+	}
+	pFw_fixed_hdr->mnTimeStamp = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (offset + 64 > pFW->size) {
+		dev_err(tas_dev->dev, "%s: mpDDCName error\n", __func__);
+		offset = -1;
+		goto out;
+	}
+	memcpy(pFw_fixed_hdr->mpDDCName, &buf[offset], 64);
+	offset  += 64;
+
+ out:
+	return offset;
+}
+
+static const unsigned char crc8_lookup_table[CRC8_TABLE_SIZE] = {
+	0x00, 0x4D, 0x9A, 0xD7, 0x79, 0x34, 0xE3, 0xAE,
+	0xF2, 0xBF, 0x68, 0x25, 0x8B, 0xC6, 0x11, 0x5C,
+	0xA9, 0xE4, 0x33, 0x7E, 0xD0, 0x9D, 0x4A, 0x07,
+	0x5B, 0x16, 0xC1, 0x8C, 0x22, 0x6F, 0xB8, 0xF5,
+	0x1F, 0x52, 0x85, 0xC8, 0x66, 0x2B, 0xFC, 0xB1,
+	0xED, 0xA0, 0x77, 0x3A, 0x94, 0xD9, 0x0E, 0x43,
+	0xB6, 0xFB, 0x2C, 0x61, 0xCF, 0x82, 0x55, 0x18,
+	0x44, 0x09, 0xDE, 0x93, 0x3D, 0x70, 0xA7, 0xEA,
+	0x3E, 0x73, 0xA4, 0xE9, 0x47, 0x0A, 0xDD, 0x90,
+	0xCC, 0x81, 0x56, 0x1B, 0xB5, 0xF8, 0x2F, 0x62,
+	0x97, 0xDA, 0x0D, 0x40, 0xEE, 0xA3, 0x74, 0x39,
+	0x65, 0x28, 0xFF, 0xB2, 0x1C, 0x51, 0x86, 0xCB,
+	0x21, 0x6C, 0xBB, 0xF6, 0x58, 0x15, 0xC2, 0x8F,
+	0xD3, 0x9E, 0x49, 0x04, 0xAA, 0xE7, 0x30, 0x7D,
+	0x88, 0xC5, 0x12, 0x5F, 0xF1, 0xBC, 0x6B, 0x26,
+	0x7A, 0x37, 0xE0, 0xAD, 0x03, 0x4E, 0x99, 0xD4,
+	0x7C, 0x31, 0xE6, 0xAB, 0x05, 0x48, 0x9F, 0xD2,
+	0x8E, 0xC3, 0x14, 0x59, 0xF7, 0xBA, 0x6D, 0x20,
+	0xD5, 0x98, 0x4F, 0x02, 0xAC, 0xE1, 0x36, 0x7B,
+	0x27, 0x6A, 0xBD, 0xF0, 0x5E, 0x13, 0xC4, 0x89,
+	0x63, 0x2E, 0xF9, 0xB4, 0x1A, 0x57, 0x80, 0xCD,
+	0x91, 0xDC, 0x0B, 0x46, 0xE8, 0xA5, 0x72, 0x3F,
+	0xCA, 0x87, 0x50, 0x1D, 0xB3, 0xFE, 0x29, 0x64,
+	0x38, 0x75, 0xA2, 0xEF, 0x41, 0x0C, 0xDB, 0x96,
+	0x42, 0x0F, 0xD8, 0x95, 0x3B, 0x76, 0xA1, 0xEC,
+	0xB0, 0xFD, 0x2A, 0x67, 0xC9, 0x84, 0x53, 0x1E,
+	0xEB, 0xA6, 0x71, 0x3C, 0x92, 0xDF, 0x08, 0x45,
+	0x19, 0x54, 0x83, 0xCE, 0x60, 0x2D, 0xFA, 0xB7,
+	0x5D, 0x10, 0xC7, 0x8A, 0x24, 0x69, 0xBE, 0xF3,
+	0xAF, 0xE2, 0x35, 0x78, 0xD6, 0x9B, 0x4C, 0x01,
+	0xF4, 0xB9, 0x6E, 0x23, 0x8D, 0xC0, 0x17, 0x5A,
+	0x06, 0x4B, 0x9C, 0xD1, 0x7F, 0x32, 0xE5, 0xA8
+};
+
+static int isInPageYRAM(struct tasdevice_priv *pTAS2781,
+	struct TYCRC *pCRCData,
+	unsigned char nBook, unsigned char nPage,
+	unsigned char nReg, unsigned char len)
+{
+	int nResult = 0;
+
+	if (nBook == TAS2781_YRAM_BOOK1) {
+		if (nPage == TAS2781_YRAM1_PAGE) {
+			if (nReg >= TAS2781_YRAM1_START_REG) {
+				pCRCData->mnOffset = nReg;
+				pCRCData->mnLen = len;
+				nResult = 1;
+			} else if ((nReg + len) > TAS2781_YRAM1_START_REG) {
+				pCRCData->mnOffset = TAS2781_YRAM1_START_REG;
+				pCRCData->mnLen =
+				len - (TAS2781_YRAM1_START_REG - nReg);
+				nResult = 1;
+			} else
+				nResult = 0;
+		} else if (nPage == TAS2781_YRAM3_PAGE) {
+			if (nReg > TAS2781_YRAM3_END_REG) {
+				nResult = 0;
+			} else if (nReg >= TAS2781_YRAM3_START_REG) {
+				if ((nReg + len) > TAS2781_YRAM3_END_REG) {
+					pCRCData->mnOffset = nReg;
+					pCRCData->mnLen =
+					TAS2781_YRAM3_END_REG - nReg + 1;
+					nResult = 1;
+				} else {
+					pCRCData->mnOffset = nReg;
+					pCRCData->mnLen = len;
+					nResult = 1;
+				}
+			} else {
+				if ((nReg + (len-1)) <
+					TAS2781_YRAM3_START_REG)
+					nResult = 0;
+				else {
+					pCRCData->mnOffset =
+					TAS2781_YRAM3_START_REG;
+					pCRCData->mnLen =
+					len - (TAS2781_YRAM3_START_REG - nReg);
+					nResult = 1;
+				}
+			}
+		}
+	} else if (nBook ==
+		TAS2781_YRAM_BOOK2) {
+		if (nPage == TAS2781_YRAM5_PAGE) {
+			if (nReg > TAS2781_YRAM5_END_REG) {
+				nResult = 0;
+			} else if (nReg >= TAS2781_YRAM5_START_REG) {
+				if ((nReg + len) > TAS2781_YRAM5_END_REG) {
+					pCRCData->mnOffset = nReg;
+					pCRCData->mnLen =
+					TAS2781_YRAM5_END_REG - nReg + 1;
+					nResult = 1;
+				} else {
+					pCRCData->mnOffset = nReg;
+					pCRCData->mnLen = len;
+					nResult = 1;
+				}
+			} else {
+				if ((nReg + (len-1)) <
+					TAS2781_YRAM5_START_REG)
+					nResult = 0;
+				else {
+					pCRCData->mnOffset =
+					TAS2781_YRAM5_START_REG;
+					pCRCData->mnLen =
+					len - (TAS2781_YRAM5_START_REG - nReg);
+					nResult = 1;
+				}
+			}
+		}
+	} else
+		nResult = 0;
+
+	return nResult;
+}
+
+static int isInBlockYRAM(struct tasdevice_priv *pTAS2781,
+	struct TYCRC *pCRCData,
+	unsigned char nBook, unsigned char nPage,
+	unsigned char nReg, unsigned char len)
+{
+	int nResult = 0;
+
+	if (nBook == TAS2781_YRAM_BOOK1) {
+		if (nPage < TAS2781_YRAM2_START_PAGE)
+			nResult = 0;
+		else if (nPage <= TAS2781_YRAM2_END_PAGE) {
+			if (nReg > TAS2781_YRAM2_END_REG)
+				nResult = 0;
+			else if (nReg >= TAS2781_YRAM2_START_REG) {
+				pCRCData->mnOffset = nReg;
+				pCRCData->mnLen = len;
+				nResult = 1;
+			} else {
+				if ((nReg + (len-1)) <
+					TAS2781_YRAM2_START_REG)
+					nResult = 0;
+				else {
+					pCRCData->mnOffset =
+					TAS2781_YRAM2_START_REG;
+					pCRCData->mnLen =
+					nReg + len - TAS2781_YRAM2_START_REG;
+					nResult = 1;
+				}
+			}
+		} else
+			nResult = 0;
+	} else if (nBook ==
+		TAS2781_YRAM_BOOK2) {
+		if (nPage < TAS2781_YRAM4_START_PAGE)
+			nResult = 0;
+		else if (nPage <= TAS2781_YRAM4_END_PAGE) {
+			if (nReg > TAS2781_YRAM2_END_REG)
+				nResult = 0;
+			else if (nReg >= TAS2781_YRAM2_START_REG) {
+				pCRCData->mnOffset = nReg;
+				pCRCData->mnLen = len;
+				nResult = 1;
+			} else {
+				if ((nReg + (len-1))
+					< TAS2781_YRAM2_START_REG)
+					nResult = 0;
+				else {
+					pCRCData->mnOffset =
+					TAS2781_YRAM2_START_REG;
+					pCRCData->mnLen =
+					nReg + len - TAS2781_YRAM2_START_REG;
+					nResult = 1;
+				}
+			}
+		} else
+			nResult = 0;
+	} else
+		nResult = 0;
+
+	return nResult;
+}
+
+static int isYRAM(struct tasdevice_priv *pTAS2781, struct TYCRC *pCRCData,
+	unsigned char nBook, unsigned char nPage,
+	unsigned char nReg, unsigned char len)
+{
+	int nResult = 0;
+
+	nResult = isInPageYRAM(pTAS2781, pCRCData, nBook, nPage, nReg, len);
+	if (nResult == 0)
+		nResult = isInBlockYRAM(pTAS2781, pCRCData, nBook,
+				nPage, nReg, len);
+
+	return nResult;
+}
+
+/*
+ * crc8-calculate a crc8 over the given input data.
+ *
+ * table: crc table used for calculation.
+ * pdata: pointer to data buffer.
+ * nbytes: number of bytes in data buffer.
+ * crc: previous returned crc8 value.
+ */
+static u8 ti_crc8(const u8 table[CRC8_TABLE_SIZE], u8 *pdata,
+			size_t nbytes, u8 crc)
+{
+	/*loop over the buffer data */
+	while (nbytes-- > 0)
+		crc = table[(crc ^ *(pdata  += 1)) & 0xff];
+
+	return crc;
+}
+
+static int doSingleRegCheckSum(struct tasdevice_priv *pTAS2781,
+	enum channel chl,
+		unsigned char nBook, unsigned char nPage,
+		unsigned char nReg, unsigned char nValue)
+{
+	int nResult = 0;
+	struct TYCRC sCRCData;
+	unsigned int nData1 = 0;
+
+	if ((nBook == TASDEVICE_BOOK_ID(TAS2781_SA_COEFF_SWAP_REG))
+		&& (nPage == TASDEVICE_PAGE_ID(TAS2781_SA_COEFF_SWAP_REG))
+		&& (nReg >= TASDEVICE_PAGE_REG(TAS2781_SA_COEFF_SWAP_REG))
+		&& (nReg <= (TASDEVICE_PAGE_REG(
+		TAS2781_SA_COEFF_SWAP_REG) + 4))) {
+		/*DSP swap command, pass */
+		nResult = 0;
+		goto end;
+	}
+
+	nResult = isYRAM(pTAS2781, &sCRCData, nBook, nPage, nReg, 1);
+	if (nResult == 1) {
+		nResult = pTAS2781->read(pTAS2781, chl,
+				TASDEVICE_REG(nBook, nPage, nReg), &nData1);
+		if (nResult < 0)
+			goto end;
+
+		if (nData1 != nValue) {
+			dev_err(pTAS2781->dev,
+				"error2, B[0x%x]P[0x%x]R[0x%x] W[0x%x], R[0x%x]\n",
+				nBook, nPage, nReg,
+				nValue, nData1);
+			nResult = -EAGAIN;
+			pTAS2781->tasdevice[chl].mnErrCode |=
+				ERROR_YRAM_CRCCHK;
+			goto end;
+		}
+
+		if (nData1 != nValue) {
+			dev_err(pTAS2781->dev,
+				"error2, B[0x%x]P[0x%x]R[0x%x] W[0x%x], R[0x%x]\n",
+				nBook, nPage, nReg,
+				nValue, nData1);
+			nResult = -EAGAIN;
+			goto end;
+		}
+
+		nResult = ti_crc8(crc8_lookup_table, &nValue, 1, 0);
+	}
+
+end:
+	return nResult;
+}
+
+static int doMultiRegCheckSum(struct tasdevice_priv *pTAS2781,
+	enum channel chn, unsigned char nBook, unsigned char nPage,
+	unsigned char nReg, unsigned int len)
+{
+	int nResult = 0, i = 0;
+	unsigned char nCRCChkSum = 0;
+	unsigned char nBuf1[128] = {0};
+	struct TYCRC TCRCData;
+
+	if ((nReg + len-1) > 127) {
+		nResult = -EINVAL;
+		dev_err(pTAS2781->dev, "firmware error\n");
+		goto end;
+	}
+
+	if ((nBook == TASDEVICE_BOOK_ID(TAS2781_SA_COEFF_SWAP_REG))
+		&& (nPage == TASDEVICE_PAGE_ID(TAS2781_SA_COEFF_SWAP_REG))
+		&& (nReg == TASDEVICE_PAGE_REG(TAS2781_SA_COEFF_SWAP_REG))
+		&& (len == 4)) {
+		/*DSP swap command, pass */
+		nResult = 0;
+		goto end;
+	}
+
+	nResult = isYRAM(pTAS2781, &TCRCData, nBook, nPage, nReg, len);
+	dev_info(pTAS2781->dev,
+		"isYRAM: nBook 0x%x, nPage 0x%x, nReg 0x%x\n",
+		nBook, nPage, nReg);
+	dev_info(pTAS2781->dev,
+		"isYRAM: TCRCData.mnLen 0x%x, len 0x%x, nResult %d\n",
+		TCRCData.mnLen, len, nResult);
+	dev_info(pTAS2781->dev, "TCRCData.mnOffset %x\n", TCRCData.mnOffset);
+	if (nResult == 1) {
+		if (len == 1) {
+			dev_err(pTAS2781->dev, "firmware error\n");
+			nResult = -EINVAL;
+			goto end;
+		} else {
+			nResult = pTAS2781->bulk_read(pTAS2781, chn,
+				TASDEVICE_REG(nBook, nPage, TCRCData.mnOffset),
+				nBuf1, TCRCData.mnLen);
+			if (nResult < 0)
+				goto end;
+
+			for (i = 0; i < TCRCData.mnLen; i++) {
+				if ((nBook == TASDEVICE_BOOK_ID(
+					TAS2781_SA_COEFF_SWAP_REG))
+					&& (nPage == TASDEVICE_PAGE_ID(
+						TAS2781_SA_COEFF_SWAP_REG))
+					&& ((i + TCRCData.mnOffset)
+					>= TASDEVICE_PAGE_REG(
+						TAS2781_SA_COEFF_SWAP_REG))
+					&& ((i + TCRCData.mnOffset)
+					<= (TASDEVICE_PAGE_REG(
+						TAS2781_SA_COEFF_SWAP_REG)
+						+ 4))) {
+					/*DSP swap command, bypass */
+					continue;
+				} else
+					nCRCChkSum  +=
+					ti_crc8(crc8_lookup_table, &nBuf1[i],
+						1, 0);
+			}
+
+			nResult = nCRCChkSum;
+		}
+	}
+
+end:
+	return nResult;
+}
+
+static int tasdevice_load_block(struct tasdevice_priv *tas_dev,
+				struct TBlock *pBlock)
+{
+	int nResult = 0;
+	unsigned int nCommand = 0;
+	unsigned char nBook = 0;
+	unsigned char nPage = 0;
+	unsigned char nOffset = 0;
+	unsigned char nData = 0;
+	unsigned int nLength = 0;
+	unsigned int nSleep = 0;
+	unsigned char nCRCChkSum = 0;
+	unsigned int nValue = 0;
+	int nRetry = 6;
+	unsigned char *pData = pBlock->mpData;
+	int chn = 0, chnend = 0;
+
+	dev_info(tas_dev->dev,
+		"TAS2781 load block: Type = %d, commands = %d\n",
+		pBlock->mnType, pBlock->mnCommands);
+	switch (pBlock->mnType) {
+	case MAIN_ALL_DEVICES:
+		chn = 0;
+		chnend = tas_dev->ndev;
+		break;
+	case MAIN_DEVICE_A:
+	case COEFF_DEVICE_A:
+	case PRE_DEVICE_A:
+		chn = 0;
+		chnend = 1;
+		break;
+	case MAIN_DEVICE_B:
+	case COEFF_DEVICE_B:
+	case PRE_DEVICE_B:
+		chn = 1;
+		chnend = 2;
+		break;
+	case MAIN_DEVICE_C:
+	case COEFF_DEVICE_C:
+	case PRE_DEVICE_C:
+		chn = 2;
+		chnend = 3;
+		break;
+	case MAIN_DEVICE_D:
+	case COEFF_DEVICE_D:
+	case PRE_DEVICE_D:
+		chn = 3;
+		chnend = 4;
+		break;
+	default:
+		dev_info(tas_dev->dev,
+			"TAS2781 load block: Other Type = 0x%02x\n",
+			pBlock->mnType);
+		break;
+	}
+
+	for (; chn < chnend; chn++) {
+		if (tas_dev->tasdevice[chn].bLoading == false)
+			continue;
+start:
+		if (pBlock->mbPChkSumPresent) {
+			nResult = tas_dev->write(tas_dev, chn,
+				TASDEVICE_I2CChecksum, 0);
+			if (nResult < 0)
+				goto end;
+		}
+
+		if (pBlock->mbYChkSumPresent)
+			nCRCChkSum = 0;
+
+		nCommand = 0;
+
+		while (nCommand < pBlock->mnCommands) {
+			pData = pBlock->mpData + nCommand * 4;
+
+			nBook = pData[0];
+			nPage = pData[1];
+			nOffset = pData[2];
+			nData = pData[3];
+
+			nCommand++;
+
+			if (nOffset <= 0x7F) {
+				nResult = tas_dev->write(tas_dev, chn,
+					TASDEVICE_REG(nBook, nPage, nOffset),
+					nData);
+				if (nResult < 0)
+					goto end;
+				if (pBlock->mbYChkSumPresent) {
+					nResult = doSingleRegCheckSum(tas_dev,
+						chn, nBook, nPage, nOffset,
+						nData);
+					if (nResult < 0)
+						goto check;
+					nCRCChkSum  += (unsigned char)nResult;
+				}
+			} else if (nOffset == 0x81) {
+				nSleep = (nBook << 8) + nPage;
+				msleep(nSleep);
+			} else if (nOffset == 0x85) {
+				pData  += 4;
+				nLength = (nBook << 8) + nPage;
+				nBook = pData[0];
+				nPage = pData[1];
+				nOffset = pData[2];
+				if (nLength > 1) {
+					nResult = tas_dev->bulk_write(tas_dev,
+						chn, TASDEVICE_REG(nBook,
+						nPage, nOffset), pData + 3,
+						nLength);
+					if (nResult < 0)
+						goto end;
+					if (pBlock->mbYChkSumPresent) {
+						nResult = doMultiRegCheckSum(
+							tas_dev, chn, nBook,
+							nPage, nOffset,
+							nLength);
+					if (nResult < 0)
+						goto check;
+					nCRCChkSum  +=
+					(unsigned char)nResult;
+					}
+				} else {
+					nResult = tas_dev->write(tas_dev, chn,
+						TASDEVICE_REG(nBook, nPage,
+						nOffset),
+						pData[3]);
+					if (nResult < 0)
+						goto end;
+					if (pBlock->mbYChkSumPresent) {
+						nResult = doSingleRegCheckSum(
+							tas_dev, chn, nBook,
+							nPage, nOffset,
+							pData[3]);
+					if (nResult < 0)
+						goto check;
+					nCRCChkSum  +=
+					(unsigned char)nResult;
+					}
+				}
+
+				nCommand++;
+				if (nLength >= 2)
+					nCommand  += ((nLength - 2) / 4) + 1;
+			}
+		}
+		if (pBlock->mbPChkSumPresent) {
+			nResult = tas_dev->read(tas_dev, chn,
+				TASDEVICE_I2CChecksum, &nValue);
+			if (nResult < 0) {
+				dev_err(tas_dev->dev, "%s: Channel %d\n",
+					__func__, chn);
+				goto check;
+			}
+			if ((nValue&0xff) != pBlock->mnPChkSum) {
+				dev_err(tas_dev->dev,
+					"Block PChkSum Channel %d Error: FW = 0x%x, Reg = 0x%x\n",
+					chn, pBlock->mnPChkSum, (nValue&0xff));
+				nResult = -EAGAIN;
+				tas_dev->tasdevice[chn].mnErrCode |=
+					ERROR_PRAM_CRCCHK;
+				goto check;
+			}
+			nResult = 0;
+			tas_dev->tasdevice[chn].mnErrCode &=
+				~ERROR_PRAM_CRCCHK;
+			dev_info(tas_dev->dev, "Block[0x%02x] PChkSum match\n",
+				pBlock->mnType);
+		}
+
+		if (pBlock->mbYChkSumPresent) {
+			//TBD, open it when FW ready
+			dev_err(tas_dev->dev, "Block YChkSum: FW = 0x%x, YCRC = 0x%x\n",
+				pBlock->mnYChkSum,
+				nCRCChkSum);
+
+			tas_dev->tasdevice[chn].mnErrCode &=
+				~ERROR_YRAM_CRCCHK;
+			nResult = 0;
+			dev_info(tas_dev->dev,
+				"Block[0x%x] YChkSum match\n", pBlock->mnType);
+		}
+check:
+		if (nResult == -EAGAIN) {
+			nRetry--;
+			if (nRetry > 0)
+				goto start;
+			else {
+				if ((pBlock->mnType == MAIN_ALL_DEVICES)
+					|| (pBlock->mnType == MAIN_DEVICE_A)
+					|| (pBlock->mnType == MAIN_DEVICE_B)
+					|| (pBlock->mnType == MAIN_DEVICE_C)
+					|| (pBlock->mnType == MAIN_DEVICE_D)) {
+					tas_dev->tasdevice[chn].mnCurrentProgram
+					= -1;
+				} else {
+					tas_dev->tasdevice[chn].mnCurrentConfiguration
+					= -1;
+				}
+				nRetry = 6;
+			}
+		}
+	}
+end:
+	if (nResult < 0) {
+		dev_err(tas_dev->dev, "Block (%d) load error\n",
+				pBlock->mnType);
+	}
+	return nResult;
+}
+
+
+static int tasdevice_load_data(struct tasdevice_priv *tas_dev,
+	struct TData *pData)
+{
+	int nResult = 0;
+	unsigned int nBlock = 0;
+	struct TBlock *pBlock = NULL;
+
+	dev_info(tas_dev->dev, "%s: TAS2781 load data: %s, Blocks = %d\n",
+		__func__,
+		pData->mpName, pData->mnBlocks);
+
+	for (nBlock = 0; nBlock < pData->mnBlocks; nBlock++) {
+		pBlock = &(pData->mpBlocks[nBlock]);
+		nResult = tas_dev->tasdevice_load_block(tas_dev, pBlock);
+		if (nResult < 0)
+			break;
+	}
+
+	return nResult;
+}
+
+static int tasdevice_load_calibrated_data(
+	struct tasdevice_priv *tas_dev, struct TData *pData)
+{
+	int nResult = 0;
+	unsigned int nBlock = 0;
+	struct TBlock *pBlock = NULL;
+
+	dev_info(tas_dev->dev, "%s: TAS2781 load data: %s, Blocks = %d\n",
+		__func__,
+		pData->mpName, pData->mnBlocks);
+
+	for (nBlock = 0; nBlock < pData->mnBlocks; nBlock++) {
+		pBlock = &(pData->mpBlocks[nBlock]);
+		nResult = tasdevice_load_block(tas_dev, pBlock);
+		if (nResult < 0)
+			break;
+	}
+
+	return nResult;
+}
+
+int tas2781_load_calibration(void *pContext,
+			char *pFileName, enum channel i)
+{
+	int ret = 0, nSize = 0, offset = 0;
+	loff_t pos = 0;
+	struct file *filp = NULL;
+	struct firmware FW;
+	const struct firmware *fw_entry = NULL;
+	char *data = NULL;
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *)pContext;
+	struct Ttasdevice *pTasdev = &(tas_dev->tasdevice[i]);
+	struct TFirmware *mpCalFirmware = NULL;
+	char pHint[256];
+
+	ret = request_firmware(&fw_entry, pFileName, tas_dev->dev);
+	if (!ret) {
+		if (!fw_entry->size) {
+			dev_err(tas_dev->dev,
+				"%s: file read error: size = %d\n",
+				__func__, fw_entry->size);
+			goto out;
+		}
+		FW.size = fw_entry->size;
+		FW.data = fw_entry->data;
+		dev_info(tas_dev->dev,
+			"%s: file = %s, file size %zd\n",
+			__func__, pFileName, fw_entry->size);
+	} else {
+		dev_info(tas_dev->dev,
+			"%s: Request firmware failed, try flip_open()\n",
+			__func__);
+
+		scnprintf(pHint, sizeof(pHint), "%s%s\n",
+			TAS2781_CAL_BIN_PATH, pFileName);
+		filp = filp_open(pHint, O_RDONLY, 664);
+		if (!IS_ERR_OR_NULL(filp)) {
+			FW.size = i_size_read(file_inode(filp));
+			dev_info(tas_dev->dev,
+				"%s: file = %s, file size %ld\n",
+				__func__, pHint, (long)FW.size);
+			data = kmalloc(FW.size, GFP_KERNEL);
+			if (data == NULL) {
+				dev_err(tas_dev->dev, "%s: malloc error\n",
+					__func__);
+				goto out;
+			}
+			nSize = (int)kernel_read(filp, data, FW.size, &pos);
+			if (!nSize) {
+				dev_err(tas_dev->dev,
+					"%s: file read error: size = %d\n",
+					__func__, nSize);
+				goto out;
+			}
+			dev_info(tas_dev->dev, "read filed nSize = %d\n",
+				nSize);
+			FW.data = data;
+		} else {
+			dev_err(tas_dev->dev,
+				"%s: cannot open calibration file: %s\n",
+				__func__, pHint);
+			goto out;
+		}
+	}
+
+	mpCalFirmware = pTasdev->mpCalFirmware = kcalloc(1,
+		sizeof(struct TFirmware), GFP_KERNEL);
+	if (pTasdev->mpCalFirmware == NULL) {
+		dev_err(tas_dev->dev, "%s: FW memory failed!\n", __func__);
+		ret = -1;
+		goto out;
+	}
+	mpCalFirmware->dev = tas_dev->dev;
+	offset = fw_parse_header(tas_dev, mpCalFirmware, &FW, offset);
+	if (offset == -1) {
+		dev_err(tas_dev->dev, "%s: fw_parse_header EXIT!\n", __func__);
+		goto out;
+	}
+	offset = fw_parse_variable_header_cal(tas_dev, mpCalFirmware, &FW,
+		offset);
+	if (offset == -1) {
+		dev_err(tas_dev->dev,
+			"%s: fw_parse_variable_header_cal EXIT!\n", __func__);
+		goto out;
+	}
+	offset = fw_parse_program_data(tas_dev, mpCalFirmware, &FW, offset);
+	if (offset == -1) {
+		dev_err(tas_dev->dev, "%s: fw_parse_program_data EXIT!\n",
+			__func__);
+		goto out;
+	}
+	offset = fw_parse_configuration_data(tas_dev, mpCalFirmware, &FW,
+		offset);
+	if (offset == -1) {
+		dev_err(tas_dev->dev,
+			"%s: fw_parse_configuration_data EXIT!\n", __func__);
+		goto out;
+	}
+	offset = fw_parse_calibration_data(tas_dev,
+		mpCalFirmware, &FW, offset);
+	if (offset == -1) {
+		dev_err(tas_dev->dev, "%s: fw_parse_calibration_data EXIT!\n",
+			__func__);
+		goto out;
+	}
+	pTasdev->mbCalibrationLoaded = true;
+out:
+	if (!IS_ERR_OR_NULL(filp)) {
+		filp_close(filp, NULL);
+		kfree(data);
+	}
+	if (fw_entry) {
+		release_firmware(fw_entry);
+		fw_entry = NULL;
+	}
+	return ret;
+}
+
+int tasdevice_dspfw_ready(const void *pVoid, void *pContext)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *) pContext;
+	const struct firmware *pFW = (const struct firmware *)pVoid;
+	struct TFirmware *pFirmware = NULL;
+	struct tasdevice_fw_fixed_hdr *pFw_fixed_hdr = NULL;
+	int offset = 0, ret = 0;
+
+	if (!pFW || !pFW->data) {
+		dev_err(tas_dev->dev, "%s: Failed to read firmware %s\n",
+			__func__, tas_dev->dsp_binaryname);
+		ret = -1;
+		goto out;
+	}
+
+	tas_dev->mpFirmware = kcalloc(1,
+		sizeof(struct TFirmware), GFP_KERNEL);
+	if (tas_dev->mpFirmware == NULL) {
+		ret = -1;
+		goto out;
+	}
+	pFirmware = tas_dev->mpFirmware;
+	pFirmware->dev = tas_dev->dev;
+	offset = fw_parse_header(tas_dev, pFirmware, pFW, offset);
+
+	if (offset == -1)
+		goto out;
+	pFw_fixed_hdr = &(pFirmware->fw_hdr.mnFixedHdr);
+	switch (pFw_fixed_hdr->mnDriverVersion) {
+	case 0x301:
+	case 0x302:
+	case 0x502:
+		tas_dev->fw_parse_variable_header =
+			fw_parse_variable_header_kernel;
+		tas_dev->fw_parse_program_data =
+			fw_parse_program_data_kernel;
+		tas_dev->fw_parse_configuration_data =
+			fw_parse_configuration_data_kernel;
+		tas_dev->tasdevice_load_block =
+			tasdevice_load_block_kernel;
+		pFirmware->bKernelFormat = true;
+		break;
+	case 0x202:
+	case 0x400:
+		tas_dev->fw_parse_variable_header =
+			fw_parse_variable_header_git;
+		tas_dev->fw_parse_program_data =
+			fw_parse_program_data;
+		tas_dev->fw_parse_configuration_data =
+			fw_parse_configuration_data;
+		tas_dev->tasdevice_load_block =
+			tasdevice_load_block;
+		pFirmware->bKernelFormat = false;
+		break;
+	default:
+	if (pFw_fixed_hdr->mnDriverVersion == 0x100) {
+		if (pFw_fixed_hdr->mnPPCVersion >= PPC3_VERSION) {
+			tas_dev->fw_parse_variable_header =
+				fw_parse_variable_header_kernel;
+			tas_dev->fw_parse_program_data =
+				fw_parse_program_data_kernel;
+			tas_dev->fw_parse_configuration_data =
+				fw_parse_configuration_data_kernel;
+			tas_dev->tasdevice_load_block =
+				tasdevice_load_block_kernel;
+			tas_dev->fw_parse_calibration_data = NULL;
+		} else {
+			switch (pFw_fixed_hdr->mnPPCVersion) {
+			case 0x00:
+				tas_dev->fw_parse_variable_header =
+					fw_parse_variable_header_git;
+				tas_dev->fw_parse_program_data =
+					fw_parse_program_data;
+				tas_dev->fw_parse_configuration_data =
+					fw_parse_configuration_data;
+				tas_dev->fw_parse_calibration_data =
+					fw_parse_calibration_data;
+				tas_dev->tasdevice_load_block =
+					tasdevice_load_block;
+				break;
+			default:
+				dev_err(tas_dev->dev,
+					"%s: PPCVersion must be 0x0 or 0x%02x Current:0x%02x\n",
+					__func__, PPC3_VERSION,
+					pFw_fixed_hdr->mnPPCVersion);
+				offset = -1;
+				break;
+			}
+		}
+	} else {
+		dev_err(tas_dev->dev,
+			"%s: DriverVersion must be 0x0, 0x230 or above 0x230:0x%02x\n",
+			__func__,
+			pFw_fixed_hdr->mnDriverVersion);
+		offset = -1;
+	}
+		break;
+	}
+
+	offset = tas_dev->fw_parse_variable_header(tas_dev, pFW, offset);
+	if (offset == -1)
+		goto out;
+
+	offset = tas_dev->fw_parse_program_data(tas_dev, pFirmware, pFW,
+		offset);
+	if (offset < 0) {
+		ret = -1;
+		goto out;
+	}
+	offset = tas_dev->fw_parse_configuration_data(tas_dev,
+		pFirmware, pFW, offset);
+	if (offset < 0)
+		ret = -1;
+
+out:
+	return ret;
+}
+
+void tasdevice_calbin_remove(void *pContext)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *) pContext;
+	struct Ttasdevice *pTasdev = NULL;
+	int i = 0;
+
+	if (tas_dev) {
+		for (i = 0; i < tas_dev->ndev; i++) {
+			pTasdev = &(tas_dev->tasdevice[i]);
+			if (pTasdev->mpCalFirmware) {
+				tas2781_clear_Calfirmware(
+					pTasdev->mpCalFirmware);
+				pTasdev->mpCalFirmware = NULL;
+			}
+		}
+	}
+}
+
+void tasdevice_dsp_remove(void *pContext)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *) pContext;
+	int i = 0;
+
+	if (tas_dev) {
+		if (tas_dev->mpFirmware) {
+			struct TFirmware *pFirmware = tas_dev->mpFirmware;
+
+			if (pFirmware->mpPrograms) {
+				struct TProgram *pProgram;
+
+				for (i = 0; i < pFirmware->mnPrograms; i++) {
+					pProgram = &(pFirmware->mpPrograms[i]);
+					if (pProgram) {
+						struct TData *pImageData =
+							&(pProgram->mData);
+
+					if (pImageData->mpBlocks) {
+						struct TBlock *pBlock;
+						unsigned int nBlock;
+
+					for (nBlock = 0;
+						nBlock <
+						pImageData->mnBlocks;
+						nBlock++) {
+						pBlock =
+							&(pImageData->mpBlocks[nBlock]);
+						kfree(pBlock->mpData);
+					}
+						kfree(pImageData->mpBlocks);
+						}
+						kfree(pProgram->mpDescription);
+					}
+				}
+				kfree(pFirmware->mpPrograms);
+			}
+
+			if (pFirmware->mpConfigurations) {
+				struct TConfiguration *pConfig;
+
+				for (i = 0; i < pFirmware->mnConfigurations;
+					i++) {
+					pConfig =
+					&(pFirmware->mpConfigurations[i]);
+					if (pConfig) {
+						struct TData *pImageData =
+							&(pConfig->mData);
+
+					if (pImageData->mpBlocks) {
+						struct TBlock *pBlock;
+						unsigned int nBlock;
+
+					for (nBlock = 0;
+						nBlock <
+							pImageData->mnBlocks;
+						nBlock++) {
+						pBlock =
+							&(pImageData->mpBlocks[nBlock]);
+						kfree(pBlock->mpData);
+					}
+						kfree(pImageData->mpBlocks);
+					}
+					kfree(pConfig->mpDescription);
+					}
+				}
+				kfree(pFirmware->mpConfigurations);
+			}
+			kfree(pFirmware->fw_hdr.mpDescription);
+			kfree(pFirmware);
+			tas_dev->mpFirmware = NULL;
+		}
+	}
+}
+
+void tasdevice_select_tuningprm_cfg(void *pContext, int prm_no,
+	int cfg_no, int regbin_conf_no)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *) pContext;
+	struct tasdevice_regbin *regbin = &(tas_dev->mtRegbin);
+	struct tasdevice_config_info **cfg_info = regbin->cfg_info;
+	struct TFirmware *pFirmware = tas_dev->mpFirmware;
+	struct TConfiguration *pConfigurations = NULL;
+	struct TProgram *pProgram = NULL;
+	int i = 0;
+	int status = 0;
+
+	if (pFirmware == NULL) {
+		dev_err(tas_dev->dev, "%s: Firmware is NULL\n", __func__);
+		goto out;
+	}
+
+	if (cfg_no >= pFirmware->mnConfigurations) {
+		dev_err(tas_dev->dev,
+			"%s: cfg(%d) is not in range of conf %u\n",
+			__func__, cfg_no, pFirmware->mnConfigurations);
+		goto out;
+	}
+
+	if (prm_no >= pFirmware->mnPrograms || prm_no == 1) {
+		dev_err(tas_dev->dev,
+			"%s: prm(%d) is not in range of Programs %u\n",
+			__func__,  prm_no, pFirmware->mnPrograms);
+		goto out;
+	}
+
+	if (regbin_conf_no > regbin->ncfgs || regbin_conf_no < 0 ||
+		cfg_info == NULL) {
+		dev_err(tas_dev->dev,
+			"conf_no:%d should be in range from 0 to %u\n",
+			regbin_conf_no, regbin->ncfgs-1);
+		goto out;
+	} else {
+		dev_info(tas_dev->dev, "%s: regbin_profile_conf_id = %d\n",
+			__func__, regbin_conf_no);
+	}
+
+	tas_dev->mnCurrentConfiguration = cfg_no;
+	tas_dev->mnCurrentProgram = prm_no;
+
+	pConfigurations = &(pFirmware->mpConfigurations[cfg_no]);
+	for (i = 0; i < tas_dev->ndev; i++) {
+		if (cfg_info[regbin_conf_no]->active_dev & (1 << i)) {
+			if (tas_dev->tasdevice[i].mnCurrentProgram != prm_no) {
+				tas_dev->tasdevice[i].mnCurrentConfiguration
+					= -1;
+				tas_dev->tasdevice[i].bLoading = true;
+				status++;
+			}
+		} else
+			tas_dev->tasdevice[i].bLoading = false;
+		tas_dev->tasdevice[i].bLoaderr = false;
+	}
+
+	if (status) {
+		pProgram = &(pFirmware->mpPrograms[prm_no]);
+		tasdevice_load_data(tas_dev, &(pProgram->mData));
+		for (i = 0; i < tas_dev->ndev; i++) {
+			if (tas_dev->tasdevice[i].bLoaderr == true)
+				continue;
+			else if (tas_dev->tasdevice[i].bLoaderr == false
+				&& tas_dev->tasdevice[i].bLoading == true) {
+				struct TFirmware *pCalFirmware =
+					tas_dev->tasdevice[i].mpCalFirmware;
+
+				if (pCalFirmware) {
+					struct TCalibration *pCalibration =
+						pCalFirmware->mpCalibrations;
+
+					if (pCalibration)
+						tasdevice_load_calibrated_data(
+							tas_dev,
+							&(pCalibration->mData));
+				}
+				tas_dev->tasdevice[i].mnCurrentProgram
+					= prm_no;
+			}
+		}
+	}
+
+	if (tas_dev->mbCalibrationLoaded == false) {
+		for (i = 0; i < tas_dev->ndev; i++)
+			tas_dev->set_calibration(tas_dev, i, 0x100);
+		tas_dev->mbCalibrationLoaded = true;
+		/* No wise to reload calibrationdata everytime,
+		 * this code will work once even if calibrated
+		 * data still failed to be got
+		 */
+	}
+
+	status = 0;
+	for (i = 0; i < tas_dev->ndev; i++) {
+		dev_info(tas_dev->dev, "%s,fun %d,%d,%d\n", __func__,
+			tas_dev->tasdevice[i].mnCurrentConfiguration,
+			cfg_info[regbin_conf_no]->active_dev,
+			tas_dev->tasdevice[i].bLoaderr);
+		if (tas_dev->tasdevice[i].mnCurrentConfiguration != cfg_no
+			&& (cfg_info[regbin_conf_no]->active_dev & (1 << i))
+			&& (tas_dev->tasdevice[i].bLoaderr == false)) {
+			status++;
+			tas_dev->tasdevice[i].bLoading = true;
+		} else
+			tas_dev->tasdevice[i].bLoading = false;
+	}
+
+	if (status) {
+		status = 0;
+		tasdevice_load_data(tas_dev, &(pConfigurations->mData));
+		for (i = 0; i < tas_dev->ndev; i++) {
+			if (tas_dev->tasdevice[i].bLoaderr == true) {
+				status |= 1 << (i + 4);
+				continue;
+			} else if (tas_dev->tasdevice[i].bLoaderr == false
+				&& tas_dev->tasdevice[i].bLoading == true) {
+				tas_dev->tasdevice[i].mnCurrentConfiguration
+					= cfg_no;
+				tas_dev->tasdevice[i].bDSPBypass = false;
+			}
+		}
+	} else {
+		dev_err(tas_dev->dev,
+			"%s: No device is in active in conf %d\n",
+			__func__, regbin_conf_no);
+	}
+
+	status |= cfg_info[regbin_conf_no]->active_dev;
+	dev_info(tas_dev->dev, "%s: DSP mode: load status is %08x\n",
+		__func__, status);
+out:
+	return;
+}
+
+int tas2781_set_calibration(void *pContext, enum channel i,
+	int nCalibration)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *) pContext;
+	int nResult = 0;
+	struct Ttasdevice *pTasdev = &(tas_dev->tasdevice[i]);
+	struct TFirmware *pCalFirmware = pTasdev->mpCalFirmware;
+
+	if ((!tas_dev->mpFirmware->mpPrograms)
+		|| (!tas_dev->mpFirmware->mpConfigurations)) {
+		dev_err(tas_dev->dev, "%s, Firmware not loaded\n\r", __func__);
+		nResult = 0;
+		goto out;
+	}
+
+	if (nCalibration == 0xFF || (nCalibration == 0x100
+		&& pTasdev->mbCalibrationLoaded == false)) {
+		if (pCalFirmware) {
+			pTasdev->mbCalibrationLoaded = false;
+			tas2781_clear_Calfirmware(pCalFirmware);
+			pCalFirmware = NULL;
+		}
+
+		scnprintf(tas_dev->cal_binaryname[i], 64, "%s_cal_0x%02x.bin",
+			tas_dev->dev_name, tas_dev->tasdevice[i].mnDevAddr);
+		nResult = tas2781_load_calibration(tas_dev,
+			tas_dev->cal_binaryname[i], i);
+		if (nResult != 0) {
+			dev_err(tas_dev->dev,
+				"%s: load %s error, no-side effect for playback\n",
+				__func__, tas_dev->cal_binaryname[i]);
+			nResult = 0;
+		}
+	}
+	pTasdev->bLoading = true;
+	pTasdev->bLoaderr = false;
+
+	if (pCalFirmware) {
+		struct TCalibration *pCalibration =
+			pCalFirmware->mpCalibrations;
+
+		if (pCalibration)
+			tasdevice_load_calibrated_data(tas_dev,
+				&(pCalibration->mData));
+	} else
+		dev_err(tas_dev->dev,
+			"%s: No calibrated data for device %d\n", __func__, i);
+
+out:
+	return nResult;
+}
diff --git a/sound/soc/codecs/tas2781-dsp.h b/sound/soc/codecs/tas2781-dsp.h
new file mode 100644
index 000000000..f7b689ad6
--- /dev/null
+++ b/sound/soc/codecs/tas2781-dsp.h
@@ -0,0 +1,213 @@
+/* SPDX-License-Identifier: GPL-2.0
+ *
+ * ALSA SoC Texas Instruments TAS2781 Audio Smart Amplifier
+ *
+ * Copyright (C) 2022 - 2023 Texas Instruments Incorporated
+ * https://www.ti.com
+ *
+ * The TAS2781 driver implements a flexible and configurable register setting
+ * for one, two, even multiple TAS2781 chips. All the register setting are in
+ * a bin file. Almost no specific register setting can be found in the code.
+ *
+ * Author: Shenghao Ding <shenghao-ding@...com>
+ *         Kevin Lu <kevin-lu@...com>
+ */
+
+#ifndef __TASDEVICE_DSP_H__
+#define __TASDEVICE_DSP_H__
+
+#define MAIN_ALL_DEVICES			(0x0d)
+#define MAIN_DEVICE_A				(0x01)
+#define MAIN_DEVICE_B				(0x08)
+#define MAIN_DEVICE_C				(0x10)
+#define MAIN_DEVICE_D				(0x14)
+#define COEFF_DEVICE_A				(0x03)
+#define COEFF_DEVICE_B				(0x0a)
+#define COEFF_DEVICE_C				(0x11)
+#define COEFF_DEVICE_D				(0x15)
+#define PRE_DEVICE_A				(0x04)
+#define PRE_DEVICE_B				(0x0b)
+#define PRE_DEVICE_C				(0x12)
+#define PRE_DEVICE_D				(0x16)
+
+#define PPC3_VERSION 0x4100
+#define REGBIN_CONFIGID_BYPASS_ALL	(0)
+#define TASDEVICE_DEVICE_SUM  (8)
+#define TASDEVICE_CONFIG_SUM  (64)
+
+enum channel {
+	TopLeftChn = 0x00,
+	TopRightChn = 0x01,
+	BottomLeftChn = 0x02,
+	BottomRightChn = 0x03,
+	MaxChn,
+};
+
+enum tasdevice_dsp_dev_idx {
+	TASDEVICE_DSP_TAS_2555 = 0,
+	TASDEVICE_DSP_TAS_2555_STEREO,
+	TASDEVICE_DSP_TAS_2557_MONO,
+	TASDEVICE_DSP_TAS_2557_DUAL_MONO,
+	TASDEVICE_DSP_TAS_2559,
+	TASDEVICE_DSP_TAS_2563,
+	TASDEVICE_DSP_TAS_2563_DUAL_MONO = 7,
+	TASDEVICE_DSP_TAS_2563_QUAD,
+	TASDEVICE_DSP_TAS_2563_21,
+	TASDEVICE_DSP_TAS_2781,
+	TASDEVICE_DSP_TAS_2781_DUAL_MONO,
+	TASDEVICE_DSP_TAS_2781_21,
+	TASDEVICE_DSP_TAS_2781_QUAD,
+	TASDEVICE_DSP_TAS_MAX_DEVICE
+};
+
+struct tasdevice_fw_fixed_hdr {
+	unsigned int mnMagicNumber;
+	unsigned int mnFWSize;
+	unsigned int mnChecksum;
+	unsigned int mnPPCVersion;
+	unsigned int mnFWVersion;
+	unsigned int mnDriverVersion;
+	unsigned int mnTimeStamp;
+	char mpDDCName[64];
+};
+
+struct tasdevice_dspfw_hdr {
+	struct tasdevice_fw_fixed_hdr mnFixedHdr;
+	unsigned int mnBinFileDocVer;
+	char *mpDescription;
+	unsigned short mnDeviceFamily;
+	unsigned short mnDevice;
+	unsigned char ndev;
+};
+
+struct TBlock {
+	unsigned int mnType;
+	unsigned char mbPChkSumPresent;
+	unsigned char mnPChkSum;
+	unsigned char mbYChkSumPresent;
+	unsigned char mnYChkSum;
+	unsigned int mnCommands;
+	unsigned int blk_size;
+	unsigned int nSublocks;
+	unsigned char *mpData;
+};
+
+struct TData {
+	char mpName[64];
+	char *mpDescription;
+	unsigned int mnBlocks;
+	struct TBlock *mpBlocks;
+};
+struct TProgram {
+	unsigned int prog_size;
+	char mpName[64];
+	char *mpDescription;
+	unsigned char mnAppMode;
+	unsigned char mnPDMI2SMode;
+	unsigned char mnISnsPD;
+	unsigned char mnVSnsPD;
+	unsigned char mnPowerLDG;
+	struct TData mData;
+};
+
+struct TConfiguration {
+	unsigned int cfg_size;
+	char mpName[64];
+	char *mpDescription;
+	unsigned char mnDevice_orientation;
+	unsigned char mnDevices;
+	unsigned int mProgram;
+	unsigned int mnSamplingRate;
+	unsigned short mnPLLSrc;
+	unsigned int mnPLLSrcRate;
+	unsigned int mnFsRate;
+	struct TData mData;
+};
+
+struct TCalibration {
+	char mpName[64];
+	char *mpDescription;
+	unsigned int mnProgram;
+	unsigned int mnConfiguration;
+	struct TData mData;
+};
+
+struct TFirmware {
+	struct tasdevice_dspfw_hdr fw_hdr;
+	unsigned int prog_start_offset;
+	unsigned short mnPrograms;
+	struct TProgram *mpPrograms;
+	unsigned int cfg_start_offset;
+	unsigned short mnConfigurations;
+	struct TConfiguration *mpConfigurations;
+	unsigned short mnCalibrations;
+	struct TCalibration *mpCalibrations;
+	bool bKernelFormat;
+	struct device *dev;
+};
+
+enum tasdevice_dsp_fw_state {
+	TASDEVICE_DSP_FW_NONE = 0,
+	TASDEVICE_DSP_FW_PENDING,
+	TASDEVICE_DSP_FW_FAIL,
+	TASDEVICE_DSP_FW_ALL_OK,
+};
+
+enum tasdevice_bin_blk_type {
+	TASDEVICE_BIN_BLK_COEFF = 1,
+	TASDEVICE_BIN_BLK_POST_POWER_UP,
+	TASDEVICE_BIN_BLK_PRE_SHUTDOWN,
+	TASDEVICE_BIN_BLK_PRE_POWER_UP,
+	TASDEVICE_BIN_BLK_POST_SHUTDOWN
+};
+
+struct tasdevice_regbin_hdr {
+	unsigned int img_sz;
+	unsigned int checksum;
+	unsigned int binary_version_num;
+	unsigned int drv_fw_version;
+	unsigned int timestamp;
+	unsigned char plat_type;
+	unsigned char dev_family;
+	unsigned char reserve;
+	unsigned char ndev;
+	unsigned char devs[TASDEVICE_DEVICE_SUM];
+	unsigned int nconfig;
+	unsigned int config_size[TASDEVICE_CONFIG_SUM];
+};
+
+struct tasdevice_block_data {
+	unsigned char dev_idx;
+	unsigned char block_type;
+	unsigned short yram_checksum;
+	unsigned int block_size;
+	unsigned int nSublocks;
+	unsigned char *regdata;
+};
+
+struct tasdevice_config_info {
+	char mpName[64];
+	unsigned int nblocks;
+	unsigned int real_nblocks;
+	unsigned char active_dev;
+	struct tasdevice_block_data **blk_data;
+};
+
+struct tasdevice_regbin {
+	struct tasdevice_regbin_hdr fw_hdr;
+	int ncfgs;
+	struct tasdevice_config_info **cfg_info;
+	int profile_cfg_id;
+};
+
+int tasdevice_dspfw_ready(const void *pVoid, void *pContext);
+void tasdevice_dsp_remove(void *pContext);
+void tasdevice_calbin_remove(void *pContext);
+int tas2781_load_calibration(void *tas_dev, char *pFileName,
+	enum channel i);
+int tas2781_set_calibration(void *pContext, enum channel i,
+	int nCalibration);
+void tasdevice_select_tuningprm_cfg(void *pContext, int prm,
+	int cfg_no, int regbin_conf_no);
+
+#endif
diff --git a/sound/soc/codecs/tas2781-i2c.c b/sound/soc/codecs/tas2781-i2c.c
new file mode 100644
index 000000000..3ee254777
--- /dev/null
+++ b/sound/soc/codecs/tas2781-i2c.c
@@ -0,0 +1,2143 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * ALSA SoC Texas Instruments TAS2781 Audio Smart Amplifier
+ *
+ * Copyright (C) 2022 - 2023 Texas Instruments Incorporated
+ * https://www.ti.com
+ *
+ * The TAS2781 driver implements a flexible and configurable register setting
+ * for one, two, even multiple TAS2781 chips. All the register setting are in
+ * a bin file. Almost no specific register setting can be found in the code.
+ *
+ * Author: Shenghao Ding <shenghao-ding@...com>
+ *         Kevin Lu <kevin-lu@...com>
+ */
+
+#include <linux/module.h>
+#include <linux/regmap.h>
+#include <linux/init.h>
+#include <linux/pm.h>
+#include <linux/i2c.h>
+#include <linux/of.h>
+#include <linux/slab.h>
+#include <linux/firmware.h>
+#include <linux/of_gpio.h>
+#include <linux/interrupt.h>
+#include <sound/soc.h>
+#include <sound/pcm_params.h>
+#include <sound/tlv.h>
+
+#include "tas2781.h"
+
+/* max. length of a alsa mixer control name */
+#define MAX_CONTROL_NAME		(48)
+#define TASDEVICE_CLK_DIR_IN		(0)
+#define TASDEVICE_CLK_DIR_OUT		(1)
+
+#define TASDEVICE_IRQ_DET_TIMEOUT		(30000)
+#define TASDEVICE_IRQ_DET_CNT_LIMIT		(500)
+#define TAS2781_REG_INT_LTCH0	TASDEVICE_REG(0X0, 0x0, 0x49)
+#define TAS2781_REG_INT_LTCH1	TASDEVICE_REG(0X0, 0x0, 0x4A)
+#define TAS2781_REG_INT_LTCH1_0	TASDEVICE_REG(0X0, 0x0, 0x4B)
+#define TAS2781_REG_INT_LTCH2	TASDEVICE_REG(0X0, 0x0, 0x4F)
+#define TAS2781_REG_INT_LTCH3	TASDEVICE_REG(0X0, 0x0, 0x50)
+#define TAS2781_REG_INT_LTCH4	TASDEVICE_REG(0X0, 0x0, 0x51)
+
+const char *blocktype[5] = {
+	"COEFF",
+	"POST_POWER_UP",
+	"PRE_SHUTDOWN",
+	"PRE_POWER_UP",
+	"POST_SHUTDOWN"
+};
+
+static const char * const dts_dev_addr_tag[] = {
+	"ti,topleft-channel",
+	"ti,topright-channel",
+	"ti,bottomleft-channel",
+	"ti,bottomright-channel"
+};
+
+static const char *dts_rst_tag = "ti,%s-gpio%d";
+
+static const char *dts_glb_addr_tag = "ti,global-address";
+
+static const struct i2c_device_id tasdevice_id[] = {
+	{ "audev", GENERAL_AUDEV },
+	{ "tas2781", TAS2781	 },
+	{}
+};
+
+MODULE_DEVICE_TABLE(i2c, tasdevice_id);
+
+static bool tasdevice_volatile(struct device *dev, unsigned int reg)
+{
+	return true;
+}
+
+static bool tasdevice_writeable(struct device *dev, unsigned int reg)
+{
+	return true;
+}
+
+static const struct regmap_config tasdevice_regmap = {
+	.reg_bits = 8,
+	.val_bits = 8,
+	.writeable_reg = tasdevice_writeable,
+	.volatile_reg = tasdevice_volatile,
+	.cache_type = REGCACHE_FLAT,
+	.max_register = 1 * 128,
+};
+
+static const struct of_device_id tasdevice_of_match[] = {
+	{ .compatible = "ti,audev" },
+	{ .compatible = "ti,tas2781" },
+	{},
+};
+
+MODULE_DEVICE_TABLE(of, tasdevice_of_match);
+
+int tasdevice_process_block(void *pContext,
+	unsigned char *data, unsigned char dev_idx, int sublocksize)
+{
+	struct tasdevice_priv *tas_dev =
+		(struct tasdevice_priv *)pContext;
+	unsigned char subblk_typ = data[1];
+	int subblk_offset = 2;
+	int chn = 0, chnend = 0;
+	int rc = 0;
+	int blktyp = dev_idx & 0xC0, idx = dev_idx & 0x3F;
+	bool bError = false;
+
+	if (idx) {
+		chn = idx-1;
+		chnend = idx;
+	} else {
+		if (tas_dev->glb_addr.mnDevAddr) {
+			chn = tas_dev->ndev;
+			chnend = tas_dev->ndev + 1;
+		} else {
+			chn = 0;
+			chnend = tas_dev->ndev;
+		}
+	}
+
+	for (; chn < chnend; chn++) {
+		if (tas_dev->glb_addr.mnDevAddr == 0 &&
+			tas_dev->tasdevice[chn].bLoading == false)
+			continue;
+
+		bError = false;
+		subblk_offset = 2;
+		switch (subblk_typ) {
+		case TASDEVICE_CMD_SING_W: {
+			int i = 0;
+			unsigned short len = SMS_HTONS(data[2], data[3]);
+
+			subblk_offset  += 2;
+			if (subblk_offset + 4 * len > sublocksize) {
+				dev_err(tas_dev->dev,
+					"process_block: Out of memory\n");
+				bError = true;
+				break;
+			}
+
+			for (i = 0; i < len; i++) {
+				rc = tasdevice_dev_write(tas_dev, chn,
+					TASDEVICE_REG(data[subblk_offset],
+						data[subblk_offset + 1],
+						data[subblk_offset + 2]),
+					data[subblk_offset + 3]);
+				if (rc < 0) {
+					bError = true;
+					dev_err(tas_dev->dev,
+						"process_block: single write error\n");
+				}
+				subblk_offset  += 4;
+			}
+		}
+			break;
+		case TASDEVICE_CMD_BURST: {
+			unsigned short len = SMS_HTONS(data[2], data[3]);
+
+			subblk_offset  += 2;
+			if (subblk_offset + 4 + len > sublocksize) {
+				dev_err(tas_dev->dev,
+					"process_block: BURST Out of memory\n");
+				bError = true;
+				break;
+			}
+			if (len % 4) {
+				dev_err(tas_dev->dev,
+				"process_block: Burst len(%u) can be divided by 4\n",
+				len);
+				break;
+			}
+
+			rc = tasdevice_dev_bulk_write(tas_dev, chn,
+				TASDEVICE_REG(data[subblk_offset],
+					data[subblk_offset + 1],
+					data[subblk_offset + 2]),
+					&(data[subblk_offset + 4]), len);
+			if (rc < 0) {
+				bError = true;
+				dev_err(tas_dev->dev,
+					"process_block: bulk_write error = %d\n",
+					rc);
+			}
+			subblk_offset  += (len + 4);
+		}
+			break;
+		case TASDEVICE_CMD_DELAY: {
+			unsigned short delay_time = 0;
+
+			if (subblk_offset + 2 > sublocksize) {
+				dev_err(tas_dev->dev,
+					"process_block: deley Out of memory\n");
+				bError = true;
+				break;
+			}
+			delay_time = SMS_HTONS(data[2], data[3]);
+			usleep_range(delay_time*1000, delay_time*1000);
+			subblk_offset  += 2;
+		}
+			break;
+		case TASDEVICE_CMD_FIELD_W:
+		if (subblk_offset + 6 > sublocksize) {
+			dev_err(tas_dev->dev,
+				"process_block: bit write Out of memory\n");
+			bError = true;
+			break;
+		}
+		rc = tasdevice_dev_update_bits(tas_dev, chn,
+			TASDEVICE_REG(data[subblk_offset + 2],
+				data[subblk_offset + 3],
+				data[subblk_offset + 4]),
+				data[subblk_offset + 1],
+				data[subblk_offset + 5]);
+		if (rc < 0) {
+			bError = true;
+			dev_err(tas_dev->dev,
+				"process_block: update_bits error = %d\n", rc);
+		}
+		subblk_offset  += 6;
+			break;
+		default:
+			break;
+		};
+		if (bError == true && blktyp != 0) {
+			if (blktyp == 0x80) {
+				tas_dev->tasdevice[chn].mnCurrentProgram = -1;
+				tas_dev->tasdevice[chn].mnCurrentConfiguration =
+				-1;
+			} else
+				tas_dev->tasdevice[chn].mnCurrentConfiguration =
+				-1;
+		}
+	}
+	return subblk_offset;
+}
+
+void tasdevice_select_cfg_blk(void *pContext, int conf_no,
+	unsigned char block_type)
+{
+	struct tasdevice_priv *tas_dev =
+		(struct tasdevice_priv *) pContext;
+	struct tasdevice_regbin *regbin = &(tas_dev->mtRegbin);
+	struct tasdevice_config_info **cfg_info = regbin->cfg_info;
+	int j = 0, k = 0, chn = 0, chnend = 0;
+
+	if (conf_no >= regbin->ncfgs || conf_no < 0 || NULL == cfg_info) {
+		dev_err(tas_dev->dev,
+			"conf_no should be not more than %u\n",
+			regbin->ncfgs);
+		goto out;
+	} else
+		dev_info(tas_dev->dev,
+			"select_cfg_blk: profile_conf_id = %d\n",
+			conf_no);
+
+	for (j = 0; j < (int)cfg_info[conf_no]->real_nblocks; j++) {
+		unsigned int length = 0, rc = 0;
+
+		if (block_type > 5 || block_type < 2) {
+			dev_err(tas_dev->dev,
+			"ERROR!!!block_type should be in range from 2 to 5\n");
+			goto out;
+		}
+		if (block_type != cfg_info[conf_no]->blk_data[j]->block_type)
+			continue;
+		dev_info(tas_dev->dev,
+		"select_cfg_blk: conf %d, block type:%s\t device idx = 0x%02x\n",
+		conf_no, blocktype[cfg_info[conf_no]->blk_data[j]
+		->block_type-1], cfg_info[conf_no]->blk_data[j]
+		->dev_idx);
+
+		for (k = 0; k < (int)cfg_info[conf_no]->blk_data[j]
+			->nSublocks; k++) {
+			if (cfg_info[conf_no]->blk_data[j]->dev_idx) {
+				chn =
+				cfg_info[conf_no]->blk_data[j]->dev_idx
+				- 1;
+				chnend =
+				cfg_info[conf_no]->blk_data[j]->dev_idx;
+			} else {
+				chn = 0;
+				chnend = tas_dev->ndev;
+			}
+			for (; chn < chnend; chn++)
+				tas_dev->tasdevice[chn].bLoading = true;
+
+			rc = tasdevice_process_block(tas_dev,
+				cfg_info[conf_no]->blk_data[j]->regdata +
+					length,
+				cfg_info[conf_no]->blk_data[j]->dev_idx,
+				cfg_info[conf_no]->blk_data[j]->block_size -
+					length);
+			length  += rc;
+			if (cfg_info[conf_no]->blk_data[j]->block_size <
+				length) {
+				dev_err(tas_dev->dev,
+					"select_cfg_blk: ERROR:%u %u out of memory\n",
+					length,
+					cfg_info[conf_no]->blk_data[j]->block_size);
+				break;
+			}
+		}
+		if (length != cfg_info[conf_no]->blk_data[j]->block_size)
+			dev_err(tas_dev->dev,
+				"select_cfg_blk: ERROR: %u %u size is not same\n",
+				length,
+				cfg_info[conf_no]->blk_data[j]->block_size);
+
+	}
+
+out:
+	return;
+}
+
+static struct tasdevice_config_info *tasdevice_add_config(
+	void *pContext, unsigned char *config_data,
+	unsigned int config_size)
+{
+	struct tasdevice_priv *tas_dev =
+		(struct tasdevice_priv *)pContext;
+	struct tasdevice_config_info *cfg_info = NULL;
+	int config_offset = 0, i = 0;
+
+	cfg_info = kzalloc(
+			sizeof(struct tasdevice_config_info), GFP_KERNEL);
+	if (!cfg_info)
+		goto out;
+
+	if (tas_dev->mtRegbin.fw_hdr.binary_version_num >= 0x105) {
+		if (config_offset + 64 > (int)config_size) {
+			dev_err(tas_dev->dev,
+				"add config: Out of memory\n");
+			goto out;
+		}
+		memcpy(cfg_info->mpName, &config_data[config_offset], 64);
+		config_offset  += 64;
+	}
+
+	if (config_offset + 4 > (int)config_size) {
+		dev_err(tas_dev->dev,
+			"add config: Out of memory\n");
+		goto out;
+	}
+	cfg_info->nblocks =
+		SMS_HTONL(config_data[config_offset],
+		config_data[config_offset + 1],
+	config_data[config_offset + 2], config_data[config_offset + 3]);
+	config_offset  +=  4;
+
+	cfg_info->blk_data = kcalloc(
+		cfg_info->nblocks, sizeof(struct tasdevice_block_data *),
+		GFP_KERNEL);
+	if (!cfg_info->blk_data) {
+		dev_err(tas_dev->dev,
+			"add config: blk_data alloc failed!\n");
+		goto out;
+	}
+	cfg_info->real_nblocks = 0;
+	for (i = 0; i < (int)cfg_info->nblocks; i++) {
+		if (config_offset + 12 > config_size) {
+			dev_err(tas_dev->dev,
+			"add config: Out of memory: i = %d nblocks = %u!\n",
+			i, cfg_info->nblocks);
+			break;
+		}
+		cfg_info->blk_data[i] = kzalloc(
+			sizeof(struct tasdevice_block_data), GFP_KERNEL);
+		if (!cfg_info->blk_data[i])
+			break;
+
+		cfg_info->blk_data[i]->dev_idx = config_data[config_offset];
+		config_offset++;
+
+		cfg_info->blk_data[i]->block_type = config_data[config_offset];
+		config_offset++;
+
+		if (cfg_info->blk_data[i]->block_type  ==
+			TASDEVICE_BIN_BLK_PRE_POWER_UP) {
+			if (cfg_info->blk_data[i]->dev_idx == 0) {
+				cfg_info->active_dev = 1;
+			} else {
+				cfg_info->active_dev =
+					1 <<
+					(cfg_info->blk_data[i]->dev_idx - 1);
+			}
+		}
+		cfg_info->blk_data[i]->yram_checksum =
+			SMS_HTONS(config_data[config_offset],
+			config_data[config_offset + 1]);
+		config_offset  += 2;
+		cfg_info->blk_data[i]->block_size =
+			SMS_HTONL(config_data[config_offset],
+			config_data[config_offset + 1],
+			config_data[config_offset + 2],
+		config_data[config_offset + 3]);
+		config_offset  += 4;
+
+		cfg_info->blk_data[i]->nSublocks =
+			SMS_HTONL(config_data[config_offset],
+			config_data[config_offset + 1],
+			config_data[config_offset + 2],
+		config_data[config_offset + 3]);
+
+		config_offset  += 4;
+		cfg_info->blk_data[i]->regdata = kzalloc(
+			cfg_info->blk_data[i]->block_size, GFP_KERNEL);
+		if (cfg_info->blk_data[i]->regdata == 0) {
+			dev_err(tas_dev->dev,
+				"add config: regdata alloc failed!\n");
+			goto out;
+		}
+		if (config_offset + cfg_info->blk_data[i]->block_size
+			> config_size) {
+			dev_err(tas_dev->dev,
+			"add config: block_size Out of memory: i = %d nblocks = %u!\n",
+			i,
+			cfg_info->nblocks);
+			break;
+		}
+		memcpy(cfg_info->blk_data[i]->regdata,
+			&config_data[config_offset],
+		cfg_info->blk_data[i]->block_size);
+		config_offset  += cfg_info->blk_data[i]->block_size;
+		cfg_info->real_nblocks  += 1;
+	}
+out:
+	return cfg_info;
+}
+static int tas2781_digital_getvol(struct snd_kcontrol *kcontrol,
+	struct snd_ctl_elem_value *ucontrol)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *tas_dev =
+		snd_soc_component_get_drvdata(codec);
+	struct soc_mixer_control *mc =
+		(struct soc_mixer_control *)kcontrol->private_value;
+	unsigned int val;
+	int ret = 0;
+
+	/* Read the primary device as the whole */
+	ret = tasdevice_dev_read(tas_dev, 0, mc->reg, &val);
+	if (ret) {
+		dev_err(tas_dev->dev,
+		"%s, get digital vol error\n",
+		__func__);
+		goto out;
+	}
+	val = (val > mc->max) ? mc->max : val;
+	val = mc->invert ? mc->max - val : val;
+	ucontrol->value.integer.value[0] = val;
+
+out:
+	return ret;
+}
+
+static int tas2781_digital_putvol(struct snd_kcontrol *kcontrol,
+	struct snd_ctl_elem_value *ucontrol)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *tas_dev =
+		snd_soc_component_get_drvdata(codec);
+	struct soc_mixer_control *mc =
+		(struct soc_mixer_control *)kcontrol->private_value;
+	unsigned int val;
+	int i, ret = 0;
+
+	val = ucontrol->value.integer.value[0];
+	val = (val > mc->max) ? mc->max : val;
+	val = mc->invert ? mc->max - val : val;
+	val = (val < 0) ? 0 : val;
+	if (tas_dev->tas2781_set_global != NULL) {
+		ret = tasdevice_dev_write(tas_dev, tas_dev->ndev,
+			mc->reg, val);
+		if (ret)
+			dev_err(tas_dev->dev,
+			"%s, set digital vol error in global mode\n",
+			__func__);
+	} else {
+		for (i = 0; i < tas_dev->ndev; i++) {
+			ret = tasdevice_dev_write(tas_dev, i,
+				mc->reg, val);
+			if (ret)
+				dev_err(tas_dev->dev,
+				"%s, set digital vol error in device %d\n",
+				__func__, i);
+		}
+	}
+
+	return ret;
+}
+
+static int tas2781_amp_getvol(struct snd_kcontrol *kcontrol,
+	struct snd_ctl_elem_value *ucontrol)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *tas_dev =
+		snd_soc_component_get_drvdata(codec);
+	struct soc_mixer_control *mc =
+		(struct soc_mixer_control *)kcontrol->private_value;
+	unsigned int val;
+	unsigned char mask = 0;
+	int ret = 0;
+
+	/* Read the primary device */
+	ret = tasdevice_dev_read(tas_dev, 0, mc->reg, &val);
+	if (ret) {
+		dev_err(tas_dev->dev,
+		"%s, get AMP vol error\n",
+		__func__);
+		goto out;
+	}
+
+	mask = (1 << fls(mc->max)) - 1;
+	mask <<= mc->shift;
+	val = (val & mask) >> mc->shift;
+	val = (val > mc->max) ? mc->max : val;
+	val = mc->invert ? mc->max - val : val;
+	ucontrol->value.integer.value[0] = val;
+
+out:
+	return ret;
+}
+
+static int tas2781_amp_putvol(struct snd_kcontrol *kcontrol,
+	struct snd_ctl_elem_value *ucontrol)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *tas_dev =
+		snd_soc_component_get_drvdata(codec);
+	struct soc_mixer_control *mc =
+		(struct soc_mixer_control *)kcontrol->private_value;
+	unsigned int val;
+	int i, ret = 0;
+	unsigned char mask = 0;
+
+	mask = (1 << fls(mc->max)) - 1;
+	mask <<= mc->shift;
+	val = ucontrol->value.integer.value[0];
+	val = (val > mc->max) ? mc->max : val;
+	val = mc->invert ? mc->max - val : val;
+	val = (val < 0) ? 0 : val;
+	for (i = 0; i < tas_dev->ndev; i++) {
+		ret = tasdevice_dev_update_bits(tas_dev, i,
+			mc->reg,
+			mask,
+			val << mc->shift);
+		if (ret)
+			dev_err(tas_dev->dev,
+			"%s, set AMP vol error in device %d\n",
+			__func__, i);
+	}
+
+	return ret;
+}
+
+static const DECLARE_TLV_DB_SCALE(dvc_tlv, -10000, 100, 0);
+static const DECLARE_TLV_DB_SCALE(amp_vol_tlv, 1100, 50, 0);
+
+static const struct snd_kcontrol_new tas2781_snd_controls[] = {
+	SOC_SINGLE_RANGE_EXT_TLV("Amp Gain Volume", TAS2781_AMP_LEVEL,
+		1, 0, 20, 0, tas2781_amp_getvol,
+		tas2781_amp_putvol, amp_vol_tlv),
+	SOC_SINGLE_RANGE_EXT_TLV("Digital Volume Control", TAS2781_DVC_LVL,
+		0, 0, 200, 1, tas2781_digital_getvol,
+		tas2781_digital_putvol, dvc_tlv),
+};
+
+static int tasdevice_set_profile_id(struct snd_kcontrol *kcontrol,
+		struct snd_ctl_elem_value *ucontrol)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *p_tasdevice =
+		snd_soc_component_get_drvdata(codec);
+
+	/* Codec Lock Hold*/
+	mutex_lock(&p_tasdevice->codec_lock);
+	p_tasdevice->mtRegbin.profile_cfg_id =
+		ucontrol->value.integer.value[0];
+	/* Codec Lock Release*/
+	mutex_unlock(&p_tasdevice->codec_lock);
+
+	return 0;
+}
+
+static int tasdevice_info_programs(struct snd_kcontrol *kcontrol,
+			struct snd_ctl_elem_info *uinfo)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *p_tasdevice =
+		snd_soc_component_get_drvdata(codec);
+	struct TFirmware *Tfw = p_tasdevice->mpFirmware;
+
+	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
+	/* Codec Lock Hold*/
+	mutex_lock(&p_tasdevice->codec_lock);
+	uinfo->count = 1;
+	uinfo->value.integer.min = 0;
+	uinfo->value.integer.max = (int)Tfw->mnPrograms;
+	/* Codec Lock Release*/
+	mutex_unlock(&p_tasdevice->codec_lock);
+	return 0;
+}
+
+static int tasdevice_info_configurations(
+	struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *p_tasdevice =
+		snd_soc_component_get_drvdata(codec);
+	struct TFirmware *Tfw = p_tasdevice->mpFirmware;
+
+	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
+	/* Codec Lock Hold*/
+	mutex_lock(&p_tasdevice->codec_lock);
+	uinfo->count = 1;
+	uinfo->value.integer.min = 0;
+	uinfo->value.integer.max = (int)Tfw->mnConfigurations - 1;
+	/* Codec Lock Release*/
+	mutex_unlock(&p_tasdevice->codec_lock);
+
+	return 0;
+}
+
+static int tasdevice_info_profile(struct snd_kcontrol *kcontrol,
+			struct snd_ctl_elem_info *uinfo)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *p_tasdevice =
+		snd_soc_component_get_drvdata(codec);
+
+	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
+	/* Codec Lock Hold*/
+	mutex_lock(&p_tasdevice->codec_lock);
+	uinfo->count = 1;
+	uinfo->value.integer.min = 0;
+	uinfo->value.integer.max = max(0, p_tasdevice->mtRegbin.ncfgs);
+	/* Codec Lock Release*/
+	mutex_unlock(&p_tasdevice->codec_lock);
+	return 0;
+}
+
+static int tasdevice_get_profile_id(struct snd_kcontrol *kcontrol,
+			  struct snd_ctl_elem_value *ucontrol)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(kcontrol);
+	struct tasdevice_priv *p_tasdevice
+		= snd_soc_component_get_drvdata(codec);
+
+	/* Codec Lock Hold*/
+	mutex_lock(&p_tasdevice->codec_lock);
+	ucontrol->value.integer.value[0] =
+		p_tasdevice->mtRegbin.profile_cfg_id;
+	/* Codec Lock Release*/
+	mutex_unlock(&p_tasdevice->codec_lock);
+
+	return 0;
+}
+
+static int tasdevice_create_controls(struct tasdevice_priv *tas_dev)
+{
+	int  nr_controls = 1, ret = 0, mix_index = 0;
+	char *name = NULL;
+	struct snd_kcontrol_new *tasdevice_profile_controls = NULL;
+
+	tasdevice_profile_controls = devm_kzalloc(tas_dev->dev,
+			nr_controls * sizeof(tasdevice_profile_controls[0]),
+			GFP_KERNEL);
+	if (tasdevice_profile_controls == NULL) {
+		ret = -ENOMEM;
+		goto out;
+	}
+
+	/* Create a mixer item for selecting the active profile */
+	name = devm_kzalloc(tas_dev->dev,
+		MAX_CONTROL_NAME, GFP_KERNEL);
+	if (!name) {
+		ret = -ENOMEM;
+		goto out;
+	}
+	scnprintf(name, MAX_CONTROL_NAME, "TASDEVICE Profile id");
+	tasdevice_profile_controls[mix_index].name = name;
+	tasdevice_profile_controls[mix_index].iface =
+		SNDRV_CTL_ELEM_IFACE_MIXER;
+	tasdevice_profile_controls[mix_index].info =
+		tasdevice_info_profile;
+	tasdevice_profile_controls[mix_index].get =
+		tasdevice_get_profile_id;
+	tasdevice_profile_controls[mix_index].put =
+		tasdevice_set_profile_id;
+	mix_index++;
+
+	ret = snd_soc_add_component_controls(tas_dev->codec,
+		tasdevice_profile_controls,
+		nr_controls < mix_index ? nr_controls : mix_index);
+
+	tas_dev->tas_ctrl.nr_controls =
+		nr_controls < mix_index ? nr_controls : mix_index;
+out:
+	return ret;
+}
+
+static int tasdevice_program_get(struct snd_kcontrol *pKcontrol,
+		struct snd_ctl_elem_value *pValue)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(pKcontrol);
+	struct tasdevice_priv *pTAS2781 = snd_soc_component_get_drvdata(codec);
+
+	mutex_lock(&pTAS2781->codec_lock);
+	pValue->value.integer.value[0] = pTAS2781->mnCurrentProgram;
+	mutex_unlock(&pTAS2781->codec_lock);
+	return 0;
+}
+
+static int tasdevice_program_put(struct snd_kcontrol *pKcontrol,
+		struct snd_ctl_elem_value *pValue)
+{
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(pKcontrol);
+	struct tasdevice_priv *pTAS2781 = snd_soc_component_get_drvdata(codec);
+	unsigned int nProgram = pValue->value.integer.value[0];
+
+	mutex_lock(&pTAS2781->codec_lock);
+	pTAS2781->mnCurrentProgram = nProgram;
+	mutex_unlock(&pTAS2781->codec_lock);
+	return 0;
+}
+
+static int tasdevice_configuration_get(
+	struct snd_kcontrol *pKcontrol,
+	struct snd_ctl_elem_value *pValue)
+{
+
+	struct snd_soc_component *codec
+		= snd_soc_kcontrol_component(pKcontrol);
+	struct tasdevice_priv *pTAS2781 = snd_soc_component_get_drvdata(codec);
+
+	mutex_lock(&pTAS2781->codec_lock);
+	pValue->value.integer.value[0] = pTAS2781->mnCurrentConfiguration;
+	mutex_unlock(&pTAS2781->codec_lock);
+	return 0;
+}
+
+static int tasdevice_configuration_put(
+	struct snd_kcontrol *pKcontrol,
+	struct snd_ctl_elem_value *pValue)
+{
+	struct snd_soc_component *codec
+					= snd_soc_kcontrol_component(pKcontrol);
+	struct tasdevice_priv *pTAS2781 = snd_soc_component_get_drvdata(codec);
+	unsigned int nConfiguration = pValue->value.integer.value[0];
+
+	mutex_lock(&pTAS2781->codec_lock);
+	pTAS2781->mnCurrentConfiguration = nConfiguration;
+	mutex_unlock(&pTAS2781->codec_lock);
+	return 0;
+}
+
+static int tasdevice_dsp_create_control(
+	struct tasdevice_priv *tas_dev)
+{
+	int  nr_controls = 2, ret = 0, mix_index = 0;
+	char *program_name = NULL;
+	char *configuration_name = NULL;
+	struct snd_kcontrol_new *tasdevice_dsp_controls = NULL;
+
+	tasdevice_dsp_controls = devm_kzalloc(tas_dev->dev,
+			nr_controls * sizeof(tasdevice_dsp_controls[0]),
+			GFP_KERNEL);
+	if (tasdevice_dsp_controls == NULL) {
+		ret = -ENOMEM;
+		goto out;
+	}
+
+	/* Create a mixer item for selecting the active profile */
+	program_name = devm_kzalloc(tas_dev->dev,
+		MAX_CONTROL_NAME, GFP_KERNEL);
+	configuration_name = devm_kzalloc(tas_dev->dev,
+		MAX_CONTROL_NAME, GFP_KERNEL);
+	if (!program_name || !configuration_name) {
+		ret = -ENOMEM;
+		goto out;
+	}
+
+	scnprintf(program_name, MAX_CONTROL_NAME, "Program");
+	tasdevice_dsp_controls[mix_index].name = program_name;
+	tasdevice_dsp_controls[mix_index].iface =
+		SNDRV_CTL_ELEM_IFACE_MIXER;
+	tasdevice_dsp_controls[mix_index].info =
+		tasdevice_info_programs;
+	tasdevice_dsp_controls[mix_index].get =
+		tasdevice_program_get;
+	tasdevice_dsp_controls[mix_index].put =
+		tasdevice_program_put;
+	mix_index++;
+
+	scnprintf(configuration_name, MAX_CONTROL_NAME, "Configuration");
+	tasdevice_dsp_controls[mix_index].name = configuration_name;
+	tasdevice_dsp_controls[mix_index].iface =
+		SNDRV_CTL_ELEM_IFACE_MIXER;
+	tasdevice_dsp_controls[mix_index].info =
+		tasdevice_info_configurations;
+	tasdevice_dsp_controls[mix_index].get =
+		tasdevice_configuration_get;
+	tasdevice_dsp_controls[mix_index].put =
+		tasdevice_configuration_put;
+	mix_index++;
+
+	ret = snd_soc_add_component_controls(tas_dev->codec,
+		tasdevice_dsp_controls,
+		nr_controls < mix_index ? nr_controls : mix_index);
+
+	tas_dev->tas_ctrl.nr_controls += nr_controls;
+out:
+	return ret;
+}
+
+void tasdevice_regbin_ready(const struct firmware *pFW,
+	void *pContext)
+{
+	struct tasdevice_priv *tas_dev =
+		(struct tasdevice_priv *) pContext;
+	struct tasdevice_regbin *regbin = NULL;
+	struct tasdevice_regbin_hdr *fw_hdr = NULL;
+	struct tasdevice_config_info **cfg_info = NULL;
+	const struct firmware *fw_entry = NULL;
+	unsigned char *buf = NULL;
+	int offset = 0, i = 0;
+	unsigned int total_config_sz = 0;
+	int ret = 0;
+
+	if (tas_dev == NULL) {
+		dev_err(tas_dev->dev,
+			"tasdev: regbin_ready: handle is NULL\n");
+		return;
+	}
+	mutex_lock(&tas_dev->codec_lock);
+	regbin = &(tas_dev->mtRegbin);
+	fw_hdr = &(regbin->fw_hdr);
+	if (unlikely(!pFW) || unlikely(!pFW->data)) {
+		dev_err(tas_dev->dev,
+		"Failed to read %s, no side - effect on driver running\n",
+		tas_dev->regbin_binaryname);
+		ret = -1;
+		goto out;
+	}
+	buf = (unsigned char *)pFW->data;
+
+	dev_info(tas_dev->dev, "tasdev: regbin_ready start\n");
+	fw_hdr->img_sz = SMS_HTONL(buf[offset], buf[offset + 1],
+		buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	if (fw_hdr->img_sz != pFW->size) {
+		dev_err(tas_dev->dev,
+			"File size not match, %d %u", (int)pFW->size,
+			fw_hdr->img_sz);
+		ret = -1;
+		goto out;
+	}
+
+	fw_hdr->checksum = SMS_HTONL(buf[offset], buf[offset + 1],
+					buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	fw_hdr->binary_version_num = SMS_HTONL(buf[offset],
+		buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+	if (fw_hdr->binary_version_num < 0x103) {
+		dev_err(tas_dev->dev,
+			"File version 0x%04x is too low",
+			fw_hdr->binary_version_num);
+		ret = -1;
+		goto out;
+	}
+	offset  += 4;
+	fw_hdr->drv_fw_version = SMS_HTONL(buf[offset], buf[offset + 1],
+					buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	fw_hdr->timestamp = SMS_HTONL(buf[offset], buf[offset + 1],
+					buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	fw_hdr->plat_type = buf[offset];
+	offset  += 1;
+	fw_hdr->dev_family = buf[offset];
+	offset  += 1;
+	fw_hdr->reserve = buf[offset];
+	offset  += 1;
+	fw_hdr->ndev = buf[offset];
+	offset  += 1;
+	if (fw_hdr->ndev != tas_dev->ndev) {
+		dev_err(tas_dev->dev,
+		"ndev(%u) from Regbin and ndev(%u) from DTS does not match\n",
+		fw_hdr->ndev,
+		tas_dev->ndev);
+		ret = -1;
+		goto out;
+	}
+	if (offset + TASDEVICE_DEVICE_SUM > fw_hdr->img_sz) {
+		dev_err(tas_dev->dev,
+		"regbin_ready: Out of Memory!\n");
+		ret = -1;
+		goto out;
+	}
+
+	for (i = 0; i < TASDEVICE_DEVICE_SUM; i++, offset++)
+		fw_hdr->devs[i] = buf[offset];
+
+	fw_hdr->nconfig = SMS_HTONL(buf[offset], buf[offset + 1],
+				buf[offset + 2], buf[offset + 3]);
+	offset  += 4;
+	dev_info(tas_dev->dev, "nconfig = %u\n", fw_hdr->nconfig);
+	for (i = 0; i < TASDEVICE_CONFIG_SUM; i++) {
+		fw_hdr->config_size[i] = SMS_HTONL(buf[offset],
+			buf[offset + 1], buf[offset + 2], buf[offset + 3]);
+		offset  += 4;
+		total_config_sz  += fw_hdr->config_size[i];
+	}
+	dev_info(tas_dev->dev,
+		"img_sz = %u total_config_sz = %u offset = %d\n",
+		fw_hdr->img_sz, total_config_sz, offset);
+	if (fw_hdr->img_sz - total_config_sz != (unsigned int)offset) {
+		dev_err(tas_dev->dev, "Bin file error!\n");
+		ret = -1;
+		goto out;
+	}
+	cfg_info = kcalloc(fw_hdr->nconfig,
+		sizeof(struct tasdevice_config_info *),
+		GFP_KERNEL);
+
+	if (!cfg_info) {
+		ret = -1;
+		dev_err(tas_dev->dev, "nconfig Memory alloc failed!\n");
+		goto out;
+	}
+	regbin->cfg_info = cfg_info;
+	regbin->ncfgs = 0;
+	for (i = 0; i < (int)fw_hdr->nconfig; i++) {
+		cfg_info[i] = tasdevice_add_config(pContext, &buf[offset],
+				fw_hdr->config_size[i]);
+		if (!cfg_info[i]) {
+			ret = -1;
+			dev_err(tas_dev->dev,
+				"add_config Memory alloc failed!\n");
+			break;
+		}
+		offset  += (int)fw_hdr->config_size[i];
+		regbin->ncfgs  += 1;
+	}
+	tasdevice_create_controls(tas_dev);
+
+	tasdevice_dsp_remove(tas_dev);
+	tasdevice_calbin_remove(tas_dev);
+	tas_dev->fw_state = TASDEVICE_DSP_FW_PENDING;
+	scnprintf(tas_dev->dsp_binaryname, 64, "%s_dsp.bin",
+		tas_dev->dev_name);
+	ret = request_firmware(&fw_entry, tas_dev->dsp_binaryname,
+		tas_dev->dev);
+	if (!ret) {
+		ret = tasdevice_dspfw_ready(fw_entry, tas_dev);
+		release_firmware(fw_entry);
+		fw_entry = NULL;
+	} else {
+		tas_dev->fw_state = TASDEVICE_DSP_FW_FAIL;
+		dev_err(tas_dev->dev, "%s: load %s error\n", __func__,
+			tas_dev->dsp_binaryname);
+		goto out;
+	}
+	tasdevice_dsp_create_control(tas_dev);
+
+	tas_dev->fw_state = TASDEVICE_DSP_FW_ALL_OK;
+	tas_dev->mbCalibrationLoaded = true;
+	for (i = 0; i < tas_dev->ndev; i++) {
+		scnprintf(tas_dev->cal_binaryname[i], 64, "%s_cal_0x%02x.bin",
+			tas_dev->dev_name, tas_dev->tasdevice[i].mnDevAddr);
+		ret = tas2781_load_calibration(tas_dev,
+			tas_dev->cal_binaryname[i], i);
+		if (ret != 0) {
+			dev_err(tas_dev->dev,
+				"%s: load %s error, no-side effect for playback\n",
+				__func__,
+				tas_dev->cal_binaryname[i]);
+			ret = 0;
+			tas_dev->mbCalibrationLoaded = false;
+		}
+	}
+
+out:
+	mutex_unlock(&tas_dev->codec_lock);
+	if (pFW)
+		release_firmware(pFW);
+	dev_info(tas_dev->dev, "Firmware init complete\n");
+}
+
+void tasdevice_config_info_remove(void *pContext)
+{
+	struct tasdevice_priv *tas_dev =
+		(struct tasdevice_priv *) pContext;
+	struct tasdevice_regbin *regbin = &(tas_dev->mtRegbin);
+	struct tasdevice_config_info **cfg_info = regbin->cfg_info;
+	int i = 0, j = 0;
+
+	mutex_lock(&tas_dev->dev_lock);
+	if (cfg_info) {
+		for (i = 0; i < regbin->ncfgs; i++) {
+			if (cfg_info[i]) {
+				for (j = 0; j < (int)cfg_info[i]->real_nblocks;
+					j++) {
+					kfree(
+					cfg_info[i]->blk_data[j]->regdata);
+					kfree(cfg_info[i]->blk_data[j]);
+				}
+				kfree(cfg_info[i]->blk_data);
+				kfree(cfg_info[i]);
+			}
+		}
+		kfree(cfg_info);
+	}
+	mutex_unlock(&tas_dev->dev_lock);
+}
+
+static int tasdevice_dac_event(struct snd_soc_dapm_widget *w,
+			struct snd_kcontrol *kcontrol, int event)
+{
+	struct snd_soc_component *codec = snd_soc_dapm_to_component(w->dapm);
+	struct tasdevice_priv *tas_dev = snd_soc_component_get_drvdata(codec);
+
+	switch (event) {
+	case SND_SOC_DAPM_POST_PMU:
+		dev_info(tas_dev->dev, "SND_SOC_DAPM_POST_PMU\n");
+		break;
+	case SND_SOC_DAPM_PRE_PMD:
+		dev_info(tas_dev->dev, "SND_SOC_DAPM_PRE_PMD\n");
+		break;
+	}
+
+	return 0;
+}
+
+static const struct snd_soc_dapm_widget tasdevice_dapm_widgets[] = {
+	SND_SOC_DAPM_AIF_IN("ASI", "ASI Playback", 0, SND_SOC_NOPM, 0, 0),
+	SND_SOC_DAPM_AIF_OUT("ASI OUT", "ASI Capture", 0, SND_SOC_NOPM, 0, 0),
+	SND_SOC_DAPM_DAC_E("DAC", NULL, SND_SOC_NOPM, 0, 0,
+		tasdevice_dac_event,
+		SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
+	SND_SOC_DAPM_OUTPUT("OUT"),
+	SND_SOC_DAPM_INPUT("DMIC"),
+	SND_SOC_DAPM_SIGGEN("VMON"),
+	SND_SOC_DAPM_SIGGEN("IMON")
+};
+
+static const struct snd_soc_dapm_route tasdevice_audio_map[] = {
+	{"DAC", NULL, "ASI"},
+	{"OUT", NULL, "DAC"},
+	{"ASI OUT", NULL, "DMIC"}
+};
+
+static int tasdevice_startup(struct snd_pcm_substream *substream,
+						struct snd_soc_dai *dai)
+{
+	struct snd_soc_component *codec = dai->component;
+	struct tasdevice_priv *tas_dev = snd_soc_component_get_drvdata(codec);
+	int ret = 0;
+
+	if (tas_dev->fw_state != TASDEVICE_DSP_FW_ALL_OK) {
+		dev_err(tas_dev->dev, "DSP bin file not loaded\n");
+		ret = -EINVAL;
+	}
+	return ret;
+}
+
+static int tasdevice_hw_params(struct snd_pcm_substream *substream,
+	struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
+{
+	struct tasdevice_priv *tas_dev = snd_soc_dai_get_drvdata(dai);
+	unsigned int fsrate;
+	unsigned int slot_width;
+	int bclk_rate;
+	int rc = 0;
+
+	dev_info(tas_dev->dev, "%s: %s\n",
+		__func__, substream->stream == SNDRV_PCM_STREAM_PLAYBACK ?
+		"Playback":"Capture");
+
+	fsrate = params_rate(params);
+	switch (fsrate) {
+	case 48000:
+		break;
+	case 44100:
+		break;
+	default:
+		dev_err(tas_dev->dev,
+			"%s: incorrect sample rate = %u\n",
+			__func__, fsrate);
+		rc = -EINVAL;
+		goto out;
+	}
+
+	slot_width = params_width(params);
+	switch (slot_width) {
+	case 16:
+		break;
+	case 20:
+		break;
+	case 24:
+		break;
+	case 32:
+		break;
+	default:
+		dev_err(tas_dev->dev,
+			"%s: incorrect slot width = %u\n",
+			__func__, slot_width);
+		rc = -EINVAL;
+		goto out;
+	}
+
+	bclk_rate = snd_soc_params_to_bclk(params);
+	if (bclk_rate < 0) {
+		dev_err(tas_dev->dev,
+			"%s: incorrect bclk rate = %d\n",
+			__func__, bclk_rate);
+		rc = bclk_rate;
+		goto out;
+	}
+	dev_info(tas_dev->dev,
+	"%s: BCLK rate = %d Channel = %d Sample rate = %u slot width = %u\n",
+	__func__, bclk_rate, params_channels(params),
+	fsrate, slot_width);
+out:
+	return rc;
+}
+
+static void tasdevice_enable_irq(
+	struct tasdevice_priv *tas_dev, bool enable)
+{
+	struct irq_desc *desc = NULL;
+
+	if (enable) {
+		if (tas_dev->mIrqInfo.mb_irq_enable)
+			return;
+		if (gpio_is_valid(tas_dev->mIrqInfo.mn_irq_gpio)) {
+			desc = irq_to_desc(tas_dev->mIrqInfo.mn_irq);
+			if (desc && desc->depth > 0)
+				enable_irq(tas_dev->mIrqInfo.mn_irq);
+			else
+				dev_info(tas_dev->dev,
+					"### irq already enabled\n");
+		}
+		tas_dev->mIrqInfo.mb_irq_enable = true;
+	} else {
+		if (gpio_is_valid(tas_dev->mIrqInfo.mn_irq_gpio))
+			disable_irq_nosync(tas_dev->mIrqInfo.mn_irq);
+		tas_dev->mIrqInfo.mb_irq_enable = false;
+	}
+}
+
+static int tasdevice_set_dai_sysclk(struct snd_soc_dai *codec_dai,
+	int clk_id, unsigned int freq, int dir)
+{
+	struct tasdevice_priv *tas_dev = snd_soc_dai_get_drvdata(codec_dai);
+
+	dev_info(tas_dev->dev,
+		"%s: clk_id = %d, freq = %u, CLK direction %s\n",
+		__func__, clk_id, freq,
+		dir == TASDEVICE_CLK_DIR_OUT ? "OUT":"IN");
+
+	return 0;
+}
+
+static void powercontrol_routine(struct work_struct *work)
+{
+	struct tasdevice_priv *tas_dev =
+		container_of(work, struct tasdevice_priv,
+		powercontrol_work.work);
+	struct TFirmware *pFw = NULL;
+	int profile_cfg_id = 0;
+
+	mutex_lock(&tas_dev->codec_lock);
+	/*mnCurrentProgram != 0 is dsp mode or tuning mode*/
+	if (tas_dev->mnCurrentProgram) {
+		/*bypass all in regbin is profile id 0*/
+		profile_cfg_id = REGBIN_CONFIGID_BYPASS_ALL;
+	} else {
+		profile_cfg_id = tas_dev->mtRegbin.profile_cfg_id;
+		pFw = tas_dev->mpFirmware;
+		dev_info(tas_dev->dev, "%s: %s\n", __func__,
+			pFw->mpConfigurations[tas_dev->mnCurrentConfiguration]
+			.mpName);
+		tasdevice_select_tuningprm_cfg(tas_dev,
+			tas_dev->mnCurrentProgram,
+			tas_dev->mnCurrentConfiguration,
+			profile_cfg_id);
+		if (tas_dev->tas2781_set_global != NULL)
+			tas_dev->tas2781_set_global(tas_dev);
+	}
+	tasdevice_select_cfg_blk(tas_dev, profile_cfg_id,
+		TASDEVICE_BIN_BLK_PRE_POWER_UP);
+
+	if (tas_dev->chip_id != GENERAL_AUDEV)
+		tasdevice_enable_irq(tas_dev, true);
+	mutex_unlock(&tas_dev->codec_lock);
+}
+
+static void tasdevice_set_power_state(
+	struct tasdevice_priv *tas_dev, int state)
+{
+	switch (state) {
+	case 0:
+		schedule_delayed_work(&tas_dev->powercontrol_work,
+			msecs_to_jiffies(20));
+		break;
+	default:
+		if (!(tas_dev->pstream || tas_dev->cstream)) {
+			if (tas_dev->chip_id != GENERAL_AUDEV)
+				tasdevice_enable_irq(tas_dev, false);
+			tasdevice_select_cfg_blk(tas_dev,
+				tas_dev->mtRegbin.profile_cfg_id,
+				TASDEVICE_BIN_BLK_PRE_SHUTDOWN);
+		}
+		break;
+	}
+}
+
+static int tasdevice_mute(struct snd_soc_dai *dai, int mute, int stream)
+{
+	struct snd_soc_component *codec = dai->component;
+	struct tasdevice_priv *tas_dev = snd_soc_component_get_drvdata(codec);
+
+	/* Codec Lock Hold */
+	mutex_lock(&tas_dev->codec_lock);
+
+	if (mute) {
+		/* stop DSP only when both playback and capture streams
+		 * are deactivated
+		 */
+		if (stream == SNDRV_PCM_STREAM_PLAYBACK)
+			tas_dev->pstream = 0;
+		else
+			tas_dev->cstream = 0;
+		if (tas_dev->pstream != 0 || tas_dev->cstream != 0)
+			goto out;
+	} else {
+		if (stream == SNDRV_PCM_STREAM_PLAYBACK)
+			tas_dev->pstream = 1;
+		else
+			tas_dev->cstream = 1;
+
+	}
+	tasdevice_set_power_state(tas_dev, mute);
+out:
+	/* Codec Lock Release*/
+	mutex_unlock(&tas_dev->codec_lock);
+
+	return 0;
+}
+
+static const struct snd_soc_dai_ops tasdevice_dai_ops = {
+	.startup = tasdevice_startup,
+	.hw_params = tasdevice_hw_params,
+	.set_sysclk = tasdevice_set_dai_sysclk,
+	.mute_stream = tasdevice_mute,
+};
+
+static struct snd_soc_dai_driver tasdevice_dai_driver[] = {
+	{
+		.name = "tas2781_codec",
+		.id = 0,
+		.playback = {
+			.stream_name	= "Playback",
+			.channels_min   = 1,
+			.channels_max   = 4,
+			.rates	 = TASDEVICE_RATES,
+			.formats	= TASDEVICE_FORMATS,
+		},
+		.capture = {
+			.stream_name	= "Capture",
+			.channels_min   = 1,
+			.channels_max   = 4,
+			.rates	 = TASDEVICE_RATES,
+			.formats	= TASDEVICE_FORMATS,
+		},
+		.ops = &tasdevice_dai_ops,
+		.symmetric_rate = 1,
+	},
+};
+
+static int tasdevice_codec_probe(
+	struct snd_soc_component *codec)
+{
+	struct tasdevice_priv *tas_dev =
+		snd_soc_component_get_drvdata(codec);
+	int ret;
+
+	/* Codec Lock Hold */
+	mutex_lock(&tas_dev->codec_lock);
+
+	tas_dev->codec = codec;
+	scnprintf(tas_dev->regbin_binaryname, 64, "%s_regbin.bin",
+		tas_dev->dev_name);
+	ret = request_firmware_nowait(THIS_MODULE, FW_ACTION_UEVENT,
+		tas_dev->regbin_binaryname, tas_dev->dev, GFP_KERNEL, tas_dev,
+		tasdevice_regbin_ready);
+	if (ret)
+		dev_err(tas_dev->dev,
+			"%s: request_firmware_nowait error:0x%08x\n",
+			__func__, ret);
+
+	/* Codec Lock Release*/
+	mutex_unlock(&tas_dev->codec_lock);
+
+	if (tas_dev->tas2781_reset != NULL)
+		tas_dev->tas2781_reset(tas_dev);
+	if (tas_dev->tas2781_set_global != NULL)
+		tas_dev->tas2781_set_global(tas_dev);
+
+	return ret;
+}
+
+static void tasdevice_deinit(void *pContext)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *) pContext;
+
+	tasdevice_config_info_remove(tas_dev);
+	tasdevice_dsp_remove(tas_dev);
+	tasdevice_calbin_remove(tas_dev);
+	tas_dev->fw_state = TASDEVICE_DSP_FW_PENDING;
+}
+
+static void tasdevice_codec_remove(
+	struct snd_soc_component *codec)
+{
+	struct tasdevice_priv *tas_dev =
+		snd_soc_component_get_drvdata(codec);
+	/* Codec Lock Hold */
+	mutex_lock(&tas_dev->codec_lock);
+	tasdevice_deinit(tas_dev);
+	/* Codec Lock Release*/
+	mutex_unlock(&tas_dev->codec_lock);
+
+	return;
+
+}
+
+static const struct snd_soc_component_driver
+	soc_codec_driver_tasdevice = {
+	.probe			= tasdevice_codec_probe,
+	.remove			= tasdevice_codec_remove,
+	.controls		= tas2781_snd_controls,
+	.num_controls		= ARRAY_SIZE(tas2781_snd_controls),
+	.dapm_widgets		= tasdevice_dapm_widgets,
+	.num_dapm_widgets	= ARRAY_SIZE(tasdevice_dapm_widgets),
+	.dapm_routes		= tasdevice_audio_map,
+	.num_dapm_routes	= ARRAY_SIZE(tasdevice_audio_map),
+	.idle_bias_on		= 1,
+	.endianness		= 1,
+};
+
+static void irq_work_routine(struct work_struct *pWork)
+{
+	struct tasdevice_priv *tas_dev =
+		container_of(pWork, struct tasdevice_priv,
+		mIrqInfo.irq_work.work);
+
+	mutex_lock(&tas_dev->codec_lock);
+	if (tas_dev->mb_runtime_suspend) {
+		dev_info(tas_dev->dev, "%s, Runtime Suspended\n", __func__);
+		goto end;
+	}
+	/*Logical Layer IRQ function, return is ignored*/
+	if (tas_dev->irq_work_func)
+		tas_dev->irq_work_func(tas_dev);
+	else
+		dev_info(tas_dev->dev,
+			"%s, irq_work_func is NULL\n", __func__);
+end:
+	mutex_unlock(&tas_dev->codec_lock);
+}
+
+static void tas2781_irq_work_func(struct tasdevice_priv *tas_dev)
+{
+	int rc = 0;
+	unsigned int reg_val = 0, array_size = 0, i = 0, ndev = 0;
+	unsigned int int_reg_array[] = {
+		TAS2781_REG_INT_LTCH0,
+		TAS2781_REG_INT_LTCH1,
+		TAS2781_REG_INT_LTCH1_0,
+		TAS2781_REG_INT_LTCH2,
+		TAS2781_REG_INT_LTCH3,
+		TAS2781_REG_INT_LTCH4};
+
+	tasdevice_enable_irq(tas_dev, false);
+
+	array_size = ARRAY_SIZE(int_reg_array);
+
+	for (ndev = 0; ndev < tas_dev->ndev; ndev++) {
+		for (i = 0; i < array_size; i++) {
+			rc = tasdevice_dev_read(tas_dev,
+				ndev, int_reg_array[i], &reg_val);
+			if (!rc)
+				dev_info(tas_dev->dev,
+					"INT STATUS REG 0x%04x=0x%02x\n",
+					int_reg_array[i], reg_val);
+			else
+				dev_err(tas_dev->dev,
+					"Read Reg 0x%04x error(rc=%d)\n",
+					int_reg_array[i], rc);
+		}
+	}
+
+}
+
+static irqreturn_t tasdevice_irq_handler(int irq,
+	void *dev_id)
+{
+	struct tasdevice_priv *tas_dev = (struct tasdevice_priv *)dev_id;
+;
+	/* get IRQ status after 100 ms */
+	schedule_delayed_work(&tas_dev->mIrqInfo.irq_work,
+		msecs_to_jiffies(100));
+	return IRQ_HANDLED;
+}
+
+int tasdevice_parse_dt(struct tasdevice_priv *tas_dev)
+{
+	struct device_node *np = tas_dev->dev->of_node;
+	int rc = 0, i = 0, ndev = 0;
+	char buf[32];
+
+
+	for (i = 0, ndev = 0; i < MaxChn; i++) {
+		rc = of_property_read_u32(np, dts_dev_addr_tag[i],
+			&(tas_dev->tasdevice[ndev].mnDevAddr));
+		if (rc)
+			dev_err(tas_dev->dev,
+			"Looking up %s property in node %s failed  %d\n",
+			dts_dev_addr_tag[i],
+			np->full_name, rc);
+		else {
+			dev_dbg(tas_dev->dev, "%s=0x%02x", dts_dev_addr_tag[i],
+				tas_dev->tasdevice[ndev].mnDevAddr);
+			ndev++;
+		}
+	}
+
+	rc = of_property_read_u32(np, dts_glb_addr_tag,
+			&(tas_dev->glb_addr.mnDevAddr));
+	if (rc) {
+		dev_err(tas_dev->dev,
+		"Looking up %s property in node %s failed %d\n",
+		dts_glb_addr_tag, np->full_name, rc);
+		tas_dev->glb_addr.mnDevAddr = 0;
+	}
+	tas_dev->ndev = (ndev == 0) ? 1 : ndev;
+
+	for (i = 0, ndev = 0; i < tas_dev->ndev; i++) {
+		scnprintf(buf, sizeof(buf), dts_rst_tag, "reset", i);
+		tas_dev->tasdevice[ndev].mnResetGpio = of_get_named_gpio(np,
+			buf, 0);
+		if (gpio_is_valid(tas_dev->tasdevice[ndev].mnResetGpio)) {
+			rc = gpio_request(tas_dev->tasdevice[ndev].mnResetGpio,
+				buf);
+			if (!rc) {
+				gpio_direction_output(
+					tas_dev->tasdevice[ndev].mnResetGpio,
+					1);
+				dev_info(tas_dev->dev, "%s = %d", buf,
+					tas_dev->tasdevice[ndev].mnResetGpio);
+				ndev++;
+			} else
+				dev_err(tas_dev->dev,
+					"%s: Failed to request dev[%d] gpio %d\n",
+					__func__, ndev,
+					tas_dev->tasdevice[ndev].mnResetGpio);
+		} else
+			dev_err(tas_dev->dev,
+				"Looking up %s property in node %s failed %d\n",
+				buf, np->full_name,
+				tas_dev->tasdevice[ndev].mnResetGpio);
+	}
+	strcpy(tas_dev->dev_name, tasdevice_id[tas_dev->chip_id].name);
+	if (tas_dev->chip_id != GENERAL_AUDEV) {
+		tas_dev->mIrqInfo.mn_irq_gpio = of_get_named_gpio(np,
+			"ti,irq-gpio", 0);
+		if (gpio_is_valid(tas_dev->mIrqInfo.mn_irq_gpio)) {
+			dev_dbg(tas_dev->dev, "irq-gpio = %d",
+				tas_dev->mIrqInfo.mn_irq_gpio);
+			INIT_DELAYED_WORK(&tas_dev->mIrqInfo.irq_work,
+				irq_work_routine);
+
+			rc = gpio_request(tas_dev->mIrqInfo.mn_irq_gpio,
+						"AUDEV-IRQ");
+			if (!rc) {
+				gpio_direction_input(
+					tas_dev->mIrqInfo.mn_irq_gpio);
+
+				tas_dev->mIrqInfo.mn_irq =
+					gpio_to_irq(
+					tas_dev->mIrqInfo.mn_irq_gpio);
+				dev_info(tas_dev->dev,
+					"irq = %d\n",
+					tas_dev->mIrqInfo.mn_irq);
+
+				rc = request_threaded_irq(
+					tas_dev->mIrqInfo.mn_irq,
+					tasdevice_irq_handler,
+					NULL, IRQF_TRIGGER_FALLING|
+					IRQF_ONESHOT,
+					SMARTAMP_MODULE_NAME, tas_dev);
+				if (!rc)
+					disable_irq_nosync(
+						tas_dev->mIrqInfo.mn_irq);
+				else
+					dev_err(tas_dev->dev,
+						"request_irq failed, %d\n",
+						rc);
+			} else
+				dev_err(tas_dev->dev,
+					"%s: GPIO %d request error\n",
+					__func__,
+					tas_dev->mIrqInfo.mn_irq_gpio);
+		} else
+			dev_err(tas_dev->dev,
+				"Looking up irq-gpio property in node %s failed %d\n",
+				np->full_name,
+				tas_dev->mIrqInfo.mn_irq_gpio);
+	}
+
+	if (tas_dev->chip_id != GENERAL_AUDEV && rc == 0) {
+		if (tas_dev->chip_id == TAS2781)
+			tas_dev->irq_work_func = tas2781_irq_work_func;
+		else
+			dev_info(tas_dev->dev, "%s: No match irq_work_func\n",
+				__func__);
+	}
+
+	return 0;
+}
+
+static int tasdevice_regmap_write(
+	struct tasdevice_priv *tas_dev,
+	unsigned int reg, unsigned int value)
+{
+	int nResult = 0;
+	int retry_count = TASDEVICE_RETRY_COUNT;
+
+	while (retry_count--) {
+		nResult = regmap_write(tas_dev->regmap, reg,
+			value);
+		if (nResult >= 0)
+			break;
+		usleep_range(5000, 5050);
+	}
+	if (retry_count == -1)
+		return TASDEVICE_ERROR_FAILED;
+	else
+		return 0;
+}
+
+static int tasdevice_regmap_bulk_write(
+	struct tasdevice_priv *tas_dev, unsigned int reg,
+	unsigned char *pData, unsigned int nLength)
+{
+	int nResult = 0;
+	int retry_count = TASDEVICE_RETRY_COUNT;
+
+	while (retry_count--) {
+		nResult = regmap_bulk_write(tas_dev->regmap, reg,
+				pData, nLength);
+		if (nResult >= 0)
+			break;
+		usleep_range(5000, 5050);
+	}
+	if (retry_count == -1)
+		return TASDEVICE_ERROR_FAILED;
+	else
+		return 0;
+}
+
+static int tasdevice_regmap_read(
+	struct tasdevice_priv *tas_dev,
+	unsigned int reg, unsigned int *value)
+{
+	int nResult = 0;
+	int retry_count = TASDEVICE_RETRY_COUNT;
+
+	while (retry_count--) {
+		nResult = regmap_read(tas_dev->regmap, reg,
+			value);
+		if (nResult >= 0)
+			break;
+		usleep_range(5000, 5050);
+	}
+	if (retry_count == -1)
+		return TASDEVICE_ERROR_FAILED;
+	else
+		return 0;
+}
+
+static int tasdevice_regmap_bulk_read(
+	struct tasdevice_priv *tas_dev, unsigned int reg,
+	unsigned char *pData, unsigned int nLength)
+{
+	int nResult = 0;
+	int retry_count = TASDEVICE_RETRY_COUNT;
+
+	while (retry_count--) {
+		nResult = regmap_bulk_read(tas_dev->regmap, reg,
+			pData, nLength);
+		if (nResult >= 0)
+			break;
+		usleep_range(5000, 5050);
+	}
+	if (retry_count == -1)
+		return TASDEVICE_ERROR_FAILED;
+	else
+		return 0;
+}
+
+static int tasdevice_regmap_update_bits(
+	struct tasdevice_priv *tas_dev, unsigned int reg,
+	unsigned int mask, unsigned int value)
+{
+	int nResult = 0;
+	int retry_count = TASDEVICE_RETRY_COUNT;
+
+	while (retry_count--) {
+		nResult = regmap_update_bits(tas_dev->regmap, reg,
+			mask, value);
+		if (nResult >= 0)
+			break;
+		usleep_range(5000, 5050);
+	}
+	if (retry_count == -1)
+		return TASDEVICE_ERROR_FAILED;
+	else
+		return 0;
+}
+
+static int tasdevice_change_chn_book_page(
+	struct tasdevice_priv *tas_dev, enum channel chn,
+	int book, int page)
+{
+	int n_result = 0;
+	struct i2c_client *pClient =
+		(struct i2c_client *)tas_dev->client;
+
+	if (chn < tas_dev->ndev) {
+		if (tas_dev->glb_addr.ref_cnt != 0) {
+			tas_dev->glb_addr.ref_cnt = 0;
+			tas_dev->glb_addr.mnBkPg.mnBook = -1;
+			tas_dev->glb_addr.mnBkPg.mnPage = -1;
+		}
+		pClient->addr = tas_dev->tasdevice[chn].mnDevAddr;
+		if (tas_dev->tasdevice[chn].mnBkPg.mnBook != book) {
+			n_result = tasdevice_regmap_write(tas_dev,
+				TASDEVICE_BOOKCTL_PAGE, 0);
+			if (n_result < 0) {
+				dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+					__func__, n_result);
+				goto out;
+			}
+			tas_dev->tasdevice[chn].mnBkPg.mnPage = 0;
+			n_result = tasdevice_regmap_write(tas_dev,
+				TASDEVICE_BOOKCTL_REG, book);
+			if (n_result < 0) {
+				dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+					__func__, n_result);
+				goto out;
+			}
+			tas_dev->tasdevice[chn].mnBkPg.mnBook = book;
+		}
+
+		if (tas_dev->tasdevice[chn].mnBkPg.mnPage != page) {
+			n_result = tasdevice_regmap_write(tas_dev,
+				TASDEVICE_BOOKCTL_PAGE, page);
+			if (n_result < 0) {
+				dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+					__func__, n_result);
+				goto out;
+			}
+			tas_dev->tasdevice[chn].mnBkPg.mnPage = page;
+		}
+	} else if (chn == tas_dev->ndev) {
+		int i = 0;
+
+		if (tas_dev->glb_addr.ref_cnt == 0)
+			for (i = 0; i < tas_dev->ndev; i++) {
+				tas_dev->tasdevice[i].mnBkPg.mnBook
+					= -1;
+				tas_dev->tasdevice[i].mnBkPg.mnPage
+					= -1;
+			}
+		pClient->addr = tas_dev->glb_addr.mnDevAddr;
+		if (tas_dev->glb_addr.mnBkPg.mnBook != book) {
+			n_result = tasdevice_regmap_write(tas_dev,
+				TASDEVICE_BOOKCTL_PAGE, 0);
+			if (n_result < 0) {
+				dev_err(tas_dev->dev,
+					"%s, 0ERROR, E=%d\n",
+					__func__, n_result);
+				goto out;
+			}
+			tas_dev->glb_addr.mnBkPg.mnPage = 0;
+			n_result = tasdevice_regmap_write(tas_dev,
+				TASDEVICE_BOOKCTL_REG, book);
+			if (n_result < 0) {
+				dev_err(tas_dev->dev,
+					"%s, book%xERROR, E=%d\n",
+					__func__, book, n_result);
+				goto out;
+			}
+			tas_dev->glb_addr.mnBkPg.mnBook = book;
+		}
+
+		if (tas_dev->glb_addr.mnBkPg.mnPage != page) {
+			n_result = tasdevice_regmap_write(tas_dev,
+				TASDEVICE_BOOKCTL_PAGE, page);
+			if (n_result < 0) {
+				dev_err(tas_dev->dev,
+					"%s, page%xERROR, E=%d\n",
+					__func__, page, n_result);
+				goto out;
+			}
+			tas_dev->glb_addr.mnBkPg.mnPage = page;
+		}
+		tas_dev->glb_addr.ref_cnt++;
+	} else
+		dev_err(tas_dev->dev,
+			"%s, ERROR, no such channel(%d)\n",
+			__func__, chn);
+
+out:
+	return n_result;
+}
+
+int tasdevice_dev_read(struct tasdevice_priv *tas_dev,
+	enum channel chn, unsigned int reg, unsigned int *pValue)
+{
+	int n_result = 0;
+
+	mutex_lock(&tas_dev->dev_lock);
+	if (chn < tas_dev->ndev) {
+		n_result = tasdevice_change_chn_book_page(tas_dev, chn,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg));
+		if (n_result < 0)
+			goto out;
+
+		n_result = tasdevice_regmap_read(tas_dev,
+			TASDEVICE_PAGE_REG(reg), pValue);
+		if (n_result < 0)
+			dev_err(tas_dev->dev, "%s, ERROR,E=%d\n",
+				__func__, n_result);
+		else
+			dev_dbg(tas_dev->dev,
+			"%s: chn:0x%02x:BOOK:PAGE:REG 0x%02x:0x%02x:0x%02x, 0x%02x\n",
+			__func__,
+			tas_dev->tasdevice[chn].mnDevAddr,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg),
+			TASDEVICE_PAGE_REG(reg), *pValue);
+	} else {
+
+		dev_err(tas_dev->dev, "%s, ERROR, no such channel(%d)\n",
+			__func__, chn);
+	}
+
+out:
+	mutex_unlock(&tas_dev->dev_lock);
+	return n_result;
+}
+
+int tasdevice_dev_write(struct tasdevice_priv *tas_dev,
+	enum channel chn, unsigned int reg, unsigned int value)
+{
+	int n_result = 0;
+
+	mutex_lock(&tas_dev->dev_lock);
+	if (chn <= tas_dev->ndev) {
+		n_result = tasdevice_change_chn_book_page(tas_dev, chn,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg));
+		if (n_result < 0)
+			goto out;
+
+		n_result = tasdevice_regmap_write(tas_dev,
+			TASDEVICE_PAGE_REG(reg), value);
+		if (n_result < 0)
+			dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+				__func__, n_result);
+		else {
+			dev_dbg(tas_dev->dev,
+			"%s: %s-0x%02x:BOOK:PAGE:REG 0x%02x:0x%02x:0x%02x, VAL: 0x%02x\n",
+			__func__, (chn == tas_dev->ndev)?"glb":"chn",
+			(chn == tas_dev->ndev) ?
+			tas_dev->glb_addr.mnDevAddr
+			: tas_dev->tasdevice[chn].mnDevAddr,
+			TASDEVICE_BOOK_ID(reg),
+			TASDEVICE_PAGE_ID(reg),
+			TASDEVICE_PAGE_REG(reg), value);
+		}
+	} else
+		dev_err(tas_dev->dev, "%s, ERROR, no such channel(%d)\n",
+			__func__, chn);
+out:
+	mutex_unlock(&tas_dev->dev_lock);
+	return n_result;
+}
+
+int tasdevice_dev_bulk_write(
+	struct tasdevice_priv *tas_dev, enum channel chn,
+	unsigned int reg, unsigned char *p_data,
+	unsigned int n_length)
+{
+	int n_result = 0;
+
+	mutex_lock(&tas_dev->dev_lock);
+	if (chn <= tas_dev->ndev) {
+		n_result = tasdevice_change_chn_book_page(tas_dev, chn,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg));
+		if (n_result < 0)
+			goto out;
+
+		n_result = tasdevice_regmap_bulk_write(tas_dev,
+			TASDEVICE_PAGE_REG(reg), p_data, n_length);
+		if (n_result < 0)
+			dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+				__func__, n_result);
+		else {
+			dev_dbg(tas_dev->dev,
+			"%s: %s-0x%02x:BOOK:PAGE:REG 0x%02x:0x%02x: 0x%02x, len: 0x%02x\n",
+			__func__,
+			(chn == tas_dev->ndev)?"glb":"chn",
+			(chn == tas_dev->ndev) ?
+			tas_dev->glb_addr.mnDevAddr
+			: tas_dev->tasdevice[chn].mnDevAddr,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg),
+			TASDEVICE_PAGE_REG(reg), n_length);
+		}
+	} else
+		dev_err(tas_dev->dev, "%s, ERROR, no such channel(%d)\n",
+			__func__, chn);
+out:
+	mutex_unlock(&tas_dev->dev_lock);
+	return n_result;
+}
+
+int tasdevice_dev_bulk_read(struct tasdevice_priv *tas_dev,
+	enum channel chn, unsigned int reg, unsigned char *p_data,
+	unsigned int n_length)
+{
+	int n_result = 0;
+
+	mutex_lock(&tas_dev->dev_lock);
+	if (chn < tas_dev->ndev) {
+		n_result = tasdevice_change_chn_book_page(tas_dev, chn,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg));
+		if (n_result < 0)
+			goto out;
+
+		n_result = tasdevice_regmap_bulk_read(tas_dev,
+			TASDEVICE_PAGE_REG(reg), p_data, n_length);
+		if (n_result < 0)
+			dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+				__func__, n_result);
+		else
+			dev_dbg(tas_dev->dev,
+			"%s: chn0x%02x:BOOK:PAGE:REG 0x%02x:0x%02x: 0x%02x, len: 0x%02x\n",
+			__func__,
+			tas_dev->tasdevice[chn].mnDevAddr,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg),
+			TASDEVICE_PAGE_REG(reg), n_length);
+	} else
+		dev_err(tas_dev->dev, "%s, ERROR, no such channel(%d)\n",
+			__func__, chn);
+
+out:
+	mutex_unlock(&tas_dev->dev_lock);
+	return n_result;
+}
+
+int tasdevice_dev_update_bits(
+	struct tasdevice_priv *tas_dev, enum channel chn,
+	unsigned int reg, unsigned int mask, unsigned int value)
+{
+	int n_result = 0;
+	struct i2c_client *pClient =
+		(struct i2c_client *)tas_dev->client;
+
+	mutex_lock(&tas_dev->dev_lock);
+	if (chn < tas_dev->ndev) {
+		n_result = tasdevice_change_chn_book_page(tas_dev, chn,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg));
+		if (n_result < 0)
+			goto out;
+		pClient->addr = tas_dev->tasdevice[chn].mnDevAddr;
+		n_result = tasdevice_regmap_update_bits(tas_dev,
+			TASDEVICE_PAGE_REG(reg), mask, value);
+		if (n_result < 0)
+			dev_err(tas_dev->dev, "%s, ERROR, E=%d\n",
+				__func__, n_result);
+		else
+			dev_dbg(tas_dev->dev,
+			"%s: chn0x%02x:BOOK:PAGE:REG 0x%02x:0x%02x:0x%02x, mask: 0x%02x, val: 0x%02x\n",
+			__func__, tas_dev->tasdevice[chn].mnDevAddr,
+			TASDEVICE_BOOK_ID(reg), TASDEVICE_PAGE_ID(reg),
+			TASDEVICE_PAGE_REG(reg), mask, value);
+	} else {
+		dev_err(tas_dev->dev, "%s, ERROR, no such channel(%d)\n",
+			__func__, chn);
+		n_result = -1;
+	}
+
+out:
+	mutex_unlock(&tas_dev->dev_lock);
+	return n_result;
+}
+
+static void tas2781_reset(struct tasdevice_priv *tas_dev)
+{
+	int ret = 0;
+	int i = 0;
+
+	for (; i < tas_dev->ndev; i++) {
+		ret = tasdevice_dev_write(tas_dev, i,
+			TAS2871_REG_SWRESET,
+			TAS2871_REG_SWRESET_RESET);
+		if (ret < 0) {
+			dev_err(tas_dev->dev, "%s: chn %d reset fail, %d\n",
+				__func__, i, ret);
+			continue;
+		}
+	}
+	usleep_range(1000, 1050);
+}
+
+static void tas2781_set_global(struct tasdevice_priv *tas_dev)
+{
+	int i = 0;
+	int ret = 0;
+
+	for (; i < tas_dev->ndev; i++) {
+		ret = tasdevice_dev_update_bits(tas_dev, i,
+			TAS2871_MISC_CFG2,
+			TAS2871_GLOBAL_MASK, 0x02);
+		if (ret < 0) {
+			dev_err(tas_dev->dev, "%s: chn %d set global fail, %d\n",
+				__func__, i, ret);
+			continue;
+		}
+	}
+}
+
+static int tasdevice_init(struct tasdevice_priv *tas_dev)
+{
+	int nResult = 0, i = 0;
+
+	tas_dev->mnCurrentProgram = -1;
+	tas_dev->mnCurrentConfiguration = -1;
+
+	for (i = 0; i < tas_dev->ndev; i++) {
+		tas_dev->tasdevice[i].mnBkPg.mnBook = -1;
+		tas_dev->tasdevice[i].mnBkPg.mnPage = -1;
+		tas_dev->tasdevice[i].mnCurrentProgram = -1;
+		tas_dev->tasdevice[i].mnCurrentConfiguration = -1;
+	}
+	mutex_init(&tas_dev->dev_lock);
+	tas_dev->read = tasdevice_dev_read;
+	tas_dev->write = tasdevice_dev_write;
+	tas_dev->bulk_read = tasdevice_dev_bulk_read;
+	tas_dev->bulk_write = tasdevice_dev_bulk_write;
+	tas_dev->set_calibration = tas2781_set_calibration;
+	tas_dev->tas2781_reset = tas2781_reset;
+	if (tas_dev->glb_addr.mnDevAddr != 0
+		&& tas_dev->glb_addr.mnDevAddr < 0x7F)
+		tas_dev->tas2781_set_global = tas2781_set_global;
+	dev_set_drvdata(tas_dev->dev, tas_dev);
+
+	INIT_DELAYED_WORK(&tas_dev->powercontrol_work,
+		powercontrol_routine);
+
+	mutex_init(&tas_dev->codec_lock);
+	nResult = devm_snd_soc_register_component(tas_dev->dev,
+		&soc_codec_driver_tasdevice,
+		tasdevice_dai_driver, ARRAY_SIZE(tasdevice_dai_driver));
+	if (nResult)
+		dev_err(tas_dev->dev, "%s: codec register error:0x%08x\n",
+			__func__, nResult);
+
+	INIT_DELAYED_WORK(&tas_dev->mIrqInfo.irq_work, irq_work_routine);
+	tas_dev->mIrqInfo.mb_irq_enable = false;
+
+	dev_info(tas_dev->dev, "i2c register success\n");
+
+	return nResult;
+}
+
+void tasdevice_remove(struct tasdevice_priv *tas_dev)
+{
+	int i = 0;
+
+	for (i = 0; i < tas_dev->mtRstGPIOs.ndev; i++) {
+		if (gpio_is_valid(tas_dev->mtRstGPIOs.mnResetGpio[i]))
+			gpio_free(tas_dev->mtRstGPIOs.mnResetGpio[i]);
+	}
+
+	for (i = 0; i < tas_dev->ndev; i++) {
+		if (gpio_is_valid(tas_dev->tasdevice[i].mnIRQGPIO))
+			gpio_free(tas_dev->tasdevice[i].mnIRQGPIO);
+	}
+
+	if (delayed_work_pending(&tas_dev->mIrqInfo.irq_work)) {
+		dev_info(tas_dev->dev, "cancel IRQ work\n");
+		cancel_delayed_work(&tas_dev->mIrqInfo.irq_work);
+	}
+	cancel_delayed_work_sync(&tas_dev->mIrqInfo.irq_work);
+
+	mutex_destroy(&tas_dev->dev_lock);
+	mutex_destroy(&tas_dev->codec_lock);
+	snd_soc_unregister_component(tas_dev->dev);
+}
+
+static int tasdevice_pm_suspend(struct device *dev)
+{
+	struct tasdevice_priv *tas_dev = dev_get_drvdata(dev);
+
+	if (!tas_dev) {
+		dev_err(tas_dev->dev, "%s: drvdata is NULL\n", __func__);
+		return -EINVAL;
+	}
+
+	mutex_lock(&tas_dev->codec_lock);
+
+	tas_dev->mb_runtime_suspend = true;
+
+	if (tas_dev->chip_id != GENERAL_AUDEV) {
+		if (delayed_work_pending(&tas_dev->mIrqInfo.irq_work)) {
+			dev_dbg(tas_dev->dev, "cancel IRQ work\n");
+			cancel_delayed_work_sync(&tas_dev->mIrqInfo.irq_work);
+		}
+	}
+	mutex_unlock(&tas_dev->codec_lock);
+	return 0;
+}
+
+static int tasdevice_pm_resume(struct device *dev)
+{
+	struct tasdevice_priv *tas_dev = dev_get_drvdata(dev);
+
+	if (!tas_dev)
+		return -EINVAL;
+
+	mutex_lock(&tas_dev->codec_lock);
+	tas_dev->mb_runtime_suspend = false;
+	mutex_unlock(&tas_dev->codec_lock);
+	return 0;
+}
+
+const struct dev_pm_ops tasdevice_pm_ops = {
+	.suspend = tasdevice_pm_suspend,
+	.resume = tasdevice_pm_resume
+};
+
+static int tasdevice_i2c_probe(struct i2c_client *i2c,
+	const struct i2c_device_id *id)
+{
+	struct tasdevice_priv *tas_dev = NULL;
+	int ret = 0;
+
+	if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_I2C)) {
+		dev_err(&i2c->dev,
+			"%s: I2C check failed\n", __func__);
+		ret = -ENODEV;
+		goto out;
+	}
+
+	tas_dev = devm_kzalloc(&i2c->dev, sizeof(*tas_dev), GFP_KERNEL);
+	if (!tas_dev) {
+		ret = -ENOMEM;
+		goto out;
+	}
+	tas_dev->dev = &i2c->dev;
+	tas_dev->client = (void *)i2c;
+	tas_dev->chip_id = id->driver_data;
+
+	if (i2c->dev.of_node)
+		ret = tasdevice_parse_dt(tas_dev);
+	else {
+		dev_err(tas_dev->dev, "No DTS info\n");
+		goto out;
+	}
+
+	tas_dev->regmap = devm_regmap_init_i2c(i2c,
+		&tasdevice_regmap);
+	if (IS_ERR(tas_dev->regmap)) {
+		ret = PTR_ERR(tas_dev->regmap);
+		dev_err(&i2c->dev, "Failed to allocate register map: %d\n",
+			ret);
+		return ret;
+	}
+	ret = tasdevice_init(tas_dev);
+
+out:
+	if (ret < 0 && tas_dev != NULL)
+		tasdevice_remove(tas_dev);
+	return ret;
+
+}
+
+static void tasdevice_i2c_remove(struct i2c_client *pClient)
+{
+	struct tasdevice_priv *tas_dev = i2c_get_clientdata(pClient);
+
+	if (tas_dev)
+		tasdevice_remove(tas_dev);
+
+}
+
+static struct i2c_driver tasdevice_i2c_driver = {
+	.driver = {
+		.name = "tas2781-codec",
+		.owner = THIS_MODULE,
+		.of_match_table = of_match_ptr(tasdevice_of_match),
+		.pm = &tasdevice_pm_ops,
+	},
+	.probe	= tasdevice_i2c_probe,
+	.remove = tasdevice_i2c_remove,
+	.id_table = tasdevice_id,
+};
+
+module_i2c_driver(tasdevice_i2c_driver);
+
+MODULE_AUTHOR("Shenghao Ding <shenghao-ding@...com>");
+MODULE_AUTHOR("Kevin Lu <kevin-lu@...com>");
+MODULE_DESCRIPTION("ASoC TAS2781 Driver");
+MODULE_LICENSE("GPL");
diff --git a/sound/soc/codecs/tas2781.h b/sound/soc/codecs/tas2781.h
new file mode 100644
index 000000000..db4ec752b
--- /dev/null
+++ b/sound/soc/codecs/tas2781.h
@@ -0,0 +1,208 @@
+/* SPDX-License-Identifier: GPL-2.0
+ *
+ * ALSA SoC Texas Instruments TAS2781 Audio Smart Amplifier
+ *
+ * Copyright (C) 2022 - 2023 Texas Instruments Incorporated
+ * https://www.ti.com
+ *
+ * The TAS2781 driver implements a flexible and configurable register setting
+ * for one, two, even multiple TAS2781 chips. All the register setting are in
+ * a bin file. Almost no specific register setting can be found in the code.
+ *
+ * Author: Shenghao Ding <shenghao-ding@...com>
+ *         Kevin Lu <kevin-lu@...com>
+ */
+
+#ifndef __TAS2781_H__
+#define __TAS2781_H__
+
+#include "tas2781-dsp.h"
+
+#define SMARTAMP_MODULE_NAME	("tas2781")
+#define MAX_LENGTH				(128)
+
+#define TASDEVICE_RETRY_COUNT	(3)
+#define TASDEVICE_ERROR_FAILED	(-2)
+
+#define TASDEVICE_RATES	(SNDRV_PCM_RATE_44100 |\
+	SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_96000 |\
+	SNDRV_PCM_RATE_88200)
+#define TASDEVICE_MAX_CHANNELS (8)
+
+#define TASDEVICE_FORMATS	(SNDRV_PCM_FMTBIT_S16_LE | \
+	SNDRV_PCM_FMTBIT_S24_LE | \
+	SNDRV_PCM_FMTBIT_S32_LE)
+
+/*PAGE Control Register (available in page0 of each book) */
+#define TASDEVICE_PAGE_SELECT	(0x00)
+#define TASDEVICE_BOOKCTL_PAGE	(0x00)
+#define TASDEVICE_BOOKCTL_REG	(127)
+#define TASDEVICE_BOOK_ID(reg)		(reg / (256 * 128))
+#define TASDEVICE_PAGE_ID(reg)		((reg % (256 * 128)) / 128)
+#define TASDEVICE_PAGE_REG(reg)		((reg % (256 * 128)) % 128)
+#define TASDEVICE_REG(book, page, reg)	(((book * 256 * 128) + \
+					(page * 128)) + reg)
+
+	/*Software Reset */
+#define TAS2871_REG_SWRESET  TASDEVICE_REG(0x0, 0X0, 0x02)
+#define TAS2871_REG_SWRESET_RESET  (0x1 << 0)
+
+	/* Enable Global addresses */
+#define TAS2871_MISC_CFG2  TASDEVICE_REG(0x0, 0X0, 0x07)
+#define TAS2871_GLOBAL_MASK (0x1 << 1)
+
+#define SMS_HTONS(a, b)  ((((a)&0x00FF)<<8) | \
+				((b)&0x00FF))
+#define SMS_HTONL(a, b, c, d) ((((a)&0x000000FF)<<24) |\
+					(((b)&0x000000FF)<<16) | \
+					(((c)&0x000000FF)<<8) | \
+					((d)&0x000000FF))
+
+	/*I2C Checksum */
+#define TASDEVICE_I2CChecksum  TASDEVICE_REG(0x0, 0x0, 0x7E)
+
+	/* Volume control */
+#define TAS2781_DVC_LVL TASDEVICE_REG(0x0, 0x0, 0x1A)
+#define TAS2781_AMP_LEVEL TASDEVICE_REG(0x0, 0x0, 0x03)
+#define TAS2781_AMP_LEVEL_MASK GENMASK(5, 1)
+
+#define TASDEVICE_CMD_SING_W  (0x1)
+#define TASDEVICE_CMD_BURST  (0x2)
+#define TASDEVICE_CMD_DELAY  (0x3)
+#define TASDEVICE_CMD_FIELD_W  (0x4)
+
+enum audio_device {
+	GENERAL_AUDEV = 0,
+	TAS2781	  = 1,
+};
+
+struct smartpa_gpio_info {
+	unsigned char ndev;
+	int mnResetGpio[MaxChn];
+};
+
+#define SMS_HTONS(a, b)  ((((a)&0x00FF)<<8) | ((b)&0x00FF))
+#define SMS_HTONL(a, b, c, d) ((((a)&0x000000FF)<<24) | \
+	(((b)&0x000000FF)<<16) | (((c)&0x000000FF)<<8) | \
+	((d)&0x000000FF))
+
+struct Tbookpage {
+	unsigned char mnBook;
+	unsigned char mnPage;
+};
+
+struct Ttasdevice {
+	unsigned int mnDevAddr;
+	unsigned int mnErrCode;
+	short mnCurrentProgram;
+	short mnCurrentConfiguration;
+	short mnCurrentRegConf;
+	int mnIRQGPIO;
+	int mnResetGpio;
+	int mn_irq;
+	int PowerStatus;
+	bool bDSPBypass;
+	bool bIrq_enabled;
+	bool bLoading;
+	bool bLoaderr;
+	bool mbCalibrationLoaded;
+	struct Tbookpage mnBkPg;
+	struct TFirmware *mpCalFirmware;
+};
+
+struct global_addr {
+	struct Tbookpage mnBkPg;
+	unsigned int mnDevAddr;
+	int ref_cnt;
+};
+
+struct tas_control {
+	struct snd_kcontrol_new *tasdevice_profile_controls;
+	int nr_controls;
+};
+
+struct tasdevice_irqinfo {
+	int mn_irq_gpio;
+	int mn_irq;
+	struct delayed_work irq_work;
+	bool mb_irq_enable;
+};
+
+struct tasdevice_priv {
+	struct device *dev;
+	void *client;
+	struct regmap *regmap;
+	struct mutex codec_lock;
+	struct mutex dev_lock;
+	struct Ttasdevice tasdevice[MaxChn];
+	struct TFirmware *mpFirmware;
+	struct tasdevice_regbin mtRegbin;
+	struct smartpa_gpio_info mtRstGPIOs;
+	struct tasdevice_irqinfo mIrqInfo;
+	struct tas_control tas_ctrl;
+	struct global_addr glb_addr;
+	int mnCurrentProgram;
+	int mnCurrentConfiguration;
+	unsigned int chip_id;
+	int (*read)(struct tasdevice_priv *tas_dev, enum channel chn,
+		unsigned int reg, unsigned int *pValue);
+	int (*write)(struct tasdevice_priv *tas_dev, enum channel chn,
+		unsigned int reg, unsigned int Value);
+	int (*bulk_read)(struct tasdevice_priv *tas_dev, enum channel chn,
+		unsigned int reg, unsigned char *pData, unsigned int len);
+	int (*bulk_write)(struct tasdevice_priv *tas_dev, enum channel chn,
+		unsigned int reg, unsigned char *pData, unsigned int len);
+	int (*set_calibration)(void *tas_dev, enum channel chl,
+		int calibration);
+	void (*tas2781_reset)(struct tasdevice_priv *tas_dev);
+	void (*tas2781_set_global)(struct tasdevice_priv *tas_dev);
+	int (*fw_parse_variable_header)(struct tasdevice_priv *tas_dev,
+		const struct firmware *pFW, int offset);
+	int (*fw_parse_program_data)(struct tasdevice_priv *tas_dev,
+		struct TFirmware *pFirmware,
+		const struct firmware *pFW, int offset);
+	int (*fw_parse_configuration_data)(struct tasdevice_priv *tas_dev,
+		struct TFirmware *pFirmware,
+		const struct firmware *pFW, int offset);
+	int (*tasdevice_load_block)(struct tasdevice_priv *tas_dev,
+		struct TBlock *pBlock);
+	int (*fw_parse_calibration_data)(struct tasdevice_priv *tas_dev,
+		struct TFirmware *pFirmware,
+		const struct firmware *pFW, int offset);
+	void (*irq_work_func)(struct tasdevice_priv *tas_dev);
+	int fw_state;
+	unsigned int magic_num;
+	unsigned char ndev;
+	unsigned char dev_name[32];
+	unsigned char regbin_binaryname[64];
+	unsigned char dsp_binaryname[64];
+	unsigned char cal_binaryname[MaxChn][64];
+	bool mb_runtime_suspend;
+	struct delayed_work powercontrol_work;
+	void *codec;
+	int sysclk;
+	int pstream;
+	int cstream;
+	bool mbCalibrationLoaded;
+};
+
+int tasdevice_dev_read(struct tasdevice_priv *pPcmdev,
+	enum channel chn, unsigned int reg, unsigned int *pValue);
+int tasdevice_process_block(void *pContext,
+	unsigned char *data, unsigned char dev_idx, int sublocksize);
+int tasdevice_dev_write(struct tasdevice_priv *pPcmdev,
+	enum channel chn, unsigned int reg, unsigned int value);
+
+int tasdevice_dev_bulk_write(
+	struct tasdevice_priv *pPcmdev, enum channel chn,
+	unsigned int reg, unsigned char *p_data, unsigned int n_length);
+
+int tasdevice_dev_bulk_read(struct tasdevice_priv *pPcmdev,
+	enum channel chn, unsigned int reg, unsigned char *p_data,
+	unsigned int n_length);
+
+int tasdevice_dev_update_bits(
+	struct tasdevice_priv *pPcmdev, enum channel chn,
+	unsigned int reg, unsigned int mask, unsigned int value);
+
+#endif /*__TAS2781_H__ */
-- 
2.17.1


Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ