2008-10-29 05:46:52 +01:00
|
|
|
/*
|
|
|
|
* This file is part of the coreboot project.
|
|
|
|
*
|
2009-01-20 23:53:10 +01:00
|
|
|
* Copyright (C) 2008-2009 coresystems GmbH
|
2008-10-29 05:46:52 +01:00
|
|
|
*
|
2009-03-11 15:54:18 +01:00
|
|
|
* This program 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; version 2 of
|
|
|
|
* the License.
|
2008-10-29 05:46:52 +01:00
|
|
|
*
|
|
|
|
* This program 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <console/console.h>
|
|
|
|
#include <device/device.h>
|
|
|
|
#include <device/pci.h>
|
|
|
|
#include <device/pci_ids.h>
|
2009-01-20 23:53:10 +01:00
|
|
|
#include <arch/io.h>
|
|
|
|
#include <delay.h>
|
2008-10-29 05:46:52 +01:00
|
|
|
#include "i82801gx.h"
|
|
|
|
|
2009-01-20 23:53:10 +01:00
|
|
|
#define NAMBAR 0x10
|
|
|
|
#define MASTER_VOL 0x02
|
|
|
|
#define PAGING 0x24
|
|
|
|
#define EXT_AUDIO 0x28
|
|
|
|
#define FUNC_SEL 0x66
|
|
|
|
#define INFO_IO 0x68
|
|
|
|
#define CONNECTOR 0x6a
|
|
|
|
#define VENDOR_ID1 0x7c
|
|
|
|
#define VENDOR_ID2 0x7e
|
|
|
|
#define SEC_VENDOR_ID1 0xfc
|
|
|
|
#define SEC_VENDOR_ID2 0xfe
|
|
|
|
|
|
|
|
#define NABMBAR 0x14
|
|
|
|
#define GLOB_CNT 0x2c
|
|
|
|
#define GLOB_STA 0x30
|
|
|
|
#define CAS 0x34
|
|
|
|
|
|
|
|
#define MMBAR 0x10
|
|
|
|
#define EXT_MODEM_ID1 0x3c
|
|
|
|
#define EXT_MODEM_ID2 0xbc
|
|
|
|
|
|
|
|
#define MBAR 0x14
|
|
|
|
#define SEC_CODEC 0x40
|
|
|
|
|
|
|
|
|
|
|
|
/* FIXME. This table is probably mainboard specific */
|
|
|
|
static u16 ac97_function[16*2][4] = {
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) },
|
|
|
|
{ (1 << 5), (2 << 11), (1 << 10), (3 << 13) }
|
|
|
|
};
|
|
|
|
|
|
|
|
static u16 nabmbar;
|
|
|
|
static u16 nambar;
|
|
|
|
|
|
|
|
static int ac97_semaphore(void)
|
|
|
|
{
|
|
|
|
int timeout;
|
|
|
|
u8 reg8;
|
|
|
|
|
|
|
|
timeout = 0xffff;
|
|
|
|
do {
|
|
|
|
reg8 = inb(nabmbar + CAS);
|
|
|
|
timeout--;
|
|
|
|
} while ((reg8 & 1) && timeout);
|
|
|
|
if (! timeout) {
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "Timeout!\n");
|
2009-01-20 23:53:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return (!timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_cnr(void)
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
|
|
|
|
static void program_sigid(struct device *dev, u32 id)
|
|
|
|
{
|
|
|
|
pci_write_config32(dev, 0x2c, id);
|
|
|
|
}
|
|
|
|
|
2008-10-29 05:46:52 +01:00
|
|
|
static void ac97_audio_init(struct device *dev)
|
|
|
|
{
|
2009-01-20 23:53:10 +01:00
|
|
|
u16 reg16;
|
|
|
|
u32 reg32;
|
|
|
|
int i;
|
|
|
|
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "Initializing AC'97 Audio.\n");
|
2009-01-20 23:53:10 +01:00
|
|
|
|
|
|
|
/* top 16 bits are zero, so don't read them */
|
|
|
|
nabmbar = pci_read_config16(dev, NABMBAR) & 0xfffe;
|
|
|
|
nambar = pci_read_config16(dev, NAMBAR) & 0xfffe;
|
|
|
|
|
|
|
|
reg16 = inw(nabmbar + GLOB_CNT);
|
|
|
|
reg16 |= (1 << 1); /* Remove AC_RESET# */
|
|
|
|
outw(reg16, nabmbar + GLOB_CNT);
|
|
|
|
|
|
|
|
/* Wait 600ms. Ouch. */
|
|
|
|
udelay(600 * 1000);
|
|
|
|
|
|
|
|
init_cnr();
|
|
|
|
|
|
|
|
/* Detect Primary AC'97 Codec */
|
|
|
|
reg32 = inl(nabmbar + GLOB_STA);
|
|
|
|
if ((reg32 & ((1 << 28) | (1 << 9) | (1 << 8))) == 0) {
|
|
|
|
/* Primary Codec not found */
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "No primary codec. Disabling AC'97 Audio.\n");
|
2009-01-20 23:53:10 +01:00
|
|
|
return;
|
|
|
|
}
|
2009-08-12 18:08:05 +02:00
|
|
|
|
2009-01-20 23:53:10 +01:00
|
|
|
ac97_semaphore();
|
|
|
|
|
|
|
|
/* Detect if codec is programmable */
|
|
|
|
outw(0x8000, nambar + MASTER_VOL);
|
|
|
|
ac97_semaphore();
|
|
|
|
if (inw(nambar + MASTER_VOL) != 0x8000) {
|
2010-03-22 12:42:32 +01:00
|
|
|
printk(BIOS_DEBUG, "Codec not programmable. Disabling AC'97 Audio.\n");
|
2009-01-20 23:53:10 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Program Vendor IDs */
|
|
|
|
reg32 = inw(nambar + VENDOR_ID1);
|
|
|
|
reg32 <<= 16;
|
|
|
|
reg32 |= (u16)inw(nambar + VENDOR_ID2);
|
|
|
|
|
|
|
|
program_sigid(dev, reg32);
|
|
|
|
|
|
|
|
/* Is Codec AC'97 2.3 compliant? */
|
|
|
|
reg16 = inw(nambar + EXT_AUDIO);
|
|
|
|
/* [11:10] = 10b -> AC'97 2.3 */
|
|
|
|
if ((reg16 & 0x0c00) != 0x0800) {
|
|
|
|
/* No 2.3 Codec. We're done */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Select Page 1 */
|
|
|
|
reg16 = inw(nambar + PAGING);
|
|
|
|
reg16 &= 0xfff0;
|
|
|
|
reg16 |= 0x0001;
|
|
|
|
outw(reg16, nambar + PAGING);
|
|
|
|
|
|
|
|
for (i = 0x0a * 2; i > 0; i--) {
|
|
|
|
outw(i, nambar + FUNC_SEL);
|
|
|
|
|
|
|
|
/* Function could not be selected. Next one */
|
|
|
|
if (inw(nambar + FUNC_SEL) != i)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
reg16 = inw(nambar + INFO_IO);
|
|
|
|
|
|
|
|
/* Function Information present? */
|
|
|
|
if (!(reg16 & (1 << 0)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Function Information valid? */
|
|
|
|
if (!(reg16 & (1 << 4)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Program Buffer Delay [9:5] */
|
|
|
|
reg16 &= 0x03e0;
|
|
|
|
reg16 |= ac97_function[i][0];
|
|
|
|
|
|
|
|
/* Program Gain [15:11] */
|
|
|
|
reg16 |= ac97_function[i][1];
|
|
|
|
|
|
|
|
/* Program Inversion [10] */
|
|
|
|
reg16 |= ac97_function[i][2];
|
|
|
|
|
|
|
|
outw(reg16, nambar + INFO_IO);
|
|
|
|
|
|
|
|
/* Program Connector / Jack Location */
|
|
|
|
reg16 = inw(nambar + CONNECTOR);
|
|
|
|
reg16 &= 0x1fff;
|
|
|
|
reg16 |= ac97_function[i][3];
|
|
|
|
outw(reg16, nambar + CONNECTOR);
|
|
|
|
}
|
2008-10-29 05:46:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ac97_modem_init(struct device *dev)
|
|
|
|
{
|
2009-01-20 23:53:10 +01:00
|
|
|
u16 reg16;
|
|
|
|
u32 reg32;
|
|
|
|
u16 mmbar, mbar;
|
|
|
|
|
|
|
|
mmbar = pci_read_config16(dev, MMBAR) & 0xfffe;
|
|
|
|
mbar = pci_read_config16(dev, MBAR) & 0xfffe;
|
|
|
|
|
|
|
|
reg16 = inw(mmbar + EXT_MODEM_ID1);
|
|
|
|
if ((reg16 & 0xc000) != 0xc000 ) {
|
|
|
|
if (reg16 & (1 << 0)) {
|
|
|
|
reg32 = inw(mmbar + VENDOR_ID2);
|
|
|
|
reg32 <<= 16;
|
|
|
|
reg32 |= (u16)inw(mmbar + VENDOR_ID1);
|
|
|
|
program_sigid(dev, reg32);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Secondary codec? */
|
|
|
|
reg16 = inw(mbar + SEC_CODEC);
|
|
|
|
if ((reg16 & (1 << 9)) == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
reg16 = inw(mmbar + EXT_MODEM_ID2);
|
|
|
|
if ((reg16 & 0xc000) == 0x4000) {
|
|
|
|
if (reg16 & (1 << 0)) {
|
|
|
|
reg32 = inw(mmbar + SEC_VENDOR_ID2);
|
|
|
|
reg32 <<= 16;
|
|
|
|
reg32 |= (u16)inw(mmbar + SEC_VENDOR_ID1);
|
|
|
|
program_sigid(dev, reg32);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2008-10-29 05:46:52 +01:00
|
|
|
}
|
|
|
|
|
2009-07-21 23:50:34 +02:00
|
|
|
static void ac97_set_subsystem(device_t dev, unsigned vendor, unsigned device)
|
|
|
|
{
|
|
|
|
if (!vendor || !device) {
|
|
|
|
pci_write_config32(dev, PCI_SUBSYSTEM_VENDOR_ID,
|
|
|
|
pci_read_config32(dev, PCI_VENDOR_ID));
|
|
|
|
} else {
|
|
|
|
pci_write_config32(dev, PCI_SUBSYSTEM_VENDOR_ID,
|
|
|
|
((device & 0xffff) << 16) | (vendor & 0xffff));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct pci_operations ac97_pci_ops = {
|
|
|
|
.set_subsystem = ac97_set_subsystem,
|
|
|
|
};
|
|
|
|
|
2008-10-29 05:46:52 +01:00
|
|
|
static struct device_operations ac97_audio_ops = {
|
|
|
|
.read_resources = pci_dev_read_resources,
|
|
|
|
.set_resources = pci_dev_set_resources,
|
|
|
|
.enable_resources = pci_dev_enable_resources,
|
|
|
|
.init = ac97_audio_init,
|
|
|
|
.scan_bus = 0,
|
|
|
|
.enable = i82801gx_enable,
|
2009-07-21 23:50:34 +02:00
|
|
|
.ops_pci = &ac97_pci_ops,
|
2008-10-29 05:46:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct device_operations ac97_modem_ops = {
|
|
|
|
.read_resources = pci_dev_read_resources,
|
|
|
|
.set_resources = pci_dev_set_resources,
|
|
|
|
.enable_resources = pci_dev_enable_resources,
|
|
|
|
.init = ac97_modem_init,
|
|
|
|
.scan_bus = 0,
|
|
|
|
.enable = i82801gx_enable,
|
2009-07-21 23:50:34 +02:00
|
|
|
.ops_pci = &ac97_pci_ops,
|
2008-10-29 05:46:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/* 82801GB/GR/GDH/GBM/GHM (ICH7/ICH7R/ICH7DH/ICH7-M/ICH7-M DH) */
|
2008-10-29 14:51:31 +01:00
|
|
|
/* Note: 82801GU (ICH7-U) doesn't have AC97 audio. */
|
|
|
|
static const struct pci_driver i82801gx_ac97_audio __pci_driver = {
|
2008-10-29 05:46:52 +01:00
|
|
|
.ops = &ac97_audio_ops,
|
|
|
|
.vendor = PCI_VENDOR_ID_INTEL,
|
2008-10-31 19:41:09 +01:00
|
|
|
.device = 0x27de,
|
2008-10-29 05:46:52 +01:00
|
|
|
};
|
|
|
|
|
2008-10-29 14:51:31 +01:00
|
|
|
/* 82801GB/GR/GDH/GBM/GHM (ICH7/ICH7R/ICH7DH/ICH7-M/ICH7-M DH) */
|
|
|
|
/* Note: 82801GU (ICH7-U) doesn't have AC97 modem. */
|
|
|
|
static const struct pci_driver i82801gx_ac97_modem __pci_driver = {
|
2008-10-29 05:46:52 +01:00
|
|
|
.ops = &ac97_modem_ops,
|
|
|
|
.vendor = PCI_VENDOR_ID_INTEL,
|
2008-10-31 19:41:09 +01:00
|
|
|
.device = 0x27dd,
|
2008-10-29 05:46:52 +01:00
|
|
|
};
|