2014-02-13 19:19:01 +00:00
|
|
|
/*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public
|
|
|
|
* License v2 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public
|
|
|
|
* License along with this program; if not, write to the
|
|
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 021110-1307, USA.
|
|
|
|
*/
|
|
|
|
|
2022-09-14 15:06:52 +00:00
|
|
|
#include "kerncompat.h"
|
|
|
|
#include <sys/ioctl.h>
|
2022-09-15 11:59:39 +00:00
|
|
|
#include <sys/statfs.h>
|
2022-09-14 15:06:52 +00:00
|
|
|
#include <linux/limits.h>
|
2023-08-28 20:12:13 +00:00
|
|
|
#include <stdbool.h>
|
2014-02-13 19:19:01 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
2017-10-23 04:44:01 +00:00
|
|
|
#include <fcntl.h>
|
2022-09-15 11:59:39 +00:00
|
|
|
#include <dirent.h>
|
|
|
|
#include <limits.h>
|
2023-02-13 09:37:42 +00:00
|
|
|
#include <uuid/uuid.h>
|
2022-09-15 11:59:39 +00:00
|
|
|
#include "kernel-lib/sizes.h"
|
2020-08-18 13:56:04 +00:00
|
|
|
#include "kernel-shared/ctree.h"
|
2020-08-18 13:56:04 +00:00
|
|
|
#include "kernel-shared/disk-io.h"
|
2021-10-06 21:22:31 +00:00
|
|
|
#include "kernel-shared/volumes.h"
|
2023-08-28 20:12:13 +00:00
|
|
|
#include "kernel-shared/accessors.h"
|
|
|
|
#include "kernel-shared/uapi/btrfs_tree.h"
|
|
|
|
#include "common/defs.h"
|
2022-09-14 15:06:52 +00:00
|
|
|
#include "common/utils.h"
|
|
|
|
#include "common/string-table.h"
|
2021-04-29 17:52:15 +00:00
|
|
|
#include "common/open-utils.h"
|
2021-04-29 21:43:24 +00:00
|
|
|
#include "common/units.h"
|
2019-06-19 23:46:21 +00:00
|
|
|
#include "common/help.h"
|
2019-07-01 22:42:23 +00:00
|
|
|
#include "common/device-utils.h"
|
2023-08-21 19:13:24 +00:00
|
|
|
#include "common/sysfs-utils.h"
|
2022-09-15 11:59:39 +00:00
|
|
|
#include "common/messages.h"
|
2023-02-13 09:37:42 +00:00
|
|
|
#include "common/path-utils.h"
|
2022-09-14 15:06:52 +00:00
|
|
|
#include "cmds/filesystem-usage.h"
|
|
|
|
#include "cmds/commands.h"
|
2014-02-13 19:19:01 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Add the chunk info to the chunk_info list
|
|
|
|
*/
|
2022-09-23 08:07:29 +00:00
|
|
|
static int add_info_to_list(struct chunk_info **chunkinfo_ret,
|
2014-02-13 19:19:01 +00:00
|
|
|
int *info_count,
|
|
|
|
struct btrfs_chunk *chunk)
|
|
|
|
{
|
|
|
|
|
|
|
|
u64 type = btrfs_stack_chunk_type(chunk);
|
|
|
|
u64 size = btrfs_stack_chunk_length(chunk);
|
|
|
|
int num_stripes = btrfs_stack_chunk_num_stripes(chunk);
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0 ; j < num_stripes ; j++) {
|
|
|
|
int i;
|
2016-01-04 01:01:30 +00:00
|
|
|
struct chunk_info *p = NULL;
|
2014-02-13 19:19:01 +00:00
|
|
|
struct btrfs_stripe *stripe;
|
|
|
|
u64 devid;
|
|
|
|
|
|
|
|
stripe = btrfs_stripe_nr(chunk, j);
|
|
|
|
devid = btrfs_stack_stripe_devid(stripe);
|
|
|
|
|
|
|
|
for (i = 0 ; i < *info_count ; i++)
|
2022-09-23 08:07:29 +00:00
|
|
|
if ((*chunkinfo_ret)[i].type == type &&
|
|
|
|
(*chunkinfo_ret)[i].devid == devid &&
|
|
|
|
(*chunkinfo_ret)[i].num_stripes == num_stripes ) {
|
|
|
|
p = (*chunkinfo_ret) + i;
|
2014-02-13 19:19:01 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!p) {
|
2016-01-12 13:34:41 +00:00
|
|
|
int tmp = sizeof(struct btrfs_chunk) * (*info_count + 1);
|
2022-09-23 08:07:29 +00:00
|
|
|
struct chunk_info *res = realloc(*chunkinfo_ret, tmp);
|
2014-02-13 19:19:01 +00:00
|
|
|
|
|
|
|
if (!res) {
|
2022-09-23 08:07:29 +00:00
|
|
|
free(*chunkinfo_ret);
|
2022-09-30 07:12:06 +00:00
|
|
|
error_msg(ERROR_MSG_MEMORY, NULL);
|
2014-04-28 16:55:05 +00:00
|
|
|
return -ENOMEM;
|
2014-02-13 19:19:01 +00:00
|
|
|
}
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
*chunkinfo_ret = res;
|
2014-02-13 19:19:01 +00:00
|
|
|
p = res + *info_count;
|
|
|
|
(*info_count)++;
|
|
|
|
|
|
|
|
p->devid = devid;
|
|
|
|
p->type = type;
|
|
|
|
p->size = 0;
|
|
|
|
p->num_stripes = num_stripes;
|
|
|
|
}
|
|
|
|
|
|
|
|
p->size += size;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helper to sort the chunk type
|
|
|
|
*/
|
|
|
|
static int cmp_chunk_block_group(u64 f1, u64 f2)
|
|
|
|
{
|
|
|
|
|
|
|
|
u64 mask;
|
|
|
|
|
|
|
|
if ((f1 & BTRFS_BLOCK_GROUP_TYPE_MASK) ==
|
|
|
|
(f2 & BTRFS_BLOCK_GROUP_TYPE_MASK))
|
|
|
|
mask = BTRFS_BLOCK_GROUP_PROFILE_MASK;
|
|
|
|
else if (f2 & BTRFS_BLOCK_GROUP_SYSTEM)
|
|
|
|
return -1;
|
|
|
|
else if (f1 & BTRFS_BLOCK_GROUP_SYSTEM)
|
|
|
|
return +1;
|
|
|
|
else
|
|
|
|
mask = BTRFS_BLOCK_GROUP_TYPE_MASK;
|
|
|
|
|
|
|
|
if ((f1 & mask) > (f2 & mask))
|
|
|
|
return +1;
|
|
|
|
else if ((f1 & mask) < (f2 & mask))
|
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helper to sort the chunk
|
|
|
|
*/
|
|
|
|
static int cmp_chunk_info(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
return cmp_chunk_block_group(
|
|
|
|
((struct chunk_info *)a)->type,
|
|
|
|
((struct chunk_info *)b)->type);
|
|
|
|
}
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
static int load_chunk_info(int fd, struct chunk_info **chunkinfo_ret,
|
|
|
|
int *chunkcount_ret)
|
2014-02-13 19:19:01 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct btrfs_ioctl_search_args args;
|
|
|
|
struct btrfs_ioctl_search_key *sk = &args.key;
|
|
|
|
struct btrfs_ioctl_search_header *sh;
|
|
|
|
unsigned long off = 0;
|
2023-08-23 14:27:50 +00:00
|
|
|
int i;
|
2014-02-13 19:19:01 +00:00
|
|
|
|
|
|
|
memset(&args, 0, sizeof(args));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* there may be more than one ROOT_ITEM key if there are
|
|
|
|
* snapshots pending deletion, we have to loop through
|
|
|
|
* them.
|
|
|
|
*/
|
|
|
|
sk->tree_id = BTRFS_CHUNK_TREE_OBJECTID;
|
|
|
|
|
|
|
|
sk->min_objectid = 0;
|
|
|
|
sk->max_objectid = (u64)-1;
|
|
|
|
sk->max_type = 0;
|
|
|
|
sk->min_type = (u8)-1;
|
|
|
|
sk->min_offset = 0;
|
|
|
|
sk->max_offset = (u64)-1;
|
|
|
|
sk->min_transid = 0;
|
|
|
|
sk->max_transid = (u64)-1;
|
|
|
|
sk->nr_items = 4096;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
ret = ioctl(fd, BTRFS_IOC_TREE_SEARCH, &args);
|
|
|
|
if (ret < 0) {
|
2023-08-23 14:27:50 +00:00
|
|
|
if (errno == EPERM)
|
|
|
|
return -errno;
|
2018-01-07 21:54:21 +00:00
|
|
|
error("cannot look up chunk tree info: %m");
|
2014-07-24 03:21:54 +00:00
|
|
|
return 1;
|
2014-02-13 19:19:01 +00:00
|
|
|
}
|
|
|
|
/* the ioctl returns the number of item it found in nr_items */
|
|
|
|
|
|
|
|
if (sk->nr_items == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
off = 0;
|
|
|
|
for (i = 0; i < sk->nr_items; i++) {
|
|
|
|
struct btrfs_chunk *item;
|
|
|
|
sh = (struct btrfs_ioctl_search_header *)(args.buf +
|
|
|
|
off);
|
|
|
|
|
|
|
|
off += sizeof(*sh);
|
|
|
|
item = (struct btrfs_chunk *)(args.buf + off);
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
ret = add_info_to_list(chunkinfo_ret, chunkcount_ret, item);
|
2014-04-28 16:55:05 +00:00
|
|
|
if (ret) {
|
2022-09-23 08:07:29 +00:00
|
|
|
*chunkinfo_ret = NULL;
|
2014-07-24 03:21:54 +00:00
|
|
|
return 1;
|
2014-02-13 19:19:01 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 14:20:52 +00:00
|
|
|
off += btrfs_search_header_len(sh);
|
2014-02-13 19:19:01 +00:00
|
|
|
|
2016-05-03 14:20:52 +00:00
|
|
|
sk->min_objectid = btrfs_search_header_objectid(sh);
|
|
|
|
sk->min_type = btrfs_search_header_type(sh);
|
|
|
|
sk->min_offset = btrfs_search_header_offset(sh)+1;
|
2014-02-13 19:19:01 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
if (!sk->min_offset) /* overflow */
|
|
|
|
sk->min_type++;
|
|
|
|
else
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!sk->min_type)
|
|
|
|
sk->min_objectid++;
|
|
|
|
else
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!sk->min_objectid)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
qsort(*chunkinfo_ret, *chunkcount_ret, sizeof(struct chunk_info),
|
2014-02-13 19:19:01 +00:00
|
|
|
cmp_chunk_info);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helper to sort the struct btrfs_ioctl_space_info
|
|
|
|
*/
|
|
|
|
static int cmp_btrfs_ioctl_space_info(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
return cmp_chunk_block_group(
|
|
|
|
((struct btrfs_ioctl_space_info *)a)->flags,
|
|
|
|
((struct btrfs_ioctl_space_info *)b)->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function load all the information about the space usage
|
|
|
|
*/
|
2018-05-16 21:38:36 +00:00
|
|
|
static struct btrfs_ioctl_space_args *load_space_info(int fd, const char *path)
|
2014-02-13 19:19:01 +00:00
|
|
|
{
|
2016-01-04 01:01:30 +00:00
|
|
|
struct btrfs_ioctl_space_args *sargs = NULL, *sargs_orig = NULL;
|
2016-01-12 10:20:18 +00:00
|
|
|
int ret, count;
|
2014-02-13 19:19:01 +00:00
|
|
|
|
2014-04-24 16:37:50 +00:00
|
|
|
sargs_orig = sargs = calloc(1, sizeof(struct btrfs_ioctl_space_args));
|
2014-02-13 19:19:01 +00:00
|
|
|
if (!sargs) {
|
2022-09-30 07:12:06 +00:00
|
|
|
error_msg(ERROR_MSG_MEMORY, NULL);
|
2014-02-13 19:19:01 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
sargs->space_slots = 0;
|
|
|
|
sargs->total_spaces = 0;
|
|
|
|
|
|
|
|
ret = ioctl(fd, BTRFS_IOC_SPACE_INFO, sargs);
|
2016-01-12 12:35:50 +00:00
|
|
|
if (ret < 0) {
|
2018-01-07 21:54:21 +00:00
|
|
|
error("cannot get space info on '%s': %m", path);
|
2014-02-13 19:19:01 +00:00
|
|
|
free(sargs);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (!sargs->total_spaces) {
|
|
|
|
free(sargs);
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "No chunks found\n");
|
2014-02-13 19:19:01 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
count = sargs->total_spaces;
|
|
|
|
|
|
|
|
sargs = realloc(sargs, sizeof(struct btrfs_ioctl_space_args) +
|
|
|
|
(count * sizeof(struct btrfs_ioctl_space_info)));
|
|
|
|
if (!sargs) {
|
|
|
|
free(sargs_orig);
|
2022-09-30 07:12:06 +00:00
|
|
|
error_msg(ERROR_MSG_MEMORY, NULL);
|
2014-02-13 19:19:01 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
sargs->space_slots = count;
|
|
|
|
sargs->total_spaces = 0;
|
|
|
|
|
|
|
|
ret = ioctl(fd, BTRFS_IOC_SPACE_INFO, sargs);
|
2016-01-12 12:35:50 +00:00
|
|
|
if (ret < 0) {
|
2018-01-07 21:54:21 +00:00
|
|
|
error("cannot get space info with %u slots: %m",
|
|
|
|
count);
|
2014-02-13 19:19:01 +00:00
|
|
|
free(sargs);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
qsort(&(sargs->spaces), count, sizeof(struct btrfs_ioctl_space_info),
|
|
|
|
cmp_btrfs_ioctl_space_info);
|
|
|
|
|
|
|
|
return sargs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2020-05-27 20:37:48 +00:00
|
|
|
* Compute the ratio between logical space used over logical space allocated
|
|
|
|
* by profile basis
|
2014-02-13 19:19:01 +00:00
|
|
|
*/
|
2020-05-27 20:37:48 +00:00
|
|
|
static void get_raid56_logical_ratio(struct btrfs_ioctl_space_args *sargs,
|
|
|
|
u64 type, double *data_ratio,
|
|
|
|
double *metadata_ratio,
|
|
|
|
double *system_ratio)
|
2014-02-13 19:19:01 +00:00
|
|
|
{
|
2020-05-27 20:37:48 +00:00
|
|
|
u64 l_data_chunk = 0, l_data_used = 0;
|
|
|
|
u64 l_metadata_chunk = 0, l_metadata_used = 0;
|
|
|
|
u64 l_system_chunk = 0, l_system_used = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < sargs->total_spaces; i++) {
|
|
|
|
u64 flags = sargs->spaces[i].flags;
|
|
|
|
|
|
|
|
if (!(flags & type))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (flags & BTRFS_BLOCK_GROUP_DATA) {
|
|
|
|
l_data_used += sargs->spaces[i].used_bytes;
|
|
|
|
l_data_chunk += sargs->spaces[i].total_bytes;
|
|
|
|
} else if (flags & BTRFS_BLOCK_GROUP_METADATA) {
|
|
|
|
l_metadata_used += sargs->spaces[i].used_bytes;
|
|
|
|
l_metadata_chunk += sargs->spaces[i].total_bytes;
|
|
|
|
} else if (flags & BTRFS_BLOCK_GROUP_SYSTEM) {
|
|
|
|
l_system_used += sargs->spaces[i].used_bytes;
|
|
|
|
l_system_chunk += sargs->spaces[i].total_bytes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*data_ratio = -1.0;
|
|
|
|
*metadata_ratio = -1.0;
|
|
|
|
*system_ratio = -1.0;
|
|
|
|
|
|
|
|
if (l_data_chunk)
|
|
|
|
*data_ratio = (double)l_data_used / l_data_chunk;
|
|
|
|
if (l_metadata_chunk)
|
|
|
|
*metadata_ratio = (double)l_metadata_used / l_metadata_chunk;
|
|
|
|
if (l_system_chunk)
|
|
|
|
*system_ratio = (double)l_system_used / l_system_chunk;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Compute the "raw" space allocated for a chunk (r_*_chunks)
|
|
|
|
* and the "raw" space used by a chunk (r_*_used)
|
|
|
|
*/
|
|
|
|
static void get_raid56_space_info(struct btrfs_ioctl_space_args *sargs,
|
2022-09-23 08:07:29 +00:00
|
|
|
struct chunk_info *chunkinfo, int chunkcount,
|
2020-05-27 20:37:48 +00:00
|
|
|
double *max_data_ratio,
|
|
|
|
u64 *r_data_chunks, u64 *r_data_used,
|
|
|
|
u64 *r_metadata_chunks, u64 *r_metadata_used,
|
|
|
|
u64 *r_system_chunks, u64 *r_system_used)
|
|
|
|
{
|
|
|
|
struct chunk_info *info_ptr;
|
|
|
|
double l_data_ratio_r5, l_metadata_ratio_r5, l_system_ratio_r5;
|
|
|
|
double l_data_ratio_r6, l_metadata_ratio_r6, l_system_ratio_r6;
|
|
|
|
|
|
|
|
get_raid56_logical_ratio(sargs, BTRFS_BLOCK_GROUP_RAID5,
|
|
|
|
&l_data_ratio_r5, &l_metadata_ratio_r5, &l_system_ratio_r5);
|
|
|
|
get_raid56_logical_ratio(sargs, BTRFS_BLOCK_GROUP_RAID6,
|
|
|
|
&l_data_ratio_r6, &l_metadata_ratio_r6, &l_system_ratio_r6);
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
for(info_ptr = chunkinfo; chunkcount > 0; chunkcount--, info_ptr++) {
|
2020-05-27 20:37:48 +00:00
|
|
|
int parities_count;
|
|
|
|
u64 size;
|
|
|
|
double l_data_ratio, l_metadata_ratio, l_system_ratio, rt;
|
|
|
|
|
2021-10-06 22:02:41 +00:00
|
|
|
parities_count = btrfs_bg_type_to_nparity(info_ptr->type);
|
2020-05-27 20:37:48 +00:00
|
|
|
if (info_ptr->type & BTRFS_BLOCK_GROUP_RAID5) {
|
|
|
|
l_data_ratio = l_data_ratio_r5;
|
|
|
|
l_metadata_ratio = l_metadata_ratio_r5;
|
|
|
|
l_system_ratio = l_system_ratio_r5;
|
|
|
|
} else if (info_ptr->type & BTRFS_BLOCK_GROUP_RAID6) {
|
|
|
|
l_data_ratio = l_data_ratio_r6;
|
|
|
|
l_metadata_ratio = l_metadata_ratio_r6;
|
|
|
|
l_system_ratio = l_system_ratio_r6;
|
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
rt = (double)info_ptr->num_stripes /
|
|
|
|
(info_ptr->num_stripes - parities_count);
|
|
|
|
if (rt > *max_data_ratio)
|
|
|
|
*max_data_ratio = rt;
|
|
|
|
|
|
|
|
/*
|
2020-12-25 19:15:11 +00:00
|
|
|
* size is the total disk(s) space occupied by a chunk
|
2020-05-27 20:37:48 +00:00
|
|
|
* the product of 'size' and '*_ratio' is "in average"
|
|
|
|
* the disk(s) space used by the data
|
|
|
|
*/
|
|
|
|
size = info_ptr->size / (info_ptr->num_stripes - parities_count);
|
|
|
|
|
|
|
|
if (info_ptr->type & BTRFS_BLOCK_GROUP_DATA) {
|
2023-04-19 21:13:46 +00:00
|
|
|
UASSERT(l_data_ratio >= 0);
|
2020-05-27 20:37:48 +00:00
|
|
|
*r_data_chunks += size;
|
|
|
|
*r_data_used += size * l_data_ratio;
|
|
|
|
} else if (info_ptr->type & BTRFS_BLOCK_GROUP_METADATA) {
|
2023-04-19 21:13:46 +00:00
|
|
|
UASSERT(l_metadata_ratio >= 0);
|
2020-05-27 20:37:48 +00:00
|
|
|
*r_metadata_chunks += size;
|
|
|
|
*r_metadata_used += size * l_metadata_ratio;
|
|
|
|
} else if (info_ptr->type & BTRFS_BLOCK_GROUP_SYSTEM) {
|
2023-04-19 21:13:46 +00:00
|
|
|
UASSERT(l_system_ratio >= 0);
|
2020-05-27 20:37:48 +00:00
|
|
|
*r_system_chunks += size;
|
|
|
|
*r_system_used += size * l_system_ratio;
|
|
|
|
}
|
2014-02-13 19:19:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-24 21:01:09 +00:00
|
|
|
static u64 get_first_device_zone_size(int fd)
|
|
|
|
{
|
|
|
|
int dirfd;
|
|
|
|
DIR *dir;
|
|
|
|
struct dirent *de;
|
|
|
|
char name[NAME_MAX] = {0};
|
|
|
|
u64 ret;
|
|
|
|
|
|
|
|
dirfd = sysfs_open_fsid_dir(fd, "devices");
|
|
|
|
if (dirfd < 0)
|
|
|
|
return 0;
|
|
|
|
dir = fdopendir(dirfd);
|
|
|
|
if (!dir) {
|
|
|
|
ret = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
while (1) {
|
|
|
|
de = readdir(dir);
|
|
|
|
if (strcmp(".", de->d_name) == 0 || strcmp("..", de->d_name) == 0)
|
|
|
|
continue;
|
|
|
|
strcpy(name, de->d_name);
|
|
|
|
name[NAME_MAX - 1] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ret = device_get_zone_size(fd, name);
|
|
|
|
ret *= 512;
|
|
|
|
|
|
|
|
out:
|
|
|
|
closedir(dir);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-09-14 19:20:43 +00:00
|
|
|
static u64 calc_slack_size(const struct device_info *devinfo)
|
|
|
|
{
|
|
|
|
if (devinfo->device_size > 0)
|
|
|
|
return devinfo->device_size - devinfo->size;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-01-24 03:03:05 +00:00
|
|
|
#define MIN_UNALOCATED_THRESH SZ_16M
|
2014-04-25 17:39:11 +00:00
|
|
|
static int print_filesystem_usage_overall(int fd, struct chunk_info *chunkinfo,
|
|
|
|
int chunkcount, struct device_info *devinfo, int devcount,
|
2018-05-16 21:38:36 +00:00
|
|
|
const char *path, unsigned unit_mode)
|
2014-02-13 19:19:01 +00:00
|
|
|
{
|
2016-01-04 01:01:30 +00:00
|
|
|
struct btrfs_ioctl_space_args *sargs = NULL;
|
2020-05-04 20:03:09 +00:00
|
|
|
char *tmp;
|
2014-02-13 19:19:01 +00:00
|
|
|
int i;
|
|
|
|
int ret = 0;
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
int width = 10; /* default 10 for human units */
|
|
|
|
/*
|
|
|
|
* r_* prefix is for raw data
|
2020-05-27 20:37:48 +00:00
|
|
|
* l_* prefix is for logical
|
|
|
|
* *_used suffix is for space used for data or metadata
|
|
|
|
* *_chunks suffix is for total space used by the chunk
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
*/
|
|
|
|
u64 r_total_size = 0; /* filesystem size, sum of device sizes */
|
|
|
|
u64 r_total_chunks = 0; /* sum of chunks sizes on disk(s) */
|
|
|
|
u64 r_total_used = 0;
|
|
|
|
u64 r_total_unused = 0;
|
2015-03-17 13:41:01 +00:00
|
|
|
u64 r_total_missing = 0; /* sum of missing devices size */
|
2022-09-14 19:20:43 +00:00
|
|
|
u64 r_total_slack = 0;
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
u64 r_data_used = 0;
|
|
|
|
u64 r_data_chunks = 0;
|
|
|
|
u64 l_data_chunks = 0;
|
|
|
|
u64 r_metadata_used = 0;
|
|
|
|
u64 r_metadata_chunks = 0;
|
|
|
|
u64 l_metadata_chunks = 0;
|
|
|
|
u64 r_system_used = 0;
|
|
|
|
u64 r_system_chunks = 0;
|
|
|
|
double data_ratio;
|
|
|
|
double metadata_ratio;
|
|
|
|
/* logical */
|
|
|
|
u64 l_global_reserve = 0;
|
|
|
|
u64 l_global_reserve_used = 0;
|
|
|
|
u64 free_estimated = 0;
|
|
|
|
u64 free_min = 0;
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
u64 zone_unusable = 0;
|
2020-05-27 20:37:48 +00:00
|
|
|
double max_data_ratio = 1.0;
|
2022-10-05 18:02:13 +00:00
|
|
|
bool mixed = false;
|
2020-11-10 00:52:21 +00:00
|
|
|
struct statfs statfs_buf;
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
struct btrfs_ioctl_feature_flags feature_flags;
|
2014-02-13 19:19:01 +00:00
|
|
|
|
2014-04-28 16:55:05 +00:00
|
|
|
sargs = load_space_info(fd, path);
|
|
|
|
if (!sargs) {
|
|
|
|
ret = 1;
|
2014-02-13 19:19:01 +00:00
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
r_total_size = 0;
|
2015-03-17 13:41:01 +00:00
|
|
|
for (i = 0; i < devcount; i++) {
|
|
|
|
r_total_size += devinfo[i].size;
|
2022-09-14 19:20:43 +00:00
|
|
|
r_total_slack += calc_slack_size(&devinfo[i]);
|
2015-03-17 13:41:01 +00:00
|
|
|
if (!devinfo[i].device_size)
|
|
|
|
r_total_missing += devinfo[i].size;
|
|
|
|
}
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
|
|
|
|
if (r_total_size == 0) {
|
2018-01-07 21:54:21 +00:00
|
|
|
error("cannot get space info on '%s': %m", path);
|
2014-02-13 19:19:01 +00:00
|
|
|
|
2014-04-28 16:55:05 +00:00
|
|
|
ret = 1;
|
2014-02-13 19:19:01 +00:00
|
|
|
goto exit;
|
|
|
|
}
|
2020-05-27 20:37:48 +00:00
|
|
|
|
|
|
|
get_raid56_space_info(sargs, chunkinfo, chunkcount, &max_data_ratio,
|
|
|
|
&r_data_chunks, &r_data_used,
|
|
|
|
&r_metadata_chunks, &r_metadata_used,
|
|
|
|
&r_system_chunks, &r_system_used);
|
2014-02-13 19:19:01 +00:00
|
|
|
|
|
|
|
for (i = 0; i < sargs->total_spaces; i++) {
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
int ratio;
|
2014-02-13 19:19:01 +00:00
|
|
|
u64 flags = sargs->spaces[i].flags;
|
|
|
|
|
2021-10-06 21:22:31 +00:00
|
|
|
ratio = btrfs_bg_type_to_ncopies(flags);
|
|
|
|
|
2014-02-13 19:19:01 +00:00
|
|
|
/*
|
2020-05-27 20:37:48 +00:00
|
|
|
* The RAID5/6 ratio depends on the number of stripes and is
|
|
|
|
* computed separately. Setting ratio to 0 will not account
|
|
|
|
* the chunks in this loop.
|
2014-02-13 19:19:01 +00:00
|
|
|
*/
|
2021-10-06 21:22:31 +00:00
|
|
|
if (flags & BTRFS_BLOCK_GROUP_RAID56_MASK)
|
2014-02-13 19:19:01 +00:00
|
|
|
ratio = 0;
|
|
|
|
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
if (ratio > max_data_ratio)
|
|
|
|
max_data_ratio = ratio;
|
|
|
|
|
2014-04-29 15:32:22 +00:00
|
|
|
if (flags & BTRFS_SPACE_INFO_GLOBAL_RSV) {
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
l_global_reserve = sargs->spaces[i].total_bytes;
|
|
|
|
l_global_reserve_used = sargs->spaces[i].used_bytes;
|
|
|
|
}
|
|
|
|
if ((flags & (BTRFS_BLOCK_GROUP_DATA | BTRFS_BLOCK_GROUP_METADATA))
|
2016-01-13 11:59:42 +00:00
|
|
|
== (BTRFS_BLOCK_GROUP_DATA | BTRFS_BLOCK_GROUP_METADATA)) {
|
2022-10-05 18:02:13 +00:00
|
|
|
mixed = true;
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* As mixed mode is not supported in zoned mode, this
|
|
|
|
* will account for all profile types
|
|
|
|
*/
|
btrfs-progs: fix all variable shadowing
There are quite some variable shadowing in btrfs-progs, most of them are
just reusing some common names like tmp.
And those are quite safe and the shadowed one are even different type.
But there are some exceptions:
- @end in traverse_tree_blocks()
There is already an @end with the same type, but a different meaning
(the end of the current extent buffer passed in).
Just rename it to @child_end.
- @start in generate_new_data_csums_range()
Just rename it to @csum_start.
- @size of fixup_chunk_tree_block()
This one is particularly bad, we declare a local @size and initialize
it to -1, then before we really utilize the variable @size, we
immediately reset it to 0, then pass it to logical_to_physical().
Then there is a location to check if @size is -1, which will always be
true.
According to the code in logical_to_physical(), @size would be clamped
down by its original value, thus our local @size will always be 0.
This patch would rename the local @size to @found_size, and only set
it to -1.
The call site is only to pass something as logical_to_physical()
requires a non-NULL pointer.
We don't really need to bother the returned value.
- duplicated @ref declaration in run_delayed_tree_ref()
- duplicated @super_flags in change_meta_csums()
Just delete the duplicated one.
Signed-off-by: Qu Wenruo <wqu@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
2023-09-11 10:40:33 +00:00
|
|
|
u64 unusable;
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
|
btrfs-progs: fix all variable shadowing
There are quite some variable shadowing in btrfs-progs, most of them are
just reusing some common names like tmp.
And those are quite safe and the shadowed one are even different type.
But there are some exceptions:
- @end in traverse_tree_blocks()
There is already an @end with the same type, but a different meaning
(the end of the current extent buffer passed in).
Just rename it to @child_end.
- @start in generate_new_data_csums_range()
Just rename it to @csum_start.
- @size of fixup_chunk_tree_block()
This one is particularly bad, we declare a local @size and initialize
it to -1, then before we really utilize the variable @size, we
immediately reset it to 0, then pass it to logical_to_physical().
Then there is a location to check if @size is -1, which will always be
true.
According to the code in logical_to_physical(), @size would be clamped
down by its original value, thus our local @size will always be 0.
This patch would rename the local @size to @found_size, and only set
it to -1.
The call site is only to pass something as logical_to_physical()
requires a non-NULL pointer.
We don't really need to bother the returned value.
- duplicated @ref declaration in run_delayed_tree_ref()
- duplicated @super_flags in change_meta_csums()
Just delete the duplicated one.
Signed-off-by: Qu Wenruo <wqu@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
2023-09-11 10:40:33 +00:00
|
|
|
unusable = device_get_zone_unusable(fd, flags);
|
|
|
|
if (unusable != DEVICE_ZONE_UNUSABLE_UNKNOWN)
|
|
|
|
zone_unusable += unusable;
|
2014-04-29 15:32:22 +00:00
|
|
|
}
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
if (flags & BTRFS_BLOCK_GROUP_DATA) {
|
|
|
|
r_data_used += sargs->spaces[i].used_bytes * ratio;
|
|
|
|
r_data_chunks += sargs->spaces[i].total_bytes * ratio;
|
|
|
|
l_data_chunks += sargs->spaces[i].total_bytes;
|
|
|
|
}
|
|
|
|
if (flags & BTRFS_BLOCK_GROUP_METADATA) {
|
|
|
|
r_metadata_used += sargs->spaces[i].used_bytes * ratio;
|
|
|
|
r_metadata_chunks += sargs->spaces[i].total_bytes * ratio;
|
|
|
|
l_metadata_chunks += sargs->spaces[i].total_bytes;
|
|
|
|
}
|
|
|
|
if (flags & BTRFS_BLOCK_GROUP_SYSTEM) {
|
|
|
|
r_system_used += sargs->spaces[i].used_bytes * ratio;
|
|
|
|
r_system_chunks += sargs->spaces[i].total_bytes * ratio;
|
|
|
|
}
|
|
|
|
}
|
2014-02-13 19:19:01 +00:00
|
|
|
|
2016-01-13 11:59:42 +00:00
|
|
|
r_total_chunks = r_data_chunks + r_system_chunks;
|
|
|
|
r_total_used = r_data_used + r_system_used;
|
|
|
|
if (!mixed) {
|
|
|
|
r_total_chunks += r_metadata_chunks;
|
|
|
|
r_total_used += r_metadata_used;
|
|
|
|
}
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
r_total_unused = r_total_size - r_total_chunks;
|
2014-02-13 19:19:01 +00:00
|
|
|
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
/* Raw / Logical = raid factor, >= 1 */
|
|
|
|
data_ratio = (double)r_data_chunks / l_data_chunks;
|
2016-01-13 11:59:42 +00:00
|
|
|
if (mixed)
|
|
|
|
metadata_ratio = data_ratio;
|
|
|
|
else
|
|
|
|
metadata_ratio = (double)r_metadata_chunks / l_metadata_chunks;
|
2014-02-13 19:19:01 +00:00
|
|
|
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
/*
|
|
|
|
* We're able to fill at least DATA for the unused space
|
|
|
|
*
|
|
|
|
* With mixed raid levels, this gives a rough estimate but more
|
|
|
|
* accurate than just counting the logical free space
|
|
|
|
* (l_data_chunks - l_data_used)
|
|
|
|
*
|
|
|
|
* In non-mixed case there's no difference.
|
|
|
|
*/
|
|
|
|
free_estimated = (r_data_chunks - r_data_used) / data_ratio;
|
2016-01-13 11:59:42 +00:00
|
|
|
/*
|
|
|
|
* For mixed-bg the metadata are left out in calculations thus global
|
|
|
|
* reserve would be lost. Part of it could be permanently allocated,
|
|
|
|
* we have to subtract the used bytes so we don't go under zero free.
|
|
|
|
*/
|
|
|
|
if (mixed)
|
|
|
|
free_estimated -= l_global_reserve - l_global_reserve_used;
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
free_min = free_estimated;
|
|
|
|
|
|
|
|
/* Chop unallocatable space */
|
|
|
|
/* FIXME: must be applied per device */
|
|
|
|
if (r_total_unused >= MIN_UNALOCATED_THRESH) {
|
|
|
|
free_estimated += r_total_unused / data_ratio;
|
|
|
|
/* Match the calculation of 'df', use the highest raid ratio */
|
|
|
|
free_min += r_total_unused / max_data_ratio;
|
|
|
|
}
|
2014-02-13 19:19:01 +00:00
|
|
|
|
2014-12-18 13:52:36 +00:00
|
|
|
if (unit_mode != UNITS_HUMAN)
|
2014-02-13 19:19:01 +00:00
|
|
|
width = 18;
|
|
|
|
|
2020-11-10 00:52:21 +00:00
|
|
|
ret = statfs(path, &statfs_buf);
|
|
|
|
if (ret) {
|
|
|
|
warning("cannot get space info with statfs() on '%s': %m", path);
|
|
|
|
memset(&statfs_buf, 0, sizeof(statfs_buf));
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "Overall:\n");
|
2014-04-25 15:24:40 +00:00
|
|
|
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device size:\t\t%*s\n", width,
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(r_total_size, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device allocated:\t\t%*s\n", width,
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(r_total_chunks, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device unallocated:\t\t%*s\n", width,
|
2017-01-03 17:26:43 +00:00
|
|
|
pretty_size_mode(r_total_unused, unit_mode | UNITS_NEGATIVE));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device missing:\t\t%*s\n", width,
|
2015-03-17 13:41:01 +00:00
|
|
|
pretty_size_mode(r_total_missing, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device slack:\t\t%*s\n", width,
|
2022-09-14 19:20:43 +00:00
|
|
|
pretty_size_mode(r_total_slack, unit_mode));
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
ret = ioctl(fd, BTRFS_IOC_GET_FEATURES, &feature_flags);
|
|
|
|
if (ret == 0 && (feature_flags.incompat_flags & BTRFS_FEATURE_INCOMPAT_ZONED)) {
|
2021-06-24 21:01:09 +00:00
|
|
|
u64 zone_size;
|
|
|
|
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device zone unusable:\t%*s\n", width,
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
pretty_size_mode(zone_unusable, unit_mode));
|
2021-06-24 21:01:09 +00:00
|
|
|
zone_size = get_first_device_zone_size(fd);
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device zone size:\t\t%*s\n", width,
|
2021-06-24 21:01:09 +00:00
|
|
|
pretty_size_mode(zone_size, unit_mode));
|
btrfs-progs: fi usage: print zone unusable in the overview
Print the total zone_unusable size in the summary for 'fi usage' for a
filesystem in zoned mode. It's a sum of all the zone_unusable values
from 'fi df'. Per-device stats are not implemented and would need more
complicated calculations from raw data, kernel does not export that (but
it could).
As of 5.12, the zone_unusable is stored only in memory so we'd have to
map raw block device zones to the block groups and the live extents in
the associated block groups to get the exact numbers.
Example:
# btrfs fi usage /mnt
Overall:
Device size: 2.00GiB
Device allocated: 768.00MiB
Device unallocated: 1.25GiB
Device missing: 0.00B
Device zone unusable: 320.00KiB
Used: 128.00KiB
Free (estimated): 1.50GiB (min: 1.50GiB)
Free (statfs, df): 1.50GiB
Data ratio: 1.00
Metadata ratio: 1.00
Global reserve: 3.25MiB (used: 32.00KiB)
Multiple profiles: no
Data,single: Size:256.00MiB, Used:0.00B (0.00%)
/dev/nullb0 256.00MiB
Metadata,single: Size:256.00MiB, Used:112.00KiB (0.04%)
/dev/nullb0 256.00MiB
System,single: Size:256.00MiB, Used:16.00KiB (0.01%)
/dev/nullb0 256.00MiB
Unallocated:
/dev/nullb0 1.25GiB
# btrfs fi df
Data, single: total=256.00MiB, used=0.00B, zone_unusable=0.00B
System, single: total=256.00MiB, used=16.00KiB, zone_unusable=160.00KiB
Metadata, single: total=256.00MiB, used=112.00KiB, zone_unusable=160.00KiB
GlobalReserve, single: total=3.25MiB, used=32.00KiB
Signed-off-by: David Sterba <dsterba@suse.com>
2021-05-06 23:13:14 +00:00
|
|
|
}
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Used:\t\t\t%*s\n", width,
|
2021-06-24 21:04:59 +00:00
|
|
|
pretty_size_mode(r_total_used, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Free (estimated):\t\t%*s\t(",
|
2014-02-13 19:19:01 +00:00
|
|
|
width,
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(free_estimated, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "min: %s)\n", pretty_size_mode(free_min, unit_mode));
|
|
|
|
pr_verbose(LOG_DEFAULT, " Free (statfs, df):\t\t%*s\n", width,
|
2020-11-10 00:52:21 +00:00
|
|
|
pretty_size_mode(statfs_buf.f_bavail * statfs_buf.f_bsize, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Data ratio:\t\t\t%*.2f\n",
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
width, data_ratio);
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Metadata ratio:\t\t%*.2f\n",
|
btrfs-progs: rework calculations of fi usage
This patch reworks the basic calculations of 'fi usage'. It does not address
all problems but should make the code more prepared to do so.
The original code tries to estimate the free space that could lead to negative
numbers for some raid profiles:
Data, RAID1: total=147.00GiB, used=141.92GiB
System, RAID1: total=32.00MiB, used=36.00KiB
Metadata, RAID1: total=2.00GiB, used=1.17GiB
GlobalReserve, single: total=404.00MiB, used=0.00B
Overall:
Device size: 279.46GiB
Device allocated: 298.06GiB
Device unallocated: 16.00EiB
Used: 286.18GiB
Free (estimated): 8.00EiB (min: 8.00EiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Eg. "Device size" - "Device allocated" = negative number or a very large
positive, hence the EiB values.
There are logical and raw numbers multiplied by ratios mixed together,
so the new code makes it explicit which kind is being used. The data and
metadata ratios are calculated separately.
Output after this patch will look like:
Overall:
Device size: 558.92GiB
Device allocated: 298.06GiB
Device unallocated: 260.86GiB
Used: 286.18GiB
Free (estimated): 135.51GiB (min: 135.51GiB)
Data ratio: 2.00
Metadata ratio: 2.00
Global reserve: 404.00MiB (used: 0.00B)
Data,RAID1: Size:147.00GiB, Used:141.92GiB
/dev/sdc 147.00GiB
/dev/sdd 147.00GiB
Metadata,RAID1: Size:2.00GiB, Used:1.17GiB
/dev/sdc 2.00GiB
/dev/sdd 2.00GiB
System,RAID1: Size:32.00MiB, Used:36.00KiB
/dev/sdc 32.00MiB
/dev/sdd 32.00MiB
Unallocated:
/dev/sdc 130.43GiB
/dev/sdd 130.43GiB
Changes:
* Device size is now the raw size, same for the following three
* Free is the logical size
* Max/min were reduced to just min
Filesystem Size Used Avail Use% Mounted on
/dev/sdc 280G 144G 141G 51% /mnt/sdc
The difference between Avail and Free is there because userspace tool does a
different guesswork than kernel.
Issues not addressed by this patch:
* RAID56 profiles are not handled
* mixed profiles are not handled
Signed-off-by: David Sterba <dsterba@suse.cz>
2014-06-26 13:42:24 +00:00
|
|
|
width, metadata_ratio);
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Global reserve:\t\t%*s\t(used: %s)\n", width,
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(l_global_reserve, unit_mode),
|
|
|
|
pretty_size_mode(l_global_reserve_used, unit_mode));
|
2020-05-04 20:03:09 +00:00
|
|
|
tmp = btrfs_test_for_multiple_profiles(fd);
|
|
|
|
if (tmp[0])
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Multiple profiles:\t\t%*s\t(%s)\n", width, "yes", tmp);
|
2020-04-04 10:32:10 +00:00
|
|
|
else
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Multiple profiles:\t\t%*s\n", width, "no");
|
2020-05-04 20:03:09 +00:00
|
|
|
free(tmp);
|
2014-02-13 19:19:01 +00:00
|
|
|
|
|
|
|
exit:
|
|
|
|
|
|
|
|
if (sargs)
|
|
|
|
free(sargs);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
/*
|
2014-04-23 17:00:22 +00:00
|
|
|
* Helper to sort the device_info structure
|
2014-02-13 19:19:50 +00:00
|
|
|
*/
|
2014-04-23 17:00:22 +00:00
|
|
|
static int cmp_device_info(const void *a, const void *b)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
2019-11-06 14:58:07 +00:00
|
|
|
const struct device_info *deva = a;
|
|
|
|
const struct device_info *devb = b;
|
|
|
|
|
|
|
|
if (deva->devid < devb->devid)
|
|
|
|
return -1;
|
|
|
|
if (deva->devid > devb->devid)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
2017-10-26 18:44:55 +00:00
|
|
|
int dev_to_fsid(const char *dev, u8 *fsid)
|
2017-10-23 04:44:01 +00:00
|
|
|
{
|
2021-10-21 01:40:19 +00:00
|
|
|
struct btrfs_super_block disk_super;
|
2017-10-23 04:44:01 +00:00
|
|
|
int ret;
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
fd = open(dev, O_RDONLY);
|
|
|
|
if (fd < 0) {
|
|
|
|
ret = -errno;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-10-21 01:40:19 +00:00
|
|
|
ret = btrfs_read_dev_super(fd, &disk_super,
|
2017-10-23 04:44:01 +00:00
|
|
|
BTRFS_SUPER_INFO_OFFSET, SBREAD_DEFAULT);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
2021-10-21 01:40:19 +00:00
|
|
|
memcpy(fsid, disk_super.fsid, BTRFS_FSID_SIZE);
|
2017-10-23 04:44:01 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
close(fd);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-02-13 09:37:42 +00:00
|
|
|
static int device_is_seed(int fd, const char *dev_path, u64 devid, const u8 *mnt_fsid)
|
2023-02-13 09:37:41 +00:00
|
|
|
{
|
2023-02-13 09:37:42 +00:00
|
|
|
char fsid_str[BTRFS_UUID_UNPARSED_SIZE];
|
|
|
|
char fsid_path[PATH_MAX];
|
|
|
|
char devid_str[20];
|
2023-02-13 09:37:41 +00:00
|
|
|
u8 fsid[BTRFS_UUID_SIZE];
|
2023-02-13 09:37:42 +00:00
|
|
|
int ret = -1;
|
|
|
|
int sysfs_fd;
|
2023-02-13 09:37:41 +00:00
|
|
|
|
2023-02-13 09:37:42 +00:00
|
|
|
snprintf(devid_str, 20, "%llu", devid);
|
|
|
|
/* devinfo/<devid>/fsid */
|
|
|
|
ret = path_cat3_out(fsid_path, "devinfo", devid_str, "fsid");
|
|
|
|
if (ret < 0)
|
2023-02-13 09:37:41 +00:00
|
|
|
return ret;
|
|
|
|
|
2023-02-13 09:37:42 +00:00
|
|
|
/* /sys/fs/btrfs/<fsid>/devinfo/<devid>/fsid */
|
|
|
|
sysfs_fd = sysfs_open_fsid_file(fd, fsid_path);
|
|
|
|
if (sysfs_fd >= 0) {
|
|
|
|
sysfs_read_file(sysfs_fd, fsid_str, BTRFS_UUID_UNPARSED_SIZE);
|
|
|
|
fsid_str[BTRFS_UUID_UNPARSED_SIZE - 1] = 0;
|
|
|
|
ret = uuid_parse(fsid_str, fsid);
|
|
|
|
close(sysfs_fd);
|
|
|
|
}
|
|
|
|
|
2023-03-15 20:03:41 +00:00
|
|
|
if (ret || sysfs_fd < 0) {
|
2023-02-13 09:37:42 +00:00
|
|
|
ret = dev_to_fsid(dev_path, fsid);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-02-13 09:37:41 +00:00
|
|
|
if (memcmp(mnt_fsid, fsid, BTRFS_FSID_SIZE) != 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
/*
|
2014-04-23 17:00:22 +00:00
|
|
|
* This function loads the device_info structure and put them in an array
|
2014-02-13 19:19:50 +00:00
|
|
|
*/
|
2022-09-23 08:07:29 +00:00
|
|
|
static int load_device_info(int fd, struct device_info **devinfo_ret,
|
|
|
|
int *devcount_ret)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
2015-10-02 15:44:21 +00:00
|
|
|
int ret, i, ndevs;
|
2014-02-13 19:19:50 +00:00
|
|
|
struct btrfs_ioctl_fs_info_args fi_args;
|
|
|
|
struct btrfs_ioctl_dev_info_args dev_info;
|
2014-04-23 17:00:22 +00:00
|
|
|
struct device_info *info;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
*devcount_ret = 0;
|
|
|
|
*devinfo_ret = NULL;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
ret = ioctl(fd, BTRFS_IOC_FS_INFO, &fi_args);
|
|
|
|
if (ret < 0) {
|
2015-10-02 15:44:21 +00:00
|
|
|
if (errno == EPERM)
|
|
|
|
return -errno;
|
2018-01-07 21:54:21 +00:00
|
|
|
error("cannot get filesystem info: %m");
|
2014-07-24 03:21:54 +00:00
|
|
|
return 1;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
2014-04-24 16:37:50 +00:00
|
|
|
info = calloc(fi_args.num_devices, sizeof(struct device_info));
|
2014-02-13 19:19:50 +00:00
|
|
|
if (!info) {
|
2022-09-30 07:12:06 +00:00
|
|
|
error_msg(ERROR_MSG_MEMORY, NULL);
|
2014-07-24 03:21:54 +00:00
|
|
|
return 1;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0, ndevs = 0 ; i <= fi_args.max_id ; i++) {
|
2016-08-26 10:53:31 +00:00
|
|
|
if (ndevs >= fi_args.num_devices) {
|
|
|
|
error("unexpected number of devices: %d >= %llu", ndevs,
|
2022-09-27 23:13:00 +00:00
|
|
|
fi_args.num_devices);
|
2017-10-23 04:45:48 +00:00
|
|
|
error(
|
|
|
|
"if seed device is used, try running this command as root");
|
2016-08-26 10:53:31 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2014-04-24 16:37:50 +00:00
|
|
|
memset(&dev_info, 0, sizeof(dev_info));
|
2023-08-22 19:07:13 +00:00
|
|
|
ret = device_get_info(fd, i, &dev_info);
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
if (ret == -ENODEV)
|
|
|
|
continue;
|
|
|
|
if (ret) {
|
2016-01-11 10:01:47 +00:00
|
|
|
error("cannot get info about device devid=%d", i);
|
2016-08-26 10:53:31 +00:00
|
|
|
goto out;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
2017-10-23 04:45:48 +00:00
|
|
|
/*
|
2023-02-13 09:37:42 +00:00
|
|
|
* Skip seed device by checking device's fsid (requires root if
|
|
|
|
* kernel is not patched to provide fsid from the sysfs).
|
|
|
|
* And we will skip only if device_is_seed is successful and dev
|
2017-10-23 04:45:48 +00:00
|
|
|
* is a seed device.
|
|
|
|
* Ignore any other error including -EACCES, which is seen when
|
|
|
|
* a non-root process calls dev_to_fsid(path)->open(path).
|
|
|
|
*/
|
2023-02-13 09:37:42 +00:00
|
|
|
ret = device_is_seed(fd, (const char *)dev_info.path, i, fi_args.fsid);
|
2023-02-13 09:37:41 +00:00
|
|
|
if (!ret)
|
2017-10-23 04:45:48 +00:00
|
|
|
continue;
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
info[ndevs].devid = dev_info.devid;
|
2015-03-17 13:41:01 +00:00
|
|
|
if (!dev_info.path[0]) {
|
|
|
|
strcpy(info[ndevs].path, "missing");
|
|
|
|
} else {
|
|
|
|
strcpy(info[ndevs].path, (char *)dev_info.path);
|
|
|
|
info[ndevs].device_size =
|
2021-04-29 16:36:44 +00:00
|
|
|
device_get_partition_size((const char *)dev_info.path);
|
2015-03-17 13:41:01 +00:00
|
|
|
}
|
2014-04-24 16:32:27 +00:00
|
|
|
info[ndevs].size = dev_info.total_bytes;
|
2014-02-13 19:19:50 +00:00
|
|
|
++ndevs;
|
|
|
|
}
|
|
|
|
|
2016-08-26 10:53:31 +00:00
|
|
|
if (ndevs != fi_args.num_devices) {
|
|
|
|
error("unexpected number of devices: %d != %llu", ndevs,
|
2022-09-27 23:13:00 +00:00
|
|
|
fi_args.num_devices);
|
2016-08-26 10:53:31 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
qsort(info, fi_args.num_devices,
|
2014-04-23 17:00:22 +00:00
|
|
|
sizeof(struct device_info), cmp_device_info);
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
*devcount_ret = fi_args.num_devices;
|
|
|
|
*devinfo_ret = info;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
return 0;
|
2016-08-26 10:53:31 +00:00
|
|
|
|
|
|
|
out:
|
|
|
|
free(info);
|
|
|
|
return ret;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
int load_chunk_and_device_info(int fd, struct chunk_info **chunkinfo_ret,
|
|
|
|
int *chunkcount_ret, struct device_info **devinfo_ret,
|
|
|
|
int *devcount_ret)
|
2014-04-25 17:39:11 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
ret = load_chunk_info(fd, chunkinfo_ret, chunkcount_ret);
|
2014-04-25 17:39:11 +00:00
|
|
|
if (ret == -EPERM) {
|
2016-01-11 10:01:47 +00:00
|
|
|
warning(
|
2018-03-29 08:22:43 +00:00
|
|
|
"cannot read detailed chunk info, per-device usage will not be shown, run as root");
|
2014-04-25 17:39:11 +00:00
|
|
|
} else if (ret) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
ret = load_device_info(fd, devinfo_ret, devcount_ret);
|
2014-04-25 17:39:11 +00:00
|
|
|
if (ret == -EPERM) {
|
2016-01-11 10:01:47 +00:00
|
|
|
warning(
|
|
|
|
"cannot get filesystem info from ioctl(FS_INFO), run as root");
|
2014-04-25 17:39:11 +00:00
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
/*
|
|
|
|
* This function computes the size of a chunk in a disk
|
|
|
|
*/
|
|
|
|
static u64 calc_chunk_size(struct chunk_info *ci)
|
|
|
|
{
|
btrfs-progs: fi usage: fix calculation of chunk size for RAID1/DUP profiles
Current formula calculates the stripe size, however that's not what we
want in the case of RAID1/DUP profiles. In those cases since chunk are
mirrored across devices we want the full size of the chunk. Without this
patch the 'btrfs fi usage' output from an fs which is using RAID1 is:
Data,RAID1: Size:2.00GiB, Used:1.00GiB (50.03%)
/dev/vdc 1.00GiB
/dev/vdf 1.00GiB
Metadata,RAID1: Size:256.00MiB, Used:1.34MiB (0.52%)
/dev/vdc 128.00MiB
/dev/vdf 128.00MiB
System,RAID1: Size:8.00MiB, Used:16.00KiB (0.20%)
/dev/vdc 4.00MiB
/dev/vdf 4.00MiB
Unallocated:
/dev/vdc 8.87GiB
/dev/vdf 8.87GiB
So a 2 gigabyte RAID1 chunk actually will take up 4 gigabytes on the
actual disks 2 each. In this case this is being miscalculated as taking
up 1GiB on each device.
This also leads to erroneously calculated unallocated space. The correct
output in this case is:
Data,RAID1: Size:2.00GiB, Used:1.00GiB (50.03%)
/dev/vdc 2.00GiB
/dev/vdf 2.00GiB
Metadata,RAID1: Size:256.00MiB, Used:1.34MiB (0.52%)
/dev/vdc 256.00MiB
/dev/vdf 256.00MiB
System,RAID1: Size:8.00MiB, Used:16.00KiB (0.20%)
/dev/vdc 8.00MiB
/dev/vdf 8.00MiB
Unallocated:
/dev/vdc 7.74GiB
/dev/vdf 7.74GiB
Fix it by only utilising the chunk formula for profiles which are not
RAID1/DUP.
Issue: #422
Signed-off-by: Nikolay Borisov <nborisov@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
2021-11-16 14:02:06 +00:00
|
|
|
u32 div = 1;
|
2021-10-12 15:29:06 +00:00
|
|
|
|
btrfs-progs: fi usage: fix calculation of chunk size for RAID1/DUP profiles
Current formula calculates the stripe size, however that's not what we
want in the case of RAID1/DUP profiles. In those cases since chunk are
mirrored across devices we want the full size of the chunk. Without this
patch the 'btrfs fi usage' output from an fs which is using RAID1 is:
Data,RAID1: Size:2.00GiB, Used:1.00GiB (50.03%)
/dev/vdc 1.00GiB
/dev/vdf 1.00GiB
Metadata,RAID1: Size:256.00MiB, Used:1.34MiB (0.52%)
/dev/vdc 128.00MiB
/dev/vdf 128.00MiB
System,RAID1: Size:8.00MiB, Used:16.00KiB (0.20%)
/dev/vdc 4.00MiB
/dev/vdf 4.00MiB
Unallocated:
/dev/vdc 8.87GiB
/dev/vdf 8.87GiB
So a 2 gigabyte RAID1 chunk actually will take up 4 gigabytes on the
actual disks 2 each. In this case this is being miscalculated as taking
up 1GiB on each device.
This also leads to erroneously calculated unallocated space. The correct
output in this case is:
Data,RAID1: Size:2.00GiB, Used:1.00GiB (50.03%)
/dev/vdc 2.00GiB
/dev/vdf 2.00GiB
Metadata,RAID1: Size:256.00MiB, Used:1.34MiB (0.52%)
/dev/vdc 256.00MiB
/dev/vdf 256.00MiB
System,RAID1: Size:8.00MiB, Used:16.00KiB (0.20%)
/dev/vdc 8.00MiB
/dev/vdf 8.00MiB
Unallocated:
/dev/vdc 7.74GiB
/dev/vdf 7.74GiB
Fix it by only utilising the chunk formula for profiles which are not
RAID1/DUP.
Issue: #422
Signed-off-by: Nikolay Borisov <nborisov@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
2021-11-16 14:02:06 +00:00
|
|
|
/*
|
|
|
|
* The formula doesn't work for RAID1/DUP types, we should just return the
|
|
|
|
* chunk size
|
|
|
|
*/
|
|
|
|
if (!(ci->type & (BTRFS_BLOCK_GROUP_RAID1_MASK|BTRFS_BLOCK_GROUP_DUP))) {
|
|
|
|
/* No parity + sub_stripes, so order of "-" and "/" does not matter */
|
|
|
|
div = (ci->num_stripes - btrfs_bg_type_to_nparity(ci->type)) /
|
|
|
|
btrfs_bg_type_to_sub_stripes(ci->type);
|
|
|
|
}
|
2021-10-12 15:29:06 +00:00
|
|
|
|
|
|
|
return ci->size / div;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2014-04-23 16:47:52 +00:00
|
|
|
* This function print the results of the command "btrfs fi usage"
|
2014-02-13 19:19:50 +00:00
|
|
|
* in tabular format
|
|
|
|
*/
|
2014-12-18 13:52:36 +00:00
|
|
|
static void _cmd_filesystem_usage_tabular(unsigned unit_mode,
|
2014-02-13 19:19:50 +00:00
|
|
|
struct btrfs_ioctl_space_args *sargs,
|
|
|
|
struct chunk_info *chunks_info_ptr,
|
|
|
|
int chunks_info_count,
|
2022-09-23 08:07:29 +00:00
|
|
|
struct device_info *devinfo,
|
|
|
|
int devcount)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
u64 total_unused = 0;
|
2022-09-14 19:20:43 +00:00
|
|
|
u64 total_total = 0;
|
|
|
|
u64 total_slack = 0;
|
2016-01-04 01:01:30 +00:00
|
|
|
struct string_table *matrix = NULL;
|
2014-02-13 19:19:50 +00:00
|
|
|
int ncols, nrows;
|
2015-11-02 17:24:28 +00:00
|
|
|
int col;
|
|
|
|
int unallocated_col;
|
2015-11-02 23:08:32 +00:00
|
|
|
int spaceinfos_col;
|
2022-09-14 19:20:43 +00:00
|
|
|
int total_col;
|
|
|
|
int slack_col;
|
|
|
|
u64 slack;
|
2015-11-02 17:54:48 +00:00
|
|
|
const int vhdr_skip = 3; /* amount of vertical header space */
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2022-09-14 19:20:43 +00:00
|
|
|
/* id, path, unallocated, total, slack */
|
|
|
|
ncols = 5;
|
2015-11-02 23:19:41 +00:00
|
|
|
spaceinfos_col = 2;
|
2015-11-02 17:44:45 +00:00
|
|
|
/* Properly count the real space infos */
|
|
|
|
for (i = 0; i < sargs->total_spaces; i++) {
|
|
|
|
if (sargs->spaces[i].flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
ncols++;
|
|
|
|
}
|
|
|
|
|
2015-11-02 17:24:28 +00:00
|
|
|
/* 2 for header, empty line, devices, ===, total, used */
|
2022-09-23 08:07:29 +00:00
|
|
|
nrows = vhdr_skip + devcount + 1 + 2;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
matrix = table_create(ncols, nrows);
|
|
|
|
if (!matrix) {
|
2022-09-30 07:12:06 +00:00
|
|
|
error_msg(ERROR_MSG_MEMORY, NULL);
|
2014-02-13 19:19:50 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-11-02 17:24:28 +00:00
|
|
|
/*
|
|
|
|
* We have to skip the global block reserve everywhere as it's an
|
|
|
|
* artificial blockgroup
|
|
|
|
*/
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
/* header */
|
2015-11-02 23:08:32 +00:00
|
|
|
for (i = 0, col = spaceinfos_col; i < sargs->total_spaces; i++) {
|
2014-02-13 19:19:50 +00:00
|
|
|
u64 flags = sargs->spaces[i].flags;
|
2014-04-29 15:32:22 +00:00
|
|
|
|
|
|
|
if (flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
|
2015-11-02 17:32:04 +00:00
|
|
|
table_printf(matrix, col, 0, "<%s",
|
|
|
|
btrfs_group_type_str(flags));
|
|
|
|
table_printf(matrix, col, 1, "<%s",
|
|
|
|
btrfs_group_profile_str(flags));
|
|
|
|
col++;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
2022-09-14 19:20:43 +00:00
|
|
|
unallocated_col = col++;
|
|
|
|
total_col = col++;
|
|
|
|
slack_col = col++;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2015-11-02 23:19:41 +00:00
|
|
|
table_printf(matrix, 0, 1, "<Id");
|
|
|
|
table_printf(matrix, 1, 1, "<Path");
|
2015-11-02 17:24:28 +00:00
|
|
|
table_printf(matrix, unallocated_col, 1, "<Unallocated");
|
2022-09-14 19:20:43 +00:00
|
|
|
table_printf(matrix, total_col, 1, "<Total");
|
|
|
|
table_printf(matrix, slack_col, 1, "<Slack");
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
/* body */
|
2022-09-23 08:07:29 +00:00
|
|
|
for (i = 0; i < devcount; i++) {
|
2015-11-02 17:24:28 +00:00
|
|
|
int k;
|
2014-02-13 19:19:50 +00:00
|
|
|
char *p;
|
|
|
|
|
|
|
|
u64 total_allocated = 0, unused;
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
p = strrchr(devinfo[i].path, '/');
|
2014-02-13 19:19:50 +00:00
|
|
|
if (!p)
|
2022-09-23 08:07:29 +00:00
|
|
|
p = devinfo[i].path;
|
2014-02-13 19:19:50 +00:00
|
|
|
else
|
|
|
|
p++;
|
|
|
|
|
2015-11-02 23:19:41 +00:00
|
|
|
table_printf(matrix, 0, vhdr_skip + i, ">%llu",
|
2022-09-23 08:07:29 +00:00
|
|
|
devinfo[i].devid);
|
2015-11-02 23:19:41 +00:00
|
|
|
table_printf(matrix, 1, vhdr_skip + i, "<%s",
|
2022-09-23 08:07:29 +00:00
|
|
|
devinfo[i].path);
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2015-11-02 23:08:32 +00:00
|
|
|
for (col = spaceinfos_col, k = 0; k < sargs->total_spaces; k++) {
|
2014-02-13 19:19:50 +00:00
|
|
|
u64 flags = sargs->spaces[k].flags;
|
2022-09-23 08:07:29 +00:00
|
|
|
u64 devid = devinfo[i].devid;
|
2014-02-13 19:19:50 +00:00
|
|
|
int j;
|
|
|
|
u64 size = 0;
|
|
|
|
|
2015-11-02 17:24:28 +00:00
|
|
|
if (flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
for (j = 0 ; j < chunks_info_count ; j++) {
|
|
|
|
if (chunks_info_ptr[j].type != flags )
|
|
|
|
continue;
|
|
|
|
if (chunks_info_ptr[j].devid != devid)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
size += calc_chunk_size(chunks_info_ptr+j);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size)
|
2015-11-02 17:54:48 +00:00
|
|
|
table_printf(matrix, col, vhdr_skip+ i,
|
2014-12-18 13:52:36 +00:00
|
|
|
">%s", pretty_size_mode(size, unit_mode));
|
2014-02-13 19:19:50 +00:00
|
|
|
else
|
2015-11-02 17:54:48 +00:00
|
|
|
table_printf(matrix, col, vhdr_skip + i, ">-");
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
total_allocated += size;
|
|
|
|
col++;
|
|
|
|
}
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
unused = device_get_partition_size(devinfo[i].path)
|
2014-04-23 17:00:22 +00:00
|
|
|
- total_allocated;
|
2022-09-23 08:07:29 +00:00
|
|
|
unused = devinfo[i].size - total_allocated;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2017-01-03 17:26:43 +00:00
|
|
|
table_printf(matrix, unallocated_col, vhdr_skip + i, ">%s",
|
|
|
|
pretty_size_mode(unused, unit_mode | UNITS_NEGATIVE));
|
2022-09-14 19:20:43 +00:00
|
|
|
table_printf(matrix, total_col, vhdr_skip + i, ">%s",
|
2022-09-23 08:07:29 +00:00
|
|
|
pretty_size_mode(devinfo[i].size,
|
2022-09-14 19:20:43 +00:00
|
|
|
unit_mode | UNITS_NEGATIVE));
|
2022-09-23 08:07:29 +00:00
|
|
|
slack = calc_slack_size(&devinfo[i]);
|
2022-09-14 19:20:43 +00:00
|
|
|
if (slack > 0) {
|
|
|
|
table_printf(matrix, slack_col, vhdr_skip + i, ">%s",
|
|
|
|
pretty_size_mode(slack,
|
|
|
|
unit_mode | UNITS_NEGATIVE));
|
|
|
|
} else {
|
|
|
|
table_printf(matrix, slack_col, vhdr_skip + i, ">-");
|
|
|
|
}
|
2014-02-13 19:19:50 +00:00
|
|
|
total_unused += unused;
|
2022-09-14 19:20:43 +00:00
|
|
|
total_slack += slack;
|
2022-09-23 08:07:29 +00:00
|
|
|
total_total += devinfo[i].size;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-11-03 00:09:50 +00:00
|
|
|
for (i = 0; i < spaceinfos_col; i++) {
|
|
|
|
table_printf(matrix, i, vhdr_skip - 1, "*-");
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, i, vhdr_skip + devcount, "*-");
|
2015-11-03 00:09:50 +00:00
|
|
|
}
|
|
|
|
|
2015-11-02 23:08:32 +00:00
|
|
|
for (i = 0, col = spaceinfos_col; i < sargs->total_spaces; i++) {
|
2015-11-02 17:24:28 +00:00
|
|
|
if (sargs->spaces[i].flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
|
2015-11-03 00:09:50 +00:00
|
|
|
table_printf(matrix, col, vhdr_skip - 1, "*-");
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, col, vhdr_skip + devcount, "*-");
|
2015-11-03 00:09:50 +00:00
|
|
|
col++;
|
2015-11-02 17:24:28 +00:00
|
|
|
}
|
2022-09-14 19:20:43 +00:00
|
|
|
/* Line under Unallocated, Total, Slack */
|
2015-11-03 00:09:50 +00:00
|
|
|
table_printf(matrix, col, vhdr_skip - 1, "*-");
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, col, vhdr_skip + devcount, "*-");
|
2022-09-14 19:20:43 +00:00
|
|
|
table_printf(matrix, col + 1, vhdr_skip - 1, "*-");
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, col + 1, vhdr_skip + devcount, "*-");
|
2022-09-14 19:20:43 +00:00
|
|
|
table_printf(matrix, col + 2, vhdr_skip - 1, "*-");
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, col + 2, vhdr_skip + devcount, "*-");
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
/* footer */
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, 1, vhdr_skip + devcount + 1, "<Total");
|
2015-11-02 23:08:32 +00:00
|
|
|
for (i = 0, col = spaceinfos_col; i < sargs->total_spaces; i++) {
|
2015-11-02 17:24:28 +00:00
|
|
|
if (sargs->spaces[i].flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, col++, vhdr_skip + devcount + 1,
|
2015-11-02 17:54:48 +00:00
|
|
|
">%s",
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(sargs->spaces[i].total_bytes, unit_mode));
|
2015-11-02 17:24:28 +00:00
|
|
|
}
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, unallocated_col, vhdr_skip + devcount + 1,
|
2017-01-03 17:26:43 +00:00
|
|
|
">%s",
|
|
|
|
pretty_size_mode(total_unused, unit_mode | UNITS_NEGATIVE));
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, total_col, vhdr_skip + devcount + 1,
|
2022-09-14 19:20:43 +00:00
|
|
|
">%s",
|
|
|
|
pretty_size_mode(total_total, unit_mode | UNITS_NEGATIVE));
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, slack_col, vhdr_skip + devcount + 1,
|
2022-09-14 19:20:43 +00:00
|
|
|
">%s",
|
|
|
|
pretty_size_mode(total_slack, unit_mode | UNITS_NEGATIVE));
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, 1, vhdr_skip + devcount + 2, "<Used");
|
2015-11-02 23:08:32 +00:00
|
|
|
for (i = 0, col = spaceinfos_col; i < sargs->total_spaces; i++) {
|
2015-11-02 17:24:28 +00:00
|
|
|
if (sargs->spaces[i].flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
table_printf(matrix, col++, vhdr_skip + devcount + 2,
|
2015-11-02 17:54:48 +00:00
|
|
|
">%s",
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(sargs->spaces[i].used_bytes, unit_mode));
|
2015-11-02 17:24:28 +00:00
|
|
|
}
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
table_dump(matrix);
|
|
|
|
table_free(matrix);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function prints the unused space per every disk
|
|
|
|
*/
|
|
|
|
static void print_unused(struct chunk_info *info_ptr,
|
|
|
|
int info_count,
|
2022-09-23 08:07:29 +00:00
|
|
|
struct device_info *devinfo,
|
|
|
|
int devcount,
|
2014-12-18 13:52:36 +00:00
|
|
|
unsigned unit_mode)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
|
|
|
int i;
|
2022-09-23 08:07:29 +00:00
|
|
|
for (i = 0; i < devcount; i++) {
|
2014-02-13 19:19:50 +00:00
|
|
|
int j;
|
|
|
|
u64 total = 0;
|
|
|
|
|
2014-04-23 17:00:22 +00:00
|
|
|
for (j = 0; j < info_count; j++)
|
2022-09-23 08:07:29 +00:00
|
|
|
if (info_ptr[j].devid == devinfo[i].devid)
|
2014-02-13 19:19:50 +00:00
|
|
|
total += calc_chunk_size(info_ptr+j);
|
|
|
|
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " %s\t%10s\n",
|
2022-09-23 08:07:29 +00:00
|
|
|
devinfo[i].path,
|
|
|
|
pretty_size_mode(devinfo[i].size - total, unit_mode));
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function prints the allocated chunk per every disk
|
|
|
|
*/
|
2014-04-23 17:00:22 +00:00
|
|
|
static void print_chunk_device(u64 chunk_type,
|
2014-02-13 19:19:50 +00:00
|
|
|
struct chunk_info *chunks_info_ptr,
|
|
|
|
int chunks_info_count,
|
2022-09-23 08:07:29 +00:00
|
|
|
struct device_info *devinfo,
|
|
|
|
int devcount,
|
2014-12-18 13:52:36 +00:00
|
|
|
unsigned unit_mode)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2022-09-23 08:07:29 +00:00
|
|
|
for (i = 0; i < devcount; i++) {
|
2014-02-13 19:19:50 +00:00
|
|
|
int j;
|
|
|
|
u64 total = 0;
|
|
|
|
|
2014-04-23 17:00:22 +00:00
|
|
|
for (j = 0; j < chunks_info_count; j++) {
|
2014-02-13 19:19:50 +00:00
|
|
|
|
|
|
|
if (chunks_info_ptr[j].type != chunk_type)
|
|
|
|
continue;
|
2022-09-23 08:07:29 +00:00
|
|
|
if (chunks_info_ptr[j].devid != devinfo[i].devid)
|
2014-02-13 19:19:50 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
total += calc_chunk_size(&(chunks_info_ptr[j]));
|
|
|
|
//total += chunks_info_ptr[j].size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (total > 0)
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " %s\t%10s\n",
|
2022-09-23 08:07:29 +00:00
|
|
|
devinfo[i].path,
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(total, unit_mode));
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2014-04-23 16:47:52 +00:00
|
|
|
* This function print the results of the command "btrfs fi usage"
|
2014-02-13 19:19:50 +00:00
|
|
|
* in linear format
|
|
|
|
*/
|
2014-12-18 13:52:36 +00:00
|
|
|
static void _cmd_filesystem_usage_linear(unsigned unit_mode,
|
2014-02-13 19:19:50 +00:00
|
|
|
struct btrfs_ioctl_space_args *sargs,
|
|
|
|
struct chunk_info *info_ptr,
|
|
|
|
int info_count,
|
2022-09-23 08:07:29 +00:00
|
|
|
struct device_info *devinfo,
|
|
|
|
int devcount)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < sargs->total_spaces; i++) {
|
|
|
|
const char *description;
|
|
|
|
const char *r_mode;
|
|
|
|
u64 flags = sargs->spaces[i].flags;
|
2014-04-29 15:32:22 +00:00
|
|
|
|
|
|
|
if (flags & BTRFS_SPACE_INFO_GLOBAL_RSV)
|
|
|
|
continue;
|
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
description = btrfs_group_type_str(flags);
|
|
|
|
r_mode = btrfs_group_profile_str(flags);
|
|
|
|
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "%s,%s: Size:%s, ",
|
2014-02-13 19:19:50 +00:00
|
|
|
description,
|
|
|
|
r_mode,
|
2014-04-28 16:13:16 +00:00
|
|
|
pretty_size_mode(sargs->spaces[i].total_bytes,
|
2014-12-18 13:52:36 +00:00
|
|
|
unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "Used:%s (%.2f%%)\n",
|
2019-11-19 18:06:55 +00:00
|
|
|
pretty_size_mode(sargs->spaces[i].used_bytes, unit_mode),
|
|
|
|
100.0f * sargs->spaces[i].used_bytes /
|
|
|
|
(sargs->spaces[i].total_bytes + 1));
|
2014-04-23 17:00:22 +00:00
|
|
|
print_chunk_device(flags, info_ptr, info_count,
|
2022-09-23 08:07:29 +00:00
|
|
|
devinfo, devcount, unit_mode);
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "\n");
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
2018-03-29 08:23:12 +00:00
|
|
|
if (info_count) {
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "Unallocated:\n");
|
2022-09-23 08:07:29 +00:00
|
|
|
print_unused(info_ptr, info_count, devinfo,
|
|
|
|
devcount, unit_mode | UNITS_NEGATIVE);
|
2018-03-29 08:23:12 +00:00
|
|
|
}
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
2014-04-25 17:39:11 +00:00
|
|
|
static int print_filesystem_usage_by_chunk(int fd,
|
|
|
|
struct chunk_info *chunkinfo, int chunkcount,
|
|
|
|
struct device_info *devinfo, int devcount,
|
2018-05-16 21:38:36 +00:00
|
|
|
const char *path, unsigned unit_mode, int tabular)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
2014-04-25 17:39:11 +00:00
|
|
|
struct btrfs_ioctl_space_args *sargs;
|
2014-02-13 19:19:50 +00:00
|
|
|
int ret = 0;
|
|
|
|
|
2014-04-25 17:39:11 +00:00
|
|
|
sargs = load_space_info(fd, path);
|
|
|
|
if (!sargs) {
|
|
|
|
ret = 1;
|
2014-04-28 16:55:05 +00:00
|
|
|
goto out;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tabular)
|
2014-12-18 13:52:36 +00:00
|
|
|
_cmd_filesystem_usage_tabular(unit_mode, sargs, chunkinfo,
|
2014-04-25 17:39:11 +00:00
|
|
|
chunkcount, devinfo, devcount);
|
2014-02-13 19:19:50 +00:00
|
|
|
else
|
2014-12-18 13:52:36 +00:00
|
|
|
_cmd_filesystem_usage_linear(unit_mode, sargs, chunkinfo,
|
2014-04-25 17:39:11 +00:00
|
|
|
chunkcount, devinfo, devcount);
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2014-04-25 17:39:11 +00:00
|
|
|
free(sargs);
|
2014-04-28 16:55:05 +00:00
|
|
|
out:
|
2014-02-13 19:19:50 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-03-07 02:07:03 +00:00
|
|
|
static const char * const cmd_filesystem_usage_usage[] = {
|
2014-12-17 17:11:58 +00:00
|
|
|
"btrfs filesystem usage [options] <path> [<path>..]",
|
|
|
|
"Show detailed information about internal filesystem usage .",
|
2019-06-21 13:36:04 +00:00
|
|
|
"",
|
2015-11-06 15:39:49 +00:00
|
|
|
HELPINFO_UNITS_SHORT_LONG,
|
2023-02-22 00:37:18 +00:00
|
|
|
OPTLINE("-T", "show data in tabular format"),
|
2014-02-13 19:19:50 +00:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2018-03-07 02:24:46 +00:00
|
|
|
static int cmd_filesystem_usage(const struct cmd_struct *cmd,
|
|
|
|
int argc, char **argv)
|
2014-02-13 19:19:50 +00:00
|
|
|
{
|
2014-04-28 16:55:05 +00:00
|
|
|
int ret = 0;
|
2015-09-01 07:12:06 +00:00
|
|
|
unsigned unit_mode;
|
|
|
|
int i;
|
|
|
|
int more_than_one = 0;
|
|
|
|
int tabular = 0;
|
|
|
|
|
|
|
|
unit_mode = get_unit_mode_from_arg(&argc, argv, 1);
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2018-06-20 00:38:38 +00:00
|
|
|
optind = 0;
|
2014-02-13 19:19:50 +00:00
|
|
|
while (1) {
|
2015-04-08 15:33:55 +00:00
|
|
|
int c;
|
2014-04-25 17:39:11 +00:00
|
|
|
|
2015-09-01 07:12:06 +00:00
|
|
|
c = getopt(argc, argv, "T");
|
2014-02-13 19:19:50 +00:00
|
|
|
if (c < 0)
|
|
|
|
break;
|
2015-09-01 07:12:06 +00:00
|
|
|
|
2014-02-13 19:19:50 +00:00
|
|
|
switch (c) {
|
2014-12-17 17:05:48 +00:00
|
|
|
case 'T':
|
2014-02-13 19:19:50 +00:00
|
|
|
tabular = 1;
|
|
|
|
break;
|
|
|
|
default:
|
2018-03-07 02:54:38 +00:00
|
|
|
usage_unknown_option(cmd, argv);
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-06 03:36:46 +00:00
|
|
|
if (check_argc_min(argc - optind, 1))
|
2019-03-04 13:49:15 +00:00
|
|
|
return 1;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2014-04-25 17:39:11 +00:00
|
|
|
for (i = optind; i < argc; i++) {
|
|
|
|
int fd;
|
|
|
|
DIR *dirstream = NULL;
|
|
|
|
struct chunk_info *chunkinfo = NULL;
|
|
|
|
struct device_info *devinfo = NULL;
|
|
|
|
int chunkcount = 0;
|
|
|
|
int devcount = 0;
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2015-10-12 13:22:55 +00:00
|
|
|
fd = btrfs_open_dir(argv[i], &dirstream, 1);
|
2014-02-13 19:19:50 +00:00
|
|
|
if (fd < 0) {
|
2014-04-28 16:55:05 +00:00
|
|
|
ret = 1;
|
|
|
|
goto out;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
2014-04-25 15:24:40 +00:00
|
|
|
if (more_than_one)
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "\n");
|
2014-04-25 15:24:40 +00:00
|
|
|
|
2014-04-25 17:39:11 +00:00
|
|
|
ret = load_chunk_and_device_info(fd, &chunkinfo, &chunkcount,
|
|
|
|
&devinfo, &devcount);
|
|
|
|
if (ret)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = print_filesystem_usage_overall(fd, chunkinfo, chunkcount,
|
2014-12-17 17:11:58 +00:00
|
|
|
devinfo, devcount, argv[i], unit_mode);
|
2014-04-25 17:39:11 +00:00
|
|
|
if (ret)
|
|
|
|
goto cleanup;
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, "\n");
|
2014-04-25 17:39:11 +00:00
|
|
|
ret = print_filesystem_usage_by_chunk(fd, chunkinfo, chunkcount,
|
2014-12-17 17:11:58 +00:00
|
|
|
devinfo, devcount, argv[i], unit_mode, tabular);
|
2014-04-25 17:39:11 +00:00
|
|
|
cleanup:
|
2014-02-13 19:19:50 +00:00
|
|
|
close_file_or_dir(fd, dirstream);
|
2014-04-25 17:39:11 +00:00
|
|
|
free(chunkinfo);
|
|
|
|
free(devinfo);
|
2014-02-13 19:19:50 +00:00
|
|
|
|
2014-04-25 17:39:11 +00:00
|
|
|
if (ret)
|
2014-04-28 16:55:05 +00:00
|
|
|
goto out;
|
2014-02-13 19:19:50 +00:00
|
|
|
more_than_one = 1;
|
|
|
|
}
|
|
|
|
|
2014-04-28 16:55:05 +00:00
|
|
|
out:
|
|
|
|
return !!ret;
|
2014-02-13 19:19:50 +00:00
|
|
|
}
|
2018-03-07 02:07:03 +00:00
|
|
|
DEFINE_SIMPLE_COMMAND(filesystem_usage, "usage");
|
2014-02-13 19:20:12 +00:00
|
|
|
|
2017-02-09 16:42:02 +00:00
|
|
|
void print_device_chunks(struct device_info *devinfo,
|
2014-04-23 17:00:22 +00:00
|
|
|
struct chunk_info *chunks_info_ptr,
|
2014-12-18 13:52:36 +00:00
|
|
|
int chunks_info_count, unsigned unit_mode)
|
2014-02-13 19:20:12 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
u64 allocated = 0;
|
|
|
|
|
|
|
|
for (i = 0 ; i < chunks_info_count ; i++) {
|
|
|
|
const char *description;
|
|
|
|
const char *r_mode;
|
|
|
|
u64 flags;
|
|
|
|
u64 size;
|
2021-06-08 15:35:20 +00:00
|
|
|
u64 num_stripes;
|
|
|
|
u64 profile;
|
2014-02-13 19:20:12 +00:00
|
|
|
|
2014-04-24 16:57:12 +00:00
|
|
|
if (chunks_info_ptr[i].devid != devinfo->devid)
|
2014-02-13 19:20:12 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
flags = chunks_info_ptr[i].type;
|
2021-06-08 15:35:20 +00:00
|
|
|
profile = (flags & BTRFS_BLOCK_GROUP_PROFILE_MASK);
|
2014-02-13 19:20:12 +00:00
|
|
|
|
|
|
|
description = btrfs_group_type_str(flags);
|
|
|
|
r_mode = btrfs_group_profile_str(flags);
|
|
|
|
size = calc_chunk_size(chunks_info_ptr+i);
|
2021-06-08 15:35:20 +00:00
|
|
|
num_stripes = chunks_info_ptr[i].num_stripes;
|
|
|
|
|
2021-10-06 23:29:47 +00:00
|
|
|
if (btrfs_bg_type_is_stripey(profile)) {
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " %s,%s/%llu:%*s%10s\n",
|
2021-06-08 15:35:20 +00:00
|
|
|
description,
|
|
|
|
r_mode,
|
|
|
|
num_stripes,
|
|
|
|
(int)(20 - strlen(description) - strlen(r_mode)
|
|
|
|
- count_digits(num_stripes) - 1), "",
|
|
|
|
pretty_size_mode(size, unit_mode));
|
2021-10-06 23:29:47 +00:00
|
|
|
} else {
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " %s,%s:%*s%10s\n",
|
2021-06-08 15:35:20 +00:00
|
|
|
description,
|
|
|
|
r_mode,
|
|
|
|
(int)(20 - strlen(description) - strlen(r_mode)), "",
|
|
|
|
pretty_size_mode(size, unit_mode));
|
|
|
|
}
|
2014-02-13 19:20:12 +00:00
|
|
|
|
|
|
|
allocated += size;
|
|
|
|
|
|
|
|
}
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Unallocated: %*s%10s\n",
|
2014-02-13 19:20:12 +00:00
|
|
|
(int)(20 - strlen("Unallocated")), "",
|
2017-01-03 17:26:43 +00:00
|
|
|
pretty_size_mode(devinfo->size - allocated,
|
|
|
|
unit_mode | UNITS_NEGATIVE));
|
2014-04-24 16:32:27 +00:00
|
|
|
}
|
2014-02-13 19:20:12 +00:00
|
|
|
|
2017-02-09 16:42:02 +00:00
|
|
|
void print_device_sizes(struct device_info *devinfo, unsigned unit_mode)
|
2014-04-24 16:32:27 +00:00
|
|
|
{
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device size: %*s%10s\n",
|
2014-04-24 16:32:27 +00:00
|
|
|
(int)(20 - strlen("Device size")), "",
|
2014-12-18 13:52:36 +00:00
|
|
|
pretty_size_mode(devinfo->device_size, unit_mode));
|
2022-10-04 14:50:09 +00:00
|
|
|
pr_verbose(LOG_DEFAULT, " Device slack: %*s%10s\n",
|
2016-05-17 14:21:00 +00:00
|
|
|
(int)(20 - strlen("Device slack")), "",
|
2022-09-14 19:20:43 +00:00
|
|
|
pretty_size_mode(calc_slack_size(devinfo), unit_mode));
|
2014-02-13 19:20:12 +00:00
|
|
|
}
|