Linux Audio

Check our new training course

Loading...
v3.1
 
   1/*
   2 * HD audio interface patch for Cirrus Logic CS420x chip
   3 *
   4 * Copyright (c) 2009 Takashi Iwai <tiwai@suse.de>
   5 *
   6 *  This driver is free software; you can redistribute it and/or modify
   7 *  it under the terms of the GNU General Public License as published by
   8 *  the Free Software Foundation; either version 2 of the License, or
   9 *  (at your option) any later version.
  10 *
  11 *  This driver is distributed in the hope that it will be useful,
  12 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
  13 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14 *  GNU General Public License for more details.
  15 *
  16 *  You should have received a copy of the GNU General Public License
  17 *  along with this program; if not, write to the Free Software
  18 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
  19 */
  20
  21#include <linux/init.h>
  22#include <linux/delay.h>
  23#include <linux/slab.h>
  24#include <linux/pci.h>
  25#include <sound/core.h>
  26#include "hda_codec.h"
  27#include "hda_local.h"
  28#include <sound/tlv.h>
 
 
 
 
 
  29
  30/*
  31 */
  32
  33struct cs_spec {
  34	int board_config;
  35	struct auto_pin_cfg autocfg;
  36	struct hda_multi_out multiout;
  37	struct snd_kcontrol *vmaster_sw;
  38	struct snd_kcontrol *vmaster_vol;
  39
  40	hda_nid_t dac_nid[AUTO_CFG_MAX_OUTS];
  41	hda_nid_t slave_dig_outs[2];
  42
  43	unsigned int input_idx[AUTO_PIN_LAST];
  44	unsigned int capsrc_idx[AUTO_PIN_LAST];
  45	hda_nid_t adc_nid[AUTO_PIN_LAST];
  46	unsigned int adc_idx[AUTO_PIN_LAST];
  47	unsigned int num_inputs;
  48	unsigned int cur_input;
  49	unsigned int automic_idx;
  50	hda_nid_t cur_adc;
  51	unsigned int cur_adc_stream_tag;
  52	unsigned int cur_adc_format;
  53	hda_nid_t dig_in;
  54
  55	const struct hda_bind_ctls *capture_bind[2];
  56
  57	unsigned int gpio_mask;
  58	unsigned int gpio_dir;
  59	unsigned int gpio_data;
 
 
  60
  61	struct hda_pcm pcm_rec[2];	/* PCM information */
  62
  63	unsigned int hp_detect:1;
  64	unsigned int mic_detect:1;
  65	/* CS421x */
  66	unsigned int spdif_detect:1;
 
  67	unsigned int sense_b:1;
  68	hda_nid_t vendor_nid;
  69	struct hda_input_mux input_mux;
  70	unsigned int last_input;
 
 
  71};
  72
  73/* available models with CS420x */
  74enum {
  75	CS420X_MBP53,
  76	CS420X_MBP55,
  77	CS420X_IMAC27,
 
 
 
 
 
  78	CS420X_AUTO,
  79	CS420X_MODELS
 
 
  80};
  81
  82/* CS421x boards */
  83enum {
  84	CS421X_CDB4210,
  85	CS421X_MODELS
 
  86};
  87
  88/* Vendor-specific processing widget */
  89#define CS420X_VENDOR_NID	0x11
  90#define CS_DIG_OUT1_PIN_NID	0x10
  91#define CS_DIG_OUT2_PIN_NID	0x15
  92#define CS_DMIC1_PIN_NID	0x12
  93#define CS_DMIC2_PIN_NID	0x0e
  94
  95/* coef indices */
  96#define IDX_SPDIF_STAT		0x0000
  97#define IDX_SPDIF_CTL		0x0001
  98#define IDX_ADC_CFG		0x0002
  99/* SZC bitmask, 4 modes below:
 100 * 0 = immediate,
 101 * 1 = digital immediate, analog zero-cross
 102 * 2 = digtail & analog soft-ramp
 103 * 3 = digital soft-ramp, analog zero-cross
 104 */
 105#define   CS_COEF_ADC_SZC_MASK		(3 << 0)
 106#define   CS_COEF_ADC_MIC_SZC_MODE	(3 << 0) /* SZC setup for mic */
 107#define   CS_COEF_ADC_LI_SZC_MODE	(3 << 0) /* SZC setup for line-in */
 108/* PGA mode: 0 = differential, 1 = signle-ended */
 109#define   CS_COEF_ADC_MIC_PGA_MODE	(1 << 5) /* PGA setup for mic */
 110#define   CS_COEF_ADC_LI_PGA_MODE	(1 << 6) /* PGA setup for line-in */
 111#define IDX_DAC_CFG		0x0003
 112/* SZC bitmask, 4 modes below:
 113 * 0 = Immediate
 114 * 1 = zero-cross
 115 * 2 = soft-ramp
 116 * 3 = soft-ramp on zero-cross
 117 */
 118#define   CS_COEF_DAC_HP_SZC_MODE	(3 << 0) /* nid 0x02 */
 119#define   CS_COEF_DAC_LO_SZC_MODE	(3 << 2) /* nid 0x03 */
 120#define   CS_COEF_DAC_SPK_SZC_MODE	(3 << 4) /* nid 0x04 */
 121
 122#define IDX_BEEP_CFG		0x0004
 123/* 0x0008 - test reg key */
 124/* 0x0009 - 0x0014 -> 12 test regs */
 125/* 0x0015 - visibility reg */
 126
 
 
 
 127/*
 128 * Cirrus Logic CS4210
 129 *
 130 * 1 DAC => HP(sense) / Speakers,
 131 * 1 ADC <= LineIn(sense) / MicIn / DMicIn,
 132 * 1 SPDIF OUT => SPDIF Trasmitter(sense)
 133*/
 134#define CS4210_DAC_NID		0x02
 135#define CS4210_ADC_NID		0x03
 136#define CS421X_VENDOR_NID	0x0B
 137#define CS421X_DMIC_PIN_NID	0x09 /* Port E */
 138#define CS421X_SPDIF_PIN_NID	0x0A /* Port H */
 139
 140#define CS421X_IDX_DEV_CFG	0x01
 141#define CS421X_IDX_ADC_CFG	0x02
 142#define CS421X_IDX_DAC_CFG	0x03
 143#define CS421X_IDX_SPK_CTL	0x04
 144
 145#define SPDIF_EVENT		0x04
 
 
 146
 147static inline int cs_vendor_coef_get(struct hda_codec *codec, unsigned int idx)
 148{
 149	struct cs_spec *spec = codec->spec;
 
 150	snd_hda_codec_write(codec, spec->vendor_nid, 0,
 151			    AC_VERB_SET_COEF_INDEX, idx);
 152	return snd_hda_codec_read(codec, spec->vendor_nid, 0,
 153				  AC_VERB_GET_PROC_COEF, 0);
 154}
 155
 156static inline void cs_vendor_coef_set(struct hda_codec *codec, unsigned int idx,
 157				      unsigned int coef)
 158{
 159	struct cs_spec *spec = codec->spec;
 
 160	snd_hda_codec_write(codec, spec->vendor_nid, 0,
 161			    AC_VERB_SET_COEF_INDEX, idx);
 162	snd_hda_codec_write(codec, spec->vendor_nid, 0,
 163			    AC_VERB_SET_PROC_COEF, coef);
 164}
 165
 166
 167#define HP_EVENT	1
 168#define MIC_EVENT	2
 169
 170/*
 171 * PCM callbacks
 172 */
 173static int cs_playback_pcm_open(struct hda_pcm_stream *hinfo,
 174				struct hda_codec *codec,
 175				struct snd_pcm_substream *substream)
 176{
 177	struct cs_spec *spec = codec->spec;
 178	return snd_hda_multi_out_analog_open(codec, &spec->multiout, substream,
 179					     hinfo);
 180}
 181
 182static int cs_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
 183				   struct hda_codec *codec,
 184				   unsigned int stream_tag,
 185				   unsigned int format,
 186				   struct snd_pcm_substream *substream)
 187{
 188	struct cs_spec *spec = codec->spec;
 189	return snd_hda_multi_out_analog_prepare(codec, &spec->multiout,
 190						stream_tag, format, substream);
 191}
 192
 193static int cs_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
 194				   struct hda_codec *codec,
 195				   struct snd_pcm_substream *substream)
 196{
 197	struct cs_spec *spec = codec->spec;
 198	return snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
 199}
 200
 201/*
 202 * Digital out
 203 */
 204static int cs_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
 205				    struct hda_codec *codec,
 206				    struct snd_pcm_substream *substream)
 207{
 208	struct cs_spec *spec = codec->spec;
 209	return snd_hda_multi_out_dig_open(codec, &spec->multiout);
 210}
 211
 212static int cs_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
 213				     struct hda_codec *codec,
 214				     struct snd_pcm_substream *substream)
 215{
 216	struct cs_spec *spec = codec->spec;
 217	return snd_hda_multi_out_dig_close(codec, &spec->multiout);
 218}
 219
 220static int cs_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
 221				       struct hda_codec *codec,
 222				       unsigned int stream_tag,
 223				       unsigned int format,
 224				       struct snd_pcm_substream *substream)
 225{
 226	struct cs_spec *spec = codec->spec;
 227	return snd_hda_multi_out_dig_prepare(codec, &spec->multiout, stream_tag,
 228					     format, substream);
 229}
 230
 231static int cs_dig_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
 232				       struct hda_codec *codec,
 233				       struct snd_pcm_substream *substream)
 234{
 235	struct cs_spec *spec = codec->spec;
 236	return snd_hda_multi_out_dig_cleanup(codec, &spec->multiout);
 237}
 238
 239/*
 240 * Analog capture
 241 */
 242static int cs_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
 243				  struct hda_codec *codec,
 244				  unsigned int stream_tag,
 245				  unsigned int format,
 246				  struct snd_pcm_substream *substream)
 247{
 248	struct cs_spec *spec = codec->spec;
 249	spec->cur_adc = spec->adc_nid[spec->cur_input];
 250	spec->cur_adc_stream_tag = stream_tag;
 251	spec->cur_adc_format = format;
 252	snd_hda_codec_setup_stream(codec, spec->cur_adc, stream_tag, 0, format);
 253	return 0;
 254}
 255
 256static int cs_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
 257				  struct hda_codec *codec,
 258				  struct snd_pcm_substream *substream)
 259{
 260	struct cs_spec *spec = codec->spec;
 261	snd_hda_codec_cleanup_stream(codec, spec->cur_adc);
 262	spec->cur_adc = 0;
 263	return 0;
 264}
 265
 266/*
 267 */
 268static const struct hda_pcm_stream cs_pcm_analog_playback = {
 269	.substreams = 1,
 270	.channels_min = 2,
 271	.channels_max = 2,
 272	.ops = {
 273		.open = cs_playback_pcm_open,
 274		.prepare = cs_playback_pcm_prepare,
 275		.cleanup = cs_playback_pcm_cleanup
 276	},
 277};
 278
 279static const struct hda_pcm_stream cs_pcm_analog_capture = {
 280	.substreams = 1,
 281	.channels_min = 2,
 282	.channels_max = 2,
 283	.ops = {
 284		.prepare = cs_capture_pcm_prepare,
 285		.cleanup = cs_capture_pcm_cleanup
 286	},
 287};
 288
 289static const struct hda_pcm_stream cs_pcm_digital_playback = {
 290	.substreams = 1,
 291	.channels_min = 2,
 292	.channels_max = 2,
 293	.ops = {
 294		.open = cs_dig_playback_pcm_open,
 295		.close = cs_dig_playback_pcm_close,
 296		.prepare = cs_dig_playback_pcm_prepare,
 297		.cleanup = cs_dig_playback_pcm_cleanup
 298	},
 299};
 300
 301static const struct hda_pcm_stream cs_pcm_digital_capture = {
 302	.substreams = 1,
 303	.channels_min = 2,
 304	.channels_max = 2,
 305};
 306
 307static int cs_build_pcms(struct hda_codec *codec)
 308{
 309	struct cs_spec *spec = codec->spec;
 310	struct hda_pcm *info = spec->pcm_rec;
 311
 312	codec->pcm_info = info;
 313	codec->num_pcms = 0;
 314
 315	info->name = "Cirrus Analog";
 316	info->stream[SNDRV_PCM_STREAM_PLAYBACK] = cs_pcm_analog_playback;
 317	info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = spec->dac_nid[0];
 318	info->stream[SNDRV_PCM_STREAM_PLAYBACK].channels_max =
 319		spec->multiout.max_channels;
 320	info->stream[SNDRV_PCM_STREAM_CAPTURE] = cs_pcm_analog_capture;
 321	info->stream[SNDRV_PCM_STREAM_CAPTURE].nid =
 322		spec->adc_nid[spec->cur_input];
 323	codec->num_pcms++;
 324
 325	if (!spec->multiout.dig_out_nid && !spec->dig_in)
 326		return 0;
 327
 328	info++;
 329	info->name = "Cirrus Digital";
 330	info->pcm_type = spec->autocfg.dig_out_type[0];
 331	if (!info->pcm_type)
 332		info->pcm_type = HDA_PCM_TYPE_SPDIF;
 333	if (spec->multiout.dig_out_nid) {
 334		info->stream[SNDRV_PCM_STREAM_PLAYBACK] =
 335			cs_pcm_digital_playback;
 336		info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid =
 337			spec->multiout.dig_out_nid;
 338	}
 339	if (spec->dig_in) {
 340		info->stream[SNDRV_PCM_STREAM_CAPTURE] =
 341			cs_pcm_digital_capture;
 342		info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->dig_in;
 343	}
 344	codec->num_pcms++;
 345
 346	return 0;
 347}
 348
 349/*
 350 * parse codec topology
 351 */
 352
 353static hda_nid_t get_dac(struct hda_codec *codec, hda_nid_t pin)
 354{
 355	hda_nid_t dac;
 356	if (!pin)
 357		return 0;
 358	if (snd_hda_get_connections(codec, pin, &dac, 1) != 1)
 359		return 0;
 360	return dac;
 361}
 362
 363static int is_ext_mic(struct hda_codec *codec, unsigned int idx)
 364{
 365	struct cs_spec *spec = codec->spec;
 366	struct auto_pin_cfg *cfg = &spec->autocfg;
 367	hda_nid_t pin = cfg->inputs[idx].pin;
 368	unsigned int val;
 369	if (!is_jack_detectable(codec, pin))
 370		return 0;
 371	val = snd_hda_codec_get_pincfg(codec, pin);
 372	return (snd_hda_get_input_pin_attr(val) != INPUT_PIN_ATTR_INT);
 373}
 374
 375static hda_nid_t get_adc(struct hda_codec *codec, hda_nid_t pin,
 376			 unsigned int *idxp)
 377{
 378	int i, idx;
 379	hda_nid_t nid;
 380
 381	nid = codec->start_nid;
 382	for (i = 0; i < codec->num_nodes; i++, nid++) {
 383		unsigned int type;
 384		type = get_wcaps_type(get_wcaps(codec, nid));
 385		if (type != AC_WID_AUD_IN)
 386			continue;
 387		idx = snd_hda_get_conn_index(codec, nid, pin, false);
 388		if (idx >= 0) {
 389			*idxp = idx;
 390			return nid;
 391		}
 392	}
 393	return 0;
 394}
 395
 396static int is_active_pin(struct hda_codec *codec, hda_nid_t nid)
 397{
 398	unsigned int val;
 399	val = snd_hda_codec_get_pincfg(codec, nid);
 400	return (get_defcfg_connect(val) != AC_JACK_PORT_NONE);
 401}
 402
 403static int parse_output(struct hda_codec *codec)
 404{
 405	struct cs_spec *spec = codec->spec;
 406	struct auto_pin_cfg *cfg = &spec->autocfg;
 407	int i, extra_nids;
 408	hda_nid_t dac;
 409
 410	for (i = 0; i < cfg->line_outs; i++) {
 411		dac = get_dac(codec, cfg->line_out_pins[i]);
 412		if (!dac)
 413			break;
 414		spec->dac_nid[i] = dac;
 415	}
 416	spec->multiout.num_dacs = i;
 417	spec->multiout.dac_nids = spec->dac_nid;
 418	spec->multiout.max_channels = i * 2;
 419
 420	/* add HP and speakers */
 421	extra_nids = 0;
 422	for (i = 0; i < cfg->hp_outs; i++) {
 423		dac = get_dac(codec, cfg->hp_pins[i]);
 424		if (!dac)
 425			break;
 426		if (!i)
 427			spec->multiout.hp_nid = dac;
 428		else
 429			spec->multiout.extra_out_nid[extra_nids++] = dac;
 430	}
 431	for (i = 0; i < cfg->speaker_outs; i++) {
 432		dac = get_dac(codec, cfg->speaker_pins[i]);
 433		if (!dac)
 434			break;
 435		spec->multiout.extra_out_nid[extra_nids++] = dac;
 436	}
 437
 438	if (cfg->line_out_type == AUTO_PIN_SPEAKER_OUT) {
 439		cfg->speaker_outs = cfg->line_outs;
 440		memcpy(cfg->speaker_pins, cfg->line_out_pins,
 441		       sizeof(cfg->speaker_pins));
 442		cfg->line_outs = 0;
 443	}
 444
 445	return 0;
 446}
 447
 448static int parse_input(struct hda_codec *codec)
 449{
 450	struct cs_spec *spec = codec->spec;
 451	struct auto_pin_cfg *cfg = &spec->autocfg;
 452	int i;
 453
 454	for (i = 0; i < cfg->num_inputs; i++) {
 455		hda_nid_t pin = cfg->inputs[i].pin;
 456		spec->input_idx[spec->num_inputs] = i;
 457		spec->capsrc_idx[i] = spec->num_inputs++;
 458		spec->cur_input = i;
 459		spec->adc_nid[i] = get_adc(codec, pin, &spec->adc_idx[i]);
 460	}
 461	if (!spec->num_inputs)
 462		return 0;
 463
 464	/* check whether the automatic mic switch is available */
 465	if (spec->num_inputs == 2 &&
 466	    cfg->inputs[0].type == AUTO_PIN_MIC &&
 467	    cfg->inputs[1].type == AUTO_PIN_MIC) {
 468		if (is_ext_mic(codec, cfg->inputs[0].pin)) {
 469			if (!is_ext_mic(codec, cfg->inputs[1].pin)) {
 470				spec->mic_detect = 1;
 471				spec->automic_idx = 0;
 472			}
 473		} else {
 474			if (is_ext_mic(codec, cfg->inputs[1].pin)) {
 475				spec->mic_detect = 1;
 476				spec->automic_idx = 1;
 477			}
 478		}
 479	}
 480	return 0;
 481}
 482
 483
 484static int parse_digital_output(struct hda_codec *codec)
 485{
 486	struct cs_spec *spec = codec->spec;
 487	struct auto_pin_cfg *cfg = &spec->autocfg;
 488	hda_nid_t nid;
 489
 490	if (!cfg->dig_outs)
 491		return 0;
 492	if (snd_hda_get_connections(codec, cfg->dig_out_pins[0], &nid, 1) < 1)
 493		return 0;
 494	spec->multiout.dig_out_nid = nid;
 495	spec->multiout.share_spdif = 1;
 496	if (cfg->dig_outs > 1 &&
 497	    snd_hda_get_connections(codec, cfg->dig_out_pins[1], &nid, 1) > 0) {
 498		spec->slave_dig_outs[0] = nid;
 499		codec->slave_dig_outs = spec->slave_dig_outs;
 500	}
 501	return 0;
 502}
 503
 504static int parse_digital_input(struct hda_codec *codec)
 505{
 506	struct cs_spec *spec = codec->spec;
 507	struct auto_pin_cfg *cfg = &spec->autocfg;
 508	int idx;
 509
 510	if (cfg->dig_in_pin)
 511		spec->dig_in = get_adc(codec, cfg->dig_in_pin, &idx);
 512	return 0;
 513}
 514
 515/*
 516 * create mixer controls
 517 */
 518
 519static const char * const dir_sfx[2] = { "Playback", "Capture" };
 520
 521static int add_mute(struct hda_codec *codec, const char *name, int index,
 522		    unsigned int pval, int dir, struct snd_kcontrol **kctlp)
 523{
 524	char tmp[44];
 525	struct snd_kcontrol_new knew =
 526		HDA_CODEC_MUTE_IDX(tmp, index, 0, 0, HDA_OUTPUT);
 527	knew.private_value = pval;
 528	snprintf(tmp, sizeof(tmp), "%s %s Switch", name, dir_sfx[dir]);
 529	*kctlp = snd_ctl_new1(&knew, codec);
 530	(*kctlp)->id.subdevice = HDA_SUBDEV_AMP_FLAG;
 531	return snd_hda_ctl_add(codec, 0, *kctlp);
 532}
 533
 534static int add_volume(struct hda_codec *codec, const char *name,
 535		      int index, unsigned int pval, int dir,
 536		      struct snd_kcontrol **kctlp)
 537{
 538	char tmp[44];
 539	struct snd_kcontrol_new knew =
 540		HDA_CODEC_VOLUME_IDX(tmp, index, 0, 0, HDA_OUTPUT);
 541	knew.private_value = pval;
 542	snprintf(tmp, sizeof(tmp), "%s %s Volume", name, dir_sfx[dir]);
 543	*kctlp = snd_ctl_new1(&knew, codec);
 544	(*kctlp)->id.subdevice = HDA_SUBDEV_AMP_FLAG;
 545	return snd_hda_ctl_add(codec, 0, *kctlp);
 546}
 547
 548static void fix_volume_caps(struct hda_codec *codec, hda_nid_t dac)
 549{
 550	unsigned int caps;
 551
 552	/* set the upper-limit for mixer amp to 0dB */
 553	caps = query_amp_caps(codec, dac, HDA_OUTPUT);
 554	caps &= ~(0x7f << AC_AMPCAP_NUM_STEPS_SHIFT);
 555	caps |= ((caps >> AC_AMPCAP_OFFSET_SHIFT) & 0x7f)
 556		<< AC_AMPCAP_NUM_STEPS_SHIFT;
 557	snd_hda_override_amp_caps(codec, dac, HDA_OUTPUT, caps);
 558}
 559
 560static int add_vmaster(struct hda_codec *codec, hda_nid_t dac)
 561{
 562	struct cs_spec *spec = codec->spec;
 563	unsigned int tlv[4];
 564	int err;
 565
 566	spec->vmaster_sw =
 567		snd_ctl_make_virtual_master("Master Playback Switch", NULL);
 568	err = snd_hda_ctl_add(codec, dac, spec->vmaster_sw);
 569	if (err < 0)
 570		return err;
 571
 572	snd_hda_set_vmaster_tlv(codec, dac, HDA_OUTPUT, tlv);
 573	spec->vmaster_vol =
 574		snd_ctl_make_virtual_master("Master Playback Volume", tlv);
 575	err = snd_hda_ctl_add(codec, dac, spec->vmaster_vol);
 576	if (err < 0)
 577		return err;
 578	return 0;
 579}
 580
 581static int add_output(struct hda_codec *codec, hda_nid_t dac, int idx,
 582		      int num_ctls, int type)
 583{
 584	struct cs_spec *spec = codec->spec;
 585	const char *name;
 586	int err, index;
 587	struct snd_kcontrol *kctl;
 588	static const char * const speakers[] = {
 589		"Front Speaker", "Surround Speaker", "Bass Speaker"
 590	};
 591	static const char * const line_outs[] = {
 592		"Front Line-Out", "Surround Line-Out", "Bass Line-Out"
 593	};
 594
 595	fix_volume_caps(codec, dac);
 596	if (!spec->vmaster_sw) {
 597		err = add_vmaster(codec, dac);
 598		if (err < 0)
 599			return err;
 600	}
 601
 602	index = 0;
 603	switch (type) {
 604	case AUTO_PIN_HP_OUT:
 605		name = "Headphone";
 606		index = idx;
 607		break;
 608	case AUTO_PIN_SPEAKER_OUT:
 609		if (num_ctls > 1)
 610			name = speakers[idx];
 611		else
 612			name = "Speaker";
 613		break;
 614	default:
 615		if (num_ctls > 1)
 616			name = line_outs[idx];
 617		else
 618			name = "Line-Out";
 619		break;
 620	}
 621
 622	err = add_mute(codec, name, index,
 623		       HDA_COMPOSE_AMP_VAL(dac, 3, 0, HDA_OUTPUT), 0, &kctl);
 624	if (err < 0)
 625		return err;
 626	err = snd_ctl_add_slave(spec->vmaster_sw, kctl);
 627	if (err < 0)
 628		return err;
 629
 630	err = add_volume(codec, name, index,
 631			 HDA_COMPOSE_AMP_VAL(dac, 3, 0, HDA_OUTPUT), 0, &kctl);
 632	if (err < 0)
 633		return err;
 634	err = snd_ctl_add_slave(spec->vmaster_vol, kctl);
 635	if (err < 0)
 636		return err;
 637
 638	return 0;
 639}		
 640
 641static int build_output(struct hda_codec *codec)
 642{
 643	struct cs_spec *spec = codec->spec;
 644	struct auto_pin_cfg *cfg = &spec->autocfg;
 645	int i, err;
 646
 647	for (i = 0; i < cfg->line_outs; i++) {
 648		err = add_output(codec, get_dac(codec, cfg->line_out_pins[i]),
 649				 i, cfg->line_outs, cfg->line_out_type);
 650		if (err < 0)
 651			return err;
 652	}
 653	for (i = 0; i < cfg->hp_outs; i++) {
 654		err = add_output(codec, get_dac(codec, cfg->hp_pins[i]),
 655				 i, cfg->hp_outs, AUTO_PIN_HP_OUT);
 656		if (err < 0)
 657			return err;
 658	}
 659	for (i = 0; i < cfg->speaker_outs; i++) {
 660		err = add_output(codec, get_dac(codec, cfg->speaker_pins[i]),
 661				 i, cfg->speaker_outs, AUTO_PIN_SPEAKER_OUT);
 662		if (err < 0)
 663			return err;
 664	}
 665	return 0;
 666}
 667
 668/*
 669 */
 670
 671static const struct snd_kcontrol_new cs_capture_ctls[] = {
 672	HDA_BIND_SW("Capture Switch", 0),
 673	HDA_BIND_VOL("Capture Volume", 0),
 674};
 675
 676static int change_cur_input(struct hda_codec *codec, unsigned int idx,
 677			    int force)
 678{
 679	struct cs_spec *spec = codec->spec;
 680	
 681	if (spec->cur_input == idx && !force)
 682		return 0;
 683	if (spec->cur_adc && spec->cur_adc != spec->adc_nid[idx]) {
 684		/* stream is running, let's swap the current ADC */
 685		__snd_hda_codec_cleanup_stream(codec, spec->cur_adc, 1);
 686		spec->cur_adc = spec->adc_nid[idx];
 687		snd_hda_codec_setup_stream(codec, spec->cur_adc,
 688					   spec->cur_adc_stream_tag, 0,
 689					   spec->cur_adc_format);
 690	}
 691	snd_hda_codec_write(codec, spec->cur_adc, 0,
 692			    AC_VERB_SET_CONNECT_SEL,
 693			    spec->adc_idx[idx]);
 694	spec->cur_input = idx;
 695	return 1;
 696}
 697
 698static int cs_capture_source_info(struct snd_kcontrol *kcontrol,
 699				  struct snd_ctl_elem_info *uinfo)
 700{
 701	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 702	struct cs_spec *spec = codec->spec;
 703	struct auto_pin_cfg *cfg = &spec->autocfg;
 704	unsigned int idx;
 705
 706	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
 707	uinfo->count = 1;
 708	uinfo->value.enumerated.items = spec->num_inputs;
 709	if (uinfo->value.enumerated.item >= spec->num_inputs)
 710		uinfo->value.enumerated.item = spec->num_inputs - 1;
 711	idx = spec->input_idx[uinfo->value.enumerated.item];
 712	strcpy(uinfo->value.enumerated.name,
 713	       hda_get_input_pin_label(codec, cfg->inputs[idx].pin, 1));
 714	return 0;
 715}
 716
 717static int cs_capture_source_get(struct snd_kcontrol *kcontrol,
 718				 struct snd_ctl_elem_value *ucontrol)
 719{
 720	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 721	struct cs_spec *spec = codec->spec;
 722	ucontrol->value.enumerated.item[0] = spec->capsrc_idx[spec->cur_input];
 723	return 0;
 724}
 725
 726static int cs_capture_source_put(struct snd_kcontrol *kcontrol,
 727				 struct snd_ctl_elem_value *ucontrol)
 728{
 729	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 730	struct cs_spec *spec = codec->spec;
 731	unsigned int idx = ucontrol->value.enumerated.item[0];
 732
 733	if (idx >= spec->num_inputs)
 734		return -EINVAL;
 735	idx = spec->input_idx[idx];
 736	return change_cur_input(codec, idx, 0);
 737}
 738
 739static const struct snd_kcontrol_new cs_capture_source = {
 740	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 741	.name = "Capture Source",
 742	.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
 743	.info = cs_capture_source_info,
 744	.get = cs_capture_source_get,
 745	.put = cs_capture_source_put,
 746};
 747
 748static const struct hda_bind_ctls *make_bind_capture(struct hda_codec *codec,
 749					       struct hda_ctl_ops *ops)
 750{
 751	struct cs_spec *spec = codec->spec;
 752	struct hda_bind_ctls *bind;
 753	int i, n;
 754
 755	bind = kzalloc(sizeof(*bind) + sizeof(long) * (spec->num_inputs + 1),
 756		       GFP_KERNEL);
 757	if (!bind)
 758		return NULL;
 759	bind->ops = ops;
 760	n = 0;
 761	for (i = 0; i < AUTO_PIN_LAST; i++) {
 762		if (!spec->adc_nid[i])
 763			continue;
 764		bind->values[n++] =
 765			HDA_COMPOSE_AMP_VAL(spec->adc_nid[i], 3,
 766					    spec->adc_idx[i], HDA_INPUT);
 767	}
 768	return bind;
 769}
 770
 771/* add a (input-boost) volume control to the given input pin */
 772static int add_input_volume_control(struct hda_codec *codec,
 773				    struct auto_pin_cfg *cfg,
 774				    int item)
 775{
 776	hda_nid_t pin = cfg->inputs[item].pin;
 777	u32 caps;
 778	const char *label;
 779	struct snd_kcontrol *kctl;
 780		
 781	if (!(get_wcaps(codec, pin) & AC_WCAP_IN_AMP))
 782		return 0;
 783	caps = query_amp_caps(codec, pin, HDA_INPUT);
 784	caps = (caps & AC_AMPCAP_NUM_STEPS) >> AC_AMPCAP_NUM_STEPS_SHIFT;
 785	if (caps <= 1)
 786		return 0;
 787	label = hda_get_autocfg_input_label(codec, cfg, item);
 788	return add_volume(codec, label, 0,
 789			  HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_INPUT), 1, &kctl);
 790}
 791
 792static int build_input(struct hda_codec *codec)
 793{
 794	struct cs_spec *spec = codec->spec;
 795	int i, err;
 796
 797	if (!spec->num_inputs)
 798		return 0;
 799
 800	/* make bind-capture */
 801	spec->capture_bind[0] = make_bind_capture(codec, &snd_hda_bind_sw);
 802	spec->capture_bind[1] = make_bind_capture(codec, &snd_hda_bind_vol);
 803	for (i = 0; i < 2; i++) {
 804		struct snd_kcontrol *kctl;
 805		int n;
 806		if (!spec->capture_bind[i])
 807			return -ENOMEM;
 808		kctl = snd_ctl_new1(&cs_capture_ctls[i], codec);
 809		if (!kctl)
 810			return -ENOMEM;
 811		kctl->private_value = (long)spec->capture_bind[i];
 812		err = snd_hda_ctl_add(codec, 0, kctl);
 813		if (err < 0)
 814			return err;
 815		for (n = 0; n < AUTO_PIN_LAST; n++) {
 816			if (!spec->adc_nid[n])
 817				continue;
 818			err = snd_hda_add_nid(codec, kctl, 0, spec->adc_nid[n]);
 819			if (err < 0)
 820				return err;
 821		}
 822	}
 823	
 824	if (spec->num_inputs > 1 && !spec->mic_detect) {
 825		err = snd_hda_ctl_add(codec, 0,
 826				      snd_ctl_new1(&cs_capture_source, codec));
 827		if (err < 0)
 828			return err;
 829	}
 830
 831	for (i = 0; i < spec->num_inputs; i++) {
 832		err = add_input_volume_control(codec, &spec->autocfg, i);
 833		if (err < 0)
 834			return err;
 835	}
 836
 837	return 0;
 838}
 839
 840/*
 841 */
 842
 843static int build_digital_output(struct hda_codec *codec)
 844{
 845	struct cs_spec *spec = codec->spec;
 846	int err;
 847
 848	if (!spec->multiout.dig_out_nid)
 849		return 0;
 850
 851	err = snd_hda_create_spdif_out_ctls(codec, spec->multiout.dig_out_nid,
 852					    spec->multiout.dig_out_nid);
 853	if (err < 0)
 854		return err;
 855	err = snd_hda_create_spdif_share_sw(codec, &spec->multiout);
 856	if (err < 0)
 857		return err;
 858	return 0;
 859}
 860
 861static int build_digital_input(struct hda_codec *codec)
 862{
 863	struct cs_spec *spec = codec->spec;
 864	if (spec->dig_in)
 865		return snd_hda_create_spdif_in_ctls(codec, spec->dig_in);
 866	return 0;
 867}
 868
 869/*
 870 * auto-mute and auto-mic switching
 871 * CS421x auto-output redirecting
 872 * HP/SPK/SPDIF
 873 */
 874
 875static void cs_automute(struct hda_codec *codec)
 876{
 877	struct cs_spec *spec = codec->spec;
 878	struct auto_pin_cfg *cfg = &spec->autocfg;
 879	unsigned int hp_present;
 880	unsigned int spdif_present;
 881	hda_nid_t nid;
 882	int i;
 883
 884	spdif_present = 0;
 885	if (cfg->dig_outs) {
 886		nid = cfg->dig_out_pins[0];
 887		if (is_jack_detectable(codec, nid)) {
 888			/*
 889			TODO: SPDIF output redirect when SENSE_B is enabled.
 890			Shared (SENSE_A) jack (e.g HP/mini-TOSLINK)
 891			assumed.
 892			*/
 893			if (snd_hda_jack_detect(codec, nid)
 894				/* && spec->sense_b */)
 895				spdif_present = 1;
 896		}
 897	}
 898
 899	hp_present = 0;
 900	for (i = 0; i < cfg->hp_outs; i++) {
 901		nid = cfg->hp_pins[i];
 902		if (!is_jack_detectable(codec, nid))
 903			continue;
 904		hp_present = snd_hda_jack_detect(codec, nid);
 905		if (hp_present)
 906			break;
 907	}
 908
 909	/* mute speakers if spdif or hp jack is plugged in */
 910	for (i = 0; i < cfg->speaker_outs; i++) {
 911		nid = cfg->speaker_pins[i];
 912		snd_hda_codec_write(codec, nid, 0,
 913				    AC_VERB_SET_PIN_WIDGET_CONTROL,
 914				    hp_present ? 0 : PIN_OUT);
 915		/* detect on spdif is specific to CS421x */
 916		if (spec->vendor_nid == CS421X_VENDOR_NID) {
 917			snd_hda_codec_write(codec, nid, 0,
 918					AC_VERB_SET_PIN_WIDGET_CONTROL,
 919					spdif_present ? 0 : PIN_OUT);
 920		}
 921	}
 922	if (spec->board_config == CS420X_MBP53 ||
 923	    spec->board_config == CS420X_MBP55 ||
 924	    spec->board_config == CS420X_IMAC27) {
 925		unsigned int gpio = hp_present ? 0x02 : 0x08;
 926		snd_hda_codec_write(codec, 0x01, 0,
 927				    AC_VERB_SET_GPIO_DATA, gpio);
 928	}
 929
 930	/* specific to CS421x */
 931	if (spec->vendor_nid == CS421X_VENDOR_NID) {
 932		/* mute HPs if spdif jack (SENSE_B) is present */
 933		for (i = 0; i < cfg->hp_outs; i++) {
 934			nid = cfg->hp_pins[i];
 935			snd_hda_codec_write(codec, nid, 0,
 936				AC_VERB_SET_PIN_WIDGET_CONTROL,
 937				(spdif_present && spec->sense_b) ? 0 : PIN_HP);
 938		}
 939
 940		/* SPDIF TX on/off */
 941		if (cfg->dig_outs) {
 942			nid = cfg->dig_out_pins[0];
 943			snd_hda_codec_write(codec, nid, 0,
 944				AC_VERB_SET_PIN_WIDGET_CONTROL,
 945				spdif_present ? PIN_OUT : 0);
 946
 947		}
 948		/* Update board GPIOs if neccessary ... */
 949	}
 950}
 951
 952/*
 953 * Auto-input redirect for CS421x
 954 * Switch max 3 inputs of a single ADC (nid 3)
 955*/
 956
 957static void cs_automic(struct hda_codec *codec)
 958{
 959	struct cs_spec *spec = codec->spec;
 960	struct auto_pin_cfg *cfg = &spec->autocfg;
 961	hda_nid_t nid;
 962	unsigned int present;
 963
 964	nid = cfg->inputs[spec->automic_idx].pin;
 965	present = snd_hda_jack_detect(codec, nid);
 966
 967	/* specific to CS421x, single ADC */
 968	if (spec->vendor_nid == CS421X_VENDOR_NID) {
 969		if (present) {
 970			spec->last_input = spec->cur_input;
 971			spec->cur_input = spec->automic_idx;
 972		} else  {
 973			spec->cur_input = spec->last_input;
 974		}
 975
 976		snd_hda_codec_write_cache(codec, spec->cur_adc, 0,
 977					AC_VERB_SET_CONNECT_SEL,
 978					spec->adc_idx[spec->cur_input]);
 979	} else {
 980		if (present)
 981			change_cur_input(codec, spec->automic_idx, 0);
 982		else
 983			change_cur_input(codec, !spec->automic_idx, 0);
 
 
 
 984	}
 985}
 986
 987/*
 988 */
 989
 990static void init_output(struct hda_codec *codec)
 991{
 992	struct cs_spec *spec = codec->spec;
 993	struct auto_pin_cfg *cfg = &spec->autocfg;
 994	int i;
 995
 996	/* mute first */
 997	for (i = 0; i < spec->multiout.num_dacs; i++)
 998		snd_hda_codec_write(codec, spec->multiout.dac_nids[i], 0,
 999				    AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
1000	if (spec->multiout.hp_nid)
1001		snd_hda_codec_write(codec, spec->multiout.hp_nid, 0,
1002				    AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
1003	for (i = 0; i < ARRAY_SIZE(spec->multiout.extra_out_nid); i++) {
1004		if (!spec->multiout.extra_out_nid[i])
1005			break;
1006		snd_hda_codec_write(codec, spec->multiout.extra_out_nid[i], 0,
1007				    AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
1008	}
1009
1010	/* set appropriate pin controls */
1011	for (i = 0; i < cfg->line_outs; i++)
1012		snd_hda_codec_write(codec, cfg->line_out_pins[i], 0,
1013				    AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
1014	/* HP */
1015	for (i = 0; i < cfg->hp_outs; i++) {
1016		hda_nid_t nid = cfg->hp_pins[i];
1017		snd_hda_codec_write(codec, nid, 0,
1018				    AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP);
1019		if (!cfg->speaker_outs)
1020			continue;
1021		if (get_wcaps(codec, nid) & AC_WCAP_UNSOL_CAP) {
1022			snd_hda_codec_write(codec, nid, 0,
1023					    AC_VERB_SET_UNSOLICITED_ENABLE,
1024					    AC_USRSP_EN | HP_EVENT);
1025			spec->hp_detect = 1;
1026		}
1027	}
1028
1029	/* Speaker */
1030	for (i = 0; i < cfg->speaker_outs; i++)
1031		snd_hda_codec_write(codec, cfg->speaker_pins[i], 0,
1032				    AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
1033
1034	/* SPDIF is enabled on presence detect for CS421x */
1035	if (spec->hp_detect || spec->spdif_detect)
1036		cs_automute(codec);
1037}
1038
1039static void init_input(struct hda_codec *codec)
1040{
1041	struct cs_spec *spec = codec->spec;
1042	struct auto_pin_cfg *cfg = &spec->autocfg;
1043	unsigned int coef;
1044	int i;
1045
1046	for (i = 0; i < cfg->num_inputs; i++) {
1047		unsigned int ctl;
1048		hda_nid_t pin = cfg->inputs[i].pin;
1049		if (!spec->adc_nid[i])
1050			continue;
1051		/* set appropriate pin control and mute first */
1052		ctl = PIN_IN;
1053		if (cfg->inputs[i].type == AUTO_PIN_MIC) {
1054			unsigned int caps = snd_hda_query_pin_caps(codec, pin);
1055			caps >>= AC_PINCAP_VREF_SHIFT;
1056			if (caps & AC_PINCAP_VREF_80)
1057				ctl = PIN_VREF80;
1058		}
1059		snd_hda_codec_write(codec, pin, 0,
1060				    AC_VERB_SET_PIN_WIDGET_CONTROL, ctl);
1061		snd_hda_codec_write(codec, spec->adc_nid[i], 0,
1062				    AC_VERB_SET_AMP_GAIN_MUTE,
1063				    AMP_IN_MUTE(spec->adc_idx[i]));
1064		if (spec->mic_detect && spec->automic_idx == i)
1065			snd_hda_codec_write(codec, pin, 0,
1066					    AC_VERB_SET_UNSOLICITED_ENABLE,
1067					    AC_USRSP_EN | MIC_EVENT);
1068	}
1069	/* specific to CS421x */
1070	if (spec->vendor_nid == CS421X_VENDOR_NID) {
1071		if (spec->mic_detect)
1072			cs_automic(codec);
1073		else  {
1074			spec->cur_adc = spec->adc_nid[spec->cur_input];
1075			snd_hda_codec_write(codec, spec->cur_adc, 0,
1076					AC_VERB_SET_CONNECT_SEL,
1077					spec->adc_idx[spec->cur_input]);
1078		}
1079	} else {
1080		change_cur_input(codec, spec->cur_input, 1);
1081		if (spec->mic_detect)
1082			cs_automic(codec);
1083
1084		coef = 0x000a; /* ADC1/2 - Digital and Analog Soft Ramp */
1085		if (is_active_pin(codec, CS_DMIC2_PIN_NID))
1086			coef |= 0x0500; /* DMIC2 2 chan on, GPIO1 off */
1087		if (is_active_pin(codec, CS_DMIC1_PIN_NID))
1088			coef |= 0x1800; /* DMIC1 2 chan on, GPIO0 off
1089					 * No effect if SPDIF_OUT2 is
1090					 * selected in IDX_SPDIF_CTL.
1091					*/
1092		cs_vendor_coef_set(codec, IDX_ADC_CFG, coef);
 
1093	}
1094}
1095
1096static const struct hda_verb cs_coef_init_verbs[] = {
1097	{0x11, AC_VERB_SET_PROC_STATE, 1},
1098	{0x11, AC_VERB_SET_COEF_INDEX, IDX_DAC_CFG},
1099	{0x11, AC_VERB_SET_PROC_COEF,
1100	 (0x002a /* DAC1/2/3 SZCMode Soft Ramp */
1101	  | 0x0040 /* Mute DACs on FIFO error */
1102	  | 0x1000 /* Enable DACs High Pass Filter */
1103	  | 0x0400 /* Disable Coefficient Auto increment */
1104	  )},
 
 
 
1105	/* Beep */
1106	{0x11, AC_VERB_SET_COEF_INDEX, IDX_DAC_CFG},
1107	{0x11, AC_VERB_SET_PROC_COEF, 0x0007}, /* Enable Beep thru DAC1/2/3 */
1108
1109	{} /* terminator */
1110};
1111
 
 
 
 
 
 
 
 
 
 
1112/* Errata: CS4207 rev C0/C1/C2 Silicon
1113 *
1114 * http://www.cirrus.com/en/pubs/errata/ER880C3.pdf
1115 *
1116 * 6. At high temperature (TA > +85°C), the digital supply current (IVD)
1117 * may be excessive (up to an additional 200 μA), which is most easily
1118 * observed while the part is being held in reset (RESET# active low).
1119 *
1120 * Root Cause: At initial powerup of the device, the logic that drives
1121 * the clock and write enable to the S/PDIF SRC RAMs is not properly
1122 * initialized.
1123 * Certain random patterns will cause a steady leakage current in those
1124 * RAM cells. The issue will resolve once the SRCs are used (turned on).
1125 *
1126 * Workaround: The following verb sequence briefly turns on the S/PDIF SRC
1127 * blocks, which will alleviate the issue.
1128 */
1129
1130static const struct hda_verb cs_errata_init_verbs[] = {
1131	{0x01, AC_VERB_SET_POWER_STATE, 0x00}, /* AFG: D0 */
1132	{0x11, AC_VERB_SET_PROC_STATE, 0x01},  /* VPW: processing on */
1133
1134	{0x11, AC_VERB_SET_COEF_INDEX, 0x0008},
1135	{0x11, AC_VERB_SET_PROC_COEF, 0x9999},
1136	{0x11, AC_VERB_SET_COEF_INDEX, 0x0017},
1137	{0x11, AC_VERB_SET_PROC_COEF, 0xa412},
1138	{0x11, AC_VERB_SET_COEF_INDEX, 0x0001},
1139	{0x11, AC_VERB_SET_PROC_COEF, 0x0009},
1140
1141	{0x07, AC_VERB_SET_POWER_STATE, 0x00}, /* S/PDIF Rx: D0 */
1142	{0x08, AC_VERB_SET_POWER_STATE, 0x00}, /* S/PDIF Tx: D0 */
1143
1144	{0x11, AC_VERB_SET_COEF_INDEX, 0x0017},
1145	{0x11, AC_VERB_SET_PROC_COEF, 0x2412},
1146	{0x11, AC_VERB_SET_COEF_INDEX, 0x0008},
1147	{0x11, AC_VERB_SET_PROC_COEF, 0x0000},
1148	{0x11, AC_VERB_SET_COEF_INDEX, 0x0001},
1149	{0x11, AC_VERB_SET_PROC_COEF, 0x0008},
1150	{0x11, AC_VERB_SET_PROC_STATE, 0x00},
1151
1152#if 0 /* Don't to set to D3 as we are in power-up sequence */
1153	{0x07, AC_VERB_SET_POWER_STATE, 0x03}, /* S/PDIF Rx: D3 */
1154	{0x08, AC_VERB_SET_POWER_STATE, 0x03}, /* S/PDIF Tx: D3 */
1155	/*{0x01, AC_VERB_SET_POWER_STATE, 0x03},*/ /* AFG: D3 This is already handled */
1156#endif
1157
1158	{} /* terminator */
1159};
1160
1161/* SPDIF setup */
1162static void init_digital(struct hda_codec *codec)
1163{
1164	unsigned int coef;
1165
1166	coef = 0x0002; /* SRC_MUTE soft-mute on SPDIF (if no lock) */
1167	coef |= 0x0008; /* Replace with mute on error */
1168	if (is_active_pin(codec, CS_DIG_OUT2_PIN_NID))
1169		coef |= 0x4000; /* RX to TX1 or TX2 Loopthru / SPDIF2
1170				 * SPDIF_OUT2 is shared with GPIO1 and
1171				 * DMIC_SDA2.
1172				 */
1173	cs_vendor_coef_set(codec, IDX_SPDIF_CTL, coef);
1174}
1175
1176static int cs_init(struct hda_codec *codec)
1177{
1178	struct cs_spec *spec = codec->spec;
1179
1180	/* init_verb sequence for C0/C1/C2 errata*/
1181	snd_hda_sequence_write(codec, cs_errata_init_verbs);
 
 
 
 
 
1182
1183	snd_hda_sequence_write(codec, cs_coef_init_verbs);
1184
1185	if (spec->gpio_mask) {
1186		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_MASK,
1187				    spec->gpio_mask);
1188		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DIRECTION,
1189				    spec->gpio_dir);
1190		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
1191				    spec->gpio_data);
1192	}
1193
1194	init_output(codec);
1195	init_input(codec);
1196	init_digital(codec);
 
 
1197	return 0;
1198}
1199
1200static int cs_build_controls(struct hda_codec *codec)
1201{
1202	int err;
1203
1204	err = build_output(codec);
1205	if (err < 0)
1206		return err;
1207	err = build_input(codec);
1208	if (err < 0)
1209		return err;
1210	err = build_digital_output(codec);
1211	if (err < 0)
1212		return err;
1213	err = build_digital_input(codec);
1214	if (err < 0)
1215		return err;
1216	return cs_init(codec);
1217}
1218
1219static void cs_free(struct hda_codec *codec)
1220{
1221	struct cs_spec *spec = codec->spec;
1222	kfree(spec->capture_bind[0]);
1223	kfree(spec->capture_bind[1]);
1224	kfree(codec->spec);
1225}
1226
1227static void cs_unsol_event(struct hda_codec *codec, unsigned int res)
1228{
1229	switch ((res >> 26) & 0x7f) {
1230	case HP_EVENT:
1231		cs_automute(codec);
1232		break;
1233	case MIC_EVENT:
1234		cs_automic(codec);
1235		break;
1236	}
1237}
1238
1239static const struct hda_codec_ops cs_patch_ops = {
1240	.build_controls = cs_build_controls,
1241	.build_pcms = cs_build_pcms,
1242	.init = cs_init,
1243	.free = cs_free,
1244	.unsol_event = cs_unsol_event,
1245};
1246
1247static int cs_parse_auto_config(struct hda_codec *codec)
1248{
1249	struct cs_spec *spec = codec->spec;
1250	int err;
 
1251
1252	err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL);
1253	if (err < 0)
1254		return err;
1255
1256	err = parse_output(codec);
1257	if (err < 0)
1258		return err;
1259	err = parse_input(codec);
1260	if (err < 0)
1261		return err;
1262	err = parse_digital_output(codec);
1263	if (err < 0)
1264		return err;
1265	err = parse_digital_input(codec);
1266	if (err < 0)
1267		return err;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1268	return 0;
1269}
1270
1271static const char * const cs420x_models[CS420X_MODELS] = {
1272	[CS420X_MBP53] = "mbp53",
1273	[CS420X_MBP55] = "mbp55",
1274	[CS420X_IMAC27] = "imac27",
1275	[CS420X_AUTO] = "auto",
 
 
 
 
 
1276};
1277
1278
1279static const struct snd_pci_quirk cs420x_cfg_tbl[] = {
1280	SND_PCI_QUIRK(0x10de, 0x0ac0, "MacBookPro 5,3", CS420X_MBP53),
1281	SND_PCI_QUIRK(0x10de, 0x0d94, "MacBookAir 3,1(2)", CS420X_MBP55),
1282	SND_PCI_QUIRK(0x10de, 0xcb79, "MacBookPro 5,5", CS420X_MBP55),
1283	SND_PCI_QUIRK(0x10de, 0xcb89, "MacBookPro 7,1", CS420X_MBP55),
1284	SND_PCI_QUIRK(0x8086, 0x7270, "IMac 27 Inch", CS420X_IMAC27),
1285	{} /* terminator */
1286};
1287
1288struct cs_pincfg {
1289	hda_nid_t nid;
1290	u32 val;
 
 
 
 
 
 
 
1291};
1292
1293static const struct cs_pincfg mbp53_pincfgs[] = {
1294	{ 0x09, 0x012b4050 },
1295	{ 0x0a, 0x90100141 },
1296	{ 0x0b, 0x90100140 },
1297	{ 0x0c, 0x018b3020 },
1298	{ 0x0d, 0x90a00110 },
1299	{ 0x0e, 0x400000f0 },
1300	{ 0x0f, 0x01cbe030 },
1301	{ 0x10, 0x014be060 },
1302	{ 0x12, 0x400000f0 },
1303	{ 0x15, 0x400000f0 },
1304	{} /* terminator */
1305};
1306
1307static const struct cs_pincfg mbp55_pincfgs[] = {
1308	{ 0x09, 0x012b4030 },
1309	{ 0x0a, 0x90100121 },
1310	{ 0x0b, 0x90100120 },
1311	{ 0x0c, 0x400000f0 },
1312	{ 0x0d, 0x90a00110 },
1313	{ 0x0e, 0x400000f0 },
1314	{ 0x0f, 0x400000f0 },
1315	{ 0x10, 0x014be040 },
1316	{ 0x12, 0x400000f0 },
1317	{ 0x15, 0x400000f0 },
1318	{} /* terminator */
1319};
1320
1321static const struct cs_pincfg imac27_pincfgs[] = {
1322	{ 0x09, 0x012b4050 },
1323	{ 0x0a, 0x90100140 },
1324	{ 0x0b, 0x90100142 },
1325	{ 0x0c, 0x018b3020 },
1326	{ 0x0d, 0x90a00110 },
1327	{ 0x0e, 0x400000f0 },
1328	{ 0x0f, 0x01cbe030 },
1329	{ 0x10, 0x014be060 },
1330	{ 0x12, 0x01ab9070 },
1331	{ 0x15, 0x400000f0 },
1332	{} /* terminator */
1333};
1334
1335static const struct cs_pincfg *cs_pincfgs[CS420X_MODELS] = {
1336	[CS420X_MBP53] = mbp53_pincfgs,
1337	[CS420X_MBP55] = mbp55_pincfgs,
1338	[CS420X_IMAC27] = imac27_pincfgs,
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1339};
1340
1341static void fix_pincfg(struct hda_codec *codec, int model,
1342		       const struct cs_pincfg **pin_configs)
1343{
1344	const struct cs_pincfg *cfg = pin_configs[model];
1345	if (!cfg)
1346		return;
1347	for (; cfg->nid; cfg++)
1348		snd_hda_codec_set_pincfg(codec, cfg->nid, cfg->val);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1349}
1350
1351static int patch_cs420x(struct hda_codec *codec)
1352{
1353	struct cs_spec *spec;
1354	int err;
1355
1356	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1357	if (!spec)
1358		return -ENOMEM;
1359	codec->spec = spec;
1360
1361	spec->vendor_nid = CS420X_VENDOR_NID;
 
 
1362
1363	spec->board_config =
1364		snd_hda_check_board_config(codec, CS420X_MODELS,
1365					   cs420x_models, cs420x_cfg_tbl);
1366	if (spec->board_config >= 0)
1367		fix_pincfg(codec, spec->board_config, cs_pincfgs);
1368
1369	switch (spec->board_config) {
1370	case CS420X_IMAC27:
1371	case CS420X_MBP53:
1372	case CS420X_MBP55:
1373		/* GPIO1 = headphones */
1374		/* GPIO3 = speakers */
1375		spec->gpio_mask = 0x0a;
1376		spec->gpio_dir = 0x0a;
1377		break;
1378	}
1379
1380	err = cs_parse_auto_config(codec);
1381	if (err < 0)
1382		goto error;
1383
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1384	codec->patch_ops = cs_patch_ops;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1385
1386	return 0;
1387
1388 error:
1389	kfree(codec->spec);
1390	codec->spec = NULL;
1391	return err;
1392}
1393
1394/*
1395 * Cirrus Logic CS4210
1396 *
1397 * 1 DAC => HP(sense) / Speakers,
1398 * 1 ADC <= LineIn(sense) / MicIn / DMicIn,
1399 * 1 SPDIF OUT => SPDIF Trasmitter(sense)
1400*/
1401
1402/* CS4210 board names */
1403static const char *cs421x_models[CS421X_MODELS] = {
1404	[CS421X_CDB4210] = "cdb4210",
 
 
1405};
1406
1407static const struct snd_pci_quirk cs421x_cfg_tbl[] = {
1408	/* Test Intel board + CDB2410  */
1409	SND_PCI_QUIRK(0x8086, 0x5001, "DP45SG/CDB4210", CS421X_CDB4210),
1410	{} /* terminator */
1411};
1412
1413/* CS4210 board pinconfigs */
1414/* Default CS4210 (CDB4210)*/
1415static const struct cs_pincfg cdb4210_pincfgs[] = {
1416	{ 0x05, 0x0321401f },
1417	{ 0x06, 0x90170010 },
1418	{ 0x07, 0x03813031 },
1419	{ 0x08, 0xb7a70037 },
1420	{ 0x09, 0xb7a6003e },
1421	{ 0x0a, 0x034510f0 },
1422	{} /* terminator */
1423};
1424
1425static const struct cs_pincfg *cs421x_pincfgs[CS421X_MODELS] = {
1426	[CS421X_CDB4210] = cdb4210_pincfgs,
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1427};
1428
1429static const struct hda_verb cs421x_coef_init_verbs[] = {
1430	{0x0B, AC_VERB_SET_PROC_STATE, 1},
1431	{0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_DEV_CFG},
1432	/*
1433	    Disable Coefficient Index Auto-Increment(DAI)=1,
1434	    PDREF=0
1435	*/
1436	{0x0B, AC_VERB_SET_PROC_COEF, 0x0001 },
1437
1438	{0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_ADC_CFG},
1439	/* ADC SZCMode = Digital Soft Ramp */
1440	{0x0B, AC_VERB_SET_PROC_COEF, 0x0002 },
1441
1442	{0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_DAC_CFG},
1443	{0x0B, AC_VERB_SET_PROC_COEF,
1444	 (0x0002 /* DAC SZCMode = Digital Soft Ramp */
1445	  | 0x0004 /* Mute DAC on FIFO error */
1446	  | 0x0008 /* Enable DAC High Pass Filter */
1447	  )},
1448	{} /* terminator */
1449};
1450
1451/* Errata: CS4210 rev A1 Silicon
1452 *
1453 * http://www.cirrus.com/en/pubs/errata/
1454 *
1455 * Description:
1456 * 1. Performance degredation is present in the ADC.
1457 * 2. Speaker output is not completely muted upon HP detect.
1458 * 3. Noise is present when clipping occurs on the amplified
1459 *    speaker outputs.
1460 *
1461 * Workaround:
1462 * The following verb sequence written to the registers during
1463 * initialization will correct the issues listed above.
1464 */
1465
1466static const struct hda_verb cs421x_coef_init_verbs_A1_silicon_fixes[] = {
1467	{0x0B, AC_VERB_SET_PROC_STATE, 0x01},  /* VPW: processing on */
1468
1469	{0x0B, AC_VERB_SET_COEF_INDEX, 0x0006},
1470	{0x0B, AC_VERB_SET_PROC_COEF, 0x9999}, /* Test mode: on */
1471
1472	{0x0B, AC_VERB_SET_COEF_INDEX, 0x000A},
1473	{0x0B, AC_VERB_SET_PROC_COEF, 0x14CB}, /* Chop double */
1474
1475	{0x0B, AC_VERB_SET_COEF_INDEX, 0x0011},
1476	{0x0B, AC_VERB_SET_PROC_COEF, 0xA2D0}, /* Increase ADC current */
1477
1478	{0x0B, AC_VERB_SET_COEF_INDEX, 0x001A},
1479	{0x0B, AC_VERB_SET_PROC_COEF, 0x02A9}, /* Mute speaker */
1480
1481	{0x0B, AC_VERB_SET_COEF_INDEX, 0x001B},
1482	{0x0B, AC_VERB_SET_PROC_COEF, 0X1006}, /* Remove noise */
1483
1484	{} /* terminator */
1485};
1486
1487/* Speaker Amp Gain is controlled by the vendor widget's coef 4 */
1488static const DECLARE_TLV_DB_SCALE(cs421x_speaker_boost_db_scale, 900, 300, 0);
1489
1490static int cs421x_boost_vol_info(struct snd_kcontrol *kcontrol,
1491				struct snd_ctl_elem_info *uinfo)
1492{
1493	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
1494	uinfo->count = 1;
1495	uinfo->value.integer.min = 0;
1496	uinfo->value.integer.max = 3;
1497	return 0;
1498}
1499
1500static int cs421x_boost_vol_get(struct snd_kcontrol *kcontrol,
1501				struct snd_ctl_elem_value *ucontrol)
1502{
1503	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1504
1505	ucontrol->value.integer.value[0] =
1506		cs_vendor_coef_get(codec, CS421X_IDX_SPK_CTL) & 0x0003;
1507	return 0;
1508}
1509
1510static int cs421x_boost_vol_put(struct snd_kcontrol *kcontrol,
1511				struct snd_ctl_elem_value *ucontrol)
1512{
1513	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1514
1515	unsigned int vol = ucontrol->value.integer.value[0];
1516	unsigned int coef =
1517		cs_vendor_coef_get(codec, CS421X_IDX_SPK_CTL);
1518	unsigned int original_coef = coef;
1519
1520	coef &= ~0x0003;
1521	coef |= (vol & 0x0003);
1522	if (original_coef == coef)
1523		return 0;
1524	else {
1525		cs_vendor_coef_set(codec, CS421X_IDX_SPK_CTL, coef);
1526		return 1;
1527	}
 
 
1528}
1529
1530static const struct snd_kcontrol_new cs421x_speaker_bost_ctl = {
1531
1532	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1533	.access = (SNDRV_CTL_ELEM_ACCESS_READWRITE |
1534			SNDRV_CTL_ELEM_ACCESS_TLV_READ),
1535	.name = "Speaker Boost Playback Volume",
1536	.info = cs421x_boost_vol_info,
1537	.get = cs421x_boost_vol_get,
1538	.put = cs421x_boost_vol_put,
1539	.tlv = { .p = cs421x_speaker_boost_db_scale },
1540};
1541
1542static void cs421x_pinmux_init(struct hda_codec *codec)
1543{
1544	struct cs_spec *spec = codec->spec;
1545	unsigned int def_conf, coef;
1546
1547	/* GPIO, DMIC_SCL, DMIC_SDA and SENSE_B are multiplexed */
1548	coef = cs_vendor_coef_get(codec, CS421X_IDX_DEV_CFG);
1549
1550	if (spec->gpio_mask)
1551		coef |= 0x0008; /* B1,B2 are GPIOs */
1552	else
1553		coef &= ~0x0008;
1554
1555	if (spec->sense_b)
1556		coef |= 0x0010; /* B2 is SENSE_B, not inverted  */
1557	else
1558		coef &= ~0x0010;
1559
1560	cs_vendor_coef_set(codec, CS421X_IDX_DEV_CFG, coef);
1561
1562	if ((spec->gpio_mask || spec->sense_b) &&
1563	    is_active_pin(codec, CS421X_DMIC_PIN_NID)) {
1564
1565		/*
1566		    GPIO or SENSE_B forced - disconnect the DMIC pin.
1567		*/
1568		def_conf = snd_hda_codec_get_pincfg(codec, CS421X_DMIC_PIN_NID);
1569		def_conf &= ~AC_DEFCFG_PORT_CONN;
1570		def_conf |= (AC_JACK_PORT_NONE << AC_DEFCFG_PORT_CONN_SHIFT);
1571		snd_hda_codec_set_pincfg(codec, CS421X_DMIC_PIN_NID, def_conf);
1572	}
1573}
1574
1575static void init_cs421x_digital(struct hda_codec *codec)
 
1576{
1577	struct cs_spec *spec = codec->spec;
1578	struct auto_pin_cfg *cfg = &spec->autocfg;
1579	int i;
1580
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1581
1582	for (i = 0; i < cfg->dig_outs; i++) {
1583		hda_nid_t nid = cfg->dig_out_pins[i];
1584		if (!cfg->speaker_outs)
1585			continue;
1586		if (get_wcaps(codec, nid) & AC_WCAP_UNSOL_CAP) {
1587
1588			snd_hda_codec_write(codec, nid, 0,
1589				    AC_VERB_SET_UNSOLICITED_ENABLE,
1590				    AC_USRSP_EN | SPDIF_EVENT);
1591			spec->spdif_detect = 1;
 
 
1592		}
1593	}
1594}
1595
1596static int cs421x_init(struct hda_codec *codec)
1597{
1598	struct cs_spec *spec = codec->spec;
1599
1600	snd_hda_sequence_write(codec, cs421x_coef_init_verbs);
1601	snd_hda_sequence_write(codec, cs421x_coef_init_verbs_A1_silicon_fixes);
 
 
 
1602
1603	cs421x_pinmux_init(codec);
1604
1605	if (spec->gpio_mask) {
1606		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_MASK,
1607				    spec->gpio_mask);
1608		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DIRECTION,
1609				    spec->gpio_dir);
1610		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
1611				    spec->gpio_data);
1612	}
1613
1614	init_output(codec);
1615	init_input(codec);
1616	init_cs421x_digital(codec);
1617
1618	return 0;
1619}
1620
1621/*
1622 * CS4210 Input MUX (1 ADC)
1623 */
1624static int cs421x_mux_enum_info(struct snd_kcontrol *kcontrol,
1625					struct snd_ctl_elem_info *uinfo)
1626{
1627	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1628	struct cs_spec *spec = codec->spec;
1629
1630	return snd_hda_input_mux_info(&spec->input_mux, uinfo);
1631}
1632
1633static int cs421x_mux_enum_get(struct snd_kcontrol *kcontrol,
1634					struct snd_ctl_elem_value *ucontrol)
1635{
1636	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1637	struct cs_spec *spec = codec->spec;
1638
1639	ucontrol->value.enumerated.item[0] = spec->cur_input;
1640	return 0;
1641}
1642
1643static int cs421x_mux_enum_put(struct snd_kcontrol *kcontrol,
1644					struct snd_ctl_elem_value *ucontrol)
1645{
1646	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
1647	struct cs_spec *spec = codec->spec;
1648
1649	return snd_hda_input_mux_put(codec, &spec->input_mux, ucontrol,
1650				spec->adc_nid[0], &spec->cur_input);
1651
1652}
1653
1654static struct snd_kcontrol_new cs421x_capture_source = {
1655
1656	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
1657	.name = "Capture Source",
1658	.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
1659	.info = cs421x_mux_enum_info,
1660	.get = cs421x_mux_enum_get,
1661	.put = cs421x_mux_enum_put,
1662};
1663
1664static int cs421x_add_input_volume_control(struct hda_codec *codec, int item)
1665{
1666	struct cs_spec *spec = codec->spec;
1667	struct auto_pin_cfg *cfg = &spec->autocfg;
1668	const struct hda_input_mux *imux = &spec->input_mux;
1669	hda_nid_t pin = cfg->inputs[item].pin;
1670	struct snd_kcontrol *kctl;
1671	u32 caps;
1672
1673	if (!(get_wcaps(codec, pin) & AC_WCAP_IN_AMP))
1674		return 0;
1675
1676	caps = query_amp_caps(codec, pin, HDA_INPUT);
1677	caps = (caps & AC_AMPCAP_NUM_STEPS) >> AC_AMPCAP_NUM_STEPS_SHIFT;
1678	if (caps <= 1)
1679		return 0;
1680
1681	return add_volume(codec,  imux->items[item].label, 0,
1682			  HDA_COMPOSE_AMP_VAL(pin, 3, 0, HDA_INPUT), 1, &kctl);
 
 
 
 
1683}
1684
1685/* add a (input-boost) volume control to the given input pin */
1686static int build_cs421x_input(struct hda_codec *codec)
1687{
1688	struct cs_spec *spec = codec->spec;
1689	struct auto_pin_cfg *cfg = &spec->autocfg;
1690	struct hda_input_mux *imux = &spec->input_mux;
1691	int i, err, type_idx;
1692	const char *label;
1693
1694	if (!spec->num_inputs)
1695		return 0;
1696
1697	/* make bind-capture */
1698	spec->capture_bind[0] = make_bind_capture(codec, &snd_hda_bind_sw);
1699	spec->capture_bind[1] = make_bind_capture(codec, &snd_hda_bind_vol);
1700	for (i = 0; i < 2; i++) {
1701		struct snd_kcontrol *kctl;
1702		int n;
1703		if (!spec->capture_bind[i])
1704			return -ENOMEM;
1705		kctl = snd_ctl_new1(&cs_capture_ctls[i], codec);
1706		if (!kctl)
1707			return -ENOMEM;
1708		kctl->private_value = (long)spec->capture_bind[i];
1709		err = snd_hda_ctl_add(codec, 0, kctl);
1710		if (err < 0)
1711			return err;
1712		for (n = 0; n < AUTO_PIN_LAST; n++) {
1713			if (!spec->adc_nid[n])
1714				continue;
1715			err = snd_hda_add_nid(codec, kctl, 0, spec->adc_nid[n]);
1716			if (err < 0)
1717				return err;
1718		}
1719	}
1720
1721	/* Add Input MUX Items + Capture Volume/Switch */
1722	for (i = 0; i < spec->num_inputs; i++) {
1723		label = hda_get_autocfg_input_label(codec, cfg, i);
1724		snd_hda_add_imux_item(imux, label, spec->adc_idx[i], &type_idx);
1725
1726		err = cs421x_add_input_volume_control(codec, i);
1727		if (err < 0)
1728			return err;
1729	}
1730
1731	/*
1732	    Add 'Capture Source' Switch if
1733		* 2 inputs and no mic detec
1734		* 3 inputs
1735	*/
1736	if ((spec->num_inputs == 2 && !spec->mic_detect) ||
1737	    (spec->num_inputs == 3)) {
1738
1739		err = snd_hda_ctl_add(codec, spec->adc_nid[0],
1740			      snd_ctl_new1(&cs421x_capture_source, codec));
1741		if (err < 0)
1742			return err;
1743	}
1744
1745	return 0;
1746}
1747
1748/* Single DAC (Mute/Gain) */
1749static int build_cs421x_output(struct hda_codec *codec)
1750{
1751	hda_nid_t dac = CS4210_DAC_NID;
1752	struct cs_spec *spec = codec->spec;
1753	struct auto_pin_cfg *cfg = &spec->autocfg;
1754	struct snd_kcontrol *kctl;
1755	int err;
1756	char *name = "HP/Speakers";
1757
1758	fix_volume_caps(codec, dac);
1759	if (!spec->vmaster_sw) {
1760		err = add_vmaster(codec, dac);
1761		if (err < 0)
1762			return err;
1763	}
1764
1765	err = add_mute(codec, name, 0,
1766			HDA_COMPOSE_AMP_VAL(dac, 3, 0, HDA_OUTPUT), 0, &kctl);
1767	if (err < 0)
1768		return err;
1769	err = snd_ctl_add_slave(spec->vmaster_sw, kctl);
1770	if (err < 0)
1771		return err;
1772
1773	err = add_volume(codec, name, 0,
1774			HDA_COMPOSE_AMP_VAL(dac, 3, 0, HDA_OUTPUT), 0, &kctl);
1775	if (err < 0)
1776		return err;
1777	err = snd_ctl_add_slave(spec->vmaster_vol, kctl);
1778	if (err < 0)
1779		return err;
1780
1781	if (cfg->speaker_outs) {
1782		err = snd_hda_ctl_add(codec, 0,
1783			snd_ctl_new1(&cs421x_speaker_bost_ctl, codec));
1784		if (err < 0)
1785			return err;
1786	}
1787	return err;
1788}
1789
1790static int cs421x_build_controls(struct hda_codec *codec)
1791{
1792	int err;
1793
1794	err = build_cs421x_output(codec);
1795	if (err < 0)
1796		return err;
1797	err = build_cs421x_input(codec);
1798	if (err < 0)
1799		return err;
1800	err = build_digital_output(codec);
1801	if (err < 0)
1802		return err;
1803	return cs421x_init(codec);
1804}
1805
1806static void cs421x_unsol_event(struct hda_codec *codec, unsigned int res)
1807{
1808	switch ((res >> 26) & 0x3f) {
1809	case HP_EVENT:
1810	case SPDIF_EVENT:
1811		cs_automute(codec);
1812		break;
1813
1814	case MIC_EVENT:
1815		cs_automic(codec);
1816		break;
1817	}
1818}
1819
1820static int parse_cs421x_input(struct hda_codec *codec)
1821{
1822	struct cs_spec *spec = codec->spec;
1823	struct auto_pin_cfg *cfg = &spec->autocfg;
1824	int i;
1825
1826	for (i = 0; i < cfg->num_inputs; i++) {
1827		hda_nid_t pin = cfg->inputs[i].pin;
1828		spec->adc_nid[i] = get_adc(codec, pin, &spec->adc_idx[i]);
1829		spec->cur_input = spec->last_input = i;
1830		spec->num_inputs++;
1831
1832		/* check whether the automatic mic switch is available */
1833		if (is_ext_mic(codec, i) && cfg->num_inputs >= 2) {
1834			spec->mic_detect = 1;
1835			spec->automic_idx = i;
1836		}
1837	}
1838	return 0;
1839}
1840
1841static int cs421x_parse_auto_config(struct hda_codec *codec)
1842{
1843	struct cs_spec *spec = codec->spec;
1844	int err;
1845
1846	err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL);
1847	if (err < 0)
1848		return err;
1849	err = parse_output(codec);
1850	if (err < 0)
1851		return err;
1852	err = parse_cs421x_input(codec);
1853	if (err < 0)
1854		return err;
1855	err = parse_digital_output(codec);
1856	if (err < 0)
1857		return err;
1858	return 0;
1859}
1860
1861#ifdef CONFIG_PM
1862/*
1863	Manage PDREF, when transitioning to D3hot
1864	(DAC,ADC) -> D3, PDREF=1, AFG->D3
1865*/
1866static int cs421x_suspend(struct hda_codec *codec, pm_message_t state)
1867{
 
1868	unsigned int coef;
1869
1870	snd_hda_shutup_pins(codec);
1871
1872	snd_hda_codec_write(codec, CS4210_DAC_NID, 0,
1873			    AC_VERB_SET_POWER_STATE,  AC_PWRST_D3);
1874	snd_hda_codec_write(codec, CS4210_ADC_NID, 0,
1875			    AC_VERB_SET_POWER_STATE,  AC_PWRST_D3);
1876
1877	coef = cs_vendor_coef_get(codec, CS421X_IDX_DEV_CFG);
1878	coef |= 0x0004; /* PDREF */
1879	cs_vendor_coef_set(codec, CS421X_IDX_DEV_CFG, coef);
 
 
1880
1881	return 0;
1882}
1883#endif
1884
1885static struct hda_codec_ops cs4210_patch_ops = {
1886	.build_controls = cs421x_build_controls,
1887	.build_pcms = cs_build_pcms,
1888	.init = cs421x_init,
1889	.free = cs_free,
1890	.unsol_event = cs421x_unsol_event,
1891#ifdef CONFIG_PM
1892	.suspend = cs421x_suspend,
1893#endif
1894};
1895
1896static int patch_cs421x(struct hda_codec *codec)
1897{
1898	struct cs_spec *spec;
1899	int err;
1900
1901	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
1902	if (!spec)
1903		return -ENOMEM;
1904	codec->spec = spec;
1905
1906	spec->vendor_nid = CS421X_VENDOR_NID;
1907
1908	spec->board_config =
1909		snd_hda_check_board_config(codec, CS421X_MODELS,
1910					   cs421x_models, cs421x_cfg_tbl);
1911	if (spec->board_config >= 0)
1912		fix_pincfg(codec, spec->board_config, cs421x_pincfgs);
1913	/*
1914	    Setup GPIO/SENSE for each board (if used)
1915	*/
1916	switch (spec->board_config) {
1917	case CS421X_CDB4210:
1918		snd_printd("CS4210 board: %s\n",
1919			cs421x_models[spec->board_config]);
1920/*		spec->gpio_mask = 3;
1921		spec->gpio_dir = 3;
1922		spec->gpio_data = 3;
1923*/
1924		spec->sense_b = 1;
1925
1926		break;
1927	}
 
1928
1929	/*
1930	    Update the GPIO/DMIC/SENSE_B pinmux before the configuration
1931	    is auto-parsed. If GPIO or SENSE_B is forced, DMIC input
1932	    is disabled.
1933	*/
1934	cs421x_pinmux_init(codec);
1935
1936	err = cs421x_parse_auto_config(codec);
1937	if (err < 0)
1938		goto error;
1939
1940	codec->patch_ops = cs4210_patch_ops;
1941
1942	return 0;
1943
1944 error:
1945	kfree(codec->spec);
1946	codec->spec = NULL;
1947	return err;
1948}
1949
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1950
1951/*
1952 * patch entries
1953 */
1954static const struct hda_codec_preset snd_hda_preset_cirrus[] = {
1955	{ .id = 0x10134206, .name = "CS4206", .patch = patch_cs420x },
1956	{ .id = 0x10134207, .name = "CS4207", .patch = patch_cs420x },
1957	{ .id = 0x10134210, .name = "CS4210", .patch = patch_cs421x },
 
 
1958	{} /* terminator */
1959};
1960
1961MODULE_ALIAS("snd-hda-codec-id:10134206");
1962MODULE_ALIAS("snd-hda-codec-id:10134207");
1963MODULE_ALIAS("snd-hda-codec-id:10134210");
1964
1965MODULE_LICENSE("GPL");
1966MODULE_DESCRIPTION("Cirrus Logic HD-audio codec");
1967
1968static struct hda_codec_preset_list cirrus_list = {
1969	.preset = snd_hda_preset_cirrus,
1970	.owner = THIS_MODULE,
1971};
1972
1973static int __init patch_cirrus_init(void)
1974{
1975	return snd_hda_add_codec_preset(&cirrus_list);
1976}
1977
1978static void __exit patch_cirrus_exit(void)
1979{
1980	snd_hda_delete_codec_preset(&cirrus_list);
1981}
1982
1983module_init(patch_cirrus_init)
1984module_exit(patch_cirrus_exit)
v6.8
   1// SPDX-License-Identifier: GPL-2.0-or-later
   2/*
   3 * HD audio interface patch for Cirrus Logic CS420x chip
   4 *
   5 * Copyright (c) 2009 Takashi Iwai <tiwai@suse.de>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
   6 */
   7
   8#include <linux/init.h>
 
   9#include <linux/slab.h>
  10#include <linux/module.h>
  11#include <sound/core.h>
  12#include <linux/pci.h>
 
  13#include <sound/tlv.h>
  14#include <sound/hda_codec.h>
  15#include "hda_local.h"
  16#include "hda_auto_parser.h"
  17#include "hda_jack.h"
  18#include "hda_generic.h"
  19
  20/*
  21 */
  22
  23struct cs_spec {
  24	struct hda_gen_spec gen;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  25
  26	unsigned int gpio_mask;
  27	unsigned int gpio_dir;
  28	unsigned int gpio_data;
  29	unsigned int gpio_eapd_hp; /* EAPD GPIO bit for headphones */
  30	unsigned int gpio_eapd_speaker; /* EAPD GPIO bit for speakers */
  31
 
 
 
 
  32	/* CS421x */
  33	unsigned int spdif_detect:1;
  34	unsigned int spdif_present:1;
  35	unsigned int sense_b:1;
  36	hda_nid_t vendor_nid;
  37
  38	/* for MBP SPDIF control */
  39	int (*spdif_sw_put)(struct snd_kcontrol *kcontrol,
  40			    struct snd_ctl_elem_value *ucontrol);
  41};
  42
  43/* available models with CS420x */
  44enum {
  45	CS420X_MBP53,
  46	CS420X_MBP55,
  47	CS420X_IMAC27,
  48	CS420X_GPIO_13,
  49	CS420X_GPIO_23,
  50	CS420X_MBP101,
  51	CS420X_MBP81,
  52	CS420X_MBA42,
  53	CS420X_AUTO,
  54	/* aliases */
  55	CS420X_IMAC27_122 = CS420X_GPIO_23,
  56	CS420X_APPLE = CS420X_GPIO_13,
  57};
  58
  59/* CS421x boards */
  60enum {
  61	CS421X_CDB4210,
  62	CS421X_SENSE_B,
  63	CS421X_STUMPY,
  64};
  65
  66/* Vendor-specific processing widget */
  67#define CS420X_VENDOR_NID	0x11
  68#define CS_DIG_OUT1_PIN_NID	0x10
  69#define CS_DIG_OUT2_PIN_NID	0x15
  70#define CS_DMIC1_PIN_NID	0x0e
  71#define CS_DMIC2_PIN_NID	0x12
  72
  73/* coef indices */
  74#define IDX_SPDIF_STAT		0x0000
  75#define IDX_SPDIF_CTL		0x0001
  76#define IDX_ADC_CFG		0x0002
  77/* SZC bitmask, 4 modes below:
  78 * 0 = immediate,
  79 * 1 = digital immediate, analog zero-cross
  80 * 2 = digtail & analog soft-ramp
  81 * 3 = digital soft-ramp, analog zero-cross
  82 */
  83#define   CS_COEF_ADC_SZC_MASK		(3 << 0)
  84#define   CS_COEF_ADC_MIC_SZC_MODE	(3 << 0) /* SZC setup for mic */
  85#define   CS_COEF_ADC_LI_SZC_MODE	(3 << 0) /* SZC setup for line-in */
  86/* PGA mode: 0 = differential, 1 = signle-ended */
  87#define   CS_COEF_ADC_MIC_PGA_MODE	(1 << 5) /* PGA setup for mic */
  88#define   CS_COEF_ADC_LI_PGA_MODE	(1 << 6) /* PGA setup for line-in */
  89#define IDX_DAC_CFG		0x0003
  90/* SZC bitmask, 4 modes below:
  91 * 0 = Immediate
  92 * 1 = zero-cross
  93 * 2 = soft-ramp
  94 * 3 = soft-ramp on zero-cross
  95 */
  96#define   CS_COEF_DAC_HP_SZC_MODE	(3 << 0) /* nid 0x02 */
  97#define   CS_COEF_DAC_LO_SZC_MODE	(3 << 2) /* nid 0x03 */
  98#define   CS_COEF_DAC_SPK_SZC_MODE	(3 << 4) /* nid 0x04 */
  99
 100#define IDX_BEEP_CFG		0x0004
 101/* 0x0008 - test reg key */
 102/* 0x0009 - 0x0014 -> 12 test regs */
 103/* 0x0015 - visibility reg */
 104
 105/* Cirrus Logic CS4208 */
 106#define CS4208_VENDOR_NID	0x24
 107
 108/*
 109 * Cirrus Logic CS4210
 110 *
 111 * 1 DAC => HP(sense) / Speakers,
 112 * 1 ADC <= LineIn(sense) / MicIn / DMicIn,
 113 * 1 SPDIF OUT => SPDIF Trasmitter(sense)
 114 */
 115#define CS4210_DAC_NID		0x02
 116#define CS4210_ADC_NID		0x03
 117#define CS4210_VENDOR_NID	0x0B
 118#define CS421X_DMIC_PIN_NID	0x09 /* Port E */
 119#define CS421X_SPDIF_PIN_NID	0x0A /* Port H */
 120
 121#define CS421X_IDX_DEV_CFG	0x01
 122#define CS421X_IDX_ADC_CFG	0x02
 123#define CS421X_IDX_DAC_CFG	0x03
 124#define CS421X_IDX_SPK_CTL	0x04
 125
 126/* Cirrus Logic CS4213 is like CS4210 but does not have SPDIF input/output */
 127#define CS4213_VENDOR_NID	0x09
 128
 129
 130static inline int cs_vendor_coef_get(struct hda_codec *codec, unsigned int idx)
 131{
 132	struct cs_spec *spec = codec->spec;
 133
 134	snd_hda_codec_write(codec, spec->vendor_nid, 0,
 135			    AC_VERB_SET_COEF_INDEX, idx);
 136	return snd_hda_codec_read(codec, spec->vendor_nid, 0,
 137				  AC_VERB_GET_PROC_COEF, 0);
 138}
 139
 140static inline void cs_vendor_coef_set(struct hda_codec *codec, unsigned int idx,
 141				      unsigned int coef)
 142{
 143	struct cs_spec *spec = codec->spec;
 144
 145	snd_hda_codec_write(codec, spec->vendor_nid, 0,
 146			    AC_VERB_SET_COEF_INDEX, idx);
 147	snd_hda_codec_write(codec, spec->vendor_nid, 0,
 148			    AC_VERB_SET_PROC_COEF, coef);
 149}
 150
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 151/*
 152 * auto-mute and auto-mic switching
 153 * CS421x auto-output redirecting
 154 * HP/SPK/SPDIF
 155 */
 156
 157static void cs_automute(struct hda_codec *codec)
 158{
 159	struct cs_spec *spec = codec->spec;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 160
 161	/* mute HPs if spdif jack (SENSE_B) is present */
 162	spec->gen.master_mute = !!(spec->spdif_present && spec->sense_b);
 
 
 
 
 
 
 
 163
 164	snd_hda_gen_update_outputs(codec);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 165
 166	if (spec->gpio_eapd_hp || spec->gpio_eapd_speaker) {
 167		if (spec->gen.automute_speaker)
 168			spec->gpio_data = spec->gen.hp_jack_present ?
 169				spec->gpio_eapd_hp : spec->gpio_eapd_speaker;
 
 
 170		else
 171			spec->gpio_data =
 172				spec->gpio_eapd_hp | spec->gpio_eapd_speaker;
 173		snd_hda_codec_write(codec, 0x01, 0,
 174				    AC_VERB_SET_GPIO_DATA, spec->gpio_data);
 175	}
 176}
 177
 178static bool is_active_pin(struct hda_codec *codec, hda_nid_t nid)
 
 
 
 179{
 180	unsigned int val;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 181
 182	val = snd_hda_codec_get_pincfg(codec, nid);
 183	return (get_defcfg_connect(val) != AC_JACK_PORT_NONE);
 
 184}
 185
 186static void init_input_coef(struct hda_codec *codec)
 187{
 188	struct cs_spec *spec = codec->spec;
 
 189	unsigned int coef;
 
 190
 191	/* CS420x has multiple ADC, CS421x has single ADC */
 192	if (spec->vendor_nid == CS420X_VENDOR_NID) {
 193		coef = cs_vendor_coef_get(codec, IDX_BEEP_CFG);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 194		if (is_active_pin(codec, CS_DMIC2_PIN_NID))
 195			coef |= 1 << 4; /* DMIC2 2 chan on, GPIO1 off */
 196		if (is_active_pin(codec, CS_DMIC1_PIN_NID))
 197			coef |= 1 << 3; /* DMIC1 2 chan on, GPIO0 off
 198					 * No effect if SPDIF_OUT2 is
 199					 * selected in IDX_SPDIF_CTL.
 200					 */
 201
 202		cs_vendor_coef_set(codec, IDX_BEEP_CFG, coef);
 203	}
 204}
 205
 206static const struct hda_verb cs_coef_init_verbs[] = {
 207	{0x11, AC_VERB_SET_PROC_STATE, 1},
 208	{0x11, AC_VERB_SET_COEF_INDEX, IDX_DAC_CFG},
 209	{0x11, AC_VERB_SET_PROC_COEF,
 210	 (0x002a /* DAC1/2/3 SZCMode Soft Ramp */
 211	  | 0x0040 /* Mute DACs on FIFO error */
 212	  | 0x1000 /* Enable DACs High Pass Filter */
 213	  | 0x0400 /* Disable Coefficient Auto increment */
 214	  )},
 215	/* ADC1/2 - Digital and Analog Soft Ramp */
 216	{0x11, AC_VERB_SET_COEF_INDEX, IDX_ADC_CFG},
 217	{0x11, AC_VERB_SET_PROC_COEF, 0x000a},
 218	/* Beep */
 219	{0x11, AC_VERB_SET_COEF_INDEX, IDX_BEEP_CFG},
 220	{0x11, AC_VERB_SET_PROC_COEF, 0x0007}, /* Enable Beep thru DAC1/2/3 */
 221
 222	{} /* terminator */
 223};
 224
 225static const struct hda_verb cs4208_coef_init_verbs[] = {
 226	{0x01, AC_VERB_SET_POWER_STATE, 0x00}, /* AFG: D0 */
 227	{0x24, AC_VERB_SET_PROC_STATE, 0x01},  /* VPW: processing on */
 228	{0x24, AC_VERB_SET_COEF_INDEX, 0x0033},
 229	{0x24, AC_VERB_SET_PROC_COEF, 0x0001}, /* A1 ICS */
 230	{0x24, AC_VERB_SET_COEF_INDEX, 0x0034},
 231	{0x24, AC_VERB_SET_PROC_COEF, 0x1C01}, /* A1 Enable, A Thresh = 300mV */
 232	{} /* terminator */
 233};
 234
 235/* Errata: CS4207 rev C0/C1/C2 Silicon
 236 *
 237 * http://www.cirrus.com/en/pubs/errata/ER880C3.pdf
 238 *
 239 * 6. At high temperature (TA > +85°C), the digital supply current (IVD)
 240 * may be excessive (up to an additional 200 μA), which is most easily
 241 * observed while the part is being held in reset (RESET# active low).
 242 *
 243 * Root Cause: At initial powerup of the device, the logic that drives
 244 * the clock and write enable to the S/PDIF SRC RAMs is not properly
 245 * initialized.
 246 * Certain random patterns will cause a steady leakage current in those
 247 * RAM cells. The issue will resolve once the SRCs are used (turned on).
 248 *
 249 * Workaround: The following verb sequence briefly turns on the S/PDIF SRC
 250 * blocks, which will alleviate the issue.
 251 */
 252
 253static const struct hda_verb cs_errata_init_verbs[] = {
 254	{0x01, AC_VERB_SET_POWER_STATE, 0x00}, /* AFG: D0 */
 255	{0x11, AC_VERB_SET_PROC_STATE, 0x01},  /* VPW: processing on */
 256
 257	{0x11, AC_VERB_SET_COEF_INDEX, 0x0008},
 258	{0x11, AC_VERB_SET_PROC_COEF, 0x9999},
 259	{0x11, AC_VERB_SET_COEF_INDEX, 0x0017},
 260	{0x11, AC_VERB_SET_PROC_COEF, 0xa412},
 261	{0x11, AC_VERB_SET_COEF_INDEX, 0x0001},
 262	{0x11, AC_VERB_SET_PROC_COEF, 0x0009},
 263
 264	{0x07, AC_VERB_SET_POWER_STATE, 0x00}, /* S/PDIF Rx: D0 */
 265	{0x08, AC_VERB_SET_POWER_STATE, 0x00}, /* S/PDIF Tx: D0 */
 266
 267	{0x11, AC_VERB_SET_COEF_INDEX, 0x0017},
 268	{0x11, AC_VERB_SET_PROC_COEF, 0x2412},
 269	{0x11, AC_VERB_SET_COEF_INDEX, 0x0008},
 270	{0x11, AC_VERB_SET_PROC_COEF, 0x0000},
 271	{0x11, AC_VERB_SET_COEF_INDEX, 0x0001},
 272	{0x11, AC_VERB_SET_PROC_COEF, 0x0008},
 273	{0x11, AC_VERB_SET_PROC_STATE, 0x00},
 
 
 
 
 
 
 
 274	{} /* terminator */
 275};
 276
 277/* SPDIF setup */
 278static void init_digital_coef(struct hda_codec *codec)
 279{
 280	unsigned int coef;
 281
 282	coef = 0x0002; /* SRC_MUTE soft-mute on SPDIF (if no lock) */
 283	coef |= 0x0008; /* Replace with mute on error */
 284	if (is_active_pin(codec, CS_DIG_OUT2_PIN_NID))
 285		coef |= 0x4000; /* RX to TX1 or TX2 Loopthru / SPDIF2
 286				 * SPDIF_OUT2 is shared with GPIO1 and
 287				 * DMIC_SDA2.
 288				 */
 289	cs_vendor_coef_set(codec, IDX_SPDIF_CTL, coef);
 290}
 291
 292static int cs_init(struct hda_codec *codec)
 293{
 294	struct cs_spec *spec = codec->spec;
 295
 296	if (spec->vendor_nid == CS420X_VENDOR_NID) {
 297		/* init_verb sequence for C0/C1/C2 errata*/
 298		snd_hda_sequence_write(codec, cs_errata_init_verbs);
 299		snd_hda_sequence_write(codec, cs_coef_init_verbs);
 300	} else if (spec->vendor_nid == CS4208_VENDOR_NID) {
 301		snd_hda_sequence_write(codec, cs4208_coef_init_verbs);
 302	}
 303
 304	snd_hda_gen_init(codec);
 305
 306	if (spec->gpio_mask) {
 307		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_MASK,
 308				    spec->gpio_mask);
 309		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DIRECTION,
 310				    spec->gpio_dir);
 311		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
 312				    spec->gpio_data);
 313	}
 314
 315	if (spec->vendor_nid == CS420X_VENDOR_NID) {
 316		init_input_coef(codec);
 317		init_digital_coef(codec);
 318	}
 319
 320	return 0;
 321}
 322
 323static int cs_build_controls(struct hda_codec *codec)
 324{
 325	int err;
 326
 327	err = snd_hda_gen_build_controls(codec);
 328	if (err < 0)
 329		return err;
 330	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_BUILD);
 331	return 0;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 332}
 333
 334#define cs_free		snd_hda_gen_free
 
 
 
 
 
 
 
 
 
 
 335
 336static const struct hda_codec_ops cs_patch_ops = {
 337	.build_controls = cs_build_controls,
 338	.build_pcms = snd_hda_gen_build_pcms,
 339	.init = cs_init,
 340	.free = cs_free,
 341	.unsol_event = snd_hda_jack_unsol_event,
 342};
 343
 344static int cs_parse_auto_config(struct hda_codec *codec)
 345{
 346	struct cs_spec *spec = codec->spec;
 347	int err;
 348	int i;
 349
 350	err = snd_hda_parse_pin_defcfg(codec, &spec->gen.autocfg, NULL, 0);
 351	if (err < 0)
 352		return err;
 353
 354	err = snd_hda_gen_parse_auto_config(codec, &spec->gen.autocfg);
 
 
 
 
 
 
 
 
 
 355	if (err < 0)
 356		return err;
 357
 358	/* keep the ADCs powered up when it's dynamically switchable */
 359	if (spec->gen.dyn_adc_switch) {
 360		unsigned int done = 0;
 361
 362		for (i = 0; i < spec->gen.input_mux.num_items; i++) {
 363			int idx = spec->gen.dyn_adc_idx[i];
 364
 365			if (done & (1 << idx))
 366				continue;
 367			snd_hda_gen_fix_pin_power(codec,
 368						  spec->gen.adc_nids[idx]);
 369			done |= 1 << idx;
 370		}
 371	}
 372
 373	return 0;
 374}
 375
 376static const struct hda_model_fixup cs420x_models[] = {
 377	{ .id = CS420X_MBP53, .name = "mbp53" },
 378	{ .id = CS420X_MBP55, .name = "mbp55" },
 379	{ .id = CS420X_IMAC27, .name = "imac27" },
 380	{ .id = CS420X_IMAC27_122, .name = "imac27_122" },
 381	{ .id = CS420X_APPLE, .name = "apple" },
 382	{ .id = CS420X_MBP101, .name = "mbp101" },
 383	{ .id = CS420X_MBP81, .name = "mbp81" },
 384	{ .id = CS420X_MBA42, .name = "mba42" },
 385	{}
 386};
 387
 388static const struct snd_pci_quirk cs420x_fixup_tbl[] = {
 
 389	SND_PCI_QUIRK(0x10de, 0x0ac0, "MacBookPro 5,3", CS420X_MBP53),
 390	SND_PCI_QUIRK(0x10de, 0x0d94, "MacBookAir 3,1(2)", CS420X_MBP55),
 391	SND_PCI_QUIRK(0x10de, 0xcb79, "MacBookPro 5,5", CS420X_MBP55),
 392	SND_PCI_QUIRK(0x10de, 0xcb89, "MacBookPro 7,1", CS420X_MBP55),
 393	/* this conflicts with too many other models */
 394	/*SND_PCI_QUIRK(0x8086, 0x7270, "IMac 27 Inch", CS420X_IMAC27),*/
 
 395
 396	/* codec SSID */
 397	SND_PCI_QUIRK(0x106b, 0x0600, "iMac 14,1", CS420X_IMAC27_122),
 398	SND_PCI_QUIRK(0x106b, 0x0900, "iMac 12,1", CS420X_IMAC27_122),
 399	SND_PCI_QUIRK(0x106b, 0x1c00, "MacBookPro 8,1", CS420X_MBP81),
 400	SND_PCI_QUIRK(0x106b, 0x2000, "iMac 12,2", CS420X_IMAC27_122),
 401	SND_PCI_QUIRK(0x106b, 0x2800, "MacBookPro 10,1", CS420X_MBP101),
 402	SND_PCI_QUIRK(0x106b, 0x5600, "MacBookAir 5,2", CS420X_MBP81),
 403	SND_PCI_QUIRK(0x106b, 0x5b00, "MacBookAir 4,2", CS420X_MBA42),
 404	SND_PCI_QUIRK_VENDOR(0x106b, "Apple", CS420X_APPLE),
 405	{} /* terminator */
 406};
 407
 408static const struct hda_pintbl mbp53_pincfgs[] = {
 409	{ 0x09, 0x012b4050 },
 410	{ 0x0a, 0x90100141 },
 411	{ 0x0b, 0x90100140 },
 412	{ 0x0c, 0x018b3020 },
 413	{ 0x0d, 0x90a00110 },
 414	{ 0x0e, 0x400000f0 },
 415	{ 0x0f, 0x01cbe030 },
 416	{ 0x10, 0x014be060 },
 417	{ 0x12, 0x400000f0 },
 418	{ 0x15, 0x400000f0 },
 419	{} /* terminator */
 420};
 421
 422static const struct hda_pintbl mbp55_pincfgs[] = {
 423	{ 0x09, 0x012b4030 },
 424	{ 0x0a, 0x90100121 },
 425	{ 0x0b, 0x90100120 },
 426	{ 0x0c, 0x400000f0 },
 427	{ 0x0d, 0x90a00110 },
 428	{ 0x0e, 0x400000f0 },
 429	{ 0x0f, 0x400000f0 },
 430	{ 0x10, 0x014be040 },
 431	{ 0x12, 0x400000f0 },
 432	{ 0x15, 0x400000f0 },
 433	{} /* terminator */
 434};
 435
 436static const struct hda_pintbl imac27_pincfgs[] = {
 437	{ 0x09, 0x012b4050 },
 438	{ 0x0a, 0x90100140 },
 439	{ 0x0b, 0x90100142 },
 440	{ 0x0c, 0x018b3020 },
 441	{ 0x0d, 0x90a00110 },
 442	{ 0x0e, 0x400000f0 },
 443	{ 0x0f, 0x01cbe030 },
 444	{ 0x10, 0x014be060 },
 445	{ 0x12, 0x01ab9070 },
 446	{ 0x15, 0x400000f0 },
 447	{} /* terminator */
 448};
 449
 450static const struct hda_pintbl mbp101_pincfgs[] = {
 451	{ 0x0d, 0x40ab90f0 },
 452	{ 0x0e, 0x90a600f0 },
 453	{ 0x12, 0x50a600f0 },
 454	{} /* terminator */
 455};
 456
 457static const struct hda_pintbl mba42_pincfgs[] = {
 458	{ 0x09, 0x012b4030 }, /* HP */
 459	{ 0x0a, 0x400000f0 },
 460	{ 0x0b, 0x90100120 }, /* speaker */
 461	{ 0x0c, 0x400000f0 },
 462	{ 0x0d, 0x90a00110 }, /* mic */
 463	{ 0x0e, 0x400000f0 },
 464	{ 0x0f, 0x400000f0 },
 465	{ 0x10, 0x400000f0 },
 466	{ 0x12, 0x400000f0 },
 467	{ 0x15, 0x400000f0 },
 468	{} /* terminator */
 469};
 470
 471static const struct hda_pintbl mba6_pincfgs[] = {
 472	{ 0x10, 0x032120f0 }, /* HP */
 473	{ 0x11, 0x500000f0 },
 474	{ 0x12, 0x90100010 }, /* Speaker */
 475	{ 0x13, 0x500000f0 },
 476	{ 0x14, 0x500000f0 },
 477	{ 0x15, 0x770000f0 },
 478	{ 0x16, 0x770000f0 },
 479	{ 0x17, 0x430000f0 },
 480	{ 0x18, 0x43ab9030 }, /* Mic */
 481	{ 0x19, 0x770000f0 },
 482	{ 0x1a, 0x770000f0 },
 483	{ 0x1b, 0x770000f0 },
 484	{ 0x1c, 0x90a00090 },
 485	{ 0x1d, 0x500000f0 },
 486	{ 0x1e, 0x500000f0 },
 487	{ 0x1f, 0x500000f0 },
 488	{ 0x20, 0x500000f0 },
 489	{ 0x21, 0x430000f0 },
 490	{ 0x22, 0x430000f0 },
 491	{} /* terminator */
 492};
 493
 494static void cs420x_fixup_gpio_13(struct hda_codec *codec,
 495				 const struct hda_fixup *fix, int action)
 496{
 497	if (action == HDA_FIXUP_ACT_PRE_PROBE) {
 498		struct cs_spec *spec = codec->spec;
 499
 500		spec->gpio_eapd_hp = 2; /* GPIO1 = headphones */
 501		spec->gpio_eapd_speaker = 8; /* GPIO3 = speakers */
 502		spec->gpio_mask = spec->gpio_dir =
 503			spec->gpio_eapd_hp | spec->gpio_eapd_speaker;
 504	}
 505}
 506
 507static void cs420x_fixup_gpio_23(struct hda_codec *codec,
 508				 const struct hda_fixup *fix, int action)
 509{
 510	if (action == HDA_FIXUP_ACT_PRE_PROBE) {
 511		struct cs_spec *spec = codec->spec;
 512
 513		spec->gpio_eapd_hp = 4; /* GPIO2 = headphones */
 514		spec->gpio_eapd_speaker = 8; /* GPIO3 = speakers */
 515		spec->gpio_mask = spec->gpio_dir =
 516			spec->gpio_eapd_hp | spec->gpio_eapd_speaker;
 517	}
 518}
 519
 520static const struct hda_fixup cs420x_fixups[] = {
 521	[CS420X_MBP53] = {
 522		.type = HDA_FIXUP_PINS,
 523		.v.pins = mbp53_pincfgs,
 524		.chained = true,
 525		.chain_id = CS420X_APPLE,
 526	},
 527	[CS420X_MBP55] = {
 528		.type = HDA_FIXUP_PINS,
 529		.v.pins = mbp55_pincfgs,
 530		.chained = true,
 531		.chain_id = CS420X_GPIO_13,
 532	},
 533	[CS420X_IMAC27] = {
 534		.type = HDA_FIXUP_PINS,
 535		.v.pins = imac27_pincfgs,
 536		.chained = true,
 537		.chain_id = CS420X_GPIO_13,
 538	},
 539	[CS420X_GPIO_13] = {
 540		.type = HDA_FIXUP_FUNC,
 541		.v.func = cs420x_fixup_gpio_13,
 542	},
 543	[CS420X_GPIO_23] = {
 544		.type = HDA_FIXUP_FUNC,
 545		.v.func = cs420x_fixup_gpio_23,
 546	},
 547	[CS420X_MBP101] = {
 548		.type = HDA_FIXUP_PINS,
 549		.v.pins = mbp101_pincfgs,
 550		.chained = true,
 551		.chain_id = CS420X_GPIO_13,
 552	},
 553	[CS420X_MBP81] = {
 554		.type = HDA_FIXUP_VERBS,
 555		.v.verbs = (const struct hda_verb[]) {
 556			/* internal mic ADC2: right only, single ended */
 557			{0x11, AC_VERB_SET_COEF_INDEX, IDX_ADC_CFG},
 558			{0x11, AC_VERB_SET_PROC_COEF, 0x102a},
 559			{}
 560		},
 561		.chained = true,
 562		.chain_id = CS420X_GPIO_13,
 563	},
 564	[CS420X_MBA42] = {
 565		.type = HDA_FIXUP_PINS,
 566		.v.pins = mba42_pincfgs,
 567		.chained = true,
 568		.chain_id = CS420X_GPIO_13,
 569	},
 570};
 571
 572static struct cs_spec *cs_alloc_spec(struct hda_codec *codec, int vendor_nid)
 573{
 574	struct cs_spec *spec;
 575
 576	spec = kzalloc(sizeof(*spec), GFP_KERNEL);
 577	if (!spec)
 578		return NULL;
 579	codec->spec = spec;
 580	spec->vendor_nid = vendor_nid;
 581	codec->power_save_node = 1;
 582	snd_hda_gen_spec_init(&spec->gen);
 583
 584	return spec;
 585}
 586
 587static int patch_cs420x(struct hda_codec *codec)
 588{
 589	struct cs_spec *spec;
 590	int err;
 591
 592	spec = cs_alloc_spec(codec, CS420X_VENDOR_NID);
 593	if (!spec)
 594		return -ENOMEM;
 
 595
 596	codec->patch_ops = cs_patch_ops;
 597	spec->gen.automute_hook = cs_automute;
 598	codec->single_adc_amp = 1;
 599
 600	snd_hda_pick_fixup(codec, cs420x_models, cs420x_fixup_tbl,
 601			   cs420x_fixups);
 602	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
 
 
 
 
 
 
 
 
 
 
 
 
 
 603
 604	err = cs_parse_auto_config(codec);
 605	if (err < 0)
 606		goto error;
 607
 608	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
 609
 610	return 0;
 611
 612 error:
 613	cs_free(codec);
 614	return err;
 615}
 616
 617/*
 618 * CS4208 support:
 619 * Its layout is no longer compatible with CS4206/CS4207
 620 */
 621enum {
 622	CS4208_MAC_AUTO,
 623	CS4208_MBA6,
 624	CS4208_MBP11,
 625	CS4208_MACMINI,
 626	CS4208_GPIO0,
 627};
 628
 629static const struct hda_model_fixup cs4208_models[] = {
 630	{ .id = CS4208_GPIO0, .name = "gpio0" },
 631	{ .id = CS4208_MBA6, .name = "mba6" },
 632	{ .id = CS4208_MBP11, .name = "mbp11" },
 633	{ .id = CS4208_MACMINI, .name = "macmini" },
 634	{}
 635};
 636
 637static const struct snd_pci_quirk cs4208_fixup_tbl[] = {
 638	SND_PCI_QUIRK_VENDOR(0x106b, "Apple", CS4208_MAC_AUTO),
 639	{} /* terminator */
 640};
 641
 642/* codec SSID matching */
 643static const struct snd_pci_quirk cs4208_mac_fixup_tbl[] = {
 644	SND_PCI_QUIRK(0x106b, 0x5e00, "MacBookPro 11,2", CS4208_MBP11),
 645	SND_PCI_QUIRK(0x106b, 0x6c00, "MacMini 7,1", CS4208_MACMINI),
 646	SND_PCI_QUIRK(0x106b, 0x7100, "MacBookAir 6,1", CS4208_MBA6),
 647	SND_PCI_QUIRK(0x106b, 0x7200, "MacBookAir 6,2", CS4208_MBA6),
 648	SND_PCI_QUIRK(0x106b, 0x7b00, "MacBookPro 12,1", CS4208_MBP11),
 649	{} /* terminator */
 650};
 651
 652static void cs4208_fixup_gpio0(struct hda_codec *codec,
 653			       const struct hda_fixup *fix, int action)
 654{
 655	if (action == HDA_FIXUP_ACT_PRE_PROBE) {
 656		struct cs_spec *spec = codec->spec;
 657
 658		spec->gpio_eapd_hp = 0;
 659		spec->gpio_eapd_speaker = 1;
 660		spec->gpio_mask = spec->gpio_dir =
 661			spec->gpio_eapd_hp | spec->gpio_eapd_speaker;
 662	}
 663}
 664
 665static const struct hda_fixup cs4208_fixups[];
 666
 667/* remap the fixup from codec SSID and apply it */
 668static void cs4208_fixup_mac(struct hda_codec *codec,
 669			     const struct hda_fixup *fix, int action)
 670{
 671	if (action != HDA_FIXUP_ACT_PRE_PROBE)
 672		return;
 673
 674	codec->fixup_id = HDA_FIXUP_ID_NOT_SET;
 675	snd_hda_pick_fixup(codec, NULL, cs4208_mac_fixup_tbl, cs4208_fixups);
 676	if (codec->fixup_id == HDA_FIXUP_ID_NOT_SET)
 677		codec->fixup_id = CS4208_GPIO0; /* default fixup */
 678	snd_hda_apply_fixup(codec, action);
 679}
 680
 681/* MacMini 7,1 has the inverted jack detection */
 682static void cs4208_fixup_macmini(struct hda_codec *codec,
 683				 const struct hda_fixup *fix, int action)
 684{
 685	static const struct hda_pintbl pincfgs[] = {
 686		{ 0x18, 0x00ab9150 }, /* mic (audio-in) jack: disable detect */
 687		{ 0x21, 0x004be140 }, /* SPDIF: disable detect */
 688		{ }
 689	};
 690
 691	if (action == HDA_FIXUP_ACT_PRE_PROBE) {
 692		/* HP pin (0x10) has an inverted detection */
 693		codec->inv_jack_detect = 1;
 694		/* disable the bogus Mic and SPDIF jack detections */
 695		snd_hda_apply_pincfgs(codec, pincfgs);
 696	}
 697}
 698
 699static int cs4208_spdif_sw_put(struct snd_kcontrol *kcontrol,
 700			       struct snd_ctl_elem_value *ucontrol)
 701{
 702	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 703	struct cs_spec *spec = codec->spec;
 704	hda_nid_t pin = spec->gen.autocfg.dig_out_pins[0];
 705	int pinctl = ucontrol->value.integer.value[0] ? PIN_OUT : 0;
 706
 707	snd_hda_set_pin_ctl_cache(codec, pin, pinctl);
 708	return spec->spdif_sw_put(kcontrol, ucontrol);
 709}
 710
 711/* hook the SPDIF switch */
 712static void cs4208_fixup_spdif_switch(struct hda_codec *codec,
 713				      const struct hda_fixup *fix, int action)
 714{
 715	if (action == HDA_FIXUP_ACT_BUILD) {
 716		struct cs_spec *spec = codec->spec;
 717		struct snd_kcontrol *kctl;
 718
 719		if (!spec->gen.autocfg.dig_out_pins[0])
 720			return;
 721		kctl = snd_hda_find_mixer_ctl(codec, "IEC958 Playback Switch");
 722		if (!kctl)
 723			return;
 724		spec->spdif_sw_put = kctl->put;
 725		kctl->put = cs4208_spdif_sw_put;
 726	}
 727}
 728
 729static const struct hda_fixup cs4208_fixups[] = {
 730	[CS4208_MBA6] = {
 731		.type = HDA_FIXUP_PINS,
 732		.v.pins = mba6_pincfgs,
 733		.chained = true,
 734		.chain_id = CS4208_GPIO0,
 735	},
 736	[CS4208_MBP11] = {
 737		.type = HDA_FIXUP_FUNC,
 738		.v.func = cs4208_fixup_spdif_switch,
 739		.chained = true,
 740		.chain_id = CS4208_GPIO0,
 741	},
 742	[CS4208_MACMINI] = {
 743		.type = HDA_FIXUP_FUNC,
 744		.v.func = cs4208_fixup_macmini,
 745		.chained = true,
 746		.chain_id = CS4208_GPIO0,
 747	},
 748	[CS4208_GPIO0] = {
 749		.type = HDA_FIXUP_FUNC,
 750		.v.func = cs4208_fixup_gpio0,
 751	},
 752	[CS4208_MAC_AUTO] = {
 753		.type = HDA_FIXUP_FUNC,
 754		.v.func = cs4208_fixup_mac,
 755	},
 756};
 757
 758/* correct the 0dB offset of input pins */
 759static void cs4208_fix_amp_caps(struct hda_codec *codec, hda_nid_t adc)
 760{
 761	unsigned int caps;
 762
 763	caps = query_amp_caps(codec, adc, HDA_INPUT);
 764	caps &= ~(AC_AMPCAP_OFFSET);
 765	caps |= 0x02;
 766	snd_hda_override_amp_caps(codec, adc, HDA_INPUT, caps);
 767}
 768
 769static int patch_cs4208(struct hda_codec *codec)
 770{
 771	struct cs_spec *spec;
 772	int err;
 773
 774	spec = cs_alloc_spec(codec, CS4208_VENDOR_NID);
 775	if (!spec)
 776		return -ENOMEM;
 777
 778	codec->patch_ops = cs_patch_ops;
 779	spec->gen.automute_hook = cs_automute;
 780	/* exclude NID 0x10 (HP) from output volumes due to different steps */
 781	spec->gen.out_vol_mask = 1ULL << 0x10;
 782
 783	snd_hda_pick_fixup(codec, cs4208_models, cs4208_fixup_tbl,
 784			   cs4208_fixups);
 785	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
 786
 787	snd_hda_override_wcaps(codec, 0x18,
 788			       get_wcaps(codec, 0x18) | AC_WCAP_STEREO);
 789	cs4208_fix_amp_caps(codec, 0x18);
 790	cs4208_fix_amp_caps(codec, 0x1b);
 791	cs4208_fix_amp_caps(codec, 0x1c);
 792
 793	err = cs_parse_auto_config(codec);
 794	if (err < 0)
 795		goto error;
 796
 797	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
 798
 799	return 0;
 800
 801 error:
 802	cs_free(codec);
 
 803	return err;
 804}
 805
 806/*
 807 * Cirrus Logic CS4210
 808 *
 809 * 1 DAC => HP(sense) / Speakers,
 810 * 1 ADC <= LineIn(sense) / MicIn / DMicIn,
 811 * 1 SPDIF OUT => SPDIF Trasmitter(sense)
 812 */
 813
 814/* CS4210 board names */
 815static const struct hda_model_fixup cs421x_models[] = {
 816	{ .id = CS421X_CDB4210, .name = "cdb4210" },
 817	{ .id = CS421X_STUMPY, .name = "stumpy" },
 818	{}
 819};
 820
 821static const struct snd_pci_quirk cs421x_fixup_tbl[] = {
 822	/* Test Intel board + CDB2410  */
 823	SND_PCI_QUIRK(0x8086, 0x5001, "DP45SG/CDB4210", CS421X_CDB4210),
 824	{} /* terminator */
 825};
 826
 827/* CS4210 board pinconfigs */
 828/* Default CS4210 (CDB4210)*/
 829static const struct hda_pintbl cdb4210_pincfgs[] = {
 830	{ 0x05, 0x0321401f },
 831	{ 0x06, 0x90170010 },
 832	{ 0x07, 0x03813031 },
 833	{ 0x08, 0xb7a70037 },
 834	{ 0x09, 0xb7a6003e },
 835	{ 0x0a, 0x034510f0 },
 836	{} /* terminator */
 837};
 838
 839/* Stumpy ChromeBox */
 840static const struct hda_pintbl stumpy_pincfgs[] = {
 841	{ 0x05, 0x022120f0 },
 842	{ 0x06, 0x901700f0 },
 843	{ 0x07, 0x02a120f0 },
 844	{ 0x08, 0x77a70037 },
 845	{ 0x09, 0x77a6003e },
 846	{ 0x0a, 0x434510f0 },
 847	{} /* terminator */
 848};
 849
 850/* Setup GPIO/SENSE for each board (if used) */
 851static void cs421x_fixup_sense_b(struct hda_codec *codec,
 852				 const struct hda_fixup *fix, int action)
 853{
 854	struct cs_spec *spec = codec->spec;
 855
 856	if (action == HDA_FIXUP_ACT_PRE_PROBE)
 857		spec->sense_b = 1;
 858}
 859
 860static const struct hda_fixup cs421x_fixups[] = {
 861	[CS421X_CDB4210] = {
 862		.type = HDA_FIXUP_PINS,
 863		.v.pins = cdb4210_pincfgs,
 864		.chained = true,
 865		.chain_id = CS421X_SENSE_B,
 866	},
 867	[CS421X_SENSE_B] = {
 868		.type = HDA_FIXUP_FUNC,
 869		.v.func = cs421x_fixup_sense_b,
 870	},
 871	[CS421X_STUMPY] = {
 872		.type = HDA_FIXUP_PINS,
 873		.v.pins = stumpy_pincfgs,
 874	},
 875};
 876
 877static const struct hda_verb cs421x_coef_init_verbs[] = {
 878	{0x0B, AC_VERB_SET_PROC_STATE, 1},
 879	{0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_DEV_CFG},
 880	/*
 881	 *  Disable Coefficient Index Auto-Increment(DAI)=1,
 882	 *  PDREF=0
 883	 */
 884	{0x0B, AC_VERB_SET_PROC_COEF, 0x0001 },
 885
 886	{0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_ADC_CFG},
 887	/* ADC SZCMode = Digital Soft Ramp */
 888	{0x0B, AC_VERB_SET_PROC_COEF, 0x0002 },
 889
 890	{0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_DAC_CFG},
 891	{0x0B, AC_VERB_SET_PROC_COEF,
 892	 (0x0002 /* DAC SZCMode = Digital Soft Ramp */
 893	  | 0x0004 /* Mute DAC on FIFO error */
 894	  | 0x0008 /* Enable DAC High Pass Filter */
 895	  )},
 896	{} /* terminator */
 897};
 898
 899/* Errata: CS4210 rev A1 Silicon
 900 *
 901 * http://www.cirrus.com/en/pubs/errata/
 902 *
 903 * Description:
 904 * 1. Performance degredation is present in the ADC.
 905 * 2. Speaker output is not completely muted upon HP detect.
 906 * 3. Noise is present when clipping occurs on the amplified
 907 *    speaker outputs.
 908 *
 909 * Workaround:
 910 * The following verb sequence written to the registers during
 911 * initialization will correct the issues listed above.
 912 */
 913
 914static const struct hda_verb cs421x_coef_init_verbs_A1_silicon_fixes[] = {
 915	{0x0B, AC_VERB_SET_PROC_STATE, 0x01},  /* VPW: processing on */
 916
 917	{0x0B, AC_VERB_SET_COEF_INDEX, 0x0006},
 918	{0x0B, AC_VERB_SET_PROC_COEF, 0x9999}, /* Test mode: on */
 919
 920	{0x0B, AC_VERB_SET_COEF_INDEX, 0x000A},
 921	{0x0B, AC_VERB_SET_PROC_COEF, 0x14CB}, /* Chop double */
 922
 923	{0x0B, AC_VERB_SET_COEF_INDEX, 0x0011},
 924	{0x0B, AC_VERB_SET_PROC_COEF, 0xA2D0}, /* Increase ADC current */
 925
 926	{0x0B, AC_VERB_SET_COEF_INDEX, 0x001A},
 927	{0x0B, AC_VERB_SET_PROC_COEF, 0x02A9}, /* Mute speaker */
 928
 929	{0x0B, AC_VERB_SET_COEF_INDEX, 0x001B},
 930	{0x0B, AC_VERB_SET_PROC_COEF, 0X1006}, /* Remove noise */
 931
 932	{} /* terminator */
 933};
 934
 935/* Speaker Amp Gain is controlled by the vendor widget's coef 4 */
 936static const DECLARE_TLV_DB_SCALE(cs421x_speaker_boost_db_scale, 900, 300, 0);
 937
 938static int cs421x_boost_vol_info(struct snd_kcontrol *kcontrol,
 939				struct snd_ctl_elem_info *uinfo)
 940{
 941	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
 942	uinfo->count = 1;
 943	uinfo->value.integer.min = 0;
 944	uinfo->value.integer.max = 3;
 945	return 0;
 946}
 947
 948static int cs421x_boost_vol_get(struct snd_kcontrol *kcontrol,
 949				struct snd_ctl_elem_value *ucontrol)
 950{
 951	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 952
 953	ucontrol->value.integer.value[0] =
 954		cs_vendor_coef_get(codec, CS421X_IDX_SPK_CTL) & 0x0003;
 955	return 0;
 956}
 957
 958static int cs421x_boost_vol_put(struct snd_kcontrol *kcontrol,
 959				struct snd_ctl_elem_value *ucontrol)
 960{
 961	struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
 962
 963	unsigned int vol = ucontrol->value.integer.value[0];
 964	unsigned int coef =
 965		cs_vendor_coef_get(codec, CS421X_IDX_SPK_CTL);
 966	unsigned int original_coef = coef;
 967
 968	coef &= ~0x0003;
 969	coef |= (vol & 0x0003);
 970	if (original_coef != coef) {
 
 
 971		cs_vendor_coef_set(codec, CS421X_IDX_SPK_CTL, coef);
 972		return 1;
 973	}
 974
 975	return 0;
 976}
 977
 978static const struct snd_kcontrol_new cs421x_speaker_boost_ctl = {
 979
 980	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
 981	.access = (SNDRV_CTL_ELEM_ACCESS_READWRITE |
 982			SNDRV_CTL_ELEM_ACCESS_TLV_READ),
 983	.name = "Speaker Boost Playback Volume",
 984	.info = cs421x_boost_vol_info,
 985	.get = cs421x_boost_vol_get,
 986	.put = cs421x_boost_vol_put,
 987	.tlv = { .p = cs421x_speaker_boost_db_scale },
 988};
 989
 990static void cs4210_pinmux_init(struct hda_codec *codec)
 991{
 992	struct cs_spec *spec = codec->spec;
 993	unsigned int def_conf, coef;
 994
 995	/* GPIO, DMIC_SCL, DMIC_SDA and SENSE_B are multiplexed */
 996	coef = cs_vendor_coef_get(codec, CS421X_IDX_DEV_CFG);
 997
 998	if (spec->gpio_mask)
 999		coef |= 0x0008; /* B1,B2 are GPIOs */
1000	else
1001		coef &= ~0x0008;
1002
1003	if (spec->sense_b)
1004		coef |= 0x0010; /* B2 is SENSE_B, not inverted  */
1005	else
1006		coef &= ~0x0010;
1007
1008	cs_vendor_coef_set(codec, CS421X_IDX_DEV_CFG, coef);
1009
1010	if ((spec->gpio_mask || spec->sense_b) &&
1011	    is_active_pin(codec, CS421X_DMIC_PIN_NID)) {
1012
1013		/*
1014		 *  GPIO or SENSE_B forced - disconnect the DMIC pin.
1015		 */
1016		def_conf = snd_hda_codec_get_pincfg(codec, CS421X_DMIC_PIN_NID);
1017		def_conf &= ~AC_DEFCFG_PORT_CONN;
1018		def_conf |= (AC_JACK_PORT_NONE << AC_DEFCFG_PORT_CONN_SHIFT);
1019		snd_hda_codec_set_pincfg(codec, CS421X_DMIC_PIN_NID, def_conf);
1020	}
1021}
1022
1023static void cs4210_spdif_automute(struct hda_codec *codec,
1024				  struct hda_jack_callback *tbl)
1025{
1026	struct cs_spec *spec = codec->spec;
1027	bool spdif_present = false;
1028	hda_nid_t spdif_pin = spec->gen.autocfg.dig_out_pins[0];
1029
1030	/* detect on spdif is specific to CS4210 */
1031	if (!spec->spdif_detect ||
1032	    spec->vendor_nid != CS4210_VENDOR_NID)
1033		return;
1034
1035	spdif_present = snd_hda_jack_detect(codec, spdif_pin);
1036	if (spdif_present == spec->spdif_present)
1037		return;
1038
1039	spec->spdif_present = spdif_present;
1040	/* SPDIF TX on/off */
1041	snd_hda_set_pin_ctl(codec, spdif_pin, spdif_present ? PIN_OUT : 0);
1042
1043	cs_automute(codec);
1044}
1045
1046static void parse_cs421x_digital(struct hda_codec *codec)
1047{
1048	struct cs_spec *spec = codec->spec;
1049	struct auto_pin_cfg *cfg = &spec->gen.autocfg;
1050	int i;
1051
1052	for (i = 0; i < cfg->dig_outs; i++) {
1053		hda_nid_t nid = cfg->dig_out_pins[i];
 
 
 
1054
1055		if (get_wcaps(codec, nid) & AC_WCAP_UNSOL_CAP) {
 
 
1056			spec->spdif_detect = 1;
1057			snd_hda_jack_detect_enable_callback(codec, nid,
1058							    cs4210_spdif_automute);
1059		}
1060	}
1061}
1062
1063static int cs421x_init(struct hda_codec *codec)
1064{
1065	struct cs_spec *spec = codec->spec;
1066
1067	if (spec->vendor_nid == CS4210_VENDOR_NID) {
1068		snd_hda_sequence_write(codec, cs421x_coef_init_verbs);
1069		snd_hda_sequence_write(codec, cs421x_coef_init_verbs_A1_silicon_fixes);
1070		cs4210_pinmux_init(codec);
1071	}
1072
1073	snd_hda_gen_init(codec);
1074
1075	if (spec->gpio_mask) {
1076		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_MASK,
1077				    spec->gpio_mask);
1078		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DIRECTION,
1079				    spec->gpio_dir);
1080		snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA,
1081				    spec->gpio_data);
1082	}
1083
1084	init_input_coef(codec);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1085
1086	cs4210_spdif_automute(codec, NULL);
 
1087
 
 
 
 
 
 
 
1088	return 0;
1089}
1090
1091static void fix_volume_caps(struct hda_codec *codec, hda_nid_t dac)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1092{
1093	unsigned int caps;
 
 
 
 
 
 
 
 
 
 
 
 
 
1094
1095	/* set the upper-limit for mixer amp to 0dB */
1096	caps = query_amp_caps(codec, dac, HDA_OUTPUT);
1097	caps &= ~(0x7f << AC_AMPCAP_NUM_STEPS_SHIFT);
1098	caps |= ((caps >> AC_AMPCAP_OFFSET_SHIFT) & 0x7f)
1099		<< AC_AMPCAP_NUM_STEPS_SHIFT;
1100	snd_hda_override_amp_caps(codec, dac, HDA_OUTPUT, caps);
1101}
1102
1103static int cs421x_parse_auto_config(struct hda_codec *codec)
 
1104{
1105	struct cs_spec *spec = codec->spec;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1106	hda_nid_t dac = CS4210_DAC_NID;
 
 
 
1107	int err;
 
1108
1109	fix_volume_caps(codec, dac);
 
 
 
 
 
1110
1111	err = snd_hda_parse_pin_defcfg(codec, &spec->gen.autocfg, NULL, 0);
 
 
 
 
1112	if (err < 0)
1113		return err;
1114
1115	err = snd_hda_gen_parse_auto_config(codec, &spec->gen.autocfg);
 
 
 
 
1116	if (err < 0)
1117		return err;
1118
1119	parse_cs421x_digital(codec);
 
 
 
 
 
 
 
1120
1121	if (spec->gen.autocfg.speaker_outs &&
1122	    spec->vendor_nid == CS4210_VENDOR_NID) {
1123		if (!snd_hda_gen_add_kctl(&spec->gen, NULL,
1124					  &cs421x_speaker_boost_ctl))
1125			return -ENOMEM;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1126	}
 
 
 
 
 
 
 
1127
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1128	return 0;
1129}
1130
1131#ifdef CONFIG_PM
1132/*
1133 *	Manage PDREF, when transitioning to D3hot
1134 *	(DAC,ADC) -> D3, PDREF=1, AFG->D3
1135 */
1136static int cs421x_suspend(struct hda_codec *codec)
1137{
1138	struct cs_spec *spec = codec->spec;
1139	unsigned int coef;
1140
1141	snd_hda_shutup_pins(codec);
1142
1143	snd_hda_codec_write(codec, CS4210_DAC_NID, 0,
1144			    AC_VERB_SET_POWER_STATE,  AC_PWRST_D3);
1145	snd_hda_codec_write(codec, CS4210_ADC_NID, 0,
1146			    AC_VERB_SET_POWER_STATE,  AC_PWRST_D3);
1147
1148	if (spec->vendor_nid == CS4210_VENDOR_NID) {
1149		coef = cs_vendor_coef_get(codec, CS421X_IDX_DEV_CFG);
1150		coef |= 0x0004; /* PDREF */
1151		cs_vendor_coef_set(codec, CS421X_IDX_DEV_CFG, coef);
1152	}
1153
1154	return 0;
1155}
1156#endif
1157
1158static const struct hda_codec_ops cs421x_patch_ops = {
1159	.build_controls = snd_hda_gen_build_controls,
1160	.build_pcms = snd_hda_gen_build_pcms,
1161	.init = cs421x_init,
1162	.free = cs_free,
1163	.unsol_event = snd_hda_jack_unsol_event,
1164#ifdef CONFIG_PM
1165	.suspend = cs421x_suspend,
1166#endif
1167};
1168
1169static int patch_cs4210(struct hda_codec *codec)
1170{
1171	struct cs_spec *spec;
1172	int err;
1173
1174	spec = cs_alloc_spec(codec, CS4210_VENDOR_NID);
1175	if (!spec)
1176		return -ENOMEM;
 
 
 
1177
1178	codec->patch_ops = cs421x_patch_ops;
1179	spec->gen.automute_hook = cs_automute;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1180
1181	snd_hda_pick_fixup(codec, cs421x_models, cs421x_fixup_tbl,
1182			   cs421x_fixups);
1183	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);
1184
1185	/*
1186	 *  Update the GPIO/DMIC/SENSE_B pinmux before the configuration
1187	 *   is auto-parsed. If GPIO or SENSE_B is forced, DMIC input
1188	 *   is disabled.
1189	 */
1190	cs4210_pinmux_init(codec);
1191
1192	err = cs421x_parse_auto_config(codec);
1193	if (err < 0)
1194		goto error;
1195
1196	snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);
1197
1198	return 0;
1199
1200 error:
1201	cs_free(codec);
 
1202	return err;
1203}
1204
1205static int patch_cs4213(struct hda_codec *codec)
1206{
1207	struct cs_spec *spec;
1208	int err;
1209
1210	spec = cs_alloc_spec(codec, CS4213_VENDOR_NID);
1211	if (!spec)
1212		return -ENOMEM;
1213
1214	codec->patch_ops = cs421x_patch_ops;
1215
1216	err = cs421x_parse_auto_config(codec);
1217	if (err < 0)
1218		goto error;
1219
1220	return 0;
1221
1222 error:
1223	cs_free(codec);
1224	return err;
1225}
1226
1227/*
1228 * patch entries
1229 */
1230static const struct hda_device_id snd_hda_id_cirrus[] = {
1231	HDA_CODEC_ENTRY(0x10134206, "CS4206", patch_cs420x),
1232	HDA_CODEC_ENTRY(0x10134207, "CS4207", patch_cs420x),
1233	HDA_CODEC_ENTRY(0x10134208, "CS4208", patch_cs4208),
1234	HDA_CODEC_ENTRY(0x10134210, "CS4210", patch_cs4210),
1235	HDA_CODEC_ENTRY(0x10134213, "CS4213", patch_cs4213),
1236	{} /* terminator */
1237};
1238MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_cirrus);
 
 
 
1239
1240MODULE_LICENSE("GPL");
1241MODULE_DESCRIPTION("Cirrus Logic HD-audio codec");
1242
1243static struct hda_codec_driver cirrus_driver = {
1244	.id = snd_hda_id_cirrus,
 
1245};
1246
1247module_hda_codec_driver(cirrus_driver);