diff options
Diffstat (limited to 'fs/xfs/xfs_iget.c')
| -rw-r--r-- | fs/xfs/xfs_iget.c | 140 |
1 files changed, 76 insertions, 64 deletions
diff --git a/fs/xfs/xfs_iget.c b/fs/xfs/xfs_iget.c index e657c5128460..b07604b94d9f 100644 --- a/fs/xfs/xfs_iget.c +++ b/fs/xfs/xfs_iget.c | |||
| @@ -593,8 +593,9 @@ xfs_iunlock_map_shared( | |||
| 593 | * XFS_IOLOCK_EXCL | XFS_ILOCK_EXCL | 593 | * XFS_IOLOCK_EXCL | XFS_ILOCK_EXCL |
| 594 | */ | 594 | */ |
| 595 | void | 595 | void |
| 596 | xfs_ilock(xfs_inode_t *ip, | 596 | xfs_ilock( |
| 597 | uint lock_flags) | 597 | xfs_inode_t *ip, |
| 598 | uint lock_flags) | ||
| 598 | { | 599 | { |
| 599 | /* | 600 | /* |
| 600 | * You can't set both SHARED and EXCL for the same lock, | 601 | * You can't set both SHARED and EXCL for the same lock, |
| @@ -607,16 +608,16 @@ xfs_ilock(xfs_inode_t *ip, | |||
| 607 | (XFS_ILOCK_SHARED | XFS_ILOCK_EXCL)); | 608 | (XFS_ILOCK_SHARED | XFS_ILOCK_EXCL)); |
| 608 | ASSERT((lock_flags & ~(XFS_LOCK_MASK | XFS_LOCK_DEP_MASK)) == 0); | 609 | ASSERT((lock_flags & ~(XFS_LOCK_MASK | XFS_LOCK_DEP_MASK)) == 0); |
| 609 | 610 | ||
| 610 | if (lock_flags & XFS_IOLOCK_EXCL) { | 611 | if (lock_flags & XFS_IOLOCK_EXCL) |
| 611 | mrupdate_nested(&ip->i_iolock, XFS_IOLOCK_DEP(lock_flags)); | 612 | mrupdate_nested(&ip->i_iolock, XFS_IOLOCK_DEP(lock_flags)); |
| 612 | } else if (lock_flags & XFS_IOLOCK_SHARED) { | 613 | else if (lock_flags & XFS_IOLOCK_SHARED) |
| 613 | mraccess_nested(&ip->i_iolock, XFS_IOLOCK_DEP(lock_flags)); | 614 | mraccess_nested(&ip->i_iolock, XFS_IOLOCK_DEP(lock_flags)); |
| 614 | } | 615 | |
| 615 | if (lock_flags & XFS_ILOCK_EXCL) { | 616 | if (lock_flags & XFS_ILOCK_EXCL) |
| 616 | mrupdate_nested(&ip->i_lock, XFS_ILOCK_DEP(lock_flags)); | 617 | mrupdate_nested(&ip->i_lock, XFS_ILOCK_DEP(lock_flags)); |
| 617 | } else if (lock_flags & XFS_ILOCK_SHARED) { | 618 | else if (lock_flags & XFS_ILOCK_SHARED) |
| 618 | mraccess_nested(&ip->i_lock, XFS_ILOCK_DEP(lock_flags)); | 619 | mraccess_nested(&ip->i_lock, XFS_ILOCK_DEP(lock_flags)); |
| 619 | } | 620 | |
| 620 | xfs_ilock_trace(ip, 1, lock_flags, (inst_t *)__return_address); | 621 | xfs_ilock_trace(ip, 1, lock_flags, (inst_t *)__return_address); |
| 621 | } | 622 | } |
| 622 | 623 | ||
| @@ -631,15 +632,12 @@ xfs_ilock(xfs_inode_t *ip, | |||
| 631 | * lock_flags -- this parameter indicates the inode's locks to be | 632 | * lock_flags -- this parameter indicates the inode's locks to be |
| 632 | * to be locked. See the comment for xfs_ilock() for a list | 633 | * to be locked. See the comment for xfs_ilock() for a list |
| 633 | * of valid values. | 634 | * of valid values. |
| 634 | * | ||
| 635 | */ | 635 | */ |
| 636 | int | 636 | int |
| 637 | xfs_ilock_nowait(xfs_inode_t *ip, | 637 | xfs_ilock_nowait( |
| 638 | uint lock_flags) | 638 | xfs_inode_t *ip, |
| 639 | uint lock_flags) | ||
| 639 | { | 640 | { |
| 640 | int iolocked; | ||
| 641 | int ilocked; | ||
| 642 | |||
| 643 | /* | 641 | /* |
| 644 | * You can't set both SHARED and EXCL for the same lock, | 642 | * You can't set both SHARED and EXCL for the same lock, |
| 645 | * and only XFS_IOLOCK_SHARED, XFS_IOLOCK_EXCL, XFS_ILOCK_SHARED, | 643 | * and only XFS_IOLOCK_SHARED, XFS_IOLOCK_EXCL, XFS_ILOCK_SHARED, |
| @@ -651,37 +649,30 @@ xfs_ilock_nowait(xfs_inode_t *ip, | |||
| 651 | (XFS_ILOCK_SHARED | XFS_ILOCK_EXCL)); | 649 | (XFS_ILOCK_SHARED | XFS_ILOCK_EXCL)); |
| 652 | ASSERT((lock_flags & ~(XFS_LOCK_MASK | XFS_LOCK_DEP_MASK)) == 0); | 650 | ASSERT((lock_flags & ~(XFS_LOCK_MASK | XFS_LOCK_DEP_MASK)) == 0); |
| 653 | 651 | ||
| 654 | iolocked = 0; | ||
| 655 | if (lock_flags & XFS_IOLOCK_EXCL) { | 652 | if (lock_flags & XFS_IOLOCK_EXCL) { |
| 656 | iolocked = mrtryupdate(&ip->i_iolock); | 653 | if (!mrtryupdate(&ip->i_iolock)) |
| 657 | if (!iolocked) { | 654 | goto out; |
| 658 | return 0; | ||
| 659 | } | ||
| 660 | } else if (lock_flags & XFS_IOLOCK_SHARED) { | 655 | } else if (lock_flags & XFS_IOLOCK_SHARED) { |
| 661 | iolocked = mrtryaccess(&ip->i_iolock); | 656 | if (!mrtryaccess(&ip->i_iolock)) |
| 662 | if (!iolocked) { | 657 | goto out; |
| 663 | return 0; | ||
| 664 | } | ||
| 665 | } | 658 | } |
| 666 | if (lock_flags & XFS_ILOCK_EXCL) { | 659 | if (lock_flags & XFS_ILOCK_EXCL) { |
| 667 | ilocked = mrtryupdate(&ip->i_lock); | 660 | if (!mrtryupdate(&ip->i_lock)) |
| 668 | if (!ilocked) { | 661 | goto out_undo_iolock; |
| 669 | if (iolocked) { | ||
| 670 | mrunlock(&ip->i_iolock); | ||
| 671 | } | ||
| 672 | return 0; | ||
| 673 | } | ||
| 674 | } else if (lock_flags & XFS_ILOCK_SHARED) { | 662 | } else if (lock_flags & XFS_ILOCK_SHARED) { |
| 675 | ilocked = mrtryaccess(&ip->i_lock); | 663 | if (!mrtryaccess(&ip->i_lock)) |
| 676 | if (!ilocked) { | 664 | goto out_undo_iolock; |
| 677 | if (iolocked) { | ||
| 678 | mrunlock(&ip->i_iolock); | ||
| 679 | } | ||
| 680 | return 0; | ||
| 681 | } | ||
| 682 | } | 665 | } |
| 683 | xfs_ilock_trace(ip, 2, lock_flags, (inst_t *)__return_address); | 666 | xfs_ilock_trace(ip, 2, lock_flags, (inst_t *)__return_address); |
| 684 | return 1; | 667 | return 1; |
| 668 | |||
| 669 | out_undo_iolock: | ||
| 670 | if (lock_flags & XFS_IOLOCK_EXCL) | ||
| 671 | mrunlock_excl(&ip->i_iolock); | ||
| 672 | else if (lock_flags & XFS_IOLOCK_SHARED) | ||
| 673 | mrunlock_shared(&ip->i_iolock); | ||
| 674 | out: | ||
| 675 | return 0; | ||
| 685 | } | 676 | } |
| 686 | 677 | ||
| 687 | /* | 678 | /* |
| @@ -697,8 +688,9 @@ xfs_ilock_nowait(xfs_inode_t *ip, | |||
| 697 | * | 688 | * |
| 698 | */ | 689 | */ |
| 699 | void | 690 | void |
| 700 | xfs_iunlock(xfs_inode_t *ip, | 691 | xfs_iunlock( |
| 701 | uint lock_flags) | 692 | xfs_inode_t *ip, |
| 693 | uint lock_flags) | ||
| 702 | { | 694 | { |
| 703 | /* | 695 | /* |
| 704 | * You can't set both SHARED and EXCL for the same lock, | 696 | * You can't set both SHARED and EXCL for the same lock, |
| @@ -713,31 +705,25 @@ xfs_iunlock(xfs_inode_t *ip, | |||
| 713 | XFS_LOCK_DEP_MASK)) == 0); | 705 | XFS_LOCK_DEP_MASK)) == 0); |
| 714 | ASSERT(lock_flags != 0); | 706 | ASSERT(lock_flags != 0); |
| 715 | 707 | ||
| 716 | if (lock_flags & (XFS_IOLOCK_SHARED | XFS_IOLOCK_EXCL)) { | 708 | if (lock_flags & XFS_IOLOCK_EXCL) |
| 717 | ASSERT(!(lock_flags & XFS_IOLOCK_SHARED) || | 709 | mrunlock_excl(&ip->i_iolock); |
| 718 | (ismrlocked(&ip->i_iolock, MR_ACCESS))); | 710 | else if (lock_flags & XFS_IOLOCK_SHARED) |
| 719 | ASSERT(!(lock_flags & XFS_IOLOCK_EXCL) || | 711 | mrunlock_shared(&ip->i_iolock); |
| 720 | (ismrlocked(&ip->i_iolock, MR_UPDATE))); | ||
| 721 | mrunlock(&ip->i_iolock); | ||
| 722 | } | ||
| 723 | 712 | ||
| 724 | if (lock_flags & (XFS_ILOCK_SHARED | XFS_ILOCK_EXCL)) { | 713 | if (lock_flags & XFS_ILOCK_EXCL) |
| 725 | ASSERT(!(lock_flags & XFS_ILOCK_SHARED) || | 714 | mrunlock_excl(&ip->i_lock); |
| 726 | (ismrlocked(&ip->i_lock, MR_ACCESS))); | 715 | else if (lock_flags & XFS_ILOCK_SHARED) |
| 727 | ASSERT(!(lock_flags & XFS_ILOCK_EXCL) || | 716 | mrunlock_shared(&ip->i_lock); |
| 728 | (ismrlocked(&ip->i_lock, MR_UPDATE))); | ||
| 729 | mrunlock(&ip->i_lock); | ||
| 730 | 717 | ||
| 718 | if ((lock_flags & (XFS_ILOCK_SHARED | XFS_ILOCK_EXCL)) && | ||
| 719 | !(lock_flags & XFS_IUNLOCK_NONOTIFY) && ip->i_itemp) { | ||
| 731 | /* | 720 | /* |
| 732 | * Let the AIL know that this item has been unlocked in case | 721 | * Let the AIL know that this item has been unlocked in case |
| 733 | * it is in the AIL and anyone is waiting on it. Don't do | 722 | * it is in the AIL and anyone is waiting on it. Don't do |
| 734 | * this if the caller has asked us not to. | 723 | * this if the caller has asked us not to. |
| 735 | */ | 724 | */ |
| 736 | if (!(lock_flags & XFS_IUNLOCK_NONOTIFY) && | 725 | xfs_trans_unlocked_item(ip->i_mount, |
| 737 | ip->i_itemp != NULL) { | 726 | (xfs_log_item_t*)(ip->i_itemp)); |
| 738 | xfs_trans_unlocked_item(ip->i_mount, | ||
| 739 | (xfs_log_item_t*)(ip->i_itemp)); | ||
| 740 | } | ||
| 741 | } | 727 | } |
| 742 | xfs_ilock_trace(ip, 3, lock_flags, (inst_t *)__return_address); | 728 | xfs_ilock_trace(ip, 3, lock_flags, (inst_t *)__return_address); |
| 743 | } | 729 | } |
| @@ -747,21 +733,47 @@ xfs_iunlock(xfs_inode_t *ip, | |||
| 747 | * if it is being demoted. | 733 | * if it is being demoted. |
| 748 | */ | 734 | */ |
| 749 | void | 735 | void |
| 750 | xfs_ilock_demote(xfs_inode_t *ip, | 736 | xfs_ilock_demote( |
| 751 | uint lock_flags) | 737 | xfs_inode_t *ip, |
| 738 | uint lock_flags) | ||
| 752 | { | 739 | { |
| 753 | ASSERT(lock_flags & (XFS_IOLOCK_EXCL|XFS_ILOCK_EXCL)); | 740 | ASSERT(lock_flags & (XFS_IOLOCK_EXCL|XFS_ILOCK_EXCL)); |
| 754 | ASSERT((lock_flags & ~(XFS_IOLOCK_EXCL|XFS_ILOCK_EXCL)) == 0); | 741 | ASSERT((lock_flags & ~(XFS_IOLOCK_EXCL|XFS_ILOCK_EXCL)) == 0); |
| 755 | 742 | ||
| 756 | if (lock_flags & XFS_ILOCK_EXCL) { | 743 | if (lock_flags & XFS_ILOCK_EXCL) |
| 757 | ASSERT(ismrlocked(&ip->i_lock, MR_UPDATE)); | ||
| 758 | mrdemote(&ip->i_lock); | 744 | mrdemote(&ip->i_lock); |
| 759 | } | 745 | if (lock_flags & XFS_IOLOCK_EXCL) |
| 760 | if (lock_flags & XFS_IOLOCK_EXCL) { | ||
| 761 | ASSERT(ismrlocked(&ip->i_iolock, MR_UPDATE)); | ||
| 762 | mrdemote(&ip->i_iolock); | 746 | mrdemote(&ip->i_iolock); |
| 747 | } | ||
| 748 | |||
| 749 | #ifdef DEBUG | ||
| 750 | /* | ||
| 751 | * Debug-only routine, without additional rw_semaphore APIs, we can | ||
| 752 | * now only answer requests regarding whether we hold the lock for write | ||
| 753 | * (reader state is outside our visibility, we only track writer state). | ||
| 754 | * | ||
| 755 | * Note: this means !xfs_isilocked would give false positives, so don't do that. | ||
| 756 | */ | ||
| 757 | int | ||
| 758 | xfs_isilocked( | ||
| 759 | xfs_inode_t *ip, | ||
| 760 | uint lock_flags) | ||
| 761 | { | ||
| 762 | if ((lock_flags & (XFS_ILOCK_EXCL|XFS_ILOCK_SHARED)) == | ||
| 763 | XFS_ILOCK_EXCL) { | ||
| 764 | if (!ip->i_lock.mr_writer) | ||
| 765 | return 0; | ||
| 763 | } | 766 | } |
| 767 | |||
| 768 | if ((lock_flags & (XFS_IOLOCK_EXCL|XFS_IOLOCK_SHARED)) == | ||
| 769 | XFS_IOLOCK_EXCL) { | ||
| 770 | if (!ip->i_iolock.mr_writer) | ||
| 771 | return 0; | ||
| 772 | } | ||
| 773 | |||
| 774 | return 1; | ||
| 764 | } | 775 | } |
| 776 | #endif | ||
| 765 | 777 | ||
| 766 | /* | 778 | /* |
| 767 | * The following three routines simply manage the i_flock | 779 | * The following three routines simply manage the i_flock |
