file.c 5.9 KB
Newer Older
1
/*
2
 *  linux/fs/ext4/file.c
3 4 5 6 7 8 9 10 11 12 13 14
 *
 * Copyright (C) 1992, 1993, 1994, 1995
 * Remy Card (card@masi.ibp.fr)
 * Laboratoire MASI - Institut Blaise Pascal
 * Universite Pierre et Marie Curie (Paris VI)
 *
 *  from
 *
 *  linux/fs/minix/file.c
 *
 *  Copyright (C) 1991, 1992  Linus Torvalds
 *
15
 *  ext4 fs regular file handling primitives
16 17 18 19 20 21 22
 *
 *  64-bit file support on 64-bit platforms by Jakub Jelinek
 *	(jj@sunsite.ms.mff.cuni.cz)
 */

#include <linux/time.h>
#include <linux/fs.h>
23
#include <linux/jbd2.h>
24 25
#include <linux/mount.h>
#include <linux/path.h>
26
#include <linux/quotaops.h>
27 28
#include "ext4.h"
#include "ext4_jbd2.h"
29 30 31 32 33
#include "xattr.h"
#include "acl.h"

/*
 * Called when an inode is released. Note that this is different
34
 * from ext4_file_open: open gets called at every open, but release
35 36
 * gets called only when /all/ the files are closed.
 */
37
static int ext4_release_file(struct inode *inode, struct file *filp)
38
{
39
	if (ext4_test_inode_state(inode, EXT4_STATE_DA_ALLOC_CLOSE)) {
40
		ext4_alloc_da_blocks(inode);
41
		ext4_clear_inode_state(inode, EXT4_STATE_DA_ALLOC_CLOSE);
42
	}
43 44
	/* if we are the last writer on the inode, drop the block reservation */
	if ((filp->f_mode & FMODE_WRITE) &&
45 46
			(atomic_read(&inode->i_writecount) == 1) &&
		        !EXT4_I(inode)->i_reserved_data_blocks)
47
	{
48
		down_write(&EXT4_I(inode)->i_data_sem);
49
		ext4_discard_preallocations(inode);
50
		up_write(&EXT4_I(inode)->i_data_sem);
51 52
	}
	if (is_dx(inode) && filp->private_data)
53
		ext4_htree_free_dir_info(filp->private_data);
54 55 56 57 58

	return 0;
}

static ssize_t
59
ext4_file_write(struct kiocb *iocb, const struct iovec *iov,
60 61
		unsigned long nr_segs, loff_t pos)
{
62
	struct inode *inode = iocb->ki_filp->f_path.dentry->d_inode;
63

64 65 66 67 68
	/*
	 * If we have encountered a bitmap-format file, the size limit
	 * is smaller than s_maxbytes, which is for extent-mapped files.
	 */

69
	if (!(ext4_test_inode_flag(inode, EXT4_INODE_EXTENTS))) {
70 71
		struct ext4_sb_info *sbi = EXT4_SB(inode->i_sb);
		size_t length = iov_length(iov, nr_segs);
72

73 74
		if ((pos > sbi->s_bitmap_maxbytes ||
		    (pos == sbi->s_bitmap_maxbytes && length > 0)))
75 76 77 78 79 80 81 82
			return -EFBIG;

		if (pos + length > sbi->s_bitmap_maxbytes) {
			nr_segs = iov_shorten((struct iovec *)iov, nr_segs,
					      sbi->s_bitmap_maxbytes - pos);
		}
	}

83
	return generic_file_aio_write(iocb, iov, nr_segs, pos);
84 85
}

86
static const struct vm_operations_struct ext4_file_vm_ops = {
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
	.fault		= filemap_fault,
	.page_mkwrite   = ext4_page_mkwrite,
};

static int ext4_file_mmap(struct file *file, struct vm_area_struct *vma)
{
	struct address_space *mapping = file->f_mapping;

	if (!mapping->a_ops->readpage)
		return -ENOEXEC;
	file_accessed(file);
	vma->vm_ops = &ext4_file_vm_ops;
	vma->vm_flags |= VM_CAN_NONLINEAR;
	return 0;
}

103 104 105 106
static int ext4_file_open(struct inode * inode, struct file * filp)
{
	struct super_block *sb = inode->i_sb;
	struct ext4_sb_info *sbi = EXT4_SB(inode->i_sb);
107
	struct ext4_inode_info *ei = EXT4_I(inode);
108 109 110 111 112 113 114 115 116 117 118 119 120 121
	struct vfsmount *mnt = filp->f_path.mnt;
	struct path path;
	char buf[64], *cp;

	if (unlikely(!(sbi->s_mount_flags & EXT4_MF_MNTDIR_SAMPLED) &&
		     !(sb->s_flags & MS_RDONLY))) {
		sbi->s_mount_flags |= EXT4_MF_MNTDIR_SAMPLED;
		/*
		 * Sample where the filesystem has been mounted and
		 * store it in the superblock for sysadmin convenience
		 * when trying to sort through large numbers of block
		 * devices or filesystem images.
		 */
		memset(buf, 0, sizeof(buf));
122 123
		path.mnt = mnt;
		path.dentry = mnt->mnt_root;
124 125 126 127
		cp = d_path(&path, buf, sizeof(buf));
		if (!IS_ERR(cp)) {
			memcpy(sbi->s_es->s_last_mounted, cp,
			       sizeof(sbi->s_es->s_last_mounted));
T
Theodore Ts'o 已提交
128
			ext4_mark_super_dirty(sb);
129 130
		}
	}
131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151
	/*
	 * Set up the jbd2_inode if we are opening the inode for
	 * writing and the journal is present
	 */
	if (sbi->s_journal && !ei->jinode && (filp->f_mode & FMODE_WRITE)) {
		struct jbd2_inode *jinode = jbd2_alloc_inode(GFP_KERNEL);

		spin_lock(&inode->i_lock);
		if (!ei->jinode) {
			if (!jinode) {
				spin_unlock(&inode->i_lock);
				return -ENOMEM;
			}
			ei->jinode = jinode;
			jbd2_journal_init_jbd_inode(ei->jinode, inode);
			jinode = NULL;
		}
		spin_unlock(&inode->i_lock);
		if (unlikely(jinode != NULL))
			jbd2_free_inode(jinode);
	}
152
	return dquot_file_open(inode, filp);
153 154
}

155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
/*
 * ext4_llseek() copied from generic_file_llseek() to handle both
 * block-mapped and extent-mapped maxbytes values. This should
 * otherwise be identical with generic_file_llseek().
 */
loff_t ext4_llseek(struct file *file, loff_t offset, int origin)
{
	struct inode *inode = file->f_mapping->host;
	loff_t maxbytes;

	if (!(ext4_test_inode_flag(inode, EXT4_INODE_EXTENTS)))
		maxbytes = EXT4_SB(inode->i_sb)->s_bitmap_maxbytes;
	else
		maxbytes = inode->i_sb->s_maxbytes;
	mutex_lock(&inode->i_mutex);
	switch (origin) {
	case SEEK_END:
		offset += inode->i_size;
		break;
	case SEEK_CUR:
		if (offset == 0) {
			mutex_unlock(&inode->i_mutex);
			return file->f_pos;
		}
		offset += file->f_pos;
		break;
	}

	if (offset < 0 || offset > maxbytes) {
		mutex_unlock(&inode->i_mutex);
		return -EINVAL;
	}

	if (offset != file->f_pos) {
		file->f_pos = offset;
		file->f_version = 0;
	}
	mutex_unlock(&inode->i_mutex);

	return offset;
}

197
const struct file_operations ext4_file_operations = {
198
	.llseek		= ext4_llseek,
199 200 201
	.read		= do_sync_read,
	.write		= do_sync_write,
	.aio_read	= generic_file_aio_read,
202
	.aio_write	= ext4_file_write,
A
Andi Kleen 已提交
203
	.unlocked_ioctl = ext4_ioctl,
204
#ifdef CONFIG_COMPAT
205
	.compat_ioctl	= ext4_compat_ioctl,
206
#endif
207
	.mmap		= ext4_file_mmap,
208
	.open		= ext4_file_open,
209 210
	.release	= ext4_release_file,
	.fsync		= ext4_sync_file,
211 212
	.splice_read	= generic_file_splice_read,
	.splice_write	= generic_file_splice_write,
213
	.fallocate	= ext4_fallocate,
214 215
};

216
const struct inode_operations ext4_file_inode_operations = {
217 218
	.truncate	= ext4_truncate,
	.setattr	= ext4_setattr,
219
	.getattr	= ext4_getattr,
T
Theodore Ts'o 已提交
220
#ifdef CONFIG_EXT4_FS_XATTR
221 222
	.setxattr	= generic_setxattr,
	.getxattr	= generic_getxattr,
223
	.listxattr	= ext4_listxattr,
224 225
	.removexattr	= generic_removexattr,
#endif
226
	.check_acl	= ext4_check_acl,
227
	.fiemap		= ext4_fiemap,
228 229
};