blob: b7c72cb140dfd38a7d4ac6757cfaca61e819cab6 [file] [log] [blame]
Stefan Reinauer23190272008-08-20 13:41:24 +00001/*
2 * inteltool - dump all registers on an Intel CPU + chipset based system.
3 *
Stefan Reinauer14e22772010-04-27 06:56:47 +00004 * Copyright (C) 2008-2010 by coresystems GmbH
Anton Kochkovc7fc4422012-07-21 06:36:47 +04005 * Copyright (C) 2012 Anton Kochkov
Stefan Reinauer14e22772010-04-27 06:56:47 +00006 *
Stefan Reinauer23190272008-08-20 13:41:24 +00007 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; version 2 of the License.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Stefan Reinauer23190272008-08-20 13:41:24 +000015 */
16
17#include <stdio.h>
18#include <stdlib.h>
Stefan Reinauera7b296d2011-11-14 12:40:34 -080019#include <inttypes.h>
Stefan Reinauer23190272008-08-20 13:41:24 +000020#include "inteltool.h"
21
Stefan Taunerdbc6fcd2013-06-20 18:05:06 +020022/* 320766 */
23static const io_register_t nehalem_dmi_registers[] = {
24 { 0x00, 4, "DMIVCH" }, // DMI Virtual Channel Capability Header
25 { 0x04, 4, "DMIVCCAP1" }, // DMI Port VC Capability Register 1
26 { 0x08, 4, "DMIVCCAP2" }, // DMI Port VC Capability Register 2
27 { 0x0C, 4, "DMIVCCTL" }, // DMI Port VC Control
28 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
29 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
30/* { 0x18, 2, "RSVD" }, // Reserved */
31 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
32 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
33 { 0x20, 4, "DMIVC1RCTL" }, // DMI VC1 Resource Control
34/* { 0x24, 2, "RSVD" }, // Reserved */
35 { 0x26, 2, "DMIVC1RSTS" }, // DMI VC1 Resource Status
36/* ... - Reserved */
37 { 0x84, 4, "DMILCAP" }, // DMI Link Capabilities
38 { 0x88, 2, "DMILCTL" }, // DMI Link Control
39 { 0x8A, 2, "DMILSTS" }, // DMI Link Status
40/* ... - Reserved */
41};
42
43/* 322812 */
44static const io_register_t westmere_dmi_registers[] = {
45 { 0x00, 4, "DMIVCECH" }, // DMI Virtual Channel Enhanced Capability
46 { 0x04, 4, "DMIPVCCAP1" }, // DMI Port VC Capability Register 1
47 { 0x08, 4, "DMIPVCCAP2" }, // DMI Port VC Capability Register 2
48 { 0x0C, 2, "DMIPVCCTL" }, // DMI Port VC Control
49/* { 0x0E, 2, "RSVD" }, // Reserved */
50 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
51 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
52/* { 0x18, 2, "RSVD" }, // Reserved */
53 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
54 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
55 { 0x20, 4, "DMIVC1RCTL1" }, // DMI VC1 Resource Control
56/* { 0x24, 2, "RSVD" }, // Reserved */
57 { 0x26, 2, "DMIC1RSTS" }, // DMI VC1 Resource Status
58/* ... - Reserved */
59 { 0x84, 4, "DMILCAP" }, // DMI Link Capabilities
60 { 0x88, 2, "DMILCTL" }, // DMI Link Control
61 { 0x8A, 2, "DMILSTS" }, // DMI Link Status
62/* ... - Reserved */
63};
64
Anton Kochkovc7fc4422012-07-21 06:36:47 +040065static const io_register_t sandybridge_dmi_registers[] = {
66 { 0x00, 4, "DMI VCECH" }, // DMI Virtual Channel Enhanced Capability
67 { 0x04, 4, "DMI PVCCAP1" }, // DMI Port VC Capability Register 1
68 { 0x08, 4, "DMI PVVAP2" }, // DMI Port VC Capability Register 2
69 { 0x0C, 2, "DMI PVCCTL" }, // DMI Port VC Control
70/* { 0x0E, 2, "RSVD" }, // Reserved */
71 { 0x10, 4, "DMI VC0RCAP" }, // DMI VC0 Resource Capability
72 { 0x14, 4, "DMI VC0RCTL" }, // DMI VC0 Resource Control
73/* { 0x18, 2, "RSVD" }, // Reserved */
74 { 0x1A, 2, "DMI VC0RSTS" }, // DMI VC0 Resource Status
75 { 0x1C, 4, "DMI VC1RCAP" }, // DMI VC1 Resource Capability
76 { 0x20, 4, "DMI VC1RCTL" }, // DMI VC1 Resource Control
77/* { 0x24, 2, "RSVD" }, // Reserved */
78 { 0x26, 2, "DMI VC1RSTS" }, // DMI VC1 Resource Status
79 { 0x28, 4, "DMI VCPRCAP" }, // DMI VCp Resource Capability
80 { 0x2C, 4, "DMI VCPRCTL" }, // DMI VCp Resource Control
81/* { 0x30, 2, "RSVD" }, // Reserved */
82 { 0x32, 2, "DMI VCPRSTS" }, // DMI VCp Resource Status
83 { 0x34, 4, "DMI VCMRCAP" }, // DMI VCm Resource Capability
84 { 0x38, 4, "DMI VCMRCTL" }, // DMI VCm Resource Control
85/* { 0x3C, 2, "RSVD" }, // Reserved */
86 { 0x3E, 2, "DMI VCMRSTS" }, // DMI VCm Resource Status
87/* { 0x40, 4, "RSVD" }, // Reserved */
88 { 0x44, 4, "DMI ESC" }, // DMI Element Self Description
89/* { 0x48, 8, "RSVD" }, // Reserved */
90 { 0x50, 4, "DMI LE1D" }, // DMI Link Entry 1 Description
91/* { 0x54, 4, "RSVD" }, // Reserved */
92 { 0x58, 4, "DMI LE1A" }, // DMI Link Entry 1 Address
93 { 0x5C, 4, "DMI LUE1A" }, // DMI Link Upper Entry 1 Address
94 { 0x60, 4, "DMI LE2D" }, // DMI Link Entry 2 Description
95/* { 0x64, 4, "RSVD" }, // Reserved */
96 { 0x68, 4, "DMI LE2A" }, // DMI Link Entry 2 Address
97/* { 0x6C, 4, "RSVD" }, // Reserved
98 { 0x70, 8, "RSVD" }, // Reserved
99 { 0x78, 8, "RSVD" }, // Reserved
100 { 0x80, 4, "RSVD" }, // Reserved */
101 { 0x84, 4, "LCAP" }, // Link Capabilities
102 { 0x88, 2, "LCTL" }, // Link Control
103 { 0x8A, 2, "LSTS" }, // Link Status
104/* { 0x8C, 4, "RSVD" }, // Reserved
105 { 0x90, 4, "RSVD" }, // Reserved
106 { 0x94, 4, "RSVD" }, // Reserved */
107 { 0x98, 2, "LCTL2" }, // Link Control 2
108 { 0x9A, 2, "LSTS2" }, // Link Status 2
109/* ... - Reserved */
110 { 0xBC0, 4, "AFE_BMUF0" }, // AFE BMU Configuration Function 0
111 { 0xBC4, 4, "RSVD" }, // Reserved
112 { 0xBC8, 4, "RSVD" }, // Reserved
113 { 0xBCC, 4, "AFE_BMUT0" }, // AFE BMU Configuration Test 0
114/* ... - Reserved */
115};
116
Stefan Reinauer23190272008-08-20 13:41:24 +0000117/*
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100118 * All Haswell DMI Registers per
119 *
120 * Mobile 4th Generation Intel Core TM Processor Family, Mobile Intel Pentium Processor Family,
121 * and Mobile Intel Celeron Processor Family
122 * Datasheet Volume 2
123 * 329002-002
124 */
125static const io_register_t haswell_ult_dmi_registers[] = {
Elyes HAOUAS94501502016-10-19 17:59:10 +0200126 { 0x00, 4, "DMIVCECH" }, // DMI Virtual Channel Enhanced Capability
127 { 0x04, 4, "DMIPVCCAP1" }, // DMI Port VC Capability Register 1
128 { 0x08, 4, "DMIPVCCAP2" }, // DMI Port VC Capability Register 2
129 { 0x0C, 2, "DMI PVCCTL" }, // DMI Port VC Control
130/* { 0x0E, 2, "RSVD" }, // Reserved */
131 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
132 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
133/* { 0x18, 2, "RSVD" }, // Reserved */
134 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
135 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
136 { 0x20, 4, "DMIVC1RCTL" }, // DMI VC1 Resource Control
137/* { 0x24, 2, "RSVD" }, // Reserved */
138 { 0x26, 2, "DMIVC1RSTS" }, // DMI VC1 Resource Status
139 { 0x28, 4, "DMIVCPRCAP" }, // DMI VCp Resource Capability
140 { 0x2C, 4, "DMIVCPRCTL" }, // DMI VCp Resource Control
141/* { 0x30, 2, "RSVD" }, // Reserved */
142 { 0x32, 2, "DMIVCPRSTS" }, // DMI VCp Resource Status
143 { 0x34, 4, "DMIVCMRCAP" }, // DMI VCm Resource Capability
144 { 0x38, 4, "DMIVCMRCTL" }, // DMI VCm Resource Control
145/* { 0x3C, 2, "RSVD" }, // Reserved */
146 { 0x3E, 2, "DMIVCMRSTS" }, // DMI VCm Resource Status
147 { 0x40, 4, "DMIRCLDECH" }, // DMI Root Complex Link Declaration */
148 { 0x44, 4, "DMIESD" }, // DMI Element Self Description
149/* { 0x48, 4, "RSVD" }, // Reserved */
150/* { 0x4C, 4, "RSVD" }, // Reserved */
151 { 0x50, 4, "DMILE1D" }, // DMI Link Entry 1 Description
152/* { 0x54, 4, "RSVD" }, // Reserved */
153 { 0x58, 4, "DMILE1A" }, // DMI Link Entry 1 Address
154 { 0x5C, 4, "DMILUE1A" }, // DMI Link Upper Entry 1 Address
155 { 0x60, 4, "DMILE2D" }, // DMI Link Entry 2 Description
156/* { 0x64, 4, "RSVD" }, // Reserved */
157 { 0x68, 4, "DMILE2A" }, // DMI Link Entry 2 Address
158/* { 0x6C, 4, "RSVD" }, // Reserved */
159/* { 0x70, 4, "RSVD" }, // Reserved */
160/* { 0x74, 4, "RSVD" }, // Reserved */
161/* { 0x78, 4, "RSVD" }, // Reserved */
162/* { 0x7C, 4, "RSVD" }, // Reserved */
163/* { 0x80, 4, "RSVD" }, // Reserved */
164/* { 0x84, 4, "RSVD" }, // Reserved */
165 { 0x88, 2, "LCTL" }, // Link Control
166 /* ... - Reserved */
167 { 0x1C4, 4, "DMIUESTS" }, // DMI Uncorrectable Error Status
168 { 0x1C8, 4, "DMIUEMSK" }, // DMI Uncorrectable Error Mask
169 { 0x1D0, 4, "DMICESTS" }, // DMI Correctable Error Status
170 { 0x1D4, 4, "DMICEMSK" }, // DMI Correctable Error Mask
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100171/* ... - Reserved */
172};
173
174/*
Maximilian Schander98c11dd2017-11-05 05:52:13 +0100175 * All Skylake-S/H DMI Registers per
176 *
177 * 6th Generation Intel Processor Families for S-Platform Volume 2 of 2
178 * Page 117
179 * 332688-003E
180 *
181 * 6th Generation Intel Processor Families for H-Platform Volume 2 of 2
182 * Page 117
183 * 332987-002EN
184 */
185static const io_register_t skylake_dmi_registers[] = {
186 { 0x00, 4, "DMIVCECH" }, // DMI Virtual Channel Enhanced Capability
187 { 0x04, 4, "DMIPVCCAP1" }, // DMI Port VC Capability Register 1
188 { 0x08, 4, "DMIPVCCAP2" }, // DMI Port VC Capability Register 2
189 { 0x0C, 2, "DMIPVCCTL" }, // DMI Port VC Control
190 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
191 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
192 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
193 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
194 { 0x20, 4, "DMIVC1RCTL" }, // DMI VC1 Resource Control
195 { 0x26, 2, "DMIVC1RSTS" }, // DMI VC1 Resource Status
196 { 0x34, 4, "DMIVCMRCAP" }, // DMI VCm Resource Capability
197 { 0x38, 4, "DMIVCMRCTL" }, // DMI VCm Resource Control
198 { 0x3E, 2, "DMIVCMRSTS" }, // DMI VCm Resource Status
199 { 0x40, 4, "DMIRCLDECH" }, // DMI Root Complex Link Declaration */
200 { 0x44, 4, "DMIESD" }, // DMI Element Self Description
201 { 0x50, 4, "DMILE1D" }, // DMI Link Entry 1 Description
202 { 0x58, 4, "DMILE1A" }, // DMI Link Entry 1 Address
203 { 0x5C, 4, "DMILUE1A" }, // DMI Link Upper Entry 1 Address
204 { 0x60, 4, "DMILE2D" }, // DMI Link Entry 2 Description
205 { 0x68, 4, "DMILE2A" }, // DMI Link Entry 2 Address
206 { 0x84, 4, "LCAP" }, // Link Capabilities
207 { 0x88, 2, "LCTL" }, // Link Control
208 { 0x8A, 2, "LSTS" }, // DMI Link Status
209 { 0x98, 2, "LCTL2" }, // Link Control 2
210 { 0x9A, 2, "LSTS2" }, // DMI Link Status 2
211 { 0x1C4, 4, "DMIUESTS" }, // DMI Uncorrectable Error Status
212 { 0x1C8, 4, "DMIUEMSK" }, // DMI Uncorrectable Error Mask
213 { 0x1CC, 4, "DMIUESEV" }, // DMI Uncorrectable Error Mask
214 { 0x1D0, 4, "DMICESTS" }, // DMI Correctable Error Status
215 { 0x1D4, 4, "DMICEMSK" }, // DMI Correctable Error Mask
216};
217
218
219/*
Stefan Reinauer23190272008-08-20 13:41:24 +0000220 * Egress Port Root Complex MMIO configuration space
221 */
222int print_epbar(struct pci_dev *nb)
223{
224 int i, size = (4 * 1024);
225 volatile uint8_t *epbar;
Stefan Reinauer1162f252008-12-04 15:18:20 +0000226 uint64_t epbar_phys;
Stefan Reinauer23190272008-08-20 13:41:24 +0000227
228 printf("\n============= EPBAR =============\n\n");
229
230 switch (nb->device_id) {
Pat Erleyca3548e2010-04-21 06:23:19 +0000231 case PCI_DEVICE_ID_INTEL_82915:
Stefan Reinauer23190272008-08-20 13:41:24 +0000232 case PCI_DEVICE_ID_INTEL_82945GM:
Björn Busse2d33dc42010-08-01 15:33:30 +0000233 case PCI_DEVICE_ID_INTEL_82945GSE:
Stefan Reinauer3d9a12f2008-11-02 11:11:40 +0000234 case PCI_DEVICE_ID_INTEL_82945P:
Stefan Tauner1a00cf02012-10-13 06:23:52 +0200235 case PCI_DEVICE_ID_INTEL_82946:
Stefan Reinauer1162f252008-12-04 15:18:20 +0000236 case PCI_DEVICE_ID_INTEL_82975X:
Stefan Reinauer23190272008-08-20 13:41:24 +0000237 epbar_phys = pci_read_long(nb, 0x40) & 0xfffffffe;
238 break;
Stefan Tauner04c06002012-10-13 02:19:30 +0200239 case PCI_DEVICE_ID_INTEL_82965PM:
240 case PCI_DEVICE_ID_INTEL_82Q965:
241 case PCI_DEVICE_ID_INTEL_82Q35:
242 case PCI_DEVICE_ID_INTEL_82G33:
243 case PCI_DEVICE_ID_INTEL_82Q33:
244 case PCI_DEVICE_ID_INTEL_82X38:
Ruud Schrampbb41f502011-04-04 07:53:19 +0200245 case PCI_DEVICE_ID_INTEL_32X0:
Damien Zammit9c986642015-08-17 21:04:41 +1000246 case PCI_DEVICE_ID_INTEL_82XX4X:
247 case PCI_DEVICE_ID_INTEL_82Q45:
248 case PCI_DEVICE_ID_INTEL_82G45:
249 case PCI_DEVICE_ID_INTEL_82G41:
250 case PCI_DEVICE_ID_INTEL_82B43:
251 case PCI_DEVICE_ID_INTEL_82B43_2:
Corey Osgood23d98c72010-07-29 19:25:31 +0000252 case PCI_DEVICE_ID_INTEL_ATOM_DXXX:
253 case PCI_DEVICE_ID_INTEL_ATOM_NXXX:
Felix Heldfac95e32014-11-09 00:11:28 +0100254 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_D:
255 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_M:
256 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_E3:
257 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_D:
258 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_M:
259 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_E3:
260 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_015c:
261 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_D:
262 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_M:
263 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_E3:
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100264 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_U:
Matt DeVillier5b667df2015-05-14 21:58:33 -0500265 case PCI_DEVICE_ID_INTEL_CORE_5TH_GEN_U:
Nico Huber54fe32f2017-10-03 16:03:07 +0200266 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_D2:
Maximilian Schander79856432017-11-05 06:14:55 +0100267 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_M:
268 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_WST:
Christoph Pomaska48ac29e2018-01-01 01:48:21 +0100269 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_D:
Maxim Polyakov13176892019-08-27 18:20:08 +0300270 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_E:
Matthew Garrett2bf28e52018-07-23 21:09:47 -0700271 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_U:
272 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_Y:
273 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_U_Q:
Christian Walter9a8c5e72019-05-06 17:50:57 +0200274 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_E3:
Stefan Tauner04c06002012-10-13 02:19:30 +0200275 epbar_phys = pci_read_long(nb, 0x40) & 0xfffffffe;
276 epbar_phys |= ((uint64_t)pci_read_long(nb, 0x44)) << 32;
277 break;
Stefan Reinauerb2aedb12009-08-29 15:45:43 +0000278 case PCI_DEVICE_ID_INTEL_82810:
Stefan Tauner04c06002012-10-13 02:19:30 +0200279 case PCI_DEVICE_ID_INTEL_82810_DC:
280 case PCI_DEVICE_ID_INTEL_82810E_DC:
Stefan Reinauer04844812010-02-22 11:26:06 +0000281 case PCI_DEVICE_ID_INTEL_82830M:
Idwer Vollering312fc962010-12-17 22:34:58 +0000282 case PCI_DEVICE_ID_INTEL_82865:
283 printf("This northbridge does not have EPBAR.\n");
Stefan Reinauer23190272008-08-20 13:41:24 +0000284 return 1;
285 default:
286 printf("Error: Dumping EPBAR on this northbridge is not (yet) supported.\n");
287 return 1;
288 }
289
Stefan Reinauer1162f252008-12-04 15:18:20 +0000290 epbar = map_physical(epbar_phys, size);
Stefan Reinauer14e22772010-04-27 06:56:47 +0000291
Stefan Reinauer1162f252008-12-04 15:18:20 +0000292 if (epbar == NULL) {
Stefan Reinauer23190272008-08-20 13:41:24 +0000293 perror("Error mapping EPBAR");
294 exit(1);
295 }
296
Stefan Reinauera7b296d2011-11-14 12:40:34 -0800297 printf("EPBAR = 0x%08" PRIx64 " (MEM)\n\n", epbar_phys);
Stefan Reinauer23190272008-08-20 13:41:24 +0000298 for (i = 0; i < size; i += 4) {
299 if (*(uint32_t *)(epbar + i))
300 printf("0x%04x: 0x%08x\n", i, *(uint32_t *)(epbar+i));
301 }
302
Stefan Reinauer1162f252008-12-04 15:18:20 +0000303 unmap_physical((void *)epbar, size);
Stefan Reinauer23190272008-08-20 13:41:24 +0000304 return 0;
305}
306
307/*
308 * MCH-ICH Serial Interconnect Ingress Root Complex MMIO configuration space
309 */
310int print_dmibar(struct pci_dev *nb)
311{
312 int i, size = (4 * 1024);
313 volatile uint8_t *dmibar;
Stefan Reinauer1162f252008-12-04 15:18:20 +0000314 uint64_t dmibar_phys;
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400315 const io_register_t *dmi_registers = NULL;
Stefan Reinauer23190272008-08-20 13:41:24 +0000316
317 printf("\n============= DMIBAR ============\n\n");
318
319 switch (nb->device_id) {
Pat Erleyca3548e2010-04-21 06:23:19 +0000320 case PCI_DEVICE_ID_INTEL_82915:
Stefan Reinauer23190272008-08-20 13:41:24 +0000321 case PCI_DEVICE_ID_INTEL_82945GM:
Björn Busse2d33dc42010-08-01 15:33:30 +0000322 case PCI_DEVICE_ID_INTEL_82945GSE:
Stefan Reinauer3d9a12f2008-11-02 11:11:40 +0000323 case PCI_DEVICE_ID_INTEL_82945P:
Stefan Reinauer1162f252008-12-04 15:18:20 +0000324 case PCI_DEVICE_ID_INTEL_82975X:
Stefan Reinauer23190272008-08-20 13:41:24 +0000325 dmibar_phys = pci_read_long(nb, 0x4c) & 0xfffffffe;
326 break;
Stefan Tauner1a00cf02012-10-13 06:23:52 +0200327 case PCI_DEVICE_ID_INTEL_82946:
Stefan Tauner04c06002012-10-13 02:19:30 +0200328 case PCI_DEVICE_ID_INTEL_82965PM:
329 case PCI_DEVICE_ID_INTEL_82Q965:
Warren Turkal53291952010-09-03 09:32:17 +0000330 case PCI_DEVICE_ID_INTEL_82Q35:
331 case PCI_DEVICE_ID_INTEL_82G33:
332 case PCI_DEVICE_ID_INTEL_82Q33:
Stefan Tauner04c06002012-10-13 02:19:30 +0200333 case PCI_DEVICE_ID_INTEL_82X38:
Ruud Schrampbb41f502011-04-04 07:53:19 +0200334 case PCI_DEVICE_ID_INTEL_32X0:
Damien Zammit9c986642015-08-17 21:04:41 +1000335 case PCI_DEVICE_ID_INTEL_82XX4X:
336 case PCI_DEVICE_ID_INTEL_82Q45:
337 case PCI_DEVICE_ID_INTEL_82G45:
338 case PCI_DEVICE_ID_INTEL_82G41:
339 case PCI_DEVICE_ID_INTEL_82B43:
340 case PCI_DEVICE_ID_INTEL_82B43_2:
Corey Osgood23d98c72010-07-29 19:25:31 +0000341 case PCI_DEVICE_ID_INTEL_ATOM_DXXX:
342 case PCI_DEVICE_ID_INTEL_ATOM_NXXX:
Warren Turkal53291952010-09-03 09:32:17 +0000343 dmibar_phys = pci_read_long(nb, 0x68) & 0xfffffffe;
344 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
345 break;
Stefan Reinauerb2aedb12009-08-29 15:45:43 +0000346 case PCI_DEVICE_ID_INTEL_82810:
Stefan Tauner04c06002012-10-13 02:19:30 +0200347 case PCI_DEVICE_ID_INTEL_82810_DC:
348 case PCI_DEVICE_ID_INTEL_82810E_DC:
Idwer Vollering312fc962010-12-17 22:34:58 +0000349 case PCI_DEVICE_ID_INTEL_82865:
350 printf("This northbridge does not have DMIBAR.\n");
Stefan Reinauer23190272008-08-20 13:41:24 +0000351 return 1;
Stefan Tauner04c06002012-10-13 02:19:30 +0200352 case PCI_DEVICE_ID_INTEL_82X58:
Warren Turkal3235eea2010-09-03 09:31:13 +0000353 dmibar_phys = pci_read_long(nb, 0x50) & 0xfffff000;
354 break;
Stefan Taunerdbc6fcd2013-06-20 18:05:06 +0200355 case PCI_DEVICE_ID_INTEL_CORE_0TH_GEN:
356 /* DMIBAR is called DMIRCBAR in Nehalem */
357 dmibar_phys = pci_read_long(nb, 0x50) & 0xfffff000; /* 31:12 */
358 dmi_registers = nehalem_dmi_registers;
359 size = ARRAY_SIZE(nehalem_dmi_registers);
360 break;
Stefan Tauner04c06002012-10-13 02:19:30 +0200361 case PCI_DEVICE_ID_INTEL_CORE_1ST_GEN:
362 dmibar_phys = pci_read_long(nb, 0x68);
363 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
364 dmibar_phys &= 0x0000000ffffff000UL; /* 35:12 */
Stefan Taunerdbc6fcd2013-06-20 18:05:06 +0200365 dmi_registers = westmere_dmi_registers;
366 size = ARRAY_SIZE(westmere_dmi_registers);
Stefan Tauner04c06002012-10-13 02:19:30 +0200367 break;
Felix Held0cc8f292014-11-05 03:18:44 +0100368 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_D:
369 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_M:
Felix Heldfac95e32014-11-09 00:11:28 +0100370 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_E3:
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400371 dmi_registers = sandybridge_dmi_registers;
372 size = ARRAY_SIZE(sandybridge_dmi_registers);
Paul Menzelceac7872017-03-19 20:12:43 +0100373 /* fall through */
Felix Heldfac95e32014-11-09 00:11:28 +0100374 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_D: /* pretty printing not implemented yet */
375 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_M:
376 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_E3:
377 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_015c:
378 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_D:
379 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_M:
380 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_E3:
Stefan Tauner04c06002012-10-13 02:19:30 +0200381 dmibar_phys = pci_read_long(nb, 0x68);
382 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
383 dmibar_phys &= 0x0000007ffffff000UL; /* 38:12 */
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400384 break;
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100385 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_U:
Matt DeVillier5b667df2015-05-14 21:58:33 -0500386 case PCI_DEVICE_ID_INTEL_CORE_5TH_GEN_U:
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100387 dmi_registers = haswell_ult_dmi_registers;
388 size = ARRAY_SIZE(haswell_ult_dmi_registers);
389 dmibar_phys = pci_read_long(nb, 0x68);
390 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
391 dmibar_phys &= 0x0000007ffffff000UL; /* 38:12 */
392 break;
Nico Huber54fe32f2017-10-03 16:03:07 +0200393 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_D2:
Maximilian Schander98c11dd2017-11-05 05:52:13 +0100394 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_M:
395 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_WST:
Christoph Pomaska48ac29e2018-01-01 01:48:21 +0100396 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_D:
Maxim Polyakov13176892019-08-27 18:20:08 +0300397 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_E:
Matthew Garrett2bf28e52018-07-23 21:09:47 -0700398 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_U:
399 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_Y:
400 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_U_Q:
Christian Walter9a8c5e72019-05-06 17:50:57 +0200401 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_E3:
Maximilian Schander98c11dd2017-11-05 05:52:13 +0100402 dmi_registers = skylake_dmi_registers;
403 size = ARRAY_SIZE(skylake_dmi_registers);
404 dmibar_phys = pci_read_long(nb, 0x68);
405 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
406 dmibar_phys &= 0x0000007ffffff000UL; /* 38:12 */
407 break;
Stefan Reinauer23190272008-08-20 13:41:24 +0000408 default:
409 printf("Error: Dumping DMIBAR on this northbridge is not (yet) supported.\n");
410 return 1;
411 }
412
Stefan Reinauer1162f252008-12-04 15:18:20 +0000413 dmibar = map_physical(dmibar_phys, size);
Stefan Reinauer14e22772010-04-27 06:56:47 +0000414
Stefan Reinauer1162f252008-12-04 15:18:20 +0000415 if (dmibar == NULL) {
Stefan Reinauer23190272008-08-20 13:41:24 +0000416 perror("Error mapping DMIBAR");
417 exit(1);
418 }
419
Stefan Reinauera7b296d2011-11-14 12:40:34 -0800420 printf("DMIBAR = 0x%08" PRIx64 " (MEM)\n\n", dmibar_phys);
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400421 if (dmi_registers != NULL) {
422 for (i = 0; i < size; i++) {
423 switch (dmi_registers[i].size) {
424 case 4:
425 printf("dmibase+0x%04x: 0x%08x (%s)\n",
426 dmi_registers[i].addr,
427 *(uint32_t *)(dmibar+dmi_registers[i].addr),
428 dmi_registers[i].name);
429 break;
430 case 2:
431 printf("dmibase+0x%04x: 0x%04x (%s)\n",
432 dmi_registers[i].addr,
433 *(uint16_t *)(dmibar+dmi_registers[i].addr),
434 dmi_registers[i].name);
435 break;
436 case 1:
437 printf("dmibase+0x%04x: 0x%02x (%s)\n",
438 dmi_registers[i].addr,
439 *(uint8_t *)(dmibar+dmi_registers[i].addr),
440 dmi_registers[i].name);
441 break;
442 }
443 }
444 } else {
445 for (i = 0; i < size; i += 4) {
446 if (*(uint32_t *)(dmibar + i))
447 printf("0x%04x: 0x%08x\n", i, *(uint32_t *)(dmibar+i));
448 }
Stefan Reinauer23190272008-08-20 13:41:24 +0000449 }
450
Stefan Reinauer1162f252008-12-04 15:18:20 +0000451 unmap_physical((void *)dmibar, size);
Stefan Reinauer23190272008-08-20 13:41:24 +0000452 return 0;
453}
454
455/*
456 * PCIe MMIO configuration space
457 */
458int print_pciexbar(struct pci_dev *nb)
459{
Stefan Reinauer1162f252008-12-04 15:18:20 +0000460 uint64_t pciexbar_reg;
461 uint64_t pciexbar_phys;
Stefan Reinauer23190272008-08-20 13:41:24 +0000462 volatile uint8_t *pciexbar;
463 int max_busses, devbase, i;
464 int bus, dev, fn;
465
466 printf("========= PCIEXBAR ========\n\n");
467
468 switch (nb->device_id) {
Pat Erleyca3548e2010-04-21 06:23:19 +0000469 case PCI_DEVICE_ID_INTEL_82915:
Stefan Reinauer23190272008-08-20 13:41:24 +0000470 case PCI_DEVICE_ID_INTEL_82945GM:
Björn Busse2d33dc42010-08-01 15:33:30 +0000471 case PCI_DEVICE_ID_INTEL_82945GSE:
Stefan Reinauer3d9a12f2008-11-02 11:11:40 +0000472 case PCI_DEVICE_ID_INTEL_82945P:
Stefan Reinauer1162f252008-12-04 15:18:20 +0000473 case PCI_DEVICE_ID_INTEL_82975X:
Stefan Reinauer23190272008-08-20 13:41:24 +0000474 pciexbar_reg = pci_read_long(nb, 0x48);
475 break;
Stefan Tauner1a00cf02012-10-13 06:23:52 +0200476 case PCI_DEVICE_ID_INTEL_82946:
Stefan Tauner04c06002012-10-13 02:19:30 +0200477 case PCI_DEVICE_ID_INTEL_82965PM:
478 case PCI_DEVICE_ID_INTEL_82Q965:
479 case PCI_DEVICE_ID_INTEL_82Q35:
480 case PCI_DEVICE_ID_INTEL_82G33:
481 case PCI_DEVICE_ID_INTEL_82Q33:
482 case PCI_DEVICE_ID_INTEL_82X38:
Ruud Schrampbb41f502011-04-04 07:53:19 +0200483 case PCI_DEVICE_ID_INTEL_32X0:
Damien Zammit9c986642015-08-17 21:04:41 +1000484 case PCI_DEVICE_ID_INTEL_82XX4X:
485 case PCI_DEVICE_ID_INTEL_82Q45:
486 case PCI_DEVICE_ID_INTEL_82G45:
487 case PCI_DEVICE_ID_INTEL_82G41:
488 case PCI_DEVICE_ID_INTEL_82B43:
489 case PCI_DEVICE_ID_INTEL_82B43_2:
Corey Osgood23d98c72010-07-29 19:25:31 +0000490 case PCI_DEVICE_ID_INTEL_ATOM_DXXX:
491 case PCI_DEVICE_ID_INTEL_ATOM_NXXX:
Felix Heldfac95e32014-11-09 00:11:28 +0100492 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_D:
493 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_M:
494 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_E3:
495 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_D:
496 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_M:
497 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_E3:
498 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_015c:
499 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_D:
500 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_M:
501 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_E3:
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100502 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_U:
Matt DeVillier5b667df2015-05-14 21:58:33 -0500503 case PCI_DEVICE_ID_INTEL_CORE_5TH_GEN_U:
Nico Huber54fe32f2017-10-03 16:03:07 +0200504 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_D2:
Maximilian Schander79856432017-11-05 06:14:55 +0100505 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_M:
506 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_WST:
Christoph Pomaska48ac29e2018-01-01 01:48:21 +0100507 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_D:
Maxim Polyakov13176892019-08-27 18:20:08 +0300508 case PCI_DEVICE_ID_INTEL_CORE_6TH_GEN_E:
Matthew Garrett2bf28e52018-07-23 21:09:47 -0700509 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_U:
510 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_Y:
511 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_U_Q:
Christian Walter9a8c5e72019-05-06 17:50:57 +0200512 case PCI_DEVICE_ID_INTEL_CORE_7TH_GEN_E3:
Stefan Tauner04c06002012-10-13 02:19:30 +0200513 pciexbar_reg = pci_read_long(nb, 0x60);
514 pciexbar_reg |= ((uint64_t)pci_read_long(nb, 0x64)) << 32;
515 break;
Stefan Reinauerb2aedb12009-08-29 15:45:43 +0000516 case PCI_DEVICE_ID_INTEL_82810:
Stefan Tauner04c06002012-10-13 02:19:30 +0200517 case PCI_DEVICE_ID_INTEL_82810_DC:
518 case PCI_DEVICE_ID_INTEL_82810E_DC:
Idwer Vollering312fc962010-12-17 22:34:58 +0000519 case PCI_DEVICE_ID_INTEL_82865:
520 printf("Error: This northbridge does not have PCIEXBAR.\n");
Stefan Reinauer23190272008-08-20 13:41:24 +0000521 return 1;
522 default:
523 printf("Error: Dumping PCIEXBAR on this northbridge is not (yet) supported.\n");
524 return 1;
525 }
526
527 if (!(pciexbar_reg & (1 << 0))) {
528 printf("PCIEXBAR register is disabled.\n");
529 return 0;
530 }
531
532 switch ((pciexbar_reg >> 1) & 3) {
533 case 0: // 256MB
Paul Menzel17c05f22013-04-03 10:00:33 +0200534 pciexbar_phys = pciexbar_reg & (0xffULL << 28);
Stefan Reinauer23190272008-08-20 13:41:24 +0000535 max_busses = 256;
536 break;
537 case 1: // 128M
Paul Menzel17c05f22013-04-03 10:00:33 +0200538 pciexbar_phys = pciexbar_reg & (0x1ffULL << 27);
Stefan Reinauer23190272008-08-20 13:41:24 +0000539 max_busses = 128;
540 break;
541 case 2: // 64M
Paul Menzel17c05f22013-04-03 10:00:33 +0200542 pciexbar_phys = pciexbar_reg & (0x3ffULL << 26);
Stefan Reinauer23190272008-08-20 13:41:24 +0000543 max_busses = 64;
544 break;
545 default: // RSVD
546 printf("Undefined address base. Bailing out.\n");
547 return 1;
Stefan Reinauer14e22772010-04-27 06:56:47 +0000548 }
Stefan Reinauer23190272008-08-20 13:41:24 +0000549
Stefan Reinauera7b296d2011-11-14 12:40:34 -0800550 printf("PCIEXBAR: 0x%08" PRIx64 "\n", pciexbar_phys);
Stefan Reinauer23190272008-08-20 13:41:24 +0000551
Stefan Reinauer1162f252008-12-04 15:18:20 +0000552 pciexbar = map_physical(pciexbar_phys, (max_busses * 1024 * 1024));
Stefan Reinauer14e22772010-04-27 06:56:47 +0000553
Stefan Reinauer1162f252008-12-04 15:18:20 +0000554 if (pciexbar == NULL) {
Stefan Reinauer23190272008-08-20 13:41:24 +0000555 perror("Error mapping PCIEXBAR");
556 exit(1);
557 }
Stefan Reinauer14e22772010-04-27 06:56:47 +0000558
Stefan Reinauer23190272008-08-20 13:41:24 +0000559 for (bus = 0; bus < max_busses; bus++) {
560 for (dev = 0; dev < 32; dev++) {
561 for (fn = 0; fn < 8; fn++) {
562 devbase = (bus * 1024 * 1024) + (dev * 32 * 1024) + (fn * 4 * 1024);
563
564 if (*(uint16_t *)(pciexbar + devbase) == 0xffff)
565 continue;
Stefan Reinauer14e22772010-04-27 06:56:47 +0000566
Stefan Reinauer23190272008-08-20 13:41:24 +0000567 /* This is a heuristics. Anyone got a better check? */
568 if( (*(uint32_t *)(pciexbar + devbase + 256) == 0xffffffff) &&
569 (*(uint32_t *)(pciexbar + devbase + 512) == 0xffffffff) ) {
570#if DEBUG
571 printf("Skipped non-PCIe device %02x:%02x.%01x\n", bus, dev, fn);
572#endif
573 continue;
574 }
575
576 printf("\nPCIe %02x:%02x.%01x extended config space:", bus, dev, fn);
577 for (i = 0; i < 4096; i++) {
578 if((i % 0x10) == 0)
579 printf("\n%04x:", i);
580 printf(" %02x", *(pciexbar+devbase+i));
581 }
582 printf("\n");
583 }
584 }
585 }
586
Stefan Reinauer1162f252008-12-04 15:18:20 +0000587 unmap_physical((void *)pciexbar, (max_busses * 1024 * 1024));
Stefan Reinauer23190272008-08-20 13:41:24 +0000588
589 return 0;
590}