2005-04-16 15:20:36 -07:00
/*
* linux / fs / hpfs / alloc . c
*
* Mikulas Patocka ( mikulas @ artax . karlin . mff . cuni . cz ) , 1998 - 1999
*
* HPFS bitmap operations
*/
# include "hpfs_fn.h"
2014-01-29 00:10:44 +01:00
static void hpfs_claim_alloc ( struct super_block * s , secno sec )
{
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
if ( sbi - > sb_n_free ! = ( unsigned ) - 1 ) {
if ( unlikely ( ! sbi - > sb_n_free ) ) {
hpfs_error ( s , " free count underflow, allocating sector %08x " , sec ) ;
sbi - > sb_n_free = - 1 ;
return ;
}
sbi - > sb_n_free - - ;
}
}
static void hpfs_claim_free ( struct super_block * s , secno sec )
{
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
if ( sbi - > sb_n_free ! = ( unsigned ) - 1 ) {
if ( unlikely ( sbi - > sb_n_free > = sbi - > sb_fs_size ) ) {
hpfs_error ( s , " free count overflow, freeing sector %08x " , sec ) ;
sbi - > sb_n_free = - 1 ;
return ;
}
sbi - > sb_n_free + + ;
}
}
static void hpfs_claim_dirband_alloc ( struct super_block * s , secno sec )
{
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
if ( sbi - > sb_n_free_dnodes ! = ( unsigned ) - 1 ) {
if ( unlikely ( ! sbi - > sb_n_free_dnodes ) ) {
hpfs_error ( s , " dirband free count underflow, allocating sector %08x " , sec ) ;
sbi - > sb_n_free_dnodes = - 1 ;
return ;
}
sbi - > sb_n_free_dnodes - - ;
}
}
static void hpfs_claim_dirband_free ( struct super_block * s , secno sec )
{
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
if ( sbi - > sb_n_free_dnodes ! = ( unsigned ) - 1 ) {
if ( unlikely ( sbi - > sb_n_free_dnodes > = sbi - > sb_dirband_size / 4 ) ) {
hpfs_error ( s , " dirband free count overflow, freeing sector %08x " , sec ) ;
sbi - > sb_n_free_dnodes = - 1 ;
return ;
}
sbi - > sb_n_free_dnodes + + ;
}
}
2005-04-16 15:20:36 -07:00
/*
* Check if a sector is allocated in bitmap
* This is really slow . Turned on only if chk = = 2
*/
static int chk_if_allocated ( struct super_block * s , secno sec , char * msg )
{
struct quad_buffer_head qbh ;
2012-04-17 15:28:51 -04:00
__le32 * bmp ;
2005-04-16 15:20:36 -07:00
if ( ! ( bmp = hpfs_map_bitmap ( s , sec > > 14 , & qbh , " chk " ) ) ) goto fail ;
2012-04-17 15:28:51 -04:00
if ( ( le32_to_cpu ( bmp [ ( sec & 0x3fff ) > > 5 ] ) > > ( sec & 0x1f ) ) & 1 ) {
2005-04-16 15:20:36 -07:00
hpfs_error ( s , " sector '%s' - %08x not allocated in bitmap " , msg , sec ) ;
goto fail1 ;
}
hpfs_brelse4 ( & qbh ) ;
if ( sec > = hpfs_sb ( s ) - > sb_dirband_start & & sec < hpfs_sb ( s ) - > sb_dirband_start + hpfs_sb ( s ) - > sb_dirband_size ) {
unsigned ssec = ( sec - hpfs_sb ( s ) - > sb_dirband_start ) / 4 ;
if ( ! ( bmp = hpfs_map_dnode_bitmap ( s , & qbh ) ) ) goto fail ;
2011-05-08 20:44:26 +02:00
if ( ( le32_to_cpu ( bmp [ ssec > > 5 ] ) > > ( ssec & 0x1f ) ) & 1 ) {
2005-04-16 15:20:36 -07:00
hpfs_error ( s , " sector '%s' - %08x not allocated in directory bitmap " , msg , sec ) ;
goto fail1 ;
}
hpfs_brelse4 ( & qbh ) ;
}
return 0 ;
fail1 :
hpfs_brelse4 ( & qbh ) ;
fail :
return 1 ;
}
/*
* Check if sector ( s ) have proper number and additionally check if they ' re
* allocated in bitmap .
*/
int hpfs_chk_sectors ( struct super_block * s , secno start , int len , char * msg )
{
if ( start + len < start | | start < 0x12 | |
start + len > hpfs_sb ( s ) - > sb_fs_size ) {
hpfs_error ( s , " sector(s) '%s' badly placed at %08x " , msg , start ) ;
return 1 ;
}
if ( hpfs_sb ( s ) - > sb_chk > = 2 ) {
int i ;
for ( i = 0 ; i < len ; i + + )
if ( chk_if_allocated ( s , start + i , msg ) ) return 1 ;
}
return 0 ;
}
static secno alloc_in_bmp ( struct super_block * s , secno near , unsigned n , unsigned forward )
{
struct quad_buffer_head qbh ;
2012-04-17 15:28:51 -04:00
__le32 * bmp ;
2005-04-16 15:20:36 -07:00
unsigned bs = near & ~ 0x3fff ;
unsigned nr = ( near & 0x3fff ) & ~ ( n - 1 ) ;
/*unsigned mnr;*/
unsigned i , q ;
int a , b ;
secno ret = 0 ;
if ( n ! = 1 & & n ! = 4 ) {
hpfs_error ( s , " Bad allocation size: %d " , n ) ;
return 0 ;
}
if ( bs ! = ~ 0x3fff ) {
if ( ! ( bmp = hpfs_map_bitmap ( s , near > > 14 , & qbh , " aib " ) ) ) goto uls ;
} else {
if ( ! ( bmp = hpfs_map_dnode_bitmap ( s , & qbh ) ) ) goto uls ;
}
if ( ! tstbits ( bmp , nr , n + forward ) ) {
ret = bs + nr ;
goto rt ;
}
q = nr + n ; b = 0 ;
while ( ( a = tstbits ( bmp , q , n + forward ) ) ! = 0 ) {
q + = a ;
if ( n ! = 1 ) q = ( ( q - 1 ) & ~ ( n - 1 ) ) + n ;
if ( ! b ) {
if ( q > > 5 ! = nr > > 5 ) {
b = 1 ;
q = nr & 0x1f ;
}
} else if ( q > nr ) break ;
}
if ( ! a ) {
ret = bs + q ;
goto rt ;
}
nr > > = 5 ;
2011-05-08 20:44:26 +02:00
/*for (i = nr + 1; i != nr; i++, i &= 0x1ff) */
2005-04-16 15:20:36 -07:00
i = nr ;
do {
2011-05-08 20:44:26 +02:00
if ( ! le32_to_cpu ( bmp [ i ] ) ) goto cont ;
if ( n + forward > = 0x3f & & le32_to_cpu ( bmp [ i ] ) ! = 0xffffffff ) goto cont ;
2005-04-16 15:20:36 -07:00
q = i < < 5 ;
if ( i > 0 ) {
2011-05-08 20:44:26 +02:00
unsigned k = le32_to_cpu ( bmp [ i - 1 ] ) ;
2005-04-16 15:20:36 -07:00
while ( k & 0x80000000 ) {
q - - ; k < < = 1 ;
}
}
if ( n ! = 1 ) q = ( ( q - 1 ) & ~ ( n - 1 ) ) + n ;
while ( ( a = tstbits ( bmp , q , n + forward ) ) ! = 0 ) {
q + = a ;
if ( n ! = 1 ) q = ( ( q - 1 ) & ~ ( n - 1 ) ) + n ;
if ( q > > 5 > i ) break ;
}
if ( ! a ) {
ret = bs + q ;
goto rt ;
}
cont :
i + + , i & = 0x1ff ;
} while ( i ! = nr ) ;
rt :
if ( ret ) {
2011-05-08 20:44:26 +02:00
if ( hpfs_sb ( s ) - > sb_chk & & ( ( ret > > 14 ) ! = ( bs > > 14 ) | | ( le32_to_cpu ( bmp [ ( ret & 0x3fff ) > > 5 ] ) | ~ ( ( ( 1 < < n ) - 1 ) < < ( ret & 0x1f ) ) ) ! = 0xffffffff ) ) {
2005-04-16 15:20:36 -07:00
hpfs_error ( s , " Allocation doesn't work! Wanted %d, allocated at %08x " , n , ret ) ;
ret = 0 ;
goto b ;
}
2011-05-08 20:44:26 +02:00
bmp [ ( ret & 0x3fff ) > > 5 ] & = cpu_to_le32 ( ~ ( ( ( 1 < < n ) - 1 ) < < ( ret & 0x1f ) ) ) ;
2005-04-16 15:20:36 -07:00
hpfs_mark_4buffers_dirty ( & qbh ) ;
}
b :
hpfs_brelse4 ( & qbh ) ;
uls :
return ret ;
}
/*
* Allocation strategy : 1 ) search place near the sector specified
* 2 ) search bitmap where free sectors last found
* 3 ) search all bitmaps
* 4 ) search all bitmaps ignoring number of pre - allocated
* sectors
*/
2011-05-08 20:43:06 +02:00
secno hpfs_alloc_sector ( struct super_block * s , secno near , unsigned n , int forward )
2005-04-16 15:20:36 -07:00
{
secno sec ;
int i ;
unsigned n_bmps ;
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
int f_p = 0 ;
int near_bmp ;
if ( forward < 0 ) {
forward = - forward ;
f_p = 1 ;
}
n_bmps = ( sbi - > sb_fs_size + 0x4000 - 1 ) > > 14 ;
if ( near & & near < sbi - > sb_fs_size ) {
if ( ( sec = alloc_in_bmp ( s , near , n , f_p ? forward : forward / 4 ) ) ) goto ret ;
near_bmp = near > > 14 ;
} else near_bmp = n_bmps / 2 ;
/*
if ( b ! = - 1 ) {
if ( ( sec = alloc_in_bmp ( s , b < < 14 , n , f_p ? forward : forward / 2 ) ) ) {
b & = 0x0fffffff ;
goto ret ;
}
if ( b > 0x10000000 ) if ( ( sec = alloc_in_bmp ( s , ( b & 0xfffffff ) < < 14 , n , f_p ? forward : 0 ) ) ) goto ret ;
*/
if ( ! f_p ) if ( forward > sbi - > sb_max_fwd_alloc ) forward = sbi - > sb_max_fwd_alloc ;
less_fwd :
for ( i = 0 ; i < n_bmps ; i + + ) {
if ( near_bmp + i < n_bmps & & ( ( sec = alloc_in_bmp ( s , ( near_bmp + i ) < < 14 , n , forward ) ) ) ) {
sbi - > sb_c_bitmap = near_bmp + i ;
goto ret ;
}
if ( ! forward ) {
if ( near_bmp - i - 1 > = 0 & & ( ( sec = alloc_in_bmp ( s , ( near_bmp - i - 1 ) < < 14 , n , forward ) ) ) ) {
sbi - > sb_c_bitmap = near_bmp - i - 1 ;
goto ret ;
}
} else {
if ( near_bmp + i > = n_bmps & & ( ( sec = alloc_in_bmp ( s , ( near_bmp + i - n_bmps ) < < 14 , n , forward ) ) ) ) {
sbi - > sb_c_bitmap = near_bmp + i - n_bmps ;
goto ret ;
}
}
if ( i = = 1 & & sbi - > sb_c_bitmap ! = - 1 & & ( ( sec = alloc_in_bmp ( s , ( sbi - > sb_c_bitmap ) < < 14 , n , forward ) ) ) ) {
goto ret ;
}
}
if ( ! f_p ) {
if ( forward ) {
sbi - > sb_max_fwd_alloc = forward * 3 / 4 ;
forward / = 2 ;
goto less_fwd ;
}
}
sec = 0 ;
ret :
2014-01-29 00:10:44 +01:00
if ( sec ) {
i = 0 ;
do
hpfs_claim_alloc ( s , sec + i ) ;
while ( unlikely ( + + i < n ) ) ;
}
2005-04-16 15:20:36 -07:00
if ( sec & & f_p ) {
for ( i = 0 ; i < forward ; i + + ) {
2014-01-29 00:10:44 +01:00
if ( ! hpfs_alloc_if_possible ( s , sec + n + i ) ) {
2005-04-16 15:20:36 -07:00
hpfs_error ( s , " Prealloc doesn't work! Wanted %d, allocated at %08x, can't allocate %d " , forward , sec , i ) ;
sec = 0 ;
break ;
}
}
}
return sec ;
}
2011-05-08 20:43:06 +02:00
static secno alloc_in_dirband ( struct super_block * s , secno near )
2005-04-16 15:20:36 -07:00
{
unsigned nr = near ;
secno sec ;
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
if ( nr < sbi - > sb_dirband_start )
nr = sbi - > sb_dirband_start ;
if ( nr > = sbi - > sb_dirband_start + sbi - > sb_dirband_size )
nr = sbi - > sb_dirband_start + sbi - > sb_dirband_size - 4 ;
nr - = sbi - > sb_dirband_start ;
nr > > = 2 ;
sec = alloc_in_bmp ( s , ( ~ 0x3fff ) | nr , 1 , 0 ) ;
if ( ! sec ) return 0 ;
2014-01-29 00:10:44 +01:00
hpfs_claim_dirband_alloc ( s , sec ) ;
2005-04-16 15:20:36 -07:00
return ( ( sec & 0x3fff ) < < 2 ) + sbi - > sb_dirband_start ;
}
/* Alloc sector if it's free */
2011-05-08 20:43:06 +02:00
int hpfs_alloc_if_possible ( struct super_block * s , secno sec )
2005-04-16 15:20:36 -07:00
{
struct quad_buffer_head qbh ;
2012-04-17 15:28:51 -04:00
__le32 * bmp ;
2005-04-16 15:20:36 -07:00
if ( ! ( bmp = hpfs_map_bitmap ( s , sec > > 14 , & qbh , " aip " ) ) ) goto end ;
2011-05-08 20:44:26 +02:00
if ( le32_to_cpu ( bmp [ ( sec & 0x3fff ) > > 5 ] ) & ( 1 < < ( sec & 0x1f ) ) ) {
bmp [ ( sec & 0x3fff ) > > 5 ] & = cpu_to_le32 ( ~ ( 1 < < ( sec & 0x1f ) ) ) ;
2005-04-16 15:20:36 -07:00
hpfs_mark_4buffers_dirty ( & qbh ) ;
hpfs_brelse4 ( & qbh ) ;
2014-01-29 00:10:44 +01:00
hpfs_claim_alloc ( s , sec ) ;
2005-04-16 15:20:36 -07:00
return 1 ;
}
hpfs_brelse4 ( & qbh ) ;
end :
return 0 ;
}
/* Free sectors in bitmaps */
void hpfs_free_sectors ( struct super_block * s , secno sec , unsigned n )
{
struct quad_buffer_head qbh ;
2012-04-17 15:28:51 -04:00
__le32 * bmp ;
2005-04-16 15:20:36 -07:00
struct hpfs_sb_info * sbi = hpfs_sb ( s ) ;
2014-06-06 14:36:34 -07:00
/*pr_info("2 - ");*/
2005-04-16 15:20:36 -07:00
if ( ! n ) return ;
if ( sec < 0x12 ) {
hpfs_error ( s , " Trying to free reserved sector %08x " , sec ) ;
return ;
}
sbi - > sb_max_fwd_alloc + = n > 0xffff ? 0xffff : n ;
if ( sbi - > sb_max_fwd_alloc > 0xffffff ) sbi - > sb_max_fwd_alloc = 0xffffff ;
new_map :
if ( ! ( bmp = hpfs_map_bitmap ( s , sec > > 14 , & qbh , " free " ) ) ) {
return ;
}
new_tst :
2011-05-08 20:44:26 +02:00
if ( ( le32_to_cpu ( bmp [ ( sec & 0x3fff ) > > 5 ] ) > > ( sec & 0x1f ) & 1 ) ) {
2005-04-16 15:20:36 -07:00
hpfs_error ( s , " sector %08x not allocated " , sec ) ;
hpfs_brelse4 ( & qbh ) ;
return ;
}
2011-05-08 20:44:26 +02:00
bmp [ ( sec & 0x3fff ) > > 5 ] | = cpu_to_le32 ( 1 < < ( sec & 0x1f ) ) ;
2014-01-29 00:10:44 +01:00
hpfs_claim_free ( s , sec ) ;
2005-04-16 15:20:36 -07:00
if ( ! - - n ) {
hpfs_mark_4buffers_dirty ( & qbh ) ;
hpfs_brelse4 ( & qbh ) ;
return ;
}
if ( ! ( + + sec & 0x3fff ) ) {
hpfs_mark_4buffers_dirty ( & qbh ) ;
hpfs_brelse4 ( & qbh ) ;
goto new_map ;
}
goto new_tst ;
}
/*
* Check if there are at least n free dnodes on the filesystem .
* Called before adding to dnode . If we run out of space while
* splitting dnodes , it would corrupt dnode tree .
*/
int hpfs_check_free_dnodes ( struct super_block * s , int n )
{
int n_bmps = ( hpfs_sb ( s ) - > sb_fs_size + 0x4000 - 1 ) > > 14 ;
int b = hpfs_sb ( s ) - > sb_c_bitmap & 0x0fffffff ;
int i , j ;
2012-04-17 15:28:51 -04:00
__le32 * bmp ;
2005-04-16 15:20:36 -07:00
struct quad_buffer_head qbh ;
if ( ( bmp = hpfs_map_dnode_bitmap ( s , & qbh ) ) ) {
for ( j = 0 ; j < 512 ; j + + ) {
unsigned k ;
2011-05-08 20:44:26 +02:00
if ( ! le32_to_cpu ( bmp [ j ] ) ) continue ;
for ( k = le32_to_cpu ( bmp [ j ] ) ; k ; k > > = 1 ) if ( k & 1 ) if ( ! - - n ) {
2005-04-16 15:20:36 -07:00
hpfs_brelse4 ( & qbh ) ;
return 0 ;
}
}
}
hpfs_brelse4 ( & qbh ) ;
i = 0 ;
if ( hpfs_sb ( s ) - > sb_c_bitmap ! = - 1 ) {
bmp = hpfs_map_bitmap ( s , b , & qbh , " chkdn1 " ) ;
goto chk_bmp ;
}
chk_next :
if ( i = = b ) i + + ;
if ( i > = n_bmps ) return 1 ;
bmp = hpfs_map_bitmap ( s , i , & qbh , " chkdn2 " ) ;
chk_bmp :
if ( bmp ) {
for ( j = 0 ; j < 512 ; j + + ) {
2011-05-08 20:44:26 +02:00
u32 k ;
if ( ! le32_to_cpu ( bmp [ j ] ) ) continue ;
2005-04-16 15:20:36 -07:00
for ( k = 0xf ; k ; k < < = 4 )
2011-05-08 20:44:26 +02:00
if ( ( le32_to_cpu ( bmp [ j ] ) & k ) = = k ) {
2005-04-16 15:20:36 -07:00
if ( ! - - n ) {
hpfs_brelse4 ( & qbh ) ;
return 0 ;
}
}
}
hpfs_brelse4 ( & qbh ) ;
}
i + + ;
goto chk_next ;
}
void hpfs_free_dnode ( struct super_block * s , dnode_secno dno )
{
if ( hpfs_sb ( s ) - > sb_chk ) if ( dno & 3 ) {
hpfs_error ( s , " hpfs_free_dnode: dnode %08x not aligned " , dno ) ;
return ;
}
if ( dno < hpfs_sb ( s ) - > sb_dirband_start | |
dno > = hpfs_sb ( s ) - > sb_dirband_start + hpfs_sb ( s ) - > sb_dirband_size ) {
hpfs_free_sectors ( s , dno , 4 ) ;
} else {
struct quad_buffer_head qbh ;
2012-04-17 15:28:51 -04:00
__le32 * bmp ;
2005-04-16 15:20:36 -07:00
unsigned ssec = ( dno - hpfs_sb ( s ) - > sb_dirband_start ) / 4 ;
if ( ! ( bmp = hpfs_map_dnode_bitmap ( s , & qbh ) ) ) {
return ;
}
2011-05-08 20:44:26 +02:00
bmp [ ssec > > 5 ] | = cpu_to_le32 ( 1 < < ( ssec & 0x1f ) ) ;
2005-04-16 15:20:36 -07:00
hpfs_mark_4buffers_dirty ( & qbh ) ;
hpfs_brelse4 ( & qbh ) ;
2014-01-29 00:10:44 +01:00
hpfs_claim_dirband_free ( s , dno ) ;
2005-04-16 15:20:36 -07:00
}
}
struct dnode * hpfs_alloc_dnode ( struct super_block * s , secno near ,
2011-05-08 20:43:06 +02:00
dnode_secno * dno , struct quad_buffer_head * qbh )
2005-04-16 15:20:36 -07:00
{
struct dnode * d ;
2014-01-29 00:10:44 +01:00
if ( hpfs_get_free_dnodes ( s ) > FREE_DNODES_ADD ) {
2011-05-08 20:43:06 +02:00
if ( ! ( * dno = alloc_in_dirband ( s , near ) ) )
if ( ! ( * dno = hpfs_alloc_sector ( s , near , 4 , 0 ) ) ) return NULL ;
2005-04-16 15:20:36 -07:00
} else {
2011-05-08 20:43:06 +02:00
if ( ! ( * dno = hpfs_alloc_sector ( s , near , 4 , 0 ) ) )
if ( ! ( * dno = alloc_in_dirband ( s , near ) ) ) return NULL ;
2005-04-16 15:20:36 -07:00
}
if ( ! ( d = hpfs_get_4sectors ( s , * dno , qbh ) ) ) {
hpfs_free_dnode ( s , * dno ) ;
return NULL ;
}
memset ( d , 0 , 2048 ) ;
2011-05-08 20:44:26 +02:00
d - > magic = cpu_to_le32 ( DNODE_MAGIC ) ;
d - > first_free = cpu_to_le32 ( 52 ) ;
2005-04-16 15:20:36 -07:00
d - > dirent [ 0 ] = 32 ;
d - > dirent [ 2 ] = 8 ;
d - > dirent [ 30 ] = 1 ;
d - > dirent [ 31 ] = 255 ;
2011-05-08 20:44:26 +02:00
d - > self = cpu_to_le32 ( * dno ) ;
2005-04-16 15:20:36 -07:00
return d ;
}
struct fnode * hpfs_alloc_fnode ( struct super_block * s , secno near , fnode_secno * fno ,
struct buffer_head * * bh )
{
struct fnode * f ;
2011-05-08 20:43:06 +02:00
if ( ! ( * fno = hpfs_alloc_sector ( s , near , 1 , FNODE_ALLOC_FWD ) ) ) return NULL ;
2005-04-16 15:20:36 -07:00
if ( ! ( f = hpfs_get_sector ( s , * fno , bh ) ) ) {
hpfs_free_sectors ( s , * fno , 1 ) ;
return NULL ;
}
memset ( f , 0 , 512 ) ;
2011-05-08 20:44:26 +02:00
f - > magic = cpu_to_le32 ( FNODE_MAGIC ) ;
f - > ea_offs = cpu_to_le16 ( 0xc4 ) ;
2005-04-16 15:20:36 -07:00
f - > btree . n_free_nodes = 8 ;
2011-05-08 20:44:26 +02:00
f - > btree . first_free = cpu_to_le16 ( 8 ) ;
2005-04-16 15:20:36 -07:00
return f ;
}
struct anode * hpfs_alloc_anode ( struct super_block * s , secno near , anode_secno * ano ,
struct buffer_head * * bh )
{
struct anode * a ;
2011-05-08 20:43:06 +02:00
if ( ! ( * ano = hpfs_alloc_sector ( s , near , 1 , ANODE_ALLOC_FWD ) ) ) return NULL ;
2005-04-16 15:20:36 -07:00
if ( ! ( a = hpfs_get_sector ( s , * ano , bh ) ) ) {
hpfs_free_sectors ( s , * ano , 1 ) ;
return NULL ;
}
memset ( a , 0 , 512 ) ;
2011-05-08 20:44:26 +02:00
a - > magic = cpu_to_le32 ( ANODE_MAGIC ) ;
a - > self = cpu_to_le32 ( * ano ) ;
2005-04-16 15:20:36 -07:00
a - > btree . n_free_nodes = 40 ;
a - > btree . n_used_nodes = 0 ;
2011-05-08 20:44:26 +02:00
a - > btree . first_free = cpu_to_le16 ( 8 ) ;
2005-04-16 15:20:36 -07:00
return a ;
}