123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 |
- /* -*- 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;
- }
- }
|