| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  | /* -*- linux-c -*- ------------------------------------------------------- *
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Copyright (C) 1991, 1992 Linus Torvalds | 
					
						
							| 
									
										
										
										
											2008-02-04 16:48:00 +01:00
										 |  |  |  *   Copyright 2007-2008 rPath, Inc. - All Rights Reserved | 
					
						
							| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  *   This file is part of the Linux kernel, and is made available under | 
					
						
							|  |  |  |  *   the terms of the GNU General Public License version 2. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-02-04 16:48:00 +01:00
										 |  |  |  * arch/x86/boot/cpu.c | 
					
						
							| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Check for obligatory CPU features and abort if the features are not | 
					
						
							|  |  |  |  * present. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "boot.h"
 | 
					
						
							| 
									
										
										
										
											2013-10-30 08:09:45 -07:00
										 |  |  | #ifdef CONFIG_X86_FEATURE_NAMES
 | 
					
						
							| 
									
										
										
										
											2008-02-04 16:48:00 +01:00
										 |  |  | #include "cpustr.h"
 | 
					
						
							| 
									
										
										
										
											2013-10-30 08:09:45 -07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-02-04 16:48:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  | static char *cpu_name(int level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static char buf[6]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (level == 64) { | 
					
						
							|  |  |  | 		return "x86-64"; | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2008-05-28 12:57:13 -04:00
										 |  |  | 		if (level == 15) | 
					
						
							|  |  |  | 			level = 6; | 
					
						
							| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  | 		sprintf(buf, "i%d86", level); | 
					
						
							|  |  |  | 		return buf; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-30 08:09:45 -07:00
										 |  |  | 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
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  | 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 (err_flags) { | 
					
						
							|  |  |  | 		puts("This kernel requires the following features " | 
					
						
							|  |  |  | 		     "not present on the CPU:\n"); | 
					
						
							| 
									
										
										
										
											2013-10-30 08:09:45 -07:00
										 |  |  | 		show_cap_strs(err_flags); | 
					
						
							| 
									
										
										
										
											2007-07-11 12:18:47 -07:00
										 |  |  | 		putchar('\n'); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |