diff options
Diffstat (limited to 'fs/ocfs2/quota_global.c')
| -rw-r--r-- | fs/ocfs2/quota_global.c | 134 |
1 files changed, 72 insertions, 62 deletions
diff --git a/fs/ocfs2/quota_global.c b/fs/ocfs2/quota_global.c index edfa60cd155c..bf7742d0ee3b 100644 --- a/fs/ocfs2/quota_global.c +++ b/fs/ocfs2/quota_global.c | |||
| @@ -69,6 +69,7 @@ static void ocfs2_global_mem2diskdqb(void *dp, struct dquot *dquot) | |||
| 69 | d->dqb_curspace = cpu_to_le64(m->dqb_curspace); | 69 | d->dqb_curspace = cpu_to_le64(m->dqb_curspace); |
| 70 | d->dqb_btime = cpu_to_le64(m->dqb_btime); | 70 | d->dqb_btime = cpu_to_le64(m->dqb_btime); |
| 71 | d->dqb_itime = cpu_to_le64(m->dqb_itime); | 71 | d->dqb_itime = cpu_to_le64(m->dqb_itime); |
| 72 | d->dqb_pad1 = d->dqb_pad2 = 0; | ||
| 72 | } | 73 | } |
| 73 | 74 | ||
| 74 | static int ocfs2_global_is_id(void *dp, struct dquot *dquot) | 75 | static int ocfs2_global_is_id(void *dp, struct dquot *dquot) |
| @@ -211,14 +212,13 @@ ssize_t ocfs2_quota_write(struct super_block *sb, int type, | |||
| 211 | 212 | ||
| 212 | mutex_lock_nested(&gqinode->i_mutex, I_MUTEX_QUOTA); | 213 | mutex_lock_nested(&gqinode->i_mutex, I_MUTEX_QUOTA); |
| 213 | if (gqinode->i_size < off + len) { | 214 | if (gqinode->i_size < off + len) { |
| 214 | down_write(&OCFS2_I(gqinode)->ip_alloc_sem); | 215 | loff_t rounded_end = |
| 215 | err = ocfs2_extend_no_holes(gqinode, off + len, off); | 216 | ocfs2_align_bytes_to_blocks(sb, off + len); |
| 216 | up_write(&OCFS2_I(gqinode)->ip_alloc_sem); | 217 | |
| 217 | if (err < 0) | 218 | /* Space is already allocated in ocfs2_global_read_dquot() */ |
| 218 | goto out; | ||
| 219 | err = ocfs2_simple_size_update(gqinode, | 219 | err = ocfs2_simple_size_update(gqinode, |
| 220 | oinfo->dqi_gqi_bh, | 220 | oinfo->dqi_gqi_bh, |
| 221 | off + len); | 221 | rounded_end); |
| 222 | if (err < 0) | 222 | if (err < 0) |
| 223 | goto out; | 223 | goto out; |
| 224 | new = 1; | 224 | new = 1; |
| @@ -234,7 +234,7 @@ ssize_t ocfs2_quota_write(struct super_block *sb, int type, | |||
| 234 | } | 234 | } |
| 235 | if (err) { | 235 | if (err) { |
| 236 | mlog_errno(err); | 236 | mlog_errno(err); |
| 237 | return err; | 237 | goto out; |
| 238 | } | 238 | } |
| 239 | lock_buffer(bh); | 239 | lock_buffer(bh); |
| 240 | if (new) | 240 | if (new) |
| @@ -342,7 +342,6 @@ int ocfs2_global_read_info(struct super_block *sb, int type) | |||
| 342 | info->dqi_bgrace = le32_to_cpu(dinfo.dqi_bgrace); | 342 | info->dqi_bgrace = le32_to_cpu(dinfo.dqi_bgrace); |
| 343 | info->dqi_igrace = le32_to_cpu(dinfo.dqi_igrace); | 343 | info->dqi_igrace = le32_to_cpu(dinfo.dqi_igrace); |
| 344 | oinfo->dqi_syncms = le32_to_cpu(dinfo.dqi_syncms); | 344 | oinfo->dqi_syncms = le32_to_cpu(dinfo.dqi_syncms); |
| 345 | oinfo->dqi_syncjiff = msecs_to_jiffies(oinfo->dqi_syncms); | ||
| 346 | oinfo->dqi_gi.dqi_blocks = le32_to_cpu(dinfo.dqi_blocks); | 345 | oinfo->dqi_gi.dqi_blocks = le32_to_cpu(dinfo.dqi_blocks); |
| 347 | oinfo->dqi_gi.dqi_free_blk = le32_to_cpu(dinfo.dqi_free_blk); | 346 | oinfo->dqi_gi.dqi_free_blk = le32_to_cpu(dinfo.dqi_free_blk); |
| 348 | oinfo->dqi_gi.dqi_free_entry = le32_to_cpu(dinfo.dqi_free_entry); | 347 | oinfo->dqi_gi.dqi_free_entry = le32_to_cpu(dinfo.dqi_free_entry); |
| @@ -352,7 +351,7 @@ int ocfs2_global_read_info(struct super_block *sb, int type) | |||
| 352 | oinfo->dqi_gi.dqi_qtree_depth = qtree_depth(&oinfo->dqi_gi); | 351 | oinfo->dqi_gi.dqi_qtree_depth = qtree_depth(&oinfo->dqi_gi); |
| 353 | INIT_DELAYED_WORK(&oinfo->dqi_sync_work, qsync_work_fn); | 352 | INIT_DELAYED_WORK(&oinfo->dqi_sync_work, qsync_work_fn); |
| 354 | queue_delayed_work(ocfs2_quota_wq, &oinfo->dqi_sync_work, | 353 | queue_delayed_work(ocfs2_quota_wq, &oinfo->dqi_sync_work, |
| 355 | oinfo->dqi_syncjiff); | 354 | msecs_to_jiffies(oinfo->dqi_syncms)); |
| 356 | 355 | ||
| 357 | out_err: | 356 | out_err: |
| 358 | mlog_exit(status); | 357 | mlog_exit(status); |
| @@ -402,13 +401,36 @@ int ocfs2_global_write_info(struct super_block *sb, int type) | |||
| 402 | return err; | 401 | return err; |
| 403 | } | 402 | } |
| 404 | 403 | ||
| 404 | static int ocfs2_global_qinit_alloc(struct super_block *sb, int type) | ||
| 405 | { | ||
| 406 | struct ocfs2_mem_dqinfo *oinfo = sb_dqinfo(sb, type)->dqi_priv; | ||
| 407 | |||
| 408 | /* | ||
| 409 | * We may need to allocate tree blocks and a leaf block but not the | ||
| 410 | * root block | ||
| 411 | */ | ||
| 412 | return oinfo->dqi_gi.dqi_qtree_depth; | ||
| 413 | } | ||
| 414 | |||
| 415 | static int ocfs2_calc_global_qinit_credits(struct super_block *sb, int type) | ||
| 416 | { | ||
| 417 | /* We modify all the allocated blocks, tree root, and info block */ | ||
| 418 | return (ocfs2_global_qinit_alloc(sb, type) + 2) * | ||
| 419 | OCFS2_QUOTA_BLOCK_UPDATE_CREDITS; | ||
| 420 | } | ||
| 421 | |||
| 405 | /* Read in information from global quota file and acquire a reference to it. | 422 | /* Read in information from global quota file and acquire a reference to it. |
| 406 | * dquot_acquire() has already started the transaction and locked quota file */ | 423 | * dquot_acquire() has already started the transaction and locked quota file */ |
| 407 | int ocfs2_global_read_dquot(struct dquot *dquot) | 424 | int ocfs2_global_read_dquot(struct dquot *dquot) |
| 408 | { | 425 | { |
| 409 | int err, err2, ex = 0; | 426 | int err, err2, ex = 0; |
| 410 | struct ocfs2_mem_dqinfo *info = | 427 | struct super_block *sb = dquot->dq_sb; |
| 411 | sb_dqinfo(dquot->dq_sb, dquot->dq_type)->dqi_priv; | 428 | int type = dquot->dq_type; |
| 429 | struct ocfs2_mem_dqinfo *info = sb_dqinfo(sb, type)->dqi_priv; | ||
| 430 | struct ocfs2_super *osb = OCFS2_SB(sb); | ||
| 431 | struct inode *gqinode = info->dqi_gqinode; | ||
| 432 | int need_alloc = ocfs2_global_qinit_alloc(sb, type); | ||
| 433 | handle_t *handle = NULL; | ||
| 412 | 434 | ||
| 413 | err = ocfs2_qinfo_lock(info, 0); | 435 | err = ocfs2_qinfo_lock(info, 0); |
| 414 | if (err < 0) | 436 | if (err < 0) |
| @@ -419,14 +441,33 @@ int ocfs2_global_read_dquot(struct dquot *dquot) | |||
| 419 | OCFS2_DQUOT(dquot)->dq_use_count++; | 441 | OCFS2_DQUOT(dquot)->dq_use_count++; |
| 420 | OCFS2_DQUOT(dquot)->dq_origspace = dquot->dq_dqb.dqb_curspace; | 442 | OCFS2_DQUOT(dquot)->dq_origspace = dquot->dq_dqb.dqb_curspace; |
| 421 | OCFS2_DQUOT(dquot)->dq_originodes = dquot->dq_dqb.dqb_curinodes; | 443 | OCFS2_DQUOT(dquot)->dq_originodes = dquot->dq_dqb.dqb_curinodes; |
| 444 | ocfs2_qinfo_unlock(info, 0); | ||
| 445 | |||
| 422 | if (!dquot->dq_off) { /* No real quota entry? */ | 446 | if (!dquot->dq_off) { /* No real quota entry? */ |
| 423 | /* Upgrade to exclusive lock for allocation */ | ||
| 424 | ocfs2_qinfo_unlock(info, 0); | ||
| 425 | err = ocfs2_qinfo_lock(info, 1); | ||
| 426 | if (err < 0) | ||
| 427 | goto out_qlock; | ||
| 428 | ex = 1; | 447 | ex = 1; |
| 448 | /* | ||
| 449 | * Add blocks to quota file before we start a transaction since | ||
| 450 | * locking allocators ranks above a transaction start | ||
| 451 | */ | ||
| 452 | WARN_ON(journal_current_handle()); | ||
| 453 | down_write(&OCFS2_I(gqinode)->ip_alloc_sem); | ||
| 454 | err = ocfs2_extend_no_holes(gqinode, | ||
| 455 | gqinode->i_size + (need_alloc << sb->s_blocksize_bits), | ||
| 456 | gqinode->i_size); | ||
| 457 | up_write(&OCFS2_I(gqinode)->ip_alloc_sem); | ||
| 458 | if (err < 0) | ||
| 459 | goto out; | ||
| 429 | } | 460 | } |
| 461 | |||
| 462 | handle = ocfs2_start_trans(osb, | ||
| 463 | ocfs2_calc_global_qinit_credits(sb, type)); | ||
| 464 | if (IS_ERR(handle)) { | ||
| 465 | err = PTR_ERR(handle); | ||
| 466 | goto out; | ||
| 467 | } | ||
| 468 | err = ocfs2_qinfo_lock(info, ex); | ||
| 469 | if (err < 0) | ||
| 470 | goto out_trans; | ||
| 430 | err = qtree_write_dquot(&info->dqi_gi, dquot); | 471 | err = qtree_write_dquot(&info->dqi_gi, dquot); |
| 431 | if (ex && info_dirty(sb_dqinfo(dquot->dq_sb, dquot->dq_type))) { | 472 | if (ex && info_dirty(sb_dqinfo(dquot->dq_sb, dquot->dq_type))) { |
| 432 | err2 = __ocfs2_global_write_info(dquot->dq_sb, dquot->dq_type); | 473 | err2 = __ocfs2_global_write_info(dquot->dq_sb, dquot->dq_type); |
| @@ -438,6 +479,9 @@ out_qlock: | |||
| 438 | ocfs2_qinfo_unlock(info, 1); | 479 | ocfs2_qinfo_unlock(info, 1); |
| 439 | else | 480 | else |
| 440 | ocfs2_qinfo_unlock(info, 0); | 481 | ocfs2_qinfo_unlock(info, 0); |
| 482 | out_trans: | ||
| 483 | if (handle) | ||
| 484 | ocfs2_commit_trans(osb, handle); | ||
| 441 | out: | 485 | out: |
| 442 | if (err < 0) | 486 | if (err < 0) |
| 443 | mlog_errno(err); | 487 | mlog_errno(err); |
| @@ -607,7 +651,7 @@ static void qsync_work_fn(struct work_struct *work) | |||
| 607 | 651 | ||
| 608 | dquot_scan_active(sb, ocfs2_sync_dquot_helper, oinfo->dqi_type); | 652 | dquot_scan_active(sb, ocfs2_sync_dquot_helper, oinfo->dqi_type); |
| 609 | queue_delayed_work(ocfs2_quota_wq, &oinfo->dqi_sync_work, | 653 | queue_delayed_work(ocfs2_quota_wq, &oinfo->dqi_sync_work, |
| 610 | oinfo->dqi_syncjiff); | 654 | msecs_to_jiffies(oinfo->dqi_syncms)); |
| 611 | } | 655 | } |
| 612 | 656 | ||
| 613 | /* | 657 | /* |
| @@ -635,20 +679,18 @@ out: | |||
| 635 | return status; | 679 | return status; |
| 636 | } | 680 | } |
| 637 | 681 | ||
| 638 | int ocfs2_calc_qdel_credits(struct super_block *sb, int type) | 682 | static int ocfs2_calc_qdel_credits(struct super_block *sb, int type) |
| 639 | { | 683 | { |
| 640 | struct ocfs2_mem_dqinfo *oinfo; | 684 | struct ocfs2_mem_dqinfo *oinfo = sb_dqinfo(sb, type)->dqi_priv; |
| 641 | int features[MAXQUOTAS] = { OCFS2_FEATURE_RO_COMPAT_USRQUOTA, | 685 | /* |
| 642 | OCFS2_FEATURE_RO_COMPAT_GRPQUOTA }; | 686 | * We modify tree, leaf block, global info, local chunk header, |
| 643 | 687 | * global and local inode; OCFS2_QINFO_WRITE_CREDITS already | |
| 644 | if (!OCFS2_HAS_RO_COMPAT_FEATURE(sb, features[type])) | 688 | * accounts for inode update |
| 645 | return 0; | 689 | */ |
| 646 | 690 | return (oinfo->dqi_gi.dqi_qtree_depth + 2) * | |
| 647 | oinfo = sb_dqinfo(sb, type)->dqi_priv; | 691 | OCFS2_QUOTA_BLOCK_UPDATE_CREDITS + |
| 648 | /* We modify tree, leaf block, global info, local chunk header, | 692 | OCFS2_QINFO_WRITE_CREDITS + |
| 649 | * global and local inode */ | 693 | OCFS2_INODE_UPDATE_CREDITS; |
| 650 | return oinfo->dqi_gi.dqi_qtree_depth + 2 + 1 + | ||
| 651 | 2 * OCFS2_INODE_UPDATE_CREDITS; | ||
| 652 | } | 694 | } |
| 653 | 695 | ||
| 654 | static int ocfs2_release_dquot(struct dquot *dquot) | 696 | static int ocfs2_release_dquot(struct dquot *dquot) |
| @@ -680,33 +722,10 @@ out: | |||
| 680 | return status; | 722 | return status; |
| 681 | } | 723 | } |
| 682 | 724 | ||
| 683 | int ocfs2_calc_qinit_credits(struct super_block *sb, int type) | ||
| 684 | { | ||
| 685 | struct ocfs2_mem_dqinfo *oinfo; | ||
| 686 | int features[MAXQUOTAS] = { OCFS2_FEATURE_RO_COMPAT_USRQUOTA, | ||
| 687 | OCFS2_FEATURE_RO_COMPAT_GRPQUOTA }; | ||
| 688 | struct ocfs2_dinode *lfe, *gfe; | ||
| 689 | |||
| 690 | if (!OCFS2_HAS_RO_COMPAT_FEATURE(sb, features[type])) | ||
| 691 | return 0; | ||
| 692 | |||
| 693 | oinfo = sb_dqinfo(sb, type)->dqi_priv; | ||
| 694 | gfe = (struct ocfs2_dinode *)oinfo->dqi_gqi_bh->b_data; | ||
| 695 | lfe = (struct ocfs2_dinode *)oinfo->dqi_lqi_bh->b_data; | ||
| 696 | /* We can extend local file + global file. In local file we | ||
| 697 | * can modify info, chunk header block and dquot block. In | ||
| 698 | * global file we can modify info, tree and leaf block */ | ||
| 699 | return ocfs2_calc_extend_credits(sb, &lfe->id2.i_list, 0) + | ||
| 700 | ocfs2_calc_extend_credits(sb, &gfe->id2.i_list, 0) + | ||
| 701 | 3 + oinfo->dqi_gi.dqi_qtree_depth + 2; | ||
| 702 | } | ||
| 703 | |||
| 704 | static int ocfs2_acquire_dquot(struct dquot *dquot) | 725 | static int ocfs2_acquire_dquot(struct dquot *dquot) |
| 705 | { | 726 | { |
| 706 | handle_t *handle; | ||
| 707 | struct ocfs2_mem_dqinfo *oinfo = | 727 | struct ocfs2_mem_dqinfo *oinfo = |
| 708 | sb_dqinfo(dquot->dq_sb, dquot->dq_type)->dqi_priv; | 728 | sb_dqinfo(dquot->dq_sb, dquot->dq_type)->dqi_priv; |
| 709 | struct ocfs2_super *osb = OCFS2_SB(dquot->dq_sb); | ||
| 710 | int status = 0; | 729 | int status = 0; |
| 711 | 730 | ||
| 712 | mlog_entry("id=%u, type=%d", dquot->dq_id, dquot->dq_type); | 731 | mlog_entry("id=%u, type=%d", dquot->dq_id, dquot->dq_type); |
| @@ -715,16 +734,7 @@ static int ocfs2_acquire_dquot(struct dquot *dquot) | |||
| 715 | status = ocfs2_lock_global_qf(oinfo, 1); | 734 | status = ocfs2_lock_global_qf(oinfo, 1); |
| 716 | if (status < 0) | 735 | if (status < 0) |
| 717 | goto out; | 736 | goto out; |
| 718 | handle = ocfs2_start_trans(osb, | ||
| 719 | ocfs2_calc_qinit_credits(dquot->dq_sb, dquot->dq_type)); | ||
| 720 | if (IS_ERR(handle)) { | ||
| 721 | status = PTR_ERR(handle); | ||
| 722 | mlog_errno(status); | ||
| 723 | goto out_ilock; | ||
| 724 | } | ||
| 725 | status = dquot_acquire(dquot); | 737 | status = dquot_acquire(dquot); |
| 726 | ocfs2_commit_trans(osb, handle); | ||
| 727 | out_ilock: | ||
| 728 | ocfs2_unlock_global_qf(oinfo, 1); | 738 | ocfs2_unlock_global_qf(oinfo, 1); |
| 729 | out: | 739 | out: |
| 730 | mlog_exit(status); | 740 | mlog_exit(status); |
