2005-04-17 02:20:36 +04:00
/*
* CMOS / NV - RAM driver for Linux
*
* Copyright ( C ) 1997 Roman Hodek < Roman . Hodek @ informatik . uni - erlangen . de >
* idea by and with help from Richard Jelinek < rj @ suse . de >
* Portions copyright ( c ) 2001 , 2002 Sun Microsystems ( thockin @ sun . com )
*
* This driver allows you to access the contents of the non - volatile memory in
* the mc146818rtc . h real - time clock . This chip is built into all PCs and into
* many Atari machines . In the former it ' s called " CMOS-RAM " , in the latter
* " NVRAM " ( NV stands for non - volatile ) .
*
* The data are supplied as a ( seekable ) character device , / dev / nvram . The
* size of this file is dependent on the controller . The usual size is 114 ,
* the number of freely available bytes in the memory ( i . e . , not used by the
* RTC itself ) .
*
* Checksums over the NVRAM contents are managed by this driver . In case of a
* bad checksum , reads and writes return - EIO . The checksum can be initialized
* to a sane state either by ioctl ( NVRAM_INIT ) ( clear whole NVRAM ) or
* ioctl ( NVRAM_SETCKS ) ( doesn ' t change contents , just makes checksum valid
* again ; use with care ! )
*
* This file also provides some functions for other parts of the kernel that
* want to access the NVRAM : nvram_ { read , write , check_checksum , set_checksum } .
* Obviously this can be used only if this driver is always configured into
* the kernel and is not a module . Since the functions are used by some Atari
* drivers , this is the case on the Atari .
*
*
* 1.1 Cesar Barros : SMP locking fixes
* added changelog
* 1.2 Erik Gilling : Cobalt Networks support
* Tim Hockin : general cleanup , Cobalt support
2008-11-11 12:56:00 +03:00
* 1.3 Wim Van Sebroeck : convert PRINT_PROC to seq_file
2005-04-17 02:20:36 +04:00
*/
2008-11-11 12:56:00 +03:00
# define NVRAM_VERSION "1.3"
2005-04-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/smp_lock.h>
# include <linux/nvram.h>
# define PC 1
# define ATARI 2
/* select machine configuration */
# if defined(CONFIG_ATARI)
# define MACH ATARI
2008-11-11 12:44:07 +03:00
# elif defined(__i386__) || defined(__x86_64__) || defined(__arm__) /* and ?? */
2007-07-16 10:41:04 +04:00
# define MACH PC
2005-04-17 02:20:36 +04:00
# else
# error Cannot build nvram driver for this machine configuration.
# endif
# if MACH == PC
/* RTC in a PC */
# define CHECK_DRIVER_INIT() 1
/* On PCs, the checksum is built only over bytes 2..31 */
# define PC_CKS_RANGE_START 2
# define PC_CKS_RANGE_END 31
# define PC_CKS_LOC 32
# define NVRAM_BYTES (128-NVRAM_FIRST_BYTE)
# define mach_check_checksum pc_check_checksum
# define mach_set_checksum pc_set_checksum
# define mach_proc_infos pc_proc_infos
# endif
# if MACH == ATARI
/* Special parameters for RTC in Atari machines */
# include <asm/atarihw.h>
# include <asm/atariints.h>
# define RTC_PORT(x) (TT_RTC_BAS + 2*(x))
# define CHECK_DRIVER_INIT() (MACH_IS_ATARI && ATARIHW_PRESENT(TT_CLK))
# define NVRAM_BYTES 50
/* On Ataris, the checksum is over all bytes except the checksum bytes
* themselves ; these are at the very end */
# define ATARI_CKS_RANGE_START 0
# define ATARI_CKS_RANGE_END 47
# define ATARI_CKS_LOC 48
# define mach_check_checksum atari_check_checksum
# define mach_set_checksum atari_set_checksum
# define mach_proc_infos atari_proc_infos
# endif
/* Note that *all* calls to CMOS_READ and CMOS_WRITE must be done with
* rtc_lock held . Due to the index - port / data - port design of the RTC , we
* don ' t want two different things trying to get to it at once . ( e . g . the
* periodic 11 min sync from time . c vs . this driver . )
*/
# include <linux/types.h>
# include <linux/errno.h>
# include <linux/miscdevice.h>
# include <linux/slab.h>
# include <linux/ioport.h>
# include <linux/fcntl.h>
# include <linux/mc146818rtc.h>
# include <linux/init.h>
# include <linux/proc_fs.h>
2008-11-11 12:56:00 +03:00
# include <linux/seq_file.h>
2005-04-17 02:20:36 +04:00
# include <linux/spinlock.h>
2008-11-11 12:44:07 +03:00
# include <linux/io.h>
# include <linux/uaccess.h>
2005-04-17 02:20:36 +04:00
# include <asm/system.h>
static DEFINE_SPINLOCK ( nvram_state_lock ) ;
static int nvram_open_cnt ; /* #times opened */
static int nvram_open_mode ; /* special open modes */
# define NVRAM_WRITE 1 /* opened for writing (exclusive) */
# define NVRAM_EXCL 2 /* opened with O_EXCL */
static int mach_check_checksum ( void ) ;
static void mach_set_checksum ( void ) ;
# ifdef CONFIG_PROC_FS
2008-11-11 12:56:00 +03:00
static void mach_proc_infos ( unsigned char * contents , struct seq_file * seq ,
void * offset ) ;
2005-04-17 02:20:36 +04:00
# endif
/*
* These functions are provided to be called internally or by other parts of
* the kernel . It ' s up to the caller to ensure correct checksum before reading
* or after writing ( needs to be done only once ) .
*
* It is worth noting that these functions all access bytes of general
* purpose memory in the NVRAM - that is to say , they all add the
2008-11-11 12:44:07 +03:00
* NVRAM_FIRST_BYTE offset . Pass them offsets into NVRAM as if you did not
2005-04-17 02:20:36 +04:00
* know about the RTC cruft .
*/
2008-11-11 12:44:07 +03:00
unsigned char __nvram_read_byte ( int i )
2005-04-17 02:20:36 +04:00
{
return CMOS_READ ( NVRAM_FIRST_BYTE + i ) ;
}
2008-11-11 12:44:07 +03:00
EXPORT_SYMBOL ( __nvram_read_byte ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:44:07 +03:00
unsigned char nvram_read_byte ( int i )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
unsigned char c ;
spin_lock_irqsave ( & rtc_lock , flags ) ;
c = __nvram_read_byte ( i ) ;
spin_unlock_irqrestore ( & rtc_lock , flags ) ;
return c ;
}
2008-11-11 12:44:07 +03:00
EXPORT_SYMBOL ( nvram_read_byte ) ;
2005-04-17 02:20:36 +04:00
/* This races nicely with trying to read with checksum checking (nvram_read) */
2008-11-11 12:44:07 +03:00
void __nvram_write_byte ( unsigned char c , int i )
2005-04-17 02:20:36 +04:00
{
CMOS_WRITE ( c , NVRAM_FIRST_BYTE + i ) ;
}
2008-11-11 12:44:07 +03:00
EXPORT_SYMBOL ( __nvram_write_byte ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:44:07 +03:00
void nvram_write_byte ( unsigned char c , int i )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
spin_lock_irqsave ( & rtc_lock , flags ) ;
__nvram_write_byte ( c , i ) ;
spin_unlock_irqrestore ( & rtc_lock , flags ) ;
}
2008-11-11 12:44:07 +03:00
EXPORT_SYMBOL ( nvram_write_byte ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:44:07 +03:00
int __nvram_check_checksum ( void )
2005-04-17 02:20:36 +04:00
{
return mach_check_checksum ( ) ;
}
2008-11-11 12:44:07 +03:00
EXPORT_SYMBOL ( __nvram_check_checksum ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:44:07 +03:00
int nvram_check_checksum ( void )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
int rv ;
spin_lock_irqsave ( & rtc_lock , flags ) ;
rv = __nvram_check_checksum ( ) ;
spin_unlock_irqrestore ( & rtc_lock , flags ) ;
return rv ;
}
2008-11-11 12:44:07 +03:00
EXPORT_SYMBOL ( nvram_check_checksum ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:44:07 +03:00
static void __nvram_set_checksum ( void )
2005-04-17 02:20:36 +04:00
{
mach_set_checksum ( ) ;
}
2005-06-26 01:59:04 +04:00
#if 0
2008-11-11 12:44:07 +03:00
void nvram_set_checksum ( void )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
spin_lock_irqsave ( & rtc_lock , flags ) ;
__nvram_set_checksum ( ) ;
spin_unlock_irqrestore ( & rtc_lock , flags ) ;
}
2005-06-26 01:59:04 +04:00
# endif /* 0 */
2005-04-17 02:20:36 +04:00
/*
* The are the file operation function for user access to / dev / nvram
*/
2008-11-11 12:44:07 +03:00
static loff_t nvram_llseek ( struct file * file , loff_t offset , int origin )
2005-04-17 02:20:36 +04:00
{
lock_kernel ( ) ;
switch ( origin ) {
case 0 :
/* nothing to do */
break ;
case 1 :
offset + = file - > f_pos ;
break ;
case 2 :
offset + = NVRAM_BYTES ;
break ;
}
unlock_kernel ( ) ;
return ( offset > = 0 ) ? ( file - > f_pos = offset ) : - EINVAL ;
}
2008-11-11 12:44:07 +03:00
static ssize_t nvram_read ( struct file * file , char __user * buf ,
size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
unsigned char contents [ NVRAM_BYTES ] ;
unsigned i = * ppos ;
unsigned char * tmp ;
spin_lock_irq ( & rtc_lock ) ;
if ( ! __nvram_check_checksum ( ) )
goto checksum_err ;
for ( tmp = contents ; count - - > 0 & & i < NVRAM_BYTES ; + + i , + + tmp )
* tmp = __nvram_read_byte ( i ) ;
spin_unlock_irq ( & rtc_lock ) ;
if ( copy_to_user ( buf , contents , tmp - contents ) )
return - EFAULT ;
* ppos = i ;
return tmp - contents ;
2008-11-11 12:44:07 +03:00
checksum_err :
2005-04-17 02:20:36 +04:00
spin_unlock_irq ( & rtc_lock ) ;
return - EIO ;
}
2008-11-11 12:44:07 +03:00
static ssize_t nvram_write ( struct file * file , const char __user * buf ,
size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
unsigned char contents [ NVRAM_BYTES ] ;
unsigned i = * ppos ;
unsigned char * tmp ;
int len ;
len = ( NVRAM_BYTES - i ) < count ? ( NVRAM_BYTES - i ) : count ;
if ( copy_from_user ( contents , buf , len ) )
return - EFAULT ;
spin_lock_irq ( & rtc_lock ) ;
if ( ! __nvram_check_checksum ( ) )
goto checksum_err ;
for ( tmp = contents ; count - - > 0 & & i < NVRAM_BYTES ; + + i , + + tmp )
__nvram_write_byte ( * tmp , i ) ;
__nvram_set_checksum ( ) ;
spin_unlock_irq ( & rtc_lock ) ;
* ppos = i ;
return tmp - contents ;
2008-11-11 12:44:07 +03:00
checksum_err :
2005-04-17 02:20:36 +04:00
spin_unlock_irq ( & rtc_lock ) ;
return - EIO ;
}
2008-11-11 12:44:07 +03:00
static int nvram_ioctl ( struct inode * inode , struct file * file ,
unsigned int cmd , unsigned long arg )
2005-04-17 02:20:36 +04:00
{
int i ;
switch ( cmd ) {
case NVRAM_INIT :
/* initialize NVRAM contents and checksum */
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
spin_lock_irq ( & rtc_lock ) ;
for ( i = 0 ; i < NVRAM_BYTES ; + + i )
__nvram_write_byte ( 0 , i ) ;
__nvram_set_checksum ( ) ;
spin_unlock_irq ( & rtc_lock ) ;
return 0 ;
case NVRAM_SETCKS :
2008-11-11 12:44:07 +03:00
/* just set checksum, contents unchanged (maybe useful after
2005-04-17 02:20:36 +04:00
* checksum garbaged somehow . . . ) */
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
spin_lock_irq ( & rtc_lock ) ;
__nvram_set_checksum ( ) ;
spin_unlock_irq ( & rtc_lock ) ;
return 0 ;
default :
return - ENOTTY ;
}
}
2008-11-11 12:44:07 +03:00
static int nvram_open ( struct inode * inode , struct file * file )
2005-04-17 02:20:36 +04:00
{
2008-05-20 21:16:26 +04:00
lock_kernel ( ) ;
2005-04-17 02:20:36 +04:00
spin_lock ( & nvram_state_lock ) ;
if ( ( nvram_open_cnt & & ( file - > f_flags & O_EXCL ) ) | |
( nvram_open_mode & NVRAM_EXCL ) | |
2008-09-02 23:28:45 +04:00
( ( file - > f_mode & FMODE_WRITE ) & & ( nvram_open_mode & NVRAM_WRITE ) ) ) {
2005-04-17 02:20:36 +04:00
spin_unlock ( & nvram_state_lock ) ;
2008-05-20 21:16:26 +04:00
unlock_kernel ( ) ;
2005-04-17 02:20:36 +04:00
return - EBUSY ;
}
if ( file - > f_flags & O_EXCL )
nvram_open_mode | = NVRAM_EXCL ;
2008-09-02 23:28:45 +04:00
if ( file - > f_mode & FMODE_WRITE )
2005-04-17 02:20:36 +04:00
nvram_open_mode | = NVRAM_WRITE ;
nvram_open_cnt + + ;
spin_unlock ( & nvram_state_lock ) ;
2008-05-20 21:16:26 +04:00
unlock_kernel ( ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2008-11-11 12:44:07 +03:00
static int nvram_release ( struct inode * inode , struct file * file )
2005-04-17 02:20:36 +04:00
{
spin_lock ( & nvram_state_lock ) ;
nvram_open_cnt - - ;
/* if only one instance is open, clear the EXCL bit */
if ( nvram_open_mode & NVRAM_EXCL )
nvram_open_mode & = ~ NVRAM_EXCL ;
2008-09-02 23:28:45 +04:00
if ( file - > f_mode & FMODE_WRITE )
2005-04-17 02:20:36 +04:00
nvram_open_mode & = ~ NVRAM_WRITE ;
spin_unlock ( & nvram_state_lock ) ;
return 0 ;
}
# ifndef CONFIG_PROC_FS
2008-11-11 12:56:00 +03:00
static int nvram_add_proc_fs ( void )
2005-04-17 02:20:36 +04:00
{
return 0 ;
}
2008-11-11 12:56:00 +03:00
2005-04-17 02:20:36 +04:00
# else
2008-11-11 12:56:00 +03:00
static int nvram_proc_read ( struct seq_file * seq , void * offset )
2005-04-17 02:20:36 +04:00
{
unsigned char contents [ NVRAM_BYTES ] ;
2008-11-11 12:56:00 +03:00
int i = 0 ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( & rtc_lock ) ;
for ( i = 0 ; i < NVRAM_BYTES ; + + i )
contents [ i ] = __nvram_read_byte ( i ) ;
spin_unlock_irq ( & rtc_lock ) ;
2008-11-11 12:56:00 +03:00
mach_proc_infos ( contents , seq , offset ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
static int nvram_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , nvram_proc_read , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-11-11 12:56:00 +03:00
static const struct file_operations nvram_proc_fops = {
. owner = THIS_MODULE ,
. open = nvram_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static int nvram_add_proc_fs ( void )
{
if ( ! proc_create ( " driver/nvram " , 0 , NULL , & nvram_proc_fops ) )
return - ENOMEM ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
# endif /* CONFIG_PROC_FS */
2006-07-03 11:24:21 +04:00
static const struct file_operations nvram_fops = {
2005-04-17 02:20:36 +04:00
. owner = THIS_MODULE ,
. llseek = nvram_llseek ,
. read = nvram_read ,
. write = nvram_write ,
. ioctl = nvram_ioctl ,
. open = nvram_open ,
. release = nvram_release ,
} ;
static struct miscdevice nvram_dev = {
NVRAM_MINOR ,
" nvram " ,
& nvram_fops
} ;
2008-11-11 12:44:07 +03:00
static int __init nvram_init ( void )
2005-04-17 02:20:36 +04:00
{
int ret ;
/* First test whether the driver should init at all */
if ( ! CHECK_DRIVER_INIT ( ) )
2008-07-17 23:16:10 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
ret = misc_register ( & nvram_dev ) ;
if ( ret ) {
printk ( KERN_ERR " nvram: can't misc_register on minor=%d \n " ,
NVRAM_MINOR ) ;
goto out ;
}
2008-11-11 12:56:00 +03:00
ret = nvram_add_proc_fs ( ) ;
if ( ret ) {
2005-04-17 02:20:36 +04:00
printk ( KERN_ERR " nvram: can't create /proc/driver/nvram \n " ) ;
goto outmisc ;
}
ret = 0 ;
printk ( KERN_INFO " Non-volatile memory driver v " NVRAM_VERSION " \n " ) ;
2008-11-11 12:44:07 +03:00
out :
2005-04-17 02:20:36 +04:00
return ret ;
2008-11-11 12:44:07 +03:00
outmisc :
2005-04-17 02:20:36 +04:00
misc_deregister ( & nvram_dev ) ;
goto out ;
}
2008-11-11 12:44:07 +03:00
static void __exit nvram_cleanup_module ( void )
2005-04-17 02:20:36 +04:00
{
remove_proc_entry ( " driver/nvram " , NULL ) ;
misc_deregister ( & nvram_dev ) ;
}
module_init ( nvram_init ) ;
module_exit ( nvram_cleanup_module ) ;
/*
* Machine specific functions
*/
# if MACH == PC
2008-11-11 12:44:07 +03:00
static int pc_check_checksum ( void )
2005-04-17 02:20:36 +04:00
{
int i ;
unsigned short sum = 0 ;
unsigned short expect ;
for ( i = PC_CKS_RANGE_START ; i < = PC_CKS_RANGE_END ; + + i )
sum + = __nvram_read_byte ( i ) ;
expect = __nvram_read_byte ( PC_CKS_LOC ) < < 8 |
__nvram_read_byte ( PC_CKS_LOC + 1 ) ;
2008-11-11 12:44:07 +03:00
return ( sum & 0xffff ) = = expect ;
2005-04-17 02:20:36 +04:00
}
2008-11-11 12:44:07 +03:00
static void pc_set_checksum ( void )
2005-04-17 02:20:36 +04:00
{
int i ;
unsigned short sum = 0 ;
for ( i = PC_CKS_RANGE_START ; i < = PC_CKS_RANGE_END ; + + i )
sum + = __nvram_read_byte ( i ) ;
__nvram_write_byte ( sum > > 8 , PC_CKS_LOC ) ;
__nvram_write_byte ( sum & 0xff , PC_CKS_LOC + 1 ) ;
}
# ifdef CONFIG_PROC_FS
static char * floppy_types [ ] = {
" none " , " 5.25'' 360k " , " 5.25'' 1.2M " , " 3.5'' 720k " , " 3.5'' 1.44M " ,
" 3.5'' 2.88M " , " 3.5'' 2.88M "
} ;
static char * gfx_types [ ] = {
" EGA, VGA, ... (with BIOS) " ,
" CGA (40 cols) " ,
" CGA (80 cols) " ,
" monochrome " ,
} ;
2008-11-11 12:56:00 +03:00
static void pc_proc_infos ( unsigned char * nvram , struct seq_file * seq ,
void * offset )
2005-04-17 02:20:36 +04:00
{
int checksum ;
int type ;
spin_lock_irq ( & rtc_lock ) ;
checksum = __nvram_check_checksum ( ) ;
spin_unlock_irq ( & rtc_lock ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " Checksum status: %svalid \n " , checksum ? " " : " not " ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " # floppies : %d \n " ,
2005-04-17 02:20:36 +04:00
( nvram [ 6 ] & 1 ) ? ( nvram [ 6 ] > > 6 ) + 1 : 0 ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " Floppy 0 type : " ) ;
2005-04-17 02:20:36 +04:00
type = nvram [ 2 ] > > 4 ;
2006-01-10 07:54:02 +03:00
if ( type < ARRAY_SIZE ( floppy_types ) )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %s \n " , floppy_types [ type ] ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %d (unknown) \n " , type ) ;
seq_printf ( seq , " Floppy 1 type : " ) ;
2005-04-17 02:20:36 +04:00
type = nvram [ 2 ] & 0x0f ;
2006-01-10 07:54:02 +03:00
if ( type < ARRAY_SIZE ( floppy_types ) )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %s \n " , floppy_types [ type ] ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %d (unknown) \n " , type ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " HD 0 type : " ) ;
2005-04-17 02:20:36 +04:00
type = nvram [ 4 ] > > 4 ;
if ( type )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %02x \n " , type = = 0x0f ? nvram [ 11 ] : type ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " none \n " ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " HD 1 type : " ) ;
2005-04-17 02:20:36 +04:00
type = nvram [ 4 ] & 0x0f ;
if ( type )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %02x \n " , type = = 0x0f ? nvram [ 12 ] : type ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " none \n " ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " HD type 48 data: %d/%d/%d C/H/S, precomp %d, lz %d \n " ,
2005-04-17 02:20:36 +04:00
nvram [ 18 ] | ( nvram [ 19 ] < < 8 ) ,
nvram [ 20 ] , nvram [ 25 ] ,
nvram [ 21 ] | ( nvram [ 22 ] < < 8 ) , nvram [ 23 ] | ( nvram [ 24 ] < < 8 ) ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " HD type 49 data: %d/%d/%d C/H/S, precomp %d, lz %d \n " ,
2005-04-17 02:20:36 +04:00
nvram [ 39 ] | ( nvram [ 40 ] < < 8 ) ,
nvram [ 41 ] , nvram [ 46 ] ,
nvram [ 42 ] | ( nvram [ 43 ] < < 8 ) , nvram [ 44 ] | ( nvram [ 45 ] < < 8 ) ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " DOS base memory: %d kB \n " , nvram [ 7 ] | ( nvram [ 8 ] < < 8 ) ) ;
seq_printf ( seq , " Extended memory: %d kB (configured), %d kB (tested) \n " ,
2005-04-17 02:20:36 +04:00
nvram [ 9 ] | ( nvram [ 10 ] < < 8 ) , nvram [ 34 ] | ( nvram [ 35 ] < < 8 ) ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " Gfx adapter : %s \n " ,
gfx_types [ ( nvram [ 6 ] > > 4 ) & 3 ] ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " FPU : %sinstalled \n " ,
2005-04-17 02:20:36 +04:00
( nvram [ 6 ] & 2 ) ? " " : " not " ) ;
2008-11-11 12:56:00 +03:00
return ;
2005-04-17 02:20:36 +04:00
}
# endif
# endif /* MACH == PC */
# if MACH == ATARI
2008-11-11 12:44:07 +03:00
static int atari_check_checksum ( void )
2005-04-17 02:20:36 +04:00
{
int i ;
unsigned char sum = 0 ;
for ( i = ATARI_CKS_RANGE_START ; i < = ATARI_CKS_RANGE_END ; + + i )
sum + = __nvram_read_byte ( i ) ;
2008-11-11 12:44:07 +03:00
return ( __nvram_read_byte ( ATARI_CKS_LOC ) = = ( ~ sum & 0xff ) ) & &
( __nvram_read_byte ( ATARI_CKS_LOC + 1 ) = = ( sum & 0xff ) ) ;
2005-04-17 02:20:36 +04:00
}
2008-11-11 12:44:07 +03:00
static void atari_set_checksum ( void )
2005-04-17 02:20:36 +04:00
{
int i ;
unsigned char sum = 0 ;
for ( i = ATARI_CKS_RANGE_START ; i < = ATARI_CKS_RANGE_END ; + + i )
sum + = __nvram_read_byte ( i ) ;
__nvram_write_byte ( ~ sum , ATARI_CKS_LOC ) ;
__nvram_write_byte ( sum , ATARI_CKS_LOC + 1 ) ;
}
# ifdef CONFIG_PROC_FS
static struct {
unsigned char val ;
char * name ;
} boot_prefs [ ] = {
{ 0x80 , " TOS " } ,
{ 0x40 , " ASV " } ,
{ 0x20 , " NetBSD (?) " } ,
{ 0x10 , " Linux " } ,
{ 0x00 , " unspecified " }
} ;
static char * languages [ ] = {
" English (US) " ,
" German " ,
" French " ,
" English (UK) " ,
" Spanish " ,
" Italian " ,
" 6 (undefined) " ,
" Swiss (French) " ,
" Swiss (German) "
} ;
static char * dateformat [ ] = {
" MM%cDD%cYY " ,
" DD%cMM%cYY " ,
" YY%cMM%cDD " ,
" YY%cDD%cMM " ,
" 4 (undefined) " ,
" 5 (undefined) " ,
" 6 (undefined) " ,
" 7 (undefined) "
} ;
static char * colors [ ] = {
" 2 " , " 4 " , " 16 " , " 256 " , " 65536 " , " ?? " , " ?? " , " ?? "
} ;
2008-11-11 12:56:00 +03:00
static void atari_proc_infos ( unsigned char * nvram , struct seq_file * seq ,
void * offset )
2005-04-17 02:20:36 +04:00
{
int checksum = nvram_check_checksum ( ) ;
int i ;
unsigned vmode ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " Checksum status : %svalid \n " , checksum ? " " : " not " ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " Boot preference : " ) ;
2006-01-10 07:54:02 +03:00
for ( i = ARRAY_SIZE ( boot_prefs ) - 1 ; i > = 0 ; - - i ) {
2005-04-17 02:20:36 +04:00
if ( nvram [ 1 ] = = boot_prefs [ i ] . val ) {
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %s \n " , boot_prefs [ i ] . name ) ;
2005-04-17 02:20:36 +04:00
break ;
}
}
if ( i < 0 )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " 0x%02x (undefined) \n " , nvram [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " SCSI arbitration : %s \n " ,
2005-04-17 02:20:36 +04:00
( nvram [ 16 ] & 0x80 ) ? " on " : " off " ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " SCSI host ID : " ) ;
2005-04-17 02:20:36 +04:00
if ( nvram [ 16 ] & 0x80 )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %d \n " , nvram [ 16 ] & 7 ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " n/a \n " ) ;
2005-04-17 02:20:36 +04:00
/* the following entries are defined only for the Falcon */
if ( ( atari_mch_cookie > > 16 ) ! = ATARI_MCH_FALCON )
2008-11-11 12:56:00 +03:00
return ;
2005-04-17 02:20:36 +04:00
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " OS language : " ) ;
2006-01-10 07:54:02 +03:00
if ( nvram [ 6 ] < ARRAY_SIZE ( languages ) )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %s \n " , languages [ nvram [ 6 ] ] ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %u (undefined) \n " , nvram [ 6 ] ) ;
seq_printf ( seq , " Keyboard language: " ) ;
2006-01-10 07:54:02 +03:00
if ( nvram [ 7 ] < ARRAY_SIZE ( languages ) )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %s \n " , languages [ nvram [ 7 ] ] ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %u (undefined) \n " , nvram [ 7 ] ) ;
seq_printf ( seq , " Date format : " ) ;
seq_printf ( seq , dateformat [ nvram [ 8 ] & 7 ] ,
2005-04-17 02:20:36 +04:00
nvram [ 9 ] ? nvram [ 9 ] : ' / ' , nvram [ 9 ] ? nvram [ 9 ] : ' / ' ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " , %dh clock \n " , nvram [ 8 ] & 16 ? 24 : 12 ) ;
seq_printf ( seq , " Boot delay : " ) ;
2005-04-17 02:20:36 +04:00
if ( nvram [ 10 ] = = 0 )
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " default " ) ;
2005-04-17 02:20:36 +04:00
else
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %ds%s \n " , nvram [ 10 ] ,
2005-04-17 02:20:36 +04:00
nvram [ 10 ] < 8 ? " , no memory test " : " " ) ;
vmode = ( nvram [ 14 ] < < 8 ) | | nvram [ 15 ] ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq ,
" Video mode : %s colors, %d columns, %s %s monitor \n " ,
2005-04-17 02:20:36 +04:00
colors [ vmode & 7 ] ,
vmode & 8 ? 80 : 40 ,
vmode & 16 ? " VGA " : " TV " , vmode & 32 ? " PAL " : " NTSC " ) ;
2008-11-11 12:56:00 +03:00
seq_printf ( seq , " %soverscan, compat. mode %s%s \n " ,
2005-04-17 02:20:36 +04:00
vmode & 64 ? " " : " no " ,
vmode & 128 ? " on " : " off " ,
vmode & 256 ?
( vmode & 16 ? " , line doubling " : " , half screen " ) : " " ) ;
2008-11-11 12:56:00 +03:00
return ;
2005-04-17 02:20:36 +04:00
}
# endif
# endif /* MACH == ATARI */
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_MISCDEV ( NVRAM_MINOR ) ;