bdk: Use Kconfig options instead of getenv()
[coreboot.git] / payloads / coreinfo / cpuinfo_module.c
blob72b941ef277aa006a867689ea6e45957d2333a05
1 /*
2 * This file is part of the coreinfo project.
4 * It is derived from the x86info project, which is GPLv2-licensed.
6 * Copyright (C) 2001-2007 Dave Jones <davej@codemonkey.org.uk>
7 * Copyright (C) 2008 Advanced Micro Devices, Inc.
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; version 2 of the License.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
19 #include "coreinfo.h"
21 #if IS_ENABLED(CONFIG_MODULE_CPUINFO)
22 #include <arch/rdtsc.h>
24 #define VENDOR_INTEL 0x756e6547
25 #define VENDOR_AMD 0x68747541
26 #define VENDOR_CYRIX 0x69727943
27 #define VENDOR_IDT 0x746e6543
28 #define VENDOR_GEODE 0x646f6547
29 #define VENDOR_RISE 0x52697365
30 #define VENDOR_RISE2 0x65736952
31 #define VENDOR_SIS 0x20536953
33 /* CPUID 0x00000001 EDX flags */
34 static const char *generic_cap_flags[] = {
35 "fpu", "vme", "de", "pse", "tsc", "msr", "pae", "mce",
36 "cx8", "apic", NULL, "sep", "mtrr", "pge", "mca", "cmov",
37 "pat", "pse36", "psn", "clflsh", NULL, "ds", "acpi", "mmx",
38 "fxsr", "sse", "sse2", "ss", "ht", "tm", NULL, "pbe"
41 /* CPUID 0x00000001 ECX flags */
42 static const char *intel_cap_generic_ecx_flags[] = {
43 "sse3", NULL, NULL, "monitor", "ds-cpl", "vmx", NULL, "est",
44 "tm2", "ssse3", "cntx-id", NULL, NULL, "cx16", "xTPR", NULL,
45 NULL, NULL, "dca", NULL, NULL, NULL, NULL, NULL,
46 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL
49 /* CPUID 0x80000001 EDX flags */
50 static const char *intel_cap_extended_edx_flags[] = {
51 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
52 NULL, NULL, NULL, "SYSCALL", NULL, NULL, NULL, NULL,
53 NULL, NULL, NULL, NULL, "xd", NULL, NULL, NULL,
54 NULL, NULL, NULL, NULL, NULL, "em64t", NULL, NULL,
57 /* CPUID 0x80000001 ECX flags */
58 static const char *intel_cap_extended_ecx_flags[] = {
59 "lahf_lm", NULL, NULL, NULL, NULL, NULL, NULL, NULL,
60 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
61 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
62 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
65 static const char *amd_cap_generic_ecx_flags[] = {
66 "sse3", NULL, NULL, "mwait", NULL, NULL, NULL, NULL,
67 NULL, NULL, NULL, NULL, NULL, "cmpxchg16b", NULL, NULL,
68 NULL, NULL, NULL, NULL, NULL, NULL, NULL, "popcnt",
69 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL
72 static const char *amd_cap_extended_edx_flags[] = {
73 "fpu", "vme", "de", "pse", "tsc", "msr", "pae", "mce",
74 "cx8", "apic", NULL, "sep", "mtrr", "pge", "mca", "cmov",
75 "pat", "pse36", NULL, "mp", "nx", NULL, "mmxext", "mmx",
76 "fxsr", "ffxsr", "page1gb", "rdtscp",
77 NULL, "lm", "3dnowext", "3dnow"
78 }; /* "mp" defined for CPUs prior to AMD family 0xf */
80 static const char *amd_cap_extended_ecx_flags[] = {
81 "lahf/sahf", "CmpLegacy", "svm", "ExtApicSpace",
82 "LockMovCr0", "abm", "sse4a", "misalignsse",
83 "3dnowPref", "osvw", "ibs", NULL, "skinit", "wdt", NULL, NULL,
84 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
85 NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL
88 static unsigned long vendor;
89 static unsigned int cpu_khz;
91 static void decode_flags(WINDOW *win, unsigned long reg, const char **flags,
92 int *row)
94 int i;
95 int lrow = *row;
97 wmove(win, lrow, 2);
99 for (i = 0; i < 32; i++) {
100 if (flags[i] == NULL)
101 continue;
103 if (reg & (1 << i))
104 wprintw(win, "%s ", flags[i]);
106 if (i && (i % 16) == 0) {
107 lrow++;
108 wmove(win, lrow, 2);
112 *row = lrow;
115 static void get_features(WINDOW *win, int *row)
117 unsigned long eax, ebx, ecx, edx;
118 int lrow = *row;
120 wmove(win, lrow++, 1);
121 wprintw(win, "Features: ");
123 docpuid(0x00000001, &eax, &ebx, &ecx, &edx);
124 decode_flags(win, edx, generic_cap_flags, &lrow);
126 lrow++;
128 switch (vendor) {
129 case VENDOR_AMD:
130 wmove(win, lrow++, 1);
131 wprintw(win, "AMD Extended Flags: ");
132 decode_flags(win, ecx, amd_cap_generic_ecx_flags, &lrow);
133 docpuid(0x80000001, &eax, &ebx, &ecx, &edx);
134 decode_flags(win, edx, amd_cap_extended_edx_flags, &lrow);
135 decode_flags(win, ecx, amd_cap_extended_ecx_flags, &lrow);
136 break;
137 case VENDOR_INTEL:
138 wmove(win, lrow++, 1);
139 wprintw(win, "Intel Extended Flags: ");
140 decode_flags(win, ecx, intel_cap_generic_ecx_flags, &lrow);
141 docpuid(0x80000001, &eax, &ebx, &ecx, &edx);
142 decode_flags(win, edx, intel_cap_extended_edx_flags, &lrow);
143 decode_flags(win, ecx, intel_cap_extended_ecx_flags, &lrow);
144 break;
147 *row = lrow;
150 static void do_name(WINDOW *win, int row)
152 char name[49], *p;
153 unsigned long eax, ebx, ecx, edx;
154 int i, t;
156 p = name;
158 for (i = 0x80000002; i <= 0x80000004; i++) {
159 docpuid(i, &eax, &ebx, &ecx, &edx);
161 if (eax == 0)
162 break;
164 for (t = 0; t < 4; t++)
165 *p++ = eax >> (8 * t);
166 for (t = 0; t < 4; t++)
167 *p++ = ebx >> (8 * t);
168 for (t = 0; t < 4; t++)
169 *p++ = ecx >> (8 * t);
170 for (t = 0; t < 4; t++)
171 *p++ = edx >> (8 * t);
174 mvwprintw(win, row, 1, "Processor: %s", name);
177 static int cpuinfo_module_redraw(WINDOW *win)
179 unsigned long eax, ebx, ecx, edx;
180 unsigned int brand;
181 char *vstr;
182 int row = 2;
184 print_module_title(win, "CPU Information");
186 docpuid(0, NULL, &vendor, NULL, NULL);
188 switch (vendor) {
189 case VENDOR_INTEL:
190 vstr = "Intel";
191 break;
192 case VENDOR_AMD:
193 vstr = "AMD";
194 break;
195 case VENDOR_CYRIX:
196 vstr = "Cyrix";
197 break;
198 case VENDOR_IDT:
199 vstr = "IDT";
200 break;
201 case VENDOR_GEODE:
202 vstr = "NatSemi Geode";
203 break;
204 case VENDOR_RISE:
205 case VENDOR_RISE2:
206 vstr = "RISE";
207 break;
208 case VENDOR_SIS:
209 vstr = "SiS";
210 break;
211 default:
212 vstr = "Unknown";
213 break;
216 mvwprintw(win, row++, 1, "Vendor: %s", vstr);
218 do_name(win, row++);
220 docpuid(0x00000001, &eax, &ebx, &ecx, &edx);
222 mvwprintw(win, row++, 1, "Family: %X", (eax >> 8) & 0x0f);
223 mvwprintw(win, row++, 1, "Model: %X",
224 ((eax >> 4) & 0xf) | ((eax >> 16) & 0xf) << 4);
226 mvwprintw(win, row++, 1, "Stepping: %X", eax & 0xf);
228 if (vendor == VENDOR_AMD) {
229 docpuid(0x80000001, &eax, &ebx, &ecx, &edx);
230 brand = ((ebx >> 9) & 0x1f);
232 mvwprintw(win, row++, 1, "Brand: %X", brand);
235 if (cpu_khz != 0)
236 mvwprintw(win, row++, 1, "CPU Speed: %d Mhz", cpu_khz / 1000);
237 else
238 mvwprintw(win, row++, 1, "CPU Speed: Error");
240 row++;
241 get_features(win, &row);
243 return 0;
246 static unsigned int getticks(void)
248 unsigned long long start, end;
250 /* Read the number of ticks during the period. */
251 start = rdtsc();
252 mdelay(100);
253 end = rdtsc();
255 return (unsigned int)((end - start) / 100);
258 static int cpuinfo_module_init(void)
260 cpu_khz = getticks();
261 return 0;
264 struct coreinfo_module cpuinfo_module = {
265 .name = "CPU Info",
266 .init = cpuinfo_module_init,
267 .redraw = cpuinfo_module_redraw,
270 #else
272 struct coreinfo_module cpuinfo_module = {
275 #endif