|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * linux/fs/ocfs2/ioctl.c | 
|  | * | 
|  | * Copyright (C) 2006 Herbert Poetzl | 
|  | * adapted from Remy Card's ext2/ioctl.c | 
|  | */ | 
|  |  | 
|  | #include <linux/fs.h> | 
|  | #include <linux/mount.h> | 
|  | #include <linux/blkdev.h> | 
|  | #include <linux/compat.h> | 
|  | #include <linux/fileattr.h> | 
|  |  | 
|  | #include <cluster/masklog.h> | 
|  |  | 
|  | #include "ocfs2.h" | 
|  | #include "alloc.h" | 
|  | #include "dlmglue.h" | 
|  | #include "file.h" | 
|  | #include "inode.h" | 
|  | #include "journal.h" | 
|  |  | 
|  | #include "ocfs2_fs.h" | 
|  | #include "ioctl.h" | 
|  | #include "resize.h" | 
|  | #include "refcounttree.h" | 
|  | #include "sysfile.h" | 
|  | #include "dir.h" | 
|  | #include "buffer_head_io.h" | 
|  | #include "suballoc.h" | 
|  | #include "move_extents.h" | 
|  |  | 
|  | #define o2info_from_user(a, b)	\ | 
|  | copy_from_user(&(a), (b), sizeof(a)) | 
|  | #define o2info_to_user(a, b)	\ | 
|  | copy_to_user((typeof(a) __user *)b, &(a), sizeof(a)) | 
|  |  | 
|  | /* | 
|  | * This is just a best-effort to tell userspace that this request | 
|  | * caused the error. | 
|  | */ | 
|  | static inline void o2info_set_request_error(struct ocfs2_info_request *kreq, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | kreq->ir_flags |= OCFS2_INFO_FL_ERROR; | 
|  | (void)put_user(kreq->ir_flags, (__u32 __user *)&(req->ir_flags)); | 
|  | } | 
|  |  | 
|  | static inline void o2info_set_request_filled(struct ocfs2_info_request *req) | 
|  | { | 
|  | req->ir_flags |= OCFS2_INFO_FL_FILLED; | 
|  | } | 
|  |  | 
|  | static inline void o2info_clear_request_filled(struct ocfs2_info_request *req) | 
|  | { | 
|  | req->ir_flags &= ~OCFS2_INFO_FL_FILLED; | 
|  | } | 
|  |  | 
|  | static inline int o2info_coherent(struct ocfs2_info_request *req) | 
|  | { | 
|  | return (!(req->ir_flags & OCFS2_INFO_FL_NON_COHERENT)); | 
|  | } | 
|  |  | 
|  | int ocfs2_fileattr_get(struct dentry *dentry, struct file_kattr *fa) | 
|  | { | 
|  | struct inode *inode = d_inode(dentry); | 
|  | unsigned int flags; | 
|  | int status; | 
|  |  | 
|  | status = ocfs2_inode_lock(inode, NULL, 0); | 
|  | if (status < 0) { | 
|  | mlog_errno(status); | 
|  | return status; | 
|  | } | 
|  | ocfs2_get_inode_flags(OCFS2_I(inode)); | 
|  | flags = OCFS2_I(inode)->ip_attr; | 
|  | ocfs2_inode_unlock(inode, 0); | 
|  |  | 
|  | fileattr_fill_flags(fa, flags & OCFS2_FL_VISIBLE); | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | int ocfs2_fileattr_set(struct mnt_idmap *idmap, | 
|  | struct dentry *dentry, struct file_kattr *fa) | 
|  | { | 
|  | struct inode *inode = d_inode(dentry); | 
|  | unsigned int flags = fa->flags; | 
|  | struct ocfs2_inode_info *ocfs2_inode = OCFS2_I(inode); | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  | handle_t *handle = NULL; | 
|  | struct buffer_head *bh = NULL; | 
|  | unsigned oldflags; | 
|  | int status; | 
|  |  | 
|  | if (fileattr_has_fsx(fa)) | 
|  | return -EOPNOTSUPP; | 
|  |  | 
|  | status = ocfs2_inode_lock(inode, &bh, 1); | 
|  | if (status < 0) { | 
|  | mlog_errno(status); | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | if (!S_ISDIR(inode->i_mode)) | 
|  | flags &= ~OCFS2_DIRSYNC_FL; | 
|  |  | 
|  | oldflags = ocfs2_inode->ip_attr; | 
|  | flags = flags & OCFS2_FL_MODIFIABLE; | 
|  | flags |= oldflags & ~OCFS2_FL_MODIFIABLE; | 
|  |  | 
|  | /* Check already done by VFS, but repeat with ocfs lock */ | 
|  | status = -EPERM; | 
|  | if ((flags ^ oldflags) & (FS_APPEND_FL | FS_IMMUTABLE_FL) && | 
|  | !capable(CAP_LINUX_IMMUTABLE)) | 
|  | goto bail_unlock; | 
|  |  | 
|  | handle = ocfs2_start_trans(osb, OCFS2_INODE_UPDATE_CREDITS); | 
|  | if (IS_ERR(handle)) { | 
|  | status = PTR_ERR(handle); | 
|  | mlog_errno(status); | 
|  | goto bail_unlock; | 
|  | } | 
|  |  | 
|  | ocfs2_inode->ip_attr = flags; | 
|  | ocfs2_set_inode_flags(inode); | 
|  | inode_set_ctime_current(inode); | 
|  |  | 
|  | status = ocfs2_mark_inode_dirty(handle, inode, bh); | 
|  | if (status < 0) | 
|  | mlog_errno(status); | 
|  |  | 
|  | ocfs2_commit_trans(osb, handle); | 
|  |  | 
|  | bail_unlock: | 
|  | ocfs2_inode_unlock(inode, 1); | 
|  | bail: | 
|  | brelse(bh); | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_blocksize(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_blocksize oib; | 
|  |  | 
|  | if (o2info_from_user(oib, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | oib.ib_blocksize = inode->i_sb->s_blocksize; | 
|  |  | 
|  | o2info_set_request_filled(&oib.ib_req); | 
|  |  | 
|  | if (o2info_to_user(oib, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_clustersize(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_clustersize oic; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  |  | 
|  | if (o2info_from_user(oic, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | oic.ic_clustersize = osb->s_clustersize; | 
|  |  | 
|  | o2info_set_request_filled(&oic.ic_req); | 
|  |  | 
|  | if (o2info_to_user(oic, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_maxslots(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_maxslots oim; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  |  | 
|  | if (o2info_from_user(oim, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | oim.im_max_slots = osb->max_slots; | 
|  |  | 
|  | o2info_set_request_filled(&oim.im_req); | 
|  |  | 
|  | if (o2info_to_user(oim, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_label(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_label oil; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  |  | 
|  | if (o2info_from_user(oil, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | memcpy(oil.il_label, osb->vol_label, OCFS2_MAX_VOL_LABEL_LEN); | 
|  |  | 
|  | o2info_set_request_filled(&oil.il_req); | 
|  |  | 
|  | if (o2info_to_user(oil, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_uuid(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_uuid oiu; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  |  | 
|  | if (o2info_from_user(oiu, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | memcpy(oiu.iu_uuid_str, osb->uuid_str, OCFS2_TEXT_UUID_LEN + 1); | 
|  |  | 
|  | o2info_set_request_filled(&oiu.iu_req); | 
|  |  | 
|  | if (o2info_to_user(oiu, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_fs_features(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_fs_features oif; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  |  | 
|  | if (o2info_from_user(oif, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | oif.if_compat_features = osb->s_feature_compat; | 
|  | oif.if_incompat_features = osb->s_feature_incompat; | 
|  | oif.if_ro_compat_features = osb->s_feature_ro_compat; | 
|  |  | 
|  | o2info_set_request_filled(&oif.if_req); | 
|  |  | 
|  | if (o2info_to_user(oif, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_journal_size(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_journal_size oij; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  |  | 
|  | if (o2info_from_user(oij, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | oij.ij_journal_size = i_size_read(osb->journal->j_inode); | 
|  |  | 
|  | o2info_set_request_filled(&oij.ij_req); | 
|  |  | 
|  | if (o2info_to_user(oij, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_scan_inode_alloc(struct ocfs2_super *osb, | 
|  | struct inode *inode_alloc, u64 blkno, | 
|  | struct ocfs2_info_freeinode *fi, | 
|  | u32 slot) | 
|  | { | 
|  | int status = 0, unlock = 0; | 
|  |  | 
|  | struct buffer_head *bh = NULL; | 
|  | struct ocfs2_dinode *dinode_alloc = NULL; | 
|  |  | 
|  | if (inode_alloc) | 
|  | inode_lock(inode_alloc); | 
|  |  | 
|  | if (inode_alloc && o2info_coherent(&fi->ifi_req)) { | 
|  | status = ocfs2_inode_lock(inode_alloc, &bh, 0); | 
|  | if (status < 0) { | 
|  | mlog_errno(status); | 
|  | goto bail; | 
|  | } | 
|  | unlock = 1; | 
|  | } else { | 
|  | status = ocfs2_read_blocks_sync(osb, blkno, 1, &bh); | 
|  | if (status < 0) { | 
|  | mlog_errno(status); | 
|  | goto bail; | 
|  | } | 
|  | } | 
|  |  | 
|  | dinode_alloc = (struct ocfs2_dinode *)bh->b_data; | 
|  |  | 
|  | fi->ifi_stat[slot].lfi_total = | 
|  | le32_to_cpu(dinode_alloc->id1.bitmap1.i_total); | 
|  | fi->ifi_stat[slot].lfi_free = | 
|  | le32_to_cpu(dinode_alloc->id1.bitmap1.i_total) - | 
|  | le32_to_cpu(dinode_alloc->id1.bitmap1.i_used); | 
|  |  | 
|  | bail: | 
|  | if (unlock) | 
|  | ocfs2_inode_unlock(inode_alloc, 0); | 
|  |  | 
|  | if (inode_alloc) | 
|  | inode_unlock(inode_alloc); | 
|  |  | 
|  | brelse(bh); | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_freeinode(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | u32 i; | 
|  | u64 blkno = -1; | 
|  | char namebuf[40]; | 
|  | int status, type = INODE_ALLOC_SYSTEM_INODE; | 
|  | struct ocfs2_info_freeinode *oifi = NULL; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  | struct inode *inode_alloc = NULL; | 
|  |  | 
|  | oifi = kzalloc(sizeof(struct ocfs2_info_freeinode), GFP_KERNEL); | 
|  | if (!oifi) { | 
|  | status = -ENOMEM; | 
|  | mlog_errno(status); | 
|  | goto out_err; | 
|  | } | 
|  |  | 
|  | if (o2info_from_user(*oifi, req)) { | 
|  | status = -EFAULT; | 
|  | goto out_free; | 
|  | } | 
|  |  | 
|  | oifi->ifi_slotnum = osb->max_slots; | 
|  |  | 
|  | for (i = 0; i < oifi->ifi_slotnum; i++) { | 
|  | if (o2info_coherent(&oifi->ifi_req)) { | 
|  | inode_alloc = ocfs2_get_system_file_inode(osb, type, i); | 
|  | if (!inode_alloc) { | 
|  | mlog(ML_ERROR, "unable to get alloc inode in " | 
|  | "slot %u\n", i); | 
|  | status = -EIO; | 
|  | goto bail; | 
|  | } | 
|  | } else { | 
|  | int len = ocfs2_sprintf_system_inode_name(namebuf, | 
|  | sizeof(namebuf), | 
|  | type, i); | 
|  | status = ocfs2_lookup_ino_from_name(osb->sys_root_inode, | 
|  | namebuf, len, &blkno); | 
|  | if (status < 0) { | 
|  | status = -ENOENT; | 
|  | goto bail; | 
|  | } | 
|  | } | 
|  |  | 
|  | status = ocfs2_info_scan_inode_alloc(osb, inode_alloc, blkno, oifi, i); | 
|  |  | 
|  | iput(inode_alloc); | 
|  | inode_alloc = NULL; | 
|  |  | 
|  | if (status < 0) | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | o2info_set_request_filled(&oifi->ifi_req); | 
|  |  | 
|  | if (o2info_to_user(*oifi, req)) { | 
|  | status = -EFAULT; | 
|  | goto out_free; | 
|  | } | 
|  |  | 
|  | status = 0; | 
|  | bail: | 
|  | if (status) | 
|  | o2info_set_request_error(&oifi->ifi_req, req); | 
|  | out_free: | 
|  | kfree(oifi); | 
|  | out_err: | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static void o2ffg_update_histogram(struct ocfs2_info_free_chunk_list *hist, | 
|  | unsigned int chunksize) | 
|  | { | 
|  | u32 index; | 
|  |  | 
|  | index = __ilog2_u32(chunksize); | 
|  | if (index >= OCFS2_INFO_MAX_HIST) | 
|  | index = OCFS2_INFO_MAX_HIST - 1; | 
|  |  | 
|  | hist->fc_chunks[index]++; | 
|  | hist->fc_clusters[index] += chunksize; | 
|  | } | 
|  |  | 
|  | static void o2ffg_update_stats(struct ocfs2_info_freefrag_stats *stats, | 
|  | unsigned int chunksize) | 
|  | { | 
|  | if (chunksize > stats->ffs_max) | 
|  | stats->ffs_max = chunksize; | 
|  |  | 
|  | if (chunksize < stats->ffs_min) | 
|  | stats->ffs_min = chunksize; | 
|  |  | 
|  | stats->ffs_avg += chunksize; | 
|  | stats->ffs_free_chunks_real++; | 
|  | } | 
|  |  | 
|  | static void ocfs2_info_update_ffg(struct ocfs2_info_freefrag *ffg, | 
|  | unsigned int chunksize) | 
|  | { | 
|  | o2ffg_update_histogram(&(ffg->iff_ffs.ffs_fc_hist), chunksize); | 
|  | o2ffg_update_stats(&(ffg->iff_ffs), chunksize); | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_freefrag_scan_chain(struct ocfs2_super *osb, | 
|  | struct inode *gb_inode, | 
|  | struct ocfs2_dinode *gb_dinode, | 
|  | struct ocfs2_chain_rec *rec, | 
|  | struct ocfs2_info_freefrag *ffg, | 
|  | u32 chunks_in_group) | 
|  | { | 
|  | int status = 0, used; | 
|  | u64 blkno; | 
|  |  | 
|  | struct buffer_head *bh = NULL; | 
|  | struct ocfs2_group_desc *bg = NULL; | 
|  |  | 
|  | unsigned int max_bits, num_clusters; | 
|  | unsigned int offset = 0, cluster, chunk; | 
|  | unsigned int chunk_free, last_chunksize = 0; | 
|  |  | 
|  | if (!le32_to_cpu(rec->c_free)) | 
|  | goto bail; | 
|  |  | 
|  | do { | 
|  | if (!bg) | 
|  | blkno = le64_to_cpu(rec->c_blkno); | 
|  | else | 
|  | blkno = le64_to_cpu(bg->bg_next_group); | 
|  |  | 
|  | if (bh) { | 
|  | brelse(bh); | 
|  | bh = NULL; | 
|  | } | 
|  |  | 
|  | if (o2info_coherent(&ffg->iff_req)) | 
|  | status = ocfs2_read_group_descriptor(gb_inode, | 
|  | gb_dinode, | 
|  | blkno, &bh); | 
|  | else | 
|  | status = ocfs2_read_blocks_sync(osb, blkno, 1, &bh); | 
|  |  | 
|  | if (status < 0) { | 
|  | mlog(ML_ERROR, "Can't read the group descriptor # " | 
|  | "%llu from device.", (unsigned long long)blkno); | 
|  | status = -EIO; | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | bg = (struct ocfs2_group_desc *)bh->b_data; | 
|  |  | 
|  | if (!le16_to_cpu(bg->bg_free_bits_count)) | 
|  | continue; | 
|  |  | 
|  | max_bits = le16_to_cpu(bg->bg_bits); | 
|  | offset = 0; | 
|  |  | 
|  | for (chunk = 0; chunk < chunks_in_group; chunk++) { | 
|  | /* | 
|  | * last chunk may be not an entire one. | 
|  | */ | 
|  | if ((offset + ffg->iff_chunksize) > max_bits) | 
|  | num_clusters = max_bits - offset; | 
|  | else | 
|  | num_clusters = ffg->iff_chunksize; | 
|  |  | 
|  | chunk_free = 0; | 
|  | for (cluster = 0; cluster < num_clusters; cluster++) { | 
|  | used = ocfs2_test_bit(offset, | 
|  | (unsigned long *)bg->bg_bitmap); | 
|  | /* | 
|  | * - chunk_free counts free clusters in #N chunk. | 
|  | * - last_chunksize records the size(in) clusters | 
|  | *   for the last real free chunk being counted. | 
|  | */ | 
|  | if (!used) { | 
|  | last_chunksize++; | 
|  | chunk_free++; | 
|  | } | 
|  |  | 
|  | if (used && last_chunksize) { | 
|  | ocfs2_info_update_ffg(ffg, | 
|  | last_chunksize); | 
|  | last_chunksize = 0; | 
|  | } | 
|  |  | 
|  | offset++; | 
|  | } | 
|  |  | 
|  | if (chunk_free == ffg->iff_chunksize) | 
|  | ffg->iff_ffs.ffs_free_chunks++; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * need to update the info for last free chunk. | 
|  | */ | 
|  | if (last_chunksize) | 
|  | ocfs2_info_update_ffg(ffg, last_chunksize); | 
|  |  | 
|  | } while (le64_to_cpu(bg->bg_next_group)); | 
|  |  | 
|  | bail: | 
|  | brelse(bh); | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_freefrag_scan_bitmap(struct ocfs2_super *osb, | 
|  | struct inode *gb_inode, u64 blkno, | 
|  | struct ocfs2_info_freefrag *ffg) | 
|  | { | 
|  | u32 chunks_in_group; | 
|  | int status = 0, unlock = 0, i; | 
|  |  | 
|  | struct buffer_head *bh = NULL; | 
|  | struct ocfs2_chain_list *cl = NULL; | 
|  | struct ocfs2_chain_rec *rec = NULL; | 
|  | struct ocfs2_dinode *gb_dinode = NULL; | 
|  |  | 
|  | if (gb_inode) | 
|  | inode_lock(gb_inode); | 
|  |  | 
|  | if (o2info_coherent(&ffg->iff_req)) { | 
|  | status = ocfs2_inode_lock(gb_inode, &bh, 0); | 
|  | if (status < 0) { | 
|  | mlog_errno(status); | 
|  | goto bail; | 
|  | } | 
|  | unlock = 1; | 
|  | } else { | 
|  | status = ocfs2_read_blocks_sync(osb, blkno, 1, &bh); | 
|  | if (status < 0) { | 
|  | mlog_errno(status); | 
|  | goto bail; | 
|  | } | 
|  | } | 
|  |  | 
|  | gb_dinode = (struct ocfs2_dinode *)bh->b_data; | 
|  | cl = &(gb_dinode->id2.i_chain); | 
|  |  | 
|  | /* | 
|  | * Chunksize(in) clusters from userspace should be | 
|  | * less than clusters in a group. | 
|  | */ | 
|  | if (ffg->iff_chunksize > le16_to_cpu(cl->cl_cpg)) { | 
|  | status = -EINVAL; | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | memset(&ffg->iff_ffs, 0, sizeof(struct ocfs2_info_freefrag_stats)); | 
|  |  | 
|  | ffg->iff_ffs.ffs_min = ~0U; | 
|  | ffg->iff_ffs.ffs_clusters = | 
|  | le32_to_cpu(gb_dinode->id1.bitmap1.i_total); | 
|  | ffg->iff_ffs.ffs_free_clusters = ffg->iff_ffs.ffs_clusters - | 
|  | le32_to_cpu(gb_dinode->id1.bitmap1.i_used); | 
|  |  | 
|  | chunks_in_group = le16_to_cpu(cl->cl_cpg) / ffg->iff_chunksize + 1; | 
|  |  | 
|  | for (i = 0; i < le16_to_cpu(cl->cl_next_free_rec); i++) { | 
|  | rec = &(cl->cl_recs[i]); | 
|  | status = ocfs2_info_freefrag_scan_chain(osb, gb_inode, | 
|  | gb_dinode, | 
|  | rec, ffg, | 
|  | chunks_in_group); | 
|  | if (status) | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | if (ffg->iff_ffs.ffs_free_chunks_real) | 
|  | ffg->iff_ffs.ffs_avg = (ffg->iff_ffs.ffs_avg / | 
|  | ffg->iff_ffs.ffs_free_chunks_real); | 
|  | bail: | 
|  | if (unlock) | 
|  | ocfs2_inode_unlock(gb_inode, 0); | 
|  |  | 
|  | if (gb_inode) | 
|  | inode_unlock(gb_inode); | 
|  |  | 
|  | iput(gb_inode); | 
|  | brelse(bh); | 
|  |  | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_freefrag(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | u64 blkno = -1; | 
|  | char namebuf[40]; | 
|  | int status, type = GLOBAL_BITMAP_SYSTEM_INODE; | 
|  |  | 
|  | struct ocfs2_info_freefrag *oiff; | 
|  | struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); | 
|  | struct inode *gb_inode = NULL; | 
|  |  | 
|  | oiff = kzalloc(sizeof(struct ocfs2_info_freefrag), GFP_KERNEL); | 
|  | if (!oiff) { | 
|  | status = -ENOMEM; | 
|  | mlog_errno(status); | 
|  | goto out_err; | 
|  | } | 
|  |  | 
|  | if (o2info_from_user(*oiff, req)) { | 
|  | status = -EFAULT; | 
|  | goto out_free; | 
|  | } | 
|  | /* | 
|  | * chunksize from userspace should be power of 2. | 
|  | */ | 
|  | if ((oiff->iff_chunksize & (oiff->iff_chunksize - 1)) || | 
|  | (!oiff->iff_chunksize)) { | 
|  | status = -EINVAL; | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | if (o2info_coherent(&oiff->iff_req)) { | 
|  | gb_inode = ocfs2_get_system_file_inode(osb, type, | 
|  | OCFS2_INVALID_SLOT); | 
|  | if (!gb_inode) { | 
|  | mlog(ML_ERROR, "unable to get global_bitmap inode\n"); | 
|  | status = -EIO; | 
|  | goto bail; | 
|  | } | 
|  | } else { | 
|  | int len = ocfs2_sprintf_system_inode_name(namebuf, sizeof(namebuf), | 
|  | type, OCFS2_INVALID_SLOT); | 
|  | status = ocfs2_lookup_ino_from_name(osb->sys_root_inode, | 
|  | namebuf, len, &blkno); | 
|  | if (status < 0) { | 
|  | status = -ENOENT; | 
|  | goto bail; | 
|  | } | 
|  | } | 
|  |  | 
|  | status = ocfs2_info_freefrag_scan_bitmap(osb, gb_inode, blkno, oiff); | 
|  | if (status < 0) | 
|  | goto bail; | 
|  |  | 
|  | o2info_set_request_filled(&oiff->iff_req); | 
|  |  | 
|  | if (o2info_to_user(*oiff, req)) { | 
|  | status = -EFAULT; | 
|  | goto out_free; | 
|  | } | 
|  |  | 
|  | status = 0; | 
|  | bail: | 
|  | if (status) | 
|  | o2info_set_request_error(&oiff->iff_req, req); | 
|  | out_free: | 
|  | kfree(oiff); | 
|  | out_err: | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int ocfs2_info_handle_unknown(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | struct ocfs2_info_request oir; | 
|  |  | 
|  | if (o2info_from_user(oir, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | o2info_clear_request_filled(&oir); | 
|  |  | 
|  | if (o2info_to_user(oir, req)) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Validate and distinguish OCFS2_IOC_INFO requests. | 
|  | * | 
|  | * - validate the magic number. | 
|  | * - distinguish different requests. | 
|  | * - validate size of different requests. | 
|  | */ | 
|  | static int ocfs2_info_handle_request(struct inode *inode, | 
|  | struct ocfs2_info_request __user *req) | 
|  | { | 
|  | int status = -EFAULT; | 
|  | struct ocfs2_info_request oir; | 
|  |  | 
|  | if (o2info_from_user(oir, req)) | 
|  | goto bail; | 
|  |  | 
|  | status = -EINVAL; | 
|  | if (oir.ir_magic != OCFS2_INFO_MAGIC) | 
|  | goto bail; | 
|  |  | 
|  | switch (oir.ir_code) { | 
|  | case OCFS2_INFO_BLOCKSIZE: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_blocksize)) | 
|  | status = ocfs2_info_handle_blocksize(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_CLUSTERSIZE: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_clustersize)) | 
|  | status = ocfs2_info_handle_clustersize(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_MAXSLOTS: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_maxslots)) | 
|  | status = ocfs2_info_handle_maxslots(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_LABEL: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_label)) | 
|  | status = ocfs2_info_handle_label(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_UUID: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_uuid)) | 
|  | status = ocfs2_info_handle_uuid(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_FS_FEATURES: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_fs_features)) | 
|  | status = ocfs2_info_handle_fs_features(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_JOURNAL_SIZE: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_journal_size)) | 
|  | status = ocfs2_info_handle_journal_size(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_FREEINODE: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_freeinode)) | 
|  | status = ocfs2_info_handle_freeinode(inode, req); | 
|  | break; | 
|  | case OCFS2_INFO_FREEFRAG: | 
|  | if (oir.ir_size == sizeof(struct ocfs2_info_freefrag)) | 
|  | status = ocfs2_info_handle_freefrag(inode, req); | 
|  | break; | 
|  | default: | 
|  | status = ocfs2_info_handle_unknown(inode, req); | 
|  | break; | 
|  | } | 
|  |  | 
|  | bail: | 
|  | return status; | 
|  | } | 
|  |  | 
|  | static int ocfs2_get_request_ptr(struct ocfs2_info *info, int idx, | 
|  | u64 *req_addr, int compat_flag) | 
|  | { | 
|  | int status = -EFAULT; | 
|  | u64 __user *bp = NULL; | 
|  |  | 
|  | if (compat_flag) { | 
|  | #ifdef CONFIG_COMPAT | 
|  | /* | 
|  | * pointer bp stores the base address of a pointers array, | 
|  | * which collects all addresses of separate request. | 
|  | */ | 
|  | bp = (u64 __user *)(unsigned long)compat_ptr(info->oi_requests); | 
|  | #else | 
|  | BUG(); | 
|  | #endif | 
|  | } else | 
|  | bp = (u64 __user *)(unsigned long)(info->oi_requests); | 
|  |  | 
|  | if (o2info_from_user(*req_addr, bp + idx)) | 
|  | goto bail; | 
|  |  | 
|  | status = 0; | 
|  | bail: | 
|  | return status; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * OCFS2_IOC_INFO handles an array of requests passed from userspace. | 
|  | * | 
|  | * ocfs2_info_handle() receives a large info aggregation, grab and | 
|  | * validate the request count from header, then break it into small | 
|  | * pieces, later specific handlers can handle them one by one. | 
|  | * | 
|  | * Idea here is to make each separate request small enough to ensure | 
|  | * a better backward&forward compatibility, since a small piece of | 
|  | * request will be less likely to be broken if disk layout get changed. | 
|  | */ | 
|  | static noinline_for_stack int | 
|  | ocfs2_info_handle(struct inode *inode, struct ocfs2_info *info, int compat_flag) | 
|  | { | 
|  | int i, status = 0; | 
|  | u64 req_addr; | 
|  | struct ocfs2_info_request __user *reqp; | 
|  |  | 
|  | if ((info->oi_count > OCFS2_INFO_MAX_REQUEST) || | 
|  | (!info->oi_requests)) { | 
|  | status = -EINVAL; | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | for (i = 0; i < info->oi_count; i++) { | 
|  |  | 
|  | status = ocfs2_get_request_ptr(info, i, &req_addr, compat_flag); | 
|  | if (status) | 
|  | break; | 
|  |  | 
|  | reqp = (struct ocfs2_info_request __user *)(unsigned long)req_addr; | 
|  | if (!reqp) { | 
|  | status = -EINVAL; | 
|  | goto bail; | 
|  | } | 
|  |  | 
|  | status = ocfs2_info_handle_request(inode, reqp); | 
|  | if (status) | 
|  | break; | 
|  | } | 
|  |  | 
|  | bail: | 
|  | return status; | 
|  | } | 
|  |  | 
|  | long ocfs2_ioctl(struct file *filp, unsigned int cmd, unsigned long arg) | 
|  | { | 
|  | struct inode *inode = file_inode(filp); | 
|  | void __user *argp = (void __user *)arg; | 
|  | int status; | 
|  |  | 
|  | switch (cmd) { | 
|  | case OCFS2_IOC_RESVSP: | 
|  | case OCFS2_IOC_RESVSP64: | 
|  | case OCFS2_IOC_UNRESVSP: | 
|  | case OCFS2_IOC_UNRESVSP64: | 
|  | { | 
|  | struct ocfs2_space_resv sr; | 
|  |  | 
|  | if (copy_from_user(&sr, (int __user *) arg, sizeof(sr))) | 
|  | return -EFAULT; | 
|  |  | 
|  | return ocfs2_change_file_space(filp, cmd, &sr); | 
|  | } | 
|  | case OCFS2_IOC_GROUP_EXTEND: | 
|  | { | 
|  | int new_clusters; | 
|  |  | 
|  | if (!capable(CAP_SYS_RESOURCE)) | 
|  | return -EPERM; | 
|  |  | 
|  | if (get_user(new_clusters, (int __user *)arg)) | 
|  | return -EFAULT; | 
|  |  | 
|  | status = mnt_want_write_file(filp); | 
|  | if (status) | 
|  | return status; | 
|  | status = ocfs2_group_extend(inode, new_clusters); | 
|  | mnt_drop_write_file(filp); | 
|  | return status; | 
|  | } | 
|  | case OCFS2_IOC_GROUP_ADD: | 
|  | case OCFS2_IOC_GROUP_ADD64: | 
|  | { | 
|  | struct ocfs2_new_group_input input; | 
|  |  | 
|  | if (!capable(CAP_SYS_RESOURCE)) | 
|  | return -EPERM; | 
|  |  | 
|  | if (copy_from_user(&input, (int __user *) arg, sizeof(input))) | 
|  | return -EFAULT; | 
|  |  | 
|  | status = mnt_want_write_file(filp); | 
|  | if (status) | 
|  | return status; | 
|  | status = ocfs2_group_add(inode, &input); | 
|  | mnt_drop_write_file(filp); | 
|  | return status; | 
|  | } | 
|  | case OCFS2_IOC_REFLINK: | 
|  | { | 
|  | struct reflink_arguments args; | 
|  | const char __user *old_path; | 
|  | const char __user *new_path; | 
|  | bool preserve; | 
|  |  | 
|  | if (copy_from_user(&args, argp, sizeof(args))) | 
|  | return -EFAULT; | 
|  | old_path = (const char __user *)(unsigned long)args.old_path; | 
|  | new_path = (const char __user *)(unsigned long)args.new_path; | 
|  | preserve = (args.preserve != 0); | 
|  |  | 
|  | return ocfs2_reflink_ioctl(inode, old_path, new_path, preserve); | 
|  | } | 
|  | case OCFS2_IOC_INFO: | 
|  | { | 
|  | struct ocfs2_info info; | 
|  |  | 
|  | if (copy_from_user(&info, argp, sizeof(struct ocfs2_info))) | 
|  | return -EFAULT; | 
|  |  | 
|  | return ocfs2_info_handle(inode, &info, 0); | 
|  | } | 
|  | case FITRIM: | 
|  | { | 
|  | struct super_block *sb = inode->i_sb; | 
|  | struct fstrim_range range; | 
|  | int ret = 0; | 
|  |  | 
|  | if (!capable(CAP_SYS_ADMIN)) | 
|  | return -EPERM; | 
|  |  | 
|  | if (!bdev_max_discard_sectors(sb->s_bdev)) | 
|  | return -EOPNOTSUPP; | 
|  |  | 
|  | if (copy_from_user(&range, argp, sizeof(range))) | 
|  | return -EFAULT; | 
|  |  | 
|  | range.minlen = max_t(u64, bdev_discard_granularity(sb->s_bdev), | 
|  | range.minlen); | 
|  | ret = ocfs2_trim_fs(sb, &range); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (copy_to_user(argp, &range, sizeof(range))) | 
|  | return -EFAULT; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | case OCFS2_IOC_MOVE_EXT: | 
|  | return ocfs2_ioctl_move_extents(filp, argp); | 
|  | default: | 
|  | return -ENOTTY; | 
|  | } | 
|  | } | 
|  |  | 
|  | #ifdef CONFIG_COMPAT | 
|  | long ocfs2_compat_ioctl(struct file *file, unsigned cmd, unsigned long arg) | 
|  | { | 
|  | bool preserve; | 
|  | struct reflink_arguments args; | 
|  | struct inode *inode = file_inode(file); | 
|  | struct ocfs2_info info; | 
|  | void __user *argp = (void __user *)arg; | 
|  |  | 
|  | switch (cmd) { | 
|  | case OCFS2_IOC_RESVSP: | 
|  | case OCFS2_IOC_RESVSP64: | 
|  | case OCFS2_IOC_UNRESVSP: | 
|  | case OCFS2_IOC_UNRESVSP64: | 
|  | case OCFS2_IOC_GROUP_EXTEND: | 
|  | case OCFS2_IOC_GROUP_ADD: | 
|  | case OCFS2_IOC_GROUP_ADD64: | 
|  | break; | 
|  | case OCFS2_IOC_REFLINK: | 
|  | if (copy_from_user(&args, argp, sizeof(args))) | 
|  | return -EFAULT; | 
|  | preserve = (args.preserve != 0); | 
|  |  | 
|  | return ocfs2_reflink_ioctl(inode, compat_ptr(args.old_path), | 
|  | compat_ptr(args.new_path), preserve); | 
|  | case OCFS2_IOC_INFO: | 
|  | if (copy_from_user(&info, argp, sizeof(struct ocfs2_info))) | 
|  | return -EFAULT; | 
|  |  | 
|  | return ocfs2_info_handle(inode, &info, 1); | 
|  | case FITRIM: | 
|  | case OCFS2_IOC_MOVE_EXT: | 
|  | break; | 
|  | default: | 
|  | return -ENOIOCTLCMD; | 
|  | } | 
|  |  | 
|  | return ocfs2_ioctl(file, cmd, arg); | 
|  | } | 
|  | #endif |