162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * Copyright 2000 by Hans Reiser, licensing governed by reiserfs/README 362306a36Sopenharmony_ci */ 462306a36Sopenharmony_ci 562306a36Sopenharmony_ci#include <linux/capability.h> 662306a36Sopenharmony_ci#include <linux/fs.h> 762306a36Sopenharmony_ci#include <linux/mount.h> 862306a36Sopenharmony_ci#include "reiserfs.h" 962306a36Sopenharmony_ci#include <linux/time.h> 1062306a36Sopenharmony_ci#include <linux/uaccess.h> 1162306a36Sopenharmony_ci#include <linux/pagemap.h> 1262306a36Sopenharmony_ci#include <linux/compat.h> 1362306a36Sopenharmony_ci#include <linux/fileattr.h> 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ciint reiserfs_fileattr_get(struct dentry *dentry, struct fileattr *fa) 1662306a36Sopenharmony_ci{ 1762306a36Sopenharmony_ci struct inode *inode = d_inode(dentry); 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci if (!reiserfs_attrs(inode->i_sb)) 2062306a36Sopenharmony_ci return -ENOTTY; 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci fileattr_fill_flags(fa, REISERFS_I(inode)->i_attrs); 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci return 0; 2562306a36Sopenharmony_ci} 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ciint reiserfs_fileattr_set(struct mnt_idmap *idmap, 2862306a36Sopenharmony_ci struct dentry *dentry, struct fileattr *fa) 2962306a36Sopenharmony_ci{ 3062306a36Sopenharmony_ci struct inode *inode = d_inode(dentry); 3162306a36Sopenharmony_ci unsigned int flags = fa->flags; 3262306a36Sopenharmony_ci int err; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci reiserfs_write_lock(inode->i_sb); 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci err = -ENOTTY; 3762306a36Sopenharmony_ci if (!reiserfs_attrs(inode->i_sb)) 3862306a36Sopenharmony_ci goto unlock; 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci err = -EOPNOTSUPP; 4162306a36Sopenharmony_ci if (fileattr_has_fsx(fa)) 4262306a36Sopenharmony_ci goto unlock; 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci /* 4562306a36Sopenharmony_ci * Is it quota file? Do not allow user to mess with it 4662306a36Sopenharmony_ci */ 4762306a36Sopenharmony_ci err = -EPERM; 4862306a36Sopenharmony_ci if (IS_NOQUOTA(inode)) 4962306a36Sopenharmony_ci goto unlock; 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci if ((flags & REISERFS_NOTAIL_FL) && S_ISREG(inode->i_mode)) { 5262306a36Sopenharmony_ci err = reiserfs_unpack(inode); 5362306a36Sopenharmony_ci if (err) 5462306a36Sopenharmony_ci goto unlock; 5562306a36Sopenharmony_ci } 5662306a36Sopenharmony_ci sd_attrs_to_i_attrs(flags, inode); 5762306a36Sopenharmony_ci REISERFS_I(inode)->i_attrs = flags; 5862306a36Sopenharmony_ci inode_set_ctime_current(inode); 5962306a36Sopenharmony_ci mark_inode_dirty(inode); 6062306a36Sopenharmony_ci err = 0; 6162306a36Sopenharmony_ciunlock: 6262306a36Sopenharmony_ci reiserfs_write_unlock(inode->i_sb); 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci return err; 6562306a36Sopenharmony_ci} 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci/* 6862306a36Sopenharmony_ci * reiserfs_ioctl - handler for ioctl for inode 6962306a36Sopenharmony_ci * supported commands: 7062306a36Sopenharmony_ci * 1) REISERFS_IOC_UNPACK - try to unpack tail from direct item into indirect 7162306a36Sopenharmony_ci * and prevent packing file (argument arg has t 7262306a36Sopenharmony_ci * be non-zero) 7362306a36Sopenharmony_ci * 2) REISERFS_IOC_[GS]ETFLAGS, REISERFS_IOC_[GS]ETVERSION 7462306a36Sopenharmony_ci * 3) That's all for a while ... 7562306a36Sopenharmony_ci */ 7662306a36Sopenharmony_cilong reiserfs_ioctl(struct file *filp, unsigned int cmd, unsigned long arg) 7762306a36Sopenharmony_ci{ 7862306a36Sopenharmony_ci struct inode *inode = file_inode(filp); 7962306a36Sopenharmony_ci int err = 0; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci reiserfs_write_lock(inode->i_sb); 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci switch (cmd) { 8462306a36Sopenharmony_ci case REISERFS_IOC_UNPACK: 8562306a36Sopenharmony_ci if (S_ISREG(inode->i_mode)) { 8662306a36Sopenharmony_ci if (arg) 8762306a36Sopenharmony_ci err = reiserfs_unpack(inode); 8862306a36Sopenharmony_ci } else 8962306a36Sopenharmony_ci err = -ENOTTY; 9062306a36Sopenharmony_ci break; 9162306a36Sopenharmony_ci /* 9262306a36Sopenharmony_ci * following two cases are taken from fs/ext2/ioctl.c by Remy 9362306a36Sopenharmony_ci * Card (card@masi.ibp.fr) 9462306a36Sopenharmony_ci */ 9562306a36Sopenharmony_ci case REISERFS_IOC_GETVERSION: 9662306a36Sopenharmony_ci err = put_user(inode->i_generation, (int __user *)arg); 9762306a36Sopenharmony_ci break; 9862306a36Sopenharmony_ci case REISERFS_IOC_SETVERSION: 9962306a36Sopenharmony_ci if (!inode_owner_or_capable(&nop_mnt_idmap, inode)) { 10062306a36Sopenharmony_ci err = -EPERM; 10162306a36Sopenharmony_ci break; 10262306a36Sopenharmony_ci } 10362306a36Sopenharmony_ci err = mnt_want_write_file(filp); 10462306a36Sopenharmony_ci if (err) 10562306a36Sopenharmony_ci break; 10662306a36Sopenharmony_ci if (get_user(inode->i_generation, (int __user *)arg)) { 10762306a36Sopenharmony_ci err = -EFAULT; 10862306a36Sopenharmony_ci goto setversion_out; 10962306a36Sopenharmony_ci } 11062306a36Sopenharmony_ci inode_set_ctime_current(inode); 11162306a36Sopenharmony_ci mark_inode_dirty(inode); 11262306a36Sopenharmony_cisetversion_out: 11362306a36Sopenharmony_ci mnt_drop_write_file(filp); 11462306a36Sopenharmony_ci break; 11562306a36Sopenharmony_ci default: 11662306a36Sopenharmony_ci err = -ENOTTY; 11762306a36Sopenharmony_ci } 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_ci reiserfs_write_unlock(inode->i_sb); 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci return err; 12262306a36Sopenharmony_ci} 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci#ifdef CONFIG_COMPAT 12562306a36Sopenharmony_cilong reiserfs_compat_ioctl(struct file *file, unsigned int cmd, 12662306a36Sopenharmony_ci unsigned long arg) 12762306a36Sopenharmony_ci{ 12862306a36Sopenharmony_ci /* 12962306a36Sopenharmony_ci * These are just misnamed, they actually 13062306a36Sopenharmony_ci * get/put from/to user an int 13162306a36Sopenharmony_ci */ 13262306a36Sopenharmony_ci switch (cmd) { 13362306a36Sopenharmony_ci case REISERFS_IOC32_UNPACK: 13462306a36Sopenharmony_ci cmd = REISERFS_IOC_UNPACK; 13562306a36Sopenharmony_ci break; 13662306a36Sopenharmony_ci case REISERFS_IOC32_GETVERSION: 13762306a36Sopenharmony_ci cmd = REISERFS_IOC_GETVERSION; 13862306a36Sopenharmony_ci break; 13962306a36Sopenharmony_ci case REISERFS_IOC32_SETVERSION: 14062306a36Sopenharmony_ci cmd = REISERFS_IOC_SETVERSION; 14162306a36Sopenharmony_ci break; 14262306a36Sopenharmony_ci default: 14362306a36Sopenharmony_ci return -ENOIOCTLCMD; 14462306a36Sopenharmony_ci } 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci return reiserfs_ioctl(file, cmd, (unsigned long) compat_ptr(arg)); 14762306a36Sopenharmony_ci} 14862306a36Sopenharmony_ci#endif 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_ciint reiserfs_commit_write(struct file *f, struct page *page, 15162306a36Sopenharmony_ci unsigned from, unsigned to); 15262306a36Sopenharmony_ci/* 15362306a36Sopenharmony_ci * reiserfs_unpack 15462306a36Sopenharmony_ci * Function try to convert tail from direct item into indirect. 15562306a36Sopenharmony_ci * It set up nopack attribute in the REISERFS_I(inode)->nopack 15662306a36Sopenharmony_ci */ 15762306a36Sopenharmony_ciint reiserfs_unpack(struct inode *inode) 15862306a36Sopenharmony_ci{ 15962306a36Sopenharmony_ci int retval = 0; 16062306a36Sopenharmony_ci int index; 16162306a36Sopenharmony_ci struct page *page; 16262306a36Sopenharmony_ci struct address_space *mapping; 16362306a36Sopenharmony_ci unsigned long write_from; 16462306a36Sopenharmony_ci unsigned long blocksize = inode->i_sb->s_blocksize; 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci if (inode->i_size == 0) { 16762306a36Sopenharmony_ci REISERFS_I(inode)->i_flags |= i_nopack_mask; 16862306a36Sopenharmony_ci return 0; 16962306a36Sopenharmony_ci } 17062306a36Sopenharmony_ci /* ioctl already done */ 17162306a36Sopenharmony_ci if (REISERFS_I(inode)->i_flags & i_nopack_mask) { 17262306a36Sopenharmony_ci return 0; 17362306a36Sopenharmony_ci } 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci /* we need to make sure nobody is changing the file size beneath us */ 17662306a36Sopenharmony_ci { 17762306a36Sopenharmony_ci int depth = reiserfs_write_unlock_nested(inode->i_sb); 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci inode_lock(inode); 18062306a36Sopenharmony_ci reiserfs_write_lock_nested(inode->i_sb, depth); 18162306a36Sopenharmony_ci } 18262306a36Sopenharmony_ci 18362306a36Sopenharmony_ci reiserfs_write_lock(inode->i_sb); 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_ci write_from = inode->i_size & (blocksize - 1); 18662306a36Sopenharmony_ci /* if we are on a block boundary, we are already unpacked. */ 18762306a36Sopenharmony_ci if (write_from == 0) { 18862306a36Sopenharmony_ci REISERFS_I(inode)->i_flags |= i_nopack_mask; 18962306a36Sopenharmony_ci goto out; 19062306a36Sopenharmony_ci } 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_ci /* 19362306a36Sopenharmony_ci * we unpack by finding the page with the tail, and calling 19462306a36Sopenharmony_ci * __reiserfs_write_begin on that page. This will force a 19562306a36Sopenharmony_ci * reiserfs_get_block to unpack the tail for us. 19662306a36Sopenharmony_ci */ 19762306a36Sopenharmony_ci index = inode->i_size >> PAGE_SHIFT; 19862306a36Sopenharmony_ci mapping = inode->i_mapping; 19962306a36Sopenharmony_ci page = grab_cache_page(mapping, index); 20062306a36Sopenharmony_ci retval = -ENOMEM; 20162306a36Sopenharmony_ci if (!page) { 20262306a36Sopenharmony_ci goto out; 20362306a36Sopenharmony_ci } 20462306a36Sopenharmony_ci retval = __reiserfs_write_begin(page, write_from, 0); 20562306a36Sopenharmony_ci if (retval) 20662306a36Sopenharmony_ci goto out_unlock; 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci /* conversion can change page contents, must flush */ 20962306a36Sopenharmony_ci flush_dcache_page(page); 21062306a36Sopenharmony_ci retval = reiserfs_commit_write(NULL, page, write_from, write_from); 21162306a36Sopenharmony_ci REISERFS_I(inode)->i_flags |= i_nopack_mask; 21262306a36Sopenharmony_ci 21362306a36Sopenharmony_ciout_unlock: 21462306a36Sopenharmony_ci unlock_page(page); 21562306a36Sopenharmony_ci put_page(page); 21662306a36Sopenharmony_ci 21762306a36Sopenharmony_ciout: 21862306a36Sopenharmony_ci inode_unlock(inode); 21962306a36Sopenharmony_ci reiserfs_write_unlock(inode->i_sb); 22062306a36Sopenharmony_ci return retval; 22162306a36Sopenharmony_ci} 222