Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 1 | #include <console/console.h> |
| 2 | #include <cpu/cpu.h> |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 3 | #include <arch/io.h> |
| 4 | #include <string.h> |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 5 | #include <cpu/x86/mtrr.h> |
| 6 | #include <cpu/x86/msr.h> |
| 7 | #include <cpu/x86/lapic.h> |
| 8 | #include <arch/cpu.h> |
| 9 | #include <device/path.h> |
| 10 | #include <device/device.h> |
| 11 | #include <smp/spinlock.h> |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 12 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 13 | /* Standard macro to see if a specific flag is changeable */ |
| 14 | static inline int flag_is_changeable_p(uint32_t flag) |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 15 | { |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 16 | uint32_t f1, f2; |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 17 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 18 | asm( |
| 19 | "pushfl\n\t" |
| 20 | "pushfl\n\t" |
| 21 | "popl %0\n\t" |
| 22 | "movl %0,%1\n\t" |
| 23 | "xorl %2,%0\n\t" |
| 24 | "pushl %0\n\t" |
| 25 | "popfl\n\t" |
| 26 | "pushfl\n\t" |
| 27 | "popl %0\n\t" |
| 28 | "popfl\n\t" |
| 29 | : "=&r" (f1), "=&r" (f2) |
| 30 | : "ir" (flag)); |
| 31 | return ((f1^f2) & flag) != 0; |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 32 | } |
| 33 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 34 | /* Probe for the CPUID instruction */ |
Rudolf Marek | 06253cd | 2012-02-25 23:51:12 +0100 | [diff] [blame] | 35 | int cpu_have_cpuid(void) |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 36 | { |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 37 | return flag_is_changeable_p(X86_EFLAGS_ID); |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 38 | } |
| 39 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 40 | /* |
| 41 | * Cyrix CPUs without cpuid or with cpuid not yet enabled can be detected |
| 42 | * by the fact that they preserve the flags across the division of 5/2. |
| 43 | * PII and PPro exhibit this behavior too, but they have cpuid available. |
| 44 | */ |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 45 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 46 | /* |
| 47 | * Perform the Cyrix 5/2 test. A Cyrix won't change |
| 48 | * the flags, while other 486 chips will. |
| 49 | */ |
| 50 | static inline int test_cyrix_52div(void) |
| 51 | { |
| 52 | unsigned int test; |
| 53 | |
| 54 | __asm__ __volatile__( |
| 55 | "sahf\n\t" /* clear flags (%eax = 0x0005) */ |
| 56 | "div %b2\n\t" /* divide 5 by 2 */ |
| 57 | "lahf" /* store flags into %ah */ |
| 58 | : "=a" (test) |
| 59 | : "0" (5), "q" (2) |
| 60 | : "cc"); |
| 61 | |
| 62 | /* AH is 0x02 on Cyrix after the divide.. */ |
| 63 | return (unsigned char) (test >> 8) == 0x02; |
| 64 | } |
| 65 | |
| 66 | /* |
| 67 | * Detect a NexGen CPU running without BIOS hypercode new enough |
| 68 | * to have CPUID. (Thanks to Herbert Oppmann) |
| 69 | */ |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 70 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 71 | static int deep_magic_nexgen_probe(void) |
| 72 | { |
| 73 | int ret; |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 74 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 75 | __asm__ __volatile__ ( |
| 76 | " movw $0x5555, %%ax\n" |
| 77 | " xorw %%dx,%%dx\n" |
| 78 | " movw $2, %%cx\n" |
| 79 | " divw %%cx\n" |
| 80 | " movl $0, %%eax\n" |
| 81 | " jnz 1f\n" |
| 82 | " movl $1, %%eax\n" |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 83 | "1:\n" |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 84 | : "=a" (ret) : : "cx", "dx" ); |
| 85 | return ret; |
| 86 | } |
| 87 | |
| 88 | /* List of cpu vendor strings along with their normalized |
| 89 | * id values. |
| 90 | */ |
| 91 | static struct { |
| 92 | int vendor; |
| 93 | const char *name; |
| 94 | } x86_vendors[] = { |
| 95 | { X86_VENDOR_INTEL, "GenuineIntel", }, |
| 96 | { X86_VENDOR_CYRIX, "CyrixInstead", }, |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 97 | { X86_VENDOR_AMD, "AuthenticAMD", }, |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 98 | { X86_VENDOR_UMC, "UMC UMC UMC ", }, |
| 99 | { X86_VENDOR_NEXGEN, "NexGenDriven", }, |
| 100 | { X86_VENDOR_CENTAUR, "CentaurHauls", }, |
| 101 | { X86_VENDOR_RISE, "RiseRiseRise", }, |
| 102 | { X86_VENDOR_TRANSMETA, "GenuineTMx86", }, |
| 103 | { X86_VENDOR_TRANSMETA, "TransmetaCPU", }, |
| 104 | { X86_VENDOR_NSC, "Geode by NSC", }, |
| 105 | { X86_VENDOR_SIS, "SiS SiS SiS ", }, |
| 106 | }; |
| 107 | |
| 108 | static const char *x86_vendor_name[] = { |
| 109 | [X86_VENDOR_INTEL] = "Intel", |
| 110 | [X86_VENDOR_CYRIX] = "Cyrix", |
| 111 | [X86_VENDOR_AMD] = "AMD", |
| 112 | [X86_VENDOR_UMC] = "UMC", |
| 113 | [X86_VENDOR_NEXGEN] = "NexGen", |
| 114 | [X86_VENDOR_CENTAUR] = "Centaur", |
| 115 | [X86_VENDOR_RISE] = "Rise", |
| 116 | [X86_VENDOR_TRANSMETA] = "Transmeta", |
| 117 | [X86_VENDOR_NSC] = "NSC", |
| 118 | [X86_VENDOR_SIS] = "SiS", |
| 119 | }; |
| 120 | |
| 121 | static const char *cpu_vendor_name(int vendor) |
| 122 | { |
| 123 | const char *name; |
| 124 | name = "<invalid cpu vendor>"; |
Carl-Daniel Hailfinger | 2ee6779 | 2008-10-01 12:52:52 +0000 | [diff] [blame] | 125 | if ((vendor < (ARRAY_SIZE(x86_vendor_name))) && |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 126 | (x86_vendor_name[vendor] != 0)) |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 127 | { |
| 128 | name = x86_vendor_name[vendor]; |
| 129 | } |
| 130 | return name; |
| 131 | } |
| 132 | |
Sven Schnelle | adfbcb79 | 2012-01-10 12:01:43 +0100 | [diff] [blame] | 133 | static int cpu_cpuid_extended_level(void) |
| 134 | { |
| 135 | return cpuid_eax(0x80000000); |
| 136 | } |
| 137 | |
| 138 | #define CPUID_FEATURE_PAE (1 << 6) |
| 139 | #define CPUID_FEATURE_PSE36 (1 << 17) |
| 140 | |
| 141 | int cpu_phys_address_size(void) |
| 142 | { |
Rudolf Marek | 06253cd | 2012-02-25 23:51:12 +0100 | [diff] [blame] | 143 | if (!(cpu_have_cpuid())) |
Sven Schnelle | adfbcb79 | 2012-01-10 12:01:43 +0100 | [diff] [blame] | 144 | return 32; |
| 145 | |
Sven Schnelle | 7363ca3 | 2012-01-31 22:10:28 +0100 | [diff] [blame] | 146 | if (cpu_cpuid_extended_level() >= 0x80000008) |
Sven Schnelle | adfbcb79 | 2012-01-10 12:01:43 +0100 | [diff] [blame] | 147 | return cpuid_eax(0x80000008) & 0xff; |
| 148 | |
Kyösti Mälkki | 730c0ee | 2012-02-29 21:17:18 +0200 | [diff] [blame] | 149 | if (cpuid_edx(1) & (CPUID_FEATURE_PAE | CPUID_FEATURE_PSE36)) |
Sven Schnelle | adfbcb79 | 2012-01-10 12:01:43 +0100 | [diff] [blame] | 150 | return 36; |
| 151 | return 32; |
| 152 | } |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 153 | static void identify_cpu(struct device *cpu) |
| 154 | { |
| 155 | char vendor_name[16]; |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 156 | int i; |
| 157 | |
| 158 | vendor_name[0] = '\0'; /* Unset */ |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 159 | |
| 160 | /* Find the id and vendor_name */ |
Rudolf Marek | 06253cd | 2012-02-25 23:51:12 +0100 | [diff] [blame] | 161 | if (!cpu_have_cpuid()) { |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 162 | /* Its a 486 if we can modify the AC flag */ |
| 163 | if (flag_is_changeable_p(X86_EFLAGS_AC)) { |
| 164 | cpu->device = 0x00000400; /* 486 */ |
| 165 | } else { |
| 166 | cpu->device = 0x00000300; /* 386 */ |
| 167 | } |
| 168 | if ((cpu->device == 0x00000400) && test_cyrix_52div()) { |
| 169 | memcpy(vendor_name, "CyrixInstead", 13); |
| 170 | /* If we ever care we can enable cpuid here */ |
| 171 | } |
| 172 | /* Detect NexGen with old hypercode */ |
| 173 | else if (deep_magic_nexgen_probe()) { |
| 174 | memcpy(vendor_name, "NexGenDriven", 13); |
| 175 | } |
| 176 | } |
Rudolf Marek | 06253cd | 2012-02-25 23:51:12 +0100 | [diff] [blame] | 177 | if (cpu_have_cpuid()) { |
Stefan Reinauer | 4b556a1 | 2009-05-26 12:33:06 +0000 | [diff] [blame] | 178 | int cpuid_level; |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 179 | struct cpuid_result result; |
| 180 | result = cpuid(0x00000000); |
| 181 | cpuid_level = result.eax; |
| 182 | vendor_name[ 0] = (result.ebx >> 0) & 0xff; |
| 183 | vendor_name[ 1] = (result.ebx >> 8) & 0xff; |
| 184 | vendor_name[ 2] = (result.ebx >> 16) & 0xff; |
| 185 | vendor_name[ 3] = (result.ebx >> 24) & 0xff; |
| 186 | vendor_name[ 4] = (result.edx >> 0) & 0xff; |
| 187 | vendor_name[ 5] = (result.edx >> 8) & 0xff; |
| 188 | vendor_name[ 6] = (result.edx >> 16) & 0xff; |
| 189 | vendor_name[ 7] = (result.edx >> 24) & 0xff; |
| 190 | vendor_name[ 8] = (result.ecx >> 0) & 0xff; |
| 191 | vendor_name[ 9] = (result.ecx >> 8) & 0xff; |
| 192 | vendor_name[10] = (result.ecx >> 16) & 0xff; |
| 193 | vendor_name[11] = (result.ecx >> 24) & 0xff; |
| 194 | vendor_name[12] = '\0'; |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 195 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 196 | /* Intel-defined flags: level 0x00000001 */ |
| 197 | if (cpuid_level >= 0x00000001) { |
| 198 | cpu->device = cpuid_eax(0x00000001); |
| 199 | } |
| 200 | else { |
| 201 | /* Have CPUID level 0 only unheard of */ |
| 202 | cpu->device = 0x00000400; |
| 203 | } |
| 204 | } |
| 205 | cpu->vendor = X86_VENDOR_UNKNOWN; |
Carl-Daniel Hailfinger | 2ee6779 | 2008-10-01 12:52:52 +0000 | [diff] [blame] | 206 | for(i = 0; i < ARRAY_SIZE(x86_vendors); i++) { |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 207 | if (memcmp(vendor_name, x86_vendors[i].name, 12) == 0) { |
| 208 | cpu->vendor = x86_vendors[i].vendor; |
| 209 | break; |
| 210 | } |
| 211 | } |
| 212 | } |
| 213 | |
Stefan Reinauer | 6293d30 | 2012-04-03 16:07:56 -0700 | [diff] [blame] | 214 | struct cpu_driver *find_cpu_driver(struct device *cpu) |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 215 | { |
| 216 | struct cpu_driver *driver; |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 217 | for (driver = cpu_drivers; driver < ecpu_drivers; driver++) { |
| 218 | struct cpu_device_id *id; |
Stefan Reinauer | 6293d30 | 2012-04-03 16:07:56 -0700 | [diff] [blame] | 219 | for (id = driver->id_table; |
| 220 | id->vendor != X86_VENDOR_INVALID; id++) { |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 221 | if ((cpu->vendor == id->vendor) && |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 222 | (cpu->device == id->device)) |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 223 | { |
Stefan Reinauer | 6293d30 | 2012-04-03 16:07:56 -0700 | [diff] [blame] | 224 | return driver; |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 225 | } |
| 226 | } |
| 227 | } |
Stefan Reinauer | 6293d30 | 2012-04-03 16:07:56 -0700 | [diff] [blame] | 228 | return NULL; |
| 229 | } |
| 230 | |
| 231 | static void set_cpu_ops(struct device *cpu) |
| 232 | { |
| 233 | struct cpu_driver *driver = find_cpu_driver(cpu); |
| 234 | cpu->ops = driver ? driver->ops : NULL; |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 235 | } |
| 236 | |
Ronald G. Minnich | 8b93059 | 2012-06-05 14:41:27 -0700 | [diff] [blame] | 237 | void cpu_initialize(unsigned int index) |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 238 | { |
| 239 | /* Because we busy wait at the printk spinlock. |
| 240 | * It is important to keep the number of printed messages |
| 241 | * from secondary cpus to a minimum, when debugging is |
| 242 | * disabled. |
| 243 | */ |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 244 | struct device *cpu; |
Sven Schnelle | 51676b1 | 2012-07-29 19:18:03 +0200 | [diff] [blame] | 245 | struct cpu_info *info; |
Yinghai Lu | d4b278c | 2006-10-04 20:46:15 +0000 | [diff] [blame] | 246 | struct cpuinfo_x86 c; |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 247 | |
Sven Schnelle | 51676b1 | 2012-07-29 19:18:03 +0200 | [diff] [blame] | 248 | info = cpu_info(); |
Eric Biederman | 83b991a | 2003-10-11 06:20:25 +0000 | [diff] [blame] | 249 | |
Ronald G. Minnich | 8b93059 | 2012-06-05 14:41:27 -0700 | [diff] [blame] | 250 | printk(BIOS_INFO, "Initializing CPU #%d\n", index); |
Kyösti Mälkki | bc8c996 | 2012-07-10 10:19:40 +0300 | [diff] [blame] | 251 | |
Sven Schnelle | 51676b1 | 2012-07-29 19:18:03 +0200 | [diff] [blame] | 252 | cpu = info->cpu; |
| 253 | if (!cpu) { |
| 254 | die("CPU: missing cpu device structure"); |
| 255 | } |
Li-Ta Lo | 8cb91dc | 2004-03-26 18:34:48 +0000 | [diff] [blame] | 256 | |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 257 | /* Find what type of cpu we are dealing with */ |
| 258 | identify_cpu(cpu); |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 259 | printk(BIOS_DEBUG, "CPU: vendor %s device %x\n", |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 260 | cpu_vendor_name(cpu->vendor), cpu->device); |
Li-Ta Lo | fd3f2d7 | 2004-05-12 16:34:46 +0000 | [diff] [blame] | 261 | |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 262 | get_fms(&c, cpu->device); |
Yinghai Lu | d4b278c | 2006-10-04 20:46:15 +0000 | [diff] [blame] | 263 | |
Stefan Reinauer | 5f5436f | 2010-04-25 20:42:02 +0000 | [diff] [blame] | 264 | printk(BIOS_DEBUG, "CPU: family %02x, model %02x, stepping %02x\n", |
| 265 | c.x86, c.x86_model, c.x86_mask); |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 266 | |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 267 | /* Lookup the cpu's operations */ |
| 268 | set_cpu_ops(cpu); |
Yinghai Lu | d4b278c | 2006-10-04 20:46:15 +0000 | [diff] [blame] | 269 | |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 270 | if(!cpu->ops) { |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 271 | /* mask out the stepping and try again */ |
| 272 | cpu->device -= c.x86_mask; |
Steven J. Magnani | eccc357 | 2005-09-14 13:53:45 +0000 | [diff] [blame] | 273 | set_cpu_ops(cpu); |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 274 | cpu->device += c.x86_mask; |
| 275 | if(!cpu->ops) die("Unknown cpu"); |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 276 | printk(BIOS_DEBUG, "Using generic cpu ops (good)\n"); |
Ronald G. Minnich | 43225bc | 2005-11-22 00:07:02 +0000 | [diff] [blame] | 277 | } |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 278 | |
Sven Schnelle | 51676b1 | 2012-07-29 19:18:03 +0200 | [diff] [blame] | 279 | |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 280 | /* Initialize the cpu */ |
| 281 | if (cpu->ops && cpu->ops->init) { |
| 282 | cpu->enabled = 1; |
| 283 | cpu->initialized = 1; |
| 284 | cpu->ops->init(cpu); |
| 285 | } |
| 286 | |
Ronald G. Minnich | 8b93059 | 2012-06-05 14:41:27 -0700 | [diff] [blame] | 287 | printk(BIOS_INFO, "CPU #%d initialized\n", index); |
Yinghai Lu | 30b4abe | 2007-04-06 19:57:42 +0000 | [diff] [blame] | 288 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 289 | return; |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 290 | } |
| 291 | |