{
        struct file_lock *fl;
        int status = false;
-       struct nfsd_file *nf = find_any_file(fp);
+       struct nfsd_file *nf;
        struct inode *inode;
        struct file_lock_context *flctx;
 
+       spin_lock(&fp->fi_lock);
+       nf = find_any_file_locked(fp);
        if (!nf) {
                /* Any valid lock stateid should have some sort of access */
                WARN_ON_ONCE(1);
-               return status;
+               goto out;
        }
 
        inode = file_inode(nf->nf_file);
                }
                spin_unlock(&flctx->flc_lock);
        }
-       nfsd_file_put(nf);
+out:
+       spin_unlock(&fp->fi_lock);
        return status;
 }
 
  * @cstate: NFSv4 COMPOUND state
  * @u: RELEASE_LOCKOWNER arguments
  *
- * The lockowner's so_count is bumped when a lock record is added
- * or when copying a conflicting lock. The latter case is brief,
- * but can lead to fleeting false positives when looking for
- * locks-in-use.
+ * Check if theree are any locks still held and if not - free the lockowner
+ * and any lock state that is owned.
  *
  * Return values:
  *   %nfs_ok: lockowner released or not found
                spin_unlock(&clp->cl_lock);
                return nfs_ok;
        }
-       if (atomic_read(&lo->lo_owner.so_count) != 2) {
-               spin_unlock(&clp->cl_lock);
-               nfs4_put_stateowner(&lo->lo_owner);
-               return nfserr_locks_held;
+
+       list_for_each_entry(stp, &lo->lo_owner.so_stateids, st_perstateowner) {
+               if (check_for_locks(stp->st_stid.sc_file, lo)) {
+                       spin_unlock(&clp->cl_lock);
+                       nfs4_put_stateowner(&lo->lo_owner);
+                       return nfserr_locks_held;
+               }
        }
        unhash_lockowner_locked(lo);
        while (!list_empty(&lo->lo_owner.so_stateids)) {