2005-04-16 15:20:36 -07:00
/*
* Implement ' Simple Boot Flag Specification 2.0 '
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/string.h>
# include <linux/slab.h>
# include <linux/spinlock.h>
# include <linux/acpi.h>
# include <asm/io.h>
# include <linux/mc146818rtc.h>
# define SBF_RESERVED (0x78)
# define SBF_PNPOS (1<<0)
# define SBF_BOOTING (1<<1)
# define SBF_DIAG (1<<2)
# define SBF_PARITY (1<<7)
int sbf_port __initdata = - 1 ; /* set via acpi_boot_init() */
static int __init parity ( u8 v )
{
int x = 0 ;
int i ;
2008-01-30 13:32:31 +01:00
for ( i = 0 ; i < 8 ; i + + ) {
x ^ = ( v & 1 ) ;
v > > = 1 ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:32:31 +01:00
2005-04-16 15:20:36 -07:00
return x ;
}
static void __init sbf_write ( u8 v )
{
unsigned long flags ;
2008-01-30 13:32:31 +01:00
if ( sbf_port ! = - 1 ) {
2005-04-16 15:20:36 -07:00
v & = ~ SBF_PARITY ;
2008-01-30 13:32:31 +01:00
if ( ! parity ( v ) )
v | = SBF_PARITY ;
2005-04-16 15:20:36 -07:00
2008-01-30 13:32:31 +01:00
printk ( KERN_INFO " Simple Boot Flag at 0x%x set to 0x%x \n " ,
sbf_port , v ) ;
2005-04-16 15:20:36 -07:00
spin_lock_irqsave ( & rtc_lock , flags ) ;
CMOS_WRITE ( v , sbf_port ) ;
spin_unlock_irqrestore ( & rtc_lock , flags ) ;
}
}
static u8 __init sbf_read ( void )
{
unsigned long flags ;
2008-01-30 13:32:31 +01:00
u8 v ;
if ( sbf_port = = - 1 )
2005-04-16 15:20:36 -07:00
return 0 ;
2008-01-30 13:32:31 +01:00
2005-04-16 15:20:36 -07:00
spin_lock_irqsave ( & rtc_lock , flags ) ;
v = CMOS_READ ( sbf_port ) ;
spin_unlock_irqrestore ( & rtc_lock , flags ) ;
2008-01-30 13:32:31 +01:00
2005-04-16 15:20:36 -07:00
return v ;
}
static int __init sbf_value_valid ( u8 v )
{
2008-01-30 13:32:31 +01:00
if ( v & SBF_RESERVED ) /* Reserved bits */
2005-04-16 15:20:36 -07:00
return 0 ;
2008-01-30 13:32:31 +01:00
if ( ! parity ( v ) )
2005-04-16 15:20:36 -07:00
return 0 ;
2008-01-30 13:32:31 +01:00
2005-04-16 15:20:36 -07:00
return 1 ;
}
static int __init sbf_init ( void )
{
u8 v ;
2008-01-30 13:32:31 +01:00
if ( sbf_port = = - 1 )
2005-04-16 15:20:36 -07:00
return 0 ;
2008-01-30 13:32:31 +01:00
2005-04-16 15:20:36 -07:00
v = sbf_read ( ) ;
2008-01-30 13:32:31 +01:00
if ( ! sbf_value_valid ( v ) ) {
printk ( KERN_WARNING " Simple Boot Flag value 0x%x read from "
" CMOS RAM was invalid \n " , v ) ;
}
2005-04-16 15:20:36 -07:00
v & = ~ SBF_RESERVED ;
v & = ~ SBF_BOOTING ;
v & = ~ SBF_DIAG ;
# if defined(CONFIG_ISAPNP)
v | = SBF_PNPOS ;
# endif
sbf_write ( v ) ;
2008-01-30 13:32:31 +01:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
module_init ( sbf_init ) ;