/*++ * linux/sound/soc/codecs/vt1603.c * WonderMedia audio driver for ALSA * * Copyright c 2010 WonderMedia Technologies, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * WonderMedia Technologies, Inc. * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C --*/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "vt1603.h" #include #define VT1603_TIMER_INTERVAL 100 // ms static struct snd_soc_codec *gvt1603_codec = NULL; // struct vt1603_priv { unsigned int sysclk; struct switch_dev hp_switch; // state of headphone insert or not struct work_struct work; struct snd_soc_codec *codec; struct timer_list timer; // timer for detecting headphone struct proc_dir_entry *proc; struct notifier_block reboot_notifier; }; struct vt1603_boardinfo { int hp_level; // 0: headset level low; 1: high int ignore_hp_event; int rec_src; // 0: AMIC-IN; 1: DMIC-IN int hpd_rm; //add 2013-11-26 int hp_debounce; //add 2014-1-14 int timer_del; //2014-1-20 tvbox int out_on; //2014-4-29 av out on for tvbox noisy. hw not good! unsigned char in_record; unsigned char trigger_stat; }; static struct vt1603_boardinfo vt1603_boardinfo; extern int wmt_getsyspara(char *varname, unsigned char *varval, int *varlen); static struct work_struct vt1603_hw_mute_work; /* codec hw configuration */ static int hw_mute_flag = 0; static const u16 vt1603_regs[] = { 0x0020, 0x00D7, 0x00D7, 0x0004, /* 0-3 */ 0x0000, 0x0040, 0x0000, 0x00c0, /* 4-7 */ 0x00c0, 0x0010, 0x0001, 0x0041, /* 8-11 */ 0x0000, 0x0000, 0x000b, 0x0093, /* 12-15 */ 0x0004, 0x0016, 0x0026, 0x0060, /* 16-19 */ 0x001a, 0x0000, 0x0002, 0x0000, /* 20-23 */ 0x0000, 0x000a, 0x0000, 0x0010, /* 24-27 */ 0x0000, 0x00e2, 0x0000, 0x0000, /* 28-31 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 32-35 */ 0x0000, 0x000c, 0x0054, 0x0000, /* 36-39 */ 0x0001, 0x000c, 0x000c, 0x000c, /* 40-43 */ 0x000c, 0x000c, 0x00c0, 0x00d5, /* 44-47 */ 0x00c5, 0x0012, 0x0085, 0x002b, /* 48-51 */ 0x00cd, 0x00cd, 0x008e, 0x008d, /* 52-55 */ 0x00e0, 0x00a6, 0x00a5, 0x0050, /* 56-59 */ 0x00e9, 0x00cf, 0x0040, 0x0000, /* 60-63 */ 0x0000, 0x0008, 0x0004, 0x0000, /* 64-67 */ 0x0040, 0x0000, 0x0040, 0x0000, /* 68-71 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 72-75 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 76-79 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 80-83 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 84-87 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 88-91 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 92-95 */ 0x00c4, 0x0079, 0x000c, 0x0024, /* 96-99 */ 0x0016, 0x0016, 0x0060, 0x0002, /* 100-103 */ 0x005b, 0x0003, 0x0039, 0x0039, /* 104-107 */ 0x00fe, 0x0012, 0x0034, 0x0000, /* 108-111 */ 0x0004, 0x00f0, 0x0000, 0x0000, /* 112-115 */ 0x0000, 0x0000, 0x0003, 0x0000, /* 116-119 */ 0x0000, 0x0000, 0x0000, 0x0000, /* 120-123 */ 0x00f2, 0x0020, 0x0023, 0x006e, /* 124-127 */ 0x0019, 0x0040, 0x0000, 0x0004, /* 128-131 */ 0x000a, 0x0075, 0x0035, 0x00b0, /* 132-135 */ 0x0040, 0x0000, 0x0000, 0x0000, /* 136-139 */ 0x0088, 0x0013, 0x000c, 0x0003, /* 140-143 */ 0x0000, 0x0000, 0x0008, 0x0000, /* 144-147 */ 0x0003, 0x0020, 0x0030, 0x0018, /* 148-151 */ 0x001b, }; /*----------------------------------------------------------------------*/ /*---------------------Control Interface Functions----------------------*/ static int vt1603_write(struct snd_soc_codec *codec, unsigned int reg, unsigned int value) { int ret = vt1603_reg_write(codec->control_data, reg, value); if (ret == 0) ((u16 *)codec->reg_cache)[reg] = value; return ret; } static unsigned int vt1603_read(struct snd_soc_codec *codec, unsigned int reg) { return ((u16 *)codec->reg_cache)[reg]; } static int vt1603_hw_write(struct snd_soc_codec *codec, unsigned int reg, unsigned int value) { return vt1603_reg_write(codec->control_data, reg, value); } static unsigned int vt1603_hw_read(struct snd_soc_codec *codec, unsigned int reg) { u8 val; int ret = vt1603_reg_read(codec->control_data, reg, &val); if (ret == 0) return val; else return ((u16 *)codec->reg_cache)[reg]; } /*----------------------------------------------------------------------*/ /*----------------------IRQ handle Functions----------------------------*/ static int vt1603_hp_power_up(struct snd_soc_codec *codec) { u16 reg; /* disable mono mixing */ reg = snd_soc_read(codec, VT1603_R0d); reg &= ~BIT4; snd_soc_write(codec, VT1603_R0d, reg); /* make sure DAC R channel is enable */ reg = snd_soc_read(codec, VT1603_R62); reg |= BIT4; snd_soc_write(codec, VT1603_R62, reg); // set HP power up reg = snd_soc_read(codec, VT1603_R68); reg &= ~BIT4; snd_soc_write(codec, VT1603_R68, reg); return 0; } static int vt1603_classd_power_up(struct snd_soc_codec *codec) { u16 reg; // set class-d power up reg = snd_soc_read(codec, VT1603_R25); reg |= BIT1; snd_soc_write(codec, VT1603_R25, reg); return 0; } static int vt1603_hp_power_off(struct snd_soc_codec *codec) { u16 reg; // set HP power off reg = snd_soc_read(codec, VT1603_R68); reg |= BIT4; snd_soc_write(codec, VT1603_R68, reg); return 0; } static int vt1603_classd_power_off(struct snd_soc_codec *codec) { u16 reg; // set class-d power off reg = snd_soc_read(codec, VT1603_R25); reg &= ~BIT1; snd_soc_write(codec, VT1603_R25, reg); return 0; } static void vt1603_switch_to_hp(struct snd_soc_codec *codec) { u16 reg; vt1603_classd_power_off(codec); if (codec->card->rtd->codec_dai->playback_active == 1) vt1603_hp_power_up(codec); } static void vt1603_switch_to_classd(struct snd_soc_codec *codec) { u16 reg; vt1603_hp_power_off(codec); if (codec->card->rtd->codec_dai->playback_active == 1) vt1603_classd_power_up(codec); } static void vt1603_set_switch_state(struct snd_soc_codec *codec) { struct vt1603_priv *vt1603 = snd_soc_codec_get_drvdata(codec); unsigned int val = vt1603_hw_read(codec, VT1603_R21); if (vt1603_boardinfo.hp_level) { if (val & BIT1) { if (!vt1603_boardinfo.ignore_hp_event) switch_set_state(&vt1603->hp_switch, 0); val &= ~BIT1; snd_soc_write(codec, VT1603_R21, val); pr_info("<<hp_switch, 1); val |= BIT1; snd_soc_write(codec, VT1603_R21, val); pr_info("<<hp_switch, 1); val &= ~BIT1; snd_soc_write(codec, VT1603_R21, val); pr_info("<<hp_switch, 0); val |= BIT1; snd_soc_write(codec, VT1603_R21, val); pr_info("<<codec; unsigned int val; static unsigned int count = 0; int hp_state = 0; //switch_get_state(&priv->hp_switch); if (!vt1603_boardinfo.hpd_rm) hp_state = switch_get_state(&priv->hp_switch); unsigned int debounce = (hp_state == 0) ? 3 : 1; // sw-debounce for headset plugin if (vt1603_boardinfo.hp_debounce > 0) debounce = (hp_state == 0) ? vt1603_boardinfo.hp_debounce : 1; val = vt1603_hw_read(codec, VT1603_R51); if (val & BIT1) { val = vt1603_hw_read(codec, VT1603_R1b); val |= BIT1; snd_soc_write(codec, VT1603_R1b, val); val &= ~BIT1; snd_soc_write(codec, VT1603_R1b, val); count++; } else { count = 0; } if (count == debounce) { pr_info("vt1603: hpdetect\n"); vt1603_set_switch_state(codec); // clear headphone irq mask val = vt1603_hw_read(codec,VT1603_R1b); val |= BIT1; snd_soc_write(codec, VT1603_R1b, val); val &= ~BIT1; snd_soc_write(codec, VT1603_R1b, val); count = 0; } val = vt1603_hw_read(codec, VT1603_R52); if (val & BIT4) { pr_info("vt1603: left cld oc\n"); val = vt1603_hw_read(codec,VT1603_R1c); val |= BIT4; snd_soc_write(codec, VT1603_R1c, val); val &= ~BIT4; snd_soc_write(codec, VT1603_R1c, val); } if (val & BIT3) { pr_info("vt1603: right cld oc\n"); val = vt1603_hw_read(codec,VT1603_R1c); val |= BIT3; snd_soc_write(codec, VT1603_R1c, val); val &= ~BIT3; snd_soc_write(codec, VT1603_R1c, val); } } static void vt1603_timer_handler(unsigned long data) { struct vt1603_priv *vt1603 = (struct vt1603_priv *)data; schedule_work(&vt1603->work); mod_timer(&vt1603->timer, jiffies + msecs_to_jiffies(VT1603_TIMER_INTERVAL)); } /*----------------------------------------------------------------------*/ /*--------------------Mixer controls & dapm-----------------------------*/ static const DECLARE_TLV_DB_SCALE(digital_tlv, -9550, 50, 1); static const DECLARE_TLV_DB_SCALE(digital_clv, -4350, 50, 1); static const DECLARE_TLV_DB_SCALE(adc_boost_clv, 0, 1000, 1); static const DECLARE_TLV_DB_SCALE(dac_boost_tlv, 0, 600, 0); static const DECLARE_TLV_DB_SCALE(adc_pga_clv, -1650, 150, 1); static const DECLARE_TLV_DB_SCALE(analog_tlv, -2100, 300, 1); static const DECLARE_TLV_DB_SCALE(hp_mixer_clv, -5700, 100, 1); static const DECLARE_TLV_DB_SCALE(cld_mixer_clv, -300, 300, 1); static const char *adc_cutoff_freq_txt[] = { "Hi-fi mode", "Voice mode 1", "Voice mode 2", "Voice mode 3", }; static const struct soc_enum adc_cutoff_freq = SOC_ENUM_SINGLE(VT1603_R00, 3, 4, adc_cutoff_freq_txt); static const char *lr_txt[] = { "Left", "Right" }; static const char *clamper_limit_voltage_txt[] = { "Vih=2.40V, Vil=0.90V, P=0.63W", "Vih=2.45V, Vil=0.85V, P=0.72W", "Vih=2.50V, ViL=0.80V, P=0.81W", "Vih=2.55V, Vil=0.75V, P=0.91W", "Vih=2.60V, Vil=0.70V, P=1.02W", "Vih=2.65V, Vil=0.65V, P=1.12W", "Vih=2.70V, Vil=0.60V, P=1.24W", "Vih=2.75V, Vil=0.55V, P=1.36W", "Vih=2.80V, Vil=0.50V, P=1.49W", "Vih=2.85V, Vil=0.45V, P=1.62W", "Vih=2.90V, Vil=0.40V, P=1.76W", "Vih=2.95V, Vil=0.35V, P=1.90W", "Vih=3.00V, Vil=0.30V, P=2.05W", "Vih=3.05V, Vil=0.25V, P=2.20W", "Vih=3.10V, Vil=0.20V, P=2.37W", "Vih=3.15V, Vil=0.15V, P=2.53W", }; static const char *clamper_work_voltage_txt[] = { "Voh=3.10V, Vol=1.90V, P=0.36W", "Voh=3.25V, Vol=1.75V, P=0.56W", "Voh=3.40V, Vol=1.60V, P=0.81W", "Voh=3.55V, Vol=1.45V, P=1.10W", "Voh=3.70V, Vol=1.30V, P=1.44W", "Voh=3.85V, Vol=1.15V, P=1.82W", "Voh=4.00V, Vol=1.00V, P=2.25W", "Voh=4.15V, Vol=0.85V, P=2.72W", }; static const struct soc_enum dacl_src = SOC_ENUM_SINGLE(VT1603_R0b, 7, 2, lr_txt); static const struct soc_enum dacr_src = SOC_ENUM_SINGLE(VT1603_R0b, 6, 2, lr_txt); static const struct soc_enum adcl_src = SOC_ENUM_SINGLE(VT1603_R03, 3, 2, lr_txt); static const struct soc_enum adcr_src = SOC_ENUM_SINGLE(VT1603_R03, 2, 2, lr_txt); static const struct soc_enum clamper_limit_voltage = SOC_ENUM_SINGLE(VT1603_R87, 4, 16, clamper_limit_voltage_txt); static const struct soc_enum clamper_work_voltage = SOC_ENUM_SINGLE(VT1603_R87, 0, 8, clamper_work_voltage_txt); static const struct snd_kcontrol_new vt1603_snd_controls[] = { /* Volume */ SOC_DOUBLE_R_TLV("Digital Capture Volume", VT1603_R01, VT1603_R02, 0, 0x7f, 0, digital_clv), SOC_DOUBLE_R_TLV("Digital Playback Volume", VT1603_R07, VT1603_R08, 0, 0xc0, 0, digital_tlv), SOC_DOUBLE_R_TLV("Analog Playback Volume", VT1603_R72, VT1603_R73, 0, 0x07, 1, analog_tlv), SOC_DOUBLE_R_TLV("Input Boost Volume", VT1603_R64, VT1603_R65, 6, 4, 0, adc_boost_clv), SOC_DOUBLE_R_TLV("Input PGA Volume", VT1603_R64, VT1603_R65, 1, 0x1f, 0, adc_pga_clv), SOC_DOUBLE_TLV("Output Boost Volume", VT1603_R06, 0, 3, 7, 0, dac_boost_tlv), SOC_DOUBLE_TLV("Output CLD Volume", VT1603_R91, 5, 2, 7, 0, cld_mixer_clv), SOC_DOUBLE_R_TLV("Output HP Volume", VT1603_R6a, VT1603_R6b, 0, 0x3f, 0, hp_mixer_clv), /* Switch */ SOC_SINGLE("EQ Switch", VT1603_R28, 0, 1, 0), SOC_SINGLE("DAC Mono Mix Switch", VT1603_R0d, 4, 1, 0), SOC_SINGLE("Left DAC Switch", VT1603_R62, 5, 1, 0), SOC_SINGLE("Right DAC Switch", VT1603_R62, 4, 1, 0), SOC_ENUM("Left DAC Source", dacl_src), SOC_ENUM("Right DAC Source", dacr_src), SOC_ENUM("Digital Capture Voice Mode", adc_cutoff_freq), SOC_ENUM("Left ADC Source", adcl_src), SOC_ENUM("Right ADC Source", adcr_src), SOC_DOUBLE("Analog Playback Switch", VT1603_R71, 7, 6, 1, 1), SOC_SINGLE("Digital Playback Switch", VT1603_R0b, 0, 1, 1), SOC_DOUBLE("Digital Capture Switch", VT1603_R63, 7, 6, 1, 0), SOC_SINGLE("Headset Switch", VT1603_R68, 4, 1, 1), SOC_SINGLE("Headfree Switch", VT1603_R25, 1, 1, 0), SOC_SINGLE("Left CLD Switch", VT1603_R82, 3, 1, 1), SOC_SINGLE("Right CLD Switch", VT1603_R82, 4, 1, 1), SOC_ENUM("Limit Voltage of Power Clamper", clamper_limit_voltage), SOC_ENUM("Voltage of Clamper Work", clamper_work_voltage), }; /* DAPM Controls */ static const struct snd_kcontrol_new linmix_controls[] = { SOC_DAPM_SINGLE("Left Mic Switch", VT1603_R8e, 5, 1, 0), SOC_DAPM_SINGLE("Left Linein Switch", VT1603_R8e, 7, 1, 0), }; static const struct snd_kcontrol_new rinmix_controls[] = { SOC_DAPM_SINGLE("Right Mic Switch", VT1603_R8e, 4, 1, 0), SOC_DAPM_SINGLE("Right Linein Switch", VT1603_R8e, 6, 1, 0), }; static const struct snd_kcontrol_new loutmix_controls[] = { SOC_DAPM_SINGLE("Left Input Mixer Switch", VT1603_R71, 3, 1, 0), SOC_DAPM_SINGLE("DACL Switch", VT1603_R71, 1, 1, 0), }; static const struct snd_kcontrol_new routmix_controls[] = { SOC_DAPM_SINGLE("Right Input Mixer Switch", VT1603_R71, 2, 1, 0), SOC_DAPM_SINGLE("DACR Switch", VT1603_R71, 0, 1, 0), }; static const struct snd_kcontrol_new lcld_controls[] = { SOC_DAPM_SINGLE("Left Output Mixer Switch", VT1603_R90, 5, 1, 0), SOC_DAPM_SINGLE("DACL Switch", VT1603_R90, 4, 1, 0), }; static const struct snd_kcontrol_new rcld_controls[] = { SOC_DAPM_SINGLE("Right Output Mixer Switch", VT1603_R90, 3, 1, 0), SOC_DAPM_SINGLE("DACR Switch", VT1603_R90, 2, 1, 0), }; static const struct snd_kcontrol_new lhp_controls[] = { SOC_DAPM_SINGLE("Left Output Mixer Switch", VT1603_R69, 5, 1, 0), SOC_DAPM_SINGLE("DACL Switch", VT1603_R69, 7, 1, 0), }; static const struct snd_kcontrol_new rhp_controls[] = { SOC_DAPM_SINGLE("Right Output Mixer Switch", VT1603_R69, 2, 1, 0), SOC_DAPM_SINGLE("DACR Switch", VT1603_R69, 4, 1, 0), }; static const struct snd_soc_dapm_widget vt1603_dapm_widgets[] = { SND_SOC_DAPM_INPUT("Mic"), SND_SOC_DAPM_INPUT("Linein"), SND_SOC_DAPM_INPUT("Internal ADC Source"), SND_SOC_DAPM_PGA("Left Input Boost", VT1603_R67, 7, 0, NULL, 0), SND_SOC_DAPM_PGA("Right Input Boost", VT1603_R67, 6, 0, NULL, 0), SND_SOC_DAPM_PGA("Left Input PGA", VT1603_R67, 5, 0, NULL, 0), SND_SOC_DAPM_PGA("Right Input PGA", VT1603_R67, 4, 0, NULL, 0), SND_SOC_DAPM_MIXER("Left Input Mixer", VT1603_R66, 6, 1, linmix_controls, ARRAY_SIZE(linmix_controls)), SND_SOC_DAPM_MIXER("Right Input Mixer", VT1603_R66, 5, 1, rinmix_controls, ARRAY_SIZE(rinmix_controls)), SND_SOC_DAPM_ADC("ADCL", "Left Capture", VT1603_R01, 7, 1), SND_SOC_DAPM_ADC("ADCR", "Right Capture", VT1603_R02, 7, 1), //SND_SOC_DAPM_DAC("DACL", "Left Playback", VT1603_R62, 5, 0), //SND_SOC_DAPM_DAC("DACR", "Right Playback", VT1603_R62, 4, 0), SND_SOC_DAPM_INPUT("DACL"), SND_SOC_DAPM_INPUT("DACR"), SND_SOC_DAPM_MIXER("Left Output Mixer", VT1603_R72, 6, 0, loutmix_controls, ARRAY_SIZE(loutmix_controls)), SND_SOC_DAPM_MIXER("Right Output Mixer", VT1603_R73, 6, 0, routmix_controls, ARRAY_SIZE(routmix_controls)), SND_SOC_DAPM_MIXER("Left CLD Mixer", VT1603_R91, 1, 0, lcld_controls, ARRAY_SIZE(lcld_controls)), SND_SOC_DAPM_MIXER("Right CLD Mixer", VT1603_R91, 0, 0, rcld_controls, ARRAY_SIZE(rcld_controls)), SND_SOC_DAPM_PGA("Left CLD PGA", VT1603_R82, 3, 1, NULL, 0), SND_SOC_DAPM_PGA("Right CLD PGA", VT1603_R82, 4, 1, NULL, 0), SND_SOC_DAPM_MIXER("Left HP Mixer", VT1603_R68, 1, 1, lhp_controls, ARRAY_SIZE(lhp_controls)), SND_SOC_DAPM_MIXER("Right HP Mixer", VT1603_R68, 0, 1, rhp_controls, ARRAY_SIZE(rhp_controls)), SND_SOC_DAPM_OUTPUT("Internal DAC Sink"), SND_SOC_DAPM_OUTPUT("Left HP"), SND_SOC_DAPM_OUTPUT("Right HP"), SND_SOC_DAPM_OUTPUT("Left SPK"), SND_SOC_DAPM_OUTPUT("Right SPK"), }; static const struct snd_soc_dapm_route audio_map[] = { /* Make DACs turn on when playing even if not mixed into any outputs */ //{"Internal DAC Sink", NULL, "DACL"}, //{"Internal DAC Sink", NULL, "DACR"}, /* Make ADCs turn on when recording even if not mixed from any inputs */ //{"ADCL", NULL, "Internal ADC Source"}, //{"ADCR", NULL, "Internal ADC Source"}, /* Capture route */ { "Left Input Mixer", "Left Linein Switch", "Linein" }, { "Left Input Mixer", "Left Mic Switch", "Mic" }, { "Right Input Mixer", "Right Linein Switch", "Linein" }, { "Right Input Mixer", "Right Mic Switch", "Mic" }, { "Left Input Boost", NULL, "Left Input Mixer"}, { "Right Input Boost", NULL, "Right Input Mixer"}, { "Left Input PGA", NULL, "Left Input Boost"}, { "Right Input PGA", NULL, "Right Input Boost"}, { "ADCL", NULL, "Left Input PGA" }, { "ADCR", NULL, "Right Input PGA" }, //{ "Left Input Mixer", NULL, "Left Input PGA" }, //{ "Right Input Mixer", NULL, "Right Input PGA" }, /* Playback route */ //{ "Left Output Mixer", "Left Input Mixer Switch", "Left Input Mixer" }, { "Left Output Mixer", "Left Input Mixer Switch", "Left Input PGA"}, { "Left Output Mixer", "DACL Switch", "DACL" }, //{ "Right Output Mixer", "Right Input Mixer Switch", "Right Input Mixer" }, { "Right Output Mixer", "Right Input Mixer Switch", "Right Input PGA"}, { "Right Output Mixer", "DACR Switch", "DACR" }, { "Left CLD Mixer", "Left Output Mixer Switch", "Left Output Mixer" }, { "Left CLD Mixer", "DACL Switch", "DACL" }, { "Right CLD Mixer", "Right Output Mixer Switch", "Right Output Mixer" }, { "Right CLD Mixer", "DACR Switch", "DACR" }, { "Left HP Mixer", "Left Output Mixer Switch", "Left Output Mixer" }, { "Left HP Mixer", "DACL Switch", "DACL" }, { "Right HP Mixer", "Right Output Mixer Switch", "Right Output Mixer" }, { "Right HP Mixer", "DACR Switch", "DACR" }, { "Left CLD PGA", NULL, "Left CLD Mixer" }, { "Right CLD PGA", NULL, "Right CLD Mixer" }, { "Left HP", NULL, "Left HP Mixer" }, { "Right HP", NULL, "Right HP Mixer" }, { "Left SPK", NULL, "Left CLD PGA" }, { "Right SPK", NULL, "Right CLD PGA" }, }; /*----------------------------------------------------------------------*/ /*----------------------codec dai functions-----------------------------*/ struct _coeff_div { u32 mclk; u32 rate; u16 fs; u8 sr; u8 bclk_div; }; /* codec hifi mclk clock divider coefficients */ static const struct _coeff_div coeff_div[] = { /* 8k */ {12288000, 8000, 1536, 0x4, 0x0}, /* 11.025k */ {11289600, 11025, 1024, 0x8, 0x0}, /* 16k */ {12288000, 16000, 768, 0x5, 0x0}, /* 22.05k */ {11289600, 22050, 512, 0x9, 0x0}, /* 32k */ {12288000, 32000, 384, 0x7, 0x0}, /* 44.1k */ {11289600, 44100, 256, 0x6, 0x07}, /* 48k */ {12288000, 48000, 256, 0x0, 0x07}, /* 96k */ {12288000, 96000, 128, 0x1, 0x04}, }; static inline int get_coeff(int mclk, int rate) { int i; for (i = 0; i < ARRAY_SIZE(coeff_div); i++) { if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk) return i; } return -EINVAL; } static int vt1603_set_dai_sysclk(struct snd_soc_dai *codec_dai, int clk_id, unsigned int freq, int dir) { struct snd_soc_codec *codec = codec_dai->codec; struct vt1603_priv *vt1603 = snd_soc_codec_get_drvdata(codec); switch (freq) { case 11289600: case 12000000: case 12288000: case 16934400: case 18432000: vt1603->sysclk = freq; return 0; } return -EINVAL; } static int vt1603_set_dai_fmt(struct snd_soc_dai *codec_dai, unsigned int fmt) { struct snd_soc_codec *codec = codec_dai->codec; u16 iface = snd_soc_read(codec, VT1603_R19); /* set master/slave audio interface */ switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { case SND_SOC_DAIFMT_CBM_CFM: iface |= BIT5 ; break; case SND_SOC_DAIFMT_CBS_CFS: break; default: return -EINVAL; } /* interface format */ switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { case SND_SOC_DAIFMT_I2S: iface |= 0x0002; break; case SND_SOC_DAIFMT_RIGHT_J: break; case SND_SOC_DAIFMT_LEFT_J: iface |= 0x0001; break; case SND_SOC_DAIFMT_DSP_A: iface |= 0x0003; break; case SND_SOC_DAIFMT_DSP_B: iface |= 0x0013; break; default: return -EINVAL; } /* clock inversion */ switch (fmt & SND_SOC_DAIFMT_INV_MASK) { case SND_SOC_DAIFMT_NB_NF: break; case SND_SOC_DAIFMT_IB_IF: iface |= 0x0090; break; case SND_SOC_DAIFMT_IB_NF: iface |= 0x0080; break; case SND_SOC_DAIFMT_NB_IF: iface |= 0x0010; break; default: return -EINVAL; } snd_soc_write(codec, VT1603_R19, iface); return 0; } static int vt1603_pcm_hw_params(struct snd_pcm_substream *substream, struct snd_pcm_hw_params *params, struct snd_soc_dai *dai) { struct snd_soc_pcm_runtime *rtd = substream->private_data; struct snd_soc_codec *codec = rtd->codec; struct vt1603_priv *vt1603 = snd_soc_codec_get_drvdata(codec); u16 val; u16 iface = snd_soc_read(codec, VT1603_R19) & 0x73; int coeff = get_coeff(vt1603->sysclk, params_rate(params)); /* bit size */ switch (params_format(params)) { case SNDRV_PCM_FORMAT_S16_LE: break; case SNDRV_PCM_FORMAT_S20_3LE: iface |= 0x04; break; case SNDRV_PCM_FORMAT_S24_LE: iface |= 0x08; break; case SNDRV_PCM_FORMAT_S32_LE: iface |= 0x0c; break; } /* set iface & srate */ snd_soc_write(codec, VT1603_R19, iface); if (coeff >= 0) { val = snd_soc_read(codec, VT1603_R42); val &= 0xf0; val |= coeff_div[coeff].bclk_div; snd_soc_write(codec, VT1603_R42, val); if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { val = snd_soc_read(codec, VT1603_R05); val &= 0xf0; val |= coeff_div[coeff].sr; snd_soc_write(codec, VT1603_R05, val); } if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) { val = snd_soc_read(codec, VT1603_R03); val &= 0x0f; val |= ((coeff_div[coeff].sr<<4) & 0xf0); snd_soc_write(codec, VT1603_R03, val); } } return 0; } static int vt1603_set_bias_level(struct snd_soc_codec *codec, enum snd_soc_bias_level level) { u16 val; switch (level) { case SND_SOC_BIAS_ON: break; case SND_SOC_BIAS_PREPARE: val = snd_soc_read(codec, VT1603_R96); val |= BIT4+BIT5; snd_soc_write(codec, VT1603_R96, val); break; case SND_SOC_BIAS_STANDBY: break; case SND_SOC_BIAS_OFF: break; } codec->dapm.bias_level = level; return 0; } static int vt1603_mute(struct snd_soc_dai *dai, int mute) { struct snd_soc_codec *codec = dai->codec; u16 reg = snd_soc_read(codec, VT1603_R0b); if (mute) reg |= BIT0; else reg &= ~BIT0; snd_soc_write(codec, VT1603_R0b, reg); return 0; } /* * The defferent between vt1603_hw_mute and vt1603_mute is that * vt1603_mute only do software mute, while vt1603_hw_mute * let L/R channel power down. */ static int vt1603_hw_mute(struct snd_soc_codec *codec, int mute) { u16 reg; /* If current output mixer connect to AA, means that it's in incall mode, * don't power down class-d channels. */ /*reg = snd_soc_read(codec, VT1603_R71); if (reg & BIT3 || reg & BIT2) return 0;*/ reg = snd_soc_read(codec, VT1603_R82); if (mute) reg |= BIT4+BIT3; else reg &= ~(BIT4+BIT3); snd_soc_write(codec, VT1603_R82, reg); return 0; } static void vt1603_do_hw_mute_work(struct work_struct *work ) { struct snd_soc_codec *codec = gvt1603_codec; u16 reg; vt1603_hw_mute(codec, hw_mute_flag); if (vt1603_boardinfo.in_record) if (vt1603_boardinfo.trigger_stat == SNDRV_PCM_TRIGGER_START) { reg = snd_soc_read(codec, VT1603_R63); reg |= (BIT6 | BIT7); snd_soc_write(codec, VT1603_R63, reg); } else if (vt1603_boardinfo.trigger_stat == SNDRV_PCM_TRIGGER_STOP) { reg = snd_soc_read(codec, VT1603_R63); reg &= ~(BIT6 | BIT7); snd_soc_write(codec, VT1603_R63, reg); vt1603_boardinfo.in_record = 0; } } /* * vt1603_trigger to disable/enable channels output while pcm stream is * pause/playing. Avoid sharp noise while pcm stream is not playing. * Note that don't add printk that will cause a loud POP. */ static int vt1603_trigger(struct snd_pcm_substream *substream, int cmd, struct snd_soc_dai *codec_dai) { switch (cmd) { case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_RESUME: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) hw_mute_flag = 0; if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) { vt1603_boardinfo.in_record = 1; vt1603_boardinfo.trigger_stat = SNDRV_PCM_TRIGGER_START; } break; case SNDRV_PCM_TRIGGER_STOP: case SNDRV_PCM_TRIGGER_SUSPEND: case SNDRV_PCM_TRIGGER_PAUSE_PUSH: if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) hw_mute_flag = 1; if (substream->stream == SNDRV_PCM_STREAM_CAPTURE) vt1603_boardinfo.trigger_stat = SNDRV_PCM_TRIGGER_STOP; break; } schedule_work(&vt1603_hw_mute_work); return 0; } static struct snd_soc_dai_ops vt1603_dai_ops = { .hw_params = vt1603_pcm_hw_params, .digital_mute = vt1603_mute, .trigger = vt1603_trigger, .set_fmt = vt1603_set_dai_fmt, .set_sysclk = vt1603_set_dai_sysclk, }; struct snd_soc_dai_driver vt1603_dai = { .name = "VT1603", .playback = { .stream_name = "Playback", .channels_min = 1, .channels_max = 2, .rates = SNDRV_PCM_RATE_8000_96000, .formats = SNDRV_PCM_FMTBIT_S16_LE, }, .capture = { .stream_name = "Capture", .channels_min = 1, .channels_max = 2, .rates = SNDRV_PCM_RATE_8000_96000, .formats = SNDRV_PCM_FMTBIT_S16_LE, }, .ops = &vt1603_dai_ops, }; /*----------------------------------------------------------------------*/ /*----------------------------------------------------------------------*/ static void vt1603_get_boardinfo(void) { int ret = 0; unsigned char buf[64]; int varlen = sizeof(buf); memset(buf, 0x0, sizeof(buf)); ret = wmt_getsyspara("wmt.audio.i2s", buf, &varlen); if (ret == 0) { sscanf(buf, "vt1603:%d:%d:%d", &vt1603_boardinfo.hp_level, &vt1603_boardinfo.ignore_hp_event, &vt1603_boardinfo.rec_src); } else { vt1603_boardinfo.hp_level = 0; vt1603_boardinfo.ignore_hp_event = 0; vt1603_boardinfo.rec_src = 0; } memset(buf, 0x0, sizeof(buf)); vt1603_boardinfo.hpd_rm = 0; ret = wmt_getsyspara("wmt.vt1603.hpd.rm", buf, &varlen); if (ret == 0) { sscanf(buf, "%d", &vt1603_boardinfo.hpd_rm); } else { vt1603_boardinfo.hpd_rm = 0; } printk("<<<<%s hpd_rm %d\n", __func__, vt1603_boardinfo.hpd_rm); memset(buf, 0x0, sizeof(buf)); vt1603_boardinfo.hp_debounce = 0; ret = wmt_getsyspara("wmt.vt1603.hp.debounce", buf, &varlen); if (ret == 0) { sscanf(buf, "%d", &vt1603_boardinfo.hp_debounce); } else { vt1603_boardinfo.hp_debounce = 0; } printk("<<<<%s hp_debounce %d\n", __func__, vt1603_boardinfo.hp_debounce); memset(buf, 0x0, sizeof(buf)); vt1603_boardinfo.timer_del = 0; ret = wmt_getsyspara("wmt.vt1603.timer.del", buf, &varlen); if (ret == 0) { sscanf(buf, "%d", &vt1603_boardinfo.timer_del); } else { vt1603_boardinfo.timer_del = 0; } printk("<<<<%s timer_del %d\n", __func__, vt1603_boardinfo.timer_del); memset(buf, 0x0, sizeof(buf)); vt1603_boardinfo.out_on = 0; ret = wmt_getsyspara("wmt.vt1603.out_on", buf, &varlen); if (ret == 0) { sscanf(buf, "%d", &vt1603_boardinfo.out_on); } else { vt1603_boardinfo.out_on = 0; } printk("<<<<%s out_on %d\n", __func__, vt1603_boardinfo.out_on); } static int vt1603_readproc(char *page, char **start, off_t off, int count, int *eof, void *data) { struct vt1603_priv *vt1603 = data; pr_info("\nvt1603 regs dump:\n"); vt1603_regs_dump(vt1603->codec->control_data); return 0; } static void vt1603_codec_reset(struct snd_soc_codec *codec) { unsigned int val; vt1603_hw_write(codec, 0xc2, 0x01); // reset all registers to default settings vt1603_hw_write(codec, VT1603_REG_RESET, 0xff); vt1603_hw_write(codec, VT1603_REG_RESET, 0x00); val = vt1603_hw_read(codec, VT1603_R60); val &= ~(BIT6 + BIT7); vt1603_hw_write(codec, VT1603_R60, val); val |= BIT6 + BIT7; vt1603_hw_write(codec, VT1603_R60, val); } static void vt1603_codec_init(struct snd_soc_codec *codec) { unsigned int val; // vt1603 touch uses irq low active??? if (vt1603_boardinfo.hp_level) { //hp high snd_soc_write(codec, VT1603_R21, 0xfd); } else { //hp low snd_soc_write(codec, VT1603_R21, 0xff); } // vt1603 codec irq settings snd_soc_write(codec, VT1603_R1b, 0xff); snd_soc_write(codec, VT1603_R1b, 0xfd); snd_soc_write(codec, VT1603_R1c, 0xff); snd_soc_write(codec, VT1603_R1d, 0xff); snd_soc_write(codec, VT1603_R24, 0x04); snd_soc_write(codec, VT1603_R95, 0x00); snd_soc_write(codec, VT1603_R60, 0xc7); snd_soc_write(codec, VT1603_R93, 0x02); snd_soc_write(codec, VT1603_R7b, 0x10); snd_soc_write(codec, VT1603_R92, 0x2c); // r4 is reserved, I don't know more about it. snd_soc_write(codec, VT1603_R04, 0x00); // r9[5] is reserved, I don't know more about it. val = snd_soc_read(codec, VT1603_R09); val |= BIT5; snd_soc_write(codec, VT1603_R09, val); // r79 is reserved, I don't know more about it. val = snd_soc_read(codec, VT1603_R79); val |= BIT2; val &= ~(BIT0+BIT1); snd_soc_write(codec, VT1603_R79, val); // r7a is reserved, I don't know more about it. val = snd_soc_read(codec, VT1603_R7a); val |= BIT3+BIT4+BIT7; snd_soc_write(codec, VT1603_R7a, val); // r87 is reserved, I don't know more about it. snd_soc_write(codec, VT1603_R87, 0x90); // r88 is reserved, I don't know more about it. snd_soc_write(codec, VT1603_R88, 0x28); // r8a is reserved, I don't know more about it. val = snd_soc_read(codec, VT1603_R8a); val |= BIT1; snd_soc_write(codec, VT1603_R8a, val); // enable CLK_SYS, CLK_DSP, CLK_AIF val = snd_soc_read(codec, VT1603_R40); val |= BIT4+BIT5+BIT6; snd_soc_write(codec, VT1603_R40, val); // set DAC sample rate divier=CLK_SYS/1, DAC clock divider=DAC_CLK_512X val = snd_soc_read(codec, VT1603_R41); val &= ~(BIT3+BIT2+BIT1+BIT0); val |= BIT1;//add for mclk 24.576M, 10Khz 17kHz sine distortion 2014-6-12 snd_soc_write(codec, VT1603_R41, val); // set BCLK rate=CLK_SYS/8 val = snd_soc_read(codec, VT1603_R42); val |= BIT0+BIT1+BIT2; val &= ~BIT3; snd_soc_write(codec, VT1603_R42, val); // enable VREF_SC_DA buffer val = snd_soc_read(codec, VT1603_R61); val |= BIT7; snd_soc_write(codec, VT1603_R61, val); // ADC VREF_SC offset voltage snd_soc_write(codec, VT1603_R93, 0x20); // set mic bias voltage = 90% * AVDD val = snd_soc_read(codec, VT1603_R92); val |= BIT2; snd_soc_write(codec, VT1603_R92, val); // enable bypass for AOW0 parameterized HPF val = snd_soc_read(codec, VT1603_R0a); val |= BIT6; snd_soc_write(codec, VT1603_R0a, val); // CPVEE=2.2uF, the ripple frequency ON CPVEE above 22kHz, that will reduce HP noise val = snd_soc_read(codec, VT1603_R7a); val |= BIT6; val &= ~BIT5; snd_soc_write(codec, VT1603_R7a, val); // set DAC soft mute mode val = snd_soc_read(codec, VT1603_R0b); val |= BIT1+BIT2; snd_soc_write(codec, VT1603_R0b, val); // DAC High Voltage Switch Control Signal enable??? val = snd_soc_read(codec, VT1603_R62); val |= BIT7+BIT6; val &= ~BIT3; snd_soc_write(codec, VT1603_R62, val); // enable microphone bias val = snd_soc_read(codec, VT1603_R60); val |= 0x0f; snd_soc_write(codec, VT1603_R60, val); // capture pga settings?? val = snd_soc_read(codec, VT1603_R8e); val |= 0x0f; snd_soc_write(codec, VT1603_R8e, val); // capture pga settings?? val = snd_soc_read(codec, VT1603_R66); val |= BIT1+BIT2+BIT3+BIT4; // snd_soc_write(codec, VT1603_R66, val); // PGA Zero Cross Detector Enable, Change gain on zero cross only val = snd_soc_read(codec, VT1603_R64); val |= BIT0; snd_soc_write(codec, VT1603_R64, val); val = snd_soc_read(codec, VT1603_R65); val |= BIT0; snd_soc_write(codec, VT1603_R65, val); // enable hp output mode val = snd_soc_read(codec, VT1603_R68); val |= BIT2; snd_soc_write(codec, VT1603_R68, val); // enable class-d, unmute channels snd_soc_write(codec, VT1603_R7c, 0xe0); // set DAC gain update val = snd_soc_read(codec, VT1603_R05); val |= BIT6+BIT7; snd_soc_write(codec, VT1603_R05, val); // set class-d AC boost gain=1.6 snd_soc_write(codec, VT1603_R97, 0x1c); val = snd_soc_read(codec, VT1603_R97); val = (val & (~0x07)) + 0x04; snd_soc_write(codec, VT1603_R97, val); // set ADC gain update, enable ADCDAT output val = snd_soc_read(codec, VT1603_R00); val |= BIT5+BIT6+BIT7; snd_soc_write(codec, VT1603_R00, val); if (vt1603_boardinfo.out_on) { val = snd_soc_read(codec, VT1603_R68); val &= ~(1<<4); snd_soc_write(codec, VT1603_R68, val); val = snd_soc_read(codec, VT1603_R69); val |= BIT2 +BIT5; snd_soc_write(codec, VT1603_R69, val); val = snd_soc_read(codec, VT1603_R25); val |= BIT1; snd_soc_write(codec, VT1603_R25, val); val = snd_soc_read(codec, VT1603_R90); val |= BIT5 +BIT3; snd_soc_write(codec, VT1603_R90, val); } // DA DRC settings //snd_soc_write(codec, VT1603_R0e, 0x0b); //snd_soc_write(codec, VT1603_R0f, 0x94); //snd_soc_write(codec, VT1603_R10, 0x02); //snd_soc_write(codec, VT1603_R11, 0x00); //snd_soc_write(codec, VT1603_R12, 0x60); } static int vt1603_reboot_notify(struct notifier_block *this, unsigned long code, void *unused) { unsigned int val; struct vt1603_priv *priv = container_of(this, struct vt1603_priv, reboot_notifier); struct snd_soc_codec *codec = priv->codec; printk("vt1603_reboot_notify [%lu]\n", code); // power down headphone val = snd_soc_read(codec, VT1603_R68); val |= BIT4; snd_soc_write(codec, VT1603_R68, val); // power down class-d val = snd_soc_read(codec, VT1603_R25); val &= ~BIT1; snd_soc_write(codec, VT1603_R25, val); return NOTIFY_DONE; } /*----------------------------------------------------------------------*/ /*---------------------codec driver functions---------------------------*/ static int vt1603_codec_suspend(struct snd_soc_codec *codec) { struct vt1603_priv *vt1603 = snd_soc_codec_get_drvdata(codec); if (!vt1603_boardinfo.timer_del) { //add 2014-1-22 tvbox del_timer_sync(&vt1603->timer); } return 0; } static int vt1603_codec_resume(struct snd_soc_codec *codec) { struct vt1603_priv *vt1603 = snd_soc_codec_get_drvdata(codec); int reg; u16 *cache = codec->reg_cache; unsigned int val; int state = 0;//switch_get_state(&vt1603->hp_switch); if (!vt1603_boardinfo.hpd_rm) state = switch_get_state(&vt1603->hp_switch); // reset codec vt1603_codec_reset(codec); // restore codec registers for (reg = 0; reg < ARRAY_SIZE(vt1603_regs); reg++) { if (reg == VT1603_REG_RESET) continue; snd_soc_write(codec, reg, cache[reg]); } // restore headphone irq status val = vt1603_hw_read(codec, VT1603_R21); if ((vt1603_boardinfo.hp_level && state) || (!vt1603_boardinfo.hp_level && !state)) { val |= BIT1; snd_soc_write(codec, VT1603_R21, val); } else { val &= ~BIT1; snd_soc_write(codec, VT1603_R21, val); } // open headphone detect val = vt1603_hw_read(codec,VT1603_R1b); val |= BIT1; snd_soc_write(codec, VT1603_R1b, val); val &= ~BIT1; snd_soc_write(codec, VT1603_R1b, val); // clear touch interrupt status??? why here??? snd_soc_write(codec, 0xca, 0x0f); if (!vt1603_boardinfo.timer_del) { //add 2014-1-22 tvbox mod_timer(&vt1603->timer, jiffies + msecs_to_jiffies(50)); } return 0; } static int vt1603_codec_probe(struct snd_soc_codec *codec) { struct vt1603_priv *vt1603; int ret; vt1603 = kzalloc(sizeof(struct vt1603_priv), GFP_KERNEL); if (vt1603 == NULL) return -ENOMEM; gvt1603_codec = codec;//add 2013-9-2 snd_soc_codec_set_drvdata(codec, vt1603); codec->control_data = dev_get_platdata(codec->dev); vt1603->codec = codec; // need close to use wmt-switch.c 2013-11-27 if (!vt1603_boardinfo.hpd_rm) { vt1603->hp_switch.name = "h2w"; switch_dev_register(&vt1603->hp_switch); } //need close vt1603_codec_reset(codec); vt1603_codec_init(codec); vt1603_set_bias_level(codec, SND_SOC_BIAS_PREPARE); INIT_WORK(&vt1603_hw_mute_work, vt1603_do_hw_mute_work); INIT_WORK(&vt1603->work, vt1603_codec_irq_handle); if (!vt1603_boardinfo.timer_del) { //add 2014-1-20 tvbox init_timer(&vt1603->timer); vt1603->timer.data = (unsigned long)vt1603; vt1603->timer.function = vt1603_timer_handler; mod_timer(&vt1603->timer, jiffies + msecs_to_jiffies(2000)); } vt1603->proc = create_proc_read_entry("vt1603_dumpregs", 0666, NULL, vt1603_readproc, vt1603); // register reboot notifier event vt1603->reboot_notifier.notifier_call = vt1603_reboot_notify; ret = register_reboot_notifier(&vt1603->reboot_notifier); if (ret != 0) pr_err("cannot register reboot notifier (err=%d)\n", ret); return 0; } static int vt1603_codec_remove(struct snd_soc_codec *codec) { struct vt1603_priv *vt1603 = snd_soc_codec_get_drvdata(codec); remove_proc_entry("vt1603_dumpregs", NULL); if (!vt1603_boardinfo.hpd_rm) { switch_dev_unregister(&vt1603->hp_switch); } del_timer_sync(&vt1603->timer); vt1603_set_bias_level(codec, SND_SOC_BIAS_OFF); kfree(vt1603); return 0; } static struct snd_soc_codec_driver soc_codec_dev_vt1603 = { .probe = vt1603_codec_probe, .remove = vt1603_codec_remove, .suspend = vt1603_codec_suspend, .resume = vt1603_codec_resume, .read = vt1603_read, .write = vt1603_write, .set_bias_level = vt1603_set_bias_level, .reg_cache_size = ARRAY_SIZE(vt1603_regs), .reg_word_size = sizeof(u16), .reg_cache_default = vt1603_regs, .controls = vt1603_snd_controls, .num_controls = ARRAY_SIZE(vt1603_snd_controls), .dapm_widgets = vt1603_dapm_widgets, .num_dapm_widgets = ARRAY_SIZE(vt1603_dapm_widgets), .dapm_routes = audio_map, .num_dapm_routes = ARRAY_SIZE(audio_map), }; static int __devinit vt1603_probe(struct platform_device *pdev) { pr_info("<<<<%s \n", __FUNCTION__); return snd_soc_register_codec(&pdev->dev, &soc_codec_dev_vt1603, &vt1603_dai, 1); } static int __devexit vt1603_remove(struct platform_device *pdev) { snd_soc_unregister_codec(&pdev->dev); return 0; } static struct platform_driver vt1603_codec_driver = { .driver = { .name = "vt1603-codec", .owner = THIS_MODULE, }, .probe = vt1603_probe, .remove = __devexit_p(vt1603_remove), }; static __init int vt1603_init(void) { vt1603_get_boardinfo(); return platform_driver_register(&vt1603_codec_driver); } module_init(vt1603_init); static __exit void vt1603_exit(void) { platform_driver_unregister(&vt1603_codec_driver); } module_exit(vt1603_exit); int vt1603_hwdep_ioctl(u8 rw_flag, u16 offset, u16 value) { struct snd_soc_codec *codec = gvt1603_codec; if (!codec){ printk("%s NULL!\n", __FUNCTION__); return -1; } //info("rw_flag=%d, offset=0x%x, value=0x%x", rw_flag, offset, value); if (rw_flag) { if ((offset >= VT1603_R00) && (offset <= VT1603_R97)) { snd_soc_write(codec, offset, value); } else { printk("write to offset 0x%x is not allowed", offset); } return 0; } else { return snd_soc_read(codec, offset); } } EXPORT_SYMBOL(vt1603_hwdep_ioctl); MODULE_DESCRIPTION("WMT [ALSA SoC/codec] driver"); MODULE_AUTHOR("WonderMedia Technologies, Inc."); MODULE_LICENSE("GPL"); MODULE_ALIAS("platform:vt1603-codec");