2005-04-16 15:20:36 -07:00
/*
* Copyright ( c ) 2000 - 2001 Christoph Hellwig .
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions , and the following disclaimer ,
* without modification .
* 2. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission .
*
* Alternatively , this software may be distributed under the terms of the
* GNU General Public License ( " GPL " ) .
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ` ` AS IS ' ' AND
* ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED . IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
* ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
* DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT
* LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE .
*/
/*
* Veritas filesystem driver - filesystem to disk block mapping .
*/
# include <linux/fs.h>
# include <linux/buffer_head.h>
# include <linux/kernel.h>
# include "vxfs.h"
# include "vxfs_inode.h"
2005-11-08 16:47:45 +01:00
# include "vxfs_extern.h"
2005-04-16 15:20:36 -07:00
# ifdef DIAGNOSTIC
static void
vxfs_typdump ( struct vxfs_typed * typ )
{
printk ( KERN_DEBUG " type=%Lu " , typ - > vt_hdr > > VXFS_TYPED_TYPESHIFT ) ;
printk ( " offset=%Lx " , typ - > vt_hdr & VXFS_TYPED_OFFSETMASK ) ;
printk ( " block=%x " , typ - > vt_block ) ;
printk ( " size=%x \n " , typ - > vt_size ) ;
}
# endif
/**
* vxfs_bmap_ext4 - do bmap for ext4 extents
* @ ip : pointer to the inode we do bmap for
* @ iblock : logical block .
*
* Description :
* vxfs_bmap_ext4 performs the bmap operation for inodes with
* ext4 - style extents ( which are much like the traditional UNIX
* inode organisation ) .
*
* Returns :
* The physical block number on success , else Zero .
*/
static daddr_t
vxfs_bmap_ext4 ( struct inode * ip , long bn )
{
struct super_block * sb = ip - > i_sb ;
struct vxfs_inode_info * vip = VXFS_INO ( ip ) ;
2016-05-31 08:45:13 +02:00
struct vxfs_sb_info * sbi = VXFS_SBI ( sb ) ;
2005-04-16 15:20:36 -07:00
unsigned long bsize = sb - > s_blocksize ;
2016-05-31 08:45:13 +02:00
u32 indsize = fs32_to_cpu ( sbi , vip - > vii_ext4 . ve4_indsize ) ;
2005-04-16 15:20:36 -07:00
int i ;
if ( indsize > sb - > s_blocksize )
goto fail_size ;
for ( i = 0 ; i < VXFS_NDADDR ; i + + ) {
struct direct * d = vip - > vii_ext4 . ve4_direct + i ;
2016-05-31 08:45:13 +02:00
if ( bn > = 0 & & bn < fs32_to_cpu ( sbi , d - > size ) )
return ( bn + fs32_to_cpu ( sbi , d - > extent ) ) ;
bn - = fs32_to_cpu ( sbi , d - > size ) ;
2005-04-16 15:20:36 -07:00
}
if ( ( bn / ( indsize * indsize * bsize / 4 ) ) = = 0 ) {
struct buffer_head * buf ;
daddr_t bno ;
2016-05-31 08:45:13 +02:00
__fs32 * indir ;
2005-04-16 15:20:36 -07:00
2016-05-31 08:45:13 +02:00
buf = sb_bread ( sb ,
fs32_to_cpu ( sbi , vip - > vii_ext4 . ve4_indir [ 0 ] ) ) ;
2005-04-16 15:20:36 -07:00
if ( ! buf | | ! buffer_mapped ( buf ) )
goto fail_buf ;
2016-05-31 08:45:13 +02:00
indir = ( __fs32 * ) buf - > b_data ;
bno = fs32_to_cpu ( sbi , indir [ ( bn / indsize ) % ( indsize * bn ) ] ) +
( bn % indsize ) ;
2005-04-16 15:20:36 -07:00
brelse ( buf ) ;
return bno ;
} else
printk ( KERN_WARNING " no matching indir? " ) ;
return 0 ;
fail_size :
2005-06-30 02:59:05 -07:00
printk ( " vxfs: indirect extent too big! \n " ) ;
2005-04-16 15:20:36 -07:00
fail_buf :
return 0 ;
}
/**
* vxfs_bmap_indir - recursion for vxfs_bmap_typed
* @ ip : pointer to the inode we do bmap for
* @ indir : indirect block we start reading at
* @ size : size of the typed area to search
* @ block : partially result from further searches
*
* Description :
* vxfs_bmap_indir reads a & struct vxfs_typed at @ indir
* and performs the type - defined action .
*
* Return Value :
* The physical block number on success , else Zero .
*
* Note :
* Kernelstack is rare . Unrecurse ?
*/
static daddr_t
vxfs_bmap_indir ( struct inode * ip , long indir , int size , long block )
{
2016-05-31 08:45:13 +02:00
struct vxfs_sb_info * sbi = VXFS_SBI ( ip - > i_sb ) ;
2005-04-16 15:20:36 -07:00
struct buffer_head * bp = NULL ;
daddr_t pblock = 0 ;
int i ;
for ( i = 0 ; i < size * VXFS_TYPED_PER_BLOCK ( ip - > i_sb ) ; i + + ) {
struct vxfs_typed * typ ;
int64_t off ;
bp = sb_bread ( ip - > i_sb ,
indir + ( i / VXFS_TYPED_PER_BLOCK ( ip - > i_sb ) ) ) ;
2007-05-08 00:24:34 -07:00
if ( ! bp | | ! buffer_mapped ( bp ) )
2005-04-16 15:20:36 -07:00
return 0 ;
typ = ( ( struct vxfs_typed * ) bp - > b_data ) +
( i % VXFS_TYPED_PER_BLOCK ( ip - > i_sb ) ) ;
2016-05-31 08:45:13 +02:00
off = fs64_to_cpu ( sbi , typ - > vt_hdr ) & VXFS_TYPED_OFFSETMASK ;
2005-04-16 15:20:36 -07:00
if ( block < off ) {
brelse ( bp ) ;
continue ;
}
2016-05-31 08:45:13 +02:00
switch ( ( u_int32_t ) ( fs64_to_cpu ( sbi , typ - > vt_hdr ) > >
VXFS_TYPED_TYPESHIFT ) ) {
2005-04-16 15:20:36 -07:00
case VXFS_TYPED_INDIRECT :
2016-05-31 08:45:13 +02:00
pblock = vxfs_bmap_indir ( ip ,
fs32_to_cpu ( sbi , typ - > vt_block ) ,
fs32_to_cpu ( sbi , typ - > vt_size ) ,
block - off ) ;
2005-04-16 15:20:36 -07:00
if ( pblock = = - 2 )
break ;
goto out ;
case VXFS_TYPED_DATA :
2016-05-31 08:45:13 +02:00
if ( ( block - off ) > = fs32_to_cpu ( sbi , typ - > vt_size ) )
2005-04-16 15:20:36 -07:00
break ;
2016-05-31 08:45:13 +02:00
pblock = fs32_to_cpu ( sbi , typ - > vt_block ) + block - off ;
2005-04-16 15:20:36 -07:00
goto out ;
case VXFS_TYPED_INDIRECT_DEV4 :
case VXFS_TYPED_DATA_DEV4 : {
struct vxfs_typed_dev4 * typ4 =
( struct vxfs_typed_dev4 * ) typ ;
printk ( KERN_INFO " \n \n TYPED_DEV4 detected! \n " ) ;
2016-05-31 08:45:13 +02:00
printk ( KERN_INFO " block: %llu \t size: %lld \t dev: %d \n " ,
fs64_to_cpu ( sbi , typ4 - > vd4_block ) ,
fs64_to_cpu ( sbi , typ4 - > vd4_size ) ,
fs32_to_cpu ( sbi , typ4 - > vd4_dev ) ) ;
2005-04-16 15:20:36 -07:00
goto fail ;
}
default :
2016-05-31 08:45:13 +02:00
printk ( KERN_ERR " %s:%d vt_hdr %llu \n " , __func__ ,
__LINE__ , fs64_to_cpu ( sbi , typ - > vt_hdr ) ) ;
2005-04-16 15:20:36 -07:00
BUG ( ) ;
}
brelse ( bp ) ;
}
fail :
pblock = 0 ;
out :
brelse ( bp ) ;
return ( pblock ) ;
}
/**
* vxfs_bmap_typed - bmap for typed extents
* @ ip : pointer to the inode we do bmap for
* @ iblock : logical block
*
* Description :
* Performs the bmap operation for typed extents .
*
* Return Value :
* The physical block number on success , else Zero .
*/
static daddr_t
vxfs_bmap_typed ( struct inode * ip , long iblock )
{
struct vxfs_inode_info * vip = VXFS_INO ( ip ) ;
2016-05-31 08:45:13 +02:00
struct vxfs_sb_info * sbi = VXFS_SBI ( ip - > i_sb ) ;
2005-04-16 15:20:36 -07:00
daddr_t pblock = 0 ;
int i ;
for ( i = 0 ; i < VXFS_NTYPED ; i + + ) {
struct vxfs_typed * typ = vip - > vii_org . typed + i ;
2016-05-31 08:45:13 +02:00
u64 hdr = fs64_to_cpu ( sbi , typ - > vt_hdr ) ;
int64_t off = ( hdr & VXFS_TYPED_OFFSETMASK ) ;
2005-04-16 15:20:36 -07:00
# ifdef DIAGNOSTIC
vxfs_typdump ( typ ) ;
# endif
if ( iblock < off )
continue ;
2016-05-31 08:45:13 +02:00
switch ( ( u32 ) ( hdr > > VXFS_TYPED_TYPESHIFT ) ) {
2005-04-16 15:20:36 -07:00
case VXFS_TYPED_INDIRECT :
2016-05-31 08:45:13 +02:00
pblock = vxfs_bmap_indir ( ip ,
fs32_to_cpu ( sbi , typ - > vt_block ) ,
fs32_to_cpu ( sbi , typ - > vt_size ) ,
iblock - off ) ;
2005-04-16 15:20:36 -07:00
if ( pblock = = - 2 )
break ;
return ( pblock ) ;
case VXFS_TYPED_DATA :
2016-05-31 08:45:13 +02:00
if ( ( iblock - off ) < fs32_to_cpu ( sbi , typ - > vt_size ) )
return ( fs32_to_cpu ( sbi , typ - > vt_block ) +
iblock - off ) ;
2005-04-16 15:20:36 -07:00
break ;
case VXFS_TYPED_INDIRECT_DEV4 :
case VXFS_TYPED_DATA_DEV4 : {
struct vxfs_typed_dev4 * typ4 =
( struct vxfs_typed_dev4 * ) typ ;
printk ( KERN_INFO " \n \n TYPED_DEV4 detected! \n " ) ;
2016-05-31 08:45:13 +02:00
printk ( KERN_INFO " block: %llu \t size: %lld \t dev: %d \n " ,
fs64_to_cpu ( sbi , typ4 - > vd4_block ) ,
fs64_to_cpu ( sbi , typ4 - > vd4_size ) ,
fs32_to_cpu ( sbi , typ4 - > vd4_dev ) ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
default :
BUG ( ) ;
}
}
return 0 ;
}
/**
* vxfs_bmap1 - vxfs - internal bmap operation
* @ ip : pointer to the inode we do bmap for
* @ iblock : logical block
*
* Description :
* vxfs_bmap1 perfoms a logical to physical block mapping
* for vxfs - internal purposes .
*
* Return Value :
* The physical block number on success , else Zero .
*/
daddr_t
vxfs_bmap1 ( struct inode * ip , long iblock )
{
struct vxfs_inode_info * vip = VXFS_INO ( ip ) ;
if ( VXFS_ISEXT4 ( vip ) )
return vxfs_bmap_ext4 ( ip , iblock ) ;
if ( VXFS_ISTYPED ( vip ) )
return vxfs_bmap_typed ( ip , iblock ) ;
if ( VXFS_ISNONE ( vip ) )
goto unsupp ;
if ( VXFS_ISIMMED ( vip ) )
goto unsupp ;
printk ( KERN_WARNING " vxfs: inode %ld has no valid orgtype (%x) \n " ,
ip - > i_ino , vip - > vii_orgtype ) ;
BUG ( ) ;
unsupp :
printk ( KERN_WARNING " vxfs: inode %ld has an unsupported orgtype (%x) \n " ,
ip - > i_ino , vip - > vii_orgtype ) ;
return 0 ;
}