2005-09-07 02:17:14 +04:00
/*
* dell_rbu . c
* Bios Update driver for Dell systems
* Author : Dell Inc
* Abhay Salunke < abhay_salunke @ dell . com >
*
* Copyright ( C ) 2005 Dell Inc .
*
* Remote BIOS Update ( rbu ) driver is used for updating DELL BIOS by
* creating entries in the / sys file systems on Linux 2.6 and higher
* kernels . The driver supports two mechanism to update the BIOS namely
* contiguous and packetized . Both these methods still require having some
* application to set the CMOS bit indicating the BIOS to update itself
* after a reboot .
*
* Contiguous method :
* This driver writes the incoming data in a monolithic image by allocating
* contiguous physical pages large enough to accommodate the incoming BIOS
* image size .
*
* Packetized method :
* The driver writes the incoming packet image by allocating a new packet
* on every time the packet data is written . This driver requires an
* application to break the BIOS image in to fixed sized packet chunks .
*
* See Documentation / dell_rbu . txt for more info .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License v2 .0 as published by
* the Free Software Foundation
*
* 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/init.h>
# include <linux/module.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/blkdev.h>
2005-10-29 22:07:23 +04:00
# include <linux/platform_device.h>
2005-09-07 02:17:14 +04:00
# include <linux/spinlock.h>
# include <linux/moduleparam.h>
# include <linux/firmware.h>
# include <linux/dma-mapping.h>
MODULE_AUTHOR ( " Abhay Salunke <abhay_salunke@dell.com> " ) ;
MODULE_DESCRIPTION ( " Driver for updating BIOS image on DELL systems " ) ;
MODULE_LICENSE ( " GPL " ) ;
2006-01-15 00:21:14 +03:00
MODULE_VERSION ( " 3.2 " ) ;
2005-09-07 02:17:14 +04:00
# define BIOS_SCAN_LIMIT 0xffffffff
# define MAX_IMAGE_LENGTH 16
static struct _rbu_data {
void * image_update_buffer ;
unsigned long image_update_buffer_size ;
unsigned long bios_image_size ;
int image_update_ordernum ;
int dma_alloc ;
spinlock_t lock ;
unsigned long packet_read_count ;
unsigned long num_packets ;
unsigned long packetsize ;
2005-10-11 19:29:02 +04:00
unsigned long imagesize ;
2005-09-17 06:28:04 +04:00
int entry_created ;
2005-09-07 02:17:14 +04:00
} rbu_data ;
2005-09-17 06:28:04 +04:00
static char image_type [ MAX_IMAGE_LENGTH + 1 ] = " mono " ;
module_param_string ( image_type , image_type , sizeof ( image_type ) , 0 ) ;
2005-10-11 19:29:02 +04:00
MODULE_PARM_DESC ( image_type ,
" BIOS image type. choose- mono or packet or init " ) ;
2005-09-07 02:17:14 +04:00
2005-11-07 11:59:26 +03:00
static unsigned long allocation_floor = 0x100000 ;
module_param ( allocation_floor , ulong , 0644 ) ;
MODULE_PARM_DESC ( allocation_floor ,
" Minimum address for allocations when using Packet mode " ) ;
2005-09-07 02:17:14 +04:00
struct packet_data {
struct list_head list ;
size_t length ;
void * data ;
int ordernum ;
} ;
static struct packet_data packet_data_head ;
static struct platform_device * rbu_device ;
static int context ;
static dma_addr_t dell_rbu_dmaaddr ;
2005-09-17 06:28:05 +04:00
static void init_packet_head ( void )
2005-09-07 02:17:14 +04:00
{
INIT_LIST_HEAD ( & packet_data_head . list ) ;
rbu_data . packet_read_count = 0 ;
rbu_data . num_packets = 0 ;
rbu_data . packetsize = 0 ;
2005-10-11 19:29:02 +04:00
rbu_data . imagesize = 0 ;
2005-09-07 02:17:14 +04:00
}
2005-10-11 19:29:02 +04:00
static int create_packet ( void * data , size_t length )
2005-09-07 02:17:14 +04:00
{
struct packet_data * newpacket ;
int ordernum = 0 ;
2005-11-07 11:59:26 +03:00
int retval = 0 ;
unsigned int packet_array_size = 0 ;
2005-12-15 12:18:00 +03:00
void * * invalid_addr_packet_array = NULL ;
void * packet_data_temp_buf = NULL ;
2005-11-07 11:59:26 +03:00
unsigned int idx = 0 ;
2005-09-07 02:17:14 +04:00
pr_debug ( " create_packet: entry \n " ) ;
if ( ! rbu_data . packetsize ) {
pr_debug ( " create_packet: packetsize not specified \n " ) ;
2005-11-07 11:59:26 +03:00
retval = - EINVAL ;
goto out_noalloc ;
2005-09-07 02:17:14 +04:00
}
2005-11-07 11:59:26 +03:00
2005-09-17 06:28:04 +04:00
spin_unlock ( & rbu_data . lock ) ;
2005-11-07 11:59:26 +03:00
newpacket = kzalloc ( sizeof ( struct packet_data ) , GFP_KERNEL ) ;
2005-09-07 02:17:14 +04:00
if ( ! newpacket ) {
printk ( KERN_WARNING
2005-09-17 06:28:04 +04:00
" dell_rbu:%s: failed to allocate new "
" packet \n " , __FUNCTION__ ) ;
2005-11-07 11:59:26 +03:00
retval = - ENOMEM ;
spin_lock ( & rbu_data . lock ) ;
goto out_noalloc ;
2005-09-07 02:17:14 +04:00
}
ordernum = get_order ( length ) ;
2005-11-07 11:59:26 +03:00
2005-09-07 02:17:14 +04:00
/*
2005-11-07 11:59:26 +03:00
* BIOS errata mean we cannot allocate packets below 1 MB or they will
* be overwritten by BIOS .
*
* array to temporarily hold packets
* that are below the allocation floor
*
* NOTE : very simplistic because we only need the floor to be at 1 MB
* due to BIOS errata . This shouldn ' t be used for higher floors
* or you will run out of mem trying to allocate the array .
2005-09-07 02:17:14 +04:00
*/
2005-11-07 11:59:26 +03:00
packet_array_size = max (
( unsigned int ) ( allocation_floor / rbu_data . packetsize ) ,
( unsigned int ) 1 ) ;
invalid_addr_packet_array = kzalloc ( packet_array_size * sizeof ( void * ) ,
GFP_KERNEL ) ;
2005-09-07 02:17:14 +04:00
2005-11-07 11:59:26 +03:00
if ( ! invalid_addr_packet_array ) {
2005-09-07 02:17:14 +04:00
printk ( KERN_WARNING
2005-11-07 11:59:26 +03:00
" dell_rbu:%s: failed to allocate "
" invalid_addr_packet_array \n " ,
__FUNCTION__ ) ;
retval = - ENOMEM ;
spin_lock ( & rbu_data . lock ) ;
goto out_alloc_packet ;
2005-09-07 02:17:14 +04:00
}
2005-11-07 11:59:26 +03:00
while ( ! packet_data_temp_buf ) {
packet_data_temp_buf = ( unsigned char * )
__get_free_pages ( GFP_KERNEL , ordernum ) ;
if ( ! packet_data_temp_buf ) {
printk ( KERN_WARNING
" dell_rbu:%s: failed to allocate new "
" packet \n " , __FUNCTION__ ) ;
retval = - ENOMEM ;
spin_lock ( & rbu_data . lock ) ;
goto out_alloc_packet_array ;
}
if ( ( unsigned long ) virt_to_phys ( packet_data_temp_buf )
< allocation_floor ) {
pr_debug ( " packet 0x%lx below floor at 0x%lx. \n " ,
( unsigned long ) virt_to_phys (
packet_data_temp_buf ) ,
allocation_floor ) ;
invalid_addr_packet_array [ idx + + ] = packet_data_temp_buf ;
2005-12-15 12:18:00 +03:00
packet_data_temp_buf = NULL ;
2005-11-07 11:59:26 +03:00
}
}
spin_lock ( & rbu_data . lock ) ;
newpacket - > data = packet_data_temp_buf ;
pr_debug ( " create_packet: newpacket at physical addr %lx \n " ,
( unsigned long ) virt_to_phys ( newpacket - > data ) ) ;
/* packets may not have fixed size */
newpacket - > length = length ;
2005-09-07 02:17:14 +04:00
newpacket - > ordernum = ordernum ;
+ + rbu_data . num_packets ;
2005-11-07 11:59:26 +03:00
/* initialize the newly created packet headers */
2005-09-07 02:17:14 +04:00
INIT_LIST_HEAD ( & newpacket - > list ) ;
list_add_tail ( & newpacket - > list , & packet_data_head . list ) ;
2005-10-11 19:29:02 +04:00
memcpy ( newpacket - > data , data , length ) ;
2005-09-07 02:17:14 +04:00
pr_debug ( " create_packet: exit \n " ) ;
2005-11-07 11:59:26 +03:00
out_alloc_packet_array :
/* always free packet array */
for ( ; idx > 0 ; idx - - ) {
pr_debug ( " freeing unused packet below floor 0x%lx. \n " ,
( unsigned long ) virt_to_phys (
invalid_addr_packet_array [ idx - 1 ] ) ) ;
free_pages ( ( unsigned long ) invalid_addr_packet_array [ idx - 1 ] ,
ordernum ) ;
}
kfree ( invalid_addr_packet_array ) ;
out_alloc_packet :
/* if error, free data */
if ( retval )
kfree ( newpacket ) ;
out_noalloc :
return retval ;
2005-09-07 02:17:14 +04:00
}
2005-09-17 06:28:05 +04:00
static int packetize_data ( void * data , size_t length )
2005-09-07 02:17:14 +04:00
{
int rc = 0 ;
2005-10-11 19:29:02 +04:00
int done = 0 ;
int packet_length ;
u8 * temp ;
u8 * end = ( u8 * ) data + length ;
2006-10-03 12:16:09 +04:00
pr_debug ( " packetize_data: data length %zd \n " , length ) ;
2005-10-11 19:29:02 +04:00
if ( ! rbu_data . packetsize ) {
printk ( KERN_WARNING
" dell_rbu: packetsize not specified \n " ) ;
return - EIO ;
}
2005-09-07 02:17:14 +04:00
2005-10-11 19:29:02 +04:00
temp = ( u8 * ) data ;
/* packetize the hunk */
while ( ! done ) {
if ( ( temp + rbu_data . packetsize ) < end )
packet_length = rbu_data . packetsize ;
else {
/* this is the last packet */
packet_length = end - temp ;
done = 1 ;
}
if ( ( rc = create_packet ( temp , packet_length ) ) )
2005-09-07 02:17:14 +04:00
return rc ;
2005-10-11 19:29:02 +04:00
2006-10-11 12:22:13 +04:00
pr_debug ( " %p:%td \n " , temp , ( end - temp ) ) ;
2005-10-11 19:29:02 +04:00
temp + = packet_length ;
2005-09-07 02:17:14 +04:00
}
2005-10-11 19:29:02 +04:00
rbu_data . imagesize = length ;
2005-09-07 02:17:14 +04:00
return rc ;
}
2005-09-17 06:28:05 +04:00
static int do_packet_read ( char * data , struct list_head * ptemp_list ,
2005-09-17 06:28:04 +04:00
int length , int bytes_read , int * list_read_count )
2005-09-07 02:17:14 +04:00
{
void * ptemp_buf ;
struct packet_data * newpacket = NULL ;
int bytes_copied = 0 ;
int j = 0 ;
newpacket = list_entry ( ptemp_list , struct packet_data , list ) ;
* list_read_count + = newpacket - > length ;
if ( * list_read_count > bytes_read ) {
/* point to the start of unread data */
j = newpacket - > length - ( * list_read_count - bytes_read ) ;
/* point to the offset in the packet buffer */
ptemp_buf = ( u8 * ) newpacket - > data + j ;
/*
* check if there is enough room in
* * the incoming buffer
*/
if ( length > ( * list_read_count - bytes_read ) )
/*
* copy what ever is there in this
* packet and move on
*/
bytes_copied = ( * list_read_count - bytes_read ) ;
else
/* copy the remaining */
bytes_copied = length ;
memcpy ( data , ptemp_buf , bytes_copied ) ;
}
return bytes_copied ;
}
2005-10-11 19:29:02 +04:00
static int packet_read_list ( char * data , size_t * pread_length )
2005-09-07 02:17:14 +04:00
{
struct list_head * ptemp_list ;
int temp_count = 0 ;
int bytes_copied = 0 ;
int bytes_read = 0 ;
int remaining_bytes = 0 ;
char * pdest = data ;
/* check if we have any packets */
if ( 0 = = rbu_data . num_packets )
return - ENOMEM ;
remaining_bytes = * pread_length ;
bytes_read = rbu_data . packet_read_count ;
ptemp_list = ( & packet_data_head . list ) - > next ;
while ( ! list_empty ( ptemp_list ) ) {
bytes_copied = do_packet_read ( pdest , ptemp_list ,
2005-09-17 06:28:04 +04:00
remaining_bytes , bytes_read , & temp_count ) ;
2005-09-07 02:17:14 +04:00
remaining_bytes - = bytes_copied ;
bytes_read + = bytes_copied ;
pdest + = bytes_copied ;
/*
* check if we reached end of buffer before reaching the
* last packet
*/
if ( remaining_bytes = = 0 )
break ;
ptemp_list = ptemp_list - > next ;
}
/*finally set the bytes read */
* pread_length = bytes_read - rbu_data . packet_read_count ;
rbu_data . packet_read_count = bytes_read ;
return 0 ;
}
2005-09-17 06:28:05 +04:00
static void packet_empty_list ( void )
2005-09-07 02:17:14 +04:00
{
struct list_head * ptemp_list ;
struct list_head * pnext_list ;
struct packet_data * newpacket ;
ptemp_list = ( & packet_data_head . list ) - > next ;
while ( ! list_empty ( ptemp_list ) ) {
newpacket =
2005-09-17 06:28:04 +04:00
list_entry ( ptemp_list , struct packet_data , list ) ;
2005-09-07 02:17:14 +04:00
pnext_list = ptemp_list - > next ;
list_del ( ptemp_list ) ;
ptemp_list = pnext_list ;
/*
* zero out the RBU packet memory before freeing
* to make sure there are no stale RBU packets left in memory
*/
memset ( newpacket - > data , 0 , rbu_data . packetsize ) ;
2005-09-17 06:28:04 +04:00
free_pages ( ( unsigned long ) newpacket - > data ,
newpacket - > ordernum ) ;
2005-09-07 02:17:14 +04:00
kfree ( newpacket ) ;
}
rbu_data . packet_read_count = 0 ;
rbu_data . num_packets = 0 ;
2005-10-11 19:29:02 +04:00
rbu_data . imagesize = 0 ;
2005-09-07 02:17:14 +04:00
}
/*
* img_update_free : Frees the buffer allocated for storing BIOS image
* Always called with lock held and returned with lock held
*/
2005-09-17 06:28:05 +04:00
static void img_update_free ( void )
2005-09-07 02:17:14 +04:00
{
if ( ! rbu_data . image_update_buffer )
return ;
/*
* zero out this buffer before freeing it to get rid of any stale
* BIOS image copied in memory .
*/
memset ( rbu_data . image_update_buffer , 0 ,
2005-09-17 06:28:04 +04:00
rbu_data . image_update_buffer_size ) ;
2005-09-07 02:17:14 +04:00
if ( rbu_data . dma_alloc = = 1 )
dma_free_coherent ( NULL , rbu_data . bios_image_size ,
2005-09-17 06:28:04 +04:00
rbu_data . image_update_buffer , dell_rbu_dmaaddr ) ;
2005-09-07 02:17:14 +04:00
else
2005-09-17 06:28:04 +04:00
free_pages ( ( unsigned long ) rbu_data . image_update_buffer ,
rbu_data . image_update_ordernum ) ;
2005-09-07 02:17:14 +04:00
/*
* Re - initialize the rbu_data variables after a free
*/
rbu_data . image_update_ordernum = - 1 ;
rbu_data . image_update_buffer = NULL ;
rbu_data . image_update_buffer_size = 0 ;
rbu_data . bios_image_size = 0 ;
rbu_data . dma_alloc = 0 ;
}
/*
* img_update_realloc : This function allocates the contiguous pages to
* accommodate the requested size of data . The memory address and size
* values are stored globally and on every call to this function the new
* size is checked to see if more data is required than the existing size .
* If true the previous memory is freed and new allocation is done to
* accommodate the new size . If the incoming size is less then than the
* already allocated size , then that memory is reused . This function is
* called with lock held and returns with lock held .
*/
2005-09-17 06:28:05 +04:00
static int img_update_realloc ( unsigned long size )
2005-09-07 02:17:14 +04:00
{
unsigned char * image_update_buffer = NULL ;
unsigned long rc ;
unsigned long img_buf_phys_addr ;
int ordernum ;
int dma_alloc = 0 ;
/*
* check if the buffer of sufficient size has been
* already allocated
*/
if ( rbu_data . image_update_buffer_size > = size ) {
/*
* check for corruption
*/
if ( ( size ! = 0 ) & & ( rbu_data . image_update_buffer = = NULL ) ) {
printk ( KERN_ERR " dell_rbu:%s: corruption "
2005-09-17 06:28:04 +04:00
" check failed \n " , __FUNCTION__ ) ;
2005-09-07 02:17:14 +04:00
return - EINVAL ;
}
/*
* we have a valid pre - allocated buffer with
* sufficient size
*/
return 0 ;
}
/*
* free any previously allocated buffer
*/
img_update_free ( ) ;
spin_unlock ( & rbu_data . lock ) ;
ordernum = get_order ( size ) ;
image_update_buffer =
2005-09-17 06:28:04 +04:00
( unsigned char * ) __get_free_pages ( GFP_KERNEL , ordernum ) ;
2005-09-07 02:17:14 +04:00
img_buf_phys_addr =
2005-09-17 06:28:04 +04:00
( unsigned long ) virt_to_phys ( image_update_buffer ) ;
2005-09-07 02:17:14 +04:00
if ( img_buf_phys_addr > BIOS_SCAN_LIMIT ) {
2005-09-17 06:28:04 +04:00
free_pages ( ( unsigned long ) image_update_buffer , ordernum ) ;
2005-09-07 02:17:14 +04:00
ordernum = - 1 ;
image_update_buffer = dma_alloc_coherent ( NULL , size ,
2005-09-17 06:28:04 +04:00
& dell_rbu_dmaaddr , GFP_KERNEL ) ;
2005-09-07 02:17:14 +04:00
dma_alloc = 1 ;
}
spin_lock ( & rbu_data . lock ) ;
if ( image_update_buffer ! = NULL ) {
rbu_data . image_update_buffer = image_update_buffer ;
rbu_data . image_update_buffer_size = size ;
rbu_data . bios_image_size =
2005-09-17 06:28:04 +04:00
rbu_data . image_update_buffer_size ;
2005-09-07 02:17:14 +04:00
rbu_data . image_update_ordernum = ordernum ;
rbu_data . dma_alloc = dma_alloc ;
rc = 0 ;
} else {
pr_debug ( " Not enough memory for image update: "
2005-09-17 06:28:04 +04:00
" size = %ld \n " , size ) ;
2005-09-07 02:17:14 +04:00
rc = - ENOMEM ;
}
return rc ;
}
2005-09-17 06:28:05 +04:00
static ssize_t read_packet_data ( char * buffer , loff_t pos , size_t count )
2005-09-07 02:17:14 +04:00
{
int retval ;
size_t bytes_left ;
size_t data_length ;
char * ptempBuf = buffer ;
/* check to see if we have something to return */
if ( rbu_data . num_packets = = 0 ) {
pr_debug ( " read_packet_data: no packets written \n " ) ;
retval = - ENOMEM ;
goto read_rbu_data_exit ;
}
2005-10-11 19:29:02 +04:00
if ( pos > rbu_data . imagesize ) {
2005-09-07 02:17:14 +04:00
retval = 0 ;
printk ( KERN_WARNING " dell_rbu:read_packet_data: "
2005-09-17 06:28:04 +04:00
" data underrun \n " ) ;
2005-09-07 02:17:14 +04:00
goto read_rbu_data_exit ;
}
2005-10-11 19:29:02 +04:00
bytes_left = rbu_data . imagesize - pos ;
2005-09-07 02:17:14 +04:00
data_length = min ( bytes_left , count ) ;
if ( ( retval = packet_read_list ( ptempBuf , & data_length ) ) < 0 )
goto read_rbu_data_exit ;
2005-10-11 19:29:02 +04:00
if ( ( pos + count ) > rbu_data . imagesize ) {
2005-09-07 02:17:14 +04:00
rbu_data . packet_read_count = 0 ;
/* this was the last copy */
retval = bytes_left ;
} else
retval = count ;
read_rbu_data_exit :
return retval ;
}
2005-09-17 06:28:05 +04:00
static ssize_t read_rbu_mono_data ( char * buffer , loff_t pos , size_t count )
2005-09-07 02:17:14 +04:00
{
unsigned char * ptemp = NULL ;
size_t bytes_left = 0 ;
size_t data_length = 0 ;
ssize_t ret_count = 0 ;
/* check to see if we have something to return */
if ( ( rbu_data . image_update_buffer = = NULL ) | |
2005-09-17 06:28:04 +04:00
( rbu_data . bios_image_size = = 0 ) ) {
2005-09-07 02:17:14 +04:00
pr_debug ( " read_rbu_data_mono: image_update_buffer %p , "
2005-09-17 06:28:04 +04:00
" bios_image_size %lu \n " ,
rbu_data . image_update_buffer ,
rbu_data . bios_image_size ) ;
2005-09-07 02:17:14 +04:00
ret_count = - ENOMEM ;
goto read_rbu_data_exit ;
}
if ( pos > rbu_data . bios_image_size ) {
ret_count = 0 ;
goto read_rbu_data_exit ;
}
bytes_left = rbu_data . bios_image_size - pos ;
data_length = min ( bytes_left , count ) ;
ptemp = rbu_data . image_update_buffer ;
memcpy ( buffer , ( ptemp + pos ) , data_length ) ;
if ( ( pos + count ) > rbu_data . bios_image_size )
/* this was the last copy */
ret_count = bytes_left ;
else
ret_count = count ;
read_rbu_data_exit :
return ret_count ;
}
2007-06-09 09:57:22 +04:00
static ssize_t read_rbu_data ( struct kobject * kobj ,
struct bin_attribute * bin_attr ,
char * buffer , loff_t pos , size_t count )
2005-09-07 02:17:14 +04:00
{
ssize_t ret_count = 0 ;
spin_lock ( & rbu_data . lock ) ;
if ( ! strcmp ( image_type , " mono " ) )
ret_count = read_rbu_mono_data ( buffer , pos , count ) ;
else if ( ! strcmp ( image_type , " packet " ) )
ret_count = read_packet_data ( buffer , pos , count ) ;
else
pr_debug ( " read_rbu_data: invalid image type specified \n " ) ;
spin_unlock ( & rbu_data . lock ) ;
return ret_count ;
}
2005-09-17 06:28:05 +04:00
static void callbackfn_rbu ( const struct firmware * fw , void * context )
2005-09-17 06:28:04 +04:00
{
2006-01-15 00:21:14 +03:00
rbu_data . entry_created = 0 ;
2005-09-17 06:28:04 +04:00
2006-01-15 00:21:14 +03:00
if ( ! fw | | ! fw - > size )
2005-09-17 06:28:04 +04:00
return ;
spin_lock ( & rbu_data . lock ) ;
if ( ! strcmp ( image_type , " mono " ) ) {
if ( ! img_update_realloc ( fw - > size ) )
memcpy ( rbu_data . image_update_buffer ,
fw - > data , fw - > size ) ;
} else if ( ! strcmp ( image_type , " packet " ) ) {
2005-10-11 19:29:02 +04:00
/*
* we need to free previous packets if a
* new hunk of packets needs to be downloaded
*/
packet_empty_list ( ) ;
if ( packetize_data ( fw - > data , fw - > size ) )
/* Incase something goes wrong when we are
* in middle of packetizing the data , we
* need to free up whatever packets might
* have been created before we quit .
*/
2005-09-17 06:28:04 +04:00
packet_empty_list ( ) ;
} else
pr_debug ( " invalid image type specified. \n " ) ;
spin_unlock ( & rbu_data . lock ) ;
}
2007-06-09 09:57:22 +04:00
static ssize_t read_rbu_image_type ( struct kobject * kobj ,
struct bin_attribute * bin_attr ,
char * buffer , loff_t pos , size_t count )
2005-09-07 02:17:14 +04:00
{
int size = 0 ;
if ( ! pos )
size = sprintf ( buffer , " %s \n " , image_type ) ;
return size ;
}
2007-06-09 09:57:22 +04:00
static ssize_t write_rbu_image_type ( struct kobject * kobj ,
struct bin_attribute * bin_attr ,
char * buffer , loff_t pos , size_t count )
2005-09-07 02:17:14 +04:00
{
int rc = count ;
2005-09-17 06:28:04 +04:00
int req_firm_rc = 0 ;
int i ;
2005-09-07 02:17:14 +04:00
spin_lock ( & rbu_data . lock ) ;
2005-09-17 06:28:04 +04:00
/*
* Find the first newline or space
*/
for ( i = 0 ; i < count ; + + i )
if ( buffer [ i ] = = ' \n ' | | buffer [ i ] = = ' ' ) {
buffer [ i ] = ' \0 ' ;
break ;
}
if ( i = = count )
buffer [ count ] = ' \0 ' ;
if ( strstr ( buffer , " mono " ) )
strcpy ( image_type , " mono " ) ;
else if ( strstr ( buffer , " packet " ) )
strcpy ( image_type , " packet " ) ;
else if ( strstr ( buffer , " init " ) ) {
/*
* If due to the user error the driver gets in a bad
* state where even though it is loaded , the
* / sys / class / firmware / dell_rbu entries are missing .
* to cover this situation the user can recreate entries
* by writing init to image_type .
*/
if ( ! rbu_data . entry_created ) {
spin_unlock ( & rbu_data . lock ) ;
req_firm_rc = request_firmware_nowait ( THIS_MODULE ,
FW_ACTION_NOHOTPLUG , " dell_rbu " ,
& rbu_device - > dev , & context ,
callbackfn_rbu ) ;
if ( req_firm_rc ) {
printk ( KERN_ERR
" dell_rbu:%s request_firmware_nowait "
" failed %d \n " , __FUNCTION__ , rc ) ;
rc = - EIO ;
} else
rbu_data . entry_created = 1 ;
spin_lock ( & rbu_data . lock ) ;
}
} else {
printk ( KERN_WARNING " dell_rbu: image_type is invalid \n " ) ;
spin_unlock ( & rbu_data . lock ) ;
return - EINVAL ;
}
2005-09-07 02:17:14 +04:00
/* we must free all previous allocations */
packet_empty_list ( ) ;
img_update_free ( ) ;
spin_unlock ( & rbu_data . lock ) ;
2005-09-17 06:28:04 +04:00
return rc ;
2005-09-07 02:17:14 +04:00
}
2007-06-09 09:57:22 +04:00
static ssize_t read_rbu_packet_size ( struct kobject * kobj ,
struct bin_attribute * bin_attr ,
char * buffer , loff_t pos , size_t count )
2005-10-11 19:29:02 +04:00
{
int size = 0 ;
if ( ! pos ) {
spin_lock ( & rbu_data . lock ) ;
size = sprintf ( buffer , " %lu \n " , rbu_data . packetsize ) ;
spin_unlock ( & rbu_data . lock ) ;
}
return size ;
}
2007-06-09 09:57:22 +04:00
static ssize_t write_rbu_packet_size ( struct kobject * kobj ,
struct bin_attribute * bin_attr ,
char * buffer , loff_t pos , size_t count )
2005-10-11 19:29:02 +04:00
{
unsigned long temp ;
spin_lock ( & rbu_data . lock ) ;
packet_empty_list ( ) ;
sscanf ( buffer , " %lu " , & temp ) ;
if ( temp < 0xffffffff )
rbu_data . packetsize = temp ;
spin_unlock ( & rbu_data . lock ) ;
return count ;
}
2005-09-07 02:17:14 +04:00
static struct bin_attribute rbu_data_attr = {
2007-06-13 22:45:17 +04:00
. attr = { . name = " data " , . mode = 0444 } ,
2005-09-07 02:17:14 +04:00
. read = read_rbu_data ,
} ;
static struct bin_attribute rbu_image_type_attr = {
2007-06-13 22:45:17 +04:00
. attr = { . name = " image_type " , . mode = 0644 } ,
2005-09-07 02:17:14 +04:00
. read = read_rbu_image_type ,
. write = write_rbu_image_type ,
} ;
2005-10-11 19:29:02 +04:00
static struct bin_attribute rbu_packet_size_attr = {
2007-06-13 22:45:17 +04:00
. attr = { . name = " packet_size " , . mode = 0644 } ,
2005-10-11 19:29:02 +04:00
. read = read_rbu_packet_size ,
. write = write_rbu_packet_size ,
} ;
2005-09-17 06:28:05 +04:00
static int __init dcdrbu_init ( void )
2005-09-07 02:17:14 +04:00
{
2006-11-16 12:19:25 +03:00
int rc ;
2005-09-07 02:17:14 +04:00
spin_lock_init ( & rbu_data . lock ) ;
init_packet_head ( ) ;
2006-11-16 12:19:25 +03:00
rbu_device = platform_device_register_simple ( " dell_rbu " , - 1 , NULL , 0 ) ;
if ( IS_ERR ( rbu_device ) ) {
2005-09-07 02:17:14 +04:00
printk ( KERN_ERR
2005-09-17 06:28:04 +04:00
" dell_rbu:%s:platform_device_register_simple "
" failed \n " , __FUNCTION__ ) ;
2006-11-16 12:19:25 +03:00
return PTR_ERR ( rbu_device ) ;
2005-09-07 02:17:14 +04:00
}
2006-10-11 12:22:20 +04:00
rc = sysfs_create_bin_file ( & rbu_device - > dev . kobj , & rbu_data_attr ) ;
if ( rc )
goto out_devreg ;
rc = sysfs_create_bin_file ( & rbu_device - > dev . kobj , & rbu_image_type_attr ) ;
if ( rc )
goto out_data ;
rc = sysfs_create_bin_file ( & rbu_device - > dev . kobj ,
2005-10-11 19:29:02 +04:00
& rbu_packet_size_attr ) ;
2006-10-11 12:22:20 +04:00
if ( rc )
goto out_imtype ;
2005-09-07 02:17:14 +04:00
2006-01-15 00:21:14 +03:00
rbu_data . entry_created = 0 ;
2006-10-11 12:22:20 +04:00
return 0 ;
2005-09-07 02:17:14 +04:00
2006-10-11 12:22:20 +04:00
out_imtype :
sysfs_remove_bin_file ( & rbu_device - > dev . kobj , & rbu_image_type_attr ) ;
out_data :
sysfs_remove_bin_file ( & rbu_device - > dev . kobj , & rbu_data_attr ) ;
out_devreg :
platform_device_unregister ( rbu_device ) ;
return rc ;
2005-09-07 02:17:14 +04:00
}
2005-09-17 06:28:05 +04:00
static __exit void dcdrbu_exit ( void )
2005-09-07 02:17:14 +04:00
{
spin_lock ( & rbu_data . lock ) ;
packet_empty_list ( ) ;
img_update_free ( ) ;
spin_unlock ( & rbu_data . lock ) ;
platform_device_unregister ( rbu_device ) ;
}
module_exit ( dcdrbu_exit ) ;
module_init ( dcdrbu_init ) ;
2005-11-07 11:59:26 +03:00
/* vim:noet:ts=8:sw=8
*/