Whamcloud - gitweb
LU-12037 mdt: add option for cross-MDT rename
[fs/lustre-release.git] / lustre / mdt / mdt_lproc.c
index d0b37fb..9bc2522 100644 (file)
@@ -23,7 +23,7 @@
  * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
  * Use is subject to license terms.
  *
- * Copyright (c) 2011, 2016, Intel Corporation.
+ * Copyright (c) 2011, 2017, Intel Corporation.
  */
 /*
  * This file is part of Lustre, http://www.lustre.org/
@@ -76,8 +76,6 @@
  *     16kB: { samples: 12, pct: 3, cum_pct: 81 }
  **/
 
-#define pct(a, b) (b ? a * 100 / b : 0)
-
 static void display_rename_stats(struct seq_file *seq, char *name,
                                  struct obd_histogram *hist)
 {
@@ -101,8 +99,8 @@ static void display_rename_stats(struct seq_file *seq, char *name,
                 else
                         seq_printf(seq, "%6s%d%s", " ", 1<<(i-20), "MB:");
 
-                seq_printf(seq, " { sample: %3lu, pct: %3lu, cum_pct: %3lu }\n",
-                           t, pct(t, tot), pct(cum, tot));
+               seq_printf(seq, " { sample: %3lu, pct: %3u, cum_pct: %3u }\n",
+                          t, pct(t, tot), pct(cum, tot));
 
                 if (cum == tot)
                         break;
@@ -128,8 +126,6 @@ static void rename_stats_show(struct seq_file *seq,
                              &rename_stats->hist[RENAME_CROSSDIR_TGT_SIZE]);
 }
 
-#undef pct
-
 static int mdt_rename_stats_seq_show(struct seq_file *seq, void *v)
 {
         struct mdt_device *mdt = seq->private;
@@ -213,7 +209,7 @@ static int mdt_identity_expire_seq_show(struct seq_file *m, void *data)
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
 
-       seq_printf(m, "%u\n", mdt->mdt_identity_cache->uc_entry_expire);
+       seq_printf(m, "%lld\n", mdt->mdt_identity_cache->uc_entry_expire);
        return 0;
 }
 
@@ -224,13 +220,14 @@ mdt_identity_expire_seq_write(struct file *file, const char __user *buffer,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       time64_t val;
        int rc;
-       __s64 val;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtoll_from_user(buffer, count, 0, &val);
        if (rc)
                return rc;
-       if (val < 0 || val > INT_MAX)
+
+       if (val < 0)
                return -ERANGE;
 
        mdt->mdt_identity_cache->uc_entry_expire = val;
@@ -244,7 +241,7 @@ static int mdt_identity_acquire_expire_seq_show(struct seq_file *m, void *data)
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
 
-       seq_printf(m, "%u\n", mdt->mdt_identity_cache->uc_acquire_expire);
+       seq_printf(m, "%lld\n", mdt->mdt_identity_cache->uc_acquire_expire);
        return 0;
 }
 
@@ -256,12 +253,13 @@ mdt_identity_acquire_expire_seq_write(struct file *file,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       time64_t val;
        int rc;
-       __s64 val;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtoll_from_user(buffer, count, 0, &val);
        if (rc)
                return rc;
+
        if (val < 0 || val > INT_MAX)
                return -ERANGE;
 
@@ -336,14 +334,12 @@ lprocfs_identity_flush_seq_write(struct file *file, const char __user *buffer,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       int uid;
        int rc;
-       __s64 uid;
 
-       rc = lprocfs_str_to_s64(buffer, count, &uid);
+       rc = kstrtoint_from_user(buffer, count, 0, &uid);
        if (rc)
                return rc;
-       if (uid < INT_MIN || uid > INT_MAX)
-               return -ERANGE;
 
        mdt_flush_identity(mdt->mdt_identity_cache, uid);
        return count;
@@ -499,13 +495,14 @@ mdt_evict_tgt_nids_seq_write(struct file *file, const char __user *buffer,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
-       __s64 val;
+       bool val;
        int rc;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtobool_from_user(buffer, count, &val);
        if (rc)
                return rc;
-       mdt->mdt_opts.mo_evict_tgt_nids = !!val;
+
+       mdt->mdt_opts.mo_evict_tgt_nids = val;
        return count;
 }
 LPROC_SEQ_FOPS(mdt_evict_tgt_nids);
@@ -526,14 +523,12 @@ mdt_cos_seq_write(struct file *file, const char __user *buffer,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
        int rc;
-       __s64 val;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtobool_from_user(buffer, count, &val);
        if (rc)
                return rc;
-       if (val < INT_MIN || val > INT_MAX)
-               return -ERANGE;
 
        mdt_enable_cos(mdt, val);
        return count;
@@ -615,16 +610,13 @@ mdt_enable_remote_dir_seq_write(struct file *file, const char __user *buffer,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
-       __s64 val;
+       bool val;
        int rc;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtobool_from_user(buffer, count, &val);
        if (rc)
                return rc;
 
-       if (val > 1 || val < 0)
-               return -ERANGE;
-
        mdt->mdt_enable_remote_dir = val;
        return count;
 }
@@ -648,10 +640,10 @@ mdt_enable_remote_dir_gid_seq_write(struct file *file,
        struct seq_file   *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
-       __s64 val;
+       int val;
        int rc;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtoint_from_user(buffer, count, 0, &val);
        if (rc)
                return rc;
 
@@ -660,25 +652,61 @@ mdt_enable_remote_dir_gid_seq_write(struct file *file,
 }
 LPROC_SEQ_FOPS(mdt_enable_remote_dir_gid);
 
-/**
- * Show MDT policy for handling dirty metadata under a lock being cancelled.
- *
- * \param[in] m                seq_file handle
- * \param[in] data     unused for single entry
- *
- * \retval             0 on success
- * \retval             negative value on error
- */
-static int mdt_slc_seq_show(struct seq_file *m, void *data)
+static int mdt_enable_striped_dir_seq_show(struct seq_file *m, void *data)
 {
        struct obd_device *obd = m->private;
-       struct lu_target *tgt = obd->u.obt.obt_lut;
-       char *slc_states[] = {"never", "blocking", "always" };
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%u\n", mdt->mdt_enable_striped_dir);
+       return 0;
+}
+
+static ssize_t
+mdt_enable_striped_dir_seq_write(struct file *file, const char __user *buffer,
+                                size_t count, loff_t *off)
+{
+       struct seq_file   *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
+       int rc;
+
+       rc = kstrtobool_from_user(buffer, count, &val);
+       if (rc)
+               return rc;
+
+       mdt->mdt_enable_striped_dir = val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_enable_striped_dir);
 
-       seq_printf(m, "%s\n", slc_states[tgt->lut_sync_lock_cancel]);
+static int mdt_enable_dir_migration_seq_show(struct seq_file *m, void *data)
+{
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%u\n", mdt->mdt_enable_dir_migration);
        return 0;
 }
-LPROC_SEQ_FOPS_RO(mdt_slc);
+
+static ssize_t
+mdt_enable_dir_migration_seq_write(struct file *file, const char __user *buffer,
+                                  size_t count, loff_t *off)
+{
+       struct seq_file   *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
+       int rc;
+
+       rc = kstrtobool_from_user(buffer, count, &val);
+       if (rc)
+               return rc;
+
+       mdt->mdt_enable_dir_migration = val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_enable_dir_migration);
 
 /**
  * Show MDT async commit count.
@@ -705,16 +733,13 @@ mdt_async_commit_count_seq_write(struct file *file, const char __user *buffer,
        struct seq_file *m = file->private_data;
        struct obd_device *obd = m->private;
        struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
-       __s64 val;
+       int val;
        int rc;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtoint_from_user(buffer, count, 0, &val);
        if (rc)
                return rc;
 
-       if (val < INT_MIN || val > INT_MAX)
-               return -ERANGE;
-
        atomic_set(&mdt->mdt_async_commit_count, val);
 
        return count;
@@ -746,22 +771,245 @@ mdt_sync_count_seq_write(struct file *file, const char __user *buffer,
        struct seq_file *m = file->private_data;
        struct obd_device *obd = m->private;
        struct lu_target *tgt = obd->u.obt.obt_lut;
-       __s64 val;
+       int val;
        int rc;
 
-       rc = lprocfs_str_to_s64(buffer, count, &val);
+       rc = kstrtoint_from_user(buffer, count, 0, &val);
        if (rc)
                return rc;
 
-       if (val < INT_MIN || val > INT_MAX)
-               return -ERANGE;
-
        atomic_set(&tgt->lut_sync_count, val);
 
        return count;
 }
 LPROC_SEQ_FOPS(mdt_sync_count);
 
+static char *dom_open_lock_modes[NUM_DOM_LOCK_ON_OPEN_MODES] = {
+       [NO_DOM_LOCK_ON_OPEN] = "never",
+       [TRYLOCK_DOM_ON_OPEN] = "trylock",
+       [ALWAYS_DOM_LOCK_ON_OPEN] = "always",
+};
+
+/* This must be longer than the longest string above */
+#define DOM_LOCK_MODES_MAXLEN 16
+
+/**
+ * Show MDT policy for data prefetch on open for DoM files..
+ *
+ * \param[in] m                seq_file handle
+ * \param[in] data     unused
+ *
+ * \retval             0 on success
+ * \retval             negative value on error
+ */
+static int mdt_dom_lock_seq_show(struct seq_file *m, void *data)
+{
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%s\n", dom_open_lock_modes[mdt->mdt_opts.mo_dom_lock]);
+       return 0;
+}
+
+/**
+ * Change MDT policy for data prefetch on open for DoM files.
+ *
+ * This variable defines how DOM lock is taken at open enqueue.
+ * There are three possible modes:
+ * 1) never - never take DoM lock on open. DoM lock will be taken as separate
+ *    IO lock with own enqueue.
+ * 2) trylock - DoM lock will be taken only if non-blocked.
+ * 3) always - DoM lock will be taken always even if it is blocking lock.
+ *
+ * If dom_read_open is enabled too then DoM lock is taken in PR mode and
+ * is paired with LAYOUT lock when possible.
+ *
+ * \param[in] file     proc file
+ * \param[in] buffer   string which represents policy
+ * \param[in] count    \a buffer length
+ * \param[in] off      unused for single entry
+ *
+ * \retval             \a count on success
+ * \retval             negative number on error
+ */
+static ssize_t
+mdt_dom_lock_seq_write(struct file *file, const char __user *buffer,
+                      size_t count, loff_t *off)
+{
+       struct seq_file *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       char kernbuf[DOM_LOCK_MODES_MAXLEN];
+       int val = -1;
+       int i, rc;
+
+       if (count == 0 || count >= sizeof(kernbuf))
+               return -EINVAL;
+
+       if (copy_from_user(kernbuf, buffer, count))
+               return -EFAULT;
+
+       kernbuf[count] = 0;
+       if (kernbuf[count - 1] == '\n')
+               kernbuf[count - 1] = 0;
+
+       for (i = 0 ; i < NUM_DOM_LOCK_ON_OPEN_MODES; i++) {
+               if (strcmp(kernbuf, dom_open_lock_modes[i]) == 0) {
+                       val = i;
+                       break;
+               }
+       }
+
+       /* Legacy numeric codes */
+       if (val == -1) {
+               rc = kstrtoint_from_user(buffer, count, 0, &val);
+               if (rc)
+                       return rc;
+       }
+
+       if (val < 0 || val >= NUM_DOM_LOCK_ON_OPEN_MODES)
+               return -EINVAL;
+
+       mdt->mdt_opts.mo_dom_lock = val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_dom_lock);
+
+/**
+ * Show MDT policy for data prefetch on open for DoM files..
+ *
+ * \param[in] m                seq_file handle
+ * \param[in] data     unused
+ *
+ * \retval             0 on success
+ * \retval             negative value on error
+ */
+static int mdt_dom_read_open_seq_show(struct seq_file *m, void *data)
+{
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%u\n", !!mdt->mdt_opts.mo_dom_read_open);
+       return 0;
+}
+
+/**
+ * Modify MDT policy for data prefetch on open for DoM files.
+ *
+ * If enabled then Data-on-MDT file data may be read during open and
+ * returned back in reply. It works only with mo_dom_lock enabled.
+ *
+ * \param[in] file     proc file
+ * \param[in] buffer   string which represents policy
+ * \param[in] count    \a buffer length
+ * \param[in] off      unused for single entry
+ *
+ * \retval             \a count on success
+ * \retval             negative number on error
+ */
+static ssize_t
+mdt_dom_read_open_seq_write(struct file *file, const char __user *buffer,
+                           size_t count, loff_t *off)
+{
+       struct seq_file *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
+       int rc;
+
+       rc = kstrtobool_from_user(buffer, count, &val);
+       if (rc)
+               return rc;
+
+       mdt->mdt_opts.mo_dom_read_open = !!val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_dom_read_open);
+
+static int mdt_migrate_hsm_allowed_seq_show(struct seq_file *m, void *data)
+{
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%u\n",  (mdt->mdt_opts.mo_migrate_hsm_allowed != 0));
+       return 0;
+}
+
+static ssize_t
+mdt_migrate_hsm_allowed_seq_write(struct file *file, const char __user *buffer,
+                                 size_t count, loff_t *off)
+{
+       struct seq_file *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
+       int rc;
+
+       rc = kstrtobool_from_user(buffer, count, &val);
+       if (rc)
+               return rc;
+
+       mdt->mdt_opts.mo_migrate_hsm_allowed = val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_migrate_hsm_allowed);
+
+static int mdt_readonly_seq_show(struct seq_file *m, void *data)
+{
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%u\n", mdt->mdt_readonly);
+       return 0;
+}
+
+static ssize_t
+mdt_readonly_seq_write(struct file *file, const char __user *buffer,
+                              size_t count, loff_t *off)
+{
+       struct seq_file   *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
+       int rc;
+
+       rc = kstrtobool_from_user(buffer, count, &val);
+       if (rc)
+               return rc;
+
+       mdt->mdt_readonly = val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_readonly);
+
+static int mdt_enable_remote_rename_seq_show(struct seq_file *m, void *data)
+{
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+
+       seq_printf(m, "%u\n",  mdt->mdt_enable_remote_rename);
+       return 0;
+}
+
+static ssize_t
+mdt_enable_remote_rename_seq_write(struct file *file, const char __user *buffer,
+                                  size_t count, loff_t *off)
+{
+       struct seq_file *m = file->private_data;
+       struct obd_device *obd = m->private;
+       struct mdt_device *mdt = mdt_dev(obd->obd_lu_dev);
+       bool val;
+       int rc;
+
+       rc = kstrtobool_from_user(buffer, count, &val);
+       if (rc)
+               return rc;
+
+       mdt->mdt_enable_remote_rename = val;
+       return count;
+}
+LPROC_SEQ_FOPS(mdt_enable_remote_rename);
+
 LPROC_SEQ_FOPS_RO_TYPE(mdt, recovery_status);
 LPROC_SEQ_FOPS_RO_TYPE(mdt, num_exports);
 LPROC_SEQ_FOPS_RO_TYPE(mdt, target_instance);
@@ -775,7 +1023,20 @@ LPROC_SEQ_FOPS(mdt_hsm_cdt_control);
 LPROC_SEQ_FOPS_RW_TYPE(mdt, recovery_time_hard);
 LPROC_SEQ_FOPS_RW_TYPE(mdt, recovery_time_soft);
 
+LPROC_SEQ_FOPS_RO(tgt_tot_dirty);
+LPROC_SEQ_FOPS_RO(tgt_tot_granted);
+LPROC_SEQ_FOPS_RO(tgt_tot_pending);
+LPROC_SEQ_FOPS(tgt_grant_compat_disable);
+
 static struct lprocfs_vars lprocfs_mdt_obd_vars[] = {
+       { .name =       "tot_dirty",
+         .fops =       &tgt_tot_dirty_fops             },
+       { .name =       "tot_pending",
+         .fops =       &tgt_tot_pending_fops           },
+       { .name =       "tot_granted",
+         .fops =       &tgt_tot_granted_fops           },
+       { .name =       "grant_compat_disable",
+         .fops =       &tgt_grant_compat_disable_fops  },
        { .name =       "recovery_status",
          .fops =       &mdt_recovery_status_fops               },
        { .name =       "num_exports",
@@ -814,18 +1075,30 @@ static struct lprocfs_vars lprocfs_mdt_obd_vars[] = {
          .fops =       &mdt_enable_remote_dir_fops             },
        { .name =       "enable_remote_dir_gid",
          .fops =       &mdt_enable_remote_dir_gid_fops         },
+       { .name =       "enable_striped_dir",
+         .fops =       &mdt_enable_striped_dir_fops            },
+       { .name =       "enable_dir_migration",
+         .fops =       &mdt_enable_dir_migration_fops          },
+       { .name =       "enable_remote_rename",
+         .fops =       &mdt_enable_remote_rename_fops          },
        { .name =       "hsm_control",
          .fops =       &mdt_hsm_cdt_control_fops               },
        { .name =       "recovery_time_hard",
          .fops =       &mdt_recovery_time_hard_fops    },
        { .name =       "recovery_time_soft",
          .fops =       &mdt_recovery_time_soft_fops    },
-       { .name =       "sync_lock_cancel",
-         .fops =       &mdt_slc_fops                           },
        { .name =       "async_commit_count",
          .fops =       &mdt_async_commit_count_fops            },
        { .name =       "sync_count",
          .fops =       &mdt_sync_count_fops                    },
+       { .name =       "dom_lock",
+         .fops =       &mdt_dom_lock_fops                      },
+       { .name =       "dom_read_open",
+         .fops =       &mdt_dom_read_open_fops                 },
+       { .name =       "migrate_hsm_allowed",
+         .fops =       &mdt_migrate_hsm_allowed_fops           },
+       { .name =       "readonly",
+         .fops =       &mdt_readonly_fops                      },
        { NULL }
 };
 
@@ -882,7 +1155,8 @@ void mdt_counter_incr(struct ptlrpc_request *req, int opcode)
        struct obd_export *exp = req->rq_export;
 
        if (exp->exp_obd && exp->exp_obd->obd_md_stats)
-               lprocfs_counter_incr(exp->exp_obd->obd_md_stats, opcode);
+               lprocfs_counter_incr(exp->exp_obd->obd_md_stats,
+                                    opcode + LPROC_MD_LAST_OPC);
        if (exp->exp_nid_stats && exp->exp_nid_stats->nid_stats != NULL)
                lprocfs_counter_incr(exp->exp_nid_stats->nid_stats, opcode);
        if (exp->exp_obd && exp->exp_obd->u.obt.obt_jobstats.ojs_hash &&
@@ -892,34 +1166,51 @@ void mdt_counter_incr(struct ptlrpc_request *req, int opcode)
                                      opcode, 1);
 }
 
+static const char * const mdt_stats[] = {
+       [LPROC_MDT_OPEN]                = "open",
+       [LPROC_MDT_CLOSE]               = "close",
+       [LPROC_MDT_MKNOD]               = "mknod",
+       [LPROC_MDT_LINK]                = "link",
+       [LPROC_MDT_UNLINK]              = "unlink",
+       [LPROC_MDT_MKDIR]               = "mkdir",
+       [LPROC_MDT_RMDIR]               = "rmdir",
+       [LPROC_MDT_RENAME]              = "rename",
+       [LPROC_MDT_GETATTR]             = "getattr",
+       [LPROC_MDT_SETATTR]             = "setattr",
+       [LPROC_MDT_GETXATTR]            = "getxattr",
+       [LPROC_MDT_SETXATTR]            = "setxattr",
+       [LPROC_MDT_STATFS]              = "statfs",
+       [LPROC_MDT_SYNC]                = "sync",
+       [LPROC_MDT_SAMEDIR_RENAME]      = "samedir_rename",
+       [LPROC_MDT_CROSSDIR_RENAME]     = "crossdir_rename",
+       [LPROC_MDT_IO_READ]             = "read_bytes",
+       [LPROC_MDT_IO_WRITE]            = "write_bytes",
+       [LPROC_MDT_IO_PUNCH]            = "punch",
+};
+
 void mdt_stats_counter_init(struct lprocfs_stats *stats)
 {
-        lprocfs_counter_init(stats, LPROC_MDT_OPEN, 0, "open", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_CLOSE, 0, "close", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_MKNOD, 0, "mknod", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_LINK, 0, "link", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_UNLINK, 0, "unlink", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_MKDIR, 0, "mkdir", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_RMDIR, 0, "rmdir", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_RENAME, 0, "rename", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_GETATTR, 0, "getattr", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_SETATTR, 0, "setattr", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_GETXATTR, 0, "getxattr", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_SETXATTR, 0, "setxattr", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_STATFS, 0, "statfs", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_SYNC, 0, "sync", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_SAMEDIR_RENAME, 0,
-                             "samedir_rename", "reqs");
-        lprocfs_counter_init(stats, LPROC_MDT_CROSSDIR_RENAME, 0,
-                             "crossdir_rename", "reqs");
+       int idx;
+
+       LASSERT(stats && stats->ls_num >= ARRAY_SIZE(mdt_stats));
+
+       for (idx = 0; idx < ARRAY_SIZE(mdt_stats); idx++) {
+               int flags = 0;
+
+               if (idx == LPROC_MDT_IO_WRITE || idx == LPROC_MDT_IO_READ)
+                       flags = LPROCFS_CNTR_AVGMINMAX;
+
+               lprocfs_counter_init(stats, idx, flags, mdt_stats[idx], "reqs");
+       }
 }
 
 int mdt_procfs_init(struct mdt_device *mdt, const char *name)
 {
-       struct obd_device               *obd = mdt2obd_dev(mdt);
-       int                              rc;
-       ENTRY;
+       struct obd_device *obd = mdt2obd_dev(mdt);
+       int rc;
+       int i;
 
+       ENTRY;
        LASSERT(name != NULL);
 
        obd->obd_vars = lprocfs_mdt_obd_vars;
@@ -930,6 +1221,13 @@ int mdt_procfs_init(struct mdt_device *mdt, const char *name)
                return rc;
        }
 
+       rc = tgt_tunables_init(&mdt->mdt_lut);
+       if (rc) {
+               CERROR("%s: failed to init target tunables: rc = %d\n",
+                      mdt_obd_name(mdt), rc);
+               return rc;
+       }
+
        rc = hsm_cdt_procfs_init(mdt);
        if (rc) {
                CERROR("%s: cannot create hsm proc entries: rc = %d\n",
@@ -942,12 +1240,24 @@ int mdt_procfs_init(struct mdt_device *mdt, const char *name)
        if (obd->obd_proc_exports_entry)
                lprocfs_add_simple(obd->obd_proc_exports_entry, "clear",
                                   obd, &mdt_nid_stats_clear_fops);
-       rc = lprocfs_alloc_md_stats(obd, LPROC_MDT_LAST);
+
+       rc = lprocfs_alloc_md_stats(obd, ARRAY_SIZE(mdt_stats));
        if (rc)
                return rc;
-       mdt_stats_counter_init(obd->obd_md_stats);
 
-       rc = lprocfs_job_stats_init(obd, LPROC_MDT_LAST,
+       /* add additional MDT md_stats after the default ones */
+       for (i = 0; i < ARRAY_SIZE(mdt_stats); i++) {
+               int idx = i + LPROC_MD_LAST_OPC;
+               int flags = 0;
+
+               if (idx == LPROC_MDT_IO_WRITE || idx == LPROC_MDT_IO_READ)
+                       flags = LPROCFS_CNTR_AVGMINMAX;
+
+               lprocfs_counter_init(obd->obd_md_stats, idx, flags,
+                                    mdt_stats[i], "reqs");
+       }
+
+       rc = lprocfs_job_stats_init(obd, ARRAY_SIZE(mdt_stats),
                                    mdt_stats_counter_init);
 
        rc = lproc_mdt_attach_rename_seqstat(mdt);
@@ -969,6 +1279,7 @@ void mdt_procfs_fini(struct mdt_device *mdt)
 
        lprocfs_free_per_client_stats(obd);
        hsm_cdt_procfs_fini(mdt);
+       tgt_tunables_fini(&mdt->mdt_lut);
        lprocfs_obd_cleanup(obd);
        lprocfs_free_md_stats(obd);
        lprocfs_free_obd_stats(obd);