staging/lustre: avoid clearing i_nlink for inodes in use
authorAndrew Perepechko <andrew.perepechko@seagate.com>
Wed, 24 Aug 2016 15:11:53 +0000 (11:11 -0400)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 1 Sep 2016 15:25:46 +0000 (17:25 +0200)
The patch removes find_cbdata callbacks and clear_nlink
from dentry_iput path, since this piece of code makes
a few races possible.

The test case reproduces one of the possible races
described in LU-7925:

1) two hard links are created for the same file
2) the test calls stat(2) for link #1
3) in the middle of 2) the test opens and closes link #2
4) in the middle of 2) the test drops the ldlm locks and
   forces dentry reclaim via vm.drop_caches=2
5) in the middle of 2) ll_d_iput() clears i_nlink for
   the inode
6) the initial stat(2) continues and copies the wrong
   i_nlink value into st_nlink

Signed-off-by: Andrew Perepechko <andrew.perepechko@seagate.com>
Seagate-bug-id: MRP-3271
Reviewed-on: http://review.whamcloud.com/19164
Intel-bug-id: https://jira.hpdd.intel.com/browse/LU-7925
Reviewed-by: Wally Wang <wang@cray.com>
Reviewed-by: Lai Siyao <lai.siyao@intel.com>
Signed-off-by: Oleg Drokin <green@linuxhacker.ru>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/staging/lustre/lustre/include/obd.h
drivers/staging/lustre/lustre/include/obd_class.h
drivers/staging/lustre/lustre/include/obd_support.h
drivers/staging/lustre/lustre/llite/dcache.c
drivers/staging/lustre/lustre/llite/file.c
drivers/staging/lustre/lustre/lmv/lmv_obd.c
drivers/staging/lustre/lustre/lov/lov_obd.c
drivers/staging/lustre/lustre/mdc/mdc_internal.h
drivers/staging/lustre/lustre/mdc/mdc_locks.c
drivers/staging/lustre/lustre/mdc/mdc_request.c
drivers/staging/lustre/lustre/osc/osc_request.c

index ed0fd4176a253b32b79f4b0690d97a7f095fbcca..f3d141b03556e1f9e1a0c5ef57875137d09c0b72 100644 (file)
@@ -896,8 +896,6 @@ struct obd_ops {
                        struct niobuf_remote *remote, int pages,
                        struct niobuf_local *local,
                        struct obd_trans_info *oti, int rc);
-       int (*find_cbdata)(struct obd_export *, struct lov_stripe_md *,
-                          ldlm_iterator_t it, void *data);
        int (*init_export)(struct obd_export *exp);
        int (*destroy_export)(struct obd_export *exp);
 
@@ -958,8 +956,6 @@ struct cl_attr;
 struct md_ops {
        int (*getstatus)(struct obd_export *, struct lu_fid *);
        int (*null_inode)(struct obd_export *, const struct lu_fid *);
-       int (*find_cbdata)(struct obd_export *, const struct lu_fid *,
-                          ldlm_iterator_t, void *);
        int (*close)(struct obd_export *, struct md_op_data *,
                     struct md_open_data *, struct ptlrpc_request **);
        int (*create)(struct obd_export *, struct md_op_data *,
index 4f4896894ff1da3f36bc4941befb9d7655ed0c5b..9702ad49dd4886465e977a472f8a646d026d206e 100644 (file)
@@ -1177,19 +1177,6 @@ static inline int obd_iocontrol(unsigned int cmd, struct obd_export *exp,
        return rc;
 }
 
-static inline int obd_find_cbdata(struct obd_export *exp,
-                                 struct lov_stripe_md *lsm,
-                                 ldlm_iterator_t it, void *data)
-{
-       int rc;
-
-       EXP_CHECK_DT_OP(exp, find_cbdata);
-       EXP_COUNTER_INCREMENT(exp, find_cbdata);
-
-       rc = OBP(exp->exp_obd, find_cbdata)(exp, lsm, it, data);
-       return rc;
-}
-
 static inline void obd_import_event(struct obd_device *obd,
                                    struct obd_import *imp,
                                    enum obd_import_event event)
@@ -1358,18 +1345,6 @@ static inline int md_null_inode(struct obd_export *exp,
        return rc;
 }
 
-static inline int md_find_cbdata(struct obd_export *exp,
-                                const struct lu_fid *fid,
-                                ldlm_iterator_t it, void *data)
-{
-       int rc;
-
-       EXP_CHECK_MD_OP(exp, find_cbdata);
-       EXP_MD_COUNTER_INCREMENT(exp, find_cbdata);
-       rc = MDP(exp->exp_obd, find_cbdata)(exp, fid, it, data);
-       return rc;
-}
-
 static inline int md_close(struct obd_export *exp, struct md_op_data *op_data,
                           struct md_open_data *mod,
                           struct ptlrpc_request **request)
index 4a9fe888b6f9858dc1eb750eefd656098bb5d4eb..4d7a5c8dfe9a7eaf90d414cf23b5ee9f7a2f97d6 100644 (file)
@@ -458,6 +458,7 @@ extern char obd_jobid_var[];
 #define OBD_FAIL_LOV_INIT                          0x1403
 #define OBD_FAIL_GLIMPSE_DELAY                     0x1404
 #define OBD_FAIL_LLITE_XATTR_ENOMEM                0x1405
+#define OBD_FAIL_GETATTR_DELAY                     0x1409
 
 #define OBD_FAIL_FID_INDIR     0x1501
 #define OBD_FAIL_FID_INLMA     0x1502
index 463b1a360733940d646766a82b03ef37056330df..f4b6f38246089174efd16a6254fa6121a9393b1e 100644 (file)
@@ -102,39 +102,6 @@ static int ll_dcompare(const struct dentry *dentry,
        return 0;
 }
 
-static inline int return_if_equal(struct ldlm_lock *lock, void *data)
-{
-       return (ldlm_is_canceling(lock) && ldlm_is_discard_data(lock)) ?
-               LDLM_ITER_CONTINUE : LDLM_ITER_STOP;
-}
-
-/* find any ldlm lock of the inode in mdc and lov
- * return 0    not find
- *     1    find one
- *      < 0    error
- */
-static int find_cbdata(struct inode *inode)
-{
-       struct ll_sb_info *sbi = ll_i2sbi(inode);
-       struct lov_stripe_md *lsm;
-       int rc = 0;
-
-       LASSERT(inode);
-       rc = md_find_cbdata(sbi->ll_md_exp, ll_inode2fid(inode),
-                           return_if_equal, NULL);
-       if (rc != 0)
-               return rc;
-
-       lsm = ccc_inode_lsm_get(inode);
-       if (!lsm)
-               return rc;
-
-       rc = obd_find_cbdata(sbi->ll_dt_exp, lsm, return_if_equal, NULL);
-       ccc_inode_lsm_put(inode, lsm);
-
-       return rc;
-}
-
 /**
  * Called when last reference to a dentry is dropped and dcache wants to know
  * whether or not it should cache it:
@@ -155,19 +122,6 @@ static int ll_ddelete(const struct dentry *de)
        /* kernel >= 2.6.38 last refcount is decreased after this function. */
        LASSERT(d_count(de) == 1);
 
-       /* Disable this piece of code temporarily because this is called
-        * inside dcache_lock so it's not appropriate to do lots of work
-        * here. ATTENTION: Before this piece of code enabling, LU-2487 must be
-        * resolved.
-        */
-#if 0
-       /* if not ldlm lock for this inode, set i_nlink to 0 so that
-        * this inode can be recycled later b=20433
-        */
-       if (d_really_is_positive(de) && !find_cbdata(d_inode(de)))
-               clear_nlink(d_inode(de));
-#endif
-
        if (d_lustre_invalid((struct dentry *)de))
                return 1;
        return 0;
@@ -347,18 +301,9 @@ static int ll_revalidate_nd(struct dentry *dentry, unsigned int flags)
        return ll_revalidate_dentry(dentry, flags);
 }
 
-static void ll_d_iput(struct dentry *de, struct inode *inode)
-{
-       LASSERT(inode);
-       if (!find_cbdata(inode))
-               clear_nlink(inode);
-       iput(inode);
-}
-
 const struct dentry_operations ll_d_ops = {
        .d_revalidate = ll_revalidate_nd,
        .d_release = ll_release,
        .d_delete  = ll_ddelete,
-       .d_iput    = ll_d_iput,
        .d_compare = ll_dcompare,
 };
index 13ff212abe038ef285d59c3f04d9acd3a3bf624b..e2e81bf33e0df70d2a0205c54e2eece8f1aab12e 100644 (file)
@@ -3191,6 +3191,8 @@ int ll_getattr(struct vfsmount *mnt, struct dentry *de, struct kstat *stat)
        if (res)
                return res;
 
+       OBD_FAIL_TIMEOUT(OBD_FAIL_GETATTR_DELAY, 30);
+
        stat->dev = inode->i_sb->s_dev;
        if (ll_need_32bit_api(sbi))
                stat->ino = cl_fid_build_ino(&lli->lli_fid, 1);
index 22b7896d5d76296724ef4d0503cf36ee5cce670b..dc752d528dac0a5c70fe84cb8fb12ab8c59c0388 100644 (file)
@@ -1609,47 +1609,6 @@ static int lmv_null_inode(struct obd_export *exp, const struct lu_fid *fid)
        return 0;
 }
 
-static int lmv_find_cbdata(struct obd_export *exp, const struct lu_fid *fid,
-                          ldlm_iterator_t it, void *data)
-{
-       struct obd_device   *obd = exp->exp_obd;
-       struct lmv_obd      *lmv = &obd->u.lmv;
-       int tgt;
-       u32 i;
-       int               rc;
-
-       rc = lmv_check_connect(obd);
-       if (rc)
-               return rc;
-
-       CDEBUG(D_INODE, "CBDATA for "DFID"\n", PFID(fid));
-
-       /*
-        * With DNE every object can have two locks in different namespaces:
-        * lookup lock in space of MDT storing direntry and update/open lock in
-        * space of MDT storing inode. Try the MDT that the FID maps to first,
-        * since this can be easily found, and only try others if that fails.
-        */
-       for (i = 0, tgt = lmv_find_target_index(lmv, fid);
-            i < lmv->desc.ld_tgt_count;
-            i++, tgt = (tgt + 1) % lmv->desc.ld_tgt_count) {
-               if (tgt < 0) {
-                       CDEBUG(D_HA, "%s: "DFID" is inaccessible: rc = %d\n",
-                              obd->obd_name, PFID(fid), tgt);
-                       tgt = 0;
-               }
-
-               if (!lmv->tgts[tgt] || !lmv->tgts[tgt]->ltd_exp)
-                       continue;
-
-               rc = md_find_cbdata(lmv->tgts[tgt]->ltd_exp, fid, it, data);
-               if (rc)
-                       return rc;
-       }
-
-       return rc;
-}
-
 static int lmv_close(struct obd_export *exp, struct md_op_data *op_data,
                     struct md_open_data *mod, struct ptlrpc_request **request)
 {
@@ -3373,7 +3332,6 @@ static struct obd_ops lmv_obd_ops = {
 static struct md_ops lmv_md_ops = {
        .getstatus              = lmv_getstatus,
        .null_inode             = lmv_null_inode,
-       .find_cbdata            = lmv_find_cbdata,
        .close                  = lmv_close,
        .create                 = lmv_create,
        .done_writing           = lmv_done_writing,
index 2817f38caf5627e2e24020b7e9e2f41fa1eb45d8..265be0893b9b20489500305ca42b8ea3ccdf861a 100644 (file)
@@ -1268,46 +1268,6 @@ static int lov_setattr_async(struct obd_export *exp, struct obd_info *oinfo,
        return 0;
 }
 
-/* find any ldlm lock of the inode in lov
- * return 0    not find
- *     1    find one
- *      < 0    error
- */
-static int lov_find_cbdata(struct obd_export *exp,
-                          struct lov_stripe_md *lsm, ldlm_iterator_t it,
-                          void *data)
-{
-       struct lov_obd *lov;
-       int rc = 0, i;
-
-       ASSERT_LSM_MAGIC(lsm);
-
-       if (!exp || !exp->exp_obd)
-               return -ENODEV;
-
-       lov = &exp->exp_obd->u.lov;
-       for (i = 0; i < lsm->lsm_stripe_count; i++) {
-               struct lov_stripe_md submd;
-               struct lov_oinfo *loi = lsm->lsm_oinfo[i];
-
-               if (lov_oinfo_is_dummy(loi))
-                       continue;
-
-               if (!lov->lov_tgts[loi->loi_ost_idx]) {
-                       CDEBUG(D_HA, "lov idx %d NULL\n", loi->loi_ost_idx);
-                       continue;
-               }
-
-               submd.lsm_oi = loi->loi_oi;
-               submd.lsm_stripe_count = 0;
-               rc = obd_find_cbdata(lov->lov_tgts[loi->loi_ost_idx]->ltd_exp,
-                                    &submd, it, data);
-               if (rc != 0)
-                       return rc;
-       }
-       return rc;
-}
-
 int lov_statfs_interpret(struct ptlrpc_request_set *rqset, void *data, int rc)
 {
        struct lov_request_set *lovset = (struct lov_request_set *)data;
@@ -2326,7 +2286,6 @@ static struct obd_ops lov_obd_ops = {
        .getattr_async  = lov_getattr_async,
        .setattr_async  = lov_setattr_async,
        .adjust_kms     = lov_adjust_kms,
-       .find_cbdata    = lov_find_cbdata,
        .iocontrol      = lov_iocontrol,
        .get_info       = lov_get_info,
        .set_info_async = lov_set_info_async,
index f778a924aa8749c004e24d0d0085017df2621df1..c10441d1eae8c5355117cfa8f2249fb4cca8807f 100644 (file)
@@ -66,9 +66,6 @@ int mdc_set_lock_data(struct obd_export *exp,
 
 int mdc_null_inode(struct obd_export *exp, const struct lu_fid *fid);
 
-int mdc_find_cbdata(struct obd_export *exp, const struct lu_fid *fid,
-                   ldlm_iterator_t it, void *data);
-
 int mdc_intent_lock(struct obd_export *exp,
                    struct md_op_data *op_data,
                    struct lookup_intent *it,
index 95dd291cbc739a8a4b76eb752a390228536590d9..54de46bee885aced446af315203cb9adaadc0c43 100644 (file)
@@ -185,28 +185,6 @@ int mdc_null_inode(struct obd_export *exp,
        return 0;
 }
 
-/* find any ldlm lock of the inode in mdc
- * return 0    not find
- *     1    find one
- *      < 0    error
- */
-int mdc_find_cbdata(struct obd_export *exp,
-                   const struct lu_fid *fid,
-                   ldlm_iterator_t it, void *data)
-{
-       struct ldlm_res_id res_id;
-       int rc = 0;
-
-       fid_build_reg_res_name((struct lu_fid *)fid, &res_id);
-       rc = ldlm_resource_iterate(class_exp2obd(exp)->obd_namespace, &res_id,
-                                  it, data);
-       if (rc == LDLM_ITER_STOP)
-               return 1;
-       else if (rc == LDLM_ITER_CONTINUE)
-               return 0;
-       return rc;
-}
-
 static inline void mdc_clear_replay_flag(struct ptlrpc_request *req, int rc)
 {
        /* Don't hold error requests for replay. */
index 313889a2be0a53f1c8c3f7ef348a38a0416c9a5b..5bf95f964438e1a05a7e0f426bd83589c599ddd8 100644 (file)
@@ -2874,7 +2874,6 @@ static struct obd_ops mdc_obd_ops = {
 static struct md_ops mdc_md_ops = {
        .getstatus              = mdc_getstatus,
        .null_inode             = mdc_null_inode,
-       .find_cbdata            = mdc_find_cbdata,
        .close                  = mdc_close,
        .create                 = mdc_create,
        .done_writing           = mdc_done_writing,
index 53c191d0c00b81d043be8a5d26813878513d61c9..bdb329d72ab8dbf2c2595b2d2d3d532bdb2be6cf 100644 (file)
@@ -2119,27 +2119,6 @@ static int osc_set_data_with_check(struct lustre_handle *lockh,
        return set;
 }
 
-/* find any ldlm lock of the inode in osc
- * return 0    not find
- *     1    find one
- *      < 0    error
- */
-static int osc_find_cbdata(struct obd_export *exp, struct lov_stripe_md *lsm,
-                          ldlm_iterator_t replace, void *data)
-{
-       struct ldlm_res_id res_id;
-       struct obd_device *obd = class_exp2obd(exp);
-       int rc = 0;
-
-       ostid_build_res_name(&lsm->lsm_oi, &res_id);
-       rc = ldlm_resource_iterate(obd->obd_namespace, &res_id, replace, data);
-       if (rc == LDLM_ITER_STOP)
-               return 1;
-       if (rc == LDLM_ITER_CONTINUE)
-               return 0;
-       return rc;
-}
-
 static int osc_enqueue_fini(struct ptlrpc_request *req,
                            osc_enqueue_upcall_f upcall, void *cookie,
                            struct lustre_handle *lockh, enum ldlm_mode mode,
@@ -3358,7 +3337,6 @@ static struct obd_ops osc_obd_ops = {
        .getattr_async  = osc_getattr_async,
        .setattr        = osc_setattr,
        .setattr_async  = osc_setattr_async,
-       .find_cbdata    = osc_find_cbdata,
        .iocontrol      = osc_iocontrol,
        .get_info       = osc_get_info,
        .set_info_async = osc_set_info_async,