mirror of
https://github.com/smaeul/u-boot.git
synced 2025-10-24 17:48:14 +01:00
If btrfs_read_fs_root() fails with -ENOENT, then we go to the next entry. Fine. But if it fails for a different reason then we need to clean up and return an error code. In the current code it doesn't clean up but instead dereferences "root" and crashes. Signed-off-by: Dan Carpenter <dan.carpenter@linaro.org> Reviewed-by: Marek Behún <kabel@kernel.org> Reviewed-by: Qu Wenruo <wqu@suse.com>
236 lines
5.4 KiB
C
236 lines
5.4 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* BTRFS filesystem implementation for U-Boot
|
|
*
|
|
* 2017 Marek Behún, CZ.NIC, kabel@kernel.org
|
|
*/
|
|
|
|
#include <malloc.h>
|
|
#include "ctree.h"
|
|
#include "btrfs.h"
|
|
#include "disk-io.h"
|
|
|
|
/*
|
|
* Resolve the path of ino inside subvolume @root into @path_ret.
|
|
*
|
|
* @path_ret must be at least PATH_MAX size.
|
|
*/
|
|
static int get_path_in_subvol(struct btrfs_root *root, u64 ino, char *path_ret)
|
|
{
|
|
struct btrfs_path path;
|
|
struct btrfs_key key;
|
|
char *tmp;
|
|
u64 cur = ino;
|
|
int ret = 0;
|
|
|
|
tmp = malloc(PATH_MAX);
|
|
if (!tmp)
|
|
return -ENOMEM;
|
|
tmp[0] = '\0';
|
|
|
|
btrfs_init_path(&path);
|
|
while (cur != BTRFS_FIRST_FREE_OBJECTID) {
|
|
struct btrfs_inode_ref *iref;
|
|
int name_len;
|
|
|
|
btrfs_release_path(&path);
|
|
key.objectid = cur;
|
|
key.type = BTRFS_INODE_REF_KEY;
|
|
key.offset = (u64)-1;
|
|
|
|
ret = btrfs_search_slot(NULL, root, &key, &path, 0, 0);
|
|
/* Impossible */
|
|
if (ret == 0)
|
|
ret = -EUCLEAN;
|
|
if (ret < 0)
|
|
goto out;
|
|
ret = btrfs_previous_item(root, &path, cur,
|
|
BTRFS_INODE_REF_KEY);
|
|
if (ret > 0)
|
|
ret = -ENOENT;
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
strncpy(tmp, path_ret, PATH_MAX);
|
|
iref = btrfs_item_ptr(path.nodes[0], path.slots[0],
|
|
struct btrfs_inode_ref);
|
|
name_len = btrfs_inode_ref_name_len(path.nodes[0],
|
|
iref);
|
|
if (name_len > BTRFS_NAME_LEN) {
|
|
ret = -ENAMETOOLONG;
|
|
goto out;
|
|
}
|
|
read_extent_buffer(path.nodes[0], path_ret,
|
|
(unsigned long)(iref + 1), name_len);
|
|
path_ret[name_len] = '/';
|
|
path_ret[name_len + 1] = '\0';
|
|
strncat(path_ret, tmp, PATH_MAX);
|
|
|
|
btrfs_item_key_to_cpu(path.nodes[0], &key, path.slots[0]);
|
|
cur = key.offset;
|
|
}
|
|
out:
|
|
btrfs_release_path(&path);
|
|
free(tmp);
|
|
return ret;
|
|
}
|
|
|
|
static int list_one_subvol(struct btrfs_root *root, char *path_ret)
|
|
{
|
|
struct btrfs_fs_info *fs_info = root->fs_info;
|
|
struct btrfs_root *tree_root = fs_info->tree_root;
|
|
struct btrfs_path path;
|
|
struct btrfs_key key;
|
|
char *tmp;
|
|
u64 cur = root->root_key.objectid;
|
|
int ret = 0;
|
|
|
|
tmp = malloc(PATH_MAX);
|
|
if (!tmp)
|
|
return -ENOMEM;
|
|
tmp[0] = '\0';
|
|
path_ret[0] = '\0';
|
|
btrfs_init_path(&path);
|
|
while (cur != BTRFS_FS_TREE_OBJECTID) {
|
|
struct btrfs_root_ref *rr;
|
|
struct btrfs_key location;
|
|
int name_len;
|
|
u64 ino;
|
|
|
|
key.objectid = cur;
|
|
key.type = BTRFS_ROOT_BACKREF_KEY;
|
|
key.offset = (u64)-1;
|
|
btrfs_release_path(&path);
|
|
|
|
ret = btrfs_search_slot(NULL, tree_root, &key, &path, 0, 0);
|
|
if (ret == 0)
|
|
ret = -EUCLEAN;
|
|
if (ret < 0)
|
|
goto out;
|
|
ret = btrfs_previous_item(tree_root, &path, cur,
|
|
BTRFS_ROOT_BACKREF_KEY);
|
|
if (ret > 0)
|
|
ret = -ENOENT;
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
/* Get the subvolume name */
|
|
rr = btrfs_item_ptr(path.nodes[0], path.slots[0],
|
|
struct btrfs_root_ref);
|
|
strncpy(tmp, path_ret, PATH_MAX);
|
|
name_len = btrfs_root_ref_name_len(path.nodes[0], rr);
|
|
if (name_len > BTRFS_NAME_LEN) {
|
|
ret = -ENAMETOOLONG;
|
|
goto out;
|
|
}
|
|
ino = btrfs_root_ref_dirid(path.nodes[0], rr);
|
|
read_extent_buffer(path.nodes[0], path_ret,
|
|
(unsigned long)(rr + 1), name_len);
|
|
path_ret[name_len] = '/';
|
|
path_ret[name_len + 1] = '\0';
|
|
strncat(path_ret, tmp, PATH_MAX);
|
|
|
|
/* Get the path inside the parent subvolume */
|
|
btrfs_item_key_to_cpu(path.nodes[0], &key, path.slots[0]);
|
|
location.objectid = key.offset;
|
|
location.type = BTRFS_ROOT_ITEM_KEY;
|
|
location.offset = (u64)-1;
|
|
root = btrfs_read_fs_root(fs_info, &location);
|
|
if (IS_ERR(root)) {
|
|
ret = PTR_ERR(root);
|
|
goto out;
|
|
}
|
|
ret = get_path_in_subvol(root, ino, path_ret);
|
|
if (ret < 0)
|
|
goto out;
|
|
cur = key.offset;
|
|
}
|
|
/* Add the leading '/' */
|
|
strncpy(tmp, path_ret, PATH_MAX);
|
|
strncpy(path_ret, "/", PATH_MAX);
|
|
strncat(path_ret, tmp, PATH_MAX);
|
|
out:
|
|
btrfs_release_path(&path);
|
|
free(tmp);
|
|
return ret;
|
|
}
|
|
|
|
static int list_subvolums(struct btrfs_fs_info *fs_info)
|
|
{
|
|
struct btrfs_root *tree_root = fs_info->tree_root;
|
|
struct btrfs_root *root;
|
|
struct btrfs_path path;
|
|
struct btrfs_key key;
|
|
char *result;
|
|
int ret = 0;
|
|
|
|
result = malloc(PATH_MAX);
|
|
if (!result)
|
|
return -ENOMEM;
|
|
|
|
ret = list_one_subvol(fs_info->fs_root, result);
|
|
if (ret < 0)
|
|
goto out;
|
|
root = fs_info->fs_root;
|
|
printf("ID %llu gen %llu path %.*s\n",
|
|
root->root_key.objectid, btrfs_root_generation(&root->root_item),
|
|
PATH_MAX, result);
|
|
|
|
key.objectid = BTRFS_FIRST_FREE_OBJECTID;
|
|
key.type = BTRFS_ROOT_ITEM_KEY;
|
|
key.offset = 0;
|
|
btrfs_init_path(&path);
|
|
ret = btrfs_search_slot(NULL, tree_root, &key, &path, 0, 0);
|
|
if (ret < 0)
|
|
goto out;
|
|
while (1) {
|
|
if (path.slots[0] >= btrfs_header_nritems(path.nodes[0]))
|
|
goto next;
|
|
|
|
btrfs_item_key_to_cpu(path.nodes[0], &key, path.slots[0]);
|
|
if (key.objectid > BTRFS_LAST_FREE_OBJECTID)
|
|
break;
|
|
if (key.objectid < BTRFS_FIRST_FREE_OBJECTID ||
|
|
key.type != BTRFS_ROOT_ITEM_KEY)
|
|
goto next;
|
|
key.offset = (u64)-1;
|
|
root = btrfs_read_fs_root(fs_info, &key);
|
|
if (IS_ERR(root)) {
|
|
ret = PTR_ERR(root);
|
|
if (ret == -ENOENT)
|
|
goto next;
|
|
goto out;
|
|
}
|
|
ret = list_one_subvol(root, result);
|
|
if (ret < 0)
|
|
goto out;
|
|
printf("ID %llu gen %llu path %.*s\n",
|
|
root->root_key.objectid,
|
|
btrfs_root_generation(&root->root_item),
|
|
PATH_MAX, result);
|
|
next:
|
|
ret = btrfs_next_item(tree_root, &path);
|
|
if (ret < 0)
|
|
goto out;
|
|
if (ret > 0) {
|
|
ret = 0;
|
|
break;
|
|
}
|
|
}
|
|
out:
|
|
free(result);
|
|
return ret;
|
|
}
|
|
|
|
void btrfs_list_subvols(void)
|
|
{
|
|
struct btrfs_fs_info *fs_info = current_fs_info;
|
|
int ret;
|
|
|
|
if (!fs_info)
|
|
return;
|
|
ret = list_subvolums(fs_info);
|
|
if (ret < 0)
|
|
error("failed to list subvolume: %d", ret);
|
|
}
|