2005-04-16 15:20:36 -07:00
/*
* Intel CPU Microcode Update Driver for Linux
*
2006-12-13 00:35:14 -08:00
* Copyright ( C ) 2000 - 2006 Tigran Aivazian < tigran @ aivazian . fsnet . co . uk >
2006-09-27 01:50:51 -07:00
* 2006 Shaohua Li < shaohua . li @ intel . com >
2005-04-16 15:20:36 -07:00
*
* This driver allows to upgrade microcode on Intel processors
* belonging to IA - 32 family - PentiumPro , Pentium II ,
* Pentium III , Xeon , Pentium 4 , etc .
*
* Reference : Section 8.10 of Volume III , Intel Pentium 4 Manual ,
* Order Number 245472 or free download from :
*
* http : //developer.intel.com/design/pentium4/manuals/245472.htm
*
* For more information , go to http : //www.urbanmyth.org/microcode
*
* 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 ; either version
* 2 of the License , or ( at your option ) any later version .
*
* 1.0 16 Feb 2000 , Tigran Aivazian < tigran @ sco . com >
* Initial release .
* 1.01 18 Feb 2000 , Tigran Aivazian < tigran @ sco . com >
* Added read ( ) support + cleanups .
* 1.02 21 Feb 2000 , Tigran Aivazian < tigran @ sco . com >
* Added ' device trimming ' support . open ( O_WRONLY ) zeroes
* and frees the saved copy of applied microcode .
* 1.03 29 Feb 2000 , Tigran Aivazian < tigran @ sco . com >
* Made to use devfs ( / dev / cpu / microcode ) + cleanups .
* 1.04 06 Jun 2000 , Simon Trimmer < simon @ veritas . com >
* Added misc device support ( now uses both devfs and misc ) .
* Added MICROCODE_IOCFREE ioctl to clear memory .
* 1.05 09 Jun 2000 , Simon Trimmer < simon @ veritas . com >
* Messages for error cases ( non Intel & no suitable microcode ) .
* 1.06 03 Aug 2000 , Tigran Aivazian < tigran @ veritas . com >
* Removed - > release ( ) . Removed exclusive open and status bitmap .
* Added microcode_rwsem to serialize read ( ) / write ( ) / ioctl ( ) .
* Removed global kernel lock usage .
* 1.07 07 Sep 2000 , Tigran Aivazian < tigran @ veritas . com >
* Write 0 to 0x8B msr and then cpuid before reading revision ,
* so that it works even if there were no update done by the
* BIOS . Otherwise , reading from 0x8B gives junk ( which happened
* to be 0 on my machine which is why it worked even when I
* disabled update by the BIOS )
* Thanks to Eric W . Biederman < ebiederman @ lnxi . com > for the fix .
* 1.08 11 Dec 2000 , Richard Schaal < richard . schaal @ intel . com > and
* Tigran Aivazian < tigran @ veritas . com >
* Intel Pentium 4 processor support and bugfixes .
* 1.09 30 Oct 2001 , Tigran Aivazian < tigran @ veritas . com >
* Bugfix for HT ( Hyper - Threading ) enabled processors
* whereby processor resources are shared by all logical processors
* in a single CPU package .
* 1.10 28 Feb 2002 Asit K Mallick < asit . k . mallick @ intel . com > and
* Tigran Aivazian < tigran @ veritas . com > ,
* Serialize updates as required on HT processors due to speculative
* nature of implementation .
* 1.11 22 Mar 2002 Tigran Aivazian < tigran @ veritas . com >
* Fix the panic when writing zero - length microcode chunk .
* 1.12 29 Sep 2003 Nitin Kamble < nitin . a . kamble @ intel . com > ,
* Jun Nakajima < jun . nakajima @ intel . com >
* Support for the microcode updates in the new format .
* 1.13 10 Oct 2003 Tigran Aivazian < tigran @ veritas . com >
* Removed - > read ( ) method and obsoleted MICROCODE_IOCFREE ioctl
* because we no longer hold a copy of applied microcode
* in kernel memory .
* 1.14 25 Jun 2004 Tigran Aivazian < tigran @ veritas . com >
* Fix sigmatch ( ) macro to handle old CPUs with pf = = 0.
* Thanks to Stuart Swales for pointing out this bug .
*/
//#define DEBUG /* pr_debug */
2006-01-11 12:17:48 -08:00
# include <linux/capability.h>
2005-04-16 15:20:36 -07:00
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/sched.h>
2006-02-28 16:58:53 -08:00
# include <linux/cpumask.h>
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/vmalloc.h>
# include <linux/miscdevice.h>
# include <linux/spinlock.h>
# include <linux/mm.h>
2007-07-30 02:36:13 +04:00
# include <linux/fs.h>
2006-03-26 01:37:14 -08:00
# include <linux/mutex.h>
2006-09-27 01:50:52 -07:00
# include <linux/cpu.h>
# include <linux/firmware.h>
# include <linux/platform_device.h>
2005-04-16 15:20:36 -07:00
# include <asm/msr.h>
# include <asm/uaccess.h>
# include <asm/processor.h>
MODULE_DESCRIPTION ( " Intel CPU (IA-32) Microcode Update Driver " ) ;
2006-12-13 00:35:14 -08:00
MODULE_AUTHOR ( " Tigran Aivazian <tigran@aivazian.fsnet.co.uk> " ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;
2006-06-23 02:04:19 -07:00
# define MICROCODE_VERSION "1.14a"
2005-04-16 15:20:36 -07:00
# define DEFAULT_UCODE_DATASIZE (2000) /* 2000 bytes */
# define MC_HEADER_SIZE (sizeof (microcode_header_t)) /* 48 bytes */
# define DEFAULT_UCODE_TOTALSIZE (DEFAULT_UCODE_DATASIZE + MC_HEADER_SIZE) /* 2048 bytes */
# define EXT_HEADER_SIZE (sizeof (struct extended_sigtable)) /* 20 bytes */
# define EXT_SIGNATURE_SIZE (sizeof (struct extended_signature)) /* 12 bytes */
# define DWSIZE (sizeof (u32))
# define get_totalsize(mc) \
( ( ( microcode_t * ) mc ) - > hdr . totalsize ? \
( ( microcode_t * ) mc ) - > hdr . totalsize : DEFAULT_UCODE_TOTALSIZE )
# define get_datasize(mc) \
( ( ( microcode_t * ) mc ) - > hdr . datasize ? \
( ( microcode_t * ) mc ) - > hdr . datasize : DEFAULT_UCODE_DATASIZE )
# define sigmatch(s1, s2, p1, p2) \
( ( ( s1 ) = = ( s2 ) ) & & ( ( ( p1 ) & ( p2 ) ) | | ( ( ( p1 ) = = 0 ) & & ( ( p2 ) = = 0 ) ) ) )
# define exttable_size(et) ((et)->count * EXT_SIGNATURE_SIZE + EXT_HEADER_SIZE)
/* serialize access to the physical write to MSR 0x79 */
static DEFINE_SPINLOCK ( microcode_update_lock ) ;
/* no concurrent ->write()s are allowed on /dev/cpu/microcode */
2006-03-26 01:37:14 -08:00
static DEFINE_MUTEX ( microcode_mutex ) ;
2005-04-16 15:20:36 -07:00
static struct ucode_cpu_info {
2006-09-27 01:50:51 -07:00
int valid ;
2005-04-16 15:20:36 -07:00
unsigned int sig ;
2006-09-27 01:50:51 -07:00
unsigned int pf ;
2005-04-16 15:20:36 -07:00
unsigned int rev ;
microcode_t * mc ;
} ucode_cpu_info [ NR_CPUS ] ;
2006-09-27 01:50:51 -07:00
static void collect_cpu_info ( int cpu_num )
2005-04-16 15:20:36 -07:00
{
2007-10-19 20:35:04 +02:00
struct cpuinfo_x86 * c = & cpu_data ( cpu_num ) ;
2005-04-16 15:20:36 -07:00
struct ucode_cpu_info * uci = ucode_cpu_info + cpu_num ;
unsigned int val [ 2 ] ;
2006-09-27 01:50:51 -07:00
/* We should bind the task to the CPU */
BUG_ON ( raw_smp_processor_id ( ) ! = cpu_num ) ;
uci - > pf = uci - > rev = 0 ;
2005-04-16 15:20:36 -07:00
uci - > mc = NULL ;
2006-09-27 01:50:51 -07:00
uci - > valid = 1 ;
2005-04-16 15:20:36 -07:00
if ( c - > x86_vendor ! = X86_VENDOR_INTEL | | c - > x86 < 6 | |
cpu_has ( c , X86_FEATURE_IA64 ) ) {
2006-09-27 01:50:51 -07:00
printk ( KERN_ERR " microcode: CPU%d not a capable Intel "
" processor \n " , cpu_num ) ;
uci - > valid = 0 ;
2005-04-16 15:20:36 -07:00
return ;
2006-09-27 01:50:51 -07:00
}
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
uci - > sig = cpuid_eax ( 0x00000001 ) ;
if ( ( c - > x86_model > = 5 ) | | ( c - > x86 > 6 ) ) {
/* get processor flags from MSR 0x17 */
rdmsr ( MSR_IA32_PLATFORM_ID , val [ 0 ] , val [ 1 ] ) ;
uci - > pf = 1 < < ( ( val [ 1 ] > > 18 ) & 7 ) ;
2005-04-16 15:20:36 -07:00
}
wrmsr ( MSR_IA32_UCODE_REV , 0 , 0 ) ;
2005-09-03 15:56:37 -07:00
/* see notes above for revision 1.07. Apparent chip bug */
2006-01-11 22:45:27 +01:00
sync_core ( ) ;
2005-04-16 15:20:36 -07:00
/* get the current revision from MSR 0x8B */
rdmsr ( MSR_IA32_UCODE_REV , val [ 0 ] , uci - > rev ) ;
pr_debug ( " microcode: collect_cpu_info : sig=0x%x, pf=0x%x, rev=0x%x \n " ,
uci - > sig , uci - > pf , uci - > rev ) ;
}
2006-09-27 01:50:51 -07:00
static inline int microcode_update_match ( int cpu_num ,
microcode_header_t * mc_header , int sig , int pf )
2005-04-16 15:20:36 -07:00
{
struct ucode_cpu_info * uci = ucode_cpu_info + cpu_num ;
2006-09-27 01:50:51 -07:00
if ( ! sigmatch ( sig , uci - > sig , pf , uci - > pf )
| | mc_header - > rev < = uci - > rev )
return 0 ;
return 1 ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
static int microcode_sanity_check ( void * mc )
2005-04-16 15:20:36 -07:00
{
2006-09-27 01:50:51 -07:00
microcode_header_t * mc_header = mc ;
struct extended_sigtable * ext_header = NULL ;
struct extended_signature * ext_sig ;
unsigned long total_size , data_size , ext_table_size ;
int sum , orig_sum , ext_sigcount = 0 , i ;
total_size = get_totalsize ( mc_header ) ;
data_size = get_datasize ( mc_header ) ;
2006-09-27 01:50:54 -07:00
if ( data_size + MC_HEADER_SIZE > total_size ) {
2006-09-27 01:50:51 -07:00
printk ( KERN_ERR " microcode: error! "
" Bad data size in microcode data file \n " ) ;
return - EINVAL ;
}
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
if ( mc_header - > ldrver ! = 1 | | mc_header - > hdrver ! = 1 ) {
printk ( KERN_ERR " microcode: error! "
" Unknown microcode update format \n " ) ;
return - EINVAL ;
}
ext_table_size = total_size - ( MC_HEADER_SIZE + data_size ) ;
if ( ext_table_size ) {
if ( ( ext_table_size < EXT_HEADER_SIZE )
| | ( ( ext_table_size - EXT_HEADER_SIZE ) % EXT_SIGNATURE_SIZE ) ) {
printk ( KERN_ERR " microcode: error! "
" Small exttable size in microcode data file \n " ) ;
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
ext_header = mc + MC_HEADER_SIZE + data_size ;
if ( ext_table_size ! = exttable_size ( ext_header ) ) {
printk ( KERN_ERR " microcode: error! "
" Bad exttable size in microcode data file \n " ) ;
return - EFAULT ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
ext_sigcount = ext_header - > count ;
}
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
/* check extended table checksum */
if ( ext_table_size ) {
int ext_table_sum = 0 ;
2006-09-27 01:50:53 -07:00
int * ext_tablep = ( int * ) ext_header ;
2006-09-27 01:50:51 -07:00
i = ext_table_size / DWSIZE ;
while ( i - - )
ext_table_sum + = ext_tablep [ i ] ;
if ( ext_table_sum ) {
printk ( KERN_WARNING " microcode: aborting, "
" bad extended signature table checksum \n " ) ;
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
}
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
/* calculate the checksum */
orig_sum = 0 ;
i = ( MC_HEADER_SIZE + data_size ) / DWSIZE ;
while ( i - - )
orig_sum + = ( ( int * ) mc ) [ i ] ;
if ( orig_sum ) {
printk ( KERN_ERR " microcode: aborting, bad checksum \n " ) ;
return - EINVAL ;
}
if ( ! ext_table_size )
return 0 ;
/* check extended signature checksum */
for ( i = 0 ; i < ext_sigcount ; i + + ) {
2008-01-30 13:33:23 +01:00
ext_sig = ( void * ) ext_header + EXT_HEADER_SIZE +
EXT_SIGNATURE_SIZE * i ;
2006-09-27 01:50:51 -07:00
sum = orig_sum
- ( mc_header - > sig + mc_header - > pf + mc_header - > cksum )
+ ( ext_sig - > sig + ext_sig - > pf + ext_sig - > cksum ) ;
if ( sum ) {
printk ( KERN_ERR " microcode: aborting, bad checksum \n " ) ;
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
}
return 0 ;
}
2006-02-28 16:58:53 -08:00
2006-09-27 01:50:51 -07:00
/*
* return 0 - no update found
* return 1 - found update
* return < 0 - error
*/
static int get_maching_microcode ( void * mc , int cpu )
{
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
microcode_header_t * mc_header = mc ;
struct extended_sigtable * ext_header ;
unsigned long total_size = get_totalsize ( mc_header ) ;
int ext_sigcount , i ;
struct extended_signature * ext_sig ;
void * new_mc ;
if ( microcode_update_match ( cpu , mc_header ,
mc_header - > sig , mc_header - > pf ) )
goto find ;
if ( total_size < = get_datasize ( mc_header ) + MC_HEADER_SIZE )
return 0 ;
2008-01-30 13:33:23 +01:00
ext_header = mc + get_datasize ( mc_header ) + MC_HEADER_SIZE ;
2006-09-27 01:50:51 -07:00
ext_sigcount = ext_header - > count ;
2008-01-30 13:33:23 +01:00
ext_sig = ( void * ) ext_header + EXT_HEADER_SIZE ;
2006-09-27 01:50:51 -07:00
for ( i = 0 ; i < ext_sigcount ; i + + ) {
if ( microcode_update_match ( cpu , mc_header ,
ext_sig - > sig , ext_sig - > pf ) )
goto find ;
ext_sig + + ;
}
return 0 ;
find :
pr_debug ( " microcode: CPU %d found a matching microcode update with "
" version 0x%x (current=0x%x) \n " , cpu , mc_header - > rev , uci - > rev ) ;
new_mc = vmalloc ( total_size ) ;
if ( ! new_mc ) {
printk ( KERN_ERR " microcode: error! Can not allocate memory \n " ) ;
return - ENOMEM ;
}
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
/* free previous update file */
vfree ( uci - > mc ) ;
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
memcpy ( new_mc , mc , total_size ) ;
uci - > mc = new_mc ;
return 1 ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
static void apply_microcode ( int cpu )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
unsigned int val [ 2 ] ;
2006-09-27 01:50:51 -07:00
int cpu_num = raw_smp_processor_id ( ) ;
2005-04-16 15:20:36 -07:00
struct ucode_cpu_info * uci = ucode_cpu_info + cpu_num ;
2006-09-27 01:50:51 -07:00
/* We should bind the task to the CPU */
BUG_ON ( cpu_num ! = cpu ) ;
if ( uci - > mc = = NULL )
2005-04-16 15:20:36 -07:00
return ;
/* serialize access to the physical write to MSR 0x79 */
spin_lock_irqsave ( & microcode_update_lock , flags ) ;
/* write microcode via MSR 0x79 */
wrmsr ( MSR_IA32_UCODE_WRITE ,
( unsigned long ) uci - > mc - > bits ,
( unsigned long ) uci - > mc - > bits > > 16 > > 16 ) ;
wrmsr ( MSR_IA32_UCODE_REV , 0 , 0 ) ;
2005-09-03 15:56:37 -07:00
/* see notes above for revision 1.07. Apparent chip bug */
2006-01-11 22:45:27 +01:00
sync_core ( ) ;
2005-09-03 15:56:37 -07:00
2005-04-16 15:20:36 -07:00
/* get the current revision from MSR 0x8B */
rdmsr ( MSR_IA32_UCODE_REV , val [ 0 ] , val [ 1 ] ) ;
spin_unlock_irqrestore ( & microcode_update_lock , flags ) ;
2006-09-27 01:50:51 -07:00
if ( val [ 1 ] ! = uci - > mc - > hdr . rev ) {
printk ( KERN_ERR " microcode: CPU%d updated from revision "
" 0x%x to 0x%x failed \n " , cpu_num , uci - > rev , val [ 1 ] ) ;
return ;
}
pr_debug ( " microcode: CPU%d updated from revision "
2005-04-16 15:20:36 -07:00
" 0x%x to 0x%x, date = %08x \n " ,
cpu_num , uci - > rev , val [ 1 ] , uci - > mc - > hdr . date ) ;
2006-09-27 01:50:51 -07:00
uci - > rev = val [ 1 ] ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
# ifdef CONFIG_MICROCODE_OLD_INTERFACE
static void __user * user_buffer ; /* user area microcode data buffer */
static unsigned int user_buffer_size ; /* it's size */
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
static long get_next_ucode ( void * * mc , long offset )
{
microcode_header_t mc_header ;
unsigned long total_size ;
/* No more data */
if ( offset > = user_buffer_size )
return 0 ;
if ( copy_from_user ( & mc_header , user_buffer + offset , MC_HEADER_SIZE ) ) {
printk ( KERN_ERR " microcode: error! Can not read user data \n " ) ;
return - EFAULT ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
total_size = get_totalsize ( & mc_header ) ;
2006-09-27 01:50:54 -07:00
if ( offset + total_size > user_buffer_size ) {
2006-09-27 01:50:51 -07:00
printk ( KERN_ERR " microcode: error! Bad total size in microcode "
" data file \n " ) ;
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
* mc = vmalloc ( total_size ) ;
if ( ! * mc )
return - ENOMEM ;
if ( copy_from_user ( * mc , user_buffer + offset , total_size ) ) {
printk ( KERN_ERR " microcode: error! Can not read user data \n " ) ;
vfree ( * mc ) ;
return - EFAULT ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
return offset + total_size ;
}
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
static int do_microcode_update ( void )
{
long cursor = 0 ;
int error = 0 ;
2007-02-13 13:26:25 +01:00
void * new_mc = NULL ;
2006-09-27 01:50:51 -07:00
int cpu ;
cpumask_t old ;
old = current - > cpus_allowed ;
while ( ( cursor = get_next_ucode ( & new_mc , cursor ) ) > 0 ) {
error = microcode_sanity_check ( new_mc ) ;
if ( error )
goto out ;
/*
* It ' s possible the data file has multiple matching ucode ,
* lets keep searching till the latest version
*/
for_each_online_cpu ( cpu ) {
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
if ( ! uci - > valid )
continue ;
set_cpus_allowed ( current , cpumask_of_cpu ( cpu ) ) ;
error = get_maching_microcode ( new_mc , cpu ) ;
if ( error < 0 )
goto out ;
if ( error = = 1 )
apply_microcode ( cpu ) ;
2005-04-16 15:20:36 -07:00
}
2006-09-27 01:50:51 -07:00
vfree ( new_mc ) ;
2005-04-16 15:20:36 -07:00
}
out :
2006-09-27 01:50:51 -07:00
if ( cursor > 0 )
vfree ( new_mc ) ;
if ( cursor < 0 )
error = cursor ;
set_cpus_allowed ( current , old ) ;
2005-04-16 15:20:36 -07:00
return error ;
}
2006-09-27 01:50:51 -07:00
static int microcode_open ( struct inode * unused1 , struct file * unused2 )
{
return capable ( CAP_SYS_RAWIO ) ? 0 : - EPERM ;
}
2005-04-16 15:20:36 -07:00
static ssize_t microcode_write ( struct file * file , const char __user * buf , size_t len , loff_t * ppos )
{
ssize_t ret ;
if ( ( len > > PAGE_SHIFT ) > num_physpages ) {
printk ( KERN_ERR " microcode: too much data (max %ld pages) \n " , num_physpages ) ;
return - EINVAL ;
}
2008-01-25 21:08:02 +01:00
get_online_cpus ( ) ;
2006-03-26 01:37:14 -08:00
mutex_lock ( & microcode_mutex ) ;
2005-04-16 15:20:36 -07:00
user_buffer = ( void __user * ) buf ;
user_buffer_size = ( int ) len ;
ret = do_microcode_update ( ) ;
if ( ! ret )
ret = ( ssize_t ) len ;
2006-03-26 01:37:14 -08:00
mutex_unlock ( & microcode_mutex ) ;
2008-01-25 21:08:02 +01:00
put_online_cpus ( ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
2007-02-12 00:55:31 -08:00
static const struct file_operations microcode_fops = {
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
. write = microcode_write ,
. open = microcode_open ,
} ;
static struct miscdevice microcode_dev = {
. minor = MICROCODE_MINOR ,
. name = " microcode " ,
. fops = & microcode_fops ,
} ;
2006-09-27 01:50:51 -07:00
static int __init microcode_dev_init ( void )
2005-04-16 15:20:36 -07:00
{
int error ;
error = misc_register ( & microcode_dev ) ;
if ( error ) {
printk ( KERN_ERR
" microcode: can't misc_register on minor=%d \n " ,
MICROCODE_MINOR ) ;
return error ;
}
2006-09-27 01:50:51 -07:00
return 0 ;
}
2007-06-01 00:47:10 -07:00
static void microcode_dev_exit ( void )
2006-09-27 01:50:51 -07:00
{
misc_deregister ( & microcode_dev ) ;
}
MODULE_ALIAS_MISCDEV ( MICROCODE_MINOR ) ;
# else
# define microcode_dev_init() 0
# define microcode_dev_exit() do { } while(0)
# endif
2006-09-27 01:50:52 -07:00
static long get_next_ucode_from_buffer ( void * * mc , void * buf ,
unsigned long size , long offset )
{
microcode_header_t * mc_header ;
unsigned long total_size ;
/* No more data */
if ( offset > = size )
return 0 ;
mc_header = ( microcode_header_t * ) ( buf + offset ) ;
total_size = get_totalsize ( mc_header ) ;
2006-09-27 01:50:54 -07:00
if ( offset + total_size > size ) {
2006-09-27 01:50:52 -07:00
printk ( KERN_ERR " microcode: error! Bad data in microcode data file \n " ) ;
return - EINVAL ;
}
* mc = vmalloc ( total_size ) ;
if ( ! * mc ) {
printk ( KERN_ERR " microcode: error! Can not allocate memory \n " ) ;
return - ENOMEM ;
}
memcpy ( * mc , buf + offset , total_size ) ;
return offset + total_size ;
}
/* fake device for request_firmware */
static struct platform_device * microcode_pdev ;
static int cpu_request_microcode ( int cpu )
{
char name [ 30 ] ;
2007-10-19 20:35:04 +02:00
struct cpuinfo_x86 * c = & cpu_data ( cpu ) ;
2006-09-27 01:50:52 -07:00
const struct firmware * firmware ;
2006-09-27 01:50:53 -07:00
void * buf ;
2006-09-27 01:50:52 -07:00
unsigned long size ;
long offset = 0 ;
int error ;
void * mc ;
/* We should bind the task to the CPU */
BUG_ON ( cpu ! = raw_smp_processor_id ( ) ) ;
sprintf ( name , " intel-ucode/%02x-%02x-%02x " ,
c - > x86 , c - > x86_model , c - > x86_mask ) ;
error = request_firmware ( & firmware , name , & microcode_pdev - > dev ) ;
if ( error ) {
pr_debug ( " ucode data file %s load failed \n " , name ) ;
return error ;
}
2008-01-30 13:33:23 +01:00
buf = firmware - > data ;
2006-09-27 01:50:52 -07:00
size = firmware - > size ;
while ( ( offset = get_next_ucode_from_buffer ( & mc , buf , size , offset ) )
> 0 ) {
error = microcode_sanity_check ( mc ) ;
if ( error )
break ;
error = get_maching_microcode ( mc , cpu ) ;
if ( error < 0 )
break ;
/*
* It ' s possible the data file has multiple matching ucode ,
* lets keep searching till the latest version
*/
if ( error = = 1 ) {
apply_microcode ( cpu ) ;
error = 0 ;
}
vfree ( mc ) ;
}
if ( offset > 0 )
vfree ( mc ) ;
if ( offset < 0 )
error = offset ;
release_firmware ( firmware ) ;
return error ;
}
2007-05-09 02:35:11 -07:00
static int apply_microcode_check_cpu ( int cpu )
2007-04-01 23:49:49 -07:00
{
2007-10-19 20:35:04 +02:00
struct cpuinfo_x86 * c = & cpu_data ( cpu ) ;
2007-04-01 23:49:49 -07:00
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
cpumask_t old ;
unsigned int val [ 2 ] ;
int err = 0 ;
2007-05-09 02:35:11 -07:00
/* Check if the microcode is available */
2007-04-01 23:49:49 -07:00
if ( ! uci - > mc )
2007-05-09 02:35:11 -07:00
return 0 ;
2007-04-01 23:49:49 -07:00
old = current - > cpus_allowed ;
set_cpus_allowed ( current , cpumask_of_cpu ( cpu ) ) ;
/* Check if the microcode we have in memory matches the CPU */
if ( c - > x86_vendor ! = X86_VENDOR_INTEL | | c - > x86 < 6 | |
cpu_has ( c , X86_FEATURE_IA64 ) | | uci - > sig ! = cpuid_eax ( 0x00000001 ) )
err = - EINVAL ;
if ( ! err & & ( ( c - > x86_model > = 5 ) | | ( c - > x86 > 6 ) ) ) {
/* get processor flags from MSR 0x17 */
rdmsr ( MSR_IA32_PLATFORM_ID , val [ 0 ] , val [ 1 ] ) ;
if ( uci - > pf ! = ( 1 < < ( ( val [ 1 ] > > 18 ) & 7 ) ) )
err = - EINVAL ;
}
if ( ! err ) {
wrmsr ( MSR_IA32_UCODE_REV , 0 , 0 ) ;
/* see notes above for revision 1.07. Apparent chip bug */
sync_core ( ) ;
/* get the current revision from MSR 0x8B */
rdmsr ( MSR_IA32_UCODE_REV , val [ 0 ] , val [ 1 ] ) ;
if ( uci - > rev ! = val [ 1 ] )
err = - EINVAL ;
}
if ( ! err )
apply_microcode ( cpu ) ;
else
printk ( KERN_ERR " microcode: Could not apply microcode to CPU%d: "
" sig=0x%x, pf=0x%x, rev=0x%x \n " ,
cpu , uci - > sig , uci - > pf , uci - > rev ) ;
set_cpus_allowed ( current , old ) ;
return err ;
}
2007-05-09 02:35:11 -07:00
static void microcode_init_cpu ( int cpu , int resume )
2006-09-27 01:50:52 -07:00
{
cpumask_t old ;
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
old = current - > cpus_allowed ;
set_cpus_allowed ( current , cpumask_of_cpu ( cpu ) ) ;
mutex_lock ( & microcode_mutex ) ;
collect_cpu_info ( cpu ) ;
2007-05-09 02:35:11 -07:00
if ( uci - > valid & & system_state = = SYSTEM_RUNNING & & ! resume )
2006-09-27 01:50:52 -07:00
cpu_request_microcode ( cpu ) ;
mutex_unlock ( & microcode_mutex ) ;
set_cpus_allowed ( current , old ) ;
}
static void microcode_fini_cpu ( int cpu )
{
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
mutex_lock ( & microcode_mutex ) ;
uci - > valid = 0 ;
vfree ( uci - > mc ) ;
uci - > mc = NULL ;
mutex_unlock ( & microcode_mutex ) ;
}
2006-09-27 01:50:53 -07:00
static ssize_t reload_store ( struct sys_device * dev , const char * buf , size_t sz )
{
struct ucode_cpu_info * uci = ucode_cpu_info + dev - > id ;
char * end ;
unsigned long val = simple_strtoul ( buf , & end , 0 ) ;
int err = 0 ;
int cpu = dev - > id ;
if ( end = = buf )
return - EINVAL ;
if ( val = = 1 ) {
cpumask_t old ;
old = current - > cpus_allowed ;
2008-01-25 21:08:02 +01:00
get_online_cpus ( ) ;
2006-09-27 01:50:53 -07:00
set_cpus_allowed ( current , cpumask_of_cpu ( cpu ) ) ;
mutex_lock ( & microcode_mutex ) ;
if ( uci - > valid )
err = cpu_request_microcode ( cpu ) ;
mutex_unlock ( & microcode_mutex ) ;
2008-01-25 21:08:02 +01:00
put_online_cpus ( ) ;
2006-09-27 01:50:53 -07:00
set_cpus_allowed ( current , old ) ;
}
if ( err )
return err ;
return sz ;
}
static ssize_t version_show ( struct sys_device * dev , char * buf )
{
struct ucode_cpu_info * uci = ucode_cpu_info + dev - > id ;
return sprintf ( buf , " 0x%x \n " , uci - > rev ) ;
}
static ssize_t pf_show ( struct sys_device * dev , char * buf )
{
struct ucode_cpu_info * uci = ucode_cpu_info + dev - > id ;
return sprintf ( buf , " 0x%x \n " , uci - > pf ) ;
}
static SYSDEV_ATTR ( reload , 0200 , NULL , reload_store ) ;
static SYSDEV_ATTR ( version , 0400 , version_show , NULL ) ;
static SYSDEV_ATTR ( processor_flags , 0400 , pf_show , NULL ) ;
static struct attribute * mc_default_attrs [ ] = {
& attr_reload . attr ,
& attr_version . attr ,
& attr_processor_flags . attr ,
NULL
} ;
static struct attribute_group mc_attr_group = {
. attrs = mc_default_attrs ,
. name = " microcode " ,
} ;
2007-05-09 02:35:11 -07:00
static int __mc_sysdev_add ( struct sys_device * sys_dev , int resume )
2006-09-27 01:50:53 -07:00
{
2006-10-11 01:21:53 -07:00
int err , cpu = sys_dev - > id ;
2006-09-27 01:50:53 -07:00
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
if ( ! cpu_online ( cpu ) )
return 0 ;
2006-10-11 01:21:53 -07:00
2006-09-27 01:50:53 -07:00
pr_debug ( " Microcode:CPU %d added \n " , cpu ) ;
2007-05-09 02:35:11 -07:00
memset ( uci , 0 , sizeof ( * uci ) ) ;
2006-10-11 01:21:53 -07:00
err = sysfs_create_group ( & sys_dev - > kobj , & mc_attr_group ) ;
if ( err )
return err ;
2006-09-27 01:50:53 -07:00
2007-05-09 02:35:11 -07:00
microcode_init_cpu ( cpu , resume ) ;
2007-04-01 23:49:49 -07:00
2006-09-27 01:50:53 -07:00
return 0 ;
}
2007-05-09 02:35:11 -07:00
static int mc_sysdev_add ( struct sys_device * sys_dev )
{
return __mc_sysdev_add ( sys_dev , 0 ) ;
}
2006-09-27 01:50:53 -07:00
static int mc_sysdev_remove ( struct sys_device * sys_dev )
{
int cpu = sys_dev - > id ;
if ( ! cpu_online ( cpu ) )
return 0 ;
2007-05-09 02:35:11 -07:00
2006-09-27 01:50:53 -07:00
pr_debug ( " Microcode:CPU %d removed \n " , cpu ) ;
2007-05-09 02:35:11 -07:00
microcode_fini_cpu ( cpu ) ;
2006-09-27 01:50:53 -07:00
sysfs_remove_group ( & sys_dev - > kobj , & mc_attr_group ) ;
return 0 ;
}
static int mc_sysdev_resume ( struct sys_device * dev )
{
int cpu = dev - > id ;
if ( ! cpu_online ( cpu ) )
return 0 ;
pr_debug ( " Microcode:CPU %d resumed \n " , cpu ) ;
/* only CPU 0 will apply ucode here */
apply_microcode ( 0 ) ;
return 0 ;
}
static struct sysdev_driver mc_sysdev_driver = {
. add = mc_sysdev_add ,
. remove = mc_sysdev_remove ,
. resume = mc_sysdev_resume ,
} ;
static __cpuinit int
mc_cpu_callback ( struct notifier_block * nb , unsigned long action , void * hcpu )
{
unsigned int cpu = ( unsigned long ) hcpu ;
struct sys_device * sys_dev ;
sys_dev = get_cpu_sysdev ( cpu ) ;
switch ( action ) {
2007-05-09 02:35:11 -07:00
case CPU_UP_CANCELED_FROZEN :
/* The CPU refused to come up during a system resume */
microcode_fini_cpu ( cpu ) ;
break ;
2006-09-27 01:50:53 -07:00
case CPU_ONLINE :
case CPU_DOWN_FAILED :
mc_sysdev_add ( sys_dev ) ;
break ;
2007-05-09 02:35:11 -07:00
case CPU_ONLINE_FROZEN :
/* System-wide resume is in progress, try to apply microcode */
if ( apply_microcode_check_cpu ( cpu ) ) {
/* The application of microcode failed */
microcode_fini_cpu ( cpu ) ;
__mc_sysdev_add ( sys_dev , 1 ) ;
break ;
}
case CPU_DOWN_FAILED_FROZEN :
if ( sysfs_create_group ( & sys_dev - > kobj , & mc_attr_group ) )
printk ( KERN_ERR " Microcode: Failed to create the sysfs "
" group for CPU%d \n " , cpu ) ;
break ;
2006-09-27 01:50:53 -07:00
case CPU_DOWN_PREPARE :
mc_sysdev_remove ( sys_dev ) ;
break ;
2007-05-09 02:35:11 -07:00
case CPU_DOWN_PREPARE_FROZEN :
/* Suspend is in progress, only remove the interface */
sysfs_remove_group ( & sys_dev - > kobj , & mc_attr_group ) ;
break ;
2006-09-27 01:50:53 -07:00
}
return NOTIFY_OK ;
}
2006-12-22 01:10:19 -08:00
static struct notifier_block __cpuinitdata mc_cpu_notifier = {
2006-09-27 01:50:53 -07:00
. notifier_call = mc_cpu_callback ,
} ;
2006-09-27 01:50:51 -07:00
static int __init microcode_init ( void )
{
int error ;
error = microcode_dev_init ( ) ;
if ( error )
return error ;
2006-09-27 01:50:52 -07:00
microcode_pdev = platform_device_register_simple ( " microcode " , - 1 ,
NULL , 0 ) ;
if ( IS_ERR ( microcode_pdev ) ) {
microcode_dev_exit ( ) ;
return PTR_ERR ( microcode_pdev ) ;
}
2006-09-27 01:50:51 -07:00
2008-01-25 21:08:02 +01:00
get_online_cpus ( ) ;
2006-09-27 01:50:53 -07:00
error = sysdev_driver_register ( & cpu_sysdev_class , & mc_sysdev_driver ) ;
2008-01-25 21:08:02 +01:00
put_online_cpus ( ) ;
2006-09-27 01:50:53 -07:00
if ( error ) {
microcode_dev_exit ( ) ;
platform_device_unregister ( microcode_pdev ) ;
return error ;
}
register_hotcpu_notifier ( & mc_cpu_notifier ) ;
2005-04-16 15:20:36 -07:00
printk ( KERN_INFO
2006-12-13 00:35:14 -08:00
" IA-32 Microcode Update Driver: v " MICROCODE_VERSION " <tigran@aivazian.fsnet.co.uk> \n " ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static void __exit microcode_exit ( void )
{
2006-09-27 01:50:51 -07:00
microcode_dev_exit ( ) ;
2006-09-27 01:50:53 -07:00
unregister_hotcpu_notifier ( & mc_cpu_notifier ) ;
2008-01-25 21:08:02 +01:00
get_online_cpus ( ) ;
2006-09-27 01:50:53 -07:00
sysdev_driver_unregister ( & cpu_sysdev_class , & mc_sysdev_driver ) ;
2008-01-25 21:08:02 +01:00
put_online_cpus ( ) ;
2006-09-27 01:50:53 -07:00
2006-09-27 01:50:52 -07:00
platform_device_unregister ( microcode_pdev ) ;
2005-04-16 15:20:36 -07:00
}
module_init ( microcode_init )
module_exit ( microcode_exit )