2009-01-05 11:46:25 +03:00
/*
* Squashfs - a compressed read only filesystem for Linux
*
* Copyright ( c ) 2002 , 2003 , 2004 , 2005 , 2006 , 2007 , 2008
2011-05-26 13:39:56 +04:00
* Phillip Lougher < phillip @ squashfs . org . uk >
2009-01-05 11:46:25 +03:00
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2 ,
* or ( at your option ) any later version .
*
* 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 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
*
* super . c
*/
/*
* This file implements code to read the superblock , read and initialise
* in - memory structures at mount time , and all the VFS glue code to register
* the filesystem .
*/
# include <linux/fs.h>
# include <linux/vfs.h>
# include <linux/slab.h>
# include <linux/mutex.h>
# include <linux/pagemap.h>
# include <linux/init.h>
# include <linux/module.h>
2009-01-16 00:51:03 +03:00
# include <linux/magic.h>
2010-05-14 23:48:47 +04:00
# include <linux/xattr.h>
2009-01-05 11:46:25 +03:00
# include "squashfs_fs.h"
# include "squashfs_fs_sb.h"
# include "squashfs_fs_i.h"
# include "squashfs.h"
2009-10-06 07:04:15 +04:00
# include "decompressor.h"
2010-05-17 22:39:02 +04:00
# include "xattr.h"
2009-01-05 11:46:25 +03:00
static struct file_system_type squashfs_fs_type ;
2009-09-22 04:01:09 +04:00
static const struct super_operations squashfs_super_ops ;
2009-01-05 11:46:25 +03:00
2009-10-06 07:04:15 +04:00
static const struct squashfs_decompressor * supported_squashfs_filesystem ( short
major , short minor , short id )
2009-01-05 11:46:25 +03:00
{
2009-10-06 07:04:15 +04:00
const struct squashfs_decompressor * decompressor ;
2009-01-05 11:46:25 +03:00
if ( major < SQUASHFS_MAJOR ) {
ERROR ( " Major/Minor mismatch, older Squashfs %d.%d "
" filesystems are unsupported \n " , major , minor ) ;
2009-10-06 07:04:15 +04:00
return NULL ;
2009-01-05 11:46:25 +03:00
} else if ( major > SQUASHFS_MAJOR | | minor > SQUASHFS_MINOR ) {
ERROR ( " Major/Minor mismatch, trying to mount newer "
" %d.%d filesystem \n " , major , minor ) ;
ERROR ( " Please update your kernel \n " ) ;
2009-10-06 07:04:15 +04:00
return NULL ;
2009-01-05 11:46:25 +03:00
}
2009-10-06 07:04:15 +04:00
decompressor = squashfs_lookup_decompressor ( id ) ;
if ( ! decompressor - > supported ) {
ERROR ( " Filesystem uses \" %s \" compression. This is not "
" supported \n " , decompressor - > name ) ;
return NULL ;
}
2009-01-05 11:46:25 +03:00
2009-10-06 07:04:15 +04:00
return decompressor ;
2009-01-05 11:46:25 +03:00
}
static int squashfs_fill_super ( struct super_block * sb , void * data , int silent )
{
struct squashfs_sb_info * msblk ;
struct squashfs_super_block * sblk = NULL ;
char b [ BDEVNAME_SIZE ] ;
struct inode * root ;
long long root_inode ;
unsigned short flags ;
unsigned int fragments ;
2011-05-24 07:05:22 +04:00
u64 lookup_table_start , xattr_id_table_start , next_table ;
2009-01-05 11:46:25 +03:00
int err ;
TRACE ( " Entered squashfs_fill_superblock \n " ) ;
sb - > s_fs_info = kzalloc ( sizeof ( * msblk ) , GFP_KERNEL ) ;
if ( sb - > s_fs_info = = NULL ) {
ERROR ( " Failed to allocate squashfs_sb_info \n " ) ;
return - ENOMEM ;
}
msblk = sb - > s_fs_info ;
2011-10-22 04:34:48 +04:00
msblk - > devblksize = sb_min_blocksize ( sb , SQUASHFS_DEVBLK_SIZE ) ;
2009-01-05 11:46:25 +03:00
msblk - > devblksize_log2 = ffz ( ~ msblk - > devblksize ) ;
mutex_init ( & msblk - > read_data_mutex ) ;
mutex_init ( & msblk - > meta_index_mutex ) ;
/*
* msblk - > bytes_used is checked in squashfs_read_table to ensure reads
* are not beyond filesystem end . But as we ' re using
* squashfs_read_table here to read the superblock ( including the value
* of bytes_used ) we need to set it to an initial sensible dummy value
*/
msblk - > bytes_used = sizeof ( * sblk ) ;
2011-05-20 05:26:43 +04:00
sblk = squashfs_read_table ( sb , SQUASHFS_START , sizeof ( * sblk ) ) ;
2009-01-05 11:46:25 +03:00
2011-05-20 05:26:43 +04:00
if ( IS_ERR ( sblk ) ) {
2009-01-05 11:46:25 +03:00
ERROR ( " unable to read squashfs_super_block \n " ) ;
2011-05-20 05:26:43 +04:00
err = PTR_ERR ( sblk ) ;
sblk = NULL ;
2009-01-05 11:46:25 +03:00
goto failed_mount ;
}
2009-10-06 07:04:15 +04:00
err = - EINVAL ;
2009-01-05 11:46:25 +03:00
/* Check it is a SQUASHFS superblock */
sb - > s_magic = le32_to_cpu ( sblk - > s_magic ) ;
if ( sb - > s_magic ! = SQUASHFS_MAGIC ) {
if ( ! silent )
ERROR ( " Can't find a SQUASHFS superblock on %s \n " ,
bdevname ( sb - > s_bdev , b ) ) ;
goto failed_mount ;
}
2009-10-06 07:04:15 +04:00
/* Check the MAJOR & MINOR versions and lookup compression type */
msblk - > decompressor = supported_squashfs_filesystem (
le16_to_cpu ( sblk - > s_major ) ,
2009-01-05 11:46:25 +03:00
le16_to_cpu ( sblk - > s_minor ) ,
le16_to_cpu ( sblk - > compression ) ) ;
2009-10-06 07:04:15 +04:00
if ( msblk - > decompressor = = NULL )
2009-01-05 11:46:25 +03:00
goto failed_mount ;
/* Check the filesystem does not extend beyond the end of the
block device */
msblk - > bytes_used = le64_to_cpu ( sblk - > bytes_used ) ;
if ( msblk - > bytes_used < 0 | | msblk - > bytes_used >
i_size_read ( sb - > s_bdev - > bd_inode ) )
goto failed_mount ;
/* Check block size for sanity */
msblk - > block_size = le32_to_cpu ( sblk - > block_size ) ;
if ( msblk - > block_size > SQUASHFS_FILE_MAX_SIZE )
goto failed_mount ;
2009-05-13 05:59:26 +04:00
/*
* Check the system page size is not larger than the filesystem
* block size ( by default 128 K ) . This is currently not supported .
*/
if ( PAGE_CACHE_SIZE > msblk - > block_size ) {
ERROR ( " Page size > filesystem block size (%d). This is "
" currently not supported! \n " , msblk - > block_size ) ;
goto failed_mount ;
}
2012-03-09 07:02:59 +04:00
/* Check block log for sanity */
2009-01-05 11:46:25 +03:00
msblk - > block_log = le16_to_cpu ( sblk - > block_log ) ;
if ( msblk - > block_log > SQUASHFS_FILE_MAX_LOG )
goto failed_mount ;
2012-03-09 07:02:59 +04:00
/* Check that block_size and block_log match */
if ( msblk - > block_size ! = ( 1 < < msblk - > block_log ) )
goto failed_mount ;
2009-01-05 11:46:25 +03:00
/* Check the root inode for sanity */
root_inode = le64_to_cpu ( sblk - > root_inode ) ;
if ( SQUASHFS_INODE_OFFSET ( root_inode ) > SQUASHFS_METADATA_SIZE )
goto failed_mount ;
msblk - > inode_table = le64_to_cpu ( sblk - > inode_table_start ) ;
msblk - > directory_table = le64_to_cpu ( sblk - > directory_table_start ) ;
msblk - > inodes = le32_to_cpu ( sblk - > inodes ) ;
flags = le16_to_cpu ( sblk - > flags ) ;
TRACE ( " Found valid superblock on %s \n " , bdevname ( sb - > s_bdev , b ) ) ;
TRACE ( " Inodes are %scompressed \n " , SQUASHFS_UNCOMPRESSED_INODES ( flags )
? " un " : " " ) ;
TRACE ( " Data is %scompressed \n " , SQUASHFS_UNCOMPRESSED_DATA ( flags )
? " un " : " " ) ;
TRACE ( " Filesystem size %lld bytes \n " , msblk - > bytes_used ) ;
TRACE ( " Block size %d \n " , msblk - > block_size ) ;
TRACE ( " Number of inodes %d \n " , msblk - > inodes ) ;
TRACE ( " Number of fragments %d \n " , le32_to_cpu ( sblk - > fragments ) ) ;
TRACE ( " Number of ids %d \n " , le16_to_cpu ( sblk - > no_ids ) ) ;
TRACE ( " sblk->inode_table_start %llx \n " , msblk - > inode_table ) ;
TRACE ( " sblk->directory_table_start %llx \n " , msblk - > directory_table ) ;
TRACE ( " sblk->fragment_table_start %llx \n " ,
( u64 ) le64_to_cpu ( sblk - > fragment_table_start ) ) ;
TRACE ( " sblk->id_table_start %llx \n " ,
( u64 ) le64_to_cpu ( sblk - > id_table_start ) ) ;
sb - > s_maxbytes = MAX_LFS_FILESIZE ;
sb - > s_flags | = MS_RDONLY ;
sb - > s_op = & squashfs_super_ops ;
err = - ENOMEM ;
msblk - > block_cache = squashfs_cache_init ( " metadata " ,
SQUASHFS_CACHED_BLKS , SQUASHFS_METADATA_SIZE ) ;
if ( msblk - > block_cache = = NULL )
goto failed_mount ;
/* Allocate read_page block */
msblk - > read_page = squashfs_cache_init ( " data " , 1 , msblk - > block_size ) ;
if ( msblk - > read_page = = NULL ) {
ERROR ( " Failed to allocate read_page block \n " ) ;
goto failed_mount ;
}
2011-02-28 04:45:42 +03:00
msblk - > stream = squashfs_decompressor_init ( sb , flags ) ;
if ( IS_ERR ( msblk - > stream ) ) {
err = PTR_ERR ( msblk - > stream ) ;
msblk - > stream = NULL ;
goto failed_mount ;
}
2011-05-24 05:57:05 +04:00
/* Handle xattrs */
sb - > s_xattr = squashfs_xattr_handlers ;
xattr_id_table_start = le64_to_cpu ( sblk - > xattr_id_table_start ) ;
2011-05-24 07:05:22 +04:00
if ( xattr_id_table_start = = SQUASHFS_INVALID_BLK ) {
next_table = msblk - > bytes_used ;
2011-05-24 05:57:05 +04:00
goto allocate_id_index_table ;
2011-05-24 07:05:22 +04:00
}
2011-05-24 05:57:05 +04:00
/* Allocate and read xattr id lookup table */
msblk - > xattr_id_table = squashfs_read_xattr_id_table ( sb ,
xattr_id_table_start , & msblk - > xattr_table , & msblk - > xattr_ids ) ;
if ( IS_ERR ( msblk - > xattr_id_table ) ) {
ERROR ( " unable to read xattr id index table \n " ) ;
err = PTR_ERR ( msblk - > xattr_id_table ) ;
msblk - > xattr_id_table = NULL ;
if ( err ! = - ENOTSUPP )
goto failed_mount ;
}
2011-05-24 07:05:22 +04:00
next_table = msblk - > xattr_table ;
2011-05-24 05:57:05 +04:00
allocate_id_index_table :
2009-01-05 11:46:25 +03:00
/* Allocate and read id index table */
msblk - > id_table = squashfs_read_id_index_table ( sb ,
2011-05-24 07:05:22 +04:00
le64_to_cpu ( sblk - > id_table_start ) , next_table ,
le16_to_cpu ( sblk - > no_ids ) ) ;
2009-01-05 11:46:25 +03:00
if ( IS_ERR ( msblk - > id_table ) ) {
2011-05-20 05:26:43 +04:00
ERROR ( " unable to read id index table \n " ) ;
2009-01-05 11:46:25 +03:00
err = PTR_ERR ( msblk - > id_table ) ;
msblk - > id_table = NULL ;
goto failed_mount ;
}
2011-05-29 03:38:46 +04:00
next_table = le64_to_cpu ( msblk - > id_table [ 0 ] ) ;
2009-01-05 11:46:25 +03:00
2011-05-24 05:57:05 +04:00
/* Handle inode lookup table */
lookup_table_start = le64_to_cpu ( sblk - > lookup_table_start ) ;
if ( lookup_table_start = = SQUASHFS_INVALID_BLK )
goto handle_fragments ;
/* Allocate and read inode lookup table */
msblk - > inode_lookup_table = squashfs_read_inode_lookup_table ( sb ,
2011-05-24 07:15:21 +04:00
lookup_table_start , next_table , msblk - > inodes ) ;
2011-05-24 05:57:05 +04:00
if ( IS_ERR ( msblk - > inode_lookup_table ) ) {
ERROR ( " unable to read inode lookup table \n " ) ;
err = PTR_ERR ( msblk - > inode_lookup_table ) ;
msblk - > inode_lookup_table = NULL ;
goto failed_mount ;
}
2011-05-29 03:38:46 +04:00
next_table = le64_to_cpu ( msblk - > inode_lookup_table [ 0 ] ) ;
2011-05-24 05:57:05 +04:00
sb - > s_export_op = & squashfs_export_ops ;
handle_fragments :
2009-01-05 11:46:25 +03:00
fragments = le32_to_cpu ( sblk - > fragments ) ;
if ( fragments = = 0 )
2011-05-24 07:45:33 +04:00
goto check_directory_table ;
2009-01-05 11:46:25 +03:00
msblk - > fragment_cache = squashfs_cache_init ( " fragment " ,
SQUASHFS_CACHED_FRAGMENTS , msblk - > block_size ) ;
if ( msblk - > fragment_cache = = NULL ) {
err = - ENOMEM ;
goto failed_mount ;
}
/* Allocate and read fragment index table */
msblk - > fragment_index = squashfs_read_fragment_index_table ( sb ,
2011-05-24 07:33:34 +04:00
le64_to_cpu ( sblk - > fragment_table_start ) , next_table , fragments ) ;
2009-01-05 11:46:25 +03:00
if ( IS_ERR ( msblk - > fragment_index ) ) {
2011-05-20 05:26:43 +04:00
ERROR ( " unable to read fragment index table \n " ) ;
2009-01-05 11:46:25 +03:00
err = PTR_ERR ( msblk - > fragment_index ) ;
msblk - > fragment_index = NULL ;
goto failed_mount ;
}
2011-05-29 03:38:46 +04:00
next_table = le64_to_cpu ( msblk - > fragment_index [ 0 ] ) ;
2009-01-05 11:46:25 +03:00
2011-05-24 07:45:33 +04:00
check_directory_table :
/* Sanity check directory_table */
2012-01-02 21:47:14 +04:00
if ( msblk - > directory_table > next_table ) {
2011-05-24 07:45:33 +04:00
err = - EINVAL ;
goto failed_mount ;
}
/* Sanity check inode_table */
if ( msblk - > inode_table > = msblk - > directory_table ) {
err = - EINVAL ;
goto failed_mount ;
}
/* allocate root */
2009-01-05 11:46:25 +03:00
root = new_inode ( sb ) ;
if ( ! root ) {
err = - ENOMEM ;
goto failed_mount ;
}
err = squashfs_read_inode ( root , root_inode ) ;
if ( err ) {
2010-04-16 04:01:36 +04:00
make_bad_inode ( root ) ;
iput ( root ) ;
2009-01-05 11:46:25 +03:00
goto failed_mount ;
}
insert_inode_hash ( root ) ;
2012-01-09 07:15:13 +04:00
sb - > s_root = d_make_root ( root ) ;
2009-01-05 11:46:25 +03:00
if ( sb - > s_root = = NULL ) {
ERROR ( " Root inode create failed \n " ) ;
err = - ENOMEM ;
goto failed_mount ;
}
TRACE ( " Leaving squashfs_fill_super \n " ) ;
kfree ( sblk ) ;
return 0 ;
failed_mount :
squashfs_cache_delete ( msblk - > block_cache ) ;
squashfs_cache_delete ( msblk - > fragment_cache ) ;
squashfs_cache_delete ( msblk - > read_page ) ;
2009-10-06 07:04:15 +04:00
squashfs_decompressor_free ( msblk , msblk - > stream ) ;
2009-01-05 11:46:25 +03:00
kfree ( msblk - > inode_lookup_table ) ;
kfree ( msblk - > fragment_index ) ;
kfree ( msblk - > id_table ) ;
2010-05-14 23:48:47 +04:00
kfree ( msblk - > xattr_id_table ) ;
2009-01-05 11:46:25 +03:00
kfree ( sb - > s_fs_info ) ;
sb - > s_fs_info = NULL ;
kfree ( sblk ) ;
return err ;
}
static int squashfs_statfs ( struct dentry * dentry , struct kstatfs * buf )
{
struct squashfs_sb_info * msblk = dentry - > d_sb - > s_fs_info ;
2009-04-03 03:59:42 +04:00
u64 id = huge_encode_dev ( dentry - > d_sb - > s_bdev - > bd_dev ) ;
2009-01-05 11:46:25 +03:00
TRACE ( " Entered squashfs_statfs \n " ) ;
buf - > f_type = SQUASHFS_MAGIC ;
buf - > f_bsize = msblk - > block_size ;
buf - > f_blocks = ( ( msblk - > bytes_used - 1 ) > > msblk - > block_log ) + 1 ;
buf - > f_bfree = buf - > f_bavail = 0 ;
buf - > f_files = msblk - > inodes ;
buf - > f_ffree = 0 ;
buf - > f_namelen = SQUASHFS_NAME_LEN ;
2009-04-03 03:59:42 +04:00
buf - > f_fsid . val [ 0 ] = ( u32 ) id ;
buf - > f_fsid . val [ 1 ] = ( u32 ) ( id > > 32 ) ;
2009-01-05 11:46:25 +03:00
return 0 ;
}
static int squashfs_remount ( struct super_block * sb , int * flags , char * data )
{
* flags | = MS_RDONLY ;
return 0 ;
}
static void squashfs_put_super ( struct super_block * sb )
{
if ( sb - > s_fs_info ) {
struct squashfs_sb_info * sbi = sb - > s_fs_info ;
squashfs_cache_delete ( sbi - > block_cache ) ;
squashfs_cache_delete ( sbi - > fragment_cache ) ;
squashfs_cache_delete ( sbi - > read_page ) ;
2009-10-06 07:04:15 +04:00
squashfs_decompressor_free ( sbi , sbi - > stream ) ;
2009-01-05 11:46:25 +03:00
kfree ( sbi - > id_table ) ;
kfree ( sbi - > fragment_index ) ;
kfree ( sbi - > meta_index ) ;
2010-04-23 03:24:22 +04:00
kfree ( sbi - > inode_lookup_table ) ;
2010-05-14 23:48:47 +04:00
kfree ( sbi - > xattr_id_table ) ;
2009-01-05 11:46:25 +03:00
kfree ( sb - > s_fs_info ) ;
sb - > s_fs_info = NULL ;
}
}
2011-02-28 21:43:48 +03:00
static struct dentry * squashfs_mount ( struct file_system_type * fs_type ,
int flags , const char * dev_name , void * data )
2009-01-05 11:46:25 +03:00
{
2010-07-25 00:46:55 +04:00
return mount_bdev ( fs_type , flags , dev_name , data , squashfs_fill_super ) ;
2009-01-05 11:46:25 +03:00
}
static struct kmem_cache * squashfs_inode_cachep ;
static void init_once ( void * foo )
{
struct squashfs_inode_info * ei = foo ;
inode_init_once ( & ei - > vfs_inode ) ;
}
static int __init init_inodecache ( void )
{
squashfs_inode_cachep = kmem_cache_create ( " squashfs_inode_cache " ,
sizeof ( struct squashfs_inode_info ) , 0 ,
SLAB_HWCACHE_ALIGN | SLAB_RECLAIM_ACCOUNT , init_once ) ;
return squashfs_inode_cachep ? 0 : - ENOMEM ;
}
static void destroy_inodecache ( void )
{
kmem_cache_destroy ( squashfs_inode_cachep ) ;
}
static int __init init_squashfs_fs ( void )
{
int err = init_inodecache ( ) ;
if ( err )
return err ;
err = register_filesystem ( & squashfs_fs_type ) ;
if ( err ) {
destroy_inodecache ( ) ;
return err ;
}
2009-03-05 03:31:12 +03:00
printk ( KERN_INFO " squashfs: version 4.0 (2009/01/31) "
2009-01-05 11:46:25 +03:00
" Phillip Lougher \n " ) ;
return 0 ;
}
static void __exit exit_squashfs_fs ( void )
{
unregister_filesystem ( & squashfs_fs_type ) ;
destroy_inodecache ( ) ;
}
static struct inode * squashfs_alloc_inode ( struct super_block * sb )
{
struct squashfs_inode_info * ei =
kmem_cache_alloc ( squashfs_inode_cachep , GFP_KERNEL ) ;
return ei ? & ei - > vfs_inode : NULL ;
}
2011-01-07 09:49:49 +03:00
static void squashfs_i_callback ( struct rcu_head * head )
2009-01-05 11:46:25 +03:00
{
2011-01-07 09:49:49 +03:00
struct inode * inode = container_of ( head , struct inode , i_rcu ) ;
2009-01-05 11:46:25 +03:00
kmem_cache_free ( squashfs_inode_cachep , squashfs_i ( inode ) ) ;
}
2011-01-07 09:49:49 +03:00
static void squashfs_destroy_inode ( struct inode * inode )
{
call_rcu ( & inode - > i_rcu , squashfs_i_callback ) ;
}
2009-01-05 11:46:25 +03:00
static struct file_system_type squashfs_fs_type = {
. owner = THIS_MODULE ,
. name = " squashfs " ,
2010-07-25 00:46:55 +04:00
. mount = squashfs_mount ,
2009-01-05 11:46:25 +03:00
. kill_sb = kill_block_super ,
. fs_flags = FS_REQUIRES_DEV
} ;
2009-09-22 04:01:09 +04:00
static const struct super_operations squashfs_super_ops = {
2009-01-05 11:46:25 +03:00
. alloc_inode = squashfs_alloc_inode ,
. destroy_inode = squashfs_destroy_inode ,
. statfs = squashfs_statfs ,
. put_super = squashfs_put_super ,
. remount_fs = squashfs_remount
} ;
module_init ( init_squashfs_fs ) ;
module_exit ( exit_squashfs_fs ) ;
MODULE_DESCRIPTION ( " squashfs 4.0, a compressed read-only filesystem " ) ;
2011-05-26 13:39:56 +04:00
MODULE_AUTHOR ( " Phillip Lougher <phillip@squashfs.org.uk> " ) ;
2009-01-05 11:46:25 +03:00
MODULE_LICENSE ( " GPL " ) ;