diff options
author | Ralf Baechle <ralf@linux-mips.org> | 1998-08-25 09:12:35 +0000 |
---|---|---|
committer | Ralf Baechle <ralf@linux-mips.org> | 1998-08-25 09:12:35 +0000 |
commit | c7fc24dc4420057f103afe8fc64524ebc25c5d37 (patch) | |
tree | 3682407a599b8f9f03fc096298134cafba1c9b2f /fs/dquot.c | |
parent | 1d793fade8b063fde3cf275bf1a5c2d381292cd9 (diff) |
o Merge with Linux 2.1.116.
o New Newport console code.
o New G364 console code.
Diffstat (limited to 'fs/dquot.c')
-rw-r--r-- | fs/dquot.c | 1079 |
1 files changed, 646 insertions, 433 deletions
diff --git a/fs/dquot.c b/fs/dquot.c index 58393b016..9ec40618e 100644 --- a/fs/dquot.c +++ b/fs/dquot.c @@ -1,41 +1,30 @@ /* * Implementation of the diskquota system for the LINUX operating - * system. QUOTA is implemented using the BSD systemcall interface as + * system. QUOTA is implemented using the BSD system call interface as * the means of communication with the user level. Currently only the - * ext2-filesystem has support for diskquotas. Other filesystems may - * be added in future time. This file contains the generic routines + * ext2 filesystem has support for disk quotas. Other filesystems may + * be added in the future. 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 the LINUX inode-subsystem with added complexity of the - * diskquota system. This implementation is not based on any BSD - * kernel sourcecode. + * based on one of the several variants of the LINUX inode-subsystem + * with added complexity of the diskquota system. * - * Version: $Id: dquot.c,v 1.11 1997/01/06 06:53:02 davem Exp $ + * Version: $Id: dquot.c,v 6.3 1996/11/17 18:35:34 mvw Exp mvw $ * - * Author: Marco van Wieringen <mvw@mcs.ow.nl> <mvw@tnix.net> - * - * Fixes: Dmitry Gorodchanin <pgmdsg@ibi.com>, 11 Feb 96 - * removed race conditions in dqput(), dqget() and iput(). - * Andi Kleen removed all verify_area() calls, 31 Dec 96 - * Nick Kralevich <nickkral@cal.alumni.berkeley.edu>, 21 Jul 97 - * Fixed a condition where user and group quotas could get mixed up. + * Author: Marco van Wieringen <mvw@planets.elm.net> * - * Chris Rankin <rankinc@bellsouth.net>, 31 Dec 97, 2-4 Jan 98 - * Fixed kernel API so that the user can get the quota for any - * group s/he belongs to. Also return useful error codes when - * turning quotas off, and fixed sync_dquot() so that all devices - * are synced when dev==NODEV. - * - * (C) Copyright 1994, 1995 Marco van Wieringen + * Fixes: Dmitry Gorodchanin <pgmdsg@ibi.com>, 11 Feb 96 * + * (C) Copyright 1994 - 1997 Marco van Wieringen */ #include <linux/errno.h> #include <linux/kernel.h> #include <linux/sched.h> + #include <linux/types.h> #include <linux/string.h> #include <linux/fcntl.h> @@ -46,129 +35,153 @@ #include <linux/mount.h> #include <linux/smp.h> #include <linux/smp_lock.h> +#include <linux/init.h> +#include <linux/slab.h> #include <asm/uaccess.h> -#define __DQUOT_VERSION__ "dquot_5.6.0" +#define __DQUOT_VERSION__ "dquot_6.4.0" + +int nr_dquots = 0, nr_free_dquots = 0; +int max_dquots = NR_DQUOTS; static char quotamessage[MAX_QUOTA_MESSAGE]; static char *quotatypes[] = INITQFNAMES; -static int nr_dquots = 0, nr_free_dquots = 0; -static struct dquot *hash_table[NR_DQHASH]; -static struct dquot *first_dquot; +static kmem_cache_t *dquot_cachep; + +static struct dquot *dquot_hash[NR_DQHASH]; +static struct free_dquot_queue { + struct dquot *head; + struct dquot **last; +} free_dquots = { NULL, &free_dquots.head }; +static struct dquot *inuse_list = NULL; +static int dquot_updating[NR_DQHASH]; + static struct dqstats dqstats; +static struct wait_queue *dquot_wait = (struct wait_queue *)NULL, + *update_wait = (struct wait_queue *)NULL; + +static inline char is_enabled(struct vfsmount *vfsmnt, short type) +{ + switch (type) { + case USRQUOTA: + return((vfsmnt->mnt_dquot.flags & DQUOT_USR_ENABLED) != 0); + case GRPQUOTA: + return((vfsmnt->mnt_dquot.flags & DQUOT_GRP_ENABLED) != 0); + } + return(0); +} -static struct wait_queue *dquot_wait = (struct wait_queue *)NULL; +static inline char sb_has_quota_enabled(struct super_block *sb, short type) +{ + struct vfsmount *vfsmnt; -extern void add_dquot_ref(kdev_t dev, short type); -extern void reset_dquot_ptrs(kdev_t dev, short type); + return((vfsmnt = lookup_vfsmnt(sb->s_dev)) != (struct vfsmount *)NULL && is_enabled(vfsmnt, type)); +} -#ifndef min -#define min(a,b) ((a) < (b)) ? (a) : (b) -#endif +static inline char dev_has_quota_enabled(kdev_t dev, short type) +{ + struct vfsmount *vfsmnt; + + return((vfsmnt = lookup_vfsmnt(dev)) != (struct vfsmount *)NULL && is_enabled(vfsmnt, type)); +} -/* - * Functions for management of the hashlist. - */ static inline int const hashfn(kdev_t dev, unsigned int id, short type) { return((HASHDEV(dev) ^ id) * (MAXQUOTAS - type)) % NR_DQHASH; } -static inline struct dquot **const hash(kdev_t dev, unsigned int id, short type) +static inline void insert_dquot_hash(struct dquot *dquot) { - return(hash_table + hashfn(dev, id, type)); + struct dquot **htable; + + htable = &dquot_hash[hashfn(dquot->dq_dev, dquot->dq_id, dquot->dq_type)]; + if ((dquot->dq_hash_next = *htable) != NULL) + (*htable)->dq_hash_pprev = &dquot->dq_hash_next; + *htable = dquot; + dquot->dq_hash_pprev = htable; } -static inline int has_quota_enabled(kdev_t dev, short type) +static inline void hash_dquot(struct dquot *dquot) { - struct vfsmount *vfsmnt; - - return((vfsmnt = lookup_vfsmnt(dev)) != (struct vfsmount *)NULL && - (vfsmnt->mnt_quotas[type] != (struct file *)NULL)); + insert_dquot_hash(dquot); } -static void insert_dquot_free(struct dquot *dquot) +static inline void unhash_dquot(struct dquot *dquot) { - dquot->dq_next = first_dquot; - dquot->dq_prev = first_dquot->dq_prev; - dquot->dq_next->dq_prev = dquot; - dquot->dq_prev->dq_next = dquot; - first_dquot = dquot; + if (dquot->dq_hash_pprev) { + if (dquot->dq_hash_next) + dquot->dq_hash_next->dq_hash_pprev = dquot->dq_hash_pprev; + *(dquot->dq_hash_pprev) = dquot->dq_hash_next; + dquot->dq_hash_pprev = NULL; + } } -static void remove_dquot_free(struct dquot *dquot) +static inline struct dquot *find_dquot(unsigned int hashent, kdev_t dev, unsigned int id, short type) { - if (first_dquot == dquot) - first_dquot = first_dquot->dq_next; - if (dquot->dq_next) - dquot->dq_next->dq_prev = dquot->dq_prev; - if (dquot->dq_prev) - dquot->dq_prev->dq_next = dquot->dq_next; - dquot->dq_next = dquot->dq_prev = NODQUOT; + struct dquot *dquot; + + for (dquot = dquot_hash[hashent]; dquot; dquot = dquot->dq_hash_next) + if (dquot->dq_dev == dev && dquot->dq_id == id && dquot->dq_type == type) + break; + return dquot; } -static void insert_dquot_hash(struct dquot *dquot) +static inline void put_dquot_head(struct dquot *dquot) { - struct dquot **hash_ent; - - hash_ent = hash(dquot->dq_dev, dquot->dq_id, dquot->dq_type); - dquot->dq_hash_next = *hash_ent; - dquot->dq_hash_prev = NODQUOT; - if (dquot->dq_hash_next) - dquot->dq_hash_next->dq_hash_prev = dquot; - *hash_ent = dquot; + if ((dquot->dq_next = free_dquots.head) != NULL) + free_dquots.head->dq_pprev = &dquot->dq_next; + else + free_dquots.last = &dquot->dq_next; + free_dquots.head = dquot; + dquot->dq_pprev = &free_dquots.head; + nr_free_dquots++; } -static void remove_dquot_hash(struct dquot *dquot) +static inline void put_dquot_last(struct dquot *dquot) { - struct dquot **hash_ent; - - hash_ent = hash(dquot->dq_dev, dquot->dq_id, dquot->dq_type); - if (*hash_ent == dquot) - *hash_ent = dquot->dq_hash_next; - if (dquot->dq_hash_next) - dquot->dq_hash_next->dq_hash_prev = dquot->dq_hash_prev; - if (dquot->dq_hash_prev) - dquot->dq_hash_prev->dq_hash_next = dquot->dq_hash_next; - dquot->dq_hash_prev = dquot->dq_hash_next = NODQUOT; + dquot->dq_next = NULL; + dquot->dq_pprev = free_dquots.last; + *free_dquots.last = dquot; + free_dquots.last = &dquot->dq_next; + nr_free_dquots++; } -static void put_last_free(struct dquot *dquot) +static inline void remove_free_dquot(struct dquot *dquot) { - remove_dquot_free(dquot); - dquot->dq_prev = first_dquot->dq_prev; - dquot->dq_prev->dq_next = dquot; - dquot->dq_next = first_dquot; - dquot->dq_next->dq_prev = dquot; + if (dquot->dq_pprev) { + if (dquot->dq_next) + dquot->dq_next->dq_pprev = dquot->dq_pprev; + else + free_dquots.last = dquot->dq_pprev; + *dquot->dq_pprev = dquot->dq_next; + dquot->dq_pprev = NULL; + nr_free_dquots--; + } } -static void grow_dquots(void) +static inline void put_inuse(struct dquot *dquot) { - struct dquot *dquot; - int cnt; + if ((dquot->dq_next = inuse_list) != NULL) + inuse_list->dq_pprev = &dquot->dq_next; + inuse_list = dquot; + dquot->dq_pprev = &inuse_list; +} - if (!(dquot = (struct dquot*) get_free_page(GFP_KERNEL))) - return; - dqstats.pages_allocated++; - cnt = PAGE_SIZE / sizeof(struct dquot); - nr_dquots += cnt; - nr_free_dquots += cnt; - if (!first_dquot) { - dquot->dq_next = dquot->dq_prev = first_dquot = dquot++; - cnt--; +static inline void remove_inuse(struct dquot *dquot) +{ + if (dquot->dq_pprev) { + if (dquot->dq_next) + dquot->dq_next->dq_pprev = dquot->dq_pprev; + *dquot->dq_pprev = dquot->dq_next; + dquot->dq_pprev = NULL; } - for (; cnt; cnt--) - insert_dquot_free(dquot++); } -/* - * Functions for locking and waiting on dquots. - */ static void __wait_on_dquot(struct dquot *dquot) { - struct wait_queue wait = {current, NULL}; + struct wait_queue wait = { current, NULL }; add_wait_queue(&dquot->dq_wait, &wait); repeat: @@ -202,39 +215,22 @@ static inline void unlock_dquot(struct dquot *dquot) wake_up(&dquot->dq_wait); } } -/* - * Note that we don't want to disturb any wait-queues when we discard - * an dquot. - * - * FIXME: As soon as we have a nice solution for the inode problem we - * can also fix this one. I.e. the volatile part. - */ -static void clear_dquot(struct dquot * dquot) -{ - struct wait_queue *wait; - - wait_on_dquot(dquot); - remove_dquot_hash(dquot); - remove_dquot_free(dquot); - wait = ((volatile struct dquot *) dquot)->dq_wait; - if (dquot->dq_count) - nr_free_dquots++; - memset(dquot, 0, sizeof(*dquot)); - ((volatile struct dquot *) dquot)->dq_wait = wait; - insert_dquot_free(dquot); -} static void write_dquot(struct dquot *dquot) { - short type = dquot->dq_type; - struct file *filp = dquot->dq_mnt->mnt_quotas[type]; + short type; + struct file *filp; mm_segment_t fs; loff_t offset; + type = dquot->dq_type; + filp = dquot->dq_mnt->mnt_dquot.files[type]; + if (!(dquot->dq_flags & DQ_MOD) || (filp == (struct file *)NULL)) return; + lock_dquot(dquot); - down(&dquot->dq_mnt->mnt_sem); + down(&dquot->dq_mnt->mnt_dquot.semaphore); offset = dqoff(dquot->dq_id); fs = get_fs(); set_fs(KERNEL_DS); @@ -242,29 +238,35 @@ static void write_dquot(struct dquot *dquot) if (filp->f_op->write(filp, (char *)&dquot->dq_dqb, sizeof(struct dqblk), &offset) == sizeof(struct dqblk)) dquot->dq_flags &= ~DQ_MOD; - up(&dquot->dq_mnt->mnt_sem); + up(&dquot->dq_mnt->mnt_dquot.semaphore); set_fs(fs); + unlock_dquot(dquot); dqstats.writes++; } static void read_dquot(struct dquot *dquot) { - short type = dquot->dq_type; - struct file *filp = dquot->dq_mnt->mnt_quotas[type]; + short type; + struct file *filp; mm_segment_t fs; loff_t offset; + type = dquot->dq_type; + filp = dquot->dq_mnt->mnt_dquot.files[type]; + if (filp == (struct file *)NULL) return; + lock_dquot(dquot); - down(&dquot->dq_mnt->mnt_sem); + down(&dquot->dq_mnt->mnt_dquot.semaphore); offset = dqoff(dquot->dq_id); fs = get_fs(); set_fs(KERNEL_DS); filp->f_op->read(filp, (char *)&dquot->dq_dqb, sizeof(struct dqblk), &offset); - up(&dquot->dq_mnt->mnt_sem); + up(&dquot->dq_mnt->mnt_dquot.semaphore); set_fs(fs); + if (dquot->dq_bhardlimit == 0 && dquot->dq_bsoftlimit == 0 && dquot->dq_ihardlimit == 0 && dquot->dq_isoftlimit == 0) dquot->dq_flags |= DQ_FAKE; @@ -272,46 +274,307 @@ static void read_dquot(struct dquot *dquot) dqstats.reads++; } +void clear_dquot(struct dquot *dquot) +{ + struct wait_queue *wait; + + /* So we don't disappear. */ + dquot->dq_count++; + + wait_on_dquot(dquot); + + if (--dquot->dq_count > 0) + remove_inuse(dquot); + else + remove_free_dquot(dquot); + unhash_dquot(dquot); + wait = dquot->dq_wait; + memset(dquot, 0, sizeof(*dquot)); barrier(); + dquot->dq_wait = wait; + put_dquot_head(dquot); +} + +void invalidate_dquots(kdev_t dev, short type) +{ + struct dquot *dquot, *next = NULL; + int pass = 0; + + dquot = free_dquots.head; +repeat: + while (dquot) { + next = dquot->dq_next; + if (dquot->dq_dev != dev || dquot->dq_type != type) + goto next; + clear_dquot(dquot); + next: + dquot = next; + } + + if (pass == 0) { + dquot = inuse_list; + pass = 1; + goto repeat; + } +} + int sync_dquots(kdev_t dev, short type) { - struct dquot *dquot = first_dquot; - int i; + struct dquot *dquot, *next; + int pass = 0; - dqstats.syncs++; - for (i = 0; i < nr_dquots * 2; i++, dquot = dquot->dq_next) { - if ((dev != NODEV && dquot->dq_dev != dev) || dquot->dq_count == 0 ) - continue; - if (type != -1 && dquot->dq_type != type) - continue; + dquot = free_dquots.head; +repeat: + while (dquot) { + next = dquot->dq_next; + if ((dev && dquot->dq_dev != dev) || + (type != -1 && dquot->dq_type != type)) + goto next; wait_on_dquot(dquot); if (dquot->dq_flags & DQ_MOD) write_dquot(dquot); + next: + dquot = next; } + + if (pass == 0) { + dquot = inuse_list; + pass = 1; + goto repeat; + } + dqstats.syncs++; return(0); } -/* - * Trash the cache for a certain type on a device. - */ -void invalidate_dquots(kdev_t dev, short type) +void dqput(struct dquot *dquot) { - struct dquot *dquot, *next; - int cnt; + if (!dquot) + return; - next = first_dquot; - for (cnt = nr_dquots ; cnt > 0 ; cnt--) { - dquot = next; - next = dquot->dq_next; - if (dquot->dq_dev != dev || dquot->dq_type != type) + /* + * If the dq_mnt pointer isn't initialized this entry needs no + * checking and doesn't need to be written. It just an empty + * dquot that is put back on to the freelist. + */ + if (dquot->dq_mnt != (struct vfsmount *)NULL) { + dqstats.drops++; + wait_on_dquot(dquot); + + if (!dquot->dq_count) { + printk("VFS: dqput: trying to free free dquot\n"); + printk("VFS: device %s, dquot of %s %d\n", kdevname(dquot->dq_dev), + quotatypes[dquot->dq_type], dquot->dq_id); + return; + } +we_slept: + if (dquot->dq_count > 1) { + dquot->dq_count--; + return; + } else { + wake_up(&dquot_wait); + + if (dquot->dq_flags & DQ_MOD) { + write_dquot(dquot); + wait_on_dquot(dquot); + goto we_slept; + } + } + } + + if (--dquot->dq_count == 0) { + remove_inuse(dquot); + put_dquot_last(dquot); /* Place at end of LRU free queue */ + } + + return; +} + +static void grow_dquots(void) +{ + struct dquot *dquot; + int cnt = 32; + + while (cnt > 0) { + dquot = kmem_cache_alloc(dquot_cachep, SLAB_KERNEL); + if(!dquot) + return; + + nr_dquots++; + memset((caddr_t)dquot, 0, sizeof(struct dquot)); + put_dquot_head(dquot); + cnt--; + } +} + +static struct dquot *find_best_candidate_weighted(struct dquot *dquot) +{ + int limit, myscore; + unsigned long bestscore; + struct dquot *best = NULL; + + if (dquot) { + bestscore = 2147483647; + limit = nr_free_dquots >> 2; + do { + if (!((dquot->dq_flags & DQ_LOCKED) || (dquot->dq_flags & DQ_MOD))) { + myscore = dquot->dq_referenced; + if (myscore < bestscore) { + bestscore = myscore; + best = dquot; + } + } + dquot = dquot->dq_next; + } while (dquot && --limit); + } + return best; +} + +static inline struct dquot *find_best_free(struct dquot *dquot) +{ + int limit; + + if (dquot) { + limit = nr_free_dquots >> 5; + do { + if (dquot->dq_referenced == 0) + return dquot; + dquot = dquot->dq_next; + } while (dquot && --limit); + } + return NULL; +} + +struct dquot *get_empty_dquot(void) +{ + struct dquot *dquot; + +repeat: + dquot = find_best_free(free_dquots.head); + if (!dquot) + goto pressure; +got_it: + dquot->dq_count++; + wait_on_dquot(dquot); + unhash_dquot(dquot); + remove_free_dquot(dquot); + + memset(dquot, 0, sizeof(*dquot)); + dquot->dq_count = 1; + + put_inuse(dquot); + return dquot; +pressure: + if (nr_dquots < max_dquots) { + grow_dquots(); + goto repeat; + } + + dquot = find_best_candidate_weighted(free_dquots.head); + if (!dquot) { + printk("VFS: No free dquots, contact mvw@planets.elm.net\n"); + sleep_on(&dquot_wait); + goto repeat; + } + if (dquot->dq_flags & DQ_LOCKED) { + wait_on_dquot(dquot); + goto repeat; + } else if (dquot->dq_flags & DQ_MOD) { + write_dquot(dquot); + goto repeat; + } + goto got_it; +} + +struct dquot *dqget(kdev_t dev, unsigned int id, short type) +{ + unsigned int hashent = hashfn(dev, id, type); + struct dquot *dquot, *empty = NULL; + struct vfsmount *vfsmnt; + + if ((vfsmnt = lookup_vfsmnt(dev)) == (struct vfsmount *)NULL || is_enabled(vfsmnt, type) == 0) + return(NODQUOT); + +we_slept: + if ((dquot = find_dquot(hashent, dev, id, type)) == NULL) { + if (empty == NULL) { + dquot_updating[hashent]++; + empty = get_empty_dquot(); + if (!--dquot_updating[hashent]) + wake_up(&update_wait); + goto we_slept; + } + dquot = empty; + dquot->dq_id = id; + dquot->dq_type = type; + dquot->dq_dev = dev; + dquot->dq_mnt = vfsmnt; + read_dquot(dquot); + hash_dquot(dquot); + } else { + if (!dquot->dq_count++) { + remove_free_dquot(dquot); + put_inuse(dquot); + } else + dqstats.cache_hits++; + wait_on_dquot(dquot); + if (empty) + dqput(empty); + } + + while (dquot_updating[hashent]) + sleep_on(&update_wait); + + dquot->dq_referenced++; + dqstats.lookups++; + + return dquot; +} + +static void add_dquot_ref(kdev_t dev, short type) +{ + struct super_block *sb = get_super(dev); + struct file *filp; + struct inode *inode; + + if (!sb || !sb->dq_op) + return; /* nothing to do */ + + for (filp = inuse_filps; filp; filp = filp->f_next) { + if (!filp->f_dentry) + continue; + if (filp->f_dentry->d_sb != sb) continue; - if (dquot->dq_flags & DQ_LOCKED) { - printk("VFS: dquot busy on removed device %s\n", kdevname(dev)); + inode = filp->f_dentry->d_inode; + if (!inode) continue; + if (filp->f_mode & FMODE_WRITE) { + sb->dq_op->initialize(inode, type); + inode->i_flags |= S_QUOTA; + } + } +} + +static void reset_dquot_ptrs(kdev_t dev, short type) +{ + struct super_block *sb = get_super(dev); + struct file *filp; + struct inode *inode; + + if (!sb || !sb->dq_op) + return; /* nothing to do */ + + for (filp = inuse_filps; filp; filp = filp->f_next) { + if (!filp->f_dentry) + continue; + if (filp->f_dentry->d_sb != sb) + continue; + inode = filp->f_dentry->d_inode; + if (!inode) + continue; + if (IS_QUOTAINIT(inode)) { + sb->dq_op->drop(inode); + inode->i_dquot[type] = NODQUOT; + inode->i_flags &= ~S_QUOTA; } - if (dquot->dq_flags & DQ_MOD) - write_dquot(dquot); - dqstats.drops++; - clear_dquot(dquot); } } @@ -359,237 +622,117 @@ static inline void dquot_decr_blocks(struct dquot *dquot, unsigned long number) unlock_dquot(dquot); } -static inline int need_print_warning(short type, struct dquot *dquot) +static inline char need_print_warning(short type, uid_t initiator, struct dquot *dquot) { switch (type) { case USRQUOTA: - return(current->fsuid == dquot->dq_id); + return(initiator == dquot->dq_id); case GRPQUOTA: - return(current->fsgid == dquot->dq_id); + return(initiator == dquot->dq_id); } return(0); } -static int check_idq(struct dquot *dquot, short type, u_long short inodes) +static inline char ignore_hardlimit(struct dquot *dquot, uid_t initiator) +{ + return(initiator == 0 && dquot->dq_mnt->mnt_dquot.rsquash[dquot->dq_type] == 0); +} + +static int check_idq(struct dquot *dquot, short type, u_long short inodes, uid_t initiator, struct tty_struct *tty) { if (inodes <= 0 || dquot->dq_flags & DQ_FAKE) return(QUOTA_OK); + if (dquot->dq_ihardlimit && - (dquot->dq_curinodes + inodes) > dquot->dq_ihardlimit && - !capable(CAP_SYS_RESOURCE)) { + (dquot->dq_curinodes + inodes) > dquot->dq_ihardlimit && + !ignore_hardlimit(dquot, initiator)) { if ((dquot->dq_flags & DQ_INODES) == 0 && - need_print_warning(type, dquot)) { - sprintf(quotamessage, "%s: write failed, %s file limit reached\r\n", + need_print_warning(type, initiator, dquot)) { + sprintf(quotamessage, "%s: write failed, %s file limit reached\n", dquot->dq_mnt->mnt_dirname, quotatypes[type]); - tty_write_message(current->tty, quotamessage); + tty_write_message(tty, quotamessage); dquot->dq_flags |= DQ_INODES; } return(NO_QUOTA); } + if (dquot->dq_isoftlimit && (dquot->dq_curinodes + inodes) > dquot->dq_isoftlimit && - dquot->dq_itime && CURRENT_TIME >= dquot->dq_itime && - !capable(CAP_SYS_RESOURCE)) { - if (need_print_warning(type, dquot)) { - sprintf(quotamessage, "%s: warning, %s file quota exceeded too long.\r\n", + dquot->dq_itime && CURRENT_TIME >= dquot->dq_itime && + !ignore_hardlimit(dquot, initiator)) { + if (need_print_warning(type, initiator, dquot)) { + sprintf(quotamessage, "%s: warning, %s file quota exceeded too long.\n", dquot->dq_mnt->mnt_dirname, quotatypes[type]); - tty_write_message(current->tty, quotamessage); + tty_write_message(tty, quotamessage); } return(NO_QUOTA); } + if (dquot->dq_isoftlimit && (dquot->dq_curinodes + inodes) > dquot->dq_isoftlimit && - dquot->dq_itime == 0 && - !capable(CAP_SYS_RESOURCE)) { - if (need_print_warning(type, dquot)) { - sprintf(quotamessage, "%s: warning, %s file quota exceeded\r\n", + dquot->dq_itime == 0) { + if (need_print_warning(type, initiator, dquot)) { + sprintf(quotamessage, "%s: warning, %s file quota exceeded\n", dquot->dq_mnt->mnt_dirname, quotatypes[type]); - tty_write_message(current->tty, quotamessage); + tty_write_message(tty, quotamessage); } - dquot->dq_itime = CURRENT_TIME + dquot->dq_mnt->mnt_iexp[type]; + dquot->dq_itime = CURRENT_TIME + dquot->dq_mnt->mnt_dquot.inode_expire[type]; } + return(QUOTA_OK); } -static int check_bdq(struct dquot *dquot, short type, u_long blocks) +static int check_bdq(struct dquot *dquot, short type, u_long blocks, uid_t initiator, struct tty_struct *tty, char warn) { if (blocks <= 0 || dquot->dq_flags & DQ_FAKE) return(QUOTA_OK); + if (dquot->dq_bhardlimit && - (dquot->dq_curblocks + blocks) > dquot->dq_bhardlimit && - !capable(CAP_SYS_RESOURCE)) { - if ((dquot->dq_flags & DQ_BLKS) == 0 && - need_print_warning(type, dquot)) { - sprintf(quotamessage, "%s: write failed, %s disk limit reached.\r\n", + (dquot->dq_curblocks + blocks) > dquot->dq_bhardlimit && + !ignore_hardlimit(dquot, initiator)) { + if (warn && (dquot->dq_flags & DQ_BLKS) == 0 && + need_print_warning(type, initiator, dquot)) { + sprintf(quotamessage, "%s: write failed, %s disk limit reached.\n", dquot->dq_mnt->mnt_dirname, quotatypes[type]); - tty_write_message(current->tty, quotamessage); + tty_write_message(tty, quotamessage); dquot->dq_flags |= DQ_BLKS; } return(NO_QUOTA); } + if (dquot->dq_bsoftlimit && (dquot->dq_curblocks + blocks) > dquot->dq_bsoftlimit && - dquot->dq_btime && CURRENT_TIME >= dquot->dq_btime && - !capable(CAP_SYS_RESOURCE)) { - if (need_print_warning(type, dquot)) { - sprintf(quotamessage, "%s: write failed, %s disk quota exceeded too long.\r\n", + dquot->dq_btime && CURRENT_TIME >= dquot->dq_btime && + !ignore_hardlimit(dquot, initiator)) { + if (warn && need_print_warning(type, initiator, dquot)) { + sprintf(quotamessage, "%s: write failed, %s disk quota exceeded too long.\n", dquot->dq_mnt->mnt_dirname, quotatypes[type]); - tty_write_message(current->tty, quotamessage); + tty_write_message(tty, quotamessage); } return(NO_QUOTA); } + if (dquot->dq_bsoftlimit && (dquot->dq_curblocks + blocks) > dquot->dq_bsoftlimit && - dquot->dq_btime == 0 && - !capable(CAP_SYS_RESOURCE)) { - if (need_print_warning(type, dquot)) { - sprintf(quotamessage, "%s: warning, %s disk quota exceeded\r\n", + dquot->dq_btime == 0) { + if (warn && need_print_warning(type, initiator, dquot)) { + sprintf(quotamessage, "%s: warning, %s disk quota exceeded\n", dquot->dq_mnt->mnt_dirname, quotatypes[type]); - tty_write_message(current->tty, quotamessage); - } - dquot->dq_btime = CURRENT_TIME + dquot->dq_mnt->mnt_bexp[type]; - } - return(QUOTA_OK); -} - -static void dqput(struct dquot *dquot) -{ - if (!dquot) - return; - /* - * If the dq_mnt pointer isn't initialized this entry needs no - * checking and doesn't need to be written. It just an empty - * dquot that is put back into the freelist. - */ - if (dquot->dq_mnt != (struct vfsmount *)NULL) { - dqstats.drops++; - wait_on_dquot(dquot); - if (!dquot->dq_count) { - printk("VFS: dqput: trying to free free dquot\n"); - printk("VFS: device %s, dquot of %s %d\n", kdevname(dquot->dq_dev), - quotatypes[dquot->dq_type], dquot->dq_id); - return; - } -repeat: - if (dquot->dq_count > 1) { - dquot->dq_count--; - return; - } - wake_up(&dquot_wait); - if (dquot->dq_flags & DQ_MOD) { - write_dquot(dquot); /* we can sleep - so do again */ - wait_on_dquot(dquot); - goto repeat; - } - } - if (dquot->dq_count) { - dquot->dq_count--; - nr_free_dquots++; - } - return; -} - -static struct dquot *get_empty_dquot(void) -{ - struct dquot *dquot, *best; - int cnt; - - if (nr_dquots < NR_DQUOTS && nr_free_dquots < (nr_dquots >> 2)) - grow_dquots(); - -repeat: - dquot = first_dquot; - best = NODQUOT; - for (cnt = 0; cnt < nr_dquots; dquot = dquot->dq_next, cnt++) { - if (!dquot->dq_count) { - if (!best) - best = dquot; - if (!(dquot->dq_flags & DQ_MOD) && !(dquot->dq_flags & DQ_LOCKED)) { - best = dquot; - break; - } - } - } - if (!best || best->dq_flags & DQ_MOD || best->dq_flags & DQ_LOCKED) - if (nr_dquots < NR_DQUOTS) { - grow_dquots(); - goto repeat; + tty_write_message(tty, quotamessage); } - dquot = best; - if (!dquot) { - printk("VFS: No free dquots - contact mvw@mcs.ow.org\n"); - sleep_on(&dquot_wait); - goto repeat; - } - if (dquot->dq_flags & DQ_LOCKED) { - wait_on_dquot(dquot); - goto repeat; - } - if (dquot->dq_flags & DQ_MOD) { - write_dquot(dquot); - goto repeat; - } - if (dquot->dq_count) - goto repeat; - clear_dquot(dquot); - dquot->dq_count = 1; - nr_free_dquots--; - if (nr_free_dquots < 0) { - printk ("VFS: get_empty_dquot: bad free dquot count.\n"); - nr_free_dquots = 0; + dquot->dq_btime = CURRENT_TIME + dquot->dq_mnt->mnt_dquot.block_expire[type]; } - return(dquot); -} - -static struct dquot *dqget(kdev_t dev, unsigned int id, short type) -{ - struct dquot *dquot, *empty; - struct vfsmount *vfsmnt; - if ((vfsmnt = lookup_vfsmnt(dev)) == (struct vfsmount *)NULL || - (vfsmnt->mnt_quotas[type] == (struct file *)0)) - return(NODQUOT); - dqstats.lookups++; - empty = get_empty_dquot(); -repeat: - dquot = *(hash(dev, id, type)); - while (dquot) { - if (dquot->dq_dev != dev || dquot->dq_id != id || - dquot->dq_type != type) { - dquot = dquot->dq_hash_next; - continue; - } - wait_on_dquot(dquot); - if (dquot->dq_dev != dev || dquot->dq_id != id || - dquot->dq_type != type) - goto repeat; - if (!dquot->dq_count) - nr_free_dquots--; - dquot->dq_count++; - if (empty) - dqput(empty); - dqstats.cache_hits++; - return(dquot); - } - if (!empty) - return(NODQUOT); - dquot = empty; - dquot->dq_id = id; - dquot->dq_type = type; - dquot->dq_dev = dev; - dquot->dq_mnt = vfsmnt; - put_last_free(dquot); - insert_dquot_hash(dquot); - read_dquot(dquot); - return(dquot); + return(QUOTA_OK); } /* * Initialize a dquot-struct with new quota info. This is used by the - * systemcall interface functions. + * system call interface functions. */ static int set_dqblk(kdev_t dev, int id, short type, int flags, struct dqblk *dqblk) { + int error; struct dquot *dquot; struct dqblk dq_dqblk; @@ -597,47 +740,49 @@ static int set_dqblk(kdev_t dev, int id, short type, int flags, struct dqblk *dq return(-EFAULT); if (flags & QUOTA_SYSCALL) { - if (copy_from_user(&dq_dqblk, dqblk, sizeof(struct dqblk))) - return -EFAULT; - } else { - memcpy(&dq_dqblk, dqblk, sizeof(struct dqblk)); - } + if ((error = copy_from_user((caddr_t)&dq_dqblk, (caddr_t)dqblk, sizeof(struct dqblk))) != 0) + return(error); + } else + memcpy((caddr_t)&dq_dqblk, (caddr_t)dqblk, sizeof(struct dqblk)); + if ((dquot = dqget(dev, id, type)) != NODQUOT) { lock_dquot(dquot); - if (id > 0 && (flags & (SET_QUOTA|SET_QLIMIT))) { + + if (id > 0 && ((flags & SET_QUOTA) || (flags & SET_QLIMIT))) { dquot->dq_bhardlimit = dq_dqblk.dqb_bhardlimit; dquot->dq_bsoftlimit = dq_dqblk.dqb_bsoftlimit; dquot->dq_ihardlimit = dq_dqblk.dqb_ihardlimit; dquot->dq_isoftlimit = dq_dqblk.dqb_isoftlimit; } - if (flags & (SET_QUOTA|SET_USE)) { + + if ((flags & SET_QUOTA) || (flags & SET_USE)) { if (dquot->dq_isoftlimit && dquot->dq_curinodes < dquot->dq_isoftlimit && dq_dqblk.dqb_curinodes >= dquot->dq_isoftlimit) - dquot->dq_itime = CURRENT_TIME + dquot->dq_mnt->mnt_iexp[type]; + dquot->dq_itime = CURRENT_TIME + dquot->dq_mnt->mnt_dquot.inode_expire[type]; dquot->dq_curinodes = dq_dqblk.dqb_curinodes; if (dquot->dq_curinodes < dquot->dq_isoftlimit) dquot->dq_flags &= ~DQ_INODES; if (dquot->dq_bsoftlimit && dquot->dq_curblocks < dquot->dq_bsoftlimit && dq_dqblk.dqb_curblocks >= dquot->dq_bsoftlimit) - dquot->dq_btime = CURRENT_TIME + dquot->dq_mnt->mnt_bexp[type]; + dquot->dq_btime = CURRENT_TIME + dquot->dq_mnt->mnt_dquot.block_expire[type]; dquot->dq_curblocks = dq_dqblk.dqb_curblocks; if (dquot->dq_curblocks < dquot->dq_bsoftlimit) dquot->dq_flags &= ~DQ_BLKS; } + if (id == 0) { - /* - * Change in expiretimes, change them in dq_mnt. - */ - dquot->dq_mnt->mnt_bexp[type] = dquot->dq_btime = dq_dqblk.dqb_btime; - dquot->dq_mnt->mnt_iexp[type] = dquot->dq_itime = dq_dqblk.dqb_itime; + dquot->dq_mnt->mnt_dquot.block_expire[type] = dquot->dq_btime = dq_dqblk.dqb_btime; + dquot->dq_mnt->mnt_dquot.inode_expire[type] = dquot->dq_itime = dq_dqblk.dqb_itime; } + if (dq_dqblk.dqb_bhardlimit == 0 && dq_dqblk.dqb_bsoftlimit == 0 && dq_dqblk.dqb_ihardlimit == 0 && dq_dqblk.dqb_isoftlimit == 0) dquot->dq_flags |= DQ_FAKE; else dquot->dq_flags &= ~DQ_FAKE; + dquot->dq_flags |= DQ_MOD; unlock_dquot(dquot); dqput(dquot); @@ -650,14 +795,14 @@ static int get_quota(kdev_t dev, int id, short type, struct dqblk *dqblk) struct dquot *dquot; int error; - if (has_quota_enabled(dev, type)) { + if (dev_has_quota_enabled(dev, type)) { if (dqblk == (struct dqblk *)NULL) return(-EFAULT); if ((dquot = dqget(dev, id, type)) != NODQUOT) { - error = copy_to_user(dqblk, (char *)&dquot->dq_dqb, sizeof(struct dqblk)); + error = copy_to_user((caddr_t)dqblk, (caddr_t)&dquot->dq_dqb, sizeof(struct dqblk)); dqput(dquot); - return error ? -EFAULT : 0; + return(error); } } return(-ESRCH); @@ -667,25 +812,67 @@ static int get_stats(caddr_t addr) { dqstats.allocated_dquots = nr_dquots; dqstats.free_dquots = nr_free_dquots; - return copy_to_user(addr, (caddr_t)&dqstats, sizeof(struct dqstats)) - ? -EFAULT : 0; + return(copy_to_user(addr, (caddr_t)&dqstats, sizeof(struct dqstats))); +} + +static int quota_root_squash(kdev_t dev, short type, int *addr) +{ + struct vfsmount *vfsmnt; + int new_value, error; + + if ((vfsmnt = lookup_vfsmnt(dev)) == (struct vfsmount *)NULL) + return(-ENODEV); + + if ((error = copy_from_user((caddr_t)&new_value, (caddr_t)addr, sizeof(int))) != 0) + return(error); + + vfsmnt->mnt_dquot.rsquash[type] = new_value; + return(0); } /* - * Initialize pointer in an inode to the right dquots. + * This is a simple algorithm that calculates the size of a file in blocks. + * This is only used on filesystems that do not have an i_blocks count. + */ +static u_long isize_to_blocks(size_t isize, size_t blksize) +{ + u_long blocks; + u_long indirect; + + if (!blksize) + blksize = BLOCK_SIZE; + blocks = (isize / blksize) + ((isize % blksize) ? 1 : 0); + if (blocks > 10) { + indirect = ((blocks - 11) >> 8) + 1; /* single indirect blocks */ + if (blocks > (10 + 256)) { + indirect += ((blocks - 267) >> 16) + 1; /* double indirect blocks */ + if (blocks > (10 + 256 + (256 << 8))) + indirect++; /* triple indirect blocks */ + } + blocks += indirect; + } + return(blocks); +} + +/* + * Externally referenced functions through dquot_operations in inode. */ void dquot_initialize(struct inode *inode, short type) { + struct dquot *dquot; unsigned int id = 0; short cnt; - struct dquot *tmp; - if (S_ISREG(inode->i_mode) || S_ISDIR(inode->i_mode) || S_ISLNK(inode->i_mode)) { + if (S_ISREG(inode->i_mode) || + S_ISDIR(inode->i_mode) || + S_ISLNK(inode->i_mode)) { for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (type != -1 && cnt != type) continue; - if (!has_quota_enabled(inode->i_dev, cnt)) + + if (!sb_has_quota_enabled(inode->i_sb, cnt)) continue; + if (inode->i_dquot[cnt] == NODQUOT) { switch (cnt) { case USRQUOTA: @@ -695,17 +882,13 @@ void dquot_initialize(struct inode *inode, short type) id = inode->i_gid; break; } - - tmp = dqget(inode->i_dev, id, cnt); - /* We may sleep in dqget(), so check it again. - * Dmitry Gorodchanin 02/11/96 - */ + dquot = dqget(inode->i_dev, id, cnt); if (inode->i_dquot[cnt] != NODQUOT) { - dqput(tmp); + dqput(dquot); continue; } - inode->i_dquot[cnt] = tmp; - inode->i_flags |= S_WRITE; + inode->i_dquot[cnt] = dquot; + inode->i_flags |= S_QUOTA; } } } @@ -713,82 +896,58 @@ void dquot_initialize(struct inode *inode, short type) void dquot_drop(struct inode *inode) { + struct dquot *dquot; short cnt; - struct dquot * tmp; + inode->i_flags &= ~S_QUOTA; for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (inode->i_dquot[cnt] == NODQUOT) continue; - /* We can sleep at dqput(). So we must do it this way. - * Dmitry Gorodchanin 02/11/96 - */ - tmp = inode->i_dquot[cnt]; + dquot = inode->i_dquot[cnt]; inode->i_dquot[cnt] = NODQUOT; - dqput(tmp); - } - inode->i_flags &= ~S_WRITE; -} - -/* - * This is a simple algorithm that calculates the size of a file in blocks. - * This is only used on filesystems that do not have an i_blocks count. - */ -static u_long isize_to_blocks(size_t isize, size_t blksize) -{ - u_long blocks; - u_long indirect; - - if (!blksize) - blksize = BLOCK_SIZE; - blocks = (isize / blksize) + ((isize % blksize) ? 1 : 0); - if (blocks > 10) { - indirect = ((blocks - 11) >> 8) + 1; /* single indirect blocks */ - if (blocks > (10 + 256)) { - indirect += ((blocks - 267) >> 16) + 1; /* double indirect blocks */ - if (blocks > (10 + 256 + (256 << 8))) - indirect++; /* triple indirect blocks */ - } - blocks += indirect; + dqput(dquot); } - return(blocks); } -/* - * Externally referenced functions through dquot_operations. - */ -int dquot_alloc_block(const struct inode *inode, unsigned long number) +int dquot_alloc_block(const struct inode *inode, unsigned long number, uid_t initiator, char warn) { unsigned short cnt; + struct tty_struct *tty = current->tty; for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (inode->i_dquot[cnt] == NODQUOT) continue; - if (check_bdq(inode->i_dquot[cnt], cnt, number)) + if (check_bdq(inode->i_dquot[cnt], cnt, number, initiator, tty, warn)) return(NO_QUOTA); } + for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (inode->i_dquot[cnt] == NODQUOT) continue; dquot_incr_blocks(inode->i_dquot[cnt], number); } + return(QUOTA_OK); } -int dquot_alloc_inode(const struct inode *inode, unsigned long number) +int dquot_alloc_inode(const struct inode *inode, unsigned long number, uid_t initiator) { unsigned short cnt; + struct tty_struct *tty = current->tty; for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (inode->i_dquot[cnt] == NODQUOT) continue; - if (check_idq(inode->i_dquot[cnt], cnt, number)) + if (check_idq(inode->i_dquot[cnt], cnt, number, initiator, tty)) return(NO_QUOTA); } + for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (inode->i_dquot[cnt] == NODQUOT) continue; dquot_incr_inodes(inode->i_dquot[cnt], number); } + return(QUOTA_OK); } @@ -817,15 +976,16 @@ void dquot_free_inode(const struct inode *inode, unsigned long number) /* * Transfer the number of inode and blocks from one diskquota to an other. */ -int dquot_transfer(struct inode *inode, struct iattr *iattr, char direction) +int dquot_transfer(struct inode *inode, struct iattr *iattr, char direction, uid_t initiator) { unsigned long blocks; struct dquot *transfer_from[MAXQUOTAS]; struct dquot *transfer_to[MAXQUOTAS]; + struct tty_struct *tty = current->tty; short cnt, disc; /* - * Find out if this filesystems uses i_blocks. + * Find out if this filesystem uses i_blocks. */ if (inode->i_blksize == 0) blocks = isize_to_blocks(inode->i_size, BLOCK_SIZE); @@ -840,7 +1000,7 @@ int dquot_transfer(struct inode *inode, struct iattr *iattr, char direction) transfer_from[cnt] = NODQUOT; transfer_to[cnt] = NODQUOT; - if (!has_quota_enabled(inode->i_dev, cnt)) + if (!sb_has_quota_enabled(inode->i_sb, cnt)) continue; switch (cnt) { @@ -858,8 +1018,8 @@ int dquot_transfer(struct inode *inode, struct iattr *iattr, char direction) break; } - if (check_idq(transfer_to[cnt], cnt, 1) == NO_QUOTA || - check_bdq(transfer_to[cnt], cnt, blocks) == NO_QUOTA) { + if (check_idq(transfer_to[cnt], cnt, 1, initiator, tty) == NO_QUOTA || + check_bdq(transfer_to[cnt], cnt, blocks, initiator, tty, 0) == NO_QUOTA) { for (disc = 0; disc <= cnt; disc++) { dqput(transfer_from[disc]); dqput(transfer_to[disc]); @@ -883,10 +1043,12 @@ int dquot_transfer(struct inode *inode, struct iattr *iattr, char direction) dquot_decr_inodes(transfer_from[cnt], 1); dquot_decr_blocks(transfer_from[cnt], blocks); } + if (transfer_to[cnt] != NODQUOT) { dquot_incr_inodes(transfer_to[cnt], 1); dquot_incr_blocks(transfer_to[cnt], blocks); } + if (inode->i_dquot[cnt] != NODQUOT) { dqput(transfer_from[cnt]); dqput(inode->i_dquot[cnt]); @@ -896,16 +1058,24 @@ int dquot_transfer(struct inode *inode, struct iattr *iattr, char direction) dqput(transfer_to[cnt]); } } + return(QUOTA_OK); } -void dquot_init(void) + +__initfunc(void dquot_init_hash(void)) { - printk(KERN_NOTICE "VFS: Diskquotas version %s initialized\r\n", - __DQUOT_VERSION__); - memset(hash_table, 0, sizeof(hash_table)); + printk(KERN_NOTICE "VFS: Diskquotas version %s initialized\n", __DQUOT_VERSION__); + + dquot_cachep = kmem_cache_create("dquot", sizeof(struct dquot), + sizeof(unsigned long) * 4, + SLAB_HWCACHE_ALIGN, NULL, NULL); + + if (!dquot_cachep) + panic("Cannot create dquot SLAB cache\n"); + + memset(dquot_hash, 0, sizeof(dquot_hash)); memset((caddr_t)&dqstats, 0, sizeof(dqstats)); - first_dquot = NODQUOT; } /* @@ -921,6 +1091,30 @@ struct dquot_operations dquot_operations = { dquot_transfer }; +static inline void set_enable_flags(struct vfsmount *vfsmnt, short type) +{ + switch (type) { + case USRQUOTA: + vfsmnt->mnt_dquot.flags |= DQUOT_USR_ENABLED; + break; + case GRPQUOTA: + vfsmnt->mnt_dquot.flags |= DQUOT_GRP_ENABLED; + break; + } +} + +static inline void reset_enable_flags(struct vfsmount *vfsmnt, short type) +{ + switch (type) { + case USRQUOTA: + vfsmnt->mnt_dquot.flags &= ~DQUOT_USR_ENABLED; + break; + case GRPQUOTA: + vfsmnt->mnt_dquot.flags &= ~DQUOT_GRP_ENABLED; + break; + } +} + /* * Turn quota off on a device. type == -1 ==> quotaoff for all types (umount) */ @@ -929,25 +1123,28 @@ int quota_off(kdev_t dev, short type) struct vfsmount *vfsmnt; short cnt; - if ( !(vfsmnt = lookup_vfsmnt(dev)) ) - return -ENODEV; - for (cnt = 0; cnt < MAXQUOTAS; cnt++) { if (type != -1 && cnt != type) continue; - if (vfsmnt->mnt_quotas[cnt] == (struct file *)NULL) - { - if(type == -1) - continue; - return -ESRCH; - } + + if ((vfsmnt = lookup_vfsmnt(dev)) == (struct vfsmount *)NULL || + is_enabled(vfsmnt, cnt) == 0 || + vfsmnt->mnt_sb == (struct super_block *)NULL) + continue; + vfsmnt->mnt_sb->dq_op = (struct dquot_operations *)NULL; + reset_dquot_ptrs(dev, cnt); invalidate_dquots(dev, cnt); - fput(vfsmnt->mnt_quotas[cnt]); - vfsmnt->mnt_quotas[cnt] = (struct file *)NULL; - vfsmnt->mnt_iexp[cnt] = vfsmnt->mnt_bexp[cnt] = (time_t)NULL; + + fput(vfsmnt->mnt_dquot.files[cnt]); + + reset_enable_flags(vfsmnt, cnt); + vfsmnt->mnt_dquot.files[cnt] = (struct file *)NULL; + vfsmnt->mnt_dquot.inode_expire[cnt] = 0; + vfsmnt->mnt_dquot.block_expire[cnt] = 0; } + return(0); } @@ -962,11 +1159,11 @@ int quota_on(kdev_t dev, short type, char *path) int error; vfsmnt = lookup_vfsmnt(dev); - if (vfsmnt == NULL) + if (vfsmnt == (struct vfsmount *)NULL) return -ENODEV; - if (vfsmnt->mnt_quotas[type] != NULL) - return -EBUSY; + if (is_enabled(vfsmnt, type)) + return(-EBUSY); tmp = getname(path); error = PTR_ERR(tmp); @@ -986,8 +1183,13 @@ int quota_on(kdev_t dev, short type, char *path) return -EACCES; } + if (inode->i_size == 0 || (inode->i_size % sizeof(struct dqblk)) != 0) { + dput(dentry); + return(-EINVAL); + } + filp = get_empty_filp(); - if (filp != NULL) { + if (filp != (struct file *)NULL) { filp->f_mode = (O_RDWR + 1) & O_ACCMODE; filp->f_flags = O_RDWR; filp->f_dentry = dentry; @@ -1000,14 +1202,18 @@ int quota_on(kdev_t dev, short type, char *path) if (filp->f_op && filp->f_op->open) error = filp->f_op->open(inode, filp); if (!error) { - vfsmnt->mnt_quotas[type] = filp; + set_enable_flags(vfsmnt, type); + vfsmnt->mnt_dquot.files[type] = filp; + dquot = dqget(dev, 0, type); - vfsmnt->mnt_iexp[type] = (dquot) ? dquot->dq_itime : MAX_IQ_TIME; - vfsmnt->mnt_bexp[type] = (dquot) ? dquot->dq_btime : MAX_DQ_TIME; + vfsmnt->mnt_dquot.inode_expire[type] = (dquot) ? dquot->dq_itime : MAX_IQ_TIME; + vfsmnt->mnt_dquot.block_expire[type] = (dquot) ? dquot->dq_btime : MAX_DQ_TIME; dqput(dquot); + vfsmnt->mnt_sb->dq_op = &dquot_operations; add_dquot_ref(dev, type); - return 0; + + return(0); } put_write_access(inode); } @@ -1016,15 +1222,17 @@ int quota_on(kdev_t dev, short type, char *path) put_filp(filp); } else error = -EMFILE; + dput(dentry); - return error; + + return(error); } /* - * Ok this is the systemcall interface, this communicates with - * the userlevel programs. Currently this only supports diskquota - * calls. Maybe we need to add the process quotas etc in the future. - * But we probably better use rlimits for that. + * This is the system call interface. This communicates with + * the user-level programs. Currently this only supports diskquota + * calls. Maybe we need to add the process quotas etc. in the future, + * but we probably should use rlimits for that. */ asmlinkage int sys_quotactl(int cmd, const char *special, int id, caddr_t addr) { @@ -1045,17 +1253,17 @@ asmlinkage int sys_quotactl(int cmd, const char *special, int id, caddr_t addr) break; case Q_GETQUOTA: if (((type == USRQUOTA && current->uid != id) || - (type == GRPQUOTA && in_group_p(id))) && - !capable(CAP_SYS_ADMIN)) + (type == GRPQUOTA && current->gid != id)) && + !capable(CAP_SYS_RESOURCE)) goto out; break; default: - if (!capable(CAP_SYS_ADMIN)) + if (!capable(CAP_SYS_RESOURCE)) goto out; } ret = -EINVAL; - dev = NODEV; + dev = 0; if (special != NULL || (cmds != Q_SYNC && cmds != Q_GETSTATS)) { mode_t mode; struct dentry * dentry; @@ -1098,12 +1306,17 @@ asmlinkage int sys_quotactl(int cmd, const char *special, int id, caddr_t addr) goto out; case Q_GETSTATS: ret = get_stats(addr); + goto out; + case Q_RSQUASH: + ret = quota_root_squash(dev, type, (int *) addr); + goto out; default: goto out; } flags |= QUOTA_SYSCALL; - if (has_quota_enabled(dev, type)) + + if (dev_has_quota_enabled(dev, type)) ret = set_dqblk(dev, id, type, flags, (struct dqblk *) addr); else ret = -ESRCH; |