Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/sound/soc/codecs/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 104 kB image not shown  

Quelle  rt5663.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-only
/*
 * rt5663.c  --  RT5663 ALSA SoC audio codec driver
 *
 * Copyright 2016 Realtek Semiconductor Corp.
 * Author: Jack Yu <jack.yu@realtek.com>
 */

#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/pm.h>
#include <linux/i2c.h>
#include <linux/platform_device.h>
#include <linux/spi/spi.h>
#include <linux/acpi.h>
#include <linux/regulator/consumer.h>
#include <linux/workqueue.h>
#include <sound/core.h>
#include <sound/pcm.h>
#include <sound/pcm_params.h>
#include <sound/jack.h>
#include <sound/soc.h>
#include <sound/soc-dapm.h>
#include <sound/initval.h>
#include <sound/tlv.h>

#include "rt5663.h"
#include "rl6231.h"

#define RT5663_DEVICE_ID_2 0x6451
#define RT5663_DEVICE_ID_1 0x6406

#define RT5663_POWER_ON_DELAY_MS 300
#define RT5663_SUPPLY_CURRENT_UA 500000

enum {
 CODEC_VER_1,
 CODEC_VER_0,
};

struct impedance_mapping_table {
 unsigned int imp_min;
 unsigned int imp_max;
 unsigned int vol;
 unsigned int dc_offset_l_manual;
 unsigned int dc_offset_r_manual;
 unsigned int dc_offset_l_manual_mic;
 unsigned int dc_offset_r_manual_mic;
};

static const char *const rt5663_supply_names[] = {
 "avdd",
 "cpvdd",
};

struct rt5663_priv {
 struct snd_soc_component *component;
 struct rt5663_platform_data pdata;
 struct regmap *regmap;
 struct delayed_work jack_detect_work, jd_unplug_work;
 struct snd_soc_jack *hs_jack;
 struct timer_list btn_check_timer;
 struct impedance_mapping_table *imp_table;
 struct regulator_bulk_data supplies[ARRAY_SIZE(rt5663_supply_names)];

 int codec_ver;
 int sysclk;
 int sysclk_src;
 int lrck;

 int pll_src;
 int pll_in;
 int pll_out;

 int jack_type;
 unsigned int irq;
};

static const struct reg_sequence rt5663_patch_list[] = {
 { 0x002a, 0x8020 },
 { 0x0086, 0x0028 },
 { 0x0100, 0xa020 },
 { 0x0117, 0x0f28 },
 { 0x02fb, 0x8089 },
};

static const struct reg_default rt5663_v2_reg[] = {
 { 0x0000, 0x0000 },
 { 0x0001, 0xc8c8 },
 { 0x0002, 0x8080 },
 { 0x0003, 0x8000 },
 { 0x0004, 0xc80a },
 { 0x0005, 0x0000 },
 { 0x0006, 0x0000 },
 { 0x0007, 0x0000 },
 { 0x000a, 0x0000 },
 { 0x000b, 0x0000 },
 { 0x000c, 0x0000 },
 { 0x000d, 0x0000 },
 { 0x000f, 0x0808 },
 { 0x0010, 0x4000 },
 { 0x0011, 0x0000 },
 { 0x0012, 0x1404 },
 { 0x0013, 0x1000 },
 { 0x0014, 0xa00a },
 { 0x0015, 0x0404 },
 { 0x0016, 0x0404 },
 { 0x0017, 0x0011 },
 { 0x0018, 0xafaf },
 { 0x0019, 0xafaf },
 { 0x001a, 0xafaf },
 { 0x001b, 0x0011 },
 { 0x001c, 0x2f2f },
 { 0x001d, 0x2f2f },
 { 0x001e, 0x2f2f },
 { 0x001f, 0x0000 },
 { 0x0020, 0x0000 },
 { 0x0021, 0x0000 },
 { 0x0022, 0x5757 },
 { 0x0023, 0x0039 },
 { 0x0024, 0x000b },
 { 0x0026, 0xc0c0 },
 { 0x0027, 0xc0c0 },
 { 0x0028, 0xc0c0 },
 { 0x0029, 0x8080 },
 { 0x002a, 0xaaaa },
 { 0x002b, 0xaaaa },
 { 0x002c, 0xaba8 },
 { 0x002d, 0x0000 },
 { 0x002e, 0x0000 },
 { 0x002f, 0x0000 },
 { 0x0030, 0x0000 },
 { 0x0031, 0x5000 },
 { 0x0032, 0x0000 },
 { 0x0033, 0x0000 },
 { 0x0034, 0x0000 },
 { 0x0035, 0x0000 },
 { 0x003a, 0x0000 },
 { 0x003b, 0x0000 },
 { 0x003c, 0x00ff },
 { 0x003d, 0x0000 },
 { 0x003e, 0x00ff },
 { 0x003f, 0x0000 },
 { 0x0040, 0x0000 },
 { 0x0041, 0x00ff },
 { 0x0042, 0x0000 },
 { 0x0043, 0x00ff },
 { 0x0044, 0x0c0c },
 { 0x0049, 0xc00b },
 { 0x004a, 0x0000 },
 { 0x004b, 0x031f },
 { 0x004d, 0x0000 },
 { 0x004e, 0x001f },
 { 0x004f, 0x0000 },
 { 0x0050, 0x001f },
 { 0x0052, 0xf000 },
 { 0x0061, 0x0000 },
 { 0x0062, 0x0000 },
 { 0x0063, 0x003e },
 { 0x0064, 0x0000 },
 { 0x0065, 0x0000 },
 { 0x0066, 0x003f },
 { 0x0067, 0x0000 },
 { 0x006b, 0x0000 },
 { 0x006d, 0xff00 },
 { 0x006e, 0x2808 },
 { 0x006f, 0x000a },
 { 0x0070, 0x8000 },
 { 0x0071, 0x8000 },
 { 0x0072, 0x8000 },
 { 0x0073, 0x7000 },
 { 0x0074, 0x7770 },
 { 0x0075, 0x0002 },
 { 0x0076, 0x0001 },
 { 0x0078, 0x00f0 },
 { 0x0079, 0x0000 },
 { 0x007a, 0x0000 },
 { 0x007b, 0x0000 },
 { 0x007c, 0x0000 },
 { 0x007d, 0x0123 },
 { 0x007e, 0x4500 },
 { 0x007f, 0x8003 },
 { 0x0080, 0x0000 },
 { 0x0081, 0x0000 },
 { 0x0082, 0x0000 },
 { 0x0083, 0x0000 },
 { 0x0084, 0x0000 },
 { 0x0085, 0x0000 },
 { 0x0086, 0x0008 },
 { 0x0087, 0x0000 },
 { 0x0088, 0x0000 },
 { 0x0089, 0x0000 },
 { 0x008a, 0x0000 },
 { 0x008b, 0x0000 },
 { 0x008c, 0x0003 },
 { 0x008e, 0x0060 },
 { 0x008f, 0x1000 },
 { 0x0091, 0x0c26 },
 { 0x0092, 0x0073 },
 { 0x0093, 0x0000 },
 { 0x0094, 0x0080 },
 { 0x0098, 0x0000 },
 { 0x0099, 0x0000 },
 { 0x009a, 0x0007 },
 { 0x009f, 0x0000 },
 { 0x00a0, 0x0000 },
 { 0x00a1, 0x0002 },
 { 0x00a2, 0x0001 },
 { 0x00a3, 0x0002 },
 { 0x00a4, 0x0001 },
 { 0x00ae, 0x2040 },
 { 0x00af, 0x0000 },
 { 0x00b6, 0x0000 },
 { 0x00b7, 0x0000 },
 { 0x00b8, 0x0000 },
 { 0x00b9, 0x0000 },
 { 0x00ba, 0x0002 },
 { 0x00bb, 0x0000 },
 { 0x00be, 0x0000 },
 { 0x00c0, 0x0000 },
 { 0x00c1, 0x0aaa },
 { 0x00c2, 0xaa80 },
 { 0x00c3, 0x0003 },
 { 0x00c4, 0x0000 },
 { 0x00d0, 0x0000 },
 { 0x00d1, 0x2244 },
 { 0x00d2, 0x0000 },
 { 0x00d3, 0x3300 },
 { 0x00d4, 0x2200 },
 { 0x00d9, 0x0809 },
 { 0x00da, 0x0000 },
 { 0x00db, 0x0008 },
 { 0x00dc, 0x00c0 },
 { 0x00dd, 0x6724 },
 { 0x00de, 0x3131 },
 { 0x00df, 0x0008 },
 { 0x00e0, 0x4000 },
 { 0x00e1, 0x3131 },
 { 0x00e2, 0x600c },
 { 0x00ea, 0xb320 },
 { 0x00eb, 0x0000 },
 { 0x00ec, 0xb300 },
 { 0x00ed, 0x0000 },
 { 0x00ee, 0xb320 },
 { 0x00ef, 0x0000 },
 { 0x00f0, 0x0201 },
 { 0x00f1, 0x0ddd },
 { 0x00f2, 0x0ddd },
 { 0x00f6, 0x0000 },
 { 0x00f7, 0x0000 },
 { 0x00f8, 0x0000 },
 { 0x00fa, 0x0000 },
 { 0x00fb, 0x0000 },
 { 0x00fc, 0x0000 },
 { 0x00fd, 0x0000 },
 { 0x00fe, 0x10ec },
 { 0x00ff, 0x6451 },
 { 0x0100, 0xaaaa },
 { 0x0101, 0x000a },
 { 0x010a, 0xaaaa },
 { 0x010b, 0xa0a0 },
 { 0x010c, 0xaeae },
 { 0x010d, 0xaaaa },
 { 0x010e, 0xaaaa },
 { 0x010f, 0xaaaa },
 { 0x0110, 0xe002 },
 { 0x0111, 0xa602 },
 { 0x0112, 0xaaaa },
 { 0x0113, 0x2000 },
 { 0x0117, 0x0f00 },
 { 0x0125, 0x0420 },
 { 0x0132, 0x0000 },
 { 0x0133, 0x0000 },
 { 0x0136, 0x5555 },
 { 0x0137, 0x5540 },
 { 0x0138, 0x3700 },
 { 0x0139, 0x79a1 },
 { 0x013a, 0x2020 },
 { 0x013b, 0x2020 },
 { 0x013c, 0x2005 },
 { 0x013f, 0x0000 },
 { 0x0145, 0x0002 },
 { 0x0146, 0x0000 },
 { 0x0147, 0x0000 },
 { 0x0148, 0x0000 },
 { 0x0160, 0x4ec0 },
 { 0x0161, 0x0080 },
 { 0x0162, 0x0200 },
 { 0x0163, 0x0800 },
 { 0x0164, 0x0000 },
 { 0x0165, 0x0000 },
 { 0x0166, 0x0000 },
 { 0x0167, 0x000f },
 { 0x0168, 0x000f },
 { 0x0170, 0x4e80 },
 { 0x0171, 0x0080 },
 { 0x0172, 0x0200 },
 { 0x0173, 0x0800 },
 { 0x0174, 0x00ff },
 { 0x0175, 0x0000 },
 { 0x0190, 0x4131 },
 { 0x0191, 0x4131 },
 { 0x0192, 0x4131 },
 { 0x0193, 0x4131 },
 { 0x0194, 0x0000 },
 { 0x0195, 0x0000 },
 { 0x0196, 0x0000 },
 { 0x0197, 0x0000 },
 { 0x0198, 0x0000 },
 { 0x0199, 0x0000 },
 { 0x01a0, 0x1e64 },
 { 0x01a1, 0x06a3 },
 { 0x01a2, 0x0000 },
 { 0x01a3, 0x0000 },
 { 0x01a4, 0x0000 },
 { 0x01a5, 0x0000 },
 { 0x01a6, 0x0000 },
 { 0x01a7, 0x0000 },
 { 0x01a8, 0x0000 },
 { 0x01a9, 0x0000 },
 { 0x01aa, 0x0000 },
 { 0x01ab, 0x0000 },
 { 0x01b5, 0x0000 },
 { 0x01b6, 0x01c3 },
 { 0x01b7, 0x02a0 },
 { 0x01b8, 0x03e9 },
 { 0x01b9, 0x1389 },
 { 0x01ba, 0xc351 },
 { 0x01bb, 0x0009 },
 { 0x01bc, 0x0018 },
 { 0x01bd, 0x002a },
 { 0x01be, 0x004c },
 { 0x01bf, 0x0097 },
 { 0x01c0, 0x433d },
 { 0x01c1, 0x0000 },
 { 0x01c2, 0x0000 },
 { 0x01c3, 0x0000 },
 { 0x01c4, 0x0000 },
 { 0x01c5, 0x0000 },
 { 0x01c6, 0x0000 },
 { 0x01c7, 0x0000 },
 { 0x01c8, 0x40af },
 { 0x01c9, 0x0702 },
 { 0x01ca, 0x0000 },
 { 0x01cb, 0x0000 },
 { 0x01cc, 0x5757 },
 { 0x01cd, 0x5757 },
 { 0x01ce, 0x5757 },
 { 0x01cf, 0x5757 },
 { 0x01d0, 0x5757 },
 { 0x01d1, 0x5757 },
 { 0x01d2, 0x5757 },
 { 0x01d3, 0x5757 },
 { 0x01d4, 0x5757 },
 { 0x01d5, 0x5757 },
 { 0x01d6, 0x003c },
 { 0x01da, 0x0000 },
 { 0x01db, 0x0000 },
 { 0x01dc, 0x0000 },
 { 0x01de, 0x7c00 },
 { 0x01df, 0x0320 },
 { 0x01e0, 0x06a1 },
 { 0x01e1, 0x0000 },
 { 0x01e2, 0x0000 },
 { 0x01e3, 0x0000 },
 { 0x01e4, 0x0000 },
 { 0x01e5, 0x0000 },
 { 0x01e6, 0x0001 },
 { 0x01e7, 0x0000 },
 { 0x01e8, 0x0000 },
 { 0x01ea, 0x0000 },
 { 0x01eb, 0x0000 },
 { 0x01ec, 0x0000 },
 { 0x01ed, 0x0000 },
 { 0x01ee, 0x0000 },
 { 0x01ef, 0x0000 },
 { 0x01f0, 0x0000 },
 { 0x01f1, 0x0000 },
 { 0x01f2, 0x0000 },
 { 0x01f3, 0x0000 },
 { 0x01f4, 0x0000 },
 { 0x0200, 0x0000 },
 { 0x0201, 0x0000 },
 { 0x0202, 0x0000 },
 { 0x0203, 0x0000 },
 { 0x0204, 0x0000 },
 { 0x0205, 0x0000 },
 { 0x0206, 0x0000 },
 { 0x0207, 0x0000 },
 { 0x0208, 0x0000 },
 { 0x0210, 0x60b1 },
 { 0x0211, 0xa000 },
 { 0x0212, 0x024c },
 { 0x0213, 0xf7ff },
 { 0x0214, 0x024c },
 { 0x0215, 0x0102 },
 { 0x0216, 0x00a3 },
 { 0x0217, 0x0048 },
 { 0x0218, 0x92c0 },
 { 0x0219, 0x0000 },
 { 0x021a, 0x00c8 },
 { 0x021b, 0x0020 },
 { 0x02fa, 0x0000 },
 { 0x02fb, 0x0000 },
 { 0x02fc, 0x0000 },
 { 0x02ff, 0x0110 },
 { 0x0300, 0x001f },
 { 0x0301, 0x032c },
 { 0x0302, 0x5f21 },
 { 0x0303, 0x4000 },
 { 0x0304, 0x4000 },
 { 0x0305, 0x06d5 },
 { 0x0306, 0x8000 },
 { 0x0307, 0x0700 },
 { 0x0310, 0x4560 },
 { 0x0311, 0xa4a8 },
 { 0x0312, 0x7418 },
 { 0x0313, 0x0000 },
 { 0x0314, 0x0006 },
 { 0x0315, 0xffff },
 { 0x0316, 0xc400 },
 { 0x0317, 0x0000 },
 { 0x0330, 0x00a6 },
 { 0x0331, 0x04c3 },
 { 0x0332, 0x27c8 },
 { 0x0333, 0xbf50 },
 { 0x0334, 0x0045 },
 { 0x0335, 0x0007 },
 { 0x0336, 0x7418 },
 { 0x0337, 0x0501 },
 { 0x0338, 0x0000 },
 { 0x0339, 0x0010 },
 { 0x033a, 0x1010 },
 { 0x03c0, 0x7e00 },
 { 0x03c1, 0x8000 },
 { 0x03c2, 0x8000 },
 { 0x03c3, 0x8000 },
 { 0x03c4, 0x8000 },
 { 0x03c5, 0x8000 },
 { 0x03c6, 0x8000 },
 { 0x03c7, 0x8000 },
 { 0x03c8, 0x8000 },
 { 0x03c9, 0x8000 },
 { 0x03ca, 0x8000 },
 { 0x03cb, 0x8000 },
 { 0x03cc, 0x8000 },
 { 0x03d0, 0x0000 },
 { 0x03d1, 0x0000 },
 { 0x03d2, 0x0000 },
 { 0x03d3, 0x0000 },
 { 0x03d4, 0x2000 },
 { 0x03d5, 0x2000 },
 { 0x03d6, 0x0000 },
 { 0x03d7, 0x0000 },
 { 0x03d8, 0x2000 },
 { 0x03d9, 0x2000 },
 { 0x03da, 0x2000 },
 { 0x03db, 0x2000 },
 { 0x03dc, 0x0000 },
 { 0x03dd, 0x0000 },
 { 0x03de, 0x0000 },
 { 0x03df, 0x2000 },
 { 0x03e0, 0x0000 },
 { 0x03e1, 0x0000 },
 { 0x03e2, 0x0000 },
 { 0x03e3, 0x0000 },
 { 0x03e4, 0x0000 },
 { 0x03e5, 0x0000 },
 { 0x03e6, 0x0000 },
 { 0x03e7, 0x0000 },
 { 0x03e8, 0x0000 },
 { 0x03e9, 0x0000 },
 { 0x03ea, 0x0000 },
 { 0x03eb, 0x0000 },
 { 0x03ec, 0x0000 },
 { 0x03ed, 0x0000 },
 { 0x03ee, 0x0000 },
 { 0x03ef, 0x0000 },
 { 0x03f0, 0x0800 },
 { 0x03f1, 0x0800 },
 { 0x03f2, 0x0800 },
 { 0x03f3, 0x0800 },
 { 0x03fe, 0x0000 },
 { 0x03ff, 0x0000 },
 { 0x07f0, 0x0000 },
 { 0x07fa, 0x0000 },
};

static const struct reg_default rt5663_reg[] = {
 { 0x0000, 0x0000 },
 { 0x0002, 0x0008 },
 { 0x0005, 0x1000 },
 { 0x0006, 0x1000 },
 { 0x000a, 0x0000 },
 { 0x0010, 0x000f },
 { 0x0015, 0x42f1 },
 { 0x0016, 0x0000 },
 { 0x0018, 0x000b },
 { 0x0019, 0xafaf },
 { 0x001c, 0x2f2f },
 { 0x001f, 0x0000 },
 { 0x0022, 0x5757 },
 { 0x0023, 0x0039 },
 { 0x0026, 0xc0c0 },
 { 0x0029, 0x8080 },
 { 0x002a, 0x8020 },
 { 0x002c, 0x000c },
 { 0x002d, 0x0000 },
 { 0x0040, 0x0808 },
 { 0x0061, 0x0000 },
 { 0x0062, 0x0000 },
 { 0x0063, 0x003e },
 { 0x0064, 0x0000 },
 { 0x0065, 0x0000 },
 { 0x0066, 0x0000 },
 { 0x006b, 0x0000 },
 { 0x006e, 0x0000 },
 { 0x006f, 0x0000 },
 { 0x0070, 0x8020 },
 { 0x0073, 0x1000 },
 { 0x0074, 0xe400 },
 { 0x0075, 0x0002 },
 { 0x0076, 0x0001 },
 { 0x0077, 0x00f0 },
 { 0x0078, 0x0000 },
 { 0x0079, 0x0000 },
 { 0x007a, 0x0123 },
 { 0x007b, 0x8003 },
 { 0x0080, 0x0000 },
 { 0x0081, 0x0000 },
 { 0x0082, 0x0000 },
 { 0x0083, 0x0000 },
 { 0x0084, 0x0000 },
 { 0x0086, 0x0028 },
 { 0x0087, 0x0000 },
 { 0x008a, 0x0000 },
 { 0x008b, 0x0000 },
 { 0x008c, 0x0003 },
 { 0x008e, 0x0008 },
 { 0x008f, 0x1000 },
 { 0x0090, 0x0646 },
 { 0x0091, 0x0e3e },
 { 0x0092, 0x1071 },
 { 0x0093, 0x0000 },
 { 0x0094, 0x0080 },
 { 0x0097, 0x0000 },
 { 0x0098, 0x0000 },
 { 0x009a, 0x0000 },
 { 0x009f, 0x0000 },
 { 0x00ae, 0x6000 },
 { 0x00af, 0x0000 },
 { 0x00b6, 0x0000 },
 { 0x00b7, 0x0000 },
 { 0x00b8, 0x0000 },
 { 0x00ba, 0x0000 },
 { 0x00bb, 0x0000 },
 { 0x00be, 0x0000 },
 { 0x00bf, 0x0000 },
 { 0x00c0, 0x0000 },
 { 0x00c1, 0x0000 },
 { 0x00c5, 0x0000 },
 { 0x00cb, 0xa02f },
 { 0x00cc, 0x0000 },
 { 0x00cd, 0x0e02 },
 { 0x00d9, 0x08f9 },
 { 0x00db, 0x0008 },
 { 0x00dc, 0x00c0 },
 { 0x00dd, 0x6729 },
 { 0x00de, 0x3131 },
 { 0x00df, 0x0008 },
 { 0x00e0, 0x4000 },
 { 0x00e1, 0x3131 },
 { 0x00e2, 0x0043 },
 { 0x00e4, 0x400b },
 { 0x00e5, 0x8031 },
 { 0x00e6, 0x3080 },
 { 0x00e7, 0x4100 },
 { 0x00e8, 0x1400 },
 { 0x00e9, 0xe00a },
 { 0x00ea, 0x0404 },
 { 0x00eb, 0x0404 },
 { 0x00ec, 0xb320 },
 { 0x00ed, 0x0000 },
 { 0x00f4, 0x0000 },
 { 0x00f6, 0x0000 },
 { 0x00f8, 0x0000 },
 { 0x00fa, 0x8000 },
 { 0x00fd, 0x0001 },
 { 0x00fe, 0x10ec },
 { 0x00ff, 0x6406 },
 { 0x0100, 0xa020 },
 { 0x0108, 0x4444 },
 { 0x0109, 0x4444 },
 { 0x010a, 0xaaaa },
 { 0x010b, 0x00a0 },
 { 0x010c, 0x8aaa },
 { 0x010d, 0xaaaa },
 { 0x010e, 0x2aaa },
 { 0x010f, 0x002a },
 { 0x0110, 0xa0a4 },
 { 0x0111, 0x4602 },
 { 0x0112, 0x0101 },
 { 0x0113, 0x2000 },
 { 0x0114, 0x0000 },
 { 0x0116, 0x0000 },
 { 0x0117, 0x0f28 },
 { 0x0118, 0x0006 },
 { 0x0125, 0x2424 },
 { 0x0126, 0x5550 },
 { 0x0127, 0x0400 },
 { 0x0128, 0x7711 },
 { 0x0132, 0x0004 },
 { 0x0137, 0x5441 },
 { 0x0139, 0x79a1 },
 { 0x013a, 0x30c0 },
 { 0x013b, 0x2000 },
 { 0x013c, 0x2005 },
 { 0x013d, 0x30c0 },
 { 0x013e, 0x0000 },
 { 0x0140, 0x3700 },
 { 0x0141, 0x1f00 },
 { 0x0144, 0x0000 },
 { 0x0145, 0x0002 },
 { 0x0146, 0x0000 },
 { 0x0160, 0x0e80 },
 { 0x0161, 0x0080 },
 { 0x0162, 0x0200 },
 { 0x0163, 0x0800 },
 { 0x0164, 0x0000 },
 { 0x0165, 0x0000 },
 { 0x0166, 0x0000 },
 { 0x0167, 0x1417 },
 { 0x0168, 0x0017 },
 { 0x0169, 0x0017 },
 { 0x0180, 0x2000 },
 { 0x0181, 0x0000 },
 { 0x0182, 0x0000 },
 { 0x0183, 0x2000 },
 { 0x0184, 0x0000 },
 { 0x0185, 0x0000 },
 { 0x01b0, 0x4b30 },
 { 0x01b1, 0x0000 },
 { 0x01b2, 0xd870 },
 { 0x01b3, 0x0000 },
 { 0x01b4, 0x0030 },
 { 0x01b5, 0x5757 },
 { 0x01b6, 0x5757 },
 { 0x01b7, 0x5757 },
 { 0x01b8, 0x5757 },
 { 0x01c0, 0x433d },
 { 0x01c1, 0x0540 },
 { 0x01c2, 0x0000 },
 { 0x01c3, 0x0000 },
 { 0x01c4, 0x0000 },
 { 0x01c5, 0x0009 },
 { 0x01c6, 0x0018 },
 { 0x01c7, 0x002a },
 { 0x01c8, 0x004c },
 { 0x01c9, 0x0097 },
 { 0x01ca, 0x01c3 },
 { 0x01cb, 0x03e9 },
 { 0x01cc, 0x1389 },
 { 0x01cd, 0xc351 },
 { 0x01ce, 0x0000 },
 { 0x01cf, 0x0000 },
 { 0x01d0, 0x0000 },
 { 0x01d1, 0x0000 },
 { 0x01d2, 0x0000 },
 { 0x01d3, 0x003c },
 { 0x01d4, 0x5757 },
 { 0x01d5, 0x5757 },
 { 0x01d6, 0x5757 },
 { 0x01d7, 0x5757 },
 { 0x01d8, 0x5757 },
 { 0x01d9, 0x5757 },
 { 0x01da, 0x0000 },
 { 0x01db, 0x0000 },
 { 0x01dd, 0x0009 },
 { 0x01de, 0x7f00 },
 { 0x01df, 0x00c8 },
 { 0x01e0, 0x0691 },
 { 0x01e1, 0x0000 },
 { 0x01e2, 0x0000 },
 { 0x01e3, 0x0000 },
 { 0x01e4, 0x0000 },
 { 0x01e5, 0x0040 },
 { 0x01e6, 0x0000 },
 { 0x01e7, 0x0000 },
 { 0x01e8, 0x0000 },
 { 0x01ea, 0x0000 },
 { 0x01eb, 0x0000 },
 { 0x01ec, 0x0000 },
 { 0x01ed, 0x0000 },
 { 0x01ee, 0x0000 },
 { 0x01ef, 0x0000 },
 { 0x01f0, 0x0000 },
 { 0x01f1, 0x0000 },
 { 0x01f2, 0x0000 },
 { 0x0200, 0x0000 },
 { 0x0201, 0x2244 },
 { 0x0202, 0xaaaa },
 { 0x0250, 0x8010 },
 { 0x0251, 0x0000 },
 { 0x0252, 0x028a },
 { 0x02fa, 0x0000 },
 { 0x02fb, 0x8089 },
 { 0x02fc, 0x0300 },
 { 0x0300, 0x0000 },
 { 0x03d0, 0x0000 },
 { 0x03d1, 0x0000 },
 { 0x03d2, 0x0000 },
 { 0x03d3, 0x0000 },
 { 0x03d4, 0x2000 },
 { 0x03d5, 0x2000 },
 { 0x03d6, 0x0000 },
 { 0x03d7, 0x0000 },
 { 0x03d8, 0x2000 },
 { 0x03d9, 0x2000 },
 { 0x03da, 0x2000 },
 { 0x03db, 0x2000 },
 { 0x03dc, 0x0000 },
 { 0x03dd, 0x0000 },
 { 0x03de, 0x0000 },
 { 0x03df, 0x2000 },
 { 0x03e0, 0x0000 },
 { 0x03e1, 0x0000 },
 { 0x03e2, 0x0000 },
 { 0x03e3, 0x0000 },
 { 0x03e4, 0x0000 },
 { 0x03e5, 0x0000 },
 { 0x03e6, 0x0000 },
 { 0x03e7, 0x0000 },
 { 0x03e8, 0x0000 },
 { 0x03e9, 0x0000 },
 { 0x03ea, 0x0000 },
 { 0x03eb, 0x0000 },
 { 0x03ec, 0x0000 },
 { 0x03ed, 0x0000 },
 { 0x03ee, 0x0000 },
 { 0x03ef, 0x0000 },
 { 0x03f0, 0x0800 },
 { 0x03f1, 0x0800 },
 { 0x03f2, 0x0800 },
 { 0x03f3, 0x0800 },
};

static bool rt5663_volatile_register(struct device *dev, unsigned int reg)
{
 switch (reg) {
 case RT5663_RESET:
 case RT5663_SIL_DET_CTL:
 case RT5663_HP_IMP_GAIN_2:
 case RT5663_AD_DA_MIXER:
 case RT5663_FRAC_DIV_2:
 case RT5663_MICBIAS_1:
 case RT5663_ASRC_11_2:
 case RT5663_ADC_EQ_1:
 case RT5663_INT_ST_1:
 case RT5663_INT_ST_2:
 case RT5663_GPIO_STA1:
 case RT5663_SIN_GEN_1:
 case RT5663_IL_CMD_1:
 case RT5663_IL_CMD_5:
 case RT5663_IL_CMD_PWRSAV1:
 case RT5663_EM_JACK_TYPE_1:
 case RT5663_EM_JACK_TYPE_2:
 case RT5663_EM_JACK_TYPE_3:
 case RT5663_JD_CTRL2:
 case RT5663_VENDOR_ID:
 case RT5663_VENDOR_ID_1:
 case RT5663_VENDOR_ID_2:
 case RT5663_PLL_INT_REG:
 case RT5663_SOFT_RAMP:
 case RT5663_STO_DRE_1:
 case RT5663_STO_DRE_5:
 case RT5663_STO_DRE_6:
 case RT5663_STO_DRE_7:
 case RT5663_MIC_DECRO_1:
 case RT5663_MIC_DECRO_4:
 case RT5663_HP_IMP_SEN_1:
 case RT5663_HP_IMP_SEN_3:
 case RT5663_HP_IMP_SEN_4:
 case RT5663_HP_IMP_SEN_5:
 case RT5663_HP_CALIB_1_1:
 case RT5663_HP_CALIB_9:
 case RT5663_HP_CALIB_ST1:
 case RT5663_HP_CALIB_ST2:
 case RT5663_HP_CALIB_ST3:
 case RT5663_HP_CALIB_ST4:
 case RT5663_HP_CALIB_ST5:
 case RT5663_HP_CALIB_ST6:
 case RT5663_HP_CALIB_ST7:
 case RT5663_HP_CALIB_ST8:
 case RT5663_HP_CALIB_ST9:
 case RT5663_ANA_JD:
  return true;
 default:
  return false;
 }
}

static bool rt5663_readable_register(struct device *dev, unsigned int reg)
{
 switch (reg) {
 case RT5663_RESET:
 case RT5663_HP_OUT_EN:
 case RT5663_HP_LCH_DRE:
 case RT5663_HP_RCH_DRE:
 case RT5663_CALIB_BST:
 case RT5663_RECMIX:
 case RT5663_SIL_DET_CTL:
 case RT5663_PWR_SAV_SILDET:
 case RT5663_SIDETONE_CTL:
 case RT5663_STO1_DAC_DIG_VOL:
 case RT5663_STO1_ADC_DIG_VOL:
 case RT5663_STO1_BOOST:
 case RT5663_HP_IMP_GAIN_1:
 case RT5663_HP_IMP_GAIN_2:
 case RT5663_STO1_ADC_MIXER:
 case RT5663_AD_DA_MIXER:
 case RT5663_STO_DAC_MIXER:
 case RT5663_DIG_SIDE_MIXER:
 case RT5663_BYPASS_STO_DAC:
 case RT5663_CALIB_REC_MIX:
 case RT5663_PWR_DIG_1:
 case RT5663_PWR_DIG_2:
 case RT5663_PWR_ANLG_1:
 case RT5663_PWR_ANLG_2:
 case RT5663_PWR_ANLG_3:
 case RT5663_PWR_MIXER:
 case RT5663_SIG_CLK_DET:
 case RT5663_PRE_DIV_GATING_1:
 case RT5663_PRE_DIV_GATING_2:
 case RT5663_I2S1_SDP:
 case RT5663_ADDA_CLK_1:
 case RT5663_ADDA_RST:
 case RT5663_FRAC_DIV_1:
 case RT5663_FRAC_DIV_2:
 case RT5663_TDM_1:
 case RT5663_TDM_2:
 case RT5663_TDM_3:
 case RT5663_TDM_4:
 case RT5663_TDM_5:
 case RT5663_GLB_CLK:
 case RT5663_PLL_1:
 case RT5663_PLL_2:
 case RT5663_ASRC_1:
 case RT5663_ASRC_2:
 case RT5663_ASRC_4:
 case RT5663_DUMMY_REG:
 case RT5663_ASRC_8:
 case RT5663_ASRC_9:
 case RT5663_ASRC_11:
 case RT5663_DEPOP_1:
 case RT5663_DEPOP_2:
 case RT5663_DEPOP_3:
 case RT5663_HP_CHARGE_PUMP_1:
 case RT5663_HP_CHARGE_PUMP_2:
 case RT5663_MICBIAS_1:
 case RT5663_RC_CLK:
 case RT5663_ASRC_11_2:
 case RT5663_DUMMY_REG_2:
 case RT5663_REC_PATH_GAIN:
 case RT5663_AUTO_1MRC_CLK:
 case RT5663_ADC_EQ_1:
 case RT5663_ADC_EQ_2:
 case RT5663_IRQ_1:
 case RT5663_IRQ_2:
 case RT5663_IRQ_3:
 case RT5663_IRQ_4:
 case RT5663_IRQ_5:
 case RT5663_INT_ST_1:
 case RT5663_INT_ST_2:
 case RT5663_GPIO_1:
 case RT5663_GPIO_2:
 case RT5663_GPIO_STA1:
 case RT5663_SIN_GEN_1:
 case RT5663_SIN_GEN_2:
 case RT5663_SIN_GEN_3:
 case RT5663_SOF_VOL_ZC1:
 case RT5663_IL_CMD_1:
 case RT5663_IL_CMD_2:
 case RT5663_IL_CMD_3:
 case RT5663_IL_CMD_4:
 case RT5663_IL_CMD_5:
 case RT5663_IL_CMD_6:
 case RT5663_IL_CMD_7:
 case RT5663_IL_CMD_8:
 case RT5663_IL_CMD_PWRSAV1:
 case RT5663_IL_CMD_PWRSAV2:
 case RT5663_EM_JACK_TYPE_1:
 case RT5663_EM_JACK_TYPE_2:
 case RT5663_EM_JACK_TYPE_3:
 case RT5663_EM_JACK_TYPE_4:
 case RT5663_EM_JACK_TYPE_5:
 case RT5663_EM_JACK_TYPE_6:
 case RT5663_STO1_HPF_ADJ1:
 case RT5663_STO1_HPF_ADJ2:
 case RT5663_FAST_OFF_MICBIAS:
 case RT5663_JD_CTRL1:
 case RT5663_JD_CTRL2:
 case RT5663_DIG_MISC:
 case RT5663_VENDOR_ID:
 case RT5663_VENDOR_ID_1:
 case RT5663_VENDOR_ID_2:
 case RT5663_DIG_VOL_ZCD:
 case RT5663_ANA_BIAS_CUR_1:
 case RT5663_ANA_BIAS_CUR_2:
 case RT5663_ANA_BIAS_CUR_3:
 case RT5663_ANA_BIAS_CUR_4:
 case RT5663_ANA_BIAS_CUR_5:
 case RT5663_ANA_BIAS_CUR_6:
 case RT5663_BIAS_CUR_5:
 case RT5663_BIAS_CUR_6:
 case RT5663_BIAS_CUR_7:
 case RT5663_BIAS_CUR_8:
 case RT5663_DACREF_LDO:
 case RT5663_DUMMY_REG_3:
 case RT5663_BIAS_CUR_9:
 case RT5663_DUMMY_REG_4:
 case RT5663_VREFADJ_OP:
 case RT5663_VREF_RECMIX:
 case RT5663_CHARGE_PUMP_1:
 case RT5663_CHARGE_PUMP_1_2:
 case RT5663_CHARGE_PUMP_1_3:
 case RT5663_CHARGE_PUMP_2:
 case RT5663_DIG_IN_PIN1:
 case RT5663_PAD_DRV_CTL:
 case RT5663_PLL_INT_REG:
 case RT5663_CHOP_DAC_L:
 case RT5663_CHOP_ADC:
 case RT5663_CALIB_ADC:
 case RT5663_CHOP_DAC_R:
 case RT5663_DUMMY_CTL_DACLR:
 case RT5663_DUMMY_REG_5:
 case RT5663_SOFT_RAMP:
 case RT5663_TEST_MODE_1:
 case RT5663_TEST_MODE_2:
 case RT5663_TEST_MODE_3:
 case RT5663_STO_DRE_1:
 case RT5663_STO_DRE_2:
 case RT5663_STO_DRE_3:
 case RT5663_STO_DRE_4:
 case RT5663_STO_DRE_5:
 case RT5663_STO_DRE_6:
 case RT5663_STO_DRE_7:
 case RT5663_STO_DRE_8:
 case RT5663_STO_DRE_9:
 case RT5663_STO_DRE_10:
 case RT5663_MIC_DECRO_1:
 case RT5663_MIC_DECRO_2:
 case RT5663_MIC_DECRO_3:
 case RT5663_MIC_DECRO_4:
 case RT5663_MIC_DECRO_5:
 case RT5663_MIC_DECRO_6:
 case RT5663_HP_DECRO_1:
 case RT5663_HP_DECRO_2:
 case RT5663_HP_DECRO_3:
 case RT5663_HP_DECRO_4:
 case RT5663_HP_DECOUP:
 case RT5663_HP_IMP_SEN_MAP8:
 case RT5663_HP_IMP_SEN_MAP9:
 case RT5663_HP_IMP_SEN_MAP10:
 case RT5663_HP_IMP_SEN_MAP11:
 case RT5663_HP_IMP_SEN_1:
 case RT5663_HP_IMP_SEN_2:
 case RT5663_HP_IMP_SEN_3:
 case RT5663_HP_IMP_SEN_4:
 case RT5663_HP_IMP_SEN_5:
 case RT5663_HP_IMP_SEN_6:
 case RT5663_HP_IMP_SEN_7:
 case RT5663_HP_IMP_SEN_8:
 case RT5663_HP_IMP_SEN_9:
 case RT5663_HP_IMP_SEN_10:
 case RT5663_HP_IMP_SEN_11:
 case RT5663_HP_IMP_SEN_12:
 case RT5663_HP_IMP_SEN_13:
 case RT5663_HP_IMP_SEN_14:
 case RT5663_HP_IMP_SEN_15:
 case RT5663_HP_IMP_SEN_16:
 case RT5663_HP_IMP_SEN_17:
 case RT5663_HP_IMP_SEN_18:
 case RT5663_HP_IMP_SEN_19:
 case RT5663_HP_IMPSEN_DIG5:
 case RT5663_HP_IMPSEN_MAP1:
 case RT5663_HP_IMPSEN_MAP2:
 case RT5663_HP_IMPSEN_MAP3:
 case RT5663_HP_IMPSEN_MAP4:
 case RT5663_HP_IMPSEN_MAP5:
 case RT5663_HP_IMPSEN_MAP7:
 case RT5663_HP_LOGIC_1:
 case RT5663_HP_LOGIC_2:
 case RT5663_HP_CALIB_1:
 case RT5663_HP_CALIB_1_1:
 case RT5663_HP_CALIB_2:
 case RT5663_HP_CALIB_3:
 case RT5663_HP_CALIB_4:
 case RT5663_HP_CALIB_5:
 case RT5663_HP_CALIB_5_1:
 case RT5663_HP_CALIB_6:
 case RT5663_HP_CALIB_7:
 case RT5663_HP_CALIB_9:
 case RT5663_HP_CALIB_10:
 case RT5663_HP_CALIB_11:
 case RT5663_HP_CALIB_ST1:
 case RT5663_HP_CALIB_ST2:
 case RT5663_HP_CALIB_ST3:
 case RT5663_HP_CALIB_ST4:
 case RT5663_HP_CALIB_ST5:
 case RT5663_HP_CALIB_ST6:
 case RT5663_HP_CALIB_ST7:
 case RT5663_HP_CALIB_ST8:
 case RT5663_HP_CALIB_ST9:
 case RT5663_HP_AMP_DET:
 case RT5663_DUMMY_REG_6:
 case RT5663_HP_BIAS:
 case RT5663_CBJ_1:
 case RT5663_CBJ_2:
 case RT5663_CBJ_3:
 case RT5663_DUMMY_1:
 case RT5663_DUMMY_2:
 case RT5663_DUMMY_3:
 case RT5663_ANA_JD:
 case RT5663_ADC_LCH_LPF1_A1:
 case RT5663_ADC_RCH_LPF1_A1:
 case RT5663_ADC_LCH_LPF1_H0:
 case RT5663_ADC_RCH_LPF1_H0:
 case RT5663_ADC_LCH_BPF1_A1:
 case RT5663_ADC_RCH_BPF1_A1:
 case RT5663_ADC_LCH_BPF1_A2:
 case RT5663_ADC_RCH_BPF1_A2:
 case RT5663_ADC_LCH_BPF1_H0:
 case RT5663_ADC_RCH_BPF1_H0:
 case RT5663_ADC_LCH_BPF2_A1:
 case RT5663_ADC_RCH_BPF2_A1:
 case RT5663_ADC_LCH_BPF2_A2:
 case RT5663_ADC_RCH_BPF2_A2:
 case RT5663_ADC_LCH_BPF2_H0:
 case RT5663_ADC_RCH_BPF2_H0:
 case RT5663_ADC_LCH_BPF3_A1:
 case RT5663_ADC_RCH_BPF3_A1:
 case RT5663_ADC_LCH_BPF3_A2:
 case RT5663_ADC_RCH_BPF3_A2:
 case RT5663_ADC_LCH_BPF3_H0:
 case RT5663_ADC_RCH_BPF3_H0:
 case RT5663_ADC_LCH_BPF4_A1:
 case RT5663_ADC_RCH_BPF4_A1:
 case RT5663_ADC_LCH_BPF4_A2:
 case RT5663_ADC_RCH_BPF4_A2:
 case RT5663_ADC_LCH_BPF4_H0:
 case RT5663_ADC_RCH_BPF4_H0:
 case RT5663_ADC_LCH_HPF1_A1:
 case RT5663_ADC_RCH_HPF1_A1:
 case RT5663_ADC_LCH_HPF1_H0:
 case RT5663_ADC_RCH_HPF1_H0:
 case RT5663_ADC_EQ_PRE_VOL_L:
 case RT5663_ADC_EQ_PRE_VOL_R:
 case RT5663_ADC_EQ_POST_VOL_L:
 case RT5663_ADC_EQ_POST_VOL_R:
  return true;
 default:
  return false;
 }
}

static bool rt5663_v2_volatile_register(struct device *dev, unsigned int reg)
{
 switch (reg) {
 case RT5663_RESET:
 case RT5663_CBJ_TYPE_2:
 case RT5663_PDM_OUT_CTL:
 case RT5663_PDM_I2C_DATA_CTL1:
 case RT5663_PDM_I2C_DATA_CTL4:
 case RT5663_ALC_BK_GAIN:
 case RT5663_PLL_2:
 case RT5663_MICBIAS_1:
 case RT5663_ADC_EQ_1:
 case RT5663_INT_ST_1:
 case RT5663_GPIO_STA2:
 case RT5663_IL_CMD_1:
 case RT5663_IL_CMD_5:
 case RT5663_A_JD_CTRL:
 case RT5663_JD_CTRL2:
 case RT5663_VENDOR_ID:
 case RT5663_VENDOR_ID_1:
 case RT5663_VENDOR_ID_2:
 case RT5663_STO_DRE_1:
 case RT5663_STO_DRE_5:
 case RT5663_STO_DRE_6:
 case RT5663_STO_DRE_7:
 case RT5663_MONO_DYNA_6:
 case RT5663_STO1_SIL_DET:
 case RT5663_MONOL_SIL_DET:
 case RT5663_MONOR_SIL_DET:
 case RT5663_STO2_DAC_SIL:
 case RT5663_MONO_AMP_CAL_ST1:
 case RT5663_MONO_AMP_CAL_ST2:
 case RT5663_MONO_AMP_CAL_ST3:
 case RT5663_MONO_AMP_CAL_ST4:
 case RT5663_HP_IMP_SEN_2:
 case RT5663_HP_IMP_SEN_3:
 case RT5663_HP_IMP_SEN_4:
 case RT5663_HP_IMP_SEN_10:
 case RT5663_HP_CALIB_1:
 case RT5663_HP_CALIB_10:
 case RT5663_HP_CALIB_ST1:
 case RT5663_HP_CALIB_ST4:
 case RT5663_HP_CALIB_ST5:
 case RT5663_HP_CALIB_ST6:
 case RT5663_HP_CALIB_ST7:
 case RT5663_HP_CALIB_ST8:
 case RT5663_HP_CALIB_ST9:
 case RT5663_HP_CALIB_ST10:
 case RT5663_HP_CALIB_ST11:
  return true;
 default:
  return false;
 }
}

static bool rt5663_v2_readable_register(struct device *dev, unsigned int reg)
{
 switch (reg) {
 case RT5663_LOUT_CTRL:
 case RT5663_HP_AMP_2:
 case RT5663_MONO_OUT:
 case RT5663_MONO_GAIN:
 case RT5663_AEC_BST:
 case RT5663_IN1_IN2:
 case RT5663_IN3_IN4:
 case RT5663_INL1_INR1:
 case RT5663_CBJ_TYPE_2:
 case RT5663_CBJ_TYPE_3:
 case RT5663_CBJ_TYPE_4:
 case RT5663_CBJ_TYPE_5:
 case RT5663_CBJ_TYPE_8:
 case RT5663_DAC3_DIG_VOL:
 case RT5663_DAC3_CTRL:
 case RT5663_MONO_ADC_DIG_VOL:
 case RT5663_STO2_ADC_DIG_VOL:
 case RT5663_MONO_ADC_BST_GAIN:
 case RT5663_STO2_ADC_BST_GAIN:
 case RT5663_SIDETONE_CTRL:
 case RT5663_MONO1_ADC_MIXER:
 case RT5663_STO2_ADC_MIXER:
 case RT5663_MONO_DAC_MIXER:
 case RT5663_DAC2_SRC_CTRL:
 case RT5663_IF_3_4_DATA_CTL:
 case RT5663_IF_5_DATA_CTL:
 case RT5663_PDM_OUT_CTL:
 case RT5663_PDM_I2C_DATA_CTL1:
 case RT5663_PDM_I2C_DATA_CTL2:
 case RT5663_PDM_I2C_DATA_CTL3:
 case RT5663_PDM_I2C_DATA_CTL4:
 case RT5663_RECMIX1_NEW:
 case RT5663_RECMIX1L_0:
 case RT5663_RECMIX1L:
 case RT5663_RECMIX1R_0:
 case RT5663_RECMIX1R:
 case RT5663_RECMIX2_NEW:
 case RT5663_RECMIX2_L_2:
 case RT5663_RECMIX2_R:
 case RT5663_RECMIX2_R_2:
 case RT5663_CALIB_REC_LR:
 case RT5663_ALC_BK_GAIN:
 case RT5663_MONOMIX_GAIN:
 case RT5663_MONOMIX_IN_GAIN:
 case RT5663_OUT_MIXL_GAIN:
 case RT5663_OUT_LMIX_IN_GAIN:
 case RT5663_OUT_RMIX_IN_GAIN:
 case RT5663_OUT_RMIX_IN_GAIN1:
 case RT5663_LOUT_MIXER_CTRL:
 case RT5663_PWR_VOL:
 case RT5663_ADCDAC_RST:
 case RT5663_I2S34_SDP:
 case RT5663_I2S5_SDP:
 case RT5663_TDM_6:
 case RT5663_TDM_7:
 case RT5663_TDM_8:
 case RT5663_TDM_9:
 case RT5663_ASRC_3:
 case RT5663_ASRC_6:
 case RT5663_ASRC_7:
 case RT5663_PLL_TRK_13:
 case RT5663_I2S_M_CLK_CTL:
 case RT5663_FDIV_I2S34_M_CLK:
 case RT5663_FDIV_I2S34_M_CLK2:
 case RT5663_FDIV_I2S5_M_CLK:
 case RT5663_FDIV_I2S5_M_CLK2:
 case RT5663_V2_IRQ_4:
 case RT5663_GPIO_3:
 case RT5663_GPIO_4:
 case RT5663_GPIO_STA2:
 case RT5663_HP_AMP_DET1:
 case RT5663_HP_AMP_DET2:
 case RT5663_HP_AMP_DET3:
 case RT5663_MID_BD_HP_AMP:
 case RT5663_LOW_BD_HP_AMP:
 case RT5663_SOF_VOL_ZC2:
 case RT5663_ADC_STO2_ADJ1:
 case RT5663_ADC_STO2_ADJ2:
 case RT5663_A_JD_CTRL:
 case RT5663_JD1_TRES_CTRL:
 case RT5663_JD2_TRES_CTRL:
 case RT5663_V2_JD_CTRL2:
 case RT5663_DUM_REG_2:
 case RT5663_DUM_REG_3:
 case RT5663_VENDOR_ID:
 case RT5663_VENDOR_ID_1:
 case RT5663_VENDOR_ID_2:
 case RT5663_DACADC_DIG_VOL2:
 case RT5663_DIG_IN_PIN2:
 case RT5663_PAD_DRV_CTL1:
 case RT5663_SOF_RAM_DEPOP:
 case RT5663_VOL_TEST:
 case RT5663_TEST_MODE_4:
 case RT5663_TEST_MODE_5:
 case RT5663_STO_DRE_9:
 case RT5663_MONO_DYNA_1:
 case RT5663_MONO_DYNA_2:
 case RT5663_MONO_DYNA_3:
 case RT5663_MONO_DYNA_4:
 case RT5663_MONO_DYNA_5:
 case RT5663_MONO_DYNA_6:
 case RT5663_STO1_SIL_DET:
 case RT5663_MONOL_SIL_DET:
 case RT5663_MONOR_SIL_DET:
 case RT5663_STO2_DAC_SIL:
 case RT5663_PWR_SAV_CTL1:
 case RT5663_PWR_SAV_CTL2:
 case RT5663_PWR_SAV_CTL3:
 case RT5663_PWR_SAV_CTL4:
 case RT5663_PWR_SAV_CTL5:
 case RT5663_PWR_SAV_CTL6:
 case RT5663_MONO_AMP_CAL1:
 case RT5663_MONO_AMP_CAL2:
 case RT5663_MONO_AMP_CAL3:
 case RT5663_MONO_AMP_CAL4:
 case RT5663_MONO_AMP_CAL5:
 case RT5663_MONO_AMP_CAL6:
 case RT5663_MONO_AMP_CAL7:
 case RT5663_MONO_AMP_CAL_ST1:
 case RT5663_MONO_AMP_CAL_ST2:
 case RT5663_MONO_AMP_CAL_ST3:
 case RT5663_MONO_AMP_CAL_ST4:
 case RT5663_MONO_AMP_CAL_ST5:
 case RT5663_V2_HP_IMP_SEN_13:
 case RT5663_V2_HP_IMP_SEN_14:
 case RT5663_V2_HP_IMP_SEN_6:
 case RT5663_V2_HP_IMP_SEN_7:
 case RT5663_V2_HP_IMP_SEN_8:
 case RT5663_V2_HP_IMP_SEN_9:
 case RT5663_V2_HP_IMP_SEN_10:
 case RT5663_HP_LOGIC_3:
 case RT5663_HP_CALIB_ST10:
 case RT5663_HP_CALIB_ST11:
 case RT5663_PRO_REG_TBL_4:
 case RT5663_PRO_REG_TBL_5:
 case RT5663_PRO_REG_TBL_6:
 case RT5663_PRO_REG_TBL_7:
 case RT5663_PRO_REG_TBL_8:
 case RT5663_PRO_REG_TBL_9:
 case RT5663_SAR_ADC_INL_1:
 case RT5663_SAR_ADC_INL_2:
 case RT5663_SAR_ADC_INL_3:
 case RT5663_SAR_ADC_INL_4:
 case RT5663_SAR_ADC_INL_5:
 case RT5663_SAR_ADC_INL_6:
 case RT5663_SAR_ADC_INL_7:
 case RT5663_SAR_ADC_INL_8:
 case RT5663_SAR_ADC_INL_9:
 case RT5663_SAR_ADC_INL_10:
 case RT5663_SAR_ADC_INL_11:
 case RT5663_SAR_ADC_INL_12:
 case RT5663_DRC_CTRL_1:
 case RT5663_DRC1_CTRL_2:
 case RT5663_DRC1_CTRL_3:
 case RT5663_DRC1_CTRL_4:
 case RT5663_DRC1_CTRL_5:
 case RT5663_DRC1_CTRL_6:
 case RT5663_DRC1_HD_CTRL_1:
 case RT5663_DRC1_HD_CTRL_2:
 case RT5663_DRC1_PRI_REG_1:
 case RT5663_DRC1_PRI_REG_2:
 case RT5663_DRC1_PRI_REG_3:
 case RT5663_DRC1_PRI_REG_4:
 case RT5663_DRC1_PRI_REG_5:
 case RT5663_DRC1_PRI_REG_6:
 case RT5663_DRC1_PRI_REG_7:
 case RT5663_DRC1_PRI_REG_8:
 case RT5663_ALC_PGA_CTL_1:
 case RT5663_ALC_PGA_CTL_2:
 case RT5663_ALC_PGA_CTL_3:
 case RT5663_ALC_PGA_CTL_4:
 case RT5663_ALC_PGA_CTL_5:
 case RT5663_ALC_PGA_CTL_6:
 case RT5663_ALC_PGA_CTL_7:
 case RT5663_ALC_PGA_CTL_8:
 case RT5663_ALC_PGA_REG_1:
 case RT5663_ALC_PGA_REG_2:
 case RT5663_ALC_PGA_REG_3:
 case RT5663_ADC_EQ_RECOV_1:
 case RT5663_ADC_EQ_RECOV_2:
 case RT5663_ADC_EQ_RECOV_3:
 case RT5663_ADC_EQ_RECOV_4:
 case RT5663_ADC_EQ_RECOV_5:
 case RT5663_ADC_EQ_RECOV_6:
 case RT5663_ADC_EQ_RECOV_7:
 case RT5663_ADC_EQ_RECOV_8:
 case RT5663_ADC_EQ_RECOV_9:
 case RT5663_ADC_EQ_RECOV_10:
 case RT5663_ADC_EQ_RECOV_11:
 case RT5663_ADC_EQ_RECOV_12:
 case RT5663_ADC_EQ_RECOV_13:
 case RT5663_VID_HIDDEN:
 case RT5663_VID_CUSTOMER:
 case RT5663_SCAN_MODE:
 case RT5663_I2C_BYPA:
  return true;
 case RT5663_TDM_1:
 case RT5663_DEPOP_3:
 case RT5663_ASRC_11_2:
 case RT5663_INT_ST_2:
 case RT5663_GPIO_STA1:
 case RT5663_SIN_GEN_1:
 case RT5663_SIN_GEN_2:
 case RT5663_SIN_GEN_3:
 case RT5663_IL_CMD_PWRSAV1:
 case RT5663_IL_CMD_PWRSAV2:
 case RT5663_EM_JACK_TYPE_1:
 case RT5663_EM_JACK_TYPE_2:
 case RT5663_EM_JACK_TYPE_3:
 case RT5663_EM_JACK_TYPE_4:
 case RT5663_FAST_OFF_MICBIAS:
 case RT5663_ANA_BIAS_CUR_1:
 case RT5663_ANA_BIAS_CUR_2:
 case RT5663_BIAS_CUR_9:
 case RT5663_DUMMY_REG_4:
 case RT5663_VREF_RECMIX:
 case RT5663_CHARGE_PUMP_1_2:
 case RT5663_CHARGE_PUMP_1_3:
 case RT5663_CHARGE_PUMP_2:
 case RT5663_CHOP_DAC_R:
 case RT5663_DUMMY_CTL_DACLR:
 case RT5663_DUMMY_REG_5:
 case RT5663_SOFT_RAMP:
 case RT5663_TEST_MODE_1:
 case RT5663_STO_DRE_10:
 case RT5663_MIC_DECRO_1:
 case RT5663_MIC_DECRO_2:
 case RT5663_MIC_DECRO_3:
 case RT5663_MIC_DECRO_4:
 case RT5663_MIC_DECRO_5:
 case RT5663_MIC_DECRO_6:
 case RT5663_HP_DECRO_1:
 case RT5663_HP_DECRO_2:
 case RT5663_HP_DECRO_3:
 case RT5663_HP_DECRO_4:
 case RT5663_HP_DECOUP:
 case RT5663_HP_IMPSEN_MAP4:
 case RT5663_HP_IMPSEN_MAP5:
 case RT5663_HP_IMPSEN_MAP7:
 case RT5663_HP_CALIB_1:
 case RT5663_CBJ_1:
 case RT5663_CBJ_2:
 case RT5663_CBJ_3:
  return false;
 default:
  return rt5663_readable_register(dev, reg);
 }
}

static const DECLARE_TLV_DB_SCALE(rt5663_hp_vol_tlv, -2400, 150, 0);
static const DECLARE_TLV_DB_SCALE(rt5663_v2_hp_vol_tlv, -2250, 150, 0);
static const DECLARE_TLV_DB_SCALE(dac_vol_tlv, -6525, 75, 0);
static const DECLARE_TLV_DB_SCALE(adc_vol_tlv, -1725, 75, 0);

/* {0, +20, +24, +30, +35, +40, +44, +50, +52} dB */
static const DECLARE_TLV_DB_RANGE(in_bst_tlv,
 0, 0, TLV_DB_SCALE_ITEM(0, 0, 0),
 1, 1, TLV_DB_SCALE_ITEM(2000, 0, 0),
 2, 2, TLV_DB_SCALE_ITEM(2400, 0, 0),
 3, 5, TLV_DB_SCALE_ITEM(3000, 500, 0),
 6, 6, TLV_DB_SCALE_ITEM(4400, 0, 0),
 7, 7, TLV_DB_SCALE_ITEM(5000, 0, 0),
 8, 8, TLV_DB_SCALE_ITEM(5200, 0, 0)
);

/* Interface data select */
static const char * const rt5663_if1_adc_data_select[] = {
 "L/R""R/L""L/L""R/R"
};

static SOC_ENUM_SINGLE_DECL(rt5663_if1_adc_enum, RT5663_TDM_2,
 RT5663_DATA_SWAP_ADCDAT1_SHIFT, rt5663_if1_adc_data_select);

static void rt5663_enable_push_button_irq(struct snd_soc_component *component,
 bool enable)
{
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);

 if (enable) {
  snd_soc_component_update_bits(component, RT5663_IL_CMD_6,
   RT5663_EN_4BTN_INL_MASK, RT5663_EN_4BTN_INL_EN);
  /* reset in-line command */
  snd_soc_component_update_bits(component, RT5663_IL_CMD_6,
   RT5663_RESET_4BTN_INL_MASK,
   RT5663_RESET_4BTN_INL_RESET);
  snd_soc_component_update_bits(component, RT5663_IL_CMD_6,
   RT5663_RESET_4BTN_INL_MASK,
   RT5663_RESET_4BTN_INL_NOR);
  switch (rt5663->codec_ver) {
  case CODEC_VER_1:
   snd_soc_component_update_bits(component, RT5663_IRQ_3,
    RT5663_V2_EN_IRQ_INLINE_MASK,
    RT5663_V2_EN_IRQ_INLINE_NOR);
   break;
  case CODEC_VER_0:
   snd_soc_component_update_bits(component, RT5663_IRQ_2,
    RT5663_EN_IRQ_INLINE_MASK,
    RT5663_EN_IRQ_INLINE_NOR);
   break;
  default:
   dev_err(component->dev, "Unknown CODEC Version\n");
  }
 } else {
  switch (rt5663->codec_ver) {
  case CODEC_VER_1:
   snd_soc_component_update_bits(component, RT5663_IRQ_3,
    RT5663_V2_EN_IRQ_INLINE_MASK,
    RT5663_V2_EN_IRQ_INLINE_BYP);
   break;
  case CODEC_VER_0:
   snd_soc_component_update_bits(component, RT5663_IRQ_2,
    RT5663_EN_IRQ_INLINE_MASK,
    RT5663_EN_IRQ_INLINE_BYP);
   break;
  default:
   dev_err(component->dev, "Unknown CODEC Version\n");
  }
  snd_soc_component_update_bits(component, RT5663_IL_CMD_6,
   RT5663_EN_4BTN_INL_MASK, RT5663_EN_4BTN_INL_DIS);
  /* reset in-line command */
  snd_soc_component_update_bits(component, RT5663_IL_CMD_6,
   RT5663_RESET_4BTN_INL_MASK,
   RT5663_RESET_4BTN_INL_RESET);
  snd_soc_component_update_bits(component, RT5663_IL_CMD_6,
   RT5663_RESET_4BTN_INL_MASK,
   RT5663_RESET_4BTN_INL_NOR);
 }
}

/**
 * rt5663_v2_jack_detect - Detect headset.
 * @component: SoC audio component device.
 * @jack_insert: Jack insert or not.
 *
 * Detect whether is headset or not when jack inserted.
 *
 * Returns detect status.
 */


static int rt5663_v2_jack_detect(struct snd_soc_component *component, int jack_insert)
{
 struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);
 int val, i = 0, sleep_time[5] = {300, 150, 100, 50, 30};

 dev_dbg(component->dev, "%s jack_insert:%d\n", __func__, jack_insert);
 if (jack_insert) {
  snd_soc_component_write(component, RT5663_CBJ_TYPE_2, 0x8040);
  snd_soc_component_write(component, RT5663_CBJ_TYPE_3, 0x1484);

  snd_soc_dapm_force_enable_pin(dapm, "MICBIAS1");
  snd_soc_dapm_force_enable_pin(dapm, "MICBIAS2");
  snd_soc_dapm_force_enable_pin(dapm, "Mic Det Power");
  snd_soc_dapm_force_enable_pin(dapm, "CBJ Power");
  snd_soc_dapm_sync(dapm);
  snd_soc_component_update_bits(component, RT5663_RC_CLK,
   RT5663_DIG_1M_CLK_MASK, RT5663_DIG_1M_CLK_EN);
  snd_soc_component_update_bits(component, RT5663_RECMIX, 0x8, 0x8);

  while (i < 5) {
   msleep(sleep_time[i]);
   val = snd_soc_component_read(component, RT5663_CBJ_TYPE_2) & 0x0003;
   if (val == 0x1 || val == 0x2 || val == 0x3)
    break;
   dev_dbg(component->dev, "%s: MX-0011 val=%x sleep %d\n",
    __func__, val, sleep_time[i]);
   i++;
  }
  dev_dbg(component->dev, "%s val = %d\n", __func__, val);
  switch (val) {
  case 1:
  case 2:
   rt5663->jack_type = SND_JACK_HEADSET;
   rt5663_enable_push_button_irq(component, true);
   break;
  default:
   snd_soc_dapm_disable_pin(dapm, "MICBIAS1");
   snd_soc_dapm_disable_pin(dapm, "MICBIAS2");
   snd_soc_dapm_disable_pin(dapm, "Mic Det Power");
   snd_soc_dapm_disable_pin(dapm, "CBJ Power");
   snd_soc_dapm_sync(dapm);
   rt5663->jack_type = SND_JACK_HEADPHONE;
   break;
  }
 } else {
  snd_soc_component_update_bits(component, RT5663_RECMIX, 0x8, 0x0);

  if (rt5663->jack_type == SND_JACK_HEADSET) {
   rt5663_enable_push_button_irq(component, false);
   snd_soc_dapm_disable_pin(dapm, "MICBIAS1");
   snd_soc_dapm_disable_pin(dapm, "MICBIAS2");
   snd_soc_dapm_disable_pin(dapm, "Mic Det Power");
   snd_soc_dapm_disable_pin(dapm, "CBJ Power");
   snd_soc_dapm_sync(dapm);
  }
  rt5663->jack_type = 0;
 }

 dev_dbg(component->dev, "jack_type = %d\n", rt5663->jack_type);
 return rt5663->jack_type;
}

/**
 * rt5663_jack_detect - Detect headset.
 * @component: SoC audio component device.
 * @jack_insert: Jack insert or not.
 *
 * Detect whether is headset or not when jack inserted.
 *
 * Returns detect status.
 */

static int rt5663_jack_detect(struct snd_soc_component *component, int jack_insert)
{
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);
 int val, i = 0;

 dev_dbg(component->dev, "%s jack_insert:%d\n", __func__, jack_insert);

 if (jack_insert) {
  snd_soc_component_update_bits(component, RT5663_DIG_MISC,
   RT5663_DIG_GATE_CTRL_MASK, RT5663_DIG_GATE_CTRL_EN);
  snd_soc_component_update_bits(component, RT5663_HP_CHARGE_PUMP_1,
   RT5663_SI_HP_MASK | RT5663_OSW_HP_L_MASK |
   RT5663_OSW_HP_R_MASK, RT5663_SI_HP_EN |
   RT5663_OSW_HP_L_DIS | RT5663_OSW_HP_R_DIS);
  snd_soc_component_update_bits(component, RT5663_DUMMY_1,
   RT5663_EMB_CLK_MASK | RT5663_HPA_CPL_BIAS_MASK |
   RT5663_HPA_CPR_BIAS_MASK, RT5663_EMB_CLK_EN |
   RT5663_HPA_CPL_BIAS_1 | RT5663_HPA_CPR_BIAS_1);
  snd_soc_component_update_bits(component, RT5663_CBJ_1,
   RT5663_INBUF_CBJ_BST1_MASK | RT5663_CBJ_SENSE_BST1_MASK,
   RT5663_INBUF_CBJ_BST1_ON | RT5663_CBJ_SENSE_BST1_L);
  snd_soc_component_update_bits(component, RT5663_IL_CMD_2,
   RT5663_PWR_MIC_DET_MASK, RT5663_PWR_MIC_DET_ON);
  /* BST1 power on for JD */
  snd_soc_component_update_bits(component, RT5663_PWR_ANLG_2,
   RT5663_PWR_BST1_MASK, RT5663_PWR_BST1_ON);
  snd_soc_component_update_bits(component, RT5663_EM_JACK_TYPE_1,
   RT5663_CBJ_DET_MASK | RT5663_EXT_JD_MASK |
   RT5663_POL_EXT_JD_MASK, RT5663_CBJ_DET_EN |
   RT5663_EXT_JD_EN | RT5663_POL_EXT_JD_EN);
  snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
   RT5663_PWR_MB_MASK | RT5663_LDO1_DVO_MASK |
   RT5663_AMP_HP_MASK, RT5663_PWR_MB |
   RT5663_LDO1_DVO_0_9V | RT5663_AMP_HP_3X);
  snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
   RT5663_PWR_VREF1_MASK | RT5663_PWR_VREF2_MASK |
   RT5663_PWR_FV1_MASK | RT5663_PWR_FV2_MASK,
   RT5663_PWR_VREF1 | RT5663_PWR_VREF2);
  msleep(20);
  snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
   RT5663_PWR_FV1_MASK | RT5663_PWR_FV2_MASK,
   RT5663_PWR_FV1 | RT5663_PWR_FV2);
  snd_soc_component_update_bits(component, RT5663_AUTO_1MRC_CLK,
   RT5663_IRQ_POW_SAV_MASK, RT5663_IRQ_POW_SAV_EN);
  snd_soc_component_update_bits(component, RT5663_IRQ_1,
   RT5663_EN_IRQ_JD1_MASK, RT5663_EN_IRQ_JD1_EN);
  snd_soc_component_update_bits(component, RT5663_EM_JACK_TYPE_1,
   RT5663_EM_JD_MASK, RT5663_EM_JD_RST);
  snd_soc_component_update_bits(component, RT5663_EM_JACK_TYPE_1,
   RT5663_EM_JD_MASK, RT5663_EM_JD_NOR);

  while (true) {
   regmap_read(rt5663->regmap, RT5663_INT_ST_2, &val);
   if (!(val & 0x80))
    usleep_range(10000, 10005);
   else
    break;

   if (i > 200)
    break;
   i++;
  }

  val = snd_soc_component_read(component, RT5663_EM_JACK_TYPE_2) & 0x0003;
  dev_dbg(component->dev, "%s val = %d\n", __func__, val);

  snd_soc_component_update_bits(component, RT5663_HP_CHARGE_PUMP_1,
   RT5663_OSW_HP_L_MASK | RT5663_OSW_HP_R_MASK,
   RT5663_OSW_HP_L_EN | RT5663_OSW_HP_R_EN);

  switch (val) {
  case 1:
  case 2:
   rt5663->jack_type = SND_JACK_HEADSET;
   rt5663_enable_push_button_irq(component, true);

   if (rt5663->pdata.impedance_sensing_num)
    break;

   if (rt5663->pdata.dc_offset_l_manual_mic) {
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_2,
     rt5663->pdata.dc_offset_l_manual_mic >>
     16);
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_3,
     rt5663->pdata.dc_offset_l_manual_mic &
     0xffff);
   }

   if (rt5663->pdata.dc_offset_r_manual_mic) {
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_5,
     rt5663->pdata.dc_offset_r_manual_mic >>
     16);
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_6,
     rt5663->pdata.dc_offset_r_manual_mic &
     0xffff);
   }
   break;
  default:
   rt5663->jack_type = SND_JACK_HEADPHONE;
   snd_soc_component_update_bits(component,
    RT5663_PWR_ANLG_1,
    RT5663_PWR_MB_MASK | RT5663_PWR_VREF1_MASK |
    RT5663_PWR_VREF2_MASK, 0);
   if (rt5663->pdata.impedance_sensing_num)
    break;

   if (rt5663->pdata.dc_offset_l_manual) {
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_2,
     rt5663->pdata.dc_offset_l_manual >> 16);
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_3,
     rt5663->pdata.dc_offset_l_manual &
     0xffff);
   }

   if (rt5663->pdata.dc_offset_r_manual) {
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_5,
     rt5663->pdata.dc_offset_r_manual >> 16);
    regmap_write(rt5663->regmap, RT5663_MIC_DECRO_6,
     rt5663->pdata.dc_offset_r_manual &
     0xffff);
   }
   break;
  }
 } else {
  if (rt5663->jack_type == SND_JACK_HEADSET)
   rt5663_enable_push_button_irq(component, false);
  rt5663->jack_type = 0;
  snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
   RT5663_PWR_MB_MASK | RT5663_PWR_VREF1_MASK |
   RT5663_PWR_VREF2_MASK, 0);
 }

 dev_dbg(component->dev, "jack_type = %d\n", rt5663->jack_type);
 return rt5663->jack_type;
}

static int rt5663_impedance_sensing(struct snd_soc_component *component)
{
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);
 unsigned int value, i, reg84, reg26, reg2fa, reg91, reg10, reg80;

 for (i = 0; i < rt5663->pdata.impedance_sensing_num; i++) {
  if (rt5663->imp_table[i].vol == 7)
   break;
 }

 if (rt5663->jack_type == SND_JACK_HEADSET) {
  snd_soc_component_write(component, RT5663_MIC_DECRO_2,
   rt5663->imp_table[i].dc_offset_l_manual_mic >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_3,
   rt5663->imp_table[i].dc_offset_l_manual_mic & 0xffff);
  snd_soc_component_write(component, RT5663_MIC_DECRO_5,
   rt5663->imp_table[i].dc_offset_r_manual_mic >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_6,
   rt5663->imp_table[i].dc_offset_r_manual_mic & 0xffff);
 } else {
  snd_soc_component_write(component, RT5663_MIC_DECRO_2,
   rt5663->imp_table[i].dc_offset_l_manual >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_3,
   rt5663->imp_table[i].dc_offset_l_manual & 0xffff);
  snd_soc_component_write(component, RT5663_MIC_DECRO_5,
   rt5663->imp_table[i].dc_offset_r_manual >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_6,
   rt5663->imp_table[i].dc_offset_r_manual & 0xffff);
 }

 reg84 = snd_soc_component_read(component, RT5663_ASRC_2);
 reg26 = snd_soc_component_read(component, RT5663_STO1_ADC_MIXER);
 reg2fa = snd_soc_component_read(component, RT5663_DUMMY_1);
 reg91 = snd_soc_component_read(component, RT5663_HP_CHARGE_PUMP_1);
 reg10 = snd_soc_component_read(component, RT5663_RECMIX);
 reg80 = snd_soc_component_read(component, RT5663_GLB_CLK);

 snd_soc_component_update_bits(component, RT5663_STO_DRE_1, 0x8000, 0);
 snd_soc_component_write(component, RT5663_ASRC_2, 0);
 snd_soc_component_write(component, RT5663_STO1_ADC_MIXER, 0x4040);
 snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
  RT5663_PWR_VREF1_MASK | RT5663_PWR_VREF2_MASK |
  RT5663_PWR_FV1_MASK | RT5663_PWR_FV2_MASK,
  RT5663_PWR_VREF1 | RT5663_PWR_VREF2);
 usleep_range(10000, 10005);
 snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
  RT5663_PWR_FV1_MASK | RT5663_PWR_FV2_MASK,
  RT5663_PWR_FV1 | RT5663_PWR_FV2);
 snd_soc_component_update_bits(component, RT5663_GLB_CLK, RT5663_SCLK_SRC_MASK,
  RT5663_SCLK_SRC_RCCLK);
 snd_soc_component_update_bits(component, RT5663_RC_CLK, RT5663_DIG_25M_CLK_MASK,
  RT5663_DIG_25M_CLK_EN);
 snd_soc_component_update_bits(component, RT5663_ADDA_CLK_1, RT5663_I2S_PD1_MASK, 0);
 snd_soc_component_write(component, RT5663_PRE_DIV_GATING_1, 0xff00);
 snd_soc_component_write(component, RT5663_PRE_DIV_GATING_2, 0xfffc);
 snd_soc_component_write(component, RT5663_HP_CHARGE_PUMP_1, 0x1232);
 snd_soc_component_write(component, RT5663_HP_LOGIC_2, 0x0005);
 snd_soc_component_write(component, RT5663_DEPOP_2, 0x3003);
 snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x0030, 0x0030);
 snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x0003, 0x0003);
 snd_soc_component_update_bits(component, RT5663_PWR_DIG_2,
  RT5663_PWR_ADC_S1F | RT5663_PWR_DAC_S1F,
  RT5663_PWR_ADC_S1F | RT5663_PWR_DAC_S1F);
 snd_soc_component_update_bits(component, RT5663_PWR_DIG_1,
  RT5663_PWR_DAC_L1 | RT5663_PWR_DAC_R1 |
  RT5663_PWR_LDO_DACREF_MASK | RT5663_PWR_ADC_L1 |
  RT5663_PWR_ADC_R1,
  RT5663_PWR_DAC_L1 | RT5663_PWR_DAC_R1 |
  RT5663_PWR_LDO_DACREF_ON | RT5663_PWR_ADC_L1 |
  RT5663_PWR_ADC_R1);
 msleep(40);
 snd_soc_component_update_bits(component, RT5663_PWR_ANLG_2,
  RT5663_PWR_RECMIX1 | RT5663_PWR_RECMIX2,
  RT5663_PWR_RECMIX1 | RT5663_PWR_RECMIX2);
 msleep(30);
 snd_soc_component_write(component, RT5663_HP_CHARGE_PUMP_2, 0x1371);
 snd_soc_component_write(component, RT5663_STO_DAC_MIXER, 0);
 snd_soc_component_write(component, RT5663_BYPASS_STO_DAC, 0x000c);
 snd_soc_component_write(component, RT5663_HP_BIAS, 0xafaa);
 snd_soc_component_write(component, RT5663_CHARGE_PUMP_1, 0x2224);
 snd_soc_component_write(component, RT5663_HP_OUT_EN, 0x8088);
 snd_soc_component_write(component, RT5663_CHOP_ADC, 0x3000);
 snd_soc_component_write(component, RT5663_ADDA_RST, 0xc000);
 snd_soc_component_write(component, RT5663_STO1_HPF_ADJ1, 0x3320);
 snd_soc_component_write(component, RT5663_HP_CALIB_2, 0x00c9);
 snd_soc_component_write(component, RT5663_DUMMY_1, 0x004c);
 snd_soc_component_write(component, RT5663_ANA_BIAS_CUR_1, 0x7733);
 snd_soc_component_write(component, RT5663_CHARGE_PUMP_2, 0x7777);
 snd_soc_component_write(component, RT5663_STO_DRE_9, 0x0007);
 snd_soc_component_write(component, RT5663_STO_DRE_10, 0x0007);
 snd_soc_component_write(component, RT5663_DUMMY_2, 0x02a4);
 snd_soc_component_write(component, RT5663_RECMIX, 0x0005);
 snd_soc_component_write(component, RT5663_HP_IMP_SEN_1, 0x4334);
 snd_soc_component_update_bits(component, RT5663_IRQ_3, 0x0004, 0x0004);
 snd_soc_component_write(component, RT5663_HP_LOGIC_1, 0x2200);
 snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x3000, 0x3000);
 snd_soc_component_write(component, RT5663_HP_LOGIC_1, 0x6200);

 for (i = 0; i < 100; i++) {
  msleep(20);
  if (snd_soc_component_read(component, RT5663_INT_ST_1) & 0x2)
   break;
 }

 value = snd_soc_component_read(component, RT5663_HP_IMP_SEN_4);

 snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x3000, 0);
 snd_soc_component_write(component, RT5663_INT_ST_1, 0);
 snd_soc_component_write(component, RT5663_HP_LOGIC_1, 0);
 snd_soc_component_update_bits(component, RT5663_RC_CLK, RT5663_DIG_25M_CLK_MASK,
  RT5663_DIG_25M_CLK_DIS);
 snd_soc_component_write(component, RT5663_GLB_CLK, reg80);
 snd_soc_component_write(component, RT5663_RECMIX, reg10);
 snd_soc_component_write(component, RT5663_DUMMY_2, 0x00a4);
 snd_soc_component_write(component, RT5663_DUMMY_1, reg2fa);
 snd_soc_component_write(component, RT5663_HP_CALIB_2, 0x00c8);
 snd_soc_component_write(component, RT5663_STO1_HPF_ADJ1, 0xb320);
 snd_soc_component_write(component, RT5663_ADDA_RST, 0xe400);
 snd_soc_component_write(component, RT5663_CHOP_ADC, 0x2000);
 snd_soc_component_write(component, RT5663_HP_OUT_EN, 0x0008);
 snd_soc_component_update_bits(component, RT5663_PWR_ANLG_2,
  RT5663_PWR_RECMIX1 | RT5663_PWR_RECMIX2, 0);
 snd_soc_component_update_bits(component, RT5663_PWR_DIG_1,
  RT5663_PWR_DAC_L1 | RT5663_PWR_DAC_R1 |
  RT5663_PWR_LDO_DACREF_MASK | RT5663_PWR_ADC_L1 |
  RT5663_PWR_ADC_R1, 0);
 snd_soc_component_update_bits(component, RT5663_PWR_DIG_2,
  RT5663_PWR_ADC_S1F | RT5663_PWR_DAC_S1F, 0);
 snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x0003, 0);
 snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x0030, 0);
 snd_soc_component_write(component, RT5663_HP_LOGIC_2, 0);
 snd_soc_component_write(component, RT5663_HP_CHARGE_PUMP_1, reg91);
 snd_soc_component_update_bits(component, RT5663_PWR_ANLG_1,
  RT5663_PWR_VREF1_MASK | RT5663_PWR_VREF2_MASK, 0);
 snd_soc_component_write(component, RT5663_STO1_ADC_MIXER, reg26);
 snd_soc_component_write(component, RT5663_ASRC_2, reg84);

 for (i = 0; i < rt5663->pdata.impedance_sensing_num; i++) {
  if (value >= rt5663->imp_table[i].imp_min &&
   value <= rt5663->imp_table[i].imp_max)
   break;
 }

 snd_soc_component_update_bits(component, RT5663_STO_DRE_9, RT5663_DRE_GAIN_HP_MASK,
  rt5663->imp_table[i].vol);
 snd_soc_component_update_bits(component, RT5663_STO_DRE_10, RT5663_DRE_GAIN_HP_MASK,
  rt5663->imp_table[i].vol);

 if (rt5663->jack_type == SND_JACK_HEADSET) {
  snd_soc_component_write(component, RT5663_MIC_DECRO_2,
   rt5663->imp_table[i].dc_offset_l_manual_mic >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_3,
   rt5663->imp_table[i].dc_offset_l_manual_mic & 0xffff);
  snd_soc_component_write(component, RT5663_MIC_DECRO_5,
   rt5663->imp_table[i].dc_offset_r_manual_mic >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_6,
   rt5663->imp_table[i].dc_offset_r_manual_mic & 0xffff);
 } else {
  snd_soc_component_write(component, RT5663_MIC_DECRO_2,
   rt5663->imp_table[i].dc_offset_l_manual >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_3,
   rt5663->imp_table[i].dc_offset_l_manual & 0xffff);
  snd_soc_component_write(component, RT5663_MIC_DECRO_5,
   rt5663->imp_table[i].dc_offset_r_manual >> 16);
  snd_soc_component_write(component, RT5663_MIC_DECRO_6,
   rt5663->imp_table[i].dc_offset_r_manual & 0xffff);
 }

 return 0;
}

static int rt5663_button_detect(struct snd_soc_component *component)
{
 int btn_type, val;

 val = snd_soc_component_read(component, RT5663_IL_CMD_5);
 dev_dbg(component->dev, "%s: val=0x%x\n", __func__, val);
 btn_type = val & 0xfff0;
 snd_soc_component_write(component, RT5663_IL_CMD_5, val);

 return btn_type;
}

static irqreturn_t rt5663_irq(int irq, void *data)
{
 struct rt5663_priv *rt5663 = data;

 dev_dbg(regmap_get_device(rt5663->regmap), "%s IRQ queue work\n",
  __func__);

 queue_delayed_work(system_wq, &rt5663->jack_detect_work,
  msecs_to_jiffies(250));

 return IRQ_HANDLED;
}

static int rt5663_set_jack_detect(struct snd_soc_component *component,
 struct snd_soc_jack *hs_jack, void *data)
{
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);

 rt5663->hs_jack = hs_jack;

 rt5663_irq(0, rt5663);

 return 0;
}

static bool rt5663_check_jd_status(struct snd_soc_component *component)
{
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);
 int val = snd_soc_component_read(component, RT5663_INT_ST_1);

 dev_dbg(component->dev, "%s val=%x\n", __func__, val);

 /* JD1 */
 switch (rt5663->codec_ver) {
 case CODEC_VER_1:
  return !(val & 0x2000);
 case CODEC_VER_0:
  return !(val & 0x1000);
 default:
  dev_err(component->dev, "Unknown CODEC Version\n");
 }

 return false;
}

static void rt5663_jack_detect_work(struct work_struct *work)
{
 struct rt5663_priv *rt5663 =
  container_of(work, struct rt5663_priv, jack_detect_work.work);
 struct snd_soc_component *component = rt5663->component;
 int btn_type, report = 0;

 if (!component)
  return;

 if (rt5663_check_jd_status(component)) {
  /* jack in */
  if (rt5663->jack_type == 0) {
   /* jack was out, report jack type */
   switch (rt5663->codec_ver) {
   case CODEC_VER_1:
    report = rt5663_v2_jack_detect(
      rt5663->component, 1);
    break;
   case CODEC_VER_0:
    report = rt5663_jack_detect(rt5663->component, 1);
    if (rt5663->pdata.impedance_sensing_num)
     rt5663_impedance_sensing(rt5663->component);
    break;
   default:
    dev_err(component->dev, "Unknown CODEC Version\n");
   }

   /* Delay the jack insert report to avoid pop noise */
   msleep(30);
  } else {
   /* jack is already in, report button event */
   report = SND_JACK_HEADSET;
   btn_type = rt5663_button_detect(rt5663->component);
   /**
 * rt5663 can report three kinds of button behavior,
 * one click, double click and hold. However,
 * currently we will report button pressed/released
 * event. So all the three button behaviors are
 * treated as button pressed.
 */

   switch (btn_type) {
   case 0x8000:
   case 0x4000:
   case 0x2000:
    report |= SND_JACK_BTN_0;
    break;
   case 0x1000:
   case 0x0800:
   case 0x0400:
    report |= SND_JACK_BTN_1;
    break;
   case 0x0200:
   case 0x0100:
   case 0x0080:
    report |= SND_JACK_BTN_2;
    break;
   case 0x0040:
   case 0x0020:
   case 0x0010:
    report |= SND_JACK_BTN_3;
    break;
   case 0x0000: /* unpressed */
    break;
   default:
    btn_type = 0;
    dev_err(rt5663->component->dev,
     "Unexpected button code 0x%04x\n",
     btn_type);
    break;
   }
   /* button release or spurious interrput*/
   if (btn_type == 0) {
    report =  rt5663->jack_type;
    cancel_delayed_work_sync(
     &rt5663->jd_unplug_work);
   } else {
    queue_delayed_work(system_wq,
     &rt5663->jd_unplug_work,
     msecs_to_jiffies(500));
   }
  }
 } else {
  /* jack out */
  switch (rt5663->codec_ver) {
  case CODEC_VER_1:
   report = rt5663_v2_jack_detect(rt5663->component, 0);
   break;
  case CODEC_VER_0:
   report = rt5663_jack_detect(rt5663->component, 0);
   break;
  default:
   dev_err(component->dev, "Unknown CODEC Version\n");
  }
 }
 dev_dbg(component->dev, "%s jack report: 0x%04x\n", __func__, report);
 snd_soc_jack_report(rt5663->hs_jack, report, SND_JACK_HEADSET |
       SND_JACK_BTN_0 | SND_JACK_BTN_1 |
       SND_JACK_BTN_2 | SND_JACK_BTN_3);
}

static void rt5663_jd_unplug_work(struct work_struct *work)
{
 struct rt5663_priv *rt5663 =
  container_of(work, struct rt5663_priv, jd_unplug_work.work);
 struct snd_soc_component *component = rt5663->component;

 if (!component)
  return;

 if (!rt5663_check_jd_status(component)) {
  /* jack out */
  switch (rt5663->codec_ver) {
  case CODEC_VER_1:
   rt5663_v2_jack_detect(rt5663->component, 0);
   break;
  case CODEC_VER_0:
   rt5663_jack_detect(rt5663->component, 0);
   break;
  default:
   dev_err(component->dev, "Unknown CODEC Version\n");
  }

  snd_soc_jack_report(rt5663->hs_jack, 0, SND_JACK_HEADSET |
        SND_JACK_BTN_0 | SND_JACK_BTN_1 |
        SND_JACK_BTN_2 | SND_JACK_BTN_3);
 } else {
  queue_delayed_work(system_wq, &rt5663->jd_unplug_work,
   msecs_to_jiffies(500));
 }
}

static const struct snd_kcontrol_new rt5663_snd_controls[] = {
 /* DAC Digital Volume */
 SOC_DOUBLE_TLV("DAC Playback Volume", RT5663_STO1_DAC_DIG_VOL,
  RT5663_DAC_L1_VOL_SHIFT + 1, RT5663_DAC_R1_VOL_SHIFT + 1,
  87, 0, dac_vol_tlv),
 /* ADC Digital Volume Control */
 SOC_DOUBLE("ADC Capture Switch", RT5663_STO1_ADC_DIG_VOL,
  RT5663_ADC_L_MUTE_SHIFT, RT5663_ADC_R_MUTE_SHIFT, 1, 1),
 SOC_DOUBLE_TLV("ADC Capture Volume", RT5663_STO1_ADC_DIG_VOL,
  RT5663_ADC_L_VOL_SHIFT + 1, RT5663_ADC_R_VOL_SHIFT + 1,
  63, 0, adc_vol_tlv),
};

static const struct snd_kcontrol_new rt5663_v2_specific_controls[] = {
 /* Headphone Output Volume */
 SOC_DOUBLE_R_TLV("Headphone Playback Volume", RT5663_HP_LCH_DRE,
  RT5663_HP_RCH_DRE, RT5663_GAIN_HP_SHIFT, 15, 1,
  rt5663_v2_hp_vol_tlv),
 /* Mic Boost Volume */
 SOC_SINGLE_TLV("IN1 Capture Volume", RT5663_AEC_BST,
  RT5663_GAIN_CBJ_SHIFT, 8, 0, in_bst_tlv),
};

static const struct snd_kcontrol_new rt5663_specific_controls[] = {
 /* Mic Boost Volume*/
 SOC_SINGLE_TLV("IN1 Capture Volume", RT5663_CBJ_2,
  RT5663_GAIN_BST1_SHIFT, 8, 0, in_bst_tlv),
 /* Data Swap for Slot0/1 in ADCDAT1 */
 SOC_ENUM("IF1 ADC Data Swap", rt5663_if1_adc_enum),
};

static const struct snd_kcontrol_new rt5663_hpvol_controls[] = {
 /* Headphone Output Volume */
 SOC_DOUBLE_R_TLV("Headphone Playback Volume", RT5663_STO_DRE_9,
  RT5663_STO_DRE_10, RT5663_DRE_GAIN_HP_SHIFT, 23, 1,
  rt5663_hp_vol_tlv),
};

static int rt5663_is_sys_clk_from_pll(struct snd_soc_dapm_widget *w,
 struct snd_soc_dapm_widget *sink)
{
 unsigned int val;
 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);

 val = snd_soc_component_read(component, RT5663_GLB_CLK);
 val &= RT5663_SCLK_SRC_MASK;
 if (val == RT5663_SCLK_SRC_PLL1)
  return 1;
 else
  return 0;
}

static int rt5663_is_using_asrc(struct snd_soc_dapm_widget *w,
 struct snd_soc_dapm_widget *sink)
{
 unsigned int reg, shift, val;
 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);

 if (rt5663->codec_ver == CODEC_VER_1) {
  switch (w->shift) {
  case RT5663_ADC_STO1_ASRC_SHIFT:
   reg = RT5663_ASRC_3;
   shift = RT5663_V2_AD_STO1_TRACK_SHIFT;
   break;
  case RT5663_DAC_STO1_ASRC_SHIFT:
   reg = RT5663_ASRC_2;
   shift = RT5663_DA_STO1_TRACK_SHIFT;
   break;
  default:
   return 0;
  }
 } else {
  switch (w->shift) {
  case RT5663_ADC_STO1_ASRC_SHIFT:
   reg = RT5663_ASRC_2;
   shift = RT5663_AD_STO1_TRACK_SHIFT;
   break;
  case RT5663_DAC_STO1_ASRC_SHIFT:
   reg = RT5663_ASRC_2;
   shift = RT5663_DA_STO1_TRACK_SHIFT;
   break;
  default:
   return 0;
  }
 }

 val = (snd_soc_component_read(component, reg) >> shift) & 0x7;

 if (val)
  return 1;

 return 0;
}

static int rt5663_i2s_use_asrc(struct snd_soc_dapm_widget *source,
 struct snd_soc_dapm_widget *sink)
{
 struct snd_soc_component *component = snd_soc_dapm_to_component(source->dapm);
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);
 int da_asrc_en, ad_asrc_en;

 da_asrc_en = (snd_soc_component_read(component, RT5663_ASRC_2) &
  RT5663_DA_STO1_TRACK_MASK) ? 1 : 0;
 switch (rt5663->codec_ver) {
 case CODEC_VER_1:
  ad_asrc_en = (snd_soc_component_read(component, RT5663_ASRC_3) &
   RT5663_V2_AD_STO1_TRACK_MASK) ? 1 : 0;
  break;
 case CODEC_VER_0:
  ad_asrc_en = (snd_soc_component_read(component, RT5663_ASRC_2) &
   RT5663_AD_STO1_TRACK_MASK) ? 1 : 0;
  break;
 default:
  dev_err(component->dev, "Unknown CODEC Version\n");
  return 1;
 }

 if (da_asrc_en || ad_asrc_en)
  if (rt5663->sysclk > rt5663->lrck * 384)
   return 1;

 dev_err(component->dev, "sysclk < 384 x fs, disable i2s asrc\n");

 return 0;
}

/**
 * rt5663_sel_asrc_clk_src - select ASRC clock source for a set of filters
 * @component: SoC audio component device.
 * @filter_mask: mask of filters.
 * @clk_src: clock source
 *
 * The ASRC function is for asynchronous MCLK and LRCK. Also, since RT5663 can
 * only support standard 32fs or 64fs i2s format, ASRC should be enabled to
 * support special i2s clock format such as Intel's 100fs(100 * sampling rate).
 * ASRC function will track i2s clock and generate a corresponding system clock
 * for codec. This function provides an API to select the clock source for a
 * set of filters specified by the mask. And the codec driver will turn on ASRC
 * for these filters if ASRC is selected as their clock source.
 */

int rt5663_sel_asrc_clk_src(struct snd_soc_component *component,
  unsigned int filter_mask, unsigned int clk_src)
{
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);
 unsigned int asrc2_mask = 0;
 unsigned int asrc2_value = 0;
 unsigned int asrc3_mask = 0;
 unsigned int asrc3_value = 0;

 switch (clk_src) {
 case RT5663_CLK_SEL_SYS:
 case RT5663_CLK_SEL_I2S1_ASRC:
  break;

 default:
  return -EINVAL;
 }

 if (filter_mask & RT5663_DA_STEREO_FILTER) {
  asrc2_mask |= RT5663_DA_STO1_TRACK_MASK;
  asrc2_value |= clk_src << RT5663_DA_STO1_TRACK_SHIFT;
 }

 if (filter_mask & RT5663_AD_STEREO_FILTER) {
  switch (rt5663->codec_ver) {
  case CODEC_VER_1:
   asrc3_mask |= RT5663_V2_AD_STO1_TRACK_MASK;
   asrc3_value |= clk_src << RT5663_V2_AD_STO1_TRACK_SHIFT;
   break;
  case CODEC_VER_0:
   asrc2_mask |= RT5663_AD_STO1_TRACK_MASK;
   asrc2_value |= clk_src << RT5663_AD_STO1_TRACK_SHIFT;
   break;
  default:
   dev_err(component->dev, "Unknown CODEC Version\n");
  }
 }

 if (asrc2_mask)
  snd_soc_component_update_bits(component, RT5663_ASRC_2, asrc2_mask,
   asrc2_value);

 if (asrc3_mask)
  snd_soc_component_update_bits(component, RT5663_ASRC_3, asrc3_mask,
   asrc3_value);

 return 0;
}
EXPORT_SYMBOL_GPL(rt5663_sel_asrc_clk_src);

/* Analog Mixer */
static const struct snd_kcontrol_new rt5663_recmix1l[] = {
 SOC_DAPM_SINGLE("BST2 Switch", RT5663_RECMIX1L,
  RT5663_RECMIX1L_BST2_SHIFT, 1, 1),
 SOC_DAPM_SINGLE("BST1 CBJ Switch", RT5663_RECMIX1L,
  RT5663_RECMIX1L_BST1_CBJ_SHIFT, 1, 1),
};

static const struct snd_kcontrol_new rt5663_recmix1r[] = {
 SOC_DAPM_SINGLE("BST2 Switch", RT5663_RECMIX1R,
  RT5663_RECMIX1R_BST2_SHIFT, 1, 1),
};

/* Digital Mixer */
static const struct snd_kcontrol_new rt5663_sto1_adc_l_mix[] = {
 SOC_DAPM_SINGLE("ADC1 Switch", RT5663_STO1_ADC_MIXER,
   RT5663_M_STO1_ADC_L1_SHIFT, 1, 1),
 SOC_DAPM_SINGLE("ADC2 Switch", RT5663_STO1_ADC_MIXER,
   RT5663_M_STO1_ADC_L2_SHIFT, 1, 1),
};

static const struct snd_kcontrol_new rt5663_sto1_adc_r_mix[] = {
 SOC_DAPM_SINGLE("ADC1 Switch", RT5663_STO1_ADC_MIXER,
   RT5663_M_STO1_ADC_R1_SHIFT, 1, 1),
 SOC_DAPM_SINGLE("ADC2 Switch", RT5663_STO1_ADC_MIXER,
   RT5663_M_STO1_ADC_R2_SHIFT, 1, 1),
};

static const struct snd_kcontrol_new rt5663_adda_l_mix[] = {
 SOC_DAPM_SINGLE("ADC L Switch", RT5663_AD_DA_MIXER,
   RT5663_M_ADCMIX_L_SHIFT, 1, 1),
 SOC_DAPM_SINGLE("DAC L Switch", RT5663_AD_DA_MIXER,
   RT5663_M_DAC1_L_SHIFT, 1, 1),
};

static const struct snd_kcontrol_new rt5663_adda_r_mix[] = {
 SOC_DAPM_SINGLE("ADC R Switch", RT5663_AD_DA_MIXER,
   RT5663_M_ADCMIX_R_SHIFT, 1, 1),
 SOC_DAPM_SINGLE("DAC R Switch", RT5663_AD_DA_MIXER,
   RT5663_M_DAC1_R_SHIFT, 1, 1),
};

static const struct snd_kcontrol_new rt5663_sto1_dac_l_mix[] = {
 SOC_DAPM_SINGLE("DAC L Switch", RT5663_STO_DAC_MIXER,
   RT5663_M_DAC_L1_STO_L_SHIFT, 1, 1),
};

static const struct snd_kcontrol_new rt5663_sto1_dac_r_mix[] = {
 SOC_DAPM_SINGLE("DAC R Switch", RT5663_STO_DAC_MIXER,
   RT5663_M_DAC_R1_STO_R_SHIFT, 1, 1),
};

/* Out Switch */
static const struct snd_kcontrol_new rt5663_hpo_switch =
 SOC_DAPM_SINGLE_AUTODISABLE("Switch", RT5663_HP_AMP_2,
  RT5663_EN_DAC_HPO_SHIFT, 1, 0);

/* Stereo ADC source */
static const char * const rt5663_sto1_adc_src[] = {
 "ADC L""ADC R"
};

static SOC_ENUM_SINGLE_DECL(rt5663_sto1_adcl_enum, RT5663_STO1_ADC_MIXER,
 RT5663_STO1_ADC_L_SRC_SHIFT, rt5663_sto1_adc_src);

static const struct snd_kcontrol_new rt5663_sto1_adcl_mux =
 SOC_DAPM_ENUM("STO1 ADC L Mux", rt5663_sto1_adcl_enum);

static SOC_ENUM_SINGLE_DECL(rt5663_sto1_adcr_enum, RT5663_STO1_ADC_MIXER,
 RT5663_STO1_ADC_R_SRC_SHIFT, rt5663_sto1_adc_src);

static const struct snd_kcontrol_new rt5663_sto1_adcr_mux =
 SOC_DAPM_ENUM("STO1 ADC R Mux", rt5663_sto1_adcr_enum);

/* RT5663: Analog DACL1 input source */
static const char * const rt5663_alg_dacl_src[] = {
 "DAC L""STO DAC MIXL"
};

static SOC_ENUM_SINGLE_DECL(rt5663_alg_dacl_enum, RT5663_BYPASS_STO_DAC,
 RT5663_DACL1_SRC_SHIFT, rt5663_alg_dacl_src);

static const struct snd_kcontrol_new rt5663_alg_dacl_mux =
 SOC_DAPM_ENUM("DAC L Mux", rt5663_alg_dacl_enum);

/* RT5663: Analog DACR1 input source */
static const char * const rt5663_alg_dacr_src[] = {
 "DAC R""STO DAC MIXR"
};

static SOC_ENUM_SINGLE_DECL(rt5663_alg_dacr_enum, RT5663_BYPASS_STO_DAC,
 RT5663_DACR1_SRC_SHIFT, rt5663_alg_dacr_src);

static const struct snd_kcontrol_new rt5663_alg_dacr_mux =
 SOC_DAPM_ENUM("DAC R Mux", rt5663_alg_dacr_enum);

static int rt5663_hp_event(struct snd_soc_dapm_widget *w,
 struct snd_kcontrol *kcontrol, int event)
{
 struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);
 struct rt5663_priv *rt5663 = snd_soc_component_get_drvdata(component);

 switch (event) {
 case SND_SOC_DAPM_POST_PMU:
  if (rt5663->codec_ver == CODEC_VER_1) {
   snd_soc_component_update_bits(component, RT5663_HP_CHARGE_PUMP_1,
    RT5663_SEL_PM_HP_SHIFT, RT5663_SEL_PM_HP_HIGH);
   snd_soc_component_update_bits(component, RT5663_HP_LOGIC_2,
    RT5663_HP_SIG_SRC1_MASK,
    RT5663_HP_SIG_SRC1_SILENCE);
  } else {
   snd_soc_component_update_bits(component,
    RT5663_DACREF_LDO, 0x3e0e, 0x3a0a);
   snd_soc_component_write(component, RT5663_DEPOP_2, 0x3003);
   snd_soc_component_update_bits(component, RT5663_HP_CHARGE_PUMP_1,
    RT5663_OVCD_HP_MASK, RT5663_OVCD_HP_DIS);
   snd_soc_component_write(component, RT5663_HP_CHARGE_PUMP_2, 0x1371);
   snd_soc_component_write(component, RT5663_HP_BIAS, 0xabba);
   snd_soc_component_write(component, RT5663_CHARGE_PUMP_1, 0x2224);
   snd_soc_component_write(component, RT5663_ANA_BIAS_CUR_1, 0x7766);
   snd_soc_component_write(component, RT5663_HP_BIAS, 0xafaa);
   snd_soc_component_write(component, RT5663_CHARGE_PUMP_2, 0x7777);
   snd_soc_component_update_bits(component, RT5663_STO_DRE_1, 0x8000,
    0x8000);
   snd_soc_component_update_bits(component, RT5663_DEPOP_1, 0x3000,
    0x3000);
   snd_soc_component_update_bits(component,
--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=92 H=98 G=94

¤ Dauer der Verarbeitung: 0.42 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.