diff options
Diffstat (limited to 'fs/isofs/inode.c')
| -rw-r--r-- | fs/isofs/inode.c | 57 | 
1 files changed, 48 insertions, 9 deletions
| diff --git a/fs/isofs/inode.c b/fs/isofs/inode.c index 09ff41a752a0..79cf7f616bbe 100644 --- a/fs/isofs/inode.c +++ b/fs/isofs/inode.c @@ -544,6 +544,34 @@ static unsigned int isofs_get_last_session(struct super_block *sb, s32 session)  }  /* + * Check if root directory is empty (has less than 3 files). + * + * Used to detect broken CDs where ISO root directory is empty but Joliet root + * directory is OK. If such CD has Rock Ridge extensions, they will be disabled + * (and Joliet used instead) or else no files would be visible. + */ +static bool rootdir_empty(struct super_block *sb, unsigned long block) +{ +	int offset = 0, files = 0, de_len; +	struct iso_directory_record *de; +	struct buffer_head *bh; + +	bh = sb_bread(sb, block); +	if (!bh) +		return true; +	while (files < 3) { +		de = (struct iso_directory_record *) (bh->b_data + offset); +		de_len = *(unsigned char *) de; +		if (de_len == 0) +			break; +		files++; +		offset += de_len; +	} +	brelse(bh); +	return files < 3; +} + +/*   * Initialize the superblock and read the root inode.   *   * Note: a check_disk_change() has been done immediately prior @@ -843,6 +871,18 @@ root_found:  		goto out_no_root;  	/* +	 * Fix for broken CDs with Rock Ridge and empty ISO root directory but +	 * correct Joliet root directory. +	 */ +	if (sbi->s_rock == 1 && joliet_level && +				rootdir_empty(s, sbi->s_firstdatazone)) { +		printk(KERN_NOTICE +			"ISOFS: primary root directory is empty. " +			"Disabling Rock Ridge and switching to Joliet."); +		sbi->s_rock = 0; +	} + +	/*  	 * If this disk has both Rock Ridge and Joliet on it, then we  	 * want to use Rock Ridge by default.  This can be overridden  	 * by using the norock mount option.  There is still one other @@ -962,25 +1002,23 @@ static int isofs_statfs (struct dentry *dentry, struct kstatfs *buf)   * or getblk() if they are not.  Returns the number of blocks inserted   * (-ve == error.)   */ -int isofs_get_blocks(struct inode *inode, sector_t iblock_s, +int isofs_get_blocks(struct inode *inode, sector_t iblock,  		     struct buffer_head **bh, unsigned long nblocks)  { -	unsigned long b_off; +	unsigned long b_off = iblock;  	unsigned offset, sect_size;  	unsigned int firstext;  	unsigned long nextblk, nextoff; -	long iblock = (long)iblock_s;  	int section, rv, error;  	struct iso_inode_info *ei = ISOFS_I(inode);  	error = -EIO;  	rv = 0; -	if (iblock < 0 || iblock != iblock_s) { +	if (iblock != b_off) {  		printk(KERN_DEBUG "%s: block number too large\n", __func__);  		goto abort;  	} -	b_off = iblock;  	offset = 0;  	firstext = ei->i_first_extent; @@ -998,8 +1036,9 @@ int isofs_get_blocks(struct inode *inode, sector_t iblock_s,  		 * I/O errors.  		 */  		if (b_off > ((inode->i_size + PAGE_CACHE_SIZE - 1) >> ISOFS_BUFFER_BITS(inode))) { -			printk(KERN_DEBUG "%s: block >= EOF (%ld, %ld)\n", -				__func__, iblock, (unsigned long) inode->i_size); +			printk(KERN_DEBUG "%s: block >= EOF (%lu, %llu)\n", +				__func__, b_off, +				(unsigned long long)inode->i_size);  			goto abort;  		} @@ -1025,9 +1064,9 @@ int isofs_get_blocks(struct inode *inode, sector_t iblock_s,  			if (++section > 100) {  				printk(KERN_DEBUG "%s: More than 100 file sections ?!?"  					" aborting...\n", __func__); -				printk(KERN_DEBUG "%s: block=%ld firstext=%u sect_size=%u " +				printk(KERN_DEBUG "%s: block=%lu firstext=%u sect_size=%u "  					"nextblk=%lu nextoff=%lu\n", __func__, -					iblock, firstext, (unsigned) sect_size, +					b_off, firstext, (unsigned) sect_size,  					nextblk, nextoff);  				goto abort;  			} |