2  *      vfsv0 quota IO operations on file
 
   5 #include <linux/errno.h>
 
   7 #include <linux/mount.h>
 
   8 #include <linux/dqblk_v2.h>
 
   9 #include <linux/kernel.h>
 
  10 #include <linux/init.h>
 
  11 #include <linux/module.h>
 
  12 #include <linux/slab.h>
 
  13 #include <linux/quotaops.h>
 
  15 #include <asm/byteorder.h>
 
  17 #include "quota_tree.h"
 
  19 MODULE_AUTHOR("Jan Kara");
 
  20 MODULE_DESCRIPTION("Quota trie support");
 
  21 MODULE_LICENSE("GPL");
 
  23 #define __QUOTA_QT_PARANOIA
 
  25 typedef char *dqbuf_t;
 
  27 static int get_index(struct qtree_mem_dqinfo *info, qid_t id, int depth)
 
  29         unsigned int epb = info->dqi_usable_bs >> 2;
 
  31         depth = info->dqi_qtree_depth - depth - 1;
 
  37 /* Number of entries in one blocks */
 
  38 static inline int qtree_dqstr_in_blk(struct qtree_mem_dqinfo *info)
 
  40         return (info->dqi_usable_bs - sizeof(struct qt_disk_dqdbheader))
 
  41                / info->dqi_entry_size;
 
  44 static dqbuf_t getdqbuf(size_t size)
 
  46         dqbuf_t buf = kmalloc(size, GFP_NOFS);
 
  48                 printk(KERN_WARNING "VFS: Not enough memory for quota buffers.\n");
 
  52 static inline void freedqbuf(dqbuf_t buf)
 
  57 static inline ssize_t read_blk(struct qtree_mem_dqinfo *info, uint blk, dqbuf_t buf)
 
  59         struct super_block *sb = info->dqi_sb;
 
  61         memset(buf, 0, info->dqi_usable_bs);
 
  62         return sb->s_op->quota_read(sb, info->dqi_type, (char *)buf,
 
  63                info->dqi_usable_bs, blk << info->dqi_blocksize_bits);
 
  66 static inline ssize_t write_blk(struct qtree_mem_dqinfo *info, uint blk, dqbuf_t buf)
 
  68         struct super_block *sb = info->dqi_sb;
 
  70         return sb->s_op->quota_write(sb, info->dqi_type, (char *)buf,
 
  71                info->dqi_usable_bs, blk << info->dqi_blocksize_bits);
 
  74 /* Remove empty block from list and return it */
 
  75 static int get_free_dqblk(struct qtree_mem_dqinfo *info)
 
  77         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
  78         struct qt_disk_dqdbheader *dh = (struct qt_disk_dqdbheader *)buf;
 
  83         if (info->dqi_free_blk) {
 
  84                 blk = info->dqi_free_blk;
 
  85                 ret = read_blk(info, blk, buf);
 
  88                 info->dqi_free_blk = le32_to_cpu(dh->dqdh_next_free);
 
  91                 memset(buf, 0, info->dqi_usable_bs);
 
  92                 /* Assure block allocation... */
 
  93                 ret = write_blk(info, info->dqi_blocks, buf);
 
  96                 blk = info->dqi_blocks++;
 
  98         mark_info_dirty(info->dqi_sb, info->dqi_type);
 
 105 /* Insert empty block to the list */
 
 106 static int put_free_dqblk(struct qtree_mem_dqinfo *info, dqbuf_t buf, uint blk)
 
 108         struct qt_disk_dqdbheader *dh = (struct qt_disk_dqdbheader *)buf;
 
 111         dh->dqdh_next_free = cpu_to_le32(info->dqi_free_blk);
 
 112         dh->dqdh_prev_free = cpu_to_le32(0);
 
 113         dh->dqdh_entries = cpu_to_le16(0);
 
 114         err = write_blk(info, blk, buf);
 
 117         info->dqi_free_blk = blk;
 
 118         mark_info_dirty(info->dqi_sb, info->dqi_type);
 
 122 /* Remove given block from the list of blocks with free entries */
 
 123 static int remove_free_dqentry(struct qtree_mem_dqinfo *info, dqbuf_t buf, uint blk)
 
 125         dqbuf_t tmpbuf = getdqbuf(info->dqi_usable_bs);
 
 126         struct qt_disk_dqdbheader *dh = (struct qt_disk_dqdbheader *)buf;
 
 127         uint nextblk = le32_to_cpu(dh->dqdh_next_free);
 
 128         uint prevblk = le32_to_cpu(dh->dqdh_prev_free);
 
 134                 err = read_blk(info, nextblk, tmpbuf);
 
 137                 ((struct qt_disk_dqdbheader *)tmpbuf)->dqdh_prev_free =
 
 139                 err = write_blk(info, nextblk, tmpbuf);
 
 144                 err = read_blk(info, prevblk, tmpbuf);
 
 147                 ((struct qt_disk_dqdbheader *)tmpbuf)->dqdh_next_free =
 
 149                 err = write_blk(info, prevblk, tmpbuf);
 
 153                 info->dqi_free_entry = nextblk;
 
 154                 mark_info_dirty(info->dqi_sb, info->dqi_type);
 
 157         dh->dqdh_next_free = dh->dqdh_prev_free = cpu_to_le32(0);
 
 158         /* No matter whether write succeeds block is out of list */
 
 159         if (write_blk(info, blk, buf) < 0)
 
 160                 printk(KERN_ERR "VFS: Can't write block (%u) with free entries.\n", blk);
 
 167 /* Insert given block to the beginning of list with free entries */
 
 168 static int insert_free_dqentry(struct qtree_mem_dqinfo *info, dqbuf_t buf, uint blk)
 
 170         dqbuf_t tmpbuf = getdqbuf(info->dqi_usable_bs);
 
 171         struct qt_disk_dqdbheader *dh = (struct qt_disk_dqdbheader *)buf;
 
 176         dh->dqdh_next_free = cpu_to_le32(info->dqi_free_entry);
 
 177         dh->dqdh_prev_free = cpu_to_le32(0);
 
 178         err = write_blk(info, blk, buf);
 
 181         if (info->dqi_free_entry) {
 
 182                 err = read_blk(info, info->dqi_free_entry, tmpbuf);
 
 185                 ((struct qt_disk_dqdbheader *)tmpbuf)->dqdh_prev_free =
 
 187                 err = write_blk(info, info->dqi_free_entry, tmpbuf);
 
 192         info->dqi_free_entry = blk;
 
 193         mark_info_dirty(info->dqi_sb, info->dqi_type);
 
 200 /* Is the entry in the block free? */
 
 201 int qtree_entry_unused(struct qtree_mem_dqinfo *info, char *disk)
 
 205         for (i = 0; i < info->dqi_entry_size; i++)
 
 210 EXPORT_SYMBOL(qtree_entry_unused);
 
 212 /* Find space for dquot */
 
 213 static uint find_free_dqentry(struct qtree_mem_dqinfo *info,
 
 214                               struct dquot *dquot, int *err)
 
 217         struct qt_disk_dqdbheader *dh;
 
 218         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
 226         dh = (struct qt_disk_dqdbheader *)buf;
 
 227         if (info->dqi_free_entry) {
 
 228                 blk = info->dqi_free_entry;
 
 229                 *err = read_blk(info, blk, buf);
 
 233                 blk = get_free_dqblk(info);
 
 239                 memset(buf, 0, info->dqi_usable_bs);
 
 240                 /* This is enough as block is already zeroed and entry list is empty... */
 
 241                 info->dqi_free_entry = blk;
 
 242                 mark_info_dirty(dquot->dq_sb, dquot->dq_type);
 
 244         /* Block will be full? */
 
 245         if (le16_to_cpu(dh->dqdh_entries) + 1 >= qtree_dqstr_in_blk(info)) {
 
 246                 *err = remove_free_dqentry(info, buf, blk);
 
 248                         printk(KERN_ERR "VFS: find_free_dqentry(): Can't "
 
 249                                "remove block (%u) from entry free list.\n",
 
 254         le16_add_cpu(&dh->dqdh_entries, 1);
 
 255         /* Find free structure in block */
 
 256         for (i = 0, ddquot = ((char *)buf) + sizeof(struct qt_disk_dqdbheader);
 
 257              i < qtree_dqstr_in_blk(info) && !qtree_entry_unused(info, ddquot);
 
 258              i++, ddquot += info->dqi_entry_size);
 
 259 #ifdef __QUOTA_QT_PARANOIA
 
 260         if (i == qtree_dqstr_in_blk(info)) {
 
 261                 printk(KERN_ERR "VFS: find_free_dqentry(): Data block full "
 
 262                                 "but it shouldn't.\n");
 
 267         *err = write_blk(info, blk, buf);
 
 269                 printk(KERN_ERR "VFS: find_free_dqentry(): Can't write quota "
 
 270                                 "data block %u.\n", blk);
 
 273         dquot->dq_off = (blk << info->dqi_blocksize_bits) +
 
 274                         sizeof(struct qt_disk_dqdbheader) +
 
 275                         i * info->dqi_entry_size;
 
 283 /* Insert reference to structure into the trie */
 
 284 static int do_insert_tree(struct qtree_mem_dqinfo *info, struct dquot *dquot,
 
 285                           uint *treeblk, int depth)
 
 287         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
 288         int ret = 0, newson = 0, newact = 0;
 
 295                 ret = get_free_dqblk(info);
 
 299                 memset(buf, 0, info->dqi_usable_bs);
 
 302                 ret = read_blk(info, *treeblk, buf);
 
 304                         printk(KERN_ERR "VFS: Can't read tree quota block "
 
 310         newblk = le32_to_cpu(ref[get_index(info, dquot->dq_id, depth)]);
 
 313         if (depth == info->dqi_qtree_depth - 1) {
 
 314 #ifdef __QUOTA_QT_PARANOIA
 
 316                         printk(KERN_ERR "VFS: Inserting already present quota "
 
 317                                         "entry (block %u).\n",
 
 318                                le32_to_cpu(ref[get_index(info,
 
 319                                                 dquot->dq_id, depth)]));
 
 324                 newblk = find_free_dqentry(info, dquot, &ret);
 
 326                 ret = do_insert_tree(info, dquot, &newblk, depth+1);
 
 328         if (newson && ret >= 0) {
 
 329                 ref[get_index(info, dquot->dq_id, depth)] =
 
 331                 ret = write_blk(info, *treeblk, buf);
 
 332         } else if (newact && ret < 0) {
 
 333                 put_free_dqblk(info, buf, *treeblk);
 
 340 /* Wrapper for inserting quota structure into tree */
 
 341 static inline int dq_insert_tree(struct qtree_mem_dqinfo *info,
 
 344         int tmp = QT_TREEOFF;
 
 345         return do_insert_tree(info, dquot, &tmp, 0);
 
 349  *      We don't have to be afraid of deadlocks as we never have quotas on quota files...
 
 351 int qtree_write_dquot(struct qtree_mem_dqinfo *info, struct dquot *dquot)
 
 353         int type = dquot->dq_type;
 
 354         struct super_block *sb = dquot->dq_sb;
 
 356         dqbuf_t ddquot = getdqbuf(info->dqi_entry_size);
 
 361         /* dq_off is guarded by dqio_mutex */
 
 362         if (!dquot->dq_off) {
 
 363                 ret = dq_insert_tree(info, dquot);
 
 365                         printk(KERN_ERR "VFS: Error %zd occurred while "
 
 366                                         "creating quota.\n", ret);
 
 371         spin_lock(&dq_data_lock);
 
 372         info->dqi_ops->mem2disk_dqblk(ddquot, dquot);
 
 373         spin_unlock(&dq_data_lock);
 
 374         ret = sb->s_op->quota_write(sb, type, (char *)ddquot,
 
 375                                         info->dqi_entry_size, dquot->dq_off);
 
 376         if (ret != info->dqi_entry_size) {
 
 377                 printk(KERN_WARNING "VFS: dquota write failed on dev %s\n",
 
 389 EXPORT_SYMBOL(qtree_write_dquot);
 
 391 /* Free dquot entry in data block */
 
 392 static int free_dqentry(struct qtree_mem_dqinfo *info, struct dquot *dquot,
 
 395         struct qt_disk_dqdbheader *dh;
 
 396         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
 401         if (dquot->dq_off >> info->dqi_blocksize_bits != blk) {
 
 402                 printk(KERN_ERR "VFS: Quota structure has offset to other "
 
 403                   "block (%u) than it should (%u).\n", blk,
 
 404                   (uint)(dquot->dq_off >> info->dqi_blocksize_bits));
 
 407         ret = read_blk(info, blk, buf);
 
 409                 printk(KERN_ERR "VFS: Can't read quota data block %u\n", blk);
 
 412         dh = (struct qt_disk_dqdbheader *)buf;
 
 413         le16_add_cpu(&dh->dqdh_entries, -1);
 
 414         if (!le16_to_cpu(dh->dqdh_entries)) {   /* Block got free? */
 
 415                 ret = remove_free_dqentry(info, buf, blk);
 
 417                         ret = put_free_dqblk(info, buf, blk);
 
 419                         printk(KERN_ERR "VFS: Can't move quota data block (%u) "
 
 420                           "to free list.\n", blk);
 
 425                        (dquot->dq_off & ((1 << info->dqi_blocksize_bits) - 1)),
 
 426                        0, info->dqi_entry_size);
 
 427                 if (le16_to_cpu(dh->dqdh_entries) ==
 
 428                     qtree_dqstr_in_blk(info) - 1) {
 
 429                         /* Insert will write block itself */
 
 430                         ret = insert_free_dqentry(info, buf, blk);
 
 432                                 printk(KERN_ERR "VFS: Can't insert quota data "
 
 433                                        "block (%u) to free entry list.\n", blk);
 
 437                         ret = write_blk(info, blk, buf);
 
 439                                 printk(KERN_ERR "VFS: Can't write quota data "
 
 445         dquot->dq_off = 0;      /* Quota is now unattached */
 
 451 /* Remove reference to dquot from tree */
 
 452 static int remove_tree(struct qtree_mem_dqinfo *info, struct dquot *dquot,
 
 453                        uint *blk, int depth)
 
 455         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
 458         __le32 *ref = (__le32 *)buf;
 
 462         ret = read_blk(info, *blk, buf);
 
 464                 printk(KERN_ERR "VFS: Can't read quota data block %u\n", *blk);
 
 467         newblk = le32_to_cpu(ref[get_index(info, dquot->dq_id, depth)]);
 
 468         if (depth == info->dqi_qtree_depth - 1) {
 
 469                 ret = free_dqentry(info, dquot, newblk);
 
 472                 ret = remove_tree(info, dquot, &newblk, depth+1);
 
 474         if (ret >= 0 && !newblk) {
 
 476                 ref[get_index(info, dquot->dq_id, depth)] = cpu_to_le32(0);
 
 477                 /* Block got empty? */
 
 479                      i < (info->dqi_usable_bs >> 2) && !ref[i];
 
 481                 /* Don't put the root block into the free block list */
 
 482                 if (i == (info->dqi_usable_bs >> 2)
 
 483                     && *blk != QT_TREEOFF) {
 
 484                         put_free_dqblk(info, buf, *blk);
 
 487                         ret = write_blk(info, *blk, buf);
 
 489                                 printk(KERN_ERR "VFS: Can't write quota tree "
 
 490                                   "block %u.\n", *blk);
 
 498 /* Delete dquot from tree */
 
 499 int qtree_delete_dquot(struct qtree_mem_dqinfo *info, struct dquot *dquot)
 
 501         uint tmp = QT_TREEOFF;
 
 503         if (!dquot->dq_off)     /* Even not allocated? */
 
 505         return remove_tree(info, dquot, &tmp, 0);
 
 507 EXPORT_SYMBOL(qtree_delete_dquot);
 
 509 /* Find entry in block */
 
 510 static loff_t find_block_dqentry(struct qtree_mem_dqinfo *info,
 
 511                                  struct dquot *dquot, uint blk)
 
 513         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
 520         ret = read_blk(info, blk, buf);
 
 522                 printk(KERN_ERR "VFS: Can't read quota tree block %u.\n", blk);
 
 525         for (i = 0, ddquot = ((char *)buf) + sizeof(struct qt_disk_dqdbheader);
 
 526              i < qtree_dqstr_in_blk(info) && !info->dqi_ops->is_id(ddquot, dquot);
 
 527              i++, ddquot += info->dqi_entry_size);
 
 528         if (i == qtree_dqstr_in_blk(info)) {
 
 529                 printk(KERN_ERR "VFS: Quota for id %u referenced "
 
 530                   "but not present.\n", dquot->dq_id);
 
 534                 ret = (blk << info->dqi_blocksize_bits) + sizeof(struct
 
 535                   qt_disk_dqdbheader) + i * info->dqi_entry_size;
 
 542 /* Find entry for given id in the tree */
 
 543 static loff_t find_tree_dqentry(struct qtree_mem_dqinfo *info,
 
 544                                 struct dquot *dquot, uint blk, int depth)
 
 546         dqbuf_t buf = getdqbuf(info->dqi_usable_bs);
 
 548         __le32 *ref = (__le32 *)buf;
 
 552         ret = read_blk(info, blk, buf);
 
 554                 printk(KERN_ERR "VFS: Can't read quota tree block %u.\n", blk);
 
 558         blk = le32_to_cpu(ref[get_index(info, dquot->dq_id, depth)]);
 
 559         if (!blk)       /* No reference? */
 
 561         if (depth < info->dqi_qtree_depth - 1)
 
 562                 ret = find_tree_dqentry(info, dquot, blk, depth+1);
 
 564                 ret = find_block_dqentry(info, dquot, blk);
 
 570 /* Find entry for given id in the tree - wrapper function */
 
 571 static inline loff_t find_dqentry(struct qtree_mem_dqinfo *info,
 
 574         return find_tree_dqentry(info, dquot, QT_TREEOFF, 0);
 
 577 int qtree_read_dquot(struct qtree_mem_dqinfo *info, struct dquot *dquot)
 
 579         int type = dquot->dq_type;
 
 580         struct super_block *sb = dquot->dq_sb;
 
 585 #ifdef __QUOTA_QT_PARANOIA
 
 586         /* Invalidated quota? */
 
 587         if (!sb_dqopt(dquot->dq_sb)->files[type]) {
 
 588                 printk(KERN_ERR "VFS: Quota invalidated while reading!\n");
 
 592         /* Do we know offset of the dquot entry in the quota file? */
 
 593         if (!dquot->dq_off) {
 
 594                 offset = find_dqentry(info, dquot);
 
 595                 if (offset <= 0) {      /* Entry not present? */
 
 597                                 printk(KERN_ERR "VFS: Can't read quota "
 
 598                                   "structure for id %u.\n", dquot->dq_id);
 
 600                         set_bit(DQ_FAKE_B, &dquot->dq_flags);
 
 601                         memset(&dquot->dq_dqb, 0, sizeof(struct mem_dqblk));
 
 605                 dquot->dq_off = offset;
 
 607         ddquot = getdqbuf(info->dqi_entry_size);
 
 610         ret = sb->s_op->quota_read(sb, type, (char *)ddquot,
 
 611                                    info->dqi_entry_size, dquot->dq_off);
 
 612         if (ret != info->dqi_entry_size) {
 
 615                 printk(KERN_ERR "VFS: Error while reading quota "
 
 616                                 "structure for id %u.\n", dquot->dq_id);
 
 617                 set_bit(DQ_FAKE_B, &dquot->dq_flags);
 
 618                 memset(&dquot->dq_dqb, 0, sizeof(struct mem_dqblk));
 
 622         spin_lock(&dq_data_lock);
 
 623         info->dqi_ops->disk2mem_dqblk(dquot, ddquot);
 
 624         if (!dquot->dq_dqb.dqb_bhardlimit &&
 
 625             !dquot->dq_dqb.dqb_bsoftlimit &&
 
 626             !dquot->dq_dqb.dqb_ihardlimit &&
 
 627             !dquot->dq_dqb.dqb_isoftlimit)
 
 628                 set_bit(DQ_FAKE_B, &dquot->dq_flags);
 
 629         spin_unlock(&dq_data_lock);
 
 635 EXPORT_SYMBOL(qtree_read_dquot);
 
 637 /* Check whether dquot should not be deleted. We know we are
 
 638  * the only one operating on dquot (thanks to dq_lock) */
 
 639 int qtree_release_dquot(struct qtree_mem_dqinfo *info, struct dquot *dquot)
 
 641         if (test_bit(DQ_FAKE_B, &dquot->dq_flags) && !(dquot->dq_dqb.dqb_curinodes | dquot->dq_dqb.dqb_curspace))
 
 642                 return qtree_delete_dquot(info, dquot);
 
 645 EXPORT_SYMBOL(qtree_release_dquot);