diff options
Diffstat (limited to 'arch/x86')
-rw-r--r-- | arch/x86/kernel/aperture_64.c | 47 |
1 files changed, 35 insertions, 12 deletions
diff --git a/arch/x86/kernel/aperture_64.c b/arch/x86/kernel/aperture_64.c index 2e93b3132dff..8c325b7f2d9b 100644 --- a/arch/x86/kernel/aperture_64.c +++ b/arch/x86/kernel/aperture_64.c | |||
@@ -83,7 +83,7 @@ static u32 __init allocate_aperture(void) | |||
83 | return (u32)__pa(p); | 83 | return (u32)__pa(p); |
84 | } | 84 | } |
85 | 85 | ||
86 | static int __init aperture_valid(u64 aper_base, u32 aper_size) | 86 | static int __init aperture_valid(u64 aper_base, u32 aper_size, u32 min_size) |
87 | { | 87 | { |
88 | if (!aper_base) | 88 | if (!aper_base) |
89 | return 0; | 89 | return 0; |
@@ -96,8 +96,9 @@ static int __init aperture_valid(u64 aper_base, u32 aper_size) | |||
96 | printk(KERN_ERR "Aperture pointing to e820 RAM. Ignoring.\n"); | 96 | printk(KERN_ERR "Aperture pointing to e820 RAM. Ignoring.\n"); |
97 | return 0; | 97 | return 0; |
98 | } | 98 | } |
99 | if (aper_size < 64*1024*1024) { | 99 | if (aper_size < min_size) { |
100 | printk(KERN_ERR "Aperture too small (%d MB)\n", aper_size>>20); | 100 | printk(KERN_ERR "Aperture too small (%d MB) than (%d MB)\n", |
101 | aper_size>>20, min_size>>20); | ||
101 | return 0; | 102 | return 0; |
102 | } | 103 | } |
103 | 104 | ||
@@ -167,7 +168,9 @@ static __u32 __init read_agp(int num, int slot, int func, int cap, u32 *order) | |||
167 | * On some sick chips, APSIZE is 0. It means it wants 4G | 168 | * On some sick chips, APSIZE is 0. It means it wants 4G |
168 | * so let double check that order, and lets trust AMD NB settings: | 169 | * so let double check that order, and lets trust AMD NB settings: |
169 | */ | 170 | */ |
170 | if (aper + (32UL<<(20 + *order)) > 0x100000000UL) { | 171 | printk(KERN_INFO "Aperture from AGP @ %Lx old size %u MB\n", |
172 | aper, 32 << old_order); | ||
173 | if (aper + (32ULL<<(20 + *order)) > 0x100000000ULL) { | ||
171 | printk(KERN_INFO "Aperture size %u MB (APSIZE %x) is not right, using settings from NB\n", | 174 | printk(KERN_INFO "Aperture size %u MB (APSIZE %x) is not right, using settings from NB\n", |
172 | 32 << *order, apsizereg); | 175 | 32 << *order, apsizereg); |
173 | *order = old_order; | 176 | *order = old_order; |
@@ -176,7 +179,7 @@ static __u32 __init read_agp(int num, int slot, int func, int cap, u32 *order) | |||
176 | printk(KERN_INFO "Aperture from AGP @ %Lx size %u MB (APSIZE %x)\n", | 179 | printk(KERN_INFO "Aperture from AGP @ %Lx size %u MB (APSIZE %x)\n", |
177 | aper, 32 << *order, apsizereg); | 180 | aper, 32 << *order, apsizereg); |
178 | 181 | ||
179 | if (!aperture_valid(aper, (32*1024*1024) << *order)) | 182 | if (!aperture_valid(aper, (32*1024*1024) << *order, 32<<20)) |
180 | return 0; | 183 | return 0; |
181 | return (u32)aper; | 184 | return (u32)aper; |
182 | } | 185 | } |
@@ -302,8 +305,8 @@ void __init early_gart_iommu_check(void) | |||
302 | fix = 1; | 305 | fix = 1; |
303 | 306 | ||
304 | if (gart_fix_e820 && !fix && aper_enabled) { | 307 | if (gart_fix_e820 && !fix && aper_enabled) { |
305 | if (e820_any_mapped(aper_base, aper_base + aper_size, | 308 | if (!e820_all_mapped(aper_base, aper_base + aper_size, |
306 | E820_RAM)) { | 309 | E820_RESERVED)) { |
307 | /* reserved it, so we can resuse it in second kernel */ | 310 | /* reserved it, so we can resuse it in second kernel */ |
308 | printk(KERN_INFO "update e820 for GART\n"); | 311 | printk(KERN_INFO "update e820 for GART\n"); |
309 | add_memory_region(aper_base, aper_size, E820_RESERVED); | 312 | add_memory_region(aper_base, aper_size, E820_RESERVED); |
@@ -324,8 +327,11 @@ void __init early_gart_iommu_check(void) | |||
324 | 327 | ||
325 | } | 328 | } |
326 | 329 | ||
330 | static int __initdata printed_gart_size_msg; | ||
331 | |||
327 | void __init gart_iommu_hole_init(void) | 332 | void __init gart_iommu_hole_init(void) |
328 | { | 333 | { |
334 | u32 agp_aper_base = 0, agp_aper_order = 0; | ||
329 | u32 aper_size, aper_alloc = 0, aper_order = 0, last_aper_order = 0; | 335 | u32 aper_size, aper_alloc = 0, aper_order = 0, last_aper_order = 0; |
330 | u64 aper_base, last_aper_base = 0; | 336 | u64 aper_base, last_aper_base = 0; |
331 | int fix, num, valid_agp = 0; | 337 | int fix, num, valid_agp = 0; |
@@ -337,6 +343,9 @@ void __init gart_iommu_hole_init(void) | |||
337 | 343 | ||
338 | printk(KERN_INFO "Checking aperture...\n"); | 344 | printk(KERN_INFO "Checking aperture...\n"); |
339 | 345 | ||
346 | if (!fallback_aper_force) | ||
347 | agp_aper_base = search_agp_bridge(&agp_aper_order, &valid_agp); | ||
348 | |||
340 | fix = 0; | 349 | fix = 0; |
341 | node = 0; | 350 | node = 0; |
342 | for (num = 24; num < 32; num++) { | 351 | for (num = 24; num < 32; num++) { |
@@ -355,9 +364,21 @@ void __init gart_iommu_hole_init(void) | |||
355 | node, aper_base, aper_size >> 20); | 364 | node, aper_base, aper_size >> 20); |
356 | node++; | 365 | node++; |
357 | 366 | ||
358 | if (!aperture_valid(aper_base, aper_size)) { | 367 | if (!aperture_valid(aper_base, aper_size, 64<<20)) { |
359 | fix = 1; | 368 | if (valid_agp && agp_aper_base && |
360 | break; | 369 | agp_aper_base == aper_base && |
370 | agp_aper_order == aper_order) { | ||
371 | /* the same between two setting from NB and agp */ | ||
372 | if (!no_iommu && end_pfn > MAX_DMA32_PFN && !printed_gart_size_msg) { | ||
373 | printk(KERN_ERR "you are using iommu with agp, but GART size is less than 64M\n"); | ||
374 | printk(KERN_ERR "please increase GART size in your BIOS setup\n"); | ||
375 | printk(KERN_ERR "if BIOS doesn't have that option, contact your HW vendor!\n"); | ||
376 | printed_gart_size_msg = 1; | ||
377 | } | ||
378 | } else { | ||
379 | fix = 1; | ||
380 | break; | ||
381 | } | ||
361 | } | 382 | } |
362 | 383 | ||
363 | if ((last_aper_order && aper_order != last_aper_order) || | 384 | if ((last_aper_order && aper_order != last_aper_order) || |
@@ -378,8 +399,10 @@ void __init gart_iommu_hole_init(void) | |||
378 | return; | 399 | return; |
379 | } | 400 | } |
380 | 401 | ||
381 | if (!fallback_aper_force) | 402 | if (!fallback_aper_force) { |
382 | aper_alloc = search_agp_bridge(&aper_order, &valid_agp); | 403 | aper_alloc = agp_aper_base; |
404 | aper_order = agp_aper_order; | ||
405 | } | ||
383 | 406 | ||
384 | if (aper_alloc) { | 407 | if (aper_alloc) { |
385 | /* Got the aperture from the AGP bridge */ | 408 | /* Got the aperture from the AGP bridge */ |