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
2008-06-10 13:15:12 +02:00
* belonging to IA - 32 family - PentiumPro , Pentium II ,
2005-04-16 15:20:36 -07:00
* Pentium III , Xeon , Pentium 4 , etc .
*
2008-06-10 13:15:12 +02:00
* Reference : Section 8.11 of Volume 3 a , IA - 32 Intel ? Architecture
* Software Developer ' s Manual
* Order Number 253668 or free download from :
*
2010-10-16 10:36:23 -07:00
* http : //developer.intel.com/Assets/PDF/manual/253668.pdf
2005-04-16 15:20:36 -07:00
*
* 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 > ,
2008-07-29 17:41:05 +02:00
* Serialize updates as required on HT processors due to
* speculative nature of implementation .
2005-04-16 15:20:36 -07:00
* 1.11 22 Mar 2002 Tigran Aivazian < tigran @ veritas . com >
* Fix the panic when writing zero - length microcode chunk .
2008-06-10 13:15:12 +02:00
* 1.12 29 Sep 2003 Nitin Kamble < nitin . a . kamble @ intel . com > ,
2005-04-16 15:20:36 -07:00
* 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
2008-06-10 13:15:12 +02:00
* because we no longer hold a copy of applied microcode
2005-04-16 15:20:36 -07:00
* 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 .
*/
2009-12-08 22:30:50 -08:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2009-03-11 11:19:46 +01:00
# include <linux/firmware.h>
# include <linux/uaccess.h>
# include <linux/kernel.h>
# include <linux/module.h>
2009-05-11 23:48:27 +02:00
# include <linux/vmalloc.h>
2005-04-16 15:20:36 -07:00
2008-07-28 18:44:13 +02:00
# include <asm/microcode.h>
2009-03-11 11:19:46 +01:00
# include <asm/processor.h>
# include <asm/msr.h>
2005-04-16 15:20:36 -07:00
2008-07-28 18:44:17 +02:00
MODULE_DESCRIPTION ( " 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 " ) ;
2008-09-23 12:08:44 +02:00
struct microcode_header_intel {
unsigned int hdrver ;
unsigned int rev ;
unsigned int date ;
unsigned int sig ;
unsigned int cksum ;
unsigned int ldrver ;
unsigned int pf ;
unsigned int datasize ;
unsigned int totalsize ;
unsigned int reserved [ 3 ] ;
} ;
struct microcode_intel {
struct microcode_header_intel hdr ;
unsigned int bits [ 0 ] ;
} ;
/* microcode format is extended from prescott processors */
struct extended_signature {
unsigned int sig ;
unsigned int pf ;
unsigned int cksum ;
} ;
struct extended_sigtable {
unsigned int count ;
unsigned int cksum ;
unsigned int reserved [ 3 ] ;
struct extended_signature sigs [ 0 ] ;
} ;
2009-03-11 11:19:46 +01:00
# define DEFAULT_UCODE_DATASIZE (2000)
2008-07-29 17:41:05 +02:00
# define MC_HEADER_SIZE (sizeof(struct microcode_header_intel))
# define DEFAULT_UCODE_TOTALSIZE (DEFAULT_UCODE_DATASIZE + MC_HEADER_SIZE)
# define EXT_HEADER_SIZE (sizeof(struct extended_sigtable))
# define EXT_SIGNATURE_SIZE (sizeof(struct extended_signature))
2008-07-28 18:44:17 +02:00
# define DWSIZE (sizeof(u32))
2009-03-11 11:19:46 +01:00
2005-04-16 15:20:36 -07:00
# define get_totalsize(mc) \
2008-07-28 18:44:18 +02:00
( ( ( struct microcode_intel * ) mc ) - > hdr . totalsize ? \
( ( struct microcode_intel * ) mc ) - > hdr . totalsize : \
DEFAULT_UCODE_TOTALSIZE )
2005-04-16 15:20:36 -07:00
# define get_datasize(mc) \
2008-07-28 18:44:18 +02:00
( ( ( struct microcode_intel * ) mc ) - > hdr . datasize ? \
( ( struct microcode_intel * ) mc ) - > hdr . datasize : DEFAULT_UCODE_DATASIZE )
2005-04-16 15:20:36 -07:00
# 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)
2008-08-20 00:22:26 +02:00
static int collect_cpu_info ( int cpu_num , struct cpu_signature * csig )
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
unsigned int val [ 2 ] ;
2008-08-20 00:22:26 +02:00
memset ( csig , 0 , sizeof ( * csig ) ) ;
2005-04-16 15:20:36 -07:00
2008-08-20 00:22:26 +02:00
csig - > sig = cpuid_eax ( 0x00000001 ) ;
2006-09-27 01:50:51 -07:00
if ( ( c - > x86_model > = 5 ) | | ( c - > x86 > 6 ) ) {
/* get processor flags from MSR 0x17 */
rdmsr ( MSR_IA32_PLATFORM_ID , val [ 0 ] , val [ 1 ] ) ;
2008-08-20 00:22:26 +02:00
csig - > pf = 1 < < ( ( val [ 1 ] > > 18 ) & 7 ) ;
2005-04-16 15:20:36 -07:00
}
2011-10-12 17:46:33 -07:00
csig - > rev = c - > microcode ;
2009-12-08 22:30:50 -08:00
pr_info ( " CPU%d sig=0x%x, pf=0x%x, revision=0x%x \n " ,
cpu_num , csig - > sig , csig - > pf , csig - > rev ) ;
2008-08-20 00:22:26 +02:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2008-09-11 23:27:52 +02:00
static inline int update_match_cpu ( struct cpu_signature * csig , int sig , int pf )
2005-04-16 15:20:36 -07:00
{
2008-09-11 23:27:52 +02:00
return ( ! sigmatch ( sig , csig - > sig , pf , csig - > pf ) ) ? 0 : 1 ;
}
2005-04-16 15:20:36 -07:00
2009-01-12 14:44:29 +05:30
static inline int
2009-03-11 11:19:46 +01:00
update_match_revision ( struct microcode_header_intel * mc_header , int rev )
2008-09-11 23:27:52 +02:00
{
return ( mc_header - > rev < = rev ) ? 0 : 1 ;
2005-04-16 15:20:36 -07:00
}
2008-07-28 18:44:21 +02:00
static int microcode_sanity_check ( void * mc )
2005-04-16 15:20:36 -07:00
{
2009-03-11 11:19:46 +01:00
unsigned long total_size , data_size , ext_table_size ;
2008-07-28 18:44:18 +02:00
struct microcode_header_intel * mc_header = mc ;
2006-09-27 01:50:51 -07:00
struct extended_sigtable * ext_header = NULL ;
int sum , orig_sum , ext_sigcount = 0 , i ;
2009-03-11 11:19:46 +01:00
struct extended_signature * ext_sig ;
2006-09-27 01:50:51 -07:00
total_size = get_totalsize ( mc_header ) ;
data_size = get_datasize ( mc_header ) ;
2009-03-11 11:19:46 +01:00
2006-09-27 01:50:54 -07:00
if ( data_size + MC_HEADER_SIZE > total_size ) {
2009-12-08 22:30:50 -08:00
pr_err ( " error! Bad data size in microcode data file \n " ) ;
2006-09-27 01:50:51 -07:00
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 ) {
2009-12-08 22:30:50 -08:00
pr_err ( " error! Unknown microcode update format \n " ) ;
2006-09-27 01:50:51 -07:00
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 ) ) {
2009-12-08 22:30:50 -08:00
pr_err ( " error! Small exttable size in microcode data file \n " ) ;
2006-09-27 01:50:51 -07:00
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 ) ) {
2009-12-08 22:30:50 -08:00
pr_err ( " error! Bad exttable size in microcode data file \n " ) ;
2006-09-27 01:50:51 -07:00
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 ) {
2009-12-08 22:30:50 -08:00
pr_warning ( " aborting, bad extended signature table checksum \n " ) ;
2006-09-27 01:50:51 -07:00
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 ) {
2009-12-08 22:30:50 -08:00
pr_err ( " aborting, bad checksum \n " ) ;
2006-09-27 01:50:51 -07:00
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 ) {
2009-12-08 22:30:50 -08:00
pr_err ( " aborting, bad checksum \n " ) ;
2006-09-27 01:50:51 -07:00
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
*/
2008-09-11 23:27:52 +02:00
static int
get_matching_microcode ( struct cpu_signature * cpu_sig , void * mc , int rev )
2006-09-27 01:50:51 -07:00
{
2008-07-28 18:44:18 +02:00
struct microcode_header_intel * mc_header = mc ;
2006-09-27 01:50:51 -07:00
struct extended_sigtable * ext_header ;
unsigned long total_size = get_totalsize ( mc_header ) ;
int ext_sigcount , i ;
struct extended_signature * ext_sig ;
2008-09-11 23:27:52 +02:00
if ( ! update_match_revision ( mc_header , rev ) )
return 0 ;
if ( update_match_cpu ( cpu_sig , mc_header - > sig , mc_header - > pf ) )
return 1 ;
2006-09-27 01:50:51 -07:00
2008-09-11 23:27:52 +02:00
/* Look for ext. headers: */
2006-09-27 01:50:51 -07:00
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 ;
2008-09-11 23:27:52 +02:00
2006-09-27 01:50:51 -07:00
for ( i = 0 ; i < ext_sigcount ; i + + ) {
2008-09-11 23:27:52 +02:00
if ( update_match_cpu ( cpu_sig , ext_sig - > sig , ext_sig - > pf ) )
return 1 ;
2006-09-27 01:50:51 -07:00
ext_sig + + ;
}
return 0 ;
2005-04-16 15:20:36 -07:00
}
2009-05-11 23:48:27 +02:00
static int apply_microcode ( int cpu )
2005-04-16 15:20:36 -07:00
{
2009-03-11 11:19:46 +01:00
struct microcode_intel * mc_intel ;
struct ucode_cpu_info * uci ;
2005-04-16 15:20:36 -07:00
unsigned int val [ 2 ] ;
2011-10-12 17:46:33 -07:00
int cpu_num = raw_smp_processor_id ( ) ;
struct cpuinfo_x86 * c = & cpu_data ( cpu_num ) ;
2009-03-11 11:19:46 +01:00
uci = ucode_cpu_info + cpu ;
mc_intel = uci - > mc ;
2005-04-16 15:20:36 -07:00
2006-09-27 01:50:51 -07:00
/* We should bind the task to the CPU */
BUG_ON ( cpu_num ! = cpu ) ;
2008-09-23 12:08:44 +02:00
if ( mc_intel = = NULL )
2009-05-11 23:48:27 +02:00
return 0 ;
2005-04-16 15:20:36 -07:00
/* write microcode via MSR 0x79 */
wrmsr ( MSR_IA32_UCODE_WRITE ,
2008-09-23 12:08:44 +02:00
( unsigned long ) mc_intel - > bits ,
( unsigned long ) mc_intel - > bits > > 16 > > 16 ) ;
2005-04-16 15:20:36 -07:00
wrmsr ( MSR_IA32_UCODE_REV , 0 , 0 ) ;
2011-10-12 17:46:33 -07:00
/* As documented in the SDM: Do a CPUID 1 here */
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 ] ) ;
2008-09-23 12:08:44 +02:00
if ( val [ 1 ] ! = mc_intel - > hdr . rev ) {
2009-12-08 22:30:50 -08:00
pr_err ( " CPU%d update to revision 0x%x failed \n " ,
cpu_num , mc_intel - > hdr . rev ) ;
2009-05-11 23:48:27 +02:00
return - 1 ;
2006-09-27 01:50:51 -07:00
}
2010-02-06 18:47:17 +01:00
pr_info ( " CPU%d updated to revision 0x%x, date = %04x-%02x-%02x \n " ,
2009-05-11 23:48:27 +02:00
cpu_num , val [ 1 ] ,
2008-09-23 12:08:44 +02:00
mc_intel - > hdr . date & 0xffff ,
mc_intel - > hdr . date > > 24 ,
( mc_intel - > hdr . date > > 16 ) & 0xff ) ;
2009-03-11 11:19:46 +01:00
2008-08-20 00:22:26 +02:00
uci - > cpu_sig . rev = val [ 1 ] ;
2011-10-12 17:46:33 -07:00
c - > microcode = val [ 1 ] ;
2009-05-11 23:48:27 +02:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2009-05-11 23:48:27 +02:00
static enum ucode_state generic_load_microcode ( int cpu , void * data , size_t size ,
int ( * get_ucode_data ) ( void * , const void * , size_t ) )
2006-09-27 01:50:51 -07:00
{
2008-09-11 23:27:52 +02:00
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
2010-03-05 11:42:03 -06:00
u8 * ucode_ptr = data , * new_mc = NULL , * mc = NULL ;
2008-09-11 23:27:52 +02:00
int new_rev = uci - > cpu_sig . rev ;
unsigned int leftover = size ;
2009-05-11 23:48:27 +02:00
enum ucode_state state = UCODE_OK ;
2010-03-05 11:42:03 -06:00
unsigned int curr_mc_size = 0 ;
2006-09-27 01:50:51 -07:00
2008-09-11 23:27:52 +02:00
while ( leftover ) {
struct microcode_header_intel mc_header ;
unsigned int mc_size ;
2006-09-27 01:50:51 -07:00
2008-09-11 23:27:52 +02:00
if ( get_ucode_data ( & mc_header , ucode_ptr , sizeof ( mc_header ) ) )
break ;
2006-09-27 01:50:52 -07:00
2008-09-11 23:27:52 +02:00
mc_size = get_totalsize ( & mc_header ) ;
if ( ! mc_size | | mc_size > leftover ) {
2009-12-08 22:30:50 -08:00
pr_err ( " error! Bad data in microcode data file \n " ) ;
2008-09-11 23:27:52 +02:00
break ;
}
2006-09-27 01:50:52 -07:00
2010-03-05 11:42:03 -06:00
/* For performance reasons, reuse mc area when possible */
if ( ! mc | | mc_size > curr_mc_size ) {
2010-12-25 19:57:41 +01:00
vfree ( mc ) ;
2010-03-05 11:42:03 -06:00
mc = vmalloc ( mc_size ) ;
if ( ! mc )
break ;
curr_mc_size = mc_size ;
}
2008-09-11 23:27:52 +02:00
if ( get_ucode_data ( mc , ucode_ptr , mc_size ) | |
microcode_sanity_check ( mc ) < 0 ) {
break ;
}
if ( get_matching_microcode ( & uci - > cpu_sig , mc , new_rev ) ) {
2010-12-25 19:57:41 +01:00
vfree ( new_mc ) ;
2008-09-11 23:27:52 +02:00
new_rev = mc_header . rev ;
new_mc = mc ;
2010-03-05 11:42:03 -06:00
mc = NULL ; /* trigger new vmalloc */
}
2008-09-11 23:27:52 +02:00
ucode_ptr + = mc_size ;
leftover - = mc_size ;
2006-09-27 01:50:52 -07:00
}
2010-12-25 19:57:41 +01:00
vfree ( mc ) ;
2010-03-05 11:42:03 -06:00
2009-05-11 23:48:27 +02:00
if ( leftover ) {
2010-12-25 19:57:41 +01:00
vfree ( new_mc ) ;
2009-05-11 23:48:27 +02:00
state = UCODE_ERROR ;
2009-03-11 11:19:46 +01:00
goto out ;
2009-05-11 23:48:27 +02:00
}
2009-03-11 11:19:46 +01:00
2009-05-11 23:48:27 +02:00
if ( ! new_mc ) {
state = UCODE_NFOUND ;
2009-03-11 11:19:46 +01:00
goto out ;
2006-09-27 01:50:52 -07:00
}
2008-09-11 23:27:52 +02:00
2010-12-25 19:57:41 +01:00
vfree ( uci - > mc ) ;
2009-03-11 11:19:46 +01:00
uci - > mc = ( struct microcode_intel * ) new_mc ;
2009-12-08 22:30:50 -08:00
pr_debug ( " CPU%d found a matching microcode update with version 0x%x (current=0x%x) \n " ,
cpu , new_rev , uci - > cpu_sig . rev ) ;
2009-05-11 23:48:27 +02:00
out :
return state ;
2006-09-27 01:50:52 -07:00
}
2008-09-11 23:27:52 +02:00
static int get_ucode_fw ( void * to , const void * from , size_t n )
{
memcpy ( to , from , n ) ;
return 0 ;
}
2006-09-27 01:50:52 -07:00
2009-05-11 23:48:27 +02:00
static enum ucode_state request_microcode_fw ( int cpu , struct device * device )
2006-09-27 01:50:52 -07:00
{
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 ;
2009-05-11 23:48:27 +02:00
enum ucode_state ret ;
2006-09-27 01:50:52 -07:00
2008-07-28 18:44:17 +02:00
sprintf ( name , " intel-ucode/%02x-%02x-%02x " ,
2006-09-27 01:50:52 -07:00
c - > x86 , c - > x86_model , c - > x86_mask ) ;
2009-05-11 23:48:27 +02:00
if ( request_firmware ( & firmware , name , device ) ) {
2009-12-08 22:30:50 -08:00
pr_debug ( " data file %s load failed \n " , name ) ;
2009-05-11 23:48:27 +02:00
return UCODE_NFOUND ;
2006-09-27 01:50:52 -07:00
}
2008-09-11 23:27:52 +02:00
2009-01-12 14:44:29 +05:30
ret = generic_load_microcode ( cpu , ( void * ) firmware - > data ,
firmware - > size , & get_ucode_fw ) ;
2008-09-11 23:27:52 +02:00
2006-09-27 01:50:52 -07:00
release_firmware ( firmware ) ;
2008-09-11 23:27:52 +02:00
return ret ;
}
static int get_ucode_user ( void * to , const void * from , size_t n )
{
return copy_from_user ( to , from , n ) ;
}
2009-05-11 23:48:27 +02:00
static enum ucode_state
request_microcode_user ( int cpu , const void __user * buf , size_t size )
2008-09-11 23:27:52 +02:00
{
2009-01-12 14:44:29 +05:30
return generic_load_microcode ( cpu , ( void * ) buf , size , & get_ucode_user ) ;
2006-09-27 01:50:52 -07:00
}
2008-07-28 18:44:21 +02:00
static void microcode_fini_cpu ( int cpu )
2006-09-27 01:50:52 -07:00
{
struct ucode_cpu_info * uci = ucode_cpu_info + cpu ;
2008-09-23 12:08:44 +02:00
vfree ( uci - > mc ) ;
uci - > mc = NULL ;
2006-09-27 01:50:52 -07:00
}
2008-07-28 18:44:21 +02:00
2008-11-23 20:49:52 +01:00
static struct microcode_ops microcode_intel_ops = {
2008-09-11 23:27:52 +02:00
. request_microcode_user = request_microcode_user ,
. request_microcode_fw = request_microcode_fw ,
2008-07-28 18:44:21 +02:00
. collect_cpu_info = collect_cpu_info ,
. apply_microcode = apply_microcode ,
. microcode_fini_cpu = microcode_fini_cpu ,
} ;
2008-09-23 12:08:44 +02:00
struct microcode_ops * __init init_intel_microcode ( void )
2008-07-28 18:44:21 +02:00
{
2012-04-16 14:12:00 +05:30
struct cpuinfo_x86 * c = & cpu_data ( 0 ) ;
if ( c - > x86_vendor ! = X86_VENDOR_INTEL | | c - > x86 < 6 | |
cpu_has ( c , X86_FEATURE_IA64 ) ) {
pr_err ( " Intel CPU family 0x%x not supported \n " , c - > x86 ) ;
return NULL ;
}
2008-09-23 12:08:44 +02:00
return & microcode_intel_ops ;
2008-07-28 18:44:21 +02:00
}