diff options
author | Jan Beulich <JBeulich@suse.com> | 2013-10-21 04:35:20 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@kernel.org> | 2013-10-26 07:34:39 -0400 |
commit | 09dc68d958c67c76cf672ec78b7391af453010f8 (patch) | |
tree | 945770f406cb85bc080b37463a75ab30e6f38c88 /arch/x86/kernel/cpu | |
parent | 88829dfe4b6ea0c1c24d415668b3bfa78e5a5b16 (diff) |
x86/cpu: Track legacy CPU model data only on 32-bit kernels
struct cpu_dev's c_models is only ever set inside CONFIG_X86_32
conditionals (or code that's being built for 32-bit only), so
there's no use of reserving the (empty) space for the model
names in a 64-bit kernel.
Similarly, c_size_cache is only used in the #else of a
CONFIG_X86_64 conditional, so reserving space for (and in one
case even initializing) that field is pointless for 64-bit
kernels too.
While moving both fields to the end of the structure, I also
noticed that:
- the c_models array size was one too small, potentially causing
table_lookup_model() to return garbage on Intel CPUs (intel.c's
instance was lacking the sentinel with family being zero), so the
patch bumps that by one,
- c_models' vendor sub-field was unused (and anyway redundant
with the base structure's c_x86_vendor field), so the patch deletes it.
Also rename the legacy fields so that their legacy nature stands out
and comment their declarations.
Signed-off-by: Jan Beulich <jbeulich@suse.com>
Link: http://lkml.kernel.org/r/5265036802000078000FC4DB@nat28.tlf.novell.com
Signed-off-by: Ingo Molnar <mingo@kernel.org>
Diffstat (limited to 'arch/x86/kernel/cpu')
-rw-r--r-- | arch/x86/kernel/cpu/amd.c | 6 | ||||
-rw-r--r-- | arch/x86/kernel/cpu/centaur.c | 8 | ||||
-rw-r--r-- | arch/x86/kernel/cpu/common.c | 12 | ||||
-rw-r--r-- | arch/x86/kernel/cpu/cpu.h | 20 | ||||
-rw-r--r-- | arch/x86/kernel/cpu/intel.c | 12 | ||||
-rw-r--r-- | arch/x86/kernel/cpu/umc.c | 4 |
6 files changed, 34 insertions, 28 deletions
diff --git a/arch/x86/kernel/cpu/amd.c b/arch/x86/kernel/cpu/amd.c index 903a264af981..3daece79a142 100644 --- a/arch/x86/kernel/cpu/amd.c +++ b/arch/x86/kernel/cpu/amd.c | |||
@@ -823,8 +823,8 @@ static const struct cpu_dev amd_cpu_dev = { | |||
823 | .c_vendor = "AMD", | 823 | .c_vendor = "AMD", |
824 | .c_ident = { "AuthenticAMD" }, | 824 | .c_ident = { "AuthenticAMD" }, |
825 | #ifdef CONFIG_X86_32 | 825 | #ifdef CONFIG_X86_32 |
826 | .c_models = { | 826 | .legacy_models = { |
827 | { .vendor = X86_VENDOR_AMD, .family = 4, .model_names = | 827 | { .family = 4, .model_names = |
828 | { | 828 | { |
829 | [3] = "486 DX/2", | 829 | [3] = "486 DX/2", |
830 | [7] = "486 DX/2-WB", | 830 | [7] = "486 DX/2-WB", |
@@ -835,7 +835,7 @@ static const struct cpu_dev amd_cpu_dev = { | |||
835 | } | 835 | } |
836 | }, | 836 | }, |
837 | }, | 837 | }, |
838 | .c_size_cache = amd_size_cache, | 838 | .legacy_cache_size = amd_size_cache, |
839 | #endif | 839 | #endif |
840 | .c_early_init = early_init_amd, | 840 | .c_early_init = early_init_amd, |
841 | .c_detect_tlb = cpu_detect_tlb_amd, | 841 | .c_detect_tlb = cpu_detect_tlb_amd, |
diff --git a/arch/x86/kernel/cpu/centaur.c b/arch/x86/kernel/cpu/centaur.c index fbf6c3bc2400..8d5652dc99dd 100644 --- a/arch/x86/kernel/cpu/centaur.c +++ b/arch/x86/kernel/cpu/centaur.c | |||
@@ -468,10 +468,10 @@ static void init_centaur(struct cpuinfo_x86 *c) | |||
468 | #endif | 468 | #endif |
469 | } | 469 | } |
470 | 470 | ||
471 | #ifdef CONFIG_X86_32 | ||
471 | static unsigned int | 472 | static unsigned int |
472 | centaur_size_cache(struct cpuinfo_x86 *c, unsigned int size) | 473 | centaur_size_cache(struct cpuinfo_x86 *c, unsigned int size) |
473 | { | 474 | { |
474 | #ifdef CONFIG_X86_32 | ||
475 | /* VIA C3 CPUs (670-68F) need further shifting. */ | 475 | /* VIA C3 CPUs (670-68F) need further shifting. */ |
476 | if ((c->x86 == 6) && ((c->x86_model == 7) || (c->x86_model == 8))) | 476 | if ((c->x86 == 6) && ((c->x86_model == 7) || (c->x86_model == 8))) |
477 | size >>= 8; | 477 | size >>= 8; |
@@ -484,16 +484,18 @@ centaur_size_cache(struct cpuinfo_x86 *c, unsigned int size) | |||
484 | if ((c->x86 == 6) && (c->x86_model == 9) && | 484 | if ((c->x86 == 6) && (c->x86_model == 9) && |
485 | (c->x86_mask == 1) && (size == 65)) | 485 | (c->x86_mask == 1) && (size == 65)) |
486 | size -= 1; | 486 | size -= 1; |
487 | #endif | ||
488 | return size; | 487 | return size; |
489 | } | 488 | } |
489 | #endif | ||
490 | 490 | ||
491 | static const struct cpu_dev centaur_cpu_dev = { | 491 | static const struct cpu_dev centaur_cpu_dev = { |
492 | .c_vendor = "Centaur", | 492 | .c_vendor = "Centaur", |
493 | .c_ident = { "CentaurHauls" }, | 493 | .c_ident = { "CentaurHauls" }, |
494 | .c_early_init = early_init_centaur, | 494 | .c_early_init = early_init_centaur, |
495 | .c_init = init_centaur, | 495 | .c_init = init_centaur, |
496 | .c_size_cache = centaur_size_cache, | 496 | #ifdef CONFIG_X86_32 |
497 | .legacy_cache_size = centaur_size_cache, | ||
498 | #endif | ||
497 | .c_x86_vendor = X86_VENDOR_CENTAUR, | 499 | .c_x86_vendor = X86_VENDOR_CENTAUR, |
498 | }; | 500 | }; |
499 | 501 | ||
diff --git a/arch/x86/kernel/cpu/common.c b/arch/x86/kernel/cpu/common.c index 2793d1f095a2..9ada0b37ae07 100644 --- a/arch/x86/kernel/cpu/common.c +++ b/arch/x86/kernel/cpu/common.c | |||
@@ -346,7 +346,8 @@ static void filter_cpuid_features(struct cpuinfo_x86 *c, bool warn) | |||
346 | /* Look up CPU names by table lookup. */ | 346 | /* Look up CPU names by table lookup. */ |
347 | static const char *table_lookup_model(struct cpuinfo_x86 *c) | 347 | static const char *table_lookup_model(struct cpuinfo_x86 *c) |
348 | { | 348 | { |
349 | const struct cpu_model_info *info; | 349 | #ifdef CONFIG_X86_32 |
350 | const struct legacy_cpu_model_info *info; | ||
350 | 351 | ||
351 | if (c->x86_model >= 16) | 352 | if (c->x86_model >= 16) |
352 | return NULL; /* Range check */ | 353 | return NULL; /* Range check */ |
@@ -354,13 +355,14 @@ static const char *table_lookup_model(struct cpuinfo_x86 *c) | |||
354 | if (!this_cpu) | 355 | if (!this_cpu) |
355 | return NULL; | 356 | return NULL; |
356 | 357 | ||
357 | info = this_cpu->c_models; | 358 | info = this_cpu->legacy_models; |
358 | 359 | ||
359 | while (info && info->family) { | 360 | while (info->family) { |
360 | if (info->family == c->x86) | 361 | if (info->family == c->x86) |
361 | return info->model_names[c->x86_model]; | 362 | return info->model_names[c->x86_model]; |
362 | info++; | 363 | info++; |
363 | } | 364 | } |
365 | #endif | ||
364 | return NULL; /* Not found */ | 366 | return NULL; /* Not found */ |
365 | } | 367 | } |
366 | 368 | ||
@@ -450,8 +452,8 @@ void cpu_detect_cache_sizes(struct cpuinfo_x86 *c) | |||
450 | c->x86_tlbsize += ((ebx >> 16) & 0xfff) + (ebx & 0xfff); | 452 | c->x86_tlbsize += ((ebx >> 16) & 0xfff) + (ebx & 0xfff); |
451 | #else | 453 | #else |
452 | /* do processor-specific cache resizing */ | 454 | /* do processor-specific cache resizing */ |
453 | if (this_cpu->c_size_cache) | 455 | if (this_cpu->legacy_cache_size) |
454 | l2size = this_cpu->c_size_cache(c, l2size); | 456 | l2size = this_cpu->legacy_cache_size(c, l2size); |
455 | 457 | ||
456 | /* Allow user to override all this if necessary. */ | 458 | /* Allow user to override all this if necessary. */ |
457 | if (cachesize_override != -1) | 459 | if (cachesize_override != -1) |
diff --git a/arch/x86/kernel/cpu/cpu.h b/arch/x86/kernel/cpu/cpu.h index 4041c24ae7db..c37dc37e8317 100644 --- a/arch/x86/kernel/cpu/cpu.h +++ b/arch/x86/kernel/cpu/cpu.h | |||
@@ -1,12 +1,6 @@ | |||
1 | #ifndef ARCH_X86_CPU_H | 1 | #ifndef ARCH_X86_CPU_H |
2 | #define ARCH_X86_CPU_H | 2 | #define ARCH_X86_CPU_H |
3 | 3 | ||
4 | struct cpu_model_info { | ||
5 | int vendor; | ||
6 | int family; | ||
7 | const char *model_names[16]; | ||
8 | }; | ||
9 | |||
10 | /* attempt to consolidate cpu attributes */ | 4 | /* attempt to consolidate cpu attributes */ |
11 | struct cpu_dev { | 5 | struct cpu_dev { |
12 | const char *c_vendor; | 6 | const char *c_vendor; |
@@ -14,15 +8,23 @@ struct cpu_dev { | |||
14 | /* some have two possibilities for cpuid string */ | 8 | /* some have two possibilities for cpuid string */ |
15 | const char *c_ident[2]; | 9 | const char *c_ident[2]; |
16 | 10 | ||
17 | struct cpu_model_info c_models[4]; | ||
18 | |||
19 | void (*c_early_init)(struct cpuinfo_x86 *); | 11 | void (*c_early_init)(struct cpuinfo_x86 *); |
20 | void (*c_bsp_init)(struct cpuinfo_x86 *); | 12 | void (*c_bsp_init)(struct cpuinfo_x86 *); |
21 | void (*c_init)(struct cpuinfo_x86 *); | 13 | void (*c_init)(struct cpuinfo_x86 *); |
22 | void (*c_identify)(struct cpuinfo_x86 *); | 14 | void (*c_identify)(struct cpuinfo_x86 *); |
23 | void (*c_detect_tlb)(struct cpuinfo_x86 *); | 15 | void (*c_detect_tlb)(struct cpuinfo_x86 *); |
24 | unsigned int (*c_size_cache)(struct cpuinfo_x86 *, unsigned int); | ||
25 | int c_x86_vendor; | 16 | int c_x86_vendor; |
17 | #ifdef CONFIG_X86_32 | ||
18 | /* Optional vendor specific routine to obtain the cache size. */ | ||
19 | unsigned int (*legacy_cache_size)(struct cpuinfo_x86 *, | ||
20 | unsigned int); | ||
21 | |||
22 | /* Family/stepping-based lookup table for model names. */ | ||
23 | struct legacy_cpu_model_info { | ||
24 | int family; | ||
25 | const char *model_names[16]; | ||
26 | } legacy_models[5]; | ||
27 | #endif | ||
26 | }; | 28 | }; |
27 | 29 | ||
28 | struct _tlb_table { | 30 | struct _tlb_table { |
diff --git a/arch/x86/kernel/cpu/intel.c b/arch/x86/kernel/cpu/intel.c index ec7299566f79..dc1ec0dff939 100644 --- a/arch/x86/kernel/cpu/intel.c +++ b/arch/x86/kernel/cpu/intel.c | |||
@@ -665,8 +665,8 @@ static const struct cpu_dev intel_cpu_dev = { | |||
665 | .c_vendor = "Intel", | 665 | .c_vendor = "Intel", |
666 | .c_ident = { "GenuineIntel" }, | 666 | .c_ident = { "GenuineIntel" }, |
667 | #ifdef CONFIG_X86_32 | 667 | #ifdef CONFIG_X86_32 |
668 | .c_models = { | 668 | .legacy_models = { |
669 | { .vendor = X86_VENDOR_INTEL, .family = 4, .model_names = | 669 | { .family = 4, .model_names = |
670 | { | 670 | { |
671 | [0] = "486 DX-25/33", | 671 | [0] = "486 DX-25/33", |
672 | [1] = "486 DX-50", | 672 | [1] = "486 DX-50", |
@@ -679,7 +679,7 @@ static const struct cpu_dev intel_cpu_dev = { | |||
679 | [9] = "486 DX/4-WB" | 679 | [9] = "486 DX/4-WB" |
680 | } | 680 | } |
681 | }, | 681 | }, |
682 | { .vendor = X86_VENDOR_INTEL, .family = 5, .model_names = | 682 | { .family = 5, .model_names = |
683 | { | 683 | { |
684 | [0] = "Pentium 60/66 A-step", | 684 | [0] = "Pentium 60/66 A-step", |
685 | [1] = "Pentium 60/66", | 685 | [1] = "Pentium 60/66", |
@@ -690,7 +690,7 @@ static const struct cpu_dev intel_cpu_dev = { | |||
690 | [8] = "Mobile Pentium MMX" | 690 | [8] = "Mobile Pentium MMX" |
691 | } | 691 | } |
692 | }, | 692 | }, |
693 | { .vendor = X86_VENDOR_INTEL, .family = 6, .model_names = | 693 | { .family = 6, .model_names = |
694 | { | 694 | { |
695 | [0] = "Pentium Pro A-step", | 695 | [0] = "Pentium Pro A-step", |
696 | [1] = "Pentium Pro", | 696 | [1] = "Pentium Pro", |
@@ -704,7 +704,7 @@ static const struct cpu_dev intel_cpu_dev = { | |||
704 | [11] = "Pentium III (Tualatin)", | 704 | [11] = "Pentium III (Tualatin)", |
705 | } | 705 | } |
706 | }, | 706 | }, |
707 | { .vendor = X86_VENDOR_INTEL, .family = 15, .model_names = | 707 | { .family = 15, .model_names = |
708 | { | 708 | { |
709 | [0] = "Pentium 4 (Unknown)", | 709 | [0] = "Pentium 4 (Unknown)", |
710 | [1] = "Pentium 4 (Willamette)", | 710 | [1] = "Pentium 4 (Willamette)", |
@@ -714,7 +714,7 @@ static const struct cpu_dev intel_cpu_dev = { | |||
714 | } | 714 | } |
715 | }, | 715 | }, |
716 | }, | 716 | }, |
717 | .c_size_cache = intel_size_cache, | 717 | .legacy_cache_size = intel_size_cache, |
718 | #endif | 718 | #endif |
719 | .c_detect_tlb = intel_detect_tlb, | 719 | .c_detect_tlb = intel_detect_tlb, |
720 | .c_early_init = early_init_intel, | 720 | .c_early_init = early_init_intel, |
diff --git a/arch/x86/kernel/cpu/umc.c b/arch/x86/kernel/cpu/umc.c index 202759a14121..75c5ad5d35cc 100644 --- a/arch/x86/kernel/cpu/umc.c +++ b/arch/x86/kernel/cpu/umc.c | |||
@@ -11,8 +11,8 @@ | |||
11 | static const struct cpu_dev umc_cpu_dev = { | 11 | static const struct cpu_dev umc_cpu_dev = { |
12 | .c_vendor = "UMC", | 12 | .c_vendor = "UMC", |
13 | .c_ident = { "UMC UMC UMC" }, | 13 | .c_ident = { "UMC UMC UMC" }, |
14 | .c_models = { | 14 | .legacy_models = { |
15 | { .vendor = X86_VENDOR_UMC, .family = 4, .model_names = | 15 | { .family = 4, .model_names = |
16 | { | 16 | { |
17 | [1] = "U5D", | 17 | [1] = "U5D", |
18 | [2] = "U5S", | 18 | [2] = "U5S", |