diff --git a/[refs] b/[refs] index ad7b807e372f..85f7cf70e176 100644 --- a/[refs] +++ b/[refs] @@ -1,2 +1,2 @@ --- -refs/heads/master: 13b137ef0367237909bb2dc38babfb8305154676 +refs/heads/master: 55309216baeb9d7f951520cf8e8bf2337cd17bad diff --git a/trunk/MAINTAINERS b/trunk/MAINTAINERS index 7bcc3951189f..29801f760b6f 100644 --- a/trunk/MAINTAINERS +++ b/trunk/MAINTAINERS @@ -533,8 +533,6 @@ L: device-drivers-devel@blackfin.uclinux.org L: alsa-devel@alsa-project.org (moderated for non-subscribers) W: http://wiki.analog.com/ S: Supported -F: sound/soc/codecs/adau* -F: sound/soc/codecs/adav* F: sound/soc/codecs/ad1* F: sound/soc/codecs/ssm* diff --git a/trunk/include/linux/pci_ids.h b/trunk/include/linux/pci_ids.h index a311008af5e1..f23f8bf02b04 100644 --- a/trunk/include/linux/pci_ids.h +++ b/trunk/include/linux/pci_ids.h @@ -1308,6 +1308,7 @@ #define PCI_SUBDEVICE_ID_CREATIVE_SB08801 0x0041 #define PCI_SUBDEVICE_ID_CREATIVE_SB08802 0x0042 #define PCI_SUBDEVICE_ID_CREATIVE_SB08803 0x0043 +#define PCI_SUBDEVICE_ID_CREATIVE_SB1270 0x0062 #define PCI_SUBDEVICE_ID_CREATIVE_HENDRIX 0x6000 #define PCI_VENDOR_ID_ECTIVA 0x1102 /* duplicate: CREATIVE */ diff --git a/trunk/include/sound/sb16_csp.h b/trunk/include/sound/sb16_csp.h index af1b49e982df..736eac71d053 100644 --- a/trunk/include/sound/sb16_csp.h +++ b/trunk/include/sound/sb16_csp.h @@ -99,14 +99,7 @@ struct snd_sb_csp_info { /* get CSP information */ #define SNDRV_SB_CSP_IOCTL_INFO _IOR('H', 0x10, struct snd_sb_csp_info) /* load microcode to CSP */ -/* NOTE: struct snd_sb_csp_microcode overflows the max size (13 bits) - * defined for some architectures like MIPS, and it leads to build errors. - * (x86 and co have 14-bit size, thus it's valid, though.) - * As a workaround for skipping the size-limit check, here we don't use the - * normal _IOW() macro but _IOC() with the manual argument. - */ -#define SNDRV_SB_CSP_IOCTL_LOAD_CODE \ - _IOC(_IOC_WRITE, 'H', 0x11, sizeof(struct snd_sb_csp_microcode)) +#define SNDRV_SB_CSP_IOCTL_LOAD_CODE _IOW('H', 0x11, struct snd_sb_csp_microcode) /* unload microcode from CSP */ #define SNDRV_SB_CSP_IOCTL_UNLOAD_CODE _IO('H', 0x12) /* start CSP */ diff --git a/trunk/include/sound/soc-dai.h b/trunk/include/sound/soc-dai.h index 5ad5f3a50c68..1bafe95dcf41 100644 --- a/trunk/include/sound/soc-dai.h +++ b/trunk/include/sound/soc-dai.h @@ -209,10 +209,6 @@ struct snd_soc_dai_driver { struct snd_soc_pcm_stream capture; struct snd_soc_pcm_stream playback; unsigned int symmetric_rates:1; - - /* probe ordering - for components with runtime dependencies */ - int probe_order; - int remove_order; }; /* diff --git a/trunk/include/sound/soc-dapm.h b/trunk/include/sound/soc-dapm.h index e09505c5a490..c46e7d89561d 100644 --- a/trunk/include/sound/soc-dapm.h +++ b/trunk/include/sound/soc-dapm.h @@ -348,8 +348,6 @@ int snd_soc_dapm_new_widgets(struct snd_soc_dapm_context *dapm); void snd_soc_dapm_free(struct snd_soc_dapm_context *dapm); int snd_soc_dapm_add_routes(struct snd_soc_dapm_context *dapm, const struct snd_soc_dapm_route *route, int num); -int snd_soc_dapm_weak_routes(struct snd_soc_dapm_context *dapm, - const struct snd_soc_dapm_route *route, int num); /* dapm events */ int snd_soc_dapm_stream_event(struct snd_soc_pcm_runtime *rtd, @@ -431,7 +429,6 @@ struct snd_soc_dapm_path { /* status */ u32 connect:1; /* source and sink widgets are connected */ u32 walked:1; /* path has been walked */ - u32 weak:1; /* path ignored for power management */ int (*connected)(struct snd_soc_dapm_widget *source, struct snd_soc_dapm_widget *sink); @@ -447,7 +444,6 @@ struct snd_soc_dapm_widget { char *name; /* widget name */ char *sname; /* stream name */ struct snd_soc_codec *codec; - struct snd_soc_platform *platform; struct list_head list; struct snd_soc_dapm_context *dapm; @@ -511,11 +507,10 @@ struct snd_soc_dapm_context { struct device *dev; /* from parent - for debug */ struct snd_soc_codec *codec; /* parent codec */ - struct snd_soc_platform *platform; /* parent platform */ struct snd_soc_card *card; /* parent card */ /* used during DAPM updates */ - enum snd_soc_bias_level target_bias_level; + int dev_power; struct list_head list; #ifdef CONFIG_DEBUG_FS diff --git a/trunk/include/sound/soc.h b/trunk/include/sound/soc.h index aa19f5a32ba8..f1de3e0c75bc 100644 --- a/trunk/include/sound/soc.h +++ b/trunk/include/sound/soc.h @@ -202,16 +202,6 @@ #define SOC_VALUE_ENUM_SINGLE_DECL(name, xreg, xshift, xmask, xtexts, xvalues) \ SOC_VALUE_ENUM_DOUBLE_DECL(name, xreg, xshift, xshift, xmask, xtexts, xvalues) -/* - * Component probe and remove ordering levels for components with runtime - * dependencies. - */ -#define SND_SOC_COMP_ORDER_FIRST -2 -#define SND_SOC_COMP_ORDER_EARLY -1 -#define SND_SOC_COMP_ORDER_NORMAL 0 -#define SND_SOC_COMP_ORDER_LATE 1 -#define SND_SOC_COMP_ORDER_LAST 2 - /* * Bias levels * @@ -224,10 +214,10 @@ * @OFF: Power Off. No restrictions on transition times. */ enum snd_soc_bias_level { - SND_SOC_BIAS_OFF = 0, - SND_SOC_BIAS_STANDBY = 1, - SND_SOC_BIAS_PREPARE = 2, - SND_SOC_BIAS_ON = 3, + SND_SOC_BIAS_OFF, + SND_SOC_BIAS_STANDBY, + SND_SOC_BIAS_PREPARE, + SND_SOC_BIAS_ON, }; struct snd_jack; @@ -258,7 +248,8 @@ typedef int (*hw_write_t)(void *,const char* ,int); extern struct snd_ac97_bus_ops soc_ac97_ops; enum snd_soc_control_type { - SND_SOC_I2C = 1, + SND_SOC_CUSTOM = 1, + SND_SOC_I2C, SND_SOC_SPI, }; @@ -268,11 +259,6 @@ enum snd_soc_compress_type { SND_SOC_RBTREE_COMPRESSION }; -enum snd_soc_pcm_subclass { - SND_SOC_PCM_CLASS_PCM = 0, - SND_SOC_PCM_CLASS_BE = 1, -}; - int snd_soc_codec_set_sysclk(struct snd_soc_codec *codec, int clk_id, unsigned int freq, int dir); int snd_soc_codec_set_pll(struct snd_soc_codec *codec, int pll_id, int source, @@ -312,10 +298,6 @@ int snd_soc_default_readable_register(struct snd_soc_codec *codec, unsigned int reg); int snd_soc_default_writable_register(struct snd_soc_codec *codec, unsigned int reg); -int snd_soc_platform_read(struct snd_soc_platform *platform, - unsigned int reg); -int snd_soc_platform_write(struct snd_soc_platform *platform, - unsigned int reg, unsigned int val); /* Utility functions to get clock rates from various things */ int snd_soc_calc_frame_size(int sample_size, int channels, int tdm_slots); @@ -368,8 +350,6 @@ struct snd_kcontrol *snd_soc_cnew(const struct snd_kcontrol_new *_template, const char *prefix); int snd_soc_add_controls(struct snd_soc_codec *codec, const struct snd_kcontrol_new *controls, int num_controls); -int snd_soc_add_platform_controls(struct snd_soc_platform *platform, - const struct snd_kcontrol_new *controls, int num_controls); int snd_soc_info_enum_double(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo); int snd_soc_info_enum_ext(struct snd_kcontrol *kcontrol, @@ -633,10 +613,6 @@ struct snd_soc_codec_driver { void (*seq_notifier)(struct snd_soc_dapm_context *, enum snd_soc_dapm_type, int); - - /* probe ordering - for components with runtime dependencies */ - int probe_order; - int remove_order; }; /* SoC platform interface */ @@ -648,17 +624,10 @@ struct snd_soc_platform_driver { int (*resume)(struct snd_soc_dai *dai); /* pcm creation and destruction */ - int (*pcm_new)(struct snd_soc_pcm_runtime *); + int (*pcm_new)(struct snd_card *, struct snd_soc_dai *, + struct snd_pcm *); void (*pcm_free)(struct snd_pcm *); - /* Default control and setup, added after probe() is run */ - const struct snd_kcontrol_new *controls; - int num_controls; - const struct snd_soc_dapm_widget *dapm_widgets; - int num_dapm_widgets; - const struct snd_soc_dapm_route *dapm_routes; - int num_dapm_routes; - /* * For platform caused delay reporting. * Optional. @@ -668,14 +637,6 @@ struct snd_soc_platform_driver { /* platform stream ops */ struct snd_pcm_ops *ops; - - /* probe ordering - for components with runtime dependencies */ - int probe_order; - int remove_order; - - /* platform IO - used for platform DAPM */ - unsigned int (*read)(struct snd_soc_platform *, unsigned int); - int (*write)(struct snd_soc_platform *, unsigned int, unsigned int); }; struct snd_soc_platform { @@ -690,8 +651,6 @@ struct snd_soc_platform { struct snd_soc_card *card; struct list_head list; struct list_head card_list; - - struct snd_soc_dapm_context dapm; }; struct snd_soc_dai_link { @@ -767,10 +726,8 @@ struct snd_soc_card { /* callbacks */ int (*set_bias_level)(struct snd_soc_card *, - struct snd_soc_dapm_context *dapm, enum snd_soc_bias_level level); int (*set_bias_level_post)(struct snd_soc_card *, - struct snd_soc_dapm_context *dapm, enum snd_soc_bias_level level); long pmdown_time; @@ -833,9 +790,6 @@ struct snd_soc_pcm_runtime { struct device dev; struct snd_soc_card *card; struct snd_soc_dai_link *dai_link; - struct mutex pcm_mutex; - enum snd_soc_pcm_subclass pcm_subclass; - struct snd_pcm_ops ops; unsigned int complete:1; unsigned int dev_registered:1; diff --git a/trunk/include/trace/events/asoc.h b/trunk/include/trace/events/asoc.h index 603f5a0f0365..ae973d2e27a1 100644 --- a/trunk/include/trace/events/asoc.h +++ b/trunk/include/trace/events/asoc.h @@ -9,7 +9,6 @@ struct snd_soc_jack; struct snd_soc_codec; -struct snd_soc_platform; struct snd_soc_card; struct snd_soc_dapm_widget; @@ -60,50 +59,6 @@ DEFINE_EVENT(snd_soc_reg, snd_soc_reg_read, ); -DECLARE_EVENT_CLASS(snd_soc_preg, - - TP_PROTO(struct snd_soc_platform *platform, unsigned int reg, - unsigned int val), - - TP_ARGS(platform, reg, val), - - TP_STRUCT__entry( - __string( name, platform->name ) - __field( int, id ) - __field( unsigned int, reg ) - __field( unsigned int, val ) - ), - - TP_fast_assign( - __assign_str(name, platform->name); - __entry->id = platform->id; - __entry->reg = reg; - __entry->val = val; - ), - - TP_printk("platform=%s.%d reg=%x val=%x", __get_str(name), - (int)__entry->id, (unsigned int)__entry->reg, - (unsigned int)__entry->val) -); - -DEFINE_EVENT(snd_soc_preg, snd_soc_preg_write, - - TP_PROTO(struct snd_soc_platform *platform, unsigned int reg, - unsigned int val), - - TP_ARGS(platform, reg, val) - -); - -DEFINE_EVENT(snd_soc_preg, snd_soc_preg_read, - - TP_PROTO(struct snd_soc_platform *platform, unsigned int reg, - unsigned int val), - - TP_ARGS(platform, reg, val) - -); - DECLARE_EVENT_CLASS(snd_soc_card, TP_PROTO(struct snd_soc_card *card, int val), diff --git a/trunk/sound/atmel/abdac.c b/trunk/sound/atmel/abdac.c index bfee60c4d4c0..6e2409181895 100644 --- a/trunk/sound/atmel/abdac.c +++ b/trunk/sound/atmel/abdac.c @@ -599,4 +599,4 @@ module_exit(atmel_abdac_exit); MODULE_LICENSE("GPL"); MODULE_DESCRIPTION("Driver for Atmel Audio Bitstream DAC (ABDAC)"); -MODULE_AUTHOR("Hans-Christian Egtvedt "); +MODULE_AUTHOR("Hans-Christian Egtvedt "); diff --git a/trunk/sound/atmel/ac97c.c b/trunk/sound/atmel/ac97c.c index ac35222ad0dd..b310702c646e 100644 --- a/trunk/sound/atmel/ac97c.c +++ b/trunk/sound/atmel/ac97c.c @@ -1199,4 +1199,4 @@ module_exit(atmel_ac97c_exit); MODULE_LICENSE("GPL"); MODULE_DESCRIPTION("Driver for Atmel AC97 controller"); -MODULE_AUTHOR("Hans-Christian Egtvedt "); +MODULE_AUTHOR("Hans-Christian Egtvedt "); diff --git a/trunk/sound/firewire/isight.c b/trunk/sound/firewire/isight.c index 440030818db7..86ee16ca365e 100644 --- a/trunk/sound/firewire/isight.c +++ b/trunk/sound/firewire/isight.c @@ -209,7 +209,6 @@ static void isight_packet(struct fw_iso_context *context, u32 cycle, isight->packet_index = -1; return; } - fw_iso_context_queue_flush(isight->context); if (++index >= QUEUE_LENGTH) index = 0; diff --git a/trunk/sound/pci/asihpi/asihpi.c b/trunk/sound/pci/asihpi/asihpi.c index e3569bdd3b64..2ca6f4f85b41 100644 --- a/trunk/sound/pci/asihpi/asihpi.c +++ b/trunk/sound/pci/asihpi/asihpi.c @@ -27,6 +27,7 @@ #include "hpioctl.h" #include +#include #include #include #include diff --git a/trunk/sound/pci/cs5535audio/cs5535audio_pcm.c b/trunk/sound/pci/cs5535audio/cs5535audio_pcm.c index e083122ca55a..f16bc8aad6ed 100644 --- a/trunk/sound/pci/cs5535audio/cs5535audio_pcm.c +++ b/trunk/sound/pci/cs5535audio/cs5535audio_pcm.c @@ -149,7 +149,7 @@ static int cs5535audio_build_dma_packets(struct cs5535audio *cs5535au, &((struct cs5535audio_dma_desc *) dma->desc_buf.area)[i]; desc->addr = cpu_to_le32(addr); desc->size = cpu_to_le32(period_bytes); - desc->ctlreserved = cpu_to_le16(PRD_EOP); + desc->ctlreserved = cpu_to_le32(PRD_EOP); desc_addr += sizeof(struct cs5535audio_dma_desc); addr += period_bytes; } @@ -157,7 +157,7 @@ static int cs5535audio_build_dma_packets(struct cs5535audio *cs5535au, lastdesc = &((struct cs5535audio_dma_desc *) dma->desc_buf.area)[periods]; lastdesc->addr = cpu_to_le32((u32) dma->desc_buf.addr); lastdesc->size = 0; - lastdesc->ctlreserved = cpu_to_le16(PRD_JMP); + lastdesc->ctlreserved = cpu_to_le32(PRD_JMP); jmpprd_addr = cpu_to_le32(lastdesc->addr + (sizeof(struct cs5535audio_dma_desc)*periods)); diff --git a/trunk/sound/pci/ctxfi/ct20k2reg.h b/trunk/sound/pci/ctxfi/ct20k2reg.h index e0394e3996e8..ca501ba03d64 100644 --- a/trunk/sound/pci/ctxfi/ct20k2reg.h +++ b/trunk/sound/pci/ctxfi/ct20k2reg.h @@ -55,6 +55,7 @@ /* GPIO Registers */ #define GPIO_DATA 0x1B7020 #define GPIO_CTRL 0x1B7024 +#define GPIO_EXT_DATA 0x1B70A0 /* Virtual memory registers */ #define VMEM_PTPAL 0x1C6300 /* 0x1C6300 + (16 * Chn) */ diff --git a/trunk/sound/pci/ctxfi/ctatc.c b/trunk/sound/pci/ctxfi/ctatc.c index 13f33c0719d3..952fd94c2666 100644 --- a/trunk/sound/pci/ctxfi/ctatc.c +++ b/trunk/sound/pci/ctxfi/ctatc.c @@ -30,7 +30,6 @@ #include #define MONO_SUM_SCALE 0x19a8 /* 2^(-0.5) in 14-bit floating format */ -#define DAIONUM 7 #define MAX_MULTI_CHN 8 #define IEC958_DEFAULT_CON ((IEC958_AES0_NONAUDIO \ @@ -53,6 +52,8 @@ static struct snd_pci_quirk __devinitdata subsys_20k1_list[] = { static struct snd_pci_quirk __devinitdata subsys_20k2_list[] = { SND_PCI_QUIRK(PCI_VENDOR_ID_CREATIVE, PCI_SUBDEVICE_ID_CREATIVE_SB0760, "SB0760", CTSB0760), + SND_PCI_QUIRK(PCI_VENDOR_ID_CREATIVE, PCI_SUBDEVICE_ID_CREATIVE_SB1270, + "SB1270", CTSB1270), SND_PCI_QUIRK(PCI_VENDOR_ID_CREATIVE, PCI_SUBDEVICE_ID_CREATIVE_SB08801, "SB0880", CTSB0880), SND_PCI_QUIRK(PCI_VENDOR_ID_CREATIVE, PCI_SUBDEVICE_ID_CREATIVE_SB08802, @@ -75,6 +76,7 @@ static const char *ct_subsys_name[NUM_CTCARDS] = { [CTSB0760] = "SB076x", [CTHENDRIX] = "Hendrix", [CTSB0880] = "SB0880", + [CTSB1270] = "SB1270", [CT20K2_UNKNOWN] = "Unknown", }; @@ -459,12 +461,12 @@ static void setup_src_node_conf(struct ct_atc *atc, struct ct_atc_pcm *apcm, apcm->substream->runtime->rate); *n_srcc = 0; - if (1 == atc->msr) { + if (1 == atc->msr) { /* FIXME: do we really need SRC here if pitch==1 */ *n_srcc = apcm->substream->runtime->channels; conf[0].pitch = pitch; conf[0].mix_msr = conf[0].imp_msr = conf[0].msr = 1; conf[0].vo = 1; - } else if (2 == atc->msr) { + } else if (2 <= atc->msr) { if (0x8000000 < pitch) { /* Need two-stage SRCs, SRCIMPs and * AMIXERs for converting format */ @@ -977,6 +979,55 @@ static int atc_have_digit_io_switch(struct ct_atc *atc) return hw->have_digit_io_switch(hw); } +static int atc_have_dedicated_mic(struct ct_atc *atc) +{ + struct hw *hw = atc->hw; + + return hw->have_dedicated_mic(hw); +} + +static int atc_have_output_switch(struct ct_atc *atc) +{ + struct hw *hw = atc->hw; + + return hw->have_output_switch(hw); +} + +static int atc_output_switch_get(struct ct_atc *atc) +{ + struct hw *hw = atc->hw; + + return hw->output_switch_get(hw); +} + +static int atc_output_switch_put(struct ct_atc *atc, int position) +{ + struct hw *hw = atc->hw; + + return hw->output_switch_put(hw, position); +} + +static int atc_have_mic_source_switch(struct ct_atc *atc) +{ + struct hw *hw = atc->hw; + + return hw->have_mic_source_switch(hw); +} + +static int atc_mic_source_switch_get(struct ct_atc *atc) +{ + struct hw *hw = atc->hw; + + return hw->mic_source_switch_get(hw); +} + +static int atc_mic_source_switch_put(struct ct_atc *atc, int position) +{ + struct hw *hw = atc->hw; + + return hw->mic_source_switch_put(hw, position); +} + static int atc_select_digit_io(struct ct_atc *atc) { struct hw *hw = atc->hw; @@ -1045,6 +1096,11 @@ static int atc_line_in_unmute(struct ct_atc *atc, unsigned char state) return atc_daio_unmute(atc, state, LINEIM); } +static int atc_mic_unmute(struct ct_atc *atc, unsigned char state) +{ + return atc_daio_unmute(atc, state, MIC); +} + static int atc_spdif_out_unmute(struct ct_atc *atc, unsigned char state) { return atc_daio_unmute(atc, state, SPDIFOO); @@ -1331,17 +1387,20 @@ static int atc_get_resources(struct ct_atc *atc) struct srcimp_mgr *srcimp_mgr; struct sum_desc sum_dsc = {0}; struct sum_mgr *sum_mgr; - int err, i; + int err, i, num_srcs, num_daios; + + num_daios = ((atc->model == CTSB1270) ? 8 : 7); + num_srcs = ((atc->model == CTSB1270) ? 6 : 4); - atc->daios = kzalloc(sizeof(void *)*(DAIONUM), GFP_KERNEL); + atc->daios = kzalloc(sizeof(void *)*num_daios, GFP_KERNEL); if (!atc->daios) return -ENOMEM; - atc->srcs = kzalloc(sizeof(void *)*(2*2), GFP_KERNEL); + atc->srcs = kzalloc(sizeof(void *)*num_srcs, GFP_KERNEL); if (!atc->srcs) return -ENOMEM; - atc->srcimps = kzalloc(sizeof(void *)*(2*2), GFP_KERNEL); + atc->srcimps = kzalloc(sizeof(void *)*num_srcs, GFP_KERNEL); if (!atc->srcimps) return -ENOMEM; @@ -1351,8 +1410,9 @@ static int atc_get_resources(struct ct_atc *atc) daio_mgr = (struct daio_mgr *)atc->rsc_mgrs[DAIO]; da_desc.msr = atc->msr; - for (i = 0, atc->n_daio = 0; i < DAIONUM-1; i++) { - da_desc.type = i; + for (i = 0, atc->n_daio = 0; i < num_daios; i++) { + da_desc.type = (atc->model != CTSB073X) ? i : + ((i == SPDIFIO) ? SPDIFI1 : i); err = daio_mgr->get_daio(daio_mgr, &da_desc, (struct daio **)&atc->daios[i]); if (err) { @@ -1362,23 +1422,12 @@ static int atc_get_resources(struct ct_atc *atc) } atc->n_daio++; } - if (atc->model == CTSB073X) - da_desc.type = SPDIFI1; - else - da_desc.type = SPDIFIO; - err = daio_mgr->get_daio(daio_mgr, &da_desc, - (struct daio **)&atc->daios[i]); - if (err) { - printk(KERN_ERR "ctxfi: Failed to get S/PDIF-in resource!!!\n"); - return err; - } - atc->n_daio++; src_mgr = atc->rsc_mgrs[SRC]; src_dsc.multi = 1; src_dsc.msr = atc->msr; src_dsc.mode = ARCRW; - for (i = 0, atc->n_src = 0; i < (2*2); i++) { + for (i = 0, atc->n_src = 0; i < num_srcs; i++) { err = src_mgr->get_src(src_mgr, &src_dsc, (struct src **)&atc->srcs[i]); if (err) @@ -1388,8 +1437,8 @@ static int atc_get_resources(struct ct_atc *atc) } srcimp_mgr = atc->rsc_mgrs[SRCIMP]; - srcimp_dsc.msr = 8; /* SRCIMPs for S/PDIFIn SRT */ - for (i = 0, atc->n_srcimp = 0; i < (2*1); i++) { + srcimp_dsc.msr = 8; + for (i = 0, atc->n_srcimp = 0; i < num_srcs; i++) { err = srcimp_mgr->get_srcimp(srcimp_mgr, &srcimp_dsc, (struct srcimp **)&atc->srcimps[i]); if (err) @@ -1397,15 +1446,6 @@ static int atc_get_resources(struct ct_atc *atc) atc->n_srcimp++; } - srcimp_dsc.msr = 8; /* SRCIMPs for LINE/MICIn SRT */ - for (i = 0; i < (2*1); i++) { - err = srcimp_mgr->get_srcimp(srcimp_mgr, &srcimp_dsc, - (struct srcimp **)&atc->srcimps[2*1+i]); - if (err) - return err; - - atc->n_srcimp++; - } sum_mgr = atc->rsc_mgrs[SUM]; sum_dsc.msr = atc->msr; @@ -1488,6 +1528,18 @@ static void atc_connect_resources(struct ct_atc *atc) src = atc->srcs[3]; mixer->set_input_right(mixer, MIX_LINE_IN, &src->rsc); + if (atc->model == CTSB1270) { + /* Titanium HD has a dedicated ADC for the Mic. */ + dai = container_of(atc->daios[MIC], struct dai, daio); + atc_connect_dai(atc->rsc_mgrs[SRC], dai, + (struct src **)&atc->srcs[4], + (struct srcimp **)&atc->srcimps[4]); + src = atc->srcs[4]; + mixer->set_input_left(mixer, MIX_MIC_IN, &src->rsc); + src = atc->srcs[5]; + mixer->set_input_right(mixer, MIX_MIC_IN, &src->rsc); + } + dai = container_of(atc->daios[SPDIFIO], struct dai, daio); atc_connect_dai(atc->rsc_mgrs[SRC], dai, (struct src **)&atc->srcs[0], @@ -1606,12 +1658,20 @@ static struct ct_atc atc_preset __devinitdata = { .line_clfe_unmute = atc_line_clfe_unmute, .line_rear_unmute = atc_line_rear_unmute, .line_in_unmute = atc_line_in_unmute, + .mic_unmute = atc_mic_unmute, .spdif_out_unmute = atc_spdif_out_unmute, .spdif_in_unmute = atc_spdif_in_unmute, .spdif_out_get_status = atc_spdif_out_get_status, .spdif_out_set_status = atc_spdif_out_set_status, .spdif_out_passthru = atc_spdif_out_passthru, .have_digit_io_switch = atc_have_digit_io_switch, + .have_dedicated_mic = atc_have_dedicated_mic, + .have_output_switch = atc_have_output_switch, + .output_switch_get = atc_output_switch_get, + .output_switch_put = atc_output_switch_put, + .have_mic_source_switch = atc_have_mic_source_switch, + .mic_source_switch_get = atc_mic_source_switch_get, + .mic_source_switch_put = atc_mic_source_switch_put, #ifdef CONFIG_PM .suspend = atc_suspend, .resume = atc_resume, diff --git a/trunk/sound/pci/ctxfi/ctatc.h b/trunk/sound/pci/ctxfi/ctatc.h index 7167c0185d52..6bad27e06f4d 100644 --- a/trunk/sound/pci/ctxfi/ctatc.h +++ b/trunk/sound/pci/ctxfi/ctatc.h @@ -115,12 +115,20 @@ struct ct_atc { int (*line_clfe_unmute)(struct ct_atc *atc, unsigned char state); int (*line_rear_unmute)(struct ct_atc *atc, unsigned char state); int (*line_in_unmute)(struct ct_atc *atc, unsigned char state); + int (*mic_unmute)(struct ct_atc *atc, unsigned char state); int (*spdif_out_unmute)(struct ct_atc *atc, unsigned char state); int (*spdif_in_unmute)(struct ct_atc *atc, unsigned char state); int (*spdif_out_get_status)(struct ct_atc *atc, unsigned int *status); int (*spdif_out_set_status)(struct ct_atc *atc, unsigned int status); int (*spdif_out_passthru)(struct ct_atc *atc, unsigned char state); int (*have_digit_io_switch)(struct ct_atc *atc); + int (*have_dedicated_mic)(struct ct_atc *atc); + int (*have_output_switch)(struct ct_atc *atc); + int (*output_switch_get)(struct ct_atc *atc); + int (*output_switch_put)(struct ct_atc *atc, int position); + int (*have_mic_source_switch)(struct ct_atc *atc); + int (*mic_source_switch_get)(struct ct_atc *atc); + int (*mic_source_switch_put)(struct ct_atc *atc, int position); /* Don't touch! Used for internal object. */ void *rsc_mgrs[NUM_RSCTYP]; /* chip resource managers */ diff --git a/trunk/sound/pci/ctxfi/ctdaio.c b/trunk/sound/pci/ctxfi/ctdaio.c index 47d9ea97de02..0c00eb4088ef 100644 --- a/trunk/sound/pci/ctxfi/ctdaio.c +++ b/trunk/sound/pci/ctxfi/ctdaio.c @@ -22,20 +22,9 @@ #include #include -#define DAIO_RESOURCE_NUM NUM_DAIOTYP #define DAIO_OUT_MAX SPDIFOO -union daio_usage { - struct { - unsigned short lineo1:1; - unsigned short lineo2:1; - unsigned short lineo3:1; - unsigned short lineo4:1; - unsigned short spdifoo:1; - unsigned short lineim:1; - unsigned short spdifio:1; - unsigned short spdifi1:1; - } bf; +struct daio_usage { unsigned short data; }; @@ -61,6 +50,7 @@ struct daio_rsc_idx idx_20k2[NUM_DAIOTYP] = { [LINEO3] = {.left = 0x50, .right = 0x51}, [LINEO4] = {.left = 0x70, .right = 0x71}, [LINEIM] = {.left = 0x45, .right = 0xc5}, + [MIC] = {.left = 0x55, .right = 0xd5}, [SPDIFOO] = {.left = 0x00, .right = 0x01}, [SPDIFIO] = {.left = 0x05, .right = 0x85}, }; @@ -138,6 +128,7 @@ static unsigned int daio_device_index(enum DAIOTYP type, struct hw *hw) case LINEO3: return 5; case LINEO4: return 6; case LINEIM: return 4; + case MIC: return 5; default: return -EINVAL; } default: @@ -519,17 +510,17 @@ static int dai_rsc_uninit(struct dai *dai) static int daio_mgr_get_rsc(struct rsc_mgr *mgr, enum DAIOTYP type) { - if (((union daio_usage *)mgr->rscs)->data & (0x1 << type)) + if (((struct daio_usage *)mgr->rscs)->data & (0x1 << type)) return -ENOENT; - ((union daio_usage *)mgr->rscs)->data |= (0x1 << type); + ((struct daio_usage *)mgr->rscs)->data |= (0x1 << type); return 0; } static int daio_mgr_put_rsc(struct rsc_mgr *mgr, enum DAIOTYP type) { - ((union daio_usage *)mgr->rscs)->data &= ~(0x1 << type); + ((struct daio_usage *)mgr->rscs)->data &= ~(0x1 << type); return 0; } @@ -712,7 +703,7 @@ int daio_mgr_create(void *hw, struct daio_mgr **rdaio_mgr) if (!daio_mgr) return -ENOMEM; - err = rsc_mgr_init(&daio_mgr->mgr, DAIO, DAIO_RESOURCE_NUM, hw); + err = rsc_mgr_init(&daio_mgr->mgr, DAIO, NUM_DAIOTYP, hw); if (err) goto error1; diff --git a/trunk/sound/pci/ctxfi/ctdaio.h b/trunk/sound/pci/ctxfi/ctdaio.h index 0f52ce571ee8..85ccb6ee1ab4 100644 --- a/trunk/sound/pci/ctxfi/ctdaio.h +++ b/trunk/sound/pci/ctxfi/ctdaio.h @@ -33,6 +33,7 @@ enum DAIOTYP { SPDIFOO, /* S/PDIF Out (Flexijack/Optical) */ LINEIM, SPDIFIO, /* S/PDIF In (Flexijack/Optical) on the card */ + MIC, /* Dedicated mic on Titanium HD */ SPDIFI1, /* S/PDIF In on internal Drive Bay */ NUM_DAIOTYP }; diff --git a/trunk/sound/pci/ctxfi/cthardware.h b/trunk/sound/pci/ctxfi/cthardware.h index af55405f5dec..de59bbf2702f 100644 --- a/trunk/sound/pci/ctxfi/cthardware.h +++ b/trunk/sound/pci/ctxfi/cthardware.h @@ -39,6 +39,7 @@ enum CTCARDS { CT20K2_MODEL_FIRST = CTSB0760, CTHENDRIX, CTSB0880, + CTSB1270, CT20K2_UNKNOWN, NUM_CTCARDS /* This should always be the last */ }; @@ -71,6 +72,13 @@ struct hw { int (*is_adc_source_selected)(struct hw *hw, enum ADCSRC source); int (*select_adc_source)(struct hw *hw, enum ADCSRC source); int (*have_digit_io_switch)(struct hw *hw); + int (*have_dedicated_mic)(struct hw *hw); + int (*have_output_switch)(struct hw *hw); + int (*output_switch_get)(struct hw *hw); + int (*output_switch_put)(struct hw *hw, int position); + int (*have_mic_source_switch)(struct hw *hw); + int (*mic_source_switch_get)(struct hw *hw); + int (*mic_source_switch_put)(struct hw *hw, int position); /* SRC operations */ int (*src_rsc_get_ctrl_blk)(void **rblk); diff --git a/trunk/sound/pci/ctxfi/cthw20k1.c b/trunk/sound/pci/ctxfi/cthw20k1.c index a5c957db5cea..9a85a84b23ab 100644 --- a/trunk/sound/pci/ctxfi/cthw20k1.c +++ b/trunk/sound/pci/ctxfi/cthw20k1.c @@ -1783,6 +1783,21 @@ static int hw_have_digit_io_switch(struct hw *hw) return !(hw->model == CTSB073X || hw->model == CTUAA); } +static int hw_have_dedicated_mic(struct hw *hw) +{ + return 0; +} + +static int hw_have_output_switch(struct hw *hw) +{ + return 0; +} + +static int hw_have_mic_source_switch(struct hw *hw) +{ + return 0; +} + #define CTLBITS(a, b, c, d) (((a) << 24) | ((b) << 16) | ((c) << 8) | (d)) #define UAA_CFG_PWRSTATUS 0x44 @@ -2173,6 +2188,9 @@ static struct hw ct20k1_preset __devinitdata = { .is_adc_source_selected = hw_is_adc_input_selected, .select_adc_source = hw_adc_input_select, .have_digit_io_switch = hw_have_digit_io_switch, + .have_dedicated_mic = hw_have_dedicated_mic, + .have_output_switch = hw_have_output_switch, + .have_mic_source_switch = hw_have_mic_source_switch, #ifdef CONFIG_PM .suspend = hw_suspend, .resume = hw_resume, diff --git a/trunk/sound/pci/ctxfi/cthw20k2.c b/trunk/sound/pci/ctxfi/cthw20k2.c index 5364164674e4..8bc6e41ce64b 100644 --- a/trunk/sound/pci/ctxfi/cthw20k2.c +++ b/trunk/sound/pci/ctxfi/cthw20k2.c @@ -8,7 +8,7 @@ * @File cthw20k2.c * * @Brief - * This file contains the implementation of hardware access methord for 20k2. + * This file contains the implementation of hardware access method for 20k2. * * @Author Liu Chun * @Date May 14 2008 @@ -38,6 +38,8 @@ struct hw20k2 { unsigned char dev_id; unsigned char addr_size; unsigned char data_size; + + int mic_source; }; static u32 hw_read_20kx(struct hw *hw, u32 reg); @@ -1163,7 +1165,12 @@ static int hw_daio_init(struct hw *hw, const struct daio_conf *info) hw_write_20kx(hw, AUDIO_IO_TX_BLRCLK, 0x01010101); hw_write_20kx(hw, AUDIO_IO_RX_BLRCLK, 0); } else if (2 == info->msr) { - hw_write_20kx(hw, AUDIO_IO_MCLK, 0x11111111); + if (hw->model != CTSB1270) { + hw_write_20kx(hw, AUDIO_IO_MCLK, 0x11111111); + } else { + /* PCM4220 on Titanium HD is different. */ + hw_write_20kx(hw, AUDIO_IO_MCLK, 0x11011111); + } /* Specify all playing 96khz * EA [0] - Enabled * RTA [4:5] - 96kHz @@ -1175,6 +1182,10 @@ static int hw_daio_init(struct hw *hw, const struct daio_conf *info) * RTD [28:29] - 96kHz */ hw_write_20kx(hw, AUDIO_IO_TX_BLRCLK, 0x11111111); hw_write_20kx(hw, AUDIO_IO_RX_BLRCLK, 0); + } else if ((4 == info->msr) && (hw->model == CTSB1270)) { + hw_write_20kx(hw, AUDIO_IO_MCLK, 0x21011111); + hw_write_20kx(hw, AUDIO_IO_TX_BLRCLK, 0x21212121); + hw_write_20kx(hw, AUDIO_IO_RX_BLRCLK, 0); } else { printk(KERN_ALERT "ctxfi: ERROR!!! Invalid sampling rate!!!\n"); return -EINVAL; @@ -1182,6 +1193,8 @@ static int hw_daio_init(struct hw *hw, const struct daio_conf *info) for (i = 0; i < 8; i++) { if (i <= 3) { + /* This comment looks wrong since loop is over 4 */ + /* channels and emu20k2 supports 4 spdif IOs. */ /* 1st 3 channels are SPDIFs (SB0960) */ if (i == 3) data = 0x1001001; @@ -1206,12 +1219,16 @@ static int hw_daio_init(struct hw *hw, const struct daio_conf *info) hw_write_20kx(hw, AUDIO_IO_TX_CSTAT_H+(0x40*i), 0x0B); } else { + /* Again, loop is over 4 channels not 5. */ /* Next 5 channels are I2S (SB0960) */ data = 0x11; hw_write_20kx(hw, AUDIO_IO_RX_CTL+(0x40*i), data); if (2 == info->msr) { /* Four channels per sample period */ data |= 0x1000; + } else if (4 == info->msr) { + /* FIXME: check this against the chip spec */ + data |= 0x2000; } hw_write_20kx(hw, AUDIO_IO_TX_CTL+(0x40*i), data); } @@ -1557,7 +1574,7 @@ static int hw20k2_i2c_write(struct hw *hw, u16 addr, u32 data) hw_write_20kx(hw, I2C_IF_STATUS, i2c_status); hw20k2_i2c_wait_data_ready(hw); - /* Dummy write to trigger the write oprtation */ + /* Dummy write to trigger the write operation */ hw_write_20kx(hw, I2C_IF_WDATA, 0); hw20k2_i2c_wait_data_ready(hw); @@ -1568,6 +1585,30 @@ static int hw20k2_i2c_write(struct hw *hw, u16 addr, u32 data) return 0; } +static void hw_dac_stop(struct hw *hw) +{ + u32 data; + data = hw_read_20kx(hw, GPIO_DATA); + data &= 0xFFFFFFFD; + hw_write_20kx(hw, GPIO_DATA, data); + mdelay(10); +} + +static void hw_dac_start(struct hw *hw) +{ + u32 data; + data = hw_read_20kx(hw, GPIO_DATA); + data |= 0x2; + hw_write_20kx(hw, GPIO_DATA, data); + mdelay(50); +} + +static void hw_dac_reset(struct hw *hw) +{ + hw_dac_stop(hw); + hw_dac_start(hw); +} + static int hw_dac_init(struct hw *hw, const struct dac_conf *info) { int err; @@ -1594,6 +1635,21 @@ static int hw_dac_init(struct hw *hw, const struct dac_conf *info) 0x00000000 /* Vol Control B4 */ }; + if (hw->model == CTSB1270) { + hw_dac_stop(hw); + data = hw_read_20kx(hw, GPIO_DATA); + data &= ~0x0600; + if (1 == info->msr) + data |= 0x0000; /* Single Speed Mode 0-50kHz */ + else if (2 == info->msr) + data |= 0x0200; /* Double Speed Mode 50-100kHz */ + else + data |= 0x0600; /* Quad Speed Mode 100-200kHz */ + hw_write_20kx(hw, GPIO_DATA, data); + hw_dac_start(hw); + return 0; + } + /* Set DAC reset bit as output */ data = hw_read_20kx(hw, GPIO_CTRL); data |= 0x02; @@ -1606,22 +1662,8 @@ static int hw_dac_init(struct hw *hw, const struct dac_conf *info) for (i = 0; i < 2; i++) { /* Reset DAC twice just in-case the chip * didn't initialized properly */ - data = hw_read_20kx(hw, GPIO_DATA); - /* GPIO data bit 1 */ - data &= 0xFFFFFFFD; - hw_write_20kx(hw, GPIO_DATA, data); - mdelay(10); - data |= 0x2; - hw_write_20kx(hw, GPIO_DATA, data); - mdelay(50); - - /* Reset the 2nd time */ - data &= 0xFFFFFFFD; - hw_write_20kx(hw, GPIO_DATA, data); - mdelay(10); - data |= 0x2; - hw_write_20kx(hw, GPIO_DATA, data); - mdelay(50); + hw_dac_reset(hw); + hw_dac_reset(hw); if (hw20k2_i2c_read(hw, CS4382_MC1, &cs_read.mode_control_1)) continue; @@ -1725,7 +1767,11 @@ static int hw_dac_init(struct hw *hw, const struct dac_conf *info) static int hw_is_adc_input_selected(struct hw *hw, enum ADCSRC type) { u32 data; - + if (hw->model == CTSB1270) { + /* Titanium HD has two ADC chips, one for line in and one */ + /* for MIC. We don't need to switch the ADC input. */ + return 1; + } data = hw_read_20kx(hw, GPIO_DATA); switch (type) { case ADC_MICIN: @@ -1742,35 +1788,47 @@ static int hw_is_adc_input_selected(struct hw *hw, enum ADCSRC type) #define MIC_BOOST_0DB 0xCF #define MIC_BOOST_STEPS_PER_DB 2 -#define MIC_BOOST_20DB (MIC_BOOST_0DB + 20 * MIC_BOOST_STEPS_PER_DB) + +static void hw_wm8775_input_select(struct hw *hw, u8 input, s8 gain_in_db) +{ + u32 adcmc, gain; + + if (input > 3) + input = 3; + + adcmc = ((u32)1 << input) | 0x100; /* Link L+R gain... */ + + hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_ADCMC, adcmc), + MAKE_WM8775_DATA(adcmc)); + + if (gain_in_db < -103) + gain_in_db = -103; + if (gain_in_db > 24) + gain_in_db = 24; + + gain = gain_in_db * MIC_BOOST_STEPS_PER_DB + MIC_BOOST_0DB; + + hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_AADCL, gain), + MAKE_WM8775_DATA(gain)); + /* ...so there should be no need for the following. */ + hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_AADCR, gain), + MAKE_WM8775_DATA(gain)); +} static int hw_adc_input_select(struct hw *hw, enum ADCSRC type) { u32 data; - data = hw_read_20kx(hw, GPIO_DATA); switch (type) { case ADC_MICIN: data |= (0x1 << 14); hw_write_20kx(hw, GPIO_DATA, data); - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_ADCMC, 0x101), - MAKE_WM8775_DATA(0x101)); /* Mic-in */ - hw20k2_i2c_write(hw, - MAKE_WM8775_ADDR(WM8775_AADCL, MIC_BOOST_20DB), - MAKE_WM8775_DATA(MIC_BOOST_20DB)); /* +20dB */ - hw20k2_i2c_write(hw, - MAKE_WM8775_ADDR(WM8775_AADCR, MIC_BOOST_20DB), - MAKE_WM8775_DATA(MIC_BOOST_20DB)); /* +20dB */ + hw_wm8775_input_select(hw, 0, 20); /* Mic, 20dB */ break; case ADC_LINEIN: data &= ~(0x1 << 14); hw_write_20kx(hw, GPIO_DATA, data); - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_ADCMC, 0x102), - MAKE_WM8775_DATA(0x102)); /* Line-in */ - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_AADCL, 0xCF), - MAKE_WM8775_DATA(0xCF)); /* No boost */ - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_AADCR, 0xCF), - MAKE_WM8775_DATA(0xCF)); /* No boost */ + hw_wm8775_input_select(hw, 1, 0); /* Line-in, 0dB */ break; default: break; @@ -1782,7 +1840,7 @@ static int hw_adc_input_select(struct hw *hw, enum ADCSRC type) static int hw_adc_init(struct hw *hw, const struct adc_conf *info) { int err; - u32 mux = 2, data, ctl; + u32 data, ctl; /* Set ADC reset bit as output */ data = hw_read_20kx(hw, GPIO_CTRL); @@ -1796,19 +1854,42 @@ static int hw_adc_init(struct hw *hw, const struct adc_conf *info) goto error; } - /* Make ADC in normal operation */ + /* Reset the ADC (reset is active low). */ data = hw_read_20kx(hw, GPIO_DATA); data &= ~(0x1 << 15); + hw_write_20kx(hw, GPIO_DATA, data); + + if (hw->model == CTSB1270) { + /* Set up the PCM4220 ADC on Titanium HD */ + data &= ~0x0C; + if (1 == info->msr) + data |= 0x00; /* Single Speed Mode 32-50kHz */ + else if (2 == info->msr) + data |= 0x08; /* Double Speed Mode 50-108kHz */ + else + data |= 0x04; /* Quad Speed Mode 108kHz-216kHz */ + hw_write_20kx(hw, GPIO_DATA, data); + } + mdelay(10); + /* Return the ADC to normal operation. */ data |= (0x1 << 15); hw_write_20kx(hw, GPIO_DATA, data); mdelay(50); + /* I2C write to register offset 0x0B to set ADC LRCLK polarity */ + /* invert bit, interface format to I2S, word length to 24-bit, */ + /* enable ADC high pass filter. Fixes bug 5323? */ + hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_IC, 0x26), + MAKE_WM8775_DATA(0x26)); + /* Set the master mode (256fs) */ if (1 == info->msr) { + /* slave mode, 128x oversampling 256fs */ hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_MMC, 0x02), MAKE_WM8775_DATA(0x02)); - } else if (2 == info->msr) { + } else if ((2 == info->msr) || (4 == info->msr)) { + /* slave mode, 64x oversampling, 256fs */ hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_MMC, 0x0A), MAKE_WM8775_DATA(0x0A)); } else { @@ -1818,47 +1899,17 @@ static int hw_adc_init(struct hw *hw, const struct adc_conf *info) goto error; } - /* Configure GPIO bit 14 change to line-in/mic-in */ - ctl = hw_read_20kx(hw, GPIO_CTRL); - ctl |= 0x1 << 14; - hw_write_20kx(hw, GPIO_CTRL, ctl); - - /* Check using Mic-in or Line-in */ - data = hw_read_20kx(hw, GPIO_DATA); - - if (mux == 1) { - /* Configures GPIO data to select Mic-in */ - data |= 0x1 << 14; - hw_write_20kx(hw, GPIO_DATA, data); - - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_ADCMC, 0x101), - MAKE_WM8775_DATA(0x101)); /* Mic-in */ - hw20k2_i2c_write(hw, - MAKE_WM8775_ADDR(WM8775_AADCL, MIC_BOOST_20DB), - MAKE_WM8775_DATA(MIC_BOOST_20DB)); /* +20dB */ - hw20k2_i2c_write(hw, - MAKE_WM8775_ADDR(WM8775_AADCR, MIC_BOOST_20DB), - MAKE_WM8775_DATA(MIC_BOOST_20DB)); /* +20dB */ - } else if (mux == 2) { - /* Configures GPIO data to select Line-in */ - data &= ~(0x1 << 14); - hw_write_20kx(hw, GPIO_DATA, data); - - /* Setup ADC */ - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_ADCMC, 0x102), - MAKE_WM8775_DATA(0x102)); /* Line-in */ - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_AADCL, 0xCF), - MAKE_WM8775_DATA(0xCF)); /* No boost */ - hw20k2_i2c_write(hw, MAKE_WM8775_ADDR(WM8775_AADCR, 0xCF), - MAKE_WM8775_DATA(0xCF)); /* No boost */ + if (hw->model != CTSB1270) { + /* Configure GPIO bit 14 change to line-in/mic-in */ + ctl = hw_read_20kx(hw, GPIO_CTRL); + ctl |= 0x1 << 14; + hw_write_20kx(hw, GPIO_CTRL, ctl); + hw_adc_input_select(hw, ADC_LINEIN); } else { - printk(KERN_ALERT "ctxfi: ERROR!!! Invalid input mux!!!\n"); - err = -EINVAL; - goto error; + hw_wm8775_input_select(hw, 0, 0); } return 0; - error: hw20k2_i2c_uninit(hw); return err; @@ -1869,6 +1920,102 @@ static int hw_have_digit_io_switch(struct hw *hw) return 0; } +static int hw_have_dedicated_mic(struct hw *hw) +{ + return hw->model == CTSB1270; +} + +static int hw_have_output_switch(struct hw *hw) +{ + return hw->model == CTSB1270; +} + +static int hw_output_switch_get(struct hw *hw) +{ + u32 data = hw_read_20kx(hw, GPIO_EXT_DATA); + + switch (data & 0x30) { + case 0x00: + return 0; + case 0x10: + return 1; + case 0x20: + return 2; + default: + return 3; + } +} + +static int hw_output_switch_put(struct hw *hw, int position) +{ + u32 data; + + if (position == hw_output_switch_get(hw)) + return 0; + + /* Mute line and headphones (intended for anti-pop). */ + data = hw_read_20kx(hw, GPIO_DATA); + data |= (0x03 << 11); + hw_write_20kx(hw, GPIO_DATA, data); + + data = hw_read_20kx(hw, GPIO_EXT_DATA) & ~0x30; + switch (position) { + case 0: + break; + case 1: + data |= 0x10; + break; + default: + data |= 0x20; + } + hw_write_20kx(hw, GPIO_EXT_DATA, data); + + /* Unmute line and headphones. */ + data = hw_read_20kx(hw, GPIO_DATA); + data &= ~(0x03 << 11); + hw_write_20kx(hw, GPIO_DATA, data); + + return 1; +} + +static int hw_have_mic_source_switch(struct hw *hw) +{ + return hw->model == CTSB1270; +} + +static int hw_mic_source_switch_get(struct hw *hw) +{ + struct hw20k2 *hw20k2 = (struct hw20k2 *)hw; + + return hw20k2->mic_source; +} + +static int hw_mic_source_switch_put(struct hw *hw, int position) +{ + struct hw20k2 *hw20k2 = (struct hw20k2 *)hw; + + if (position == hw20k2->mic_source) + return 0; + + switch (position) { + case 0: + hw_wm8775_input_select(hw, 0, 0); /* Mic, 0dB */ + break; + case 1: + hw_wm8775_input_select(hw, 1, 0); /* FP Mic, 0dB */ + break; + case 2: + hw_wm8775_input_select(hw, 3, 0); /* Aux Ext, 0dB */ + break; + default: + return 0; + } + + hw20k2->mic_source = position; + + return 1; +} + static irqreturn_t ct_20k2_interrupt(int irq, void *dev_id) { struct hw *hw = dev_id; @@ -2023,13 +2170,16 @@ static int hw_card_init(struct hw *hw, struct card_conf *info) /* Reset all SRC pending interrupts */ hw_write_20kx(hw, SRC_IP, 0); - /* TODO: detect the card ID and configure GPIO accordingly. */ - /* Configures GPIO (0xD802 0x98028) */ - /*hw_write_20kx(hw, GPIO_CTRL, 0x7F07);*/ - /* Configures GPIO (SB0880) */ - /*hw_write_20kx(hw, GPIO_CTRL, 0xFF07);*/ - hw_write_20kx(hw, GPIO_CTRL, 0xD802); - + if (hw->model != CTSB1270) { + /* TODO: detect the card ID and configure GPIO accordingly. */ + /* Configures GPIO (0xD802 0x98028) */ + /*hw_write_20kx(hw, GPIO_CTRL, 0x7F07);*/ + /* Configures GPIO (SB0880) */ + /*hw_write_20kx(hw, GPIO_CTRL, 0xFF07);*/ + hw_write_20kx(hw, GPIO_CTRL, 0xD802); + } else { + hw_write_20kx(hw, GPIO_CTRL, 0x9E5F); + } /* Enable audio ring */ hw_write_20kx(hw, MIXER_AR_ENABLE, 0x01); @@ -2107,6 +2257,13 @@ static struct hw ct20k2_preset __devinitdata = { .is_adc_source_selected = hw_is_adc_input_selected, .select_adc_source = hw_adc_input_select, .have_digit_io_switch = hw_have_digit_io_switch, + .have_dedicated_mic = hw_have_dedicated_mic, + .have_output_switch = hw_have_output_switch, + .output_switch_get = hw_output_switch_get, + .output_switch_put = hw_output_switch_put, + .have_mic_source_switch = hw_have_mic_source_switch, + .mic_source_switch_get = hw_mic_source_switch_get, + .mic_source_switch_put = hw_mic_source_switch_put, #ifdef CONFIG_PM .suspend = hw_suspend, .resume = hw_resume, diff --git a/trunk/sound/pci/ctxfi/ctmixer.c b/trunk/sound/pci/ctxfi/ctmixer.c index c3519ff42fbb..388235c43789 100644 --- a/trunk/sound/pci/ctxfi/ctmixer.c +++ b/trunk/sound/pci/ctxfi/ctmixer.c @@ -86,9 +86,7 @@ enum CTALSA_MIXER_CTL { MIXER_LINEIN_C_S, MIXER_MIC_C_S, MIXER_SPDIFI_C_S, - MIXER_LINEIN_P_S, MIXER_SPDIFO_P_S, - MIXER_SPDIFI_P_S, MIXER_WAVEF_P_S, MIXER_WAVER_P_S, MIXER_WAVEC_P_S, @@ -137,11 +135,11 @@ ct_kcontrol_init_table[NUM_CTALSA_MIXERS] = { }, [MIXER_LINEIN_P] = { .ctl = 1, - .name = "Line-in Playback Volume", + .name = "Line Playback Volume", }, [MIXER_LINEIN_C] = { .ctl = 1, - .name = "Line-in Capture Volume", + .name = "Line Capture Volume", }, [MIXER_MIC_P] = { .ctl = 1, @@ -153,15 +151,15 @@ ct_kcontrol_init_table[NUM_CTALSA_MIXERS] = { }, [MIXER_SPDIFI_P] = { .ctl = 1, - .name = "S/PDIF-in Playback Volume", + .name = "IEC958 Playback Volume", }, [MIXER_SPDIFI_C] = { .ctl = 1, - .name = "S/PDIF-in Capture Volume", + .name = "IEC958 Capture Volume", }, [MIXER_SPDIFO_P] = { .ctl = 1, - .name = "S/PDIF-out Playback Volume", + .name = "Digital Playback Volume", }, [MIXER_WAVEF_P] = { .ctl = 1, @@ -179,14 +177,13 @@ ct_kcontrol_init_table[NUM_CTALSA_MIXERS] = { .ctl = 1, .name = "Surround Playback Volume", }, - [MIXER_PCM_C_S] = { .ctl = 1, .name = "PCM Capture Switch", }, [MIXER_LINEIN_C_S] = { .ctl = 1, - .name = "Line-in Capture Switch", + .name = "Line Capture Switch", }, [MIXER_MIC_C_S] = { .ctl = 1, @@ -194,19 +191,11 @@ ct_kcontrol_init_table[NUM_CTALSA_MIXERS] = { }, [MIXER_SPDIFI_C_S] = { .ctl = 1, - .name = "S/PDIF-in Capture Switch", - }, - [MIXER_LINEIN_P_S] = { - .ctl = 1, - .name = "Line-in Playback Switch", + .name = "IEC958 Capture Switch", }, [MIXER_SPDIFO_P_S] = { .ctl = 1, - .name = "S/PDIF-out Playback Switch", - }, - [MIXER_SPDIFI_P_S] = { - .ctl = 1, - .name = "S/PDIF-in Playback Switch", + .name = "Digital Playback Switch", }, [MIXER_WAVEF_P_S] = { .ctl = 1, @@ -236,6 +225,8 @@ ct_mixer_recording_select(struct ct_mixer *mixer, enum CT_AMIXER_CTL type); static void ct_mixer_recording_unselect(struct ct_mixer *mixer, enum CT_AMIXER_CTL type); +/* FIXME: this static looks like it would fail if more than one card was */ +/* installed. */ static struct snd_kcontrol *kctls[2] = {NULL}; static enum CT_AMIXER_CTL get_amixer_index(enum CTALSA_MIXER_CTL alsa_index) @@ -420,6 +411,77 @@ static struct snd_kcontrol_new vol_ctl = { .tlv = { .p = ct_vol_db_scale }, }; +static int output_switch_info(struct snd_kcontrol *kcontrol, + struct snd_ctl_elem_info *info) +{ + static const char *const names[3] = { + "FP Headphones", "Headphones", "Speakers" + }; + + return snd_ctl_enum_info(info, 1, 3, names); +} + +static int output_switch_get(struct snd_kcontrol *kcontrol, + struct snd_ctl_elem_value *ucontrol) +{ + struct ct_atc *atc = snd_kcontrol_chip(kcontrol); + ucontrol->value.enumerated.item[0] = atc->output_switch_get(atc); + return 0; +} + +static int output_switch_put(struct snd_kcontrol *kcontrol, + struct snd_ctl_elem_value *ucontrol) +{ + struct ct_atc *atc = snd_kcontrol_chip(kcontrol); + if (ucontrol->value.enumerated.item[0] > 2) + return -EINVAL; + return atc->output_switch_put(atc, ucontrol->value.enumerated.item[0]); +} + +static struct snd_kcontrol_new output_ctl = { + .iface = SNDRV_CTL_ELEM_IFACE_MIXER, + .name = "Analog Output Playback Enum", + .info = output_switch_info, + .get = output_switch_get, + .put = output_switch_put, +}; + +static int mic_source_switch_info(struct snd_kcontrol *kcontrol, + struct snd_ctl_elem_info *info) +{ + static const char *const names[3] = { + "Mic", "FP Mic", "Aux" + }; + + return snd_ctl_enum_info(info, 1, 3, names); +} + +static int mic_source_switch_get(struct snd_kcontrol *kcontrol, + struct snd_ctl_elem_value *ucontrol) +{ + struct ct_atc *atc = snd_kcontrol_chip(kcontrol); + ucontrol->value.enumerated.item[0] = atc->mic_source_switch_get(atc); + return 0; +} + +static int mic_source_switch_put(struct snd_kcontrol *kcontrol, + struct snd_ctl_elem_value *ucontrol) +{ + struct ct_atc *atc = snd_kcontrol_chip(kcontrol); + if (ucontrol->value.enumerated.item[0] > 2) + return -EINVAL; + return atc->mic_source_switch_put(atc, + ucontrol->value.enumerated.item[0]); +} + +static struct snd_kcontrol_new mic_source_ctl = { + .iface = SNDRV_CTL_ELEM_IFACE_MIXER, + .name = "Mic Source Capture Enum", + .info = mic_source_switch_info, + .get = mic_source_switch_get, + .put = mic_source_switch_put, +}; + static void do_line_mic_switch(struct ct_atc *atc, enum CTALSA_MIXER_CTL type) { @@ -465,6 +527,7 @@ do_digit_io_switch(struct ct_atc *atc, int state) static void do_switch(struct ct_atc *atc, enum CTALSA_MIXER_CTL type, int state) { struct ct_mixer *mixer = atc->mixer; + int have_dedicated_mic = atc->have_dedicated_mic(atc); /* Do changes in mixer. */ if ((SWH_CAPTURE_START <= type) && (SWH_CAPTURE_END >= type)) { @@ -477,8 +540,17 @@ static void do_switch(struct ct_atc *atc, enum CTALSA_MIXER_CTL type, int state) } } /* Do changes out of mixer. */ - if (state && (MIXER_LINEIN_C_S == type || MIXER_MIC_C_S == type)) - do_line_mic_switch(atc, type); + if (!have_dedicated_mic && + (MIXER_LINEIN_C_S == type || MIXER_MIC_C_S == type)) { + if (state) + do_line_mic_switch(atc, type); + atc->line_in_unmute(atc, state); + } else if (have_dedicated_mic && (MIXER_LINEIN_C_S == type)) + atc->line_in_unmute(atc, state); + else if (have_dedicated_mic && (MIXER_MIC_C_S == type)) + atc->mic_unmute(atc, state); + else if (MIXER_SPDIFI_C_S == type) + atc->spdif_in_unmute(atc, state); else if (MIXER_WAVEF_P_S == type) atc->line_front_unmute(atc, state); else if (MIXER_WAVES_P_S == type) @@ -487,12 +559,8 @@ static void do_switch(struct ct_atc *atc, enum CTALSA_MIXER_CTL type, int state) atc->line_clfe_unmute(atc, state); else if (MIXER_WAVER_P_S == type) atc->line_rear_unmute(atc, state); - else if (MIXER_LINEIN_P_S == type) - atc->line_in_unmute(atc, state); else if (MIXER_SPDIFO_P_S == type) atc->spdif_out_unmute(atc, state); - else if (MIXER_SPDIFI_P_S == type) - atc->spdif_in_unmute(atc, state); else if (MIXER_DIGITAL_IO_S == type) do_digit_io_switch(atc, state); @@ -686,6 +754,7 @@ static int ct_mixer_kcontrols_create(struct ct_mixer *mixer) ct_kcontrol_init_table[MIXER_DIGITAL_IO_S].ctl = atc->have_digit_io_switch(atc); + for (type = SWH_MIXER_START; type <= SWH_MIXER_END; type++) { if (ct_kcontrol_init_table[type].ctl) { swh_ctl.name = ct_kcontrol_init_table[type].name; @@ -708,6 +777,17 @@ static int ct_mixer_kcontrols_create(struct ct_mixer *mixer) if (err) return err; + if (atc->have_output_switch(atc)) { + err = ct_mixer_kcontrol_new(mixer, &output_ctl); + if (err) + return err; + } + + if (atc->have_mic_source_switch(atc)) { + err = ct_mixer_kcontrol_new(mixer, &mic_source_ctl); + if (err) + return err; + } atc->line_front_unmute(atc, 1); set_switch_state(mixer, MIXER_WAVEF_P_S, 1); atc->line_surround_unmute(atc, 0); @@ -719,13 +799,12 @@ static int ct_mixer_kcontrols_create(struct ct_mixer *mixer) atc->spdif_out_unmute(atc, 0); set_switch_state(mixer, MIXER_SPDIFO_P_S, 0); atc->line_in_unmute(atc, 0); - set_switch_state(mixer, MIXER_LINEIN_P_S, 0); + if (atc->have_dedicated_mic(atc)) + atc->mic_unmute(atc, 0); atc->spdif_in_unmute(atc, 0); - set_switch_state(mixer, MIXER_SPDIFI_P_S, 0); - - set_switch_state(mixer, MIXER_PCM_C_S, 1); - set_switch_state(mixer, MIXER_LINEIN_C_S, 1); - set_switch_state(mixer, MIXER_SPDIFI_C_S, 1); + set_switch_state(mixer, MIXER_PCM_C_S, 0); + set_switch_state(mixer, MIXER_LINEIN_C_S, 0); + set_switch_state(mixer, MIXER_SPDIFI_C_S, 0); return 0; } diff --git a/trunk/sound/pci/ctxfi/xfi.c b/trunk/sound/pci/ctxfi/xfi.c index f42e7e1a1074..7afa765afa3c 100644 --- a/trunk/sound/pci/ctxfi/xfi.c +++ b/trunk/sound/pci/ctxfi/xfi.c @@ -80,11 +80,11 @@ ct_card_probe(struct pci_dev *pci, const struct pci_device_id *pci_id) "are 48000 and 44100, Value 48000 is assumed.\n"); reference_rate = 48000; } - if ((multiple != 1) && (multiple != 2)) { + if ((multiple != 1) && (multiple != 2) && (multiple != 4)) { printk(KERN_ERR "ctxfi: Invalid multiple value %u!!!\n", multiple); printk(KERN_ERR "ctxfi: The valid values for multiple are " - "1 and 2, Value 2 is assumed.\n"); + "1, 2 and 4, Value 2 is assumed.\n"); multiple = 2; } err = ct_atc_create(card, pci, reference_rate, multiple, diff --git a/trunk/sound/pci/hda/hda_beep.h b/trunk/sound/pci/hda/hda_beep.h index 55f0647458c7..4967eabe774e 100644 --- a/trunk/sound/pci/hda/hda_beep.h +++ b/trunk/sound/pci/hda/hda_beep.h @@ -54,7 +54,7 @@ static inline int snd_hda_attach_beep_device(struct hda_codec *codec, int nid) { return 0; } -static inline void snd_hda_detach_beep_device(struct hda_codec *codec) +void snd_hda_detach_beep_device(struct hda_codec *codec) { } #endif diff --git a/trunk/sound/pci/hda/hda_eld.c b/trunk/sound/pci/hda/hda_eld.c index e3e853153d14..b05f7be9dc1b 100644 --- a/trunk/sound/pci/hda/hda_eld.c +++ b/trunk/sound/pci/hda/hda_eld.c @@ -294,7 +294,7 @@ static int hdmi_update_eld(struct hdmi_eld *e, snd_printd(KERN_INFO "HDMI: out of range MNL %d\n", mnl); goto out_fail; } else - strlcpy(e->monitor_name, buf + ELD_FIXED_BYTES, mnl + 1); + strlcpy(e->monitor_name, buf + ELD_FIXED_BYTES, mnl); for (i = 0; i < e->sad_count; i++) { if (ELD_FIXED_BYTES + mnl + 3 * (i + 1) > size) { diff --git a/trunk/sound/pci/hda/patch_conexant.c b/trunk/sound/pci/hda/patch_conexant.c index 7bbc5f237a5e..694b9daf691f 100644 --- a/trunk/sound/pci/hda/patch_conexant.c +++ b/trunk/sound/pci/hda/patch_conexant.c @@ -3074,7 +3074,6 @@ static const char * const cxt5066_models[CXT5066_MODELS] = { }; static const struct snd_pci_quirk cxt5066_cfg_tbl[] = { - SND_PCI_QUIRK(0x1025, 0x054c, "Acer Aspire 3830TG", CXT5066_AUTO), SND_PCI_QUIRK_MASK(0x1025, 0xff00, 0x0400, "Acer", CXT5066_IDEAPAD), SND_PCI_QUIRK(0x1028, 0x02d8, "Dell Vostro", CXT5066_DELL_VOSTRO), SND_PCI_QUIRK(0x1028, 0x02f5, "Dell Vostro 320", CXT5066_IDEAPAD), @@ -4390,8 +4389,6 @@ static const struct hda_codec_preset snd_hda_preset_conexant[] = { .patch = patch_cxt5066 }, { .id = 0x14f15069, .name = "CX20585", .patch = patch_cxt5066 }, - { .id = 0x14f1506c, .name = "CX20588", - .patch = patch_cxt5066 }, { .id = 0x14f1506e, .name = "CX20590", .patch = patch_cxt5066 }, { .id = 0x14f15097, .name = "CX20631", @@ -4420,7 +4417,6 @@ MODULE_ALIAS("snd-hda-codec-id:14f15066"); MODULE_ALIAS("snd-hda-codec-id:14f15067"); MODULE_ALIAS("snd-hda-codec-id:14f15068"); MODULE_ALIAS("snd-hda-codec-id:14f15069"); -MODULE_ALIAS("snd-hda-codec-id:14f1506c"); MODULE_ALIAS("snd-hda-codec-id:14f1506e"); MODULE_ALIAS("snd-hda-codec-id:14f15097"); MODULE_ALIAS("snd-hda-codec-id:14f15098"); diff --git a/trunk/sound/pci/hda/patch_realtek.c b/trunk/sound/pci/hda/patch_realtek.c index b48fb43b5448..61a774b3d3cb 100644 --- a/trunk/sound/pci/hda/patch_realtek.c +++ b/trunk/sound/pci/hda/patch_realtek.c @@ -2715,30 +2715,17 @@ typedef int (*getput_call_t)(struct snd_kcontrol *kcontrol, static int alc_cap_getput_caller(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol, - getput_call_t func, bool check_adc_switch) + getput_call_t func) { struct hda_codec *codec = snd_kcontrol_chip(kcontrol); struct alc_spec *spec = codec->spec; - int i, err = 0; + unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id); + int err; mutex_lock(&codec->control_mutex); - if (check_adc_switch && spec->dual_adc_switch) { - for (i = 0; i < spec->num_adc_nids; i++) { - kcontrol->private_value = - HDA_COMPOSE_AMP_VAL(spec->adc_nids[i], - 3, 0, HDA_INPUT); - err = func(kcontrol, ucontrol); - if (err < 0) - goto error; - } - } else { - i = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id); - kcontrol->private_value = - HDA_COMPOSE_AMP_VAL(spec->adc_nids[i], - 3, 0, HDA_INPUT); - err = func(kcontrol, ucontrol); - } - error: + kcontrol->private_value = HDA_COMPOSE_AMP_VAL(spec->adc_nids[adc_idx], + 3, 0, HDA_INPUT); + err = func(kcontrol, ucontrol); mutex_unlock(&codec->control_mutex); return err; } @@ -2747,14 +2734,14 @@ static int alc_cap_vol_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { return alc_cap_getput_caller(kcontrol, ucontrol, - snd_hda_mixer_amp_volume_get, false); + snd_hda_mixer_amp_volume_get); } static int alc_cap_vol_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { return alc_cap_getput_caller(kcontrol, ucontrol, - snd_hda_mixer_amp_volume_put, true); + snd_hda_mixer_amp_volume_put); } /* capture mixer elements */ @@ -2764,14 +2751,14 @@ static int alc_cap_sw_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { return alc_cap_getput_caller(kcontrol, ucontrol, - snd_hda_mixer_amp_switch_get, false); + snd_hda_mixer_amp_switch_get); } static int alc_cap_sw_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) { return alc_cap_getput_caller(kcontrol, ucontrol, - snd_hda_mixer_amp_switch_put, true); + snd_hda_mixer_amp_switch_put); } #define _DEFINE_CAPMIX(num) \ @@ -4896,6 +4883,7 @@ static const struct snd_pci_quirk alc880_cfg_tbl[] = { SND_PCI_QUIRK(0x1025, 0xe309, "ULI", ALC880_3ST_DIG), SND_PCI_QUIRK(0x1025, 0xe310, "ULI", ALC880_3ST), SND_PCI_QUIRK(0x1039, 0x1234, NULL, ALC880_6ST_DIG), + SND_PCI_QUIRK(0x103c, 0x2a09, "HP", ALC880_5ST), SND_PCI_QUIRK(0x1043, 0x10b3, "ASUS W1V", ALC880_ASUS_W1V), SND_PCI_QUIRK(0x1043, 0x10c2, "ASUS W6A", ALC880_ASUS_DIG), SND_PCI_QUIRK(0x1043, 0x10c3, "ASUS Wxx", ALC880_ASUS_DIG), @@ -12612,7 +12600,6 @@ static const struct hda_verb alc262_toshiba_rx1_unsol_verbs[] = { */ enum { PINFIX_FSC_H270, - PINFIX_HP_Z200, }; static const struct alc_fixup alc262_fixups[] = { @@ -12625,17 +12612,9 @@ static const struct alc_fixup alc262_fixups[] = { { } } }, - [PINFIX_HP_Z200] = { - .type = ALC_FIXUP_PINS, - .v.pins = (const struct alc_pincfg[]) { - { 0x16, 0x99130120 }, /* internal speaker */ - { } - } - }, }; static const struct snd_pci_quirk alc262_fixup_tbl[] = { - SND_PCI_QUIRK(0x103c, 0x170b, "HP Z200", PINFIX_HP_Z200), SND_PCI_QUIRK(0x1734, 0x1147, "FSC Celsius H270", PINFIX_FSC_H270), {} }; @@ -12752,8 +12731,6 @@ static const struct snd_pci_quirk alc262_cfg_tbl[] = { ALC262_HP_BPC), SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x1500, "HP z series", ALC262_HP_BPC), - SND_PCI_QUIRK(0x103c, 0x170b, "HP Z200", - ALC262_AUTO), SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x1700, "HP xw series", ALC262_HP_BPC), SND_PCI_QUIRK(0x103c, 0x2800, "HP D7000", ALC262_HP_BPC_D7000_WL), @@ -13895,6 +13872,7 @@ static const struct snd_pci_quirk alc268_cfg_tbl[] = { SND_PCI_QUIRK(0x1043, 0x1205, "ASUS W7J", ALC268_3ST), SND_PCI_QUIRK(0x1170, 0x0040, "ZEPTO", ALC268_ZEPTO), SND_PCI_QUIRK(0x14c0, 0x0025, "COMPAL IFL90/JFL-92", ALC268_TOSHIBA), + SND_PCI_QUIRK(0x152d, 0x0763, "Diverse (CPR2000)", ALC268_ACER), SND_PCI_QUIRK(0x152d, 0x0771, "Quanta IL1", ALC267_QUANTA_IL1), {} }; diff --git a/trunk/sound/pci/hda/patch_via.c b/trunk/sound/pci/hda/patch_via.c index f43bb0eaed8b..605c99e1e520 100644 --- a/trunk/sound/pci/hda/patch_via.c +++ b/trunk/sound/pci/hda/patch_via.c @@ -745,23 +745,12 @@ static int via_independent_hp_put(struct snd_kcontrol *kcontrol, struct via_spec *spec = codec->spec; hda_nid_t nid = kcontrol->private_value; unsigned int pinsel = ucontrol->value.enumerated.item[0]; - unsigned int parm0, parm1; /* Get Independent Mode index of headphone pin widget */ spec->hp_independent_mode = spec->hp_independent_mode_index == pinsel ? 1 : 0; - if (spec->codec_type == VT1718S) { + if (spec->codec_type == VT1718S) snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, pinsel ? 2 : 0); - /* Set correct mute switch for MW3 */ - parm0 = spec->hp_independent_mode ? - AMP_IN_UNMUTE(0) : AMP_IN_MUTE(0); - parm1 = spec->hp_independent_mode ? - AMP_IN_MUTE(1) : AMP_IN_UNMUTE(1); - snd_hda_codec_write(codec, 0x1b, 0, - AC_VERB_SET_AMP_GAIN_MUTE, parm0); - snd_hda_codec_write(codec, 0x1b, 0, - AC_VERB_SET_AMP_GAIN_MUTE, parm1); - } else snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_CONNECT_SEL, pinsel); @@ -843,13 +832,10 @@ static int via_hp_build(struct hda_codec *codec) knew->subdevice = HDA_SUBDEV_NID_FLAG | nid; knew->private_value = nid; - nid = side_mute_channel(spec); - if (nid) { - knew = via_clone_control(spec, &via_hp_mixer[1]); - if (knew == NULL) - return -ENOMEM; - knew->subdevice = nid; - } + knew = via_clone_control(spec, &via_hp_mixer[1]); + if (knew == NULL) + return -ENOMEM; + knew->subdevice = side_mute_channel(spec); return 0; } @@ -4294,6 +4280,9 @@ static const struct hda_verb vt1718S_volume_init_verbs[] = { {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, {0x21, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(5)}, + + /* Setup default input of Front HP to MW9 */ + {0x28, AC_VERB_SET_CONNECT_SEL, 0x1}, /* PW9 PW10 Output enable */ {0x2d, AC_VERB_SET_PIN_WIDGET_CONTROL, AC_PINCTL_OUT_EN}, {0x2e, AC_VERB_SET_PIN_WIDGET_CONTROL, AC_PINCTL_OUT_EN}, @@ -4302,10 +4291,10 @@ static const struct hda_verb vt1718S_volume_init_verbs[] = { /* Enable Boost Volume backdoor */ {0x1, 0xf88, 0x8}, /* MW0/1/2/3/4: un-mute index 0 (AOWx), mute index 1 (MW9) */ - {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, + {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, {0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, - {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, + {0x1b, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, {0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1)}, {0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, @@ -4315,6 +4304,8 @@ static const struct hda_verb vt1718S_volume_init_verbs[] = { /* set MUX1 = 2 (AOW4), MUX2 = 1 (AOW3) */ {0x34, AC_VERB_SET_CONNECT_SEL, 0x2}, {0x35, AC_VERB_SET_CONNECT_SEL, 0x1}, + /* Unmute MW4's index 0 */ + {0x1c, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, { } }; @@ -4462,19 +4453,6 @@ static int vt1718S_auto_create_multi_out_ctls(struct via_spec *spec, if (err < 0) return err; } else if (i == AUTO_SEQ_FRONT) { - /* add control to mixer index 0 */ - err = via_add_control(spec, VIA_CTL_WIDGET_VOL, - "Master Front Playback Volume", - HDA_COMPOSE_AMP_VAL(0x21, 3, 5, - HDA_INPUT)); - if (err < 0) - return err; - err = via_add_control(spec, VIA_CTL_WIDGET_MUTE, - "Master Front Playback Switch", - HDA_COMPOSE_AMP_VAL(0x21, 3, 5, - HDA_INPUT)); - if (err < 0) - return err; /* Front */ sprintf(name, "%s Playback Volume", chname[i]); err = via_add_control( diff --git a/trunk/sound/pci/lola/lola.c b/trunk/sound/pci/lola/lola.c index 2692e5ae5f2d..34b24286d279 100644 --- a/trunk/sound/pci/lola/lola.c +++ b/trunk/sound/pci/lola/lola.c @@ -445,7 +445,7 @@ static void lola_reset_setups(struct lola *chip) lola_setup_all_analog_gains(chip, PLAY, false); /* output, update */ } -static int __devinit lola_parse_tree(struct lola *chip) +static int lola_parse_tree(struct lola *chip) { unsigned int val; int nid, err; diff --git a/trunk/sound/pci/rme9652/hdspm.c b/trunk/sound/pci/rme9652/hdspm.c index c8e402fc3782..3f08afc0f0d3 100644 --- a/trunk/sound/pci/rme9652/hdspm.c +++ b/trunk/sound/pci/rme9652/hdspm.c @@ -896,11 +896,11 @@ struct hdspm { unsigned char max_channels_in; unsigned char max_channels_out; - signed char *channel_map_in; - signed char *channel_map_out; + char *channel_map_in; + char *channel_map_out; - signed char *channel_map_in_ss, *channel_map_in_ds, *channel_map_in_qs; - signed char *channel_map_out_ss, *channel_map_out_ds, *channel_map_out_qs; + char *channel_map_in_ss, *channel_map_in_ds, *channel_map_in_qs; + char *channel_map_out_ss, *channel_map_out_ds, *channel_map_out_qs; char **port_names_in; char **port_names_out; diff --git a/trunk/sound/soc/Makefile b/trunk/sound/soc/Makefile index 4f913876f332..1ed61c5df2c5 100644 --- a/trunk/sound/soc/Makefile +++ b/trunk/sound/soc/Makefile @@ -1,5 +1,4 @@ snd-soc-core-objs := soc-core.o soc-dapm.o soc-jack.o soc-cache.o soc-utils.o -snd-soc-core-objs += soc-pcm.o soc-io.o obj-$(CONFIG_SND_SOC) += snd-soc-core.o obj-$(CONFIG_SND_SOC) += codecs/ diff --git a/trunk/sound/soc/atmel/atmel-pcm.c b/trunk/sound/soc/atmel/atmel-pcm.c index f81d4c3f8956..d0e75323ec19 100644 --- a/trunk/sound/soc/atmel/atmel-pcm.c +++ b/trunk/sound/soc/atmel/atmel-pcm.c @@ -364,11 +364,9 @@ static struct snd_pcm_ops atmel_pcm_ops = { \*--------------------------------------------------------------------------*/ static u64 atmel_pcm_dmamask = 0xffffffff; -static int atmel_pcm_new(struct snd_soc_pcm_runtime *rtd) +static int atmel_pcm_new(struct snd_card *card, + struct snd_soc_dai *dai, struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; if (!card->dev->dma_mask) @@ -384,7 +382,7 @@ static int atmel_pcm_new(struct snd_soc_pcm_runtime *rtd) } if (dai->driver->capture.channels_min) { - pr_debug("atmel-pcm:" + pr_debug("at32-pcm:" "Allocating PCM capture DMA buffer\n"); ret = atmel_pcm_preallocate_dma_buffer(pcm, SNDRV_PCM_STREAM_CAPTURE); diff --git a/trunk/sound/soc/atmel/atmel-pcm.h b/trunk/sound/soc/atmel/atmel-pcm.h index 5e0a95e64329..2597329302e7 100644 --- a/trunk/sound/soc/atmel/atmel-pcm.h +++ b/trunk/sound/soc/atmel/atmel-pcm.h @@ -60,7 +60,7 @@ struct atmel_ssc_mask { * This structure, shared between the PCM driver and the interface, * contains all information required by the PCM driver to perform the * PDC DMA operation. All fields except dma_intr_handler() are initialized - * by the interface. The dma_intr_handler() pointer is set by the PCM + * by the interface. The dms_intr_handler() pointer is set by the PCM * driver and called by the interface SSC interrupt handler if it is * non-NULL. */ diff --git a/trunk/sound/soc/atmel/atmel_ssc_dai.c b/trunk/sound/soc/atmel/atmel_ssc_dai.c index 71225090c49f..eda955b15834 100644 --- a/trunk/sound/soc/atmel/atmel_ssc_dai.c +++ b/trunk/sound/soc/atmel/atmel_ssc_dai.c @@ -402,7 +402,7 @@ static int atmel_ssc_hw_params(struct snd_pcm_substream *substream, if ((ssc_p->daifmt & SND_SOC_DAIFMT_FORMAT_MASK) == SND_SOC_DAIFMT_I2S && bits > 16) { printk(KERN_WARNING - "atmel_ssc_dai: sample size %d " + "atmel_ssc_dai: sample size %d" "is too large for I2S\n", bits); return -EINVAL; } @@ -838,8 +838,10 @@ int atmel_ssc_set_audio(int ssc_id) } ssc_pdev = platform_device_alloc("atmel-ssc-dai", ssc_id); - if (!ssc_pdev) + if (!ssc_pdev) { + ssc_free(ssc); return -ENOMEM; + } /* If we can grab the SSC briefly to parent the DAI device off it */ ssc = ssc_request(ssc_id); diff --git a/trunk/sound/soc/atmel/sam9g20_wm8731.c b/trunk/sound/soc/atmel/sam9g20_wm8731.c index bad3aa14d5b3..95572d290c27 100644 --- a/trunk/sound/soc/atmel/sam9g20_wm8731.c +++ b/trunk/sound/soc/atmel/sam9g20_wm8731.c @@ -92,7 +92,6 @@ static struct snd_soc_ops at91sam9g20ek_ops = { }; static int at91sam9g20ek_set_bias_level(struct snd_soc_card *card, - struct snd_soc_dapm_context *dapm, enum snd_soc_bias_level level) { static int mclk_on; diff --git a/trunk/sound/soc/au1x/dbdma2.c b/trunk/sound/soc/au1x/dbdma2.c index 20bb53a837b1..10fdd2854e58 100644 --- a/trunk/sound/soc/au1x/dbdma2.c +++ b/trunk/sound/soc/au1x/dbdma2.c @@ -319,11 +319,10 @@ static void au1xpsc_pcm_free_dma_buffers(struct snd_pcm *pcm) snd_pcm_lib_preallocate_free_for_all(pcm); } -static int au1xpsc_pcm_new(struct snd_soc_pcm_runtime *rtd) +static int au1xpsc_pcm_new(struct snd_card *card, + struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_pcm *pcm = rtd->pcm; - snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV, card->dev, AU1XPSC_BUFFER_MIN_BYTES, (4096 * 1024) - 1); diff --git a/trunk/sound/soc/blackfin/Kconfig b/trunk/sound/soc/blackfin/Kconfig index fe9d548a6837..ae403597fd31 100644 --- a/trunk/sound/soc/blackfin/Kconfig +++ b/trunk/sound/soc/blackfin/Kconfig @@ -10,35 +10,12 @@ config SND_BF5XX_I2S config SND_BF5XX_SOC_SSM2602 tristate "SoC SSM2602 Audio support for BF52x ezkit" - depends on SND_BF5XX_I2S && (SPI_MASTER || I2C) - select SND_BF5XX_SOC_I2S - select SND_SOC_SSM2602 - help - Say Y if you want to add support for SoC audio on BF527-EZKIT. - -config SND_SOC_BFIN_EVAL_ADAU1701 - tristate "Support for the EVAL-ADAU1701MINIZ board on Blackfin eval boards" depends on SND_BF5XX_I2S select SND_BF5XX_SOC_I2S - select SND_SOC_ADAU1701 + select SND_SOC_SSM2602 select I2C help - Say Y if you want to add support for the Analog Devices EVAL-ADAU1701MINIZ - board connected to one of the Blackfin evaluation boards like the - BF5XX-STAMP or BF5XX-EZKIT. - -config SND_SOC_BFIN_EVAL_ADAV80X - tristate "Support for the EVAL-ADAV80X boards on Blackfin eval boards" - depends on SND_BF5XX_I2S && (SPI_MASTER || I2C) - select SND_BF5XX_SOC_I2S - select SND_SOC_ADAV80X - help - Say Y if you want to add support for the Analog Devices EVAL-ADAV801 or - EVAL-ADAV803 board connected to one of the Blackfin evaluation boards - like the BF5XX-STAMP or BF5XX-EZKIT. - - Note: This driver assumes that the ADAV80X digital record and playback - interfaces are connected to the first SPORT port on the BF5XX board. + Say Y if you want to add support for SoC audio on BF527-EZKIT. config SND_BF5XX_SOC_AD73311 tristate "SoC AD73311 Audio support for Blackfin" diff --git a/trunk/sound/soc/blackfin/Makefile b/trunk/sound/soc/blackfin/Makefile index 6018bf52a234..49af3f32aec8 100644 --- a/trunk/sound/soc/blackfin/Makefile +++ b/trunk/sound/soc/blackfin/Makefile @@ -21,13 +21,9 @@ snd-ad1980-objs := bf5xx-ad1980.o snd-ssm2602-objs := bf5xx-ssm2602.o snd-ad73311-objs := bf5xx-ad73311.o snd-ad193x-objs := bf5xx-ad193x.o -snd-soc-bfin-eval-adau1701-objs := bfin-eval-adau1701.o -snd-soc-bfin-eval-adav80x-objs := bfin-eval-adav80x.o obj-$(CONFIG_SND_BF5XX_SOC_AD1836) += snd-ad1836.o obj-$(CONFIG_SND_BF5XX_SOC_AD1980) += snd-ad1980.o obj-$(CONFIG_SND_BF5XX_SOC_SSM2602) += snd-ssm2602.o obj-$(CONFIG_SND_BF5XX_SOC_AD73311) += snd-ad73311.o obj-$(CONFIG_SND_BF5XX_SOC_AD193X) += snd-ad193x.o -obj-$(CONFIG_SND_SOC_BFIN_EVAL_ADAU1701) += snd-soc-bfin-eval-adau1701.o -obj-$(CONFIG_SND_SOC_BFIN_EVAL_ADAV80X) += snd-soc-bfin-eval-adav80x.o diff --git a/trunk/sound/soc/blackfin/bf5xx-ac97-pcm.c b/trunk/sound/soc/blackfin/bf5xx-ac97-pcm.c index 9e59f680bc19..98b44b316e78 100644 --- a/trunk/sound/soc/blackfin/bf5xx-ac97-pcm.c +++ b/trunk/sound/soc/blackfin/bf5xx-ac97-pcm.c @@ -418,11 +418,9 @@ static void bf5xx_pcm_free_dma_buffers(struct snd_pcm *pcm) static u64 bf5xx_pcm_dmamask = DMA_BIT_MASK(32); -int bf5xx_pcm_ac97_new(struct snd_soc_pcm_runtime *rtd) +int bf5xx_pcm_ac97_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; pr_debug("%s enter\n", __func__); diff --git a/trunk/sound/soc/blackfin/bf5xx-i2s-pcm.c b/trunk/sound/soc/blackfin/bf5xx-i2s-pcm.c index 61ddf942fd4d..b5101efd1c87 100644 --- a/trunk/sound/soc/blackfin/bf5xx-i2s-pcm.c +++ b/trunk/sound/soc/blackfin/bf5xx-i2s-pcm.c @@ -138,20 +138,11 @@ static snd_pcm_uframes_t bf5xx_pcm_pointer(struct snd_pcm_substream *substream) pr_debug("%s enter\n", __func__); if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { diff = sport_curr_offset_tx(sport); + frames = bytes_to_frames(substream->runtime, diff); } else { diff = sport_curr_offset_rx(sport); + frames = bytes_to_frames(substream->runtime, diff); } - - /* - * TX at least can report one frame beyond the end of the - * buffer if we hit the wraparound case - clamp to within the - * buffer as the ALSA APIs require. - */ - if (diff == snd_pcm_lib_buffer_bytes(substream)) - diff = 0; - - frames = bytes_to_frames(substream->runtime, diff); - return frames; } @@ -168,7 +159,7 @@ static int bf5xx_pcm_open(struct snd_pcm_substream *substream) snd_soc_set_runtime_hwparams(substream, &bf5xx_pcm_hardware); - ret = snd_pcm_hw_constraint_integer(runtime, + ret = snd_pcm_hw_constraint_integer(runtime, \ SNDRV_PCM_HW_PARAM_PERIODS); if (ret < 0) goto out; @@ -257,11 +248,9 @@ static void bf5xx_pcm_free_dma_buffers(struct snd_pcm *pcm) static u64 bf5xx_pcm_dmamask = DMA_BIT_MASK(32); -int bf5xx_pcm_i2s_new(struct snd_soc_pcm_runtime *rtd) +int bf5xx_pcm_i2s_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; pr_debug("%s enter\n", __func__); @@ -306,8 +295,8 @@ static int __devexit bfin_i2s_soc_platform_remove(struct platform_device *pdev) static struct platform_driver bfin_i2s_pcm_driver = { .driver = { - .name = "bfin-i2s-pcm-audio", - .owner = THIS_MODULE, + .name = "bfin-i2s-pcm-audio", + .owner = THIS_MODULE, }, .probe = bfin_i2s_soc_platform_probe, diff --git a/trunk/sound/soc/blackfin/bf5xx-tdm-pcm.c b/trunk/sound/soc/blackfin/bf5xx-tdm-pcm.c index c95cc03d583d..07cfc7a9e49a 100644 --- a/trunk/sound/soc/blackfin/bf5xx-tdm-pcm.c +++ b/trunk/sound/soc/blackfin/bf5xx-tdm-pcm.c @@ -283,11 +283,9 @@ static void bf5xx_pcm_free_dma_buffers(struct snd_pcm *pcm) static u64 bf5xx_pcm_dmamask = DMA_BIT_MASK(32); -static int bf5xx_pcm_tdm_new(struct snd_soc_pcm_runtime *rtd) +static int bf5xx_pcm_tdm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; if (!card->dev->dma_mask) diff --git a/trunk/sound/soc/blackfin/bfin-eval-adau1701.c b/trunk/sound/soc/blackfin/bfin-eval-adau1701.c deleted file mode 100644 index e5550acba2c2..000000000000 --- a/trunk/sound/soc/blackfin/bfin-eval-adau1701.c +++ /dev/null @@ -1,139 +0,0 @@ -/* - * Machine driver for EVAL-ADAU1701MINIZ on Analog Devices bfin - * evaluation boards. - * - * Copyright 2011 Analog Devices Inc. - * Author: Lars-Peter Clausen - * - * Licensed under the GPL-2 or later. - */ - -#include -#include -#include -#include -#include -#include - -#include "../codecs/adau1701.h" - -static const struct snd_soc_dapm_widget bfin_eval_adau1701_dapm_widgets[] = { - SND_SOC_DAPM_SPK("Speaker", NULL), - SND_SOC_DAPM_LINE("Line Out", NULL), - SND_SOC_DAPM_LINE("Line In", NULL), -}; - -static const struct snd_soc_dapm_route bfin_eval_adau1701_dapm_routes[] = { - { "Speaker", NULL, "OUT0" }, - { "Speaker", NULL, "OUT1" }, - { "Line Out", NULL, "OUT2" }, - { "Line Out", NULL, "OUT3" }, - - { "IN0", NULL, "Line In" }, - { "IN1", NULL, "Line In" }, -}; - -static int bfin_eval_adau1701_hw_params(struct snd_pcm_substream *substream, - struct snd_pcm_hw_params *params) -{ - struct snd_soc_pcm_runtime *rtd = substream->private_data; - struct snd_soc_dai *cpu_dai = rtd->cpu_dai; - struct snd_soc_dai *codec_dai = rtd->codec_dai; - int ret; - - ret = snd_soc_dai_set_fmt(cpu_dai, SND_SOC_DAIFMT_I2S | - SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBM_CFM); - if (ret) - return ret; - - ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_I2S | - SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBM_CFM); - if (ret) - return ret; - - ret = snd_soc_dai_set_sysclk(codec_dai, ADAU1701_CLK_SRC_OSC, 12288000, - SND_SOC_CLOCK_IN); - - return ret; -} - -static struct snd_soc_ops bfin_eval_adau1701_ops = { - .hw_params = bfin_eval_adau1701_hw_params, -}; - -static struct snd_soc_dai_link bfin_eval_adau1701_dai[] = { - { - .name = "adau1701", - .stream_name = "adau1701", - .cpu_dai_name = "bfin-i2s.0", - .codec_dai_name = "adau1701", - .platform_name = "bfin-i2s-pcm-audio", - .codec_name = "adau1701.0-0034", - .ops = &bfin_eval_adau1701_ops, - }, - { - .name = "adau1701", - .stream_name = "adau1701", - .cpu_dai_name = "bfin-i2s.1", - .codec_dai_name = "adau1701", - .platform_name = "bfin-i2s-pcm-audio", - .codec_name = "adau1701.0-0034", - .ops = &bfin_eval_adau1701_ops, - }, -}; - -static struct snd_soc_card bfin_eval_adau1701 = { - .name = "bfin-eval-adau1701", - .dai_link = &bfin_eval_adau1701_dai[CONFIG_SND_BF5XX_SPORT_NUM], - .num_links = 1, - - .dapm_widgets = bfin_eval_adau1701_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(bfin_eval_adau1701_dapm_widgets), - .dapm_routes = bfin_eval_adau1701_dapm_routes, - .num_dapm_routes = ARRAY_SIZE(bfin_eval_adau1701_dapm_routes), -}; - -static int bfin_eval_adau1701_probe(struct platform_device *pdev) -{ - struct snd_soc_card *card = &bfin_eval_adau1701; - - card->dev = &pdev->dev; - - return snd_soc_register_card(&bfin_eval_adau1701); -} - -static int __devexit bfin_eval_adau1701_remove(struct platform_device *pdev) -{ - struct snd_soc_card *card = platform_get_drvdata(pdev); - - snd_soc_unregister_card(card); - - return 0; -} - -static struct platform_driver bfin_eval_adau1701_driver = { - .driver = { - .name = "bfin-eval-adau1701", - .owner = THIS_MODULE, - .pm = &snd_soc_pm_ops, - }, - .probe = bfin_eval_adau1701_probe, - .remove = __devexit_p(bfin_eval_adau1701_remove), -}; - -static int __init bfin_eval_adau1701_init(void) -{ - return platform_driver_register(&bfin_eval_adau1701_driver); -} -module_init(bfin_eval_adau1701_init); - -static void __exit bfin_eval_adau1701_exit(void) -{ - platform_driver_unregister(&bfin_eval_adau1701_driver); -} -module_exit(bfin_eval_adau1701_exit); - -MODULE_AUTHOR("Lars-Peter Clausen "); -MODULE_DESCRIPTION("ALSA SoC bfin ADAU1701 driver"); -MODULE_LICENSE("GPL"); -MODULE_ALIAS("platform:bfin-eval-adau1701"); diff --git a/trunk/sound/soc/blackfin/bfin-eval-adav80x.c b/trunk/sound/soc/blackfin/bfin-eval-adav80x.c deleted file mode 100644 index 8d014d01626e..000000000000 --- a/trunk/sound/soc/blackfin/bfin-eval-adav80x.c +++ /dev/null @@ -1,173 +0,0 @@ -/* - * Machine driver for EVAL-ADAV801 and EVAL-ADAV803 on Analog Devices bfin - * evaluation boards. - * - * Copyright 2011 Analog Devices Inc. - * Author: Lars-Peter Clausen - * - * Licensed under the GPL-2 or later. - */ - -#include -#include -#include -#include -#include - -#include "../codecs/adav80x.h" - -static const struct snd_soc_dapm_widget bfin_eval_adav80x_dapm_widgets[] = { - SND_SOC_DAPM_LINE("Line Out", NULL), - SND_SOC_DAPM_LINE("Line In", NULL), -}; - -static const struct snd_soc_dapm_route bfin_eval_adav80x_dapm_routes[] = { - { "Line Out", NULL, "VOUTL" }, - { "Line Out", NULL, "VOUTR" }, - - { "VINL", NULL, "Line In" }, - { "VINR", NULL, "Line In" }, -}; - -static int bfin_eval_adav80x_hw_params(struct snd_pcm_substream *substream, - struct snd_pcm_hw_params *params) -{ - struct snd_soc_pcm_runtime *rtd = substream->private_data; - struct snd_soc_dai *cpu_dai = rtd->cpu_dai; - struct snd_soc_dai *codec_dai = rtd->codec_dai; - int ret; - - ret = snd_soc_dai_set_fmt(cpu_dai, SND_SOC_DAIFMT_I2S | - SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBM_CFM); - if (ret) - return ret; - - ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_I2S | - SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBM_CFM); - if (ret) - return ret; - - ret = snd_soc_dai_set_pll(codec_dai, ADAV80X_PLL1, ADAV80X_PLL_SRC_XTAL, - 27000000, params_rate(params) * 256); - if (ret) - return ret; - - ret = snd_soc_dai_set_sysclk(codec_dai, ADAV80X_CLK_PLL1, - params_rate(params) * 256, SND_SOC_CLOCK_IN); - - return ret; -} - -static int bfin_eval_adav80x_codec_init(struct snd_soc_pcm_runtime *rtd) -{ - struct snd_soc_dai *codec_dai = rtd->codec_dai; - - snd_soc_dai_set_sysclk(codec_dai, ADAV80X_CLK_SYSCLK1, 0, - SND_SOC_CLOCK_OUT); - snd_soc_dai_set_sysclk(codec_dai, ADAV80X_CLK_SYSCLK2, 0, - SND_SOC_CLOCK_OUT); - snd_soc_dai_set_sysclk(codec_dai, ADAV80X_CLK_SYSCLK3, 0, - SND_SOC_CLOCK_OUT); - - snd_soc_dai_set_sysclk(codec_dai, ADAV80X_CLK_XTAL, 2700000, 0); - - return 0; -} - -static struct snd_soc_ops bfin_eval_adav80x_ops = { - .hw_params = bfin_eval_adav80x_hw_params, -}; - -static struct snd_soc_dai_link bfin_eval_adav80x_dais[] = { - { - .name = "adav80x", - .stream_name = "ADAV80x HiFi", - .cpu_dai_name = "bfin-i2s.0", - .codec_dai_name = "adav80x-hifi", - .platform_name = "bfin-i2s-pcm-audio", - .init = bfin_eval_adav80x_codec_init, - .ops = &bfin_eval_adav80x_ops, - }, -}; - -static struct snd_soc_card bfin_eval_adav80x = { - .name = "bfin-eval-adav80x", - .dai_link = bfin_eval_adav80x_dais, - .num_links = ARRAY_SIZE(bfin_eval_adav80x_dais), - - .dapm_widgets = bfin_eval_adav80x_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(bfin_eval_adav80x_dapm_widgets), - .dapm_routes = bfin_eval_adav80x_dapm_routes, - .num_dapm_routes = ARRAY_SIZE(bfin_eval_adav80x_dapm_routes), -}; - -enum bfin_eval_adav80x_type { - BFIN_EVAL_ADAV801, - BFIN_EVAL_ADAV803, -}; - -static int bfin_eval_adav80x_probe(struct platform_device *pdev) -{ - struct snd_soc_card *card = &bfin_eval_adav80x; - const char *codec_name; - - switch (platform_get_device_id(pdev)->driver_data) { - case BFIN_EVAL_ADAV801: - codec_name = "spi0.1"; - break; - case BFIN_EVAL_ADAV803: - codec_name = "adav803.0-0034"; - break; - default: - return -EINVAL; - } - - bfin_eval_adav80x_dais[0].codec_name = codec_name; - - card->dev = &pdev->dev; - - return snd_soc_register_card(&bfin_eval_adav80x); -} - -static int __devexit bfin_eval_adav80x_remove(struct platform_device *pdev) -{ - struct snd_soc_card *card = platform_get_drvdata(pdev); - - snd_soc_unregister_card(card); - - return 0; -} - -static const struct platform_device_id bfin_eval_adav80x_ids[] = { - { "bfin-eval-adav801", BFIN_EVAL_ADAV801 }, - { "bfin-eval-adav803", BFIN_EVAL_ADAV803 }, - { }, -}; -MODULE_DEVICE_TABLE(platform, bfin_eval_adav80x_ids); - -static struct platform_driver bfin_eval_adav80x_driver = { - .driver = { - .name = "bfin-eval-adav80x", - .owner = THIS_MODULE, - .pm = &snd_soc_pm_ops, - }, - .probe = bfin_eval_adav80x_probe, - .remove = __devexit_p(bfin_eval_adav80x_remove), - .id_table = bfin_eval_adav80x_ids, -}; - -static int __init bfin_eval_adav80x_init(void) -{ - return platform_driver_register(&bfin_eval_adav80x_driver); -} -module_init(bfin_eval_adav80x_init); - -static void __exit bfin_eval_adav80x_exit(void) -{ - platform_driver_unregister(&bfin_eval_adav80x_driver); -} -module_exit(bfin_eval_adav80x_exit); - -MODULE_AUTHOR("Lars-Peter Clausen "); -MODULE_DESCRIPTION("ALSA SoC bfin adav80x driver"); -MODULE_LICENSE("GPL"); diff --git a/trunk/sound/soc/codecs/Kconfig b/trunk/sound/soc/codecs/Kconfig index 36a030f1d1f5..98175a096df2 100644 --- a/trunk/sound/soc/codecs/Kconfig +++ b/trunk/sound/soc/codecs/Kconfig @@ -17,7 +17,6 @@ config SND_SOC_ALL_CODECS select SND_SOC_AD193X if SND_SOC_I2C_AND_SPI select SND_SOC_AD1980 if SND_SOC_AC97_BUS select SND_SOC_AD73311 - select SND_SOC_ADAV80X select SND_SOC_ADS117X select SND_SOC_AK4104 if SPI_MASTER select SND_SOC_AK4535 if I2C @@ -43,7 +42,6 @@ config SND_SOC_ALL_CODECS select SND_SOC_SN95031 if INTEL_SCU_IPC select SND_SOC_SPDIF select SND_SOC_SSM2602 if SND_SOC_I2C_AND_SPI - select SND_SOC_STA32X if I2C select SND_SOC_STAC9766 if SND_SOC_AC97_BUS select SND_SOC_TLV320AIC23 if I2C select SND_SOC_TLV320AIC26 if SPI_MASTER @@ -73,7 +71,6 @@ config SND_SOC_ALL_CODECS select SND_SOC_WM8753 if SND_SOC_I2C_AND_SPI select SND_SOC_WM8770 if SPI_MASTER select SND_SOC_WM8776 if SND_SOC_I2C_AND_SPI - select SND_SOC_WM8782 select SND_SOC_WM8804 if SND_SOC_I2C_AND_SPI select SND_SOC_WM8900 if I2C select SND_SOC_WM8903 if I2C @@ -87,7 +84,6 @@ config SND_SOC_ALL_CODECS select SND_SOC_WM8971 if I2C select SND_SOC_WM8974 if I2C select SND_SOC_WM8978 if I2C - select SND_SOC_WM8983 if SND_SOC_I2C_AND_SPI select SND_SOC_WM8985 if SND_SOC_I2C_AND_SPI select SND_SOC_WM8988 if SND_SOC_I2C_AND_SPI select SND_SOC_WM8990 if I2C @@ -134,14 +130,7 @@ config SND_SOC_AD1980 config SND_SOC_AD73311 tristate - -config SND_SOC_ADAU1701 - select SIGMA - tristate - -config SND_SOC_ADAV80X - tristate - + config SND_SOC_ADS117X tristate @@ -227,9 +216,6 @@ config SND_SOC_SPDIF config SND_SOC_SSM2602 tristate -config SND_SOC_STA32X - tristate - config SND_SOC_STAC9766 tristate @@ -313,9 +299,6 @@ config SND_SOC_WM8770 config SND_SOC_WM8776 tristate -config SND_SOC_WM8782 - tristate - config SND_SOC_WM8804 tristate @@ -355,9 +338,6 @@ config SND_SOC_WM8974 config SND_SOC_WM8978 tristate -config SND_SOC_WM8983 - tristate - config SND_SOC_WM8985 tristate diff --git a/trunk/sound/soc/codecs/Makefile b/trunk/sound/soc/codecs/Makefile index da9990fb8569..fd8558406ef0 100644 --- a/trunk/sound/soc/codecs/Makefile +++ b/trunk/sound/soc/codecs/Makefile @@ -4,8 +4,6 @@ snd-soc-ad1836-objs := ad1836.o snd-soc-ad193x-objs := ad193x.o snd-soc-ad1980-objs := ad1980.o snd-soc-ad73311-objs := ad73311.o -snd-soc-adau1701-objs := adau1701.o -snd-soc-adav80x-objs := adav80x.o snd-soc-ads117x-objs := ads117x.o snd-soc-ak4104-objs := ak4104.o snd-soc-ak4535-objs := ak4535.o @@ -30,7 +28,6 @@ snd-soc-alc5623-objs := alc5623.o snd-soc-sn95031-objs := sn95031.o snd-soc-spdif-objs := spdif_transciever.o snd-soc-ssm2602-objs := ssm2602.o -snd-soc-sta32x-objs := sta32x.o snd-soc-stac9766-objs := stac9766.o snd-soc-tlv320aic23-objs := tlv320aic23.o snd-soc-tlv320aic26-objs := tlv320aic26.o @@ -58,7 +55,6 @@ snd-soc-wm8750-objs := wm8750.o snd-soc-wm8753-objs := wm8753.o snd-soc-wm8770-objs := wm8770.o snd-soc-wm8776-objs := wm8776.o -snd-soc-wm8782-objs := wm8782.o snd-soc-wm8804-objs := wm8804.o snd-soc-wm8900-objs := wm8900.o snd-soc-wm8903-objs := wm8903.o @@ -72,7 +68,6 @@ snd-soc-wm8962-objs := wm8962.o snd-soc-wm8971-objs := wm8971.o snd-soc-wm8974-objs := wm8974.o snd-soc-wm8978-objs := wm8978.o -snd-soc-wm8983-objs := wm8983.o snd-soc-wm8985-objs := wm8985.o snd-soc-wm8988-objs := wm8988.o snd-soc-wm8990-objs := wm8990.o @@ -100,8 +95,6 @@ obj-$(CONFIG_SND_SOC_AD1836) += snd-soc-ad1836.o obj-$(CONFIG_SND_SOC_AD193X) += snd-soc-ad193x.o obj-$(CONFIG_SND_SOC_AD1980) += snd-soc-ad1980.o obj-$(CONFIG_SND_SOC_AD73311) += snd-soc-ad73311.o -obj-$(CONFIG_SND_SOC_ADAU1701) += snd-soc-adau1701.o -obj-$(CONFIG_SND_SOC_ADAV80X) += snd-soc-adav80x.o obj-$(CONFIG_SND_SOC_ADS117X) += snd-soc-ads117x.o obj-$(CONFIG_SND_SOC_AK4104) += snd-soc-ak4104.o obj-$(CONFIG_SND_SOC_AK4535) += snd-soc-ak4535.o @@ -127,7 +120,6 @@ obj-$(CONFIG_SND_SOC_SGTL5000) += snd-soc-sgtl5000.o obj-$(CONFIG_SND_SOC_SN95031) +=snd-soc-sn95031.o obj-$(CONFIG_SND_SOC_SPDIF) += snd-soc-spdif.o obj-$(CONFIG_SND_SOC_SSM2602) += snd-soc-ssm2602.o -obj-$(CONFIG_SND_SOC_STA32X) += snd-soc-sta32x.o obj-$(CONFIG_SND_SOC_STAC9766) += snd-soc-stac9766.o obj-$(CONFIG_SND_SOC_TLV320AIC23) += snd-soc-tlv320aic23.o obj-$(CONFIG_SND_SOC_TLV320AIC26) += snd-soc-tlv320aic26.o @@ -155,7 +147,6 @@ obj-$(CONFIG_SND_SOC_WM8750) += snd-soc-wm8750.o obj-$(CONFIG_SND_SOC_WM8753) += snd-soc-wm8753.o obj-$(CONFIG_SND_SOC_WM8770) += snd-soc-wm8770.o obj-$(CONFIG_SND_SOC_WM8776) += snd-soc-wm8776.o -obj-$(CONFIG_SND_SOC_WM8782) += snd-soc-wm8782.o obj-$(CONFIG_SND_SOC_WM8804) += snd-soc-wm8804.o obj-$(CONFIG_SND_SOC_WM8900) += snd-soc-wm8900.o obj-$(CONFIG_SND_SOC_WM8903) += snd-soc-wm8903.o @@ -169,7 +160,6 @@ obj-$(CONFIG_SND_SOC_WM8962) += snd-soc-wm8962.o obj-$(CONFIG_SND_SOC_WM8971) += snd-soc-wm8971.o obj-$(CONFIG_SND_SOC_WM8974) += snd-soc-wm8974.o obj-$(CONFIG_SND_SOC_WM8978) += snd-soc-wm8978.o -obj-$(CONFIG_SND_SOC_WM8983) += snd-soc-wm8983.o obj-$(CONFIG_SND_SOC_WM8985) += snd-soc-wm8985.o obj-$(CONFIG_SND_SOC_WM8988) += snd-soc-wm8988.o obj-$(CONFIG_SND_SOC_WM8990) += snd-soc-wm8990.o diff --git a/trunk/sound/soc/codecs/ad1836.c b/trunk/sound/soc/codecs/ad1836.c index 4e5c5726366b..754c496412bd 100644 --- a/trunk/sound/soc/codecs/ad1836.c +++ b/trunk/sound/soc/codecs/ad1836.c @@ -1,10 +1,19 @@ - /* - * Audio Codec driver supporting: - * AD1835A, AD1836, AD1837A, AD1838A, AD1839A +/* + * File: sound/soc/codecs/ad1836.c + * Author: Barry Song + * + * Created: Aug 04 2009 + * Description: Driver for AD1836 sound chip + * + * Modified: + * Copyright 2009 Analog Devices Inc. * - * Copyright 2009-2011 Analog Devices Inc. + * Bugs: Enter bugs at http://blackfin.uclinux.org/ * - * Licensed under the GPL-2 or later. + * 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. */ #include @@ -21,15 +30,10 @@ #include #include "ad1836.h" -enum ad1836_type { - AD1835, - AD1836, - AD1838, -}; - /* codec private data */ struct ad1836_priv { - enum ad1836_type type; + enum snd_soc_control_type control_type; + void *control_data; }; /* @@ -40,60 +44,29 @@ static const char *ad1836_deemp[] = {"None", "44.1kHz", "32kHz", "48kHz"}; static const struct soc_enum ad1836_deemp_enum = SOC_ENUM_SINGLE(AD1836_DAC_CTRL1, 8, 4, ad1836_deemp); -#define AD1836_DAC_VOLUME(x) \ - SOC_DOUBLE_R("DAC" #x " Playback Volume", AD1836_DAC_L_VOL(x), \ - AD1836_DAC_R_VOL(x), 0, 0x3FF, 0) - -#define AD1836_DAC_SWITCH(x) \ - SOC_DOUBLE("DAC" #x " Playback Switch", AD1836_DAC_CTRL2, \ - AD1836_MUTE_LEFT(x), AD1836_MUTE_RIGHT(x), 1, 1) - -#define AD1836_ADC_SWITCH(x) \ - SOC_DOUBLE("ADC" #x " Capture Switch", AD1836_ADC_CTRL2, \ - AD1836_MUTE_LEFT(x), AD1836_MUTE_RIGHT(x), 1, 1) - -static const struct snd_kcontrol_new ad183x_dac_controls[] = { - AD1836_DAC_VOLUME(1), - AD1836_DAC_SWITCH(1), - AD1836_DAC_VOLUME(2), - AD1836_DAC_SWITCH(2), - AD1836_DAC_VOLUME(3), - AD1836_DAC_SWITCH(3), - AD1836_DAC_VOLUME(4), - AD1836_DAC_SWITCH(4), -}; - -static const struct snd_soc_dapm_widget ad183x_dac_dapm_widgets[] = { - SND_SOC_DAPM_OUTPUT("DAC1OUT"), - SND_SOC_DAPM_OUTPUT("DAC2OUT"), - SND_SOC_DAPM_OUTPUT("DAC3OUT"), - SND_SOC_DAPM_OUTPUT("DAC4OUT"), -}; - -static const struct snd_soc_dapm_route ad183x_dac_routes[] = { - { "DAC1OUT", NULL, "DAC" }, - { "DAC2OUT", NULL, "DAC" }, - { "DAC3OUT", NULL, "DAC" }, - { "DAC4OUT", NULL, "DAC" }, -}; - -static const struct snd_kcontrol_new ad183x_adc_controls[] = { - AD1836_ADC_SWITCH(1), - AD1836_ADC_SWITCH(2), - AD1836_ADC_SWITCH(3), -}; - -static const struct snd_soc_dapm_widget ad183x_adc_dapm_widgets[] = { - SND_SOC_DAPM_INPUT("ADC1IN"), - SND_SOC_DAPM_INPUT("ADC2IN"), -}; - -static const struct snd_soc_dapm_route ad183x_adc_routes[] = { - { "ADC", NULL, "ADC1IN" }, - { "ADC", NULL, "ADC2IN" }, -}; +static const struct snd_kcontrol_new ad1836_snd_controls[] = { + /* DAC volume control */ + SOC_DOUBLE_R("DAC1 Volume", AD1836_DAC_L1_VOL, + AD1836_DAC_R1_VOL, 0, 0x3FF, 0), + SOC_DOUBLE_R("DAC2 Volume", AD1836_DAC_L2_VOL, + AD1836_DAC_R2_VOL, 0, 0x3FF, 0), + SOC_DOUBLE_R("DAC3 Volume", AD1836_DAC_L3_VOL, + AD1836_DAC_R3_VOL, 0, 0x3FF, 0), + + /* ADC switch control */ + SOC_DOUBLE("ADC1 Switch", AD1836_ADC_CTRL2, AD1836_ADCL1_MUTE, + AD1836_ADCR1_MUTE, 1, 1), + SOC_DOUBLE("ADC2 Switch", AD1836_ADC_CTRL2, AD1836_ADCL2_MUTE, + AD1836_ADCR2_MUTE, 1, 1), + + /* DAC switch control */ + SOC_DOUBLE("DAC1 Switch", AD1836_DAC_CTRL2, AD1836_DACL1_MUTE, + AD1836_DACR1_MUTE, 1, 1), + SOC_DOUBLE("DAC2 Switch", AD1836_DAC_CTRL2, AD1836_DACL2_MUTE, + AD1836_DACR2_MUTE, 1, 1), + SOC_DOUBLE("DAC3 Switch", AD1836_DAC_CTRL2, AD1836_DACL3_MUTE, + AD1836_DACR3_MUTE, 1, 1), -static const struct snd_kcontrol_new ad183x_controls[] = { /* ADC high-pass filter */ SOC_SINGLE("ADC High Pass Filter Switch", AD1836_ADC_CTRL1, AD1836_ADC_HIGHPASS_FILTER, 1, 0), @@ -102,24 +75,27 @@ static const struct snd_kcontrol_new ad183x_controls[] = { SOC_ENUM("Playback Deemphasis", ad1836_deemp_enum), }; -static const struct snd_soc_dapm_widget ad183x_dapm_widgets[] = { +static const struct snd_soc_dapm_widget ad1836_dapm_widgets[] = { SND_SOC_DAPM_DAC("DAC", "Playback", AD1836_DAC_CTRL1, AD1836_DAC_POWERDOWN, 1), SND_SOC_DAPM_ADC("ADC", "Capture", SND_SOC_NOPM, 0, 0), SND_SOC_DAPM_SUPPLY("ADC_PWR", AD1836_ADC_CTRL1, AD1836_ADC_POWERDOWN, 1, NULL, 0), + SND_SOC_DAPM_OUTPUT("DAC1OUT"), + SND_SOC_DAPM_OUTPUT("DAC2OUT"), + SND_SOC_DAPM_OUTPUT("DAC3OUT"), + SND_SOC_DAPM_INPUT("ADC1IN"), + SND_SOC_DAPM_INPUT("ADC2IN"), }; -static const struct snd_soc_dapm_route ad183x_dapm_routes[] = { +static const struct snd_soc_dapm_route audio_paths[] = { { "DAC", NULL, "ADC_PWR" }, { "ADC", NULL, "ADC_PWR" }, -}; - -static const DECLARE_TLV_DB_SCALE(ad1836_in_tlv, 0, 300, 0); - -static const struct snd_kcontrol_new ad1836_controls[] = { - SOC_DOUBLE_TLV("ADC2 Capture Volume", AD1836_ADC_CTRL1, 3, 0, 4, 0, - ad1836_in_tlv), + { "DAC1OUT", "DAC1 Switch", "DAC" }, + { "DAC2OUT", "DAC2 Switch", "DAC" }, + { "DAC3OUT", "DAC3 Switch", "DAC" }, + { "ADC", "ADC1 Switch", "ADC1IN" }, + { "ADC", "ADC2 Switch", "ADC2IN" }, }; /* @@ -189,69 +165,64 @@ static int ad1836_hw_params(struct snd_pcm_substream *substream, return 0; } -static struct snd_soc_dai_ops ad1836_dai_ops = { - .hw_params = ad1836_hw_params, - .set_fmt = ad1836_set_dai_fmt, -}; - -#define AD183X_DAI(_name, num_dacs, num_adcs) \ -{ \ - .name = _name "-hifi", \ - .playback = { \ - .stream_name = "Playback", \ - .channels_min = 2, \ - .channels_max = (num_dacs) * 2, \ - .rates = SNDRV_PCM_RATE_48000, \ - .formats = SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S16_LE | \ - SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE, \ - }, \ - .capture = { \ - .stream_name = "Capture", \ - .channels_min = 2, \ - .channels_max = (num_adcs) * 2, \ - .rates = SNDRV_PCM_RATE_48000, \ - .formats = SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S16_LE | \ - SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE, \ - }, \ - .ops = &ad1836_dai_ops, \ -} - -static struct snd_soc_dai_driver ad183x_dais[] = { - [AD1835] = AD183X_DAI("ad1835", 4, 1), - [AD1836] = AD183X_DAI("ad1836", 3, 2), - [AD1838] = AD183X_DAI("ad1838", 3, 1), -}; - #ifdef CONFIG_PM -static int ad1836_suspend(struct snd_soc_codec *codec, pm_message_t state) +static int ad1836_soc_suspend(struct snd_soc_codec *codec, + pm_message_t state) { /* reset clock control mode */ - return snd_soc_update_bits(codec, AD1836_ADC_CTRL2, - AD1836_ADC_SERFMT_MASK, 0); + u16 adc_ctrl2 = snd_soc_read(codec, AD1836_ADC_CTRL2); + adc_ctrl2 &= ~AD1836_ADC_SERFMT_MASK; + + return snd_soc_write(codec, AD1836_ADC_CTRL2, adc_ctrl2); } -static int ad1836_resume(struct snd_soc_codec *codec) +static int ad1836_soc_resume(struct snd_soc_codec *codec) { /* restore clock control mode */ - return snd_soc_update_bits(codec, AD1836_ADC_CTRL2, - AD1836_ADC_SERFMT_MASK, AD1836_ADC_AUX); + u16 adc_ctrl2 = snd_soc_read(codec, AD1836_ADC_CTRL2); + adc_ctrl2 |= AD1836_ADC_AUX; + + return snd_soc_write(codec, AD1836_ADC_CTRL2, adc_ctrl2); } #else -#define ad1836_suspend NULL -#define ad1836_resume NULL +#define ad1836_soc_suspend NULL +#define ad1836_soc_resume NULL #endif +static struct snd_soc_dai_ops ad1836_dai_ops = { + .hw_params = ad1836_hw_params, + .set_fmt = ad1836_set_dai_fmt, +}; + +/* codec DAI instance */ +static struct snd_soc_dai_driver ad1836_dai = { + .name = "ad1836-hifi", + .playback = { + .stream_name = "Playback", + .channels_min = 2, + .channels_max = 6, + .rates = SNDRV_PCM_RATE_48000, + .formats = SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S16_LE | + SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE, + }, + .capture = { + .stream_name = "Capture", + .channels_min = 2, + .channels_max = 4, + .rates = SNDRV_PCM_RATE_48000, + .formats = SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S16_LE | + SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE, + }, + .ops = &ad1836_dai_ops, +}; + static int ad1836_probe(struct snd_soc_codec *codec) { struct ad1836_priv *ad1836 = snd_soc_codec_get_drvdata(codec); struct snd_soc_dapm_context *dapm = &codec->dapm; - int num_dacs, num_adcs; int ret = 0; - int i; - - num_dacs = ad183x_dais[ad1836->type].playback.channels_max / 2; - num_adcs = ad183x_dais[ad1836->type].capture.channels_max / 2; + codec->control_data = ad1836->control_data; ret = snd_soc_codec_set_cache_io(codec, 4, 12, SND_SOC_SPI); if (ret < 0) { dev_err(codec->dev, "failed to set cache I/O: %d\n", @@ -268,46 +239,21 @@ static int ad1836_probe(struct snd_soc_codec *codec) snd_soc_write(codec, AD1836_ADC_CTRL1, 0x100); /* unmute adc channles, adc aux mode */ snd_soc_write(codec, AD1836_ADC_CTRL2, 0x180); + /* left/right diff:PGA/MUX */ + snd_soc_write(codec, AD1836_ADC_CTRL3, 0x3A); /* volume */ - for (i = 1; i <= num_dacs; ++i) { - snd_soc_write(codec, AD1836_DAC_L_VOL(i), 0x3FF); - snd_soc_write(codec, AD1836_DAC_R_VOL(i), 0x3FF); - } - - if (ad1836->type == AD1836) { - /* left/right diff:PGA/MUX */ - snd_soc_write(codec, AD1836_ADC_CTRL3, 0x3A); - ret = snd_soc_add_controls(codec, ad1836_controls, - ARRAY_SIZE(ad1836_controls)); - if (ret) - return ret; - } else { - snd_soc_write(codec, AD1836_ADC_CTRL3, 0x00); - } - - ret = snd_soc_add_controls(codec, ad183x_dac_controls, num_dacs * 2); - if (ret) - return ret; - - ret = snd_soc_add_controls(codec, ad183x_adc_controls, num_adcs); - if (ret) - return ret; - - ret = snd_soc_dapm_new_controls(dapm, ad183x_dac_dapm_widgets, num_dacs); - if (ret) - return ret; - - ret = snd_soc_dapm_new_controls(dapm, ad183x_adc_dapm_widgets, num_adcs); - if (ret) - return ret; - - ret = snd_soc_dapm_add_routes(dapm, ad183x_dac_routes, num_dacs); - if (ret) - return ret; - - ret = snd_soc_dapm_add_routes(dapm, ad183x_adc_routes, num_adcs); - if (ret) - return ret; + snd_soc_write(codec, AD1836_DAC_L1_VOL, 0x3FF); + snd_soc_write(codec, AD1836_DAC_R1_VOL, 0x3FF); + snd_soc_write(codec, AD1836_DAC_L2_VOL, 0x3FF); + snd_soc_write(codec, AD1836_DAC_R2_VOL, 0x3FF); + snd_soc_write(codec, AD1836_DAC_L3_VOL, 0x3FF); + snd_soc_write(codec, AD1836_DAC_R3_VOL, 0x3FF); + + snd_soc_add_controls(codec, ad1836_snd_controls, + ARRAY_SIZE(ad1836_snd_controls)); + snd_soc_dapm_new_controls(dapm, ad1836_dapm_widgets, + ARRAY_SIZE(ad1836_dapm_widgets)); + snd_soc_dapm_add_routes(dapm, audio_paths, ARRAY_SIZE(audio_paths)); return ret; } @@ -316,24 +262,19 @@ static int ad1836_probe(struct snd_soc_codec *codec) static int ad1836_remove(struct snd_soc_codec *codec) { /* reset clock control mode */ - return snd_soc_update_bits(codec, AD1836_ADC_CTRL2, - AD1836_ADC_SERFMT_MASK, 0); + u16 adc_ctrl2 = snd_soc_read(codec, AD1836_ADC_CTRL2); + adc_ctrl2 &= ~AD1836_ADC_SERFMT_MASK; + + return snd_soc_write(codec, AD1836_ADC_CTRL2, adc_ctrl2); } static struct snd_soc_codec_driver soc_codec_dev_ad1836 = { - .probe = ad1836_probe, - .remove = ad1836_remove, - .suspend = ad1836_suspend, - .resume = ad1836_resume, + .probe = ad1836_probe, + .remove = ad1836_remove, + .suspend = ad1836_soc_suspend, + .resume = ad1836_soc_resume, .reg_cache_size = AD1836_NUM_REGS, .reg_word_size = sizeof(u16), - - .controls = ad183x_controls, - .num_controls = ARRAY_SIZE(ad183x_controls), - .dapm_widgets = ad183x_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(ad183x_dapm_widgets), - .dapm_routes = ad183x_dapm_routes, - .num_dapm_routes = ARRAY_SIZE(ad183x_dapm_routes), }; static int __devinit ad1836_spi_probe(struct spi_device *spi) @@ -345,12 +286,12 @@ static int __devinit ad1836_spi_probe(struct spi_device *spi) if (ad1836 == NULL) return -ENOMEM; - ad1836->type = spi_get_device_id(spi)->driver_data; - spi_set_drvdata(spi, ad1836); + ad1836->control_data = spi; + ad1836->control_type = SND_SOC_SPI; ret = snd_soc_register_codec(&spi->dev, - &soc_codec_dev_ad1836, &ad183x_dais[ad1836->type], 1); + &soc_codec_dev_ad1836, &ad1836_dai, 1); if (ret < 0) kfree(ad1836); return ret; @@ -362,29 +303,27 @@ static int __devexit ad1836_spi_remove(struct spi_device *spi) kfree(spi_get_drvdata(spi)); return 0; } -static const struct spi_device_id ad1836_ids[] = { - { "ad1835", AD1835 }, - { "ad1836", AD1836 }, - { "ad1837", AD1835 }, - { "ad1838", AD1838 }, - { "ad1839", AD1838 }, - { }, -}; -MODULE_DEVICE_TABLE(spi, ad1836_ids); static struct spi_driver ad1836_spi_driver = { .driver = { - .name = "ad1836", + .name = "ad1836-codec", .owner = THIS_MODULE, }, .probe = ad1836_spi_probe, .remove = __devexit_p(ad1836_spi_remove), - .id_table = ad1836_ids, }; static int __init ad1836_init(void) { - return spi_register_driver(&ad1836_spi_driver); + int ret; + + ret = spi_register_driver(&ad1836_spi_driver); + if (ret != 0) { + printk(KERN_ERR "Failed to register ad1836 SPI driver: %d\n", + ret); + } + + return ret; } module_init(ad1836_init); diff --git a/trunk/sound/soc/codecs/ad1836.h b/trunk/sound/soc/codecs/ad1836.h index 444747f0db26..9d6a3f8f8aaf 100644 --- a/trunk/sound/soc/codecs/ad1836.h +++ b/trunk/sound/soc/codecs/ad1836.h @@ -1,10 +1,19 @@ /* - * Audio Codec driver supporting: - * AD1835A, AD1836, AD1837A, AD1838A, AD1839A + * File: sound/soc/codecs/ad1836.h + * Based on: + * Author: Barry Song * - * Copyright 2009-2011 Analog Devices Inc. + * Created: Aug 04, 2009 + * Description: definitions for AD1836 registers * - * Licensed under the GPL-2 or later. + * Modified: + * + * Bugs: Enter bugs at http://blackfin.uclinux.org/ + * + * 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. */ #ifndef __AD1836_H__ @@ -12,30 +21,39 @@ #define AD1836_DAC_CTRL1 0 #define AD1836_DAC_POWERDOWN 2 -#define AD1836_DAC_SERFMT_MASK 0xE0 +#define AD1836_DAC_SERFMT_MASK 0xE0 #define AD1836_DAC_SERFMT_PCK256 (0x4 << 5) #define AD1836_DAC_SERFMT_PCK128 (0x5 << 5) #define AD1836_DAC_WORD_LEN_MASK 0x18 #define AD1836_DAC_WORD_LEN_OFFSET 3 #define AD1836_DAC_CTRL2 1 +#define AD1836_DACL1_MUTE 0 +#define AD1836_DACR1_MUTE 1 +#define AD1836_DACL2_MUTE 2 +#define AD1836_DACR2_MUTE 3 +#define AD1836_DACL3_MUTE 4 +#define AD1836_DACR3_MUTE 5 -/* These macros are one-based. So AD183X_MUTE_LEFT(1) will return the mute bit - * for the first ADC/DAC */ -#define AD1836_MUTE_LEFT(x) (((x) * 2) - 2) -#define AD1836_MUTE_RIGHT(x) (((x) * 2) - 1) - -#define AD1836_DAC_L_VOL(x) ((x) * 2) -#define AD1836_DAC_R_VOL(x) (1 + ((x) * 2)) +#define AD1836_DAC_L1_VOL 2 +#define AD1836_DAC_R1_VOL 3 +#define AD1836_DAC_L2_VOL 4 +#define AD1836_DAC_R2_VOL 5 +#define AD1836_DAC_L3_VOL 6 +#define AD1836_DAC_R3_VOL 7 #define AD1836_ADC_CTRL1 12 #define AD1836_ADC_POWERDOWN 7 #define AD1836_ADC_HIGHPASS_FILTER 8 #define AD1836_ADC_CTRL2 13 +#define AD1836_ADCL1_MUTE 0 +#define AD1836_ADCR1_MUTE 1 +#define AD1836_ADCL2_MUTE 2 +#define AD1836_ADCR2_MUTE 3 #define AD1836_ADC_WORD_LEN_MASK 0x30 #define AD1836_ADC_WORD_OFFSET 5 -#define AD1836_ADC_SERFMT_MASK (7 << 6) +#define AD1836_ADC_SERFMT_MASK (7 << 6) #define AD1836_ADC_SERFMT_PCK256 (0x4 << 6) #define AD1836_ADC_SERFMT_PCK128 (0x5 << 6) #define AD1836_ADC_AUX (0x6 << 6) diff --git a/trunk/sound/soc/codecs/adau1701.c b/trunk/sound/soc/codecs/adau1701.c deleted file mode 100644 index 2758d5fc60d6..000000000000 --- a/trunk/sound/soc/codecs/adau1701.c +++ /dev/null @@ -1,549 +0,0 @@ -/* - * Driver for ADAU1701 SigmaDSP processor - * - * Copyright 2011 Analog Devices Inc. - * Author: Lars-Peter Clausen - * based on an inital version by Cliff Cai - * - * Licensed under the GPL-2 or later. - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "adau1701.h" - -#define ADAU1701_DSPCTRL 0x1c -#define ADAU1701_SEROCTL 0x1e -#define ADAU1701_SERICTL 0x1f - -#define ADAU1701_AUXNPOW 0x22 - -#define ADAU1701_OSCIPOW 0x26 -#define ADAU1701_DACSET 0x27 - -#define ADAU1701_NUM_REGS 0x28 - -#define ADAU1701_DSPCTRL_CR (1 << 2) -#define ADAU1701_DSPCTRL_DAM (1 << 3) -#define ADAU1701_DSPCTRL_ADM (1 << 4) -#define ADAU1701_DSPCTRL_SR_48 0x00 -#define ADAU1701_DSPCTRL_SR_96 0x01 -#define ADAU1701_DSPCTRL_SR_192 0x02 -#define ADAU1701_DSPCTRL_SR_MASK 0x03 - -#define ADAU1701_SEROCTL_INV_LRCLK 0x2000 -#define ADAU1701_SEROCTL_INV_BCLK 0x1000 -#define ADAU1701_SEROCTL_MASTER 0x0800 - -#define ADAU1701_SEROCTL_OBF16 0x0000 -#define ADAU1701_SEROCTL_OBF8 0x0200 -#define ADAU1701_SEROCTL_OBF4 0x0400 -#define ADAU1701_SEROCTL_OBF2 0x0600 -#define ADAU1701_SEROCTL_OBF_MASK 0x0600 - -#define ADAU1701_SEROCTL_OLF1024 0x0000 -#define ADAU1701_SEROCTL_OLF512 0x0080 -#define ADAU1701_SEROCTL_OLF256 0x0100 -#define ADAU1701_SEROCTL_OLF_MASK 0x0180 - -#define ADAU1701_SEROCTL_MSB_DEALY1 0x0000 -#define ADAU1701_SEROCTL_MSB_DEALY0 0x0004 -#define ADAU1701_SEROCTL_MSB_DEALY8 0x0008 -#define ADAU1701_SEROCTL_MSB_DEALY12 0x000c -#define ADAU1701_SEROCTL_MSB_DEALY16 0x0010 -#define ADAU1701_SEROCTL_MSB_DEALY_MASK 0x001c - -#define ADAU1701_SEROCTL_WORD_LEN_24 0x0000 -#define ADAU1701_SEROCTL_WORD_LEN_20 0x0001 -#define ADAU1701_SEROCTL_WORD_LEN_16 0x0010 -#define ADAU1701_SEROCTL_WORD_LEN_MASK 0x0003 - -#define ADAU1701_AUXNPOW_VBPD 0x40 -#define ADAU1701_AUXNPOW_VRPD 0x20 - -#define ADAU1701_SERICTL_I2S 0 -#define ADAU1701_SERICTL_LEFTJ 1 -#define ADAU1701_SERICTL_TDM 2 -#define ADAU1701_SERICTL_RIGHTJ_24 3 -#define ADAU1701_SERICTL_RIGHTJ_20 4 -#define ADAU1701_SERICTL_RIGHTJ_18 5 -#define ADAU1701_SERICTL_RIGHTJ_16 6 -#define ADAU1701_SERICTL_MODE_MASK 7 -#define ADAU1701_SERICTL_INV_BCLK BIT(3) -#define ADAU1701_SERICTL_INV_LRCLK BIT(4) - -#define ADAU1701_OSCIPOW_OPD 0x04 -#define ADAU1701_DACSET_DACINIT 1 - -#define ADAU1701_FIRMWARE "adau1701.bin" - -struct adau1701 { - unsigned int dai_fmt; -}; - -static const struct snd_kcontrol_new adau1701_controls[] = { - SOC_SINGLE("Master Capture Switch", ADAU1701_DSPCTRL, 4, 1, 0), -}; - -static const struct snd_soc_dapm_widget adau1701_dapm_widgets[] = { - SND_SOC_DAPM_DAC("DAC0", "Playback", ADAU1701_AUXNPOW, 3, 1), - SND_SOC_DAPM_DAC("DAC1", "Playback", ADAU1701_AUXNPOW, 2, 1), - SND_SOC_DAPM_DAC("DAC2", "Playback", ADAU1701_AUXNPOW, 1, 1), - SND_SOC_DAPM_DAC("DAC3", "Playback", ADAU1701_AUXNPOW, 0, 1), - SND_SOC_DAPM_ADC("ADC", "Capture", ADAU1701_AUXNPOW, 7, 1), - - SND_SOC_DAPM_OUTPUT("OUT0"), - SND_SOC_DAPM_OUTPUT("OUT1"), - SND_SOC_DAPM_OUTPUT("OUT2"), - SND_SOC_DAPM_OUTPUT("OUT3"), - SND_SOC_DAPM_INPUT("IN0"), - SND_SOC_DAPM_INPUT("IN1"), -}; - -static const struct snd_soc_dapm_route adau1701_dapm_routes[] = { - { "OUT0", NULL, "DAC0" }, - { "OUT1", NULL, "DAC1" }, - { "OUT2", NULL, "DAC2" }, - { "OUT3", NULL, "DAC3" }, - - { "ADC", NULL, "IN0" }, - { "ADC", NULL, "IN1" }, -}; - -static unsigned int adau1701_register_size(struct snd_soc_codec *codec, - unsigned int reg) -{ - switch (reg) { - case ADAU1701_DSPCTRL: - case ADAU1701_SEROCTL: - case ADAU1701_AUXNPOW: - case ADAU1701_OSCIPOW: - case ADAU1701_DACSET: - return 2; - case ADAU1701_SERICTL: - return 1; - } - - dev_err(codec->dev, "Unsupported register address: %d\n", reg); - return 0; -} - -static int adau1701_write(struct snd_soc_codec *codec, unsigned int reg, - unsigned int value) -{ - unsigned int i; - unsigned int size; - uint8_t buf[4]; - int ret; - - size = adau1701_register_size(codec, reg); - if (size == 0) - return -EINVAL; - - snd_soc_cache_write(codec, reg, value); - - buf[0] = 0x08; - buf[1] = reg; - - for (i = size + 1; i >= 2; --i) { - buf[i] = value; - value >>= 8; - } - - ret = i2c_master_send(to_i2c_client(codec->dev), buf, size + 2); - if (ret == size + 2) - return 0; - else if (ret < 0) - return ret; - else - return -EIO; -} - -static unsigned int adau1701_read(struct snd_soc_codec *codec, unsigned int reg) -{ - unsigned int value; - unsigned int ret; - - ret = snd_soc_cache_read(codec, reg, &value); - if (ret) - return ret; - - return value; -} - -static int adau1701_load_firmware(struct snd_soc_codec *codec) -{ - return process_sigma_firmware(codec->control_data, ADAU1701_FIRMWARE); -} - -static int adau1701_set_capture_pcm_format(struct snd_soc_codec *codec, - snd_pcm_format_t format) -{ - struct adau1701 *adau1701 = snd_soc_codec_get_drvdata(codec); - unsigned int mask = ADAU1701_SEROCTL_WORD_LEN_MASK; - unsigned int val; - - switch (format) { - case SNDRV_PCM_FORMAT_S16_LE: - val = ADAU1701_SEROCTL_WORD_LEN_16; - break; - case SNDRV_PCM_FORMAT_S20_3LE: - val = ADAU1701_SEROCTL_WORD_LEN_20; - break; - case SNDRV_PCM_FORMAT_S24_LE: - val = ADAU1701_SEROCTL_WORD_LEN_24; - break; - default: - return -EINVAL; - } - - if (adau1701->dai_fmt == SND_SOC_DAIFMT_RIGHT_J) { - switch (format) { - case SNDRV_PCM_FORMAT_S16_LE: - val |= ADAU1701_SEROCTL_MSB_DEALY16; - break; - case SNDRV_PCM_FORMAT_S20_3LE: - val |= ADAU1701_SEROCTL_MSB_DEALY12; - break; - case SNDRV_PCM_FORMAT_S24_LE: - val |= ADAU1701_SEROCTL_MSB_DEALY8; - break; - } - mask |= ADAU1701_SEROCTL_MSB_DEALY_MASK; - } - - snd_soc_update_bits(codec, ADAU1701_SEROCTL, mask, val); - - return 0; -} - -static int adau1701_set_playback_pcm_format(struct snd_soc_codec *codec, - snd_pcm_format_t format) -{ - struct adau1701 *adau1701 = snd_soc_codec_get_drvdata(codec); - unsigned int val; - - if (adau1701->dai_fmt != SND_SOC_DAIFMT_RIGHT_J) - return 0; - - switch (format) { - case SNDRV_PCM_FORMAT_S16_LE: - val = ADAU1701_SERICTL_RIGHTJ_16; - break; - case SNDRV_PCM_FORMAT_S20_3LE: - val = ADAU1701_SERICTL_RIGHTJ_20; - break; - case SNDRV_PCM_FORMAT_S24_LE: - val = ADAU1701_SERICTL_RIGHTJ_24; - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, ADAU1701_SERICTL, - ADAU1701_SERICTL_MODE_MASK, val); - - return 0; -} - -static int adau1701_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; - snd_pcm_format_t format; - unsigned int val; - - switch (params_rate(params)) { - case 192000: - val = ADAU1701_DSPCTRL_SR_192; - break; - case 96000: - val = ADAU1701_DSPCTRL_SR_96; - break; - case 48000: - val = ADAU1701_DSPCTRL_SR_48; - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, ADAU1701_DSPCTRL, - ADAU1701_DSPCTRL_SR_MASK, val); - - format = params_format(params); - if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) - return adau1701_set_playback_pcm_format(codec, format); - else - return adau1701_set_capture_pcm_format(codec, format); -} - -static int adau1701_set_dai_fmt(struct snd_soc_dai *codec_dai, - unsigned int fmt) -{ - struct snd_soc_codec *codec = codec_dai->codec; - struct adau1701 *adau1701 = snd_soc_codec_get_drvdata(codec); - unsigned int serictl = 0x00, seroctl = 0x00; - bool invert_lrclk; - - switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { - case SND_SOC_DAIFMT_CBM_CFM: - /* master, 64-bits per sample, 1 frame per sample */ - seroctl |= ADAU1701_SEROCTL_MASTER | ADAU1701_SEROCTL_OBF16 - | ADAU1701_SEROCTL_OLF1024; - break; - case SND_SOC_DAIFMT_CBS_CFS: - break; - default: - return -EINVAL; - } - - /* clock inversion */ - switch (fmt & SND_SOC_DAIFMT_INV_MASK) { - case SND_SOC_DAIFMT_NB_NF: - invert_lrclk = false; - break; - case SND_SOC_DAIFMT_NB_IF: - invert_lrclk = true; - break; - case SND_SOC_DAIFMT_IB_NF: - invert_lrclk = false; - serictl |= ADAU1701_SERICTL_INV_BCLK; - seroctl |= ADAU1701_SEROCTL_INV_BCLK; - break; - case SND_SOC_DAIFMT_IB_IF: - invert_lrclk = true; - serictl |= ADAU1701_SERICTL_INV_BCLK; - seroctl |= ADAU1701_SEROCTL_INV_BCLK; - break; - default: - return -EINVAL; - } - - switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { - case SND_SOC_DAIFMT_I2S: - break; - case SND_SOC_DAIFMT_LEFT_J: - serictl |= ADAU1701_SERICTL_LEFTJ; - seroctl |= ADAU1701_SEROCTL_MSB_DEALY0; - invert_lrclk = !invert_lrclk; - break; - case SND_SOC_DAIFMT_RIGHT_J: - serictl |= ADAU1701_SERICTL_RIGHTJ_24; - seroctl |= ADAU1701_SEROCTL_MSB_DEALY8; - invert_lrclk = !invert_lrclk; - break; - default: - return -EINVAL; - } - - if (invert_lrclk) { - seroctl |= ADAU1701_SEROCTL_INV_LRCLK; - serictl |= ADAU1701_SERICTL_INV_LRCLK; - } - - adau1701->dai_fmt = fmt & SND_SOC_DAIFMT_FORMAT_MASK; - - snd_soc_write(codec, ADAU1701_SERICTL, serictl); - snd_soc_update_bits(codec, ADAU1701_SEROCTL, - ~ADAU1701_SEROCTL_WORD_LEN_MASK, seroctl); - - return 0; -} - -static int adau1701_set_bias_level(struct snd_soc_codec *codec, - enum snd_soc_bias_level level) -{ - unsigned int mask = ADAU1701_AUXNPOW_VBPD | ADAU1701_AUXNPOW_VRPD; - - switch (level) { - case SND_SOC_BIAS_ON: - break; - case SND_SOC_BIAS_PREPARE: - break; - case SND_SOC_BIAS_STANDBY: - /* Enable VREF and VREF buffer */ - snd_soc_update_bits(codec, ADAU1701_AUXNPOW, mask, 0x00); - break; - case SND_SOC_BIAS_OFF: - /* Disable VREF and VREF buffer */ - snd_soc_update_bits(codec, ADAU1701_AUXNPOW, mask, mask); - break; - } - - codec->dapm.bias_level = level; - return 0; -} - -static int adau1701_digital_mute(struct snd_soc_dai *dai, int mute) -{ - struct snd_soc_codec *codec = dai->codec; - unsigned int mask = ADAU1701_DSPCTRL_DAM; - unsigned int val; - - if (mute) - val = 0; - else - val = mask; - - snd_soc_update_bits(codec, ADAU1701_DSPCTRL, mask, val); - - return 0; -} - -static int adau1701_set_sysclk(struct snd_soc_codec *codec, int clk_id, - unsigned int freq, int dir) -{ - unsigned int val; - - switch (clk_id) { - case ADAU1701_CLK_SRC_OSC: - val = 0x0; - break; - case ADAU1701_CLK_SRC_MCLK: - val = ADAU1701_OSCIPOW_OPD; - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, ADAU1701_OSCIPOW, ADAU1701_OSCIPOW_OPD, val); - - return 0; -} - -#define ADAU1701_RATES (SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_96000 | \ - SNDRV_PCM_RATE_192000) - -#define ADAU1701_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\ - SNDRV_PCM_FMTBIT_S24_LE) - -static const struct snd_soc_dai_ops adau1701_dai_ops = { - .set_fmt = adau1701_set_dai_fmt, - .hw_params = adau1701_hw_params, - .digital_mute = adau1701_digital_mute, -}; - -static struct snd_soc_dai_driver adau1701_dai = { - .name = "adau1701", - .playback = { - .stream_name = "Playback", - .channels_min = 2, - .channels_max = 8, - .rates = ADAU1701_RATES, - .formats = ADAU1701_FORMATS, - }, - .capture = { - .stream_name = "Capture", - .channels_min = 2, - .channels_max = 8, - .rates = ADAU1701_RATES, - .formats = ADAU1701_FORMATS, - }, - .ops = &adau1701_dai_ops, - .symmetric_rates = 1, -}; - -static int adau1701_probe(struct snd_soc_codec *codec) -{ - int ret; - - codec->dapm.idle_bias_off = 1; - - ret = adau1701_load_firmware(codec); - if (ret) - dev_warn(codec->dev, "Failed to load firmware\n"); - - snd_soc_write(codec, ADAU1701_DACSET, ADAU1701_DACSET_DACINIT); - snd_soc_write(codec, ADAU1701_DSPCTRL, ADAU1701_DSPCTRL_CR); - - return 0; -} - -static struct snd_soc_codec_driver adau1701_codec_drv = { - .probe = adau1701_probe, - .set_bias_level = adau1701_set_bias_level, - - .reg_cache_size = ADAU1701_NUM_REGS, - .reg_word_size = sizeof(u16), - - .controls = adau1701_controls, - .num_controls = ARRAY_SIZE(adau1701_controls), - .dapm_widgets = adau1701_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(adau1701_dapm_widgets), - .dapm_routes = adau1701_dapm_routes, - .num_dapm_routes = ARRAY_SIZE(adau1701_dapm_routes), - - .write = adau1701_write, - .read = adau1701_read, - - .set_sysclk = adau1701_set_sysclk, -}; - -static __devinit int adau1701_i2c_probe(struct i2c_client *client, - const struct i2c_device_id *id) -{ - struct adau1701 *adau1701; - int ret; - - adau1701 = kzalloc(sizeof(*adau1701), GFP_KERNEL); - if (!adau1701) - return -ENOMEM; - - i2c_set_clientdata(client, adau1701); - ret = snd_soc_register_codec(&client->dev, &adau1701_codec_drv, - &adau1701_dai, 1); - if (ret < 0) - kfree(adau1701); - - return ret; -} - -static __devexit int adau1701_i2c_remove(struct i2c_client *client) -{ - snd_soc_unregister_codec(&client->dev); - kfree(i2c_get_clientdata(client)); - return 0; -} - -static const struct i2c_device_id adau1701_i2c_id[] = { - { "adau1701", 0 }, - { } -}; -MODULE_DEVICE_TABLE(i2c, adau1701_i2c_id); - -static struct i2c_driver adau1701_i2c_driver = { - .driver = { - .name = "adau1701", - .owner = THIS_MODULE, - }, - .probe = adau1701_i2c_probe, - .remove = __devexit_p(adau1701_i2c_remove), - .id_table = adau1701_i2c_id, -}; - -static int __init adau1701_init(void) -{ - return i2c_add_driver(&adau1701_i2c_driver); -} -module_init(adau1701_init); - -static void __exit adau1701_exit(void) -{ - i2c_del_driver(&adau1701_i2c_driver); -} -module_exit(adau1701_exit); - -MODULE_DESCRIPTION("ASoC ADAU1701 SigmaDSP driver"); -MODULE_AUTHOR("Cliff Cai "); -MODULE_AUTHOR("Lars-Peter Clausen "); -MODULE_LICENSE("GPL"); diff --git a/trunk/sound/soc/codecs/adau1701.h b/trunk/sound/soc/codecs/adau1701.h deleted file mode 100644 index 8d0949a2aec9..000000000000 --- a/trunk/sound/soc/codecs/adau1701.h +++ /dev/null @@ -1,17 +0,0 @@ -/* - * header file for ADAU1701 SigmaDSP processor - * - * Copyright 2011 Analog Devices Inc. - * - * Licensed under the GPL-2 or later. - */ - -#ifndef _ADAU1701_H -#define _ADAU1701_H - -enum adau1701_clk_src { - ADAU1701_CLK_SRC_OSC, - ADAU1701_CLK_SRC_MCLK, -}; - -#endif diff --git a/trunk/sound/soc/codecs/adav80x.c b/trunk/sound/soc/codecs/adav80x.c deleted file mode 100644 index 300c04b70e71..000000000000 --- a/trunk/sound/soc/codecs/adav80x.c +++ /dev/null @@ -1,951 +0,0 @@ -/* - * ADAV80X Audio Codec driver supporting ADAV801, ADAV803 - * - * Copyright 2011 Analog Devices Inc. - * Author: Yi Li - * Author: Lars-Peter Clausen - * - * Licensed under the GPL-2 or later. - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "adav80x.h" - -#define ADAV80X_PLAYBACK_CTRL 0x04 -#define ADAV80X_AUX_IN_CTRL 0x05 -#define ADAV80X_REC_CTRL 0x06 -#define ADAV80X_AUX_OUT_CTRL 0x07 -#define ADAV80X_DPATH_CTRL1 0x62 -#define ADAV80X_DPATH_CTRL2 0x63 -#define ADAV80X_DAC_CTRL1 0x64 -#define ADAV80X_DAC_CTRL2 0x65 -#define ADAV80X_DAC_CTRL3 0x66 -#define ADAV80X_DAC_L_VOL 0x68 -#define ADAV80X_DAC_R_VOL 0x69 -#define ADAV80X_PGA_L_VOL 0x6c -#define ADAV80X_PGA_R_VOL 0x6d -#define ADAV80X_ADC_CTRL1 0x6e -#define ADAV80X_ADC_CTRL2 0x6f -#define ADAV80X_ADC_L_VOL 0x70 -#define ADAV80X_ADC_R_VOL 0x71 -#define ADAV80X_PLL_CTRL1 0x74 -#define ADAV80X_PLL_CTRL2 0x75 -#define ADAV80X_ICLK_CTRL1 0x76 -#define ADAV80X_ICLK_CTRL2 0x77 -#define ADAV80X_PLL_CLK_SRC 0x78 -#define ADAV80X_PLL_OUTE 0x7a - -#define ADAV80X_PLL_CLK_SRC_PLL_XIN(pll) 0x00 -#define ADAV80X_PLL_CLK_SRC_PLL_MCLKI(pll) (0x40 << (pll)) -#define ADAV80X_PLL_CLK_SRC_PLL_MASK(pll) (0x40 << (pll)) - -#define ADAV80X_ICLK_CTRL1_DAC_SRC(src) ((src) << 5) -#define ADAV80X_ICLK_CTRL1_ADC_SRC(src) ((src) << 2) -#define ADAV80X_ICLK_CTRL1_ICLK2_SRC(src) (src) -#define ADAV80X_ICLK_CTRL2_ICLK1_SRC(src) ((src) << 3) - -#define ADAV80X_PLL_CTRL1_PLLDIV 0x10 -#define ADAV80X_PLL_CTRL1_PLLPD(pll) (0x04 << (pll)) -#define ADAV80X_PLL_CTRL1_XTLPD 0x02 - -#define ADAV80X_PLL_CTRL2_FIELD(pll, x) ((x) << ((pll) * 4)) - -#define ADAV80X_PLL_CTRL2_FS_48(pll) ADAV80X_PLL_CTRL2_FIELD((pll), 0x00) -#define ADAV80X_PLL_CTRL2_FS_32(pll) ADAV80X_PLL_CTRL2_FIELD((pll), 0x08) -#define ADAV80X_PLL_CTRL2_FS_44(pll) ADAV80X_PLL_CTRL2_FIELD((pll), 0x0c) - -#define ADAV80X_PLL_CTRL2_SEL(pll) ADAV80X_PLL_CTRL2_FIELD((pll), 0x02) -#define ADAV80X_PLL_CTRL2_DOUB(pll) ADAV80X_PLL_CTRL2_FIELD((pll), 0x01) -#define ADAV80X_PLL_CTRL2_PLL_MASK(pll) ADAV80X_PLL_CTRL2_FIELD((pll), 0x0f) - -#define ADAV80X_ADC_CTRL1_MODULATOR_MASK 0x80 -#define ADAV80X_ADC_CTRL1_MODULATOR_128FS 0x00 -#define ADAV80X_ADC_CTRL1_MODULATOR_64FS 0x80 - -#define ADAV80X_DAC_CTRL1_PD 0x80 - -#define ADAV80X_DAC_CTRL2_DIV1 0x00 -#define ADAV80X_DAC_CTRL2_DIV1_5 0x10 -#define ADAV80X_DAC_CTRL2_DIV2 0x20 -#define ADAV80X_DAC_CTRL2_DIV3 0x30 -#define ADAV80X_DAC_CTRL2_DIV_MASK 0x30 - -#define ADAV80X_DAC_CTRL2_INTERPOL_256FS 0x00 -#define ADAV80X_DAC_CTRL2_INTERPOL_128FS 0x40 -#define ADAV80X_DAC_CTRL2_INTERPOL_64FS 0x80 -#define ADAV80X_DAC_CTRL2_INTERPOL_MASK 0xc0 - -#define ADAV80X_DAC_CTRL2_DEEMPH_NONE 0x00 -#define ADAV80X_DAC_CTRL2_DEEMPH_44 0x01 -#define ADAV80X_DAC_CTRL2_DEEMPH_32 0x02 -#define ADAV80X_DAC_CTRL2_DEEMPH_48 0x03 -#define ADAV80X_DAC_CTRL2_DEEMPH_MASK 0x01 - -#define ADAV80X_CAPTURE_MODE_MASTER 0x20 -#define ADAV80X_CAPTURE_WORD_LEN24 0x00 -#define ADAV80X_CAPTURE_WORD_LEN20 0x04 -#define ADAV80X_CAPTRUE_WORD_LEN18 0x08 -#define ADAV80X_CAPTURE_WORD_LEN16 0x0c -#define ADAV80X_CAPTURE_WORD_LEN_MASK 0x0c - -#define ADAV80X_CAPTURE_MODE_LEFT_J 0x00 -#define ADAV80X_CAPTURE_MODE_I2S 0x01 -#define ADAV80X_CAPTURE_MODE_RIGHT_J 0x03 -#define ADAV80X_CAPTURE_MODE_MASK 0x03 - -#define ADAV80X_PLAYBACK_MODE_MASTER 0x10 -#define ADAV80X_PLAYBACK_MODE_LEFT_J 0x00 -#define ADAV80X_PLAYBACK_MODE_I2S 0x01 -#define ADAV80X_PLAYBACK_MODE_RIGHT_J_24 0x04 -#define ADAV80X_PLAYBACK_MODE_RIGHT_J_20 0x05 -#define ADAV80X_PLAYBACK_MODE_RIGHT_J_18 0x06 -#define ADAV80X_PLAYBACK_MODE_RIGHT_J_16 0x07 -#define ADAV80X_PLAYBACK_MODE_MASK 0x07 - -#define ADAV80X_PLL_OUTE_SYSCLKPD(x) BIT(2 - (x)) - -static u8 adav80x_default_regs[] = { - 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x02, 0x01, 0x80, 0x26, 0x00, 0x00, - 0x02, 0x40, 0x20, 0x00, 0x09, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x04, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0xd1, 0x92, 0xb1, 0x37, - 0x48, 0xd2, 0xfb, 0xca, 0xd2, 0x15, 0xe8, 0x29, 0xb9, 0x6a, 0xda, 0x2b, - 0xb7, 0xc0, 0x11, 0x65, 0x5c, 0xf6, 0xff, 0x8d, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xa5, 0x00, 0x00, - 0x00, 0xe8, 0x46, 0xe1, 0x5b, 0xd3, 0x43, 0x77, 0x93, 0xa7, 0x44, 0xee, - 0x32, 0x12, 0xc0, 0x11, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x3f, 0x3f, - 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x00, 0x1d, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x52, 0x00, -}; - -struct adav80x { - enum snd_soc_control_type control_type; - - enum adav80x_clk_src clk_src; - unsigned int sysclk; - enum adav80x_pll_src pll_src; - - unsigned int dai_fmt[2]; - unsigned int rate; - bool deemph; - bool sysclk_pd[3]; -}; - -static const char *adav80x_mux_text[] = { - "ADC", - "Playback", - "Aux Playback", -}; - -static const unsigned int adav80x_mux_values[] = { - 0, 2, 3, -}; - -#define ADAV80X_MUX_ENUM_DECL(name, reg, shift) \ - SOC_VALUE_ENUM_DOUBLE_DECL(name, reg, shift, 7, \ - ARRAY_SIZE(adav80x_mux_text), adav80x_mux_text, \ - adav80x_mux_values) - -static ADAV80X_MUX_ENUM_DECL(adav80x_aux_capture_enum, ADAV80X_DPATH_CTRL1, 0); -static ADAV80X_MUX_ENUM_DECL(adav80x_capture_enum, ADAV80X_DPATH_CTRL1, 3); -static ADAV80X_MUX_ENUM_DECL(adav80x_dac_enum, ADAV80X_DPATH_CTRL2, 3); - -static const struct snd_kcontrol_new adav80x_aux_capture_mux_ctrl = - SOC_DAPM_VALUE_ENUM("Route", adav80x_aux_capture_enum); -static const struct snd_kcontrol_new adav80x_capture_mux_ctrl = - SOC_DAPM_VALUE_ENUM("Route", adav80x_capture_enum); -static const struct snd_kcontrol_new adav80x_dac_mux_ctrl = - SOC_DAPM_VALUE_ENUM("Route", adav80x_dac_enum); - -#define ADAV80X_MUX(name, ctrl) \ - SND_SOC_DAPM_VALUE_MUX(name, SND_SOC_NOPM, 0, 0, ctrl) - -static const struct snd_soc_dapm_widget adav80x_dapm_widgets[] = { - SND_SOC_DAPM_DAC("DAC", NULL, ADAV80X_DAC_CTRL1, 7, 1), - SND_SOC_DAPM_ADC("ADC", NULL, ADAV80X_ADC_CTRL1, 5, 1), - - SND_SOC_DAPM_PGA("Right PGA", ADAV80X_ADC_CTRL1, 0, 1, NULL, 0), - SND_SOC_DAPM_PGA("Left PGA", ADAV80X_ADC_CTRL1, 1, 1, NULL, 0), - - SND_SOC_DAPM_AIF_OUT("AIFOUT", "HiFi Capture", 0, SND_SOC_NOPM, 0, 0), - SND_SOC_DAPM_AIF_IN("AIFIN", "HiFi Playback", 0, SND_SOC_NOPM, 0, 0), - - SND_SOC_DAPM_AIF_OUT("AIFAUXOUT", "Aux Capture", 0, SND_SOC_NOPM, 0, 0), - SND_SOC_DAPM_AIF_IN("AIFAUXIN", "Aux Playback", 0, SND_SOC_NOPM, 0, 0), - - ADAV80X_MUX("Aux Capture Select", &adav80x_aux_capture_mux_ctrl), - ADAV80X_MUX("Capture Select", &adav80x_capture_mux_ctrl), - ADAV80X_MUX("DAC Select", &adav80x_dac_mux_ctrl), - - SND_SOC_DAPM_INPUT("VINR"), - SND_SOC_DAPM_INPUT("VINL"), - SND_SOC_DAPM_OUTPUT("VOUTR"), - SND_SOC_DAPM_OUTPUT("VOUTL"), - - SND_SOC_DAPM_SUPPLY("SYSCLK", SND_SOC_NOPM, 0, 0, NULL, 0), - SND_SOC_DAPM_SUPPLY("PLL1", ADAV80X_PLL_CTRL1, 2, 1, NULL, 0), - SND_SOC_DAPM_SUPPLY("PLL2", ADAV80X_PLL_CTRL1, 3, 1, NULL, 0), - SND_SOC_DAPM_SUPPLY("OSC", ADAV80X_PLL_CTRL1, 1, 1, NULL, 0), -}; - -static int adav80x_dapm_sysclk_check(struct snd_soc_dapm_widget *source, - struct snd_soc_dapm_widget *sink) -{ - struct snd_soc_codec *codec = source->codec; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - const char *clk; - - switch (adav80x->clk_src) { - case ADAV80X_CLK_PLL1: - clk = "PLL1"; - break; - case ADAV80X_CLK_PLL2: - clk = "PLL2"; - break; - case ADAV80X_CLK_XTAL: - clk = "OSC"; - break; - default: - return 0; - } - - return strcmp(source->name, clk) == 0; -} - -static int adav80x_dapm_pll_check(struct snd_soc_dapm_widget *source, - struct snd_soc_dapm_widget *sink) -{ - struct snd_soc_codec *codec = source->codec; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - - return adav80x->pll_src == ADAV80X_PLL_SRC_XTAL; -} - - -static const struct snd_soc_dapm_route adav80x_dapm_routes[] = { - { "DAC Select", "ADC", "ADC" }, - { "DAC Select", "Playback", "AIFIN" }, - { "DAC Select", "Aux Playback", "AIFAUXIN" }, - { "DAC", NULL, "DAC Select" }, - - { "Capture Select", "ADC", "ADC" }, - { "Capture Select", "Playback", "AIFIN" }, - { "Capture Select", "Aux Playback", "AIFAUXIN" }, - { "AIFOUT", NULL, "Capture Select" }, - - { "Aux Capture Select", "ADC", "ADC" }, - { "Aux Capture Select", "Playback", "AIFIN" }, - { "Aux Capture Select", "Aux Playback", "AIFAUXIN" }, - { "AIFAUXOUT", NULL, "Aux Capture Select" }, - - { "VOUTR", NULL, "DAC" }, - { "VOUTL", NULL, "DAC" }, - - { "Left PGA", NULL, "VINL" }, - { "Right PGA", NULL, "VINR" }, - { "ADC", NULL, "Left PGA" }, - { "ADC", NULL, "Right PGA" }, - - { "SYSCLK", NULL, "PLL1", adav80x_dapm_sysclk_check }, - { "SYSCLK", NULL, "PLL2", adav80x_dapm_sysclk_check }, - { "SYSCLK", NULL, "OSC", adav80x_dapm_sysclk_check }, - { "PLL1", NULL, "OSC", adav80x_dapm_pll_check }, - { "PLL2", NULL, "OSC", adav80x_dapm_pll_check }, - - { "ADC", NULL, "SYSCLK" }, - { "DAC", NULL, "SYSCLK" }, - { "AIFOUT", NULL, "SYSCLK" }, - { "AIFAUXOUT", NULL, "SYSCLK" }, - { "AIFIN", NULL, "SYSCLK" }, - { "AIFAUXIN", NULL, "SYSCLK" }, -}; - -static int adav80x_set_deemph(struct snd_soc_codec *codec) -{ - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - unsigned int val; - - if (adav80x->deemph) { - switch (adav80x->rate) { - case 32000: - val = ADAV80X_DAC_CTRL2_DEEMPH_32; - break; - case 44100: - val = ADAV80X_DAC_CTRL2_DEEMPH_44; - break; - case 48000: - case 64000: - case 88200: - case 96000: - val = ADAV80X_DAC_CTRL2_DEEMPH_48; - break; - default: - val = ADAV80X_DAC_CTRL2_DEEMPH_NONE; - break; - } - } else { - val = ADAV80X_DAC_CTRL2_DEEMPH_NONE; - } - - return snd_soc_update_bits(codec, ADAV80X_DAC_CTRL2, - ADAV80X_DAC_CTRL2_DEEMPH_MASK, val); -} - -static int adav80x_put_deemph(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol) -{ - struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - unsigned int deemph = ucontrol->value.enumerated.item[0]; - - if (deemph > 1) - return -EINVAL; - - adav80x->deemph = deemph; - - return adav80x_set_deemph(codec); -} - -static int adav80x_get_deemph(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol) -{ - struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - - ucontrol->value.enumerated.item[0] = adav80x->deemph; - return 0; -}; - -static const DECLARE_TLV_DB_SCALE(adav80x_inpga_tlv, 0, 50, 0); -static const DECLARE_TLV_DB_MINMAX(adav80x_digital_tlv, -9563, 0); - -static const struct snd_kcontrol_new adav80x_controls[] = { - SOC_DOUBLE_R_TLV("Master Playback Volume", ADAV80X_DAC_L_VOL, - ADAV80X_DAC_R_VOL, 0, 0xff, 0, adav80x_digital_tlv), - SOC_DOUBLE_R_TLV("Master Capture Volume", ADAV80X_ADC_L_VOL, - ADAV80X_ADC_R_VOL, 0, 0xff, 0, adav80x_digital_tlv), - - SOC_DOUBLE_R_TLV("PGA Capture Volume", ADAV80X_PGA_L_VOL, - ADAV80X_PGA_R_VOL, 0, 0x30, 0, adav80x_inpga_tlv), - - SOC_DOUBLE("Master Playback Switch", ADAV80X_DAC_CTRL1, 0, 1, 1, 0), - SOC_DOUBLE("Master Capture Switch", ADAV80X_ADC_CTRL1, 2, 3, 1, 1), - - SOC_SINGLE("ADC High Pass Filter Switch", ADAV80X_ADC_CTRL1, 6, 1, 0), - - SOC_SINGLE_BOOL_EXT("Playback De-emphasis Switch", 0, - adav80x_get_deemph, adav80x_put_deemph), -}; - -static unsigned int adav80x_port_ctrl_regs[2][2] = { - { ADAV80X_REC_CTRL, ADAV80X_PLAYBACK_CTRL, }, - { ADAV80X_AUX_OUT_CTRL, ADAV80X_AUX_IN_CTRL }, -}; - -static int adav80x_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt) -{ - struct snd_soc_codec *codec = dai->codec; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - unsigned int capture = 0x00; - unsigned int playback = 0x00; - - switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { - case SND_SOC_DAIFMT_CBM_CFM: - capture |= ADAV80X_CAPTURE_MODE_MASTER; - playback |= ADAV80X_PLAYBACK_MODE_MASTER; - case SND_SOC_DAIFMT_CBS_CFS: - break; - default: - return -EINVAL; - } - - switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { - case SND_SOC_DAIFMT_I2S: - capture |= ADAV80X_CAPTURE_MODE_I2S; - playback |= ADAV80X_PLAYBACK_MODE_I2S; - break; - case SND_SOC_DAIFMT_LEFT_J: - capture |= ADAV80X_CAPTURE_MODE_LEFT_J; - playback |= ADAV80X_PLAYBACK_MODE_LEFT_J; - break; - case SND_SOC_DAIFMT_RIGHT_J: - capture |= ADAV80X_CAPTURE_MODE_RIGHT_J; - playback |= ADAV80X_PLAYBACK_MODE_RIGHT_J_24; - break; - default: - return -EINVAL; - } - - switch (fmt & SND_SOC_DAIFMT_INV_MASK) { - case SND_SOC_DAIFMT_NB_NF: - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, adav80x_port_ctrl_regs[dai->id][0], - ADAV80X_CAPTURE_MODE_MASK | ADAV80X_CAPTURE_MODE_MASTER, - capture); - snd_soc_write(codec, adav80x_port_ctrl_regs[dai->id][1], playback); - - adav80x->dai_fmt[dai->id] = fmt & SND_SOC_DAIFMT_FORMAT_MASK; - - return 0; -} - -static int adav80x_set_adc_clock(struct snd_soc_codec *codec, - unsigned int sample_rate) -{ - unsigned int val; - - if (sample_rate <= 48000) - val = ADAV80X_ADC_CTRL1_MODULATOR_128FS; - else - val = ADAV80X_ADC_CTRL1_MODULATOR_64FS; - - snd_soc_update_bits(codec, ADAV80X_ADC_CTRL1, - ADAV80X_ADC_CTRL1_MODULATOR_MASK, val); - - return 0; -} - -static int adav80x_set_dac_clock(struct snd_soc_codec *codec, - unsigned int sample_rate) -{ - unsigned int val; - - if (sample_rate <= 48000) - val = ADAV80X_DAC_CTRL2_DIV1 | ADAV80X_DAC_CTRL2_INTERPOL_256FS; - else - val = ADAV80X_DAC_CTRL2_DIV2 | ADAV80X_DAC_CTRL2_INTERPOL_128FS; - - snd_soc_update_bits(codec, ADAV80X_DAC_CTRL2, - ADAV80X_DAC_CTRL2_DIV_MASK | ADAV80X_DAC_CTRL2_INTERPOL_MASK, - val); - - return 0; -} - -static int adav80x_set_capture_pcm_format(struct snd_soc_codec *codec, - struct snd_soc_dai *dai, snd_pcm_format_t format) -{ - unsigned int val; - - switch (format) { - case SNDRV_PCM_FORMAT_S16_LE: - val = ADAV80X_CAPTURE_WORD_LEN16; - break; - case SNDRV_PCM_FORMAT_S18_3LE: - val = ADAV80X_CAPTRUE_WORD_LEN18; - break; - case SNDRV_PCM_FORMAT_S20_3LE: - val = ADAV80X_CAPTURE_WORD_LEN20; - break; - case SNDRV_PCM_FORMAT_S24_LE: - val = ADAV80X_CAPTURE_WORD_LEN24; - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, adav80x_port_ctrl_regs[dai->id][0], - ADAV80X_CAPTURE_WORD_LEN_MASK, val); - - return 0; -} - -static int adav80x_set_playback_pcm_format(struct snd_soc_codec *codec, - struct snd_soc_dai *dai, snd_pcm_format_t format) -{ - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - unsigned int val; - - if (adav80x->dai_fmt[dai->id] != SND_SOC_DAIFMT_RIGHT_J) - return 0; - - switch (format) { - case SNDRV_PCM_FORMAT_S16_LE: - val = ADAV80X_PLAYBACK_MODE_RIGHT_J_16; - break; - case SNDRV_PCM_FORMAT_S18_3LE: - val = ADAV80X_PLAYBACK_MODE_RIGHT_J_18; - break; - case SNDRV_PCM_FORMAT_S20_3LE: - val = ADAV80X_PLAYBACK_MODE_RIGHT_J_20; - break; - case SNDRV_PCM_FORMAT_S24_LE: - val = ADAV80X_PLAYBACK_MODE_RIGHT_J_24; - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, adav80x_port_ctrl_regs[dai->id][1], - ADAV80X_PLAYBACK_MODE_MASK, val); - - return 0; -} - -static int adav80x_hw_params(struct snd_pcm_substream *substream, - struct snd_pcm_hw_params *params, struct snd_soc_dai *dai) -{ - struct snd_soc_codec *codec = dai->codec; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - unsigned int rate = params_rate(params); - - if (rate * 256 != adav80x->sysclk) - return -EINVAL; - - if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { - adav80x_set_playback_pcm_format(codec, dai, - params_format(params)); - adav80x_set_dac_clock(codec, rate); - } else { - adav80x_set_capture_pcm_format(codec, dai, - params_format(params)); - adav80x_set_adc_clock(codec, rate); - } - adav80x->rate = rate; - adav80x_set_deemph(codec); - - return 0; -} - -static int adav80x_set_sysclk(struct snd_soc_codec *codec, - int clk_id, unsigned int freq, int dir) -{ - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - - if (dir == SND_SOC_CLOCK_IN) { - switch (clk_id) { - case ADAV80X_CLK_XIN: - case ADAV80X_CLK_XTAL: - case ADAV80X_CLK_MCLKI: - case ADAV80X_CLK_PLL1: - case ADAV80X_CLK_PLL2: - break; - default: - return -EINVAL; - } - - adav80x->sysclk = freq; - - if (adav80x->clk_src != clk_id) { - unsigned int iclk_ctrl1, iclk_ctrl2; - - adav80x->clk_src = clk_id; - if (clk_id == ADAV80X_CLK_XTAL) - clk_id = ADAV80X_CLK_XIN; - - iclk_ctrl1 = ADAV80X_ICLK_CTRL1_DAC_SRC(clk_id) | - ADAV80X_ICLK_CTRL1_ADC_SRC(clk_id) | - ADAV80X_ICLK_CTRL1_ICLK2_SRC(clk_id); - iclk_ctrl2 = ADAV80X_ICLK_CTRL2_ICLK1_SRC(clk_id); - - snd_soc_write(codec, ADAV80X_ICLK_CTRL1, iclk_ctrl1); - snd_soc_write(codec, ADAV80X_ICLK_CTRL2, iclk_ctrl2); - - snd_soc_dapm_sync(&codec->dapm); - } - } else { - unsigned int mask; - - switch (clk_id) { - case ADAV80X_CLK_SYSCLK1: - case ADAV80X_CLK_SYSCLK2: - case ADAV80X_CLK_SYSCLK3: - break; - default: - return -EINVAL; - } - - clk_id -= ADAV80X_CLK_SYSCLK1; - mask = ADAV80X_PLL_OUTE_SYSCLKPD(clk_id); - - if (freq == 0) { - snd_soc_update_bits(codec, ADAV80X_PLL_OUTE, mask, mask); - adav80x->sysclk_pd[clk_id] = true; - } else { - snd_soc_update_bits(codec, ADAV80X_PLL_OUTE, mask, 0); - adav80x->sysclk_pd[clk_id] = false; - } - - if (adav80x->sysclk_pd[0]) - snd_soc_dapm_disable_pin(&codec->dapm, "PLL1"); - else - snd_soc_dapm_force_enable_pin(&codec->dapm, "PLL1"); - - if (adav80x->sysclk_pd[1] || adav80x->sysclk_pd[2]) - snd_soc_dapm_disable_pin(&codec->dapm, "PLL2"); - else - snd_soc_dapm_force_enable_pin(&codec->dapm, "PLL2"); - - snd_soc_dapm_sync(&codec->dapm); - } - - return 0; -} - -static int adav80x_set_pll(struct snd_soc_codec *codec, int pll_id, - int source, unsigned int freq_in, unsigned int freq_out) -{ - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - unsigned int pll_ctrl1 = 0; - unsigned int pll_ctrl2 = 0; - unsigned int pll_src; - - switch (source) { - case ADAV80X_PLL_SRC_XTAL: - case ADAV80X_PLL_SRC_XIN: - case ADAV80X_PLL_SRC_MCLKI: - break; - default: - return -EINVAL; - } - - if (!freq_out) - return 0; - - switch (freq_in) { - case 27000000: - break; - case 54000000: - if (source == ADAV80X_PLL_SRC_XIN) { - pll_ctrl1 |= ADAV80X_PLL_CTRL1_PLLDIV; - break; - } - default: - return -EINVAL; - } - - if (freq_out > 12288000) { - pll_ctrl2 |= ADAV80X_PLL_CTRL2_DOUB(pll_id); - freq_out /= 2; - } - - /* freq_out = sample_rate * 256 */ - switch (freq_out) { - case 8192000: - pll_ctrl2 |= ADAV80X_PLL_CTRL2_FS_32(pll_id); - break; - case 11289600: - pll_ctrl2 |= ADAV80X_PLL_CTRL2_FS_44(pll_id); - break; - case 12288000: - pll_ctrl2 |= ADAV80X_PLL_CTRL2_FS_48(pll_id); - break; - default: - return -EINVAL; - } - - snd_soc_update_bits(codec, ADAV80X_PLL_CTRL1, ADAV80X_PLL_CTRL1_PLLDIV, - pll_ctrl1); - snd_soc_update_bits(codec, ADAV80X_PLL_CTRL2, - ADAV80X_PLL_CTRL2_PLL_MASK(pll_id), pll_ctrl2); - - if (source != adav80x->pll_src) { - if (source == ADAV80X_PLL_SRC_MCLKI) - pll_src = ADAV80X_PLL_CLK_SRC_PLL_MCLKI(pll_id); - else - pll_src = ADAV80X_PLL_CLK_SRC_PLL_XIN(pll_id); - - snd_soc_update_bits(codec, ADAV80X_PLL_CLK_SRC, - ADAV80X_PLL_CLK_SRC_PLL_MASK(pll_id), pll_src); - - adav80x->pll_src = source; - - snd_soc_dapm_sync(&codec->dapm); - } - - return 0; -} - -static int adav80x_set_bias_level(struct snd_soc_codec *codec, - enum snd_soc_bias_level level) -{ - unsigned int mask = ADAV80X_DAC_CTRL1_PD; - - switch (level) { - case SND_SOC_BIAS_ON: - break; - case SND_SOC_BIAS_PREPARE: - break; - case SND_SOC_BIAS_STANDBY: - snd_soc_update_bits(codec, ADAV80X_DAC_CTRL1, mask, 0x00); - break; - case SND_SOC_BIAS_OFF: - snd_soc_update_bits(codec, ADAV80X_DAC_CTRL1, mask, mask); - break; - } - - codec->dapm.bias_level = level; - return 0; -} - -/* Enforce the same sample rate on all audio interfaces */ -static int adav80x_dai_startup(struct snd_pcm_substream *substream, - struct snd_soc_dai *dai) -{ - struct snd_soc_codec *codec = dai->codec; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - - if (!codec->active || !adav80x->rate) - return 0; - - return snd_pcm_hw_constraint_minmax(substream->runtime, - SNDRV_PCM_HW_PARAM_RATE, adav80x->rate, adav80x->rate); -} - -static void adav80x_dai_shutdown(struct snd_pcm_substream *substream, - struct snd_soc_dai *dai) -{ - struct snd_soc_codec *codec = dai->codec; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - - if (!codec->active) - adav80x->rate = 0; -} - -static const struct snd_soc_dai_ops adav80x_dai_ops = { - .set_fmt = adav80x_set_dai_fmt, - .hw_params = adav80x_hw_params, - .startup = adav80x_dai_startup, - .shutdown = adav80x_dai_shutdown, -}; - -#define ADAV80X_PLAYBACK_RATES (SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | \ - SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_64000 | SNDRV_PCM_RATE_88200 | \ - SNDRV_PCM_RATE_96000) - -#define ADAV80X_CAPTURE_RATES (SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_96000) - -#define ADAV80X_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S18_3LE | \ - SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_LE) - -static struct snd_soc_dai_driver adav80x_dais[] = { - { - .name = "adav80x-hifi", - .id = 0, - .playback = { - .stream_name = "HiFi Playback", - .channels_min = 2, - .channels_max = 2, - .rates = ADAV80X_PLAYBACK_RATES, - .formats = ADAV80X_FORMATS, - }, - .capture = { - .stream_name = "HiFi Capture", - .channels_min = 2, - .channels_max = 2, - .rates = ADAV80X_CAPTURE_RATES, - .formats = ADAV80X_FORMATS, - }, - .ops = &adav80x_dai_ops, - }, - { - .name = "adav80x-aux", - .id = 1, - .playback = { - .stream_name = "Aux Playback", - .channels_min = 2, - .channels_max = 2, - .rates = ADAV80X_PLAYBACK_RATES, - .formats = ADAV80X_FORMATS, - }, - .capture = { - .stream_name = "Aux Capture", - .channels_min = 2, - .channels_max = 2, - .rates = ADAV80X_CAPTURE_RATES, - .formats = ADAV80X_FORMATS, - }, - .ops = &adav80x_dai_ops, - }, -}; - -static int adav80x_probe(struct snd_soc_codec *codec) -{ - int ret; - struct adav80x *adav80x = snd_soc_codec_get_drvdata(codec); - - ret = snd_soc_codec_set_cache_io(codec, 7, 9, adav80x->control_type); - if (ret) { - dev_err(codec->dev, "failed to set cache I/O: %d\n", ret); - return ret; - } - - /* Force PLLs on for SYSCLK output */ - snd_soc_dapm_force_enable_pin(&codec->dapm, "PLL1"); - snd_soc_dapm_force_enable_pin(&codec->dapm, "PLL2"); - - /* Power down S/PDIF receiver, since it is currently not supported */ - snd_soc_write(codec, ADAV80X_PLL_OUTE, 0x20); - /* Disable DAC zero flag */ - snd_soc_write(codec, ADAV80X_DAC_CTRL3, 0x6); - - return adav80x_set_bias_level(codec, SND_SOC_BIAS_STANDBY); -} - -static int adav80x_suspend(struct snd_soc_codec *codec, pm_message_t state) -{ - return adav80x_set_bias_level(codec, SND_SOC_BIAS_OFF); -} - -static int adav80x_resume(struct snd_soc_codec *codec) -{ - adav80x_set_bias_level(codec, SND_SOC_BIAS_STANDBY); - codec->cache_sync = 1; - snd_soc_cache_sync(codec); - - return 0; -} - -static int adav80x_remove(struct snd_soc_codec *codec) -{ - return adav80x_set_bias_level(codec, SND_SOC_BIAS_OFF); -} - -static struct snd_soc_codec_driver adav80x_codec_driver = { - .probe = adav80x_probe, - .remove = adav80x_remove, - .suspend = adav80x_suspend, - .resume = adav80x_resume, - .set_bias_level = adav80x_set_bias_level, - - .set_pll = adav80x_set_pll, - .set_sysclk = adav80x_set_sysclk, - - .reg_word_size = sizeof(u8), - .reg_cache_size = ARRAY_SIZE(adav80x_default_regs), - .reg_cache_default = adav80x_default_regs, - - .controls = adav80x_controls, - .num_controls = ARRAY_SIZE(adav80x_controls), - .dapm_widgets = adav80x_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(adav80x_dapm_widgets), - .dapm_routes = adav80x_dapm_routes, - .num_dapm_routes = ARRAY_SIZE(adav80x_dapm_routes), -}; - -static int __devinit adav80x_bus_probe(struct device *dev, - enum snd_soc_control_type control_type) -{ - struct adav80x *adav80x; - int ret; - - adav80x = kzalloc(sizeof(*adav80x), GFP_KERNEL); - if (!adav80x) - return -ENOMEM; - - dev_set_drvdata(dev, adav80x); - adav80x->control_type = control_type; - - ret = snd_soc_register_codec(dev, &adav80x_codec_driver, - adav80x_dais, ARRAY_SIZE(adav80x_dais)); - if (ret) - kfree(adav80x); - - return ret; -} - -static int __devexit adav80x_bus_remove(struct device *dev) -{ - snd_soc_unregister_codec(dev); - kfree(dev_get_drvdata(dev)); - return 0; -} - -#if defined(CONFIG_SPI_MASTER) -static int __devinit adav80x_spi_probe(struct spi_device *spi) -{ - return adav80x_bus_probe(&spi->dev, SND_SOC_SPI); -} - -static int __devexit adav80x_spi_remove(struct spi_device *spi) -{ - return adav80x_bus_remove(&spi->dev); -} - -static struct spi_driver adav80x_spi_driver = { - .driver = { - .name = "adav801", - .owner = THIS_MODULE, - }, - .probe = adav80x_spi_probe, - .remove = __devexit_p(adav80x_spi_remove), -}; -#endif - -#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) -static const struct i2c_device_id adav80x_id[] = { - { "adav803", 0 }, - { } -}; -MODULE_DEVICE_TABLE(i2c, adav80x_id); - -static int __devinit adav80x_i2c_probe(struct i2c_client *client, - const struct i2c_device_id *id) -{ - return adav80x_bus_probe(&client->dev, SND_SOC_I2C); -} - -static int __devexit adav80x_i2c_remove(struct i2c_client *client) -{ - return adav80x_bus_remove(&client->dev); -} - -static struct i2c_driver adav80x_i2c_driver = { - .driver = { - .name = "adav803", - .owner = THIS_MODULE, - }, - .probe = adav80x_i2c_probe, - .remove = __devexit_p(adav80x_i2c_remove), - .id_table = adav80x_id, -}; -#endif - -static int __init adav80x_init(void) -{ - int ret = 0; - -#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) - ret = i2c_add_driver(&adav80x_i2c_driver); - if (ret) - return ret; -#endif - -#if defined(CONFIG_SPI_MASTER) - ret = spi_register_driver(&adav80x_spi_driver); -#endif - - return ret; -} -module_init(adav80x_init); - -static void __exit adav80x_exit(void) -{ -#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) - i2c_del_driver(&adav80x_i2c_driver); -#endif -#if defined(CONFIG_SPI_MASTER) - spi_unregister_driver(&adav80x_spi_driver); -#endif -} -module_exit(adav80x_exit); - -MODULE_DESCRIPTION("ASoC ADAV80x driver"); -MODULE_AUTHOR("Lars-Peter Clausen "); -MODULE_AUTHOR("Yi Li >"); -MODULE_LICENSE("GPL"); diff --git a/trunk/sound/soc/codecs/adav80x.h b/trunk/sound/soc/codecs/adav80x.h deleted file mode 100644 index adb0fc76d4e3..000000000000 --- a/trunk/sound/soc/codecs/adav80x.h +++ /dev/null @@ -1,35 +0,0 @@ -/* - * header file for ADAV80X parts - * - * Copyright 2011 Analog Devices Inc. - * - * Licensed under the GPL-2 or later. - */ - -#ifndef _ADAV80X_H -#define _ADAV80X_H - -enum adav80x_pll_src { - ADAV80X_PLL_SRC_XIN, - ADAV80X_PLL_SRC_XTAL, - ADAV80X_PLL_SRC_MCLKI, -}; - -enum adav80x_pll { - ADAV80X_PLL1 = 0, - ADAV80X_PLL2 = 1, -}; - -enum adav80x_clk_src { - ADAV80X_CLK_XIN = 0, - ADAV80X_CLK_MCLKI = 1, - ADAV80X_CLK_PLL1 = 2, - ADAV80X_CLK_PLL2 = 3, - ADAV80X_CLK_XTAL = 6, - - ADAV80X_CLK_SYSCLK1 = 6, - ADAV80X_CLK_SYSCLK2 = 7, - ADAV80X_CLK_SYSCLK3 = 8, -}; - -#endif diff --git a/trunk/sound/soc/codecs/ak4641.c b/trunk/sound/soc/codecs/ak4641.c index 7a64e58cddc4..ed96f247c2da 100644 --- a/trunk/sound/soc/codecs/ak4641.c +++ b/trunk/sound/soc/codecs/ak4641.c @@ -457,7 +457,7 @@ static struct snd_soc_dai_ops ak4641_pcm_dai_ops = { .set_sysclk = ak4641_set_dai_sysclk, }; -static struct snd_soc_dai_driver ak4641_dai[] = { +struct snd_soc_dai_driver ak4641_dai[] = { { .name = "ak4641-hifi", .id = 1, diff --git a/trunk/sound/soc/codecs/ak4642.c b/trunk/sound/soc/codecs/ak4642.c index 65f46047b1cb..4be0570e3f1f 100644 --- a/trunk/sound/soc/codecs/ak4642.c +++ b/trunk/sound/soc/codecs/ak4642.c @@ -357,7 +357,7 @@ static int ak4642_dai_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) default: return -EINVAL; } - snd_soc_update_bits(codec, PW_MGMT2, MS | MCKO | PMPLL, data); + snd_soc_update_bits(codec, PW_MGMT2, MS, data); snd_soc_update_bits(codec, MD_CTL1, BCKO_MASK, bcko); /* format type */ diff --git a/trunk/sound/soc/codecs/cs4270.c b/trunk/sound/soc/codecs/cs4270.c index 6cc8678f49f3..0206a17d7283 100644 --- a/trunk/sound/soc/codecs/cs4270.c +++ b/trunk/sound/soc/codecs/cs4270.c @@ -636,7 +636,10 @@ static int cs4270_soc_resume(struct snd_soc_codec *codec) #endif /* CONFIG_PM */ /* - * ASoC codec driver structure + * ASoC codec device structure + * + * Assign this variable to the codec_dev field of the machine driver's + * snd_soc_device structure. */ static const struct snd_soc_codec_driver soc_codec_device_cs4270 = { .probe = cs4270_probe, diff --git a/trunk/sound/soc/codecs/max98088.c b/trunk/sound/soc/codecs/max98088.c index ac65a2d36408..4173b67c94d1 100644 --- a/trunk/sound/soc/codecs/max98088.c +++ b/trunk/sound/soc/codecs/max98088.c @@ -1397,6 +1397,8 @@ static int max98088_dai_set_sysclk(struct snd_soc_dai *dai, if (freq == max98088->sysclk) return 0; + max98088->sysclk = freq; /* remember current sysclk */ + /* Setup clocks for slave mode, and using the PLL * PSCLK = 0x01 (when master clk is 10MHz to 20MHz) * 0x02 (when master clk is 20MHz to 30MHz).. diff --git a/trunk/sound/soc/codecs/max98095.c b/trunk/sound/soc/codecs/max98095.c index 668434d44303..e1d282d477da 100644 --- a/trunk/sound/soc/codecs/max98095.c +++ b/trunk/sound/soc/codecs/max98095.c @@ -1517,6 +1517,8 @@ static int max98095_dai_set_sysclk(struct snd_soc_dai *dai, if (freq == max98095->sysclk) return 0; + max98095->sysclk = freq; /* remember current sysclk */ + /* Setup clocks for slave mode, and using the PLL * PSCLK = 0x01 (when master clk is 10MHz to 20MHz) * 0x02 (when master clk is 20MHz to 40MHz).. @@ -2259,11 +2261,11 @@ static int max98095_probe(struct snd_soc_codec *codec) ret = snd_soc_read(codec, M98095_0FF_REV_ID); if (ret < 0) { - dev_err(codec->dev, "Failure reading hardware revision: %d\n", + dev_err(codec->dev, "Failed to read device revision: %d\n", ret); goto err_access; } - dev_info(codec->dev, "Hardware revision: %c\n", ret - 0x40 + 'A'); + dev_info(codec->dev, "revision %c\n", ret + 'A'); snd_soc_write(codec, M98095_097_PWR_SYS, M98095_PWRSV); @@ -2340,8 +2342,8 @@ static int max98095_i2c_probe(struct i2c_client *i2c, max98095->control_data = i2c; max98095->pdata = i2c->dev.platform_data; - ret = snd_soc_register_codec(&i2c->dev, &soc_codec_dev_max98095, - max98095_dai, ARRAY_SIZE(max98095_dai)); + ret = snd_soc_register_codec(&i2c->dev, + &soc_codec_dev_max98095, &max98095_dai[0], 3); if (ret < 0) kfree(max98095); return ret; diff --git a/trunk/sound/soc/codecs/sta32x.c b/trunk/sound/soc/codecs/sta32x.c deleted file mode 100644 index 409d89d1f34c..000000000000 --- a/trunk/sound/soc/codecs/sta32x.c +++ /dev/null @@ -1,917 +0,0 @@ -/* - * Codec driver for ST STA32x 2.1-channel high-efficiency digital audio system - * - * Copyright: 2011 Raumfeld GmbH - * Author: Johannes Stezenbach - * - * based on code from: - * Wolfson Microelectronics PLC. - * Mark Brown - * Freescale Semiconductor, Inc. - * Timur Tabi - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - */ - -#define pr_fmt(fmt) KBUILD_MODNAME ":%s:%d: " fmt, __func__, __LINE__ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "sta32x.h" - -#define STA32X_RATES (SNDRV_PCM_RATE_32000 | \ - SNDRV_PCM_RATE_44100 | \ - SNDRV_PCM_RATE_48000 | \ - SNDRV_PCM_RATE_88200 | \ - SNDRV_PCM_RATE_96000 | \ - SNDRV_PCM_RATE_176400 | \ - SNDRV_PCM_RATE_192000) - -#define STA32X_FORMATS \ - (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S16_BE | \ - SNDRV_PCM_FMTBIT_S18_3LE | SNDRV_PCM_FMTBIT_S18_3BE | \ - SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S20_3BE | \ - SNDRV_PCM_FMTBIT_S24_3LE | SNDRV_PCM_FMTBIT_S24_3BE | \ - SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S24_BE | \ - SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S32_BE) - -/* Power-up register defaults */ -static const u8 sta32x_regs[STA32X_REGISTER_COUNT] = { - 0x63, 0x80, 0xc2, 0x40, 0xc2, 0x5c, 0x10, 0xff, 0x60, 0x60, - 0x60, 0x80, 0x00, 0x00, 0x00, 0x40, 0x80, 0x77, 0x6a, 0x69, - 0x6a, 0x69, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x2d, - 0xc0, 0xf3, 0x33, 0x00, 0x0c, -}; - -/* regulator power supply names */ -static const char *sta32x_supply_names[] = { - "Vdda", /* analog supply, 3.3VV */ - "Vdd3", /* digital supply, 3.3V */ - "Vcc" /* power amp spply, 10V - 36V */ -}; - -/* codec private data */ -struct sta32x_priv { - struct regulator_bulk_data supplies[ARRAY_SIZE(sta32x_supply_names)]; - struct snd_soc_codec *codec; - - unsigned int mclk; - unsigned int format; -}; - -static const DECLARE_TLV_DB_SCALE(mvol_tlv, -12700, 50, 1); -static const DECLARE_TLV_DB_SCALE(chvol_tlv, -7950, 50, 1); -static const DECLARE_TLV_DB_SCALE(tone_tlv, -120, 200, 0); - -static const char *sta32x_drc_ac[] = { - "Anti-Clipping", "Dynamic Range Compression" }; -static const char *sta32x_auto_eq_mode[] = { - "User", "Preset", "Loudness" }; -static const char *sta32x_auto_gc_mode[] = { - "User", "AC no clipping", "AC limited clipping (10%)", - "DRC nighttime listening mode" }; -static const char *sta32x_auto_xo_mode[] = { - "User", "80Hz", "100Hz", "120Hz", "140Hz", "160Hz", "180Hz", "200Hz", - "220Hz", "240Hz", "260Hz", "280Hz", "300Hz", "320Hz", "340Hz", "360Hz" }; -static const char *sta32x_preset_eq_mode[] = { - "Flat", "Rock", "Soft Rock", "Jazz", "Classical", "Dance", "Pop", "Soft", - "Hard", "Party", "Vocal", "Hip-Hop", "Dialog", "Bass-boost #1", - "Bass-boost #2", "Bass-boost #3", "Loudness 1", "Loudness 2", - "Loudness 3", "Loudness 4", "Loudness 5", "Loudness 6", "Loudness 7", - "Loudness 8", "Loudness 9", "Loudness 10", "Loudness 11", "Loudness 12", - "Loudness 13", "Loudness 14", "Loudness 15", "Loudness 16" }; -static const char *sta32x_limiter_select[] = { - "Limiter Disabled", "Limiter #1", "Limiter #2" }; -static const char *sta32x_limiter_attack_rate[] = { - "3.1584", "2.7072", "2.2560", "1.8048", "1.3536", "0.9024", - "0.4512", "0.2256", "0.1504", "0.1123", "0.0902", "0.0752", - "0.0645", "0.0564", "0.0501", "0.0451" }; -static const char *sta32x_limiter_release_rate[] = { - "0.5116", "0.1370", "0.0744", "0.0499", "0.0360", "0.0299", - "0.0264", "0.0208", "0.0198", "0.0172", "0.0147", "0.0137", - "0.0134", "0.0117", "0.0110", "0.0104" }; - -static const unsigned int sta32x_limiter_ac_attack_tlv[] = { - TLV_DB_RANGE_HEAD(2), - 0, 7, TLV_DB_SCALE_ITEM(-1200, 200, 0), - 8, 16, TLV_DB_SCALE_ITEM(300, 100, 0), -}; - -static const unsigned int sta32x_limiter_ac_release_tlv[] = { - TLV_DB_RANGE_HEAD(5), - 0, 0, TLV_DB_SCALE_ITEM(TLV_DB_GAIN_MUTE, 0, 0), - 1, 1, TLV_DB_SCALE_ITEM(-2900, 0, 0), - 2, 2, TLV_DB_SCALE_ITEM(-2000, 0, 0), - 3, 8, TLV_DB_SCALE_ITEM(-1400, 200, 0), - 8, 16, TLV_DB_SCALE_ITEM(-700, 100, 0), -}; - -static const unsigned int sta32x_limiter_drc_attack_tlv[] = { - TLV_DB_RANGE_HEAD(3), - 0, 7, TLV_DB_SCALE_ITEM(-3100, 200, 0), - 8, 13, TLV_DB_SCALE_ITEM(-1600, 100, 0), - 14, 16, TLV_DB_SCALE_ITEM(-1000, 300, 0), -}; - -static const unsigned int sta32x_limiter_drc_release_tlv[] = { - TLV_DB_RANGE_HEAD(5), - 0, 0, TLV_DB_SCALE_ITEM(TLV_DB_GAIN_MUTE, 0, 0), - 1, 2, TLV_DB_SCALE_ITEM(-3800, 200, 0), - 3, 4, TLV_DB_SCALE_ITEM(-3300, 200, 0), - 5, 12, TLV_DB_SCALE_ITEM(-3000, 200, 0), - 13, 16, TLV_DB_SCALE_ITEM(-1500, 300, 0), -}; - -static const struct soc_enum sta32x_drc_ac_enum = - SOC_ENUM_SINGLE(STA32X_CONFD, STA32X_CONFD_DRC_SHIFT, - 2, sta32x_drc_ac); -static const struct soc_enum sta32x_auto_eq_enum = - SOC_ENUM_SINGLE(STA32X_AUTO1, STA32X_AUTO1_AMEQ_SHIFT, - 3, sta32x_auto_eq_mode); -static const struct soc_enum sta32x_auto_gc_enum = - SOC_ENUM_SINGLE(STA32X_AUTO1, STA32X_AUTO1_AMGC_SHIFT, - 4, sta32x_auto_gc_mode); -static const struct soc_enum sta32x_auto_xo_enum = - SOC_ENUM_SINGLE(STA32X_AUTO2, STA32X_AUTO2_XO_SHIFT, - 16, sta32x_auto_xo_mode); -static const struct soc_enum sta32x_preset_eq_enum = - SOC_ENUM_SINGLE(STA32X_AUTO3, STA32X_AUTO3_PEQ_SHIFT, - 32, sta32x_preset_eq_mode); -static const struct soc_enum sta32x_limiter_ch1_enum = - SOC_ENUM_SINGLE(STA32X_C1CFG, STA32X_CxCFG_LS_SHIFT, - 3, sta32x_limiter_select); -static const struct soc_enum sta32x_limiter_ch2_enum = - SOC_ENUM_SINGLE(STA32X_C2CFG, STA32X_CxCFG_LS_SHIFT, - 3, sta32x_limiter_select); -static const struct soc_enum sta32x_limiter_ch3_enum = - SOC_ENUM_SINGLE(STA32X_C3CFG, STA32X_CxCFG_LS_SHIFT, - 3, sta32x_limiter_select); -static const struct soc_enum sta32x_limiter1_attack_rate_enum = - SOC_ENUM_SINGLE(STA32X_L1AR, STA32X_LxA_SHIFT, - 16, sta32x_limiter_attack_rate); -static const struct soc_enum sta32x_limiter2_attack_rate_enum = - SOC_ENUM_SINGLE(STA32X_L2AR, STA32X_LxA_SHIFT, - 16, sta32x_limiter_attack_rate); -static const struct soc_enum sta32x_limiter1_release_rate_enum = - SOC_ENUM_SINGLE(STA32X_L1AR, STA32X_LxR_SHIFT, - 16, sta32x_limiter_release_rate); -static const struct soc_enum sta32x_limiter2_release_rate_enum = - SOC_ENUM_SINGLE(STA32X_L2AR, STA32X_LxR_SHIFT, - 16, sta32x_limiter_release_rate); - -/* byte array controls for setting biquad, mixer, scaling coefficients; - * for biquads all five coefficients need to be set in one go, - * mixer and pre/postscale coefs can be set individually; - * each coef is 24bit, the bytes are ordered in the same way - * as given in the STA32x data sheet (big endian; b1, b2, a1, a2, b0) - */ - -static int sta32x_coefficient_info(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_info *uinfo) -{ - int numcoef = kcontrol->private_value >> 16; - uinfo->type = SNDRV_CTL_ELEM_TYPE_BYTES; - uinfo->count = 3 * numcoef; - return 0; -} - -static int sta32x_coefficient_get(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol) -{ - struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); - int numcoef = kcontrol->private_value >> 16; - int index = kcontrol->private_value & 0xffff; - unsigned int cfud; - int i; - - /* preserve reserved bits in STA32X_CFUD */ - cfud = snd_soc_read(codec, STA32X_CFUD) & 0xf0; - /* chip documentation does not say if the bits are self clearing, - * so do it explicitly */ - snd_soc_write(codec, STA32X_CFUD, cfud); - - snd_soc_write(codec, STA32X_CFADDR2, index); - if (numcoef == 1) - snd_soc_write(codec, STA32X_CFUD, cfud | 0x04); - else if (numcoef == 5) - snd_soc_write(codec, STA32X_CFUD, cfud | 0x08); - else - return -EINVAL; - for (i = 0; i < 3 * numcoef; i++) - ucontrol->value.bytes.data[i] = - snd_soc_read(codec, STA32X_B1CF1 + i); - - return 0; -} - -static int sta32x_coefficient_put(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol) -{ - struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); - int numcoef = kcontrol->private_value >> 16; - int index = kcontrol->private_value & 0xffff; - unsigned int cfud; - int i; - - /* preserve reserved bits in STA32X_CFUD */ - cfud = snd_soc_read(codec, STA32X_CFUD) & 0xf0; - /* chip documentation does not say if the bits are self clearing, - * so do it explicitly */ - snd_soc_write(codec, STA32X_CFUD, cfud); - - snd_soc_write(codec, STA32X_CFADDR2, index); - for (i = 0; i < 3 * numcoef; i++) - snd_soc_write(codec, STA32X_B1CF1 + i, - ucontrol->value.bytes.data[i]); - if (numcoef == 1) - snd_soc_write(codec, STA32X_CFUD, cfud | 0x01); - else if (numcoef == 5) - snd_soc_write(codec, STA32X_CFUD, cfud | 0x02); - else - return -EINVAL; - - return 0; -} - -#define SINGLE_COEF(xname, index) \ -{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ - .info = sta32x_coefficient_info, \ - .get = sta32x_coefficient_get,\ - .put = sta32x_coefficient_put, \ - .private_value = index | (1 << 16) } - -#define BIQUAD_COEFS(xname, index) \ -{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ - .info = sta32x_coefficient_info, \ - .get = sta32x_coefficient_get,\ - .put = sta32x_coefficient_put, \ - .private_value = index | (5 << 16) } - -static const struct snd_kcontrol_new sta32x_snd_controls[] = { -SOC_SINGLE_TLV("Master Volume", STA32X_MVOL, 0, 0xff, 1, mvol_tlv), -SOC_SINGLE("Master Switch", STA32X_MMUTE, 0, 1, 1), -SOC_SINGLE("Ch1 Switch", STA32X_MMUTE, 1, 1, 1), -SOC_SINGLE("Ch2 Switch", STA32X_MMUTE, 2, 1, 1), -SOC_SINGLE("Ch3 Switch", STA32X_MMUTE, 3, 1, 1), -SOC_SINGLE_TLV("Ch1 Volume", STA32X_C1VOL, 0, 0xff, 1, chvol_tlv), -SOC_SINGLE_TLV("Ch2 Volume", STA32X_C2VOL, 0, 0xff, 1, chvol_tlv), -SOC_SINGLE_TLV("Ch3 Volume", STA32X_C3VOL, 0, 0xff, 1, chvol_tlv), -SOC_SINGLE("De-emphasis Filter Switch", STA32X_CONFD, STA32X_CONFD_DEMP_SHIFT, 1, 0), -SOC_ENUM("Compressor/Limiter Switch", sta32x_drc_ac_enum), -SOC_SINGLE("Miami Mode Switch", STA32X_CONFD, STA32X_CONFD_MME_SHIFT, 1, 0), -SOC_SINGLE("Zero Cross Switch", STA32X_CONFE, STA32X_CONFE_ZCE_SHIFT, 1, 0), -SOC_SINGLE("Soft Ramp Switch", STA32X_CONFE, STA32X_CONFE_SVE_SHIFT, 1, 0), -SOC_SINGLE("Auto-Mute Switch", STA32X_CONFF, STA32X_CONFF_IDE_SHIFT, 1, 0), -SOC_ENUM("Automode EQ", sta32x_auto_eq_enum), -SOC_ENUM("Automode GC", sta32x_auto_gc_enum), -SOC_ENUM("Automode XO", sta32x_auto_xo_enum), -SOC_ENUM("Preset EQ", sta32x_preset_eq_enum), -SOC_SINGLE("Ch1 Tone Control Bypass Switch", STA32X_C1CFG, STA32X_CxCFG_TCB_SHIFT, 1, 0), -SOC_SINGLE("Ch2 Tone Control Bypass Switch", STA32X_C2CFG, STA32X_CxCFG_TCB_SHIFT, 1, 0), -SOC_SINGLE("Ch1 EQ Bypass Switch", STA32X_C1CFG, STA32X_CxCFG_EQBP_SHIFT, 1, 0), -SOC_SINGLE("Ch2 EQ Bypass Switch", STA32X_C2CFG, STA32X_CxCFG_EQBP_SHIFT, 1, 0), -SOC_SINGLE("Ch1 Master Volume Bypass Switch", STA32X_C1CFG, STA32X_CxCFG_VBP_SHIFT, 1, 0), -SOC_SINGLE("Ch2 Master Volume Bypass Switch", STA32X_C1CFG, STA32X_CxCFG_VBP_SHIFT, 1, 0), -SOC_SINGLE("Ch3 Master Volume Bypass Switch", STA32X_C1CFG, STA32X_CxCFG_VBP_SHIFT, 1, 0), -SOC_ENUM("Ch1 Limiter Select", sta32x_limiter_ch1_enum), -SOC_ENUM("Ch2 Limiter Select", sta32x_limiter_ch2_enum), -SOC_ENUM("Ch3 Limiter Select", sta32x_limiter_ch3_enum), -SOC_SINGLE_TLV("Bass Tone Control", STA32X_TONE, STA32X_TONE_BTC_SHIFT, 15, 0, tone_tlv), -SOC_SINGLE_TLV("Treble Tone Control", STA32X_TONE, STA32X_TONE_TTC_SHIFT, 15, 0, tone_tlv), -SOC_ENUM("Limiter1 Attack Rate (dB/ms)", sta32x_limiter1_attack_rate_enum), -SOC_ENUM("Limiter2 Attack Rate (dB/ms)", sta32x_limiter2_attack_rate_enum), -SOC_ENUM("Limiter1 Release Rate (dB/ms)", sta32x_limiter1_release_rate_enum), -SOC_ENUM("Limiter2 Release Rate (dB/ms)", sta32x_limiter1_release_rate_enum), - -/* depending on mode, the attack/release thresholds have - * two different enum definitions; provide both - */ -SOC_SINGLE_TLV("Limiter1 Attack Threshold (AC Mode)", STA32X_L1ATRT, STA32X_LxA_SHIFT, - 16, 0, sta32x_limiter_ac_attack_tlv), -SOC_SINGLE_TLV("Limiter2 Attack Threshold (AC Mode)", STA32X_L2ATRT, STA32X_LxA_SHIFT, - 16, 0, sta32x_limiter_ac_attack_tlv), -SOC_SINGLE_TLV("Limiter1 Release Threshold (AC Mode)", STA32X_L1ATRT, STA32X_LxR_SHIFT, - 16, 0, sta32x_limiter_ac_release_tlv), -SOC_SINGLE_TLV("Limiter2 Release Threshold (AC Mode)", STA32X_L2ATRT, STA32X_LxR_SHIFT, - 16, 0, sta32x_limiter_ac_release_tlv), -SOC_SINGLE_TLV("Limiter1 Attack Threshold (DRC Mode)", STA32X_L1ATRT, STA32X_LxA_SHIFT, - 16, 0, sta32x_limiter_drc_attack_tlv), -SOC_SINGLE_TLV("Limiter2 Attack Threshold (DRC Mode)", STA32X_L2ATRT, STA32X_LxA_SHIFT, - 16, 0, sta32x_limiter_drc_attack_tlv), -SOC_SINGLE_TLV("Limiter1 Release Threshold (DRC Mode)", STA32X_L1ATRT, STA32X_LxR_SHIFT, - 16, 0, sta32x_limiter_drc_release_tlv), -SOC_SINGLE_TLV("Limiter2 Release Threshold (DRC Mode)", STA32X_L2ATRT, STA32X_LxR_SHIFT, - 16, 0, sta32x_limiter_drc_release_tlv), - -BIQUAD_COEFS("Ch1 - Biquad 1", 0), -BIQUAD_COEFS("Ch1 - Biquad 2", 5), -BIQUAD_COEFS("Ch1 - Biquad 3", 10), -BIQUAD_COEFS("Ch1 - Biquad 4", 15), -BIQUAD_COEFS("Ch2 - Biquad 1", 20), -BIQUAD_COEFS("Ch2 - Biquad 2", 25), -BIQUAD_COEFS("Ch2 - Biquad 3", 30), -BIQUAD_COEFS("Ch2 - Biquad 4", 35), -BIQUAD_COEFS("High-pass", 40), -BIQUAD_COEFS("Low-pass", 45), -SINGLE_COEF("Ch1 - Prescale", 50), -SINGLE_COEF("Ch2 - Prescale", 51), -SINGLE_COEF("Ch1 - Postscale", 52), -SINGLE_COEF("Ch2 - Postscale", 53), -SINGLE_COEF("Ch3 - Postscale", 54), -SINGLE_COEF("Thermal warning - Postscale", 55), -SINGLE_COEF("Ch1 - Mix 1", 56), -SINGLE_COEF("Ch1 - Mix 2", 57), -SINGLE_COEF("Ch2 - Mix 1", 58), -SINGLE_COEF("Ch2 - Mix 2", 59), -SINGLE_COEF("Ch3 - Mix 1", 60), -SINGLE_COEF("Ch3 - Mix 2", 61), -}; - -static const struct snd_soc_dapm_widget sta32x_dapm_widgets[] = { -SND_SOC_DAPM_DAC("DAC", "Playback", SND_SOC_NOPM, 0, 0), -SND_SOC_DAPM_OUTPUT("LEFT"), -SND_SOC_DAPM_OUTPUT("RIGHT"), -SND_SOC_DAPM_OUTPUT("SUB"), -}; - -static const struct snd_soc_dapm_route sta32x_dapm_routes[] = { - { "LEFT", NULL, "DAC" }, - { "RIGHT", NULL, "DAC" }, - { "SUB", NULL, "DAC" }, -}; - -/* MCLK interpolation ratio per fs */ -static struct { - int fs; - int ir; -} interpolation_ratios[] = { - { 32000, 0 }, - { 44100, 0 }, - { 48000, 0 }, - { 88200, 1 }, - { 96000, 1 }, - { 176400, 2 }, - { 192000, 2 }, -}; - -/* MCLK to fs clock ratios */ -static struct { - int ratio; - int mcs; -} mclk_ratios[3][7] = { - { { 768, 0 }, { 512, 1 }, { 384, 2 }, { 256, 3 }, - { 128, 4 }, { 576, 5 }, { 0, 0 } }, - { { 384, 2 }, { 256, 3 }, { 192, 4 }, { 128, 5 }, {64, 0 }, { 0, 0 } }, - { { 384, 2 }, { 256, 3 }, { 192, 4 }, { 128, 5 }, {64, 0 }, { 0, 0 } }, -}; - - -/** - * sta32x_set_dai_sysclk - configure MCLK - * @codec_dai: the codec DAI - * @clk_id: the clock ID (ignored) - * @freq: the MCLK input frequency - * @dir: the clock direction (ignored) - * - * The value of MCLK is used to determine which sample rates are supported - * by the STA32X, based on the mclk_ratios table. - * - * This function must be called by the machine driver's 'startup' function, - * otherwise the list of supported sample rates will not be available in - * time for ALSA. - * - * For setups with variable MCLKs, pass 0 as 'freq' argument. This will cause - * theoretically possible sample rates to be enabled. Call it again with a - * proper value set one the external clock is set (most probably you would do - * that from a machine's driver 'hw_param' hook. - */ -static int sta32x_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 sta32x_priv *sta32x = snd_soc_codec_get_drvdata(codec); - int i, j, ir, fs; - unsigned int rates = 0; - unsigned int rate_min = -1; - unsigned int rate_max = 0; - - pr_debug("mclk=%u\n", freq); - sta32x->mclk = freq; - - if (sta32x->mclk) { - for (i = 0; i < ARRAY_SIZE(interpolation_ratios); i++) { - ir = interpolation_ratios[i].ir; - fs = interpolation_ratios[i].fs; - for (j = 0; mclk_ratios[ir][j].ratio; j++) { - if (mclk_ratios[ir][j].ratio * fs == freq) { - rates |= snd_pcm_rate_to_rate_bit(fs); - if (fs < rate_min) - rate_min = fs; - if (fs > rate_max) - rate_max = fs; - } - } - } - /* FIXME: soc should support a rate list */ - rates &= ~SNDRV_PCM_RATE_KNOT; - - if (!rates) { - dev_err(codec->dev, "could not find a valid sample rate\n"); - return -EINVAL; - } - } else { - /* enable all possible rates */ - rates = STA32X_RATES; - rate_min = 32000; - rate_max = 192000; - } - - codec_dai->driver->playback.rates = rates; - codec_dai->driver->playback.rate_min = rate_min; - codec_dai->driver->playback.rate_max = rate_max; - return 0; -} - -/** - * sta32x_set_dai_fmt - configure the codec for the selected audio format - * @codec_dai: the codec DAI - * @fmt: a SND_SOC_DAIFMT_x value indicating the data format - * - * This function takes a bitmask of SND_SOC_DAIFMT_x bits and programs the - * codec accordingly. - */ -static int sta32x_set_dai_fmt(struct snd_soc_dai *codec_dai, - unsigned int fmt) -{ - struct snd_soc_codec *codec = codec_dai->codec; - struct sta32x_priv *sta32x = snd_soc_codec_get_drvdata(codec); - u8 confb = snd_soc_read(codec, STA32X_CONFB); - - pr_debug("\n"); - confb &= ~(STA32X_CONFB_C1IM | STA32X_CONFB_C2IM); - - switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { - case SND_SOC_DAIFMT_CBS_CFS: - break; - default: - return -EINVAL; - } - - switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { - case SND_SOC_DAIFMT_I2S: - case SND_SOC_DAIFMT_RIGHT_J: - case SND_SOC_DAIFMT_LEFT_J: - sta32x->format = fmt & SND_SOC_DAIFMT_FORMAT_MASK; - break; - default: - return -EINVAL; - } - - switch (fmt & SND_SOC_DAIFMT_INV_MASK) { - case SND_SOC_DAIFMT_NB_NF: - confb |= STA32X_CONFB_C2IM; - break; - case SND_SOC_DAIFMT_NB_IF: - confb |= STA32X_CONFB_C1IM; - break; - default: - return -EINVAL; - } - - snd_soc_write(codec, STA32X_CONFB, confb); - return 0; -} - -/** - * sta32x_hw_params - program the STA32X with the given hardware parameters. - * @substream: the audio stream - * @params: the hardware parameters to set - * @dai: the SOC DAI (ignored) - * - * This function programs the hardware with the values provided. - * Specifically, the sample rate and the data format. - */ -static int sta32x_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 sta32x_priv *sta32x = snd_soc_codec_get_drvdata(codec); - unsigned int rate; - int i, mcs = -1, ir = -1; - u8 confa, confb; - - rate = params_rate(params); - pr_debug("rate: %u\n", rate); - for (i = 0; i < ARRAY_SIZE(interpolation_ratios); i++) - if (interpolation_ratios[i].fs == rate) - ir = interpolation_ratios[i].ir; - if (ir < 0) - return -EINVAL; - for (i = 0; mclk_ratios[ir][i].ratio; i++) - if (mclk_ratios[ir][i].ratio * rate == sta32x->mclk) - mcs = mclk_ratios[ir][i].mcs; - if (mcs < 0) - return -EINVAL; - - confa = snd_soc_read(codec, STA32X_CONFA); - confa &= ~(STA32X_CONFA_MCS_MASK | STA32X_CONFA_IR_MASK); - confa |= (ir << STA32X_CONFA_IR_SHIFT) | (mcs << STA32X_CONFA_MCS_SHIFT); - - confb = snd_soc_read(codec, STA32X_CONFB); - confb &= ~(STA32X_CONFB_SAI_MASK | STA32X_CONFB_SAIFB); - switch (params_format(params)) { - case SNDRV_PCM_FORMAT_S24_LE: - case SNDRV_PCM_FORMAT_S24_BE: - case SNDRV_PCM_FORMAT_S24_3LE: - case SNDRV_PCM_FORMAT_S24_3BE: - pr_debug("24bit\n"); - /* fall through */ - case SNDRV_PCM_FORMAT_S32_LE: - case SNDRV_PCM_FORMAT_S32_BE: - pr_debug("24bit or 32bit\n"); - switch (sta32x->format) { - case SND_SOC_DAIFMT_I2S: - confb |= 0x0; - break; - case SND_SOC_DAIFMT_LEFT_J: - confb |= 0x1; - break; - case SND_SOC_DAIFMT_RIGHT_J: - confb |= 0x2; - break; - } - - break; - case SNDRV_PCM_FORMAT_S20_3LE: - case SNDRV_PCM_FORMAT_S20_3BE: - pr_debug("20bit\n"); - switch (sta32x->format) { - case SND_SOC_DAIFMT_I2S: - confb |= 0x4; - break; - case SND_SOC_DAIFMT_LEFT_J: - confb |= 0x5; - break; - case SND_SOC_DAIFMT_RIGHT_J: - confb |= 0x6; - break; - } - - break; - case SNDRV_PCM_FORMAT_S18_3LE: - case SNDRV_PCM_FORMAT_S18_3BE: - pr_debug("18bit\n"); - switch (sta32x->format) { - case SND_SOC_DAIFMT_I2S: - confb |= 0x8; - break; - case SND_SOC_DAIFMT_LEFT_J: - confb |= 0x9; - break; - case SND_SOC_DAIFMT_RIGHT_J: - confb |= 0xa; - break; - } - - break; - case SNDRV_PCM_FORMAT_S16_LE: - case SNDRV_PCM_FORMAT_S16_BE: - pr_debug("16bit\n"); - switch (sta32x->format) { - case SND_SOC_DAIFMT_I2S: - confb |= 0x0; - break; - case SND_SOC_DAIFMT_LEFT_J: - confb |= 0xd; - break; - case SND_SOC_DAIFMT_RIGHT_J: - confb |= 0xe; - break; - } - - break; - default: - return -EINVAL; - } - - snd_soc_write(codec, STA32X_CONFA, confa); - snd_soc_write(codec, STA32X_CONFB, confb); - return 0; -} - -/** - * sta32x_set_bias_level - DAPM callback - * @codec: the codec device - * @level: DAPM power level - * - * This is called by ALSA to put the codec into low power mode - * or to wake it up. If the codec is powered off completely - * all registers must be restored after power on. - */ -static int sta32x_set_bias_level(struct snd_soc_codec *codec, - enum snd_soc_bias_level level) -{ - int ret; - struct sta32x_priv *sta32x = snd_soc_codec_get_drvdata(codec); - - pr_debug("level = %d\n", level); - switch (level) { - case SND_SOC_BIAS_ON: - break; - - case SND_SOC_BIAS_PREPARE: - /* Full power on */ - snd_soc_update_bits(codec, STA32X_CONFF, - STA32X_CONFF_PWDN | STA32X_CONFF_EAPD, - STA32X_CONFF_PWDN | STA32X_CONFF_EAPD); - break; - - case SND_SOC_BIAS_STANDBY: - if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) { - ret = regulator_bulk_enable(ARRAY_SIZE(sta32x->supplies), - sta32x->supplies); - if (ret != 0) { - dev_err(codec->dev, - "Failed to enable supplies: %d\n", ret); - return ret; - } - - snd_soc_cache_sync(codec); - } - - /* Power up to mute */ - /* FIXME */ - snd_soc_update_bits(codec, STA32X_CONFF, - STA32X_CONFF_PWDN | STA32X_CONFF_EAPD, - STA32X_CONFF_PWDN | STA32X_CONFF_EAPD); - - break; - - case SND_SOC_BIAS_OFF: - /* The chip runs through the power down sequence for us. */ - snd_soc_update_bits(codec, STA32X_CONFF, - STA32X_CONFF_PWDN | STA32X_CONFF_EAPD, - STA32X_CONFF_PWDN); - msleep(300); - - regulator_bulk_disable(ARRAY_SIZE(sta32x->supplies), - sta32x->supplies); - break; - } - codec->dapm.bias_level = level; - return 0; -} - -static struct snd_soc_dai_ops sta32x_dai_ops = { - .hw_params = sta32x_hw_params, - .set_sysclk = sta32x_set_dai_sysclk, - .set_fmt = sta32x_set_dai_fmt, -}; - -static struct snd_soc_dai_driver sta32x_dai = { - .name = "STA32X", - .playback = { - .stream_name = "Playback", - .channels_min = 2, - .channels_max = 2, - .rates = STA32X_RATES, - .formats = STA32X_FORMATS, - }, - .ops = &sta32x_dai_ops, -}; - -#ifdef CONFIG_PM -static int sta32x_suspend(struct snd_soc_codec *codec, pm_message_t state) -{ - sta32x_set_bias_level(codec, SND_SOC_BIAS_OFF); - return 0; -} - -static int sta32x_resume(struct snd_soc_codec *codec) -{ - sta32x_set_bias_level(codec, SND_SOC_BIAS_STANDBY); - return 0; -} -#else -#define sta32x_suspend NULL -#define sta32x_resume NULL -#endif - -static int sta32x_probe(struct snd_soc_codec *codec) -{ - struct sta32x_priv *sta32x = snd_soc_codec_get_drvdata(codec); - int i, ret = 0; - - sta32x->codec = codec; - - /* regulators */ - for (i = 0; i < ARRAY_SIZE(sta32x->supplies); i++) - sta32x->supplies[i].supply = sta32x_supply_names[i]; - - ret = regulator_bulk_get(codec->dev, ARRAY_SIZE(sta32x->supplies), - sta32x->supplies); - if (ret != 0) { - dev_err(codec->dev, "Failed to request supplies: %d\n", ret); - goto err; - } - - ret = regulator_bulk_enable(ARRAY_SIZE(sta32x->supplies), - sta32x->supplies); - if (ret != 0) { - dev_err(codec->dev, "Failed to enable supplies: %d\n", ret); - goto err_get; - } - - /* Tell ASoC what kind of I/O to use to read the registers. ASoC will - * then do the I2C transactions itself. - */ - ret = snd_soc_codec_set_cache_io(codec, 8, 8, SND_SOC_I2C); - if (ret < 0) { - dev_err(codec->dev, "failed to set cache I/O (ret=%i)\n", ret); - return ret; - } - - /* read reg reset values into cache */ - for (i = 0; i < STA32X_REGISTER_COUNT; i++) - snd_soc_cache_write(codec, i, sta32x_regs[i]); - - /* preserve reset values of reserved register bits */ - snd_soc_cache_write(codec, STA32X_CONFC, - codec->hw_read(codec, STA32X_CONFC)); - snd_soc_cache_write(codec, STA32X_CONFE, - codec->hw_read(codec, STA32X_CONFE)); - snd_soc_cache_write(codec, STA32X_CONFF, - codec->hw_read(codec, STA32X_CONFF)); - snd_soc_cache_write(codec, STA32X_MMUTE, - codec->hw_read(codec, STA32X_MMUTE)); - snd_soc_cache_write(codec, STA32X_AUTO1, - codec->hw_read(codec, STA32X_AUTO1)); - snd_soc_cache_write(codec, STA32X_AUTO3, - codec->hw_read(codec, STA32X_AUTO3)); - snd_soc_cache_write(codec, STA32X_C3CFG, - codec->hw_read(codec, STA32X_C3CFG)); - - /* FIXME enable thermal warning adjustment and recovery */ - snd_soc_update_bits(codec, STA32X_CONFA, - STA32X_CONFA_TWAB | STA32X_CONFA_TWRB, 0); - - /* FIXME select 2.1 mode */ - snd_soc_update_bits(codec, STA32X_CONFF, - STA32X_CONFF_OCFG_MASK, - 1 << STA32X_CONFF_OCFG_SHIFT); - - /* FIXME channel to output mapping */ - snd_soc_update_bits(codec, STA32X_C1CFG, - STA32X_CxCFG_OM_MASK, - 0 << STA32X_CxCFG_OM_SHIFT); - snd_soc_update_bits(codec, STA32X_C2CFG, - STA32X_CxCFG_OM_MASK, - 1 << STA32X_CxCFG_OM_SHIFT); - snd_soc_update_bits(codec, STA32X_C3CFG, - STA32X_CxCFG_OM_MASK, - 2 << STA32X_CxCFG_OM_SHIFT); - - sta32x_set_bias_level(codec, SND_SOC_BIAS_STANDBY); - /* Bias level configuration will have done an extra enable */ - regulator_bulk_disable(ARRAY_SIZE(sta32x->supplies), sta32x->supplies); - - return 0; - -err_get: - regulator_bulk_free(ARRAY_SIZE(sta32x->supplies), sta32x->supplies); -err: - return ret; -} - -static int sta32x_remove(struct snd_soc_codec *codec) -{ - struct sta32x_priv *sta32x = snd_soc_codec_get_drvdata(codec); - - regulator_bulk_disable(ARRAY_SIZE(sta32x->supplies), sta32x->supplies); - regulator_bulk_free(ARRAY_SIZE(sta32x->supplies), sta32x->supplies); - - return 0; -} - -static int sta32x_reg_is_volatile(struct snd_soc_codec *codec, - unsigned int reg) -{ - switch (reg) { - case STA32X_CONFA ... STA32X_L2ATRT: - case STA32X_MPCC1 ... STA32X_FDRC2: - return 0; - } - return 1; -} - -static const struct snd_soc_codec_driver sta32x_codec = { - .probe = sta32x_probe, - .remove = sta32x_remove, - .suspend = sta32x_suspend, - .resume = sta32x_resume, - .reg_cache_size = STA32X_REGISTER_COUNT, - .reg_word_size = sizeof(u8), - .volatile_register = sta32x_reg_is_volatile, - .set_bias_level = sta32x_set_bias_level, - .controls = sta32x_snd_controls, - .num_controls = ARRAY_SIZE(sta32x_snd_controls), - .dapm_widgets = sta32x_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(sta32x_dapm_widgets), - .dapm_routes = sta32x_dapm_routes, - .num_dapm_routes = ARRAY_SIZE(sta32x_dapm_routes), -}; - -static __devinit int sta32x_i2c_probe(struct i2c_client *i2c, - const struct i2c_device_id *id) -{ - struct sta32x_priv *sta32x; - int ret; - - sta32x = kzalloc(sizeof(struct sta32x_priv), GFP_KERNEL); - if (!sta32x) - return -ENOMEM; - - i2c_set_clientdata(i2c, sta32x); - - ret = snd_soc_register_codec(&i2c->dev, &sta32x_codec, &sta32x_dai, 1); - if (ret != 0) { - dev_err(&i2c->dev, "Failed to register codec (%d)\n", ret); - return ret; - } - - return 0; -} - -static __devexit int sta32x_i2c_remove(struct i2c_client *client) -{ - struct sta32x_priv *sta32x = i2c_get_clientdata(client); - struct snd_soc_codec *codec = sta32x->codec; - - if (codec) - sta32x_set_bias_level(codec, SND_SOC_BIAS_OFF); - - regulator_bulk_free(ARRAY_SIZE(sta32x->supplies), sta32x->supplies); - - if (codec) { - snd_soc_unregister_codec(&client->dev); - snd_soc_codec_set_drvdata(codec, NULL); - } - - kfree(sta32x); - return 0; -} - -static const struct i2c_device_id sta32x_i2c_id[] = { - { "sta326", 0 }, - { "sta328", 0 }, - { "sta329", 0 }, - { } -}; -MODULE_DEVICE_TABLE(i2c, sta32x_i2c_id); - -static struct i2c_driver sta32x_i2c_driver = { - .driver = { - .name = "sta32x", - .owner = THIS_MODULE, - }, - .probe = sta32x_i2c_probe, - .remove = __devexit_p(sta32x_i2c_remove), - .id_table = sta32x_i2c_id, -}; - -static int __init sta32x_init(void) -{ - return i2c_add_driver(&sta32x_i2c_driver); -} -module_init(sta32x_init); - -static void __exit sta32x_exit(void) -{ - i2c_del_driver(&sta32x_i2c_driver); -} -module_exit(sta32x_exit); - -MODULE_DESCRIPTION("ASoC STA32X driver"); -MODULE_AUTHOR("Johannes Stezenbach "); -MODULE_LICENSE("GPL"); diff --git a/trunk/sound/soc/codecs/sta32x.h b/trunk/sound/soc/codecs/sta32x.h deleted file mode 100644 index b97ee5a75667..000000000000 --- a/trunk/sound/soc/codecs/sta32x.h +++ /dev/null @@ -1,210 +0,0 @@ -/* - * Codec driver for ST STA32x 2.1-channel high-efficiency digital audio system - * - * Copyright: 2011 Raumfeld GmbH - * Author: Johannes Stezenbach - * - * based on code from: - * Wolfson Microelectronics PLC. - * Mark Brown - * - * 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. - */ -#ifndef _ASOC_STA_32X_H -#define _ASOC_STA_32X_H - -/* STA326 register addresses */ - -#define STA32X_REGISTER_COUNT 0x2d - -#define STA32X_CONFA 0x00 -#define STA32X_CONFB 0x01 -#define STA32X_CONFC 0x02 -#define STA32X_CONFD 0x03 -#define STA32X_CONFE 0x04 -#define STA32X_CONFF 0x05 -#define STA32X_MMUTE 0x06 -#define STA32X_MVOL 0x07 -#define STA32X_C1VOL 0x08 -#define STA32X_C2VOL 0x09 -#define STA32X_C3VOL 0x0a -#define STA32X_AUTO1 0x0b -#define STA32X_AUTO2 0x0c -#define STA32X_AUTO3 0x0d -#define STA32X_C1CFG 0x0e -#define STA32X_C2CFG 0x0f -#define STA32X_C3CFG 0x10 -#define STA32X_TONE 0x11 -#define STA32X_L1AR 0x12 -#define STA32X_L1ATRT 0x13 -#define STA32X_L2AR 0x14 -#define STA32X_L2ATRT 0x15 -#define STA32X_CFADDR2 0x16 -#define STA32X_B1CF1 0x17 -#define STA32X_B1CF2 0x18 -#define STA32X_B1CF3 0x19 -#define STA32X_B2CF1 0x1a -#define STA32X_B2CF2 0x1b -#define STA32X_B2CF3 0x1c -#define STA32X_A1CF1 0x1d -#define STA32X_A1CF2 0x1e -#define STA32X_A1CF3 0x1f -#define STA32X_A2CF1 0x20 -#define STA32X_A2CF2 0x21 -#define STA32X_A2CF3 0x22 -#define STA32X_B0CF1 0x23 -#define STA32X_B0CF2 0x24 -#define STA32X_B0CF3 0x25 -#define STA32X_CFUD 0x26 -#define STA32X_MPCC1 0x27 -#define STA32X_MPCC2 0x28 -/* Reserved 0x29 */ -/* Reserved 0x2a */ -#define STA32X_Reserved 0x2a -#define STA32X_FDRC1 0x2b -#define STA32X_FDRC2 0x2c -/* Reserved 0x2d */ - - -/* STA326 register field definitions */ - -/* 0x00 CONFA */ -#define STA32X_CONFA_MCS_MASK 0x03 -#define STA32X_CONFA_MCS_SHIFT 0 -#define STA32X_CONFA_IR_MASK 0x18 -#define STA32X_CONFA_IR_SHIFT 3 -#define STA32X_CONFA_TWRB 0x20 -#define STA32X_CONFA_TWAB 0x40 -#define STA32X_CONFA_FDRB 0x80 - -/* 0x01 CONFB */ -#define STA32X_CONFB_SAI_MASK 0x0f -#define STA32X_CONFB_SAI_SHIFT 0 -#define STA32X_CONFB_SAIFB 0x10 -#define STA32X_CONFB_DSCKE 0x20 -#define STA32X_CONFB_C1IM 0x40 -#define STA32X_CONFB_C2IM 0x80 - -/* 0x02 CONFC */ -#define STA32X_CONFC_OM_MASK 0x03 -#define STA32X_CONFC_OM_SHIFT 0 -#define STA32X_CONFC_CSZ_MASK 0x7c -#define STA32X_CONFC_CSZ_SHIFT 2 - -/* 0x03 CONFD */ -#define STA32X_CONFD_HPB 0x01 -#define STA32X_CONFD_HPB_SHIFT 0 -#define STA32X_CONFD_DEMP 0x02 -#define STA32X_CONFD_DEMP_SHIFT 1 -#define STA32X_CONFD_DSPB 0x04 -#define STA32X_CONFD_DSPB_SHIFT 2 -#define STA32X_CONFD_PSL 0x08 -#define STA32X_CONFD_PSL_SHIFT 3 -#define STA32X_CONFD_BQL 0x10 -#define STA32X_CONFD_BQL_SHIFT 4 -#define STA32X_CONFD_DRC 0x20 -#define STA32X_CONFD_DRC_SHIFT 5 -#define STA32X_CONFD_ZDE 0x40 -#define STA32X_CONFD_ZDE_SHIFT 6 -#define STA32X_CONFD_MME 0x80 -#define STA32X_CONFD_MME_SHIFT 7 - -/* 0x04 CONFE */ -#define STA32X_CONFE_MPCV 0x01 -#define STA32X_CONFE_MPCV_SHIFT 0 -#define STA32X_CONFE_MPC 0x02 -#define STA32X_CONFE_MPC_SHIFT 1 -#define STA32X_CONFE_AME 0x08 -#define STA32X_CONFE_AME_SHIFT 3 -#define STA32X_CONFE_PWMS 0x10 -#define STA32X_CONFE_PWMS_SHIFT 4 -#define STA32X_CONFE_ZCE 0x40 -#define STA32X_CONFE_ZCE_SHIFT 6 -#define STA32X_CONFE_SVE 0x80 -#define STA32X_CONFE_SVE_SHIFT 7 - -/* 0x05 CONFF */ -#define STA32X_CONFF_OCFG_MASK 0x03 -#define STA32X_CONFF_OCFG_SHIFT 0 -#define STA32X_CONFF_IDE 0x04 -#define STA32X_CONFF_IDE_SHIFT 3 -#define STA32X_CONFF_BCLE 0x08 -#define STA32X_CONFF_ECLE 0x20 -#define STA32X_CONFF_PWDN 0x40 -#define STA32X_CONFF_EAPD 0x80 - -/* 0x06 MMUTE */ -#define STA32X_MMUTE_MMUTE 0x01 - -/* 0x0b AUTO1 */ -#define STA32X_AUTO1_AMEQ_MASK 0x03 -#define STA32X_AUTO1_AMEQ_SHIFT 0 -#define STA32X_AUTO1_AMV_MASK 0xc0 -#define STA32X_AUTO1_AMV_SHIFT 2 -#define STA32X_AUTO1_AMGC_MASK 0x30 -#define STA32X_AUTO1_AMGC_SHIFT 4 -#define STA32X_AUTO1_AMPS 0x80 - -/* 0x0c AUTO2 */ -#define STA32X_AUTO2_AMAME 0x01 -#define STA32X_AUTO2_AMAM_MASK 0x0e -#define STA32X_AUTO2_AMAM_SHIFT 1 -#define STA32X_AUTO2_XO_MASK 0xf0 -#define STA32X_AUTO2_XO_SHIFT 4 - -/* 0x0d AUTO3 */ -#define STA32X_AUTO3_PEQ_MASK 0x1f -#define STA32X_AUTO3_PEQ_SHIFT 0 - -/* 0x0e 0x0f 0x10 CxCFG */ -#define STA32X_CxCFG_TCB 0x01 /* only C1 and C2 */ -#define STA32X_CxCFG_TCB_SHIFT 0 -#define STA32X_CxCFG_EQBP 0x02 /* only C1 and C2 */ -#define STA32X_CxCFG_EQBP_SHIFT 1 -#define STA32X_CxCFG_VBP 0x03 -#define STA32X_CxCFG_VBP_SHIFT 2 -#define STA32X_CxCFG_BO 0x04 -#define STA32X_CxCFG_LS_MASK 0x30 -#define STA32X_CxCFG_LS_SHIFT 4 -#define STA32X_CxCFG_OM_MASK 0xc0 -#define STA32X_CxCFG_OM_SHIFT 6 - -/* 0x11 TONE */ -#define STA32X_TONE_BTC_SHIFT 0 -#define STA32X_TONE_TTC_SHIFT 4 - -/* 0x12 0x13 0x14 0x15 limiter attack/release */ -#define STA32X_LxA_SHIFT 0 -#define STA32X_LxR_SHIFT 4 - -/* 0x26 CFUD */ -#define STA32X_CFUD_W1 0x01 -#define STA32X_CFUD_WA 0x02 -#define STA32X_CFUD_R1 0x04 -#define STA32X_CFUD_RA 0x08 - - -/* biquad filter coefficient table offsets */ -#define STA32X_C1_BQ_BASE 0 -#define STA32X_C2_BQ_BASE 20 -#define STA32X_CH_BQ_NUM 4 -#define STA32X_BQ_NUM_COEF 5 -#define STA32X_XO_HP_BQ_BASE 40 -#define STA32X_XO_LP_BQ_BASE 45 -#define STA32X_C1_PRESCALE 50 -#define STA32X_C2_PRESCALE 51 -#define STA32X_C1_POSTSCALE 52 -#define STA32X_C2_POSTSCALE 53 -#define STA32X_C3_POSTSCALE 54 -#define STA32X_TW_POSTSCALE 55 -#define STA32X_C1_MIX1 56 -#define STA32X_C1_MIX2 57 -#define STA32X_C2_MIX1 58 -#define STA32X_C2_MIX2 59 -#define STA32X_C3_MIX1 60 -#define STA32X_C3_MIX2 61 - -#endif /* _ASOC_STA_32X_H */ diff --git a/trunk/sound/soc/codecs/tlv320aic26.c b/trunk/sound/soc/codecs/tlv320aic26.c index 7859bdcc93db..e2a7608d3944 100644 --- a/trunk/sound/soc/codecs/tlv320aic26.c +++ b/trunk/sound/soc/codecs/tlv320aic26.c @@ -161,18 +161,10 @@ static int aic26_hw_params(struct snd_pcm_substream *substream, dev_dbg(&aic26->spi->dev, "bad format\n"); return -EINVAL; } - /** - * Configure PLL - * fsref = (mclk * PLLM) / 2048 - * where PLLM = J.DDDD (DDDD register ranges from 0 to 9999, decimal) - */ + /* Configure PLL */ pval = 1; - /* compute J portion of multiplier */ - jval = fsref / (aic26->mclk / 2048); - /* compute fractional DDDD component of multiplier */ - dval = fsref - (jval * (aic26->mclk / 2048)); - dval = (10000 * dval) / (aic26->mclk / 2048); - dev_dbg(&aic26->spi->dev, "Setting PLLM to %d.%04d\n", jval, dval); + jval = (fsref == 44100) ? 7 : 8; + dval = (fsref == 44100) ? 5264 : 1920; qval = 0; reg = 0x8000 | qval << 11 | pval << 8 | jval << 2; aic26_reg_write(codec, AIC26_REG_PLL_PROG1, reg); diff --git a/trunk/sound/soc/codecs/tlv320aic3x.c b/trunk/sound/soc/codecs/tlv320aic3x.c index 0963c4c7a83f..c3d96fc8c267 100644 --- a/trunk/sound/soc/codecs/tlv320aic3x.c +++ b/trunk/sound/soc/codecs/tlv320aic3x.c @@ -226,13 +226,11 @@ static const char *aic3x_adc_hpf[] = #define RDAC_ENUM 1 #define LHPCOM_ENUM 2 #define RHPCOM_ENUM 3 -#define LINE1L_2_L_ENUM 4 -#define LINE1L_2_R_ENUM 5 -#define LINE1R_2_L_ENUM 6 -#define LINE1R_2_R_ENUM 7 -#define LINE2L_ENUM 8 -#define LINE2R_ENUM 9 -#define ADC_HPF_ENUM 10 +#define LINE1L_ENUM 4 +#define LINE1R_ENUM 5 +#define LINE2L_ENUM 6 +#define LINE2R_ENUM 7 +#define ADC_HPF_ENUM 8 static const struct soc_enum aic3x_enum[] = { SOC_ENUM_SINGLE(DAC_LINE_MUX, 6, 3, aic3x_left_dac_mux), @@ -240,8 +238,6 @@ static const struct soc_enum aic3x_enum[] = { SOC_ENUM_SINGLE(HPLCOM_CFG, 4, 3, aic3x_left_hpcom_mux), SOC_ENUM_SINGLE(HPRCOM_CFG, 3, 5, aic3x_right_hpcom_mux), SOC_ENUM_SINGLE(LINE1L_2_LADC_CTRL, 7, 2, aic3x_linein_mode_mux), - SOC_ENUM_SINGLE(LINE1L_2_RADC_CTRL, 7, 2, aic3x_linein_mode_mux), - SOC_ENUM_SINGLE(LINE1R_2_LADC_CTRL, 7, 2, aic3x_linein_mode_mux), SOC_ENUM_SINGLE(LINE1R_2_RADC_CTRL, 7, 2, aic3x_linein_mode_mux), SOC_ENUM_SINGLE(LINE2L_2_LADC_CTRL, 7, 2, aic3x_linein_mode_mux), SOC_ENUM_SINGLE(LINE2R_2_RADC_CTRL, 7, 2, aic3x_linein_mode_mux), @@ -494,16 +490,12 @@ static const struct snd_kcontrol_new aic3x_right_pga_mixer_controls[] = { }; /* Left Line1 Mux */ -static const struct snd_kcontrol_new aic3x_left_line1l_mux_controls = -SOC_DAPM_ENUM("Route", aic3x_enum[LINE1L_2_L_ENUM]); -static const struct snd_kcontrol_new aic3x_right_line1l_mux_controls = -SOC_DAPM_ENUM("Route", aic3x_enum[LINE1L_2_R_ENUM]); +static const struct snd_kcontrol_new aic3x_left_line1_mux_controls = +SOC_DAPM_ENUM("Route", aic3x_enum[LINE1L_ENUM]); /* Right Line1 Mux */ -static const struct snd_kcontrol_new aic3x_right_line1r_mux_controls = -SOC_DAPM_ENUM("Route", aic3x_enum[LINE1R_2_R_ENUM]); -static const struct snd_kcontrol_new aic3x_left_line1r_mux_controls = -SOC_DAPM_ENUM("Route", aic3x_enum[LINE1R_2_L_ENUM]); +static const struct snd_kcontrol_new aic3x_right_line1_mux_controls = +SOC_DAPM_ENUM("Route", aic3x_enum[LINE1R_ENUM]); /* Left Line2 Mux */ static const struct snd_kcontrol_new aic3x_left_line2_mux_controls = @@ -543,9 +535,9 @@ static const struct snd_soc_dapm_widget aic3x_dapm_widgets[] = { &aic3x_left_pga_mixer_controls[0], ARRAY_SIZE(aic3x_left_pga_mixer_controls)), SND_SOC_DAPM_MUX("Left Line1L Mux", SND_SOC_NOPM, 0, 0, - &aic3x_left_line1l_mux_controls), + &aic3x_left_line1_mux_controls), SND_SOC_DAPM_MUX("Left Line1R Mux", SND_SOC_NOPM, 0, 0, - &aic3x_left_line1r_mux_controls), + &aic3x_left_line1_mux_controls), SND_SOC_DAPM_MUX("Left Line2L Mux", SND_SOC_NOPM, 0, 0, &aic3x_left_line2_mux_controls), @@ -556,9 +548,9 @@ static const struct snd_soc_dapm_widget aic3x_dapm_widgets[] = { &aic3x_right_pga_mixer_controls[0], ARRAY_SIZE(aic3x_right_pga_mixer_controls)), SND_SOC_DAPM_MUX("Right Line1L Mux", SND_SOC_NOPM, 0, 0, - &aic3x_right_line1l_mux_controls), + &aic3x_right_line1_mux_controls), SND_SOC_DAPM_MUX("Right Line1R Mux", SND_SOC_NOPM, 0, 0, - &aic3x_right_line1r_mux_controls), + &aic3x_right_line1_mux_controls), SND_SOC_DAPM_MUX("Right Line2R Mux", SND_SOC_NOPM, 0, 0, &aic3x_right_line2_mux_controls), @@ -1122,19 +1114,12 @@ static int aic3x_set_power(struct snd_soc_codec *codec, int power) /* Sync reg_cache with the hardware */ codec->cache_only = 0; - for (i = AIC3X_SAMPLE_RATE_SEL_REG; i < ARRAY_SIZE(aic3x_reg); i++) + for (i = 0; i < ARRAY_SIZE(aic3x_reg); i++) snd_soc_write(codec, i, cache[i]); if (aic3x->model == AIC3X_MODEL_3007) aic3x_init_3007(codec); codec->cache_sync = 0; } else { - /* - * Do soft reset to this codec instance in order to clear - * possible VDD leakage currents in case the supply regulators - * remain on - */ - snd_soc_write(codec, AIC3X_RESET, SOFT_RESET); - codec->cache_sync = 1; aic3x->power = 0; /* HW writes are needless when bias is off */ codec->cache_only = 1; diff --git a/trunk/sound/soc/codecs/twl6040.c b/trunk/sound/soc/codecs/twl6040.c index cd63bba623df..4c336636d4f5 100644 --- a/trunk/sound/soc/codecs/twl6040.c +++ b/trunk/sound/soc/codecs/twl6040.c @@ -954,9 +954,9 @@ static DECLARE_TLV_DB_SCALE(mic_preamp_tlv, -600, 600, 0); /* * MICGAIN volume control: - * from 6 to 30 dB in 6 dB steps + * from -6 to 30 dB in 6 dB steps */ -static DECLARE_TLV_DB_SCALE(mic_amp_tlv, 600, 600, 0); +static DECLARE_TLV_DB_SCALE(mic_amp_tlv, -600, 600, 0); /* * AFMGAIN volume control: diff --git a/trunk/sound/soc/codecs/wm8731.c b/trunk/sound/soc/codecs/wm8731.c index 76b4361e9b80..2dc964b55e4f 100644 --- a/trunk/sound/soc/codecs/wm8731.c +++ b/trunk/sound/soc/codecs/wm8731.c @@ -175,7 +175,6 @@ static const struct snd_kcontrol_new wm8731_input_mux_controls = SOC_DAPM_ENUM("Input Select", wm8731_insel_enum); static const struct snd_soc_dapm_widget wm8731_dapm_widgets[] = { -SND_SOC_DAPM_SUPPLY("ACTIVE",WM8731_ACTIVE, 0, 0, NULL, 0), SND_SOC_DAPM_SUPPLY("OSC", WM8731_PWR, 5, 1, NULL, 0), SND_SOC_DAPM_MIXER("Output Mixer", WM8731_PWR, 4, 1, &wm8731_output_mixer_controls[0], @@ -205,8 +204,6 @@ static int wm8731_check_osc(struct snd_soc_dapm_widget *source, static const struct snd_soc_dapm_route wm8731_intercon[] = { {"DAC", NULL, "OSC", wm8731_check_osc}, {"ADC", NULL, "OSC", wm8731_check_osc}, - {"DAC", NULL, "ACTIVE"}, - {"ADC", NULL, "ACTIVE"}, /* output mixer */ {"Output Mixer", "Line Bypass Switch", "Line Input"}, @@ -318,6 +315,29 @@ static int wm8731_hw_params(struct snd_pcm_substream *substream, return 0; } +static int wm8731_pcm_prepare(struct snd_pcm_substream *substream, + struct snd_soc_dai *dai) +{ + struct snd_soc_codec *codec = dai->codec; + + /* set active */ + snd_soc_write(codec, WM8731_ACTIVE, 0x0001); + + return 0; +} + +static void wm8731_shutdown(struct snd_pcm_substream *substream, + struct snd_soc_dai *dai) +{ + struct snd_soc_codec *codec = dai->codec; + + /* deactivate */ + if (!codec->active) { + udelay(50); + snd_soc_write(codec, WM8731_ACTIVE, 0x0); + } +} + static int wm8731_mute(struct snd_soc_dai *dai, int mute) { struct snd_soc_codec *codec = dai->codec; @@ -460,6 +480,7 @@ static int wm8731_set_bias_level(struct snd_soc_codec *codec, snd_soc_write(codec, WM8731_PWR, reg | 0x0040); break; case SND_SOC_BIAS_OFF: + snd_soc_write(codec, WM8731_ACTIVE, 0x0); snd_soc_write(codec, WM8731_PWR, 0xffff); regulator_bulk_disable(ARRAY_SIZE(wm8731->supplies), wm8731->supplies); @@ -475,7 +496,9 @@ static int wm8731_set_bias_level(struct snd_soc_codec *codec, SNDRV_PCM_FMTBIT_S24_LE) static struct snd_soc_dai_ops wm8731_dai_ops = { + .prepare = wm8731_pcm_prepare, .hw_params = wm8731_hw_params, + .shutdown = wm8731_shutdown, .digital_mute = wm8731_mute, .set_sysclk = wm8731_set_dai_sysclk, .set_fmt = wm8731_set_dai_fmt, diff --git a/trunk/sound/soc/codecs/wm8782.c b/trunk/sound/soc/codecs/wm8782.c deleted file mode 100644 index a2a09f85ea99..000000000000 --- a/trunk/sound/soc/codecs/wm8782.c +++ /dev/null @@ -1,80 +0,0 @@ -/* - * sound/soc/codecs/wm8782.c - * simple, strap-pin configured 24bit 2ch ADC - * - * Copyright: 2011 Raumfeld GmbH - * Author: Johannes Stezenbach - * - * based on ad73311.c - * Copyright: Analog Device Inc. - * Author: Cliff Cai - * - * 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. - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -static struct snd_soc_dai_driver wm8782_dai = { - .name = "wm8782", - .capture = { - .stream_name = "Capture", - .channels_min = 2, - .channels_max = 2, - /* For configurations with FSAMPEN=0 */ - .rates = SNDRV_PCM_RATE_8000_48000, - .formats = SNDRV_PCM_FMTBIT_S16_LE | - SNDRV_PCM_FMTBIT_S20_3LE | - SNDRV_PCM_FMTBIT_S24_LE, - }, -}; - -static struct snd_soc_codec_driver soc_codec_dev_wm8782; - -static __devinit int wm8782_probe(struct platform_device *pdev) -{ - return snd_soc_register_codec(&pdev->dev, - &soc_codec_dev_wm8782, &wm8782_dai, 1); -} - -static int __devexit wm8782_remove(struct platform_device *pdev) -{ - snd_soc_unregister_codec(&pdev->dev); - return 0; -} - -static struct platform_driver wm8782_codec_driver = { - .driver = { - .name = "wm8782", - .owner = THIS_MODULE, - }, - .probe = wm8782_probe, - .remove = wm8782_remove, -}; - -static int __init wm8782_init(void) -{ - return platform_driver_register(&wm8782_codec_driver); -} -module_init(wm8782_init); - -static void __exit wm8782_exit(void) -{ - platform_driver_unregister(&wm8782_codec_driver); -} -module_exit(wm8782_exit); - -MODULE_DESCRIPTION("ASoC WM8782 driver"); -MODULE_AUTHOR("Johannes Stezenbach "); -MODULE_LICENSE("GPL"); diff --git a/trunk/sound/soc/codecs/wm8900.c b/trunk/sound/soc/codecs/wm8900.c index 082040eda8a2..449ea09a193d 100644 --- a/trunk/sound/soc/codecs/wm8900.c +++ b/trunk/sound/soc/codecs/wm8900.c @@ -1167,7 +1167,6 @@ static int wm8900_resume(struct snd_soc_codec *codec) ret = wm8900_set_fll(codec, 0, fll_in, fll_out); if (ret != 0) { dev_err(codec->dev, "Failed to restart FLL\n"); - kfree(cache); return ret; } } diff --git a/trunk/sound/soc/codecs/wm8904.c b/trunk/sound/soc/codecs/wm8904.c index b085575d4aa5..9b3bba4df5b3 100644 --- a/trunk/sound/soc/codecs/wm8904.c +++ b/trunk/sound/soc/codecs/wm8904.c @@ -2560,7 +2560,6 @@ static __devexit int wm8904_i2c_remove(struct i2c_client *client) static const struct i2c_device_id wm8904_i2c_id[] = { { "wm8904", WM8904 }, { "wm8912", WM8912 }, - { "wm8918", WM8904 }, /* Actually a subset, updates to follow */ { } }; MODULE_DEVICE_TABLE(i2c, wm8904_i2c_id); diff --git a/trunk/sound/soc/codecs/wm8915.c b/trunk/sound/soc/codecs/wm8915.c index 423baa9be241..e2ab4fac2819 100644 --- a/trunk/sound/soc/codecs/wm8915.c +++ b/trunk/sound/soc/codecs/wm8915.c @@ -41,12 +41,14 @@ #define HPOUT2L 4 #define HPOUT2R 8 -#define WM8915_NUM_SUPPLIES 4 +#define WM8915_NUM_SUPPLIES 6 static const char *wm8915_supply_names[WM8915_NUM_SUPPLIES] = { + "DCVDD", "DBVDD", "AVDD1", "AVDD2", "CPVDD", + "MICVDD", }; struct wm8915_priv { @@ -55,7 +57,6 @@ struct wm8915_priv { int ldo1ena; int sysclk; - int sysclk_src; int fll_src; int fll_fref; @@ -75,7 +76,6 @@ struct wm8915_priv { struct wm8915_pdata pdata; int rx_rate[WM8915_AIFS]; - int bclk_rate[WM8915_AIFS]; /* Platform dependant ReTune mobile configuration */ int num_retune_mobile_texts; @@ -113,6 +113,8 @@ WM8915_REGULATOR_EVENT(0) WM8915_REGULATOR_EVENT(1) WM8915_REGULATOR_EVENT(2) WM8915_REGULATOR_EVENT(3) +WM8915_REGULATOR_EVENT(4) +WM8915_REGULATOR_EVENT(5) static const u16 wm8915_reg[WM8915_MAX_REGISTER] = { [WM8915_SOFTWARE_RESET] = 0x8915, @@ -1563,50 +1565,6 @@ static int wm8915_reset(struct snd_soc_codec *codec) return snd_soc_write(codec, WM8915_SOFTWARE_RESET, 0x8915); } -static const int bclk_divs[] = { - 1, 2, 3, 4, 6, 8, 12, 16, 24, 32, 48, 64, 96 -}; - -static void wm8915_update_bclk(struct snd_soc_codec *codec) -{ - struct wm8915_priv *wm8915 = snd_soc_codec_get_drvdata(codec); - int aif, best, cur_val, bclk_rate, bclk_reg, i; - - /* Don't bother if we're in a low frequency idle mode that - * can't support audio. - */ - if (wm8915->sysclk < 64000) - return; - - for (aif = 0; aif < WM8915_AIFS; aif++) { - switch (aif) { - case 0: - bclk_reg = WM8915_AIF1_BCLK; - break; - case 1: - bclk_reg = WM8915_AIF2_BCLK; - break; - } - - bclk_rate = wm8915->bclk_rate[aif]; - - /* Pick a divisor for BCLK as close as we can get to ideal */ - best = 0; - for (i = 0; i < ARRAY_SIZE(bclk_divs); i++) { - cur_val = (wm8915->sysclk / bclk_divs[i]) - bclk_rate; - if (cur_val < 0) /* BCLK table is sorted */ - break; - best = i; - } - bclk_rate = wm8915->sysclk / bclk_divs[best]; - dev_dbg(codec->dev, "Using BCLK_DIV %d for actual BCLK %dHz\n", - bclk_divs[best], bclk_rate); - - snd_soc_update_bits(codec, bclk_reg, - WM8915_AIF1_BCLK_DIV_MASK, best); - } -} - static int wm8915_set_bias_level(struct snd_soc_codec *codec, enum snd_soc_bias_level level) { @@ -1759,6 +1717,10 @@ static int wm8915_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) return 0; } +static const int bclk_divs[] = { + 1, 2, 3, 4, 6, 8, 12, 16, 24, 32, 48, 64, 96 +}; + static const int dsp_divs[] = { 48000, 32000, 16000, 8000 }; @@ -1769,11 +1731,17 @@ static int wm8915_hw_params(struct snd_pcm_substream *substream, { struct snd_soc_codec *codec = dai->codec; struct wm8915_priv *wm8915 = snd_soc_codec_get_drvdata(codec); - int bits, i, bclk_rate; + int bits, i, bclk_rate, best, cur_val; int aifdata = 0; + int bclk = 0; int lrclk = 0; int dsp = 0; - int aifdata_reg, lrclk_reg, dsp_shift; + int aifdata_reg, bclk_reg, lrclk_reg, dsp_shift; + + if (!wm8915->sysclk) { + dev_err(codec->dev, "SYSCLK not configured\n"); + return -EINVAL; + } switch (dai->id) { case 0: @@ -1785,6 +1753,7 @@ static int wm8915_hw_params(struct snd_pcm_substream *substream, aifdata_reg = WM8915_AIF1TX_DATA_CONFIGURATION_1; lrclk_reg = WM8915_AIF1_TX_LRCLK_1; } + bclk_reg = WM8915_AIF1_BCLK; dsp_shift = 0; break; case 1: @@ -1796,6 +1765,7 @@ static int wm8915_hw_params(struct snd_pcm_substream *substream, aifdata_reg = WM8915_AIF2TX_DATA_CONFIGURATION_1; lrclk_reg = WM8915_AIF2_TX_LRCLK_1; } + bclk_reg = WM8915_AIF2_BCLK; dsp_shift = WM8915_DSP2_DIV_SHIFT; break; default: @@ -1809,9 +1779,6 @@ static int wm8915_hw_params(struct snd_pcm_substream *substream, return bclk_rate; } - wm8915->bclk_rate[dai->id] = bclk_rate; - wm8915->rx_rate[dai->id] = params_rate(params); - /* Needs looking at for TDM */ bits = snd_pcm_format_width(params_format(params)); if (bits < 0) @@ -1829,7 +1796,18 @@ static int wm8915_hw_params(struct snd_pcm_substream *substream, } dsp |= i << dsp_shift; - wm8915_update_bclk(codec); + /* Pick a divisor for BCLK as close as we can get to ideal */ + best = 0; + for (i = 0; i < ARRAY_SIZE(bclk_divs); i++) { + cur_val = (wm8915->sysclk / bclk_divs[i]) - bclk_rate; + if (cur_val < 0) /* BCLK table is sorted */ + break; + best = i; + } + bclk_rate = wm8915->sysclk / bclk_divs[best]; + dev_dbg(dai->dev, "Using BCLK_DIV %d for actual BCLK %dHz\n", + bclk_divs[best], bclk_rate); + bclk |= best; lrclk = bclk_rate / params_rate(params); dev_dbg(dai->dev, "Using LRCLK rate %d for actual LRCLK %dHz\n", @@ -1839,11 +1817,14 @@ static int wm8915_hw_params(struct snd_pcm_substream *substream, WM8915_AIF1TX_WL_MASK | WM8915_AIF1TX_SLOT_LEN_MASK, aifdata); + snd_soc_update_bits(codec, bclk_reg, WM8915_AIF1_BCLK_DIV_MASK, bclk); snd_soc_update_bits(codec, lrclk_reg, WM8915_AIF1RX_RATE_MASK, lrclk); snd_soc_update_bits(codec, WM8915_AIF_CLOCKING_2, WM8915_DSP1_DIV_SHIFT << dsp_shift, dsp); + wm8915->rx_rate[dai->id] = params_rate(params); + return 0; } @@ -1857,9 +1838,6 @@ static int wm8915_set_sysclk(struct snd_soc_dai *dai, int src; int old; - if (freq == wm8915->sysclk && clk_id == wm8915->sysclk_src) - return 0; - /* Disable SYSCLK while we reconfigure */ old = snd_soc_read(codec, WM8915_AIF_CLOCKING_1) & WM8915_SYSCLK_ENA; snd_soc_update_bits(codec, WM8915_AIF_CLOCKING_1, @@ -1904,8 +1882,6 @@ static int wm8915_set_sysclk(struct snd_soc_dai *dai, return -EINVAL; } - wm8915_update_bclk(codec); - snd_soc_update_bits(codec, WM8915_AIF_CLOCKING_1, WM8915_SYSCLK_SRC_MASK | WM8915_SYSCLK_DIV_MASK, src << WM8915_SYSCLK_SRC_SHIFT | ratediv); @@ -1913,8 +1889,6 @@ static int wm8915_set_sysclk(struct snd_soc_dai *dai, snd_soc_update_bits(codec, WM8915_AIF_CLOCKING_1, WM8915_SYSCLK_ENA, old); - wm8915->sysclk_src = clk_id; - return 0; } @@ -2033,7 +2007,6 @@ static int wm8915_set_fll(struct snd_soc_codec *codec, int fll_id, int source, unsigned int Fref, unsigned int Fout) { struct wm8915_priv *wm8915 = snd_soc_codec_get_drvdata(codec); - struct i2c_client *i2c = to_i2c_client(codec->dev); struct _fll_div fll_div; unsigned long timeout; int ret, reg; @@ -2120,18 +2093,7 @@ static int wm8915_set_fll(struct snd_soc_codec *codec, int fll_id, int source, else timeout = msecs_to_jiffies(2); - /* Allow substantially longer if we've actually got the IRQ */ - if (i2c->irq) - timeout *= 1000; - - ret = wait_for_completion_timeout(&wm8915->fll_lock, timeout); - - if (ret == 0 && i2c->irq) { - dev_err(codec->dev, "Timed out waiting for FLL\n"); - ret = -ETIMEDOUT; - } else { - ret = 0; - } + wait_for_completion_timeout(&wm8915->fll_lock, timeout); dev_dbg(codec->dev, "FLL configured for %dHz->%dHz\n", Fref, Fout); @@ -2139,7 +2101,7 @@ static int wm8915_set_fll(struct snd_soc_codec *codec, int fll_id, int source, wm8915->fll_fout = Fout; wm8915->fll_src = source; - return ret; + return 0; } #ifdef CONFIG_GPIOLIB @@ -2331,12 +2293,6 @@ static void wm8915_micd(struct snd_soc_codec *codec) SND_JACK_HEADSET | SND_JACK_BTN_0); wm8915->jack_mic = true; wm8915->detecting = false; - - /* Increase poll rate to give better responsiveness - * for buttons */ - snd_soc_update_bits(codec, WM8915_MIC_DETECT_1, - WM8915_MICD_RATE_MASK, - 5 << WM8915_MICD_RATE_SHIFT); } /* If we detected a lower impedence during initial startup @@ -2377,17 +2333,15 @@ static void wm8915_micd(struct snd_soc_codec *codec) SND_JACK_HEADPHONE, SND_JACK_HEADSET | SND_JACK_BTN_0); - - /* Increase the detection rate a bit for - * responsiveness. - */ - snd_soc_update_bits(codec, WM8915_MIC_DETECT_1, - WM8915_MICD_RATE_MASK, - 7 << WM8915_MICD_RATE_SHIFT); - wm8915->detecting = false; } } + + /* Increase poll rate to give better responsiveness for buttons */ + if (!wm8915->detecting) + snd_soc_update_bits(codec, WM8915_MIC_DETECT_1, + WM8915_MICD_RATE_MASK, + 5 << WM8915_MICD_RATE_SHIFT); } static irqreturn_t wm8915_irq(int irq, void *data) @@ -2429,20 +2383,6 @@ static irqreturn_t wm8915_irq(int irq, void *data) } } -static irqreturn_t wm8915_edge_irq(int irq, void *data) -{ - irqreturn_t ret = IRQ_NONE; - irqreturn_t val; - - do { - val = wm8915_irq(irq, data); - if (val != IRQ_NONE) - ret = val; - } while (val != IRQ_NONE); - - return ret; -} - static void wm8915_retune_mobile_pdata(struct snd_soc_codec *codec) { struct wm8915_priv *wm8915 = snd_soc_codec_get_drvdata(codec); @@ -2542,6 +2482,8 @@ static int wm8915_probe(struct snd_soc_codec *codec) wm8915->disable_nb[1].notifier_call = wm8915_regulator_event_1; wm8915->disable_nb[2].notifier_call = wm8915_regulator_event_2; wm8915->disable_nb[3].notifier_call = wm8915_regulator_event_3; + wm8915->disable_nb[4].notifier_call = wm8915_regulator_event_4; + wm8915->disable_nb[5].notifier_call = wm8915_regulator_event_5; /* This should really be moved into the regulator core */ for (i = 0; i < ARRAY_SIZE(wm8915->supplies); i++) { @@ -2767,14 +2709,8 @@ static int wm8915_probe(struct snd_soc_codec *codec) irq_flags |= IRQF_ONESHOT; - if (irq_flags & (IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING)) - ret = request_threaded_irq(i2c->irq, NULL, - wm8915_edge_irq, - irq_flags, "wm8915", codec); - else - ret = request_threaded_irq(i2c->irq, NULL, wm8915_irq, - irq_flags, "wm8915", codec); - + ret = request_threaded_irq(i2c->irq, NULL, wm8915_irq, + irq_flags, "wm8915", codec); if (ret == 0) { /* Unmask the interrupt */ snd_soc_update_bits(codec, WM8915_INTERRUPT_CONTROL, diff --git a/trunk/sound/soc/codecs/wm8940.c b/trunk/sound/soc/codecs/wm8940.c index 056daa0010f9..25580e3ee7c4 100644 --- a/trunk/sound/soc/codecs/wm8940.c +++ b/trunk/sound/soc/codecs/wm8940.c @@ -297,6 +297,8 @@ static int wm8940_add_widgets(struct snd_soc_codec *codec) if (ret) goto error_ret; ret = snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map)); + if (ret) + goto error_ret; error_ret: return ret; @@ -681,6 +683,8 @@ static int wm8940_resume(struct snd_soc_codec *codec) } } ret = wm8940_set_bias_level(codec, SND_SOC_BIAS_STANDBY); + if (ret) + goto error_ret; error_ret: return ret; @@ -726,6 +730,9 @@ static int wm8940_probe(struct snd_soc_codec *codec) if (ret) return ret; ret = wm8940_add_widgets(codec); + if (ret) + return ret; + return ret; } diff --git a/trunk/sound/soc/codecs/wm8962.c b/trunk/sound/soc/codecs/wm8962.c index 8499c563a9b5..5e05eed96c38 100644 --- a/trunk/sound/soc/codecs/wm8962.c +++ b/trunk/sound/soc/codecs/wm8962.c @@ -78,8 +78,6 @@ struct wm8962_priv { #ifdef CONFIG_GPIOLIB struct gpio_chip gpio_chip; #endif - - int irq; }; /* We can't use the same notifier block for more than one supply and @@ -1984,7 +1982,6 @@ static const unsigned int classd_tlv[] = { 0, 6, TLV_DB_SCALE_ITEM(0, 150, 0), 7, 7, TLV_DB_SCALE_ITEM(1200, 0, 0), }; -static const DECLARE_TLV_DB_SCALE(eq_tlv, -1200, 100, 0); /* The VU bits for the headphones are in a different register to the mute * bits and only take effect on the PGA if it is actually powered. @@ -2122,18 +2119,6 @@ SOC_SINGLE_TLV("HPMIXR MIXINR Volume", WM8962_HEADPHONE_MIXER_4, SOC_SINGLE_TLV("Speaker Boost Volume", WM8962_CLASS_D_CONTROL_2, 0, 7, 0, classd_tlv), - -SOC_SINGLE("EQ Switch", WM8962_EQ1, WM8962_EQ_ENA_SHIFT, 1, 0), -SOC_DOUBLE_R_TLV("EQ1 Volume", WM8962_EQ2, WM8962_EQ22, - WM8962_EQL_B1_GAIN_SHIFT, 31, 0, eq_tlv), -SOC_DOUBLE_R_TLV("EQ2 Volume", WM8962_EQ2, WM8962_EQ22, - WM8962_EQL_B2_GAIN_SHIFT, 31, 0, eq_tlv), -SOC_DOUBLE_R_TLV("EQ3 Volume", WM8962_EQ2, WM8962_EQ22, - WM8962_EQL_B3_GAIN_SHIFT, 31, 0, eq_tlv), -SOC_DOUBLE_R_TLV("EQ4 Volume", WM8962_EQ3, WM8962_EQ23, - WM8962_EQL_B4_GAIN_SHIFT, 31, 0, eq_tlv), -SOC_DOUBLE_R_TLV("EQ5 Volume", WM8962_EQ3, WM8962_EQ23, - WM8962_EQL_B5_GAIN_SHIFT, 31, 0, eq_tlv), }; static const struct snd_kcontrol_new wm8962_spk_mono_controls[] = { @@ -2199,8 +2184,6 @@ static int sysclk_event(struct snd_soc_dapm_widget *w, struct snd_kcontrol *kcontrol, int event) { struct snd_soc_codec *codec = w->codec; - struct wm8962_priv *wm8962 = snd_soc_codec_get_drvdata(codec); - unsigned long timeout; int src; int fll; @@ -2220,19 +2203,9 @@ static int sysclk_event(struct snd_soc_dapm_widget *w, switch (event) { case SND_SOC_DAPM_PRE_PMU: - if (fll) { + if (fll) snd_soc_update_bits(codec, WM8962_FLL_CONTROL_1, WM8962_FLL_ENA, WM8962_FLL_ENA); - if (wm8962->irq) { - timeout = msecs_to_jiffies(5); - timeout = wait_for_completion_timeout(&wm8962->fll_lock, - timeout); - - if (timeout == 0) - dev_err(codec->dev, - "Timed out starting FLL\n"); - } - } break; case SND_SOC_DAPM_POST_PMD: @@ -2790,44 +2763,18 @@ static const int bclk_divs[] = { 1, -1, 2, 3, 4, -1, 6, 8, -1, 12, 16, 24, -1, 32, 32, 32 }; -static const int sysclk_rates[] = { - 64, 128, 192, 256, 384, 512, 768, 1024, 1408, 1536, -}; - static void wm8962_configure_bclk(struct snd_soc_codec *codec) { struct wm8962_priv *wm8962 = snd_soc_codec_get_drvdata(codec); int dspclk, i; int clocking2 = 0; - int clocking4 = 0; int aif2 = 0; - if (!wm8962->sysclk_rate) { - dev_dbg(codec->dev, "No SYSCLK configured\n"); - return; - } - - if (!wm8962->bclk || !wm8962->lrclk) { - dev_dbg(codec->dev, "No audio clocks configured\n"); + if (!wm8962->bclk) { + dev_dbg(codec->dev, "No BCLK rate configured\n"); return; } - for (i = 0; i < ARRAY_SIZE(sysclk_rates); i++) { - if (sysclk_rates[i] == wm8962->sysclk_rate / wm8962->lrclk) { - clocking4 |= i << WM8962_SYSCLK_RATE_SHIFT; - break; - } - } - - if (i == ARRAY_SIZE(sysclk_rates)) { - dev_err(codec->dev, "Unsupported sysclk ratio %d\n", - wm8962->sysclk_rate / wm8962->lrclk); - return; - } - - snd_soc_update_bits(codec, WM8962_CLOCKING_4, - WM8962_SYSCLK_RATE_MASK, clocking4); - dspclk = snd_soc_read(codec, WM8962_CLOCKING1); if (dspclk < 0) { dev_err(codec->dev, "Failed to read DSPCLK: %d\n", dspclk); @@ -2897,8 +2844,6 @@ static int wm8962_set_bias_level(struct snd_soc_codec *codec, /* VMID 2*50k */ snd_soc_update_bits(codec, WM8962_PWR_MGMT_1, WM8962_VMID_SEL_MASK, 0x80); - - wm8962_configure_bclk(codec); break; case SND_SOC_BIAS_STANDBY: @@ -2931,6 +2876,8 @@ static int wm8962_set_bias_level(struct snd_soc_codec *codec, snd_soc_update_bits(codec, WM8962_CLOCKING2, WM8962_CLKREG_OVD, WM8962_CLKREG_OVD); + + wm8962_configure_bclk(codec); } /* VMID 2*250k */ @@ -2971,6 +2918,10 @@ static const struct { { 96000, 6 }, }; +static const int sysclk_rates[] = { + 64, 128, 192, 256, 384, 512, 768, 1024, 1408, 1536, +}; + static int wm8962_hw_params(struct snd_pcm_substream *substream, struct snd_pcm_hw_params *params, struct snd_soc_dai *dai) @@ -2978,27 +2929,41 @@ static int wm8962_hw_params(struct snd_pcm_substream *substream, struct snd_soc_pcm_runtime *rtd = substream->private_data; struct snd_soc_codec *codec = rtd->codec; struct wm8962_priv *wm8962 = snd_soc_codec_get_drvdata(codec); + int rate = params_rate(params); int i; int aif0 = 0; int adctl3 = 0; + int clocking4 = 0; wm8962->bclk = snd_soc_params_to_bclk(params); wm8962->lrclk = params_rate(params); for (i = 0; i < ARRAY_SIZE(sr_vals); i++) { - if (sr_vals[i].rate == wm8962->lrclk) { + if (sr_vals[i].rate == rate) { adctl3 |= sr_vals[i].reg; break; } } if (i == ARRAY_SIZE(sr_vals)) { - dev_err(codec->dev, "Unsupported rate %dHz\n", wm8962->lrclk); + dev_err(codec->dev, "Unsupported rate %dHz\n", rate); return -EINVAL; } - if (wm8962->lrclk % 8000 == 0) + if (rate % 8000 == 0) adctl3 |= WM8962_SAMPLE_RATE_INT_MODE; + for (i = 0; i < ARRAY_SIZE(sysclk_rates); i++) { + if (sysclk_rates[i] == wm8962->sysclk_rate / rate) { + clocking4 |= i << WM8962_SYSCLK_RATE_SHIFT; + break; + } + } + if (i == ARRAY_SIZE(sysclk_rates)) { + dev_err(codec->dev, "Unsupported sysclk ratio %d\n", + wm8962->sysclk_rate / rate); + return -EINVAL; + } + switch (params_format(params)) { case SNDRV_PCM_FORMAT_S16_LE: break; @@ -3020,6 +2985,8 @@ static int wm8962_hw_params(struct snd_pcm_substream *substream, snd_soc_update_bits(codec, WM8962_ADDITIONAL_CONTROL_3, WM8962_SAMPLE_RATE_INT_MODE | WM8962_SAMPLE_RATE_MASK, adctl3); + snd_soc_update_bits(codec, WM8962_CLOCKING_4, + WM8962_SYSCLK_RATE_MASK, clocking4); wm8962_configure_bclk(codec); @@ -3294,31 +3261,16 @@ static int wm8962_set_fll(struct snd_soc_codec *codec, int fll_id, int source, dev_dbg(codec->dev, "FLL configured for %dHz->%dHz\n", Fref, Fout); - ret = 0; - - if (fll1 & WM8962_FLL_ENA) { - /* This should be a massive overestimate but go even - * higher if we'll error out - */ - if (wm8962->irq) - timeout = msecs_to_jiffies(5); - else - timeout = msecs_to_jiffies(1); - - timeout = wait_for_completion_timeout(&wm8962->fll_lock, - timeout); + /* This should be a massive overestimate */ + timeout = msecs_to_jiffies(1); - if (timeout == 0 && wm8962->irq) { - dev_err(codec->dev, "FLL lock timed out"); - ret = -ETIMEDOUT; - } - } + wait_for_completion_timeout(&wm8962->fll_lock, timeout); wm8962->fll_fref = Fref; wm8962->fll_fout = Fout; wm8962->fll_src = source; - return ret; + return 0; } static int wm8962_mute(struct snd_soc_dai *dai, int mute) @@ -3779,6 +3731,8 @@ static int wm8962_probe(struct snd_soc_codec *codec) int ret; struct wm8962_priv *wm8962 = snd_soc_codec_get_drvdata(codec); struct wm8962_pdata *pdata = dev_get_platdata(codec->dev); + struct i2c_client *i2c = container_of(codec->dev, struct i2c_client, + dev); u16 *reg_cache = codec->reg_cache; int i, trigger, irq_pol; bool dmicclk, dmicdat; @@ -3917,9 +3871,6 @@ static int wm8962_probe(struct snd_soc_codec *codec) snd_soc_update_bits(codec, WM8962_HPOUTR_VOLUME, WM8962_HPOUT_VU, WM8962_HPOUT_VU); - /* Stereo control for EQ */ - snd_soc_update_bits(codec, WM8962_EQ1, WM8962_EQ_SHARED_COEFF, 0); - wm8962_add_widgets(codec); /* Save boards having to disable DMIC when not in use */ @@ -3948,7 +3899,7 @@ static int wm8962_probe(struct snd_soc_codec *codec) wm8962_init_beep(codec); wm8962_init_gpio(codec); - if (wm8962->irq) { + if (i2c->irq) { if (pdata && pdata->irq_active_low) { trigger = IRQF_TRIGGER_LOW; irq_pol = WM8962_IRQ_POL; @@ -3960,13 +3911,12 @@ static int wm8962_probe(struct snd_soc_codec *codec) snd_soc_update_bits(codec, WM8962_INTERRUPT_CONTROL, WM8962_IRQ_POL, irq_pol); - ret = request_threaded_irq(wm8962->irq, NULL, wm8962_irq, + ret = request_threaded_irq(i2c->irq, NULL, wm8962_irq, trigger | IRQF_ONESHOT, "wm8962", codec); if (ret != 0) { dev_err(codec->dev, "Failed to request IRQ %d: %d\n", - wm8962->irq, ret); - wm8962->irq = 0; + i2c->irq, ret); /* Non-fatal */ } else { /* Enable some IRQs by default */ @@ -3991,10 +3941,12 @@ static int wm8962_probe(struct snd_soc_codec *codec) static int wm8962_remove(struct snd_soc_codec *codec) { struct wm8962_priv *wm8962 = snd_soc_codec_get_drvdata(codec); + struct i2c_client *i2c = container_of(codec->dev, struct i2c_client, + dev); int i; - if (wm8962->irq) - free_irq(wm8962->irq, codec); + if (i2c->irq) + free_irq(i2c->irq, codec); cancel_delayed_work_sync(&wm8962->mic_work); @@ -4034,8 +3986,6 @@ static __devinit int wm8962_i2c_probe(struct i2c_client *i2c, i2c_set_clientdata(i2c, wm8962); - wm8962->irq = i2c->irq; - ret = snd_soc_register_codec(&i2c->dev, &soc_codec_dev_wm8962, &wm8962_dai, 1); if (ret < 0) diff --git a/trunk/sound/soc/codecs/wm8983.c b/trunk/sound/soc/codecs/wm8983.c deleted file mode 100644 index 17f04ec2b940..000000000000 --- a/trunk/sound/soc/codecs/wm8983.c +++ /dev/null @@ -1,1203 +0,0 @@ -/* - * wm8983.c -- WM8983 ALSA SoC Audio driver - * - * Copyright 2011 Wolfson Microelectronics plc - * - * Author: Dimitris Papastamos - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License version 2 as - * published by the Free Software Foundation. - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "wm8983.h" - -static const u16 wm8983_reg_defs[WM8983_MAX_REGISTER + 1] = { - [0x00] = 0x0000, /* R0 - Software Reset */ - [0x01] = 0x0000, /* R1 - Power management 1 */ - [0x02] = 0x0000, /* R2 - Power management 2 */ - [0x03] = 0x0000, /* R3 - Power management 3 */ - [0x04] = 0x0050, /* R4 - Audio Interface */ - [0x05] = 0x0000, /* R5 - Companding control */ - [0x06] = 0x0140, /* R6 - Clock Gen control */ - [0x07] = 0x0000, /* R7 - Additional control */ - [0x08] = 0x0000, /* R8 - GPIO Control */ - [0x09] = 0x0000, /* R9 - Jack Detect Control 1 */ - [0x0A] = 0x0000, /* R10 - DAC Control */ - [0x0B] = 0x00FF, /* R11 - Left DAC digital Vol */ - [0x0C] = 0x00FF, /* R12 - Right DAC digital vol */ - [0x0D] = 0x0000, /* R13 - Jack Detect Control 2 */ - [0x0E] = 0x0100, /* R14 - ADC Control */ - [0x0F] = 0x00FF, /* R15 - Left ADC Digital Vol */ - [0x10] = 0x00FF, /* R16 - Right ADC Digital Vol */ - [0x12] = 0x012C, /* R18 - EQ1 - low shelf */ - [0x13] = 0x002C, /* R19 - EQ2 - peak 1 */ - [0x14] = 0x002C, /* R20 - EQ3 - peak 2 */ - [0x15] = 0x002C, /* R21 - EQ4 - peak 3 */ - [0x16] = 0x002C, /* R22 - EQ5 - high shelf */ - [0x18] = 0x0032, /* R24 - DAC Limiter 1 */ - [0x19] = 0x0000, /* R25 - DAC Limiter 2 */ - [0x1B] = 0x0000, /* R27 - Notch Filter 1 */ - [0x1C] = 0x0000, /* R28 - Notch Filter 2 */ - [0x1D] = 0x0000, /* R29 - Notch Filter 3 */ - [0x1E] = 0x0000, /* R30 - Notch Filter 4 */ - [0x20] = 0x0038, /* R32 - ALC control 1 */ - [0x21] = 0x000B, /* R33 - ALC control 2 */ - [0x22] = 0x0032, /* R34 - ALC control 3 */ - [0x23] = 0x0000, /* R35 - Noise Gate */ - [0x24] = 0x0008, /* R36 - PLL N */ - [0x25] = 0x000C, /* R37 - PLL K 1 */ - [0x26] = 0x0093, /* R38 - PLL K 2 */ - [0x27] = 0x00E9, /* R39 - PLL K 3 */ - [0x29] = 0x0000, /* R41 - 3D control */ - [0x2A] = 0x0000, /* R42 - OUT4 to ADC */ - [0x2B] = 0x0000, /* R43 - Beep control */ - [0x2C] = 0x0033, /* R44 - Input ctrl */ - [0x2D] = 0x0010, /* R45 - Left INP PGA gain ctrl */ - [0x2E] = 0x0010, /* R46 - Right INP PGA gain ctrl */ - [0x2F] = 0x0100, /* R47 - Left ADC BOOST ctrl */ - [0x30] = 0x0100, /* R48 - Right ADC BOOST ctrl */ - [0x31] = 0x0002, /* R49 - Output ctrl */ - [0x32] = 0x0001, /* R50 - Left mixer ctrl */ - [0x33] = 0x0001, /* R51 - Right mixer ctrl */ - [0x34] = 0x0039, /* R52 - LOUT1 (HP) volume ctrl */ - [0x35] = 0x0039, /* R53 - ROUT1 (HP) volume ctrl */ - [0x36] = 0x0039, /* R54 - LOUT2 (SPK) volume ctrl */ - [0x37] = 0x0039, /* R55 - ROUT2 (SPK) volume ctrl */ - [0x38] = 0x0001, /* R56 - OUT3 mixer ctrl */ - [0x39] = 0x0001, /* R57 - OUT4 (MONO) mix ctrl */ - [0x3D] = 0x0000 /* R61 - BIAS CTRL */ -}; - -static const struct wm8983_reg_access { - u16 read; /* Mask of readable bits */ - u16 write; /* Mask of writable bits */ -} wm8983_access_masks[WM8983_MAX_REGISTER + 1] = { - [0x00] = { 0x0000, 0x01FF }, /* R0 - Software Reset */ - [0x01] = { 0x0000, 0x01FF }, /* R1 - Power management 1 */ - [0x02] = { 0x0000, 0x01FF }, /* R2 - Power management 2 */ - [0x03] = { 0x0000, 0x01EF }, /* R3 - Power management 3 */ - [0x04] = { 0x0000, 0x01FF }, /* R4 - Audio Interface */ - [0x05] = { 0x0000, 0x003F }, /* R5 - Companding control */ - [0x06] = { 0x0000, 0x01FD }, /* R6 - Clock Gen control */ - [0x07] = { 0x0000, 0x000F }, /* R7 - Additional control */ - [0x08] = { 0x0000, 0x003F }, /* R8 - GPIO Control */ - [0x09] = { 0x0000, 0x0070 }, /* R9 - Jack Detect Control 1 */ - [0x0A] = { 0x0000, 0x004F }, /* R10 - DAC Control */ - [0x0B] = { 0x0000, 0x01FF }, /* R11 - Left DAC digital Vol */ - [0x0C] = { 0x0000, 0x01FF }, /* R12 - Right DAC digital vol */ - [0x0D] = { 0x0000, 0x00FF }, /* R13 - Jack Detect Control 2 */ - [0x0E] = { 0x0000, 0x01FB }, /* R14 - ADC Control */ - [0x0F] = { 0x0000, 0x01FF }, /* R15 - Left ADC Digital Vol */ - [0x10] = { 0x0000, 0x01FF }, /* R16 - Right ADC Digital Vol */ - [0x12] = { 0x0000, 0x017F }, /* R18 - EQ1 - low shelf */ - [0x13] = { 0x0000, 0x017F }, /* R19 - EQ2 - peak 1 */ - [0x14] = { 0x0000, 0x017F }, /* R20 - EQ3 - peak 2 */ - [0x15] = { 0x0000, 0x017F }, /* R21 - EQ4 - peak 3 */ - [0x16] = { 0x0000, 0x007F }, /* R22 - EQ5 - high shelf */ - [0x18] = { 0x0000, 0x01FF }, /* R24 - DAC Limiter 1 */ - [0x19] = { 0x0000, 0x007F }, /* R25 - DAC Limiter 2 */ - [0x1B] = { 0x0000, 0x01FF }, /* R27 - Notch Filter 1 */ - [0x1C] = { 0x0000, 0x017F }, /* R28 - Notch Filter 2 */ - [0x1D] = { 0x0000, 0x017F }, /* R29 - Notch Filter 3 */ - [0x1E] = { 0x0000, 0x017F }, /* R30 - Notch Filter 4 */ - [0x20] = { 0x0000, 0x01BF }, /* R32 - ALC control 1 */ - [0x21] = { 0x0000, 0x00FF }, /* R33 - ALC control 2 */ - [0x22] = { 0x0000, 0x01FF }, /* R34 - ALC control 3 */ - [0x23] = { 0x0000, 0x000F }, /* R35 - Noise Gate */ - [0x24] = { 0x0000, 0x001F }, /* R36 - PLL N */ - [0x25] = { 0x0000, 0x003F }, /* R37 - PLL K 1 */ - [0x26] = { 0x0000, 0x01FF }, /* R38 - PLL K 2 */ - [0x27] = { 0x0000, 0x01FF }, /* R39 - PLL K 3 */ - [0x29] = { 0x0000, 0x000F }, /* R41 - 3D control */ - [0x2A] = { 0x0000, 0x01E7 }, /* R42 - OUT4 to ADC */ - [0x2B] = { 0x0000, 0x01BF }, /* R43 - Beep control */ - [0x2C] = { 0x0000, 0x0177 }, /* R44 - Input ctrl */ - [0x2D] = { 0x0000, 0x01FF }, /* R45 - Left INP PGA gain ctrl */ - [0x2E] = { 0x0000, 0x01FF }, /* R46 - Right INP PGA gain ctrl */ - [0x2F] = { 0x0000, 0x0177 }, /* R47 - Left ADC BOOST ctrl */ - [0x30] = { 0x0000, 0x0177 }, /* R48 - Right ADC BOOST ctrl */ - [0x31] = { 0x0000, 0x007F }, /* R49 - Output ctrl */ - [0x32] = { 0x0000, 0x01FF }, /* R50 - Left mixer ctrl */ - [0x33] = { 0x0000, 0x01FF }, /* R51 - Right mixer ctrl */ - [0x34] = { 0x0000, 0x01FF }, /* R52 - LOUT1 (HP) volume ctrl */ - [0x35] = { 0x0000, 0x01FF }, /* R53 - ROUT1 (HP) volume ctrl */ - [0x36] = { 0x0000, 0x01FF }, /* R54 - LOUT2 (SPK) volume ctrl */ - [0x37] = { 0x0000, 0x01FF }, /* R55 - ROUT2 (SPK) volume ctrl */ - [0x38] = { 0x0000, 0x004F }, /* R56 - OUT3 mixer ctrl */ - [0x39] = { 0x0000, 0x00FF }, /* R57 - OUT4 (MONO) mix ctrl */ - [0x3D] = { 0x0000, 0x0100 } /* R61 - BIAS CTRL */ -}; - -/* vol/gain update regs */ -static const int vol_update_regs[] = { - WM8983_LEFT_DAC_DIGITAL_VOL, - WM8983_RIGHT_DAC_DIGITAL_VOL, - WM8983_LEFT_ADC_DIGITAL_VOL, - WM8983_RIGHT_ADC_DIGITAL_VOL, - WM8983_LOUT1_HP_VOLUME_CTRL, - WM8983_ROUT1_HP_VOLUME_CTRL, - WM8983_LOUT2_SPK_VOLUME_CTRL, - WM8983_ROUT2_SPK_VOLUME_CTRL, - WM8983_LEFT_INP_PGA_GAIN_CTRL, - WM8983_RIGHT_INP_PGA_GAIN_CTRL -}; - -struct wm8983_priv { - enum snd_soc_control_type control_type; - u32 sysclk; - u32 bclk; -}; - -static const struct { - int div; - int ratio; -} fs_ratios[] = { - { 10, 128 }, - { 15, 192 }, - { 20, 256 }, - { 30, 384 }, - { 40, 512 }, - { 60, 768 }, - { 80, 1024 }, - { 120, 1536 } -}; - -static const int srates[] = { 48000, 32000, 24000, 16000, 12000, 8000 }; - -static const int bclk_divs[] = { - 1, 2, 4, 8, 16, 32 -}; - -static int eqmode_get(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol); -static int eqmode_put(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol); - -static const DECLARE_TLV_DB_SCALE(dac_tlv, -12700, 50, 1); -static const DECLARE_TLV_DB_SCALE(adc_tlv, -12700, 50, 1); -static const DECLARE_TLV_DB_SCALE(out_tlv, -5700, 100, 0); -static const DECLARE_TLV_DB_SCALE(lim_thresh_tlv, -600, 100, 0); -static const DECLARE_TLV_DB_SCALE(lim_boost_tlv, 0, 100, 0); -static const DECLARE_TLV_DB_SCALE(alc_min_tlv, -1200, 600, 0); -static const DECLARE_TLV_DB_SCALE(alc_max_tlv, -675, 600, 0); -static const DECLARE_TLV_DB_SCALE(alc_tar_tlv, -2250, 150, 0); -static const DECLARE_TLV_DB_SCALE(pga_vol_tlv, -1200, 75, 0); -static const DECLARE_TLV_DB_SCALE(boost_tlv, -1200, 300, 1); -static const DECLARE_TLV_DB_SCALE(eq_tlv, -1200, 100, 0); -static const DECLARE_TLV_DB_SCALE(aux_tlv, -1500, 300, 0); -static const DECLARE_TLV_DB_SCALE(bypass_tlv, -1500, 300, 0); -static const DECLARE_TLV_DB_SCALE(pga_boost_tlv, 0, 2000, 0); - -static const char *alc_sel_text[] = { "Off", "Right", "Left", "Stereo" }; -static const SOC_ENUM_SINGLE_DECL(alc_sel, WM8983_ALC_CONTROL_1, 7, - alc_sel_text); - -static const char *alc_mode_text[] = { "ALC", "Limiter" }; -static const SOC_ENUM_SINGLE_DECL(alc_mode, WM8983_ALC_CONTROL_3, 8, - alc_mode_text); - -static const char *filter_mode_text[] = { "Audio", "Application" }; -static const SOC_ENUM_SINGLE_DECL(filter_mode, WM8983_ADC_CONTROL, 7, - filter_mode_text); - -static const char *eq_bw_text[] = { "Narrow", "Wide" }; -static const char *eqmode_text[] = { "Capture", "Playback" }; -static const SOC_ENUM_SINGLE_EXT_DECL(eqmode, eqmode_text); - -static const char *eq1_cutoff_text[] = { - "80Hz", "105Hz", "135Hz", "175Hz" -}; -static const SOC_ENUM_SINGLE_DECL(eq1_cutoff, WM8983_EQ1_LOW_SHELF, 5, - eq1_cutoff_text); -static const char *eq2_cutoff_text[] = { - "230Hz", "300Hz", "385Hz", "500Hz" -}; -static const SOC_ENUM_SINGLE_DECL(eq2_bw, WM8983_EQ2_PEAK_1, 8, eq_bw_text); -static const SOC_ENUM_SINGLE_DECL(eq2_cutoff, WM8983_EQ2_PEAK_1, 5, - eq2_cutoff_text); -static const char *eq3_cutoff_text[] = { - "650Hz", "850Hz", "1.1kHz", "1.4kHz" -}; -static const SOC_ENUM_SINGLE_DECL(eq3_bw, WM8983_EQ3_PEAK_2, 8, eq_bw_text); -static const SOC_ENUM_SINGLE_DECL(eq3_cutoff, WM8983_EQ3_PEAK_2, 5, - eq3_cutoff_text); -static const char *eq4_cutoff_text[] = { - "1.8kHz", "2.4kHz", "3.2kHz", "4.1kHz" -}; -static const SOC_ENUM_SINGLE_DECL(eq4_bw, WM8983_EQ4_PEAK_3, 8, eq_bw_text); -static const SOC_ENUM_SINGLE_DECL(eq4_cutoff, WM8983_EQ4_PEAK_3, 5, - eq4_cutoff_text); -static const char *eq5_cutoff_text[] = { - "5.3kHz", "6.9kHz", "9kHz", "11.7kHz" -}; -static const SOC_ENUM_SINGLE_DECL(eq5_cutoff, WM8983_EQ5_HIGH_SHELF, 5, - eq5_cutoff_text); - -static const char *speaker_mode_text[] = { "Class A/B", "Class D" }; -static const SOC_ENUM_SINGLE_DECL(speaker_mode, 0x17, 8, speaker_mode_text); - -static const char *depth_3d_text[] = { - "Off", - "6.67%", - "13.3%", - "20%", - "26.7%", - "33.3%", - "40%", - "46.6%", - "53.3%", - "60%", - "66.7%", - "73.3%", - "80%", - "86.7%", - "93.3%", - "100%" -}; -static const SOC_ENUM_SINGLE_DECL(depth_3d, WM8983_3D_CONTROL, 0, - depth_3d_text); - -static const struct snd_kcontrol_new wm8983_snd_controls[] = { - SOC_SINGLE("Digital Loopback Switch", WM8983_COMPANDING_CONTROL, - 0, 1, 0), - - SOC_ENUM("ALC Capture Function", alc_sel), - SOC_SINGLE_TLV("ALC Capture Max Volume", WM8983_ALC_CONTROL_1, - 3, 7, 0, alc_max_tlv), - SOC_SINGLE_TLV("ALC Capture Min Volume", WM8983_ALC_CONTROL_1, - 0, 7, 0, alc_min_tlv), - SOC_SINGLE_TLV("ALC Capture Target Volume", WM8983_ALC_CONTROL_2, - 0, 15, 0, alc_tar_tlv), - SOC_SINGLE("ALC Capture Attack", WM8983_ALC_CONTROL_3, 0, 10, 0), - SOC_SINGLE("ALC Capture Hold", WM8983_ALC_CONTROL_2, 4, 10, 0), - SOC_SINGLE("ALC Capture Decay", WM8983_ALC_CONTROL_3, 4, 10, 0), - SOC_ENUM("ALC Mode", alc_mode), - SOC_SINGLE("ALC Capture NG Switch", WM8983_NOISE_GATE, - 3, 1, 0), - SOC_SINGLE("ALC Capture NG Threshold", WM8983_NOISE_GATE, - 0, 7, 1), - - SOC_DOUBLE_R_TLV("Capture Volume", WM8983_LEFT_ADC_DIGITAL_VOL, - WM8983_RIGHT_ADC_DIGITAL_VOL, 0, 255, 0, adc_tlv), - SOC_DOUBLE_R("Capture PGA ZC Switch", WM8983_LEFT_INP_PGA_GAIN_CTRL, - WM8983_RIGHT_INP_PGA_GAIN_CTRL, 7, 1, 0), - SOC_DOUBLE_R_TLV("Capture PGA Volume", WM8983_LEFT_INP_PGA_GAIN_CTRL, - WM8983_RIGHT_INP_PGA_GAIN_CTRL, 0, 63, 0, pga_vol_tlv), - - SOC_DOUBLE_R_TLV("Capture PGA Boost Volume", - WM8983_LEFT_ADC_BOOST_CTRL, WM8983_RIGHT_ADC_BOOST_CTRL, - 8, 1, 0, pga_boost_tlv), - - SOC_DOUBLE("ADC Inversion Switch", WM8983_ADC_CONTROL, 0, 1, 1, 0), - SOC_SINGLE("ADC 128x Oversampling Switch", WM8983_ADC_CONTROL, 8, 1, 0), - - SOC_DOUBLE_R_TLV("Playback Volume", WM8983_LEFT_DAC_DIGITAL_VOL, - WM8983_RIGHT_DAC_DIGITAL_VOL, 0, 255, 0, dac_tlv), - - SOC_SINGLE("DAC Playback Limiter Switch", WM8983_DAC_LIMITER_1, 8, 1, 0), - SOC_SINGLE("DAC Playback Limiter Decay", WM8983_DAC_LIMITER_1, 4, 10, 0), - SOC_SINGLE("DAC Playback Limiter Attack", WM8983_DAC_LIMITER_1, 0, 11, 0), - SOC_SINGLE_TLV("DAC Playback Limiter Threshold", WM8983_DAC_LIMITER_2, - 4, 7, 1, lim_thresh_tlv), - SOC_SINGLE_TLV("DAC Playback Limiter Boost Volume", WM8983_DAC_LIMITER_2, - 0, 12, 0, lim_boost_tlv), - SOC_DOUBLE("DAC Inversion Switch", WM8983_DAC_CONTROL, 0, 1, 1, 0), - SOC_SINGLE("DAC Auto Mute Switch", WM8983_DAC_CONTROL, 2, 1, 0), - SOC_SINGLE("DAC 128x Oversampling Switch", WM8983_DAC_CONTROL, 3, 1, 0), - - SOC_DOUBLE_R_TLV("Headphone Playback Volume", WM8983_LOUT1_HP_VOLUME_CTRL, - WM8983_ROUT1_HP_VOLUME_CTRL, 0, 63, 0, out_tlv), - SOC_DOUBLE_R("Headphone Playback ZC Switch", WM8983_LOUT1_HP_VOLUME_CTRL, - WM8983_ROUT1_HP_VOLUME_CTRL, 7, 1, 0), - SOC_DOUBLE_R("Headphone Switch", WM8983_LOUT1_HP_VOLUME_CTRL, - WM8983_ROUT1_HP_VOLUME_CTRL, 6, 1, 1), - - SOC_DOUBLE_R_TLV("Speaker Playback Volume", WM8983_LOUT2_SPK_VOLUME_CTRL, - WM8983_ROUT2_SPK_VOLUME_CTRL, 0, 63, 0, out_tlv), - SOC_DOUBLE_R("Speaker Playback ZC Switch", WM8983_LOUT2_SPK_VOLUME_CTRL, - WM8983_ROUT2_SPK_VOLUME_CTRL, 7, 1, 0), - SOC_DOUBLE_R("Speaker Switch", WM8983_LOUT2_SPK_VOLUME_CTRL, - WM8983_ROUT2_SPK_VOLUME_CTRL, 6, 1, 1), - - SOC_SINGLE("OUT3 Switch", WM8983_OUT3_MIXER_CTRL, - 6, 1, 1), - - SOC_SINGLE("OUT4 Switch", WM8983_OUT4_MONO_MIX_CTRL, - 6, 1, 1), - - SOC_SINGLE("High Pass Filter Switch", WM8983_ADC_CONTROL, 8, 1, 0), - SOC_ENUM("High Pass Filter Mode", filter_mode), - SOC_SINGLE("High Pass Filter Cutoff", WM8983_ADC_CONTROL, 4, 7, 0), - - SOC_DOUBLE_R_TLV("Aux Bypass Volume", - WM8983_LEFT_MIXER_CTRL, WM8983_RIGHT_MIXER_CTRL, 6, 7, 0, - aux_tlv), - - SOC_DOUBLE_R_TLV("Input PGA Bypass Volume", - WM8983_LEFT_MIXER_CTRL, WM8983_RIGHT_MIXER_CTRL, 2, 7, 0, - bypass_tlv), - - SOC_ENUM_EXT("Equalizer Function", eqmode, eqmode_get, eqmode_put), - SOC_ENUM("EQ1 Cutoff", eq1_cutoff), - SOC_SINGLE_TLV("EQ1 Volume", WM8983_EQ1_LOW_SHELF, 0, 24, 1, eq_tlv), - SOC_ENUM("EQ2 Bandwith", eq2_bw), - SOC_ENUM("EQ2 Cutoff", eq2_cutoff), - SOC_SINGLE_TLV("EQ2 Volume", WM8983_EQ2_PEAK_1, 0, 24, 1, eq_tlv), - SOC_ENUM("EQ3 Bandwith", eq3_bw), - SOC_ENUM("EQ3 Cutoff", eq3_cutoff), - SOC_SINGLE_TLV("EQ3 Volume", WM8983_EQ3_PEAK_2, 0, 24, 1, eq_tlv), - SOC_ENUM("EQ4 Bandwith", eq4_bw), - SOC_ENUM("EQ4 Cutoff", eq4_cutoff), - SOC_SINGLE_TLV("EQ4 Volume", WM8983_EQ4_PEAK_3, 0, 24, 1, eq_tlv), - SOC_ENUM("EQ5 Cutoff", eq5_cutoff), - SOC_SINGLE_TLV("EQ5 Volume", WM8983_EQ5_HIGH_SHELF, 0, 24, 1, eq_tlv), - - SOC_ENUM("3D Depth", depth_3d), - - SOC_ENUM("Speaker Mode", speaker_mode) -}; - -static const struct snd_kcontrol_new left_out_mixer[] = { - SOC_DAPM_SINGLE("Line Switch", WM8983_LEFT_MIXER_CTRL, 1, 1, 0), - SOC_DAPM_SINGLE("Aux Switch", WM8983_LEFT_MIXER_CTRL, 5, 1, 0), - SOC_DAPM_SINGLE("PCM Switch", WM8983_LEFT_MIXER_CTRL, 0, 1, 0), -}; - -static const struct snd_kcontrol_new right_out_mixer[] = { - SOC_DAPM_SINGLE("Line Switch", WM8983_RIGHT_MIXER_CTRL, 1, 1, 0), - SOC_DAPM_SINGLE("Aux Switch", WM8983_RIGHT_MIXER_CTRL, 5, 1, 0), - SOC_DAPM_SINGLE("PCM Switch", WM8983_RIGHT_MIXER_CTRL, 0, 1, 0), -}; - -static const struct snd_kcontrol_new left_input_mixer[] = { - SOC_DAPM_SINGLE("L2 Switch", WM8983_INPUT_CTRL, 2, 1, 0), - SOC_DAPM_SINGLE("MicN Switch", WM8983_INPUT_CTRL, 1, 1, 0), - SOC_DAPM_SINGLE("MicP Switch", WM8983_INPUT_CTRL, 0, 1, 0), -}; - -static const struct snd_kcontrol_new right_input_mixer[] = { - SOC_DAPM_SINGLE("R2 Switch", WM8983_INPUT_CTRL, 6, 1, 0), - SOC_DAPM_SINGLE("MicN Switch", WM8983_INPUT_CTRL, 5, 1, 0), - SOC_DAPM_SINGLE("MicP Switch", WM8983_INPUT_CTRL, 4, 1, 0), -}; - -static const struct snd_kcontrol_new left_boost_mixer[] = { - SOC_DAPM_SINGLE_TLV("L2 Volume", WM8983_LEFT_ADC_BOOST_CTRL, - 4, 7, 0, boost_tlv), - SOC_DAPM_SINGLE_TLV("AUXL Volume", WM8983_LEFT_ADC_BOOST_CTRL, - 0, 7, 0, boost_tlv) -}; - -static const struct snd_kcontrol_new out3_mixer[] = { - SOC_DAPM_SINGLE("LMIX2OUT3 Switch", WM8983_OUT3_MIXER_CTRL, - 1, 1, 0), - SOC_DAPM_SINGLE("LDAC2OUT3 Switch", WM8983_OUT3_MIXER_CTRL, - 0, 1, 0), -}; - -static const struct snd_kcontrol_new out4_mixer[] = { - SOC_DAPM_SINGLE("LMIX2OUT4 Switch", WM8983_OUT4_MONO_MIX_CTRL, - 4, 1, 0), - SOC_DAPM_SINGLE("RMIX2OUT4 Switch", WM8983_OUT4_MONO_MIX_CTRL, - 1, 1, 0), - SOC_DAPM_SINGLE("LDAC2OUT4 Switch", WM8983_OUT4_MONO_MIX_CTRL, - 3, 1, 0), - SOC_DAPM_SINGLE("RDAC2OUT4 Switch", WM8983_OUT4_MONO_MIX_CTRL, - 0, 1, 0), -}; - -static const struct snd_kcontrol_new right_boost_mixer[] = { - SOC_DAPM_SINGLE_TLV("R2 Volume", WM8983_RIGHT_ADC_BOOST_CTRL, - 4, 7, 0, boost_tlv), - SOC_DAPM_SINGLE_TLV("AUXR Volume", WM8983_RIGHT_ADC_BOOST_CTRL, - 0, 7, 0, boost_tlv) -}; - -static const struct snd_soc_dapm_widget wm8983_dapm_widgets[] = { - SND_SOC_DAPM_DAC("Left DAC", "Left Playback", WM8983_POWER_MANAGEMENT_3, - 0, 0), - SND_SOC_DAPM_DAC("Right DAC", "Right Playback", WM8983_POWER_MANAGEMENT_3, - 1, 0), - SND_SOC_DAPM_ADC("Left ADC", "Left Capture", WM8983_POWER_MANAGEMENT_2, - 0, 0), - SND_SOC_DAPM_ADC("Right ADC", "Right Capture", WM8983_POWER_MANAGEMENT_2, - 1, 0), - - SND_SOC_DAPM_MIXER("Left Output Mixer", WM8983_POWER_MANAGEMENT_3, - 2, 0, left_out_mixer, ARRAY_SIZE(left_out_mixer)), - SND_SOC_DAPM_MIXER("Right Output Mixer", WM8983_POWER_MANAGEMENT_3, - 3, 0, right_out_mixer, ARRAY_SIZE(right_out_mixer)), - - SND_SOC_DAPM_MIXER("Left Input Mixer", WM8983_POWER_MANAGEMENT_2, - 2, 0, left_input_mixer, ARRAY_SIZE(left_input_mixer)), - SND_SOC_DAPM_MIXER("Right Input Mixer", WM8983_POWER_MANAGEMENT_2, - 3, 0, right_input_mixer, ARRAY_SIZE(right_input_mixer)), - - SND_SOC_DAPM_MIXER("Left Boost Mixer", WM8983_POWER_MANAGEMENT_2, - 4, 0, left_boost_mixer, ARRAY_SIZE(left_boost_mixer)), - SND_SOC_DAPM_MIXER("Right Boost Mixer", WM8983_POWER_MANAGEMENT_2, - 5, 0, right_boost_mixer, ARRAY_SIZE(right_boost_mixer)), - - SND_SOC_DAPM_MIXER("OUT3 Mixer", WM8983_POWER_MANAGEMENT_1, - 6, 0, out3_mixer, ARRAY_SIZE(out3_mixer)), - - SND_SOC_DAPM_MIXER("OUT4 Mixer", WM8983_POWER_MANAGEMENT_1, - 7, 0, out4_mixer, ARRAY_SIZE(out4_mixer)), - - SND_SOC_DAPM_PGA("Left Capture PGA", WM8983_LEFT_INP_PGA_GAIN_CTRL, - 6, 1, NULL, 0), - SND_SOC_DAPM_PGA("Right Capture PGA", WM8983_RIGHT_INP_PGA_GAIN_CTRL, - 6, 1, NULL, 0), - - SND_SOC_DAPM_PGA("Left Headphone Out", WM8983_POWER_MANAGEMENT_2, - 7, 0, NULL, 0), - SND_SOC_DAPM_PGA("Right Headphone Out", WM8983_POWER_MANAGEMENT_2, - 8, 0, NULL, 0), - - SND_SOC_DAPM_PGA("Left Speaker Out", WM8983_POWER_MANAGEMENT_3, - 5, 0, NULL, 0), - SND_SOC_DAPM_PGA("Right Speaker Out", WM8983_POWER_MANAGEMENT_3, - 6, 0, NULL, 0), - - SND_SOC_DAPM_PGA("OUT3 Out", WM8983_POWER_MANAGEMENT_3, - 7, 0, NULL, 0), - - SND_SOC_DAPM_PGA("OUT4 Out", WM8983_POWER_MANAGEMENT_3, - 8, 0, NULL, 0), - - SND_SOC_DAPM_MICBIAS("Mic Bias", WM8983_POWER_MANAGEMENT_1, 4, 0), - - SND_SOC_DAPM_INPUT("LIN"), - SND_SOC_DAPM_INPUT("LIP"), - SND_SOC_DAPM_INPUT("RIN"), - SND_SOC_DAPM_INPUT("RIP"), - SND_SOC_DAPM_INPUT("AUXL"), - SND_SOC_DAPM_INPUT("AUXR"), - SND_SOC_DAPM_INPUT("L2"), - SND_SOC_DAPM_INPUT("R2"), - SND_SOC_DAPM_OUTPUT("HPL"), - SND_SOC_DAPM_OUTPUT("HPR"), - SND_SOC_DAPM_OUTPUT("SPKL"), - SND_SOC_DAPM_OUTPUT("SPKR"), - SND_SOC_DAPM_OUTPUT("OUT3"), - SND_SOC_DAPM_OUTPUT("OUT4") -}; - -static const struct snd_soc_dapm_route wm8983_audio_map[] = { - { "OUT3 Mixer", "LMIX2OUT3 Switch", "Left Output Mixer" }, - { "OUT3 Mixer", "LDAC2OUT3 Switch", "Left DAC" }, - - { "OUT3 Out", NULL, "OUT3 Mixer" }, - { "OUT3", NULL, "OUT3 Out" }, - - { "OUT4 Mixer", "LMIX2OUT4 Switch", "Left Output Mixer" }, - { "OUT4 Mixer", "RMIX2OUT4 Switch", "Right Output Mixer" }, - { "OUT4 Mixer", "LDAC2OUT4 Switch", "Left DAC" }, - { "OUT4 Mixer", "RDAC2OUT4 Switch", "Right DAC" }, - - { "OUT4 Out", NULL, "OUT4 Mixer" }, - { "OUT4", NULL, "OUT4 Out" }, - - { "Right Output Mixer", "PCM Switch", "Right DAC" }, - { "Right Output Mixer", "Aux Switch", "AUXR" }, - { "Right Output Mixer", "Line Switch", "Right Boost Mixer" }, - - { "Left Output Mixer", "PCM Switch", "Left DAC" }, - { "Left Output Mixer", "Aux Switch", "AUXL" }, - { "Left Output Mixer", "Line Switch", "Left Boost Mixer" }, - - { "Right Headphone Out", NULL, "Right Output Mixer" }, - { "HPR", NULL, "Right Headphone Out" }, - - { "Left Headphone Out", NULL, "Left Output Mixer" }, - { "HPL", NULL, "Left Headphone Out" }, - - { "Right Speaker Out", NULL, "Right Output Mixer" }, - { "SPKR", NULL, "Right Speaker Out" }, - - { "Left Speaker Out", NULL, "Left Output Mixer" }, - { "SPKL", NULL, "Left Speaker Out" }, - - { "Right ADC", NULL, "Right Boost Mixer" }, - - { "Right Boost Mixer", "AUXR Volume", "AUXR" }, - { "Right Boost Mixer", NULL, "Right Capture PGA" }, - { "Right Boost Mixer", "R2 Volume", "R2" }, - - { "Left ADC", NULL, "Left Boost Mixer" }, - - { "Left Boost Mixer", "AUXL Volume", "AUXL" }, - { "Left Boost Mixer", NULL, "Left Capture PGA" }, - { "Left Boost Mixer", "L2 Volume", "L2" }, - - { "Right Capture PGA", NULL, "Right Input Mixer" }, - { "Left Capture PGA", NULL, "Left Input Mixer" }, - - { "Right Input Mixer", "R2 Switch", "R2" }, - { "Right Input Mixer", "MicN Switch", "RIN" }, - { "Right Input Mixer", "MicP Switch", "RIP" }, - - { "Left Input Mixer", "L2 Switch", "L2" }, - { "Left Input Mixer", "MicN Switch", "LIN" }, - { "Left Input Mixer", "MicP Switch", "LIP" }, -}; - -static int eqmode_get(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol) -{ - struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); - unsigned int reg; - - reg = snd_soc_read(codec, WM8983_EQ1_LOW_SHELF); - if (reg & WM8983_EQ3DMODE) - ucontrol->value.integer.value[0] = 1; - else - ucontrol->value.integer.value[0] = 0; - - return 0; -} - -static int eqmode_put(struct snd_kcontrol *kcontrol, - struct snd_ctl_elem_value *ucontrol) -{ - struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); - unsigned int regpwr2, regpwr3; - unsigned int reg_eq; - - if (ucontrol->value.integer.value[0] != 0 - && ucontrol->value.integer.value[0] != 1) - return -EINVAL; - - reg_eq = snd_soc_read(codec, WM8983_EQ1_LOW_SHELF); - switch ((reg_eq & WM8983_EQ3DMODE) >> WM8983_EQ3DMODE_SHIFT) { - case 0: - if (!ucontrol->value.integer.value[0]) - return 0; - break; - case 1: - if (ucontrol->value.integer.value[0]) - return 0; - break; - } - - regpwr2 = snd_soc_read(codec, WM8983_POWER_MANAGEMENT_2); - regpwr3 = snd_soc_read(codec, WM8983_POWER_MANAGEMENT_3); - /* disable the DACs and ADCs */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_2, - WM8983_ADCENR_MASK | WM8983_ADCENL_MASK, 0); - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_3, - WM8983_DACENR_MASK | WM8983_DACENL_MASK, 0); - /* set the desired eqmode */ - snd_soc_update_bits(codec, WM8983_EQ1_LOW_SHELF, - WM8983_EQ3DMODE_MASK, - ucontrol->value.integer.value[0] - << WM8983_EQ3DMODE_SHIFT); - /* restore DAC/ADC configuration */ - snd_soc_write(codec, WM8983_POWER_MANAGEMENT_2, regpwr2); - snd_soc_write(codec, WM8983_POWER_MANAGEMENT_3, regpwr3); - return 0; -} - -static int wm8983_readable(struct snd_soc_codec *codec, unsigned int reg) -{ - if (reg > WM8983_MAX_REGISTER) - return 0; - - return wm8983_access_masks[reg].read != 0; -} - -static int wm8983_dac_mute(struct snd_soc_dai *dai, int mute) -{ - struct snd_soc_codec *codec = dai->codec; - - return snd_soc_update_bits(codec, WM8983_DAC_CONTROL, - WM8983_SOFTMUTE_MASK, - !!mute << WM8983_SOFTMUTE_SHIFT); -} - -static int wm8983_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) -{ - struct snd_soc_codec *codec = dai->codec; - u16 format, master, bcp, lrp; - - switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { - case SND_SOC_DAIFMT_I2S: - format = 0x2; - break; - case SND_SOC_DAIFMT_RIGHT_J: - format = 0x0; - break; - case SND_SOC_DAIFMT_LEFT_J: - format = 0x1; - break; - case SND_SOC_DAIFMT_DSP_A: - case SND_SOC_DAIFMT_DSP_B: - format = 0x3; - break; - default: - dev_err(dai->dev, "Unknown dai format\n"); - return -EINVAL; - } - - snd_soc_update_bits(codec, WM8983_AUDIO_INTERFACE, - WM8983_FMT_MASK, format << WM8983_FMT_SHIFT); - - switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) { - case SND_SOC_DAIFMT_CBM_CFM: - master = 1; - break; - case SND_SOC_DAIFMT_CBS_CFS: - master = 0; - break; - default: - dev_err(dai->dev, "Unknown master/slave configuration\n"); - return -EINVAL; - } - - snd_soc_update_bits(codec, WM8983_CLOCK_GEN_CONTROL, - WM8983_MS_MASK, master << WM8983_MS_SHIFT); - - /* FIXME: We don't currently support DSP A/B modes */ - switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) { - case SND_SOC_DAIFMT_DSP_A: - case SND_SOC_DAIFMT_DSP_B: - dev_err(dai->dev, "DSP A/B modes are not supported\n"); - return -EINVAL; - default: - break; - } - - bcp = lrp = 0; - switch (fmt & SND_SOC_DAIFMT_INV_MASK) { - case SND_SOC_DAIFMT_NB_NF: - break; - case SND_SOC_DAIFMT_IB_IF: - bcp = lrp = 1; - break; - case SND_SOC_DAIFMT_IB_NF: - bcp = 1; - break; - case SND_SOC_DAIFMT_NB_IF: - lrp = 1; - break; - default: - dev_err(dai->dev, "Unknown polarity configuration\n"); - return -EINVAL; - } - - snd_soc_update_bits(codec, WM8983_AUDIO_INTERFACE, - WM8983_LRCP_MASK, lrp << WM8983_LRCP_SHIFT); - snd_soc_update_bits(codec, WM8983_AUDIO_INTERFACE, - WM8983_BCP_MASK, bcp << WM8983_BCP_SHIFT); - return 0; -} - -static int wm8983_hw_params(struct snd_pcm_substream *substream, - struct snd_pcm_hw_params *params, - struct snd_soc_dai *dai) -{ - int i; - struct snd_soc_codec *codec = dai->codec; - struct wm8983_priv *wm8983 = snd_soc_codec_get_drvdata(codec); - u16 blen, srate_idx; - u32 tmp; - int srate_best; - int ret; - - ret = snd_soc_params_to_bclk(params); - if (ret < 0) { - dev_err(codec->dev, "Failed to convert params to bclk: %d\n", ret); - return ret; - } - - wm8983->bclk = ret; - - switch (params_format(params)) { - case SNDRV_PCM_FORMAT_S16_LE: - blen = 0x0; - break; - case SNDRV_PCM_FORMAT_S20_3LE: - blen = 0x1; - break; - case SNDRV_PCM_FORMAT_S24_LE: - blen = 0x2; - break; - case SNDRV_PCM_FORMAT_S32_LE: - blen = 0x3; - break; - default: - dev_err(dai->dev, "Unsupported word length %u\n", - params_format(params)); - return -EINVAL; - } - - snd_soc_update_bits(codec, WM8983_AUDIO_INTERFACE, - WM8983_WL_MASK, blen << WM8983_WL_SHIFT); - - /* - * match to the nearest possible sample rate and rely - * on the array index to configure the SR register - */ - srate_idx = 0; - srate_best = abs(srates[0] - params_rate(params)); - for (i = 1; i < ARRAY_SIZE(srates); ++i) { - if (abs(srates[i] - params_rate(params)) >= srate_best) - continue; - srate_idx = i; - srate_best = abs(srates[i] - params_rate(params)); - } - - dev_dbg(dai->dev, "Selected SRATE = %d\n", srates[srate_idx]); - snd_soc_update_bits(codec, WM8983_ADDITIONAL_CONTROL, - WM8983_SR_MASK, srate_idx << WM8983_SR_SHIFT); - - dev_dbg(dai->dev, "Target BCLK = %uHz\n", wm8983->bclk); - dev_dbg(dai->dev, "SYSCLK = %uHz\n", wm8983->sysclk); - - for (i = 0; i < ARRAY_SIZE(fs_ratios); ++i) { - if (wm8983->sysclk / params_rate(params) - == fs_ratios[i].ratio) - break; - } - - if (i == ARRAY_SIZE(fs_ratios)) { - dev_err(dai->dev, "Unable to configure MCLK ratio %u/%u\n", - wm8983->sysclk, params_rate(params)); - return -EINVAL; - } - - dev_dbg(dai->dev, "MCLK ratio = %dfs\n", fs_ratios[i].ratio); - snd_soc_update_bits(codec, WM8983_CLOCK_GEN_CONTROL, - WM8983_MCLKDIV_MASK, i << WM8983_MCLKDIV_SHIFT); - - /* select the appropriate bclk divider */ - tmp = (wm8983->sysclk / fs_ratios[i].div) * 10; - for (i = 0; i < ARRAY_SIZE(bclk_divs); ++i) { - if (wm8983->bclk == tmp / bclk_divs[i]) - break; - } - - if (i == ARRAY_SIZE(bclk_divs)) { - dev_err(dai->dev, "No matching BCLK divider found\n"); - return -EINVAL; - } - - dev_dbg(dai->dev, "BCLK div = %d\n", i); - snd_soc_update_bits(codec, WM8983_CLOCK_GEN_CONTROL, - WM8983_BCLKDIV_MASK, i << WM8983_BCLKDIV_SHIFT); - - return 0; -} - -struct pll_div { - u32 div2:1; - u32 n:4; - u32 k:24; -}; - -#define FIXED_PLL_SIZE ((1ULL << 24) * 10) -static int pll_factors(struct pll_div *pll_div, unsigned int target, - unsigned int source) -{ - u64 Kpart; - unsigned long int K, Ndiv, Nmod; - - pll_div->div2 = 0; - Ndiv = target / source; - if (Ndiv < 6) { - source >>= 1; - pll_div->div2 = 1; - Ndiv = target / source; - } - - if (Ndiv < 6 || Ndiv > 12) { - printk(KERN_ERR "%s: WM8983 N value is not within" - " the recommended range: %lu\n", __func__, Ndiv); - return -EINVAL; - } - pll_div->n = Ndiv; - - Nmod = target % source; - Kpart = FIXED_PLL_SIZE * (u64)Nmod; - - do_div(Kpart, source); - - K = Kpart & 0xffffffff; - if ((K % 10) >= 5) - K += 5; - K /= 10; - pll_div->k = K; - return 0; -} - -static int wm8983_set_pll(struct snd_soc_dai *dai, int pll_id, - int source, unsigned int freq_in, - unsigned int freq_out) -{ - int ret; - struct snd_soc_codec *codec; - struct pll_div pll_div; - - codec = dai->codec; - if (freq_in && freq_out) { - ret = pll_factors(&pll_div, freq_out * 4 * 2, freq_in); - if (ret) - return ret; - } - - /* disable the PLL before re-programming it */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_PLLEN_MASK, 0); - - if (!freq_in || !freq_out) - return 0; - - /* set PLLN and PRESCALE */ - snd_soc_write(codec, WM8983_PLL_N, - (pll_div.div2 << WM8983_PLL_PRESCALE_SHIFT) - | pll_div.n); - /* set PLLK */ - snd_soc_write(codec, WM8983_PLL_K_3, pll_div.k & 0x1ff); - snd_soc_write(codec, WM8983_PLL_K_2, (pll_div.k >> 9) & 0x1ff); - snd_soc_write(codec, WM8983_PLL_K_1, (pll_div.k >> 18)); - /* enable the PLL */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_PLLEN_MASK, WM8983_PLLEN); - return 0; -} - -static int wm8983_set_sysclk(struct snd_soc_dai *dai, - int clk_id, unsigned int freq, int dir) -{ - struct snd_soc_codec *codec = dai->codec; - struct wm8983_priv *wm8983 = snd_soc_codec_get_drvdata(codec); - - switch (clk_id) { - case WM8983_CLKSRC_MCLK: - snd_soc_update_bits(codec, WM8983_CLOCK_GEN_CONTROL, - WM8983_CLKSEL_MASK, 0); - break; - case WM8983_CLKSRC_PLL: - snd_soc_update_bits(codec, WM8983_CLOCK_GEN_CONTROL, - WM8983_CLKSEL_MASK, WM8983_CLKSEL); - break; - default: - dev_err(dai->dev, "Unknown clock source: %d\n", clk_id); - return -EINVAL; - } - - wm8983->sysclk = freq; - return 0; -} - -static int wm8983_set_bias_level(struct snd_soc_codec *codec, - enum snd_soc_bias_level level) -{ - int ret; - - switch (level) { - case SND_SOC_BIAS_ON: - case SND_SOC_BIAS_PREPARE: - /* VMID at 100k */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_VMIDSEL_MASK, - 1 << WM8983_VMIDSEL_SHIFT); - break; - case SND_SOC_BIAS_STANDBY: - if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) { - ret = snd_soc_cache_sync(codec); - if (ret < 0) { - dev_err(codec->dev, "Failed to sync cache: %d\n", ret); - return ret; - } - /* enable anti-pop features */ - snd_soc_update_bits(codec, WM8983_OUT4_TO_ADC, - WM8983_POBCTRL_MASK | WM8983_DELEN_MASK, - WM8983_POBCTRL | WM8983_DELEN); - /* enable thermal shutdown */ - snd_soc_update_bits(codec, WM8983_OUTPUT_CTRL, - WM8983_TSDEN_MASK, WM8983_TSDEN); - /* enable BIASEN */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_BIASEN_MASK, WM8983_BIASEN); - /* VMID at 100k */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_VMIDSEL_MASK, - 1 << WM8983_VMIDSEL_SHIFT); - msleep(250); - /* disable anti-pop features */ - snd_soc_update_bits(codec, WM8983_OUT4_TO_ADC, - WM8983_POBCTRL_MASK | - WM8983_DELEN_MASK, 0); - } - - /* VMID at 500k */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_VMIDSEL_MASK, - 2 << WM8983_VMIDSEL_SHIFT); - break; - case SND_SOC_BIAS_OFF: - /* disable thermal shutdown */ - snd_soc_update_bits(codec, WM8983_OUTPUT_CTRL, - WM8983_TSDEN_MASK, 0); - /* disable VMIDSEL and BIASEN */ - snd_soc_update_bits(codec, WM8983_POWER_MANAGEMENT_1, - WM8983_VMIDSEL_MASK | WM8983_BIASEN_MASK, - 0); - /* wait for VMID to discharge */ - msleep(100); - snd_soc_write(codec, WM8983_POWER_MANAGEMENT_1, 0); - snd_soc_write(codec, WM8983_POWER_MANAGEMENT_2, 0); - snd_soc_write(codec, WM8983_POWER_MANAGEMENT_3, 0); - break; - } - - codec->dapm.bias_level = level; - return 0; -} - -#ifdef CONFIG_PM -static int wm8983_suspend(struct snd_soc_codec *codec, pm_message_t state) -{ - wm8983_set_bias_level(codec, SND_SOC_BIAS_OFF); - return 0; -} - -static int wm8983_resume(struct snd_soc_codec *codec) -{ - wm8983_set_bias_level(codec, SND_SOC_BIAS_STANDBY); - return 0; -} -#else -#define wm8983_suspend NULL -#define wm8983_resume NULL -#endif - -static int wm8983_remove(struct snd_soc_codec *codec) -{ - wm8983_set_bias_level(codec, SND_SOC_BIAS_OFF); - return 0; -} - -static int wm8983_probe(struct snd_soc_codec *codec) -{ - int ret; - struct wm8983_priv *wm8983 = snd_soc_codec_get_drvdata(codec); - int i; - - ret = snd_soc_codec_set_cache_io(codec, 7, 9, wm8983->control_type); - if (ret < 0) { - dev_err(codec->dev, "Failed to set cache i/o: %d\n", ret); - return ret; - } - - ret = snd_soc_write(codec, WM8983_SOFTWARE_RESET, 0x8983); - if (ret < 0) { - dev_err(codec->dev, "Failed to issue reset: %d\n", ret); - return ret; - } - - /* set the vol/gain update bits */ - for (i = 0; i < ARRAY_SIZE(vol_update_regs); ++i) - snd_soc_update_bits(codec, vol_update_regs[i], - 0x100, 0x100); - - /* mute all outputs and set PGAs to minimum gain */ - for (i = WM8983_LOUT1_HP_VOLUME_CTRL; - i <= WM8983_OUT4_MONO_MIX_CTRL; ++i) - snd_soc_update_bits(codec, i, 0x40, 0x40); - - /* enable soft mute */ - snd_soc_update_bits(codec, WM8983_DAC_CONTROL, - WM8983_SOFTMUTE_MASK, - WM8983_SOFTMUTE); - - /* enable BIASCUT */ - snd_soc_update_bits(codec, WM8983_BIAS_CTRL, - WM8983_BIASCUT, WM8983_BIASCUT); - return 0; -} - -static struct snd_soc_dai_ops wm8983_dai_ops = { - .digital_mute = wm8983_dac_mute, - .hw_params = wm8983_hw_params, - .set_fmt = wm8983_set_fmt, - .set_sysclk = wm8983_set_sysclk, - .set_pll = wm8983_set_pll -}; - -#define WM8983_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE | \ - SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S32_LE) - -static struct snd_soc_dai_driver wm8983_dai = { - .name = "wm8983-hifi", - .playback = { - .stream_name = "Playback", - .channels_min = 2, - .channels_max = 2, - .rates = SNDRV_PCM_RATE_8000_48000, - .formats = WM8983_FORMATS, - }, - .capture = { - .stream_name = "Capture", - .channels_min = 2, - .channels_max = 2, - .rates = SNDRV_PCM_RATE_8000_48000, - .formats = WM8983_FORMATS, - }, - .ops = &wm8983_dai_ops, - .symmetric_rates = 1 -}; - -static struct snd_soc_codec_driver soc_codec_dev_wm8983 = { - .probe = wm8983_probe, - .remove = wm8983_remove, - .suspend = wm8983_suspend, - .resume = wm8983_resume, - .set_bias_level = wm8983_set_bias_level, - .reg_cache_size = ARRAY_SIZE(wm8983_reg_defs), - .reg_word_size = sizeof(u16), - .reg_cache_default = wm8983_reg_defs, - .controls = wm8983_snd_controls, - .num_controls = ARRAY_SIZE(wm8983_snd_controls), - .dapm_widgets = wm8983_dapm_widgets, - .num_dapm_widgets = ARRAY_SIZE(wm8983_dapm_widgets), - .dapm_routes = wm8983_audio_map, - .num_dapm_routes = ARRAY_SIZE(wm8983_audio_map), - .readable_register = wm8983_readable -}; - -#if defined(CONFIG_SPI_MASTER) -static int __devinit wm8983_spi_probe(struct spi_device *spi) -{ - struct wm8983_priv *wm8983; - int ret; - - wm8983 = kzalloc(sizeof *wm8983, GFP_KERNEL); - if (!wm8983) - return -ENOMEM; - - wm8983->control_type = SND_SOC_SPI; - spi_set_drvdata(spi, wm8983); - - ret = snd_soc_register_codec(&spi->dev, - &soc_codec_dev_wm8983, &wm8983_dai, 1); - if (ret < 0) - kfree(wm8983); - return ret; -} - -static int __devexit wm8983_spi_remove(struct spi_device *spi) -{ - snd_soc_unregister_codec(&spi->dev); - kfree(spi_get_drvdata(spi)); - return 0; -} - -static struct spi_driver wm8983_spi_driver = { - .driver = { - .name = "wm8983", - .owner = THIS_MODULE, - }, - .probe = wm8983_spi_probe, - .remove = __devexit_p(wm8983_spi_remove) -}; -#endif - -#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) -static __devinit int wm8983_i2c_probe(struct i2c_client *i2c, - const struct i2c_device_id *id) -{ - struct wm8983_priv *wm8983; - int ret; - - wm8983 = kzalloc(sizeof *wm8983, GFP_KERNEL); - if (!wm8983) - return -ENOMEM; - - wm8983->control_type = SND_SOC_I2C; - i2c_set_clientdata(i2c, wm8983); - - ret = snd_soc_register_codec(&i2c->dev, - &soc_codec_dev_wm8983, &wm8983_dai, 1); - if (ret < 0) - kfree(wm8983); - return ret; -} - -static __devexit int wm8983_i2c_remove(struct i2c_client *client) -{ - snd_soc_unregister_codec(&client->dev); - kfree(i2c_get_clientdata(client)); - return 0; -} - -static const struct i2c_device_id wm8983_i2c_id[] = { - { "wm8983", 0 }, - { } -}; -MODULE_DEVICE_TABLE(i2c, wm8983_i2c_id); - -static struct i2c_driver wm8983_i2c_driver = { - .driver = { - .name = "wm8983", - .owner = THIS_MODULE, - }, - .probe = wm8983_i2c_probe, - .remove = __devexit_p(wm8983_i2c_remove), - .id_table = wm8983_i2c_id -}; -#endif - -static int __init wm8983_modinit(void) -{ - int ret = 0; - -#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) - ret = i2c_add_driver(&wm8983_i2c_driver); - if (ret) { - printk(KERN_ERR "Failed to register wm8983 I2C driver: %d\n", - ret); - } -#endif -#if defined(CONFIG_SPI_MASTER) - ret = spi_register_driver(&wm8983_spi_driver); - if (ret != 0) { - printk(KERN_ERR "Failed to register wm8983 SPI driver: %d\n", - ret); - } -#endif - return ret; -} -module_init(wm8983_modinit); - -static void __exit wm8983_exit(void) -{ -#if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) - i2c_del_driver(&wm8983_i2c_driver); -#endif -#if defined(CONFIG_SPI_MASTER) - spi_unregister_driver(&wm8983_spi_driver); -#endif -} -module_exit(wm8983_exit); - -MODULE_DESCRIPTION("ASoC WM8983 driver"); -MODULE_AUTHOR("Dimitris Papastamos "); -MODULE_LICENSE("GPL"); diff --git a/trunk/sound/soc/codecs/wm8983.h b/trunk/sound/soc/codecs/wm8983.h deleted file mode 100644 index 71ee619c2742..000000000000 --- a/trunk/sound/soc/codecs/wm8983.h +++ /dev/null @@ -1,1029 +0,0 @@ -/* - * wm8983.h -- WM8983 ALSA SoC Audio driver - * - * Copyright 2011 Wolfson Microelectronics plc - * - * Author: Dimitris Papastamos - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License version 2 as - * published by the Free Software Foundation. - */ - -#ifndef _WM8983_H -#define _WM8983_H - -/* - * Register values. - */ -#define WM8983_SOFTWARE_RESET 0x00 -#define WM8983_POWER_MANAGEMENT_1 0x01 -#define WM8983_POWER_MANAGEMENT_2 0x02 -#define WM8983_POWER_MANAGEMENT_3 0x03 -#define WM8983_AUDIO_INTERFACE 0x04 -#define WM8983_COMPANDING_CONTROL 0x05 -#define WM8983_CLOCK_GEN_CONTROL 0x06 -#define WM8983_ADDITIONAL_CONTROL 0x07 -#define WM8983_GPIO_CONTROL 0x08 -#define WM8983_JACK_DETECT_CONTROL_1 0x09 -#define WM8983_DAC_CONTROL 0x0A -#define WM8983_LEFT_DAC_DIGITAL_VOL 0x0B -#define WM8983_RIGHT_DAC_DIGITAL_VOL 0x0C -#define WM8983_JACK_DETECT_CONTROL_2 0x0D -#define WM8983_ADC_CONTROL 0x0E -#define WM8983_LEFT_ADC_DIGITAL_VOL 0x0F -#define WM8983_RIGHT_ADC_DIGITAL_VOL 0x10 -#define WM8983_EQ1_LOW_SHELF 0x12 -#define WM8983_EQ2_PEAK_1 0x13 -#define WM8983_EQ3_PEAK_2 0x14 -#define WM8983_EQ4_PEAK_3 0x15 -#define WM8983_EQ5_HIGH_SHELF 0x16 -#define WM8983_DAC_LIMITER_1 0x18 -#define WM8983_DAC_LIMITER_2 0x19 -#define WM8983_NOTCH_FILTER_1 0x1B -#define WM8983_NOTCH_FILTER_2 0x1C -#define WM8983_NOTCH_FILTER_3 0x1D -#define WM8983_NOTCH_FILTER_4 0x1E -#define WM8983_ALC_CONTROL_1 0x20 -#define WM8983_ALC_CONTROL_2 0x21 -#define WM8983_ALC_CONTROL_3 0x22 -#define WM8983_NOISE_GATE 0x23 -#define WM8983_PLL_N 0x24 -#define WM8983_PLL_K_1 0x25 -#define WM8983_PLL_K_2 0x26 -#define WM8983_PLL_K_3 0x27 -#define WM8983_3D_CONTROL 0x29 -#define WM8983_OUT4_TO_ADC 0x2A -#define WM8983_BEEP_CONTROL 0x2B -#define WM8983_INPUT_CTRL 0x2C -#define WM8983_LEFT_INP_PGA_GAIN_CTRL 0x2D -#define WM8983_RIGHT_INP_PGA_GAIN_CTRL 0x2E -#define WM8983_LEFT_ADC_BOOST_CTRL 0x2F -#define WM8983_RIGHT_ADC_BOOST_CTRL 0x30 -#define WM8983_OUTPUT_CTRL 0x31 -#define WM8983_LEFT_MIXER_CTRL 0x32 -#define WM8983_RIGHT_MIXER_CTRL 0x33 -#define WM8983_LOUT1_HP_VOLUME_CTRL 0x34 -#define WM8983_ROUT1_HP_VOLUME_CTRL 0x35 -#define WM8983_LOUT2_SPK_VOLUME_CTRL 0x36 -#define WM8983_ROUT2_SPK_VOLUME_CTRL 0x37 -#define WM8983_OUT3_MIXER_CTRL 0x38 -#define WM8983_OUT4_MONO_MIX_CTRL 0x39 -#define WM8983_BIAS_CTRL 0x3D - -#define WM8983_REGISTER_COUNT 59 -#define WM8983_MAX_REGISTER 0x3F - -/* - * Field Definitions. - */ - -/* - * R0 (0x00) - Software Reset - */ -#define WM8983_SOFTWARE_RESET_MASK 0x01FF /* SOFTWARE_RESET - [8:0] */ -#define WM8983_SOFTWARE_RESET_SHIFT 0 /* SOFTWARE_RESET - [8:0] */ -#define WM8983_SOFTWARE_RESET_WIDTH 9 /* SOFTWARE_RESET - [8:0] */ - -/* - * R1 (0x01) - Power management 1 - */ -#define WM8983_BUFDCOPEN 0x0100 /* BUFDCOPEN */ -#define WM8983_BUFDCOPEN_MASK 0x0100 /* BUFDCOPEN */ -#define WM8983_BUFDCOPEN_SHIFT 8 /* BUFDCOPEN */ -#define WM8983_BUFDCOPEN_WIDTH 1 /* BUFDCOPEN */ -#define WM8983_OUT4MIXEN 0x0080 /* OUT4MIXEN */ -#define WM8983_OUT4MIXEN_MASK 0x0080 /* OUT4MIXEN */ -#define WM8983_OUT4MIXEN_SHIFT 7 /* OUT4MIXEN */ -#define WM8983_OUT4MIXEN_WIDTH 1 /* OUT4MIXEN */ -#define WM8983_OUT3MIXEN 0x0040 /* OUT3MIXEN */ -#define WM8983_OUT3MIXEN_MASK 0x0040 /* OUT3MIXEN */ -#define WM8983_OUT3MIXEN_SHIFT 6 /* OUT3MIXEN */ -#define WM8983_OUT3MIXEN_WIDTH 1 /* OUT3MIXEN */ -#define WM8983_PLLEN 0x0020 /* PLLEN */ -#define WM8983_PLLEN_MASK 0x0020 /* PLLEN */ -#define WM8983_PLLEN_SHIFT 5 /* PLLEN */ -#define WM8983_PLLEN_WIDTH 1 /* PLLEN */ -#define WM8983_MICBEN 0x0010 /* MICBEN */ -#define WM8983_MICBEN_MASK 0x0010 /* MICBEN */ -#define WM8983_MICBEN_SHIFT 4 /* MICBEN */ -#define WM8983_MICBEN_WIDTH 1 /* MICBEN */ -#define WM8983_BIASEN 0x0008 /* BIASEN */ -#define WM8983_BIASEN_MASK 0x0008 /* BIASEN */ -#define WM8983_BIASEN_SHIFT 3 /* BIASEN */ -#define WM8983_BIASEN_WIDTH 1 /* BIASEN */ -#define WM8983_BUFIOEN 0x0004 /* BUFIOEN */ -#define WM8983_BUFIOEN_MASK 0x0004 /* BUFIOEN */ -#define WM8983_BUFIOEN_SHIFT 2 /* BUFIOEN */ -#define WM8983_BUFIOEN_WIDTH 1 /* BUFIOEN */ -#define WM8983_VMIDSEL_MASK 0x0003 /* VMIDSEL - [1:0] */ -#define WM8983_VMIDSEL_SHIFT 0 /* VMIDSEL - [1:0] */ -#define WM8983_VMIDSEL_WIDTH 2 /* VMIDSEL - [1:0] */ - -/* - * R2 (0x02) - Power management 2 - */ -#define WM8983_ROUT1EN 0x0100 /* ROUT1EN */ -#define WM8983_ROUT1EN_MASK 0x0100 /* ROUT1EN */ -#define WM8983_ROUT1EN_SHIFT 8 /* ROUT1EN */ -#define WM8983_ROUT1EN_WIDTH 1 /* ROUT1EN */ -#define WM8983_LOUT1EN 0x0080 /* LOUT1EN */ -#define WM8983_LOUT1EN_MASK 0x0080 /* LOUT1EN */ -#define WM8983_LOUT1EN_SHIFT 7 /* LOUT1EN */ -#define WM8983_LOUT1EN_WIDTH 1 /* LOUT1EN */ -#define WM8983_SLEEP 0x0040 /* SLEEP */ -#define WM8983_SLEEP_MASK 0x0040 /* SLEEP */ -#define WM8983_SLEEP_SHIFT 6 /* SLEEP */ -#define WM8983_SLEEP_WIDTH 1 /* SLEEP */ -#define WM8983_BOOSTENR 0x0020 /* BOOSTENR */ -#define WM8983_BOOSTENR_MASK 0x0020 /* BOOSTENR */ -#define WM8983_BOOSTENR_SHIFT 5 /* BOOSTENR */ -#define WM8983_BOOSTENR_WIDTH 1 /* BOOSTENR */ -#define WM8983_BOOSTENL 0x0010 /* BOOSTENL */ -#define WM8983_BOOSTENL_MASK 0x0010 /* BOOSTENL */ -#define WM8983_BOOSTENL_SHIFT 4 /* BOOSTENL */ -#define WM8983_BOOSTENL_WIDTH 1 /* BOOSTENL */ -#define WM8983_INPGAENR 0x0008 /* INPGAENR */ -#define WM8983_INPGAENR_MASK 0x0008 /* INPGAENR */ -#define WM8983_INPGAENR_SHIFT 3 /* INPGAENR */ -#define WM8983_INPGAENR_WIDTH 1 /* INPGAENR */ -#define WM8983_INPPGAENL 0x0004 /* INPPGAENL */ -#define WM8983_INPPGAENL_MASK 0x0004 /* INPPGAENL */ -#define WM8983_INPPGAENL_SHIFT 2 /* INPPGAENL */ -#define WM8983_INPPGAENL_WIDTH 1 /* INPPGAENL */ -#define WM8983_ADCENR 0x0002 /* ADCENR */ -#define WM8983_ADCENR_MASK 0x0002 /* ADCENR */ -#define WM8983_ADCENR_SHIFT 1 /* ADCENR */ -#define WM8983_ADCENR_WIDTH 1 /* ADCENR */ -#define WM8983_ADCENL 0x0001 /* ADCENL */ -#define WM8983_ADCENL_MASK 0x0001 /* ADCENL */ -#define WM8983_ADCENL_SHIFT 0 /* ADCENL */ -#define WM8983_ADCENL_WIDTH 1 /* ADCENL */ - -/* - * R3 (0x03) - Power management 3 - */ -#define WM8983_OUT4EN 0x0100 /* OUT4EN */ -#define WM8983_OUT4EN_MASK 0x0100 /* OUT4EN */ -#define WM8983_OUT4EN_SHIFT 8 /* OUT4EN */ -#define WM8983_OUT4EN_WIDTH 1 /* OUT4EN */ -#define WM8983_OUT3EN 0x0080 /* OUT3EN */ -#define WM8983_OUT3EN_MASK 0x0080 /* OUT3EN */ -#define WM8983_OUT3EN_SHIFT 7 /* OUT3EN */ -#define WM8983_OUT3EN_WIDTH 1 /* OUT3EN */ -#define WM8983_LOUT2EN 0x0040 /* LOUT2EN */ -#define WM8983_LOUT2EN_MASK 0x0040 /* LOUT2EN */ -#define WM8983_LOUT2EN_SHIFT 6 /* LOUT2EN */ -#define WM8983_LOUT2EN_WIDTH 1 /* LOUT2EN */ -#define WM8983_ROUT2EN 0x0020 /* ROUT2EN */ -#define WM8983_ROUT2EN_MASK 0x0020 /* ROUT2EN */ -#define WM8983_ROUT2EN_SHIFT 5 /* ROUT2EN */ -#define WM8983_ROUT2EN_WIDTH 1 /* ROUT2EN */ -#define WM8983_RMIXEN 0x0008 /* RMIXEN */ -#define WM8983_RMIXEN_MASK 0x0008 /* RMIXEN */ -#define WM8983_RMIXEN_SHIFT 3 /* RMIXEN */ -#define WM8983_RMIXEN_WIDTH 1 /* RMIXEN */ -#define WM8983_LMIXEN 0x0004 /* LMIXEN */ -#define WM8983_LMIXEN_MASK 0x0004 /* LMIXEN */ -#define WM8983_LMIXEN_SHIFT 2 /* LMIXEN */ -#define WM8983_LMIXEN_WIDTH 1 /* LMIXEN */ -#define WM8983_DACENR 0x0002 /* DACENR */ -#define WM8983_DACENR_MASK 0x0002 /* DACENR */ -#define WM8983_DACENR_SHIFT 1 /* DACENR */ -#define WM8983_DACENR_WIDTH 1 /* DACENR */ -#define WM8983_DACENL 0x0001 /* DACENL */ -#define WM8983_DACENL_MASK 0x0001 /* DACENL */ -#define WM8983_DACENL_SHIFT 0 /* DACENL */ -#define WM8983_DACENL_WIDTH 1 /* DACENL */ - -/* - * R4 (0x04) - Audio Interface - */ -#define WM8983_BCP 0x0100 /* BCP */ -#define WM8983_BCP_MASK 0x0100 /* BCP */ -#define WM8983_BCP_SHIFT 8 /* BCP */ -#define WM8983_BCP_WIDTH 1 /* BCP */ -#define WM8983_LRCP 0x0080 /* LRCP */ -#define WM8983_LRCP_MASK 0x0080 /* LRCP */ -#define WM8983_LRCP_SHIFT 7 /* LRCP */ -#define WM8983_LRCP_WIDTH 1 /* LRCP */ -#define WM8983_WL_MASK 0x0060 /* WL - [6:5] */ -#define WM8983_WL_SHIFT 5 /* WL - [6:5] */ -#define WM8983_WL_WIDTH 2 /* WL - [6:5] */ -#define WM8983_FMT_MASK 0x0018 /* FMT - [4:3] */ -#define WM8983_FMT_SHIFT 3 /* FMT - [4:3] */ -#define WM8983_FMT_WIDTH 2 /* FMT - [4:3] */ -#define WM8983_DLRSWAP 0x0004 /* DLRSWAP */ -#define WM8983_DLRSWAP_MASK 0x0004 /* DLRSWAP */ -#define WM8983_DLRSWAP_SHIFT 2 /* DLRSWAP */ -#define WM8983_DLRSWAP_WIDTH 1 /* DLRSWAP */ -#define WM8983_ALRSWAP 0x0002 /* ALRSWAP */ -#define WM8983_ALRSWAP_MASK 0x0002 /* ALRSWAP */ -#define WM8983_ALRSWAP_SHIFT 1 /* ALRSWAP */ -#define WM8983_ALRSWAP_WIDTH 1 /* ALRSWAP */ -#define WM8983_MONO 0x0001 /* MONO */ -#define WM8983_MONO_MASK 0x0001 /* MONO */ -#define WM8983_MONO_SHIFT 0 /* MONO */ -#define WM8983_MONO_WIDTH 1 /* MONO */ - -/* - * R5 (0x05) - Companding control - */ -#define WM8983_WL8 0x0020 /* WL8 */ -#define WM8983_WL8_MASK 0x0020 /* WL8 */ -#define WM8983_WL8_SHIFT 5 /* WL8 */ -#define WM8983_WL8_WIDTH 1 /* WL8 */ -#define WM8983_DAC_COMP_MASK 0x0018 /* DAC_COMP - [4:3] */ -#define WM8983_DAC_COMP_SHIFT 3 /* DAC_COMP - [4:3] */ -#define WM8983_DAC_COMP_WIDTH 2 /* DAC_COMP - [4:3] */ -#define WM8983_ADC_COMP_MASK 0x0006 /* ADC_COMP - [2:1] */ -#define WM8983_ADC_COMP_SHIFT 1 /* ADC_COMP - [2:1] */ -#define WM8983_ADC_COMP_WIDTH 2 /* ADC_COMP - [2:1] */ -#define WM8983_LOOPBACK 0x0001 /* LOOPBACK */ -#define WM8983_LOOPBACK_MASK 0x0001 /* LOOPBACK */ -#define WM8983_LOOPBACK_SHIFT 0 /* LOOPBACK */ -#define WM8983_LOOPBACK_WIDTH 1 /* LOOPBACK */ - -/* - * R6 (0x06) - Clock Gen control - */ -#define WM8983_CLKSEL 0x0100 /* CLKSEL */ -#define WM8983_CLKSEL_MASK 0x0100 /* CLKSEL */ -#define WM8983_CLKSEL_SHIFT 8 /* CLKSEL */ -#define WM8983_CLKSEL_WIDTH 1 /* CLKSEL */ -#define WM8983_MCLKDIV_MASK 0x00E0 /* MCLKDIV - [7:5] */ -#define WM8983_MCLKDIV_SHIFT 5 /* MCLKDIV - [7:5] */ -#define WM8983_MCLKDIV_WIDTH 3 /* MCLKDIV - [7:5] */ -#define WM8983_BCLKDIV_MASK 0x001C /* BCLKDIV - [4:2] */ -#define WM8983_BCLKDIV_SHIFT 2 /* BCLKDIV - [4:2] */ -#define WM8983_BCLKDIV_WIDTH 3 /* BCLKDIV - [4:2] */ -#define WM8983_MS 0x0001 /* MS */ -#define WM8983_MS_MASK 0x0001 /* MS */ -#define WM8983_MS_SHIFT 0 /* MS */ -#define WM8983_MS_WIDTH 1 /* MS */ - -/* - * R7 (0x07) - Additional control - */ -#define WM8983_SR_MASK 0x000E /* SR - [3:1] */ -#define WM8983_SR_SHIFT 1 /* SR - [3:1] */ -#define WM8983_SR_WIDTH 3 /* SR - [3:1] */ -#define WM8983_SLOWCLKEN 0x0001 /* SLOWCLKEN */ -#define WM8983_SLOWCLKEN_MASK 0x0001 /* SLOWCLKEN */ -#define WM8983_SLOWCLKEN_SHIFT 0 /* SLOWCLKEN */ -#define WM8983_SLOWCLKEN_WIDTH 1 /* SLOWCLKEN */ - -/* - * R8 (0x08) - GPIO Control - */ -#define WM8983_OPCLKDIV_MASK 0x0030 /* OPCLKDIV - [5:4] */ -#define WM8983_OPCLKDIV_SHIFT 4 /* OPCLKDIV - [5:4] */ -#define WM8983_OPCLKDIV_WIDTH 2 /* OPCLKDIV - [5:4] */ -#define WM8983_GPIO1POL 0x0008 /* GPIO1POL */ -#define WM8983_GPIO1POL_MASK 0x0008 /* GPIO1POL */ -#define WM8983_GPIO1POL_SHIFT 3 /* GPIO1POL */ -#define WM8983_GPIO1POL_WIDTH 1 /* GPIO1POL */ -#define WM8983_GPIO1SEL_MASK 0x0007 /* GPIO1SEL - [2:0] */ -#define WM8983_GPIO1SEL_SHIFT 0 /* GPIO1SEL - [2:0] */ -#define WM8983_GPIO1SEL_WIDTH 3 /* GPIO1SEL - [2:0] */ - -/* - * R9 (0x09) - Jack Detect Control 1 - */ -#define WM8983_JD_VMID1 0x0100 /* JD_VMID1 */ -#define WM8983_JD_VMID1_MASK 0x0100 /* JD_VMID1 */ -#define WM8983_JD_VMID1_SHIFT 8 /* JD_VMID1 */ -#define WM8983_JD_VMID1_WIDTH 1 /* JD_VMID1 */ -#define WM8983_JD_VMID0 0x0080 /* JD_VMID0 */ -#define WM8983_JD_VMID0_MASK 0x0080 /* JD_VMID0 */ -#define WM8983_JD_VMID0_SHIFT 7 /* JD_VMID0 */ -#define WM8983_JD_VMID0_WIDTH 1 /* JD_VMID0 */ -#define WM8983_JD_EN 0x0040 /* JD_EN */ -#define WM8983_JD_EN_MASK 0x0040 /* JD_EN */ -#define WM8983_JD_EN_SHIFT 6 /* JD_EN */ -#define WM8983_JD_EN_WIDTH 1 /* JD_EN */ -#define WM8983_JD_SEL_MASK 0x0030 /* JD_SEL - [5:4] */ -#define WM8983_JD_SEL_SHIFT 4 /* JD_SEL - [5:4] */ -#define WM8983_JD_SEL_WIDTH 2 /* JD_SEL - [5:4] */ - -/* - * R10 (0x0A) - DAC Control - */ -#define WM8983_SOFTMUTE 0x0040 /* SOFTMUTE */ -#define WM8983_SOFTMUTE_MASK 0x0040 /* SOFTMUTE */ -#define WM8983_SOFTMUTE_SHIFT 6 /* SOFTMUTE */ -#define WM8983_SOFTMUTE_WIDTH 1 /* SOFTMUTE */ -#define WM8983_DACOSR128 0x0008 /* DACOSR128 */ -#define WM8983_DACOSR128_MASK 0x0008 /* DACOSR128 */ -#define WM8983_DACOSR128_SHIFT 3 /* DACOSR128 */ -#define WM8983_DACOSR128_WIDTH 1 /* DACOSR128 */ -#define WM8983_AMUTE 0x0004 /* AMUTE */ -#define WM8983_AMUTE_MASK 0x0004 /* AMUTE */ -#define WM8983_AMUTE_SHIFT 2 /* AMUTE */ -#define WM8983_AMUTE_WIDTH 1 /* AMUTE */ -#define WM8983_DACRPOL 0x0002 /* DACRPOL */ -#define WM8983_DACRPOL_MASK 0x0002 /* DACRPOL */ -#define WM8983_DACRPOL_SHIFT 1 /* DACRPOL */ -#define WM8983_DACRPOL_WIDTH 1 /* DACRPOL */ -#define WM8983_DACLPOL 0x0001 /* DACLPOL */ -#define WM8983_DACLPOL_MASK 0x0001 /* DACLPOL */ -#define WM8983_DACLPOL_SHIFT 0 /* DACLPOL */ -#define WM8983_DACLPOL_WIDTH 1 /* DACLPOL */ - -/* - * R11 (0x0B) - Left DAC digital Vol - */ -#define WM8983_DACVU 0x0100 /* DACVU */ -#define WM8983_DACVU_MASK 0x0100 /* DACVU */ -#define WM8983_DACVU_SHIFT 8 /* DACVU */ -#define WM8983_DACVU_WIDTH 1 /* DACVU */ -#define WM8983_DACLVOL_MASK 0x00FF /* DACLVOL - [7:0] */ -#define WM8983_DACLVOL_SHIFT 0 /* DACLVOL - [7:0] */ -#define WM8983_DACLVOL_WIDTH 8 /* DACLVOL - [7:0] */ - -/* - * R12 (0x0C) - Right DAC digital vol - */ -#define WM8983_DACVU 0x0100 /* DACVU */ -#define WM8983_DACVU_MASK 0x0100 /* DACVU */ -#define WM8983_DACVU_SHIFT 8 /* DACVU */ -#define WM8983_DACVU_WIDTH 1 /* DACVU */ -#define WM8983_DACRVOL_MASK 0x00FF /* DACRVOL - [7:0] */ -#define WM8983_DACRVOL_SHIFT 0 /* DACRVOL - [7:0] */ -#define WM8983_DACRVOL_WIDTH 8 /* DACRVOL - [7:0] */ - -/* - * R13 (0x0D) - Jack Detect Control 2 - */ -#define WM8983_JD_EN1_MASK 0x00F0 /* JD_EN1 - [7:4] */ -#define WM8983_JD_EN1_SHIFT 4 /* JD_EN1 - [7:4] */ -#define WM8983_JD_EN1_WIDTH 4 /* JD_EN1 - [7:4] */ -#define WM8983_JD_EN0_MASK 0x000F /* JD_EN0 - [3:0] */ -#define WM8983_JD_EN0_SHIFT 0 /* JD_EN0 - [3:0] */ -#define WM8983_JD_EN0_WIDTH 4 /* JD_EN0 - [3:0] */ - -/* - * R14 (0x0E) - ADC Control - */ -#define WM8983_HPFEN 0x0100 /* HPFEN */ -#define WM8983_HPFEN_MASK 0x0100 /* HPFEN */ -#define WM8983_HPFEN_SHIFT 8 /* HPFEN */ -#define WM8983_HPFEN_WIDTH 1 /* HPFEN */ -#define WM8983_HPFAPP 0x0080 /* HPFAPP */ -#define WM8983_HPFAPP_MASK 0x0080 /* HPFAPP */ -#define WM8983_HPFAPP_SHIFT 7 /* HPFAPP */ -#define WM8983_HPFAPP_WIDTH 1 /* HPFAPP */ -#define WM8983_HPFCUT_MASK 0x0070 /* HPFCUT - [6:4] */ -#define WM8983_HPFCUT_SHIFT 4 /* HPFCUT - [6:4] */ -#define WM8983_HPFCUT_WIDTH 3 /* HPFCUT - [6:4] */ -#define WM8983_ADCOSR128 0x0008 /* ADCOSR128 */ -#define WM8983_ADCOSR128_MASK 0x0008 /* ADCOSR128 */ -#define WM8983_ADCOSR128_SHIFT 3 /* ADCOSR128 */ -#define WM8983_ADCOSR128_WIDTH 1 /* ADCOSR128 */ -#define WM8983_ADCRPOL 0x0002 /* ADCRPOL */ -#define WM8983_ADCRPOL_MASK 0x0002 /* ADCRPOL */ -#define WM8983_ADCRPOL_SHIFT 1 /* ADCRPOL */ -#define WM8983_ADCRPOL_WIDTH 1 /* ADCRPOL */ -#define WM8983_ADCLPOL 0x0001 /* ADCLPOL */ -#define WM8983_ADCLPOL_MASK 0x0001 /* ADCLPOL */ -#define WM8983_ADCLPOL_SHIFT 0 /* ADCLPOL */ -#define WM8983_ADCLPOL_WIDTH 1 /* ADCLPOL */ - -/* - * R15 (0x0F) - Left ADC Digital Vol - */ -#define WM8983_ADCVU 0x0100 /* ADCVU */ -#define WM8983_ADCVU_MASK 0x0100 /* ADCVU */ -#define WM8983_ADCVU_SHIFT 8 /* ADCVU */ -#define WM8983_ADCVU_WIDTH 1 /* ADCVU */ -#define WM8983_ADCLVOL_MASK 0x00FF /* ADCLVOL - [7:0] */ -#define WM8983_ADCLVOL_SHIFT 0 /* ADCLVOL - [7:0] */ -#define WM8983_ADCLVOL_WIDTH 8 /* ADCLVOL - [7:0] */ - -/* - * R16 (0x10) - Right ADC Digital Vol - */ -#define WM8983_ADCVU 0x0100 /* ADCVU */ -#define WM8983_ADCVU_MASK 0x0100 /* ADCVU */ -#define WM8983_ADCVU_SHIFT 8 /* ADCVU */ -#define WM8983_ADCVU_WIDTH 1 /* ADCVU */ -#define WM8983_ADCRVOL_MASK 0x00FF /* ADCRVOL - [7:0] */ -#define WM8983_ADCRVOL_SHIFT 0 /* ADCRVOL - [7:0] */ -#define WM8983_ADCRVOL_WIDTH 8 /* ADCRVOL - [7:0] */ - -/* - * R18 (0x12) - EQ1 - low shelf - */ -#define WM8983_EQ3DMODE 0x0100 /* EQ3DMODE */ -#define WM8983_EQ3DMODE_MASK 0x0100 /* EQ3DMODE */ -#define WM8983_EQ3DMODE_SHIFT 8 /* EQ3DMODE */ -#define WM8983_EQ3DMODE_WIDTH 1 /* EQ3DMODE */ -#define WM8983_EQ1C_MASK 0x0060 /* EQ1C - [6:5] */ -#define WM8983_EQ1C_SHIFT 5 /* EQ1C - [6:5] */ -#define WM8983_EQ1C_WIDTH 2 /* EQ1C - [6:5] */ -#define WM8983_EQ1G_MASK 0x001F /* EQ1G - [4:0] */ -#define WM8983_EQ1G_SHIFT 0 /* EQ1G - [4:0] */ -#define WM8983_EQ1G_WIDTH 5 /* EQ1G - [4:0] */ - -/* - * R19 (0x13) - EQ2 - peak 1 - */ -#define WM8983_EQ2BW 0x0100 /* EQ2BW */ -#define WM8983_EQ2BW_MASK 0x0100 /* EQ2BW */ -#define WM8983_EQ2BW_SHIFT 8 /* EQ2BW */ -#define WM8983_EQ2BW_WIDTH 1 /* EQ2BW */ -#define WM8983_EQ2C_MASK 0x0060 /* EQ2C - [6:5] */ -#define WM8983_EQ2C_SHIFT 5 /* EQ2C - [6:5] */ -#define WM8983_EQ2C_WIDTH 2 /* EQ2C - [6:5] */ -#define WM8983_EQ2G_MASK 0x001F /* EQ2G - [4:0] */ -#define WM8983_EQ2G_SHIFT 0 /* EQ2G - [4:0] */ -#define WM8983_EQ2G_WIDTH 5 /* EQ2G - [4:0] */ - -/* - * R20 (0x14) - EQ3 - peak 2 - */ -#define WM8983_EQ3BW 0x0100 /* EQ3BW */ -#define WM8983_EQ3BW_MASK 0x0100 /* EQ3BW */ -#define WM8983_EQ3BW_SHIFT 8 /* EQ3BW */ -#define WM8983_EQ3BW_WIDTH 1 /* EQ3BW */ -#define WM8983_EQ3C_MASK 0x0060 /* EQ3C - [6:5] */ -#define WM8983_EQ3C_SHIFT 5 /* EQ3C - [6:5] */ -#define WM8983_EQ3C_WIDTH 2 /* EQ3C - [6:5] */ -#define WM8983_EQ3G_MASK 0x001F /* EQ3G - [4:0] */ -#define WM8983_EQ3G_SHIFT 0 /* EQ3G - [4:0] */ -#define WM8983_EQ3G_WIDTH 5 /* EQ3G - [4:0] */ - -/* - * R21 (0x15) - EQ4 - peak 3 - */ -#define WM8983_EQ4BW 0x0100 /* EQ4BW */ -#define WM8983_EQ4BW_MASK 0x0100 /* EQ4BW */ -#define WM8983_EQ4BW_SHIFT 8 /* EQ4BW */ -#define WM8983_EQ4BW_WIDTH 1 /* EQ4BW */ -#define WM8983_EQ4C_MASK 0x0060 /* EQ4C - [6:5] */ -#define WM8983_EQ4C_SHIFT 5 /* EQ4C - [6:5] */ -#define WM8983_EQ4C_WIDTH 2 /* EQ4C - [6:5] */ -#define WM8983_EQ4G_MASK 0x001F /* EQ4G - [4:0] */ -#define WM8983_EQ4G_SHIFT 0 /* EQ4G - [4:0] */ -#define WM8983_EQ4G_WIDTH 5 /* EQ4G - [4:0] */ - -/* - * R22 (0x16) - EQ5 - high shelf - */ -#define WM8983_EQ5C_MASK 0x0060 /* EQ5C - [6:5] */ -#define WM8983_EQ5C_SHIFT 5 /* EQ5C - [6:5] */ -#define WM8983_EQ5C_WIDTH 2 /* EQ5C - [6:5] */ -#define WM8983_EQ5G_MASK 0x001F /* EQ5G - [4:0] */ -#define WM8983_EQ5G_SHIFT 0 /* EQ5G - [4:0] */ -#define WM8983_EQ5G_WIDTH 5 /* EQ5G - [4:0] */ - -/* - * R24 (0x18) - DAC Limiter 1 - */ -#define WM8983_LIMEN 0x0100 /* LIMEN */ -#define WM8983_LIMEN_MASK 0x0100 /* LIMEN */ -#define WM8983_LIMEN_SHIFT 8 /* LIMEN */ -#define WM8983_LIMEN_WIDTH 1 /* LIMEN */ -#define WM8983_LIMDCY_MASK 0x00F0 /* LIMDCY - [7:4] */ -#define WM8983_LIMDCY_SHIFT 4 /* LIMDCY - [7:4] */ -#define WM8983_LIMDCY_WIDTH 4 /* LIMDCY - [7:4] */ -#define WM8983_LIMATK_MASK 0x000F /* LIMATK - [3:0] */ -#define WM8983_LIMATK_SHIFT 0 /* LIMATK - [3:0] */ -#define WM8983_LIMATK_WIDTH 4 /* LIMATK - [3:0] */ - -/* - * R25 (0x19) - DAC Limiter 2 - */ -#define WM8983_LIMLVL_MASK 0x0070 /* LIMLVL - [6:4] */ -#define WM8983_LIMLVL_SHIFT 4 /* LIMLVL - [6:4] */ -#define WM8983_LIMLVL_WIDTH 3 /* LIMLVL - [6:4] */ -#define WM8983_LIMBOOST_MASK 0x000F /* LIMBOOST - [3:0] */ -#define WM8983_LIMBOOST_SHIFT 0 /* LIMBOOST - [3:0] */ -#define WM8983_LIMBOOST_WIDTH 4 /* LIMBOOST - [3:0] */ - -/* - * R27 (0x1B) - Notch Filter 1 - */ -#define WM8983_NFU 0x0100 /* NFU */ -#define WM8983_NFU_MASK 0x0100 /* NFU */ -#define WM8983_NFU_SHIFT 8 /* NFU */ -#define WM8983_NFU_WIDTH 1 /* NFU */ -#define WM8983_NFEN 0x0080 /* NFEN */ -#define WM8983_NFEN_MASK 0x0080 /* NFEN */ -#define WM8983_NFEN_SHIFT 7 /* NFEN */ -#define WM8983_NFEN_WIDTH 1 /* NFEN */ -#define WM8983_NFA0_13_7_MASK 0x007F /* NFA0(13:7) - [6:0] */ -#define WM8983_NFA0_13_7_SHIFT 0 /* NFA0(13:7) - [6:0] */ -#define WM8983_NFA0_13_7_WIDTH 7 /* NFA0(13:7) - [6:0] */ - -/* - * R28 (0x1C) - Notch Filter 2 - */ -#define WM8983_NFU 0x0100 /* NFU */ -#define WM8983_NFU_MASK 0x0100 /* NFU */ -#define WM8983_NFU_SHIFT 8 /* NFU */ -#define WM8983_NFU_WIDTH 1 /* NFU */ -#define WM8983_NFA0_6_0_MASK 0x007F /* NFA0(6:0) - [6:0] */ -#define WM8983_NFA0_6_0_SHIFT 0 /* NFA0(6:0) - [6:0] */ -#define WM8983_NFA0_6_0_WIDTH 7 /* NFA0(6:0) - [6:0] */ - -/* - * R29 (0x1D) - Notch Filter 3 - */ -#define WM8983_NFU 0x0100 /* NFU */ -#define WM8983_NFU_MASK 0x0100 /* NFU */ -#define WM8983_NFU_SHIFT 8 /* NFU */ -#define WM8983_NFU_WIDTH 1 /* NFU */ -#define WM8983_NFA1_13_7_MASK 0x007F /* NFA1(13:7) - [6:0] */ -#define WM8983_NFA1_13_7_SHIFT 0 /* NFA1(13:7) - [6:0] */ -#define WM8983_NFA1_13_7_WIDTH 7 /* NFA1(13:7) - [6:0] */ - -/* - * R30 (0x1E) - Notch Filter 4 - */ -#define WM8983_NFU 0x0100 /* NFU */ -#define WM8983_NFU_MASK 0x0100 /* NFU */ -#define WM8983_NFU_SHIFT 8 /* NFU */ -#define WM8983_NFU_WIDTH 1 /* NFU */ -#define WM8983_NFA1_6_0_MASK 0x007F /* NFA1(6:0) - [6:0] */ -#define WM8983_NFA1_6_0_SHIFT 0 /* NFA1(6:0) - [6:0] */ -#define WM8983_NFA1_6_0_WIDTH 7 /* NFA1(6:0) - [6:0] */ - -/* - * R32 (0x20) - ALC control 1 - */ -#define WM8983_ALCSEL_MASK 0x0180 /* ALCSEL - [8:7] */ -#define WM8983_ALCSEL_SHIFT 7 /* ALCSEL - [8:7] */ -#define WM8983_ALCSEL_WIDTH 2 /* ALCSEL - [8:7] */ -#define WM8983_ALCMAX_MASK 0x0038 /* ALCMAX - [5:3] */ -#define WM8983_ALCMAX_SHIFT 3 /* ALCMAX - [5:3] */ -#define WM8983_ALCMAX_WIDTH 3 /* ALCMAX - [5:3] */ -#define WM8983_ALCMIN_MASK 0x0007 /* ALCMIN - [2:0] */ -#define WM8983_ALCMIN_SHIFT 0 /* ALCMIN - [2:0] */ -#define WM8983_ALCMIN_WIDTH 3 /* ALCMIN - [2:0] */ - -/* - * R33 (0x21) - ALC control 2 - */ -#define WM8983_ALCHLD_MASK 0x00F0 /* ALCHLD - [7:4] */ -#define WM8983_ALCHLD_SHIFT 4 /* ALCHLD - [7:4] */ -#define WM8983_ALCHLD_WIDTH 4 /* ALCHLD - [7:4] */ -#define WM8983_ALCLVL_MASK 0x000F /* ALCLVL - [3:0] */ -#define WM8983_ALCLVL_SHIFT 0 /* ALCLVL - [3:0] */ -#define WM8983_ALCLVL_WIDTH 4 /* ALCLVL - [3:0] */ - -/* - * R34 (0x22) - ALC control 3 - */ -#define WM8983_ALCMODE 0x0100 /* ALCMODE */ -#define WM8983_ALCMODE_MASK 0x0100 /* ALCMODE */ -#define WM8983_ALCMODE_SHIFT 8 /* ALCMODE */ -#define WM8983_ALCMODE_WIDTH 1 /* ALCMODE */ -#define WM8983_ALCDCY_MASK 0x00F0 /* ALCDCY - [7:4] */ -#define WM8983_ALCDCY_SHIFT 4 /* ALCDCY - [7:4] */ -#define WM8983_ALCDCY_WIDTH 4 /* ALCDCY - [7:4] */ -#define WM8983_ALCATK_MASK 0x000F /* ALCATK - [3:0] */ -#define WM8983_ALCATK_SHIFT 0 /* ALCATK - [3:0] */ -#define WM8983_ALCATK_WIDTH 4 /* ALCATK - [3:0] */ - -/* - * R35 (0x23) - Noise Gate - */ -#define WM8983_NGEN 0x0008 /* NGEN */ -#define WM8983_NGEN_MASK 0x0008 /* NGEN */ -#define WM8983_NGEN_SHIFT 3 /* NGEN */ -#define WM8983_NGEN_WIDTH 1 /* NGEN */ -#define WM8983_NGTH_MASK 0x0007 /* NGTH - [2:0] */ -#define WM8983_NGTH_SHIFT 0 /* NGTH - [2:0] */ -#define WM8983_NGTH_WIDTH 3 /* NGTH - [2:0] */ - -/* - * R36 (0x24) - PLL N - */ -#define WM8983_PLL_PRESCALE 0x0010 /* PLL_PRESCALE */ -#define WM8983_PLL_PRESCALE_MASK 0x0010 /* PLL_PRESCALE */ -#define WM8983_PLL_PRESCALE_SHIFT 4 /* PLL_PRESCALE */ -#define WM8983_PLL_PRESCALE_WIDTH 1 /* PLL_PRESCALE */ -#define WM8983_PLLN_MASK 0x000F /* PLLN - [3:0] */ -#define WM8983_PLLN_SHIFT 0 /* PLLN - [3:0] */ -#define WM8983_PLLN_WIDTH 4 /* PLLN - [3:0] */ - -/* - * R37 (0x25) - PLL K 1 - */ -#define WM8983_PLLK_23_18_MASK 0x003F /* PLLK(23:18) - [5:0] */ -#define WM8983_PLLK_23_18_SHIFT 0 /* PLLK(23:18) - [5:0] */ -#define WM8983_PLLK_23_18_WIDTH 6 /* PLLK(23:18) - [5:0] */ - -/* - * R38 (0x26) - PLL K 2 - */ -#define WM8983_PLLK_17_9_MASK 0x01FF /* PLLK(17:9) - [8:0] */ -#define WM8983_PLLK_17_9_SHIFT 0 /* PLLK(17:9) - [8:0] */ -#define WM8983_PLLK_17_9_WIDTH 9 /* PLLK(17:9) - [8:0] */ - -/* - * R39 (0x27) - PLL K 3 - */ -#define WM8983_PLLK_8_0_MASK 0x01FF /* PLLK(8:0) - [8:0] */ -#define WM8983_PLLK_8_0_SHIFT 0 /* PLLK(8:0) - [8:0] */ -#define WM8983_PLLK_8_0_WIDTH 9 /* PLLK(8:0) - [8:0] */ - -/* - * R41 (0x29) - 3D control - */ -#define WM8983_DEPTH3D_MASK 0x000F /* DEPTH3D - [3:0] */ -#define WM8983_DEPTH3D_SHIFT 0 /* DEPTH3D - [3:0] */ -#define WM8983_DEPTH3D_WIDTH 4 /* DEPTH3D - [3:0] */ - -/* - * R42 (0x2A) - OUT4 to ADC - */ -#define WM8983_OUT4_2ADCVOL_MASK 0x01C0 /* OUT4_2ADCVOL - [8:6] */ -#define WM8983_OUT4_2ADCVOL_SHIFT 6 /* OUT4_2ADCVOL - [8:6] */ -#define WM8983_OUT4_2ADCVOL_WIDTH 3 /* OUT4_2ADCVOL - [8:6] */ -#define WM8983_OUT4_2LNR 0x0020 /* OUT4_2LNR */ -#define WM8983_OUT4_2LNR_MASK 0x0020 /* OUT4_2LNR */ -#define WM8983_OUT4_2LNR_SHIFT 5 /* OUT4_2LNR */ -#define WM8983_OUT4_2LNR_WIDTH 1 /* OUT4_2LNR */ -#define WM8983_POBCTRL 0x0004 /* POBCTRL */ -#define WM8983_POBCTRL_MASK 0x0004 /* POBCTRL */ -#define WM8983_POBCTRL_SHIFT 2 /* POBCTRL */ -#define WM8983_POBCTRL_WIDTH 1 /* POBCTRL */ -#define WM8983_DELEN 0x0002 /* DELEN */ -#define WM8983_DELEN_MASK 0x0002 /* DELEN */ -#define WM8983_DELEN_SHIFT 1 /* DELEN */ -#define WM8983_DELEN_WIDTH 1 /* DELEN */ -#define WM8983_OUT1DEL 0x0001 /* OUT1DEL */ -#define WM8983_OUT1DEL_MASK 0x0001 /* OUT1DEL */ -#define WM8983_OUT1DEL_SHIFT 0 /* OUT1DEL */ -#define WM8983_OUT1DEL_WIDTH 1 /* OUT1DEL */ - -/* - * R43 (0x2B) - Beep control - */ -#define WM8983_BYPL2RMIX 0x0100 /* BYPL2RMIX */ -#define WM8983_BYPL2RMIX_MASK 0x0100 /* BYPL2RMIX */ -#define WM8983_BYPL2RMIX_SHIFT 8 /* BYPL2RMIX */ -#define WM8983_BYPL2RMIX_WIDTH 1 /* BYPL2RMIX */ -#define WM8983_BYPR2LMIX 0x0080 /* BYPR2LMIX */ -#define WM8983_BYPR2LMIX_MASK 0x0080 /* BYPR2LMIX */ -#define WM8983_BYPR2LMIX_SHIFT 7 /* BYPR2LMIX */ -#define WM8983_BYPR2LMIX_WIDTH 1 /* BYPR2LMIX */ -#define WM8983_MUTERPGA2INV 0x0020 /* MUTERPGA2INV */ -#define WM8983_MUTERPGA2INV_MASK 0x0020 /* MUTERPGA2INV */ -#define WM8983_MUTERPGA2INV_SHIFT 5 /* MUTERPGA2INV */ -#define WM8983_MUTERPGA2INV_WIDTH 1 /* MUTERPGA2INV */ -#define WM8983_INVROUT2 0x0010 /* INVROUT2 */ -#define WM8983_INVROUT2_MASK 0x0010 /* INVROUT2 */ -#define WM8983_INVROUT2_SHIFT 4 /* INVROUT2 */ -#define WM8983_INVROUT2_WIDTH 1 /* INVROUT2 */ -#define WM8983_BEEPVOL_MASK 0x000E /* BEEPVOL - [3:1] */ -#define WM8983_BEEPVOL_SHIFT 1 /* BEEPVOL - [3:1] */ -#define WM8983_BEEPVOL_WIDTH 3 /* BEEPVOL - [3:1] */ -#define WM8983_BEEPEN 0x0001 /* BEEPEN */ -#define WM8983_BEEPEN_MASK 0x0001 /* BEEPEN */ -#define WM8983_BEEPEN_SHIFT 0 /* BEEPEN */ -#define WM8983_BEEPEN_WIDTH 1 /* BEEPEN */ - -/* - * R44 (0x2C) - Input ctrl - */ -#define WM8983_MBVSEL 0x0100 /* MBVSEL */ -#define WM8983_MBVSEL_MASK 0x0100 /* MBVSEL */ -#define WM8983_MBVSEL_SHIFT 8 /* MBVSEL */ -#define WM8983_MBVSEL_WIDTH 1 /* MBVSEL */ -#define WM8983_R2_2INPPGA 0x0040 /* R2_2INPPGA */ -#define WM8983_R2_2INPPGA_MASK 0x0040 /* R2_2INPPGA */ -#define WM8983_R2_2INPPGA_SHIFT 6 /* R2_2INPPGA */ -#define WM8983_R2_2INPPGA_WIDTH 1 /* R2_2INPPGA */ -#define WM8983_RIN2INPPGA 0x0020 /* RIN2INPPGA */ -#define WM8983_RIN2INPPGA_MASK 0x0020 /* RIN2INPPGA */ -#define WM8983_RIN2INPPGA_SHIFT 5 /* RIN2INPPGA */ -#define WM8983_RIN2INPPGA_WIDTH 1 /* RIN2INPPGA */ -#define WM8983_RIP2INPPGA 0x0010 /* RIP2INPPGA */ -#define WM8983_RIP2INPPGA_MASK 0x0010 /* RIP2INPPGA */ -#define WM8983_RIP2INPPGA_SHIFT 4 /* RIP2INPPGA */ -#define WM8983_RIP2INPPGA_WIDTH 1 /* RIP2INPPGA */ -#define WM8983_L2_2INPPGA 0x0004 /* L2_2INPPGA */ -#define WM8983_L2_2INPPGA_MASK 0x0004 /* L2_2INPPGA */ -#define WM8983_L2_2INPPGA_SHIFT 2 /* L2_2INPPGA */ -#define WM8983_L2_2INPPGA_WIDTH 1 /* L2_2INPPGA */ -#define WM8983_LIN2INPPGA 0x0002 /* LIN2INPPGA */ -#define WM8983_LIN2INPPGA_MASK 0x0002 /* LIN2INPPGA */ -#define WM8983_LIN2INPPGA_SHIFT 1 /* LIN2INPPGA */ -#define WM8983_LIN2INPPGA_WIDTH 1 /* LIN2INPPGA */ -#define WM8983_LIP2INPPGA 0x0001 /* LIP2INPPGA */ -#define WM8983_LIP2INPPGA_MASK 0x0001 /* LIP2INPPGA */ -#define WM8983_LIP2INPPGA_SHIFT 0 /* LIP2INPPGA */ -#define WM8983_LIP2INPPGA_WIDTH 1 /* LIP2INPPGA */ - -/* - * R45 (0x2D) - Left INP PGA gain ctrl - */ -#define WM8983_INPGAVU 0x0100 /* INPGAVU */ -#define WM8983_INPGAVU_MASK 0x0100 /* INPGAVU */ -#define WM8983_INPGAVU_SHIFT 8 /* INPGAVU */ -#define WM8983_INPGAVU_WIDTH 1 /* INPGAVU */ -#define WM8983_INPPGAZCL 0x0080 /* INPPGAZCL */ -#define WM8983_INPPGAZCL_MASK 0x0080 /* INPPGAZCL */ -#define WM8983_INPPGAZCL_SHIFT 7 /* INPPGAZCL */ -#define WM8983_INPPGAZCL_WIDTH 1 /* INPPGAZCL */ -#define WM8983_INPPGAMUTEL 0x0040 /* INPPGAMUTEL */ -#define WM8983_INPPGAMUTEL_MASK 0x0040 /* INPPGAMUTEL */ -#define WM8983_INPPGAMUTEL_SHIFT 6 /* INPPGAMUTEL */ -#define WM8983_INPPGAMUTEL_WIDTH 1 /* INPPGAMUTEL */ -#define WM8983_INPPGAVOLL_MASK 0x003F /* INPPGAVOLL - [5:0] */ -#define WM8983_INPPGAVOLL_SHIFT 0 /* INPPGAVOLL - [5:0] */ -#define WM8983_INPPGAVOLL_WIDTH 6 /* INPPGAVOLL - [5:0] */ - -/* - * R46 (0x2E) - Right INP PGA gain ctrl - */ -#define WM8983_INPGAVU 0x0100 /* INPGAVU */ -#define WM8983_INPGAVU_MASK 0x0100 /* INPGAVU */ -#define WM8983_INPGAVU_SHIFT 8 /* INPGAVU */ -#define WM8983_INPGAVU_WIDTH 1 /* INPGAVU */ -#define WM8983_INPPGAZCR 0x0080 /* INPPGAZCR */ -#define WM8983_INPPGAZCR_MASK 0x0080 /* INPPGAZCR */ -#define WM8983_INPPGAZCR_SHIFT 7 /* INPPGAZCR */ -#define WM8983_INPPGAZCR_WIDTH 1 /* INPPGAZCR */ -#define WM8983_INPPGAMUTER 0x0040 /* INPPGAMUTER */ -#define WM8983_INPPGAMUTER_MASK 0x0040 /* INPPGAMUTER */ -#define WM8983_INPPGAMUTER_SHIFT 6 /* INPPGAMUTER */ -#define WM8983_INPPGAMUTER_WIDTH 1 /* INPPGAMUTER */ -#define WM8983_INPPGAVOLR_MASK 0x003F /* INPPGAVOLR - [5:0] */ -#define WM8983_INPPGAVOLR_SHIFT 0 /* INPPGAVOLR - [5:0] */ -#define WM8983_INPPGAVOLR_WIDTH 6 /* INPPGAVOLR - [5:0] */ - -/* - * R47 (0x2F) - Left ADC BOOST ctrl - */ -#define WM8983_PGABOOSTL 0x0100 /* PGABOOSTL */ -#define WM8983_PGABOOSTL_MASK 0x0100 /* PGABOOSTL */ -#define WM8983_PGABOOSTL_SHIFT 8 /* PGABOOSTL */ -#define WM8983_PGABOOSTL_WIDTH 1 /* PGABOOSTL */ -#define WM8983_L2_2BOOSTVOL_MASK 0x0070 /* L2_2BOOSTVOL - [6:4] */ -#define WM8983_L2_2BOOSTVOL_SHIFT 4 /* L2_2BOOSTVOL - [6:4] */ -#define WM8983_L2_2BOOSTVOL_WIDTH 3 /* L2_2BOOSTVOL - [6:4] */ -#define WM8983_AUXL2BOOSTVOL_MASK 0x0007 /* AUXL2BOOSTVOL - [2:0] */ -#define WM8983_AUXL2BOOSTVOL_SHIFT 0 /* AUXL2BOOSTVOL - [2:0] */ -#define WM8983_AUXL2BOOSTVOL_WIDTH 3 /* AUXL2BOOSTVOL - [2:0] */ - -/* - * R48 (0x30) - Right ADC BOOST ctrl - */ -#define WM8983_PGABOOSTR 0x0100 /* PGABOOSTR */ -#define WM8983_PGABOOSTR_MASK 0x0100 /* PGABOOSTR */ -#define WM8983_PGABOOSTR_SHIFT 8 /* PGABOOSTR */ -#define WM8983_PGABOOSTR_WIDTH 1 /* PGABOOSTR */ -#define WM8983_R2_2BOOSTVOL_MASK 0x0070 /* R2_2BOOSTVOL - [6:4] */ -#define WM8983_R2_2BOOSTVOL_SHIFT 4 /* R2_2BOOSTVOL - [6:4] */ -#define WM8983_R2_2BOOSTVOL_WIDTH 3 /* R2_2BOOSTVOL - [6:4] */ -#define WM8983_AUXR2BOOSTVOL_MASK 0x0007 /* AUXR2BOOSTVOL - [2:0] */ -#define WM8983_AUXR2BOOSTVOL_SHIFT 0 /* AUXR2BOOSTVOL - [2:0] */ -#define WM8983_AUXR2BOOSTVOL_WIDTH 3 /* AUXR2BOOSTVOL - [2:0] */ - -/* - * R49 (0x31) - Output ctrl - */ -#define WM8983_DACL2RMIX 0x0040 /* DACL2RMIX */ -#define WM8983_DACL2RMIX_MASK 0x0040 /* DACL2RMIX */ -#define WM8983_DACL2RMIX_SHIFT 6 /* DACL2RMIX */ -#define WM8983_DACL2RMIX_WIDTH 1 /* DACL2RMIX */ -#define WM8983_DACR2LMIX 0x0020 /* DACR2LMIX */ -#define WM8983_DACR2LMIX_MASK 0x0020 /* DACR2LMIX */ -#define WM8983_DACR2LMIX_SHIFT 5 /* DACR2LMIX */ -#define WM8983_DACR2LMIX_WIDTH 1 /* DACR2LMIX */ -#define WM8983_OUT4BOOST 0x0010 /* OUT4BOOST */ -#define WM8983_OUT4BOOST_MASK 0x0010 /* OUT4BOOST */ -#define WM8983_OUT4BOOST_SHIFT 4 /* OUT4BOOST */ -#define WM8983_OUT4BOOST_WIDTH 1 /* OUT4BOOST */ -#define WM8983_OUT3BOOST 0x0008 /* OUT3BOOST */ -#define WM8983_OUT3BOOST_MASK 0x0008 /* OUT3BOOST */ -#define WM8983_OUT3BOOST_SHIFT 3 /* OUT3BOOST */ -#define WM8983_OUT3BOOST_WIDTH 1 /* OUT3BOOST */ -#define WM8983_SPKBOOST 0x0004 /* SPKBOOST */ -#define WM8983_SPKBOOST_MASK 0x0004 /* SPKBOOST */ -#define WM8983_SPKBOOST_SHIFT 2 /* SPKBOOST */ -#define WM8983_SPKBOOST_WIDTH 1 /* SPKBOOST */ -#define WM8983_TSDEN 0x0002 /* TSDEN */ -#define WM8983_TSDEN_MASK 0x0002 /* TSDEN */ -#define WM8983_TSDEN_SHIFT 1 /* TSDEN */ -#define WM8983_TSDEN_WIDTH 1 /* TSDEN */ -#define WM8983_VROI 0x0001 /* VROI */ -#define WM8983_VROI_MASK 0x0001 /* VROI */ -#define WM8983_VROI_SHIFT 0 /* VROI */ -#define WM8983_VROI_WIDTH 1 /* VROI */ - -/* - * R50 (0x32) - Left mixer ctrl - */ -#define WM8983_AUXLMIXVOL_MASK 0x01C0 /* AUXLMIXVOL - [8:6] */ -#define WM8983_AUXLMIXVOL_SHIFT 6 /* AUXLMIXVOL - [8:6] */ -#define WM8983_AUXLMIXVOL_WIDTH 3 /* AUXLMIXVOL - [8:6] */ -#define WM8983_AUXL2LMIX 0x0020 /* AUXL2LMIX */ -#define WM8983_AUXL2LMIX_MASK 0x0020 /* AUXL2LMIX */ -#define WM8983_AUXL2LMIX_SHIFT 5 /* AUXL2LMIX */ -#define WM8983_AUXL2LMIX_WIDTH 1 /* AUXL2LMIX */ -#define WM8983_BYPLMIXVOL_MASK 0x001C /* BYPLMIXVOL - [4:2] */ -#define WM8983_BYPLMIXVOL_SHIFT 2 /* BYPLMIXVOL - [4:2] */ -#define WM8983_BYPLMIXVOL_WIDTH 3 /* BYPLMIXVOL - [4:2] */ -#define WM8983_BYPL2LMIX 0x0002 /* BYPL2LMIX */ -#define WM8983_BYPL2LMIX_MASK 0x0002 /* BYPL2LMIX */ -#define WM8983_BYPL2LMIX_SHIFT 1 /* BYPL2LMIX */ -#define WM8983_BYPL2LMIX_WIDTH 1 /* BYPL2LMIX */ -#define WM8983_DACL2LMIX 0x0001 /* DACL2LMIX */ -#define WM8983_DACL2LMIX_MASK 0x0001 /* DACL2LMIX */ -#define WM8983_DACL2LMIX_SHIFT 0 /* DACL2LMIX */ -#define WM8983_DACL2LMIX_WIDTH 1 /* DACL2LMIX */ - -/* - * R51 (0x33) - Right mixer ctrl - */ -#define WM8983_AUXRMIXVOL_MASK 0x01C0 /* AUXRMIXVOL - [8:6] */ -#define WM8983_AUXRMIXVOL_SHIFT 6 /* AUXRMIXVOL - [8:6] */ -#define WM8983_AUXRMIXVOL_WIDTH 3 /* AUXRMIXVOL - [8:6] */ -#define WM8983_AUXR2RMIX 0x0020 /* AUXR2RMIX */ -#define WM8983_AUXR2RMIX_MASK 0x0020 /* AUXR2RMIX */ -#define WM8983_AUXR2RMIX_SHIFT 5 /* AUXR2RMIX */ -#define WM8983_AUXR2RMIX_WIDTH 1 /* AUXR2RMIX */ -#define WM8983_BYPRMIXVOL_MASK 0x001C /* BYPRMIXVOL - [4:2] */ -#define WM8983_BYPRMIXVOL_SHIFT 2 /* BYPRMIXVOL - [4:2] */ -#define WM8983_BYPRMIXVOL_WIDTH 3 /* BYPRMIXVOL - [4:2] */ -#define WM8983_BYPR2RMIX 0x0002 /* BYPR2RMIX */ -#define WM8983_BYPR2RMIX_MASK 0x0002 /* BYPR2RMIX */ -#define WM8983_BYPR2RMIX_SHIFT 1 /* BYPR2RMIX */ -#define WM8983_BYPR2RMIX_WIDTH 1 /* BYPR2RMIX */ -#define WM8983_DACR2RMIX 0x0001 /* DACR2RMIX */ -#define WM8983_DACR2RMIX_MASK 0x0001 /* DACR2RMIX */ -#define WM8983_DACR2RMIX_SHIFT 0 /* DACR2RMIX */ -#define WM8983_DACR2RMIX_WIDTH 1 /* DACR2RMIX */ - -/* - * R52 (0x34) - LOUT1 (HP) volume ctrl - */ -#define WM8983_OUT1VU 0x0100 /* OUT1VU */ -#define WM8983_OUT1VU_MASK 0x0100 /* OUT1VU */ -#define WM8983_OUT1VU_SHIFT 8 /* OUT1VU */ -#define WM8983_OUT1VU_WIDTH 1 /* OUT1VU */ -#define WM8983_LOUT1ZC 0x0080 /* LOUT1ZC */ -#define WM8983_LOUT1ZC_MASK 0x0080 /* LOUT1ZC */ -#define WM8983_LOUT1ZC_SHIFT 7 /* LOUT1ZC */ -#define WM8983_LOUT1ZC_WIDTH 1 /* LOUT1ZC */ -#define WM8983_LOUT1MUTE 0x0040 /* LOUT1MUTE */ -#define WM8983_LOUT1MUTE_MASK 0x0040 /* LOUT1MUTE */ -#define WM8983_LOUT1MUTE_SHIFT 6 /* LOUT1MUTE */ -#define WM8983_LOUT1MUTE_WIDTH 1 /* LOUT1MUTE */ -#define WM8983_LOUT1VOL_MASK 0x003F /* LOUT1VOL - [5:0] */ -#define WM8983_LOUT1VOL_SHIFT 0 /* LOUT1VOL - [5:0] */ -#define WM8983_LOUT1VOL_WIDTH 6 /* LOUT1VOL - [5:0] */ - -/* - * R53 (0x35) - ROUT1 (HP) volume ctrl - */ -#define WM8983_OUT1VU 0x0100 /* OUT1VU */ -#define WM8983_OUT1VU_MASK 0x0100 /* OUT1VU */ -#define WM8983_OUT1VU_SHIFT 8 /* OUT1VU */ -#define WM8983_OUT1VU_WIDTH 1 /* OUT1VU */ -#define WM8983_ROUT1ZC 0x0080 /* ROUT1ZC */ -#define WM8983_ROUT1ZC_MASK 0x0080 /* ROUT1ZC */ -#define WM8983_ROUT1ZC_SHIFT 7 /* ROUT1ZC */ -#define WM8983_ROUT1ZC_WIDTH 1 /* ROUT1ZC */ -#define WM8983_ROUT1MUTE 0x0040 /* ROUT1MUTE */ -#define WM8983_ROUT1MUTE_MASK 0x0040 /* ROUT1MUTE */ -#define WM8983_ROUT1MUTE_SHIFT 6 /* ROUT1MUTE */ -#define WM8983_ROUT1MUTE_WIDTH 1 /* ROUT1MUTE */ -#define WM8983_ROUT1VOL_MASK 0x003F /* ROUT1VOL - [5:0] */ -#define WM8983_ROUT1VOL_SHIFT 0 /* ROUT1VOL - [5:0] */ -#define WM8983_ROUT1VOL_WIDTH 6 /* ROUT1VOL - [5:0] */ - -/* - * R54 (0x36) - LOUT2 (SPK) volume ctrl - */ -#define WM8983_OUT2VU 0x0100 /* OUT2VU */ -#define WM8983_OUT2VU_MASK 0x0100 /* OUT2VU */ -#define WM8983_OUT2VU_SHIFT 8 /* OUT2VU */ -#define WM8983_OUT2VU_WIDTH 1 /* OUT2VU */ -#define WM8983_LOUT2ZC 0x0080 /* LOUT2ZC */ -#define WM8983_LOUT2ZC_MASK 0x0080 /* LOUT2ZC */ -#define WM8983_LOUT2ZC_SHIFT 7 /* LOUT2ZC */ -#define WM8983_LOUT2ZC_WIDTH 1 /* LOUT2ZC */ -#define WM8983_LOUT2MUTE 0x0040 /* LOUT2MUTE */ -#define WM8983_LOUT2MUTE_MASK 0x0040 /* LOUT2MUTE */ -#define WM8983_LOUT2MUTE_SHIFT 6 /* LOUT2MUTE */ -#define WM8983_LOUT2MUTE_WIDTH 1 /* LOUT2MUTE */ -#define WM8983_LOUT2VOL_MASK 0x003F /* LOUT2VOL - [5:0] */ -#define WM8983_LOUT2VOL_SHIFT 0 /* LOUT2VOL - [5:0] */ -#define WM8983_LOUT2VOL_WIDTH 6 /* LOUT2VOL - [5:0] */ - -/* - * R55 (0x37) - ROUT2 (SPK) volume ctrl - */ -#define WM8983_OUT2VU 0x0100 /* OUT2VU */ -#define WM8983_OUT2VU_MASK 0x0100 /* OUT2VU */ -#define WM8983_OUT2VU_SHIFT 8 /* OUT2VU */ -#define WM8983_OUT2VU_WIDTH 1 /* OUT2VU */ -#define WM8983_ROUT2ZC 0x0080 /* ROUT2ZC */ -#define WM8983_ROUT2ZC_MASK 0x0080 /* ROUT2ZC */ -#define WM8983_ROUT2ZC_SHIFT 7 /* ROUT2ZC */ -#define WM8983_ROUT2ZC_WIDTH 1 /* ROUT2ZC */ -#define WM8983_ROUT2MUTE 0x0040 /* ROUT2MUTE */ -#define WM8983_ROUT2MUTE_MASK 0x0040 /* ROUT2MUTE */ -#define WM8983_ROUT2MUTE_SHIFT 6 /* ROUT2MUTE */ -#define WM8983_ROUT2MUTE_WIDTH 1 /* ROUT2MUTE */ -#define WM8983_ROUT2VOL_MASK 0x003F /* ROUT2VOL - [5:0] */ -#define WM8983_ROUT2VOL_SHIFT 0 /* ROUT2VOL - [5:0] */ -#define WM8983_ROUT2VOL_WIDTH 6 /* ROUT2VOL - [5:0] */ - -/* - * R56 (0x38) - OUT3 mixer ctrl - */ -#define WM8983_OUT3MUTE 0x0040 /* OUT3MUTE */ -#define WM8983_OUT3MUTE_MASK 0x0040 /* OUT3MUTE */ -#define WM8983_OUT3MUTE_SHIFT 6 /* OUT3MUTE */ -#define WM8983_OUT3MUTE_WIDTH 1 /* OUT3MUTE */ -#define WM8983_OUT4_2OUT3 0x0008 /* OUT4_2OUT3 */ -#define WM8983_OUT4_2OUT3_MASK 0x0008 /* OUT4_2OUT3 */ -#define WM8983_OUT4_2OUT3_SHIFT 3 /* OUT4_2OUT3 */ -#define WM8983_OUT4_2OUT3_WIDTH 1 /* OUT4_2OUT3 */ -#define WM8983_BYPL2OUT3 0x0004 /* BYPL2OUT3 */ -#define WM8983_BYPL2OUT3_MASK 0x0004 /* BYPL2OUT3 */ -#define WM8983_BYPL2OUT3_SHIFT 2 /* BYPL2OUT3 */ -#define WM8983_BYPL2OUT3_WIDTH 1 /* BYPL2OUT3 */ -#define WM8983_LMIX2OUT3 0x0002 /* LMIX2OUT3 */ -#define WM8983_LMIX2OUT3_MASK 0x0002 /* LMIX2OUT3 */ -#define WM8983_LMIX2OUT3_SHIFT 1 /* LMIX2OUT3 */ -#define WM8983_LMIX2OUT3_WIDTH 1 /* LMIX2OUT3 */ -#define WM8983_LDAC2OUT3 0x0001 /* LDAC2OUT3 */ -#define WM8983_LDAC2OUT3_MASK 0x0001 /* LDAC2OUT3 */ -#define WM8983_LDAC2OUT3_SHIFT 0 /* LDAC2OUT3 */ -#define WM8983_LDAC2OUT3_WIDTH 1 /* LDAC2OUT3 */ - -/* - * R57 (0x39) - OUT4 (MONO) mix ctrl - */ -#define WM8983_OUT3_2OUT4 0x0080 /* OUT3_2OUT4 */ -#define WM8983_OUT3_2OUT4_MASK 0x0080 /* OUT3_2OUT4 */ -#define WM8983_OUT3_2OUT4_SHIFT 7 /* OUT3_2OUT4 */ -#define WM8983_OUT3_2OUT4_WIDTH 1 /* OUT3_2OUT4 */ -#define WM8983_OUT4MUTE 0x0040 /* OUT4MUTE */ -#define WM8983_OUT4MUTE_MASK 0x0040 /* OUT4MUTE */ -#define WM8983_OUT4MUTE_SHIFT 6 /* OUT4MUTE */ -#define WM8983_OUT4MUTE_WIDTH 1 /* OUT4MUTE */ -#define WM8983_OUT4ATTN 0x0020 /* OUT4ATTN */ -#define WM8983_OUT4ATTN_MASK 0x0020 /* OUT4ATTN */ -#define WM8983_OUT4ATTN_SHIFT 5 /* OUT4ATTN */ -#define WM8983_OUT4ATTN_WIDTH 1 /* OUT4ATTN */ -#define WM8983_LMIX2OUT4 0x0010 /* LMIX2OUT4 */ -#define WM8983_LMIX2OUT4_MASK 0x0010 /* LMIX2OUT4 */ -#define WM8983_LMIX2OUT4_SHIFT 4 /* LMIX2OUT4 */ -#define WM8983_LMIX2OUT4_WIDTH 1 /* LMIX2OUT4 */ -#define WM8983_LDAC2OUT4 0x0008 /* LDAC2OUT4 */ -#define WM8983_LDAC2OUT4_MASK 0x0008 /* LDAC2OUT4 */ -#define WM8983_LDAC2OUT4_SHIFT 3 /* LDAC2OUT4 */ -#define WM8983_LDAC2OUT4_WIDTH 1 /* LDAC2OUT4 */ -#define WM8983_BYPR2OUT4 0x0004 /* BYPR2OUT4 */ -#define WM8983_BYPR2OUT4_MASK 0x0004 /* BYPR2OUT4 */ -#define WM8983_BYPR2OUT4_SHIFT 2 /* BYPR2OUT4 */ -#define WM8983_BYPR2OUT4_WIDTH 1 /* BYPR2OUT4 */ -#define WM8983_RMIX2OUT4 0x0002 /* RMIX2OUT4 */ -#define WM8983_RMIX2OUT4_MASK 0x0002 /* RMIX2OUT4 */ -#define WM8983_RMIX2OUT4_SHIFT 1 /* RMIX2OUT4 */ -#define WM8983_RMIX2OUT4_WIDTH 1 /* RMIX2OUT4 */ -#define WM8983_RDAC2OUT4 0x0001 /* RDAC2OUT4 */ -#define WM8983_RDAC2OUT4_MASK 0x0001 /* RDAC2OUT4 */ -#define WM8983_RDAC2OUT4_SHIFT 0 /* RDAC2OUT4 */ -#define WM8983_RDAC2OUT4_WIDTH 1 /* RDAC2OUT4 */ - -/* - * R61 (0x3D) - BIAS CTRL - */ -#define WM8983_BIASCUT 0x0100 /* BIASCUT */ -#define WM8983_BIASCUT_MASK 0x0100 /* BIASCUT */ -#define WM8983_BIASCUT_SHIFT 8 /* BIASCUT */ -#define WM8983_BIASCUT_WIDTH 1 /* BIASCUT */ -#define WM8983_HALFIPBIAS 0x0080 /* HALFIPBIAS */ -#define WM8983_HALFIPBIAS_MASK 0x0080 /* HALFIPBIAS */ -#define WM8983_HALFIPBIAS_SHIFT 7 /* HALFIPBIAS */ -#define WM8983_HALFIPBIAS_WIDTH 1 /* HALFIPBIAS */ -#define WM8983_VBBIASTST_MASK 0x0060 /* VBBIASTST - [6:5] */ -#define WM8983_VBBIASTST_SHIFT 5 /* VBBIASTST - [6:5] */ -#define WM8983_VBBIASTST_WIDTH 2 /* VBBIASTST - [6:5] */ -#define WM8983_BUFBIAS_MASK 0x0018 /* BUFBIAS - [4:3] */ -#define WM8983_BUFBIAS_SHIFT 3 /* BUFBIAS - [4:3] */ -#define WM8983_BUFBIAS_WIDTH 2 /* BUFBIAS - [4:3] */ -#define WM8983_ADCBIAS_MASK 0x0006 /* ADCBIAS - [2:1] */ -#define WM8983_ADCBIAS_SHIFT 1 /* ADCBIAS - [2:1] */ -#define WM8983_ADCBIAS_WIDTH 2 /* ADCBIAS - [2:1] */ -#define WM8983_HALFOPBIAS 0x0001 /* HALFOPBIAS */ -#define WM8983_HALFOPBIAS_MASK 0x0001 /* HALFOPBIAS */ -#define WM8983_HALFOPBIAS_SHIFT 0 /* HALFOPBIAS */ -#define WM8983_HALFOPBIAS_WIDTH 1 /* HALFOPBIAS */ - -enum clk_src { - WM8983_CLKSRC_MCLK, - WM8983_CLKSRC_PLL -}; - -#endif /* _WM8983_H */ diff --git a/trunk/sound/soc/codecs/wm8991.c b/trunk/sound/soc/codecs/wm8991.c index 6af23d06870f..3c2ee1bb73cd 100644 --- a/trunk/sound/soc/codecs/wm8991.c +++ b/trunk/sound/soc/codecs/wm8991.c @@ -13,6 +13,7 @@ #include #include +#include #include #include #include diff --git a/trunk/sound/soc/codecs/wm8993.c b/trunk/sound/soc/codecs/wm8993.c index 6e85b8869af7..9e5ff789b805 100644 --- a/trunk/sound/soc/codecs/wm8993.c +++ b/trunk/sound/soc/codecs/wm8993.c @@ -876,7 +876,7 @@ SND_SOC_DAPM_MIXER("SPKL", WM8993_POWER_MANAGEMENT_3, 8, 0, left_speaker_mixer, ARRAY_SIZE(left_speaker_mixer)), SND_SOC_DAPM_MIXER("SPKR", WM8993_POWER_MANAGEMENT_3, 9, 0, right_speaker_mixer, ARRAY_SIZE(right_speaker_mixer)), -SND_SOC_DAPM_PGA("Direct Voice", SND_SOC_NOPM, 0, 0, NULL, 0), + }; static const struct snd_soc_dapm_route routes[] = { @@ -1434,7 +1434,6 @@ static int wm8993_probe(struct snd_soc_codec *codec) wm8993->hubs_data.hp_startup_mode = 1; wm8993->hubs_data.dcs_codes = -2; - wm8993->hubs_data.series_startup = 1; ret = snd_soc_codec_set_cache_io(codec, 8, 16, SND_SOC_I2C); if (ret != 0) { diff --git a/trunk/sound/soc/codecs/wm8994.c b/trunk/sound/soc/codecs/wm8994.c index 09e680ae88b2..970a95c5360b 100644 --- a/trunk/sound/soc/codecs/wm8994.c +++ b/trunk/sound/soc/codecs/wm8994.c @@ -195,6 +195,10 @@ static int configure_aif_clock(struct snd_soc_codec *codec, int aif) aif + 1, rate); } + if (rate && rate < 3000000) + dev_warn(codec->dev, "AIF%dCLK is %dHz, should be >=3MHz for optimal performance\n", + aif + 1, rate); + wm8994->aifclk[aif] = rate; snd_soc_update_bits(codec, WM8994_AIF1_CLOCKING_1 + offset, @@ -1142,33 +1146,13 @@ SND_SOC_DAPM_PGA_E("Late DAC2L Enable PGA", SND_SOC_NOPM, 0, 0, NULL, 0, late_enable_ev, SND_SOC_DAPM_PRE_PMU), SND_SOC_DAPM_PGA_E("Late DAC2R Enable PGA", SND_SOC_NOPM, 0, 0, NULL, 0, late_enable_ev, SND_SOC_DAPM_PRE_PMU), -SND_SOC_DAPM_PGA_E("Direct Voice", SND_SOC_NOPM, 0, 0, NULL, 0, - late_enable_ev, SND_SOC_DAPM_PRE_PMU), - -SND_SOC_DAPM_MIXER_E("SPKL", WM8994_POWER_MANAGEMENT_3, 8, 0, - left_speaker_mixer, ARRAY_SIZE(left_speaker_mixer), - late_enable_ev, SND_SOC_DAPM_PRE_PMU), -SND_SOC_DAPM_MIXER_E("SPKR", WM8994_POWER_MANAGEMENT_3, 9, 0, - right_speaker_mixer, ARRAY_SIZE(right_speaker_mixer), - late_enable_ev, SND_SOC_DAPM_PRE_PMU), -SND_SOC_DAPM_MUX_E("Left Headphone Mux", SND_SOC_NOPM, 0, 0, &hpl_mux, - late_enable_ev, SND_SOC_DAPM_PRE_PMU), -SND_SOC_DAPM_MUX_E("Right Headphone Mux", SND_SOC_NOPM, 0, 0, &hpr_mux, - late_enable_ev, SND_SOC_DAPM_PRE_PMU), SND_SOC_DAPM_POST("Late Disable PGA", late_disable_ev) }; static const struct snd_soc_dapm_widget wm8994_lateclk_widgets[] = { SND_SOC_DAPM_SUPPLY("AIF1CLK", WM8994_AIF1_CLOCKING_1, 0, 0, NULL, 0), -SND_SOC_DAPM_SUPPLY("AIF2CLK", WM8994_AIF2_CLOCKING_1, 0, 0, NULL, 0), -SND_SOC_DAPM_PGA("Direct Voice", SND_SOC_NOPM, 0, 0, NULL, 0), -SND_SOC_DAPM_MIXER("SPKL", WM8994_POWER_MANAGEMENT_3, 8, 0, - left_speaker_mixer, ARRAY_SIZE(left_speaker_mixer)), -SND_SOC_DAPM_MIXER("SPKR", WM8994_POWER_MANAGEMENT_3, 9, 0, - right_speaker_mixer, ARRAY_SIZE(right_speaker_mixer)), -SND_SOC_DAPM_MUX("Left Headphone Mux", SND_SOC_NOPM, 0, 0, &hpl_mux), -SND_SOC_DAPM_MUX("Right Headphone Mux", SND_SOC_NOPM, 0, 0, &hpr_mux), +SND_SOC_DAPM_SUPPLY("AIF2CLK", WM8994_AIF2_CLOCKING_1, 0, 0, NULL, 0) }; static const struct snd_soc_dapm_widget wm8994_dac_revd_widgets[] = { @@ -1206,6 +1190,7 @@ SND_SOC_DAPM_INPUT("DMIC1DAT"), SND_SOC_DAPM_INPUT("DMIC2DAT"), SND_SOC_DAPM_INPUT("Clock"), +SND_SOC_DAPM_MICBIAS("MICBIAS", WM8994_MICBIAS, 2, 0), SND_SOC_DAPM_SUPPLY_S("MICBIAS Supply", 1, SND_SOC_NOPM, 0, 0, micbias_ev, SND_SOC_DAPM_PRE_PMU), @@ -1298,6 +1283,14 @@ SND_SOC_DAPM_ADC("DMIC1R", NULL, WM8994_POWER_MANAGEMENT_4, 2, 0), SND_SOC_DAPM_ADC("ADCL", NULL, SND_SOC_NOPM, 1, 0), SND_SOC_DAPM_ADC("ADCR", NULL, SND_SOC_NOPM, 0, 0), +SND_SOC_DAPM_MUX("Left Headphone Mux", SND_SOC_NOPM, 0, 0, &hpl_mux), +SND_SOC_DAPM_MUX("Right Headphone Mux", SND_SOC_NOPM, 0, 0, &hpr_mux), + +SND_SOC_DAPM_MIXER("SPKL", WM8994_POWER_MANAGEMENT_3, 8, 0, + left_speaker_mixer, ARRAY_SIZE(left_speaker_mixer)), +SND_SOC_DAPM_MIXER("SPKR", WM8994_POWER_MANAGEMENT_3, 9, 0, + right_speaker_mixer, ARRAY_SIZE(right_speaker_mixer)), + SND_SOC_DAPM_POST("Debug log", post_ev), }; @@ -1516,10 +1509,8 @@ static const struct snd_soc_dapm_route wm8994_revd_intercon[] = { { "AIF2DACDAT", NULL, "AIF1DACDAT" }, { "AIF1ADCDAT", NULL, "AIF2ADCDAT" }, { "AIF2ADCDAT", NULL, "AIF1ADCDAT" }, - { "MICBIAS1", NULL, "CLK_SYS" }, - { "MICBIAS1", NULL, "MICBIAS Supply" }, - { "MICBIAS2", NULL, "CLK_SYS" }, - { "MICBIAS2", NULL, "MICBIAS Supply" }, + { "MICBIAS", NULL, "CLK_SYS" }, + { "MICBIAS", NULL, "MICBIAS Supply" }, }; static const struct snd_soc_dapm_route wm8994_intercon[] = { @@ -1632,7 +1623,6 @@ static int _wm8994_set_fll(struct snd_soc_codec *codec, int id, int src, int reg_offset, ret; struct fll_div fll; u16 reg, aif1, aif2; - unsigned long timeout; aif1 = snd_soc_read(codec, WM8994_AIF1_CLOCKING_1) & WM8994_AIF1CLK_ENA; @@ -1714,9 +1704,6 @@ static int _wm8994_set_fll(struct snd_soc_codec *codec, int id, int src, (fll.clk_ref_div << WM8994_FLL1_REFCLK_DIV_SHIFT) | (src - 1)); - /* Clear any pending completion from a previous failure */ - try_wait_for_completion(&wm8994->fll_locked[id]); - /* Enable (with fractional mode if required) */ if (freq_out) { if (fll.k) @@ -1726,16 +1713,6 @@ static int _wm8994_set_fll(struct snd_soc_codec *codec, int id, int src, snd_soc_update_bits(codec, WM8994_FLL1_CONTROL_1 + reg_offset, WM8994_FLL1_ENA | WM8994_FLL1_FRAC, reg); - - if (wm8994->fll_locked_irq) { - timeout = wait_for_completion_timeout(&wm8994->fll_locked[id], - msecs_to_jiffies(10)); - if (timeout == 0) - dev_warn(codec->dev, - "Timed out waiting for FLL lock\n"); - } else { - msleep(5); - } } wm8994->fll[id].in = freq_in; @@ -1753,14 +1730,6 @@ static int _wm8994_set_fll(struct snd_soc_codec *codec, int id, int src, return 0; } -static irqreturn_t wm8994_fll_locked_irq(int irq, void *data) -{ - struct completion *completion = data; - - complete(completion); - - return IRQ_HANDLED; -} static int opclk_divs[] = { 10, 20, 30, 40, 55, 60, 80, 120, 160 }; @@ -2300,33 +2269,6 @@ static int wm8994_aif3_hw_params(struct snd_pcm_substream *substream, return snd_soc_update_bits(codec, aif1_reg, WM8994_AIF1_WL_MASK, aif1); } -static void wm8994_aif_shutdown(struct snd_pcm_substream *substream, - struct snd_soc_dai *dai) -{ - struct snd_soc_codec *codec = dai->codec; - int rate_reg = 0; - - switch (dai->id) { - case 1: - rate_reg = WM8994_AIF1_RATE; - break; - case 2: - rate_reg = WM8994_AIF1_RATE; - break; - default: - break; - } - - /* If the DAI is idle then configure the divider tree for the - * lowest output rate to save a little power if the clock is - * still active (eg, because it is system clock). - */ - if (rate_reg && !dai->playback_active && !dai->capture_active) - snd_soc_update_bits(codec, rate_reg, - WM8994_AIF1_SR_MASK | - WM8994_AIF1CLK_RATE_MASK, 0x9); -} - static int wm8994_aif_mute(struct snd_soc_dai *codec_dai, int mute) { struct snd_soc_codec *codec = codec_dai->codec; @@ -2393,7 +2335,6 @@ static struct snd_soc_dai_ops wm8994_aif1_dai_ops = { .set_sysclk = wm8994_set_dai_sysclk, .set_fmt = wm8994_set_dai_fmt, .hw_params = wm8994_hw_params, - .shutdown = wm8994_aif_shutdown, .digital_mute = wm8994_aif_mute, .set_pll = wm8994_set_fll, .set_tristate = wm8994_set_tristate, @@ -2403,7 +2344,6 @@ static struct snd_soc_dai_ops wm8994_aif2_dai_ops = { .set_sysclk = wm8994_set_dai_sysclk, .set_fmt = wm8994_set_dai_fmt, .hw_params = wm8994_hw_params, - .shutdown = wm8994_aif_shutdown, .digital_mute = wm8994_aif_mute, .set_pll = wm8994_set_fll, .set_tristate = wm8994_set_tristate, @@ -2821,7 +2761,7 @@ static void wm8958_default_micdet(u16 status, void *data) report = SND_JACK_MICROPHONE; /* Everything else is buttons; just assign slots */ - if (status & 0x1c) + if (status & 0x1c0) report |= SND_JACK_BTN_0; done: @@ -2907,15 +2847,6 @@ static irqreturn_t wm8958_mic_irq(int irq, void *data) return IRQ_HANDLED; } -static irqreturn_t wm8994_fifo_error(int irq, void *data) -{ - struct snd_soc_codec *codec = data; - - dev_err(codec->dev, "FIFO error\n"); - - return IRQ_HANDLED; -} - static int wm8994_codec_probe(struct snd_soc_codec *codec) { struct wm8994 *control; @@ -2934,9 +2865,6 @@ static int wm8994_codec_probe(struct snd_soc_codec *codec) wm8994->pdata = dev_get_platdata(codec->dev->parent); wm8994->codec = codec; - for (i = 0; i < ARRAY_SIZE(wm8994->fll_locked); i++) - init_completion(&wm8994->fll_locked[i]); - if (wm8994->pdata && wm8994->pdata->micdet_irq) wm8994->micdet_irq = wm8994->pdata->micdet_irq; else if (wm8994->pdata && wm8994->pdata->irq_base) @@ -2975,7 +2903,6 @@ static int wm8994_codec_probe(struct snd_soc_codec *codec) wm8994->hubs.dcs_codes = -5; wm8994->hubs.hp_startup_mode = 1; wm8994->hubs.dcs_readback_mode = 1; - wm8994->hubs.series_startup = 1; break; default: wm8994->hubs.dcs_readback_mode = 1; @@ -2990,15 +2917,6 @@ static int wm8994_codec_probe(struct snd_soc_codec *codec) break; } - wm8994_request_irq(codec->control_data, WM8994_IRQ_FIFOS_ERR, - wm8994_fifo_error, "FIFO error", codec); - - ret = wm8994_request_irq(codec->control_data, WM8994_IRQ_DCS_DONE, - wm_hubs_dcs_done, "DC servo done", - &wm8994->hubs); - if (ret == 0) - wm8994->hubs.dcs_done_irq = true; - switch (control->type) { case WM8994: if (wm8994->micdet_irq) { @@ -3055,16 +2973,6 @@ static int wm8994_codec_probe(struct snd_soc_codec *codec) } } - wm8994->fll_locked_irq = true; - for (i = 0; i < ARRAY_SIZE(wm8994->fll_locked); i++) { - ret = wm8994_request_irq(codec->control_data, - WM8994_IRQ_FLL1_LOCK + i, - wm8994_fll_locked_irq, "FLL lock", - &wm8994->fll_locked[i]); - if (ret != 0) - wm8994->fll_locked_irq = false; - } - /* Remember if AIFnLRCLK is configured as a GPIO. This should be * configured on init - if a system wants to do this dynamically * at runtime we can deal with that then. @@ -3140,18 +3048,10 @@ static int wm8994_codec_probe(struct snd_soc_codec *codec) 1 << WM8994_AIF2DAC_3D_GAIN_SHIFT, 1 << WM8994_AIF2DAC_3D_GAIN_SHIFT); - /* Unconditionally enable AIF1 ADC TDM mode on chips which can - * use this; it only affects behaviour on idle TDM clock - * cycles. */ - switch (control->type) { - case WM8994: - case WM8958: - snd_soc_update_bits(codec, WM8994_AIF1_CONTROL_1, - WM8994_AIF1ADC_TDM, WM8994_AIF1ADC_TDM); - break; - default: - break; - } + /* Unconditionally enable AIF1 ADC TDM mode; it only affects + * behaviour on idle TDM clock cycles. */ + snd_soc_update_bits(codec, WM8994_AIF1_CONTROL_1, + WM8994_AIF1ADC_TDM, WM8994_AIF1ADC_TDM); wm8994_update_class_w(codec); @@ -3250,12 +3150,6 @@ static int wm8994_codec_probe(struct snd_soc_codec *codec) wm8994_free_irq(codec->control_data, WM8994_IRQ_MIC1_SHRT, wm8994); if (wm8994->micdet_irq) free_irq(wm8994->micdet_irq, wm8994); - for (i = 0; i < ARRAY_SIZE(wm8994->fll_locked); i++) - wm8994_free_irq(codec->control_data, WM8994_IRQ_FLL1_LOCK + i, - &wm8994->fll_locked[i]); - wm8994_free_irq(codec->control_data, WM8994_IRQ_DCS_DONE, - &wm8994->hubs); - wm8994_free_irq(codec->control_data, WM8994_IRQ_FIFOS_ERR, codec); err: kfree(wm8994); return ret; @@ -3265,20 +3159,11 @@ static int wm8994_codec_remove(struct snd_soc_codec *codec) { struct wm8994_priv *wm8994 = snd_soc_codec_get_drvdata(codec); struct wm8994 *control = codec->control_data; - int i; wm8994_set_bias_level(codec, SND_SOC_BIAS_OFF); pm_runtime_disable(codec->dev); - for (i = 0; i < ARRAY_SIZE(wm8994->fll_locked); i++) - wm8994_free_irq(codec->control_data, WM8994_IRQ_FLL1_LOCK + i, - &wm8994->fll_locked[i]); - - wm8994_free_irq(codec->control_data, WM8994_IRQ_DCS_DONE, - &wm8994->hubs); - wm8994_free_irq(codec->control_data, WM8994_IRQ_FIFOS_ERR, codec); - switch (control->type) { case WM8994: if (wm8994->micdet_irq) diff --git a/trunk/sound/soc/codecs/wm8994.h b/trunk/sound/soc/codecs/wm8994.h index 1ab2266039f7..0a1db04b73bd 100644 --- a/trunk/sound/soc/codecs/wm8994.h +++ b/trunk/sound/soc/codecs/wm8994.h @@ -11,7 +11,6 @@ #include #include -#include #include "wm_hubs.h" @@ -80,8 +79,6 @@ struct wm8994_priv { int mclk[2]; int aifclk[2]; struct wm8994_fll_config fll[2], fll_suspend[2]; - struct completion fll_locked[2]; - bool fll_locked_irq; int dac_rates[2]; int lrclk_shared[2]; diff --git a/trunk/sound/soc/codecs/wm9081.c b/trunk/sound/soc/codecs/wm9081.c index a4691321f9b3..91c6b39de50c 100644 --- a/trunk/sound/soc/codecs/wm9081.c +++ b/trunk/sound/soc/codecs/wm9081.c @@ -727,7 +727,7 @@ SND_SOC_DAPM_MIXER_NAMED_CTL("Mixer", SND_SOC_NOPM, 0, 0, SND_SOC_DAPM_PGA("LINEOUT PGA", WM9081_POWER_MANAGEMENT, 4, 0, NULL, 0), SND_SOC_DAPM_PGA("Speaker PGA", WM9081_POWER_MANAGEMENT, 2, 0, NULL, 0), -SND_SOC_DAPM_OUT_DRV("Speaker", WM9081_POWER_MANAGEMENT, 1, 0, NULL, 0), +SND_SOC_DAPM_PGA("Speaker", WM9081_POWER_MANAGEMENT, 1, 0, NULL, 0), SND_SOC_DAPM_OUTPUT("LINEOUT"), SND_SOC_DAPM_OUTPUT("SPKN"), diff --git a/trunk/sound/soc/codecs/wm_hubs.c b/trunk/sound/soc/codecs/wm_hubs.c index 4cc2d567f22f..9e370d14ad88 100644 --- a/trunk/sound/soc/codecs/wm_hubs.c +++ b/trunk/sound/soc/codecs/wm_hubs.c @@ -63,10 +63,8 @@ static const struct soc_enum speaker_mode = static void wait_for_dc_servo(struct snd_soc_codec *codec, unsigned int op) { - struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); unsigned int reg; int count = 0; - int timeout; unsigned int val; val = op | WM8993_DCS_ENA_CHAN_0 | WM8993_DCS_ENA_CHAN_1; @@ -76,39 +74,18 @@ static void wait_for_dc_servo(struct snd_soc_codec *codec, unsigned int op) dev_dbg(codec->dev, "Waiting for DC servo...\n"); - if (hubs->dcs_done_irq) - timeout = 4; - else - timeout = 400; - do { count++; - - if (hubs->dcs_done_irq) - wait_for_completion_timeout(&hubs->dcs_done, - msecs_to_jiffies(250)); - else - msleep(1); - + msleep(1); reg = snd_soc_read(codec, WM8993_DC_SERVO_0); dev_dbg(codec->dev, "DC servo: %x\n", reg); - } while (reg & op && count < timeout); + } while (reg & op && count < 400); if (reg & op) dev_err(codec->dev, "Timed out waiting for DC Servo %x\n", op); } -irqreturn_t wm_hubs_dcs_done(int irq, void *data) -{ - struct wm_hubs_data *hubs = data; - - complete(&hubs->dcs_done); - - return IRQ_HANDLED; -} -EXPORT_SYMBOL_GPL(wm_hubs_dcs_done); - /* * Startup calibration of the DC servo */ @@ -130,7 +107,8 @@ static void calibrate_dc_servo(struct snd_soc_codec *codec) return; } - if (hubs->series_startup) { + /* Devices not using a DCS code correction have startup mode */ + if (hubs->dcs_codes) { /* Set for 32 series updates */ snd_soc_update_bits(codec, WM8993_DC_SERVO_1, WM8993_DCS_SERIES_NO_01_MASK, @@ -156,9 +134,9 @@ static void calibrate_dc_servo(struct snd_soc_codec *codec) break; case 1: reg = snd_soc_read(codec, WM8993_DC_SERVO_3); - reg_r = (reg & WM8993_DCS_DAC_WR_VAL_1_MASK) + reg_l = (reg & WM8993_DCS_DAC_WR_VAL_1_MASK) >> WM8993_DCS_DAC_WR_VAL_1_SHIFT; - reg_l = reg & WM8993_DCS_DAC_WR_VAL_0_MASK; + reg_r = reg & WM8993_DCS_DAC_WR_VAL_0_MASK; break; default: WARN(1, "Unknown DCS readback method\n"); @@ -172,13 +150,13 @@ static void calibrate_dc_servo(struct snd_soc_codec *codec) dev_dbg(codec->dev, "Applying %d code DC servo correction\n", hubs->dcs_codes); - /* HPOUT1R */ - offset = reg_r; + /* HPOUT1L */ + offset = reg_l; offset += hubs->dcs_codes; dcs_cfg = (u8)offset << WM8993_DCS_DAC_WR_VAL_1_SHIFT; - /* HPOUT1L */ - offset = reg_l; + /* HPOUT1R */ + offset = reg_r; offset += hubs->dcs_codes; dcs_cfg |= (u8)offset; @@ -190,8 +168,8 @@ static void calibrate_dc_servo(struct snd_soc_codec *codec) WM8993_DCS_TRIG_DAC_WR_0 | WM8993_DCS_TRIG_DAC_WR_1); } else { - dcs_cfg = reg_r << WM8993_DCS_DAC_WR_VAL_1_SHIFT; - dcs_cfg |= reg_l; + dcs_cfg = reg_l << WM8993_DCS_DAC_WR_VAL_1_SHIFT; + dcs_cfg |= reg_r; } /* Save the callibrated offset if we're in class W mode and @@ -217,7 +195,7 @@ static int wm8993_put_dc_servo(struct snd_kcontrol *kcontrol, /* If we're applying an offset correction then updating the * callibration would be likely to introduce further offsets. */ - if (hubs->dcs_codes || hubs->no_series_update) + if (hubs->dcs_codes) return ret; /* Only need to do this if the outputs are active */ @@ -621,6 +599,9 @@ SND_SOC_DAPM_MIXER("IN2L PGA", WM8993_POWER_MANAGEMENT_2, 7, 0, SND_SOC_DAPM_MIXER("IN2R PGA", WM8993_POWER_MANAGEMENT_2, 5, 0, in2r_pga, ARRAY_SIZE(in2r_pga)), +/* Dummy widgets to represent differential paths */ +SND_SOC_DAPM_PGA("Direct Voice", SND_SOC_NOPM, 0, 0, NULL, 0), + SND_SOC_DAPM_MIXER("MIXINL", WM8993_POWER_MANAGEMENT_2, 9, 0, mixinl, ARRAY_SIZE(mixinl)), SND_SOC_DAPM_MIXER("MIXINR", WM8993_POWER_MANAGEMENT_2, 8, 0, @@ -886,11 +867,8 @@ EXPORT_SYMBOL_GPL(wm_hubs_add_analogue_controls); int wm_hubs_add_analogue_routes(struct snd_soc_codec *codec, int lineout1_diff, int lineout2_diff) { - struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); struct snd_soc_dapm_context *dapm = &codec->dapm; - init_completion(&hubs->dcs_done); - snd_soc_dapm_add_routes(dapm, analogue_routes, ARRAY_SIZE(analogue_routes)); diff --git a/trunk/sound/soc/codecs/wm_hubs.h b/trunk/sound/soc/codecs/wm_hubs.h index 676b1252ab91..f8a5e976b5e6 100644 --- a/trunk/sound/soc/codecs/wm_hubs.h +++ b/trunk/sound/soc/codecs/wm_hubs.h @@ -14,9 +14,6 @@ #ifndef _WM_HUBS_H #define _WM_HUBS_H -#include -#include - struct snd_soc_codec; extern const unsigned int wm_hubs_spkmix_tlv[]; @@ -26,14 +23,9 @@ struct wm_hubs_data { int dcs_codes; int dcs_readback_mode; int hp_startup_mode; - int series_startup; - int no_series_update; bool class_w; u16 class_w_dcs; - - bool dcs_done_irq; - struct completion dcs_done; }; extern int wm_hubs_add_analogue_controls(struct snd_soc_codec *); @@ -44,6 +36,4 @@ extern int wm_hubs_handle_analogue_pdata(struct snd_soc_codec *, int jd_scthr, int jd_thr, int micbias1_lvl, int micbias2_lvl); -extern irqreturn_t wm_hubs_dcs_done(int irq, void *data); - #endif diff --git a/trunk/sound/soc/davinci/davinci-pcm.c b/trunk/sound/soc/davinci/davinci-pcm.c index a49e667373bc..9d35b8c1a624 100644 --- a/trunk/sound/soc/davinci/davinci-pcm.c +++ b/trunk/sound/soc/davinci/davinci-pcm.c @@ -46,28 +46,11 @@ static void print_buf_info(int slot, char *name) } #endif -#define DAVINCI_PCM_FMTBITS (\ - SNDRV_PCM_FMTBIT_S8 |\ - SNDRV_PCM_FMTBIT_U8 |\ - SNDRV_PCM_FMTBIT_S16_LE |\ - SNDRV_PCM_FMTBIT_S16_BE |\ - SNDRV_PCM_FMTBIT_U16_LE |\ - SNDRV_PCM_FMTBIT_U16_BE |\ - SNDRV_PCM_FMTBIT_S24_LE |\ - SNDRV_PCM_FMTBIT_S24_BE |\ - SNDRV_PCM_FMTBIT_U24_LE |\ - SNDRV_PCM_FMTBIT_U24_BE |\ - SNDRV_PCM_FMTBIT_S32_LE |\ - SNDRV_PCM_FMTBIT_S32_BE |\ - SNDRV_PCM_FMTBIT_U32_LE |\ - SNDRV_PCM_FMTBIT_U32_BE) - static struct snd_pcm_hardware pcm_hardware_playback = { .info = (SNDRV_PCM_INFO_INTERLEAVED | SNDRV_PCM_INFO_BLOCK_TRANSFER | SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_MMAP_VALID | - SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_RESUME| - SNDRV_PCM_INFO_BATCH), - .formats = DAVINCI_PCM_FMTBITS, + SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_RESUME), + .formats = (SNDRV_PCM_FMTBIT_S16_LE), .rates = (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 | @@ -76,7 +59,7 @@ static struct snd_pcm_hardware pcm_hardware_playback = { .rate_min = 8000, .rate_max = 96000, .channels_min = 2, - .channels_max = 384, + .channels_max = 2, .buffer_bytes_max = 128 * 1024, .period_bytes_min = 32, .period_bytes_max = 8 * 1024, @@ -88,9 +71,8 @@ static struct snd_pcm_hardware pcm_hardware_playback = { static struct snd_pcm_hardware pcm_hardware_capture = { .info = (SNDRV_PCM_INFO_INTERLEAVED | SNDRV_PCM_INFO_BLOCK_TRANSFER | SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_MMAP_VALID | - SNDRV_PCM_INFO_PAUSE | - SNDRV_PCM_INFO_BATCH), - .formats = DAVINCI_PCM_FMTBITS, + SNDRV_PCM_INFO_PAUSE), + .formats = (SNDRV_PCM_FMTBIT_S16_LE), .rates = (SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 | SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000 | @@ -99,7 +81,7 @@ static struct snd_pcm_hardware pcm_hardware_capture = { .rate_min = 8000, .rate_max = 96000, .channels_min = 2, - .channels_max = 384, + .channels_max = 2, .buffer_bytes_max = 128 * 1024, .period_bytes_min = 32, .period_bytes_max = 8 * 1024, @@ -157,22 +139,6 @@ struct davinci_runtime_data { struct edmacc_param ram_params; }; -static void davinci_pcm_period_elapsed(struct snd_pcm_substream *substream) -{ - struct davinci_runtime_data *prtd = substream->runtime->private_data; - struct snd_pcm_runtime *runtime = substream->runtime; - - prtd->period++; - if (unlikely(prtd->period >= runtime->periods)) - prtd->period = 0; -} - -static void davinci_pcm_period_reset(struct snd_pcm_substream *substream) -{ - struct davinci_runtime_data *prtd = substream->runtime->private_data; - - prtd->period = 0; -} /* * Not used with ping/pong */ @@ -233,6 +199,10 @@ static void davinci_pcm_enqueue_dma(struct snd_pcm_substream *substream) else edma_set_transfer_params(link, acnt, fifo_level, count, fifo_level, ABSYNC); + + prtd->period++; + if (unlikely(prtd->period >= runtime->periods)) + prtd->period = 0; } static void davinci_pcm_dma_irq(unsigned link, u16 ch_status, void *data) @@ -247,13 +217,12 @@ static void davinci_pcm_dma_irq(unsigned link, u16 ch_status, void *data) return; if (snd_pcm_running(substream)) { - spin_lock(&prtd->lock); if (prtd->ram_channel < 0) { /* No ping/pong must fix up link dma data*/ + spin_lock(&prtd->lock); davinci_pcm_enqueue_dma(substream); + spin_unlock(&prtd->lock); } - davinci_pcm_period_elapsed(substream); - spin_unlock(&prtd->lock); snd_pcm_period_elapsed(substream); } } @@ -456,8 +425,7 @@ static int request_ping_pong(struct snd_pcm_substream *substream, edma_read_slot(link, &prtd->asp_params); prtd->asp_params.opt &= ~(TCCMODE | EDMA_TCC(0x3f) | TCINTEN); - prtd->asp_params.opt |= TCCHEN | - EDMA_TCC(prtd->ram_channel & 0x3f); + prtd->asp_params.opt |= TCCHEN | EDMA_TCC(prtd->ram_channel & 0x3f); edma_write_slot(link, &prtd->asp_params); /* pong */ @@ -471,7 +439,7 @@ static int request_ping_pong(struct snd_pcm_substream *substream, prtd->asp_params.opt &= ~(TCCMODE | EDMA_TCC(0x3f)); /* interrupt after every pong completion */ prtd->asp_params.opt |= TCINTEN | TCCHEN | - EDMA_TCC(prtd->ram_channel & 0x3f); + EDMA_TCC(EDMA_CHAN_SLOT(prtd->ram_channel)); edma_write_slot(link, &prtd->asp_params); /* ram */ @@ -559,13 +527,6 @@ static int davinci_pcm_trigger(struct snd_pcm_substream *substream, int cmd) switch (cmd) { case SNDRV_PCM_TRIGGER_START: - edma_start(prtd->asp_channel); - if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK && - prtd->ram_channel >= 0) { - /* copy 1st iram buffer */ - edma_start(prtd->ram_channel); - } - break; case SNDRV_PCM_TRIGGER_RESUME: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: edma_resume(prtd->asp_channel); @@ -589,7 +550,6 @@ static int davinci_pcm_prepare(struct snd_pcm_substream *substream) { struct davinci_runtime_data *prtd = substream->runtime->private_data; - davinci_pcm_period_reset(substream); if (prtd->ram_channel >= 0) { int ret = ping_pong_dma_setup(substream); if (ret < 0) @@ -605,31 +565,21 @@ static int davinci_pcm_prepare(struct snd_pcm_substream *substream) print_buf_info(prtd->asp_link[0], "asp_link[0]"); print_buf_info(prtd->asp_link[1], "asp_link[1]"); - /* - * There is a phase offset of 2 periods between the position - * used by dma setup and the position reported in the pointer - * function. - * - * The phase offset, when not using ping-pong buffers, is due to - * the two consecutive calls to davinci_pcm_enqueue_dma() below. - * - * Whereas here, with ping-pong buffers, the phase is due to - * there being an entire buffer transfer complete before the - * first dma completion event triggers davinci_pcm_dma_irq(). - */ - davinci_pcm_period_elapsed(substream); - davinci_pcm_period_elapsed(substream); - + if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { + /* copy 1st iram buffer */ + edma_start(prtd->ram_channel); + } + edma_start(prtd->asp_channel); return 0; } + prtd->period = 0; davinci_pcm_enqueue_dma(substream); - davinci_pcm_period_elapsed(substream); /* Copy self-linked parameter RAM entry into master channel */ edma_read_slot(prtd->asp_link[0], &prtd->asp_params); edma_write_slot(prtd->asp_channel, &prtd->asp_params); davinci_pcm_enqueue_dma(substream); - davinci_pcm_period_elapsed(substream); + edma_start(prtd->asp_channel); return 0; } @@ -641,23 +591,51 @@ davinci_pcm_pointer(struct snd_pcm_substream *substream) struct davinci_runtime_data *prtd = runtime->private_data; unsigned int offset; int asp_count; - unsigned int period_size = snd_pcm_lib_period_bytes(substream); - - /* - * There is a phase offset of 2 periods between the position used by dma - * setup and the position reported in the pointer function. Either +2 in - * the dma setup or -2 here in the pointer function (with wrapping, - * both) accounts for this offset -- choose the latter since it makes - * the first-time setup clearer. - */ + dma_addr_t asp_src, asp_dst; + spin_lock(&prtd->lock); - asp_count = prtd->period - 2; + if (prtd->ram_channel >= 0) { + int ram_count; + int mod_ram; + dma_addr_t ram_src, ram_dst; + unsigned int period_size = snd_pcm_lib_period_bytes(substream); + if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { + /* reading ram before asp should be safe + * as long as the asp transfers less than a ping size + * of bytes between the 2 reads + */ + edma_get_position(prtd->ram_channel, + &ram_src, &ram_dst); + edma_get_position(prtd->asp_channel, + &asp_src, &asp_dst); + asp_count = asp_src - prtd->asp_params.src; + ram_count = ram_src - prtd->ram_params.src; + mod_ram = ram_count % period_size; + mod_ram -= asp_count; + if (mod_ram < 0) + mod_ram += period_size; + else if (mod_ram == 0) { + if (snd_pcm_running(substream)) + mod_ram += period_size; + } + ram_count -= mod_ram; + if (ram_count < 0) + ram_count += period_size * runtime->periods; + } else { + edma_get_position(prtd->ram_channel, + &ram_src, &ram_dst); + ram_count = ram_dst - prtd->ram_params.dst; + } + asp_count = ram_count; + } else { + edma_get_position(prtd->asp_channel, &asp_src, &asp_dst); + if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) + asp_count = asp_src - runtime->dma_addr; + else + asp_count = asp_dst - runtime->dma_addr; + } spin_unlock(&prtd->lock); - if (asp_count < 0) - asp_count += runtime->periods; - asp_count *= period_size; - offset = bytes_to_frames(runtime, asp_count); if (offset >= runtime->buffer_size) offset = 0; @@ -833,11 +811,9 @@ static void davinci_pcm_free(struct snd_pcm *pcm) static u64 davinci_pcm_dmamask = 0xffffffff; -static int davinci_pcm_new(struct snd_soc_pcm_runtime *rtd) +static int davinci_pcm_new(struct snd_card *card, + struct snd_soc_dai *dai, struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret; if (!card->dev->dma_mask) diff --git a/trunk/sound/soc/ep93xx/ep93xx-pcm.c b/trunk/sound/soc/ep93xx/ep93xx-pcm.c index e27c417da437..a456e491155f 100644 --- a/trunk/sound/soc/ep93xx/ep93xx-pcm.c +++ b/trunk/sound/soc/ep93xx/ep93xx-pcm.c @@ -266,11 +266,9 @@ static void ep93xx_pcm_free_dma_buffers(struct snd_pcm *pcm) static u64 ep93xx_pcm_dmamask = 0xffffffff; -static int ep93xx_pcm_new(struct snd_soc_pcm_runtime *rtd) +static int ep93xx_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; if (!card->dev->dma_mask) diff --git a/trunk/sound/soc/fsl/fsl_dma.c b/trunk/sound/soc/fsl/fsl_dma.c index 732208c8c0b4..6680c0b4d203 100644 --- a/trunk/sound/soc/fsl/fsl_dma.c +++ b/trunk/sound/soc/fsl/fsl_dma.c @@ -294,11 +294,9 @@ static irqreturn_t fsl_dma_isr(int irq, void *dev_id) * Regardless of where the memory is actually allocated, since the device can * technically DMA to any 36-bit address, we do need to set the DMA mask to 36. */ -static int fsl_dma_new(struct snd_soc_pcm_runtime *rtd) +static int fsl_dma_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; static u64 fsl_dma_dmamask = DMA_BIT_MASK(36); int ret; @@ -941,7 +939,7 @@ static int __devinit fsl_soc_dma_probe(struct platform_device *pdev) iprop = of_get_property(ssi_np, "fsl,fifo-depth", NULL); if (iprop) - dma->ssi_fifo_depth = be32_to_cpup(iprop); + dma->ssi_fifo_depth = *iprop; else /* Older 8610 DTs didn't have the fifo-depth property */ dma->ssi_fifo_depth = 8; diff --git a/trunk/sound/soc/fsl/fsl_ssi.c b/trunk/sound/soc/fsl/fsl_ssi.c index d48afea5d93d..313e0ccedd5b 100644 --- a/trunk/sound/soc/fsl/fsl_ssi.c +++ b/trunk/sound/soc/fsl/fsl_ssi.c @@ -678,12 +678,7 @@ static int __devinit fsl_ssi_probe(struct platform_device *pdev) kfree(ssi_private); return ret; } - ssi_private->ssi = of_iomap(np, 0); - if (!ssi_private->ssi) { - dev_err(&pdev->dev, "could not map device resources\n"); - kfree(ssi_private); - return -ENOMEM; - } + ssi_private->ssi = ioremap(res.start, 1 + res.end - res.start); ssi_private->ssi_phys = res.start; ssi_private->irq = irq_of_parse_and_map(np, 0); @@ -696,7 +691,7 @@ static int __devinit fsl_ssi_probe(struct platform_device *pdev) /* Determine the FIFO depth. */ iprop = of_get_property(np, "fsl,fifo-depth", NULL); if (iprop) - ssi_private->fifo_depth = be32_to_cpup(iprop); + ssi_private->fifo_depth = *iprop; else /* Older 8610 DTs didn't have the fifo-depth property */ ssi_private->fifo_depth = 8; diff --git a/trunk/sound/soc/fsl/mpc5200_dma.c b/trunk/sound/soc/fsl/mpc5200_dma.c index 19ad0c1be67e..fff695ccdd3e 100644 --- a/trunk/sound/soc/fsl/mpc5200_dma.c +++ b/trunk/sound/soc/fsl/mpc5200_dma.c @@ -299,11 +299,10 @@ static struct snd_pcm_ops psc_dma_ops = { }; static u64 psc_dma_dmamask = 0xffffffff; -static int psc_dma_new(struct snd_soc_pcm_runtime *rtd) +static int psc_dma_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; + struct snd_soc_pcm_runtime *rtd = pcm->private_data; struct psc_dma *psc_dma = snd_soc_dai_get_drvdata(rtd->cpu_dai); size_t size = psc_dma_hardware.buffer_bytes_max; int rc = 0; diff --git a/trunk/sound/soc/fsl/mpc8610_hpcd.c b/trunk/sound/soc/fsl/mpc8610_hpcd.c index a19297959587..c16c6b2eff95 100644 --- a/trunk/sound/soc/fsl/mpc8610_hpcd.c +++ b/trunk/sound/soc/fsl/mpc8610_hpcd.c @@ -233,7 +233,7 @@ static int get_parent_cell_index(struct device_node *np) if (!iprop) return -1; - return be32_to_cpup(iprop); + return *iprop; } /** @@ -258,7 +258,7 @@ static int codec_node_dev_name(struct device_node *np, char *buf, size_t len) if (!iprop) return -EINVAL; - addr = be32_to_cpup(iprop); + addr = *iprop; bus = get_parent_cell_index(np); if (bus < 0) @@ -305,7 +305,7 @@ static int get_dma_channel(struct device_node *ssi_np, return -EINVAL; } - *dma_channel_id = be32_to_cpup(iprop); + *dma_channel_id = *iprop; *dma_id = get_parent_cell_index(dma_channel_np); of_node_put(dma_channel_np); @@ -379,7 +379,7 @@ static int mpc8610_hpcd_probe(struct platform_device *pdev) ret = -EINVAL; goto error; } - machine_data->ssi_id = be32_to_cpup(iprop); + machine_data->ssi_id = *iprop; /* Get the serial format and clock direction. */ sprop = of_get_property(np, "fsl,mode", NULL); @@ -405,7 +405,7 @@ static int mpc8610_hpcd_probe(struct platform_device *pdev) ret = -EINVAL; goto error; } - machine_data->clk_frequency = be32_to_cpup(iprop); + machine_data->clk_frequency = *iprop; } else if (strcasecmp(sprop, "i2s-master") == 0) { machine_data->dai_format = SND_SOC_DAIFMT_I2S; machine_data->codec_clk_direction = SND_SOC_CLOCK_IN; diff --git a/trunk/sound/soc/fsl/p1022_ds.c b/trunk/sound/soc/fsl/p1022_ds.c index 8fa4d5f8eda1..66e0b68af147 100644 --- a/trunk/sound/soc/fsl/p1022_ds.c +++ b/trunk/sound/soc/fsl/p1022_ds.c @@ -232,7 +232,7 @@ static int get_parent_cell_index(struct device_node *np) iprop = of_get_property(parent, "cell-index", NULL); if (iprop) - ret = be32_to_cpup(iprop); + ret = *iprop; of_node_put(parent); @@ -261,7 +261,7 @@ static int codec_node_dev_name(struct device_node *np, char *buf, size_t len) if (!iprop) return -EINVAL; - addr = be32_to_cpup(iprop); + addr = *iprop; bus = get_parent_cell_index(np); if (bus < 0) @@ -308,7 +308,7 @@ static int get_dma_channel(struct device_node *ssi_np, return -EINVAL; } - *dma_channel_id = be32_to_cpup(iprop); + *dma_channel_id = *iprop; *dma_id = get_parent_cell_index(dma_channel_np); of_node_put(dma_channel_np); @@ -379,7 +379,7 @@ static int p1022_ds_probe(struct platform_device *pdev) ret = -EINVAL; goto error; } - mdata->ssi_id = be32_to_cpup(iprop); + mdata->ssi_id = *iprop; /* Get the serial format and clock direction. */ sprop = of_get_property(np, "fsl,mode", NULL); @@ -405,7 +405,7 @@ static int p1022_ds_probe(struct platform_device *pdev) ret = -EINVAL; goto error; } - mdata->clk_frequency = be32_to_cpup(iprop); + mdata->clk_frequency = *iprop; } else if (strcasecmp(sprop, "i2s-master") == 0) { mdata->dai_format = SND_SOC_DAIFMT_I2S; mdata->codec_clk_direction = SND_SOC_CLOCK_IN; diff --git a/trunk/sound/soc/imx/Kconfig b/trunk/sound/soc/imx/Kconfig index bb699bb55a50..d8f130d39dd9 100644 --- a/trunk/sound/soc/imx/Kconfig +++ b/trunk/sound/soc/imx/Kconfig @@ -11,6 +11,9 @@ menuconfig SND_IMX_SOC if SND_IMX_SOC +config SND_MXC_SOC_SSI + tristate + config SND_MXC_SOC_FIQ tristate @@ -21,6 +24,7 @@ config SND_MXC_SOC_WM1133_EV1 tristate "Audio on the the i.MX31ADS with WM1133-EV1 fitted" depends on MACH_MX31ADS_WM1133_EV1 && EXPERIMENTAL select SND_SOC_WM8350 + select SND_MXC_SOC_SSI select SND_MXC_SOC_FIQ help Enable support for audio on the i.MX31ADS with the WM1133-EV1 @@ -30,6 +34,7 @@ config SND_SOC_MX27VIS_AIC32X4 tristate "SoC audio support for Visstrim M10 boards" depends on MACH_IMX27_VISSTRIM_M10 select SND_SOC_TVL320AIC32X4 + select SND_MXC_SOC_SSI select SND_MXC_SOC_MX2 help Say Y if you want to add support for SoC audio on Visstrim SM10 @@ -39,6 +44,7 @@ config SND_SOC_PHYCORE_AC97 tristate "SoC Audio support for Phytec phyCORE (and phyCARD) boards" depends on MACH_PCM043 || MACH_PCA100 select SND_SOC_WM9712 + select SND_MXC_SOC_SSI select SND_MXC_SOC_FIQ help Say Y if you want to add support for SoC audio on Phytec phyCORE @@ -51,6 +57,7 @@ config SND_SOC_EUKREA_TLV320 || MACH_EUKREA_MBIMXSD35_BASEBOARD \ || MACH_EUKREA_MBIMXSD51_BASEBOARD select SND_SOC_TLV320AIC23 + select SND_MXC_SOC_SSI select SND_MXC_SOC_FIQ help Enable I2S based access to the TLV320AIC23B codec attached diff --git a/trunk/sound/soc/imx/imx-pcm-dma-mx2.c b/trunk/sound/soc/imx/imx-pcm-dma-mx2.c index 4173b3d87f97..aab7765f401a 100644 --- a/trunk/sound/soc/imx/imx-pcm-dma-mx2.c +++ b/trunk/sound/soc/imx/imx-pcm-dma-mx2.c @@ -337,5 +337,3 @@ static void __exit snd_imx_pcm_exit(void) platform_driver_unregister(&imx_pcm_driver); } module_exit(snd_imx_pcm_exit); -MODULE_LICENSE("GPL"); -MODULE_ALIAS("platform:imx-pcm-audio"); diff --git a/trunk/sound/soc/imx/imx-pcm-fiq.c b/trunk/sound/soc/imx/imx-pcm-fiq.c index 309c59e6fb6c..413b78da248f 100644 --- a/trunk/sound/soc/imx/imx-pcm-fiq.c +++ b/trunk/sound/soc/imx/imx-pcm-fiq.c @@ -238,14 +238,12 @@ static struct snd_pcm_ops imx_pcm_ops = { static int ssi_irq = 0; -static int imx_pcm_fiq_new(struct snd_soc_pcm_runtime *rtd) +static int imx_pcm_fiq_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret; - ret = imx_pcm_new(rtd); + ret = imx_pcm_new(card, dai, pcm); if (ret) return ret; diff --git a/trunk/sound/soc/imx/imx-ssi.c b/trunk/sound/soc/imx/imx-ssi.c index 10a8e2783751..5b13feca7537 100644 --- a/trunk/sound/soc/imx/imx-ssi.c +++ b/trunk/sound/soc/imx/imx-ssi.c @@ -388,11 +388,10 @@ static int imx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) static u64 imx_pcm_dmamask = DMA_BIT_MASK(32); -int imx_pcm_new(struct snd_soc_pcm_runtime *rtd) +int imx_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; + int ret = 0; if (!card->dev->dma_mask) @@ -775,4 +774,4 @@ module_exit(imx_ssi_exit); MODULE_AUTHOR("Sascha Hauer, "); MODULE_DESCRIPTION("i.MX I2S/ac97 SoC Interface"); MODULE_LICENSE("GPL"); -MODULE_ALIAS("platform:imx-ssi"); + diff --git a/trunk/sound/soc/imx/imx-ssi.h b/trunk/sound/soc/imx/imx-ssi.h index 0a84cec3599e..dc8a87530e3e 100644 --- a/trunk/sound/soc/imx/imx-ssi.h +++ b/trunk/sound/soc/imx/imx-ssi.h @@ -225,7 +225,8 @@ struct snd_soc_platform *imx_ssi_dma_mx2_init(struct platform_device *pdev, struct imx_ssi *ssi); int snd_imx_pcm_mmap(struct snd_pcm_substream *substream, struct vm_area_struct *vma); -int imx_pcm_new(struct snd_soc_pcm_runtime *rtd); +int imx_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm); void imx_pcm_free(struct snd_pcm *pcm); /* diff --git a/trunk/sound/soc/jz4740/jz4740-pcm.c b/trunk/sound/soc/jz4740/jz4740-pcm.c index a7c9578be983..fb1483f7c966 100644 --- a/trunk/sound/soc/jz4740/jz4740-pcm.c +++ b/trunk/sound/soc/jz4740/jz4740-pcm.c @@ -299,11 +299,9 @@ static void jz4740_pcm_free(struct snd_pcm *pcm) static u64 jz4740_pcm_dmamask = DMA_BIT_MASK(32); -int jz4740_pcm_new(struct snd_soc_pcm_runtime *rtd) +int jz4740_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; if (!card->dev->dma_mask) diff --git a/trunk/sound/soc/kirkwood/kirkwood-dma.c b/trunk/sound/soc/kirkwood/kirkwood-dma.c index cd33de1c5b7a..e13c6ce46328 100644 --- a/trunk/sound/soc/kirkwood/kirkwood-dma.c +++ b/trunk/sound/soc/kirkwood/kirkwood-dma.c @@ -312,11 +312,9 @@ static int kirkwood_dma_preallocate_dma_buffer(struct snd_pcm *pcm, return 0; } -static int kirkwood_dma_new(struct snd_soc_pcm_runtime *rtd) +static int kirkwood_dma_new(struct snd_card *card, + struct snd_soc_dai *dai, struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret; if (!card->dev->dma_mask) diff --git a/trunk/sound/soc/mid-x86/sst_platform.c b/trunk/sound/soc/mid-x86/sst_platform.c index 3e7826058efe..5a946b4115a2 100644 --- a/trunk/sound/soc/mid-x86/sst_platform.c +++ b/trunk/sound/soc/mid-x86/sst_platform.c @@ -402,10 +402,9 @@ static void sst_pcm_free(struct snd_pcm *pcm) snd_pcm_lib_preallocate_free_for_all(pcm); } -int sst_pcm_new(struct snd_soc_pcm_runtime *rtd) +int sst_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int retval = 0; pr_debug("sst_pcm_new called\n"); diff --git a/trunk/sound/soc/nuc900/nuc900-ac97.c b/trunk/sound/soc/nuc900/nuc900-ac97.c index 9c0edad90d8b..dac6732da969 100644 --- a/trunk/sound/soc/nuc900/nuc900-ac97.c +++ b/trunk/sound/soc/nuc900/nuc900-ac97.c @@ -356,7 +356,7 @@ static int __devinit nuc900_ac97_drvprobe(struct platform_device *pdev) nuc900_audio->irq_num = platform_get_irq(pdev, 0); if (!nuc900_audio->irq_num) { ret = -EBUSY; - goto out3; + goto out2; } nuc900_ac97_data = nuc900_audio; diff --git a/trunk/sound/soc/nuc900/nuc900-pcm.c b/trunk/sound/soc/nuc900/nuc900-pcm.c index d589ef14e917..8263f56dc665 100644 --- a/trunk/sound/soc/nuc900/nuc900-pcm.c +++ b/trunk/sound/soc/nuc900/nuc900-pcm.c @@ -315,12 +315,9 @@ static void nuc900_dma_free_dma_buffers(struct snd_pcm *pcm) } static u64 nuc900_pcm_dmamask = DMA_BIT_MASK(32); -static int nuc900_dma_new(struct snd_soc_pcm_runtime *rtd) +static int nuc900_dma_new(struct snd_card *card, + struct snd_soc_dai *dai, struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; - if (!card->dev->dma_mask) card->dev->dma_mask = &nuc900_pcm_dmamask; if (!card->dev->coherent_dma_mask) diff --git a/trunk/sound/soc/omap/Kconfig b/trunk/sound/soc/omap/Kconfig index fe83d0d176be..99054cf1f68f 100644 --- a/trunk/sound/soc/omap/Kconfig +++ b/trunk/sound/soc/omap/Kconfig @@ -9,9 +9,6 @@ config SND_OMAP_SOC_MCBSP config SND_OMAP_SOC_MCPDM tristate -config SND_OMAP_SOC_HDMI - tristate - config SND_OMAP_SOC_N810 tristate "SoC Audio support for Nokia N810" depends on SND_OMAP_SOC && MACH_NOKIA_N810 && I2C @@ -103,14 +100,6 @@ config SND_OMAP_SOC_SDP4430 Say Y if you want to add support for SoC audio on Texas Instruments SDP4430. -config SND_OMAP_SOC_OMAP4_HDMI - tristate "SoC Audio support for Texas Instruments OMAP4 HDMI" - depends on SND_OMAP_SOC && OMAP4_DSS_HDMI && OMAP2_DSS && ARCH_OMAP4 - select SND_OMAP_SOC_HDMI - help - Say Y if you want to add support for SoC HDMI audio on Texas Instruments - OMAP4 chips - config SND_OMAP_SOC_OMAP3_PANDORA tristate "SoC Audio support for OMAP3 Pandora" depends on TWL4030_CORE && SND_OMAP_SOC && MACH_OMAP3_PANDORA diff --git a/trunk/sound/soc/omap/Makefile b/trunk/sound/soc/omap/Makefile index 59e2c8d1e38d..6c2c87eed5bb 100644 --- a/trunk/sound/soc/omap/Makefile +++ b/trunk/sound/soc/omap/Makefile @@ -2,12 +2,10 @@ snd-soc-omap-objs := omap-pcm.o snd-soc-omap-mcbsp-objs := omap-mcbsp.o snd-soc-omap-mcpdm-objs := omap-mcpdm.o mcpdm.o -snd-soc-omap-hdmi-objs := omap-hdmi.o obj-$(CONFIG_SND_OMAP_SOC) += snd-soc-omap.o obj-$(CONFIG_SND_OMAP_SOC_MCBSP) += snd-soc-omap-mcbsp.o obj-$(CONFIG_SND_OMAP_SOC_MCPDM) += snd-soc-omap-mcpdm.o -obj-$(CONFIG_SND_OMAP_SOC_HDMI) += snd-soc-omap-hdmi.o # OMAP Machine Support snd-soc-n810-objs := n810.o @@ -23,7 +21,6 @@ snd-soc-omap3pandora-objs := omap3pandora.o snd-soc-omap3beagle-objs := omap3beagle.o snd-soc-zoom2-objs := zoom2.o snd-soc-igep0020-objs := igep0020.o -snd-soc-omap4-hdmi-objs := omap4-hdmi-card.o obj-$(CONFIG_SND_OMAP_SOC_N810) += snd-soc-n810.o obj-$(CONFIG_SND_OMAP_SOC_RX51) += snd-soc-rx51.o @@ -39,4 +36,3 @@ obj-$(CONFIG_SND_OMAP_SOC_OMAP3_PANDORA) += snd-soc-omap3pandora.o obj-$(CONFIG_SND_OMAP_SOC_OMAP3_BEAGLE) += snd-soc-omap3beagle.o obj-$(CONFIG_SND_OMAP_SOC_ZOOM2) += snd-soc-zoom2.o obj-$(CONFIG_SND_OMAP_SOC_IGEP0020) += snd-soc-igep0020.o -obj-$(CONFIG_SND_OMAP_SOC_OMAP4_HDMI) += snd-soc-omap4-hdmi.o diff --git a/trunk/sound/soc/omap/ams-delta.c b/trunk/sound/soc/omap/ams-delta.c index b40095a19883..462cbcbea74a 100644 --- a/trunk/sound/soc/omap/ams-delta.c +++ b/trunk/sound/soc/omap/ams-delta.c @@ -427,8 +427,7 @@ static struct snd_soc_ops ams_delta_ops = { /* Board specific codec bias level control */ static int ams_delta_set_bias_level(struct snd_soc_card *card, - struct snd_soc_dapm_context *dapm, - enum snd_soc_bias_level level) + enum snd_soc_bias_level level) { struct snd_soc_codec *codec = card->rtd->codec; diff --git a/trunk/sound/soc/omap/omap-hdmi.c b/trunk/sound/soc/omap/omap-hdmi.c deleted file mode 100644 index 36c6eaeffb02..000000000000 --- a/trunk/sound/soc/omap/omap-hdmi.c +++ /dev/null @@ -1,158 +0,0 @@ -/* - * omap-hdmi.c - * - * OMAP ALSA SoC DAI driver for HDMI audio on OMAP4 processors. - * Copyright (C) 2010-2011 Texas Instruments Incorporated - http://www.ti.com/ - * Authors: Jorge Candelaria - * Ricardo Neri - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * version 2 as published by the Free Software Foundation. - * - * 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, write to the Free Software - * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA - * 02110-1301 USA - * - */ - -#include -#include -#include -#include -#include -#include -#include -#include - -#include -#include "omap-pcm.h" -#include "omap-hdmi.h" - -#define DRV_NAME "hdmi-audio-dai" - -static struct omap_pcm_dma_data omap_hdmi_dai_dma_params = { - .name = "HDMI playback", - .sync_mode = OMAP_DMA_SYNC_PACKET, -}; - -static int omap_hdmi_dai_startup(struct snd_pcm_substream *substream, - struct snd_soc_dai *dai) -{ - int err; - /* - * Make sure that the period bytes are multiple of the DMA packet size. - * Largest packet size we use is 32 32-bit words = 128 bytes - */ - err = snd_pcm_hw_constraint_step(substream->runtime, 0, - SNDRV_PCM_HW_PARAM_PERIOD_BYTES, 128); - if (err < 0) - return err; - - return 0; -} - -static int omap_hdmi_dai_hw_params(struct snd_pcm_substream *substream, - struct snd_pcm_hw_params *params, - struct snd_soc_dai *dai) -{ - int err = 0; - - switch (params_format(params)) { - case SNDRV_PCM_FORMAT_S16_LE: - omap_hdmi_dai_dma_params.packet_size = 16; - break; - case SNDRV_PCM_FORMAT_S24_LE: - omap_hdmi_dai_dma_params.packet_size = 32; - break; - default: - err = -EINVAL; - } - - omap_hdmi_dai_dma_params.data_type = OMAP_DMA_DATA_TYPE_S32; - - snd_soc_dai_set_dma_data(dai, substream, - &omap_hdmi_dai_dma_params); - - return err; -} - -static struct snd_soc_dai_ops omap_hdmi_dai_ops = { - .startup = omap_hdmi_dai_startup, - .hw_params = omap_hdmi_dai_hw_params, -}; - -static struct snd_soc_dai_driver omap_hdmi_dai = { - .playback = { - .channels_min = 2, - .channels_max = 2, - .rates = OMAP_HDMI_RATES, - .formats = OMAP_HDMI_FORMATS, - }, - .ops = &omap_hdmi_dai_ops, -}; - -static __devinit int omap_hdmi_probe(struct platform_device *pdev) -{ - int ret; - struct resource *hdmi_rsrc; - - hdmi_rsrc = platform_get_resource(pdev, IORESOURCE_MEM, 0); - if (!hdmi_rsrc) { - dev_err(&pdev->dev, "Cannot obtain IORESOURCE_MEM HDMI\n"); - return -EINVAL; - } - - omap_hdmi_dai_dma_params.port_addr = hdmi_rsrc->start - + OMAP_HDMI_AUDIO_DMA_PORT; - - hdmi_rsrc = platform_get_resource(pdev, IORESOURCE_DMA, 0); - if (!hdmi_rsrc) { - dev_err(&pdev->dev, "Cannot obtain IORESOURCE_DMA HDMI\n"); - return -EINVAL; - } - - omap_hdmi_dai_dma_params.dma_req = hdmi_rsrc->start; - - ret = snd_soc_register_dai(&pdev->dev, &omap_hdmi_dai); - return ret; -} - -static int __devexit omap_hdmi_remove(struct platform_device *pdev) -{ - snd_soc_unregister_dai(&pdev->dev); - return 0; -} - -static struct platform_driver hdmi_dai_driver = { - .driver = { - .name = DRV_NAME, - .owner = THIS_MODULE, - }, - .probe = omap_hdmi_probe, - .remove = __devexit_p(omap_hdmi_remove), -}; - -static int __init hdmi_dai_init(void) -{ - return platform_driver_register(&hdmi_dai_driver); -} -module_init(hdmi_dai_init); - -static void __exit hdmi_dai_exit(void) -{ - platform_driver_unregister(&hdmi_dai_driver); -} -module_exit(hdmi_dai_exit); - -MODULE_AUTHOR("Jorge Candelaria "); -MODULE_AUTHOR("Ricardo Neri "); -MODULE_DESCRIPTION("OMAP HDMI SoC Interface"); -MODULE_LICENSE("GPL"); -MODULE_ALIAS("platform:" DRV_NAME); diff --git a/trunk/sound/soc/omap/omap-hdmi.h b/trunk/sound/soc/omap/omap-hdmi.h deleted file mode 100644 index 34c298d5057e..000000000000 --- a/trunk/sound/soc/omap/omap-hdmi.h +++ /dev/null @@ -1,36 +0,0 @@ -/* - * omap-hdmi.h - * - * Definitions for OMAP ALSA SoC DAI driver for HDMI audio on OMAP4 processors. - * Copyright (C) 2010-2011 Texas Instruments Incorporated - http://www.ti.com/ - * Authors: Jorge Candelaria - * Ricardo Neri - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * version 2 as published by the Free Software Foundation. - * - * 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, write to the Free Software - * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA - * 02110-1301 USA - * - */ - -#ifndef __OMAP_HDMI_H__ -#define __OMAP_HDMI_H__ - -#define OMAP_HDMI_AUDIO_DMA_PORT 0x8c - -#define OMAP_HDMI_RATES (SNDRV_PCM_RATE_32000 | \ - SNDRV_PCM_RATE_44100 | SNDRV_PCM_RATE_48000) - -#define OMAP_HDMI_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | \ - SNDRV_PCM_FMTBIT_S24_LE) - -#endif diff --git a/trunk/sound/soc/omap/omap-pcm.c b/trunk/sound/soc/omap/omap-pcm.c index b2f5751edae3..e6a6b991d05f 100644 --- a/trunk/sound/soc/omap/omap-pcm.c +++ b/trunk/sound/soc/omap/omap-pcm.c @@ -366,11 +366,9 @@ static void omap_pcm_free_dma_buffers(struct snd_pcm *pcm) } } -static int omap_pcm_new(struct snd_soc_pcm_runtime *rtd) +static int omap_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, + struct snd_pcm *pcm) { - struct snd_card *card = rtd->card->snd_card; - struct snd_soc_dai *dai = rtd->cpu_dai; - struct snd_pcm *pcm = rtd->pcm; int ret = 0; if (!card->dev->dma_mask) diff --git a/trunk/sound/soc/omap/omap4-hdmi-card.c b/trunk/sound/soc/omap/omap4-hdmi-card.c deleted file mode 100644 index 9f32615b81f7..000000000000 --- a/trunk/sound/soc/omap/omap4-hdmi-card.c +++ /dev/null @@ -1,129 +0,0 @@ -/* - * omap4-hdmi-card.c - * - * OMAP ALSA SoC machine driver for TI OMAP4 HDMI - * Copyright (C) 2011 Texas Instruments Incorporated - http://www.ti.com/ - * Author: Ricardo Neri - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * version 2 as published by the Free Software Foundation. - * - * 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, write to the Free Software - * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA - * 02110-1301 USA - * - */ - -#include -#include -#include -#include