blob: 4122dbb3643e750b4624982d4249fa555ce358f4 [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 codecs_init(u8 *base, u32 codec_mask)
Aaron Durbin76c37002012-10-30 09:03:43 -050015{
16 int i;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070017
18 /* Can support up to 4 codecs */
Aaron Durbin76c37002012-10-30 09:03:43 -050019 for (i = 3; i >= 0; i--) {
20 if (codec_mask & (1 << i))
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070021 hda_codec_init(base, i,
22 cim_verb_data_size,
23 cim_verb_data);
Aaron Durbin76c37002012-10-30 09:03:43 -050024 }
25
Vladimir Serbinenko75c83872014-09-05 01:01:31 +020026 if (pc_beep_verbs_size)
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070027 hda_codec_write(base, pc_beep_verbs_size, pc_beep_verbs);
Aaron Durbin76c37002012-10-30 09:03:43 -050028}
29
Kevin Paul Herbertbde6d302014-12-24 18:43:20 -080030static void azalia_pch_init(struct device *dev, u8 *base)
Aaron Durbin76c37002012-10-30 09:03:43 -050031{
Aaron Durbin76c37002012-10-30 09:03:43 -050032 u8 reg8;
33 u16 reg16;
34 u32 reg32;
35
Ryan Salsamendi0d9b3602017-06-30 17:15:57 -070036 if (RCBA32(0x2030) & (1UL << 31)) {
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030037 reg32 = pci_read_config32(dev, 0x120);
Aaron Durbin76c37002012-10-30 09:03:43 -050038 reg32 &= 0xf8ffff01;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070039 reg32 |= (1 << 25);
Aaron Durbin76c37002012-10-30 09:03:43 -050040 reg32 |= RCBA32(0x2030) & 0xfe;
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030041 pci_write_config32(dev, 0x120, reg32);
Aaron Durbin76c37002012-10-30 09:03:43 -050042
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070043 if (!pch_is_lp()) {
44 reg16 = pci_read_config16(dev, 0x78);
45 reg16 &= ~(1 << 11);
46 pci_write_config16(dev, 0x78, reg16);
47 }
Aaron Durbin76c37002012-10-30 09:03:43 -050048 } else
49 printk(BIOS_DEBUG, "Azalia: V1CTL disabled.\n");
50
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030051 reg32 = pci_read_config32(dev, 0x114);
Aaron Durbin76c37002012-10-30 09:03:43 -050052 reg32 &= ~0xfe;
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030053 pci_write_config32(dev, 0x114, reg32);
Aaron Durbin76c37002012-10-30 09:03:43 -050054
55 // Set VCi enable bit
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030056 if (pci_read_config32(dev, 0x120) & ((1 << 24) |
Aaron Durbin76c37002012-10-30 09:03:43 -050057 (1 << 25) | (1 << 26))) {
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030058 reg32 = pci_read_config32(dev, 0x120);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070059 if (pch_is_lp())
Ryan Salsamendi0d9b3602017-06-30 17:15:57 -070060 reg32 &= ~(1UL << 31);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070061 else
Ryan Salsamendi0d9b3602017-06-30 17:15:57 -070062 reg32 |= (1UL << 31);
Kyösti Mälkki386b3e62013-07-26 08:52:49 +030063 pci_write_config32(dev, 0x120, reg32);
Aaron Durbin76c37002012-10-30 09:03:43 -050064 }
65
Aaron Durbin76c37002012-10-30 09:03:43 -050066 reg8 = pci_read_config8(dev, 0x43);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070067 if (pch_is_lp())
68 reg8 &= ~(1 << 6);
69 else
70 reg8 |= (1 << 4);
Aaron Durbin76c37002012-10-30 09:03:43 -050071 pci_write_config8(dev, 0x43, reg8);
72
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070073 if (!pch_is_lp()) {
74 reg32 = pci_read_config32(dev, 0xc0);
75 reg32 |= (1 << 17);
76 pci_write_config32(dev, 0xc0, reg32);
77 }
Aaron Durbin76c37002012-10-30 09:03:43 -050078
79 /* Additional programming steps */
80 reg32 = pci_read_config32(dev, 0xc4);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070081 if (pch_is_lp())
82 reg32 |= (1 << 24);
83 else
84 reg32 |= (1 << 14);
Aaron Durbin76c37002012-10-30 09:03:43 -050085 pci_write_config32(dev, 0xc4, reg32);
86
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070087 if (!pch_is_lp()) {
88 reg32 = pci_read_config32(dev, 0xd0);
Ryan Salsamendi0d9b3602017-06-30 17:15:57 -070089 reg32 &= ~(1UL << 31);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070090 pci_write_config32(dev, 0xd0, reg32);
91 }
Aaron Durbin76c37002012-10-30 09:03:43 -050092
Aaron Durbin76c37002012-10-30 09:03:43 -050093 reg8 = pci_read_config8(dev, 0x40); // Audio Control
Duncan Laurie0a7c49e2013-06-20 12:40:55 -070094 reg8 |= 1; // Select Azalia mode
Aaron Durbin76c37002012-10-30 09:03:43 -050095 pci_write_config8(dev, 0x40, reg8);
96
97 reg8 = pci_read_config8(dev, 0x4d); // Docking Status
98 reg8 &= ~(1 << 7); // Docking not supported
99 pci_write_config8(dev, 0x4d, reg8);
100
Duncan Laurie0a7c49e2013-06-20 12:40:55 -0700101 if (pch_is_lp()) {
102 reg16 = read32(base + 0x0012);
103 reg16 |= (1 << 0);
104 write32(base + 0x0012, reg16);
105
106 /* disable Auto Voltage Detector */
107 reg8 = pci_read_config8(dev, 0x42);
108 reg8 |= (1 << 2);
109 pci_write_config8(dev, 0x42, reg8);
110 }
111}
112
113static void azalia_init(struct device *dev)
114{
Kevin Paul Herbertbde6d302014-12-24 18:43:20 -0800115 u8 *base;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -0700116 struct resource *res;
117 u32 codec_mask;
Duncan Laurie0a7c49e2013-06-20 12:40:55 -0700118
119 /* Find base address */
120 res = find_resource(dev, PCI_BASE_ADDRESS_0);
121 if (!res)
122 return;
123
Kevin Paul Herbertbde6d302014-12-24 18:43:20 -0800124 base = res2mmio(res, 0, 0);
125 printk(BIOS_DEBUG, "Azalia: base = %p\n", base);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -0700126
127 /* Set Bus Master */
Elyes HAOUAS73ae0762020-04-28 10:13:05 +0200128 pci_or_config16(dev, PCI_COMMAND, PCI_COMMAND_MASTER);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -0700129
130 azalia_pch_init(dev, base);
131
132 codec_mask = hda_codec_detect(base);
Aaron Durbin76c37002012-10-30 09:03:43 -0500133
134 if (codec_mask) {
135 printk(BIOS_DEBUG, "Azalia: codec_mask = %02x\n", codec_mask);
Duncan Laurie0a7c49e2013-06-20 12:40:55 -0700136 codecs_init(base, codec_mask);
Aaron Durbin76c37002012-10-30 09:03:43 -0500137 }
Aaron Durbin76c37002012-10-30 09:03:43 -0500138}
139
Aaron Durbin76c37002012-10-30 09:03:43 -0500140static struct pci_operations azalia_pci_ops = {
Subrata Banik4a0f0712019-03-20 14:29:47 +0530141 .set_subsystem = pci_dev_set_subsystem,
Aaron Durbin76c37002012-10-30 09:03:43 -0500142};
143
144static struct device_operations azalia_ops = {
145 .read_resources = pci_dev_read_resources,
146 .set_resources = pci_dev_set_resources,
147 .enable_resources = pci_dev_enable_resources,
148 .init = azalia_init,
Aaron Durbin76c37002012-10-30 09:03:43 -0500149 .ops_pci = &azalia_pci_ops,
150};
151
Aaron Durbined095ca2013-05-20 15:57:16 -0500152static const unsigned short pci_device_ids[] = { 0x8c20, 0x9c20, 0 };
Aaron Durbin76c37002012-10-30 09:03:43 -0500153
154static const struct pci_driver pch_azalia __pci_driver = {
155 .ops = &azalia_ops,
156 .vendor = PCI_VENDOR_ID_INTEL,
157 .devices = pci_device_ids,
158};