#include "crc32c.h"
static void print_usage(void);
-static void dump_superblock(struct btrfs_super_block *sb);
+static void dump_superblock(struct btrfs_super_block *sb, int full);
int main(int argc, char **argv);
-static int load_and_dump_sb(char *, int fd, u64 sb_bytenr);
+static int load_and_dump_sb(char *, int fd, u64 sb_bytenr, int full, int force);
static void print_usage(void)
{
fprintf(stderr,
- "usage: btrfs-show-super [-i super_mirror|-a] dev [dev..]\n");
- fprintf(stderr, "\tThe super_mirror number is between 0 and %d.\n",
- BTRFS_SUPER_MIRROR_MAX - 1);
- fprintf(stderr, "\tIf -a is passed all the superblocks are showed.\n");
+ "usage: btrfs-show-super [-i super_mirror|-a|-f|-F] dev [dev..]\n");
+ fprintf(stderr, "\t-f : print full superblock information\n");
+ fprintf(stderr, "\t-a : print information of all superblocks\n");
+ fprintf(stderr, "\t-i <super_mirror> : specify which mirror to print out\n");
+ fprintf(stderr, "\t-F : attempt to dump superblocks with bad magic\n");
fprintf(stderr, "%s\n", BTRFS_BUILD_VERSION);
}
{
int opt;
int all = 0;
+ int full = 0;
+ int force = 0;
char *filename;
int fd = -1;
- int arg, i;
+ int i;
+ u64 arg;
u64 sb_bytenr = btrfs_sb_offset(0);
- while ((opt = getopt(argc, argv, "ai:")) != -1) {
+ while ((opt = getopt(argc, argv, "fFai:")) != -1) {
switch (opt) {
case 'i':
- arg = atoi(optarg);
-
- if (arg < 0 || arg >= BTRFS_SUPER_MIRROR_MAX) {
+ arg = arg_strtou64(optarg);
+ if (arg >= BTRFS_SUPER_MIRROR_MAX) {
fprintf(stderr,
- "Illegal super_mirror %d\n",
+ "Illegal super_mirror %llu\n",
arg);
print_usage();
exit(1);
case 'a':
all = 1;
break;
-
+ case 'f':
+ full = 1;
+ break;
+ case 'F':
+ force = 1;
+ break;
default:
print_usage();
exit(1);
fd = open(filename, O_RDONLY, 0666);
if (fd < 0) {
fprintf(stderr, "Could not open %s\n", filename);
- close(fd);
exit(1);
}
int idx;
for (idx = 0; idx < BTRFS_SUPER_MIRROR_MAX; idx++) {
sb_bytenr = btrfs_sb_offset(idx);
- if (load_and_dump_sb(filename, fd, sb_bytenr)) {
+ if (load_and_dump_sb(filename, fd,
+ sb_bytenr, full, force)) {
close(fd);
exit(1);
}
putchar('\n');
}
} else {
- load_and_dump_sb(filename, fd, sb_bytenr);
+ load_and_dump_sb(filename, fd, sb_bytenr, full, force);
putchar('\n');
}
close(fd);
exit(0);
}
-static int load_and_dump_sb(char *filename, int fd, u64 sb_bytenr)
+static int load_and_dump_sb(char *filename, int fd, u64 sb_bytenr, int full,
+ int force)
{
u8 super_block_data[BTRFS_SUPER_INFO_SIZE];
struct btrfs_super_block *sb;
}
printf("superblock: bytenr=%llu, device=%s\n", sb_bytenr, filename);
printf("---------------------------------------------------------\n");
- dump_superblock(sb);
+ if (btrfs_super_magic(sb) != BTRFS_MAGIC && !force) {
+ fprintf(stderr,
+ "ERROR: bad magic on superblock on %s at %llu\n",
+ filename, (unsigned long long)sb_bytenr);
+ } else {
+ dump_superblock(sb, full);
+ }
return 0;
}
static int check_csum_sblock(void *sb, int csum_size)
{
- char result[csum_size];
+ char result[BTRFS_CSUM_SIZE];
u32 crc = ~(u32)0;
crc = btrfs_csum_data(NULL, (char *)sb + BTRFS_CSUM_SIZE,
return !memcmp(sb, &result, csum_size);
}
-static void dump_superblock(struct btrfs_super_block *sb)
+static void print_sys_chunk_array(struct btrfs_super_block *sb)
+{
+ struct extent_buffer *buf;
+ struct btrfs_disk_key *disk_key;
+ struct btrfs_chunk *chunk;
+ struct btrfs_key key;
+ u8 *ptr, *array_end;
+ u32 num_stripes;
+ u32 len = 0;
+ int i = 0;
+
+ buf = malloc(sizeof(*buf) + sizeof(*sb));
+ if (!buf) {
+ fprintf(stderr, "%s\n", strerror(ENOMEM));
+ exit(1);
+ }
+ write_extent_buffer(buf, sb, 0, sizeof(*sb));
+ ptr = sb->sys_chunk_array;
+ array_end = ptr + btrfs_super_sys_array_size(sb);
+
+ while (ptr < array_end) {
+ disk_key = (struct btrfs_disk_key *)ptr;
+ btrfs_disk_key_to_cpu(&key, disk_key);
+
+ printf("\titem %d ", i);
+ btrfs_print_key(disk_key);
+
+ len = sizeof(*disk_key);
+ putchar('\n');
+ ptr += len;
+
+ if (key.type == BTRFS_CHUNK_ITEM_KEY) {
+ chunk = (struct btrfs_chunk *)(ptr - (u8 *)sb);
+ print_chunk(buf, chunk);
+ num_stripes = btrfs_chunk_num_stripes(buf, chunk);
+ len = btrfs_chunk_item_size(num_stripes);
+ } else {
+ BUG();
+ }
+
+ ptr += len;
+ i++;
+ }
+
+ free(buf);
+}
+
+static int empty_backup(struct btrfs_root_backup *backup)
{
+ if (backup == NULL ||
+ (backup->tree_root == 0 &&
+ backup->tree_root_gen == 0))
+ return 1;
+ return 0;
+}
+
+static void print_root_backup(struct btrfs_root_backup *backup)
+{
+ printf("\t\tbackup_tree_root:\t%llu\tgen: %llu\tlevel: %d\n",
+ btrfs_backup_tree_root(backup),
+ btrfs_backup_tree_root_gen(backup),
+ btrfs_backup_tree_root_level(backup));
+ printf("\t\tbackup_chunk_root:\t%llu\tgen: %llu\tlevel: %d\n",
+ btrfs_backup_chunk_root(backup),
+ btrfs_backup_chunk_root_gen(backup),
+ btrfs_backup_chunk_root_level(backup));
+ printf("\t\tbackup_extent_root:\t%llu\tgen: %llu\tlevel: %d\n",
+ btrfs_backup_extent_root(backup),
+ btrfs_backup_extent_root_gen(backup),
+ btrfs_backup_extent_root_level(backup));
+ printf("\t\tbackup_fs_root:\t\t%llu\tgen: %llu\tlevel: %d\n",
+ btrfs_backup_fs_root(backup),
+ btrfs_backup_fs_root_gen(backup),
+ btrfs_backup_fs_root_level(backup));
+ printf("\t\tbackup_dev_root:\t%llu\tgen: %llu\tlevel: %d\n",
+ btrfs_backup_dev_root(backup),
+ btrfs_backup_dev_root_gen(backup),
+ btrfs_backup_dev_root_level(backup));
+ printf("\t\tbackup_csum_root:\t%llu\tgen: %llu\tlevel: %d\n",
+ btrfs_backup_csum_root(backup),
+ btrfs_backup_csum_root_gen(backup),
+ btrfs_backup_csum_root_level(backup));
+
+ printf("\t\tbackup_total_bytes:\t%llu\n",
+ btrfs_backup_total_bytes(backup));
+ printf("\t\tbackup_bytes_used:\t%llu\n",
+ btrfs_backup_bytes_used(backup));
+ printf("\t\tbackup_num_devices:\t%llu\n",
+ btrfs_backup_num_devices(backup));
+ putchar('\n');
+}
+
+static void print_backup_roots(struct btrfs_super_block *sb)
+{
+ struct btrfs_root_backup *backup;
int i;
- char *s, buf[36+1];
+
+ for (i = 0; i < BTRFS_NUM_BACKUP_ROOTS; i++) {
+ backup = sb->super_roots + i;
+ if (!empty_backup(backup)) {
+ printf("\tbackup %d:\n", i);
+ print_root_backup(backup);
+ }
+ }
+}
+
+static void dump_superblock(struct btrfs_super_block *sb, int full)
+{
+ int i;
+ char *s, buf[BTRFS_UUID_UNPARSED_SIZE];
u8 *p;
printf("csum\t\t\t0x");
s = (char *) &sb->magic;
for (i = 0; i < 8; i++)
putchar(isprint(s[i]) ? s[i] : '.');
- if (sb->magic == cpu_to_le64(BTRFS_MAGIC))
+ if (btrfs_super_magic(sb) == BTRFS_MAGIC)
printf(" [match]\n");
else
printf(" [DON'T MATCH]\n");
(unsigned long long)btrfs_super_csum_size(sb));
printf("cache_generation\t%llu\n",
(unsigned long long)btrfs_super_cache_generation(sb));
+ printf("uuid_tree_generation\t%llu\n",
+ (unsigned long long)btrfs_super_uuid_tree_generation(sb));
uuid_unparse(sb->dev_item.uuid, buf);
printf("dev_item.uuid\t\t%s\n", buf);
btrfs_stack_device_bandwidth(&sb->dev_item));
printf("dev_item.generation\t%llu\n", (unsigned long long)
btrfs_stack_device_generation(&sb->dev_item));
+ if (full) {
+ printf("sys_chunk_array[%d]:\n", BTRFS_SYSTEM_CHUNK_ARRAY_SIZE);
+ print_sys_chunk_array(sb);
+ printf("backup_roots[%d]:\n", BTRFS_NUM_BACKUP_ROOTS);
+ print_backup_roots(sb);
+ }
}