Merge to Fedora kernel-2.6.18-1.2224_FC5 patched with stable patch-2.6.18.1-vs2.0...
[linux-2.6.git] / fs / isofs / inode.c
index 884719b..e5fbb03 100644 (file)
@@ -7,41 +7,26 @@
  *      1995  Mark Dobie - allow mounting of some weird VideoCDs and PhotoCDs.
  *     1997  Gordon Chaffee - Joliet CDs
  *     1998  Eric Lammerts - ISO 9660 Level 3
+ *     2004  Paul Serice - Inode Support pushed out from 4GB to 128GB
+ *     2004  Paul Serice - NFS Export Operations
  */
 
-#include <linux/config.h>
+#include <linux/init.h>
 #include <linux/module.h>
 
-#include <linux/stat.h>
-#include <linux/time.h>
-#include <linux/iso_fs.h>
-#include <linux/kernel.h>
-#include <linux/major.h>
-#include <linux/mm.h>
-#include <linux/string.h>
 #include <linux/slab.h>
-#include <linux/errno.h>
-#include <linux/cdrom.h>
-#include <linux/init.h>
 #include <linux/nls.h>
 #include <linux/ctype.h>
 #include <linux/smp_lock.h>
-#include <linux/blkdev.h>
-#include <linux/buffer_head.h>
-#include <linux/vfs.h>
+#include <linux/statfs.h>
+#include <linux/cdrom.h>
 #include <linux/parser.h>
-#include <asm/system.h>
-#include <asm/uaccess.h>
 
+#include "isofs.h"
 #include "zisofs.h"
 
 #define BEQUIET
 
-#ifdef LEAK_CHECK
-static int check_malloc;
-static int check_bread;
-#endif
-
 static int isofs_hashi(struct dentry *parent, struct qstr *qstr);
 static int isofs_hash(struct dentry *parent, struct qstr *qstr);
 static int isofs_dentry_cmpi(struct dentry *dentry, struct qstr *a, struct qstr *b);
@@ -64,25 +49,20 @@ static void isofs_put_super(struct super_block *sb)
        }
 #endif
 
-#ifdef LEAK_CHECK
-       printk("Outstanding mallocs:%d, outstanding buffers: %d\n",
-              check_malloc, check_bread);
-#endif
-
        kfree(sbi);
        sb->s_fs_info = NULL;
        return;
 }
 
 static void isofs_read_inode(struct inode *);
-static int isofs_statfs (struct super_block *, struct kstatfs *);
+static int isofs_statfs (struct dentry *, struct kstatfs *);
 
 static kmem_cache_t *isofs_inode_cachep;
 
 static struct inode *isofs_alloc_inode(struct super_block *sb)
 {
        struct iso_inode_info *ei;
-       ei = (struct iso_inode_info *)kmem_cache_alloc(isofs_inode_cachep, SLAB_KERNEL);
+       ei = kmem_cache_alloc(isofs_inode_cachep, SLAB_KERNEL);
        if (!ei)
                return NULL;
        return &ei->vfs_inode;
@@ -93,9 +73,9 @@ static void isofs_destroy_inode(struct inode *inode)
        kmem_cache_free(isofs_inode_cachep, ISOFS_I(inode));
 }
 
-static void init_once(void * foo, kmem_cache_t * cachep, unsigned long flags)
+static void init_once(void *foo, kmem_cache_t * cachep, unsigned long flags)
 {
-       struct iso_inode_info *ei = (struct iso_inode_info *) foo;
+       struct iso_inode_info *ei = foo;
 
        if ((flags & (SLAB_CTOR_VERIFY|SLAB_CTOR_CONSTRUCTOR)) ==
            SLAB_CTOR_CONSTRUCTOR)
@@ -106,7 +86,8 @@ static int init_inodecache(void)
 {
        isofs_inode_cachep = kmem_cache_create("isofs_inode_cache",
                                             sizeof(struct iso_inode_info),
-                                            0, SLAB_HWCACHE_ALIGN|SLAB_RECLAIM_ACCOUNT,
+                                            0, (SLAB_RECLAIM_ACCOUNT|
+                                               SLAB_MEM_SPREAD),
                                             init_once, NULL);
        if (isofs_inode_cachep == NULL)
                return -ENOMEM;
@@ -116,7 +97,8 @@ static int init_inodecache(void)
 static void destroy_inodecache(void)
 {
        if (kmem_cache_destroy(isofs_inode_cachep))
-               printk(KERN_INFO "iso_inode_cache: not all structures were freed\n");
+               printk(KERN_INFO "iso_inode_cache: not all structures were "
+                                       "freed\n");
 }
 
 static int isofs_remount(struct super_block *sb, int *flags, char *data)
@@ -135,20 +117,6 @@ static struct super_operations isofs_sops = {
        .remount_fs     = isofs_remount,
 };
 
-/* the export_operations structure for describing
- * how to export (e.g. via kNFSd) is deliberately
- * empty.
- * This means that the filesystem want to use iget
- * to map an inode number into an inode.
- * The lack of a get_parent operation means that 
- * if something isn't in the cache, then you cannot
- * access it.
- * It should be possible to write a get_parent,
- * but it would be a bit hairy...
- */
-static struct export_operations isofs_export_ops = {
-};
-
 
 static struct dentry_operations isofs_dentry_ops[] = {
        {
@@ -167,7 +135,7 @@ static struct dentry_operations isofs_dentry_ops[] = {
        {
                .d_hash         = isofs_hashi_ms,
                .d_compare      = isofs_dentry_cmpi_ms,
-       }
+       },
 #endif
 };
 
@@ -176,7 +144,8 @@ struct iso9660_options{
        char rock;
        char joliet;
        char cruft;
-       char unhide;
+       char hide;
+       char showassoc;
        char nocompress;
        unsigned char check;
        unsigned int blocksize;
@@ -242,8 +211,8 @@ isofs_hashi_common(struct dentry *dentry, struct qstr *qstr, int ms)
 /*
  * Case insensitive compare of two isofs names.
  */
-static int
-isofs_dentry_cmpi_common(struct dentry *dentry,struct qstr *a,struct qstr *b,int ms)
+static int isofs_dentry_cmpi_common(struct dentry *dentry, struct qstr *a,
+                               struct qstr *b, int ms)
 {
        int alen, blen;
 
@@ -266,8 +235,8 @@ isofs_dentry_cmpi_common(struct dentry *dentry,struct qstr *a,struct qstr *b,int
 /*
  * Case sensitive compare of two isofs names.
  */
-static int
-isofs_dentry_cmp_common(struct dentry *dentry,struct qstr *a,struct qstr *b,int ms)
+static int isofs_dentry_cmp_common(struct dentry *dentry, struct qstr *a,
+                                       struct qstr *b, int ms)
 {
        int alen, blen;
 
@@ -341,13 +310,15 @@ enum {
        Opt_block, Opt_check_r, Opt_check_s, Opt_cruft, Opt_gid, Opt_ignore,
        Opt_iocharset, Opt_map_a, Opt_map_n, Opt_map_o, Opt_mode, Opt_nojoliet,
        Opt_norock, Opt_sb, Opt_session, Opt_uid, Opt_unhide, Opt_utf8, Opt_err,
-       Opt_nocompress,
+       Opt_nocompress, Opt_hide, Opt_showassoc,
 };
 
 static match_table_t tokens = {
        {Opt_norock, "norock"},
        {Opt_nojoliet, "nojoliet"},
        {Opt_unhide, "unhide"},
+       {Opt_hide, "hide"},
+       {Opt_showassoc, "showassoc"},
        {Opt_cruft, "cruft"},
        {Opt_utf8, "utf8"},
        {Opt_iocharset, "iocharset=%s"},
@@ -379,7 +350,7 @@ static match_table_t tokens = {
        {Opt_err, NULL}
 };
 
-static int parse_options(char *options, struct iso9660_options * popt)
+static int parse_options(char *options, struct iso9660_options *popt)
 {
        char *p;
        int option;
@@ -388,7 +359,8 @@ static int parse_options(char *options, struct iso9660_options * popt)
        popt->rock = 'y';
        popt->joliet = 'y';
        popt->cruft = 'n';
-       popt->unhide = 'n';
+       popt->hide = 'n';
+       popt->showassoc = 'n';
        popt->check = 'u';              /* unset */
        popt->nocompress = 0;
        popt->blocksize = 1024;
@@ -421,8 +393,12 @@ static int parse_options(char *options, struct iso9660_options * popt)
                case Opt_nojoliet:
                        popt->joliet = 'n';
                        break;
+               case Opt_hide:
+                       popt->hide = 'y';
+                       break;
                case Opt_unhide:
-                       popt->unhide = 'y';
+               case Opt_showassoc:
+                       popt->showassoc = 'y';
                        break;
                case Opt_cruft:
                        popt->cruft = 'y';
@@ -516,7 +492,7 @@ static int parse_options(char *options, struct iso9660_options * popt)
  */
 #define WE_OBEY_THE_WRITTEN_STANDARDS 1
 
-static unsigned int isofs_get_last_session(struct super_block *sb,s32 session )
+static unsigned int isofs_get_last_session(struct super_block *sb, s32 session)
 {
        struct cdrom_multisession ms_info;
        unsigned int vol_desc_start;
@@ -541,7 +517,8 @@ static unsigned int isofs_get_last_session(struct super_block *sb,s32 session )
                printk(KERN_ERR "Invalid session number or type of track\n");
        }
        i = ioctl_by_bdev(bdev, CDROMMULTISESSION, (unsigned long) &ms_info);
-       if(session > 0) printk(KERN_ERR "Invalid session number\n");
+       if (session > 0)
+               printk(KERN_ERR "Invalid session number\n");
 #if 0
        printk("isofs.inode: CDROMMULTISESSION: rc=%d\n",i);
        if (i==0) {
@@ -580,13 +557,13 @@ static int isofs_fill_super(struct super_block *s, void *data, int silent)
        struct iso9660_options          opt;
        struct isofs_sb_info          * sbi;
 
-       sbi = kmalloc(sizeof(struct isofs_sb_info), GFP_KERNEL);
+       sbi = kmalloc(sizeof(*sbi), GFP_KERNEL);
        if (!sbi)
                return -ENOMEM;
        s->s_fs_info = sbi;
-       memset(sbi, 0, sizeof(struct isofs_sb_info));
+       memset(sbi, 0, sizeof(*sbi));
 
-       if (!parse_options((char *) data, &opt))
+       if (!parse_options((char *)data, &opt))
                goto out_freesbi;
 
        /*
@@ -697,6 +674,8 @@ root_found:
          sbi->s_log_zone_size = isonum_723 (h_pri->logical_block_size);
          sbi->s_max_size = isonum_733(h_pri->volume_space_size);
        } else {
+         if (!pri)
+           goto out_freebh;
          rootp = (struct iso_directory_record *) pri->root_directory_record;
          sbi->s_nzones = isonum_733 (pri->volume_space_size);
          sbi->s_log_zone_size = isonum_723 (pri->logical_block_size);
@@ -727,6 +706,7 @@ root_found:
          }
 
        s->s_magic = ISOFS_SUPER_MAGIC;
+       s->s_maxbytes = 0xffffffff; /* We can handle files up to 4 GB */
 
        /* The CDROM is read-only, has no nodes (devices) on it, and since
           all of the files appear to be owned by root, we really do not want
@@ -738,19 +718,14 @@ root_found:
        /* Set this for reference. Its not currently used except on write
           which we don't have .. */
           
-       /* RDE: data zone now byte offset! */
-
-       first_data_zone = ((isonum_733 (rootp->extent) +
-                         isonum_711 (rootp->ext_attr_length))
-                        << sbi->s_log_zone_size);
+       first_data_zone = isonum_733 (rootp->extent) +
+                         isonum_711 (rootp->ext_attr_length);
        sbi->s_firstdatazone = first_data_zone;
 #ifndef BEQUIET
        printk(KERN_DEBUG "Max size:%ld   Log zone size:%ld\n",
               sbi->s_max_size,
               1UL << sbi->s_log_zone_size);
-       printk(KERN_DEBUG "First datazone:%ld   Root inode number:%ld\n",
-              sbi->s_firstdatazone >> sbi->s_log_zone_size,
-              sbi->s_firstdatazone);
+       printk(KERN_DEBUG "First datazone:%ld\n", sbi->s_firstdatazone);
        if(sbi->s_high_sierra)
                printk(KERN_DEBUG "Disc in High Sierra format.\n");
 #endif
@@ -767,9 +742,8 @@ root_found:
                pri = (struct iso_primary_descriptor *) sec;
                rootp = (struct iso_directory_record *)
                        pri->root_directory_record;
-               first_data_zone = ((isonum_733 (rootp->extent) +
-                               isonum_711 (rootp->ext_attr_length))
-                                << sbi->s_log_zone_size);
+               first_data_zone = isonum_733 (rootp->extent) +
+                               isonum_711 (rootp->ext_attr_length);
        }
 
        /*
@@ -818,7 +792,8 @@ root_found:
        sbi->s_rock = (opt.rock == 'y' ? 2 : 0);
        sbi->s_rock_offset = -1; /* initial offset, will guess until SP is found*/
        sbi->s_cruft = opt.cruft;
-       sbi->s_unhide = opt.unhide;
+       sbi->s_hide = opt.hide;
+       sbi->s_showassoc = opt.showassoc;
        sbi->s_uid = opt.uid;
        sbi->s_gid = opt.gid;
        sbi->s_utf8 = opt.utf8;
@@ -835,7 +810,7 @@ root_found:
         * the s_rock flag. Once we have the final s_rock value,
         * we then decide whether to use the Joliet descriptor.
         */
-       inode = iget(s, sbi->s_firstdatazone);
+       inode = isofs_iget(s, sbi->s_firstdatazone, 0);
 
        /*
         * If this disk has both Rock Ridge and Joliet on it, then we
@@ -854,7 +829,7 @@ root_found:
                        printk(KERN_DEBUG 
                                "ISOFS: changing to secondary root\n");
                        iput(inode);
-                       inode = iget(s, sbi->s_firstdatazone);
+                       inode = isofs_iget(s, sbi->s_firstdatazone, 0);
                }
        }
 
@@ -880,8 +855,7 @@ root_found:
        if (opt.check == 'r') table++;
        s->s_root->d_op = &isofs_dentry_ops[table];
 
-       if (opt.iocharset)
-               kfree(opt.iocharset);
+       kfree(opt.iocharset);
 
        return 0;
 
@@ -920,15 +894,16 @@ out_unknown_format:
 out_freebh:
        brelse(bh);
 out_freesbi:
-       if (opt.iocharset)
-               kfree(opt.iocharset);
+       kfree(opt.iocharset);
        kfree(sbi);
        s->s_fs_info = NULL;
        return -EINVAL;
 }
 
-static int isofs_statfs (struct super_block *sb, struct kstatfs *buf)
+static int isofs_statfs (struct dentry *dentry, struct kstatfs *buf)
 {
+       struct super_block *sb = dentry->d_sb;
+
        buf->f_type = ISOFS_SUPER_MAGIC;
        buf->f_bsize = sb->s_blocksize;
        buf->f_blocks = (ISOFS_SB(sb)->s_nzones
@@ -952,7 +927,7 @@ int isofs_get_blocks(struct inode *inode, sector_t iblock_s,
        unsigned long b_off;
        unsigned offset, sect_size;
        unsigned int firstext;
-       unsigned long nextino;
+       unsigned long nextblk, nextoff;
        long iblock = (long)iblock_s;
        int section, rv;
        struct iso_inode_info *ei = ISOFS_I(inode);
@@ -970,7 +945,8 @@ int isofs_get_blocks(struct inode *inode, sector_t iblock_s,
        offset    = 0;
        firstext  = ei->i_first_extent;
        sect_size = ei->i_section_size >> ISOFS_BUFFER_BITS(inode);
-       nextino   = ei->i_next_section_ino;
+       nextblk   = ei->i_next_section_block;
+       nextoff   = ei->i_next_section_offset;
        section   = 0;
 
        while ( nblocks ) {
@@ -987,25 +963,28 @@ int isofs_get_blocks(struct inode *inode, sector_t iblock_s,
                        goto abort;
                }
                
-               if (nextino) {
+               if (nextblk) {
                        while (b_off >= (offset + sect_size)) {
                                struct inode *ninode;
                                
                                offset += sect_size;
-                               if (nextino == 0)
+                               if (nextblk == 0)
                                        goto abort;
-                               ninode = iget(inode->i_sb, nextino);
+                               ninode = isofs_iget(inode->i_sb, nextblk, nextoff);
                                if (!ninode)
                                        goto abort;
                                firstext  = ISOFS_I(ninode)->i_first_extent;
-                               sect_size = ISOFS_I(ninode)->i_section_size;
-                               nextino   = ISOFS_I(ninode)->i_next_section_ino;
+                               sect_size = ISOFS_I(ninode)->i_section_size >> ISOFS_BUFFER_BITS(ninode);
+                               nextblk   = ISOFS_I(ninode)->i_next_section_block;
+                               nextoff   = ISOFS_I(ninode)->i_next_section_offset;
                                iput(ninode);
                                
                                if (++section > 100) {
                                        printk("isofs_get_blocks: More than 100 file sections ?!?, aborting...\n");
-                                       printk("isofs_get_blocks: ino=%lu block=%ld firstext=%u sect_size=%u nextino=%lu\n",
-                                              inode->i_ino, iblock, firstext, (unsigned) sect_size, nextino);
+                                       printk("isofs_get_blocks: block=%ld firstext=%u sect_size=%u "
+                                              "nextblk=%lu nextoff=%lu\n",
+                                              iblock, firstext, (unsigned) sect_size,
+                                              nextblk, nextoff);
                                        goto abort;
                                }
                        }
@@ -1024,7 +1003,6 @@ int isofs_get_blocks(struct inode *inode, sector_t iblock_s,
                rv++;
        }
 
-
 abort:
        unlock_kernel();
        return rv;
@@ -1036,7 +1014,7 @@ abort:
 static int isofs_get_block(struct inode *inode, sector_t iblock,
                    struct buffer_head *bh_result, int create)
 {
-       if ( create ) {
+       if (create) {
                printk("isofs_get_block: Kernel tries to allocate a block\n");
                return -EROFS;
        }
@@ -1044,7 +1022,7 @@ static int isofs_get_block(struct inode *inode, sector_t iblock,
        return isofs_get_blocks(inode, iblock, &bh_result, 1) ? 0 : -EIO;
 }
 
-static int isofs_bmap(struct inode *inode, int block)
+static int isofs_bmap(struct inode *inode, sector_t block)
 {
        struct buffer_head dummy;
        int error;
@@ -1075,7 +1053,7 @@ static sector_t _isofs_bmap(struct address_space *mapping, sector_t block)
        return generic_block_bmap(mapping,block,isofs_get_block);
 }
 
-static struct address_space_operations isofs_aops = {
+static const struct address_space_operations isofs_aops = {
        .readpage = isofs_readpage,
        .sync_page = block_sync_page,
        .bmap = _isofs_bmap
@@ -1083,35 +1061,37 @@ static struct address_space_operations isofs_aops = {
 
 static inline void test_and_set_uid(uid_t *p, uid_t value)
 {
-       if(value) {
+       if (value)
                *p = value;
-       }
 }
 
 static inline void test_and_set_gid(gid_t *p, gid_t value)
 {
-        if(value) {
+        if (value)
                 *p = value;
-        }
 }
 
-static int isofs_read_level3_size(struct inode * inode)
+static int isofs_read_level3_size(struct inode *inode)
 {
-       unsigned long f_pos = inode->i_ino;
        unsigned long bufsize = ISOFS_BUFFER_SIZE(inode);
        int high_sierra = ISOFS_SB(inode->i_sb)->s_high_sierra;
        struct buffer_head * bh = NULL;
-       unsigned long block, offset;
+       unsigned long block, offset, block_saved, offset_saved;
        int i = 0;
        int more_entries = 0;
        struct iso_directory_record * tmpde = NULL;
        struct iso_inode_info *ei = ISOFS_I(inode);
 
        inode->i_size = 0;
-       ei->i_next_section_ino = 0;
 
-       block = f_pos >> ISOFS_BUFFER_BITS(inode);
-       offset = f_pos & (bufsize-1);
+       /* The first 16 blocks are reserved as the System Area.  Thus,
+        * no inodes can appear in block 0.  We use this to flag that
+        * this is the last section. */
+       ei->i_next_section_block = 0;
+       ei->i_next_section_offset = 0;
+
+       block = ei->i_iget5_block;
+       offset = ei->i_iget5_offset;
 
        do {
                struct iso_directory_record * de;
@@ -1128,12 +1108,13 @@ static int isofs_read_level3_size(struct inode * inode)
                if (de_len == 0) {
                        brelse(bh);
                        bh = NULL;
-                       f_pos = (f_pos + ISOFS_BLOCK_SIZE) & ~(ISOFS_BLOCK_SIZE - 1);
-                       block = f_pos >> ISOFS_BUFFER_BITS(inode);
+                       ++block;
                        offset = 0;
                        continue;
                }
 
+               block_saved = block;
+               offset_saved = offset;
                offset += de_len;
 
                /* Make sure we have a full directory entry */
@@ -1153,25 +1134,25 @@ static int isofs_read_level3_size(struct inode * inode)
                                bh = sb_bread(inode->i_sb, block);
                                if (!bh)
                                        goto out_noread;
-                               memcpy((void *) tmpde + slop, bh->b_data, offset);
+                               memcpy((void *)tmpde+slop, bh->b_data, offset);
                        }
                        de = tmpde;
                }
 
                inode->i_size += isonum_733(de->size);
-               if (i == 1)
-                       ei->i_next_section_ino = f_pos;
+               if (i == 1) {
+                       ei->i_next_section_block = block_saved;
+                       ei->i_next_section_offset = offset_saved;
+               }
 
                more_entries = de->flags[-high_sierra] & 0x80;
 
-               f_pos += de_len;
                i++;
-               if(i > 100)
+               if (i > 100)
                        goto out_toomany;
-       } while(more_entries);
+       } while (more_entries);
 out:
-       if (tmpde)
-               kfree(tmpde);
+       kfree(tmpde);
        if (bh)
                brelse(bh);
        return 0;
@@ -1183,38 +1164,38 @@ out_nomem:
 
 out_noread:
        printk(KERN_INFO "ISOFS: unable to read i-node block %lu\n", block);
-       if (tmpde)
-               kfree(tmpde);
+       kfree(tmpde);
        return -EIO;
 
 out_toomany:
        printk(KERN_INFO "isofs_read_level3_size: "
                "More than 100 file sections ?!?, aborting...\n"
-               "isofs_read_level3_size: inode=%lu ino=%lu\n",
-               inode->i_ino, f_pos);
+               "isofs_read_level3_size: inode=%lu\n",
+               inode->i_ino);
        goto out;
 }
 
-static void isofs_read_inode(struct inode * inode)
+static void isofs_read_inode(struct inode *inode)
 {
        struct super_block *sb = inode->i_sb;
        struct isofs_sb_info *sbi = ISOFS_SB(sb);
        unsigned long bufsize = ISOFS_BUFFER_SIZE(inode);
-       int block = inode->i_ino >> ISOFS_BUFFER_BITS(inode);
+       unsigned long block;
        int high_sierra = sbi->s_high_sierra;
        struct buffer_head * bh = NULL;
        struct iso_directory_record * de;
        struct iso_directory_record * tmpde = NULL;
        unsigned int de_len;
        unsigned long offset;
-       int volume_seq_no, i;
        struct iso_inode_info *ei = ISOFS_I(inode);
 
+       block = ei->i_iget5_block;
        bh = sb_bread(inode->i_sb, block);
        if (!bh)
                goto out_badread;
 
-       offset = (inode->i_ino & (bufsize - 1));
+       offset = ei->i_iget5_offset;
+
        de = (struct iso_directory_record *) (bh->b_data + offset);
        de_len = *(unsigned char *) de;
 
@@ -1235,6 +1216,10 @@ static void isofs_read_inode(struct inode * inode)
                de = tmpde;
        }
 
+       inode->i_ino = isofs_get_ino(ei->i_iget5_block,
+                                    ei->i_iget5_offset,
+                                    ISOFS_BUFFER_BITS(inode));
+
        /* Assume it is a normal-format file unless told otherwise */
        ei->i_file_format = isofs_file_normal;
 
@@ -1250,56 +1235,32 @@ static void isofs_read_inode(struct inode * inode)
                inode->i_mode = sbi->s_mode;
                inode->i_nlink = 1;
                inode->i_mode |= S_IFREG;
-               /* If there are no periods in the name,
-                * then set the execute permission bit
-                */
-               for(i=0; i< de->name_len[0]; i++)
-                       if(de->name[i]=='.' || de->name[i]==';')
-                               break;
-               if(i == de->name_len[0] || de->name[i] == ';')
-                       inode->i_mode |= S_IXUGO; /* execute permission */
        }
        inode->i_uid = sbi->s_uid;
        inode->i_gid = sbi->s_gid;
-       inode->i_blocks = inode->i_blksize = 0;
+       inode->i_blocks = 0;
 
        ei->i_format_parm[0] = 0;
        ei->i_format_parm[1] = 0;
        ei->i_format_parm[2] = 0;
 
        ei->i_section_size = isonum_733 (de->size);
-       if(de->flags[-high_sierra] & 0x80) {
+       if (de->flags[-high_sierra] & 0x80) {
                if(isofs_read_level3_size(inode)) goto fail;
        } else {
-               ei->i_next_section_ino = 0;
+               ei->i_next_section_block = 0;
+               ei->i_next_section_offset = 0;
                inode->i_size = isonum_733 (de->size);
        }
 
-       /*
-        * The ISO-9660 filesystem only stores 32 bits for file size.
-        * mkisofs handles files up to 2GB-2 = 2147483646 = 0x7FFFFFFE bytes
-        * in size. This is according to the large file summit paper from 1996.
-        * WARNING: ISO-9660 filesystems > 1 GB and even > 2 GB are fully
-        *          legal. Do not prevent to use DVD's schilling@fokus.gmd.de
-        */
-       if ((inode->i_size < 0 || inode->i_size > 0x7FFFFFFE) &&
-           sbi->s_cruft == 'n') {
-               printk(KERN_WARNING "Warning: defective CD-ROM.  "
-                      "Enabling \"cruft\" mount option.\n");
-               sbi->s_cruft = 'y';
-       }
-
        /*
         * Some dipshit decided to store some other bit of information
-        * in the high byte of the file length.  Catch this and holler.
-        * WARNING: this will make it impossible for a file to be > 16MB
-        * on the CDROM.
+        * in the high byte of the file length.  Truncate size in case
+        * this CDROM was mounted with the cruft option.
         */
 
-       if (sbi->s_cruft == 'y' &&
-           inode->i_size & 0xff000000) {
+       if (sbi->s_cruft == 'y')
                inode->i_size &= 0x00ffffff;
-       }
 
        if (de->interleave[0]) {
                printk("Interleaved files not (yet) supported.\n");
@@ -1333,7 +1294,6 @@ static void isofs_read_inode(struct inode * inode)
                              isonum_711 (de->ext_attr_length));
 
        /* Set the number of blocks for stat() - should be done before RR */
-       inode->i_blksize = PAGE_CACHE_SIZE; /* For stat() only */
        inode->i_blocks  = (inode->i_size + 511) >> 9;
 
        /*
@@ -1348,9 +1308,6 @@ static void isofs_read_inode(struct inode * inode)
                test_and_set_gid(&inode->i_gid, sbi->s_gid);
        }
 
-       /* get the volume sequence number */
-       volume_seq_no = isonum_723 (de->volume_sequence_number) ;
-
        /* Install the inode operations vector */
        if (S_ISREG(inode->i_mode)) {
                inode->i_fop = &generic_ro_fops;
@@ -1374,54 +1331,79 @@ static void isofs_read_inode(struct inode * inode)
                /* XXX - parse_rock_ridge_inode() had already set i_rdev. */
                init_special_inode(inode, inode->i_mode, inode->i_rdev);
 
- out:
-       if (tmpde)
-               kfree(tmpde);
+out:
+       kfree(tmpde);
        if (bh)
                brelse(bh);
        return;
 
- out_badread:
+out_badread:
        printk(KERN_WARNING "ISOFS: unable to read i-node block\n");
- fail:
+fail:
        make_bad_inode(inode);
        goto out;
 }
 
-#ifdef LEAK_CHECK
-#undef malloc
-#undef free_s
-#undef sb_bread
-#undef brelse
-
-void * leak_check_malloc(unsigned int size){
-  void * tmp;
-  check_malloc++;
-  tmp = kmalloc(size, GFP_KERNEL);
-  return tmp;
-}
+struct isofs_iget5_callback_data {
+       unsigned long block;
+       unsigned long offset;
+};
 
-void leak_check_free_s(void * obj, int size){
-  check_malloc--;
-  return kfree(obj);
+static int isofs_iget5_test(struct inode *ino, void *data)
+{
+       struct iso_inode_info *i = ISOFS_I(ino);
+       struct isofs_iget5_callback_data *d =
+               (struct isofs_iget5_callback_data*)data;
+       return (i->i_iget5_block == d->block)
+              && (i->i_iget5_offset == d->offset);
 }
 
-struct buffer_head * leak_check_bread(struct super_block *sb, int block){
-  check_bread++;
-  return sb_bread(sb, block);
+static int isofs_iget5_set(struct inode *ino, void *data)
+{
+       struct iso_inode_info *i = ISOFS_I(ino);
+       struct isofs_iget5_callback_data *d =
+               (struct isofs_iget5_callback_data*)data;
+       i->i_iget5_block = d->block;
+       i->i_iget5_offset = d->offset;
+       return 0;
 }
 
-void leak_check_brelse(struct buffer_head * bh){
-  check_bread--;
-  return brelse(bh);
-}
+/* Store, in the inode's containing structure, the block and block
+ * offset that point to the underlying meta-data for the inode.  The
+ * code below is otherwise similar to the iget() code in
+ * include/linux/fs.h */
+struct inode *isofs_iget(struct super_block *sb,
+                        unsigned long block,
+                        unsigned long offset)
+{
+       unsigned long hashval;
+       struct inode *inode;
+       struct isofs_iget5_callback_data data;
 
-#endif
+       if (offset >= 1ul << sb->s_blocksize_bits)
+               return NULL;
+
+       data.block = block;
+       data.offset = offset;
+
+       hashval = (block << sb->s_blocksize_bits) | offset;
+
+       inode = iget5_locked(sb, hashval, &isofs_iget5_test,
+                            &isofs_iget5_set, &data);
+
+       if (inode && (inode->i_state & I_NEW)) {
+               sb->s_op->read_inode(inode);
+               unlock_new_inode(inode);
+       }
+
+       return inode;
+}
 
-static struct super_block *isofs_get_sb(struct file_system_type *fs_type,
-       int flags, const char *dev_name, void *data)
+static int isofs_get_sb(struct file_system_type *fs_type,
+       int flags, const char *dev_name, void *data, struct vfsmount *mnt)
 {
-       return get_sb_bdev(fs_type, flags, dev_name, data, isofs_fill_super);
+       return get_sb_bdev(fs_type, flags, dev_name, data, isofs_fill_super,
+                          mnt);
 }
 
 static struct file_system_type iso9660_fs_type = {