Blame view

fs/logfs/file.c 7.06 KB
5db53f3e8   Joern Engel   [LogFS] add new f...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
  /*
   * fs/logfs/file.c	- prepare_write, commit_write and friends
   *
   * As should be obvious for Linux kernel code, license is GPLv2
   *
   * Copyright (c) 2005-2008 Joern Engel <joern@logfs.org>
   */
  #include "logfs.h"
  #include <linux/sched.h>
  #include <linux/writeback.h>
  
  static int logfs_write_begin(struct file *file, struct address_space *mapping,
  		loff_t pos, unsigned len, unsigned flags,
  		struct page **pagep, void **fsdata)
  {
  	struct inode *inode = mapping->host;
  	struct page *page;
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
18
  	pgoff_t index = pos >> PAGE_SHIFT;
5db53f3e8   Joern Engel   [LogFS] add new f...
19
20
21
22
23
  
  	page = grab_cache_page_write_begin(mapping, index, flags);
  	if (!page)
  		return -ENOMEM;
  	*pagep = page;
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
24
  	if ((len == PAGE_SIZE) || PageUptodate(page))
5db53f3e8   Joern Engel   [LogFS] add new f...
25
  		return 0;
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
26
27
  	if ((pos & PAGE_MASK) >= i_size_read(inode)) {
  		unsigned start = pos & (PAGE_SIZE - 1);
5db53f3e8   Joern Engel   [LogFS] add new f...
28
29
30
  		unsigned end = start + len;
  
  		/* Reading beyond i_size is simple: memset to zero */
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
31
  		zero_user_segments(page, 0, start, end, PAGE_SIZE);
5db53f3e8   Joern Engel   [LogFS] add new f...
32
33
34
35
36
37
38
39
40
41
42
  		return 0;
  	}
  	return logfs_readpage_nolock(page);
  }
  
  static int logfs_write_end(struct file *file, struct address_space *mapping,
  		loff_t pos, unsigned len, unsigned copied, struct page *page,
  		void *fsdata)
  {
  	struct inode *inode = mapping->host;
  	pgoff_t index = page->index;
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
43
  	unsigned start = pos & (PAGE_SIZE - 1);
5db53f3e8   Joern Engel   [LogFS] add new f...
44
45
  	unsigned end = start + copied;
  	int ret = 0;
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
46
  	BUG_ON(PAGE_SIZE != inode->i_sb->s_blocksize);
5db53f3e8   Joern Engel   [LogFS] add new f...
47
48
49
50
51
52
53
54
55
56
57
58
59
60
  	BUG_ON(page->index > I3_BLOCKS);
  
  	if (copied < len) {
  		/*
  		 * Short write of a non-initialized paged.  Just tell userspace
  		 * to retry the entire page.
  		 */
  		if (!PageUptodate(page)) {
  			copied = 0;
  			goto out;
  		}
  	}
  	if (copied == 0)
  		goto out; /* FIXME: do we need to update inode? */
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
61
62
  	if (i_size_read(inode) < (index << PAGE_SHIFT) + end) {
  		i_size_write(inode, (index << PAGE_SHIFT) + end);
5db53f3e8   Joern Engel   [LogFS] add new f...
63
64
65
66
67
68
69
70
71
72
73
74
  		mark_inode_dirty_sync(inode);
  	}
  
  	SetPageUptodate(page);
  	if (!PageDirty(page)) {
  		if (!get_page_reserve(inode, page))
  			__set_page_dirty_nobuffers(page);
  		else
  			ret = logfs_write_buf(inode, page, WF_LOCK);
  	}
  out:
  	unlock_page(page);
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
75
  	put_page(page);
5db53f3e8   Joern Engel   [LogFS] add new f...
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
  	return ret ? ret : copied;
  }
  
  int logfs_readpage(struct file *file, struct page *page)
  {
  	int ret;
  
  	ret = logfs_readpage_nolock(page);
  	unlock_page(page);
  	return ret;
  }
  
  /* Clear the page's dirty flag in the radix tree. */
  /* TODO: mucking with PageWriteback is silly.  Add a generic function to clear
   * the dirty bit from the radix tree for filesystems that don't have to wait
   * for page writeback to finish (i.e. any compressing filesystem).
   */
  static void clear_radix_tree_dirty(struct page *page)
  {
  	BUG_ON(PagePrivate(page) || page->private);
  	set_page_writeback(page);
  	end_page_writeback(page);
  }
  
  static int __logfs_writepage(struct page *page)
  {
  	struct inode *inode = page->mapping->host;
  	int err;
  
  	err = logfs_write_buf(inode, page, WF_LOCK);
  	if (err)
  		set_page_dirty(page);
  	else
  		clear_radix_tree_dirty(page);
  	unlock_page(page);
  	return err;
  }
  
  static int logfs_writepage(struct page *page, struct writeback_control *wbc)
  {
  	struct inode *inode = page->mapping->host;
  	loff_t i_size = i_size_read(inode);
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
118
  	pgoff_t end_index = i_size >> PAGE_SHIFT;
5db53f3e8   Joern Engel   [LogFS] add new f...
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
  	unsigned offset;
  	u64 bix;
  	level_t level;
  
  	log_file("logfs_writepage(%lx, %lx, %p)
  ", inode->i_ino, page->index,
  			page);
  
  	logfs_unpack_index(page->index, &bix, &level);
  
  	/* Indirect blocks are never truncated */
  	if (level != 0)
  		return __logfs_writepage(page);
  
  	/*
  	 * TODO: everything below is a near-verbatim copy of nobh_writepage().
  	 * The relevant bits should be factored out after logfs is merged.
  	 */
  
  	/* Is the page fully inside i_size? */
  	if (bix < end_index)
  		return __logfs_writepage(page);
  
  	 /* Is the page fully outside i_size? (truncate in progress) */
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
143
  	offset = i_size & (PAGE_SIZE-1);
5db53f3e8   Joern Engel   [LogFS] add new f...
144
145
146
147
148
149
150
151
152
153
154
155
  	if (bix > end_index || offset == 0) {
  		unlock_page(page);
  		return 0; /* don't care */
  	}
  
  	/*
  	 * The page straddles i_size.  It must be zeroed out on each and every
  	 * writepage invokation because it may be mmapped.  "A file is mapped
  	 * in multiples of the page size.  For a file that is not a multiple of
  	 * the  page size, the remaining memory is zeroed when mapped, and
  	 * writes to that region are not written out to the file."
  	 */
09cbfeaf1   Kirill A. Shutemov   mm, fs: get rid o...
156
  	zero_user_segment(page, offset, PAGE_SIZE);
5db53f3e8   Joern Engel   [LogFS] add new f...
157
158
  	return __logfs_writepage(page);
  }
d47992f86   Lukas Czerner   mm: change invali...
159
160
  static void logfs_invalidatepage(struct page *page, unsigned int offset,
  				 unsigned int length)
5db53f3e8   Joern Engel   [LogFS] add new f...
161
  {
05ebad852   Joern Engel   logfs: commit res...
162
163
164
165
166
167
168
169
170
171
172
  	struct logfs_block *block = logfs_block(page);
  
  	if (block->reserved_bytes) {
  		struct super_block *sb = page->mapping->host->i_sb;
  		struct logfs_super *super = logfs_super(sb);
  
  		super->s_dirty_pages -= block->reserved_bytes;
  		block->ops->free_block(sb, block);
  		BUG_ON(bitmap_weight(block->alias_map, LOGFS_BLOCK_FACTOR));
  	} else
  		move_page_to_btree(page);
5db53f3e8   Joern Engel   [LogFS] add new f...
173
174
175
176
177
178
179
  	BUG_ON(PagePrivate(page) || page->private);
  }
  
  static int logfs_releasepage(struct page *page, gfp_t only_xfs_uses_this)
  {
  	return 0; /* None of these are easy to release */
  }
02d6d685f   Arnd Bergmann   logfs: kill BKL
180
  long logfs_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
5db53f3e8   Joern Engel   [LogFS] add new f...
181
  {
496ad9aa8   Al Viro   new helper: file_...
182
  	struct inode *inode = file_inode(file);
5db53f3e8   Joern Engel   [LogFS] add new f...
183
184
185
186
187
188
189
190
191
192
193
  	struct logfs_inode *li = logfs_inode(inode);
  	unsigned int oldflags, flags;
  	int err;
  
  	switch (cmd) {
  	case FS_IOC_GETFLAGS:
  		flags = li->li_flags & LOGFS_FL_USER_VISIBLE;
  		return put_user(flags, (int __user *)arg);
  	case FS_IOC_SETFLAGS:
  		if (IS_RDONLY(inode))
  			return -EROFS;
2e1496707   Serge E. Hallyn   userns: rename is...
194
  		if (!inode_owner_or_capable(inode))
5db53f3e8   Joern Engel   [LogFS] add new f...
195
196
197
198
199
  			return -EACCES;
  
  		err = get_user(flags, (int __user *)arg);
  		if (err)
  			return err;
5955102c9   Al Viro   wrappers for ->i_...
200
  		inode_lock(inode);
5db53f3e8   Joern Engel   [LogFS] add new f...
201
202
203
204
  		oldflags = li->li_flags;
  		flags &= LOGFS_FL_USER_MODIFIABLE;
  		flags |= oldflags & ~LOGFS_FL_USER_MODIFIABLE;
  		li->li_flags = flags;
5955102c9   Al Viro   wrappers for ->i_...
205
  		inode_unlock(inode);
5db53f3e8   Joern Engel   [LogFS] add new f...
206

078cd8279   Deepa Dinamani   fs: Replace CURRE...
207
  		inode->i_ctime = current_time(inode);
5db53f3e8   Joern Engel   [LogFS] add new f...
208
209
210
211
212
213
214
  		mark_inode_dirty_sync(inode);
  		return 0;
  
  	default:
  		return -ENOTTY;
  	}
  }
02c24a821   Josef Bacik   fs: push i_mutex ...
215
  int logfs_fsync(struct file *file, loff_t start, loff_t end, int datasync)
5db53f3e8   Joern Engel   [LogFS] add new f...
216
  {
7ea808591   Christoph Hellwig   drop unused dentr...
217
  	struct super_block *sb = file->f_mapping->host->i_sb;
02c24a821   Josef Bacik   fs: push i_mutex ...
218
219
220
221
222
223
  	struct inode *inode = file->f_mapping->host;
  	int ret;
  
  	ret = filemap_write_and_wait_range(inode->i_mapping, start, end);
  	if (ret)
  		return ret;
5db53f3e8   Joern Engel   [LogFS] add new f...
224

5955102c9   Al Viro   wrappers for ->i_...
225
  	inode_lock(inode);
13ced29cb   Prasad Joshi   logfs: take write...
226
  	logfs_get_wblocks(sb, NULL, WF_LOCK);
c0c79c31c   Joern Engel   logfs: fix sync
227
  	logfs_write_anchor(sb);
13ced29cb   Prasad Joshi   logfs: take write...
228
  	logfs_put_wblocks(sb, NULL, WF_LOCK);
5955102c9   Al Viro   wrappers for ->i_...
229
  	inode_unlock(inode);
02c24a821   Josef Bacik   fs: push i_mutex ...
230

5db53f3e8   Joern Engel   [LogFS] add new f...
231
232
233
234
235
  	return 0;
  }
  
  static int logfs_setattr(struct dentry *dentry, struct iattr *attr)
  {
2b0143b5c   David Howells   VFS: normal files...
236
  	struct inode *inode = d_inode(dentry);
5db53f3e8   Joern Engel   [LogFS] add new f...
237
  	int err = 0;
31051c85b   Jan Kara   fs: Give dentry t...
238
  	err = setattr_prepare(dentry, attr);
db78b877f   Christoph Hellwig   always call inode...
239
240
  	if (err)
  		return err;
1025774ce   Christoph Hellwig   remove inode_setattr
241
  	if (attr->ia_valid & ATTR_SIZE) {
5db53f3e8   Joern Engel   [LogFS] add new f...
242
  		err = logfs_truncate(inode, attr->ia_size);
1025774ce   Christoph Hellwig   remove inode_setattr
243
244
245
  		if (err)
  			return err;
  	}
5db53f3e8   Joern Engel   [LogFS] add new f...
246

1025774ce   Christoph Hellwig   remove inode_setattr
247
248
249
  	setattr_copy(inode, attr);
  	mark_inode_dirty(inode);
  	return 0;
5db53f3e8   Joern Engel   [LogFS] add new f...
250
251
252
253
254
255
256
  }
  
  const struct inode_operations logfs_reg_iops = {
  	.setattr	= logfs_setattr,
  };
  
  const struct file_operations logfs_reg_fops = {
aad4f8bb4   Al Viro   switch simple gen...
257
  	.read_iter	= generic_file_read_iter,
8174202b3   Al Viro   write_iter varian...
258
  	.write_iter	= generic_file_write_iter,
5db53f3e8   Joern Engel   [LogFS] add new f...
259
  	.fsync		= logfs_fsync,
02d6d685f   Arnd Bergmann   logfs: kill BKL
260
  	.unlocked_ioctl	= logfs_ioctl,
5db53f3e8   Joern Engel   [LogFS] add new f...
261
262
263
  	.llseek		= generic_file_llseek,
  	.mmap		= generic_file_readonly_mmap,
  	.open		= generic_file_open,
5db53f3e8   Joern Engel   [LogFS] add new f...
264
265
266
267
268
269
270
271
272
273
274
275
  };
  
  const struct address_space_operations logfs_reg_aops = {
  	.invalidatepage	= logfs_invalidatepage,
  	.readpage	= logfs_readpage,
  	.releasepage	= logfs_releasepage,
  	.set_page_dirty	= __set_page_dirty_nobuffers,
  	.writepage	= logfs_writepage,
  	.writepages	= generic_writepages,
  	.write_begin	= logfs_write_begin,
  	.write_end	= logfs_write_end,
  };