Commit 1ec9307f authored by Darrick J. Wong's avatar Darrick J. Wong Committed by Chandan Babu R

xfs: allow unlinked symlinks and dirs with zero size

For a very very long time, inode inactivation has set the inode size to
zero before unmapping the extents associated with the data fork.
Unfortunately, commit 3c6f46ea changed the inode verifier to
prohibit zero-length symlinks and directories.  If an inode happens to
get logged in this state and the system crashes before freeing the
inode, log recovery will also fail on the broken inode.

Therefore, allow zero-size symlinks and directories as long as the link
count is zero; nobody will be able to open these files by handle so
there isn't any risk of data exposure.

Fixes: 3c6f46ea ("xfs: sanity check directory inode di_size")
Signed-off-by: default avatarDarrick J. Wong <djwong@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Signed-off-by: default avatarChandan Babu R <chandanbabu@kernel.org>
parent 288e1f69
...@@ -379,10 +379,13 @@ xfs_dinode_verify_fork( ...@@ -379,10 +379,13 @@ xfs_dinode_verify_fork(
/* /*
* A directory small enough to fit in the inode must be stored * A directory small enough to fit in the inode must be stored
* in local format. The directory sf <-> extents conversion * in local format. The directory sf <-> extents conversion
* code updates the directory size accordingly. * code updates the directory size accordingly. Directories
* being truncated have zero size and are not subject to this
* check.
*/ */
if (S_ISDIR(mode)) { if (S_ISDIR(mode)) {
if (be64_to_cpu(dip->di_size) <= fork_size && if (dip->di_size &&
be64_to_cpu(dip->di_size) <= fork_size &&
fork_format != XFS_DINODE_FMT_LOCAL) fork_format != XFS_DINODE_FMT_LOCAL)
return __this_address; return __this_address;
} }
...@@ -528,9 +531,19 @@ xfs_dinode_verify( ...@@ -528,9 +531,19 @@ xfs_dinode_verify(
if (mode && xfs_mode_to_ftype(mode) == XFS_DIR3_FT_UNKNOWN) if (mode && xfs_mode_to_ftype(mode) == XFS_DIR3_FT_UNKNOWN)
return __this_address; return __this_address;
/* No zero-length symlinks/dirs. */ /*
if ((S_ISLNK(mode) || S_ISDIR(mode)) && di_size == 0) * No zero-length symlinks/dirs unless they're unlinked and hence being
return __this_address; * inactivated.
*/
if ((S_ISLNK(mode) || S_ISDIR(mode)) && di_size == 0) {
if (dip->di_version > 1) {
if (dip->di_nlink)
return __this_address;
} else {
if (dip->di_onlink)
return __this_address;
}
}
fa = xfs_dinode_verify_nrext64(mp, dip); fa = xfs_dinode_verify_nrext64(mp, dip);
if (fa) if (fa)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment