2005-04-16 15:20:36 -07:00
/*
* blacklist . c
*
* Check to see if the given machine has a known bad ACPI BIOS
* or if the BIOS is too old .
2016-05-03 16:49:01 +08:00
* Check given machine against acpi_rev_dmi_table [ ] .
2005-04-16 15:20:36 -07: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 .
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*/
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/acpi.h>
# include <linux/dmi.h>
2009-07-28 16:45:54 -04:00
# include "internal.h"
2005-08-05 00:44:28 -04:00
enum acpi_blacklist_predicates {
all_versions ,
less_than_or_equal ,
equal ,
greater_than_or_equal ,
2005-04-16 15:20:36 -07:00
} ;
2005-08-05 00: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 00:44:28 -04:00
enum acpi_blacklist_predicates oem_revision_predicate ;
char * reason ;
u32 is_critical_error ;
2005-04-16 15:20:36 -07:00
} ;
2016-05-03 16:49:01 +08:00
static struct dmi_system_id acpi_rev_dmi_table [ ] __initdata ;
2008-01-23 20:50:56 -05:00
2005-04-16 15:20:36 -07: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 00:44:28 -04:00
static struct acpi_blacklist_item acpi_blacklist [ ] __initdata = {
2005-04-16 15:20:36 -07: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 00:44:28 -04:00
" Multiple problems " , 1 } ,
2005-04-16 15:20:36 -07: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 00:44:28 -04:00
" ACPI driver problem " , 1 } ,
2005-04-16 15:20:36 -07: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 00:44:28 -04:00
" Does not use _REG to protect EC OpRegions " , 1 } ,
2005-04-16 15:20:36 -07: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 00:44:28 -04:00
" Incorrect _ADR " , 1 } ,
2005-04-16 15:20:36 -07:00
{ " " }
} ;
2005-08-05 00:44:28 -04:00
int __init acpi_blacklisted ( void )
2005-04-16 15:20:36 -07:00
{
int i = 0 ;
int blacklisted = 0 ;
2007-02-02 19:48:22 +03:00
struct acpi_table_header table_header ;
2005-04-16 15:20:36 -07:00
2005-08-05 00: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-16 15:20:36 -07: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-16 15:20:36 -07:00
i + + ;
continue ;
}
2005-08-05 00: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 00:44:28 -04:00
8 ) ) {
2005-04-16 15:20:36 -07:00
i + + ;
continue ;
}
if ( ( acpi_blacklist [ i ] . oem_revision_predicate = = all_versions )
2005-08-05 00: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 00: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 00:44:28 -04:00
acpi_blacklist [ i ] . oem_revision )
2005-04-16 15:20:36 -07:00
| | ( acpi_blacklist [ i ] . oem_revision_predicate = = equal
2007-02-02 19:48:22 +03:00
& & table_header . oem_revision = =
2005-08-05 00: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-16 15:20:36 -07:00
blacklisted = acpi_blacklist [ i ] . is_critical_error ;
break ;
2005-08-05 00:44:28 -04:00
} else {
2005-04-16 15:20:36 -07:00
i + + ;
}
}
2016-05-03 16:49:01 +08:00
( void ) early_acpi_osi_init ( ) ;
dmi_check_system ( acpi_rev_dmi_table ) ;
2008-01-23 20:50:56 -05:00
2005-04-16 15:20:36 -07:00
return blacklisted ;
}
2008-01-23 20:50:56 -05:00
# ifdef CONFIG_DMI
2015-07-03 01:06:00 +02:00
# ifdef CONFIG_ACPI_REV_OVERRIDE_POSSIBLE
static int __init dmi_enable_rev_override ( const struct dmi_system_id * d )
{
printk ( KERN_NOTICE PREFIX " DMI detected: %s (force ACPI _REV to 5) \n " ,
d - > ident ) ;
acpi_rev_override_setup ( NULL ) ;
return 0 ;
}
# endif
2008-01-23 21:19:27 -05:00
2016-05-03 16:49:01 +08:00
static struct dmi_system_id acpi_rev_dmi_table [ ] __initdata = {
2015-07-03 01:06:00 +02:00
# ifdef CONFIG_ACPI_REV_OVERRIDE_POSSIBLE
/*
* DELL XPS 13 ( 2015 ) switches sound between HDA and I2S
* depending on the ACPI _REV callback . If userspace supports
* I2S sufficiently ( or if you do not care about sound ) , you
* can safely disable this quirk .
*/
{
. callback = dmi_enable_rev_override ,
. ident = " DELL XPS 13 (2015) " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " XPS 13 9343 " ) ,
} ,
} ,
# endif
2008-01-23 20:50:56 -05:00
{ }
} ;
# endif /* CONFIG_DMI */