aboutsummaryrefslogtreecommitdiffstats
path: root/fs/xfs/xfs_qm.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/xfs/xfs_qm.c')
-rw-r--r--fs/xfs/xfs_qm.c64
1 files changed, 16 insertions, 48 deletions
diff --git a/fs/xfs/xfs_qm.c b/fs/xfs/xfs_qm.c
index 22360bb26af9..3c553454443c 100644
--- a/fs/xfs/xfs_qm.c
+++ b/fs/xfs/xfs_qm.c
@@ -650,11 +650,7 @@ xfs_qm_dqattach_one(
650 650
651/* 651/*
652 * Given a udquot and gdquot, attach a ptr to the group dquot in the 652 * Given a udquot and gdquot, attach a ptr to the group dquot in the
653 * udquot as a hint for future lookups. The idea sounds simple, but the 653 * udquot as a hint for future lookups.
654 * execution isn't, because the udquot might have a group dquot attached
655 * already and getting rid of that gets us into lock ordering constraints.
656 * The process is complicated more by the fact that the dquots may or may not
657 * be locked on entry.
658 */ 654 */
659STATIC void 655STATIC void
660xfs_qm_dqattach_grouphint( 656xfs_qm_dqattach_grouphint(
@@ -665,45 +661,21 @@ xfs_qm_dqattach_grouphint(
665 661
666 xfs_dqlock(udq); 662 xfs_dqlock(udq);
667 663
668 if ((tmp = udq->q_gdquot)) { 664 tmp = udq->q_gdquot;
669 if (tmp == gdq) { 665 if (tmp) {
670 xfs_dqunlock(udq); 666 if (tmp == gdq)
671 return; 667 goto done;
672 }
673 668
674 udq->q_gdquot = NULL; 669 udq->q_gdquot = NULL;
675 /*
676 * We can't keep any dqlocks when calling dqrele,
677 * because the freelist lock comes before dqlocks.
678 */
679 xfs_dqunlock(udq);
680 /*
681 * we took a hard reference once upon a time in dqget,
682 * so give it back when the udquot no longer points at it
683 * dqput() does the unlocking of the dquot.
684 */
685 xfs_qm_dqrele(tmp); 670 xfs_qm_dqrele(tmp);
686
687 xfs_dqlock(udq);
688 xfs_dqlock(gdq);
689
690 } else {
691 ASSERT(XFS_DQ_IS_LOCKED(udq));
692 xfs_dqlock(gdq);
693 }
694
695 ASSERT(XFS_DQ_IS_LOCKED(udq));
696 ASSERT(XFS_DQ_IS_LOCKED(gdq));
697 /*
698 * Somebody could have attached a gdquot here,
699 * when we dropped the uqlock. If so, just do nothing.
700 */
701 if (udq->q_gdquot == NULL) {
702 XFS_DQHOLD(gdq);
703 udq->q_gdquot = gdq;
704 } 671 }
705 672
673 xfs_dqlock(gdq);
674 XFS_DQHOLD(gdq);
706 xfs_dqunlock(gdq); 675 xfs_dqunlock(gdq);
676
677 udq->q_gdquot = gdq;
678done:
707 xfs_dqunlock(udq); 679 xfs_dqunlock(udq);
708} 680}
709 681
@@ -770,17 +742,13 @@ xfs_qm_dqattach_locked(
770 ASSERT(ip->i_gdquot); 742 ASSERT(ip->i_gdquot);
771 743
772 /* 744 /*
773 * We may or may not have the i_udquot locked at this point, 745 * We do not have i_udquot locked at this point, but this check
774 * but this check is OK since we don't depend on the i_gdquot to 746 * is OK since we don't depend on the i_gdquot to be accurate
775 * be accurate 100% all the time. It is just a hint, and this 747 * 100% all the time. It is just a hint, and this will
776 * will succeed in general. 748 * succeed in general.
777 */
778 if (ip->i_udquot->q_gdquot == ip->i_gdquot)
779 goto done;
780 /*
781 * Attach i_gdquot to the gdquot hint inside the i_udquot.
782 */ 749 */
783 xfs_qm_dqattach_grouphint(ip->i_udquot, ip->i_gdquot); 750 if (ip->i_udquot->q_gdquot != ip->i_gdquot)
751 xfs_qm_dqattach_grouphint(ip->i_udquot, ip->i_gdquot);
784 } 752 }
785 753
786 done: 754 done: