diff options
Diffstat (limited to 'arch/x86/kernel/amd_nb.c')
-rw-r--r-- | arch/x86/kernel/amd_nb.c | 41 |
1 files changed, 41 insertions, 0 deletions
diff --git a/arch/x86/kernel/amd_nb.c b/arch/x86/kernel/amd_nb.c index 458da8509b75..6db28f17ff28 100644 --- a/arch/x86/kernel/amd_nb.c +++ b/arch/x86/kernel/amd_nb.c | |||
@@ -27,6 +27,8 @@ static const struct pci_device_id amd_root_ids[] = { | |||
27 | {} | 27 | {} |
28 | }; | 28 | }; |
29 | 29 | ||
30 | #define PCI_DEVICE_ID_AMD_CNB17H_F4 0x1704 | ||
31 | |||
30 | const struct pci_device_id amd_nb_misc_ids[] = { | 32 | const struct pci_device_id amd_nb_misc_ids[] = { |
31 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_K8_NB_MISC) }, | 33 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_K8_NB_MISC) }, |
32 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_10H_NB_MISC) }, | 34 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_10H_NB_MISC) }, |
@@ -37,6 +39,7 @@ const struct pci_device_id amd_nb_misc_ids[] = { | |||
37 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_NB_F3) }, | 39 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_NB_F3) }, |
38 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_M30H_NB_F3) }, | 40 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_M30H_NB_F3) }, |
39 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_17H_DF_F3) }, | 41 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_17H_DF_F3) }, |
42 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_CNB17H_F3) }, | ||
40 | {} | 43 | {} |
41 | }; | 44 | }; |
42 | EXPORT_SYMBOL_GPL(amd_nb_misc_ids); | 45 | EXPORT_SYMBOL_GPL(amd_nb_misc_ids); |
@@ -48,6 +51,7 @@ static const struct pci_device_id amd_nb_link_ids[] = { | |||
48 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_NB_F4) }, | 51 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_NB_F4) }, |
49 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_M30H_NB_F4) }, | 52 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_16H_M30H_NB_F4) }, |
50 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_17H_DF_F4) }, | 53 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_17H_DF_F4) }, |
54 | { PCI_DEVICE(PCI_VENDOR_ID_AMD, PCI_DEVICE_ID_AMD_CNB17H_F4) }, | ||
51 | {} | 55 | {} |
52 | }; | 56 | }; |
53 | 57 | ||
@@ -402,11 +406,48 @@ void amd_flush_garts(void) | |||
402 | } | 406 | } |
403 | EXPORT_SYMBOL_GPL(amd_flush_garts); | 407 | EXPORT_SYMBOL_GPL(amd_flush_garts); |
404 | 408 | ||
409 | static void __fix_erratum_688(void *info) | ||
410 | { | ||
411 | #define MSR_AMD64_IC_CFG 0xC0011021 | ||
412 | |||
413 | msr_set_bit(MSR_AMD64_IC_CFG, 3); | ||
414 | msr_set_bit(MSR_AMD64_IC_CFG, 14); | ||
415 | } | ||
416 | |||
417 | /* Apply erratum 688 fix so machines without a BIOS fix work. */ | ||
418 | static __init void fix_erratum_688(void) | ||
419 | { | ||
420 | struct pci_dev *F4; | ||
421 | u32 val; | ||
422 | |||
423 | if (boot_cpu_data.x86 != 0x14) | ||
424 | return; | ||
425 | |||
426 | if (!amd_northbridges.num) | ||
427 | return; | ||
428 | |||
429 | F4 = node_to_amd_nb(0)->link; | ||
430 | if (!F4) | ||
431 | return; | ||
432 | |||
433 | if (pci_read_config_dword(F4, 0x164, &val)) | ||
434 | return; | ||
435 | |||
436 | if (val & BIT(2)) | ||
437 | return; | ||
438 | |||
439 | on_each_cpu(__fix_erratum_688, NULL, 0); | ||
440 | |||
441 | pr_info("x86/cpu/AMD: CPU erratum 688 worked around\n"); | ||
442 | } | ||
443 | |||
405 | static __init int init_amd_nbs(void) | 444 | static __init int init_amd_nbs(void) |
406 | { | 445 | { |
407 | amd_cache_northbridges(); | 446 | amd_cache_northbridges(); |
408 | amd_cache_gart(); | 447 | amd_cache_gart(); |
409 | 448 | ||
449 | fix_erratum_688(); | ||
450 | |||
410 | return 0; | 451 | return 0; |
411 | } | 452 | } |
412 | 453 | ||