[GFS2] Fix bug in directory expansion code

We didn't properly check that leaf splitting was allowed. We do
now.

Signed-off-by: Steven Whitehouse <swhiteho@redhat.com>
This commit is contained in:
Steven Whitehouse 2006-03-29 19:02:15 -05:00
parent d0dc80dbaf
commit e90deff533

View File

@ -940,10 +940,15 @@ static int dir_split_leaf(struct inode *inode, const struct qstr *name)
/* Get the old leaf block */ /* Get the old leaf block */
error = get_leaf(dip, leaf_no, &obh); error = get_leaf(dip, leaf_no, &obh);
if (error) if (error)
goto fail; return error;
oleaf = (struct gfs2_leaf *)obh->b_data;
if (dip->i_di.di_depth == be16_to_cpu(oleaf->lf_depth)) {
brelse(obh);
return 1; /* can't split */
}
gfs2_trans_add_bh(dip->i_gl, obh, 1); gfs2_trans_add_bh(dip->i_gl, obh, 1);
oleaf = (struct gfs2_leaf *)obh->b_data;
nleaf = new_leaf(inode, &nbh, be16_to_cpu(oleaf->lf_depth) + 1); nleaf = new_leaf(inode, &nbh, be16_to_cpu(oleaf->lf_depth) + 1);
if (!nleaf) { if (!nleaf) {
@ -956,6 +961,7 @@ static int dir_split_leaf(struct inode *inode, const struct qstr *name)
len = 1 << (dip->i_di.di_depth - be16_to_cpu(oleaf->lf_depth)); len = 1 << (dip->i_di.di_depth - be16_to_cpu(oleaf->lf_depth));
half_len = len >> 1; half_len = len >> 1;
if (!half_len) { if (!half_len) {
printk(KERN_WARNING "di_depth %u lf_depth %u index %u\n", dip->i_di.di_depth, be16_to_cpu(oleaf->lf_depth), index);
gfs2_consist_inode(dip); gfs2_consist_inode(dip);
error = -EIO; error = -EIO;
goto fail_brelse; goto fail_brelse;
@ -1038,13 +1044,11 @@ static int dir_split_leaf(struct inode *inode, const struct qstr *name)
return error; return error;
fail_lpfree: fail_lpfree:
kfree(lp); kfree(lp);
fail_brelse: fail_brelse:
brelse(obh); brelse(obh);
fail:
brelse(nbh); brelse(nbh);
return error; return error;
} }
@ -1570,16 +1574,17 @@ int gfs2_dir_add(struct inode *inode, const struct qstr *name,
error = dir_split_leaf(inode, name); error = dir_split_leaf(inode, name);
if (error == 0) if (error == 0)
continue; continue;
if (error != -ENOSPC) if (error < 0)
break; break;
if (ip->i_di.di_depth < GFS2_DIR_MAX_DEPTH) { if (ip->i_di.di_depth < GFS2_DIR_MAX_DEPTH) {
error = dir_double_exhash(ip); error = dir_double_exhash(ip);
if (error) if (error)
break; break;
error = dir_split_leaf(inode, name); error = dir_split_leaf(inode, name);
if (error) if (error < 0)
break; break;
continue; if (error == 0)
continue;
} }
error = dir_new_leaf(inode, name); error = dir_new_leaf(inode, name);
if (!error) if (!error)