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