/* * Copyright (c) 2016 Jeff Mahoney <jeffm@suse.com> * Copyright (c) 2016-2021 The strace developers. * All rights reserved. * * SPDX-License-Identifier: LGPL-2.1-or-later */ #include "defs.h" #include DEF_MPERS_TYPE(struct_btrfs_ioctl_dev_replace_args) #include DEF_MPERS_TYPE(struct_btrfs_ioctl_send_args) #include DEF_MPERS_TYPE(struct_btrfs_ioctl_received_subvol_args) #include DEF_MPERS_TYPE(struct_btrfs_ioctl_timespec) #include DEF_MPERS_TYPE(struct_btrfs_ioctl_vol_args_v2) #include <linux/btrfs_tree.h> typedef struct btrfs_ioctl_dev_replace_args struct_btrfs_ioctl_dev_replace_args; typedef struct btrfs_ioctl_send_args struct_btrfs_ioctl_send_args; typedef struct btrfs_ioctl_received_subvol_args struct_btrfs_ioctl_received_subvol_args; typedef struct btrfs_ioctl_timespec struct_btrfs_ioctl_timespec; typedef struct btrfs_ioctl_vol_args_v2 struct_btrfs_ioctl_vol_args_v2; #include MPERS_DEFS #include <linux/fs.h> #include "xlat/btrfs_balance_args.h" #include "xlat/btrfs_balance_ctl_cmds.h" #include "xlat/btrfs_balance_flags.h" #include "xlat/btrfs_balance_state.h" #include "xlat/btrfs_compress_types.h" #include "xlat/btrfs_cont_reading_from_srcdev_mode.h" #include "xlat/btrfs_csum_types.h" #include "xlat/btrfs_defrag_flags.h" #include "xlat/btrfs_dev_replace_cmds.h" #include "xlat/btrfs_dev_replace_results.h" #include "xlat/btrfs_dev_replace_state.h" #include "xlat/btrfs_dev_stats_flags.h" #include "xlat/btrfs_dev_stats_values.h" #include "xlat/btrfs_features_compat.h" #include "xlat/btrfs_features_compat_ro.h" #include "xlat/btrfs_features_incompat.h" #include "xlat/btrfs_fs_info_flags.h" #include "xlat/btrfs_key_types.h" #include "xlat/btrfs_logical_ino_args_flags.h" #include "xlat/btrfs_qgroup_ctl_cmds.h" #include "xlat/btrfs_qgroup_inherit_flags.h" #include "xlat/btrfs_qgroup_limit_flags.h" #include "xlat/btrfs_qgroup_status_flags.h" #include "xlat/btrfs_scrub_flags.h" #include "xlat/btrfs_send_flags.h" #include "xlat/btrfs_snap_flags_v2.h" #include "xlat/btrfs_space_info_flags.h" #include "xlat/btrfs_tree_objectids.h" static void btrfs_print_balance_args(const struct btrfs_balance_args *const bba) { tprint_struct_begin(); PRINT_FIELD_FLAGS(*bba, profiles, btrfs_space_info_flags, "BTRFS_BLOCK_GROUP_???"); tprint_struct_next(); PRINT_FIELD_U64(*bba, usage); tprint_struct_next(); PRINT_FIELD_DEV(*bba, devid); tprint_struct_next(); PRINT_FIELD_U64(*bba, pstart); tprint_struct_next(); PRINT_FIELD_U64(*bba, pend); tprint_struct_next(); PRINT_FIELD_U64(*bba, vstart); tprint_struct_next(); PRINT_FIELD_U64(*bba, vend); tprint_struct_next(); PRINT_FIELD_U64(*bba, target); tprint_struct_next(); PRINT_FIELD_FLAGS(*bba, flags, btrfs_balance_args, "BTRFS_BALANCE_ARGS_???"); tprint_struct_end(); } static void btrfs_print_balance(struct tcb *const tcp, const kernel_ulong_t arg, bool out) { struct btrfs_ioctl_balance_args balance_args; if (umove_or_printaddr(tcp, arg, &balance_args)) return; tprint_struct_begin(); PRINT_FIELD_FLAGS(balance_args, flags, btrfs_balance_flags, "BTRFS_BALANCE_???"); if (out) { tprint_struct_next(); PRINT_FIELD_FLAGS(balance_args, state, btrfs_balance_state, "BTRFS_BALANCE_STATE_???"); } if (balance_args.flags & BTRFS_BALANCE_DATA) { tprint_struct_next(); PRINT_FIELD_OBJ_PTR(balance_args, data, btrfs_print_balance_args); } if (balance_args.flags & BTRFS_BALANCE_METADATA) { tprint_struct_next(); PRINT_FIELD_OBJ_PTR(balance_args, meta, btrfs_print_balance_args); } if (balance_args.flags & BTRFS_BALANCE_SYSTEM) { tprint_struct_next(); PRINT_FIELD_OBJ_PTR(balance_args, sys, btrfs_print_balance_args); } tprint_struct_end(); } static void btrfs_print_features(const struct btrfs_ioctl_feature_flags *flags) { tprint_struct_begin(); PRINT_FIELD_FLAGS(*flags, compat_flags, btrfs_features_compat, "BTRFS_FEATURE_COMPAT_???"); tprint_struct_next(); PRINT_FIELD_FLAGS(*flags, compat_ro_flags, btrfs_features_compat_ro, "BTRFS_FEATURE_COMPAT_RO_???"); tprint_struct_next(); PRINT_FIELD_FLAGS(*flags, incompat_flags, btrfs_features_incompat, "BTRFS_FEATURE_INCOMPAT_???"); tprint_struct_end(); } static void btrfs_print_qgroup_limit(const struct btrfs_qgroup_limit *lim) { tprint_struct_begin(); PRINT_FIELD_FLAGS(*lim, flags, btrfs_qgroup_limit_flags, "BTRFS_QGROUP_LIMIT_???"); tprint_struct_next(); PRINT_FIELD_U(*lim, max_rfer); tprint_struct_next(); PRINT_FIELD_U(*lim, max_excl); tprint_struct_next(); PRINT_FIELD_U(*lim, rsv_rfer); tprint_struct_next(); PRINT_FIELD_U(*lim, rsv_excl); tprint_struct_end(); } #define btrfs_print_key_type(where_, field_) \ PRINT_FIELD_XVAL_U((where_), field_, btrfs_key_types, NULL) #define btrfs_print_objectid(where_, field_) \ PRINT_FIELD_XVAL_U((where_), field_, btrfs_tree_objectids, NULL) static void btrfs_print_data_container_header(const struct btrfs_data_container *container) { tprint_struct_begin(); PRINT_FIELD_U(*container, bytes_left); tprint_struct_next(); PRINT_FIELD_U(*container, bytes_missing); tprint_struct_next(); PRINT_FIELD_U(*container, elem_cnt); tprint_struct_next(); PRINT_FIELD_U(*container, elem_missed); } static void btrfs_print_data_container_footer(void) { tprint_struct_end(); } static bool print_btrfs_data_container_logical_ino(struct tcb *tcp, void *elem_buf, size_t elem_size, void *data) { const struct { uint64_t inum; uint64_t offset; uint64_t root; } *const record = elem_buf; tprint_struct_begin(); PRINT_FIELD_U(*record, inum); tprint_struct_next(); PRINT_FIELD_U(*record, offset); tprint_struct_next(); PRINT_FIELD_U(*record, root); tprint_struct_end(); return true; } static void btrfs_print_logical_ino_container(struct tcb *tcp, const uint64_t inodes_addr) { struct btrfs_data_container container; if (umove_or_printaddr(tcp, inodes_addr, &container)) return; btrfs_print_data_container_header(&container); if (abbrev(tcp)) { tprint_struct_next(); tprint_more_data_follows(); } else { const uint64_t val_addr = inodes_addr + offsetof(typeof(container), val); uint64_t record[3]; tprint_struct_next(); tprints_field_name("val"); print_array(tcp, val_addr, container.elem_cnt / 3, record, sizeof(record), tfetch_mem, print_btrfs_data_container_logical_ino, 0); } btrfs_print_data_container_footer(); } static bool print_btrfs_data_container_ino_path(struct tcb *tcp, void *elem_buf, size_t elem_size, void *data) { const uint64_t *const offset = elem_buf; const uint64_t *const val_addr = data; printpath(tcp, *val_addr + *offset); return true; } static void btrfs_print_ino_path_container(struct tcb *tcp, const uint64_t fspath_addr) { struct btrfs_data_container container; if (umove_or_printaddr(tcp, fspath_addr, &container)) return; btrfs_print_data_container_header(&container); if (abbrev(tcp)) { tprint_struct_next(); tprint_more_data_follows(); } else { uint64_t val_addr = fspath_addr + offsetof(typeof(container), val); uint64_t offset; tprint_struct_next(); tprints_field_name("val"); print_array(tcp, val_addr, container.elem_cnt, &offset, sizeof(offset), tfetch_mem, print_btrfs_data_container_ino_path, &val_addr); } btrfs_print_data_container_footer(); } static void btrfs_print_qgroup_inherit(struct tcb *const tcp, const kernel_ulong_t qgi_addr) { struct btrfs_qgroup_inherit inherit; if (umove_or_printaddr(tcp, qgi_addr, &inherit)) return; tprint_struct_begin(); PRINT_FIELD_FLAGS(inherit, flags, btrfs_qgroup_inherit_flags, "BTRFS_QGROUP_INHERIT_???"); tprint_struct_next(); PRINT_FIELD_U(inherit, num_qgroups); tprint_struct_next(); PRINT_FIELD_U(inherit, num_ref_copies); tprint_struct_next(); PRINT_FIELD_U(inherit, num_excl_copies); tprint_struct_next(); PRINT_FIELD_OBJ_PTR(inherit, lim, btrfs_print_qgroup_limit); if (abbrev(tcp)) { tprint_struct_next(); tprint_more_data_follows(); } else { uint64_t record; tprint_struct_next(); tprints_field_name("qgroups"); print_array(tcp, qgi_addr + offsetof(typeof(inherit), qgroups), inherit.num_qgroups, &record, sizeof(record), tfetch_mem, print_uint_array_member, 0); } tprint_struct_end(); } static void print_btrfs_ioctl_search_key(const struct btrfs_ioctl_search_key *const key, const bool is_entering, const bool is_not_abbrev) { tprint_struct_begin(); if (is_entering) { btrfs_print_objectid(*key, tree_id); if (key->min_objectid != BTRFS_FIRST_FREE_OBJECTID || is_not_abbrev) { tprint_struct_next(); btrfs_print_objectid(*key, min_objectid); } if (key->max_objectid != BTRFS_LAST_FREE_OBJECTID || is_not_abbrev) { tprint_struct_next(); btrfs_print_objectid(*key, max_objectid); } tprint_struct_next(); PRINT_FIELD_U64(*key, min_offset); tprint_struct_next(); PRINT_FIELD_U64(*key, max_offset); tprint_struct_next(); PRINT_FIELD_U64(*key, min_transid); tprint_struct_next(); PRINT_FIELD_U64(*key, max_transid); tprint_struct_next(); btrfs_print_key_type(*key, min_type); tprint_struct_next(); btrfs_print_key_type(*key, max_type); tprint_struct_next(); PRINT_FIELD_U(*key, nr_items); } else { PRINT_FIELD_U(*key, nr_items); } tprint_struct_end(); } static void print_btrfs_ioctl_search_header(const struct btrfs_ioctl_search_header *p) { tprint_struct_begin(); PRINT_FIELD_U(*p, transid); tprint_struct_next(); btrfs_print_objectid(*p, objectid); tprint_struct_next(); PRINT_FIELD_U(*p, offset); tprint_struct_next(); btrfs_print_key_type(*p, type); tprint_struct_next(); PRINT_FIELD_U(*p, len); tprint_struct_end(); } static void decode_search_arg_buf(struct tcb *tcp, kernel_ulong_t buf_addr, uint64_t buf_size, unsigned int nr_items) { if (entering(tcp)) return; tprint_struct_next(); if (abbrev(tcp)) { tprint_more_data_follows(); } else { tprints_field_name("buf"); tprint_array_begin(); uint64_t off = 0; for (unsigned int i = 0; i < nr_items; ++i) { if (i) tprint_array_next(); struct btrfs_ioctl_search_header sh; uint64_t addr = buf_addr + off; if (addr < buf_addr || off + sizeof(sh) > buf_size || i > max_strlen) { tprint_more_data_follows(); break; } if (!tfetch_mem(tcp, addr, sizeof(sh), &sh)) { tprint_more_data_follows(); printaddr_comment(addr); break; } print_btrfs_ioctl_search_header(&sh); off += sizeof(sh) + sh.len; } tprint_array_end(); } } static bool print_objectid_callback(struct tcb *tcp, void *elem_buf, size_t elem_size, void *data) { printxvals_ex(*(uint64_t *) elem_buf, NULL, XLAT_STYLE_FMT_U, btrfs_tree_objectids, NULL); return true; } static bool print_btrfs_ioctl_space_info(struct tcb *tcp, void *elem_buf, size_t elem_size, void *data) { const struct btrfs_ioctl_space_info *info = elem_buf; tprint_struct_begin(); PRINT_FIELD_FLAGS(*info, flags, btrfs_space_info_flags, "BTRFS_SPACE_INFO_???"); tprint_struct_next(); PRINT_FIELD_U(*info, total_bytes); tprint_struct_next(); PRINT_FIELD_U(*info, used_bytes); tprint_struct_end(); return true; } static void print_btrfs_timespec(const MPERS_PTR_ARG(struct_btrfs_ioctl_timespec *) const arg) { const struct_btrfs_ioctl_timespec *const p = arg; tprint_struct_begin(); PRINT_FIELD_U(*p, sec); tprint_struct_next(); PRINT_FIELD_U(*p, nsec); tprint_struct_end(); tprints_comment(sprinttime_nsec(p->sec, p->nsec)); } static void print_btrfs_scrub_progress(const struct btrfs_scrub_progress *const p) { tprint_struct_begin(); PRINT_FIELD_U(*p, data_extents_scrubbed); tprint_struct_next(); PRINT_FIELD_U(*p, tree_extents_scrubbed); tprint_struct_next(); PRINT_FIELD_U(*p, data_bytes_scrubbed); tprint_struct_next(); PRINT_FIELD_U(*p, tree_bytes_scrubbed); tprint_struct_next(); PRINT_FIELD_U(*p, read_errors); tprint_struct_next(); PRINT_FIELD_U(*p, csum_errors); tprint_struct_next(); PRINT_FIELD_U(*p, verify_errors); tprint_struct_next(); PRINT_FIELD_U(*p, no_csum); tprint_struct_next(); PRINT_FIELD_U(*p, csum_discards); tprint_struct_next(); PRINT_FIELD_U(*p, super_errors); tprint_struct_next(); PRINT_FIELD_U(*p, malloc_errors); tprint_struct_next(); PRINT_FIELD_U(*p, uncorrectable_errors); tprint_struct_next(); PRINT_FIELD_U(*p, corrected_errors); tprint_struct_next(); PRINT_FIELD_U(*p, last_physical); tprint_struct_next(); PRINT_FIELD_U(*p, unverified_errors); tprint_struct_end(); } static void print_btrfs_replace_start_params(const typeof_field(struct_btrfs_ioctl_dev_replace_args, start) *const p) { tprint_struct_begin(); PRINT_FIELD_DEV(*p, srcdevid); tprint_struct_next(); PRINT_FIELD_XVAL(*p, cont_reading_from_srcdev_mode, btrfs_cont_reading_from_srcdev_mode, "BTRFS_IOCTL_DEV_REPLACE_CONT_READING" "_FROM_SRCDEV_MODE_???"); tprint_struct_next(); PRINT_FIELD_CSTRING(*p, srcdev_name); tprint_struct_next(); PRINT_FIELD_CSTRING(*p, tgtdev_name); tprint_struct_end(); } static void print_btrfs_replace_status_params(const typeof_field(struct_btrfs_ioctl_dev_replace_args, status) *const p) { tprint_struct_begin(); PRINT_FIELD_XVAL(*p, replace_state, btrfs_dev_replace_state, "BTRFS_IOCTL_DEV_REPLACE_STATE_???"); tprint_struct_next(); PRINT_FIELD_U(*p, progress_1000); if (p->progress_1000 <= 1000) tprintf_comment("%u.%u%%", (unsigned) p->progress_1000 / 10, (unsigned) p->progress_1000 % 10); tprint_struct_next(); PRINT_FIELD_U(*p, time_started); tprints_comment(sprinttime(p->time_started)); tprint_struct_next(); PRINT_FIELD_U(*p, time_stopped); tprints_comment(sprinttime(p->time_stopped)); tprint_struct_next(); PRINT_FIELD_U(*p, num_write_errors); tprint_struct_next(); PRINT_FIELD_U(*p, num_uncorrectable_read_errors); tprint_struct_end(); } MPERS_PRINTER_DECL(int, btrfs_ioctl, struct tcb *const tcp, const unsigned int code, const kernel_ulong_t arg) { switch (code) { /* Take no arguments; command only. */ case BTRFS_IOC_TRANS_START: case BTRFS_IOC_TRANS_END: case BTRFS_IOC_SYNC: case BTRFS_IOC_SCRUB_CANCEL: case BTRFS_IOC_QUOTA_RESCAN_WAIT: /* * The codes for these ioctls are based on each accepting a * vol_args but none of them actually consume an argument. */ case BTRFS_IOC_DEFRAG: case BTRFS_IOC_BALANCE: break; /* takes a signed int */ case BTRFS_IOC_BALANCE_CTL: tprint_arg_next(); printxval(btrfs_balance_ctl_cmds, arg, "BTRFS_BALANCE_CTL_???"); break; /* returns a 64 */ case BTRFS_IOC_START_SYNC: /* R */ if (entering(tcp)) return 0; ATTRIBUTE_FALLTHROUGH; /* takes a u64 */ case BTRFS_IOC_DEFAULT_SUBVOL: /* W */ case BTRFS_IOC_WAIT_SYNC: /* W */ tprint_arg_next(); printnum_int64(tcp, arg, "%" PRIu64); break; /* u64 but describe a flags bitfield; we can make that symbolic */ case BTRFS_IOC_SUBVOL_GETFLAGS: { /* R */ uint64_t flags; if (entering(tcp)) return 0; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &flags)) break; printflags64(btrfs_snap_flags_v2, flags, "BTRFS_SUBVOL_???"); break; } case BTRFS_IOC_SUBVOL_SETFLAGS: { /* W */ uint64_t flags; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &flags)) break; printflags64(btrfs_snap_flags_v2, flags, "BTRFS_SUBVOL_???"); break; } /* More complex types */ case BTRFS_IOC_BALANCE_V2: /* RW */ if (entering(tcp)) { tprint_arg_next(); btrfs_print_balance(tcp, arg, false); return 0; } if (syserror(tcp)) break; tprint_value_changed(); btrfs_print_balance(tcp, arg, true); break; case BTRFS_IOC_BALANCE_PROGRESS: /* R */ if (entering(tcp)) return 0; tprint_arg_next(); btrfs_print_balance(tcp, arg, true); break; case BTRFS_IOC_DEFRAG_RANGE: { /* W */ struct btrfs_ioctl_defrag_range_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, start); tprint_struct_next(); PRINT_FIELD_U64(args, len); tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_defrag_flags, "BTRFS_DEFRAG_RANGE_???"); tprint_struct_next(); PRINT_FIELD_U(args, extent_thresh); tprint_struct_next(); PRINT_FIELD_XVAL(args, compress_type, btrfs_compress_types, "BTRFS_COMPRESS_???"); tprint_struct_end(); break; } case BTRFS_IOC_DEV_INFO: { /* RW */ struct btrfs_ioctl_dev_info_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_DEV(args, devid); if (!IS_ARRAY_ZERO(args.uuid)) { tprint_struct_next(); PRINT_FIELD_UUID(args, uuid); } tprint_struct_end(); return 0; } tprint_struct_begin(); if (!IS_ARRAY_ZERO(args.uuid)) { PRINT_FIELD_UUID(args, uuid); tprint_struct_next(); } PRINT_FIELD_U(args, bytes_used); tprint_struct_next(); PRINT_FIELD_U(args, total_bytes); tprint_struct_next(); PRINT_FIELD_CSTRING(args, path); tprint_struct_end(); break; } case BTRFS_IOC_DEV_REPLACE: { /* RW */ struct_btrfs_ioctl_dev_replace_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_XVAL(args, cmd, btrfs_dev_replace_cmds, "BTRFS_IOCTL_DEV_REPLACE_CMD_???"); if (args.cmd == BTRFS_IOCTL_DEV_REPLACE_CMD_START) { tprint_struct_next(); PRINT_FIELD_OBJ_PTR(args, start, print_btrfs_replace_start_params); } tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_XVAL(args, result, btrfs_dev_replace_results, "BTRFS_IOCTL_DEV_REPLACE_RESULT_???"); if (args.cmd == BTRFS_IOCTL_DEV_REPLACE_CMD_STATUS) { tprint_struct_next(); PRINT_FIELD_OBJ_PTR(args, status, print_btrfs_replace_status_params); } tprint_struct_end(); break; } case BTRFS_IOC_GET_FEATURES: { /* R */ struct btrfs_ioctl_feature_flags flags; if (entering(tcp)) return 0; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &flags)) break; btrfs_print_features(&flags); break; } case BTRFS_IOC_SET_FEATURES: { /* W */ struct btrfs_ioctl_feature_flags flarg[2]; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &flarg)) break; tprint_array_begin(); btrfs_print_features(&flarg[0]); tprint_array_next(); btrfs_print_features(&flarg[1]); tprint_array_end(); break; } case BTRFS_IOC_GET_SUPPORTED_FEATURES: { /* R */ struct btrfs_ioctl_feature_flags flarg[3]; if (entering(tcp)) return 0; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &flarg)) break; tprint_array_begin(); btrfs_print_features(&flarg[0]); tprints_comment("supported"); tprint_array_next(); btrfs_print_features(&flarg[1]); tprints_comment("safe to set"); tprint_array_next(); btrfs_print_features(&flarg[2]); tprints_comment("safe to clear"); tprint_array_end(); break; } case BTRFS_IOC_FS_INFO: { /* R */ struct btrfs_ioctl_fs_info_args args; if (entering(tcp)) return 0; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, max_id); tprint_struct_next(); PRINT_FIELD_U(args, num_devices); tprint_struct_next(); PRINT_FIELD_UUID(args, fsid); tprint_struct_next(); PRINT_FIELD_U(args, nodesize); tprint_struct_next(); PRINT_FIELD_U(args, sectorsize); tprint_struct_next(); PRINT_FIELD_U(args, clone_alignment); if (args.flags & BTRFS_FS_INFO_FLAG_CSUM_INFO) { tprint_struct_next(); PRINT_FIELD_XVAL(args, csum_type, btrfs_csum_types, "BTRFS_CSUM_TYPE_???"); tprint_struct_next(); PRINT_FIELD_U(args, csum_size); } tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_fs_info_flags, "BTRFS_FS_INFO_FLAG_???"); if (args.flags & BTRFS_FS_INFO_FLAG_GENERATION) { tprint_struct_next(); PRINT_FIELD_U(args, generation); } if (args.flags & BTRFS_FS_INFO_FLAG_METADATA_UUID) { tprint_struct_next(); PRINT_FIELD_UUID(args, metadata_uuid); } tprint_struct_end(); break; } case BTRFS_IOC_GET_DEV_STATS: { /* RW */ struct btrfs_ioctl_get_dev_stats args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); if (entering(tcp)) { PRINT_FIELD_DEV(args, devid); tprint_struct_next(); } PRINT_FIELD_U(args, nr_items); tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_dev_stats_flags, "BTRFS_DEV_STATS_???"); if (entering(tcp)) { tprint_struct_end(); return 0; } /* * The structure has a 1k limit; Let's make sure we don't * go off into the middle of nowhere with a bad nr_items * value. */ tprint_struct_next(); tprint_array_begin(); for (uint64_t i = 0; i < args.nr_items; ++i) { if (i) tprint_array_next(); if (i >= ARRAY_SIZE(args.values)) { tprint_more_data_follows(); break; } tprint_array_index_begin(); printxval_u(btrfs_dev_stats_values, i, NULL); tprint_array_index_end(); PRINT_VAL_U(args.values[i]); } tprint_array_end(); tprint_struct_end(); break; } case BTRFS_IOC_INO_LOOKUP: { /* RW */ struct btrfs_ioctl_ino_lookup_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { /* Use subvolume id of the containing root */ if (args.treeid == 0) set_tcb_priv_ulong(tcp, 1); tprint_struct_begin(); btrfs_print_objectid(args, treeid); tprint_struct_next(); btrfs_print_objectid(args, objectid); tprint_struct_end(); return 0; } tprint_struct_begin(); if (get_tcb_priv_ulong(tcp)) { btrfs_print_objectid(args, treeid); tprint_struct_next(); } PRINT_FIELD_CSTRING(args, name); tprint_struct_end(); break; } case BTRFS_IOC_INO_PATHS: { /* RW */ struct btrfs_ioctl_ino_path_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_U(args, inum); tprint_struct_next(); PRINT_FIELD_U(args, size); tprint_struct_next(); PRINT_FIELD_ADDR64(args, fspath); tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_OBJ_TCB_VAL(args, fspath, tcp, btrfs_print_ino_path_container); tprint_struct_end(); break; } case BTRFS_IOC_LOGICAL_INO: { /* RW */ struct btrfs_ioctl_logical_ino_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_U(args, logical); tprint_struct_next(); PRINT_FIELD_U(args, size); if (!IS_ARRAY_ZERO(args.reserved)) { tprint_struct_next(); PRINT_FIELD_X_ARRAY(args, reserved); } tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_logical_ino_args_flags, "BTRFS_LOGICAL_INO_ARGS_???"); tprint_struct_next(); PRINT_FIELD_ADDR64(args, inodes); tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_OBJ_TCB_VAL(args, inodes, tcp, btrfs_print_logical_ino_container); tprint_struct_end(); break; } case BTRFS_IOC_QGROUP_ASSIGN: { /* W */ struct btrfs_ioctl_qgroup_assign_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, assign); tprint_struct_next(); PRINT_FIELD_U(args, src); tprint_struct_next(); PRINT_FIELD_U(args, dst); tprint_struct_end(); break; } case BTRFS_IOC_QGROUP_CREATE: { /* W */ struct btrfs_ioctl_qgroup_create_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, create); tprint_struct_next(); PRINT_FIELD_U(args, qgroupid); tprint_struct_end(); break; } case BTRFS_IOC_QGROUP_LIMIT: { /* R */ struct btrfs_ioctl_qgroup_limit_args args; if (entering(tcp)) return 0; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, qgroupid); tprint_struct_next(); PRINT_FIELD_OBJ_PTR(args, lim, btrfs_print_qgroup_limit); tprint_struct_end(); break; } case BTRFS_IOC_QUOTA_CTL: { /* W */ struct btrfs_ioctl_quota_ctl_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_XVAL(args, cmd, btrfs_qgroup_ctl_cmds, "BTRFS_QUOTA_CTL_???"); tprint_struct_end(); break; } case BTRFS_IOC_QUOTA_RESCAN: { /* W */ struct btrfs_ioctl_quota_rescan_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, flags); tprint_struct_end(); break; } case BTRFS_IOC_QUOTA_RESCAN_STATUS: { /* R */ struct btrfs_ioctl_quota_rescan_args args; if (entering(tcp)) return 0; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_U(args, flags); tprint_struct_next(); btrfs_print_objectid(args, progress); tprint_struct_end(); break; } case BTRFS_IOC_SET_RECEIVED_SUBVOL: { /* RW */ struct_btrfs_ioctl_received_subvol_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_UUID(args, uuid); tprint_struct_next(); PRINT_FIELD_U(args, stransid); tprint_struct_next(); PRINT_FIELD_OBJ_PTR(args, stime, print_btrfs_timespec); tprint_struct_next(); PRINT_FIELD_U(args, flags); tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_U(args, rtransid); tprint_struct_next(); PRINT_FIELD_OBJ_PTR(args, rtime, print_btrfs_timespec); tprint_struct_end(); break; } case BTRFS_IOC_SCRUB: /* RW */ case BTRFS_IOC_SCRUB_PROGRESS: { /* RW */ struct btrfs_ioctl_scrub_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_DEV(args, devid); if (code == BTRFS_IOC_SCRUB) { tprint_struct_next(); PRINT_FIELD_U(args, start); tprint_struct_next(); PRINT_FIELD_U64(args, end); tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_scrub_flags, "BTRFS_SCRUB_???"); } tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_OBJ_PTR(args, progress, print_btrfs_scrub_progress); tprint_struct_end(); break; } case BTRFS_IOC_TREE_SEARCH: { /* RW */ struct btrfs_ioctl_search_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args.key)) break; tprint_struct_begin(); PRINT_FIELD_OBJ_PTR(args, key, print_btrfs_ioctl_search_key, entering(tcp), !abbrev(tcp)); decode_search_arg_buf(tcp, arg + offsetof(typeof(args), buf), sizeof(args.buf), args.key.nr_items); tprint_struct_end(); if (entering(tcp)) return 0; break; } case BTRFS_IOC_TREE_SEARCH_V2: { /* RW */ struct btrfs_ioctl_search_args_v2 args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) { if (tcp->u_error == EOVERFLOW) { tprint_value_changed(); if (!umove_or_printaddr_ignore_syserror(tcp, arg, &args)) { tprint_struct_begin(); PRINT_FIELD_U(args, buf_size); tprint_struct_end(); } } break; } else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_OBJ_PTR(args, key, print_btrfs_ioctl_search_key, entering(tcp), !abbrev(tcp)); tprint_struct_next(); PRINT_FIELD_U(args, buf_size); decode_search_arg_buf(tcp, arg + offsetof(typeof(args), buf), args.buf_size, args.key.nr_items); tprint_struct_end(); if (entering(tcp)) return 0; break; } case BTRFS_IOC_SEND: { /* W */ struct_btrfs_ioctl_send_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_FD(args, send_fd, tcp); tprint_struct_next(); PRINT_FIELD_U(args, clone_sources_count); if (abbrev(tcp)) { tprint_struct_next(); PRINT_FIELD_PTR(args, clone_sources); } else { tprint_struct_next(); tprints_field_name("clone_sources"); uint64_t record; print_array(tcp, ptr_to_kulong(args.clone_sources), args.clone_sources_count, &record, sizeof(record), tfetch_mem, print_objectid_callback, 0); } tprint_struct_next(); btrfs_print_objectid(args, parent_root); tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_send_flags, "BTRFS_SEND_FLAGS_???"); tprint_struct_end(); break; } case BTRFS_IOC_SPACE_INFO: { /* RW */ struct btrfs_ioctl_space_args args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_U(args, space_slots); tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_U(args, total_spaces); if (args.space_slots == 0 && args.total_spaces) { tprint_struct_end(); break; } if (abbrev(tcp)) { tprint_struct_next(); tprint_more_data_follows(); } else { struct btrfs_ioctl_space_info info; tprint_struct_next(); tprints_field_name("spaces"); print_array(tcp, arg + offsetof(typeof(args), spaces), args.total_spaces, &info, sizeof(info), tfetch_mem, print_btrfs_ioctl_space_info, 0); } tprint_struct_end(); break; } case BTRFS_IOC_SNAP_CREATE: case BTRFS_IOC_RESIZE: case BTRFS_IOC_SCAN_DEV: case BTRFS_IOC_FORGET_DEV: case BTRFS_IOC_ADD_DEV: case BTRFS_IOC_RM_DEV: case BTRFS_IOC_SUBVOL_CREATE: case BTRFS_IOC_SNAP_DESTROY: case BTRFS_IOC_DEVICES_READY: { /* W */ struct btrfs_ioctl_vol_args args; tprint_arg_next(); if (umove_or_printaddr(tcp, arg, &args)) break; tprint_struct_begin(); PRINT_FIELD_FD(args, fd, tcp); tprint_struct_next(); PRINT_FIELD_CSTRING(args, name); tprint_struct_end(); break; } case BTRFS_IOC_SNAP_CREATE_V2: case BTRFS_IOC_SUBVOL_CREATE_V2: { /* code is W, but is actually RW */ struct_btrfs_ioctl_vol_args_v2 args; if (entering(tcp)) tprint_arg_next(); else if (syserror(tcp)) break; else tprint_value_changed(); if (umove_or_printaddr(tcp, arg, &args)) break; if (entering(tcp)) { tprint_struct_begin(); PRINT_FIELD_FD(args, fd, tcp); tprint_struct_next(); PRINT_FIELD_FLAGS(args, flags, btrfs_snap_flags_v2, "BTRFS_SUBVOL_???"); if (args.flags & BTRFS_SUBVOL_QGROUP_INHERIT) { tprint_struct_next(); PRINT_FIELD_U(args, size); tprint_struct_next(); tprints_field_name("qgroup_inherit"); btrfs_print_qgroup_inherit(tcp, ptr_to_kulong(args.qgroup_inherit)); } tprint_struct_next(); PRINT_FIELD_CSTRING(args, name); tprint_struct_end(); return 0; } tprint_struct_begin(); PRINT_FIELD_U(args, transid); tprint_struct_end(); break; } default: return RVAL_DECODED; }; return RVAL_IOCTL_DECODED; }