udf: Make udf_read_inode() and udf_iget() return error
Currently __udf_read_inode() wasn't returning anything and we found out whether we succeeded reading inode by checking whether inode is bad or not. udf_iget() returned NULL on failure and inode pointer otherwise. Make these two functions properly propagate errors up the call stack and use the return value in callers. Signed-off-by: Jan Kara <jack@suse.cz>
This commit is contained in:
101
fs/udf/inode.c
101
fs/udf/inode.c
@@ -1277,7 +1277,7 @@ update_time:
|
||||
*/
|
||||
#define UDF_MAX_ICB_NESTING 1024
|
||||
|
||||
static void __udf_read_inode(struct inode *inode)
|
||||
static int udf_read_inode(struct inode *inode)
|
||||
{
|
||||
struct buffer_head *bh = NULL;
|
||||
struct fileEntry *fe;
|
||||
@@ -1285,10 +1285,19 @@ static void __udf_read_inode(struct inode *inode)
|
||||
uint16_t ident;
|
||||
struct udf_inode_info *iinfo = UDF_I(inode);
|
||||
struct udf_sb_info *sbi = UDF_SB(inode->i_sb);
|
||||
struct kernel_lb_addr *iloc = &iinfo->i_location;
|
||||
unsigned int link_count;
|
||||
unsigned int indirections = 0;
|
||||
int ret = -EIO;
|
||||
|
||||
reread:
|
||||
if (iloc->logicalBlockNum >=
|
||||
sbi->s_partmaps[iloc->partitionReferenceNum].s_partition_len) {
|
||||
udf_debug("block=%d, partition=%d out of range\n",
|
||||
iloc->logicalBlockNum, iloc->partitionReferenceNum);
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
/*
|
||||
* Set defaults, but the inode is still incomplete!
|
||||
* Note: get_new_inode() sets the following on a new inode:
|
||||
@@ -1301,20 +1310,17 @@ reread:
|
||||
* i_nlink = 1
|
||||
* i_op = NULL;
|
||||
*/
|
||||
bh = udf_read_ptagged(inode->i_sb, &iinfo->i_location, 0, &ident);
|
||||
bh = udf_read_ptagged(inode->i_sb, iloc, 0, &ident);
|
||||
if (!bh) {
|
||||
udf_err(inode->i_sb, "(ino %ld) failed !bh\n", inode->i_ino);
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
if (ident != TAG_IDENT_FE && ident != TAG_IDENT_EFE &&
|
||||
ident != TAG_IDENT_USE) {
|
||||
udf_err(inode->i_sb, "(ino %ld) failed ident=%d\n",
|
||||
inode->i_ino, ident);
|
||||
brelse(bh);
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
goto out;
|
||||
}
|
||||
|
||||
fe = (struct fileEntry *)bh->b_data;
|
||||
@@ -1323,8 +1329,7 @@ reread:
|
||||
if (fe->icbTag.strategyType == cpu_to_le16(4096)) {
|
||||
struct buffer_head *ibh;
|
||||
|
||||
ibh = udf_read_ptagged(inode->i_sb, &iinfo->i_location, 1,
|
||||
&ident);
|
||||
ibh = udf_read_ptagged(inode->i_sb, iloc, 1, &ident);
|
||||
if (ident == TAG_IDENT_IE && ibh) {
|
||||
struct kernel_lb_addr loc;
|
||||
struct indirectEntry *ie;
|
||||
@@ -1333,7 +1338,6 @@ reread:
|
||||
loc = lelb_to_cpu(ie->indirectICB.extLocation);
|
||||
|
||||
if (ie->indirectICB.extLength) {
|
||||
brelse(bh);
|
||||
brelse(ibh);
|
||||
memcpy(&iinfo->i_location, &loc,
|
||||
sizeof(struct kernel_lb_addr));
|
||||
@@ -1342,9 +1346,9 @@ reread:
|
||||
"too many ICBs in ICB hierarchy"
|
||||
" (max %d supported)\n",
|
||||
UDF_MAX_ICB_NESTING);
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
goto out;
|
||||
}
|
||||
brelse(bh);
|
||||
goto reread;
|
||||
}
|
||||
}
|
||||
@@ -1352,9 +1356,7 @@ reread:
|
||||
} else if (fe->icbTag.strategyType != cpu_to_le16(4)) {
|
||||
udf_err(inode->i_sb, "unsupported strategy type: %d\n",
|
||||
le16_to_cpu(fe->icbTag.strategyType));
|
||||
brelse(bh);
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
goto out;
|
||||
}
|
||||
if (fe->icbTag.strategyType == cpu_to_le16(4))
|
||||
iinfo->i_strat4096 = 0;
|
||||
@@ -1372,11 +1374,10 @@ reread:
|
||||
if (fe->descTag.tagIdent == cpu_to_le16(TAG_IDENT_EFE)) {
|
||||
iinfo->i_efe = 1;
|
||||
iinfo->i_use = 0;
|
||||
if (udf_alloc_i_data(inode, inode->i_sb->s_blocksize -
|
||||
sizeof(struct extendedFileEntry))) {
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
}
|
||||
ret = udf_alloc_i_data(inode, inode->i_sb->s_blocksize -
|
||||
sizeof(struct extendedFileEntry));
|
||||
if (ret)
|
||||
goto out;
|
||||
memcpy(iinfo->i_ext.i_data,
|
||||
bh->b_data + sizeof(struct extendedFileEntry),
|
||||
inode->i_sb->s_blocksize -
|
||||
@@ -1384,11 +1385,10 @@ reread:
|
||||
} else if (fe->descTag.tagIdent == cpu_to_le16(TAG_IDENT_FE)) {
|
||||
iinfo->i_efe = 0;
|
||||
iinfo->i_use = 0;
|
||||
if (udf_alloc_i_data(inode, inode->i_sb->s_blocksize -
|
||||
sizeof(struct fileEntry))) {
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
}
|
||||
ret = udf_alloc_i_data(inode, inode->i_sb->s_blocksize -
|
||||
sizeof(struct fileEntry));
|
||||
if (ret)
|
||||
goto out;
|
||||
memcpy(iinfo->i_ext.i_data,
|
||||
bh->b_data + sizeof(struct fileEntry),
|
||||
inode->i_sb->s_blocksize - sizeof(struct fileEntry));
|
||||
@@ -1398,18 +1398,18 @@ reread:
|
||||
iinfo->i_lenAlloc = le32_to_cpu(
|
||||
((struct unallocSpaceEntry *)bh->b_data)->
|
||||
lengthAllocDescs);
|
||||
if (udf_alloc_i_data(inode, inode->i_sb->s_blocksize -
|
||||
sizeof(struct unallocSpaceEntry))) {
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
}
|
||||
ret = udf_alloc_i_data(inode, inode->i_sb->s_blocksize -
|
||||
sizeof(struct unallocSpaceEntry));
|
||||
if (ret)
|
||||
goto out;
|
||||
memcpy(iinfo->i_ext.i_data,
|
||||
bh->b_data + sizeof(struct unallocSpaceEntry),
|
||||
inode->i_sb->s_blocksize -
|
||||
sizeof(struct unallocSpaceEntry));
|
||||
return;
|
||||
return 0;
|
||||
}
|
||||
|
||||
ret = -EIO;
|
||||
read_lock(&sbi->s_cred_lock);
|
||||
i_uid_write(inode, le32_to_cpu(fe->uid));
|
||||
if (!uid_valid(inode->i_uid) ||
|
||||
@@ -1531,8 +1531,7 @@ reread:
|
||||
default:
|
||||
udf_err(inode->i_sb, "(ino %ld) failed unknown file type=%d\n",
|
||||
inode->i_ino, fe->icbTag.fileType);
|
||||
make_bad_inode(inode);
|
||||
return;
|
||||
goto out;
|
||||
}
|
||||
if (S_ISCHR(inode->i_mode) || S_ISBLK(inode->i_mode)) {
|
||||
struct deviceSpec *dsea =
|
||||
@@ -1543,9 +1542,12 @@ reread:
|
||||
le32_to_cpu(dsea->minorDeviceIdent)));
|
||||
/* Developer ID ??? */
|
||||
} else
|
||||
make_bad_inode(inode);
|
||||
goto out;
|
||||
}
|
||||
ret = 0;
|
||||
out:
|
||||
brelse(bh);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int udf_alloc_i_data(struct inode *inode, size_t size)
|
||||
@@ -1825,32 +1827,23 @@ struct inode *udf_iget(struct super_block *sb, struct kernel_lb_addr *ino)
|
||||
{
|
||||
unsigned long block = udf_get_lb_pblock(sb, ino, 0);
|
||||
struct inode *inode = iget_locked(sb, block);
|
||||
int err;
|
||||
|
||||
if (!inode)
|
||||
return NULL;
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
if (inode->i_state & I_NEW) {
|
||||
memcpy(&UDF_I(inode)->i_location, ino, sizeof(struct kernel_lb_addr));
|
||||
__udf_read_inode(inode);
|
||||
unlock_new_inode(inode);
|
||||
}
|
||||
|
||||
if (is_bad_inode(inode))
|
||||
goto out_iput;
|
||||
|
||||
if (ino->logicalBlockNum >= UDF_SB(sb)->
|
||||
s_partmaps[ino->partitionReferenceNum].s_partition_len) {
|
||||
udf_debug("block=%d, partition=%d out of range\n",
|
||||
ino->logicalBlockNum, ino->partitionReferenceNum);
|
||||
make_bad_inode(inode);
|
||||
goto out_iput;
|
||||
if (!(inode->i_state & I_NEW))
|
||||
return inode;
|
||||
|
||||
memcpy(&UDF_I(inode)->i_location, ino, sizeof(struct kernel_lb_addr));
|
||||
err = udf_read_inode(inode);
|
||||
if (err < 0) {
|
||||
iget_failed(inode);
|
||||
return ERR_PTR(err);
|
||||
}
|
||||
unlock_new_inode(inode);
|
||||
|
||||
return inode;
|
||||
|
||||
out_iput:
|
||||
iput(inode);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int udf_add_aext(struct inode *inode, struct extent_position *epos,
|
||||
|
Reference in New Issue
Block a user