]>
Commit | Line | Data |
---|---|---|
31b54f40 PA |
1 | /* -*- linux-c -*- ------------------------------------------------------- * |
2 | * | |
3 | * Copyright (C) 1991, 1992 Linus Torvalds | |
f0be6c6a | 4 | * Copyright 2007-2008 rPath, Inc. - All Rights Reserved |
31b54f40 PA |
5 | * |
6 | * This file is part of the Linux kernel, and is made available under | |
7 | * the terms of the GNU General Public License version 2. | |
8 | * | |
9 | * ----------------------------------------------------------------------- */ | |
10 | ||
11 | /* | |
f0be6c6a | 12 | * arch/x86/boot/cpu.c |
31b54f40 PA |
13 | * |
14 | * Check for obligatory CPU features and abort if the features are not | |
15 | * present. | |
16 | */ | |
17 | ||
18 | #include "boot.h" | |
9def39be | 19 | #ifdef CONFIG_X86_FEATURE_NAMES |
f0be6c6a | 20 | #include "cpustr.h" |
9def39be | 21 | #endif |
f0be6c6a | 22 | |
31b54f40 PA |
23 | static char *cpu_name(int level) |
24 | { | |
25 | static char buf[6]; | |
26 | ||
27 | if (level == 64) { | |
28 | return "x86-64"; | |
29 | } else { | |
c7d624d1 DJ |
30 | if (level == 15) |
31 | level = 6; | |
31b54f40 PA |
32 | sprintf(buf, "i%d86", level); |
33 | return buf; | |
34 | } | |
35 | } | |
36 | ||
9def39be JT |
37 | static void show_cap_strs(u32 *err_flags) |
38 | { | |
39 | int i, j; | |
40 | #ifdef CONFIG_X86_FEATURE_NAMES | |
41 | const unsigned char *msg_strs = (const unsigned char *)x86_cap_strs; | |
42 | for (i = 0; i < NCAPINTS; i++) { | |
43 | u32 e = err_flags[i]; | |
44 | for (j = 0; j < 32; j++) { | |
45 | if (msg_strs[0] < i || | |
46 | (msg_strs[0] == i && msg_strs[1] < j)) { | |
47 | /* Skip to the next string */ | |
48 | msg_strs += 2; | |
49 | while (*msg_strs++) | |
50 | ; | |
51 | } | |
52 | if (e & 1) { | |
53 | if (msg_strs[0] == i && | |
54 | msg_strs[1] == j && | |
55 | msg_strs[2]) | |
56 | printf("%s ", msg_strs+2); | |
57 | else | |
58 | printf("%d:%d ", i, j); | |
59 | } | |
60 | e >>= 1; | |
61 | } | |
62 | } | |
63 | #else | |
64 | for (i = 0; i < NCAPINTS; i++) { | |
65 | u32 e = err_flags[i]; | |
66 | for (j = 0; j < 32; j++) { | |
67 | if (e & 1) | |
68 | printf("%d:%d ", i, j); | |
69 | e >>= 1; | |
70 | } | |
71 | } | |
72 | #endif | |
73 | } | |
74 | ||
31b54f40 PA |
75 | int validate_cpu(void) |
76 | { | |
77 | u32 *err_flags; | |
78 | int cpu_level, req_level; | |
79 | ||
80 | check_cpu(&cpu_level, &req_level, &err_flags); | |
81 | ||
82 | if (cpu_level < req_level) { | |
83 | printf("This kernel requires an %s CPU, ", | |
84 | cpu_name(req_level)); | |
85 | printf("but only detected an %s CPU.\n", | |
86 | cpu_name(cpu_level)); | |
87 | return -1; | |
88 | } | |
89 | ||
ed68d7e9 AL |
90 | if (CONFIG_X86_MINIMUM_CPU_FAMILY <= 4 && !IS_ENABLED(CONFIG_M486) && |
91 | !has_eflag(X86_EFLAGS_ID)) { | |
92 | printf("This kernel requires a CPU with the CPUID instruction. Build with CONFIG_M486=y to run on this CPU.\n"); | |
93 | return -1; | |
94 | } | |
95 | ||
31b54f40 | 96 | if (err_flags) { |
31b54f40 PA |
97 | puts("This kernel requires the following features " |
98 | "not present on the CPU:\n"); | |
9def39be | 99 | show_cap_strs(err_flags); |
31b54f40 PA |
100 | putchar('\n'); |
101 | return -1; | |
e4a84be6 DH |
102 | } else if (check_knl_erratum()) { |
103 | return -1; | |
31b54f40 PA |
104 | } else { |
105 | return 0; | |
106 | } | |
107 | } |