blob: 766cf929542e6f8e51a457fbd0c8e4f0c67b0c9c [file] [log] [blame]
/*
* This file is part of the coreboot project.
*
* Copyright (C) 2006 coresystems GmbH
* (Written by Stefan Reinauer <stepan@coresystems.de> for coresystems GmbH)
* Copyright (C) 2007 University of Mannheim
* (Written by Sven Kapferer <skapfere@rumms.uni-mannheim.de> for Uni Ma.)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; version 2 of the License.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA, 02110-1301 USA
*/
/*
* This code sets the Processor Name String for AMD64 CPUs.
*
* Written by Stefan Reinauer <stepan@coresystems.de> using
* the Revision Guide for AMD Athlon™ 64 and AMD Opteron™ Processors
* Document ID 25759 Rev 3.59, April 2006 (Public Version).
*
* Updated by Sven Kapferer <skapfere@rumms.uni-mannheim.de> using
* the Revision Guide for AMD NPT Family 0Fh Processors
* Document ID 33610 Rev 3.00, October 2006 (Public Version).
*/
#include <console/console.h>
#include <string.h>
#include <cpu/x86/msr.h>
#include <cpu/amd/mtrr.h>
#include <cpu/amd/model_fxx_rev.h>
/* The maximum length of CPU names is 48 bytes, including the final NULL byte.
* If you change these names your BIOS will _NOT_ pass the AMD validation and
* your mainboard will not be posted on the AMD Recommended Motherboard Website
*/
#if !CONFIG_K8_REV_F_SUPPORT
static const char *processor_names[]={
/* 0x00 */ "AMD Engineering Sample",
/* 0x01-0x03 */ NULL, NULL, NULL,
/* 0x04 */ "AMD Athlon(tm) 64 Processor XX00+",
/* 0x05 */ "AMD Athlon(tm) 64 X2 Dual Core Processor XX00+",
/* 0x06-0x07 */ NULL, NULL,
/* 0x08 */ "Mobile AMD Athlon(tm) 64 Processor XX00+",
/* 0x09 */ "Mobile AMD Athlon(tm) 64 Processor XX00+",
/* 0x0A */ "AMD Turion(tm) 64 Mobile Technology ML-XX",
/* 0x0B */ "AMD Turion(tm) 64 Mobile Technology MT-XX",
/* 0x0C */ "AMD Opteron(tm) Processor 1YY",
/* 0x0D */ "AMD Opteron(tm) Processor 1YY",
/* 0x0E */ "AMD Opteron(tm) Processor 1YY HE",
/* 0x0F */ "AMD Opteron(tm) Processor 1YY EE",
/* 0x10 */ "AMD Opteron(tm) Processor 2YY",
/* 0x11 */ "AMD Opteron(tm) Processor 2YY",
/* 0x12 */ "AMD Opteron(tm) Processor 2YY HE",
/* 0x13 */ "AMD Opteron(tm) Processor 2YY EE",
/* 0x14 */ "AMD Opteron(tm) Processor 8YY",
/* 0x15 */ "AMD Opteron(tm) Processor 8YY",
/* 0x16 */ "AMD Opteron(tm) Processor 8YY HE",
/* 0x17 */ "AMD Opteron(tm) Processor 8YY EE",
/* 0x18 */ "AMD Athlon(tm) 64 Processor VV00+",
/* 0x19-0x1C */ NULL, NULL, NULL, NULL,
/* 0x1D */ "Mobile AMD Athlon(tm) XP-M Processor XX00+",
/* 0x1E */ "Mobile AMD Athlon(tm) XP-M Processor XX00+",
/* 0x1F */ NULL,
/* 0x20 */ "AMD Athlon(tm) XP Processor XX00+",
/* 0x21 */ "Mobile AMD Sempron(tm) Processor TT00+",
/* 0x22 */ "AMD Sempron(tm) Processor TT00+",
/* 0x23 */ "Mobile AMD Sempron(tm) Processor TT00+",
/* 0x24 */ "AMD Athlon(tm) 64 FX-ZZ Processor",
/* 0x25 */ NULL,
/* 0x26 */ "AMD Sempron(tm) Processor TT00+",
/* 0x27-0x28 */ NULL, NULL,
/* 0x29 */ "Dual Core AMD Opteron(tm) Processor 1RR SE",
/* 0x2A */ "Dual Core AMD Opteron(tm) Processor 2RR SE",
/* 0x2B */ "Dual Core AMD Opteron(tm) Processor 8RR SE",
/* 0x2C */ "Dual Core AMD Opteron(tm) Processor 1RR",
/* 0x2D */ "Dual Core AMD Opteron(tm) Processor 1RR",
/* 0x2E */ "Dual Core AMD Opteron(tm) Processor 1RR HE",
/* 0x2F */ "Dual Core AMD Opteron(tm) Processor 1RR EE",
/* 0x30 */ "Dual Core AMD Opteron(tm) Processor 2RR",
/* 0x31 */ "Dual Core AMD Opteron(tm) Processor 2RR",
/* 0x32 */ "Dual Core AMD Opteron(tm) Processor 2RR HE",
/* 0x33 */ "Dual Core AMD Opteron(tm) Processor 2RR EE",
/* 0x34 */ "Dual Core AMD Opteron(tm) Processor 8RR",
/* 0x35 */ "Dual Core AMD Opteron(tm) Processor 8RR",
/* 0x36 */ "Dual Core AMD Opteron(tm) Processor 8RR HE",
/* 0x37 */ "Dual Core AMD Opteron(tm) Processor 8RR EE",
/* 0x38 */ "Dual Core AMD Opteron(tm) Processor 1RR",
/* 0x39 */ "Dual Core AMD Opteron(tm) Processor 2RR",
/* 0x3A */ "Dual Core AMD Opteron(tm) Processor 8RR"
#define MAX_CPU_NUMBER 0x3A
};
#endif
int init_processor_name(void)
{
#if !CONFIG_K8_REV_F_SUPPORT
u32 EightBitBrandId;
#endif
u32 BrandId;
u32 BrandTableIndex;
u32 NN;
u32 ModelNumber=0;
msr_t progmsr;
int i;
const char *processor_name_string=NULL;
char program_string[48];
unsigned int *program_values = (unsigned int *)program_string;
#if !CONFIG_K8_REV_F_SUPPORT
/* Find out which CPU brand it is */
EightBitBrandId = cpuid_ebx(0x00000001) & 0xff;
BrandId = cpuid_ebx(0x80000001) & 0xffff;
if(!EightBitBrandId && !BrandId) {
BrandTableIndex = 0;
NN = 0xffffff;
} else if(!EightBitBrandId) {
BrandTableIndex = (BrandId >> 6) & 0x3f; // BrandId[11:6]
NN = BrandId & 0x3f; // // BrandId[6:0]
} else {
BrandTableIndex = EightBitBrandId >> (5-2) & 0xfc; // { 0b, 8BitBrandId[7:5], 00b }
NN = EightBitBrandId & 0x1f; // 8BitBrandId[4:0]
}
/* Look up the CPU brand in above table */
if (BrandTableIndex <= MAX_CPU_NUMBER)
processor_name_string = processor_names[BrandTableIndex];
if (!processor_name_string)
processor_name_string = "AMD Processor model unknown";
#endif
#if CONFIG_K8_REV_F_SUPPORT
u32 Socket;
u32 CmpCap;
u32 PwrLmt;
BrandId = cpuid_ebx(0x80000001) & 0xffff;
Socket = (cpuid_eax(0x80000001) & 0x00000030) >> 4; // 00b = S1g1, 01b = F (1207), 11b = AM2
CmpCap = cpuid_ecx(0x80000008) & 0x03; // Number of CPU cores
PwrLmt = ((BrandId >> 14) & 0x01) | ((BrandId >> 5) & 0x0e); // BrandId[8:6,14]
BrandTableIndex = (BrandId >> 9) & 0x1f; // BrandId[13:9]
NN = ((BrandId >> 15) & 0x01) | (BrandId & 0x3f); // BrandId[15,5:0]
if (((BrandTableIndex == 0) && (PwrLmt == 0)) || (NN == 0)) {
processor_name_string = "AMD Engineering Sample";
} else {
/* Use all fields to identify CPU */
switch ((Socket << 16) | (CmpCap << 12) | (BrandTableIndex << 4)
| PwrLmt) {
/* Socket F */
case 0x10012:
processor_name_string =
"AMD Opteron(tm) Processor 22RR EE";
break;
case 0x11002:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 12RR EE";
break;
case 0x11006:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 12RR HE";
break;
case 0x11012:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 22RR EE";
break;
case 0x11016:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 22RR HE";
break;
case 0x1101a:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 22RR";
break;
case 0x1101c:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 22RR SE";
break;
case 0x11042:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 82RR EE";
break;
case 0x11046:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 82RR HE";
break;
case 0x1104a:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 82RR";
break;
case 0x1104c:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 82RR SE";
break;
case 0x1106e:
processor_name_string =
"AMD Athlon(tm) 64 Processor FX-ZZ Processor";
break;
/* Socket AM2 */
/* single core */
case 0x30015:
processor_name_string =
"AMD Sempron(tm) Processor LE-1RR0";
break;
case 0x30026:
processor_name_string =
"AMD Athlon(tm) Processor LE-1ZZ0";
break;
case 0x30036:
processor_name_string =
"AMD Athlon(tm) Processor 1ZZ0B";
break;
case 0x30041:
case 0x30042:
case 0x30043:
case 0x30044:
case 0x30045:
case 0x30048:
processor_name_string =
"AMD Athlon(tm) 64 Processor TT00+";
break;
case 0x30052:
processor_name_string =
"AMD Sempron(tm) Processor RR50p";
case 0x30064:
case 0x30068:
processor_name_string =
"AMD Sempron(tm) Processor TT00+";
break;
case 0x30071:
case 0x30072:
processor_name_string =
"AMD Sempron(tm) Processor TT0U";
break;
case 0x30082:
case 0x30083:
processor_name_string =
"AMD Athlon(tm) Processor TT50e";
break;
case 0x30092:
processor_name_string =
"AMD Athlon(tm) Neo Processor MV-TT";
break;
case 0x300c2:
processor_name_string =
"AMD Sempron(tm) Processor 2RRU";
break;
/* dual-core */
case 0x31016:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 12RR HE";
break;
case 0x3101a:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 12RR";
break;
case 0x3101c:
processor_name_string =
"Dual-Core AMD Opteron(tm) Processor 12RR SE";
break;
case 0x31033:
processor_name_string =
"AMD Athlon(tm) X2 Dual Core Processor BE-2TT0";
break;
case 0x31041:
case 0x31042:
case 0x31046:
case 0x31048:
case 0x3104c:
processor_name_string =
"AMD Athlon(tm) 64 X2 Dual Core Processor TT00+";
break;
case 0x3105c:
processor_name_string =
"AMD Athlon(tm) 64 FX-ZZ Dual Core Processor";
break;
case 0x31066:
processor_name_string =
"AMD Sempron(tm) Dual Core Processor RR00";
break;
case 0x31073:
processor_name_string =
"AMD Athlon(tm) Dual Core Processor TT50e";
break;
case 0x31076:
case 0x31077:
processor_name_string =
"AMD Athlon(tm) Dual Core Processor TT00B";
break;
case 0x31083:
processor_name_string =
"AMD Athlon(tm) Dual Core Processor TT50B";
break;
case 0x31091:
processor_name_string =
"AMD Athlon(tm) X2 Dual Core Processor TT50e";
break;
case 0x310a1:
case 0x310a2:
processor_name_string =
"AMD Athlon(tm) Neo X2 Dual Core Processor TT50e";
break;
case 0x310b0:
case 0x310c0:
processor_name_string =
"AMD Turion(tm) Neo X2 Dual Core Processor L6RR";
break;
/* Socket S1g1 */
/* single core */
case 0x00012:
processor_name_string =
"AMD Athlon(tm) 64 Processor TT00+";
break;
case 0x0002c:
processor_name_string =
"AMD Turion(tm) 64 Mobile Technology MK-YY";
break;
case 0x00031:
processor_name_string =
"Mobile AMD Sempron(tm) Processor TT00+";
break;
case 0x00036:
processor_name_string =
"Mobile AMD Sempron(tm) Processor PP00+";
break;
case 0x0003c:
processor_name_string =
"Mobile AMD Sempron(tm) Processor PP00+";
break;
case 0x00042:
processor_name_string =
"AMD Sempron(tm) Processor TT00+";
break;
case 0x00064:
case 0x00066:
case 0x0006c:
processor_name_string =
"AMD Athlon(tm) Processor TF-TT";
break;
case 0x00073:
processor_name_string =
"AMD Athlon(tm) Processor L1RR";
break;
/* dual-core */
case 0x0101c:
processor_name_string =
"AMD Sempron(tm) Dual Core Processor TJ-YY";
break;
case 0x0102c:
processor_name_string =
"AMD Turion(tm) 64 X2 Mobile Technology TL-YY";
break;
case 0x01034:
case 0x0103c:
processor_name_string =
"AMD Athlon(tm) 64 X2 Dual-Core Processor TK-YY";
break;
case 0x01054:
processor_name_string =
"AMD Athlon(tm) 64 X2 Dual Core Processor TT00+";
break;
case 0x01062:
processor_name_string =
"AMD Athlon(tm) X2 Dual Core Processor L3RR";
break;
case 0x01074:
processor_name_string =
"AMD Athlon(tm) X2 Dual Core Processor L5RR";
break;
default:
processor_name_string = "AMD Processor model unknown";
}
}
#endif
memset(program_string, 0, 48);
strcpy(program_string, processor_name_string);
/* Now create a model number - See Table 4. Model Number Calculation
* in the Revision Guide. NOTE: #6, EE was changed to VV because
* otherwise it clashes with the brand names.
*/
for (i=0; i<47; i++) { // 48 -1
if(program_string[i] == program_string[i+1]) {
switch (program_string[i]) {
#if !CONFIG_K8_REV_F_SUPPORT
case 'X': ModelNumber = 22+ NN; break;
case 'Y': ModelNumber = 38 + (2*NN); break;
case 'Z':
case 'T': ModelNumber = 24 + NN; break;
case 'R': ModelNumber = 45 + (5*NN); break;
case 'V': ModelNumber = 9 + NN; break;
#endif
#if CONFIG_K8_REV_F_SUPPORT
case 'R': ModelNumber = NN - 1; break;
case 'P': ModelNumber = 26 + NN; break;
case 'T': ModelNumber = 15 + (CmpCap * 10) + NN; break;
case 'Z': ModelNumber = 57 + NN; break;
case 'Y': ModelNumber = 29 + NN; break;
#endif
}
if(ModelNumber && ModelNumber < 100) {
// No idea what to do with RR=100. According
// to the revision guide this is possible.
//
// --> "AMD Opteron(tm) Processor 8100"?
program_string[i]=(ModelNumber/10)+'0';
program_string[i+1]=(ModelNumber%10)+'0';
break;
}
}
}
printk(BIOS_DEBUG, "CPU model %s\n", program_string);
for (i=0; i<6; i++) {
progmsr.lo = program_values[(2*i)+0];
progmsr.hi = program_values[(2*i)+1];
wrmsr_amd(0xc0010030+i, progmsr);
}
return 0;
}