btrfs: avoid potential out-of-bounds in btrfs_encode_fh()

The function btrfs_encode_fh() does not properly account for the three
cases it handles.

Before writing to the file handle (fh), the function only returns to the
user BTRFS_FID_SIZE_NON_CONNECTABLE (5 dwords, 20 bytes) or
BTRFS_FID_SIZE_CONNECTABLE (8 dwords, 32 bytes).

However, when a parent exists and the root ID of the parent and the
inode are different, the function writes BTRFS_FID_SIZE_CONNECTABLE_ROOT
(10 dwords, 40 bytes).

If *max_len is not large enough, this write goes out of bounds because
BTRFS_FID_SIZE_CONNECTABLE_ROOT is greater than
BTRFS_FID_SIZE_CONNECTABLE originally returned.

This results in an 8-byte out-of-bounds write at
fid->parent_root_objectid = parent_root_id.

A previous attempt to fix this issue was made but was lost.

https://lore.kernel.org/all/4CADAEEC020000780001B32C@vpn.id2.novell.com/

Although this issue does not seem to be easily triggerable, it is a
potential memory corruption bug that should be fixed. This patch
resolves the issue by ensuring the function returns the appropriate size
for all three cases and validates that *max_len is large enough before
writing any data.

Fixes: be6e8dc0ba84 ("NFS support for btrfs - v3")
CC: stable@vger.kernel.org # 3.0+
Signed-off-by: Anderson Nascimento <anderson@allelesecurity.com>
Reviewed-by: David Sterba <dsterba@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>

authored by Anderson Nascimento and committed by David Sterba dff4f9ff 45c22246

+7 -1
+7 -1
fs/btrfs/export.c
··· 23 23 int type; 24 24 25 25 if (parent && (len < BTRFS_FID_SIZE_CONNECTABLE)) { 26 - *max_len = BTRFS_FID_SIZE_CONNECTABLE; 26 + if (btrfs_root_id(BTRFS_I(inode)->root) != 27 + btrfs_root_id(BTRFS_I(parent)->root)) 28 + *max_len = BTRFS_FID_SIZE_CONNECTABLE_ROOT; 29 + else 30 + *max_len = BTRFS_FID_SIZE_CONNECTABLE; 27 31 return FILEID_INVALID; 28 32 } else if (len < BTRFS_FID_SIZE_NON_CONNECTABLE) { 29 33 *max_len = BTRFS_FID_SIZE_NON_CONNECTABLE; ··· 49 45 parent_root_id = btrfs_root_id(BTRFS_I(parent)->root); 50 46 51 47 if (parent_root_id != fid->root_objectid) { 48 + if (*max_len < BTRFS_FID_SIZE_CONNECTABLE_ROOT) 49 + return FILEID_INVALID; 52 50 fid->parent_root_objectid = parent_root_id; 53 51 len = BTRFS_FID_SIZE_CONNECTABLE_ROOT; 54 52 type = FILEID_BTRFS_WITH_PARENT_ROOT;