blob: ebe087737895fa32db512b7b4397faf8eec73c96 [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.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
Patrick Georgib890a122015-03-26 15:17:45 +010018 * Foundation, Inc.
Stefan Reinauer23190272008-08-20 13:41:24 +000019 */
20
21#include <stdio.h>
22#include <stdlib.h>
Stefan Reinauera7b296d2011-11-14 12:40:34 -080023#include <inttypes.h>
Stefan Reinauer23190272008-08-20 13:41:24 +000024#include "inteltool.h"
25
Stefan Taunerdbc6fcd2013-06-20 18:05:06 +020026/* 320766 */
27static const io_register_t nehalem_dmi_registers[] = {
28 { 0x00, 4, "DMIVCH" }, // DMI Virtual Channel Capability Header
29 { 0x04, 4, "DMIVCCAP1" }, // DMI Port VC Capability Register 1
30 { 0x08, 4, "DMIVCCAP2" }, // DMI Port VC Capability Register 2
31 { 0x0C, 4, "DMIVCCTL" }, // DMI Port VC Control
32 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
33 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
34/* { 0x18, 2, "RSVD" }, // Reserved */
35 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
36 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
37 { 0x20, 4, "DMIVC1RCTL" }, // DMI VC1 Resource Control
38/* { 0x24, 2, "RSVD" }, // Reserved */
39 { 0x26, 2, "DMIVC1RSTS" }, // DMI VC1 Resource Status
40/* ... - Reserved */
41 { 0x84, 4, "DMILCAP" }, // DMI Link Capabilities
42 { 0x88, 2, "DMILCTL" }, // DMI Link Control
43 { 0x8A, 2, "DMILSTS" }, // DMI Link Status
44/* ... - Reserved */
45};
46
47/* 322812 */
48static const io_register_t westmere_dmi_registers[] = {
49 { 0x00, 4, "DMIVCECH" }, // DMI Virtual Channel Enhanced Capability
50 { 0x04, 4, "DMIPVCCAP1" }, // DMI Port VC Capability Register 1
51 { 0x08, 4, "DMIPVCCAP2" }, // DMI Port VC Capability Register 2
52 { 0x0C, 2, "DMIPVCCTL" }, // DMI Port VC Control
53/* { 0x0E, 2, "RSVD" }, // Reserved */
54 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
55 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
56/* { 0x18, 2, "RSVD" }, // Reserved */
57 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
58 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
59 { 0x20, 4, "DMIVC1RCTL1" }, // DMI VC1 Resource Control
60/* { 0x24, 2, "RSVD" }, // Reserved */
61 { 0x26, 2, "DMIC1RSTS" }, // DMI VC1 Resource Status
62/* ... - Reserved */
63 { 0x84, 4, "DMILCAP" }, // DMI Link Capabilities
64 { 0x88, 2, "DMILCTL" }, // DMI Link Control
65 { 0x8A, 2, "DMILSTS" }, // DMI Link Status
66/* ... - Reserved */
67};
68
Anton Kochkovc7fc4422012-07-21 06:36:47 +040069static const io_register_t sandybridge_dmi_registers[] = {
70 { 0x00, 4, "DMI VCECH" }, // DMI Virtual Channel Enhanced Capability
71 { 0x04, 4, "DMI PVCCAP1" }, // DMI Port VC Capability Register 1
72 { 0x08, 4, "DMI PVVAP2" }, // DMI Port VC Capability Register 2
73 { 0x0C, 2, "DMI PVCCTL" }, // DMI Port VC Control
74/* { 0x0E, 2, "RSVD" }, // Reserved */
75 { 0x10, 4, "DMI VC0RCAP" }, // DMI VC0 Resource Capability
76 { 0x14, 4, "DMI VC0RCTL" }, // DMI VC0 Resource Control
77/* { 0x18, 2, "RSVD" }, // Reserved */
78 { 0x1A, 2, "DMI VC0RSTS" }, // DMI VC0 Resource Status
79 { 0x1C, 4, "DMI VC1RCAP" }, // DMI VC1 Resource Capability
80 { 0x20, 4, "DMI VC1RCTL" }, // DMI VC1 Resource Control
81/* { 0x24, 2, "RSVD" }, // Reserved */
82 { 0x26, 2, "DMI VC1RSTS" }, // DMI VC1 Resource Status
83 { 0x28, 4, "DMI VCPRCAP" }, // DMI VCp Resource Capability
84 { 0x2C, 4, "DMI VCPRCTL" }, // DMI VCp Resource Control
85/* { 0x30, 2, "RSVD" }, // Reserved */
86 { 0x32, 2, "DMI VCPRSTS" }, // DMI VCp Resource Status
87 { 0x34, 4, "DMI VCMRCAP" }, // DMI VCm Resource Capability
88 { 0x38, 4, "DMI VCMRCTL" }, // DMI VCm Resource Control
89/* { 0x3C, 2, "RSVD" }, // Reserved */
90 { 0x3E, 2, "DMI VCMRSTS" }, // DMI VCm Resource Status
91/* { 0x40, 4, "RSVD" }, // Reserved */
92 { 0x44, 4, "DMI ESC" }, // DMI Element Self Description
93/* { 0x48, 8, "RSVD" }, // Reserved */
94 { 0x50, 4, "DMI LE1D" }, // DMI Link Entry 1 Description
95/* { 0x54, 4, "RSVD" }, // Reserved */
96 { 0x58, 4, "DMI LE1A" }, // DMI Link Entry 1 Address
97 { 0x5C, 4, "DMI LUE1A" }, // DMI Link Upper Entry 1 Address
98 { 0x60, 4, "DMI LE2D" }, // DMI Link Entry 2 Description
99/* { 0x64, 4, "RSVD" }, // Reserved */
100 { 0x68, 4, "DMI LE2A" }, // DMI Link Entry 2 Address
101/* { 0x6C, 4, "RSVD" }, // Reserved
102 { 0x70, 8, "RSVD" }, // Reserved
103 { 0x78, 8, "RSVD" }, // Reserved
104 { 0x80, 4, "RSVD" }, // Reserved */
105 { 0x84, 4, "LCAP" }, // Link Capabilities
106 { 0x88, 2, "LCTL" }, // Link Control
107 { 0x8A, 2, "LSTS" }, // Link Status
108/* { 0x8C, 4, "RSVD" }, // Reserved
109 { 0x90, 4, "RSVD" }, // Reserved
110 { 0x94, 4, "RSVD" }, // Reserved */
111 { 0x98, 2, "LCTL2" }, // Link Control 2
112 { 0x9A, 2, "LSTS2" }, // Link Status 2
113/* ... - Reserved */
114 { 0xBC0, 4, "AFE_BMUF0" }, // AFE BMU Configuration Function 0
115 { 0xBC4, 4, "RSVD" }, // Reserved
116 { 0xBC8, 4, "RSVD" }, // Reserved
117 { 0xBCC, 4, "AFE_BMUT0" }, // AFE BMU Configuration Test 0
118/* ... - Reserved */
119};
120
Stefan Reinauer23190272008-08-20 13:41:24 +0000121/*
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100122 * All Haswell DMI Registers per
123 *
124 * Mobile 4th Generation Intel Core TM Processor Family, Mobile Intel Pentium Processor Family,
125 * and Mobile Intel Celeron Processor Family
126 * Datasheet Volume 2
127 * 329002-002
128 */
129static const io_register_t haswell_ult_dmi_registers[] = {
130 { 0x00, 4, "DMIVCECH" }, // DMI Virtual Channel Enhanced Capability
131 { 0x04, 4, "DMIPVCCAP1" }, // DMI Port VC Capability Register 1
132 { 0x08, 4, "DMIPVCCAP2" }, // DMI Port VC Capability Register 2
133 { 0x0C, 2, "DMI PVCCTL" }, // DMI Port VC Control
134/* { 0x0E, 2, "RSVD" }, // Reserved */
135 { 0x10, 4, "DMIVC0RCAP" }, // DMI VC0 Resource Capability
136 { 0x14, 4, "DMIVC0RCTL" }, // DMI VC0 Resource Control
137/* { 0x18, 2, "RSVD" }, // Reserved */
138 { 0x1A, 2, "DMIVC0RSTS" }, // DMI VC0 Resource Status
139 { 0x1C, 4, "DMIVC1RCAP" }, // DMI VC1 Resource Capability
140 { 0x20, 4, "DMIVC1RCTL" }, // DMI VC1 Resource Control
141/* { 0x24, 2, "RSVD" }, // Reserved */
142 { 0x26, 2, "DMIVC1RSTS" }, // DMI VC1 Resource Status
143 { 0x28, 4, "DMIVCPRCAP" }, // DMI VCp Resource Capability
144 { 0x2C, 4, "DMIVCPRCTL" }, // DMI VCp Resource Control
145/* { 0x30, 2, "RSVD" }, // Reserved */
146 { 0x32, 2, "DMIVCPRSTS" }, // DMI VCp Resource Status
147 { 0x34, 4, "DMIVCMRCAP" }, // DMI VCm Resource Capability
148 { 0x38, 4, "DMIVCMRCTL" }, // DMI VCm Resource Control
149/* { 0x3C, 2, "RSVD" }, // Reserved */
150 { 0x3E, 2, "DMIVCMRSTS" }, // DMI VCm Resource Status
151 { 0x40, 4, "DMIRCLDECH" }, // DMI Root Complex Link Declaration */
152 { 0x44, 4, "DMIESD" }, // DMI Element Self Description
153/* { 0x48, 4, "RSVD" }, // Reserved */
154/* { 0x4C, 4, "RSVD" }, // Reserved */
155 { 0x50, 4, "DMILE1D" }, // DMI Link Entry 1 Description
156/* { 0x54, 4, "RSVD" }, // Reserved */
157 { 0x58, 4, "DMILE1A" }, // DMI Link Entry 1 Address
158 { 0x5C, 4, "DMILUE1A" }, // DMI Link Upper Entry 1 Address
159 { 0x60, 4, "DMILE2D" }, // DMI Link Entry 2 Description
160/* { 0x64, 4, "RSVD" }, // Reserved */
161 { 0x68, 4, "DMILE2A" }, // DMI Link Entry 2 Address
162/* { 0x6C, 4, "RSVD" }, // Reserved */
163/* { 0x70, 4, "RSVD" }, // Reserved */
164/* { 0x74, 4, "RSVD" }, // Reserved */
165/* { 0x78, 4, "RSVD" }, // Reserved */
166/* { 0x7C, 4, "RSVD" }, // Reserved */
167/* { 0x80, 4, "RSVD" }, // Reserved */
168/* { 0x84, 4, "RSVD" }, // Reserved */
169 { 0x88, 2, "LCTL" }, // Link Control
170 /* ... - Reserved */
171 { 0x1C4, 4, "DMIUESTS" }, // DMI Uncorrectable Error Status
172 { 0x1C8, 4, "DMIUEMSK" }, // DMI Uncorrectable Error Mask
173 { 0x1D0, 4, "DMICESTS" }, // DMI Correctable Error Status
174 { 0x1D4, 4, "DMICEMSK" }, // DMI Correctable Error Mask
175/* ... - Reserved */
176};
177
178/*
Stefan Reinauer23190272008-08-20 13:41:24 +0000179 * Egress Port Root Complex MMIO configuration space
180 */
181int print_epbar(struct pci_dev *nb)
182{
183 int i, size = (4 * 1024);
184 volatile uint8_t *epbar;
Stefan Reinauer1162f252008-12-04 15:18:20 +0000185 uint64_t epbar_phys;
Stefan Reinauer23190272008-08-20 13:41:24 +0000186
187 printf("\n============= EPBAR =============\n\n");
188
189 switch (nb->device_id) {
Pat Erleyca3548e2010-04-21 06:23:19 +0000190 case PCI_DEVICE_ID_INTEL_82915:
Stefan Reinauer23190272008-08-20 13:41:24 +0000191 case PCI_DEVICE_ID_INTEL_82945GM:
Björn Busse2d33dc42010-08-01 15:33:30 +0000192 case PCI_DEVICE_ID_INTEL_82945GSE:
Stefan Reinauer3d9a12f2008-11-02 11:11:40 +0000193 case PCI_DEVICE_ID_INTEL_82945P:
Stefan Tauner1a00cf02012-10-13 06:23:52 +0200194 case PCI_DEVICE_ID_INTEL_82946:
Stefan Reinauer1162f252008-12-04 15:18:20 +0000195 case PCI_DEVICE_ID_INTEL_82975X:
Stefan Reinauer23190272008-08-20 13:41:24 +0000196 epbar_phys = pci_read_long(nb, 0x40) & 0xfffffffe;
197 break;
Stefan Tauner04c06002012-10-13 02:19:30 +0200198 case PCI_DEVICE_ID_INTEL_82965PM:
199 case PCI_DEVICE_ID_INTEL_82Q965:
200 case PCI_DEVICE_ID_INTEL_82Q35:
201 case PCI_DEVICE_ID_INTEL_82G33:
202 case PCI_DEVICE_ID_INTEL_82Q33:
203 case PCI_DEVICE_ID_INTEL_82X38:
Ruud Schrampbb41f502011-04-04 07:53:19 +0200204 case PCI_DEVICE_ID_INTEL_32X0:
Damien Zammit9c986642015-08-17 21:04:41 +1000205 case PCI_DEVICE_ID_INTEL_82XX4X:
206 case PCI_DEVICE_ID_INTEL_82Q45:
207 case PCI_DEVICE_ID_INTEL_82G45:
208 case PCI_DEVICE_ID_INTEL_82G41:
209 case PCI_DEVICE_ID_INTEL_82B43:
210 case PCI_DEVICE_ID_INTEL_82B43_2:
Corey Osgood23d98c72010-07-29 19:25:31 +0000211 case PCI_DEVICE_ID_INTEL_ATOM_DXXX:
212 case PCI_DEVICE_ID_INTEL_ATOM_NXXX:
Felix Heldfac95e32014-11-09 00:11:28 +0100213 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_D:
214 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_M:
215 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_E3:
216 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_D:
217 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_M:
218 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_E3:
219 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_015c:
220 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_D:
221 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_M:
222 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_E3:
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100223 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_U:
Matt DeVillier5b667df2015-05-14 21:58:33 -0500224 case PCI_DEVICE_ID_INTEL_CORE_5TH_GEN_U:
Stefan Tauner04c06002012-10-13 02:19:30 +0200225 epbar_phys = pci_read_long(nb, 0x40) & 0xfffffffe;
226 epbar_phys |= ((uint64_t)pci_read_long(nb, 0x44)) << 32;
227 break;
Stefan Reinauerb2aedb12009-08-29 15:45:43 +0000228 case PCI_DEVICE_ID_INTEL_82810:
Stefan Tauner04c06002012-10-13 02:19:30 +0200229 case PCI_DEVICE_ID_INTEL_82810_DC:
230 case PCI_DEVICE_ID_INTEL_82810E_DC:
Stefan Reinauer04844812010-02-22 11:26:06 +0000231 case PCI_DEVICE_ID_INTEL_82830M:
Idwer Vollering312fc962010-12-17 22:34:58 +0000232 case PCI_DEVICE_ID_INTEL_82865:
233 printf("This northbridge does not have EPBAR.\n");
Stefan Reinauer23190272008-08-20 13:41:24 +0000234 return 1;
235 default:
236 printf("Error: Dumping EPBAR on this northbridge is not (yet) supported.\n");
237 return 1;
238 }
239
Stefan Reinauer1162f252008-12-04 15:18:20 +0000240 epbar = map_physical(epbar_phys, size);
Stefan Reinauer14e22772010-04-27 06:56:47 +0000241
Stefan Reinauer1162f252008-12-04 15:18:20 +0000242 if (epbar == NULL) {
Stefan Reinauer23190272008-08-20 13:41:24 +0000243 perror("Error mapping EPBAR");
244 exit(1);
245 }
246
Stefan Reinauera7b296d2011-11-14 12:40:34 -0800247 printf("EPBAR = 0x%08" PRIx64 " (MEM)\n\n", epbar_phys);
Stefan Reinauer23190272008-08-20 13:41:24 +0000248 for (i = 0; i < size; i += 4) {
249 if (*(uint32_t *)(epbar + i))
250 printf("0x%04x: 0x%08x\n", i, *(uint32_t *)(epbar+i));
251 }
252
Stefan Reinauer1162f252008-12-04 15:18:20 +0000253 unmap_physical((void *)epbar, size);
Stefan Reinauer23190272008-08-20 13:41:24 +0000254 return 0;
255}
256
257/*
258 * MCH-ICH Serial Interconnect Ingress Root Complex MMIO configuration space
259 */
260int print_dmibar(struct pci_dev *nb)
261{
262 int i, size = (4 * 1024);
263 volatile uint8_t *dmibar;
Stefan Reinauer1162f252008-12-04 15:18:20 +0000264 uint64_t dmibar_phys;
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400265 const io_register_t *dmi_registers = NULL;
Stefan Reinauer23190272008-08-20 13:41:24 +0000266
267 printf("\n============= DMIBAR ============\n\n");
268
269 switch (nb->device_id) {
Pat Erleyca3548e2010-04-21 06:23:19 +0000270 case PCI_DEVICE_ID_INTEL_82915:
Stefan Reinauer23190272008-08-20 13:41:24 +0000271 case PCI_DEVICE_ID_INTEL_82945GM:
Björn Busse2d33dc42010-08-01 15:33:30 +0000272 case PCI_DEVICE_ID_INTEL_82945GSE:
Stefan Reinauer3d9a12f2008-11-02 11:11:40 +0000273 case PCI_DEVICE_ID_INTEL_82945P:
Stefan Reinauer1162f252008-12-04 15:18:20 +0000274 case PCI_DEVICE_ID_INTEL_82975X:
Stefan Reinauer23190272008-08-20 13:41:24 +0000275 dmibar_phys = pci_read_long(nb, 0x4c) & 0xfffffffe;
276 break;
Stefan Tauner1a00cf02012-10-13 06:23:52 +0200277 case PCI_DEVICE_ID_INTEL_82946:
Stefan Tauner04c06002012-10-13 02:19:30 +0200278 case PCI_DEVICE_ID_INTEL_82965PM:
279 case PCI_DEVICE_ID_INTEL_82Q965:
Warren Turkal53291952010-09-03 09:32:17 +0000280 case PCI_DEVICE_ID_INTEL_82Q35:
281 case PCI_DEVICE_ID_INTEL_82G33:
282 case PCI_DEVICE_ID_INTEL_82Q33:
Stefan Tauner04c06002012-10-13 02:19:30 +0200283 case PCI_DEVICE_ID_INTEL_82X38:
Ruud Schrampbb41f502011-04-04 07:53:19 +0200284 case PCI_DEVICE_ID_INTEL_32X0:
Damien Zammit9c986642015-08-17 21:04:41 +1000285 case PCI_DEVICE_ID_INTEL_82XX4X:
286 case PCI_DEVICE_ID_INTEL_82Q45:
287 case PCI_DEVICE_ID_INTEL_82G45:
288 case PCI_DEVICE_ID_INTEL_82G41:
289 case PCI_DEVICE_ID_INTEL_82B43:
290 case PCI_DEVICE_ID_INTEL_82B43_2:
Corey Osgood23d98c72010-07-29 19:25:31 +0000291 case PCI_DEVICE_ID_INTEL_ATOM_DXXX:
292 case PCI_DEVICE_ID_INTEL_ATOM_NXXX:
Warren Turkal53291952010-09-03 09:32:17 +0000293 dmibar_phys = pci_read_long(nb, 0x68) & 0xfffffffe;
294 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
295 break;
Stefan Reinauerb2aedb12009-08-29 15:45:43 +0000296 case PCI_DEVICE_ID_INTEL_82810:
Stefan Tauner04c06002012-10-13 02:19:30 +0200297 case PCI_DEVICE_ID_INTEL_82810_DC:
298 case PCI_DEVICE_ID_INTEL_82810E_DC:
Idwer Vollering312fc962010-12-17 22:34:58 +0000299 case PCI_DEVICE_ID_INTEL_82865:
300 printf("This northbridge does not have DMIBAR.\n");
Stefan Reinauer23190272008-08-20 13:41:24 +0000301 return 1;
Stefan Tauner04c06002012-10-13 02:19:30 +0200302 case PCI_DEVICE_ID_INTEL_82X58:
Warren Turkal3235eea2010-09-03 09:31:13 +0000303 dmibar_phys = pci_read_long(nb, 0x50) & 0xfffff000;
304 break;
Stefan Taunerdbc6fcd2013-06-20 18:05:06 +0200305 case PCI_DEVICE_ID_INTEL_CORE_0TH_GEN:
306 /* DMIBAR is called DMIRCBAR in Nehalem */
307 dmibar_phys = pci_read_long(nb, 0x50) & 0xfffff000; /* 31:12 */
308 dmi_registers = nehalem_dmi_registers;
309 size = ARRAY_SIZE(nehalem_dmi_registers);
310 break;
Stefan Tauner04c06002012-10-13 02:19:30 +0200311 case PCI_DEVICE_ID_INTEL_CORE_1ST_GEN:
312 dmibar_phys = pci_read_long(nb, 0x68);
313 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
314 dmibar_phys &= 0x0000000ffffff000UL; /* 35:12 */
Stefan Taunerdbc6fcd2013-06-20 18:05:06 +0200315 dmi_registers = westmere_dmi_registers;
316 size = ARRAY_SIZE(westmere_dmi_registers);
Stefan Tauner04c06002012-10-13 02:19:30 +0200317 break;
Felix Held0cc8f292014-11-05 03:18:44 +0100318 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_D:
319 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_M:
Felix Heldfac95e32014-11-09 00:11:28 +0100320 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_E3:
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400321 dmi_registers = sandybridge_dmi_registers;
322 size = ARRAY_SIZE(sandybridge_dmi_registers);
Felix Heldfac95e32014-11-09 00:11:28 +0100323 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_D: /* pretty printing not implemented yet */
324 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_M:
325 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_E3:
326 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_015c:
327 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_D:
328 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_M:
329 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_E3:
Stefan Tauner04c06002012-10-13 02:19:30 +0200330 dmibar_phys = pci_read_long(nb, 0x68);
331 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
332 dmibar_phys &= 0x0000007ffffff000UL; /* 38:12 */
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400333 break;
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100334 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_U:
Matt DeVillier5b667df2015-05-14 21:58:33 -0500335 case PCI_DEVICE_ID_INTEL_CORE_5TH_GEN_U:
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100336 dmi_registers = haswell_ult_dmi_registers;
337 size = ARRAY_SIZE(haswell_ult_dmi_registers);
338 dmibar_phys = pci_read_long(nb, 0x68);
339 dmibar_phys |= ((uint64_t)pci_read_long(nb, 0x6c)) << 32;
340 dmibar_phys &= 0x0000007ffffff000UL; /* 38:12 */
341 break;
342
Stefan Reinauer23190272008-08-20 13:41:24 +0000343 default:
344 printf("Error: Dumping DMIBAR on this northbridge is not (yet) supported.\n");
345 return 1;
346 }
347
Stefan Reinauer1162f252008-12-04 15:18:20 +0000348 dmibar = map_physical(dmibar_phys, size);
Stefan Reinauer14e22772010-04-27 06:56:47 +0000349
Stefan Reinauer1162f252008-12-04 15:18:20 +0000350 if (dmibar == NULL) {
Stefan Reinauer23190272008-08-20 13:41:24 +0000351 perror("Error mapping DMIBAR");
352 exit(1);
353 }
354
Stefan Reinauera7b296d2011-11-14 12:40:34 -0800355 printf("DMIBAR = 0x%08" PRIx64 " (MEM)\n\n", dmibar_phys);
Anton Kochkovc7fc4422012-07-21 06:36:47 +0400356 if (dmi_registers != NULL) {
357 for (i = 0; i < size; i++) {
358 switch (dmi_registers[i].size) {
359 case 4:
360 printf("dmibase+0x%04x: 0x%08x (%s)\n",
361 dmi_registers[i].addr,
362 *(uint32_t *)(dmibar+dmi_registers[i].addr),
363 dmi_registers[i].name);
364 break;
365 case 2:
366 printf("dmibase+0x%04x: 0x%04x (%s)\n",
367 dmi_registers[i].addr,
368 *(uint16_t *)(dmibar+dmi_registers[i].addr),
369 dmi_registers[i].name);
370 break;
371 case 1:
372 printf("dmibase+0x%04x: 0x%02x (%s)\n",
373 dmi_registers[i].addr,
374 *(uint8_t *)(dmibar+dmi_registers[i].addr),
375 dmi_registers[i].name);
376 break;
377 }
378 }
379 } else {
380 for (i = 0; i < size; i += 4) {
381 if (*(uint32_t *)(dmibar + i))
382 printf("0x%04x: 0x%08x\n", i, *(uint32_t *)(dmibar+i));
383 }
Stefan Reinauer23190272008-08-20 13:41:24 +0000384 }
385
Stefan Reinauer1162f252008-12-04 15:18:20 +0000386 unmap_physical((void *)dmibar, size);
Stefan Reinauer23190272008-08-20 13:41:24 +0000387 return 0;
388}
389
390/*
391 * PCIe MMIO configuration space
392 */
393int print_pciexbar(struct pci_dev *nb)
394{
Stefan Reinauer1162f252008-12-04 15:18:20 +0000395 uint64_t pciexbar_reg;
396 uint64_t pciexbar_phys;
Stefan Reinauer23190272008-08-20 13:41:24 +0000397 volatile uint8_t *pciexbar;
398 int max_busses, devbase, i;
399 int bus, dev, fn;
400
401 printf("========= PCIEXBAR ========\n\n");
402
403 switch (nb->device_id) {
Pat Erleyca3548e2010-04-21 06:23:19 +0000404 case PCI_DEVICE_ID_INTEL_82915:
Stefan Reinauer23190272008-08-20 13:41:24 +0000405 case PCI_DEVICE_ID_INTEL_82945GM:
Björn Busse2d33dc42010-08-01 15:33:30 +0000406 case PCI_DEVICE_ID_INTEL_82945GSE:
Stefan Reinauer3d9a12f2008-11-02 11:11:40 +0000407 case PCI_DEVICE_ID_INTEL_82945P:
Stefan Reinauer1162f252008-12-04 15:18:20 +0000408 case PCI_DEVICE_ID_INTEL_82975X:
Stefan Reinauer23190272008-08-20 13:41:24 +0000409 pciexbar_reg = pci_read_long(nb, 0x48);
410 break;
Stefan Tauner1a00cf02012-10-13 06:23:52 +0200411 case PCI_DEVICE_ID_INTEL_82946:
Stefan Tauner04c06002012-10-13 02:19:30 +0200412 case PCI_DEVICE_ID_INTEL_82965PM:
413 case PCI_DEVICE_ID_INTEL_82Q965:
414 case PCI_DEVICE_ID_INTEL_82Q35:
415 case PCI_DEVICE_ID_INTEL_82G33:
416 case PCI_DEVICE_ID_INTEL_82Q33:
417 case PCI_DEVICE_ID_INTEL_82X38:
Ruud Schrampbb41f502011-04-04 07:53:19 +0200418 case PCI_DEVICE_ID_INTEL_32X0:
Damien Zammit9c986642015-08-17 21:04:41 +1000419 case PCI_DEVICE_ID_INTEL_82XX4X:
420 case PCI_DEVICE_ID_INTEL_82Q45:
421 case PCI_DEVICE_ID_INTEL_82G45:
422 case PCI_DEVICE_ID_INTEL_82G41:
423 case PCI_DEVICE_ID_INTEL_82B43:
424 case PCI_DEVICE_ID_INTEL_82B43_2:
Corey Osgood23d98c72010-07-29 19:25:31 +0000425 case PCI_DEVICE_ID_INTEL_ATOM_DXXX:
426 case PCI_DEVICE_ID_INTEL_ATOM_NXXX:
Felix Heldfac95e32014-11-09 00:11:28 +0100427 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_D:
428 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_M:
429 case PCI_DEVICE_ID_INTEL_CORE_2ND_GEN_E3:
430 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_D:
431 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_M:
432 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_E3:
433 case PCI_DEVICE_ID_INTEL_CORE_3RD_GEN_015c:
434 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_D:
435 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_M:
436 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_E3:
Dennis Wassenbergae6685f2014-10-30 10:30:40 +0100437 case PCI_DEVICE_ID_INTEL_CORE_4TH_GEN_U:
Matt DeVillier5b667df2015-05-14 21:58:33 -0500438 case PCI_DEVICE_ID_INTEL_CORE_5TH_GEN_U:
Stefan Tauner04c06002012-10-13 02:19:30 +0200439 pciexbar_reg = pci_read_long(nb, 0x60);
440 pciexbar_reg |= ((uint64_t)pci_read_long(nb, 0x64)) << 32;
441 break;
Stefan Reinauerb2aedb12009-08-29 15:45:43 +0000442 case PCI_DEVICE_ID_INTEL_82810:
Stefan Tauner04c06002012-10-13 02:19:30 +0200443 case PCI_DEVICE_ID_INTEL_82810_DC:
444 case PCI_DEVICE_ID_INTEL_82810E_DC:
Idwer Vollering312fc962010-12-17 22:34:58 +0000445 case PCI_DEVICE_ID_INTEL_82865:
446 printf("Error: This northbridge does not have PCIEXBAR.\n");
Stefan Reinauer23190272008-08-20 13:41:24 +0000447 return 1;
448 default:
449 printf("Error: Dumping PCIEXBAR on this northbridge is not (yet) supported.\n");
450 return 1;
451 }
452
453 if (!(pciexbar_reg & (1 << 0))) {
454 printf("PCIEXBAR register is disabled.\n");
455 return 0;
456 }
457
458 switch ((pciexbar_reg >> 1) & 3) {
459 case 0: // 256MB
Paul Menzel17c05f22013-04-03 10:00:33 +0200460 pciexbar_phys = pciexbar_reg & (0xffULL << 28);
Stefan Reinauer23190272008-08-20 13:41:24 +0000461 max_busses = 256;
462 break;
463 case 1: // 128M
Paul Menzel17c05f22013-04-03 10:00:33 +0200464 pciexbar_phys = pciexbar_reg & (0x1ffULL << 27);
Stefan Reinauer23190272008-08-20 13:41:24 +0000465 max_busses = 128;
466 break;
467 case 2: // 64M
Paul Menzel17c05f22013-04-03 10:00:33 +0200468 pciexbar_phys = pciexbar_reg & (0x3ffULL << 26);
Stefan Reinauer23190272008-08-20 13:41:24 +0000469 max_busses = 64;
470 break;
471 default: // RSVD
472 printf("Undefined address base. Bailing out.\n");
473 return 1;
Stefan Reinauer14e22772010-04-27 06:56:47 +0000474 }
Stefan Reinauer23190272008-08-20 13:41:24 +0000475
Stefan Reinauera7b296d2011-11-14 12:40:34 -0800476 printf("PCIEXBAR: 0x%08" PRIx64 "\n", pciexbar_phys);
Stefan Reinauer23190272008-08-20 13:41:24 +0000477
Stefan Reinauer1162f252008-12-04 15:18:20 +0000478 pciexbar = map_physical(pciexbar_phys, (max_busses * 1024 * 1024));
Stefan Reinauer14e22772010-04-27 06:56:47 +0000479
Stefan Reinauer1162f252008-12-04 15:18:20 +0000480 if (pciexbar == NULL) {
Stefan Reinauer23190272008-08-20 13:41:24 +0000481 perror("Error mapping PCIEXBAR");
482 exit(1);
483 }
Stefan Reinauer14e22772010-04-27 06:56:47 +0000484
Stefan Reinauer23190272008-08-20 13:41:24 +0000485 for (bus = 0; bus < max_busses; bus++) {
486 for (dev = 0; dev < 32; dev++) {
487 for (fn = 0; fn < 8; fn++) {
488 devbase = (bus * 1024 * 1024) + (dev * 32 * 1024) + (fn * 4 * 1024);
489
490 if (*(uint16_t *)(pciexbar + devbase) == 0xffff)
491 continue;
Stefan Reinauer14e22772010-04-27 06:56:47 +0000492
Stefan Reinauer23190272008-08-20 13:41:24 +0000493 /* This is a heuristics. Anyone got a better check? */
494 if( (*(uint32_t *)(pciexbar + devbase + 256) == 0xffffffff) &&
495 (*(uint32_t *)(pciexbar + devbase + 512) == 0xffffffff) ) {
496#if DEBUG
497 printf("Skipped non-PCIe device %02x:%02x.%01x\n", bus, dev, fn);
498#endif
499 continue;
500 }
501
502 printf("\nPCIe %02x:%02x.%01x extended config space:", bus, dev, fn);
503 for (i = 0; i < 4096; i++) {
504 if((i % 0x10) == 0)
505 printf("\n%04x:", i);
506 printf(" %02x", *(pciexbar+devbase+i));
507 }
508 printf("\n");
509 }
510 }
511 }
512
Stefan Reinauer1162f252008-12-04 15:18:20 +0000513 unmap_physical((void *)pciexbar, (max_busses * 1024 * 1024));
Stefan Reinauer23190272008-08-20 13:41:24 +0000514
515 return 0;
516}