Blame view

fs/jfs/jfs_metapage.c 19.6 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
  /*
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
2
   *   Copyright (C) International Business Machines Corp., 2000-2005
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
3
4
5
6
   *   Portions Copyright (C) Christoph Hellwig, 2001-2002
   *
   *   This program is free software;  you can redistribute it and/or modify
   *   it under the terms of the GNU General Public License as published by
63f83c9fc   Dave Kleikamp   JFS: White space ...
7
   *   the Free Software Foundation; either version 2 of the License, or
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
8
   *   (at your option) any later version.
63f83c9fc   Dave Kleikamp   JFS: White space ...
9
   *
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
10
11
12
13
14
15
   *   This program is distributed in the hope that it will be useful,
   *   but WITHOUT ANY WARRANTY;  without even the implied warranty of
   *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
   *   the GNU General Public License for more details.
   *
   *   You should have received a copy of the GNU General Public License
63f83c9fc   Dave Kleikamp   JFS: White space ...
16
   *   along with this program;  if not, write to the Free Software
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
17
18
19
20
   *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
   */
  
  #include <linux/fs.h>
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
21
  #include <linux/mm.h>
b2e03ca74   Alexey Dobriyan   JFS: switch to se...
22
  #include <linux/module.h>
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
23
  #include <linux/bio.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
24
  #include <linux/slab.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
25
26
27
  #include <linux/init.h>
  #include <linux/buffer_head.h>
  #include <linux/mempool.h>
b2e03ca74   Alexey Dobriyan   JFS: switch to se...
28
  #include <linux/seq_file.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
29
30
31
32
33
34
  #include "jfs_incore.h"
  #include "jfs_superblock.h"
  #include "jfs_filsys.h"
  #include "jfs_metapage.h"
  #include "jfs_txnmgr.h"
  #include "jfs_debug.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35
36
37
38
39
40
41
  #ifdef CONFIG_JFS_STATISTICS
  static struct {
  	uint	pagealloc;	/* # of page allocations */
  	uint	pagefree;	/* # of page frees */
  	uint	lockwait;	/* # of sleeping lock_metapage() calls */
  } mpStat;
  #endif
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
42
  #define metapage_locked(mp) test_bit(META_locked, &(mp)->flag)
54af6233d   Nick Piggin   JFS is missing a ...
43
  #define trylock_metapage(mp) test_and_set_bit_lock(META_locked, &(mp)->flag)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
44
45
46
  
  static inline void unlock_metapage(struct metapage *mp)
  {
54af6233d   Nick Piggin   JFS is missing a ...
47
  	clear_bit_unlock(META_locked, &mp->flag);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
48
49
  	wake_up(&mp->wait);
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
50
  static inline void __lock_metapage(struct metapage *mp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
51
52
  {
  	DECLARE_WAITQUEUE(wait, current);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
53
  	INCREMENT(mpStat.lockwait);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
54
55
56
57
  	add_wait_queue_exclusive(&mp->wait, &wait);
  	do {
  		set_current_state(TASK_UNINTERRUPTIBLE);
  		if (metapage_locked(mp)) {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
58
  			unlock_page(mp->page);
4aa0d230c   Dave Kleikamp   JFS: call io_sche...
59
  			io_schedule();
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
60
  			lock_page(mp->page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
61
62
63
64
65
  		}
  	} while (trylock_metapage(mp));
  	__set_current_state(TASK_RUNNING);
  	remove_wait_queue(&mp->wait, &wait);
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
66
67
68
  /*
   * Must have mp->page locked
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
69
70
71
72
73
74
75
  static inline void lock_metapage(struct metapage *mp)
  {
  	if (trylock_metapage(mp))
  		__lock_metapage(mp);
  }
  
  #define METAPOOL_MIN_PAGES 32
e18b890bb   Christoph Lameter   [PATCH] slab: rem...
76
  static struct kmem_cache *metapage_cache;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
77
  static mempool_t *metapage_mempool;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
78
79
80
81
82
83
84
85
86
  #define MPS_PER_PAGE (PAGE_CACHE_SIZE >> L2PSIZE)
  
  #if MPS_PER_PAGE > 1
  
  struct meta_anchor {
  	int mp_count;
  	atomic_t io_count;
  	struct metapage *mp[MPS_PER_PAGE];
  };
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
87
  #define mp_anchor(page) ((struct meta_anchor *)page_private(page))
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
88

967c9ec4e   Dave Kleikamp   JFS: simplify typ...
89
  static inline struct metapage *page_to_mp(struct page *page, int offset)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
  {
  	if (!PagePrivate(page))
  		return NULL;
  	return mp_anchor(page)->mp[offset >> L2PSIZE];
  }
  
  static inline int insert_metapage(struct page *page, struct metapage *mp)
  {
  	struct meta_anchor *a;
  	int index;
  	int l2mp_blocks;	/* log2 blocks per metapage */
  
  	if (PagePrivate(page))
  		a = mp_anchor(page);
  	else {
5b3030e39   Eric Sesterhenn   JFS: kzalloc conv...
105
  		a = kzalloc(sizeof(struct meta_anchor), GFP_NOFS);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
106
107
  		if (!a)
  			return -ENOMEM;
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
108
  		set_page_private(page, (unsigned long)a);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
  		SetPagePrivate(page);
  		kmap(page);
  	}
  
  	if (mp) {
  		l2mp_blocks = L2PSIZE - page->mapping->host->i_blkbits;
  		index = (mp->index >> l2mp_blocks) & (MPS_PER_PAGE - 1);
  		a->mp_count++;
  		a->mp[index] = mp;
  	}
  
  	return 0;
  }
  
  static inline void remove_metapage(struct page *page, struct metapage *mp)
  {
  	struct meta_anchor *a = mp_anchor(page);
  	int l2mp_blocks = L2PSIZE - page->mapping->host->i_blkbits;
  	int index;
  
  	index = (mp->index >> l2mp_blocks) & (MPS_PER_PAGE - 1);
  
  	BUG_ON(a->mp[index] != mp);
  
  	a->mp[index] = NULL;
  	if (--a->mp_count == 0) {
  		kfree(a);
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
136
  		set_page_private(page, 0);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
  		ClearPagePrivate(page);
  		kunmap(page);
  	}
  }
  
  static inline void inc_io(struct page *page)
  {
  	atomic_inc(&mp_anchor(page)->io_count);
  }
  
  static inline void dec_io(struct page *page, void (*handler) (struct page *))
  {
  	if (atomic_dec_and_test(&mp_anchor(page)->io_count))
  		handler(page);
  }
  
  #else
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
154
  static inline struct metapage *page_to_mp(struct page *page, int offset)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
155
  {
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
156
  	return PagePrivate(page) ? (struct metapage *)page_private(page) : NULL;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
157
158
159
160
161
  }
  
  static inline int insert_metapage(struct page *page, struct metapage *mp)
  {
  	if (mp) {
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
162
  		set_page_private(page, (unsigned long)mp);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
163
164
165
166
167
168
169
170
  		SetPagePrivate(page);
  		kmap(page);
  	}
  	return 0;
  }
  
  static inline void remove_metapage(struct page *page, struct metapage *mp)
  {
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
171
  	set_page_private(page, 0);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
172
173
174
175
176
177
178
179
  	ClearPagePrivate(page);
  	kunmap(page);
  }
  
  #define inc_io(page) do {} while(0)
  #define dec_io(page, handler) handler(page)
  
  #endif
51cc50685   Alexey Dobriyan   SL*B: drop kmem c...
180
  static void init_once(void *foo)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
181
182
  {
  	struct metapage *mp = (struct metapage *)foo;
a35afb830   Christoph Lameter   Remove SLAB_CTOR_...
183
184
185
186
187
188
189
190
  	mp->lid = 0;
  	mp->lsn = 0;
  	mp->flag = 0;
  	mp->data = NULL;
  	mp->clsn = 0;
  	mp->log = NULL;
  	set_bit(META_free, &mp->flag);
  	init_waitqueue_head(&mp->wait);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
191
  }
27496a8c6   Al Viro   [PATCH] gfp_t: fs/*
192
  static inline struct metapage *alloc_metapage(gfp_t gfp_mask)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
  {
  	return mempool_alloc(metapage_mempool, gfp_mask);
  }
  
  static inline void free_metapage(struct metapage *mp)
  {
  	mp->flag = 0;
  	set_bit(META_free, &mp->flag);
  
  	mempool_free(mp, metapage_mempool);
  }
  
  int __init metapage_init(void)
  {
  	/*
  	 * Allocate the metapage structures
  	 */
  	metapage_cache = kmem_cache_create("jfs_mp", sizeof(struct metapage),
20c2df83d   Paul Mundt   mm: Remove slab d...
211
  					   0, 0, init_once);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
212
213
  	if (metapage_cache == NULL)
  		return -ENOMEM;
93d2341c7   Matthew Dobson   [PATCH] mempool: ...
214
215
  	metapage_mempool = mempool_create_slab_pool(METAPOOL_MIN_PAGES,
  						    metapage_cache);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
216
217
218
219
220
  
  	if (metapage_mempool == NULL) {
  		kmem_cache_destroy(metapage_cache);
  		return -ENOMEM;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
221
222
223
224
225
226
227
228
229
  
  	return 0;
  }
  
  void metapage_exit(void)
  {
  	mempool_destroy(metapage_mempool);
  	kmem_cache_destroy(metapage_cache);
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
230
231
232
233
234
235
236
237
238
  static inline void drop_metapage(struct page *page, struct metapage *mp)
  {
  	if (mp->count || mp->nohomeok || test_bit(META_dirty, &mp->flag) ||
  	    test_bit(META_io, &mp->flag))
  		return;
  	remove_metapage(page, mp);
  	INCREMENT(mpStat.pagefree);
  	free_metapage(mp);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
239
  /*
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
240
   * Metapage address space operations
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
241
   */
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
242
243
  
  static sector_t metapage_get_blocks(struct inode *inode, sector_t lblock,
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
244
  				    int *len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
245
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
246
247
248
  	int rc = 0;
  	int xflag;
  	s64 xaddr;
ba52de123   Theodore Ts'o   [PATCH] inode-die...
249
  	sector_t file_blocks = (inode->i_size + inode->i_sb->s_blocksize - 1) >>
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
  			       inode->i_blkbits;
  
  	if (lblock >= file_blocks)
  		return 0;
  	if (lblock + *len > file_blocks)
  		*len = file_blocks - lblock;
  
  	if (inode->i_ino) {
  		rc = xtLookup(inode, (s64)lblock, *len, &xflag, &xaddr, len, 0);
  		if ((rc == 0) && *len)
  			lblock = (sector_t)xaddr;
  		else
  			lblock = 0;
  	} /* else no mapping */
  
  	return lblock;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
266
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
267
  static void last_read_complete(struct page *page)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
268
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
269
270
271
272
  	if (!PageError(page))
  		SetPageUptodate(page);
  	unlock_page(page);
  }
6712ecf8f   NeilBrown   Drop 'size' argum...
273
  static void metapage_read_end_io(struct bio *bio, int err)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
274
275
  {
  	struct page *page = bio->bi_private;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
276
277
278
279
  	if (!test_bit(BIO_UPTODATE, &bio->bi_flags)) {
  		printk(KERN_ERR "metapage_read_end_io: I/O error
  ");
  		SetPageError(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
280
  	}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
281
282
  	dec_io(page, last_read_complete);
  	bio_put(bio);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
283
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
284
  static void remove_from_logsync(struct metapage *mp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
285
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
  	struct jfs_log *log = mp->log;
  	unsigned long flags;
  /*
   * This can race.  Recheck that log hasn't been set to null, and after
   * acquiring logsync lock, recheck lsn
   */
  	if (!log)
  		return;
  
  	LOGSYNC_LOCK(log, flags);
  	if (mp->lsn) {
  		mp->log = NULL;
  		mp->lsn = 0;
  		mp->clsn = 0;
  		log->count--;
  		list_del(&mp->synclist);
  	}
  	LOGSYNC_UNLOCK(log, flags);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
305

7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
  static void last_write_complete(struct page *page)
  {
  	struct metapage *mp;
  	unsigned int offset;
  
  	for (offset = 0; offset < PAGE_CACHE_SIZE; offset += PSIZE) {
  		mp = page_to_mp(page, offset);
  		if (mp && test_bit(META_io, &mp->flag)) {
  			if (mp->lsn)
  				remove_from_logsync(mp);
  			clear_bit(META_io, &mp->flag);
  		}
  		/*
  		 * I'd like to call drop_metapage here, but I don't think it's
  		 * safe unless I have the page locked
  		 */
  	}
  	end_page_writeback(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
324
  }
6712ecf8f   NeilBrown   Drop 'size' argum...
325
  static void metapage_write_end_io(struct bio *bio, int err)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
326
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
327
328
329
  	struct page *page = bio->bi_private;
  
  	BUG_ON(!PagePrivate(page));
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
330
331
332
333
334
335
336
  	if (! test_bit(BIO_UPTODATE, &bio->bi_flags)) {
  		printk(KERN_ERR "metapage_write_end_io: I/O error
  ");
  		SetPageError(page);
  	}
  	dec_io(page, last_write_complete);
  	bio_put(bio);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
337
338
339
340
341
  }
  
  static int metapage_writepage(struct page *page, struct writeback_control *wbc)
  {
  	struct bio *bio = NULL;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
342
  	int block_offset;	/* block offset of mp within page */
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
343
  	struct inode *inode = page->mapping->host;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
344
345
346
  	int blocks_per_mp = JFS_SBI(inode->i_sb)->nbperpage;
  	int len;
  	int xlen;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
347
348
349
  	struct metapage *mp;
  	int redirty = 0;
  	sector_t lblock;
29a424f28   Dave Kleikamp   JFS: clear PAGECA...
350
  	int nr_underway = 0;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
351
352
353
354
355
  	sector_t pblock;
  	sector_t next_block = 0;
  	sector_t page_start;
  	unsigned long bio_bytes = 0;
  	unsigned long bio_offset = 0;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
356
  	int offset;
1ad53a98c   Dave Kleikamp   jfs: Fix error ha...
357
  	int bad_blocks = 0;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
358
359
360
361
362
  
  	page_start = (sector_t)page->index <<
  		     (PAGE_CACHE_SHIFT - inode->i_blkbits);
  	BUG_ON(!PageLocked(page));
  	BUG_ON(PageWriteback(page));
29a424f28   Dave Kleikamp   JFS: clear PAGECA...
363
  	set_page_writeback(page);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
364
365
366
367
368
369
370
371
372
  
  	for (offset = 0; offset < PAGE_CACHE_SIZE; offset += PSIZE) {
  		mp = page_to_mp(page, offset);
  
  		if (!mp || !test_bit(META_dirty, &mp->flag))
  			continue;
  
  		if (mp->nohomeok && !test_bit(META_forcewrite, &mp->flag)) {
  			redirty = 1;
ac17b8b57   Dave Kleikamp   JFS: make special...
373
374
375
376
377
378
  			/*
  			 * Make sure this page isn't blocked indefinitely.
  			 * If the journal isn't undergoing I/O, push it
  			 */
  			if (mp->log && !(mp->log->cflag & logGC_PAGEOUT))
  				jfs_flush_journal(mp->log, 0);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
379
380
381
382
  			continue;
  		}
  
  		clear_bit(META_dirty, &mp->flag);
1ad53a98c   Dave Kleikamp   jfs: Fix error ha...
383
  		set_bit(META_io, &mp->flag);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
384
385
386
387
388
389
390
391
  		block_offset = offset >> inode->i_blkbits;
  		lblock = page_start + block_offset;
  		if (bio) {
  			if (xlen && lblock == next_block) {
  				/* Contiguous, in memory & on disk */
  				len = min(xlen, blocks_per_mp);
  				xlen -= len;
  				bio_bytes += len << inode->i_blkbits;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
392
393
394
395
396
397
398
399
400
401
402
403
404
405
  				continue;
  			}
  			/* Not contiguous */
  			if (bio_add_page(bio, page, bio_bytes, bio_offset) <
  			    bio_bytes)
  				goto add_failed;
  			/*
  			 * Increment counter before submitting i/o to keep
  			 * count from hitting zero before we're through
  			 */
  			inc_io(page);
  			if (!bio->bi_size)
  				goto dump_bio;
  			submit_bio(WRITE, bio);
29a424f28   Dave Kleikamp   JFS: clear PAGECA...
406
  			nr_underway++;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
407
  			bio = NULL;
29a424f28   Dave Kleikamp   JFS: clear PAGECA...
408
  		} else
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
409
  			inc_io(page);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
410
411
412
  		xlen = (PAGE_CACHE_SIZE - offset) >> inode->i_blkbits;
  		pblock = metapage_get_blocks(inode, lblock, &xlen);
  		if (!pblock) {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
413
414
  			printk(KERN_ERR "JFS: metapage_get_blocks failed
  ");
1ad53a98c   Dave Kleikamp   jfs: Fix error ha...
415
416
417
418
419
  			/*
  			 * We already called inc_io(), but can't cancel it
  			 * with dec_io() until we're done with the page
  			 */
  			bad_blocks++;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
420
421
  			continue;
  		}
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
422
  		len = min(xlen, (int)JFS_SBI(inode->i_sb)->nbperpage);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
  
  		bio = bio_alloc(GFP_NOFS, 1);
  		bio->bi_bdev = inode->i_sb->s_bdev;
  		bio->bi_sector = pblock << (inode->i_blkbits - 9);
  		bio->bi_end_io = metapage_write_end_io;
  		bio->bi_private = page;
  
  		/* Don't call bio_add_page yet, we may add to this vec */
  		bio_offset = offset;
  		bio_bytes = len << inode->i_blkbits;
  
  		xlen -= len;
  		next_block = lblock + len;
  	}
  	if (bio) {
  		if (bio_add_page(bio, page, bio_bytes, bio_offset) < bio_bytes)
  				goto add_failed;
  		if (!bio->bi_size)
  			goto dump_bio;
63f83c9fc   Dave Kleikamp   JFS: White space ...
442

7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
443
  		submit_bio(WRITE, bio);
29a424f28   Dave Kleikamp   JFS: clear PAGECA...
444
  		nr_underway++;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
445
446
447
448
449
  	}
  	if (redirty)
  		redirty_page_for_writepage(wbc, page);
  
  	unlock_page(page);
1ad53a98c   Dave Kleikamp   jfs: Fix error ha...
450
451
  	if (bad_blocks)
  		goto err_out;
29a424f28   Dave Kleikamp   JFS: clear PAGECA...
452
453
  	if (nr_underway == 0)
  		end_page_writeback(page);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
454
455
456
457
458
459
460
  	return 0;
  add_failed:
  	/* We should never reach here, since we're only adding one vec */
  	printk(KERN_ERR "JFS: bio_add_page failed unexpectedly
  ");
  	goto skip;
  dump_bio:
288e4d838   Dave Kleikamp   JFS: Update print...
461
462
  	print_hex_dump(KERN_ERR, "JFS: dump of bio: ", DUMP_PREFIX_ADDRESS, 16,
  		       4, bio, sizeof(*bio), 0);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
463
464
465
466
  skip:
  	bio_put(bio);
  	unlock_page(page);
  	dec_io(page, last_write_complete);
1ad53a98c   Dave Kleikamp   jfs: Fix error ha...
467
468
469
  err_out:
  	while (bad_blocks--)
  		dec_io(page, last_write_complete);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
470
471
472
473
474
475
476
  	return -EIO;
  }
  
  static int metapage_readpage(struct file *fp, struct page *page)
  {
  	struct inode *inode = page->mapping->host;
  	struct bio *bio = NULL;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
477
478
  	int block_offset;
  	int blocks_per_page = PAGE_CACHE_SIZE >> inode->i_blkbits;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
479
480
  	sector_t page_start;	/* address of page in fs blocks */
  	sector_t pblock;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
481
  	int xlen;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
482
  	unsigned int len;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
483
  	int offset;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
  
  	BUG_ON(!PageLocked(page));
  	page_start = (sector_t)page->index <<
  		     (PAGE_CACHE_SHIFT - inode->i_blkbits);
  
  	block_offset = 0;
  	while (block_offset < blocks_per_page) {
  		xlen = blocks_per_page - block_offset;
  		pblock = metapage_get_blocks(inode, page_start + block_offset,
  					     &xlen);
  		if (pblock) {
  			if (!PagePrivate(page))
  				insert_metapage(page, NULL);
  			inc_io(page);
  			if (bio)
  				submit_bio(READ, bio);
  
  			bio = bio_alloc(GFP_NOFS, 1);
  			bio->bi_bdev = inode->i_sb->s_bdev;
  			bio->bi_sector = pblock << (inode->i_blkbits - 9);
  			bio->bi_end_io = metapage_read_end_io;
  			bio->bi_private = page;
  			len = xlen << inode->i_blkbits;
  			offset = block_offset << inode->i_blkbits;
  			if (bio_add_page(bio, page, len, offset) < len)
  				goto add_failed;
  			block_offset += xlen;
  		} else
  			block_offset++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
513
  	}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
514
515
516
517
518
519
  	if (bio)
  		submit_bio(READ, bio);
  	else
  		unlock_page(page);
  
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
520

7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
521
522
523
524
525
526
  add_failed:
  	printk(KERN_ERR "JFS: bio_add_page failed unexpectedly
  ");
  	bio_put(bio);
  	dec_io(page, last_read_complete);
  	return -EIO;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527
  }
27496a8c6   Al Viro   [PATCH] gfp_t: fs/*
528
  static int metapage_releasepage(struct page *page, gfp_t gfp_mask)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
529
530
  {
  	struct metapage *mp;
b964638ff   Dave Kleikamp   JFS: Fix multiple...
531
  	int ret = 1;
967c9ec4e   Dave Kleikamp   JFS: simplify typ...
532
  	int offset;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
533
534
535
536
537
538
539
540
  
  	for (offset = 0; offset < PAGE_CACHE_SIZE; offset += PSIZE) {
  		mp = page_to_mp(page, offset);
  
  		if (!mp)
  			continue;
  
  		jfs_info("metapage_releasepage: mp = 0x%p", mp);
b964638ff   Dave Kleikamp   JFS: Fix multiple...
541
542
  		if (mp->count || mp->nohomeok ||
  		    test_bit(META_dirty, &mp->flag)) {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
543
544
  			jfs_info("count = %ld, nohomeok = %d", mp->count,
  				 mp->nohomeok);
b964638ff   Dave Kleikamp   JFS: Fix multiple...
545
  			ret = 0;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
546
547
  			continue;
  		}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
548
549
550
551
552
553
  		if (mp->lsn)
  			remove_from_logsync(mp);
  		remove_metapage(page, mp);
  		INCREMENT(mpStat.pagefree);
  		free_metapage(mp);
  	}
b964638ff   Dave Kleikamp   JFS: Fix multiple...
554
  	return ret;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
555
  }
2ff28e22b   NeilBrown   [PATCH] Make addr...
556
  static void metapage_invalidatepage(struct page *page, unsigned long offset)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
557
558
  {
  	BUG_ON(offset);
2ff28e22b   NeilBrown   [PATCH] Make addr...
559
  	BUG_ON(PageWriteback(page));
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
560

2ff28e22b   NeilBrown   [PATCH] Make addr...
561
  	metapage_releasepage(page, 0);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
562
  }
f5e54d6e5   Christoph Hellwig   [PATCH] mark addr...
563
  const struct address_space_operations jfs_metapage_aops = {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
564
565
  	.readpage	= metapage_readpage,
  	.writepage	= metapage_writepage,
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
566
567
568
569
  	.releasepage	= metapage_releasepage,
  	.invalidatepage	= metapage_invalidatepage,
  	.set_page_dirty	= __set_page_dirty_nobuffers,
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
570
571
572
573
  struct metapage *__get_metapage(struct inode *inode, unsigned long lblock,
  				unsigned int size, int absolute,
  				unsigned long new)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
574
575
576
  	int l2BlocksPerPage;
  	int l2bsize;
  	struct address_space *mapping;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
577
578
  	struct metapage *mp = NULL;
  	struct page *page;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
579
580
  	unsigned long page_index;
  	unsigned long page_offset;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
581
582
583
584
585
586
587
588
589
590
591
592
593
  	jfs_info("__get_metapage: ino = %ld, lblock = 0x%lx, abs=%d",
  		 inode->i_ino, lblock, absolute);
  
  	l2bsize = inode->i_blkbits;
  	l2BlocksPerPage = PAGE_CACHE_SHIFT - l2bsize;
  	page_index = lblock >> l2BlocksPerPage;
  	page_offset = (lblock - (page_index << l2BlocksPerPage)) << l2bsize;
  	if ((page_offset + size) > PAGE_CACHE_SIZE) {
  		jfs_err("MetaData crosses page boundary!!");
  		jfs_err("lblock = %lx, size  = %d", lblock, size);
  		dump_stack();
  		return NULL;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
594
  	if (absolute)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
595
  		mapping = JFS_SBI(inode->i_sb)->direct_inode->i_mapping;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
596
597
598
599
600
601
602
603
604
605
  	else {
  		/*
  		 * If an nfs client tries to read an inode that is larger
  		 * than any existing inodes, we may try to read past the
  		 * end of the inode map
  		 */
  		if ((lblock << inode->i_blkbits) >= inode->i_size)
  			return NULL;
  		mapping = inode->i_mapping;
  	}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
606
607
608
609
610
611
612
613
  	if (new && (PSIZE == PAGE_CACHE_SIZE)) {
  		page = grab_cache_page(mapping, page_index);
  		if (!page) {
  			jfs_err("grab_cache_page failed!");
  			return NULL;
  		}
  		SetPageUptodate(page);
  	} else {
090d2b185   Pekka Enberg   [PATCH] read_mapp...
614
  		page = read_mapping_page(mapping, page_index, NULL);
c40c20249   Dave Kleikamp   JFS: Fix typo in ...
615
  		if (IS_ERR(page) || !PageUptodate(page)) {
090d2b185   Pekka Enberg   [PATCH] read_mapp...
616
  			jfs_err("read_mapping_page failed!");
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
617
618
619
620
621
622
  			return NULL;
  		}
  		lock_page(page);
  	}
  
  	mp = page_to_mp(page, page_offset);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
623
  	if (mp) {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
624
625
626
627
628
629
  		if (mp->logical_size != size) {
  			jfs_error(inode->i_sb,
  				  "__get_metapage: mp->logical_size != size");
  			jfs_err("logical_size = %d, size = %d",
  				mp->logical_size, size);
  			dump_stack();
63f83c9fc   Dave Kleikamp   JFS: White space ...
630
  			goto unlock;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
631
632
633
  		}
  		mp->count++;
  		lock_metapage(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
634
635
636
637
638
  		if (test_bit(META_discard, &mp->flag)) {
  			if (!new) {
  				jfs_error(inode->i_sb,
  					  "__get_metapage: using a "
  					  "discarded metapage");
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
639
  				discard_metapage(mp);
63f83c9fc   Dave Kleikamp   JFS: White space ...
640
  				goto unlock;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
641
642
643
  			}
  			clear_bit(META_discard, &mp->flag);
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
644
  	} else {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
645
646
647
  		INCREMENT(mpStat.pagealloc);
  		mp = alloc_metapage(GFP_NOFS);
  		mp->page = page;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
648
  		mp->flag = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
649
650
  		mp->xflag = COMMIT_PAGE;
  		mp->count = 1;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
651
  		mp->nohomeok = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
652
  		mp->logical_size = size;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
653
654
655
656
657
  		mp->data = page_address(page) + page_offset;
  		mp->index = lblock;
  		if (unlikely(insert_metapage(page, mp))) {
  			free_metapage(mp);
  			goto unlock;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
658
  		}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
659
  		lock_metapage(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
660
  	}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
661
662
  	if (new) {
  		jfs_info("zeroing mp = 0x%p", mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
663
  		memset(mp->data, 0, PSIZE);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
664
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
665

7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
666
667
  	unlock_page(page);
  	jfs_info("__get_metapage: returning = 0x%p data = 0x%p", mp, mp->data);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
668
  	return mp;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
669
670
  unlock:
  	unlock_page(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
671
672
  	return NULL;
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
673
  void grab_metapage(struct metapage * mp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
674
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
675
676
677
  	jfs_info("grab_metapage: mp = 0x%p", mp);
  	page_cache_get(mp->page);
  	lock_page(mp->page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
678
  	mp->count++;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
679
680
  	lock_metapage(mp);
  	unlock_page(mp->page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
681
  }
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
682
  void force_metapage(struct metapage *mp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
683
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
684
685
686
687
688
689
690
691
692
693
694
  	struct page *page = mp->page;
  	jfs_info("force_metapage: mp = 0x%p", mp);
  	set_bit(META_forcewrite, &mp->flag);
  	clear_bit(META_sync, &mp->flag);
  	page_cache_get(page);
  	lock_page(page);
  	set_page_dirty(page);
  	write_one_page(page, 1);
  	clear_bit(META_forcewrite, &mp->flag);
  	page_cache_release(page);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
695

1868f4aa5   Dave Kleikamp   JFS: fix sparse w...
696
  void hold_metapage(struct metapage *mp)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
697
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
698
  	lock_page(mp->page);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
699
  }
1868f4aa5   Dave Kleikamp   JFS: fix sparse w...
700
  void put_metapage(struct metapage *mp)
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
701
702
703
  {
  	if (mp->count || mp->nohomeok) {
  		/* Someone else will release this */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
704
  		unlock_page(mp->page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
705
706
  		return;
  	}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
707
708
709
  	page_cache_get(mp->page);
  	mp->count++;
  	lock_metapage(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
710
  	unlock_page(mp->page);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
711
  	release_metapage(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
712
713
714
715
  }
  
  void release_metapage(struct metapage * mp)
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
716
  	struct page *page = mp->page;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
717
  	jfs_info("release_metapage: mp = 0x%p, flag = 0x%lx", mp, mp->flag);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
718
719
720
721
  	BUG_ON(!page);
  
  	lock_page(page);
  	unlock_metapage(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
722
723
  
  	assert(mp->count);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
724
725
726
  	if (--mp->count || mp->nohomeok) {
  		unlock_page(page);
  		page_cache_release(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
727
728
  		return;
  	}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
729
730
  	if (test_bit(META_dirty, &mp->flag)) {
  		set_page_dirty(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
731
  		if (test_bit(META_sync, &mp->flag)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
732
  			clear_bit(META_sync, &mp->flag);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
733
734
  			write_one_page(page, 1);
  			lock_page(page); /* write_one_page unlocks the page */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
735
  		}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
736
737
  	} else if (mp->lsn)	/* discard_metapage doesn't remove it */
  		remove_from_logsync(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
738

7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
739
740
  	/* Try to keep metapages from using up too much memory */
  	drop_metapage(page, mp);
d0e671a93   Dave Kleikamp   [PATCH] Fix JFS a...
741

7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
742
743
  	unlock_page(page);
  	page_cache_release(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
744
745
746
747
  }
  
  void __invalidate_metapages(struct inode *ip, s64 addr, int len)
  {
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
748
  	sector_t lblock;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
749
  	int l2BlocksPerPage = PAGE_CACHE_SHIFT - ip->i_blkbits;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
750
  	int BlocksPerPage = 1 << l2BlocksPerPage;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
751
  	/* All callers are interested in block device's mapping */
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
752
753
  	struct address_space *mapping =
  		JFS_SBI(ip->i_sb)->direct_inode->i_mapping;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
754
755
  	struct metapage *mp;
  	struct page *page;
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
756
  	unsigned int offset;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
757
758
  
  	/*
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
759
  	 * Mark metapages to discard.  They will eventually be
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
760
761
  	 * released, but should not be written.
  	 */
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
762
763
764
765
766
767
768
769
770
771
772
773
774
  	for (lblock = addr & ~(BlocksPerPage - 1); lblock < addr + len;
  	     lblock += BlocksPerPage) {
  		page = find_lock_page(mapping, lblock >> l2BlocksPerPage);
  		if (!page)
  			continue;
  		for (offset = 0; offset < PAGE_CACHE_SIZE; offset += PSIZE) {
  			mp = page_to_mp(page, offset);
  			if (!mp)
  				continue;
  			if (mp->index < addr)
  				continue;
  			if (mp->index >= addr + len)
  				break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
775
776
777
  
  			clear_bit(META_dirty, &mp->flag);
  			set_bit(META_discard, &mp->flag);
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
778
779
  			if (mp->lsn)
  				remove_from_logsync(mp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
780
  		}
7fab479be   Dave Kleikamp   [PATCH] JFS: Supp...
781
782
  		unlock_page(page);
  		page_cache_release(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
783
784
785
786
  	}
  }
  
  #ifdef CONFIG_JFS_STATISTICS
b2e03ca74   Alexey Dobriyan   JFS: switch to se...
787
  static int jfs_mpstat_proc_show(struct seq_file *m, void *v)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
788
  {
b2e03ca74   Alexey Dobriyan   JFS: switch to se...
789
  	seq_printf(m,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
790
791
792
793
794
795
796
797
798
799
800
801
802
  		       "JFS Metapage statistics
  "
  		       "=======================
  "
  		       "page allocations = %d
  "
  		       "page frees = %d
  "
  		       "lock waits = %d
  ",
  		       mpStat.pagealloc,
  		       mpStat.pagefree,
  		       mpStat.lockwait);
b2e03ca74   Alexey Dobriyan   JFS: switch to se...
803
804
  	return 0;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
805

b2e03ca74   Alexey Dobriyan   JFS: switch to se...
806
807
808
  static int jfs_mpstat_proc_open(struct inode *inode, struct file *file)
  {
  	return single_open(file, jfs_mpstat_proc_show, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
809
  }
b2e03ca74   Alexey Dobriyan   JFS: switch to se...
810
811
812
813
814
815
816
817
  
  const struct file_operations jfs_mpstat_proc_fops = {
  	.owner		= THIS_MODULE,
  	.open		= jfs_mpstat_proc_open,
  	.read		= seq_read,
  	.llseek		= seq_lseek,
  	.release	= single_release,
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
818
  #endif