[ALSA] intel8x0 - Add spdif_aclink option
Added spdif_aclink module option to specify whether the board has SPDIF over AC-link or a direct connection from the controller chip. NForce and ICH4 (or newer) boards may be equipped with SPDIF through AC97 codec. In such a case, SPDIF should be handled as if the old ICH style (the same slot for analog and digital). A quirk list is added to detect this automatically for known hardwares. Corresponds to ALSA bug#2637. Signed-off-by: Takashi Iwai <tiwai@suse.de> Signed-off-by: Jaroslav Kysela <perex@suse.cz>
This commit is contained in:
parent
5da8fa2516
commit
a9e996604f
1 changed files with 60 additions and 26 deletions
|
@ -71,6 +71,7 @@ static char *ac97_quirk;
|
||||||
static int buggy_semaphore;
|
static int buggy_semaphore;
|
||||||
static int buggy_irq = -1; /* auto-check */
|
static int buggy_irq = -1; /* auto-check */
|
||||||
static int xbox;
|
static int xbox;
|
||||||
|
static int spdif_aclink = -1;
|
||||||
|
|
||||||
module_param(index, int, 0444);
|
module_param(index, int, 0444);
|
||||||
MODULE_PARM_DESC(index, "Index value for Intel i8x0 soundcard.");
|
MODULE_PARM_DESC(index, "Index value for Intel i8x0 soundcard.");
|
||||||
|
@ -86,6 +87,8 @@ module_param(buggy_irq, bool, 0444);
|
||||||
MODULE_PARM_DESC(buggy_irq, "Enable workaround for buggy interrupts on some motherboards.");
|
MODULE_PARM_DESC(buggy_irq, "Enable workaround for buggy interrupts on some motherboards.");
|
||||||
module_param(xbox, bool, 0444);
|
module_param(xbox, bool, 0444);
|
||||||
MODULE_PARM_DESC(xbox, "Set to 1 for Xbox, if you have problems with the AC'97 codec detection.");
|
MODULE_PARM_DESC(xbox, "Set to 1 for Xbox, if you have problems with the AC'97 codec detection.");
|
||||||
|
module_param(spdif_aclink, int, 0444);
|
||||||
|
MODULE_PARM_DESC(spdif_aclink, "S/PDIF over AC-link.");
|
||||||
|
|
||||||
/* just for backward compatibility */
|
/* just for backward compatibility */
|
||||||
static int enable;
|
static int enable;
|
||||||
|
@ -1578,10 +1581,14 @@ static int __devinit snd_intel8x0_pcm(struct intel8x0 *chip)
|
||||||
case DEVICE_INTEL_ICH4:
|
case DEVICE_INTEL_ICH4:
|
||||||
tbl = intel_pcms;
|
tbl = intel_pcms;
|
||||||
tblsize = ARRAY_SIZE(intel_pcms);
|
tblsize = ARRAY_SIZE(intel_pcms);
|
||||||
|
if (spdif_aclink)
|
||||||
|
tblsize--;
|
||||||
break;
|
break;
|
||||||
case DEVICE_NFORCE:
|
case DEVICE_NFORCE:
|
||||||
tbl = nforce_pcms;
|
tbl = nforce_pcms;
|
||||||
tblsize = ARRAY_SIZE(nforce_pcms);
|
tblsize = ARRAY_SIZE(nforce_pcms);
|
||||||
|
if (spdif_aclink)
|
||||||
|
tblsize--;
|
||||||
break;
|
break;
|
||||||
case DEVICE_ALI:
|
case DEVICE_ALI:
|
||||||
tbl = ali_pcms;
|
tbl = ali_pcms;
|
||||||
|
@ -2040,17 +2047,19 @@ static int __devinit snd_intel8x0_mixer(struct intel8x0 *chip, int ac97_clock,
|
||||||
};
|
};
|
||||||
|
|
||||||
chip->spdif_idx = -1; /* use PCMOUT (or disabled) */
|
chip->spdif_idx = -1; /* use PCMOUT (or disabled) */
|
||||||
switch (chip->device_type) {
|
if (!spdif_aclink) {
|
||||||
case DEVICE_NFORCE:
|
switch (chip->device_type) {
|
||||||
chip->spdif_idx = NVD_SPBAR;
|
case DEVICE_NFORCE:
|
||||||
break;
|
chip->spdif_idx = NVD_SPBAR;
|
||||||
case DEVICE_ALI:
|
break;
|
||||||
chip->spdif_idx = ALID_AC97SPDIFOUT;
|
case DEVICE_ALI:
|
||||||
break;
|
chip->spdif_idx = ALID_AC97SPDIFOUT;
|
||||||
case DEVICE_INTEL_ICH4:
|
break;
|
||||||
chip->spdif_idx = ICHD_SPBAR;
|
case DEVICE_INTEL_ICH4:
|
||||||
break;
|
chip->spdif_idx = ICHD_SPBAR;
|
||||||
};
|
break;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
chip->in_ac97_init = 1;
|
chip->in_ac97_init = 1;
|
||||||
|
|
||||||
|
@ -2173,11 +2182,11 @@ static int __devinit snd_intel8x0_mixer(struct intel8x0 *chip, int ac97_clock,
|
||||||
if ((igetdword(chip, ICHREG(GLOB_STA)) & ICH_SAMPLE_CAP) == ICH_SAMPLE_16_20)
|
if ((igetdword(chip, ICHREG(GLOB_STA)) & ICH_SAMPLE_CAP) == ICH_SAMPLE_16_20)
|
||||||
chip->smp20bit = 1;
|
chip->smp20bit = 1;
|
||||||
}
|
}
|
||||||
if (chip->device_type == DEVICE_NFORCE) {
|
if (chip->device_type == DEVICE_NFORCE && !spdif_aclink) {
|
||||||
/* 48kHz only */
|
/* 48kHz only */
|
||||||
chip->ichd[chip->spdif_idx].pcm->rates = SNDRV_PCM_RATE_48000;
|
chip->ichd[chip->spdif_idx].pcm->rates = SNDRV_PCM_RATE_48000;
|
||||||
}
|
}
|
||||||
if (chip->device_type == DEVICE_INTEL_ICH4) {
|
if (chip->device_type == DEVICE_INTEL_ICH4 && !spdif_aclink) {
|
||||||
/* use slot 10/11 for SPDIF */
|
/* use slot 10/11 for SPDIF */
|
||||||
u32 val;
|
u32 val;
|
||||||
val = igetdword(chip, ICHREG(GLOB_CNT)) & ~ICH_PCM_SPDIF_MASK;
|
val = igetdword(chip, ICHREG(GLOB_CNT)) & ~ICH_PCM_SPDIF_MASK;
|
||||||
|
@ -2305,7 +2314,7 @@ static int snd_intel8x0_ich_chip_init(struct intel8x0 *chip, int probing)
|
||||||
/* unmute the output on SIS7012 */
|
/* unmute the output on SIS7012 */
|
||||||
iputword(chip, 0x4c, igetword(chip, 0x4c) | 1);
|
iputword(chip, 0x4c, igetword(chip, 0x4c) | 1);
|
||||||
}
|
}
|
||||||
if (chip->device_type == DEVICE_NFORCE) {
|
if (chip->device_type == DEVICE_NFORCE && !spdif_aclink) {
|
||||||
/* enable SPDIF interrupt */
|
/* enable SPDIF interrupt */
|
||||||
unsigned int val;
|
unsigned int val;
|
||||||
pci_read_config_dword(chip->pci, 0x4c, &val);
|
pci_read_config_dword(chip->pci, 0x4c, &val);
|
||||||
|
@ -2398,7 +2407,7 @@ static int snd_intel8x0_free(struct intel8x0 *chip)
|
||||||
/* reset channels */
|
/* reset channels */
|
||||||
for (i = 0; i < chip->bdbars_count; i++)
|
for (i = 0; i < chip->bdbars_count; i++)
|
||||||
iputbyte(chip, ICH_REG_OFF_CR + chip->ichd[i].reg_offset, ICH_RESETREGS);
|
iputbyte(chip, ICH_REG_OFF_CR + chip->ichd[i].reg_offset, ICH_RESETREGS);
|
||||||
if (chip->device_type == DEVICE_NFORCE) {
|
if (chip->device_type == DEVICE_NFORCE && !spdif_aclink) {
|
||||||
/* stop the spdif interrupt */
|
/* stop the spdif interrupt */
|
||||||
unsigned int val;
|
unsigned int val;
|
||||||
pci_read_config_dword(chip->pci, 0x4c, &val);
|
pci_read_config_dword(chip->pci, 0x4c, &val);
|
||||||
|
@ -2492,7 +2501,7 @@ static int intel8x0_resume(struct pci_dev *pci)
|
||||||
snd_intel8x0_chip_init(chip, 0);
|
snd_intel8x0_chip_init(chip, 0);
|
||||||
|
|
||||||
/* re-initialize mixer stuff */
|
/* re-initialize mixer stuff */
|
||||||
if (chip->device_type == DEVICE_INTEL_ICH4) {
|
if (chip->device_type == DEVICE_INTEL_ICH4 && !spdif_aclink) {
|
||||||
/* enable separate SDINs for ICH4 */
|
/* enable separate SDINs for ICH4 */
|
||||||
iputbyte(chip, ICHREG(SDM), chip->sdm_saved);
|
iputbyte(chip, ICHREG(SDM), chip->sdm_saved);
|
||||||
/* use slot 10/11 for SPDIF */
|
/* use slot 10/11 for SPDIF */
|
||||||
|
@ -2928,6 +2937,29 @@ static struct shortname_table {
|
||||||
{ 0, NULL },
|
{ 0, NULL },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct snd_pci_quirk spdif_aclink_defaults[] __devinitdata = {
|
||||||
|
SND_PCI_QUIRK(0x147b, 0x1c1a, "ASUS KN8", 1),
|
||||||
|
{ } /* end */
|
||||||
|
};
|
||||||
|
|
||||||
|
/* look up white/black list for SPDIF over ac-link */
|
||||||
|
static int __devinit check_default_spdif_aclink(struct pci_dev *pci)
|
||||||
|
{
|
||||||
|
const struct snd_pci_quirk *w;
|
||||||
|
|
||||||
|
w = snd_pci_quirk_lookup(pci, spdif_aclink_defaults);
|
||||||
|
if (w) {
|
||||||
|
if (w->value)
|
||||||
|
snd_printdd(KERN_INFO "intel8x0: Using SPDIF over "
|
||||||
|
"AC-Link for %s\n", w->name);
|
||||||
|
else
|
||||||
|
snd_printdd(KERN_INFO "intel8x0: Using integrated "
|
||||||
|
"SPDIF DMA for %s\n", w->name);
|
||||||
|
return w->value;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int __devinit snd_intel8x0_probe(struct pci_dev *pci,
|
static int __devinit snd_intel8x0_probe(struct pci_dev *pci,
|
||||||
const struct pci_device_id *pci_id)
|
const struct pci_device_id *pci_id)
|
||||||
{
|
{
|
||||||
|
@ -2940,16 +2972,18 @@ static int __devinit snd_intel8x0_probe(struct pci_dev *pci,
|
||||||
if (card == NULL)
|
if (card == NULL)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
switch (pci_id->driver_data) {
|
if (spdif_aclink < 0)
|
||||||
case DEVICE_NFORCE:
|
spdif_aclink = check_default_spdif_aclink(pci);
|
||||||
strcpy(card->driver, "NFORCE");
|
|
||||||
break;
|
strcpy(card->driver, "ICH");
|
||||||
case DEVICE_INTEL_ICH4:
|
if (!spdif_aclink) {
|
||||||
strcpy(card->driver, "ICH4");
|
switch (pci_id->driver_data) {
|
||||||
break;
|
case DEVICE_NFORCE:
|
||||||
default:
|
strcpy(card->driver, "NFORCE");
|
||||||
strcpy(card->driver, "ICH");
|
break;
|
||||||
break;
|
case DEVICE_INTEL_ICH4:
|
||||||
|
strcpy(card->driver, "ICH4");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
strcpy(card->shortname, "Intel ICH");
|
strcpy(card->shortname, "Intel ICH");
|
||||||
|
|
Loading…
Reference in a new issue