aboutsummaryrefslogtreecommitdiffstats
path: root/arch/i386/pci
diff options
context:
space:
mode:
authorOGAWA Hirofumi <hirofumi@mail.parknet.co.jp>2007-02-13 07:26:20 -0500
committerAndi Kleen <andi@basil.nowhere.org>2007-02-13 07:26:20 -0500
commit44de0203fab205417b24322272c53ee0883c36e7 (patch)
treec5dbfa7e373ab07f517fc91fd97ff45045ccbb1b /arch/i386/pci
parentfaed197b7b44a6c4e6b81dd2db649fd452b0a7ef (diff)
[PATCH] mmconfig: Reject a broken MCFG tables on Asus etc
This rejects broken MCFG tables on Asus. When the table looks bogus just disable mmconfig Arjan and Andi suggested this. Signed-off-by: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp> Signed-off-by: Andi Kleen <ak@suse.de>
Diffstat (limited to 'arch/i386/pci')
-rw-r--r--arch/i386/pci/mmconfig-shared.c24
-rw-r--r--arch/i386/pci/mmconfig.c9
2 files changed, 23 insertions, 10 deletions
diff --git a/arch/i386/pci/mmconfig-shared.c b/arch/i386/pci/mmconfig-shared.c
index 4757554b08cf..77de6de94f1f 100644
--- a/arch/i386/pci/mmconfig-shared.c
+++ b/arch/i386/pci/mmconfig-shared.c
@@ -197,6 +197,26 @@ static __init void pci_mmcfg_insert_resources(void)
197 } 197 }
198} 198}
199 199
200static void __init pci_mmcfg_reject_broken(void)
201{
202 typeof(pci_mmcfg_config[0]) *cfg = &pci_mmcfg_config[0];
203
204 /*
205 * Handle more broken MCFG tables on Asus etc.
206 * They only contain a single entry for bus 0-0.
207 */
208 if (pci_mmcfg_config_num == 1 &&
209 cfg->pci_segment == 0 &&
210 (cfg->start_bus_number | cfg->end_bus_number) == 0) {
211 kfree(pci_mmcfg_config);
212 pci_mmcfg_config = NULL;
213 pci_mmcfg_config_num = 0;
214
215 printk(KERN_ERR "PCI: start and end of bus number is 0. "
216 "Rejected as broken MCFG.");
217 }
218}
219
200void __init pci_mmcfg_init(int type) 220void __init pci_mmcfg_init(int type)
201{ 221{
202 int known_bridge = 0; 222 int known_bridge = 0;
@@ -207,8 +227,10 @@ void __init pci_mmcfg_init(int type)
207 if (type == 1 && pci_mmcfg_check_hostbridge()) 227 if (type == 1 && pci_mmcfg_check_hostbridge())
208 known_bridge = 1; 228 known_bridge = 1;
209 229
210 if (!known_bridge) 230 if (!known_bridge) {
211 acpi_table_parse(ACPI_SIG_MCFG, acpi_parse_mcfg); 231 acpi_table_parse(ACPI_SIG_MCFG, acpi_parse_mcfg);
232 pci_mmcfg_reject_broken();
233 }
212 234
213 if ((pci_mmcfg_config_num == 0) || 235 if ((pci_mmcfg_config_num == 0) ||
214 (pci_mmcfg_config == NULL) || 236 (pci_mmcfg_config == NULL) ||
diff --git a/arch/i386/pci/mmconfig.c b/arch/i386/pci/mmconfig.c
index 97dcaaa0de0f..3325b79e651c 100644
--- a/arch/i386/pci/mmconfig.c
+++ b/arch/i386/pci/mmconfig.c
@@ -47,15 +47,6 @@ static u32 get_base_addr(unsigned int seg, int bus, unsigned devfn)
47 return cfg->address; 47 return cfg->address;
48 } 48 }
49 49
50 /* Handle more broken MCFG tables on Asus etc.
51 They only contain a single entry for bus 0-0. Assume
52 this applies to all busses. */
53 cfg = &pci_mmcfg_config[0];
54 if (pci_mmcfg_config_num == 1 &&
55 cfg->pci_segment == 0 &&
56 (cfg->start_bus_number | cfg->end_bus_number) == 0)
57 return cfg->address;
58
59 /* Fall back to type 0 */ 50 /* Fall back to type 0 */
60 return 0; 51 return 0;
61} 52}