}
                                nfs4_put_stid(stid);
                                spin_lock(&nn->client_lock);
+                               if (clp->cl_minorversion == 0)
+                                       /* Allow cleanup after a lease period.
+                                        * store_release ensures cleanup will
+                                        * see any newly revoked states if it
+                                        * sees the time updated.
+                                        */
+                                       nn->nfs40_last_revoke =
+                                               ktime_get_boottime_seconds();
                                goto retry;
                        }
                }
        return ret;
 }
 
+static void nfsd4_drop_revoked_stid(struct nfs4_stid *s)
+       __releases(&s->sc_client->cl_lock)
+{
+       struct nfs4_client *cl = s->sc_client;
+
+       switch (s->sc_type) {
+       default:
+               spin_unlock(&cl->cl_lock);
+       }
+}
+
+static void nfsd40_drop_revoked_stid(struct nfs4_client *cl,
+                                   stateid_t *stid)
+{
+       /* NFSv4.0 has no way for the client to tell the server
+        * that it can forget an admin-revoked stateid.
+        * So we keep it around until the first time that the
+        * client uses it, and drop it the first time
+        * nfserr_admin_revoked is returned.
+        * For v4.1 and later we wait until explicitly told
+        * to free the stateid.
+        */
+       if (cl->cl_minorversion == 0) {
+               struct nfs4_stid *st;
+
+               spin_lock(&cl->cl_lock);
+               st = find_stateid_locked(cl, stid);
+               if (st)
+                       nfsd4_drop_revoked_stid(st);
+               else
+                       spin_unlock(&cl->cl_lock);
+       }
+}
+
 static __be32
 nfsd4_verify_open_stid(struct nfs4_stid *s)
 {
 
        mutex_lock_nested(&stp->st_mutex, LOCK_STATEID_MUTEX);
        ret = nfsd4_verify_open_stid(&stp->st_stid);
+       if (ret == nfserr_admin_revoked)
+               nfsd40_drop_revoked_stid(stp->st_stid.sc_client,
+                                       &stp->st_stid.sc_stateid);
+
        if (ret != nfs_ok)
                mutex_unlock(&stp->st_mutex);
        return ret;
        }
        if (deleg->dl_stid.sc_status & SC_STATUS_REVOKED) {
                nfs4_put_stid(&deleg->dl_stid);
+               nfsd40_drop_revoked_stid(cl, &open->op_delegate_stateid);
                status = nfserr_deleg_revoked;
                goto out;
        }
        }
 }
 
+static void nfs40_clean_admin_revoked(struct nfsd_net *nn,
+                                     struct laundry_time *lt)
+{
+       struct nfs4_client *clp;
+
+       spin_lock(&nn->client_lock);
+       if (nn->nfs40_last_revoke == 0 ||
+           nn->nfs40_last_revoke > lt->cutoff) {
+               spin_unlock(&nn->client_lock);
+               return;
+       }
+       nn->nfs40_last_revoke = 0;
+
+retry:
+       list_for_each_entry(clp, &nn->client_lru, cl_lru) {
+               unsigned long id, tmp;
+               struct nfs4_stid *stid;
+
+               if (atomic_read(&clp->cl_admin_revoked) == 0)
+                       continue;
+
+               spin_lock(&clp->cl_lock);
+               idr_for_each_entry_ul(&clp->cl_stateids, stid, tmp, id)
+                       if (stid->sc_status & SC_STATUS_ADMIN_REVOKED) {
+                               refcount_inc(&stid->sc_count);
+                               spin_unlock(&nn->client_lock);
+                               /* this function drops ->cl_lock */
+                               nfsd4_drop_revoked_stid(stid);
+                               nfs4_put_stid(stid);
+                               spin_lock(&nn->client_lock);
+                               goto retry;
+                       }
+               spin_unlock(&clp->cl_lock);
+       }
+       spin_unlock(&nn->client_lock);
+}
+
 static time64_t
 nfs4_laundromat(struct nfsd_net *nn)
 {
        nfs4_get_client_reaplist(nn, &reaplist, <);
        nfs4_process_client_reaplist(&reaplist);
 
+       nfs40_clean_admin_revoked(nn, <);
+
        spin_lock(&state_lock);
        list_for_each_safe(pos, next, &nn->del_recall_lru) {
                dp = list_entry (pos, struct nfs4_delegation, dl_recall_lru);
        if (ret == nfs_ok)
                ret = check_stateid_generation(in, &s->sc_stateid, has_session);
        spin_unlock(&s->sc_lock);
+       if (ret == nfserr_admin_revoked)
+               nfsd40_drop_revoked_stid(s->sc_client,
+                                       &s->sc_stateid);
        return ret;
 }
 
        }
 out_unlock:
        spin_unlock(&cl->cl_lock);
+       if (status == nfserr_admin_revoked)
+               nfsd40_drop_revoked_stid(cl, stateid);
        return status;
 }
 
                return nfserr_deleg_revoked;
        }
        if (stid->sc_status & SC_STATUS_ADMIN_REVOKED) {
+               nfsd40_drop_revoked_stid(cstate->clp, stateid);
                nfs4_put_stid(stid);
                return nfserr_admin_revoked;
        }
        s = find_stateid_locked(cl, stateid);
        if (!s || s->sc_status & SC_STATUS_CLOSED)
                goto out_unlock;
+       if (s->sc_status & SC_STATUS_ADMIN_REVOKED) {
+               nfsd4_drop_revoked_stid(s);
+               ret = nfs_ok;
+               goto out;
+       }
        spin_lock(&s->sc_lock);
        switch (s->sc_type) {
        case SC_TYPE_DELEG:
                spin_unlock(&cl->cl_lock);
                ret = nfsd4_free_lock_stateid(stateid, s);
                goto out;
-       /* Default falls through and returns nfserr_bad_stateid */
        }
        spin_unlock(&s->sc_lock);
 out_unlock: