2005-04-13 05:45:30 -07:00
|
|
|
/*
|
|
|
|
* Universal Interface for Intel High Definition Audio Codec
|
|
|
|
*
|
|
|
|
* HD audio interface patch for SigmaTel STAC92xx
|
|
|
|
*
|
|
|
|
* Copyright (c) 2005 Embedded Alley Solutions, Inc.
|
2005-11-29 07:00:51 -07:00
|
|
|
* Matt Porter <mporter@embeddedalley.com>
|
2005-04-13 05:45:30 -07:00
|
|
|
*
|
|
|
|
* Based on patch_cmedia.c and patch_realtek.c
|
|
|
|
* Copyright (c) 2004 Takashi Iwai <tiwai@suse.de>
|
|
|
|
*
|
|
|
|
* This driver is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This driver is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sound/driver.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/pci.h>
|
|
|
|
#include <sound/core.h>
|
2005-06-27 05:59:41 -07:00
|
|
|
#include <sound/asoundef.h>
|
2005-04-13 05:45:30 -07:00
|
|
|
#include "hda_codec.h"
|
|
|
|
#include "hda_local.h"
|
|
|
|
|
2005-07-04 08:51:39 -07:00
|
|
|
#define NUM_CONTROL_ALLOC 32
|
|
|
|
#define STAC_HP_EVENT 0x37
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
enum {
|
|
|
|
STAC_REF,
|
2007-08-23 10:04:28 -07:00
|
|
|
STAC_9200_DELL_D21,
|
|
|
|
STAC_9200_DELL_D22,
|
|
|
|
STAC_9200_DELL_D23,
|
|
|
|
STAC_9200_DELL_M21,
|
|
|
|
STAC_9200_DELL_M22,
|
|
|
|
STAC_9200_DELL_M23,
|
|
|
|
STAC_9200_DELL_M24,
|
|
|
|
STAC_9200_DELL_M25,
|
|
|
|
STAC_9200_DELL_M26,
|
|
|
|
STAC_9200_DELL_M27,
|
2007-10-10 01:04:26 -07:00
|
|
|
STAC_9200_GATEWAY,
|
2006-11-24 09:07:44 -07:00
|
|
|
STAC_9200_MODELS
|
|
|
|
};
|
|
|
|
|
|
|
|
enum {
|
|
|
|
STAC_9205_REF,
|
2007-08-23 10:04:28 -07:00
|
|
|
STAC_9205_DELL_M42,
|
2007-08-13 06:50:29 -07:00
|
|
|
STAC_9205_DELL_M43,
|
|
|
|
STAC_9205_DELL_M44,
|
2006-11-24 09:07:44 -07:00
|
|
|
STAC_9205_MODELS
|
|
|
|
};
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
enum {
|
|
|
|
STAC_92HD71BXX_REF,
|
|
|
|
STAC_92HD71BXX_MODELS
|
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
enum {
|
|
|
|
STAC_925x_REF,
|
|
|
|
STAC_M2_2,
|
|
|
|
STAC_MA6,
|
2007-05-17 00:36:34 -07:00
|
|
|
STAC_PA6,
|
2007-01-08 03:04:17 -07:00
|
|
|
STAC_925x_MODELS
|
|
|
|
};
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
enum {
|
|
|
|
STAC_D945_REF,
|
|
|
|
STAC_D945GTP3,
|
|
|
|
STAC_D945GTP5,
|
2007-05-29 07:03:00 -07:00
|
|
|
STAC_INTEL_MAC_V1,
|
|
|
|
STAC_INTEL_MAC_V2,
|
|
|
|
STAC_INTEL_MAC_V3,
|
|
|
|
STAC_INTEL_MAC_V4,
|
|
|
|
STAC_INTEL_MAC_V5,
|
2007-08-23 10:04:28 -07:00
|
|
|
/* for backward compatibility */
|
2006-11-24 09:07:44 -07:00
|
|
|
STAC_MACMINI,
|
2007-02-16 05:27:18 -07:00
|
|
|
STAC_MACBOOK,
|
2007-03-15 04:38:15 -07:00
|
|
|
STAC_MACBOOK_PRO_V1,
|
|
|
|
STAC_MACBOOK_PRO_V2,
|
2007-04-27 05:22:36 -07:00
|
|
|
STAC_IMAC_INTEL,
|
2007-05-21 03:41:29 -07:00
|
|
|
STAC_IMAC_INTEL_20,
|
2007-08-23 10:04:28 -07:00
|
|
|
STAC_922X_DELL_D81,
|
|
|
|
STAC_922X_DELL_D82,
|
|
|
|
STAC_922X_DELL_M81,
|
|
|
|
STAC_922X_DELL_M82,
|
2006-11-24 09:07:44 -07:00
|
|
|
STAC_922X_MODELS
|
|
|
|
};
|
|
|
|
|
|
|
|
enum {
|
|
|
|
STAC_D965_REF,
|
|
|
|
STAC_D965_3ST,
|
|
|
|
STAC_D965_5ST,
|
2007-08-07 02:48:12 -07:00
|
|
|
STAC_DELL_3ST,
|
2006-11-24 09:07:44 -07:00
|
|
|
STAC_927X_MODELS
|
|
|
|
};
|
2005-11-29 07:00:51 -07:00
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
struct sigmatel_spec {
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_kcontrol_new *mixers[4];
|
2005-06-27 05:59:41 -07:00
|
|
|
unsigned int num_mixers;
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
int board_config;
|
2005-06-27 05:59:41 -07:00
|
|
|
unsigned int surr_switch: 1;
|
2005-11-29 07:00:51 -07:00
|
|
|
unsigned int line_switch: 1;
|
|
|
|
unsigned int mic_switch: 1;
|
2006-01-23 07:27:49 -07:00
|
|
|
unsigned int alt_switch: 1;
|
2006-03-21 03:24:42 -07:00
|
|
|
unsigned int hp_detect: 1;
|
2006-05-10 06:09:17 -07:00
|
|
|
unsigned int gpio_mute: 1;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
2007-07-31 06:56:24 -07:00
|
|
|
unsigned int gpio_mask, gpio_data;
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
/* playback */
|
|
|
|
struct hda_multi_out multiout;
|
2006-01-23 07:27:49 -07:00
|
|
|
hda_nid_t dac_nids[5];
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
/* capture */
|
|
|
|
hda_nid_t *adc_nids;
|
|
|
|
unsigned int num_adcs;
|
2005-06-14 01:19:34 -07:00
|
|
|
hda_nid_t *mux_nids;
|
|
|
|
unsigned int num_muxes;
|
2006-10-26 08:12:59 -07:00
|
|
|
hda_nid_t *dmic_nids;
|
|
|
|
unsigned int num_dmics;
|
|
|
|
hda_nid_t dmux_nid;
|
2005-06-14 01:19:34 -07:00
|
|
|
hda_nid_t dig_in_nid;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
/* pin widgets */
|
|
|
|
hda_nid_t *pin_nids;
|
|
|
|
unsigned int num_pins;
|
|
|
|
unsigned int *pin_configs;
|
2006-08-23 09:31:34 -07:00
|
|
|
unsigned int *bios_pin_configs;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
/* codec specific stuff */
|
|
|
|
struct hda_verb *init;
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_kcontrol_new *mixer;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
/* capture source */
|
2006-10-26 08:12:59 -07:00
|
|
|
struct hda_input_mux *dinput_mux;
|
|
|
|
unsigned int cur_dmux;
|
2005-06-27 05:59:41 -07:00
|
|
|
struct hda_input_mux *input_mux;
|
2006-01-23 07:27:49 -07:00
|
|
|
unsigned int cur_mux[3];
|
2005-04-13 05:45:30 -07:00
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
/* i/o switches */
|
|
|
|
unsigned int io_switch[2];
|
2007-09-03 06:29:04 -07:00
|
|
|
unsigned int clfe_swap;
|
2007-09-03 06:29:37 -07:00
|
|
|
unsigned int aloopback;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
struct hda_pcm pcm_rec[2]; /* PCM information */
|
|
|
|
|
|
|
|
/* dynamic controls and input_mux */
|
|
|
|
struct auto_pin_cfg autocfg;
|
|
|
|
unsigned int num_kctl_alloc, num_kctl_used;
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_kcontrol_new *kctl_alloc;
|
2006-10-26 08:12:59 -07:00
|
|
|
struct hda_input_mux private_dimux;
|
2005-06-27 05:59:41 -07:00
|
|
|
struct hda_input_mux private_imux;
|
2005-04-13 05:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac9200_adc_nids[1] = {
|
|
|
|
0x03,
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac9200_mux_nids[1] = {
|
|
|
|
0x0c,
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac9200_dac_nids[1] = {
|
|
|
|
0x02,
|
|
|
|
};
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
static hda_nid_t stac92hd71bxx_adc_nids[2] = {
|
|
|
|
0x12, 0x13,
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac92hd71bxx_mux_nids[2] = {
|
|
|
|
0x1a, 0x1b
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac92hd71bxx_dac_nids[2] = {
|
|
|
|
0x10, /*0x11, */
|
|
|
|
};
|
|
|
|
|
|
|
|
#define STAC92HD71BXX_NUM_DMICS 2
|
|
|
|
static hda_nid_t stac92hd71bxx_dmic_nids[STAC92HD71BXX_NUM_DMICS + 1] = {
|
|
|
|
0x18, 0x19, 0
|
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static hda_nid_t stac925x_adc_nids[1] = {
|
|
|
|
0x03,
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac925x_mux_nids[1] = {
|
|
|
|
0x0f,
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac925x_dac_nids[1] = {
|
|
|
|
0x02,
|
|
|
|
};
|
|
|
|
|
2007-10-16 05:27:04 -07:00
|
|
|
#define STAC925X_NUM_DMICS 1
|
|
|
|
static hda_nid_t stac925x_dmic_nids[STAC925X_NUM_DMICS + 1] = {
|
|
|
|
0x15, 0
|
2007-05-17 00:36:34 -07:00
|
|
|
};
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static hda_nid_t stac922x_adc_nids[2] = {
|
|
|
|
0x06, 0x07,
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac922x_mux_nids[2] = {
|
|
|
|
0x12, 0x13,
|
|
|
|
};
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
static hda_nid_t stac927x_adc_nids[3] = {
|
|
|
|
0x07, 0x08, 0x09
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac927x_mux_nids[3] = {
|
|
|
|
0x15, 0x16, 0x17
|
|
|
|
};
|
|
|
|
|
2007-10-18 08:38:17 -07:00
|
|
|
#define STAC927X_NUM_DMICS 2
|
|
|
|
static hda_nid_t stac927x_dmic_nids[STAC927X_NUM_DMICS + 1] = {
|
|
|
|
0x13, 0x14, 0
|
|
|
|
};
|
|
|
|
|
2006-07-31 03:49:34 -07:00
|
|
|
static hda_nid_t stac9205_adc_nids[2] = {
|
|
|
|
0x12, 0x13
|
|
|
|
};
|
|
|
|
|
|
|
|
static hda_nid_t stac9205_mux_nids[2] = {
|
|
|
|
0x19, 0x1a
|
|
|
|
};
|
|
|
|
|
2007-10-16 05:27:04 -07:00
|
|
|
#define STAC9205_NUM_DMICS 2
|
|
|
|
static hda_nid_t stac9205_dmic_nids[STAC9205_NUM_DMICS + 1] = {
|
|
|
|
0x17, 0x18, 0
|
2006-10-26 08:12:59 -07:00
|
|
|
};
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
static hda_nid_t stac9200_pin_nids[8] = {
|
2006-09-01 12:03:12 -07:00
|
|
|
0x08, 0x09, 0x0d, 0x0e,
|
|
|
|
0x0f, 0x10, 0x11, 0x12,
|
2005-04-13 05:45:30 -07:00
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static hda_nid_t stac925x_pin_nids[8] = {
|
|
|
|
0x07, 0x08, 0x0a, 0x0b,
|
|
|
|
0x0c, 0x0d, 0x10, 0x11,
|
|
|
|
};
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static hda_nid_t stac922x_pin_nids[10] = {
|
|
|
|
0x0a, 0x0b, 0x0c, 0x0d, 0x0e,
|
|
|
|
0x0f, 0x10, 0x11, 0x15, 0x1b,
|
|
|
|
};
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
static hda_nid_t stac92hd71bxx_pin_nids[10] = {
|
|
|
|
0x0a, 0x0b, 0x0c, 0x0d, 0x0e,
|
|
|
|
0x0f, 0x14, 0x18, 0x19, 0x1e,
|
|
|
|
};
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
static hda_nid_t stac927x_pin_nids[14] = {
|
|
|
|
0x0a, 0x0b, 0x0c, 0x0d, 0x0e,
|
|
|
|
0x0f, 0x10, 0x11, 0x12, 0x13,
|
|
|
|
0x14, 0x21, 0x22, 0x23,
|
|
|
|
};
|
|
|
|
|
2006-07-31 03:49:34 -07:00
|
|
|
static hda_nid_t stac9205_pin_nids[12] = {
|
|
|
|
0x0a, 0x0b, 0x0c, 0x0d, 0x0e,
|
|
|
|
0x0f, 0x14, 0x16, 0x17, 0x18,
|
|
|
|
0x21, 0x22,
|
|
|
|
};
|
|
|
|
|
2006-10-26 08:12:59 -07:00
|
|
|
static int stac92xx_dmux_enum_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
return snd_hda_input_mux_info(spec->dinput_mux, uinfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_dmux_enum_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
ucontrol->value.enumerated.item[0] = spec->cur_dmux;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_dmux_enum_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
return snd_hda_input_mux_put(codec, spec->dinput_mux, ucontrol,
|
|
|
|
spec->dmux_nid, &spec->cur_dmux);
|
|
|
|
}
|
|
|
|
|
2005-11-17 06:57:47 -07:00
|
|
|
static int stac92xx_mux_enum_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2005-06-27 05:59:41 -07:00
|
|
|
return snd_hda_input_mux_info(spec->input_mux, uinfo);
|
2005-04-13 05:45:30 -07:00
|
|
|
}
|
|
|
|
|
2005-11-17 06:57:47 -07:00
|
|
|
static int stac92xx_mux_enum_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
|
|
|
|
|
|
|
|
ucontrol->value.enumerated.item[0] = spec->cur_mux[adc_idx];
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 06:57:47 -07:00
|
|
|
static int stac92xx_mux_enum_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
unsigned int adc_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
return snd_hda_input_mux_put(codec, spec->input_mux, ucontrol,
|
2005-04-13 05:45:30 -07:00
|
|
|
spec->mux_nids[adc_idx], &spec->cur_mux[adc_idx]);
|
|
|
|
}
|
|
|
|
|
2007-09-03 06:29:37 -07:00
|
|
|
#define stac92xx_aloopback_info snd_ctl_boolean_mono_info
|
|
|
|
|
|
|
|
static int stac92xx_aloopback_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
ucontrol->value.integer.value[0] = spec->aloopback;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_aloopback_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
unsigned int dac_mode;
|
|
|
|
|
|
|
|
if (spec->aloopback == ucontrol->value.integer.value[0])
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
spec->aloopback = ucontrol->value.integer.value[0];
|
|
|
|
|
|
|
|
|
|
|
|
dac_mode = snd_hda_codec_read(codec, codec->afg, 0,
|
|
|
|
kcontrol->private_value & 0xFFFF, 0x0);
|
|
|
|
|
|
|
|
if (spec->aloopback) {
|
|
|
|
snd_hda_power_up(codec);
|
|
|
|
dac_mode |= 0x40;
|
|
|
|
} else {
|
|
|
|
snd_hda_power_down(codec);
|
|
|
|
dac_mode &= ~0x40;
|
|
|
|
}
|
|
|
|
|
|
|
|
snd_hda_codec_write_cache(codec, codec->afg, 0,
|
|
|
|
kcontrol->private_value >> 16, dac_mode);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
static struct hda_verb stac9200_core_init[] = {
|
2005-04-13 05:45:30 -07:00
|
|
|
/* set dac0mux for dac converter */
|
2005-06-27 05:59:41 -07:00
|
|
|
{ 0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
|
2005-04-13 05:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2007-10-10 01:04:26 -07:00
|
|
|
static struct hda_verb stac9200_eapd_init[] = {
|
|
|
|
/* set dac0mux for dac converter */
|
|
|
|
{0x07, AC_VERB_SET_CONNECT_SEL, 0x00},
|
|
|
|
{0x08, AC_VERB_SET_EAPD_BTLENABLE, 0x02},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
static struct hda_verb stac92hd71bxx_core_init[] = {
|
|
|
|
/* set master volume and direct control */
|
|
|
|
{ 0x28, AC_VERB_SET_VOLUME_KNOB_CONTROL, 0xff},
|
|
|
|
/* connect headphone jack to dac1 */
|
|
|
|
{ 0x0a, AC_VERB_SET_CONNECT_SEL, 0x01},
|
2007-11-07 05:03:12 -07:00
|
|
|
/* connect ports 0d and 0f to audio mixer */
|
|
|
|
{ 0x0d, AC_VERB_SET_CONNECT_SEL, 0x2},
|
|
|
|
{ 0x0f, AC_VERB_SET_CONNECT_SEL, 0x2},
|
|
|
|
/* unmute dac0 input in audio mixer */
|
|
|
|
{ 0x17, AC_VERB_SET_AMP_GAIN_MUTE, 0x701f},
|
2007-11-06 03:53:55 -07:00
|
|
|
/* unmute right and left channels for nodes 0x0a, 0xd, 0x0f */
|
|
|
|
{ 0x0a, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
|
|
|
|
{ 0x0d, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
|
|
|
|
{ 0x0f, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
|
|
|
|
/* unmute mono out node */
|
|
|
|
{ 0x14, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static struct hda_verb stac925x_core_init[] = {
|
|
|
|
/* set dac0mux for dac converter */
|
|
|
|
{ 0x06, AC_VERB_SET_CONNECT_SEL, 0x00},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
static struct hda_verb stac922x_core_init[] = {
|
2005-04-13 05:45:30 -07:00
|
|
|
/* set master volume and direct control */
|
2005-06-27 05:59:41 -07:00
|
|
|
{ 0x16, AC_VERB_SET_VOLUME_KNOB_CONTROL, 0xff},
|
2005-04-13 05:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-09-01 12:03:12 -07:00
|
|
|
static struct hda_verb d965_core_init[] = {
|
2006-06-28 06:52:16 -07:00
|
|
|
/* set master volume and direct control */
|
2006-09-01 12:03:12 -07:00
|
|
|
{ 0x24, AC_VERB_SET_VOLUME_KNOB_CONTROL, 0xff},
|
2006-06-28 06:52:16 -07:00
|
|
|
/* unmute node 0x1b */
|
|
|
|
{ 0x1b, AC_VERB_SET_AMP_GAIN_MUTE, 0xb000},
|
|
|
|
/* select node 0x03 as DAC */
|
|
|
|
{ 0x0b, AC_VERB_SET_CONNECT_SEL, 0x01},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
static struct hda_verb stac927x_core_init[] = {
|
|
|
|
/* set master volume and direct control */
|
|
|
|
{ 0x24, AC_VERB_SET_VOLUME_KNOB_CONTROL, 0xff},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-07-31 03:49:34 -07:00
|
|
|
static struct hda_verb stac9205_core_init[] = {
|
|
|
|
/* set master volume and direct control */
|
|
|
|
{ 0x24, AC_VERB_SET_VOLUME_KNOB_CONTROL, 0xff},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2007-10-18 23:19:56 -07:00
|
|
|
#define STAC_DIGITAL_INPUT_SOURCE(cnt) \
|
|
|
|
{ \
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
|
|
|
|
.name = "Digital Input Source", \
|
|
|
|
.count = cnt, \
|
|
|
|
.info = stac92xx_dmux_enum_info, \
|
|
|
|
.get = stac92xx_dmux_enum_get, \
|
|
|
|
.put = stac92xx_dmux_enum_put,\
|
|
|
|
}
|
|
|
|
|
2007-09-03 06:31:02 -07:00
|
|
|
#define STAC_INPUT_SOURCE(cnt) \
|
2007-08-31 03:52:19 -07:00
|
|
|
{ \
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
|
|
|
|
.name = "Input Source", \
|
2007-09-03 06:31:02 -07:00
|
|
|
.count = cnt, \
|
2007-08-31 03:52:19 -07:00
|
|
|
.info = stac92xx_mux_enum_info, \
|
|
|
|
.get = stac92xx_mux_enum_get, \
|
|
|
|
.put = stac92xx_mux_enum_put, \
|
|
|
|
}
|
|
|
|
|
2007-09-03 06:29:37 -07:00
|
|
|
#define STAC_ANALOG_LOOPBACK(verb_read,verb_write) \
|
|
|
|
{ \
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
|
|
|
|
.name = "Analog Loopback", \
|
|
|
|
.count = 1, \
|
|
|
|
.info = stac92xx_aloopback_info, \
|
|
|
|
.get = stac92xx_aloopback_get, \
|
|
|
|
.put = stac92xx_aloopback_put, \
|
|
|
|
.private_value = verb_read | (verb_write << 16), \
|
|
|
|
}
|
|
|
|
|
2007-11-07 05:03:12 -07:00
|
|
|
#define STAC_VOLKNOB(knob_nid) \
|
|
|
|
{ \
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
|
|
|
|
.name = "Master Playback Volume", \
|
|
|
|
.count = 1, \
|
|
|
|
.info = stac92xx_volknob_info, \
|
|
|
|
.get = stac92xx_volknob_get, \
|
|
|
|
.put = stac92xx_volknob_put, \
|
|
|
|
.private_value = 127 | (knob_nid << 16), \
|
|
|
|
}
|
|
|
|
|
2005-11-17 06:57:47 -07:00
|
|
|
static struct snd_kcontrol_new stac9200_mixer[] = {
|
2005-04-13 05:45:30 -07:00
|
|
|
HDA_CODEC_VOLUME("Master Playback Volume", 0xb, 0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_MUTE("Master Playback Switch", 0xb, 0, HDA_OUTPUT),
|
2007-09-03 06:31:02 -07:00
|
|
|
STAC_INPUT_SOURCE(1),
|
2005-04-13 05:45:30 -07:00
|
|
|
HDA_CODEC_VOLUME("Capture Volume", 0x0a, 0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_MUTE("Capture Switch", 0x0a, 0, HDA_OUTPUT),
|
2005-06-27 05:59:41 -07:00
|
|
|
HDA_CODEC_VOLUME("Capture Mux Volume", 0x0c, 0, HDA_OUTPUT),
|
2005-04-13 05:45:30 -07:00
|
|
|
{ } /* end */
|
|
|
|
};
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
static struct snd_kcontrol_new stac92hd71bxx_mixer[] = {
|
|
|
|
STAC_DIGITAL_INPUT_SOURCE(1),
|
|
|
|
STAC_INPUT_SOURCE(2),
|
|
|
|
STAC_VOLKNOB(0x28),
|
|
|
|
|
|
|
|
/* hardware gain controls */
|
2007-11-07 05:03:12 -07:00
|
|
|
HDA_CODEC_VOLUME_IDX("Digital Mic Volume", 0x0, 0x18, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Digital Mic Volume", 0x1, 0x19, 0x0, HDA_OUTPUT),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x0, 0x1c, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x0, 0x1c, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Mux Volume", 0x0, 0x1a, 0x0, HDA_OUTPUT),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x1, 0x1d, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x1, 0x1d, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Mux Volume", 0x1, 0x1b, 0x0, HDA_OUTPUT),
|
2007-11-06 03:53:55 -07:00
|
|
|
|
2007-11-07 05:03:12 -07:00
|
|
|
HDA_CODEC_MUTE("Analog Loopback 1", 0x17, 0x3, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE("Analog Loopback 2", 0x17, 0x4, HDA_INPUT),
|
2007-11-06 03:53:55 -07:00
|
|
|
{ } /* end */
|
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static struct snd_kcontrol_new stac925x_mixer[] = {
|
2007-09-03 06:31:02 -07:00
|
|
|
STAC_INPUT_SOURCE(1),
|
2007-01-08 03:04:17 -07:00
|
|
|
HDA_CODEC_VOLUME("Capture Volume", 0x09, 0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_MUTE("Capture Switch", 0x09, 0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME("Capture Mux Volume", 0x0f, 0, HDA_OUTPUT),
|
|
|
|
{ } /* end */
|
|
|
|
};
|
|
|
|
|
2007-09-03 06:31:02 -07:00
|
|
|
static struct snd_kcontrol_new stac9205_mixer[] = {
|
2007-10-18 23:19:56 -07:00
|
|
|
STAC_DIGITAL_INPUT_SOURCE(1),
|
2007-09-03 06:31:02 -07:00
|
|
|
STAC_INPUT_SOURCE(2),
|
|
|
|
STAC_ANALOG_LOOPBACK(0xFE0, 0x7E0),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x0, 0x1b, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x0, 0x1d, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x0, 0x19, 0x0, HDA_OUTPUT),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x1, 0x1c, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x1, 0x1e, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x1, 0x1A, 0x0, HDA_OUTPUT),
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
{ } /* end */
|
|
|
|
};
|
|
|
|
|
2006-06-28 06:52:16 -07:00
|
|
|
/* This needs to be generated dynamically based on sequence */
|
2007-09-03 06:31:02 -07:00
|
|
|
static struct snd_kcontrol_new stac922x_mixer[] = {
|
|
|
|
STAC_INPUT_SOURCE(2),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x0, 0x17, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x0, 0x17, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x0, 0x12, 0x0, HDA_OUTPUT),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x1, 0x18, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x1, 0x18, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x1, 0x13, 0x0, HDA_OUTPUT),
|
2006-06-28 06:52:16 -07:00
|
|
|
{ } /* end */
|
|
|
|
};
|
|
|
|
|
2007-09-03 06:31:02 -07:00
|
|
|
|
2006-11-23 11:27:12 -07:00
|
|
|
static struct snd_kcontrol_new stac927x_mixer[] = {
|
2007-10-18 23:19:56 -07:00
|
|
|
STAC_DIGITAL_INPUT_SOURCE(1),
|
2007-09-03 06:31:02 -07:00
|
|
|
STAC_INPUT_SOURCE(3),
|
2007-09-03 06:29:37 -07:00
|
|
|
STAC_ANALOG_LOOPBACK(0xFEB, 0x7EB),
|
2006-01-23 07:27:49 -07:00
|
|
|
|
2007-09-03 06:31:02 -07:00
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x0, 0x18, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x0, 0x1b, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x0, 0x15, 0x0, HDA_OUTPUT),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x1, 0x19, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x1, 0x1c, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x1, 0x16, 0x0, HDA_OUTPUT),
|
|
|
|
|
|
|
|
HDA_CODEC_VOLUME_IDX("Capture Volume", 0x2, 0x1A, 0x0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE_IDX("Capture Switch", 0x2, 0x1d, 0x0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME_IDX("Mux Capture Volume", 0x2, 0x17, 0x0, HDA_OUTPUT),
|
2006-07-31 03:49:34 -07:00
|
|
|
{ } /* end */
|
|
|
|
};
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static int stac92xx_build_controls(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int err;
|
2005-06-27 05:59:41 -07:00
|
|
|
int i;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
err = snd_hda_add_new_ctls(codec, spec->mixer);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
for (i = 0; i < spec->num_mixers; i++) {
|
|
|
|
err = snd_hda_add_new_ctls(codec, spec->mixers[i]);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2005-06-14 01:19:34 -07:00
|
|
|
if (spec->multiout.dig_out_nid) {
|
|
|
|
err = snd_hda_create_spdif_out_ctls(codec, spec->multiout.dig_out_nid);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
if (spec->dig_in_nid) {
|
|
|
|
err = snd_hda_create_spdif_in_ctls(codec, spec->dig_in_nid);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
return 0;
|
2005-04-13 05:45:30 -07:00
|
|
|
}
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
static unsigned int ref9200_pin_configs[8] = {
|
2005-06-14 01:19:34 -07:00
|
|
|
0x01c47010, 0x01447010, 0x0221401f, 0x01114010,
|
2005-04-13 05:45:30 -07:00
|
|
|
0x02a19020, 0x01a19021, 0x90100140, 0x01813122,
|
|
|
|
};
|
|
|
|
|
2007-08-23 10:04:28 -07:00
|
|
|
/*
|
|
|
|
STAC 9200 pin configs for
|
|
|
|
102801A8
|
|
|
|
102801DE
|
|
|
|
102801E8
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_d21_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x400001f0, 0x400001f1, 0x02214030, 0x01014010,
|
|
|
|
0x02a19020, 0x01a19021, 0x90100140, 0x01813122,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200 pin configs for
|
|
|
|
102801C0
|
|
|
|
102801C1
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_d22_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x400001f0, 0x400001f1, 0x0221401f, 0x01014010,
|
|
|
|
0x01813020, 0x02a19021, 0x90100140, 0x400001f2,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200 pin configs for
|
|
|
|
102801C4 (Dell Dimension E310)
|
|
|
|
102801C5
|
|
|
|
102801C7
|
|
|
|
102801D9
|
|
|
|
102801DA
|
|
|
|
102801E3
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_d23_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x400001f0, 0x400001f1, 0x0221401f, 0x01014010,
|
|
|
|
0x01813020, 0x01a19021, 0x90100140, 0x400001f2,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32 pin configs for
|
|
|
|
102801B5 (Dell Inspiron 630m)
|
|
|
|
102801D8 (Dell Inspiron 640m)
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m21_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x40c003fa, 0x03441340, 0x0321121f, 0x90170310,
|
|
|
|
0x408003fb, 0x03a11020, 0x401003fc, 0x403003fd,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32 pin configs for
|
|
|
|
102801C2 (Dell Latitude D620)
|
|
|
|
102801C8
|
|
|
|
102801CC (Dell Latitude D820)
|
|
|
|
102801D4
|
|
|
|
102801D6
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m22_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x40c003fa, 0x0144131f, 0x0321121f, 0x90170310,
|
|
|
|
0x90a70321, 0x03a11020, 0x401003fb, 0x40f000fc,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32 pin configs for
|
|
|
|
102801CE (Dell XPS M1710)
|
|
|
|
102801CF (Dell Precision M90)
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m23_pin_configs[8] = {
|
|
|
|
0x40c003fa, 0x01441340, 0x0421421f, 0x90170310,
|
|
|
|
0x408003fb, 0x04a1102e, 0x90170311, 0x403003fc,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32 pin configs for
|
|
|
|
102801C9
|
|
|
|
102801CA
|
|
|
|
102801CB (Dell Latitude 120L)
|
|
|
|
102801D3
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m24_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x40c003fa, 0x404003fb, 0x0321121f, 0x90170310,
|
|
|
|
0x408003fc, 0x03a11020, 0x401003fd, 0x403003fe,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32 pin configs for
|
|
|
|
102801BD (Dell Inspiron E1505n)
|
|
|
|
102801EE
|
|
|
|
102801EF
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m25_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x40c003fa, 0x01441340, 0x0421121f, 0x90170310,
|
|
|
|
0x408003fb, 0x04a11020, 0x401003fc, 0x403003fd,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32 pin configs for
|
|
|
|
102801F5 (Dell Inspiron 1501)
|
|
|
|
102801F6
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m26_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x40c003fa, 0x404003fb, 0x0421121f, 0x90170310,
|
|
|
|
0x408003fc, 0x04a11020, 0x401003fd, 0x403003fe,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9200-32
|
|
|
|
102801CD (Dell Inspiron E1705/9400)
|
|
|
|
*/
|
|
|
|
static unsigned int dell9200_m27_pin_configs[8] = {
|
2007-09-05 14:46:03 -07:00
|
|
|
0x40c003fa, 0x01441340, 0x0421121f, 0x90170310,
|
|
|
|
0x90170310, 0x04a11020, 0x90170310, 0x40f003fc,
|
2007-08-23 10:04:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
static unsigned int *stac9200_brd_tbl[STAC_9200_MODELS] = {
|
|
|
|
[STAC_REF] = ref9200_pin_configs,
|
2007-08-23 10:04:28 -07:00
|
|
|
[STAC_9200_DELL_D21] = dell9200_d21_pin_configs,
|
|
|
|
[STAC_9200_DELL_D22] = dell9200_d22_pin_configs,
|
|
|
|
[STAC_9200_DELL_D23] = dell9200_d23_pin_configs,
|
|
|
|
[STAC_9200_DELL_M21] = dell9200_m21_pin_configs,
|
|
|
|
[STAC_9200_DELL_M22] = dell9200_m22_pin_configs,
|
|
|
|
[STAC_9200_DELL_M23] = dell9200_m23_pin_configs,
|
|
|
|
[STAC_9200_DELL_M24] = dell9200_m24_pin_configs,
|
|
|
|
[STAC_9200_DELL_M25] = dell9200_m25_pin_configs,
|
|
|
|
[STAC_9200_DELL_M26] = dell9200_m26_pin_configs,
|
|
|
|
[STAC_9200_DELL_M27] = dell9200_m27_pin_configs,
|
2005-11-29 07:00:51 -07:00
|
|
|
};
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
static const char *stac9200_models[STAC_9200_MODELS] = {
|
|
|
|
[STAC_REF] = "ref",
|
2007-08-23 10:04:28 -07:00
|
|
|
[STAC_9200_DELL_D21] = "dell-d21",
|
|
|
|
[STAC_9200_DELL_D22] = "dell-d22",
|
|
|
|
[STAC_9200_DELL_D23] = "dell-d23",
|
|
|
|
[STAC_9200_DELL_M21] = "dell-m21",
|
|
|
|
[STAC_9200_DELL_M22] = "dell-m22",
|
|
|
|
[STAC_9200_DELL_M23] = "dell-m23",
|
|
|
|
[STAC_9200_DELL_M24] = "dell-m24",
|
|
|
|
[STAC_9200_DELL_M25] = "dell-m25",
|
|
|
|
[STAC_9200_DELL_M26] = "dell-m26",
|
|
|
|
[STAC_9200_DELL_M27] = "dell-m27",
|
2007-10-10 01:04:26 -07:00
|
|
|
[STAC_9200_GATEWAY] = "gateway",
|
2006-11-24 09:07:44 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac9200_cfg_tbl[] = {
|
|
|
|
/* SigmaTel reference board */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2668,
|
|
|
|
"DFI LanParty", STAC_REF),
|
2006-11-06 03:20:38 -07:00
|
|
|
/* Dell laptops have BIOS problem */
|
2007-08-23 10:04:28 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01a8,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D21),
|
2006-11-24 09:07:44 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01b5,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Inspiron 630m", STAC_9200_DELL_M21),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01bd,
|
|
|
|
"Dell Inspiron E1505n", STAC_9200_DELL_M25),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c0,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D22),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c1,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D22),
|
2006-11-24 09:07:44 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c2,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Latitude D620", STAC_9200_DELL_M22),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c5,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D23),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c7,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D23),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c8,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M22),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01c9,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M24),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ca,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M24),
|
2006-11-24 09:07:44 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01cb,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Latitude 120L", STAC_9200_DELL_M24),
|
2007-01-30 09:30:55 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01cc,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Latitude D820", STAC_9200_DELL_M22),
|
2007-02-13 04:46:16 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01cd,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Inspiron E1705/9400", STAC_9200_DELL_M27),
|
2007-02-13 04:46:16 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ce,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell XPS M1710", STAC_9200_DELL_M23),
|
2007-02-13 16:59:17 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01cf,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Precision M90", STAC_9200_DELL_M23),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d3,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M22),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d4,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M22),
|
2007-05-15 02:46:23 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d6,
|
2007-08-23 10:04:28 -07:00
|
|
|
"unknown Dell", STAC_9200_DELL_M22),
|
2007-05-17 00:38:24 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d8,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Inspiron 640m", STAC_9200_DELL_M21),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d9,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D23),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01da,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D23),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01de,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D21),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01e3,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D23),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01e8,
|
|
|
|
"unknown Dell", STAC_9200_DELL_D21),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ee,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M25),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ef,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M25),
|
2007-05-17 00:38:24 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f5,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Inspiron 1501", STAC_9200_DELL_M26),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f6,
|
|
|
|
"unknown Dell", STAC_9200_DELL_M26),
|
2007-05-17 00:38:24 -07:00
|
|
|
/* Panasonic */
|
|
|
|
SND_PCI_QUIRK(0x10f7, 0x8338, "Panasonic CF-74", STAC_REF),
|
2007-10-10 01:04:26 -07:00
|
|
|
/* Gateway machines needs EAPD to be set on resume */
|
|
|
|
SND_PCI_QUIRK(0x107b, 0x0205, "Gateway S-7110M", STAC_9200_GATEWAY),
|
|
|
|
SND_PCI_QUIRK(0x107b, 0x0317, "Gateway MT3423, MX341*",
|
|
|
|
STAC_9200_GATEWAY),
|
|
|
|
SND_PCI_QUIRK(0x107b, 0x0318, "Gateway ML3019, MT3707",
|
|
|
|
STAC_9200_GATEWAY),
|
2005-11-29 07:00:51 -07:00
|
|
|
{} /* terminator */
|
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static unsigned int ref925x_pin_configs[8] = {
|
|
|
|
0x40c003f0, 0x424503f2, 0x01813022, 0x02a19021,
|
|
|
|
0x90a70320, 0x02214210, 0x400003f1, 0x9033032e,
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned int stac925x_MA6_pin_configs[8] = {
|
|
|
|
0x40c003f0, 0x424503f2, 0x01813022, 0x02a19021,
|
|
|
|
0x90a70320, 0x90100211, 0x400003f1, 0x9033032e,
|
|
|
|
};
|
|
|
|
|
2007-05-17 00:36:34 -07:00
|
|
|
static unsigned int stac925x_PA6_pin_configs[8] = {
|
|
|
|
0x40c003f0, 0x424503f2, 0x01813022, 0x02a19021,
|
|
|
|
0x50a103f0, 0x90100211, 0x400003f1, 0x9033032e,
|
|
|
|
};
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static unsigned int stac925xM2_2_pin_configs[8] = {
|
2007-05-29 05:36:17 -07:00
|
|
|
0x40c003f3, 0x424503f2, 0x04180011, 0x02a19020,
|
|
|
|
0x50a103f0, 0x90100212, 0x400003f1, 0x9033032e,
|
2007-01-08 03:04:17 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned int *stac925x_brd_tbl[STAC_925x_MODELS] = {
|
|
|
|
[STAC_REF] = ref925x_pin_configs,
|
|
|
|
[STAC_M2_2] = stac925xM2_2_pin_configs,
|
|
|
|
[STAC_MA6] = stac925x_MA6_pin_configs,
|
2007-05-17 00:36:34 -07:00
|
|
|
[STAC_PA6] = stac925x_PA6_pin_configs,
|
2007-01-08 03:04:17 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static const char *stac925x_models[STAC_925x_MODELS] = {
|
|
|
|
[STAC_REF] = "ref",
|
|
|
|
[STAC_M2_2] = "m2-2",
|
|
|
|
[STAC_MA6] = "m6",
|
2007-05-17 00:36:34 -07:00
|
|
|
[STAC_PA6] = "pa6",
|
2007-01-08 03:04:17 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac925x_cfg_tbl[] = {
|
|
|
|
/* SigmaTel reference board */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2668, "DFI LanParty", STAC_REF),
|
2007-05-17 00:36:34 -07:00
|
|
|
SND_PCI_QUIRK(0x8384, 0x7632, "Stac9202 Reference Board", STAC_REF),
|
2007-01-08 03:04:17 -07:00
|
|
|
SND_PCI_QUIRK(0x107b, 0x0316, "Gateway M255", STAC_REF),
|
|
|
|
SND_PCI_QUIRK(0x107b, 0x0366, "Gateway MP6954", STAC_REF),
|
|
|
|
SND_PCI_QUIRK(0x107b, 0x0461, "Gateway NX560XL", STAC_MA6),
|
2007-05-17 00:36:34 -07:00
|
|
|
SND_PCI_QUIRK(0x107b, 0x0681, "Gateway NX860", STAC_PA6),
|
2007-01-08 03:04:17 -07:00
|
|
|
SND_PCI_QUIRK(0x1002, 0x437b, "Gateway MX6453", STAC_M2_2),
|
|
|
|
{} /* terminator */
|
|
|
|
};
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
static unsigned int ref92hd71bxx_pin_configs[10] = {
|
|
|
|
0x02214030, 0x02a19040, 0x01a19020, 0x01014010,
|
|
|
|
0x0181302e, 0x01114010, 0x01a19020, 0x90a000f0,
|
|
|
|
0x90a000f0, 0x01452050,
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned int *stac92hd71bxx_brd_tbl[STAC_92HD71BXX_MODELS] = {
|
|
|
|
[STAC_92HD71BXX_REF] = ref92hd71bxx_pin_configs,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *stac92hd71bxx_models[STAC_92HD71BXX_MODELS] = {
|
|
|
|
[STAC_92HD71BXX_REF] = "ref",
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac92hd71bxx_cfg_tbl[] = {
|
|
|
|
/* SigmaTel reference board */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2668,
|
|
|
|
"DFI LanParty", STAC_92HD71BXX_REF),
|
|
|
|
{} /* terminator */
|
|
|
|
};
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
static unsigned int ref922x_pin_configs[10] = {
|
|
|
|
0x01014010, 0x01016011, 0x01012012, 0x0221401f,
|
|
|
|
0x01813122, 0x01011014, 0x01441030, 0x01c41030,
|
2005-04-13 05:45:30 -07:00
|
|
|
0x40000100, 0x40000100,
|
|
|
|
};
|
|
|
|
|
2007-08-23 10:04:28 -07:00
|
|
|
/*
|
|
|
|
STAC 922X pin configs for
|
|
|
|
102801A7
|
|
|
|
102801AB
|
|
|
|
102801A9
|
|
|
|
102801D1
|
|
|
|
102801D2
|
|
|
|
*/
|
|
|
|
static unsigned int dell_922x_d81_pin_configs[10] = {
|
|
|
|
0x02214030, 0x01a19021, 0x01111012, 0x01114010,
|
|
|
|
0x02a19020, 0x01117011, 0x400001f0, 0x400001f1,
|
|
|
|
0x01813122, 0x400001f2,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 922X pin configs for
|
|
|
|
102801AC
|
|
|
|
102801D0
|
|
|
|
*/
|
|
|
|
static unsigned int dell_922x_d82_pin_configs[10] = {
|
|
|
|
0x02214030, 0x01a19021, 0x01111012, 0x01114010,
|
|
|
|
0x02a19020, 0x01117011, 0x01451140, 0x400001f0,
|
|
|
|
0x01813122, 0x400001f1,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 922X pin configs for
|
|
|
|
102801BF
|
|
|
|
*/
|
|
|
|
static unsigned int dell_922x_m81_pin_configs[10] = {
|
|
|
|
0x0321101f, 0x01112024, 0x01111222, 0x91174220,
|
|
|
|
0x03a11050, 0x01116221, 0x90a70330, 0x01452340,
|
|
|
|
0x40C003f1, 0x405003f0,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9221 A1 pin configs for
|
|
|
|
102801D7 (Dell XPS M1210)
|
|
|
|
*/
|
|
|
|
static unsigned int dell_922x_m82_pin_configs[10] = {
|
|
|
|
0x0221121f, 0x408103ff, 0x02111212, 0x90100310,
|
|
|
|
0x408003f1, 0x02111211, 0x03451340, 0x40c003f2,
|
|
|
|
0x508003f3, 0x405003f4,
|
|
|
|
};
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
static unsigned int d945gtp3_pin_configs[10] = {
|
2006-01-25 11:20:50 -07:00
|
|
|
0x0221401f, 0x01a19022, 0x01813021, 0x01014010,
|
2005-11-29 07:00:51 -07:00
|
|
|
0x40000100, 0x40000100, 0x40000100, 0x40000100,
|
|
|
|
0x02a19120, 0x40000100,
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned int d945gtp5_pin_configs[10] = {
|
2006-01-25 11:20:50 -07:00
|
|
|
0x0221401f, 0x01011012, 0x01813024, 0x01014010,
|
|
|
|
0x01a19021, 0x01016011, 0x01452130, 0x40000100,
|
2005-11-29 07:00:51 -07:00
|
|
|
0x02a19320, 0x40000100,
|
|
|
|
};
|
|
|
|
|
2007-05-29 07:03:00 -07:00
|
|
|
static unsigned int intel_mac_v1_pin_configs[10] = {
|
|
|
|
0x0121e21f, 0x400000ff, 0x9017e110, 0x400000fd,
|
|
|
|
0x400000fe, 0x0181e020, 0x1145e030, 0x11c5e240,
|
|
|
|
0x400000fc, 0x400000fb,
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned int intel_mac_v2_pin_configs[10] = {
|
|
|
|
0x0121e21f, 0x90a7012e, 0x9017e110, 0x400000fd,
|
|
|
|
0x400000fe, 0x0181e020, 0x1145e230, 0x500000fa,
|
|
|
|
0x400000fc, 0x400000fb,
|
2007-03-15 04:38:15 -07:00
|
|
|
};
|
|
|
|
|
2007-05-29 07:03:00 -07:00
|
|
|
static unsigned int intel_mac_v3_pin_configs[10] = {
|
|
|
|
0x0121e21f, 0x90a7012e, 0x9017e110, 0x400000fd,
|
|
|
|
0x400000fe, 0x0181e020, 0x1145e230, 0x11c5e240,
|
2007-02-16 05:27:18 -07:00
|
|
|
0x400000fc, 0x400000fb,
|
|
|
|
};
|
|
|
|
|
2007-05-29 07:03:00 -07:00
|
|
|
static unsigned int intel_mac_v4_pin_configs[10] = {
|
|
|
|
0x0321e21f, 0x03a1e02e, 0x9017e110, 0x9017e11f,
|
|
|
|
0x400000fe, 0x0381e020, 0x1345e230, 0x13c5e240,
|
2007-04-27 05:22:36 -07:00
|
|
|
0x400000fc, 0x400000fb,
|
|
|
|
};
|
|
|
|
|
2007-05-29 07:03:00 -07:00
|
|
|
static unsigned int intel_mac_v5_pin_configs[10] = {
|
|
|
|
0x0321e21f, 0x03a1e02e, 0x9017e110, 0x9017e11f,
|
|
|
|
0x400000fe, 0x0381e020, 0x1345e230, 0x13c5e240,
|
|
|
|
0x400000fc, 0x400000fb,
|
2007-05-21 03:41:29 -07:00
|
|
|
};
|
|
|
|
|
2007-06-19 03:17:42 -07:00
|
|
|
|
2006-06-28 06:52:16 -07:00
|
|
|
static unsigned int *stac922x_brd_tbl[STAC_922X_MODELS] = {
|
2006-11-24 09:07:44 -07:00
|
|
|
[STAC_D945_REF] = ref922x_pin_configs,
|
2006-06-28 06:52:16 -07:00
|
|
|
[STAC_D945GTP3] = d945gtp3_pin_configs,
|
|
|
|
[STAC_D945GTP5] = d945gtp5_pin_configs,
|
2007-05-29 07:03:00 -07:00
|
|
|
[STAC_INTEL_MAC_V1] = intel_mac_v1_pin_configs,
|
|
|
|
[STAC_INTEL_MAC_V2] = intel_mac_v2_pin_configs,
|
|
|
|
[STAC_INTEL_MAC_V3] = intel_mac_v3_pin_configs,
|
|
|
|
[STAC_INTEL_MAC_V4] = intel_mac_v4_pin_configs,
|
|
|
|
[STAC_INTEL_MAC_V5] = intel_mac_v5_pin_configs,
|
2007-08-23 10:04:28 -07:00
|
|
|
/* for backward compatibility */
|
2007-05-29 07:03:00 -07:00
|
|
|
[STAC_MACMINI] = intel_mac_v3_pin_configs,
|
|
|
|
[STAC_MACBOOK] = intel_mac_v5_pin_configs,
|
|
|
|
[STAC_MACBOOK_PRO_V1] = intel_mac_v3_pin_configs,
|
|
|
|
[STAC_MACBOOK_PRO_V2] = intel_mac_v3_pin_configs,
|
|
|
|
[STAC_IMAC_INTEL] = intel_mac_v2_pin_configs,
|
|
|
|
[STAC_IMAC_INTEL_20] = intel_mac_v3_pin_configs,
|
2007-08-23 10:04:28 -07:00
|
|
|
[STAC_922X_DELL_D81] = dell_922x_d81_pin_configs,
|
|
|
|
[STAC_922X_DELL_D82] = dell_922x_d82_pin_configs,
|
|
|
|
[STAC_922X_DELL_M81] = dell_922x_m81_pin_configs,
|
|
|
|
[STAC_922X_DELL_M82] = dell_922x_m82_pin_configs,
|
2005-11-29 07:00:51 -07:00
|
|
|
};
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
static const char *stac922x_models[STAC_922X_MODELS] = {
|
|
|
|
[STAC_D945_REF] = "ref",
|
|
|
|
[STAC_D945GTP5] = "5stack",
|
|
|
|
[STAC_D945GTP3] = "3stack",
|
2007-05-29 07:03:00 -07:00
|
|
|
[STAC_INTEL_MAC_V1] = "intel-mac-v1",
|
|
|
|
[STAC_INTEL_MAC_V2] = "intel-mac-v2",
|
|
|
|
[STAC_INTEL_MAC_V3] = "intel-mac-v3",
|
|
|
|
[STAC_INTEL_MAC_V4] = "intel-mac-v4",
|
|
|
|
[STAC_INTEL_MAC_V5] = "intel-mac-v5",
|
2007-08-23 10:04:28 -07:00
|
|
|
/* for backward compatibility */
|
2006-11-24 09:07:44 -07:00
|
|
|
[STAC_MACMINI] = "macmini",
|
2007-02-16 05:27:18 -07:00
|
|
|
[STAC_MACBOOK] = "macbook",
|
2007-03-15 04:38:15 -07:00
|
|
|
[STAC_MACBOOK_PRO_V1] = "macbook-pro-v1",
|
|
|
|
[STAC_MACBOOK_PRO_V2] = "macbook-pro",
|
2007-04-27 05:22:36 -07:00
|
|
|
[STAC_IMAC_INTEL] = "imac-intel",
|
2007-05-21 03:41:29 -07:00
|
|
|
[STAC_IMAC_INTEL_20] = "imac-intel-20",
|
2007-08-23 10:04:28 -07:00
|
|
|
[STAC_922X_DELL_D81] = "dell-d81",
|
|
|
|
[STAC_922X_DELL_D82] = "dell-d82",
|
|
|
|
[STAC_922X_DELL_M81] = "dell-m81",
|
|
|
|
[STAC_922X_DELL_M82] = "dell-m82",
|
2006-11-24 09:07:44 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac922x_cfg_tbl[] = {
|
|
|
|
/* SigmaTel reference board */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2668,
|
|
|
|
"DFI LanParty", STAC_D945_REF),
|
|
|
|
/* Intel 945G based systems */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0101,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0202,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0606,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0601,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0111,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x1115,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x1116,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x1117,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x1118,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x1119,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x8826,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x5049,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x5055,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x5048,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0110,
|
|
|
|
"Intel D945G", STAC_D945GTP3),
|
|
|
|
/* Intel D945G 5-stack systems */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0404,
|
|
|
|
"Intel D945G", STAC_D945GTP5),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0303,
|
|
|
|
"Intel D945G", STAC_D945GTP5),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0013,
|
|
|
|
"Intel D945G", STAC_D945GTP5),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0417,
|
|
|
|
"Intel D945G", STAC_D945GTP5),
|
|
|
|
/* Intel 945P based systems */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0b0b,
|
|
|
|
"Intel D945P", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0112,
|
|
|
|
"Intel D945P", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0d0d,
|
|
|
|
"Intel D945P", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0909,
|
|
|
|
"Intel D945P", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0505,
|
|
|
|
"Intel D945P", STAC_D945GTP3),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x0707,
|
|
|
|
"Intel D945P", STAC_D945GTP5),
|
|
|
|
/* other systems */
|
|
|
|
/* Apple Mac Mini (early 2006) */
|
|
|
|
SND_PCI_QUIRK(0x8384, 0x7680,
|
2007-05-29 07:03:00 -07:00
|
|
|
"Mac Mini", STAC_INTEL_MAC_V3),
|
2007-08-23 10:04:28 -07:00
|
|
|
/* Dell systems */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01a7,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D81),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01a9,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D81),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ab,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D81),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ac,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D82),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01bf,
|
|
|
|
"unknown Dell", STAC_922X_DELL_M81),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d0,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D82),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d1,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D81),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d2,
|
|
|
|
"unknown Dell", STAC_922X_DELL_D81),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01d7,
|
|
|
|
"Dell XPS M1210", STAC_922X_DELL_M82),
|
2005-11-29 07:00:51 -07:00
|
|
|
{} /* terminator */
|
|
|
|
};
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
static unsigned int ref927x_pin_configs[14] = {
|
2006-09-01 12:03:12 -07:00
|
|
|
0x02214020, 0x02a19080, 0x0181304e, 0x01014010,
|
|
|
|
0x01a19040, 0x01011012, 0x01016011, 0x0101201f,
|
|
|
|
0x183301f0, 0x18a001f0, 0x18a001f0, 0x01442070,
|
|
|
|
0x01c42190, 0x40000100,
|
2006-01-23 07:27:49 -07:00
|
|
|
};
|
|
|
|
|
2006-09-01 12:03:12 -07:00
|
|
|
static unsigned int d965_3st_pin_configs[14] = {
|
2006-08-22 10:44:45 -07:00
|
|
|
0x0221401f, 0x02a19120, 0x40000100, 0x01014011,
|
|
|
|
0x01a19021, 0x01813024, 0x40000100, 0x40000100,
|
|
|
|
0x40000100, 0x40000100, 0x40000100, 0x40000100,
|
|
|
|
0x40000100, 0x40000100
|
|
|
|
};
|
|
|
|
|
2006-09-01 12:03:12 -07:00
|
|
|
static unsigned int d965_5st_pin_configs[14] = {
|
|
|
|
0x02214020, 0x02a19080, 0x0181304e, 0x01014010,
|
|
|
|
0x01a19040, 0x01011012, 0x01016011, 0x40000100,
|
|
|
|
0x40000100, 0x40000100, 0x40000100, 0x01442070,
|
|
|
|
0x40000100, 0x40000100
|
|
|
|
};
|
|
|
|
|
2007-08-07 02:48:12 -07:00
|
|
|
static unsigned int dell_3st_pin_configs[14] = {
|
|
|
|
0x02211230, 0x02a11220, 0x01a19040, 0x01114210,
|
|
|
|
0x01111212, 0x01116211, 0x01813050, 0x01112214,
|
|
|
|
0x403003fa, 0x40000100, 0x40000100, 0x404003fb,
|
|
|
|
0x40c003fc, 0x40000100
|
|
|
|
};
|
|
|
|
|
2006-09-01 12:03:12 -07:00
|
|
|
static unsigned int *stac927x_brd_tbl[STAC_927X_MODELS] = {
|
2006-11-24 09:07:44 -07:00
|
|
|
[STAC_D965_REF] = ref927x_pin_configs,
|
2006-09-01 12:03:12 -07:00
|
|
|
[STAC_D965_3ST] = d965_3st_pin_configs,
|
|
|
|
[STAC_D965_5ST] = d965_5st_pin_configs,
|
2007-08-07 02:48:12 -07:00
|
|
|
[STAC_DELL_3ST] = dell_3st_pin_configs,
|
2006-01-23 07:27:49 -07:00
|
|
|
};
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
static const char *stac927x_models[STAC_927X_MODELS] = {
|
|
|
|
[STAC_D965_REF] = "ref",
|
|
|
|
[STAC_D965_3ST] = "3stack",
|
|
|
|
[STAC_D965_5ST] = "5stack",
|
2007-08-07 02:48:12 -07:00
|
|
|
[STAC_DELL_3ST] = "dell-3stack",
|
2006-11-24 09:07:44 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac927x_cfg_tbl[] = {
|
|
|
|
/* SigmaTel reference board */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2668,
|
|
|
|
"DFI LanParty", STAC_D965_REF),
|
2006-08-22 10:44:45 -07:00
|
|
|
/* Intel 946 based systems */
|
2006-11-24 09:07:44 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x3d01, "Intel D946", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0xa301, "Intel D946", STAC_D965_3ST),
|
2006-09-01 12:03:12 -07:00
|
|
|
/* 965 based 3 stack systems */
|
2006-11-24 09:07:44 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2116, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2115, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2114, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2113, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2112, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2111, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2110, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2009, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2008, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2007, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2006, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2005, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2004, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2003, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2002, "Intel D965", STAC_D965_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2001, "Intel D965", STAC_D965_3ST),
|
2007-10-10 01:42:00 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f3, "Dell Inspiron 1420", STAC_D965_3ST),
|
2007-08-07 02:48:12 -07:00
|
|
|
/* Dell 3 stack systems */
|
2007-08-23 10:04:28 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01dd, "Dell Dimension E520", STAC_DELL_3ST),
|
2007-08-07 02:48:12 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ed, "Dell ", STAC_DELL_3ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f4, "Dell ", STAC_DELL_3ST),
|
2006-09-01 12:03:12 -07:00
|
|
|
/* 965 based 5 stack systems */
|
2007-10-10 01:42:00 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x0209, "Dell XPS 1330", STAC_D965_5ST),
|
2006-11-24 09:07:44 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2301, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2302, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2303, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2304, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2305, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2501, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2502, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2503, "Intel D965", STAC_D965_5ST),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2504, "Intel D965", STAC_D965_5ST),
|
2006-01-23 07:27:49 -07:00
|
|
|
{} /* terminator */
|
|
|
|
};
|
|
|
|
|
2006-07-31 03:49:34 -07:00
|
|
|
static unsigned int ref9205_pin_configs[12] = {
|
|
|
|
0x40000100, 0x40000100, 0x01016011, 0x01014010,
|
2006-10-26 08:12:59 -07:00
|
|
|
0x01813122, 0x01a19021, 0x40000100, 0x40000100,
|
|
|
|
0x90a000f0, 0x90a000f0, 0x01441030, 0x01c41030
|
2006-07-31 03:49:34 -07:00
|
|
|
};
|
|
|
|
|
2007-08-23 10:04:28 -07:00
|
|
|
/*
|
|
|
|
STAC 9205 pin configs for
|
|
|
|
102801F1
|
|
|
|
102801F2
|
|
|
|
102801FC
|
|
|
|
102801FD
|
|
|
|
10280204
|
|
|
|
1028021F
|
|
|
|
*/
|
|
|
|
static unsigned int dell_9205_m42_pin_configs[12] = {
|
|
|
|
0x0321101F, 0x03A11020, 0x400003FA, 0x90170310,
|
|
|
|
0x400003FB, 0x400003FC, 0x400003FD, 0x40F000F9,
|
|
|
|
0x90A60330, 0x400003FF, 0x0144131F, 0x40C003FE,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
STAC 9205 pin configs for
|
|
|
|
102801F9
|
|
|
|
102801FA
|
|
|
|
102801FE
|
|
|
|
102801FF (Dell Precision M4300)
|
|
|
|
10280206
|
|
|
|
10280200
|
|
|
|
10280201
|
|
|
|
*/
|
|
|
|
static unsigned int dell_9205_m43_pin_configs[12] = {
|
2007-08-13 06:50:29 -07:00
|
|
|
0x0321101f, 0x03a11020, 0x90a70330, 0x90170310,
|
|
|
|
0x400000fe, 0x400000ff, 0x400000fd, 0x40f000f9,
|
|
|
|
0x400000fa, 0x400000fc, 0x0144131f, 0x40c003f8,
|
|
|
|
};
|
|
|
|
|
2007-08-23 10:04:28 -07:00
|
|
|
static unsigned int dell_9205_m44_pin_configs[12] = {
|
2007-08-13 06:50:29 -07:00
|
|
|
0x0421101f, 0x04a11020, 0x400003fa, 0x90170310,
|
|
|
|
0x400003fb, 0x400003fc, 0x400003fd, 0x400003f9,
|
|
|
|
0x90a60330, 0x400003ff, 0x01441340, 0x40c003fe,
|
|
|
|
};
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
static unsigned int *stac9205_brd_tbl[STAC_9205_MODELS] = {
|
2007-08-13 06:50:29 -07:00
|
|
|
[STAC_9205_REF] = ref9205_pin_configs,
|
2007-08-23 10:04:28 -07:00
|
|
|
[STAC_9205_DELL_M42] = dell_9205_m42_pin_configs,
|
|
|
|
[STAC_9205_DELL_M43] = dell_9205_m43_pin_configs,
|
|
|
|
[STAC_9205_DELL_M44] = dell_9205_m44_pin_configs,
|
2006-07-31 03:49:34 -07:00
|
|
|
};
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
static const char *stac9205_models[STAC_9205_MODELS] = {
|
|
|
|
[STAC_9205_REF] = "ref",
|
2007-08-23 10:04:28 -07:00
|
|
|
[STAC_9205_DELL_M42] = "dell-m42",
|
2007-08-13 06:50:29 -07:00
|
|
|
[STAC_9205_DELL_M43] = "dell-m43",
|
|
|
|
[STAC_9205_DELL_M44] = "dell-m44",
|
2006-11-24 09:07:44 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac9205_cfg_tbl[] = {
|
|
|
|
/* SigmaTel reference board */
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_INTEL, 0x2668,
|
|
|
|
"DFI LanParty", STAC_9205_REF),
|
2007-08-23 10:04:28 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f1,
|
|
|
|
"unknown Dell", STAC_9205_DELL_M42),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f2,
|
|
|
|
"unknown Dell", STAC_9205_DELL_M42),
|
2007-08-13 06:50:29 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f8,
|
2007-09-17 15:52:38 -07:00
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x021c,
|
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
2007-08-13 06:50:29 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f9,
|
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
2007-09-10 14:09:42 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x021b,
|
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
2007-08-13 06:50:29 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01fa,
|
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
2007-08-23 10:04:28 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01fc,
|
|
|
|
"unknown Dell", STAC_9205_DELL_M42),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01fd,
|
|
|
|
"unknown Dell", STAC_9205_DELL_M42),
|
2007-08-13 06:50:29 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01fe,
|
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01ff,
|
2007-08-23 10:04:28 -07:00
|
|
|
"Dell Precision M4300", STAC_9205_DELL_M43),
|
2007-08-13 06:50:29 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x0206,
|
|
|
|
"Dell Precision", STAC_9205_DELL_M43),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f1,
|
|
|
|
"Dell Inspiron", STAC_9205_DELL_M44),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01f2,
|
|
|
|
"Dell Inspiron", STAC_9205_DELL_M44),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01fc,
|
|
|
|
"Dell Inspiron", STAC_9205_DELL_M44),
|
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x01fd,
|
|
|
|
"Dell Inspiron", STAC_9205_DELL_M44),
|
2007-08-23 10:04:28 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x0204,
|
|
|
|
"unknown Dell", STAC_9205_DELL_M42),
|
2007-08-13 06:50:29 -07:00
|
|
|
SND_PCI_QUIRK(PCI_VENDOR_ID_DELL, 0x021f,
|
|
|
|
"Dell Inspiron", STAC_9205_DELL_M44),
|
2006-07-31 03:49:34 -07:00
|
|
|
{} /* terminator */
|
|
|
|
};
|
|
|
|
|
2006-08-23 09:31:34 -07:00
|
|
|
static int stac92xx_save_bios_config_regs(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
if (! spec->bios_pin_configs) {
|
|
|
|
spec->bios_pin_configs = kcalloc(spec->num_pins,
|
|
|
|
sizeof(*spec->bios_pin_configs), GFP_KERNEL);
|
|
|
|
if (! spec->bios_pin_configs)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < spec->num_pins; i++) {
|
|
|
|
hda_nid_t nid = spec->pin_nids[i];
|
|
|
|
unsigned int pin_cfg;
|
|
|
|
|
|
|
|
pin_cfg = snd_hda_codec_read(codec, nid, 0,
|
|
|
|
AC_VERB_GET_CONFIG_DEFAULT, 0x00);
|
|
|
|
snd_printdd(KERN_INFO "hda_codec: pin nid %2.2x bios pin config %8.8x\n",
|
|
|
|
nid, pin_cfg);
|
|
|
|
spec->bios_pin_configs[i] = pin_cfg;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-17 02:52:24 -07:00
|
|
|
static void stac92xx_set_config_reg(struct hda_codec *codec,
|
|
|
|
hda_nid_t pin_nid, unsigned int pin_config)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
snd_hda_codec_write(codec, pin_nid, 0,
|
|
|
|
AC_VERB_SET_CONFIG_DEFAULT_BYTES_0,
|
|
|
|
pin_config & 0x000000ff);
|
|
|
|
snd_hda_codec_write(codec, pin_nid, 0,
|
|
|
|
AC_VERB_SET_CONFIG_DEFAULT_BYTES_1,
|
|
|
|
(pin_config & 0x0000ff00) >> 8);
|
|
|
|
snd_hda_codec_write(codec, pin_nid, 0,
|
|
|
|
AC_VERB_SET_CONFIG_DEFAULT_BYTES_2,
|
|
|
|
(pin_config & 0x00ff0000) >> 16);
|
|
|
|
snd_hda_codec_write(codec, pin_nid, 0,
|
|
|
|
AC_VERB_SET_CONFIG_DEFAULT_BYTES_3,
|
|
|
|
pin_config >> 24);
|
|
|
|
i = snd_hda_codec_read(codec, pin_nid, 0,
|
|
|
|
AC_VERB_GET_CONFIG_DEFAULT,
|
|
|
|
0x00);
|
|
|
|
snd_printdd(KERN_INFO "hda_codec: pin nid %2.2x pin config %8.8x\n",
|
|
|
|
pin_nid, i);
|
|
|
|
}
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static void stac92xx_set_config_regs(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
2007-07-17 02:52:24 -07:00
|
|
|
if (!spec->pin_configs)
|
|
|
|
return;
|
2006-08-23 09:31:34 -07:00
|
|
|
|
2007-07-17 02:52:24 -07:00
|
|
|
for (i = 0; i < spec->num_pins; i++)
|
|
|
|
stac92xx_set_config_reg(codec, spec->pin_nids[i],
|
|
|
|
spec->pin_configs[i]);
|
2005-04-13 05:45:30 -07:00
|
|
|
}
|
|
|
|
|
2007-07-31 06:56:24 -07:00
|
|
|
static void stac92xx_enable_gpio_mask(struct hda_codec *codec)
|
2007-06-19 07:48:28 -07:00
|
|
|
{
|
2007-07-31 06:56:24 -07:00
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2007-07-17 02:52:24 -07:00
|
|
|
/* Configure GPIOx as output */
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, codec->afg, 0,
|
|
|
|
AC_VERB_SET_GPIO_DIRECTION, spec->gpio_mask);
|
2007-07-17 02:52:24 -07:00
|
|
|
/* Configure GPIOx as CMOS */
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, codec->afg, 0, 0x7e7, 0x00000000);
|
2007-07-17 02:52:24 -07:00
|
|
|
/* Assert GPIOx */
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, codec->afg, 0,
|
|
|
|
AC_VERB_SET_GPIO_DATA, spec->gpio_data);
|
2007-07-17 02:52:24 -07:00
|
|
|
/* Enable GPIOx */
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, codec->afg, 0,
|
|
|
|
AC_VERB_SET_GPIO_MASK, spec->gpio_mask);
|
2007-06-19 07:48:28 -07:00
|
|
|
}
|
|
|
|
|
2005-06-14 01:19:34 -07:00
|
|
|
/*
|
2005-06-27 05:59:41 -07:00
|
|
|
* Analog playback callbacks
|
2005-06-14 01:19:34 -07:00
|
|
|
*/
|
2005-06-27 05:59:41 -07:00
|
|
|
static int stac92xx_playback_pcm_open(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
2005-06-14 01:19:34 -07:00
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2005-06-27 05:59:41 -07:00
|
|
|
return snd_hda_multi_out_analog_open(codec, &spec->multiout, substream);
|
2005-04-13 05:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
|
|
|
unsigned int stream_tag,
|
|
|
|
unsigned int format,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2005-11-29 07:00:51 -07:00
|
|
|
return snd_hda_multi_out_analog_prepare(codec, &spec->multiout, stream_tag, format, substream);
|
2005-04-13 05:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_playback_pcm_cleanup(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
return snd_hda_multi_out_analog_cleanup(codec, &spec->multiout);
|
|
|
|
}
|
|
|
|
|
2005-06-14 01:19:34 -07:00
|
|
|
/*
|
|
|
|
* Digital playback callbacks
|
|
|
|
*/
|
|
|
|
static int stac92xx_dig_playback_pcm_open(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-06-14 01:19:34 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
return snd_hda_multi_out_dig_open(codec, &spec->multiout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_dig_playback_pcm_close(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-06-14 01:19:34 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
return snd_hda_multi_out_dig_close(codec, &spec->multiout);
|
|
|
|
}
|
|
|
|
|
2007-04-05 05:51:48 -07:00
|
|
|
static int stac92xx_dig_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
|
|
|
unsigned int stream_tag,
|
|
|
|
unsigned int format,
|
|
|
|
struct snd_pcm_substream *substream)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
|
|
|
|
stream_tag, format, substream);
|
|
|
|
}
|
|
|
|
|
2005-06-14 01:19:34 -07:00
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
/*
|
|
|
|
* Analog capture callbacks
|
|
|
|
*/
|
|
|
|
static int stac92xx_capture_pcm_prepare(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
|
|
|
unsigned int stream_tag,
|
|
|
|
unsigned int format,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number],
|
|
|
|
stream_tag, 0, format);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_capture_pcm_cleanup(struct hda_pcm_stream *hinfo,
|
|
|
|
struct hda_codec *codec,
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_pcm_substream *substream)
|
2005-04-13 05:45:30 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
snd_hda_codec_setup_stream(codec, spec->adc_nids[substream->number], 0, 0, 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-06-14 01:19:34 -07:00
|
|
|
static struct hda_pcm_stream stac92xx_pcm_digital_playback = {
|
|
|
|
.substreams = 1,
|
|
|
|
.channels_min = 2,
|
|
|
|
.channels_max = 2,
|
|
|
|
/* NID is set in stac92xx_build_pcms */
|
|
|
|
.ops = {
|
|
|
|
.open = stac92xx_dig_playback_pcm_open,
|
2007-04-05 05:51:48 -07:00
|
|
|
.close = stac92xx_dig_playback_pcm_close,
|
|
|
|
.prepare = stac92xx_dig_playback_pcm_prepare
|
2005-06-14 01:19:34 -07:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct hda_pcm_stream stac92xx_pcm_digital_capture = {
|
|
|
|
.substreams = 1,
|
|
|
|
.channels_min = 2,
|
|
|
|
.channels_max = 2,
|
|
|
|
/* NID is set in stac92xx_build_pcms */
|
|
|
|
};
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static struct hda_pcm_stream stac92xx_pcm_analog_playback = {
|
|
|
|
.substreams = 1,
|
|
|
|
.channels_min = 2,
|
2005-06-27 05:59:41 -07:00
|
|
|
.channels_max = 8,
|
2005-04-13 05:45:30 -07:00
|
|
|
.nid = 0x02, /* NID to query formats and rates */
|
|
|
|
.ops = {
|
|
|
|
.open = stac92xx_playback_pcm_open,
|
|
|
|
.prepare = stac92xx_playback_pcm_prepare,
|
|
|
|
.cleanup = stac92xx_playback_pcm_cleanup
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
static struct hda_pcm_stream stac92xx_pcm_analog_alt_playback = {
|
|
|
|
.substreams = 1,
|
|
|
|
.channels_min = 2,
|
|
|
|
.channels_max = 2,
|
|
|
|
.nid = 0x06, /* NID to query formats and rates */
|
|
|
|
.ops = {
|
|
|
|
.open = stac92xx_playback_pcm_open,
|
|
|
|
.prepare = stac92xx_playback_pcm_prepare,
|
|
|
|
.cleanup = stac92xx_playback_pcm_cleanup
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static struct hda_pcm_stream stac92xx_pcm_analog_capture = {
|
|
|
|
.channels_min = 2,
|
|
|
|
.channels_max = 2,
|
2007-09-03 06:31:02 -07:00
|
|
|
/* NID + .substreams is set in stac92xx_build_pcms */
|
2005-04-13 05:45:30 -07:00
|
|
|
.ops = {
|
|
|
|
.prepare = stac92xx_capture_pcm_prepare,
|
|
|
|
.cleanup = stac92xx_capture_pcm_cleanup
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static int stac92xx_build_pcms(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
struct hda_pcm *info = spec->pcm_rec;
|
|
|
|
|
|
|
|
codec->num_pcms = 1;
|
|
|
|
codec->pcm_info = info;
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
info->name = "STAC92xx Analog";
|
2005-04-13 05:45:30 -07:00
|
|
|
info->stream[SNDRV_PCM_STREAM_PLAYBACK] = stac92xx_pcm_analog_playback;
|
|
|
|
info->stream[SNDRV_PCM_STREAM_CAPTURE] = stac92xx_pcm_analog_capture;
|
2006-01-23 07:27:49 -07:00
|
|
|
info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->adc_nids[0];
|
2007-09-03 06:31:02 -07:00
|
|
|
info->stream[SNDRV_PCM_STREAM_CAPTURE].substreams = spec->num_adcs;
|
2006-01-23 07:27:49 -07:00
|
|
|
|
|
|
|
if (spec->alt_switch) {
|
|
|
|
codec->num_pcms++;
|
|
|
|
info++;
|
|
|
|
info->name = "STAC92xx Analog Alt";
|
|
|
|
info->stream[SNDRV_PCM_STREAM_PLAYBACK] = stac92xx_pcm_analog_alt_playback;
|
|
|
|
}
|
2005-04-13 05:45:30 -07:00
|
|
|
|
2005-06-14 01:19:34 -07:00
|
|
|
if (spec->multiout.dig_out_nid || spec->dig_in_nid) {
|
|
|
|
codec->num_pcms++;
|
|
|
|
info++;
|
|
|
|
info->name = "STAC92xx Digital";
|
|
|
|
if (spec->multiout.dig_out_nid) {
|
|
|
|
info->stream[SNDRV_PCM_STREAM_PLAYBACK] = stac92xx_pcm_digital_playback;
|
|
|
|
info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = spec->multiout.dig_out_nid;
|
|
|
|
}
|
|
|
|
if (spec->dig_in_nid) {
|
|
|
|
info->stream[SNDRV_PCM_STREAM_CAPTURE] = stac92xx_pcm_digital_capture;
|
|
|
|
info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = spec->dig_in_nid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-03-23 09:06:28 -07:00
|
|
|
static unsigned int stac92xx_get_vref(struct hda_codec *codec, hda_nid_t nid)
|
|
|
|
{
|
|
|
|
unsigned int pincap = snd_hda_param_read(codec, nid,
|
|
|
|
AC_PAR_PIN_CAP);
|
|
|
|
pincap = (pincap & AC_PINCAP_VREF) >> AC_PINCAP_VREF_SHIFT;
|
|
|
|
if (pincap & AC_PINCAP_VREF_100)
|
|
|
|
return AC_PINCTL_VREF_100;
|
|
|
|
if (pincap & AC_PINCAP_VREF_80)
|
|
|
|
return AC_PINCTL_VREF_80;
|
|
|
|
if (pincap & AC_PINCAP_VREF_50)
|
|
|
|
return AC_PINCTL_VREF_50;
|
|
|
|
if (pincap & AC_PINCAP_VREF_GRD)
|
|
|
|
return AC_PINCTL_VREF_GRD;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
static void stac92xx_auto_set_pinctl(struct hda_codec *codec, hda_nid_t nid, int pin_type)
|
|
|
|
|
|
|
|
{
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, nid, 0,
|
|
|
|
AC_VERB_SET_PIN_WIDGET_CONTROL, pin_type);
|
2005-11-29 07:00:51 -07:00
|
|
|
}
|
|
|
|
|
2007-07-23 06:42:26 -07:00
|
|
|
#define stac92xx_io_switch_info snd_ctl_boolean_mono_info
|
2005-11-29 07:00:51 -07:00
|
|
|
|
|
|
|
static int stac92xx_io_switch_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int io_idx = kcontrol-> private_value & 0xff;
|
|
|
|
|
|
|
|
ucontrol->value.integer.value[0] = spec->io_switch[io_idx];
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_io_switch_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
hda_nid_t nid = kcontrol->private_value >> 8;
|
|
|
|
int io_idx = kcontrol-> private_value & 0xff;
|
|
|
|
unsigned short val = ucontrol->value.integer.value[0];
|
|
|
|
|
|
|
|
spec->io_switch[io_idx] = val;
|
|
|
|
|
|
|
|
if (val)
|
|
|
|
stac92xx_auto_set_pinctl(codec, nid, AC_PINCTL_OUT_EN);
|
2006-03-23 09:06:28 -07:00
|
|
|
else {
|
|
|
|
unsigned int pinctl = AC_PINCTL_IN_EN;
|
|
|
|
if (io_idx) /* set VREF for mic */
|
|
|
|
pinctl |= stac92xx_get_vref(codec, nid);
|
|
|
|
stac92xx_auto_set_pinctl(codec, nid, pinctl);
|
|
|
|
}
|
2005-11-29 07:00:51 -07:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2007-09-03 06:29:04 -07:00
|
|
|
#define stac92xx_clfe_switch_info snd_ctl_boolean_mono_info
|
|
|
|
|
|
|
|
static int stac92xx_clfe_switch_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
|
|
|
ucontrol->value.integer.value[0] = spec->clfe_swap;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int stac92xx_clfe_switch_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
|
|
|
{
|
|
|
|
struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
hda_nid_t nid = kcontrol->private_value & 0xff;
|
|
|
|
|
|
|
|
if (spec->clfe_swap == ucontrol->value.integer.value[0])
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
spec->clfe_swap = ucontrol->value.integer.value[0];
|
|
|
|
|
|
|
|
snd_hda_codec_write_cache(codec, nid, 0, AC_VERB_SET_EAPD_BTLENABLE,
|
|
|
|
spec->clfe_swap ? 0x4 : 0x0);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
#define STAC_CODEC_IO_SWITCH(xname, xpval) \
|
|
|
|
{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
|
|
|
|
.name = xname, \
|
|
|
|
.index = 0, \
|
|
|
|
.info = stac92xx_io_switch_info, \
|
|
|
|
.get = stac92xx_io_switch_get, \
|
|
|
|
.put = stac92xx_io_switch_put, \
|
|
|
|
.private_value = xpval, \
|
|
|
|
}
|
|
|
|
|
2007-09-03 06:29:04 -07:00
|
|
|
#define STAC_CODEC_CLFE_SWITCH(xname, xpval) \
|
|
|
|
{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \
|
|
|
|
.name = xname, \
|
|
|
|
.index = 0, \
|
|
|
|
.info = stac92xx_clfe_switch_info, \
|
|
|
|
.get = stac92xx_clfe_switch_get, \
|
|
|
|
.put = stac92xx_clfe_switch_put, \
|
|
|
|
.private_value = xpval, \
|
|
|
|
}
|
2005-11-29 07:00:51 -07:00
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
enum {
|
|
|
|
STAC_CTL_WIDGET_VOL,
|
|
|
|
STAC_CTL_WIDGET_MUTE,
|
2005-11-29 07:00:51 -07:00
|
|
|
STAC_CTL_WIDGET_IO_SWITCH,
|
2007-09-03 06:29:04 -07:00
|
|
|
STAC_CTL_WIDGET_CLFE_SWITCH
|
2005-06-27 05:59:41 -07:00
|
|
|
};
|
|
|
|
|
2005-11-17 06:57:47 -07:00
|
|
|
static struct snd_kcontrol_new stac92xx_control_templates[] = {
|
2005-06-27 05:59:41 -07:00
|
|
|
HDA_CODEC_VOLUME(NULL, 0, 0, 0),
|
|
|
|
HDA_CODEC_MUTE(NULL, 0, 0, 0),
|
2005-11-29 07:00:51 -07:00
|
|
|
STAC_CODEC_IO_SWITCH(NULL, 0),
|
2007-09-03 06:29:04 -07:00
|
|
|
STAC_CODEC_CLFE_SWITCH(NULL, 0),
|
2005-06-27 05:59:41 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/* add dynamic controls */
|
|
|
|
static int stac92xx_add_control(struct sigmatel_spec *spec, int type, const char *name, unsigned long val)
|
|
|
|
{
|
2005-11-17 06:57:47 -07:00
|
|
|
struct snd_kcontrol_new *knew;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
if (spec->num_kctl_used >= spec->num_kctl_alloc) {
|
|
|
|
int num = spec->num_kctl_alloc + NUM_CONTROL_ALLOC;
|
|
|
|
|
|
|
|
knew = kcalloc(num + 1, sizeof(*knew), GFP_KERNEL); /* array + terminator */
|
|
|
|
if (! knew)
|
|
|
|
return -ENOMEM;
|
|
|
|
if (spec->kctl_alloc) {
|
|
|
|
memcpy(knew, spec->kctl_alloc, sizeof(*knew) * spec->num_kctl_alloc);
|
|
|
|
kfree(spec->kctl_alloc);
|
|
|
|
}
|
|
|
|
spec->kctl_alloc = knew;
|
|
|
|
spec->num_kctl_alloc = num;
|
|
|
|
}
|
|
|
|
|
|
|
|
knew = &spec->kctl_alloc[spec->num_kctl_used];
|
|
|
|
*knew = stac92xx_control_templates[type];
|
2005-06-30 01:54:33 -07:00
|
|
|
knew->name = kstrdup(name, GFP_KERNEL);
|
2005-06-27 05:59:41 -07:00
|
|
|
if (! knew->name)
|
|
|
|
return -ENOMEM;
|
|
|
|
knew->private_value = val;
|
|
|
|
spec->num_kctl_used++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
/* flag inputs as additional dynamic lineouts */
|
|
|
|
static int stac92xx_add_dyn_out_pins(struct hda_codec *codec, struct auto_pin_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
unsigned int wcaps, wtype;
|
|
|
|
int i, num_dacs = 0;
|
|
|
|
|
|
|
|
/* use the wcaps cache to count all DACs available for line-outs */
|
|
|
|
for (i = 0; i < codec->num_nodes; i++) {
|
|
|
|
wcaps = codec->wcaps[i];
|
|
|
|
wtype = (wcaps & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT;
|
|
|
|
if (wtype == AC_WID_AUD_OUT && !(wcaps & AC_WCAP_DIGITAL))
|
|
|
|
num_dacs++;
|
|
|
|
}
|
2005-11-29 07:00:51 -07:00
|
|
|
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
snd_printdd("%s: total dac count=%d\n", __func__, num_dacs);
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
switch (cfg->line_outs) {
|
|
|
|
case 3:
|
|
|
|
/* add line-in as side */
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (cfg->input_pins[AUTO_PIN_LINE] && num_dacs > 3) {
|
2007-09-03 00:43:38 -07:00
|
|
|
cfg->line_out_pins[cfg->line_outs] =
|
|
|
|
cfg->input_pins[AUTO_PIN_LINE];
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->line_switch = 1;
|
|
|
|
cfg->line_outs++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* add line-in as clfe and mic as side */
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (cfg->input_pins[AUTO_PIN_LINE] && num_dacs > 2) {
|
2007-09-03 00:43:38 -07:00
|
|
|
cfg->line_out_pins[cfg->line_outs] =
|
|
|
|
cfg->input_pins[AUTO_PIN_LINE];
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->line_switch = 1;
|
|
|
|
cfg->line_outs++;
|
|
|
|
}
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (cfg->input_pins[AUTO_PIN_MIC] && num_dacs > 3) {
|
2007-09-03 00:43:38 -07:00
|
|
|
cfg->line_out_pins[cfg->line_outs] =
|
|
|
|
cfg->input_pins[AUTO_PIN_MIC];
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->mic_switch = 1;
|
|
|
|
cfg->line_outs++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
/* add line-in as surr and mic as clfe */
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (cfg->input_pins[AUTO_PIN_LINE] && num_dacs > 1) {
|
2007-09-03 00:43:38 -07:00
|
|
|
cfg->line_out_pins[cfg->line_outs] =
|
|
|
|
cfg->input_pins[AUTO_PIN_LINE];
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->line_switch = 1;
|
|
|
|
cfg->line_outs++;
|
|
|
|
}
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (cfg->input_pins[AUTO_PIN_MIC] && num_dacs > 2) {
|
2007-09-03 00:43:38 -07:00
|
|
|
cfg->line_out_pins[cfg->line_outs] =
|
|
|
|
cfg->input_pins[AUTO_PIN_MIC];
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->mic_switch = 1;
|
|
|
|
cfg->line_outs++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
|
|
|
|
static int is_in_dac_nids(struct sigmatel_spec *spec, hda_nid_t nid)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < spec->multiout.num_dacs; i++) {
|
|
|
|
if (spec->multiout.dac_nids[i] == nid)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
/*
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
* Fill in the dac_nids table from the parsed pin configuration
|
|
|
|
* This function only works when every pin in line_out_pins[]
|
|
|
|
* contains atleast one DAC in its connection list. Some 92xx
|
|
|
|
* codecs are not connected directly to a DAC, such as the 9200
|
|
|
|
* and 9202/925x. For those, dac_nids[] must be hard-coded.
|
2006-01-23 07:27:49 -07:00
|
|
|
*/
|
2006-06-28 06:52:16 -07:00
|
|
|
static int stac92xx_auto_fill_dac_nids(struct hda_codec *codec,
|
2007-07-02 10:18:00 -07:00
|
|
|
struct auto_pin_cfg *cfg)
|
2005-06-27 05:59:41 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
int i, j, conn_len = 0;
|
|
|
|
hda_nid_t nid, conn[HDA_MAX_CONNECTIONS];
|
|
|
|
unsigned int wcaps, wtype;
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
for (i = 0; i < cfg->line_outs; i++) {
|
|
|
|
nid = cfg->line_out_pins[i];
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
conn_len = snd_hda_get_connections(codec, nid, conn,
|
|
|
|
HDA_MAX_CONNECTIONS);
|
|
|
|
for (j = 0; j < conn_len; j++) {
|
|
|
|
wcaps = snd_hda_param_read(codec, conn[j],
|
|
|
|
AC_PAR_AUDIO_WIDGET_CAP);
|
|
|
|
wtype = (wcaps & AC_WCAP_TYPE) >> AC_WCAP_TYPE_SHIFT;
|
|
|
|
|
|
|
|
if (wtype != AC_WID_AUD_OUT ||
|
|
|
|
(wcaps & AC_WCAP_DIGITAL))
|
|
|
|
continue;
|
|
|
|
/* conn[j] is a DAC routed to this line-out */
|
|
|
|
if (!is_in_dac_nids(spec, conn[j]))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (j == conn_len) {
|
2007-07-02 10:18:00 -07:00
|
|
|
if (spec->multiout.num_dacs > 0) {
|
|
|
|
/* we have already working output pins,
|
|
|
|
* so let's drop the broken ones again
|
|
|
|
*/
|
|
|
|
cfg->line_outs = spec->multiout.num_dacs;
|
|
|
|
break;
|
|
|
|
}
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
/* error out, no available DAC found */
|
|
|
|
snd_printk(KERN_ERR
|
|
|
|
"%s: No available DAC for pin 0x%x\n",
|
|
|
|
__func__, nid);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
spec->multiout.dac_nids[i] = conn[j];
|
|
|
|
spec->multiout.num_dacs++;
|
|
|
|
if (conn_len > 1) {
|
|
|
|
/* select this DAC in the pin's input mux */
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, nid, 0,
|
|
|
|
AC_VERB_SET_CONNECT_SEL, j);
|
2005-06-27 05:59:41 -07:00
|
|
|
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
}
|
|
|
|
}
|
2005-06-27 05:59:41 -07:00
|
|
|
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
snd_printd("dac_nids=%d (0x%x/0x%x/0x%x/0x%x/0x%x)\n",
|
|
|
|
spec->multiout.num_dacs,
|
|
|
|
spec->multiout.dac_nids[0],
|
|
|
|
spec->multiout.dac_nids[1],
|
|
|
|
spec->multiout.dac_nids[2],
|
|
|
|
spec->multiout.dac_nids[3],
|
|
|
|
spec->multiout.dac_nids[4]);
|
2005-06-27 05:59:41 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-09-20 08:10:27 -07:00
|
|
|
/* create volume control/switch for the given prefx type */
|
|
|
|
static int create_controls(struct sigmatel_spec *spec, const char *pfx, hda_nid_t nid, int chs)
|
|
|
|
{
|
|
|
|
char name[32];
|
|
|
|
int err;
|
|
|
|
|
|
|
|
sprintf(name, "%s Playback Volume", pfx);
|
|
|
|
err = stac92xx_add_control(spec, STAC_CTL_WIDGET_VOL, name,
|
|
|
|
HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
sprintf(name, "%s Playback Switch", pfx);
|
|
|
|
err = stac92xx_add_control(spec, STAC_CTL_WIDGET_MUTE, name,
|
|
|
|
HDA_COMPOSE_AMP_VAL(nid, chs, 0, HDA_OUTPUT));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
/* add playback controls from the parsed DAC table */
|
2007-09-03 06:29:04 -07:00
|
|
|
static int stac92xx_auto_create_multi_out_ctls(struct hda_codec *codec,
|
2006-06-28 06:52:16 -07:00
|
|
|
const struct auto_pin_cfg *cfg)
|
2005-06-27 05:59:41 -07:00
|
|
|
{
|
2006-06-28 06:52:16 -07:00
|
|
|
static const char *chname[4] = {
|
|
|
|
"Front", "Surround", NULL /*CLFE*/, "Side"
|
|
|
|
};
|
2005-06-27 05:59:41 -07:00
|
|
|
hda_nid_t nid;
|
|
|
|
int i, err;
|
|
|
|
|
2007-09-03 06:29:04 -07:00
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
unsigned int wid_caps;
|
|
|
|
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
for (i = 0; i < cfg->line_outs; i++) {
|
2005-11-29 07:00:51 -07:00
|
|
|
if (!spec->multiout.dac_nids[i])
|
2005-06-27 05:59:41 -07:00
|
|
|
continue;
|
|
|
|
|
|
|
|
nid = spec->multiout.dac_nids[i];
|
|
|
|
|
|
|
|
if (i == 2) {
|
|
|
|
/* Center/LFE */
|
2006-09-20 08:10:27 -07:00
|
|
|
err = create_controls(spec, "Center", nid, 1);
|
|
|
|
if (err < 0)
|
2005-06-27 05:59:41 -07:00
|
|
|
return err;
|
2006-09-20 08:10:27 -07:00
|
|
|
err = create_controls(spec, "LFE", nid, 2);
|
|
|
|
if (err < 0)
|
2005-06-27 05:59:41 -07:00
|
|
|
return err;
|
2007-09-03 06:29:04 -07:00
|
|
|
|
|
|
|
wid_caps = get_wcaps(codec, nid);
|
|
|
|
|
|
|
|
if (wid_caps & AC_WCAP_LR_SWAP) {
|
|
|
|
err = stac92xx_add_control(spec,
|
|
|
|
STAC_CTL_WIDGET_CLFE_SWITCH,
|
|
|
|
"Swap Center/LFE Playback Switch", nid);
|
|
|
|
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
} else {
|
2006-09-20 08:10:27 -07:00
|
|
|
err = create_controls(spec, chname[i], nid, 3);
|
|
|
|
if (err < 0)
|
2005-06-27 05:59:41 -07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
if (spec->line_switch)
|
|
|
|
if ((err = stac92xx_add_control(spec, STAC_CTL_WIDGET_IO_SWITCH, "Line In as Output Switch", cfg->input_pins[AUTO_PIN_LINE] << 8)) < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (spec->mic_switch)
|
|
|
|
if ((err = stac92xx_add_control(spec, STAC_CTL_WIDGET_IO_SWITCH, "Mic as Output Switch", (cfg->input_pins[AUTO_PIN_MIC] << 8) | 1)) < 0)
|
|
|
|
return err;
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-09-20 08:10:27 -07:00
|
|
|
static int check_in_dac_nids(struct sigmatel_spec *spec, hda_nid_t nid)
|
2005-06-27 05:59:41 -07:00
|
|
|
{
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (is_in_dac_nids(spec, nid))
|
|
|
|
return 1;
|
2006-09-20 08:10:27 -07:00
|
|
|
if (spec->multiout.hp_nid == nid)
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-06-27 05:59:41 -07:00
|
|
|
|
2006-09-20 08:10:27 -07:00
|
|
|
static int add_spec_dacs(struct sigmatel_spec *spec, hda_nid_t nid)
|
|
|
|
{
|
|
|
|
if (!spec->multiout.hp_nid)
|
|
|
|
spec->multiout.hp_nid = nid;
|
|
|
|
else if (spec->multiout.num_dacs > 4) {
|
|
|
|
printk(KERN_WARNING "stac92xx: No space for DAC 0x%x\n", nid);
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
spec->multiout.dac_nids[spec->multiout.num_dacs] = nid;
|
|
|
|
spec->multiout.num_dacs++;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2005-07-04 08:51:39 -07:00
|
|
|
|
2006-09-20 08:10:27 -07:00
|
|
|
/* add playback controls for Speaker and HP outputs */
|
|
|
|
static int stac92xx_auto_create_hp_ctls(struct hda_codec *codec,
|
|
|
|
struct auto_pin_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
hda_nid_t nid;
|
|
|
|
int i, old_num_dacs, err;
|
|
|
|
|
|
|
|
old_num_dacs = spec->multiout.num_dacs;
|
|
|
|
for (i = 0; i < cfg->hp_outs; i++) {
|
|
|
|
unsigned int wid_caps = get_wcaps(codec, cfg->hp_pins[i]);
|
|
|
|
if (wid_caps & AC_WCAP_UNSOL_CAP)
|
|
|
|
spec->hp_detect = 1;
|
|
|
|
nid = snd_hda_codec_read(codec, cfg->hp_pins[i], 0,
|
|
|
|
AC_VERB_GET_CONNECT_LIST, 0) & 0xff;
|
|
|
|
if (check_in_dac_nids(spec, nid))
|
|
|
|
nid = 0;
|
|
|
|
if (! nid)
|
2005-06-27 05:59:41 -07:00
|
|
|
continue;
|
2006-09-20 08:10:27 -07:00
|
|
|
add_spec_dacs(spec, nid);
|
|
|
|
}
|
|
|
|
for (i = 0; i < cfg->speaker_outs; i++) {
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
nid = snd_hda_codec_read(codec, cfg->speaker_pins[i], 0,
|
2006-09-20 08:10:27 -07:00
|
|
|
AC_VERB_GET_CONNECT_LIST, 0) & 0xff;
|
|
|
|
if (check_in_dac_nids(spec, nid))
|
|
|
|
nid = 0;
|
|
|
|
if (! nid)
|
|
|
|
continue;
|
|
|
|
add_spec_dacs(spec, nid);
|
2005-06-27 05:59:41 -07:00
|
|
|
}
|
2007-07-12 06:17:34 -07:00
|
|
|
for (i = 0; i < cfg->line_outs; i++) {
|
|
|
|
nid = snd_hda_codec_read(codec, cfg->line_out_pins[i], 0,
|
|
|
|
AC_VERB_GET_CONNECT_LIST, 0) & 0xff;
|
|
|
|
if (check_in_dac_nids(spec, nid))
|
|
|
|
nid = 0;
|
|
|
|
if (! nid)
|
|
|
|
continue;
|
|
|
|
add_spec_dacs(spec, nid);
|
|
|
|
}
|
2006-09-20 08:10:27 -07:00
|
|
|
for (i = old_num_dacs; i < spec->multiout.num_dacs; i++) {
|
|
|
|
static const char *pfxs[] = {
|
|
|
|
"Speaker", "External Speaker", "Speaker2",
|
|
|
|
};
|
|
|
|
err = create_controls(spec, pfxs[i - old_num_dacs],
|
|
|
|
spec->multiout.dac_nids[i], 3);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
if (spec->multiout.hp_nid) {
|
|
|
|
const char *pfx;
|
2007-11-19 03:56:26 -07:00
|
|
|
if (old_num_dacs == spec->multiout.num_dacs)
|
2006-09-20 08:10:27 -07:00
|
|
|
pfx = "Master";
|
|
|
|
else
|
|
|
|
pfx = "Headphone";
|
|
|
|
err = create_controls(spec, pfx, spec->multiout.hp_nid, 3);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-10-26 08:12:59 -07:00
|
|
|
/* labels for dmic mux inputs */
|
2006-11-20 04:03:44 -07:00
|
|
|
static const char *stac92xx_dmic_labels[5] = {
|
2006-10-26 08:12:59 -07:00
|
|
|
"Analog Inputs", "Digital Mic 1", "Digital Mic 2",
|
|
|
|
"Digital Mic 3", "Digital Mic 4"
|
|
|
|
};
|
|
|
|
|
|
|
|
/* create playback/capture controls for input pins on dmic capable codecs */
|
|
|
|
static int stac92xx_auto_create_dmic_input_ctls(struct hda_codec *codec,
|
|
|
|
const struct auto_pin_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
struct hda_input_mux *dimux = &spec->private_dimux;
|
|
|
|
hda_nid_t con_lst[HDA_MAX_NUM_INPUTS];
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
dimux->items[dimux->num_items].label = stac92xx_dmic_labels[0];
|
|
|
|
dimux->items[dimux->num_items].index = 0;
|
|
|
|
dimux->num_items++;
|
|
|
|
|
|
|
|
for (i = 0; i < spec->num_dmics; i++) {
|
|
|
|
int index;
|
|
|
|
int num_cons;
|
|
|
|
unsigned int def_conf;
|
|
|
|
|
|
|
|
def_conf = snd_hda_codec_read(codec,
|
|
|
|
spec->dmic_nids[i],
|
|
|
|
0,
|
|
|
|
AC_VERB_GET_CONFIG_DEFAULT,
|
|
|
|
0);
|
|
|
|
if (get_defcfg_connect(def_conf) == AC_JACK_PORT_NONE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
num_cons = snd_hda_get_connections(codec,
|
|
|
|
spec->dmux_nid,
|
|
|
|
con_lst,
|
|
|
|
HDA_MAX_NUM_INPUTS);
|
|
|
|
for (j = 0; j < num_cons; j++)
|
|
|
|
if (con_lst[j] == spec->dmic_nids[i]) {
|
|
|
|
index = j;
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
found:
|
|
|
|
dimux->items[dimux->num_items].label =
|
|
|
|
stac92xx_dmic_labels[dimux->num_items];
|
|
|
|
dimux->items[dimux->num_items].index = index;
|
|
|
|
dimux->num_items++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
/* create playback/capture controls for input pins */
|
|
|
|
static int stac92xx_auto_create_analog_input_ctls(struct hda_codec *codec, const struct auto_pin_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
struct hda_input_mux *imux = &spec->private_imux;
|
|
|
|
hda_nid_t con_lst[HDA_MAX_NUM_INPUTS];
|
|
|
|
int i, j, k;
|
|
|
|
|
|
|
|
for (i = 0; i < AUTO_PIN_LAST; i++) {
|
2006-09-21 02:56:18 -07:00
|
|
|
int index;
|
|
|
|
|
|
|
|
if (!cfg->input_pins[i])
|
|
|
|
continue;
|
|
|
|
index = -1;
|
|
|
|
for (j = 0; j < spec->num_muxes; j++) {
|
|
|
|
int num_cons;
|
|
|
|
num_cons = snd_hda_get_connections(codec,
|
|
|
|
spec->mux_nids[j],
|
|
|
|
con_lst,
|
|
|
|
HDA_MAX_NUM_INPUTS);
|
|
|
|
for (k = 0; k < num_cons; k++)
|
|
|
|
if (con_lst[k] == cfg->input_pins[i]) {
|
|
|
|
index = k;
|
|
|
|
goto found;
|
|
|
|
}
|
2005-06-27 05:59:41 -07:00
|
|
|
}
|
2006-09-21 02:56:18 -07:00
|
|
|
continue;
|
|
|
|
found:
|
|
|
|
imux->items[imux->num_items].label = auto_pin_cfg_labels[i];
|
|
|
|
imux->items[imux->num_items].index = index;
|
|
|
|
imux->num_items++;
|
2005-06-27 05:59:41 -07:00
|
|
|
}
|
|
|
|
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
if (imux->num_items) {
|
2006-05-10 06:09:17 -07:00
|
|
|
/*
|
|
|
|
* Set the current input for the muxes.
|
|
|
|
* The STAC9221 has two input muxes with identical source
|
|
|
|
* NID lists. Hopefully this won't get confused.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < spec->num_muxes; i++) {
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, spec->mux_nids[i], 0,
|
|
|
|
AC_VERB_SET_CONNECT_SEL,
|
|
|
|
imux->items[0].index);
|
2006-05-10 06:09:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stac92xx_auto_init_multi_out(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < spec->autocfg.line_outs; i++) {
|
|
|
|
hda_nid_t nid = spec->autocfg.line_out_pins[i];
|
|
|
|
stac92xx_auto_set_pinctl(codec, nid, AC_PINCTL_OUT_EN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stac92xx_auto_init_hp_out(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2006-09-20 08:10:27 -07:00
|
|
|
int i;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
2006-09-20 08:10:27 -07:00
|
|
|
for (i = 0; i < spec->autocfg.hp_outs; i++) {
|
|
|
|
hda_nid_t pin;
|
|
|
|
pin = spec->autocfg.hp_pins[i];
|
|
|
|
if (pin) /* connect to front */
|
|
|
|
stac92xx_auto_set_pinctl(codec, pin, AC_PINCTL_OUT_EN | AC_PINCTL_HP_EN);
|
|
|
|
}
|
|
|
|
for (i = 0; i < spec->autocfg.speaker_outs; i++) {
|
|
|
|
hda_nid_t pin;
|
|
|
|
pin = spec->autocfg.speaker_pins[i];
|
|
|
|
if (pin) /* connect to front */
|
|
|
|
stac92xx_auto_set_pinctl(codec, pin, AC_PINCTL_OUT_EN);
|
|
|
|
}
|
2005-06-27 05:59:41 -07:00
|
|
|
}
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
static int stac92xx_parse_auto_config(struct hda_codec *codec, hda_nid_t dig_out, hda_nid_t dig_in)
|
2005-06-27 05:59:41 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int err;
|
|
|
|
|
2006-10-26 08:12:59 -07:00
|
|
|
if ((err = snd_hda_parse_pin_def_config(codec,
|
|
|
|
&spec->autocfg,
|
|
|
|
spec->dmic_nids)) < 0)
|
2005-06-27 05:59:41 -07:00
|
|
|
return err;
|
2006-03-21 03:24:42 -07:00
|
|
|
if (! spec->autocfg.line_outs)
|
2006-01-25 11:20:50 -07:00
|
|
|
return 0; /* can't find valid pin config */
|
2006-06-28 06:52:16 -07:00
|
|
|
|
2005-11-29 07:00:51 -07:00
|
|
|
if ((err = stac92xx_add_dyn_out_pins(codec, &spec->autocfg)) < 0)
|
|
|
|
return err;
|
2006-06-28 06:52:16 -07:00
|
|
|
if (spec->multiout.num_dacs == 0)
|
|
|
|
if ((err = stac92xx_auto_fill_dac_nids(codec, &spec->autocfg)) < 0)
|
|
|
|
return err;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
2007-09-03 06:29:04 -07:00
|
|
|
err = stac92xx_auto_create_multi_out_ctls(codec, &spec->autocfg);
|
|
|
|
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = stac92xx_auto_create_hp_ctls(codec, &spec->autocfg);
|
|
|
|
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = stac92xx_auto_create_analog_input_ctls(codec, &spec->autocfg);
|
|
|
|
|
|
|
|
if (err < 0)
|
2005-06-27 05:59:41 -07:00
|
|
|
return err;
|
|
|
|
|
2006-10-26 08:12:59 -07:00
|
|
|
if (spec->num_dmics > 0)
|
|
|
|
if ((err = stac92xx_auto_create_dmic_input_ctls(codec,
|
|
|
|
&spec->autocfg)) < 0)
|
|
|
|
return err;
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
spec->multiout.max_channels = spec->multiout.num_dacs * 2;
|
2005-11-29 07:00:51 -07:00
|
|
|
if (spec->multiout.max_channels > 2)
|
2005-06-27 05:59:41 -07:00
|
|
|
spec->surr_switch = 1;
|
|
|
|
|
2006-03-21 03:24:42 -07:00
|
|
|
if (spec->autocfg.dig_out_pin)
|
2006-01-23 07:27:49 -07:00
|
|
|
spec->multiout.dig_out_nid = dig_out;
|
2006-03-21 03:24:42 -07:00
|
|
|
if (spec->autocfg.dig_in_pin)
|
2006-01-23 07:27:49 -07:00
|
|
|
spec->dig_in_nid = dig_in;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
if (spec->kctl_alloc)
|
|
|
|
spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
|
|
|
|
|
|
|
|
spec->input_mux = &spec->private_imux;
|
2006-10-26 08:12:59 -07:00
|
|
|
spec->dinput_mux = &spec->private_dimux;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-03-21 03:24:42 -07:00
|
|
|
/* add playback controls for HP output */
|
|
|
|
static int stac9200_auto_create_hp_ctls(struct hda_codec *codec,
|
|
|
|
struct auto_pin_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2006-09-20 08:10:27 -07:00
|
|
|
hda_nid_t pin = cfg->hp_pins[0];
|
2006-03-21 03:24:42 -07:00
|
|
|
unsigned int wid_caps;
|
|
|
|
|
|
|
|
if (! pin)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
wid_caps = get_wcaps(codec, pin);
|
2006-03-27 03:51:52 -07:00
|
|
|
if (wid_caps & AC_WCAP_UNSOL_CAP)
|
2006-03-21 03:24:42 -07:00
|
|
|
spec->hp_detect = 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-09-06 04:58:25 -07:00
|
|
|
/* add playback controls for LFE output */
|
|
|
|
static int stac9200_auto_create_lfe_ctls(struct hda_codec *codec,
|
|
|
|
struct auto_pin_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int err;
|
|
|
|
hda_nid_t lfe_pin = 0x0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* search speaker outs and line outs for a mono speaker pin
|
|
|
|
* with an amp. If one is found, add LFE controls
|
|
|
|
* for it.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < spec->autocfg.speaker_outs && lfe_pin == 0x0; i++) {
|
|
|
|
hda_nid_t pin = spec->autocfg.speaker_pins[i];
|
|
|
|
unsigned long wcaps = get_wcaps(codec, pin);
|
|
|
|
wcaps &= (AC_WCAP_STEREO | AC_WCAP_OUT_AMP);
|
|
|
|
if (wcaps == AC_WCAP_OUT_AMP)
|
|
|
|
/* found a mono speaker with an amp, must be lfe */
|
|
|
|
lfe_pin = pin;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if speaker_outs is 0, then speakers may be in line_outs */
|
|
|
|
if (lfe_pin == 0 && spec->autocfg.speaker_outs == 0) {
|
|
|
|
for (i = 0; i < spec->autocfg.line_outs && lfe_pin == 0x0; i++) {
|
|
|
|
hda_nid_t pin = spec->autocfg.line_out_pins[i];
|
|
|
|
unsigned long cfg;
|
|
|
|
cfg = snd_hda_codec_read(codec, pin, 0,
|
|
|
|
AC_VERB_GET_CONFIG_DEFAULT,
|
|
|
|
0x00);
|
|
|
|
if (get_defcfg_device(cfg) == AC_JACK_SPEAKER) {
|
|
|
|
unsigned long wcaps = get_wcaps(codec, pin);
|
|
|
|
wcaps &= (AC_WCAP_STEREO | AC_WCAP_OUT_AMP);
|
|
|
|
if (wcaps == AC_WCAP_OUT_AMP)
|
|
|
|
/* found a mono speaker with an amp,
|
|
|
|
must be lfe */
|
|
|
|
lfe_pin = pin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lfe_pin) {
|
2006-09-20 08:10:27 -07:00
|
|
|
err = create_controls(spec, "LFE", lfe_pin, 1);
|
2006-09-06 04:58:25 -07:00
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
static int stac9200_parse_auto_config(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int err;
|
|
|
|
|
2005-12-05 11:42:22 -07:00
|
|
|
if ((err = snd_hda_parse_pin_def_config(codec, &spec->autocfg, NULL)) < 0)
|
2005-06-27 05:59:41 -07:00
|
|
|
return err;
|
|
|
|
|
|
|
|
if ((err = stac92xx_auto_create_analog_input_ctls(codec, &spec->autocfg)) < 0)
|
|
|
|
return err;
|
|
|
|
|
2006-03-21 03:24:42 -07:00
|
|
|
if ((err = stac9200_auto_create_hp_ctls(codec, &spec->autocfg)) < 0)
|
|
|
|
return err;
|
|
|
|
|
2006-09-06 04:58:25 -07:00
|
|
|
if ((err = stac9200_auto_create_lfe_ctls(codec, &spec->autocfg)) < 0)
|
|
|
|
return err;
|
|
|
|
|
2006-03-21 03:24:42 -07:00
|
|
|
if (spec->autocfg.dig_out_pin)
|
2005-06-27 05:59:41 -07:00
|
|
|
spec->multiout.dig_out_nid = 0x05;
|
2006-03-21 03:24:42 -07:00
|
|
|
if (spec->autocfg.dig_in_pin)
|
2005-06-27 05:59:41 -07:00
|
|
|
spec->dig_in_nid = 0x04;
|
|
|
|
|
|
|
|
if (spec->kctl_alloc)
|
|
|
|
spec->mixers[spec->num_mixers++] = spec->kctl_alloc;
|
|
|
|
|
|
|
|
spec->input_mux = &spec->private_imux;
|
2006-10-26 08:12:59 -07:00
|
|
|
spec->dinput_mux = &spec->private_dimux;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-05-10 06:09:17 -07:00
|
|
|
/*
|
|
|
|
* Early 2006 Intel Macintoshes with STAC9220X5 codecs seem to have a
|
|
|
|
* funky external mute control using GPIO pins.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void stac922x_gpio_mute(struct hda_codec *codec, int pin, int muted)
|
|
|
|
{
|
|
|
|
unsigned int gpiostate, gpiomask, gpiodir;
|
|
|
|
|
|
|
|
gpiostate = snd_hda_codec_read(codec, codec->afg, 0,
|
|
|
|
AC_VERB_GET_GPIO_DATA, 0);
|
|
|
|
|
|
|
|
if (!muted)
|
|
|
|
gpiostate |= (1 << pin);
|
|
|
|
else
|
|
|
|
gpiostate &= ~(1 << pin);
|
|
|
|
|
|
|
|
gpiomask = snd_hda_codec_read(codec, codec->afg, 0,
|
|
|
|
AC_VERB_GET_GPIO_MASK, 0);
|
|
|
|
gpiomask |= (1 << pin);
|
|
|
|
|
|
|
|
gpiodir = snd_hda_codec_read(codec, codec->afg, 0,
|
|
|
|
AC_VERB_GET_GPIO_DIRECTION, 0);
|
|
|
|
gpiodir |= (1 << pin);
|
|
|
|
|
|
|
|
/* AppleHDA seems to do this -- WTF is this verb?? */
|
|
|
|
snd_hda_codec_write(codec, codec->afg, 0, 0x7e7, 0);
|
|
|
|
|
|
|
|
snd_hda_codec_write(codec, codec->afg, 0,
|
|
|
|
AC_VERB_SET_GPIO_MASK, gpiomask);
|
|
|
|
snd_hda_codec_write(codec, codec->afg, 0,
|
|
|
|
AC_VERB_SET_GPIO_DIRECTION, gpiodir);
|
|
|
|
|
|
|
|
msleep(1);
|
|
|
|
|
|
|
|
snd_hda_codec_write(codec, codec->afg, 0,
|
|
|
|
AC_VERB_SET_GPIO_DATA, gpiostate);
|
|
|
|
}
|
|
|
|
|
2006-09-21 02:56:18 -07:00
|
|
|
static void enable_pin_detect(struct hda_codec *codec, hda_nid_t nid,
|
|
|
|
unsigned int event)
|
|
|
|
{
|
|
|
|
if (get_wcaps(codec, nid) & AC_WCAP_UNSOL_CAP)
|
2007-09-03 00:36:36 -07:00
|
|
|
snd_hda_codec_write_cache(codec, nid, 0,
|
|
|
|
AC_VERB_SET_UNSOLICITED_ENABLE,
|
|
|
|
(AC_USRSP_EN | event));
|
2006-09-21 02:56:18 -07:00
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
static int stac92xx_init(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
2006-03-21 03:24:42 -07:00
|
|
|
struct auto_pin_cfg *cfg = &spec->autocfg;
|
|
|
|
int i;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
snd_hda_sequence_write(codec, spec->init);
|
|
|
|
|
2006-03-21 03:24:42 -07:00
|
|
|
/* set up pins */
|
|
|
|
if (spec->hp_detect) {
|
2006-03-27 03:51:52 -07:00
|
|
|
/* Enable unsolicited responses on the HP widget */
|
2006-09-20 08:10:27 -07:00
|
|
|
for (i = 0; i < cfg->hp_outs; i++)
|
2006-09-21 02:56:18 -07:00
|
|
|
enable_pin_detect(codec, cfg->hp_pins[i],
|
|
|
|
STAC_HP_EVENT);
|
2007-03-13 02:40:23 -07:00
|
|
|
/* force to enable the first line-out; the others are set up
|
|
|
|
* in unsol_event
|
|
|
|
*/
|
|
|
|
stac92xx_auto_set_pinctl(codec, spec->autocfg.line_out_pins[0],
|
|
|
|
AC_PINCTL_OUT_EN);
|
2006-09-21 05:28:21 -07:00
|
|
|
stac92xx_auto_init_hp_out(codec);
|
2006-03-21 03:24:42 -07:00
|
|
|
/* fake event to set up pins */
|
|
|
|
codec->patch_ops.unsol_event(codec, STAC_HP_EVENT << 26);
|
|
|
|
} else {
|
|
|
|
stac92xx_auto_init_multi_out(codec);
|
|
|
|
stac92xx_auto_init_hp_out(codec);
|
|
|
|
}
|
|
|
|
for (i = 0; i < AUTO_PIN_LAST; i++) {
|
2006-03-23 09:06:28 -07:00
|
|
|
hda_nid_t nid = cfg->input_pins[i];
|
|
|
|
if (nid) {
|
|
|
|
unsigned int pinctl = AC_PINCTL_IN_EN;
|
|
|
|
if (i == AUTO_PIN_MIC || i == AUTO_PIN_FRONT_MIC)
|
|
|
|
pinctl |= stac92xx_get_vref(codec, nid);
|
|
|
|
stac92xx_auto_set_pinctl(codec, nid, pinctl);
|
|
|
|
}
|
2006-03-21 03:24:42 -07:00
|
|
|
}
|
2006-10-26 08:12:59 -07:00
|
|
|
if (spec->num_dmics > 0)
|
|
|
|
for (i = 0; i < spec->num_dmics; i++)
|
|
|
|
stac92xx_auto_set_pinctl(codec, spec->dmic_nids[i],
|
|
|
|
AC_PINCTL_IN_EN);
|
|
|
|
|
2006-03-21 03:24:42 -07:00
|
|
|
if (cfg->dig_out_pin)
|
|
|
|
stac92xx_auto_set_pinctl(codec, cfg->dig_out_pin,
|
|
|
|
AC_PINCTL_OUT_EN);
|
|
|
|
if (cfg->dig_in_pin)
|
|
|
|
stac92xx_auto_set_pinctl(codec, cfg->dig_in_pin,
|
|
|
|
AC_PINCTL_IN_EN);
|
|
|
|
|
2006-05-10 06:09:17 -07:00
|
|
|
if (spec->gpio_mute) {
|
|
|
|
stac922x_gpio_mute(codec, 0, 0);
|
|
|
|
stac922x_gpio_mute(codec, 1, 0);
|
|
|
|
}
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static void stac92xx_free(struct hda_codec *codec)
|
|
|
|
{
|
2005-06-27 05:59:41 -07:00
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (! spec)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (spec->kctl_alloc) {
|
|
|
|
for (i = 0; i < spec->num_kctl_used; i++)
|
|
|
|
kfree(spec->kctl_alloc[i].name);
|
|
|
|
kfree(spec->kctl_alloc);
|
|
|
|
}
|
|
|
|
|
2006-08-23 09:31:34 -07:00
|
|
|
if (spec->bios_pin_configs)
|
|
|
|
kfree(spec->bios_pin_configs);
|
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
kfree(spec);
|
2005-04-13 05:45:30 -07:00
|
|
|
}
|
|
|
|
|
2005-07-04 08:51:39 -07:00
|
|
|
static void stac92xx_set_pinctl(struct hda_codec *codec, hda_nid_t nid,
|
|
|
|
unsigned int flag)
|
|
|
|
{
|
|
|
|
unsigned int pin_ctl = snd_hda_codec_read(codec, nid,
|
|
|
|
0, AC_VERB_GET_PIN_WIDGET_CONTROL, 0x00);
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
|
2007-05-29 09:01:06 -07:00
|
|
|
if (pin_ctl & AC_PINCTL_IN_EN) {
|
|
|
|
/*
|
|
|
|
* we need to check the current set-up direction of
|
|
|
|
* shared input pins since they can be switched via
|
|
|
|
* "xxx as Output" mixer switch
|
|
|
|
*/
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
struct auto_pin_cfg *cfg = &spec->autocfg;
|
|
|
|
if ((nid == cfg->input_pins[AUTO_PIN_LINE] &&
|
|
|
|
spec->line_switch) ||
|
|
|
|
(nid == cfg->input_pins[AUTO_PIN_MIC] &&
|
|
|
|
spec->mic_switch))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
[ALSA] hda-codec - bug fixes for stac92xx HDA codecs.
* fixed surround playback on stac922x. Pin direction control bits were
not being set correctly in stac92xx_set_pinctl(). Specifically it
would refuse to set the port as an output if the port was already
configured as an input. Last hunk (#8).
* fixed an input mux bug on 92xx codecs. When there is more than one
possible input calculated for the muxes, the actual mux widget never
gets set from its reset default, which is index 0, in the stac9221
case that is port E. So alsamixer/amixer/gnome-mixer report the Mic
as being the selected input source, but in fact is something else
(line-in port E in stac9221 case). Another problem with this is that
if you actually try to set the mux input to 'Mic', nothing happens
because *cur_val == idx (see snd_hda_input_mux_put). You have to
actually toggle input source to line-in then back to mic to actually
set the mux widget. Hunk #7.
* fixed some typos in patch_sigmatel.c. Hunk #6.
* fix to stac92xx_add_dyn_out_pins() that fixes surround playback on
codecs with less that 4 DACs (stac9205 for example). It reads the widget
caps cache created by hda_codec to count the total number of analog DACs
found. It then uses that to determine whether there will be enough
independent DACs available for line/mic switch controls. Hunk #1, #2,
and #3.
* improvements to stac92xx_auto_fill_dac_nids() to make it more general.
This fixes surround playback on some codecs in combination with the
fix to stac92xx_add_dyn_out_pins() above. It reads the full connection
list now, instead of just the first entry, and then locates an analog
DAC in the list. If one is found and it's free, assign it to that line-out.
If no free DAC is found for the line-out, return -ENODEV. It also makes
sure to actually select the chosen DAC if more than one DAC is input to
the pin. Hunks #4, #5.
Signed-off-by: Steve Longerbeam <stevel@embeddedalley.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2007-05-03 11:50:03 -07:00
|
|
|
/* if setting pin direction bits, clear the current
|
|
|
|
direction bits first */
|
|
|
|
if (flag & (AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN))
|
|
|
|
pin_ctl &= ~(AC_PINCTL_IN_EN | AC_PINCTL_OUT_EN);
|
|
|
|
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, nid, 0,
|
2005-07-04 08:51:39 -07:00
|
|
|
AC_VERB_SET_PIN_WIDGET_CONTROL,
|
|
|
|
pin_ctl | flag);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stac92xx_reset_pinctl(struct hda_codec *codec, hda_nid_t nid,
|
|
|
|
unsigned int flag)
|
|
|
|
{
|
|
|
|
unsigned int pin_ctl = snd_hda_codec_read(codec, nid,
|
|
|
|
0, AC_VERB_GET_PIN_WIDGET_CONTROL, 0x00);
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_write_cache(codec, nid, 0,
|
2005-07-04 08:51:39 -07:00
|
|
|
AC_VERB_SET_PIN_WIDGET_CONTROL,
|
|
|
|
pin_ctl & ~flag);
|
|
|
|
}
|
|
|
|
|
2006-09-21 02:56:18 -07:00
|
|
|
static int get_pin_presence(struct hda_codec *codec, hda_nid_t nid)
|
|
|
|
{
|
|
|
|
if (!nid)
|
|
|
|
return 0;
|
|
|
|
if (snd_hda_codec_read(codec, nid, 0, AC_VERB_GET_PIN_SENSE, 0x00)
|
|
|
|
& (1 << 31))
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stac92xx_hp_detect(struct hda_codec *codec, unsigned int res)
|
2005-07-04 08:51:39 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
struct auto_pin_cfg *cfg = &spec->autocfg;
|
|
|
|
int i, presence;
|
|
|
|
|
2006-09-20 08:10:27 -07:00
|
|
|
presence = 0;
|
|
|
|
for (i = 0; i < cfg->hp_outs; i++) {
|
2006-09-21 02:56:18 -07:00
|
|
|
presence = get_pin_presence(codec, cfg->hp_pins[i]);
|
|
|
|
if (presence)
|
|
|
|
break;
|
2006-09-20 08:10:27 -07:00
|
|
|
}
|
2005-07-04 08:51:39 -07:00
|
|
|
|
|
|
|
if (presence) {
|
|
|
|
/* disable lineouts, enable hp */
|
|
|
|
for (i = 0; i < cfg->line_outs; i++)
|
|
|
|
stac92xx_reset_pinctl(codec, cfg->line_out_pins[i],
|
|
|
|
AC_PINCTL_OUT_EN);
|
2006-09-20 08:10:27 -07:00
|
|
|
for (i = 0; i < cfg->speaker_outs; i++)
|
|
|
|
stac92xx_reset_pinctl(codec, cfg->speaker_pins[i],
|
|
|
|
AC_PINCTL_OUT_EN);
|
2005-07-04 08:51:39 -07:00
|
|
|
} else {
|
|
|
|
/* enable lineouts, disable hp */
|
|
|
|
for (i = 0; i < cfg->line_outs; i++)
|
|
|
|
stac92xx_set_pinctl(codec, cfg->line_out_pins[i],
|
|
|
|
AC_PINCTL_OUT_EN);
|
2006-09-20 08:10:27 -07:00
|
|
|
for (i = 0; i < cfg->speaker_outs; i++)
|
|
|
|
stac92xx_set_pinctl(codec, cfg->speaker_pins[i],
|
|
|
|
AC_PINCTL_OUT_EN);
|
2005-07-04 08:51:39 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-09-21 02:56:18 -07:00
|
|
|
static void stac92xx_unsol_event(struct hda_codec *codec, unsigned int res)
|
|
|
|
{
|
|
|
|
switch (res >> 26) {
|
|
|
|
case STAC_HP_EVENT:
|
|
|
|
stac92xx_hp_detect(codec, res);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-10 08:21:45 -07:00
|
|
|
#ifdef SND_HDA_NEEDS_RESUME
|
2005-06-27 06:06:52 -07:00
|
|
|
static int stac92xx_resume(struct hda_codec *codec)
|
|
|
|
{
|
2007-09-03 00:36:36 -07:00
|
|
|
struct sigmatel_spec *spec = codec->spec;
|
|
|
|
|
2006-08-23 09:31:34 -07:00
|
|
|
stac92xx_set_config_regs(codec);
|
2007-09-03 00:36:36 -07:00
|
|
|
snd_hda_sequence_write(codec, spec->init);
|
|
|
|
if (spec->gpio_mute) {
|
|
|
|
stac922x_gpio_mute(codec, 0, 0);
|
|
|
|
stac922x_gpio_mute(codec, 1, 0);
|
|
|
|
}
|
2007-08-10 08:09:26 -07:00
|
|
|
snd_hda_codec_resume_amp(codec);
|
|
|
|
snd_hda_codec_resume_cache(codec);
|
2007-09-03 00:36:36 -07:00
|
|
|
/* invoke unsolicited event to reset the HP state */
|
|
|
|
if (spec->hp_detect)
|
|
|
|
codec->patch_ops.unsol_event(codec, STAC_HP_EVENT << 26);
|
2005-06-27 06:06:52 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static struct hda_codec_ops stac92xx_patch_ops = {
|
|
|
|
.build_controls = stac92xx_build_controls,
|
|
|
|
.build_pcms = stac92xx_build_pcms,
|
|
|
|
.init = stac92xx_init,
|
|
|
|
.free = stac92xx_free,
|
2005-07-04 08:51:39 -07:00
|
|
|
.unsol_event = stac92xx_unsol_event,
|
2007-08-10 08:21:45 -07:00
|
|
|
#ifdef SND_HDA_NEEDS_RESUME
|
2005-06-27 06:06:52 -07:00
|
|
|
.resume = stac92xx_resume,
|
|
|
|
#endif
|
2005-04-13 05:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static int patch_stac9200(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
2005-06-27 05:59:41 -07:00
|
|
|
int err;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
[ALSA] Replace with kzalloc() - pci stuff
AD1889 driver,ATIIXP driver,ATIIXP-modem driver,AZT3328 driver
BT87x driver,CMIPCI driver,CS4281 driver,ENS1370/1+ driver
ES1938 driver,ES1968 driver,FM801 driver,Intel8x0 driver
Intel8x0-modem driver,Maestro3 driver,SonicVibes driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,AK4531 codec,au88x0 driver
CA0106 driver,CS46xx driver,EMU10K1/EMU10K2 driver,HDA Codec driver
HDA generic driver,HDA Intel driver,ICE1712 driver,ICE1724 driver
KORG1212 driver,MIXART driver,NM256 driver,Trident driver,YMFPCI driver
Replace kcalloc(1,..) with kzalloc().
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2005-09-09 05:21:46 -07:00
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
2005-04-13 05:45:30 -07:00
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
2007-07-06 09:17:04 -07:00
|
|
|
spec->num_pins = ARRAY_SIZE(stac9200_pin_nids);
|
2006-08-23 09:31:34 -07:00
|
|
|
spec->pin_nids = stac9200_pin_nids;
|
2006-11-24 09:07:44 -07:00
|
|
|
spec->board_config = snd_hda_check_board_config(codec, STAC_9200_MODELS,
|
|
|
|
stac9200_models,
|
|
|
|
stac9200_cfg_tbl);
|
2006-08-23 09:31:34 -07:00
|
|
|
if (spec->board_config < 0) {
|
|
|
|
snd_printdd(KERN_INFO "hda_codec: Unknown model for STAC9200, using BIOS defaults\n");
|
|
|
|
err = stac92xx_save_bios_config_regs(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
spec->pin_configs = spec->bios_pin_configs;
|
|
|
|
} else {
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->pin_configs = stac9200_brd_tbl[spec->board_config];
|
|
|
|
stac92xx_set_config_regs(codec);
|
|
|
|
}
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
spec->multiout.max_channels = 2;
|
|
|
|
spec->multiout.num_dacs = 1;
|
|
|
|
spec->multiout.dac_nids = stac9200_dac_nids;
|
|
|
|
spec->adc_nids = stac9200_adc_nids;
|
|
|
|
spec->mux_nids = stac9200_mux_nids;
|
2005-06-14 01:19:34 -07:00
|
|
|
spec->num_muxes = 1;
|
2006-10-26 08:12:59 -07:00
|
|
|
spec->num_dmics = 0;
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = 1;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
2007-10-10 01:04:26 -07:00
|
|
|
if (spec->board_config == STAC_9200_GATEWAY)
|
|
|
|
spec->init = stac9200_eapd_init;
|
|
|
|
else
|
|
|
|
spec->init = stac9200_core_init;
|
2005-04-13 05:45:30 -07:00
|
|
|
spec->mixer = stac9200_mixer;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
err = stac9200_parse_auto_config(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
codec->patch_ops = stac92xx_patch_ops;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-01-08 03:04:17 -07:00
|
|
|
static int patch_stac925x(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
2007-07-06 09:17:04 -07:00
|
|
|
spec->num_pins = ARRAY_SIZE(stac925x_pin_nids);
|
2007-01-08 03:04:17 -07:00
|
|
|
spec->pin_nids = stac925x_pin_nids;
|
|
|
|
spec->board_config = snd_hda_check_board_config(codec, STAC_925x_MODELS,
|
|
|
|
stac925x_models,
|
|
|
|
stac925x_cfg_tbl);
|
2007-02-08 09:50:10 -07:00
|
|
|
again:
|
2007-01-08 03:04:17 -07:00
|
|
|
if (spec->board_config < 0) {
|
2007-05-17 00:36:34 -07:00
|
|
|
snd_printdd(KERN_INFO "hda_codec: Unknown model for STAC925x,"
|
|
|
|
"using BIOS defaults\n");
|
2007-01-08 03:04:17 -07:00
|
|
|
err = stac92xx_save_bios_config_regs(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
spec->pin_configs = spec->bios_pin_configs;
|
|
|
|
} else if (stac925x_brd_tbl[spec->board_config] != NULL){
|
|
|
|
spec->pin_configs = stac925x_brd_tbl[spec->board_config];
|
|
|
|
stac92xx_set_config_regs(codec);
|
|
|
|
}
|
|
|
|
|
|
|
|
spec->multiout.max_channels = 2;
|
|
|
|
spec->multiout.num_dacs = 1;
|
|
|
|
spec->multiout.dac_nids = stac925x_dac_nids;
|
|
|
|
spec->adc_nids = stac925x_adc_nids;
|
|
|
|
spec->mux_nids = stac925x_mux_nids;
|
|
|
|
spec->num_muxes = 1;
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = 1;
|
2007-05-17 00:36:34 -07:00
|
|
|
switch (codec->vendor_id) {
|
|
|
|
case 0x83847632: /* STAC9202 */
|
|
|
|
case 0x83847633: /* STAC9202D */
|
|
|
|
case 0x83847636: /* STAC9251 */
|
|
|
|
case 0x83847637: /* STAC9251D */
|
2007-10-16 05:27:04 -07:00
|
|
|
spec->num_dmics = STAC925X_NUM_DMICS;
|
2007-05-17 00:36:34 -07:00
|
|
|
spec->dmic_nids = stac925x_dmic_nids;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
spec->num_dmics = 0;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-08 03:04:17 -07:00
|
|
|
|
|
|
|
spec->init = stac925x_core_init;
|
|
|
|
spec->mixer = stac925x_mixer;
|
|
|
|
|
|
|
|
err = stac92xx_parse_auto_config(codec, 0x8, 0x7);
|
2007-02-08 09:50:10 -07:00
|
|
|
if (!err) {
|
|
|
|
if (spec->board_config < 0) {
|
|
|
|
printk(KERN_WARNING "hda_codec: No auto-config is "
|
|
|
|
"available, default to model=ref\n");
|
|
|
|
spec->board_config = STAC_925x_REF;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
2007-01-08 03:04:17 -07:00
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
codec->patch_ops = stac92xx_patch_ops;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-11-06 03:53:55 -07:00
|
|
|
static int patch_stac92hd71bxx(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
|
|
|
spec->num_pins = ARRAY_SIZE(stac92hd71bxx_pin_nids);
|
|
|
|
spec->pin_nids = stac92hd71bxx_pin_nids;
|
|
|
|
spec->board_config = snd_hda_check_board_config(codec,
|
|
|
|
STAC_92HD71BXX_MODELS,
|
|
|
|
stac92hd71bxx_models,
|
|
|
|
stac92hd71bxx_cfg_tbl);
|
|
|
|
again:
|
|
|
|
if (spec->board_config < 0) {
|
|
|
|
snd_printdd(KERN_INFO "hda_codec: Unknown model for"
|
|
|
|
" STAC92HD71BXX, using BIOS defaults\n");
|
|
|
|
err = stac92xx_save_bios_config_regs(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
spec->pin_configs = spec->bios_pin_configs;
|
|
|
|
} else {
|
|
|
|
spec->pin_configs = stac92hd71bxx_brd_tbl[spec->board_config];
|
|
|
|
stac92xx_set_config_regs(codec);
|
|
|
|
}
|
|
|
|
|
|
|
|
spec->gpio_mask = spec->gpio_data = 0x00000001; /* GPIO0 High = EAPD */
|
|
|
|
stac92xx_enable_gpio_mask(codec);
|
|
|
|
|
|
|
|
spec->init = stac92hd71bxx_core_init;
|
|
|
|
spec->mixer = stac92hd71bxx_mixer;
|
|
|
|
|
|
|
|
spec->mux_nids = stac92hd71bxx_mux_nids;
|
|
|
|
spec->adc_nids = stac92hd71bxx_adc_nids;
|
|
|
|
spec->dmic_nids = stac92hd71bxx_dmic_nids;
|
|
|
|
spec->dmux_nid = 0x1c;
|
|
|
|
|
|
|
|
spec->num_muxes = ARRAY_SIZE(stac92hd71bxx_mux_nids);
|
|
|
|
spec->num_adcs = ARRAY_SIZE(stac92hd71bxx_adc_nids);
|
|
|
|
spec->num_dmics = STAC92HD71BXX_NUM_DMICS;
|
|
|
|
|
|
|
|
spec->multiout.num_dacs = 2;
|
|
|
|
spec->multiout.hp_nid = 0x11;
|
|
|
|
spec->multiout.dac_nids = stac92hd71bxx_dac_nids;
|
|
|
|
|
|
|
|
err = stac92xx_parse_auto_config(codec, 0x21, 0x23);
|
|
|
|
if (!err) {
|
|
|
|
if (spec->board_config < 0) {
|
|
|
|
printk(KERN_WARNING "hda_codec: No auto-config is "
|
|
|
|
"available, default to model=ref\n");
|
|
|
|
spec->board_config = STAC_92HD71BXX_REF;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
codec->patch_ops = stac92xx_patch_ops;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
};
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
static int patch_stac922x(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
2005-06-27 05:59:41 -07:00
|
|
|
int err;
|
2005-04-13 05:45:30 -07:00
|
|
|
|
[ALSA] Replace with kzalloc() - pci stuff
AD1889 driver,ATIIXP driver,ATIIXP-modem driver,AZT3328 driver
BT87x driver,CMIPCI driver,CS4281 driver,ENS1370/1+ driver
ES1938 driver,ES1968 driver,FM801 driver,Intel8x0 driver
Intel8x0-modem driver,Maestro3 driver,SonicVibes driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,AK4531 codec,au88x0 driver
CA0106 driver,CS46xx driver,EMU10K1/EMU10K2 driver,HDA Codec driver
HDA generic driver,HDA Intel driver,ICE1712 driver,ICE1724 driver
KORG1212 driver,MIXART driver,NM256 driver,Trident driver,YMFPCI driver
Replace kcalloc(1,..) with kzalloc().
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2005-09-09 05:21:46 -07:00
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
2005-04-13 05:45:30 -07:00
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
2007-07-06 09:17:04 -07:00
|
|
|
spec->num_pins = ARRAY_SIZE(stac922x_pin_nids);
|
2006-08-23 09:31:34 -07:00
|
|
|
spec->pin_nids = stac922x_pin_nids;
|
2006-11-24 09:07:44 -07:00
|
|
|
spec->board_config = snd_hda_check_board_config(codec, STAC_922X_MODELS,
|
|
|
|
stac922x_models,
|
|
|
|
stac922x_cfg_tbl);
|
2007-05-29 07:03:00 -07:00
|
|
|
if (spec->board_config == STAC_INTEL_MAC_V3) {
|
2007-02-16 05:27:18 -07:00
|
|
|
spec->gpio_mute = 1;
|
|
|
|
/* Intel Macs have all same PCI SSID, so we need to check
|
|
|
|
* codec SSID to distinguish the exact models
|
|
|
|
*/
|
2007-03-15 04:38:15 -07:00
|
|
|
printk(KERN_INFO "hda_codec: STAC922x, Apple subsys_id=%x\n", codec->subsystem_id);
|
2007-02-16 05:27:18 -07:00
|
|
|
switch (codec->subsystem_id) {
|
2007-05-29 07:03:00 -07:00
|
|
|
|
|
|
|
case 0x106b0800:
|
|
|
|
spec->board_config = STAC_INTEL_MAC_V1;
|
2007-04-17 02:57:16 -07:00
|
|
|
break;
|
2007-05-29 07:03:00 -07:00
|
|
|
case 0x106b0600:
|
|
|
|
case 0x106b0700:
|
|
|
|
spec->board_config = STAC_INTEL_MAC_V2;
|
2007-03-15 04:38:15 -07:00
|
|
|
break;
|
2007-05-29 07:03:00 -07:00
|
|
|
case 0x106b0e00:
|
|
|
|
case 0x106b0f00:
|
|
|
|
case 0x106b1600:
|
|
|
|
case 0x106b1700:
|
|
|
|
case 0x106b0200:
|
|
|
|
case 0x106b1e00:
|
|
|
|
spec->board_config = STAC_INTEL_MAC_V3;
|
2007-02-16 05:27:18 -07:00
|
|
|
break;
|
2007-05-29 07:03:00 -07:00
|
|
|
case 0x106b1a00:
|
|
|
|
case 0x00000100:
|
|
|
|
spec->board_config = STAC_INTEL_MAC_V4;
|
2007-04-27 05:22:36 -07:00
|
|
|
break;
|
2007-05-29 07:03:00 -07:00
|
|
|
case 0x106b0a00:
|
|
|
|
case 0x106b2200:
|
|
|
|
spec->board_config = STAC_INTEL_MAC_V5;
|
2007-05-21 03:41:29 -07:00
|
|
|
break;
|
2007-02-16 05:27:18 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-02-08 09:50:10 -07:00
|
|
|
again:
|
2006-08-23 09:31:34 -07:00
|
|
|
if (spec->board_config < 0) {
|
|
|
|
snd_printdd(KERN_INFO "hda_codec: Unknown model for STAC922x, "
|
|
|
|
"using BIOS defaults\n");
|
|
|
|
err = stac92xx_save_bios_config_regs(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
spec->pin_configs = spec->bios_pin_configs;
|
|
|
|
} else if (stac922x_brd_tbl[spec->board_config] != NULL) {
|
2005-11-29 07:00:51 -07:00
|
|
|
spec->pin_configs = stac922x_brd_tbl[spec->board_config];
|
|
|
|
stac92xx_set_config_regs(codec);
|
|
|
|
}
|
2005-04-13 05:45:30 -07:00
|
|
|
|
2005-06-27 05:59:41 -07:00
|
|
|
spec->adc_nids = stac922x_adc_nids;
|
|
|
|
spec->mux_nids = stac922x_mux_nids;
|
2007-03-12 04:36:16 -07:00
|
|
|
spec->num_muxes = ARRAY_SIZE(stac922x_mux_nids);
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = ARRAY_SIZE(stac922x_adc_nids);
|
2006-10-26 08:12:59 -07:00
|
|
|
spec->num_dmics = 0;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
spec->init = stac922x_core_init;
|
2005-04-13 05:45:30 -07:00
|
|
|
spec->mixer = stac922x_mixer;
|
2005-06-27 05:59:41 -07:00
|
|
|
|
|
|
|
spec->multiout.dac_nids = spec->dac_nids;
|
2006-06-28 06:52:16 -07:00
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
err = stac92xx_parse_auto_config(codec, 0x08, 0x09);
|
2007-02-08 09:50:10 -07:00
|
|
|
if (!err) {
|
|
|
|
if (spec->board_config < 0) {
|
|
|
|
printk(KERN_WARNING "hda_codec: No auto-config is "
|
|
|
|
"available, default to model=ref\n");
|
|
|
|
spec->board_config = STAC_D945_REF;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
2006-01-23 07:27:49 -07:00
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
codec->patch_ops = stac92xx_patch_ops;
|
|
|
|
|
2007-05-29 10:01:37 -07:00
|
|
|
/* Fix Mux capture level; max to 2 */
|
|
|
|
snd_hda_override_amp_caps(codec, 0x12, HDA_OUTPUT,
|
|
|
|
(0 << AC_AMPCAP_OFFSET_SHIFT) |
|
|
|
|
(2 << AC_AMPCAP_NUM_STEPS_SHIFT) |
|
|
|
|
(0x27 << AC_AMPCAP_STEP_SIZE_SHIFT) |
|
|
|
|
(0 << AC_AMPCAP_MUTE_SHIFT));
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int patch_stac927x(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
2007-07-06 09:17:04 -07:00
|
|
|
spec->num_pins = ARRAY_SIZE(stac927x_pin_nids);
|
2006-08-23 09:31:34 -07:00
|
|
|
spec->pin_nids = stac927x_pin_nids;
|
2006-11-24 09:07:44 -07:00
|
|
|
spec->board_config = snd_hda_check_board_config(codec, STAC_927X_MODELS,
|
|
|
|
stac927x_models,
|
|
|
|
stac927x_cfg_tbl);
|
2007-02-08 09:50:10 -07:00
|
|
|
again:
|
2006-08-23 09:31:34 -07:00
|
|
|
if (spec->board_config < 0) {
|
2006-01-23 07:27:49 -07:00
|
|
|
snd_printdd(KERN_INFO "hda_codec: Unknown model for STAC927x, using BIOS defaults\n");
|
2006-08-23 09:31:34 -07:00
|
|
|
err = stac92xx_save_bios_config_regs(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
spec->pin_configs = spec->bios_pin_configs;
|
|
|
|
} else if (stac927x_brd_tbl[spec->board_config] != NULL) {
|
2006-01-23 07:27:49 -07:00
|
|
|
spec->pin_configs = stac927x_brd_tbl[spec->board_config];
|
|
|
|
stac92xx_set_config_regs(codec);
|
|
|
|
}
|
|
|
|
|
2006-08-22 10:44:45 -07:00
|
|
|
switch (spec->board_config) {
|
2006-09-01 12:03:12 -07:00
|
|
|
case STAC_D965_3ST:
|
2006-08-22 10:44:45 -07:00
|
|
|
spec->adc_nids = stac927x_adc_nids;
|
|
|
|
spec->mux_nids = stac927x_mux_nids;
|
2007-03-12 04:36:16 -07:00
|
|
|
spec->num_muxes = ARRAY_SIZE(stac927x_mux_nids);
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = ARRAY_SIZE(stac927x_adc_nids);
|
2006-09-01 12:03:12 -07:00
|
|
|
spec->init = d965_core_init;
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->mixer = stac927x_mixer;
|
2006-08-22 10:44:45 -07:00
|
|
|
break;
|
2006-09-01 12:03:12 -07:00
|
|
|
case STAC_D965_5ST:
|
|
|
|
spec->adc_nids = stac927x_adc_nids;
|
|
|
|
spec->mux_nids = stac927x_mux_nids;
|
2007-03-12 04:36:16 -07:00
|
|
|
spec->num_muxes = ARRAY_SIZE(stac927x_mux_nids);
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = ARRAY_SIZE(stac927x_adc_nids);
|
2006-09-01 12:03:12 -07:00
|
|
|
spec->init = d965_core_init;
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->mixer = stac927x_mixer;
|
2006-08-22 10:44:45 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
spec->adc_nids = stac927x_adc_nids;
|
|
|
|
spec->mux_nids = stac927x_mux_nids;
|
2007-03-12 04:36:16 -07:00
|
|
|
spec->num_muxes = ARRAY_SIZE(stac927x_mux_nids);
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = ARRAY_SIZE(stac927x_adc_nids);
|
2006-08-22 10:44:45 -07:00
|
|
|
spec->init = stac927x_core_init;
|
|
|
|
spec->mixer = stac927x_mixer;
|
|
|
|
}
|
2006-01-23 07:27:49 -07:00
|
|
|
|
2007-10-18 08:38:17 -07:00
|
|
|
switch (codec->subsystem_id) {
|
2007-11-07 07:54:45 -07:00
|
|
|
case 0x10280242: /* STAC 9228 */
|
|
|
|
case 0x102801f3:
|
|
|
|
case 0x1028020A:
|
|
|
|
case 0x10280209:
|
2007-10-18 08:38:17 -07:00
|
|
|
spec->dmic_nids = stac927x_dmic_nids;
|
|
|
|
spec->num_dmics = STAC927X_NUM_DMICS;
|
2007-10-22 03:27:10 -07:00
|
|
|
spec->dmux_nid = 0x1b;
|
2007-11-05 07:30:13 -07:00
|
|
|
|
|
|
|
/* Enable DMIC0 */
|
|
|
|
stac92xx_set_config_reg(codec, 0x13, 0x90a60040);
|
|
|
|
|
|
|
|
/* GPIO2 High = Enable EAPD */
|
|
|
|
spec->gpio_mask = spec->gpio_data = 0x00000004;
|
2007-10-18 08:38:17 -07:00
|
|
|
break;
|
|
|
|
default:
|
2007-11-05 07:30:13 -07:00
|
|
|
spec->num_dmics = 0;
|
|
|
|
|
|
|
|
/* GPIO0 High = Enable EAPD */
|
|
|
|
spec->gpio_mask = spec->gpio_data = 0x00000001;
|
2007-10-18 08:38:17 -07:00
|
|
|
}
|
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
spec->multiout.dac_nids = spec->dac_nids;
|
2007-07-31 06:56:24 -07:00
|
|
|
stac92xx_enable_gpio_mask(codec);
|
2007-06-19 07:48:28 -07:00
|
|
|
|
2006-01-23 07:27:49 -07:00
|
|
|
err = stac92xx_parse_auto_config(codec, 0x1e, 0x20);
|
2007-02-08 09:50:10 -07:00
|
|
|
if (!err) {
|
|
|
|
if (spec->board_config < 0) {
|
|
|
|
printk(KERN_WARNING "hda_codec: No auto-config is "
|
|
|
|
"available, default to model=ref\n");
|
|
|
|
spec->board_config = STAC_D965_REF;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
2005-06-27 05:59:41 -07:00
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
2005-04-13 05:45:30 -07:00
|
|
|
|
|
|
|
codec->patch_ops = stac92xx_patch_ops;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-07-31 03:49:34 -07:00
|
|
|
static int patch_stac9205(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
2007-07-31 06:56:24 -07:00
|
|
|
int err;
|
2006-07-31 03:49:34 -07:00
|
|
|
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
2007-07-06 09:17:04 -07:00
|
|
|
spec->num_pins = ARRAY_SIZE(stac9205_pin_nids);
|
2006-08-23 09:31:34 -07:00
|
|
|
spec->pin_nids = stac9205_pin_nids;
|
2006-11-24 09:07:44 -07:00
|
|
|
spec->board_config = snd_hda_check_board_config(codec, STAC_9205_MODELS,
|
|
|
|
stac9205_models,
|
|
|
|
stac9205_cfg_tbl);
|
2007-02-08 09:50:10 -07:00
|
|
|
again:
|
2006-08-23 09:31:34 -07:00
|
|
|
if (spec->board_config < 0) {
|
|
|
|
snd_printdd(KERN_INFO "hda_codec: Unknown model for STAC9205, using BIOS defaults\n");
|
|
|
|
err = stac92xx_save_bios_config_regs(codec);
|
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
spec->pin_configs = spec->bios_pin_configs;
|
|
|
|
} else {
|
2006-07-31 03:49:34 -07:00
|
|
|
spec->pin_configs = stac9205_brd_tbl[spec->board_config];
|
|
|
|
stac92xx_set_config_regs(codec);
|
|
|
|
}
|
|
|
|
|
|
|
|
spec->adc_nids = stac9205_adc_nids;
|
2007-09-03 06:31:02 -07:00
|
|
|
spec->num_adcs = ARRAY_SIZE(stac9205_adc_nids);
|
2006-07-31 03:49:34 -07:00
|
|
|
spec->mux_nids = stac9205_mux_nids;
|
2007-03-12 04:36:16 -07:00
|
|
|
spec->num_muxes = ARRAY_SIZE(stac9205_mux_nids);
|
2006-10-26 08:12:59 -07:00
|
|
|
spec->dmic_nids = stac9205_dmic_nids;
|
2007-10-16 05:27:04 -07:00
|
|
|
spec->num_dmics = STAC9205_NUM_DMICS;
|
2006-10-26 08:12:59 -07:00
|
|
|
spec->dmux_nid = 0x1d;
|
2006-07-31 03:49:34 -07:00
|
|
|
|
|
|
|
spec->init = stac9205_core_init;
|
|
|
|
spec->mixer = stac9205_mixer;
|
|
|
|
|
|
|
|
spec->multiout.dac_nids = spec->dac_nids;
|
2007-07-17 02:52:24 -07:00
|
|
|
|
2007-08-13 06:50:29 -07:00
|
|
|
switch (spec->board_config){
|
|
|
|
case STAC_9205_DELL_M43:
|
2007-07-17 02:52:24 -07:00
|
|
|
/* Enable SPDIF in/out */
|
|
|
|
stac92xx_set_config_reg(codec, 0x1f, 0x01441030);
|
|
|
|
stac92xx_set_config_reg(codec, 0x20, 0x1c410030);
|
|
|
|
|
2007-07-31 06:56:24 -07:00
|
|
|
spec->gpio_mask = 0x00000007; /* GPIO0-2 */
|
2007-07-17 02:52:24 -07:00
|
|
|
/* GPIO0 High = EAPD, GPIO1 Low = DRM,
|
|
|
|
* GPIO2 High = Headphone Mute
|
|
|
|
*/
|
2007-07-31 06:56:24 -07:00
|
|
|
spec->gpio_data = 0x00000005;
|
2007-08-13 06:50:29 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* GPIO0 High = EAPD */
|
|
|
|
spec->gpio_mask = spec->gpio_data = 0x00000001;
|
|
|
|
break;
|
|
|
|
}
|
2006-12-18 05:17:28 -07:00
|
|
|
|
2007-07-31 06:56:24 -07:00
|
|
|
stac92xx_enable_gpio_mask(codec);
|
2006-07-31 03:49:34 -07:00
|
|
|
err = stac92xx_parse_auto_config(codec, 0x1f, 0x20);
|
2007-02-08 09:50:10 -07:00
|
|
|
if (!err) {
|
|
|
|
if (spec->board_config < 0) {
|
|
|
|
printk(KERN_WARNING "hda_codec: No auto-config is "
|
|
|
|
"available, default to model=ref\n");
|
|
|
|
spec->board_config = STAC_9205_REF;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
2006-07-31 03:49:34 -07:00
|
|
|
if (err < 0) {
|
|
|
|
stac92xx_free(codec);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
codec->patch_ops = stac92xx_patch_ops;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-03-16 08:04:58 -07:00
|
|
|
/*
|
2006-08-22 08:15:47 -07:00
|
|
|
* STAC9872 hack
|
2006-03-16 08:04:58 -07:00
|
|
|
*/
|
|
|
|
|
2006-08-16 10:35:12 -07:00
|
|
|
/* static config for Sony VAIO FE550G and Sony VAIO AR */
|
2006-03-16 08:04:58 -07:00
|
|
|
static hda_nid_t vaio_dacs[] = { 0x2 };
|
|
|
|
#define VAIO_HP_DAC 0x5
|
|
|
|
static hda_nid_t vaio_adcs[] = { 0x8 /*,0x6*/ };
|
|
|
|
static hda_nid_t vaio_mux_nids[] = { 0x15 };
|
|
|
|
|
|
|
|
static struct hda_input_mux vaio_mux = {
|
2007-10-11 02:21:21 -07:00
|
|
|
.num_items = 3,
|
2006-03-16 08:04:58 -07:00
|
|
|
.items = {
|
2006-04-25 04:05:43 -07:00
|
|
|
/* { "HP", 0x0 }, */
|
2007-07-05 04:10:51 -07:00
|
|
|
{ "Mic Jack", 0x1 },
|
|
|
|
{ "Internal Mic", 0x2 },
|
2006-03-16 08:04:58 -07:00
|
|
|
{ "PCM", 0x3 },
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct hda_verb vaio_init[] = {
|
|
|
|
{0x0a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP }, /* HP <- 0x2 */
|
2007-08-16 08:33:55 -07:00
|
|
|
{0x0a, AC_VERB_SET_UNSOLICITED_ENABLE, AC_USRSP_EN | STAC_HP_EVENT},
|
2006-03-16 08:04:58 -07:00
|
|
|
{0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Speaker <- 0x5 */
|
|
|
|
{0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Mic? (<- 0x2) */
|
|
|
|
{0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN }, /* CD */
|
|
|
|
{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Mic? */
|
2007-07-05 04:10:51 -07:00
|
|
|
{0x15, AC_VERB_SET_CONNECT_SEL, 0x1}, /* mic-sel: 0a,0d,14,02 */
|
2006-03-16 08:04:58 -07:00
|
|
|
{0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* HP */
|
|
|
|
{0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* Speaker */
|
|
|
|
{0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* capture sw/vol -> 0x8 */
|
|
|
|
{0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, /* CD-in -> 0x6 */
|
|
|
|
{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Mic-in -> 0x9 */
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-08-22 08:15:47 -07:00
|
|
|
static struct hda_verb vaio_ar_init[] = {
|
|
|
|
{0x0a, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_HP }, /* HP <- 0x2 */
|
|
|
|
{0x0f, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT }, /* Speaker <- 0x5 */
|
|
|
|
{0x0d, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Mic? (<- 0x2) */
|
|
|
|
{0x0e, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_IN }, /* CD */
|
|
|
|
/* {0x11, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT },*/ /* Optical Out */
|
|
|
|
{0x14, AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_VREF80 }, /* Mic? */
|
2007-07-05 04:10:51 -07:00
|
|
|
{0x15, AC_VERB_SET_CONNECT_SEL, 0x1}, /* mic-sel: 0a,0d,14,02 */
|
2006-08-22 08:15:47 -07:00
|
|
|
{0x02, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* HP */
|
|
|
|
{0x05, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE}, /* Speaker */
|
|
|
|
/* {0x10, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE},*/ /* Optical Out */
|
|
|
|
{0x09, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_MUTE(0)}, /* capture sw/vol -> 0x8 */
|
|
|
|
{0x07, AC_VERB_SET_AMP_GAIN_MUTE, AMP_IN_UNMUTE(0)}, /* CD-in -> 0x6 */
|
|
|
|
{0x15, AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE}, /* Mic-in -> 0x9 */
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-03-16 08:04:58 -07:00
|
|
|
/* bind volumes of both NID 0x02 and 0x05 */
|
2007-08-10 08:12:15 -07:00
|
|
|
static struct hda_bind_ctls vaio_bind_master_vol = {
|
|
|
|
.ops = &snd_hda_bind_vol,
|
|
|
|
.values = {
|
|
|
|
HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT),
|
|
|
|
HDA_COMPOSE_AMP_VAL(0x05, 3, 0, HDA_OUTPUT),
|
|
|
|
0
|
|
|
|
},
|
|
|
|
};
|
2006-03-16 08:04:58 -07:00
|
|
|
|
|
|
|
/* bind volumes of both NID 0x02 and 0x05 */
|
2007-08-10 08:12:15 -07:00
|
|
|
static struct hda_bind_ctls vaio_bind_master_sw = {
|
|
|
|
.ops = &snd_hda_bind_sw,
|
|
|
|
.values = {
|
|
|
|
HDA_COMPOSE_AMP_VAL(0x02, 3, 0, HDA_OUTPUT),
|
|
|
|
HDA_COMPOSE_AMP_VAL(0x05, 3, 0, HDA_OUTPUT),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
};
|
2006-03-16 08:04:58 -07:00
|
|
|
|
|
|
|
static struct snd_kcontrol_new vaio_mixer[] = {
|
2007-08-10 08:12:15 -07:00
|
|
|
HDA_BIND_VOL("Master Playback Volume", &vaio_bind_master_vol),
|
|
|
|
HDA_BIND_SW("Master Playback Switch", &vaio_bind_master_sw),
|
2006-03-16 08:04:58 -07:00
|
|
|
/* HDA_CODEC_VOLUME("CD Capture Volume", 0x07, 0, HDA_INPUT), */
|
|
|
|
HDA_CODEC_VOLUME("Capture Volume", 0x09, 0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE("Capture Switch", 0x09, 0, HDA_INPUT),
|
|
|
|
{
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Capture Source",
|
|
|
|
.count = 1,
|
|
|
|
.info = stac92xx_mux_enum_info,
|
|
|
|
.get = stac92xx_mux_enum_get,
|
|
|
|
.put = stac92xx_mux_enum_put,
|
|
|
|
},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-08-22 08:15:47 -07:00
|
|
|
static struct snd_kcontrol_new vaio_ar_mixer[] = {
|
2007-08-10 08:12:15 -07:00
|
|
|
HDA_BIND_VOL("Master Playback Volume", &vaio_bind_master_vol),
|
|
|
|
HDA_BIND_SW("Master Playback Switch", &vaio_bind_master_sw),
|
2006-08-22 08:15:47 -07:00
|
|
|
/* HDA_CODEC_VOLUME("CD Capture Volume", 0x07, 0, HDA_INPUT), */
|
|
|
|
HDA_CODEC_VOLUME("Capture Volume", 0x09, 0, HDA_INPUT),
|
|
|
|
HDA_CODEC_MUTE("Capture Switch", 0x09, 0, HDA_INPUT),
|
|
|
|
/*HDA_CODEC_MUTE("Optical Out Switch", 0x10, 0, HDA_OUTPUT),
|
|
|
|
HDA_CODEC_VOLUME("Optical Out Volume", 0x10, 0, HDA_OUTPUT),*/
|
|
|
|
{
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Capture Source",
|
|
|
|
.count = 1,
|
|
|
|
.info = stac92xx_mux_enum_info,
|
|
|
|
.get = stac92xx_mux_enum_get,
|
|
|
|
.put = stac92xx_mux_enum_put,
|
|
|
|
},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct hda_codec_ops stac9872_patch_ops = {
|
2006-03-16 08:04:58 -07:00
|
|
|
.build_controls = stac92xx_build_controls,
|
|
|
|
.build_pcms = stac92xx_build_pcms,
|
|
|
|
.init = stac92xx_init,
|
|
|
|
.free = stac92xx_free,
|
2007-08-10 08:21:45 -07:00
|
|
|
#ifdef SND_HDA_NEEDS_RESUME
|
2006-03-16 08:04:58 -07:00
|
|
|
.resume = stac92xx_resume,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2007-08-16 08:33:55 -07:00
|
|
|
static int stac9872_vaio_init(struct hda_codec *codec)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = stac92xx_init(codec);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
if (codec->patch_ops.unsol_event)
|
|
|
|
codec->patch_ops.unsol_event(codec, STAC_HP_EVENT << 26);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stac9872_vaio_hp_detect(struct hda_codec *codec, unsigned int res)
|
|
|
|
{
|
|
|
|
if (get_pin_presence(codec, 0x0a)) {
|
|
|
|
stac92xx_reset_pinctl(codec, 0x0f, AC_PINCTL_OUT_EN);
|
|
|
|
stac92xx_set_pinctl(codec, 0x0a, AC_PINCTL_OUT_EN);
|
|
|
|
} else {
|
|
|
|
stac92xx_reset_pinctl(codec, 0x0a, AC_PINCTL_OUT_EN);
|
|
|
|
stac92xx_set_pinctl(codec, 0x0f, AC_PINCTL_OUT_EN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stac9872_vaio_unsol_event(struct hda_codec *codec, unsigned int res)
|
|
|
|
{
|
|
|
|
switch (res >> 26) {
|
|
|
|
case STAC_HP_EVENT:
|
|
|
|
stac9872_vaio_hp_detect(codec, res);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct hda_codec_ops stac9872_vaio_patch_ops = {
|
|
|
|
.build_controls = stac92xx_build_controls,
|
|
|
|
.build_pcms = stac92xx_build_pcms,
|
|
|
|
.init = stac9872_vaio_init,
|
|
|
|
.free = stac92xx_free,
|
|
|
|
.unsol_event = stac9872_vaio_unsol_event,
|
|
|
|
#ifdef CONFIG_PM
|
|
|
|
.resume = stac92xx_resume,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2006-08-22 08:15:47 -07:00
|
|
|
enum { /* FE and SZ series. id=0x83847661 and subsys=0x104D0700 or 104D1000. */
|
|
|
|
CXD9872RD_VAIO,
|
|
|
|
/* Unknown. id=0x83847662 and subsys=0x104D1200 or 104D1000. */
|
|
|
|
STAC9872AK_VAIO,
|
|
|
|
/* Unknown. id=0x83847661 and subsys=0x104D1200. */
|
|
|
|
STAC9872K_VAIO,
|
|
|
|
/* AR Series. id=0x83847664 and subsys=104D1300 */
|
2006-11-24 09:07:44 -07:00
|
|
|
CXD9872AKD_VAIO,
|
|
|
|
STAC_9872_MODELS,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *stac9872_models[STAC_9872_MODELS] = {
|
|
|
|
[CXD9872RD_VAIO] = "vaio",
|
|
|
|
[CXD9872AKD_VAIO] = "vaio-ar",
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct snd_pci_quirk stac9872_cfg_tbl[] = {
|
|
|
|
SND_PCI_QUIRK(0x104d, 0x81e6, "Sony VAIO F/S", CXD9872RD_VAIO),
|
|
|
|
SND_PCI_QUIRK(0x104d, 0x81ef, "Sony VAIO F/S", CXD9872RD_VAIO),
|
|
|
|
SND_PCI_QUIRK(0x104d, 0x81fd, "Sony VAIO AR", CXD9872AKD_VAIO),
|
2007-03-12 03:36:39 -07:00
|
|
|
SND_PCI_QUIRK(0x104d, 0x8205, "Sony VAIO AR", CXD9872AKD_VAIO),
|
2006-03-16 08:04:58 -07:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2006-08-22 08:15:47 -07:00
|
|
|
static int patch_stac9872(struct hda_codec *codec)
|
2006-03-16 08:04:58 -07:00
|
|
|
{
|
|
|
|
struct sigmatel_spec *spec;
|
|
|
|
int board_config;
|
|
|
|
|
2006-11-24 09:07:44 -07:00
|
|
|
board_config = snd_hda_check_board_config(codec, STAC_9872_MODELS,
|
|
|
|
stac9872_models,
|
|
|
|
stac9872_cfg_tbl);
|
2006-03-16 08:04:58 -07:00
|
|
|
if (board_config < 0)
|
|
|
|
/* unknown config, let generic-parser do its job... */
|
|
|
|
return snd_hda_parse_generic_codec(codec);
|
|
|
|
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
|
|
if (spec == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
codec->spec = spec;
|
|
|
|
switch (board_config) {
|
2006-08-22 08:15:47 -07:00
|
|
|
case CXD9872RD_VAIO:
|
|
|
|
case STAC9872AK_VAIO:
|
|
|
|
case STAC9872K_VAIO:
|
2006-03-16 08:04:58 -07:00
|
|
|
spec->mixer = vaio_mixer;
|
|
|
|
spec->init = vaio_init;
|
|
|
|
spec->multiout.max_channels = 2;
|
|
|
|
spec->multiout.num_dacs = ARRAY_SIZE(vaio_dacs);
|
|
|
|
spec->multiout.dac_nids = vaio_dacs;
|
|
|
|
spec->multiout.hp_nid = VAIO_HP_DAC;
|
|
|
|
spec->num_adcs = ARRAY_SIZE(vaio_adcs);
|
|
|
|
spec->adc_nids = vaio_adcs;
|
|
|
|
spec->input_mux = &vaio_mux;
|
|
|
|
spec->mux_nids = vaio_mux_nids;
|
2007-08-16 08:33:55 -07:00
|
|
|
codec->patch_ops = stac9872_vaio_patch_ops;
|
2006-03-16 08:04:58 -07:00
|
|
|
break;
|
2006-08-22 08:15:47 -07:00
|
|
|
|
|
|
|
case CXD9872AKD_VAIO:
|
|
|
|
spec->mixer = vaio_ar_mixer;
|
|
|
|
spec->init = vaio_ar_init;
|
|
|
|
spec->multiout.max_channels = 2;
|
|
|
|
spec->multiout.num_dacs = ARRAY_SIZE(vaio_dacs);
|
|
|
|
spec->multiout.dac_nids = vaio_dacs;
|
|
|
|
spec->multiout.hp_nid = VAIO_HP_DAC;
|
|
|
|
spec->num_adcs = ARRAY_SIZE(vaio_adcs);
|
|
|
|
spec->adc_nids = vaio_adcs;
|
|
|
|
spec->input_mux = &vaio_mux;
|
|
|
|
spec->mux_nids = vaio_mux_nids;
|
2007-08-16 08:33:55 -07:00
|
|
|
codec->patch_ops = stac9872_patch_ops;
|
2006-08-22 08:15:47 -07:00
|
|
|
break;
|
2006-03-16 08:04:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-04-13 05:45:30 -07:00
|
|
|
/*
|
|
|
|
* patch entries
|
|
|
|
*/
|
|
|
|
struct hda_codec_preset snd_hda_preset_sigmatel[] = {
|
|
|
|
{ .id = 0x83847690, .name = "STAC9200", .patch = patch_stac9200 },
|
|
|
|
{ .id = 0x83847882, .name = "STAC9220 A1", .patch = patch_stac922x },
|
|
|
|
{ .id = 0x83847680, .name = "STAC9221 A1", .patch = patch_stac922x },
|
|
|
|
{ .id = 0x83847880, .name = "STAC9220 A2", .patch = patch_stac922x },
|
|
|
|
{ .id = 0x83847681, .name = "STAC9220D/9223D A2", .patch = patch_stac922x },
|
|
|
|
{ .id = 0x83847682, .name = "STAC9221 A2", .patch = patch_stac922x },
|
|
|
|
{ .id = 0x83847683, .name = "STAC9221D A2", .patch = patch_stac922x },
|
2006-07-06 09:49:10 -07:00
|
|
|
{ .id = 0x83847618, .name = "STAC9227", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847619, .name = "STAC9227", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847616, .name = "STAC9228", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847617, .name = "STAC9228", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847614, .name = "STAC9229", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847615, .name = "STAC9229", .patch = patch_stac927x },
|
2006-01-23 07:27:49 -07:00
|
|
|
{ .id = 0x83847620, .name = "STAC9274", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847621, .name = "STAC9274D", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847622, .name = "STAC9273X", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847623, .name = "STAC9273D", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847624, .name = "STAC9272X", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847625, .name = "STAC9272D", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847626, .name = "STAC9271X", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847627, .name = "STAC9271D", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847628, .name = "STAC9274X5NH", .patch = patch_stac927x },
|
|
|
|
{ .id = 0x83847629, .name = "STAC9274D5NH", .patch = patch_stac927x },
|
2007-01-08 03:04:17 -07:00
|
|
|
{ .id = 0x83847632, .name = "STAC9202", .patch = patch_stac925x },
|
|
|
|
{ .id = 0x83847633, .name = "STAC9202D", .patch = patch_stac925x },
|
|
|
|
{ .id = 0x83847634, .name = "STAC9250", .patch = patch_stac925x },
|
|
|
|
{ .id = 0x83847635, .name = "STAC9250D", .patch = patch_stac925x },
|
|
|
|
{ .id = 0x83847636, .name = "STAC9251", .patch = patch_stac925x },
|
|
|
|
{ .id = 0x83847637, .name = "STAC9250D", .patch = patch_stac925x },
|
2006-08-22 08:15:47 -07:00
|
|
|
/* The following does not take into account .id=0x83847661 when subsys =
|
|
|
|
* 104D0C00 which is STAC9225s. Because of this, some SZ Notebooks are
|
|
|
|
* currently not fully supported.
|
|
|
|
*/
|
|
|
|
{ .id = 0x83847661, .name = "CXD9872RD/K", .patch = patch_stac9872 },
|
|
|
|
{ .id = 0x83847662, .name = "STAC9872AK", .patch = patch_stac9872 },
|
|
|
|
{ .id = 0x83847664, .name = "CXD9872AKD", .patch = patch_stac9872 },
|
2006-07-31 03:49:34 -07:00
|
|
|
{ .id = 0x838476a0, .name = "STAC9205", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a1, .name = "STAC9205D", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a2, .name = "STAC9204", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a3, .name = "STAC9204D", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a4, .name = "STAC9255", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a5, .name = "STAC9255D", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a6, .name = "STAC9254", .patch = patch_stac9205 },
|
|
|
|
{ .id = 0x838476a7, .name = "STAC9254D", .patch = patch_stac9205 },
|
2007-11-06 03:53:55 -07:00
|
|
|
{ .id = 0x111d76b0, .name = "92HD71BXX", .patch = patch_stac92hd71bxx },
|
2005-04-13 05:45:30 -07:00
|
|
|
{} /* terminator */
|
|
|
|
};
|