blob: f41b3c2f046cd41f4cb0c53de13325086d7281a5 [file] [log] [blame]
Angel Pons182dbde2020-04-02 23:49:05 +02001/* SPDX-License-Identifier: GPL-2.0-only */
Aaron Durbin76c37002012-10-30 09:03:43 -05002
3#include <console/console.h>
4#include <device/device.h>
5#include <device/pci.h>
6#include <device/pci_ids.h>
7#include <device/pci_ops.h>
Kyösti Mälkki13f66502019-03-03 08:01:05 +02008#include <device/mmio.h>
Vladimir Serbinenko75c83872014-09-05 01:01:31 +02009#include <device/azalia_device.h>
Elyes HAOUASbf0970e2019-03-21 11:10:03 +010010
Aaron Durbin76c37002012-10-30 09:03:43 -050011#include "pch.h"
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070012#include "hda_verb.h"
Aaron Durbin76c37002012-10-30 09:03:43 -050013
Kevin Paul Herbertbde6d302014-12-24 18:43:20 -080014static void azalia_pch_init(struct device *dev, u8 *base)
Aaron Durbin76c37002012-10-30 09:03:43 -050015{
Aaron Durbin76c37002012-10-30 09:03:43 -050016 u8 reg8;
17 u16 reg16;
18 u32 reg32;
19
Angel Pons8963f7d2020-10-24 12:20:28 +020020 if (RCBA32(0x2030) & (1 << 31)) {
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030021 reg32 = pci_read_config32(dev, 0x120);
Aaron Durbin76c37002012-10-30 09:03:43 -050022 reg32 &= 0xf8ffff01;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070023 reg32 |= (1 << 25);
Aaron Durbin76c37002012-10-30 09:03:43 -050024 reg32 |= RCBA32(0x2030) & 0xfe;
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030025 pci_write_config32(dev, 0x120, reg32);
Aaron Durbin76c37002012-10-30 09:03:43 -050026
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070027 if (!pch_is_lp()) {
Angel Ponsbf9bc502020-06-08 00:12:43 +020028 pci_and_config16(dev, 0x78, ~(1 << 11));
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070029 }
Aaron Durbin76c37002012-10-30 09:03:43 -050030 } else
31 printk(BIOS_DEBUG, "Azalia: V1CTL disabled.\n");
32
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030033 reg32 = pci_read_config32(dev, 0x114);
Aaron Durbin76c37002012-10-30 09:03:43 -050034 reg32 &= ~0xfe;
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030035 pci_write_config32(dev, 0x114, reg32);
Aaron Durbin76c37002012-10-30 09:03:43 -050036
37 // Set VCi enable bit
Angel Ponsbf9bc502020-06-08 00:12:43 +020038 if (pci_read_config32(dev, 0x120) & ((1 << 24) | (1 << 25) | (1 << 26))) {
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030039 reg32 = pci_read_config32(dev, 0x120);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070040 if (pch_is_lp())
Angel Pons8963f7d2020-10-24 12:20:28 +020041 reg32 &= ~(1 << 31);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070042 else
Angel Pons8963f7d2020-10-24 12:20:28 +020043 reg32 |= (1 << 31);
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030044 pci_write_config32(dev, 0x120, reg32);
Aaron Durbin76c37002012-10-30 09:03:43 -050045 }
46
Aaron Durbin76c37002012-10-30 09:03:43 -050047 reg8 = pci_read_config8(dev, 0x43);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070048 if (pch_is_lp())
49 reg8 &= ~(1 << 6);
50 else
51 reg8 |= (1 << 4);
Aaron Durbin76c37002012-10-30 09:03:43 -050052 pci_write_config8(dev, 0x43, reg8);
53
Angel Ponsbf9bc502020-06-08 00:12:43 +020054 if (!pch_is_lp())
55 pci_or_config32(dev, 0xc0, 1 << 17);
Aaron Durbin76c37002012-10-30 09:03:43 -050056
57 /* Additional programming steps */
58 reg32 = pci_read_config32(dev, 0xc4);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070059 if (pch_is_lp())
60 reg32 |= (1 << 24);
61 else
62 reg32 |= (1 << 14);
Aaron Durbin76c37002012-10-30 09:03:43 -050063 pci_write_config32(dev, 0xc4, reg32);
64
Angel Ponsbf9bc502020-06-08 00:12:43 +020065 if (!pch_is_lp())
Angel Pons8963f7d2020-10-24 12:20:28 +020066 pci_and_config32(dev, 0xd0, ~(1 << 31));
Aaron Durbin76c37002012-10-30 09:03:43 -050067
Angel Ponsbf9bc502020-06-08 00:12:43 +020068 // Docking not supported
69 pci_and_config8(dev, 0x4d, (u8)~(1 << 7)); // Docking Status
Aaron Durbin76c37002012-10-30 09:03:43 -050070
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070071 if (pch_is_lp()) {
72 reg16 = read32(base + 0x0012);
73 reg16 |= (1 << 0);
74 write32(base + 0x0012, reg16);
75
76 /* disable Auto Voltage Detector */
Angel Ponsbf9bc502020-06-08 00:12:43 +020077 pci_or_config8(dev, 0x42, 1 << 2);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070078 }
79}
80
81static void azalia_init(struct device *dev)
82{
Kevin Paul Herbertbde6d302014-12-24 18:43:20 -080083 u8 *base;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070084 struct resource *res;
85 u32 codec_mask;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070086
87 /* Find base address */
Angel Ponsf32ae102021-11-03 13:07:14 +010088 res = probe_resource(dev, PCI_BASE_ADDRESS_0);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070089 if (!res)
90 return;
91
Kevin Paul Herbertbde6d302014-12-24 18:43:20 -080092 base = res2mmio(res, 0, 0);
93 printk(BIOS_DEBUG, "Azalia: base = %p\n", base);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070094
95 /* Set Bus Master */
Elyes HAOUAS73ae0762020-04-28 10:13:05 +020096 pci_or_config16(dev, PCI_COMMAND, PCI_COMMAND_MASTER);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070097
98 azalia_pch_init(dev, base);
99
100 codec_mask = hda_codec_detect(base);
Aaron Durbin76c37002012-10-30 09:03:43 -0500101
102 if (codec_mask) {
103 printk(BIOS_DEBUG, "Azalia: codec_mask = %02x\n", codec_mask);
Angel Pons4f8aea02021-11-10 18:09:04 +0100104 azalia_codecs_init(base, codec_mask);
Aaron Durbin76c37002012-10-30 09:03:43 -0500105 }
Aaron Durbin76c37002012-10-30 09:03:43 -0500106}
107
Angel Ponsc1301dd2021-03-18 20:35:19 +0100108static void azalia_final(struct device *dev)
109{
110 /* Set HDCFG.BCLD */
111 pci_or_config16(dev, 0x40, 1 << 1);
112}
113
Aaron Durbin76c37002012-10-30 09:03:43 -0500114static struct device_operations azalia_ops = {
115 .read_resources = pci_dev_read_resources,
116 .set_resources = pci_dev_set_resources,
117 .enable_resources = pci_dev_enable_resources,
118 .init = azalia_init,
Angel Ponsc1301dd2021-03-18 20:35:19 +0100119 .final = azalia_final,
Angel Pons1fc0edd2020-05-31 00:03:28 +0200120 .ops_pci = &pci_dev_ops_pci,
Aaron Durbin76c37002012-10-30 09:03:43 -0500121};
122
Felix Singer4ea08f92020-11-20 12:56:44 +0000123static const unsigned short pci_device_ids[] = {
Felix Singer43b7f412022-03-07 04:34:52 +0100124 PCI_DID_INTEL_LPT_H_AUDIO,
125 PCI_DID_INTEL_LPT_LP_AUDIO,
Felix Singer4ea08f92020-11-20 12:56:44 +0000126 0
127};
Aaron Durbin76c37002012-10-30 09:03:43 -0500128
129static const struct pci_driver pch_azalia __pci_driver = {
130 .ops = &azalia_ops,
Felix Singer43b7f412022-03-07 04:34:52 +0100131 .vendor = PCI_VID_INTEL,
Aaron Durbin76c37002012-10-30 09:03:43 -0500132 .devices = pci_device_ids,
133};