2005-04-17 02:20:36 +04:00
/*
* blacklist . c
*
* Check to see if the given machine has a known bad ACPI BIOS
* or if the BIOS is too old .
2008-01-24 04:50:56 +03:00
* Check given machine against acpi_osi_dmi_table [ ] .
2005-04-17 02:20:36 +04:00
*
* Copyright ( C ) 2004 Len Brown < len . brown @ intel . com >
* Copyright ( C ) 2002 Andy Grover < andrew . grover @ intel . com >
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*
* 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 .
*
* 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 . ,
* 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA .
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/acpi.h>
# include <acpi/acpi_bus.h>
# include <linux/dmi.h>
2005-08-05 08:44:28 +04:00
enum acpi_blacklist_predicates {
all_versions ,
less_than_or_equal ,
equal ,
greater_than_or_equal ,
2005-04-17 02:20:36 +04:00
} ;
2005-08-05 08:44:28 +04:00
struct acpi_blacklist_item {
char oem_id [ 7 ] ;
char oem_table_id [ 9 ] ;
u32 oem_revision ;
2007-02-02 19:48:19 +03:00
char * table ;
2005-08-05 08:44:28 +04:00
enum acpi_blacklist_predicates oem_revision_predicate ;
char * reason ;
u32 is_critical_error ;
2005-04-17 02:20:36 +04:00
} ;
2008-01-24 04:50:56 +03:00
static struct dmi_system_id acpi_osi_dmi_table [ ] __initdata ;
2005-04-17 02:20:36 +04:00
/*
* POLICY : If * anything * doesn ' t work , put it on the blacklist .
* If they are critical errors , mark it critical , and abort driver load .
*/
2005-08-05 08:44:28 +04:00
static struct acpi_blacklist_item acpi_blacklist [ ] __initdata = {
2005-04-17 02:20:36 +04:00
/* Compaq Presario 1700 */
2007-02-02 19:48:19 +03:00
{ " PTLTD " , " DSDT " , 0x06040000 , ACPI_SIG_DSDT , less_than_or_equal ,
2005-08-05 08:44:28 +04:00
" Multiple problems " , 1 } ,
2005-04-17 02:20:36 +04:00
/* Sony FX120, FX140, FX150? */
2007-02-02 19:48:19 +03:00
{ " SONY " , " U0 " , 0x20010313 , ACPI_SIG_DSDT , less_than_or_equal ,
2005-08-05 08:44:28 +04:00
" ACPI driver problem " , 1 } ,
2005-04-17 02:20:36 +04:00
/* Compaq Presario 800, Insyde BIOS */
2007-02-02 19:48:19 +03:00
{ " INT440 " , " SYSFexxx " , 0x00001001 , ACPI_SIG_DSDT , less_than_or_equal ,
2005-08-05 08:44:28 +04:00
" Does not use _REG to protect EC OpRegions " , 1 } ,
2005-04-17 02:20:36 +04:00
/* IBM 600E - _ADR should return 7, but it returns 1 */
2007-02-02 19:48:19 +03:00
{ " IBM " , " TP600E " , 0x00000105 , ACPI_SIG_DSDT , less_than_or_equal ,
2005-08-05 08:44:28 +04:00
" Incorrect _ADR " , 1 } ,
2005-04-17 02:20:36 +04:00
{ " " }
} ;
# if CONFIG_ACPI_BLACKLIST_YEAR
2005-08-05 08:44:28 +04:00
static int __init blacklist_by_year ( void )
2005-04-17 02:20:36 +04:00
{
2009-08-16 16:02:36 +04:00
int year ;
2006-03-25 18:30:19 +03:00
/* Doesn't exist? Likely an old system */
2009-08-16 16:02:36 +04:00
if ( ! dmi_get_date ( DMI_BIOS_DATE , & year , NULL , NULL ) ) {
2007-03-10 05:19:05 +03:00
printk ( KERN_ERR PREFIX " no DMI BIOS year, "
" acpi=force is required to enable ACPI \n " ) ;
2006-03-25 18:30:19 +03:00
return 1 ;
2007-03-10 05:19:05 +03:00
}
2006-03-25 18:30:19 +03:00
/* 0? Likely a buggy new BIOS */
2007-03-10 05:19:05 +03:00
if ( year = = 0 ) {
printk ( KERN_ERR PREFIX " DMI BIOS year==0, "
" assuming ACPI-capable machine \n " ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
2007-03-10 05:19:05 +03:00
}
2005-04-17 02:20:36 +04:00
if ( year < CONFIG_ACPI_BLACKLIST_YEAR ) {
2005-08-05 08:44:28 +04:00
printk ( KERN_ERR PREFIX " BIOS age (%d) fails cutoff (%d), "
" acpi=force is required to enable ACPI \n " ,
year , CONFIG_ACPI_BLACKLIST_YEAR ) ;
2005-04-17 02:20:36 +04:00
return 1 ;
}
return 0 ;
}
# else
2005-08-05 08:44:28 +04:00
static inline int blacklist_by_year ( void )
{
return 0 ;
}
2005-04-17 02:20:36 +04:00
# endif
2005-08-05 08:44:28 +04:00
int __init acpi_blacklisted ( void )
2005-04-17 02:20:36 +04:00
{
int i = 0 ;
int blacklisted = 0 ;
2007-02-02 19:48:22 +03:00
struct acpi_table_header table_header ;
2005-04-17 02:20:36 +04:00
2005-08-05 08:44:28 +04:00
while ( acpi_blacklist [ i ] . oem_id [ 0 ] ! = ' \0 ' ) {
2007-02-02 19:48:19 +03:00
if ( acpi_get_table_header ( acpi_blacklist [ i ] . table , 0 , & table_header ) ) {
2005-04-17 02:20:36 +04:00
i + + ;
continue ;
}
2007-02-02 19:48:22 +03:00
if ( strncmp ( acpi_blacklist [ i ] . oem_id , table_header . oem_id , 6 ) ) {
2005-04-17 02:20:36 +04:00
i + + ;
continue ;
}
2005-08-05 08:44:28 +04:00
if ( strncmp
2007-02-02 19:48:22 +03:00
( acpi_blacklist [ i ] . oem_table_id , table_header . oem_table_id ,
2005-08-05 08:44:28 +04:00
8 ) ) {
2005-04-17 02:20:36 +04:00
i + + ;
continue ;
}
if ( ( acpi_blacklist [ i ] . oem_revision_predicate = = all_versions )
2005-08-05 08:44:28 +04:00
| | ( acpi_blacklist [ i ] . oem_revision_predicate = =
less_than_or_equal
2007-02-02 19:48:22 +03:00
& & table_header . oem_revision < =
2005-08-05 08:44:28 +04:00
acpi_blacklist [ i ] . oem_revision )
| | ( acpi_blacklist [ i ] . oem_revision_predicate = =
greater_than_or_equal
2007-02-02 19:48:22 +03:00
& & table_header . oem_revision > =
2005-08-05 08:44:28 +04:00
acpi_blacklist [ i ] . oem_revision )
2005-04-17 02:20:36 +04:00
| | ( acpi_blacklist [ i ] . oem_revision_predicate = = equal
2007-02-02 19:48:22 +03:00
& & table_header . oem_revision = =
2005-08-05 08:44:28 +04:00
acpi_blacklist [ i ] . oem_revision ) ) {
printk ( KERN_ERR PREFIX
" Vendor \" %6.6s \" System \" %8.8s \" "
" Revision 0x%x has a known ACPI BIOS problem. \n " ,
acpi_blacklist [ i ] . oem_id ,
acpi_blacklist [ i ] . oem_table_id ,
acpi_blacklist [ i ] . oem_revision ) ;
printk ( KERN_ERR PREFIX
" Reason: %s. This is a %s error \n " ,
acpi_blacklist [ i ] . reason ,
( acpi_blacklist [ i ] .
is_critical_error ? " non-recoverable " :
" recoverable " ) ) ;
2005-04-17 02:20:36 +04:00
blacklisted = acpi_blacklist [ i ] . is_critical_error ;
break ;
2005-08-05 08:44:28 +04:00
} else {
2005-04-17 02:20:36 +04:00
i + + ;
}
}
blacklisted + = blacklist_by_year ( ) ;
2008-01-24 04:50:56 +03:00
dmi_check_system ( acpi_osi_dmi_table ) ;
2005-04-17 02:20:36 +04:00
return blacklisted ;
}
2008-01-24 04:50:56 +03:00
# ifdef CONFIG_DMI
2008-01-24 04:56:18 +03:00
static int __init dmi_enable_osi_linux ( const struct dmi_system_id * d )
{
acpi_dmi_osi_linux ( 1 , d ) ; /* enable */
return 0 ;
}
2008-02-14 07:13:25 +03:00
static int __init dmi_disable_osi_vista ( const struct dmi_system_id * d )
{
printk ( KERN_NOTICE PREFIX " DMI detected: %s \n " , d - > ident ) ;
acpi_osi_setup ( " !Windows 2006 " ) ;
return 0 ;
}
2008-01-24 05:19:27 +03:00
2008-01-24 04:50:56 +03:00
static struct dmi_system_id acpi_osi_dmi_table [ ] __initdata = {
2008-02-14 07:13:25 +03:00
{
. callback = dmi_disable_osi_vista ,
. ident = " Fujitsu Siemens " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " FUJITSU SIEMENS " ) ,
2008-11-08 09:21:10 +03:00
DMI_MATCH ( DMI_PRODUCT_NAME , " ESPRIMO Mobile V5505 " ) ,
2008-02-14 07:13:25 +03:00
} ,
} ,
2009-06-22 07:31:17 +04:00
{
. callback = dmi_disable_osi_vista ,
. ident = " Sony VGN-NS10J_S " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Sony Corporation " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " VGN-NS10J_S " ) ,
} ,
} ,
{
. callback = dmi_disable_osi_vista ,
. ident = " Sony VGN-SR290J " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Sony Corporation " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Sony VGN-SR290J " ) ,
} ,
} ,
2008-11-08 09:21:10 +03:00
2008-01-24 05:19:27 +03:00
/*
2008-11-08 09:21:10 +03:00
* BIOS invocation of _OSI ( Linux ) is almost always a BIOS bug .
* Linux ignores it , except for the machines enumerated below .
2008-01-24 05:19:27 +03:00
*/
2008-11-08 09:21:10 +03:00
2008-01-24 05:19:27 +03:00
/*
* Lenovo has a mix of systems OSI ( Linux ) situations
* and thus we can not wildcard the vendor .
*
2008-01-24 04:56:18 +03:00
* _OSI ( Linux ) helps sound
* DMI_MATCH ( DMI_PRODUCT_VERSION , " ThinkPad R61 " ) ,
* DMI_MATCH ( DMI_PRODUCT_VERSION , " ThinkPad T61 " ) ,
2008-02-11 05:29:56 +03:00
* _OSI ( Linux ) has Linux specific hooks
* DMI_MATCH ( DMI_PRODUCT_VERSION , " ThinkPad X61 " ) ,
2008-01-24 05:19:27 +03:00
* _OSI ( Linux ) is a NOP :
* DMI_MATCH ( DMI_PRODUCT_VERSION , " 3000 N100 " ) ,
2008-02-11 05:29:56 +03:00
* DMI_MATCH ( DMI_PRODUCT_VERSION , " LENOVO3000 V100 " ) ,
2008-01-24 04:56:18 +03:00
*/
{
. callback = dmi_enable_osi_linux ,
. ident = " Lenovo ThinkPad R61 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " LENOVO " ) ,
DMI_MATCH ( DMI_PRODUCT_VERSION , " ThinkPad R61 " ) ,
} ,
} ,
{
. callback = dmi_enable_osi_linux ,
. ident = " Lenovo ThinkPad T61 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " LENOVO " ) ,
DMI_MATCH ( DMI_PRODUCT_VERSION , " ThinkPad T61 " ) ,
} ,
} ,
2008-01-24 05:19:27 +03:00
{
2008-02-11 05:29:56 +03:00
. callback = dmi_enable_osi_linux ,
2008-02-08 00:23:00 +03:00
. ident = " Lenovo ThinkPad X61 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " LENOVO " ) ,
DMI_MATCH ( DMI_PRODUCT_VERSION , " ThinkPad X61 " ) ,
} ,
} ,
2008-01-24 04:50:56 +03:00
{ }
} ;
# endif /* CONFIG_DMI */