diff options
Diffstat (limited to 'sound/soc/codecs/wm_hubs.c')
| -rw-r--r-- | sound/soc/codecs/wm_hubs.c | 714 | 
1 files changed, 564 insertions, 150 deletions
diff --git a/sound/soc/codecs/wm_hubs.c b/sound/soc/codecs/wm_hubs.c index 19ca782ac97..916817fe663 100644 --- a/sound/soc/codecs/wm_hubs.c +++ b/sound/soc/codecs/wm_hubs.c @@ -1,7 +1,7 @@  /*   * wm_hubs.c  --  WM8993/4 common code   * - * Copyright 2009 Wolfson Microelectronics plc + * Copyright 2009-12 Wolfson Microelectronics plc   *   * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>   * @@ -17,12 +17,11 @@  #include <linux/delay.h>  #include <linux/pm.h>  #include <linux/i2c.h> -#include <linux/platform_device.h> +#include <linux/mfd/wm8994/registers.h>  #include <sound/core.h>  #include <sound/pcm.h>  #include <sound/pcm_params.h>  #include <sound/soc.h> -#include <sound/soc-dapm.h>  #include <sound/initval.h>  #include <sound/tlv.h> @@ -40,7 +39,7 @@ static const DECLARE_TLV_DB_SCALE(outmix_tlv, -2100, 300, 0);  static const DECLARE_TLV_DB_SCALE(spkmixout_tlv, -1800, 600, 1);  static const DECLARE_TLV_DB_SCALE(outpga_tlv, -5700, 100, 0);  static const unsigned int spkboost_tlv[] = { -	TLV_DB_RANGE_HEAD(7), +	TLV_DB_RANGE_HEAD(2),  	0, 6, TLV_DB_SCALE_ITEM(0, 150, 0),  	7, 7, TLV_DB_SCALE_ITEM(1200, 0, 0),  }; @@ -51,21 +50,23 @@ static const char *speaker_ref_text[] = {  	"VMID",  }; -static const struct soc_enum speaker_ref = -	SOC_ENUM_SINGLE(WM8993_SPEAKER_MIXER, 8, 2, speaker_ref_text); +static SOC_ENUM_SINGLE_DECL(speaker_ref, +			    WM8993_SPEAKER_MIXER, 8, speaker_ref_text);  static const char *speaker_mode_text[] = {  	"Class D",  	"Class AB",  }; -static const struct soc_enum speaker_mode = -	SOC_ENUM_SINGLE(WM8993_SPKMIXR_ATTENUATION, 8, 2, speaker_mode_text); +static SOC_ENUM_SINGLE_DECL(speaker_mode, +			    WM8993_SPKMIXR_ATTENUATION, 8, speaker_mode_text);  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; @@ -75,80 +76,259 @@ 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++; -		msleep(1); + +		if (hubs->dcs_done_irq) +			wait_for_completion_timeout(&hubs->dcs_done, +						    msecs_to_jiffies(250)); +		else +			msleep(1); +  		reg = snd_soc_read(codec, WM8993_DC_SERVO_0);  		dev_dbg(codec->dev, "DC servo: %x\n", reg); -	} while (reg & op && count < 400); +	} while (reg & op && count < timeout);  	if (reg & op) -		dev_err(codec->dev, "Timed out waiting for DC Servo\n"); +		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); + +static bool wm_hubs_dac_hp_direct(struct snd_soc_codec *codec) +{ +	int reg; + +	/* If we're going via the mixer we'll need to do additional checks */ +	reg = snd_soc_read(codec, WM8993_OUTPUT_MIXER1); +	if (!(reg & WM8993_DACL_TO_HPOUT1L)) { +		if (reg & ~WM8993_DACL_TO_MIXOUTL) { +			dev_vdbg(codec->dev, "Analogue paths connected: %x\n", +				 reg & ~WM8993_DACL_TO_HPOUT1L); +			return false; +		} else { +			dev_vdbg(codec->dev, "HPL connected to mixer\n"); +		} +	} else { +		dev_vdbg(codec->dev, "HPL connected to DAC\n"); +	} + +	reg = snd_soc_read(codec, WM8993_OUTPUT_MIXER2); +	if (!(reg & WM8993_DACR_TO_HPOUT1R)) { +		if (reg & ~WM8993_DACR_TO_MIXOUTR) { +			dev_vdbg(codec->dev, "Analogue paths connected: %x\n", +				 reg & ~WM8993_DACR_TO_HPOUT1R); +			return false; +		} else { +			dev_vdbg(codec->dev, "HPR connected to mixer\n"); +		} +	} else { +		dev_vdbg(codec->dev, "HPR connected to DAC\n"); +	} + +	return true; +} + +struct wm_hubs_dcs_cache { +	struct list_head list; +	unsigned int left; +	unsigned int right; +	u16 dcs_cfg; +}; + +static bool wm_hubs_dcs_cache_get(struct snd_soc_codec *codec, +				  struct wm_hubs_dcs_cache **entry) +{ +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	struct wm_hubs_dcs_cache *cache; +	unsigned int left, right; + +	left = snd_soc_read(codec, WM8993_LEFT_OUTPUT_VOLUME); +	left &= WM8993_HPOUT1L_VOL_MASK; + +	right = snd_soc_read(codec, WM8993_RIGHT_OUTPUT_VOLUME); +	right &= WM8993_HPOUT1R_VOL_MASK; + +	list_for_each_entry(cache, &hubs->dcs_cache, list) { +		if (cache->left != left || cache->right != right) +			continue; + +		*entry = cache; +		return true; +	} + +	return false; +} + +static void wm_hubs_dcs_cache_set(struct snd_soc_codec *codec, u16 dcs_cfg) +{ +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	struct wm_hubs_dcs_cache *cache; + +	if (hubs->no_cache_dac_hp_direct) +		return; + +	cache = devm_kzalloc(codec->dev, sizeof(*cache), GFP_KERNEL); +	if (!cache) { +		dev_err(codec->dev, "Failed to allocate DCS cache entry\n"); +		return; +	} + +	cache->left = snd_soc_read(codec, WM8993_LEFT_OUTPUT_VOLUME); +	cache->left &= WM8993_HPOUT1L_VOL_MASK; + +	cache->right = snd_soc_read(codec, WM8993_RIGHT_OUTPUT_VOLUME); +	cache->right &= WM8993_HPOUT1R_VOL_MASK; + +	cache->dcs_cfg = dcs_cfg; + +	list_add_tail(&cache->list, &hubs->dcs_cache); +} + +static int wm_hubs_read_dc_servo(struct snd_soc_codec *codec, +				  u16 *reg_l, u16 *reg_r) +{ +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	u16 dcs_reg, reg; +	int ret = 0; + +	switch (hubs->dcs_readback_mode) { +	case 2: +		dcs_reg = WM8994_DC_SERVO_4E; +		break; +	case 1: +		dcs_reg = WM8994_DC_SERVO_READBACK; +		break; +	default: +		dcs_reg = WM8993_DC_SERVO_3; +		break; +	} + +	/* Different chips in the family support different readback +	 * methods. +	 */ +	switch (hubs->dcs_readback_mode) { +	case 0: +		*reg_l = snd_soc_read(codec, WM8993_DC_SERVO_READBACK_1) +			& WM8993_DCS_INTEG_CHAN_0_MASK; +		*reg_r = snd_soc_read(codec, WM8993_DC_SERVO_READBACK_2) +			& WM8993_DCS_INTEG_CHAN_1_MASK; +		break; +	case 2: +	case 1: +		reg = snd_soc_read(codec, dcs_reg); +		*reg_r = (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; +		break; +	default: +		WARN(1, "Unknown DCS readback method\n"); +		ret = -1; +	} +	return ret;  }  /*   * Startup calibration of the DC servo   */ -static void calibrate_dc_servo(struct snd_soc_codec *codec) +static void enable_dc_servo(struct snd_soc_codec *codec)  {  	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); -	u16 reg, reg_l, reg_r, dcs_cfg; +	struct wm_hubs_dcs_cache *cache; +	s8 offset; +	u16 reg_l, reg_r, dcs_cfg, dcs_reg; -	/* Set for 32 series updates */ -	snd_soc_update_bits(codec, WM8993_DC_SERVO_1, -			    WM8993_DCS_SERIES_NO_01_MASK, -			    32 << WM8993_DCS_SERIES_NO_01_SHIFT); -	wait_for_dc_servo(codec, -			  WM8993_DCS_TRIG_SERIES_0 | WM8993_DCS_TRIG_SERIES_1); +	switch (hubs->dcs_readback_mode) { +	case 2: +		dcs_reg = WM8994_DC_SERVO_4E; +		break; +	default: +		dcs_reg = WM8993_DC_SERVO_3; +		break; +	} -	/* Apply correction to DC servo result */ -	if (hubs->dcs_codes) { -		dev_dbg(codec->dev, "Applying %d code DC servo correction\n", -			hubs->dcs_codes); - -		/* Different chips in the family support different -		 * readback methods. -		 */ -		switch (hubs->dcs_readback_mode) { -		case 0: -			reg_l = snd_soc_read(codec, WM8993_DC_SERVO_READBACK_1) -				& WM8993_DCS_INTEG_CHAN_0_MASK;; -			reg_r = snd_soc_read(codec, WM8993_DC_SERVO_READBACK_2) -				& WM8993_DCS_INTEG_CHAN_1_MASK; -			break; -		case 1: -			reg = snd_soc_read(codec, WM8993_DC_SERVO_3); -			reg_l = (reg & WM8993_DCS_DAC_WR_VAL_1_MASK) -				>> WM8993_DCS_DAC_WR_VAL_1_SHIFT; -			reg_r = reg & WM8993_DCS_DAC_WR_VAL_0_MASK; -			break; -		default: -			WARN(1, "Unknown DCS readback method\n"); -			break; -		} +	/* If we're using a digital only path and have a previously +	 * callibrated DC servo offset stored then use that. */ +	if (wm_hubs_dac_hp_direct(codec) && +	    wm_hubs_dcs_cache_get(codec, &cache)) { +		dev_dbg(codec->dev, "Using cached DCS offset %x for %d,%d\n", +			cache->dcs_cfg, cache->left, cache->right); +		snd_soc_write(codec, dcs_reg, cache->dcs_cfg); +		wait_for_dc_servo(codec, +				  WM8993_DCS_TRIG_DAC_WR_0 | +				  WM8993_DCS_TRIG_DAC_WR_1); +		return; +	} + +	if (hubs->series_startup) { +		/* Set for 32 series updates */ +		snd_soc_update_bits(codec, WM8993_DC_SERVO_1, +				    WM8993_DCS_SERIES_NO_01_MASK, +				    32 << WM8993_DCS_SERIES_NO_01_SHIFT); +		wait_for_dc_servo(codec, +				  WM8993_DCS_TRIG_SERIES_0 | +				  WM8993_DCS_TRIG_SERIES_1); +	} else { +		wait_for_dc_servo(codec, +				  WM8993_DCS_TRIG_STARTUP_0 | +				  WM8993_DCS_TRIG_STARTUP_1); +	} -		dev_dbg(codec->dev, "DCS input: %x %x\n", reg_l, reg_r); +	if (wm_hubs_read_dc_servo(codec, ®_l, ®_r) < 0) +		return; -		/* HPOUT1L */ -		if (reg_l + hubs->dcs_codes > 0 && -		    reg_l + hubs->dcs_codes < 0xff) -			reg_l += hubs->dcs_codes; -		dcs_cfg = reg_l << WM8993_DCS_DAC_WR_VAL_1_SHIFT; +	dev_dbg(codec->dev, "DCS input: %x %x\n", reg_l, reg_r); + +	/* Apply correction to DC servo result */ +	if (hubs->dcs_codes_l || hubs->dcs_codes_r) { +		dev_dbg(codec->dev, +			"Applying %d/%d code DC servo correction\n", +			hubs->dcs_codes_l, hubs->dcs_codes_r);  		/* HPOUT1R */ -		if (reg_r + hubs->dcs_codes > 0 && -		    reg_r + hubs->dcs_codes < 0xff) -			reg_r += hubs->dcs_codes; -		dcs_cfg |= reg_r; +		offset = (s8)reg_r; +		dev_dbg(codec->dev, "DCS right %d->%d\n", offset, +			offset + hubs->dcs_codes_r); +		offset += hubs->dcs_codes_r; +		dcs_cfg = (u8)offset << WM8993_DCS_DAC_WR_VAL_1_SHIFT; + +		/* HPOUT1L */ +		offset = (s8)reg_l; +		dev_dbg(codec->dev, "DCS left %d->%d\n", offset, +			offset + hubs->dcs_codes_l); +		offset += hubs->dcs_codes_l; +		dcs_cfg |= (u8)offset;  		dev_dbg(codec->dev, "DCS result: %x\n", dcs_cfg);  		/* Do it */ -		snd_soc_write(codec, WM8993_DC_SERVO_3, dcs_cfg); +		snd_soc_write(codec, dcs_reg, dcs_cfg);  		wait_for_dc_servo(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;  	} + +	/* Save the callibrated offset if we're in class W mode and +	 * therefore don't have any analogue signal mixed in. */ +	if (wm_hubs_dac_hp_direct(codec)) +		wm_hubs_dcs_cache_set(codec, dcs_cfg);  }  /* @@ -157,15 +337,15 @@ static void calibrate_dc_servo(struct snd_soc_codec *codec)  static int wm8993_put_dc_servo(struct snd_kcontrol *kcontrol,  			       struct snd_ctl_elem_value *ucontrol)  { -	struct snd_soc_codec *codec = snd_kcontrol_chip(kcontrol); +	struct snd_soc_codec *codec = snd_soc_kcontrol_codec(kcontrol);  	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec);  	int ret; -	ret = snd_soc_put_volsw_2r(kcontrol, ucontrol); +	ret = snd_soc_put_volsw(kcontrol, ucontrol);  	/* If we're applying an offset correction then updating the  	 * callibration would be likely to introduce further offsets. */ -	if (hubs->dcs_codes) +	if (hubs->dcs_codes_l || hubs->dcs_codes_r || hubs->no_series_update)  		return ret;  	/* Only need to do this if the outputs are active */ @@ -185,23 +365,23 @@ static const struct snd_kcontrol_new analogue_snd_controls[] = {  SOC_SINGLE_TLV("IN1L Volume", WM8993_LEFT_LINE_INPUT_1_2_VOLUME, 0, 31, 0,  	       inpga_tlv),  SOC_SINGLE("IN1L Switch", WM8993_LEFT_LINE_INPUT_1_2_VOLUME, 7, 1, 1), -SOC_SINGLE("IN1L ZC Switch", WM8993_LEFT_LINE_INPUT_1_2_VOLUME, 7, 1, 0), +SOC_SINGLE("IN1L ZC Switch", WM8993_LEFT_LINE_INPUT_1_2_VOLUME, 6, 1, 0),  SOC_SINGLE_TLV("IN1R Volume", WM8993_RIGHT_LINE_INPUT_1_2_VOLUME, 0, 31, 0,  	       inpga_tlv),  SOC_SINGLE("IN1R Switch", WM8993_RIGHT_LINE_INPUT_1_2_VOLUME, 7, 1, 1), -SOC_SINGLE("IN1R ZC Switch", WM8993_RIGHT_LINE_INPUT_1_2_VOLUME, 7, 1, 0), +SOC_SINGLE("IN1R ZC Switch", WM8993_RIGHT_LINE_INPUT_1_2_VOLUME, 6, 1, 0),  SOC_SINGLE_TLV("IN2L Volume", WM8993_LEFT_LINE_INPUT_3_4_VOLUME, 0, 31, 0,  	       inpga_tlv),  SOC_SINGLE("IN2L Switch", WM8993_LEFT_LINE_INPUT_3_4_VOLUME, 7, 1, 1), -SOC_SINGLE("IN2L ZC Switch", WM8993_LEFT_LINE_INPUT_3_4_VOLUME, 7, 1, 0), +SOC_SINGLE("IN2L ZC Switch", WM8993_LEFT_LINE_INPUT_3_4_VOLUME, 6, 1, 0),  SOC_SINGLE_TLV("IN2R Volume", WM8993_RIGHT_LINE_INPUT_3_4_VOLUME, 0, 31, 0,  	       inpga_tlv),  SOC_SINGLE("IN2R Switch", WM8993_RIGHT_LINE_INPUT_3_4_VOLUME, 7, 1, 1), -SOC_SINGLE("IN2R ZC Switch", WM8993_RIGHT_LINE_INPUT_3_4_VOLUME, 7, 1, 0), +SOC_SINGLE("IN2R ZC Switch", WM8993_RIGHT_LINE_INPUT_3_4_VOLUME, 6, 1, 0),  SOC_SINGLE_TLV("MIXINL IN2L Volume", WM8993_INPUT_MIXER3, 7, 1, 0,  	       inmix_sw_tlv), @@ -293,24 +473,16 @@ SOC_DOUBLE_R("Speaker Switch",  SOC_DOUBLE_R("Speaker ZC Switch",  	     WM8993_SPEAKER_VOLUME_LEFT, WM8993_SPEAKER_VOLUME_RIGHT,  	     7, 1, 0), -SOC_DOUBLE_TLV("Speaker Boost Volume", WM8993_SPKOUT_BOOST, 0, 3, 7, 0, +SOC_DOUBLE_TLV("Speaker Boost Volume", WM8993_SPKOUT_BOOST, 3, 0, 7, 0,  	       spkboost_tlv),  SOC_ENUM("Speaker Reference", speaker_ref),  SOC_ENUM("Speaker Mode", speaker_mode), -{ -	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = "Headphone Volume", -	.access = SNDRV_CTL_ELEM_ACCESS_TLV_READ | -		 SNDRV_CTL_ELEM_ACCESS_READWRITE, -	.tlv.p = outpga_tlv, -	.info = snd_soc_info_volsw_2r, -	.get = snd_soc_get_volsw_2r, .put = wm8993_put_dc_servo, -	.private_value = (unsigned long)&(struct soc_mixer_control) { -		.reg = WM8993_LEFT_OUTPUT_VOLUME, -		.rreg = WM8993_RIGHT_OUTPUT_VOLUME, -		.shift = 0, .max = 63 -	}, -}, +SOC_DOUBLE_R_EXT_TLV("Headphone Volume", +		     WM8993_LEFT_OUTPUT_VOLUME, WM8993_RIGHT_OUTPUT_VOLUME, +		     0, 63, 0, snd_soc_get_volsw, wm8993_put_dc_servo, +		     outpga_tlv), +  SOC_DOUBLE_R("Headphone Switch", WM8993_LEFT_OUTPUT_VOLUME,  	     WM8993_RIGHT_OUTPUT_VOLUME, 6, 1, 0),  SOC_DOUBLE_R("Headphone ZC Switch", WM8993_LEFT_OUTPUT_VOLUME, @@ -358,6 +530,7 @@ static int hp_supply_event(struct snd_soc_dapm_widget *w,  				hubs->hp_startup_mode);  			break;  		} +		break;  	case SND_SOC_DAPM_PRE_PMD:  		snd_soc_update_bits(codec, WM8993_CHARGE_PUMP_1, @@ -388,11 +561,10 @@ static int hp_event(struct snd_soc_dapm_widget *w,  		reg |= WM8993_HPOUT1L_DLY | WM8993_HPOUT1R_DLY;  		snd_soc_write(codec, WM8993_ANALOGUE_HP_0, reg); -		/* Smallest supported update interval */  		snd_soc_update_bits(codec, WM8993_DC_SERVO_1, -				    WM8993_DCS_TIMER_PERIOD_01_MASK, 1); +				    WM8993_DCS_TIMER_PERIOD_01_MASK, 0); -		calibrate_dc_servo(codec); +		enable_dc_servo(codec);  		reg |= WM8993_HPOUT1R_OUTP | WM8993_HPOUT1R_RMV_SHORT |  			WM8993_HPOUT1L_OUTP | WM8993_HPOUT1L_RMV_SHORT; @@ -439,13 +611,144 @@ static int earpiece_event(struct snd_soc_dapm_widget *w,  		break;  	default: -		BUG(); +		WARN(1, "Invalid event %d\n", event); +		break; +	} + +	return 0; +} + +static int lineout_event(struct snd_soc_dapm_widget *w, +			 struct snd_kcontrol *control, int event) +{ +	struct snd_soc_codec *codec = w->codec; +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	bool *flag; + +	switch (w->shift) { +	case WM8993_LINEOUT1N_ENA_SHIFT: +		flag = &hubs->lineout1n_ena; +		break; +	case WM8993_LINEOUT1P_ENA_SHIFT: +		flag = &hubs->lineout1p_ena; +		break; +	case WM8993_LINEOUT2N_ENA_SHIFT: +		flag = &hubs->lineout2n_ena; +		break; +	case WM8993_LINEOUT2P_ENA_SHIFT: +		flag = &hubs->lineout2p_ena; +		break; +	default: +		WARN(1, "Unknown line output"); +		return -EINVAL; +	} + +	*flag = SND_SOC_DAPM_EVENT_ON(event); + +	return 0; +} + +static int micbias_event(struct snd_soc_dapm_widget *w, +			 struct snd_kcontrol *kcontrol, int event) +{ +	struct snd_soc_codec *codec = w->codec; +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); + +	switch (w->shift) { +	case WM8993_MICB1_ENA_SHIFT: +		if (hubs->micb1_delay) +			msleep(hubs->micb1_delay); +		break; +	case WM8993_MICB2_ENA_SHIFT: +		if (hubs->micb2_delay) +			msleep(hubs->micb2_delay);  		break; +	default: +		return -EINVAL;  	}  	return 0;  } +void wm_hubs_update_class_w(struct snd_soc_codec *codec) +{ +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	int enable = WM8993_CP_DYN_V | WM8993_CP_DYN_FREQ; + +	if (!wm_hubs_dac_hp_direct(codec)) +		enable = false; + +	if (hubs->check_class_w_digital && !hubs->check_class_w_digital(codec)) +		enable = false; + +	dev_vdbg(codec->dev, "Class W %s\n", enable ? "enabled" : "disabled"); + +	snd_soc_update_bits(codec, WM8993_CLASS_W_0, +			    WM8993_CP_DYN_V | WM8993_CP_DYN_FREQ, enable); + +	snd_soc_write(codec, WM8993_LEFT_OUTPUT_VOLUME, +		      snd_soc_read(codec, WM8993_LEFT_OUTPUT_VOLUME)); +	snd_soc_write(codec, WM8993_RIGHT_OUTPUT_VOLUME, +		      snd_soc_read(codec, WM8993_RIGHT_OUTPUT_VOLUME)); +} +EXPORT_SYMBOL_GPL(wm_hubs_update_class_w); + +#define WM_HUBS_SINGLE_W(xname, reg, shift, max, invert) \ +	SOC_SINGLE_EXT(xname, reg, shift, max, invert, \ +		snd_soc_dapm_get_volsw, class_w_put_volsw) + +static int class_w_put_volsw(struct snd_kcontrol *kcontrol, +			      struct snd_ctl_elem_value *ucontrol) +{ +	struct snd_soc_codec *codec = snd_soc_dapm_kcontrol_codec(kcontrol); +	int ret; + +	ret = snd_soc_dapm_put_volsw(kcontrol, ucontrol); + +	wm_hubs_update_class_w(codec); + +	return ret; +} + +#define WM_HUBS_ENUM_W(xname, xenum) \ +{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \ +	.info = snd_soc_info_enum_double, \ +	.get = snd_soc_dapm_get_enum_double, \ +	.put = class_w_put_double, \ +	.private_value = (unsigned long)&xenum } + +static int class_w_put_double(struct snd_kcontrol *kcontrol, +			      struct snd_ctl_elem_value *ucontrol) +{ +	struct snd_soc_codec *codec = snd_soc_dapm_kcontrol_codec(kcontrol); +	int ret; + +	ret = snd_soc_dapm_put_enum_double(kcontrol, ucontrol); + +	wm_hubs_update_class_w(codec); + +	return ret; +} + +static const char *hp_mux_text[] = { +	"Mixer", +	"DAC", +}; + +static SOC_ENUM_SINGLE_DECL(hpl_enum, +			    WM8993_OUTPUT_MIXER1, 8, hp_mux_text); + +const struct snd_kcontrol_new wm_hubs_hpl_mux = +	WM_HUBS_ENUM_W("Left Headphone Mux", hpl_enum); +EXPORT_SYMBOL_GPL(wm_hubs_hpl_mux); + +static SOC_ENUM_SINGLE_DECL(hpr_enum, +			    WM8993_OUTPUT_MIXER2, 8, hp_mux_text); + +const struct snd_kcontrol_new wm_hubs_hpr_mux = +	WM_HUBS_ENUM_W("Right Headphone Mux", hpr_enum); +EXPORT_SYMBOL_GPL(wm_hubs_hpr_mux); +  static const struct snd_kcontrol_new in1l_pga[] = {  SOC_DAPM_SINGLE("IN1LP Switch", WM8993_INPUT_MIXER2, 5, 1, 0),  SOC_DAPM_SINGLE("IN1LN Switch", WM8993_INPUT_MIXER2, 4, 1, 0), @@ -477,25 +780,25 @@ SOC_DAPM_SINGLE("IN1R Switch", WM8993_INPUT_MIXER4, 5, 1, 0),  };  static const struct snd_kcontrol_new left_output_mixer[] = { -SOC_DAPM_SINGLE("Right Input Switch", WM8993_OUTPUT_MIXER1, 7, 1, 0), -SOC_DAPM_SINGLE("Left Input Switch", WM8993_OUTPUT_MIXER1, 6, 1, 0), -SOC_DAPM_SINGLE("IN2RN Switch", WM8993_OUTPUT_MIXER1, 5, 1, 0), -SOC_DAPM_SINGLE("IN2LN Switch", WM8993_OUTPUT_MIXER1, 4, 1, 0), -SOC_DAPM_SINGLE("IN2LP Switch", WM8993_OUTPUT_MIXER1, 1, 1, 0), -SOC_DAPM_SINGLE("IN1R Switch", WM8993_OUTPUT_MIXER1, 3, 1, 0), -SOC_DAPM_SINGLE("IN1L Switch", WM8993_OUTPUT_MIXER1, 2, 1, 0), -SOC_DAPM_SINGLE("DAC Switch", WM8993_OUTPUT_MIXER1, 0, 1, 0), +WM_HUBS_SINGLE_W("Right Input Switch", WM8993_OUTPUT_MIXER1, 7, 1, 0), +WM_HUBS_SINGLE_W("Left Input Switch", WM8993_OUTPUT_MIXER1, 6, 1, 0), +WM_HUBS_SINGLE_W("IN2RN Switch", WM8993_OUTPUT_MIXER1, 5, 1, 0), +WM_HUBS_SINGLE_W("IN2LN Switch", WM8993_OUTPUT_MIXER1, 4, 1, 0), +WM_HUBS_SINGLE_W("IN2LP Switch", WM8993_OUTPUT_MIXER1, 1, 1, 0), +WM_HUBS_SINGLE_W("IN1R Switch", WM8993_OUTPUT_MIXER1, 3, 1, 0), +WM_HUBS_SINGLE_W("IN1L Switch", WM8993_OUTPUT_MIXER1, 2, 1, 0), +WM_HUBS_SINGLE_W("DAC Switch", WM8993_OUTPUT_MIXER1, 0, 1, 0),  };  static const struct snd_kcontrol_new right_output_mixer[] = { -SOC_DAPM_SINGLE("Left Input Switch", WM8993_OUTPUT_MIXER2, 7, 1, 0), -SOC_DAPM_SINGLE("Right Input Switch", WM8993_OUTPUT_MIXER2, 6, 1, 0), -SOC_DAPM_SINGLE("IN2LN Switch", WM8993_OUTPUT_MIXER2, 5, 1, 0), -SOC_DAPM_SINGLE("IN2RN Switch", WM8993_OUTPUT_MIXER2, 4, 1, 0), -SOC_DAPM_SINGLE("IN1L Switch", WM8993_OUTPUT_MIXER2, 3, 1, 0), -SOC_DAPM_SINGLE("IN1R Switch", WM8993_OUTPUT_MIXER2, 2, 1, 0), -SOC_DAPM_SINGLE("IN2RP Switch", WM8993_OUTPUT_MIXER2, 1, 1, 0), -SOC_DAPM_SINGLE("DAC Switch", WM8993_OUTPUT_MIXER2, 0, 1, 0), +WM_HUBS_SINGLE_W("Left Input Switch", WM8993_OUTPUT_MIXER2, 7, 1, 0), +WM_HUBS_SINGLE_W("Right Input Switch", WM8993_OUTPUT_MIXER2, 6, 1, 0), +WM_HUBS_SINGLE_W("IN2LN Switch", WM8993_OUTPUT_MIXER2, 5, 1, 0), +WM_HUBS_SINGLE_W("IN2RN Switch", WM8993_OUTPUT_MIXER2, 4, 1, 0), +WM_HUBS_SINGLE_W("IN1L Switch", WM8993_OUTPUT_MIXER2, 3, 1, 0), +WM_HUBS_SINGLE_W("IN1R Switch", WM8993_OUTPUT_MIXER2, 2, 1, 0), +WM_HUBS_SINGLE_W("IN2RP Switch", WM8993_OUTPUT_MIXER2, 1, 1, 0), +WM_HUBS_SINGLE_W("DAC Switch", WM8993_OUTPUT_MIXER2, 0, 1, 0),  };  static const struct snd_kcontrol_new earpiece_mixer[] = { @@ -532,14 +835,14 @@ SOC_DAPM_SINGLE("Left Output Switch", WM8993_LINE_MIXER1, 0, 1, 0),  };  static const struct snd_kcontrol_new line2_mix[] = { -SOC_DAPM_SINGLE("IN2R Switch", WM8993_LINE_MIXER2, 2, 1, 0), -SOC_DAPM_SINGLE("IN2L Switch", WM8993_LINE_MIXER2, 1, 1, 0), +SOC_DAPM_SINGLE("IN1L Switch", WM8993_LINE_MIXER2, 2, 1, 0), +SOC_DAPM_SINGLE("IN1R Switch", WM8993_LINE_MIXER2, 1, 1, 0),  SOC_DAPM_SINGLE("Output Switch", WM8993_LINE_MIXER2, 0, 1, 0),  };  static const struct snd_kcontrol_new line2n_mix[] = { -SOC_DAPM_SINGLE("Left Output Switch", WM8993_LINE_MIXER2, 6, 1, 0), -SOC_DAPM_SINGLE("Right Output Switch", WM8993_LINE_MIXER2, 5, 1, 0), +SOC_DAPM_SINGLE("Left Output Switch", WM8993_LINE_MIXER2, 5, 1, 0), +SOC_DAPM_SINGLE("Right Output Switch", WM8993_LINE_MIXER2, 6, 1, 0),  };  static const struct snd_kcontrol_new line2p_mix[] = { @@ -556,8 +859,10 @@ SND_SOC_DAPM_INPUT("IN1RP"),  SND_SOC_DAPM_INPUT("IN2RN"),  SND_SOC_DAPM_INPUT("IN2RP:VXRP"), -SND_SOC_DAPM_MICBIAS("MICBIAS2", WM8993_POWER_MANAGEMENT_1, 5, 0), -SND_SOC_DAPM_MICBIAS("MICBIAS1", WM8993_POWER_MANAGEMENT_1, 4, 0), +SND_SOC_DAPM_SUPPLY("MICBIAS2", WM8993_POWER_MANAGEMENT_1, 5, 0, +		    micbias_event, SND_SOC_DAPM_POST_PMU), +SND_SOC_DAPM_SUPPLY("MICBIAS1", WM8993_POWER_MANAGEMENT_1, 4, 0, +		    micbias_event, SND_SOC_DAPM_POST_PMU),  SND_SOC_DAPM_MIXER("IN1L PGA", WM8993_POWER_MANAGEMENT_2, 6, 0,  		   in1l_pga, ARRAY_SIZE(in1l_pga)), @@ -569,9 +874,6 @@ 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, @@ -587,9 +889,8 @@ SND_SOC_DAPM_PGA("Right Output PGA", WM8993_POWER_MANAGEMENT_3, 6, 0, NULL, 0),  SND_SOC_DAPM_SUPPLY("Headphone Supply", SND_SOC_NOPM, 0, 0, hp_supply_event,   		    SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_PRE_PMD), -SND_SOC_DAPM_PGA_E("Headphone PGA", SND_SOC_NOPM, 0, 0, -		   NULL, 0, -		   hp_event, SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), +SND_SOC_DAPM_OUT_DRV_E("Headphone PGA", SND_SOC_NOPM, 0, 0, NULL, 0, +		       hp_event, SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),  SND_SOC_DAPM_MIXER("Earpiece Mixer", SND_SOC_NOPM, 0, 0,  		   earpiece_mixer, ARRAY_SIZE(earpiece_mixer)), @@ -602,10 +903,11 @@ SND_SOC_DAPM_MIXER("SPKL Boost", SND_SOC_NOPM, 0, 0,  SND_SOC_DAPM_MIXER("SPKR Boost", SND_SOC_NOPM, 0, 0,  		   right_speaker_boost, ARRAY_SIZE(right_speaker_boost)), -SND_SOC_DAPM_PGA("SPKL Driver", WM8993_POWER_MANAGEMENT_1, 12, 0, -		 NULL, 0), -SND_SOC_DAPM_PGA("SPKR Driver", WM8993_POWER_MANAGEMENT_1, 13, 0, -		 NULL, 0), +SND_SOC_DAPM_SUPPLY("TSHUT", WM8993_POWER_MANAGEMENT_2, 14, 0, NULL, 0), +SND_SOC_DAPM_OUT_DRV("SPKL Driver", WM8993_POWER_MANAGEMENT_1, 12, 0, +		     NULL, 0), +SND_SOC_DAPM_OUT_DRV("SPKR Driver", WM8993_POWER_MANAGEMENT_1, 13, 0, +		     NULL, 0),  SND_SOC_DAPM_MIXER("LINEOUT1 Mixer", SND_SOC_NOPM, 0, 0,  		   line1_mix, ARRAY_SIZE(line1_mix)), @@ -621,14 +923,18 @@ SND_SOC_DAPM_MIXER("LINEOUT2N Mixer", SND_SOC_NOPM, 0, 0,  SND_SOC_DAPM_MIXER("LINEOUT2P Mixer", SND_SOC_NOPM, 0, 0,  		   line2p_mix, ARRAY_SIZE(line2p_mix)), -SND_SOC_DAPM_PGA("LINEOUT1N Driver", WM8993_POWER_MANAGEMENT_3, 13, 0, -		 NULL, 0), -SND_SOC_DAPM_PGA("LINEOUT1P Driver", WM8993_POWER_MANAGEMENT_3, 12, 0, -		 NULL, 0), -SND_SOC_DAPM_PGA("LINEOUT2N Driver", WM8993_POWER_MANAGEMENT_3, 11, 0, -		 NULL, 0), -SND_SOC_DAPM_PGA("LINEOUT2P Driver", WM8993_POWER_MANAGEMENT_3, 10, 0, -		 NULL, 0), +SND_SOC_DAPM_OUT_DRV_E("LINEOUT1N Driver", WM8993_POWER_MANAGEMENT_3, 13, 0, +		       NULL, 0, lineout_event, +		     SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), +SND_SOC_DAPM_OUT_DRV_E("LINEOUT1P Driver", WM8993_POWER_MANAGEMENT_3, 12, 0, +		       NULL, 0, lineout_event, +		       SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), +SND_SOC_DAPM_OUT_DRV_E("LINEOUT2N Driver", WM8993_POWER_MANAGEMENT_3, 11, 0, +		       NULL, 0, lineout_event, +		       SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), +SND_SOC_DAPM_OUT_DRV_E("LINEOUT2P Driver", WM8993_POWER_MANAGEMENT_3, 10, 0, +		       NULL, 0, lineout_event, +		       SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),  SND_SOC_DAPM_OUTPUT("SPKOUTLP"),  SND_SOC_DAPM_OUTPUT("SPKOUTLN"), @@ -645,9 +951,17 @@ SND_SOC_DAPM_OUTPUT("LINEOUT2N"),  };  static const struct snd_soc_dapm_route analogue_routes[] = { +	{ "MICBIAS1", NULL, "CLK_SYS" }, +	{ "MICBIAS2", NULL, "CLK_SYS" }, +  	{ "IN1L PGA", "IN1LP Switch", "IN1LP" },  	{ "IN1L PGA", "IN1LN Switch", "IN1LN" }, +	{ "IN1L PGA", NULL, "VMID" }, +	{ "IN1R PGA", NULL, "VMID" }, +	{ "IN2L PGA", NULL, "VMID" }, +	{ "IN2R PGA", NULL, "VMID" }, +  	{ "IN1R PGA", "IN1RP Switch", "IN1RP" },  	{ "IN1R PGA", "IN1RN Switch", "IN1RN" }, @@ -665,12 +979,14 @@ static const struct snd_soc_dapm_route analogue_routes[] = {  	{ "MIXINL", NULL, "Direct Voice" },  	{ "MIXINL", NULL, "IN1LP" },  	{ "MIXINL", NULL, "Left Output Mixer" }, +	{ "MIXINL", NULL, "VMID" },  	{ "MIXINR", "IN1R Switch", "IN1R PGA" },  	{ "MIXINR", "IN2R Switch", "IN2R PGA" },  	{ "MIXINR", NULL, "Direct Voice" },  	{ "MIXINR", NULL, "IN1RP" },  	{ "MIXINR", NULL, "Right Output Mixer" }, +	{ "MIXINR", NULL, "VMID" },  	{ "ADCL", NULL, "MIXINL" },  	{ "ADCR", NULL, "MIXINR" }, @@ -701,18 +1017,19 @@ static const struct snd_soc_dapm_route analogue_routes[] = {  	{ "Earpiece Mixer", "Left Output Switch", "Left Output PGA" },  	{ "Earpiece Mixer", "Right Output Switch", "Right Output PGA" }, +	{ "Earpiece Driver", NULL, "VMID" },  	{ "Earpiece Driver", NULL, "Earpiece Mixer" },  	{ "HPOUT2N", NULL, "Earpiece Driver" },  	{ "HPOUT2P", NULL, "Earpiece Driver" },  	{ "SPKL", "Input Switch", "MIXINL" },  	{ "SPKL", "IN1LP Switch", "IN1LP" }, -	{ "SPKL", "Output Switch", "Left Output Mixer" }, +	{ "SPKL", "Output Switch", "Left Output PGA" },  	{ "SPKL", NULL, "TOCLK" },  	{ "SPKR", "Input Switch", "MIXINR" },  	{ "SPKR", "IN1RP Switch", "IN1RP" }, -	{ "SPKR", "Output Switch", "Right Output Mixer" }, +	{ "SPKR", "Output Switch", "Right Output PGA" },  	{ "SPKR", NULL, "TOCLK" },  	{ "SPKL Boost", "Direct Voice Switch", "Direct Voice" }, @@ -723,28 +1040,38 @@ static const struct snd_soc_dapm_route analogue_routes[] = {  	{ "SPKR Boost", "SPKR Switch", "SPKR" },  	{ "SPKR Boost", "SPKL Switch", "SPKL" }, +	{ "SPKL Driver", NULL, "VMID" },  	{ "SPKL Driver", NULL, "SPKL Boost" },  	{ "SPKL Driver", NULL, "CLK_SYS" }, +	{ "SPKL Driver", NULL, "TSHUT" }, +	{ "SPKR Driver", NULL, "VMID" },  	{ "SPKR Driver", NULL, "SPKR Boost" },  	{ "SPKR Driver", NULL, "CLK_SYS" }, +	{ "SPKR Driver", NULL, "TSHUT" },  	{ "SPKOUTLP", NULL, "SPKL Driver" },  	{ "SPKOUTLN", NULL, "SPKL Driver" },  	{ "SPKOUTRP", NULL, "SPKR Driver" },  	{ "SPKOUTRN", NULL, "SPKR Driver" }, -	{ "Left Headphone Mux", "Mixer", "Left Output Mixer" }, -	{ "Right Headphone Mux", "Mixer", "Right Output Mixer" }, +	{ "Left Headphone Mux", "Mixer", "Left Output PGA" }, +	{ "Right Headphone Mux", "Mixer", "Right Output PGA" },  	{ "Headphone PGA", NULL, "Left Headphone Mux" },  	{ "Headphone PGA", NULL, "Right Headphone Mux" }, +	{ "Headphone PGA", NULL, "VMID" },  	{ "Headphone PGA", NULL, "CLK_SYS" },  	{ "Headphone PGA", NULL, "Headphone Supply" },  	{ "HPOUT1L", NULL, "Headphone PGA" },  	{ "HPOUT1R", NULL, "Headphone PGA" }, +	{ "LINEOUT1N Driver", NULL, "VMID" }, +	{ "LINEOUT1P Driver", NULL, "VMID" }, +	{ "LINEOUT2N Driver", NULL, "VMID" }, +	{ "LINEOUT2P Driver", NULL, "VMID" }, +  	{ "LINEOUT1N", NULL, "LINEOUT1N Driver" },  	{ "LINEOUT1P", NULL, "LINEOUT1P Driver" },  	{ "LINEOUT2N", NULL, "LINEOUT2N Driver" }, @@ -754,36 +1081,36 @@ static const struct snd_soc_dapm_route analogue_routes[] = {  static const struct snd_soc_dapm_route lineout1_diff_routes[] = {  	{ "LINEOUT1 Mixer", "IN1L Switch", "IN1L PGA" },  	{ "LINEOUT1 Mixer", "IN1R Switch", "IN1R PGA" }, -	{ "LINEOUT1 Mixer", "Output Switch", "Left Output Mixer" }, +	{ "LINEOUT1 Mixer", "Output Switch", "Left Output PGA" },  	{ "LINEOUT1N Driver", NULL, "LINEOUT1 Mixer" },  	{ "LINEOUT1P Driver", NULL, "LINEOUT1 Mixer" },  };  static const struct snd_soc_dapm_route lineout1_se_routes[] = { -	{ "LINEOUT1N Mixer", "Left Output Switch", "Left Output Mixer" }, -	{ "LINEOUT1N Mixer", "Right Output Switch", "Left Output Mixer" }, +	{ "LINEOUT1N Mixer", "Left Output Switch", "Left Output PGA" }, +	{ "LINEOUT1N Mixer", "Right Output Switch", "Right Output PGA" }, -	{ "LINEOUT1P Mixer", "Left Output Switch", "Left Output Mixer" }, +	{ "LINEOUT1P Mixer", "Left Output Switch", "Left Output PGA" },  	{ "LINEOUT1N Driver", NULL, "LINEOUT1N Mixer" },  	{ "LINEOUT1P Driver", NULL, "LINEOUT1P Mixer" },  };  static const struct snd_soc_dapm_route lineout2_diff_routes[] = { -	{ "LINEOUT2 Mixer", "IN2L Switch", "IN2L PGA" }, -	{ "LINEOUT2 Mixer", "IN2R Switch", "IN2R PGA" }, -	{ "LINEOUT2 Mixer", "Output Switch", "Right Output Mixer" }, +	{ "LINEOUT2 Mixer", "IN1L Switch", "IN1L PGA" }, +	{ "LINEOUT2 Mixer", "IN1R Switch", "IN1R PGA" }, +	{ "LINEOUT2 Mixer", "Output Switch", "Right Output PGA" },  	{ "LINEOUT2N Driver", NULL, "LINEOUT2 Mixer" },  	{ "LINEOUT2P Driver", NULL, "LINEOUT2 Mixer" },  };  static const struct snd_soc_dapm_route lineout2_se_routes[] = { -	{ "LINEOUT2N Mixer", "Left Output Switch", "Left Output Mixer" }, -	{ "LINEOUT2N Mixer", "Right Output Switch", "Left Output Mixer" }, +	{ "LINEOUT2N Mixer", "Left Output Switch", "Left Output PGA" }, +	{ "LINEOUT2N Mixer", "Right Output Switch", "Right Output PGA" }, -	{ "LINEOUT2P Mixer", "Right Output Switch", "Right Output Mixer" }, +	{ "LINEOUT2P Mixer", "Right Output Switch", "Right Output PGA" },  	{ "LINEOUT2N Driver", NULL, "LINEOUT2N Mixer" },  	{ "LINEOUT2P Driver", NULL, "LINEOUT2P Mixer" }, @@ -791,6 +1118,8 @@ static const struct snd_soc_dapm_route lineout2_se_routes[] = {  int wm_hubs_add_analogue_controls(struct snd_soc_codec *codec)  { +	struct snd_soc_dapm_context *dapm = &codec->dapm; +  	/* Latch volume update bits & default ZC on */  	snd_soc_update_bits(codec, WM8993_LEFT_LINE_INPUT_1_2_VOLUME,  			    WM8993_IN1_VU, WM8993_IN1_VU); @@ -801,25 +1130,29 @@ int wm_hubs_add_analogue_controls(struct snd_soc_codec *codec)  	snd_soc_update_bits(codec, WM8993_RIGHT_LINE_INPUT_3_4_VOLUME,  			    WM8993_IN2_VU, WM8993_IN2_VU); +	snd_soc_update_bits(codec, WM8993_SPEAKER_VOLUME_LEFT, +			    WM8993_SPKOUT_VU, WM8993_SPKOUT_VU);  	snd_soc_update_bits(codec, WM8993_SPEAKER_VOLUME_RIGHT,  			    WM8993_SPKOUT_VU, WM8993_SPKOUT_VU);  	snd_soc_update_bits(codec, WM8993_LEFT_OUTPUT_VOLUME, -			    WM8993_HPOUT1L_ZC, WM8993_HPOUT1L_ZC); +			    WM8993_HPOUT1_VU | WM8993_HPOUT1L_ZC, +			    WM8993_HPOUT1_VU | WM8993_HPOUT1L_ZC);  	snd_soc_update_bits(codec, WM8993_RIGHT_OUTPUT_VOLUME,  			    WM8993_HPOUT1_VU | WM8993_HPOUT1R_ZC,  			    WM8993_HPOUT1_VU | WM8993_HPOUT1R_ZC);  	snd_soc_update_bits(codec, WM8993_LEFT_OPGA_VOLUME, -			    WM8993_MIXOUTL_ZC, WM8993_MIXOUTL_ZC); +			    WM8993_MIXOUTL_ZC | WM8993_MIXOUT_VU, +			    WM8993_MIXOUTL_ZC | WM8993_MIXOUT_VU);  	snd_soc_update_bits(codec, WM8993_RIGHT_OPGA_VOLUME,  			    WM8993_MIXOUTR_ZC | WM8993_MIXOUT_VU,  			    WM8993_MIXOUTR_ZC | WM8993_MIXOUT_VU); -	snd_soc_add_controls(codec, analogue_snd_controls, +	snd_soc_add_codec_controls(codec, analogue_snd_controls,  			     ARRAY_SIZE(analogue_snd_controls)); -	snd_soc_dapm_new_controls(codec, analogue_dapm_widgets, +	snd_soc_dapm_new_controls(dapm, analogue_dapm_widgets,  				  ARRAY_SIZE(analogue_dapm_widgets));  	return 0;  } @@ -828,24 +1161,32 @@ 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)  { -	snd_soc_dapm_add_routes(codec, analogue_routes, +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	struct snd_soc_dapm_context *dapm = &codec->dapm; + +	hubs->codec = codec; + +	INIT_LIST_HEAD(&hubs->dcs_cache); +	init_completion(&hubs->dcs_done); + +	snd_soc_dapm_add_routes(dapm, analogue_routes,  				ARRAY_SIZE(analogue_routes));  	if (lineout1_diff) -		snd_soc_dapm_add_routes(codec, +		snd_soc_dapm_add_routes(dapm,  					lineout1_diff_routes,  					ARRAY_SIZE(lineout1_diff_routes));  	else -		snd_soc_dapm_add_routes(codec, +		snd_soc_dapm_add_routes(dapm,  					lineout1_se_routes,  					ARRAY_SIZE(lineout1_se_routes));  	if (lineout2_diff) -		snd_soc_dapm_add_routes(codec, +		snd_soc_dapm_add_routes(dapm,  					lineout2_diff_routes,  					ARRAY_SIZE(lineout2_diff_routes));  	else -		snd_soc_dapm_add_routes(codec, +		snd_soc_dapm_add_routes(dapm,  					lineout2_se_routes,  					ARRAY_SIZE(lineout2_se_routes)); @@ -856,9 +1197,17 @@ EXPORT_SYMBOL_GPL(wm_hubs_add_analogue_routes);  int wm_hubs_handle_analogue_pdata(struct snd_soc_codec *codec,  				  int lineout1_diff, int lineout2_diff,  				  int lineout1fb, int lineout2fb, -				  int jd_scthr, int jd_thr, int micbias1_lvl, -				  int micbias2_lvl) +				  int jd_scthr, int jd_thr, +				  int micbias1_delay, int micbias2_delay, +				  int micbias1_lvl, int micbias2_lvl)  { +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); + +	hubs->lineout1_se = !lineout1_diff; +	hubs->lineout2_se = !lineout2_diff; +	hubs->micb1_delay = micbias1_delay; +	hubs->micb2_delay = micbias2_delay; +  	if (!lineout1_diff)  		snd_soc_update_bits(codec, WM8993_LINE_MIXER1,  				    WM8993_LINEOUT1_MODE, @@ -868,11 +1217,10 @@ int wm_hubs_handle_analogue_pdata(struct snd_soc_codec *codec,  				    WM8993_LINEOUT2_MODE,  				    WM8993_LINEOUT2_MODE); -	/* If the line outputs are differential then we aren't presenting -	 * VMID as an output and can disable it. -	 */ -	if (lineout1_diff && lineout2_diff) -		codec->idle_bias_off = 1; +	if (!lineout1_diff && !lineout2_diff) +		snd_soc_update_bits(codec, WM8993_ANTIPOP1, +				    WM8993_LINEOUT_VMID_BUF_ENA, +				    WM8993_LINEOUT_VMID_BUF_ENA);  	if (lineout1fb)  		snd_soc_update_bits(codec, WM8993_ADDITIONAL_CONTROL, @@ -894,6 +1242,72 @@ int wm_hubs_handle_analogue_pdata(struct snd_soc_codec *codec,  }  EXPORT_SYMBOL_GPL(wm_hubs_handle_analogue_pdata); +void wm_hubs_vmid_ena(struct snd_soc_codec *codec) +{ +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	int val = 0; + +	if (hubs->lineout1_se) +		val |= WM8993_LINEOUT1N_ENA | WM8993_LINEOUT1P_ENA; + +	if (hubs->lineout2_se) +		val |= WM8993_LINEOUT2N_ENA | WM8993_LINEOUT2P_ENA; + +	/* Enable the line outputs while we power up */ +	snd_soc_update_bits(codec, WM8993_POWER_MANAGEMENT_3, val, val); +} +EXPORT_SYMBOL_GPL(wm_hubs_vmid_ena); + +void wm_hubs_set_bias_level(struct snd_soc_codec *codec, +			    enum snd_soc_bias_level level) +{ +	struct wm_hubs_data *hubs = snd_soc_codec_get_drvdata(codec); +	int mask, val; + +	switch (level) { +	case SND_SOC_BIAS_STANDBY: +		/* Clamp the inputs to VMID while we ramp to charge caps */ +		snd_soc_update_bits(codec, WM8993_INPUTS_CLAMP_REG, +				    WM8993_INPUTS_CLAMP, WM8993_INPUTS_CLAMP); +		break; + +	case SND_SOC_BIAS_ON: +		/* Turn off any unneded single ended outputs */ +		val = 0; +		mask = 0; + +		if (hubs->lineout1_se) +			mask |= WM8993_LINEOUT1N_ENA | WM8993_LINEOUT1P_ENA; + +		if (hubs->lineout2_se) +			mask |= WM8993_LINEOUT2N_ENA | WM8993_LINEOUT2P_ENA; + +		if (hubs->lineout1_se && hubs->lineout1n_ena) +			val |= WM8993_LINEOUT1N_ENA; + +		if (hubs->lineout1_se && hubs->lineout1p_ena) +			val |= WM8993_LINEOUT1P_ENA; + +		if (hubs->lineout2_se && hubs->lineout2n_ena) +			val |= WM8993_LINEOUT2N_ENA; + +		if (hubs->lineout2_se && hubs->lineout2p_ena) +			val |= WM8993_LINEOUT2P_ENA; + +		snd_soc_update_bits(codec, WM8993_POWER_MANAGEMENT_3, +				    mask, val); + +		/* Remove the input clamps */ +		snd_soc_update_bits(codec, WM8993_INPUTS_CLAMP_REG, +				    WM8993_INPUTS_CLAMP, 0); +		break; + +	default: +		break; +	} +} +EXPORT_SYMBOL_GPL(wm_hubs_set_bias_level); +  MODULE_DESCRIPTION("Shared support for Wolfson hubs products");  MODULE_AUTHOR("Mark Brown <broonie@opensource.wolfsonmicro.com>");  MODULE_LICENSE("GPL");  | 
