Linux Audio

Check our new training course

Loading...
v3.1
 
   1/*
   2 * HD audio interface patch for Conexant HDA audio codec
   3 *
   4 * Copyright (c) 2006 Pototskiy Akex <alex.pototskiy@gmail.com>
   5 * 		      Takashi Iwai <tiwai@suse.de>
   6 * 		      Tobin Davis  <tdavis@dsl-only.net>
   7 *
   8 *  This driver is free software; you can redistribute it and/or modify
   9 *  it under the terms of the GNU General Public License as published by
  10 *  the Free Software Foundation; either version 2 of the License, or
  11 *  (at your option) any later version.
  12 *
  13 *  This driver is distributed in the hope that it will be useful,
  14 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
  15 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  16 *  GNU General Public License for more details.
  17 *
  18 *  You should have received a copy of the GNU General Public License
  19 *  along with this program; if not, write to the Free Software
  20 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
  21 */
  22
  23#include <linux/init.h>
  24#include <linux/delay.h>
  25#include <linux/slab.h>
  26#include <linux/pci.h>
  27#include <sound/core.h>
  28#include <sound/jack.h>
  29
  30#include "hda_codec.h"
  31#include "hda_local.h"
 
  32#include "hda_beep.h"
 
 
  33
  34#define CXT_PIN_DIR_IN              0x00
  35#define CXT_PIN_DIR_OUT             0x01
  36#define CXT_PIN_DIR_INOUT           0x02
  37#define CXT_PIN_DIR_IN_NOMICBIAS    0x03
  38#define CXT_PIN_DIR_INOUT_NOMICBIAS 0x04
  39
  40#define CONEXANT_HP_EVENT	0x37
  41#define CONEXANT_MIC_EVENT	0x38
  42#define CONEXANT_LINE_EVENT	0x39
  43
  44/* Conexant 5051 specific */
  45
  46#define CXT5051_SPDIF_OUT	0x12
  47#define CXT5051_PORTB_EVENT	0x38
  48#define CXT5051_PORTC_EVENT	0x39
  49
  50#define AUTO_MIC_PORTB		(1 << 1)
  51#define AUTO_MIC_PORTC		(1 << 2)
  52
  53struct pin_dac_pair {
  54	hda_nid_t pin;
  55	hda_nid_t dac;
  56	int type;
  57};
  58
  59struct imux_info {
  60	hda_nid_t pin;		/* input pin NID */
  61	hda_nid_t adc;		/* connected ADC NID */	
  62	hda_nid_t boost;	/* optional boost volume NID */
  63	int index;		/* corresponding to autocfg.input */
  64};
  65
  66struct conexant_spec {
  67
  68	const struct snd_kcontrol_new *mixers[5];
  69	int num_mixers;
  70	hda_nid_t vmaster_nid;
  71
  72	const struct hda_verb *init_verbs[5];	/* initialization verbs
  73						 * don't forget NULL
  74						 * termination!
  75						 */
  76	unsigned int num_init_verbs;
  77
  78	/* playback */
  79	struct hda_multi_out multiout;	/* playback set-up
  80					 * max_channels, dacs must be set
  81					 * dig_out_nid and hp_nid are optional
  82					 */
  83	unsigned int cur_eapd;
  84	unsigned int hp_present;
  85	unsigned int line_present;
  86	unsigned int auto_mic;
  87	int auto_mic_ext;		/* imux_pins[] index for ext mic */
  88	int auto_mic_dock;		/* imux_pins[] index for dock mic */
  89	int auto_mic_int;		/* imux_pins[] index for int mic */
  90	unsigned int need_dac_fix;
  91	hda_nid_t slave_dig_outs[2];
  92
  93	/* capture */
  94	unsigned int num_adc_nids;
  95	const hda_nid_t *adc_nids;
  96	hda_nid_t dig_in_nid;		/* digital-in NID; optional */
  97
  98	unsigned int cur_adc_idx;
  99	hda_nid_t cur_adc;
 100	unsigned int cur_adc_stream_tag;
 101	unsigned int cur_adc_format;
 102
 103	const struct hda_pcm_stream *capture_stream;
 104
 105	/* capture source */
 106	const struct hda_input_mux *input_mux;
 107	const hda_nid_t *capsrc_nids;
 108	unsigned int cur_mux[3];
 109
 110	/* channel model */
 111	const struct hda_channel_mode *channel_mode;
 112	int num_channel_mode;
 113
 114	/* PCM information */
 115	struct hda_pcm pcm_rec[2];	/* used in build_pcms() */
 116
 117	unsigned int spdif_route;
 118
 119	/* dynamic controls, init_verbs and input_mux */
 120	struct auto_pin_cfg autocfg;
 121	struct hda_input_mux private_imux;
 122	struct imux_info imux_info[HDA_MAX_NUM_INPUTS];
 123	hda_nid_t private_adc_nids[HDA_MAX_NUM_INPUTS];
 124	hda_nid_t private_dac_nids[AUTO_CFG_MAX_OUTS];
 125	struct pin_dac_pair dac_info[8];
 126	int dac_info_filled;
 127
 128	unsigned int port_d_mode;
 129	unsigned int auto_mute:1;	/* used in auto-parser */
 130	unsigned int detect_line:1;	/* Line-out detection enabled */
 131	unsigned int automute_lines:1;	/* automute line-out as well */
 132	unsigned int automute_hp_lo:1;	/* both HP and LO available */
 133	unsigned int dell_automute:1;
 134	unsigned int dell_vostro:1;
 135	unsigned int ideapad:1;
 136	unsigned int thinkpad:1;
 137	unsigned int hp_laptop:1;
 138	unsigned int asus:1;
 139
 140	unsigned int adc_switching:1;
 141
 142	unsigned int ext_mic_present;
 143	unsigned int recording;
 144	void (*capture_prepare)(struct hda_codec *codec);
 145	void (*capture_cleanup)(struct hda_codec *codec);
 146
 147	/* OLPC XO-1.5 supports DC input mode (e.g. for use with analog sensors)
 148	 * through the microphone jack.
 149	 * When the user enables this through a mixer switch, both internal and
 150	 * external microphones are disabled. Gain is fixed at 0dB. In this mode,
 151	 * we also allow the bias to be configured through a separate mixer
 152	 * control. */
 153	unsigned int dc_enable;
 154	unsigned int dc_input_bias; /* offset into cxt5066_olpc_dc_bias */
 155	unsigned int mic_boost; /* offset into cxt5066_analog_mic_boost */
 156
 157	unsigned int beep_amp;
 158
 159	/* extra EAPD pins */
 160	unsigned int num_eapds;
 161	hda_nid_t eapds[4];
 162};
 163
 164static int conexant_playback_pcm_open(struct hda_pcm_stream *hinfo,
 165				      struct hda_codec *codec,
 166				      struct snd_pcm_substream *substream)
 167{
 168	struct conexant_spec *spec = codec->spec;
 169	return snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
 170					     hinfo);
 171}
 172
 173static int conexant_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
 174					 struct hda_codec *codec,
 175					 unsigned int stream_tag,
 176					 unsigned int format,
 177					 struct snd_pcm_substream *substream)
 178{
 179	struct conexant_spec *spec = codec->spec;
 180	return snd_hda_multi_out_analog_prepare(codec, &spec->multiout,
 181						stream_tag,
 182						format, substream);
 183}
 184
 185static int conexant_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
 186					 struct hda_codec *codec,
 187					 struct snd_pcm_substream *substream)
 188{
 189	struct conexant_spec *spec = codec->spec;
 190	return snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
 191}
 192
 193/*
 194 * Digital out
 195 */
 196static int conexant_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
 197					  struct hda_codec *codec,
 198					  struct snd_pcm_substream *substream)
 199{
 200	struct conexant_spec *spec = codec->spec;
 201	return snd_hda_multi_out_dig_open(codec, &spec->multiout);
 202}
 203
 204static int conexant_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
 205					 struct hda_codec *codec,
 206					 struct snd_pcm_substream *substream)
 207{
 208	struct conexant_spec *spec = codec->spec;
 209	return snd_hda_multi_out_dig_close(codec, &spec->multiout);
 210}
 211
 212static int conexant_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
 213					 struct hda_codec *codec,
 214					 unsigned int stream_tag,
 215					 unsigned int format,
 216					 struct snd_pcm_substream *substream)
 217{
 218	struct conexant_spec *spec = codec->spec;
 219	return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
 220					     stream_tag,
 221					     format, substream);
 222}
 223
 224/*
 225 * Analog capture
 226 */
 227static int conexant_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
 228				      struct hda_codec *codec,
 229				      unsigned int stream_tag,
 230				      unsigned int format,
 231				      struct snd_pcm_substream *substream)
 232{
 233	struct conexant_spec *spec = codec->spec;
 234	if (spec->capture_prepare)
 235		spec->capture_prepare(codec);
 236	snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number],
 237				   stream_tag, 0, format);
 238	return 0;
 239}
 240
 241static int conexant_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
 242				      struct hda_codec *codec,
 243				      struct snd_pcm_substream *substream)
 244{
 245	struct conexant_spec *spec = codec->spec;
 246	snd_hda_codec_cleanup_stream(codec, spec->adc_nids[substream->number]);
 247	if (spec->capture_cleanup)
 248		spec->capture_cleanup(codec);
 249	return 0;
 250}
 251
 252
 253
 254static const struct hda_pcm_stream conexant_pcm_analog_playback = {
 255	.substreams = 1,
 256	.channels_min = 2,
 257	.channels_max = 2,
 258	.nid = 0, /* fill later */
 259	.ops = {
 260		.open = conexant_playback_pcm_open,
 261		.prepare = conexant_playback_pcm_prepare,
 262		.cleanup = conexant_playback_pcm_cleanup
 263	},
 264};
 265
 266static const struct hda_pcm_stream conexant_pcm_analog_capture = {
 267	.substreams = 1,
 268	.channels_min = 2,
 269	.channels_max = 2,
 270	.nid = 0, /* fill later */
 271	.ops = {
 272		.prepare = conexant_capture_pcm_prepare,
 273		.cleanup = conexant_capture_pcm_cleanup
 274	},
 275};
 276
 277
 278static const struct hda_pcm_stream conexant_pcm_digital_playback = {
 279	.substreams = 1,
 280	.channels_min = 2,
 281	.channels_max = 2,
 282	.nid = 0, /* fill later */
 283	.ops = {
 284		.open = conexant_dig_playback_pcm_open,
 285		.close = conexant_dig_playback_pcm_close,
 286		.prepare = conexant_dig_playback_pcm_prepare
 287	},
 288};
 289
 290static const struct hda_pcm_stream conexant_pcm_digital_capture = {
 291	.substreams = 1,
 292	.channels_min = 2,
 293	.channels_max = 2,
 294	/* NID is set in alc_build_pcms */
 295};
 296
 297static int cx5051_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
 298				      struct hda_codec *codec,
 299				      unsigned int stream_tag,
 300				      unsigned int format,
 301				      struct snd_pcm_substream *substream)
 302{
 303	struct conexant_spec *spec = codec->spec;
 304	spec->cur_adc = spec->adc_nids[spec->cur_adc_idx];
 305	spec->cur_adc_stream_tag = stream_tag;
 306	spec->cur_adc_format = format;
 307	snd_hda_codec_setup_stream(codec, spec->cur_adc, stream_tag, 0, format);
 308	return 0;
 309}
 310
 311static int cx5051_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
 312				      struct hda_codec *codec,
 313				      struct snd_pcm_substream *substream)
 314{
 315	struct conexant_spec *spec = codec->spec;
 316	snd_hda_codec_cleanup_stream(codec, spec->cur_adc);
 317	spec->cur_adc = 0;
 318	return 0;
 319}
 320
 321static const struct hda_pcm_stream cx5051_pcm_analog_capture = {
 322	.substreams = 1,
 323	.channels_min = 2,
 324	.channels_max = 2,
 325	.nid = 0, /* fill later */
 326	.ops = {
 327		.prepare = cx5051_capture_pcm_prepare,
 328		.cleanup = cx5051_capture_pcm_cleanup
 329	},
 330};
 331
 332static int conexant_build_pcms(struct hda_codec *codec)
 333{
 334	struct conexant_spec *spec = codec->spec;
 335	struct hda_pcm *info = spec->pcm_rec;
 336
 337	codec->num_pcms = 1;
 338	codec->pcm_info = info;
 339
 340	info->name = "CONEXANT Analog";
 341	info->stream[SNDRV_PCM_STREAM_PLAYBACK] = conexant_pcm_analog_playback;
 342	info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max =
 343		spec->multiout.max_channels;
 344	info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
 345		spec->multiout.dac_nids[0];
 346	if (spec->capture_stream)
 347		info->stream[SNDRV_PCM_STREAM_CAPTURE] = *spec->capture_stream;
 348	else {
 349		if (codec->vendor_id == 0x14f15051)
 350			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
 351				cx5051_pcm_analog_capture;
 352		else {
 353			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
 354				conexant_pcm_analog_capture;
 355			info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams =
 356				spec->num_adc_nids;
 357		}
 358	}
 359	info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->adc_nids[0];
 360
 361	if (spec->multiout.dig_out_nid) {
 362		info++;
 363		codec->num_pcms++;
 364		info->name = "Conexant Digital";
 365		info->pcm_type = HDA_PCM_TYPE_SPDIF;
 366		info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
 367			conexant_pcm_digital_playback;
 368		info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
 369			spec->multiout.dig_out_nid;
 370		if (spec->dig_in_nid) {
 371			info->stream[SNDRV_PCM_STREAM_CAPTURE] =
 372				conexant_pcm_digital_capture;
 373			info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
 374				spec->dig_in_nid;
 375		}
 376		if (spec->slave_dig_outs[0])
 377			codec->slave_dig_outs = spec->slave_dig_outs;
 378	}
 379
 380	return 0;
 381}
 382
 383static int conexant_mux_enum_info(struct snd_kcontrol *kcontrol,
 384	       			  struct snd_ctl_elem_info *uinfo)
 385{
 386	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 387	struct conexant_spec *spec = codec->spec;
 388
 389	return snd_hda_input_mux_info(spec->input_mux, uinfo);
 390}
 391
 392static int conexant_mux_enum_get(struct snd_kcontrol *kcontrol,
 393				 struct snd_ctl_elem_value *ucontrol)
 394{
 395	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 396	struct conexant_spec *spec = codec->spec;
 397	unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
 398
 399	ucontrol->value.enumerated.item[0] = spec->cur_mux[adc_idx];
 400	return 0;
 401}
 402
 403static int conexant_mux_enum_put(struct snd_kcontrol *kcontrol,
 404				 struct snd_ctl_elem_value *ucontrol)
 405{
 406	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 407	struct conexant_spec *spec = codec->spec;
 408	unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
 409
 410	return snd_hda_input_mux_put(codec, spec->input_mux, ucontrol,
 411				     spec->capsrc_nids[adc_idx],
 412				     &spec->cur_mux[adc_idx]);
 413}
 414
 415static int conexant_init_jacks(struct hda_codec *codec)
 416{
 417#ifdef CONFIG_SND_HDA_INPUT_JACK
 418	struct conexant_spec *spec = codec->spec;
 419	int i;
 420
 421	for (i = 0; i < spec->num_init_verbs; i++) {
 422		const struct hda_verb *hv;
 423
 424		hv = spec->init_verbs[i];
 425		while (hv->nid) {
 426			int err = 0;
 427			switch (hv->param ^ AC_USRSP_EN) {
 428			case CONEXANT_HP_EVENT:
 429				err = snd_hda_input_jack_add(codec, hv->nid,
 430						SND_JACK_HEADPHONE, NULL);
 431				snd_hda_input_jack_report(codec, hv->nid);
 432				break;
 433			case CXT5051_PORTC_EVENT:
 434			case CONEXANT_MIC_EVENT:
 435				err = snd_hda_input_jack_add(codec, hv->nid,
 436						SND_JACK_MICROPHONE, NULL);
 437				snd_hda_input_jack_report(codec, hv->nid);
 438				break;
 439			}
 440			if (err < 0)
 441				return err;
 442			++hv;
 443		}
 444	}
 445#endif /* CONFIG_SND_HDA_INPUT_JACK */
 446	return 0;
 447}
 448
 449static void conexant_set_power(struct hda_codec *codec, hda_nid_t fg,
 450			       unsigned int power_state)
 451{
 452	if (power_state == AC_PWRST_D3)
 453		msleep(100);
 454	snd_hda_codec_read(codec, fg, 0, AC_VERB_SET_POWER_STATE,
 455			    power_state);
 456	/* partial workaround for "azx_get_response timeout" */
 457	if (power_state == AC_PWRST_D0)
 458		msleep(10);
 459	snd_hda_codec_set_power_to_all(codec, fg, power_state, true);
 460}
 461
 462static int conexant_init(struct hda_codec *codec)
 463{
 464	struct conexant_spec *spec = codec->spec;
 465	int i;
 466
 467	for (i = 0; i < spec->num_init_verbs; i++)
 468		snd_hda_sequence_write(codec, spec->init_verbs[i]);
 469	return 0;
 470}
 471
 472static void conexant_free(struct hda_codec *codec)
 473{
 474	snd_hda_input_jack_free(codec);
 475	snd_hda_detach_beep_device(codec);
 476	kfree(codec->spec);
 477}
 478
 479static const struct snd_kcontrol_new cxt_capture_mixers[] = {
 480	{
 481		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 482		.name = "Capture Source",
 483		.info = conexant_mux_enum_info,
 484		.get = conexant_mux_enum_get,
 485		.put = conexant_mux_enum_put
 486	},
 487	{}
 488};
 489
 490#ifdef CONFIG_SND_HDA_INPUT_BEEP
 491/* additional beep mixers; the actual parameters are overwritten at build */
 492static const struct snd_kcontrol_new cxt_beep_mixer[] = {
 493	HDA_CODEC_VOLUME_MONO("Beep Playback Volume", 0, 1, 0, HDA_OUTPUT),
 494	HDA_CODEC_MUTE_BEEP_MONO("Beep Playback Switch", 0, 1, 0, HDA_OUTPUT),
 495	{ } /* end */
 496};
 497#endif
 498
 499static const char * const slave_vols[] = {
 500	"Headphone Playback Volume",
 501	"Speaker Playback Volume",
 502	"Front Playback Volume",
 503	"Surround Playback Volume",
 504	"CLFE Playback Volume",
 505	NULL
 506};
 507
 508static const char * const slave_sws[] = {
 509	"Headphone Playback Switch",
 510	"Speaker Playback Switch",
 511	"Front Playback Switch",
 512	"Surround Playback Switch",
 513	"CLFE Playback Switch",
 514	NULL
 515};
 516
 517static int conexant_build_controls(struct hda_codec *codec)
 518{
 519	struct conexant_spec *spec = codec->spec;
 520	unsigned int i;
 521	int err;
 522
 523	for (i = 0; i < spec->num_mixers; i++) {
 524		err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
 525		if (err < 0)
 526			return err;
 527	}
 528	if (spec->multiout.dig_out_nid) {
 529		err = snd_hda_create_spdif_out_ctls(codec,
 530						    spec->multiout.dig_out_nid,
 531						    spec->multiout.dig_out_nid);
 532		if (err < 0)
 533			return err;
 534		err = snd_hda_create_spdif_share_sw(codec,
 535						    &spec->multiout);
 536		if (err < 0)
 537			return err;
 538		spec->multiout.share_spdif = 1;
 539	} 
 540	if (spec->dig_in_nid) {
 541		err = snd_hda_create_spdif_in_ctls(codec,spec->dig_in_nid);
 542		if (err < 0)
 543			return err;
 544	}
 545
 546	/* if we have no master control, let's create it */
 547	if (spec->vmaster_nid &&
 548	    !snd_hda_find_mixer_ctl(codec, "Master Playback Volume")) {
 549		unsigned int vmaster_tlv[4];
 550		snd_hda_set_vmaster_tlv(codec, spec->vmaster_nid,
 551					HDA_OUTPUT, vmaster_tlv);
 552		err = snd_hda_add_vmaster(codec, "Master Playback Volume",
 553					  vmaster_tlv, slave_vols);
 554		if (err < 0)
 555			return err;
 556	}
 557	if (spec->vmaster_nid &&
 558	    !snd_hda_find_mixer_ctl(codec, "Master Playback Switch")) {
 559		err = snd_hda_add_vmaster(codec, "Master Playback Switch",
 560					  NULL, slave_sws);
 561		if (err < 0)
 562			return err;
 563	}
 564
 565	if (spec->input_mux) {
 566		err = snd_hda_add_new_ctls(codec, cxt_capture_mixers);
 567		if (err < 0)
 568			return err;
 569	}
 570
 571#ifdef CONFIG_SND_HDA_INPUT_BEEP
 572	/* create beep controls if needed */
 573	if (spec->beep_amp) {
 574		const struct snd_kcontrol_new *knew;
 575		for (knew = cxt_beep_mixer; knew->name; knew++) {
 576			struct snd_kcontrol *kctl;
 577			kctl = snd_ctl_new1(knew, codec);
 578			if (!kctl)
 579				return -ENOMEM;
 580			kctl->private_value = spec->beep_amp;
 581			err = snd_hda_ctl_add(codec, 0, kctl);
 582			if (err < 0)
 583				return err;
 584		}
 585	}
 586#endif
 587
 588	return 0;
 589}
 590
 591#ifdef CONFIG_SND_HDA_POWER_SAVE
 592static int conexant_suspend(struct hda_codec *codec, pm_message_t state)
 593{
 594	snd_hda_shutup_pins(codec);
 
 
 
 
 
 595	return 0;
 596}
 597#endif
 598
 599static const struct hda_codec_ops conexant_patch_ops = {
 600	.build_controls = conexant_build_controls,
 601	.build_pcms = conexant_build_pcms,
 602	.init = conexant_init,
 603	.free = conexant_free,
 604	.set_power_state = conexant_set_power,
 605#ifdef CONFIG_SND_HDA_POWER_SAVE
 606	.suspend = conexant_suspend,
 607#endif
 608	.reboot_notify = snd_hda_shutup_pins,
 609};
 610
 611#ifdef CONFIG_SND_HDA_INPUT_BEEP
 612#define set_beep_amp(spec, nid, idx, dir) \
 613	((spec)->beep_amp = HDA_COMPOSE_AMP_VAL(nid, 1, idx, dir))
 614#else
 615#define set_beep_amp(spec, nid, idx, dir) /* NOP */
 616#endif
 617
 618static int patch_conexant_auto(struct hda_codec *codec);
 619/*
 620 * EAPD control
 621 * the private value = nid | (invert << 8)
 622 */
 623
 624#define cxt_eapd_info		snd_ctl_boolean_mono_info
 625
 626static int cxt_eapd_get(struct snd_kcontrol *kcontrol,
 627			     struct snd_ctl_elem_value *ucontrol)
 628{
 629	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 630	struct conexant_spec *spec = codec->spec;
 631	int invert = (kcontrol->private_value >> 8) & 1;
 632	if (invert)
 633		ucontrol->value.integer.value[0] = !spec->cur_eapd;
 634	else
 635		ucontrol->value.integer.value[0] = spec->cur_eapd;
 636	return 0;
 637
 638}
 
 
 
 
 
 
 
 
 639
 640static int cxt_eapd_put(struct snd_kcontrol *kcontrol,
 641			     struct snd_ctl_elem_value *ucontrol)
 642{
 643	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 644	struct conexant_spec *spec = codec->spec;
 645	int invert = (kcontrol->private_value >> 8) & 1;
 646	hda_nid_t nid = kcontrol->private_value & 0xff;
 647	unsigned int eapd;
 648
 649	eapd = !!ucontrol->value.integer.value[0];
 650	if (invert)
 651		eapd = !eapd;
 652	if (eapd == spec->cur_eapd)
 653		return 0;
 654	
 655	spec->cur_eapd = eapd;
 656	snd_hda_codec_write_cache(codec, nid,
 657				  0, AC_VERB_SET_EAPD_BTLENABLE,
 658				  eapd ? 0x02 : 0x00);
 659	return 1;
 660}
 661
 662/* controls for test mode */
 663#ifdef CONFIG_SND_DEBUG
 664
 665#define CXT_EAPD_SWITCH(xname, nid, mask) \
 666	{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0,  \
 667	  .info = cxt_eapd_info, \
 668	  .get = cxt_eapd_get, \
 669	  .put = cxt_eapd_put, \
 670	  .private_value = nid | (mask<<16) }
 671
 672
 673
 674static int conexant_ch_mode_info(struct snd_kcontrol *kcontrol,
 675				 struct snd_ctl_elem_info *uinfo)
 676{
 677	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 678	struct conexant_spec *spec = codec->spec;
 679	return snd_hda_ch_mode_info(codec, uinfo, spec->channel_mode,
 680				    spec->num_channel_mode);
 
 
 
 681}
 682
 683static int conexant_ch_mode_get(struct snd_kcontrol *kcontrol,
 684				struct snd_ctl_elem_value *ucontrol)
 685{
 686	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 687	struct conexant_spec *spec = codec->spec;
 688	return snd_hda_ch_mode_get(codec, ucontrol, spec->channel_mode,
 689				   spec->num_channel_mode,
 690				   spec->multiout.max_channels);
 691}
 692
 693static int conexant_ch_mode_put(struct snd_kcontrol *kcontrol,
 694				struct snd_ctl_elem_value *ucontrol)
 
 695{
 696	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 697	struct conexant_spec *spec = codec->spec;
 698	int err = snd_hda_ch_mode_put(codec, ucontrol, spec->channel_mode,
 699				      spec->num_channel_mode,
 700				      &spec->multiout.max_channels);
 701	if (err >= 0 && spec->need_dac_fix)
 702		spec->multiout.num_dacs = spec->multiout.max_channels / 2;
 703	return err;
 704}
 705
 706#define CXT_PIN_MODE(xname, nid, dir) \
 707	{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = 0,  \
 708	  .info = conexant_ch_mode_info, \
 709	  .get = conexant_ch_mode_get, \
 710	  .put = conexant_ch_mode_put, \
 711	  .private_value = nid | (dir<<16) }
 712
 713#endif /* CONFIG_SND_DEBUG */
 714
 715/* Conexant 5045 specific */
 716
 717static const hda_nid_t cxt5045_dac_nids[1] = { 0x19 };
 718static const hda_nid_t cxt5045_adc_nids[1] = { 0x1a };
 719static const hda_nid_t cxt5045_capsrc_nids[1] = { 0x1a };
 720#define CXT5045_SPDIF_OUT	0x18
 721
 722static const struct hda_channel_mode cxt5045_modes[1] = {
 723	{ 2, NULL },
 724};
 725
 726static const struct hda_input_mux cxt5045_capture_source = {
 727	.num_items = 2,
 728	.items = {
 729		{ "IntMic", 0x1 },
 730		{ "ExtMic", 0x2 },
 731	}
 732};
 733
 734static const struct hda_input_mux cxt5045_capture_source_benq = {
 735	.num_items = 5,
 736	.items = {
 737		{ "IntMic", 0x1 },
 738		{ "ExtMic", 0x2 },
 739		{ "LineIn", 0x3 },
 740		{ "CD",     0x4 },
 741		{ "Mixer",  0x0 },
 742	}
 743};
 744
 745static const struct hda_input_mux cxt5045_capture_source_hp530 = {
 746	.num_items = 2,
 747	.items = {
 748		{ "ExtMic", 0x1 },
 749		{ "IntMic", 0x2 },
 750	}
 751};
 752
 753/* turn on/off EAPD (+ mute HP) as a master switch */
 754static int cxt5045_hp_master_sw_put(struct snd_kcontrol *kcontrol,
 755				    struct snd_ctl_elem_value *ucontrol)
 756{
 757	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 758	struct conexant_spec *spec = codec->spec;
 759	unsigned int bits;
 760
 761	if (!cxt_eapd_put(kcontrol, ucontrol))
 762		return 0;
 763
 764	/* toggle internal speakers mute depending of presence of
 765	 * the headphone jack
 766	 */
 767	bits = (!spec->hp_present && spec->cur_eapd) ? 0 : HDA_AMP_MUTE;
 768	snd_hda_codec_amp_stereo(codec, 0x10, HDA_OUTPUT, 0,
 769				 HDA_AMP_MUTE, bits);
 770
 771	bits = spec->cur_eapd ? 0 : HDA_AMP_MUTE;
 772	snd_hda_codec_amp_stereo(codec, 0x11, HDA_OUTPUT, 0,
 773				 HDA_AMP_MUTE, bits);
 774	return 1;
 775}
 776
 777/* bind volumes of both NID 0x10 and 0x11 */
 778static const struct hda_bind_ctls cxt5045_hp_bind_master_vol = {
 779	.ops = &snd_hda_bind_vol,
 780	.values = {
 781		HDA_COMPOSE_AMP_VAL(0x10, 3, 0, HDA_OUTPUT),
 782		HDA_COMPOSE_AMP_VAL(0x11, 3, 0, HDA_OUTPUT),
 783		0
 784	},
 785};
 786
 787/* toggle input of built-in and mic jack appropriately */
 788static void cxt5045_hp_automic(struct hda_codec *codec)
 789{
 790	static const struct hda_verb mic_jack_on[] = {
 791		{0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0xb080},
 792		{0x12, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
 793		{}
 794	};
 795	static const struct hda_verb mic_jack_off[] = {
 796		{0x12, AC_VERB_SET_AMP_GAIN_MUTE, 0xb080},
 797		{0x14, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
 798		{}
 799	};
 800	unsigned int present;
 801
 802	present = snd_hda_jack_detect(codec, 0x12);
 803	if (present)
 804		snd_hda_sequence_write(codec, mic_jack_on);
 
 
 
 
 
 
 
 
 
 805	else
 806		snd_hda_sequence_write(codec, mic_jack_off);
 
 807}
 808
 809
 810/* mute internal speaker if HP is plugged */
 811static void cxt5045_hp_automute(struct hda_codec *codec)
 812{
 813	struct conexant_spec *spec = codec->spec;
 814	unsigned int bits;
 
 
 815
 816	spec->hp_present = snd_hda_jack_detect(codec, 0x11);
 
 817
 818	bits = (spec->hp_present || !spec->cur_eapd) ? HDA_AMP_MUTE : 0; 
 819	snd_hda_codec_amp_stereo(codec, 0x10, HDA_OUTPUT, 0,
 820				 HDA_AMP_MUTE, bits);
 
 821}
 822
 823/* unsolicited event for HP jack sensing */
 824static void cxt5045_hp_unsol_event(struct hda_codec *codec,
 825				   unsigned int res)
 826{
 827	res >>= 26;
 828	switch (res) {
 829	case CONEXANT_HP_EVENT:
 830		cxt5045_hp_automute(codec);
 831		break;
 832	case CONEXANT_MIC_EVENT:
 833		cxt5045_hp_automic(codec);
 834		break;
 835
 836	}
 
 
 837}
 838
 839static const struct snd_kcontrol_new cxt5045_mixers[] = {
 840	HDA_CODEC_VOLUME("Internal Mic Capture Volume", 0x1a, 0x01, HDA_INPUT),
 841	HDA_CODEC_MUTE("Internal Mic Capture Switch", 0x1a, 0x01, HDA_INPUT),
 842	HDA_CODEC_VOLUME("Mic Capture Volume", 0x1a, 0x02, HDA_INPUT),
 843	HDA_CODEC_MUTE("Mic Capture Switch", 0x1a, 0x02, HDA_INPUT),
 844	HDA_CODEC_VOLUME("PCM Playback Volume", 0x17, 0x0, HDA_INPUT),
 845	HDA_CODEC_MUTE("PCM Playback Switch", 0x17, 0x0, HDA_INPUT),
 846	HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x17, 0x1, HDA_INPUT),
 847	HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x17, 0x1, HDA_INPUT),
 848	HDA_CODEC_VOLUME("Mic Playback Volume", 0x17, 0x2, HDA_INPUT),
 849	HDA_CODEC_MUTE("Mic Playback Switch", 0x17, 0x2, HDA_INPUT),
 850	HDA_BIND_VOL("Master Playback Volume", &cxt5045_hp_bind_master_vol),
 851	{
 852		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 853		.name = "Master Playback Switch",
 854		.info = cxt_eapd_info,
 855		.get = cxt_eapd_get,
 856		.put = cxt5045_hp_master_sw_put,
 857		.private_value = 0x10,
 858	},
 859
 860	{}
 861};
 862
 863static const struct snd_kcontrol_new cxt5045_benq_mixers[] = {
 864	HDA_CODEC_VOLUME("CD Capture Volume", 0x1a, 0x04, HDA_INPUT),
 865	HDA_CODEC_MUTE("CD Capture Switch", 0x1a, 0x04, HDA_INPUT),
 866	HDA_CODEC_VOLUME("CD Playback Volume", 0x17, 0x4, HDA_INPUT),
 867	HDA_CODEC_MUTE("CD Playback Switch", 0x17, 0x4, HDA_INPUT),
 868
 869	HDA_CODEC_VOLUME("Line In Capture Volume", 0x1a, 0x03, HDA_INPUT),
 870	HDA_CODEC_MUTE("Line In Capture Switch", 0x1a, 0x03, HDA_INPUT),
 871	HDA_CODEC_VOLUME("Line In Playback Volume", 0x17, 0x3, HDA_INPUT),
 872	HDA_CODEC_MUTE("Line In Playback Switch", 0x17, 0x3, HDA_INPUT),
 873
 874	HDA_CODEC_VOLUME("Mixer Capture Volume", 0x1a, 0x0, HDA_INPUT),
 875	HDA_CODEC_MUTE("Mixer Capture Switch", 0x1a, 0x0, HDA_INPUT),
 876
 877	{}
 878};
 879
 880static const struct snd_kcontrol_new cxt5045_mixers_hp530[] = {
 881	HDA_CODEC_VOLUME("Internal Mic Capture Volume", 0x1a, 0x02, HDA_INPUT),
 882	HDA_CODEC_MUTE("Internal Mic Capture Switch", 0x1a, 0x02, HDA_INPUT),
 883	HDA_CODEC_VOLUME("Mic Capture Volume", 0x1a, 0x01, HDA_INPUT),
 884	HDA_CODEC_MUTE("Mic Capture Switch", 0x1a, 0x01, HDA_INPUT),
 885	HDA_CODEC_VOLUME("PCM Playback Volume", 0x17, 0x0, HDA_INPUT),
 886	HDA_CODEC_MUTE("PCM Playback Switch", 0x17, 0x0, HDA_INPUT),
 887	HDA_CODEC_VOLUME("Internal Mic Playback Volume", 0x17, 0x2, HDA_INPUT),
 888	HDA_CODEC_MUTE("Internal Mic Playback Switch", 0x17, 0x2, HDA_INPUT),
 889	HDA_CODEC_VOLUME("Mic Playback Volume", 0x17, 0x1, HDA_INPUT),
 890	HDA_CODEC_MUTE("Mic Playback Switch", 0x17, 0x1, HDA_INPUT),
 891	HDA_BIND_VOL("Master Playback Volume", &cxt5045_hp_bind_master_vol),
 892	{
 893		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 894		.name = "Master Playback Switch",
 895		.info = cxt_eapd_info,
 896		.get = cxt_eapd_get,
 897		.put = cxt5045_hp_master_sw_put,
 898		.private_value = 0x10,
 899	},
 900
 901	{}
 902};
 903
 904static const struct hda_verb cxt5045_init_verbs[] = {
 905	/* Line in, Mic */
 906	{0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN|AC_PINCTL_VREF_80 },
 907	{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN|AC_PINCTL_VREF_80 },
 908	/* HP, Amp  */
 909	{0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
 910	{0x10, AC_VERB_SET_CONNECT_SEL, 0x1},
 911	{0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
 912	{0x11, AC_VERB_SET_CONNECT_SEL, 0x1},
 913	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
 914	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
 915	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
 916	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
 917	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
 918	/* Record selector: Internal mic */
 919	{0x1a, AC_VERB_SET_CONNECT_SEL,0x1},
 920	{0x1a, AC_VERB_SET_AMP_GAIN_MUTE,
 921	 AC_AMP_SET_INPUT|AC_AMP_SET_RIGHT|AC_AMP_SET_LEFT|0x17},
 922	/* SPDIF route: PCM */
 923	{0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
 924	{ 0x13, AC_VERB_SET_CONNECT_SEL, 0x0 },
 925	/* EAPD */
 926	{0x10, AC_VERB_SET_EAPD_BTLENABLE, 0x2 }, /* default on */ 
 927	{ } /* end */
 928};
 929
 930static const struct hda_verb cxt5045_benq_init_verbs[] = {
 931	/* Internal Mic, Mic */
 932	{0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN|AC_PINCTL_VREF_80 },
 933	{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN|AC_PINCTL_VREF_80 },
 934	/* Line In,HP, Amp  */
 935	{0x10, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
 936	{0x10, AC_VERB_SET_CONNECT_SEL, 0x1},
 937	{0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
 938	{0x11, AC_VERB_SET_CONNECT_SEL, 0x1},
 939	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
 940	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
 941	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
 942	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
 943	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
 944	/* Record selector: Internal mic */
 945	{0x1a, AC_VERB_SET_CONNECT_SEL, 0x1},
 946	{0x1a, AC_VERB_SET_AMP_GAIN_MUTE,
 947	 AC_AMP_SET_INPUT|AC_AMP_SET_RIGHT|AC_AMP_SET_LEFT|0x17},
 948	/* SPDIF route: PCM */
 949	{0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
 950	{0x13, AC_VERB_SET_CONNECT_SEL, 0x0},
 951	/* EAPD */
 952	{0x10, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
 953	{ } /* end */
 954};
 955
 956static const struct hda_verb cxt5045_hp_sense_init_verbs[] = {
 957	/* pin sensing on HP jack */
 958	{0x11, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
 959	{ } /* end */
 960};
 961
 962static const struct hda_verb cxt5045_mic_sense_init_verbs[] = {
 963	/* pin sensing on HP jack */
 964	{0x12, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
 965	{ } /* end */
 966};
 967
 968#ifdef CONFIG_SND_DEBUG
 969/* Test configuration for debugging, modelled after the ALC260 test
 970 * configuration.
 971 */
 972static const struct hda_input_mux cxt5045_test_capture_source = {
 973	.num_items = 5,
 974	.items = {
 975		{ "MIXER", 0x0 },
 976		{ "MIC1 pin", 0x1 },
 977		{ "LINE1 pin", 0x2 },
 978		{ "HP-OUT pin", 0x3 },
 979		{ "CD pin", 0x4 },
 980        },
 981};
 982
 983static const struct snd_kcontrol_new cxt5045_test_mixer[] = {
 984
 985	/* Output controls */
 986	HDA_CODEC_VOLUME("Speaker Playback Volume", 0x10, 0x0, HDA_OUTPUT),
 987	HDA_CODEC_MUTE("Speaker Playback Switch", 0x10, 0x0, HDA_OUTPUT),
 988	HDA_CODEC_VOLUME("Node 11 Playback Volume", 0x11, 0x0, HDA_OUTPUT),
 989	HDA_CODEC_MUTE("Node 11 Playback Switch", 0x11, 0x0, HDA_OUTPUT),
 990	HDA_CODEC_VOLUME("Node 12 Playback Volume", 0x12, 0x0, HDA_OUTPUT),
 991	HDA_CODEC_MUTE("Node 12 Playback Switch", 0x12, 0x0, HDA_OUTPUT),
 992	
 993	/* Modes for retasking pin widgets */
 994	CXT_PIN_MODE("HP-OUT pin mode", 0x11, CXT_PIN_DIR_INOUT),
 995	CXT_PIN_MODE("LINE1 pin mode", 0x12, CXT_PIN_DIR_INOUT),
 996
 997	/* EAPD Switch Control */
 998	CXT_EAPD_SWITCH("External Amplifier", 0x10, 0x0),
 999
1000	/* Loopback mixer controls */
1001
1002	HDA_CODEC_VOLUME("Mixer-1 Volume", 0x17, 0x0, HDA_INPUT),
1003	HDA_CODEC_MUTE("Mixer-1 Switch", 0x17, 0x0, HDA_INPUT),
1004	HDA_CODEC_VOLUME("Mixer-2 Volume", 0x17, 0x1, HDA_INPUT),
1005	HDA_CODEC_MUTE("Mixer-2 Switch", 0x17, 0x1, HDA_INPUT),
1006	HDA_CODEC_VOLUME("Mixer-3 Volume", 0x17, 0x2, HDA_INPUT),
1007	HDA_CODEC_MUTE("Mixer-3 Switch", 0x17, 0x2, HDA_INPUT),
1008	HDA_CODEC_VOLUME("Mixer-4 Volume", 0x17, 0x3, HDA_INPUT),
1009	HDA_CODEC_MUTE("Mixer-4 Switch", 0x17, 0x3, HDA_INPUT),
1010	HDA_CODEC_VOLUME("Mixer-5 Volume", 0x17, 0x4, HDA_INPUT),
1011	HDA_CODEC_MUTE("Mixer-5 Switch", 0x17, 0x4, HDA_INPUT),
1012	{
1013		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1014		.name = "Input Source",
1015		.info = conexant_mux_enum_info,
1016		.get = conexant_mux_enum_get,
1017		.put = conexant_mux_enum_put,
1018	},
1019	/* Audio input controls */
1020	HDA_CODEC_VOLUME("Input-1 Volume", 0x1a, 0x0, HDA_INPUT),
1021	HDA_CODEC_MUTE("Input-1 Switch", 0x1a, 0x0, HDA_INPUT),
1022	HDA_CODEC_VOLUME("Input-2 Volume", 0x1a, 0x1, HDA_INPUT),
1023	HDA_CODEC_MUTE("Input-2 Switch", 0x1a, 0x1, HDA_INPUT),
1024	HDA_CODEC_VOLUME("Input-3 Volume", 0x1a, 0x2, HDA_INPUT),
1025	HDA_CODEC_MUTE("Input-3 Switch", 0x1a, 0x2, HDA_INPUT),
1026	HDA_CODEC_VOLUME("Input-4 Volume", 0x1a, 0x3, HDA_INPUT),
1027	HDA_CODEC_MUTE("Input-4 Switch", 0x1a, 0x3, HDA_INPUT),
1028	HDA_CODEC_VOLUME("Input-5 Volume", 0x1a, 0x4, HDA_INPUT),
1029	HDA_CODEC_MUTE("Input-5 Switch", 0x1a, 0x4, HDA_INPUT),
1030	{ } /* end */
1031};
1032
1033static const struct hda_verb cxt5045_test_init_verbs[] = {
1034	/* Set connections */
1035	{ 0x10, AC_VERB_SET_CONNECT_SEL, 0x0 },
1036	{ 0x11, AC_VERB_SET_CONNECT_SEL, 0x0 },
1037	{ 0x12, AC_VERB_SET_CONNECT_SEL, 0x0 },
1038	/* Enable retasking pins as output, initially without power amp */
1039	{0x12, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1040	{0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1041
1042	/* Disable digital (SPDIF) pins initially, but users can enable
1043	 * them via a mixer switch.  In the case of SPDIF-out, this initverb
1044	 * payload also sets the generation to 0, output to be in "consumer"
1045	 * PCM format, copyright asserted, no pre-emphasis and no validity
1046	 * control.
1047	 */
1048	{0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1049	{0x18, AC_VERB_SET_DIGI_CONVERT_1, 0},
1050
1051	/* Start with output sum widgets muted and their output gains at min */
1052	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1053	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
1054
1055	/* Unmute retasking pin widget output buffers since the default
1056	 * state appears to be output.  As the pin mode is changed by the
1057	 * user the pin mode control will take care of enabling the pin's
1058	 * input/output buffers as needed.
1059	 */
1060	{0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1061	{0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1062
1063	/* Mute capture amp left and right */
1064	{0x1a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1065
1066	/* Set ADC connection select to match default mixer setting (mic1
1067	 * pin)
1068	 */
1069	{0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
1070	{0x17, AC_VERB_SET_CONNECT_SEL, 0x00},
1071
1072	/* Mute all inputs to mixer widget (even unconnected ones) */
1073	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* Mixer pin */
1074	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* Mic1 pin */
1075	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* Line pin */
1076	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* HP pin */
1077	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
1078
1079	{ }
1080};
1081#endif
1082
1083
1084/* initialize jack-sensing, too */
1085static int cxt5045_init(struct hda_codec *codec)
1086{
1087	conexant_init(codec);
1088	cxt5045_hp_automute(codec);
1089	return 0;
1090}
1091
1092
1093enum {
1094	CXT5045_LAPTOP_HPSENSE,
1095	CXT5045_LAPTOP_MICSENSE,
1096	CXT5045_LAPTOP_HPMICSENSE,
1097	CXT5045_BENQ,
1098	CXT5045_LAPTOP_HP530,
1099#ifdef CONFIG_SND_DEBUG
1100	CXT5045_TEST,
1101#endif
1102	CXT5045_AUTO,
1103	CXT5045_MODELS
1104};
1105
1106static const char * const cxt5045_models[CXT5045_MODELS] = {
1107	[CXT5045_LAPTOP_HPSENSE]	= "laptop-hpsense",
1108	[CXT5045_LAPTOP_MICSENSE]	= "laptop-micsense",
1109	[CXT5045_LAPTOP_HPMICSENSE]	= "laptop-hpmicsense",
1110	[CXT5045_BENQ]			= "benq",
1111	[CXT5045_LAPTOP_HP530]		= "laptop-hp530",
1112#ifdef CONFIG_SND_DEBUG
1113	[CXT5045_TEST]		= "test",
1114#endif
1115	[CXT5045_AUTO]			= "auto",
1116};
1117
1118static const struct snd_pci_quirk cxt5045_cfg_tbl[] = {
1119	SND_PCI_QUIRK(0x103c, 0x30d5, "HP 530", CXT5045_LAPTOP_HP530),
1120	SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x3000, "HP DV Series",
1121			   CXT5045_LAPTOP_HPSENSE),
1122	SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba P105", CXT5045_LAPTOP_MICSENSE),
1123	SND_PCI_QUIRK(0x152d, 0x0753, "Benq R55E", CXT5045_BENQ),
1124	SND_PCI_QUIRK(0x1734, 0x10ad, "Fujitsu Si1520", CXT5045_LAPTOP_MICSENSE),
1125	SND_PCI_QUIRK(0x1734, 0x10cb, "Fujitsu Si3515", CXT5045_LAPTOP_HPMICSENSE),
1126	SND_PCI_QUIRK(0x1734, 0x110e, "Fujitsu V5505",
1127		      CXT5045_LAPTOP_HPMICSENSE),
1128	SND_PCI_QUIRK(0x1509, 0x1e40, "FIC", CXT5045_LAPTOP_HPMICSENSE),
1129	SND_PCI_QUIRK(0x1509, 0x2f05, "FIC", CXT5045_LAPTOP_HPMICSENSE),
1130	SND_PCI_QUIRK(0x1509, 0x2f06, "FIC", CXT5045_LAPTOP_HPMICSENSE),
1131	SND_PCI_QUIRK_MASK(0x1631, 0xff00, 0xc100, "Packard Bell",
1132			   CXT5045_LAPTOP_HPMICSENSE),
1133	SND_PCI_QUIRK(0x8086, 0x2111, "Conexant Reference board", CXT5045_LAPTOP_HPSENSE),
1134	{}
1135};
1136
1137static int patch_cxt5045(struct hda_codec *codec)
1138{
1139	struct conexant_spec *spec;
1140	int board_config;
1141
1142	board_config = snd_hda_check_board_config(codec, CXT5045_MODELS,
1143						  cxt5045_models,
1144						  cxt5045_cfg_tbl);
1145	if (board_config < 0)
1146		board_config = CXT5045_AUTO; /* model=auto as default */
1147	if (board_config == CXT5045_AUTO)
1148		return patch_conexant_auto(codec);
1149
1150	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1151	if (!spec)
1152		return -ENOMEM;
1153	codec->spec = spec;
1154	codec->pin_amp_workaround = 1;
1155
1156	spec->multiout.max_channels = 2;
1157	spec->multiout.num_dacs = ARRAY_SIZE(cxt5045_dac_nids);
1158	spec->multiout.dac_nids = cxt5045_dac_nids;
1159	spec->multiout.dig_out_nid = CXT5045_SPDIF_OUT;
1160	spec->num_adc_nids = 1;
1161	spec->adc_nids = cxt5045_adc_nids;
1162	spec->capsrc_nids = cxt5045_capsrc_nids;
1163	spec->input_mux = &cxt5045_capture_source;
1164	spec->num_mixers = 1;
1165	spec->mixers[0] = cxt5045_mixers;
1166	spec->num_init_verbs = 1;
1167	spec->init_verbs[0] = cxt5045_init_verbs;
1168	spec->spdif_route = 0;
1169	spec->num_channel_mode = ARRAY_SIZE(cxt5045_modes);
1170	spec->channel_mode = cxt5045_modes;
1171
1172	set_beep_amp(spec, 0x16, 0, 1);
1173
1174	codec->patch_ops = conexant_patch_ops;
1175
1176	switch (board_config) {
1177	case CXT5045_LAPTOP_HPSENSE:
1178		codec->patch_ops.unsol_event = cxt5045_hp_unsol_event;
1179		spec->input_mux = &cxt5045_capture_source;
1180		spec->num_init_verbs = 2;
1181		spec->init_verbs[1] = cxt5045_hp_sense_init_verbs;
1182		spec->mixers[0] = cxt5045_mixers;
1183		codec->patch_ops.init = cxt5045_init;
1184		break;
1185	case CXT5045_LAPTOP_MICSENSE:
1186		codec->patch_ops.unsol_event = cxt5045_hp_unsol_event;
1187		spec->input_mux = &cxt5045_capture_source;
1188		spec->num_init_verbs = 2;
1189		spec->init_verbs[1] = cxt5045_mic_sense_init_verbs;
1190		spec->mixers[0] = cxt5045_mixers;
1191		codec->patch_ops.init = cxt5045_init;
1192		break;
1193	default:
1194	case CXT5045_LAPTOP_HPMICSENSE:
1195		codec->patch_ops.unsol_event = cxt5045_hp_unsol_event;
1196		spec->input_mux = &cxt5045_capture_source;
1197		spec->num_init_verbs = 3;
1198		spec->init_verbs[1] = cxt5045_hp_sense_init_verbs;
1199		spec->init_verbs[2] = cxt5045_mic_sense_init_verbs;
1200		spec->mixers[0] = cxt5045_mixers;
1201		codec->patch_ops.init = cxt5045_init;
1202		break;
1203	case CXT5045_BENQ:
1204		codec->patch_ops.unsol_event = cxt5045_hp_unsol_event;
1205		spec->input_mux = &cxt5045_capture_source_benq;
1206		spec->num_init_verbs = 1;
1207		spec->init_verbs[0] = cxt5045_benq_init_verbs;
1208		spec->mixers[0] = cxt5045_mixers;
1209		spec->mixers[1] = cxt5045_benq_mixers;
1210		spec->num_mixers = 2;
1211		codec->patch_ops.init = cxt5045_init;
1212		break;
1213	case CXT5045_LAPTOP_HP530:
1214		codec->patch_ops.unsol_event = cxt5045_hp_unsol_event;
1215		spec->input_mux = &cxt5045_capture_source_hp530;
1216		spec->num_init_verbs = 2;
1217		spec->init_verbs[1] = cxt5045_hp_sense_init_verbs;
1218		spec->mixers[0] = cxt5045_mixers_hp530;
1219		codec->patch_ops.init = cxt5045_init;
1220		break;
1221#ifdef CONFIG_SND_DEBUG
1222	case CXT5045_TEST:
1223		spec->input_mux = &cxt5045_test_capture_source;
1224		spec->mixers[0] = cxt5045_test_mixer;
1225		spec->init_verbs[0] = cxt5045_test_init_verbs;
1226		break;
1227		
1228#endif	
1229	}
1230
1231	switch (codec->subsystem_id >> 16) {
1232	case 0x103c:
1233	case 0x1631:
1234	case 0x1734:
1235	case 0x17aa:
1236		/* HP, Packard Bell, Fujitsu-Siemens & Lenovo laptops have
1237		 * really bad sound over 0dB on NID 0x17. Fix max PCM level to
1238		 * 0 dB (originally it has 0x2b steps with 0dB offset 0x14)
1239		 */
1240		snd_hda_override_amp_caps(codec, 0x17, HDA_INPUT,
1241					  (0x14 << AC_AMPCAP_OFFSET_SHIFT) |
1242					  (0x14 << AC_AMPCAP_NUM_STEPS_SHIFT) |
1243					  (0x05 << AC_AMPCAP_STEP_SIZE_SHIFT) |
1244					  (1 << AC_AMPCAP_MUTE_SHIFT));
1245		break;
 
 
 
 
1246	}
1247
1248	if (spec->beep_amp)
1249		snd_hda_attach_beep_device(codec, spec->beep_amp);
1250
1251	return 0;
1252}
1253
1254
1255/* Conexant 5047 specific */
1256#define CXT5047_SPDIF_OUT	0x11
1257
1258static const hda_nid_t cxt5047_dac_nids[1] = { 0x10 }; /* 0x1c */
1259static const hda_nid_t cxt5047_adc_nids[1] = { 0x12 };
1260static const hda_nid_t cxt5047_capsrc_nids[1] = { 0x1a };
1261
1262static const struct hda_channel_mode cxt5047_modes[1] = {
1263	{ 2, NULL },
1264};
1265
1266static const struct hda_input_mux cxt5047_toshiba_capture_source = {
1267	.num_items = 2,
1268	.items = {
1269		{ "ExtMic", 0x2 },
1270		{ "Line-In", 0x1 },
1271	}
1272};
1273
1274/* turn on/off EAPD (+ mute HP) as a master switch */
1275static int cxt5047_hp_master_sw_put(struct snd_kcontrol *kcontrol,
1276				    struct snd_ctl_elem_value *ucontrol)
1277{
1278	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1279	struct conexant_spec *spec = codec->spec;
1280	unsigned int bits;
1281
1282	if (!cxt_eapd_put(kcontrol, ucontrol))
1283		return 0;
1284
1285	/* toggle internal speakers mute depending of presence of
1286	 * the headphone jack
1287	 */
1288	bits = (!spec->hp_present && spec->cur_eapd) ? 0 : HDA_AMP_MUTE;
1289	/* NOTE: Conexat codec needs the index for *OUTPUT* amp of
1290	 * pin widgets unlike other codecs.  In this case, we need to
1291	 * set index 0x01 for the volume from the mixer amp 0x19.
1292	 */
1293	snd_hda_codec_amp_stereo(codec, 0x1d, HDA_OUTPUT, 0x01,
1294				 HDA_AMP_MUTE, bits);
1295	bits = spec->cur_eapd ? 0 : HDA_AMP_MUTE;
1296	snd_hda_codec_amp_stereo(codec, 0x13, HDA_OUTPUT, 0,
1297				 HDA_AMP_MUTE, bits);
1298	return 1;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1299}
1300
1301/* mute internal speaker if HP is plugged */
1302static void cxt5047_hp_automute(struct hda_codec *codec)
1303{
1304	struct conexant_spec *spec = codec->spec;
1305	unsigned int bits;
1306
1307	spec->hp_present = snd_hda_jack_detect(codec, 0x13);
 
1308
1309	bits = (spec->hp_present || !spec->cur_eapd) ? HDA_AMP_MUTE : 0;
1310	/* See the note in cxt5047_hp_master_sw_put */
1311	snd_hda_codec_amp_stereo(codec, 0x1d, HDA_OUTPUT, 0x01,
1312				 HDA_AMP_MUTE, bits);
1313}
1314
1315/* toggle input of built-in and mic jack appropriately */
1316static void cxt5047_hp_automic(struct hda_codec *codec)
1317{
1318	static const struct hda_verb mic_jack_on[] = {
1319		{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1320		{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1321		{}
1322	};
1323	static const struct hda_verb mic_jack_off[] = {
1324		{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
1325		{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1326		{}
1327	};
1328	unsigned int present;
1329
1330	present = snd_hda_jack_detect(codec, 0x15);
1331	if (present)
1332		snd_hda_sequence_write(codec, mic_jack_on);
1333	else
1334		snd_hda_sequence_write(codec, mic_jack_off);
1335}
1336
1337/* unsolicited event for HP jack sensing */
1338static void cxt5047_hp_unsol_event(struct hda_codec *codec,
1339				  unsigned int res)
1340{
1341	switch (res >> 26) {
1342	case CONEXANT_HP_EVENT:
1343		cxt5047_hp_automute(codec);
1344		break;
1345	case CONEXANT_MIC_EVENT:
1346		cxt5047_hp_automic(codec);
1347		break;
1348	}
1349}
1350
1351static const struct snd_kcontrol_new cxt5047_base_mixers[] = {
1352	HDA_CODEC_VOLUME("Mic Playback Volume", 0x19, 0x02, HDA_INPUT),
1353	HDA_CODEC_MUTE("Mic Playback Switch", 0x19, 0x02, HDA_INPUT),
1354	HDA_CODEC_VOLUME("Mic Boost Volume", 0x1a, 0x0, HDA_OUTPUT),
1355	HDA_CODEC_VOLUME("Capture Volume", 0x12, 0x03, HDA_INPUT),
1356	HDA_CODEC_MUTE("Capture Switch", 0x12, 0x03, HDA_INPUT),
1357	HDA_CODEC_VOLUME("PCM Volume", 0x10, 0x00, HDA_OUTPUT),
1358	HDA_CODEC_MUTE("PCM Switch", 0x10, 0x00, HDA_OUTPUT),
1359	{
1360		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1361		.name = "Master Playback Switch",
1362		.info = cxt_eapd_info,
1363		.get = cxt_eapd_get,
1364		.put = cxt5047_hp_master_sw_put,
1365		.private_value = 0x13,
1366	},
1367
1368	{}
1369};
1370
1371static const struct snd_kcontrol_new cxt5047_hp_spk_mixers[] = {
1372	/* See the note in cxt5047_hp_master_sw_put */
1373	HDA_CODEC_VOLUME("Speaker Playback Volume", 0x1d, 0x01, HDA_OUTPUT),
1374	HDA_CODEC_VOLUME("Headphone Playback Volume", 0x13, 0x00, HDA_OUTPUT),
1375	{}
1376};
1377
1378static const struct snd_kcontrol_new cxt5047_hp_only_mixers[] = {
1379	HDA_CODEC_VOLUME("Master Playback Volume", 0x13, 0x00, HDA_OUTPUT),
1380	{ } /* end */
1381};
1382
1383static const struct hda_verb cxt5047_init_verbs[] = {
1384	/* Line in, Mic, Built-in Mic */
1385	{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN },
1386	{0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN|AC_PINCTL_VREF_50 },
1387	{0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN|AC_PINCTL_VREF_50 },
1388	/* HP, Speaker  */
1389	{0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP },
1390	{0x13, AC_VERB_SET_CONNECT_SEL, 0x0}, /* mixer(0x19) */
1391	{0x1d, AC_VERB_SET_CONNECT_SEL, 0x1}, /* mixer(0x19) */
1392	/* Record selector: Mic */
1393	{0x12, AC_VERB_SET_CONNECT_SEL,0x03},
1394	{0x19, AC_VERB_SET_AMP_GAIN_MUTE,
1395	 AC_AMP_SET_INPUT|AC_AMP_SET_RIGHT|AC_AMP_SET_LEFT|0x17},
1396	{0x1A, AC_VERB_SET_CONNECT_SEL,0x02},
1397	{0x1A, AC_VERB_SET_AMP_GAIN_MUTE,
1398	 AC_AMP_SET_OUTPUT|AC_AMP_SET_RIGHT|AC_AMP_SET_LEFT|0x00},
1399	{0x1A, AC_VERB_SET_AMP_GAIN_MUTE,
1400	 AC_AMP_SET_OUTPUT|AC_AMP_SET_RIGHT|AC_AMP_SET_LEFT|0x03},
1401	/* SPDIF route: PCM */
1402	{ 0x18, AC_VERB_SET_CONNECT_SEL, 0x0 },
1403	/* Enable unsolicited events */
1404	{0x13, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
1405	{0x15, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
1406	{ } /* end */
1407};
1408
1409/* configuration for Toshiba Laptops */
1410static const struct hda_verb cxt5047_toshiba_init_verbs[] = {
1411	{0x13, AC_VERB_SET_EAPD_BTLENABLE, 0x0}, /* default off */
1412	{}
1413};
1414
1415/* Test configuration for debugging, modelled after the ALC260 test
1416 * configuration.
1417 */
1418#ifdef CONFIG_SND_DEBUG
1419static const struct hda_input_mux cxt5047_test_capture_source = {
1420	.num_items = 4,
1421	.items = {
1422		{ "LINE1 pin", 0x0 },
1423		{ "MIC1 pin", 0x1 },
1424		{ "MIC2 pin", 0x2 },
1425		{ "CD pin", 0x3 },
1426        },
1427};
1428
1429static const struct snd_kcontrol_new cxt5047_test_mixer[] = {
1430
1431	/* Output only controls */
1432	HDA_CODEC_VOLUME("OutAmp-1 Volume", 0x10, 0x0, HDA_OUTPUT),
1433	HDA_CODEC_MUTE("OutAmp-1 Switch", 0x10,0x0, HDA_OUTPUT),
1434	HDA_CODEC_VOLUME("OutAmp-2 Volume", 0x1c, 0x0, HDA_OUTPUT),
1435	HDA_CODEC_MUTE("OutAmp-2 Switch", 0x1c, 0x0, HDA_OUTPUT),
1436	HDA_CODEC_VOLUME("Speaker Playback Volume", 0x1d, 0x0, HDA_OUTPUT),
1437	HDA_CODEC_MUTE("Speaker Playback Switch", 0x1d, 0x0, HDA_OUTPUT),
1438	HDA_CODEC_VOLUME("HeadPhone Playback Volume", 0x13, 0x0, HDA_OUTPUT),
1439	HDA_CODEC_MUTE("HeadPhone Playback Switch", 0x13, 0x0, HDA_OUTPUT),
1440	HDA_CODEC_VOLUME("Line1-Out Playback Volume", 0x14, 0x0, HDA_OUTPUT),
1441	HDA_CODEC_MUTE("Line1-Out Playback Switch", 0x14, 0x0, HDA_OUTPUT),
1442	HDA_CODEC_VOLUME("Line2-Out Playback Volume", 0x15, 0x0, HDA_OUTPUT),
1443	HDA_CODEC_MUTE("Line2-Out Playback Switch", 0x15, 0x0, HDA_OUTPUT),
1444
1445	/* Modes for retasking pin widgets */
1446	CXT_PIN_MODE("LINE1 pin mode", 0x14, CXT_PIN_DIR_INOUT),
1447	CXT_PIN_MODE("MIC1 pin mode", 0x15, CXT_PIN_DIR_INOUT),
1448
1449	/* EAPD Switch Control */
1450	CXT_EAPD_SWITCH("External Amplifier", 0x13, 0x0),
1451
1452	/* Loopback mixer controls */
1453	HDA_CODEC_VOLUME("MIC1 Playback Volume", 0x12, 0x01, HDA_INPUT),
1454	HDA_CODEC_MUTE("MIC1 Playback Switch", 0x12, 0x01, HDA_INPUT),
1455	HDA_CODEC_VOLUME("MIC2 Playback Volume", 0x12, 0x02, HDA_INPUT),
1456	HDA_CODEC_MUTE("MIC2 Playback Switch", 0x12, 0x02, HDA_INPUT),
1457	HDA_CODEC_VOLUME("LINE Playback Volume", 0x12, 0x0, HDA_INPUT),
1458	HDA_CODEC_MUTE("LINE Playback Switch", 0x12, 0x0, HDA_INPUT),
1459	HDA_CODEC_VOLUME("CD Playback Volume", 0x12, 0x04, HDA_INPUT),
1460	HDA_CODEC_MUTE("CD Playback Switch", 0x12, 0x04, HDA_INPUT),
1461
1462	HDA_CODEC_VOLUME("Capture-1 Volume", 0x19, 0x0, HDA_INPUT),
1463	HDA_CODEC_MUTE("Capture-1 Switch", 0x19, 0x0, HDA_INPUT),
1464	HDA_CODEC_VOLUME("Capture-2 Volume", 0x19, 0x1, HDA_INPUT),
1465	HDA_CODEC_MUTE("Capture-2 Switch", 0x19, 0x1, HDA_INPUT),
1466	HDA_CODEC_VOLUME("Capture-3 Volume", 0x19, 0x2, HDA_INPUT),
1467	HDA_CODEC_MUTE("Capture-3 Switch", 0x19, 0x2, HDA_INPUT),
1468	HDA_CODEC_VOLUME("Capture-4 Volume", 0x19, 0x3, HDA_INPUT),
1469	HDA_CODEC_MUTE("Capture-4 Switch", 0x19, 0x3, HDA_INPUT),
1470	{
1471		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1472		.name = "Input Source",
1473		.info = conexant_mux_enum_info,
1474		.get = conexant_mux_enum_get,
1475		.put = conexant_mux_enum_put,
1476	},
1477	HDA_CODEC_VOLUME("Mic Boost Volume", 0x1a, 0x0, HDA_OUTPUT),
1478
1479	{ } /* end */
1480};
1481
1482static const struct hda_verb cxt5047_test_init_verbs[] = {
1483	/* Enable retasking pins as output, initially without power amp */
1484	{0x15, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1485	{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1486	{0x13, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1487
1488	/* Disable digital (SPDIF) pins initially, but users can enable
1489	 * them via a mixer switch.  In the case of SPDIF-out, this initverb
1490	 * payload also sets the generation to 0, output to be in "consumer"
1491	 * PCM format, copyright asserted, no pre-emphasis and no validity
1492	 * control.
1493	 */
1494	{0x18, AC_VERB_SET_DIGI_CONVERT_1, 0},
1495
1496	/* Ensure mic1, mic2, line1 pin widgets take input from the 
1497	 * OUT1 sum bus when acting as an output.
1498	 */
1499	{0x1a, AC_VERB_SET_CONNECT_SEL, 0},
1500	{0x1b, AC_VERB_SET_CONNECT_SEL, 0},
1501
1502	/* Start with output sum widgets muted and their output gains at min */
1503	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1504	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
1505
1506	/* Unmute retasking pin widget output buffers since the default
1507	 * state appears to be output.  As the pin mode is changed by the
1508	 * user the pin mode control will take care of enabling the pin's
1509	 * input/output buffers as needed.
1510	 */
1511	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1512	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1513	{0x13, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1514
1515	/* Mute capture amp left and right */
1516	{0x12, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
1517
1518	/* Set ADC connection select to match default mixer setting (mic1
1519	 * pin)
1520	 */
1521	{0x12, AC_VERB_SET_CONNECT_SEL, 0x00},
1522
1523	/* Mute all inputs to mixer widget (even unconnected ones) */
1524	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* mic1 pin */
1525	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)}, /* mic2 pin */
1526	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)}, /* line1 pin */
1527	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)}, /* line2 pin */
1528	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)}, /* CD pin */
1529	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(5)}, /* Beep-gen pin */
1530	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(6)}, /* Line-out pin */
1531	{0x19, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(7)}, /* HP-pin pin */
1532
1533	{ }
1534};
1535#endif
1536
1537
1538/* initialize jack-sensing, too */
1539static int cxt5047_hp_init(struct hda_codec *codec)
1540{
1541	conexant_init(codec);
1542	cxt5047_hp_automute(codec);
1543	return 0;
1544}
1545
1546
1547enum {
1548	CXT5047_LAPTOP,		/* Laptops w/o EAPD support */
1549	CXT5047_LAPTOP_HP,	/* Some HP laptops */
1550	CXT5047_LAPTOP_EAPD,	/* Laptops with EAPD support */
1551#ifdef CONFIG_SND_DEBUG
1552	CXT5047_TEST,
1553#endif
1554	CXT5047_AUTO,
1555	CXT5047_MODELS
1556};
1557
1558static const char * const cxt5047_models[CXT5047_MODELS] = {
1559	[CXT5047_LAPTOP]	= "laptop",
1560	[CXT5047_LAPTOP_HP]	= "laptop-hp",
1561	[CXT5047_LAPTOP_EAPD]	= "laptop-eapd",
1562#ifdef CONFIG_SND_DEBUG
1563	[CXT5047_TEST]		= "test",
 
 
 
1564#endif
1565	[CXT5047_AUTO]		= "auto",
1566};
1567
1568static const struct snd_pci_quirk cxt5047_cfg_tbl[] = {
1569	SND_PCI_QUIRK(0x103c, 0x30a5, "HP DV5200T/DV8000T", CXT5047_LAPTOP_HP),
1570	SND_PCI_QUIRK_MASK(0x103c, 0xff00, 0x3000, "HP DV Series",
1571			   CXT5047_LAPTOP),
1572	SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba P100", CXT5047_LAPTOP_EAPD),
1573	{}
1574};
1575
1576static int patch_cxt5047(struct hda_codec *codec)
1577{
1578	struct conexant_spec *spec;
1579	int board_config;
1580
1581	board_config = snd_hda_check_board_config(codec, CXT5047_MODELS,
1582						  cxt5047_models,
1583						  cxt5047_cfg_tbl);
1584	if (board_config < 0)
1585		board_config = CXT5047_AUTO; /* model=auto as default */
1586	if (board_config == CXT5047_AUTO)
1587		return patch_conexant_auto(codec);
1588
1589	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1590	if (!spec)
1591		return -ENOMEM;
1592	codec->spec = spec;
1593	codec->pin_amp_workaround = 1;
1594
1595	spec->multiout.max_channels = 2;
1596	spec->multiout.num_dacs = ARRAY_SIZE(cxt5047_dac_nids);
1597	spec->multiout.dac_nids = cxt5047_dac_nids;
1598	spec->multiout.dig_out_nid = CXT5047_SPDIF_OUT;
1599	spec->num_adc_nids = 1;
1600	spec->adc_nids = cxt5047_adc_nids;
1601	spec->capsrc_nids = cxt5047_capsrc_nids;
1602	spec->num_mixers = 1;
1603	spec->mixers[0] = cxt5047_base_mixers;
1604	spec->num_init_verbs = 1;
1605	spec->init_verbs[0] = cxt5047_init_verbs;
1606	spec->spdif_route = 0;
1607	spec->num_channel_mode = ARRAY_SIZE(cxt5047_modes),
1608	spec->channel_mode = cxt5047_modes,
1609
1610	codec->patch_ops = conexant_patch_ops;
1611
1612	switch (board_config) {
1613	case CXT5047_LAPTOP:
1614		spec->num_mixers = 2;
1615		spec->mixers[1] = cxt5047_hp_spk_mixers;
1616		codec->patch_ops.unsol_event = cxt5047_hp_unsol_event;
1617		break;
1618	case CXT5047_LAPTOP_HP:
1619		spec->num_mixers = 2;
1620		spec->mixers[1] = cxt5047_hp_only_mixers;
1621		codec->patch_ops.unsol_event = cxt5047_hp_unsol_event;
1622		codec->patch_ops.init = cxt5047_hp_init;
1623		break;
1624	case CXT5047_LAPTOP_EAPD:
1625		spec->input_mux = &cxt5047_toshiba_capture_source;
1626		spec->num_mixers = 2;
1627		spec->mixers[1] = cxt5047_hp_spk_mixers;
1628		spec->num_init_verbs = 2;
1629		spec->init_verbs[1] = cxt5047_toshiba_init_verbs;
1630		codec->patch_ops.unsol_event = cxt5047_hp_unsol_event;
1631		break;
1632#ifdef CONFIG_SND_DEBUG
1633	case CXT5047_TEST:
1634		spec->input_mux = &cxt5047_test_capture_source;
1635		spec->mixers[0] = cxt5047_test_mixer;
1636		spec->init_verbs[0] = cxt5047_test_init_verbs;
1637		codec->patch_ops.unsol_event = cxt5047_hp_unsol_event;
1638#endif	
1639	}
1640	spec->vmaster_nid = 0x13;
1641
1642	switch (codec->subsystem_id >> 16) {
1643	case 0x103c:
1644		/* HP laptops have really bad sound over 0 dB on NID 0x10.
1645		 * Fix max PCM level to 0 dB (originally it has 0x1e steps
1646		 * with 0 dB offset 0x17)
1647		 */
1648		snd_hda_override_amp_caps(codec, 0x10, HDA_INPUT,
1649					  (0x17 << AC_AMPCAP_OFFSET_SHIFT) |
1650					  (0x17 << AC_AMPCAP_NUM_STEPS_SHIFT) |
1651					  (0x05 << AC_AMPCAP_STEP_SIZE_SHIFT) |
1652					  (1 << AC_AMPCAP_MUTE_SHIFT));
1653		break;
1654	}
1655
1656	return 0;
1657}
1658
1659/* Conexant 5051 specific */
1660static const hda_nid_t cxt5051_dac_nids[1] = { 0x10 };
1661static const hda_nid_t cxt5051_adc_nids[2] = { 0x14, 0x15 };
1662
1663static const struct hda_channel_mode cxt5051_modes[1] = {
1664	{ 2, NULL },
1665};
1666
1667static void cxt5051_update_speaker(struct hda_codec *codec)
1668{
1669	struct conexant_spec *spec = codec->spec;
1670	unsigned int pinctl;
1671	/* headphone pin */
1672	pinctl = (spec->hp_present && spec->cur_eapd) ? PIN_HP : 0;
1673	snd_hda_codec_write(codec, 0x16, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
1674			    pinctl);
1675	/* speaker pin */
1676	pinctl = (!spec->hp_present && spec->cur_eapd) ? PIN_OUT : 0;
1677	snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
1678			    pinctl);
1679	/* on ideapad there is an aditional speaker (subwoofer) to mute */
1680	if (spec->ideapad)
1681		snd_hda_codec_write(codec, 0x1b, 0,
1682				    AC_VERB_SET_PIN_WIDGET_CONTROL,
1683				    pinctl);
1684}
1685
1686/* turn on/off EAPD (+ mute HP) as a master switch */
1687static int cxt5051_hp_master_sw_put(struct snd_kcontrol *kcontrol,
1688				    struct snd_ctl_elem_value *ucontrol)
1689{
1690	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 
1691
1692	if (!cxt_eapd_put(kcontrol, ucontrol))
1693		return 0;
1694	cxt5051_update_speaker(codec);
1695	return 1;
 
1696}
1697
1698/* toggle input of built-in and mic jack appropriately */
1699static void cxt5051_portb_automic(struct hda_codec *codec)
1700{
 
 
 
1701	struct conexant_spec *spec = codec->spec;
1702	unsigned int present;
1703
1704	if (!(spec->auto_mic & AUTO_MIC_PORTB))
1705		return;
1706	present = snd_hda_jack_detect(codec, 0x17);
1707	snd_hda_codec_write(codec, 0x14, 0,
1708			    AC_VERB_SET_CONNECT_SEL,
1709			    present ? 0x01 : 0x00);
1710}
1711
1712/* switch the current ADC according to the jack state */
1713static void cxt5051_portc_automic(struct hda_codec *codec)
1714{
1715	struct conexant_spec *spec = codec->spec;
1716	unsigned int present;
1717	hda_nid_t new_adc;
1718
1719	if (!(spec->auto_mic & AUTO_MIC_PORTC))
1720		return;
1721	present = snd_hda_jack_detect(codec, 0x18);
1722	if (present)
1723		spec->cur_adc_idx = 1;
1724	else
1725		spec->cur_adc_idx = 0;
1726	new_adc = spec->adc_nids[spec->cur_adc_idx];
1727	if (spec->cur_adc && spec->cur_adc != new_adc) {
1728		/* stream is running, let's swap the current ADC */
1729		__snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
1730		spec->cur_adc = new_adc;
1731		snd_hda_codec_setup_stream(codec, new_adc,
1732					   spec->cur_adc_stream_tag, 0,
1733					   spec->cur_adc_format);
1734	}
1735}
1736
1737/* mute internal speaker if HP is plugged */
1738static void cxt5051_hp_automute(struct hda_codec *codec)
1739{
1740	struct conexant_spec *spec = codec->spec;
1741
1742	spec->hp_present = snd_hda_jack_detect(codec, 0x16);
1743	cxt5051_update_speaker(codec);
1744}
1745
1746/* unsolicited event for HP jack sensing */
1747static void cxt5051_hp_unsol_event(struct hda_codec *codec,
1748				   unsigned int res)
1749{
1750	int nid = (res & AC_UNSOL_RES_SUBTAG) >> 20;
1751	switch (res >> 26) {
1752	case CONEXANT_HP_EVENT:
1753		cxt5051_hp_automute(codec);
1754		break;
1755	case CXT5051_PORTB_EVENT:
1756		cxt5051_portb_automic(codec);
1757		break;
1758	case CXT5051_PORTC_EVENT:
1759		cxt5051_portc_automic(codec);
1760		break;
1761	}
1762	snd_hda_input_jack_report(codec, nid);
1763}
1764
1765static const struct snd_kcontrol_new cxt5051_playback_mixers[] = {
1766	HDA_CODEC_VOLUME("Master Playback Volume", 0x10, 0x00, HDA_OUTPUT),
1767	{
1768		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1769		.name = "Master Playback Switch",
1770		.info = cxt_eapd_info,
1771		.get = cxt_eapd_get,
1772		.put = cxt5051_hp_master_sw_put,
1773		.private_value = 0x1a,
1774	},
1775	{}
1776};
1777
1778static const struct snd_kcontrol_new cxt5051_capture_mixers[] = {
1779	HDA_CODEC_VOLUME("Internal Mic Volume", 0x14, 0x00, HDA_INPUT),
1780	HDA_CODEC_MUTE("Internal Mic Switch", 0x14, 0x00, HDA_INPUT),
1781	HDA_CODEC_VOLUME("Mic Volume", 0x14, 0x01, HDA_INPUT),
1782	HDA_CODEC_MUTE("Mic Switch", 0x14, 0x01, HDA_INPUT),
1783	HDA_CODEC_VOLUME("Docking Mic Volume", 0x15, 0x00, HDA_INPUT),
1784	HDA_CODEC_MUTE("Docking Mic Switch", 0x15, 0x00, HDA_INPUT),
1785	{}
1786};
1787
1788static const struct snd_kcontrol_new cxt5051_hp_mixers[] = {
1789	HDA_CODEC_VOLUME("Internal Mic Volume", 0x14, 0x00, HDA_INPUT),
1790	HDA_CODEC_MUTE("Internal Mic Switch", 0x14, 0x00, HDA_INPUT),
1791	HDA_CODEC_VOLUME("Mic Volume", 0x15, 0x00, HDA_INPUT),
1792	HDA_CODEC_MUTE("Mic Switch", 0x15, 0x00, HDA_INPUT),
1793	{}
1794};
1795
1796static const struct snd_kcontrol_new cxt5051_hp_dv6736_mixers[] = {
1797	HDA_CODEC_VOLUME("Capture Volume", 0x14, 0x00, HDA_INPUT),
1798	HDA_CODEC_MUTE("Capture Switch", 0x14, 0x00, HDA_INPUT),
1799	{}
1800};
1801
1802static const struct snd_kcontrol_new cxt5051_f700_mixers[] = {
1803	HDA_CODEC_VOLUME("Capture Volume", 0x14, 0x01, HDA_INPUT),
1804	HDA_CODEC_MUTE("Capture Switch", 0x14, 0x01, HDA_INPUT),
1805	{}
1806};
1807
1808static const struct snd_kcontrol_new cxt5051_toshiba_mixers[] = {
1809	HDA_CODEC_VOLUME("Internal Mic Volume", 0x14, 0x00, HDA_INPUT),
1810	HDA_CODEC_MUTE("Internal Mic Switch", 0x14, 0x00, HDA_INPUT),
1811	HDA_CODEC_VOLUME("Mic Volume", 0x14, 0x01, HDA_INPUT),
1812	HDA_CODEC_MUTE("Mic Switch", 0x14, 0x01, HDA_INPUT),
1813	{}
1814};
1815
1816static const struct hda_verb cxt5051_init_verbs[] = {
1817	/* Line in, Mic */
1818	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1819	{0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1820	{0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1821	{0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1822	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1823	{0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1824	/* SPK  */
1825	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1826	{0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
1827	/* HP, Amp  */
1828	{0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1829	{0x16, AC_VERB_SET_CONNECT_SEL, 0x00},
1830	/* DAC1 */	
1831	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1832	/* Record selector: Internal mic */
1833	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x44},
1834	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1) | 0x44},
1835	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x44},
1836	/* SPDIF route: PCM */
1837	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1838	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x0},
1839	/* EAPD */
1840	{0x1a, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */ 
1841	{0x16, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|CONEXANT_HP_EVENT},
1842	{ } /* end */
1843};
1844
1845static const struct hda_verb cxt5051_hp_dv6736_init_verbs[] = {
1846	/* Line in, Mic */
1847	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1848	{0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1849	{0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0},
1850	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0},
1851	/* SPK  */
1852	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1853	{0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
1854	/* HP, Amp  */
1855	{0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1856	{0x16, AC_VERB_SET_CONNECT_SEL, 0x00},
1857	/* DAC1 */
1858	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1859	/* Record selector: Internal mic */
1860	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1) | 0x44},
1861	{0x14, AC_VERB_SET_CONNECT_SEL, 0x1},
1862	/* SPDIF route: PCM */
1863	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x0},
1864	/* EAPD */
1865	{0x1a, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
1866	{0x16, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|CONEXANT_HP_EVENT},
1867	{ } /* end */
1868};
1869
1870static const struct hda_verb cxt5051_lenovo_x200_init_verbs[] = {
1871	/* Line in, Mic */
1872	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1873	{0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1874	{0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1875	{0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1876	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
1877	{0x1d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1878	/* SPK  */
1879	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1880	{0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
1881	/* HP, Amp  */
1882	{0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1883	{0x16, AC_VERB_SET_CONNECT_SEL, 0x00},
1884	/* Docking HP */
1885	{0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1886	{0x19, AC_VERB_SET_CONNECT_SEL, 0x00},
1887	/* DAC1 */
1888	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1889	/* Record selector: Internal mic */
1890	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x44},
1891	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1) | 0x44},
1892	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x44},
1893	/* SPDIF route: PCM */
1894	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT}, /* needed for W500 Advanced Mini Dock 250410 */
1895	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x0},
1896	/* EAPD */
1897	{0x1a, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
1898	{0x16, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|CONEXANT_HP_EVENT},
1899	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|CONEXANT_HP_EVENT},
1900	{ } /* end */
1901};
1902
1903static const struct hda_verb cxt5051_f700_init_verbs[] = {
1904	/* Line in, Mic */
1905	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x03},
1906	{0x17, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
1907	{0x18, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0},
1908	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x0},
1909	/* SPK  */
1910	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1911	{0x1a, AC_VERB_SET_CONNECT_SEL, 0x00},
1912	/* HP, Amp  */
1913	{0x16, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
1914	{0x16, AC_VERB_SET_CONNECT_SEL, 0x00},
1915	/* DAC1 */
1916	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
1917	/* Record selector: Internal mic */
1918	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(1) | 0x44},
1919	{0x14, AC_VERB_SET_CONNECT_SEL, 0x1},
1920	/* SPDIF route: PCM */
1921	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x0},
1922	/* EAPD */
1923	{0x1a, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
1924	{0x16, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN|CONEXANT_HP_EVENT},
1925	{ } /* end */
1926};
1927
1928static void cxt5051_init_mic_port(struct hda_codec *codec, hda_nid_t nid,
1929				 unsigned int event)
1930{
1931	snd_hda_codec_write(codec, nid, 0,
1932			    AC_VERB_SET_UNSOLICITED_ENABLE,
1933			    AC_USRSP_EN | event);
1934	snd_hda_input_jack_add(codec, nid, SND_JACK_MICROPHONE, NULL);
1935	snd_hda_input_jack_report(codec, nid);
1936}
1937
1938static const struct hda_verb cxt5051_ideapad_init_verbs[] = {
1939	/* Subwoofer */
1940	{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
1941	{0x1b, AC_VERB_SET_CONNECT_SEL, 0x00},
1942	{ } /* end */
1943};
1944
1945/* initialize jack-sensing, too */
1946static int cxt5051_init(struct hda_codec *codec)
1947{
1948	struct conexant_spec *spec = codec->spec;
1949
1950	conexant_init(codec);
1951	conexant_init_jacks(codec);
1952
1953	if (spec->auto_mic & AUTO_MIC_PORTB)
1954		cxt5051_init_mic_port(codec, 0x17, CXT5051_PORTB_EVENT);
1955	if (spec->auto_mic & AUTO_MIC_PORTC)
1956		cxt5051_init_mic_port(codec, 0x18, CXT5051_PORTC_EVENT);
1957
1958	if (codec->patch_ops.unsol_event) {
1959		cxt5051_hp_automute(codec);
1960		cxt5051_portb_automic(codec);
1961		cxt5051_portc_automic(codec);
1962	}
1963	return 0;
1964}
1965
1966
1967enum {
1968	CXT5051_LAPTOP,	 /* Laptops w/ EAPD support */
1969	CXT5051_HP,	/* no docking */
1970	CXT5051_HP_DV6736,	/* HP without mic switch */
1971	CXT5051_LENOVO_X200,	/* Lenovo X200 laptop, also used for Advanced Mini Dock 250410 */
1972	CXT5051_F700,       /* HP Compaq Presario F700 */
1973	CXT5051_TOSHIBA,	/* Toshiba M300 & co */
1974	CXT5051_IDEAPAD,	/* Lenovo IdeaPad Y430 */
1975	CXT5051_AUTO,		/* auto-parser */
1976	CXT5051_MODELS
1977};
1978
1979static const char *const cxt5051_models[CXT5051_MODELS] = {
1980	[CXT5051_LAPTOP]	= "laptop",
1981	[CXT5051_HP]		= "hp",
1982	[CXT5051_HP_DV6736]	= "hp-dv6736",
1983	[CXT5051_LENOVO_X200]	= "lenovo-x200",
1984	[CXT5051_F700]          = "hp-700",
1985	[CXT5051_TOSHIBA]	= "toshiba",
1986	[CXT5051_IDEAPAD]	= "ideapad",
1987	[CXT5051_AUTO]		= "auto",
1988};
1989
1990static const struct snd_pci_quirk cxt5051_cfg_tbl[] = {
1991	SND_PCI_QUIRK(0x103c, 0x30cf, "HP DV6736", CXT5051_HP_DV6736),
1992	SND_PCI_QUIRK(0x103c, 0x360b, "Compaq Presario CQ60", CXT5051_HP),
1993	SND_PCI_QUIRK(0x103c, 0x30ea, "Compaq Presario F700", CXT5051_F700),
1994	SND_PCI_QUIRK(0x1179, 0xff50, "Toshiba M30x", CXT5051_TOSHIBA),
1995	SND_PCI_QUIRK(0x14f1, 0x0101, "Conexant Reference board",
1996		      CXT5051_LAPTOP),
1997	SND_PCI_QUIRK(0x14f1, 0x5051, "HP Spartan 1.1", CXT5051_HP),
1998	SND_PCI_QUIRK(0x17aa, 0x20f2, "Lenovo X200", CXT5051_LENOVO_X200),
1999	SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo IdeaPad", CXT5051_IDEAPAD),
2000	{}
2001};
2002
2003static int patch_cxt5051(struct hda_codec *codec)
2004{
2005	struct conexant_spec *spec;
2006	int board_config;
2007
2008	board_config = snd_hda_check_board_config(codec, CXT5051_MODELS,
2009						  cxt5051_models,
2010						  cxt5051_cfg_tbl);
2011	if (board_config < 0)
2012		board_config = CXT5051_AUTO; /* model=auto as default */
2013	if (board_config == CXT5051_AUTO)
2014		return patch_conexant_auto(codec);
2015
2016	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
2017	if (!spec)
2018		return -ENOMEM;
2019	codec->spec = spec;
2020	codec->pin_amp_workaround = 1;
2021
2022	codec->patch_ops = conexant_patch_ops;
2023	codec->patch_ops.init = cxt5051_init;
2024
2025	spec->multiout.max_channels = 2;
2026	spec->multiout.num_dacs = ARRAY_SIZE(cxt5051_dac_nids);
2027	spec->multiout.dac_nids = cxt5051_dac_nids;
2028	spec->multiout.dig_out_nid = CXT5051_SPDIF_OUT;
2029	spec->num_adc_nids = 1; /* not 2; via auto-mic switch */
2030	spec->adc_nids = cxt5051_adc_nids;
2031	spec->num_mixers = 2;
2032	spec->mixers[0] = cxt5051_capture_mixers;
2033	spec->mixers[1] = cxt5051_playback_mixers;
2034	spec->num_init_verbs = 1;
2035	spec->init_verbs[0] = cxt5051_init_verbs;
2036	spec->spdif_route = 0;
2037	spec->num_channel_mode = ARRAY_SIZE(cxt5051_modes);
2038	spec->channel_mode = cxt5051_modes;
2039	spec->cur_adc = 0;
2040	spec->cur_adc_idx = 0;
2041
2042	set_beep_amp(spec, 0x13, 0, HDA_OUTPUT);
2043
2044	codec->patch_ops.unsol_event = cxt5051_hp_unsol_event;
2045
2046	spec->auto_mic = AUTO_MIC_PORTB | AUTO_MIC_PORTC;
2047	switch (board_config) {
2048	case CXT5051_HP:
2049		spec->mixers[0] = cxt5051_hp_mixers;
2050		break;
2051	case CXT5051_HP_DV6736:
2052		spec->init_verbs[0] = cxt5051_hp_dv6736_init_verbs;
2053		spec->mixers[0] = cxt5051_hp_dv6736_mixers;
2054		spec->auto_mic = 0;
2055		break;
2056	case CXT5051_LENOVO_X200:
2057		spec->init_verbs[0] = cxt5051_lenovo_x200_init_verbs;
2058		/* Thinkpad X301 does not have S/PDIF wired and no ability
2059		   to use a docking station. */
2060		if (codec->subsystem_id == 0x17aa211f)
2061			spec->multiout.dig_out_nid = 0;
2062		break;
2063	case CXT5051_F700:
2064		spec->init_verbs[0] = cxt5051_f700_init_verbs;
2065		spec->mixers[0] = cxt5051_f700_mixers;
2066		spec->auto_mic = 0;
2067		break;
2068	case CXT5051_TOSHIBA:
2069		spec->mixers[0] = cxt5051_toshiba_mixers;
2070		spec->auto_mic = AUTO_MIC_PORTB;
2071		break;
2072	case CXT5051_IDEAPAD:
2073		spec->init_verbs[spec->num_init_verbs++] =
2074			cxt5051_ideapad_init_verbs;
2075		spec->ideapad = 1;
2076		break;
2077	}
2078
2079	if (spec->beep_amp)
2080		snd_hda_attach_beep_device(codec, spec->beep_amp);
2081
2082	return 0;
2083}
2084
2085/* Conexant 5066 specific */
2086
2087static const hda_nid_t cxt5066_dac_nids[1] = { 0x10 };
2088static const hda_nid_t cxt5066_adc_nids[3] = { 0x14, 0x15, 0x16 };
2089static const hda_nid_t cxt5066_capsrc_nids[1] = { 0x17 };
2090static const hda_nid_t cxt5066_digout_pin_nids[2] = { 0x20, 0x22 };
2091
2092/* OLPC's microphone port is DC coupled for use with external sensors,
2093 * therefore we use a 50% mic bias in order to center the input signal with
2094 * the DC input range of the codec. */
2095#define CXT5066_OLPC_EXT_MIC_BIAS PIN_VREF50
2096
2097static const struct hda_channel_mode cxt5066_modes[1] = {
2098	{ 2, NULL },
2099};
2100
2101#define HP_PRESENT_PORT_A	(1 << 0)
2102#define HP_PRESENT_PORT_D	(1 << 1)
2103#define hp_port_a_present(spec)	((spec)->hp_present & HP_PRESENT_PORT_A)
2104#define hp_port_d_present(spec)	((spec)->hp_present & HP_PRESENT_PORT_D)
2105
2106static void cxt5066_update_speaker(struct hda_codec *codec)
2107{
2108	struct conexant_spec *spec = codec->spec;
2109	unsigned int pinctl;
2110
2111	snd_printdd("CXT5066: update speaker, hp_present=%d, cur_eapd=%d\n",
2112		    spec->hp_present, spec->cur_eapd);
2113
2114	/* Port A (HP) */
2115	pinctl = (hp_port_a_present(spec) && spec->cur_eapd) ? PIN_HP : 0;
2116	snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
2117			pinctl);
2118
2119	/* Port D (HP/LO) */
2120	pinctl = spec->cur_eapd ? spec->port_d_mode : 0;
2121	if (spec->dell_automute || spec->thinkpad) {
2122		/* Mute if Port A is connected */
2123		if (hp_port_a_present(spec))
2124			pinctl = 0;
2125	} else {
2126		/* Thinkpad/Dell doesn't give pin-D status */
2127		if (!hp_port_d_present(spec))
2128			pinctl = 0;
2129	}
2130	snd_hda_codec_write(codec, 0x1c, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
2131			pinctl);
2132
2133	/* CLASS_D AMP */
2134	pinctl = (!spec->hp_present && spec->cur_eapd) ? PIN_OUT : 0;
2135	snd_hda_codec_write(codec, 0x1f, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
2136			pinctl);
2137}
2138
2139/* turn on/off EAPD (+ mute HP) as a master switch */
2140static int cxt5066_hp_master_sw_put(struct snd_kcontrol *kcontrol,
2141				    struct snd_ctl_elem_value *ucontrol)
2142{
2143	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2144
2145	if (!cxt_eapd_put(kcontrol, ucontrol))
2146		return 0;
 
 
 
 
2147
2148	cxt5066_update_speaker(codec);
2149	return 1;
2150}
2151
2152static const struct hda_input_mux cxt5066_olpc_dc_bias = {
2153	.num_items = 3,
2154	.items = {
2155		{ "Off", PIN_IN },
2156		{ "50%", PIN_VREF50 },
2157		{ "80%", PIN_VREF80 },
2158	},
2159};
2160
2161static int cxt5066_set_olpc_dc_bias(struct hda_codec *codec)
2162{
2163	struct conexant_spec *spec = codec->spec;
2164	/* Even though port F is the DC input, the bias is controlled on port B.
2165	 * we also leave that port as an active input (but unselected) in DC mode
2166	 * just in case that is necessary to make the bias setting take effect. */
2167	return snd_hda_codec_write_cache(codec, 0x1a, 0,
2168		AC_VERB_SET_PIN_WIDGET_CONTROL,
2169		cxt5066_olpc_dc_bias.items[spec->dc_input_bias].index);
2170}
2171
2172/* OLPC defers mic widget control until when capture is started because the
2173 * microphone LED comes on as soon as these settings are put in place. if we
2174 * did this before recording, it would give the false indication that recording
2175 * is happening when it is not. */
2176static void cxt5066_olpc_select_mic(struct hda_codec *codec)
2177{
2178	struct conexant_spec *spec = codec->spec;
2179	if (!spec->recording)
2180		return;
2181
2182	if (spec->dc_enable) {
2183		/* in DC mode we ignore presence detection and just use the jack
2184		 * through our special DC port */
2185		const struct hda_verb enable_dc_mode[] = {
2186			/* disble internal mic, port C */
2187			{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2188
2189			/* enable DC capture, port F */
2190			{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2191			{},
2192		};
2193
2194		snd_hda_sequence_write(codec, enable_dc_mode);
2195		/* port B input disabled (and bias set) through the following call */
2196		cxt5066_set_olpc_dc_bias(codec);
2197		return;
2198	}
2199
2200	/* disable DC (port F) */
2201	snd_hda_codec_write(codec, 0x1e, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, 0);
2202
2203	/* external mic, port B */
2204	snd_hda_codec_write(codec, 0x1a, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
2205		spec->ext_mic_present ? CXT5066_OLPC_EXT_MIC_BIAS : 0);
2206
2207	/* internal mic, port C */
2208	snd_hda_codec_write(codec, 0x1b, 0, AC_VERB_SET_PIN_WIDGET_CONTROL,
2209		spec->ext_mic_present ? 0 : PIN_VREF80);
2210}
2211
2212/* toggle input of built-in and mic jack appropriately */
2213static void cxt5066_olpc_automic(struct hda_codec *codec)
2214{
2215	struct conexant_spec *spec = codec->spec;
2216	unsigned int present;
2217
2218	if (spec->dc_enable) /* don't do presence detection in DC mode */
2219		return;
2220
2221	present = snd_hda_codec_read(codec, 0x1a, 0,
2222				     AC_VERB_GET_PIN_SENSE, 0) & 0x80000000;
2223	if (present)
2224		snd_printdd("CXT5066: external microphone detected\n");
2225	else
2226		snd_printdd("CXT5066: external microphone absent\n");
2227
2228	snd_hda_codec_write(codec, 0x17, 0, AC_VERB_SET_CONNECT_SEL,
2229		present ? 0 : 1);
2230	spec->ext_mic_present = !!present;
2231
2232	cxt5066_olpc_select_mic(codec);
2233}
2234
2235/* toggle input of built-in digital mic and mic jack appropriately */
2236static void cxt5066_vostro_automic(struct hda_codec *codec)
2237{
2238	unsigned int present;
2239
2240	struct hda_verb ext_mic_present[] = {
2241		/* enable external mic, port B */
2242		{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2243
2244		/* switch to external mic input */
2245		{0x17, AC_VERB_SET_CONNECT_SEL, 0},
2246		{0x14, AC_VERB_SET_CONNECT_SEL, 0},
2247
2248		/* disable internal digital mic */
2249		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2250		{}
2251	};
2252	static const struct hda_verb ext_mic_absent[] = {
2253		/* enable internal mic, port C */
2254		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2255
2256		/* switch to internal mic input */
2257		{0x14, AC_VERB_SET_CONNECT_SEL, 2},
2258
2259		/* disable external mic, port B */
2260		{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2261		{}
2262	};
2263
2264	present = snd_hda_jack_detect(codec, 0x1a);
2265	if (present) {
2266		snd_printdd("CXT5066: external microphone detected\n");
2267		snd_hda_sequence_write(codec, ext_mic_present);
2268	} else {
2269		snd_printdd("CXT5066: external microphone absent\n");
2270		snd_hda_sequence_write(codec, ext_mic_absent);
2271	}
2272}
2273
2274/* toggle input of built-in digital mic and mic jack appropriately */
2275static void cxt5066_ideapad_automic(struct hda_codec *codec)
2276{
2277	unsigned int present;
2278
2279	struct hda_verb ext_mic_present[] = {
2280		{0x14, AC_VERB_SET_CONNECT_SEL, 0},
2281		{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2282		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2283		{}
2284	};
2285	static const struct hda_verb ext_mic_absent[] = {
2286		{0x14, AC_VERB_SET_CONNECT_SEL, 2},
2287		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2288		{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2289		{}
2290	};
2291
2292	present = snd_hda_jack_detect(codec, 0x1b);
2293	if (present) {
2294		snd_printdd("CXT5066: external microphone detected\n");
2295		snd_hda_sequence_write(codec, ext_mic_present);
2296	} else {
2297		snd_printdd("CXT5066: external microphone absent\n");
2298		snd_hda_sequence_write(codec, ext_mic_absent);
2299	}
2300}
2301
2302
2303/* toggle input of built-in digital mic and mic jack appropriately */
2304static void cxt5066_asus_automic(struct hda_codec *codec)
2305{
2306	unsigned int present;
2307
2308	present = snd_hda_jack_detect(codec, 0x1b);
2309	snd_printdd("CXT5066: external microphone present=%d\n", present);
2310	snd_hda_codec_write(codec, 0x17, 0, AC_VERB_SET_CONNECT_SEL,
2311			    present ? 1 : 0);
2312}
2313
2314
2315/* toggle input of built-in digital mic and mic jack appropriately */
2316static void cxt5066_hp_laptop_automic(struct hda_codec *codec)
2317{
2318	unsigned int present;
2319
2320	present = snd_hda_jack_detect(codec, 0x1b);
2321	snd_printdd("CXT5066: external microphone present=%d\n", present);
2322	snd_hda_codec_write(codec, 0x17, 0, AC_VERB_SET_CONNECT_SEL,
2323			    present ? 1 : 3);
2324}
2325
2326
2327/* toggle input of built-in digital mic and mic jack appropriately
2328   order is: external mic -> dock mic -> interal mic */
2329static void cxt5066_thinkpad_automic(struct hda_codec *codec)
2330{
2331	unsigned int ext_present, dock_present;
2332
2333	static const struct hda_verb ext_mic_present[] = {
2334		{0x14, AC_VERB_SET_CONNECT_SEL, 0},
2335		{0x17, AC_VERB_SET_CONNECT_SEL, 1},
2336		{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2337		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2338		{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2339		{}
2340	};
2341	static const struct hda_verb dock_mic_present[] = {
2342		{0x14, AC_VERB_SET_CONNECT_SEL, 0},
2343		{0x17, AC_VERB_SET_CONNECT_SEL, 0},
2344		{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80},
2345		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2346		{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2347		{}
2348	};
2349	static const struct hda_verb ext_mic_absent[] = {
2350		{0x14, AC_VERB_SET_CONNECT_SEL, 2},
2351		{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2352		{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2353		{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2354		{}
2355	};
2356
2357	ext_present = snd_hda_jack_detect(codec, 0x1b);
2358	dock_present = snd_hda_jack_detect(codec, 0x1a);
2359	if (ext_present) {
2360		snd_printdd("CXT5066: external microphone detected\n");
2361		snd_hda_sequence_write(codec, ext_mic_present);
2362	} else if (dock_present) {
2363		snd_printdd("CXT5066: dock microphone detected\n");
2364		snd_hda_sequence_write(codec, dock_mic_present);
2365	} else {
2366		snd_printdd("CXT5066: external microphone absent\n");
2367		snd_hda_sequence_write(codec, ext_mic_absent);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
2368	}
2369}
2370
2371/* mute internal speaker if HP is plugged */
2372static void cxt5066_hp_automute(struct hda_codec *codec)
2373{
2374	struct conexant_spec *spec = codec->spec;
2375	unsigned int portA, portD;
2376
2377	/* Port A */
2378	portA = snd_hda_jack_detect(codec, 0x19);
2379
2380	/* Port D */
2381	portD = snd_hda_jack_detect(codec, 0x1c);
2382
2383	spec->hp_present = portA ? HP_PRESENT_PORT_A : 0;
2384	spec->hp_present |= portD ? HP_PRESENT_PORT_D : 0;
2385	snd_printdd("CXT5066: hp automute portA=%x portD=%x present=%d\n",
2386		portA, portD, spec->hp_present);
2387	cxt5066_update_speaker(codec);
2388}
2389
2390/* Dispatch the right mic autoswitch function */
2391static void cxt5066_automic(struct hda_codec *codec)
2392{
2393	struct conexant_spec *spec = codec->spec;
2394
2395	if (spec->dell_vostro)
2396		cxt5066_vostro_automic(codec);
2397	else if (spec->ideapad)
2398		cxt5066_ideapad_automic(codec);
2399	else if (spec->thinkpad)
2400		cxt5066_thinkpad_automic(codec);
2401	else if (spec->hp_laptop)
2402		cxt5066_hp_laptop_automic(codec);
2403	else if (spec->asus)
2404		cxt5066_asus_automic(codec);
2405}
2406
2407/* unsolicited event for jack sensing */
2408static void cxt5066_olpc_unsol_event(struct hda_codec *codec, unsigned int res)
 
 
 
2409{
2410	struct conexant_spec *spec = codec->spec;
2411	snd_printdd("CXT5066: unsol event %x (%x)\n", res, res >> 26);
2412	switch (res >> 26) {
2413	case CONEXANT_HP_EVENT:
2414		cxt5066_hp_automute(codec);
2415		break;
2416	case CONEXANT_MIC_EVENT:
2417		/* ignore mic events in DC mode; we're always using the jack */
2418		if (!spec->dc_enable)
2419			cxt5066_olpc_automic(codec);
2420		break;
2421	}
2422}
2423
2424/* unsolicited event for jack sensing */
2425static void cxt5066_unsol_event(struct hda_codec *codec, unsigned int res)
2426{
2427	snd_printdd("CXT5066: unsol event %x (%x)\n", res, res >> 26);
2428	switch (res >> 26) {
2429	case CONEXANT_HP_EVENT:
2430		cxt5066_hp_automute(codec);
2431		break;
2432	case CONEXANT_MIC_EVENT:
2433		cxt5066_automic(codec);
 
2434		break;
2435	}
2436}
2437
2438
2439static const struct hda_input_mux cxt5066_analog_mic_boost = {
2440	.num_items = 5,
2441	.items = {
2442		{ "0dB",  0 },
2443		{ "10dB", 1 },
2444		{ "20dB", 2 },
2445		{ "30dB", 3 },
2446		{ "40dB", 4 },
2447	},
2448};
2449
2450static void cxt5066_set_mic_boost(struct hda_codec *codec)
2451{
2452	struct conexant_spec *spec = codec->spec;
2453	snd_hda_codec_write_cache(codec, 0x17, 0,
2454		AC_VERB_SET_AMP_GAIN_MUTE,
2455		AC_AMP_SET_RIGHT | AC_AMP_SET_LEFT | AC_AMP_SET_OUTPUT |
2456			cxt5066_analog_mic_boost.items[spec->mic_boost].index);
2457	if (spec->ideapad || spec->thinkpad) {
2458		/* adjust the internal mic as well...it is not through 0x17 */
2459		snd_hda_codec_write_cache(codec, 0x23, 0,
2460			AC_VERB_SET_AMP_GAIN_MUTE,
2461			AC_AMP_SET_RIGHT | AC_AMP_SET_LEFT | AC_AMP_SET_INPUT |
2462				cxt5066_analog_mic_boost.
2463					items[spec->mic_boost].index);
2464	}
2465}
2466
2467static int cxt5066_mic_boost_mux_enum_info(struct snd_kcontrol *kcontrol,
2468					   struct snd_ctl_elem_info *uinfo)
2469{
2470	return snd_hda_input_mux_info(&cxt5066_analog_mic_boost, uinfo);
2471}
2472
2473static int cxt5066_mic_boost_mux_enum_get(struct snd_kcontrol *kcontrol,
2474					  struct snd_ctl_elem_value *ucontrol)
2475{
2476	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2477	struct conexant_spec *spec = codec->spec;
2478	ucontrol->value.enumerated.item[0] = spec->mic_boost;
2479	return 0;
2480}
2481
2482static int cxt5066_mic_boost_mux_enum_put(struct snd_kcontrol *kcontrol,
2483					  struct snd_ctl_elem_value *ucontrol)
2484{
2485	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2486	struct conexant_spec *spec = codec->spec;
2487	const struct hda_input_mux *imux = &cxt5066_analog_mic_boost;
2488	unsigned int idx;
2489	idx = ucontrol->value.enumerated.item[0];
2490	if (idx >= imux->num_items)
2491		idx = imux->num_items - 1;
2492
2493	spec->mic_boost = idx;
2494	if (!spec->dc_enable)
2495		cxt5066_set_mic_boost(codec);
2496	return 1;
2497}
2498
2499static void cxt5066_enable_dc(struct hda_codec *codec)
2500{
2501	const struct hda_verb enable_dc_mode[] = {
2502		/* disable gain */
2503		{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2504
2505		/* switch to DC input */
2506		{0x17, AC_VERB_SET_CONNECT_SEL, 3},
2507		{}
2508	};
2509
2510	/* configure as input source */
2511	snd_hda_sequence_write(codec, enable_dc_mode);
2512	cxt5066_olpc_select_mic(codec); /* also sets configured bias */
2513}
2514
2515static void cxt5066_disable_dc(struct hda_codec *codec)
2516{
2517	/* reconfigure input source */
2518	cxt5066_set_mic_boost(codec);
2519	/* automic also selects the right mic if we're recording */
2520	cxt5066_olpc_automic(codec);
2521}
2522
2523static int cxt5066_olpc_dc_get(struct snd_kcontrol *kcontrol,
2524			     struct snd_ctl_elem_value *ucontrol)
2525{
2526	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2527	struct conexant_spec *spec = codec->spec;
2528	ucontrol->value.integer.value[0] = spec->dc_enable;
2529	return 0;
2530}
2531
2532static int cxt5066_olpc_dc_put(struct snd_kcontrol *kcontrol,
2533			     struct snd_ctl_elem_value *ucontrol)
2534{
2535	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2536	struct conexant_spec *spec = codec->spec;
2537	int dc_enable = !!ucontrol->value.integer.value[0];
2538
2539	if (dc_enable == spec->dc_enable)
2540		return 0;
2541
2542	spec->dc_enable = dc_enable;
2543	if (dc_enable)
2544		cxt5066_enable_dc(codec);
2545	else
2546		cxt5066_disable_dc(codec);
2547
2548	return 1;
2549}
2550
2551static int cxt5066_olpc_dc_bias_enum_info(struct snd_kcontrol *kcontrol,
2552					   struct snd_ctl_elem_info *uinfo)
2553{
2554	return snd_hda_input_mux_info(&cxt5066_olpc_dc_bias, uinfo);
2555}
2556
2557static int cxt5066_olpc_dc_bias_enum_get(struct snd_kcontrol *kcontrol,
2558					  struct snd_ctl_elem_value *ucontrol)
2559{
2560	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2561	struct conexant_spec *spec = codec->spec;
2562	ucontrol->value.enumerated.item[0] = spec->dc_input_bias;
2563	return 0;
2564}
2565
2566static int cxt5066_olpc_dc_bias_enum_put(struct snd_kcontrol *kcontrol,
2567					  struct snd_ctl_elem_value *ucontrol)
 
 
 
 
 
 
2568{
2569	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
2570	struct conexant_spec *spec = codec->spec;
2571	const struct hda_input_mux *imux = &cxt5066_analog_mic_boost;
2572	unsigned int idx;
2573
2574	idx = ucontrol->value.enumerated.item[0];
2575	if (idx >= imux->num_items)
2576		idx = imux->num_items - 1;
 
 
2577
2578	spec->dc_input_bias = idx;
2579	if (spec->dc_enable)
2580		cxt5066_set_olpc_dc_bias(codec);
2581	return 1;
2582}
2583
2584static void cxt5066_olpc_capture_prepare(struct hda_codec *codec)
2585{
2586	struct conexant_spec *spec = codec->spec;
2587	/* mark as recording and configure the microphone widget so that the
2588	 * recording LED comes on. */
2589	spec->recording = 1;
2590	cxt5066_olpc_select_mic(codec);
2591}
2592
2593static void cxt5066_olpc_capture_cleanup(struct hda_codec *codec)
2594{
2595	struct conexant_spec *spec = codec->spec;
2596	const struct hda_verb disable_mics[] = {
2597		/* disable external mic, port B */
2598		{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2599
2600		/* disble internal mic, port C */
2601		{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2602
2603		/* disable DC capture, port F */
2604		{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2605		{},
2606	};
2607
2608	snd_hda_sequence_write(codec, disable_mics);
2609	spec->recording = 0;
2610}
2611
2612static void conexant_check_dig_outs(struct hda_codec *codec,
2613				    const hda_nid_t *dig_pins,
2614				    int num_pins)
2615{
2616	struct conexant_spec *spec = codec->spec;
2617	hda_nid_t *nid_loc = &spec->multiout.dig_out_nid;
2618	int i;
2619
2620	for (i = 0; i < num_pins; i++, dig_pins++) {
2621		unsigned int cfg = snd_hda_codec_get_pincfg(codec, *dig_pins);
2622		if (get_defcfg_connect(cfg) == AC_JACK_PORT_NONE)
2623			continue;
2624		if (snd_hda_get_connections(codec, *dig_pins, nid_loc, 1) != 1)
2625			continue;
2626		if (spec->slave_dig_outs[0])
2627			nid_loc++;
2628		else
2629			nid_loc = spec->slave_dig_outs;
2630	}
2631}
2632
2633static const struct hda_input_mux cxt5066_capture_source = {
2634	.num_items = 4,
2635	.items = {
2636		{ "Mic B", 0 },
2637		{ "Mic C", 1 },
2638		{ "Mic E", 2 },
2639		{ "Mic F", 3 },
2640	},
2641};
2642
2643static const struct hda_bind_ctls cxt5066_bind_capture_vol_others = {
2644	.ops = &snd_hda_bind_vol,
2645	.values = {
2646		HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_INPUT),
2647		HDA_COMPOSE_AMP_VAL(0x14, 3, 2, HDA_INPUT),
2648		0
2649	},
2650};
2651
2652static const struct hda_bind_ctls cxt5066_bind_capture_sw_others = {
2653	.ops = &snd_hda_bind_sw,
2654	.values = {
2655		HDA_COMPOSE_AMP_VAL(0x14, 3, 0, HDA_INPUT),
2656		HDA_COMPOSE_AMP_VAL(0x14, 3, 2, HDA_INPUT),
2657		0
2658	},
2659};
2660
2661static const struct snd_kcontrol_new cxt5066_mixer_master[] = {
2662	HDA_CODEC_VOLUME("Master Playback Volume", 0x10, 0x00, HDA_OUTPUT),
2663	{}
2664};
2665
2666static const struct snd_kcontrol_new cxt5066_mixer_master_olpc[] = {
2667	{
2668		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2669		.name = "Master Playback Volume",
2670		.access = SNDRV_CTL_ELEM_ACCESS_READWRITE |
2671				  SNDRV_CTL_ELEM_ACCESS_TLV_READ |
2672				  SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK,
2673		.subdevice = HDA_SUBDEV_AMP_FLAG,
2674		.info = snd_hda_mixer_amp_volume_info,
2675		.get = snd_hda_mixer_amp_volume_get,
2676		.put = snd_hda_mixer_amp_volume_put,
2677		.tlv = { .c = snd_hda_mixer_amp_tlv },
2678		/* offset by 28 volume steps to limit minimum gain to -46dB */
2679		.private_value =
2680			HDA_COMPOSE_AMP_VAL_OFS(0x10, 3, 0, HDA_OUTPUT, 28),
2681	},
2682	{}
2683};
2684
2685static const struct snd_kcontrol_new cxt5066_mixer_olpc_dc[] = {
2686	{
2687		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2688		.name = "DC Mode Enable Switch",
2689		.info = snd_ctl_boolean_mono_info,
2690		.get = cxt5066_olpc_dc_get,
2691		.put = cxt5066_olpc_dc_put,
2692	},
2693	{
2694		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2695		.name = "DC Input Bias Enum",
2696		.info = cxt5066_olpc_dc_bias_enum_info,
2697		.get = cxt5066_olpc_dc_bias_enum_get,
2698		.put = cxt5066_olpc_dc_bias_enum_put,
2699	},
2700	{}
2701};
2702
2703static const struct snd_kcontrol_new cxt5066_mixers[] = {
2704	{
2705		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2706		.name = "Master Playback Switch",
2707		.info = cxt_eapd_info,
2708		.get = cxt_eapd_get,
2709		.put = cxt5066_hp_master_sw_put,
2710		.private_value = 0x1d,
2711	},
2712
2713	{
2714		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2715		.name = "Analog Mic Boost Capture Enum",
2716		.info = cxt5066_mic_boost_mux_enum_info,
2717		.get = cxt5066_mic_boost_mux_enum_get,
2718		.put = cxt5066_mic_boost_mux_enum_put,
2719	},
2720
2721	HDA_BIND_VOL("Capture Volume", &cxt5066_bind_capture_vol_others),
2722	HDA_BIND_SW("Capture Switch", &cxt5066_bind_capture_sw_others),
2723	{}
2724};
2725
2726static const struct snd_kcontrol_new cxt5066_vostro_mixers[] = {
2727	{
2728		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
2729		.name = "Internal Mic Boost Capture Enum",
2730		.info = cxt5066_mic_boost_mux_enum_info,
2731		.get = cxt5066_mic_boost_mux_enum_get,
2732		.put = cxt5066_mic_boost_mux_enum_put,
2733		.private_value = 0x23 | 0x100,
2734	},
2735	{}
2736};
2737
2738static const struct hda_verb cxt5066_init_verbs[] = {
2739	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80}, /* Port B */
2740	{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80}, /* Port C */
2741	{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Port F */
2742	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Port E */
2743
2744	/* Speakers  */
2745	{0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2746	{0x1f, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2747
2748	/* HP, Amp  */
2749	{0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2750	{0x19, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2751
2752	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2753	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2754
2755	/* DAC1 */
2756	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2757
2758	/* Node 14 connections: 0x17 0x18 0x23 0x24 0x27 */
2759	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x50},
2760	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2761	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2) | 0x50},
2762	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2763	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
2764
2765	/* no digital microphone support yet */
2766	{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2767
2768	/* Audio input selector */
2769	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE | 0x3},
2770
2771	/* SPDIF route: PCM */
2772	{0x20, AC_VERB_SET_CONNECT_SEL, 0x0},
2773	{0x22, AC_VERB_SET_CONNECT_SEL, 0x0},
2774
2775	{0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2776	{0x22, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2777
2778	/* EAPD */
2779	{0x1d, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
2780
2781	/* not handling these yet */
2782	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2783	{0x1a, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2784	{0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2785	{0x1c, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2786	{0x1d, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2787	{0x1e, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2788	{0x20, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2789	{0x22, AC_VERB_SET_UNSOLICITED_ENABLE, 0},
2790	{ } /* end */
2791};
2792
2793static const struct hda_verb cxt5066_init_verbs_olpc[] = {
2794	/* Port A: headphones */
2795	{0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2796	{0x19, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2797
2798	/* Port B: external microphone */
2799	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2800
2801	/* Port C: internal microphone */
2802	{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2803
2804	/* Port D: unused */
2805	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2806
2807	/* Port E: unused, but has primary EAPD */
2808	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2809	{0x1d, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
2810
2811	/* Port F: external DC input through microphone port */
2812	{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2813
2814	/* Port G: internal speakers */
2815	{0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2816	{0x1f, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2817
2818	/* DAC1 */
2819	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2820
2821	/* DAC2: unused */
2822	{0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2823
2824	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x50},
2825	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2826	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2827	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2828	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2829	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2830	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2831	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2832	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2833	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2834	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2835	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2836
2837	/* Disable digital microphone port */
2838	{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2839
2840	/* Audio input selectors */
2841	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE | 0x3},
2842	{0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
2843
2844	/* Disable SPDIF */
2845	{0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2846	{0x22, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2847
2848	/* enable unsolicited events for Port A and B */
2849	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
2850	{0x1a, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
2851	{ } /* end */
2852};
2853
2854static const struct hda_verb cxt5066_init_verbs_vostro[] = {
2855	/* Port A: headphones */
2856	{0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2857	{0x19, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2858
2859	/* Port B: external microphone */
2860	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2861
2862	/* Port C: unused */
2863	{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2864
2865	/* Port D: unused */
2866	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2867
2868	/* Port E: unused, but has primary EAPD */
2869	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2870	{0x1d, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
2871
2872	/* Port F: unused */
2873	{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2874
2875	/* Port G: internal speakers */
2876	{0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2877	{0x1f, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2878
2879	/* DAC1 */
2880	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2881
2882	/* DAC2: unused */
2883	{0x11, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},
2884
2885	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2886	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2887	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2888	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2889	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2890	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2891	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2892	{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2893	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)},
2894	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2895	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(2)},
2896	{0x16, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2897
2898	/* Digital microphone port */
2899	{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN},
2900
2901	/* Audio input selectors */
2902	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE | 0x3},
2903	{0x18, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE },
2904
2905	/* Disable SPDIF */
2906	{0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2907	{0x22, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2908
2909	/* enable unsolicited events for Port A and B */
2910	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
2911	{0x1a, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
2912	{ } /* end */
2913};
2914
2915static const struct hda_verb cxt5066_init_verbs_ideapad[] = {
2916	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80}, /* Port B */
2917	{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80}, /* Port C */
2918	{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Port F */
2919	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Port E */
2920
2921	/* Speakers  */
2922	{0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2923	{0x1f, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2924
2925	/* HP, Amp  */
2926	{0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2927	{0x19, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2928
2929	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP},
2930	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2931
2932	/* DAC1 */
2933	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2934
2935	/* Node 14 connections: 0x17 0x18 0x23 0x24 0x27 */
2936	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x50},
2937	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2938	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2) | 0x50},
2939	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2940	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
2941	{0x14, AC_VERB_SET_CONNECT_SEL, 2},	/* default to internal mic */
2942
2943	/* Audio input selector */
2944	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE | 0x2},
2945	{0x17, AC_VERB_SET_CONNECT_SEL, 1},	/* route ext mic */
2946
2947	/* SPDIF route: PCM */
2948	{0x20, AC_VERB_SET_CONNECT_SEL, 0x0},
2949	{0x22, AC_VERB_SET_CONNECT_SEL, 0x0},
2950
2951	{0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2952	{0x22, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2953
2954	/* internal microphone */
2955	{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* enable internal mic */
2956
2957	/* EAPD */
2958	{0x1d, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
2959
2960	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
2961	{0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
2962	{ } /* end */
2963};
2964
2965static const struct hda_verb cxt5066_init_verbs_thinkpad[] = {
2966	{0x1e, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Port F */
2967	{0x1d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* Port E */
2968
2969	/* Port G: internal speakers  */
2970	{0x1f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
2971	{0x1f, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2972
2973	/* Port A: HP, Amp  */
2974	{0x19, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2975	{0x19, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2976
2977	/* Port B: Mic Dock */
2978	{0x1a, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2979
2980	/* Port C: Mic */
2981	{0x1b, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2982
2983	/* Port D: HP Dock, Amp */
2984	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, 0},
2985	{0x1c, AC_VERB_SET_CONNECT_SEL, 0x00}, /* DAC1 */
2986
2987	/* DAC1 */
2988	{0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE},
2989
2990	/* Node 14 connections: 0x17 0x18 0x23 0x24 0x27 */
2991	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0) | 0x50},
2992	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(1)},
2993	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(2) | 0x50},
2994	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(3)},
2995	{0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(4)},
2996	{0x14, AC_VERB_SET_CONNECT_SEL, 2},	/* default to internal mic */
2997
2998	/* Audio input selector */
2999	{0x17, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE | 0x2},
3000	{0x17, AC_VERB_SET_CONNECT_SEL, 1},	/* route ext mic */
3001
3002	/* SPDIF route: PCM */
3003	{0x20, AC_VERB_SET_CONNECT_SEL, 0x0},
3004	{0x22, AC_VERB_SET_CONNECT_SEL, 0x0},
3005
3006	{0x20, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3007	{0x22, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3008
3009	/* internal microphone */
3010	{0x23, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN}, /* enable internal mic */
3011
3012	/* EAPD */
3013	{0x1d, AC_VERB_SET_EAPD_BTLENABLE, 0x2}, /* default on */
3014
3015	/* enable unsolicited events for Port A, B, C and D */
3016	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
3017	{0x1c, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
3018	{0x1a, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
3019	{0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
3020	{ } /* end */
3021};
3022
3023static const struct hda_verb cxt5066_init_verbs_portd_lo[] = {
3024	{0x1c, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT},
3025	{ } /* end */
3026};
3027
3028
3029static const struct hda_verb cxt5066_init_verbs_hp_laptop[] = {
3030	{0x14, AC_VERB_SET_CONNECT_SEL, 0x0},
3031	{0x19, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_HP_EVENT},
3032	{0x1b, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | CONEXANT_MIC_EVENT},
3033	{ } /* end */
3034};
3035
3036/* initialize jack-sensing, too */
3037static int cxt5066_init(struct hda_codec *codec)
3038{
3039	snd_printdd("CXT5066: init\n");
3040	conexant_init(codec);
3041	if (codec->patch_ops.unsol_event) {
3042		cxt5066_hp_automute(codec);
3043		cxt5066_automic(codec);
3044	}
3045	cxt5066_set_mic_boost(codec);
3046	return 0;
3047}
3048
3049static int cxt5066_olpc_init(struct hda_codec *codec)
3050{
3051	struct conexant_spec *spec = codec->spec;
3052	snd_printdd("CXT5066: init\n");
3053	conexant_init(codec);
3054	cxt5066_hp_automute(codec);
3055	if (!spec->dc_enable) {
3056		cxt5066_set_mic_boost(codec);
3057		cxt5066_olpc_automic(codec);
3058	} else {
3059		cxt5066_enable_dc(codec);
3060	}
3061	return 0;
3062}
3063
3064enum {
3065	CXT5066_LAPTOP,		/* Laptops w/ EAPD support */
3066	CXT5066_DELL_LAPTOP,	/* Dell Laptop */
3067	CXT5066_OLPC_XO_1_5,	/* OLPC XO 1.5 */
3068	CXT5066_DELL_VOSTRO,	/* Dell Vostro 1015i */
3069	CXT5066_IDEAPAD,	/* Lenovo IdeaPad U150 */
3070	CXT5066_THINKPAD,	/* Lenovo ThinkPad T410s, others? */
3071	CXT5066_ASUS,		/* Asus K52JU, Lenovo G560 - Int mic at 0x1a and Ext mic at 0x1b */
3072	CXT5066_HP_LAPTOP,      /* HP Laptop */
3073	CXT5066_AUTO,		/* BIOS auto-parser */
3074	CXT5066_MODELS
3075};
3076
3077static const char * const cxt5066_models[CXT5066_MODELS] = {
3078	[CXT5066_LAPTOP]	= "laptop",
3079	[CXT5066_DELL_LAPTOP]	= "dell-laptop",
3080	[CXT5066_OLPC_XO_1_5]	= "olpc-xo-1_5",
3081	[CXT5066_DELL_VOSTRO]	= "dell-vostro",
3082	[CXT5066_IDEAPAD]	= "ideapad",
3083	[CXT5066_THINKPAD]	= "thinkpad",
3084	[CXT5066_ASUS]		= "asus",
3085	[CXT5066_HP_LAPTOP]	= "hp-laptop",
3086	[CXT5066_AUTO]		= "auto",
3087};
3088
3089static const struct snd_pci_quirk cxt5066_cfg_tbl[] = {
3090	SND_PCI_QUIRK(0x1025, 0x054c, "Acer Aspire 3830TG", CXT5066_AUTO),
3091	SND_PCI_QUIRK_MASK(0x1025, 0xff00, 0x0400, "Acer", CXT5066_IDEAPAD),
3092	SND_PCI_QUIRK(0x1028, 0x02d8, "Dell Vostro", CXT5066_DELL_VOSTRO),
3093	SND_PCI_QUIRK(0x1028, 0x02f5, "Dell Vostro 320", CXT5066_IDEAPAD),
3094	SND_PCI_QUIRK(0x1028, 0x0401, "Dell Vostro 1014", CXT5066_DELL_VOSTRO),
3095	SND_PCI_QUIRK(0x1028, 0x0402, "Dell Vostro", CXT5066_DELL_VOSTRO),
3096	SND_PCI_QUIRK(0x1028, 0x0408, "Dell Inspiron One 19T", CXT5066_IDEAPAD),
3097	SND_PCI_QUIRK(0x1028, 0x050f, "Dell Inspiron", CXT5066_IDEAPAD),
3098	SND_PCI_QUIRK(0x1028, 0x0510, "Dell Vostro", CXT5066_IDEAPAD),
3099	SND_PCI_QUIRK(0x103c, 0x360b, "HP G60", CXT5066_HP_LAPTOP),
3100	SND_PCI_QUIRK(0x1043, 0x13f3, "Asus A52J", CXT5066_ASUS),
3101	SND_PCI_QUIRK(0x1043, 0x1643, "Asus K52JU", CXT5066_ASUS),
3102	SND_PCI_QUIRK(0x1043, 0x1993, "Asus U50F", CXT5066_ASUS),
3103	SND_PCI_QUIRK(0x1179, 0xff1e, "Toshiba Satellite C650D", CXT5066_IDEAPAD),
3104	SND_PCI_QUIRK(0x1179, 0xff50, "Toshiba Satellite P500-PSPGSC-01800T", CXT5066_OLPC_XO_1_5),
3105	SND_PCI_QUIRK(0x1179, 0xffe0, "Toshiba Satellite Pro T130-15F", CXT5066_OLPC_XO_1_5),
3106	SND_PCI_QUIRK(0x14f1, 0x0101, "Conexant Reference board",
3107		      CXT5066_LAPTOP),
3108	SND_PCI_QUIRK(0x152d, 0x0833, "OLPC XO-1.5", CXT5066_OLPC_XO_1_5),
3109	SND_PCI_QUIRK(0x17aa, 0x20f2, "Lenovo T400s", CXT5066_THINKPAD),
3110	SND_PCI_QUIRK(0x17aa, 0x21c5, "Thinkpad Edge 13", CXT5066_THINKPAD),
3111	SND_PCI_QUIRK(0x17aa, 0x21c6, "Thinkpad Edge 13", CXT5066_ASUS),
3112 	SND_PCI_QUIRK(0x17aa, 0x215e, "Lenovo Thinkpad", CXT5066_THINKPAD),
3113	SND_PCI_QUIRK(0x17aa, 0x21cf, "Lenovo T520 & W520", CXT5066_AUTO),
3114	SND_PCI_QUIRK(0x17aa, 0x21da, "Lenovo X220", CXT5066_THINKPAD),
3115	SND_PCI_QUIRK(0x17aa, 0x21db, "Lenovo X220-tablet", CXT5066_THINKPAD),
3116	SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo U350", CXT5066_ASUS),
3117	SND_PCI_QUIRK(0x17aa, 0x38af, "Lenovo G560", CXT5066_ASUS),
3118	SND_PCI_QUIRK(0x17aa, 0x3938, "Lenovo G565", CXT5066_AUTO),
3119	SND_PCI_QUIRK_VENDOR(0x17aa, "Lenovo", CXT5066_IDEAPAD), /* Fallback for Lenovos without dock mic */
3120	SND_PCI_QUIRK(0x1b0a, 0x2092, "CyberpowerPC Gamer Xplorer N57001", CXT5066_AUTO),
3121	{}
3122};
3123
3124static int patch_cxt5066(struct hda_codec *codec)
3125{
3126	struct conexant_spec *spec;
3127	int board_config;
3128
3129	board_config = snd_hda_check_board_config(codec, CXT5066_MODELS,
3130						  cxt5066_models, cxt5066_cfg_tbl);
3131	if (board_config < 0)
3132		board_config = CXT5066_AUTO; /* model=auto as default */
3133	if (board_config == CXT5066_AUTO)
3134		return patch_conexant_auto(codec);
3135
3136	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
3137	if (!spec)
3138		return -ENOMEM;
3139	codec->spec = spec;
3140
3141	codec->patch_ops = conexant_patch_ops;
3142	codec->patch_ops.init = conexant_init;
3143
3144	spec->dell_automute = 0;
3145	spec->multiout.max_channels = 2;
3146	spec->multiout.num_dacs = ARRAY_SIZE(cxt5066_dac_nids);
3147	spec->multiout.dac_nids = cxt5066_dac_nids;
3148	conexant_check_dig_outs(codec, cxt5066_digout_pin_nids,
3149	    ARRAY_SIZE(cxt5066_digout_pin_nids));
3150	spec->num_adc_nids = 1;
3151	spec->adc_nids = cxt5066_adc_nids;
3152	spec->capsrc_nids = cxt5066_capsrc_nids;
3153	spec->input_mux = &cxt5066_capture_source;
3154
3155	spec->port_d_mode = PIN_HP;
3156
3157	spec->num_init_verbs = 1;
3158	spec->init_verbs[0] = cxt5066_init_verbs;
3159	spec->num_channel_mode = ARRAY_SIZE(cxt5066_modes);
3160	spec->channel_mode = cxt5066_modes;
3161	spec->cur_adc = 0;
3162	spec->cur_adc_idx = 0;
3163
3164	set_beep_amp(spec, 0x13, 0, HDA_OUTPUT);
3165
3166	switch (board_config) {
3167	default:
3168	case CXT5066_LAPTOP:
3169		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master;
3170		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3171		break;
3172	case CXT5066_DELL_LAPTOP:
3173		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master;
3174		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3175
3176		spec->port_d_mode = PIN_OUT;
3177		spec->init_verbs[spec->num_init_verbs] = cxt5066_init_verbs_portd_lo;
3178		spec->num_init_verbs++;
3179		spec->dell_automute = 1;
3180		break;
3181	case CXT5066_ASUS:
3182	case CXT5066_HP_LAPTOP:
3183		codec->patch_ops.init = cxt5066_init;
3184		codec->patch_ops.unsol_event = cxt5066_unsol_event;
3185		spec->init_verbs[spec->num_init_verbs] =
3186			cxt5066_init_verbs_hp_laptop;
3187		spec->num_init_verbs++;
3188		spec->hp_laptop = board_config == CXT5066_HP_LAPTOP;
3189		spec->asus = board_config == CXT5066_ASUS;
3190		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master;
3191		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3192		/* no S/PDIF out */
3193		if (board_config == CXT5066_HP_LAPTOP)
3194			spec->multiout.dig_out_nid = 0;
3195		/* input source automatically selected */
3196		spec->input_mux = NULL;
3197		spec->port_d_mode = 0;
3198		spec->mic_boost = 3; /* default 30dB gain */
3199		break;
3200
3201	case CXT5066_OLPC_XO_1_5:
3202		codec->patch_ops.init = cxt5066_olpc_init;
3203		codec->patch_ops.unsol_event = cxt5066_olpc_unsol_event;
3204		spec->init_verbs[0] = cxt5066_init_verbs_olpc;
3205		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master_olpc;
3206		spec->mixers[spec->num_mixers++] = cxt5066_mixer_olpc_dc;
3207		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3208		spec->port_d_mode = 0;
3209		spec->mic_boost = 3; /* default 30dB gain */
3210
3211		/* no S/PDIF out */
3212		spec->multiout.dig_out_nid = 0;
3213
3214		/* input source automatically selected */
3215		spec->input_mux = NULL;
3216
3217		/* our capture hooks which allow us to turn on the microphone LED
3218		 * at the right time */
3219		spec->capture_prepare = cxt5066_olpc_capture_prepare;
3220		spec->capture_cleanup = cxt5066_olpc_capture_cleanup;
3221		break;
3222	case CXT5066_DELL_VOSTRO:
3223		codec->patch_ops.init = cxt5066_init;
3224		codec->patch_ops.unsol_event = cxt5066_unsol_event;
3225		spec->init_verbs[0] = cxt5066_init_verbs_vostro;
3226		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master_olpc;
3227		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3228		spec->mixers[spec->num_mixers++] = cxt5066_vostro_mixers;
3229		spec->port_d_mode = 0;
3230		spec->dell_vostro = 1;
3231		spec->mic_boost = 3; /* default 30dB gain */
3232
3233		/* no S/PDIF out */
3234		spec->multiout.dig_out_nid = 0;
3235
3236		/* input source automatically selected */
3237		spec->input_mux = NULL;
3238		break;
3239	case CXT5066_IDEAPAD:
3240		codec->patch_ops.init = cxt5066_init;
3241		codec->patch_ops.unsol_event = cxt5066_unsol_event;
3242		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master;
3243		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3244		spec->init_verbs[0] = cxt5066_init_verbs_ideapad;
3245		spec->port_d_mode = 0;
3246		spec->ideapad = 1;
3247		spec->mic_boost = 2;	/* default 20dB gain */
3248
3249		/* no S/PDIF out */
3250		spec->multiout.dig_out_nid = 0;
3251
3252		/* input source automatically selected */
3253		spec->input_mux = NULL;
3254		break;
3255	case CXT5066_THINKPAD:
3256		codec->patch_ops.init = cxt5066_init;
3257		codec->patch_ops.unsol_event = cxt5066_unsol_event;
3258		spec->mixers[spec->num_mixers++] = cxt5066_mixer_master;
3259		spec->mixers[spec->num_mixers++] = cxt5066_mixers;
3260		spec->init_verbs[0] = cxt5066_init_verbs_thinkpad;
3261		spec->thinkpad = 1;
3262		spec->port_d_mode = PIN_OUT;
3263		spec->mic_boost = 2;	/* default 20dB gain */
3264
3265		/* no S/PDIF out */
3266		spec->multiout.dig_out_nid = 0;
3267
3268		/* input source automatically selected */
3269		spec->input_mux = NULL;
3270		break;
3271	}
3272
3273	if (spec->beep_amp)
3274		snd_hda_attach_beep_device(codec, spec->beep_amp);
3275
3276	return 0;
3277}
3278
3279/*
3280 * Automatic parser for CX20641 & co
3281 */
3282
3283static int cx_auto_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
3284				       struct hda_codec *codec,
3285				       unsigned int stream_tag,
3286				       unsigned int format,
3287				       struct snd_pcm_substream *substream)
3288{
 
3289	struct conexant_spec *spec = codec->spec;
3290	hda_nid_t adc = spec->imux_info[spec->cur_mux[0]].adc;
3291	if (spec->adc_switching) {
3292		spec->cur_adc = adc;
3293		spec->cur_adc_stream_tag = stream_tag;
3294		spec->cur_adc_format = format;
3295	}
3296	snd_hda_codec_setup_stream(codec, adc, stream_tag, 0, format);
3297	return 0;
3298}
3299
3300static int cx_auto_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
3301				       struct hda_codec *codec,
3302				       struct snd_pcm_substream *substream)
3303{
3304	struct conexant_spec *spec = codec->spec;
3305	snd_hda_codec_cleanup_stream(codec, spec->cur_adc);
3306	spec->cur_adc = 0;
3307	return 0;
3308}
3309
3310static const struct hda_pcm_stream cx_auto_pcm_analog_capture = {
3311	.substreams = 1,
3312	.channels_min = 2,
3313	.channels_max = 2,
3314	.nid = 0, /* fill later */
3315	.ops = {
3316		.prepare = cx_auto_capture_pcm_prepare,
3317		.cleanup = cx_auto_capture_pcm_cleanup
3318	},
3319};
3320
3321static const hda_nid_t cx_auto_adc_nids[] = { 0x14 };
3322
3323#define get_connection_index(codec, mux, nid)\
3324	snd_hda_get_conn_index(codec, mux, nid, 0)
3325
3326/* get an unassigned DAC from the given list.
3327 * Return the nid if found and reduce the DAC list, or return zero if
3328 * not found
3329 */
3330static hda_nid_t get_unassigned_dac(struct hda_codec *codec, hda_nid_t pin,
3331				    hda_nid_t *dacs, int *num_dacs)
3332{
3333	int i, nums = *num_dacs;
3334	hda_nid_t ret = 0;
 
 
3335
3336	for (i = 0; i < nums; i++) {
3337		if (get_connection_index(codec, pin, dacs[i]) >= 0) {
3338			ret = dacs[i];
 
3339			break;
3340		}
3341	}
3342	if (!ret)
3343		return 0;
3344	if (--nums > 0)
3345		memmove(dacs, dacs + 1, nums * sizeof(hda_nid_t));
3346	*num_dacs = nums;
3347	return ret;
3348}
3349
3350#define MAX_AUTO_DACS	5
3351
3352#define DAC_SLAVE_FLAG	0x8000	/* filled dac is a slave */
3353
3354/* fill analog DAC list from the widget tree */
3355static int fill_cx_auto_dacs(struct hda_codec *codec, hda_nid_t *dacs)
3356{
3357	hda_nid_t nid, end_nid;
3358	int nums = 0;
3359
3360	end_nid = codec->start_nid + codec->num_nodes;
3361	for (nid = codec->start_nid; nid < end_nid; nid++) {
3362		unsigned int wcaps = get_wcaps(codec, nid);
3363		unsigned int type = get_wcaps_type(wcaps);
3364		if (type == AC_WID_AUD_OUT && !(wcaps & AC_WCAP_DIGITAL)) {
3365			dacs[nums++] = nid;
3366			if (nums >= MAX_AUTO_DACS)
3367				break;
3368		}
3369	}
3370	return nums;
3371}
3372
3373/* fill pin_dac_pair list from the pin and dac list */
3374static int fill_dacs_for_pins(struct hda_codec *codec, hda_nid_t *pins,
3375			      int num_pins, hda_nid_t *dacs, int *rest,
3376			      struct pin_dac_pair *filled, int nums, 
3377			      int type)
3378{
3379	int i, start = nums;
3380
3381	for (i = 0; i < num_pins; i++, nums++) {
3382		filled[nums].pin = pins[i];
3383		filled[nums].type = type;
3384		filled[nums].dac = get_unassigned_dac(codec, pins[i], dacs, rest);
3385		if (filled[nums].dac) 
3386			continue;
3387		if (filled[start].dac && get_connection_index(codec, pins[i], filled[start].dac) >= 0) {
3388			filled[nums].dac = filled[start].dac | DAC_SLAVE_FLAG;
3389			continue;
3390		}
3391		if (filled[0].dac && get_connection_index(codec, pins[i], filled[0].dac) >= 0) {
3392			filled[nums].dac = filled[0].dac | DAC_SLAVE_FLAG;
3393			continue;
3394		}
3395		snd_printdd("Failed to find a DAC for pin 0x%x", pins[i]);
3396	}
3397	return nums;
3398}
3399
3400/* parse analog output paths */
3401static void cx_auto_parse_output(struct hda_codec *codec)
3402{
3403	struct conexant_spec *spec = codec->spec;
3404	struct auto_pin_cfg *cfg = &spec->autocfg;
3405	hda_nid_t dacs[MAX_AUTO_DACS];
3406	int i, j, nums, rest;
3407
3408	rest = fill_cx_auto_dacs(codec, dacs);
3409	/* parse all analog output pins */
3410	nums = fill_dacs_for_pins(codec, cfg->line_out_pins, cfg->line_outs,
3411			  dacs, &rest, spec->dac_info, 0,
3412			  AUTO_PIN_LINE_OUT);
3413	nums = fill_dacs_for_pins(codec, cfg->hp_pins, cfg->hp_outs,
3414			  dacs, &rest, spec->dac_info, nums,
3415			  AUTO_PIN_HP_OUT);
3416	nums = fill_dacs_for_pins(codec, cfg->speaker_pins, cfg->speaker_outs,
3417			  dacs, &rest, spec->dac_info, nums,
3418			  AUTO_PIN_SPEAKER_OUT);
3419	spec->dac_info_filled = nums;
3420	/* fill multiout struct */
3421	for (i = 0; i < nums; i++) {
3422		hda_nid_t dac = spec->dac_info[i].dac;
3423		if (!dac || (dac & DAC_SLAVE_FLAG))
3424			continue;
3425		switch (spec->dac_info[i].type) {
3426		case AUTO_PIN_LINE_OUT:
3427			spec->private_dac_nids[spec->multiout.num_dacs] = dac;
3428			spec->multiout.num_dacs++;
3429			break;
3430		case AUTO_PIN_HP_OUT:
3431		case AUTO_PIN_SPEAKER_OUT:
3432			if (!spec->multiout.hp_nid) {
3433				spec->multiout.hp_nid = dac;
3434				break;
3435			}
3436			for (j = 0; j < ARRAY_SIZE(spec->multiout.extra_out_nid); j++)
3437				if (!spec->multiout.extra_out_nid[j]) {
3438					spec->multiout.extra_out_nid[j] = dac;
3439					break;
3440				}
3441			break;
3442		}
3443	}
3444	spec->multiout.dac_nids = spec->private_dac_nids;
3445	spec->multiout.max_channels = spec->multiout.num_dacs * 2;
3446
3447	for (i = 0; i < cfg->hp_outs; i++) {
3448		if (is_jack_detectable(codec, cfg->hp_pins[i])) {
3449			spec->auto_mute = 1;
3450			break;
3451		}
3452	}
3453	if (spec->auto_mute &&
3454	    cfg->line_out_pins[0] &&
3455	    cfg->line_out_type != AUTO_PIN_SPEAKER_OUT &&
3456	    cfg->line_out_pins[0] != cfg->hp_pins[0] &&
3457	    cfg->line_out_pins[0] != cfg->speaker_pins[0]) {
3458		for (i = 0; i < cfg->line_outs; i++) {
3459			if (is_jack_detectable(codec, cfg->line_out_pins[i])) {
3460				spec->detect_line = 1;
3461				break;
3462			}
3463		}
3464		spec->automute_lines = spec->detect_line;
3465	}
3466
3467	spec->vmaster_nid = spec->private_dac_nids[0];
3468}
3469
3470static void cx_auto_turn_eapd(struct hda_codec *codec, int num_pins,
3471			      hda_nid_t *pins, bool on);
3472
3473static void do_automute(struct hda_codec *codec, int num_pins,
3474			hda_nid_t *pins, bool on)
3475{
3476	int i;
3477	for (i = 0; i < num_pins; i++)
3478		snd_hda_codec_write(codec, pins[i], 0,
3479				    AC_VERB_SET_PIN_WIDGET_CONTROL,
3480				    on ? PIN_OUT : 0);
3481	cx_auto_turn_eapd(codec, num_pins, pins, on);
3482}
3483
3484static int detect_jacks(struct hda_codec *codec, int num_pins, hda_nid_t *pins)
3485{
3486	int i, present = 0;
3487
3488	for (i = 0; i < num_pins; i++) {
3489		hda_nid_t nid = pins[i];
3490		if (!nid || !is_jack_detectable(codec, nid))
3491			break;
3492		snd_hda_input_jack_report(codec, nid);
3493		present |= snd_hda_jack_detect(codec, nid);
3494	}
3495	return present;
3496}
3497
3498/* auto-mute/unmute speaker and line outs according to headphone jack */
3499static void cx_auto_update_speakers(struct hda_codec *codec)
 
 
 
 
3500{
3501	struct conexant_spec *spec = codec->spec;
3502	struct auto_pin_cfg *cfg = &spec->autocfg;
3503	int on = 1;
3504
3505	/* turn on HP EAPD when HP jacks are present */
3506	if (spec->auto_mute)
3507		on = spec->hp_present;
3508	cx_auto_turn_eapd(codec, cfg->hp_outs, cfg->hp_pins, on);
3509	/* mute speakers in auto-mode if HP or LO jacks are plugged */
3510	if (spec->auto_mute)
3511		on = !(spec->hp_present ||
3512		       (spec->detect_line && spec->line_present));
3513	do_automute(codec, cfg->speaker_outs, cfg->speaker_pins, on);
3514
3515	/* toggle line-out mutes if needed, too */
3516	/* if LO is a copy of either HP or Speaker, don't need to handle it */
3517	if (cfg->line_out_pins[0] == cfg->hp_pins[0] ||
3518	    cfg->line_out_pins[0] == cfg->speaker_pins[0])
3519		return;
3520	if (spec->auto_mute) {
3521		/* mute LO in auto-mode when HP jack is present */
3522		if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT ||
3523		    spec->automute_lines)
3524			on = !spec->hp_present;
3525		else
3526			on = 1;
3527	}
3528	do_automute(codec, cfg->line_outs, cfg->line_out_pins, on);
3529}
3530
3531static void cx_auto_hp_automute(struct hda_codec *codec)
 
 
 
 
 
3532{
3533	struct conexant_spec *spec = codec->spec;
3534	struct auto_pin_cfg *cfg = &spec->autocfg;
3535
3536	if (!spec->auto_mute)
3537		return;
3538	spec->hp_present = detect_jacks(codec, cfg->hp_outs, cfg->hp_pins);
3539	cx_auto_update_speakers(codec);
3540}
3541
3542static void cx_auto_line_automute(struct hda_codec *codec)
3543{
3544	struct conexant_spec *spec = codec->spec;
3545	struct auto_pin_cfg *cfg = &spec->autocfg;
3546
3547	if (!spec->auto_mute || !spec->detect_line)
3548		return;
3549	spec->line_present = detect_jacks(codec, cfg->line_outs,
3550					  cfg->line_out_pins);
3551	cx_auto_update_speakers(codec);
3552}
3553
3554static int cx_automute_mode_info(struct snd_kcontrol *kcontrol,
3555				 struct snd_ctl_elem_info *uinfo)
 
3556{
3557	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3558	struct conexant_spec *spec = codec->spec;
3559	static const char * const texts2[] = {
3560		"Disabled", "Enabled"
3561	};
3562	static const char * const texts3[] = {
3563		"Disabled", "Speaker Only", "Line-Out+Speaker"
3564	};
3565	const char * const *texts;
3566
3567	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
3568	uinfo->count = 1;
3569	if (spec->automute_hp_lo) {
3570		uinfo->value.enumerated.items = 3;
3571		texts = texts3;
3572	} else {
3573		uinfo->value.enumerated.items = 2;
3574		texts = texts2;
3575	}
3576	if (uinfo->value.enumerated.item >= uinfo->value.enumerated.items)
3577		uinfo->value.enumerated.item = uinfo->value.enumerated.items - 1;
3578	strcpy(uinfo->value.enumerated.name,
3579	       texts[uinfo->value.enumerated.item]);
3580	return 0;
3581}
3582
3583static int cx_automute_mode_get(struct snd_kcontrol *kcontrol,
3584				struct snd_ctl_elem_value *ucontrol)
3585{
3586	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3587	struct conexant_spec *spec = codec->spec;
3588	unsigned int val;
3589	if (!spec->auto_mute)
3590		val = 0;
3591	else if (!spec->automute_lines)
3592		val = 1;
3593	else
3594		val = 2;
3595	ucontrol->value.enumerated.item[0] = val;
3596	return 0;
 
 
 
3597}
3598
3599static int cx_automute_mode_put(struct snd_kcontrol *kcontrol,
3600				struct snd_ctl_elem_value *ucontrol)
 
3601{
3602	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3603	struct conexant_spec *spec = codec->spec;
3604
3605	switch (ucontrol->value.enumerated.item[0]) {
3606	case 0:
3607		if (!spec->auto_mute)
3608			return 0;
3609		spec->auto_mute = 0;
3610		break;
3611	case 1:
3612		if (spec->auto_mute && !spec->automute_lines)
3613			return 0;
3614		spec->auto_mute = 1;
3615		spec->automute_lines = 0;
3616		break;
3617	case 2:
3618		if (!spec->automute_hp_lo)
3619			return -EINVAL;
3620		if (spec->auto_mute && spec->automute_lines)
3621			return 0;
3622		spec->auto_mute = 1;
3623		spec->automute_lines = 1;
3624		break;
3625	default:
3626		return -EINVAL;
3627	}
3628	cx_auto_update_speakers(codec);
3629	return 1;
3630}
3631
3632static const struct snd_kcontrol_new cx_automute_mode_enum[] = {
3633	{
3634		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3635		.name = "Auto-Mute Mode",
3636		.info = cx_automute_mode_info,
3637		.get = cx_automute_mode_get,
3638		.put = cx_automute_mode_put,
3639	},
3640	{ }
3641};
3642
3643static int cx_auto_mux_enum_info(struct snd_kcontrol *kcontrol,
3644				 struct snd_ctl_elem_info *uinfo)
3645{
3646	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3647	struct conexant_spec *spec = codec->spec;
3648
3649	return snd_hda_input_mux_info(&spec->private_imux, uinfo);
 
3650}
3651
3652static int cx_auto_mux_enum_get(struct snd_kcontrol *kcontrol,
3653				struct snd_ctl_elem_value *ucontrol)
3654{
3655	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3656	struct conexant_spec *spec = codec->spec;
3657
3658	ucontrol->value.enumerated.item[0] = spec->cur_mux[0];
3659	return 0;
3660}
3661
3662/* look for the route the given pin from mux and return the index;
3663 * if do_select is set, actually select the route.
3664 */
3665static int __select_input_connection(struct hda_codec *codec, hda_nid_t mux,
3666				     hda_nid_t pin, hda_nid_t *srcp,
3667				     bool do_select, int depth)
3668{
3669	hda_nid_t conn[HDA_MAX_NUM_INPUTS];
3670	int i, nums;
3671
3672	switch (get_wcaps_type(get_wcaps(codec, mux))) {
3673	case AC_WID_AUD_IN:
3674	case AC_WID_AUD_SEL:
3675	case AC_WID_AUD_MIX:
3676		break;
3677	default:
3678		return -1;
3679	}
3680
3681	nums = snd_hda_get_connections(codec, mux, conn, ARRAY_SIZE(conn));
3682	for (i = 0; i < nums; i++)
3683		if (conn[i] == pin) {
3684			if (do_select)
3685				snd_hda_codec_write(codec, mux, 0,
3686						    AC_VERB_SET_CONNECT_SEL, i);
3687			if (srcp)
3688				*srcp = mux;
3689			return i;
3690		}
3691	depth++;
3692	if (depth == 2)
3693		return -1;
3694	for (i = 0; i < nums; i++) {
3695		int ret  = __select_input_connection(codec, conn[i], pin, srcp,
3696						     do_select, depth);
3697		if (ret >= 0) {
3698			if (do_select)
3699				snd_hda_codec_write(codec, mux, 0,
3700						    AC_VERB_SET_CONNECT_SEL, i);
3701			return i;
3702		}
3703	}
3704	return -1;
3705}
3706
3707static void select_input_connection(struct hda_codec *codec, hda_nid_t mux,
3708				   hda_nid_t pin)
3709{
3710	__select_input_connection(codec, mux, pin, NULL, true, 0);
 
3711}
3712
3713static int get_input_connection(struct hda_codec *codec, hda_nid_t mux,
3714				hda_nid_t pin)
3715{
3716	return __select_input_connection(codec, mux, pin, NULL, false, 0);
3717}
3718
3719static int cx_auto_mux_enum_update(struct hda_codec *codec,
3720				   const struct hda_input_mux *imux,
3721				   unsigned int idx)
3722{
3723	struct conexant_spec *spec = codec->spec;
3724	hda_nid_t adc;
3725	int changed = 1;
3726
3727	if (!imux->num_items)
3728		return 0;
3729	if (idx >= imux->num_items)
3730		idx = imux->num_items - 1;
3731	if (spec->cur_mux[0] == idx)
3732		changed = 0;
3733	adc = spec->imux_info[idx].adc;
3734	select_input_connection(codec, spec->imux_info[idx].adc,
3735				spec->imux_info[idx].pin);
3736	if (spec->cur_adc && spec->cur_adc != adc) {
3737		/* stream is running, let's swap the current ADC */
3738		__snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
3739		spec->cur_adc = adc;
3740		snd_hda_codec_setup_stream(codec, adc,
3741					   spec->cur_adc_stream_tag, 0,
3742					   spec->cur_adc_format);
3743	}
3744	spec->cur_mux[0] = idx;
3745	return changed;
3746}
3747
3748static int cx_auto_mux_enum_put(struct snd_kcontrol *kcontrol,
3749				struct snd_ctl_elem_value *ucontrol)
3750{
3751	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
3752	struct conexant_spec *spec = codec->spec;
3753
3754	return cx_auto_mux_enum_update(codec, &spec->private_imux,
3755				       ucontrol->value.enumerated.item[0]);
3756}
3757
3758static const struct snd_kcontrol_new cx_auto_capture_mixers[] = {
3759	{
3760		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
3761		.name = "Capture Source",
3762		.info = cx_auto_mux_enum_info,
3763		.get = cx_auto_mux_enum_get,
3764		.put = cx_auto_mux_enum_put
3765	},
3766	{}
3767};
3768
3769static bool select_automic(struct hda_codec *codec, int idx, bool detect)
3770{
3771	struct conexant_spec *spec = codec->spec;
3772	if (idx < 0)
3773		return false;
3774	if (detect && !snd_hda_jack_detect(codec, spec->imux_info[idx].pin))
3775		return false;
3776	cx_auto_mux_enum_update(codec, &spec->private_imux, idx);
3777	return true;
3778}
3779
3780/* automatic switch internal and external mic */
3781static void cx_auto_automic(struct hda_codec *codec)
3782{
3783	struct conexant_spec *spec = codec->spec;
3784
3785	if (!spec->auto_mic)
3786		return;
3787	if (!select_automic(codec, spec->auto_mic_ext, true))
3788		if (!select_automic(codec, spec->auto_mic_dock, true))
3789			select_automic(codec, spec->auto_mic_int, false);
3790}
3791
3792static void cx_auto_unsol_event(struct hda_codec *codec, unsigned int res)
3793{
3794	int nid = (res & AC_UNSOL_RES_SUBTAG) >> 20;
3795	switch (res >> 26) {
3796	case CONEXANT_HP_EVENT:
3797		cx_auto_hp_automute(codec);
3798		break;
3799	case CONEXANT_LINE_EVENT:
3800		cx_auto_line_automute(codec);
3801		break;
3802	case CONEXANT_MIC_EVENT:
3803		cx_auto_automic(codec);
3804		snd_hda_input_jack_report(codec, nid);
3805		break;
3806	}
3807}
3808
3809/* check whether the pin config is suitable for auto-mic switching;
3810 * auto-mic is enabled only when one int-mic and one ext- and/or
3811 * one dock-mic exist
3812 */
3813static void cx_auto_check_auto_mic(struct hda_codec *codec)
3814{
3815	struct conexant_spec *spec = codec->spec;
3816	int pset[INPUT_PIN_ATTR_NORMAL + 1];
3817	int i;
 
 
3818
3819	for (i = 0; i < ARRAY_SIZE(pset); i++)
3820		pset[i] = -1;
3821	for (i = 0; i < spec->private_imux.num_items; i++) {
3822		hda_nid_t pin = spec->imux_info[i].pin;
3823		unsigned int def_conf = snd_hda_codec_get_pincfg(codec, pin);
3824		int type, attr;
3825		attr = snd_hda_get_input_pin_attr(def_conf);
3826		if (attr == INPUT_PIN_ATTR_UNUSED)
3827			return; /* invalid entry */
3828		if (attr > INPUT_PIN_ATTR_NORMAL)
3829			attr = INPUT_PIN_ATTR_NORMAL;
3830		if (attr != INPUT_PIN_ATTR_INT &&
3831		    !is_jack_detectable(codec, pin))
3832			return; /* non-detectable pin */
3833		type = get_defcfg_device(def_conf);
3834		if (type != AC_JACK_MIC_IN &&
3835		    (attr != INPUT_PIN_ATTR_DOCK || type != AC_JACK_LINE_IN))
3836			return; /* no valid input type */
3837		if (pset[attr] >= 0)
3838			return; /* already occupied */
3839		pset[attr] = i;
3840	}
3841	if (pset[INPUT_PIN_ATTR_INT] < 0 ||
3842	    (pset[INPUT_PIN_ATTR_NORMAL] < 0 && pset[INPUT_PIN_ATTR_DOCK]))
3843		return; /* no input to switch*/
3844	spec->auto_mic = 1;
3845	spec->auto_mic_ext = pset[INPUT_PIN_ATTR_NORMAL];
3846	spec->auto_mic_dock = pset[INPUT_PIN_ATTR_DOCK];
3847	spec->auto_mic_int = pset[INPUT_PIN_ATTR_INT];
3848}
3849
3850static void cx_auto_parse_input(struct hda_codec *codec)
3851{
3852	struct conexant_spec *spec = codec->spec;
3853	struct auto_pin_cfg *cfg = &spec->autocfg;
3854	struct hda_input_mux *imux;
3855	int i, j;
3856
3857	imux = &spec->private_imux;
3858	for (i = 0; i < cfg->num_inputs; i++) {
3859		for (j = 0; j < spec->num_adc_nids; j++) {
3860			hda_nid_t adc = spec->adc_nids[j];
3861			int idx = get_input_connection(codec, adc,
3862						       cfg->inputs[i].pin);
3863			if (idx >= 0) {
3864				const char *label;
3865				label = hda_get_autocfg_input_label(codec, cfg, i);
3866				spec->imux_info[imux->num_items].index = i;
3867				spec->imux_info[imux->num_items].boost = 0;
3868				spec->imux_info[imux->num_items].adc = adc;
3869				spec->imux_info[imux->num_items].pin =
3870					cfg->inputs[i].pin;
3871				snd_hda_add_imux_item(imux, label, idx, NULL);
3872				break;
3873			}
3874		}
3875	}
3876	if (imux->num_items >= 2 && cfg->num_inputs == imux->num_items)
3877		cx_auto_check_auto_mic(codec);
3878	if (imux->num_items > 1) {
3879		for (i = 1; i < imux->num_items; i++) {
3880			if (spec->imux_info[i].adc != spec->imux_info[0].adc) {
3881				spec->adc_switching = 1;
3882				break;
3883			}
3884		}
3885	}
3886}
3887
3888/* get digital-input audio widget corresponding to the given pin */
3889static hda_nid_t cx_auto_get_dig_in(struct hda_codec *codec, hda_nid_t pin)
3890{
3891	hda_nid_t nid, end_nid;
3892
3893	end_nid = codec->start_nid + codec->num_nodes;
3894	for (nid = codec->start_nid; nid < end_nid; nid++) {
3895		unsigned int wcaps = get_wcaps(codec, nid);
3896		unsigned int type = get_wcaps_type(wcaps);
3897		if (type == AC_WID_AUD_IN && (wcaps & AC_WCAP_DIGITAL)) {
3898			if (get_connection_index(codec, nid, pin) >= 0)
3899				return nid;
 
 
 
 
 
 
 
 
 
3900		}
3901	}
3902	return 0;
3903}
3904
3905static void cx_auto_parse_digital(struct hda_codec *codec)
3906{
3907	struct conexant_spec *spec = codec->spec;
3908	struct auto_pin_cfg *cfg = &spec->autocfg;
3909	hda_nid_t nid;
3910
3911	if (cfg->dig_outs &&
3912	    snd_hda_get_connections(codec, cfg->dig_out_pins[0], &nid, 1) == 1)
3913		spec->multiout.dig_out_nid = nid;
3914	if (cfg->dig_in_pin)
3915		spec->dig_in_nid = cx_auto_get_dig_in(codec, cfg->dig_in_pin);
3916}
3917
3918#ifdef CONFIG_SND_HDA_INPUT_BEEP
3919static void cx_auto_parse_beep(struct hda_codec *codec)
3920{
3921	struct conexant_spec *spec = codec->spec;
3922	hda_nid_t nid, end_nid;
3923
3924	end_nid = codec->start_nid + codec->num_nodes;
3925	for (nid = codec->start_nid; nid < end_nid; nid++)
3926		if (get_wcaps_type(get_wcaps(codec, nid)) == AC_WID_BEEP) {
3927			set_beep_amp(spec, nid, 0, HDA_OUTPUT);
3928			break;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3929		}
3930}
3931#else
3932#define cx_auto_parse_beep(codec)
3933#endif
3934
3935static bool found_in_nid_list(hda_nid_t nid, const hda_nid_t *list, int nums)
3936{
3937	int i;
3938	for (i = 0; i < nums; i++)
3939		if (list[i] == nid)
3940			return true;
3941	return false;
3942}
3943
3944/* parse extra-EAPD that aren't assigned to any pins */
3945static void cx_auto_parse_eapd(struct hda_codec *codec)
3946{
3947	struct conexant_spec *spec = codec->spec;
3948	struct auto_pin_cfg *cfg = &spec->autocfg;
3949	hda_nid_t nid, end_nid;
3950
3951	end_nid = codec->start_nid + codec->num_nodes;
3952	for (nid = codec->start_nid; nid < end_nid; nid++) {
3953		if (get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_PIN)
3954			continue;
3955		if (!(snd_hda_query_pin_caps(codec, nid) & AC_PINCAP_EAPD))
3956			continue;
3957		if (found_in_nid_list(nid, cfg->line_out_pins, cfg->line_outs) ||
3958		    found_in_nid_list(nid, cfg->hp_pins, cfg->hp_outs) ||
3959		    found_in_nid_list(nid, cfg->speaker_pins, cfg->speaker_outs))
3960			continue;
3961		spec->eapds[spec->num_eapds++] = nid;
3962		if (spec->num_eapds >= ARRAY_SIZE(spec->eapds))
3963			break;
3964	}
3965}
3966
3967static int cx_auto_parse_auto_config(struct hda_codec *codec)
3968{
3969	struct conexant_spec *spec = codec->spec;
3970	int err;
3971
3972	err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL);
3973	if (err < 0)
3974		return err;
3975
3976	cx_auto_parse_output(codec);
3977	cx_auto_parse_input(codec);
3978	cx_auto_parse_digital(codec);
3979	cx_auto_parse_beep(codec);
3980	cx_auto_parse_eapd(codec);
3981	return 0;
3982}
3983
3984static void cx_auto_turn_eapd(struct hda_codec *codec, int num_pins,
3985			      hda_nid_t *pins, bool on)
3986{
3987	int i;
3988	for (i = 0; i < num_pins; i++) {
3989		if (snd_hda_query_pin_caps(codec, pins[i]) & AC_PINCAP_EAPD)
3990			snd_hda_codec_write(codec, pins[i], 0,
3991					    AC_VERB_SET_EAPD_BTLENABLE,
3992					    on ? 0x02 : 0);
3993	}
3994}
3995
3996static void select_connection(struct hda_codec *codec, hda_nid_t pin,
3997			      hda_nid_t src)
3998{
3999	int idx = get_connection_index(codec, pin, src);
4000	if (idx >= 0)
4001		snd_hda_codec_write(codec, pin, 0,
4002				    AC_VERB_SET_CONNECT_SEL, idx);
4003}
4004
4005static void mute_outputs(struct hda_codec *codec, int num_nids,
4006			 const hda_nid_t *nids)
4007{
4008	int i, val;
4009
4010	for (i = 0; i < num_nids; i++) {
4011		hda_nid_t nid = nids[i];
4012		if (!(get_wcaps(codec, nid) & AC_WCAP_OUT_AMP))
4013			continue;
4014		if (query_amp_caps(codec, nid, HDA_OUTPUT) & AC_AMPCAP_MUTE)
4015			val = AMP_OUT_MUTE;
4016		else
4017			val = AMP_OUT_ZERO;
4018		snd_hda_codec_write(codec, nid, 0,
4019				    AC_VERB_SET_AMP_GAIN_MUTE, val);
4020	}
4021}
4022
4023static void enable_unsol_pins(struct hda_codec *codec, int num_pins,
4024			      hda_nid_t *pins, unsigned int tag)
4025{
4026	int i;
4027	for (i = 0; i < num_pins; i++)
4028		snd_hda_codec_write(codec, pins[i], 0,
4029				    AC_VERB_SET_UNSOLICITED_ENABLE,
4030				    AC_USRSP_EN | tag);
4031}
4032
4033static void cx_auto_init_output(struct hda_codec *codec)
4034{
4035	struct conexant_spec *spec = codec->spec;
4036	struct auto_pin_cfg *cfg = &spec->autocfg;
4037	hda_nid_t nid;
4038	int i;
 
 
 
 
 
 
4039
4040	mute_outputs(codec, spec->multiout.num_dacs, spec->multiout.dac_nids);
4041	for (i = 0; i < cfg->hp_outs; i++)
4042		snd_hda_codec_write(codec, cfg->hp_pins[i], 0,
4043				    AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP);
4044	mute_outputs(codec, cfg->hp_outs, cfg->hp_pins);
4045	mute_outputs(codec, cfg->line_outs, cfg->line_out_pins);
4046	mute_outputs(codec, cfg->speaker_outs, cfg->speaker_pins);
4047	for (i = 0; i < spec->dac_info_filled; i++) {
4048		nid = spec->dac_info[i].dac;
4049		if (!nid)
4050			nid = spec->multiout.dac_nids[0];
4051		else if (nid & DAC_SLAVE_FLAG)
4052			nid &= ~DAC_SLAVE_FLAG;
4053		select_connection(codec, spec->dac_info[i].pin, nid);
4054	}
4055	if (spec->auto_mute) {
4056		enable_unsol_pins(codec, cfg->hp_outs, cfg->hp_pins,
4057				  CONEXANT_HP_EVENT);
4058		spec->hp_present = detect_jacks(codec, cfg->hp_outs,
4059						cfg->hp_pins);
4060		if (spec->detect_line) {
4061			enable_unsol_pins(codec, cfg->line_outs,
4062					  cfg->line_out_pins,
4063					  CONEXANT_LINE_EVENT);
4064			spec->line_present =
4065				detect_jacks(codec, cfg->line_outs,
4066					     cfg->line_out_pins);
4067		}
4068	}
4069	cx_auto_update_speakers(codec);
4070	/* turn on/off extra EAPDs, too */
4071	cx_auto_turn_eapd(codec, spec->num_eapds, spec->eapds, true);
4072}
4073
4074static void cx_auto_init_input(struct hda_codec *codec)
4075{
4076	struct conexant_spec *spec = codec->spec;
4077	struct auto_pin_cfg *cfg = &spec->autocfg;
4078	int i, val;
 
4079
4080	for (i = 0; i < spec->num_adc_nids; i++) {
4081		hda_nid_t nid = spec->adc_nids[i];
4082		if (!(get_wcaps(codec, nid) & AC_WCAP_IN_AMP))
4083			continue;
4084		if (query_amp_caps(codec, nid, HDA_INPUT) & AC_AMPCAP_MUTE)
4085			val = AMP_IN_MUTE(0);
4086		else
4087			val = AMP_IN_UNMUTE(0);
4088		snd_hda_codec_write(codec, nid, 0, AC_VERB_SET_AMP_GAIN_MUTE,
4089				    val);
4090	}
4091
4092	for (i = 0; i < cfg->num_inputs; i++) {
4093		unsigned int type;
4094		if (cfg->inputs[i].type == AUTO_PIN_MIC)
4095			type = PIN_VREF80;
4096		else
4097			type = PIN_IN;
4098		snd_hda_codec_write(codec, cfg->inputs[i].pin, 0,
4099				    AC_VERB_SET_PIN_WIDGET_CONTROL, type);
4100	}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
4101
4102	if (spec->auto_mic) {
4103		if (spec->auto_mic_ext >= 0) {
4104			snd_hda_codec_write(codec,
4105				cfg->inputs[spec->auto_mic_ext].pin, 0,
4106				AC_VERB_SET_UNSOLICITED_ENABLE,
4107				AC_USRSP_EN | CONEXANT_MIC_EVENT);
4108		}
4109		if (spec->auto_mic_dock >= 0) {
4110			snd_hda_codec_write(codec,
4111				cfg->inputs[spec->auto_mic_dock].pin, 0,
4112				AC_VERB_SET_UNSOLICITED_ENABLE,
4113				AC_USRSP_EN | CONEXANT_MIC_EVENT);
4114		}
4115		cx_auto_automic(codec);
4116	} else {
4117		select_input_connection(codec, spec->imux_info[0].adc,
4118					spec->imux_info[0].pin);
4119	}
4120}
4121
4122static void cx_auto_init_digital(struct hda_codec *codec)
 
 
 
4123{
4124	struct conexant_spec *spec = codec->spec;
4125	struct auto_pin_cfg *cfg = &spec->autocfg;
4126
4127	if (spec->multiout.dig_out_nid)
4128		snd_hda_codec_write(codec, cfg->dig_out_pins[0], 0,
4129				    AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
4130	if (spec->dig_in_nid)
4131		snd_hda_codec_write(codec, cfg->dig_in_pin, 0,
4132				    AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN);
4133}
4134
4135static int cx_auto_init(struct hda_codec *codec)
4136{
4137	/*snd_hda_sequence_write(codec, cx_auto_init_verbs);*/
4138	cx_auto_init_output(codec);
4139	cx_auto_init_input(codec);
4140	cx_auto_init_digital(codec);
4141	return 0;
4142}
4143
4144static int cx_auto_add_volume_idx(struct hda_codec *codec, const char *basename,
4145			      const char *dir, int cidx,
4146			      hda_nid_t nid, int hda_dir, int amp_idx)
4147{
4148	static char name[32];
4149	static struct snd_kcontrol_new knew[] = {
4150		HDA_CODEC_VOLUME(name, 0, 0, 0),
4151		HDA_CODEC_MUTE(name, 0, 0, 0),
4152	};
4153	static const char * const sfx[2] = { "Volume", "Switch" };
4154	int i, err;
4155
4156	for (i = 0; i < 2; i++) {
4157		struct snd_kcontrol *kctl;
4158		knew[i].private_value = HDA_COMPOSE_AMP_VAL(nid, 3, amp_idx,
4159							    hda_dir);
4160		knew[i].subdevice = HDA_SUBDEV_AMP_FLAG;
4161		knew[i].index = cidx;
4162		snprintf(name, sizeof(name), "%s%s %s", basename, dir, sfx[i]);
4163		kctl = snd_ctl_new1(&knew[i], codec);
4164		if (!kctl)
4165			return -ENOMEM;
4166		err = snd_hda_ctl_add(codec, nid, kctl);
4167		if (err < 0)
4168			return err;
4169		if (!(query_amp_caps(codec, nid, hda_dir) & AC_AMPCAP_MUTE))
4170			break;
4171	}
4172	return 0;
4173}
4174
4175#define cx_auto_add_volume(codec, str, dir, cidx, nid, hda_dir)		\
4176	cx_auto_add_volume_idx(codec, str, dir, cidx, nid, hda_dir, 0)
4177
4178#define cx_auto_add_pb_volume(codec, nid, str, idx)			\
4179	cx_auto_add_volume(codec, str, " Playback", idx, nid, HDA_OUTPUT)
4180
4181static int try_add_pb_volume(struct hda_codec *codec, hda_nid_t dac,
4182			     hda_nid_t pin, const char *name, int idx)
4183{
4184	unsigned int caps;
4185	if (dac && !(dac & DAC_SLAVE_FLAG)) {
4186		caps = query_amp_caps(codec, dac, HDA_OUTPUT);
4187		if (caps & AC_AMPCAP_NUM_STEPS)
4188			return cx_auto_add_pb_volume(codec, dac, name, idx);
4189	}
4190	caps = query_amp_caps(codec, pin, HDA_OUTPUT);
4191	if (caps & AC_AMPCAP_NUM_STEPS)
4192		return cx_auto_add_pb_volume(codec, pin, name, idx);
4193	return 0;
4194}
4195
4196static int cx_auto_build_output_controls(struct hda_codec *codec)
4197{
4198	struct conexant_spec *spec = codec->spec;
4199	int i, err;
4200	int num_line = 0, num_hp = 0, num_spk = 0;
4201	static const char * const texts[3] = { "Front", "Surround", "CLFE" };
4202
4203	if (spec->dac_info_filled == 1)
4204		return try_add_pb_volume(codec, spec->dac_info[0].dac,
4205					 spec->dac_info[0].pin,
4206					 "Master", 0);
4207
4208	for (i = 0; i < spec->dac_info_filled; i++) {
4209		const char *label;
4210		int idx, type;
4211		hda_nid_t dac = spec->dac_info[i].dac;
4212		type = spec->dac_info[i].type;
4213		if (type == AUTO_PIN_LINE_OUT)
4214			type = spec->autocfg.line_out_type;
4215		switch (type) {
4216		case AUTO_PIN_LINE_OUT:
4217		default:
4218			label = texts[num_line++];
4219			idx = 0;
4220			break;
4221		case AUTO_PIN_HP_OUT:
4222			label = "Headphone";
4223			idx = num_hp++;
4224			break;
4225		case AUTO_PIN_SPEAKER_OUT:
4226			label = "Speaker";
4227			idx = num_spk++;
4228			break;
4229		}
4230		err = try_add_pb_volume(codec, dac,
4231					spec->dac_info[i].pin,
4232					label, idx);
4233		if (err < 0)
4234			return err;
4235	}
4236
4237	if (spec->auto_mute) {
4238		err = snd_hda_add_new_ctls(codec, cx_automute_mode_enum);
4239		if (err < 0)
4240			return err;
4241	}
4242	
4243	return 0;
4244}
4245
4246static int cx_auto_add_capture_volume(struct hda_codec *codec, hda_nid_t nid,
4247				      const char *label, const char *pfx,
4248				      int cidx)
4249{
4250	struct conexant_spec *spec = codec->spec;
4251	int i;
4252
4253	for (i = 0; i < spec->num_adc_nids; i++) {
4254		hda_nid_t adc_nid = spec->adc_nids[i];
4255		int idx = get_input_connection(codec, adc_nid, nid);
4256		if (idx < 0)
4257			continue;
4258		return cx_auto_add_volume_idx(codec, label, pfx,
4259					      cidx, adc_nid, HDA_INPUT, idx);
4260	}
4261	return 0;
4262}
4263
4264static int cx_auto_add_boost_volume(struct hda_codec *codec, int idx,
4265				    const char *label, int cidx)
4266{
4267	struct conexant_spec *spec = codec->spec;
4268	hda_nid_t mux, nid;
4269	int i, con;
4270
4271	nid = spec->imux_info[idx].pin;
4272	if (get_wcaps(codec, nid) & AC_WCAP_IN_AMP)
4273		return cx_auto_add_volume(codec, label, " Boost", cidx,
4274					  nid, HDA_INPUT);
4275	con = __select_input_connection(codec, spec->imux_info[idx].adc, nid,
4276					&mux, false, 0);
4277	if (con < 0)
4278		return 0;
4279	for (i = 0; i < idx; i++) {
4280		if (spec->imux_info[i].boost == mux)
4281			return 0; /* already present */
4282	}
4283
4284	if (get_wcaps(codec, mux) & AC_WCAP_OUT_AMP) {
4285		spec->imux_info[idx].boost = mux;
4286		return cx_auto_add_volume(codec, label, " Boost", 0,
4287					  mux, HDA_OUTPUT);
4288	}
4289	return 0;
4290}
4291
4292static int cx_auto_build_input_controls(struct hda_codec *codec)
4293{
4294	struct conexant_spec *spec = codec->spec;
4295	struct hda_input_mux *imux = &spec->private_imux;
4296	const char *prev_label;
4297	int input_conn[HDA_MAX_NUM_INPUTS];
4298	int i, err, cidx;
4299	int multi_connection;
4300
4301	multi_connection = 0;
4302	for (i = 0; i < imux->num_items; i++) {
4303		cidx = get_input_connection(codec, spec->imux_info[i].adc,
4304					    spec->imux_info[i].pin);
4305		input_conn[i] = (spec->imux_info[i].adc << 8) | cidx;
4306		if (i > 0 && input_conn[i] != input_conn[0])
4307			multi_connection = 1;
4308	}
4309
4310	prev_label = NULL;
4311	cidx = 0;
4312	for (i = 0; i < imux->num_items; i++) {
4313		hda_nid_t nid = spec->imux_info[i].pin;
4314		const char *label;
4315
4316		label = hda_get_autocfg_input_label(codec, &spec->autocfg,
4317						    spec->imux_info[i].index);
4318		if (label == prev_label)
4319			cidx++;
4320		else
4321			cidx = 0;
4322		prev_label = label;
4323
4324		err = cx_auto_add_boost_volume(codec, i, label, cidx);
4325		if (err < 0)
4326			return err;
4327
4328		if (!multi_connection) {
4329			if (i > 0)
4330				continue;
4331			err = cx_auto_add_capture_volume(codec, nid,
4332							 "Capture", "", cidx);
4333		} else {
4334			err = cx_auto_add_capture_volume(codec, nid,
4335							 label, " Capture", cidx);
4336		}
4337		if (err < 0)
4338			return err;
4339	}
4340
4341	if (spec->private_imux.num_items > 1 && !spec->auto_mic) {
4342		err = snd_hda_add_new_ctls(codec, cx_auto_capture_mixers);
4343		if (err < 0)
4344			return err;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
4345	}
4346
4347	return 0;
4348}
4349
4350static int cx_auto_build_controls(struct hda_codec *codec)
4351{
4352	int err;
4353
4354	err = cx_auto_build_output_controls(codec);
 
4355	if (err < 0)
4356		return err;
4357	err = cx_auto_build_input_controls(codec);
 
4358	if (err < 0)
4359		return err;
4360	return conexant_build_controls(codec);
4361}
4362
4363static int cx_auto_search_adcs(struct hda_codec *codec)
4364{
4365	struct conexant_spec *spec = codec->spec;
4366	hda_nid_t nid, end_nid;
4367
4368	end_nid = codec->start_nid + codec->num_nodes;
4369	for (nid = codec->start_nid; nid < end_nid; nid++) {
4370		unsigned int caps = get_wcaps(codec, nid);
4371		if (get_wcaps_type(caps) != AC_WID_AUD_IN)
4372			continue;
4373		if (caps & AC_WCAP_DIGITAL)
4374			continue;
4375		if (snd_BUG_ON(spec->num_adc_nids >=
4376			       ARRAY_SIZE(spec->private_adc_nids)))
4377			break;
4378		spec->private_adc_nids[spec->num_adc_nids++] = nid;
4379	}
4380	spec->adc_nids = spec->private_adc_nids;
4381	return 0;
4382}
4383
 
4384
4385static const struct hda_codec_ops cx_auto_patch_ops = {
4386	.build_controls = cx_auto_build_controls,
4387	.build_pcms = conexant_build_pcms,
4388	.init = cx_auto_init,
4389	.free = conexant_free,
4390	.unsol_event = cx_auto_unsol_event,
4391#ifdef CONFIG_SND_HDA_POWER_SAVE
4392	.suspend = conexant_suspend,
4393#endif
4394	.reboot_notify = snd_hda_shutup_pins,
4395};
4396
4397static int patch_conexant_auto(struct hda_codec *codec)
4398{
4399	struct conexant_spec *spec;
4400	int err;
4401
4402	printk(KERN_INFO "hda_codec: %s: BIOS auto-probing.\n",
4403	       codec->chip_name);
4404
4405	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
4406	if (!spec)
4407		return -ENOMEM;
4408	codec->spec = spec;
4409	codec->pin_amp_workaround = 1;
4410	err = cx_auto_search_adcs(codec);
4411	if (err < 0)
4412		return err;
4413	err = cx_auto_parse_auto_config(codec);
4414	if (err < 0) {
4415		kfree(codec->spec);
4416		codec->spec = NULL;
4417		return err;
4418	}
4419	spec->capture_stream = &cx_auto_pcm_analog_capture;
4420	codec->patch_ops = cx_auto_patch_ops;
4421	if (spec->beep_amp)
4422		snd_hda_attach_beep_device(codec, spec->beep_amp);
4423	return 0;
 
 
 
 
4424}
4425
4426/*
4427 */
4428
4429static const struct hda_codec_preset snd_hda_preset_conexant[] = {
4430	{ .id = 0x14f15045, .name = "CX20549 (Venice)",
4431	  .patch = patch_cxt5045 },
4432	{ .id = 0x14f15047, .name = "CX20551 (Waikiki)",
4433	  .patch = patch_cxt5047 },
4434	{ .id = 0x14f15051, .name = "CX20561 (Hermosa)",
4435	  .patch = patch_cxt5051 },
4436	{ .id = 0x14f15066, .name = "CX20582 (Pebble)",
4437	  .patch = patch_cxt5066 },
4438	{ .id = 0x14f15067, .name = "CX20583 (Pebble HSF)",
4439	  .patch = patch_cxt5066 },
4440	{ .id = 0x14f15068, .name = "CX20584",
4441	  .patch = patch_cxt5066 },
4442	{ .id = 0x14f15069, .name = "CX20585",
4443	  .patch = patch_cxt5066 },
4444	{ .id = 0x14f1506c, .name = "CX20588",
4445	  .patch = patch_cxt5066 },
4446	{ .id = 0x14f1506e, .name = "CX20590",
4447	  .patch = patch_cxt5066 },
4448	{ .id = 0x14f15097, .name = "CX20631",
4449	  .patch = patch_conexant_auto },
4450	{ .id = 0x14f15098, .name = "CX20632",
4451	  .patch = patch_conexant_auto },
4452	{ .id = 0x14f150a1, .name = "CX20641",
4453	  .patch = patch_conexant_auto },
4454	{ .id = 0x14f150a2, .name = "CX20642",
4455	  .patch = patch_conexant_auto },
4456	{ .id = 0x14f150ab, .name = "CX20651",
4457	  .patch = patch_conexant_auto },
4458	{ .id = 0x14f150ac, .name = "CX20652",
4459	  .patch = patch_conexant_auto },
4460	{ .id = 0x14f150b8, .name = "CX20664",
4461	  .patch = patch_conexant_auto },
4462	{ .id = 0x14f150b9, .name = "CX20665",
4463	  .patch = patch_conexant_auto },
4464	{} /* terminator */
4465};
4466
4467MODULE_ALIAS("snd-hda-codec-id:14f15045");
4468MODULE_ALIAS("snd-hda-codec-id:14f15047");
4469MODULE_ALIAS("snd-hda-codec-id:14f15051");
4470MODULE_ALIAS("snd-hda-codec-id:14f15066");
4471MODULE_ALIAS("snd-hda-codec-id:14f15067");
4472MODULE_ALIAS("snd-hda-codec-id:14f15068");
4473MODULE_ALIAS("snd-hda-codec-id:14f15069");
4474MODULE_ALIAS("snd-hda-codec-id:14f1506c");
4475MODULE_ALIAS("snd-hda-codec-id:14f1506e");
4476MODULE_ALIAS("snd-hda-codec-id:14f15097");
4477MODULE_ALIAS("snd-hda-codec-id:14f15098");
4478MODULE_ALIAS("snd-hda-codec-id:14f150a1");
4479MODULE_ALIAS("snd-hda-codec-id:14f150a2");
4480MODULE_ALIAS("snd-hda-codec-id:14f150ab");
4481MODULE_ALIAS("snd-hda-codec-id:14f150ac");
4482MODULE_ALIAS("snd-hda-codec-id:14f150b8");
4483MODULE_ALIAS("snd-hda-codec-id:14f150b9");
4484
4485MODULE_LICENSE("GPL");
4486MODULE_DESCRIPTION("Conexant HD-audio codec");
4487
4488static struct hda_codec_preset_list conexant_list = {
4489	.preset = snd_hda_preset_conexant,
4490	.owner = THIS_MODULE,
4491};
4492
4493static int __init patch_conexant_init(void)
4494{
4495	return snd_hda_add_codec_preset(&conexant_list);
4496}
4497
4498static void __exit patch_conexant_exit(void)
4499{
4500	snd_hda_delete_codec_preset(&conexant_list);
4501}
4502
4503module_init(patch_conexant_init)
4504module_exit(patch_conexant_exit)
v6.8
   1// SPDX-License-Identifier: GPL-2.0-or-later
   2/*
   3 * HD audio interface patch for Conexant HDA audio codec
   4 *
   5 * Copyright (c) 2006 Pototskiy Akex <alex.pototskiy@gmail.com>
   6 * 		      Takashi Iwai <tiwai@suse.de>
   7 * 		      Tobin Davis  <tdavis@dsl-only.net>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
   8 */
   9
  10#include <linux/init.h>
  11#include <linux/delay.h>
  12#include <linux/slab.h>
  13#include <linux/module.h>
  14#include <sound/core.h>
  15#include <sound/jack.h>
  16
  17#include <sound/hda_codec.h>
  18#include "hda_local.h"
  19#include "hda_auto_parser.h"
  20#include "hda_beep.h"
  21#include "hda_jack.h"
  22#include "hda_generic.h"
  23
  24enum {
  25	CX_HEADSET_NOPRESENT = 0,
  26	CX_HEADSET_PARTPRESENT,
  27	CX_HEADSET_ALLPRESENT,
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  28};
  29
  30struct conexant_spec {
  31	struct hda_gen_spec gen;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  32
  33	/* extra EAPD pins */
  34	unsigned int num_eapds;
  35	hda_nid_t eapds[4];
  36	bool dynamic_eapd;
  37	hda_nid_t mute_led_eapd;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  38
  39	unsigned int parse_flags; /* flag for snd_hda_parse_pin_defcfg() */
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  40
  41	/* OPLC XO specific */
  42	bool recording;
  43	bool dc_enable;
  44	unsigned int dc_input_bias; /* offset into olpc_xo_dc_bias */
  45	struct nid_path *dc_mode_path;
  46
  47	int mute_led_polarity;
  48	unsigned int gpio_led;
  49	unsigned int gpio_mute_led_mask;
  50	unsigned int gpio_mic_led_mask;
  51	unsigned int headset_present_flag;
  52	bool is_cx8070_sn6140;
 
 
 
 
 
 
 
  53};
  54
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  55
  56#ifdef CONFIG_SND_HDA_INPUT_BEEP
  57/* additional beep mixers; private_value will be overwritten */
  58static const struct snd_kcontrol_new cxt_beep_mixer[] = {
  59	HDA_CODEC_VOLUME_MONO("Beep Playback Volume", 0, 1, 0, HDA_OUTPUT),
  60	HDA_CODEC_MUTE_BEEP_MONO("Beep Playback Switch", 0, 1, 0, HDA_OUTPUT),
 
  61};
 
  62
  63static int set_beep_amp(struct conexant_spec *spec, hda_nid_t nid,
  64			int idx, int dir)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  65{
  66	struct snd_kcontrol_new *knew;
  67	unsigned int beep_amp = HDA_COMPOSE_AMP_VAL(nid, 1, idx, dir);
  68	int i;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  69
  70	spec->gen.beep_nid = nid;
  71	for (i = 0; i < ARRAY_SIZE(cxt_beep_mixer); i++) {
  72		knew = snd_hda_gen_add_kctl(&spec->gen, NULL,
  73					    &cxt_beep_mixer[i]);
  74		if (!knew)
  75			return -ENOMEM;
  76		knew->private_value = beep_amp;
 
 
 
 
 
 
 
  77	}
 
 
  78	return 0;
  79}
  80
  81static int cx_auto_parse_beep(struct hda_codec *codec)
 
  82{
  83	struct conexant_spec *spec = codec->spec;
  84	hda_nid_t nid;
  85
  86	for_each_hda_codec_node(nid, codec)
  87		if (get_wcaps_type(get_wcaps(codec, nid)) == AC_WID_BEEP)
  88			return set_beep_amp(spec, nid, 0, HDA_OUTPUT);
  89	return 0;
  90}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  91#else
  92#define cx_auto_parse_beep(codec)	0
  93#endif
  94
 
  95/*
  96 * Automatic parser for CX20641 & co
 
  97 */
  98
  99/* parse EAPDs */
 100static void cx_auto_parse_eapd(struct hda_codec *codec)
 
 
 101{
 
 102	struct conexant_spec *spec = codec->spec;
 103	hda_nid_t nid;
 
 
 
 
 
 104
 105	for_each_hda_codec_node(nid, codec) {
 106		if (get_wcaps_type(get_wcaps(codec, nid)) != AC_WID_PIN)
 107			continue;
 108		if (!(snd_hda_query_pin_caps(codec, nid) & AC_PINCAP_EAPD))
 109			continue;
 110		spec->eapds[spec->num_eapds++] = nid;
 111		if (spec->num_eapds >= ARRAY_SIZE(spec->eapds))
 112			break;
 113	}
 114
 115	/* NOTE: below is a wild guess; if we have more than two EAPDs,
 116	 * it's a new chip, where EAPDs are supposed to be associated to
 117	 * pins, and we can control EAPD per pin.
 118	 * OTOH, if only one or two EAPDs are found, it's an old chip,
 119	 * thus it might control over all pins.
 120	 */
 121	if (spec->num_eapds > 2)
 122		spec->dynamic_eapd = 1;
 
 
 
 
 
 
 
 
 
 
 
 
 123}
 124
 125static void cx_auto_turn_eapd(struct hda_codec *codec, int num_pins,
 126			      const hda_nid_t *pins, bool on)
 
 
 
 
 
 
 
 
 
 
 
 
 127{
 128	int i;
 129	for (i = 0; i < num_pins; i++) {
 130		if (snd_hda_query_pin_caps(codec, pins[i]) & AC_PINCAP_EAPD)
 131			snd_hda_codec_write(codec, pins[i], 0,
 132					    AC_VERB_SET_EAPD_BTLENABLE,
 133					    on ? 0x02 : 0);
 134	}
 135}
 136
 137/* turn on/off EAPD according to Master switch */
 138static void cx_auto_vmaster_hook(void *private_data, int enabled)
 139{
 140	struct hda_codec *codec = private_data;
 141	struct conexant_spec *spec = codec->spec;
 142
 143	cx_auto_turn_eapd(codec, spec->num_eapds, spec->eapds, enabled);
 
 144}
 145
 146/* turn on/off EAPD according to Master switch (inversely!) for mute LED */
 147static int cx_auto_vmaster_mute_led(struct led_classdev *led_cdev,
 148				    enum led_brightness brightness)
 149{
 150	struct hda_codec *codec = dev_to_hda_codec(led_cdev->dev->parent);
 151	struct conexant_spec *spec = codec->spec;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 152
 153	snd_hda_codec_write(codec, spec->mute_led_eapd, 0,
 154			    AC_VERB_SET_EAPD_BTLENABLE,
 155			    brightness ? 0x02 : 0x00);
 156	return 0;
 157}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 158
 159static void cxt_init_gpio_led(struct hda_codec *codec)
 
 
 160{
 
 161	struct conexant_spec *spec = codec->spec;
 162	unsigned int mask = spec->gpio_mute_led_mask | spec->gpio_mic_led_mask;
 
 
 
 163
 164	if (mask) {
 165		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_MASK,
 166				    mask);
 167		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DIRECTION,
 168				    mask);
 169		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
 170				    spec->gpio_led);
 171	}
 
 
 
 172}
 173
 174static void cx_fixup_headset_recog(struct hda_codec *codec)
 
 
 
 
 
 
 
 
 
 
 
 175{
 176	unsigned int mic_persent;
 
 
 
 
 
 
 
 
 
 
 177
 178	/* fix some headset type recognize fail issue, such as EDIFIER headset */
 179	/* set micbiasd output current comparator threshold from 66% to 55%. */
 180	snd_hda_codec_write(codec, 0x1c, 0, 0x320, 0x010);
 181	/* set OFF voltage for DFET from -1.2V to -0.8V, set headset micbias registor
 182	 * value adjustment trim from 2.2K ohms to 2.0K ohms.
 183	 */
 184	snd_hda_codec_write(codec, 0x1c, 0, 0x3b0, 0xe10);
 185	/* fix reboot headset type recognize fail issue */
 186	mic_persent = snd_hda_codec_read(codec, 0x19, 0, AC_VERB_GET_PIN_SENSE, 0x0);
 187	if (mic_persent & AC_PINSENSE_PRESENCE)
 188		/* enable headset mic VREF */
 189		snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24);
 190	else
 191		/* disable headset mic VREF */
 192		snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20);
 193}
 194
 195static int cx_auto_init(struct hda_codec *codec)
 
 
 196{
 197	struct conexant_spec *spec = codec->spec;
 198	snd_hda_gen_init(codec);
 199	if (!spec->dynamic_eapd)
 200		cx_auto_turn_eapd(codec, spec->num_eapds, spec->eapds, true);
 201
 202	cxt_init_gpio_led(codec);
 203	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_INIT);
 204
 205	if (spec->is_cx8070_sn6140)
 206		cx_fixup_headset_recog(codec);
 207
 208	return 0;
 209}
 210
 211static void cx_auto_shutdown(struct hda_codec *codec)
 
 
 212{
 213	struct conexant_spec *spec = codec->spec;
 
 
 
 
 
 
 
 214
 215	/* Turn the problematic codec into D3 to avoid spurious noises
 216	   from the internal speaker during (and after) reboot */
 217	cx_auto_turn_eapd(codec, spec->num_eapds, spec->eapds, false);
 218}
 219
 220static void cx_auto_free(struct hda_codec *codec)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 221{
 222	cx_auto_shutdown(codec);
 223	snd_hda_gen_free(codec);
 
 224}
 225
 226static void cx_process_headset_plugin(struct hda_codec *codec)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 227{
 228	unsigned int val;
 229	unsigned int count = 0;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 230
 231	/* Wait headset detect done. */
 232	do {
 233		val = snd_hda_codec_read(codec, 0x1c, 0, 0xca0, 0x0);
 234		if (val & 0x080) {
 235			codec_dbg(codec, "headset type detect done!\n");
 236			break;
 237		}
 238		msleep(20);
 239		count++;
 240	} while (count < 3);
 241	val = snd_hda_codec_read(codec, 0x1c, 0, 0xcb0, 0x0);
 242	if (val & 0x800) {
 243		codec_dbg(codec, "headset plugin, type is CTIA\n");
 244		snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24);
 245	} else if (val & 0x400) {
 246		codec_dbg(codec, "headset plugin, type is OMTP\n");
 247		snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x24);
 248	} else {
 249		codec_dbg(codec, "headphone plugin\n");
 250	}
 
 
 
 
 
 251}
 252
 253static void cx_update_headset_mic_vref(struct hda_codec *codec, unsigned int res)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 254{
 255	unsigned int phone_present, mic_persent, phone_tag, mic_tag;
 256	struct conexant_spec *spec = codec->spec;
 
 257
 258	/* In cx8070 and sn6140, the node 16 can only be config to headphone or disabled,
 259	 * the node 19 can only be config to microphone or disabled.
 260	 * Check hp&mic tag to process headset pulgin&plugout.
 
 
 
 
 
 
 
 261	 */
 262	phone_tag = snd_hda_codec_read(codec, 0x16, 0, AC_VERB_GET_UNSOLICITED_RESPONSE, 0x0);
 263	mic_tag = snd_hda_codec_read(codec, 0x19, 0, AC_VERB_GET_UNSOLICITED_RESPONSE, 0x0);
 264	if ((phone_tag & (res >> AC_UNSOL_RES_TAG_SHIFT)) ||
 265	    (mic_tag & (res >> AC_UNSOL_RES_TAG_SHIFT))) {
 266		phone_present = snd_hda_codec_read(codec, 0x16, 0, AC_VERB_GET_PIN_SENSE, 0x0);
 267		if (!(phone_present & AC_PINSENSE_PRESENCE)) {/* headphone plugout */
 268			spec->headset_present_flag = CX_HEADSET_NOPRESENT;
 269			snd_hda_codec_write(codec, 0x19, 0, AC_VERB_SET_PIN_WIDGET_CONTROL, 0x20);
 270			return;
 271		}
 272		if (spec->headset_present_flag == CX_HEADSET_NOPRESENT) {
 273			spec->headset_present_flag = CX_HEADSET_PARTPRESENT;
 274		} else if (spec->headset_present_flag == CX_HEADSET_PARTPRESENT) {
 275			mic_persent = snd_hda_codec_read(codec, 0x19, 0,
 276							 AC_VERB_GET_PIN_SENSE, 0x0);
 277			/* headset is present */
 278			if ((phone_present & AC_PINSENSE_PRESENCE) &&
 279			    (mic_persent & AC_PINSENSE_PRESENCE)) {
 280				cx_process_headset_plugin(codec);
 281				spec->headset_present_flag = CX_HEADSET_ALLPRESENT;
 282			}
 283		}
 284	}
 285}
 286
 287static void cx_jack_unsol_event(struct hda_codec *codec, unsigned int res)
 
 288{
 289	struct conexant_spec *spec = codec->spec;
 
 290
 291	if (spec->is_cx8070_sn6140)
 292		cx_update_headset_mic_vref(codec, res);
 293
 294	snd_hda_jack_unsol_event(codec, res);
 
 
 
 295}
 296
 297#ifdef CONFIG_PM
 298static int cx_auto_suspend(struct hda_codec *codec)
 299{
 300	cx_auto_shutdown(codec);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 301	return 0;
 302}
 
 
 
 
 
 
 
 
 303#endif
 
 
 
 304
 305static const struct hda_codec_ops cx_auto_patch_ops = {
 306	.build_controls = snd_hda_gen_build_controls,
 307	.build_pcms = snd_hda_gen_build_pcms,
 308	.init = cx_auto_init,
 309	.free = cx_auto_free,
 310	.unsol_event = cx_jack_unsol_event,
 311#ifdef CONFIG_PM
 312	.suspend = cx_auto_suspend,
 313	.check_power_status = snd_hda_gen_check_power_status,
 314#endif
 
 315};
 316
 317/*
 318 * pin fix-up
 319 */
 320enum {
 321	CXT_PINCFG_LENOVO_X200,
 322	CXT_PINCFG_LENOVO_TP410,
 323	CXT_PINCFG_LEMOTE_A1004,
 324	CXT_PINCFG_LEMOTE_A1205,
 325	CXT_PINCFG_COMPAQ_CQ60,
 326	CXT_FIXUP_STEREO_DMIC,
 327	CXT_PINCFG_LENOVO_NOTEBOOK,
 328	CXT_FIXUP_INC_MIC_BOOST,
 329	CXT_FIXUP_HEADPHONE_MIC_PIN,
 330	CXT_FIXUP_HEADPHONE_MIC,
 331	CXT_FIXUP_GPIO1,
 332	CXT_FIXUP_ASPIRE_DMIC,
 333	CXT_FIXUP_THINKPAD_ACPI,
 334	CXT_FIXUP_OLPC_XO,
 335	CXT_FIXUP_CAP_MIX_AMP,
 336	CXT_FIXUP_TOSHIBA_P105,
 337	CXT_FIXUP_HP_530,
 338	CXT_FIXUP_CAP_MIX_AMP_5047,
 339	CXT_FIXUP_MUTE_LED_EAPD,
 340	CXT_FIXUP_HP_DOCK,
 341	CXT_FIXUP_HP_SPECTRE,
 342	CXT_FIXUP_HP_GATE_MIC,
 343	CXT_FIXUP_MUTE_LED_GPIO,
 344	CXT_FIXUP_HP_ZBOOK_MUTE_LED,
 345	CXT_FIXUP_HEADSET_MIC,
 346	CXT_FIXUP_HP_MIC_NO_PRESENCE,
 347	CXT_PINCFG_SWS_JS201D,
 348};
 349
 350/* for hda_fixup_thinkpad_acpi() */
 351#include "thinkpad_helper.c"
 352
 353static void cxt_fixup_stereo_dmic(struct hda_codec *codec,
 354				  const struct hda_fixup *fix, int action)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 355{
 356	struct conexant_spec *spec = codec->spec;
 357	spec->gen.inv_dmic_split = 1;
 
 
 
 
 
 
 
 
 
 
 
 
 
 358}
 359
 360static void cxt5066_increase_mic_boost(struct hda_codec *codec,
 361				   const struct hda_fixup *fix, int action)
 
 362{
 363	if (action != HDA_FIXUP_ACT_PRE_PROBE)
 364		return;
 365
 366	snd_hda_override_amp_caps(codec, 0x17, HDA_OUTPUT,
 367				  (0x3 << AC_AMPCAP_OFFSET_SHIFT) |
 368				  (0x4 << AC_AMPCAP_NUM_STEPS_SHIFT) |
 369				  (0x27 << AC_AMPCAP_STEP_SIZE_SHIFT) |
 370				  (0 << AC_AMPCAP_MUTE_SHIFT));
 371}
 372
 373static void cxt_update_headset_mode(struct hda_codec *codec)
 
 374{
 375	/* The verbs used in this function were tested on a Conexant CX20751/2 codec. */
 376	int i;
 377	bool mic_mode = false;
 378	struct conexant_spec *spec = codec->spec;
 379	struct auto_pin_cfg *cfg = &spec->gen.autocfg;
 380
 381	hda_nid_t mux_pin = spec->gen.imux_pins[spec->gen.cur_mux[0]];
 
 
 
 
 
 
 382
 383	for (i = 0; i < cfg->num_inputs; i++)
 384		if (cfg->inputs[i].pin == mux_pin) {
 385			mic_mode = !!cfg->inputs[i].is_headphone_mic;
 386			break;
 387		}
 
 388
 389	if (mic_mode) {
 390		snd_hda_codec_write_cache(codec, 0x1c, 0, 0x410, 0x7c); /* enable merged mode for analog int-mic */
 391		spec->gen.hp_jack_present = false;
 392	} else {
 393		snd_hda_codec_write_cache(codec, 0x1c, 0, 0x410, 0x54); /* disable merged mode for analog int-mic */
 394		spec->gen.hp_jack_present = snd_hda_jack_detect(codec, spec->gen.autocfg.hp_pins[0]);
 
 
 
 
 
 
 
 
 
 395	}
 
 
 
 
 
 
 
 
 
 
 396
 397	snd_hda_gen_update_outputs(codec);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 398}
 399
 400static void cxt_update_headset_mode_hook(struct hda_codec *codec,
 401					 struct snd_kcontrol *kcontrol,
 402					 struct snd_ctl_elem_value *ucontrol)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 403{
 404	cxt_update_headset_mode(codec);
 
 
 
 
 405}
 406
 407static void cxt_fixup_headphone_mic(struct hda_codec *codec,
 408				    const struct hda_fixup *fix, int action)
 
 
 
 
 
 
 
 409{
 410	struct conexant_spec *spec = codec->spec;
 411
 412	switch (action) {
 413	case HDA_FIXUP_ACT_PRE_PROBE:
 414		spec->parse_flags |= HDA_PINCFG_HEADPHONE_MIC;
 415		snd_hdac_regmap_add_vendor_verb(&codec->core, 0x410);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 416		break;
 417	case HDA_FIXUP_ACT_PROBE:
 418		WARN_ON(spec->gen.cap_sync_hook);
 419		spec->gen.cap_sync_hook = cxt_update_headset_mode_hook;
 420		spec->gen.automute_hook = cxt_update_headset_mode;
 
 
 421		break;
 422	case HDA_FIXUP_ACT_INIT:
 423		cxt_update_headset_mode(codec);
 
 
 
 
 
 
 
 
 
 
 
 424		break;
 425	}
 
 
 
 
 
 426}
 427
 428static void cxt_fixup_headset_mic(struct hda_codec *codec,
 429				    const struct hda_fixup *fix, int action)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 430{
 431	struct conexant_spec *spec = codec->spec;
 
 
 
 
 432
 433	switch (action) {
 434	case HDA_FIXUP_ACT_PRE_PROBE:
 435		spec->parse_flags |= HDA_PINCFG_HEADSET_MIC;
 436		break;
 
 
 
 
 
 
 
 
 
 
 
 437	}
 
 
 
 
 
 
 
 438}
 439
 440/* OPLC XO 1.5 fixup */
 
 
 
 
 441
 442/* OLPC XO-1.5 supports DC input mode (e.g. for use with analog sensors)
 443 * through the microphone jack.
 444 * When the user enables this through a mixer switch, both internal and
 445 * external microphones are disabled. Gain is fixed at 0dB. In this mode,
 446 * we also allow the bias to be configured through a separate mixer
 447 * control. */
 448
 449#define update_mic_pin(codec, nid, val)					\
 450	snd_hda_codec_write_cache(codec, nid, 0,			\
 451				   AC_VERB_SET_PIN_WIDGET_CONTROL, val)
 452
 453static const struct hda_input_mux olpc_xo_dc_bias = {
 454	.num_items = 3,
 455	.items = {
 456		{ "Off", PIN_IN },
 457		{ "50%", PIN_VREF50 },
 458		{ "80%", PIN_VREF80 },
 459	},
 460};
 461
 462static void olpc_xo_update_mic_boost(struct hda_codec *codec)
 463{
 464	struct conexant_spec *spec = codec->spec;
 465	int ch, val;
 
 
 
 
 
 
 466
 467	for (ch = 0; ch < 2; ch++) {
 468		val = AC_AMP_SET_OUTPUT |
 469			(ch ? AC_AMP_SET_RIGHT : AC_AMP_SET_LEFT);
 470		if (!spec->dc_enable)
 471			val |= snd_hda_codec_amp_read(codec, 0x17, ch, HDA_OUTPUT, 0);
 472		snd_hda_codec_write(codec, 0x17, 0,
 473				    AC_VERB_SET_AMP_GAIN_MUTE, val);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 474	}
 
 
 
 
 
 
 
 
 
 
 
 475}
 476
 477static void olpc_xo_update_mic_pins(struct hda_codec *codec)
 
 478{
 479	struct conexant_spec *spec = codec->spec;
 480	int cur_input, val;
 481	struct nid_path *path;
 
 
 482
 483	cur_input = spec->gen.input_paths[0][spec->gen.cur_mux[0]];
 
 
 
 
 
 
 
 
 
 484
 485	/* Set up mic pins for port-B, C and F dynamically as the recording
 486	 * LED is turned on/off by these pin controls
 487	 */
 488	if (!spec->dc_enable) {
 489		/* disable DC bias path and pin for port F */
 490		update_mic_pin(codec, 0x1e, 0);
 491		snd_hda_activate_path(codec, spec->dc_mode_path, false, false);
 492
 493		/* update port B (ext mic) and C (int mic) */
 494		/* OLPC defers mic widget control until when capture is
 495		 * started because the microphone LED comes on as soon as
 496		 * these settings are put in place. if we did this before
 497		 * recording, it would give the false indication that
 498		 * recording is happening when it is not.
 499		 */
 500		update_mic_pin(codec, 0x1a, spec->recording ?
 501			       snd_hda_codec_get_pin_target(codec, 0x1a) : 0);
 502		update_mic_pin(codec, 0x1b, spec->recording ?
 503			       snd_hda_codec_get_pin_target(codec, 0x1b) : 0);
 504		/* enable normal mic path */
 505		path = snd_hda_get_path_from_idx(codec, cur_input);
 506		if (path)
 507			snd_hda_activate_path(codec, path, true, false);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 508	} else {
 509		/* disable normal mic path */
 510		path = snd_hda_get_path_from_idx(codec, cur_input);
 511		if (path)
 512			snd_hda_activate_path(codec, path, false, false);
 513
 514		/* Even though port F is the DC input, the bias is controlled
 515		 * on port B.  We also leave that port as an active input (but
 516		 * unselected) in DC mode just in case that is necessary to
 517		 * make the bias setting take effect.
 518		 */
 519		if (spec->recording)
 520			val = olpc_xo_dc_bias.items[spec->dc_input_bias].index;
 521		else
 522			val = 0;
 523		update_mic_pin(codec, 0x1a, val);
 524		update_mic_pin(codec, 0x1b, 0);
 525		/* enable DC bias path and pin */
 526		update_mic_pin(codec, 0x1e, spec->recording ? PIN_IN : 0);
 527		snd_hda_activate_path(codec, spec->dc_mode_path, true, false);
 528	}
 529}
 530
 531/* mic_autoswitch hook */
 532static void olpc_xo_automic(struct hda_codec *codec,
 533			    struct hda_jack_callback *jack)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 534{
 535	struct conexant_spec *spec = codec->spec;
 536
 537	/* in DC mode, we don't handle automic */
 538	if (!spec->dc_enable)
 539		snd_hda_gen_mic_autoswitch(codec, jack);
 540	olpc_xo_update_mic_pins(codec);
 541	if (spec->dc_enable)
 542		olpc_xo_update_mic_boost(codec);
 
 
 
 
 543}
 544
 545/* pcm_capture hook */
 546static void olpc_xo_capture_hook(struct hda_pcm_stream *hinfo,
 547				 struct hda_codec *codec,
 548				 struct snd_pcm_substream *substream,
 549				 int action)
 550{
 551	struct conexant_spec *spec = codec->spec;
 
 
 
 
 
 
 
 
 
 
 
 
 552
 553	/* toggle spec->recording flag and update mic pins accordingly
 554	 * for turning on/off LED
 555	 */
 556	switch (action) {
 557	case HDA_GEN_PCM_ACT_PREPARE:
 558		spec->recording = 1;
 559		olpc_xo_update_mic_pins(codec);
 560		break;
 561	case HDA_GEN_PCM_ACT_CLEANUP:
 562		spec->recording = 0;
 563		olpc_xo_update_mic_pins(codec);
 564		break;
 565	}
 566}
 567
 568static int olpc_xo_dc_mode_get(struct snd_kcontrol *kcontrol,
 569			       struct snd_ctl_elem_value *ucontrol)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 570{
 571	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 572	struct conexant_spec *spec = codec->spec;
 573	ucontrol->value.integer.value[0] = spec->dc_enable;
 574	return 0;
 575}
 576
 577static int olpc_xo_dc_mode_put(struct snd_kcontrol *kcontrol,
 578			       struct snd_ctl_elem_value *ucontrol)
 579{
 580	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 581	struct conexant_spec *spec = codec->spec;
 582	int dc_enable = !!ucontrol->value.integer.value[0];
 583
 584	if (dc_enable == spec->dc_enable)
 585		return 0;
 586
 587	spec->dc_enable = dc_enable;
 588	olpc_xo_update_mic_pins(codec);
 589	olpc_xo_update_mic_boost(codec);
 
 
 
 590	return 1;
 591}
 592
 593static int olpc_xo_dc_bias_enum_get(struct snd_kcontrol *kcontrol,
 594				    struct snd_ctl_elem_value *ucontrol)
 
 
 
 
 
 
 595{
 596	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 597	struct conexant_spec *spec = codec->spec;
 598	ucontrol->value.enumerated.item[0] = spec->dc_input_bias;
 599	return 0;
 600}
 601
 602static int olpc_xo_dc_bias_enum_info(struct snd_kcontrol *kcontrol,
 603				     struct snd_ctl_elem_info *uinfo)
 604{
 605	return snd_hda_input_mux_info(&olpc_xo_dc_bias, uinfo);
 606}
 607
 608static int olpc_xo_dc_bias_enum_put(struct snd_kcontrol *kcontrol,
 609				    struct snd_ctl_elem_value *ucontrol)
 610{
 611	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 612	struct conexant_spec *spec = codec->spec;
 613	const struct hda_input_mux *imux = &olpc_xo_dc_bias;
 614	unsigned int idx;
 615
 616	idx = ucontrol->value.enumerated.item[0];
 617	if (idx >= imux->num_items)
 618		idx = imux->num_items - 1;
 619	if (spec->dc_input_bias == idx)
 620		return 0;
 621
 622	spec->dc_input_bias = idx;
 623	if (spec->dc_enable)
 624		olpc_xo_update_mic_pins(codec);
 625	return 1;
 626}
 627
 628static const struct snd_kcontrol_new olpc_xo_mixers[] = {
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 629	{
 630		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 631		.name = "DC Mode Enable Switch",
 632		.info = snd_ctl_boolean_mono_info,
 633		.get = olpc_xo_dc_mode_get,
 634		.put = olpc_xo_dc_mode_put,
 635	},
 636	{
 637		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 638		.name = "DC Input Bias Enum",
 639		.info = olpc_xo_dc_bias_enum_info,
 640		.get = olpc_xo_dc_bias_enum_get,
 641		.put = olpc_xo_dc_bias_enum_put,
 642	},
 643	{}
 644};
 645
 646/* overriding mic boost put callback; update mic boost volume only when
 647 * DC mode is disabled
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 648 */
 649static int olpc_xo_mic_boost_put(struct snd_kcontrol *kcontrol,
 650				 struct snd_ctl_elem_value *ucontrol)
 
 
 
 
 651{
 652	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 653	struct conexant_spec *spec = codec->spec;
 654	int ret = snd_hda_mixer_amp_volume_put(kcontrol, ucontrol);
 655	if (ret > 0 && spec->dc_enable)
 656		olpc_xo_update_mic_boost(codec);
 657	return ret;
 
 
 
 
 658}
 659
 660static void cxt_fixup_olpc_xo(struct hda_codec *codec,
 661				    const struct hda_fixup *fix, int action)
 
 662{
 663	struct conexant_spec *spec = codec->spec;
 664	struct snd_kcontrol_new *kctl;
 665	int i;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 666
 667	if (action != HDA_FIXUP_ACT_PROBE)
 668		return;
 669
 670	spec->gen.mic_autoswitch_hook = olpc_xo_automic;
 671	spec->gen.pcm_capture_hook = olpc_xo_capture_hook;
 672	spec->dc_mode_path = snd_hda_add_new_path(codec, 0x1e, 0x14, 0);
 673
 674	snd_hda_add_new_ctls(codec, olpc_xo_mixers);
 675
 676	/* OLPC's microphone port is DC coupled for use with external sensors,
 677	 * therefore we use a 50% mic bias in order to center the input signal
 678	 * with the DC input range of the codec.
 679	 */
 680	snd_hda_codec_set_pin_target(codec, 0x1a, PIN_VREF50);
 681
 682	/* override mic boost control */
 683	snd_array_for_each(&spec->gen.kctls, i, kctl) {
 684		if (!strcmp(kctl->name, "Mic Boost Volume")) {
 685			kctl->put = olpc_xo_mic_boost_put;
 686			break;
 687		}
 688	}
 
 
 
 
 
 
 689}
 690
 691static void cxt_fixup_mute_led_eapd(struct hda_codec *codec,
 692				    const struct hda_fixup *fix, int action)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 693{
 694	struct conexant_spec *spec = codec->spec;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 695
 696	if (action == HDA_FIXUP_ACT_PRE_PROBE) {
 697		spec->mute_led_eapd = 0x1b;
 698		spec->dynamic_eapd = true;
 699		snd_hda_gen_add_mute_led_cdev(codec, cx_auto_vmaster_mute_led);
 
 
 700	}
 
 701}
 702
 703/*
 704 * Fix max input level on mixer widget to 0dB
 705 * (originally it has 0x2b steps with 0dB offset 0x14)
 706 */
 707static void cxt_fixup_cap_mix_amp(struct hda_codec *codec,
 708				  const struct hda_fixup *fix, int action)
 709{
 710	snd_hda_override_amp_caps(codec, 0x17, HDA_INPUT,
 711				  (0x14 << AC_AMPCAP_OFFSET_SHIFT) |
 712				  (0x14 << AC_AMPCAP_NUM_STEPS_SHIFT) |
 713				  (0x05 << AC_AMPCAP_STEP_SIZE_SHIFT) |
 714				  (1 << AC_AMPCAP_MUTE_SHIFT));
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 715}
 716
 717/*
 718 * Fix max input level on mixer widget to 0dB
 719 * (originally it has 0x1e steps with 0 dB offset 0x17)
 720 */
 721static void cxt_fixup_cap_mix_amp_5047(struct hda_codec *codec,
 722				  const struct hda_fixup *fix, int action)
 723{
 724	snd_hda_override_amp_caps(codec, 0x10, HDA_INPUT,
 725				  (0x17 << AC_AMPCAP_OFFSET_SHIFT) |
 726				  (0x17 << AC_AMPCAP_NUM_STEPS_SHIFT) |
 727				  (0x05 << AC_AMPCAP_STEP_SIZE_SHIFT) |
 728				  (1 << AC_AMPCAP_MUTE_SHIFT));
 729}
 730
 731static void cxt_fixup_hp_gate_mic_jack(struct hda_codec *codec,
 732				       const struct hda_fixup *fix,
 733				       int action)
 734{
 735	/* the mic pin (0x19) doesn't give an unsolicited event;
 736	 * probe the mic pin together with the headphone pin (0x16)
 737	 */
 738	if (action == HDA_FIXUP_ACT_PROBE)
 739		snd_hda_jack_set_gating_jack(codec, 0x19, 0x16);
 
 
 
 740}
 741
 742/* update LED status via GPIO */
 743static void cxt_update_gpio_led(struct hda_codec *codec, unsigned int mask,
 744				bool led_on)
 745{
 
 746	struct conexant_spec *spec = codec->spec;
 747	unsigned int oldval = spec->gpio_led;
 
 
 
 
 
 
 748
 749	if (spec->mute_led_polarity)
 750		led_on = !led_on;
 
 
 
 
 
 
 
 
 
 
 
 
 
 751
 752	if (led_on)
 753		spec->gpio_led |= mask;
 
 
 
 
 
 
 
 
 754	else
 755		spec->gpio_led &= ~mask;
 756	codec_dbg(codec, "mask:%d enabled:%d gpio_led:%d\n",
 757			mask, led_on, spec->gpio_led);
 758	if (spec->gpio_led != oldval)
 759		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
 760				    spec->gpio_led);
 761}
 762
 763/* turn on/off mute LED via GPIO per vmaster hook */
 764static int cxt_gpio_mute_update(struct led_classdev *led_cdev,
 765				enum led_brightness brightness)
 766{
 767	struct hda_codec *codec = dev_to_hda_codec(led_cdev->dev->parent);
 768	struct conexant_spec *spec = codec->spec;
 769
 770	cxt_update_gpio_led(codec, spec->gpio_mute_led_mask, brightness);
 771	return 0;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 772}
 773
 774/* turn on/off mic-mute LED via GPIO per capture hook */
 775static int cxt_gpio_micmute_update(struct led_classdev *led_cdev,
 776				   enum led_brightness brightness)
 
 
 
 
 
 
 
 
 
 
 777{
 778	struct hda_codec *codec = dev_to_hda_codec(led_cdev->dev->parent);
 779	struct conexant_spec *spec = codec->spec;
 780
 781	cxt_update_gpio_led(codec, spec->gpio_mic_led_mask, brightness);
 782	return 0;
 783}
 784
 785static void cxt_setup_mute_led(struct hda_codec *codec,
 786			       unsigned int mute, unsigned int mic_mute)
 787{
 
 788	struct conexant_spec *spec = codec->spec;
 789
 790	spec->gpio_led = 0;
 791	spec->mute_led_polarity = 0;
 792	if (mute) {
 793		snd_hda_gen_add_mute_led_cdev(codec, cxt_gpio_mute_update);
 794		spec->gpio_mute_led_mask = mute;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 795	}
 796	if (mic_mute) {
 797		snd_hda_gen_add_micmute_led_cdev(codec, cxt_gpio_micmute_update);
 798		spec->gpio_mic_led_mask = mic_mute;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 799	}
 
 800}
 801
 802static void cxt_fixup_mute_led_gpio(struct hda_codec *codec,
 803				const struct hda_fixup *fix, int action)
 804{
 805	if (action == HDA_FIXUP_ACT_PRE_PROBE)
 806		cxt_setup_mute_led(codec, 0x01, 0x02);
 807}
 808
 809static void cxt_fixup_hp_zbook_mute_led(struct hda_codec *codec,
 810					const struct hda_fixup *fix, int action)
 811{
 812	if (action == HDA_FIXUP_ACT_PRE_PROBE)
 813		cxt_setup_mute_led(codec, 0x10, 0x20);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 814}
 815
 816/* ThinkPad X200 & co with cxt5051 */
 817static const struct hda_pintbl cxt_pincfg_lenovo_x200[] = {
 818	{ 0x16, 0x042140ff }, /* HP (seq# overridden) */
 819	{ 0x17, 0x21a11000 }, /* dock-mic */
 820	{ 0x19, 0x2121103f }, /* dock-HP */
 821	{ 0x1c, 0x21440100 }, /* dock SPDIF out */
 
 
 
 
 
 
 
 
 
 
 
 
 822	{}
 823};
 824
 825/* ThinkPad 410/420/510/520, X201 & co with cxt5066 */
 826static const struct hda_pintbl cxt_pincfg_lenovo_tp410[] = {
 827	{ 0x19, 0x042110ff }, /* HP (seq# overridden) */
 828	{ 0x1a, 0x21a190f0 }, /* dock-mic */
 829	{ 0x1c, 0x212140ff }, /* dock-HP */
 830	{}
 831};
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 832
 833/* Lemote A1004/A1205 with cxt5066 */
 834static const struct hda_pintbl cxt_pincfg_lemote[] = {
 835	{ 0x1a, 0x90a10020 }, /* Internal mic */
 836	{ 0x1b, 0x03a11020 }, /* External mic */
 837	{ 0x1d, 0x400101f0 }, /* Not used */
 838	{ 0x1e, 0x40a701f0 }, /* Not used */
 839	{ 0x20, 0x404501f0 }, /* Not used */
 840	{ 0x22, 0x404401f0 }, /* Not used */
 841	{ 0x23, 0x40a701f0 }, /* Not used */
 842	{}
 843};
 844
 845/* SuoWoSi/South-holding JS201D with sn6140 */
 846static const struct hda_pintbl cxt_pincfg_sws_js201d[] = {
 847	{ 0x16, 0x03211040 }, /* hp out */
 848	{ 0x17, 0x91170110 }, /* SPK/Class_D */
 849	{ 0x18, 0x95a70130 }, /* Internal mic */
 850	{ 0x19, 0x03a11020 }, /* Headset Mic */
 851	{ 0x1a, 0x40f001f0 }, /* Not used */
 852	{ 0x21, 0x40f001f0 }, /* Not used */
 853	{}
 854};
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 855
 856static const struct hda_fixup cxt_fixups[] = {
 857	[CXT_PINCFG_LENOVO_X200] = {
 858		.type = HDA_FIXUP_PINS,
 859		.v.pins = cxt_pincfg_lenovo_x200,
 860	},
 861	[CXT_PINCFG_LENOVO_TP410] = {
 862		.type = HDA_FIXUP_PINS,
 863		.v.pins = cxt_pincfg_lenovo_tp410,
 864		.chained = true,
 865		.chain_id = CXT_FIXUP_THINKPAD_ACPI,
 866	},
 867	[CXT_PINCFG_LEMOTE_A1004] = {
 868		.type = HDA_FIXUP_PINS,
 869		.chained = true,
 870		.chain_id = CXT_FIXUP_INC_MIC_BOOST,
 871		.v.pins = cxt_pincfg_lemote,
 872	},
 873	[CXT_PINCFG_LEMOTE_A1205] = {
 874		.type = HDA_FIXUP_PINS,
 875		.v.pins = cxt_pincfg_lemote,
 876	},
 877	[CXT_PINCFG_COMPAQ_CQ60] = {
 878		.type = HDA_FIXUP_PINS,
 879		.v.pins = (const struct hda_pintbl[]) {
 880			/* 0x17 was falsely set up as a mic, it should 0x1d */
 881			{ 0x17, 0x400001f0 },
 882			{ 0x1d, 0x97a70120 },
 883			{ }
 
 
 
 
 
 
 884		}
 885	},
 886	[CXT_FIXUP_STEREO_DMIC] = {
 887		.type = HDA_FIXUP_FUNC,
 888		.v.func = cxt_fixup_stereo_dmic,
 889	},
 890	[CXT_PINCFG_LENOVO_NOTEBOOK] = {
 891		.type = HDA_FIXUP_PINS,
 892		.v.pins = (const struct hda_pintbl[]) {
 893			{ 0x1a, 0x05d71030 },
 894			{ }
 895		},
 896		.chain_id = CXT_FIXUP_STEREO_DMIC,
 897	},
 898	[CXT_FIXUP_INC_MIC_BOOST] = {
 899		.type = HDA_FIXUP_FUNC,
 900		.v.func = cxt5066_increase_mic_boost,
 901	},
 902	[CXT_FIXUP_HEADPHONE_MIC_PIN] = {
 903		.type = HDA_FIXUP_PINS,
 904		.chained = true,
 905		.chain_id = CXT_FIXUP_HEADPHONE_MIC,
 906		.v.pins = (const struct hda_pintbl[]) {
 907			{ 0x18, 0x03a1913d }, /* use as headphone mic, without its own jack detect */
 908			{ }
 909		}
 910	},
 911	[CXT_FIXUP_HEADPHONE_MIC] = {
 912		.type = HDA_FIXUP_FUNC,
 913		.v.func = cxt_fixup_headphone_mic,
 914	},
 915	[CXT_FIXUP_GPIO1] = {
 916		.type = HDA_FIXUP_VERBS,
 917		.v.verbs = (const struct hda_verb[]) {
 918			{ 0x01, AC_VERB_SET_GPIO_MASK, 0x01 },
 919			{ 0x01, AC_VERB_SET_GPIO_DIRECTION, 0x01 },
 920			{ 0x01, AC_VERB_SET_GPIO_DATA, 0x01 },
 921			{ }
 922		},
 923	},
 924	[CXT_FIXUP_ASPIRE_DMIC] = {
 925		.type = HDA_FIXUP_FUNC,
 926		.v.func = cxt_fixup_stereo_dmic,
 927		.chained = true,
 928		.chain_id = CXT_FIXUP_GPIO1,
 929	},
 930	[CXT_FIXUP_THINKPAD_ACPI] = {
 931		.type = HDA_FIXUP_FUNC,
 932		.v.func = hda_fixup_thinkpad_acpi,
 933	},
 934	[CXT_FIXUP_OLPC_XO] = {
 935		.type = HDA_FIXUP_FUNC,
 936		.v.func = cxt_fixup_olpc_xo,
 937	},
 938	[CXT_FIXUP_CAP_MIX_AMP] = {
 939		.type = HDA_FIXUP_FUNC,
 940		.v.func = cxt_fixup_cap_mix_amp,
 941	},
 942	[CXT_FIXUP_TOSHIBA_P105] = {
 943		.type = HDA_FIXUP_PINS,
 944		.v.pins = (const struct hda_pintbl[]) {
 945			{ 0x10, 0x961701f0 }, /* speaker/hp */
 946			{ 0x12, 0x02a1901e }, /* ext mic */
 947			{ 0x14, 0x95a70110 }, /* int mic */
 948			{}
 949		},
 950	},
 951	[CXT_FIXUP_HP_530] = {
 952		.type = HDA_FIXUP_PINS,
 953		.v.pins = (const struct hda_pintbl[]) {
 954			{ 0x12, 0x90a60160 }, /* int mic */
 955			{}
 956		},
 957		.chained = true,
 958		.chain_id = CXT_FIXUP_CAP_MIX_AMP,
 959	},
 960	[CXT_FIXUP_CAP_MIX_AMP_5047] = {
 961		.type = HDA_FIXUP_FUNC,
 962		.v.func = cxt_fixup_cap_mix_amp_5047,
 963	},
 964	[CXT_FIXUP_MUTE_LED_EAPD] = {
 965		.type = HDA_FIXUP_FUNC,
 966		.v.func = cxt_fixup_mute_led_eapd,
 967	},
 968	[CXT_FIXUP_HP_DOCK] = {
 969		.type = HDA_FIXUP_PINS,
 970		.v.pins = (const struct hda_pintbl[]) {
 971			{ 0x16, 0x21011020 }, /* line-out */
 972			{ 0x18, 0x2181103f }, /* line-in */
 973			{ }
 974		},
 975		.chained = true,
 976		.chain_id = CXT_FIXUP_MUTE_LED_GPIO,
 977	},
 978	[CXT_FIXUP_HP_SPECTRE] = {
 979		.type = HDA_FIXUP_PINS,
 980		.v.pins = (const struct hda_pintbl[]) {
 981			/* enable NID 0x1d for the speaker on top */
 982			{ 0x1d, 0x91170111 },
 983			{ }
 984		}
 985	},
 986	[CXT_FIXUP_HP_GATE_MIC] = {
 987		.type = HDA_FIXUP_FUNC,
 988		.v.func = cxt_fixup_hp_gate_mic_jack,
 989	},
 990	[CXT_FIXUP_MUTE_LED_GPIO] = {
 991		.type = HDA_FIXUP_FUNC,
 992		.v.func = cxt_fixup_mute_led_gpio,
 993	},
 994	[CXT_FIXUP_HP_ZBOOK_MUTE_LED] = {
 995		.type = HDA_FIXUP_FUNC,
 996		.v.func = cxt_fixup_hp_zbook_mute_led,
 997	},
 998	[CXT_FIXUP_HEADSET_MIC] = {
 999		.type = HDA_FIXUP_FUNC,
1000		.v.func = cxt_fixup_headset_mic,
1001	},
1002	[CXT_FIXUP_HP_MIC_NO_PRESENCE] = {
1003		.type = HDA_FIXUP_PINS,
1004		.v.pins = (const struct hda_pintbl[]) {
1005			{ 0x1a, 0x02a1113c },
1006			{ }
1007		},
1008		.chained = true,
1009		.chain_id = CXT_FIXUP_HEADSET_MIC,
1010	},
1011	[CXT_PINCFG_SWS_JS201D] = {
1012		.type = HDA_FIXUP_PINS,
1013		.v.pins = cxt_pincfg_sws_js201d,
1014	},
1015};
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1016
1017static const struct snd_pci_quirk cxt5045_fixups[] = {
1018	SND_PCI_QUIRK(0x103c, 0x30d5, "HP 530", CXT_FIXUP_HP_530),
1019	SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba P105", CXT_FIXUP_TOSHIBA_P105),
1020	/* HP, Packard Bell, Fujitsu-Siemens & Lenovo laptops have
1021	 * really bad sound over 0dB on NID 0x17.
1022	 */
1023	SND_PCI_QUIRK_VENDOR(0x103c, "HP", CXT_FIXUP_CAP_MIX_AMP),
1024	SND_PCI_QUIRK_VENDOR(0x1631, "Packard Bell", CXT_FIXUP_CAP_MIX_AMP),
1025	SND_PCI_QUIRK_VENDOR(0x1734, "Fujitsu", CXT_FIXUP_CAP_MIX_AMP),
1026	SND_PCI_QUIRK_VENDOR(0x17aa, "Lenovo", CXT_FIXUP_CAP_MIX_AMP),
1027	{}
1028};
1029
1030static const struct hda_model_fixup cxt5045_fixup_models[] = {
1031	{ .id = CXT_FIXUP_CAP_MIX_AMP, .name = "cap-mix-amp" },
1032	{ .id = CXT_FIXUP_TOSHIBA_P105, .name = "toshiba-p105" },
1033	{ .id = CXT_FIXUP_HP_530, .name = "hp-530" },
1034	{}
1035};
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1036
1037static const struct snd_pci_quirk cxt5047_fixups[] = {
1038	/* HP laptops have really bad sound over 0 dB on NID 0x10.
1039	 */
1040	SND_PCI_QUIRK_VENDOR(0x103c, "HP", CXT_FIXUP_CAP_MIX_AMP_5047),
1041	{}
1042};
1043
1044static const struct hda_model_fixup cxt5047_fixup_models[] = {
1045	{ .id = CXT_FIXUP_CAP_MIX_AMP_5047, .name = "cap-mix-amp" },
1046	{}
1047};
 
 
 
 
 
 
 
1048
1049static const struct snd_pci_quirk cxt5051_fixups[] = {
1050	SND_PCI_QUIRK(0x103c, 0x360b, "Compaq CQ60", CXT_PINCFG_COMPAQ_CQ60),
1051	SND_PCI_QUIRK(0x17aa, 0x20f2, "Lenovo X200", CXT_PINCFG_LENOVO_X200),
1052	{}
1053};
1054
1055static const struct hda_model_fixup cxt5051_fixup_models[] = {
1056	{ .id = CXT_PINCFG_LENOVO_X200, .name = "lenovo-x200" },
1057	{}
1058};
1059
1060static const struct snd_pci_quirk cxt5066_fixups[] = {
1061	SND_PCI_QUIRK(0x1025, 0x0543, "Acer Aspire One 522", CXT_FIXUP_STEREO_DMIC),
1062	SND_PCI_QUIRK(0x1025, 0x054c, "Acer Aspire 3830TG", CXT_FIXUP_ASPIRE_DMIC),
1063	SND_PCI_QUIRK(0x1025, 0x054f, "Acer Aspire 4830T", CXT_FIXUP_ASPIRE_DMIC),
1064	SND_PCI_QUIRK(0x103c, 0x8079, "HP EliteBook 840 G3", CXT_FIXUP_HP_DOCK),
1065	SND_PCI_QUIRK(0x103c, 0x807C, "HP EliteBook 820 G3", CXT_FIXUP_HP_DOCK),
1066	SND_PCI_QUIRK(0x103c, 0x80FD, "HP ProBook 640 G2", CXT_FIXUP_HP_DOCK),
1067	SND_PCI_QUIRK(0x103c, 0x8115, "HP Z1 Gen3", CXT_FIXUP_HP_GATE_MIC),
1068	SND_PCI_QUIRK(0x103c, 0x814f, "HP ZBook 15u G3", CXT_FIXUP_MUTE_LED_GPIO),
1069	SND_PCI_QUIRK(0x103c, 0x8174, "HP Spectre x360", CXT_FIXUP_HP_SPECTRE),
1070	SND_PCI_QUIRK(0x103c, 0x822e, "HP ProBook 440 G4", CXT_FIXUP_MUTE_LED_GPIO),
1071	SND_PCI_QUIRK(0x103c, 0x828c, "HP EliteBook 840 G4", CXT_FIXUP_HP_DOCK),
1072	SND_PCI_QUIRK(0x103c, 0x8299, "HP 800 G3 SFF", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1073	SND_PCI_QUIRK(0x103c, 0x829a, "HP 800 G3 DM", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1074	SND_PCI_QUIRK(0x103c, 0x82b4, "HP ProDesk 600 G3", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1075	SND_PCI_QUIRK(0x103c, 0x836e, "HP ProBook 455 G5", CXT_FIXUP_MUTE_LED_GPIO),
1076	SND_PCI_QUIRK(0x103c, 0x837f, "HP ProBook 470 G5", CXT_FIXUP_MUTE_LED_GPIO),
1077	SND_PCI_QUIRK(0x103c, 0x83b2, "HP EliteBook 840 G5", CXT_FIXUP_HP_DOCK),
1078	SND_PCI_QUIRK(0x103c, 0x83b3, "HP EliteBook 830 G5", CXT_FIXUP_HP_DOCK),
1079	SND_PCI_QUIRK(0x103c, 0x83d3, "HP ProBook 640 G4", CXT_FIXUP_HP_DOCK),
1080	SND_PCI_QUIRK(0x103c, 0x8402, "HP ProBook 645 G4", CXT_FIXUP_MUTE_LED_GPIO),
1081	SND_PCI_QUIRK(0x103c, 0x8427, "HP ZBook Studio G5", CXT_FIXUP_HP_ZBOOK_MUTE_LED),
1082	SND_PCI_QUIRK(0x103c, 0x844f, "HP ZBook Studio G5", CXT_FIXUP_HP_ZBOOK_MUTE_LED),
1083	SND_PCI_QUIRK(0x103c, 0x8455, "HP Z2 G4", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1084	SND_PCI_QUIRK(0x103c, 0x8456, "HP Z2 G4 SFF", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1085	SND_PCI_QUIRK(0x103c, 0x8457, "HP Z2 G4 mini", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1086	SND_PCI_QUIRK(0x103c, 0x8458, "HP Z2 G4 mini premium", CXT_FIXUP_HP_MIC_NO_PRESENCE),
1087	SND_PCI_QUIRK(0x1043, 0x138d, "Asus", CXT_FIXUP_HEADPHONE_MIC_PIN),
1088	SND_PCI_QUIRK(0x14f1, 0x0265, "SWS JS201D", CXT_PINCFG_SWS_JS201D),
1089	SND_PCI_QUIRK(0x152d, 0x0833, "OLPC XO-1.5", CXT_FIXUP_OLPC_XO),
1090	SND_PCI_QUIRK(0x17aa, 0x20f2, "Lenovo T400", CXT_PINCFG_LENOVO_TP410),
1091	SND_PCI_QUIRK(0x17aa, 0x215e, "Lenovo T410", CXT_PINCFG_LENOVO_TP410),
1092	SND_PCI_QUIRK(0x17aa, 0x215f, "Lenovo T510", CXT_PINCFG_LENOVO_TP410),
1093	SND_PCI_QUIRK(0x17aa, 0x21ce, "Lenovo T420", CXT_PINCFG_LENOVO_TP410),
1094	SND_PCI_QUIRK(0x17aa, 0x21cf, "Lenovo T520", CXT_PINCFG_LENOVO_TP410),
1095	SND_PCI_QUIRK(0x17aa, 0x21d2, "Lenovo T420s", CXT_PINCFG_LENOVO_TP410),
1096	SND_PCI_QUIRK(0x17aa, 0x21da, "Lenovo X220", CXT_PINCFG_LENOVO_TP410),
1097	SND_PCI_QUIRK(0x17aa, 0x21db, "Lenovo X220-tablet", CXT_PINCFG_LENOVO_TP410),
1098	SND_PCI_QUIRK(0x17aa, 0x38af, "Lenovo IdeaPad Z560", CXT_FIXUP_MUTE_LED_EAPD),
1099	SND_PCI_QUIRK(0x17aa, 0x3905, "Lenovo G50-30", CXT_FIXUP_STEREO_DMIC),
1100	SND_PCI_QUIRK(0x17aa, 0x390b, "Lenovo G50-80", CXT_FIXUP_STEREO_DMIC),
1101	SND_PCI_QUIRK(0x17aa, 0x3975, "Lenovo U300s", CXT_FIXUP_STEREO_DMIC),
1102	/* NOTE: we'd need to extend the quirk for 17aa:3977 as the same
1103	 * PCI SSID is used on multiple Lenovo models
1104	 */
1105	SND_PCI_QUIRK(0x17aa, 0x3977, "Lenovo IdeaPad U310", CXT_FIXUP_STEREO_DMIC),
1106	SND_PCI_QUIRK(0x17aa, 0x3978, "Lenovo G50-70", CXT_FIXUP_STEREO_DMIC),
1107	SND_PCI_QUIRK(0x17aa, 0x397b, "Lenovo S205", CXT_FIXUP_STEREO_DMIC),
1108	SND_PCI_QUIRK_VENDOR(0x17aa, "Thinkpad", CXT_FIXUP_THINKPAD_ACPI),
1109	SND_PCI_QUIRK(0x1c06, 0x2011, "Lemote A1004", CXT_PINCFG_LEMOTE_A1004),
1110	SND_PCI_QUIRK(0x1c06, 0x2012, "Lemote A1205", CXT_PINCFG_LEMOTE_A1205),
1111	{}
1112};
1113
1114static const struct hda_model_fixup cxt5066_fixup_models[] = {
1115	{ .id = CXT_FIXUP_STEREO_DMIC, .name = "stereo-dmic" },
1116	{ .id = CXT_FIXUP_GPIO1, .name = "gpio1" },
1117	{ .id = CXT_FIXUP_HEADPHONE_MIC_PIN, .name = "headphone-mic-pin" },
1118	{ .id = CXT_PINCFG_LENOVO_TP410, .name = "tp410" },
1119	{ .id = CXT_FIXUP_THINKPAD_ACPI, .name = "thinkpad" },
1120	{ .id = CXT_PINCFG_LEMOTE_A1004, .name = "lemote-a1004" },
1121	{ .id = CXT_PINCFG_LEMOTE_A1205, .name = "lemote-a1205" },
1122	{ .id = CXT_FIXUP_OLPC_XO, .name = "olpc-xo" },
1123	{ .id = CXT_FIXUP_MUTE_LED_EAPD, .name = "mute-led-eapd" },
1124	{ .id = CXT_FIXUP_HP_DOCK, .name = "hp-dock" },
1125	{ .id = CXT_FIXUP_MUTE_LED_GPIO, .name = "mute-led-gpio" },
1126	{ .id = CXT_FIXUP_HP_ZBOOK_MUTE_LED, .name = "hp-zbook-mute-led" },
1127	{ .id = CXT_FIXUP_HP_MIC_NO_PRESENCE, .name = "hp-mic-fix" },
1128	{ .id = CXT_PINCFG_LENOVO_NOTEBOOK, .name = "lenovo-20149" },
1129	{ .id = CXT_PINCFG_SWS_JS201D, .name = "sws-js201d" },
1130	{}
1131};
 
1132
1133/* add "fake" mute amp-caps to DACs on cx5051 so that mixer mute switches
1134 * can be created (bko#42825)
1135 */
1136static void add_cx5051_fake_mutes(struct hda_codec *codec)
1137{
1138	struct conexant_spec *spec = codec->spec;
1139	static const hda_nid_t out_nids[] = {
1140		0x10, 0x11, 0
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1141	};
1142	const hda_nid_t *p;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1143
1144	for (p = out_nids; *p; p++)
1145		snd_hda_override_amp_caps(codec, *p, HDA_OUTPUT,
1146					  AC_AMPCAP_MIN_MUTE |
1147					  query_amp_caps(codec, *p, HDA_OUTPUT));
1148	spec->gen.dac_min_mute = true;
 
 
 
 
1149}
1150
1151static int patch_conexant_auto(struct hda_codec *codec)
 
1152{
1153	struct conexant_spec *spec;
1154	int err;
 
1155
1156	codec_info(codec, "%s: BIOS auto-probing.\n", codec->core.chip_name);
 
 
 
 
 
 
 
 
 
 
 
1157
1158	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1159	if (!spec)
1160		return -ENOMEM;
1161	snd_hda_gen_spec_init(&spec->gen);
1162	codec->spec = spec;
1163	codec->patch_ops = cx_auto_patch_ops;
 
1164
1165	/* init cx8070/sn6140 flag and reset headset_present_flag */
1166	switch (codec->core.vendor_id) {
1167	case 0x14f11f86:
1168	case 0x14f11f87:
1169		spec->is_cx8070_sn6140 = true;
1170		spec->headset_present_flag = CX_HEADSET_NOPRESENT;
1171		break;
 
 
 
 
 
 
 
 
 
1172	}
1173
1174	cx_auto_parse_eapd(codec);
1175	spec->gen.own_eapd_ctl = 1;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1176
1177	switch (codec->core.vendor_id) {
1178	case 0x14f15045:
1179		codec->single_adc_amp = 1;
1180		spec->gen.mixer_nid = 0x17;
1181		spec->gen.add_stereo_mix_input = HDA_HINT_STEREO_MIX_AUTO;
1182		snd_hda_pick_fixup(codec, cxt5045_fixup_models,
1183				   cxt5045_fixups, cxt_fixups);
1184		break;
1185	case 0x14f15047:
1186		codec->pin_amp_workaround = 1;
1187		spec->gen.mixer_nid = 0x19;
1188		spec->gen.add_stereo_mix_input = HDA_HINT_STEREO_MIX_AUTO;
1189		snd_hda_pick_fixup(codec, cxt5047_fixup_models,
1190				   cxt5047_fixups, cxt_fixups);
1191		break;
1192	case 0x14f15051:
1193		add_cx5051_fake_mutes(codec);
1194		codec->pin_amp_workaround = 1;
1195		snd_hda_pick_fixup(codec, cxt5051_fixup_models,
1196				   cxt5051_fixups, cxt_fixups);
1197		break;
1198	case 0x14f15098:
1199		codec->pin_amp_workaround = 1;
1200		spec->gen.mixer_nid = 0x22;
1201		spec->gen.add_stereo_mix_input = HDA_HINT_STEREO_MIX_AUTO;
1202		snd_hda_pick_fixup(codec, cxt5066_fixup_models,
1203				   cxt5066_fixups, cxt_fixups);
1204		break;
1205	case 0x14f150f2:
1206		codec->power_save_node = 1;
1207		fallthrough;
1208	default:
1209		codec->pin_amp_workaround = 1;
1210		snd_hda_pick_fixup(codec, cxt5066_fixup_models,
1211				   cxt5066_fixups, cxt_fixups);
1212		break;
1213	}
1214
1215	if (!spec->gen.vmaster_mute.hook && spec->dynamic_eapd)
1216		spec->gen.vmaster_mute.hook = cx_auto_vmaster_hook;
1217
1218	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
 
 
1219
1220	err = snd_hda_parse_pin_defcfg(codec, &spec->gen.autocfg, NULL,
1221				       spec->parse_flags);
1222	if (err < 0)
1223		goto error;
1224
1225	err = cx_auto_parse_beep(codec);
1226	if (err < 0)
1227		goto error;
 
 
1228
1229	err = snd_hda_gen_parse_auto_config(codec, &spec->gen.autocfg);
1230	if (err < 0)
1231		goto error;
 
1232
1233	/* Some laptops with Conexant chips show stalls in S3 resume,
1234	 * which falls into the single-cmd mode.
1235	 * Better to make reset, then.
1236	 */
1237	if (!codec->bus->core.sync_write) {
1238		codec_info(codec,
1239			   "Enable sync_write for stable communication\n");
1240		codec->bus->core.sync_write = 1;
1241		codec->bus->allow_bus_reset = 1;
 
 
1242	}
 
 
 
1243
1244	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
1245
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1246	return 0;
1247
1248 error:
1249	cx_auto_free(codec);
1250	return err;
1251}
1252
1253/*
1254 */
1255
1256static const struct hda_device_id snd_hda_id_conexant[] = {
1257	HDA_CODEC_ENTRY(0x14f11f86, "CX8070", patch_conexant_auto),
1258	HDA_CODEC_ENTRY(0x14f11f87, "SN6140", patch_conexant_auto),
1259	HDA_CODEC_ENTRY(0x14f12008, "CX8200", patch_conexant_auto),
1260	HDA_CODEC_ENTRY(0x14f120d0, "CX11970", patch_conexant_auto),
1261	HDA_CODEC_ENTRY(0x14f120d1, "SN6180", patch_conexant_auto),
1262	HDA_CODEC_ENTRY(0x14f15045, "CX20549 (Venice)", patch_conexant_auto),
1263	HDA_CODEC_ENTRY(0x14f15047, "CX20551 (Waikiki)", patch_conexant_auto),
1264	HDA_CODEC_ENTRY(0x14f15051, "CX20561 (Hermosa)", patch_conexant_auto),
1265	HDA_CODEC_ENTRY(0x14f15066, "CX20582 (Pebble)", patch_conexant_auto),
1266	HDA_CODEC_ENTRY(0x14f15067, "CX20583 (Pebble HSF)", patch_conexant_auto),
1267	HDA_CODEC_ENTRY(0x14f15068, "CX20584", patch_conexant_auto),
1268	HDA_CODEC_ENTRY(0x14f15069, "CX20585", patch_conexant_auto),
1269	HDA_CODEC_ENTRY(0x14f1506c, "CX20588", patch_conexant_auto),
1270	HDA_CODEC_ENTRY(0x14f1506e, "CX20590", patch_conexant_auto),
1271	HDA_CODEC_ENTRY(0x14f15097, "CX20631", patch_conexant_auto),
1272	HDA_CODEC_ENTRY(0x14f15098, "CX20632", patch_conexant_auto),
1273	HDA_CODEC_ENTRY(0x14f150a1, "CX20641", patch_conexant_auto),
1274	HDA_CODEC_ENTRY(0x14f150a2, "CX20642", patch_conexant_auto),
1275	HDA_CODEC_ENTRY(0x14f150ab, "CX20651", patch_conexant_auto),
1276	HDA_CODEC_ENTRY(0x14f150ac, "CX20652", patch_conexant_auto),
1277	HDA_CODEC_ENTRY(0x14f150b8, "CX20664", patch_conexant_auto),
1278	HDA_CODEC_ENTRY(0x14f150b9, "CX20665", patch_conexant_auto),
1279	HDA_CODEC_ENTRY(0x14f150f1, "CX21722", patch_conexant_auto),
1280	HDA_CODEC_ENTRY(0x14f150f2, "CX20722", patch_conexant_auto),
1281	HDA_CODEC_ENTRY(0x14f150f3, "CX21724", patch_conexant_auto),
1282	HDA_CODEC_ENTRY(0x14f150f4, "CX20724", patch_conexant_auto),
1283	HDA_CODEC_ENTRY(0x14f1510f, "CX20751/2", patch_conexant_auto),
1284	HDA_CODEC_ENTRY(0x14f15110, "CX20751/2", patch_conexant_auto),
1285	HDA_CODEC_ENTRY(0x14f15111, "CX20753/4", patch_conexant_auto),
1286	HDA_CODEC_ENTRY(0x14f15113, "CX20755", patch_conexant_auto),
1287	HDA_CODEC_ENTRY(0x14f15114, "CX20756", patch_conexant_auto),
1288	HDA_CODEC_ENTRY(0x14f15115, "CX20757", patch_conexant_auto),
1289	HDA_CODEC_ENTRY(0x14f151d7, "CX20952", patch_conexant_auto),
 
1290	{} /* terminator */
1291};
1292MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_conexant);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1293
1294MODULE_LICENSE("GPL");
1295MODULE_DESCRIPTION("Conexant HD-audio codec");
1296
1297static struct hda_codec_driver conexant_driver = {
1298	.id = snd_hda_id_conexant,
 
1299};
1300
1301module_hda_codec_driver(conexant_driver);