Blame view

fs/dquot.c 67.5 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
  /*
   * Implementation of the diskquota system for the LINUX operating system. QUOTA
   * is implemented using the BSD system call interface as the means of
   * communication with the user level. This file contains the generic routines
   * called by the different filesystems on allocation of an inode or block.
   * These routines take care of the administration needed to have a consistent
   * diskquota tracking system. The ideas of both user and group quotas are based
   * on the Melbourne quota system as used on BSD derived systems. The internal
   * implementation is based on one of the several variants of the LINUX
   * inode-subsystem with added complexity of the diskquota system.
   * 
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
   * Author:	Marco van Wieringen <mvw@planets.elm.net>
   *
   * Fixes:   Dmitry Gorodchanin <pgmdsg@ibi.com>, 11 Feb 96
   *
   *		Revised list management to avoid races
   *		-- Bill Hawes, <whawes@star.net>, 9/98
   *
   *		Fixed races in dquot_transfer(), dqget() and dquot_alloc_...().
   *		As the consequence the locking was moved from dquot_decr_...(),
   *		dquot_incr_...() to calling functions.
   *		invalidate_dquots() now writes modified dquots.
   *		Serialized quota_off() and quota_on() for mount point.
   *		Fixed a few bugs in grow_dquots().
   *		Fixed deadlock in write_dquot() - we no longer account quotas on
   *		quota files
   *		remove_dquot_ref() moved to inode.c - it now traverses through inodes
   *		add_dquot_ref() restarts after blocking
   *		Added check for bogus uid and fixed check for group in quotactl.
   *		Jan Kara, <jack@suse.cz>, sponsored by SuSE CR, 10-11/99
   *
   *		Used struct list_head instead of own list struct
   *		Invalidation of referenced dquots is no longer possible
   *		Improved free_dquots list management
   *		Quota and i_blocks are now updated in one place to avoid races
   *		Warnings are now delayed so we won't block in critical section
   *		Write updated not to require dquot lock
   *		Jan Kara, <jack@suse.cz>, 9/2000
   *
   *		Added dynamic quota structure allocation
   *		Jan Kara <jack@suse.cz> 12/2000
   *
   *		Rewritten quota interface. Implemented new quota format and
   *		formats registering.
   *		Jan Kara, <jack@suse.cz>, 2001,2002
   *
   *		New SMP locking.
   *		Jan Kara, <jack@suse.cz>, 10/2002
   *
   *		Added journalled quota support, fix lock inversion problems
   *		Jan Kara, <jack@suse.cz>, 2003,2004
   *
   * (C) Copyright 1994 - 1997 Marco van Wieringen 
   */
  
  #include <linux/errno.h>
  #include <linux/kernel.h>
  #include <linux/fs.h>
  #include <linux/mount.h>
  #include <linux/mm.h>
  #include <linux/time.h>
  #include <linux/types.h>
  #include <linux/string.h>
  #include <linux/fcntl.h>
  #include <linux/stat.h>
  #include <linux/tty.h>
  #include <linux/file.h>
  #include <linux/slab.h>
  #include <linux/sysctl.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
70
71
72
73
74
75
76
  #include <linux/init.h>
  #include <linux/module.h>
  #include <linux/proc_fs.h>
  #include <linux/security.h>
  #include <linux/kmod.h>
  #include <linux/namei.h>
  #include <linux/buffer_head.h>
16f7e0fe2   Randy Dunlap   [PATCH] capable/c...
77
  #include <linux/capability.h>
be586bab8   Adrian Bunk   [PATCH] quota: sm...
78
  #include <linux/quotaops.h>
fb58b7316   Christoph Hellwig   [PATCH] move remo...
79
  #include <linux/writeback.h> /* for inode_lock, oddly enough.. */
8e8934695   Jan Kara   quota: send messa...
80
81
82
83
  #ifdef CONFIG_QUOTA_NETLINK_INTERFACE
  #include <net/netlink.h>
  #include <net/genetlink.h>
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
84
85
86
87
88
89
  
  #include <asm/uaccess.h>
  
  #define __DQUOT_PARANOIA
  
  /*
cc33412fb   Jan Kara   quota: Improve lo...
90
91
92
93
   * There are three quota SMP locks. dq_list_lock protects all lists with quotas
   * and quota formats, dqstats structure containing statistics about the lists
   * dq_data_lock protects data from dq_dqb and also mem_dqinfo structures and
   * also guards consistency of dquot->dq_dqb with inode->i_blocks, i_bytes.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
94
   * i_blocks and i_bytes updates itself are guarded by i_lock acquired directly
cc33412fb   Jan Kara   quota: Improve lo...
95
96
97
   * in inode_add_bytes() and inode_sub_bytes(). dq_state_lock protects
   * modifications of quota state (on quotaon and quotaoff) and readers who care
   * about latest values take it as well.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
98
   *
cc33412fb   Jan Kara   quota: Improve lo...
99
100
   * The spinlock ordering is hence: dq_data_lock > dq_list_lock > i_lock,
   *   dq_list_lock > dq_state_lock
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
101
102
103
104
105
106
107
108
   *
   * Note that some things (eg. sb pointer, type, id) doesn't change during
   * the life of the dquot structure and so needn't to be protected by a lock
   *
   * Any operation working on dquots via inode pointers must hold dqptr_sem.  If
   * operation is just reading pointers from inode (or not using them at all) the
   * read lock is enough. If pointers are altered function must hold write lock
   * (these locking rules also apply for S_NOQUOTA flag in the inode - note that
cc33412fb   Jan Kara   quota: Improve lo...
109
   * for altering the flag i_mutex is also needed).
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
110
   *
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
111
   * Each dquot has its dq_lock mutex. Locked dquots might not be referenced
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
112
113
114
115
116
117
118
119
120
   * from inodes (dquot_alloc_space() and such don't check the dq_lock).
   * Currently dquot is locked only when it is being read to memory (or space for
   * it is being allocated) on the first dqget() and when it is being released on
   * the last dqput(). The allocation and release oparations are serialized by
   * the dq_lock and by checking the use count in dquot_release().  Write
   * operations on dquots don't hold dq_lock as they copy data under dq_data_lock
   * spinlock to internal buffers before writing.
   *
   * Lock ordering (including related VFS locks) is the following:
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
121
122
   *   i_mutex > dqonoff_sem > journal_lock > dqptr_sem > dquot->dq_lock >
   *   dqio_mutex
cc33412fb   Jan Kara   quota: Improve lo...
123
124
125
126
127
128
   * The lock ordering of dqptr_sem imposed by quota code is only dqonoff_sem >
   * dqptr_sem. But filesystem has to count with the fact that functions such as
   * dquot_alloc_space() acquire dqptr_sem and they usually have to be called
   * from inside a transaction to keep filesystem consistency after a crash. Also
   * filesystems usually want to do some IO on dquot from ->mark_dirty which is
   * called with dqptr_sem held.
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
129
   * i_mutex on quota files is special (it's below dqio_mutex)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
130
131
132
   */
  
  static DEFINE_SPINLOCK(dq_list_lock);
cc33412fb   Jan Kara   quota: Improve lo...
133
  static DEFINE_SPINLOCK(dq_state_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
134
135
136
137
138
139
140
  DEFINE_SPINLOCK(dq_data_lock);
  
  static char *quotatypes[] = INITQFNAMES;
  static struct quota_format_type *quota_formats;	/* List of registered formats */
  static struct quota_module_name module_names[] = INIT_QUOTA_MODULE_NAMES;
  
  /* SLAB cache for dquot structures */
e18b890bb   Christoph Lameter   [PATCH] slab: rem...
141
  static struct kmem_cache *dquot_cachep;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
142
143
144
145
146
147
148
149
150
151
152
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
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
  
  int register_quota_format(struct quota_format_type *fmt)
  {
  	spin_lock(&dq_list_lock);
  	fmt->qf_next = quota_formats;
  	quota_formats = fmt;
  	spin_unlock(&dq_list_lock);
  	return 0;
  }
  
  void unregister_quota_format(struct quota_format_type *fmt)
  {
  	struct quota_format_type **actqf;
  
  	spin_lock(&dq_list_lock);
  	for (actqf = &quota_formats; *actqf && *actqf != fmt; actqf = &(*actqf)->qf_next);
  	if (*actqf)
  		*actqf = (*actqf)->qf_next;
  	spin_unlock(&dq_list_lock);
  }
  
  static struct quota_format_type *find_quota_format(int id)
  {
  	struct quota_format_type *actqf;
  
  	spin_lock(&dq_list_lock);
  	for (actqf = quota_formats; actqf && actqf->qf_fmt_id != id; actqf = actqf->qf_next);
  	if (!actqf || !try_module_get(actqf->qf_owner)) {
  		int qm;
  
  		spin_unlock(&dq_list_lock);
  		
  		for (qm = 0; module_names[qm].qm_fmt_id && module_names[qm].qm_fmt_id != id; qm++);
  		if (!module_names[qm].qm_fmt_id || request_module(module_names[qm].qm_mod_name))
  			return NULL;
  
  		spin_lock(&dq_list_lock);
  		for (actqf = quota_formats; actqf && actqf->qf_fmt_id != id; actqf = actqf->qf_next);
  		if (actqf && !try_module_get(actqf->qf_owner))
  			actqf = NULL;
  	}
  	spin_unlock(&dq_list_lock);
  	return actqf;
  }
  
  static void put_quota_format(struct quota_format_type *fmt)
  {
  	module_put(fmt->qf_owner);
  }
  
  /*
   * Dquot List Management:
   * The quota code uses three lists for dquot management: the inuse_list,
   * free_dquots, and dquot_hash[] array. A single dquot structure may be
   * on all three lists, depending on its current state.
   *
   * All dquots are placed to the end of inuse_list when first created, and this
   * list is used for invalidate operation, which must look at every dquot.
   *
   * Unused dquots (dq_count == 0) are added to the free_dquots list when freed,
   * and this list is searched whenever we need an available dquot.  Dquots are
   * removed from the list as soon as they are used again, and
   * dqstats.free_dquots gives the number of dquots on the list. When
   * dquot is invalidated it's completely released from memory.
   *
   * Dquots with a specific identity (device, type and id) are placed on
   * one of the dquot_hash[] hash chains. The provides an efficient search
   * mechanism to locate a specific dquot.
   */
  
  static LIST_HEAD(inuse_list);
  static LIST_HEAD(free_dquots);
  static unsigned int dq_hash_bits, dq_hash_mask;
  static struct hlist_head *dquot_hash;
  
  struct dqstats dqstats;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
  static inline unsigned int
  hashfn(const struct super_block *sb, unsigned int id, int type)
  {
  	unsigned long tmp;
  
  	tmp = (((unsigned long)sb>>L1_CACHE_SHIFT) ^ id) * (MAXQUOTAS - type);
  	return (tmp + (tmp >> dq_hash_bits)) & dq_hash_mask;
  }
  
  /*
   * Following list functions expect dq_list_lock to be held
   */
  static inline void insert_dquot_hash(struct dquot *dquot)
  {
  	struct hlist_head *head = dquot_hash + hashfn(dquot->dq_sb, dquot->dq_id, dquot->dq_type);
  	hlist_add_head(&dquot->dq_hash, head);
  }
  
  static inline void remove_dquot_hash(struct dquot *dquot)
  {
  	hlist_del_init(&dquot->dq_hash);
  }
  
  static inline struct dquot *find_dquot(unsigned int hashent, struct super_block *sb, unsigned int id, int type)
  {
  	struct hlist_node *node;
  	struct dquot *dquot;
  
  	hlist_for_each (node, dquot_hash+hashent) {
  		dquot = hlist_entry(node, struct dquot, dq_hash);
  		if (dquot->dq_sb == sb && dquot->dq_id == id && dquot->dq_type == type)
  			return dquot;
  	}
  	return NODQUOT;
  }
  
  /* Add a dquot to the tail of the free list */
  static inline void put_dquot_last(struct dquot *dquot)
  {
8e13059a3   Akinobu Mita   [PATCH] use list_...
257
  	list_add_tail(&dquot->dq_free, &free_dquots);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
  	dqstats.free_dquots++;
  }
  
  static inline void remove_free_dquot(struct dquot *dquot)
  {
  	if (list_empty(&dquot->dq_free))
  		return;
  	list_del_init(&dquot->dq_free);
  	dqstats.free_dquots--;
  }
  
  static inline void put_inuse(struct dquot *dquot)
  {
  	/* We add to the back of inuse list so we don't have to restart
  	 * when traversing this list and we block */
8e13059a3   Akinobu Mita   [PATCH] use list_...
273
  	list_add_tail(&dquot->dq_inuse, &inuse_list);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
274
275
276
277
278
279
280
281
282
283
284
285
286
287
  	dqstats.allocated_dquots++;
  }
  
  static inline void remove_inuse(struct dquot *dquot)
  {
  	dqstats.allocated_dquots--;
  	list_del(&dquot->dq_inuse);
  }
  /*
   * End of list functions needing dq_list_lock
   */
  
  static void wait_on_dquot(struct dquot *dquot)
  {
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
288
289
  	mutex_lock(&dquot->dq_lock);
  	mutex_unlock(&dquot->dq_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
290
  }
03f6e92bd   Jan Kara   quota: various st...
291
292
293
294
295
296
297
298
299
  static inline int dquot_dirty(struct dquot *dquot)
  {
  	return test_bit(DQ_MOD_B, &dquot->dq_flags);
  }
  
  static inline int mark_dquot_dirty(struct dquot *dquot)
  {
  	return dquot->dq_sb->dq_op->mark_dirty(dquot);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
  
  int dquot_mark_dquot_dirty(struct dquot *dquot)
  {
  	spin_lock(&dq_list_lock);
  	if (!test_and_set_bit(DQ_MOD_B, &dquot->dq_flags))
  		list_add(&dquot->dq_dirty, &sb_dqopt(dquot->dq_sb)->
  				info[dquot->dq_type].dqi_dirty_list);
  	spin_unlock(&dq_list_lock);
  	return 0;
  }
  
  /* This function needs dq_list_lock */
  static inline int clear_dquot_dirty(struct dquot *dquot)
  {
  	if (!test_and_clear_bit(DQ_MOD_B, &dquot->dq_flags))
  		return 0;
  	list_del_init(&dquot->dq_dirty);
  	return 1;
  }
  
  void mark_info_dirty(struct super_block *sb, int type)
  {
  	set_bit(DQF_INFO_DIRTY_B, &sb_dqopt(sb)->info[type].dqi_flags);
  }
  EXPORT_SYMBOL(mark_info_dirty);
  
  /*
   *	Read dquot from disk and alloc space for it
   */
  
  int dquot_acquire(struct dquot *dquot)
  {
  	int ret = 0, ret2 = 0;
  	struct quota_info *dqopt = sb_dqopt(dquot->dq_sb);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
334
335
  	mutex_lock(&dquot->dq_lock);
  	mutex_lock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
  	if (!test_bit(DQ_READ_B, &dquot->dq_flags))
  		ret = dqopt->ops[dquot->dq_type]->read_dqblk(dquot);
  	if (ret < 0)
  		goto out_iolock;
  	set_bit(DQ_READ_B, &dquot->dq_flags);
  	/* Instantiate dquot if needed */
  	if (!test_bit(DQ_ACTIVE_B, &dquot->dq_flags) && !dquot->dq_off) {
  		ret = dqopt->ops[dquot->dq_type]->commit_dqblk(dquot);
  		/* Write the info if needed */
  		if (info_dirty(&dqopt->info[dquot->dq_type]))
  			ret2 = dqopt->ops[dquot->dq_type]->write_file_info(dquot->dq_sb, dquot->dq_type);
  		if (ret < 0)
  			goto out_iolock;
  		if (ret2 < 0) {
  			ret = ret2;
  			goto out_iolock;
  		}
  	}
  	set_bit(DQ_ACTIVE_B, &dquot->dq_flags);
  out_iolock:
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
356
357
  	mutex_unlock(&dqopt->dqio_mutex);
  	mutex_unlock(&dquot->dq_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
358
359
360
361
362
363
364
365
366
367
  	return ret;
  }
  
  /*
   *	Write dquot to disk
   */
  int dquot_commit(struct dquot *dquot)
  {
  	int ret = 0, ret2 = 0;
  	struct quota_info *dqopt = sb_dqopt(dquot->dq_sb);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
368
  	mutex_lock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
  	spin_lock(&dq_list_lock);
  	if (!clear_dquot_dirty(dquot)) {
  		spin_unlock(&dq_list_lock);
  		goto out_sem;
  	}
  	spin_unlock(&dq_list_lock);
  	/* Inactive dquot can be only if there was error during read/init
  	 * => we have better not writing it */
  	if (test_bit(DQ_ACTIVE_B, &dquot->dq_flags)) {
  		ret = dqopt->ops[dquot->dq_type]->commit_dqblk(dquot);
  		if (info_dirty(&dqopt->info[dquot->dq_type]))
  			ret2 = dqopt->ops[dquot->dq_type]->write_file_info(dquot->dq_sb, dquot->dq_type);
  		if (ret >= 0)
  			ret = ret2;
  	}
  out_sem:
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
385
  	mutex_unlock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
386
387
388
389
390
391
392
393
394
395
  	return ret;
  }
  
  /*
   *	Release dquot
   */
  int dquot_release(struct dquot *dquot)
  {
  	int ret = 0, ret2 = 0;
  	struct quota_info *dqopt = sb_dqopt(dquot->dq_sb);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
396
  	mutex_lock(&dquot->dq_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
397
398
399
  	/* Check whether we are not racing with some other dqget() */
  	if (atomic_read(&dquot->dq_count) > 1)
  		goto out_dqlock;
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
400
  	mutex_lock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
401
402
403
404
405
406
407
408
409
  	if (dqopt->ops[dquot->dq_type]->release_dqblk) {
  		ret = dqopt->ops[dquot->dq_type]->release_dqblk(dquot);
  		/* Write the info */
  		if (info_dirty(&dqopt->info[dquot->dq_type]))
  			ret2 = dqopt->ops[dquot->dq_type]->write_file_info(dquot->dq_sb, dquot->dq_type);
  		if (ret >= 0)
  			ret = ret2;
  	}
  	clear_bit(DQ_ACTIVE_B, &dquot->dq_flags);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
410
  	mutex_unlock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
411
  out_dqlock:
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
412
  	mutex_unlock(&dquot->dq_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
413
414
  	return ret;
  }
7d9056ba2   Jan Kara   quota: Export dqu...
415
  void dquot_destroy(struct dquot *dquot)
74f783af9   Jan Kara   quota: Add callba...
416
417
418
  {
  	kmem_cache_free(dquot_cachep, dquot);
  }
7d9056ba2   Jan Kara   quota: Export dqu...
419
  EXPORT_SYMBOL(dquot_destroy);
74f783af9   Jan Kara   quota: Add callba...
420
421
422
423
424
  
  static inline void do_destroy_dquot(struct dquot *dquot)
  {
  	dquot->dq_sb->dq_op->destroy_dquot(dquot);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
425
426
  /* Invalidate all dquots on the list. Note that this function is called after
   * quota is disabled and pointers from inodes removed so there cannot be new
6362e4d4e   Jan Kara   [PATCH] Fix oops ...
427
428
   * quota users. There can still be some users of quotas due to inodes being
   * just deleted or pruned by prune_icache() (those are not attached to any
cc33412fb   Jan Kara   quota: Improve lo...
429
   * list) or parallel quotactl call. We have to wait for such users.
6362e4d4e   Jan Kara   [PATCH] Fix oops ...
430
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
431
432
  static void invalidate_dquots(struct super_block *sb, int type)
  {
c33ed2712   Domen Puncer   [PATCH] list_for_...
433
  	struct dquot *dquot, *tmp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
434

6362e4d4e   Jan Kara   [PATCH] Fix oops ...
435
  restart:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
436
  	spin_lock(&dq_list_lock);
c33ed2712   Domen Puncer   [PATCH] list_for_...
437
  	list_for_each_entry_safe(dquot, tmp, &inuse_list, dq_inuse) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
438
439
440
441
  		if (dquot->dq_sb != sb)
  			continue;
  		if (dquot->dq_type != type)
  			continue;
6362e4d4e   Jan Kara   [PATCH] Fix oops ...
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
  		/* Wait for dquot users */
  		if (atomic_read(&dquot->dq_count)) {
  			DEFINE_WAIT(wait);
  
  			atomic_inc(&dquot->dq_count);
  			prepare_to_wait(&dquot->dq_wait_unused, &wait,
  					TASK_UNINTERRUPTIBLE);
  			spin_unlock(&dq_list_lock);
  			/* Once dqput() wakes us up, we know it's time to free
  			 * the dquot.
  			 * IMPORTANT: we rely on the fact that there is always
  			 * at most one process waiting for dquot to free.
  			 * Otherwise dq_count would be > 1 and we would never
  			 * wake up.
  			 */
  			if (atomic_read(&dquot->dq_count) > 1)
  				schedule();
  			finish_wait(&dquot->dq_wait_unused, &wait);
  			dqput(dquot);
  			/* At this moment dquot() need not exist (it could be
  			 * reclaimed by prune_dqcache(). Hence we must
  			 * restart. */
  			goto restart;
  		}
  		/*
  		 * Quota now has no users and it has been written on last
  		 * dqput()
  		 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
470
471
472
  		remove_dquot_hash(dquot);
  		remove_free_dquot(dquot);
  		remove_inuse(dquot);
74f783af9   Jan Kara   quota: Add callba...
473
  		do_destroy_dquot(dquot);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
474
475
476
  	}
  	spin_unlock(&dq_list_lock);
  }
12c77527e   Jan Kara   quota: Implement ...
477
478
479
480
481
482
483
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
  /* Call callback for every active dquot on given filesystem */
  int dquot_scan_active(struct super_block *sb,
  		      int (*fn)(struct dquot *dquot, unsigned long priv),
  		      unsigned long priv)
  {
  	struct dquot *dquot, *old_dquot = NULL;
  	int ret = 0;
  
  	mutex_lock(&sb_dqopt(sb)->dqonoff_mutex);
  	spin_lock(&dq_list_lock);
  	list_for_each_entry(dquot, &inuse_list, dq_inuse) {
  		if (!test_bit(DQ_ACTIVE_B, &dquot->dq_flags))
  			continue;
  		if (dquot->dq_sb != sb)
  			continue;
  		/* Now we have active dquot so we can just increase use count */
  		atomic_inc(&dquot->dq_count);
  		dqstats.lookups++;
  		spin_unlock(&dq_list_lock);
  		dqput(old_dquot);
  		old_dquot = dquot;
  		ret = fn(dquot, priv);
  		if (ret < 0)
  			goto out;
  		spin_lock(&dq_list_lock);
  		/* We are safe to continue now because our dquot could not
  		 * be moved out of the inuse list while we hold the reference */
  	}
  	spin_unlock(&dq_list_lock);
  out:
  	dqput(old_dquot);
  	mutex_unlock(&sb_dqopt(sb)->dqonoff_mutex);
  	return ret;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
511
512
513
514
515
516
  int vfs_quota_sync(struct super_block *sb, int type)
  {
  	struct list_head *dirty;
  	struct dquot *dquot;
  	struct quota_info *dqopt = sb_dqopt(sb);
  	int cnt;
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
517
  	mutex_lock(&dqopt->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
518
519
520
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (type != -1 && cnt != type)
  			continue;
f55abc0fb   Jan Kara   quota: Allow to s...
521
  		if (!sb_has_quota_active(sb, cnt))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
522
523
524
525
  			continue;
  		spin_lock(&dq_list_lock);
  		dirty = &dqopt->info[cnt].dqi_dirty_list;
  		while (!list_empty(dirty)) {
b5e618181   Pavel Emelianov   Introduce a handy...
526
  			dquot = list_first_entry(dirty, struct dquot, dq_dirty);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
  			/* Dirty and inactive can be only bad dquot... */
  			if (!test_bit(DQ_ACTIVE_B, &dquot->dq_flags)) {
  				clear_dquot_dirty(dquot);
  				continue;
  			}
  			/* Now we have active dquot from which someone is
   			 * holding reference so we can safely just increase
  			 * use count */
  			atomic_inc(&dquot->dq_count);
  			dqstats.lookups++;
  			spin_unlock(&dq_list_lock);
  			sb->dq_op->write_dquot(dquot);
  			dqput(dquot);
  			spin_lock(&dq_list_lock);
  		}
  		spin_unlock(&dq_list_lock);
  	}
  
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
f55abc0fb   Jan Kara   quota: Allow to s...
546
547
  		if ((cnt == type || type == -1) && sb_has_quota_active(sb, cnt)
  		    && info_dirty(&dqopt->info[cnt]))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
548
549
550
551
  			sb->dq_op->write_info(sb, cnt);
  	spin_lock(&dq_list_lock);
  	dqstats.syncs++;
  	spin_unlock(&dq_list_lock);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
552
  	mutex_unlock(&dqopt->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
  
  	return 0;
  }
  
  /* Free unused dquots from cache */
  static void prune_dqcache(int count)
  {
  	struct list_head *head;
  	struct dquot *dquot;
  
  	head = free_dquots.prev;
  	while (head != &free_dquots && count) {
  		dquot = list_entry(head, struct dquot, dq_free);
  		remove_dquot_hash(dquot);
  		remove_free_dquot(dquot);
  		remove_inuse(dquot);
74f783af9   Jan Kara   quota: Add callba...
569
  		do_destroy_dquot(dquot);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
570
571
572
573
574
575
576
577
578
  		count--;
  		head = free_dquots.prev;
  	}
  }
  
  /*
   * This is called from kswapd when we think we need some
   * more memory
   */
27496a8c6   Al Viro   [PATCH] gfp_t: fs/*
579
  static int shrink_dqcache_memory(int nr, gfp_t gfp_mask)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
580
581
582
583
584
585
586
587
  {
  	if (nr) {
  		spin_lock(&dq_list_lock);
  		prune_dqcache(nr);
  		spin_unlock(&dq_list_lock);
  	}
  	return (dqstats.free_dquots / 100) * sysctl_vfs_cache_pressure;
  }
8e1f936b7   Rusty Russell   mm: clean up and ...
588
589
590
591
  static struct shrinker dqcache_shrinker = {
  	.shrink = shrink_dqcache_memory,
  	.seeks = DEFAULT_SEEKS,
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
592
593
594
  /*
   * Put reference to dquot
   * NOTE: If you change this function please check whether dqput_blocks() works right...
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
595
   */
3d9ea253a   Jan Kara   quota: Add helper...
596
  void dqput(struct dquot *dquot)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
597
  {
b48d38054   Jan Kara   quota: fix possib...
598
  	int ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
  	if (!dquot)
  		return;
  #ifdef __DQUOT_PARANOIA
  	if (!atomic_read(&dquot->dq_count)) {
  		printk("VFS: dqput: trying to free free dquot
  ");
  		printk("VFS: device %s, dquot of %s %d
  ",
  			dquot->dq_sb->s_id,
  			quotatypes[dquot->dq_type],
  			dquot->dq_id);
  		BUG();
  	}
  #endif
  	
  	spin_lock(&dq_list_lock);
  	dqstats.drops++;
  	spin_unlock(&dq_list_lock);
  we_slept:
  	spin_lock(&dq_list_lock);
  	if (atomic_read(&dquot->dq_count) > 1) {
  		/* We have more than one user... nothing to do */
  		atomic_dec(&dquot->dq_count);
6362e4d4e   Jan Kara   [PATCH] Fix oops ...
622
  		/* Releasing dquot during quotaoff phase? */
f55abc0fb   Jan Kara   quota: Allow to s...
623
  		if (!sb_has_quota_active(dquot->dq_sb, dquot->dq_type) &&
6362e4d4e   Jan Kara   [PATCH] Fix oops ...
624
625
  		    atomic_read(&dquot->dq_count) == 1)
  			wake_up(&dquot->dq_wait_unused);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
626
627
628
629
630
631
632
  		spin_unlock(&dq_list_lock);
  		return;
  	}
  	/* Need to release dquot? */
  	if (test_bit(DQ_ACTIVE_B, &dquot->dq_flags) && dquot_dirty(dquot)) {
  		spin_unlock(&dq_list_lock);
  		/* Commit dquot before releasing */
b48d38054   Jan Kara   quota: fix possib...
633
634
635
636
637
638
639
640
641
642
643
644
645
646
  		ret = dquot->dq_sb->dq_op->write_dquot(dquot);
  		if (ret < 0) {
  			printk(KERN_ERR "VFS: cannot write quota structure on "
  				"device %s (error %d). Quota may get out of "
  				"sync!
  ", dquot->dq_sb->s_id, ret);
  			/*
  			 * We clear dirty bit anyway, so that we avoid
  			 * infinite loop here
  			 */
  			spin_lock(&dq_list_lock);
  			clear_dquot_dirty(dquot);
  			spin_unlock(&dq_list_lock);
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
647
648
649
650
651
652
653
654
655
656
657
658
  		goto we_slept;
  	}
  	/* Clear flag in case dquot was inactive (something bad happened) */
  	clear_dquot_dirty(dquot);
  	if (test_bit(DQ_ACTIVE_B, &dquot->dq_flags)) {
  		spin_unlock(&dq_list_lock);
  		dquot->dq_sb->dq_op->release_dquot(dquot);
  		goto we_slept;
  	}
  	atomic_dec(&dquot->dq_count);
  #ifdef __DQUOT_PARANOIA
  	/* sanity check */
8abf6a470   Eric Sesterhenn   BUG_ON() Conversi...
659
  	BUG_ON(!list_empty(&dquot->dq_free));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
660
661
662
663
  #endif
  	put_dquot_last(dquot);
  	spin_unlock(&dq_list_lock);
  }
7d9056ba2   Jan Kara   quota: Export dqu...
664
  struct dquot *dquot_alloc(struct super_block *sb, int type)
74f783af9   Jan Kara   quota: Add callba...
665
666
667
  {
  	return kmem_cache_zalloc(dquot_cachep, GFP_NOFS);
  }
7d9056ba2   Jan Kara   quota: Export dqu...
668
  EXPORT_SYMBOL(dquot_alloc);
74f783af9   Jan Kara   quota: Add callba...
669

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
670
671
672
  static struct dquot *get_empty_dquot(struct super_block *sb, int type)
  {
  	struct dquot *dquot;
74f783af9   Jan Kara   quota: Add callba...
673
  	dquot = sb->dq_op->alloc_dquot(sb, type);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
674
675
  	if(!dquot)
  		return NODQUOT;
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
676
  	mutex_init(&dquot->dq_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
677
678
679
680
  	INIT_LIST_HEAD(&dquot->dq_free);
  	INIT_LIST_HEAD(&dquot->dq_inuse);
  	INIT_HLIST_NODE(&dquot->dq_hash);
  	INIT_LIST_HEAD(&dquot->dq_dirty);
6362e4d4e   Jan Kara   [PATCH] Fix oops ...
681
  	init_waitqueue_head(&dquot->dq_wait_unused);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
682
683
684
685
686
687
688
689
690
  	dquot->dq_sb = sb;
  	dquot->dq_type = type;
  	atomic_set(&dquot->dq_count, 1);
  
  	return dquot;
  }
  
  /*
   * Get reference to dquot
cc33412fb   Jan Kara   quota: Improve lo...
691
692
693
694
695
   *
   * Locking is slightly tricky here. We are guarded from parallel quotaoff()
   * destroying our dquot by:
   *   a) checking for quota flags under dq_list_lock and
   *   b) getting a reference to dquot before we release dq_list_lock
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
696
   */
3d9ea253a   Jan Kara   quota: Add helper...
697
  struct dquot *dqget(struct super_block *sb, unsigned int id, int type)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
698
699
  {
  	unsigned int hashent = hashfn(sb, id, type);
cc33412fb   Jan Kara   quota: Improve lo...
700
  	struct dquot *dquot = NODQUOT, *empty = NODQUOT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
701

f55abc0fb   Jan Kara   quota: Allow to s...
702
          if (!sb_has_quota_active(sb, type))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
703
704
705
  		return NODQUOT;
  we_slept:
  	spin_lock(&dq_list_lock);
cc33412fb   Jan Kara   quota: Improve lo...
706
707
708
709
710
711
712
  	spin_lock(&dq_state_lock);
  	if (!sb_has_quota_active(sb, type)) {
  		spin_unlock(&dq_state_lock);
  		spin_unlock(&dq_list_lock);
  		goto out;
  	}
  	spin_unlock(&dq_state_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
713
714
715
716
717
718
719
720
  	if ((dquot = find_dquot(hashent, sb, id, type)) == NODQUOT) {
  		if (empty == NODQUOT) {
  			spin_unlock(&dq_list_lock);
  			if ((empty = get_empty_dquot(sb, type)) == NODQUOT)
  				schedule();	/* Try to wait for a moment... */
  			goto we_slept;
  		}
  		dquot = empty;
cc33412fb   Jan Kara   quota: Improve lo...
721
  		empty = NODQUOT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
722
723
724
725
726
727
728
729
730
731
732
733
734
735
  		dquot->dq_id = id;
  		/* all dquots go on the inuse_list */
  		put_inuse(dquot);
  		/* hash it first so it can be found */
  		insert_dquot_hash(dquot);
  		dqstats.lookups++;
  		spin_unlock(&dq_list_lock);
  	} else {
  		if (!atomic_read(&dquot->dq_count))
  			remove_free_dquot(dquot);
  		atomic_inc(&dquot->dq_count);
  		dqstats.cache_hits++;
  		dqstats.lookups++;
  		spin_unlock(&dq_list_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
736
737
738
739
740
741
742
  	}
  	/* Wait for dq_lock - after this we know that either dquot_release() is already
  	 * finished or it will be canceled due to dq_count > 1 test */
  	wait_on_dquot(dquot);
  	/* Read the dquot and instantiate it (everything done only if needed) */
  	if (!test_bit(DQ_ACTIVE_B, &dquot->dq_flags) && sb->dq_op->acquire_dquot(dquot) < 0) {
  		dqput(dquot);
cc33412fb   Jan Kara   quota: Improve lo...
743
744
  		dquot = NODQUOT;
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
745
746
  	}
  #ifdef __DQUOT_PARANOIA
8abf6a470   Eric Sesterhenn   BUG_ON() Conversi...
747
  	BUG_ON(!dquot->dq_sb);	/* Has somebody invalidated entry under us? */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
748
  #endif
cc33412fb   Jan Kara   quota: Improve lo...
749
750
751
  out:
  	if (empty)
  		do_destroy_dquot(empty);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
  
  	return dquot;
  }
  
  static int dqinit_needed(struct inode *inode, int type)
  {
  	int cnt;
  
  	if (IS_NOQUOTA(inode))
  		return 0;
  	if (type != -1)
  		return inode->i_dquot[type] == NODQUOT;
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		if (inode->i_dquot[cnt] == NODQUOT)
  			return 1;
  	return 0;
  }
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
769
  /* This routine is guarded by dqonoff_mutex mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
770
771
  static void add_dquot_ref(struct super_block *sb, int type)
  {
941d2380e   Jan Kara   quota: improve in...
772
  	struct inode *inode, *old_inode = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
773

d003fb70f   Christoph Hellwig   [PATCH] remove sb...
774
775
776
777
778
779
780
781
782
783
784
  	spin_lock(&inode_lock);
  	list_for_each_entry(inode, &sb->s_inodes, i_sb_list) {
  		if (!atomic_read(&inode->i_writecount))
  			continue;
  		if (!dqinit_needed(inode, type))
  			continue;
  		if (inode->i_state & (I_FREEING|I_WILL_FREE))
  			continue;
  
  		__iget(inode);
  		spin_unlock(&inode_lock);
941d2380e   Jan Kara   quota: improve in...
785
  		iput(old_inode);
d003fb70f   Christoph Hellwig   [PATCH] remove sb...
786
  		sb->dq_op->initialize(inode, type);
941d2380e   Jan Kara   quota: improve in...
787
788
789
790
791
792
793
  		/* We hold a reference to 'inode' so it couldn't have been
  		 * removed from s_inodes list while we dropped the inode_lock.
  		 * We cannot iput the inode now as we can be holding the last
  		 * reference and we cannot iput it under inode_lock. So we
  		 * keep the reference and iput it later. */
  		old_inode = inode;
  		spin_lock(&inode_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
794
  	}
d003fb70f   Christoph Hellwig   [PATCH] remove sb...
795
  	spin_unlock(&inode_lock);
941d2380e   Jan Kara   quota: improve in...
796
  	iput(old_inode);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
797
798
799
800
801
802
803
804
805
806
807
808
  }
  
  /* Return 0 if dqput() won't block (note that 1 doesn't necessarily mean blocking) */
  static inline int dqput_blocks(struct dquot *dquot)
  {
  	if (atomic_read(&dquot->dq_count) <= 1)
  		return 1;
  	return 0;
  }
  
  /* Remove references to dquots from inode - add dquot to list for freeing if needed */
  /* We can't race with anybody because we hold dqptr_sem for writing... */
e5f00f42f   Adrian Bunk   make remove_inode...
809
810
  static int remove_inode_dquot_ref(struct inode *inode, int type,
  				  struct list_head *tofree_head)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
  {
  	struct dquot *dquot = inode->i_dquot[type];
  
  	inode->i_dquot[type] = NODQUOT;
  	if (dquot != NODQUOT) {
  		if (dqput_blocks(dquot)) {
  #ifdef __DQUOT_PARANOIA
  			if (atomic_read(&dquot->dq_count) != 1)
  				printk(KERN_WARNING "VFS: Adding dquot with dq_count %d to dispose list.
  ", atomic_read(&dquot->dq_count));
  #endif
  			spin_lock(&dq_list_lock);
  			list_add(&dquot->dq_free, tofree_head);	/* As dquot must have currently users it can't be on the free list... */
  			spin_unlock(&dq_list_lock);
  			return 1;
  		}
  		else
  			dqput(dquot);   /* We have guaranteed we won't block */
  	}
  	return 0;
  }
  
  /* Free list of dquots - called from inode.c */
  /* dquots are removed from inodes, no new references can be got so we are the only ones holding reference */
  static void put_dquot_list(struct list_head *tofree_head)
  {
  	struct list_head *act_head;
  	struct dquot *dquot;
  
  	act_head = tofree_head->next;
  	/* So now we have dquots on the list... Just free them */
  	while (act_head != tofree_head) {
  		dquot = list_entry(act_head, struct dquot, dq_free);
  		act_head = act_head->next;
  		list_del_init(&dquot->dq_free);	/* Remove dquot from the list so we won't have problems... */
  		dqput(dquot);
  	}
  }
fb58b7316   Christoph Hellwig   [PATCH] move remo...
849
850
851
852
853
854
855
856
857
858
859
860
  static void remove_dquot_ref(struct super_block *sb, int type,
  		struct list_head *tofree_head)
  {
  	struct inode *inode;
  
  	spin_lock(&inode_lock);
  	list_for_each_entry(inode, &sb->s_inodes, i_sb_list) {
  		if (!IS_NOQUOTA(inode))
  			remove_inode_dquot_ref(inode, type, tofree_head);
  	}
  	spin_unlock(&inode_lock);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
861
862
863
864
  /* Gather all references from inodes and drop them */
  static void drop_dquot_ref(struct super_block *sb, int type)
  {
  	LIST_HEAD(tofree_head);
fb58b7316   Christoph Hellwig   [PATCH] move remo...
865
866
867
868
869
870
  	if (sb->dq_op) {
  		down_write(&sb_dqopt(sb)->dqptr_sem);
  		remove_dquot_ref(sb, type, &tofree_head);
  		up_write(&sb_dqopt(sb)->dqptr_sem);
  		put_dquot_list(&tofree_head);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
871
  }
12095460f   Jan Kara   quota: Increase s...
872
  static inline void dquot_incr_inodes(struct dquot *dquot, qsize_t number)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
873
874
875
876
877
878
879
880
  {
  	dquot->dq_dqb.dqb_curinodes += number;
  }
  
  static inline void dquot_incr_space(struct dquot *dquot, qsize_t number)
  {
  	dquot->dq_dqb.dqb_curspace += number;
  }
12095460f   Jan Kara   quota: Increase s...
881
  static inline void dquot_decr_inodes(struct dquot *dquot, qsize_t number)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
882
  {
db49d2df4   Jan Kara   quota: Allow nega...
883
884
  	if (sb_dqopt(dquot->dq_sb)->flags & DQUOT_NEGATIVE_USAGE ||
  	    dquot->dq_dqb.dqb_curinodes >= number)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
885
886
887
888
889
890
891
892
893
894
  		dquot->dq_dqb.dqb_curinodes -= number;
  	else
  		dquot->dq_dqb.dqb_curinodes = 0;
  	if (dquot->dq_dqb.dqb_curinodes <= dquot->dq_dqb.dqb_isoftlimit)
  		dquot->dq_dqb.dqb_itime = (time_t) 0;
  	clear_bit(DQ_INODES_B, &dquot->dq_flags);
  }
  
  static inline void dquot_decr_space(struct dquot *dquot, qsize_t number)
  {
db49d2df4   Jan Kara   quota: Allow nega...
895
896
  	if (sb_dqopt(dquot->dq_sb)->flags & DQUOT_NEGATIVE_USAGE ||
  	    dquot->dq_dqb.dqb_curspace >= number)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
897
898
899
  		dquot->dq_dqb.dqb_curspace -= number;
  	else
  		dquot->dq_dqb.dqb_curspace = 0;
12095460f   Jan Kara   quota: Increase s...
900
  	if (dquot->dq_dqb.dqb_curspace <= dquot->dq_dqb.dqb_bsoftlimit)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
901
902
903
  		dquot->dq_dqb.dqb_btime = (time_t) 0;
  	clear_bit(DQ_BLKS_B, &dquot->dq_flags);
  }
c525460e2   Jan Kara   Don't send quota ...
904
905
906
907
908
909
910
911
912
913
914
  static int warning_issued(struct dquot *dquot, const int warntype)
  {
  	int flag = (warntype == QUOTA_NL_BHARDWARN ||
  		warntype == QUOTA_NL_BSOFTLONGWARN) ? DQ_BLKS_B :
  		((warntype == QUOTA_NL_IHARDWARN ||
  		warntype == QUOTA_NL_ISOFTLONGWARN) ? DQ_INODES_B : 0);
  
  	if (!flag)
  		return 0;
  	return test_and_set_bit(flag, &dquot->dq_flags);
  }
8e8934695   Jan Kara   quota: send messa...
915
  #ifdef CONFIG_PRINT_QUOTA_WARNING
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
916
917
918
919
920
921
922
923
924
  static int flag_print_warnings = 1;
  
  static inline int need_print_warning(struct dquot *dquot)
  {
  	if (!flag_print_warnings)
  		return 0;
  
  	switch (dquot->dq_type) {
  		case USRQUOTA:
da9592ede   David Howells   CRED: Wrap task c...
925
  			return current_fsuid() == dquot->dq_id;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
926
927
928
929
930
  		case GRPQUOTA:
  			return in_group_p(dquot->dq_id);
  	}
  	return 0;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
931
  /* Print warning to user which exceeded quota */
c525460e2   Jan Kara   Don't send quota ...
932
  static void print_warning(struct dquot *dquot, const int warntype)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
933
934
  {
  	char *msg = NULL;
24ec839c4   Peter Zijlstra   [PATCH] tty: ->si...
935
  	struct tty_struct *tty;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
936

657d3bfa9   Jan Kara   quota: implement ...
937
938
939
940
  	if (warntype == QUOTA_NL_IHARDBELOW ||
  	    warntype == QUOTA_NL_ISOFTBELOW ||
  	    warntype == QUOTA_NL_BHARDBELOW ||
  	    warntype == QUOTA_NL_BSOFTBELOW || !need_print_warning(dquot))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
941
  		return;
24ec839c4   Peter Zijlstra   [PATCH] tty: ->si...
942
943
  	tty = get_current_tty();
  	if (!tty)
452a00d2e   Alan Cox   tty: Make get_cur...
944
  		return;
24ec839c4   Peter Zijlstra   [PATCH] tty: ->si...
945
  	tty_write_message(tty, dquot->dq_sb->s_id);
8e8934695   Jan Kara   quota: send messa...
946
  	if (warntype == QUOTA_NL_ISOFTWARN || warntype == QUOTA_NL_BSOFTWARN)
24ec839c4   Peter Zijlstra   [PATCH] tty: ->si...
947
  		tty_write_message(tty, ": warning, ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
948
  	else
24ec839c4   Peter Zijlstra   [PATCH] tty: ->si...
949
950
  		tty_write_message(tty, ": write failed, ");
  	tty_write_message(tty, quotatypes[dquot->dq_type]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
951
  	switch (warntype) {
8e8934695   Jan Kara   quota: send messa...
952
  		case QUOTA_NL_IHARDWARN:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
953
954
955
  			msg = " file limit reached.\r
  ";
  			break;
8e8934695   Jan Kara   quota: send messa...
956
  		case QUOTA_NL_ISOFTLONGWARN:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
957
958
959
  			msg = " file quota exceeded too long.\r
  ";
  			break;
8e8934695   Jan Kara   quota: send messa...
960
  		case QUOTA_NL_ISOFTWARN:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
961
962
963
  			msg = " file quota exceeded.\r
  ";
  			break;
8e8934695   Jan Kara   quota: send messa...
964
  		case QUOTA_NL_BHARDWARN:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
965
966
967
  			msg = " block limit reached.\r
  ";
  			break;
8e8934695   Jan Kara   quota: send messa...
968
  		case QUOTA_NL_BSOFTLONGWARN:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
969
970
971
  			msg = " block quota exceeded too long.\r
  ";
  			break;
8e8934695   Jan Kara   quota: send messa...
972
  		case QUOTA_NL_BSOFTWARN:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
973
974
975
976
  			msg = " block quota exceeded.\r
  ";
  			break;
  	}
24ec839c4   Peter Zijlstra   [PATCH] tty: ->si...
977
  	tty_write_message(tty, msg);
452a00d2e   Alan Cox   tty: Make get_cur...
978
  	tty_kref_put(tty);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
979
  }
8e8934695   Jan Kara   quota: send messa...
980
981
982
  #endif
  
  #ifdef CONFIG_QUOTA_NETLINK_INTERFACE
8e8934695   Jan Kara   quota: send messa...
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
  /* Netlink family structure for quota */
  static struct genl_family quota_genl_family = {
  	.id = GENL_ID_GENERATE,
  	.hdrsize = 0,
  	.name = "VFS_DQUOT",
  	.version = 1,
  	.maxattr = QUOTA_NL_A_MAX,
  };
  
  /* Send warning to userspace about user which exceeded quota */
  static void send_warning(const struct dquot *dquot, const char warntype)
  {
  	static atomic_t seq;
  	struct sk_buff *skb;
  	void *msg_head;
  	int ret;
22dd48372   Jan Kara   Fix computation o...
999
1000
  	int msg_size = 4 * nla_total_size(sizeof(u32)) +
  		       2 * nla_total_size(sizeof(u64));
8e8934695   Jan Kara   quota: send messa...
1001
1002
1003
1004
  
  	/* We have to allocate using GFP_NOFS as we are called from a
  	 * filesystem performing write and thus further recursion into
  	 * the fs to free some data could cause deadlocks. */
22dd48372   Jan Kara   Fix computation o...
1005
  	skb = genlmsg_new(msg_size, GFP_NOFS);
8e8934695   Jan Kara   quota: send messa...
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
  	if (!skb) {
  		printk(KERN_ERR
  		  "VFS: Not enough memory to send quota warning.
  ");
  		return;
  	}
  	msg_head = genlmsg_put(skb, 0, atomic_add_return(1, &seq),
  			&quota_genl_family, 0, QUOTA_NL_C_WARNING);
  	if (!msg_head) {
  		printk(KERN_ERR
  		  "VFS: Cannot store netlink header in quota warning.
  ");
  		goto err_out;
  	}
  	ret = nla_put_u32(skb, QUOTA_NL_A_QTYPE, dquot->dq_type);
  	if (ret)
  		goto attr_err_out;
  	ret = nla_put_u64(skb, QUOTA_NL_A_EXCESS_ID, dquot->dq_id);
  	if (ret)
  		goto attr_err_out;
  	ret = nla_put_u32(skb, QUOTA_NL_A_WARNING, warntype);
  	if (ret)
  		goto attr_err_out;
  	ret = nla_put_u32(skb, QUOTA_NL_A_DEV_MAJOR,
  		MAJOR(dquot->dq_sb->s_dev));
  	if (ret)
  		goto attr_err_out;
  	ret = nla_put_u32(skb, QUOTA_NL_A_DEV_MINOR,
  		MINOR(dquot->dq_sb->s_dev));
  	if (ret)
  		goto attr_err_out;
da9592ede   David Howells   CRED: Wrap task c...
1037
  	ret = nla_put_u64(skb, QUOTA_NL_A_CAUSED_ID, current_uid());
8e8934695   Jan Kara   quota: send messa...
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
  	if (ret)
  		goto attr_err_out;
  	genlmsg_end(skb, msg_head);
  
  	ret = genlmsg_multicast(skb, 0, quota_genl_family.id, GFP_NOFS);
  	if (ret < 0 && ret != -ESRCH)
  		printk(KERN_ERR
  			"VFS: Failed to send notification message: %d
  ", ret);
  	return;
  attr_err_out:
22dd48372   Jan Kara   Fix computation o...
1049
1050
  	printk(KERN_ERR "VFS: Not enough space to compose quota message!
  ");
8e8934695   Jan Kara   quota: send messa...
1051
1052
1053
1054
  err_out:
  	kfree_skb(skb);
  }
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1055

087ee8d5b   Jan Kara   Fix compilation w...
1056
  static inline void flush_warnings(struct dquot * const *dquots, char *warntype)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1057
1058
1059
1060
  {
  	int i;
  
  	for (i = 0; i < MAXQUOTAS; i++)
c525460e2   Jan Kara   Don't send quota ...
1061
1062
  		if (dquots[i] != NODQUOT && warntype[i] != QUOTA_NL_NOWARN &&
  		    !warning_issued(dquots[i], warntype[i])) {
8e8934695   Jan Kara   quota: send messa...
1063
  #ifdef CONFIG_PRINT_QUOTA_WARNING
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1064
  			print_warning(dquots[i], warntype[i]);
8e8934695   Jan Kara   quota: send messa...
1065
1066
1067
1068
1069
  #endif
  #ifdef CONFIG_QUOTA_NETLINK_INTERFACE
  			send_warning(dquots[i], warntype[i]);
  #endif
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
  }
  
  static inline char ignore_hardlimit(struct dquot *dquot)
  {
  	struct mem_dqinfo *info = &sb_dqopt(dquot->dq_sb)->info[dquot->dq_type];
  
  	return capable(CAP_SYS_RESOURCE) &&
  	    (info->dqi_format->qf_fmt_id != QFMT_VFS_OLD || !(info->dqi_flags & V1_DQF_RSQUASH));
  }
  
  /* needs dq_data_lock */
12095460f   Jan Kara   quota: Increase s...
1081
  static int check_idq(struct dquot *dquot, qsize_t inodes, char *warntype)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1082
  {
8e8934695   Jan Kara   quota: send messa...
1083
  	*warntype = QUOTA_NL_NOWARN;
f55abc0fb   Jan Kara   quota: Allow to s...
1084
1085
  	if (!sb_has_quota_limits_enabled(dquot->dq_sb, dquot->dq_type) ||
  	    test_bit(DQ_FAKE_B, &dquot->dq_flags))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1086
1087
1088
1089
1090
  		return QUOTA_OK;
  
  	if (dquot->dq_dqb.dqb_ihardlimit &&
  	   (dquot->dq_dqb.dqb_curinodes + inodes) > dquot->dq_dqb.dqb_ihardlimit &&
              !ignore_hardlimit(dquot)) {
8e8934695   Jan Kara   quota: send messa...
1091
  		*warntype = QUOTA_NL_IHARDWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1092
1093
1094
1095
1096
1097
1098
  		return NO_QUOTA;
  	}
  
  	if (dquot->dq_dqb.dqb_isoftlimit &&
  	   (dquot->dq_dqb.dqb_curinodes + inodes) > dquot->dq_dqb.dqb_isoftlimit &&
  	    dquot->dq_dqb.dqb_itime && get_seconds() >= dquot->dq_dqb.dqb_itime &&
              !ignore_hardlimit(dquot)) {
8e8934695   Jan Kara   quota: send messa...
1099
  		*warntype = QUOTA_NL_ISOFTLONGWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1100
1101
1102
1103
1104
1105
  		return NO_QUOTA;
  	}
  
  	if (dquot->dq_dqb.dqb_isoftlimit &&
  	   (dquot->dq_dqb.dqb_curinodes + inodes) > dquot->dq_dqb.dqb_isoftlimit &&
  	    dquot->dq_dqb.dqb_itime == 0) {
8e8934695   Jan Kara   quota: send messa...
1106
  		*warntype = QUOTA_NL_ISOFTWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1107
1108
1109
1110
1111
1112
1113
1114
1115
  		dquot->dq_dqb.dqb_itime = get_seconds() + sb_dqopt(dquot->dq_sb)->info[dquot->dq_type].dqi_igrace;
  	}
  
  	return QUOTA_OK;
  }
  
  /* needs dq_data_lock */
  static int check_bdq(struct dquot *dquot, qsize_t space, int prealloc, char *warntype)
  {
8e8934695   Jan Kara   quota: send messa...
1116
  	*warntype = QUOTA_NL_NOWARN;
f55abc0fb   Jan Kara   quota: Allow to s...
1117
1118
  	if (!sb_has_quota_limits_enabled(dquot->dq_sb, dquot->dq_type) ||
  	    test_bit(DQ_FAKE_B, &dquot->dq_flags))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1119
1120
1121
  		return QUOTA_OK;
  
  	if (dquot->dq_dqb.dqb_bhardlimit &&
12095460f   Jan Kara   quota: Increase s...
1122
  	    dquot->dq_dqb.dqb_curspace + space > dquot->dq_dqb.dqb_bhardlimit &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1123
1124
              !ignore_hardlimit(dquot)) {
  		if (!prealloc)
8e8934695   Jan Kara   quota: send messa...
1125
  			*warntype = QUOTA_NL_BHARDWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1126
1127
1128
1129
  		return NO_QUOTA;
  	}
  
  	if (dquot->dq_dqb.dqb_bsoftlimit &&
12095460f   Jan Kara   quota: Increase s...
1130
  	    dquot->dq_dqb.dqb_curspace + space > dquot->dq_dqb.dqb_bsoftlimit &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1131
1132
1133
  	    dquot->dq_dqb.dqb_btime && get_seconds() >= dquot->dq_dqb.dqb_btime &&
              !ignore_hardlimit(dquot)) {
  		if (!prealloc)
8e8934695   Jan Kara   quota: send messa...
1134
  			*warntype = QUOTA_NL_BSOFTLONGWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1135
1136
1137
1138
  		return NO_QUOTA;
  	}
  
  	if (dquot->dq_dqb.dqb_bsoftlimit &&
12095460f   Jan Kara   quota: Increase s...
1139
  	    dquot->dq_dqb.dqb_curspace + space > dquot->dq_dqb.dqb_bsoftlimit &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1140
1141
  	    dquot->dq_dqb.dqb_btime == 0) {
  		if (!prealloc) {
8e8934695   Jan Kara   quota: send messa...
1142
  			*warntype = QUOTA_NL_BSOFTWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
  			dquot->dq_dqb.dqb_btime = get_seconds() + sb_dqopt(dquot->dq_sb)->info[dquot->dq_type].dqi_bgrace;
  		}
  		else
  			/*
  			 * We don't allow preallocation to exceed softlimit so exceeding will
  			 * be always printed
  			 */
  			return NO_QUOTA;
  	}
  
  	return QUOTA_OK;
  }
12095460f   Jan Kara   quota: Increase s...
1155
  static int info_idq_free(struct dquot *dquot, qsize_t inodes)
657d3bfa9   Jan Kara   quota: implement ...
1156
1157
  {
  	if (test_bit(DQ_FAKE_B, &dquot->dq_flags) ||
f55abc0fb   Jan Kara   quota: Allow to s...
1158
1159
  	    dquot->dq_dqb.dqb_curinodes <= dquot->dq_dqb.dqb_isoftlimit ||
  	    !sb_has_quota_limits_enabled(dquot->dq_sb, dquot->dq_type))
657d3bfa9   Jan Kara   quota: implement ...
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
  		return QUOTA_NL_NOWARN;
  
  	if (dquot->dq_dqb.dqb_curinodes - inodes <= dquot->dq_dqb.dqb_isoftlimit)
  		return QUOTA_NL_ISOFTBELOW;
  	if (dquot->dq_dqb.dqb_curinodes >= dquot->dq_dqb.dqb_ihardlimit &&
  	    dquot->dq_dqb.dqb_curinodes - inodes < dquot->dq_dqb.dqb_ihardlimit)
  		return QUOTA_NL_IHARDBELOW;
  	return QUOTA_NL_NOWARN;
  }
  
  static int info_bdq_free(struct dquot *dquot, qsize_t space)
  {
  	if (test_bit(DQ_FAKE_B, &dquot->dq_flags) ||
12095460f   Jan Kara   quota: Increase s...
1173
  	    dquot->dq_dqb.dqb_curspace <= dquot->dq_dqb.dqb_bsoftlimit)
657d3bfa9   Jan Kara   quota: implement ...
1174
  		return QUOTA_NL_NOWARN;
12095460f   Jan Kara   quota: Increase s...
1175
  	if (dquot->dq_dqb.dqb_curspace - space <= dquot->dq_dqb.dqb_bsoftlimit)
657d3bfa9   Jan Kara   quota: implement ...
1176
  		return QUOTA_NL_BSOFTBELOW;
12095460f   Jan Kara   quota: Increase s...
1177
1178
  	if (dquot->dq_dqb.dqb_curspace >= dquot->dq_dqb.dqb_bhardlimit &&
  	    dquot->dq_dqb.dqb_curspace - space < dquot->dq_dqb.dqb_bhardlimit)
657d3bfa9   Jan Kara   quota: implement ...
1179
1180
1181
  		return QUOTA_NL_BHARDBELOW;
  	return QUOTA_NL_NOWARN;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1182
1183
  /*
   *	Initialize quota pointers in inode
cc33412fb   Jan Kara   quota: Improve lo...
1184
1185
   *	We do things in a bit complicated way but by that we avoid calling
   *	dqget() and thus filesystem callbacks under dqptr_sem.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1186
1187
1188
1189
1190
   */
  int dquot_initialize(struct inode *inode, int type)
  {
  	unsigned int id = 0;
  	int cnt, ret = 0;
cc33412fb   Jan Kara   quota: Improve lo...
1191
1192
  	struct dquot *got[MAXQUOTAS] = { NODQUOT, NODQUOT };
  	struct super_block *sb = inode->i_sb;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1193

d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1194
1195
  	/* First test before acquiring mutex - solves deadlocks when we
           * re-enter the quota code and are already holding the mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1196
1197
  	if (IS_NOQUOTA(inode))
  		return 0;
cc33412fb   Jan Kara   quota: Improve lo...
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
  
  	/* First get references to structures we might need. */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (type != -1 && cnt != type)
  			continue;
  		switch (cnt) {
  		case USRQUOTA:
  			id = inode->i_uid;
  			break;
  		case GRPQUOTA:
  			id = inode->i_gid;
  			break;
  		}
  		got[cnt] = dqget(sb, id, cnt);
  	}
  
  	down_write(&sb_dqopt(sb)->dqptr_sem);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1215
1216
1217
1218
1219
1220
  	/* Having dqptr_sem we know NOQUOTA flags can't be altered... */
  	if (IS_NOQUOTA(inode))
  		goto out_err;
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (type != -1 && cnt != type)
  			continue;
cc33412fb   Jan Kara   quota: Improve lo...
1221
1222
1223
  		/* Avoid races with quotaoff() */
  		if (!sb_has_quota_active(sb, cnt))
  			continue;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1224
  		if (inode->i_dquot[cnt] == NODQUOT) {
cc33412fb   Jan Kara   quota: Improve lo...
1225
1226
  			inode->i_dquot[cnt] = got[cnt];
  			got[cnt] = NODQUOT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1227
1228
1229
  		}
  	}
  out_err:
cc33412fb   Jan Kara   quota: Improve lo...
1230
1231
1232
1233
  	up_write(&sb_dqopt(sb)->dqptr_sem);
  	/* Drop unused references */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		dqput(got[cnt]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1234
1235
1236
1237
1238
  	return ret;
  }
  
  /*
   * 	Release all quotas referenced by inode
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1239
   */
cc33412fb   Jan Kara   quota: Improve lo...
1240
  int dquot_drop(struct inode *inode)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1241
1242
  {
  	int cnt;
cc33412fb   Jan Kara   quota: Improve lo...
1243
  	struct dquot *put[MAXQUOTAS];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1244

cc33412fb   Jan Kara   quota: Improve lo...
1245
  	down_write(&sb_dqopt(inode->i_sb)->dqptr_sem);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1246
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
cc33412fb   Jan Kara   quota: Improve lo...
1247
1248
  		put[cnt] = inode->i_dquot[cnt];
  		inode->i_dquot[cnt] = NODQUOT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1249
1250
  	}
  	up_write(&sb_dqopt(inode->i_sb)->dqptr_sem);
cc33412fb   Jan Kara   quota: Improve lo...
1251
1252
1253
  
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		dqput(put[cnt]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1254
1255
  	return 0;
  }
b85f4b87a   Jan Kara   quota: rename quo...
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
  /* Wrapper to remove references to quota structures from inode */
  void vfs_dq_drop(struct inode *inode)
  {
  	/* Here we can get arbitrary inode from clear_inode() so we have
  	 * to be careful. OTOH we don't need locking as quota operations
  	 * are allowed to change only at mount time */
  	if (!IS_NOQUOTA(inode) && inode->i_sb && inode->i_sb->dq_op
  	    && inode->i_sb->dq_op->drop) {
  		int cnt;
  		/* Test before calling to rule out calls from proc and such
                   * where we are not allowed to block. Note that this is
  		 * actually reliable test even without the lock - the caller
  		 * must assure that nobody can come after the DQUOT_DROP and
  		 * add quota pointers back anyway */
  		for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  			if (inode->i_dquot[cnt] != NODQUOT)
  				break;
  		if (cnt < MAXQUOTAS)
  			inode->i_sb->dq_op->drop(inode);
  	}
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
  /*
   * Following four functions update i_blocks+i_bytes fields and
   * quota information (together with appropriate checks)
   * NOTE: We absolutely rely on the fact that caller dirties
   * the inode (usually macros in quotaops.h care about this) and
   * holds a handle for the current transaction so that dquot write and
   * inode write go into the same transaction.
   */
  
  /*
   * This operation can block, but only after everything is updated
   */
  int dquot_alloc_space(struct inode *inode, qsize_t number, int warn)
  {
  	int cnt, ret = NO_QUOTA;
  	char warntype[MAXQUOTAS];
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1293
1294
  	/* First test before acquiring mutex - solves deadlocks when we
           * re-enter the quota code and are already holding the mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1295
1296
1297
1298
1299
1300
  	if (IS_NOQUOTA(inode)) {
  out_add:
  		inode_add_bytes(inode, number);
  		return QUOTA_OK;
  	}
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
8e8934695   Jan Kara   quota: send messa...
1301
  		warntype[cnt] = QUOTA_NL_NOWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
  
  	down_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	if (IS_NOQUOTA(inode)) {	/* Now we can do reliable test... */
  		up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  		goto out_add;
  	}
  	spin_lock(&dq_data_lock);
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (inode->i_dquot[cnt] == NODQUOT)
  			continue;
  		if (check_bdq(inode->i_dquot[cnt], number, warn, warntype+cnt) == NO_QUOTA)
  			goto warn_put_all;
  	}
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (inode->i_dquot[cnt] == NODQUOT)
  			continue;
  		dquot_incr_space(inode->i_dquot[cnt], number);
  	}
  	inode_add_bytes(inode, number);
  	ret = QUOTA_OK;
  warn_put_all:
  	spin_unlock(&dq_data_lock);
  	if (ret == QUOTA_OK)
  		/* Dirtify all the dquots - this can block when journalling */
  		for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  			if (inode->i_dquot[cnt])
  				mark_dquot_dirty(inode->i_dquot[cnt]);
  	flush_warnings(inode->i_dquot, warntype);
  	up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	return ret;
  }
  
  /*
   * This operation can block, but only after everything is updated
   */
12095460f   Jan Kara   quota: Increase s...
1337
  int dquot_alloc_inode(const struct inode *inode, qsize_t number)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1338
1339
1340
  {
  	int cnt, ret = NO_QUOTA;
  	char warntype[MAXQUOTAS];
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1341
1342
  	/* First test before acquiring mutex - solves deadlocks when we
           * re-enter the quota code and are already holding the mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1343
1344
1345
  	if (IS_NOQUOTA(inode))
  		return QUOTA_OK;
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
8e8934695   Jan Kara   quota: send messa...
1346
  		warntype[cnt] = QUOTA_NL_NOWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
  	down_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	if (IS_NOQUOTA(inode)) {
  		up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  		return QUOTA_OK;
  	}
  	spin_lock(&dq_data_lock);
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (inode->i_dquot[cnt] == NODQUOT)
  			continue;
  		if (check_idq(inode->i_dquot[cnt], number, warntype+cnt) == NO_QUOTA)
  			goto warn_put_all;
  	}
  
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (inode->i_dquot[cnt] == NODQUOT)
  			continue;
  		dquot_incr_inodes(inode->i_dquot[cnt], number);
  	}
  	ret = QUOTA_OK;
  warn_put_all:
  	spin_unlock(&dq_data_lock);
  	if (ret == QUOTA_OK)
  		/* Dirtify all the dquots - this can block when journalling */
  		for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  			if (inode->i_dquot[cnt])
  				mark_dquot_dirty(inode->i_dquot[cnt]);
087ee8d5b   Jan Kara   Fix compilation w...
1373
  	flush_warnings(inode->i_dquot, warntype);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
  	up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	return ret;
  }
  
  /*
   * This operation can block, but only after everything is updated
   */
  int dquot_free_space(struct inode *inode, qsize_t number)
  {
  	unsigned int cnt;
657d3bfa9   Jan Kara   quota: implement ...
1384
  	char warntype[MAXQUOTAS];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1385

d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1386
1387
  	/* First test before acquiring mutex - solves deadlocks when we
           * re-enter the quota code and are already holding the mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1388
1389
1390
1391
1392
  	if (IS_NOQUOTA(inode)) {
  out_sub:
  		inode_sub_bytes(inode, number);
  		return QUOTA_OK;
  	}
657d3bfa9   Jan Kara   quota: implement ...
1393

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
  	down_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	/* Now recheck reliably when holding dqptr_sem */
  	if (IS_NOQUOTA(inode)) {
  		up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  		goto out_sub;
  	}
  	spin_lock(&dq_data_lock);
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (inode->i_dquot[cnt] == NODQUOT)
  			continue;
657d3bfa9   Jan Kara   quota: implement ...
1404
  		warntype[cnt] = info_bdq_free(inode->i_dquot[cnt], number);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1405
1406
1407
1408
1409
1410
1411
1412
  		dquot_decr_space(inode->i_dquot[cnt], number);
  	}
  	inode_sub_bytes(inode, number);
  	spin_unlock(&dq_data_lock);
  	/* Dirtify all the dquots - this can block when journalling */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		if (inode->i_dquot[cnt])
  			mark_dquot_dirty(inode->i_dquot[cnt]);
657d3bfa9   Jan Kara   quota: implement ...
1413
  	flush_warnings(inode->i_dquot, warntype);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1414
1415
1416
1417
1418
1419
1420
  	up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	return QUOTA_OK;
  }
  
  /*
   * This operation can block, but only after everything is updated
   */
12095460f   Jan Kara   quota: Increase s...
1421
  int dquot_free_inode(const struct inode *inode, qsize_t number)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1422
1423
  {
  	unsigned int cnt;
657d3bfa9   Jan Kara   quota: implement ...
1424
  	char warntype[MAXQUOTAS];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1425

d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1426
1427
  	/* First test before acquiring mutex - solves deadlocks when we
           * re-enter the quota code and are already holding the mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1428
1429
  	if (IS_NOQUOTA(inode))
  		return QUOTA_OK;
657d3bfa9   Jan Kara   quota: implement ...
1430

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
  	down_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	/* Now recheck reliably when holding dqptr_sem */
  	if (IS_NOQUOTA(inode)) {
  		up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  		return QUOTA_OK;
  	}
  	spin_lock(&dq_data_lock);
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (inode->i_dquot[cnt] == NODQUOT)
  			continue;
657d3bfa9   Jan Kara   quota: implement ...
1441
  		warntype[cnt] = info_idq_free(inode->i_dquot[cnt], number);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1442
1443
1444
1445
1446
1447
1448
  		dquot_decr_inodes(inode->i_dquot[cnt], number);
  	}
  	spin_unlock(&dq_data_lock);
  	/* Dirtify all the dquots - this can block when journalling */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		if (inode->i_dquot[cnt])
  			mark_dquot_dirty(inode->i_dquot[cnt]);
657d3bfa9   Jan Kara   quota: implement ...
1449
  	flush_warnings(inode->i_dquot, warntype);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464
  	up_read(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	return QUOTA_OK;
  }
  
  /*
   * Transfer the number of inode and blocks from one diskquota to an other.
   *
   * This operation can block, but only after everything is updated
   * A transaction must be started when entering this function.
   */
  int dquot_transfer(struct inode *inode, struct iattr *iattr)
  {
  	qsize_t space;
  	struct dquot *transfer_from[MAXQUOTAS];
  	struct dquot *transfer_to[MAXQUOTAS];
cc33412fb   Jan Kara   quota: Improve lo...
1465
1466
1467
  	int cnt, ret = QUOTA_OK;
  	int chuid = iattr->ia_valid & ATTR_UID && inode->i_uid != iattr->ia_uid,
  	    chgid = iattr->ia_valid & ATTR_GID && inode->i_gid != iattr->ia_gid;
657d3bfa9   Jan Kara   quota: implement ...
1468
1469
  	char warntype_to[MAXQUOTAS];
  	char warntype_from_inodes[MAXQUOTAS], warntype_from_space[MAXQUOTAS];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1470

d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1471
1472
  	/* First test before acquiring mutex - solves deadlocks when we
           * re-enter the quota code and are already holding the mutex */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1473
1474
  	if (IS_NOQUOTA(inode))
  		return QUOTA_OK;
cc33412fb   Jan Kara   quota: Improve lo...
1475
  	/* Initialize the arrays */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1476
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
cc33412fb   Jan Kara   quota: Improve lo...
1477
1478
  		transfer_from[cnt] = NODQUOT;
  		transfer_to[cnt] = NODQUOT;
657d3bfa9   Jan Kara   quota: implement ...
1479
  		warntype_to[cnt] = QUOTA_NL_NOWARN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
  		switch (cnt) {
  			case USRQUOTA:
  				if (!chuid)
  					continue;
  				transfer_to[cnt] = dqget(inode->i_sb, iattr->ia_uid, cnt);
  				break;
  			case GRPQUOTA:
  				if (!chgid)
  					continue;
  				transfer_to[cnt] = dqget(inode->i_sb, iattr->ia_gid, cnt);
  				break;
  		}
  	}
cc33412fb   Jan Kara   quota: Improve lo...
1493
1494
1495
1496
1497
1498
1499
  
  	down_write(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	/* Now recheck reliably when holding dqptr_sem */
  	if (IS_NOQUOTA(inode)) {	/* File without quota accounting? */
  		up_write(&sb_dqopt(inode->i_sb)->dqptr_sem);
  		goto put_all;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1500
1501
1502
1503
1504
1505
1506
  	spin_lock(&dq_data_lock);
  	space = inode_get_bytes(inode);
  	/* Build the transfer_from list and check the limits */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (transfer_to[cnt] == NODQUOT)
  			continue;
  		transfer_from[cnt] = inode->i_dquot[cnt];
657d3bfa9   Jan Kara   quota: implement ...
1507
1508
1509
  		if (check_idq(transfer_to[cnt], 1, warntype_to + cnt) ==
  		    NO_QUOTA || check_bdq(transfer_to[cnt], space, 0,
  		    warntype_to + cnt) == NO_QUOTA)
cc33412fb   Jan Kara   quota: Improve lo...
1510
  			goto over_quota;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
  	}
  
  	/*
  	 * Finally perform the needed transfer from transfer_from to transfer_to
  	 */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		/*
  		 * Skip changes for same uid or gid or for turned off quota-type.
  		 */
  		if (transfer_to[cnt] == NODQUOT)
  			continue;
  
  		/* Due to IO error we might not have transfer_from[] structure */
  		if (transfer_from[cnt]) {
657d3bfa9   Jan Kara   quota: implement ...
1525
1526
1527
1528
  			warntype_from_inodes[cnt] =
  				info_idq_free(transfer_from[cnt], 1);
  			warntype_from_space[cnt] =
  				info_bdq_free(transfer_from[cnt], space);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1529
1530
1531
1532
1533
1534
1535
1536
1537
  			dquot_decr_inodes(transfer_from[cnt], 1);
  			dquot_decr_space(transfer_from[cnt], space);
  		}
  
  		dquot_incr_inodes(transfer_to[cnt], 1);
  		dquot_incr_space(transfer_to[cnt], space);
  
  		inode->i_dquot[cnt] = transfer_to[cnt];
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1538
  	spin_unlock(&dq_data_lock);
cc33412fb   Jan Kara   quota: Improve lo...
1539
  	up_write(&sb_dqopt(inode->i_sb)->dqptr_sem);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1540
1541
1542
1543
  	/* Dirtify all the dquots - this can block when journalling */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		if (transfer_from[cnt])
  			mark_dquot_dirty(transfer_from[cnt]);
cc33412fb   Jan Kara   quota: Improve lo...
1544
  		if (transfer_to[cnt]) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1545
  			mark_dquot_dirty(transfer_to[cnt]);
cc33412fb   Jan Kara   quota: Improve lo...
1546
1547
1548
  			/* The reference we got is transferred to the inode */
  			transfer_to[cnt] = NODQUOT;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1549
  	}
cc33412fb   Jan Kara   quota: Improve lo...
1550
  warn_put_all:
657d3bfa9   Jan Kara   quota: implement ...
1551
1552
1553
  	flush_warnings(transfer_to, warntype_to);
  	flush_warnings(transfer_from, warntype_from_inodes);
  	flush_warnings(transfer_from, warntype_from_space);
cc33412fb   Jan Kara   quota: Improve lo...
1554
  put_all:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1555
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
cc33412fb   Jan Kara   quota: Improve lo...
1556
1557
  		dqput(transfer_from[cnt]);
  		dqput(transfer_to[cnt]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1558
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1559
  	return ret;
cc33412fb   Jan Kara   quota: Improve lo...
1560
1561
1562
1563
1564
1565
1566
1567
  over_quota:
  	spin_unlock(&dq_data_lock);
  	up_write(&sb_dqopt(inode->i_sb)->dqptr_sem);
  	/* Clear dquot pointers we don't want to dqput() */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		transfer_from[cnt] = NODQUOT;
  	ret = NO_QUOTA;
  	goto warn_put_all;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1568
  }
b85f4b87a   Jan Kara   quota: rename quo...
1569
1570
1571
  /* Wrapper for transferring ownership of an inode */
  int vfs_dq_transfer(struct inode *inode, struct iattr *iattr)
  {
f55abc0fb   Jan Kara   quota: Allow to s...
1572
  	if (sb_any_quota_active(inode->i_sb) && !IS_NOQUOTA(inode)) {
b85f4b87a   Jan Kara   quota: rename quo...
1573
1574
1575
1576
1577
1578
  		vfs_dq_init(inode);
  		if (inode->i_sb->dq_op->transfer(inode, iattr) == NO_QUOTA)
  			return 1;
  	}
  	return 0;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1579
1580
1581
1582
1583
1584
1585
  /*
   * Write info of quota file to disk
   */
  int dquot_commit_info(struct super_block *sb, int type)
  {
  	int ret;
  	struct quota_info *dqopt = sb_dqopt(sb);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1586
  	mutex_lock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1587
  	ret = dqopt->ops[type]->write_file_info(sb, type);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1588
  	mutex_unlock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
  	return ret;
  }
  
  /*
   * Definitions of diskquota operations.
   */
  struct dquot_operations dquot_operations = {
  	.initialize	= dquot_initialize,
  	.drop		= dquot_drop,
  	.alloc_space	= dquot_alloc_space,
  	.alloc_inode	= dquot_alloc_inode,
  	.free_space	= dquot_free_space,
  	.free_inode	= dquot_free_inode,
  	.transfer	= dquot_transfer,
  	.write_dquot	= dquot_commit,
  	.acquire_dquot	= dquot_acquire,
  	.release_dquot	= dquot_release,
  	.mark_dirty	= dquot_mark_dquot_dirty,
74f783af9   Jan Kara   quota: Add callba...
1607
1608
1609
  	.write_info	= dquot_commit_info,
  	.alloc_dquot	= dquot_alloc,
  	.destroy_dquot	= dquot_destroy,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1610
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1611
1612
1613
  /*
   * Turn quota off on a device. type == -1 ==> quotaoff for all types (umount)
   */
f55abc0fb   Jan Kara   quota: Allow to s...
1614
  int vfs_quota_disable(struct super_block *sb, int type, unsigned int flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1615
  {
0ff5af834   Jan Kara   quota: quota core...
1616
  	int cnt, ret = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1617
1618
  	struct quota_info *dqopt = sb_dqopt(sb);
  	struct inode *toputinode[MAXQUOTAS];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1619

f55abc0fb   Jan Kara   quota: Allow to s...
1620
1621
1622
1623
1624
1625
  	/* Cannot turn off usage accounting without turning off limits, or
  	 * suspend quotas and simultaneously turn quotas off. */
  	if ((flags & DQUOT_USAGE_ENABLED && !(flags & DQUOT_LIMITS_ENABLED))
  	    || (flags & DQUOT_SUSPENDED && flags & (DQUOT_LIMITS_ENABLED |
  	    DQUOT_USAGE_ENABLED)))
  		return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1626
  	/* We need to serialize quota_off() for device */
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1627
  	mutex_lock(&dqopt->dqonoff_mutex);
9377abd02   Jan Kara   quota: don't call...
1628
1629
1630
1631
1632
1633
  
  	/*
  	 * Skip everything if there's nothing to do. We have to do this because
  	 * sometimes we are called when fill_super() failed and calling
  	 * sync_fs() in such cases does no good.
  	 */
f55abc0fb   Jan Kara   quota: Allow to s...
1634
  	if (!sb_any_quota_loaded(sb)) {
9377abd02   Jan Kara   quota: don't call...
1635
1636
1637
  		mutex_unlock(&dqopt->dqonoff_mutex);
  		return 0;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1638
1639
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		toputinode[cnt] = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1640
1641
  		if (type != -1 && cnt != type)
  			continue;
f55abc0fb   Jan Kara   quota: Allow to s...
1642
  		if (!sb_has_quota_loaded(sb, cnt))
0ff5af834   Jan Kara   quota: quota core...
1643
  			continue;
f55abc0fb   Jan Kara   quota: Allow to s...
1644
1645
  
  		if (flags & DQUOT_SUSPENDED) {
cc33412fb   Jan Kara   quota: Improve lo...
1646
  			spin_lock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1647
1648
  			dqopt->flags |=
  				dquot_state_flag(DQUOT_SUSPENDED, cnt);
cc33412fb   Jan Kara   quota: Improve lo...
1649
  			spin_unlock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1650
  		} else {
cc33412fb   Jan Kara   quota: Improve lo...
1651
  			spin_lock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1652
1653
1654
1655
1656
1657
  			dqopt->flags &= ~dquot_state_flag(flags, cnt);
  			/* Turning off suspended quotas? */
  			if (!sb_has_quota_loaded(sb, cnt) &&
  			    sb_has_quota_suspended(sb, cnt)) {
  				dqopt->flags &=	~dquot_state_flag(
  							DQUOT_SUSPENDED, cnt);
cc33412fb   Jan Kara   quota: Improve lo...
1658
  				spin_unlock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1659
1660
1661
1662
  				iput(dqopt->files[cnt]);
  				dqopt->files[cnt] = NULL;
  				continue;
  			}
cc33412fb   Jan Kara   quota: Improve lo...
1663
  			spin_unlock(&dq_state_lock);
0ff5af834   Jan Kara   quota: quota core...
1664
  		}
f55abc0fb   Jan Kara   quota: Allow to s...
1665
1666
1667
  
  		/* We still have to keep quota loaded? */
  		if (sb_has_quota_loaded(sb, cnt) && !(flags & DQUOT_SUSPENDED))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1668
  			continue;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1669
1670
1671
1672
1673
1674
1675
1676
1677
1678
1679
1680
1681
1682
1683
  
  		/* Note: these are blocking operations */
  		drop_dquot_ref(sb, cnt);
  		invalidate_dquots(sb, cnt);
  		/*
  		 * Now all dquots should be invalidated, all writes done so we should be only
  		 * users of the info. No locks needed.
  		 */
  		if (info_dirty(&dqopt->info[cnt]))
  			sb->dq_op->write_info(sb, cnt);
  		if (dqopt->ops[cnt]->free_file_info)
  			dqopt->ops[cnt]->free_file_info(sb, cnt);
  		put_quota_format(dqopt->info[cnt].dqi_format);
  
  		toputinode[cnt] = dqopt->files[cnt];
f55abc0fb   Jan Kara   quota: Allow to s...
1684
  		if (!sb_has_quota_loaded(sb, cnt))
0ff5af834   Jan Kara   quota: quota core...
1685
  			dqopt->files[cnt] = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1686
1687
1688
1689
1690
  		dqopt->info[cnt].dqi_flags = 0;
  		dqopt->info[cnt].dqi_igrace = 0;
  		dqopt->info[cnt].dqi_bgrace = 0;
  		dqopt->ops[cnt] = NULL;
  	}
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1691
  	mutex_unlock(&dqopt->dqonoff_mutex);
ca785ec66   Jan Kara   quota: Introduce ...
1692
1693
1694
1695
  
  	/* Skip syncing and setting flags if quota files are hidden */
  	if (dqopt->flags & DQUOT_QUOTA_SYS_FILE)
  		goto put_inodes;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1696
  	/* Sync the superblock so that buffers with quota data are written to
7b7b1ace2   Al Viro   [PATCH] saner han...
1697
  	 * disk (and so userspace sees correct data afterwards). */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1698
1699
1700
1701
1702
1703
1704
1705
1706
1707
  	if (sb->s_op->sync_fs)
  		sb->s_op->sync_fs(sb, 1);
  	sync_blockdev(sb->s_bdev);
  	/* Now the quota files are just ordinary files and we can set the
  	 * inode flags back. Moreover we discard the pagecache so that
  	 * userspace sees the writes we did bypassing the pagecache. We
  	 * must also discard the blockdev buffers so that we see the
  	 * changes done by userspace on the next quotaon() */
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		if (toputinode[cnt]) {
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1708
  			mutex_lock(&dqopt->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1709
1710
  			/* If quota was reenabled in the meantime, we have
  			 * nothing to do */
f55abc0fb   Jan Kara   quota: Allow to s...
1711
  			if (!sb_has_quota_loaded(sb, cnt)) {
7925409e2   Jan Kara   circular locking ...
1712
  				mutex_lock_nested(&toputinode[cnt]->i_mutex, I_MUTEX_QUOTA);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1713
1714
1715
  				toputinode[cnt]->i_flags &= ~(S_IMMUTABLE |
  				  S_NOATIME | S_NOQUOTA);
  				truncate_inode_pages(&toputinode[cnt]->i_data, 0);
1b1dcc1b5   Jes Sorensen   [PATCH] mutex sub...
1716
  				mutex_unlock(&toputinode[cnt]->i_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1717
  				mark_inode_dirty(toputinode[cnt]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1718
  			}
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1719
  			mutex_unlock(&dqopt->dqonoff_mutex);
ca785ec66   Jan Kara   quota: Introduce ...
1720
1721
1722
1723
1724
1725
  		}
  	if (sb->s_bdev)
  		invalidate_bdev(sb->s_bdev);
  put_inodes:
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++)
  		if (toputinode[cnt]) {
0ff5af834   Jan Kara   quota: quota core...
1726
  			/* On remount RO, we keep the inode pointer so that we
f55abc0fb   Jan Kara   quota: Allow to s...
1727
1728
1729
1730
1731
1732
1733
  			 * can reenable quota on the subsequent remount RW. We
  			 * have to check 'flags' variable and not use sb_has_
  			 * function because another quotaon / quotaoff could
  			 * change global state before we got here. We refuse
  			 * to suspend quotas when there is pending delete on
  			 * the quota file... */
  			if (!(flags & DQUOT_SUSPENDED))
0ff5af834   Jan Kara   quota: quota core...
1734
1735
1736
  				iput(toputinode[cnt]);
  			else if (!toputinode[cnt]->i_nlink)
  				ret = -EBUSY;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1737
  		}
0ff5af834   Jan Kara   quota: quota core...
1738
  	return ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1739
  }
f55abc0fb   Jan Kara   quota: Allow to s...
1740
1741
1742
1743
1744
  int vfs_quota_off(struct super_block *sb, int type, int remount)
  {
  	return vfs_quota_disable(sb, type, remount ? DQUOT_SUSPENDED :
  				 (DQUOT_USAGE_ENABLED | DQUOT_LIMITS_ENABLED));
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1745
1746
1747
  /*
   *	Turn quotas on on a device
   */
f55abc0fb   Jan Kara   quota: Allow to s...
1748
1749
1750
1751
1752
1753
  /*
   * Helper function to turn quotas on when we already have the inode of
   * quota file and no quota information is loaded.
   */
  static int vfs_load_quota_inode(struct inode *inode, int type, int format_id,
  	unsigned int flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
  {
  	struct quota_format_type *fmt = find_quota_format(format_id);
  	struct super_block *sb = inode->i_sb;
  	struct quota_info *dqopt = sb_dqopt(sb);
  	int error;
  	int oldflags = -1;
  
  	if (!fmt)
  		return -ESRCH;
  	if (!S_ISREG(inode->i_mode)) {
  		error = -EACCES;
  		goto out_fmt;
  	}
  	if (IS_RDONLY(inode)) {
  		error = -EROFS;
  		goto out_fmt;
  	}
  	if (!sb->s_op->quota_write || !sb->s_op->quota_read) {
  		error = -EINVAL;
  		goto out_fmt;
  	}
f55abc0fb   Jan Kara   quota: Allow to s...
1775
1776
1777
1778
1779
  	/* Usage always has to be set... */
  	if (!(flags & DQUOT_USAGE_ENABLED)) {
  		error = -EINVAL;
  		goto out_fmt;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1780

ca785ec66   Jan Kara   quota: Introduce ...
1781
1782
1783
1784
1785
1786
1787
1788
  	if (!(dqopt->flags & DQUOT_QUOTA_SYS_FILE)) {
  		/* As we bypass the pagecache we must now flush the inode so
  		 * that we see all the changes from userspace... */
  		write_inode_now(inode, 1);
  		/* And now flush the block cache so that kernel sees the
  		 * changes */
  		invalidate_bdev(sb->s_bdev);
  	}
1b1dcc1b5   Jes Sorensen   [PATCH] mutex sub...
1789
  	mutex_lock(&inode->i_mutex);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1790
  	mutex_lock(&dqopt->dqonoff_mutex);
f55abc0fb   Jan Kara   quota: Allow to s...
1791
  	if (sb_has_quota_loaded(sb, type)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1792
1793
1794
  		error = -EBUSY;
  		goto out_lock;
  	}
ca785ec66   Jan Kara   quota: Introduce ...
1795
1796
1797
1798
1799
1800
1801
1802
1803
1804
1805
  
  	if (!(dqopt->flags & DQUOT_QUOTA_SYS_FILE)) {
  		/* We don't want quota and atime on quota files (deadlocks
  		 * possible) Also nobody should write to the file - we use
  		 * special IO operations which ignore the immutable bit. */
  		down_write(&dqopt->dqptr_sem);
  		oldflags = inode->i_flags & (S_NOATIME | S_IMMUTABLE | S_NOQUOTA);
  		inode->i_flags |= S_NOQUOTA | S_NOATIME | S_IMMUTABLE;
  		up_write(&dqopt->dqptr_sem);
  		sb->dq_op->drop(inode);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1806
1807
1808
1809
1810
1811
1812
1813
1814
1815
1816
  
  	error = -EIO;
  	dqopt->files[type] = igrab(inode);
  	if (!dqopt->files[type])
  		goto out_lock;
  	error = -EINVAL;
  	if (!fmt->qf_ops->check_quota_file(sb, type))
  		goto out_file_init;
  
  	dqopt->ops[type] = fmt->qf_ops;
  	dqopt->info[type].dqi_format = fmt;
0ff5af834   Jan Kara   quota: quota core...
1817
  	dqopt->info[type].dqi_fmt_id = format_id;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1818
  	INIT_LIST_HEAD(&dqopt->info[type].dqi_dirty_list);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1819
  	mutex_lock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1820
  	if ((error = dqopt->ops[type]->read_file_info(sb, type)) < 0) {
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1821
  		mutex_unlock(&dqopt->dqio_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1822
1823
  		goto out_file_init;
  	}
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1824
  	mutex_unlock(&dqopt->dqio_mutex);
1b1dcc1b5   Jes Sorensen   [PATCH] mutex sub...
1825
  	mutex_unlock(&inode->i_mutex);
cc33412fb   Jan Kara   quota: Improve lo...
1826
  	spin_lock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1827
  	dqopt->flags |= dquot_state_flag(flags, type);
cc33412fb   Jan Kara   quota: Improve lo...
1828
  	spin_unlock(&dq_state_lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1829
1830
  
  	add_dquot_ref(sb, type);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1831
  	mutex_unlock(&dqopt->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1832
1833
1834
1835
1836
1837
1838
  
  	return 0;
  
  out_file_init:
  	dqopt->files[type] = NULL;
  	iput(inode);
  out_lock:
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
1839
  	mutex_unlock(&dqopt->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1840
1841
1842
1843
1844
1845
1846
1847
  	if (oldflags != -1) {
  		down_write(&dqopt->dqptr_sem);
  		/* Set the flags back (in the case of accidental quotaon()
  		 * on a wrong file we don't want to mess up the flags) */
  		inode->i_flags &= ~(S_NOATIME | S_NOQUOTA | S_IMMUTABLE);
  		inode->i_flags |= oldflags;
  		up_write(&dqopt->dqptr_sem);
  	}
1b1dcc1b5   Jes Sorensen   [PATCH] mutex sub...
1848
  	mutex_unlock(&inode->i_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1849
1850
1851
1852
1853
  out_fmt:
  	put_quota_format(fmt);
  
  	return error; 
  }
0ff5af834   Jan Kara   quota: quota core...
1854
1855
1856
1857
1858
1859
  /* Reenable quotas on remount RW */
  static int vfs_quota_on_remount(struct super_block *sb, int type)
  {
  	struct quota_info *dqopt = sb_dqopt(sb);
  	struct inode *inode;
  	int ret;
f55abc0fb   Jan Kara   quota: Allow to s...
1860
  	unsigned int flags;
0ff5af834   Jan Kara   quota: quota core...
1861
1862
1863
1864
1865
1866
  
  	mutex_lock(&dqopt->dqonoff_mutex);
  	if (!sb_has_quota_suspended(sb, type)) {
  		mutex_unlock(&dqopt->dqonoff_mutex);
  		return 0;
  	}
0ff5af834   Jan Kara   quota: quota core...
1867
1868
  	inode = dqopt->files[type];
  	dqopt->files[type] = NULL;
cc33412fb   Jan Kara   quota: Improve lo...
1869
  	spin_lock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1870
1871
1872
  	flags = dqopt->flags & dquot_state_flag(DQUOT_USAGE_ENABLED |
  						DQUOT_LIMITS_ENABLED, type);
  	dqopt->flags &= ~dquot_state_flag(DQUOT_STATE_FLAGS, type);
cc33412fb   Jan Kara   quota: Improve lo...
1873
  	spin_unlock(&dq_state_lock);
0ff5af834   Jan Kara   quota: quota core...
1874
  	mutex_unlock(&dqopt->dqonoff_mutex);
f55abc0fb   Jan Kara   quota: Allow to s...
1875
1876
1877
  	flags = dquot_generic_flag(flags, type);
  	ret = vfs_load_quota_inode(inode, type, dqopt->info[type].dqi_fmt_id,
  				   flags);
0ff5af834   Jan Kara   quota: quota core...
1878
1879
1880
1881
  	iput(inode);
  
  	return ret;
  }
77e69dac3   Al Viro   [PATCH] fix races...
1882
1883
1884
1885
1886
1887
1888
1889
1890
1891
  int vfs_quota_on_path(struct super_block *sb, int type, int format_id,
  		      struct path *path)
  {
  	int error = security_quota_on(path->dentry);
  	if (error)
  		return error;
  	/* Quota file not on the same filesystem? */
  	if (path->mnt->mnt_sb != sb)
  		error = -EXDEV;
  	else
f55abc0fb   Jan Kara   quota: Allow to s...
1892
1893
1894
  		error = vfs_load_quota_inode(path->dentry->d_inode, type,
  					     format_id, DQUOT_USAGE_ENABLED |
  					     DQUOT_LIMITS_ENABLED);
77e69dac3   Al Viro   [PATCH] fix races...
1895
1896
  	return error;
  }
8264613de   Al Viro   [PATCH] switch qu...
1897
  int vfs_quota_on(struct super_block *sb, int type, int format_id, char *name,
0ff5af834   Jan Kara   quota: quota core...
1898
  		 int remount)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1899
  {
8264613de   Al Viro   [PATCH] switch qu...
1900
  	struct path path;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1901
  	int error;
0ff5af834   Jan Kara   quota: quota core...
1902
1903
  	if (remount)
  		return vfs_quota_on_remount(sb, type);
8264613de   Al Viro   [PATCH] switch qu...
1904
  	error = kern_path(name, LOOKUP_FOLLOW, &path);
77e69dac3   Al Viro   [PATCH] fix races...
1905
  	if (!error) {
8264613de   Al Viro   [PATCH] switch qu...
1906
1907
  		error = vfs_quota_on_path(sb, type, format_id, &path);
  		path_put(&path);
77e69dac3   Al Viro   [PATCH] fix races...
1908
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1909
1910
1911
1912
  	return error;
  }
  
  /*
f55abc0fb   Jan Kara   quota: Allow to s...
1913
1914
1915
1916
1917
1918
1919
1920
1921
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931
1932
1933
1934
1935
1936
1937
1938
1939
1940
1941
1942
1943
1944
1945
   * More powerful function for turning on quotas allowing setting
   * of individual quota flags
   */
  int vfs_quota_enable(struct inode *inode, int type, int format_id,
  		unsigned int flags)
  {
  	int ret = 0;
  	struct super_block *sb = inode->i_sb;
  	struct quota_info *dqopt = sb_dqopt(sb);
  
  	/* Just unsuspend quotas? */
  	if (flags & DQUOT_SUSPENDED)
  		return vfs_quota_on_remount(sb, type);
  	if (!flags)
  		return 0;
  	/* Just updating flags needed? */
  	if (sb_has_quota_loaded(sb, type)) {
  		mutex_lock(&dqopt->dqonoff_mutex);
  		/* Now do a reliable test... */
  		if (!sb_has_quota_loaded(sb, type)) {
  			mutex_unlock(&dqopt->dqonoff_mutex);
  			goto load_quota;
  		}
  		if (flags & DQUOT_USAGE_ENABLED &&
  		    sb_has_quota_usage_enabled(sb, type)) {
  			ret = -EBUSY;
  			goto out_lock;
  		}
  		if (flags & DQUOT_LIMITS_ENABLED &&
  		    sb_has_quota_limits_enabled(sb, type)) {
  			ret = -EBUSY;
  			goto out_lock;
  		}
cc33412fb   Jan Kara   quota: Improve lo...
1946
  		spin_lock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1947
  		sb_dqopt(sb)->flags |= dquot_state_flag(flags, type);
cc33412fb   Jan Kara   quota: Improve lo...
1948
  		spin_unlock(&dq_state_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
1949
1950
1951
1952
1953
1954
1955
1956
1957
1958
  out_lock:
  		mutex_unlock(&dqopt->dqonoff_mutex);
  		return ret;
  	}
  
  load_quota:
  	return vfs_load_quota_inode(inode, type, format_id, flags);
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1959
1960
1961
   * This function is used when filesystem needs to initialize quotas
   * during mount time.
   */
84de856ed   Christoph Hellwig   [PATCH] quota: co...
1962
1963
  int vfs_quota_on_mount(struct super_block *sb, char *qf_name,
  		int format_id, int type)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1964
  {
84de856ed   Christoph Hellwig   [PATCH] quota: co...
1965
  	struct dentry *dentry;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1966
  	int error;
2fa389c5e   Christoph Hellwig   [PATCH] quota: sa...
1967
  	dentry = lookup_one_len(qf_name, sb->s_root, strlen(qf_name));
84de856ed   Christoph Hellwig   [PATCH] quota: co...
1968
1969
  	if (IS_ERR(dentry))
  		return PTR_ERR(dentry);
154f484b9   Jan Kara   [PATCH] Fix oops ...
1970
1971
1972
1973
  	if (!dentry->d_inode) {
  		error = -ENOENT;
  		goto out;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1974
  	error = security_quota_on(dentry);
84de856ed   Christoph Hellwig   [PATCH] quota: co...
1975
  	if (!error)
f55abc0fb   Jan Kara   quota: Allow to s...
1976
1977
  		error = vfs_load_quota_inode(dentry->d_inode, type, format_id,
  				DQUOT_USAGE_ENABLED | DQUOT_LIMITS_ENABLED);
84de856ed   Christoph Hellwig   [PATCH] quota: co...
1978

154f484b9   Jan Kara   [PATCH] Fix oops ...
1979
  out:
84de856ed   Christoph Hellwig   [PATCH] quota: co...
1980
1981
  	dput(dentry);
  	return error;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1982
  }
b85f4b87a   Jan Kara   quota: rename quo...
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
  /* Wrapper to turn on quotas when remounting rw */
  int vfs_dq_quota_on_remount(struct super_block *sb)
  {
  	int cnt;
  	int ret = 0, err;
  
  	if (!sb->s_qcop || !sb->s_qcop->quota_on)
  		return -ENOSYS;
  	for (cnt = 0; cnt < MAXQUOTAS; cnt++) {
  		err = sb->s_qcop->quota_on(sb, cnt, 0, NULL, 1);
  		if (err < 0 && !ret)
  			ret = err;
  	}
  	return ret;
  }
12095460f   Jan Kara   quota: Increase s...
1998
1999
2000
2001
2002
2003
2004
2005
2006
  static inline qsize_t qbtos(qsize_t blocks)
  {
  	return blocks << QIF_DQBLKSIZE_BITS;
  }
  
  static inline qsize_t stoqb(qsize_t space)
  {
  	return (space + QIF_DQBLKSIZE - 1) >> QIF_DQBLKSIZE_BITS;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2007
2008
2009
2010
2011
2012
  /* Generic routine for getting common part of quota structure */
  static void do_get_dqblk(struct dquot *dquot, struct if_dqblk *di)
  {
  	struct mem_dqblk *dm = &dquot->dq_dqb;
  
  	spin_lock(&dq_data_lock);
12095460f   Jan Kara   quota: Increase s...
2013
2014
  	di->dqb_bhardlimit = stoqb(dm->dqb_bhardlimit);
  	di->dqb_bsoftlimit = stoqb(dm->dqb_bsoftlimit);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
2027
  	di->dqb_curspace = dm->dqb_curspace;
  	di->dqb_ihardlimit = dm->dqb_ihardlimit;
  	di->dqb_isoftlimit = dm->dqb_isoftlimit;
  	di->dqb_curinodes = dm->dqb_curinodes;
  	di->dqb_btime = dm->dqb_btime;
  	di->dqb_itime = dm->dqb_itime;
  	di->dqb_valid = QIF_ALL;
  	spin_unlock(&dq_data_lock);
  }
  
  int vfs_get_dqblk(struct super_block *sb, int type, qid_t id, struct if_dqblk *di)
  {
  	struct dquot *dquot;
cc33412fb   Jan Kara   quota: Improve lo...
2028
2029
  	dquot = dqget(sb, id, type);
  	if (dquot == NODQUOT)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2030
  		return -ESRCH;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2031
2032
  	do_get_dqblk(dquot, di);
  	dqput(dquot);
cc33412fb   Jan Kara   quota: Improve lo...
2033

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2034
2035
2036
2037
  	return 0;
  }
  
  /* Generic routine for setting common part of quota structure */
338bf9afd   Andrew Perepechko   quota: do not all...
2038
  static int do_set_dqblk(struct dquot *dquot, struct if_dqblk *di)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2039
2040
2041
  {
  	struct mem_dqblk *dm = &dquot->dq_dqb;
  	int check_blim = 0, check_ilim = 0;
338bf9afd   Andrew Perepechko   quota: do not all...
2042
2043
2044
2045
2046
2047
2048
2049
2050
  	struct mem_dqinfo *dqi = &sb_dqopt(dquot->dq_sb)->info[dquot->dq_type];
  
  	if ((di->dqb_valid & QIF_BLIMITS &&
  	     (di->dqb_bhardlimit > dqi->dqi_maxblimit ||
  	      di->dqb_bsoftlimit > dqi->dqi_maxblimit)) ||
  	    (di->dqb_valid & QIF_ILIMITS &&
  	     (di->dqb_ihardlimit > dqi->dqi_maxilimit ||
  	      di->dqb_isoftlimit > dqi->dqi_maxilimit)))
  		return -ERANGE;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2051
2052
2053
2054
2055
  
  	spin_lock(&dq_data_lock);
  	if (di->dqb_valid & QIF_SPACE) {
  		dm->dqb_curspace = di->dqb_curspace;
  		check_blim = 1;
4d59bce4f   Jan Kara   quota: Keep which...
2056
  		__set_bit(DQ_LASTSET_B + QIF_SPACE_B, &dquot->dq_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2057
2058
  	}
  	if (di->dqb_valid & QIF_BLIMITS) {
12095460f   Jan Kara   quota: Increase s...
2059
2060
  		dm->dqb_bsoftlimit = qbtos(di->dqb_bsoftlimit);
  		dm->dqb_bhardlimit = qbtos(di->dqb_bhardlimit);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2061
  		check_blim = 1;
4d59bce4f   Jan Kara   quota: Keep which...
2062
  		__set_bit(DQ_LASTSET_B + QIF_BLIMITS_B, &dquot->dq_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2063
2064
2065
2066
  	}
  	if (di->dqb_valid & QIF_INODES) {
  		dm->dqb_curinodes = di->dqb_curinodes;
  		check_ilim = 1;
4d59bce4f   Jan Kara   quota: Keep which...
2067
  		__set_bit(DQ_LASTSET_B + QIF_INODES_B, &dquot->dq_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2068
2069
2070
2071
2072
  	}
  	if (di->dqb_valid & QIF_ILIMITS) {
  		dm->dqb_isoftlimit = di->dqb_isoftlimit;
  		dm->dqb_ihardlimit = di->dqb_ihardlimit;
  		check_ilim = 1;
4d59bce4f   Jan Kara   quota: Keep which...
2073
  		__set_bit(DQ_LASTSET_B + QIF_ILIMITS_B, &dquot->dq_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2074
  	}
4d59bce4f   Jan Kara   quota: Keep which...
2075
  	if (di->dqb_valid & QIF_BTIME) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2076
  		dm->dqb_btime = di->dqb_btime;
e04a88a92   Jan Kara   quota: don't set ...
2077
  		check_blim = 1;
4d59bce4f   Jan Kara   quota: Keep which...
2078
2079
2080
  		__set_bit(DQ_LASTSET_B + QIF_BTIME_B, &dquot->dq_flags);
  	}
  	if (di->dqb_valid & QIF_ITIME) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2081
  		dm->dqb_itime = di->dqb_itime;
e04a88a92   Jan Kara   quota: don't set ...
2082
  		check_ilim = 1;
4d59bce4f   Jan Kara   quota: Keep which...
2083
2084
  		__set_bit(DQ_LASTSET_B + QIF_ITIME_B, &dquot->dq_flags);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2085
2086
  
  	if (check_blim) {
12095460f   Jan Kara   quota: Increase s...
2087
  		if (!dm->dqb_bsoftlimit || dm->dqb_curspace < dm->dqb_bsoftlimit) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2088
2089
2090
2091
  			dm->dqb_btime = 0;
  			clear_bit(DQ_BLKS_B, &dquot->dq_flags);
  		}
  		else if (!(di->dqb_valid & QIF_BTIME))	/* Set grace only if user hasn't provided his own... */
338bf9afd   Andrew Perepechko   quota: do not all...
2092
  			dm->dqb_btime = get_seconds() + dqi->dqi_bgrace;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2093
2094
2095
2096
2097
2098
2099
  	}
  	if (check_ilim) {
  		if (!dm->dqb_isoftlimit || dm->dqb_curinodes < dm->dqb_isoftlimit) {
  			dm->dqb_itime = 0;
  			clear_bit(DQ_INODES_B, &dquot->dq_flags);
  		}
  		else if (!(di->dqb_valid & QIF_ITIME))	/* Set grace only if user hasn't provided his own... */
338bf9afd   Andrew Perepechko   quota: do not all...
2100
  			dm->dqb_itime = get_seconds() + dqi->dqi_igrace;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2101
2102
2103
2104
2105
2106
2107
  	}
  	if (dm->dqb_bhardlimit || dm->dqb_bsoftlimit || dm->dqb_ihardlimit || dm->dqb_isoftlimit)
  		clear_bit(DQ_FAKE_B, &dquot->dq_flags);
  	else
  		set_bit(DQ_FAKE_B, &dquot->dq_flags);
  	spin_unlock(&dq_data_lock);
  	mark_dquot_dirty(dquot);
338bf9afd   Andrew Perepechko   quota: do not all...
2108
2109
  
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2110
2111
2112
2113
2114
  }
  
  int vfs_set_dqblk(struct super_block *sb, int type, qid_t id, struct if_dqblk *di)
  {
  	struct dquot *dquot;
338bf9afd   Andrew Perepechko   quota: do not all...
2115
  	int rc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2116

f55abc0fb   Jan Kara   quota: Allow to s...
2117
2118
2119
2120
  	dquot = dqget(sb, id, type);
  	if (!dquot) {
  		rc = -ESRCH;
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2121
  	}
338bf9afd   Andrew Perepechko   quota: do not all...
2122
  	rc = do_set_dqblk(dquot, di);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2123
  	dqput(dquot);
f55abc0fb   Jan Kara   quota: Allow to s...
2124
  out:
338bf9afd   Andrew Perepechko   quota: do not all...
2125
  	return rc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2126
2127
2128
2129
2130
2131
2132
  }
  
  /* Generic routine for getting common part of quota file information */
  int vfs_get_dqinfo(struct super_block *sb, int type, struct if_dqinfo *ii)
  {
  	struct mem_dqinfo *mi;
    
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
2133
  	mutex_lock(&sb_dqopt(sb)->dqonoff_mutex);
f55abc0fb   Jan Kara   quota: Allow to s...
2134
  	if (!sb_has_quota_active(sb, type)) {
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
2135
  		mutex_unlock(&sb_dqopt(sb)->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2136
2137
2138
2139
2140
2141
2142
2143
2144
  		return -ESRCH;
  	}
  	mi = sb_dqopt(sb)->info + type;
  	spin_lock(&dq_data_lock);
  	ii->dqi_bgrace = mi->dqi_bgrace;
  	ii->dqi_igrace = mi->dqi_igrace;
  	ii->dqi_flags = mi->dqi_flags & DQF_MASK;
  	ii->dqi_valid = IIF_ALL;
  	spin_unlock(&dq_data_lock);
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
2145
  	mutex_unlock(&sb_dqopt(sb)->dqonoff_mutex);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2146
2147
2148
2149
2150
2151
2152
  	return 0;
  }
  
  /* Generic routine for setting common part of quota file information */
  int vfs_set_dqinfo(struct super_block *sb, int type, struct if_dqinfo *ii)
  {
  	struct mem_dqinfo *mi;
f55abc0fb   Jan Kara   quota: Allow to s...
2153
  	int err = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2154

d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
2155
  	mutex_lock(&sb_dqopt(sb)->dqonoff_mutex);
f55abc0fb   Jan Kara   quota: Allow to s...
2156
2157
2158
  	if (!sb_has_quota_active(sb, type)) {
  		err = -ESRCH;
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
  	}
  	mi = sb_dqopt(sb)->info + type;
  	spin_lock(&dq_data_lock);
  	if (ii->dqi_valid & IIF_BGRACE)
  		mi->dqi_bgrace = ii->dqi_bgrace;
  	if (ii->dqi_valid & IIF_IGRACE)
  		mi->dqi_igrace = ii->dqi_igrace;
  	if (ii->dqi_valid & IIF_FLAGS)
  		mi->dqi_flags = (mi->dqi_flags & ~DQF_MASK) | (ii->dqi_flags & DQF_MASK);
  	spin_unlock(&dq_data_lock);
  	mark_info_dirty(sb, type);
  	/* Force write to disk */
  	sb->dq_op->write_info(sb, type);
f55abc0fb   Jan Kara   quota: Allow to s...
2172
  out:
d3be915fc   Ingo Molnar   [PATCH] sem2mutex...
2173
  	mutex_unlock(&sb_dqopt(sb)->dqonoff_mutex);
f55abc0fb   Jan Kara   quota: Allow to s...
2174
  	return err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
2190
2191
2192
2193
2194
2195
2196
2197
2198
2199
2200
2201
2202
2203
2204
2205
2206
2207
2208
2209
2210
2211
2212
2213
2214
2215
2216
2217
2218
2219
2220
2221
2222
2223
2224
2225
2226
2227
2228
2229
2230
2231
2232
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
2244
2245
2246
2247
2248
2249
2250
2251
  }
  
  struct quotactl_ops vfs_quotactl_ops = {
  	.quota_on	= vfs_quota_on,
  	.quota_off	= vfs_quota_off,
  	.quota_sync	= vfs_quota_sync,
  	.get_info	= vfs_get_dqinfo,
  	.set_info	= vfs_set_dqinfo,
  	.get_dqblk	= vfs_get_dqblk,
  	.set_dqblk	= vfs_set_dqblk
  };
  
  static ctl_table fs_dqstats_table[] = {
  	{
  		.ctl_name	= FS_DQ_LOOKUPS,
  		.procname	= "lookups",
  		.data		= &dqstats.lookups,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_DROPS,
  		.procname	= "drops",
  		.data		= &dqstats.drops,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_READS,
  		.procname	= "reads",
  		.data		= &dqstats.reads,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_WRITES,
  		.procname	= "writes",
  		.data		= &dqstats.writes,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_CACHE_HITS,
  		.procname	= "cache_hits",
  		.data		= &dqstats.cache_hits,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_ALLOCATED,
  		.procname	= "allocated_dquots",
  		.data		= &dqstats.allocated_dquots,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_FREE,
  		.procname	= "free_dquots",
  		.data		= &dqstats.free_dquots,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
  	{
  		.ctl_name	= FS_DQ_SYNCS,
  		.procname	= "syncs",
  		.data		= &dqstats.syncs,
  		.maxlen		= sizeof(int),
  		.mode		= 0444,
  		.proc_handler	= &proc_dointvec,
  	},
8e8934695   Jan Kara   quota: send messa...
2252
  #ifdef CONFIG_PRINT_QUOTA_WARNING
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2253
2254
2255
2256
2257
2258
2259
2260
  	{
  		.ctl_name	= FS_DQ_WARNINGS,
  		.procname	= "warnings",
  		.data		= &flag_print_warnings,
  		.maxlen		= sizeof(int),
  		.mode		= 0644,
  		.proc_handler	= &proc_dointvec,
  	},
8e8934695   Jan Kara   quota: send messa...
2261
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2262
2263
2264
2265
2266
2267
2268
2269
2270
2271
2272
2273
2274
2275
2276
2277
2278
2279
2280
2281
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
  	{ .ctl_name = 0 },
  };
  
  static ctl_table fs_table[] = {
  	{
  		.ctl_name	= FS_DQSTATS,
  		.procname	= "quota",
  		.mode		= 0555,
  		.child		= fs_dqstats_table,
  	},
  	{ .ctl_name = 0 },
  };
  
  static ctl_table sys_table[] = {
  	{
  		.ctl_name	= CTL_FS,
  		.procname	= "fs",
  		.mode		= 0555,
  		.child		= fs_table,
  	},
  	{ .ctl_name = 0 },
  };
  
  static int __init dquot_init(void)
  {
  	int i;
  	unsigned long nr_hash, order;
  
  	printk(KERN_NOTICE "VFS: Disk quotas %s
  ", __DQUOT_VERSION__);
0b4d41471   Eric W. Biederman   [PATCH] sysctl: r...
2292
  	register_sysctl_table(sys_table);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2293

20c2df83d   Paul Mundt   mm: Remove slab d...
2294
  	dquot_cachep = kmem_cache_create("dquot",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2295
  			sizeof(struct dquot), sizeof(unsigned long) * 4,
fffb60f93   Paul Jackson   [PATCH] cpuset me...
2296
2297
  			(SLAB_HWCACHE_ALIGN|SLAB_RECLAIM_ACCOUNT|
  				SLAB_MEM_SPREAD|SLAB_PANIC),
20c2df83d   Paul Mundt   mm: Remove slab d...
2298
  			NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2299
2300
2301
2302
2303
2304
2305
2306
2307
2308
2309
2310
2311
2312
2313
2314
2315
2316
2317
2318
2319
2320
  
  	order = 0;
  	dquot_hash = (struct hlist_head *)__get_free_pages(GFP_ATOMIC, order);
  	if (!dquot_hash)
  		panic("Cannot create dquot hash table");
  
  	/* Find power-of-two hlist_heads which can fit into allocation */
  	nr_hash = (1UL << order) * PAGE_SIZE / sizeof(struct hlist_head);
  	dq_hash_bits = 0;
  	do {
  		dq_hash_bits++;
  	} while (nr_hash >> dq_hash_bits);
  	dq_hash_bits--;
  
  	nr_hash = 1UL << dq_hash_bits;
  	dq_hash_mask = nr_hash - 1;
  	for (i = 0; i < nr_hash; i++)
  		INIT_HLIST_HEAD(dquot_hash + i);
  
  	printk("Dquot-cache hash table entries: %ld (order %ld, %ld bytes)
  ",
  			nr_hash, order, (PAGE_SIZE << order));
8e1f936b7   Rusty Russell   mm: clean up and ...
2321
  	register_shrinker(&dqcache_shrinker);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2322

8e8934695   Jan Kara   quota: send messa...
2323
2324
2325
2326
2327
  #ifdef CONFIG_QUOTA_NETLINK_INTERFACE
  	if (genl_register_family(&quota_genl_family) != 0)
  		printk(KERN_ERR "VFS: Failed to create quota netlink interface.
  ");
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2328
2329
2330
2331
2332
2333
2334
2335
  	return 0;
  }
  module_init(dquot_init);
  
  EXPORT_SYMBOL(register_quota_format);
  EXPORT_SYMBOL(unregister_quota_format);
  EXPORT_SYMBOL(dqstats);
  EXPORT_SYMBOL(dq_data_lock);
f55abc0fb   Jan Kara   quota: Allow to s...
2336
  EXPORT_SYMBOL(vfs_quota_enable);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2337
  EXPORT_SYMBOL(vfs_quota_on);
77e69dac3   Al Viro   [PATCH] fix races...
2338
  EXPORT_SYMBOL(vfs_quota_on_path);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2339
  EXPORT_SYMBOL(vfs_quota_on_mount);
f55abc0fb   Jan Kara   quota: Allow to s...
2340
  EXPORT_SYMBOL(vfs_quota_disable);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2341
  EXPORT_SYMBOL(vfs_quota_off);
12c77527e   Jan Kara   quota: Implement ...
2342
  EXPORT_SYMBOL(dquot_scan_active);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2343
2344
2345
2346
2347
2348
2349
2350
2351
2352
2353
2354
  EXPORT_SYMBOL(vfs_quota_sync);
  EXPORT_SYMBOL(vfs_get_dqinfo);
  EXPORT_SYMBOL(vfs_set_dqinfo);
  EXPORT_SYMBOL(vfs_get_dqblk);
  EXPORT_SYMBOL(vfs_set_dqblk);
  EXPORT_SYMBOL(dquot_commit);
  EXPORT_SYMBOL(dquot_commit_info);
  EXPORT_SYMBOL(dquot_acquire);
  EXPORT_SYMBOL(dquot_release);
  EXPORT_SYMBOL(dquot_mark_dquot_dirty);
  EXPORT_SYMBOL(dquot_initialize);
  EXPORT_SYMBOL(dquot_drop);
b85f4b87a   Jan Kara   quota: rename quo...
2355
  EXPORT_SYMBOL(vfs_dq_drop);
3d9ea253a   Jan Kara   quota: Add helper...
2356
2357
  EXPORT_SYMBOL(dqget);
  EXPORT_SYMBOL(dqput);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2358
2359
2360
2361
2362
  EXPORT_SYMBOL(dquot_alloc_space);
  EXPORT_SYMBOL(dquot_alloc_inode);
  EXPORT_SYMBOL(dquot_free_space);
  EXPORT_SYMBOL(dquot_free_inode);
  EXPORT_SYMBOL(dquot_transfer);
b85f4b87a   Jan Kara   quota: rename quo...
2363
2364
  EXPORT_SYMBOL(vfs_dq_transfer);
  EXPORT_SYMBOL(vfs_dq_quota_on_remount);